xref: /reactos/ntoskrnl/ke/i386/thrdini.c (revision 92a36b36)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         ReactOS kernel
4  * FILE:            ntoskrnl/ke/i386/thrdini.c
5  * PURPOSE:         i386 Thread Context Creation
6  * PROGRAMMER:      Alex Ionescu (alex@relsoft.net)
7  */
8 
9 /* INCLUDES ******************************************************************/
10 
11 #include <ntoskrnl.h>
12 #define NDEBUG
13 #include <debug.h>
14 
15 typedef struct _KSWITCHFRAME
16 {
17     PVOID ExceptionList;
18     BOOLEAN ApcBypassDisable;
19     PVOID RetAddr;
20 } KSWITCHFRAME, *PKSWITCHFRAME;
21 
22 typedef struct _KSTART_FRAME
23 {
24     PKSYSTEM_ROUTINE SystemRoutine;
25     PKSTART_ROUTINE StartRoutine;
26     PVOID StartContext;
27     BOOLEAN UserThread;
28 } KSTART_FRAME, *PKSTART_FRAME;
29 
30 typedef struct _KUINIT_FRAME
31 {
32     KSWITCHFRAME CtxSwitchFrame;
33     KSTART_FRAME StartFrame;
34     KTRAP_FRAME TrapFrame;
35     FX_SAVE_AREA FxSaveArea;
36 } KUINIT_FRAME, *PKUINIT_FRAME;
37 
38 typedef struct _KKINIT_FRAME
39 {
40     KSWITCHFRAME CtxSwitchFrame;
41     KSTART_FRAME StartFrame;
42     FX_SAVE_AREA FxSaveArea;
43 } KKINIT_FRAME, *PKKINIT_FRAME;
44 
45 VOID
46 FASTCALL
47 KiSwitchThreads(
48     IN PKTHREAD OldThread,
49     IN PKTHREAD NewThread
50 );
51 
52 VOID
53 FASTCALL
54 KiRetireDpcListInDpcStack(
55     IN PKPRCB Prcb,
56     IN PVOID DpcStack
57 );
58 
59 /* FUNCTIONS *****************************************************************/
60 
61 VOID
62 NTAPI
63 KiThreadStartup(VOID)
64 {
65     PKTRAP_FRAME TrapFrame;
66     PKSTART_FRAME StartFrame;
67     PKUINIT_FRAME InitFrame;
68 
69     /* Get the start and trap frames */
70     InitFrame = KeGetCurrentThread()->KernelStack;
71     StartFrame = &InitFrame->StartFrame;
72     TrapFrame = &InitFrame->TrapFrame;
73 
74     /* Lower to APC level */
75     KfLowerIrql(APC_LEVEL);
76 
77     /* Call the system routine */
78     StartFrame->SystemRoutine(StartFrame->StartRoutine, StartFrame->StartContext);
79 
80     /* If we returned, we better be a user thread */
81     if (!StartFrame->UserThread)
82     {
83         KeBugCheck(NO_USER_MODE_CONTEXT);
84     }
85 
86     /* Exit to user-mode */
87     KiServiceExit2(TrapFrame);
88 }
89 
90 VOID
91 NTAPI
92 KiInitializeContextThread(IN PKTHREAD Thread,
93                           IN PKSYSTEM_ROUTINE SystemRoutine,
94                           IN PKSTART_ROUTINE StartRoutine,
95                           IN PVOID StartContext,
96                           IN PCONTEXT ContextPointer)
97 {
98     PFX_SAVE_AREA FxSaveArea;
99     PFXSAVE_FORMAT FxSaveFormat;
100     PKSTART_FRAME StartFrame;
101     PKSWITCHFRAME CtxSwitchFrame;
102     PKTRAP_FRAME TrapFrame;
103     CONTEXT LocalContext;
104     PCONTEXT Context = NULL;
105     ULONG ContextFlags;
106 
107     /* Check if this is a With-Context Thread */
108     if (ContextPointer)
109     {
110         /* Set up the Initial Frame */
111         PKUINIT_FRAME InitFrame;
112         InitFrame = (PKUINIT_FRAME)((ULONG_PTR)Thread->InitialStack -
113                                     sizeof(KUINIT_FRAME));
114 
115         /* Copy over the context we got */
116         RtlCopyMemory(&LocalContext, ContextPointer, sizeof(CONTEXT));
117         Context = &LocalContext;
118         ContextFlags = CONTEXT_CONTROL;
119 
120         /* Zero out the trap frame and save area */
121         RtlZeroMemory(&InitFrame->TrapFrame,
122                       KTRAP_FRAME_LENGTH + sizeof(FX_SAVE_AREA));
123 
124         /* Setup the Fx Area */
125         FxSaveArea = &InitFrame->FxSaveArea;
126 
127         /* Check if we support FXsr */
128         if (KeI386FxsrPresent)
129         {
130             /* Get the FX Save Format Area */
131             FxSaveFormat = (PFXSAVE_FORMAT)Context->ExtendedRegisters;
132 
133             /* Set an initial state */
134             FxSaveFormat->ControlWord = 0x27F;
135             FxSaveFormat->StatusWord = 0;
136             FxSaveFormat->TagWord = 0;
137             FxSaveFormat->ErrorOffset = 0;
138             FxSaveFormat->ErrorSelector = 0;
139             FxSaveFormat->DataOffset = 0;
140             FxSaveFormat->DataSelector = 0;
141             FxSaveFormat->MXCsr = 0x1F80;
142         }
143         else
144         {
145             /* Setup the regular save area */
146             Context->FloatSave.ControlWord = 0x27F;
147             Context->FloatSave.StatusWord = 0;
148             Context->FloatSave.TagWord = -1;
149             Context->FloatSave.ErrorOffset = 0;
150             Context->FloatSave.ErrorSelector = 0;
151             Context->FloatSave.DataOffset =0;
152             Context->FloatSave.DataSelector = 0;
153         }
154 
155         /* Set an intial NPX State */
156         Context->FloatSave.Cr0NpxState = 0;
157         FxSaveArea->Cr0NpxState = 0;
158         FxSaveArea->NpxSavedCpu = 0;
159 
160         /* Now set the context flags depending on XMM support */
161         ContextFlags |= (KeI386FxsrPresent) ? CONTEXT_EXTENDED_REGISTERS : CONTEXT_FLOATING_POINT;
162 
163         /* Set the Thread's NPX State */
164         Thread->NpxState = NPX_STATE_NOT_LOADED;
165         Thread->Header.NpxIrql = PASSIVE_LEVEL;
166 
167         /* Disable any debug registers */
168         Context->ContextFlags &= ~CONTEXT_DEBUG_REGISTERS;
169 
170         /* Setup the Trap Frame */
171         TrapFrame = &InitFrame->TrapFrame;
172 
173         /* Set up a trap frame from the context. */
174         KeContextToTrapFrame(Context,
175                              NULL,
176                              TrapFrame,
177                              Context->ContextFlags | ContextFlags,
178                              UserMode);
179 
180         /* Set SS, DS, ES's RPL Mask properly */
181         TrapFrame->HardwareSegSs |= RPL_MASK;
182         TrapFrame->SegDs |= RPL_MASK;
183         TrapFrame->SegEs |= RPL_MASK;
184         TrapFrame->Dr7 = 0;
185 
186         /* Set the debug mark */
187         TrapFrame->DbgArgMark = 0xBADB0D00;
188 
189         /* Set the previous mode as user */
190         TrapFrame->PreviousPreviousMode = UserMode;
191 
192         /* Terminate the Exception Handler List */
193         TrapFrame->ExceptionList = EXCEPTION_CHAIN_END;
194 
195         /* Setup the Stack for KiThreadStartup and Context Switching */
196         StartFrame = &InitFrame->StartFrame;
197         CtxSwitchFrame = &InitFrame->CtxSwitchFrame;
198 
199         /* Tell the thread it will run in User Mode */
200         Thread->PreviousMode = UserMode;
201 
202         /* Tell KiThreadStartup of that too */
203         StartFrame->UserThread = TRUE;
204     }
205     else
206     {
207         /* Set up the Initial Frame for the system thread */
208         PKKINIT_FRAME InitFrame;
209         InitFrame = (PKKINIT_FRAME)((ULONG_PTR)Thread->InitialStack -
210                                     sizeof(KKINIT_FRAME));
211 
212         /* Setup the Fx Area */
213         FxSaveArea = &InitFrame->FxSaveArea;
214         RtlZeroMemory(FxSaveArea, sizeof(FX_SAVE_AREA));
215 
216         /* Check if we have Fxsr support */
217         if (KeI386FxsrPresent)
218         {
219             /* Set the stub FX area */
220             FxSaveArea->U.FxArea.ControlWord = 0x27F;
221             FxSaveArea->U.FxArea.MXCsr = 0x1F80;
222         }
223         else
224         {
225             /* Set the stub FN area */
226             FxSaveArea->U.FnArea.ControlWord = 0x27F;
227             FxSaveArea->U.FnArea.TagWord = -1;
228         }
229 
230         /* No NPX State */
231         Thread->NpxState = NPX_STATE_NOT_LOADED;
232 
233         /* Setup the Stack for KiThreadStartup and Context Switching */
234         StartFrame = &InitFrame->StartFrame;
235         CtxSwitchFrame = &InitFrame->CtxSwitchFrame;
236 
237         /* Tell the thread it will run in Kernel Mode */
238         Thread->PreviousMode = KernelMode;
239 
240         /* Tell KiThreadStartup of that too */
241         StartFrame->UserThread = FALSE;
242     }
243 
244     /* Now setup the remaining data for KiThreadStartup */
245     StartFrame->StartContext = StartContext;
246     StartFrame->StartRoutine = StartRoutine;
247     StartFrame->SystemRoutine = SystemRoutine;
248 
249     /* And set up the Context Switch Frame */
250     CtxSwitchFrame->RetAddr = KiThreadStartup;
251     CtxSwitchFrame->ApcBypassDisable = TRUE;
252     CtxSwitchFrame->ExceptionList = EXCEPTION_CHAIN_END;
253 
254     /* Save back the new value of the kernel stack. */
255     Thread->KernelStack = (PVOID)CtxSwitchFrame;
256 }
257 
258 DECLSPEC_NORETURN
259 VOID
260 KiIdleLoop(VOID)
261 {
262     PKPRCB Prcb = KeGetCurrentPrcb();
263     PKTHREAD OldThread, NewThread;
264 
265     /* Now loop forever */
266     while (TRUE)
267     {
268         /* Start of the idle loop: disable interrupts */
269         _enable();
270         YieldProcessor();
271         YieldProcessor();
272         _disable();
273 
274         /* Check for pending timers, pending DPCs, or pending ready threads */
275         if ((Prcb->DpcData[0].DpcQueueDepth) ||
276             (Prcb->TimerRequest) ||
277             (Prcb->DeferredReadyListHead.Next))
278         {
279             /* Quiesce the DPC software interrupt */
280             HalClearSoftwareInterrupt(DISPATCH_LEVEL);
281 
282             /* Handle it */
283             KiRetireDpcList(Prcb);
284         }
285 
286         /* Check if a new thread is scheduled for execution */
287         if (Prcb->NextThread)
288         {
289             /* Enable interrupts */
290             _enable();
291 
292             /* Capture current thread data */
293             OldThread = Prcb->CurrentThread;
294             NewThread = Prcb->NextThread;
295 
296             /* Set new thread data */
297             Prcb->NextThread = NULL;
298             Prcb->CurrentThread = NewThread;
299 
300             /* The thread is now running */
301             NewThread->State = Running;
302 
303 #ifdef CONFIG_SMP
304             /* Do the swap at SYNCH_LEVEL */
305             KfRaiseIrql(SYNCH_LEVEL);
306 #endif
307 
308             /* Switch away from the idle thread */
309             KiSwapContext(APC_LEVEL, OldThread);
310 
311 #ifdef CONFIG_SMP
312             /* Go back to DISPATCH_LEVEL */
313             KeLowerIrql(DISPATCH_LEVEL);
314 #endif
315         }
316         else
317         {
318             /* Continue staying idle. Note the HAL returns with interrupts on */
319             Prcb->PowerState.IdleFunction(&Prcb->PowerState);
320         }
321     }
322 }
323 
324 BOOLEAN
325 FASTCALL
326 KiSwapContextExit(IN PKTHREAD OldThread,
327                   IN PKSWITCHFRAME SwitchFrame)
328 {
329     PKIPCR Pcr = (PKIPCR)KeGetPcr();
330     PKPROCESS OldProcess, NewProcess;
331     PKTHREAD NewThread;
332 
333     /* We are on the new thread stack now */
334     NewThread = Pcr->PrcbData.CurrentThread;
335 
336     /* Now we are the new thread. Check if it's in a new process */
337     OldProcess = OldThread->ApcState.Process;
338     NewProcess = NewThread->ApcState.Process;
339     if (OldProcess != NewProcess)
340     {
341         /* Check if there is a different LDT */
342         if (*(PULONGLONG)&OldProcess->LdtDescriptor != *(PULONGLONG)&NewProcess->LdtDescriptor)
343         {
344             if (NewProcess->LdtDescriptor.LimitLow)
345             {
346                 KeSetGdtSelector(KGDT_LDT,
347                                  ((PULONG)&NewProcess->LdtDescriptor)[0],
348                                  ((PULONG)&NewProcess->LdtDescriptor)[1]);
349                 Ke386SetLocalDescriptorTable(KGDT_LDT);
350             }
351             else
352             {
353                 Ke386SetLocalDescriptorTable(0);
354             }
355         }
356 
357         /* Switch address space and flush TLB */
358         __writecr3(NewProcess->DirectoryTableBase[0]);
359     }
360 
361     /* Clear GS */
362     Ke386SetGs(0);
363 
364     /* Set the TEB */
365     KiSetTebBase((PKPCR)Pcr, &NewThread->Teb->NtTib);
366 
367     /* Set new TSS fields */
368     Pcr->TSS->Esp0 = (ULONG_PTR)NewThread->InitialStack;
369     if (!((KeGetTrapFrame(NewThread))->EFlags & EFLAGS_V86_MASK))
370     {
371         Pcr->TSS->Esp0 -= sizeof(KTRAP_FRAME) - FIELD_OFFSET(KTRAP_FRAME, V86Es);
372     }
373     Pcr->TSS->Esp0 -= NPX_FRAME_LENGTH;
374     Pcr->TSS->IoMapBase = NewProcess->IopmOffset;
375 
376     /* Increase thread context switches */
377     NewThread->ContextSwitches++;
378 
379     /* Load data from switch frame */
380     Pcr->NtTib.ExceptionList = SwitchFrame->ExceptionList;
381 
382     /* DPCs shouldn't be active */
383     if (Pcr->PrcbData.DpcRoutineActive)
384     {
385         /* Crash the machine */
386         KeBugCheckEx(ATTEMPTED_SWITCH_FROM_DPC,
387                      (ULONG_PTR)OldThread,
388                      (ULONG_PTR)NewThread,
389                      (ULONG_PTR)OldThread->InitialStack,
390                      0);
391     }
392 
393     /* Kernel APCs may be pending */
394     if (NewThread->ApcState.KernelApcPending)
395     {
396         /* Are APCs enabled? */
397         if (!NewThread->SpecialApcDisable)
398         {
399             /* Request APC delivery */
400             if (SwitchFrame->ApcBypassDisable)
401                 HalRequestSoftwareInterrupt(APC_LEVEL);
402             else
403                 return TRUE;
404         }
405     }
406 
407     /* Return stating that no kernel APCs are pending*/
408     return FALSE;
409 }
410 
411 VOID
412 FASTCALL
413 KiSwapContextEntry(IN PKSWITCHFRAME SwitchFrame,
414                    IN ULONG_PTR OldThreadAndApcFlag)
415 {
416     PKIPCR Pcr = (PKIPCR)KeGetPcr();
417     PKTHREAD OldThread, NewThread;
418     ULONG Cr0, NewCr0;
419 
420     /* Save APC bypass disable */
421     SwitchFrame->ApcBypassDisable = OldThreadAndApcFlag & 3;
422     SwitchFrame->ExceptionList = Pcr->NtTib.ExceptionList;
423 
424     /* Increase context switch count and check if tracing is enabled */
425     Pcr->ContextSwitches++;
426     if (Pcr->PerfGlobalGroupMask)
427     {
428         /* We don't support this yet on x86 either */
429         DPRINT1("WMI Tracing not supported\n");
430         ASSERT(FALSE);
431     }
432 
433     /* Get thread pointers */
434     OldThread = (PKTHREAD)(OldThreadAndApcFlag & ~3);
435     NewThread = Pcr->PrcbData.CurrentThread;
436 
437     /* Get the old thread and set its kernel stack */
438     OldThread->KernelStack = SwitchFrame;
439 
440     /* Set swapbusy to false for the new thread */
441     NewThread->SwapBusy = FALSE;
442 
443     /* ISRs can change FPU state, so disable interrupts while checking */
444     _disable();
445 
446     /* Get current and new CR0 and check if they've changed */
447     Cr0 = __readcr0();
448     NewCr0 = NewThread->NpxState |
449              (Cr0 & ~(CR0_MP | CR0_EM | CR0_TS)) |
450              KiGetThreadNpxArea(NewThread)->Cr0NpxState;
451     if (Cr0 != NewCr0)  __writecr0(NewCr0);
452 
453     /* Now enable interrupts and do the switch */
454     _enable();
455     KiSwitchThreads(OldThread, NewThread->KernelStack);
456 }
457 
458 VOID
459 NTAPI
460 KiDispatchInterrupt(VOID)
461 {
462     PKIPCR Pcr = (PKIPCR)KeGetPcr();
463     PKPRCB Prcb = &Pcr->PrcbData;
464     PVOID OldHandler;
465     PKTHREAD NewThread, OldThread;
466 
467     /* Disable interrupts */
468     _disable();
469 
470     /* Check for pending timers, pending DPCs, or pending ready threads */
471     if ((Prcb->DpcData[0].DpcQueueDepth) ||
472         (Prcb->TimerRequest) ||
473         (Prcb->DeferredReadyListHead.Next))
474     {
475         /* Switch to safe execution context */
476         OldHandler = Pcr->NtTib.ExceptionList;
477         Pcr->NtTib.ExceptionList = EXCEPTION_CHAIN_END;
478 
479         /* Retire DPCs while under the DPC stack */
480         KiRetireDpcListInDpcStack(Prcb, Prcb->DpcStack);
481 
482         /* Restore context */
483         Pcr->NtTib.ExceptionList = OldHandler;
484     }
485 
486     /* Re-enable interrupts */
487     _enable();
488 
489     /* Check for quantum end */
490     if (Prcb->QuantumEnd)
491     {
492         /* Handle quantum end */
493         Prcb->QuantumEnd = FALSE;
494         KiQuantumEnd();
495     }
496     else if (Prcb->NextThread)
497     {
498         /* Acquire the PRCB lock */
499         KiAcquirePrcbLock(Prcb);
500 
501         /* Capture current thread data */
502         OldThread = Prcb->CurrentThread;
503         NewThread = Prcb->NextThread;
504 
505         /* Set new thread data */
506         Prcb->NextThread = NULL;
507         Prcb->CurrentThread = NewThread;
508 
509         /* The thread is now running */
510         NewThread->State = Running;
511         OldThread->WaitReason = WrDispatchInt;
512 
513         /* Make the old thread ready */
514         KxQueueReadyThread(OldThread, Prcb);
515 
516         /* Swap to the new thread */
517         KiSwapContext(APC_LEVEL, OldThread);
518     }
519 }
520 
521 
522 /* EOF */
523