* top.c (gdb_readline): Allow CRLF line termination on systems

which define CRLF_SOURCE_FILES.
* win32-nat.c: 1) Add thread support, 2) fix ability to attach to
a running process, and 3) implement limited support for cygwin
signals.
(thread_rec): New function.
(child_add_thread): Ditto.
(child_init_thread_list): Ditto.
(child_delete_thread): Ditto.
(do_child_fetch_inferior_registers): Ditto.
(do_child_store_inferior_registers): Ditto.
(handle_output_debug_string): Ditto.
(child_fetch_inferior_registers): Use do_* function to perform
operation.
(child_store_inferior_registers): Ditto.
(child_continue): Ditto.
(child_thread_alive): Ditto.
(cygwin_pid_to_str): Ditto.
(handle_load_dll): Reorganize, add first attempt at reading
dll names from attached processes.  Change info messages to provide
more information when dll is already loaded.
(handle_exception): Changes mandated by new thread-aware structures.
(child_wait): Track thread creation/destruction.  Handle cygwin
signals.
(child_create_inferior): Ditto.
(child_resume): Ditto.
(child_kill_inferior): Ditto.  Close child process handle to avoid a
handle leak.
(child_ops): Fill out child_ops fields that deal with threads.
* config/i386/tm-cygwin32.h: Declare function and macro needed
for converting a cygwin "pid" to a string.
* config/i386/xm-cygwin32.h: define HAVE_SIGSETMASK as 0 since
sigsetmask is not defined in cygwin.
This commit is contained in:
Christopher Faylor 1998-11-05 14:08:48 +00:00
parent 8015bd27ec
commit 3cee93ac7a
6 changed files with 2535 additions and 2290 deletions

View file

@ -1,3 +1,39 @@
Thu Nov 5 08:41:33 1998 Christopher Faylor <cgf@cygnus.com>
* top.c (gdb_readline): Allow CRLF line termination on systems
which define CRLF_SOURCE_FILES.
* win32-nat.c: 1) Add thread support, 2) fix ability to attach to
a running process, and 3) implement limited support for cygwin
signals.
(thread_rec): New function.
(child_add_thread): Ditto.
(child_init_thread_list): Ditto.
(child_delete_thread): Ditto.
(do_child_fetch_inferior_registers): Ditto.
(do_child_store_inferior_registers): Ditto.
(handle_output_debug_string): Ditto.
(child_fetch_inferior_registers): Use do_* function to perform
operation.
(child_store_inferior_registers): Ditto.
(child_continue): Ditto.
(child_thread_alive): Ditto.
(cygwin_pid_to_str): Ditto.
(handle_load_dll): Reorganize, add first attempt at reading
dll names from attached processes. Change info messages to provide
more information when dll is already loaded.
(handle_exception): Changes mandated by new thread-aware structures.
(child_wait): Track thread creation/destruction. Handle cygwin
signals.
(child_create_inferior): Ditto.
(child_resume): Ditto.
(child_kill_inferior): Ditto. Close child process handle to avoid a
handle leak.
(child_ops): Fill out child_ops fields that deal with threads.
* config/i386/tm-cygwin32.h: Declare function and macro needed
for converting a cygwin "pid" to a string.
* config/i386/xm-cygwin32.h: define HAVE_SIGSETMASK as 0 since
sigsetmask is not defined in cygwin.
Thu Nov 5 08:38:18 1998 Christopher Faylor <cgf@cygnus.com> Thu Nov 5 08:38:18 1998 Christopher Faylor <cgf@cygnus.com>
* win32-nat.c: Remove obsolete PPC conditionals. * win32-nat.c: Remove obsolete PPC conditionals.

View file

@ -119,7 +119,9 @@ double_to_i387 PARAMS ((char *, char *));
#define NAMES_HAVE_UNDERSCORE #define NAMES_HAVE_UNDERSCORE
#define IN_SOLIB_CALL_TRAMPOLINE(pc, name) skip_trampoline_code (pc, name) #define IN_SOLIB_CALL_TRAMPOLINE(pc, name) skip_trampoline_code (pc, name)
#define SKIP_TRAMPOLINE_CODE(pc) skip_trampoline_code (pc, 0) #define SKIP_TRAMPOLINE_CODE(pc) skip_trampoline_code (pc, 0)
extern CORE_ADDR skip_trampoline_code PARAMS ((CORE_ADDR pc, char *name)); extern CORE_ADDR skip_trampoline_code PARAMS ((CORE_ADDR pc, char *name));
extern char *cygwin_pid_to_str PARAMS ((int pid));
#define target_pid_to_str(PID) cygwin_pid_to_str (PID)

View file

@ -34,3 +34,5 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
/* Define this if source files use \r\n rather than just \n. */ /* Define this if source files use \r\n rather than just \n. */
#define CRLF_SOURCE_FILES #define CRLF_SOURCE_FILES
#define HAVE_SIGSETMASK 0

File diff suppressed because it is too large Load diff

View file

@ -1447,7 +1447,15 @@ gdb_readline (prrompt)
} }
if (c == '\n') if (c == '\n')
#ifndef CRLF_SOURCE_FILES
break; break;
#else
{
if (input_index > 0 && result[input_index - 1] == '\r')
input_index--;
break;
}
#endif
result[input_index++] = c; result[input_index++] = c;
while (input_index >= result_size) while (input_index >= result_size)

View file

@ -21,7 +21,7 @@
/* by Steve Chamberlain, sac@cygnus.com */ /* by Steve Chamberlain, sac@cygnus.com */
/* We assume we're being built with and will be used for cygwin32. */ /* We assume we're being built with and will be used for cygwin. */
#include "defs.h" #include "defs.h"
#include "frame.h" /* required by inferior.h */ #include "frame.h" /* required by inferior.h */
@ -50,6 +50,10 @@
#include <sys/param.h> #include <sys/param.h>
#include <unistd.h> #include <unistd.h>
/* The string sent by cygwin when it processes a signal.
FIXME: This should be in a cygwin include file. */
#define CYGWIN_SIGNAL_STRING "cygwin: signal"
#define CHECK(x) check (x, __FILE__,__LINE__) #define CHECK(x) check (x, __FILE__,__LINE__)
#define DEBUG_EXEC(x) if (debug_exec) printf x #define DEBUG_EXEC(x) if (debug_exec) printf x
#define DEBUG_EVENTS(x) if (debug_events) printf x #define DEBUG_EVENTS(x) if (debug_events) printf x
@ -60,18 +64,32 @@
extern struct target_ops child_ops; extern struct target_ops child_ops;
static void child_stop PARAMS ((void)); static void child_stop PARAMS ((void));
static int child_thread_alive PARAMS ((int));
/* The most recently read context. Inspect ContextFlags to see what static int last_sig = 0; /* Set if a signal was received from the
bits are valid. */ debugged process */
static CONTEXT context; /* Thread information structure used to track information that is
not available in gdb's thread structure. */
typedef struct thread_info_struct
{
struct thread_info_struct *next;
DWORD id;
HANDLE h;
char *name;
int suspend_count;
CONTEXT context;
} thread_info;
static thread_info thread_head = {NULL};
/* The process and thread handles for the above context. */ /* The process and thread handles for the above context. */
static HANDLE current_process; static DEBUG_EVENT current_event; /* The current debug event from
static HANDLE current_thread; WaitForDebugEvent */
static int current_process_id; static HANDLE current_process_handle; /* Currently executing process */
static int current_thread_id; static thread_info *current_thread; /* Info on currently selected thread */
static DWORD main_thread_id; /* Thread ID of the main thread */
/* Counts of things. */ /* Counts of things. */
static int exception_count = 0; static int exception_count = 0;
@ -98,40 +116,33 @@ static int debug_exceptions = 0; /* show target exceptions */
the other regs of the group, and then we copy the info in and set the other regs of the group, and then we copy the info in and set
out bit. */ out bit. */
struct regmappings #define context_offset(x) ((int)&(((CONTEXT *)NULL)->x))
{ static const int mappings[] =
char *incontext;
int mask;
};
static const struct regmappings mappings[] =
{ {
#ifdef i386 context_offset(Eax),
{(char *) &context.Eax, CONTEXT_INTEGER}, context_offset(Ecx),
{(char *) &context.Ecx, CONTEXT_INTEGER}, context_offset(Edx),
{(char *) &context.Edx, CONTEXT_INTEGER}, context_offset(Ebx),
{(char *) &context.Ebx, CONTEXT_INTEGER}, context_offset(Esp),
{(char *) &context.Esp, CONTEXT_CONTROL}, context_offset(Ebp),
{(char *) &context.Ebp, CONTEXT_CONTROL}, context_offset(Esi),
{(char *) &context.Esi, CONTEXT_INTEGER}, context_offset(Edi),
{(char *) &context.Edi, CONTEXT_INTEGER}, context_offset(Eip),
{(char *) &context.Eip, CONTEXT_CONTROL}, context_offset(EFlags),
{(char *) &context.EFlags, CONTEXT_CONTROL}, context_offset(SegCs),
{(char *) &context.SegCs, CONTEXT_SEGMENTS}, context_offset(SegSs),
{(char *) &context.SegSs, CONTEXT_SEGMENTS}, context_offset(SegDs),
{(char *) &context.SegDs, CONTEXT_SEGMENTS}, context_offset(SegEs),
{(char *) &context.SegEs, CONTEXT_SEGMENTS}, context_offset(SegFs),
{(char *) &context.SegFs, CONTEXT_SEGMENTS}, context_offset(SegGs),
{(char *) &context.SegGs, CONTEXT_SEGMENTS}, context_offset(FloatSave.RegisterArea[0 * 10]),
{&context.FloatSave.RegisterArea[0 * 10], CONTEXT_FLOATING_POINT}, context_offset(FloatSave.RegisterArea[1 * 10]),
{&context.FloatSave.RegisterArea[1 * 10], CONTEXT_FLOATING_POINT}, context_offset(FloatSave.RegisterArea[2 * 10]),
{&context.FloatSave.RegisterArea[2 * 10], CONTEXT_FLOATING_POINT}, context_offset(FloatSave.RegisterArea[3 * 10]),
{&context.FloatSave.RegisterArea[3 * 10], CONTEXT_FLOATING_POINT}, context_offset(FloatSave.RegisterArea[4 * 10]),
{&context.FloatSave.RegisterArea[4 * 10], CONTEXT_FLOATING_POINT}, context_offset(FloatSave.RegisterArea[5 * 10]),
{&context.FloatSave.RegisterArea[5 * 10], CONTEXT_FLOATING_POINT}, context_offset(FloatSave.RegisterArea[6 * 10]),
{&context.FloatSave.RegisterArea[6 * 10], CONTEXT_FLOATING_POINT}, context_offset(FloatSave.RegisterArea[7 * 10]),
{&context.FloatSave.RegisterArea[7 * 10], CONTEXT_FLOATING_POINT},
#endif
}; };
/* This vector maps the target's idea of an exception (extracted /* This vector maps the target's idea of an exception (extracted
@ -153,6 +164,94 @@ static const struct xlate_exception
{EXCEPTION_SINGLE_STEP, TARGET_SIGNAL_TRAP}, {EXCEPTION_SINGLE_STEP, TARGET_SIGNAL_TRAP},
{-1, -1}}; {-1, -1}};
/* Find a thread record given a thread id.
If get_context then also retrieve the context for this
thread. */
static thread_info *
thread_rec (DWORD id, int get_context)
{
thread_info *th;
for (th = &thread_head; (th = th->next) != NULL; )
if (th->id == id)
{
if (!th->suspend_count && get_context)
{
if (get_context > 0)
th->suspend_count = SuspendThread (th->h) + 1;
else if (get_context < 0)
th->suspend_count = -1;
th->context.ContextFlags = CONTEXT_DEBUGGER;
GetThreadContext (th->h, &th->context);
}
return th;
}
return NULL;
}
/* Add a thread to the thread list */
static thread_info *
child_add_thread(DWORD id, HANDLE h)
{
thread_info *th;
if ((th = thread_rec (id, FALSE)))
return th;
th = (thread_info *) xmalloc (sizeof (*th));
memset(th, 0, sizeof (*th));
th->id = id;
th->h = h;
th->next = thread_head.next;
thread_head.next = th;
add_thread (id);
return th;
}
/* Clear out any old thread list and reintialize it to a
pristine state. */
static void
child_init_thread_list ()
{
thread_info *th = &thread_head;
DEBUG_EVENTS (("gdb: child_init_thread_list\n"));
init_thread_list ();
while (th->next != NULL)
{
thread_info *here = th->next;
th->next = here->next;
(void) CloseHandle (here->h);
free (here);
}
}
/* Delete a thread from the list of threads */
static void
child_delete_thread (DWORD id)
{
thread_info *th;
if (info_verbose)
printf_unfiltered ("[Deleting %s]\n", target_pid_to_str (id));
delete_thread (id);
for (th = &thread_head;
th->next != NULL && th->next->id != id;
th = th->next)
continue;
if (th->next != NULL)
{
thread_info *here = th->next;
th->next = here->next;
CloseHandle (here->h);
free (here);
}
}
static void static void
check (BOOL ok, const char *file, int line) check (BOOL ok, const char *file, int line)
{ {
@ -161,64 +260,103 @@ check (BOOL ok, const char *file, int line)
} }
static void static void
child_fetch_inferior_registers (int r) do_child_fetch_inferior_registers (int r)
{ {
if (r < 0) if (r >= 0)
{ supply_register (r, ((char *) &current_thread->context) + mappings[r]);
for (r = 0; r < NUM_REGS; r++)
child_fetch_inferior_registers (r);
}
else else
{ {
supply_register (r, mappings[r].incontext); for (r = 0; r < NUM_REGS; r++)
do_child_fetch_inferior_registers (r);
} }
} }
static void static void
child_store_inferior_registers (int r) child_fetch_inferior_registers (int r)
{ {
if (r < 0) current_thread = thread_rec (inferior_pid, TRUE);
{ do_child_fetch_inferior_registers (r);
for (r = 0; r < NUM_REGS; r++) }
child_store_inferior_registers (r);
} static void
do_child_store_inferior_registers (int r)
{
if (r >= 0)
read_register_gen (r, ((char *) &current_thread->context) + mappings[r]);
else else
{ {
read_register_gen (r, mappings[r].incontext); for (r = 0; r < NUM_REGS; r++)
do_child_store_inferior_registers (r);
} }
} }
/* Store a new register value into the current thread context */
static void
child_store_inferior_registers (int r)
{
current_thread = thread_rec (inferior_pid, TRUE);
do_child_store_inferior_registers (r);
}
/* Wait for child to do something. Return pid of child, or -1 in case /* Wait for child to do something. Return pid of child, or -1 in case
of error; store status through argument pointer OURSTATUS. */ of error; store status through argument pointer OURSTATUS. */
static int static int
handle_load_dll (char *eventp) handle_load_dll (char *dummy)
{ {
DEBUG_EVENT * event = (DEBUG_EVENT *)eventp; LOAD_DLL_DEBUG_INFO * event = &current_event.u.LoadDll;
DWORD dll_name_ptr; DWORD dll_name_ptr;
DWORD done; DWORD done;
char dll_buf[MAX_PATH + 1];
char *p, *dll_name = NULL, *dll_basename;
struct objfile *objfile;
MEMORY_BASIC_INFORMATION minfo;
ReadProcessMemory (current_process, dll_buf[0] = dll_buf[sizeof(dll_buf) - 1] = '\0';
(DWORD) event->u.LoadDll.lpImageName,
/* The following code attempts to find the name of the dll by reading the
name from the processes memory. Unfortunately it doesn't work right.
Doing this the "right way" for Windows is very difficult. FIXME */
#ifdef DOESNT_WORK
memset (&minfo, 0, sizeof minfo);
if (VirtualQueryEx (current_process_handle, (LPCVOID) event->lpBaseOfDll,
&minfo, sizeof(minfo)) && minfo.BaseAddress) {
DWORD len;
IMAGE_DOS_HEADER *hmm0 = (IMAGE_DOS_HEADER *) minfo.BaseAddress;
HMODULE hmm = (HMODULE) (((DWORD) hmm0) + hmm0->e_lfanew);
if ((len = GetModuleFileName (hmm, dll_buf, MAX_PATH)))
{
dll_name = dll_buf;
dll_name[len] = '\0';
}
}
#endif
/* Attempt to read the name of the dll that was detected.
This is documented to work only when actively debugging
a program. It will not work for attached processes. */
if (dll_name == NULL || *dll_name == '\0')
{
int size = event->fUnicode ? sizeof (WCHAR) : sizeof (char);
int len = 0;
char b[2];
ReadProcessMemory (current_process_handle,
(LPCVOID) event->lpImageName,
(char *) &dll_name_ptr, (char *) &dll_name_ptr,
sizeof (dll_name_ptr), &done); sizeof (dll_name_ptr), &done);
/* See if we could read the address of a string, and that the /* See if we could read the address of a string, and that the
address isn't null. */ address isn't null. */
if (done == sizeof (dll_name_ptr) && dll_name_ptr) if (done != sizeof (dll_name_ptr) || !dll_name_ptr)
{ return 1;
char *dll_name, *dll_basename;
struct objfile *objfile;
char unix_dll_name[MAX_PATH];
int size = event->u.LoadDll.fUnicode ? sizeof (WCHAR) : sizeof (char);
int len = 0;
char b[2];
do do
{ {
ReadProcessMemory (current_process, ReadProcessMemory (current_process_handle,
dll_name_ptr + len * size, (LPCVOID) (dll_name_ptr + len * size),
&b, &b,
size, size,
&done); &done);
@ -228,11 +366,11 @@ handle_load_dll (char *eventp)
dll_name = alloca (len); dll_name = alloca (len);
if (event->u.LoadDll.fUnicode) if (event->fUnicode)
{ {
WCHAR *unicode_dll_name = (WCHAR *) alloca (len * sizeof (WCHAR)); WCHAR *unicode_dll_name = (WCHAR *) alloca (len * sizeof (WCHAR));
ReadProcessMemory (current_process, ReadProcessMemory (current_process_handle,
dll_name_ptr, (LPCVOID) dll_name_ptr,
unicode_dll_name, unicode_dll_name,
len * sizeof (WCHAR), len * sizeof (WCHAR),
&done); &done);
@ -243,211 +381,273 @@ handle_load_dll (char *eventp)
} }
else else
{ {
ReadProcessMemory (current_process, ReadProcessMemory (current_process_handle,
dll_name_ptr, (LPCVOID) dll_name_ptr,
dll_name, dll_name,
len, len,
&done); &done);
} }
}
/* FIXME: Can we delete this call? */ if (!dll_name)
cygwin32_conv_to_posix_path (dll_name, unix_dll_name); return 1;
while ((p = strchr (dll_name, '\\')))
*p = '/';
/* FIXME!! It would be nice to define one symbol which pointed to the /* FIXME!! It would be nice to define one symbol which pointed to the
front of the dll if we can't find any symbols. */ front of the dll if we can't find any symbols. */
if (!(dll_basename = strrchr(dll_name, '\\'))) if (!(dll_basename = strrchr(dll_name, '/')))
dll_basename = strrchr(dll_name, '/'); dll_basename = dll_name;
else
dll_basename++;
ALL_OBJFILES(objfile) ALL_OBJFILES(objfile)
{ {
char *objfile_basename; char *objfile_basename;
if (!(objfile_basename = strrchr(objfile->name, '\\')))
objfile_basename = strrchr(objfile->name, '/'); objfile_basename = strrchr(objfile->name, '/');
if (dll_basename && objfile_basename && if (objfile_basename &&
strcmp(dll_basename+1, objfile_basename+1) == 0) strcmp(dll_basename, objfile_basename + 1) == 0)
{ {
printf_unfiltered ("%s (symbols previously loaded)\n", printf_unfiltered ("%x:%s (symbols previously loaded)\n",
dll_basename + 1); event->lpBaseOfDll, dll_name);
return 1; goto out;
} }
} }
context.ContextFlags = CONTEXT_FULL | CONTEXT_FLOATING_POINT;
GetThreadContext (current_thread, &context);
/* The symbols in a dll are offset by 0x1000, which is the /* The symbols in a dll are offset by 0x1000, which is the
the offset from 0 of the first byte in an image - because the offset from 0 of the first byte in an image - because
of the file header and the section alignment. of the file header and the section alignment.
FIXME: Is this the real reason that we need the 0x1000 ? */ FIXME: Is this the real reason that we need the 0x1000 ? */
printf_unfiltered ("%x:%s", event->lpBaseOfDll, dll_name);
symbol_file_add (dll_name, 0, (int) event->lpBaseOfDll + 0x1000, 0, 0, 0);
printf_unfiltered ("\n");
symbol_file_add (unix_dll_name, 0, out:
(int) event->u.LoadDll.lpBaseOfDll + 0x1000, 0, 0, 0);
printf_unfiltered ("%x:%s\n", event->u.LoadDll.lpBaseOfDll,
unix_dll_name);
}
return 1; return 1;
} }
/* Handle DEBUG_STRING output from child process.
Cygwin prepends its messages with a "cygwin:". Interpret this as
a Cygwin signal. Otherwise just print the string as a warning. */
static int
handle_output_debug_string (struct target_waitstatus *ourstatus)
{
char *s;
int gotasig = FALSE;
if (!target_read_string
((CORE_ADDR) current_event.u.DebugString.lpDebugStringData, &s, 1024, 0)
|| !s || !*s)
return gotasig;
if (strncmp(s, CYGWIN_SIGNAL_STRING, sizeof(CYGWIN_SIGNAL_STRING) - 1))
{
warning (s);
}
else
{
char *p;
/*last_sig = */strtol(s + sizeof(CYGWIN_SIGNAL_STRING) - 1, &p, 0);
if (gotasig = (ourstatus->value.sig = target_signal_from_host (last_sig)))
ourstatus->kind = TARGET_WAITKIND_STOPPED;
}
free (s);
return gotasig;
}
static int static int
handle_exception (DEBUG_EVENT * event, struct target_waitstatus *ourstatus) handle_exception (struct target_waitstatus *ourstatus)
{ {
int i; int i;
int done = 0; int done = 0;
thread_info *th;
ourstatus->kind = TARGET_WAITKIND_STOPPED; ourstatus->kind = TARGET_WAITKIND_STOPPED;
/* Record the context of the current thread */
th = thread_rec (current_event.dwThreadId, -1);
switch (event->u.Exception.ExceptionRecord.ExceptionCode) switch (current_event.u.Exception.ExceptionRecord.ExceptionCode)
{ {
case EXCEPTION_ACCESS_VIOLATION: case EXCEPTION_ACCESS_VIOLATION:
DEBUG_EXCEPT (("gdb: Target exception ACCESS_VIOLATION at 0x%08x\n", DEBUG_EXCEPT (("gdb: Target exception ACCESS_VIOLATION at 0x%08x\n",
event->u.Exception.ExceptionRecord.ExceptionAddress)); current_event.u.Exception.ExceptionRecord.ExceptionAddress));
ourstatus->value.sig = TARGET_SIGNAL_SEGV; ourstatus->value.sig = TARGET_SIGNAL_SEGV;
break; break;
case STATUS_STACK_OVERFLOW: case STATUS_STACK_OVERFLOW:
DEBUG_EXCEPT (("gdb: Target exception STACK_OVERFLOW at 0x%08x\n", DEBUG_EXCEPT (("gdb: Target exception STACK_OVERFLOW at 0x%08x\n",
event->u.Exception.ExceptionRecord.ExceptionAddress)); current_event.u.Exception.ExceptionRecord.ExceptionAddress));
ourstatus->value.sig = TARGET_SIGNAL_SEGV; ourstatus->value.sig = TARGET_SIGNAL_SEGV;
break; break;
case EXCEPTION_BREAKPOINT: case EXCEPTION_BREAKPOINT:
DEBUG_EXCEPT (("gdb: Target exception BREAKPOINT at 0x%08x\n", DEBUG_EXCEPT (("gdb: Target exception BREAKPOINT at 0x%08x\n",
event->u.Exception.ExceptionRecord.ExceptionAddress)); current_event.u.Exception.ExceptionRecord.ExceptionAddress));
ourstatus->value.sig = TARGET_SIGNAL_TRAP; ourstatus->value.sig = TARGET_SIGNAL_TRAP;
break; break;
case DBG_CONTROL_C: case DBG_CONTROL_C:
DEBUG_EXCEPT (("gdb: Target exception CONTROL_C at 0x%08x\n", DEBUG_EXCEPT (("gdb: Target exception CONTROL_C at 0x%08x\n",
event->u.Exception.ExceptionRecord.ExceptionAddress)); current_event.u.Exception.ExceptionRecord.ExceptionAddress));
ourstatus->value.sig = TARGET_SIGNAL_INT; ourstatus->value.sig = TARGET_SIGNAL_INT;
/* User typed CTRL-C. Continue with this status */
last_sig = SIGINT; /* FIXME - should check pass state */
break; break;
case EXCEPTION_SINGLE_STEP: case EXCEPTION_SINGLE_STEP:
DEBUG_EXCEPT (("gdb: Target exception SINGLE_STEP at 0x%08x\n", DEBUG_EXCEPT (("gdb: Target exception SINGLE_STEP at 0x%08x\n",
event->u.Exception.ExceptionRecord.ExceptionAddress)); current_event.u.Exception.ExceptionRecord.ExceptionAddress));
ourstatus->value.sig = TARGET_SIGNAL_TRAP; ourstatus->value.sig = TARGET_SIGNAL_TRAP;
break; break;
default: default:
/* This may be a structured exception handling exception. In /* This may be a structured exception handling exception. In
that case, we want to let the program try to handle it, and that case, we want to let the program try to handle it, and
only break if we see the exception a second time. */ only break if we see the exception a second time. */
if (event->u.Exception.dwFirstChance) if (current_event.u.Exception.dwFirstChance)
return 0; return 0;
printf_unfiltered ("gdb: unknown target exception 0x%08x at 0x%08x\n", printf_unfiltered ("gdb: unknown target exception 0x%08x at 0x%08x\n",
event->u.Exception.ExceptionRecord.ExceptionCode, current_event.u.Exception.ExceptionRecord.ExceptionCode,
event->u.Exception.ExceptionRecord.ExceptionAddress); current_event.u.Exception.ExceptionRecord.ExceptionAddress);
ourstatus->value.sig = TARGET_SIGNAL_UNKNOWN; ourstatus->value.sig = TARGET_SIGNAL_UNKNOWN;
break; break;
} }
context.ContextFlags = CONTEXT_FULL | CONTEXT_FLOATING_POINT;
GetThreadContext (current_thread, &context);
exception_count++; exception_count++;
return 1; return 1;
} }
/* Resume all artificially suspended threads if we are continuing
execution */
static BOOL
child_continue (DWORD continue_status, int id)
{
int i;
thread_info *th;
BOOL res;
DEBUG_EVENTS (("ContinueDebugEvent (cpid=%d, ctid=%d, DBG_CONTINUE);\n",
current_event.dwProcessId, current_event.dwThreadId));
if (res = ContinueDebugEvent (current_event.dwProcessId,
current_event.dwThreadId,
continue_status))
for (th = &thread_head; (th = th->next) != NULL; )
if (((id == -1) || (id == th->id)) && th->suspend_count)
{
for (i = 0; i < th->suspend_count; i++)
(void) ResumeThread (th->h);
th->suspend_count = 0;
}
return res;
}
static int static int
child_wait (int pid, struct target_waitstatus *ourstatus) child_wait (int pid, struct target_waitstatus *ourstatus)
{ {
/* We loop when we get a non-standard exception rather than return /* We loop when we get a non-standard exception rather than return
with a SPURIOUS because resume can try and step or modify things, with a SPURIOUS because resume can try and step or modify things,
which needs a current_thread. But some of these exceptions mark which needs a current_thread->h. But some of these exceptions mark
the birth or death of threads, which mean that the current thread the birth or death of threads, which mean that the current thread
isn't necessarily what you think it is. */ isn't necessarily what you think it is. */
while (1) while (1)
{ {
DEBUG_EVENT event;
BOOL t = WaitForDebugEvent (&event, INFINITE);
char *p;
DWORD continue_status; DWORD continue_status;
BOOL t = WaitForDebugEvent (&current_event, INFINITE);
char *p;
thread_info *th;
int sig;
event_count++; event_count++;
current_thread_id = event.dwThreadId;
current_process_id = event.dwProcessId;
continue_status = DBG_CONTINUE; continue_status = DBG_CONTINUE;
switch (event.dwDebugEventCode) switch (current_event.dwDebugEventCode)
{ {
case CREATE_THREAD_DEBUG_EVENT: case CREATE_THREAD_DEBUG_EVENT:
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%x code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"CREATE_THREAD_DEBUG_EVENT")); "CREATE_THREAD_DEBUG_EVENT"));
/* Record the existence of this thread */
child_add_thread (current_event.dwThreadId,
current_event.u.CreateThread.hThread);
if (info_verbose)
printf_unfiltered ("[New %s]\n",
target_pid_to_str (current_event.dwThreadId));
break; break;
case EXIT_THREAD_DEBUG_EVENT: case EXIT_THREAD_DEBUG_EVENT:
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"EXIT_THREAD_DEBUG_EVENT")); "EXIT_THREAD_DEBUG_EVENT"));
child_delete_thread (current_event.dwThreadId);
break; break;
case CREATE_PROCESS_DEBUG_EVENT: case CREATE_PROCESS_DEBUG_EVENT:
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"CREATE_PROCESS_DEBUG_EVENT")); "CREATE_PROCESS_DEBUG_EVENT"));
current_process_handle = current_event.u.CreateProcessInfo.hProcess;
main_thread_id = inferior_pid = current_event.dwThreadId;
/* Add the main thread */
current_thread = child_add_thread (inferior_pid,
current_event.u.CreateProcessInfo.hThread);
break; break;
case EXIT_PROCESS_DEBUG_EVENT: case EXIT_PROCESS_DEBUG_EVENT:
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"EXIT_PROCESS_DEBUG_EVENT")); "EXIT_PROCESS_DEBUG_EVENT"));
ourstatus->kind = TARGET_WAITKIND_EXITED; ourstatus->kind = TARGET_WAITKIND_EXITED;
ourstatus->value.integer = event.u.ExitProcess.dwExitCode; ourstatus->value.integer = current_event.u.ExitProcess.dwExitCode;
CloseHandle (current_process); CloseHandle (current_process_handle);
CloseHandle (current_thread); return current_event.dwProcessId;
return current_process_id;
break; break;
case LOAD_DLL_DEBUG_EVENT: case LOAD_DLL_DEBUG_EVENT:
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"LOAD_DLL_DEBUG_EVENT")); "LOAD_DLL_DEBUG_EVENT"));
catch_errors (handle_load_dll, catch_errors (handle_load_dll, NULL, "", RETURN_MASK_ALL);
(char*) &event,
"\n[failed reading symbols from DLL]\n",
RETURN_MASK_ALL);
registers_changed(); /* mark all regs invalid */ registers_changed(); /* mark all regs invalid */
break; break;
case UNLOAD_DLL_DEBUG_EVENT: case UNLOAD_DLL_DEBUG_EVENT:
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"UNLOAD_DLL_DEBUG_EVENT")); "UNLOAD_DLL_DEBUG_EVENT"));
break; /* FIXME: don't know what to do here */ break; /* FIXME: don't know what to do here */
case EXCEPTION_DEBUG_EVENT: case EXCEPTION_DEBUG_EVENT:
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"EXCEPTION_DEBUG_EVENT")); "EXCEPTION_DEBUG_EVENT"));
if (handle_exception (&event, ourstatus)) if (handle_exception (ourstatus))
return current_process_id; return current_event.dwThreadId;
continue_status = DBG_EXCEPTION_NOT_HANDLED; continue_status = DBG_EXCEPTION_NOT_HANDLED;
break; break;
case OUTPUT_DEBUG_STRING_EVENT: /* message from the kernel */ case OUTPUT_DEBUG_STRING_EVENT: /* message from the kernel */
DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n", DEBUG_EVENTS (("gdb: kernel event for pid=%d tid=%d code=%s)\n",
event.dwProcessId, event.dwThreadId, current_event.dwProcessId, current_event.dwThreadId,
"OUTPUT_DEBUG_STRING_EVENT")); "OUTPUT_DEBUG_STRING_EVENT"));
if (target_read_string if (handle_output_debug_string (ourstatus))
((CORE_ADDR) event.u.DebugString.lpDebugStringData, return main_thread_id;
&p, 1024, 0) && p && *p)
{
warning(p);
free(p);
}
break; break;
default: default:
printf_unfiltered ("gdb: kernel event for pid=%d tid=%d\n", printf_unfiltered ("gdb: kernel event for pid=%d tid=%d\n",
event.dwProcessId, event.dwThreadId); current_event.dwProcessId,
current_event.dwThreadId);
printf_unfiltered (" unknown event code %d\n", printf_unfiltered (" unknown event code %d\n",
event.dwDebugEventCode); current_event.dwDebugEventCode);
break; break;
} }
DEBUG_EVENTS (("ContinueDebugEvent (cpid=%d, ctid=%d, DBG_CONTINUE);\n",
current_process_id, current_thread_id)); CHECK (child_continue (continue_status, -1));
CHECK (ContinueDebugEvent (current_process_id,
current_thread_id,
continue_status));
} }
} }
@ -463,9 +663,9 @@ child_attach (args, from_tty)
if (!args) if (!args)
error_no_arg ("process-id to attach"); error_no_arg ("process-id to attach");
current_process_id = strtoul (args, 0, 0); current_event.dwProcessId = strtoul (args, 0, 0);
ok = DebugActiveProcess (current_process_id); ok = DebugActiveProcess (current_event.dwProcessId);
if (!ok) if (!ok)
error ("Can't attach to process."); error ("Can't attach to process.");
@ -479,15 +679,14 @@ child_attach (args, from_tty)
if (exec_file) if (exec_file)
printf_unfiltered ("Attaching to program `%s', %s\n", exec_file, printf_unfiltered ("Attaching to program `%s', %s\n", exec_file,
target_pid_to_str (current_process_id)); target_pid_to_str (current_event.dwProcessId));
else else
printf_unfiltered ("Attaching to %s\n", printf_unfiltered ("Attaching to %s\n",
target_pid_to_str (current_process_id)); target_pid_to_str (current_event.dwProcessId));
gdb_flush (gdb_stdout); gdb_flush (gdb_stdout);
} }
inferior_pid = current_process_id;
push_target (&child_ops); push_target (&child_ops);
} }
@ -560,7 +759,7 @@ child_create_inferior (exec_file, allargs, env)
memset (&si, 0, sizeof (si)); memset (&si, 0, sizeof (si));
si.cb = sizeof (si); si.cb = sizeof (si);
cygwin32_conv_to_win32_path (exec_file, real_path); cygwin_conv_to_win32_path (exec_file, real_path);
flags = DEBUG_ONLY_THIS_PROCESS; flags = DEBUG_ONLY_THIS_PROCESS;
@ -606,9 +805,9 @@ child_create_inferior (exec_file, allargs, env)
len = strlen (conv_path_names[j]); len = strlen (conv_path_names[j]);
if (strncmp (conv_path_names[j], env[i], len) == 0) if (strncmp (conv_path_names[j], env[i], len) == 0)
{ {
if (cygwin32_posix_path_list_p (env[i] + len)) if (cygwin_posix_path_list_p (env[i] + len))
envlen += len envlen += len
+ cygwin32_posix_to_win32_path_list_buf_size (env[i] + len); + cygwin_posix_to_win32_path_list_buf_size (env[i] + len);
else else
envlen += strlen (env[i]) + 1; envlen += strlen (env[i]) + 1;
break; break;
@ -630,10 +829,10 @@ child_create_inferior (exec_file, allargs, env)
len = strlen (conv_path_names[j]); len = strlen (conv_path_names[j]);
if (strncmp (conv_path_names[j], env[i], len) == 0) if (strncmp (conv_path_names[j], env[i], len) == 0)
{ {
if (cygwin32_posix_path_list_p (env[i] + len)) if (cygwin_posix_path_list_p (env[i] + len))
{ {
memcpy (temp, env[i], len); memcpy (temp, env[i], len);
cygwin32_posix_to_win32_path_list (env[i] + len, temp + len); cygwin_posix_to_win32_path_list (env[i] + len, temp + len);
} }
else else
strcpy (temp, env[i]); strcpy (temp, env[i]);
@ -666,13 +865,12 @@ child_create_inferior (exec_file, allargs, env)
exception_count = 0; exception_count = 0;
event_count = 0; event_count = 0;
inferior_pid = pi.dwProcessId; current_process_handle = pi.hProcess;
current_process = pi.hProcess; current_event.dwProcessId = pi.dwProcessId;
current_thread = pi.hThread; memset (&current_event, 0, sizeof (current_event));
current_process_id = pi.dwProcessId; inferior_pid = current_event.dwThreadId = pi.dwThreadId;
current_thread_id = pi.dwThreadId;
push_target (&child_ops); push_target (&child_ops);
init_thread_list (); child_init_thread_list ();
init_wait_for_inferior (); init_wait_for_inferior ();
clear_proceed_status (); clear_proceed_status ();
target_terminal_init (); target_terminal_init ();
@ -687,9 +885,7 @@ child_create_inferior (exec_file, allargs, env)
static void static void
child_mourn_inferior () child_mourn_inferior ()
{ {
(void) ContinueDebugEvent (current_process_id, (void) child_continue (DBG_CONTINUE, -1);
current_thread_id,
DBG_CONTINUE);
unpush_target (&child_ops); unpush_target (&child_ops);
generic_mourn_inferior (); generic_mourn_inferior ();
} }
@ -714,14 +910,16 @@ child_xfer_memory (CORE_ADDR memaddr, char *our, int len,
{ {
DEBUG_MEM (("gdb: write target memory, %d bytes at 0x%08x\n", DEBUG_MEM (("gdb: write target memory, %d bytes at 0x%08x\n",
len, memaddr)); len, memaddr));
WriteProcessMemory (current_process, memaddr, our, len, &done); WriteProcessMemory (current_process_handle, (LPVOID) memaddr, our,
FlushInstructionCache (current_process, memaddr, len); len, &done);
FlushInstructionCache (current_process_handle, (LPCVOID) memaddr, len);
} }
else else
{ {
DEBUG_MEM (("gdb: read target memory, %d bytes at 0x%08x\n", DEBUG_MEM (("gdb: read target memory, %d bytes at 0x%08x\n",
len, memaddr)); len, memaddr));
ReadProcessMemory (current_process, memaddr, our, len, &done); ReadProcessMemory (current_process_handle, (LPCVOID) memaddr, our, len,
&done);
} }
return done; return done;
} }
@ -729,59 +927,60 @@ child_xfer_memory (CORE_ADDR memaddr, char *our, int len,
void void
child_kill_inferior (void) child_kill_inferior (void)
{ {
CHECK (TerminateProcess (current_process, 0)); CHECK (TerminateProcess (current_process_handle, 0));
for (;;) for (;;)
{ {
DEBUG_EVENT event; if (!child_continue (DBG_CONTINUE, -1))
if (!ContinueDebugEvent (current_process_id,
current_thread_id,
DBG_CONTINUE))
break; break;
if (!WaitForDebugEvent (&event, INFINITE)) if (!WaitForDebugEvent (&current_event, INFINITE))
break; break;
current_thread_id = event.dwThreadId; if (current_event.dwDebugEventCode == EXIT_PROCESS_DEBUG_EVENT)
current_process_id = event.dwProcessId;
if (event.dwDebugEventCode == EXIT_PROCESS_DEBUG_EVENT)
break; break;
} }
CHECK (CloseHandle (current_process)); CHECK (CloseHandle (current_process_handle));
CHECK (CloseHandle (current_thread));
/* this may fail in an attached process so don't check. */
(void) CloseHandle (current_thread->h);
target_mourn_inferior(); /* or just child_mourn_inferior? */ target_mourn_inferior(); /* or just child_mourn_inferior? */
} }
void void
child_resume (int pid, int step, enum target_signal signal) child_resume (int pid, int step, enum target_signal sig)
{ {
DEBUG_EXEC (("gdb: child_resume (pid=%d, step=%d, signal=%d);\n", int i;
pid, step, signal)); thread_info *th;
DWORD continue_status = last_sig > 0 && last_sig < NSIG ?
DBG_EXCEPTION_NOT_HANDLED : DBG_CONTINUE;
DEBUG_EXEC (("gdb: child_resume (pid=%d, step=%d, sig=%d);\n",
pid, step, sig));
/* Get context for currently selected thread */
th = thread_rec (current_event.dwThreadId, FALSE);
if (step) if (step)
{ {
#ifdef i386 #ifdef i386
/* Single step by setting t bit */ /* Single step by setting t bit */
child_fetch_inferior_registers (PS_REGNUM); child_fetch_inferior_registers (PS_REGNUM);
context.EFlags |= FLAG_TRACE_BIT; th->context.EFlags |= FLAG_TRACE_BIT;
#endif #endif
} }
if (context.ContextFlags) if (th->context.ContextFlags)
{ {
CHECK (SetThreadContext (current_thread, &context)); CHECK (SetThreadContext (th->h, &th->context));
context.ContextFlags = 0; th->context.ContextFlags = 0;
} }
if (signal) /* Allow continuing with the same signal that interrupted us.
{ Otherwise complain. */
fprintf_unfiltered (gdb_stderr, "Can't send signals to the child.\n"); if (sig && sig != last_sig)
} fprintf_unfiltered (gdb_stderr, "Can't send signals to the child. signal %d\n", sig);
DEBUG_EVENTS (("gdb: ContinueDebugEvent (cpid=%d, ctid=%d, DBG_CONTINUE);\n", last_sig = 0;
current_process_id, current_thread_id)); child_continue (continue_status, pid);
CHECK (ContinueDebugEvent (current_process_id,
current_thread_id,
DBG_CONTINUE));
} }
static void static void
@ -834,7 +1033,7 @@ static void init_child_ops(void)
child_ops.to_mourn_inferior = child_mourn_inferior; child_ops.to_mourn_inferior = child_mourn_inferior;
child_ops.to_can_run = child_can_run; child_ops.to_can_run = child_can_run;
child_ops.to_notice_signals = 0; child_ops.to_notice_signals = 0;
child_ops.to_thread_alive = 0; child_ops.to_thread_alive = child_thread_alive;
child_ops.to_stop = child_stop; child_ops.to_stop = child_stop;
child_ops.to_stratum = process_stratum; child_ops.to_stratum = process_stratum;
child_ops.DONT_USE = 0; child_ops.DONT_USE = 0;
@ -898,3 +1097,25 @@ _initialize_inftarg ()
add_target (&child_ops); add_target (&child_ops);
} }
/* Determine if the thread referenced by "pid" is alive
by "polling" it. If WaitForSingleObject returns WAIT_OBJECT_0
it means that the pid has died. Otherwise it is assumed to be alive. */
static int
child_thread_alive (int pid)
{
return WaitForSingleObject(thread_rec (pid, FALSE)->h, 0) == WAIT_OBJECT_0 ?
FALSE : TRUE;
}
/* Convert pid to printable format. */
char *
cygwin_pid_to_str (int pid)
{
static char buf[80];
if (pid == current_event.dwProcessId)
sprintf (buf, "process %d", pid);
else
sprintf (buf, "thread %d.0x%x", current_event.dwProcessId, pid);
return buf;
}