Skip to content

Commit be52ea7

Browse files
ricardonOpenEmbedded
authored andcommitted
cacheinfo: Allocate memory during CPU hotplug if not done from the primary CPU
commit b3fce42 upstream. Commit 5944ce0 ("arch_topology: Build cacheinfo from primary CPU") adds functionality that architectures can use to optionally allocate and build cacheinfo early during boot. Commit 6539cff ("cacheinfo: Add arch specific early level initializer") lets secondary CPUs correct (and reallocate memory) cacheinfo data if needed. If the early build functionality is not used and cacheinfo does not need correction, memory for cacheinfo is never allocated. x86 does not use the early build functionality. Consequently, during the cacheinfo CPU hotplug callback, last_level_cache_is_valid() attempts to dereference a NULL pointer: BUG: kernel NULL pointer dereference, address: 0000000000000100 #PF: supervisor read access in kernel mode #PF: error_code(0x0000) - not present page PGD 0 P4D 0 Oops: 0000 [openbmc#1] PREEPMT SMP NOPTI CPU: 0 PID 19 Comm: cpuhp/0 Not tainted 6.4.0-rc2 openbmc#1 RIP: 0010: last_level_cache_is_valid+0x95/0xe0a Allocate memory for cacheinfo during the cacheinfo CPU hotplug callback if not done earlier. Moreover, before determining the validity of the last-level cache info, ensure that it has been allocated. Simply checking for non-zero cache_leaves() is not sufficient, as some architectures (e.g., Intel processors) have non-zero cache_leaves() before allocation. Dereferencing NULL cacheinfo can occur in update_per_cpu_data_slice_size(). This function iterates over all online CPUs. However, a CPU may have come online recently, but its cacheinfo may not have been allocated yet. While here, remove an unnecessary indentation in allocate_cache_info(). [ bp: Massage. ] Fixes: 6539cff ("cacheinfo: Add arch specific early level initializer") Signed-off-by: Ricardo Neri <[email protected]> Signed-off-by: Borislav Petkov (AMD) <[email protected]> Reviewed-by: Radu Rendec <[email protected]> Reviewed-by: Nikolay Borisov <[email protected]> Reviewed-by: Andreas Herrmann <[email protected]> Reviewed-by: Sudeep Holla <[email protected]> Cc: [email protected] # 6.3+ Link: https://lore.kernel.org/r/[email protected] Signed-off-by: Greg Kroah-Hartman <[email protected]>
1 parent b1a1341 commit be52ea7

File tree

1 file changed

+8
-6
lines changed

1 file changed

+8
-6
lines changed

drivers/base/cacheinfo.c

Lines changed: 8 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ bool last_level_cache_is_valid(unsigned int cpu)
5858
{
5959
struct cacheinfo *llc;
6060

61-
if (!cache_leaves(cpu))
61+
if (!cache_leaves(cpu) || !per_cpu_cacheinfo(cpu))
6262
return false;
6363

6464
llc = per_cpu_cacheinfo_idx(cpu, cache_leaves(cpu) - 1);
@@ -478,11 +478,9 @@ int __weak populate_cache_leaves(unsigned int cpu)
478478
return -ENOENT;
479479
}
480480

481-
static inline
482-
int allocate_cache_info(int cpu)
481+
static inline int allocate_cache_info(int cpu)
483482
{
484-
per_cpu_cacheinfo(cpu) = kcalloc(cache_leaves(cpu),
485-
sizeof(struct cacheinfo), GFP_ATOMIC);
483+
per_cpu_cacheinfo(cpu) = kcalloc(cache_leaves(cpu), sizeof(struct cacheinfo), GFP_ATOMIC);
486484
if (!per_cpu_cacheinfo(cpu)) {
487485
cache_leaves(cpu) = 0;
488486
return -ENOMEM;
@@ -554,7 +552,11 @@ static inline int init_level_allocate_ci(unsigned int cpu)
554552
*/
555553
ci_cacheinfo(cpu)->early_ci_levels = false;
556554

557-
if (cache_leaves(cpu) <= early_leaves)
555+
/*
556+
* Some architectures (e.g., x86) do not use early initialization.
557+
* Allocate memory now in such case.
558+
*/
559+
if (cache_leaves(cpu) <= early_leaves && per_cpu_cacheinfo(cpu))
558560
return 0;
559561

560562
kfree(per_cpu_cacheinfo(cpu));

0 commit comments

Comments
 (0)