Skip to content

Commit 10c091b

Browse files
committed
Merge tag 'efi-urgent-2020-08-23' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip
Pull EFI fixes from Thomas Gleixner: - Enforce NX on RO data in mixed EFI mode - Destroy workqueue in an error handling path to prevent UAF - Stop argument parser at '--' which is the delimiter for init - Treat a NULL command line pointer as empty instead of dereferncing it unconditionally. - Handle an unterminated command line correctly - Cleanup the 32bit code leftovers and remove obsolete documentation * tag 'efi-urgent-2020-08-23' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: Documentation: efi: remove description of efi=old_map efi/x86: Move 32-bit code into efi_32.c efi/libstub: Handle unterminated cmdline efi/libstub: Handle NULL cmdline efi/libstub: Stop parsing arguments at "--" efi: add missed destroy_workqueue when efisubsys_init fails efi/x86: Mark kernel rodata non-executable for mixed mode
2 parents e99b250 + fb1201a commit 10c091b

File tree

7 files changed

+52
-92
lines changed

7 files changed

+52
-92
lines changed

Documentation/admin-guide/kernel-parameters.txt

+1-4
Original file line numberDiff line numberDiff line change
@@ -1233,8 +1233,7 @@
12331233
efi= [EFI]
12341234
Format: { "debug", "disable_early_pci_dma",
12351235
"nochunk", "noruntime", "nosoftreserve",
1236-
"novamap", "no_disable_early_pci_dma",
1237-
"old_map" }
1236+
"novamap", "no_disable_early_pci_dma" }
12381237
debug: enable misc debug output.
12391238
disable_early_pci_dma: disable the busmaster bit on all
12401239
PCI bridges while in the EFI boot stub.
@@ -1251,8 +1250,6 @@
12511250
novamap: do not call SetVirtualAddressMap().
12521251
no_disable_early_pci_dma: Leave the busmaster bit set
12531252
on all PCI bridges while in the EFI boot stub
1254-
old_map [X86-64]: switch to the old ioremap-based EFI
1255-
runtime services mapping. [Needs CONFIG_X86_UV=y]
12561253

12571254
efi_no_storage_paranoia [EFI; X86]
12581255
Using this parameter you can use more than 50% of

arch/x86/include/asm/efi.h

-10
Original file line numberDiff line numberDiff line change
@@ -81,11 +81,8 @@ extern unsigned long efi_fw_vendor, efi_config_table;
8181
kernel_fpu_end(); \
8282
})
8383

84-
8584
#define arch_efi_call_virt(p, f, args...) p->f(args)
8685

87-
#define efi_ioremap(addr, size, type, attr) ioremap_cache(addr, size)
88-
8986
#else /* !CONFIG_X86_32 */
9087

9188
#define EFI_LOADER_SIGNATURE "EL64"
@@ -125,9 +122,6 @@ struct efi_scratch {
125122
kernel_fpu_end(); \
126123
})
127124

128-
extern void __iomem *__init efi_ioremap(unsigned long addr, unsigned long size,
129-
u32 type, u64 attribute);
130-
131125
#ifdef CONFIG_KASAN
132126
/*
133127
* CONFIG_KASAN may redefine memset to __memset. __memset function is present
@@ -143,17 +137,13 @@ extern void __iomem *__init efi_ioremap(unsigned long addr, unsigned long size,
143137
#endif /* CONFIG_X86_32 */
144138

145139
extern struct efi_scratch efi_scratch;
146-
extern void __init efi_set_executable(efi_memory_desc_t *md, bool executable);
147140
extern int __init efi_memblock_x86_reserve_range(void);
148141
extern void __init efi_print_memmap(void);
149-
extern void __init efi_memory_uc(u64 addr, unsigned long size);
150142
extern void __init efi_map_region(efi_memory_desc_t *md);
151143
extern void __init efi_map_region_fixed(efi_memory_desc_t *md);
152144
extern void efi_sync_low_kernel_mappings(void);
153145
extern int __init efi_alloc_page_tables(void);
154146
extern int __init efi_setup_page_tables(unsigned long pa_memmap, unsigned num_pages);
155-
extern void __init old_map_region(efi_memory_desc_t *md);
156-
extern void __init runtime_code_page_mkexec(void);
157147
extern void __init efi_runtime_update_mappings(void);
158148
extern void __init efi_dump_pagetable(void);
159149
extern void __init efi_apply_memmap_quirks(void);

arch/x86/platform/efi/efi.c

-69
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,6 @@
4949
#include <asm/efi.h>
5050
#include <asm/e820/api.h>
5151
#include <asm/time.h>
52-
#include <asm/set_memory.h>
5352
#include <asm/tlbflush.h>
5453
#include <asm/x86_init.h>
5554
#include <asm/uv/uv.h>
@@ -496,74 +495,6 @@ void __init efi_init(void)
496495
efi_print_memmap();
497496
}
498497

499-
#if defined(CONFIG_X86_32)
500-
501-
void __init efi_set_executable(efi_memory_desc_t *md, bool executable)
502-
{
503-
u64 addr, npages;
504-
505-
addr = md->virt_addr;
506-
npages = md->num_pages;
507-
508-
memrange_efi_to_native(&addr, &npages);
509-
510-
if (executable)
511-
set_memory_x(addr, npages);
512-
else
513-
set_memory_nx(addr, npages);
514-
}
515-
516-
void __init runtime_code_page_mkexec(void)
517-
{
518-
efi_memory_desc_t *md;
519-
520-
/* Make EFI runtime service code area executable */
521-
for_each_efi_memory_desc(md) {
522-
if (md->type != EFI_RUNTIME_SERVICES_CODE)
523-
continue;
524-
525-
efi_set_executable(md, true);
526-
}
527-
}
528-
529-
void __init efi_memory_uc(u64 addr, unsigned long size)
530-
{
531-
unsigned long page_shift = 1UL << EFI_PAGE_SHIFT;
532-
u64 npages;
533-
534-
npages = round_up(size, page_shift) / page_shift;
535-
memrange_efi_to_native(&addr, &npages);
536-
set_memory_uc(addr, npages);
537-
}
538-
539-
void __init old_map_region(efi_memory_desc_t *md)
540-
{
541-
u64 start_pfn, end_pfn, end;
542-
unsigned long size;
543-
void *va;
544-
545-
start_pfn = PFN_DOWN(md->phys_addr);
546-
size = md->num_pages << PAGE_SHIFT;
547-
end = md->phys_addr + size;
548-
end_pfn = PFN_UP(end);
549-
550-
if (pfn_range_is_mapped(start_pfn, end_pfn)) {
551-
va = __va(md->phys_addr);
552-
553-
if (!(md->attribute & EFI_MEMORY_WB))
554-
efi_memory_uc((u64)(unsigned long)va, size);
555-
} else
556-
va = efi_ioremap(md->phys_addr, size,
557-
md->type, md->attribute);
558-
559-
md->virt_addr = (u64) (unsigned long) va;
560-
if (!va)
561-
pr_err("ioremap of 0x%llX failed!\n",
562-
(unsigned long long)md->phys_addr);
563-
}
564-
565-
#endif
566-
567498
/* Merge contiguous regions of the same type and attribute */
568499
static void __init efi_merge_regions(void)
569500
{

arch/x86/platform/efi/efi_32.c

+37-7
Original file line numberDiff line numberDiff line change
@@ -29,9 +29,35 @@
2929
#include <asm/io.h>
3030
#include <asm/desc.h>
3131
#include <asm/page.h>
32+
#include <asm/set_memory.h>
3233
#include <asm/tlbflush.h>
3334
#include <asm/efi.h>
3435

36+
void __init efi_map_region(efi_memory_desc_t *md)
37+
{
38+
u64 start_pfn, end_pfn, end;
39+
unsigned long size;
40+
void *va;
41+
42+
start_pfn = PFN_DOWN(md->phys_addr);
43+
size = md->num_pages << PAGE_SHIFT;
44+
end = md->phys_addr + size;
45+
end_pfn = PFN_UP(end);
46+
47+
if (pfn_range_is_mapped(start_pfn, end_pfn)) {
48+
va = __va(md->phys_addr);
49+
50+
if (!(md->attribute & EFI_MEMORY_WB))
51+
set_memory_uc((unsigned long)va, md->num_pages);
52+
} else {
53+
va = ioremap_cache(md->phys_addr, size);
54+
}
55+
56+
md->virt_addr = (unsigned long)va;
57+
if (!va)
58+
pr_err("ioremap of 0x%llX failed!\n", md->phys_addr);
59+
}
60+
3561
/*
3662
* To make EFI call EFI runtime service in physical addressing mode we need
3763
* prolog/epilog before/after the invocation to claim the EFI runtime service
@@ -58,11 +84,6 @@ int __init efi_setup_page_tables(unsigned long pa_memmap, unsigned num_pages)
5884
return 0;
5985
}
6086

61-
void __init efi_map_region(efi_memory_desc_t *md)
62-
{
63-
old_map_region(md);
64-
}
65-
6687
void __init efi_map_region_fixed(efi_memory_desc_t *md) {}
6788
void __init parse_efi_setup(u64 phys_addr, u32 data_len) {}
6889

@@ -107,6 +128,15 @@ efi_status_t __init efi_set_virtual_address_map(unsigned long memory_map_size,
107128

108129
void __init efi_runtime_update_mappings(void)
109130
{
110-
if (__supported_pte_mask & _PAGE_NX)
111-
runtime_code_page_mkexec();
131+
if (__supported_pte_mask & _PAGE_NX) {
132+
efi_memory_desc_t *md;
133+
134+
/* Make EFI runtime service code area executable */
135+
for_each_efi_memory_desc(md) {
136+
if (md->type != EFI_RUNTIME_SERVICES_CODE)
137+
continue;
138+
139+
set_memory_x(md->virt_addr, md->num_pages);
140+
}
141+
}
112142
}

arch/x86/platform/efi/efi_64.c

+2
Original file line numberDiff line numberDiff line change
@@ -259,6 +259,8 @@ int __init efi_setup_page_tables(unsigned long pa_memmap, unsigned num_pages)
259259
npages = (__end_rodata - __start_rodata) >> PAGE_SHIFT;
260260
rodata = __pa(__start_rodata);
261261
pfn = rodata >> PAGE_SHIFT;
262+
263+
pf = _PAGE_NX | _PAGE_ENC;
262264
if (kernel_map_pages_in_pgd(pgd, pfn, rodata, npages, pf)) {
263265
pr_err("Failed to map kernel rodata 1:1\n");
264266
return 1;

drivers/firmware/efi/efi.c

+2
Original file line numberDiff line numberDiff line change
@@ -381,6 +381,7 @@ static int __init efisubsys_init(void)
381381
efi_kobj = kobject_create_and_add("efi", firmware_kobj);
382382
if (!efi_kobj) {
383383
pr_err("efi: Firmware registration failed.\n");
384+
destroy_workqueue(efi_rts_wq);
384385
return -ENOMEM;
385386
}
386387

@@ -424,6 +425,7 @@ static int __init efisubsys_init(void)
424425
generic_ops_unregister();
425426
err_put:
426427
kobject_put(efi_kobj);
428+
destroy_workqueue(efi_rts_wq);
427429
return error;
428430
}
429431

drivers/firmware/efi/libstub/efi-stub-helper.c

+10-2
Original file line numberDiff line numberDiff line change
@@ -187,20 +187,28 @@ int efi_printk(const char *fmt, ...)
187187
*/
188188
efi_status_t efi_parse_options(char const *cmdline)
189189
{
190-
size_t len = strlen(cmdline) + 1;
190+
size_t len;
191191
efi_status_t status;
192192
char *str, *buf;
193193

194+
if (!cmdline)
195+
return EFI_SUCCESS;
196+
197+
len = strnlen(cmdline, COMMAND_LINE_SIZE - 1) + 1;
194198
status = efi_bs_call(allocate_pool, EFI_LOADER_DATA, len, (void **)&buf);
195199
if (status != EFI_SUCCESS)
196200
return status;
197201

198-
str = skip_spaces(memcpy(buf, cmdline, len));
202+
memcpy(buf, cmdline, len - 1);
203+
buf[len - 1] = '\0';
204+
str = skip_spaces(buf);
199205

200206
while (*str) {
201207
char *param, *val;
202208

203209
str = next_arg(str, &param, &val);
210+
if (!val && !strcmp(param, "--"))
211+
break;
204212

205213
if (!strcmp(param, "nokaslr")) {
206214
efi_nokaslr = true;

0 commit comments

Comments
 (0)