1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS kernel 4 * FILE: ntoskrnl/rtl/libsupp.c 5 * PURPOSE: RTL Support Routines 6 * PROGRAMMERS: Alex Ionescu (alex@relsoft.net) 7 * Gunnar Dalsnes 8 */ 9 10 /* INCLUDES ******************************************************************/ 11 12 #include <ntoskrnl.h> 13 #define NDEBUG 14 #include <debug.h> 15 16 #define TAG_ATMT 'TotA' /* Atom table */ 17 #define TAG_RTHL 'LHtR' /* Heap Lock */ 18 19 extern ULONG NtGlobalFlag; 20 21 typedef struct _RTL_RANGE_ENTRY 22 { 23 LIST_ENTRY Entry; 24 RTL_RANGE Range; 25 } RTL_RANGE_ENTRY, *PRTL_RANGE_ENTRY; 26 27 PAGED_LOOKASIDE_LIST RtlpRangeListEntryLookasideList; 28 SIZE_T RtlpAllocDeallocQueryBufferSize = 128; 29 30 /* FUNCTIONS *****************************************************************/ 31 32 PVOID 33 NTAPI 34 RtlPcToFileHeader( 35 IN PVOID PcValue, 36 OUT PVOID *BaseOfImage) 37 { 38 PLDR_DATA_TABLE_ENTRY LdrEntry; 39 BOOLEAN InSystem; 40 41 /* Get the base for this file */ 42 if ((ULONG_PTR)PcValue > (ULONG_PTR)MmHighestUserAddress) 43 { 44 /* We are in kernel */ 45 *BaseOfImage = KiPcToFileHeader(PcValue, &LdrEntry, FALSE, &InSystem); 46 } 47 else 48 { 49 /* We are in user land */ 50 *BaseOfImage = KiRosPcToUserFileHeader(PcValue, &LdrEntry); 51 } 52 53 return *BaseOfImage; 54 } 55 56 VOID 57 NTAPI 58 RtlInitializeRangeListPackage(VOID) 59 { 60 /* Setup the lookaside list for allocations (not used yet) */ 61 ExInitializePagedLookasideList(&RtlpRangeListEntryLookasideList, 62 NULL, 63 NULL, 64 POOL_COLD_ALLOCATION, 65 sizeof(RTL_RANGE_ENTRY), 66 'elRR', 67 16); 68 } 69 70 BOOLEAN 71 NTAPI 72 RtlpCheckForActiveDebugger(VOID) 73 { 74 /* This check is meaningless in kernel-mode */ 75 return FALSE; 76 } 77 78 BOOLEAN 79 NTAPI 80 RtlpSetInDbgPrint(VOID) 81 { 82 /* Nothing to set in kernel mode */ 83 return FALSE; 84 } 85 86 VOID 87 NTAPI 88 RtlpClearInDbgPrint(VOID) 89 { 90 /* Nothing to clear in kernel mode */ 91 } 92 93 KPROCESSOR_MODE 94 NTAPI 95 RtlpGetMode(VOID) 96 { 97 return KernelMode; 98 } 99 100 PVOID 101 NTAPI 102 RtlpAllocateMemory(ULONG Bytes, 103 ULONG Tag) 104 { 105 return ExAllocatePoolWithTag(PagedPool, 106 (SIZE_T)Bytes, 107 Tag); 108 } 109 110 111 #define TAG_USTR 'RTSU' 112 #define TAG_ASTR 'RTSA' 113 #define TAG_OSTR 'RTSO' 114 VOID 115 NTAPI 116 RtlpFreeMemory(PVOID Mem, 117 ULONG Tag) 118 { 119 if (Tag == TAG_ASTR || Tag == TAG_OSTR || Tag == TAG_USTR) 120 ExFreePool(Mem); 121 else 122 ExFreePoolWithTag(Mem, Tag); 123 } 124 125 /* 126 * @implemented 127 */ 128 VOID NTAPI 129 RtlAcquirePebLock(VOID) 130 { 131 132 } 133 134 /* 135 * @implemented 136 */ 137 VOID NTAPI 138 RtlReleasePebLock(VOID) 139 { 140 141 } 142 143 NTSTATUS 144 NTAPI 145 LdrShutdownThread(VOID) 146 { 147 return STATUS_SUCCESS; 148 } 149 150 151 PPEB 152 NTAPI 153 RtlGetCurrentPeb(VOID) 154 { 155 return ((PEPROCESS)(KeGetCurrentThread()->ApcState.Process))->Peb; 156 } 157 158 NTSTATUS 159 NTAPI 160 RtlDeleteHeapLock(IN OUT PHEAP_LOCK Lock) 161 { 162 ExDeleteResourceLite(&Lock->Resource); 163 ExFreePoolWithTag(Lock, TAG_RTHL); 164 165 return STATUS_SUCCESS; 166 } 167 168 NTSTATUS 169 NTAPI 170 RtlEnterHeapLock(IN OUT PHEAP_LOCK Lock, IN BOOLEAN Exclusive) 171 { 172 KeEnterCriticalRegion(); 173 174 if (Exclusive) 175 ExAcquireResourceExclusiveLite(&Lock->Resource, TRUE); 176 else 177 ExAcquireResourceSharedLite(&Lock->Resource, TRUE); 178 179 return STATUS_SUCCESS; 180 } 181 182 BOOLEAN 183 NTAPI 184 RtlTryEnterHeapLock(IN OUT PHEAP_LOCK Lock, IN BOOLEAN Exclusive) 185 { 186 BOOLEAN Success; 187 KeEnterCriticalRegion(); 188 189 if (Exclusive) 190 Success = ExAcquireResourceExclusiveLite(&Lock->Resource, FALSE); 191 else 192 Success = ExAcquireResourceSharedLite(&Lock->Resource, FALSE); 193 194 if (!Success) 195 KeLeaveCriticalRegion(); 196 197 return Success; 198 } 199 200 NTSTATUS 201 NTAPI 202 RtlInitializeHeapLock(IN OUT PHEAP_LOCK *Lock) 203 { 204 PHEAP_LOCK HeapLock = ExAllocatePoolWithTag(NonPagedPool, 205 sizeof(HEAP_LOCK), 206 TAG_RTHL); 207 if (HeapLock == NULL) 208 return STATUS_NO_MEMORY; 209 210 ExInitializeResourceLite(&HeapLock->Resource); 211 *Lock = HeapLock; 212 213 return STATUS_SUCCESS; 214 } 215 216 NTSTATUS 217 NTAPI 218 RtlLeaveHeapLock(IN OUT PHEAP_LOCK Lock) 219 { 220 ExReleaseResourceLite(&Lock->Resource); 221 KeLeaveCriticalRegion(); 222 223 return STATUS_SUCCESS; 224 } 225 226 struct _HEAP; 227 228 VOID 229 NTAPI 230 RtlpAddHeapToProcessList(struct _HEAP *Heap) 231 { 232 UNREFERENCED_PARAMETER(Heap); 233 } 234 235 VOID 236 NTAPI 237 RtlpRemoveHeapFromProcessList(struct _HEAP *Heap) 238 { 239 UNREFERENCED_PARAMETER(Heap); 240 } 241 242 VOID 243 RtlInitializeHeapManager(VOID) 244 { 245 } 246 247 #if DBG 248 VOID FASTCALL 249 CHECK_PAGED_CODE_RTL(char *file, int line) 250 { 251 if(KeGetCurrentIrql() > APC_LEVEL) 252 { 253 DbgPrint("%s:%i: Pagable code called at IRQL > APC_LEVEL (%u)\n", file, line, KeGetCurrentIrql()); 254 ASSERT(FALSE); 255 } 256 } 257 #endif 258 259 VOID 260 NTAPI 261 RtlpSetHeapParameters(IN PRTL_HEAP_PARAMETERS Parameters) 262 { 263 /* Apply defaults for non-set parameters */ 264 if (!Parameters->SegmentCommit) Parameters->SegmentCommit = MmHeapSegmentCommit; 265 if (!Parameters->SegmentReserve) Parameters->SegmentReserve = MmHeapSegmentReserve; 266 if (!Parameters->DeCommitFreeBlockThreshold) Parameters->DeCommitFreeBlockThreshold = MmHeapDeCommitFreeBlockThreshold; 267 if (!Parameters->DeCommitTotalFreeThreshold) Parameters->DeCommitTotalFreeThreshold = MmHeapDeCommitTotalFreeThreshold; 268 } 269 270 VOID 271 NTAPI 272 RtlpCheckLogException(IN PEXCEPTION_RECORD ExceptionRecord, 273 IN PCONTEXT ContextRecord, 274 IN PVOID ContextData, 275 IN ULONG Size) 276 { 277 /* Check the global flag */ 278 if (NtGlobalFlag & FLG_ENABLE_EXCEPTION_LOGGING) 279 { 280 /* FIXME: Log this exception */ 281 } 282 } 283 284 BOOLEAN 285 NTAPI 286 RtlpHandleDpcStackException(IN PEXCEPTION_REGISTRATION_RECORD RegistrationFrame, 287 IN ULONG_PTR RegistrationFrameEnd, 288 IN OUT PULONG_PTR StackLow, 289 IN OUT PULONG_PTR StackHigh) 290 { 291 PKPRCB Prcb; 292 ULONG_PTR DpcStack; 293 294 /* Check if we are at DISPATCH or higher */ 295 if (KeGetCurrentIrql() >= DISPATCH_LEVEL) 296 { 297 /* Get the PRCB and DPC Stack */ 298 Prcb = KeGetCurrentPrcb(); 299 DpcStack = (ULONG_PTR)Prcb->DpcStack; 300 301 /* Check if we are in a DPC and the stack matches */ 302 if ((Prcb->DpcRoutineActive) && 303 (RegistrationFrameEnd <= DpcStack) && 304 ((ULONG_PTR)RegistrationFrame >= DpcStack - KERNEL_STACK_SIZE)) 305 { 306 /* Update the limits to the DPC Stack's */ 307 *StackHigh = DpcStack; 308 *StackLow = DpcStack - KERNEL_STACK_SIZE; 309 return TRUE; 310 } 311 } 312 313 /* Not in DPC stack */ 314 return FALSE; 315 } 316 317 #if !defined(_ARM_) && !defined(_AMD64_) 318 319 BOOLEAN 320 NTAPI 321 RtlpCaptureStackLimits(IN ULONG_PTR Ebp, 322 IN ULONG_PTR *StackBegin, 323 IN ULONG_PTR *StackEnd) 324 { 325 PKTHREAD Thread = KeGetCurrentThread(); 326 327 /* Don't even try at ISR level or later */ 328 if (KeGetCurrentIrql() > DISPATCH_LEVEL) return FALSE; 329 330 /* Start with defaults */ 331 *StackBegin = Thread->StackLimit; 332 *StackEnd = (ULONG_PTR)Thread->StackBase; 333 334 /* Check if EBP is inside the stack */ 335 if ((*StackBegin <= Ebp) && (Ebp <= *StackEnd)) 336 { 337 /* Then make the stack start at EBP */ 338 *StackBegin = Ebp; 339 } 340 else 341 { 342 /* Now we're going to assume we're on the DPC stack */ 343 *StackEnd = (ULONG_PTR)(KeGetPcr()->Prcb->DpcStack); 344 *StackBegin = *StackEnd - KERNEL_STACK_SIZE; 345 346 /* Check if we seem to be on the DPC stack */ 347 if ((*StackEnd) && (*StackBegin < Ebp) && (Ebp <= *StackEnd)) 348 { 349 /* We're on the DPC stack */ 350 *StackBegin = Ebp; 351 } 352 else 353 { 354 /* We're somewhere else entirely... use EBP for safety */ 355 *StackBegin = Ebp; 356 *StackEnd = (ULONG_PTR)PAGE_ALIGN(*StackBegin); 357 } 358 } 359 360 /* Return success */ 361 return TRUE; 362 } 363 364 /* 365 * @implemented 366 */ 367 ULONG 368 NTAPI 369 RtlWalkFrameChain(OUT PVOID *Callers, 370 IN ULONG Count, 371 IN ULONG Flags) 372 { 373 ULONG_PTR Stack, NewStack, StackBegin, StackEnd = 0; 374 ULONG Eip; 375 BOOLEAN Result, StopSearch = FALSE; 376 ULONG i = 0; 377 PETHREAD Thread = PsGetCurrentThread(); 378 PTEB Teb; 379 PKTRAP_FRAME TrapFrame; 380 381 /* Get current EBP */ 382 #if defined(_M_IX86) 383 #if defined __GNUC__ 384 __asm__("mov %%ebp, %0" : "=r" (Stack) : ); 385 #elif defined(_MSC_VER) 386 __asm mov Stack, ebp 387 #endif 388 #elif defined(_M_MIPS) 389 __asm__("move $sp, %0" : "=r" (Stack) : ); 390 #elif defined(_M_PPC) 391 __asm__("mr %0,1" : "=r" (Stack) : ); 392 #elif defined(_M_ARM) 393 __asm__("mov sp, %0" : "=r"(Stack) : ); 394 #else 395 #error Unknown architecture 396 #endif 397 398 /* Set it as the stack begin limit as well */ 399 StackBegin = (ULONG_PTR)Stack; 400 401 /* Check if we're called for non-logging mode */ 402 if (!Flags) 403 { 404 /* Get the actual safe limits */ 405 Result = RtlpCaptureStackLimits((ULONG_PTR)Stack, 406 &StackBegin, 407 &StackEnd); 408 if (!Result) return 0; 409 } 410 411 /* Use a SEH block for maximum protection */ 412 _SEH2_TRY 413 { 414 /* Check if we want the user-mode stack frame */ 415 if (Flags == 1) 416 { 417 /* Get the trap frame and TEB */ 418 TrapFrame = KeGetTrapFrame(&Thread->Tcb); 419 Teb = Thread->Tcb.Teb; 420 421 /* Make sure we can trust the TEB and trap frame */ 422 if (!(Teb) || 423 (KeIsAttachedProcess()) || 424 (KeGetCurrentIrql() >= DISPATCH_LEVEL)) 425 { 426 /* Invalid or unsafe attempt to get the stack */ 427 _SEH2_YIELD(return 0;) 428 } 429 430 /* Get the stack limits */ 431 StackBegin = (ULONG_PTR)Teb->NtTib.StackLimit; 432 StackEnd = (ULONG_PTR)Teb->NtTib.StackBase; 433 #ifdef _M_IX86 434 Stack = TrapFrame->Ebp; 435 #elif defined(_M_PPC) 436 Stack = TrapFrame->Gpr1; 437 #else 438 #error Unknown architecture 439 #endif 440 441 /* Validate them */ 442 if (StackEnd <= StackBegin) _SEH2_YIELD(return 0); 443 ProbeForRead((PVOID)StackBegin, 444 StackEnd - StackBegin, 445 sizeof(CHAR)); 446 } 447 448 /* Loop the frames */ 449 for (i = 0; i < Count; i++) 450 { 451 /* 452 * Leave if we're past the stack, 453 * if we're before the stack, 454 * or if we've reached ourselves. 455 */ 456 if ((Stack >= StackEnd) || 457 (!i ? (Stack < StackBegin) : (Stack <= StackBegin)) || 458 ((StackEnd - Stack) < (2 * sizeof(ULONG_PTR)))) 459 { 460 /* We're done or hit a bad address */ 461 break; 462 } 463 464 /* Get new stack and EIP */ 465 NewStack = *(PULONG_PTR)Stack; 466 Eip = *(PULONG_PTR)(Stack + sizeof(ULONG_PTR)); 467 468 /* Check if the new pointer is above the oldone and past the end */ 469 if (!((Stack < NewStack) && (NewStack < StackEnd))) 470 { 471 /* Stop searching after this entry */ 472 StopSearch = TRUE; 473 } 474 475 /* Also make sure that the EIP isn't a stack address */ 476 if ((StackBegin < Eip) && (Eip < StackEnd)) break; 477 478 /* Check if we reached a user-mode address */ 479 if (!(Flags) && !(Eip & 0x80000000)) break; // FIXME: 3GB breakage 480 481 /* Save this frame */ 482 Callers[i] = (PVOID)Eip; 483 484 /* Check if we should continue */ 485 if (StopSearch) 486 { 487 /* Return the next index */ 488 i++; 489 break; 490 } 491 492 /* Move to the next stack */ 493 Stack = NewStack; 494 } 495 } 496 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 497 { 498 /* No index */ 499 i = 0; 500 } 501 _SEH2_END; 502 503 /* Return frames parsed */ 504 return i; 505 } 506 507 #endif 508 509 #if defined(_M_AMD64) || defined(_M_ARM) 510 VOID 511 NTAPI 512 RtlpGetStackLimits( 513 OUT PULONG_PTR LowLimit, 514 OUT PULONG_PTR HighLimit) 515 { 516 PKTHREAD CurrentThread = KeGetCurrentThread(); 517 *HighLimit = (ULONG_PTR)CurrentThread->InitialStack; 518 *LowLimit = (ULONG_PTR)CurrentThread->StackLimit; 519 } 520 #endif 521 522 /* RTL Atom Tables ************************************************************/ 523 524 NTSTATUS 525 RtlpInitAtomTableLock(PRTL_ATOM_TABLE AtomTable) 526 { 527 ExInitializeFastMutex(&AtomTable->FastMutex); 528 529 return STATUS_SUCCESS; 530 } 531 532 533 VOID 534 RtlpDestroyAtomTableLock(PRTL_ATOM_TABLE AtomTable) 535 { 536 } 537 538 539 BOOLEAN 540 RtlpLockAtomTable(PRTL_ATOM_TABLE AtomTable) 541 { 542 ExAcquireFastMutex(&AtomTable->FastMutex); 543 return TRUE; 544 } 545 546 VOID 547 RtlpUnlockAtomTable(PRTL_ATOM_TABLE AtomTable) 548 { 549 ExReleaseFastMutex(&AtomTable->FastMutex); 550 } 551 552 BOOLEAN 553 RtlpCreateAtomHandleTable(PRTL_ATOM_TABLE AtomTable) 554 { 555 AtomTable->ExHandleTable = ExCreateHandleTable(NULL); 556 return (AtomTable->ExHandleTable != NULL); 557 } 558 559 BOOLEAN 560 NTAPI 561 RtlpCloseHandleCallback( 562 IN PHANDLE_TABLE_ENTRY HandleTableEntry, 563 IN HANDLE Handle, 564 IN PVOID HandleTable) 565 { 566 /* Destroy and unlock the handle entry */ 567 return ExDestroyHandle(HandleTable, Handle, HandleTableEntry); 568 } 569 570 VOID 571 RtlpDestroyAtomHandleTable(PRTL_ATOM_TABLE AtomTable) 572 { 573 if (AtomTable->ExHandleTable) 574 { 575 ExSweepHandleTable(AtomTable->ExHandleTable, 576 RtlpCloseHandleCallback, 577 AtomTable->ExHandleTable); 578 ExDestroyHandleTable(AtomTable->ExHandleTable, NULL); 579 AtomTable->ExHandleTable = NULL; 580 } 581 } 582 583 PRTL_ATOM_TABLE 584 RtlpAllocAtomTable(ULONG Size) 585 { 586 PRTL_ATOM_TABLE Table = ExAllocatePoolWithTag(NonPagedPool, 587 Size, 588 TAG_ATMT); 589 if (Table != NULL) 590 { 591 RtlZeroMemory(Table, 592 Size); 593 } 594 595 return Table; 596 } 597 598 VOID 599 RtlpFreeAtomTable(PRTL_ATOM_TABLE AtomTable) 600 { 601 ExFreePoolWithTag(AtomTable, TAG_ATMT); 602 } 603 604 PRTL_ATOM_TABLE_ENTRY 605 RtlpAllocAtomTableEntry(ULONG Size) 606 { 607 PRTL_ATOM_TABLE_ENTRY Entry; 608 609 Entry = ExAllocatePoolWithTag(NonPagedPool, Size, TAG_ATMT); 610 if (Entry != NULL) 611 { 612 RtlZeroMemory(Entry, Size); 613 } 614 615 return Entry; 616 } 617 618 VOID 619 RtlpFreeAtomTableEntry(PRTL_ATOM_TABLE_ENTRY Entry) 620 { 621 ExFreePoolWithTag(Entry, TAG_ATMT); 622 } 623 624 VOID 625 RtlpFreeAtomHandle(PRTL_ATOM_TABLE AtomTable, PRTL_ATOM_TABLE_ENTRY Entry) 626 { 627 ExDestroyHandle(AtomTable->ExHandleTable, 628 (HANDLE)((ULONG_PTR)Entry->HandleIndex << 2), 629 NULL); 630 } 631 632 BOOLEAN 633 RtlpCreateAtomHandle(PRTL_ATOM_TABLE AtomTable, PRTL_ATOM_TABLE_ENTRY Entry) 634 { 635 HANDLE_TABLE_ENTRY ExEntry; 636 HANDLE Handle; 637 USHORT HandleIndex; 638 639 /* Initialize ex handle table entry */ 640 ExEntry.Object = Entry; 641 ExEntry.GrantedAccess = 0x1; /* FIXME - valid handle */ 642 643 /* Create ex handle */ 644 Handle = ExCreateHandle(AtomTable->ExHandleTable, 645 &ExEntry); 646 if (!Handle) return FALSE; 647 648 /* Calculate HandleIndex (by getting rid of the first two bits) */ 649 HandleIndex = (USHORT)((ULONG_PTR)Handle >> 2); 650 651 /* Index must be less than 0xC000 */ 652 if (HandleIndex >= 0xC000) 653 { 654 /* Destroy ex handle */ 655 ExDestroyHandle(AtomTable->ExHandleTable, 656 Handle, 657 NULL); 658 659 /* Return failure */ 660 return FALSE; 661 } 662 663 /* Initialize atom table entry */ 664 Entry->HandleIndex = HandleIndex; 665 Entry->Atom = 0xC000 + HandleIndex; 666 667 /* Return success */ 668 return TRUE; 669 } 670 671 PRTL_ATOM_TABLE_ENTRY 672 RtlpGetAtomEntry(PRTL_ATOM_TABLE AtomTable, ULONG Index) 673 { 674 PHANDLE_TABLE_ENTRY ExEntry; 675 PRTL_ATOM_TABLE_ENTRY Entry = NULL; 676 677 /* NOTE: There's no need to explicitly enter a critical region because it's 678 guaranteed that we're in a critical region right now (as we hold 679 the atom table lock) */ 680 681 ExEntry = ExMapHandleToPointer(AtomTable->ExHandleTable, 682 (HANDLE)((ULONG_PTR)Index << 2)); 683 if (ExEntry != NULL) 684 { 685 Entry = ExEntry->Object; 686 687 ExUnlockHandleTableEntry(AtomTable->ExHandleTable, 688 ExEntry); 689 } 690 691 return Entry; 692 } 693 694 /* 695 * Ldr Resource support code 696 */ 697 698 IMAGE_RESOURCE_DIRECTORY *find_entry_by_name( IMAGE_RESOURCE_DIRECTORY *dir, 699 LPCWSTR name, void *root, 700 int want_dir ); 701 IMAGE_RESOURCE_DIRECTORY *find_entry_by_id( IMAGE_RESOURCE_DIRECTORY *dir, 702 USHORT id, void *root, int want_dir ); 703 IMAGE_RESOURCE_DIRECTORY *find_first_entry( IMAGE_RESOURCE_DIRECTORY *dir, 704 void *root, int want_dir ); 705 706 /********************************************************************** 707 * find_entry 708 * 709 * Find a resource entry 710 */ 711 NTSTATUS find_entry( PVOID BaseAddress, LDR_RESOURCE_INFO *info, 712 ULONG level, void **ret, int want_dir ) 713 { 714 ULONG size; 715 void *root; 716 IMAGE_RESOURCE_DIRECTORY *resdirptr; 717 718 root = RtlImageDirectoryEntryToData( BaseAddress, TRUE, IMAGE_DIRECTORY_ENTRY_RESOURCE, &size ); 719 if (!root) return STATUS_RESOURCE_DATA_NOT_FOUND; 720 if (size < sizeof(*resdirptr)) return STATUS_RESOURCE_DATA_NOT_FOUND; 721 resdirptr = root; 722 723 if (!level--) goto done; 724 if (!(*ret = find_entry_by_name( resdirptr, (LPCWSTR)info->Type, root, want_dir || level ))) 725 return STATUS_RESOURCE_TYPE_NOT_FOUND; 726 if (!level--) return STATUS_SUCCESS; 727 728 resdirptr = *ret; 729 if (!(*ret = find_entry_by_name( resdirptr, (LPCWSTR)info->Name, root, want_dir || level ))) 730 return STATUS_RESOURCE_NAME_NOT_FOUND; 731 if (!level--) return STATUS_SUCCESS; 732 if (level) return STATUS_INVALID_PARAMETER; /* level > 3 */ 733 734 resdirptr = *ret; 735 736 if ((*ret = find_first_entry( resdirptr, root, want_dir ))) return STATUS_SUCCESS; 737 738 return STATUS_RESOURCE_DATA_NOT_FOUND; 739 740 done: 741 *ret = resdirptr; 742 return STATUS_SUCCESS; 743 } 744 745 NTSTATUS 746 NTAPI 747 RtlpSafeCopyMemory( 748 _Out_writes_bytes_all_(Length) VOID UNALIGNED *Destination, 749 _In_reads_bytes_(Length) CONST VOID UNALIGNED *Source, 750 _In_ SIZE_T Length) 751 { 752 _SEH2_TRY 753 { 754 RtlCopyMemory(Destination, Source, Length); 755 } 756 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 757 { 758 _SEH2_YIELD(return _SEH2_GetExceptionCode()); 759 } 760 _SEH2_END; 761 762 return STATUS_SUCCESS; 763 } 764 765 BOOLEAN 766 NTAPI 767 RtlCallVectoredExceptionHandlers(_In_ PEXCEPTION_RECORD ExceptionRecord, 768 _In_ PCONTEXT Context) 769 { 770 /* In the kernel we don't have vectored exception handlers */ 771 return FALSE; 772 } 773 774 VOID 775 NTAPI 776 RtlCallVectoredContinueHandlers(_In_ PEXCEPTION_RECORD ExceptionRecord, 777 _In_ PCONTEXT Context) 778 { 779 /* No vectored continue handlers either in kernel mode */ 780 return; 781 } 782 783 /* EOF */ 784