Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

bpf: add kfunc for populating cpumask bits

Add a helper kfunc that sets the bitmap of a bpf_cpumask from BPF memory.

Signed-off-by: Emil Tsalapatis (Meta) <emil@etsalapatis.com>
Acked-by: Hou Tao <houtao1@huawei.com>
Acked-by: Tejun Heo <tj@kernel.org>
Link: https://lore.kernel.org/r/20250309230427.26603-2-emil@etsalapatis.com
Signed-off-by: Alexei Starovoitov <ast@kernel.org>

authored by

Emil Tsalapatis and committed by
Alexei Starovoitov
950ad93d 103b9ab9

+33
+33
kernel/bpf/cpumask.c
··· 420 420 return cpumask_weight(cpumask); 421 421 } 422 422 423 + /** 424 + * bpf_cpumask_populate() - Populate the CPU mask from the contents of 425 + * a BPF memory region. 426 + * 427 + * @cpumask: The cpumask being populated. 428 + * @src: The BPF memory holding the bit pattern. 429 + * @src__sz: Length of the BPF memory region in bytes. 430 + * 431 + * Return: 432 + * * 0 if the struct cpumask * instance was populated successfully. 433 + * * -EACCES if the memory region is too small to populate the cpumask. 434 + * * -EINVAL if the memory region is not aligned to the size of a long 435 + * and the architecture does not support efficient unaligned accesses. 436 + */ 437 + __bpf_kfunc int bpf_cpumask_populate(struct cpumask *cpumask, void *src, size_t src__sz) 438 + { 439 + unsigned long source = (unsigned long)src; 440 + 441 + /* The memory region must be large enough to populate the entire CPU mask. */ 442 + if (src__sz < bitmap_size(nr_cpu_ids)) 443 + return -EACCES; 444 + 445 + /* If avoiding unaligned accesses, the input region must be aligned to the nearest long. */ 446 + if (!IS_ENABLED(CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS) && 447 + !IS_ALIGNED(source, sizeof(long))) 448 + return -EINVAL; 449 + 450 + bitmap_copy(cpumask_bits(cpumask), src, nr_cpu_ids); 451 + 452 + return 0; 453 + } 454 + 423 455 __bpf_kfunc_end_defs(); 424 456 425 457 BTF_KFUNCS_START(cpumask_kfunc_btf_ids) ··· 480 448 BTF_ID_FLAGS(func, bpf_cpumask_any_distribute, KF_RCU) 481 449 BTF_ID_FLAGS(func, bpf_cpumask_any_and_distribute, KF_RCU) 482 450 BTF_ID_FLAGS(func, bpf_cpumask_weight, KF_RCU) 451 + BTF_ID_FLAGS(func, bpf_cpumask_populate, KF_RCU) 483 452 BTF_KFUNCS_END(cpumask_kfunc_btf_ids) 484 453 485 454 static const struct btf_kfunc_id_set cpumask_kfunc_set = {