diff options
Diffstat (limited to 'lib')
-rw-r--r-- | lib/lmb.c | 56 |
1 files changed, 30 insertions, 26 deletions
@@ -230,20 +230,23 @@ static u64 lmb_align_up(u64 addr, u64 size) | |||
230 | static u64 __init lmb_alloc_nid_unreserved(u64 start, u64 end, | 230 | static u64 __init lmb_alloc_nid_unreserved(u64 start, u64 end, |
231 | u64 size, u64 align) | 231 | u64 size, u64 align) |
232 | { | 232 | { |
233 | u64 base; | 233 | u64 base, res_base; |
234 | long j; | 234 | long j; |
235 | 235 | ||
236 | base = lmb_align_down((end - size), align); | 236 | base = lmb_align_down((end - size), align); |
237 | while (start <= base && | 237 | while (start <= base) { |
238 | ((j = lmb_overlaps_region(&lmb.reserved, base, size)) >= 0)) | 238 | j = lmb_overlaps_region(&lmb.reserved, base, size); |
239 | base = lmb_align_down(lmb.reserved.region[j].base - size, | 239 | if (j < 0) { |
240 | align); | 240 | /* this area isn't reserved, take it */ |
241 | 241 | if (lmb_add_region(&lmb.reserved, base, | |
242 | if (base != 0 && start <= base) { | 242 | lmb_align_up(size, align)) < 0) |
243 | if (lmb_add_region(&lmb.reserved, base, | 243 | base = ~(u64)0; |
244 | lmb_align_up(size, align)) < 0) | 244 | return base; |
245 | base = ~(u64)0; | 245 | } |
246 | return base; | 246 | res_base = lmb.reserved.region[j].base; |
247 | if (res_base < size) | ||
248 | break; | ||
249 | base = lmb_align_down(res_base - size, align); | ||
247 | } | 250 | } |
248 | 251 | ||
249 | return ~(u64)0; | 252 | return ~(u64)0; |
@@ -315,10 +318,12 @@ u64 __init __lmb_alloc_base(u64 size, u64 align, u64 max_addr) | |||
315 | { | 318 | { |
316 | long i, j; | 319 | long i, j; |
317 | u64 base = 0; | 320 | u64 base = 0; |
321 | u64 res_base; | ||
318 | 322 | ||
319 | BUG_ON(0 == size); | 323 | BUG_ON(0 == size); |
320 | 324 | ||
321 | /* On some platforms, make sure we allocate lowmem */ | 325 | /* On some platforms, make sure we allocate lowmem */ |
326 | /* Note that LMB_REAL_LIMIT may be LMB_ALLOC_ANYWHERE */ | ||
322 | if (max_addr == LMB_ALLOC_ANYWHERE) | 327 | if (max_addr == LMB_ALLOC_ANYWHERE) |
323 | max_addr = LMB_REAL_LIMIT; | 328 | max_addr = LMB_REAL_LIMIT; |
324 | 329 | ||
@@ -326,6 +331,8 @@ u64 __init __lmb_alloc_base(u64 size, u64 align, u64 max_addr) | |||
326 | u64 lmbbase = lmb.memory.region[i].base; | 331 | u64 lmbbase = lmb.memory.region[i].base; |
327 | u64 lmbsize = lmb.memory.region[i].size; | 332 | u64 lmbsize = lmb.memory.region[i].size; |
328 | 333 | ||
334 | if (lmbsize < size) | ||
335 | continue; | ||
329 | if (max_addr == LMB_ALLOC_ANYWHERE) | 336 | if (max_addr == LMB_ALLOC_ANYWHERE) |
330 | base = lmb_align_down(lmbbase + lmbsize - size, align); | 337 | base = lmb_align_down(lmbbase + lmbsize - size, align); |
331 | else if (lmbbase < max_addr) { | 338 | else if (lmbbase < max_addr) { |
@@ -334,25 +341,22 @@ u64 __init __lmb_alloc_base(u64 size, u64 align, u64 max_addr) | |||
334 | } else | 341 | } else |
335 | continue; | 342 | continue; |
336 | 343 | ||
337 | while (lmbbase <= base) { | 344 | while (base && lmbbase <= base) { |
338 | j = lmb_overlaps_region(&lmb.reserved, base, size); | 345 | j = lmb_overlaps_region(&lmb.reserved, base, size); |
339 | if (j < 0) | 346 | if (j < 0) { |
347 | /* this area isn't reserved, take it */ | ||
348 | if (lmb_add_region(&lmb.reserved, base, | ||
349 | size) < 0) | ||
350 | return 0; | ||
351 | return base; | ||
352 | } | ||
353 | res_base = lmb.reserved.region[j].base; | ||
354 | if (res_base < size) | ||
340 | break; | 355 | break; |
341 | base = lmb_align_down(lmb.reserved.region[j].base - size, | 356 | base = lmb_align_down(res_base - size, align); |
342 | align); | ||
343 | } | 357 | } |
344 | |||
345 | if ((base != 0) && (lmbbase <= base)) | ||
346 | break; | ||
347 | } | 358 | } |
348 | 359 | return 0; | |
349 | if (i < 0) | ||
350 | return 0; | ||
351 | |||
352 | if (lmb_add_region(&lmb.reserved, base, lmb_align_up(size, align)) < 0) | ||
353 | return 0; | ||
354 | |||
355 | return base; | ||
356 | } | 360 | } |
357 | 361 | ||
358 | /* You must call lmb_analyze() before this. */ | 362 | /* You must call lmb_analyze() before this. */ |