diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2014-06-05 11:16:29 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2014-06-05 11:16:29 -0400 |
commit | 046f153343e33dcad1be7f6249ea6ff1c6fd9b58 (patch) | |
tree | 46da03ea84e1a4fe8eff53891332e715cbf31f3e /drivers/firmware | |
parent | a0abcf2e8f8017051830f738ac1bf5ef42703243 (diff) | |
parent | e33655a386ed3b26ad36fb97a47ebb1c2ca1e928 (diff) |
Merge branch 'x86-efi-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip into next
Pull x86 EFI updates from Peter Anvin:
"A collection of EFI changes. The perhaps most important one is to
fully save and restore the FPU state around each invocation of EFI
runtime, and to not choke on non-ASCII characters in the boot stub"
* 'x86-efi-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip:
efivars: Add compatibility code for compat tasks
efivars: Refactor sanity checking code into separate function
efivars: Stop passing a struct argument to efivar_validate()
efivars: Check size of user object
efivars: Use local variables instead of a pointer dereference
x86/efi: Save and restore FPU context around efi_calls (i386)
x86/efi: Save and restore FPU context around efi_calls (x86_64)
x86/efi: Implement a __efi_call_virt macro
x86, fpu: Extend the use of static_cpu_has_safe
x86/efi: Delete most of the efi_call* macros
efi: x86: Handle arbitrary Unicode characters
efi: Add get_dram_base() helper function
efi: Add shared printk wrapper for consistent prefixing
efi: create memory map iteration helper
efi: efi-stub-helper cleanup
Diffstat (limited to 'drivers/firmware')
-rw-r--r-- | drivers/firmware/efi/efi-stub-helper.c | 144 | ||||
-rw-r--r-- | drivers/firmware/efi/efivars.c | 192 | ||||
-rw-r--r-- | drivers/firmware/efi/vars.c | 30 |
3 files changed, 284 insertions, 82 deletions
diff --git a/drivers/firmware/efi/efi-stub-helper.c b/drivers/firmware/efi/efi-stub-helper.c index 2c41eaece2c1..eb6d4be9e722 100644 --- a/drivers/firmware/efi/efi-stub-helper.c +++ b/drivers/firmware/efi/efi-stub-helper.c | |||
@@ -11,6 +11,10 @@ | |||
11 | */ | 11 | */ |
12 | #define EFI_READ_CHUNK_SIZE (1024 * 1024) | 12 | #define EFI_READ_CHUNK_SIZE (1024 * 1024) |
13 | 13 | ||
14 | /* error code which can't be mistaken for valid address */ | ||
15 | #define EFI_ERROR (~0UL) | ||
16 | |||
17 | |||
14 | struct file_info { | 18 | struct file_info { |
15 | efi_file_handle_t *handle; | 19 | efi_file_handle_t *handle; |
16 | u64 size; | 20 | u64 size; |
@@ -33,6 +37,9 @@ static void efi_printk(efi_system_table_t *sys_table_arg, char *str) | |||
33 | } | 37 | } |
34 | } | 38 | } |
35 | 39 | ||
40 | #define pr_efi(sys_table, msg) efi_printk(sys_table, "EFI stub: "msg) | ||
41 | #define pr_efi_err(sys_table, msg) efi_printk(sys_table, "EFI stub: ERROR: "msg) | ||
42 | |||
36 | 43 | ||
37 | static efi_status_t efi_get_memory_map(efi_system_table_t *sys_table_arg, | 44 | static efi_status_t efi_get_memory_map(efi_system_table_t *sys_table_arg, |
38 | efi_memory_desc_t **map, | 45 | efi_memory_desc_t **map, |
@@ -80,6 +87,32 @@ fail: | |||
80 | return status; | 87 | return status; |
81 | } | 88 | } |
82 | 89 | ||
90 | |||
91 | static unsigned long __init get_dram_base(efi_system_table_t *sys_table_arg) | ||
92 | { | ||
93 | efi_status_t status; | ||
94 | unsigned long map_size; | ||
95 | unsigned long membase = EFI_ERROR; | ||
96 | struct efi_memory_map map; | ||
97 | efi_memory_desc_t *md; | ||
98 | |||
99 | status = efi_get_memory_map(sys_table_arg, (efi_memory_desc_t **)&map.map, | ||
100 | &map_size, &map.desc_size, NULL, NULL); | ||
101 | if (status != EFI_SUCCESS) | ||
102 | return membase; | ||
103 | |||
104 | map.map_end = map.map + map_size; | ||
105 | |||
106 | for_each_efi_memory_desc(&map, md) | ||
107 | if (md->attribute & EFI_MEMORY_WB) | ||
108 | if (membase > md->phys_addr) | ||
109 | membase = md->phys_addr; | ||
110 | |||
111 | efi_call_early(free_pool, map.map); | ||
112 | |||
113 | return membase; | ||
114 | } | ||
115 | |||
83 | /* | 116 | /* |
84 | * Allocate at the highest possible address that is not above 'max'. | 117 | * Allocate at the highest possible address that is not above 'max'. |
85 | */ | 118 | */ |
@@ -267,7 +300,7 @@ static efi_status_t handle_cmdline_files(efi_system_table_t *sys_table_arg, | |||
267 | struct file_info *files; | 300 | struct file_info *files; |
268 | unsigned long file_addr; | 301 | unsigned long file_addr; |
269 | u64 file_size_total; | 302 | u64 file_size_total; |
270 | efi_file_handle_t *fh; | 303 | efi_file_handle_t *fh = NULL; |
271 | efi_status_t status; | 304 | efi_status_t status; |
272 | int nr_files; | 305 | int nr_files; |
273 | char *str; | 306 | char *str; |
@@ -310,7 +343,7 @@ static efi_status_t handle_cmdline_files(efi_system_table_t *sys_table_arg, | |||
310 | status = efi_call_early(allocate_pool, EFI_LOADER_DATA, | 343 | status = efi_call_early(allocate_pool, EFI_LOADER_DATA, |
311 | nr_files * sizeof(*files), (void **)&files); | 344 | nr_files * sizeof(*files), (void **)&files); |
312 | if (status != EFI_SUCCESS) { | 345 | if (status != EFI_SUCCESS) { |
313 | efi_printk(sys_table_arg, "Failed to alloc mem for file handle list\n"); | 346 | pr_efi_err(sys_table_arg, "Failed to alloc mem for file handle list\n"); |
314 | goto fail; | 347 | goto fail; |
315 | } | 348 | } |
316 | 349 | ||
@@ -374,13 +407,13 @@ static efi_status_t handle_cmdline_files(efi_system_table_t *sys_table_arg, | |||
374 | status = efi_high_alloc(sys_table_arg, file_size_total, 0x1000, | 407 | status = efi_high_alloc(sys_table_arg, file_size_total, 0x1000, |
375 | &file_addr, max_addr); | 408 | &file_addr, max_addr); |
376 | if (status != EFI_SUCCESS) { | 409 | if (status != EFI_SUCCESS) { |
377 | efi_printk(sys_table_arg, "Failed to alloc highmem for files\n"); | 410 | pr_efi_err(sys_table_arg, "Failed to alloc highmem for files\n"); |
378 | goto close_handles; | 411 | goto close_handles; |
379 | } | 412 | } |
380 | 413 | ||
381 | /* We've run out of free low memory. */ | 414 | /* We've run out of free low memory. */ |
382 | if (file_addr > max_addr) { | 415 | if (file_addr > max_addr) { |
383 | efi_printk(sys_table_arg, "We've run out of free low memory\n"); | 416 | pr_efi_err(sys_table_arg, "We've run out of free low memory\n"); |
384 | status = EFI_INVALID_PARAMETER; | 417 | status = EFI_INVALID_PARAMETER; |
385 | goto free_file_total; | 418 | goto free_file_total; |
386 | } | 419 | } |
@@ -401,7 +434,7 @@ static efi_status_t handle_cmdline_files(efi_system_table_t *sys_table_arg, | |||
401 | &chunksize, | 434 | &chunksize, |
402 | (void *)addr); | 435 | (void *)addr); |
403 | if (status != EFI_SUCCESS) { | 436 | if (status != EFI_SUCCESS) { |
404 | efi_printk(sys_table_arg, "Failed to read file\n"); | 437 | pr_efi_err(sys_table_arg, "Failed to read file\n"); |
405 | goto free_file_total; | 438 | goto free_file_total; |
406 | } | 439 | } |
407 | addr += chunksize; | 440 | addr += chunksize; |
@@ -486,7 +519,7 @@ static efi_status_t efi_relocate_kernel(efi_system_table_t *sys_table_arg, | |||
486 | &new_addr); | 519 | &new_addr); |
487 | } | 520 | } |
488 | if (status != EFI_SUCCESS) { | 521 | if (status != EFI_SUCCESS) { |
489 | efi_printk(sys_table_arg, "ERROR: Failed to allocate usable memory for kernel.\n"); | 522 | pr_efi_err(sys_table_arg, "Failed to allocate usable memory for kernel.\n"); |
490 | return status; | 523 | return status; |
491 | } | 524 | } |
492 | 525 | ||
@@ -503,62 +536,99 @@ static efi_status_t efi_relocate_kernel(efi_system_table_t *sys_table_arg, | |||
503 | } | 536 | } |
504 | 537 | ||
505 | /* | 538 | /* |
539 | * Get the number of UTF-8 bytes corresponding to an UTF-16 character. | ||
540 | * This overestimates for surrogates, but that is okay. | ||
541 | */ | ||
542 | static int efi_utf8_bytes(u16 c) | ||
543 | { | ||
544 | return 1 + (c >= 0x80) + (c >= 0x800); | ||
545 | } | ||
546 | |||
547 | /* | ||
548 | * Convert an UTF-16 string, not necessarily null terminated, to UTF-8. | ||
549 | */ | ||
550 | static u8 *efi_utf16_to_utf8(u8 *dst, const u16 *src, int n) | ||
551 | { | ||
552 | unsigned int c; | ||
553 | |||
554 | while (n--) { | ||
555 | c = *src++; | ||
556 | if (n && c >= 0xd800 && c <= 0xdbff && | ||
557 | *src >= 0xdc00 && *src <= 0xdfff) { | ||
558 | c = 0x10000 + ((c & 0x3ff) << 10) + (*src & 0x3ff); | ||
559 | src++; | ||
560 | n--; | ||
561 | } | ||
562 | if (c >= 0xd800 && c <= 0xdfff) | ||
563 | c = 0xfffd; /* Unmatched surrogate */ | ||
564 | if (c < 0x80) { | ||
565 | *dst++ = c; | ||
566 | continue; | ||
567 | } | ||
568 | if (c < 0x800) { | ||
569 | *dst++ = 0xc0 + (c >> 6); | ||
570 | goto t1; | ||
571 | } | ||
572 | if (c < 0x10000) { | ||
573 | *dst++ = 0xe0 + (c >> 12); | ||
574 | goto t2; | ||
575 | } | ||
576 | *dst++ = 0xf0 + (c >> 18); | ||
577 | *dst++ = 0x80 + ((c >> 12) & 0x3f); | ||
578 | t2: | ||
579 | *dst++ = 0x80 + ((c >> 6) & 0x3f); | ||
580 | t1: | ||
581 | *dst++ = 0x80 + (c & 0x3f); | ||
582 | } | ||
583 | |||
584 | return dst; | ||
585 | } | ||
586 | |||
587 | /* | ||
506 | * Convert the unicode UEFI command line to ASCII to pass to kernel. | 588 | * Convert the unicode UEFI command line to ASCII to pass to kernel. |
507 | * Size of memory allocated return in *cmd_line_len. | 589 | * Size of memory allocated return in *cmd_line_len. |
508 | * Returns NULL on error. | 590 | * Returns NULL on error. |
509 | */ | 591 | */ |
510 | static char *efi_convert_cmdline_to_ascii(efi_system_table_t *sys_table_arg, | 592 | static char *efi_convert_cmdline(efi_system_table_t *sys_table_arg, |
511 | efi_loaded_image_t *image, | 593 | efi_loaded_image_t *image, |
512 | int *cmd_line_len) | 594 | int *cmd_line_len) |
513 | { | 595 | { |
514 | u16 *s2; | 596 | const u16 *s2; |
515 | u8 *s1 = NULL; | 597 | u8 *s1 = NULL; |
516 | unsigned long cmdline_addr = 0; | 598 | unsigned long cmdline_addr = 0; |
517 | int load_options_size = image->load_options_size / 2; /* ASCII */ | 599 | int load_options_chars = image->load_options_size / 2; /* UTF-16 */ |
518 | void *options = image->load_options; | 600 | const u16 *options = image->load_options; |
519 | int options_size = 0; | 601 | int options_bytes = 0; /* UTF-8 bytes */ |
602 | int options_chars = 0; /* UTF-16 chars */ | ||
520 | efi_status_t status; | 603 | efi_status_t status; |
521 | int i; | ||
522 | u16 zero = 0; | 604 | u16 zero = 0; |
523 | 605 | ||
524 | if (options) { | 606 | if (options) { |
525 | s2 = options; | 607 | s2 = options; |
526 | while (*s2 && *s2 != '\n' && options_size < load_options_size) { | 608 | while (*s2 && *s2 != '\n' |
527 | s2++; | 609 | && options_chars < load_options_chars) { |
528 | options_size++; | 610 | options_bytes += efi_utf8_bytes(*s2++); |
611 | options_chars++; | ||
529 | } | 612 | } |
530 | } | 613 | } |
531 | 614 | ||
532 | if (options_size == 0) { | 615 | if (!options_chars) { |
533 | /* No command line options, so return empty string*/ | 616 | /* No command line options, so return empty string*/ |
534 | options_size = 1; | ||
535 | options = &zero; | 617 | options = &zero; |
536 | } | 618 | } |
537 | 619 | ||
538 | options_size++; /* NUL termination */ | 620 | options_bytes++; /* NUL termination */ |
539 | #ifdef CONFIG_ARM | 621 | |
540 | /* | 622 | status = efi_low_alloc(sys_table_arg, options_bytes, 0, &cmdline_addr); |
541 | * For ARM, allocate at a high address to avoid reserved | ||
542 | * regions at low addresses that we don't know the specfics of | ||
543 | * at the time we are processing the command line. | ||
544 | */ | ||
545 | status = efi_high_alloc(sys_table_arg, options_size, 0, | ||
546 | &cmdline_addr, 0xfffff000); | ||
547 | #else | ||
548 | status = efi_low_alloc(sys_table_arg, options_size, 0, | ||
549 | &cmdline_addr); | ||
550 | #endif | ||
551 | if (status != EFI_SUCCESS) | 623 | if (status != EFI_SUCCESS) |
552 | return NULL; | 624 | return NULL; |
553 | 625 | ||
554 | s1 = (u8 *)cmdline_addr; | 626 | s1 = (u8 *)cmdline_addr; |
555 | s2 = (u16 *)options; | 627 | s2 = (const u16 *)options; |
556 | |||
557 | for (i = 0; i < options_size - 1; i++) | ||
558 | *s1++ = *s2++; | ||
559 | 628 | ||
629 | s1 = efi_utf16_to_utf8(s1, s2, options_chars); | ||
560 | *s1 = '\0'; | 630 | *s1 = '\0'; |
561 | 631 | ||
562 | *cmd_line_len = options_size; | 632 | *cmd_line_len = options_bytes; |
563 | return (char *)cmdline_addr; | 633 | return (char *)cmdline_addr; |
564 | } | 634 | } |
diff --git a/drivers/firmware/efi/efivars.c b/drivers/firmware/efi/efivars.c index 50ea412a25e6..463c56545ae8 100644 --- a/drivers/firmware/efi/efivars.c +++ b/drivers/firmware/efi/efivars.c | |||
@@ -69,6 +69,7 @@ | |||
69 | #include <linux/module.h> | 69 | #include <linux/module.h> |
70 | #include <linux/slab.h> | 70 | #include <linux/slab.h> |
71 | #include <linux/ucs2_string.h> | 71 | #include <linux/ucs2_string.h> |
72 | #include <linux/compat.h> | ||
72 | 73 | ||
73 | #define EFIVARS_VERSION "0.08" | 74 | #define EFIVARS_VERSION "0.08" |
74 | #define EFIVARS_DATE "2004-May-17" | 75 | #define EFIVARS_DATE "2004-May-17" |
@@ -86,6 +87,15 @@ static struct kset *efivars_kset; | |||
86 | static struct bin_attribute *efivars_new_var; | 87 | static struct bin_attribute *efivars_new_var; |
87 | static struct bin_attribute *efivars_del_var; | 88 | static struct bin_attribute *efivars_del_var; |
88 | 89 | ||
90 | struct compat_efi_variable { | ||
91 | efi_char16_t VariableName[EFI_VAR_NAME_LEN/sizeof(efi_char16_t)]; | ||
92 | efi_guid_t VendorGuid; | ||
93 | __u32 DataSize; | ||
94 | __u8 Data[1024]; | ||
95 | __u32 Status; | ||
96 | __u32 Attributes; | ||
97 | } __packed; | ||
98 | |||
89 | struct efivar_attribute { | 99 | struct efivar_attribute { |
90 | struct attribute attr; | 100 | struct attribute attr; |
91 | ssize_t (*show) (struct efivar_entry *entry, char *buf); | 101 | ssize_t (*show) (struct efivar_entry *entry, char *buf); |
@@ -189,45 +199,107 @@ efivar_data_read(struct efivar_entry *entry, char *buf) | |||
189 | memcpy(buf, var->Data, var->DataSize); | 199 | memcpy(buf, var->Data, var->DataSize); |
190 | return var->DataSize; | 200 | return var->DataSize; |
191 | } | 201 | } |
192 | /* | ||
193 | * We allow each variable to be edited via rewriting the | ||
194 | * entire efi variable structure. | ||
195 | */ | ||
196 | static ssize_t | ||
197 | efivar_store_raw(struct efivar_entry *entry, const char *buf, size_t count) | ||
198 | { | ||
199 | struct efi_variable *new_var, *var = &entry->var; | ||
200 | int err; | ||
201 | 202 | ||
202 | if (count != sizeof(struct efi_variable)) | 203 | static inline int |
203 | return -EINVAL; | 204 | sanity_check(struct efi_variable *var, efi_char16_t *name, efi_guid_t vendor, |
204 | 205 | unsigned long size, u32 attributes, u8 *data) | |
205 | new_var = (struct efi_variable *)buf; | 206 | { |
206 | /* | 207 | /* |
207 | * If only updating the variable data, then the name | 208 | * If only updating the variable data, then the name |
208 | * and guid should remain the same | 209 | * and guid should remain the same |
209 | */ | 210 | */ |
210 | if (memcmp(new_var->VariableName, var->VariableName, sizeof(var->VariableName)) || | 211 | if (memcmp(name, var->VariableName, sizeof(var->VariableName)) || |
211 | efi_guidcmp(new_var->VendorGuid, var->VendorGuid)) { | 212 | efi_guidcmp(vendor, var->VendorGuid)) { |
212 | printk(KERN_ERR "efivars: Cannot edit the wrong variable!\n"); | 213 | printk(KERN_ERR "efivars: Cannot edit the wrong variable!\n"); |
213 | return -EINVAL; | 214 | return -EINVAL; |
214 | } | 215 | } |
215 | 216 | ||
216 | if ((new_var->DataSize <= 0) || (new_var->Attributes == 0)){ | 217 | if ((size <= 0) || (attributes == 0)){ |
217 | printk(KERN_ERR "efivars: DataSize & Attributes must be valid!\n"); | 218 | printk(KERN_ERR "efivars: DataSize & Attributes must be valid!\n"); |
218 | return -EINVAL; | 219 | return -EINVAL; |
219 | } | 220 | } |
220 | 221 | ||
221 | if ((new_var->Attributes & ~EFI_VARIABLE_MASK) != 0 || | 222 | if ((attributes & ~EFI_VARIABLE_MASK) != 0 || |
222 | efivar_validate(new_var, new_var->Data, new_var->DataSize) == false) { | 223 | efivar_validate(name, data, size) == false) { |
223 | printk(KERN_ERR "efivars: Malformed variable content\n"); | 224 | printk(KERN_ERR "efivars: Malformed variable content\n"); |
224 | return -EINVAL; | 225 | return -EINVAL; |
225 | } | 226 | } |
226 | 227 | ||
227 | memcpy(&entry->var, new_var, count); | 228 | return 0; |
229 | } | ||
230 | |||
231 | static inline bool is_compat(void) | ||
232 | { | ||
233 | if (IS_ENABLED(CONFIG_COMPAT) && is_compat_task()) | ||
234 | return true; | ||
235 | |||
236 | return false; | ||
237 | } | ||
238 | |||
239 | static void | ||
240 | copy_out_compat(struct efi_variable *dst, struct compat_efi_variable *src) | ||
241 | { | ||
242 | memcpy(dst->VariableName, src->VariableName, EFI_VAR_NAME_LEN); | ||
243 | memcpy(dst->Data, src->Data, sizeof(src->Data)); | ||
244 | |||
245 | dst->VendorGuid = src->VendorGuid; | ||
246 | dst->DataSize = src->DataSize; | ||
247 | dst->Attributes = src->Attributes; | ||
248 | } | ||
249 | |||
250 | /* | ||
251 | * We allow each variable to be edited via rewriting the | ||
252 | * entire efi variable structure. | ||
253 | */ | ||
254 | static ssize_t | ||
255 | efivar_store_raw(struct efivar_entry *entry, const char *buf, size_t count) | ||
256 | { | ||
257 | struct efi_variable *new_var, *var = &entry->var; | ||
258 | efi_char16_t *name; | ||
259 | unsigned long size; | ||
260 | efi_guid_t vendor; | ||
261 | u32 attributes; | ||
262 | u8 *data; | ||
263 | int err; | ||
264 | |||
265 | if (is_compat()) { | ||
266 | struct compat_efi_variable *compat; | ||
267 | |||
268 | if (count != sizeof(*compat)) | ||
269 | return -EINVAL; | ||
270 | |||
271 | compat = (struct compat_efi_variable *)buf; | ||
272 | attributes = compat->Attributes; | ||
273 | vendor = compat->VendorGuid; | ||
274 | name = compat->VariableName; | ||
275 | size = compat->DataSize; | ||
276 | data = compat->Data; | ||
277 | |||
278 | err = sanity_check(var, name, vendor, size, attributes, data); | ||
279 | if (err) | ||
280 | return err; | ||
281 | |||
282 | copy_out_compat(&entry->var, compat); | ||
283 | } else { | ||
284 | if (count != sizeof(struct efi_variable)) | ||
285 | return -EINVAL; | ||
286 | |||
287 | new_var = (struct efi_variable *)buf; | ||
228 | 288 | ||
229 | err = efivar_entry_set(entry, new_var->Attributes, | 289 | attributes = new_var->Attributes; |
230 | new_var->DataSize, new_var->Data, NULL); | 290 | vendor = new_var->VendorGuid; |
291 | name = new_var->VariableName; | ||
292 | size = new_var->DataSize; | ||
293 | data = new_var->Data; | ||
294 | |||
295 | err = sanity_check(var, name, vendor, size, attributes, data); | ||
296 | if (err) | ||
297 | return err; | ||
298 | |||
299 | memcpy(&entry->var, new_var, count); | ||
300 | } | ||
301 | |||
302 | err = efivar_entry_set(entry, attributes, size, data, NULL); | ||
231 | if (err) { | 303 | if (err) { |
232 | printk(KERN_WARNING "efivars: set_variable() failed: status=%d\n", err); | 304 | printk(KERN_WARNING "efivars: set_variable() failed: status=%d\n", err); |
233 | return -EIO; | 305 | return -EIO; |
@@ -240,6 +312,8 @@ static ssize_t | |||
240 | efivar_show_raw(struct efivar_entry *entry, char *buf) | 312 | efivar_show_raw(struct efivar_entry *entry, char *buf) |
241 | { | 313 | { |
242 | struct efi_variable *var = &entry->var; | 314 | struct efi_variable *var = &entry->var; |
315 | struct compat_efi_variable *compat; | ||
316 | size_t size; | ||
243 | 317 | ||
244 | if (!entry || !buf) | 318 | if (!entry || !buf) |
245 | return 0; | 319 | return 0; |
@@ -249,9 +323,23 @@ efivar_show_raw(struct efivar_entry *entry, char *buf) | |||
249 | &entry->var.DataSize, entry->var.Data)) | 323 | &entry->var.DataSize, entry->var.Data)) |
250 | return -EIO; | 324 | return -EIO; |
251 | 325 | ||
252 | memcpy(buf, var, sizeof(*var)); | 326 | if (is_compat()) { |
327 | compat = (struct compat_efi_variable *)buf; | ||
328 | |||
329 | size = sizeof(*compat); | ||
330 | memcpy(compat->VariableName, var->VariableName, | ||
331 | EFI_VAR_NAME_LEN); | ||
332 | memcpy(compat->Data, var->Data, sizeof(compat->Data)); | ||
333 | |||
334 | compat->VendorGuid = var->VendorGuid; | ||
335 | compat->DataSize = var->DataSize; | ||
336 | compat->Attributes = var->Attributes; | ||
337 | } else { | ||
338 | size = sizeof(*var); | ||
339 | memcpy(buf, var, size); | ||
340 | } | ||
253 | 341 | ||
254 | return sizeof(*var); | 342 | return size; |
255 | } | 343 | } |
256 | 344 | ||
257 | /* | 345 | /* |
@@ -326,15 +414,39 @@ static ssize_t efivar_create(struct file *filp, struct kobject *kobj, | |||
326 | struct bin_attribute *bin_attr, | 414 | struct bin_attribute *bin_attr, |
327 | char *buf, loff_t pos, size_t count) | 415 | char *buf, loff_t pos, size_t count) |
328 | { | 416 | { |
417 | struct compat_efi_variable *compat = (struct compat_efi_variable *)buf; | ||
329 | struct efi_variable *new_var = (struct efi_variable *)buf; | 418 | struct efi_variable *new_var = (struct efi_variable *)buf; |
330 | struct efivar_entry *new_entry; | 419 | struct efivar_entry *new_entry; |
420 | bool need_compat = is_compat(); | ||
421 | efi_char16_t *name; | ||
422 | unsigned long size; | ||
423 | u32 attributes; | ||
424 | u8 *data; | ||
331 | int err; | 425 | int err; |
332 | 426 | ||
333 | if (!capable(CAP_SYS_ADMIN)) | 427 | if (!capable(CAP_SYS_ADMIN)) |
334 | return -EACCES; | 428 | return -EACCES; |
335 | 429 | ||
336 | if ((new_var->Attributes & ~EFI_VARIABLE_MASK) != 0 || | 430 | if (need_compat) { |
337 | efivar_validate(new_var, new_var->Data, new_var->DataSize) == false) { | 431 | if (count != sizeof(*compat)) |
432 | return -EINVAL; | ||
433 | |||
434 | attributes = compat->Attributes; | ||
435 | name = compat->VariableName; | ||
436 | size = compat->DataSize; | ||
437 | data = compat->Data; | ||
438 | } else { | ||
439 | if (count != sizeof(*new_var)) | ||
440 | return -EINVAL; | ||
441 | |||
442 | attributes = new_var->Attributes; | ||
443 | name = new_var->VariableName; | ||
444 | size = new_var->DataSize; | ||
445 | data = new_var->Data; | ||
446 | } | ||
447 | |||
448 | if ((attributes & ~EFI_VARIABLE_MASK) != 0 || | ||
449 | efivar_validate(name, data, size) == false) { | ||
338 | printk(KERN_ERR "efivars: Malformed variable content\n"); | 450 | printk(KERN_ERR "efivars: Malformed variable content\n"); |
339 | return -EINVAL; | 451 | return -EINVAL; |
340 | } | 452 | } |
@@ -343,10 +455,13 @@ static ssize_t efivar_create(struct file *filp, struct kobject *kobj, | |||
343 | if (!new_entry) | 455 | if (!new_entry) |
344 | return -ENOMEM; | 456 | return -ENOMEM; |
345 | 457 | ||
346 | memcpy(&new_entry->var, new_var, sizeof(*new_var)); | 458 | if (need_compat) |
459 | copy_out_compat(&new_entry->var, compat); | ||
460 | else | ||
461 | memcpy(&new_entry->var, new_var, sizeof(*new_var)); | ||
347 | 462 | ||
348 | err = efivar_entry_set(new_entry, new_var->Attributes, new_var->DataSize, | 463 | err = efivar_entry_set(new_entry, attributes, size, |
349 | new_var->Data, &efivar_sysfs_list); | 464 | data, &efivar_sysfs_list); |
350 | if (err) { | 465 | if (err) { |
351 | if (err == -EEXIST) | 466 | if (err == -EEXIST) |
352 | err = -EINVAL; | 467 | err = -EINVAL; |
@@ -369,15 +484,32 @@ static ssize_t efivar_delete(struct file *filp, struct kobject *kobj, | |||
369 | char *buf, loff_t pos, size_t count) | 484 | char *buf, loff_t pos, size_t count) |
370 | { | 485 | { |
371 | struct efi_variable *del_var = (struct efi_variable *)buf; | 486 | struct efi_variable *del_var = (struct efi_variable *)buf; |
487 | struct compat_efi_variable *compat; | ||
372 | struct efivar_entry *entry; | 488 | struct efivar_entry *entry; |
489 | efi_char16_t *name; | ||
490 | efi_guid_t vendor; | ||
373 | int err = 0; | 491 | int err = 0; |
374 | 492 | ||
375 | if (!capable(CAP_SYS_ADMIN)) | 493 | if (!capable(CAP_SYS_ADMIN)) |
376 | return -EACCES; | 494 | return -EACCES; |
377 | 495 | ||
496 | if (is_compat()) { | ||
497 | if (count != sizeof(*compat)) | ||
498 | return -EINVAL; | ||
499 | |||
500 | compat = (struct compat_efi_variable *)buf; | ||
501 | name = compat->VariableName; | ||
502 | vendor = compat->VendorGuid; | ||
503 | } else { | ||
504 | if (count != sizeof(*del_var)) | ||
505 | return -EINVAL; | ||
506 | |||
507 | name = del_var->VariableName; | ||
508 | vendor = del_var->VendorGuid; | ||
509 | } | ||
510 | |||
378 | efivar_entry_iter_begin(); | 511 | efivar_entry_iter_begin(); |
379 | entry = efivar_entry_find(del_var->VariableName, del_var->VendorGuid, | 512 | entry = efivar_entry_find(name, vendor, &efivar_sysfs_list, true); |
380 | &efivar_sysfs_list, true); | ||
381 | if (!entry) | 513 | if (!entry) |
382 | err = -EINVAL; | 514 | err = -EINVAL; |
383 | else if (__efivar_entry_delete(entry)) | 515 | else if (__efivar_entry_delete(entry)) |
diff --git a/drivers/firmware/efi/vars.c b/drivers/firmware/efi/vars.c index b22659cccca4..f0a43646a2f3 100644 --- a/drivers/firmware/efi/vars.c +++ b/drivers/firmware/efi/vars.c | |||
@@ -42,7 +42,7 @@ DECLARE_WORK(efivar_work, NULL); | |||
42 | EXPORT_SYMBOL_GPL(efivar_work); | 42 | EXPORT_SYMBOL_GPL(efivar_work); |
43 | 43 | ||
44 | static bool | 44 | static bool |
45 | validate_device_path(struct efi_variable *var, int match, u8 *buffer, | 45 | validate_device_path(efi_char16_t *var_name, int match, u8 *buffer, |
46 | unsigned long len) | 46 | unsigned long len) |
47 | { | 47 | { |
48 | struct efi_generic_dev_path *node; | 48 | struct efi_generic_dev_path *node; |
@@ -75,7 +75,7 @@ validate_device_path(struct efi_variable *var, int match, u8 *buffer, | |||
75 | } | 75 | } |
76 | 76 | ||
77 | static bool | 77 | static bool |
78 | validate_boot_order(struct efi_variable *var, int match, u8 *buffer, | 78 | validate_boot_order(efi_char16_t *var_name, int match, u8 *buffer, |
79 | unsigned long len) | 79 | unsigned long len) |
80 | { | 80 | { |
81 | /* An array of 16-bit integers */ | 81 | /* An array of 16-bit integers */ |
@@ -86,18 +86,18 @@ validate_boot_order(struct efi_variable *var, int match, u8 *buffer, | |||
86 | } | 86 | } |
87 | 87 | ||
88 | static bool | 88 | static bool |
89 | validate_load_option(struct efi_variable *var, int match, u8 *buffer, | 89 | validate_load_option(efi_char16_t *var_name, int match, u8 *buffer, |
90 | unsigned long len) | 90 | unsigned long len) |
91 | { | 91 | { |
92 | u16 filepathlength; | 92 | u16 filepathlength; |
93 | int i, desclength = 0, namelen; | 93 | int i, desclength = 0, namelen; |
94 | 94 | ||
95 | namelen = ucs2_strnlen(var->VariableName, sizeof(var->VariableName)); | 95 | namelen = ucs2_strnlen(var_name, EFI_VAR_NAME_LEN); |
96 | 96 | ||
97 | /* Either "Boot" or "Driver" followed by four digits of hex */ | 97 | /* Either "Boot" or "Driver" followed by four digits of hex */ |
98 | for (i = match; i < match+4; i++) { | 98 | for (i = match; i < match+4; i++) { |
99 | if (var->VariableName[i] > 127 || | 99 | if (var_name[i] > 127 || |
100 | hex_to_bin(var->VariableName[i] & 0xff) < 0) | 100 | hex_to_bin(var_name[i] & 0xff) < 0) |
101 | return true; | 101 | return true; |
102 | } | 102 | } |
103 | 103 | ||
@@ -132,12 +132,12 @@ validate_load_option(struct efi_variable *var, int match, u8 *buffer, | |||
132 | /* | 132 | /* |
133 | * And, finally, check the filepath | 133 | * And, finally, check the filepath |
134 | */ | 134 | */ |
135 | return validate_device_path(var, match, buffer + desclength + 6, | 135 | return validate_device_path(var_name, match, buffer + desclength + 6, |
136 | filepathlength); | 136 | filepathlength); |
137 | } | 137 | } |
138 | 138 | ||
139 | static bool | 139 | static bool |
140 | validate_uint16(struct efi_variable *var, int match, u8 *buffer, | 140 | validate_uint16(efi_char16_t *var_name, int match, u8 *buffer, |
141 | unsigned long len) | 141 | unsigned long len) |
142 | { | 142 | { |
143 | /* A single 16-bit integer */ | 143 | /* A single 16-bit integer */ |
@@ -148,7 +148,7 @@ validate_uint16(struct efi_variable *var, int match, u8 *buffer, | |||
148 | } | 148 | } |
149 | 149 | ||
150 | static bool | 150 | static bool |
151 | validate_ascii_string(struct efi_variable *var, int match, u8 *buffer, | 151 | validate_ascii_string(efi_char16_t *var_name, int match, u8 *buffer, |
152 | unsigned long len) | 152 | unsigned long len) |
153 | { | 153 | { |
154 | int i; | 154 | int i; |
@@ -166,7 +166,7 @@ validate_ascii_string(struct efi_variable *var, int match, u8 *buffer, | |||
166 | 166 | ||
167 | struct variable_validate { | 167 | struct variable_validate { |
168 | char *name; | 168 | char *name; |
169 | bool (*validate)(struct efi_variable *var, int match, u8 *data, | 169 | bool (*validate)(efi_char16_t *var_name, int match, u8 *data, |
170 | unsigned long len); | 170 | unsigned long len); |
171 | }; | 171 | }; |
172 | 172 | ||
@@ -189,10 +189,10 @@ static const struct variable_validate variable_validate[] = { | |||
189 | }; | 189 | }; |
190 | 190 | ||
191 | bool | 191 | bool |
192 | efivar_validate(struct efi_variable *var, u8 *data, unsigned long len) | 192 | efivar_validate(efi_char16_t *var_name, u8 *data, unsigned long len) |
193 | { | 193 | { |
194 | int i; | 194 | int i; |
195 | u16 *unicode_name = var->VariableName; | 195 | u16 *unicode_name = var_name; |
196 | 196 | ||
197 | for (i = 0; variable_validate[i].validate != NULL; i++) { | 197 | for (i = 0; variable_validate[i].validate != NULL; i++) { |
198 | const char *name = variable_validate[i].name; | 198 | const char *name = variable_validate[i].name; |
@@ -208,7 +208,7 @@ efivar_validate(struct efi_variable *var, u8 *data, unsigned long len) | |||
208 | 208 | ||
209 | /* Wildcard in the matching name means we've matched */ | 209 | /* Wildcard in the matching name means we've matched */ |
210 | if (c == '*') | 210 | if (c == '*') |
211 | return variable_validate[i].validate(var, | 211 | return variable_validate[i].validate(var_name, |
212 | match, data, len); | 212 | match, data, len); |
213 | 213 | ||
214 | /* Case sensitive match */ | 214 | /* Case sensitive match */ |
@@ -217,7 +217,7 @@ efivar_validate(struct efi_variable *var, u8 *data, unsigned long len) | |||
217 | 217 | ||
218 | /* Reached the end of the string while matching */ | 218 | /* Reached the end of the string while matching */ |
219 | if (!c) | 219 | if (!c) |
220 | return variable_validate[i].validate(var, | 220 | return variable_validate[i].validate(var_name, |
221 | match, data, len); | 221 | match, data, len); |
222 | } | 222 | } |
223 | } | 223 | } |
@@ -805,7 +805,7 @@ int efivar_entry_set_get_size(struct efivar_entry *entry, u32 attributes, | |||
805 | 805 | ||
806 | *set = false; | 806 | *set = false; |
807 | 807 | ||
808 | if (efivar_validate(&entry->var, data, *size) == false) | 808 | if (efivar_validate(name, data, *size) == false) |
809 | return -EINVAL; | 809 | return -EINVAL; |
810 | 810 | ||
811 | /* | 811 | /* |