Skip to content

Commit 964124a

Browse files
nivedita76Ingo Molnar
authored andcommitted
efi/x86: Remove extra headroom for setup block
The following commit: 223e3ee ("efi/x86: add headroom to decompressor BSS to account for setup block") added headroom to the PE image to account for the setup block, which wasn't used for the decompression buffer. Now that the decompression buffer is located at the start of the image, and includes the setup block, this is no longer required. Add a check to make sure that the head section of the compressed kernel won't overwrite itself while relocating. This is only for future-proofing as with current limits on the setup and the actual size of the head section, this can never happen. Signed-off-by: Arvind Sankar <[email protected]> Signed-off-by: Ard Biesheuvel <[email protected]> Signed-off-by: Ingo Molnar <[email protected]> Link: https://lore.kernel.org/r/[email protected] Link: https://lore.kernel.org/r/[email protected]
1 parent 2672519 commit 964124a

File tree

1 file changed

+26
-2
lines changed

1 file changed

+26
-2
lines changed

arch/x86/boot/tools/build.c

Lines changed: 26 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,8 @@ unsigned long efi_pe_entry;
6565
unsigned long efi32_pe_entry;
6666
unsigned long kernel_info;
6767
unsigned long startup_64;
68+
unsigned long _ehead;
69+
unsigned long _end;
6870

6971
/*----------------------------------------------------------------------*/
7072

@@ -232,7 +234,7 @@ static void update_pecoff_text(unsigned int text_start, unsigned int file_sz,
232234
{
233235
unsigned int pe_header;
234236
unsigned int text_sz = file_sz - text_start;
235-
unsigned int bss_sz = init_sz + text_start - file_sz;
237+
unsigned int bss_sz = init_sz - file_sz;
236238

237239
pe_header = get_unaligned_le32(&buf[0x3c]);
238240

@@ -259,7 +261,7 @@ static void update_pecoff_text(unsigned int text_start, unsigned int file_sz,
259261
put_unaligned_le32(file_sz - 512 + bss_sz, &buf[pe_header + 0x1c]);
260262

261263
/* Size of image */
262-
put_unaligned_le32(init_sz + text_start, &buf[pe_header + 0x50]);
264+
put_unaligned_le32(init_sz, &buf[pe_header + 0x50]);
263265

264266
/*
265267
* Address of entry point for PE/COFF executable
@@ -360,6 +362,8 @@ static void parse_zoffset(char *fname)
360362
PARSE_ZOFS(p, efi32_pe_entry);
361363
PARSE_ZOFS(p, kernel_info);
362364
PARSE_ZOFS(p, startup_64);
365+
PARSE_ZOFS(p, _ehead);
366+
PARSE_ZOFS(p, _end);
363367

364368
p = strchr(p, '\n');
365369
while (p && (*p == '\r' || *p == '\n'))
@@ -444,6 +448,26 @@ int main(int argc, char ** argv)
444448
put_unaligned_le32(sys_size, &buf[0x1f4]);
445449

446450
init_sz = get_unaligned_le32(&buf[0x260]);
451+
#ifdef CONFIG_EFI_STUB
452+
/*
453+
* The decompression buffer will start at ImageBase. When relocating
454+
* the compressed kernel to its end, we must ensure that the head
455+
* section does not get overwritten. The head section occupies
456+
* [i, i + _ehead), and the destination is [init_sz - _end, init_sz).
457+
*
458+
* At present these should never overlap, because 'i' is at most 32k
459+
* because of SETUP_SECT_MAX, '_ehead' is less than 1k, and the
460+
* calculation of INIT_SIZE in boot/header.S ensures that
461+
* 'init_sz - _end' is at least 64k.
462+
*
463+
* For future-proofing, increase init_sz if necessary.
464+
*/
465+
466+
if (init_sz - _end < i + _ehead) {
467+
init_sz = (i + _ehead + _end + 4095) & ~4095;
468+
put_unaligned_le32(init_sz, &buf[0x260]);
469+
}
470+
#endif
447471
update_pecoff_text(setup_sectors * 512, i + (sys_size * 16), init_sz);
448472

449473
efi_stub_entry_update();

0 commit comments

Comments
 (0)