/** @file IA32 CPU Exception Handler functons. Copyright (c) 2012 - 2022, Intel Corporation. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent **/ #include "CpuExceptionCommon.h" /** Return address map of exception handler template so that C code can generate exception tables. @param IdtEntry Pointer to IDT entry to be updated. @param InterruptHandler IDT handler value. **/ VOID ArchUpdateIdtEntry ( OUT IA32_IDT_GATE_DESCRIPTOR *IdtEntry, IN UINTN InterruptHandler ) { IdtEntry->Bits.OffsetLow = (UINT16)(UINTN)InterruptHandler; IdtEntry->Bits.OffsetHigh = (UINT16)((UINTN)InterruptHandler >> 16); IdtEntry->Bits.GateType = IA32_IDT_GATE_TYPE_INTERRUPT_32; } /** Read IDT handler value from IDT entry. @param IdtEntry Pointer to IDT entry to be read. **/ UINTN ArchGetIdtHandler ( IN IA32_IDT_GATE_DESCRIPTOR *IdtEntry ) { return (UINTN)IdtEntry->Bits.OffsetLow + (((UINTN)IdtEntry->Bits.OffsetHigh) << 16); } /** Save CPU exception context when handling EFI_VECTOR_HANDOFF_HOOK_AFTER case. @param[in] ExceptionType Exception type. @param[in] SystemContext Pointer to EFI_SYSTEM_CONTEXT. @param[in] ExceptionHandlerData Pointer to exception handler data. **/ VOID ArchSaveExceptionContext ( IN UINTN ExceptionType, IN EFI_SYSTEM_CONTEXT SystemContext, IN EXCEPTION_HANDLER_DATA *ExceptionHandlerData ) { IA32_EFLAGS32 Eflags; RESERVED_VECTORS_DATA *ReservedVectors; ReservedVectors = ExceptionHandlerData->ReservedVectors; // // Save Exception context in global variable in first entry of the exception handler. // So when original exception handler returns to the new exception handler (second entry), // the Eflags/Cs/Eip/ExceptionData can be used. // ReservedVectors[ExceptionType].OldFlags = SystemContext.SystemContextIa32->Eflags; ReservedVectors[ExceptionType].OldCs = SystemContext.SystemContextIa32->Cs; ReservedVectors[ExceptionType].OldIp = SystemContext.SystemContextIa32->Eip; ReservedVectors[ExceptionType].ExceptionData = SystemContext.SystemContextIa32->ExceptionData; // // Clear IF flag to avoid old IDT handler enable interrupt by IRET // Eflags.UintN = SystemContext.SystemContextIa32->Eflags; Eflags.Bits.IF = 0; SystemContext.SystemContextIa32->Eflags = Eflags.UintN; // // Modify the EIP in stack, then old IDT handler will return to HookAfterStubBegin. // SystemContext.SystemContextIa32->Eip = (UINTN)ReservedVectors[ExceptionType].HookAfterStubHeaderCode; } /** Restore CPU exception context when handling EFI_VECTOR_HANDOFF_HOOK_AFTER case. @param[in] ExceptionType Exception type. @param[in] SystemContext Pointer to EFI_SYSTEM_CONTEXT. @param[in] ExceptionHandlerData Pointer to exception handler data. **/ VOID ArchRestoreExceptionContext ( IN UINTN ExceptionType, IN EFI_SYSTEM_CONTEXT SystemContext, IN EXCEPTION_HANDLER_DATA *ExceptionHandlerData ) { RESERVED_VECTORS_DATA *ReservedVectors; ReservedVectors = ExceptionHandlerData->ReservedVectors; SystemContext.SystemContextIa32->Eflags = ReservedVectors[ExceptionType].OldFlags; SystemContext.SystemContextIa32->Cs = ReservedVectors[ExceptionType].OldCs; SystemContext.SystemContextIa32->Eip = ReservedVectors[ExceptionType].OldIp; SystemContext.SystemContextIa32->ExceptionData = ReservedVectors[ExceptionType].ExceptionData; } /** Setup separate stacks for certain exception handlers. @param[in] Buffer Point to buffer used to separate exception stack. @param[in, out] BufferSize On input, it indicates the byte size of Buffer. If the size is not enough, the return status will be EFI_BUFFER_TOO_SMALL, and output BufferSize will be the size it needs. @retval EFI_SUCCESS The stacks are assigned successfully. @retval EFI_BUFFER_TOO_SMALL This BufferSize is too small. **/ EFI_STATUS ArchSetupExceptionStack ( IN VOID *Buffer, IN OUT UINTN *BufferSize ) { IA32_DESCRIPTOR Gdtr; IA32_DESCRIPTOR Idtr; IA32_IDT_GATE_DESCRIPTOR *IdtTable; IA32_TSS_DESCRIPTOR *TssDesc; IA32_TSS_DESCRIPTOR *TssDescBase; IA32_TASK_STATE_SEGMENT *Tss; VOID *NewGdtTable; UINTN StackTop; UINTN Index; UINTN Vector; UINTN TssBase; UINT8 *StackSwitchExceptions; UINTN NeedBufferSize; EXCEPTION_HANDLER_TEMPLATE_MAP TemplateMap; if (BufferSize == NULL) { return EFI_INVALID_PARAMETER; } // // Total needed size includes stack size, new GDT table size, TSS size. // Add another DESCRIPTOR size for alignment requiremet. // // Layout of memory needed for each processor: // -------------------------------- // | | // | Stack Size | X ExceptionNumber // | | // -------------------------------- // | Alignment | (just in case) // -------------------------------- // | | // | Original GDT | // | | // -------------------------------- // | Current task descriptor | // -------------------------------- // | | // | Exception task descriptors | X ExceptionNumber // | | // -------------------------------- // | Current task-state segment | // -------------------------------- // | | // | Exception task-state segment | X ExceptionNumber // | | // -------------------------------- // AsmReadGdtr (&Gdtr); NeedBufferSize = CPU_STACK_SWITCH_EXCEPTION_NUMBER * CPU_KNOWN_GOOD_STACK_SIZE + sizeof (IA32_TSS_DESCRIPTOR) + Gdtr.Limit + 1 + CPU_TSS_DESC_SIZE + CPU_TSS_SIZE; if (*BufferSize < NeedBufferSize) { *BufferSize = NeedBufferSize; return EFI_BUFFER_TOO_SMALL; } if (Buffer == NULL) { return EFI_INVALID_PARAMETER; } AsmReadIdtr (&Idtr); StackSwitchExceptions = CPU_STACK_SWITCH_EXCEPTION_LIST; StackTop = (UINTN)Buffer + CPU_STACK_SWITCH_EXCEPTION_NUMBER * CPU_KNOWN_GOOD_STACK_SIZE; NewGdtTable = ALIGN_POINTER (StackTop, sizeof (IA32_TSS_DESCRIPTOR)); TssDesc = (IA32_TSS_DESCRIPTOR *)((UINTN)NewGdtTable + Gdtr.Limit + 1); Tss = (IA32_TASK_STATE_SEGMENT *)((UINTN)TssDesc + CPU_TSS_DESC_SIZE); TssDescBase = TssDesc; CopyMem (NewGdtTable, (VOID *)Gdtr.Base, Gdtr.Limit + 1); Gdtr.Base = (UINTN)NewGdtTable; Gdtr.Limit = (UINT16)(Gdtr.Limit + CPU_TSS_DESC_SIZE); // // Fixup current task descriptor. Task-state segment for current task will // be filled by processor during task switching. // TssBase = (UINTN)Tss; TssDesc->Uint64 = 0; TssDesc->Bits.LimitLow = sizeof (IA32_TASK_STATE_SEGMENT) - 1; TssDesc->Bits.BaseLow = (UINT16)TssBase; TssDesc->Bits.BaseMid = (UINT8)(TssBase >> 16); TssDesc->Bits.Type = IA32_GDT_TYPE_TSS; TssDesc->Bits.P = 1; TssDesc->Bits.LimitHigh = 0; TssDesc->Bits.BaseHigh = (UINT8)(TssBase >> 24); // // Fixup exception task descriptor and task-state segment // AsmGetTssTemplateMap (&TemplateMap); // // Plus 1 byte is for compact stack layout in case StackTop is already aligned. // StackTop = StackTop - CPU_STACK_ALIGNMENT + 1; StackTop = (UINTN)ALIGN_POINTER (StackTop, CPU_STACK_ALIGNMENT); IdtTable = (IA32_IDT_GATE_DESCRIPTOR *)Idtr.Base; for (Index = 0; Index < CPU_STACK_SWITCH_EXCEPTION_NUMBER; ++Index) { TssDesc += 1; Tss += 1; // // Fixup TSS descriptor // TssBase = (UINTN)Tss; TssDesc->Uint64 = 0; TssDesc->Bits.LimitLow = sizeof (IA32_TASK_STATE_SEGMENT) - 1; TssDesc->Bits.BaseLow = (UINT16)TssBase; TssDesc->Bits.BaseMid = (UINT8)(TssBase >> 16); TssDesc->Bits.Type = IA32_GDT_TYPE_TSS; TssDesc->Bits.P = 1; TssDesc->Bits.LimitHigh = 0; TssDesc->Bits.BaseHigh = (UINT8)(TssBase >> 24); // // Fixup TSS // Vector = StackSwitchExceptions[Index]; if ((Vector >= CPU_EXCEPTION_NUM) || (Vector >= (Idtr.Limit + 1) / sizeof (IA32_IDT_GATE_DESCRIPTOR))) { continue; } ZeroMem (Tss, sizeof (*Tss)); Tss->EIP = (UINT32)(TemplateMap.ExceptionStart + Vector * TemplateMap.ExceptionStubHeaderSize); Tss->EFLAGS = 0x2; Tss->ESP = StackTop; Tss->CR3 = AsmReadCr3 (); Tss->ES = AsmReadEs (); Tss->CS = AsmReadCs (); Tss->SS = AsmReadSs (); Tss->DS = AsmReadDs (); Tss->FS = AsmReadFs (); Tss->GS = AsmReadGs (); StackTop -= CPU_KNOWN_GOOD_STACK_SIZE; // // Update IDT to use Task Gate for given exception // IdtTable[Vector].Bits.OffsetLow = 0; IdtTable[Vector].Bits.Selector = (UINT16)((UINTN)TssDesc - Gdtr.Base); IdtTable[Vector].Bits.Reserved_0 = 0; IdtTable[Vector].Bits.GateType = IA32_IDT_GATE_TYPE_TASK; IdtTable[Vector].Bits.OffsetHigh = 0; } // // Publish GDT // AsmWriteGdtr (&Gdtr); // // Load current task // AsmWriteTr ((UINT16)((UINTN)TssDescBase - Gdtr.Base)); return EFI_SUCCESS; } /** Display processor context. @param[in] ExceptionType Exception type. @param[in] SystemContext Processor context to be display. **/ VOID EFIAPI DumpCpuContext ( IN EFI_EXCEPTION_TYPE ExceptionType, IN EFI_SYSTEM_CONTEXT SystemContext ) { InternalPrintMessage ( "!!!! IA32 Exception Type - %02x(%a) CPU Apic ID - %08x !!!!\n", ExceptionType, GetExceptionNameStr (ExceptionType), GetApicId () ); if ((mErrorCodeFlag & (1 << ExceptionType)) != 0) { InternalPrintMessage ( "ExceptionData - %08x", SystemContext.SystemContextIa32->ExceptionData ); if (ExceptionType == EXCEPT_IA32_PAGE_FAULT) { InternalPrintMessage ( " I:%x R:%x U:%x W:%x P:%x PK:%x SS:%x SGX:%x", (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_ID) != 0, (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_RSVD) != 0, (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_US) != 0, (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_WR) != 0, (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_P) != 0, (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_PK) != 0, (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_SS) != 0, (SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_SGX) != 0 ); } InternalPrintMessage ("\n"); } InternalPrintMessage ( "EIP - %08x, CS - %08x, EFLAGS - %08x\n", SystemContext.SystemContextIa32->Eip, SystemContext.SystemContextIa32->Cs, SystemContext.SystemContextIa32->Eflags ); InternalPrintMessage ( "EAX - %08x, ECX - %08x, EDX - %08x, EBX - %08x\n", SystemContext.SystemContextIa32->Eax, SystemContext.SystemContextIa32->Ecx, SystemContext.SystemContextIa32->Edx, SystemContext.SystemContextIa32->Ebx ); InternalPrintMessage ( "ESP - %08x, EBP - %08x, ESI - %08x, EDI - %08x\n", SystemContext.SystemContextIa32->Esp, SystemContext.SystemContextIa32->Ebp, SystemContext.SystemContextIa32->Esi, SystemContext.SystemContextIa32->Edi ); InternalPrintMessage ( "DS - %08x, ES - %08x, FS - %08x, GS - %08x, SS - %08x\n", SystemContext.SystemContextIa32->Ds, SystemContext.SystemContextIa32->Es, SystemContext.SystemContextIa32->Fs, SystemContext.SystemContextIa32->Gs, SystemContext.SystemContextIa32->Ss ); InternalPrintMessage ( "CR0 - %08x, CR2 - %08x, CR3 - %08x, CR4 - %08x\n", SystemContext.SystemContextIa32->Cr0, SystemContext.SystemContextIa32->Cr2, SystemContext.SystemContextIa32->Cr3, SystemContext.SystemContextIa32->Cr4 ); InternalPrintMessage ( "DR0 - %08x, DR1 - %08x, DR2 - %08x, DR3 - %08x\n", SystemContext.SystemContextIa32->Dr0, SystemContext.SystemContextIa32->Dr1, SystemContext.SystemContextIa32->Dr2, SystemContext.SystemContextIa32->Dr3 ); InternalPrintMessage ( "DR6 - %08x, DR7 - %08x\n", SystemContext.SystemContextIa32->Dr6, SystemContext.SystemContextIa32->Dr7 ); InternalPrintMessage ( "GDTR - %08x %08x, IDTR - %08x %08x\n", SystemContext.SystemContextIa32->Gdtr[0], SystemContext.SystemContextIa32->Gdtr[1], SystemContext.SystemContextIa32->Idtr[0], SystemContext.SystemContextIa32->Idtr[1] ); InternalPrintMessage ( "LDTR - %08x, TR - %08x\n", SystemContext.SystemContextIa32->Ldtr, SystemContext.SystemContextIa32->Tr ); InternalPrintMessage ( "FXSAVE_STATE - %08x\n", &SystemContext.SystemContextIa32->FxSaveState ); } /** Display CPU information. @param ExceptionType Exception type. @param SystemContext Pointer to EFI_SYSTEM_CONTEXT. **/ VOID DumpImageAndCpuContent ( IN EFI_EXCEPTION_TYPE ExceptionType, IN EFI_SYSTEM_CONTEXT SystemContext ) { DumpCpuContext (ExceptionType, SystemContext); // // Dump module image base and module entry point by EIP // if ((ExceptionType == EXCEPT_IA32_PAGE_FAULT) && ((SystemContext.SystemContextIa32->ExceptionData & IA32_PF_EC_ID) != 0)) { // // The EIP in SystemContext could not be used // if it is page fault with I/D set. // DumpModuleImageInfo ((*(UINTN *)(UINTN)SystemContext.SystemContextIa32->Esp)); } else { DumpModuleImageInfo (SystemContext.SystemContextIa32->Eip); } }