2005-10-06 12:23:33 +10:00
/*
* Procedures for maintaining information about logical memory blocks .
*
* Peter Bergner , IBM Corp . June 2001.
* Copyright ( C ) 2001 Peter Bergner .
2008-02-13 16:56:49 -08:00
*
2005-10-06 12:23:33 +10:00
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/bitops.h>
2008-02-13 16:56:49 -08:00
# include <linux/lmb.h>
2005-10-06 12:23:33 +10:00
2006-01-25 21:31:30 +13:00
# define LMB_ALLOC_ANYWHERE 0
2005-11-15 14:49:22 +11:00
struct lmb lmb ;
2008-05-12 17:21:55 -07:00
static int lmb_debug ;
static int __init early_lmb ( char * p )
{
if ( p & & strstr ( p , " debug " ) )
lmb_debug = 1 ;
return 0 ;
}
early_param ( " lmb " , early_lmb ) ;
2009-01-14 20:46:02 +00:00
static void lmb_dump ( struct lmb_region * region , char * name )
2005-10-06 12:23:33 +10:00
{
2009-01-14 20:46:02 +00:00
unsigned long long base , size ;
int i ;
pr_info ( " %s.cnt = 0x%lx \n " , name , region - > cnt ) ;
for ( i = 0 ; i < region - > cnt ; i + + ) {
base = region - > region [ i ] . base ;
size = region - > region [ i ] . size ;
pr_info ( " %s[0x%x] \t 0x%016llx - 0x%016llx, 0x%llx bytes \n " ,
name , i , base , base + size - 1 , size ) ;
}
}
2005-10-06 12:23:33 +10:00
2009-01-14 20:46:02 +00:00
void lmb_dump_all ( void )
{
2008-05-12 17:21:55 -07:00
if ( ! lmb_debug )
return ;
2009-01-14 20:46:02 +00:00
pr_info ( " LMB configuration: \n " ) ;
pr_info ( " rmo_size = 0x%llx \n " , ( unsigned long long ) lmb . rmo_size ) ;
pr_info ( " memory.size = 0x%llx \n " , ( unsigned long long ) lmb . memory . size ) ;
2005-10-06 12:23:33 +10:00
2009-01-14 20:46:02 +00:00
lmb_dump ( & lmb . memory , " memory " ) ;
lmb_dump ( & lmb . reserved , " reserved " ) ;
2005-10-06 12:23:33 +10:00
}
2008-04-18 13:33:52 -07:00
static unsigned long lmb_addrs_overlap ( u64 base1 , u64 size1 , u64 base2 ,
u64 size2 )
2005-10-06 12:23:33 +10:00
{
2008-04-12 15:20:59 +10:00
return ( ( base1 < ( base2 + size2 ) ) & & ( base2 < ( base1 + size1 ) ) ) ;
2005-10-06 12:23:33 +10:00
}
2008-04-18 13:33:52 -07:00
static long lmb_addrs_adjacent ( u64 base1 , u64 size1 , u64 base2 , u64 size2 )
2005-10-06 12:23:33 +10:00
{
if ( base2 = = base1 + size1 )
return 1 ;
else if ( base1 = = base2 + size2 )
return - 1 ;
return 0 ;
}
2008-04-18 13:33:52 -07:00
static long lmb_regions_adjacent ( struct lmb_region * rgn ,
2005-10-06 12:23:33 +10:00
unsigned long r1 , unsigned long r2 )
{
2008-02-13 16:58:39 -08:00
u64 base1 = rgn - > region [ r1 ] . base ;
u64 size1 = rgn - > region [ r1 ] . size ;
u64 base2 = rgn - > region [ r2 ] . base ;
u64 size2 = rgn - > region [ r2 ] . size ;
2005-10-06 12:23:33 +10:00
return lmb_addrs_adjacent ( base1 , size1 , base2 , size2 ) ;
}
2008-04-18 13:33:52 -07:00
static void lmb_remove_region ( struct lmb_region * rgn , unsigned long r )
2005-10-06 12:23:33 +10:00
{
unsigned long i ;
2006-05-17 18:00:46 +10:00
for ( i = r ; i < rgn - > cnt - 1 ; i + + ) {
rgn - > region [ i ] . base = rgn - > region [ i + 1 ] . base ;
rgn - > region [ i ] . size = rgn - > region [ i + 1 ] . size ;
2005-10-06 12:23:33 +10:00
}
rgn - > cnt - - ;
}
2006-05-17 18:00:46 +10:00
/* Assumption: base addr of region 1 < base addr of region 2 */
2008-04-18 13:33:52 -07:00
static void lmb_coalesce_regions ( struct lmb_region * rgn ,
2006-05-17 18:00:46 +10:00
unsigned long r1 , unsigned long r2 )
{
rgn - > region [ r1 ] . size + = rgn - > region [ r2 ] . size ;
lmb_remove_region ( rgn , r2 ) ;
}
2005-10-06 12:23:33 +10:00
void __init lmb_init ( void )
{
/* Create a dummy zero size LMB which will get coalesced away later.
* This simplifies the lmb_add ( ) code below . . .
*/
lmb . memory . region [ 0 ] . base = 0 ;
lmb . memory . region [ 0 ] . size = 0 ;
lmb . memory . cnt = 1 ;
/* Ditto. */
lmb . reserved . region [ 0 ] . base = 0 ;
lmb . reserved . region [ 0 ] . size = 0 ;
lmb . reserved . cnt = 1 ;
}
void __init lmb_analyze ( void )
{
int i ;
lmb . memory . size = 0 ;
for ( i = 0 ; i < lmb . memory . cnt ; i + + )
lmb . memory . size + = lmb . memory . region [ i ] . size ;
}
2008-04-18 13:33:52 -07:00
static long lmb_add_region ( struct lmb_region * rgn , u64 base , u64 size )
2005-10-06 12:23:33 +10:00
{
2007-07-10 05:03:45 +10:00
unsigned long coalesced = 0 ;
long adjacent , i ;
2005-10-06 12:23:33 +10:00
2008-02-13 16:58:11 -08:00
if ( ( rgn - > cnt = = 1 ) & & ( rgn - > region [ 0 ] . size = = 0 ) ) {
rgn - > region [ 0 ] . base = base ;
rgn - > region [ 0 ] . size = size ;
return 0 ;
}
2005-10-06 12:23:33 +10:00
/* First try and coalesce this LMB with another. */
2008-04-12 15:20:59 +10:00
for ( i = 0 ; i < rgn - > cnt ; i + + ) {
2008-02-13 16:58:39 -08:00
u64 rgnbase = rgn - > region [ i ] . base ;
u64 rgnsize = rgn - > region [ i ] . size ;
2005-10-06 12:23:33 +10:00
2007-02-28 14:12:29 +11:00
if ( ( rgnbase = = base ) & & ( rgnsize = = size ) )
/* Already have this region, so we're done */
return 0 ;
2008-04-12 15:20:59 +10:00
adjacent = lmb_addrs_adjacent ( base , size , rgnbase , rgnsize ) ;
if ( adjacent > 0 ) {
2005-10-06 12:23:33 +10:00
rgn - > region [ i ] . base - = size ;
rgn - > region [ i ] . size + = size ;
coalesced + + ;
break ;
2008-04-12 15:20:59 +10:00
} else if ( adjacent < 0 ) {
2005-10-06 12:23:33 +10:00
rgn - > region [ i ] . size + = size ;
coalesced + + ;
break ;
}
}
2008-04-12 15:20:59 +10:00
if ( ( i < rgn - > cnt - 1 ) & & lmb_regions_adjacent ( rgn , i , i + 1 ) ) {
2005-10-06 12:23:33 +10:00
lmb_coalesce_regions ( rgn , i , i + 1 ) ;
coalesced + + ;
}
if ( coalesced )
return coalesced ;
if ( rgn - > cnt > = MAX_LMB_REGIONS )
return - 1 ;
/* Couldn't coalesce the LMB, so add it to the sorted table. */
2008-04-12 15:20:59 +10:00
for ( i = rgn - > cnt - 1 ; i > = 0 ; i - - ) {
2005-10-06 12:23:33 +10:00
if ( base < rgn - > region [ i ] . base ) {
rgn - > region [ i + 1 ] . base = rgn - > region [ i ] . base ;
rgn - > region [ i + 1 ] . size = rgn - > region [ i ] . size ;
} else {
rgn - > region [ i + 1 ] . base = base ;
rgn - > region [ i + 1 ] . size = size ;
break ;
}
}
2008-02-19 21:28:18 -08:00
if ( base < rgn - > region [ 0 ] . base ) {
rgn - > region [ 0 ] . base = base ;
rgn - > region [ 0 ] . size = size ;
}
2005-10-06 12:23:33 +10:00
rgn - > cnt + + ;
return 0 ;
}
2008-04-18 13:33:52 -07:00
long lmb_add ( u64 base , u64 size )
2005-10-06 12:23:33 +10:00
{
2008-04-12 15:20:59 +10:00
struct lmb_region * _rgn = & lmb . memory ;
2005-10-06 12:23:33 +10:00
/* On pSeries LPAR systems, the first LMB is our RMO region. */
if ( base = = 0 )
lmb . rmo_size = size ;
return lmb_add_region ( _rgn , base , size ) ;
}
2008-04-18 13:33:52 -07:00
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 ) ;
}
2008-02-13 16:58:39 -08:00
long __init lmb_reserve ( u64 base , u64 size )
2005-10-06 12:23:33 +10:00
{
2008-04-12 15:20:59 +10:00
struct lmb_region * _rgn = & lmb . reserved ;
2005-10-06 12:23:33 +10:00
2006-01-25 21:31:26 +13:00
BUG_ON ( 0 = = size ) ;
2005-10-06 12:23:33 +10:00
return lmb_add_region ( _rgn , base , size ) ;
}
2008-04-12 15:20:59 +10:00
long __init lmb_overlaps_region ( struct lmb_region * rgn , u64 base , u64 size )
2005-10-06 12:23:33 +10:00
{
unsigned long i ;
2008-04-12 15:20:59 +10:00
for ( i = 0 ; i < rgn - > cnt ; i + + ) {
2008-02-13 16:58:39 -08:00
u64 rgnbase = rgn - > region [ i ] . base ;
u64 rgnsize = rgn - > region [ i ] . size ;
2008-04-12 15:20:59 +10:00
if ( lmb_addrs_overlap ( base , size , rgnbase , rgnsize ) )
2005-10-06 12:23:33 +10:00
break ;
}
return ( i < rgn - > cnt ) ? i : - 1 ;
}
2008-03-24 20:50:48 +11:00
static u64 lmb_align_down ( u64 addr , u64 size )
{
return addr & ~ ( size - 1 ) ;
}
static u64 lmb_align_up ( u64 addr , u64 size )
{
return ( addr + ( size - 1 ) ) & ~ ( size - 1 ) ;
}
static u64 __init lmb_alloc_nid_unreserved ( u64 start , u64 end ,
u64 size , u64 align )
{
2008-04-12 15:20:59 +10:00
u64 base , res_base ;
2008-03-24 20:50:48 +11:00
long j ;
base = lmb_align_down ( ( end - size ) , align ) ;
2008-04-12 15:20:59 +10:00
while ( start < = base ) {
j = lmb_overlaps_region ( & lmb . reserved , base , size ) ;
if ( j < 0 ) {
/* this area isn't reserved, take it */
2008-05-12 16:51:15 -07:00
if ( lmb_add_region ( & lmb . reserved , base , size ) < 0 )
2008-04-12 15:20:59 +10:00
base = ~ ( u64 ) 0 ;
return base ;
}
res_base = lmb . reserved . region [ j ] . base ;
if ( res_base < size )
break ;
base = lmb_align_down ( res_base - size , align ) ;
2008-03-24 20:50:48 +11:00
}
return ~ ( u64 ) 0 ;
}
static u64 __init lmb_alloc_nid_region ( struct lmb_property * mp ,
u64 ( * nid_range ) ( u64 , u64 , int * ) ,
u64 size , u64 align , int nid )
{
u64 start , end ;
start = mp - > base ;
end = start + mp - > size ;
start = lmb_align_up ( start , align ) ;
while ( start < end ) {
u64 this_end ;
int this_nid ;
this_end = nid_range ( start , end , & this_nid ) ;
if ( this_nid = = nid ) {
u64 ret = lmb_alloc_nid_unreserved ( start , this_end ,
size , align ) ;
if ( ret ! = ~ ( u64 ) 0 )
return ret ;
}
start = this_end ;
}
return ~ ( u64 ) 0 ;
}
u64 __init lmb_alloc_nid ( u64 size , u64 align , int nid ,
u64 ( * nid_range ) ( u64 start , u64 end , int * nid ) )
{
struct lmb_region * mem = & lmb . memory ;
int i ;
2008-05-12 16:51:15 -07:00
BUG_ON ( 0 = = size ) ;
size = lmb_align_up ( size , align ) ;
2008-03-24 20:50:48 +11:00
for ( i = 0 ; i < mem - > cnt ; i + + ) {
u64 ret = lmb_alloc_nid_region ( & mem - > region [ i ] ,
nid_range ,
size , align , nid ) ;
if ( ret ! = ~ ( u64 ) 0 )
return ret ;
}
return lmb_alloc ( size , align ) ;
}
2008-02-13 16:58:39 -08:00
u64 __init lmb_alloc ( u64 size , u64 align )
2005-10-06 12:23:33 +10:00
{
return lmb_alloc_base ( size , align , LMB_ALLOC_ANYWHERE ) ;
}
2008-02-13 16:58:39 -08:00
u64 __init lmb_alloc_base ( u64 size , u64 align , u64 max_addr )
2006-01-25 21:31:28 +13:00
{
2008-02-13 16:58:39 -08:00
u64 alloc ;
2006-01-25 21:31:28 +13:00
alloc = __lmb_alloc_base ( size , align , max_addr ) ;
2006-03-16 14:47:20 +11:00
if ( alloc = = 0 )
2008-02-13 16:58:39 -08:00
panic ( " ERROR: Failed to allocate 0x%llx bytes below 0x%llx. \n " ,
( unsigned long long ) size , ( unsigned long long ) max_addr ) ;
2006-01-25 21:31:28 +13:00
return alloc ;
}
2008-02-13 16:58:39 -08:00
u64 __init __lmb_alloc_base ( u64 size , u64 align , u64 max_addr )
2005-10-06 12:23:33 +10:00
{
long i , j ;
2008-02-13 16:58:39 -08:00
u64 base = 0 ;
2008-04-12 15:20:59 +10:00
u64 res_base ;
2005-10-06 12:23:33 +10:00
2006-01-25 21:31:26 +13:00
BUG_ON ( 0 = = size ) ;
2008-05-12 16:51:15 -07:00
size = lmb_align_up ( size , align ) ;
2008-02-13 16:56:49 -08:00
/* On some platforms, make sure we allocate lowmem */
2008-04-12 15:20:59 +10:00
/* Note that LMB_REAL_LIMIT may be LMB_ALLOC_ANYWHERE */
2005-10-06 12:23:33 +10:00
if ( max_addr = = LMB_ALLOC_ANYWHERE )
2008-02-13 16:56:49 -08:00
max_addr = LMB_REAL_LIMIT ;
2008-04-12 15:20:59 +10:00
for ( i = lmb . memory . cnt - 1 ; i > = 0 ; i - - ) {
2008-02-13 16:58:39 -08:00
u64 lmbbase = lmb . memory . region [ i ] . base ;
u64 lmbsize = lmb . memory . region [ i ] . size ;
2005-10-06 12:23:33 +10:00
2008-04-12 15:20:59 +10:00
if ( lmbsize < size )
continue ;
2005-10-06 12:23:33 +10:00
if ( max_addr = = LMB_ALLOC_ANYWHERE )
2008-02-13 16:56:49 -08:00
base = lmb_align_down ( lmbbase + lmbsize - size , align ) ;
2005-10-06 12:23:33 +10:00
else if ( lmbbase < max_addr ) {
base = min ( lmbbase + lmbsize , max_addr ) ;
2008-02-13 16:56:49 -08:00
base = lmb_align_down ( base - size , align ) ;
2005-10-06 12:23:33 +10:00
} else
continue ;
2008-04-12 15:20:59 +10:00
while ( base & & lmbbase < = base ) {
2008-04-12 15:20:59 +10:00
j = lmb_overlaps_region ( & lmb . reserved , base , size ) ;
2008-04-12 15:20:59 +10:00
if ( j < 0 ) {
/* this area isn't reserved, take it */
2008-05-12 16:51:15 -07:00
if ( lmb_add_region ( & lmb . reserved , base , size ) < 0 )
2008-04-12 15:20:59 +10:00
return 0 ;
return base ;
}
res_base = lmb . reserved . region [ j ] . base ;
if ( res_base < size )
2008-04-12 15:20:59 +10:00
break ;
2008-04-12 15:20:59 +10:00
base = lmb_align_down ( res_base - size , align ) ;
2008-04-12 15:20:59 +10:00
}
2005-10-06 12:23:33 +10:00
}
2008-04-12 15:20:59 +10:00
return 0 ;
2005-10-06 12:23:33 +10:00
}
/* You must call lmb_analyze() before this. */
2008-02-13 16:58:39 -08:00
u64 __init lmb_phys_mem_size ( void )
2005-10-06 12:23:33 +10:00
{
return lmb . memory . size ;
}
2009-08-27 17:20:30 +10:00
u64 lmb_end_of_DRAM ( void )
2005-10-06 12:23:33 +10:00
{
int idx = lmb . memory . cnt - 1 ;
return ( lmb . memory . region [ idx ] . base + lmb . memory . region [ idx ] . size ) ;
}
2006-05-17 18:00:46 +10:00
/* You must call lmb_analyze() after this. */
2008-02-13 16:58:39 -08:00
void __init lmb_enforce_memory_limit ( u64 memory_limit )
2005-10-06 12:23:33 +10:00
{
2008-02-13 16:58:39 -08:00
unsigned long i ;
u64 limit ;
2006-05-17 18:00:46 +10:00
struct lmb_property * p ;
2005-10-06 12:23:33 +10:00
2008-04-12 15:20:59 +10:00
if ( ! memory_limit )
2005-10-06 12:23:33 +10:00
return ;
2006-05-17 18:00:46 +10:00
/* Truncate the lmb regions to satisfy the memory limit. */
2005-10-06 12:23:33 +10:00
limit = memory_limit ;
for ( i = 0 ; i < lmb . memory . cnt ; i + + ) {
if ( limit > lmb . memory . region [ i ] . size ) {
limit - = lmb . memory . region [ i ] . size ;
continue ;
}
lmb . memory . region [ i ] . size = limit ;
lmb . memory . cnt = i + 1 ;
break ;
}
2006-05-17 18:00:46 +10:00
2006-07-04 17:13:23 +10:00
if ( lmb . memory . region [ 0 ] . size < lmb . rmo_size )
lmb . rmo_size = lmb . memory . region [ 0 ] . size ;
2006-05-17 18:00:46 +10:00
2008-08-15 19:57:57 -07:00
memory_limit = lmb_end_of_DRAM ( ) ;
2006-05-17 18:00:46 +10:00
/* And truncate any reserves above the limit also. */
for ( i = 0 ; i < lmb . reserved . cnt ; i + + ) {
p = & lmb . reserved . region [ i ] ;
if ( p - > base > memory_limit )
p - > size = 0 ;
else if ( ( p - > base + p - > size ) > memory_limit )
p - > size = memory_limit - p - > base ;
if ( p - > size = = 0 ) {
lmb_remove_region ( & lmb . reserved , i ) ;
i - - ;
}
}
2005-10-06 12:23:33 +10:00
}
2008-01-09 11:27:23 -06:00
2008-02-13 16:58:39 -08:00
int __init lmb_is_reserved ( u64 addr )
2008-01-09 11:27:23 -06:00
{
int i ;
for ( i = 0 ; i < lmb . reserved . cnt ; i + + ) {
2008-02-13 16:58:39 -08:00
u64 upper = lmb . reserved . region [ i ] . base +
lmb . reserved . region [ i ] . size - 1 ;
2008-01-09 11:27:23 -06:00
if ( ( addr > = lmb . reserved . region [ i ] . base ) & & ( addr < = upper ) )
return 1 ;
}
return 0 ;
}
2008-04-18 13:33:53 -07:00
/*
* Given a < base , len > , find which memory regions belong to this range .
* Adjust the request and return a contiguous chunk .
*/
int lmb_find ( struct lmb_property * res )
{
int i ;
u64 rstart , rend ;
rstart = res - > base ;
rend = rstart + res - > size - 1 ;
for ( i = 0 ; i < lmb . memory . cnt ; i + + ) {
u64 start = lmb . memory . region [ i ] . base ;
u64 end = start + lmb . memory . region [ i ] . size - 1 ;
if ( start > rend )
return - 1 ;
if ( ( end > = rstart ) & & ( start < rend ) ) {
/* adjust the request */
if ( rstart < start )
rstart = start ;
if ( rend > end )
rend = end ;
res - > base = rstart ;
res - > size = rend - rstart + 1 ;
return 0 ;
}
}
return - 1 ;
}