2005-04-16 15:20:36 -07:00
/*
* Written by : Patricia Gaughen < gone @ us . ibm . com > , IBM Corporation
* August 2002 : added remote node KVA remap - Martin J . Bligh
*
* Copyright ( C ) 2002 , IBM Corp .
*
* All rights reserved .
*
* 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 .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE , GOOD TITLE or
* NON INFRINGEMENT . See the GNU General Public License for more
* details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include <linux/mm.h>
# include <linux/bootmem.h>
# include <linux/mmzone.h>
# include <linux/highmem.h>
# include <linux/initrd.h>
# include <linux/nodemask.h>
2005-06-23 00:08:33 -07:00
# include <linux/module.h>
2005-06-25 14:58:01 -07:00
# include <linux/kexec.h>
2006-03-27 01:16:04 -08:00
# include <linux/pfn.h>
2007-05-15 18:45:49 -07:00
# include <linux/swap.h>
2008-01-30 13:33:25 +01:00
# include <linux/acpi.h>
2005-06-25 14:58:01 -07:00
2005-04-16 15:20:36 -07:00
# include <asm/e820.h>
# include <asm/setup.h>
# include <asm/mmzone.h>
2008-03-17 22:08:17 +03:00
# include <asm/bios_ebda.h>
2008-06-01 21:06:31 -07:00
# include <asm/proto.h>
2005-04-16 15:20:36 -07:00
2005-09-06 15:17:45 -07:00
struct pglist_data * node_data [ MAX_NUMNODES ] __read_mostly ;
2005-06-23 00:08:33 -07:00
EXPORT_SYMBOL ( node_data ) ;
2005-04-16 15:20:36 -07:00
/*
2006-06-30 18:29:51 +02:00
* numa interface - we expect the numa architecture specific code to have
2005-04-16 15:20:36 -07:00
* populated the following initialisation .
*
* 1 ) node_online_map - the map of all nodes configured ( online ) in the system
2005-06-23 00:07:57 -07:00
* 2 ) node_start_pfn - the starting page frame number for a node
2005-04-16 15:20:36 -07:00
* 3 ) node_end_pfn - the ending page fram number for a node
*/
2005-09-06 15:17:45 -07:00
unsigned long node_start_pfn [ MAX_NUMNODES ] __read_mostly ;
unsigned long node_end_pfn [ MAX_NUMNODES ] __read_mostly ;
2005-06-23 00:07:57 -07:00
2005-04-16 15:20:36 -07:00
2005-06-23 00:07:57 -07:00
# ifdef CONFIG_DISCONTIGMEM
2005-04-16 15:20:36 -07:00
/*
2005-06-23 00:07:57 -07:00
* 4 ) physnode_map - the mapping between a pfn and owning node
2005-04-16 15:20:36 -07:00
* physnode_map keeps track of the physical memory layout of a generic
2008-05-31 22:51:51 -07:00
* numa node on a 64 Mb break ( each element of the array will
* represent 64 Mb of memory and will be marked by the node id . so ,
2005-04-16 15:20:36 -07:00
* if the first gig is on node 0 , and the second gig is on node 1
* physnode_map will contain :
*
2008-05-31 22:51:51 -07:00
* physnode_map [ 0 - 15 ] = 0 ;
* physnode_map [ 16 - 31 ] = 1 ;
* physnode_map [ 32 - ] = - 1 ;
2005-04-16 15:20:36 -07:00
*/
2005-09-06 15:17:45 -07:00
s8 physnode_map [ MAX_ELEMENTS ] __read_mostly = { [ 0 . . . ( MAX_ELEMENTS - 1 ) ] = - 1 } ;
2005-06-23 00:08:33 -07:00
EXPORT_SYMBOL ( physnode_map ) ;
2005-04-16 15:20:36 -07:00
void memory_present ( int nid , unsigned long start , unsigned long end )
{
unsigned long pfn ;
2008-06-23 16:41:30 -07:00
printk ( KERN_INFO " Node: %d, start_pfn: %lx, end_pfn: %lx \n " ,
2005-04-16 15:20:36 -07:00
nid , start , end ) ;
printk ( KERN_DEBUG " Setting physnode_map array to node %d for pfns: \n " , nid ) ;
printk ( KERN_DEBUG " " ) ;
for ( pfn = start ; pfn < end ; pfn + = PAGES_PER_ELEMENT ) {
physnode_map [ pfn / PAGES_PER_ELEMENT ] = nid ;
2008-06-23 16:41:30 -07:00
printk ( KERN_CONT " %lx " , pfn ) ;
2005-04-16 15:20:36 -07:00
}
2008-05-31 22:51:51 -07:00
printk ( KERN_CONT " \n " ) ;
2005-04-16 15:20:36 -07:00
}
unsigned long node_memmap_size_bytes ( int nid , unsigned long start_pfn ,
unsigned long end_pfn )
{
unsigned long nr_pages = end_pfn - start_pfn ;
if ( ! nr_pages )
return 0 ;
return ( nr_pages + 1 ) * sizeof ( struct page ) ;
}
2005-06-23 00:07:57 -07:00
# endif
2005-04-16 15:20:36 -07:00
extern unsigned long find_max_low_pfn ( void ) ;
extern unsigned long highend_pfn , highstart_pfn ;
# define LARGE_PAGE_BYTES (PTRS_PER_PTE * PAGE_SIZE)
unsigned long node_remap_size [ MAX_NUMNODES ] ;
2007-10-17 18:04:36 +02:00
static void * node_remap_start_vaddr [ MAX_NUMNODES ] ;
2005-04-16 15:20:36 -07:00
void set_pmd_pfn ( unsigned long vaddr , unsigned long pfn , pgprot_t flags ) ;
2006-09-25 23:31:03 -07:00
static unsigned long kva_start_pfn ;
static unsigned long kva_pages ;
2005-04-16 15:20:36 -07:00
/*
* FLAT - support for basic PC memory model with discontig enabled , essentially
* a single node with all available processors in it with a flat
* memory map .
*/
int __init get_memcfg_numa_flat ( void )
{
2008-06-23 16:41:30 -07:00
printk ( KERN_DEBUG " NUMA - single node, flat memory mode \n " ) ;
2005-04-16 15:20:36 -07:00
node_start_pfn [ 0 ] = 0 ;
node_end_pfn [ 0 ] = max_pfn ;
2008-06-03 19:35:04 -07:00
e820_register_active_regions ( 0 , 0 , max_pfn ) ;
2005-04-16 15:20:36 -07:00
memory_present ( 0 , 0 , max_pfn ) ;
2008-05-31 22:53:47 -07:00
node_remap_size [ 0 ] = node_memmap_size_bytes ( 0 , 0 , max_pfn ) ;
2005-04-16 15:20:36 -07:00
/* Indicate there is one node available. */
nodes_clear ( node_online_map ) ;
node_set_online ( 0 ) ;
return 1 ;
}
/*
* Find the highest page frame number we have available for the node
*/
2008-04-16 02:29:42 +02:00
static void __init propagate_e820_map_node ( int nid )
2005-04-16 15:20:36 -07:00
{
if ( node_end_pfn [ nid ] > max_pfn )
node_end_pfn [ nid ] = max_pfn ;
/*
* if a user has given mem = XXXX , then we need to make sure
* that the node _starts_ before that , too , not just ends
*/
if ( node_start_pfn [ nid ] > max_pfn )
node_start_pfn [ nid ] = max_pfn ;
2006-10-03 23:34:58 +02:00
BUG_ON ( node_start_pfn [ nid ] > node_end_pfn [ nid ] ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Allocate memory for the pg_data_t for this node via a crude pre - bootmem
* method . For node zero take this from the bottom of memory , for
* subsequent nodes place them at node_remap_start_vaddr which contains
* node local data in physically node local memory . See setup_memory ( )
* for details .
*/
static void __init allocate_pgdat ( int nid )
{
2008-06-30 18:34:58 -07:00
char buf [ 16 ] ;
if ( node_has_online_mem ( nid ) & & node_remap_start_vaddr [ nid ] )
2005-04-16 15:20:36 -07:00
NODE_DATA ( nid ) = ( pg_data_t * ) node_remap_start_vaddr [ nid ] ;
else {
2008-05-29 12:57:22 -07:00
unsigned long pgdat_phys ;
pgdat_phys = find_e820_area ( min_low_pfn < < PAGE_SHIFT ,
2008-06-30 18:34:58 -07:00
max_pfn_mapped < < PAGE_SHIFT ,
2008-06-01 23:53:50 -07:00
sizeof ( pg_data_t ) ,
2008-05-29 12:57:22 -07:00
PAGE_SIZE ) ;
NODE_DATA ( nid ) = ( pg_data_t * ) ( pfn_to_kaddr ( pgdat_phys > > PAGE_SHIFT ) ) ;
2008-06-30 18:34:58 -07:00
memset ( buf , 0 , sizeof ( buf ) ) ;
sprintf ( buf , " NODE_DATA %d " , nid ) ;
reserve_early ( pgdat_phys , pgdat_phys + sizeof ( pg_data_t ) , buf ) ;
2005-04-16 15:20:36 -07:00
}
2008-06-01 13:15:22 -07:00
printk ( KERN_DEBUG " allocate_pgdat: node %d NODE_DATA %08lx \n " ,
nid , ( unsigned long ) NODE_DATA ( nid ) ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-30 13:33:25 +01:00
/*
2008-05-20 11:01:08 +01:00
* In the DISCONTIGMEM and SPARSEMEM memory model , a portion of the kernel
* virtual address space ( KVA ) is reserved and portions of nodes are mapped
* using it . This is to allow node - local memory to be allocated for
* structures that would normally require ZONE_NORMAL . The memory is
* allocated with alloc_remap ( ) and callers should be prepared to allocate
* from the bootmem allocator instead .
2008-01-30 13:33:25 +01:00
*/
static unsigned long node_remap_start_pfn [ MAX_NUMNODES ] ;
static void * node_remap_end_vaddr [ MAX_NUMNODES ] ;
static void * node_remap_alloc_vaddr [ MAX_NUMNODES ] ;
static unsigned long node_remap_offset [ MAX_NUMNODES ] ;
2005-06-23 00:07:39 -07:00
void * alloc_remap ( int nid , unsigned long size )
{
void * allocation = node_remap_alloc_vaddr [ nid ] ;
size = ALIGN ( size , L1_CACHE_BYTES ) ;
if ( ! allocation | | ( allocation + size ) > = node_remap_end_vaddr [ nid ] )
return 0 ;
node_remap_alloc_vaddr [ nid ] + = size ;
memset ( allocation , 0 , size ) ;
return allocation ;
}
2008-06-24 12:19:41 -07:00
static void __init remap_numa_kva ( void )
2005-04-16 15:20:36 -07:00
{
void * vaddr ;
unsigned long pfn ;
int node ;
for_each_online_node ( node ) {
2008-06-01 13:15:22 -07:00
printk ( KERN_DEBUG " remap_numa_kva: node %d \n " , node ) ;
2005-04-16 15:20:36 -07:00
for ( pfn = 0 ; pfn < node_remap_size [ node ] ; pfn + = PTRS_PER_PTE ) {
vaddr = node_remap_start_vaddr [ node ] + ( pfn < < PAGE_SHIFT ) ;
2008-06-01 13:15:22 -07:00
printk ( KERN_DEBUG " remap_numa_kva: %08lx to pfn %08lx \n " ,
( unsigned long ) vaddr ,
node_remap_start_pfn [ node ] + pfn ) ;
2005-04-16 15:20:36 -07:00
set_pmd_pfn ( ( ulong ) vaddr ,
node_remap_start_pfn [ node ] + pfn ,
PAGE_KERNEL_LARGE ) ;
}
}
}
2008-11-12 23:22:35 +01:00
# ifdef CONFIG_HIBERNATION
/**
* resume_map_numa_kva - add KVA mapping to the temporary page tables created
* during resume from hibernation
* @ pgd_base - temporary resume page directory
*/
void resume_map_numa_kva ( pgd_t * pgd_base )
{
int node ;
for_each_online_node ( node ) {
unsigned long start_va , start_pfn , size , pfn ;
start_va = ( unsigned long ) node_remap_start_vaddr [ node ] ;
start_pfn = node_remap_start_pfn [ node ] ;
size = node_remap_size [ node ] ;
printk ( KERN_DEBUG " %s: node %d \n " , __FUNCTION__ , node ) ;
for ( pfn = 0 ; pfn < size ; pfn + = PTRS_PER_PTE ) {
unsigned long vaddr = start_va + ( pfn < < PAGE_SHIFT ) ;
pgd_t * pgd = pgd_base + pgd_index ( vaddr ) ;
pud_t * pud = pud_offset ( pgd , vaddr ) ;
pmd_t * pmd = pmd_offset ( pud , vaddr ) ;
set_pmd ( pmd , pfn_pmd ( start_pfn + pfn ,
PAGE_KERNEL_LARGE_EXEC ) ) ;
printk ( KERN_DEBUG " %s: %08lx -> pfn %08lx \n " ,
__FUNCTION__ , vaddr , start_pfn + pfn ) ;
}
}
}
# endif
2005-04-16 15:20:36 -07:00
static unsigned long calculate_numa_remap_pages ( void )
{
int nid ;
unsigned long size , reserve_pages = 0 ;
for_each_online_node ( nid ) {
2008-06-13 19:08:52 -07:00
u64 node_kva_target ;
u64 node_kva_final ;
2006-09-27 01:49:51 -07:00
2005-04-16 15:20:36 -07:00
/*
* The acpi / srat node info can show hot - add memroy zones
* where memory could be added but not currently present .
*/
2008-06-23 16:41:30 -07:00
printk ( KERN_DEBUG " node %d pfn: [%lx - %lx] \n " ,
2008-06-06 18:53:33 -07:00
nid , node_start_pfn [ nid ] , node_end_pfn [ nid ] ) ;
2005-04-16 15:20:36 -07:00
if ( node_start_pfn [ nid ] > max_pfn )
continue ;
2008-06-06 18:53:33 -07:00
if ( ! node_end_pfn [ nid ] )
continue ;
2005-04-16 15:20:36 -07:00
if ( node_end_pfn [ nid ] > max_pfn )
node_end_pfn [ nid ] = max_pfn ;
/* ensure the remap includes space for the pgdat. */
size = node_remap_size [ nid ] + sizeof ( pg_data_t ) ;
/* convert size to large (pmd size) pages, rounding up */
size = ( size + LARGE_PAGE_BYTES - 1 ) / LARGE_PAGE_BYTES ;
/* now the roundup is correct, convert to PAGE_SIZE pages */
size = size * PTRS_PER_PTE ;
2005-06-23 00:07:41 -07:00
2008-06-13 19:08:52 -07:00
node_kva_target = round_down ( node_end_pfn [ nid ] - size ,
2008-06-06 18:53:33 -07:00
PTRS_PER_PTE ) ;
2008-06-13 19:08:52 -07:00
node_kva_target < < = PAGE_SHIFT ;
2008-06-06 18:53:33 -07:00
do {
2008-06-13 19:08:52 -07:00
node_kva_final = find_e820_area ( node_kva_target ,
2008-06-06 18:53:33 -07:00
( ( u64 ) node_end_pfn [ nid ] ) < < PAGE_SHIFT ,
( ( u64 ) size ) < < PAGE_SHIFT ,
LARGE_PAGE_BYTES ) ;
2008-06-13 19:08:52 -07:00
node_kva_target - = LARGE_PAGE_BYTES ;
} while ( node_kva_final = = - 1ULL & &
( node_kva_target > > PAGE_SHIFT ) > ( node_start_pfn [ nid ] ) ) ;
2008-06-06 18:53:33 -07:00
2008-06-13 19:08:52 -07:00
if ( node_kva_final = = - 1ULL )
2008-06-06 18:53:33 -07:00
panic ( " Can not get kva ram \n " ) ;
2005-06-23 00:07:41 -07:00
2005-04-16 15:20:36 -07:00
node_remap_size [ nid ] = size ;
node_remap_offset [ nid ] = reserve_pages ;
2005-06-23 00:07:39 -07:00
reserve_pages + = size ;
2008-06-23 16:41:30 -07:00
printk ( KERN_DEBUG " Reserving %ld pages of KVA for lmem_map of "
" node %d at %llx \n " ,
2008-06-13 19:08:52 -07:00
size , nid , node_kva_final > > PAGE_SHIFT ) ;
2008-06-06 18:53:33 -07:00
/*
* prevent kva address below max_low_pfn want it on system
* with less memory later .
* layout will be : KVA address , KVA RAM
2008-06-13 19:08:52 -07:00
*
* we are supposed to only record the one less then max_low_pfn
* but we could have some hole in high memory , and it will only
* check page_is_ram ( pfn ) & & ! page_is_reserved_early ( pfn ) to decide
* to use it as free .
* So reserve_early here , hope we don ' t run out of that array
2008-06-06 18:53:33 -07:00
*/
2008-06-13 19:08:52 -07:00
reserve_early ( node_kva_final ,
node_kva_final + ( ( ( u64 ) size ) < < PAGE_SHIFT ) ,
" KVA RAM " ) ;
2005-08-07 09:42:50 -07:00
2008-06-13 19:08:52 -07:00
node_remap_start_pfn [ nid ] = node_kva_final > > PAGE_SHIFT ;
remove_active_range ( nid , node_remap_start_pfn [ nid ] ,
node_remap_start_pfn [ nid ] + size ) ;
2005-04-16 15:20:36 -07:00
}
2008-06-23 16:41:30 -07:00
printk ( KERN_INFO " Reserving total of %lx pages for numa KVA remap \n " ,
2005-04-16 15:20:36 -07:00
reserve_pages ) ;
return reserve_pages ;
}
2008-01-30 13:33:25 +01:00
static void init_remap_allocator ( int nid )
{
node_remap_start_vaddr [ nid ] = pfn_to_kaddr (
kva_start_pfn + node_remap_offset [ nid ] ) ;
node_remap_end_vaddr [ nid ] = node_remap_start_vaddr [ nid ] +
( node_remap_size [ nid ] * PAGE_SIZE ) ;
node_remap_alloc_vaddr [ nid ] = node_remap_start_vaddr [ nid ] +
ALIGN ( sizeof ( pg_data_t ) , PAGE_SIZE ) ;
2008-06-23 16:41:30 -07:00
printk ( KERN_DEBUG " node %d will remap to vaddr %08lx - %08lx \n " , nid ,
2008-01-30 13:33:25 +01:00
( ulong ) node_remap_start_vaddr [ nid ] ,
2008-06-01 13:15:22 -07:00
( ulong ) node_remap_end_vaddr [ nid ] ) ;
2008-01-30 13:33:25 +01:00
}
2008-06-23 03:05:30 -07:00
void __init initmem_init ( unsigned long start_pfn ,
2008-06-22 02:45:39 -07:00
unsigned long end_pfn )
2005-04-16 15:20:36 -07:00
{
int nid ;
2008-06-03 19:32:30 -07:00
long kva_target_pfn ;
2005-04-16 15:20:36 -07:00
/*
* When mapping a NUMA machine we allocate the node_mem_map arrays
* from node local memory . They are then mapped directly into KVA
* between zone normal and vmalloc space . Calculate the size of
2007-10-20 01:13:56 +02:00
* this space and use it to adjust the boundary between ZONE_NORMAL
2005-04-16 15:20:36 -07:00
* and ZONE_HIGHMEM .
*/
2008-06-03 19:35:04 -07:00
2005-04-16 15:20:36 -07:00
get_memcfg_numa ( ) ;
2008-07-25 16:48:59 +02:00
kva_pages = roundup ( calculate_numa_remap_pages ( ) , PTRS_PER_PTE ) ;
2005-04-16 15:20:36 -07:00
2008-06-03 19:32:30 -07:00
kva_target_pfn = round_down ( max_low_pfn - kva_pages , PTRS_PER_PTE ) ;
do {
kva_start_pfn = find_e820_area ( kva_target_pfn < < PAGE_SHIFT ,
max_low_pfn < < PAGE_SHIFT ,
kva_pages < < PAGE_SHIFT ,
PTRS_PER_PTE < < PAGE_SHIFT ) > > PAGE_SHIFT ;
kva_target_pfn - = PTRS_PER_PTE ;
} while ( kva_start_pfn = = - 1UL & & kva_target_pfn > min_low_pfn ) ;
2008-01-30 13:32:54 +01:00
2008-06-03 19:32:30 -07:00
if ( kva_start_pfn = = - 1UL )
panic ( " Can not get kva space \n " ) ;
2006-09-25 23:31:03 -07:00
2008-06-23 16:41:30 -07:00
printk ( KERN_INFO " kva_start_pfn ~ %lx max_low_pfn ~ %lx \n " ,
2006-09-25 23:31:03 -07:00
kva_start_pfn , max_low_pfn ) ;
2008-06-23 16:41:30 -07:00
printk ( KERN_INFO " max_pfn = %lx \n " , max_pfn ) ;
2008-05-29 12:58:37 -07:00
/* avoid clash with initrd */
reserve_early ( kva_start_pfn < < PAGE_SHIFT ,
( kva_start_pfn + kva_pages ) < < PAGE_SHIFT ,
" KVA PG " ) ;
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_HIGHMEM
highstart_pfn = highend_pfn = max_pfn ;
2008-06-23 03:05:30 -07:00
if ( max_pfn > max_low_pfn )
highstart_pfn = max_low_pfn ;
2005-04-16 15:20:36 -07:00
printk ( KERN_NOTICE " %ldMB HIGHMEM available. \n " ,
pages_to_mb ( highend_pfn - highstart_pfn ) ) ;
2006-09-26 10:52:31 +02:00
num_physpages = highend_pfn ;
high_memory = ( void * ) __va ( highstart_pfn * PAGE_SIZE - 1 ) + 1 ;
# else
2008-06-23 03:05:30 -07:00
num_physpages = max_low_pfn ;
high_memory = ( void * ) __va ( max_low_pfn * PAGE_SIZE - 1 ) + 1 ;
2005-04-16 15:20:36 -07:00
# endif
printk ( KERN_NOTICE " %ldMB LOWMEM available. \n " ,
2008-06-23 03:05:30 -07:00
pages_to_mb ( max_low_pfn ) ) ;
2008-06-23 16:41:30 -07:00
printk ( KERN_DEBUG " max_low_pfn = %lx, highstart_pfn = %lx \n " ,
max_low_pfn , highstart_pfn ) ;
2005-04-16 15:20:36 -07:00
2008-06-23 16:41:30 -07:00
printk ( KERN_DEBUG " Low memory ends at vaddr %08lx \n " ,
2005-04-16 15:20:36 -07:00
( ulong ) pfn_to_kaddr ( max_low_pfn ) ) ;
for_each_online_node ( nid ) {
2008-01-30 13:33:25 +01:00
init_remap_allocator ( nid ) ;
2005-06-23 00:07:39 -07:00
2005-04-16 15:20:36 -07:00
allocate_pgdat ( nid ) ;
}
2008-06-24 12:19:41 -07:00
remap_numa_kva ( ) ;
2008-06-23 16:41:30 -07:00
printk ( KERN_DEBUG " High memory starts at vaddr %08lx \n " ,
2005-04-16 15:20:36 -07:00
( ulong ) pfn_to_kaddr ( highstart_pfn ) ) ;
for_each_online_node ( nid )
2008-04-16 02:29:42 +02:00
propagate_e820_map_node ( nid ) ;
2005-04-16 15:20:36 -07:00
2008-06-24 12:19:41 -07:00
for_each_online_node ( nid )
memset ( NODE_DATA ( nid ) , 0 , sizeof ( struct pglist_data ) ) ;
2008-07-23 21:26:55 -07:00
NODE_DATA ( 0 ) - > bdata = & bootmem_node_data [ 0 ] ;
2005-04-16 15:20:36 -07:00
setup_bootmem_allocator ( ) ;
}
2008-06-16 16:11:08 -07:00
void __init set_highmem_pages_init ( void )
2005-04-16 15:20:36 -07:00
{
# ifdef CONFIG_HIGHMEM
struct zone * zone ;
2008-06-14 18:32:52 -07:00
int nid ;
2005-04-16 15:20:36 -07:00
for_each_zone ( zone ) {
2008-06-14 18:32:52 -07:00
unsigned long zone_start_pfn , zone_end_pfn ;
2005-06-23 00:07:57 -07:00
2005-04-16 15:20:36 -07:00
if ( ! is_highmem ( zone ) )
continue ;
zone_start_pfn = zone - > zone_start_pfn ;
2005-06-23 00:07:57 -07:00
zone_end_pfn = zone_start_pfn + zone - > spanned_pages ;
2008-06-14 18:32:52 -07:00
nid = zone_to_nid ( zone ) ;
2008-06-23 16:41:30 -07:00
printk ( KERN_INFO " Initializing %s for node %d (%08lx:%08lx) \n " ,
2008-06-14 18:32:52 -07:00
zone - > name , nid , zone_start_pfn , zone_end_pfn ) ;
add_highpages_with_active_regions ( nid , zone_start_pfn ,
2008-06-16 16:11:08 -07:00
zone_end_pfn ) ;
2005-04-16 15:20:36 -07:00
}
totalram_pages + = totalhigh_pages ;
# endif
}
2006-12-22 01:11:13 -08:00
# ifdef CONFIG_MEMORY_HOTPLUG
2007-10-24 18:24:47 +02:00
static int paddr_to_nid ( u64 addr )
2006-12-22 01:11:13 -08:00
{
int nid ;
unsigned long pfn = PFN_DOWN ( addr ) ;
for_each_node ( nid )
if ( node_start_pfn [ nid ] < = pfn & &
pfn < node_end_pfn [ nid ] )
return nid ;
return - 1 ;
}
/*
* This function is used to ask node id BEFORE memmap and mem_section ' s
* initialization ( pfn_to_nid ( ) can ' t be used yet ) .
* If _PXM is not defined on ACPI ' s DSDT , node id must be found by this .
*/
int memory_add_physaddr_to_nid ( u64 addr )
{
int nid = paddr_to_nid ( addr ) ;
return ( nid > = 0 ) ? nid : 0 ;
}
EXPORT_SYMBOL_GPL ( memory_add_physaddr_to_nid ) ;
# endif
2008-06-03 10:09:45 +02:00