aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/boot/compressed
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/boot/compressed')
-rw-r--r--arch/x86/boot/compressed/eboot.c319
-rw-r--r--arch/x86/boot/compressed/eboot.h16
-rw-r--r--arch/x86/boot/compressed/head_32.S48
-rw-r--r--arch/x86/boot/compressed/head_64.S57
4 files changed, 339 insertions, 101 deletions
diff --git a/arch/x86/boot/compressed/eboot.c b/arch/x86/boot/compressed/eboot.c
index a7677babf946..42548168bdc3 100644
--- a/arch/x86/boot/compressed/eboot.c
+++ b/arch/x86/boot/compressed/eboot.c
@@ -19,10 +19,145 @@
19 19
20static efi_system_table_t *sys_table; 20static efi_system_table_t *sys_table;
21 21
22static struct efi_config *efi_early;
23
24#define BOOT_SERVICES(bits) \
25static void setup_boot_services##bits(struct efi_config *c) \
26{ \
27 efi_system_table_##bits##_t *table; \
28 efi_boot_services_##bits##_t *bt; \
29 \
30 table = (typeof(table))sys_table; \
31 \
32 c->text_output = table->con_out; \
33 \
34 bt = (typeof(bt))(unsigned long)(table->boottime); \
35 \
36 c->allocate_pool = bt->allocate_pool; \
37 c->allocate_pages = bt->allocate_pages; \
38 c->get_memory_map = bt->get_memory_map; \
39 c->free_pool = bt->free_pool; \
40 c->free_pages = bt->free_pages; \
41 c->locate_handle = bt->locate_handle; \
42 c->handle_protocol = bt->handle_protocol; \
43 c->exit_boot_services = bt->exit_boot_services; \
44}
45BOOT_SERVICES(32);
46BOOT_SERVICES(64);
22 47
23#include "../../../../drivers/firmware/efi/efi-stub-helper.c" 48static void efi_printk(efi_system_table_t *, char *);
49static void efi_char16_printk(efi_system_table_t *, efi_char16_t *);
50
51static efi_status_t
52efi_file_size(efi_system_table_t *sys_table, void *__fh,
53 efi_char16_t *filename_16, void **handle, u64 *file_sz)
54{
55 efi_file_handle_t *h, *fh = __fh;
56 efi_file_info_t *info;
57 efi_status_t status;
58 efi_guid_t info_guid = EFI_FILE_INFO_ID;
59 u32 info_sz;
60
61 status = efi_early->call((unsigned long)fh->open, fh, &h, filename_16,
62 EFI_FILE_MODE_READ, (u64)0);
63 if (status != EFI_SUCCESS) {
64 efi_printk(sys_table, "Failed to open file: ");
65 efi_char16_printk(sys_table, filename_16);
66 efi_printk(sys_table, "\n");
67 return status;
68 }
69
70 *handle = h;
71
72 info_sz = 0;
73 status = efi_early->call((unsigned long)h->get_info, h, &info_guid,
74 &info_sz, NULL);
75 if (status != EFI_BUFFER_TOO_SMALL) {
76 efi_printk(sys_table, "Failed to get file info size\n");
77 return status;
78 }
79
80grow:
81 status = efi_early->call(efi_early->allocate_pool, EFI_LOADER_DATA,
82 info_sz, (void **)&info);
83 if (status != EFI_SUCCESS) {
84 efi_printk(sys_table, "Failed to alloc mem for file info\n");
85 return status;
86 }
87
88 status = efi_early->call((unsigned long)h->get_info, h, &info_guid,
89 &info_sz, info);
90 if (status == EFI_BUFFER_TOO_SMALL) {
91 efi_early->call(efi_early->free_pool, info);
92 goto grow;
93 }
94
95 *file_sz = info->file_size;
96 efi_early->call(efi_early->free_pool, info);
97
98 if (status != EFI_SUCCESS)
99 efi_printk(sys_table, "Failed to get initrd info\n");
100
101 return status;
102}
103
104static inline efi_status_t
105efi_file_read(void *__fh, void *handle, unsigned long *size, void *addr)
106{
107 efi_file_handle_t *fh = __fh;
108 return efi_early->call((unsigned long)fh->read, handle, size, addr);
109}
110
111static inline efi_status_t efi_file_close(void *__fh, void *handle)
112{
113 efi_file_handle_t *fh = __fh;
24 114
115 return efi_early->call((unsigned long)fh->close, handle);
116}
117
118static inline efi_status_t
119efi_open_volume(efi_system_table_t *sys_table, void *__image, void **__fh)
120{
121 efi_file_io_interface_t *io;
122 efi_loaded_image_t *image = __image;
123 efi_file_handle_t *fh;
124 efi_guid_t fs_proto = EFI_FILE_SYSTEM_GUID;
125 efi_status_t status;
126 void *handle = (void *)(unsigned long)image->device_handle;
127 u32 func;
128
129 status = efi_early->call(efi_early->handle_protocol, handle,
130 &fs_proto, (void **)&io);
131 if (status != EFI_SUCCESS) {
132 efi_printk(sys_table, "Failed to handle fs_proto\n");
133 return status;
134 }
25 135
136 func = (unsigned long)io->open_volume;
137 status = efi_early->call(func, io, &fh);
138 if (status != EFI_SUCCESS)
139 efi_printk(sys_table, "Failed to open volume\n");
140
141 *__fh = fh;
142 return status;
143}
144
145static inline void
146efi_char16_printk(efi_system_table_t *table, efi_char16_t *str)
147{
148 struct efi_simple_text_output_protocol *out;
149 unsigned long output_string;
150 size_t offset;
151 unsigned long *func;
152
153 offset = offsetof(typeof(*out), output_string);
154 output_string = efi_early->text_output + offset;
155 func = (unsigned long *)output_string;
156
157 efi_early->call(*func, efi_early->text_output, str);
158}
159
160#include "../../../../drivers/firmware/efi/efi-stub-helper.c"
26 161
27static void find_bits(unsigned long mask, u8 *pos, u8 *size) 162static void find_bits(unsigned long mask, u8 *pos, u8 *size)
28{ 163{
@@ -51,7 +186,7 @@ static efi_status_t setup_efi_pci(struct boot_params *params)
51{ 186{
52 efi_pci_io_protocol *pci; 187 efi_pci_io_protocol *pci;
53 efi_status_t status; 188 efi_status_t status;
54 void **pci_handle; 189 void **pci_handle = NULL;
55 efi_guid_t pci_proto = EFI_PCI_IO_PROTOCOL_GUID; 190 efi_guid_t pci_proto = EFI_PCI_IO_PROTOCOL_GUID;
56 unsigned long nr_pci, size = 0; 191 unsigned long nr_pci, size = 0;
57 int i; 192 int i;
@@ -62,20 +197,21 @@ static efi_status_t setup_efi_pci(struct boot_params *params)
62 while (data && data->next) 197 while (data && data->next)
63 data = (struct setup_data *)(unsigned long)data->next; 198 data = (struct setup_data *)(unsigned long)data->next;
64 199
65 status = efi_call_phys5(sys_table->boottime->locate_handle, 200 status = efi_early->call(efi_early->locate_handle,
66 EFI_LOCATE_BY_PROTOCOL, &pci_proto, 201 EFI_LOCATE_BY_PROTOCOL,
67 NULL, &size, pci_handle); 202 &pci_proto, NULL, &size, pci_handle);
68 203
69 if (status == EFI_BUFFER_TOO_SMALL) { 204 if (status == EFI_BUFFER_TOO_SMALL) {
70 status = efi_call_phys3(sys_table->boottime->allocate_pool, 205 status = efi_early->call(efi_early->allocate_pool,
71 EFI_LOADER_DATA, size, &pci_handle); 206 EFI_LOADER_DATA,
207 size, (void **)&pci_handle);
72 208
73 if (status != EFI_SUCCESS) 209 if (status != EFI_SUCCESS)
74 return status; 210 return status;
75 211
76 status = efi_call_phys5(sys_table->boottime->locate_handle, 212 status = efi_early->call(efi_early->locate_handle,
77 EFI_LOCATE_BY_PROTOCOL, &pci_proto, 213 EFI_LOCATE_BY_PROTOCOL, &pci_proto,
78 NULL, &size, pci_handle); 214 NULL, &size, pci_handle);
79 } 215 }
80 216
81 if (status != EFI_SUCCESS) 217 if (status != EFI_SUCCESS)
@@ -87,8 +223,8 @@ static efi_status_t setup_efi_pci(struct boot_params *params)
87 uint64_t attributes; 223 uint64_t attributes;
88 struct pci_setup_rom *rom; 224 struct pci_setup_rom *rom;
89 225
90 status = efi_call_phys3(sys_table->boottime->handle_protocol, 226 status = efi_early->call(efi_early->handle_protocol, h,
91 h, &pci_proto, &pci); 227 &pci_proto, (void **)&pci);
92 228
93 if (status != EFI_SUCCESS) 229 if (status != EFI_SUCCESS)
94 continue; 230 continue;
@@ -97,13 +233,13 @@ static efi_status_t setup_efi_pci(struct boot_params *params)
97 continue; 233 continue;
98 234
99#ifdef CONFIG_X86_64 235#ifdef CONFIG_X86_64
100 status = efi_call_phys4(pci->attributes, pci, 236 status = efi_early->call((unsigned long)pci->attributes, pci,
101 EfiPciIoAttributeOperationGet, 0, 237 EfiPciIoAttributeOperationGet, 0,
102 &attributes); 238 &attributes);
103#else 239#else
104 status = efi_call_phys5(pci->attributes, pci, 240 status = efi_early->call((unsigned long)pci->attributes, pci,
105 EfiPciIoAttributeOperationGet, 0, 0, 241 EfiPciIoAttributeOperationGet, 0, 0,
106 &attributes); 242 &attributes);
107#endif 243#endif
108 if (status != EFI_SUCCESS) 244 if (status != EFI_SUCCESS)
109 continue; 245 continue;
@@ -113,8 +249,8 @@ static efi_status_t setup_efi_pci(struct boot_params *params)
113 249
114 size = pci->romsize + sizeof(*rom); 250 size = pci->romsize + sizeof(*rom);
115 251
116 status = efi_call_phys3(sys_table->boottime->allocate_pool, 252 status = efi_early->call(efi_early->allocate_pool,
117 EFI_LOADER_DATA, size, &rom); 253 EFI_LOADER_DATA, size, &rom);
118 254
119 if (status != EFI_SUCCESS) 255 if (status != EFI_SUCCESS)
120 continue; 256 continue;
@@ -124,23 +260,23 @@ static efi_status_t setup_efi_pci(struct boot_params *params)
124 rom->data.next = 0; 260 rom->data.next = 0;
125 rom->pcilen = pci->romsize; 261 rom->pcilen = pci->romsize;
126 262
127 status = efi_call_phys5(pci->pci.read, pci, 263 status = efi_early->call((unsigned long)pci->pci.read, pci,
128 EfiPciIoWidthUint16, PCI_VENDOR_ID, 264 EfiPciIoWidthUint16, PCI_VENDOR_ID,
129 1, &(rom->vendor)); 265 1, &(rom->vendor));
130 266
131 if (status != EFI_SUCCESS) 267 if (status != EFI_SUCCESS)
132 goto free_struct; 268 goto free_struct;
133 269
134 status = efi_call_phys5(pci->pci.read, pci, 270 status = efi_early->call((unsigned long)pci->pci.read, pci,
135 EfiPciIoWidthUint16, PCI_DEVICE_ID, 271 EfiPciIoWidthUint16, PCI_DEVICE_ID,
136 1, &(rom->devid)); 272 1, &(rom->devid));
137 273
138 if (status != EFI_SUCCESS) 274 if (status != EFI_SUCCESS)
139 goto free_struct; 275 goto free_struct;
140 276
141 status = efi_call_phys5(pci->get_location, pci, 277 status = efi_early->call((unsigned long)pci->get_location, pci,
142 &(rom->segment), &(rom->bus), 278 &(rom->segment), &(rom->bus),
143 &(rom->device), &(rom->function)); 279 &(rom->device), &(rom->function));
144 280
145 if (status != EFI_SUCCESS) 281 if (status != EFI_SUCCESS)
146 goto free_struct; 282 goto free_struct;
@@ -156,11 +292,11 @@ static efi_status_t setup_efi_pci(struct boot_params *params)
156 292
157 continue; 293 continue;
158 free_struct: 294 free_struct:
159 efi_call_phys1(sys_table->boottime->free_pool, rom); 295 efi_early->call(efi_early->free_pool, rom);
160 } 296 }
161 297
162free_handle: 298free_handle:
163 efi_call_phys1(sys_table->boottime->free_pool, pci_handle); 299 efi_early->call(efi_early->free_pool, pci_handle);
164 return status; 300 return status;
165} 301}
166 302
@@ -174,21 +310,21 @@ static efi_status_t setup_gop(struct screen_info *si, efi_guid_t *proto,
174 struct efi_pixel_bitmask pixel_info; 310 struct efi_pixel_bitmask pixel_info;
175 unsigned long nr_gops; 311 unsigned long nr_gops;
176 efi_status_t status; 312 efi_status_t status;
177 void **gop_handle; 313 void **gop_handle = NULL;
178 u16 width, height; 314 u16 width, height;
179 u32 fb_base, fb_size; 315 u32 fb_base, fb_size;
180 u32 pixels_per_scan_line; 316 u32 pixels_per_scan_line;
181 int pixel_format; 317 int pixel_format;
182 int i; 318 int i;
183 319
184 status = efi_call_phys3(sys_table->boottime->allocate_pool, 320 status = efi_early->call(efi_early->allocate_pool, EFI_LOADER_DATA,
185 EFI_LOADER_DATA, size, &gop_handle); 321 size, (void **)&gop_handle);
186 if (status != EFI_SUCCESS) 322 if (status != EFI_SUCCESS)
187 return status; 323 return status;
188 324
189 status = efi_call_phys5(sys_table->boottime->locate_handle, 325 status = efi_early->call(efi_early->locate_handle,
190 EFI_LOCATE_BY_PROTOCOL, proto, 326 EFI_LOCATE_BY_PROTOCOL,
191 NULL, &size, gop_handle); 327 proto, NULL, &size, gop_handle);
192 if (status != EFI_SUCCESS) 328 if (status != EFI_SUCCESS)
193 goto free_handle; 329 goto free_handle;
194 330
@@ -202,19 +338,18 @@ static efi_status_t setup_gop(struct screen_info *si, efi_guid_t *proto,
202 void *dummy; 338 void *dummy;
203 void *h = gop_handle[i]; 339 void *h = gop_handle[i];
204 340
205 status = efi_call_phys3(sys_table->boottime->handle_protocol, 341 status = efi_early->call(efi_early->handle_protocol, h,
206 h, proto, &gop); 342 proto, (void **)&gop);
207 if (status != EFI_SUCCESS) 343 if (status != EFI_SUCCESS)
208 continue; 344 continue;
209 345
210 status = efi_call_phys3(sys_table->boottime->handle_protocol, 346 status = efi_early->call(efi_early->handle_protocol, h,
211 h, &conout_proto, &dummy); 347 &conout_proto, &dummy);
212
213 if (status == EFI_SUCCESS) 348 if (status == EFI_SUCCESS)
214 conout_found = true; 349 conout_found = true;
215 350
216 status = efi_call_phys4(gop->query_mode, gop, 351 status = efi_early->call((unsigned long)gop->query_mode, gop,
217 gop->mode->mode, &size, &info); 352 gop->mode->mode, &size, &info);
218 if (status == EFI_SUCCESS && (!first_gop || conout_found)) { 353 if (status == EFI_SUCCESS && (!first_gop || conout_found)) {
219 /* 354 /*
220 * Systems that use the UEFI Console Splitter may 355 * Systems that use the UEFI Console Splitter may
@@ -303,7 +438,7 @@ static efi_status_t setup_gop(struct screen_info *si, efi_guid_t *proto,
303 si->capabilities |= VIDEO_CAPABILITY_SKIP_QUIRKS; 438 si->capabilities |= VIDEO_CAPABILITY_SKIP_QUIRKS;
304 439
305free_handle: 440free_handle:
306 efi_call_phys1(sys_table->boottime->free_pool, gop_handle); 441 efi_early->call(efi_early->free_pool, gop_handle);
307 return status; 442 return status;
308} 443}
309 444
@@ -320,14 +455,14 @@ static efi_status_t setup_uga(struct screen_info *si, efi_guid_t *uga_proto,
320 void **uga_handle = NULL; 455 void **uga_handle = NULL;
321 int i; 456 int i;
322 457
323 status = efi_call_phys3(sys_table->boottime->allocate_pool, 458 status = efi_early->call(efi_early->allocate_pool, EFI_LOADER_DATA,
324 EFI_LOADER_DATA, size, &uga_handle); 459 size, (void **)&uga_handle);
325 if (status != EFI_SUCCESS) 460 if (status != EFI_SUCCESS)
326 return status; 461 return status;
327 462
328 status = efi_call_phys5(sys_table->boottime->locate_handle, 463 status = efi_early->call(efi_early->locate_handle,
329 EFI_LOCATE_BY_PROTOCOL, uga_proto, 464 EFI_LOCATE_BY_PROTOCOL,
330 NULL, &size, uga_handle); 465 uga_proto, NULL, &size, uga_handle);
331 if (status != EFI_SUCCESS) 466 if (status != EFI_SUCCESS)
332 goto free_handle; 467 goto free_handle;
333 468
@@ -340,16 +475,16 @@ static efi_status_t setup_uga(struct screen_info *si, efi_guid_t *uga_proto,
340 u32 w, h, depth, refresh; 475 u32 w, h, depth, refresh;
341 void *pciio; 476 void *pciio;
342 477
343 status = efi_call_phys3(sys_table->boottime->handle_protocol, 478 status = efi_early->call(efi_early->handle_protocol, handle,
344 handle, uga_proto, &uga); 479 uga_proto, (void **)&uga);
345 if (status != EFI_SUCCESS) 480 if (status != EFI_SUCCESS)
346 continue; 481 continue;
347 482
348 efi_call_phys3(sys_table->boottime->handle_protocol, 483 efi_early->call(efi_early->handle_protocol, handle,
349 handle, &pciio_proto, &pciio); 484 &pciio_proto, &pciio);
350 485
351 status = efi_call_phys5(uga->get_mode, uga, &w, &h, 486 status = efi_early->call((unsigned long)uga->get_mode, uga,
352 &depth, &refresh); 487 &w, &h, &depth, &refresh);
353 if (status == EFI_SUCCESS && (!first_uga || pciio)) { 488 if (status == EFI_SUCCESS && (!first_uga || pciio)) {
354 width = w; 489 width = w;
355 height = h; 490 height = h;
@@ -386,7 +521,7 @@ static efi_status_t setup_uga(struct screen_info *si, efi_guid_t *uga_proto,
386 521
387 522
388free_handle: 523free_handle:
389 efi_call_phys1(sys_table->boottime->free_pool, uga_handle); 524 efi_early->call(efi_early->free_pool, uga_handle);
390 return status; 525 return status;
391} 526}
392 527
@@ -404,29 +539,28 @@ void setup_graphics(struct boot_params *boot_params)
404 memset(si, 0, sizeof(*si)); 539 memset(si, 0, sizeof(*si));
405 540
406 size = 0; 541 size = 0;
407 status = efi_call_phys5(sys_table->boottime->locate_handle, 542 status = efi_early->call(efi_early->locate_handle,
408 EFI_LOCATE_BY_PROTOCOL, &graphics_proto, 543 EFI_LOCATE_BY_PROTOCOL,
409 NULL, &size, gop_handle); 544 &graphics_proto, NULL, &size, gop_handle);
410 if (status == EFI_BUFFER_TOO_SMALL) 545 if (status == EFI_BUFFER_TOO_SMALL)
411 status = setup_gop(si, &graphics_proto, size); 546 status = setup_gop(si, &graphics_proto, size);
412 547
413 if (status != EFI_SUCCESS) { 548 if (status != EFI_SUCCESS) {
414 size = 0; 549 size = 0;
415 status = efi_call_phys5(sys_table->boottime->locate_handle, 550 status = efi_early->call(efi_early->locate_handle,
416 EFI_LOCATE_BY_PROTOCOL, &uga_proto, 551 EFI_LOCATE_BY_PROTOCOL,
417 NULL, &size, uga_handle); 552 &uga_proto, NULL, &size, uga_handle);
418 if (status == EFI_BUFFER_TOO_SMALL) 553 if (status == EFI_BUFFER_TOO_SMALL)
419 setup_uga(si, &uga_proto, size); 554 setup_uga(si, &uga_proto, size);
420 } 555 }
421} 556}
422 557
423
424/* 558/*
425 * Because the x86 boot code expects to be passed a boot_params we 559 * Because the x86 boot code expects to be passed a boot_params we
426 * need to create one ourselves (usually the bootloader would create 560 * need to create one ourselves (usually the bootloader would create
427 * one for us). 561 * one for us).
428 */ 562 */
429struct boot_params *make_boot_params(void *handle, efi_system_table_t *_table) 563struct boot_params *make_boot_params(struct efi_config *c)
430{ 564{
431 struct boot_params *boot_params; 565 struct boot_params *boot_params;
432 struct sys_desc_table *sdt; 566 struct sys_desc_table *sdt;
@@ -434,7 +568,7 @@ struct boot_params *make_boot_params(void *handle, efi_system_table_t *_table)
434 struct setup_header *hdr; 568 struct setup_header *hdr;
435 struct efi_info *efi; 569 struct efi_info *efi;
436 efi_loaded_image_t *image; 570 efi_loaded_image_t *image;
437 void *options; 571 void *options, *handle;
438 efi_guid_t proto = LOADED_IMAGE_PROTOCOL_GUID; 572 efi_guid_t proto = LOADED_IMAGE_PROTOCOL_GUID;
439 int options_size = 0; 573 int options_size = 0;
440 efi_status_t status; 574 efi_status_t status;
@@ -445,14 +579,21 @@ struct boot_params *make_boot_params(void *handle, efi_system_table_t *_table)
445 unsigned long ramdisk_addr; 579 unsigned long ramdisk_addr;
446 unsigned long ramdisk_size; 580 unsigned long ramdisk_size;
447 581
448 sys_table = _table; 582 efi_early = c;
583 sys_table = (efi_system_table_t *)(unsigned long)efi_early->table;
584 handle = (void *)(unsigned long)efi_early->image_handle;
449 585
450 /* Check if we were booted by the EFI firmware */ 586 /* Check if we were booted by the EFI firmware */
451 if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE) 587 if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE)
452 return NULL; 588 return NULL;
453 589
454 status = efi_call_phys3(sys_table->boottime->handle_protocol, 590 if (efi_early->is64)
455 handle, &proto, (void *)&image); 591 setup_boot_services64(efi_early);
592 else
593 setup_boot_services32(efi_early);
594
595 status = efi_early->call(efi_early->handle_protocol, handle,
596 &proto, (void *)&image);
456 if (status != EFI_SUCCESS) { 597 if (status != EFI_SUCCESS) {
457 efi_printk(sys_table, "Failed to get handle for LOADED_IMAGE_PROTOCOL\n"); 598 efi_printk(sys_table, "Failed to get handle for LOADED_IMAGE_PROTOCOL\n");
458 return NULL; 599 return NULL;
@@ -641,14 +782,13 @@ static efi_status_t alloc_e820ext(u32 nr_desc, struct setup_data **e820ext,
641 sizeof(struct e820entry) * nr_desc; 782 sizeof(struct e820entry) * nr_desc;
642 783
643 if (*e820ext) { 784 if (*e820ext) {
644 efi_call_phys1(sys_table->boottime->free_pool, *e820ext); 785 efi_early->call(efi_early->free_pool, *e820ext);
645 *e820ext = NULL; 786 *e820ext = NULL;
646 *e820ext_size = 0; 787 *e820ext_size = 0;
647 } 788 }
648 789
649 status = efi_call_phys3(sys_table->boottime->allocate_pool, 790 status = efi_early->call(efi_early->allocate_pool, EFI_LOADER_DATA,
650 EFI_LOADER_DATA, size, e820ext); 791 size, (void **)e820ext);
651
652 if (status == EFI_SUCCESS) 792 if (status == EFI_SUCCESS)
653 *e820ext_size = size; 793 *e820ext_size = size;
654 794
@@ -691,7 +831,7 @@ get_map:
691 if (status != EFI_SUCCESS) 831 if (status != EFI_SUCCESS)
692 goto free_mem_map; 832 goto free_mem_map;
693 833
694 efi_call_phys1(sys_table->boottime->free_pool, mem_map); 834 efi_early->call(efi_early->free_pool, mem_map);
695 goto get_map; /* Allocated memory, get map again */ 835 goto get_map; /* Allocated memory, get map again */
696 } 836 }
697 837
@@ -708,8 +848,7 @@ get_map:
708#endif 848#endif
709 849
710 /* Might as well exit boot services now */ 850 /* Might as well exit boot services now */
711 status = efi_call_phys2(sys_table->boottime->exit_boot_services, 851 status = efi_early->call(efi_early->exit_boot_services, handle, key);
712 handle, key);
713 if (status != EFI_SUCCESS) { 852 if (status != EFI_SUCCESS) {
714 /* 853 /*
715 * ExitBootServices() will fail if any of the event 854 * ExitBootServices() will fail if any of the event
@@ -722,7 +861,7 @@ get_map:
722 goto free_mem_map; 861 goto free_mem_map;
723 862
724 called_exit = true; 863 called_exit = true;
725 efi_call_phys1(sys_table->boottime->free_pool, mem_map); 864 efi_early->call(efi_early->free_pool, mem_map);
726 goto get_map; 865 goto get_map;
727 } 866 }
728 867
@@ -736,23 +875,31 @@ get_map:
736 return EFI_SUCCESS; 875 return EFI_SUCCESS;
737 876
738free_mem_map: 877free_mem_map:
739 efi_call_phys1(sys_table->boottime->free_pool, mem_map); 878 efi_early->call(efi_early->free_pool, mem_map);
740 return status; 879 return status;
741} 880}
742 881
743
744/* 882/*
745 * On success we return a pointer to a boot_params structure, and NULL 883 * On success we return a pointer to a boot_params structure, and NULL
746 * on failure. 884 * on failure.
747 */ 885 */
748struct boot_params *efi_main(void *handle, efi_system_table_t *_table, 886struct boot_params *efi_main(struct efi_config *c,
749 struct boot_params *boot_params) 887 struct boot_params *boot_params)
750{ 888{
751 struct desc_ptr *gdt; 889 struct desc_ptr *gdt = NULL;
752 efi_loaded_image_t *image; 890 efi_loaded_image_t *image;
753 struct setup_header *hdr = &boot_params->hdr; 891 struct setup_header *hdr = &boot_params->hdr;
754 efi_status_t status; 892 efi_status_t status;
755 struct desc_struct *desc; 893 struct desc_struct *desc;
894 void *handle;
895 efi_system_table_t *_table;
896 bool is64;
897
898 efi_early = c;
899
900 _table = (efi_system_table_t *)(unsigned long)efi_early->table;
901 handle = (void *)(unsigned long)efi_early->image_handle;
902 is64 = efi_early->is64;
756 903
757 sys_table = _table; 904 sys_table = _table;
758 905
@@ -760,13 +907,17 @@ struct boot_params *efi_main(void *handle, efi_system_table_t *_table,
760 if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE) 907 if (sys_table->hdr.signature != EFI_SYSTEM_TABLE_SIGNATURE)
761 goto fail; 908 goto fail;
762 909
910 if (is64)
911 setup_boot_services64(efi_early);
912 else
913 setup_boot_services32(efi_early);
914
763 setup_graphics(boot_params); 915 setup_graphics(boot_params);
764 916
765 setup_efi_pci(boot_params); 917 setup_efi_pci(boot_params);
766 918
767 status = efi_call_phys3(sys_table->boottime->allocate_pool, 919 status = efi_early->call(efi_early->allocate_pool, EFI_LOADER_DATA,
768 EFI_LOADER_DATA, sizeof(*gdt), 920 sizeof(*gdt), (void **)&gdt);
769 (void **)&gdt);
770 if (status != EFI_SUCCESS) { 921 if (status != EFI_SUCCESS) {
771 efi_printk(sys_table, "Failed to alloc mem for gdt structure\n"); 922 efi_printk(sys_table, "Failed to alloc mem for gdt structure\n");
772 goto fail; 923 goto fail;
diff --git a/arch/x86/boot/compressed/eboot.h b/arch/x86/boot/compressed/eboot.h
index d487e727f1ec..c88c31ecad12 100644
--- a/arch/x86/boot/compressed/eboot.h
+++ b/arch/x86/boot/compressed/eboot.h
@@ -103,4 +103,20 @@ struct efi_uga_draw_protocol {
103 void *blt; 103 void *blt;
104}; 104};
105 105
106struct efi_config {
107 u64 image_handle;
108 u64 table;
109 u64 allocate_pool;
110 u64 allocate_pages;
111 u64 get_memory_map;
112 u64 free_pool;
113 u64 free_pages;
114 u64 locate_handle;
115 u64 handle_protocol;
116 u64 exit_boot_services;
117 u64 text_output;
118 efi_status_t (*call)(unsigned long, ...);
119 bool is64;
120} __packed;
121
106#endif /* BOOT_COMPRESSED_EBOOT_H */ 122#endif /* BOOT_COMPRESSED_EBOOT_H */
diff --git a/arch/x86/boot/compressed/head_32.S b/arch/x86/boot/compressed/head_32.S
index 9116aac232c7..eed23c087d6c 100644
--- a/arch/x86/boot/compressed/head_32.S
+++ b/arch/x86/boot/compressed/head_32.S
@@ -42,26 +42,53 @@ ENTRY(startup_32)
42ENTRY(efi_pe_entry) 42ENTRY(efi_pe_entry)
43 add $0x4, %esp 43 add $0x4, %esp
44 44
45 call 1f
461: popl %esi
47 subl $1b, %esi
48
49 popl %ecx
50 movl %ecx, efi32_config(%esi) /* Handle */
51 popl %ecx
52 movl %ecx, efi32_config+8(%esi) /* EFI System table pointer */
53
54 /* Relocate efi_config->call() */
55 leal efi32_config(%esi), %eax
56 add %esi, 88(%eax)
57 pushl %eax
58
45 call make_boot_params 59 call make_boot_params
46 cmpl $0, %eax 60 cmpl $0, %eax
47 je 1f 61 je fail
48 movl 0x4(%esp), %esi 62 popl %ecx
49 movl (%esp), %ecx
50 pushl %eax 63 pushl %eax
51 pushl %esi
52 pushl %ecx 64 pushl %ecx
53 sub $0x4, %esp 65 jmp 2f /* Skip efi_config initialization */
54 66
55ENTRY(efi_stub_entry) 67ENTRY(efi_stub_entry)
56 add $0x4, %esp 68 add $0x4, %esp
69 popl %ecx
70 popl %edx
71
72 call 1f
731: popl %esi
74 subl $1b, %esi
75
76 movl %ecx, efi32_config(%esi) /* Handle */
77 movl %edx, efi32_config+8(%esi) /* EFI System table pointer */
78
79 /* Relocate efi_config->call() */
80 leal efi32_config(%esi), %eax
81 add %esi, 88(%eax)
82 pushl %eax
832:
57 call efi_main 84 call efi_main
58 cmpl $0, %eax 85 cmpl $0, %eax
59 movl %eax, %esi 86 movl %eax, %esi
60 jne 2f 87 jne 2f
611: 88fail:
62 /* EFI init failed, so hang. */ 89 /* EFI init failed, so hang. */
63 hlt 90 hlt
64 jmp 1b 91 jmp fail
652: 922:
66 call 3f 93 call 3f
673: 943:
@@ -202,6 +229,13 @@ relocated:
202 xorl %ebx, %ebx 229 xorl %ebx, %ebx
203 jmp *%eax 230 jmp *%eax
204 231
232 .data
233efi32_config:
234 .fill 11,8,0
235 .long efi_call_phys
236 .long 0
237 .byte 0
238
205/* 239/*
206 * Stack and heap for uncompression 240 * Stack and heap for uncompression
207 */ 241 */
diff --git a/arch/x86/boot/compressed/head_64.S b/arch/x86/boot/compressed/head_64.S
index c5c1ae0997e7..1bc206fa4bd0 100644
--- a/arch/x86/boot/compressed/head_64.S
+++ b/arch/x86/boot/compressed/head_64.S
@@ -209,26 +209,55 @@ ENTRY(startup_64)
209 jmp preferred_addr 209 jmp preferred_addr
210 210
211ENTRY(efi_pe_entry) 211ENTRY(efi_pe_entry)
212 mov %rcx, %rdi 212 movq %rcx, efi64_config(%rip) /* Handle */
213 mov %rdx, %rsi 213 movq %rdx, efi64_config+8(%rip) /* EFI System table pointer */
214 pushq %rdi 214
215 pushq %rsi 215 leaq efi64_config(%rip), %rax
216 movq %rax, efi_config(%rip)
217
218 call 1f
2191: popq %rbp
220 subq $1b, %rbp
221
222 /*
223 * Relocate efi_config->call().
224 */
225 addq %rbp, efi64_config+88(%rip)
226
227 movq %rax, %rdi
216 call make_boot_params 228 call make_boot_params
217 cmpq $0,%rax 229 cmpq $0,%rax
218 je 1f 230 je fail
219 mov %rax, %rdx 231 mov %rax, %rsi
220 popq %rsi 232 jmp 2f /* Skip the relocation */
221 popq %rdi
222 233
223ENTRY(efi_stub_entry) 234ENTRY(efi_stub_entry)
235 movq %rdi, efi64_config(%rip) /* Handle */
236 movq %rsi, efi64_config+8(%rip) /* EFI System table pointer */
237
238 leaq efi64_config(%rip), %rax
239 movq %rax, efi_config(%rip)
240
241 call 1f
2421: popq %rbp
243 subq $1b, %rbp
244
245 /*
246 * Relocate efi_config->call().
247 */
248 movq efi_config(%rip), %rax
249 addq %rbp, 88(%rax)
250 movq %rdx, %rsi
2512:
252 movq efi_config(%rip), %rdi
224 call efi_main 253 call efi_main
225 movq %rax,%rsi 254 movq %rax,%rsi
226 cmpq $0,%rax 255 cmpq $0,%rax
227 jne 2f 256 jne 2f
2281: 257fail:
229 /* EFI init failed, so hang. */ 258 /* EFI init failed, so hang. */
230 hlt 259 hlt
231 jmp 1b 260 jmp fail
2322: 2612:
233 call 3f 262 call 3f
2343: 2633:
@@ -372,6 +401,14 @@ gdt:
372 .quad 0x0000000000000000 /* TS continued */ 401 .quad 0x0000000000000000 /* TS continued */
373gdt_end: 402gdt_end:
374 403
404efi_config:
405 .quad 0
406
407 .global efi64_config
408efi64_config:
409 .fill 11,8,0
410 .quad efi_call6
411 .byte 1
375/* 412/*
376 * Stack and heap for uncompression 413 * Stack and heap for uncompression
377 */ 414 */