summaryrefslogtreecommitdiff
path: root/kern/ipc_sched.c
diff options
context:
space:
mode:
Diffstat (limited to 'kern/ipc_sched.c')
-rw-r--r--kern/ipc_sched.c287
1 files changed, 287 insertions, 0 deletions
diff --git a/kern/ipc_sched.c b/kern/ipc_sched.c
new file mode 100644
index 0000000..a2f4c35
--- /dev/null
+++ b/kern/ipc_sched.c
@@ -0,0 +1,287 @@
+/*
+ * Mach Operating System
+ * Copyright (c) 1993, 1992,1991,1990 Carnegie Mellon University
+ * All Rights Reserved.
+ *
+ * Permission to use, copy, modify and distribute this software and its
+ * documentation is hereby granted, provided that both the copyright
+ * notice and this permission notice appear in all copies of the
+ * software, derivative works or modified versions, and any portions
+ * thereof, and that both notices appear in supporting documentation.
+ *
+ * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
+ * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
+ * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
+ *
+ * Carnegie Mellon requests users of this software to return to
+ *
+ * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
+ * School of Computer Science
+ * Carnegie Mellon University
+ * Pittsburgh PA 15213-3890
+ *
+ * any improvements or extensions that they make and grant Carnegie Mellon
+ * the rights to redistribute these changes.
+ */
+
+#include <cpus.h>
+#include <mach_host.h>
+
+#include <mach/message.h>
+#include <kern/counters.h>
+#include "cpu_number.h"
+#include <kern/lock.h>
+#include <kern/thread.h>
+#include <kern/sched_prim.h>
+#include <kern/processor.h>
+#include <kern/time_out.h>
+#include <kern/thread_swap.h>
+#include <kern/ipc_sched.h>
+#include <machine/machspl.h> /* for splsched/splx */
+#include <machine/pmap.h>
+
+
+
+/*
+ * These functions really belong in kern/sched_prim.c.
+ */
+
+/*
+ * Routine: thread_go
+ * Purpose:
+ * Start a thread running.
+ * Conditions:
+ * IPC locks may be held.
+ */
+
+void
+thread_go(
+ thread_t thread)
+{
+ int state;
+ spl_t s;
+
+ s = splsched();
+ thread_lock(thread);
+
+ reset_timeout_check(&thread->timer);
+
+ state = thread->state;
+ switch (state & TH_SCHED_STATE) {
+
+ case TH_WAIT | TH_SUSP | TH_UNINT:
+ case TH_WAIT | TH_UNINT:
+ case TH_WAIT:
+ /*
+ * Sleeping and not suspendable - put
+ * on run queue.
+ */
+ thread->state = (state &~ TH_WAIT) | TH_RUN;
+ thread->wait_result = THREAD_AWAKENED;
+ thread_setrun(thread, TRUE);
+ break;
+
+ case TH_WAIT | TH_SUSP:
+ case TH_RUN | TH_WAIT:
+ case TH_RUN | TH_WAIT | TH_SUSP:
+ case TH_RUN | TH_WAIT | TH_UNINT:
+ case TH_RUN | TH_WAIT | TH_SUSP | TH_UNINT:
+ /*
+ * Either already running, or suspended.
+ */
+ thread->state = state & ~TH_WAIT;
+ thread->wait_result = THREAD_AWAKENED;
+ break;
+
+ default:
+ /*
+ * Not waiting.
+ */
+ break;
+ }
+
+ thread_unlock(thread);
+ splx(s);
+}
+
+/*
+ * Routine: thread_will_wait
+ * Purpose:
+ * Assert that the thread intends to block.
+ */
+
+void
+thread_will_wait(
+ thread_t thread)
+{
+ spl_t s;
+
+ s = splsched();
+ thread_lock(thread);
+
+ assert(thread->wait_result = -1); /* for later assertions */
+ thread->state |= TH_WAIT;
+
+ thread_unlock(thread);
+ splx(s);
+}
+
+/*
+ * Routine: thread_will_wait_with_timeout
+ * Purpose:
+ * Assert that the thread intends to block,
+ * with a timeout.
+ */
+
+void
+thread_will_wait_with_timeout(
+ thread_t thread,
+ mach_msg_timeout_t msecs)
+{
+ natural_t ticks = convert_ipc_timeout_to_ticks(msecs);
+ spl_t s;
+
+ s = splsched();
+ thread_lock(thread);
+
+ assert(thread->wait_result = -1); /* for later assertions */
+ thread->state |= TH_WAIT;
+
+ set_timeout(&thread->timer, ticks);
+
+ thread_unlock(thread);
+ splx(s);
+}
+
+#if MACH_HOST
+#define check_processor_set(thread) \
+ (current_processor()->processor_set == (thread)->processor_set)
+#else /* MACH_HOST */
+#define check_processor_set(thread) TRUE
+#endif /* MACH_HOST */
+
+#if NCPUS > 1
+#define check_bound_processor(thread) \
+ ((thread)->bound_processor == PROCESSOR_NULL || \
+ (thread)->bound_processor == current_processor())
+#else /* NCPUS > 1 */
+#define check_bound_processor(thread) TRUE
+#endif /* NCPUS > 1 */
+
+#ifdef CONTINUATIONS
+/*
+ * Routine: thread_handoff
+ * Purpose:
+ * Switch to a new thread (new), leaving the current
+ * thread (old) blocked. If successful, moves the
+ * kernel stack from old to new and returns as the
+ * new thread. An explicit continuation for the old thread
+ * must be supplied.
+ *
+ * NOTE: Although we wakeup new, we don't set new->wait_result.
+ * Returns:
+ * TRUE if the handoff happened.
+ */
+
+boolean_t
+thread_handoff(
+ register thread_t old,
+ register continuation_t continuation,
+ register thread_t new)
+{
+ spl_t s;
+
+ assert(current_thread() == old);
+
+ /*
+ * XXX Dubious things here:
+ * I don't check the idle_count on the processor set.
+ * No scheduling priority or policy checks.
+ * I assume the new thread is interruptible.
+ */
+
+ s = splsched();
+ thread_lock(new);
+
+ /*
+ * The first thing we must do is check the state
+ * of the threads, to ensure we can handoff.
+ * This check uses current_processor()->processor_set,
+ * which we can read without locking.
+ */
+
+ if ((old->stack_privilege == current_stack()) ||
+ (new->state != (TH_WAIT|TH_SWAPPED)) ||
+ !check_processor_set(new) ||
+ !check_bound_processor(new)) {
+ thread_unlock(new);
+ (void) splx(s);
+
+ counter_always(c_thread_handoff_misses++);
+ return FALSE;
+ }
+
+ reset_timeout_check(&new->timer);
+
+ new->state = TH_RUN;
+ thread_unlock(new);
+
+#if NCPUS > 1
+ new->last_processor = current_processor();
+#endif /* NCPUS > 1 */
+
+ ast_context(new, cpu_number());
+ timer_switch(&new->system_timer);
+
+ /*
+ * stack_handoff is machine-dependent. It does the
+ * machine-dependent components of a context-switch, like
+ * changing address spaces. It updates active_threads.
+ */
+
+ stack_handoff(old, new);
+
+ /*
+ * Now we must dispose of the old thread.
+ * This is like thread_continue, except
+ * that the old thread isn't waiting yet.
+ */
+
+ thread_lock(old);
+ old->swap_func = continuation;
+ assert(old->wait_result = -1); /* for later assertions */
+
+ if (old->state == TH_RUN) {
+ /*
+ * This is our fast path.
+ */
+
+ old->state = TH_WAIT|TH_SWAPPED;
+ }
+ else if (old->state == (TH_RUN|TH_SUSP)) {
+ /*
+ * Somebody is trying to suspend the thread.
+ */
+
+ old->state = TH_WAIT|TH_SUSP|TH_SWAPPED;
+ if (old->wake_active) {
+ /*
+ * Someone wants to know when the thread
+ * really stops.
+ */
+ old->wake_active = FALSE;
+ thread_unlock(old);
+ thread_wakeup((event_t)&old->wake_active);
+ goto after_old_thread;
+ }
+ } else
+ panic("thread_handoff");
+
+ thread_unlock(old);
+ after_old_thread:
+ (void) splx(s);
+
+ counter_always(c_thread_handoff_hits++);
+ return TRUE;
+}
+#endif /* CONTINUATIONS */