diff options
| author | Benjamin Herrenschmidt <benh@kernel.crashing.org> | 2010-07-06 18:39:14 -0400 |
|---|---|---|
| committer | Benjamin Herrenschmidt <benh@kernel.crashing.org> | 2010-08-04 22:56:21 -0400 |
| commit | d2cd563ba82c424083b78e0ce97d68bfb04d1242 (patch) | |
| tree | ff4ab6de8b6906a6c955aa51c4ee53868dfafc0a | |
| parent | 142b45a72e221537c1bb1995497fef7cdc439e26 (diff) | |
memblock: Add arch function to control coalescing of memblock memory regions
Some archs such as ARM want to avoid coalescing accross things such
as the lowmem/highmem boundary or similar. This provides the option
to control it via an arch callback for which a weak default is provided
which always allows coalescing.
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
| -rw-r--r-- | include/linux/memblock.h | 2 | ||||
| -rw-r--r-- | mm/memblock.c | 19 |
2 files changed, 20 insertions, 1 deletions
diff --git a/include/linux/memblock.h b/include/linux/memblock.h index 150be938b910..e5e8f9db3a84 100644 --- a/include/linux/memblock.h +++ b/include/linux/memblock.h | |||
| @@ -72,6 +72,8 @@ extern void memblock_dump_all(void); | |||
| 72 | 72 | ||
| 73 | /* Provided by the architecture */ | 73 | /* Provided by the architecture */ |
| 74 | extern phys_addr_t memblock_nid_range(phys_addr_t start, phys_addr_t end, int *nid); | 74 | extern phys_addr_t memblock_nid_range(phys_addr_t start, phys_addr_t end, int *nid); |
| 75 | extern int memblock_memory_can_coalesce(phys_addr_t addr1, phys_addr_t size1, | ||
| 76 | phys_addr_t addr2, phys_addr_t size2); | ||
| 75 | 77 | ||
| 76 | /** | 78 | /** |
| 77 | * memblock_set_current_limit - Set the current allocation limit to allow | 79 | * memblock_set_current_limit - Set the current allocation limit to allow |
diff --git a/mm/memblock.c b/mm/memblock.c index 0787790b1ce0..8715f09434df 100644 --- a/mm/memblock.c +++ b/mm/memblock.c | |||
| @@ -241,6 +241,12 @@ static int memblock_double_array(struct memblock_type *type) | |||
| 241 | return 0; | 241 | return 0; |
| 242 | } | 242 | } |
| 243 | 243 | ||
| 244 | extern int __weak memblock_memory_can_coalesce(phys_addr_t addr1, phys_addr_t size1, | ||
| 245 | phys_addr_t addr2, phys_addr_t size2) | ||
| 246 | { | ||
| 247 | return 1; | ||
| 248 | } | ||
| 249 | |||
| 244 | static long memblock_add_region(struct memblock_type *type, phys_addr_t base, phys_addr_t size) | 250 | static long memblock_add_region(struct memblock_type *type, phys_addr_t base, phys_addr_t size) |
| 245 | { | 251 | { |
| 246 | unsigned long coalesced = 0; | 252 | unsigned long coalesced = 0; |
| @@ -262,6 +268,10 @@ static long memblock_add_region(struct memblock_type *type, phys_addr_t base, ph | |||
| 262 | return 0; | 268 | return 0; |
| 263 | 269 | ||
| 264 | adjacent = memblock_addrs_adjacent(base, size, rgnbase, rgnsize); | 270 | adjacent = memblock_addrs_adjacent(base, size, rgnbase, rgnsize); |
| 271 | /* Check if arch allows coalescing */ | ||
| 272 | if (adjacent != 0 && type == &memblock.memory && | ||
| 273 | !memblock_memory_can_coalesce(base, size, rgnbase, rgnsize)) | ||
| 274 | break; | ||
| 265 | if (adjacent > 0) { | 275 | if (adjacent > 0) { |
| 266 | type->regions[i].base -= size; | 276 | type->regions[i].base -= size; |
| 267 | type->regions[i].size += size; | 277 | type->regions[i].size += size; |
| @@ -274,7 +284,14 @@ static long memblock_add_region(struct memblock_type *type, phys_addr_t base, ph | |||
| 274 | } | 284 | } |
| 275 | } | 285 | } |
| 276 | 286 | ||
| 277 | if ((i < type->cnt - 1) && memblock_regions_adjacent(type, i, i+1)) { | 287 | /* If we plugged a hole, we may want to also coalesce with the |
| 288 | * next region | ||
| 289 | */ | ||
| 290 | if ((i < type->cnt - 1) && memblock_regions_adjacent(type, i, i+1) && | ||
| 291 | ((type != &memblock.memory || memblock_memory_can_coalesce(type->regions[i].base, | ||
| 292 | type->regions[i].size, | ||
| 293 | type->regions[i+1].base, | ||
| 294 | type->regions[i+1].size)))) { | ||
| 278 | memblock_coalesce_regions(type, i, i+1); | 295 | memblock_coalesce_regions(type, i, i+1); |
| 279 | coalesced++; | 296 | coalesced++; |
| 280 | } | 297 | } |
