taskqueues
John Baldwin
jhb at freebsd.org
Wed May 29 18:16:45 UTC 2013
On Thursday, May 23, 2013 9:47:14 am Orit Moskovich wrote:
> Hi,
>
> Can you please specify the difference between interrupt threads and regular
kernel threads in the context of the different default taskqueues?
> Meaning, I saw that the taskqueue taskqueue_thread is actually a kernel
thread running the function taskqueue_thread_loop,
> And the 3 other default taskqueues are working with ithreads.
>
> Which of the above preempts the other? What should I use if performance is
critical (something equivalent to Linux softirq...)?
ithreads use the most important priority range:
#define PRI_MIN (0) /* Highest priority. */
#define PRI_MAX (255) /* Lowest priority. */
#define PRI_MIN_ITHD (PRI_MIN)
#define PRI_MAX_ITHD (PRI_MIN_REALTIME - 1)
#define PI_REALTIME (PRI_MIN_ITHD + 0)
#define PI_AV (PRI_MIN_ITHD + 4)
#define PI_NET (PRI_MIN_ITHD + 8)
#define PI_DISK (PRI_MIN_ITHD + 12)
#define PI_TTY (PRI_MIN_ITHD + 16)
#define PI_DULL (PRI_MIN_ITHD + 20)
#define PI_SOFT (PRI_MIN_ITHD + 24)
#define PI_SWI(x) (PI_SOFT + (x) * RQ_PPQ)
For example, an INTR_TYPE_NET interrupt will generally run in an ithread with
the PI_NET priority. For software interrupt threads the priority is generally
PI_SWI(SWI_xxx). The SWI_xxx constants are:
/*
* Software interrupt numbers in priority order. The priority determines
* the priority of the corresponding interrupt thread.
*/
#define SWI_TTY 0
#define SWI_NET 1
#define SWI_CAMBIO 2
#define SWI_VM 3
#define SWI_CLOCK 4
#define SWI_TQ_FAST 5
#define SWI_TQ 6
#define SWI_TQ_GIANT 6
For taskqueues you have three global taskqueues to choose from:
taskqueue_fast: runs at PI_SWI(SWI_TQ_FAST) = 44
taskqueue_swi: runs at PI_SWI(SWI_TQ) = 48
taskqueue_thread: runs at PWAIT = 108
However, you can create your own thread-backed taskqueue that can run at
whatever priority you choose. There is nothing really special about having
the taskqueue be an SWI vs a normal thread. If you wanted a taskqueue that
ran at PI_NET you could do:
TASKQUEUE_DEFINE(foo, taskqueue_thread_enqueue, &taskqueue_foo,
taskqueue_start_threads(&taskqueue_foo, 1, PI_NET, "foo taskq");
If you needed to queue tasks from an interrupt filter you would want to use
a fast taskqueue instead:
TASKQUEUE_FAST_DEFINE(< same args as above >)
--
John Baldwin
More information about the freebsd-drivers
mailing list