1 /* 2 * PROJECT: ReactOS win32 kernel mode subsystem 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: win32ss/gdi/ntgdi/gdidbg.c 5 * PURPOSE: Special debugging functions for GDI 6 * PROGRAMMERS: Timo Kreuzer 7 */ 8 9 /** INCLUDES ******************************************************************/ 10 11 #if DBG 12 #include <win32k.h> 13 #define NDEBUG 14 #include <debug.h> 15 16 extern ULONG gulFirstFree; 17 extern ULONG gulFirstUnused; 18 extern PENTRY gpentHmgr; 19 20 ULONG gulLogUnique = 0; 21 22 /* Note: the following values need to be sorted */ 23 DBG_CHANNEL DbgChannels[DbgChCount] = { 24 {L"EngBlt", DbgChEngBlt}, 25 {L"EngBrush", DbgChEngBrush}, 26 {L"EngClip", DbgChEngClip}, 27 {L"EngCursor", DbgChEngCursor}, 28 {L"EngDev", DbgChEngDev}, 29 {L"EngErr", DbgChEngErr}, 30 {L"EngEvent", DbgChEngEvent}, 31 {L"EngGrad", DbgChEngGrad}, 32 {L"EngLDev", DbgChEngLDev}, 33 {L"EngLine", DbgChEngLine}, 34 {L"EngMapping", DbgChEngMapping}, 35 {L"EngPDev", DbgChEngPDev}, 36 {L"EngSurface", DbgChEngSurface}, 37 {L"EngWnd", DbgChEngWnd}, 38 {L"EngXlate", DbgChEngXlate}, 39 {L"GdiBitmap", DbgChGdiBitmap}, 40 {L"GdiBlt", DbgChGdiBlt}, 41 {L"GdiBrush", DbgChGdiBrush}, 42 {L"GdiClipRgn", DbgChGdiClipRgn}, 43 {L"GdiCoord", DbgChGdiCoord}, 44 {L"GdiDC", DbgChGdiDC}, 45 {L"GdiDCAttr", DbgChGdiDCAttr}, 46 {L"GdiDCState", DbgChGdiDCState}, 47 {L"GdiDev", DbgChGdiDev}, 48 {L"GdiDib", DbgChGdiDib}, 49 {L"GdiFont", DbgChGdiFont}, 50 {L"GdiLine", DbgChGdiLine}, 51 {L"GdiObj", DbgChGdiObj}, 52 {L"GdiPalette", DbgChGdiPalette}, 53 {L"GdiPath", DbgChGdiPath}, 54 {L"GdiPen", DbgChGdiPen}, 55 {L"GdiPool", DbgChGdiPool}, 56 {L"GdiRgn", DbgChGdiRgn}, 57 {L"GdiText", DbgChGdiText}, 58 {L"GdiXFormObj", DbgChGdiXFormObj}, 59 {L"UserAccel", DbgChUserAccel}, 60 {L"UserCallback", DbgChUserCallback}, 61 {L"UserCallProc", DbgChUserCallProc}, 62 {L"UserCaret", DbgChUserCaret}, 63 {L"UserClass", DbgChUserClass}, 64 {L"UserClipbrd", DbgChUserClipbrd}, 65 {L"UserCsr", DbgChUserCsr}, 66 {L"UserDce", DbgChUserDce}, 67 {L"UserDefwnd", DbgChUserDefwnd}, 68 {L"UserDesktop", DbgChUserDesktop}, 69 {L"UserDisplay",DbgChUserDisplay}, 70 {L"UserEvent", DbgChUserEvent}, 71 {L"UserFocus", DbgChUserFocus}, 72 {L"UserHook", DbgChUserHook}, 73 {L"UserHotkey", DbgChUserHotkey}, 74 {L"UserIcon", DbgChUserIcon}, 75 {L"UserInput", DbgChUserInput}, 76 {L"UserKbd", DbgChUserKbd}, 77 {L"UserKbdLayout", DbgChUserKbdLayout}, 78 {L"UserMenu", DbgChUserMenu}, 79 {L"UserMetric", DbgChUserMetric}, 80 {L"UserMisc", DbgChUserMisc}, 81 {L"UserMonitor", DbgChUserMonitor}, 82 {L"UserMsg", DbgChUserMsg}, 83 {L"UserMsgQ", DbgChUserMsgQ}, 84 {L"UserObj", DbgChUserObj}, 85 {L"UserPainting", DbgChUserPainting}, 86 {L"UserProcess", DbgChUserProcess}, 87 {L"UserProp", DbgChUserProp}, 88 {L"UserScrollbar", DbgChUserScrollbar}, 89 {L"UserShutdown", DbgChUserShutdown}, 90 {L"UserSysparams", DbgChUserSysparams}, 91 {L"UserTimer", DbgChUserTimer}, 92 {L"UserThread", DbgChUserThread}, 93 {L"UserWinpos", DbgChUserWinpos}, 94 {L"UserWinsta", DbgChUserWinsta}, 95 {L"UserWnd", DbgChUserWnd} 96 }; 97 98 ULONG 99 NTAPI 100 DbgCaptureStackBackTace( 101 _Out_writes_(cFramesToCapture) PVOID* ppvFrames, 102 _In_ ULONG cFramesToSkip, 103 _In_ ULONG cFramesToCapture) 104 { 105 ULONG cFrameCount; 106 PVOID apvTemp[30]; 107 NT_ASSERT(cFramesToCapture <= _countof(apvTemp)); 108 109 /* Zero it out */ 110 RtlZeroMemory(ppvFrames, cFramesToCapture * sizeof(PVOID)); 111 112 /* Capture kernel stack */ 113 cFrameCount = RtlWalkFrameChain(apvTemp, _countof(apvTemp), 0); 114 115 /* If we should skip more than we have, we are done */ 116 if (cFramesToSkip > cFrameCount) 117 return 0; 118 119 /* Copy, but skip frames */ 120 cFrameCount -= cFramesToSkip; 121 cFrameCount = min(cFrameCount, cFramesToCapture); 122 RtlCopyMemory(ppvFrames, &apvTemp[cFramesToSkip], cFrameCount * sizeof(PVOID)); 123 124 /* Check if there is still space left */ 125 if (cFrameCount < cFramesToCapture) 126 { 127 /* Capture user stack */ 128 cFrameCount += RtlWalkFrameChain(&ppvFrames[cFrameCount], 129 cFramesToCapture - cFrameCount, 130 1); 131 } 132 133 return cFrameCount; 134 } 135 136 #if DBG_ENABLE_GDIOBJ_BACKTRACES 137 138 static 139 BOOL 140 CompareBacktraces( 141 USHORT idx1, 142 USHORT idx2) 143 { 144 POBJ pobj1, pobj2; 145 ULONG iLevel; 146 147 /* Get the objects */ 148 pobj1 = gpentHmgr[idx1].einfo.pobj; 149 pobj2 = gpentHmgr[idx2].einfo.pobj; 150 151 /* Loop all stack levels */ 152 for (iLevel = 0; iLevel < GDI_OBJECT_STACK_LEVELS; iLevel++) 153 { 154 /* If one level doesn't match we are done */ 155 if (pobj1->apvBackTrace[iLevel] != pobj2->apvBackTrace[iLevel]) 156 { 157 return FALSE; 158 } 159 } 160 161 return TRUE; 162 } 163 164 typedef struct 165 { 166 USHORT idx; 167 USHORT iCount; 168 } GDI_DBG_HANDLE_BT; 169 170 VOID 171 NTAPI 172 DbgDumpGdiHandleTableWithBT(void) 173 { 174 static BOOL bLeakReported = FALSE; 175 ULONG idx, j; 176 BOOL bAlreadyPresent; 177 GDI_DBG_HANDLE_BT aBacktraceTable[GDI_DBG_MAX_BTS]; 178 USHORT iCount; 179 KIRQL OldIrql; 180 POBJ pobj; 181 ULONG iLevel, ulObj; 182 183 /* Only report once */ 184 if (bLeakReported) 185 { 186 DPRINT1("GDI handle abusers already reported!\n"); 187 return; 188 } 189 190 bLeakReported = TRUE; 191 DPRINT1("Reporting GDI handle abusers:\n"); 192 193 /* Zero out the table */ 194 RtlZeroMemory(aBacktraceTable, sizeof(aBacktraceTable)); 195 196 /* We've got serious business to do */ 197 KeRaiseIrql(DISPATCH_LEVEL, &OldIrql); 198 199 /* Step through GDI handle table and find out who our culprit is... */ 200 for (idx = RESERVE_ENTRIES_COUNT; idx < GDI_HANDLE_COUNT; idx++) 201 { 202 /* If the handle is free, continue */ 203 if (gpentHmgr[idx].einfo.pobj == 0) continue; 204 205 /* Check if this backtrace is already covered */ 206 bAlreadyPresent = FALSE; 207 for (j = RESERVE_ENTRIES_COUNT; j < idx; j++) 208 { 209 if (CompareBacktraces(idx, j)) 210 { 211 bAlreadyPresent = TRUE; 212 break; 213 } 214 } 215 216 if (bAlreadyPresent) continue; 217 218 /* We don't have this BT yet, count how often it is present */ 219 iCount = 1; 220 for (j = idx + 1; j < GDI_HANDLE_COUNT; j++) 221 { 222 if (CompareBacktraces(idx, j)) 223 { 224 iCount++; 225 } 226 } 227 228 /* Now add this backtrace */ 229 for (j = 0; j < GDI_DBG_MAX_BTS; j++) 230 { 231 /* Insert it below the next smaller count */ 232 if (aBacktraceTable[j].iCount < iCount) 233 { 234 /* Check if there are entries above */ 235 if (j < GDI_DBG_MAX_BTS - 1) 236 { 237 /* Move the following entries up by 1 */ 238 RtlMoveMemory(&aBacktraceTable[j], 239 &aBacktraceTable[j + 1], 240 GDI_DBG_MAX_BTS - j - 1); 241 } 242 243 /* Set this entry */ 244 aBacktraceTable[j].idx = idx; 245 aBacktraceTable[j].iCount = iCount; 246 247 /* We are done here */ 248 break; 249 } 250 } 251 } 252 253 /* Print the worst offenders... */ 254 DbgPrint("Count Handle Backtrace\n"); 255 DbgPrint("------------------------------------------------\n"); 256 for (j = 0; j < GDI_DBG_MAX_BTS; j++) 257 { 258 idx = aBacktraceTable[j].idx; 259 if (idx == 0) 260 break; 261 262 ulObj = ((ULONG)gpentHmgr[idx].FullUnique << 16) | idx; 263 pobj = gpentHmgr[idx].einfo.pobj; 264 265 DbgPrint("%5d %08lx ", aBacktraceTable[j].iCount, ulObj); 266 for (iLevel = 0; iLevel < GDI_OBJECT_STACK_LEVELS; iLevel++) 267 { 268 DbgPrint("%p,", pobj->apvBackTrace[iLevel]); 269 } 270 DbgPrint("\n"); 271 } 272 273 __debugbreak(); 274 275 KeLowerIrql(OldIrql); 276 } 277 278 #endif /* DBG_ENABLE_GDIOBJ_BACKTRACES */ 279 280 #if DBG 281 282 BOOL 283 NTAPI 284 DbgGdiHTIntegrityCheck(VOID) 285 { 286 ULONG i, nDeleted = 0, nFree = 0, nUsed = 0; 287 PGDI_TABLE_ENTRY pEntry; 288 BOOL r = 1; 289 290 KeEnterCriticalRegion(); 291 292 /* FIXME: Check reserved entries */ 293 294 /* Now go through the deleted objects */ 295 i = gulFirstFree & 0xffff; 296 while (i) 297 { 298 pEntry = &GdiHandleTable->Entries[i]; 299 if (i >= GDI_HANDLE_COUNT) 300 { 301 DPRINT1("nDeleted=%lu\n", nDeleted); 302 ASSERT(FALSE); 303 } 304 305 nDeleted++; 306 307 /* Check the entry */ 308 if ((pEntry->Type & GDI_ENTRY_BASETYPE_MASK) != 0) 309 { 310 r = 0; 311 DPRINT1("Deleted Entry has a type != 0\n"); 312 } 313 if ((ULONG_PTR)pEntry->KernelData >= GDI_HANDLE_COUNT) 314 { 315 r = 0; 316 DPRINT1("Deleted entries KernelPointer too big\n"); 317 } 318 if (pEntry->UserData != NULL) 319 { 320 r = 0; 321 DPRINT1("Deleted entry has UserData != 0\n"); 322 } 323 if (pEntry->ProcessId != 0) 324 { 325 r = 0; 326 DPRINT1("Deleted entry has ProcessId != 0\n"); 327 } 328 329 i = (ULONG_PTR)pEntry->KernelData & 0xffff; 330 }; 331 332 for (i = gulFirstUnused; 333 i < GDI_HANDLE_COUNT; 334 i++) 335 { 336 pEntry = &GdiHandleTable->Entries[i]; 337 338 if ((pEntry->Type) != 0) 339 { 340 r = 0; 341 DPRINT1("Free Entry has a type != 0\n"); 342 } 343 if ((ULONG_PTR)pEntry->KernelData != 0) 344 { 345 r = 0; 346 DPRINT1("Free entries KernelPointer != 0\n"); 347 } 348 if (pEntry->UserData != NULL) 349 { 350 r = 0; 351 DPRINT1("Free entry has UserData != 0\n"); 352 } 353 if (pEntry->ProcessId != 0) 354 { 355 r = 0; 356 DPRINT1("Free entry has ProcessId != 0\n"); 357 } 358 nFree++; 359 } 360 361 for (i = RESERVE_ENTRIES_COUNT; i < GDI_HANDLE_COUNT; i++) 362 { 363 HGDIOBJ Handle; 364 ULONG Type; 365 366 pEntry = &GdiHandleTable->Entries[i]; 367 Type = pEntry->Type; 368 Handle = (HGDIOBJ)(((ULONG_PTR)Type << GDI_ENTRY_UPPER_SHIFT) + i); 369 370 if (Type & GDI_ENTRY_BASETYPE_MASK) 371 { 372 if (pEntry->KernelData == NULL) 373 { 374 r = 0; 375 DPRINT1("Used entry has KernelData == 0\n"); 376 } 377 else if (pEntry->KernelData <= MmHighestUserAddress) 378 { 379 r = 0; 380 DPRINT1("Used entry invalid KernelData\n"); 381 } 382 else if (((POBJ)(pEntry->KernelData))->hHmgr != Handle) 383 { 384 r = 0; 385 DPRINT1("Used entry %lu, has invalid hHmg %p (expected: %p)\n", 386 i, ((POBJ)(pEntry->KernelData))->hHmgr, Handle); 387 } 388 nUsed++; 389 } 390 } 391 392 if (RESERVE_ENTRIES_COUNT + nDeleted + nFree + nUsed != GDI_HANDLE_COUNT) 393 { 394 r = 0; 395 DPRINT1("Number of all entries incorrect: RESERVE_ENTRIES_COUNT = %lu, nDeleted = %lu, nFree = %lu, nUsed = %lu\n", 396 RESERVE_ENTRIES_COUNT, nDeleted, nFree, nUsed); 397 } 398 399 KeLeaveCriticalRegion(); 400 401 return r; 402 } 403 404 #endif /* DBG */ 405 406 407 #if DBG_ENABLE_EVENT_LOGGING 408 409 VOID 410 NTAPI 411 DbgLogEvent(PSLIST_HEADER pslh, LOG_EVENT_TYPE nEventType, LPARAM lParam) 412 { 413 PLOGENTRY pLogEntry; 414 415 /* Log a maximum of 100 events */ 416 if (QueryDepthSList(pslh) >= 1000) return; 417 418 /* Allocate a logentry */ 419 pLogEntry = EngAllocMem(0, sizeof(LOGENTRY), 'golG'); 420 if (!pLogEntry) return; 421 422 /* Set type */ 423 pLogEntry->nEventType = nEventType; 424 pLogEntry->ulUnique = InterlockedIncrement((LONG*)&gulLogUnique); 425 pLogEntry->dwProcessId = HandleToUlong(PsGetCurrentProcessId()); 426 pLogEntry->dwThreadId = HandleToUlong(PsGetCurrentThreadId()); 427 pLogEntry->lParam = lParam; 428 429 /* Capture a backtrace */ 430 DbgCaptureStackBackTace(pLogEntry->apvBackTrace, 1, 20); 431 432 switch (nEventType) 433 { 434 case EVENT_ALLOCATE: 435 case EVENT_CREATE_HANDLE: 436 case EVENT_REFERENCE: 437 case EVENT_DEREFERENCE: 438 case EVENT_LOCK: 439 case EVENT_UNLOCK: 440 case EVENT_DELETE: 441 case EVENT_FREE: 442 case EVENT_SET_OWNER: 443 default: 444 break; 445 } 446 447 /* Push it on the list */ 448 InterlockedPushEntrySList(pslh, &pLogEntry->sleLink); 449 } 450 451 #define REL_ADDR(va) ((ULONG_PTR)va - (ULONG_PTR)&__ImageBase) 452 453 VOID 454 NTAPI 455 DbgPrintEvent(PLOGENTRY pLogEntry) 456 { 457 PSTR pstr; 458 459 switch (pLogEntry->nEventType) 460 { 461 case EVENT_ALLOCATE: pstr = "Allocate"; break; 462 case EVENT_CREATE_HANDLE: pstr = "CreatHdl"; break; 463 case EVENT_REFERENCE: pstr = "Ref"; break; 464 case EVENT_DEREFERENCE: pstr = "Deref"; break; 465 case EVENT_LOCK: pstr = "Lock"; break; 466 case EVENT_UNLOCK: pstr = "Unlock"; break; 467 case EVENT_DELETE: pstr = "Delete"; break; 468 case EVENT_FREE: pstr = "Free"; break; 469 case EVENT_SET_OWNER: pstr = "SetOwner"; break; 470 default: pstr = "Unknown"; break; 471 } 472 473 DbgPrint("[%lu] %03x:%03x %.8s val=%p <%lx,%lx,%lx,%lx>\n", 474 pLogEntry->ulUnique, 475 pLogEntry->dwProcessId, 476 pLogEntry->dwThreadId, 477 pstr, 478 (PVOID)pLogEntry->lParam, 479 REL_ADDR(pLogEntry->apvBackTrace[2]), 480 REL_ADDR(pLogEntry->apvBackTrace[3]), 481 REL_ADDR(pLogEntry->apvBackTrace[4]), 482 REL_ADDR(pLogEntry->apvBackTrace[5])); 483 } 484 485 VOID 486 NTAPI 487 DbgDumpEventList(PSLIST_HEADER pslh) 488 { 489 PSLIST_ENTRY psle; 490 PLOGENTRY pLogEntry; 491 492 while ((psle = InterlockedPopEntrySList(pslh))) 493 { 494 pLogEntry = CONTAINING_RECORD(psle, LOGENTRY, sleLink); 495 DbgPrintEvent(pLogEntry); 496 } 497 } 498 499 VOID 500 NTAPI 501 DbgCleanupEventList(PSLIST_HEADER pslh) 502 { 503 PSLIST_ENTRY psle; 504 PLOGENTRY pLogEntry; 505 506 while ((psle = InterlockedPopEntrySList(pslh))) 507 { 508 pLogEntry = CONTAINING_RECORD(psle, LOGENTRY, sleLink); 509 EngFreeMem(pLogEntry); 510 } 511 } 512 513 #endif /* DBG_ENABLE_EVENT_LOGGING */ 514 515 #if 1 || DBG_ENABLE_SERVICE_HOOKS 516 517 VOID 518 NTAPI 519 DbgDumpLockedGdiHandles(VOID) 520 { 521 ULONG i; 522 523 for (i = RESERVE_ENTRIES_COUNT; i < GDI_HANDLE_COUNT; i++) 524 { 525 PENTRY pentry = &gpentHmgr[i]; 526 527 if (pentry->Objt) 528 { 529 POBJ pobj = pentry->einfo.pobj; 530 if (pobj->cExclusiveLock > 0) 531 { 532 DPRINT1("Locked object: %lx, type = %lx. allocated from:\n", 533 i, pentry->Objt); 534 DBG_DUMP_EVENT_LIST(&pobj->slhLog); 535 } 536 } 537 } 538 } 539 540 void 541 NTAPI 542 GdiDbgPreServiceHook(ULONG ulSyscallId, PULONG_PTR pulArguments) 543 { 544 PTHREADINFO pti = (PTHREADINFO)PsGetCurrentThreadWin32Thread(); 545 if (pti && pti->cExclusiveLocks != 0) 546 { 547 DbgPrint("FATAL: Win32DbgPreServiceHook(0x%lx): There are %lu exclusive locks!\n", 548 ulSyscallId, pti->cExclusiveLocks); 549 DbgDumpLockedGdiHandles(); 550 ASSERT(FALSE); 551 } 552 553 } 554 555 ULONG_PTR 556 NTAPI 557 GdiDbgPostServiceHook(ULONG ulSyscallId, ULONG_PTR ulResult) 558 { 559 PTHREADINFO pti = (PTHREADINFO)PsGetCurrentThreadWin32Thread(); 560 if (pti && pti->cExclusiveLocks != 0) 561 { 562 DbgPrint("FATAL: Win32DbgPostServiceHook(0x%lx): There are %lu exclusive locks!\n", 563 ulSyscallId, pti->cExclusiveLocks); 564 DbgDumpLockedGdiHandles(); 565 ASSERT(FALSE); 566 } 567 return ulResult; 568 } 569 570 #endif /* DBG_ENABLE_SERVICE_HOOKS */ 571 572 573 NTSTATUS NTAPI 574 QueryEnvironmentVariable(PUNICODE_STRING Name, 575 PUNICODE_STRING Value) 576 { 577 NTSTATUS Status; 578 PWSTR wcs; 579 UNICODE_STRING var; 580 PWSTR val; 581 PPEB Peb; 582 PWSTR Environment; 583 584 /* Ugly HACK for ReactOS system threads */ 585 if(!NtCurrentTeb()) 586 { 587 return(STATUS_VARIABLE_NOT_FOUND); 588 } 589 590 Peb = NtCurrentPeb(); 591 592 if (Peb == NULL) 593 { 594 return(STATUS_VARIABLE_NOT_FOUND); 595 } 596 597 Environment = Peb->ProcessParameters->Environment; 598 599 if (Environment == NULL) 600 { 601 return(STATUS_VARIABLE_NOT_FOUND); 602 } 603 604 Value->Length = 0; 605 606 wcs = Environment; 607 while (*wcs) 608 { 609 var.Buffer = wcs++; 610 wcs = wcschr(wcs, L'='); 611 if (wcs == NULL) 612 { 613 wcs = var.Buffer + wcslen(var.Buffer); 614 } 615 if (*wcs) 616 { 617 var.Length = var.MaximumLength = (wcs - var.Buffer) * sizeof(WCHAR); 618 val = ++wcs; 619 wcs += wcslen(wcs); 620 621 if (RtlEqualUnicodeString(&var, Name, TRUE)) 622 { 623 Value->Length = (wcs - val) * sizeof(WCHAR); 624 if (Value->Length <= Value->MaximumLength) 625 { 626 memcpy(Value->Buffer, val, 627 min(Value->Length + sizeof(WCHAR), Value->MaximumLength)); 628 Status = STATUS_SUCCESS; 629 } 630 else 631 { 632 Status = STATUS_BUFFER_TOO_SMALL; 633 } 634 635 return(Status); 636 } 637 } 638 wcs++; 639 } 640 641 return(STATUS_VARIABLE_NOT_FOUND); 642 } 643 644 static int __cdecl 645 DbgCompareChannels(const void * a, const void * b) 646 { 647 return wcscmp((WCHAR*)a, ((DBG_CHANNEL*)b)->Name); 648 } 649 650 static BOOL 651 DbgAddDebugChannel(PPROCESSINFO ppi, WCHAR* channel, WCHAR* level, WCHAR op) 652 { 653 DBG_CHANNEL *ChannelEntry; 654 UINT iLevel, iChannel; 655 656 /* Special treatment for the "all" channel */ 657 if (wcscmp(channel, L"all") == 0) 658 { 659 for (iChannel = 0; iChannel < DbgChCount; iChannel++) 660 { 661 DbgAddDebugChannel(ppi, DbgChannels[iChannel].Name, level, op); 662 } 663 return TRUE; 664 } 665 666 ChannelEntry = (DBG_CHANNEL*)bsearch(channel, 667 DbgChannels, 668 DbgChCount, 669 sizeof(DBG_CHANNEL), 670 DbgCompareChannels); 671 if(ChannelEntry == NULL) 672 { 673 return FALSE; 674 } 675 676 iChannel = ChannelEntry->Id; 677 ASSERT(iChannel < DbgChCount); 678 679 if(level == NULL || *level == L'\0' ||wcslen(level) == 0 ) 680 iLevel = MAX_LEVEL; 681 else if(wcsncmp(level, L"err", 3) == 0) 682 iLevel = ERR_LEVEL; 683 else if(wcsncmp(level, L"fixme", 5) == 0) 684 iLevel = FIXME_LEVEL; 685 else if(wcsncmp(level, L"warn", 4) == 0) 686 iLevel = WARN_LEVEL; 687 else if (wcsncmp(level, L"trace", 4) == 0) 688 iLevel = TRACE_LEVEL; 689 else 690 return FALSE; 691 692 if(op==L'+') 693 { 694 DBG_ENABLE_CHANNEL(ppi, iChannel, iLevel); 695 } 696 else 697 { 698 DBG_DISABLE_CHANNEL(ppi, iChannel, iLevel); 699 } 700 701 return TRUE; 702 } 703 704 static BOOL 705 DbgParseDebugChannels(PPROCESSINFO ppi, PUNICODE_STRING Value) 706 { 707 WCHAR *str, *separator, *c, op; 708 709 str = Value->Buffer; 710 711 do 712 { 713 separator = wcschr(str, L','); 714 if(separator != NULL) 715 *separator = L'\0'; 716 717 c = wcschr(str, L'+'); 718 if(c == NULL) 719 c = wcschr(str, L'-'); 720 721 if(c != NULL) 722 { 723 op = *c; 724 *c = L'\0'; 725 c++; 726 727 DbgAddDebugChannel(ppi, c, str, op); 728 } 729 730 str = separator + 1; 731 }while(separator != NULL); 732 733 return TRUE; 734 } 735 736 BOOL DbgInitDebugChannels(VOID) 737 { 738 WCHAR valBuffer[100]; 739 UNICODE_STRING Value; 740 UNICODE_STRING Name = RTL_CONSTANT_STRING(L"DEBUGCHANNEL"); 741 NTSTATUS Status; 742 PPROCESSINFO ppi; 743 BOOL ret; 744 745 /* Initialize all channels to ERROR */ 746 ppi = PsGetCurrentProcessWin32Process(); 747 RtlFillMemory( ppi->DbgChannelLevel, 748 sizeof(ppi->DbgChannelLevel), 749 ERR_LEVEL); 750 751 /* Find DEBUGCHANNEL env var */ 752 Value.Buffer = valBuffer; 753 Value.Length = 0; 754 Value.MaximumLength = sizeof(valBuffer); 755 Status = QueryEnvironmentVariable(&Name, &Value); 756 757 /* It does not exist */ 758 if(Status == STATUS_VARIABLE_NOT_FOUND) 759 { 760 /* There is nothing more to do */ 761 return TRUE; 762 } 763 764 /* If the buffer in the stack is not enough allocate it */ 765 if(Status == STATUS_BUFFER_TOO_SMALL) 766 { 767 Value.Buffer = ExAllocatePool(PagedPool, Value.MaximumLength); 768 if(Value.Buffer == NULL) 769 { 770 return FALSE; 771 } 772 773 /* Get the env var again */ 774 Status = QueryEnvironmentVariable(&Name, &Value); 775 } 776 777 /* Check for error */ 778 if(!NT_SUCCESS(Status)) 779 { 780 if(Value.Buffer != valBuffer) 781 { 782 ExFreePool(Value.Buffer); 783 } 784 785 return FALSE; 786 } 787 788 /* Parse the variable */ 789 ret = DbgParseDebugChannels(ppi, &Value); 790 791 /* Clean up */ 792 if(Value.Buffer != valBuffer) 793 { 794 ExFreePool(Value.Buffer); 795 } 796 797 return ret; 798 } 799 800 801 #endif // DBG 802 803 /* EOF */ 804