I started from the beginning with the patches, applying them on top of each other.
2. If that sounds like too much work, we could print the instructions in the kernel when we know we're going to return true. (Sorry the formatting of this will likely be messed up): diff --git a/arch/sparc/kernel/traps_64.c b/arch/sparc/kernel/traps_64.c index 27778b65a965..77e31d7c4097 100644 --- a/arch/sparc/kernel/traps_64.c +++ b/arch/sparc/kernel/traps_64.c @@ -277,11 +277,13 @@ bool is_no_fault_exception(struct pt_regs *regs) asi = (insn >> 5); /* immediate asi */ if ((asi & 0xf2) == ASI_PNF) { if (insn & 0x1000000) { /* op3[5:4]=3 */ + printk(KERN_ALERT "fixing up no fault insn %x\n", insn); handle_ldf_stq(insn, regs); return true; } else if (insn & 0x200000) { /* op3[2], stores */ return false; } + printk(KERN_ALERT "fixing up no fault insn %x\n", insn); handle_ld_nf(insn, regs); return true; } 3. I have a theory that the instruction may be something like this: sta %f0, [ %g0 ] #ASI_PNF which should assemble to 0xc1a01040. You could just try this instruction.
Tried it with this instruction. The following message appears 26 times again and then it hangs: fixing up no fault insn c1a01040 Starting now, my test program has 0xc1a01040 as first command and oxcf95d1ef as second.
4. If this does result in a crash, this patch might be the fix: diff --git a/arch/sparc/kernel/traps_64.c b/arch/sparc/kernel/traps_64.c index 77e31d7c4097..c0d2e3665e69 100644 --- a/arch/sparc/kernel/traps_64.c +++ b/arch/sparc/kernel/traps_64.c @@ -276,12 +276,12 @@ bool is_no_fault_exception(struct pt_regs *regs) else asi = (insn >> 5); /* immediate asi */ if ((asi & 0xf2) == ASI_PNF) { + if (insn & 0x200000) /* op3[2], stores */ + return false; if (insn & 0x1000000) { /* op3[5:4]=3 */ printk(KERN_ALERT "fixing up no fault insn %x\n", insn); handle_ldf_stq(insn, regs); return true; - } else if (insn & 0x200000) { /* op3[2], stores */ - return false; } printk(KERN_ALERT "fixing up no fault insn %x\n", insn); handle_ld_nf(insn, regs); 5. Try the patch in #4 regardless of the outcome of step #3
With this patch applied on top top of previous, it works with nothing in dmesg.
5. Here is another patch to try after the others: diff --git a/arch/sparc/kernel/traps_64.c b/arch/sparc/kernel/traps_64.c index c0d2e3665e69..e383738fdd9f 100644 --- a/arch/sparc/kernel/traps_64.c +++ b/arch/sparc/kernel/traps_64.c @@ -275,7 +275,7 @@ bool is_no_fault_exception(struct pt_regs *regs) asi = (regs->tstate >> 24); /* saved %asi */ else asi = (insn >> 5); /* immediate asi */ - if ((asi & 0xf2) == ASI_PNF) { + if (asi == ASI_PNF) { if (insn & 0x200000) /* op3[2], stores */ return false; if (insn & 0x1000000) { /* op3[5:4]=3 */
This works too without any hang or warnings. -- Meelis Roos <mroos@xxxxxxxx>