diff options
Diffstat (limited to 'mm/memblock.c')
-rw-r--r-- | mm/memblock.c | 36 |
1 files changed, 32 insertions, 4 deletions
diff --git a/mm/memblock.c b/mm/memblock.c index b65b687e2362..d4382095f8bd 100644 --- a/mm/memblock.c +++ b/mm/memblock.c | |||
@@ -184,7 +184,24 @@ static void __init_memblock memblock_remove_region(struct memblock_type *type, u | |||
184 | } | 184 | } |
185 | } | 185 | } |
186 | 186 | ||
187 | static int __init_memblock memblock_double_array(struct memblock_type *type) | 187 | /** |
188 | * memblock_double_array - double the size of the memblock regions array | ||
189 | * @type: memblock type of the regions array being doubled | ||
190 | * @new_area_start: starting address of memory range to avoid overlap with | ||
191 | * @new_area_size: size of memory range to avoid overlap with | ||
192 | * | ||
193 | * Double the size of the @type regions array. If memblock is being used to | ||
194 | * allocate memory for a new reserved regions array and there is a previously | ||
195 | * allocated memory range [@new_area_start,@new_area_start+@new_area_size] | ||
196 | * waiting to be reserved, ensure the memory used by the new array does | ||
197 | * not overlap. | ||
198 | * | ||
199 | * RETURNS: | ||
200 | * 0 on success, -1 on failure. | ||
201 | */ | ||
202 | static int __init_memblock memblock_double_array(struct memblock_type *type, | ||
203 | phys_addr_t new_area_start, | ||
204 | phys_addr_t new_area_size) | ||
188 | { | 205 | { |
189 | struct memblock_region *new_array, *old_array; | 206 | struct memblock_region *new_array, *old_array; |
190 | phys_addr_t old_size, new_size, addr; | 207 | phys_addr_t old_size, new_size, addr; |
@@ -222,7 +239,18 @@ static int __init_memblock memblock_double_array(struct memblock_type *type) | |||
222 | new_array = kmalloc(new_size, GFP_KERNEL); | 239 | new_array = kmalloc(new_size, GFP_KERNEL); |
223 | addr = new_array ? __pa(new_array) : 0; | 240 | addr = new_array ? __pa(new_array) : 0; |
224 | } else { | 241 | } else { |
225 | addr = memblock_find_in_range(0, MEMBLOCK_ALLOC_ACCESSIBLE, new_size, sizeof(phys_addr_t)); | 242 | /* only exclude range when trying to double reserved.regions */ |
243 | if (type != &memblock.reserved) | ||
244 | new_area_start = new_area_size = 0; | ||
245 | |||
246 | addr = memblock_find_in_range(new_area_start + new_area_size, | ||
247 | memblock.current_limit, | ||
248 | new_size, sizeof(phys_addr_t)); | ||
249 | if (!addr && new_area_size) | ||
250 | addr = memblock_find_in_range(0, | ||
251 | min(new_area_start, memblock.current_limit), | ||
252 | new_size, sizeof(phys_addr_t)); | ||
253 | |||
226 | new_array = addr ? __va(addr) : 0; | 254 | new_array = addr ? __va(addr) : 0; |
227 | } | 255 | } |
228 | if (!addr) { | 256 | if (!addr) { |
@@ -399,7 +427,7 @@ repeat: | |||
399 | */ | 427 | */ |
400 | if (!insert) { | 428 | if (!insert) { |
401 | while (type->cnt + nr_new > type->max) | 429 | while (type->cnt + nr_new > type->max) |
402 | if (memblock_double_array(type) < 0) | 430 | if (memblock_double_array(type, obase, size) < 0) |
403 | return -ENOMEM; | 431 | return -ENOMEM; |
404 | insert = true; | 432 | insert = true; |
405 | goto repeat; | 433 | goto repeat; |
@@ -450,7 +478,7 @@ static int __init_memblock memblock_isolate_range(struct memblock_type *type, | |||
450 | 478 | ||
451 | /* we'll create at most two more regions */ | 479 | /* we'll create at most two more regions */ |
452 | while (type->cnt + 2 > type->max) | 480 | while (type->cnt + 2 > type->max) |
453 | if (memblock_double_array(type) < 0) | 481 | if (memblock_double_array(type, base, size) < 0) |
454 | return -ENOMEM; | 482 | return -ENOMEM; |
455 | 483 | ||
456 | for (i = 0; i < type->cnt; i++) { | 484 | for (i = 0; i < type->cnt; i++) { |