diff options
Diffstat (limited to 'arch/powerpc/kernel/prom.c')
| -rw-r--r-- | arch/powerpc/kernel/prom.c | 33 | 
1 files changed, 29 insertions, 4 deletions
| diff --git a/arch/powerpc/kernel/prom.c b/arch/powerpc/kernel/prom.c index 05e7fb47a7a4..c4d7078e5295 100644 --- a/arch/powerpc/kernel/prom.c +++ b/arch/powerpc/kernel/prom.c @@ -23,7 +23,6 @@  #include <linux/spinlock.h>  #include <linux/types.h>  #include <linux/pci.h> -#include <linux/stringify.h>  #include <linux/delay.h>  #include <linux/initrd.h>  #include <linux/bitops.h> @@ -440,6 +439,29 @@ static int __init early_init_dt_scan_chosen_ppc(unsigned long node,  	return 1;  } +/* + * Compare the range against max mem limit and update + * size if it cross the limit. + */ + +#ifdef CONFIG_SPARSEMEM +static bool validate_mem_limit(u64 base, u64 *size) +{ +	u64 max_mem = 1UL << (MAX_PHYSMEM_BITS); + +	if (base >= max_mem) +		return false; +	if ((base + *size) > max_mem) +		*size = max_mem - base; +	return true; +} +#else +static bool validate_mem_limit(u64 base, u64 *size) +{ +	return true; +} +#endif +  #ifdef CONFIG_PPC_PSERIES  /*   * Interpret the ibm dynamic reconfiguration memory LMBs. @@ -494,7 +516,8 @@ static void __init early_init_drmem_lmb(struct drmem_lmb *lmb,  		}  		DBG("Adding: %llx -> %llx\n", base, size); -		memblock_add(base, size); +		if (validate_mem_limit(base, &size)) +			memblock_add(base, size);  	} while (--rngs);  }  #endif /* CONFIG_PPC_PSERIES */ @@ -548,8 +571,10 @@ void __init early_init_dt_add_memory_arch(u64 base, u64 size)  	}  	/* Add the chunk to the MEMBLOCK list */ -	if (add_mem_to_memblock) -		memblock_add(base, size); +	if (add_mem_to_memblock) { +		if (validate_mem_limit(base, &size)) +			memblock_add(base, size); +	}  }  static void __init early_reserve_mem_dt(void) | 
