1 /******************************************************************************
2 * tasklet.h
3 *
4 * Tasklets are dynamically-allocatable tasks run in either VCPU context
5 * (specifically, the idle VCPU's context) or in softirq context, on at most
6 * one CPU at a time. Softirq versus VCPU context execution is specified
7 * during per-tasklet initialisation.
8 */
9
10 #ifndef __XEN_TASKLET_H__
11 #define __XEN_TASKLET_H__
12
13 #include <xen/types.h>
14 #include <xen/list.h>
15 #include <xen/percpu.h>
16
17 struct tasklet
18 {
19 struct list_head list;
20 int scheduled_on;
21 bool_t is_softirq;
22 bool_t is_running;
23 bool_t is_dead;
24 void (*func)(void *);
25 void *data;
26 };
27
28 #define _DECLARE_TASKLET(name, func, data, softirq) \
29 struct tasklet name = { \
30 LIST_HEAD_INIT(name.list), -1, softirq, 0, 0, func, data }
31 #define DECLARE_TASKLET(name, func, data) \
32 _DECLARE_TASKLET(name, func, data, 0)
33 #define DECLARE_SOFTIRQ_TASKLET(name, func, data) \
34 _DECLARE_TASKLET(name, func, data, 1)
35
36 /* Indicates status of tasklet work on each CPU. */
37 DECLARE_PER_CPU(unsigned long, tasklet_work_to_do);
38 #define _TASKLET_enqueued 0 /* Tasklet work is enqueued for this CPU. */
39 #define _TASKLET_scheduled 1 /* Scheduler has scheduled do_tasklet(). */
40 #define TASKLET_enqueued (1ul << _TASKLET_enqueued)
41 #define TASKLET_scheduled (1ul << _TASKLET_scheduled)
42
tasklet_work_to_do(unsigned int cpu)43 static inline bool tasklet_work_to_do(unsigned int cpu)
44 {
45 /*
46 * Work must be enqueued *and* scheduled. Otherwise there is no work to
47 * do, and/or scheduler needs to run to update idle vcpu priority.
48 */
49 return per_cpu(tasklet_work_to_do, cpu) == (TASKLET_enqueued|
50 TASKLET_scheduled);
51 }
52
tasklet_is_scheduled(const struct tasklet * t)53 static inline bool tasklet_is_scheduled(const struct tasklet *t)
54 {
55 return t->scheduled_on != -1;
56 }
57
58 void tasklet_schedule_on_cpu(struct tasklet *t, unsigned int cpu);
59 void tasklet_schedule(struct tasklet *t);
60 void do_tasklet(void);
61 void tasklet_kill(struct tasklet *t);
62 void tasklet_init(struct tasklet *t, void (*func)(void *), void *data);
63 void softirq_tasklet_init(struct tasklet *t, void (*func)(void *), void *data);
64 void tasklet_subsys_init(void);
65
66 #endif /* __XEN_TASKLET_H__ */
67