Re: [PATCH 1/2] cpumask: Introduce possible_cpu_safe()

From: Peter Zijlstra
Date: Thu Apr 04 2019 - 06:45:36 EST


On Thu, Apr 04, 2019 at 01:02:19PM +0300, Dan Carpenter wrote:
> There have been two cases recently where we pass user a controlled "cpu"
> to possible_cpus(). That's not allowed. If it's invalid, it will
> trigger a WARN_ONCE() and an out of bounds read which could result in an
> Oops.

> +/**
> + * cpumask_test_cpu_safe - test for a cpu in a cpumask
> + * @cpu: cpu number
> + * @cpumask: the cpumask pointer
> + *
> + * Returns 1 if @cpu is valid and set in @cpumask, else returns 0
> + */
> +static inline int cpumask_test_cpu_safe(int cpu, const struct cpumask *cpumask)
> +{
> + if ((unsigned int)cpu >= nr_cpu_ids)
> + return 0;
> + cpu = array_index_nospec(cpu, NR_CPUS);

That should be:

cpu = array_index_nospec(cpu, nr_cpu_ids);

NR_CPUS might still be out-of-bounds for dynamically allocated cpumasks.

> + return test_bit(cpu, cpumask_bits(cpumask));
> +}

That said; I don't particularly like this interface not its naming, how
about something like:

static inline unsigned int cpumask_validate_cpu(unsigned int cpu)
{
if (cpu >= nr_cpumask_bits)
return nr_cpumask_bits;
return array_index_nospec(cpu, nr_cpumask_bits);
}

Which you can then use like:

cpu = cpumask_validate_cpu(user_cpu);
if (cpu >= nr_cpu_ids)
return -ENORANGE;

/* @cpu is valid, do what needs doing */