summaryrefslogtreecommitdiff
path: root/sys/kern/subr_smp.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/kern/subr_smp.c')
-rw-r--r--sys/kern/subr_smp.c41
1 files changed, 41 insertions, 0 deletions
diff --git a/sys/kern/subr_smp.c b/sys/kern/subr_smp.c
index 818858909d717..93df59f32ee09 100644
--- a/sys/kern/subr_smp.c
+++ b/sys/kern/subr_smp.c
@@ -884,6 +884,47 @@ smp_no_rendezvous_barrier(void *dummy)
#endif
}
+void
+smp_rendezvous_cpus_retry(cpuset_t map,
+ void (* setup_func)(void *),
+ void (* action_func)(void *),
+ void (* teardown_func)(void *),
+ void (* wait_func)(void *, int),
+ struct smp_rendezvous_cpus_retry_arg *arg)
+{
+ int cpu;
+
+ /*
+ * Execute an action on all specified CPUs while retrying until they
+ * all acknowledge completion.
+ */
+ CPU_COPY(&map, &arg->cpus);
+ for (;;) {
+ smp_rendezvous_cpus(
+ arg->cpus,
+ setup_func,
+ action_func,
+ teardown_func,
+ arg);
+
+ if (CPU_EMPTY(&arg->cpus))
+ break;
+
+ CPU_FOREACH(cpu) {
+ if (!CPU_ISSET(cpu, &arg->cpus))
+ continue;
+ wait_func(arg, cpu);
+ }
+ }
+}
+
+void
+smp_rendezvous_cpus_done(struct smp_rendezvous_cpus_retry_arg *arg)
+{
+
+ CPU_CLR_ATOMIC(curcpu, &arg->cpus);
+}
+
/*
* Wait for specified idle threads to switch once. This ensures that even
* preempted threads have cycled through the switch function once,