Code Search for Developers
 
 
  

server.c from Gdb at Krugle


Show server.c syntax highlighted

/* server-main.c

   Copyright 2000, 2001, 2002 Red Hat, Inc.

   This file is part of RDA, the Red Hat Debug Agent (and library).

   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 2 of the License, or
   (at your option) any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with this program; if not, write to the Free Software
   Foundation, Inc., 59 Temple Place - Suite 330,
   Boston, MA 02111-1307, USA.
   
   Alternative licenses for RDA may be arranged by contacting Red Hat,
   Inc.  */

#include "config.h"

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <signal.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
#include <ctype.h>
#include <termios.h>

#include "gdbsocket.h"
#include "gdbloop.h"
#include "gdbserv.h"
#include "server.h"

/* Signal a request to terminate main loop. */
int server_quit_p = 0;

/* FIXME -- move to header file.  */
struct gdbserv_target *linux_attach (struct gdbserv* serv, void* data);

/* We have an empty SIGCHLD handler in order to make sure that
   the timed, but blocking call to select will be interrupted
   by SIGCHLD signals.  */

static void
chld_handler (int sig)
{
}

static struct termios save_termios;

static void
close_device (int infd, int outfd)
{
  if (isatty (infd))
    {
      tcsetattr (infd, TCSAFLUSH, &save_termios);
    }
  close (infd);
  close (outfd);
}

/* Put a tty into "raw" mode.  Mostly taken from Stevens' book, "Advanced
   Programming in the UNIX Environment.  */
static int
tty_raw (int fd, speed_t speed)
{
  struct termios buf;

  /* Don't do anything for non-tty devices.  */
  if (!isatty (fd))
    return 0;

  if (tcgetattr (fd, &save_termios) < 0)
    return -1;

  buf = save_termios;

  /* Set the following local modes: echo off, canonical mode off, extended
     input processing off, signal chars off.  */
  buf.c_lflag &= ~(ECHO | ICANON | IEXTEN | ISIG);

  /* Set the following input modes: no SIGINT on BREAK, CR-to-NL off, input
     parity check off, don't strip 7th bit on input, output flow control
     off.  */
  buf.c_iflag &= ~(BRKINT | ICRNL | INPCK | ISTRIP | IXON);

  /* Set the following control modes: clear size bits, parity checking off.  */
  buf.c_cflag &= ~(CSIZE | PARENB); 

  /* Enable 8 bits/char.  */
  buf.c_cflag |= CS8;

  /* Turn output processing off.  */
  buf.c_oflag &= ~(OPOST);

  /* Case B: 1 byte at a time, no timer */
  buf.c_cc[VMIN] = 1;
  buf.c_cc[VTIME] = 0;

  if (speed != B0)
    {
      cfsetispeed (&buf, speed);
      cfsetospeed (&buf, speed);
    }

  if (tcsetattr(fd, TCSAFLUSH, &buf) < 0)
    return -1;
  return 0;
}

/* Table of serial port speed values.  */
static struct
  {
    long speed;		/* speed value as an integer */
    speed_t bspeed;	/* speed value as one of the termios.h constants
    			   suitable for passing to cfsetispeed() or
			   cfsetospeed().  */
  }
speed_table[] = {
  { 50, B50 },
  { 75, B75 },
  { 110, B110 },
  { 134, B134 },
  { 150, B150 },
  { 200, B200 },
  { 300, B300 },
  { 600, B600 },
  { 1200, B1200 },
  { 1800, B1800 },
  { 2400, B2400 },
  { 4800, B4800 },
  { 9600, B9600 },
  { 19200, B19200 },
  { 38400, B38400 }
#ifdef B57600 
  ,{ 57600, B57600 }
#endif
#ifdef B115200
  ,{ 115200, B115200 }
#endif
#ifdef B230400
  ,{ 230400, B230400 }
#endif
#ifdef B460800
  ,{ 460800, B460800 }
#endif
#ifdef B500000
  ,{ 500000, B500000 }
#endif
#ifdef B576000
  ,{ 576000, B576000 }
#endif
#ifdef B921600
  ,{ 921600, B921600 }
#endif
#ifdef B2000000
  ,{ 1000000, B1000000 }
#endif
#ifdef B1152000
  ,{ 1152000, B1152000 }
#endif
#ifdef B1500000
  ,{ 1500000, B1500000 }
#endif
#ifdef B2000000
  ,{ 2000000, B2000000 }
#endif
#ifdef B2500000
  ,{ 2500000, B2500000 }
#endif
#ifdef B3000000
  ,{ 3000000, B3000000 }
#endif
#ifdef B3500000
  ,{ 3500000, B3500000 }
#endif
#ifdef B4000000
  ,{ 4000000, B4000000 }
#endif
};

/* Print error for erroneous -s switch and exit.  */
static void
invalid_speed (char *str)
{
  int i;
  int ll;

  fprintf (stderr, "Error: Invalid -s switch \"%s\".\n", str);
  ll = fprintf (stderr, "Valid speeds are:");
  
  for (i = 0; i < sizeof (speed_table) / sizeof (speed_table[0]); i++)
    {
      char str[20];
      int cnt;
      cnt = snprintf (str, sizeof (str), "%ld", speed_table[i].speed);
      if (cnt + ll + 1 > 80)
	{
	  fprintf (stderr, "\n");
	  ll = 0;
	}
      else
	{
	  fprintf (stderr, " ");
	  ll++;
	}
      fprintf (stderr, "%s", str);
      ll += cnt;
    }
  fprintf (stderr, "\n");
  exit (1);
}

/* Attempt to parse a speed and return the speed value.  */
static speed_t
parse_speed (char *str)
{
  int i;
  long speed;
  char *endptr;

  speed = strtol (str, &endptr, 10);
  if (speed == 0 || *endptr != '\0')
    {
      invalid_speed (str);
      /* won't return */
    }

  for (i = 0; i < sizeof (speed_table) / sizeof (speed_table[0]); i++)
    if (speed_table[i].speed == speed)
      return speed_table[i].bspeed;

  /* Speed not found.  Error out.  */
  invalid_speed (str);
  return 0;	/* won't actually return */
}

/* Print a usage message and exit.  */
static void
usage (char *progname)
{
  /* Remove any leading slashes from the executable name.  */
  if (strrchr (progname, '/'))
    progname = strrchr (progname, '/') + 1;

  fprintf (stderr,
    "Usage: %s [-v] tcp-port-num executable-file [arguments ...]\n"
    "   or: %s -a [-v] tcp-port-num process-id\n"
    "   or: %s [-v] [-s speed] device-name executable-file [arguments ...]\n"
    "   or: %s -a [-v] [-s speed] device-name process-id\n"
    "   or: %s -h\n"
    "Start the Red Hat debug agent for use with a remote debugger.\n"
    "Options and arguments:\n"
    "  -a               Attach to already running process.\n"
    "  -h               Print this usage message.\n"
    "  -s speed         Set speed (e.g. 115200) at which device \"device-name\"\n"
    "                   will communicate with remote debugger.\n"
    "  -v               Increase verbosity.  One -v flag enables informational\n"
    "                   messages.  Two -v flags turn on internal debugging\n"
    "                   messages.\n"
    "  arguments ...    Command line arguments with which to start program\n"
    "                   being debugged.\n"
    "  device-name      Name of serial device over which RDA will communicate\n"
    "                   with remote debugger.\n"
    "  executable-file  Name of program to debug.\n"
    "  process-id       Process ID (PID) of process to attach to.\n"
    "  tcp-port-num     Port number to which debugger connects for purpose\n"
    "                   of communicating with the debug agent using the GDB\n"
    "                   remote protocol.\n",
    progname, progname, progname, progname, progname);
  exit (1);
}

int
main (int argc, char **argv)
{
  int portno = 0;
  char *endptr;
  int verbose = 0;
  int attach = 0;
  int optidx;
  int infd = 0, outfd = 0;
  struct child_process *process;
  char *devicename = "";
  speed_t speed = B0;

  /* Parse options.  */
  for (optidx = 1; optidx < argc; optidx++)
    {
      if (argv[optidx][0] == '-' && argv[optidx][1] != '\0')
	{
	  switch (argv[optidx][1])
	    {
	      case 'a':
		attach = 1;
		break;
	      case 'h':
		usage (argv[0]);
		/* not reached */
		break;
	      case 's':
		optidx++;
		if (optidx >= argc)
		  usage(argv[0]);
		speed = parse_speed (argv[optidx]);
		break;
	      case 'v':
		verbose++;
		break;
	      default:
		usage (argv[0]);
		/* not reached */
		break;
	    }
	}
      else
	break;
    }

  if (argc - optidx < 2)
    usage (argv[0]);

  if (isdigit (*argv[optidx]))
    {
      errno = 0;
      portno = strtol (argv[optidx], &endptr, 10);
      if (errno != 0 || portno == 0)
	usage (argv[0]);
    }
  else if (strcmp (argv[optidx], "-") == 0)
    {
      infd = STDIN_FILENO;
      outfd = STDOUT_FILENO;
      devicename = "stdin/stdout";
    }
  else
    {
      devicename = argv[optidx];
      infd = open (devicename, O_RDWR);
      if (infd < 0)
	{
	  fprintf (stderr, "Error opening device %s: %s\n", devicename,
	           strerror (errno));
	  exit (1);
	}
      outfd = infd;
    }

  process = malloc (sizeof (struct child_process));
  memset (process, 0, sizeof (struct child_process));
  if (attach)
    {
      process->pid = strtol (argv[optidx + 1], &endptr, 10);
      if (errno != 0 || endptr == argv[optidx] || process->pid <= 0)
	usage (argv[0]);
    }
  else
    {
      process->argv       = &argv[optidx + 1];
      process->executable =  argv[optidx + 1];
    }

  if (verbose > 1)
    process->debug_backend = 1;
  if (verbose)
    process->debug_informational = 1;

  signal (SIGCHLD, chld_handler);

#if defined(__SIGRTMIN) && defined(__SIGRTMAX)
  {
    int sig;
    /* Ignore realtime signals.  We do this so as to not terminate
       RDA if we inadvertently receive one of these signals.  The
       realtime signals are used for thread support, and, for some
       reason, some environments send these signals to RDA as well
       as the application.  (And some don't.)  */
    for (sig = __SIGRTMIN; sig <= __SIGRTMAX; sig++)
      signal (sig, SIG_IGN);
  }
#endif

  if (portno != 0)
    {
      if (gdbsocket_startup (portno, gdbserver.attach, process) < 0)
        {
          fprintf (stderr, "Error listening on port %d: %s\n",
                   portno, strerror (errno));
          return 2;
        }
      if (process->debug_informational)
	fprintf (stderr, "Started listening socket on port %d.\n", portno);
    }
  else
    {
      tty_raw (infd, speed);
      if (process->debug_informational)
	fprintf (stderr, "Waiting for input on %s.\n", devicename);
      gdbsocket_reopen (infd, outfd, close_device, gdbserver.attach, process);
    }

  /* Poll for socket traffic. */
  while (! server_quit_p)
    {
      gdbloop_poll (10 /* milliseconds */);
      if (! server_quit_p)
	{
	  if (gdbserver.check_child_state (process))
	    {
	      switch (process->stop_status) {
	      case 'T':
		gdbserver.fromtarget_break (process);
		break;
	      case 'X':
		gdbserver.fromtarget_terminate (process);
		server_quit_p = 1;		/* See below.  */
		break;
	      case 'W':
		gdbserver.fromtarget_exit (process);
		/* If we're connected to a serial port which is not
		   observing the modem control signals (e.g. "stty
		   clocal"), then we'll never notice that the port has
		   been closed.  For this reason, when we notice that
		   the target has exited or terminated, we need to set
		   ``server_quit_p'' explicitly.  */
		server_quit_p = 1;
		break;
	      }
	    }
	}
    }
  gdbsocket_shutdown ();
  if (process->debug_informational)
    fprintf (stderr, "Shut down sockets.\n");

  return 0;
}




See more files for this project here

Gdb

GDB, the GNU Project debugger, allows you to see what is going on `inside' another program while it executes -- or what another program was doing at the moment it crashed.

Project homepage: http://sources.redhat.com/gdb/
Programming language(s): Assembly,C,C++,Expect
License: other

  ChangeLog
  Makefile.am
  Makefile.in
  acinclude.m4
  aclocal.m4
  arch.h
  config.in
  configure
  configure.in
  diagnostics.c
  diagnostics.h
  dummy-target.c
  gdb_proc_service.h
  gdbserv-thread-db.h
  linux-target.c
  lwp-ctrl.h
  lwp-pool.c
  lwp-pool.h
  no-threads.c
  ptrace-target.c
  ptrace-target.h
  server.c
  server.h
  solaris-target.c
  stock-breakpoints.c
  stock-breakpoints.h
  thread-db.c