/* General utility routines for the remote server for GDB. Copyright (C) 1986-2014 Free Software Foundation, Inc. This file is part of GDB. 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 3 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, see . */ #include "server.h" #include #include #if HAVE_ERRNO_H #include #endif #ifdef IN_PROCESS_AGENT # define PREFIX "ipa: " # define TOOLNAME "GDBserver in-process agent" #else # define PREFIX "gdbserver: " # define TOOLNAME "GDBserver" #endif /* Generally useful subroutines used throughout the program. */ void malloc_failure (long size) { fprintf (stderr, PREFIX "ran out of memory while trying to allocate %lu bytes\n", (unsigned long) size); exit (1); } /* Copy a string into a memory buffer. If malloc fails, this will print a message to stderr and exit. */ char * xstrdup (const char *s) { char *ret = strdup (s); if (ret == NULL) malloc_failure (strlen (s) + 1); return ret; } /* Print the system error message for errno, and also mention STRING as the file name for which the error was encountered. Then return to command level. */ void perror_with_name (const char *string) { const char *err; char *combined; err = strerror (errno); if (err == NULL) err = "unknown error"; combined = (char *) alloca (strlen (err) + strlen (string) + 3); strcpy (combined, string); strcat (combined, ": "); strcat (combined, err); error ("%s.", combined); } /* Print an error message and return to command level. STRING is the error message, used as a fprintf string, and ARG is passed as an argument to it. */ void error (const char *string,...) { #ifndef IN_PROCESS_AGENT extern jmp_buf toplevel; #endif va_list args; va_start (args, string); fflush (stdout); vfprintf (stderr, string, args); fprintf (stderr, "\n"); #ifndef IN_PROCESS_AGENT longjmp (toplevel, 1); #else exit (1); #endif } /* Print an error message and exit reporting failure. This is for a error that we cannot continue from. STRING and ARG are passed to fprintf. */ /* VARARGS */ void fatal (const char *string,...) { va_list args; va_start (args, string); fprintf (stderr, PREFIX); vfprintf (stderr, string, args); fprintf (stderr, "\n"); va_end (args); exit (1); } /* VARARGS */ void warning (const char *string,...) { va_list args; va_start (args, string); fprintf (stderr, PREFIX); vfprintf (stderr, string, args); fprintf (stderr, "\n"); va_end (args); } /* Report a problem internal to GDBserver, and exit. */ void internal_error (const char *file, int line, const char *fmt, ...) { va_list args; va_start (args, fmt); fprintf (stderr, "\ %s:%d: A problem internal to " TOOLNAME " has been detected.\n", file, line); vfprintf (stderr, fmt, args); fprintf (stderr, "\n"); va_end (args); exit (1); } /* Convert a CORE_ADDR into a HEX string, like %lx. The result is stored in a circular static buffer, NUMCELLS deep. */ char * paddress (CORE_ADDR addr) { return phex_nz (addr, sizeof (CORE_ADDR)); } /* Convert a file descriptor into a printable string. */ char * pfildes (gdb_fildes_t fd) { #if USE_WIN32API return phex_nz (fd, sizeof (gdb_fildes_t)); #else return plongest (fd); #endif }