xref: /reactos/dll/ntdll/ldr/ldrinit.c (revision 5140a990)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         ReactOS NT User-Mode Library
4  * FILE:            dll/ntdll/ldr/ldrinit.c
5  * PURPOSE:         User-Mode Process/Thread Startup
6  * PROGRAMMERS:     Alex Ionescu (alex@relsoft.net)
7  *                  Aleksey Bragin (aleksey@reactos.org)
8  */
9 
10 /* INCLUDES *****************************************************************/
11 
12 #include <ntdll.h>
13 #include <compat_undoc.h>
14 #include <compatguid_undoc.h>
15 
16 #define NDEBUG
17 #include <debug.h>
18 
19 
20 /* GLOBALS *******************************************************************/
21 
22 HANDLE ImageExecOptionsKey;
23 HANDLE Wow64ExecOptionsKey;
24 UNICODE_STRING ImageExecOptionsString = RTL_CONSTANT_STRING(L"\\Registry\\Machine\\Software\\Microsoft\\Windows NT\\CurrentVersion\\Image File Execution Options");
25 UNICODE_STRING Wow64OptionsString = RTL_CONSTANT_STRING(L"");
26 UNICODE_STRING NtDllString = RTL_CONSTANT_STRING(L"ntdll.dll");
27 UNICODE_STRING Kernel32String = RTL_CONSTANT_STRING(L"kernel32.dll");
28 const UNICODE_STRING LdrpDotLocal = RTL_CONSTANT_STRING(L".Local");
29 
30 BOOLEAN LdrpInLdrInit;
31 LONG LdrpProcessInitialized;
32 BOOLEAN LdrpLoaderLockInit;
33 BOOLEAN LdrpLdrDatabaseIsSetup;
34 BOOLEAN LdrpShutdownInProgress;
35 HANDLE LdrpShutdownThreadId;
36 
37 BOOLEAN LdrpDllValidation;
38 
39 PLDR_DATA_TABLE_ENTRY LdrpImageEntry;
40 PUNICODE_STRING LdrpTopLevelDllBeingLoaded;
41 WCHAR StringBuffer[156];
42 extern PTEB LdrpTopLevelDllBeingLoadedTeb; // defined in rtlsupp.c!
43 PLDR_DATA_TABLE_ENTRY LdrpCurrentDllInitializer;
44 PLDR_DATA_TABLE_ENTRY LdrpNtDllDataTableEntry;
45 
46 static NTSTATUS (WINAPI *Kernel32ProcessInitPostImportFunction)(VOID);
47 static BOOL (WINAPI *Kernel32BaseQueryModuleData)(IN LPSTR ModuleName, IN LPSTR Unk1, IN PVOID Unk2, IN PVOID Unk3, IN PVOID Unk4);
48 
49 RTL_BITMAP TlsBitMap;
50 RTL_BITMAP TlsExpansionBitMap;
51 RTL_BITMAP FlsBitMap;
52 BOOLEAN LdrpImageHasTls;
53 LIST_ENTRY LdrpTlsList;
54 ULONG LdrpNumberOfTlsEntries;
55 ULONG LdrpNumberOfProcessors;
56 PVOID NtDllBase;
57 extern LARGE_INTEGER RtlpTimeout;
58 extern BOOLEAN RtlpTimeoutDisable;
59 PVOID LdrpHeap;
60 LIST_ENTRY LdrpHashTable[LDR_HASH_TABLE_ENTRIES];
61 LIST_ENTRY LdrpDllNotificationList;
62 HANDLE LdrpKnownDllObjectDirectory;
63 UNICODE_STRING LdrpKnownDllPath;
64 WCHAR LdrpKnownDllPathBuffer[128];
65 UNICODE_STRING LdrpDefaultPath;
66 
67 PEB_LDR_DATA PebLdr;
68 
69 RTL_CRITICAL_SECTION_DEBUG LdrpLoaderLockDebug;
70 RTL_CRITICAL_SECTION LdrpLoaderLock =
71 {
72     &LdrpLoaderLockDebug,
73     -1,
74     0,
75     0,
76     0,
77     0
78 };
79 RTL_CRITICAL_SECTION FastPebLock;
80 
81 BOOLEAN ShowSnaps;
82 
83 ULONG LdrpFatalHardErrorCount;
84 ULONG LdrpActiveUnloadCount;
85 
86 //extern LIST_ENTRY RtlCriticalSectionList;
87 
88 VOID NTAPI RtlpInitializeVectoredExceptionHandling(VOID);
89 VOID NTAPI RtlpInitDeferredCriticalSection(VOID);
90 VOID NTAPI RtlInitializeHeapManager(VOID);
91 
92 ULONG RtlpDisableHeapLookaside; // TODO: Move to heap.c
93 ULONG RtlpShutdownProcessFlags; // TODO: Use it
94 
95 NTSTATUS LdrPerformRelocations(PIMAGE_NT_HEADERS NTHeaders, PVOID ImageBase);
96 void actctx_init(PVOID* pOldShimData);
97 extern BOOLEAN RtlpUse16ByteSLists;
98 
99 #ifdef _WIN64
100 #define DEFAULT_SECURITY_COOKIE 0x00002B992DDFA232ll
101 #else
102 #define DEFAULT_SECURITY_COOKIE 0xBB40E64E
103 #endif
104 
105 /* FUNCTIONS *****************************************************************/
106 
107 /*
108  * @implemented
109  */
110 NTSTATUS
111 NTAPI
112 LdrOpenImageFileOptionsKey(
113     _In_ PUNICODE_STRING SubKey,
114     _In_ BOOLEAN Wow64,
115     _Out_ PHANDLE NewKeyHandle)
116 {
117     PHANDLE RootKeyLocation;
118     HANDLE RootKey;
119     UNICODE_STRING SubKeyString;
120     OBJECT_ATTRIBUTES ObjectAttributes;
121     NTSTATUS Status;
122     PWCHAR p1;
123 
124     /* Check which root key to open */
125     if (Wow64)
126         RootKeyLocation = &Wow64ExecOptionsKey;
127     else
128         RootKeyLocation = &ImageExecOptionsKey;
129 
130     /* Get the current key */
131     RootKey = *RootKeyLocation;
132 
133     /* Setup the object attributes */
134     InitializeObjectAttributes(&ObjectAttributes,
135                                Wow64 ?
136                                &Wow64OptionsString : &ImageExecOptionsString,
137                                OBJ_CASE_INSENSITIVE,
138                                NULL,
139                                NULL);
140 
141     /* Open the root key */
142     Status = ZwOpenKey(&RootKey, KEY_ENUMERATE_SUB_KEYS, &ObjectAttributes);
143     if (NT_SUCCESS(Status))
144     {
145         /* Write the key handle */
146         if (InterlockedCompareExchangePointer(RootKeyLocation, RootKey, NULL) != NULL)
147         {
148             /* Someone already opened it, use it instead */
149             NtClose(RootKey);
150             RootKey = *RootKeyLocation;
151         }
152 
153         /* Extract the name */
154         SubKeyString = *SubKey;
155         p1 = (PWCHAR)((ULONG_PTR)SubKeyString.Buffer + SubKeyString.Length);
156         while (SubKeyString.Length)
157         {
158             if (p1[-1] == L'\\') break;
159             p1--;
160             SubKeyString.Length -= sizeof(*p1);
161         }
162         SubKeyString.Buffer = p1;
163         SubKeyString.Length = SubKey->Length - SubKeyString.Length;
164 
165         /* Setup the object attributes */
166         InitializeObjectAttributes(&ObjectAttributes,
167                                    &SubKeyString,
168                                    OBJ_CASE_INSENSITIVE,
169                                    RootKey,
170                                    NULL);
171 
172         /* Open the setting key */
173         Status = ZwOpenKey((PHANDLE)NewKeyHandle, GENERIC_READ, &ObjectAttributes);
174     }
175 
176     /* Return to caller */
177     return Status;
178 }
179 
180 /*
181  * @implemented
182  */
183 NTSTATUS
184 NTAPI
185 LdrQueryImageFileKeyOption(
186     _In_ HANDLE KeyHandle,
187     _In_ PCWSTR ValueName,
188     _In_ ULONG Type,
189     _Out_ PVOID Buffer,
190     _In_ ULONG BufferSize,
191     _Out_opt_ PULONG ReturnedLength)
192 {
193     ULONG KeyInfo[256];
194     UNICODE_STRING ValueNameString, IntegerString;
195     ULONG KeyInfoSize, ResultSize;
196     PKEY_VALUE_PARTIAL_INFORMATION KeyValueInformation = (PKEY_VALUE_PARTIAL_INFORMATION)&KeyInfo;
197     BOOLEAN FreeHeap = FALSE;
198     NTSTATUS Status;
199 
200     /* Build a string for the value name */
201     Status = RtlInitUnicodeStringEx(&ValueNameString, ValueName);
202     if (!NT_SUCCESS(Status)) return Status;
203 
204     /* Query the value */
205     Status = ZwQueryValueKey(KeyHandle,
206                              &ValueNameString,
207                              KeyValuePartialInformation,
208                              KeyValueInformation,
209                              sizeof(KeyInfo),
210                              &ResultSize);
211     if (Status == STATUS_BUFFER_OVERFLOW)
212     {
213         /* Our local buffer wasn't enough, allocate one */
214         KeyInfoSize = sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
215                       KeyValueInformation->DataLength;
216         KeyValueInformation = RtlAllocateHeap(RtlGetProcessHeap(),
217                                               0,
218                                               KeyInfoSize);
219         if (KeyValueInformation != NULL)
220         {
221             /* Try again */
222             Status = ZwQueryValueKey(KeyHandle,
223                                      &ValueNameString,
224                                      KeyValuePartialInformation,
225                                      KeyValueInformation,
226                                      KeyInfoSize,
227                                      &ResultSize);
228             FreeHeap = TRUE;
229         }
230         else
231         {
232             /* Give up this time */
233             Status = STATUS_NO_MEMORY;
234         }
235     }
236 
237     /* Check for success */
238     if (NT_SUCCESS(Status))
239     {
240         /* Handle binary data */
241         if (KeyValueInformation->Type == REG_BINARY)
242         {
243             /* Check validity */
244             if ((Buffer) && (KeyValueInformation->DataLength <= BufferSize))
245             {
246                 /* Copy into buffer */
247                 RtlMoveMemory(Buffer,
248                               &KeyValueInformation->Data,
249                               KeyValueInformation->DataLength);
250             }
251             else
252             {
253                 Status = STATUS_BUFFER_OVERFLOW;
254             }
255 
256             /* Copy the result length */
257             if (ReturnedLength) *ReturnedLength = KeyValueInformation->DataLength;
258         }
259         else if (KeyValueInformation->Type == REG_DWORD)
260         {
261             /* Check for valid type */
262             if (KeyValueInformation->Type != Type)
263             {
264                 /* Error */
265                 Status = STATUS_OBJECT_TYPE_MISMATCH;
266             }
267             else
268             {
269                 /* Check validity */
270                 if ((Buffer) &&
271                     (BufferSize == sizeof(ULONG)) &&
272                     (KeyValueInformation->DataLength <= BufferSize))
273                 {
274                     /* Copy into buffer */
275                     RtlMoveMemory(Buffer,
276                                   &KeyValueInformation->Data,
277                                   KeyValueInformation->DataLength);
278                 }
279                 else
280                 {
281                     Status = STATUS_BUFFER_OVERFLOW;
282                 }
283 
284                 /* Copy the result length */
285                 if (ReturnedLength) *ReturnedLength = KeyValueInformation->DataLength;
286             }
287         }
288         else if (KeyValueInformation->Type != REG_SZ)
289         {
290             /* We got something weird */
291             Status = STATUS_OBJECT_TYPE_MISMATCH;
292         }
293         else
294         {
295             /*  String, check what you requested */
296             if (Type == REG_DWORD)
297             {
298                 /* Validate */
299                 if (BufferSize != sizeof(ULONG))
300                 {
301                     /* Invalid size */
302                     BufferSize = 0;
303                     Status = STATUS_INFO_LENGTH_MISMATCH;
304                 }
305                 else
306                 {
307                     /* OK, we know what you want... */
308                     IntegerString.Buffer = (PWSTR)KeyValueInformation->Data;
309                     IntegerString.Length = (USHORT)KeyValueInformation->DataLength -
310                                            sizeof(WCHAR);
311                     IntegerString.MaximumLength = (USHORT)KeyValueInformation->DataLength;
312                     Status = RtlUnicodeStringToInteger(&IntegerString, 0, (PULONG)Buffer);
313                 }
314             }
315             else
316             {
317                 /* Validate */
318                 if (KeyValueInformation->DataLength > BufferSize)
319                 {
320                     /* Invalid */
321                     Status = STATUS_BUFFER_OVERFLOW;
322                 }
323                 else
324                 {
325                     /* Set the size */
326                     BufferSize = KeyValueInformation->DataLength;
327                 }
328 
329                 /* Copy the string */
330                 RtlMoveMemory(Buffer, &KeyValueInformation->Data, BufferSize);
331             }
332 
333             /* Copy the result length */
334             if (ReturnedLength) *ReturnedLength = KeyValueInformation->DataLength;
335         }
336     }
337 
338     /* Check if buffer was in heap */
339     if (FreeHeap) RtlFreeHeap(RtlGetProcessHeap(), 0, KeyValueInformation);
340 
341     /* Return status */
342     return Status;
343 }
344 
345 /*
346  * @implemented
347  */
348 NTSTATUS
349 NTAPI
350 LdrQueryImageFileExecutionOptionsEx(
351     _In_ PUNICODE_STRING SubKey,
352     _In_ PCWSTR ValueName,
353     _In_ ULONG Type,
354     _Out_ PVOID Buffer,
355     _In_ ULONG BufferSize,
356     _Out_opt_ PULONG ReturnedLength,
357     _In_ BOOLEAN Wow64)
358 {
359     NTSTATUS Status;
360     HANDLE KeyHandle;
361 
362     /* Open a handle to the key */
363     Status = LdrOpenImageFileOptionsKey(SubKey, Wow64, &KeyHandle);
364 
365     /* Check for success */
366     if (NT_SUCCESS(Status))
367     {
368         /* Query the data */
369         Status = LdrQueryImageFileKeyOption(KeyHandle,
370                                             ValueName,
371                                             Type,
372                                             Buffer,
373                                             BufferSize,
374                                             ReturnedLength);
375 
376         /* Close the key */
377         NtClose(KeyHandle);
378     }
379 
380     /* Return to caller */
381     return Status;
382 }
383 
384 /*
385  * @implemented
386  */
387 NTSTATUS
388 NTAPI
389 LdrQueryImageFileExecutionOptions(
390     _In_ PUNICODE_STRING SubKey,
391     _In_ PCWSTR ValueName,
392     _In_ ULONG Type,
393     _Out_ PVOID Buffer,
394     _In_ ULONG BufferSize,
395     _Out_opt_ PULONG ReturnedLength)
396 {
397     /* Call the newer function */
398     return LdrQueryImageFileExecutionOptionsEx(SubKey,
399                                                ValueName,
400                                                Type,
401                                                Buffer,
402                                                BufferSize,
403                                                ReturnedLength,
404                                                FALSE);
405 }
406 
407 VOID
408 NTAPI
409 LdrpEnsureLoaderLockIsHeld(VOID)
410 {
411     // Ignored atm
412 }
413 
414 PVOID
415 NTAPI
416 LdrpFetchAddressOfSecurityCookie(PVOID BaseAddress, ULONG SizeOfImage)
417 {
418     PIMAGE_LOAD_CONFIG_DIRECTORY ConfigDir;
419     ULONG DirSize;
420     PVOID Cookie = NULL;
421 
422     /* Check NT header first */
423     if (!RtlImageNtHeader(BaseAddress)) return NULL;
424 
425     /* Get the pointer to the config directory */
426     ConfigDir = RtlImageDirectoryEntryToData(BaseAddress,
427                                              TRUE,
428                                              IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG,
429                                              &DirSize);
430 
431     /* Check for sanity */
432     if (!ConfigDir ||
433         (DirSize != 64 && ConfigDir->Size != DirSize) ||
434         (ConfigDir->Size < 0x48))
435         return NULL;
436 
437     /* Now get the cookie */
438     Cookie = (PVOID)ConfigDir->SecurityCookie;
439 
440     /* Check this cookie */
441     if ((PCHAR)Cookie <= (PCHAR)BaseAddress ||
442         (PCHAR)Cookie >= (PCHAR)BaseAddress + SizeOfImage)
443     {
444         Cookie = NULL;
445     }
446 
447     /* Return validated security cookie */
448     return Cookie;
449 }
450 
451 PVOID
452 NTAPI
453 LdrpInitSecurityCookie(PLDR_DATA_TABLE_ENTRY LdrEntry)
454 {
455     PULONG_PTR Cookie;
456     LARGE_INTEGER Counter;
457     ULONG_PTR NewCookie;
458 
459     /* Fetch address of the cookie */
460     Cookie = LdrpFetchAddressOfSecurityCookie(LdrEntry->DllBase, LdrEntry->SizeOfImage);
461 
462     if (Cookie)
463     {
464         /* Check if it's a default one */
465         if ((*Cookie == DEFAULT_SECURITY_COOKIE) ||
466             (*Cookie == 0xBB40))
467         {
468             /* Make up a cookie from a bunch of values which may uniquely represent
469                current moment of time, environment, etc */
470             NtQueryPerformanceCounter(&Counter, NULL);
471 
472             NewCookie = Counter.LowPart ^ Counter.HighPart;
473             NewCookie ^= (ULONG_PTR)NtCurrentTeb()->ClientId.UniqueProcess;
474             NewCookie ^= (ULONG_PTR)NtCurrentTeb()->ClientId.UniqueThread;
475 
476             /* Loop like it's done in KeQueryTickCount(). We don't want to call it directly. */
477             while (SharedUserData->SystemTime.High1Time != SharedUserData->SystemTime.High2Time)
478             {
479                 YieldProcessor();
480             };
481 
482             /* Calculate the milliseconds value and xor it to the cookie */
483             NewCookie ^= Int64ShrlMod32(UInt32x32To64(SharedUserData->TickCountMultiplier, SharedUserData->TickCount.LowPart), 24) +
484                 (SharedUserData->TickCountMultiplier * (SharedUserData->TickCount.High1Time << 8));
485 
486             /* Make the cookie 16bit if necessary */
487             if (*Cookie == 0xBB40) NewCookie &= 0xFFFF;
488 
489             /* If the result is 0 or the same as we got, just subtract one from the existing value
490                and that's it */
491             if ((NewCookie == 0) || (NewCookie == *Cookie))
492             {
493                 NewCookie = *Cookie - 1;
494             }
495 
496             /* Set the new cookie value */
497             *Cookie = NewCookie;
498         }
499     }
500 
501     return Cookie;
502 }
503 
504 VOID
505 NTAPI
506 LdrpInitializeThread(IN PCONTEXT Context)
507 {
508     PPEB Peb = NtCurrentPeb();
509     PLDR_DATA_TABLE_ENTRY LdrEntry;
510     PLIST_ENTRY NextEntry, ListHead;
511     RTL_CALLER_ALLOCATED_ACTIVATION_CONTEXT_STACK_FRAME_EXTENDED ActCtx;
512     NTSTATUS Status;
513     PVOID EntryPoint;
514 
515     DPRINT("LdrpInitializeThread() called for %wZ (%p/%p)\n",
516             &LdrpImageEntry->BaseDllName,
517             NtCurrentTeb()->RealClientId.UniqueProcess,
518             NtCurrentTeb()->RealClientId.UniqueThread);
519 
520     /* Allocate an Activation Context Stack */
521     DPRINT("ActivationContextStack %p\n", NtCurrentTeb()->ActivationContextStackPointer);
522     Status = RtlAllocateActivationContextStack(&NtCurrentTeb()->ActivationContextStackPointer);
523     if (!NT_SUCCESS(Status))
524     {
525         DPRINT1("Warning: Unable to allocate ActivationContextStack\n");
526     }
527 
528     /* Make sure we are not shutting down */
529     if (LdrpShutdownInProgress) return;
530 
531     /* Allocate TLS */
532     LdrpAllocateTls();
533 
534     /* Start at the beginning */
535     ListHead = &Peb->Ldr->InMemoryOrderModuleList;
536     NextEntry = ListHead->Flink;
537     while (NextEntry != ListHead)
538     {
539         /* Get the current entry */
540         LdrEntry = CONTAINING_RECORD(NextEntry, LDR_DATA_TABLE_ENTRY, InMemoryOrderLinks);
541 
542         /* Make sure it's not ourselves */
543         if (Peb->ImageBaseAddress != LdrEntry->DllBase)
544         {
545             /* Check if we should call */
546             if (!(LdrEntry->Flags & LDRP_DONT_CALL_FOR_THREADS))
547             {
548                 /* Get the entrypoint */
549                 EntryPoint = LdrEntry->EntryPoint;
550 
551                 /* Check if we are ready to call it */
552                 if ((EntryPoint) &&
553                     (LdrEntry->Flags & LDRP_PROCESS_ATTACH_CALLED) &&
554                     (LdrEntry->Flags & LDRP_IMAGE_DLL))
555                 {
556                     /* Set up the Act Ctx */
557                     ActCtx.Size = sizeof(ActCtx);
558                     ActCtx.Format = 1;
559                     RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
560 
561                     /* Activate the ActCtx */
562                     RtlActivateActivationContextUnsafeFast(&ActCtx,
563                                                            LdrEntry->EntryPointActivationContext);
564 
565                     _SEH2_TRY
566                     {
567                         /* Check if it has TLS */
568                         if (LdrEntry->TlsIndex)
569                         {
570                             /* Make sure we're not shutting down */
571                             if (!LdrpShutdownInProgress)
572                             {
573                                 /* Call TLS */
574                                 LdrpCallTlsInitializers(LdrEntry, DLL_THREAD_ATTACH);
575                             }
576                         }
577 
578                         /* Make sure we're not shutting down */
579                         if (!LdrpShutdownInProgress)
580                         {
581                             /* Call the Entrypoint */
582                             DPRINT("%wZ - Calling entry point at %p for thread attaching, %p/%p\n",
583                                    &LdrEntry->BaseDllName, LdrEntry->EntryPoint,
584                                    NtCurrentTeb()->RealClientId.UniqueProcess,
585                                    NtCurrentTeb()->RealClientId.UniqueThread);
586                             LdrpCallInitRoutine(LdrEntry->EntryPoint,
587                                                 LdrEntry->DllBase,
588                                                 DLL_THREAD_ATTACH,
589                                                 NULL);
590                         }
591                     }
592                     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
593                     {
594                         DPRINT1("WARNING: Exception 0x%x during LdrpCallInitRoutine(DLL_THREAD_ATTACH) for %wZ\n",
595                                 _SEH2_GetExceptionCode(), &LdrEntry->BaseDllName);
596                     }
597                     _SEH2_END;
598 
599                     /* Deactivate the ActCtx */
600                     RtlDeactivateActivationContextUnsafeFast(&ActCtx);
601                 }
602             }
603         }
604 
605         /* Next entry */
606         NextEntry = NextEntry->Flink;
607     }
608 
609     /* Check for TLS */
610     if (LdrpImageHasTls && !LdrpShutdownInProgress)
611     {
612         /* Set up the Act Ctx */
613         ActCtx.Size = sizeof(ActCtx);
614         ActCtx.Format = 1;
615         RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
616 
617         /* Activate the ActCtx */
618         RtlActivateActivationContextUnsafeFast(&ActCtx,
619                                                LdrpImageEntry->EntryPointActivationContext);
620 
621         _SEH2_TRY
622         {
623             /* Do TLS callbacks */
624             LdrpCallTlsInitializers(LdrpImageEntry, DLL_THREAD_ATTACH);
625         }
626         _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
627         {
628             /* Do nothing */
629         }
630         _SEH2_END;
631 
632         /* Deactivate the ActCtx */
633         RtlDeactivateActivationContextUnsafeFast(&ActCtx);
634     }
635 
636     DPRINT("LdrpInitializeThread() done\n");
637 }
638 
639 NTSTATUS
640 NTAPI
641 LdrpRunInitializeRoutines(IN PCONTEXT Context OPTIONAL)
642 {
643     PLDR_DATA_TABLE_ENTRY LocalArray[16];
644     PLIST_ENTRY ListHead;
645     PLIST_ENTRY NextEntry;
646     PLDR_DATA_TABLE_ENTRY LdrEntry, *LdrRootEntry, OldInitializer;
647     PVOID EntryPoint;
648     ULONG Count, i;
649     //ULONG BreakOnInit;
650     NTSTATUS Status = STATUS_SUCCESS;
651     PPEB Peb = NtCurrentPeb();
652     RTL_CALLER_ALLOCATED_ACTIVATION_CONTEXT_STACK_FRAME_EXTENDED ActCtx;
653     ULONG BreakOnDllLoad;
654     PTEB OldTldTeb;
655     BOOLEAN DllStatus;
656 
657     DPRINT("LdrpRunInitializeRoutines() called for %wZ (%p/%p)\n",
658         &LdrpImageEntry->BaseDllName,
659         NtCurrentTeb()->RealClientId.UniqueProcess,
660         NtCurrentTeb()->RealClientId.UniqueThread);
661 
662     /* Check the Loader Lock */
663     LdrpEnsureLoaderLockIsHeld();
664 
665      /* Get the number of entries to call */
666     if ((Count = LdrpClearLoadInProgress()))
667     {
668         /* Check if we can use our local buffer */
669         if (Count > 16)
670         {
671             /* Allocate space for all the entries */
672             LdrRootEntry = RtlAllocateHeap(LdrpHeap,
673                                            0,
674                                            Count * sizeof(*LdrRootEntry));
675             if (!LdrRootEntry) return STATUS_NO_MEMORY;
676         }
677         else
678         {
679             /* Use our local array */
680             LdrRootEntry = LocalArray;
681         }
682     }
683     else
684     {
685         /* Don't need one */
686         LdrRootEntry = NULL;
687     }
688 
689     /* Show debug message */
690     if (ShowSnaps)
691     {
692         DPRINT1("[%p,%p] LDR: Real INIT LIST for Process %wZ\n",
693                 NtCurrentTeb()->RealClientId.UniqueThread,
694                 NtCurrentTeb()->RealClientId.UniqueProcess,
695                 &Peb->ProcessParameters->ImagePathName);
696     }
697 
698     /* Loop in order */
699     ListHead = &Peb->Ldr->InInitializationOrderModuleList;
700     NextEntry = ListHead->Flink;
701     i = 0;
702     while (NextEntry != ListHead)
703     {
704         /* Get the Data Entry */
705         LdrEntry = CONTAINING_RECORD(NextEntry, LDR_DATA_TABLE_ENTRY, InInitializationOrderLinks);
706 
707         /* Check if we have a Root Entry */
708         if (LdrRootEntry)
709         {
710             /* Check flags */
711             if (!(LdrEntry->Flags & LDRP_ENTRY_PROCESSED))
712             {
713                 /* Setup the Cookie for the DLL */
714                 LdrpInitSecurityCookie(LdrEntry);
715 
716                 /* Check for valid entrypoint */
717                 if (LdrEntry->EntryPoint)
718                 {
719                     /* Write in array */
720                     ASSERT(i < Count);
721                     LdrRootEntry[i] = LdrEntry;
722 
723                     /* Display debug message */
724                     if (ShowSnaps)
725                     {
726                         DPRINT1("[%p,%p] LDR: %wZ init routine %p\n",
727                                 NtCurrentTeb()->RealClientId.UniqueThread,
728                                 NtCurrentTeb()->RealClientId.UniqueProcess,
729                                 &LdrEntry->FullDllName,
730                                 LdrEntry->EntryPoint);
731                     }
732                     i++;
733                 }
734             }
735         }
736 
737         /* Set the flag */
738         LdrEntry->Flags |= LDRP_ENTRY_PROCESSED;
739         NextEntry = NextEntry->Flink;
740     }
741 
742     Status = STATUS_SUCCESS;
743 
744     /* If we got a context, then we have to call Kernel32 for TS support */
745     if (Context)
746     {
747         /* Check if we have one */
748         if (Kernel32ProcessInitPostImportFunction)
749         {
750             /* Call it */
751             Status = Kernel32ProcessInitPostImportFunction();
752             if (!NT_SUCCESS(Status))
753             {
754                 DPRINT1("LDR: LdrpRunInitializeRoutines - Failed running kernel32 post-import function, Status=0x%08lx\n", Status);
755             }
756         }
757         /* Clear it */
758         Kernel32ProcessInitPostImportFunction = NULL;
759     }
760 
761     /* No root entry? return */
762     if (!LdrRootEntry)
763         return Status;
764 
765     /* Set the TLD TEB */
766     OldTldTeb = LdrpTopLevelDllBeingLoadedTeb;
767     LdrpTopLevelDllBeingLoadedTeb = NtCurrentTeb();
768 
769     /* Loop */
770     i = 0;
771     while (i < Count)
772     {
773         /* Get an entry */
774         LdrEntry = LdrRootEntry[i];
775 
776         /* FIXME: Verify NX Compat */
777 
778         /* Move to next entry */
779         i++;
780 
781         /* Get its entrypoint */
782         EntryPoint = LdrEntry->EntryPoint;
783 
784         /* Are we being debugged? */
785         BreakOnDllLoad = 0;
786         if (Peb->BeingDebugged || Peb->ReadImageFileExecOptions)
787         {
788             /* Check if we should break on load */
789             Status = LdrQueryImageFileExecutionOptions(&LdrEntry->BaseDllName,
790                                                        L"BreakOnDllLoad",
791                                                        REG_DWORD,
792                                                        &BreakOnDllLoad,
793                                                        sizeof(ULONG),
794                                                        NULL);
795             if (!NT_SUCCESS(Status)) BreakOnDllLoad = 0;
796 
797             /* Reset status back to STATUS_SUCCESS */
798             Status = STATUS_SUCCESS;
799         }
800 
801         /* Break if aksed */
802         if (BreakOnDllLoad)
803         {
804             /* Check if we should show a message */
805             if (ShowSnaps)
806             {
807                 DPRINT1("LDR: %wZ loaded.", &LdrEntry->BaseDllName);
808                 DPRINT1(" - About to call init routine at %p\n", EntryPoint);
809             }
810 
811             /* Break in debugger */
812             DbgBreakPoint();
813         }
814 
815         /* Make sure we have an entrypoint */
816         if (EntryPoint)
817         {
818             /* Save the old Dll Initializer and write the current one */
819             OldInitializer = LdrpCurrentDllInitializer;
820             LdrpCurrentDllInitializer = LdrEntry;
821 
822             /* Set up the Act Ctx */
823             ActCtx.Size = sizeof(ActCtx);
824             ActCtx.Format = 1;
825             RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
826 
827             /* Activate the ActCtx */
828             RtlActivateActivationContextUnsafeFast(&ActCtx,
829                                                    LdrEntry->EntryPointActivationContext);
830 
831             _SEH2_TRY
832             {
833                 /* Check if it has TLS */
834                 if (LdrEntry->TlsIndex && Context)
835                 {
836                     /* Call TLS */
837                     LdrpCallTlsInitializers(LdrEntry, DLL_PROCESS_ATTACH);
838                 }
839 
840                 /* Call the Entrypoint */
841                 if (ShowSnaps)
842                 {
843                     DPRINT1("%wZ - Calling entry point at %p for DLL_PROCESS_ATTACH\n",
844                             &LdrEntry->BaseDllName, EntryPoint);
845                 }
846                 DllStatus = LdrpCallInitRoutine(EntryPoint,
847                                                 LdrEntry->DllBase,
848                                                 DLL_PROCESS_ATTACH,
849                                                 Context);
850             }
851             _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
852             {
853                 DllStatus = FALSE;
854                 DPRINT1("WARNING: Exception 0x%x during LdrpCallInitRoutine(DLL_PROCESS_ATTACH) for %wZ\n",
855                         _SEH2_GetExceptionCode(), &LdrEntry->BaseDllName);
856             }
857             _SEH2_END;
858 
859             /* Deactivate the ActCtx */
860             RtlDeactivateActivationContextUnsafeFast(&ActCtx);
861 
862             /* Save the Current DLL Initializer */
863             LdrpCurrentDllInitializer = OldInitializer;
864 
865             /* Mark the entry as processed */
866             LdrEntry->Flags |= LDRP_PROCESS_ATTACH_CALLED;
867 
868             /* Fail if DLL init failed */
869             if (!DllStatus)
870             {
871                 DPRINT1("LDR: DLL_PROCESS_ATTACH for dll \"%wZ\" (InitRoutine: %p) failed\n",
872                     &LdrEntry->BaseDllName, EntryPoint);
873 
874                 Status = STATUS_DLL_INIT_FAILED;
875                 goto Quickie;
876             }
877         }
878     }
879 
880     /* Loop in order */
881     ListHead = &Peb->Ldr->InInitializationOrderModuleList;
882     NextEntry = NextEntry->Flink;
883     while (NextEntry != ListHead)
884     {
885         /* Get the Data Entry */
886         LdrEntry = CONTAINING_RECORD(NextEntry, LDR_DATA_TABLE_ENTRY, InInitializationOrderLinks);
887 
888         /* FIXME: Verify NX Compat */
889         // LdrpCheckNXCompatibility()
890 
891         /* Next entry */
892         NextEntry = NextEntry->Flink;
893     }
894 
895     /* Check for TLS */
896     if (LdrpImageHasTls && Context)
897     {
898         /* Set up the Act Ctx */
899         ActCtx.Size = sizeof(ActCtx);
900         ActCtx.Format = 1;
901         RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
902 
903         /* Activate the ActCtx */
904         RtlActivateActivationContextUnsafeFast(&ActCtx,
905                                                LdrpImageEntry->EntryPointActivationContext);
906 
907         _SEH2_TRY
908         {
909             /* Do TLS callbacks */
910             LdrpCallTlsInitializers(LdrpImageEntry, DLL_PROCESS_ATTACH);
911         }
912         _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
913         {
914             /* Do nothing */
915         }
916         _SEH2_END;
917 
918         /* Deactivate the ActCtx */
919         RtlDeactivateActivationContextUnsafeFast(&ActCtx);
920     }
921 
922 Quickie:
923     /* Restore old TEB */
924     LdrpTopLevelDllBeingLoadedTeb = OldTldTeb;
925 
926     /* Check if the array is in the heap */
927     if (LdrRootEntry != LocalArray)
928     {
929         /* Free the array */
930         RtlFreeHeap(LdrpHeap, 0, LdrRootEntry);
931     }
932 
933     /* Return to caller */
934     DPRINT("LdrpRunInitializeRoutines() done\n");
935     return Status;
936 }
937 
938 /*
939  * @implemented
940  */
941 NTSTATUS
942 NTAPI
943 LdrShutdownProcess(VOID)
944 {
945     PPEB Peb = NtCurrentPeb();
946     PLDR_DATA_TABLE_ENTRY LdrEntry;
947     PLIST_ENTRY NextEntry, ListHead;
948     RTL_CALLER_ALLOCATED_ACTIVATION_CONTEXT_STACK_FRAME_EXTENDED ActCtx;
949     PVOID EntryPoint;
950 
951     DPRINT("LdrShutdownProcess() called for %wZ\n", &LdrpImageEntry->BaseDllName);
952     if (LdrpShutdownInProgress) return STATUS_SUCCESS;
953 
954     /* Tell the Shim Engine */
955     if (g_ShimsEnabled)
956     {
957         VOID(NTAPI *SE_ProcessDying)();
958         SE_ProcessDying = RtlDecodeSystemPointer(g_pfnSE_ProcessDying);
959         SE_ProcessDying();
960     }
961 
962     /* Tell the world */
963     if (ShowSnaps)
964     {
965         DPRINT1("\n");
966     }
967 
968     /* Set the shutdown variables */
969     LdrpShutdownThreadId = NtCurrentTeb()->RealClientId.UniqueThread;
970     LdrpShutdownInProgress = TRUE;
971 
972     /* Enter the Loader Lock */
973     RtlEnterCriticalSection(&LdrpLoaderLock);
974 
975     /* Cleanup trace logging data (Etw) */
976     if (SharedUserData->TraceLogging)
977     {
978         /* FIXME */
979         DPRINT1("We don't support Etw yet.\n");
980     }
981 
982     /* Start at the end */
983     ListHead = &Peb->Ldr->InInitializationOrderModuleList;
984     NextEntry = ListHead->Blink;
985     while (NextEntry != ListHead)
986     {
987         /* Get the current entry */
988         LdrEntry = CONTAINING_RECORD(NextEntry, LDR_DATA_TABLE_ENTRY, InInitializationOrderLinks);
989         NextEntry = NextEntry->Blink;
990 
991         /* Make sure it's not ourselves */
992         if (Peb->ImageBaseAddress != LdrEntry->DllBase)
993         {
994             /* Get the entrypoint */
995             EntryPoint = LdrEntry->EntryPoint;
996 
997             /* Check if we are ready to call it */
998             if (EntryPoint &&
999                 (LdrEntry->Flags & LDRP_PROCESS_ATTACH_CALLED) &&
1000                 LdrEntry->Flags)
1001             {
1002                 /* Set up the Act Ctx */
1003                 ActCtx.Size = sizeof(ActCtx);
1004                 ActCtx.Format = 1;
1005                 RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
1006 
1007                 /* Activate the ActCtx */
1008                 RtlActivateActivationContextUnsafeFast(&ActCtx,
1009                                                        LdrEntry->EntryPointActivationContext);
1010 
1011                 _SEH2_TRY
1012                 {
1013                     /* Check if it has TLS */
1014                     if (LdrEntry->TlsIndex)
1015                     {
1016                         /* Call TLS */
1017                         LdrpCallTlsInitializers(LdrEntry, DLL_PROCESS_DETACH);
1018                     }
1019 
1020                     /* Call the Entrypoint */
1021                     DPRINT("%wZ - Calling entry point at %p for thread detaching\n",
1022                            &LdrEntry->BaseDllName, LdrEntry->EntryPoint);
1023                     LdrpCallInitRoutine(EntryPoint,
1024                                         LdrEntry->DllBase,
1025                                         DLL_PROCESS_DETACH,
1026                                         (PVOID)1);
1027                 }
1028                 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
1029                 {
1030                     DPRINT1("WARNING: Exception 0x%x during LdrpCallInitRoutine(DLL_PROCESS_DETACH) for %wZ\n",
1031                             _SEH2_GetExceptionCode(), &LdrEntry->BaseDllName);
1032                 }
1033                 _SEH2_END;
1034 
1035                 /* Deactivate the ActCtx */
1036                 RtlDeactivateActivationContextUnsafeFast(&ActCtx);
1037             }
1038         }
1039     }
1040 
1041     /* Check for TLS */
1042     if (LdrpImageHasTls)
1043     {
1044         /* Set up the Act Ctx */
1045         ActCtx.Size = sizeof(ActCtx);
1046         ActCtx.Format = 1;
1047         RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
1048 
1049         /* Activate the ActCtx */
1050         RtlActivateActivationContextUnsafeFast(&ActCtx,
1051                                                LdrpImageEntry->EntryPointActivationContext);
1052 
1053         _SEH2_TRY
1054         {
1055             /* Do TLS callbacks */
1056             LdrpCallTlsInitializers(LdrpImageEntry, DLL_PROCESS_DETACH);
1057         }
1058         _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
1059         {
1060             /* Do nothing */
1061         }
1062         _SEH2_END;
1063 
1064         /* Deactivate the ActCtx */
1065         RtlDeactivateActivationContextUnsafeFast(&ActCtx);
1066     }
1067 
1068     /* FIXME: Do Heap detection and Etw final shutdown */
1069 
1070     /* Release the lock */
1071     RtlLeaveCriticalSection(&LdrpLoaderLock);
1072     DPRINT("LdrpShutdownProcess() done\n");
1073 
1074     return STATUS_SUCCESS;
1075 }
1076 
1077 /*
1078  * @implemented
1079  */
1080 NTSTATUS
1081 NTAPI
1082 LdrShutdownThread(VOID)
1083 {
1084     PPEB Peb = NtCurrentPeb();
1085     PTEB Teb = NtCurrentTeb();
1086     PLDR_DATA_TABLE_ENTRY LdrEntry;
1087     PLIST_ENTRY NextEntry, ListHead;
1088     RTL_CALLER_ALLOCATED_ACTIVATION_CONTEXT_STACK_FRAME_EXTENDED ActCtx;
1089     PVOID EntryPoint;
1090 
1091     DPRINT("LdrShutdownThread() called for %wZ\n",
1092             &LdrpImageEntry->BaseDllName);
1093 
1094     /* Cleanup trace logging data (Etw) */
1095     if (SharedUserData->TraceLogging)
1096     {
1097         /* FIXME */
1098         DPRINT1("We don't support Etw yet.\n");
1099     }
1100 
1101     /* Get the Ldr Lock */
1102     RtlEnterCriticalSection(&LdrpLoaderLock);
1103 
1104     /* Start at the end */
1105     ListHead = &Peb->Ldr->InInitializationOrderModuleList;
1106     NextEntry = ListHead->Blink;
1107     while (NextEntry != ListHead)
1108     {
1109         /* Get the current entry */
1110         LdrEntry = CONTAINING_RECORD(NextEntry, LDR_DATA_TABLE_ENTRY, InInitializationOrderLinks);
1111         NextEntry = NextEntry->Blink;
1112 
1113         /* Make sure it's not ourselves */
1114         if (Peb->ImageBaseAddress != LdrEntry->DllBase)
1115         {
1116             /* Check if we should call */
1117             if (!(LdrEntry->Flags & LDRP_DONT_CALL_FOR_THREADS) &&
1118                 (LdrEntry->Flags & LDRP_PROCESS_ATTACH_CALLED) &&
1119                 (LdrEntry->Flags & LDRP_IMAGE_DLL))
1120             {
1121                 /* Get the entrypoint */
1122                 EntryPoint = LdrEntry->EntryPoint;
1123 
1124                 /* Check if we are ready to call it */
1125                 if (EntryPoint)
1126                 {
1127                     /* Set up the Act Ctx */
1128                     ActCtx.Size = sizeof(ActCtx);
1129                     ActCtx.Format = 1;
1130                     RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
1131 
1132                     /* Activate the ActCtx */
1133                     RtlActivateActivationContextUnsafeFast(&ActCtx,
1134                                                            LdrEntry->EntryPointActivationContext);
1135 
1136                     _SEH2_TRY
1137                     {
1138                         /* Check if it has TLS */
1139                         if (LdrEntry->TlsIndex)
1140                         {
1141                             /* Make sure we're not shutting down */
1142                             if (!LdrpShutdownInProgress)
1143                             {
1144                                 /* Call TLS */
1145                                 LdrpCallTlsInitializers(LdrEntry, DLL_THREAD_DETACH);
1146                             }
1147                         }
1148 
1149                         /* Make sure we're not shutting down */
1150                         if (!LdrpShutdownInProgress)
1151                         {
1152                             /* Call the Entrypoint */
1153                             DPRINT("%wZ - Calling entry point at %p for thread detaching\n",
1154                                    &LdrEntry->BaseDllName, LdrEntry->EntryPoint);
1155                             LdrpCallInitRoutine(EntryPoint,
1156                                                 LdrEntry->DllBase,
1157                                                 DLL_THREAD_DETACH,
1158                                                 NULL);
1159                         }
1160                     }
1161                     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
1162                     {
1163                         DPRINT1("WARNING: Exception 0x%x during LdrpCallInitRoutine(DLL_THREAD_DETACH) for %wZ\n",
1164                                 _SEH2_GetExceptionCode(), &LdrEntry->BaseDllName);
1165                     }
1166                     _SEH2_END;
1167 
1168                     /* Deactivate the ActCtx */
1169                     RtlDeactivateActivationContextUnsafeFast(&ActCtx);
1170                 }
1171             }
1172         }
1173     }
1174 
1175     /* Check for TLS */
1176     if (LdrpImageHasTls)
1177     {
1178         /* Set up the Act Ctx */
1179         ActCtx.Size = sizeof(ActCtx);
1180         ActCtx.Format = 1;
1181         RtlZeroMemory(&ActCtx.Frame, sizeof(RTL_ACTIVATION_CONTEXT_STACK_FRAME));
1182 
1183         /* Activate the ActCtx */
1184         RtlActivateActivationContextUnsafeFast(&ActCtx,
1185                                                LdrpImageEntry->EntryPointActivationContext);
1186 
1187         _SEH2_TRY
1188         {
1189             /* Do TLS callbacks */
1190             LdrpCallTlsInitializers(LdrpImageEntry, DLL_THREAD_DETACH);
1191         }
1192         _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
1193         {
1194             /* Do nothing */
1195         }
1196         _SEH2_END;
1197 
1198         /* Deactivate the ActCtx */
1199         RtlDeactivateActivationContextUnsafeFast(&ActCtx);
1200     }
1201 
1202     /* Free TLS */
1203     LdrpFreeTls();
1204     RtlLeaveCriticalSection(&LdrpLoaderLock);
1205 
1206     /* Check for expansion slots */
1207     if (Teb->TlsExpansionSlots)
1208     {
1209         /* Free expansion slots */
1210         RtlFreeHeap(RtlGetProcessHeap(), 0, Teb->TlsExpansionSlots);
1211     }
1212 
1213     /* Check for FLS Data */
1214     if (Teb->FlsData)
1215     {
1216         /* Mimic BaseRundownFls */
1217         ULONG n, FlsHighIndex;
1218         PRTL_FLS_DATA pFlsData;
1219         PFLS_CALLBACK_FUNCTION lpCallback;
1220 
1221         pFlsData = Teb->FlsData;
1222 
1223         RtlAcquirePebLock();
1224         FlsHighIndex = NtCurrentPeb()->FlsHighIndex;
1225         RemoveEntryList(&pFlsData->ListEntry);
1226         RtlReleasePebLock();
1227 
1228         for (n = 1; n <= FlsHighIndex; ++n)
1229         {
1230             lpCallback = NtCurrentPeb()->FlsCallback[n];
1231             if (lpCallback && pFlsData->Data[n])
1232             {
1233                 lpCallback(pFlsData->Data[n]);
1234             }
1235         }
1236 
1237         RtlFreeHeap(RtlGetProcessHeap(), 0, pFlsData);
1238         Teb->FlsData = NULL;
1239     }
1240 
1241     /* Check for Fiber data */
1242     if (Teb->HasFiberData)
1243     {
1244         /* Free Fiber data*/
1245         RtlFreeHeap(RtlGetProcessHeap(), 0, Teb->NtTib.FiberData);
1246         Teb->NtTib.FiberData = NULL;
1247     }
1248 
1249     /* Free the activation context stack */
1250     RtlFreeThreadActivationContextStack();
1251     DPRINT("LdrShutdownThread() done\n");
1252 
1253     return STATUS_SUCCESS;
1254 }
1255 
1256 NTSTATUS
1257 NTAPI
1258 LdrpInitializeTls(VOID)
1259 {
1260     PLIST_ENTRY NextEntry, ListHead;
1261     PLDR_DATA_TABLE_ENTRY LdrEntry;
1262     PIMAGE_TLS_DIRECTORY TlsDirectory;
1263     PLDRP_TLS_DATA TlsData;
1264     ULONG Size;
1265 
1266     /* Initialize the TLS List */
1267     InitializeListHead(&LdrpTlsList);
1268 
1269     /* Loop all the modules */
1270     ListHead = &NtCurrentPeb()->Ldr->InLoadOrderModuleList;
1271     NextEntry = ListHead->Flink;
1272     while (ListHead != NextEntry)
1273     {
1274         /* Get the entry */
1275         LdrEntry = CONTAINING_RECORD(NextEntry, LDR_DATA_TABLE_ENTRY, InLoadOrderLinks);
1276         NextEntry = NextEntry->Flink;
1277 
1278         /* Get the TLS directory */
1279         TlsDirectory = RtlImageDirectoryEntryToData(LdrEntry->DllBase,
1280                                                     TRUE,
1281                                                     IMAGE_DIRECTORY_ENTRY_TLS,
1282                                                     &Size);
1283 
1284         /* Check if we have a directory */
1285         if (!TlsDirectory) continue;
1286 
1287         /* Check if the image has TLS */
1288         if (!LdrpImageHasTls) LdrpImageHasTls = TRUE;
1289 
1290         /* Show debug message */
1291         if (ShowSnaps)
1292         {
1293             DPRINT1("LDR: Tls Found in %wZ at %p\n",
1294                     &LdrEntry->BaseDllName,
1295                     TlsDirectory);
1296         }
1297 
1298         /* Allocate an entry */
1299         TlsData = RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(LDRP_TLS_DATA));
1300         if (!TlsData) return STATUS_NO_MEMORY;
1301 
1302         /* Lock the DLL and mark it for TLS Usage */
1303         LdrEntry->LoadCount = -1;
1304         LdrEntry->TlsIndex = -1;
1305 
1306         /* Save the cached TLS data */
1307         TlsData->TlsDirectory = *TlsDirectory;
1308         InsertTailList(&LdrpTlsList, &TlsData->TlsLinks);
1309 
1310         /* Update the index */
1311         *(PLONG)TlsData->TlsDirectory.AddressOfIndex = LdrpNumberOfTlsEntries;
1312         TlsData->TlsDirectory.Characteristics = LdrpNumberOfTlsEntries++;
1313     }
1314 
1315     /* Done setting up TLS, allocate entries */
1316     return LdrpAllocateTls();
1317 }
1318 
1319 NTSTATUS
1320 NTAPI
1321 LdrpAllocateTls(VOID)
1322 {
1323     PTEB Teb = NtCurrentTeb();
1324     PLIST_ENTRY NextEntry, ListHead;
1325     PLDRP_TLS_DATA TlsData;
1326     SIZE_T TlsDataSize;
1327     PVOID *TlsVector;
1328 
1329     /* Check if we have any entries */
1330     if (!LdrpNumberOfTlsEntries)
1331         return STATUS_SUCCESS;
1332 
1333     /* Allocate the vector array */
1334     TlsVector = RtlAllocateHeap(RtlGetProcessHeap(),
1335                                     0,
1336                                     LdrpNumberOfTlsEntries * sizeof(PVOID));
1337     if (!TlsVector) return STATUS_NO_MEMORY;
1338     Teb->ThreadLocalStoragePointer = TlsVector;
1339 
1340     /* Loop the TLS Array */
1341     ListHead = &LdrpTlsList;
1342     NextEntry = ListHead->Flink;
1343     while (NextEntry != ListHead)
1344     {
1345         /* Get the entry */
1346         TlsData = CONTAINING_RECORD(NextEntry, LDRP_TLS_DATA, TlsLinks);
1347         NextEntry = NextEntry->Flink;
1348 
1349         /* Allocate this vector */
1350         TlsDataSize = TlsData->TlsDirectory.EndAddressOfRawData -
1351                       TlsData->TlsDirectory.StartAddressOfRawData;
1352         TlsVector[TlsData->TlsDirectory.Characteristics] = RtlAllocateHeap(RtlGetProcessHeap(),
1353                                                                            0,
1354                                                                            TlsDataSize);
1355         if (!TlsVector[TlsData->TlsDirectory.Characteristics])
1356         {
1357             /* Out of memory */
1358             return STATUS_NO_MEMORY;
1359         }
1360 
1361         /* Show debug message */
1362         if (ShowSnaps)
1363         {
1364             DPRINT1("LDR: TlsVector %p Index %lu = %p copied from %x to %p\n",
1365                     TlsVector,
1366                     TlsData->TlsDirectory.Characteristics,
1367                     &TlsVector[TlsData->TlsDirectory.Characteristics],
1368                     TlsData->TlsDirectory.StartAddressOfRawData,
1369                     TlsVector[TlsData->TlsDirectory.Characteristics]);
1370         }
1371 
1372         /* Copy the data */
1373         RtlCopyMemory(TlsVector[TlsData->TlsDirectory.Characteristics],
1374                       (PVOID)TlsData->TlsDirectory.StartAddressOfRawData,
1375                       TlsDataSize);
1376     }
1377 
1378     /* Done */
1379     return STATUS_SUCCESS;
1380 }
1381 
1382 VOID
1383 NTAPI
1384 LdrpFreeTls(VOID)
1385 {
1386     PLIST_ENTRY ListHead, NextEntry;
1387     PLDRP_TLS_DATA TlsData;
1388     PVOID *TlsVector;
1389     PTEB Teb = NtCurrentTeb();
1390 
1391     /* Get a pointer to the vector array */
1392     TlsVector = Teb->ThreadLocalStoragePointer;
1393     if (!TlsVector) return;
1394 
1395     /* Loop through it */
1396     ListHead = &LdrpTlsList;
1397     NextEntry = ListHead->Flink;
1398     while (NextEntry != ListHead)
1399     {
1400         TlsData = CONTAINING_RECORD(NextEntry, LDRP_TLS_DATA, TlsLinks);
1401         NextEntry = NextEntry->Flink;
1402 
1403         /* Free each entry */
1404         if (TlsVector[TlsData->TlsDirectory.Characteristics])
1405         {
1406             RtlFreeHeap(RtlGetProcessHeap(),
1407                         0,
1408                         TlsVector[TlsData->TlsDirectory.Characteristics]);
1409         }
1410     }
1411 
1412     /* Free the array itself */
1413     RtlFreeHeap(RtlGetProcessHeap(),
1414                 0,
1415                 TlsVector);
1416 }
1417 
1418 NTSTATUS
1419 NTAPI
1420 LdrpInitializeExecutionOptions(PUNICODE_STRING ImagePathName, PPEB Peb, PHANDLE OptionsKey)
1421 {
1422     NTSTATUS Status;
1423     HANDLE KeyHandle;
1424     ULONG ExecuteOptions, MinimumStackCommit = 0, GlobalFlag;
1425 
1426     /* Return error if we were not provided a pointer where to save the options key handle */
1427     if (!OptionsKey) return STATUS_INVALID_HANDLE;
1428 
1429     /* Zero initialize the options key pointer */
1430     *OptionsKey = NULL;
1431 
1432     /* Open the options key */
1433     Status = LdrOpenImageFileOptionsKey(ImagePathName, 0, &KeyHandle);
1434 
1435     /* Save it if it was opened successfully */
1436     if (NT_SUCCESS(Status))
1437         *OptionsKey = KeyHandle;
1438 
1439     if (KeyHandle)
1440     {
1441         /* There are image specific options, read them starting with NXCOMPAT */
1442         Status = LdrQueryImageFileKeyOption(KeyHandle,
1443                                             L"ExecuteOptions",
1444                                             4,
1445                                             &ExecuteOptions,
1446                                             sizeof(ExecuteOptions),
1447                                             0);
1448 
1449         if (NT_SUCCESS(Status))
1450         {
1451             /* TODO: Set execution options for the process */
1452             /*
1453             if (ExecuteOptions == 0)
1454                 ExecuteOptions = 1;
1455             else
1456                 ExecuteOptions = 2;
1457             ZwSetInformationProcess(NtCurrentProcess(),
1458                                     ProcessExecuteFlags,
1459                                     &ExecuteOptions,
1460                                     sizeof(ULONG));*/
1461 
1462         }
1463 
1464         /* Check if this image uses large pages */
1465         if (Peb->ImageUsesLargePages)
1466         {
1467             /* TODO: If it does, open large page key */
1468             UNIMPLEMENTED;
1469         }
1470 
1471         /* Get various option values */
1472         LdrQueryImageFileKeyOption(KeyHandle,
1473                                    L"DisableHeapLookaside",
1474                                    REG_DWORD,
1475                                    &RtlpDisableHeapLookaside,
1476                                    sizeof(RtlpDisableHeapLookaside),
1477                                    NULL);
1478 
1479         LdrQueryImageFileKeyOption(KeyHandle,
1480                                    L"ShutdownFlags",
1481                                    REG_DWORD,
1482                                    &RtlpShutdownProcessFlags,
1483                                    sizeof(RtlpShutdownProcessFlags),
1484                                    NULL);
1485 
1486         LdrQueryImageFileKeyOption(KeyHandle,
1487                                    L"MinimumStackCommitInBytes",
1488                                    REG_DWORD,
1489                                    &MinimumStackCommit,
1490                                    sizeof(MinimumStackCommit),
1491                                    NULL);
1492 
1493         /* Update PEB's minimum stack commit if it's lower */
1494         if (Peb->MinimumStackCommit < MinimumStackCommit)
1495             Peb->MinimumStackCommit = MinimumStackCommit;
1496 
1497         /* Set the global flag */
1498         Status = LdrQueryImageFileKeyOption(KeyHandle,
1499                                             L"GlobalFlag",
1500                                             REG_DWORD,
1501                                             &GlobalFlag,
1502                                             sizeof(GlobalFlag),
1503                                             NULL);
1504 
1505         if (NT_SUCCESS(Status))
1506             Peb->NtGlobalFlag = GlobalFlag;
1507         else
1508             GlobalFlag = 0;
1509 
1510         /* Call AVRF if necessary */
1511         if (Peb->NtGlobalFlag & (FLG_APPLICATION_VERIFIER | FLG_HEAP_PAGE_ALLOCS))
1512         {
1513             Status = LdrpInitializeApplicationVerifierPackage(KeyHandle, Peb, TRUE, FALSE);
1514             if (!NT_SUCCESS(Status))
1515             {
1516                 DPRINT1("AVRF: LdrpInitializeApplicationVerifierPackage failed with %08X\n", Status);
1517             }
1518         }
1519     }
1520     else
1521     {
1522         /* There are no image-specific options, so perform global initialization */
1523         if (Peb->NtGlobalFlag & (FLG_APPLICATION_VERIFIER | FLG_HEAP_PAGE_ALLOCS))
1524         {
1525             /* Initialize app verifier package */
1526             Status = LdrpInitializeApplicationVerifierPackage(KeyHandle, Peb, TRUE, FALSE);
1527             if (!NT_SUCCESS(Status))
1528             {
1529                 DPRINT1("AVRF: LdrpInitializeApplicationVerifierPackage failed with %08X\n", Status);
1530             }
1531         }
1532     }
1533 
1534     return STATUS_SUCCESS;
1535 }
1536 
1537 VOID
1538 NTAPI
1539 LdrpValidateImageForMp(IN PLDR_DATA_TABLE_ENTRY LdrDataTableEntry)
1540 {
1541     UNIMPLEMENTED;
1542 }
1543 
1544 BOOLEAN
1545 NTAPI
1546 LdrpDisableProcessCompatGuidDetection(VOID)
1547 {
1548     UNICODE_STRING PolicyKey = RTL_CONSTANT_STRING(L"\\Registry\\MACHINE\\Software\\Policies\\Microsoft\\Windows\\AppCompat");
1549     UNICODE_STRING DisableDetection = RTL_CONSTANT_STRING(L"DisableCompatGuidDetection");
1550     OBJECT_ATTRIBUTES PolicyKeyAttributes = RTL_CONSTANT_OBJECT_ATTRIBUTES(&PolicyKey, OBJ_CASE_INSENSITIVE);
1551     KEY_VALUE_PARTIAL_INFORMATION KeyInfo;
1552     ULONG ResultLength;
1553     NTSTATUS Status;
1554     HANDLE KeyHandle;
1555 
1556     Status = NtOpenKey(&KeyHandle, KEY_QUERY_VALUE, &PolicyKeyAttributes);
1557     if (NT_SUCCESS(Status))
1558     {
1559         Status = NtQueryValueKey(KeyHandle,
1560                                  &DisableDetection,
1561                                  KeyValuePartialInformation,
1562                                  &KeyInfo,
1563                                  sizeof(KeyInfo),
1564                                  &ResultLength);
1565         NtClose(KeyHandle);
1566         if ((NT_SUCCESS(Status)) &&
1567             (KeyInfo.Type == REG_DWORD) &&
1568             (KeyInfo.DataLength == sizeof(ULONG)) &&
1569             (KeyInfo.Data[0] == TRUE))
1570         {
1571             return TRUE;
1572         }
1573     }
1574     return FALSE;
1575 }
1576 
1577 
1578 VOID
1579 NTAPI
1580 LdrpInitializeProcessCompat(PVOID pProcessActctx, PVOID* pOldShimData)
1581 {
1582     static const struct
1583     {
1584         const GUID* Guid;
1585         const DWORD Version;
1586     } KnownCompatGuids[] = {
1587         { &COMPAT_GUID_WIN10, _WIN32_WINNT_WIN10 },
1588         { &COMPAT_GUID_WIN81, _WIN32_WINNT_WINBLUE },
1589         { &COMPAT_GUID_WIN8, _WIN32_WINNT_WIN8 },
1590         { &COMPAT_GUID_WIN7, _WIN32_WINNT_WIN7 },
1591         { &COMPAT_GUID_VISTA, _WIN32_WINNT_VISTA },
1592     };
1593 
1594     ULONG Buffer[(sizeof(COMPATIBILITY_CONTEXT_ELEMENT) * 10 + sizeof(ACTIVATION_CONTEXT_COMPATIBILITY_INFORMATION)) / sizeof(ULONG)];
1595     ACTIVATION_CONTEXT_COMPATIBILITY_INFORMATION* ContextCompatInfo;
1596     SIZE_T SizeRequired;
1597     NTSTATUS Status;
1598     DWORD n, cur;
1599     ReactOS_ShimData* pShimData = *pOldShimData;
1600 
1601     if (pShimData)
1602     {
1603         if (pShimData->dwMagic != REACTOS_SHIMDATA_MAGIC ||
1604             pShimData->dwSize != sizeof(ReactOS_ShimData))
1605         {
1606             DPRINT1("LdrpInitializeProcessCompat: Corrupt pShimData (0x%x, %u)\n", pShimData->dwMagic, pShimData->dwSize);
1607             return;
1608         }
1609         if (pShimData->dwRosProcessCompatVersion)
1610         {
1611             if (pShimData->dwRosProcessCompatVersion == REACTOS_COMPATVERSION_IGNOREMANIFEST)
1612             {
1613                 DPRINT1("LdrpInitializeProcessCompat: ProcessCompatVersion set to ignore manifest\n");
1614             }
1615             else
1616             {
1617                 DPRINT1("LdrpInitializeProcessCompat: ProcessCompatVersion already set to 0x%x\n", pShimData->dwRosProcessCompatVersion);
1618             }
1619             return;
1620         }
1621     }
1622 
1623     SizeRequired = sizeof(Buffer);
1624     Status = RtlQueryInformationActivationContext(RTL_QUERY_ACTIVATION_CONTEXT_FLAG_NO_ADDREF,
1625                                                   pProcessActctx,
1626                                                   NULL,
1627                                                   CompatibilityInformationInActivationContext,
1628                                                   Buffer,
1629                                                   sizeof(Buffer),
1630                                                   &SizeRequired);
1631 
1632     if (!NT_SUCCESS(Status))
1633     {
1634         DPRINT1("LdrpInitializeProcessCompat: Unable to query process actctx with status %x\n", Status);
1635         return;
1636     }
1637 
1638     ContextCompatInfo = (ACTIVATION_CONTEXT_COMPATIBILITY_INFORMATION*)Buffer;
1639     /* No Compatibility elements present, bail out */
1640     if (ContextCompatInfo->ElementCount == 0)
1641         return;
1642 
1643     /* Search for known GUIDs, starting from oldest to newest.
1644        Note that on Windows it is somewhat reversed, starting from the latest known
1645        version, going down. But we are not Windows, trying to allow a lower version,
1646        we are ReactOS trying to fake a higher version. So we interpret what Windows
1647        does as "try the closest version to the actual version", so we start with the
1648        lowest version, which is closest to Windows 2003, which we mostly are. */
1649     for (cur = RTL_NUMBER_OF(KnownCompatGuids) - 1; cur != -1; --cur)
1650     {
1651         for (n = 0; n < ContextCompatInfo->ElementCount; ++n)
1652         {
1653             if (ContextCompatInfo->Elements[n].Type == ACTCTX_COMPATIBILITY_ELEMENT_TYPE_OS &&
1654                 RtlCompareMemory(&ContextCompatInfo->Elements[n].Id, KnownCompatGuids[cur].Guid, sizeof(GUID)) == sizeof(GUID))
1655             {
1656                 if (LdrpDisableProcessCompatGuidDetection())
1657                 {
1658                     DPRINT1("LdrpInitializeProcessCompat: Not applying automatic fix for winver 0x%x due to policy\n", KnownCompatGuids[cur].Version);
1659                     return;
1660                 }
1661 
1662                 /* If this process did not need shim data before, allocate and store it */
1663                 if (pShimData == NULL)
1664                 {
1665                     PPEB Peb = NtCurrentPeb();
1666 
1667                     ASSERT(Peb->pShimData == NULL);
1668                     pShimData = RtlAllocateHeap(Peb->ProcessHeap, HEAP_ZERO_MEMORY, sizeof(*pShimData));
1669 
1670                     if (!pShimData)
1671                     {
1672                         DPRINT1("LdrpInitializeProcessCompat: Unable to allocated %u bytes\n", sizeof(*pShimData));
1673                         return;
1674                     }
1675 
1676                     pShimData->dwSize = sizeof(*pShimData);
1677                     pShimData->dwMagic = REACTOS_SHIMDATA_MAGIC;
1678 
1679                     Peb->pShimData = pShimData;
1680                     *pOldShimData = pShimData;
1681                 }
1682 
1683                 /* Store the lowest found version, and bail out. */
1684                 pShimData->dwRosProcessCompatVersion = KnownCompatGuids[cur].Version;
1685                 DPRINT1("LdrpInitializeProcessCompat: Found guid for winver 0x%x in manifest from %wZ\n",
1686                         KnownCompatGuids[cur].Version,
1687                         &(NtCurrentPeb()->ProcessParameters->ImagePathName));
1688                 return;
1689             }
1690         }
1691     }
1692 }
1693 
1694 VOID
1695 NTAPI
1696 LdrpInitializeDotLocalSupport(PRTL_USER_PROCESS_PARAMETERS ProcessParameters)
1697 {
1698     UNICODE_STRING ImagePathName = ProcessParameters->ImagePathName;
1699     WCHAR LocalBuffer[MAX_PATH];
1700     UNICODE_STRING DotLocal;
1701     NTSTATUS Status;
1702     ULONG RequiredSize;
1703 
1704     RequiredSize = ImagePathName.Length + LdrpDotLocal.Length + sizeof(UNICODE_NULL);
1705     if (RequiredSize <= sizeof(LocalBuffer))
1706     {
1707         RtlInitEmptyUnicodeString(&DotLocal, LocalBuffer, sizeof(LocalBuffer));
1708     }
1709     else if (RequiredSize <= UNICODE_STRING_MAX_BYTES)
1710     {
1711         DotLocal.Buffer = RtlAllocateHeap(RtlGetProcessHeap(), 0, RequiredSize);
1712         DotLocal.Length = 0;
1713         DotLocal.MaximumLength = RequiredSize;
1714         if (!DotLocal.Buffer)
1715             DPRINT1("LDR: Failed to allocate memory for .local check\n");
1716     }
1717     else
1718     {
1719         DotLocal.Buffer = NULL;
1720         DotLocal.Length = 0;
1721         DotLocal.MaximumLength = 0;
1722         DPRINT1("LDR: String too big for .local check\n");
1723     }
1724 
1725     if (DotLocal.Buffer)
1726     {
1727         Status = RtlAppendUnicodeStringToString(&DotLocal, &ImagePathName);
1728         ASSERT(NT_SUCCESS(Status));
1729         if (NT_SUCCESS(Status))
1730         {
1731             Status = RtlAppendUnicodeStringToString(&DotLocal, &LdrpDotLocal);
1732             ASSERT(NT_SUCCESS(Status));
1733         }
1734 
1735         if (NT_SUCCESS(Status))
1736         {
1737             if (RtlDoesFileExists_UStr(&DotLocal))
1738             {
1739                 ProcessParameters->Flags |= RTL_USER_PROCESS_PARAMETERS_PRIVATE_DLL_PATH;
1740             }
1741         }
1742         else
1743         {
1744             DPRINT1("LDR: Failed to append: 0x%lx\n", Status);
1745         }
1746 
1747         if (DotLocal.Buffer != LocalBuffer)
1748         {
1749             RtlFreeHeap(RtlGetProcessHeap(), 0, DotLocal.Buffer);
1750         }
1751     }
1752 }
1753 
1754 
1755 NTSTATUS
1756 NTAPI
1757 LdrpInitializeProcess(IN PCONTEXT Context,
1758                       IN PVOID SystemArgument1)
1759 {
1760     RTL_HEAP_PARAMETERS HeapParameters;
1761     ULONG ComSectionSize;
1762     ANSI_STRING BaseProcessInitPostImportName = RTL_CONSTANT_STRING("BaseProcessInitPostImport");
1763     ANSI_STRING BaseQueryModuleDataName = RTL_CONSTANT_STRING("BaseQueryModuleData");
1764     PVOID OldShimData;
1765     OBJECT_ATTRIBUTES ObjectAttributes;
1766     //UNICODE_STRING LocalFileName, FullImageName;
1767     HANDLE SymLinkHandle;
1768     //ULONG DebugHeapOnly;
1769     UNICODE_STRING CommandLine, NtSystemRoot, ImagePathName, FullPath, ImageFileName, KnownDllString;
1770     PPEB Peb = NtCurrentPeb();
1771     BOOLEAN IsDotNetImage = FALSE;
1772     BOOLEAN FreeCurDir = FALSE;
1773     //HANDLE CompatKey;
1774     PRTL_USER_PROCESS_PARAMETERS ProcessParameters;
1775     //LPWSTR ImagePathBuffer;
1776     ULONG ConfigSize;
1777     UNICODE_STRING CurrentDirectory;
1778     HANDLE OptionsKey;
1779     ULONG HeapFlags;
1780     PIMAGE_NT_HEADERS NtHeader;
1781     LPWSTR NtDllName = NULL;
1782     NTSTATUS Status, ImportStatus;
1783     NLSTABLEINFO NlsTable;
1784     PIMAGE_LOAD_CONFIG_DIRECTORY LoadConfig;
1785     PTEB Teb = NtCurrentTeb();
1786     PLIST_ENTRY ListHead;
1787     PLIST_ENTRY NextEntry;
1788     ULONG i;
1789     PWSTR ImagePath;
1790     ULONG DebugProcessHeapOnly = 0;
1791     PLDR_DATA_TABLE_ENTRY NtLdrEntry;
1792     PWCHAR Current;
1793     ULONG ExecuteOptions = 0;
1794     PVOID ViewBase;
1795 
1796     /* Set a NULL SEH Filter */
1797     RtlSetUnhandledExceptionFilter(NULL);
1798 
1799     /* Get the image path */
1800     ImagePath = Peb->ProcessParameters->ImagePathName.Buffer;
1801 
1802     /* Check if it's not normalized */
1803     if (!(Peb->ProcessParameters->Flags & RTL_USER_PROCESS_PARAMETERS_NORMALIZED))
1804     {
1805         /* Normalize it*/
1806         ImagePath = (PWSTR)((ULONG_PTR)ImagePath + (ULONG_PTR)Peb->ProcessParameters);
1807     }
1808 
1809     /* Create a unicode string for the Image Path */
1810     ImagePathName.Length = Peb->ProcessParameters->ImagePathName.Length;
1811     ImagePathName.MaximumLength = ImagePathName.Length + sizeof(WCHAR);
1812     ImagePathName.Buffer = ImagePath;
1813 
1814     /* Get the NT Headers */
1815     NtHeader = RtlImageNtHeader(Peb->ImageBaseAddress);
1816 
1817     /* Get the execution options */
1818     Status = LdrpInitializeExecutionOptions(&ImagePathName, Peb, &OptionsKey);
1819 
1820     /* Check if this is a .NET executable */
1821     if (RtlImageDirectoryEntryToData(Peb->ImageBaseAddress,
1822                                      TRUE,
1823                                      IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR,
1824                                      &ComSectionSize))
1825     {
1826         /* Remember this for later */
1827         IsDotNetImage = TRUE;
1828     }
1829 
1830     /* Save the NTDLL Base address */
1831     NtDllBase = SystemArgument1;
1832 
1833     /* If this is a Native Image */
1834     if (NtHeader->OptionalHeader.Subsystem == IMAGE_SUBSYSTEM_NATIVE)
1835     {
1836         /* Then do DLL Validation */
1837         LdrpDllValidation = TRUE;
1838     }
1839 
1840     /* Save the old Shim Data */
1841     OldShimData = Peb->pShimData;
1842 
1843     /* ReactOS specific: do not clear it. (Windows starts doing the same in later versions) */
1844     //Peb->pShimData = NULL;
1845 
1846     /* Save the number of processors and CS timeout */
1847     LdrpNumberOfProcessors = Peb->NumberOfProcessors;
1848     RtlpTimeout = Peb->CriticalSectionTimeout;
1849 
1850     /* Normalize the parameters */
1851     ProcessParameters = RtlNormalizeProcessParams(Peb->ProcessParameters);
1852     if (ProcessParameters)
1853     {
1854         /* Save the Image and Command Line Names */
1855         ImageFileName = ProcessParameters->ImagePathName;
1856         CommandLine = ProcessParameters->CommandLine;
1857     }
1858     else
1859     {
1860         /* It failed, initialize empty strings */
1861         RtlInitUnicodeString(&ImageFileName, NULL);
1862         RtlInitUnicodeString(&CommandLine, NULL);
1863     }
1864 
1865     /* Initialize NLS data */
1866     RtlInitNlsTables(Peb->AnsiCodePageData,
1867                      Peb->OemCodePageData,
1868                      Peb->UnicodeCaseTableData,
1869                      &NlsTable);
1870 
1871     /* Reset NLS Translations */
1872     RtlResetRtlTranslations(&NlsTable);
1873 
1874     /* Get the Image Config Directory */
1875     LoadConfig = RtlImageDirectoryEntryToData(Peb->ImageBaseAddress,
1876                                               TRUE,
1877                                               IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG,
1878                                               &ConfigSize);
1879 
1880     /* Setup the Heap Parameters */
1881     RtlZeroMemory(&HeapParameters, sizeof(HeapParameters));
1882     HeapFlags = HEAP_GROWABLE;
1883     HeapParameters.Length = sizeof(HeapParameters);
1884 
1885     /* Check if we have Configuration Data */
1886 #define VALID_CONFIG_FIELD(Name) (ConfigSize >= RTL_SIZEOF_THROUGH_FIELD(IMAGE_LOAD_CONFIG_DIRECTORY, Name))
1887     /* The 'original' load config ends after SecurityCookie */
1888     if ((LoadConfig) && ConfigSize && (VALID_CONFIG_FIELD(SecurityCookie) || ConfigSize == LoadConfig->Size))
1889     {
1890         if (ConfigSize != sizeof(IMAGE_LOAD_CONFIG_DIRECTORY))
1891             DPRINT1("WARN: Accepting different LOAD_CONFIG size!\n");
1892         else
1893             DPRINT1("Applying LOAD_CONFIG\n");
1894 
1895         if (VALID_CONFIG_FIELD(GlobalFlagsSet) && LoadConfig->GlobalFlagsSet)
1896             Peb->NtGlobalFlag |= LoadConfig->GlobalFlagsSet;
1897 
1898         if (VALID_CONFIG_FIELD(GlobalFlagsClear) && LoadConfig->GlobalFlagsClear)
1899             Peb->NtGlobalFlag &= ~LoadConfig->GlobalFlagsClear;
1900 
1901         /* Convert the default CS timeout from milliseconds to 100ns units */
1902         if (VALID_CONFIG_FIELD(CriticalSectionDefaultTimeout) && LoadConfig->CriticalSectionDefaultTimeout)
1903             RtlpTimeout.QuadPart = Int32x32To64(LoadConfig->CriticalSectionDefaultTimeout, -10000);
1904 
1905         if (VALID_CONFIG_FIELD(DeCommitFreeBlockThreshold) && LoadConfig->DeCommitFreeBlockThreshold)
1906             HeapParameters.DeCommitFreeBlockThreshold = LoadConfig->DeCommitFreeBlockThreshold;
1907 
1908         if (VALID_CONFIG_FIELD(DeCommitTotalFreeThreshold) && LoadConfig->DeCommitTotalFreeThreshold)
1909             HeapParameters.DeCommitTotalFreeThreshold = LoadConfig->DeCommitTotalFreeThreshold;
1910 
1911         if (VALID_CONFIG_FIELD(MaximumAllocationSize) && LoadConfig->MaximumAllocationSize)
1912             HeapParameters.MaximumAllocationSize = LoadConfig->MaximumAllocationSize;
1913 
1914         if (VALID_CONFIG_FIELD(VirtualMemoryThreshold) && LoadConfig->VirtualMemoryThreshold)
1915             HeapParameters.VirtualMemoryThreshold = LoadConfig->VirtualMemoryThreshold;
1916 
1917         if (VALID_CONFIG_FIELD(ProcessHeapFlags) && LoadConfig->ProcessHeapFlags)
1918             HeapFlags = LoadConfig->ProcessHeapFlags;
1919     }
1920 #undef VALID_CONFIG_FIELD
1921 
1922     /* Check for custom affinity mask */
1923     if (Peb->ImageProcessAffinityMask)
1924     {
1925         /* Set it */
1926         Status = NtSetInformationProcess(NtCurrentProcess(),
1927                                          ProcessAffinityMask,
1928                                          &Peb->ImageProcessAffinityMask,
1929                                          sizeof(Peb->ImageProcessAffinityMask));
1930     }
1931 
1932     /* Check if verbose debugging (ShowSnaps) was requested */
1933     ShowSnaps = Peb->NtGlobalFlag & FLG_SHOW_LDR_SNAPS;
1934 
1935     /* Start verbose debugging messages right now if they were requested */
1936     if (ShowSnaps)
1937     {
1938         DPRINT1("LDR: PID: 0x%p started - '%wZ'\n",
1939                 Teb->ClientId.UniqueProcess,
1940                 &CommandLine);
1941     }
1942 
1943     /* If the CS timeout is longer than 1 hour, disable it */
1944     if (RtlpTimeout.QuadPart < Int32x32To64(3600, -10000000))
1945         RtlpTimeoutDisable = TRUE;
1946 
1947     /* Initialize Critical Section Data */
1948     RtlpInitDeferredCriticalSection();
1949 
1950     /* Initialize VEH Call lists */
1951     RtlpInitializeVectoredExceptionHandling();
1952 
1953     /* Set TLS/FLS Bitmap data */
1954     Peb->FlsBitmap = &FlsBitMap;
1955     Peb->TlsBitmap = &TlsBitMap;
1956     Peb->TlsExpansionBitmap = &TlsExpansionBitMap;
1957 
1958     /* Initialize FLS Bitmap */
1959     RtlInitializeBitMap(&FlsBitMap,
1960                         Peb->FlsBitmapBits,
1961                         FLS_MAXIMUM_AVAILABLE);
1962     RtlSetBit(&FlsBitMap, 0);
1963     InitializeListHead(&Peb->FlsListHead);
1964 
1965     /* Initialize TLS Bitmap */
1966     RtlInitializeBitMap(&TlsBitMap,
1967                         Peb->TlsBitmapBits,
1968                         TLS_MINIMUM_AVAILABLE);
1969     RtlSetBit(&TlsBitMap, 0);
1970     RtlInitializeBitMap(&TlsExpansionBitMap,
1971                         Peb->TlsExpansionBitmapBits,
1972                         TLS_EXPANSION_SLOTS);
1973     RtlSetBit(&TlsExpansionBitMap, 0);
1974 
1975     /* Initialize the Hash Table */
1976     for (i = 0; i < LDR_HASH_TABLE_ENTRIES; i++)
1977     {
1978         InitializeListHead(&LdrpHashTable[i]);
1979     }
1980 
1981     /* Initialize the Loader Lock */
1982     // FIXME: What's the point of initing it manually, if two lines lower
1983     //        a call to RtlInitializeCriticalSection() is being made anyway?
1984     //InsertTailList(&RtlCriticalSectionList, &LdrpLoaderLock.DebugInfo->ProcessLocksList);
1985     //LdrpLoaderLock.DebugInfo->CriticalSection = &LdrpLoaderLock;
1986     RtlInitializeCriticalSection(&LdrpLoaderLock);
1987     LdrpLoaderLockInit = TRUE;
1988 
1989     /* Check if User Stack Trace Database support was requested */
1990     if (Peb->NtGlobalFlag & FLG_USER_STACK_TRACE_DB)
1991     {
1992         DPRINT1("We don't support user stack trace databases yet\n");
1993     }
1994 
1995     /* Setup Fast PEB Lock */
1996     RtlInitializeCriticalSection(&FastPebLock);
1997     Peb->FastPebLock = &FastPebLock;
1998     //Peb->FastPebLockRoutine = (PPEBLOCKROUTINE)RtlEnterCriticalSection;
1999     //Peb->FastPebUnlockRoutine = (PPEBLOCKROUTINE)RtlLeaveCriticalSection;
2000 
2001     /* Setup Callout Lock and Notification list */
2002     //RtlInitializeCriticalSection(&RtlpCalloutEntryLock);
2003     InitializeListHead(&LdrpDllNotificationList);
2004 
2005     /* For old executables, use 16-byte aligned heap */
2006     if ((NtHeader->OptionalHeader.MajorSubsystemVersion <= 3) &&
2007         (NtHeader->OptionalHeader.MinorSubsystemVersion < 51))
2008     {
2009         HeapFlags |= HEAP_CREATE_ALIGN_16;
2010     }
2011 
2012     /* Setup the Heap */
2013     RtlInitializeHeapManager();
2014     Peb->ProcessHeap = RtlCreateHeap(HeapFlags,
2015                                      NULL,
2016                                      NtHeader->OptionalHeader.SizeOfHeapReserve,
2017                                      NtHeader->OptionalHeader.SizeOfHeapCommit,
2018                                      NULL,
2019                                      &HeapParameters);
2020 
2021     if (!Peb->ProcessHeap)
2022     {
2023         DPRINT1("Failed to create process heap\n");
2024         return STATUS_NO_MEMORY;
2025     }
2026 
2027     /* Allocate an Activation Context Stack */
2028     Status = RtlAllocateActivationContextStack(&Teb->ActivationContextStackPointer);
2029     if (!NT_SUCCESS(Status)) return Status;
2030 
2031     RtlZeroMemory(&HeapParameters, sizeof(HeapParameters));
2032     HeapFlags = HEAP_GROWABLE | HEAP_CLASS_1;
2033     HeapParameters.Length = sizeof(HeapParameters);
2034     LdrpHeap = RtlCreateHeap(HeapFlags, 0, 0x10000, 0x6000, 0, &HeapParameters);
2035     if (!LdrpHeap)
2036     {
2037         DPRINT1("Failed to create loader private heap\n");
2038         return STATUS_NO_MEMORY;
2039     }
2040 
2041     /* Check for Debug Heap */
2042     if (OptionsKey)
2043     {
2044         /* Query the setting */
2045         Status = LdrQueryImageFileKeyOption(OptionsKey,
2046                                             L"DebugProcessHeapOnly",
2047                                             REG_DWORD,
2048                                             &DebugProcessHeapOnly,
2049                                             sizeof(ULONG),
2050                                             NULL);
2051 
2052         if (NT_SUCCESS(Status))
2053         {
2054             /* Reset DPH if requested */
2055             if (RtlpPageHeapEnabled && DebugProcessHeapOnly)
2056             {
2057                 RtlpDphGlobalFlags &= ~DPH_FLAG_DLL_NOTIFY;
2058                 RtlpPageHeapEnabled = FALSE;
2059             }
2060         }
2061     }
2062 
2063     /* Build the NTDLL Path */
2064     FullPath.Buffer = StringBuffer;
2065     FullPath.Length = 0;
2066     FullPath.MaximumLength = sizeof(StringBuffer);
2067     RtlInitUnicodeString(&NtSystemRoot, SharedUserData->NtSystemRoot);
2068     RtlAppendUnicodeStringToString(&FullPath, &NtSystemRoot);
2069     RtlAppendUnicodeToString(&FullPath, L"\\System32\\");
2070 
2071     /* Open the Known DLLs directory */
2072     RtlInitUnicodeString(&KnownDllString, L"\\KnownDlls");
2073     InitializeObjectAttributes(&ObjectAttributes,
2074                                &KnownDllString,
2075                                OBJ_CASE_INSENSITIVE,
2076                                NULL,
2077                                NULL);
2078     Status = ZwOpenDirectoryObject(&LdrpKnownDllObjectDirectory,
2079                                    DIRECTORY_QUERY | DIRECTORY_TRAVERSE,
2080                                    &ObjectAttributes);
2081 
2082     /* Check if it exists */
2083     if (NT_SUCCESS(Status))
2084     {
2085         /* Open the Known DLLs Path */
2086         RtlInitUnicodeString(&KnownDllString, L"KnownDllPath");
2087         InitializeObjectAttributes(&ObjectAttributes,
2088                                    &KnownDllString,
2089                                    OBJ_CASE_INSENSITIVE,
2090                                    LdrpKnownDllObjectDirectory,
2091                                    NULL);
2092         Status = NtOpenSymbolicLinkObject(&SymLinkHandle,
2093                                           SYMBOLIC_LINK_QUERY,
2094                                           &ObjectAttributes);
2095         if (NT_SUCCESS(Status))
2096         {
2097             /* Query the path */
2098             LdrpKnownDllPath.Length = 0;
2099             LdrpKnownDllPath.MaximumLength = sizeof(LdrpKnownDllPathBuffer);
2100             LdrpKnownDllPath.Buffer = LdrpKnownDllPathBuffer;
2101             Status = ZwQuerySymbolicLinkObject(SymLinkHandle, &LdrpKnownDllPath, NULL);
2102             NtClose(SymLinkHandle);
2103             if (!NT_SUCCESS(Status))
2104             {
2105                 DPRINT1("LDR: %s - failed call to ZwQuerySymbolicLinkObject with status %x\n", "", Status);
2106                 return Status;
2107             }
2108         }
2109     }
2110 
2111     /* Check if we failed */
2112     if (!NT_SUCCESS(Status))
2113     {
2114         /* Assume System32 */
2115         LdrpKnownDllObjectDirectory = NULL;
2116         RtlInitUnicodeString(&LdrpKnownDllPath, StringBuffer);
2117         LdrpKnownDllPath.Length -= sizeof(WCHAR);
2118     }
2119 
2120     /* If we have process parameters, get the default path and current path */
2121     if (ProcessParameters)
2122     {
2123         /* Check if we have a Dll Path */
2124         if (ProcessParameters->DllPath.Length)
2125         {
2126             /* Get the path */
2127             LdrpDefaultPath = *(PUNICODE_STRING)&ProcessParameters->DllPath;
2128         }
2129         else
2130         {
2131             /* We need a valid path */
2132             DPRINT1("No valid DllPath was given!\n");
2133             LdrpInitFailure(STATUS_INVALID_PARAMETER);
2134         }
2135 
2136         /* Set the current directory */
2137         CurrentDirectory = ProcessParameters->CurrentDirectory.DosPath;
2138 
2139         /* Check if it's empty or invalid */
2140         if ((!CurrentDirectory.Buffer) ||
2141             (CurrentDirectory.Buffer[0] == UNICODE_NULL) ||
2142             (!CurrentDirectory.Length))
2143         {
2144             /* Allocate space for the buffer */
2145             CurrentDirectory.Buffer = RtlAllocateHeap(Peb->ProcessHeap,
2146                                                       0,
2147                                                       3 * sizeof(WCHAR) +
2148                                                       sizeof(UNICODE_NULL));
2149             if (!CurrentDirectory.Buffer)
2150             {
2151                 DPRINT1("LDR: LdrpInitializeProcess - unable to allocate current working directory buffer\n");
2152                 // FIXME: And what?
2153             }
2154 
2155             /* Copy the drive of the system root */
2156             RtlMoveMemory(CurrentDirectory.Buffer,
2157                           SharedUserData->NtSystemRoot,
2158                           3 * sizeof(WCHAR));
2159             CurrentDirectory.Buffer[3] = UNICODE_NULL;
2160             CurrentDirectory.Length = 3 * sizeof(WCHAR);
2161             CurrentDirectory.MaximumLength = CurrentDirectory.Length + sizeof(WCHAR);
2162 
2163             FreeCurDir = TRUE;
2164             DPRINT("Using dynamically allocd curdir\n");
2165         }
2166         else
2167         {
2168             /* Use the local buffer */
2169             DPRINT("Using local system root\n");
2170         }
2171     }
2172 
2173     /* Setup Loader Data */
2174     Peb->Ldr = &PebLdr;
2175     InitializeListHead(&PebLdr.InLoadOrderModuleList);
2176     InitializeListHead(&PebLdr.InMemoryOrderModuleList);
2177     InitializeListHead(&PebLdr.InInitializationOrderModuleList);
2178     PebLdr.Length = sizeof(PEB_LDR_DATA);
2179     PebLdr.Initialized = TRUE;
2180 
2181     /* Allocate a data entry for the Image */
2182     LdrpImageEntry = LdrpAllocateDataTableEntry(Peb->ImageBaseAddress);
2183 
2184     /* Set it up */
2185     LdrpImageEntry->EntryPoint = LdrpFetchAddressOfEntryPoint(LdrpImageEntry->DllBase);
2186     LdrpImageEntry->LoadCount = -1;
2187     LdrpImageEntry->EntryPointActivationContext = 0;
2188     LdrpImageEntry->FullDllName = ImageFileName;
2189 
2190     if (IsDotNetImage)
2191         LdrpImageEntry->Flags = LDRP_COR_IMAGE;
2192     else
2193         LdrpImageEntry->Flags = 0;
2194 
2195     /* Check if the name is empty */
2196     if (!ImageFileName.Buffer[0])
2197     {
2198         /* Use the same Base name */
2199         LdrpImageEntry->BaseDllName = LdrpImageEntry->FullDllName;
2200     }
2201     else
2202     {
2203         /* Find the last slash */
2204         Current = ImageFileName.Buffer;
2205         while (*Current)
2206         {
2207             if (*Current++ == '\\')
2208             {
2209                 /* Set this path */
2210                 NtDllName = Current;
2211             }
2212         }
2213 
2214         /* Did we find anything? */
2215         if (!NtDllName)
2216         {
2217             /* Use the same Base name */
2218             LdrpImageEntry->BaseDllName = LdrpImageEntry->FullDllName;
2219         }
2220         else
2221         {
2222             /* Setup the name */
2223             LdrpImageEntry->BaseDllName.Length = (USHORT)((ULONG_PTR)ImageFileName.Buffer + ImageFileName.Length - (ULONG_PTR)NtDllName);
2224             LdrpImageEntry->BaseDllName.MaximumLength = LdrpImageEntry->BaseDllName.Length + sizeof(WCHAR);
2225             LdrpImageEntry->BaseDllName.Buffer = (PWSTR)((ULONG_PTR)ImageFileName.Buffer +
2226                                                      (ImageFileName.Length - LdrpImageEntry->BaseDllName.Length));
2227         }
2228     }
2229 
2230     /* Processing done, insert it */
2231     LdrpInsertMemoryTableEntry(LdrpImageEntry);
2232     LdrpImageEntry->Flags |= LDRP_ENTRY_PROCESSED;
2233 
2234     /* Now add an entry for NTDLL */
2235     NtLdrEntry = LdrpAllocateDataTableEntry(SystemArgument1);
2236     NtLdrEntry->Flags = LDRP_IMAGE_DLL;
2237     NtLdrEntry->EntryPoint = LdrpFetchAddressOfEntryPoint(NtLdrEntry->DllBase);
2238     NtLdrEntry->LoadCount = -1;
2239     NtLdrEntry->EntryPointActivationContext = 0;
2240 
2241     NtLdrEntry->FullDllName.Length = FullPath.Length;
2242     NtLdrEntry->FullDllName.MaximumLength = FullPath.MaximumLength;
2243     NtLdrEntry->FullDllName.Buffer = StringBuffer;
2244     RtlAppendUnicodeStringToString(&NtLdrEntry->FullDllName, &NtDllString);
2245 
2246     NtLdrEntry->BaseDllName.Length = NtDllString.Length;
2247     NtLdrEntry->BaseDllName.MaximumLength = NtDllString.MaximumLength;
2248     NtLdrEntry->BaseDllName.Buffer = NtDllString.Buffer;
2249 
2250     /* Processing done, insert it */
2251     LdrpNtDllDataTableEntry = NtLdrEntry;
2252     LdrpInsertMemoryTableEntry(NtLdrEntry);
2253 
2254     /* Let the world know */
2255     if (ShowSnaps)
2256     {
2257         DPRINT1("LDR: NEW PROCESS\n");
2258         DPRINT1("     Image Path: %wZ (%wZ)\n", &LdrpImageEntry->FullDllName, &LdrpImageEntry->BaseDllName);
2259         DPRINT1("     Current Directory: %wZ\n", &CurrentDirectory);
2260         DPRINT1("     Search Path: %wZ\n", &LdrpDefaultPath);
2261     }
2262 
2263     /* Link the Init Order List */
2264     InsertHeadList(&Peb->Ldr->InInitializationOrderModuleList,
2265                    &LdrpNtDllDataTableEntry->InInitializationOrderLinks);
2266 
2267     /* Initialize Wine's active context implementation for the current process */
2268     actctx_init(&OldShimData);
2269 
2270     /* Set the current directory */
2271     Status = RtlSetCurrentDirectory_U(&CurrentDirectory);
2272     if (!NT_SUCCESS(Status))
2273     {
2274         /* We failed, check if we should free it */
2275         if (FreeCurDir) RtlFreeUnicodeString(&CurrentDirectory);
2276 
2277         /* Set it to the NT Root */
2278         CurrentDirectory = NtSystemRoot;
2279         RtlSetCurrentDirectory_U(&CurrentDirectory);
2280     }
2281     else
2282     {
2283         /* We're done with it, free it */
2284         if (FreeCurDir) RtlFreeUnicodeString(&CurrentDirectory);
2285     }
2286 
2287     /* Check if we should look for a .local file */
2288     if (ProcessParameters && !(ProcessParameters->Flags & RTL_USER_PROCESS_PARAMETERS_LOCAL_DLL_PATH))
2289     {
2290         LdrpInitializeDotLocalSupport(ProcessParameters);
2291     }
2292 
2293     /* Check if the Application Verifier was enabled */
2294     if (Peb->NtGlobalFlag & FLG_APPLICATION_VERIFIER)
2295     {
2296         Status = AVrfInitializeVerifier();
2297         if (!NT_SUCCESS(Status))
2298         {
2299             DPRINT1("LDR: AVrfInitializeVerifier failed (ntstatus 0x%x)\n", Status);
2300             return Status;
2301         }
2302 
2303     }
2304 
2305     if (IsDotNetImage)
2306     {
2307         /* FIXME */
2308         DPRINT1("We don't support .NET applications yet\n");
2309     }
2310 
2311     if (NtHeader->OptionalHeader.Subsystem == IMAGE_SUBSYSTEM_WINDOWS_GUI ||
2312         NtHeader->OptionalHeader.Subsystem == IMAGE_SUBSYSTEM_WINDOWS_CUI)
2313     {
2314         PVOID Kernel32BaseAddress;
2315         PVOID FunctionAddress;
2316 
2317         Status = LdrLoadDll(NULL, NULL, &Kernel32String, &Kernel32BaseAddress);
2318 
2319         if (!NT_SUCCESS(Status))
2320         {
2321             if (ShowSnaps)
2322                 DPRINT1("LDR: Unable to load %wZ, Status=0x%08lx\n", &Kernel32String, Status);
2323             return Status;
2324         }
2325 
2326         Status = LdrGetProcedureAddress(Kernel32BaseAddress,
2327                                         &BaseProcessInitPostImportName,
2328                                         0,
2329                                         &FunctionAddress);
2330 
2331         if (!NT_SUCCESS(Status))
2332         {
2333             if (ShowSnaps)
2334                 DPRINT1("LDR: Unable to find post-import process init function, Status=0x%08lx\n", Status);
2335             return Status;
2336         }
2337         Kernel32ProcessInitPostImportFunction = FunctionAddress;
2338 
2339         Status = LdrGetProcedureAddress(Kernel32BaseAddress,
2340                                         &BaseQueryModuleDataName,
2341                                         0,
2342                                         &FunctionAddress);
2343 
2344         if (!NT_SUCCESS(Status))
2345         {
2346             if (ShowSnaps)
2347                 DPRINT1("LDR: Unable to find BaseQueryModuleData, Status=0x%08lx\n", Status);
2348             return Status;
2349         }
2350         Kernel32BaseQueryModuleData = FunctionAddress;
2351     }
2352 
2353     /* Walk the IAT and load all the DLLs */
2354     ImportStatus = LdrpWalkImportDescriptor(LdrpDefaultPath.Buffer, LdrpImageEntry);
2355 
2356     /* Check if relocation is needed */
2357     if (Peb->ImageBaseAddress != (PVOID)NtHeader->OptionalHeader.ImageBase)
2358     {
2359         DPRINT1("LDR: Performing EXE relocation\n");
2360 
2361         /* Change the protection to prepare for relocation */
2362         ViewBase = Peb->ImageBaseAddress;
2363         Status = LdrpSetProtection(ViewBase, FALSE);
2364         if (!NT_SUCCESS(Status)) return Status;
2365 
2366         /* Do the relocation */
2367         Status = LdrRelocateImageWithBias(ViewBase,
2368                                           0LL,
2369                                           NULL,
2370                                           STATUS_SUCCESS,
2371                                           STATUS_CONFLICTING_ADDRESSES,
2372                                           STATUS_INVALID_IMAGE_FORMAT);
2373         if (!NT_SUCCESS(Status))
2374         {
2375             DPRINT1("LdrRelocateImageWithBias() failed\n");
2376             return Status;
2377         }
2378 
2379         /* Check if a start context was provided */
2380         if (Context)
2381         {
2382             DPRINT1("WARNING: Relocated EXE Context");
2383             UNIMPLEMENTED; // We should support this
2384             return STATUS_INVALID_IMAGE_FORMAT;
2385         }
2386 
2387         /* Restore the protection */
2388         Status = LdrpSetProtection(ViewBase, TRUE);
2389         if (!NT_SUCCESS(Status)) return Status;
2390     }
2391 
2392     /* Lock the DLLs */
2393     ListHead = &Peb->Ldr->InLoadOrderModuleList;
2394     NextEntry = ListHead->Flink;
2395     while (ListHead != NextEntry)
2396     {
2397         NtLdrEntry = CONTAINING_RECORD(NextEntry, LDR_DATA_TABLE_ENTRY, InLoadOrderLinks);
2398         NtLdrEntry->LoadCount = -1;
2399         NextEntry = NextEntry->Flink;
2400     }
2401 
2402     /* Phase 0 is done */
2403     LdrpLdrDatabaseIsSetup = TRUE;
2404 
2405     /* Check whether all static imports were properly loaded and return here */
2406     if (!NT_SUCCESS(ImportStatus)) return ImportStatus;
2407 
2408 #if (DLL_EXPORT_VERSION >= _WIN32_WINNT_VISTA)
2409     /* Initialize the keyed event for condition variables */
2410     RtlpInitializeKeyedEvent();
2411 #endif
2412 
2413     /* Initialize TLS */
2414     Status = LdrpInitializeTls();
2415     if (!NT_SUCCESS(Status))
2416     {
2417         DPRINT1("LDR: LdrpProcessInitialization failed to initialize TLS slots; status %x\n",
2418                 Status);
2419         return Status;
2420     }
2421 
2422     /* FIXME Mark the DLL Ranges for Stack Traces later */
2423 
2424     /* Notify the debugger now */
2425     if (Peb->BeingDebugged)
2426     {
2427         /* Break */
2428         DbgBreakPoint();
2429 
2430         /* Update show snaps again */
2431         ShowSnaps = Peb->NtGlobalFlag & FLG_SHOW_LDR_SNAPS;
2432     }
2433 
2434     /* Validate the Image for MP Usage */
2435     if (LdrpNumberOfProcessors > 1) LdrpValidateImageForMp(LdrpImageEntry);
2436 
2437     /* Check NX options and set them */
2438     if (SharedUserData->NXSupportPolicy == NX_SUPPORT_POLICY_ALWAYSON)
2439     {
2440         ExecuteOptions = MEM_EXECUTE_OPTION_DISABLE |
2441                          MEM_EXECUTE_OPTION_DISABLE_THUNK_EMULATION |
2442                          MEM_EXECUTE_OPTION_PERMANENT;
2443     }
2444     else if (SharedUserData->NXSupportPolicy == NX_SUPPORT_POLICY_ALWAYSOFF)
2445     {
2446         ExecuteOptions = MEM_EXECUTE_OPTION_ENABLE | MEM_EXECUTE_OPTION_PERMANENT;
2447     }
2448     Status = NtSetInformationProcess(NtCurrentProcess(),
2449                                      ProcessExecuteFlags,
2450                                      &ExecuteOptions,
2451                                      sizeof(ExecuteOptions));
2452     if (!NT_SUCCESS(Status))
2453     {
2454         DPRINT1("LDR: Could not set process execute flags 0x%x; status %x\n",
2455                 ExecuteOptions, Status);
2456     }
2457 
2458     // FIXME: Should be done by Application Compatibility features,
2459     // by reading the registry, etc...
2460     // For now, this is the old code from ntdll!RtlGetVersion().
2461     RtlInitEmptyUnicodeString(&Peb->CSDVersion, NULL, 0);
2462     if (((Peb->OSCSDVersion >> 8) & 0xFF) != 0)
2463     {
2464         WCHAR szCSDVersion[128];
2465         LONG i;
2466         USHORT Length = (USHORT)ARRAYSIZE(szCSDVersion) - 1;
2467         i = _snwprintf(szCSDVersion, Length,
2468                        L"Service Pack %d",
2469                        ((Peb->OSCSDVersion >> 8) & 0xFF));
2470         if (i < 0)
2471         {
2472             /* Null-terminate if it was overflowed */
2473             szCSDVersion[Length] = UNICODE_NULL;
2474         }
2475 
2476         Length *= sizeof(WCHAR);
2477         Peb->CSDVersion.Buffer = RtlAllocateHeap(RtlGetProcessHeap(),
2478                                                  0,
2479                                                  Length + sizeof(UNICODE_NULL));
2480         if (Peb->CSDVersion.Buffer)
2481         {
2482             Peb->CSDVersion.Length = Length;
2483             Peb->CSDVersion.MaximumLength = Length + sizeof(UNICODE_NULL);
2484 
2485             RtlCopyMemory(Peb->CSDVersion.Buffer,
2486                           szCSDVersion,
2487                           Peb->CSDVersion.MaximumLength);
2488             Peb->CSDVersion.Buffer[Peb->CSDVersion.Length / sizeof(WCHAR)] = UNICODE_NULL;
2489         }
2490     }
2491 
2492     /* Check if we had Shim Data */
2493     if (OldShimData)
2494     {
2495         /* Load the Shim Engine */
2496         Peb->AppCompatInfo = NULL;
2497         LdrpLoadShimEngine(OldShimData, &ImagePathName, OldShimData);
2498     }
2499     else
2500     {
2501         /* Check for Application Compatibility Goo */
2502         //LdrQueryApplicationCompatibilityGoo(hKey);
2503         DPRINT("Querying app compat hacks is missing!\n");
2504     }
2505 
2506     /*
2507      * FIXME: Check for special images, SecuROM, SafeDisc and other NX-
2508      * incompatible images.
2509      */
2510 
2511     /* Now call the Init Routines */
2512     Status = LdrpRunInitializeRoutines(Context);
2513     if (!NT_SUCCESS(Status))
2514     {
2515         DPRINT1("LDR: LdrpProcessInitialization failed running initialization routines; status %x\n",
2516                 Status);
2517         return Status;
2518     }
2519 
2520     /* Notify Shim Engine */
2521     if (g_ShimsEnabled)
2522     {
2523         VOID(NTAPI *SE_InstallAfterInit)(PUNICODE_STRING, PVOID);
2524         SE_InstallAfterInit = RtlDecodeSystemPointer(g_pfnSE_InstallAfterInit);
2525         SE_InstallAfterInit(&ImagePathName, OldShimData);
2526     }
2527 
2528     /* Check if we have a user-defined Post Process Routine */
2529     if (NT_SUCCESS(Status) && Peb->PostProcessInitRoutine)
2530     {
2531         /* Call it */
2532         Peb->PostProcessInitRoutine();
2533     }
2534 
2535     /* Close the key if we have one opened */
2536     if (OptionsKey) NtClose(OptionsKey);
2537 
2538     /* Return status */
2539     return Status;
2540 }
2541 
2542 VOID
2543 NTAPI
2544 LdrpInitFailure(NTSTATUS Status)
2545 {
2546     ULONG Response;
2547     PPEB Peb = NtCurrentPeb();
2548 
2549     /* Print a debug message */
2550     DPRINT1("LDR: Process initialization failure for %wZ; NTSTATUS = %08lx\n",
2551             &Peb->ProcessParameters->ImagePathName, Status);
2552 
2553     /* Raise a hard error */
2554     if (!LdrpFatalHardErrorCount)
2555     {
2556         ZwRaiseHardError(STATUS_APP_INIT_FAILURE, 1, 0, (PULONG_PTR)&Status, OptionOk, &Response);
2557     }
2558 }
2559 
2560 VOID
2561 NTAPI
2562 LdrpInit(PCONTEXT Context,
2563          PVOID SystemArgument1,
2564          PVOID SystemArgument2)
2565 {
2566     LARGE_INTEGER Timeout;
2567     PTEB Teb = NtCurrentTeb();
2568     NTSTATUS Status, LoaderStatus = STATUS_SUCCESS;
2569     MEMORY_BASIC_INFORMATION MemoryBasicInfo;
2570     PPEB Peb = NtCurrentPeb();
2571 
2572     DPRINT("LdrpInit() %p/%p\n",
2573         NtCurrentTeb()->RealClientId.UniqueProcess,
2574         NtCurrentTeb()->RealClientId.UniqueThread);
2575 
2576 #ifdef _WIN64
2577     /* Set the SList header usage */
2578     RtlpUse16ByteSLists = SharedUserData->ProcessorFeatures[PF_COMPARE_EXCHANGE128];
2579 #endif /* _WIN64 */
2580 
2581     /* Check if we have a deallocation stack */
2582     if (!Teb->DeallocationStack)
2583     {
2584         /* We don't, set one */
2585         Status = NtQueryVirtualMemory(NtCurrentProcess(),
2586                                       Teb->NtTib.StackLimit,
2587                                       MemoryBasicInformation,
2588                                       &MemoryBasicInfo,
2589                                       sizeof(MEMORY_BASIC_INFORMATION),
2590                                       NULL);
2591         if (!NT_SUCCESS(Status))
2592         {
2593             /* Fail */
2594             LdrpInitFailure(Status);
2595             RtlRaiseStatus(Status);
2596             return;
2597         }
2598 
2599         /* Set the stack */
2600         Teb->DeallocationStack = MemoryBasicInfo.AllocationBase;
2601     }
2602 
2603     /* Now check if the process is already being initialized */
2604     while (_InterlockedCompareExchange(&LdrpProcessInitialized,
2605                                       1,
2606                                       0) == 1)
2607     {
2608         /* Set the timeout to 30 milliseconds */
2609         Timeout.QuadPart = Int32x32To64(30, -10000);
2610 
2611         /* Make sure the status hasn't changed */
2612         while (LdrpProcessInitialized == 1)
2613         {
2614             /* Do the wait */
2615             ZwDelayExecution(FALSE, &Timeout);
2616         }
2617     }
2618 
2619     /* Check if we have already setup LDR data */
2620     if (!Peb->Ldr)
2621     {
2622         /* Setup the Loader Lock */
2623         Peb->LoaderLock = &LdrpLoaderLock;
2624 
2625         /* Let other code know we're initializing */
2626         LdrpInLdrInit = TRUE;
2627 
2628         /* Protect with SEH */
2629         _SEH2_TRY
2630         {
2631             /* Initialize the Process */
2632             LoaderStatus = LdrpInitializeProcess(Context,
2633                                                  SystemArgument1);
2634 
2635             /* Check for success and if MinimumStackCommit was requested */
2636             if (NT_SUCCESS(LoaderStatus) && Peb->MinimumStackCommit)
2637             {
2638                 /* Enforce the limit */
2639                 //LdrpTouchThreadStack(Peb->MinimumStackCommit);
2640                 UNIMPLEMENTED;
2641             }
2642         }
2643         _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
2644         {
2645             /* Fail with the SEH error */
2646             LoaderStatus = _SEH2_GetExceptionCode();
2647         }
2648         _SEH2_END;
2649 
2650         /* We're not initializing anymore */
2651         LdrpInLdrInit = FALSE;
2652 
2653         /* Check if init worked */
2654         if (NT_SUCCESS(LoaderStatus))
2655         {
2656             /* Set the process as Initialized */
2657             _InterlockedIncrement(&LdrpProcessInitialized);
2658         }
2659     }
2660     else
2661     {
2662         /* Loader data is there... is this a fork() ? */
2663         if(Peb->InheritedAddressSpace)
2664         {
2665             /* Handle the fork() */
2666             //LoaderStatus = LdrpForkProcess();
2667             LoaderStatus = STATUS_NOT_IMPLEMENTED;
2668             UNIMPLEMENTED;
2669         }
2670         else
2671         {
2672             /* This is a new thread initializing */
2673             LdrpInitializeThread(Context);
2674         }
2675     }
2676 
2677     /* All done, test alert the thread */
2678     NtTestAlert();
2679 
2680     /* Return */
2681     if (!NT_SUCCESS(LoaderStatus))
2682     {
2683         /* Fail */
2684         LdrpInitFailure(LoaderStatus);
2685         RtlRaiseStatus(LoaderStatus);
2686     }
2687 }
2688 
2689 /* EOF */
2690