arm-runtime.c 3.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
/*
 * Extensible Firmware Interface
 *
 * Based on Extensible Firmware Interface Specification version 2.4
 *
 * Copyright (C) 2013, 2014 Linaro Ltd.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 *
 */

#include <linux/efi.h>
15
#include <linux/io.h>
16 17 18 19 20 21 22 23 24 25 26 27
#include <linux/memblock.h>
#include <linux/mm_types.h>
#include <linux/preempt.h>
#include <linux/rbtree.h>
#include <linux/rwsem.h>
#include <linux/sched.h>
#include <linux/slab.h>
#include <linux/spinlock.h>

#include <asm/cacheflush.h>
#include <asm/efi.h>
#include <asm/mmu.h>
28
#include <asm/pgalloc.h>
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
#include <asm/pgtable.h>

extern u64 efi_system_table;

static struct mm_struct efi_mm = {
	.mm_rb			= RB_ROOT,
	.mm_users		= ATOMIC_INIT(2),
	.mm_count		= ATOMIC_INIT(1),
	.mmap_sem		= __RWSEM_INITIALIZER(efi_mm.mmap_sem),
	.page_table_lock	= __SPIN_LOCK_UNLOCKED(efi_mm.page_table_lock),
	.mmlist			= LIST_HEAD_INIT(efi_mm.mmlist),
};

static bool __init efi_virtmap_init(void)
{
	efi_memory_desc_t *md;
45
	bool systab_found;
46

47
	efi_mm.pgd = pgd_alloc(&efi_mm);
48 49
	init_new_context(NULL, &efi_mm);

50
	systab_found = false;
51
	for_each_efi_memory_desc(md) {
52 53
		phys_addr_t phys = md->phys_addr;
		int ret;
54 55 56 57 58 59

		if (!(md->attribute & EFI_MEMORY_RUNTIME))
			continue;
		if (md->virt_addr == 0)
			return false;

60 61 62 63 64 65 66 67 68
		ret = efi_create_mapping(&efi_mm, md);
		if  (!ret) {
			pr_info("  EFI remap %pa => %p\n",
				&phys, (void *)(unsigned long)md->virt_addr);
		} else {
			pr_warn("  EFI remap %pa: failed to create mapping (%d)\n",
				&phys, ret);
			return false;
		}
69 70 71 72 73 74 75 76 77 78
		/*
		 * If this entry covers the address of the UEFI system table,
		 * calculate and record its virtual address.
		 */
		if (efi_system_table >= phys &&
		    efi_system_table < phys + (md->num_pages * EFI_PAGE_SIZE)) {
			efi.systab = (void *)(unsigned long)(efi_system_table -
							     phys + md->virt_addr);
			systab_found = true;
		}
79
	}
80
	if (!systab_found) {
81
		pr_err("No virtual mapping found for the UEFI System Table\n");
82 83 84 85 86 87 88
		return false;
	}

	if (efi_memattr_apply_permissions(&efi_mm, efi_set_mapping_permissions))
		return false;

	return true;
89 90 91 92 93 94 95
}

/*
 * Enable the UEFI Runtime Services if all prerequisites are in place, i.e.,
 * non-early mapping of the UEFI system table and virtual mappings for all
 * EFI_MEMORY_RUNTIME regions.
 */
96
static int __init arm_enable_runtime_services(void)
97 98 99 100 101 102 103 104 105 106 107 108 109
{
	u64 mapsize;

	if (!efi_enabled(EFI_BOOT)) {
		pr_info("EFI services will not be available.\n");
		return 0;
	}

	if (efi_runtime_disabled()) {
		pr_info("EFI runtime services will be disabled.\n");
		return 0;
	}

110 111 112 113 114
	if (efi_enabled(EFI_RUNTIME_SERVICES)) {
		pr_info("EFI runtime services access via paravirt.\n");
		return 0;
	}

115 116
	pr_info("Remapping and enabling EFI services.\n");

117 118
	mapsize = efi.memmap.map_end - efi.memmap.map;

119
	efi.memmap.map = memremap(efi.memmap.phys_map, mapsize, MEMREMAP_WB);
120
	if (!efi.memmap.map) {
121 122 123
		pr_err("Failed to remap EFI memory map\n");
		return -ENOMEM;
	}
124
	efi.memmap.map_end = efi.memmap.map + mapsize;
125 126

	if (!efi_virtmap_init()) {
127
		pr_err("UEFI virtual mapping missing or invalid -- runtime services will not be available\n");
128 129 130 131 132 133 134 135 136
		return -ENOMEM;
	}

	/* Set up runtime services function pointers */
	efi_native_runtime_setup();
	set_bit(EFI_RUNTIME_SERVICES, &efi.flags);

	return 0;
}
137
early_initcall(arm_enable_runtime_services);
138 139 140 141 142 143 144 145 146 147 148 149

void efi_virtmap_load(void)
{
	preempt_disable();
	efi_set_pgd(&efi_mm);
}

void efi_virtmap_unload(void)
{
	efi_set_pgd(current->active_mm);
	preempt_enable();
}