memblock: introduce saner 'memblock_free_ptr()' interface
The boot-time allocation interface for memblock is a mess, with
'memblock_alloc()' returning a virtual pointer, but then you are
supposed to free it with 'memblock_free()' that takes a _physical_
address.
Not only is that all kinds of strange and illogical, but it actually
causes bugs, when people then use it like a normal allocation function,
and it fails spectacularly on a NULL pointer:
https://lore.kernel.org/all/20210912140820.GD25450@xsang-OptiPlex-9020/
or just random memory corruption if the debug checks don't catch it:
https://lore.kernel.org/all/61ab2d0c-3313-aaab-514c-e15b7aa054a0@suse.cz/
I really don't want to apply patches that treat the symptoms, when the
fundamental cause is this horribly confusing interface.
I started out looking at just automating a sane replacement sequence,
but because of this mix or virtual and physical addresses, and because
people have used the "__pa()" macro that can take either a regular
kernel pointer, or just the raw "unsigned long" address, it's all quite
messy.
So this just introduces a new saner interface for freeing a virtual
address that was allocated using 'memblock_alloc()', and that was kept
as a regular kernel pointer. And then it converts a couple of users
that are obvious and easy to test, including the 'xbc_nodes' case in
lib/bootconfig.c that caused problems.
Reported-by: kernel test robot <oliver.sang@intel.com>
Fixes: 40caa127f3
("init: bootconfig: Remove all bootconfig data when the init memory is removed")
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Mike Rapoport <rppt@kernel.org>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Ingo Molnar <mingo@kernel.org>
Cc: Masami Hiramatsu <mhiramat@kernel.org>
Cc: Vlastimil Babka <vbabka@suse.cz>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
6a4746ba06
commit
77e02cf57b
@ -135,7 +135,7 @@ static void * __init pcpu_fc_alloc(unsigned int cpu, size_t size, size_t align)
|
|||||||
|
|
||||||
static void __init pcpu_fc_free(void *ptr, size_t size)
|
static void __init pcpu_fc_free(void *ptr, size_t size)
|
||||||
{
|
{
|
||||||
memblock_free(__pa(ptr), size);
|
memblock_free_ptr(ptr, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int __init pcpu_cpu_distance(unsigned int from, unsigned int to)
|
static int __init pcpu_cpu_distance(unsigned int from, unsigned int to)
|
||||||
|
@ -49,8 +49,7 @@ static void __init kasan_populate_pmd(pmd_t *pmd, unsigned long addr,
|
|||||||
p = early_alloc(PMD_SIZE, nid, false);
|
p = early_alloc(PMD_SIZE, nid, false);
|
||||||
if (p && pmd_set_huge(pmd, __pa(p), PAGE_KERNEL))
|
if (p && pmd_set_huge(pmd, __pa(p), PAGE_KERNEL))
|
||||||
return;
|
return;
|
||||||
else if (p)
|
memblock_free_ptr(p, PMD_SIZE);
|
||||||
memblock_free(__pa(p), PMD_SIZE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
p = early_alloc(PAGE_SIZE, nid, true);
|
p = early_alloc(PAGE_SIZE, nid, true);
|
||||||
@ -86,8 +85,7 @@ static void __init kasan_populate_pud(pud_t *pud, unsigned long addr,
|
|||||||
p = early_alloc(PUD_SIZE, nid, false);
|
p = early_alloc(PUD_SIZE, nid, false);
|
||||||
if (p && pud_set_huge(pud, __pa(p), PAGE_KERNEL))
|
if (p && pud_set_huge(pud, __pa(p), PAGE_KERNEL))
|
||||||
return;
|
return;
|
||||||
else if (p)
|
memblock_free_ptr(p, PUD_SIZE);
|
||||||
memblock_free(__pa(p), PUD_SIZE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
p = early_alloc(PAGE_SIZE, nid, true);
|
p = early_alloc(PAGE_SIZE, nid, true);
|
||||||
|
@ -355,7 +355,7 @@ void __init numa_reset_distance(void)
|
|||||||
|
|
||||||
/* numa_distance could be 1LU marking allocation failure, test cnt */
|
/* numa_distance could be 1LU marking allocation failure, test cnt */
|
||||||
if (numa_distance_cnt)
|
if (numa_distance_cnt)
|
||||||
memblock_free(__pa(numa_distance), size);
|
memblock_free_ptr(numa_distance, size);
|
||||||
numa_distance_cnt = 0;
|
numa_distance_cnt = 0;
|
||||||
numa_distance = NULL; /* enable table creation */
|
numa_distance = NULL; /* enable table creation */
|
||||||
}
|
}
|
||||||
|
@ -517,8 +517,7 @@ void __init numa_emulation(struct numa_meminfo *numa_meminfo, int numa_dist_cnt)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* free the copied physical distance table */
|
/* free the copied physical distance table */
|
||||||
if (phys_dist)
|
memblock_free_ptr(phys_dist, phys_size);
|
||||||
memblock_free(__pa(phys_dist), phys_size);
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
no_emu:
|
no_emu:
|
||||||
|
@ -264,7 +264,7 @@ void __init numa_free_distance(void)
|
|||||||
size = numa_distance_cnt * numa_distance_cnt *
|
size = numa_distance_cnt * numa_distance_cnt *
|
||||||
sizeof(numa_distance[0]);
|
sizeof(numa_distance[0]);
|
||||||
|
|
||||||
memblock_free(__pa(numa_distance), size);
|
memblock_free_ptr(numa_distance, size);
|
||||||
numa_distance_cnt = 0;
|
numa_distance_cnt = 0;
|
||||||
numa_distance = NULL;
|
numa_distance = NULL;
|
||||||
}
|
}
|
||||||
|
@ -570,7 +570,7 @@ fail_msg_node:
|
|||||||
fail_db_node:
|
fail_db_node:
|
||||||
of_node_put(smu->db_node);
|
of_node_put(smu->db_node);
|
||||||
fail_bootmem:
|
fail_bootmem:
|
||||||
memblock_free(__pa(smu), sizeof(struct smu_device));
|
memblock_free_ptr(smu, sizeof(struct smu_device));
|
||||||
smu = NULL;
|
smu = NULL;
|
||||||
fail_np:
|
fail_np:
|
||||||
of_node_put(np);
|
of_node_put(np);
|
||||||
|
@ -118,6 +118,7 @@ int memblock_mark_nomap(phys_addr_t base, phys_addr_t size);
|
|||||||
int memblock_clear_nomap(phys_addr_t base, phys_addr_t size);
|
int memblock_clear_nomap(phys_addr_t base, phys_addr_t size);
|
||||||
|
|
||||||
void memblock_free_all(void);
|
void memblock_free_all(void);
|
||||||
|
void memblock_free_ptr(void *ptr, size_t size);
|
||||||
void reset_node_managed_pages(pg_data_t *pgdat);
|
void reset_node_managed_pages(pg_data_t *pgdat);
|
||||||
void reset_all_zones_managed_pages(void);
|
void reset_all_zones_managed_pages(void);
|
||||||
|
|
||||||
|
@ -924,7 +924,7 @@ static void __init print_unknown_bootoptions(void)
|
|||||||
end += sprintf(end, " %s", *p);
|
end += sprintf(end, " %s", *p);
|
||||||
|
|
||||||
pr_notice("Unknown command line parameters:%s\n", unknown_options);
|
pr_notice("Unknown command line parameters:%s\n", unknown_options);
|
||||||
memblock_free(__pa(unknown_options), len);
|
memblock_free_ptr(unknown_options, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
asmlinkage __visible void __init __no_sanitize_address start_kernel(void)
|
asmlinkage __visible void __init __no_sanitize_address start_kernel(void)
|
||||||
|
@ -1166,9 +1166,9 @@ void __init setup_log_buf(int early)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
err_free_descs:
|
err_free_descs:
|
||||||
memblock_free(__pa(new_descs), new_descs_size);
|
memblock_free_ptr(new_descs, new_descs_size);
|
||||||
err_free_log_buf:
|
err_free_log_buf:
|
||||||
memblock_free(__pa(new_log_buf), new_log_buf_len);
|
memblock_free_ptr(new_log_buf, new_log_buf_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool __read_mostly ignore_loglevel;
|
static bool __read_mostly ignore_loglevel;
|
||||||
|
@ -792,7 +792,7 @@ void __init xbc_destroy_all(void)
|
|||||||
xbc_data = NULL;
|
xbc_data = NULL;
|
||||||
xbc_data_size = 0;
|
xbc_data_size = 0;
|
||||||
xbc_node_num = 0;
|
xbc_node_num = 0;
|
||||||
memblock_free(__pa(xbc_nodes), sizeof(struct xbc_node) * XBC_NODE_MAX);
|
memblock_free_ptr(xbc_nodes, sizeof(struct xbc_node) * XBC_NODE_MAX);
|
||||||
xbc_nodes = NULL;
|
xbc_nodes = NULL;
|
||||||
brace_index = 0;
|
brace_index = 0;
|
||||||
}
|
}
|
||||||
|
@ -472,7 +472,7 @@ static int __init_memblock memblock_double_array(struct memblock_type *type,
|
|||||||
kfree(old_array);
|
kfree(old_array);
|
||||||
else if (old_array != memblock_memory_init_regions &&
|
else if (old_array != memblock_memory_init_regions &&
|
||||||
old_array != memblock_reserved_init_regions)
|
old_array != memblock_reserved_init_regions)
|
||||||
memblock_free(__pa(old_array), old_alloc_size);
|
memblock_free_ptr(old_array, old_alloc_size);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reserve the new array if that comes from the memblock. Otherwise, we
|
* Reserve the new array if that comes from the memblock. Otherwise, we
|
||||||
@ -795,6 +795,20 @@ int __init_memblock memblock_remove(phys_addr_t base, phys_addr_t size)
|
|||||||
return memblock_remove_range(&memblock.memory, base, size);
|
return memblock_remove_range(&memblock.memory, base, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* memblock_free_ptr - free boot memory allocation
|
||||||
|
* @ptr: starting address of the boot memory allocation
|
||||||
|
* @size: size of the boot memory block in bytes
|
||||||
|
*
|
||||||
|
* Free boot memory block previously allocated by memblock_alloc_xx() API.
|
||||||
|
* The freeing memory will not be released to the buddy allocator.
|
||||||
|
*/
|
||||||
|
void __init_memblock memblock_free_ptr(void *ptr, size_t size)
|
||||||
|
{
|
||||||
|
if (ptr)
|
||||||
|
memblock_free(__pa(ptr), size);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* memblock_free - free boot memory block
|
* memblock_free - free boot memory block
|
||||||
* @base: phys starting address of the boot memory block
|
* @base: phys starting address of the boot memory block
|
||||||
|
Loading…
Reference in New Issue
Block a user