|
|
|
@ -5,6 +5,9 @@
|
|
|
|
|
* Cpumasks provide a bitmap suitable for representing the
|
|
|
|
|
* set of CPU's in a system, one bit position per CPU number.
|
|
|
|
|
*
|
|
|
|
|
* The new cpumask_ ops take a "struct cpumask *"; the old ones
|
|
|
|
|
* use cpumask_t.
|
|
|
|
|
*
|
|
|
|
|
* See detailed comments in the file linux/bitmap.h describing the
|
|
|
|
|
* data type on which these cpumasks are based.
|
|
|
|
|
*
|
|
|
|
@ -31,7 +34,7 @@
|
|
|
|
|
* will span the entire range of NR_CPUS.
|
|
|
|
|
* . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
|
|
|
|
|
*
|
|
|
|
|
* The available cpumask operations are:
|
|
|
|
|
* The obsolescent cpumask operations are:
|
|
|
|
|
*
|
|
|
|
|
* void cpu_set(cpu, mask) turn on bit 'cpu' in mask
|
|
|
|
|
* void cpu_clear(cpu, mask) turn off bit 'cpu' in mask
|
|
|
|
@ -138,7 +141,7 @@
|
|
|
|
|
#include <linux/threads.h>
|
|
|
|
|
#include <linux/bitmap.h>
|
|
|
|
|
|
|
|
|
|
typedef struct { DECLARE_BITMAP(bits, NR_CPUS); } cpumask_t;
|
|
|
|
|
typedef struct cpumask { DECLARE_BITMAP(bits, NR_CPUS); } cpumask_t;
|
|
|
|
|
extern cpumask_t _unused_cpumask_arg_;
|
|
|
|
|
|
|
|
|
|
#define cpu_set(cpu, dst) __cpu_set((cpu), &(dst))
|
|
|
|
@ -527,4 +530,556 @@ extern cpumask_t cpu_active_map;
|
|
|
|
|
#define for_each_online_cpu(cpu) for_each_cpu_mask_nr((cpu), cpu_online_map)
|
|
|
|
|
#define for_each_present_cpu(cpu) for_each_cpu_mask_nr((cpu), cpu_present_map)
|
|
|
|
|
|
|
|
|
|
/* These are the new versions of the cpumask operators: passed by pointer.
|
|
|
|
|
* The older versions will be implemented in terms of these, then deleted. */
|
|
|
|
|
#define cpumask_bits(maskp) ((maskp)->bits)
|
|
|
|
|
|
|
|
|
|
#if NR_CPUS <= BITS_PER_LONG
|
|
|
|
|
#define CPU_BITS_ALL \
|
|
|
|
|
{ \
|
|
|
|
|
[BITS_TO_LONGS(NR_CPUS)-1] = CPU_MASK_LAST_WORD \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This produces more efficient code. */
|
|
|
|
|
#define nr_cpumask_bits NR_CPUS
|
|
|
|
|
|
|
|
|
|
#else /* NR_CPUS > BITS_PER_LONG */
|
|
|
|
|
|
|
|
|
|
#define CPU_BITS_ALL \
|
|
|
|
|
{ \
|
|
|
|
|
[0 ... BITS_TO_LONGS(NR_CPUS)-2] = ~0UL, \
|
|
|
|
|
[BITS_TO_LONGS(NR_CPUS)-1] = CPU_MASK_LAST_WORD \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define nr_cpumask_bits nr_cpu_ids
|
|
|
|
|
#endif /* NR_CPUS > BITS_PER_LONG */
|
|
|
|
|
|
|
|
|
|
/* verify cpu argument to cpumask_* operators */
|
|
|
|
|
static inline unsigned int cpumask_check(unsigned int cpu)
|
|
|
|
|
{
|
|
|
|
|
#ifdef CONFIG_DEBUG_PER_CPU_MAPS
|
|
|
|
|
WARN_ON_ONCE(cpu >= nr_cpumask_bits);
|
|
|
|
|
#endif /* CONFIG_DEBUG_PER_CPU_MAPS */
|
|
|
|
|
return cpu;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if NR_CPUS == 1
|
|
|
|
|
/* Uniprocessor. Assume all masks are "1". */
|
|
|
|
|
static inline unsigned int cpumask_first(const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Valid inputs for n are -1 and 0. */
|
|
|
|
|
static inline unsigned int cpumask_next(int n, const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
return n+1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline unsigned int cpumask_next_zero(int n, const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
return n+1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline unsigned int cpumask_next_and(int n,
|
|
|
|
|
const struct cpumask *srcp,
|
|
|
|
|
const struct cpumask *andp)
|
|
|
|
|
{
|
|
|
|
|
return n+1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* cpu must be a valid cpu, ie 0, so there's no other choice. */
|
|
|
|
|
static inline unsigned int cpumask_any_but(const struct cpumask *mask,
|
|
|
|
|
unsigned int cpu)
|
|
|
|
|
{
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define for_each_cpu(cpu, mask) \
|
|
|
|
|
for ((cpu) = 0; (cpu) < 1; (cpu)++, (void)mask)
|
|
|
|
|
#define for_each_cpu_and(cpu, mask, and) \
|
|
|
|
|
for ((cpu) = 0; (cpu) < 1; (cpu)++, (void)mask, (void)and)
|
|
|
|
|
#else
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_first - get the first cpu in a cpumask
|
|
|
|
|
* @srcp: the cpumask pointer
|
|
|
|
|
*
|
|
|
|
|
* Returns >= nr_cpu_ids if no cpus set.
|
|
|
|
|
*/
|
|
|
|
|
static inline unsigned int cpumask_first(const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
return find_first_bit(cpumask_bits(srcp), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_next - get the next cpu in a cpumask
|
|
|
|
|
* @n: the cpu prior to the place to search (ie. return will be > @n)
|
|
|
|
|
* @srcp: the cpumask pointer
|
|
|
|
|
*
|
|
|
|
|
* Returns >= nr_cpu_ids if no further cpus set.
|
|
|
|
|
*/
|
|
|
|
|
static inline unsigned int cpumask_next(int n, const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
/* -1 is a legal arg here. */
|
|
|
|
|
if (n != -1)
|
|
|
|
|
cpumask_check(n);
|
|
|
|
|
return find_next_bit(cpumask_bits(srcp), nr_cpumask_bits, n+1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_next_zero - get the next unset cpu in a cpumask
|
|
|
|
|
* @n: the cpu prior to the place to search (ie. return will be > @n)
|
|
|
|
|
* @srcp: the cpumask pointer
|
|
|
|
|
*
|
|
|
|
|
* Returns >= nr_cpu_ids if no further cpus unset.
|
|
|
|
|
*/
|
|
|
|
|
static inline unsigned int cpumask_next_zero(int n, const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
/* -1 is a legal arg here. */
|
|
|
|
|
if (n != -1)
|
|
|
|
|
cpumask_check(n);
|
|
|
|
|
return find_next_zero_bit(cpumask_bits(srcp), nr_cpumask_bits, n+1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int cpumask_next_and(int n, const struct cpumask *, const struct cpumask *);
|
|
|
|
|
int cpumask_any_but(const struct cpumask *mask, unsigned int cpu);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* for_each_cpu - iterate over every cpu in a mask
|
|
|
|
|
* @cpu: the (optionally unsigned) integer iterator
|
|
|
|
|
* @mask: the cpumask pointer
|
|
|
|
|
*
|
|
|
|
|
* After the loop, cpu is >= nr_cpu_ids.
|
|
|
|
|
*/
|
|
|
|
|
#define for_each_cpu(cpu, mask) \
|
|
|
|
|
for ((cpu) = -1; \
|
|
|
|
|
(cpu) = cpumask_next((cpu), (mask)), \
|
|
|
|
|
(cpu) < nr_cpu_ids;)
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* for_each_cpu_and - iterate over every cpu in both masks
|
|
|
|
|
* @cpu: the (optionally unsigned) integer iterator
|
|
|
|
|
* @mask: the first cpumask pointer
|
|
|
|
|
* @and: the second cpumask pointer
|
|
|
|
|
*
|
|
|
|
|
* This saves a temporary CPU mask in many places. It is equivalent to:
|
|
|
|
|
* struct cpumask tmp;
|
|
|
|
|
* cpumask_and(&tmp, &mask, &and);
|
|
|
|
|
* for_each_cpu(cpu, &tmp)
|
|
|
|
|
* ...
|
|
|
|
|
*
|
|
|
|
|
* After the loop, cpu is >= nr_cpu_ids.
|
|
|
|
|
*/
|
|
|
|
|
#define for_each_cpu_and(cpu, mask, and) \
|
|
|
|
|
for ((cpu) = -1; \
|
|
|
|
|
(cpu) = cpumask_next_and((cpu), (mask), (and)), \
|
|
|
|
|
(cpu) < nr_cpu_ids;)
|
|
|
|
|
#endif /* SMP */
|
|
|
|
|
|
|
|
|
|
#define CPU_BITS_NONE \
|
|
|
|
|
{ \
|
|
|
|
|
[0 ... BITS_TO_LONGS(NR_CPUS)-1] = 0UL \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define CPU_BITS_CPU0 \
|
|
|
|
|
{ \
|
|
|
|
|
[0] = 1UL \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_set_cpu - set a cpu in a cpumask
|
|
|
|
|
* @cpu: cpu number (< nr_cpu_ids)
|
|
|
|
|
* @dstp: the cpumask pointer
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_set_cpu(unsigned int cpu, struct cpumask *dstp)
|
|
|
|
|
{
|
|
|
|
|
set_bit(cpumask_check(cpu), cpumask_bits(dstp));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_clear_cpu - clear a cpu in a cpumask
|
|
|
|
|
* @cpu: cpu number (< nr_cpu_ids)
|
|
|
|
|
* @dstp: the cpumask pointer
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_clear_cpu(int cpu, struct cpumask *dstp)
|
|
|
|
|
{
|
|
|
|
|
clear_bit(cpumask_check(cpu), cpumask_bits(dstp));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_test_cpu - test for a cpu in a cpumask
|
|
|
|
|
* @cpu: cpu number (< nr_cpu_ids)
|
|
|
|
|
* @cpumask: the cpumask pointer
|
|
|
|
|
*
|
|
|
|
|
* No static inline type checking - see Subtlety (1) above.
|
|
|
|
|
*/
|
|
|
|
|
#define cpumask_test_cpu(cpu, cpumask) \
|
|
|
|
|
test_bit(cpumask_check(cpu), (cpumask)->bits)
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_test_and_set_cpu - atomically test and set a cpu in a cpumask
|
|
|
|
|
* @cpu: cpu number (< nr_cpu_ids)
|
|
|
|
|
* @cpumask: the cpumask pointer
|
|
|
|
|
*
|
|
|
|
|
* test_and_set_bit wrapper for cpumasks.
|
|
|
|
|
*/
|
|
|
|
|
static inline int cpumask_test_and_set_cpu(int cpu, struct cpumask *cpumask)
|
|
|
|
|
{
|
|
|
|
|
return test_and_set_bit(cpumask_check(cpu), cpumask_bits(cpumask));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_setall - set all cpus (< nr_cpu_ids) in a cpumask
|
|
|
|
|
* @dstp: the cpumask pointer
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_setall(struct cpumask *dstp)
|
|
|
|
|
{
|
|
|
|
|
bitmap_fill(cpumask_bits(dstp), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_clear - clear all cpus (< nr_cpu_ids) in a cpumask
|
|
|
|
|
* @dstp: the cpumask pointer
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_clear(struct cpumask *dstp)
|
|
|
|
|
{
|
|
|
|
|
bitmap_zero(cpumask_bits(dstp), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_and - *dstp = *src1p & *src2p
|
|
|
|
|
* @dstp: the cpumask result
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_and(struct cpumask *dstp,
|
|
|
|
|
const struct cpumask *src1p,
|
|
|
|
|
const struct cpumask *src2p)
|
|
|
|
|
{
|
|
|
|
|
bitmap_and(cpumask_bits(dstp), cpumask_bits(src1p),
|
|
|
|
|
cpumask_bits(src2p), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_or - *dstp = *src1p | *src2p
|
|
|
|
|
* @dstp: the cpumask result
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_or(struct cpumask *dstp, const struct cpumask *src1p,
|
|
|
|
|
const struct cpumask *src2p)
|
|
|
|
|
{
|
|
|
|
|
bitmap_or(cpumask_bits(dstp), cpumask_bits(src1p),
|
|
|
|
|
cpumask_bits(src2p), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_xor - *dstp = *src1p ^ *src2p
|
|
|
|
|
* @dstp: the cpumask result
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_xor(struct cpumask *dstp,
|
|
|
|
|
const struct cpumask *src1p,
|
|
|
|
|
const struct cpumask *src2p)
|
|
|
|
|
{
|
|
|
|
|
bitmap_xor(cpumask_bits(dstp), cpumask_bits(src1p),
|
|
|
|
|
cpumask_bits(src2p), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_andnot - *dstp = *src1p & ~*src2p
|
|
|
|
|
* @dstp: the cpumask result
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_andnot(struct cpumask *dstp,
|
|
|
|
|
const struct cpumask *src1p,
|
|
|
|
|
const struct cpumask *src2p)
|
|
|
|
|
{
|
|
|
|
|
bitmap_andnot(cpumask_bits(dstp), cpumask_bits(src1p),
|
|
|
|
|
cpumask_bits(src2p), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_complement - *dstp = ~*srcp
|
|
|
|
|
* @dstp: the cpumask result
|
|
|
|
|
* @srcp: the input to invert
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_complement(struct cpumask *dstp,
|
|
|
|
|
const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
bitmap_complement(cpumask_bits(dstp), cpumask_bits(srcp),
|
|
|
|
|
nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_equal - *src1p == *src2p
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*/
|
|
|
|
|
static inline bool cpumask_equal(const struct cpumask *src1p,
|
|
|
|
|
const struct cpumask *src2p)
|
|
|
|
|
{
|
|
|
|
|
return bitmap_equal(cpumask_bits(src1p), cpumask_bits(src2p),
|
|
|
|
|
nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_intersects - (*src1p & *src2p) != 0
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*/
|
|
|
|
|
static inline bool cpumask_intersects(const struct cpumask *src1p,
|
|
|
|
|
const struct cpumask *src2p)
|
|
|
|
|
{
|
|
|
|
|
return bitmap_intersects(cpumask_bits(src1p), cpumask_bits(src2p),
|
|
|
|
|
nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_subset - (*src1p & ~*src2p) == 0
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*/
|
|
|
|
|
static inline int cpumask_subset(const struct cpumask *src1p,
|
|
|
|
|
const struct cpumask *src2p)
|
|
|
|
|
{
|
|
|
|
|
return bitmap_subset(cpumask_bits(src1p), cpumask_bits(src2p),
|
|
|
|
|
nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_empty - *srcp == 0
|
|
|
|
|
* @srcp: the cpumask to that all cpus < nr_cpu_ids are clear.
|
|
|
|
|
*/
|
|
|
|
|
static inline bool cpumask_empty(const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
return bitmap_empty(cpumask_bits(srcp), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_full - *srcp == 0xFFFFFFFF...
|
|
|
|
|
* @srcp: the cpumask to that all cpus < nr_cpu_ids are set.
|
|
|
|
|
*/
|
|
|
|
|
static inline bool cpumask_full(const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
return bitmap_full(cpumask_bits(srcp), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_weight - Count of bits in *srcp
|
|
|
|
|
* @srcp: the cpumask to count bits (< nr_cpu_ids) in.
|
|
|
|
|
*/
|
|
|
|
|
static inline unsigned int cpumask_weight(const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
return bitmap_weight(cpumask_bits(srcp), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_shift_right - *dstp = *srcp >> n
|
|
|
|
|
* @dstp: the cpumask result
|
|
|
|
|
* @srcp: the input to shift
|
|
|
|
|
* @n: the number of bits to shift by
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_shift_right(struct cpumask *dstp,
|
|
|
|
|
const struct cpumask *srcp, int n)
|
|
|
|
|
{
|
|
|
|
|
bitmap_shift_right(cpumask_bits(dstp), cpumask_bits(srcp), n,
|
|
|
|
|
nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_shift_left - *dstp = *srcp << n
|
|
|
|
|
* @dstp: the cpumask result
|
|
|
|
|
* @srcp: the input to shift
|
|
|
|
|
* @n: the number of bits to shift by
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_shift_left(struct cpumask *dstp,
|
|
|
|
|
const struct cpumask *srcp, int n)
|
|
|
|
|
{
|
|
|
|
|
bitmap_shift_left(cpumask_bits(dstp), cpumask_bits(srcp), n,
|
|
|
|
|
nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_copy - *dstp = *srcp
|
|
|
|
|
* @dstp: the result
|
|
|
|
|
* @srcp: the input cpumask
|
|
|
|
|
*/
|
|
|
|
|
static inline void cpumask_copy(struct cpumask *dstp,
|
|
|
|
|
const struct cpumask *srcp)
|
|
|
|
|
{
|
|
|
|
|
bitmap_copy(cpumask_bits(dstp), cpumask_bits(srcp), nr_cpumask_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_any - pick a "random" cpu from *srcp
|
|
|
|
|
* @srcp: the input cpumask
|
|
|
|
|
*
|
|
|
|
|
* Returns >= nr_cpu_ids if no cpus set.
|
|
|
|
|
*/
|
|
|
|
|
#define cpumask_any(srcp) cpumask_first(srcp)
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_first_and - return the first cpu from *srcp1 & *srcp2
|
|
|
|
|
* @src1p: the first input
|
|
|
|
|
* @src2p: the second input
|
|
|
|
|
*
|
|
|
|
|
* Returns >= nr_cpu_ids if no cpus set in both. See also cpumask_next_and().
|
|
|
|
|
*/
|
|
|
|
|
#define cpumask_first_and(src1p, src2p) cpumask_next_and(-1, (src1p), (src2p))
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_any_and - pick a "random" cpu from *mask1 & *mask2
|
|
|
|
|
* @mask1: the first input cpumask
|
|
|
|
|
* @mask2: the second input cpumask
|
|
|
|
|
*
|
|
|
|
|
* Returns >= nr_cpu_ids if no cpus set.
|
|
|
|
|
*/
|
|
|
|
|
#define cpumask_any_and(mask1, mask2) cpumask_first_and((mask1), (mask2))
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_of - the cpumask containing just a given cpu
|
|
|
|
|
* @cpu: the cpu (<= nr_cpu_ids)
|
|
|
|
|
*/
|
|
|
|
|
#define cpumask_of(cpu) (get_cpu_mask(cpu))
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* to_cpumask - convert an NR_CPUS bitmap to a struct cpumask *
|
|
|
|
|
* @bitmap: the bitmap
|
|
|
|
|
*
|
|
|
|
|
* There are a few places where cpumask_var_t isn't appropriate and
|
|
|
|
|
* static cpumasks must be used (eg. very early boot), yet we don't
|
|
|
|
|
* expose the definition of 'struct cpumask'.
|
|
|
|
|
*
|
|
|
|
|
* This does the conversion, and can be used as a constant initializer.
|
|
|
|
|
*/
|
|
|
|
|
#define to_cpumask(bitmap) \
|
|
|
|
|
((struct cpumask *)(1 ? (bitmap) \
|
|
|
|
|
: (void *)sizeof(__check_is_bitmap(bitmap))))
|
|
|
|
|
|
|
|
|
|
static inline int __check_is_bitmap(const unsigned long *bitmap)
|
|
|
|
|
{
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* cpumask_size - size to allocate for a 'struct cpumask' in bytes
|
|
|
|
|
*
|
|
|
|
|
* This will eventually be a runtime variable, depending on nr_cpu_ids.
|
|
|
|
|
*/
|
|
|
|
|
static inline size_t cpumask_size(void)
|
|
|
|
|
{
|
|
|
|
|
/* FIXME: Once all cpumask assignments are eliminated, this
|
|
|
|
|
* can be nr_cpumask_bits */
|
|
|
|
|
return BITS_TO_LONGS(NR_CPUS) * sizeof(long);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* cpumask_var_t: struct cpumask for stack usage.
|
|
|
|
|
*
|
|
|
|
|
* Oh, the wicked games we play! In order to make kernel coding a
|
|
|
|
|
* little more difficult, we typedef cpumask_var_t to an array or a
|
|
|
|
|
* pointer: doing &mask on an array is a noop, so it still works.
|
|
|
|
|
*
|
|
|
|
|
* ie.
|
|
|
|
|
* cpumask_var_t tmpmask;
|
|
|
|
|
* if (!alloc_cpumask_var(&tmpmask, GFP_KERNEL))
|
|
|
|
|
* return -ENOMEM;
|
|
|
|
|
*
|
|
|
|
|
* ... use 'tmpmask' like a normal struct cpumask * ...
|
|
|
|
|
*
|
|
|
|
|
* free_cpumask_var(tmpmask);
|
|
|
|
|
*/
|
|
|
|
|
#ifdef CONFIG_CPUMASK_OFFSTACK
|
|
|
|
|
typedef struct cpumask *cpumask_var_t;
|
|
|
|
|
|
|
|
|
|
bool alloc_cpumask_var(cpumask_var_t *mask, gfp_t flags);
|
|
|
|
|
void alloc_bootmem_cpumask_var(cpumask_var_t *mask);
|
|
|
|
|
void free_cpumask_var(cpumask_var_t mask);
|
|
|
|
|
void free_bootmem_cpumask_var(cpumask_var_t mask);
|
|
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
typedef struct cpumask cpumask_var_t[1];
|
|
|
|
|
|
|
|
|
|
static inline bool alloc_cpumask_var(cpumask_var_t *mask, gfp_t flags)
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void alloc_bootmem_cpumask_var(cpumask_var_t *mask)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void free_cpumask_var(cpumask_var_t mask)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void free_bootmem_cpumask_var(cpumask_var_t mask)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
#endif /* CONFIG_CPUMASK_OFFSTACK */
|
|
|
|
|
|
|
|
|
|
/* The pointer versions of the maps, these will become the primary versions. */
|
|
|
|
|
#define cpu_possible_mask ((const struct cpumask *)&cpu_possible_map)
|
|
|
|
|
#define cpu_online_mask ((const struct cpumask *)&cpu_online_map)
|
|
|
|
|
#define cpu_present_mask ((const struct cpumask *)&cpu_present_map)
|
|
|
|
|
#define cpu_active_mask ((const struct cpumask *)&cpu_active_map)
|
|
|
|
|
|
|
|
|
|
/* It's common to want to use cpu_all_mask in struct member initializers,
|
|
|
|
|
* so it has to refer to an address rather than a pointer. */
|
|
|
|
|
extern const DECLARE_BITMAP(cpu_all_bits, NR_CPUS);
|
|
|
|
|
#define cpu_all_mask to_cpumask(cpu_all_bits)
|
|
|
|
|
|
|
|
|
|
/* First bits of cpu_bit_bitmap are in fact unset. */
|
|
|
|
|
#define cpu_none_mask to_cpumask(cpu_bit_bitmap[0])
|
|
|
|
|
|
|
|
|
|
/* Wrappers for arch boot code to manipulate normally-constant masks */
|
|
|
|
|
static inline void set_cpu_possible(unsigned int cpu, bool possible)
|
|
|
|
|
{
|
|
|
|
|
if (possible)
|
|
|
|
|
cpumask_set_cpu(cpu, &cpu_possible_map);
|
|
|
|
|
else
|
|
|
|
|
cpumask_clear_cpu(cpu, &cpu_possible_map);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void set_cpu_present(unsigned int cpu, bool present)
|
|
|
|
|
{
|
|
|
|
|
if (present)
|
|
|
|
|
cpumask_set_cpu(cpu, &cpu_present_map);
|
|
|
|
|
else
|
|
|
|
|
cpumask_clear_cpu(cpu, &cpu_present_map);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void set_cpu_online(unsigned int cpu, bool online)
|
|
|
|
|
{
|
|
|
|
|
if (online)
|
|
|
|
|
cpumask_set_cpu(cpu, &cpu_online_map);
|
|
|
|
|
else
|
|
|
|
|
cpumask_clear_cpu(cpu, &cpu_online_map);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void set_cpu_active(unsigned int cpu, bool active)
|
|
|
|
|
{
|
|
|
|
|
if (active)
|
|
|
|
|
cpumask_set_cpu(cpu, &cpu_active_map);
|
|
|
|
|
else
|
|
|
|
|
cpumask_clear_cpu(cpu, &cpu_active_map);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void init_cpu_present(const struct cpumask *src)
|
|
|
|
|
{
|
|
|
|
|
cpumask_copy(&cpu_present_map, src);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void init_cpu_possible(const struct cpumask *src)
|
|
|
|
|
{
|
|
|
|
|
cpumask_copy(&cpu_possible_map, src);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void init_cpu_online(const struct cpumask *src)
|
|
|
|
|
{
|
|
|
|
|
cpumask_copy(&cpu_online_map, src);
|
|
|
|
|
}
|
|
|
|
|
#endif /* __LINUX_CPUMASK_H */
|
|
|
|
|