Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

2374 lines
76 KiB

/*++
Copyright (c) 1995 Intel Corporation
Module Name:
i64efi.c
Abstract:
This module implements the routines that transfer control
from the kernel to the EFI code.
Author:
Bernard Lint
M. Jayakumar ([email protected])
Environment:
Kernel mode
Revision History:
Neal Vu ([email protected]), 03-Apr-2001:
Added HalpGetSmBiosVersion.
--*/
#include "halp.h"
#include "arc.h"
#include "arccodes.h"
#include "i64fw.h"
#include "floatem.h"
#include "fpswa.h"
#include <smbios.h>
extern ULONGLONG PhysicalIOBase;
BOOLEAN
HalpCompareEfiGuid (
IN EFI_GUID CheckGuid,
IN EFI_GUID ReferenceGuid
);
BOOLEAN
MmSetPageProtection(
IN PVOID VirtualAddress,
IN SIZE_T NumberOfBytes,
IN ULONG NewProtect
);
EFI_STATUS
HalpCallEfiPhysical(
IN ULONGLONG Arg1,
IN ULONGLONG Arg2,
IN ULONGLONG Arg3,
IN ULONGLONG Arg4,
IN ULONGLONG Arg5,
IN ULONGLONG Arg6,
IN ULONGLONG EP,
IN ULONGLONG GP
);
EFI_STATUS
HalpCallEfiVirtual(
IN ULONGLONG Arg1,
IN ULONGLONG Arg2,
IN ULONGLONG Arg3,
IN ULONGLONG Arg4,
IN ULONGLONG Arg5,
IN ULONGLONG Arg6,
IN ULONGLONG EP,
IN ULONGLONG GP
);
LONG
HalFpEmulate(
ULONG trap_type,
BUNDLE *pbundle,
ULONGLONG *pipsr,
ULONGLONG *pfpsr,
ULONGLONG *pisr,
ULONGLONG *ppreds,
ULONGLONG *pifs,
FP_STATE *fp_state
);
BOOLEAN
HalEFIFpSwa(
VOID
);
VOID
HalpFpswaPlabelFixup(
EFI_MEMORY_DESCRIPTOR *EfiVirtualMemoryMapPtr,
ULONGLONG MapEntries,
ULONGLONG EfiDescriptorSize,
PPLABEL_DESCRIPTOR PlabelPointer
);
PUCHAR
HalpGetSmBiosVersion (
IN PLOADER_PARAMETER_BLOCK LoaderBlock
);
//
// External global data
//
extern HALP_SAL_PAL_DATA HalpSalPalData;
extern ULONGLONG HalpPalProcPointer;
extern ULONGLONG HalpSalProcPointer;
extern ULONGLONG HalpSalProcGlobalPointer;
extern KSPIN_LOCK HalpSalSpinLock;
extern KSPIN_LOCK HalpSalStateInfoSpinLock;
extern KSPIN_LOCK HalpMcaSpinLock;
extern KSPIN_LOCK HalpInitSpinLock;
extern KSPIN_LOCK HalpCmcSpinLock;
extern KSPIN_LOCK HalpCpeSpinLock;
#define VENDOR_SPECIFIC_GUID \
{ 0xa3c72e56, 0x4c35, 0x11d3, 0x8a, 0x03, 0x0, 0xa0, 0xc9, 0x06, 0xad, 0xec }
#define ConfigGuidOffset 0x100
#define ConfigTableOffset 0x200
#define VariableNameOffset 0x100
#define VendorGuidOffset 0x200
#define AttributeOffset 0x300
#define DataSizeOffset 0x400
#define DataBufferOffset 0x500
#define EndOfCommonDataOffset 0x600
//
// Read Variable and Write Variable will not be called till the copying out of
// Memory Descriptors is done. Because the lock is released before copying and we are using
// the same offset for read/write variable as well as memory layout calls.
//
#define MemoryMapSizeOffset 0x100
#define DescriptorSizeOffset 0x200
#define DescriptorVersionOffset 0x300
#define MemoryMapOffset 0x400
#define OptionROMAddress 0xC0000
#define FP_EMUL_ERROR -1
SST_MEMORY_LIST PalCode;
NTSTATUS EfiInitStatus;
ULONGLONG PalTrMask;
PUCHAR HalpConfigGuidVirtualPtr, HalpConfigGuidPhysPtr;
PUCHAR HalpConfigTableVirtualPtr;
PUCHAR HalpConfigTablePhysPtr;
EFI_CONFIGURATION_TABLE *EfiConfigTableVirtualPtr;
EFI_CONFIGURATION_TABLE EfiConfigTable;
EFI_GUID CheckGuid;
EFI_GUID SalGuid = SAL_SYSTEM_TABLE_GUID;
EFI_GUID VendorGuid;
PUCHAR HalpVendorGuidPhysPtr;
PUCHAR HalpVendorGuidVirtualPtr;
EFI_SYSTEM_TABLE *EfiSysTableVirtualPtr;
EFI_SYSTEM_TABLE *EfiSysTableVirtualPtrCpy;
EFI_RUNTIME_SERVICES *EfiRSVirtualPtr;
EFI_BOOT_SERVICES *EfiBootVirtualPtr;
PLABEL_DESCRIPTOR *EfiVirtualGetVariablePtr; // Get Variable
PLABEL_DESCRIPTOR *EfiVirtualGetNextVariableNamePtr; // Get NextVariable Name
PLABEL_DESCRIPTOR *EfiVirtualSetVariablePtr; // Set Variable
PLABEL_DESCRIPTOR *EfiVirtualGetTimePtr; // Get Time
PLABEL_DESCRIPTOR *EfiVirtualSetTimePtr; // Set Time
PLABEL_DESCRIPTOR *EfiSetVirtualAddressMapPtr; // Set Virtual Address Map
PLABEL_DESCRIPTOR *EfiResetSystemPtr; // Reboot
PULONGLONG AttributePtr;
ULONGLONG EfiAttribute;
PULONGLONG DataSizePtr;
ULONGLONG EfiDataSize;
ULONGLONG EfiConfigTableSize, EfiMemoryMapSize,EfiDescriptorSize,EfiMapEntries;
ULONG EfiDescriptorVersion;
PULONGLONG HalpPhysBSPointer;
PULONGLONG HalpPhysStackPointer;
PUCHAR HalpVirtualEfiSalPalDataPointer;
PUCHAR HalpPhysEfiSalPalDataPointer;
PUCHAR HalpVirtualCommonDataPointer;
PUCHAR HalpPhysCommonDataPointer;
PUCHAR HalpVariableNamePhysPtr;
PUCHAR HalpVariableAttributesPhysPtr;
PUCHAR HalpDataSizePhysPtr;
PUCHAR HalpDataPhysPtr;
PUCHAR HalpMemoryMapSizePhysPtr;
PUCHAR HalpMemoryMapPhysPtr;
PUCHAR HalpDescriptorSizePhysPtr;
PUCHAR HalpDescriptorVersionPhysPtr;
PUCHAR HalpVariableNameVirtualPtr;
PUCHAR HalpVariableAttributesVirtualPtr;
PUCHAR HalpDataSizeVirtualPtr;
PUCHAR HalpDataVirtualPtr;
PUCHAR HalpCommonDataEndPtr;
PUCHAR HalpMemoryMapSizeVirtualPtr;
PVOID HalpMemoryMapVirtualPtr;
PUCHAR HalpDescriptorSizeVirtualPtr;
PUCHAR HalpDescriptorVersionVirtualPtr;
EFI_FPSWA HalpFpEmulate;
KSPIN_LOCK EFIMPLock;
ULONG HalpOsBootRendezVector;
BOOLEAN
HalpCompareEfiGuid (
IN EFI_GUID CheckGuid,
IN EFI_GUID ReferenceGuid
)
/*++
--*/
{
USHORT i;
USHORT TotalArrayLength = 8;
if (CheckGuid.Data1 != ReferenceGuid.Data1) {
return FALSE;
} else if (CheckGuid.Data2 != ReferenceGuid.Data2) {
return FALSE;
} else if (CheckGuid.Data3 != ReferenceGuid.Data3) {
return FALSE;
}
for (i = 0; i != TotalArrayLength; i++) {
if (CheckGuid.Data4[i] != ReferenceGuid.Data4[i])
return FALSE;
}
return TRUE;
} // HalpCompareEfiGuid()
VOID
HalpInitSalPalWorkArounds(
VOID
)
/*++
Routine Description:
This function determines and initializes the FW workarounds.
Arguments:
None.
Return Value:
None.
Globals:
Notes: This function is being called at the end of HalpInitSalPal.
It should not access SST members if this SAL table is unmapped.
--*/
{
NTSTATUS status;
extern FADT HalpFixedAcpiDescTable;
#define HalpIsIntelOEM() \
( !_strnicmp( HalpFixedAcpiDescTable.Header.OEMID, "INTEL", 5 ) )
#define HalpIsBigSur() \
( !strncmp( HalpFixedAcpiDescTable.Header.OEMTableID, "W460GXBS", 8 ) )
#define HalpIsIntelBigSur() \
( HalpIsIntelOEM() && HalpIsBigSur() )
if ( HalpIsIntelOEM() ) {
//
// If Intel BigSur and FW build < 103 (checked as Pal_A_Revision < 0x20),
// enable the SAL_GET_STATE_INFO log id increment workaround.
//
if ( HalpIsBigSur() ) {
if ( HalpSalPalData.PalVersion.PAL_A_Revision < 0x20 ) {
HalpSalPalData.Flags |= HALP_SALPAL_FIX_MCE_LOG_ID;
HalpSalPalData.Flags |= HALP_SALPAL_FIX_MP_SAFE;
}
//
// HALP_SALPAL_MCE_MODINFO_CPUINDINFO_OMITTED 04/12/2001:
// FW omits completely ERROR_PROCESSOR.CpuIdInfo field in the log.
// Softsur: This is fixed with developer FW SoftMca3 05/09/2001.
// Lion : ?
//
// HALP_SALPAL_MCE_PROCESSOR_STATICINFO_PARTIAL 04/12/2001:
// FW does not pad invalid processor staticinfo fields.
// We do not know when it is going to be fixed.
//
if ( HalpSalPalData.PalVersion.PAL_A_Revision < 0x21 ) {
HalpSalPalData.Flags |= HALP_SALPAL_MCE_PROCESSOR_CPUIDINFO_OMITTED;
}
HalpSalPalData.Flags |= HALP_SALPAL_MCE_PROCESSOR_STATICINFO_PARTIAL;
}
else {
//
// If Intel Lion and FW build < 78b (checked as SalRevision < 0x300),
// enable the SAL_GET_STATE_INFO log id increment workaround.
//
if ( HalpSalPalData.SalRevision.Revision < 0x300 ) {
HalpSalPalData.Flags |= HALP_SALPAL_FIX_MCE_LOG_ID;
HalpSalPalData.Flags |= HALP_SALPAL_FIX_MP_SAFE;
}
HalpSalPalData.Flags |= HALP_SALPAL_MCE_PROCESSOR_CPUIDINFO_OMITTED;
HalpSalPalData.Flags |= HALP_SALPAL_MCE_PROCESSOR_STATICINFO_PARTIAL;
}
}
return;
} // HalpInitSalPalWorkArounds()
NTSTATUS
HalpInitSalPal(
PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
This function extracts data from the SAL system table, virtually maps the
SAL code, SAL data, and PAL code areas. PAL requires a TR mapping, and
is mapped using an architected TR, using the smallest page size to map
the entire PAL code region. If SAL data or SAL code areas can be mapped in
the same page, it uses the same translation. Otherwise, it uses MmMapIoSpace.
Arguments:
LoaderBlock - Supplies a pointer to the Loader parameter block, containing the
physical address of the SAL system table.
Return Value:
STATUS_SUCCESS is returned if the mappings were successful, and SAL/PAL calls can
be made. Otherwise, STATUS_UNSUCCESSFUL is returned if it cannot virtually map
the areas or if PAL requires a page larger than 16MB.
Assumptions: The EfiSysTableVirtualPtr is initialized prior by EfiInitialization.
--*/
{
//
// Local declarations
//
PSAL_PAL_ENTRY_POINT FwEntry;
PSST_MEMORY_DESCRIPTOR SstMemoryDescriptor;
PAP_WAKEUP_DESCRIPTOR ApWakeupDescriptor;
PVOID NextEntry, FwCodeSpace;
UCHAR Checksum;
PUCHAR CheckBuf;
ULONG index,i,SstLength;
SAL_PAL_RETURN_VALUES RetVals;
PHYSICAL_ADDRESS physicalAddr;
SAL_STATUS SALstatus;
BOOLEAN LargerThan16Mb, MmMappedSalCode, MmMappedSalData;
ULONGLONG physicalEfiST,physicalSST;
ULONGLONG physicalPAL, physicalSAL, physicalSALGP;
ULONGLONG PhysicalConfigPtr;
SST_MEMORY_LIST SalCode,SalData, FwCode;
BOOLEAN FoundSalTable;
ULONGLONG palStatus;
PAL_VERSION_STRUCT minimumPalVersion;
ULONG PalPageShift;
ULONGLONG PalPteUlong;
HalpVirtualEfiSalPalDataPointer = (PUCHAR)(ExAllocatePool (NonPagedPool, PAGE_SIZE));
HalpConfigTableVirtualPtr = HalpVirtualEfiSalPalDataPointer + ConfigTableOffset;
HalpConfigGuidVirtualPtr = HalpVirtualEfiSalPalDataPointer + ConfigGuidOffset;
HalpPhysEfiSalPalDataPointer = (PUCHAR)((MmGetPhysicalAddress(HalpVirtualEfiSalPalDataPointer)).QuadPart);
HalpConfigTablePhysPtr = HalpPhysEfiSalPalDataPointer + ConfigTableOffset;
HalpConfigGuidPhysPtr = HalpPhysEfiSalPalDataPointer + ConfigGuidOffset;
EfiConfigTableVirtualPtr = &EfiConfigTable;
EfiConfigTableSize = (ULONGLONG) sizeof(EFI_CONFIGURATION_TABLE);
HalDebugPrint(( HAL_INFO, "SAL_PAL: Entering HalpInitSalPal\n" ));
FoundSalTable = FALSE;
//
// Extract SST physical address out of EFI table
//
PhysicalConfigPtr = (ULONGLONG) EfiSysTableVirtualPtrCpy -> ConfigurationTable;
for (index = 0; (FoundSalTable == FALSE) && (index <= (EfiSysTableVirtualPtr->NumberOfTableEntries)); index = index + 1) {
physicalAddr.QuadPart = PhysicalConfigPtr;
EfiConfigTableVirtualPtr = MmMapIoSpace(physicalAddr, sizeof(EFI_CONFIGURATION_TABLE),MmCached);
if (EfiConfigTableVirtualPtr == NULL) {
HalDebugPrint(( HAL_ERROR, "SAL_PAL: Efi Config Table Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
CheckGuid = EfiConfigTableVirtualPtr -> VendorGuid;
PhysicalConfigPtr = PhysicalConfigPtr + EfiConfigTableSize;
HalDebugPrint(( HAL_INFO,
"SAL_PAL: Found new ConfigPhysPtr 0x%I64x points to tablesize 0x%I64x\n",
PhysicalConfigPtr, EfiConfigTableSize ));
// if (CheckGuid.Data1 == 0xeb9d2d32)
if (HalpCompareEfiGuid(CheckGuid, SalGuid)) {
FoundSalTable = TRUE;
}
}
physicalSST = (ULONGLONG) EfiConfigTableVirtualPtr -> VendorTable;
//
// Zero out internal structures
//
RtlZeroMemory(&HalpSalPalData, sizeof(HalpSalPalData));
RtlZeroMemory(&SalCode, sizeof(SST_MEMORY_LIST));
RtlZeroMemory(&PalCode, sizeof(SST_MEMORY_LIST));
RtlZeroMemory(&SalData, sizeof(SST_MEMORY_LIST));
RtlZeroMemory(&FwCode, sizeof(SST_MEMORY_LIST));
//
// Initialize flags
//
MmMappedSalCode = FALSE;
MmMappedSalData = FALSE;
HalpSalPalData.Status = STATUS_SUCCESS;
//
// Map the SST, get the length of entire SST to remap the rest of it
//
physicalAddr.QuadPart = physicalSST;
HalpSalPalData.SalSystemTable = MmMapIoSpace(physicalAddr,sizeof(SST_HEADER),MmCached);
if (HalpSalPalData.SalSystemTable) {
//
// Save SAL revision in case we decide to unmap it later on.
//
HalpSalPalData.SalRevision.Revision = HalpSalPalData.SalSystemTable->SalRev;
HalDebugPrint(( HAL_INFO,
"SAL_PAL: Found SST SAL v%d.%d (XXTF:%d) at PhysAddr 0x%I64x & mapped to VirtAddr 0x%I64x\n",
HalpSalPalData.SalRevision.Major,
HalpSalPalData.SalRevision.Minor,
HalpSalPalData.SalRevision.Revision,
physicalSST,
HalpSalPalData.SalSystemTable));
SstLength=HalpSalPalData.SalSystemTable->Length;
MmUnmapIoSpace((PVOID)HalpSalPalData.SalSystemTable,sizeof(SST_HEADER));
HalpSalPalData.SalSystemTable = MmMapIoSpace(physicalAddr,SstLength,MmCached);
} else {
HalDebugPrint(( HAL_INFO, "SAL_PAL: Couldn't virtually map SST (PhysAddr: 0x%I64x)\n", physicalSST ));
HalpSalPalData.Status = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO,
"SAL_PAL: Mapped all of SST for PhysAddr 0x%I64x at VirtAddr 0x%I64x for %d bytes\n",
physicalSST,
HalpSalPalData.SalSystemTable, SstLength ));
//
// Because the checksum is maybe not computed and verified by the OS loader before the
// hand-off to the kernel, let's do a checksum and verify it.
//
Checksum=0;
CheckBuf= (PUCHAR) HalpSalPalData.SalSystemTable;
for (i=0; i < SstLength; i++) {
Checksum += CheckBuf[i];
}
if (Checksum) {
HalDebugPrint(( HAL_ERROR, "SAL_PAL: Checksum is BAD with value of %d, should be 0\n", Checksum ));
MmUnmapIoSpace((PVOID)HalpSalPalData.SalSystemTable,SstLength);
HalpSalPalData.Status = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "SAL_PAL: Checksum is GOOD\n" ));
//
// Pull out the SAL/PAL entrypoint and memory descriptor information we need by iterating
// over the entire SST.
//
NextEntry = (PUCHAR)(((PUCHAR) HalpSalPalData.SalSystemTable) + sizeof(SST_HEADER));
for (i = 0; i < HalpSalPalData.SalSystemTable->EntryCount; i++) {
switch ( *(PUCHAR)NextEntry ) {
case SAL_PAL_ENTRY_POINT_TYPE:
//
// It is assumed only one Entry Point Type in the SST but we do not currently
// impose this in this code.
//
FwEntry = (PSAL_PAL_ENTRY_POINT)NextEntry;
physicalSAL = FwEntry->SalEntryPoint;
physicalPAL = FwEntry->PalEntryPoint;
physicalSALGP = FwEntry->GlobalPointer;
HalDebugPrint(( HAL_INFO,
"SAL_PAL: SAL_PROC PhysAddr at 0x%I64x\n"
"SAL_PAL: PAL_PROC PhysAddr at 0x%I64x\n"
"SAL_PAL: SAL_GP PhysAddr at 0x%I64x\n",
physicalSAL,
physicalPAL,
physicalSALGP
));
((PSAL_PAL_ENTRY_POINT)NextEntry)++;
break;
case SST_MEMORY_DESCRIPTOR_TYPE:
//
// It is assumed only one REGULAR_MEMORY entry for PAL_CODE_MEM, SAL_CODE_MEM and
// SAL_DATA_MEM. However, we do not currently impose this in the code.
//
SstMemoryDescriptor = (PSST_MEMORY_DESCRIPTOR)NextEntry;
HalDebugPrint(( HAL_INFO,
"SAL_PAL: MemDesc for PhysAddr 0x%I64x (%I64d KB) - VaReg:%d,Type:%d,Use:%d\n",
SstMemoryDescriptor->MemoryAddress,(ULONGLONG)SstMemoryDescriptor->Length * 4,
SstMemoryDescriptor->NeedVaReg,SstMemoryDescriptor->MemoryType,SstMemoryDescriptor->MemoryUsage
));
if (SstMemoryDescriptor->MemoryType == REGULAR_MEMORY) {
switch(SstMemoryDescriptor->MemoryUsage) {
case PAL_CODE_MEM:
PalCode.PhysicalAddress = SstMemoryDescriptor->MemoryAddress;
PalCode.Length = SstMemoryDescriptor->Length << 12;
PalCode.NeedVaReg = SstMemoryDescriptor->NeedVaReg;
break;
case SAL_CODE_MEM:
SalCode.PhysicalAddress = SstMemoryDescriptor->MemoryAddress;
SalCode.Length = SstMemoryDescriptor->Length << 12;
SalCode.NeedVaReg = SstMemoryDescriptor->NeedVaReg;
break;
case SAL_DATA_MEM:
SalData.PhysicalAddress = SstMemoryDescriptor->MemoryAddress;
SalData.Length = SstMemoryDescriptor->Length << 12;
SalData.NeedVaReg = SstMemoryDescriptor->NeedVaReg;
break;
case FW_RESERVED:
// We need to virtually map and count these areas for NVM access.
// We should keep a list of all such regions and would allocate an
// array of virtual address mappings for each NVM region for later
// use when accessing variables.
// No NVM support currently, so ignore for now.
break;
default:
break;
}
} else if ((SstMemoryDescriptor->MemoryType == FIRMWARE_CODE) &&
(SstMemoryDescriptor->MemoryUsage == FW_SAL_PAL)) {
FwCode.PhysicalAddress = SstMemoryDescriptor->MemoryAddress;
FwCode.Length = SstMemoryDescriptor->Length << 12;
FwCode.NeedVaReg = SstMemoryDescriptor->NeedVaReg;
}
((PSST_MEMORY_DESCRIPTOR)NextEntry)++;
break;
case PLATFORM_FEATURES_TYPE:
// We ignore this entry for now...
((PPLATFORM_FEATURES)NextEntry)++;
break;
case TRANSLATION_REGISTER_TYPE:
// We ignore this entry for now...
((PTRANSLATION_REGISTER)NextEntry)++;
break;
case PTC_COHERENCE_DOMAIN_TYPE:
// We ignore this entry for now...
((PPTC_COHERENCE_DOMAIN)NextEntry)++;
break;
case AP_WAKEUP_DESCRIPTOR_TYPE:
ApWakeupDescriptor = (PAP_WAKEUP_DESCRIPTOR)NextEntry;
HalpOsBootRendezVector = (ULONG) ApWakeupDescriptor->WakeupVector;
if ((HalpOsBootRendezVector < 0x100 ) && (HalpOsBootRendezVector > 0xF)) {
HalDebugPrint(( HAL_INFO, "SAL_PAL: Found Valid WakeupVector: 0x%x\n",
HalpOsBootRendezVector ));
} else {
HalDebugPrint(( HAL_INFO, "SAL_PAL: Invalid WakeupVector.Using Default: 0x%x\n",
DEFAULT_OS_RENDEZ_VECTOR ));
HalpOsBootRendezVector = DEFAULT_OS_RENDEZ_VECTOR;
}
((PAP_WAKEUP_DESCRIPTOR)NextEntry)++;
break;
default:
HalDebugPrint(( HAL_ERROR,
"SAL_PAL: EntryType %d is bad at VirtAddr 0x%I64x, Aborting\n",
*(PUCHAR)NextEntry,
NextEntry ));
// An unknown SST Type was found, so we can't continue
MmUnmapIoSpace((PVOID)HalpSalPalData.SalSystemTable,SstLength);
HalpSalPalData.Status = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
}
//
// Initialize the HAL private spinlocks
//
// - HalpSalSpinLock, HalpSalStateInfoSpinLock are used for MP synchronization of the
// SAL calls that are not MP-safe.
// - HalpMcaSpinLock is used for defining an MCA monarch and MP synchrnonization of shared
// HAL MCA resources during OS_MCA calls.
//
KeInitializeSpinLock(&HalpSalSpinLock);
KeInitializeSpinLock(&HalpSalStateInfoSpinLock);
KeInitializeSpinLock(&HalpMcaSpinLock);
KeInitializeSpinLock(&HalpInitSpinLock);
KeInitializeSpinLock(&HalpCmcSpinLock);
KeInitializeSpinLock(&HalpCpeSpinLock);
//
// Unmap the SalSystemTable since we will be mapping all PAL and SAL code, and SAL data area
// which includes SST.
//
// TEMP TEMP TEMP. we are not unmapping
//
// MmUnmapIoSpace((PVOID)HalpSalPalData.SalSystemTable,SstLength);
//
// Make sure we found all of our entries for mapping
//
if (!(SalCode.Length && SalData.Length && PalCode.Length && FwCode.Length)) {
HalDebugPrint(( HAL_ERROR, "SAL_PAL: One or more of the SalCode, SalData, PalCode, or FwCode spaces not in SST\n" ));
HalpSalPalData.Status = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
//
// Since PAL will need to use only 1 TR, figure out the page size to use for mapping PAL,
// starting at 16KB, and trying 64KB, 256KB, 1MB, 4MB, and 16MB.
//
HalpSalPalData.PalTrSize = SIZE_IN_BYTES_16KB;
PalTrMask = MASK_16KB;
PalPageShift = 14;
LargerThan16Mb = 1;
while (PalTrMask >= MASK_16MB) {
// Check if it entirely fits in the given page
if ( (PalCode.PhysicalAddress + PalCode.Length) <=
( (PalCode.PhysicalAddress & PalTrMask) + HalpSalPalData.PalTrSize) ) {
LargerThan16Mb = 0;
break;
}
// Try the next page size, incrementing in multiples of 4 from 16KB to 16MB
PalTrMask <<= 2;
HalpSalPalData.PalTrSize <<= 2;
PalPageShift += 2;
}
//
// If PAL requires a page size of larger than 16MB, fail.
//
if (LargerThan16Mb) {
HalDebugPrint(( HAL_ERROR, "SAL_PAL: More than 16MB was required to map PAL" ));
HalpSalPalData.Status = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
//
// Set the page size aligned address that fits PAL
//
HalpSalPalData.PalTrBase = PalCode.PhysicalAddress & PalTrMask;
HalDebugPrint(( HAL_INFO,
"SAL_PAL: For the PAL code located at 0x%I64x - length 0x%I64x, the TrMask is 0x%I64x and TrSize is %d Kbytes\n",
PalCode.PhysicalAddress,
PalCode.Length,
PalTrMask,
HalpSalPalData.PalTrSize/1024 ));
//
// Map the PAL code at a architected address reserved for SAL/PAL
//
// PAL is known to have an alignment of 256KB.
//
PalCode.VirtualAddress = HAL_PAL_VIRTUAL_ADDRESS + (PalCode.PhysicalAddress - HalpSalPalData.PalTrBase);
//
// Setup the ITR and DTRs to map PAL
//
PalPteUlong = HalpSalPalData.PalTrBase | VALID_KERNEL_EXECUTE_PTE;
KeFillFixedEntryTb((PHARDWARE_PTE)&PalPteUlong,
(PVOID)HAL_PAL_VIRTUAL_ADDRESS,
PalPageShift,
INST_TB_PAL_INDEX);
KeFillFixedEntryTb((PHARDWARE_PTE)&PalPteUlong,
(PVOID)HAL_PAL_VIRTUAL_ADDRESS,
PalPageShift,
DATA_TB_PAL_INDEX);
//
// SAL has no specific virtual mapping translation requirement. It can use TCs to map the
// SAL code regions and the SAL data regions.
// SAL requires virtual address translations for descriptors like PAL code areas (to be able
// to call PAL for SAL), SAL code, SAL data and NVM areas. SAL_UPDATE_PAL requires a virtual
// mapping of the FW ROM area to perform the update. The latter could be done using TCs.
//
// First check if the SAL code fits in the same page as PAL.
// If it does, then use that mapping, otherwise, create a mapping using MmMapIoSpace and
// mark it executable.
//
if ((SalCode.PhysicalAddress >= HalpSalPalData.PalTrBase) &&
((SalCode.PhysicalAddress + SalCode.Length) <= (HalpSalPalData.PalTrBase + HalpSalPalData.PalTrSize))) {
SalCode.VirtualAddress = HAL_PAL_VIRTUAL_ADDRESS + (SalCode.PhysicalAddress - HalpSalPalData.PalTrBase);
}
else {
physicalAddr.QuadPart = SalCode.PhysicalAddress;
SalCode.VirtualAddress = (ULONGLONG) MmMapIoSpace(physicalAddr, (ULONG)SalCode.Length, MmCached);
if (!SalCode.VirtualAddress) {
HalDebugPrint(( HAL_ERROR, "SAL_PAL: Failed to map the SAL code region\n" ));
goto SalPalCleanup;
}
HalDebugPrint(( HAL_INFO, "SAL_PAL: Starting to mark %d bytes of SAL code executable\n", SalCode.Length ));
//
// Temporarily commented out for Mm bug of clearing the dirty bit.
// We will enable this.
//
// MmSetPageProtection((PVOID)SalCode.VirtualAddress,
// SalCode.Length,
// 0x40 /* PAGE_EXECUTE_READWRITE */);
//
MmMappedSalCode=TRUE;
}
//
// Same check for SAL data:
// Check if the SAL data fits in the same page as PAL.
// If it does, then use that mapping, otherwise, create a mapping using MmMapIoSpace
//
if ((SalData.PhysicalAddress >= HalpSalPalData.PalTrBase) &&
((SalData.PhysicalAddress + SalData.Length) <= (HalpSalPalData.PalTrBase + HalpSalPalData.PalTrSize))) {
SalData.VirtualAddress = HAL_PAL_VIRTUAL_ADDRESS + (SalData.PhysicalAddress - HalpSalPalData.PalTrBase);
} else {
physicalAddr.QuadPart = SalData.PhysicalAddress;
SalData.VirtualAddress = (ULONGLONG) MmMapIoSpace(physicalAddr, (ULONG)SalData.Length, MmCached);
if (!SalData.VirtualAddress) {
HalDebugPrint(( HAL_ERROR, "SAL_PAL: Failed to map the SAL data region\n" ));
goto SalPalCleanup;
}
MmMappedSalData=TRUE;
}
HalDebugPrint(( HAL_INFO, "SAL_PAL: Mapped SAL code for PhysAddr 0x%I64x at VirtAddr 0x%I64x for %d bytes\n"
"SAL_PAL: Mapped PAL code for PhysAddr 0x%I64x at VirtAddr 0x%I64x for %d bytes\n"
"SAL_PAL: Mapped SAL data for PhysAddr 0x%I64x at VirtAddr 0x%I64x for %d bytes\n",
SalCode.PhysicalAddress, SalCode.VirtualAddress, SalCode.Length,
PalCode.PhysicalAddress, PalCode.VirtualAddress, PalCode.Length,
SalData.PhysicalAddress, SalData.VirtualAddress, SalData.Length
));
// Store virtual address pointers of SST, SAL_PROC, PAL_PROC, and GP for use later
// (VirtualAddr = VirtualAddrBase + PhysicalOffset) where (PhysicalOffset = physicalAddr - PhysicalBase)
// We are not ensuring SST is within the SAL code area
// ASSERT((physicalSST > SalCode.PhysicalAddress) && (physicalSST < (SalCode.PhysicalAddress + SalCode.Length)));
// HalpSalPalData.SalSystemTable = (PSST_HEADER) (SalCode.VirtualAddress + (physicalSST - SalCode.PhysicalAddress));
HalpSalProcPointer = (ULONGLONG) (SalCode.VirtualAddress + (physicalSAL - SalCode.PhysicalAddress));
HalpSalProcGlobalPointer = (ULONGLONG) (SalCode.VirtualAddress + (physicalSALGP - SalCode.PhysicalAddress));
HalpPalProcPointer = (ULONGLONG) (PalCode.VirtualAddress + (physicalPAL - PalCode.PhysicalAddress));
//
// Go map the firmware code space
//
// SAL_UPDATE_PAL requires a virtual mapping of the ROM area to perform the update.
//
HalDebugPrint(( HAL_INFO, "SAL_PAL: Mapping and Registering Virtual Address of FwCode area\n" ));
physicalAddr.QuadPart = FwCode.PhysicalAddress;
FwCodeSpace = MmMapIoSpace(physicalAddr, (ULONG) FwCode.Length, MmCached);
//
// Get the PAL version.
//
palStatus = HalCallPal(PAL_VERSION,
0,
0,
0,
NULL,
&minimumPalVersion.ReturnValue,
&HalpSalPalData.PalVersion.ReturnValue,
NULL);
if (palStatus != SAL_STATUS_SUCCESS) {
HalDebugPrint(( HAL_ERROR, "SAL_PAL: Get PAL version number failed. Status = d\n" ));
}
//
// Retrieve SmBiosVersion and save the pointer into HalpSalPalData. Note:
// HalpGetSmBiosVersion will allocate a buffer for SmBiosVersion.
//
HalpSalPalData.SmBiosVersion = HalpGetSmBiosVersion(LoaderBlock);
//
// Determine and Initialize HAL private SAL/PAL WorkArounds if any.
//
HalpInitSalPalWorkArounds();
// We completed initialization
HalDebugPrint(( HAL_INFO, "SAL_PAL: Exiting HalpSalPalInitialization with SUCCESS\n" ));
return HalpSalPalData.Status;
// Cleanup the mappings and get out of here...
SalPalCleanup:
if (MmMappedSalData) {
MmUnmapIoSpace((PVOID) SalData.VirtualAddress, (ULONG) SalData.Length);
}
if (MmMappedSalCode) {
MmUnmapIoSpace((PVOID) SalCode.VirtualAddress, (ULONG) SalCode.Length);
}
HalDebugPrint(( HAL_ERROR, "SAL_PAL: Exiting HalpSalPalInitialization with ERROR!!!\n" ));
HalpSalPalData.Status = STATUS_UNSUCCESSFUL;
return HalpSalPalData.Status;
} // HalpInitSalPal()
PUCHAR
HalpGetSmBiosVersion (
IN PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
This function retrieves the SmBiosVersion string from the BIOS structure
table, allocates memory for the buffer, copies the string to the buffer,
and returns a pointer to this buffer. If unsuccessful, this function
returns a NULL.
Arguments:
LoaderBlock - Pointer to the loader parameter block.
Return Value:
Pointer to a buffer that contains SmBiosVersion string.
--*/
{
PSMBIOS_EPS_HEADER SMBiosEPSHeader;
PDMIBIOS_EPS_HEADER DMIBiosEPSHeader;
USHORT SMBiosTableLength;
USHORT SMBiosTableNumberStructures;
PUCHAR SmBiosVersion;
PHYSICAL_ADDRESS SMBiosTablePhysicalAddress;
PUCHAR SMBiosDataVirtualAddress;
UCHAR Type;
UCHAR Length;
UCHAR BiosVersionStringNumber;
UCHAR chr;
USHORT i;
PUCHAR pBuffer;
BOOLEAN Found;
if (LoaderBlock->Extension->Size < sizeof(LOADER_PARAMETER_EXTENSION)) {
HalDebugPrint((HAL_ERROR, "HalpGetSmBiosVersion: Invalid LoaderBlock extension size\n"));
return NULL;
}
SMBiosEPSHeader = (PSMBIOS_EPS_HEADER)LoaderBlock->Extension->SMBiosEPSHeader;
//
// Verify SM Bios Header signature
//
if ((SMBiosEPSHeader == NULL) || (strncmp((PUCHAR)SMBiosEPSHeader, "_SM_", 4) != 0)) {
HalDebugPrint((HAL_ERROR, "HalpGetSmBiosVersion: Invalid SMBiosEPSHeader\n"));
return NULL;
}
DMIBiosEPSHeader = (PDMIBIOS_EPS_HEADER)&SMBiosEPSHeader->Signature2[0];
//
// Verify DMI Bios Header signature
//
if ((DMIBiosEPSHeader == NULL) || (strncmp((PUCHAR)DMIBiosEPSHeader, "_DMI_", 5) != 0)) {
HalDebugPrint((HAL_ERROR, "HalpGetSmBiosVersion: Invalid DMIBiosEPSHeader\n"));
return NULL;
}
SMBiosTablePhysicalAddress.HighPart = 0;
SMBiosTablePhysicalAddress.LowPart = DMIBiosEPSHeader->StructureTableAddress;
SMBiosTableLength = DMIBiosEPSHeader->StructureTableLength;
SMBiosTableNumberStructures = DMIBiosEPSHeader->NumberStructures;
//
// Map SMBiosTable to virtual address
//
SMBiosDataVirtualAddress = MmMapIoSpace(SMBiosTablePhysicalAddress,
SMBiosTableLength,
MmCached
);
if (!SMBiosDataVirtualAddress) {
HalDebugPrint((HAL_ERROR, "HalpGetSmBiosVersion: Failed to map SMBiosTablePhysicalAddress\n"));
return NULL;
}
//
// The Spec doesn't say that SmBios Type 0 structure has to be the first
// structure at this entry point... so we have to traverse through memory
// to find the right one.
//
i = 0;
Found = FALSE;
while (i < SMBiosTableNumberStructures && !Found) {
i++;
Type = (UCHAR)SMBiosDataVirtualAddress[SMBIOS_STRUCT_HEADER_TYPE_FIELD];
if (Type == 0) {
Found = TRUE;
}
else {
//
// Advance to the next structure
//
SMBiosDataVirtualAddress += SMBiosDataVirtualAddress[SMBIOS_STRUCT_HEADER_LENGTH_FIELD];
// Get pass trailing string-list by looking for a double-null
while (*(PUSHORT)SMBiosDataVirtualAddress != 0) {
SMBiosDataVirtualAddress++;
}
SMBiosDataVirtualAddress += 2;
}
}
if (!Found) {
HalDebugPrint((HAL_ERROR, "HalpGetSmBiosVersion: Could not find Type 0 structure\n"));
return NULL;
}
//
// Extract BIOS Version string from the SmBios Type 0 Structure
//
Length = SMBiosDataVirtualAddress[SMBIOS_STRUCT_HEADER_LENGTH_FIELD];
BiosVersionStringNumber = SMBiosDataVirtualAddress[SMBIOS_TYPE0_STRUCT_BIOSVER_FIELD];
//
// Text strings begin right after the formatted portion of the structure.
//
pBuffer = (PUCHAR)&SMBiosDataVirtualAddress[Length];
//
// Get to the beginning of SmBiosVersion string
//
for (i = 0; i < BiosVersionStringNumber - 1; i++) {
do {
chr = *pBuffer;
pBuffer++;
} while (chr != '\0');
}
//
// Allocate memory for SmBiosVersion string and copy content of
// pBuffer to SmBiosVersion.
//
SmBiosVersion = ExAllocatePool(NonPagedPool, strlen(pBuffer)+1);
if (!SmBiosVersion) {
HalDebugPrint((HAL_ERROR, "HalpGetSmBiosVersion: Failed to allocate memory for SmBiosVersion\n"));
return NULL;
}
strcpy(SmBiosVersion, pBuffer);
MmUnmapIoSpace(SMBiosDataVirtualAddress,
SMBiosTableLength
);
return SmBiosVersion;
}
VOID
HalpInitSalPalNonBsp(
VOID
)
/*++
Routine Description:
This function is called for the non-BSP processors to simply set up the same
TR registers that HalpInitSalPal does for the BSP processor.
Arguments:
None
Return Value:
None
--*/
{
ULONG PalPageShift;
ULONGLONG PalPteUlong;
ULONGLONG PalTrSize;
// If we successfully initialized in HalpSalPalInitialization, then set-up the TR
if (!NT_SUCCESS(HalpSalPalData.Status)) {
return;
}
PalTrSize = HalpSalPalData.PalTrSize;
PalPageShift = 14;
while (PalTrSize > ((ULONGLONG)1 << PalPageShift)) {
PalPageShift += 2;
}
PalPteUlong = HalpSalPalData.PalTrBase | VALID_KERNEL_EXECUTE_PTE;
KeFillFixedEntryTb((PHARDWARE_PTE)&PalPteUlong,
(PVOID)HAL_PAL_VIRTUAL_ADDRESS,
PalPageShift,
INST_TB_PAL_INDEX);
KeFillFixedEntryTb((PHARDWARE_PTE)&PalPteUlong,
(PVOID)HAL_PAL_VIRTUAL_ADDRESS,
PalPageShift,
DATA_TB_PAL_INDEX);
} // HalpInitSalPalNonBsp()
NTSTATUS
HalpEfiInitialization(
PLOADER_PARAMETER_BLOCK LoaderBlock
)
/*++
Routine Description:
This function
Arguments:
LoaderBlock - Supplies a pointer to the Loader parameter block, containing the
physical address of the EFI system table.
Return Value:
STATUS_SUCCESS is returned if the mappings were successful, and EFI calls can
be made. Otherwise, STATUS_UNSUCCESSFUL is returned.
--*/
{
//
// Local declarations
//
EFI_MEMORY_DESCRIPTOR *efiMapEntryPtr, *efiVirtualMemoryMapPtr;
EFI_STATUS status;
ULONGLONG index, mapEntries;
ULONGLONG physicalEfiST, physicalEfiMemoryMapPtr, physicalRunTimeServicesPtr;
ULONGLONG physicalEfiGetVariable, physicalEfiGetNextVariableName, physicalEfiSetVariable;
ULONGLONG physicalEfiGetTime, physicalEfiSetTime;
ULONGLONG physicalEfiSetVirtualAddressMap, physicalEfiResetSystem;
PHYSICAL_ADDRESS physicalAddr;
ULONGLONG physicalPlabel_Fpswa;
FPSWA_INTERFACE *interfacePtr;
PVOID tmpPtr;
//
// First, get the physical address of the fpswa entry point PLABEL.
//
if (LoaderBlock->u.Ia64.FpswaInterface != (ULONG_PTR) NULL) {
physicalAddr.QuadPart = LoaderBlock->u.Ia64.FpswaInterface;
interfacePtr = MmMapIoSpace(physicalAddr,
sizeof(FPSWA_INTERFACE),
MmCached
);
if (interfacePtr == NULL) {
HalDebugPrint(( HAL_FATAL_ERROR, "FpswaInterfacePtr is Null. Efi handle not available\n"));
KeBugCheckEx(FP_EMULATION_ERROR, 0, 0, 0, 0);
return STATUS_UNSUCCESSFUL;
}
physicalPlabel_Fpswa = (ULONGLONG)(interfacePtr->Fpswa);
}
else {
HalDebugPrint(( HAL_FATAL_ERROR, "HAL: EFI FpswaInterface is not available\n"));
KeBugCheckEx(FP_EMULATION_ERROR, 0, 0, 0, 0);
return STATUS_UNSUCCESSFUL;
}
physicalEfiST = LoaderBlock->u.Ia64.EfiSystemTable;
physicalAddr.QuadPart = physicalEfiST;
EfiSysTableVirtualPtr = MmMapIoSpace( physicalAddr, sizeof(EFI_SYSTEM_TABLE), MmCached);
if (EfiSysTableVirtualPtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: EfiSystem Table Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
EfiSysTableVirtualPtrCpy = EfiSysTableVirtualPtr;
physicalRunTimeServicesPtr = (ULONGLONG) EfiSysTableVirtualPtr->RuntimeServices;
physicalAddr.QuadPart = physicalRunTimeServicesPtr;
EfiRSVirtualPtr = MmMapIoSpace(physicalAddr, sizeof(EFI_RUNTIME_SERVICES),MmCached);
if (EfiRSVirtualPtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: Run Time Table Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
EfiMemoryMapSize = LoaderBlock->u.Ia64.EfiMemMapParam.MemoryMapSize;
EfiDescriptorSize = LoaderBlock->u.Ia64.EfiMemMapParam.DescriptorSize;
EfiDescriptorVersion = LoaderBlock->u.Ia64.EfiMemMapParam.DescriptorVersion;
physicalEfiMemoryMapPtr = (ULONGLONG)LoaderBlock->u.Ia64.EfiMemMapParam.MemoryMap;
physicalAddr.QuadPart = physicalEfiMemoryMapPtr;
efiVirtualMemoryMapPtr = MmMapIoSpace (physicalAddr, EfiMemoryMapSize, MmCached);
if (efiVirtualMemoryMapPtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: Virtual Set Memory Map Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
//
// #define VENDOR_SPECIFIC_GUID \
// { 0xa3c72e56, 0x4c35, 0x11d3, 0x8a, 0x03, 0x0, 0xa0, 0xc9, 0x06, 0xad, 0xec }
//
VendorGuid.Data1 = 0x8be4df61;
VendorGuid.Data2 = 0x93ca;
VendorGuid.Data3 = 0x11d2;
VendorGuid.Data4[0] = 0xaa;
VendorGuid.Data4[1] = 0x0d;
VendorGuid.Data4[2] = 0x00;
VendorGuid.Data4[3] = 0xe0;
VendorGuid.Data4[4] = 0x98;
VendorGuid.Data4[5] = 0x03;
VendorGuid.Data4[6] = 0x2b;
VendorGuid.Data4[7] = 0x8c;
HalDebugPrint(( HAL_INFO,
"HAL: EFI SystemTable VA = 0x%I64x, PA = 0x%I64x\n"
"HAL: EFI RunTimeServices VA = 0x%I64x, PA = 0x%I64x\n"
"HAL: EFI MemoryMapPtr VA = 0x%I64x, PA = 0x%I64x\n"
"HAL: EFI MemoryMap Size = 0x%I64x\n"
"HAL: EFI Descriptor Size = 0x%I64x\n",
EfiSysTableVirtualPtr,
physicalEfiST,
EfiRSVirtualPtr,
physicalRunTimeServicesPtr,
efiVirtualMemoryMapPtr,
physicalEfiMemoryMapPtr,
EfiMemoryMapSize,
EfiDescriptorSize
));
// GetVariable
physicalEfiGetVariable = (ULONGLONG) (EfiRSVirtualPtr -> GetVariable);
physicalAddr.QuadPart = physicalEfiGetVariable;
EfiVirtualGetVariablePtr = MmMapIoSpace (physicalAddr, sizeof(PLABEL_DESCRIPTOR), MmCached);
if (EfiVirtualGetVariablePtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: EfiGetVariable Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "HAL: EFI GetVariable VA = 0x%I64x, PA = 0x%I64x\n",
EfiVirtualGetVariablePtr, physicalEfiGetVariable ));
// GetNextVariableName
physicalEfiGetNextVariableName = (ULONGLONG) (EfiRSVirtualPtr -> GetNextVariableName);
physicalAddr.QuadPart = physicalEfiGetNextVariableName;
EfiVirtualGetNextVariableNamePtr = MmMapIoSpace (physicalAddr,sizeof(PLABEL_DESCRIPTOR),MmCached);
if (EfiVirtualGetNextVariableNamePtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: EfiVirtual Get Next Variable Name Ptr Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
//SetVariable
physicalEfiSetVariable = (ULONGLONG) (EfiRSVirtualPtr -> SetVariable);
physicalAddr.QuadPart = physicalEfiSetVariable;
EfiVirtualSetVariablePtr = MmMapIoSpace (physicalAddr, sizeof(PLABEL_DESCRIPTOR), MmCached);
if (EfiVirtualSetVariablePtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: EfiVariableSetVariable Pointer dr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "HAL: EFI Set Variable VA = 0x%I64x, PA = 0x%I64x\n",
EfiVirtualSetVariablePtr, physicalEfiSetVariable ));
//GetTime
physicalEfiGetTime = (ULONGLONG) (EfiRSVirtualPtr -> GetTime);
physicalAddr.QuadPart = physicalEfiGetTime;
EfiVirtualGetTimePtr = MmMapIoSpace (physicalAddr, sizeof(PLABEL_DESCRIPTOR), MmCached);
if (EfiVirtualGetTimePtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: EfiGetTime Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "HAL: EFI GetTime VA = 0x%I64x, PA = 0x%I64x\n",
EfiVirtualGetTimePtr, physicalEfiGetTime ));
//SetTime
physicalEfiSetTime = (ULONGLONG) (EfiRSVirtualPtr -> SetTime);
physicalAddr.QuadPart = physicalEfiSetTime;
EfiVirtualSetTimePtr = MmMapIoSpace (physicalAddr, sizeof(PLABEL_DESCRIPTOR), MmCached);
if (EfiVirtualSetTimePtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: EfiSetTime Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "HAL: EFI SetTime VA = 0x%I64x, PA = 0x%I64x\n",
EfiVirtualSetTimePtr, physicalEfiSetTime ));
//SetVirtualAddressMap
physicalEfiSetVirtualAddressMap = (ULONGLONG) (EfiRSVirtualPtr -> SetVirtualAddressMap);
physicalAddr.QuadPart = physicalEfiSetVirtualAddressMap;
EfiSetVirtualAddressMapPtr = MmMapIoSpace (physicalAddr, sizeof(PLABEL_DESCRIPTOR), MmCached);
if (EfiSetVirtualAddressMapPtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: Efi Set VirtualMapPointer Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "HAL: EFI SetVirtualAddressMap VA = 0x%I64x, PA = 0x%I64x\n",
EfiSetVirtualAddressMapPtr, physicalEfiSetVirtualAddressMap ));
//ResetSystem
physicalEfiResetSystem = (ULONGLONG) (EfiRSVirtualPtr -> ResetSystem);
physicalAddr.QuadPart = physicalEfiResetSystem;
EfiResetSystemPtr = MmMapIoSpace (physicalAddr, sizeof(PLABEL_DESCRIPTOR), MmCached);
if (EfiResetSystemPtr == NULL) {
HalDebugPrint(( HAL_ERROR,"HAL: Efi Reset System Virtual Addr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "HAL: EFI ResetSystem VA = 0x%I64x, PA = 0x%I64x\n",
EfiResetSystemPtr, physicalEfiResetSystem ));
//
// Allocate one page for Backing store
//
physicalAddr.QuadPart = 0xffffffffffffffffI64;
tmpPtr = MmAllocateContiguousMemory( PAGE_SIZE, physicalAddr );
HalpPhysBSPointer = (PULONGLONG)( ( MmGetPhysicalAddress( tmpPtr ) ).QuadPart );
//
// Allocate one page for the Stack, then make sure the base pointer
// is assigned to the high address of this block since the stack
// grows down.
//
tmpPtr = MmAllocateContiguousMemory( PAGE_SIZE, physicalAddr );
HalpPhysStackPointer = (PULONGLONG)( ( MmGetPhysicalAddress( tmpPtr ) ).QuadPart + PAGE_SIZE - 16);
HalpVirtualCommonDataPointer = (PUCHAR)(ExAllocatePool (NonPagedPool, PAGE_SIZE));
HalpVariableNameVirtualPtr = HalpVirtualCommonDataPointer + VariableNameOffset;
HalpVendorGuidVirtualPtr = HalpVirtualCommonDataPointer + VendorGuidOffset;
HalpVariableAttributesVirtualPtr = HalpVirtualCommonDataPointer + AttributeOffset;
HalpDataSizeVirtualPtr = HalpVirtualCommonDataPointer + DataSizeOffset;
HalpDataVirtualPtr = HalpVirtualCommonDataPointer + DataBufferOffset;
HalpCommonDataEndPtr = HalpVirtualCommonDataPointer + EndOfCommonDataOffset;
HalpMemoryMapSizeVirtualPtr = HalpVirtualCommonDataPointer + MemoryMapSizeOffset;
HalpMemoryMapVirtualPtr = (PUCHAR)(HalpVirtualCommonDataPointer + MemoryMapOffset);
HalpDescriptorSizeVirtualPtr = HalpVirtualCommonDataPointer + DescriptorSizeOffset;
HalpDescriptorVersionVirtualPtr = HalpVirtualCommonDataPointer + DescriptorVersionOffset;
HalpPhysCommonDataPointer = (PUCHAR)((MmGetPhysicalAddress(HalpVirtualCommonDataPointer)).QuadPart);
HalpVariableNamePhysPtr = HalpPhysCommonDataPointer + VariableNameOffset;
HalpVendorGuidPhysPtr = HalpPhysCommonDataPointer + VendorGuidOffset;
HalpVariableAttributesPhysPtr = HalpPhysCommonDataPointer + AttributeOffset;
HalpDataSizePhysPtr = HalpPhysCommonDataPointer + DataSizeOffset;
HalpDataPhysPtr = HalpPhysCommonDataPointer + DataBufferOffset;
HalpMemoryMapSizePhysPtr = HalpPhysCommonDataPointer + MemoryMapSizeOffset;
HalpMemoryMapPhysPtr = HalpPhysCommonDataPointer + MemoryMapOffset;
HalpDescriptorSizePhysPtr = HalpPhysCommonDataPointer + DescriptorSizeOffset;
HalpDescriptorVersionPhysPtr = HalpPhysCommonDataPointer + DescriptorVersionOffset;
AttributePtr = &EfiAttribute;
DataSizePtr = &EfiDataSize;
RtlCopyMemory ((PULONGLONG)HalpMemoryMapVirtualPtr,
efiVirtualMemoryMapPtr,
(ULONG)(EfiMemoryMapSize)
);
//
// Now, extract SAL, PAL information from the loader parameter block and
// initializes HAL SAL, PAL definitions.
//
// N.B 10/2000:
// We do not check the return status of HalpInitSalPal(). We should. FIXFIX.
// In case of failure, we currently flag HalpSalPalData.Status as unsuccessful.
//
HalpInitSalPal(LoaderBlock);
//
// Initialize Spin Lock
//
KeInitializeSpinLock(&EFIMPLock);
ASSERT (EfiDescriptorVersion == EFI_MEMORY_DESCRIPTOR_VERSION);
// if (EfiDescriptorVersion != EFI_MEMORY_DESCRIPTION_VERSION) {
// HalDebugPrint(HAL_ERROR,("Efi Memory Map Pointer VAddr is NULL\n"));
// EfiInitStatus = STATUS_UNSUCCESSFUL;
// return STATUS_UNSUCCESSFUL;
// }
HalDebugPrint(( HAL_INFO, "HAL: Creating EFI virtual address mapping\n" ));
efiMapEntryPtr = efiVirtualMemoryMapPtr;
if (efiMapEntryPtr == NULL) {
HalDebugPrint(( HAL_ERROR, "HAL: Efi Memory Map Pointer VAddr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
mapEntries = EfiMemoryMapSize/EfiDescriptorSize;
HalDebugPrint(( HAL_INFO, "HAL: EfiMemoryMapSize: 0x%I64x & EfiDescriptorSize: 0x%I64x & #of entries: 0x%I64x\n",
EfiMemoryMapSize,
EfiDescriptorSize,
mapEntries ));
HalDebugPrint(( HAL_INFO, "HAL: Efi RunTime Attribute will be printed as 1\n" ));
for (index = 0; index < mapEntries; index= index + 1) {
BOOLEAN attribute = 0;
physicalAddr.QuadPart = efiMapEntryPtr -> PhysicalStart;
if ((efiMapEntryPtr->Attribute) & EFI_MEMORY_RUNTIME) {
attribute = 1;
switch (efiMapEntryPtr->Type) {
case EfiRuntimeServicesData:
case EfiReservedMemoryType:
efiMapEntryPtr->VirtualStart = (ULONGLONG) (MmMapIoSpace (physicalAddr,
(SIZE_T)((EFI_PAGE_SIZE)*(efiMapEntryPtr->NumberOfPages)),
(efiMapEntryPtr->Attribute & EFI_MEMORY_WB) ? MmCached : MmNonCached
));
if ((efiMapEntryPtr->VirtualStart) == 0) {
HalDebugPrint(( HAL_ERROR, "HAL: Efi RunTimeSrvceData/RsrvdMemory area VAddr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO,
"HAL: Efi attribute %d & Type 0x%I64x with # of 4k pages 0x%I64x at PA 0x%I64x & mapped to VA 0x%I64x\n",
attribute,
efiMapEntryPtr->Type,
efiMapEntryPtr->NumberOfPages,
efiMapEntryPtr->PhysicalStart,
efiMapEntryPtr->VirtualStart ));
break;
case EfiPalCode:
efiMapEntryPtr->VirtualStart = PalCode.VirtualAddress;
HalDebugPrint(( HAL_INFO,
"HAL: Efi attribute %d & Type 0x%I64x with # of 4K pages 0x%I64x at PA 0x%I64x & mapped to VA 0x%I64x\n",
attribute,
efiMapEntryPtr->Type,
efiMapEntryPtr->NumberOfPages,
efiMapEntryPtr->PhysicalStart,
efiMapEntryPtr->VirtualStart ));
break;
case EfiRuntimeServicesCode:
if(physicalAddr.QuadPart != OptionROMAddress) {
efiMapEntryPtr->VirtualStart = (ULONGLONG) (MmMapIoSpace (physicalAddr,
(EFI_PAGE_SIZE) * (efiMapEntryPtr->NumberOfPages),
MmCached
));
if ((efiMapEntryPtr->VirtualStart) == 0) {
HalDebugPrint(( HAL_ERROR, "HAL: Efi RunTimeSrvceCode area VAddr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
//
// Give Execution previlege
//
//
// Temporarily commented out. Will be enabled once Mm does not clear dirty bit here.
//
// MmSetPageProtection ((PVOID)(efiMapEntryPtr->VirtualStart),
// (EFI_PAGE_SIZE) * (efiMapEntryPtr->NumberOfPages),
// 0x40 /* PAGE_EXECUTE_READWRITE */
// );
HalDebugPrint(( HAL_INFO,
"HAL: Efi attribute %d & Type 0x%I64x with # of 4K pages 0x%I64x at PA 0x%I64x & mapped to VA 0x%I64x\n",
attribute,
efiMapEntryPtr->Type,
efiMapEntryPtr->NumberOfPages,
efiMapEntryPtr->PhysicalStart,
efiMapEntryPtr->VirtualStart));
break;
} else {
efiMapEntryPtr->VirtualStart = OptionROMAddress;
HalDebugPrint(( HAL_INFO,
"HAL: Efi attribute %d & Type 0x%I64x with # of 4K pages 0x%I64x at PA 0x%I64x. NOT MAPPED\n",
attribute,
efiMapEntryPtr->Type,
efiMapEntryPtr->NumberOfPages,
efiMapEntryPtr->PhysicalStart ));
break;
}
case EfiMemoryMappedIO:
efiMapEntryPtr->VirtualStart = (ULONGLONG) (MmMapIoSpace (physicalAddr,
(EFI_PAGE_SIZE) * (efiMapEntryPtr->NumberOfPages),
MmNonCached
));
if ((efiMapEntryPtr->VirtualStart) == 0) {
HalDebugPrint(( HAL_ERROR, "HAL: Efi MemoryMappedIO VAddr is NULL\n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO,
"HAL: Efi attribute %d & Type 0x%I64x with # of 4K pages 0x%I64x at PA 0x%I64x & mapped to VA 0x%I64x\n",
attribute,
efiMapEntryPtr->Type,
efiMapEntryPtr->NumberOfPages,
efiMapEntryPtr->PhysicalStart,
efiMapEntryPtr->VirtualStart ));
break;
case EfiMemoryMappedIOPortSpace:
efiMapEntryPtr->VirtualStart = VIRTUAL_IO_BASE;
HalDebugPrint(( HAL_INFO,
"HAL: Efi attribute %d & Type 0x%I64x with # of 4K pages 0x%I64x at PA 0x%I64x ALREADY mapped to VA 0x%I64x\n",
attribute,
efiMapEntryPtr->Type,
efiMapEntryPtr->NumberOfPages,
efiMapEntryPtr->PhysicalStart,
efiMapEntryPtr->VirtualStart ));
break;
default:
HalDebugPrint(( HAL_INFO, "HAL: Efi CONTROL SHOULD NOT COME HERE \n" ));
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
break;
}
} else {
HalDebugPrint(( HAL_INFO,
"HAL: Efi attribute %d & Type 0x%I64x with # of 4K pages 0x%I64x at PA 0x%I64x ALREADY mapped to VA 0x%I64x\n",
attribute,
efiMapEntryPtr->Type,
efiMapEntryPtr->NumberOfPages,
efiMapEntryPtr->PhysicalStart,
efiMapEntryPtr->VirtualStart ));
}
efiMapEntryPtr = NextMemoryDescriptor(efiMapEntryPtr,EfiDescriptorSize);
}
status = HalpCallEfi(EFI_SET_VIRTUAL_ADDRESS_MAP_INDEX,
(ULONGLONG)EfiMemoryMapSize,
(ULONGLONG)EfiDescriptorSize,
(ULONGLONG)EfiDescriptorVersion,
(ULONGLONG)efiVirtualMemoryMapPtr,
0,
0,
0,
0
);
HalDebugPrint(( HAL_INFO, "HAL: Returned from SetVirtualAddressMap: 0x%Ix\n", status ));
if (EFI_ERROR( status )) {
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
HalDebugPrint(( HAL_INFO, "HAL: EFI Virtual Address mapping done...\n" ));
EfiInitStatus = STATUS_SUCCESS;
//
// Execute some validity checks on the floating point software assist.
//
if (LoaderBlock->u.Ia64.FpswaInterface != (ULONG_PTR) NULL) {
PPLABEL_DESCRIPTOR plabelPointer;
HalpFpEmulate = interfacePtr->Fpswa;
if (HalpFpEmulate == NULL ) {
HalDebugPrint(( HAL_FATAL_ERROR, "HAL: EfiFpswa Virtual Addr is NULL\n" ));
KeBugCheckEx(FP_EMULATION_ERROR, 0, 0, 0, 0);
EfiInitStatus = STATUS_UNSUCCESSFUL;
return STATUS_UNSUCCESSFUL;
}
plabelPointer = (PPLABEL_DESCRIPTOR) HalpFpEmulate;
if ((plabelPointer->EntryPoint & 0xe000000000000000) == 0) {
HalDebugPrint(( HAL_FATAL_ERROR, "HAL: EfiFpswa Instruction Addr is bougus\n" ));
KeBugCheckEx(FP_EMULATION_ERROR, 0, 0, 0, 0);
}
}
return STATUS_SUCCESS;
} // HalpEfiInitialization()
EFI_STATUS
HalpCallEfi(
IN ULONGLONG FunctionId,
IN ULONGLONG Arg1,
IN ULONGLONG Arg2,
IN ULONGLONG Arg3,
IN ULONGLONG Arg4,
IN ULONGLONG Arg5,
IN ULONGLONG Arg6,
IN ULONGLONG Arg7,
IN ULONGLONG Arg8
)
/*++
Routine Description:
:9
This function is a wrapper function for making a EFI call. Callers within the
HAL must use this function to call the EFI.
Arguments:
FunctionId - The EFI function
Arg1-Arg7 - EFI defined arguments for each call
ReturnValues - A pointer to an array of 4 64-bit return values
Return Value:
SAL's return status, return value 0, is returned in addition to the ReturnValues structure
being filled
--*/
{
ULONGLONG EP, GP;
EFI_STATUS efiStatus;
//
// Storage for old level
//
KIRQL OldLevel;
if (((FunctionId != EFI_SET_VIRTUAL_ADDRESS_MAP_INDEX) || (FunctionId != EFI_RESET_SYSTEM_INDEX))
&& (!NT_SUCCESS(EfiInitStatus))) {
return EFI_UNSUPPORTED;
}
switch (FunctionId) {
case EFI_GET_VARIABLE_INDEX:
//
// Dereference the pointer to get the function arguements
//
EP = ((PPLABEL_DESCRIPTOR)EfiVirtualGetVariablePtr) -> EntryPoint;
GP = ((PPLABEL_DESCRIPTOR)EfiVirtualGetVariablePtr) -> GlobalPointer;
//
// Acquire MP Lock
//
KeAcquireSpinLock (&EFIMPLock, &OldLevel);
efiStatus = (HalpCallEfiVirtual( (ULONGLONG)Arg1, // VariableNamePtr
(ULONGLONG)Arg2, // VendorGuidPtr
(ULONGLONG)Arg3, // VariableAttributesPtr,
(ULONGLONG)Arg4, // DataSizePtr,
(ULONGLONG)Arg5, // DataPtr,
Arg6,
EP,
GP
));
//
// Release the MP Lock
//
KeReleaseSpinLock (&EFIMPLock, OldLevel);
break;
case EFI_SET_VARIABLE_INDEX:
//
// Dereference the pointer to get the function arguements
//
EP = ((PPLABEL_DESCRIPTOR)EfiVirtualSetVariablePtr) -> EntryPoint;
GP = ((PPLABEL_DESCRIPTOR)EfiVirtualSetVariablePtr) -> GlobalPointer;
//
// Acquire MP Lock
//
KeAcquireSpinLock (&EFIMPLock, &OldLevel);
efiStatus = (HalpCallEfiVirtual( Arg1,
Arg2,
Arg3,
Arg4,
Arg5,
Arg6,
EP,
GP
));
//
// Release the MP Lock
//
KeReleaseSpinLock (&EFIMPLock, OldLevel);
break;
case EFI_GET_NEXT_VARIABLE_NAME_INDEX:
//
// Dereference the pointer to get the function arguements
//
EP = ((PPLABEL_DESCRIPTOR)EfiVirtualGetNextVariableNamePtr) -> EntryPoint;
GP = ((PPLABEL_DESCRIPTOR)EfiVirtualGetNextVariableNamePtr) -> GlobalPointer;
//
// Acquire MP Lock
//
KeAcquireSpinLock (&EFIMPLock, &OldLevel);
efiStatus = (HalpCallEfiVirtual( Arg1,
Arg2,
Arg3,
Arg4,
Arg5,
Arg6,
EP,
GP
));
//
// Release the MP Lock
//
KeReleaseSpinLock (&EFIMPLock, OldLevel);
break;
case EFI_GET_TIME_INDEX:
//
// Dereference the pointer to get the function arguements
//
EP = ((PPLABEL_DESCRIPTOR)EfiVirtualGetTimePtr) -> EntryPoint;
GP = ((PPLABEL_DESCRIPTOR)EfiVirtualGetTimePtr) -> GlobalPointer;
//
// Acquire MP Lock
//
KeAcquireSpinLock (&EFIMPLock, &OldLevel);
efiStatus = (HalpCallEfiVirtual ((ULONGLONG)Arg1, //EFI_TIME
(ULONGLONG)Arg2, //EFI_TIME Capabilities
Arg3,
Arg4,
Arg5,
Arg6,
EP,
GP
));
//
// Release the MP Lock
//
KeReleaseSpinLock (&EFIMPLock, OldLevel);
break;
case EFI_SET_TIME_INDEX:
//
// Dereference the pointer to get the function arguements
//
EP = ((PPLABEL_DESCRIPTOR)EfiVirtualSetTimePtr) -> EntryPoint;
GP = ((PPLABEL_DESCRIPTOR)EfiVirtualSetTimePtr) -> GlobalPointer;
//
// Acquire MP Lock
//
KeAcquireSpinLock (&EFIMPLock, &OldLevel);
efiStatus = (HalpCallEfiVirtual ((ULONGLONG)Arg1, //EFI_TIME
Arg2,
Arg3,
Arg4,
Arg5,
Arg6,
EP,
GP
));
//
// Release the MP Lock
//
KeReleaseSpinLock (&EFIMPLock, OldLevel);
break;
case EFI_SET_VIRTUAL_ADDRESS_MAP_INDEX:
//
// Dereference the pointer to get the function arguements
//
EP = ((PPLABEL_DESCRIPTOR)EfiSetVirtualAddressMapPtr) -> EntryPoint;
GP = ((PPLABEL_DESCRIPTOR)EfiSetVirtualAddressMapPtr) -> GlobalPointer;
//
// Arg 1 and 5 are virtual mode pointers. We need to convert to physical
//
RtlCopyMemory (HalpMemoryMapVirtualPtr,
(PULONGLONG)Arg4,
(ULONG)EfiMemoryMapSize
);
//
// Acquire MP Lock
//
KeAcquireSpinLock (&EFIMPLock, &OldLevel);
efiStatus = (HalpCallEfiPhysical ((ULONGLONG)EfiMemoryMapSize,
(ULONGLONG)EfiDescriptorSize,
(ULONGLONG)EfiDescriptorVersion,
(ULONGLONG)HalpMemoryMapPhysPtr,
Arg5,
Arg6,
EP,
GP
));
//
// Release the MP Lock
//
KeReleaseSpinLock (&EFIMPLock, OldLevel);
break;
case EFI_RESET_SYSTEM_INDEX:
//
// Dereference the pointer to get the function arguements
//
EP = ((PPLABEL_DESCRIPTOR)EfiResetSystemPtr) -> EntryPoint;
GP = ((PPLABEL_DESCRIPTOR)EfiResetSystemPtr) -> GlobalPointer;
//
// Acquire MP Lock
//
KeAcquireSpinLock (&EFIMPLock, &OldLevel);
efiStatus = ((HalpCallEfiVirtual ( Arg1,
Arg2,
Arg3,
Arg4,
Arg5,
Arg6,
EP,
GP
)));
//
// Release the MP Lock
//
KeReleaseSpinLock (&EFIMPLock, OldLevel);
break;
default:
//
// DebugPrint("EFI: Not supported now\n");
//
efiStatus = EFI_UNSUPPORTED;
break;
}
return efiStatus;
} // HalpCallEfi()
HalpFpErrorPrint (PAL_RETURN pal_ret)
{
EM_uint64_t err_nr;
unsigned int qp;
EM_uint64_t OpCode;
unsigned int rc;
unsigned int significand_size;
unsigned int ISRlow;
unsigned int f1;
unsigned int sign;
unsigned int exponent;
EM_uint64_t significand;
unsigned int new_trap_type;
err_nr = pal_ret.err1 >> 56;
switch (err_nr) {
case 1:
// err_nr = 1 in err1, bits 63-56
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: template FXX is invalid\n"));
break;
case 2:
// err_nr = 2 in err1, bits 63-56
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: instruction slot 3 is not valid \n"));
break;
case 3:
// err_nr = 3 in err1, bits 63-56
// qp in err1, bits 31-0
qp = (unsigned int) pal_ret.err1 & 0xffffffff;
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: qualifying predicate PR[%ud] = 0 \n",qp));
break;
case 4:
// err_nr = 4 in err1, bits 63-56
// OpCode in err2, bits 63-0
OpCode = pal_ret.err2;
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: instruction opcode %8x%8x not recognized \n",
(unsigned int)((OpCode >> 32) & 0xffffffff),(unsigned int)(OpCode & 0xffffffff)));
break;
case 5:
// err_nr = 5 in err1, bits 63-56
// rc in err1, bits 31-0 (1-0)
rc = (unsigned int) pal_ret.err1 & 0xffffffff;
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: invalid rc = %ud\n", rc));
break;
case 6:
// err_nr = 6 in err1, bits 63-56
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: cannot determine the computation model \n"));
break;
case 7:
// err_nr = 7 in err1, bits 63-56
// significand_size in err1, bits 55-32
// ISRlow in err1, bits 31-0
// f1 in err2, bits 63-32
// tmp_fp.sign in err2, bit 17
// tmp_fp.exponent in err2, bits 16-0
// tmp_fp.significand in err3
significand_size = (unsigned int)((pal_ret.err1 >> 32) & 0xffffff);
ISRlow = (unsigned int) (pal_ret.err1 & 0xffffffff);
f1 = (unsigned int) ((pal_ret.err2 >> 32) & 0xffffffff);
sign = (unsigned int) ((pal_ret.err2 >> 17) & 0x01);
exponent = (unsigned int) (pal_ret.err2 & 0x1ffff);
significand = pal_ret.err3;
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: incorrect significand \
size %ud for ISRlow = %4.4x and FR[%ud] = %1.1x %5.5x %8x%8x\n",
significand_size, ISRlow, f1, sign, exponent,
(unsigned int)((significand >> 32) & 0xffffffff),
(unsigned int)(significand & 0xffffffff)));
break;
case 8:
// err_nr = 8 in err1, bits 63-56
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: non-tiny result\n"));
break;
case 9:
// err_nr = 9 in err1, bits 63-56
// significand_size in err1, bits 31-0
significand_size = (unsigned int) pal_ret.err1 & 0xffffffff;
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: incorrect significand \
size %ud\n", significand_size));
break;
case 10:
// err_nr = 10 in err1, bits 63-56
// rc in err1, bits 31-0
rc = (unsigned int) (pal_ret.err1 & 0xffffffff);
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: invalid rc = %ud for \
non-SIMD F1 instruction\n", rc));
break;
case 11:
// err_nr = 11 in err1, bits 63-56
// ISRlow & 0x0ffff in err1, bits 31-0
ISRlow = (unsigned int) (pal_ret.err1 & 0xffffffff);
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: SWA trap code invoked \
with F1 instruction, w/o O or U set in ISR.code = %x\n", ISRlow));
break;
case 12:
// err_nr = 12 in err1, bits 63-56
// ISRlow & 0x0ffff in err1, bits 31-0
ISRlow = (unsigned int) (pal_ret.err1 & 0xffffffff);
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: SWA trap code invoked \
with SIMD F1 instruction, w/o O or U set in ISR.code = %x\n", ISRlow));
break;
case 13:
// err_nr = 13 in err1, bits 63-56
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: non-tiny result low\n"));
break;
case 14:
// err_nr = 14 in err1, bits 63-56
// rc in err1, bits 31-0
rc = (unsigned int) (pal_ret.err1 & 0xffffffff);
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: invalid rc = %ud for \
SIMD F1 instruction\n", rc));
break;
case 15:
// err_nr = 15 in err1, bits 63-56
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: non-tiny result high\n"));
break;
case 16:
// err_nr = 16 in err1, bits 63-56
// OpCode in err2, bits 63-0
OpCode = pal_ret.err2;
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: instruction opcode %8x%8x \
not valid for SWA trap\n", (unsigned int)((OpCode >> 32) & 0xffffffff),
(unsigned int)(OpCode & 0xffffffff)));
break;
case 17:
// err_nr = 17 in err1, bits 63-56
// OpCode in err2, bits 63-0
// ISRlow in err3, bits 31-0
OpCode = pal_ret.err2;
ISRlow = (unsigned int) (pal_ret.err3 & 0xffffffff);
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: fp_emulate () called w/o \
trap_type FPFLT or FPTRAP, OpCode = %8x%8x, and ISR code = %x\n",
(unsigned int)((OpCode >> 32) & 0xffffffff),
(unsigned int)(OpCode & 0xffffffff), ISRlow));
break;
case 18:
// err_nr = 18 in err1, bits 63-56
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: SWA fault repeated\n"));
break;
case 19:
// err_nr = 19 in err1, bits 63-56
// new_trap_type in err1, bits 31-0
new_trap_type = (unsigned int) (pal_ret.err1 & 0xffffffff);
HalDebugPrint(( HAL_ERROR, "fp_emulate () Internal Error: new_trap_type = %x\n",
new_trap_type));
break;
default:
// error
HalDebugPrint(( HAL_ERROR, "Incorrect err_nr = %8x%8x from fp_emulate ()\n",
(unsigned int)((err_nr >> 32) & 0xffffffff),
(unsigned int)(err_nr & 0xffffffff)));
}
}
LONG
HalFpEmulate (
ULONG trap_type,
BUNDLE *pbundle,
ULONGLONG *pipsr,
ULONGLONG *pfpsr,
ULONGLONG *pisr,
ULONGLONG *ppreds,
ULONGLONG *pifs,
FP_STATE *fp_state
)
/*++
Routine Description:
This function is a wrapper function to make fp_emulate() call
to EFI FPSWA driver.
Arguments:
trap_type - indicating which FP trap it is.
pbundle - bundle where this trap occurred
pipsr - IPSR value
pfpsr - FPSR value
pisr - ISR value
ppreds - value of predicate registers
pifs - IFS value
fp_state - floating point registers
Return Value:
return IEEE result of the floating point operation
--*/
{
PAL_RETURN ret;
ret = (*HalpFpEmulate) (
trap_type,
pbundle,
pipsr,
pfpsr,
pisr,
ppreds,
pifs,
fp_state
);
if (ret.retval == FP_EMUL_ERROR) {
HalpFpErrorPrint (ret);
}
return ((LONG) (ret.retval));
}