f724f903d41e5900c8aee5d4d9437f184a32851b
[fio.git] / workqueue.h
1 #ifndef FIO_RATE_H
2 #define FIO_RATE_H
3
4 #include "flist.h"
5
6 struct workqueue_work {
7         struct flist_head list;
8 };
9
10 struct submit_worker {
11         pthread_t thread;
12         pthread_mutex_t lock;
13         pthread_cond_t cond;
14         struct flist_head work_list;
15         unsigned int flags;
16         unsigned int index;
17         uint64_t seq;
18         struct workqueue *wq;
19         void *private;
20 };
21
22 typedef void (workqueue_work_fn)(struct submit_worker *, struct workqueue_work *);
23 typedef bool (workqueue_pre_sleep_flush_fn)(struct submit_worker *);
24 typedef void (workqueue_pre_sleep_fn)(struct submit_worker *);
25 typedef int (workqueue_alloc_worker_fn)(struct submit_worker *);
26 typedef void (workqueue_free_worker_fn)(struct submit_worker *);
27 typedef int (workqueue_init_worker_fn)(struct submit_worker *);
28
29 struct workqueue_ops {
30         workqueue_work_fn *fn;
31         workqueue_pre_sleep_flush_fn *pre_sleep_flush_fn;
32         workqueue_pre_sleep_fn *pre_sleep_fn;
33         workqueue_alloc_worker_fn *alloc_worker_fn;
34         workqueue_free_worker_fn *free_worker_fn;
35         workqueue_init_worker_fn *init_worker_fn;
36 };
37
38 struct workqueue {
39         unsigned int max_workers;
40
41         struct thread_data *td;
42         struct workqueue_ops ops;
43
44         uint64_t work_seq;
45         struct submit_worker *workers;
46         unsigned int next_free_worker;
47
48         pthread_cond_t flush_cond;
49         pthread_mutex_t flush_lock;
50         pthread_mutex_t stat_lock;
51         volatile int wake_idle;
52 };
53
54 int workqueue_init(struct thread_data *td, struct workqueue *wq, struct workqueue_ops *ops, unsigned int max_workers);
55 void workqueue_exit(struct workqueue *wq);
56
57 bool workqueue_enqueue(struct workqueue *wq, struct workqueue_work *work);
58 void workqueue_flush(struct workqueue *wq);
59
60 static inline bool workqueue_pre_sleep_check(struct submit_worker *sw)
61 {
62         struct workqueue *wq = sw->wq;
63
64         if (!wq->ops.pre_sleep_flush_fn)
65                 return false;
66
67         return wq->ops.pre_sleep_flush_fn(sw);
68 }
69
70 static inline void workqueue_pre_sleep(struct submit_worker *sw)
71 {
72         struct workqueue *wq = sw->wq;
73
74         if (wq->ops.pre_sleep_fn)
75                 wq->ops.pre_sleep_fn(sw);
76 }
77
78 static inline int workqueue_init_worker(struct submit_worker *sw)
79 {
80         struct workqueue *wq = sw->wq;
81
82         if (!wq->ops.init_worker_fn)
83                 return 0;
84
85         return wq->ops.init_worker_fn(sw);
86 }
87
88 #endif