| 1 | /* $Id: oslibdebug.cpp,v 1.3 2000-03-13 13:10:10 sandervl Exp $ */ | 
|---|
| 2 |  | 
|---|
| 3 | /* | 
|---|
| 4 | * OS/2 debug apis | 
|---|
| 5 | * | 
|---|
| 6 | * Copyright 1999 Edgar Buerkle | 
|---|
| 7 | * | 
|---|
| 8 | * NOTE: Including os2.h, so not automatic protection against FS trashing! | 
|---|
| 9 | * | 
|---|
| 10 | * Project Odin Software License can be found in LICENSE.TXT | 
|---|
| 11 | * | 
|---|
| 12 | */ | 
|---|
| 13 | #define INCL_DOSPROCESS | 
|---|
| 14 | #define INCL_DOSSEMAPHORES | 
|---|
| 15 | #define INCL_DOSQUEUES | 
|---|
| 16 | #define INCL_DOSMODULEMGR | 
|---|
| 17 | #define INCL_DOSEXCEPTIONS | 
|---|
| 18 | #define INCL_DOSERRORS | 
|---|
| 19 | #include <os2wrap.h> | 
|---|
| 20 | #include <process.h> | 
|---|
| 21 | #include <os2sel.h> | 
|---|
| 22 | #include <stdlib.h> | 
|---|
| 23 | #include <string.h> | 
|---|
| 24 | #include <misc.h> | 
|---|
| 25 | #include <windllbase.h> | 
|---|
| 26 | #include <winconst.h> | 
|---|
| 27 | #include "oslibdebug.h" | 
|---|
| 28 |  | 
|---|
| 29 | #define DBG_LOCALLOG    DBG_oslibdebug | 
|---|
| 30 | #include "dbglocal.h" | 
|---|
| 31 |  | 
|---|
| 32 | #define DEBUG_QUEUENAME "\\QUEUES\\ODINTRACE\\" | 
|---|
| 33 | #define DEBUG_QSEMNAME  "\\SEM32\\ODINTRACEQ\\" | 
|---|
| 34 | #define DEBUG_SEMNAME   "\\SEM32\\ODINTRACE\\" | 
|---|
| 35 |  | 
|---|
| 36 | //****************************************************************************** | 
|---|
| 37 | //****************************************************************************** | 
|---|
| 38 | VOID _Optlink DebugThread(VOID *argpid) | 
|---|
| 39 | { | 
|---|
| 40 | CHAR   QueueName[30]=DEBUG_QUEUENAME; | 
|---|
| 41 | CHAR   SemName[30]=DEBUG_SEMNAME; | 
|---|
| 42 | CHAR   QSemName[30]=DEBUG_QSEMNAME; | 
|---|
| 43 | HQUEUE QueueHandle=0; | 
|---|
| 44 | HEV    hevSem=0, hevQSem=0; | 
|---|
| 45 | int    rc; | 
|---|
| 46 | uDB_t  DbgBuf={0}; | 
|---|
| 47 | char   path[CCHMAXPATH]; | 
|---|
| 48 | Win32DllBase *winmod; | 
|---|
| 49 | LPDEBUG_EVENT lpde; | 
|---|
| 50 | ULONG  *pid = (ULONG*)argpid; | 
|---|
| 51 | char   tmp[12]; | 
|---|
| 52 |  | 
|---|
| 53 | dprintf(("KERNEL32: DebugThread pid:%d", *pid)); | 
|---|
| 54 |  | 
|---|
| 55 | strcat(QueueName, itoa(getpid(), tmp, 10)); | 
|---|
| 56 | rc = DosCreateQueue( &QueueHandle , QUE_FIFO, QueueName); | 
|---|
| 57 | if(rc != 0) | 
|---|
| 58 | { | 
|---|
| 59 | dprintf(("DebugThread: Could not create queue:%s rc:%d", QueueName, rc)); | 
|---|
| 60 | return; | 
|---|
| 61 | } | 
|---|
| 62 | strcat(SemName, itoa(getpid(), tmp, 10)); | 
|---|
| 63 | rc = DosCreateEventSem(SemName, &hevSem, 0, TRUE); | 
|---|
| 64 | if(rc != 0) | 
|---|
| 65 | { | 
|---|
| 66 | dprintf(("DebugThread: Could not create event sem:%s rc:%d", SemName, rc)); | 
|---|
| 67 | DosCloseQueue(QueueHandle); | 
|---|
| 68 | return; | 
|---|
| 69 | } | 
|---|
| 70 | strcat(QSemName, itoa(getpid(), tmp, 10)); | 
|---|
| 71 | rc = DosCreateEventSem(QSemName, &hevQSem, 0, FALSE); | 
|---|
| 72 | if(rc != 0) | 
|---|
| 73 | { | 
|---|
| 74 | dprintf(("DebugThread: Could not create event sem:%s rc:%d", QSemName, rc)); | 
|---|
| 75 | DosCloseEventSem(hevSem); | 
|---|
| 76 | DosCloseQueue(QueueHandle); | 
|---|
| 77 | return; | 
|---|
| 78 | } | 
|---|
| 79 |  | 
|---|
| 80 | // connect to debuggee | 
|---|
| 81 | DbgBuf.Cmd = DBG_C_Connect; | 
|---|
| 82 | DbgBuf.Pid = *pid; | 
|---|
| 83 | DbgBuf.Tid = 0; | 
|---|
| 84 | DbgBuf.Value = DBG_L_386; | 
|---|
| 85 | DbgBuf.Addr  = 1; | 
|---|
| 86 | rc = DosDebug(&DbgBuf); | 
|---|
| 87 | if (rc != 0) | 
|---|
| 88 | { | 
|---|
| 89 | dprintf(("DosDebug error: rc = %d error:%d", rc, DbgBuf.Value)); | 
|---|
| 90 | DosCloseQueue(QueueHandle); | 
|---|
| 91 | DosCloseEventSem(hevSem); | 
|---|
| 92 | return; | 
|---|
| 93 | } | 
|---|
| 94 |  | 
|---|
| 95 | while(rc == 0) | 
|---|
| 96 | { | 
|---|
| 97 | DosWaitEventSem(hevSem, SEM_INDEFINITE_WAIT); | 
|---|
| 98 |  | 
|---|
| 99 | DosDebug_GO: | 
|---|
| 100 | DbgBuf.Cmd = DBG_C_Go; | 
|---|
| 101 | DbgBuf.Pid = *pid; | 
|---|
| 102 |  | 
|---|
| 103 | DebugApi: | 
|---|
| 104 | rc = DosDebug(&DbgBuf); | 
|---|
| 105 | if (rc != 0) | 
|---|
| 106 | dprintf(("DosDebug error: rc = %d", rc)); | 
|---|
| 107 |  | 
|---|
| 108 | switch (DbgBuf.Cmd) | 
|---|
| 109 | { | 
|---|
| 110 | case DBG_N_Success: | 
|---|
| 111 | dprintf(("DosDebug: GO ok")); | 
|---|
| 112 | goto DosDebug_GO; | 
|---|
| 113 | case DBG_N_Error: | 
|---|
| 114 | dprintf(("DosDebug: Error %d", DbgBuf.Value)); | 
|---|
| 115 | // if(DbgBuf.Value == ERROR_INVALID_PROCID) connect ? | 
|---|
| 116 | if(DbgBuf.Value == ERROR_EXCL_SEM_ALREADY_OWNED) | 
|---|
| 117 | { | 
|---|
| 118 | rc = 0; // continue | 
|---|
| 119 | goto DosDebug_GO; | 
|---|
| 120 | } | 
|---|
| 121 | break;    // end thread !!! | 
|---|
| 122 | case DBG_N_ProcTerm: | 
|---|
| 123 | dprintf(("DosDebug: Process terminated with rc %d\n",DbgBuf.Value)); | 
|---|
| 124 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 125 | lpde->dwDebugEventCode = EXIT_PROCESS_DEBUG_EVENT_W; | 
|---|
| 126 | lpde->dwProcessId = *pid; | 
|---|
| 127 | lpde->dwThreadId = 0; | 
|---|
| 128 | lpde->u.ExitThread.dwExitCode = DbgBuf.Value; | 
|---|
| 129 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 130 | break; | 
|---|
| 131 | case DBG_N_Exception: | 
|---|
| 132 | dprintf(("DosDebug: Exception")); | 
|---|
| 133 | // lpde = malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 134 | // TODO: fill union | 
|---|
| 135 | // DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 136 | // break; | 
|---|
| 137 | DbgBuf.Cmd = DBG_C_Continue; | 
|---|
| 138 | DbgBuf.Value = XCPT_CONTINUE_SEARCH; | 
|---|
| 139 | goto DebugApi; | 
|---|
| 140 | case DBG_N_ModuleLoad: | 
|---|
| 141 | DosQueryModuleName(DbgBuf.Value, CCHMAXPATH, path); | 
|---|
| 142 | dprintf(("DosDebug: module loaded [%s]", path)); | 
|---|
| 143 |  | 
|---|
| 144 | winmod = Win32DllBase::findModule(path); | 
|---|
| 145 | // only odin32(win32) modules, hide OS/2 ones | 
|---|
| 146 | if(!winmod) | 
|---|
| 147 | { | 
|---|
| 148 | dprintf(("DosDebug: os/2 module [%s], suppress", path)); | 
|---|
| 149 | goto DosDebug_GO; | 
|---|
| 150 | } | 
|---|
| 151 | dprintf(("DosDebug: win32 module [%s], inform", path)); | 
|---|
| 152 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 153 | lpde->dwDebugEventCode = LOAD_DLL_DEBUG_EVENT_W; | 
|---|
| 154 | lpde->dwProcessId = *pid; | 
|---|
| 155 | lpde->dwThreadId = 0; | 
|---|
| 156 | lpde->u.LoadDll.hFile = 0; | 
|---|
| 157 | // TODO: make a pe fakeheader in our DLL's (kernel32,...) | 
|---|
| 158 | lpde->u.LoadDll.lpBaseOfDll = WINIMAGE_LOOKUPADDR(winmod); | 
|---|
| 159 | lpde->u.LoadDll.dwDebugInfoFileOffset = 0; | 
|---|
| 160 | lpde->u.LoadDll.nDebugInfoSize = 0; | 
|---|
| 161 | lpde->u.LoadDll.lpImageName = path; | 
|---|
| 162 | lpde->u.LoadDll.fUnicode = FALSE; | 
|---|
| 163 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 164 | break; | 
|---|
| 165 | case DBG_N_CoError: | 
|---|
| 166 | dprintf(("DosDebug: Coprocessor Error")); | 
|---|
| 167 | // TODO: create an exception ? | 
|---|
| 168 | goto DosDebug_GO; | 
|---|
| 169 | break; | 
|---|
| 170 | case DBG_N_ThreadTerm: | 
|---|
| 171 | dprintf(("DosDebug: Thread %d terminated with rc %d", DbgBuf.Tid,DbgBuf.Value)); | 
|---|
| 172 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 173 | lpde->dwDebugEventCode = EXIT_THREAD_DEBUG_EVENT_W; | 
|---|
| 174 | lpde->dwProcessId = *pid; | 
|---|
| 175 | lpde->dwThreadId = DbgBuf.Tid; | 
|---|
| 176 | lpde->u.ExitThread.dwExitCode = DbgBuf.Value; | 
|---|
| 177 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 178 | break; | 
|---|
| 179 | case DBG_N_AsyncStop: | 
|---|
| 180 | dprintf(("DosDebug: Async stop")); | 
|---|
| 181 | goto DosDebug_GO; | 
|---|
| 182 | break; | 
|---|
| 183 | case DBG_N_NewProc: | 
|---|
| 184 | dprintf(("DosDebug: Debuggee started new Pid %d",DbgBuf.Value)); | 
|---|
| 185 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 186 | lpde->dwDebugEventCode = CREATE_PROCESS_DEBUG_EVENT_W; | 
|---|
| 187 | lpde->dwProcessId = *pid; | 
|---|
| 188 | lpde->dwThreadId = 0; | 
|---|
| 189 | //TODO: fill union | 
|---|
| 190 | lpde->u.CreateProcessInfo.hFile = 0; | 
|---|
| 191 | lpde->u.CreateProcessInfo.hProcess = 0; | 
|---|
| 192 | lpde->u.CreateProcessInfo.hThread = 0; | 
|---|
| 193 | lpde->u.CreateProcessInfo.lpBaseOfImage = NULL; | 
|---|
| 194 | lpde->u.CreateProcessInfo.dwDebugInfoFileOffset = 0; | 
|---|
| 195 | lpde->u.CreateProcessInfo.nDebugInfoSize = 0; | 
|---|
| 196 | lpde->u.CreateProcessInfo.lpThreadLocalBase = NULL; | 
|---|
| 197 | lpde->u.CreateProcessInfo.lpStartAddress = NULL; | 
|---|
| 198 | lpde->u.CreateProcessInfo.lpImageName = NULL; | 
|---|
| 199 | lpde->u.CreateProcessInfo.fUnicode = FALSE; | 
|---|
| 200 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 201 | break; | 
|---|
| 202 | case DBG_N_AliasFree: | 
|---|
| 203 | dprintf(("DosDebug: AliasFree")); | 
|---|
| 204 | goto DosDebug_GO; | 
|---|
| 205 | break; | 
|---|
| 206 | case DBG_N_Watchpoint: | 
|---|
| 207 | dprintf(("DosDebug: WatchPoint")); | 
|---|
| 208 | goto DosDebug_GO; | 
|---|
| 209 | break; | 
|---|
| 210 | case DBG_N_ThreadCreate: | 
|---|
| 211 | dprintf(("DosDebug: Thread %d created",DbgBuf.Tid)); | 
|---|
| 212 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 213 | lpde->dwDebugEventCode = CREATE_THREAD_DEBUG_EVENT_W; | 
|---|
| 214 | lpde->dwProcessId = *pid; | 
|---|
| 215 | lpde->dwThreadId = DbgBuf.Tid; | 
|---|
| 216 | //TODO: fill union | 
|---|
| 217 | lpde->u.CreateThread.hThread = 0; | 
|---|
| 218 | lpde->u.CreateThread.lpThreadLocalBase = NULL; | 
|---|
| 219 | lpde->u.CreateThread.lpStartAddress = NULL; | 
|---|
| 220 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 221 | break; | 
|---|
| 222 | case DBG_N_ModuleFree: | 
|---|
| 223 | DosQueryModuleName(DbgBuf.Value, CCHMAXPATH, path); | 
|---|
| 224 | dprintf(("DosDebug: ModuleFree [%s]", path)); | 
|---|
| 225 | winmod = Win32DllBase::findModule(path); | 
|---|
| 226 | // only odin32(win32) modules, hide OS/2 ones | 
|---|
| 227 | if(!winmod) | 
|---|
| 228 | { | 
|---|
| 229 | dprintf(("DosDebug: os/2 module [%s], suppress", path)); | 
|---|
| 230 | break; | 
|---|
| 231 | } | 
|---|
| 232 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 233 | lpde->dwDebugEventCode = UNLOAD_DLL_DEBUG_EVENT_W; | 
|---|
| 234 | lpde->dwProcessId = *pid; | 
|---|
| 235 | lpde->dwThreadId = 0; | 
|---|
| 236 | lpde->u.UnloadDll.lpBaseOfDll = WINIMAGE_LOOKUPADDR(winmod); | 
|---|
| 237 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 238 | break; | 
|---|
| 239 | case DBG_N_RangeStep: | 
|---|
| 240 | dprintf(("DosDebug: RangeStep")); | 
|---|
| 241 | goto DosDebug_GO; | 
|---|
| 242 | break; | 
|---|
| 243 | default: | 
|---|
| 244 | dprintf(("DosDebug: Unkown Notify %d", DbgBuf.Cmd)); | 
|---|
| 245 | goto DosDebug_GO; | 
|---|
| 246 | break; | 
|---|
| 247 | } | 
|---|
| 248 | } | 
|---|
| 249 |  | 
|---|
| 250 | DosCloseQueue(QueueHandle); | 
|---|
| 251 | DosCloseEventSem(hevSem); | 
|---|
| 252 | DosCloseEventSem(hevQSem); | 
|---|
| 253 | *pid = 0; | 
|---|
| 254 |  | 
|---|
| 255 | } | 
|---|
| 256 | //****************************************************************************** | 
|---|
| 257 | //****************************************************************************** | 
|---|
| 258 | BOOL OSLibWaitForDebugEvent(LPDEBUG_EVENT lpde, DWORD dwTimeout) | 
|---|
| 259 | { | 
|---|
| 260 | CHAR QueueName[30]=DEBUG_QUEUENAME; | 
|---|
| 261 | CHAR   SemName[30]=DEBUG_QSEMNAME; | 
|---|
| 262 | LPDEBUG_EVENT lpde_queue; | 
|---|
| 263 | int    rc, req; | 
|---|
| 264 | PID    pidOwner; | 
|---|
| 265 | HQUEUE QueueHandle=0; | 
|---|
| 266 | REQUESTDATA Request = {0}; | 
|---|
| 267 | ULONG  len; | 
|---|
| 268 | BYTE   prio; | 
|---|
| 269 | HEV    hevQSem=0; | 
|---|
| 270 | char   tmp[12]; | 
|---|
| 271 | USHORT sel = RestoreOS2FS(); | 
|---|
| 272 |  | 
|---|
| 273 | strcat(SemName, itoa(getpid(),tmp, 10)); | 
|---|
| 274 | rc = DosOpenEventSem(SemName, &hevQSem); | 
|---|
| 275 | if(rc != 0) | 
|---|
| 276 | goto fail; | 
|---|
| 277 |  | 
|---|
| 278 | // get a DebugEvent from our DebugThread | 
|---|
| 279 | strcat(QueueName, itoa(getpid(), tmp, 10)); | 
|---|
| 280 | rc = DosOpenQueue(&pidOwner, &QueueHandle, QueueName); | 
|---|
| 281 | Request.pid = pidOwner; | 
|---|
| 282 | rc = DosReadQueue(QueueHandle, &Request, &len, (PPVOID) &lpde_queue, 0, DCWW_NOWAIT, | 
|---|
| 283 | &prio, hevQSem); | 
|---|
| 284 | if(rc == ERROR_QUE_EMPTY) | 
|---|
| 285 | { | 
|---|
| 286 | if(DosWaitEventSem(hevQSem, dwTimeout) == 0) | 
|---|
| 287 | rc = DosReadQueue(QueueHandle, &Request, &len, (PPVOID) &lpde_queue, 0, DCWW_NOWAIT, | 
|---|
| 288 | &prio, hevQSem); | 
|---|
| 289 | } | 
|---|
| 290 | if(rc != 0) | 
|---|
| 291 | goto fail; | 
|---|
| 292 |  | 
|---|
| 293 | // copy DebugEvent to user space and free queue pointer | 
|---|
| 294 | memcpy(lpde, lpde_queue, len); | 
|---|
| 295 | free(lpde_queue); | 
|---|
| 296 | // DosCloseEventSem(hevSem); | 
|---|
| 297 | SetFS(sel); | 
|---|
| 298 | return TRUE; | 
|---|
| 299 |  | 
|---|
| 300 | fail: | 
|---|
| 301 | // DosCloseEventSem(hevSem); | 
|---|
| 302 | SetFS(sel); | 
|---|
| 303 | return FALSE; | 
|---|
| 304 | } | 
|---|
| 305 | //****************************************************************************** | 
|---|
| 306 | //****************************************************************************** | 
|---|
| 307 | BOOL OSLibContinueDebugEvent(DWORD dwProcessId, DWORD dwThreadId, DWORD dwContinueStatus) | 
|---|
| 308 | { | 
|---|
| 309 | CHAR   SemName[30]=DEBUG_SEMNAME; | 
|---|
| 310 | CHAR   QueueName[30]=DEBUG_QUEUENAME; | 
|---|
| 311 | PID    pidOwner; | 
|---|
| 312 | HQUEUE QueueHandle=0; | 
|---|
| 313 | HEV    hev=0; | 
|---|
| 314 | int    rc; | 
|---|
| 315 | char   tmp[12]; | 
|---|
| 316 | ULONG  QEntries=0; | 
|---|
| 317 | USHORT sel = RestoreOS2FS(); | 
|---|
| 318 |  | 
|---|
| 319 | // only continue DebugThread, if queue is empty | 
|---|
| 320 | strcat(QueueName, itoa(getpid(), tmp, 10)); | 
|---|
| 321 | rc = DosOpenQueue(&pidOwner, &QueueHandle, QueueName); | 
|---|
| 322 | rc = DosQueryQueue(QueueHandle, &QEntries); | 
|---|
| 323 | if(QEntries > 0) { | 
|---|
| 324 | SetFS(sel); | 
|---|
| 325 | return TRUE; | 
|---|
| 326 | } | 
|---|
| 327 | // continue DebugThread | 
|---|
| 328 | strcat(SemName, itoa(getpid(), tmp, 10)); | 
|---|
| 329 | rc = DosOpenEventSem(SemName, &hev); | 
|---|
| 330 | rc = DosPostEventSem(hev); | 
|---|
| 331 | // DosCloseEventSem(hev); | 
|---|
| 332 | SetFS(sel); | 
|---|
| 333 | return (rc == 0) ? TRUE : FALSE; | 
|---|
| 334 | } | 
|---|
| 335 | //****************************************************************************** | 
|---|
| 336 | //****************************************************************************** | 
|---|
| 337 | BOOL OSLibAddModuleDebugEvent(char *name, BOOL fLoad) | 
|---|
| 338 | { | 
|---|
| 339 | Win32DllBase  *winmod; | 
|---|
| 340 | LPDEBUG_EVENT lpde; | 
|---|
| 341 | int           rc; | 
|---|
| 342 | CHAR          QueueName[30]=DEBUG_QUEUENAME; | 
|---|
| 343 | PID           pidOwner; | 
|---|
| 344 | HQUEUE        QueueHandle=0; | 
|---|
| 345 | char          tmp[12]; | 
|---|
| 346 | USHORT        sel = RestoreOS2FS(); | 
|---|
| 347 |  | 
|---|
| 348 | winmod = Win32DllBase::findModule(name); | 
|---|
| 349 | if(!winmod) | 
|---|
| 350 | { | 
|---|
| 351 | dprintf(("OSLibAddModuleDebugEvent: ERROR could not find module [%s]", name)); | 
|---|
| 352 | SetFS(sel); | 
|---|
| 353 | return FALSE; | 
|---|
| 354 | } | 
|---|
| 355 |  | 
|---|
| 356 | //  strcat(QueueName, itoa(getDebuggerPID(), tmp, 10)); | 
|---|
| 357 | rc = DosOpenQueue(&pidOwner, &QueueHandle, QueueName); | 
|---|
| 358 |  | 
|---|
| 359 | if(fLoad == TRUE) | 
|---|
| 360 | { | 
|---|
| 361 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 362 | lpde->dwDebugEventCode = LOAD_DLL_DEBUG_EVENT_W; | 
|---|
| 363 | lpde->dwProcessId = getpid(); // debuggee pid | 
|---|
| 364 | lpde->dwThreadId = 0; | 
|---|
| 365 | lpde->u.LoadDll.hFile = 0; | 
|---|
| 366 | lpde->u.LoadDll.lpBaseOfDll = WINIMAGE_LOOKUPADDR(winmod); | 
|---|
| 367 | lpde->u.LoadDll.dwDebugInfoFileOffset = 0; | 
|---|
| 368 | lpde->u.LoadDll.nDebugInfoSize = 0; | 
|---|
| 369 | lpde->u.LoadDll.lpImageName = name; | 
|---|
| 370 | lpde->u.LoadDll.fUnicode = FALSE; | 
|---|
| 371 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 372 | } | 
|---|
| 373 | else | 
|---|
| 374 | { | 
|---|
| 375 | lpde = (LPDEBUG_EVENT) malloc(sizeof(DEBUG_EVENT)); | 
|---|
| 376 | lpde->dwDebugEventCode = UNLOAD_DLL_DEBUG_EVENT_W; | 
|---|
| 377 | lpde->dwProcessId = getpid(); // debuggee pid | 
|---|
| 378 | lpde->dwThreadId = 0; | 
|---|
| 379 | lpde->u.UnloadDll.lpBaseOfDll = WINIMAGE_LOOKUPADDR(winmod); | 
|---|
| 380 | DosWriteQueue(QueueHandle, 0, sizeof(DEBUG_EVENT), lpde, 0); | 
|---|
| 381 | } | 
|---|
| 382 | SetFS(sel); | 
|---|
| 383 | return TRUE; | 
|---|
| 384 | } | 
|---|
| 385 | //****************************************************************************** | 
|---|
| 386 | //****************************************************************************** | 
|---|
| 387 | VOID OSLibStartDebugger(ULONG *pid) | 
|---|
| 388 | { | 
|---|
| 389 | USHORT sel = RestoreOS2FS(); | 
|---|
| 390 | TID tid; | 
|---|
| 391 |  | 
|---|
| 392 | tid = _beginthread(DebugThread, NULL, 1024, (PVOID) pid); | 
|---|
| 393 | if(tid == 0) | 
|---|
| 394 | { | 
|---|
| 395 | dprintf(("OSLibStartDebugger: Could create debug thread")); | 
|---|
| 396 | SetFS(sel); | 
|---|
| 397 | return; | 
|---|
| 398 | } | 
|---|
| 399 | DosSleep(128); | 
|---|
| 400 | SetFS(sel); | 
|---|
| 401 | } | 
|---|
| 402 | //****************************************************************************** | 
|---|
| 403 | //****************************************************************************** | 
|---|