Blame view

kernel/async.c 8.96 KB
22a9d6456   Arjan van de Ven   async: Asynchrono...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
  /*
   * async.c: Asynchronous function calls for boot performance
   *
   * (C) Copyright 2009 Intel Corporation
   * Author: Arjan van de Ven <arjan@linux.intel.com>
   *
   * This program is free software; you can redistribute it and/or
   * modify it under the terms of the GNU General Public License
   * as published by the Free Software Foundation; version 2
   * of the License.
   */
  
  
  /*
  
  Goals and Theory of Operation
  
  The primary goal of this feature is to reduce the kernel boot time,
  by doing various independent hardware delays and discovery operations
  decoupled and not strictly serialized.
  
  More specifically, the asynchronous function call concept allows
  certain operations (primarily during system boot) to happen
  asynchronously, out of order, while these operations still
  have their externally visible parts happen sequentially and in-order.
  (not unlike how out-of-order CPUs retire their instructions in order)
  
  Key to the asynchronous function call implementation is the concept of
  a "sequence cookie" (which, although it has an abstracted type, can be
  thought of as a monotonically incrementing number).
  
  The async core will assign each scheduled event such a sequence cookie and
  pass this to the called functions.
  
  The asynchronously called function should before doing a globally visible
  operation, such as registering device numbers, call the
  async_synchronize_cookie() function and pass in its own cookie. The
  async_synchronize_cookie() function will make sure that all asynchronous
  operations that were scheduled prior to the operation corresponding with the
  cookie have completed.
  
  Subsystem/driver initialization code that scheduled asynchronous probe
  functions, but which shares global resources with other drivers/subsystems
  that do not use the asynchronous call feature, need to do a full
  synchronization with the async_synchronize_full() function, before returning
  from their init function. This is to maintain strict ordering between the
  asynchronous and synchronous parts of the kernel.
  
  */
  
  #include <linux/async.h>
  #include <linux/module.h>
  #include <linux/wait.h>
  #include <linux/sched.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
55
  #include <linux/slab.h>
083b804c4   Tejun Heo   async: use workqu...
56
  #include <linux/workqueue.h>
22a9d6456   Arjan van de Ven   async: Asynchrono...
57
58
59
  #include <asm/atomic.h>
  
  static async_cookie_t next_cookie = 1;
22a9d6456   Arjan van de Ven   async: Asynchrono...
60
61
62
63
64
65
66
  #define MAX_WORK	32768
  
  static LIST_HEAD(async_pending);
  static LIST_HEAD(async_running);
  static DEFINE_SPINLOCK(async_lock);
  
  struct async_entry {
083b804c4   Tejun Heo   async: use workqu...
67
68
69
70
71
72
  	struct list_head	list;
  	struct work_struct	work;
  	async_cookie_t		cookie;
  	async_func_ptr		*func;
  	void			*data;
  	struct list_head	*running;
22a9d6456   Arjan van de Ven   async: Asynchrono...
73
74
75
  };
  
  static DECLARE_WAIT_QUEUE_HEAD(async_done);
22a9d6456   Arjan van de Ven   async: Asynchrono...
76
77
  
  static atomic_t entry_count;
22a9d6456   Arjan van de Ven   async: Asynchrono...
78
79
80
81
82
83
84
85
86
87
  
  extern int initcall_debug;
  
  
  /*
   * MUST be called with the lock held!
   */
  static async_cookie_t  __lowest_in_progress(struct list_head *running)
  {
  	struct async_entry *entry;
d5a877e8d   James Bottomley   async: make sure ...
88

37a76bd4f   Arjan van de Ven   async: fix __lowe...
89
90
  	if (!list_empty(running)) {
  		entry = list_first_entry(running,
22a9d6456   Arjan van de Ven   async: Asynchrono...
91
  			struct async_entry, list);
3af968e06   Linus Torvalds   async: Fix lack o...
92
  		return entry->cookie;
22a9d6456   Arjan van de Ven   async: Asynchrono...
93
  	}
3af968e06   Linus Torvalds   async: Fix lack o...
94
95
96
  	list_for_each_entry(entry, &async_pending, list)
  		if (entry->running == running)
  			return entry->cookie;
d5a877e8d   James Bottomley   async: make sure ...
97

3af968e06   Linus Torvalds   async: Fix lack o...
98
  	return next_cookie;	/* "infinity" value */
22a9d6456   Arjan van de Ven   async: Asynchrono...
99
  }
37a76bd4f   Arjan van de Ven   async: fix __lowe...
100
101
102
103
104
105
106
107
108
109
110
  
  static async_cookie_t  lowest_in_progress(struct list_head *running)
  {
  	unsigned long flags;
  	async_cookie_t ret;
  
  	spin_lock_irqsave(&async_lock, flags);
  	ret = __lowest_in_progress(running);
  	spin_unlock_irqrestore(&async_lock, flags);
  	return ret;
  }
083b804c4   Tejun Heo   async: use workqu...
111

22a9d6456   Arjan van de Ven   async: Asynchrono...
112
113
114
  /*
   * pick the first pending entry and run it
   */
083b804c4   Tejun Heo   async: use workqu...
115
  static void async_run_entry_fn(struct work_struct *work)
22a9d6456   Arjan van de Ven   async: Asynchrono...
116
  {
083b804c4   Tejun Heo   async: use workqu...
117
118
  	struct async_entry *entry =
  		container_of(work, struct async_entry, work);
22a9d6456   Arjan van de Ven   async: Asynchrono...
119
  	unsigned long flags;
22a9d6456   Arjan van de Ven   async: Asynchrono...
120
  	ktime_t calltime, delta, rettime;
083b804c4   Tejun Heo   async: use workqu...
121
  	/* 1) move self to the running queue */
22a9d6456   Arjan van de Ven   async: Asynchrono...
122
  	spin_lock_irqsave(&async_lock, flags);
f7de7621f   Stefan Richter   async: use list_m...
123
  	list_move_tail(&entry->list, entry->running);
22a9d6456   Arjan van de Ven   async: Asynchrono...
124
  	spin_unlock_irqrestore(&async_lock, flags);
083b804c4   Tejun Heo   async: use workqu...
125
  	/* 2) run (and print duration) */
ad160d231   Arjan van de Ven   async: don't do t...
126
  	if (initcall_debug && system_state == SYSTEM_BOOTING) {
58763a297   Andrew Morton   kernel/async.c: f...
127
128
129
  		printk("calling  %lli_%pF @ %i
  ", (long long)entry->cookie,
  			entry->func, task_pid_nr(current));
22a9d6456   Arjan van de Ven   async: Asynchrono...
130
131
132
  		calltime = ktime_get();
  	}
  	entry->func(entry->data, entry->cookie);
ad160d231   Arjan van de Ven   async: don't do t...
133
  	if (initcall_debug && system_state == SYSTEM_BOOTING) {
22a9d6456   Arjan van de Ven   async: Asynchrono...
134
135
  		rettime = ktime_get();
  		delta = ktime_sub(rettime, calltime);
58763a297   Andrew Morton   kernel/async.c: f...
136
137
138
139
140
  		printk("initcall %lli_%pF returned 0 after %lld usecs
  ",
  			(long long)entry->cookie,
  			entry->func,
  			(long long)ktime_to_ns(delta) >> 10);
22a9d6456   Arjan van de Ven   async: Asynchrono...
141
  	}
083b804c4   Tejun Heo   async: use workqu...
142
  	/* 3) remove self from the running queue */
22a9d6456   Arjan van de Ven   async: Asynchrono...
143
144
  	spin_lock_irqsave(&async_lock, flags);
  	list_del(&entry->list);
083b804c4   Tejun Heo   async: use workqu...
145
  	/* 4) free the entry */
22a9d6456   Arjan van de Ven   async: Asynchrono...
146
147
148
149
  	kfree(entry);
  	atomic_dec(&entry_count);
  
  	spin_unlock_irqrestore(&async_lock, flags);
083b804c4   Tejun Heo   async: use workqu...
150
  	/* 5) wake up any waiters */
22a9d6456   Arjan van de Ven   async: Asynchrono...
151
  	wake_up(&async_done);
22a9d6456   Arjan van de Ven   async: Asynchrono...
152
  }
22a9d6456   Arjan van de Ven   async: Asynchrono...
153
154
155
156
157
  static async_cookie_t __async_schedule(async_func_ptr *ptr, void *data, struct list_head *running)
  {
  	struct async_entry *entry;
  	unsigned long flags;
  	async_cookie_t newcookie;
22a9d6456   Arjan van de Ven   async: Asynchrono...
158
159
160
161
162
163
164
165
  
  	/* allow irq-off callers */
  	entry = kzalloc(sizeof(struct async_entry), GFP_ATOMIC);
  
  	/*
  	 * If we're out of memory or if there's too much work
  	 * pending already, we execute synchronously.
  	 */
083b804c4   Tejun Heo   async: use workqu...
166
  	if (!entry || atomic_read(&entry_count) > MAX_WORK) {
22a9d6456   Arjan van de Ven   async: Asynchrono...
167
168
169
170
171
172
173
174
175
  		kfree(entry);
  		spin_lock_irqsave(&async_lock, flags);
  		newcookie = next_cookie++;
  		spin_unlock_irqrestore(&async_lock, flags);
  
  		/* low on memory.. run synchronously */
  		ptr(data, newcookie);
  		return newcookie;
  	}
083b804c4   Tejun Heo   async: use workqu...
176
  	INIT_WORK(&entry->work, async_run_entry_fn);
22a9d6456   Arjan van de Ven   async: Asynchrono...
177
178
179
180
181
182
183
184
185
  	entry->func = ptr;
  	entry->data = data;
  	entry->running = running;
  
  	spin_lock_irqsave(&async_lock, flags);
  	newcookie = entry->cookie = next_cookie++;
  	list_add_tail(&entry->list, &async_pending);
  	atomic_inc(&entry_count);
  	spin_unlock_irqrestore(&async_lock, flags);
083b804c4   Tejun Heo   async: use workqu...
186
187
188
  
  	/* schedule for execution */
  	queue_work(system_unbound_wq, &entry->work);
22a9d6456   Arjan van de Ven   async: Asynchrono...
189
190
  	return newcookie;
  }
f30d5b307   Cornelia Huck   async: Add some d...
191
192
193
194
195
196
197
198
  /**
   * async_schedule - schedule a function for asynchronous execution
   * @ptr: function to execute asynchronously
   * @data: data pointer to pass to the function
   *
   * Returns an async_cookie_t that may be used for checkpointing later.
   * Note: This function may be called from atomic or non-atomic contexts.
   */
22a9d6456   Arjan van de Ven   async: Asynchrono...
199
200
  async_cookie_t async_schedule(async_func_ptr *ptr, void *data)
  {
7a89bbc74   Cornelia Huck   async: Fix runnin...
201
  	return __async_schedule(ptr, data, &async_running);
22a9d6456   Arjan van de Ven   async: Asynchrono...
202
203
  }
  EXPORT_SYMBOL_GPL(async_schedule);
f30d5b307   Cornelia Huck   async: Add some d...
204
  /**
766ccb9ed   Cornelia Huck   async: Rename _sp...
205
   * async_schedule_domain - schedule a function for asynchronous execution within a certain domain
f30d5b307   Cornelia Huck   async: Add some d...
206
207
   * @ptr: function to execute asynchronously
   * @data: data pointer to pass to the function
766ccb9ed   Cornelia Huck   async: Rename _sp...
208
   * @running: running list for the domain
f30d5b307   Cornelia Huck   async: Add some d...
209
210
   *
   * Returns an async_cookie_t that may be used for checkpointing later.
766ccb9ed   Cornelia Huck   async: Rename _sp...
211
212
213
   * @running may be used in the async_synchronize_*_domain() functions
   * to wait within a certain synchronization domain rather than globally.
   * A synchronization domain is specified via the running queue @running to use.
f30d5b307   Cornelia Huck   async: Add some d...
214
215
   * Note: This function may be called from atomic or non-atomic contexts.
   */
766ccb9ed   Cornelia Huck   async: Rename _sp...
216
217
  async_cookie_t async_schedule_domain(async_func_ptr *ptr, void *data,
  				     struct list_head *running)
22a9d6456   Arjan van de Ven   async: Asynchrono...
218
219
220
  {
  	return __async_schedule(ptr, data, running);
  }
766ccb9ed   Cornelia Huck   async: Rename _sp...
221
  EXPORT_SYMBOL_GPL(async_schedule_domain);
22a9d6456   Arjan van de Ven   async: Asynchrono...
222

f30d5b307   Cornelia Huck   async: Add some d...
223
224
225
226
227
  /**
   * async_synchronize_full - synchronize all asynchronous function calls
   *
   * This function waits until all asynchronous function calls have been done.
   */
22a9d6456   Arjan van de Ven   async: Asynchrono...
228
229
  void async_synchronize_full(void)
  {
33b04b930   Arjan van de Ven   async: make async...
230
231
232
  	do {
  		async_synchronize_cookie(next_cookie);
  	} while (!list_empty(&async_running) || !list_empty(&async_pending));
22a9d6456   Arjan van de Ven   async: Asynchrono...
233
234
  }
  EXPORT_SYMBOL_GPL(async_synchronize_full);
f30d5b307   Cornelia Huck   async: Add some d...
235
  /**
766ccb9ed   Cornelia Huck   async: Rename _sp...
236
   * async_synchronize_full_domain - synchronize all asynchronous function within a certain domain
f30d5b307   Cornelia Huck   async: Add some d...
237
238
   * @list: running list to synchronize on
   *
766ccb9ed   Cornelia Huck   async: Rename _sp...
239
240
   * This function waits until all asynchronous function calls for the
   * synchronization domain specified by the running list @list have been done.
f30d5b307   Cornelia Huck   async: Add some d...
241
   */
766ccb9ed   Cornelia Huck   async: Rename _sp...
242
  void async_synchronize_full_domain(struct list_head *list)
22a9d6456   Arjan van de Ven   async: Asynchrono...
243
  {
766ccb9ed   Cornelia Huck   async: Rename _sp...
244
  	async_synchronize_cookie_domain(next_cookie, list);
22a9d6456   Arjan van de Ven   async: Asynchrono...
245
  }
766ccb9ed   Cornelia Huck   async: Rename _sp...
246
  EXPORT_SYMBOL_GPL(async_synchronize_full_domain);
22a9d6456   Arjan van de Ven   async: Asynchrono...
247

f30d5b307   Cornelia Huck   async: Add some d...
248
  /**
766ccb9ed   Cornelia Huck   async: Rename _sp...
249
   * async_synchronize_cookie_domain - synchronize asynchronous function calls within a certain domain with cookie checkpointing
f30d5b307   Cornelia Huck   async: Add some d...
250
251
252
   * @cookie: async_cookie_t to use as checkpoint
   * @running: running list to synchronize on
   *
766ccb9ed   Cornelia Huck   async: Rename _sp...
253
254
255
   * This function waits until all asynchronous function calls for the
   * synchronization domain specified by the running list @list submitted
   * prior to @cookie have been done.
f30d5b307   Cornelia Huck   async: Add some d...
256
   */
766ccb9ed   Cornelia Huck   async: Rename _sp...
257
258
  void async_synchronize_cookie_domain(async_cookie_t cookie,
  				     struct list_head *running)
22a9d6456   Arjan van de Ven   async: Asynchrono...
259
260
  {
  	ktime_t starttime, delta, endtime;
ad160d231   Arjan van de Ven   async: don't do t...
261
  	if (initcall_debug && system_state == SYSTEM_BOOTING) {
22a9d6456   Arjan van de Ven   async: Asynchrono...
262
263
264
265
  		printk("async_waiting @ %i
  ", task_pid_nr(current));
  		starttime = ktime_get();
  	}
37a76bd4f   Arjan van de Ven   async: fix __lowe...
266
  	wait_event(async_done, lowest_in_progress(running) >= cookie);
22a9d6456   Arjan van de Ven   async: Asynchrono...
267

ad160d231   Arjan van de Ven   async: don't do t...
268
  	if (initcall_debug && system_state == SYSTEM_BOOTING) {
22a9d6456   Arjan van de Ven   async: Asynchrono...
269
270
271
272
273
  		endtime = ktime_get();
  		delta = ktime_sub(endtime, starttime);
  
  		printk("async_continuing @ %i after %lli usec
  ",
58763a297   Andrew Morton   kernel/async.c: f...
274
275
  			task_pid_nr(current),
  			(long long)ktime_to_ns(delta) >> 10);
22a9d6456   Arjan van de Ven   async: Asynchrono...
276
277
  	}
  }
766ccb9ed   Cornelia Huck   async: Rename _sp...
278
  EXPORT_SYMBOL_GPL(async_synchronize_cookie_domain);
22a9d6456   Arjan van de Ven   async: Asynchrono...
279

f30d5b307   Cornelia Huck   async: Add some d...
280
281
282
283
284
285
286
  /**
   * async_synchronize_cookie - synchronize asynchronous function calls with cookie checkpointing
   * @cookie: async_cookie_t to use as checkpoint
   *
   * This function waits until all asynchronous function calls prior to @cookie
   * have been done.
   */
22a9d6456   Arjan van de Ven   async: Asynchrono...
287
288
  void async_synchronize_cookie(async_cookie_t cookie)
  {
766ccb9ed   Cornelia Huck   async: Rename _sp...
289
  	async_synchronize_cookie_domain(cookie, &async_running);
22a9d6456   Arjan van de Ven   async: Asynchrono...
290
291
  }
  EXPORT_SYMBOL_GPL(async_synchronize_cookie);