diff options
Diffstat (limited to 'mm/hugetlb.c')
-rw-r--r-- | mm/hugetlb.c | 10 |
1 files changed, 7 insertions, 3 deletions
diff --git a/mm/hugetlb.c b/mm/hugetlb.c index 15fc7b000772..2d7611cf276a 100644 --- a/mm/hugetlb.c +++ b/mm/hugetlb.c | |||
@@ -104,15 +104,19 @@ static int alloc_fresh_huge_page(void) | |||
104 | { | 104 | { |
105 | static int prev_nid; | 105 | static int prev_nid; |
106 | struct page *page; | 106 | struct page *page; |
107 | static DEFINE_SPINLOCK(nid_lock); | ||
108 | int nid; | 107 | int nid; |
109 | 108 | ||
110 | spin_lock(&nid_lock); | 109 | /* |
110 | * Copy static prev_nid to local nid, work on that, then copy it | ||
111 | * back to prev_nid afterwards: otherwise there's a window in which | ||
112 | * a racer might pass invalid nid MAX_NUMNODES to alloc_pages_node. | ||
113 | * But we don't need to use a spin_lock here: it really doesn't | ||
114 | * matter if occasionally a racer chooses the same nid as we do. | ||
115 | */ | ||
111 | nid = next_node(prev_nid, node_online_map); | 116 | nid = next_node(prev_nid, node_online_map); |
112 | if (nid == MAX_NUMNODES) | 117 | if (nid == MAX_NUMNODES) |
113 | nid = first_node(node_online_map); | 118 | nid = first_node(node_online_map); |
114 | prev_nid = nid; | 119 | prev_nid = nid; |
115 | spin_unlock(&nid_lock); | ||
116 | 120 | ||
117 | page = alloc_pages_node(nid, htlb_alloc_mask|__GFP_COMP|__GFP_NOWARN, | 121 | page = alloc_pages_node(nid, htlb_alloc_mask|__GFP_COMP|__GFP_NOWARN, |
118 | HUGETLB_PAGE_ORDER); | 122 | HUGETLB_PAGE_ORDER); |