Commit 5ac47f7a authored by Anton Blanchard's avatar Anton Blanchard Committed by Benjamin Herrenschmidt

powerpc: Relocate prom_init.c on 64bit

The ppc64 kernel can get loaded at any address which means
our very early init code in prom_init.c must be relocatable. We do
this with a pretty nasty RELOC() macro that we wrap accesses of
variables with. It is very fragile and sometimes we forget to add a
RELOC() to an uncommon path or sometimes a compiler change breaks it.

32bit has a much more elegant solution where we build prom_init.c
with -mrelocatable and then process the relocations manually.
Unfortunately we can't do the equivalent on 64bit and we would
have to build the entire kernel relocatable (-pie), resulting in a
large increase in kernel footprint (megabytes of relocation data).
The relocation data will be marked __initdata but it still creates
more pressure on our already tight memory layout at boot.

Alan Modra pointed out that the 64bit ABI is relocatable even
if we don't build with -pie, we just need to relocate the TOC.
This patch implements that idea and relocates the TOC entries of
prom_init.c. An added bonus is there are very few relocations to
process which helps keep boot times on simulators down.

gcc does not put 64bit integer constants into the TOC but to be
safe we may want a build time script which passes through the
prom_init.c TOC entries to make sure everything looks reasonable.
Signed-off-by: default avatarAnton Blanchard <anton@samba.org>
Signed-off-by: default avatarBenjamin Herrenschmidt <benh@kernel.crashing.org>
parent e95e09ce
...@@ -136,6 +136,7 @@ head-$(CONFIG_FSL_BOOKE) := arch/powerpc/kernel/head_fsl_booke.o ...@@ -136,6 +136,7 @@ head-$(CONFIG_FSL_BOOKE) := arch/powerpc/kernel/head_fsl_booke.o
head-$(CONFIG_PPC64) += arch/powerpc/kernel/entry_64.o head-$(CONFIG_PPC64) += arch/powerpc/kernel/entry_64.o
head-$(CONFIG_PPC_FPU) += arch/powerpc/kernel/fpu.o head-$(CONFIG_PPC_FPU) += arch/powerpc/kernel/fpu.o
head-$(CONFIG_ALTIVEC) += arch/powerpc/kernel/vector.o head-$(CONFIG_ALTIVEC) += arch/powerpc/kernel/vector.o
head-$(CONFIG_PPC_OF_BOOT_TRAMPOLINE) += arch/powerpc/kernel/prom_init.o
core-y += arch/powerpc/kernel/ \ core-y += arch/powerpc/kernel/ \
arch/powerpc/mm/ \ arch/powerpc/mm/ \
......
...@@ -10,6 +10,9 @@ ...@@ -10,6 +10,9 @@
extern char __end_interrupts[]; extern char __end_interrupts[];
extern char __prom_init_toc_start[];
extern char __prom_init_toc_end[];
static inline int in_kernel_text(unsigned long addr) static inline int in_kernel_text(unsigned long addr)
{ {
if (addr >= (unsigned long)_stext && addr < (unsigned long)__init_end) if (addr >= (unsigned long)_stext && addr < (unsigned long)__init_end)
......
...@@ -91,7 +91,6 @@ obj-$(CONFIG_RELOCATABLE_PPC32) += reloc_32.o ...@@ -91,7 +91,6 @@ obj-$(CONFIG_RELOCATABLE_PPC32) += reloc_32.o
obj-$(CONFIG_PPC32) += entry_32.o setup_32.o obj-$(CONFIG_PPC32) += entry_32.o setup_32.o
obj-$(CONFIG_PPC64) += dma-iommu.o iommu.o obj-$(CONFIG_PPC64) += dma-iommu.o iommu.o
obj-$(CONFIG_KGDB) += kgdb.o obj-$(CONFIG_KGDB) += kgdb.o
obj-$(CONFIG_PPC_OF_BOOT_TRAMPOLINE) += prom_init.o
obj-$(CONFIG_MODULES) += ppc_ksyms.o obj-$(CONFIG_MODULES) += ppc_ksyms.o
obj-$(CONFIG_BOOTX_TEXT) += btext.o obj-$(CONFIG_BOOTX_TEXT) += btext.o
obj-$(CONFIG_SMP) += smp.o obj-$(CONFIG_SMP) += smp.o
...@@ -142,6 +141,7 @@ GCOV_PROFILE_kprobes.o := n ...@@ -142,6 +141,7 @@ GCOV_PROFILE_kprobes.o := n
extra-$(CONFIG_PPC_FPU) += fpu.o extra-$(CONFIG_PPC_FPU) += fpu.o
extra-$(CONFIG_ALTIVEC) += vector.o extra-$(CONFIG_ALTIVEC) += vector.o
extra-$(CONFIG_PPC64) += entry_64.o extra-$(CONFIG_PPC64) += entry_64.o
extra-$(CONFIG_PPC_OF_BOOT_TRAMPOLINE) += prom_init.o
extra-y += systbl_chk.i extra-y += systbl_chk.i
$(obj)/systbl.o: systbl_chk $(obj)/systbl.o: systbl_chk
......
...@@ -66,8 +66,8 @@ ...@@ -66,8 +66,8 @@
* is running at whatever address it has been loaded at. * is running at whatever address it has been loaded at.
* On ppc32 we compile with -mrelocatable, which means that references * On ppc32 we compile with -mrelocatable, which means that references
* to extern and static variables get relocated automatically. * to extern and static variables get relocated automatically.
* On ppc64 we have to relocate the references explicitly with * ppc64 objects are always relocatable, we just need to relocate the
* RELOC. (Note that strings count as static variables.) * TOC.
* *
* Because OF may have mapped I/O devices into the area starting at * Because OF may have mapped I/O devices into the area starting at
* KERNELBASE, particularly on CHRP machines, we can't safely call * KERNELBASE, particularly on CHRP machines, we can't safely call
...@@ -79,13 +79,12 @@ ...@@ -79,13 +79,12 @@
* On ppc64, 64 bit values are truncated to 32 bits (and * On ppc64, 64 bit values are truncated to 32 bits (and
* fortunately don't get interpreted as two arguments). * fortunately don't get interpreted as two arguments).
*/ */
#define RELOC(x) (x)
#define ADDR(x) (u32)(unsigned long)(x)
#ifdef CONFIG_PPC64 #ifdef CONFIG_PPC64
#define RELOC(x) (*PTRRELOC(&(x)))
#define ADDR(x) (u32) add_reloc_offset((unsigned long)(x))
#define OF_WORKAROUNDS 0 #define OF_WORKAROUNDS 0
#else #else
#define RELOC(x) (x)
#define ADDR(x) (u32) (x)
#define OF_WORKAROUNDS of_workarounds #define OF_WORKAROUNDS of_workarounds
int of_workarounds; int of_workarounds;
#endif #endif
...@@ -334,9 +333,6 @@ static void __init prom_printf(const char *format, ...) ...@@ -334,9 +333,6 @@ static void __init prom_printf(const char *format, ...)
struct prom_t *_prom = &RELOC(prom); struct prom_t *_prom = &RELOC(prom);
va_start(args, format); va_start(args, format);
#ifdef CONFIG_PPC64
format = PTRRELOC(format);
#endif
for (p = format; *p != 0; p = q) { for (p = format; *p != 0; p = q) {
for (q = p; *q != 0 && *q != '\n' && *q != '%'; ++q) for (q = p; *q != 0 && *q != '\n' && *q != '%'; ++q)
; ;
...@@ -437,9 +433,6 @@ static unsigned int __init prom_claim(unsigned long virt, unsigned long size, ...@@ -437,9 +433,6 @@ static unsigned int __init prom_claim(unsigned long virt, unsigned long size,
static void __init __attribute__((noreturn)) prom_panic(const char *reason) static void __init __attribute__((noreturn)) prom_panic(const char *reason)
{ {
#ifdef CONFIG_PPC64
reason = PTRRELOC(reason);
#endif
prom_print(reason); prom_print(reason);
/* Do not call exit because it clears the screen on pmac /* Do not call exit because it clears the screen on pmac
* it also causes some sort of double-fault on early pmacs */ * it also causes some sort of double-fault on early pmacs */
...@@ -929,7 +922,7 @@ static void __init prom_send_capabilities(void) ...@@ -929,7 +922,7 @@ static void __init prom_send_capabilities(void)
* (we assume this is the same for all cores) and use it to * (we assume this is the same for all cores) and use it to
* divide NR_CPUS. * divide NR_CPUS.
*/ */
cores = (u32 *)PTRRELOC(&ibm_architecture_vec[IBM_ARCH_VEC_NRCORES_OFFSET]); cores = (u32 *)&ibm_architecture_vec[IBM_ARCH_VEC_NRCORES_OFFSET];
if (*cores != NR_CPUS) { if (*cores != NR_CPUS) {
prom_printf("WARNING ! " prom_printf("WARNING ! "
"ibm_architecture_vec structure inconsistent: %lu!\n", "ibm_architecture_vec structure inconsistent: %lu!\n",
...@@ -2850,6 +2843,53 @@ static void __init prom_check_initrd(unsigned long r3, unsigned long r4) ...@@ -2850,6 +2843,53 @@ static void __init prom_check_initrd(unsigned long r3, unsigned long r4)
#endif /* CONFIG_BLK_DEV_INITRD */ #endif /* CONFIG_BLK_DEV_INITRD */
} }
#ifdef CONFIG_PPC64
#ifdef CONFIG_RELOCATABLE
static void reloc_toc(void)
{
}
static void unreloc_toc(void)
{
}
#else
static void __reloc_toc(void *tocstart, unsigned long offset,
unsigned long nr_entries)
{
unsigned long i;
unsigned long *toc_entry = (unsigned long *)tocstart;
for (i = 0; i < nr_entries; i++) {
*toc_entry = *toc_entry + offset;
toc_entry++;
}
}
static void reloc_toc(void)
{
unsigned long offset = reloc_offset();
unsigned long nr_entries =
(__prom_init_toc_end - __prom_init_toc_start) / sizeof(long);
/* Need to add offset to get at __prom_init_toc_start */
__reloc_toc(__prom_init_toc_start + offset, offset, nr_entries);
mb();
}
static void unreloc_toc(void)
{
unsigned long offset = reloc_offset();
unsigned long nr_entries =
(__prom_init_toc_end - __prom_init_toc_start) / sizeof(long);
mb();
/* __prom_init_toc_start has been relocated, no need to add offset */
__reloc_toc(__prom_init_toc_start, -offset, nr_entries);
}
#endif
#endif
/* /*
* We enter here early on, when the Open Firmware prom is still * We enter here early on, when the Open Firmware prom is still
...@@ -2867,6 +2907,8 @@ unsigned long __init prom_init(unsigned long r3, unsigned long r4, ...@@ -2867,6 +2907,8 @@ unsigned long __init prom_init(unsigned long r3, unsigned long r4,
#ifdef CONFIG_PPC32 #ifdef CONFIG_PPC32
unsigned long offset = reloc_offset(); unsigned long offset = reloc_offset();
reloc_got2(offset); reloc_got2(offset);
#else
reloc_toc();
#endif #endif
_prom = &RELOC(prom); _prom = &RELOC(prom);
...@@ -3061,6 +3103,8 @@ unsigned long __init prom_init(unsigned long r3, unsigned long r4, ...@@ -3061,6 +3103,8 @@ unsigned long __init prom_init(unsigned long r3, unsigned long r4,
#ifdef CONFIG_PPC32 #ifdef CONFIG_PPC32
reloc_got2(-offset); reloc_got2(-offset);
#else
unreloc_toc();
#endif #endif
#ifdef CONFIG_PPC_EARLY_DEBUG_OPAL #ifdef CONFIG_PPC_EARLY_DEBUG_OPAL
......
...@@ -22,7 +22,7 @@ __secondary_hold_acknowledge __secondary_hold_spinloop __start ...@@ -22,7 +22,7 @@ __secondary_hold_acknowledge __secondary_hold_spinloop __start
strcmp strcpy strlcpy strlen strncmp strstr logo_linux_clut224 strcmp strcpy strlcpy strlen strncmp strstr logo_linux_clut224
reloc_got2 kernstart_addr memstart_addr linux_banner _stext reloc_got2 kernstart_addr memstart_addr linux_banner _stext
opal_query_takeover opal_do_takeover opal_enter_rtas opal_secondary_entry opal_query_takeover opal_do_takeover opal_enter_rtas opal_secondary_entry
boot_command_line" boot_command_line __prom_init_toc_start __prom_init_toc_end"
NM="$1" NM="$1"
OBJ="$2" OBJ="$2"
......
...@@ -218,6 +218,11 @@ SECTIONS ...@@ -218,6 +218,11 @@ SECTIONS
.got : AT(ADDR(.got) - LOAD_OFFSET) { .got : AT(ADDR(.got) - LOAD_OFFSET) {
__toc_start = .; __toc_start = .;
#ifndef CONFIG_RELOCATABLE
__prom_init_toc_start = .;
arch/powerpc/kernel/prom_init.o*(.toc .got)
__prom_init_toc_end = .;
#endif
*(.got) *(.got)
*(.toc) *(.toc)
} }
......
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