Release 4.7 kernel/smp.c
/*
* Generic helpers for smp ipi calls
*
* (C) Jens Axboe <jens.axboe@oracle.com> 2008
*/
#include <linux/irq_work.h>
#include <linux/rcupdate.h>
#include <linux/rculist.h>
#include <linux/kernel.h>
#include <linux/export.h>
#include <linux/percpu.h>
#include <linux/init.h>
#include <linux/gfp.h>
#include <linux/smp.h>
#include <linux/cpu.h>
#include <linux/sched.h>
#include "smpboot.h"
enum {
CSD_FLAG_LOCK = 0x01,
CSD_FLAG_SYNCHRONOUS = 0x02,
};
struct call_function_data {
struct call_single_data __percpu *csd;
cpumask_var_t cpumask;
};
static DEFINE_PER_CPU_SHARED_ALIGNED(struct call_function_data, cfd_data);
static DEFINE_PER_CPU_SHARED_ALIGNED(struct llist_head, call_single_queue);
static void flush_smp_call_function_queue(bool warn_cpu_offline);
static int
hotplug_cfd(struct notifier_block *nfb, unsigned long action, void *hcpu)
{
long cpu = (long)hcpu;
struct call_function_data *cfd = &per_cpu(cfd_data, cpu);
switch (action) {
case CPU_UP_PREPARE:
case CPU_UP_PREPARE_FROZEN:
if (!zalloc_cpumask_var_node(&cfd->cpumask, GFP_KERNEL,
cpu_to_node(cpu)))
return notifier_from_errno(-ENOMEM);
cfd->csd = alloc_percpu(struct call_single_data);
if (!cfd->csd) {
free_cpumask_var(cfd->cpumask);
return notifier_from_errno(-ENOMEM);
}
break;
#ifdef CONFIG_HOTPLUG_CPU
case CPU_UP_CANCELED:
case CPU_UP_CANCELED_FROZEN:
/* Fall-through to the CPU_DEAD[_FROZEN] case. */
case CPU_DEAD:
case CPU_DEAD_FROZEN:
free_cpumask_var(cfd->cpumask);
free_percpu(cfd->csd);
break;
case CPU_DYING:
case CPU_DYING_FROZEN:
/*
* The IPIs for the smp-call-function callbacks queued by other
* CPUs might arrive late, either due to hardware latencies or
* because this CPU disabled interrupts (inside stop-machine)
* before the IPIs were sent. So flush out any pending callbacks
* explicitly (without waiting for the IPIs to arrive), to
* ensure that the outgoing CPU doesn't go offline with work
* still pending.
*/
flush_smp_call_function_queue(false);
break;
#endif
};
return NOTIFY_OK;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
peter zijlstra | peter zijlstra | 100 | 62.11% | 1 | 16.67% |
shaohua li | shaohua li | 40 | 24.84% | 1 | 16.67% |
srivatsa s. bhat | srivatsa s. bhat | 14 | 8.70% | 1 | 16.67% |
akinobu mita | akinobu mita | 5 | 3.11% | 1 | 16.67% |
yinghai lu | yinghai lu | 1 | 0.62% | 1 | 16.67% |
xiao guangrong | xiao guangrong | 1 | 0.62% | 1 | 16.67% |
| Total | 161 | 100.00% | 6 | 100.00% |
static struct notifier_block hotplug_cfd_notifier = {
.notifier_call = hotplug_cfd,
};
void __init call_function_init(void)
{
void *cpu = (void *)(long)smp_processor_id();
int i;
for_each_possible_cpu(i)
init_llist_head(&per_cpu(call_single_queue, i));
hotplug_cfd(&hotplug_cfd_notifier, CPU_UP_PREPARE, cpu);
register_cpu_notifier(&hotplug_cfd_notifier);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
peter zijlstra | peter zijlstra | 30 | 53.57% | 1 | 25.00% |
jens axboe | jens axboe | 21 | 37.50% | 1 | 25.00% |
takao indoh | takao indoh | 3 | 5.36% | 1 | 25.00% |
christoph hellwig | christoph hellwig | 2 | 3.57% | 1 | 25.00% |
| Total | 56 | 100.00% | 4 | 100.00% |
/*
* csd_lock/csd_unlock used to serialize access to per-cpu csd resources
*
* For non-synchronous ipi calls the csd can still be in use by the
* previous function call. For multi-cpu calls its even more interesting
* as we'll have to ensure no other cpu is observing our csd.
*/
static __always_inline void csd_lock_wait(struct call_single_data *csd)
{
smp_cond_acquire(!(csd->flags & CSD_FLAG_LOCK));
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
peter zijlstra | peter zijlstra | 13 | 54.17% | 2 | 28.57% |
davidlohr bueso | davidlohr bueso | 6 | 25.00% | 2 | 28.57% |
jens axboe | jens axboe | 2 | 8.33% | 1 | 14.29% |
andrew morton | andrew morton | 2 | 8.33% | 1 | 14.29% |
linus torvalds | linus torvalds | 1 | 4.17% | 1 | 14.29% |
| Total | 24 | 100.00% | 7 | 100.00% |
static __always_inline void csd_lock(struct call_single_data *csd)
{
csd_lock_wait(csd);
csd->flags |= CSD_FLAG_LOCK;
/*
* prevent CPU from reordering the above assignment
* to ->flags with any subsequent assignments to other
* fields of the specified call_single_data structure:
*/
smp_wmb();
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
peter zijlstra | peter zijlstra | 20 | 74.07% | 2 | 28.57% |
andrew morton | andrew morton | 3 | 11.11% | 1 | 14.29% |
ingo molnar | ingo molnar | 1 | 3.70% | 1 | 14.29% |
liguang | liguang | 1 | 3.70% | 1 | 14.29% |
linus torvalds | linus torvalds | 1 | 3.70% | 1 | 14.29% |
davidlohr bueso | davidlohr bueso | 1 | 3.70% | 1 | 14.29% |
| Total | 27 | 100.00% | 7 | 100.00% |
static __always_inline void csd_unlock(struct call_single_data *csd)
{
WARN_ON(!(csd->flags & CSD_FLAG_LOCK));
/*
* ensure we're all done before releasing data:
*/
smp_store_release(&csd->flags, 0);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
peter zijlstra | peter zijlstra | 23 | 65.71% | 1 | 16.67% |
linus torvalds | linus torvalds | 6 | 17.14% | 1 | 16.67% |
andrew morton | andrew morton | 3 | 8.57% | 1 | 16.67% |
davidlohr bueso | davidlohr bueso | 1 | 2.86% | 1 | 16.67% |
jens axboe | jens axboe | 1 | 2.86% | 1 | 16.67% |
ingo molnar | ingo molnar | 1 | 2.86% | 1 | 16.67% |
| Total | 35 | 100.00% | 6 | 100.00% |
static DEFINE_PER_CPU_SHARED_ALIGNED(struct call_single_data, csd_data);
/*
* Insert a previously allocated call_single_data element
* for execution on the given CPU. data must already have
* ->func, ->info, and ->flags set.
*/
static int generic_exec_single(int cpu, struct call_single_data *csd,
smp_call_func_t func, void *info)
{
if (cpu == smp_processor_id()) {
unsigned long flags;
/*
* We can unlock early even for the synchronous on-stack case,
* since we're doing this from the same CPU..
*/
csd_unlock(csd);
local_irq_save(flags);
func(info);
local_irq_restore(flags);
return 0;
}
if ((unsigned)cpu >= nr_cpu_ids || !cpu_online(cpu)) {
csd_unlock(csd);
return -ENXIO;
}
csd->func = func;
csd->info = info;
/*
* The list addition should be visible before sending the IPI
* handler locks the list to pull the entry off it because of
* normal cache coherency rules implied by spinlocks.
*
* If IPIs can go out of order to the cache coherency protocol
* in an architecture, sufficient synchronisation should be added
* to arch code to make it appear to obey cache coherency WRT
* locking and barrier primitives. Generic code isn't really
* equipped to do the right thing...
*/
if (llist_add(&csd->llist, &per_cpu(call_single_queue, cpu)))
arch_send_call_function_single_ipi(cpu);
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
frederic weisbecker | frederic weisbecker | 68 | 55.28% | 1 | 12.50% |
jens axboe | jens axboe | 25 | 20.33% | 1 | 12.50% |
linus torvalds | linus torvalds | 17 | 13.82% | 2 | 25.00% |
christoph hellwig | christoph hellwig | 10 | 8.13% | 2 | 25.00% |
andrew morton | andrew morton | 2 | 1.63% | 1 | 12.50% |
peter zijlstra | peter zijlstra | 1 | 0.81% | 1 | 12.50% |
| Total | 123 | 100.00% | 8 | 100.00% |
/**
* generic_smp_call_function_single_interrupt - Execute SMP IPI callbacks
*
* Invoked by arch to handle an IPI for call function single.
* Must be called with interrupts disabled.
*/
void generic_smp_call_function_single_interrupt(void)
{
flush_smp_call_function_queue(true);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
srivatsa s. bhat | srivatsa s. bhat | 6 | 50.00% | 1 | 50.00% |
jens axboe | jens axboe | 6 | 50.00% | 1 | 50.00% |
| Total | 12 | 100.00% | 2 | 100.00% |
/**
* flush_smp_call_function_queue - Flush pending smp-call-function callbacks
*
* @warn_cpu_offline: If set to 'true', warn if callbacks were queued on an
* offline CPU. Skip this check if set to 'false'.
*
* Flush any pending smp-call-function callbacks queued on this CPU. This is
* invoked by the generic IPI handler, as well as by a CPU about to go offline,
* to ensure that all pending IPI callbacks are run before it goes completely
* offline.
*
* Loop through the call_single_queue and run all the queued callbacks.
* Must be called with interrupts disabled.
*/
static void flush_smp_call_function_queue(bool warn_cpu_offline)
{
struct llist_head *head;
struct llist_node *entry;
struct call_single_data *csd, *csd_next;
static bool warned;
WARN_ON(!irqs_disabled());
head = this_cpu_ptr(&call_single_queue);
entry = llist_del_all(head);
entry = llist_reverse_order(entry);
/* There shouldn't be any pending callbacks on an offline CPU. */
if (unlikely(warn_cpu_offline && !cpu_online(smp_processor_id()) &&
!warned && !llist_empty(head))) {
warned = true;
WARN(1, "IPI on offline CPU %d\n", smp_processor_id());
/*
* We don't have to use the _safe() variant here
* because we are not invoking the IPI handlers yet.
*/
llist_for_each_entry(csd, entry, llist)
pr_warn("IPI callback %pS sent to offline CPU\n",
csd->func);
}
llist_for_each_entry_safe(csd, csd_next, entry, llist) {
smp_call_func_t func = csd->func;
void *info = csd->info;
/* Do we wait until *after* callback? */
if (csd->flags & CSD_FLAG_SYNCHRONOUS) {
func(info);
csd_unlock(csd);
} else {
csd_unlock(csd);
func(info);
}
}
/*
* Handle irq works queued remotely by irq_work_queue_on().
* Smp functions above are typically synchronous so they
* better run first since some other CPUs may be busy waiting
* for them.
*/
irq_work_run();
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
srivatsa s. bhat | srivatsa s. bhat | 89 | 48.63% | 2 | 20.00% |
linus torvalds | linus torvalds | 36 | 19.67% | 1 | 10.00% |
jens axboe | jens axboe | 25 | 13.66% | 1 | 10.00% |
jan kara | jan kara | 13 | 7.10% | 1 | 10.00% |
christoph hellwig | christoph hellwig | 10 | 5.46% | 1 | 10.00% |
frederic weisbecker | frederic weisbecker | 4 | 2.19% | 1 | 10.00% |
andrew morton | andrew morton | 3 | 1.64% | 1 | 10.00% |
christoph lameter | christoph lameter | 2 | 1.09% | 1 | 10.00% |
peter zijlstra | peter zijlstra | 1 | 0.55% | 1 | 10.00% |
| Total | 183 | 100.00% | 10 | 100.00% |
/*
* smp_call_function_single - Run a function on a specific CPU
* @func: The function to run. This must be fast and non-blocking.
* @info: An arbitrary pointer to pass to the function.
* @wait: If true, wait until function has completed on other CPUs.
*
* Returns 0 on success, else a negative status code.
*/
int smp_call_function_single(int cpu, smp_call_func_t func, void *info,
int wait)
{
struct call_single_data *csd;
struct call_single_data csd_stack = { .flags = CSD_FLAG_LOCK | CSD_FLAG_SYNCHRONOUS };
int this_cpu;
int err;
/*
* prevent preemption and reschedule on another processor,
* as well as CPU removal
*/
this_cpu = get_cpu();
/*
* Can deadlock when called with interrupts disabled.
* We allow cpu's that are not yet online though, as no one else can
* send smp call function interrupt to this cpu and as such deadlocks
* can't happen.
*/
WARN_ON_ONCE(cpu_online(this_cpu) && irqs_disabled()
&& !oops_in_progress);
csd = &csd_stack;
if (!wait) {
csd = this_cpu_ptr(&csd_data);
csd_lock(csd);
}
err = generic_exec_single(cpu, csd, func, info);
if (wait)
csd_lock_wait(csd);
put_cpu();
return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
linus torvalds | linus torvalds | 52 | 44.83% | 1 | 11.11% |
jens axboe | jens axboe | 36 | 31.03% | 1 | 11.11% |
ingo molnar | ingo molnar | 11 | 9.48% | 2 | 22.22% |
suresh siddha | suresh siddha | 6 | 5.17% | 1 | 11.11% |
frederic weisbecker | frederic weisbecker | 5 | 4.31% | 1 | 11.11% |
h. peter anvin | h. peter anvin | 4 | 3.45% | 1 | 11.11% |
peter zijlstra | peter zijlstra | 1 | 0.86% | 1 | 11.11% |
david howells | david howells | 1 | 0.86% | 1 | 11.11% |
| Total | 116 | 100.00% | 9 | 100.00% |
EXPORT_SYMBOL(smp_call_function_single);
/**
* smp_call_function_single_async(): Run an asynchronous function on a
* specific CPU.
* @cpu: The CPU to run on.
* @csd: Pre-allocated and setup data structure
*
* Like smp_call_function_single(), but the call is asynchonous and
* can thus be done from contexts with disabled interrupts.
*
* The caller passes his own pre-allocated data structure
* (ie: embedded in an object) and is responsible for synchronizing it
* such that the IPIs performed on the @csd are strictly serialized.
*
* NOTE: Be careful, there is unfortunately no current debugging facility to
* validate the correctness of this serialization.
*/
int smp_call_function_single_async(int cpu, struct call_single_data *csd)
{
int err = 0;
preempt_disable();
/* We could deadlock if we have to wait here with interrupts disabled! */
if (WARN_ON_ONCE(csd->flags & CSD_FLAG_LOCK))
csd_lock_wait(csd);
csd->flags = CSD_FLAG_LOCK;
smp_wmb();
err = generic_exec_single(cpu, csd, csd->func, csd->info);
preempt_enable();
return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
frederic weisbecker | frederic weisbecker | 44 | 62.86% | 3 | 75.00% |
linus torvalds | linus torvalds | 26 | 37.14% | 1 | 25.00% |
| Total | 70 | 100.00% | 4 | 100.00% |
EXPORT_SYMBOL_GPL(smp_call_function_single_async);
/*
* smp_call_function_any - Run a function on any of the given cpus
* @mask: The mask of cpus it can run on.
* @func: The function to run. This must be fast and non-blocking.
* @info: An arbitrary pointer to pass to the function.
* @wait: If true, wait until function has completed.
*
* Returns 0 on success, else a negative status code (if no cpus were online).
*
* Selection preference:
* 1) current cpu if in @mask
* 2) any cpu of current node if in @mask
* 3) any other online cpu in @mask
*/
int smp_call_function_any(const struct cpumask *mask,
smp_call_func_t func, void *info, int wait)
{
unsigned int cpu;
const struct cpumask *nodemask;
int ret;
/* Try for same CPU (cheapest) */
cpu = get_cpu();
if (cpumask_test_cpu(cpu, mask))
goto call;
/* Try for same node. */
nodemask = cpumask_of_node(cpu_to_node(cpu));
for (cpu = cpumask_first_and(nodemask, mask); cpu < nr_cpu_ids;
cpu = cpumask_next_and(cpu, nodemask, mask)) {
if (cpu_online(cpu))
goto call;
}
/* Any online will do: smp_call_function_single handles nr_cpu_ids. */
cpu = cpumask_any_and(mask, cpu_online_mask);
call:
ret = smp_call_function_single(cpu, func, info, wait);
put_cpu();
return ret;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
rusty russell | rusty russell | 128 | 96.97% | 1 | 33.33% |
david john | david john | 3 | 2.27% | 1 | 33.33% |
david howells | david howells | 1 | 0.76% | 1 | 33.33% |
| Total | 132 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL_GPL(smp_call_function_any);
/**
* smp_call_function_many(): Run a function on a set of other CPUs.
* @mask: The set of cpus to run on (only runs on online subset).
* @func: The function to run. This must be fast and non-blocking.
* @info: An arbitrary pointer to pass to the function.
* @wait: If true, wait (atomically) until function has completed
* on other CPUs.
*
* If @wait is true, then returns once @func has returned.
*
* You must not call this function with disabled interrupts or from a
* hardware interrupt handler or from a bottom half handler. Preemption
* must be disabled when calling this function.
*/
void smp_call_function_many(const struct cpumask *mask,
smp_call_func_t func, void *info, bool wait)
{
struct call_function_data *cfd;
int cpu, next_cpu, this_cpu = smp_processor_id();
/*
* Can deadlock when called with interrupts disabled.
* We allow cpu's that are not yet online though, as no one else can
* send smp call function interrupt to this cpu and as such deadlocks
* can't happen.
*/
WARN_ON_ONCE(cpu_online(this_cpu) && irqs_disabled()
&& !oops_in_progress && !early_boot_irqs_disabled);
/* Try to fastpath. So, what's a CPU they want? Ignoring this one. */
cpu = cpumask_first_and(mask, cpu_online_mask);
if (cpu == this_cpu)
cpu = cpumask_next_and(cpu, mask, cpu_online_mask);
/* No online cpus? We're done. */
if (cpu >= nr_cpu_ids)
return;
/* Do we have another CPU which isn't us? */
next_cpu = cpumask_next_and(cpu, mask, cpu_online_mask);
if (next_cpu == this_cpu)
next_cpu = cpumask_next_and(next_cpu, mask, cpu_online_mask);
/* Fastpath: do that cpu by itself. */
if (next_cpu >= nr_cpu_ids) {
smp_call_function_single(cpu, func, info, wait);
return;
}
cfd = this_cpu_ptr(&cfd_data);
cpumask_and(cfd->cpumask, mask, cpu_online_mask);
cpumask_clear_cpu(this_cpu, cfd->cpumask);
/* Some callers race with other cpus changing the passed mask */
if (unlikely(!cpumask_weight(cfd->cpumask)))
return;
for_each_cpu(cpu, cfd->cpumask) {
struct call_single_data *csd = per_cpu_ptr(cfd->csd, cpu);
csd_lock(csd);
if (wait)
csd->flags |= CSD_FLAG_SYNCHRONOUS;
csd->func = func;
csd->info = info;
llist_add(&csd->llist, &per_cpu(call_single_queue, cpu));
}
/* Send a message to all CPUs in the map */
arch_send_call_function_ipi_mask(cfd->cpumask);
if (wait) {
for_each_cpu(cpu, cfd->cpumask) {
struct call_single_data *csd;
csd = per_cpu_ptr(cfd->csd, cpu);
csd_lock_wait(csd);
}
}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
jens axboe | jens axboe | 89 | 30.69% | 1 | 5.56% |
rusty russell | rusty russell | 63 | 21.72% | 2 | 11.11% |
shaohua li | shaohua li | 60 | 20.69% | 1 | 5.56% |
milton d. miller | milton d. miller | 15 | 5.17% | 2 | 11.11% |
andrew morton | andrew morton | 14 | 4.83% | 1 | 5.56% |
linus torvalds | linus torvalds | 10 | 3.45% | 1 | 5.56% |
peter zijlstra | peter zijlstra | 8 | 2.76% | 1 | 5.56% |
ingo molnar | ingo molnar | 8 | 2.76% | 2 | 11.11% |
christoph hellwig | christoph hellwig | 6 | 2.07% | 1 | 5.56% |
suresh siddha | suresh siddha | 6 | 2.07% | 1 | 5.56% |
nick piggin | nick piggin | 4 | 1.38% | 1 | 5.56% |
tejun heo | tejun heo | 3 | 1.03% | 1 | 5.56% |
christoph lameter | christoph lameter | 2 | 0.69% | 1 | 5.56% |
david howells | david howells | 1 | 0.34% | 1 | 5.56% |
roman gushchin | roman gushchin | 1 | 0.34% | 1 | 5.56% |
| Total | 290 | 100.00% | 18 | 100.00% |
EXPORT_SYMBOL(smp_call_function_many);
/**
* smp_call_function(): Run a function on all other CPUs.
* @func: The function to run. This must be fast and non-blocking.
* @info: An arbitrary pointer to pass to the function.
* @wait: If true, wait (atomically) until function has completed
* on other CPUs.
*
* Returns 0.
*
* If @wait is true, then returns once @func has returned; otherwise
* it returns just before the target cpu calls @func.
*
* You must not call this function with disabled interrupts or from a
* hardware interrupt handler or from a bottom half handler.
*/
int smp_call_function(smp_call_func_t func, void *info, int wait)
{
preempt_disable();
smp_call_function_many(cpu_online_mask, func, info, wait);
preempt_enable();
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
jens axboe | jens axboe | 31 | 88.57% | 1 | 33.33% |
rusty russell | rusty russell | 3 | 8.57% | 1 | 33.33% |
david howells | david howells | 1 | 2.86% | 1 | 33.33% |
| Total | 35 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL(smp_call_function);
/* Setup configured maximum number of CPUs to activate */
unsigned int setup_max_cpus = NR_CPUS;
EXPORT_SYMBOL(setup_max_cpus);
/*
* Setup routine for controlling SMP activation
*
* Command-line option of "nosmp" or "maxcpus=0" will disable SMP
* activation entirely (the MPS table probe still happens, though).
*
* Command-line option of "maxcpus=<NUM>", where <NUM> is an integer
* greater than 0, limits the maximum number of CPUs activated in
* SMP mode to <NUM>.
*/
void __weak arch_disable_smp_support(void) { }
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 7 | 100.00% | 1 | 100.00% |
| Total | 7 | 100.00% | 1 | 100.00% |
static int __init nosmp(char *str)
{
setup_max_cpus = 0;
arch_disable_smp_support();
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 21 | 100.00% | 1 | 100.00% |
| Total | 21 | 100.00% | 1 | 100.00% |
early_param("nosmp", nosmp);
/* this is hard limit */
static int __init nrcpus(char *str)
{
int nr_cpus;
get_option(&str, &nr_cpus);
if (nr_cpus > 0 && nr_cpus < nr_cpu_ids)
nr_cpu_ids = nr_cpus;
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 40 | 100.00% | 1 | 100.00% |
| Total | 40 | 100.00% | 1 | 100.00% |
early_param("nr_cpus", nrcpus);
static int __init maxcpus(char *str)
{
get_option(&str, &setup_max_cpus);
if (setup_max_cpus == 0)
arch_disable_smp_support();
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 32 | 100.00% | 1 | 100.00% |
| Total | 32 | 100.00% | 1 | 100.00% |
early_param("maxcpus", maxcpus);
/* Setup number of possible processor ids */
int nr_cpu_ids __read_mostly = NR_CPUS;
EXPORT_SYMBOL(nr_cpu_ids);
/* An arch may set nr_cpu_ids earlier if needed, so this would be redundant */
void __init setup_nr_cpu_ids(void)
{
nr_cpu_ids = find_last_bit(cpumask_bits(cpu_possible_mask),NR_CPUS) + 1;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 22 | 100.00% | 1 | 100.00% |
| Total | 22 | 100.00% | 1 | 100.00% |
void __weak smp_announce(void)
{
printk(KERN_INFO "Brought up %d CPUs\n", num_online_cpus());
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
borislav petkov | borislav petkov | 17 | 100.00% | 1 | 100.00% |
| Total | 17 | 100.00% | 1 | 100.00% |
/* Called by boot processor to activate the rest. */
void __init smp_init(void)
{
unsigned int cpu;
idle_threads_init();
cpuhp_threads_init();
/* FIXME: This should be done in userspace --RR */
for_each_present_cpu(cpu) {
if (num_online_cpus() >= setup_max_cpus)
break;
if (!cpu_online(cpu))
cpu_up(cpu);
}
/* Any cleanup work */
smp_announce();
smp_cpus_done(setup_max_cpus);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 48 | 87.27% | 1 | 25.00% |
suresh siddha | suresh siddha | 3 | 5.45% | 1 | 25.00% |
thomas gleixner | thomas gleixner | 3 | 5.45% | 1 | 25.00% |
borislav petkov | borislav petkov | 1 | 1.82% | 1 | 25.00% |
| Total | 55 | 100.00% | 4 | 100.00% |
/*
* Call a function on all processors. May be used during early boot while
* early_boot_irqs_disabled is set. Use local_irq_save/restore() instead
* of local_irq_disable/enable().
*/
int on_each_cpu(void (*func) (void *info), void *info, int wait)
{
unsigned long flags;
int ret = 0;
preempt_disable();
ret = smp_call_function(func, info, wait);
local_irq_save(flags);
func(info);
local_irq_restore(flags);
preempt_enable();
return ret;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 55 | 82.09% | 1 | 50.00% |
tejun heo | tejun heo | 12 | 17.91% | 1 | 50.00% |
| Total | 67 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(on_each_cpu);
/**
* on_each_cpu_mask(): Run a function on processors specified by
* cpumask, which may include the local processor.
* @mask: The set of cpus to run on (only runs on online subset).
* @func: The function to run. This must be fast and non-blocking.
* @info: An arbitrary pointer to pass to the function.
* @wait: If true, wait (atomically) until function has completed
* on other CPUs.
*
* If @wait is true, then returns once @func has returned.
*
* You must not call this function with disabled interrupts or from a
* hardware interrupt handler or from a bottom half handler. The
* exception is that it may be used during early boot while
* early_boot_irqs_disabled is set.
*/
void on_each_cpu_mask(const struct cpumask *mask, smp_call_func_t func,
void *info, bool wait)
{
int cpu = get_cpu();
smp_call_function_many(mask, func, info, wait);
if (cpumask_test_cpu(cpu, mask)) {
unsigned long flags;
local_irq_save(flags);
func(info);
local_irq_restore(flags);
}
put_cpu();
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
gilad ben-yossef | gilad ben-yossef | 59 | 83.10% | 1 | 50.00% |
david daney | david daney | 12 | 16.90% | 1 | 50.00% |
| Total | 71 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(on_each_cpu_mask);
/*
* on_each_cpu_cond(): Call a function on each processor for which
* the supplied function cond_func returns true, optionally waiting
* for all the required CPUs to finish. This may include the local
* processor.
* @cond_func: A callback function that is passed a cpu id and
* the the info parameter. The function is called
* with preemption disabled. The function should
* return a blooean value indicating whether to IPI
* the specified CPU.
* @func: The function to run on all applicable CPUs.
* This must be fast and non-blocking.
* @info: An arbitrary pointer to pass to both functions.
* @wait: If true, wait (atomically) until function has
* completed on other CPUs.
* @gfp_flags: GFP flags to use when allocating the cpumask
* used internally by the function.
*
* The function might sleep if the GFP flags indicates a non
* atomic allocation is allowed.
*
* Preemption is disabled to protect against CPUs going offline but not online.
* CPUs going online during the call will not be seen or sent an IPI.
*
* You must not call this function with disabled interrupts or
* from a hardware interrupt handler or from a bottom half handler.
*/
void on_each_cpu_cond(bool (*cond_func)(int cpu, void *info),
smp_call_func_t func, void *info, bool wait,
gfp_t gfp_flags)
{
cpumask_var_t cpus;
int cpu, ret;
might_sleep_if(gfpflags_allow_blocking(gfp_flags));
if (likely(zalloc_cpumask_var(&cpus, (gfp_flags|__GFP_NOWARN)))) {
preempt_disable();
for_each_online_cpu(cpu)
if (cond_func(cpu, info))
cpumask_set_cpu(cpu, cpus);
on_each_cpu_mask(cpus, func, info, wait);
preempt_enable();
free_cpumask_var(cpus);
} else {
/*
* No free cpumask, bother. No matter, we'll
* just have to IPI them one by one.
*/
preempt_disable();
for_each_online_cpu(cpu)
if (cond_func(cpu, info)) {
ret = smp_call_function_single(cpu, func,
info, wait);
WARN_ON_ONCE(ret);
}
preempt_enable();
}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
gilad ben-yossef | gilad ben-yossef | 149 | 98.03% | 1 | 50.00% |
mel gorman | mel gorman | 3 | 1.97% | 1 | 50.00% |
| Total | 152 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(on_each_cpu_cond);
static void do_nothing(void *unused)
{
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
thomas gleixner | thomas gleixner | 9 | 100.00% | 1 | 100.00% |
| Total | 9 | 100.00% | 1 | 100.00% |
/**
* kick_all_cpus_sync - Force all cpus out of idle
*
* Used to synchronize the update of pm_idle function pointer. It's
* called after the pointer is updated and returns after the dummy
* callback function has been executed on all cpus. The execution of
* the function can only happen on the remote cpus after they have
* left the idle function which had been called via pm_idle function
* pointer. So it's guaranteed that nothing uses the previous pointer
* anymore.
*/
void kick_all_cpus_sync(void)
{
/* Make sure the change is visible before we kick the cpus */
smp_mb();
smp_call_function(do_nothing, NULL, 1);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
thomas gleixner | thomas gleixner | 20 | 100.00% | 1 | 100.00% |
| Total | 20 | 100.00% | 1 | 100.00% |
EXPORT_SYMBOL_GPL(kick_all_cpus_sync);
/**
* wake_up_all_idle_cpus - break all cpus out of idle
* wake_up_all_idle_cpus try to break all cpus which is in idle state even
* including idle polling cpus, for non-idle cpus, we will do nothing
* for them.
*/
void wake_up_all_idle_cpus(void)
{
int cpu;
preempt_disable();
for_each_online_cpu(cpu) {
if (cpu == smp_processor_id())
continue;
wake_up_if_idle(cpu);
}
preempt_enable();
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
liu chuansheng | liu chuansheng | 35 | 100.00% | 1 | 100.00% |
| Total | 35 | 100.00% | 1 | 100.00% |
EXPORT_SYMBOL_GPL(wake_up_all_idle_cpus);
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp |
americo wang | americo wang | 279 | 13.72% | 2 | 3.45% |
jens axboe | jens axboe | 277 | 13.62% | 2 | 3.45% |
peter zijlstra | peter zijlstra | 221 | 10.87% | 2 | 3.45% |
gilad ben-yossef | gilad ben-yossef | 219 | 10.77% | 2 | 3.45% |
rusty russell | rusty russell | 201 | 9.88% | 3 | 5.17% |
linus torvalds | linus torvalds | 152 | 7.47% | 3 | 5.17% |
frederic weisbecker | frederic weisbecker | 139 | 6.83% | 5 | 8.62% |
srivatsa s. bhat | srivatsa s. bhat | 119 | 5.85% | 2 | 3.45% |
shaohua li | shaohua li | 102 | 5.01% | 1 | 1.72% |
liu chuansheng | liu chuansheng | 44 | 2.16% | 1 | 1.72% |
thomas gleixner | thomas gleixner | 38 | 1.87% | 2 | 3.45% |
ingo molnar | ingo molnar | 31 | 1.52% | 2 | 3.45% |
christoph hellwig | christoph hellwig | 29 | 1.43% | 2 | 3.45% |
andrew morton | andrew morton | 27 | 1.33% | 1 | 1.72% |
milton d. miller | milton d. miller | 26 | 1.28% | 3 | 5.17% |
tejun heo | tejun heo | 19 | 0.93% | 2 | 3.45% |
borislav petkov | borislav petkov | 18 | 0.88% | 1 | 1.72% |
suresh siddha | suresh siddha | 18 | 0.88% | 2 | 3.45% |
david daney | david daney | 13 | 0.64% | 1 | 1.72% |
jan kara | jan kara | 13 | 0.64% | 1 | 1.72% |
davidlohr bueso | davidlohr bueso | 8 | 0.39% | 2 | 3.45% |
akinobu mita | akinobu mita | 5 | 0.25% | 1 | 1.72% |
h. peter anvin | h. peter anvin | 4 | 0.20% | 1 | 1.72% |
christoph lameter | christoph lameter | 4 | 0.20% | 1 | 1.72% |
nick piggin | nick piggin | 4 | 0.20% | 1 | 1.72% |
david howells | david howells | 4 | 0.20% | 1 | 1.72% |
sheng yang | sheng yang | 3 | 0.15% | 1 | 1.72% |
takao indoh | takao indoh | 3 | 0.15% | 1 | 1.72% |
david john | david john | 3 | 0.15% | 1 | 1.72% |
mel gorman | mel gorman | 3 | 0.15% | 1 | 1.72% |
steven rostedt | steven rostedt | 2 | 0.10% | 1 | 1.72% |
roman gushchin | roman gushchin | 1 | 0.05% | 1 | 1.72% |
xiao guangrong | xiao guangrong | 1 | 0.05% | 1 | 1.72% |
yinghai lu | yinghai lu | 1 | 0.05% | 1 | 1.72% |
liguang | liguang | 1 | 0.05% | 1 | 1.72% |
paul gortmaker | paul gortmaker | 1 | 0.05% | 1 | 1.72% |
xie xiuqi | xie xiuqi | 1 | 0.05% | 1 | 1.72% |
| Total | 2034 | 100.00% | 58 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.