Commit 1642285e authored by Alexander Gordeev's avatar Alexander Gordeev Committed by Vasily Gorbik

s390/boot: Fix KASLR base offset off by __START_KERNEL bytes

Symbol offsets to the KASLR base do not match symbol address in
the vmlinux image. That is the result of setting the KASLR base
to the beginning of .text section as result of an optimization.

Revert that optimization and allocate virtual memory for the
whole kernel image including __START_KERNEL bytes as per the
linker script. That allows keeping the semantics of the KASLR
base offset in sync with other architectures.

Rename __START_KERNEL to TEXT_OFFSET, since it represents the
offset of the .text section within the kernel image, rather than
a virtual address.

Still skip mapping TEXT_OFFSET bytes to save memory on pgtables
and provoke exceptions in case an attempt to access this area is
made, as no kernel symbol may reside there.

In case CONFIG_KASAN is enabled the location counter might exceed
the value of TEXT_OFFSET, while the decompressor linker script
forcefully resets it to TEXT_OFFSET, which leads to a sections
overlap link failure. Use MAX() expression to avoid that.
Reported-by: default avatarOmar Sandoval <osandov@osandov.com>
Closes: https://lore.kernel.org/linux-s390/ZnS8dycxhtXBZVky@telecaster.dhcp.thefacebook.com/
Fixes: 56b1069c ("s390/boot: Rework deployment of the kernel image")
Signed-off-by: default avatarAlexander Gordeev <agordeev@linux.ibm.com>
Acked-by: default avatarVasily Gorbik <gor@linux.ibm.com>
Signed-off-by: default avatarVasily Gorbik <gor@linux.ibm.com>
parent d7fd2941
...@@ -162,7 +162,7 @@ static void kaslr_adjust_relocs(unsigned long min_addr, unsigned long max_addr, ...@@ -162,7 +162,7 @@ static void kaslr_adjust_relocs(unsigned long min_addr, unsigned long max_addr,
loc = (long)*reloc + phys_offset; loc = (long)*reloc + phys_offset;
if (loc < min_addr || loc > max_addr) if (loc < min_addr || loc > max_addr)
error("64-bit relocation outside of kernel!\n"); error("64-bit relocation outside of kernel!\n");
*(u64 *)loc += offset - __START_KERNEL; *(u64 *)loc += offset;
} }
} }
...@@ -177,7 +177,7 @@ static void kaslr_adjust_got(unsigned long offset) ...@@ -177,7 +177,7 @@ static void kaslr_adjust_got(unsigned long offset)
*/ */
for (entry = (u64 *)vmlinux.got_start; entry < (u64 *)vmlinux.got_end; entry++) { for (entry = (u64 *)vmlinux.got_start; entry < (u64 *)vmlinux.got_end; entry++) {
if (*entry) if (*entry)
*entry += offset - __START_KERNEL; *entry += offset;
} }
} }
...@@ -252,7 +252,7 @@ static unsigned long setup_kernel_memory_layout(unsigned long kernel_size) ...@@ -252,7 +252,7 @@ static unsigned long setup_kernel_memory_layout(unsigned long kernel_size)
vmemmap_size = SECTION_ALIGN_UP(pages) * sizeof(struct page); vmemmap_size = SECTION_ALIGN_UP(pages) * sizeof(struct page);
/* choose kernel address space layout: 4 or 3 levels. */ /* choose kernel address space layout: 4 or 3 levels. */
BUILD_BUG_ON(!IS_ALIGNED(__START_KERNEL, THREAD_SIZE)); BUILD_BUG_ON(!IS_ALIGNED(TEXT_OFFSET, THREAD_SIZE));
BUILD_BUG_ON(!IS_ALIGNED(__NO_KASLR_START_KERNEL, THREAD_SIZE)); BUILD_BUG_ON(!IS_ALIGNED(__NO_KASLR_START_KERNEL, THREAD_SIZE));
BUILD_BUG_ON(__NO_KASLR_END_KERNEL > _REGION1_SIZE); BUILD_BUG_ON(__NO_KASLR_END_KERNEL > _REGION1_SIZE);
vsize = get_vmem_size(ident_map_size, vmemmap_size, vmalloc_size, _REGION3_SIZE); vsize = get_vmem_size(ident_map_size, vmemmap_size, vmalloc_size, _REGION3_SIZE);
...@@ -389,31 +389,25 @@ static void kaslr_adjust_vmlinux_info(long offset) ...@@ -389,31 +389,25 @@ static void kaslr_adjust_vmlinux_info(long offset)
#endif #endif
} }
static void fixup_vmlinux_info(void)
{
vmlinux.entry -= __START_KERNEL;
kaslr_adjust_vmlinux_info(-__START_KERNEL);
}
void startup_kernel(void) void startup_kernel(void)
{ {
unsigned long kernel_size = vmlinux.image_size + vmlinux.bss_size; unsigned long vmlinux_size = vmlinux.image_size + vmlinux.bss_size;
unsigned long nokaslr_offset_phys, kaslr_large_page_offset; unsigned long nokaslr_text_lma, text_lma = 0, amode31_lma = 0;
unsigned long amode31_lma = 0; unsigned long kernel_size = TEXT_OFFSET + vmlinux_size;
unsigned long kaslr_large_page_offset;
unsigned long max_physmem_end; unsigned long max_physmem_end;
unsigned long asce_limit; unsigned long asce_limit;
unsigned long safe_addr; unsigned long safe_addr;
psw_t psw; psw_t psw;
fixup_vmlinux_info();
setup_lpp(); setup_lpp();
/* /*
* Non-randomized kernel physical start address must be _SEGMENT_SIZE * Non-randomized kernel physical start address must be _SEGMENT_SIZE
* aligned (see blow). * aligned (see blow).
*/ */
nokaslr_offset_phys = ALIGN(mem_safe_offset(), _SEGMENT_SIZE); nokaslr_text_lma = ALIGN(mem_safe_offset(), _SEGMENT_SIZE);
safe_addr = PAGE_ALIGN(nokaslr_offset_phys + kernel_size); safe_addr = PAGE_ALIGN(nokaslr_text_lma + vmlinux_size);
/* /*
* Reserve decompressor memory together with decompression heap, * Reserve decompressor memory together with decompression heap,
...@@ -457,16 +451,27 @@ void startup_kernel(void) ...@@ -457,16 +451,27 @@ void startup_kernel(void)
*/ */
kaslr_large_page_offset = __kaslr_offset & ~_SEGMENT_MASK; kaslr_large_page_offset = __kaslr_offset & ~_SEGMENT_MASK;
if (kaslr_enabled()) { if (kaslr_enabled()) {
unsigned long size = kernel_size + kaslr_large_page_offset; unsigned long size = vmlinux_size + kaslr_large_page_offset;
__kaslr_offset_phys = randomize_within_range(size, _SEGMENT_SIZE, 0, ident_map_size); text_lma = randomize_within_range(size, _SEGMENT_SIZE, TEXT_OFFSET, ident_map_size);
} }
if (!__kaslr_offset_phys) if (!text_lma)
__kaslr_offset_phys = nokaslr_offset_phys; text_lma = nokaslr_text_lma;
__kaslr_offset_phys |= kaslr_large_page_offset; text_lma |= kaslr_large_page_offset;
/*
* [__kaslr_offset_phys..__kaslr_offset_phys + TEXT_OFFSET] region is
* never accessed via the kernel image mapping as per the linker script:
*
* . = TEXT_OFFSET;
*
* Therefore, this region could be used for something else and does
* not need to be reserved. See how it is skipped in setup_vmem().
*/
__kaslr_offset_phys = text_lma - TEXT_OFFSET;
kaslr_adjust_vmlinux_info(__kaslr_offset_phys); kaslr_adjust_vmlinux_info(__kaslr_offset_phys);
physmem_reserve(RR_VMLINUX, __kaslr_offset_phys, kernel_size); physmem_reserve(RR_VMLINUX, text_lma, vmlinux_size);
deploy_kernel((void *)__kaslr_offset_phys); deploy_kernel((void *)text_lma);
/* vmlinux decompression is done, shrink reserved low memory */ /* vmlinux decompression is done, shrink reserved low memory */
physmem_reserve(RR_DECOMPRESSOR, 0, (unsigned long)_decompressor_end); physmem_reserve(RR_DECOMPRESSOR, 0, (unsigned long)_decompressor_end);
...@@ -489,7 +494,7 @@ void startup_kernel(void) ...@@ -489,7 +494,7 @@ void startup_kernel(void)
amode31_lma = randomize_within_range(vmlinux.amode31_size, PAGE_SIZE, amode31_min, SZ_2G); amode31_lma = randomize_within_range(vmlinux.amode31_size, PAGE_SIZE, amode31_min, SZ_2G);
} }
if (!amode31_lma) if (!amode31_lma)
amode31_lma = __kaslr_offset_phys - vmlinux.amode31_size; amode31_lma = text_lma - vmlinux.amode31_size;
physmem_reserve(RR_AMODE31, amode31_lma, vmlinux.amode31_size); physmem_reserve(RR_AMODE31, amode31_lma, vmlinux.amode31_size);
/* /*
...@@ -505,8 +510,8 @@ void startup_kernel(void) ...@@ -505,8 +510,8 @@ void startup_kernel(void)
* - copy_bootdata() must follow setup_vmem() to propagate changes * - copy_bootdata() must follow setup_vmem() to propagate changes
* to bootdata made by setup_vmem() * to bootdata made by setup_vmem()
*/ */
clear_bss_section(__kaslr_offset_phys); clear_bss_section(text_lma);
kaslr_adjust_relocs(__kaslr_offset_phys, __kaslr_offset_phys + vmlinux.image_size, kaslr_adjust_relocs(text_lma, text_lma + vmlinux.image_size,
__kaslr_offset, __kaslr_offset_phys); __kaslr_offset, __kaslr_offset_phys);
kaslr_adjust_got(__kaslr_offset); kaslr_adjust_got(__kaslr_offset);
setup_vmem(__kaslr_offset, __kaslr_offset + kernel_size, asce_limit); setup_vmem(__kaslr_offset, __kaslr_offset + kernel_size, asce_limit);
......
...@@ -90,7 +90,7 @@ static void kasan_populate_shadow(unsigned long kernel_start, unsigned long kern ...@@ -90,7 +90,7 @@ static void kasan_populate_shadow(unsigned long kernel_start, unsigned long kern
} }
memgap_start = end; memgap_start = end;
} }
kasan_populate(kernel_start, kernel_end, POPULATE_KASAN_MAP_SHADOW); kasan_populate(kernel_start + TEXT_OFFSET, kernel_end, POPULATE_KASAN_MAP_SHADOW);
kasan_populate(0, (unsigned long)__identity_va(0), POPULATE_KASAN_ZERO_SHADOW); kasan_populate(0, (unsigned long)__identity_va(0), POPULATE_KASAN_ZERO_SHADOW);
kasan_populate(AMODE31_START, AMODE31_END, POPULATE_KASAN_ZERO_SHADOW); kasan_populate(AMODE31_START, AMODE31_END, POPULATE_KASAN_ZERO_SHADOW);
if (IS_ENABLED(CONFIG_KASAN_VMALLOC)) { if (IS_ENABLED(CONFIG_KASAN_VMALLOC)) {
...@@ -475,7 +475,17 @@ void setup_vmem(unsigned long kernel_start, unsigned long kernel_end, unsigned l ...@@ -475,7 +475,17 @@ void setup_vmem(unsigned long kernel_start, unsigned long kernel_end, unsigned l
(unsigned long)__identity_va(end), (unsigned long)__identity_va(end),
POPULATE_IDENTITY); POPULATE_IDENTITY);
} }
pgtable_populate(kernel_start, kernel_end, POPULATE_KERNEL);
/*
* [kernel_start..kernel_start + TEXT_OFFSET] region is never
* accessed as per the linker script:
*
* . = TEXT_OFFSET;
*
* Therefore, skip mapping TEXT_OFFSET bytes to prevent access to
* [__kaslr_offset_phys..__kaslr_offset_phys + TEXT_OFFSET] region.
*/
pgtable_populate(kernel_start + TEXT_OFFSET, kernel_end, POPULATE_KERNEL);
pgtable_populate(AMODE31_START, AMODE31_END, POPULATE_DIRECT); pgtable_populate(AMODE31_START, AMODE31_END, POPULATE_DIRECT);
pgtable_populate(__abs_lowcore, __abs_lowcore + sizeof(struct lowcore), pgtable_populate(__abs_lowcore, __abs_lowcore + sizeof(struct lowcore),
POPULATE_ABS_LOWCORE); POPULATE_ABS_LOWCORE);
......
...@@ -109,7 +109,12 @@ SECTIONS ...@@ -109,7 +109,12 @@ SECTIONS
#ifdef CONFIG_KERNEL_UNCOMPRESSED #ifdef CONFIG_KERNEL_UNCOMPRESSED
. = ALIGN(PAGE_SIZE); . = ALIGN(PAGE_SIZE);
. += AMODE31_SIZE; /* .amode31 section */ . += AMODE31_SIZE; /* .amode31 section */
. = ALIGN(1 << 20); /* _SEGMENT_SIZE */
/*
* Make sure the location counter is not less than TEXT_OFFSET.
* _SEGMENT_SIZE is not available, use ALIGN(1 << 20) instead.
*/
. = MAX(TEXT_OFFSET, ALIGN(1 << 20));
#else #else
. = ALIGN(8); . = ALIGN(8);
#endif #endif
......
...@@ -279,8 +279,9 @@ static inline unsigned long virt_to_pfn(const void *kaddr) ...@@ -279,8 +279,9 @@ static inline unsigned long virt_to_pfn(const void *kaddr)
#define AMODE31_SIZE (3 * PAGE_SIZE) #define AMODE31_SIZE (3 * PAGE_SIZE)
#define KERNEL_IMAGE_SIZE (512 * 1024 * 1024) #define KERNEL_IMAGE_SIZE (512 * 1024 * 1024)
#define __START_KERNEL 0x100000
#define __NO_KASLR_START_KERNEL CONFIG_KERNEL_IMAGE_BASE #define __NO_KASLR_START_KERNEL CONFIG_KERNEL_IMAGE_BASE
#define __NO_KASLR_END_KERNEL (__NO_KASLR_START_KERNEL + KERNEL_IMAGE_SIZE) #define __NO_KASLR_END_KERNEL (__NO_KASLR_START_KERNEL + KERNEL_IMAGE_SIZE)
#define TEXT_OFFSET 0x100000
#endif /* _S390_PAGE_H */ #endif /* _S390_PAGE_H */
...@@ -39,7 +39,7 @@ PHDRS { ...@@ -39,7 +39,7 @@ PHDRS {
SECTIONS SECTIONS
{ {
. = __START_KERNEL; . = TEXT_OFFSET;
.text : { .text : {
_stext = .; /* Start of text section */ _stext = .; /* Start of text section */
_text = .; /* Text and read-only data */ _text = .; /* Text and read-only data */
......
...@@ -280,7 +280,7 @@ static int do_reloc(struct section *sec, Elf_Rel *rel) ...@@ -280,7 +280,7 @@ static int do_reloc(struct section *sec, Elf_Rel *rel)
case R_390_GOTOFF64: case R_390_GOTOFF64:
break; break;
case R_390_64: case R_390_64:
add_reloc(&relocs64, offset - ehdr.e_entry); add_reloc(&relocs64, offset);
break; break;
default: default:
die("Unsupported relocation type: %d\n", r_type); die("Unsupported relocation type: %d\n", r_type);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment