xref: /reactos/ntoskrnl/ke/amd64/stubs.c (revision fbdff437)
1 /*
2  * PROJECT:         ReactOS Kernel
3  * LICENSE:         GPL - See COPYING in the top level directory
4  * PURPOSE:         stubs
5  * PROGRAMMERS:     Timo Kreuzer (timo.kreuzer@reactos.org)
6  */
7 
8 /* INCLUDES ******************************************************************/
9 
10 #include <ntoskrnl.h>
11 #include <fltkernel.h>
12 
13 #define NDEBUG
14 #include <debug.h>
15 
16 ULONG ProcessCount;
17 BOOLEAN CcPfEnablePrefetcher;
18 SIZE_T KeXStateLength = sizeof(XSAVE_FORMAT);
19 
20 VOID
21 KiRetireDpcListInDpcStack(
22     PKPRCB Prcb,
23     PVOID DpcStack);
24 
25 NTSTATUS
26 KiConvertToGuiThread(
27     VOID);
28 
29 VOID
30 NTAPI
31 KiDpcInterruptHandler(VOID)
32 {
33     PKPRCB Prcb = KeGetCurrentPrcb();
34     PKTHREAD NewThread, OldThread;
35     KIRQL OldIrql;
36 
37     /* Raise to DISPATCH_LEVEL */
38     OldIrql = KfRaiseIrql(DISPATCH_LEVEL);
39 
40     /* Send an EOI */
41     KiSendEOI();
42 
43     /* Check for pending timers, pending DPCs, or pending ready threads */
44     if ((Prcb->DpcData[0].DpcQueueDepth) ||
45         (Prcb->TimerRequest) ||
46         (Prcb->DeferredReadyListHead.Next))
47     {
48         /* Retire DPCs while under the DPC stack */
49         KiRetireDpcListInDpcStack(Prcb, Prcb->DpcStack);
50     }
51 
52     /* Enable interrupts */
53     _enable();
54 
55     /* Check for quantum end */
56     if (Prcb->QuantumEnd)
57     {
58         /* Handle quantum end */
59         Prcb->QuantumEnd = FALSE;
60         KiQuantumEnd();
61     }
62     else if (Prcb->NextThread)
63     {
64         /* Capture current thread data */
65         OldThread = Prcb->CurrentThread;
66         NewThread = Prcb->NextThread;
67 
68         /* Set new thread data */
69         Prcb->NextThread = NULL;
70         Prcb->CurrentThread = NewThread;
71 
72         /* The thread is now running */
73         NewThread->State = Running;
74         OldThread->WaitReason = WrDispatchInt;
75 
76         /* Make the old thread ready */
77         KxQueueReadyThread(OldThread, Prcb);
78 
79         /* Swap to the new thread */
80         KiSwapContext(APC_LEVEL, OldThread);
81     }
82 
83     /* Disable interrupts and go back to old irql */
84     _disable();
85     KeLowerIrql(OldIrql);
86 }
87 
88 
89 VOID
90 FASTCALL
91 KeZeroPages(IN PVOID Address,
92             IN ULONG Size)
93 {
94     /* Not using XMMI in this routine */
95     RtlZeroMemory(Address, Size);
96 }
97 
98 PVOID
99 KiSwitchKernelStackHelper(
100     LONG_PTR StackOffset,
101     PVOID OldStackBase);
102 
103 /*
104  * Kernel stack layout (example pointers):
105  * 0xFFFFFC0F'2D008000 KTHREAD::StackBase
106  *    [XSAVE_AREA size == KeXStateLength = 0x440]
107  * 0xFFFFFC0F'2D007BC0 KTHREAD::StateSaveArea _XSAVE_FORMAT
108  * 0xFFFFFC0F'2D007B90 KTHREAD::InitialStack
109  *    [0x190 bytes KTRAP_FRAME]
110  * 0xFFFFFC0F'2D007A00 KTHREAD::TrapFrame
111  *    [KSTART_FRAME] or ...
112  *    [KSWITCH_FRAME]
113  * 0xFFFFFC0F'2D007230 KTHREAD::KernelStack
114  */
115 
116 PVOID
117 NTAPI
118 KiSwitchKernelStack(PVOID StackBase, PVOID StackLimit)
119 {
120     PKTHREAD CurrentThread;
121     PVOID OldStackBase;
122     LONG_PTR StackOffset;
123     SIZE_T StackSize;
124     PKIPCR Pcr;
125 
126     /* Get the current thread */
127     CurrentThread = KeGetCurrentThread();
128 
129     /* Save the old stack base */
130     OldStackBase = CurrentThread->StackBase;
131 
132     /* Get the size of the current stack */
133     StackSize = (ULONG_PTR)CurrentThread->StackBase - CurrentThread->StackLimit;
134     ASSERT(StackSize <= (ULONG_PTR)StackBase - (ULONG_PTR)StackLimit);
135 
136     /* Copy the current stack contents to the new stack */
137     RtlCopyMemory((PUCHAR)StackBase - StackSize,
138                   (PVOID)CurrentThread->StackLimit,
139                   StackSize);
140 
141     /* Calculate the offset between the old and the new stack */
142     StackOffset = (PUCHAR)StackBase - (PUCHAR)CurrentThread->StackBase;
143 
144     /* Disable interrupts while messing with the stack */
145     _disable();
146 
147     /* Set the new trap frame */
148     CurrentThread->TrapFrame = (PKTRAP_FRAME)Add2Ptr(CurrentThread->TrapFrame,
149                                                      StackOffset);
150 
151     /* Set the new initial stack */
152     CurrentThread->InitialStack = Add2Ptr(CurrentThread->InitialStack,
153                                           StackOffset);
154 
155     /* Set the new stack limits */
156     CurrentThread->StackBase = StackBase;
157     CurrentThread->StackLimit = (ULONG_PTR)StackLimit;
158     CurrentThread->LargeStack = TRUE;
159 
160     /* Adjust RspBase in the PCR */
161     Pcr = (PKIPCR)KeGetPcr();
162     Pcr->Prcb.RspBase += StackOffset;
163 
164     /* Adjust Rsp0 in the TSS */
165     Pcr->TssBase->Rsp0 += StackOffset;
166 
167     return OldStackBase;
168 }
169 
170 
171 NTSTATUS
172 NTAPI
173 KeUserModeCallback(IN ULONG RoutineIndex,
174                    IN PVOID Argument,
175                    IN ULONG ArgumentLength,
176                    OUT PVOID *Result,
177                    OUT PULONG ResultLength)
178 {
179     UNIMPLEMENTED;
180     __debugbreak();
181     return STATUS_UNSUCCESSFUL;
182 }
183 
184 VOID
185 FASTCALL
186 KiIdleLoop(VOID)
187 {
188     PKPRCB Prcb = KeGetCurrentPrcb();
189     PKTHREAD OldThread, NewThread;
190 
191     /* Now loop forever */
192     while (TRUE)
193     {
194         /* Start of the idle loop: disable interrupts */
195         _enable();
196         YieldProcessor();
197         YieldProcessor();
198         _disable();
199 
200         /* Check for pending timers, pending DPCs, or pending ready threads */
201         if ((Prcb->DpcData[0].DpcQueueDepth) ||
202             (Prcb->TimerRequest) ||
203             (Prcb->DeferredReadyListHead.Next))
204         {
205             /* Quiesce the DPC software interrupt */
206             HalClearSoftwareInterrupt(DISPATCH_LEVEL);
207 
208             /* Handle it */
209             KiRetireDpcList(Prcb);
210         }
211 
212         /* Check if a new thread is scheduled for execution */
213         if (Prcb->NextThread)
214         {
215             /* Enable interrupts */
216             _enable();
217 
218             /* Capture current thread data */
219             OldThread = Prcb->CurrentThread;
220             NewThread = Prcb->NextThread;
221 
222             /* Set new thread data */
223             Prcb->NextThread = NULL;
224             Prcb->CurrentThread = NewThread;
225 
226             /* The thread is now running */
227             NewThread->State = Running;
228 
229             /* Do the swap at SYNCH_LEVEL */
230             KfRaiseIrql(SYNCH_LEVEL);
231 
232             /* Switch away from the idle thread */
233             KiSwapContext(APC_LEVEL, OldThread);
234 
235             /* Go back to DISPATCH_LEVEL */
236             KeLowerIrql(DISPATCH_LEVEL);
237         }
238         else
239         {
240             /* Continue staying idle. Note the HAL returns with interrupts on */
241             Prcb->PowerState.IdleFunction(&Prcb->PowerState);
242         }
243     }
244 }
245 
246 
247 /*! \name KiInitializeUserApc
248  *
249  *  \brief
250  *      Prepares the current trap frame (which must have come from user mode)
251  *      with the ntdll.KiUserApcDispatcher entrypoint, copying a CONTEXT
252  *      record with the context from the old trap frame to the threads user
253  *      mode stack.
254  *
255  *  \param ExceptionFrame
256  *  \param TrapFrame
257  *  \param NormalRoutine
258  *  \param NormalContext
259  *  \param SystemArgument1
260  *  \param SystemArgument2
261  *
262  *  \remarks
263  *      This function is called from KiDeliverApc, when the trap frame came
264  *      from user mode. This happens before a systemcall or interrupt exits back
265  *      to usermode or when a thread is started from PspUserThreadstartup.
266  *      The trap exit code will then leave to KiUserApcDispatcher which in turn
267  *      calls the NormalRoutine, passing NormalContext, SystemArgument1 and
268  *      SystemArgument2 as parameters. When that function returns, it calls
269  *      NtContinue to return back to the kernel, where the old context that was
270  *      saved on the usermode stack is restored and execution is transferred
271  *      back to usermode, where the original trap originated from.
272  *
273  *--*/
274 VOID
275 NTAPI
276 KiInitializeUserApc(IN PKEXCEPTION_FRAME ExceptionFrame,
277                     IN PKTRAP_FRAME TrapFrame,
278                     IN PKNORMAL_ROUTINE NormalRoutine,
279                     IN PVOID NormalContext,
280                     IN PVOID SystemArgument1,
281                     IN PVOID SystemArgument2)
282 {
283     CONTEXT Context = { 0 };
284     ULONG64 AlignedRsp, Stack;
285     EXCEPTION_RECORD SehExceptRecord;
286 
287     /* Sanity check, that the trap frame is from user mode */
288     ASSERT((TrapFrame->SegCs & MODE_MASK) != KernelMode);
289 
290     /* Convert the current trap frame to a context */
291     Context.ContextFlags = CONTEXT_FULL | CONTEXT_DEBUG_REGISTERS;
292     KeTrapFrameToContext(TrapFrame, ExceptionFrame, &Context);
293 
294     /* We jump to KiUserApcDispatcher in ntdll */
295     TrapFrame->Rip = (ULONG64)KeUserApcDispatcher;
296 
297     /* Setup Ring 3 segments */
298     TrapFrame->SegCs = KGDT64_R3_CODE | RPL_MASK;
299     TrapFrame->SegDs = KGDT64_R3_DATA | RPL_MASK;
300     TrapFrame->SegEs = KGDT64_R3_DATA | RPL_MASK;
301     TrapFrame->SegFs = KGDT64_R3_CMTEB | RPL_MASK;
302     TrapFrame->SegGs = KGDT64_R3_DATA | RPL_MASK;
303     TrapFrame->SegSs = KGDT64_R3_DATA | RPL_MASK;
304 
305     /* Sanitize EFLAGS, enable interrupts */
306     TrapFrame->EFlags = (Context.EFlags & EFLAGS_USER_SANITIZE);
307     TrapFrame->EFlags |= EFLAGS_INTERRUPT_MASK;
308 
309     /* Set parameters for KiUserApcDispatcher */
310     Context.P1Home = (ULONG64)NormalContext;
311     Context.P2Home = (ULONG64)SystemArgument1;
312     Context.P3Home = (ULONG64)SystemArgument2;
313     Context.P4Home = (ULONG64)NormalRoutine;
314 
315     /* Check if thread has IOPL and force it enabled if so */
316     //if (KeGetCurrentThread()->Iopl) TrapFrame->EFlags |= EFLAGS_IOPL;
317 
318     /* Align Stack to 16 bytes and allocate space */
319     AlignedRsp = Context.Rsp & ~15;
320     Stack = AlignedRsp - sizeof(CONTEXT);
321     TrapFrame->Rsp = Stack;
322 
323     /* The stack must be 16 byte aligned for KiUserApcDispatcher */
324     ASSERT((Stack & 15) == 0);
325 
326     /* Protect with SEH */
327     _SEH2_TRY
328     {
329          /* Probe the stack */
330         ProbeForWrite((PCONTEXT)Stack,  sizeof(CONTEXT), 8);
331 
332         /* Copy the context */
333         RtlCopyMemory((PCONTEXT)Stack, &Context, sizeof(CONTEXT));
334     }
335     _SEH2_EXCEPT((RtlCopyMemory(&SehExceptRecord, _SEH2_GetExceptionInformation()->ExceptionRecord, sizeof(EXCEPTION_RECORD)), EXCEPTION_EXECUTE_HANDLER))
336     {
337         /* Dispatch the exception */
338         SehExceptRecord.ExceptionAddress = (PVOID)TrapFrame->Rip;
339         KiDispatchException(&SehExceptRecord,
340                             ExceptionFrame,
341                             TrapFrame,
342                             UserMode,
343                             TRUE);
344     }
345     _SEH2_END;
346 }
347 
348 VOID
349 NTAPI
350 KiSwapProcess(IN PKPROCESS NewProcess,
351               IN PKPROCESS OldProcess)
352 {
353     PKIPCR Pcr = (PKIPCR)KeGetPcr();
354 
355 #ifdef CONFIG_SMP
356     /* Update active processor mask */
357     InterlockedXor64((PLONG64)&NewProcess->ActiveProcessors, Pcr->Prcb.SetMember);
358     InterlockedXor64((PLONG64)&OldProcess->ActiveProcessors, Pcr->Prcb.SetMember);
359 #endif
360 
361     /* Update CR3 */
362     __writecr3(NewProcess->DirectoryTableBase[0]);
363 
364     /* Update IOPM offset */
365     Pcr->TssBase->IoMapBase = NewProcess->IopmOffset;
366 }
367 
368 #define MAX_SYSCALL_PARAMS 16
369 
370 NTSTATUS
371 NtSyscallFailure(void)
372 {
373     /* This is the failure function */
374     return (NTSTATUS)KeGetCurrentThread()->TrapFrame->Rax;
375 }
376 
377 PVOID
378 KiSystemCallHandler(
379     _In_ ULONG64 ReturnAddress,
380     _In_ ULONG64 P2,
381     _In_ ULONG64 P3,
382     _In_ ULONG64 P4)
383 {
384     PKTRAP_FRAME TrapFrame;
385     PKSERVICE_TABLE_DESCRIPTOR DescriptorTable;
386     PKTHREAD Thread;
387     PULONG64 KernelParams, UserParams;
388     ULONG ServiceNumber, Offset, Count;
389     ULONG64 UserRsp;
390 
391     /* Get a pointer to the trap frame */
392     TrapFrame = (PKTRAP_FRAME)((PULONG64)_AddressOfReturnAddress() + 1 + MAX_SYSCALL_PARAMS);
393 
394     /* Save some values in the trap frame */
395     TrapFrame->Rip = ReturnAddress;
396     TrapFrame->Rdx = P2;
397     TrapFrame->R8 = P3;
398     TrapFrame->R9 = P4;
399 
400     /* Increase system call count */
401     __addgsdword(FIELD_OFFSET(KIPCR, Prcb.KeSystemCalls), 1);
402 
403     /* Get the current thread */
404     Thread = KeGetCurrentThread();
405 
406     /* Set previous mode */
407     Thread->PreviousMode = TrapFrame->PreviousMode = UserMode;
408 
409     /* Save the old trap frame and set the new */
410     TrapFrame->TrapFrame = (ULONG64)Thread->TrapFrame;
411     Thread->TrapFrame = TrapFrame;
412 
413     /* We don't have an exception frame yet */
414     TrapFrame->ExceptionFrame = 0;
415 
416     /* Before enabling interrupts get the user rsp from the KPCR */
417     UserRsp = __readgsqword(FIELD_OFFSET(KIPCR, UserRsp));
418     TrapFrame->Rsp = UserRsp;
419 
420     /* Enable interrupts */
421     _enable();
422 
423     /* If the usermode rsp was not a usermode address, prepare an exception */
424     if (UserRsp > MmUserProbeAddress) UserRsp = MmUserProbeAddress;
425 
426     /* Get the address of the usermode and kernelmode parameters */
427     UserParams = (PULONG64)UserRsp + 1;
428     KernelParams = (PULONG64)TrapFrame - MAX_SYSCALL_PARAMS;
429 
430     /* Get the system call number from the trap frame and decode it */
431     ServiceNumber = (ULONG)TrapFrame->Rax;
432     Offset = (ServiceNumber >> SERVICE_TABLE_SHIFT) & SERVICE_TABLE_MASK;
433     ServiceNumber &= SERVICE_NUMBER_MASK;
434 
435     /* Check for win32k system calls */
436     if (Offset & SERVICE_TABLE_TEST)
437     {
438         ULONG GdiBatchCount;
439 
440         /* Read the GDI batch count from the TEB */
441         _SEH2_TRY
442         {
443             GdiBatchCount = NtCurrentTeb()->GdiBatchCount;
444         }
445         _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
446         {
447             GdiBatchCount = 0;
448         }
449 
450         /* Flush batch, if there are entries */
451         if (GdiBatchCount != 0)
452         {
453             KeGdiFlushUserBatch();
454         }
455     }
456 
457     /* Get descriptor table */
458     DescriptorTable = (PVOID)((ULONG_PTR)Thread->ServiceTable + Offset);
459 
460     /* Validate the system call number */
461     if (ServiceNumber >= DescriptorTable->Limit)
462     {
463         /* Check if this is a GUI call */
464         if (!(Offset & SERVICE_TABLE_TEST))
465         {
466             /* Fail the call */
467             TrapFrame->Rax = STATUS_INVALID_SYSTEM_SERVICE;
468             return (PVOID)NtSyscallFailure;
469         }
470 
471         /* Convert us to a GUI thread
472            To be entirely correct. we return KiConvertToGuiThread,
473            which allocates a new stack, switches to it, calls
474            PsConvertToGuiThread and resumes in the middle of
475            KiSystemCallEntry64 to restart the system call handling. */
476         return (PVOID)KiConvertToGuiThread;
477     }
478 
479     /* Get stack bytes and calculate argument count */
480     Count = DescriptorTable->Number[ServiceNumber] / 8;
481 
482     __try
483     {
484         switch (Count)
485         {
486             case 16: KernelParams[15] = UserParams[15];
487             case 15: KernelParams[14] = UserParams[14];
488             case 14: KernelParams[13] = UserParams[13];
489             case 13: KernelParams[12] = UserParams[12];
490             case 12: KernelParams[11] = UserParams[11];
491             case 11: KernelParams[10] = UserParams[10];
492             case 10: KernelParams[9] = UserParams[9];
493             case 9: KernelParams[8] = UserParams[8];
494             case 8: KernelParams[7] = UserParams[7];
495             case 7: KernelParams[6] = UserParams[6];
496             case 6: KernelParams[5] = UserParams[5];
497             case 5: KernelParams[4] = UserParams[4];
498             case 4:
499             case 3:
500             case 2:
501             case 1:
502             case 0:
503                 break;
504 
505             default:
506                 __debugbreak();
507                 break;
508         }
509     }
510     __except(1)
511     {
512         TrapFrame->Rax = _SEH2_GetExceptionCode();
513         return (PVOID)NtSyscallFailure;
514     }
515 
516 
517     return (PVOID)DescriptorTable->Base[ServiceNumber];
518 }
519 
520 
521 // FIXME: we need to
522 VOID
523 KiSystemService(IN PKTHREAD Thread,
524                 IN PKTRAP_FRAME TrapFrame,
525                 IN ULONG Instruction)
526 {
527     UNIMPLEMENTED;
528     __debugbreak();
529 }
530 
531 NTSYSAPI
532 NTSTATUS
533 NTAPI
534 NtCallbackReturn
535 ( IN PVOID Result OPTIONAL, IN ULONG ResultLength, IN NTSTATUS Status )
536 {
537     UNIMPLEMENTED;
538     __debugbreak();
539     return STATUS_UNSUCCESSFUL;
540 }
541 
542 NTSTATUS
543 NTAPI
544 NtSetLdtEntries
545 (ULONG Selector1, LDT_ENTRY LdtEntry1, ULONG Selector2, LDT_ENTRY LdtEntry2)
546 {
547     UNIMPLEMENTED;
548     __debugbreak();
549     return STATUS_UNSUCCESSFUL;
550 }
551 
552 NTSTATUS
553 NTAPI
554 NtVdmControl(IN ULONG ControlCode,
555              IN PVOID ControlData)
556 {
557     /* Not supported */
558     return STATUS_NOT_IMPLEMENTED;
559 }
560 
561 NTSTATUS
562 NTAPI
563 KiCallUserMode(
564     IN PVOID *OutputBuffer,
565     IN PULONG OutputLength)
566 {
567     UNIMPLEMENTED;
568     __debugbreak();
569     return STATUS_UNSUCCESSFUL;
570 }
571 
572 ULONG ProcessCount;
573 BOOLEAN CcPfEnablePrefetcher;
574 
575 
576