358 lines
8.6 KiB
C
358 lines
8.6 KiB
C
/*++
|
||
|
||
Copyright (c) 1997 Microsoft Corporation
|
||
All rights reserved
|
||
|
||
Module Name:
|
||
|
||
acpisetd.c
|
||
|
||
Abstract:
|
||
|
||
This module detects an ACPI system. It
|
||
is included into setup so that setup
|
||
can figure out which HAL to load
|
||
|
||
Author:
|
||
|
||
Jake Oshins (jakeo) - Feb. 7, 1997.
|
||
|
||
Environment:
|
||
|
||
Textmode setup.
|
||
|
||
Revision History:
|
||
|
||
|
||
--*/
|
||
#include "bootx86.h"
|
||
|
||
#include "stdlib.h"
|
||
#include "string.h"
|
||
|
||
VOID
|
||
BlPrint(
|
||
PCHAR cp,
|
||
...
|
||
);
|
||
|
||
|
||
#ifdef DEBUG
|
||
#undef DEBUG_PRINT
|
||
#define DEBUG_PRINT BlPrint
|
||
#else
|
||
#define DEBUG_PRINT
|
||
#endif
|
||
|
||
typedef struct _ACPI_BIOS_INSTALLATION_CHECK {
|
||
UCHAR Signature[8]; // "RSD PTR" (ascii)
|
||
UCHAR Checksum;
|
||
UCHAR OemId[6]; // An OEM-supplied string
|
||
UCHAR reserved; // must be 0
|
||
ULONG RsdtAddress; // 32-bit physical address of RSDT
|
||
} ACPI_BIOS_INSTALLATION_CHECK, *PACPI_BIOS_INSTALLATION_CHECK;
|
||
|
||
#include "acpitabl.h"
|
||
|
||
PRSDP BlRsdp;
|
||
PRSDT BlRsdt;
|
||
PXSDT BlXsdt;
|
||
BOOLEAN BlLegacyFree = FALSE;
|
||
|
||
PDESCRIPTION_HEADER
|
||
BlFindACPITable(
|
||
IN PCHAR TableName,
|
||
IN ULONG TableLength
|
||
);
|
||
|
||
// from boot\detect\i386\acpibios.h
|
||
//
|
||
// Acpi BIOS Installation check
|
||
//
|
||
#define ACPI_BIOS_START 0xE0000
|
||
#define ACPI_BIOS_END 0xFFFFF
|
||
#define ACPI_BIOS_HEADER_INCREMENT 16
|
||
|
||
VOID
|
||
BlFindRsdp (
|
||
VOID
|
||
)
|
||
#define EBDA_SEGMENT_PTR 0x40e
|
||
{
|
||
ULONG romAddr = 0;
|
||
ULONG romEnd = 0;
|
||
PACPI_BIOS_INSTALLATION_CHECK header;
|
||
UCHAR sum;
|
||
USHORT i;
|
||
ULONG EbdaSegmentPtr;
|
||
ULONG EbdaPhysicalAdd = 0;
|
||
PUCHAR EbdaVirtualAdd = 0;
|
||
PHYSICAL_ADDRESS paddr;
|
||
enum PASS { PASS1 = 0, PASS2, MAX_PASSES } pass;
|
||
|
||
//
|
||
// Search on 16 byte boundaries for the signature of the
|
||
// Root System Description Table structure.
|
||
//
|
||
for (pass = PASS1; pass < MAX_PASSES; pass++) {
|
||
|
||
if (pass == PASS1) {
|
||
|
||
//
|
||
// On the first pass, we search the first 1K of the
|
||
// Extended BIOS data area. The EBDA segment address
|
||
// is available at physical address 40:0E.
|
||
//
|
||
|
||
paddr.QuadPart = 0;
|
||
EbdaSegmentPtr = (ULONG) MmMapIoSpace( paddr,
|
||
PAGE_SIZE,
|
||
TRUE);
|
||
|
||
EbdaSegmentPtr += EBDA_SEGMENT_PTR;
|
||
EbdaPhysicalAdd = *((PUSHORT)EbdaSegmentPtr);
|
||
EbdaPhysicalAdd = EbdaPhysicalAdd << 4;
|
||
|
||
if (EbdaPhysicalAdd) {
|
||
paddr.HighPart = 0;
|
||
paddr.LowPart = EbdaPhysicalAdd;
|
||
EbdaVirtualAdd = MmMapIoSpace( paddr,
|
||
2 * PAGE_SIZE,
|
||
TRUE);
|
||
}
|
||
|
||
if (!EbdaVirtualAdd) {
|
||
continue;
|
||
}
|
||
|
||
romAddr = (ULONG)EbdaVirtualAdd;
|
||
romEnd = romAddr + 1024;
|
||
|
||
} else {
|
||
//
|
||
// On the second pass, we search (physical) memory 0xE0000
|
||
// to 0xF0000.
|
||
|
||
paddr.LowPart = ACPI_BIOS_START;
|
||
romAddr = (ULONG)MmMapIoSpace(paddr,
|
||
ACPI_BIOS_END - ACPI_BIOS_START,
|
||
TRUE);
|
||
|
||
romEnd = romAddr + (ACPI_BIOS_END - ACPI_BIOS_START);
|
||
}
|
||
|
||
while (romAddr < romEnd) {
|
||
|
||
header = (PACPI_BIOS_INSTALLATION_CHECK)romAddr;
|
||
|
||
//
|
||
// Signature to match is the string "RSD PTR ".
|
||
//
|
||
|
||
if (header->Signature[0] == 'R' && header->Signature[1] == 'S' &&
|
||
header->Signature[2] == 'D' && header->Signature[3] == ' ' &&
|
||
header->Signature[4] == 'P' && header->Signature[5] == 'T' &&
|
||
header->Signature[6] == 'R' && header->Signature[7] == ' ' ) {
|
||
|
||
sum = 0;
|
||
for (i = 0; i < sizeof(ACPI_BIOS_INSTALLATION_CHECK); i++) {
|
||
sum = sum + ((PUCHAR) romAddr)[i];
|
||
}
|
||
if (sum == 0) {
|
||
pass = MAX_PASSES; // leave 'for' loop
|
||
break; // leave 'while' loop
|
||
}
|
||
}
|
||
|
||
romAddr += ACPI_BIOS_HEADER_INCREMENT;
|
||
}
|
||
}
|
||
|
||
if (romAddr >= romEnd) {
|
||
BlRsdp = NULL;
|
||
BlRsdt = NULL;
|
||
return;
|
||
}
|
||
|
||
BlRsdp = (PRSDP)romAddr;
|
||
paddr.LowPart = BlRsdp->RsdtAddress;
|
||
BlRsdt = MmMapIoSpace(paddr, sizeof(RSDT), TRUE);
|
||
BlRsdt = MmMapIoSpace(paddr, BlRsdt->Header.Length, TRUE);
|
||
|
||
#ifdef ACPI_20_COMPLIANT
|
||
if (BlRsdp->Revision > 1) {
|
||
|
||
//
|
||
// ACPI 2.0 BIOS
|
||
//
|
||
|
||
BlXsdt = MmMapIoSpace(paddr, sizeof(XSDT), TRUE);
|
||
BlXsdt = MmMapIoSpace(paddr, BlXsdt->Header.Length, TRUE);
|
||
}
|
||
#endif
|
||
return;
|
||
}
|
||
|
||
|
||
//
|
||
// Make the FADT table global.
|
||
// It will be used when resetting legacy free machines.
|
||
//
|
||
PFADT fadt = NULL;
|
||
|
||
|
||
BOOLEAN
|
||
BlDetectLegacyFreeBios(
|
||
VOID
|
||
)
|
||
{
|
||
|
||
if (BlLegacyFree) {
|
||
return TRUE;
|
||
}
|
||
|
||
BlFindRsdp();
|
||
|
||
if (BlRsdt) {
|
||
|
||
fadt = (PFADT)BlFindACPITable("FACP", sizeof(FADT));
|
||
|
||
if (fadt == NULL) {
|
||
return FALSE;
|
||
}
|
||
|
||
if ((fadt->Header.Revision < 2) ||
|
||
(fadt->Header.Length <= 116)) {
|
||
|
||
//
|
||
// The BIOS is earlier than the legacy-free
|
||
// additions.
|
||
//
|
||
return FALSE;
|
||
}
|
||
|
||
if (!(fadt->boot_arch & I8042)) {
|
||
BlLegacyFree = TRUE;
|
||
return TRUE;
|
||
}
|
||
}
|
||
|
||
return FALSE;
|
||
}
|
||
|
||
PDESCRIPTION_HEADER
|
||
BlFindACPITable(
|
||
IN PCHAR TableName,
|
||
IN ULONG TableLength
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Given a table name, finds that table in the ACPI BIOS
|
||
|
||
Arguments:
|
||
|
||
TableName - Supplies the table name
|
||
|
||
TableLength - Supplies the length of the table to map
|
||
|
||
Return Value:
|
||
|
||
Pointer to the table if found
|
||
|
||
NULL if the table is not found
|
||
|
||
--*/
|
||
|
||
{
|
||
ULONG Signature;
|
||
PFADT Fadt;
|
||
PDESCRIPTION_HEADER Header;
|
||
ULONG TableCount;
|
||
ULONG i;
|
||
PHYSICAL_ADDRESS paddr = {0};
|
||
|
||
Signature = *((ULONG UNALIGNED *)TableName);
|
||
if (Signature == RSDT_SIGNATURE) {
|
||
return(&BlRsdt->Header);
|
||
} else if (Signature == XSDT_SIGNATURE) {
|
||
return(&BlXsdt->Header);
|
||
} else if (Signature == DSDT_SIGNATURE) {
|
||
Fadt = (PFADT)BlFindACPITable("FACP", sizeof(PFADT));
|
||
if (Fadt == NULL) {
|
||
return(NULL);
|
||
}
|
||
if (BlXsdt) {
|
||
paddr = Fadt->x_dsdt;
|
||
} else {
|
||
#if defined(_X86_)
|
||
paddr.LowPart = Fadt->dsdt;
|
||
#else
|
||
paddr.QuadPart = Fadt->dsdt;
|
||
#endif
|
||
}
|
||
Header = MmMapIoSpace(paddr, TableLength, TRUE);
|
||
return(Header);
|
||
} else {
|
||
|
||
//
|
||
// Make sure...
|
||
//
|
||
if( !BlRsdt ) {
|
||
BlFindRsdp();
|
||
}
|
||
|
||
if( BlRsdt ) {
|
||
|
||
TableCount = BlXsdt ?
|
||
NumTableEntriesFromXSDTPointer(BlXsdt) :
|
||
NumTableEntriesFromRSDTPointer(BlRsdt);
|
||
|
||
//
|
||
// Sanity check.
|
||
//
|
||
if( TableCount > 0x100 ) {
|
||
return(NULL);
|
||
}
|
||
|
||
for (i=0;i<TableCount;i++) {
|
||
|
||
|
||
if (BlXsdt) {
|
||
|
||
paddr = BlXsdt->Tables[i];
|
||
|
||
} else {
|
||
|
||
#if defined(_X86_)
|
||
paddr.HighPart = 0;
|
||
paddr.LowPart = BlRsdt->Tables[i];
|
||
#else
|
||
paddr.QuadPart = BlRsdt->Tables[i];
|
||
#endif
|
||
}
|
||
|
||
Header = MmMapIoSpace(paddr, sizeof(DESCRIPTION_HEADER), TRUE);
|
||
if (Header == NULL) {
|
||
return(NULL);
|
||
}
|
||
if (Header->Signature == Signature) {
|
||
//
|
||
// if we need to map more than just the DESCRIPTION_HEADER, do that before
|
||
// returning. Check to see if the end of the table lies past the page
|
||
// boundary the header lies on. If so, we will have to map it.
|
||
//
|
||
if ( ((paddr.LowPart + TableLength) & ~(PAGE_SIZE - 1)) >
|
||
((paddr.LowPart + sizeof(DESCRIPTION_HEADER)) & ~(PAGE_SIZE - 1)) ) {
|
||
Header = MmMapIoSpace(paddr, TableLength, TRUE);
|
||
}
|
||
return(Header);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return(NULL);
|
||
}
|