2008-01-30 12:31:19 +00: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.
|
|
|
|
*/
|
|
|
|
|
2012-02-12 21:24:26 +00:00
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
2008-01-30 12:31:19 +00:00
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/efi.h>
|
2011-05-26 16:22:53 +00:00
|
|
|
#include <linux/export.h>
|
2008-01-30 12:31:19 +00:00
|
|
|
#include <linux/bootmem.h>
|
2010-08-25 20:39:17 +00:00
|
|
|
#include <linux/memblock.h>
|
2008-01-30 12:31:19 +00:00
|
|
|
#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 12:33:55 +00:00
|
|
|
#include <asm/cacheflush.h>
|
|
|
|
#include <asm/tlbflush.h>
|
2009-09-10 02:48:56 +00:00
|
|
|
#include <asm/x86_init.h>
|
2008-01-30 12:31:19 +00:00
|
|
|
|
|
|
|
#define EFI_DEBUG 1
|
|
|
|
|
|
|
|
int efi_enabled;
|
|
|
|
EXPORT_SYMBOL(efi_enabled);
|
|
|
|
|
2011-07-05 11:22:18 +00:00
|
|
|
struct efi __read_mostly efi = {
|
|
|
|
.mps = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.acpi = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.acpi20 = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.smbios = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.sal_systab = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.boot_info = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.hcdp = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.uga = EFI_INVALID_TABLE_ADDR,
|
|
|
|
.uv_systab = EFI_INVALID_TABLE_ADDR,
|
|
|
|
};
|
2008-01-30 12:31:19 +00:00
|
|
|
EXPORT_SYMBOL(efi);
|
|
|
|
|
|
|
|
struct efi_memory_map memmap;
|
|
|
|
|
2012-02-12 21:24:29 +00:00
|
|
|
bool efi_64bit;
|
|
|
|
static bool efi_native;
|
|
|
|
|
2008-02-13 21:26:13 +00:00
|
|
|
static struct efi efi_phys __initdata;
|
2008-01-30 12:31:19 +00:00
|
|
|
static efi_system_table_t efi_systab __initdata;
|
|
|
|
|
2008-01-30 12:32:11 +00:00
|
|
|
static int __init setup_noefi(char *arg)
|
|
|
|
{
|
|
|
|
efi_enabled = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
early_param("noefi", setup_noefi);
|
|
|
|
|
2008-06-25 12:44:46 +00: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 12:31:19 +00:00
|
|
|
static efi_status_t virt_efi_get_time(efi_time_t *tm, efi_time_cap_t *tc)
|
|
|
|
{
|
2011-07-19 10:53:07 +00:00
|
|
|
unsigned long flags;
|
|
|
|
efi_status_t status;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&rtc_lock, flags);
|
|
|
|
status = efi_call_virt2(get_time, tm, tc);
|
|
|
|
spin_unlock_irqrestore(&rtc_lock, flags);
|
|
|
|
return status;
|
2008-01-30 12:31:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_set_time(efi_time_t *tm)
|
|
|
|
{
|
2011-07-19 10:53:07 +00:00
|
|
|
unsigned long flags;
|
|
|
|
efi_status_t status;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&rtc_lock, flags);
|
|
|
|
status = efi_call_virt1(set_time, tm);
|
|
|
|
spin_unlock_irqrestore(&rtc_lock, flags);
|
|
|
|
return status;
|
2008-01-30 12:31:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_get_wakeup_time(efi_bool_t *enabled,
|
|
|
|
efi_bool_t *pending,
|
|
|
|
efi_time_t *tm)
|
|
|
|
{
|
2011-07-19 10:53:07 +00:00
|
|
|
unsigned long flags;
|
|
|
|
efi_status_t status;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&rtc_lock, flags);
|
|
|
|
status = efi_call_virt3(get_wakeup_time,
|
|
|
|
enabled, pending, tm);
|
|
|
|
spin_unlock_irqrestore(&rtc_lock, flags);
|
|
|
|
return status;
|
2008-01-30 12:31:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_set_wakeup_time(efi_bool_t enabled, efi_time_t *tm)
|
|
|
|
{
|
2011-07-19 10:53:07 +00:00
|
|
|
unsigned long flags;
|
|
|
|
efi_status_t status;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&rtc_lock, flags);
|
|
|
|
status = efi_call_virt2(set_wakeup_time,
|
|
|
|
enabled, tm);
|
|
|
|
spin_unlock_irqrestore(&rtc_lock, flags);
|
|
|
|
return status;
|
2008-01-30 12:31:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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,
|
2011-06-06 19:36:24 +00:00
|
|
|
u32 attr,
|
2008-01-30 12:31:19 +00:00
|
|
|
unsigned long data_size,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
return efi_call_virt5(set_variable,
|
|
|
|
name, vendor, attr,
|
|
|
|
data_size, data);
|
|
|
|
}
|
|
|
|
|
2011-06-06 19:36:25 +00:00
|
|
|
static efi_status_t virt_efi_query_variable_info(u32 attr,
|
|
|
|
u64 *storage_space,
|
|
|
|
u64 *remaining_space,
|
|
|
|
u64 *max_variable_size)
|
|
|
|
{
|
|
|
|
if (efi.runtime_version < EFI_2_00_SYSTEM_TABLE_REVISION)
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
|
|
|
|
return efi_call_virt4(query_variable_info, attr, storage_space,
|
|
|
|
remaining_space, max_variable_size);
|
|
|
|
}
|
|
|
|
|
2008-01-30 12:31:19 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2011-06-06 19:36:25 +00:00
|
|
|
static efi_status_t virt_efi_update_capsule(efi_capsule_header_t **capsules,
|
|
|
|
unsigned long count,
|
|
|
|
unsigned long sg_list)
|
|
|
|
{
|
|
|
|
if (efi.runtime_version < EFI_2_00_SYSTEM_TABLE_REVISION)
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
|
|
|
|
return efi_call_virt3(update_capsule, capsules, count, sg_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
static efi_status_t virt_efi_query_capsule_caps(efi_capsule_header_t **capsules,
|
|
|
|
unsigned long count,
|
|
|
|
u64 *max_size,
|
|
|
|
int *reset_type)
|
|
|
|
{
|
|
|
|
if (efi.runtime_version < EFI_2_00_SYSTEM_TABLE_REVISION)
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
|
|
|
|
return efi_call_virt4(query_capsule_caps, capsules, count, max_size,
|
|
|
|
reset_type);
|
|
|
|
}
|
|
|
|
|
2008-01-30 12:31:19 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2012-08-14 16:53:38 +00:00
|
|
|
static efi_status_t __init phys_efi_get_time(efi_time_t *tm,
|
|
|
|
efi_time_cap_t *tc)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
efi_status_t status;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&rtc_lock, flags);
|
|
|
|
efi_call_phys_prelog();
|
|
|
|
status = efi_call_phys2(efi_phys.get_time, virt_to_phys(tm),
|
|
|
|
virt_to_phys(tc));
|
|
|
|
efi_call_phys_epilog();
|
|
|
|
spin_unlock_irqrestore(&rtc_lock, flags);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
int efi_set_rtc_mmss(unsigned long nowtime)
|
2008-01-30 12:31:19 +00:00
|
|
|
{
|
|
|
|
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) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Oops: efitime: can't read time!\n");
|
2008-01-30 12:31:19 +00:00
|
|
|
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) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Oops: efitime: can't write time!\n");
|
2008-01-30 12:31:19 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-08-14 16:53:38 +00:00
|
|
|
unsigned long efi_get_time(void)
|
2008-01-30 12:31:19 +00:00
|
|
|
{
|
|
|
|
efi_status_t status;
|
|
|
|
efi_time_t eft;
|
|
|
|
efi_time_cap_t cap;
|
|
|
|
|
|
|
|
status = efi.get_time(&eft, &cap);
|
|
|
|
if (status != EFI_SUCCESS)
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Oops: efitime: can't read time!\n");
|
2008-01-30 12:31:19 +00:00
|
|
|
|
|
|
|
return mktime(eft.year, eft.month, eft.day, eft.hour,
|
|
|
|
eft.minute, eft.second);
|
|
|
|
}
|
|
|
|
|
2008-05-14 15:15:58 +00: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 12:44:46 +00:00
|
|
|
static void __init do_add_efi_memmap(void)
|
2008-05-14 15:15:58 +00: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-16 21:43:40 +00: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 15:15:58 +00:00
|
|
|
e820_type = E820_RESERVED;
|
2009-06-16 21:43:40 +00:00
|
|
|
break;
|
|
|
|
}
|
2008-06-16 01:58:51 +00:00
|
|
|
e820_add_region(start, size, e820_type);
|
2008-05-14 15:15:58 +00:00
|
|
|
}
|
|
|
|
sanitize_e820_map(e820.map, ARRAY_SIZE(e820.map), &e820.nr_map);
|
|
|
|
}
|
|
|
|
|
2012-02-12 21:24:29 +00:00
|
|
|
int __init efi_memblock_x86_reserve_range(void)
|
2008-06-02 06:26:21 +00:00
|
|
|
{
|
|
|
|
unsigned long pmap;
|
|
|
|
|
2008-06-22 14:22:02 +00:00
|
|
|
#ifdef CONFIG_X86_32
|
2012-02-12 21:24:29 +00:00
|
|
|
/* Can't handle data above 4GB at this time */
|
|
|
|
if (boot_params.efi_info.efi_memmap_hi) {
|
|
|
|
pr_err("Memory map is above 4GB, disabling EFI.\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2008-06-02 06:26:21 +00:00
|
|
|
pmap = boot_params.efi_info.efi_memmap;
|
2008-06-22 14:22:02 +00:00
|
|
|
#else
|
|
|
|
pmap = (boot_params.efi_info.efi_memmap |
|
|
|
|
((__u64)boot_params.efi_info.efi_memmap_hi<<32));
|
2008-06-02 06:26:21 +00: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;
|
2011-07-12 09:16:06 +00:00
|
|
|
memblock_reserve(pmap, memmap.nr_map * memmap.desc_size);
|
2012-02-12 21:24:29 +00:00
|
|
|
|
|
|
|
return 0;
|
2008-06-02 06:26:21 +00:00
|
|
|
}
|
|
|
|
|
2008-01-30 12:31:19 +00: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;
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_info("mem%02u: type=%u, attr=0x%llx, "
|
2008-01-30 12:31:19 +00:00
|
|
|
"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 */
|
|
|
|
|
x86, efi: Retain boot service code until after switching to virtual mode
UEFI stands for "Unified Extensible Firmware Interface", where "Firmware"
is an ancient African word meaning "Why do something right when you can
do it so wrong that children will weep and brave adults will cower before
you", and "UEI" is Celtic for "We missed DOS so we burned it into your
ROMs". The UEFI specification provides for runtime services (ie, another
way for the operating system to be forced to depend on the firmware) and
we rely on these for certain trivial tasks such as setting up the
bootloader. But some hardware fails to work if we attempt to use these
runtime services from physical mode, and so we have to switch into virtual
mode. So far so dreadful.
The specification makes it clear that the operating system is free to do
whatever it wants with boot services code after ExitBootServices() has been
called. SetVirtualAddressMap() can't be called until ExitBootServices() has
been. So, obviously, a whole bunch of EFI implementations call into boot
services code when we do that. Since we've been charmingly naive and
trusted that the specification may be somehow relevant to the real world,
we've already stuffed a picture of a penguin or something in that address
space. And just to make things more entertaining, we've also marked it
non-executable.
This patch allocates the boot services regions during EFI init and makes
sure that they're executable. Then, after SetVirtualAddressMap(), it
discards them and everyone lives happily ever after. Except for the ones
who have to work on EFI, who live sad lives haunted by the knowledge that
someone's eventually going to write yet another firmware specification.
[ hpa: adding this to urgent with a stable tag since it fixes currently-broken
hardware. However, I do not know what the dependencies are and so I do
not know which -stable versions this may be a candidate for. ]
Signed-off-by: Matthew Garrett <mjg@redhat.com>
Link: http://lkml.kernel.org/r/1306331593-28715-1-git-send-email-mjg@redhat.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: <stable@kernel.org>
2011-05-25 13:53:13 +00:00
|
|
|
void __init efi_reserve_boot_services(void)
|
|
|
|
{
|
|
|
|
void *p;
|
|
|
|
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
efi_memory_desc_t *md = p;
|
2011-06-14 17:53:09 +00:00
|
|
|
u64 start = md->phys_addr;
|
|
|
|
u64 size = md->num_pages << EFI_PAGE_SHIFT;
|
x86, efi: Retain boot service code until after switching to virtual mode
UEFI stands for "Unified Extensible Firmware Interface", where "Firmware"
is an ancient African word meaning "Why do something right when you can
do it so wrong that children will weep and brave adults will cower before
you", and "UEI" is Celtic for "We missed DOS so we burned it into your
ROMs". The UEFI specification provides for runtime services (ie, another
way for the operating system to be forced to depend on the firmware) and
we rely on these for certain trivial tasks such as setting up the
bootloader. But some hardware fails to work if we attempt to use these
runtime services from physical mode, and so we have to switch into virtual
mode. So far so dreadful.
The specification makes it clear that the operating system is free to do
whatever it wants with boot services code after ExitBootServices() has been
called. SetVirtualAddressMap() can't be called until ExitBootServices() has
been. So, obviously, a whole bunch of EFI implementations call into boot
services code when we do that. Since we've been charmingly naive and
trusted that the specification may be somehow relevant to the real world,
we've already stuffed a picture of a penguin or something in that address
space. And just to make things more entertaining, we've also marked it
non-executable.
This patch allocates the boot services regions during EFI init and makes
sure that they're executable. Then, after SetVirtualAddressMap(), it
discards them and everyone lives happily ever after. Except for the ones
who have to work on EFI, who live sad lives haunted by the knowledge that
someone's eventually going to write yet another firmware specification.
[ hpa: adding this to urgent with a stable tag since it fixes currently-broken
hardware. However, I do not know what the dependencies are and so I do
not know which -stable versions this may be a candidate for. ]
Signed-off-by: Matthew Garrett <mjg@redhat.com>
Link: http://lkml.kernel.org/r/1306331593-28715-1-git-send-email-mjg@redhat.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: <stable@kernel.org>
2011-05-25 13:53:13 +00:00
|
|
|
|
|
|
|
if (md->type != EFI_BOOT_SERVICES_CODE &&
|
|
|
|
md->type != EFI_BOOT_SERVICES_DATA)
|
|
|
|
continue;
|
2011-06-14 17:53:09 +00:00
|
|
|
/* Only reserve where possible:
|
|
|
|
* - Not within any already allocated areas
|
|
|
|
* - Not over any memory area (really needed, if above?)
|
|
|
|
* - Not within any part of the kernel
|
|
|
|
* - Not the bios reserved area
|
|
|
|
*/
|
|
|
|
if ((start+size >= virt_to_phys(_text)
|
|
|
|
&& start <= virt_to_phys(_end)) ||
|
|
|
|
!e820_all_mapped(start, start+size, E820_RAM) ||
|
2011-07-12 07:58:05 +00:00
|
|
|
memblock_is_region_reserved(start, size)) {
|
2011-06-14 17:53:09 +00:00
|
|
|
/* Could not reserve, skip it */
|
|
|
|
md->num_pages = 0;
|
2012-02-12 21:24:26 +00:00
|
|
|
memblock_dbg("Could not reserve boot range "
|
2011-06-14 17:53:09 +00:00
|
|
|
"[0x%010llx-0x%010llx]\n",
|
|
|
|
start, start+size-1);
|
|
|
|
} else
|
2011-07-12 09:16:06 +00:00
|
|
|
memblock_reserve(start, size);
|
x86, efi: Retain boot service code until after switching to virtual mode
UEFI stands for "Unified Extensible Firmware Interface", where "Firmware"
is an ancient African word meaning "Why do something right when you can
do it so wrong that children will weep and brave adults will cower before
you", and "UEI" is Celtic for "We missed DOS so we burned it into your
ROMs". The UEFI specification provides for runtime services (ie, another
way for the operating system to be forced to depend on the firmware) and
we rely on these for certain trivial tasks such as setting up the
bootloader. But some hardware fails to work if we attempt to use these
runtime services from physical mode, and so we have to switch into virtual
mode. So far so dreadful.
The specification makes it clear that the operating system is free to do
whatever it wants with boot services code after ExitBootServices() has been
called. SetVirtualAddressMap() can't be called until ExitBootServices() has
been. So, obviously, a whole bunch of EFI implementations call into boot
services code when we do that. Since we've been charmingly naive and
trusted that the specification may be somehow relevant to the real world,
we've already stuffed a picture of a penguin or something in that address
space. And just to make things more entertaining, we've also marked it
non-executable.
This patch allocates the boot services regions during EFI init and makes
sure that they're executable. Then, after SetVirtualAddressMap(), it
discards them and everyone lives happily ever after. Except for the ones
who have to work on EFI, who live sad lives haunted by the knowledge that
someone's eventually going to write yet another firmware specification.
[ hpa: adding this to urgent with a stable tag since it fixes currently-broken
hardware. However, I do not know what the dependencies are and so I do
not know which -stable versions this may be a candidate for. ]
Signed-off-by: Matthew Garrett <mjg@redhat.com>
Link: http://lkml.kernel.org/r/1306331593-28715-1-git-send-email-mjg@redhat.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: <stable@kernel.org>
2011-05-25 13:53:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __init efi_free_boot_services(void)
|
|
|
|
{
|
|
|
|
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;
|
|
|
|
|
|
|
|
if (md->type != EFI_BOOT_SERVICES_CODE &&
|
|
|
|
md->type != EFI_BOOT_SERVICES_DATA)
|
|
|
|
continue;
|
|
|
|
|
2011-06-14 17:53:09 +00:00
|
|
|
/* Could not reserve boot area */
|
|
|
|
if (!size)
|
|
|
|
continue;
|
|
|
|
|
x86, efi: Retain boot service code until after switching to virtual mode
UEFI stands for "Unified Extensible Firmware Interface", where "Firmware"
is an ancient African word meaning "Why do something right when you can
do it so wrong that children will weep and brave adults will cower before
you", and "UEI" is Celtic for "We missed DOS so we burned it into your
ROMs". The UEFI specification provides for runtime services (ie, another
way for the operating system to be forced to depend on the firmware) and
we rely on these for certain trivial tasks such as setting up the
bootloader. But some hardware fails to work if we attempt to use these
runtime services from physical mode, and so we have to switch into virtual
mode. So far so dreadful.
The specification makes it clear that the operating system is free to do
whatever it wants with boot services code after ExitBootServices() has been
called. SetVirtualAddressMap() can't be called until ExitBootServices() has
been. So, obviously, a whole bunch of EFI implementations call into boot
services code when we do that. Since we've been charmingly naive and
trusted that the specification may be somehow relevant to the real world,
we've already stuffed a picture of a penguin or something in that address
space. And just to make things more entertaining, we've also marked it
non-executable.
This patch allocates the boot services regions during EFI init and makes
sure that they're executable. Then, after SetVirtualAddressMap(), it
discards them and everyone lives happily ever after. Except for the ones
who have to work on EFI, who live sad lives haunted by the knowledge that
someone's eventually going to write yet another firmware specification.
[ hpa: adding this to urgent with a stable tag since it fixes currently-broken
hardware. However, I do not know what the dependencies are and so I do
not know which -stable versions this may be a candidate for. ]
Signed-off-by: Matthew Garrett <mjg@redhat.com>
Link: http://lkml.kernel.org/r/1306331593-28715-1-git-send-email-mjg@redhat.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: <stable@kernel.org>
2011-05-25 13:53:13 +00:00
|
|
|
free_bootmem_late(start, size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-12 21:24:28 +00:00
|
|
|
static int __init efi_systab_init(void *phys)
|
2008-01-30 12:31:19 +00:00
|
|
|
{
|
2012-02-12 21:24:29 +00:00
|
|
|
if (efi_64bit) {
|
|
|
|
efi_system_table_64_t *systab64;
|
|
|
|
u64 tmp = 0;
|
|
|
|
|
|
|
|
systab64 = early_ioremap((unsigned long)phys,
|
|
|
|
sizeof(*systab64));
|
|
|
|
if (systab64 == NULL) {
|
|
|
|
pr_err("Couldn't map the system table!\n");
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
efi_systab.hdr = systab64->hdr;
|
|
|
|
efi_systab.fw_vendor = systab64->fw_vendor;
|
|
|
|
tmp |= systab64->fw_vendor;
|
|
|
|
efi_systab.fw_revision = systab64->fw_revision;
|
|
|
|
efi_systab.con_in_handle = systab64->con_in_handle;
|
|
|
|
tmp |= systab64->con_in_handle;
|
|
|
|
efi_systab.con_in = systab64->con_in;
|
|
|
|
tmp |= systab64->con_in;
|
|
|
|
efi_systab.con_out_handle = systab64->con_out_handle;
|
|
|
|
tmp |= systab64->con_out_handle;
|
|
|
|
efi_systab.con_out = systab64->con_out;
|
|
|
|
tmp |= systab64->con_out;
|
|
|
|
efi_systab.stderr_handle = systab64->stderr_handle;
|
|
|
|
tmp |= systab64->stderr_handle;
|
|
|
|
efi_systab.stderr = systab64->stderr;
|
|
|
|
tmp |= systab64->stderr;
|
|
|
|
efi_systab.runtime = (void *)(unsigned long)systab64->runtime;
|
|
|
|
tmp |= systab64->runtime;
|
|
|
|
efi_systab.boottime = (void *)(unsigned long)systab64->boottime;
|
|
|
|
tmp |= systab64->boottime;
|
|
|
|
efi_systab.nr_tables = systab64->nr_tables;
|
|
|
|
efi_systab.tables = systab64->tables;
|
|
|
|
tmp |= systab64->tables;
|
|
|
|
|
|
|
|
early_iounmap(systab64, sizeof(*systab64));
|
|
|
|
#ifdef CONFIG_X86_32
|
|
|
|
if (tmp >> 32) {
|
|
|
|
pr_err("EFI data located above 4GB, disabling EFI.\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
efi_system_table_32_t *systab32;
|
|
|
|
|
|
|
|
systab32 = early_ioremap((unsigned long)phys,
|
|
|
|
sizeof(*systab32));
|
|
|
|
if (systab32 == NULL) {
|
|
|
|
pr_err("Couldn't map the system table!\n");
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
efi_systab.hdr = systab32->hdr;
|
|
|
|
efi_systab.fw_vendor = systab32->fw_vendor;
|
|
|
|
efi_systab.fw_revision = systab32->fw_revision;
|
|
|
|
efi_systab.con_in_handle = systab32->con_in_handle;
|
|
|
|
efi_systab.con_in = systab32->con_in;
|
|
|
|
efi_systab.con_out_handle = systab32->con_out_handle;
|
|
|
|
efi_systab.con_out = systab32->con_out;
|
|
|
|
efi_systab.stderr_handle = systab32->stderr_handle;
|
|
|
|
efi_systab.stderr = systab32->stderr;
|
|
|
|
efi_systab.runtime = (void *)(unsigned long)systab32->runtime;
|
|
|
|
efi_systab.boottime = (void *)(unsigned long)systab32->boottime;
|
|
|
|
efi_systab.nr_tables = systab32->nr_tables;
|
|
|
|
efi_systab.tables = systab32->tables;
|
|
|
|
|
|
|
|
early_iounmap(systab32, sizeof(*systab32));
|
2012-02-12 21:24:28 +00:00
|
|
|
}
|
2012-02-12 21:24:29 +00:00
|
|
|
|
2008-01-30 12:31:19 +00:00
|
|
|
efi.systab = &efi_systab;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify the EFI Table
|
|
|
|
*/
|
2012-02-12 21:24:28 +00:00
|
|
|
if (efi.systab->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("System table signature incorrect!\n");
|
2012-02-12 21:24:28 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2008-01-30 12:31:19 +00:00
|
|
|
if ((efi.systab->hdr.revision >> 16) == 0)
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Warning: System table version "
|
2008-01-30 12:31:19 +00:00
|
|
|
"%d.%02d, expected 1.00 or greater!\n",
|
|
|
|
efi.systab->hdr.revision >> 16,
|
|
|
|
efi.systab->hdr.revision & 0xffff);
|
2012-02-12 21:24:28 +00:00
|
|
|
|
|
|
|
return 0;
|
2012-02-12 21:24:25 +00:00
|
|
|
}
|
2008-01-30 12:31:19 +00:00
|
|
|
|
2012-02-12 21:24:28 +00:00
|
|
|
static int __init efi_config_init(u64 tables, int nr_tables)
|
2012-02-12 21:24:25 +00:00
|
|
|
{
|
2012-02-12 21:24:29 +00:00
|
|
|
void *config_tables, *tablep;
|
|
|
|
int i, sz;
|
|
|
|
|
|
|
|
if (efi_64bit)
|
|
|
|
sz = sizeof(efi_config_table_64_t);
|
|
|
|
else
|
|
|
|
sz = sizeof(efi_config_table_32_t);
|
2008-01-30 12:31:19 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Let's see what config tables the firmware passed to us.
|
|
|
|
*/
|
2012-02-12 21:24:29 +00:00
|
|
|
config_tables = early_ioremap(tables, nr_tables * sz);
|
2012-02-12 21:24:28 +00:00
|
|
|
if (config_tables == NULL) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Could not map Configuration table!\n");
|
2012-02-12 21:24:28 +00:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2008-01-30 12:31:19 +00:00
|
|
|
|
2012-02-12 21:24:29 +00:00
|
|
|
tablep = config_tables;
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_info("");
|
2008-01-30 12:31:19 +00:00
|
|
|
for (i = 0; i < efi.systab->nr_tables; i++) {
|
2012-02-12 21:24:29 +00:00
|
|
|
efi_guid_t guid;
|
|
|
|
unsigned long table;
|
|
|
|
|
|
|
|
if (efi_64bit) {
|
|
|
|
u64 table64;
|
|
|
|
guid = ((efi_config_table_64_t *)tablep)->guid;
|
|
|
|
table64 = ((efi_config_table_64_t *)tablep)->table;
|
|
|
|
table = table64;
|
|
|
|
#ifdef CONFIG_X86_32
|
|
|
|
if (table64 >> 32) {
|
|
|
|
pr_cont("\n");
|
|
|
|
pr_err("Table located above 4GB, disabling EFI.\n");
|
|
|
|
early_iounmap(config_tables,
|
|
|
|
efi.systab->nr_tables * sz);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
guid = ((efi_config_table_32_t *)tablep)->guid;
|
|
|
|
table = ((efi_config_table_32_t *)tablep)->table;
|
|
|
|
}
|
2012-02-12 21:24:27 +00:00
|
|
|
if (!efi_guidcmp(guid, MPS_TABLE_GUID)) {
|
|
|
|
efi.mps = table;
|
|
|
|
pr_cont(" MPS=0x%lx ", table);
|
|
|
|
} else if (!efi_guidcmp(guid, ACPI_20_TABLE_GUID)) {
|
|
|
|
efi.acpi20 = table;
|
|
|
|
pr_cont(" ACPI 2.0=0x%lx ", table);
|
|
|
|
} else if (!efi_guidcmp(guid, ACPI_TABLE_GUID)) {
|
|
|
|
efi.acpi = table;
|
|
|
|
pr_cont(" ACPI=0x%lx ", table);
|
|
|
|
} else if (!efi_guidcmp(guid, SMBIOS_TABLE_GUID)) {
|
|
|
|
efi.smbios = table;
|
|
|
|
pr_cont(" SMBIOS=0x%lx ", table);
|
2009-01-20 03:36:04 +00:00
|
|
|
#ifdef CONFIG_X86_UV
|
2012-02-12 21:24:27 +00:00
|
|
|
} else if (!efi_guidcmp(guid, UV_SYSTEM_TABLE_GUID)) {
|
|
|
|
efi.uv_systab = table;
|
|
|
|
pr_cont(" UVsystab=0x%lx ", table);
|
2009-01-20 03:36:04 +00:00
|
|
|
#endif
|
2012-02-12 21:24:27 +00:00
|
|
|
} else if (!efi_guidcmp(guid, HCDP_TABLE_GUID)) {
|
|
|
|
efi.hcdp = table;
|
|
|
|
pr_cont(" HCDP=0x%lx ", table);
|
|
|
|
} else if (!efi_guidcmp(guid, UGA_IO_PROTOCOL_GUID)) {
|
|
|
|
efi.uga = table;
|
|
|
|
pr_cont(" UGA=0x%lx ", table);
|
2008-01-30 12:31:19 +00:00
|
|
|
}
|
2012-02-12 21:24:29 +00:00
|
|
|
tablep += sz;
|
2008-01-30 12:31:19 +00:00
|
|
|
}
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_cont("\n");
|
2012-02-12 21:24:27 +00:00
|
|
|
early_iounmap(config_tables, efi.systab->nr_tables * sz);
|
2012-02-12 21:24:28 +00:00
|
|
|
return 0;
|
2012-02-12 21:24:25 +00:00
|
|
|
}
|
|
|
|
|
2012-02-12 21:24:28 +00:00
|
|
|
static int __init efi_runtime_init(void)
|
2012-02-12 21:24:25 +00:00
|
|
|
{
|
|
|
|
efi_runtime_services_t *runtime;
|
2008-01-30 12:31:19 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* 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 12:33:44 +00:00
|
|
|
runtime = early_ioremap((unsigned long)efi.systab->runtime,
|
|
|
|
sizeof(efi_runtime_services_t));
|
2012-02-12 21:24:28 +00:00
|
|
|
if (!runtime) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Could not map the runtime service table!\n");
|
2012-02-12 21:24:28 +00:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* We will only need *early* access to the following
|
2012-08-14 16:53:38 +00:00
|
|
|
* two EFI runtime services before set_virtual_address_map
|
2012-02-12 21:24:28 +00:00
|
|
|
* is invoked.
|
|
|
|
*/
|
2012-08-14 16:53:38 +00:00
|
|
|
efi_phys.get_time = (efi_get_time_t *)runtime->get_time;
|
2012-02-12 21:24:28 +00:00
|
|
|
efi_phys.set_virtual_address_map =
|
|
|
|
(efi_set_virtual_address_map_t *)
|
|
|
|
runtime->set_virtual_address_map;
|
2012-08-14 16:53:38 +00:00
|
|
|
/*
|
|
|
|
* Make efi_get_time can be called before entering
|
|
|
|
* virtual mode.
|
|
|
|
*/
|
|
|
|
efi.get_time = phys_efi_get_time;
|
2008-01-30 12:33:44 +00:00
|
|
|
early_iounmap(runtime, sizeof(efi_runtime_services_t));
|
2012-02-12 21:24:28 +00:00
|
|
|
|
|
|
|
return 0;
|
2012-02-12 21:24:25 +00:00
|
|
|
}
|
2008-01-30 12:31:19 +00:00
|
|
|
|
2012-02-12 21:24:28 +00:00
|
|
|
static int __init efi_memmap_init(void)
|
2012-02-12 21:24:25 +00:00
|
|
|
{
|
2008-01-30 12:31:19 +00:00
|
|
|
/* Map the EFI memory map */
|
2008-01-30 12:33:44 +00:00
|
|
|
memmap.map = early_ioremap((unsigned long)memmap.phys_map,
|
|
|
|
memmap.nr_map * memmap.desc_size);
|
2012-02-12 21:24:28 +00:00
|
|
|
if (memmap.map == NULL) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Could not map the memory map!\n");
|
2012-02-12 21:24:28 +00:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2008-01-30 12:31:19 +00:00
|
|
|
memmap.map_end = memmap.map + (memmap.nr_map * memmap.desc_size);
|
2008-10-02 22:32:06 +00:00
|
|
|
|
2008-06-25 12:44:46 +00:00
|
|
|
if (add_efi_memmap)
|
|
|
|
do_add_efi_memmap();
|
2012-02-12 21:24:28 +00:00
|
|
|
|
|
|
|
return 0;
|
2012-02-12 21:24:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void __init efi_init(void)
|
|
|
|
{
|
|
|
|
efi_char16_t *c16;
|
|
|
|
char vendor[100] = "unknown";
|
|
|
|
int i = 0;
|
|
|
|
void *tmp;
|
|
|
|
|
|
|
|
#ifdef CONFIG_X86_32
|
2012-02-12 21:24:29 +00:00
|
|
|
if (boot_params.efi_info.efi_systab_hi ||
|
|
|
|
boot_params.efi_info.efi_memmap_hi) {
|
|
|
|
pr_info("Table located above 4GB, disabling EFI.\n");
|
|
|
|
efi_enabled = 0;
|
|
|
|
return;
|
|
|
|
}
|
2012-02-12 21:24:25 +00:00
|
|
|
efi_phys.systab = (efi_system_table_t *)boot_params.efi_info.efi_systab;
|
2012-02-12 21:24:29 +00:00
|
|
|
efi_native = !efi_64bit;
|
2012-02-12 21:24:25 +00:00
|
|
|
#else
|
|
|
|
efi_phys.systab = (efi_system_table_t *)
|
2012-02-12 21:24:29 +00:00
|
|
|
(boot_params.efi_info.efi_systab |
|
|
|
|
((__u64)boot_params.efi_info.efi_systab_hi<<32));
|
|
|
|
efi_native = efi_64bit;
|
2012-02-12 21:24:25 +00:00
|
|
|
#endif
|
|
|
|
|
2012-02-12 21:24:28 +00:00
|
|
|
if (efi_systab_init(efi_phys.systab)) {
|
|
|
|
efi_enabled = 0;
|
|
|
|
return;
|
|
|
|
}
|
2012-02-12 21:24:25 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Show what we know for posterity
|
|
|
|
*/
|
|
|
|
c16 = tmp = early_ioremap(efi.systab->fw_vendor, 2);
|
|
|
|
if (c16) {
|
|
|
|
for (i = 0; i < sizeof(vendor) - 1 && *c16; ++i)
|
|
|
|
vendor[i] = *c16++;
|
|
|
|
vendor[i] = '\0';
|
|
|
|
} else
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("Could not map the firmware vendor!\n");
|
2012-02-12 21:24:25 +00:00
|
|
|
early_iounmap(tmp, 2);
|
|
|
|
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_info("EFI v%u.%.02u by %s\n",
|
|
|
|
efi.systab->hdr.revision >> 16,
|
|
|
|
efi.systab->hdr.revision & 0xffff, vendor);
|
2012-02-12 21:24:25 +00:00
|
|
|
|
2012-02-12 21:24:28 +00:00
|
|
|
if (efi_config_init(efi.systab->tables, efi.systab->nr_tables)) {
|
|
|
|
efi_enabled = 0;
|
|
|
|
return;
|
|
|
|
}
|
2012-02-12 21:24:25 +00:00
|
|
|
|
2012-02-12 21:24:29 +00:00
|
|
|
/*
|
|
|
|
* Note: We currently don't support runtime services on an EFI
|
|
|
|
* that doesn't match the kernel 32/64-bit mode.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!efi_native)
|
|
|
|
pr_info("No EFI runtime due to 32/64-bit mismatch with kernel\n");
|
|
|
|
else if (efi_runtime_init()) {
|
2012-02-12 21:24:28 +00:00
|
|
|
efi_enabled = 0;
|
|
|
|
return;
|
|
|
|
}
|
2012-02-12 21:24:25 +00:00
|
|
|
|
2012-02-12 21:24:28 +00:00
|
|
|
if (efi_memmap_init()) {
|
|
|
|
efi_enabled = 0;
|
|
|
|
return;
|
|
|
|
}
|
2012-08-14 16:53:38 +00:00
|
|
|
#ifdef CONFIG_X86_32
|
2012-02-12 21:24:29 +00:00
|
|
|
if (efi_native) {
|
|
|
|
x86_platform.get_wallclock = efi_get_time;
|
|
|
|
x86_platform.set_wallclock = efi_set_rtc_mmss;
|
|
|
|
}
|
2012-08-14 16:53:38 +00:00
|
|
|
#endif
|
2009-09-10 02:48:56 +00:00
|
|
|
|
2008-01-30 12:31:19 +00:00
|
|
|
#if EFI_DEBUG
|
|
|
|
print_efi_memmap();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-05-05 19:19:43 +00:00
|
|
|
void __init efi_set_executable(efi_memory_desc_t *md, bool executable)
|
|
|
|
{
|
|
|
|
u64 addr, npages;
|
|
|
|
|
|
|
|
addr = md->virt_addr;
|
|
|
|
npages = md->num_pages;
|
|
|
|
|
|
|
|
memrange_efi_to_native(&addr, &npages);
|
|
|
|
|
|
|
|
if (executable)
|
|
|
|
set_memory_x(addr, npages);
|
|
|
|
else
|
|
|
|
set_memory_nx(addr, npages);
|
|
|
|
}
|
|
|
|
|
2008-01-30 12:33:55 +00:00
|
|
|
static void __init runtime_code_page_mkexec(void)
|
|
|
|
{
|
|
|
|
efi_memory_desc_t *md;
|
|
|
|
void *p;
|
|
|
|
|
|
|
|
/* Make EFI runtime service code area executable */
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
md = p;
|
2008-02-04 15:48:06 +00:00
|
|
|
|
|
|
|
if (md->type != EFI_RUNTIME_SERVICES_CODE)
|
|
|
|
continue;
|
|
|
|
|
2011-05-05 19:19:43 +00:00
|
|
|
efi_set_executable(md, true);
|
2008-01-30 12:33:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-30 12:31:19 +00: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)
|
|
|
|
{
|
2011-05-05 19:19:44 +00:00
|
|
|
efi_memory_desc_t *md, *prev_md = NULL;
|
2008-01-30 12:31:19 +00:00
|
|
|
efi_status_t status;
|
2008-02-04 15:48:06 +00:00
|
|
|
unsigned long size;
|
2009-03-04 02:58:33 +00:00
|
|
|
u64 end, systab, addr, npages, end_pfn;
|
2011-05-05 19:19:45 +00:00
|
|
|
void *p, *va, *new_memmap = NULL;
|
|
|
|
int count = 0;
|
2008-01-30 12:31:19 +00:00
|
|
|
|
|
|
|
efi.systab = NULL;
|
2011-05-05 19:19:44 +00:00
|
|
|
|
2012-02-12 21:24:29 +00:00
|
|
|
/*
|
|
|
|
* We don't do virtual mode, since we don't do runtime services, on
|
|
|
|
* non-native EFI
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!efi_native)
|
|
|
|
goto out;
|
|
|
|
|
2011-05-05 19:19:44 +00:00
|
|
|
/* Merge contiguous regions of the same type and attribute */
|
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
u64 prev_size;
|
|
|
|
md = p;
|
|
|
|
|
|
|
|
if (!prev_md) {
|
|
|
|
prev_md = md;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prev_md->type != md->type ||
|
|
|
|
prev_md->attribute != md->attribute) {
|
|
|
|
prev_md = md;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
prev_size = prev_md->num_pages << EFI_PAGE_SHIFT;
|
|
|
|
|
|
|
|
if (md->phys_addr == (prev_md->phys_addr + prev_size)) {
|
|
|
|
prev_md->num_pages += md->num_pages;
|
|
|
|
md->type = EFI_RESERVED_TYPE;
|
|
|
|
md->attribute = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
prev_md = md;
|
|
|
|
}
|
|
|
|
|
2008-01-30 12:31:19 +00:00
|
|
|
for (p = memmap.map; p < memmap.map_end; p += memmap.desc_size) {
|
|
|
|
md = p;
|
x86, efi: Retain boot service code until after switching to virtual mode
UEFI stands for "Unified Extensible Firmware Interface", where "Firmware"
is an ancient African word meaning "Why do something right when you can
do it so wrong that children will weep and brave adults will cower before
you", and "UEI" is Celtic for "We missed DOS so we burned it into your
ROMs". The UEFI specification provides for runtime services (ie, another
way for the operating system to be forced to depend on the firmware) and
we rely on these for certain trivial tasks such as setting up the
bootloader. But some hardware fails to work if we attempt to use these
runtime services from physical mode, and so we have to switch into virtual
mode. So far so dreadful.
The specification makes it clear that the operating system is free to do
whatever it wants with boot services code after ExitBootServices() has been
called. SetVirtualAddressMap() can't be called until ExitBootServices() has
been. So, obviously, a whole bunch of EFI implementations call into boot
services code when we do that. Since we've been charmingly naive and
trusted that the specification may be somehow relevant to the real world,
we've already stuffed a picture of a penguin or something in that address
space. And just to make things more entertaining, we've also marked it
non-executable.
This patch allocates the boot services regions during EFI init and makes
sure that they're executable. Then, after SetVirtualAddressMap(), it
discards them and everyone lives happily ever after. Except for the ones
who have to work on EFI, who live sad lives haunted by the knowledge that
someone's eventually going to write yet another firmware specification.
[ hpa: adding this to urgent with a stable tag since it fixes currently-broken
hardware. However, I do not know what the dependencies are and so I do
not know which -stable versions this may be a candidate for. ]
Signed-off-by: Matthew Garrett <mjg@redhat.com>
Link: http://lkml.kernel.org/r/1306331593-28715-1-git-send-email-mjg@redhat.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: <stable@kernel.org>
2011-05-25 13:53:13 +00:00
|
|
|
if (!(md->attribute & EFI_MEMORY_RUNTIME) &&
|
|
|
|
md->type != EFI_BOOT_SERVICES_CODE &&
|
|
|
|
md->type != EFI_BOOT_SERVICES_DATA)
|
2008-01-30 12:31:19 +00:00
|
|
|
continue;
|
2008-02-04 15:48:06 +00:00
|
|
|
|
|
|
|
size = md->num_pages << EFI_PAGE_SHIFT;
|
|
|
|
end = md->phys_addr + size;
|
|
|
|
|
2009-03-04 02:58:33 +00: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 15:48:06 +00:00
|
|
|
va = __va(md->phys_addr);
|
2008-01-30 12:31:19 +00: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 12:38:10 +00:00
|
|
|
va = efi_ioremap(md->phys_addr, size, md->type);
|
2008-02-04 15:48:06 +00:00
|
|
|
|
|
|
|
md->virt_addr = (u64) (unsigned long) va;
|
|
|
|
|
|
|
|
if (!va) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_err("ioremap of 0x%llX failed!\n",
|
2008-01-30 12:31:19 +00:00
|
|
|
(unsigned long long)md->phys_addr);
|
2008-02-04 15:48:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-02-25 07:18:37 +00: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-12 18:46:48 +00:00
|
|
|
|
2008-02-04 15:48:06 +00: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;
|
|
|
|
}
|
2011-05-05 19:19:45 +00:00
|
|
|
new_memmap = krealloc(new_memmap,
|
|
|
|
(count + 1) * memmap.desc_size,
|
|
|
|
GFP_KERNEL);
|
|
|
|
memcpy(new_memmap + (count * memmap.desc_size), md,
|
|
|
|
memmap.desc_size);
|
|
|
|
count++;
|
2008-01-30 12:31:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BUG_ON(!efi.systab);
|
|
|
|
|
|
|
|
status = phys_efi_set_virtual_address_map(
|
2011-05-05 19:19:45 +00:00
|
|
|
memmap.desc_size * count,
|
2008-01-30 12:31:19 +00:00
|
|
|
memmap.desc_size,
|
|
|
|
memmap.desc_version,
|
2011-05-05 19:19:45 +00:00
|
|
|
(efi_memory_desc_t *)__pa(new_memmap));
|
2008-01-30 12:31:19 +00:00
|
|
|
|
|
|
|
if (status != EFI_SUCCESS) {
|
2012-02-12 21:24:26 +00:00
|
|
|
pr_alert("Unable to switch EFI into virtual mode "
|
|
|
|
"(status=%lx)!\n", status);
|
2008-01-30 12:31:19 +00:00
|
|
|
panic("EFI call to SetVirtualAddressMap() failed!");
|
|
|
|
}
|
|
|
|
|
x86, efi: Retain boot service code until after switching to virtual mode
UEFI stands for "Unified Extensible Firmware Interface", where "Firmware"
is an ancient African word meaning "Why do something right when you can
do it so wrong that children will weep and brave adults will cower before
you", and "UEI" is Celtic for "We missed DOS so we burned it into your
ROMs". The UEFI specification provides for runtime services (ie, another
way for the operating system to be forced to depend on the firmware) and
we rely on these for certain trivial tasks such as setting up the
bootloader. But some hardware fails to work if we attempt to use these
runtime services from physical mode, and so we have to switch into virtual
mode. So far so dreadful.
The specification makes it clear that the operating system is free to do
whatever it wants with boot services code after ExitBootServices() has been
called. SetVirtualAddressMap() can't be called until ExitBootServices() has
been. So, obviously, a whole bunch of EFI implementations call into boot
services code when we do that. Since we've been charmingly naive and
trusted that the specification may be somehow relevant to the real world,
we've already stuffed a picture of a penguin or something in that address
space. And just to make things more entertaining, we've also marked it
non-executable.
This patch allocates the boot services regions during EFI init and makes
sure that they're executable. Then, after SetVirtualAddressMap(), it
discards them and everyone lives happily ever after. Except for the ones
who have to work on EFI, who live sad lives haunted by the knowledge that
someone's eventually going to write yet another firmware specification.
[ hpa: adding this to urgent with a stable tag since it fixes currently-broken
hardware. However, I do not know what the dependencies are and so I do
not know which -stable versions this may be a candidate for. ]
Signed-off-by: Matthew Garrett <mjg@redhat.com>
Link: http://lkml.kernel.org/r/1306331593-28715-1-git-send-email-mjg@redhat.com
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: <stable@kernel.org>
2011-05-25 13:53:13 +00:00
|
|
|
/*
|
|
|
|
* Thankfully, it does seem that no runtime services other than
|
|
|
|
* SetVirtualAddressMap() will touch boot services code, so we can
|
|
|
|
* get rid of it all at this point
|
|
|
|
*/
|
|
|
|
efi_free_boot_services();
|
|
|
|
|
2008-01-30 12:31:19 +00:00
|
|
|
/*
|
|
|
|
* 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;
|
2011-05-05 19:19:42 +00:00
|
|
|
efi.set_virtual_address_map = NULL;
|
2011-06-06 19:36:25 +00:00
|
|
|
efi.query_variable_info = virt_efi_query_variable_info;
|
|
|
|
efi.update_capsule = virt_efi_update_capsule;
|
|
|
|
efi.query_capsule_caps = virt_efi_query_capsule_caps;
|
2008-02-13 09:22:41 +00:00
|
|
|
if (__supported_pte_mask & _PAGE_NX)
|
|
|
|
runtime_code_page_mkexec();
|
2012-02-12 21:24:29 +00:00
|
|
|
|
|
|
|
out:
|
2008-01-30 12:34:10 +00:00
|
|
|
early_iounmap(memmap.map, memmap.nr_map * memmap.desc_size);
|
|
|
|
memmap.map = NULL;
|
2011-05-05 19:19:45 +00:00
|
|
|
kfree(new_memmap);
|
2008-01-30 12:31:19 +00: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;
|
|
|
|
}
|