On Fri, 2009-08-28 at 17:06 -0500, Brian King wrote: > The SLB can change sizes across a live migration, which was not > being handled, resulting in possible machine crashes during > migration if migrating to a machine which has a smaller max SLB > size than the source machine. Fix this by first reducing the > SLB size to the minimum possible value, which is 32, prior to > migration. Then during the device tree update which occurs after > migration, we make the call to ensure the SLB gets updated. Also > add the slb_size to the lparcfg output so that the migration > tools can check to make sure the kernel has this capability > before allowing migration in scenarios where the SLB size will change. > > Signed-off-by: Brian King <brk...@linux.vnet.ibm.com> > ---
Thanks. I'll apply to -next hopefully tomorrow, and then we can send it to the various -stable. Cheers, Ben. > arch/powerpc/include/asm/mmu-hash64.h | 2 ++ > arch/powerpc/kernel/lparcfg.c | 3 +++ > arch/powerpc/kernel/rtas.c | 7 ++++++- > arch/powerpc/mm/slb.c | 16 ++++++++++++---- > arch/powerpc/platforms/pseries/reconfig.c | 9 ++++++++- > 5 files changed, 31 insertions(+), 6 deletions(-) > > diff -puN arch/powerpc/kernel/rtas.c~powerpc_slb_resize > arch/powerpc/kernel/rtas.c > --- linux-2.6/arch/powerpc/kernel/rtas.c~powerpc_slb_resize 2009-08-21 > 16:14:41.000000000 -0500 > +++ linux-2.6-bjking1/arch/powerpc/kernel/rtas.c 2009-08-21 > 16:14:41.000000000 -0500 > @@ -39,6 +39,7 @@ > #include <asm/smp.h> > #include <asm/atomic.h> > #include <asm/time.h> > +#include <asm/mmu-hash64.h> > > struct rtas_t rtas = { > .lock = __RAW_SPIN_LOCK_UNLOCKED > @@ -713,6 +714,7 @@ static void rtas_percpu_suspend_me(void > { > long rc = H_SUCCESS; > unsigned long msr_save; > + u16 slb_size = mmu_slb_size; > int cpu; > struct rtas_suspend_me_data *data = > (struct rtas_suspend_me_data *)info; > @@ -735,13 +737,16 @@ static void rtas_percpu_suspend_me(void > /* All other cpus are in H_JOIN, this cpu does > * the suspend. > */ > + slb_set_size(SLB_MIN_SIZE); > printk(KERN_DEBUG "calling ibm,suspend-me on cpu %i\n", > smp_processor_id()); > data->error = rtas_call(data->token, 0, 1, NULL); > > - if (data->error) > + if (data->error) { > printk(KERN_DEBUG "ibm,suspend-me returned %d\n", > data->error); > + slb_set_size(slb_size); > + } > } else { > printk(KERN_ERR "H_JOIN on cpu %i failed with rc = %ld\n", > smp_processor_id(), rc); > diff -puN arch/powerpc/include/asm/mmu-hash64.h~powerpc_slb_resize > arch/powerpc/include/asm/mmu-hash64.h > --- linux-2.6/arch/powerpc/include/asm/mmu-hash64.h~powerpc_slb_resize > 2009-08-21 16:14:41.000000000 -0500 > +++ linux-2.6-bjking1/arch/powerpc/include/asm/mmu-hash64.h 2009-08-21 > 16:14:41.000000000 -0500 > @@ -41,6 +41,7 @@ extern char initial_stab[]; > > #define SLB_NUM_BOLTED 3 > #define SLB_CACHE_ENTRIES 8 > +#define SLB_MIN_SIZE 32 > > /* Bits in the SLB ESID word */ > #define SLB_ESID_V ASM_CONST(0x0000000008000000) /* valid */ > @@ -296,6 +297,7 @@ extern void slb_flush_and_rebolt(void); > extern void stab_initialize(unsigned long stab); > > extern void slb_vmalloc_update(void); > +extern void slb_set_size(u16 size); > #endif /* __ASSEMBLY__ */ > > /* > diff -puN arch/powerpc/mm/slb.c~powerpc_slb_resize arch/powerpc/mm/slb.c > --- linux-2.6/arch/powerpc/mm/slb.c~powerpc_slb_resize 2009-08-21 > 16:14:41.000000000 -0500 > +++ linux-2.6-bjking1/arch/powerpc/mm/slb.c 2009-08-21 16:14:41.000000000 > -0500 > @@ -240,14 +240,22 @@ void switch_slb(struct task_struct *tsk, > static inline void patch_slb_encoding(unsigned int *insn_addr, > unsigned int immed) > { > - /* Assume the instruction had a "0" immediate value, just > - * "or" in the new value > - */ > - *insn_addr |= immed; > + *insn_addr = (*insn_addr & 0xffff0000) | immed; > flush_icache_range((unsigned long)insn_addr, 4+ > (unsigned long)insn_addr); > } > > +void slb_set_size(u16 size) > +{ > + extern unsigned int *slb_compare_rr_to_size; > + > + if (mmu_slb_size == size) > + return; > + > + mmu_slb_size = size; > + patch_slb_encoding(slb_compare_rr_to_size, mmu_slb_size); > +} > + > void slb_initialize(void) > { > unsigned long linear_llp, vmalloc_llp, io_llp; > diff -puN arch/powerpc/platforms/pseries/reconfig.c~powerpc_slb_resize > arch/powerpc/platforms/pseries/reconfig.c > --- linux-2.6/arch/powerpc/platforms/pseries/reconfig.c~powerpc_slb_resize > 2009-08-21 16:14:41.000000000 -0500 > +++ linux-2.6-bjking1/arch/powerpc/platforms/pseries/reconfig.c > 2009-08-28 13:36:55.000000000 -0500 > @@ -20,6 +20,7 @@ > #include <asm/machdep.h> > #include <asm/uaccess.h> > #include <asm/pSeries_reconfig.h> > +#include <asm/mmu-hash64.h> > > > > @@ -439,9 +440,15 @@ static int do_update_property(char *buf, > if (!newprop) > return -ENOMEM; > > + if (!strcmp(name, "slb-size") || !strcmp(name, "ibm,slb-size")) > + slb_set_size(*(int *)value); > + > oldprop = of_find_property(np, name,NULL); > - if (!oldprop) > + if (!oldprop) { > + if (strlen(name)) > + return prom_add_property(np, newprop); > return -ENODEV; > + } > > rc = prom_update_property(np, newprop, oldprop); > if (rc) > diff -puN arch/powerpc/kernel/lparcfg.c~powerpc_slb_resize > arch/powerpc/kernel/lparcfg.c > --- linux-2.6/arch/powerpc/kernel/lparcfg.c~powerpc_slb_resize > 2009-08-21 16:14:41.000000000 -0500 > +++ linux-2.6-bjking1/arch/powerpc/kernel/lparcfg.c 2009-08-21 > 16:14:41.000000000 -0500 > @@ -35,6 +35,7 @@ > #include <asm/prom.h> > #include <asm/vdso_datapage.h> > #include <asm/vio.h> > +#include <asm/mmu-hash64.h> > > #define MODULE_VERS "1.8" > #define MODULE_NAME "lparcfg" > @@ -537,6 +538,8 @@ static int pseries_lparcfg_data(struct s > > seq_printf(m, "shared_processor_mode=%d\n", lppaca[0].shared_proc); > > + seq_printf(m, "slb_size=%d\n", mmu_slb_size); > + > return 0; > } > > _ _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev