This is just to see if the modification to fix_alignment is ok with people. Still need to fill in the actual implementation of emulate_spe().
diff --git a/arch/powerpc/kernel/align.c b/arch/powerpc/kernel/align.c index 4c47f9c..a9fcf25 100644 --- a/arch/powerpc/kernel/align.c +++ b/arch/powerpc/kernel/align.c @@ -189,6 +189,76 @@ static struct aligninfo aligninfo[128] = { INVALID, /* 11 1 1111 */ }; +#ifdef CONFIG_SPE +static struct aligninfo spe_aligninfo[64] = { + { 8, LD }, /* 0 00 00 0: evlddx */ + { 8, LD }, /* 0 00 00 1: evldd */ + { 8, LD }, /* 0 00 01 0: evldwx */ + { 8, LD }, /* 0 00 01 1: evldw */ + { 8, LD }, /* 0 00 10 0: evldhx */ + { 8, LD }, /* 0 00 10 1: evldh */ + INVALID, /* 0 00 11 0 */ + INVALID, /* 0 00 11 1 */ + { 2, LD }, /* 0 01 00 0: evlhhesplatx */ + { 2, LD }, /* 0 01 00 1: evlhhesplat */ + INVALID, /* 0 01 01 0 */ + INVALID, /* 0 01 01 1 */ + { 2, LD }, /* 0 01 10 0: evlhhousplatx */ + { 2, LD }, /* 0 01 10 1: evlhhousplat */ + { 2, LD }, /* 0 01 11 0: evlhhossplatx */ + { 2, LD }, /* 0 01 11 1: evlhhossplat */ + { 4, LD }, /* 0 10 00 0: evlwhex */ + { 4, LD }, /* 0 10 00 1: evlwhe */ + INVALID, /* 0 10 01 0 */ + INVALID, /* 0 10 01 1 */ + { 4, LD }, /* 0 10 10 0: evlwhoux */ + { 4, LD }, /* 0 10 10 1: evlwhou */ + { 4, LD }, /* 0 10 11 0: evlwhosx */ + { 4, LD }, /* 0 10 11 1: evlwhos */ + { 4, LD }, /* 0 11 00 0: evlwwsplatx */ + { 4, LD }, /* 0 11 00 1: evlwwsplat */ + INVALID, /* 0 11 01 0 */ + INVALID, /* 0 11 01 1 */ + { 4, LD }, /* 0 11 10 0: evlwhsplatx */ + { 4, LD }, /* 0 11 10 1: evlwhsplat */ + INVALID, /* 0 11 11 0 */ + INVALID, /* 0 11 11 1 */ + + { 8, ST }, /* 1 00 00 0: evstddx */ + { 8, ST }, /* 1 00 00 1: evstdd */ + { 8, ST }, /* 1 00 01 0: evstdwx */ + { 8, ST }, /* 1 00 01 1: evstdw */ + { 8, ST }, /* 1 00 10 0: evstdhx */ + { 8, ST }, /* 1 00 10 1: evstdh */ + INVALID, /* 1 00 11 0 */ + INVALID, /* 1 00 11 1 */ + INVALID, /* 1 01 00 0 */ + INVALID, /* 1 01 00 1 */ + INVALID, /* 1 01 01 0 */ + INVALID, /* 1 01 01 1 */ + INVALID, /* 1 01 10 0 */ + INVALID, /* 1 01 10 1 */ + INVALID, /* 1 01 11 0 */ + INVALID, /* 1 01 11 1 */ + { 4, ST }, /* 1 10 00 0: evstwhex */ + { 4, ST }, /* 1 10 00 1: evstwhe */ + INVALID, /* 1 10 01 0 */ + INVALID, /* 1 10 01 1 */ + { 4, ST }, /* 1 10 10 0: evstwhoux */ + { 4, ST }, /* 1 10 10 1: evstwhou */ + INVALID, /* 1 10 11 0 */ + INVALID, /* 1 10 11 1 */ + { 4, ST }, /* 1 11 00 0: evstwwex */ + { 4, ST }, /* 1 11 00 1: evstwwe */ + INVALID, /* 1 11 01 0 */ + INVALID, /* 1 11 01 1 */ + { 4, ST }, /* 1 11 10 0: evstwwox */ + { 4, ST }, /* 1 11 10 1: evstwwo */ + INVALID, /* 1 11 11 0 */ + INVALID, /* 1 11 11 1 */ +}; +#endif + /* * Create a DSISR value from the instruction */ @@ -392,6 +462,103 @@ static int emulate_fp_pair(struct pt_regs *regs, unsigned char __user *addr, return 1; /* exception handled and fixed up */ } +#ifdef CONFIG_SPE +/* + * Emulate SPE loads and stores. + * Only Book-E has these instructions, and it does true little-endian, + * so we don't need the address swizzling. + */ +static int emulate_spe(struct pt_regs *regs, unsigned char __user *addr, + unsigned int reg, unsigned int nb, + unsigned int flags, unsigned int instr) +{ + int i, ret; + + /* userland only */ + if (unlikely(!user_mode(regs))) + return 0; + + flush_spe_to_thread(current); + + + return 1; +} +#endif /* CONFIG_SPE */ /* * Called on alignment exception. Attempts to fixup @@ -408,7 +575,7 @@ int fix_alignment(struct pt_regs *regs) unsigned int dsisr; unsigned char __user *addr; unsigned long p, swiz; - int ret, t; + int ret, t, is_spe = 0; union { u64 ll; double dd; @@ -445,17 +612,31 @@ int fix_alignment(struct pt_regs *regs) if (cpu_has_feature(CPU_FTR_REAL_LE) && (regs->msr & MSR_LE)) instr = cpu_to_le32(instr); dsisr = make_dsisr(instr); +#ifdef CONFIG_SPE + if ((instr >> 26) == 0x4) + is_spe = 1; +#endif } /* extract the operation and registers from the dsisr */ reg = (dsisr >> 5) & 0x1f; /* source/dest register */ areg = dsisr & 0x1f; /* register to update */ - instr = (dsisr >> 10) & 0x7f; - instr |= (dsisr >> 13) & 0x60; + if (!is_spe) { + instr = (dsisr >> 10) & 0x7f; + instr |= (dsisr >> 13) & 0x60; - /* Lookup the operation in our table */ - nb = aligninfo[instr].len; - flags = aligninfo[instr].flags; + /* Lookup the operation in our table */ + nb = aligninfo[instr].len; + flags = aligninfo[instr].flags; + } +#ifdef CONFIG_SPE + else { + instr &= 0x3f; + /* Lookup the operation in our table */ + nb = spe_aligninfo[instr].len; + flags = spe_aligninfo[instr].flags; + } +#endif /* Byteswap little endian loads and stores */ swiz = 0; @@ -507,6 +688,14 @@ int fix_alignment(struct pt_regs *regs) flush_fp_to_thread(current); } +#ifdef CONFIG_SPE + /* Force the fprs into the save area so we can reference them */ + if (is_spe) { + return emulate_spe(regs, addr, reg, nb, flags, instr); + + } +#endif + /* Special case for 16-byte FP loads and stores */ if (nb == 16) return emulate_fp_pair(regs, addr, reg, flags); _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@ozlabs.org https://ozlabs.org/mailman/listinfo/linuxppc-dev