On Thu, 2008-03-27 at 16:39 -0800, Badari Pulavarty wrote:
> ppc kernel maintains information about logical memory blocks in
> lmb.memory structure at the boot time. Its not updated for
> hotplug memory add/remove. hotplug memory notifier for memory
> add/remove now updates lmb.memory.
> 
> This information is useful for eHEA driver to find out the memory 
> layout and holes.
> 
> NOTE: No special locking is needed for lmb_add() and lmb_remove().
> Calls to these are serialized by caller. (pSeries_reconfig_chain).

My worry here is the lack of locking...

> Signed-off-by: Badari Pulavarty <[EMAIL PROTECTED]>
> ---
>  arch/powerpc/platforms/pseries/hotplug-memory.c |   43 +++++++++++++++
>  include/linux/lmb.h                             |    3 -
>  lib/lmb.c                                       |   66 
> ++++++++++++++++++++----
>  3 files changed, 102 insertions(+), 10 deletions(-)
> 
> Index: linux-2.6.25-rc3/lib/lmb.c
> ===================================================================
> --- linux-2.6.25-rc3.orig/lib/lmb.c   2008-03-05 10:44:29.000000000 -0800
> +++ linux-2.6.25-rc3/lib/lmb.c        2008-03-05 10:44:56.000000000 -0800
> @@ -54,14 +54,13 @@ void lmb_dump_all(void)
>  #endif /* DEBUG */
>  }
>  
> -static unsigned long __init lmb_addrs_overlap(u64 base1,
> -             u64 size1, u64 base2, u64 size2)
> +static unsigned long lmb_addrs_overlap(u64 base1, u64 size1, u64 base2,
> +             u64 size2)
>  {
>       return ((base1 < (base2+size2)) && (base2 < (base1+size1)));
>  }
>  
> -static long __init lmb_addrs_adjacent(u64 base1, u64 size1,
> -             u64 base2, u64 size2)
> +static long lmb_addrs_adjacent(u64 base1, u64 size1, u64 base2, u64 size2)
>  {
>       if (base2 == base1 + size1)
>               return 1;
> @@ -71,7 +70,7 @@ static long __init lmb_addrs_adjacent(u6
>       return 0;
>  }
>  
> -static long __init lmb_regions_adjacent(struct lmb_region *rgn,
> +static long lmb_regions_adjacent(struct lmb_region *rgn,
>               unsigned long r1, unsigned long r2)
>  {
>       u64 base1 = rgn->region[r1].base;
> @@ -82,7 +81,7 @@ static long __init lmb_regions_adjacent(
>       return lmb_addrs_adjacent(base1, size1, base2, size2);
>  }
>  
> -static void __init lmb_remove_region(struct lmb_region *rgn, unsigned long r)
> +static void lmb_remove_region(struct lmb_region *rgn, unsigned long r)
>  {
>       unsigned long i;
>  
> @@ -94,7 +93,7 @@ static void __init lmb_remove_region(str
>  }
>  
>  /* Assumption: base addr of region 1 < base addr of region 2 */
> -static void __init lmb_coalesce_regions(struct lmb_region *rgn,
> +static void lmb_coalesce_regions(struct lmb_region *rgn,
>               unsigned long r1, unsigned long r2)
>  {
>       rgn->region[r1].size += rgn->region[r2].size;
> @@ -129,7 +128,7 @@ void __init lmb_analyze(void)
>  }
>  
>  /* This routine called with relocation disabled. */
> -static long __init lmb_add_region(struct lmb_region *rgn, u64 base, u64 size)
> +static long lmb_add_region(struct lmb_region *rgn, u64 base, u64 size)
>  {
>       unsigned long coalesced = 0;
>       long adjacent, i;
> @@ -195,7 +194,7 @@ static long __init lmb_add_region(struct
>  }
>  
>  /* This routine may be called with relocation disabled. */
> -long __init lmb_add(u64 base, u64 size)
> +long lmb_add(u64 base, u64 size)
>  {
>       struct lmb_region *_rgn = &(lmb.memory);
>  
> @@ -207,6 +206,55 @@ long __init lmb_add(u64 base, u64 size)
>  
>  }
>  
> +long lmb_remove(u64 base, u64 size)
> +{
> +     struct lmb_region *rgn = &(lmb.memory);
> +     u64 rgnbegin, rgnend;
> +     u64 end = base + size;
> +     int i;
> +
> +     rgnbegin = rgnend = 0; /* supress gcc warnings */
> +
> +     /* Find the region where (base, size) belongs to */
> +     for (i=0; i < rgn->cnt; i++) {
> +             rgnbegin = rgn->region[i].base;
> +             rgnend = rgnbegin + rgn->region[i].size;
> +
> +             if ((rgnbegin <= base) && (end <= rgnend))
> +                     break;
> +     }
> +
> +     /* Didn't find the region */
> +     if (i == rgn->cnt)
> +             return -1;
> +
> +     /* Check to see if we are removing entire region */
> +     if ((rgnbegin == base) && (rgnend == end)) {
> +             lmb_remove_region(rgn, i);
> +             return 0;
> +     }
> +
> +     /* Check to see if region is matching at the front */
> +     if (rgnbegin == base) {
> +             rgn->region[i].base = end;
> +             rgn->region[i].size -= size;
> +             return 0;
> +     }
> +
> +     /* Check to see if the region is matching at the end */
> +     if (rgnend == end) {
> +             rgn->region[i].size -= size;
> +             return 0;
> +     }
> +
> +     /*
> +      * We need to split the entry -  adjust the current one to the
> +      * beginging of the hole and add the region after hole.
> +      */
> +     rgn->region[i].size = base - rgn->region[i].base;
> +     return lmb_add_region(rgn, end, rgnend - end);
> +}
> +
>  long __init lmb_reserve(u64 base, u64 size)
>  {
>       struct lmb_region *_rgn = &(lmb.reserved);
> Index: linux-2.6.25-rc3/arch/powerpc/platforms/pseries/hotplug-memory.c
> ===================================================================
> --- linux-2.6.25-rc3.orig/arch/powerpc/platforms/pseries/hotplug-memory.c     
> 2008-03-05 10:44:51.000000000 -0800
> +++ linux-2.6.25-rc3/arch/powerpc/platforms/pseries/hotplug-memory.c  
> 2008-03-05 10:45:06.000000000 -0800
> @@ -10,6 +10,7 @@
>   */
>  
>  #include <linux/of.h>
> +#include <linux/lmb.h>
>  #include <asm/firmware.h>
>  #include <asm/machdep.h>
>  #include <asm/pSeries_reconfig.h>
> @@ -58,6 +59,11 @@ static int pseries_remove_memory(struct 
>               return ret;
>  
>       /*
> +      * Update memory regions for memory remove
> +      */
> +     lmb_remove(start_pfn << PAGE_SHIFT, regs[3]);
> +
> +     /*
>        * Remove htab bolted mappings for this section of memory
>        */
>       start = (unsigned long)__va(start_pfn << PAGE_SHIFT);
> @@ -65,6 +71,41 @@ static int pseries_remove_memory(struct 
>       return ret;
>  }
>  
> +static int pseries_add_memory(struct device_node *np)
> +{
> +     const char *type;
> +     const unsigned int *my_index;
> +     const unsigned int *regs;
> +     u64 start_pfn;
> +     int ret = -EINVAL;
> +
> +     /*
> +      * Check to see if we are actually adding memory
> +      */
> +     type = of_get_property(np, "device_type", NULL);
> +     if (type == NULL || strcmp(type, "memory") != 0)
> +             return 0;
> +
> +     /*
> +      * Find the memory index and size of the removing section
> +      */
> +     my_index = of_get_property(np, "ibm,my-drc-index", NULL);
> +     if (!my_index)
> +             return ret;
> +
> +     regs = of_get_property(np, "reg", NULL);
> +     if (!regs)
> +             return ret;
> +
> +     start_pfn = section_nr_to_pfn(*my_index & 0xffff);
> +
> +     /*
> +      * Update memory region to represent the memory add
> +      */
> +     lmb_add(start_pfn << PAGE_SHIFT, regs[3]);
> +     return 0;
> +}
> +
>  static int pseries_memory_notifier(struct notifier_block *nb,
>                               unsigned long action, void *node)
>  {
> @@ -72,6 +113,8 @@ static int pseries_memory_notifier(struc
>  
>       switch (action) {
>       case PSERIES_RECONFIG_ADD:
> +             if (pseries_add_memory(node))
> +                     err = NOTIFY_BAD;
>               break;
>       case PSERIES_RECONFIG_REMOVE:
>               if (pseries_remove_memory(node))
> Index: linux-2.6.25-rc3/include/linux/lmb.h
> ===================================================================
> --- linux-2.6.25-rc3.orig/include/linux/lmb.h 2008-03-05 10:44:29.000000000 
> -0800
> +++ linux-2.6.25-rc3/include/linux/lmb.h      2008-03-05 10:44:56.000000000 
> -0800
> @@ -40,7 +40,8 @@ extern struct lmb lmb;
>  
>  extern void __init lmb_init(void);
>  extern void __init lmb_analyze(void);
> -extern long __init lmb_add(u64 base, u64 size);
> +extern long lmb_add(u64 base, u64 size);
> +extern long lmb_remove(u64 base, u64 size);
>  extern long __init lmb_reserve(u64 base, u64 size);
>  extern u64 __init lmb_alloc(u64 size, u64 align);
>  extern u64 __init lmb_alloc_base(u64 size,
> 
> 
> _______________________________________________
> Linuxppc-dev mailing list
> Linuxppc-dev@ozlabs.org
> https://ozlabs.org/mailman/listinfo/linuxppc-dev

_______________________________________________
Linuxppc-dev mailing list
Linuxppc-dev@ozlabs.org
https://ozlabs.org/mailman/listinfo/linuxppc-dev

Reply via email to