1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS system libraries 4 * FILE: dll/win32/kernel32/client/utils.c 5 * PURPOSE: Utility and Support Functions 6 * PROGRAMMER: Alex Ionescu (alex@relsoft.net) 7 * Pierre Schweitzer (pierre.schweitzer@reactos.org) 8 */ 9 10 /* INCLUDES ******************************************************************/ 11 12 #include <k32.h> 13 #ifdef _M_IX86 14 #include "i386/ketypes.h" 15 #elif defined _M_AMD64 16 #include "amd64/ketypes.h" 17 #endif 18 19 #define NDEBUG 20 #include <debug.h> 21 22 /* GLOBALS ********************************************************************/ 23 24 UNICODE_STRING Restricted = RTL_CONSTANT_STRING(L"Restricted"); 25 BOOL bIsFileApiAnsi = TRUE; // set the file api to ansi or oem 26 PRTL_CONVERT_STRING Basep8BitStringToUnicodeString = RtlAnsiStringToUnicodeString; 27 PRTL_CONVERT_STRINGA BasepUnicodeStringTo8BitString = RtlUnicodeStringToAnsiString; 28 PRTL_COUNT_STRING BasepUnicodeStringTo8BitSize = BasepUnicodeStringToAnsiSize; 29 PRTL_COUNT_STRINGA Basep8BitStringToUnicodeSize = BasepAnsiStringToUnicodeSize; 30 31 /* FUNCTIONS ******************************************************************/ 32 33 ULONG 34 NTAPI 35 BasepUnicodeStringToOemSize(IN PUNICODE_STRING String) 36 { 37 return RtlUnicodeStringToOemSize(String); 38 } 39 40 ULONG 41 NTAPI 42 BasepOemStringToUnicodeSize(IN PANSI_STRING String) 43 { 44 return RtlOemStringToUnicodeSize(String); 45 } 46 47 ULONG 48 NTAPI 49 BasepUnicodeStringToAnsiSize(IN PUNICODE_STRING String) 50 { 51 return RtlUnicodeStringToAnsiSize(String); 52 } 53 54 ULONG 55 NTAPI 56 BasepAnsiStringToUnicodeSize(IN PANSI_STRING String) 57 { 58 return RtlAnsiStringToUnicodeSize(String); 59 } 60 61 HANDLE 62 WINAPI 63 BaseGetNamedObjectDirectory(VOID) 64 { 65 OBJECT_ATTRIBUTES ObjectAttributes; 66 NTSTATUS Status; 67 HANDLE DirHandle, BnoHandle, Token, NewToken; 68 69 if (BaseNamedObjectDirectory) return BaseNamedObjectDirectory; 70 71 if (NtCurrentTeb()->IsImpersonating) 72 { 73 Status = NtOpenThreadToken(NtCurrentThread(), 74 TOKEN_IMPERSONATE, 75 TRUE, 76 &Token); 77 if (!NT_SUCCESS(Status)) return BaseNamedObjectDirectory; 78 79 NewToken = NULL; 80 Status = NtSetInformationThread(NtCurrentThread(), 81 ThreadImpersonationToken, 82 &NewToken, 83 sizeof(HANDLE)); 84 if (!NT_SUCCESS (Status)) 85 { 86 NtClose(Token); 87 return BaseNamedObjectDirectory; 88 } 89 } 90 else 91 { 92 Token = NULL; 93 } 94 95 RtlAcquirePebLock(); 96 if (BaseNamedObjectDirectory) goto Quickie; 97 98 InitializeObjectAttributes(&ObjectAttributes, 99 &BaseStaticServerData->NamedObjectDirectory, 100 OBJ_CASE_INSENSITIVE, 101 NULL, 102 NULL); 103 104 Status = NtOpenDirectoryObject(&BnoHandle, 105 DIRECTORY_QUERY | 106 DIRECTORY_TRAVERSE | 107 DIRECTORY_CREATE_OBJECT | 108 DIRECTORY_CREATE_SUBDIRECTORY, 109 &ObjectAttributes); 110 if (!NT_SUCCESS(Status)) 111 { 112 Status = NtOpenDirectoryObject(&DirHandle, 113 DIRECTORY_TRAVERSE, 114 &ObjectAttributes); 115 116 if (NT_SUCCESS(Status)) 117 { 118 InitializeObjectAttributes(&ObjectAttributes, 119 (PUNICODE_STRING)&Restricted, 120 OBJ_CASE_INSENSITIVE, 121 DirHandle, 122 NULL); 123 124 Status = NtOpenDirectoryObject(&BnoHandle, 125 DIRECTORY_QUERY | 126 DIRECTORY_TRAVERSE | 127 DIRECTORY_CREATE_OBJECT | 128 DIRECTORY_CREATE_SUBDIRECTORY, 129 &ObjectAttributes); 130 NtClose(DirHandle); 131 132 } 133 } 134 135 if (NT_SUCCESS(Status)) BaseNamedObjectDirectory = BnoHandle; 136 137 Quickie: 138 139 RtlReleasePebLock(); 140 141 if (Token) 142 { 143 NtSetInformationThread(NtCurrentThread(), 144 ThreadImpersonationToken, 145 &Token, 146 sizeof(Token)); 147 148 NtClose(Token); 149 } 150 151 return BaseNamedObjectDirectory; 152 } 153 154 VOID 155 NTAPI 156 BasepLocateExeLdrEntry(IN PLDR_DATA_TABLE_ENTRY Entry, 157 IN PVOID Context, 158 OUT BOOLEAN *StopEnumeration) 159 { 160 /* Make sure we get Entry, Context and valid StopEnumeration pointer */ 161 ASSERT(Entry); 162 ASSERT(Context); 163 ASSERT(StopEnumeration); 164 165 /* If entry is already found - signal to stop */ 166 if (BasepExeLdrEntry) 167 { 168 *StopEnumeration = TRUE; 169 return; 170 } 171 172 /* Otherwise keep enumerating until we find a match */ 173 if (Entry->DllBase == Context) 174 { 175 /* It matches, so remember the ldr entry */ 176 BasepExeLdrEntry = Entry; 177 178 /* And stop enumeration */ 179 *StopEnumeration = TRUE; 180 } 181 } 182 183 /* 184 * Converts an ANSI or OEM String to the TEB StaticUnicodeString 185 */ 186 PUNICODE_STRING 187 WINAPI 188 Basep8BitStringToStaticUnicodeString(IN LPCSTR String) 189 { 190 PUNICODE_STRING StaticString = &(NtCurrentTeb()->StaticUnicodeString); 191 ANSI_STRING AnsiString; 192 NTSTATUS Status; 193 194 /* Initialize an ANSI String */ 195 Status = RtlInitAnsiStringEx(&AnsiString, String); 196 if (!NT_SUCCESS(Status)) 197 { 198 Status = STATUS_BUFFER_OVERFLOW; 199 } 200 else 201 { 202 /* Convert it */ 203 Status = Basep8BitStringToUnicodeString(StaticString, &AnsiString, FALSE); 204 } 205 206 if (NT_SUCCESS(Status)) return StaticString; 207 208 if (Status == STATUS_BUFFER_OVERFLOW) 209 { 210 SetLastError(ERROR_FILENAME_EXCED_RANGE); 211 } 212 else 213 { 214 BaseSetLastNTError(Status); 215 } 216 217 return NULL; 218 } 219 220 /* 221 * Allocates space from the Heap and converts an Unicode String into it 222 */ 223 BOOLEAN 224 WINAPI 225 Basep8BitStringToDynamicUnicodeString(OUT PUNICODE_STRING UnicodeString, 226 IN LPCSTR String) 227 { 228 ANSI_STRING AnsiString; 229 NTSTATUS Status; 230 231 /* Initialize an ANSI String */ 232 Status = RtlInitAnsiStringEx(&AnsiString, String); 233 if (!NT_SUCCESS(Status)) 234 { 235 Status = STATUS_BUFFER_OVERFLOW; 236 } 237 else 238 { 239 /* Convert it */ 240 Status = Basep8BitStringToUnicodeString(UnicodeString, &AnsiString, TRUE); 241 } 242 243 if (NT_SUCCESS(Status)) return TRUE; 244 245 if (Status == STATUS_BUFFER_OVERFLOW) 246 { 247 SetLastError(ERROR_FILENAME_EXCED_RANGE); 248 } 249 else 250 { 251 BaseSetLastNTError(Status); 252 } 253 254 return FALSE; 255 } 256 257 /* 258 * Allocates space from the Heap and converts an Ansi String into it 259 */ 260 /*NOTE: API IS A HACK */ 261 VOID 262 WINAPI 263 BasepAnsiStringToHeapUnicodeString(IN LPCSTR AnsiString, 264 OUT LPWSTR* UnicodeString) 265 { 266 ANSI_STRING AnsiTemp; 267 UNICODE_STRING UnicodeTemp; 268 269 DPRINT("BasepAnsiStringToHeapUnicodeString\n"); 270 271 /* First create the ANSI_STRING */ 272 RtlInitAnsiString(&AnsiTemp, AnsiString); 273 274 if (NT_SUCCESS(RtlAnsiStringToUnicodeString(&UnicodeTemp, 275 &AnsiTemp, 276 TRUE))) 277 { 278 *UnicodeString = UnicodeTemp.Buffer; 279 } 280 else 281 { 282 *UnicodeString = NULL; 283 } 284 } 285 286 PLARGE_INTEGER 287 WINAPI 288 BaseFormatTimeOut(OUT PLARGE_INTEGER Timeout, 289 IN DWORD dwMilliseconds) 290 { 291 /* Check if this is an infinite wait, which means no timeout argument */ 292 if (dwMilliseconds == INFINITE) return NULL; 293 294 /* Otherwise, convert the time to NT Format */ 295 Timeout->QuadPart = dwMilliseconds * -10000LL; 296 return Timeout; 297 } 298 299 /* 300 * Converts lpSecurityAttributes + Object Name into ObjectAttributes. 301 */ 302 POBJECT_ATTRIBUTES 303 WINAPI 304 BaseFormatObjectAttributes(OUT POBJECT_ATTRIBUTES ObjectAttributes, 305 IN PSECURITY_ATTRIBUTES SecurityAttributes OPTIONAL, 306 IN PUNICODE_STRING ObjectName) 307 { 308 ULONG Attributes; 309 HANDLE RootDirectory; 310 PVOID SecurityDescriptor; 311 DPRINT("BaseFormatObjectAttributes. Security: %p, Name: %p\n", 312 SecurityAttributes, ObjectName); 313 314 /* Get the attributes if present */ 315 if (SecurityAttributes) 316 { 317 Attributes = SecurityAttributes->bInheritHandle ? OBJ_INHERIT : 0; 318 SecurityDescriptor = SecurityAttributes->lpSecurityDescriptor; 319 } 320 else 321 { 322 if (!ObjectName) return NULL; 323 Attributes = 0; 324 SecurityDescriptor = NULL; 325 } 326 327 if (ObjectName) 328 { 329 Attributes |= OBJ_OPENIF; 330 RootDirectory = BaseGetNamedObjectDirectory(); 331 } 332 else 333 { 334 RootDirectory = NULL; 335 } 336 337 /* Create the Object Attributes */ 338 InitializeObjectAttributes(ObjectAttributes, 339 ObjectName, 340 Attributes, 341 RootDirectory, 342 SecurityDescriptor); 343 DPRINT("Attributes: %lx, RootDirectory: %p, SecurityDescriptor: %p\n", 344 Attributes, RootDirectory, SecurityDescriptor); 345 return ObjectAttributes; 346 } 347 348 /* 349 * Creates a stack for a thread or fiber. 350 * NOTE: Adapted from sdk/lib/rtl/thread.c:RtlpCreateUserStack(). 351 */ 352 NTSTATUS 353 WINAPI 354 BaseCreateStack( 355 _In_ HANDLE hProcess, 356 _In_opt_ SIZE_T StackCommit, 357 _In_opt_ SIZE_T StackReserve, 358 _Out_ PINITIAL_TEB InitialTeb) 359 { 360 NTSTATUS Status; 361 PIMAGE_NT_HEADERS Headers; 362 ULONG_PTR Stack; 363 BOOLEAN UseGuard; 364 ULONG PageSize, AllocationGranularity, Dummy; 365 SIZE_T MinimumStackCommit, GuardPageSize; 366 367 DPRINT("BaseCreateStack(hProcess: 0x%p, Max: 0x%lx, Current: 0x%lx)\n", 368 hProcess, StackReserve, StackCommit); 369 370 /* Read page size */ 371 PageSize = BaseStaticServerData->SysInfo.PageSize; 372 AllocationGranularity = BaseStaticServerData->SysInfo.AllocationGranularity; 373 374 /* Get the Image Headers */ 375 Headers = RtlImageNtHeader(NtCurrentPeb()->ImageBaseAddress); 376 if (!Headers) return STATUS_INVALID_IMAGE_FORMAT; 377 378 if (StackReserve == 0) 379 StackReserve = Headers->OptionalHeader.SizeOfStackReserve; 380 381 if (StackCommit == 0) 382 { 383 StackCommit = Headers->OptionalHeader.SizeOfStackCommit; 384 } 385 /* Check if the commit is higher than the reserve */ 386 else if (StackCommit >= StackReserve) 387 { 388 /* Grow the reserve beyond the commit, up to 1MB alignment */ 389 StackReserve = ROUND_UP(StackCommit, 1024 * 1024); 390 } 391 392 /* Align everything to Page Size */ 393 StackCommit = ROUND_UP(StackCommit, PageSize); 394 StackReserve = ROUND_UP(StackReserve, AllocationGranularity); 395 396 MinimumStackCommit = NtCurrentPeb()->MinimumStackCommit; 397 if ((MinimumStackCommit != 0) && (StackCommit < MinimumStackCommit)) 398 { 399 StackCommit = MinimumStackCommit; 400 } 401 402 /* Check if the commit is higher than the reserve */ 403 if (StackCommit >= StackReserve) 404 { 405 /* Grow the reserve beyond the commit, up to 1MB alignment */ 406 StackReserve = ROUND_UP(StackCommit, 1024 * 1024); 407 } 408 409 /* Align everything to Page Size */ 410 StackCommit = ROUND_UP(StackCommit, PageSize); 411 StackReserve = ROUND_UP(StackReserve, AllocationGranularity); 412 413 /* Reserve memory for the stack */ 414 Stack = 0; 415 Status = NtAllocateVirtualMemory(hProcess, 416 (PVOID*)&Stack, 417 0, 418 &StackReserve, 419 MEM_RESERVE, 420 PAGE_READWRITE); 421 if (!NT_SUCCESS(Status)) 422 { 423 DPRINT1("Failure to reserve stack: %lx\n", Status); 424 return Status; 425 } 426 427 /* Now set up some basic Initial TEB Parameters */ 428 InitialTeb->AllocatedStackBase = (PVOID)Stack; 429 InitialTeb->StackBase = (PVOID)(Stack + StackReserve); 430 InitialTeb->PreviousStackBase = NULL; 431 InitialTeb->PreviousStackLimit = NULL; 432 433 /* Update the stack position */ 434 Stack += StackReserve - StackCommit; 435 436 /* Check if we can add a guard page */ 437 if (StackReserve >= StackCommit + PageSize) 438 { 439 Stack -= PageSize; 440 StackCommit += PageSize; 441 UseGuard = TRUE; 442 } 443 else 444 { 445 UseGuard = FALSE; 446 } 447 448 /* Allocate memory for the stack */ 449 Status = NtAllocateVirtualMemory(hProcess, 450 (PVOID*)&Stack, 451 0, 452 &StackCommit, 453 MEM_COMMIT, 454 PAGE_READWRITE); 455 if (!NT_SUCCESS(Status)) 456 { 457 DPRINT1("Failure to allocate stack\n"); 458 GuardPageSize = 0; 459 NtFreeVirtualMemory(hProcess, (PVOID*)&Stack, &GuardPageSize, MEM_RELEASE); 460 return Status; 461 } 462 463 /* Now set the current Stack Limit */ 464 InitialTeb->StackLimit = (PVOID)Stack; 465 466 /* Create a guard page if needed */ 467 if (UseGuard) 468 { 469 GuardPageSize = PageSize; 470 Status = NtProtectVirtualMemory(hProcess, 471 (PVOID*)&Stack, 472 &GuardPageSize, 473 PAGE_GUARD | PAGE_READWRITE, 474 &Dummy); 475 if (!NT_SUCCESS(Status)) 476 { 477 DPRINT1("Failure to set guard page\n"); 478 return Status; 479 } 480 481 /* Update the Stack Limit keeping in mind the Guard Page */ 482 InitialTeb->StackLimit = (PVOID)((ULONG_PTR)InitialTeb->StackLimit + 483 GuardPageSize); 484 } 485 486 /* We are done! */ 487 return STATUS_SUCCESS; 488 } 489 490 /* 491 * NOTE: Adapted from sdk/lib/rtl/thread.c:RtlpFreeUserStack(). 492 */ 493 VOID 494 WINAPI 495 BaseFreeThreadStack( 496 _In_ HANDLE hProcess, 497 _In_ PINITIAL_TEB InitialTeb) 498 { 499 SIZE_T Dummy = 0; 500 501 /* Free the Stack */ 502 NtFreeVirtualMemory(hProcess, 503 &InitialTeb->AllocatedStackBase, 504 &Dummy, 505 MEM_RELEASE); 506 } 507 508 /* 509 * Creates the Initial Context for a Thread or Fiber 510 */ 511 VOID 512 WINAPI 513 BaseInitializeContext(IN PCONTEXT Context, 514 IN PVOID Parameter, 515 IN PVOID StartAddress, 516 IN PVOID StackAddress, 517 IN ULONG ContextType) 518 { 519 #ifdef _M_IX86 520 ULONG ContextFlags; 521 DPRINT("BaseInitializeContext: %p\n", Context); 522 523 /* Setup the Initial Win32 Thread Context */ 524 Context->Eax = (ULONG)StartAddress; 525 Context->Ebx = (ULONG)Parameter; 526 Context->Esp = (ULONG)StackAddress; 527 /* The other registers are undefined */ 528 529 /* Setup the Segments */ 530 Context->SegFs = KGDT_R3_TEB; 531 Context->SegEs = KGDT_R3_DATA; 532 Context->SegDs = KGDT_R3_DATA; 533 Context->SegCs = KGDT_R3_CODE; 534 Context->SegSs = KGDT_R3_DATA; 535 Context->SegGs = 0; 536 537 /* Set the Context Flags */ 538 ContextFlags = Context->ContextFlags; 539 Context->ContextFlags = CONTEXT_FULL; 540 541 /* Give it some room for the Parameter */ 542 Context->Esp -= sizeof(PVOID); 543 544 /* Set the EFLAGS */ 545 Context->EFlags = 0x3000; /* IOPL 3 */ 546 547 /* What kind of context is being created? */ 548 if (ContextType == 1) 549 { 550 /* For Threads */ 551 Context->Eip = (ULONG)BaseThreadStartupThunk; 552 } 553 else if (ContextType == 2) 554 { 555 /* This is a fiber: make space for the return address */ 556 Context->Esp -= sizeof(PVOID); 557 *((PVOID*)Context->Esp) = BaseFiberStartup; 558 559 /* Is FPU state required? */ 560 Context->ContextFlags |= ContextFlags; 561 if ((ContextFlags & CONTEXT_FLOATING_POINT) == CONTEXT_FLOATING_POINT) 562 { 563 /* Set an initial state */ 564 Context->FloatSave.ControlWord = 0x27F; 565 Context->FloatSave.StatusWord = 0; 566 Context->FloatSave.TagWord = 0xFFFF; 567 Context->FloatSave.ErrorOffset = 0; 568 Context->FloatSave.ErrorSelector = 0; 569 Context->FloatSave.DataOffset = 0; 570 Context->FloatSave.DataSelector = 0; 571 if (SharedUserData->ProcessorFeatures[PF_XMMI_INSTRUCTIONS_AVAILABLE]) 572 Context->Dr6 = 0x1F80; 573 } 574 } 575 else 576 { 577 /* For first thread in a Process */ 578 Context->Eip = (ULONG)BaseProcessStartThunk; 579 } 580 581 #elif defined(_M_AMD64) 582 DPRINT("BaseInitializeContext: %p\n", Context); 583 ASSERT(((ULONG_PTR)StackAddress & 15) == 0); 584 585 RtlZeroMemory(Context, sizeof(*Context)); 586 587 /* Setup the Initial Win32 Thread Context */ 588 Context->Rcx = (ULONG_PTR)StartAddress; 589 Context->Rdx = (ULONG_PTR)Parameter; 590 Context->Rsp = (ULONG_PTR)StackAddress - 5 * sizeof(PVOID); 591 592 /* Setup the Segments */ 593 Context->SegGs = KGDT64_R3_DATA | RPL_MASK; 594 Context->SegEs = KGDT64_R3_DATA | RPL_MASK; 595 Context->SegDs = KGDT64_R3_DATA | RPL_MASK; 596 Context->SegCs = KGDT64_R3_CODE | RPL_MASK; 597 Context->SegSs = KGDT64_R3_DATA | RPL_MASK; 598 Context->SegFs = KGDT64_R3_CMTEB | RPL_MASK; 599 600 /* Set the EFLAGS */ 601 Context->EFlags = 0x3000 | EFLAGS_INTERRUPT_MASK; /* IOPL 3 */ 602 603 if (ContextType == 1) /* For Threads */ 604 { 605 Context->Rip = (ULONG_PTR)BaseThreadStartup; 606 } 607 else if (ContextType == 2) /* For Fibers */ 608 { 609 Context->Rip = (ULONG_PTR)BaseFiberStartup; 610 } 611 else /* For first thread in a Process */ 612 { 613 Context->Rip = (ULONG_PTR)BaseProcessStartup; 614 } 615 616 /* Set the Context Flags */ 617 Context->ContextFlags = CONTEXT_FULL; 618 #elif defined(_M_ARM) 619 DPRINT("BaseInitializeContext: %p\n", Context); 620 621 // FIXME: check if this is correct! 622 /* Setup the Initial Win32 Thread Context */ 623 Context->R0 = (ULONG_PTR)StartAddress; 624 Context->R1 = (ULONG_PTR)Parameter; 625 Context->Sp = (ULONG_PTR)StackAddress; 626 627 if (ContextType == 1) /* For Threads */ 628 { 629 Context->Pc = (ULONG_PTR)BaseThreadStartupThunk; 630 } 631 else if (ContextType == 2) /* For Fibers */ 632 { 633 Context->Pc = (ULONG_PTR)BaseFiberStartup; 634 } 635 else /* For first thread in a Process */ 636 { 637 Context->Pc = (ULONG_PTR)BaseProcessStartThunk; 638 } 639 640 /* Set the Context Flags */ 641 Context->ContextFlags = CONTEXT_FULL; 642 643 /* Give it some room for the Parameter */ 644 Context->Sp -= sizeof(PVOID); 645 #else 646 #warning Unknown architecture 647 UNIMPLEMENTED; 648 DbgBreakPoint(); 649 #endif 650 } 651 652 /* 653 * Checks if the privilege for Real-Time Priority is there 654 * Beware about this function behavior: 655 * - In case Keep is set to FALSE, then the function will only check 656 * whether real time is allowed and won't grant the privilege. In that case 657 * it will return TRUE if allowed, FALSE otherwise. Not a state! 658 * It means you don't have to release privilege when calling with FALSE. 659 */ 660 PVOID 661 WINAPI 662 BasepIsRealtimeAllowed(IN BOOLEAN Keep) 663 { 664 ULONG Privilege = SE_INC_BASE_PRIORITY_PRIVILEGE; 665 PVOID State; 666 NTSTATUS Status; 667 668 Status = RtlAcquirePrivilege(&Privilege, 1, 0, &State); 669 if (!NT_SUCCESS(Status)) return NULL; 670 671 if (!Keep) 672 { 673 RtlReleasePrivilege(State); 674 State = (PVOID)TRUE; 675 } 676 677 return State; 678 } 679 680 /* 681 * Maps an image file into a section 682 */ 683 NTSTATUS 684 WINAPI 685 BasepMapFile(IN LPCWSTR lpApplicationName, 686 OUT PHANDLE hSection, 687 IN PUNICODE_STRING ApplicationName) 688 { 689 RTL_RELATIVE_NAME_U RelativeName; 690 OBJECT_ATTRIBUTES ObjectAttributes; 691 NTSTATUS Status; 692 HANDLE hFile = NULL; 693 IO_STATUS_BLOCK IoStatusBlock; 694 695 DPRINT("BasepMapFile\n"); 696 697 /* Zero out the Relative Directory */ 698 RelativeName.ContainingDirectory = NULL; 699 700 /* Find the application name */ 701 if (!RtlDosPathNameToNtPathName_U(lpApplicationName, 702 ApplicationName, 703 NULL, 704 &RelativeName)) 705 { 706 return STATUS_OBJECT_PATH_NOT_FOUND; 707 } 708 709 DPRINT("ApplicationName %wZ\n", ApplicationName); 710 DPRINT("RelativeName %wZ\n", &RelativeName.RelativeName); 711 712 /* Did we get a relative name? */ 713 if (RelativeName.RelativeName.Length) 714 { 715 ApplicationName = &RelativeName.RelativeName; 716 } 717 718 /* Initialize the Object Attributes */ 719 InitializeObjectAttributes(&ObjectAttributes, 720 ApplicationName, 721 OBJ_CASE_INSENSITIVE, 722 RelativeName.ContainingDirectory, 723 NULL); 724 725 /* Try to open the executable */ 726 Status = NtOpenFile(&hFile, 727 SYNCHRONIZE | FILE_EXECUTE | FILE_READ_DATA, 728 &ObjectAttributes, 729 &IoStatusBlock, 730 FILE_SHARE_DELETE | FILE_SHARE_READ, 731 FILE_SYNCHRONOUS_IO_NONALERT | FILE_NON_DIRECTORY_FILE); 732 if (!NT_SUCCESS(Status)) 733 { 734 DPRINT1("Failed to open file\n"); 735 BaseSetLastNTError(Status); 736 return Status; 737 } 738 739 /* Create a section for this file */ 740 Status = NtCreateSection(hSection, 741 SECTION_ALL_ACCESS, 742 NULL, 743 NULL, 744 PAGE_EXECUTE, 745 SEC_IMAGE, 746 hFile); 747 NtClose(hFile); 748 749 /* Return status */ 750 DPRINT("Section: %p for file: %p\n", *hSection, hFile); 751 return Status; 752 } 753 754 /* 755 * @implemented 756 */ 757 BOOLEAN 758 WINAPI 759 Wow64EnableWow64FsRedirection(IN BOOLEAN Wow64EnableWow64FsRedirection) 760 { 761 NTSTATUS Status; 762 BOOL Result; 763 764 Status = RtlWow64EnableFsRedirection(Wow64EnableWow64FsRedirection); 765 if (NT_SUCCESS(Status)) 766 { 767 Result = TRUE; 768 } 769 else 770 { 771 BaseSetLastNTError(Status); 772 Result = FALSE; 773 } 774 return Result; 775 } 776 777 /* 778 * @implemented 779 */ 780 BOOL 781 WINAPI 782 Wow64DisableWow64FsRedirection(IN PVOID *OldValue) 783 { 784 NTSTATUS Status; 785 BOOL Result; 786 787 Status = RtlWow64EnableFsRedirectionEx((PVOID)TRUE, OldValue); 788 if (NT_SUCCESS(Status)) 789 { 790 Result = TRUE; 791 } 792 else 793 { 794 BaseSetLastNTError(Status); 795 Result = FALSE; 796 } 797 return Result; 798 } 799 800 /* 801 * @implemented 802 */ 803 BOOL 804 WINAPI 805 Wow64RevertWow64FsRedirection(IN PVOID OldValue) 806 { 807 NTSTATUS Status; 808 BOOL Result; 809 810 Status = RtlWow64EnableFsRedirectionEx(OldValue, &OldValue); 811 if (NT_SUCCESS(Status)) 812 { 813 Result = TRUE; 814 } 815 else 816 { 817 BaseSetLastNTError(Status); 818 Result = FALSE; 819 } 820 return Result; 821 } 822 823 /* 824 * @implemented 825 */ 826 VOID 827 WINAPI 828 SetFileApisToOEM(VOID) 829 { 830 /* Set the correct Base Api */ 831 Basep8BitStringToUnicodeString = (PRTL_CONVERT_STRING)RtlOemStringToUnicodeString; 832 BasepUnicodeStringTo8BitString = RtlUnicodeStringToOemString; 833 BasepUnicodeStringTo8BitSize = BasepUnicodeStringToOemSize; 834 Basep8BitStringToUnicodeSize = BasepOemStringToUnicodeSize; 835 836 /* FIXME: Old, deprecated way */ 837 bIsFileApiAnsi = FALSE; 838 } 839 840 841 /* 842 * @implemented 843 */ 844 VOID 845 WINAPI 846 SetFileApisToANSI(VOID) 847 { 848 /* Set the correct Base Api */ 849 Basep8BitStringToUnicodeString = RtlAnsiStringToUnicodeString; 850 BasepUnicodeStringTo8BitString = RtlUnicodeStringToAnsiString; 851 BasepUnicodeStringTo8BitSize = BasepUnicodeStringToAnsiSize; 852 Basep8BitStringToUnicodeSize = BasepAnsiStringToUnicodeSize; 853 854 /* FIXME: Old, deprecated way */ 855 bIsFileApiAnsi = TRUE; 856 } 857 858 /* 859 * @implemented 860 */ 861 BOOL 862 WINAPI 863 AreFileApisANSI(VOID) 864 { 865 return Basep8BitStringToUnicodeString == RtlAnsiStringToUnicodeString; 866 } 867 868 /* 869 * @implemented 870 */ 871 VOID 872 WINAPI 873 BaseMarkFileForDelete(IN HANDLE FileHandle, 874 IN ULONG FileAttributes) 875 { 876 IO_STATUS_BLOCK IoStatusBlock; 877 FILE_BASIC_INFORMATION FileBasicInfo; 878 FILE_DISPOSITION_INFORMATION FileDispositionInfo; 879 880 /* If no attributes were given, get them */ 881 if (!FileAttributes) 882 { 883 FileBasicInfo.FileAttributes = 0; 884 NtQueryInformationFile(FileHandle, 885 &IoStatusBlock, 886 &FileBasicInfo, 887 sizeof(FileBasicInfo), 888 FileBasicInformation); 889 FileAttributes = FileBasicInfo.FileAttributes; 890 } 891 892 /* If file is marked as RO, reset its attributes */ 893 if (FileAttributes & FILE_ATTRIBUTE_READONLY) 894 { 895 RtlZeroMemory(&FileBasicInfo, sizeof(FileBasicInfo)); 896 FileBasicInfo.FileAttributes = FILE_ATTRIBUTE_NORMAL; 897 NtSetInformationFile(FileHandle, 898 &IoStatusBlock, 899 &FileBasicInfo, 900 sizeof(FileBasicInfo), 901 FileBasicInformation); 902 } 903 904 /* Finally, mark the file for deletion */ 905 FileDispositionInfo.DeleteFile = TRUE; 906 NtSetInformationFile(FileHandle, 907 &IoStatusBlock, 908 &FileDispositionInfo, 909 sizeof(FileDispositionInfo), 910 FileDispositionInformation); 911 } 912 913 /* 914 * @unimplemented 915 */ 916 NTSTATUS 917 WINAPI 918 BasepCheckWinSaferRestrictions(IN HANDLE UserToken, 919 IN LPWSTR ApplicationName, 920 IN HANDLE FileHandle, 921 OUT PBOOLEAN InJob, 922 OUT PHANDLE NewToken, 923 OUT PHANDLE JobHandle) 924 { 925 NTSTATUS Status; 926 927 /* Validate that there's a name */ 928 if ((ApplicationName) && *(ApplicationName)) 929 { 930 /* Validate that the required output parameters are there */ 931 if ((InJob) && (NewToken) && (JobHandle)) 932 { 933 /* Do the work (one day...) */ 934 DPRINT("BasepCheckWinSaferRestrictions is UNIMPLEMENTED\n"); 935 Status = STATUS_SUCCESS; 936 } 937 else 938 { 939 /* Act as if SEH hit this */ 940 Status = STATUS_ACCESS_VIOLATION; 941 } 942 } 943 else 944 { 945 /* Input is invalid */ 946 Status = STATUS_INVALID_PARAMETER; 947 } 948 949 /* Return the status */ 950 return Status; 951 } 952