2008-01-30 19:31:19 +07:00
|
|
|
/*
|
|
|
|
* Common EFI (Extensible Firmware Interface) support functions
|
|
|
|
* Based on Extensible Firmware Interface Specification version 1.0
|
|
|
|
*
|
|
|
|
* Copyright (C) 1999 VA Linux Systems
|
|
|
|
* Copyright (C) 1999 Walt Drummond <drummond@valinux.com>
|
|
|
|
* Copyright (C) 1999-2002 Hewlett-Packard Co.
|
|
|
|
* David Mosberger-Tang <davidm@hpl.hp.com>
|
|
|
|
* Stephane Eranian <eranian@hpl.hp.com>
|
|
|
|
* Copyright (C) 2005-2008 Intel Co.
|
|
|
|
* Fenghua Yu <fenghua.yu@intel.com>
|
|
|
|
* Bibo Mao <bibo.mao@intel.com>
|
|
|
|
* Chandramouli Narayanan <mouli@linux.intel.com>
|
|
|
|
* Huang Ying <ying.huang@intel.com>
|
|
|
|
*
|
|
|
|
* Copied from efi_32.c to eliminate the duplicated code between EFI
|
|
|
|
* 32/64 support code. --ying 2007-10-26
|
|
|
|
*
|
|
|
|
* All EFI Runtime Services are not implemented yet as EFI only
|
|
|
|
* supports physical mode addressing on SoftSDV. This is to be fixed
|
|
|
|
* in a future version. --drummond 1999-07-20
|
|
|
|
*
|
|
|
|
* Implemented EFI runtime services and virtual mode calls. --davidm
|
|
|
|
*
|
|
|
|
* Goutham Rao: <goutham.rao@intel.com>
|
|
|
|
* Skip non-WB memory and ignore empty memory ranges.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/efi.h>
|
|
|
|
#include <linux/bootmem.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/uaccess.h>
|
|
|
|
#include <linux/time.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <linux/reboot.h>
|
|
|
|
#include <linux/bcd.h>
|
|
|
|
|
|
|
|
#include <asm/setup.h>
|
|
|
|
#include <asm/efi.h>
|
|
|
|
#include <asm/time.h>
|
2008-01-30 19:33:55 +07:00
|
|
|
#include <asm/cacheflush.h>
|
|
|
|
#include <asm/tlbflush.h>
|
2009-09-10 09:48:56 +07:00
|
|
|
#include <asm/x86_init.h>
|
2008-01-30 19:31:19 +07:00
|
|
|
|
|
|
|
#define EFI_DEBUG 1
|
|
|
|
#define PFX "EFI: "
|
|
|
|
|
|
|
|
int efi_enabled;
|
|
|
|
EXPORT_SYMBOL(efi_enabled);
|
|
|
|
|
|
|
|
struct efi efi;
|
|
|
|
EXPORT_SYMBOL(efi);
|
|
|
|
|
|
|
|
struct efi_memory_map memmap;
|
|
|
|
|
2008-02-14 04:26:13 +07:00
|
|
|
static struct efi efi_phys __initdata;
|
2008-01-30 19:31:19 +07:00
|
|
|
static efi_system_table_t efi_systab __initdata;
|
|
|
|
|
2008-01-30 19:32:11 +07:00
|
|
|
static int __init setup_noefi(char *arg)
|
|
|
|
{
|
|
|
|
efi_enabled = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
early_param("noefi", setup_noefi);
|
|
|
|
|
2008-06-25 19:44:46 +07:00
|
|
|
int add_efi_memmap;
|
|
|
|
EXPORT_SYMBOL(add_efi_memmap);
|
|
|
|
|
|
|
|
static int __init setup_add_efi_memmap(char *arg)
|
|
|
|
{
|
|
|
|
add_efi_memmap = 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
early_param("add_efi_memmap", setup_add_efi_memmap);
|
|
|
|
|
|
|
|
|
2008-01-30 19:31:19 +07:00
|
|
|
static efi_status_t virt_efi_get_time(efi_time_t *tm, efi_time_cap_t *tc)
|
|
|
|
{
|
|
|
|
return efi_call_virt2(get_time, tm, tc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_set_time(efi_time_t *tm)
|
|
|
|
{
|
|
|
|
return efi_call_virt1(set_time, tm);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_get_wakeup_time(efi_bool_t *enabled,
|
|
|
|
efi_bool_t *pending,
|
|
|
|
efi_time_t *tm)
|
|
|
|
{
|
|
|
|
return efi_call_virt3(get_wakeup_time,
|
|
|
|
enabled, pending, tm);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_set_wakeup_time(efi_bool_t enabled, efi_time_t *tm)
|
|
|
|
{
|
|
|
|
return efi_call_virt2(set_wakeup_time,
|
|
|
|
enabled, tm);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_get_variable(efi_char16_t *name,
|
|
|
|
efi_guid_t *vendor,
|
|
|
|
u32 *attr,
|
|
|
|
unsigned long *data_size,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
return efi_call_virt5(get_variable,
|
|
|
|
name, vendor, attr,
|
|
|
|
data_size, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_get_next_variable(unsigned long *name_size,
|
|
|
|
efi_char16_t *name,
|
|
|
|
efi_guid_t *vendor)
|
|
|
|
{
|
|
|
|
return efi_call_virt3(get_next_variable,
|
|
|
|
name_size, name, vendor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_set_variable(efi_char16_t *name,
|
|
|
|
efi_guid_t *vendor,
|
|
|
|
unsigned long attr,
|
|
|
|
unsigned long data_size,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
return efi_call_virt5(set_variable,
|
|
|
|
name, vendor, attr,
|
|
|
|
data_size, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_get_next_high_mono_count(u32 *count)
|
|
|
|
{
|
|
|
|
return efi_call_virt1(get_next_high_mono_count, count);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void virt_efi_reset_system(int reset_type,
|
|
|
|
efi_status_t status,
|
|
|
|
unsigned long data_size,
|
|
|
|
efi_char16_t *data)
|
|
|
|
{
|
|
|
|
efi_call_virt4(reset_system, reset_type, status,
|
|
|
|
data_size, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_set_virtual_address_map(
|
|
|
|
unsigned long memory_map_size,
|
|
|
|
unsigned long descriptor_size,
|
|
|
|
u32 descriptor_version,
|
|
|
|
efi_memory_desc_t *virtual_map)
|
|
|
|
{
|
|
|
|
return efi_call_virt4(set_virtual_address_map,
|
|
|
|
memory_map_size, descriptor_size,
|
|
|
|
descriptor_version, virtual_map);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t __init phys_efi_set_virtual_address_map(
|
|
|
|
unsigned long memory_map_size,
|
|
|
|
unsigned long descriptor_size,
|
|
|
|
u32 descriptor_version,
|
|
|
|
efi_memory_desc_t *virtual_map)
|
|
|
|
{
|
|
|
|
efi_status_t status;
|
|
|
|
|
|
|
|
efi_call_phys_prelog();
|
|
|
|
status = efi_call_phys4(efi_phys.set_virtual_address_map,
|
|
|
|
memory_map_size, descriptor_size,
|
|
|
|
descriptor_version, virtual_map);
|
|
|
|
efi_call_phys_epilog();
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t __init phys_efi_get_time(efi_time_t *tm,
|
|
|
|
efi_time_cap_t *tc)
|
|
|
|
{
|
|
|
|
efi_status_t status;
|
|
|
|
|
|
|
|
efi_call_phys_prelog();
|
|
|
|
status = efi_call_phys2(efi_phys.get_time, tm, tc);
|
|
|
|
efi_call_phys_epilog();
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efi_set_rtc_mmss(unsigned long nowtime)
|
|
|
|
{
|
|
|
|
int real_seconds, real_minutes;
|
|
|
|
efi_status_t status;
|
|
|
|
efi_time_t eft;
|
|
|
|
efi_time_cap_t cap;
|
|
|
|
|
|
|
|
status = efi.get_time(&eft, &cap);
|
|
|
|
if (status != EFI_SUCCESS) {
|
|
|
|
printk(KERN_ERR "Oops: efitime: can't read time!\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
real_seconds = nowtime % 60;
|
|
|
|
real_minutes = nowtime / 60;
|
|
|
|
if (((abs(real_minutes - eft.minute) + 15)/30) & 1)
|
|
|
|
real_minutes += 30;
|
|
|
|
real_minutes %= 60;
|
|
|
|
eft.minute = real_minutes;
|
|
|
|
eft.second = real_seconds;
|
|
|
|
|
|
|
|
status = efi.set_time(&eft);
|
|
|
|
if (status != EFI_SUCCESS) {
|
|
|
|
printk(KERN_ERR "Oops: efitime: can't write time!\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned long efi_get_time(void)
|
|
|
|
{
|
|
|
|
efi_status_t status;
|
|
|
|
efi_time_t eft;
|
|
|
|
efi_time_cap_t cap;
|
|
|
|
|
|
|
|
status = efi.get_time(&eft, &cap);
|
|
|
|
if (status != EFI_SUCCESS)
|
|
|
|
printk(KERN_ERR "Oops: efitime: can't read time!\n");
|
|
|
|
|
|
|
|
return mktime(eft.year, eft.month, eft.day, eft.hour,
|
|
|
|
eft.minute, eft.second);
|
|
|
|
}
|
|
|
|
|
2008-05-14 22:15:58 +07:00
|
|
|
/*
|
|
|
|
* Tell the kernel about the EFI memory map. This might include
|
|
|
|
* more than the max 128 entries that can fit in the e820 legacy
|
|
|
|
* (zeropage) memory map.
|
|
|
|
*/
|
|
|
|
|
2008-06-25 19:44:46 +07:00
|
|
|
static void __init do_add_efi_memmap(void)
|
2008-05-14 22:15:58 +07:00
|
|
|
{
|
|
|
|
void *p;
|
|
|
|
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
efi_memory_desc_t *md = p;
|
|
|
|
unsigned long long start = md->phys_addr;
|
|
|
|
unsigned long long size = md->num_pages << EFI_PAGE_SHIFT;
|
|
|
|
int e820_type;
|
|
|
|
|
2009-06-17 04:43:40 +07:00
|
|
|
switch (md->type) {
|
|
|
|
case EFI_LOADER_CODE:
|
|
|
|
case EFI_LOADER_DATA:
|
|
|
|
case EFI_BOOT_SERVICES_CODE:
|
|
|
|
case EFI_BOOT_SERVICES_DATA:
|
|
|
|
case EFI_CONVENTIONAL_MEMORY:
|
|
|
|
if (md->attribute & EFI_MEMORY_WB)
|
|
|
|
e820_type = E820_RAM;
|
|
|
|
else
|
|
|
|
e820_type = E820_RESERVED;
|
|
|
|
break;
|
|
|
|
case EFI_ACPI_RECLAIM_MEMORY:
|
|
|
|
e820_type = E820_ACPI;
|
|
|
|
break;
|
|
|
|
case EFI_ACPI_MEMORY_NVS:
|
|
|
|
e820_type = E820_NVS;
|
|
|
|
break;
|
|
|
|
case EFI_UNUSABLE_MEMORY:
|
|
|
|
e820_type = E820_UNUSABLE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/*
|
|
|
|
* EFI_RESERVED_TYPE EFI_RUNTIME_SERVICES_CODE
|
|
|
|
* EFI_RUNTIME_SERVICES_DATA EFI_MEMORY_MAPPED_IO
|
|
|
|
* EFI_MEMORY_MAPPED_IO_PORT_SPACE EFI_PAL_CODE
|
|
|
|
*/
|
2008-05-14 22:15:58 +07:00
|
|
|
e820_type = E820_RESERVED;
|
2009-06-17 04:43:40 +07:00
|
|
|
break;
|
|
|
|
}
|
2008-06-16 08:58:51 +07:00
|
|
|
e820_add_region(start, size, e820_type);
|
2008-05-14 22:15:58 +07:00
|
|
|
}
|
|
|
|
sanitize_e820_map(e820.map, ARRAY_SIZE(e820.map), &e820.nr_map);
|
|
|
|
}
|
|
|
|
|
2008-06-02 13:26:21 +07:00
|
|
|
void __init efi_reserve_early(void)
|
|
|
|
{
|
|
|
|
unsigned long pmap;
|
|
|
|
|
2008-06-22 21:22:02 +07:00
|
|
|
#ifdef CONFIG_X86_32
|
2008-06-02 13:26:21 +07:00
|
|
|
pmap = boot_params.efi_info.efi_memmap;
|
2008-06-22 21:22:02 +07:00
|
|
|
#else
|
|
|
|
pmap = (boot_params.efi_info.efi_memmap |
|
|
|
|
((__u64)boot_params.efi_info.efi_memmap_hi<<32));
|
2008-06-02 13:26:21 +07:00
|
|
|
#endif
|
|
|
|
memmap.phys_map = (void *)pmap;
|
|
|
|
memmap.nr_map = boot_params.efi_info.efi_memmap_size /
|
|
|
|
boot_params.efi_info.efi_memdesc_size;
|
|
|
|
memmap.desc_version = boot_params.efi_info.efi_memdesc_version;
|
|
|
|
memmap.desc_size = boot_params.efi_info.efi_memdesc_size;
|
|
|
|
reserve_early(pmap, pmap + memmap.nr_map * memmap.desc_size,
|
|
|
|
"EFI memmap");
|
|
|
|
}
|
|
|
|
|
2008-01-30 19:31:19 +07:00
|
|
|
#if EFI_DEBUG
|
|
|
|
static void __init print_efi_memmap(void)
|
|
|
|
{
|
|
|
|
efi_memory_desc_t *md;
|
|
|
|
void *p;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (p = memmap.map, i = 0;
|
|
|
|
p < memmap.map_end;
|
|
|
|
p += memmap.desc_size, i++) {
|
|
|
|
md = p;
|
|
|
|
printk(KERN_INFO PFX "mem%02u: type=%u, attr=0x%llx, "
|
|
|
|
"range=[0x%016llx-0x%016llx) (%lluMB)\n",
|
|
|
|
i, md->type, md->attribute, md->phys_addr,
|
|
|
|
md->phys_addr + (md->num_pages << EFI_PAGE_SHIFT),
|
|
|
|
(md->num_pages >> (20 - EFI_PAGE_SHIFT)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* EFI_DEBUG */
|
|
|
|
|
|
|
|
void __init efi_init(void)
|
|
|
|
{
|
|
|
|
efi_config_table_t *config_tables;
|
|
|
|
efi_runtime_services_t *runtime;
|
|
|
|
efi_char16_t *c16;
|
|
|
|
char vendor[100] = "unknown";
|
|
|
|
int i = 0;
|
|
|
|
void *tmp;
|
|
|
|
|
2008-06-22 21:22:02 +07:00
|
|
|
#ifdef CONFIG_X86_32
|
2008-01-30 19:31:19 +07:00
|
|
|
efi_phys.systab = (efi_system_table_t *)boot_params.efi_info.efi_systab;
|
2008-06-22 21:22:02 +07:00
|
|
|
#else
|
|
|
|
efi_phys.systab = (efi_system_table_t *)
|
|
|
|
(boot_params.efi_info.efi_systab |
|
|
|
|
((__u64)boot_params.efi_info.efi_systab_hi<<32));
|
2008-01-30 19:31:19 +07:00
|
|
|
#endif
|
|
|
|
|
2008-01-30 19:33:44 +07:00
|
|
|
efi.systab = early_ioremap((unsigned long)efi_phys.systab,
|
|
|
|
sizeof(efi_system_table_t));
|
2008-01-30 19:31:19 +07:00
|
|
|
if (efi.systab == NULL)
|
|
|
|
printk(KERN_ERR "Couldn't map the EFI system table!\n");
|
|
|
|
memcpy(&efi_systab, efi.systab, sizeof(efi_system_table_t));
|
2008-01-30 19:33:44 +07:00
|
|
|
early_iounmap(efi.systab, sizeof(efi_system_table_t));
|
2008-01-30 19:31:19 +07:00
|
|
|
efi.systab = &efi_systab;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify the EFI Table
|
|
|
|
*/
|
|
|
|
if (efi.systab->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE)
|
|
|
|
printk(KERN_ERR "EFI system table signature incorrect!\n");
|
|
|
|
if ((efi.systab->hdr.revision >> 16) == 0)
|
|
|
|
printk(KERN_ERR "Warning: EFI system table version "
|
|
|
|
"%d.%02d, expected 1.00 or greater!\n",
|
|
|
|
efi.systab->hdr.revision >> 16,
|
|
|
|
efi.systab->hdr.revision & 0xffff);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Show what we know for posterity
|
|
|
|
*/
|
2008-01-30 19:33:44 +07:00
|
|
|
c16 = tmp = early_ioremap(efi.systab->fw_vendor, 2);
|
2008-01-30 19:31:19 +07:00
|
|
|
if (c16) {
|
2009-08-07 05:58:13 +07:00
|
|
|
for (i = 0; i < sizeof(vendor) - 1 && *c16; ++i)
|
2008-01-30 19:31:19 +07:00
|
|
|
vendor[i] = *c16++;
|
|
|
|
vendor[i] = '\0';
|
|
|
|
} else
|
|
|
|
printk(KERN_ERR PFX "Could not map the firmware vendor!\n");
|
2008-01-30 19:33:44 +07:00
|
|
|
early_iounmap(tmp, 2);
|
2008-01-30 19:31:19 +07:00
|
|
|
|
|
|
|
printk(KERN_INFO "EFI v%u.%.02u by %s \n",
|
|
|
|
efi.systab->hdr.revision >> 16,
|
|
|
|
efi.systab->hdr.revision & 0xffff, vendor);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Let's see what config tables the firmware passed to us.
|
|
|
|
*/
|
2008-01-30 19:33:44 +07:00
|
|
|
config_tables = early_ioremap(
|
2008-01-30 19:31:19 +07:00
|
|
|
efi.systab->tables,
|
|
|
|
efi.systab->nr_tables * sizeof(efi_config_table_t));
|
|
|
|
if (config_tables == NULL)
|
|
|
|
printk(KERN_ERR "Could not map EFI Configuration Table!\n");
|
|
|
|
|
|
|
|
printk(KERN_INFO);
|
|
|
|
for (i = 0; i < efi.systab->nr_tables; i++) {
|
|
|
|
if (!efi_guidcmp(config_tables[i].guid, MPS_TABLE_GUID)) {
|
|
|
|
efi.mps = config_tables[i].table;
|
|
|
|
printk(" MPS=0x%lx ", config_tables[i].table);
|
|
|
|
} else if (!efi_guidcmp(config_tables[i].guid,
|
|
|
|
ACPI_20_TABLE_GUID)) {
|
|
|
|
efi.acpi20 = config_tables[i].table;
|
|
|
|
printk(" ACPI 2.0=0x%lx ", config_tables[i].table);
|
|
|
|
} else if (!efi_guidcmp(config_tables[i].guid,
|
|
|
|
ACPI_TABLE_GUID)) {
|
|
|
|
efi.acpi = config_tables[i].table;
|
|
|
|
printk(" ACPI=0x%lx ", config_tables[i].table);
|
|
|
|
} else if (!efi_guidcmp(config_tables[i].guid,
|
|
|
|
SMBIOS_TABLE_GUID)) {
|
|
|
|
efi.smbios = config_tables[i].table;
|
|
|
|
printk(" SMBIOS=0x%lx ", config_tables[i].table);
|
2009-01-20 10:36:04 +07:00
|
|
|
#ifdef CONFIG_X86_UV
|
2008-10-03 23:58:54 +07:00
|
|
|
} else if (!efi_guidcmp(config_tables[i].guid,
|
|
|
|
UV_SYSTEM_TABLE_GUID)) {
|
|
|
|
efi.uv_systab = config_tables[i].table;
|
|
|
|
printk(" UVsystab=0x%lx ", config_tables[i].table);
|
2009-01-20 10:36:04 +07:00
|
|
|
#endif
|
2008-01-30 19:31:19 +07:00
|
|
|
} else if (!efi_guidcmp(config_tables[i].guid,
|
|
|
|
HCDP_TABLE_GUID)) {
|
|
|
|
efi.hcdp = config_tables[i].table;
|
|
|
|
printk(" HCDP=0x%lx ", config_tables[i].table);
|
|
|
|
} else if (!efi_guidcmp(config_tables[i].guid,
|
|
|
|
UGA_IO_PROTOCOL_GUID)) {
|
|
|
|
efi.uga = config_tables[i].table;
|
|
|
|
printk(" UGA=0x%lx ", config_tables[i].table);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printk("\n");
|
2008-01-30 19:33:44 +07:00
|
|
|
early_iounmap(config_tables,
|
2008-01-30 19:31:19 +07:00
|
|
|
efi.systab->nr_tables * sizeof(efi_config_table_t));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check out the runtime services table. We need to map
|
|
|
|
* the runtime services table so that we can grab the physical
|
|
|
|
* address of several of the EFI runtime functions, needed to
|
|
|
|
* set the firmware into virtual mode.
|
|
|
|
*/
|
2008-01-30 19:33:44 +07:00
|
|
|
runtime = early_ioremap((unsigned long)efi.systab->runtime,
|
|
|
|
sizeof(efi_runtime_services_t));
|
2008-01-30 19:31:19 +07:00
|
|
|
if (runtime != NULL) {
|
|
|
|
/*
|
|
|
|
* We will only need *early* access to the following
|
|
|
|
* two EFI runtime services before set_virtual_address_map
|
|
|
|
* is invoked.
|
|
|
|
*/
|
|
|
|
efi_phys.get_time = (efi_get_time_t *)runtime->get_time;
|
|
|
|
efi_phys.set_virtual_address_map =
|
|
|
|
(efi_set_virtual_address_map_t *)
|
|
|
|
runtime->set_virtual_address_map;
|
|
|
|
/*
|
|
|
|
* Make efi_get_time can be called before entering
|
|
|
|
* virtual mode.
|
|
|
|
*/
|
|
|
|
efi.get_time = phys_efi_get_time;
|
|
|
|
} else
|
|
|
|
printk(KERN_ERR "Could not map the EFI runtime service "
|
|
|
|
"table!\n");
|
2008-01-30 19:33:44 +07:00
|
|
|
early_iounmap(runtime, sizeof(efi_runtime_services_t));
|
2008-01-30 19:31:19 +07:00
|
|
|
|
|
|
|
/* Map the EFI memory map */
|
2008-01-30 19:33:44 +07:00
|
|
|
memmap.map = early_ioremap((unsigned long)memmap.phys_map,
|
|
|
|
memmap.nr_map * memmap.desc_size);
|
2008-01-30 19:31:19 +07:00
|
|
|
if (memmap.map == NULL)
|
|
|
|
printk(KERN_ERR "Could not map the EFI memory map!\n");
|
|
|
|
memmap.map_end = memmap.map + (memmap.nr_map * memmap.desc_size);
|
2008-10-03 05:32:06 +07:00
|
|
|
|
2008-01-30 19:31:19 +07:00
|
|
|
if (memmap.desc_size != sizeof(efi_memory_desc_t))
|
2008-10-03 05:32:06 +07:00
|
|
|
printk(KERN_WARNING
|
|
|
|
"Kernel-defined memdesc doesn't match the one from EFI!\n");
|
|
|
|
|
2008-06-25 19:44:46 +07:00
|
|
|
if (add_efi_memmap)
|
|
|
|
do_add_efi_memmap();
|
2008-01-30 19:31:19 +07:00
|
|
|
|
2009-09-10 09:48:56 +07:00
|
|
|
x86_platform.get_wallclock = efi_get_time;
|
|
|
|
x86_platform.set_wallclock = efi_set_rtc_mmss;
|
|
|
|
|
2008-01-30 19:31:19 +07:00
|
|
|
/* Setup for EFI runtime service */
|
|
|
|
reboot_type = BOOT_EFI;
|
|
|
|
|
|
|
|
#if EFI_DEBUG
|
|
|
|
print_efi_memmap();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2008-01-30 19:33:55 +07:00
|
|
|
static void __init runtime_code_page_mkexec(void)
|
|
|
|
{
|
|
|
|
efi_memory_desc_t *md;
|
|
|
|
void *p;
|
2008-02-25 14:18:37 +07:00
|
|
|
u64 addr, npages;
|
2008-01-30 19:33:55 +07:00
|
|
|
|
|
|
|
/* Make EFI runtime service code area executable */
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
md = p;
|
2008-02-04 22:48:06 +07:00
|
|
|
|
|
|
|
if (md->type != EFI_RUNTIME_SERVICES_CODE)
|
|
|
|
continue;
|
|
|
|
|
2008-02-25 14:18:37 +07:00
|
|
|
addr = md->virt_addr;
|
|
|
|
npages = md->num_pages;
|
|
|
|
memrange_efi_to_native(&addr, &npages);
|
|
|
|
set_memory_x(addr, npages);
|
2008-01-30 19:33:55 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-30 19:31:19 +07:00
|
|
|
/*
|
|
|
|
* This function will switch the EFI runtime services to virtual mode.
|
|
|
|
* Essentially, look through the EFI memmap and map every region that
|
|
|
|
* has the runtime attribute bit set in its memory descriptor and update
|
|
|
|
* that memory descriptor with the virtual address obtained from ioremap().
|
|
|
|
* This enables the runtime services to be called without having to
|
|
|
|
* thunk back into physical mode for every invocation.
|
|
|
|
*/
|
|
|
|
void __init efi_enter_virtual_mode(void)
|
|
|
|
{
|
|
|
|
efi_memory_desc_t *md;
|
|
|
|
efi_status_t status;
|
2008-02-04 22:48:06 +07:00
|
|
|
unsigned long size;
|
2009-03-04 09:58:33 +07:00
|
|
|
u64 end, systab, addr, npages, end_pfn;
|
2008-02-04 22:48:06 +07:00
|
|
|
void *p, *va;
|
2008-01-30 19:31:19 +07:00
|
|
|
|
|
|
|
efi.systab = NULL;
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
md = p;
|
|
|
|
if (!(md->attribute & EFI_MEMORY_RUNTIME))
|
|
|
|
continue;
|
2008-02-04 22:48:06 +07:00
|
|
|
|
|
|
|
size = md->num_pages << EFI_PAGE_SHIFT;
|
|
|
|
end = md->phys_addr + size;
|
|
|
|
|
2009-03-04 09:58:33 +07:00
|
|
|
end_pfn = PFN_UP(end);
|
|
|
|
if (end_pfn <= max_low_pfn_mapped
|
|
|
|
|| (end_pfn > (1UL << (32 - PAGE_SHIFT))
|
|
|
|
&& end_pfn <= max_pfn_mapped))
|
2008-02-04 22:48:06 +07:00
|
|
|
va = __va(md->phys_addr);
|
2008-01-30 19:31:19 +07:00
|
|
|
else
|
x86: Make 64-bit efi_ioremap use ioremap on MMIO regions
Booting current 64-bit x86 kernels on the latest Apple MacBook
(MacBook5,2) via EFI gives the following warning:
[ 0.182209] ------------[ cut here ]------------
[ 0.182222] WARNING: at arch/x86/mm/pageattr.c:581 __cpa_process_fault+0x44/0xa0()
[ 0.182227] Hardware name: MacBook5,2
[ 0.182231] CPA: called for zero pte. vaddr = ffff8800ffe00000 cpa->vaddr = ffff8800ffe00000
[ 0.182236] Modules linked in:
[ 0.182242] Pid: 0, comm: swapper Not tainted 2.6.31-rc4 #6
[ 0.182246] Call Trace:
[ 0.182254] [<ffffffff8102c754>] ? __cpa_process_fault+0x44/0xa0
[ 0.182261] [<ffffffff81048668>] warn_slowpath_common+0x78/0xd0
[ 0.182266] [<ffffffff81048744>] warn_slowpath_fmt+0x64/0x70
[ 0.182272] [<ffffffff8102c7ec>] ? update_page_count+0x3c/0x50
[ 0.182280] [<ffffffff818d25c5>] ? phys_pmd_init+0x140/0x22e
[ 0.182286] [<ffffffff8102c754>] __cpa_process_fault+0x44/0xa0
[ 0.182292] [<ffffffff8102ce60>] __change_page_attr_set_clr+0x5f0/0xb40
[ 0.182301] [<ffffffff810d1035>] ? vm_unmap_aliases+0x175/0x190
[ 0.182307] [<ffffffff8102d4ae>] change_page_attr_set_clr+0xfe/0x3d0
[ 0.182314] [<ffffffff8102dcca>] _set_memory_uc+0x2a/0x30
[ 0.182319] [<ffffffff8102dd4b>] set_memory_uc+0x7b/0xb0
[ 0.182327] [<ffffffff818afe31>] efi_enter_virtual_mode+0x2ad/0x2c9
[ 0.182334] [<ffffffff818a1c66>] start_kernel+0x2db/0x3f4
[ 0.182340] [<ffffffff818a1289>] x86_64_start_reservations+0x99/0xb9
[ 0.182345] [<ffffffff818a1389>] x86_64_start_kernel+0xe0/0xf2
[ 0.182357] ---[ end trace 4eaa2a86a8e2da22 ]---
[ 0.182982] init_memory_mapping: 00000000ffffc000-0000000100000000
[ 0.182993] 00ffffc000 - 0100000000 page 4k
This happens because the 64-bit version of efi_ioremap calls
init_memory_mapping for all addresses, regardless of whether they are
RAM or MMIO. The EFI tables on this machine ask for runtime access to
some MMIO regions:
[ 0.000000] EFI: mem195: type=11, attr=0x8000000000000000, range=[0x0000000093400000-0x0000000093401000) (0MB)
[ 0.000000] EFI: mem196: type=11, attr=0x8000000000000000, range=[0x00000000ffc00000-0x00000000ffc40000) (0MB)
[ 0.000000] EFI: mem197: type=11, attr=0x8000000000000000, range=[0x00000000ffc40000-0x00000000ffc80000) (0MB)
[ 0.000000] EFI: mem198: type=11, attr=0x8000000000000000, range=[0x00000000ffc80000-0x00000000ffca4000) (0MB)
[ 0.000000] EFI: mem199: type=11, attr=0x8000000000000000, range=[0x00000000ffca4000-0x00000000ffcb4000) (0MB)
[ 0.000000] EFI: mem200: type=11, attr=0x8000000000000000, range=[0x00000000ffcb4000-0x00000000ffffc000) (3MB)
[ 0.000000] EFI: mem201: type=11, attr=0x8000000000000000, range=[0x00000000ffffc000-0x0000000100000000) (0MB)
This arranges to pass the EFI memory type through to efi_ioremap, and
makes efi_ioremap use ioremap rather than init_memory_mapping if the
type is EFI_MEMORY_MAPPED_IO. With this, the above warning goes away.
Signed-off-by: Paul Mackerras <paulus@samba.org>
LKML-Reference: <19062.55858.533494.471153@cargo.ozlabs.ibm.com>
Cc: Huang Ying <ying.huang@intel.com>
Signed-off-by: H. Peter Anvin <hpa@zytor.com>
2009-08-03 19:38:10 +07:00
|
|
|
va = efi_ioremap(md->phys_addr, size, md->type);
|
2008-02-04 22:48:06 +07:00
|
|
|
|
|
|
|
md->virt_addr = (u64) (unsigned long) va;
|
|
|
|
|
|
|
|
if (!va) {
|
2008-01-30 19:31:19 +07:00
|
|
|
printk(KERN_ERR PFX "ioremap of 0x%llX failed!\n",
|
|
|
|
(unsigned long long)md->phys_addr);
|
2008-02-04 22:48:06 +07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-02-25 14:18:37 +07:00
|
|
|
if (!(md->attribute & EFI_MEMORY_WB)) {
|
|
|
|
addr = md->virt_addr;
|
|
|
|
npages = md->num_pages;
|
|
|
|
memrange_efi_to_native(&addr, &npages);
|
|
|
|
set_memory_uc(addr, npages);
|
|
|
|
}
|
2008-02-13 01:46:48 +07:00
|
|
|
|
2008-02-04 22:48:06 +07:00
|
|
|
systab = (u64) (unsigned long) efi_phys.systab;
|
|
|
|
if (md->phys_addr <= systab && systab < end) {
|
|
|
|
systab += md->virt_addr - md->phys_addr;
|
|
|
|
efi.systab = (efi_system_table_t *) (unsigned long) systab;
|
|
|
|
}
|
2008-01-30 19:31:19 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
BUG_ON(!efi.systab);
|
|
|
|
|
|
|
|
status = phys_efi_set_virtual_address_map(
|
|
|
|
memmap.desc_size * memmap.nr_map,
|
|
|
|
memmap.desc_size,
|
|
|
|
memmap.desc_version,
|
|
|
|
memmap.phys_map);
|
|
|
|
|
|
|
|
if (status != EFI_SUCCESS) {
|
|
|
|
printk(KERN_ALERT "Unable to switch EFI into virtual mode "
|
|
|
|
"(status=%lx)!\n", status);
|
|
|
|
panic("EFI call to SetVirtualAddressMap() failed!");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now that EFI is in virtual mode, update the function
|
|
|
|
* pointers in the runtime service table to the new virtual addresses.
|
|
|
|
*
|
|
|
|
* Call EFI services through wrapper functions.
|
|
|
|
*/
|
|
|
|
efi.get_time = virt_efi_get_time;
|
|
|
|
efi.set_time = virt_efi_set_time;
|
|
|
|
efi.get_wakeup_time = virt_efi_get_wakeup_time;
|
|
|
|
efi.set_wakeup_time = virt_efi_set_wakeup_time;
|
|
|
|
efi.get_variable = virt_efi_get_variable;
|
|
|
|
efi.get_next_variable = virt_efi_get_next_variable;
|
|
|
|
efi.set_variable = virt_efi_set_variable;
|
|
|
|
efi.get_next_high_mono_count = virt_efi_get_next_high_mono_count;
|
|
|
|
efi.reset_system = virt_efi_reset_system;
|
|
|
|
efi.set_virtual_address_map = virt_efi_set_virtual_address_map;
|
2008-02-13 16:22:41 +07:00
|
|
|
if (__supported_pte_mask & _PAGE_NX)
|
|
|
|
runtime_code_page_mkexec();
|
2008-01-30 19:34:10 +07:00
|
|
|
early_iounmap(memmap.map, memmap.nr_map * memmap.desc_size);
|
|
|
|
memmap.map = NULL;
|
2008-01-30 19:31:19 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convenience functions to obtain memory types and attributes
|
|
|
|
*/
|
|
|
|
u32 efi_mem_type(unsigned long phys_addr)
|
|
|
|
{
|
|
|
|
efi_memory_desc_t *md;
|
|
|
|
void *p;
|
|
|
|
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
md = p;
|
|
|
|
if ((md->phys_addr <= phys_addr) &&
|
|
|
|
(phys_addr < (md->phys_addr +
|
|
|
|
(md->num_pages << EFI_PAGE_SHIFT))))
|
|
|
|
return md->type;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 efi_mem_attributes(unsigned long phys_addr)
|
|
|
|
{
|
|
|
|
efi_memory_desc_t *md;
|
|
|
|
void *p;
|
|
|
|
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
md = p;
|
|
|
|
if ((md->phys_addr <= phys_addr) &&
|
|
|
|
(phys_addr < (md->phys_addr +
|
|
|
|
(md->num_pages << EFI_PAGE_SHIFT))))
|
|
|
|
return md->attribute;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|