409 lines
11 KiB
C
409 lines
11 KiB
C
/* $NetBSD: kern_kcont.c,v 1.10 2004/04/25 16:42:41 simonb Exp $ */
|
|
|
|
/*
|
|
* Copyright 2003 Jonathan Stone.
|
|
* All rights reserved.
|
|
* Copyright (c) 2004 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by Jonathan Stone.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of The NetBSD Foundation nor the names of its
|
|
* contributors may be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/*
|
|
*/
|
|
#include <sys/cdefs.h>
|
|
__KERNEL_RCSID(0, "$NetBSD: kern_kcont.c,v 1.10 2004/04/25 16:42:41 simonb Exp $ ");
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#include <sys/queue.h>
|
|
#include <sys/errno.h>
|
|
#include <sys/malloc.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/pool.h>
|
|
#include <sys/kthread.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/systm.h>
|
|
#include <lib/libkern/libkern.h>
|
|
|
|
#include <machine/intr.h> /* IPL_*, and schedsoftnet() */
|
|
/* XXX: schedsofnet() should die. */
|
|
|
|
#include <sys/kcont.h>
|
|
|
|
|
|
/* Accessors for struct kc_queue */
|
|
static __inline struct kc *kc_set(struct kc *,
|
|
void (*func)(void *, void *, int),
|
|
void *env_arg, int ipl);
|
|
|
|
static __inline void kcont_enqueue_atomic(kcq_t *kcq, struct kc *kc);
|
|
static __inline struct kc *kcont_dequeue_atomic(kcq_t *kcq);
|
|
|
|
|
|
static void kcont_worker(void * /*arg*/);
|
|
static void kcont_create_worker(void *);
|
|
|
|
|
|
#ifdef __HAVE_GENERIC_SOFT_INTERRUPTS
|
|
/*
|
|
* Software-interrupt priority continuation queues.
|
|
*/
|
|
static kcq_t kcq_softnet;
|
|
static kcq_t kcq_softclock;
|
|
static kcq_t kcq_softserial;
|
|
|
|
static void *kc_si_softnet;
|
|
static void *kc_si_softclock;
|
|
static void *kc_si_softserial;
|
|
#endif /* __HAVE_GENERIC_SOFT_INTERRUPTS */
|
|
|
|
/*
|
|
* Pool allocator structure.
|
|
*/
|
|
POOL_INIT(kc_pool, sizeof(struct kc), 0, 0, 0, "kcpl", NULL);
|
|
|
|
/*
|
|
* Process-context continuation queue.
|
|
*/
|
|
static kcq_t kcq_process_ctxt;
|
|
|
|
|
|
/*
|
|
* Insert/Remove a fully-formed struct kc * into the kc_queue *
|
|
* of some kernel object (e.g., a struct buf *).
|
|
* For fine-grained SMP, both enqueueing and dequeueing will
|
|
* need a locking mechanism.
|
|
*/
|
|
static __inline void
|
|
kcont_enqueue_atomic(kcq_t *kcq, struct kc *kc)
|
|
{
|
|
int s;
|
|
|
|
s = splvm();
|
|
SIMPLEQ_INSERT_TAIL(kcq, kc, kc_next);
|
|
splx(s);
|
|
}
|
|
|
|
static __inline struct kc *
|
|
kcont_dequeue_atomic(kcq_t *kcq)
|
|
{
|
|
struct kc *kc;
|
|
int s;
|
|
|
|
s = splvm();
|
|
kc = SIMPLEQ_FIRST(kcq);
|
|
if (kc != NULL) {
|
|
SIMPLEQ_REMOVE_HEAD(kcq, kc_next);
|
|
SIMPLEQ_NEXT(kc, kc_next) = NULL;
|
|
}
|
|
splx(s);
|
|
return kc;
|
|
}
|
|
|
|
/*
|
|
* Construct a continuation object from pre-allocated memory.
|
|
* Used by functions that are about call an asynchronous operation,
|
|
* to build a continuation to be called once the operation completes.
|
|
*/
|
|
static __inline struct kc *
|
|
kc_set(struct kc *kc, void (*func)(void *, void *, int),
|
|
void *env_arg, int ipl)
|
|
{
|
|
|
|
kc->kc_fn = func;
|
|
kc->kc_env_arg = env_arg;
|
|
kc->kc_ipl = ipl;
|
|
kc->kc_flags = 0;
|
|
#ifdef DEBUG
|
|
kc->kc_obj = NULL;
|
|
kc->kc_status = -1;
|
|
SIMPLEQ_NEXT(kc, kc_next) = NULL;
|
|
#endif
|
|
return kc;
|
|
}
|
|
|
|
/*
|
|
* Request a continuation. Caller provides space for the struct kc *.
|
|
*/
|
|
struct kc *
|
|
kcont(struct kc *kc, void (*func)(void *, void *, int),
|
|
void *env_arg, int continue_ipl)
|
|
{
|
|
|
|
/* Just save the arguments in the kcont *. */
|
|
return kc_set(kc, func, env_arg, continue_ipl);
|
|
}
|
|
|
|
/*
|
|
* Request a malloc'ed/auto-freed continuation. The kcont framework
|
|
* mallocs the struct kc, and initializes it with the caller-supplied args.
|
|
* Once the asynchronous operation completes and the continuation function
|
|
* has been called, the kcont framework will free the struct kc *
|
|
* immediately after the continuation function returns.
|
|
*/
|
|
struct kc *
|
|
kcont_malloc(int malloc_flags,
|
|
void (*func)(void *, void *, int),
|
|
void *env_arg, int continue_ipl)
|
|
{
|
|
struct kc *kc;
|
|
int pool_flags;
|
|
|
|
pool_flags = (malloc_flags & M_NOWAIT) ? 0 : PR_WAITOK;
|
|
pool_flags |= (malloc_flags & M_CANFAIL) ? PR_LIMITFAIL : 0;
|
|
|
|
kc = pool_get(&kc_pool, pool_flags);
|
|
if (kc == NULL)
|
|
return kc;
|
|
return kc_set(kc, func, env_arg, continue_ipl);
|
|
}
|
|
|
|
/*
|
|
* Dispatch a dequeued continuation which requested deferral
|
|
* into the appropriate lower-priority queue.
|
|
* Public API entry to defer execution of a pre-built kcont.
|
|
*/
|
|
void
|
|
kcont_defer(struct kc *kc, void *obj, int status)
|
|
{
|
|
/*
|
|
* IPL at which to synchronize access to object is
|
|
* above the continuer's requested callback IPL,
|
|
* (e.g., continuer wants IPL_SOFTNET but the object
|
|
* holding this continuation incurred the wakeup()able
|
|
* event whilst at IPL_BIO).
|
|
* Defer this kc to a lower-priority kc queue,
|
|
* to be serviced slightly later at a lower IPL.
|
|
*/
|
|
|
|
/*
|
|
* If we already deferred this kcont, don't clobber
|
|
* the previously-saved kc_object and kc_status.
|
|
* (The obj/status arguments passed in by ck_run() should
|
|
* be the same as kc_object/kc_status, but don't rely on that.)
|
|
*/
|
|
if ((kc->kc_flags & KC_DEFERRED) == 0) {
|
|
kc->kc_flags |= KC_DEFERRED;
|
|
kc->kc_obj = obj;
|
|
kc->kc_status = status;
|
|
}
|
|
|
|
switch (kc->kc_ipl) {
|
|
#ifdef __HAVE_GENERIC_SOFT_INTERRUPTS
|
|
case KC_IPL_DEFER_SOFTCLOCK:
|
|
kcont_enqueue_atomic(&kcq_softclock, kc);
|
|
softintr_schedule(kc_si_softclock);
|
|
break;
|
|
case KC_IPL_DEFER_SOFTNET:
|
|
kcont_enqueue_atomic(&kcq_softnet, kc);
|
|
softintr_schedule(kc_si_softnet);
|
|
break;
|
|
case KC_IPL_DEFER_SOFTSERIAL:
|
|
kcont_enqueue_atomic(&kcq_softserial, kc);
|
|
softintr_schedule(kc_si_softserial);
|
|
break;
|
|
|
|
#else /* !__HAVE_GENERIC_SOFT_INTERRUPTS */
|
|
/* What to do? For now, punt to process context */
|
|
case KC_IPL_DEFER_SOFTCLOCK:
|
|
case KC_IPL_DEFER_SOFTSERIAL:
|
|
case KC_IPL_DEFER_SOFTNET:
|
|
/*FALLTHROUGH*/
|
|
#endif /* __HAVE_GENERIC_SOFT_INTERRUPTS */
|
|
|
|
case KC_IPL_DEFER_PROCESS:
|
|
kcont_enqueue_atomic(&kcq_process_ctxt, kc);
|
|
wakeup(&kcq_process_ctxt);
|
|
break;
|
|
default:
|
|
KASSERT(0);
|
|
}
|
|
}
|
|
|
|
void
|
|
kcont_defer_malloc(int mallocflags,
|
|
void (*func)(void *, void *, int),
|
|
void *obj, void *env_arg, int status, int ipl)
|
|
{
|
|
struct kc *kc;
|
|
|
|
kc = kcont_malloc(mallocflags, func, env_arg, ipl);
|
|
if (kc != NULL)
|
|
kcont_defer(kc, obj, status);
|
|
}
|
|
|
|
/*
|
|
* Enqueue a pre-existing kcont onto a struct kcq completion queue
|
|
* of some pre-existing kernel object.
|
|
*/
|
|
void
|
|
kcont_enqueue(kcq_t *kcq, struct kc *kc)
|
|
{
|
|
|
|
kcont_enqueue_atomic(kcq, kc);
|
|
}
|
|
|
|
|
|
/*
|
|
* Run through a list of continuations, calling (or handing off)
|
|
* continuation functions.
|
|
* If the caller-provided IPL is the same as the requested IPL,
|
|
* deliver the callback.
|
|
* If the caller-provided IPL is higher than the requested
|
|
* callback IPL, re-enqueue the continuation to a lower-priority queue.
|
|
*/
|
|
void
|
|
kcont_run(kcq_t *kcq, void *obj, int status, int curipl)
|
|
{
|
|
struct kc *kc;
|
|
|
|
while ((kc = kcont_dequeue_atomic(kcq)) != NULL) {
|
|
|
|
/* If execution of kc was already deferred, restore context. */
|
|
if (kc->kc_flags & KC_DEFERRED) {
|
|
KASSERT(obj == NULL);
|
|
obj = kc->kc_obj;
|
|
status = kc->kc_status;
|
|
}
|
|
|
|
/* Check whether to execute now or to defer. */
|
|
if (kc->kc_ipl == KC_IPL_IMMED || curipl <= kc->kc_ipl) {
|
|
int saved_flags = kc->kc_flags; /* XXX see below */
|
|
|
|
/* Satisfy our raison d'e^tre */
|
|
(*kc->kc_fn)(obj, kc->kc_env_arg, status);
|
|
|
|
/*
|
|
* We must not touch (*kc) after calling
|
|
* (*kc->kc_fn), unless we were specifically
|
|
* asked to free it. The memory for (*kc) may
|
|
* be a sub-field of some other object (for example,
|
|
* of kc->kc_env_arg) and (*kc_fn)() may already
|
|
* have freed it by the time we get here. So save
|
|
* kc->kc_flags (above) and use that saved copy
|
|
* to test for auto-free.
|
|
*/
|
|
if (saved_flags & KC_AUTOFREE)
|
|
pool_put(&kc_pool, kc);
|
|
} else {
|
|
kcont_defer(kc, obj, status);
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef __HAVE_GENERIC_SOFT_INTERRUPTS
|
|
/*
|
|
* Trampolines for processing software-interrupt kcont queues.
|
|
*/
|
|
static void
|
|
kcont_run_softclock(void *arg)
|
|
{
|
|
|
|
kcont_run((struct kcqueue *)arg, NULL, 0, KC_IPL_DEFER_SOFTCLOCK);
|
|
}
|
|
|
|
static void
|
|
kcont_run_softnet(void *arg)
|
|
{
|
|
|
|
kcont_run((struct kcqueue *)arg, NULL, 0, KC_IPL_DEFER_SOFTNET);
|
|
}
|
|
|
|
static void
|
|
kcont_run_softserial(void *arg)
|
|
{
|
|
|
|
kcont_run((struct kcqueue *)arg, NULL, 0, KC_IPL_DEFER_SOFTSERIAL);
|
|
}
|
|
#endif /* __HAVE_GENERIC_SOFT_INTERRUPTS */
|
|
|
|
|
|
static void
|
|
kcont_create_worker(void *arg)
|
|
{
|
|
if (kthread_create1(kcont_worker, NULL, NULL, "kcont"))
|
|
panic("fork kcont");
|
|
}
|
|
|
|
/*
|
|
* Main entrypoint for kcont worker kthreads to execute
|
|
* a continuation which requested deferral to process context.
|
|
*/
|
|
static void
|
|
kcont_worker(void *arg)
|
|
{
|
|
int status;
|
|
|
|
(void)arg; /* kill GCC warning */
|
|
|
|
while (1) {
|
|
status = ltsleep(&kcq_process_ctxt, PCATCH, "kcont", hz, NULL);
|
|
if (status != 0 && status != EWOULDBLOCK)
|
|
break;
|
|
kcont_run(&kcq_process_ctxt, NULL, 0, KC_IPL_DEFER_PROCESS);
|
|
}
|
|
kthread_exit(0);
|
|
}
|
|
|
|
|
|
/*
|
|
* Initialize kcont subsystem.
|
|
*/
|
|
void
|
|
kcont_init(void)
|
|
{
|
|
|
|
#ifdef __HAVE_GENERIC_SOFT_INTERRUPTS
|
|
/*
|
|
* Initialize kc_queue and callout for soft-int deferred
|
|
* continuations. (If not available, deferrals fall back
|
|
* to deferring all the way to process context).
|
|
*/
|
|
SIMPLEQ_INIT(&kcq_softclock);
|
|
kc_si_softclock = softintr_establish(IPL_SOFTCLOCK,
|
|
kcont_run_softclock, &kcq_softnet);
|
|
|
|
SIMPLEQ_INIT(&kcq_softnet);
|
|
kc_si_softnet = softintr_establish(IPL_SOFTNET,
|
|
kcont_run_softnet, &kcq_softnet);
|
|
|
|
SIMPLEQ_INIT(&kcq_softserial);
|
|
kc_si_softserial = softintr_establish(IPL_SOFTSERIAL,
|
|
kcont_run_softserial, &kcq_softserial);
|
|
#endif /* __HAVE_GENERIC_SOFT_INTERRUPTS */
|
|
|
|
/*
|
|
* Create kc_queue for process-context continuations, and
|
|
* a worker kthread to process the queue. (Fine-grained SMP
|
|
* locking should have at least one worker kthread per CPU).
|
|
*/
|
|
SIMPLEQ_INIT(&kcq_process_ctxt);
|
|
kthread_create(kcont_create_worker, NULL);
|
|
}
|