Update iflib to support more NIC designs
- Move group task queue into kern/subr_gtaskqueue.c - Change intr_enable to return an int so it can be detected if it's not implemented - Allow different TX/RX queues per set to be different sizes - Don't split up TX mbufs before transmit - Allow a completion queue for TX as well as RX - Pass the RX budget to isc_rxd_available() to allow an earlier return and avoid multiple calls Submitted by: shurd Reviewed by: gallatin Approved by: scottl Differential Revision: https://reviews.freebsd.org/D7393
This commit is contained in:
parent
bda1739fd3
commit
181dc4875b
@ -3349,6 +3349,7 @@ kern/subr_disk.c standard
|
||||
kern/subr_eventhandler.c standard
|
||||
kern/subr_fattime.c standard
|
||||
kern/subr_firmware.c optional firmware
|
||||
kern/subr_gtaskqueue.c standard
|
||||
kern/subr_hash.c standard
|
||||
kern/subr_hints.c standard
|
||||
kern/subr_kdb.c standard
|
||||
|
864
sys/kern/subr_gtaskqueue.c
Normal file
864
sys/kern/subr_gtaskqueue.c
Normal file
@ -0,0 +1,864 @@
|
||||
/*-
|
||||
* Copyright (c) 2000 Doug Rabson
|
||||
* Copyright (c) 2014 Jeff Roberson
|
||||
* Copyright (c) 2016 Matthew Macy
|
||||
* All rights reserved.
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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>
|
||||
__FBSDID("$FreeBSD$");
|
||||
|
||||
#include <sys/param.h>
|
||||
#include <sys/systm.h>
|
||||
#include <sys/bus.h>
|
||||
#include <sys/cpuset.h>
|
||||
#include <sys/interrupt.h>
|
||||
#include <sys/kernel.h>
|
||||
#include <sys/kthread.h>
|
||||
#include <sys/libkern.h>
|
||||
#include <sys/limits.h>
|
||||
#include <sys/lock.h>
|
||||
#include <sys/malloc.h>
|
||||
#include <sys/mutex.h>
|
||||
#include <sys/proc.h>
|
||||
#include <sys/sched.h>
|
||||
#include <sys/smp.h>
|
||||
#include <sys/gtaskqueue.h>
|
||||
#include <sys/unistd.h>
|
||||
#include <machine/stdarg.h>
|
||||
|
||||
static MALLOC_DEFINE(M_GTASKQUEUE, "taskqueue", "Task Queues");
|
||||
static void gtaskqueue_thread_enqueue(void *);
|
||||
static void gtaskqueue_thread_loop(void *arg);
|
||||
|
||||
|
||||
struct gtaskqueue_busy {
|
||||
struct gtask *tb_running;
|
||||
TAILQ_ENTRY(gtaskqueue_busy) tb_link;
|
||||
};
|
||||
|
||||
static struct gtask * const TB_DRAIN_WAITER = (struct gtask *)0x1;
|
||||
|
||||
struct gtaskqueue {
|
||||
STAILQ_HEAD(, gtask) tq_queue;
|
||||
gtaskqueue_enqueue_fn tq_enqueue;
|
||||
void *tq_context;
|
||||
char *tq_name;
|
||||
TAILQ_HEAD(, gtaskqueue_busy) tq_active;
|
||||
struct mtx tq_mutex;
|
||||
struct thread **tq_threads;
|
||||
int tq_tcount;
|
||||
int tq_spin;
|
||||
int tq_flags;
|
||||
int tq_callouts;
|
||||
taskqueue_callback_fn tq_callbacks[TASKQUEUE_NUM_CALLBACKS];
|
||||
void *tq_cb_contexts[TASKQUEUE_NUM_CALLBACKS];
|
||||
};
|
||||
|
||||
#define TQ_FLAGS_ACTIVE (1 << 0)
|
||||
#define TQ_FLAGS_BLOCKED (1 << 1)
|
||||
#define TQ_FLAGS_UNLOCKED_ENQUEUE (1 << 2)
|
||||
|
||||
#define DT_CALLOUT_ARMED (1 << 0)
|
||||
|
||||
#define TQ_LOCK(tq) \
|
||||
do { \
|
||||
if ((tq)->tq_spin) \
|
||||
mtx_lock_spin(&(tq)->tq_mutex); \
|
||||
else \
|
||||
mtx_lock(&(tq)->tq_mutex); \
|
||||
} while (0)
|
||||
#define TQ_ASSERT_LOCKED(tq) mtx_assert(&(tq)->tq_mutex, MA_OWNED)
|
||||
|
||||
#define TQ_UNLOCK(tq) \
|
||||
do { \
|
||||
if ((tq)->tq_spin) \
|
||||
mtx_unlock_spin(&(tq)->tq_mutex); \
|
||||
else \
|
||||
mtx_unlock(&(tq)->tq_mutex); \
|
||||
} while (0)
|
||||
#define TQ_ASSERT_UNLOCKED(tq) mtx_assert(&(tq)->tq_mutex, MA_NOTOWNED)
|
||||
|
||||
static __inline int
|
||||
TQ_SLEEP(struct gtaskqueue *tq, void *p, struct mtx *m, int pri, const char *wm,
|
||||
int t)
|
||||
{
|
||||
if (tq->tq_spin)
|
||||
return (msleep_spin(p, m, wm, t));
|
||||
return (msleep(p, m, pri, wm, t));
|
||||
}
|
||||
|
||||
static struct gtaskqueue *
|
||||
_gtaskqueue_create(const char *name, int mflags,
|
||||
taskqueue_enqueue_fn enqueue, void *context,
|
||||
int mtxflags, const char *mtxname __unused)
|
||||
{
|
||||
struct gtaskqueue *queue;
|
||||
char *tq_name;
|
||||
|
||||
tq_name = malloc(TASKQUEUE_NAMELEN, M_GTASKQUEUE, mflags | M_ZERO);
|
||||
if (!tq_name)
|
||||
return (NULL);
|
||||
|
||||
snprintf(tq_name, TASKQUEUE_NAMELEN, "%s", (name) ? name : "taskqueue");
|
||||
|
||||
queue = malloc(sizeof(struct gtaskqueue), M_GTASKQUEUE, mflags | M_ZERO);
|
||||
if (!queue)
|
||||
return (NULL);
|
||||
|
||||
STAILQ_INIT(&queue->tq_queue);
|
||||
TAILQ_INIT(&queue->tq_active);
|
||||
queue->tq_enqueue = enqueue;
|
||||
queue->tq_context = context;
|
||||
queue->tq_name = tq_name;
|
||||
queue->tq_spin = (mtxflags & MTX_SPIN) != 0;
|
||||
queue->tq_flags |= TQ_FLAGS_ACTIVE;
|
||||
if (enqueue == gtaskqueue_thread_enqueue)
|
||||
queue->tq_flags |= TQ_FLAGS_UNLOCKED_ENQUEUE;
|
||||
mtx_init(&queue->tq_mutex, tq_name, NULL, mtxflags);
|
||||
|
||||
return (queue);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* Signal a taskqueue thread to terminate.
|
||||
*/
|
||||
static void
|
||||
gtaskqueue_terminate(struct thread **pp, struct gtaskqueue *tq)
|
||||
{
|
||||
|
||||
while (tq->tq_tcount > 0 || tq->tq_callouts > 0) {
|
||||
wakeup(tq);
|
||||
TQ_SLEEP(tq, pp, &tq->tq_mutex, PWAIT, "taskqueue_destroy", 0);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
gtaskqueue_free(struct gtaskqueue *queue)
|
||||
{
|
||||
|
||||
TQ_LOCK(queue);
|
||||
queue->tq_flags &= ~TQ_FLAGS_ACTIVE;
|
||||
gtaskqueue_terminate(queue->tq_threads, queue);
|
||||
KASSERT(TAILQ_EMPTY(&queue->tq_active), ("Tasks still running?"));
|
||||
KASSERT(queue->tq_callouts == 0, ("Armed timeout tasks"));
|
||||
mtx_destroy(&queue->tq_mutex);
|
||||
free(queue->tq_threads, M_GTASKQUEUE);
|
||||
free(queue->tq_name, M_GTASKQUEUE);
|
||||
free(queue, M_GTASKQUEUE);
|
||||
}
|
||||
|
||||
int
|
||||
grouptaskqueue_enqueue(struct gtaskqueue *queue, struct gtask *gtask)
|
||||
{
|
||||
TQ_LOCK(queue);
|
||||
if (gtask->ta_flags & TASK_ENQUEUED) {
|
||||
TQ_UNLOCK(queue);
|
||||
return (0);
|
||||
}
|
||||
STAILQ_INSERT_TAIL(&queue->tq_queue, gtask, ta_link);
|
||||
gtask->ta_flags |= TASK_ENQUEUED;
|
||||
TQ_UNLOCK(queue);
|
||||
if ((queue->tq_flags & TQ_FLAGS_BLOCKED) == 0)
|
||||
queue->tq_enqueue(queue->tq_context);
|
||||
return (0);
|
||||
}
|
||||
|
||||
static void
|
||||
gtaskqueue_task_nop_fn(void *context)
|
||||
{
|
||||
}
|
||||
|
||||
/*
|
||||
* Block until all currently queued tasks in this taskqueue
|
||||
* have begun execution. Tasks queued during execution of
|
||||
* this function are ignored.
|
||||
*/
|
||||
static void
|
||||
gtaskqueue_drain_tq_queue(struct gtaskqueue *queue)
|
||||
{
|
||||
struct gtask t_barrier;
|
||||
|
||||
if (STAILQ_EMPTY(&queue->tq_queue))
|
||||
return;
|
||||
|
||||
/*
|
||||
* Enqueue our barrier after all current tasks, but with
|
||||
* the highest priority so that newly queued tasks cannot
|
||||
* pass it. Because of the high priority, we can not use
|
||||
* taskqueue_enqueue_locked directly (which drops the lock
|
||||
* anyway) so just insert it at tail while we have the
|
||||
* queue lock.
|
||||
*/
|
||||
GTASK_INIT(&t_barrier, 0, USHRT_MAX, gtaskqueue_task_nop_fn, &t_barrier);
|
||||
STAILQ_INSERT_TAIL(&queue->tq_queue, &t_barrier, ta_link);
|
||||
t_barrier.ta_flags |= TASK_ENQUEUED;
|
||||
|
||||
/*
|
||||
* Once the barrier has executed, all previously queued tasks
|
||||
* have completed or are currently executing.
|
||||
*/
|
||||
while (t_barrier.ta_flags & TASK_ENQUEUED)
|
||||
TQ_SLEEP(queue, &t_barrier, &queue->tq_mutex, PWAIT, "-", 0);
|
||||
}
|
||||
|
||||
/*
|
||||
* Block until all currently executing tasks for this taskqueue
|
||||
* complete. Tasks that begin execution during the execution
|
||||
* of this function are ignored.
|
||||
*/
|
||||
static void
|
||||
gtaskqueue_drain_tq_active(struct gtaskqueue *queue)
|
||||
{
|
||||
struct gtaskqueue_busy tb_marker, *tb_first;
|
||||
|
||||
if (TAILQ_EMPTY(&queue->tq_active))
|
||||
return;
|
||||
|
||||
/* Block taskq_terminate().*/
|
||||
queue->tq_callouts++;
|
||||
|
||||
/*
|
||||
* Wait for all currently executing taskqueue threads
|
||||
* to go idle.
|
||||
*/
|
||||
tb_marker.tb_running = TB_DRAIN_WAITER;
|
||||
TAILQ_INSERT_TAIL(&queue->tq_active, &tb_marker, tb_link);
|
||||
while (TAILQ_FIRST(&queue->tq_active) != &tb_marker)
|
||||
TQ_SLEEP(queue, &tb_marker, &queue->tq_mutex, PWAIT, "-", 0);
|
||||
TAILQ_REMOVE(&queue->tq_active, &tb_marker, tb_link);
|
||||
|
||||
/*
|
||||
* Wakeup any other drain waiter that happened to queue up
|
||||
* without any intervening active thread.
|
||||
*/
|
||||
tb_first = TAILQ_FIRST(&queue->tq_active);
|
||||
if (tb_first != NULL && tb_first->tb_running == TB_DRAIN_WAITER)
|
||||
wakeup(tb_first);
|
||||
|
||||
/* Release taskqueue_terminate(). */
|
||||
queue->tq_callouts--;
|
||||
if ((queue->tq_flags & TQ_FLAGS_ACTIVE) == 0)
|
||||
wakeup_one(queue->tq_threads);
|
||||
}
|
||||
|
||||
void
|
||||
gtaskqueue_block(struct gtaskqueue *queue)
|
||||
{
|
||||
|
||||
TQ_LOCK(queue);
|
||||
queue->tq_flags |= TQ_FLAGS_BLOCKED;
|
||||
TQ_UNLOCK(queue);
|
||||
}
|
||||
|
||||
void
|
||||
gtaskqueue_unblock(struct gtaskqueue *queue)
|
||||
{
|
||||
|
||||
TQ_LOCK(queue);
|
||||
queue->tq_flags &= ~TQ_FLAGS_BLOCKED;
|
||||
if (!STAILQ_EMPTY(&queue->tq_queue))
|
||||
queue->tq_enqueue(queue->tq_context);
|
||||
TQ_UNLOCK(queue);
|
||||
}
|
||||
|
||||
static void
|
||||
gtaskqueue_run_locked(struct gtaskqueue *queue)
|
||||
{
|
||||
struct gtaskqueue_busy tb;
|
||||
struct gtaskqueue_busy *tb_first;
|
||||
struct gtask *gtask;
|
||||
|
||||
KASSERT(queue != NULL, ("tq is NULL"));
|
||||
TQ_ASSERT_LOCKED(queue);
|
||||
tb.tb_running = NULL;
|
||||
|
||||
while (STAILQ_FIRST(&queue->tq_queue)) {
|
||||
TAILQ_INSERT_TAIL(&queue->tq_active, &tb, tb_link);
|
||||
|
||||
/*
|
||||
* Carefully remove the first task from the queue and
|
||||
* clear its TASK_ENQUEUED flag
|
||||
*/
|
||||
gtask = STAILQ_FIRST(&queue->tq_queue);
|
||||
KASSERT(gtask != NULL, ("task is NULL"));
|
||||
STAILQ_REMOVE_HEAD(&queue->tq_queue, ta_link);
|
||||
gtask->ta_flags &= ~TASK_ENQUEUED;
|
||||
tb.tb_running = gtask;
|
||||
TQ_UNLOCK(queue);
|
||||
|
||||
KASSERT(gtask->ta_func != NULL, ("task->ta_func is NULL"));
|
||||
gtask->ta_func(gtask->ta_context);
|
||||
|
||||
TQ_LOCK(queue);
|
||||
tb.tb_running = NULL;
|
||||
wakeup(gtask);
|
||||
|
||||
TAILQ_REMOVE(&queue->tq_active, &tb, tb_link);
|
||||
tb_first = TAILQ_FIRST(&queue->tq_active);
|
||||
if (tb_first != NULL &&
|
||||
tb_first->tb_running == TB_DRAIN_WAITER)
|
||||
wakeup(tb_first);
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
task_is_running(struct gtaskqueue *queue, struct gtask *gtask)
|
||||
{
|
||||
struct gtaskqueue_busy *tb;
|
||||
|
||||
TQ_ASSERT_LOCKED(queue);
|
||||
TAILQ_FOREACH(tb, &queue->tq_active, tb_link) {
|
||||
if (tb->tb_running == gtask)
|
||||
return (1);
|
||||
}
|
||||
return (0);
|
||||
}
|
||||
|
||||
static int
|
||||
gtaskqueue_cancel_locked(struct gtaskqueue *queue, struct gtask *gtask)
|
||||
{
|
||||
|
||||
if (gtask->ta_flags & TASK_ENQUEUED)
|
||||
STAILQ_REMOVE(&queue->tq_queue, gtask, gtask, ta_link);
|
||||
gtask->ta_flags &= ~TASK_ENQUEUED;
|
||||
return (task_is_running(queue, gtask) ? EBUSY : 0);
|
||||
}
|
||||
|
||||
int
|
||||
gtaskqueue_cancel(struct gtaskqueue *queue, struct gtask *gtask)
|
||||
{
|
||||
int error;
|
||||
|
||||
TQ_LOCK(queue);
|
||||
error = gtaskqueue_cancel_locked(queue, gtask);
|
||||
TQ_UNLOCK(queue);
|
||||
|
||||
return (error);
|
||||
}
|
||||
|
||||
void
|
||||
gtaskqueue_drain(struct gtaskqueue *queue, struct gtask *gtask)
|
||||
{
|
||||
|
||||
if (!queue->tq_spin)
|
||||
WITNESS_WARN(WARN_GIANTOK | WARN_SLEEPOK, NULL, __func__);
|
||||
|
||||
TQ_LOCK(queue);
|
||||
while ((gtask->ta_flags & TASK_ENQUEUED) || task_is_running(queue, gtask))
|
||||
TQ_SLEEP(queue, gtask, &queue->tq_mutex, PWAIT, "-", 0);
|
||||
TQ_UNLOCK(queue);
|
||||
}
|
||||
|
||||
void
|
||||
gtaskqueue_drain_all(struct gtaskqueue *queue)
|
||||
{
|
||||
|
||||
if (!queue->tq_spin)
|
||||
WITNESS_WARN(WARN_GIANTOK | WARN_SLEEPOK, NULL, __func__);
|
||||
|
||||
TQ_LOCK(queue);
|
||||
gtaskqueue_drain_tq_queue(queue);
|
||||
gtaskqueue_drain_tq_active(queue);
|
||||
TQ_UNLOCK(queue);
|
||||
}
|
||||
|
||||
static int
|
||||
_gtaskqueue_start_threads(struct gtaskqueue **tqp, int count, int pri,
|
||||
cpuset_t *mask, const char *name, va_list ap)
|
||||
{
|
||||
char ktname[MAXCOMLEN + 1];
|
||||
struct thread *td;
|
||||
struct gtaskqueue *tq;
|
||||
int i, error;
|
||||
|
||||
if (count <= 0)
|
||||
return (EINVAL);
|
||||
|
||||
vsnprintf(ktname, sizeof(ktname), name, ap);
|
||||
tq = *tqp;
|
||||
|
||||
tq->tq_threads = malloc(sizeof(struct thread *) * count, M_GTASKQUEUE,
|
||||
M_NOWAIT | M_ZERO);
|
||||
if (tq->tq_threads == NULL) {
|
||||
printf("%s: no memory for %s threads\n", __func__, ktname);
|
||||
return (ENOMEM);
|
||||
}
|
||||
|
||||
for (i = 0; i < count; i++) {
|
||||
if (count == 1)
|
||||
error = kthread_add(gtaskqueue_thread_loop, tqp, NULL,
|
||||
&tq->tq_threads[i], RFSTOPPED, 0, "%s", ktname);
|
||||
else
|
||||
error = kthread_add(gtaskqueue_thread_loop, tqp, NULL,
|
||||
&tq->tq_threads[i], RFSTOPPED, 0,
|
||||
"%s_%d", ktname, i);
|
||||
if (error) {
|
||||
/* should be ok to continue, taskqueue_free will dtrt */
|
||||
printf("%s: kthread_add(%s): error %d", __func__,
|
||||
ktname, error);
|
||||
tq->tq_threads[i] = NULL; /* paranoid */
|
||||
} else
|
||||
tq->tq_tcount++;
|
||||
}
|
||||
for (i = 0; i < count; i++) {
|
||||
if (tq->tq_threads[i] == NULL)
|
||||
continue;
|
||||
td = tq->tq_threads[i];
|
||||
if (mask) {
|
||||
error = cpuset_setthread(td->td_tid, mask);
|
||||
/*
|
||||
* Failing to pin is rarely an actual fatal error;
|
||||
* it'll just affect performance.
|
||||
*/
|
||||
if (error)
|
||||
printf("%s: curthread=%llu: can't pin; "
|
||||
"error=%d\n",
|
||||
__func__,
|
||||
(unsigned long long) td->td_tid,
|
||||
error);
|
||||
}
|
||||
thread_lock(td);
|
||||
sched_prio(td, pri);
|
||||
sched_add(td, SRQ_BORING);
|
||||
thread_unlock(td);
|
||||
}
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
static int
|
||||
gtaskqueue_start_threads(struct gtaskqueue **tqp, int count, int pri,
|
||||
const char *name, ...)
|
||||
{
|
||||
va_list ap;
|
||||
int error;
|
||||
|
||||
va_start(ap, name);
|
||||
error = _gtaskqueue_start_threads(tqp, count, pri, NULL, name, ap);
|
||||
va_end(ap);
|
||||
return (error);
|
||||
}
|
||||
|
||||
static inline void
|
||||
gtaskqueue_run_callback(struct gtaskqueue *tq,
|
||||
enum taskqueue_callback_type cb_type)
|
||||
{
|
||||
taskqueue_callback_fn tq_callback;
|
||||
|
||||
TQ_ASSERT_UNLOCKED(tq);
|
||||
tq_callback = tq->tq_callbacks[cb_type];
|
||||
if (tq_callback != NULL)
|
||||
tq_callback(tq->tq_cb_contexts[cb_type]);
|
||||
}
|
||||
|
||||
static void
|
||||
gtaskqueue_thread_loop(void *arg)
|
||||
{
|
||||
struct gtaskqueue **tqp, *tq;
|
||||
|
||||
tqp = arg;
|
||||
tq = *tqp;
|
||||
gtaskqueue_run_callback(tq, TASKQUEUE_CALLBACK_TYPE_INIT);
|
||||
TQ_LOCK(tq);
|
||||
while ((tq->tq_flags & TQ_FLAGS_ACTIVE) != 0) {
|
||||
/* XXX ? */
|
||||
gtaskqueue_run_locked(tq);
|
||||
/*
|
||||
* Because taskqueue_run() can drop tq_mutex, we need to
|
||||
* check if the TQ_FLAGS_ACTIVE flag wasn't removed in the
|
||||
* meantime, which means we missed a wakeup.
|
||||
*/
|
||||
if ((tq->tq_flags & TQ_FLAGS_ACTIVE) == 0)
|
||||
break;
|
||||
TQ_SLEEP(tq, tq, &tq->tq_mutex, 0, "-", 0);
|
||||
}
|
||||
gtaskqueue_run_locked(tq);
|
||||
/*
|
||||
* This thread is on its way out, so just drop the lock temporarily
|
||||
* in order to call the shutdown callback. This allows the callback
|
||||
* to look at the taskqueue, even just before it dies.
|
||||
*/
|
||||
TQ_UNLOCK(tq);
|
||||
gtaskqueue_run_callback(tq, TASKQUEUE_CALLBACK_TYPE_SHUTDOWN);
|
||||
TQ_LOCK(tq);
|
||||
|
||||
/* rendezvous with thread that asked us to terminate */
|
||||
tq->tq_tcount--;
|
||||
wakeup_one(tq->tq_threads);
|
||||
TQ_UNLOCK(tq);
|
||||
kthread_exit();
|
||||
}
|
||||
|
||||
static void
|
||||
gtaskqueue_thread_enqueue(void *context)
|
||||
{
|
||||
struct gtaskqueue **tqp, *tq;
|
||||
|
||||
tqp = context;
|
||||
tq = *tqp;
|
||||
wakeup_one(tq);
|
||||
}
|
||||
|
||||
|
||||
static struct gtaskqueue *
|
||||
gtaskqueue_create_fast(const char *name, int mflags,
|
||||
taskqueue_enqueue_fn enqueue, void *context)
|
||||
{
|
||||
return _gtaskqueue_create(name, mflags, enqueue, context,
|
||||
MTX_SPIN, "fast_taskqueue");
|
||||
}
|
||||
|
||||
|
||||
struct taskqgroup_cpu {
|
||||
LIST_HEAD(, grouptask) tgc_tasks;
|
||||
struct gtaskqueue *tgc_taskq;
|
||||
int tgc_cnt;
|
||||
int tgc_cpu;
|
||||
};
|
||||
|
||||
struct taskqgroup {
|
||||
struct taskqgroup_cpu tqg_queue[MAXCPU];
|
||||
struct mtx tqg_lock;
|
||||
char * tqg_name;
|
||||
int tqg_adjusting;
|
||||
int tqg_stride;
|
||||
int tqg_cnt;
|
||||
};
|
||||
|
||||
struct taskq_bind_task {
|
||||
struct gtask bt_task;
|
||||
int bt_cpuid;
|
||||
};
|
||||
|
||||
static void
|
||||
taskqgroup_cpu_create(struct taskqgroup *qgroup, int idx)
|
||||
{
|
||||
struct taskqgroup_cpu *qcpu;
|
||||
|
||||
qcpu = &qgroup->tqg_queue[idx];
|
||||
LIST_INIT(&qcpu->tgc_tasks);
|
||||
qcpu->tgc_taskq = gtaskqueue_create_fast(NULL, M_WAITOK,
|
||||
taskqueue_thread_enqueue, &qcpu->tgc_taskq);
|
||||
gtaskqueue_start_threads(&qcpu->tgc_taskq, 1, PI_SOFT,
|
||||
"%s_%d", qgroup->tqg_name, idx);
|
||||
qcpu->tgc_cpu = idx * qgroup->tqg_stride;
|
||||
}
|
||||
|
||||
static void
|
||||
taskqgroup_cpu_remove(struct taskqgroup *qgroup, int idx)
|
||||
{
|
||||
|
||||
gtaskqueue_free(qgroup->tqg_queue[idx].tgc_taskq);
|
||||
}
|
||||
|
||||
/*
|
||||
* Find the taskq with least # of tasks that doesn't currently have any
|
||||
* other queues from the uniq identifier.
|
||||
*/
|
||||
static int
|
||||
taskqgroup_find(struct taskqgroup *qgroup, void *uniq)
|
||||
{
|
||||
struct grouptask *n;
|
||||
int i, idx, mincnt;
|
||||
int strict;
|
||||
|
||||
mtx_assert(&qgroup->tqg_lock, MA_OWNED);
|
||||
if (qgroup->tqg_cnt == 0)
|
||||
return (0);
|
||||
idx = -1;
|
||||
mincnt = INT_MAX;
|
||||
/*
|
||||
* Two passes; First scan for a queue with the least tasks that
|
||||
* does not already service this uniq id. If that fails simply find
|
||||
* the queue with the least total tasks;
|
||||
*/
|
||||
for (strict = 1; mincnt == INT_MAX; strict = 0) {
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++) {
|
||||
if (qgroup->tqg_queue[i].tgc_cnt > mincnt)
|
||||
continue;
|
||||
if (strict) {
|
||||
LIST_FOREACH(n,
|
||||
&qgroup->tqg_queue[i].tgc_tasks, gt_list)
|
||||
if (n->gt_uniq == uniq)
|
||||
break;
|
||||
if (n != NULL)
|
||||
continue;
|
||||
}
|
||||
mincnt = qgroup->tqg_queue[i].tgc_cnt;
|
||||
idx = i;
|
||||
}
|
||||
}
|
||||
if (idx == -1)
|
||||
panic("taskqgroup_find: Failed to pick a qid.");
|
||||
|
||||
return (idx);
|
||||
}
|
||||
|
||||
void
|
||||
taskqgroup_attach(struct taskqgroup *qgroup, struct grouptask *gtask,
|
||||
void *uniq, int irq, char *name)
|
||||
{
|
||||
cpuset_t mask;
|
||||
int qid;
|
||||
|
||||
gtask->gt_uniq = uniq;
|
||||
gtask->gt_name = name;
|
||||
gtask->gt_irq = irq;
|
||||
gtask->gt_cpu = -1;
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
qid = taskqgroup_find(qgroup, uniq);
|
||||
qgroup->tqg_queue[qid].tgc_cnt++;
|
||||
LIST_INSERT_HEAD(&qgroup->tqg_queue[qid].tgc_tasks, gtask, gt_list);
|
||||
gtask->gt_taskqueue = qgroup->tqg_queue[qid].tgc_taskq;
|
||||
if (irq != -1 && smp_started) {
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(qgroup->tqg_queue[qid].tgc_cpu, &mask);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
intr_setaffinity(irq, &mask);
|
||||
} else
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
}
|
||||
|
||||
int
|
||||
taskqgroup_attach_cpu(struct taskqgroup *qgroup, struct grouptask *gtask,
|
||||
void *uniq, int cpu, int irq, char *name)
|
||||
{
|
||||
cpuset_t mask;
|
||||
int i, qid;
|
||||
|
||||
qid = -1;
|
||||
gtask->gt_uniq = uniq;
|
||||
gtask->gt_name = name;
|
||||
gtask->gt_irq = irq;
|
||||
gtask->gt_cpu = cpu;
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
if (smp_started) {
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++)
|
||||
if (qgroup->tqg_queue[i].tgc_cpu == cpu) {
|
||||
qid = i;
|
||||
break;
|
||||
}
|
||||
if (qid == -1) {
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
return (EINVAL);
|
||||
}
|
||||
} else
|
||||
qid = 0;
|
||||
qgroup->tqg_queue[qid].tgc_cnt++;
|
||||
LIST_INSERT_HEAD(&qgroup->tqg_queue[qid].tgc_tasks, gtask, gt_list);
|
||||
gtask->gt_taskqueue = qgroup->tqg_queue[qid].tgc_taskq;
|
||||
if (irq != -1 && smp_started) {
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(qgroup->tqg_queue[qid].tgc_cpu, &mask);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
intr_setaffinity(irq, &mask);
|
||||
} else
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
return (0);
|
||||
}
|
||||
|
||||
void
|
||||
taskqgroup_detach(struct taskqgroup *qgroup, struct grouptask *gtask)
|
||||
{
|
||||
int i;
|
||||
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++)
|
||||
if (qgroup->tqg_queue[i].tgc_taskq == gtask->gt_taskqueue)
|
||||
break;
|
||||
if (i == qgroup->tqg_cnt)
|
||||
panic("taskqgroup_detach: task not in group\n");
|
||||
qgroup->tqg_queue[i].tgc_cnt--;
|
||||
LIST_REMOVE(gtask, gt_list);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
gtask->gt_taskqueue = NULL;
|
||||
}
|
||||
|
||||
static void
|
||||
taskqgroup_binder(void *ctx)
|
||||
{
|
||||
struct taskq_bind_task *gtask = (struct taskq_bind_task *)ctx;
|
||||
cpuset_t mask;
|
||||
int error;
|
||||
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(gtask->bt_cpuid, &mask);
|
||||
error = cpuset_setthread(curthread->td_tid, &mask);
|
||||
thread_lock(curthread);
|
||||
sched_bind(curthread, gtask->bt_cpuid);
|
||||
thread_unlock(curthread);
|
||||
|
||||
if (error)
|
||||
printf("taskqgroup_binder: setaffinity failed: %d\n",
|
||||
error);
|
||||
free(gtask, M_DEVBUF);
|
||||
}
|
||||
|
||||
static void
|
||||
taskqgroup_bind(struct taskqgroup *qgroup)
|
||||
{
|
||||
struct taskq_bind_task *gtask;
|
||||
int i;
|
||||
|
||||
/*
|
||||
* Bind taskqueue threads to specific CPUs, if they have been assigned
|
||||
* one.
|
||||
*/
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++) {
|
||||
gtask = malloc(sizeof (*gtask), M_DEVBUF, M_NOWAIT);
|
||||
GTASK_INIT(>ask->bt_task, 0, 0, taskqgroup_binder, gtask);
|
||||
gtask->bt_cpuid = qgroup->tqg_queue[i].tgc_cpu;
|
||||
grouptaskqueue_enqueue(qgroup->tqg_queue[i].tgc_taskq,
|
||||
>ask->bt_task);
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
_taskqgroup_adjust(struct taskqgroup *qgroup, int cnt, int stride)
|
||||
{
|
||||
LIST_HEAD(, grouptask) gtask_head = LIST_HEAD_INITIALIZER(NULL);
|
||||
cpuset_t mask;
|
||||
struct grouptask *gtask;
|
||||
int i, old_cnt, qid;
|
||||
|
||||
mtx_assert(&qgroup->tqg_lock, MA_OWNED);
|
||||
|
||||
if (cnt < 1 || cnt * stride > mp_ncpus || !smp_started) {
|
||||
printf("taskqgroup_adjust failed cnt: %d stride: %d mp_ncpus: %d smp_started: %d\n",
|
||||
cnt, stride, mp_ncpus, smp_started);
|
||||
return (EINVAL);
|
||||
}
|
||||
if (qgroup->tqg_adjusting) {
|
||||
printf("taskqgroup_adjust failed: adjusting\n");
|
||||
return (EBUSY);
|
||||
}
|
||||
qgroup->tqg_adjusting = 1;
|
||||
old_cnt = qgroup->tqg_cnt;
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
/*
|
||||
* Set up queue for tasks added before boot.
|
||||
*/
|
||||
if (old_cnt == 0) {
|
||||
LIST_SWAP(>ask_head, &qgroup->tqg_queue[0].tgc_tasks,
|
||||
grouptask, gt_list);
|
||||
qgroup->tqg_queue[0].tgc_cnt = 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* If new taskq threads have been added.
|
||||
*/
|
||||
for (i = old_cnt; i < cnt; i++)
|
||||
taskqgroup_cpu_create(qgroup, i);
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
qgroup->tqg_cnt = cnt;
|
||||
qgroup->tqg_stride = stride;
|
||||
|
||||
/*
|
||||
* Adjust drivers to use new taskqs.
|
||||
*/
|
||||
for (i = 0; i < old_cnt; i++) {
|
||||
while ((gtask = LIST_FIRST(&qgroup->tqg_queue[i].tgc_tasks))) {
|
||||
LIST_REMOVE(gtask, gt_list);
|
||||
qgroup->tqg_queue[i].tgc_cnt--;
|
||||
LIST_INSERT_HEAD(>ask_head, gtask, gt_list);
|
||||
}
|
||||
}
|
||||
|
||||
while ((gtask = LIST_FIRST(>ask_head))) {
|
||||
LIST_REMOVE(gtask, gt_list);
|
||||
if (gtask->gt_cpu == -1)
|
||||
qid = taskqgroup_find(qgroup, gtask->gt_uniq);
|
||||
else {
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++)
|
||||
if (qgroup->tqg_queue[i].tgc_cpu == gtask->gt_cpu) {
|
||||
qid = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
qgroup->tqg_queue[qid].tgc_cnt++;
|
||||
LIST_INSERT_HEAD(&qgroup->tqg_queue[qid].tgc_tasks, gtask,
|
||||
gt_list);
|
||||
gtask->gt_taskqueue = qgroup->tqg_queue[qid].tgc_taskq;
|
||||
}
|
||||
/*
|
||||
* Set new CPU and IRQ affinity
|
||||
*/
|
||||
for (i = 0; i < cnt; i++) {
|
||||
qgroup->tqg_queue[i].tgc_cpu = i * qgroup->tqg_stride;
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(qgroup->tqg_queue[i].tgc_cpu, &mask);
|
||||
LIST_FOREACH(gtask, &qgroup->tqg_queue[i].tgc_tasks, gt_list) {
|
||||
if (gtask->gt_irq == -1)
|
||||
continue;
|
||||
intr_setaffinity(gtask->gt_irq, &mask);
|
||||
}
|
||||
}
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
|
||||
/*
|
||||
* If taskq thread count has been reduced.
|
||||
*/
|
||||
for (i = cnt; i < old_cnt; i++)
|
||||
taskqgroup_cpu_remove(qgroup, i);
|
||||
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
qgroup->tqg_adjusting = 0;
|
||||
|
||||
taskqgroup_bind(qgroup);
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
int
|
||||
taskqgroup_adjust(struct taskqgroup *qgroup, int cpu, int stride)
|
||||
{
|
||||
int error;
|
||||
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
error = _taskqgroup_adjust(qgroup, cpu, stride);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
|
||||
return (error);
|
||||
}
|
||||
|
||||
struct taskqgroup *
|
||||
taskqgroup_create(char *name)
|
||||
{
|
||||
struct taskqgroup *qgroup;
|
||||
|
||||
qgroup = malloc(sizeof(*qgroup), M_GTASKQUEUE, M_WAITOK | M_ZERO);
|
||||
mtx_init(&qgroup->tqg_lock, "taskqgroup", NULL, MTX_DEF);
|
||||
qgroup->tqg_name = name;
|
||||
LIST_INIT(&qgroup->tqg_queue[0].tgc_tasks);
|
||||
|
||||
return (qgroup);
|
||||
}
|
||||
|
||||
void
|
||||
taskqgroup_destroy(struct taskqgroup *qgroup)
|
||||
{
|
||||
|
||||
}
|
@ -260,22 +260,6 @@ taskqueue_enqueue_locked(struct taskqueue *queue, struct task *task)
|
||||
return (0);
|
||||
}
|
||||
|
||||
int
|
||||
grouptaskqueue_enqueue(struct taskqueue *queue, struct task *task)
|
||||
{
|
||||
TQ_LOCK(queue);
|
||||
if (task->ta_pending) {
|
||||
TQ_UNLOCK(queue);
|
||||
return (0);
|
||||
}
|
||||
STAILQ_INSERT_TAIL(&queue->tq_queue, task, ta_link);
|
||||
task->ta_pending = 1;
|
||||
TQ_UNLOCK(queue);
|
||||
if ((queue->tq_flags & TQ_FLAGS_BLOCKED) == 0)
|
||||
queue->tq_enqueue(queue->tq_context);
|
||||
return (0);
|
||||
}
|
||||
|
||||
int
|
||||
taskqueue_enqueue(struct taskqueue *queue, struct task *task)
|
||||
{
|
||||
@ -806,347 +790,3 @@ taskqueue_member(struct taskqueue *queue, struct thread *td)
|
||||
}
|
||||
return (ret);
|
||||
}
|
||||
|
||||
struct taskqgroup_cpu {
|
||||
LIST_HEAD(, grouptask) tgc_tasks;
|
||||
struct taskqueue *tgc_taskq;
|
||||
int tgc_cnt;
|
||||
int tgc_cpu;
|
||||
};
|
||||
|
||||
struct taskqgroup {
|
||||
struct taskqgroup_cpu tqg_queue[MAXCPU];
|
||||
struct mtx tqg_lock;
|
||||
char * tqg_name;
|
||||
int tqg_adjusting;
|
||||
int tqg_stride;
|
||||
int tqg_cnt;
|
||||
};
|
||||
|
||||
struct taskq_bind_task {
|
||||
struct task bt_task;
|
||||
int bt_cpuid;
|
||||
};
|
||||
|
||||
static void
|
||||
taskqgroup_cpu_create(struct taskqgroup *qgroup, int idx)
|
||||
{
|
||||
struct taskqgroup_cpu *qcpu;
|
||||
int i, j;
|
||||
|
||||
qcpu = &qgroup->tqg_queue[idx];
|
||||
LIST_INIT(&qcpu->tgc_tasks);
|
||||
qcpu->tgc_taskq = taskqueue_create_fast(NULL, M_WAITOK,
|
||||
taskqueue_thread_enqueue, &qcpu->tgc_taskq);
|
||||
taskqueue_start_threads(&qcpu->tgc_taskq, 1, PI_SOFT,
|
||||
"%s_%d", qgroup->tqg_name, idx);
|
||||
|
||||
for (i = CPU_FIRST(), j = 0; j < idx * qgroup->tqg_stride;
|
||||
j++, i = CPU_NEXT(i)) {
|
||||
/*
|
||||
* Wait: evaluate the idx * qgroup->tqg_stride'th CPU,
|
||||
* potentially wrapping the actual count
|
||||
*/
|
||||
}
|
||||
qcpu->tgc_cpu = i;
|
||||
}
|
||||
|
||||
static void
|
||||
taskqgroup_cpu_remove(struct taskqgroup *qgroup, int idx)
|
||||
{
|
||||
|
||||
taskqueue_free(qgroup->tqg_queue[idx].tgc_taskq);
|
||||
}
|
||||
|
||||
/*
|
||||
* Find the taskq with least # of tasks that doesn't currently have any
|
||||
* other queues from the uniq identifier.
|
||||
*/
|
||||
static int
|
||||
taskqgroup_find(struct taskqgroup *qgroup, void *uniq)
|
||||
{
|
||||
struct grouptask *n;
|
||||
int i, idx, mincnt;
|
||||
int strict;
|
||||
|
||||
mtx_assert(&qgroup->tqg_lock, MA_OWNED);
|
||||
if (qgroup->tqg_cnt == 0)
|
||||
return (0);
|
||||
idx = -1;
|
||||
mincnt = INT_MAX;
|
||||
/*
|
||||
* Two passes; First scan for a queue with the least tasks that
|
||||
* does not already service this uniq id. If that fails simply find
|
||||
* the queue with the least total tasks;
|
||||
*/
|
||||
for (strict = 1; mincnt == INT_MAX; strict = 0) {
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++) {
|
||||
if (qgroup->tqg_queue[i].tgc_cnt > mincnt)
|
||||
continue;
|
||||
if (strict) {
|
||||
LIST_FOREACH(n,
|
||||
&qgroup->tqg_queue[i].tgc_tasks, gt_list)
|
||||
if (n->gt_uniq == uniq)
|
||||
break;
|
||||
if (n != NULL)
|
||||
continue;
|
||||
}
|
||||
mincnt = qgroup->tqg_queue[i].tgc_cnt;
|
||||
idx = i;
|
||||
}
|
||||
}
|
||||
if (idx == -1)
|
||||
panic("taskqgroup_find: Failed to pick a qid.");
|
||||
|
||||
return (idx);
|
||||
}
|
||||
|
||||
void
|
||||
taskqgroup_attach(struct taskqgroup *qgroup, struct grouptask *gtask,
|
||||
void *uniq, int irq, char *name)
|
||||
{
|
||||
cpuset_t mask;
|
||||
int qid;
|
||||
|
||||
gtask->gt_uniq = uniq;
|
||||
gtask->gt_name = name;
|
||||
gtask->gt_irq = irq;
|
||||
gtask->gt_cpu = -1;
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
qid = taskqgroup_find(qgroup, uniq);
|
||||
qgroup->tqg_queue[qid].tgc_cnt++;
|
||||
LIST_INSERT_HEAD(&qgroup->tqg_queue[qid].tgc_tasks, gtask, gt_list);
|
||||
gtask->gt_taskqueue = qgroup->tqg_queue[qid].tgc_taskq;
|
||||
if (irq != -1 && smp_started) {
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(qgroup->tqg_queue[qid].tgc_cpu, &mask);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
intr_setaffinity(irq, &mask);
|
||||
} else
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
}
|
||||
|
||||
int
|
||||
taskqgroup_attach_cpu(struct taskqgroup *qgroup, struct grouptask *gtask,
|
||||
void *uniq, int cpu, int irq, char *name)
|
||||
{
|
||||
cpuset_t mask;
|
||||
int i, qid;
|
||||
|
||||
qid = -1;
|
||||
gtask->gt_uniq = uniq;
|
||||
gtask->gt_name = name;
|
||||
gtask->gt_irq = irq;
|
||||
gtask->gt_cpu = cpu;
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
if (smp_started) {
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++)
|
||||
if (qgroup->tqg_queue[i].tgc_cpu == cpu) {
|
||||
qid = i;
|
||||
break;
|
||||
}
|
||||
if (qid == -1) {
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
return (EINVAL);
|
||||
}
|
||||
} else
|
||||
qid = 0;
|
||||
qgroup->tqg_queue[qid].tgc_cnt++;
|
||||
LIST_INSERT_HEAD(&qgroup->tqg_queue[qid].tgc_tasks, gtask, gt_list);
|
||||
gtask->gt_taskqueue = qgroup->tqg_queue[qid].tgc_taskq;
|
||||
if (irq != -1 && smp_started) {
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(qgroup->tqg_queue[qid].tgc_cpu, &mask);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
intr_setaffinity(irq, &mask);
|
||||
} else
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
return (0);
|
||||
}
|
||||
|
||||
void
|
||||
taskqgroup_detach(struct taskqgroup *qgroup, struct grouptask *gtask)
|
||||
{
|
||||
int i;
|
||||
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++)
|
||||
if (qgroup->tqg_queue[i].tgc_taskq == gtask->gt_taskqueue)
|
||||
break;
|
||||
if (i == qgroup->tqg_cnt)
|
||||
panic("taskqgroup_detach: task not in group\n");
|
||||
qgroup->tqg_queue[i].tgc_cnt--;
|
||||
LIST_REMOVE(gtask, gt_list);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
gtask->gt_taskqueue = NULL;
|
||||
}
|
||||
|
||||
static void
|
||||
taskqgroup_binder(void *ctx, int pending)
|
||||
{
|
||||
struct taskq_bind_task *task = (struct taskq_bind_task *)ctx;
|
||||
cpuset_t mask;
|
||||
int error;
|
||||
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(task->bt_cpuid, &mask);
|
||||
error = cpuset_setthread(curthread->td_tid, &mask);
|
||||
thread_lock(curthread);
|
||||
sched_bind(curthread, task->bt_cpuid);
|
||||
thread_unlock(curthread);
|
||||
|
||||
if (error)
|
||||
printf("taskqgroup_binder: setaffinity failed: %d\n",
|
||||
error);
|
||||
free(task, M_DEVBUF);
|
||||
}
|
||||
|
||||
static void
|
||||
taskqgroup_bind(struct taskqgroup *qgroup)
|
||||
{
|
||||
struct taskq_bind_task *task;
|
||||
int i;
|
||||
|
||||
/*
|
||||
* Bind taskqueue threads to specific CPUs, if they have been assigned
|
||||
* one.
|
||||
*/
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++) {
|
||||
task = malloc(sizeof (*task), M_DEVBUF, M_NOWAIT);
|
||||
TASK_INIT(&task->bt_task, 0, taskqgroup_binder, task);
|
||||
task->bt_cpuid = qgroup->tqg_queue[i].tgc_cpu;
|
||||
taskqueue_enqueue(qgroup->tqg_queue[i].tgc_taskq,
|
||||
&task->bt_task);
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
_taskqgroup_adjust(struct taskqgroup *qgroup, int cnt, int stride)
|
||||
{
|
||||
LIST_HEAD(, grouptask) gtask_head = LIST_HEAD_INITIALIZER(NULL);
|
||||
cpuset_t mask;
|
||||
struct grouptask *gtask;
|
||||
int i, k, old_cnt, qid, cpu;
|
||||
|
||||
mtx_assert(&qgroup->tqg_lock, MA_OWNED);
|
||||
|
||||
if (cnt < 1 || cnt * stride > mp_ncpus || !smp_started) {
|
||||
printf("taskqgroup_adjust failed cnt: %d stride: %d "
|
||||
"mp_ncpus: %d smp_started: %d\n", cnt, stride, mp_ncpus,
|
||||
smp_started);
|
||||
return (EINVAL);
|
||||
}
|
||||
if (qgroup->tqg_adjusting) {
|
||||
printf("taskqgroup_adjust failed: adjusting\n");
|
||||
return (EBUSY);
|
||||
}
|
||||
qgroup->tqg_adjusting = 1;
|
||||
old_cnt = qgroup->tqg_cnt;
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
/*
|
||||
* Set up queue for tasks added before boot.
|
||||
*/
|
||||
if (old_cnt == 0) {
|
||||
LIST_SWAP(>ask_head, &qgroup->tqg_queue[0].tgc_tasks,
|
||||
grouptask, gt_list);
|
||||
qgroup->tqg_queue[0].tgc_cnt = 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* If new taskq threads have been added.
|
||||
*/
|
||||
for (i = old_cnt; i < cnt; i++)
|
||||
taskqgroup_cpu_create(qgroup, i);
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
qgroup->tqg_cnt = cnt;
|
||||
qgroup->tqg_stride = stride;
|
||||
|
||||
/*
|
||||
* Adjust drivers to use new taskqs.
|
||||
*/
|
||||
for (i = 0; i < old_cnt; i++) {
|
||||
while ((gtask = LIST_FIRST(&qgroup->tqg_queue[i].tgc_tasks))) {
|
||||
LIST_REMOVE(gtask, gt_list);
|
||||
qgroup->tqg_queue[i].tgc_cnt--;
|
||||
LIST_INSERT_HEAD(>ask_head, gtask, gt_list);
|
||||
}
|
||||
}
|
||||
|
||||
while ((gtask = LIST_FIRST(>ask_head))) {
|
||||
LIST_REMOVE(gtask, gt_list);
|
||||
if (gtask->gt_cpu == -1)
|
||||
qid = taskqgroup_find(qgroup, gtask->gt_uniq);
|
||||
else {
|
||||
for (i = 0; i < qgroup->tqg_cnt; i++)
|
||||
if (qgroup->tqg_queue[i].tgc_cpu == gtask->gt_cpu) {
|
||||
qid = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
qgroup->tqg_queue[qid].tgc_cnt++;
|
||||
LIST_INSERT_HEAD(&qgroup->tqg_queue[qid].tgc_tasks, gtask,
|
||||
gt_list);
|
||||
gtask->gt_taskqueue = qgroup->tqg_queue[qid].tgc_taskq;
|
||||
}
|
||||
/*
|
||||
* Set new CPU and IRQ affinity
|
||||
*/
|
||||
cpu = CPU_FIRST();
|
||||
for (i = 0; i < cnt; i++) {
|
||||
qgroup->tqg_queue[i].tgc_cpu = cpu;
|
||||
for (k = 0; k < qgroup->tqg_stride; k++)
|
||||
cpu = CPU_NEXT(cpu);
|
||||
CPU_ZERO(&mask);
|
||||
CPU_SET(qgroup->tqg_queue[i].tgc_cpu, &mask);
|
||||
LIST_FOREACH(gtask, &qgroup->tqg_queue[i].tgc_tasks, gt_list) {
|
||||
if (gtask->gt_irq == -1)
|
||||
continue;
|
||||
intr_setaffinity(gtask->gt_irq, &mask);
|
||||
}
|
||||
}
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
|
||||
/*
|
||||
* If taskq thread count has been reduced.
|
||||
*/
|
||||
for (i = cnt; i < old_cnt; i++)
|
||||
taskqgroup_cpu_remove(qgroup, i);
|
||||
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
qgroup->tqg_adjusting = 0;
|
||||
|
||||
taskqgroup_bind(qgroup);
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
int
|
||||
taskqgroup_adjust(struct taskqgroup *qgroup, int cpu, int stride)
|
||||
{
|
||||
int error;
|
||||
|
||||
mtx_lock(&qgroup->tqg_lock);
|
||||
error = _taskqgroup_adjust(qgroup, cpu, stride);
|
||||
mtx_unlock(&qgroup->tqg_lock);
|
||||
|
||||
return (error);
|
||||
}
|
||||
|
||||
struct taskqgroup *
|
||||
taskqgroup_create(char *name)
|
||||
{
|
||||
struct taskqgroup *qgroup;
|
||||
|
||||
qgroup = malloc(sizeof(*qgroup), M_TASKQUEUE, M_WAITOK | M_ZERO);
|
||||
mtx_init(&qgroup->tqg_lock, "taskqgroup", NULL, MTX_DEF);
|
||||
qgroup->tqg_name = name;
|
||||
LIST_INIT(&qgroup->tqg_queue[0].tgc_tasks);
|
||||
|
||||
return (qgroup);
|
||||
}
|
||||
|
||||
void
|
||||
taskqgroup_destroy(struct taskqgroup *qgroup)
|
||||
{
|
||||
|
||||
}
|
||||
|
@ -60,9 +60,10 @@ CODE {
|
||||
return (0);
|
||||
}
|
||||
|
||||
static void
|
||||
static int
|
||||
null_queue_intr_enable(if_ctx_t _ctx __unused, uint16_t _qid __unused)
|
||||
{
|
||||
return (ENOTSUP);
|
||||
}
|
||||
|
||||
static void
|
||||
@ -194,7 +195,7 @@ METHOD void intr_disable {
|
||||
if_ctx_t _ctx;
|
||||
};
|
||||
|
||||
METHOD void queue_intr_enable {
|
||||
METHOD int queue_intr_enable {
|
||||
if_ctx_t _ctx;
|
||||
uint16_t _qid;
|
||||
} DEFAULT null_queue_intr_enable;
|
||||
|
539
sys/net/iflib.c
539
sys/net/iflib.c
File diff suppressed because it is too large
Load Diff
@ -35,6 +35,7 @@
|
||||
#include <machine/bus.h>
|
||||
#include <sys/bus_dma.h>
|
||||
#include <sys/nv.h>
|
||||
#include <sys/gtaskqueue.h>
|
||||
|
||||
|
||||
/*
|
||||
@ -63,12 +64,14 @@ typedef struct if_int_delay_info *if_int_delay_info_t;
|
||||
typedef struct if_rxd_frag {
|
||||
uint8_t irf_flid;
|
||||
uint16_t irf_idx;
|
||||
uint16_t irf_len;
|
||||
} *if_rxd_frag_t;
|
||||
|
||||
typedef struct if_rxd_info {
|
||||
/* set by iflib */
|
||||
uint16_t iri_qsidx; /* qset index */
|
||||
uint16_t iri_vtag; /* vlan tag - if flag set */
|
||||
/* XXX redundant with the new irf_len field */
|
||||
uint16_t iri_len; /* packet length */
|
||||
uint16_t iri_cidx; /* consumer index of cq */
|
||||
struct ifnet *iri_ifp; /* some drivers >1 interface per softc */
|
||||
@ -156,10 +159,11 @@ typedef struct if_txrx {
|
||||
void (*ift_txd_flush) (void *, uint16_t, uint32_t);
|
||||
int (*ift_txd_credits_update) (void *, uint16_t, uint32_t, bool);
|
||||
|
||||
int (*ift_rxd_available) (void *, uint16_t qsidx, uint32_t pidx);
|
||||
int (*ift_rxd_available) (void *, uint16_t qsidx, uint32_t pidx,
|
||||
int budget);
|
||||
int (*ift_rxd_pkt_get) (void *, if_rxd_info_t ri);
|
||||
void (*ift_rxd_refill) (void * , uint16_t qsidx, uint8_t flidx, uint32_t pidx,
|
||||
uint64_t *paddrs, caddr_t *vaddrs, uint16_t count);
|
||||
uint64_t *paddrs, caddr_t *vaddrs, uint16_t count, uint16_t buf_size);
|
||||
void (*ift_rxd_flush) (void *, uint16_t qsidx, uint8_t flidx, uint32_t pidx);
|
||||
int (*ift_legacy_intr) (void *);
|
||||
} *if_txrx_t;
|
||||
@ -170,11 +174,20 @@ typedef struct if_softc_ctx {
|
||||
int isc_ntxqsets;
|
||||
int isc_msix_bar; /* can be model specific - initialize in attach_pre */
|
||||
int isc_tx_nsegments; /* can be model specific - initialize in attach_pre */
|
||||
int isc_ntxd[8];
|
||||
int isc_nrxd[8];
|
||||
|
||||
uint32_t isc_txqsizes[8];
|
||||
uint32_t isc_rxqsizes[8];
|
||||
int isc_max_txqsets;
|
||||
int isc_max_rxqsets;
|
||||
int isc_tx_tso_segments_max;
|
||||
int isc_tx_tso_size_max;
|
||||
int isc_tx_tso_segsize_max;
|
||||
int isc_rss_table_size;
|
||||
int isc_rss_table_mask;
|
||||
int isc_nrxqsets_max;
|
||||
int isc_ntxqsets_max;
|
||||
|
||||
iflib_intr_mode_t isc_intr;
|
||||
uint16_t isc_max_frame_size; /* set at init time by driver */
|
||||
@ -188,8 +201,6 @@ struct if_shared_ctx {
|
||||
int isc_magic;
|
||||
if_txrx_t isc_txrx;
|
||||
driver_t *isc_driver;
|
||||
int isc_ntxd;
|
||||
int isc_nrxd;
|
||||
int isc_nfl;
|
||||
int isc_flags;
|
||||
bus_size_t isc_q_align;
|
||||
@ -199,14 +210,11 @@ struct if_shared_ctx {
|
||||
bus_size_t isc_rx_maxsegsize;
|
||||
int isc_rx_nsegments;
|
||||
int isc_rx_process_limit;
|
||||
|
||||
|
||||
uint32_t isc_txqsizes[8];
|
||||
int isc_ntxqs; /* # of tx queues per tx qset - usually 1 */
|
||||
uint32_t isc_rxqsizes[8];
|
||||
int isc_nrxqs; /* # of rx queues per rx qset - intel 1, chelsio 2, broadcom 3 */
|
||||
int isc_admin_intrcnt; /* # of admin/link interrupts */
|
||||
|
||||
|
||||
int isc_tx_reclaim_thresh;
|
||||
|
||||
/* fields necessary for probe */
|
||||
@ -215,6 +223,12 @@ struct if_shared_ctx {
|
||||
/* optional function to transform the read values to match the table*/
|
||||
void (*isc_parse_devinfo) (uint16_t *device_id, uint16_t *subvendor_id,
|
||||
uint16_t *subdevice_id, uint16_t *rev_id);
|
||||
int isc_nrxd_min[8];
|
||||
int isc_nrxd_default[8];
|
||||
int isc_nrxd_max[8];
|
||||
int isc_ntxd_min[8];
|
||||
int isc_ntxd_default[8];
|
||||
int isc_ntxd_max[8];
|
||||
};
|
||||
|
||||
typedef struct iflib_dma_info {
|
||||
@ -240,9 +254,9 @@ typedef enum {
|
||||
|
||||
|
||||
/*
|
||||
* Interface has a separate command queue
|
||||
* Interface has a separate command queue for RX
|
||||
*/
|
||||
#define IFLIB_HAS_CQ 0x1
|
||||
#define IFLIB_HAS_RXCQ 0x1
|
||||
/*
|
||||
* Driver has already allocated vectors
|
||||
*/
|
||||
@ -252,6 +266,10 @@ typedef enum {
|
||||
* Interface is a virtual function
|
||||
*/
|
||||
#define IFLIB_IS_VF 0x4
|
||||
/*
|
||||
* Interface has a separate command queue for TX
|
||||
*/
|
||||
#define IFLIB_HAS_TXCQ 0x8
|
||||
|
||||
|
||||
/*
|
||||
@ -308,7 +326,10 @@ void iflib_irq_free(if_ctx_t ctx, if_irq_t irq);
|
||||
void iflib_io_tqg_attach(struct grouptask *gt, void *uniq, int cpu, char *name);
|
||||
|
||||
void iflib_config_gtask_init(if_ctx_t ctx, struct grouptask *gtask,
|
||||
task_fn_t *fn, char *name);
|
||||
gtask_fn_t *fn, char *name);
|
||||
|
||||
void iflib_config_gtask_deinit(struct grouptask *gtask);
|
||||
|
||||
|
||||
|
||||
void iflib_tx_intr_deferred(if_ctx_t ctx, int txqid);
|
||||
@ -317,7 +338,7 @@ void iflib_admin_intr_deferred(if_ctx_t ctx);
|
||||
void iflib_iov_intr_deferred(if_ctx_t ctx);
|
||||
|
||||
|
||||
void iflib_link_state_change(if_ctx_t ctx, int linkstate);
|
||||
void iflib_link_state_change(if_ctx_t ctx, int linkstate, uint64_t baudrate);
|
||||
|
||||
int iflib_dma_alloc(if_ctx_t ctx, int size, iflib_dma_info_t dma, int mapflags);
|
||||
void iflib_dma_free(iflib_dma_info_t dma);
|
||||
|
@ -42,6 +42,7 @@
|
||||
* (q) taskqueue lock
|
||||
*/
|
||||
typedef void task_fn_t(void *context, int pending);
|
||||
typedef void gtask_fn_t(void *context);
|
||||
|
||||
struct task {
|
||||
STAILQ_ENTRY(task) ta_link; /* (q) link for queue */
|
||||
@ -51,8 +52,16 @@ struct task {
|
||||
void *ta_context; /* (c) argument for handler */
|
||||
};
|
||||
|
||||
struct gtask {
|
||||
STAILQ_ENTRY(gtask) ta_link; /* (q) link for queue */
|
||||
uint16_t ta_flags; /* (q) state flags */
|
||||
u_short ta_priority; /* (c) Priority */
|
||||
gtask_fn_t *ta_func; /* (c) task handler */
|
||||
void *ta_context; /* (c) argument for handler */
|
||||
};
|
||||
|
||||
struct grouptask {
|
||||
struct task gt_task;
|
||||
struct gtask gt_task;
|
||||
void *gt_taskqueue;
|
||||
LIST_ENTRY(grouptask) gt_list;
|
||||
void *gt_uniq;
|
||||
|
125
sys/sys/gtaskqueue.h
Normal file
125
sys/sys/gtaskqueue.h
Normal file
@ -0,0 +1,125 @@
|
||||
/*-
|
||||
* Copyright (c) 2014 Jeffrey Roberson <jeff@freebsd.org>
|
||||
* Copyright (c) 2016 Matthew Macy <mmacy@nextbsd.org>
|
||||
* All rights reserved.
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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.
|
||||
*
|
||||
* $FreeBSD$
|
||||
*/
|
||||
|
||||
#ifndef _SYS_GTASKQUEUE_H_
|
||||
#define _SYS_GTASKQUEUE_H_
|
||||
#include <sys/taskqueue.h>
|
||||
|
||||
#ifndef _KERNEL
|
||||
#error "no user-servicable parts inside"
|
||||
#endif
|
||||
|
||||
struct gtaskqueue;
|
||||
typedef void (*gtaskqueue_enqueue_fn)(void *context);
|
||||
|
||||
/*
|
||||
* Taskqueue groups. Manages dynamic thread groups and irq binding for
|
||||
* device and other tasks.
|
||||
*/
|
||||
|
||||
void gtaskqueue_block(struct gtaskqueue *queue);
|
||||
void gtaskqueue_unblock(struct gtaskqueue *queue);
|
||||
|
||||
int gtaskqueue_cancel(struct gtaskqueue *queue, struct gtask *gtask);
|
||||
void gtaskqueue_drain(struct gtaskqueue *queue, struct gtask *task);
|
||||
void gtaskqueue_drain_all(struct gtaskqueue *queue);
|
||||
|
||||
int grouptaskqueue_enqueue(struct gtaskqueue *queue, struct gtask *task);
|
||||
void taskqgroup_attach(struct taskqgroup *qgroup, struct grouptask *grptask,
|
||||
void *uniq, int irq, char *name);
|
||||
int taskqgroup_attach_cpu(struct taskqgroup *qgroup, struct grouptask *grptask,
|
||||
void *uniq, int cpu, int irq, char *name);
|
||||
void taskqgroup_detach(struct taskqgroup *qgroup, struct grouptask *gtask);
|
||||
struct taskqgroup *taskqgroup_create(char *name);
|
||||
void taskqgroup_destroy(struct taskqgroup *qgroup);
|
||||
int taskqgroup_adjust(struct taskqgroup *qgroup, int cnt, int stride);
|
||||
|
||||
#define TASK_ENQUEUED 0x1
|
||||
#define TASK_SKIP_WAKEUP 0x2
|
||||
|
||||
|
||||
#define GTASK_INIT(task, flags, priority, func, context) do { \
|
||||
(task)->ta_flags = flags; \
|
||||
(task)->ta_priority = (priority); \
|
||||
(task)->ta_func = (func); \
|
||||
(task)->ta_context = (context); \
|
||||
} while (0)
|
||||
|
||||
#define GROUPTASK_INIT(gtask, priority, func, context) \
|
||||
GTASK_INIT(&(gtask)->gt_task, TASK_SKIP_WAKEUP, priority, func, context)
|
||||
|
||||
#define GROUPTASK_ENQUEUE(gtask) \
|
||||
grouptaskqueue_enqueue((gtask)->gt_taskqueue, &(gtask)->gt_task)
|
||||
|
||||
#define TASKQGROUP_DECLARE(name) \
|
||||
extern struct taskqgroup *qgroup_##name
|
||||
|
||||
|
||||
#ifdef EARLY_AP_STARTUP
|
||||
#define TASKQGROUP_DEFINE(name, cnt, stride) \
|
||||
\
|
||||
struct taskqgroup *qgroup_##name; \
|
||||
\
|
||||
static void \
|
||||
taskqgroup_define_##name(void *arg) \
|
||||
{ \
|
||||
qgroup_##name = taskqgroup_create(#name); \
|
||||
taskqgroup_adjust(qgroup_##name, (cnt), (stride)); \
|
||||
} \
|
||||
\
|
||||
SYSINIT(taskqgroup_##name, SI_SUB_INIT_IF, SI_ORDER_FIRST, \
|
||||
taskqgroup_define_##name, NULL)
|
||||
#else
|
||||
#define TASKQGROUP_DEFINE(name, cnt, stride) \
|
||||
\
|
||||
struct taskqgroup *qgroup_##name; \
|
||||
\
|
||||
static void \
|
||||
taskqgroup_define_##name(void *arg) \
|
||||
{ \
|
||||
qgroup_##name = taskqgroup_create(#name); \
|
||||
} \
|
||||
\
|
||||
SYSINIT(taskqgroup_##name, SI_SUB_INIT_IF, SI_ORDER_FIRST, \
|
||||
taskqgroup_define_##name, NULL); \
|
||||
\
|
||||
static void \
|
||||
taskqgroup_adjust_##name(void *arg) \
|
||||
{ \
|
||||
taskqgroup_adjust(qgroup_##name, (cnt), (stride)); \
|
||||
} \
|
||||
\
|
||||
SYSINIT(taskqgroup_adj_##name, SI_SUB_SMP, SI_ORDER_ANY, \
|
||||
taskqgroup_adjust_##name, NULL); \
|
||||
\
|
||||
struct __hack
|
||||
#endif
|
||||
TASKQGROUP_DECLARE(net);
|
||||
|
||||
#endif /* !_SYS_GTASKQUEUE_H_ */
|
@ -204,78 +204,4 @@ struct taskqueue *taskqueue_create_fast(const char *name, int mflags,
|
||||
taskqueue_enqueue_fn enqueue,
|
||||
void *context);
|
||||
|
||||
/*
|
||||
* Taskqueue groups. Manages dynamic thread groups and irq binding for
|
||||
* device and other tasks.
|
||||
*/
|
||||
int grouptaskqueue_enqueue(struct taskqueue *queue, struct task *task);
|
||||
void taskqgroup_attach(struct taskqgroup *qgroup, struct grouptask *gtask,
|
||||
void *uniq, int irq, char *name);
|
||||
int taskqgroup_attach_cpu(struct taskqgroup *qgroup, struct grouptask *gtask,
|
||||
void *uniq, int cpu, int irq, char *name);
|
||||
void taskqgroup_detach(struct taskqgroup *qgroup, struct grouptask *gtask);
|
||||
struct taskqgroup *taskqgroup_create(char *name);
|
||||
void taskqgroup_destroy(struct taskqgroup *qgroup);
|
||||
int taskqgroup_adjust(struct taskqgroup *qgroup, int cnt, int stride);
|
||||
|
||||
#define TASK_SKIP_WAKEUP 0x1
|
||||
|
||||
#define GTASK_INIT(task, priority, func, context) do { \
|
||||
(task)->ta_pending = 0; \
|
||||
(task)->ta_priority = (priority); \
|
||||
(task)->ta_func = (func); \
|
||||
(task)->ta_context = (context); \
|
||||
} while (0)
|
||||
|
||||
#define GROUPTASK_INIT(gtask, priority, func, context) \
|
||||
GTASK_INIT(&(gtask)->gt_task, priority, func, context)
|
||||
|
||||
#define GROUPTASK_ENQUEUE(gtask) \
|
||||
grouptaskqueue_enqueue((gtask)->gt_taskqueue, &(gtask)->gt_task)
|
||||
|
||||
#define TASKQGROUP_DECLARE(name) \
|
||||
extern struct taskqgroup *qgroup_##name
|
||||
|
||||
#ifdef EARLY_AP_STARTUP
|
||||
#define TASKQGROUP_DEFINE(name, cnt, stride) \
|
||||
\
|
||||
struct taskqgroup *qgroup_##name; \
|
||||
\
|
||||
static void \
|
||||
taskqgroup_define_##name(void *arg) \
|
||||
{ \
|
||||
qgroup_##name = taskqgroup_create(#name); \
|
||||
taskqgroup_adjust(qgroup_##name, (cnt), (stride)); \
|
||||
} \
|
||||
\
|
||||
SYSINIT(taskqgroup_##name, SI_SUB_INIT_IF, SI_ORDER_FIRST, \
|
||||
taskqgroup_define_##name, NULL)
|
||||
#else
|
||||
#define TASKQGROUP_DEFINE(name, cnt, stride) \
|
||||
\
|
||||
struct taskqgroup *qgroup_##name; \
|
||||
\
|
||||
static void \
|
||||
taskqgroup_define_##name(void *arg) \
|
||||
{ \
|
||||
qgroup_##name = taskqgroup_create(#name); \
|
||||
} \
|
||||
\
|
||||
SYSINIT(taskqgroup_##name, SI_SUB_INIT_IF, SI_ORDER_FIRST, \
|
||||
taskqgroup_define_##name, NULL); \
|
||||
\
|
||||
static void \
|
||||
taskqgroup_adjust_##name(void *arg) \
|
||||
{ \
|
||||
taskqgroup_adjust(qgroup_##name, (cnt), (stride)); \
|
||||
} \
|
||||
\
|
||||
SYSINIT(taskqgroup_adj_##name, SI_SUB_SMP, SI_ORDER_ANY, \
|
||||
taskqgroup_adjust_##name, NULL); \
|
||||
\
|
||||
struct __hack
|
||||
#endif
|
||||
|
||||
TASKQGROUP_DECLARE(net);
|
||||
|
||||
#endif /* !_SYS_TASKQUEUE_H_ */
|
||||
|
Loading…
x
Reference in New Issue
Block a user