> -----Original Message----- > From: > linuxppc-dev-bounces+tiejun.chen=windriver....@lists.ozlabs.or > g > [mailto:linuxppc-dev-bounces+tiejun.chen=windriver.com@lists.o zlabs.org] On Behalf Of Tiejun Chen > Sent: Friday, July 08, 2011 6:05 PM > To: b...@kernel.crashing.org; linuxppc-...@ozlabs.org > Subject: [PATCH 1/2] booke/kprobe: make program exception to > use one dedicated exception stack > > When kprobe these operations such as store-and-update-word for SP(r1), > > stwu r1, -A(r1) > > The program exception is triggered, and PPC always allocate > an exception frame as shown as the follows: > > old r1 ---------- > ... > nip > gpr[2] ~ gpr[31] > gpr[1] <--------- old r1 is stored. > gpr[0] > -------- <--------- pr_regs @offset 16 bytes > padding > STACK_FRAME_REGS_MARKER > LR > back chain > new r1 ---------- > Then emulate_step() will emulate this instruction, 'stwu'. > Actually its equivalent to: > 1> Update pr_regs->gpr[1] = mem[old r1 + (-A)] > 2> stw [old r1], mem[old r1 + (-A)] > > Please notice the stack based on new r1 may be covered with mem[old r1 > +(-A)] when addr[old r1 + (-A)] < addr[old r1 + sizeof(an > exception frame0]. > So the above 2# operation will overwirte something to break > this exception frame then unexpected kernel problem will be issued. > > So looks we have to implement independed interrupt stack for > PPC program exception when CONFIG_BOOKE is enabled. Here we > can use EXC_LEVEL_EXCEPTION_PROLOG to replace original > NORMAL_EXCEPTION_PROLOG for program exception if > CONFIG_BOOKE. Then its always safe for kprobe with independed > exc stack from one pre-allocated and dedicated thread_info. > Actually this is just waht we did for critical/machine check > exceptions on PPC.
Update: You can understand this problem completely from our previous email thread, about '[BUG?] 3.0-rc4+ftrace+kprobe: set kprobe at instruction 'stwu' lead to system crash/freeze'. > > Signed-off-by: Tiejun Chen <tiejun.c...@windriver.com> > --- > arch/powerpc/include/asm/irq.h | 3 +++ > arch/powerpc/include/asm/reg.h | 4 ++++ > arch/powerpc/kernel/head_booke.h | 12 +++++++++++- > arch/powerpc/kernel/irq.c | 11 +++++++++++ > arch/powerpc/kernel/setup_32.c | 4 ++++ > 5 files changed, 33 insertions(+), 1 deletions(-) > > diff --git a/arch/powerpc/include/asm/irq.h > b/arch/powerpc/include/asm/irq.h index 1bff591..6d12169 100644 > --- a/arch/powerpc/include/asm/irq.h > +++ b/arch/powerpc/include/asm/irq.h > @@ -313,6 +313,9 @@ struct pt_regs; > extern struct thread_info *critirq_ctx[NR_CPUS]; extern > struct thread_info *dbgirq_ctx[NR_CPUS]; extern struct > thread_info *mcheckirq_ctx[NR_CPUS]; > +#if defined(CONFIG_KPROBES) && defined(CONFIG_BOOKE) extern struct > +thread_info *pgirq_ctx[NR_CPUS]; #endif > extern void exc_lvl_ctx_init(void); > #else > #define exc_lvl_ctx_init() > diff --git a/arch/powerpc/include/asm/reg.h > b/arch/powerpc/include/asm/reg.h index c5cae0d..34d6178 100644 > --- a/arch/powerpc/include/asm/reg.h > +++ b/arch/powerpc/include/asm/reg.h > @@ -885,6 +885,10 @@ > #endif > #define SPRN_SPRG_RVCPU SPRN_SPRG1 > #define SPRN_SPRG_WVCPU SPRN_SPRG1 > +#ifdef CONFIG_KPROBES > +#define SPRN_SPRG_RSCRATCH_PG SPRN_SPRG0 > +#define SPRN_SPRG_WSCRATCH_PG SPRN_SPRG0 > +#endif > #endif > > #ifdef CONFIG_8xx > diff --git a/arch/powerpc/kernel/head_booke.h > b/arch/powerpc/kernel/head_booke.h > index a0bf158..cf6cb1e 100644 > --- a/arch/powerpc/kernel/head_booke.h > +++ b/arch/powerpc/kernel/head_booke.h > @@ -79,6 +79,10 @@ > /* only on e500mc/e200 */ > #define DBG_STACK_BASE dbgirq_ctx > > +#if defined(CONFIG_KPROBES) > +#define PG_STACK_BASE pgirq_ctx > +#endif > + > #define EXC_LVL_FRAME_OVERHEAD (THREAD_SIZE - > INT_FRAME_SIZE - EXC_LVL_SIZE) > > #ifdef CONFIG_SMP > @@ -158,6 +162,12 @@ > EXC_LEVEL_EXCEPTION_PROLOG(DBG, SPRN_DSRR0, > SPRN_DSRR1) #define MCHECK_EXCEPTION_PROLOG \ > EXC_LEVEL_EXCEPTION_PROLOG(MC, SPRN_MCSRR0, SPRN_MCSRR1) > +#if defined(CONFIG_KPROBES) > +#define PROGRAM_EXCEPTION_PROLOG \ > + EXC_LEVEL_EXCEPTION_PROLOG(PG, SPRN_SRR0, > SPRN_SRR1) #else > +#define PROGRAM_EXCEPTION_PROLOG NORMAL_EXCEPTION_PROLOG > +#endif > > /* > * Exception vectors. > @@ -370,7 +380,7 @@ label: > > #define PROGRAM_EXCEPTION > \ > START_EXCEPTION(Program) > \ > - NORMAL_EXCEPTION_PROLOG; > \ > + PROGRAM_EXCEPTION_PROLOG; > \ > mfspr r4,SPRN_ESR; /* Grab the ESR and > save it */ \ > stw r4,_ESR(r11); > \ > addi r3,r1,STACK_FRAME_OVERHEAD; > \ > diff --git a/arch/powerpc/kernel/irq.c > b/arch/powerpc/kernel/irq.c index 5b428e3..ff5b8dd 100644 > --- a/arch/powerpc/kernel/irq.c > +++ b/arch/powerpc/kernel/irq.c > @@ -397,6 +397,10 @@ struct thread_info > *critirq_ctx[NR_CPUS] __read_mostly; > struct thread_info *dbgirq_ctx[NR_CPUS] __read_mostly; > struct thread_info *mcheckirq_ctx[NR_CPUS] __read_mostly; > > +#if defined(CONFIG_KPROBES) && defined(CONFIG_BOOKE) > +struct thread_info *pgirq_ctx[NR_CPUS] __read_mostly; > +#endif > + > void exc_lvl_ctx_init(void) > { > struct thread_info *tp; > @@ -423,6 +427,13 @@ void exc_lvl_ctx_init(void) > tp = mcheckirq_ctx[cpu_nr]; > tp->cpu = cpu_nr; > tp->preempt_count = HARDIRQ_OFFSET; > + > +#if defined(CONFIG_KPROBES) > + memset((void *)pgirq_ctx[i], 0, THREAD_SIZE); > + tp = pgirq_ctx[i]; > + tp->cpu = i; > + tp->preempt_count = 0; > +#endif > #endif > } > } > diff --git a/arch/powerpc/kernel/setup_32.c > b/arch/powerpc/kernel/setup_32.c index 620d792..f0c62ae 100644 > --- a/arch/powerpc/kernel/setup_32.c > +++ b/arch/powerpc/kernel/setup_32.c > @@ -272,6 +272,10 @@ static void __init exc_lvl_early_init(void) > __va(memblock_alloc(THREAD_SIZE, THREAD_SIZE)); > mcheckirq_ctx[hw_cpu] = (struct thread_info *) > __va(memblock_alloc(THREAD_SIZE, THREAD_SIZE)); > +#ifdef CONFIG_KPROBES > + pgirq_ctx[i] = (struct thread_info *) ^ Next version I should replace 'i' with 'hw_cpu' like the above usage :) Tiejun > + __va(memblock_alloc(THREAD_SIZE, > THREAD_SIZE)); #endif > #endif > } > } > -- > 1.5.6 _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev