diff options
author | Dave Hansen <haveblue@us.ibm.com> | 2005-10-29 21:16:52 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2005-10-30 00:40:44 -0400 |
commit | 208d54e5513c0c02d85af0990901354c74364d5c (patch) | |
tree | 83922f1d4a83f19bffcbff299044f421bd7e9c73 /include | |
parent | c6a57e19e464db118dc4ab9cfe9e9748c6d630a0 (diff) |
[PATCH] memory hotplug locking: node_size_lock
pgdat->node_size_lock is basically only neeeded in one place in the normal
code: show_mem(), which is the arch-specific sysrq-m printing function.
Strictly speaking, the architectures not doing memory hotplug do no need this
locking in show_mem(). However, they are all included for completeness. This
should also make any future consolidation of all of the implementations a
little more straightforward.
This lock is also held in the sparsemem code during a memory removal, as
sections are invalidated. This is the place there pfn_valid() is made false
for a memory area that's being removed. The lock is only required when doing
pfn_valid() operations on memory which the user does not already have a
reference on the page, such as in show_mem().
Signed-off-by: Dave Hansen <haveblue@us.ibm.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'include')
-rw-r--r-- | include/linux/memory_hotplug.h | 34 | ||||
-rw-r--r-- | include/linux/mmzone.h | 12 |
2 files changed, 46 insertions, 0 deletions
diff --git a/include/linux/memory_hotplug.h b/include/linux/memory_hotplug.h new file mode 100644 index 000000000000..e8103be9d528 --- /dev/null +++ b/include/linux/memory_hotplug.h | |||
@@ -0,0 +1,34 @@ | |||
1 | #ifndef __LINUX_MEMORY_HOTPLUG_H | ||
2 | #define __LINUX_MEMORY_HOTPLUG_H | ||
3 | |||
4 | #include <linux/mmzone.h> | ||
5 | #include <linux/spinlock.h> | ||
6 | |||
7 | #ifdef CONFIG_MEMORY_HOTPLUG | ||
8 | /* | ||
9 | * pgdat resizing functions | ||
10 | */ | ||
11 | static inline | ||
12 | void pgdat_resize_lock(struct pglist_data *pgdat, unsigned long *flags) | ||
13 | { | ||
14 | spin_lock_irqsave(&pgdat->node_size_lock, *flags); | ||
15 | } | ||
16 | static inline | ||
17 | void pgdat_resize_unlock(struct pglist_data *pgdat, unsigned long *flags) | ||
18 | { | ||
19 | spin_lock_irqrestore(&pgdat->node_size_lock, *flags); | ||
20 | } | ||
21 | static inline | ||
22 | void pgdat_resize_init(struct pglist_data *pgdat) | ||
23 | { | ||
24 | spin_lock_init(&pgdat->node_size_lock); | ||
25 | } | ||
26 | #else /* ! CONFIG_MEMORY_HOTPLUG */ | ||
27 | /* | ||
28 | * Stub functions for when hotplug is off | ||
29 | */ | ||
30 | static inline void pgdat_resize_lock(struct pglist_data *p, unsigned long *f) {} | ||
31 | static inline void pgdat_resize_unlock(struct pglist_data *p, unsigned long *f) {} | ||
32 | static inline void pgdat_resize_init(struct pglist_data *pgdat) {} | ||
33 | #endif | ||
34 | #endif /* __LINUX_MEMORY_HOTPLUG_H */ | ||
diff --git a/include/linux/mmzone.h b/include/linux/mmzone.h index 4674145bb63d..e050d68963a1 100644 --- a/include/linux/mmzone.h +++ b/include/linux/mmzone.h | |||
@@ -273,6 +273,16 @@ typedef struct pglist_data { | |||
273 | struct page *node_mem_map; | 273 | struct page *node_mem_map; |
274 | #endif | 274 | #endif |
275 | struct bootmem_data *bdata; | 275 | struct bootmem_data *bdata; |
276 | #ifdef CONFIG_MEMORY_HOTPLUG | ||
277 | /* | ||
278 | * Must be held any time you expect node_start_pfn, node_present_pages | ||
279 | * or node_spanned_pages stay constant. Holding this will also | ||
280 | * guarantee that any pfn_valid() stays that way. | ||
281 | * | ||
282 | * Nests above zone->lock and zone->size_seqlock. | ||
283 | */ | ||
284 | spinlock_t node_size_lock; | ||
285 | #endif | ||
276 | unsigned long node_start_pfn; | 286 | unsigned long node_start_pfn; |
277 | unsigned long node_present_pages; /* total number of physical pages */ | 287 | unsigned long node_present_pages; /* total number of physical pages */ |
278 | unsigned long node_spanned_pages; /* total size of physical page | 288 | unsigned long node_spanned_pages; /* total size of physical page |
@@ -293,6 +303,8 @@ typedef struct pglist_data { | |||
293 | #endif | 303 | #endif |
294 | #define nid_page_nr(nid, pagenr) pgdat_page_nr(NODE_DATA(nid),(pagenr)) | 304 | #define nid_page_nr(nid, pagenr) pgdat_page_nr(NODE_DATA(nid),(pagenr)) |
295 | 305 | ||
306 | #include <linux/memory_hotplug.h> | ||
307 | |||
296 | extern struct pglist_data *pgdat_list; | 308 | extern struct pglist_data *pgdat_list; |
297 | 309 | ||
298 | void __get_zone_counts(unsigned long *active, unsigned long *inactive, | 310 | void __get_zone_counts(unsigned long *active, unsigned long *inactive, |