2  * Win32 debugger functions
 
   4  * Copyright (C) 1999 Alexandre Julliard
 
   6  * This library is free software; you can redistribute it and/or
 
   7  * modify it under the terms of the GNU Lesser General Public
 
   8  * License as published by the Free Software Foundation; either
 
   9  * version 2.1 of the License, or (at your option) any later version.
 
  11  * This library is distributed in the hope that it will be useful,
 
  12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 
  14  * Lesser General Public License for more details.
 
  16  * You should have received a copy of the GNU Lesser General Public
 
  17  * License along with this library; if not, write to the Free Software
 
  18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
 
  25 #include "wine/winbase16.h"
 
  26 #include "wine/server.h"
 
  27 #include "kernel_private.h"
 
  28 #include "kernel16_private.h"
 
  29 #include "wine/debug.h"
 
  31 WINE_DEFAULT_DEBUG_CHANNEL(debugstr);
 
  34 /******************************************************************************
 
  35  *           WaitForDebugEvent   (KERNEL32.@)
 
  37  *  Waits for a debugging event to occur in a process being debugged before
 
  38  *  filling out the debug event structure.
 
  41  *  event   [O] Address of structure for event information.
 
  42  *  timeout [I] Number of milliseconds to wait for event.
 
  46  *  Returns true if a debug event occurred and false if the call timed out.
 
  48 BOOL WINAPI WaitForDebugEvent(
 
  59         SERVER_START_REQ( wait_debug_event )
 
  61             req->get_handle = (timeout != 0);
 
  62             wine_server_set_reply( req, &data, sizeof(data) );
 
  63             if (!(ret = !wine_server_call_err( req ))) goto done;
 
  65             if (!wine_server_reply_size(reply))  /* timeout */
 
  71             event->dwDebugEventCode = data.code;
 
  72             event->dwProcessId      = (DWORD)reply->pid;
 
  73             event->dwThreadId       = (DWORD)reply->tid;
 
  76             case EXCEPTION_DEBUG_EVENT:
 
  77                 event->u.Exception.ExceptionRecord = data.info.exception.record;
 
  78                 event->u.Exception.dwFirstChance   = data.info.exception.first;
 
  80             case CREATE_THREAD_DEBUG_EVENT:
 
  81                 event->u.CreateThread.hThread           = data.info.create_thread.handle;
 
  82                 event->u.CreateThread.lpThreadLocalBase = data.info.create_thread.teb;
 
  83                 event->u.CreateThread.lpStartAddress    = data.info.create_thread.start;
 
  85             case CREATE_PROCESS_DEBUG_EVENT:
 
  86                 event->u.CreateProcessInfo.hFile                 = data.info.create_process.file;
 
  87                 event->u.CreateProcessInfo.hProcess              = data.info.create_process.process;
 
  88                 event->u.CreateProcessInfo.hThread               = data.info.create_process.thread;
 
  89                 event->u.CreateProcessInfo.lpBaseOfImage         = data.info.create_process.base;
 
  90                 event->u.CreateProcessInfo.dwDebugInfoFileOffset = data.info.create_process.dbg_offset;
 
  91                 event->u.CreateProcessInfo.nDebugInfoSize        = data.info.create_process.dbg_size;
 
  92                 event->u.CreateProcessInfo.lpThreadLocalBase     = data.info.create_process.teb;
 
  93                 event->u.CreateProcessInfo.lpStartAddress        = data.info.create_process.start;
 
  94                 event->u.CreateProcessInfo.lpImageName           = data.info.create_process.name;
 
  95                 event->u.CreateProcessInfo.fUnicode              = data.info.create_process.unicode;
 
  97             case EXIT_THREAD_DEBUG_EVENT:
 
  98                 event->u.ExitThread.dwExitCode = data.info.exit.exit_code;
 
 100             case EXIT_PROCESS_DEBUG_EVENT:
 
 101                 event->u.ExitProcess.dwExitCode = data.info.exit.exit_code;
 
 103             case LOAD_DLL_DEBUG_EVENT:
 
 104                 event->u.LoadDll.hFile                 = data.info.load_dll.handle;
 
 105                 event->u.LoadDll.lpBaseOfDll           = data.info.load_dll.base;
 
 106                 event->u.LoadDll.dwDebugInfoFileOffset = data.info.load_dll.dbg_offset;
 
 107                 event->u.LoadDll.nDebugInfoSize        = data.info.load_dll.dbg_size;
 
 108                 event->u.LoadDll.lpImageName           = data.info.load_dll.name;
 
 109                 event->u.LoadDll.fUnicode              = data.info.load_dll.unicode;
 
 111             case UNLOAD_DLL_DEBUG_EVENT:
 
 112                 event->u.UnloadDll.lpBaseOfDll = data.info.unload_dll.base;
 
 114             case OUTPUT_DEBUG_STRING_EVENT:
 
 115                 event->u.DebugString.lpDebugStringData  = data.info.output_string.string;
 
 116                 event->u.DebugString.fUnicode           = data.info.output_string.unicode;
 
 117                 event->u.DebugString.nDebugStringLength = data.info.output_string.length;
 
 120                 event->u.RipInfo.dwError = data.info.rip_info.error;
 
 121                 event->u.RipInfo.dwType  = data.info.rip_info.type;
 
 128         if (ret) return TRUE;
 
 130         res = WaitForSingleObject( wait, timeout );
 
 132         if (res != STATUS_WAIT_0) break;
 
 134     SetLastError( ERROR_SEM_TIMEOUT );
 
 139 /**********************************************************************
 
 140  *           ContinueDebugEvent   (KERNEL32.@)
 
 142  *  Enables a thread that previously produced a debug event to continue.
 
 145  *  pid    [I] The id of the process to continue.
 
 146  *  tid    [I] The id of the thread to continue.
 
 147  *  status [I] The rule to apply to unhandled exeptions.
 
 151  *  True if the debugger is listed as the processes owner and the process
 
 152  *  and thread are valid.
 
 154 BOOL WINAPI ContinueDebugEvent(
 
 160     SERVER_START_REQ( continue_debug_event )
 
 164         req->status = status;
 
 165         ret = !wine_server_call_err( req );
 
 172 /**********************************************************************
 
 173  *           DebugActiveProcess   (KERNEL32.@)
 
 175  *  Attempts to attach the debugger to a process.
 
 178  *  pid [I] The process to be debugged.
 
 182  *  True if the debugger was attached to process.
 
 184 BOOL WINAPI DebugActiveProcess( DWORD pid )
 
 187     SERVER_START_REQ( debug_process )
 
 191         ret = !wine_server_call_err( req );
 
 197 /**********************************************************************
 
 198  *           DebugActiveProcessStop   (KERNEL32.@)
 
 200  *  Attempts to detach the debugger from a process.
 
 203  *  pid [I] The process to be detached.
 
 207  *  True if the debugger was detached from the process.
 
 209 BOOL WINAPI DebugActiveProcessStop( DWORD pid )
 
 212     SERVER_START_REQ( debug_process )
 
 216         ret = !wine_server_call_err( req );
 
 223 /***********************************************************************
 
 224  *           OutputDebugStringA   (KERNEL32.@)
 
 226  *  Output by an application of an ascii string to a debugger (if attached)
 
 230  *  str [I] The message to be logged and given to the debugger.
 
 236 void WINAPI OutputDebugStringA( LPCSTR str )
 
 238     SERVER_START_REQ( output_debug_string )
 
 240         req->string  = (void *)str;
 
 242         req->length  = strlen(str) + 1;
 
 243         wine_server_call( req );
 
 250 /***********************************************************************
 
 251  *           OutputDebugStringW   (KERNEL32.@)
 
 253  *  Output by an application of a unicode string to a debugger (if attached)
 
 257  *  str [I] The message to be logged and given to the debugger.
 
 263 void WINAPI OutputDebugStringW( LPCWSTR str )
 
 265     SERVER_START_REQ( output_debug_string )
 
 267         req->string  = (void *)str;
 
 269         req->length  = (lstrlenW(str) + 1) * sizeof(WCHAR);
 
 270         wine_server_call( req );
 
 273     WARN("%s\n", debugstr_w(str));
 
 277 /***********************************************************************
 
 278  *           OutputDebugString   (KERNEL.115)
 
 280  *  Output by a 16 bit application of an ascii string to a debugger (if attached)
 
 284  *  str [I] The message to be logged and given to the debugger.
 
 288 void WINAPI OutputDebugString16( LPCSTR str )
 
 290     OutputDebugStringA( str );
 
 294 /***********************************************************************
 
 295  *           DebugBreak   (KERNEL32.@)
 
 297  *  Raises an exception so that a debugger (if attached)
 
 298  *  can take some action.
 
 304 void WINAPI DebugBreak(void)
 
 309 /***********************************************************************
 
 310  *           DebugBreakProcess   (KERNEL32.@)
 
 312  *  Raises an exception so that a debugger (if attached)
 
 313  *  can take some action. Same as DebugBreak, but applies to any process.
 
 316  *  hProc [I] Process to break into.
 
 320  *  True if successful.
 
 322 BOOL WINAPI DebugBreakProcess(HANDLE hProc)
 
 326     TRACE("(%p)\n", hProc);
 
 328     SERVER_START_REQ( debug_break )
 
 331         ret = !wine_server_call_err( req );
 
 332         self = ret && reply->self;
 
 335     if (self) DbgBreakPoint();
 
 340 /***********************************************************************
 
 341  *           DebugBreak   (KERNEL.203)
 
 343  *  Raises an expection in a 16 bit application so that a debugger (if attached)
 
 344  *  can take some action.
 
 352  *  Only 386 compatible processors implemented.
 
 354 void WINAPI DebugBreak16(
 
 355     CONTEXT86 *context) /* [in/out] A pointer to the 386 compatible processor state. */
 
 358     EXCEPTION_RECORD rec;
 
 360     rec.ExceptionCode    = EXCEPTION_BREAKPOINT;
 
 361     rec.ExceptionFlags   = 0;
 
 362     rec.ExceptionRecord  = NULL;
 
 363     rec.ExceptionAddress = (LPVOID)context->Eip;
 
 364     rec.NumberParameters = 0;
 
 365     NtRaiseException( &rec, context, TRUE );
 
 366 #endif  /* defined(__i386__) */
 
 370 /***********************************************************************
 
 371  *           IsDebuggerPresent   (KERNEL32.@)
 
 373  *  Allows a process to determine if there is a debugger attached.
 
 379  *  True if there is a debugger attached.
 
 381 BOOL WINAPI IsDebuggerPresent(void)
 
 383     return NtCurrentTeb()->Peb->BeingDebugged;
 
 386 /***********************************************************************
 
 387  *           CheckRemoteDebuggerPresent   (KERNEL32.@)
 
 389  *  Allows a process to determine if there is a remote debugger
 
 396  *  TRUE because it is a stub.
 
 398 BOOL WINAPI CheckRemoteDebuggerPresent(HANDLE process, PBOOL DebuggerPresent)
 
 400     FIXME("(%p)->(%p): Stub!\n", process, DebuggerPresent);
 
 401     *DebuggerPresent = FALSE;
 
 405 /***********************************************************************
 
 406  *           _DebugOutput                    (KERNEL.328)
 
 408 void WINAPIV _DebugOutput( WORD flags, LPCSTR spec, VA_LIST16 valist )
 
 412     /* Decode caller address */
 
 413     if (!GetModuleName16( GetExePtr(CURRENT_STACK16->cs), caller, sizeof(caller) ))
 
 414         sprintf( caller, "%04X:%04X", CURRENT_STACK16->cs, CURRENT_STACK16->ip );
 
 416     /* FIXME: cannot use wvsnprintf16 from kernel */
 
 417     /* wvsnprintf16( temp, sizeof(temp), spec, valist ); */
 
 420     FIXME("%s %04x %s\n", caller, flags, debugstr_a(spec) );
 
 423 /***********************************************************************
 
 424  *           DebugSetProcessKillOnExit                    (KERNEL32.@)
 
 426  * Let a debugger decide whether a debuggee will be killed upon debugger
 
 430  *  kill [I] If set to true then kill the process on exit.
 
 433  *  True if successful, false otherwise.
 
 435 BOOL WINAPI DebugSetProcessKillOnExit(BOOL kill)
 
 439     SERVER_START_REQ( set_debugger_kill_on_exit )
 
 441         req->kill_on_exit = kill;
 
 442         ret = !wine_server_call_err( req );