workqueue.h 6.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * workqueue.h --- work queue handling for Linux.
 */

#ifndef _LINUX_WORKQUEUE_H
#define _LINUX_WORKQUEUE_H

#include <linux/timer.h>
#include <linux/linkage.h>
#include <linux/bitops.h>

struct workqueue_struct;

14 15
struct work_struct;
typedef void (*work_func_t)(struct work_struct *work);
16

L
Linus Torvalds 已提交
17
struct work_struct {
18 19
	/* the first word is the work queue pointer and the flags rolled into
	 * one */
20 21
	unsigned long management;
#define WORK_STRUCT_PENDING 0		/* T if work item pending execution */
22
#define WORK_STRUCT_NOAUTOREL 1		/* F if work item automatically released on exec */
23 24
#define WORK_STRUCT_FLAG_MASK (3UL)
#define WORK_STRUCT_WQ_DATA_MASK (~WORK_STRUCT_FLAG_MASK)
L
Linus Torvalds 已提交
25
	struct list_head entry;
26
	work_func_t func;
27 28 29 30
};

struct delayed_work {
	struct work_struct work;
L
Linus Torvalds 已提交
31 32 33
	struct timer_list timer;
};

34 35 36 37
struct execute_work {
	struct work_struct work;
};

38 39
#define __WORK_INITIALIZER(n, f) {				\
	.management = 0,					\
L
Linus Torvalds 已提交
40 41
        .entry	= { &(n).entry, &(n).entry },			\
	.func = (f),						\
42 43
	}

44 45 46 47 48 49 50 51 52 53 54 55 56
#define __WORK_INITIALIZER_NAR(n, f) {				\
	.management = (1 << WORK_STRUCT_NOAUTOREL),		\
        .entry	= { &(n).entry, &(n).entry },			\
	.func = (f),						\
	}

#define __DELAYED_WORK_INITIALIZER(n, f) {			\
	.work = __WORK_INITIALIZER((n).work, (f)),		\
	.timer = TIMER_INITIALIZER(NULL, 0, 0),			\
	}

#define __DELAYED_WORK_INITIALIZER_NAR(n, f) {			\
	.work = __WORK_INITIALIZER_NAR((n).work, (f)),		\
L
Linus Torvalds 已提交
57 58 59
	.timer = TIMER_INITIALIZER(NULL, 0, 0),			\
	}

60 61 62 63 64
#define DECLARE_WORK(n, f)					\
	struct work_struct n = __WORK_INITIALIZER(n, f)

#define DECLARE_WORK_NAR(n, f)					\
	struct work_struct n = __WORK_INITIALIZER_NAR(n, f)
L
Linus Torvalds 已提交
65

66 67 68 69 70
#define DECLARE_DELAYED_WORK(n, f)				\
	struct delayed_work n = __DELAYED_WORK_INITIALIZER(n, f)

#define DECLARE_DELAYED_WORK_NAR(n, f)			\
	struct dwork_struct n = __DELAYED_WORK_INITIALIZER_NAR(n, f)
71

L
Linus Torvalds 已提交
72
/*
73
 * initialize a work item's function pointer
L
Linus Torvalds 已提交
74
 */
75
#define PREPARE_WORK(_work, _func)				\
L
Linus Torvalds 已提交
76
	do {							\
77
		(_work)->func = (_func);			\
L
Linus Torvalds 已提交
78 79
	} while (0)

80 81
#define PREPARE_DELAYED_WORK(_work, _func)			\
	PREPARE_WORK(&(_work)->work, (_func))
82

L
Linus Torvalds 已提交
83
/*
84
 * initialize all of a work item in one go
L
Linus Torvalds 已提交
85
 */
86
#define INIT_WORK(_work, _func)					\
L
Linus Torvalds 已提交
87
	do {							\
88
		(_work)->management = 0;			\
89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
		INIT_LIST_HEAD(&(_work)->entry);		\
		PREPARE_WORK((_work), (_func));			\
	} while (0)

#define INIT_WORK_NAR(_work, _func)					\
	do {								\
		(_work)->management = (1 << WORK_STRUCT_NOAUTOREL);	\
		INIT_LIST_HEAD(&(_work)->entry);			\
		PREPARE_WORK((_work), (_func));				\
	} while (0)

#define INIT_DELAYED_WORK(_work, _func)				\
	do {							\
		INIT_WORK(&(_work)->work, (_func));		\
		init_timer(&(_work)->timer);			\
104 105
	} while (0)

106
#define INIT_DELAYED_WORK_NAR(_work, _func)			\
107
	do {							\
108
		INIT_WORK_NAR(&(_work)->work, (_func));		\
L
Linus Torvalds 已提交
109 110 111
		init_timer(&(_work)->timer);			\
	} while (0)

112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
/**
 * work_pending - Find out whether a work item is currently pending
 * @work: The work item in question
 */
#define work_pending(work) \
	test_bit(WORK_STRUCT_PENDING, &(work)->management)

/**
 * delayed_work_pending - Find out whether a delayable work item is currently
 * pending
 * @work: The work item in question
 */
#define delayed_work_pending(work) \
	test_bit(WORK_STRUCT_PENDING, &(work)->work.management)

127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
/**
 * work_release - Release a work item under execution
 * @work: The work item to release
 *
 * This is used to release a work item that has been initialised with automatic
 * release mode disabled (WORK_STRUCT_NOAUTOREL is set).  This gives the work
 * function the opportunity to grab auxiliary data from the container of the
 * work_struct before clearing the pending bit as the work_struct may be
 * subject to deallocation the moment the pending bit is cleared.
 *
 * In such a case, this should be called in the work function after it has
 * fetched any data it may require from the containter of the work_struct.
 * After this function has been called, the work_struct may be scheduled for
 * further execution or it may be deallocated unless other precautions are
 * taken.
 *
 * This should also be used to release a delayed work item.
 */
#define work_release(work) \
	clear_bit(WORK_STRUCT_PENDING, &(work)->management)

148

L
Linus Torvalds 已提交
149
extern struct workqueue_struct *__create_workqueue(const char *name,
150 151 152 153 154
						    int singlethread,
						    int freezeable);
#define create_workqueue(name) __create_workqueue((name), 0, 0)
#define create_freezeable_workqueue(name) __create_workqueue((name), 0, 1)
#define create_singlethread_workqueue(name) __create_workqueue((name), 1, 0)
L
Linus Torvalds 已提交
155 156 157 158

extern void destroy_workqueue(struct workqueue_struct *wq);

extern int FASTCALL(queue_work(struct workqueue_struct *wq, struct work_struct *work));
159
extern int FASTCALL(queue_delayed_work(struct workqueue_struct *wq, struct delayed_work *work, unsigned long delay));
160
extern int queue_delayed_work_on(int cpu, struct workqueue_struct *wq,
161
	struct delayed_work *work, unsigned long delay);
L
Linus Torvalds 已提交
162 163 164
extern void FASTCALL(flush_workqueue(struct workqueue_struct *wq));

extern int FASTCALL(schedule_work(struct work_struct *work));
165
extern int FASTCALL(run_scheduled_work(struct work_struct *work));
166
extern int FASTCALL(schedule_delayed_work(struct delayed_work *work, unsigned long delay));
L
Linus Torvalds 已提交
167

168
extern int schedule_delayed_work_on(int cpu, struct delayed_work *work, unsigned long delay);
169
extern int schedule_on_each_cpu(work_func_t func);
L
Linus Torvalds 已提交
170 171 172 173 174
extern void flush_scheduled_work(void);
extern int current_is_keventd(void);
extern int keventd_up(void);

extern void init_workqueues(void);
175
void cancel_rearming_delayed_work(struct delayed_work *work);
176
void cancel_rearming_delayed_workqueue(struct workqueue_struct *,
177
				       struct delayed_work *);
178
int execute_in_process_context(work_func_t fn, struct execute_work *);
L
Linus Torvalds 已提交
179 180 181 182 183 184

/*
 * Kill off a pending schedule_delayed_work().  Note that the work callback
 * function may still be running on return from cancel_delayed_work().  Run
 * flush_scheduled_work() to wait on it.
 */
185
static inline int cancel_delayed_work(struct delayed_work *work)
L
Linus Torvalds 已提交
186 187 188 189 190
{
	int ret;

	ret = del_timer_sync(&work->timer);
	if (ret)
191
		clear_bit(WORK_STRUCT_PENDING, &work->work.management);
L
Linus Torvalds 已提交
192 193 194 195
	return ret;
}

#endif