On Tue, 2008-07-01 at 00:32 -0500, Kumar Gala wrote: > To allow for a single kernel image on e500 v1/v2/mc we need to fixup lwsync > at runtime. On e500v1/v2 lwsync causes an illop so we need to patch up > the code. We default to 'sync' since that is always safe and if the cpu > is capable we will replace 'sync' with 'lwsync'. > > We introduce CPU_FTR_LWSYNC as a way to determine at runtime if this is > needed. This flag could be moved elsewhere since we dont really use it > for the normal CPU_FTR purpose. > > Finally we only store the relative offset in the fixup section to keep it > as small as possible rather than using a full fixup_entry. > > Signed-off-by: Kumar Gala <[EMAIL PROTECTED]>
I'd rather have some more generic "alternatives" stuff but in the meantime, Ack. > --- > > Rebased against latest powerpc-next tree. > > - k > > arch/powerpc/kernel/module.c | 6 ++++++ > arch/powerpc/kernel/setup_32.c | 4 ++++ > arch/powerpc/kernel/setup_64.c | 2 ++ > arch/powerpc/kernel/vdso.c | 10 ++++++++++ > arch/powerpc/kernel/vdso32/vdso32.lds.S | 3 +++ > arch/powerpc/kernel/vdso64/vdso64.lds.S | 3 +++ > arch/powerpc/kernel/vmlinux.lds.S | 6 ++++++ > arch/powerpc/lib/feature-fixups.c | 16 ++++++++++++++++ > include/asm-powerpc/code-patching.h | 3 ++- > include/asm-powerpc/cputable.h | 21 +++++++++++---------- > include/asm-powerpc/synch.h | 27 ++++++++++++++++++++++----- > 11 files changed, 85 insertions(+), 16 deletions(-) > > diff --git a/arch/powerpc/kernel/module.c b/arch/powerpc/kernel/module.c > index 40dd52d..af07003 100644 > --- a/arch/powerpc/kernel/module.c > +++ b/arch/powerpc/kernel/module.c > @@ -86,6 +86,12 @@ int module_finalize(const Elf_Ehdr *hdr, > (void *)sect->sh_addr + sect->sh_size); > #endif > > + sect = find_section(hdr, sechdrs, "__lwsync_fixup"); > + if (sect != NULL) > + do_lwsync_fixups(cur_cpu_spec->cpu_features, > + (void *)sect->sh_addr, > + (void *)sect->sh_addr + sect->sh_size); > + > return 0; > } > > diff --git a/arch/powerpc/kernel/setup_32.c b/arch/powerpc/kernel/setup_32.c > index 9e83add..0109e7f 100644 > --- a/arch/powerpc/kernel/setup_32.c > +++ b/arch/powerpc/kernel/setup_32.c > @@ -101,6 +101,10 @@ unsigned long __init early_init(unsigned long dt_ptr) > PTRRELOC(&__start___ftr_fixup), > PTRRELOC(&__stop___ftr_fixup)); > > + do_lwsync_fixups(spec->cpu_features, > + PTRRELOC(&__start___lwsync_fixup), > + PTRRELOC(&__stop___lwsync_fixup)); > + > return KERNELBASE + offset; > } > > diff --git a/arch/powerpc/kernel/setup_64.c b/arch/powerpc/kernel/setup_64.c > index 098fd96..04d8de9 100644 > --- a/arch/powerpc/kernel/setup_64.c > +++ b/arch/powerpc/kernel/setup_64.c > @@ -363,6 +363,8 @@ void __init setup_system(void) > &__start___ftr_fixup, &__stop___ftr_fixup); > do_feature_fixups(powerpc_firmware_features, > &__start___fw_ftr_fixup, &__stop___fw_ftr_fixup); > + do_lwsync_fixups(cur_cpu_spec->cpu_features, > + &__start___lwsync_fixup, &__stop___lwsync_fixup); > > /* > * Unflatten the device-tree passed by prom_init or kexec > diff --git a/arch/powerpc/kernel/vdso.c b/arch/powerpc/kernel/vdso.c > index ce245a8..f177c60 100644 > --- a/arch/powerpc/kernel/vdso.c > +++ b/arch/powerpc/kernel/vdso.c > @@ -571,6 +571,11 @@ static __init int vdso_fixup_features(struct > lib32_elfinfo *v32, > if (start64) > do_feature_fixups(powerpc_firmware_features, > start64, start64 + size64); > + > + start64 = find_section64(v64->hdr, "__lwsync_fixup", &size64); > + if (start64) > + do_lwsync_fixups(cur_cpu_spec->cpu_features, > + start64, start64 + size64); > #endif /* CONFIG_PPC64 */ > > start32 = find_section32(v32->hdr, "__ftr_fixup", &size32); > @@ -585,6 +590,11 @@ static __init int vdso_fixup_features(struct > lib32_elfinfo *v32, > start32, start32 + size32); > #endif /* CONFIG_PPC64 */ > > + start32 = find_section32(v32->hdr, "__lwsync_fixup", &size32); > + if (start32) > + do_lwsync_fixups(cur_cpu_spec->cpu_features, > + start32, start32 + size32); > + > return 0; > } > > diff --git a/arch/powerpc/kernel/vdso32/vdso32.lds.S > b/arch/powerpc/kernel/vdso32/vdso32.lds.S > index 2717935..be3b6a4 100644 > --- a/arch/powerpc/kernel/vdso32/vdso32.lds.S > +++ b/arch/powerpc/kernel/vdso32/vdso32.lds.S > @@ -33,6 +33,9 @@ SECTIONS > . = ALIGN(8); > __ftr_fixup : { *(__ftr_fixup) } > > + . = ALIGN(8); > + __lwsync_fixup : { *(__lwsync_fixup) } > + > #ifdef CONFIG_PPC64 > . = ALIGN(8); > __fw_ftr_fixup : { *(__fw_ftr_fixup) } > diff --git a/arch/powerpc/kernel/vdso64/vdso64.lds.S > b/arch/powerpc/kernel/vdso64/vdso64.lds.S > index e608d1b..d0b2526 100644 > --- a/arch/powerpc/kernel/vdso64/vdso64.lds.S > +++ b/arch/powerpc/kernel/vdso64/vdso64.lds.S > @@ -35,6 +35,9 @@ SECTIONS > __ftr_fixup : { *(__ftr_fixup) } > > . = ALIGN(8); > + __lwsync_fixup : { *(__lwsync_fixup) } > + > + . = ALIGN(8); > __fw_ftr_fixup : { *(__fw_ftr_fixup) } > > /* > diff --git a/arch/powerpc/kernel/vmlinux.lds.S > b/arch/powerpc/kernel/vmlinux.lds.S > index 3c07811..6856f6c 100644 > --- a/arch/powerpc/kernel/vmlinux.lds.S > +++ b/arch/powerpc/kernel/vmlinux.lds.S > @@ -127,6 +127,12 @@ SECTIONS > *(__ftr_fixup) > __stop___ftr_fixup = .; > } > + . = ALIGN(8); > + __lwsync_fixup : AT(ADDR(__lwsync_fixup) - LOAD_OFFSET) { > + __start___lwsync_fixup = .; > + *(__lwsync_fixup) > + __stop___lwsync_fixup = .; > + } > #ifdef CONFIG_PPC64 > . = ALIGN(8); > __fw_ftr_fixup : AT(ADDR(__fw_ftr_fixup) - LOAD_OFFSET) { > diff --git a/arch/powerpc/lib/feature-fixups.c > b/arch/powerpc/lib/feature-fixups.c > index 48e1ed8..ac5f5a1 100644 > --- a/arch/powerpc/lib/feature-fixups.c > +++ b/arch/powerpc/lib/feature-fixups.c > @@ -110,6 +110,22 @@ void do_feature_fixups(unsigned long value, void > *fixup_start, void *fixup_end) > } > } > > +void do_lwsync_fixups(unsigned long value, void *fixup_start, void > *fixup_end) > +{ > + unsigned int *start, *end, *dest; > + > + if (!(value & CPU_FTR_LWSYNC)) > + return ; > + > + start = fixup_start; > + end = fixup_end; > + > + for (; start < end; start++) { > + dest = (void *)start + *start; > + patch_instruction(dest, PPC_LWSYNC_INSTR); > + } > +} > + > #ifdef CONFIG_FTR_FIXUP_SELFTEST > > #define check(x) \ > diff --git a/include/asm-powerpc/code-patching.h > b/include/asm-powerpc/code-patching.h > index ef3a5d1..107d9b9 100644 > --- a/include/asm-powerpc/code-patching.h > +++ b/include/asm-powerpc/code-patching.h > @@ -12,7 +12,8 @@ > > #include <asm/types.h> > > -#define PPC_NOP_INSTR 0x60000000 > +#define PPC_NOP_INSTR 0x60000000 > +#define PPC_LWSYNC_INSTR 0x7c2004ac > > /* Flags for create_branch: > * "b" == create_branch(addr, target, 0); > diff --git a/include/asm-powerpc/cputable.h b/include/asm-powerpc/cputable.h > index 4e4491c..3171ac9 100644 > --- a/include/asm-powerpc/cputable.h > +++ b/include/asm-powerpc/cputable.h > @@ -156,6 +156,7 @@ extern void do_feature_fixups(unsigned long value, void > *fixup_start, > #define CPU_FTR_UNIFIED_ID_CACHE ASM_CONST(0x0000000001000000) > #define CPU_FTR_SPE ASM_CONST(0x0000000002000000) > #define CPU_FTR_NEED_PAIRED_STWCX ASM_CONST(0x0000000004000000) > +#define CPU_FTR_LWSYNC ASM_CONST(0x0000000008000000) > > /* > * Add the 64-bit processor unique features in the top half of the word; > @@ -369,43 +370,43 @@ extern void do_feature_fixups(unsigned long value, void > *fixup_start, > CPU_FTR_NODSISRALIGN) > #define CPU_FTRS_E500MC (CPU_FTR_MAYBE_CAN_DOZE | CPU_FTR_USE_TB | \ > CPU_FTR_MAYBE_CAN_NAP | CPU_FTR_BIG_PHYS | CPU_FTR_NODSISRALIGN | \ > - CPU_FTR_L2CSR) > + CPU_FTR_L2CSR | CPU_FTR_LWSYNC) > #define CPU_FTRS_GENERIC_32 (CPU_FTR_COMMON | CPU_FTR_NODSISRALIGN) > > /* 64-bit CPUs */ > -#define CPU_FTRS_POWER3 (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_POWER3 (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_IABR | CPU_FTR_PPC_LE) > -#define CPU_FTRS_RS64 (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_RS64 (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_IABR | \ > CPU_FTR_MMCRA | CPU_FTR_CTRL) > -#define CPU_FTRS_POWER4 (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_POWER4 (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | CPU_FTR_CTRL | \ > CPU_FTR_MMCRA) > -#define CPU_FTRS_PPC970 (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_PPC970 (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | CPU_FTR_CTRL | \ > CPU_FTR_ALTIVEC_COMP | CPU_FTR_CAN_NAP | CPU_FTR_MMCRA) > -#define CPU_FTRS_POWER5 (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_POWER5 (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | CPU_FTR_CTRL | \ > CPU_FTR_MMCRA | CPU_FTR_SMT | \ > CPU_FTR_COHERENT_ICACHE | CPU_FTR_LOCKLESS_TLBIE | \ > CPU_FTR_PURR) > -#define CPU_FTRS_POWER6 (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_POWER6 (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | CPU_FTR_CTRL | \ > CPU_FTR_MMCRA | CPU_FTR_SMT | \ > CPU_FTR_COHERENT_ICACHE | CPU_FTR_LOCKLESS_TLBIE | \ > CPU_FTR_PURR | CPU_FTR_SPURR | CPU_FTR_REAL_LE | \ > CPU_FTR_DSCR) > -#define CPU_FTRS_POWER7 (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_POWER7 (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | CPU_FTR_CTRL | \ > CPU_FTR_MMCRA | CPU_FTR_SMT | \ > CPU_FTR_COHERENT_ICACHE | CPU_FTR_LOCKLESS_TLBIE | \ > CPU_FTR_PURR | CPU_FTR_SPURR | CPU_FTR_REAL_LE | \ > CPU_FTR_DSCR) > -#define CPU_FTRS_CELL (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_CELL (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | CPU_FTR_CTRL | \ > CPU_FTR_ALTIVEC_COMP | CPU_FTR_MMCRA | CPU_FTR_SMT | \ > CPU_FTR_PAUSE_ZERO | CPU_FTR_CI_LARGE_PAGE | CPU_FTR_CELL_TB_BUG) > -#define CPU_FTRS_PA6T (CPU_FTR_USE_TB | \ > +#define CPU_FTRS_PA6T (CPU_FTR_USE_TB | CPU_FTR_LWSYNC | \ > CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | \ > CPU_FTR_ALTIVEC_COMP | CPU_FTR_CI_LARGE_PAGE | \ > CPU_FTR_PURR | CPU_FTR_REAL_LE | CPU_FTR_NO_SLBIE_B) > diff --git a/include/asm-powerpc/synch.h b/include/asm-powerpc/synch.h > index 42a1ef5..4737c61 100644 > --- a/include/asm-powerpc/synch.h > +++ b/include/asm-powerpc/synch.h > @@ -3,20 +3,37 @@ > #ifdef __KERNEL__ > > #include <linux/stringify.h> > +#include <asm/feature-fixups.h> > > -#if defined(__powerpc64__) || defined(CONFIG_PPC_E500MC) > -#define __SUBARCH_HAS_LWSYNC > -#endif > +#ifndef __ASSEMBLY__ > +extern unsigned int __start___lwsync_fixup, __stop___lwsync_fixup; > +extern void do_lwsync_fixups(unsigned long value, void *fixup_start, > + void *fixup_end); > +#endif /* __ASSEMBLY__ */ > + > +#define START_LWSYNC_SECTION(label) label##1: > +#define MAKE_LWSYNC_SECTION_ENTRY(label, sect) \ > +label##4: \ > + .pushsection sect,"a"; \ > + .align 2; \ > +label##5: \ > + .long label##1b-label##5b; \ > + .popsection; > > -#ifdef __SUBARCH_HAS_LWSYNC > +#if defined(__powerpc64__) > # define LWSYNC lwsync > +#elif defined(CONFIG_E500) > +# define LWSYNC \ > + START_LWSYNC_SECTION(97); \ > + sync; \ > + MAKE_LWSYNC_SECTION_ENTRY(97, __lwsync_fixup); > #else > # define LWSYNC sync > #endif > > #ifdef CONFIG_SMP > #define ISYNC_ON_SMP "\n\tisync\n" > -#define LWSYNC_ON_SMP __stringify(LWSYNC) "\n" > +#define LWSYNC_ON_SMP stringify_in_c(LWSYNC) "\n" > #else > #define ISYNC_ON_SMP > #define LWSYNC_ON_SMP _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@ozlabs.org https://ozlabs.org/mailman/listinfo/linuxppc-dev