1999-04-16 09:35:26 +08:00
|
|
|
/* General utility routines for the remote server for GDB.
|
2007-01-10 01:59:20 +08:00
|
|
|
Copyright (C) 1986, 1989, 1993, 1995, 1996, 1997, 1999, 2000, 2002, 2003,
|
2010-01-01 15:32:07 +08:00
|
|
|
2007, 2008, 2009, 2010 Free Software Foundation, Inc.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
This file is part of GDB.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
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
|
2007-08-24 02:08:50 +08:00
|
|
|
the Free Software Foundation; either version 3 of the License, or
|
1999-07-08 04:19:36 +08:00
|
|
|
(at your option) any later version.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
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.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
You should have received a copy of the GNU General Public License
|
2007-08-24 02:08:50 +08:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
#include "server.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
gdb/
* arm-wince-tdep.c: New.
* config/arm/wince.mt (DEPRECATED_TM_FILE): Use tm-arm.h.
(MT_CFLAGS): Delete.
(TM_CLIBS): Delete.
(TDEPFILES): Add arm-wince-tdep.o, corelow.o, solib.o,
solib-legacy.o, solib-svr4.o, and remove wince.o.
* configure.tgt (arm*-*-mingw32ce*): Add.
* signals/signals.c [HAVE_SIGNAL_H]: Check.
(do_target_signal_to_host): Silence 'not used' warning.
* config/arm/tm-wince.h: Remove.
gdb/gdbserver/
* gdbserver/configure.ac: Add errno checking.
(AC_CHECK_HEADERS): Add errno.h, fcntl.h, signal.h,
sys/file.h and malloc.h.
(AC_CHECK_DECLS): Add perror.
(srv_mingwce): Handle.
* gdbserver/configure.srv (i[34567]86-*-cygwin*): Add
win32-i386-low.o to srv_tgtobj.
(i[34567]86-*-mingw*): Likewise.
(arm*-*-mingw32ce*): Add case.
* gdbreplay.c [HAVE_SYS_FILE_H, HAVE_SIGNAL_H,
HAVE_FCNTL_H, HAVE_ERRNO_H, HAVE_MALLOC_H]: Check.
[__MINGW32CE__] (strerror): New function.
[__MINGW32CE__] (errno): Define to GetLastError.
[__MINGW32CE__] (COUNTOF): New macro.
(remote_open): Remove extra close call.
* mem-break.c (delete_breakpoint_at): New function.
* mem-break.h (delete_breakpoint_at): Declare.
* remote-utils.c [HAVE_SYS_FILE_H, HAVE_SIGNAL_H,
HAVE_FCNTL_H, HAVE_UNISTD_H, HAVE_ERRNO_H]: Check.
[USE_WIN32API] (read, write): Add char* casts.
* server.c [HAVE_UNISTD_H, HAVE_SIGNAL_H]: Check.
* server.h: Include wincecompat.h on Windows CE.
[HAVE_ERRNO_H]: Check.
(perror): Declare if not declared.
* utils.c: Add stdlib.h, errno.h and malloc.h includes.
(perror_with_name): Remove errno declaration.
* wincecompat.h: New.
* wincecompat.c: New.
* win32-low.h: New.
* win32-arm-low.c: New.
* win32-i386-low.c: New.
(win32-low.c): Include mem-break.h and win32-low.h, and winnt.h.
(OUTMSG2): Make it safe.
(_T): New macro.
(COUNTOF): New macro.
(NUM_REGS): Get it from the low target.
(CONTEXT_EXTENDED_REGISTERS, CONTEXT_FLOATING_POINT,
CONTEXT_DEBUG_REGISTERS): Add fallbacks to 0.
(thread_rec): Let low target handle debug registers.
(child_add_thread): Likewise.
(child_init_thread_list): Likewise.
(continue_one_thread): Likewise.
(regptr): New.
(do_child_fetch_inferior_registers): Move to ...
* win32-i386-low.c: ... here, and rename to ...
(do_fetch_inferior_registers): ... this.
* win32-low.c (child_fetch_inferior_registers):
Go through the low target.
(do_child_store_inferior_registers): Use regptr.
(strwinerror): New function.
(win32_create_inferior): Handle Windows CE.
Use strwinerror instead of strerror on Windows error
codes. Add program to the error output.
Don't close the main thread handle on Windows CE.
(win32_attach): Use coredll.dll on Windows CE.
(win32_kill): Close current process and current
thread handles.
(win32_detach): Use coredll.dll on Windows CE.
(win32_resume): Let low target handle debug registers, and
step request.
(handle_exception): Add/Remove initial breakpoint. Avoid
non-existant WSTOPSIG on Windows CE.
(win32_read_inferior_memory): Cast to remove warning.
(win32_arch_string): Go through the low target.
(initialize_low): Call set_breakpoint_data with the low
target's breakpoint.
* win32-low.c (dr, FLAG_TRACE_BIT, FCS_REGNUM,
FOP_REGNUM, mappings): Move to ...
* win32-i386-low.c: ... here.
* win32-low.c (win32_thread_info): Move to ...
* win32-low.h: ... here.
* Makefile.in (SFILES): Add win32-low.c, win32-i386-low.c,
win32-arm-low.c and wincecompat.c.
(all:): Add $EXEEXT.
(install-only:): Likewise.
(gdbserver:): Likewise.
(gdbreplay:): Likewise.
* config.in: Regenerate.
* configure: Regenerate.
2007-03-29 09:06:48 +08:00
|
|
|
#include <stdlib.h>
|
|
|
|
#if HAVE_ERRNO_H
|
|
|
|
#include <errno.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_MALLOC_H
|
|
|
|
#include <malloc.h>
|
|
|
|
#endif
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
/* Generally useful subroutines used throughout the program. */
|
|
|
|
|
2008-12-15 04:51:04 +08:00
|
|
|
static void malloc_failure (size_t size) ATTR_NORETURN;
|
|
|
|
|
|
|
|
static void
|
|
|
|
malloc_failure (size_t size)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "gdbserver: ran out of memory while trying to allocate %lu bytes\n",
|
|
|
|
(unsigned long) size);
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate memory without fail.
|
|
|
|
If malloc fails, this will print a message to stderr and exit. */
|
|
|
|
|
|
|
|
void *
|
|
|
|
xmalloc (size_t size)
|
|
|
|
{
|
|
|
|
void *newmem;
|
|
|
|
|
|
|
|
if (size == 0)
|
|
|
|
size = 1;
|
|
|
|
newmem = malloc (size);
|
|
|
|
if (!newmem)
|
|
|
|
malloc_failure (size);
|
|
|
|
|
|
|
|
return newmem;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate memory without fail and set it to zero.
|
|
|
|
If malloc fails, this will print a message to stderr and exit. */
|
|
|
|
|
|
|
|
void *
|
|
|
|
xcalloc (size_t nelem, size_t elsize)
|
|
|
|
{
|
|
|
|
void *newmem;
|
|
|
|
|
|
|
|
if (nelem == 0 || elsize == 0)
|
|
|
|
nelem = elsize = 1;
|
|
|
|
|
|
|
|
newmem = calloc (nelem, elsize);
|
|
|
|
if (!newmem)
|
|
|
|
malloc_failure (nelem * elsize);
|
|
|
|
|
|
|
|
return newmem;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
2009-01-19 08:16:46 +08:00
|
|
|
/* Free a standard argv vector. */
|
|
|
|
|
|
|
|
void
|
|
|
|
freeargv (char **vector)
|
|
|
|
{
|
|
|
|
char **scan;
|
|
|
|
|
|
|
|
if (vector != NULL)
|
|
|
|
{
|
|
|
|
for (scan = vector; *scan != NULL; scan++)
|
|
|
|
{
|
|
|
|
free (*scan);
|
|
|
|
}
|
|
|
|
free (vector);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-04-16 09:35:26 +08:00
|
|
|
/* 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
|
2009-01-16 06:07:20 +08:00
|
|
|
perror_with_name (const char *string)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
1999-11-02 12:44:47 +08:00
|
|
|
const char *err;
|
1999-04-16 09:35:26 +08:00
|
|
|
char *combined;
|
|
|
|
|
2003-06-29 12:01:39 +08:00
|
|
|
err = strerror (errno);
|
|
|
|
if (err == NULL)
|
1999-04-16 09:35:26 +08:00
|
|
|
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. */
|
|
|
|
|
2002-04-10 05:11:35 +08:00
|
|
|
void
|
1999-07-08 04:19:36 +08:00
|
|
|
error (const char *string,...)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
extern jmp_buf toplevel;
|
|
|
|
va_list args;
|
|
|
|
va_start (args, string);
|
|
|
|
fflush (stdout);
|
|
|
|
vfprintf (stderr, string, args);
|
|
|
|
fprintf (stderr, "\n");
|
1999-07-08 04:19:36 +08:00
|
|
|
longjmp (toplevel, 1);
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* 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 */
|
2002-04-10 05:11:35 +08:00
|
|
|
void
|
2002-02-14 14:21:24 +08:00
|
|
|
fatal (const char *string,...)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start (args, string);
|
* inferiors.c (change_inferior_id): Delete.
(add_pid_to_list, pull_pid_from_list): New.
* linux-low.c (PTRACE_SETOPTIONS, PTRACE_GETEVENTMSG)
(PTRACE_O_TRACESYSGOOD, PTRACE_O_TRACEFORK, PTRACE_O_TRACEVFORK)
(PTRACE_O_TRACECLONE, PTRACE_O_TRACEEXEC, PTRACE_O_TRACEVFORKDONE)
(PTRACE_O_TRACEEXIT, PTRACE_EVENT_FORK, PTRACE_EVENT_VFORK)
(PTRACE_EVENT_CLONE, PTRACE_EVENT_EXEC, PTRACE_EVENT_VFORK_DONE)
(PTRACE_EVENT_EXIT, __WALL): Provide default definitions.
(stopped_pids, thread_db_active, must_set_ptrace_flags): New variables.
(using_threads): Always set to 1.
(handle_extended_wait): New.
(add_process): Do not set TID.
(linux_create_inferior): Set must_set_ptrace_flags.
(linux_attach_lwp): Remove TID argument. Do not check using_threads.
Use PTRACE_SETOPTIONS. Call new_thread_notify. Update all callers.
(linux_thread_alive): Rename TID argument to LWPID.
(linux_wait_for_process): Handle unknown processes. Do not use TID.
(linux_wait_for_event): Do not use TID or check using_threads. Update
call to dead_thread_notify. Call handle_extended_wait.
(linux_create_inferior): Use PTRACE_SETOPTIONS.
(send_sigstop): Delete sigstop_sent.
(wait_for_sigstop): Avoid TID.
(linux_supports_tracefork_flag, linux_tracefork_child, my_waitpid)
(linux_test_for_tracefork): New.
(linux_lookup_signals): Use thread_db_active and
linux_supports_tracefork_flag.
(initialize_low): Use thread_db_active and linux_test_for_tracefork.
* linux-low.h (get_process_thread): Avoid TID.
(struct process_ifo): Move thread_known and tid to the end. Remove
sigstop_sent.
(linux_attach_lwp, thread_db_init): Update prototypes.
* server.h (change_inferior_id): Delete prototype.
(add_pid_to_list, pull_pid_from_list): New prototypes.
* thread-db.c (thread_db_use_events): New.
(find_first_thread): Rename to...
(find_one_thread): ...this. Update callers and messages. Do not
call fatal. Check thread_db_use_events. Do not call
change_inferior_id or new_thread_notify.
(maybe_attach_thread): Update. Do not call new_thread_notify.
(thread_db_init): Set thread_db_use_events. Check use_events.
* utils.c (fatal, warning): Correct message prefix.
2007-10-24 04:05:03 +08:00
|
|
|
fprintf (stderr, "gdbserver: ");
|
1999-04-16 09:35:26 +08:00
|
|
|
vfprintf (stderr, string, args);
|
|
|
|
fprintf (stderr, "\n");
|
|
|
|
va_end (args);
|
|
|
|
exit (1);
|
|
|
|
}
|
2002-02-14 14:21:24 +08:00
|
|
|
|
|
|
|
/* VARARGS */
|
|
|
|
void
|
|
|
|
warning (const char *string,...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start (args, string);
|
* inferiors.c (change_inferior_id): Delete.
(add_pid_to_list, pull_pid_from_list): New.
* linux-low.c (PTRACE_SETOPTIONS, PTRACE_GETEVENTMSG)
(PTRACE_O_TRACESYSGOOD, PTRACE_O_TRACEFORK, PTRACE_O_TRACEVFORK)
(PTRACE_O_TRACECLONE, PTRACE_O_TRACEEXEC, PTRACE_O_TRACEVFORKDONE)
(PTRACE_O_TRACEEXIT, PTRACE_EVENT_FORK, PTRACE_EVENT_VFORK)
(PTRACE_EVENT_CLONE, PTRACE_EVENT_EXEC, PTRACE_EVENT_VFORK_DONE)
(PTRACE_EVENT_EXIT, __WALL): Provide default definitions.
(stopped_pids, thread_db_active, must_set_ptrace_flags): New variables.
(using_threads): Always set to 1.
(handle_extended_wait): New.
(add_process): Do not set TID.
(linux_create_inferior): Set must_set_ptrace_flags.
(linux_attach_lwp): Remove TID argument. Do not check using_threads.
Use PTRACE_SETOPTIONS. Call new_thread_notify. Update all callers.
(linux_thread_alive): Rename TID argument to LWPID.
(linux_wait_for_process): Handle unknown processes. Do not use TID.
(linux_wait_for_event): Do not use TID or check using_threads. Update
call to dead_thread_notify. Call handle_extended_wait.
(linux_create_inferior): Use PTRACE_SETOPTIONS.
(send_sigstop): Delete sigstop_sent.
(wait_for_sigstop): Avoid TID.
(linux_supports_tracefork_flag, linux_tracefork_child, my_waitpid)
(linux_test_for_tracefork): New.
(linux_lookup_signals): Use thread_db_active and
linux_supports_tracefork_flag.
(initialize_low): Use thread_db_active and linux_test_for_tracefork.
* linux-low.h (get_process_thread): Avoid TID.
(struct process_ifo): Move thread_known and tid to the end. Remove
sigstop_sent.
(linux_attach_lwp, thread_db_init): Update prototypes.
* server.h (change_inferior_id): Delete prototype.
(add_pid_to_list, pull_pid_from_list): New prototypes.
* thread-db.c (thread_db_use_events): New.
(find_first_thread): Rename to...
(find_one_thread): ...this. Update callers and messages. Do not
call fatal. Check thread_db_use_events. Do not call
change_inferior_id or new_thread_notify.
(maybe_attach_thread): Update. Do not call new_thread_notify.
(thread_db_init): Set thread_db_use_events. Check use_events.
* utils.c (fatal, warning): Correct message prefix.
2007-10-24 04:05:03 +08:00
|
|
|
fprintf (stderr, "gdbserver: ");
|
2002-02-14 14:21:24 +08:00
|
|
|
vfprintf (stderr, string, args);
|
|
|
|
fprintf (stderr, "\n");
|
|
|
|
va_end (args);
|
|
|
|
}
|
Add h/w watchpoint support to x86-linux, win32-i386.
* Makefile.in (SFILES): Add i386-low.c
(i386_low_h): Define.
(i386-low.o): Add dependencies.
(linux-x86-low.o): Add i386-low.h dependency.
(win32-i386-low.o): Ditto.
* i386-low.c: New file.
* i386-low.h: New file.
* configure.srv (i[34567]86-*-cygwin*): Add i386-low.o to srv_tgtobj.
(i[34567]86-*-linux*, i[34567]86-*-mingw*, x86_64-*-linux*): Ditto.
* linux-low.c (linux_add_process): Initialize arch_private.
(linux_remove_process): Free arch_private.
(add_lwp): Initialize arch_private.
(delete_lwp): Free arch_private.
(linux_resume_one_lwp): Call the_low_target.prepare_to_resume if
provided.
* linux-low.h (process_info_private): New member arch_private.
(lwp_info): New member arch_private.
(linux_target_ops): New members new_process, new_thread,
prepare_to_resume.
(ptid_of): New macro.
* linux-x86-low.c: Include stddef.h, i386-low.h.
(arch_process_info): New struct.
(arch_lwp_info): New struct.
(x86_linux_dr_get, x86_linux_dr_set): New functions.
(i386_dr_low_set_addr, i386_dr_low_set_control): New functions.
(i386_dr_low_get_status): New function.
(x86_insert_point, x86_remove_point): New functions.
(x86_stopped_by_watchpoint): New function.
(x86_stopped_data_address): New function.
(x86_linux_new_process, x86_linux_new_thread): New functions.
(x86_linux_prepare_to_resume): New function.
(the_low_target): Add entries for insert_point, remove_point,
stopped_by_watchpoint, stopped_data_address, new_process, new_thread,
prepare_to_resume.
* server.c (debug_hw_points): New global.
(monitor_show_help): Document set debug-hw-points.
(handle_query): Process "set debug-hw-points".
* server.h (debug_hw_points): Declare.
(paddress): Declare.
* utils.c (NUMCELLS, CELLSIZE): New macros.
(get_sell, xsnprintf, paddress): New functions.
* win32-arm-low.c (the_low_target): Add entries for insert_point,
remove_point, stopped_by_watchpoint, stopped_data_address.
* win32-i386-low.c: Include i386-low.h.
(debug_reg_state): Replaces dr.
(i386_dr_low_set_addr, i386_dr_low_set_control): New functions.
(i386_dr_low_get_status): New function.
(i386_insert_point, i386_remove_point): New functions.
(i386_stopped_by_watchpoint): New function.
(i386_stopped_data_address): New function.
(i386_initial_stuff): Update.
(get_thread_context,set_thread_context,i386_thread_added): Update.
(the_low_target): Add entries for insert_point,
remove_point, stopped_by_watchpoint, stopped_data_address.
* win32-low.c (win32_insert_watchpoint): New function.
(win32_remove_watchpoint): New function.
(win32_stopped_by_watchpoint): New function.
(win32_stopped_data_address): New function.
(win32_target_ops): Add entries for insert_watchpoint,
remove_watchpoint, stopped_by_watchpoint, stopped_data_address.
* win32-low.h (win32_target_ops): New members insert_point,
remove_point, stopped_by_watchpoint, stopped_data_address.
2009-07-01 00:35:25 +08:00
|
|
|
|
|
|
|
/* Temporary storage using circular buffer. */
|
|
|
|
#define NUMCELLS 4
|
|
|
|
#define CELLSIZE 50
|
|
|
|
|
|
|
|
/* Return the next entry in the circular buffer. */
|
|
|
|
|
|
|
|
static char *
|
|
|
|
get_cell (void)
|
|
|
|
{
|
|
|
|
static char buf[NUMCELLS][CELLSIZE];
|
|
|
|
static int cell = 0;
|
|
|
|
if (++cell >= NUMCELLS)
|
|
|
|
cell = 0;
|
|
|
|
return buf[cell];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Stdarg wrapper around vsnprintf.
|
|
|
|
SIZE is the size of the buffer pointed to by STR. */
|
|
|
|
|
|
|
|
static int
|
|
|
|
xsnprintf (char *str, size_t size, const char *format, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
va_start (args, format);
|
|
|
|
ret = vsnprintf (str, size, format, args);
|
|
|
|
va_end (args);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 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)
|
|
|
|
{
|
|
|
|
char *str = get_cell ();
|
|
|
|
xsnprintf (str, CELLSIZE, "%lx", (long) addr);
|
|
|
|
return str;
|
|
|
|
}
|