RISC-V: Improve init_resources()

The kernel region is always present and we know where it is, no need to
look for it inside the loop, just ignore it like the rest of the
reserved regions within system's memory.

Additionally, we don't need to call memblock_free inside the loop, as if
called it'll split the region of pre-allocated resources in two parts,
messing things up, just re-use the previous pre-allocated resource and
free any unused resources after both loops finish.

Signed-off-by: Nick Kossifidis <mick@ics.forth.gr>
[Palmer: commit text]
Signed-off-by: Palmer Dabbelt <palmerdabbelt@google.com>
This commit is contained in:
Nick Kossifidis 2021-04-19 03:55:37 +03:00 committed by Palmer Dabbelt
parent fba8a8674f
commit ffe0e52612
No known key found for this signature in database
GPG Key ID: 2E1319F35FBB1889
1 changed files with 56 additions and 54 deletions

View File

@ -60,6 +60,7 @@ static DEFINE_PER_CPU(struct cpu, cpu_devices);
* also add "System RAM" regions for compatibility with other * also add "System RAM" regions for compatibility with other
* archs, and the rest of the known regions for completeness. * archs, and the rest of the known regions for completeness.
*/ */
static struct resource kimage_res = { .name = "Kernel image", };
static struct resource code_res = { .name = "Kernel code", }; static struct resource code_res = { .name = "Kernel code", };
static struct resource data_res = { .name = "Kernel data", }; static struct resource data_res = { .name = "Kernel data", };
static struct resource rodata_res = { .name = "Kernel rodata", }; static struct resource rodata_res = { .name = "Kernel rodata", };
@ -80,56 +81,16 @@ static int __init add_resource(struct resource *parent,
return 1; return 1;
} }
static int __init add_kernel_resources(struct resource *res) static int __init add_kernel_resources(void)
{ {
int ret = 0; int ret = 0;
/* /*
* The memory region of the kernel image is continuous and * The memory region of the kernel image is continuous and
* was reserved on setup_bootmem, find it here and register * was reserved on setup_bootmem, register it here as a
* it as a resource, then register the various segments of * resource, with the various segments of the image as
* the image as child nodes * child nodes.
*/ */
if (!(res->start <= code_res.start && res->end >= data_res.end))
return 0;
res->name = "Kernel image";
res->flags = IORESOURCE_SYSTEM_RAM | IORESOURCE_BUSY;
/*
* We removed a part of this region on setup_bootmem so
* we need to expand the resource for the bss to fit in.
*/
res->end = bss_res.end;
ret = add_resource(&iomem_resource, res);
if (ret < 0)
return ret;
ret = add_resource(res, &code_res);
if (ret < 0)
return ret;
ret = add_resource(res, &rodata_res);
if (ret < 0)
return ret;
ret = add_resource(res, &data_res);
if (ret < 0)
return ret;
ret = add_resource(res, &bss_res);
return ret;
}
static void __init init_resources(void)
{
struct memblock_region *region = NULL;
struct resource *res = NULL;
struct resource *mem_res = NULL;
size_t mem_res_sz = 0;
int ret = 0, i = 0;
code_res.start = __pa_symbol(_text); code_res.start = __pa_symbol(_text);
code_res.end = __pa_symbol(_etext) - 1; code_res.end = __pa_symbol(_etext) - 1;
@ -147,35 +108,73 @@ static void __init init_resources(void)
bss_res.end = __pa_symbol(__bss_stop) - 1; bss_res.end = __pa_symbol(__bss_stop) - 1;
bss_res.flags = IORESOURCE_SYSTEM_RAM | IORESOURCE_BUSY; bss_res.flags = IORESOURCE_SYSTEM_RAM | IORESOURCE_BUSY;
mem_res_sz = (memblock.memory.cnt + memblock.reserved.cnt) * sizeof(*mem_res); kimage_res.start = code_res.start;
kimage_res.end = bss_res.end;
kimage_res.flags = IORESOURCE_SYSTEM_RAM | IORESOURCE_BUSY;
ret = add_resource(&iomem_resource, &kimage_res);
if (ret < 0)
return ret;
ret = add_resource(&kimage_res, &code_res);
if (ret < 0)
return ret;
ret = add_resource(&kimage_res, &rodata_res);
if (ret < 0)
return ret;
ret = add_resource(&kimage_res, &data_res);
if (ret < 0)
return ret;
ret = add_resource(&kimage_res, &bss_res);
return ret;
}
static void __init init_resources(void)
{
struct memblock_region *region = NULL;
struct resource *res = NULL;
struct resource *mem_res = NULL;
size_t mem_res_sz = 0;
int num_resources = 0, res_idx = 0;
int ret = 0;
/* + 1 as memblock_alloc() might increase memblock.reserved.cnt */
num_resources = memblock.memory.cnt + memblock.reserved.cnt + 1;
res_idx = num_resources - 1;
mem_res_sz = num_resources * sizeof(*mem_res);
mem_res = memblock_alloc(mem_res_sz, SMP_CACHE_BYTES); mem_res = memblock_alloc(mem_res_sz, SMP_CACHE_BYTES);
if (!mem_res) if (!mem_res)
panic("%s: Failed to allocate %zu bytes\n", __func__, mem_res_sz); panic("%s: Failed to allocate %zu bytes\n", __func__, mem_res_sz);
/* /*
* Start by adding the reserved regions, if they overlap * Start by adding the reserved regions, if they overlap
* with /memory regions, insert_resource later on will take * with /memory regions, insert_resource later on will take
* care of it. * care of it.
*/ */
ret = add_kernel_resources();
if (ret < 0)
goto error;
for_each_reserved_mem_region(region) { for_each_reserved_mem_region(region) {
res = &mem_res[i++]; res = &mem_res[res_idx--];
res->name = "Reserved"; res->name = "Reserved";
res->flags = IORESOURCE_MEM | IORESOURCE_BUSY; res->flags = IORESOURCE_MEM | IORESOURCE_BUSY;
res->start = __pfn_to_phys(memblock_region_reserved_base_pfn(region)); res->start = __pfn_to_phys(memblock_region_reserved_base_pfn(region));
res->end = __pfn_to_phys(memblock_region_reserved_end_pfn(region)) - 1; res->end = __pfn_to_phys(memblock_region_reserved_end_pfn(region)) - 1;
ret = add_kernel_resources(res);
if (ret < 0)
goto error;
else if (ret)
continue;
/* /*
* Ignore any other reserved regions within * Ignore any other reserved regions within
* system memory. * system memory.
*/ */
if (memblock_is_memory(res->start)) { if (memblock_is_memory(res->start)) {
memblock_free((phys_addr_t) res, sizeof(struct resource)); /* Re-use this pre-allocated resource */
res_idx++;
continue; continue;
} }
@ -186,7 +185,7 @@ static void __init init_resources(void)
/* Add /memory regions to the resource tree */ /* Add /memory regions to the resource tree */
for_each_mem_region(region) { for_each_mem_region(region) {
res = &mem_res[i++]; res = &mem_res[res_idx--];
if (unlikely(memblock_is_nomap(region))) { if (unlikely(memblock_is_nomap(region))) {
res->name = "Reserved"; res->name = "Reserved";
@ -204,6 +203,9 @@ static void __init init_resources(void)
goto error; goto error;
} }
/* Clean-up any unused pre-allocated resources */
mem_res_sz = (num_resources - res_idx + 1) * sizeof(*mem_res);
memblock_free((phys_addr_t) mem_res, mem_res_sz);
return; return;
error: error: