Blame view

kernel/async.c 9.04 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
  /*
   * 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>
84c15027a   Paul McQuade   async: Fixed an i...
52
53
  #include <linux/atomic.h>
  #include <linux/ktime.h>
9984de1a5   Paul Gortmaker   kernel: Map most ...
54
  #include <linux/export.h>
22a9d6456   Arjan van de Ven   async: Asynchrono...
55
56
  #include <linux/wait.h>
  #include <linux/sched.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
57
  #include <linux/slab.h>
083b804c4   Tejun Heo   async: use workqu...
58
  #include <linux/workqueue.h>
22a9d6456   Arjan van de Ven   async: Asynchrono...
59
60
  
  static async_cookie_t next_cookie = 1;
22a9d6456   Arjan van de Ven   async: Asynchrono...
61
62
63
64
65
66
67
  #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...
68
69
70
71
72
73
  	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...
74
75
76
  };
  
  static DECLARE_WAIT_QUEUE_HEAD(async_done);
22a9d6456   Arjan van de Ven   async: Asynchrono...
77
78
  
  static atomic_t entry_count;
22a9d6456   Arjan van de Ven   async: Asynchrono...
79

22a9d6456   Arjan van de Ven   async: Asynchrono...
80
81
82
83
84
85
86
  
  /*
   * 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 ...
87

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

3af968e06   Linus Torvalds   async: Fix lack o...
97
  	return next_cookie;	/* "infinity" value */
22a9d6456   Arjan van de Ven   async: Asynchrono...
98
  }
37a76bd4f   Arjan van de Ven   async: fix __lowe...
99
100
101
102
103
104
105
106
107
108
109
  
  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...
110

22a9d6456   Arjan van de Ven   async: Asynchrono...
111
112
113
  /*
   * pick the first pending entry and run it
   */
083b804c4   Tejun Heo   async: use workqu...
114
  static void async_run_entry_fn(struct work_struct *work)
22a9d6456   Arjan van de Ven   async: Asynchrono...
115
  {
083b804c4   Tejun Heo   async: use workqu...
116
117
  	struct async_entry *entry =
  		container_of(work, struct async_entry, work);
22a9d6456   Arjan van de Ven   async: Asynchrono...
118
  	unsigned long flags;
124ff4e53   Vitaliy Ivanov   async: uninitiali...
119
  	ktime_t uninitialized_var(calltime), delta, rettime;
22a9d6456   Arjan van de Ven   async: Asynchrono...
120

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) {
84c15027a   Paul McQuade   async: Fixed an i...
127
128
129
  		printk(KERN_DEBUG "calling  %lli_%pF @ %i
  ",
  			(long long)entry->cookie,
58763a297   Andrew Morton   kernel/async.c: f...
130
  			entry->func, task_pid_nr(current));
22a9d6456   Arjan van de Ven   async: Asynchrono...
131
132
133
  		calltime = ktime_get();
  	}
  	entry->func(entry->data, entry->cookie);
ad160d231   Arjan van de Ven   async: don't do t...
134
  	if (initcall_debug && system_state == SYSTEM_BOOTING) {
22a9d6456   Arjan van de Ven   async: Asynchrono...
135
136
  		rettime = ktime_get();
  		delta = ktime_sub(rettime, calltime);
84c15027a   Paul McQuade   async: Fixed an i...
137
138
  		printk(KERN_DEBUG "initcall %lli_%pF returned 0 after %lld usecs
  ",
58763a297   Andrew Morton   kernel/async.c: f...
139
140
141
  			(long long)entry->cookie,
  			entry->func,
  			(long long)ktime_to_ns(delta) >> 10);
22a9d6456   Arjan van de Ven   async: Asynchrono...
142
  	}
083b804c4   Tejun Heo   async: use workqu...
143
  	/* 3) remove self from the running queue */
22a9d6456   Arjan van de Ven   async: Asynchrono...
144
145
  	spin_lock_irqsave(&async_lock, flags);
  	list_del(&entry->list);
083b804c4   Tejun Heo   async: use workqu...
146
  	/* 4) free the entry */
22a9d6456   Arjan van de Ven   async: Asynchrono...
147
148
149
150
  	kfree(entry);
  	atomic_dec(&entry_count);
  
  	spin_unlock_irqrestore(&async_lock, flags);
083b804c4   Tejun Heo   async: use workqu...
151
  	/* 5) wake up any waiters */
22a9d6456   Arjan van de Ven   async: Asynchrono...
152
  	wake_up(&async_done);
22a9d6456   Arjan van de Ven   async: Asynchrono...
153
  }
22a9d6456   Arjan van de Ven   async: Asynchrono...
154
155
156
157
158
  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...
159
160
161
162
163
164
165
166
  
  	/* 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...
167
  	if (!entry || atomic_read(&entry_count) > MAX_WORK) {
22a9d6456   Arjan van de Ven   async: Asynchrono...
168
169
170
171
172
173
174
175
176
  		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...
177
  	INIT_WORK(&entry->work, async_run_entry_fn);
22a9d6456   Arjan van de Ven   async: Asynchrono...
178
179
180
181
182
183
184
185
186
  	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...
187
188
189
  
  	/* schedule for execution */
  	queue_work(system_unbound_wq, &entry->work);
22a9d6456   Arjan van de Ven   async: Asynchrono...
190
191
  	return newcookie;
  }
f30d5b307   Cornelia Huck   async: Add some d...
192
193
194
195
196
197
198
199
  /**
   * 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...
200
201
  async_cookie_t async_schedule(async_func_ptr *ptr, void *data)
  {
7a89bbc74   Cornelia Huck   async: Fix runnin...
202
  	return __async_schedule(ptr, data, &async_running);
22a9d6456   Arjan van de Ven   async: Asynchrono...
203
204
  }
  EXPORT_SYMBOL_GPL(async_schedule);
f30d5b307   Cornelia Huck   async: Add some d...
205
  /**
766ccb9ed   Cornelia Huck   async: Rename _sp...
206
   * async_schedule_domain - schedule a function for asynchronous execution within a certain domain
f30d5b307   Cornelia Huck   async: Add some d...
207
208
   * @ptr: function to execute asynchronously
   * @data: data pointer to pass to the function
766ccb9ed   Cornelia Huck   async: Rename _sp...
209
   * @running: running list for the domain
f30d5b307   Cornelia Huck   async: Add some d...
210
211
   *
   * Returns an async_cookie_t that may be used for checkpointing later.
766ccb9ed   Cornelia Huck   async: Rename _sp...
212
213
214
   * @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...
215
216
   * Note: This function may be called from atomic or non-atomic contexts.
   */
766ccb9ed   Cornelia Huck   async: Rename _sp...
217
218
  async_cookie_t async_schedule_domain(async_func_ptr *ptr, void *data,
  				     struct list_head *running)
22a9d6456   Arjan van de Ven   async: Asynchrono...
219
220
221
  {
  	return __async_schedule(ptr, data, running);
  }
766ccb9ed   Cornelia Huck   async: Rename _sp...
222
  EXPORT_SYMBOL_GPL(async_schedule_domain);
22a9d6456   Arjan van de Ven   async: Asynchrono...
223

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

f30d5b307   Cornelia Huck   async: Add some d...
249
  /**
766ccb9ed   Cornelia Huck   async: Rename _sp...
250
   * async_synchronize_cookie_domain - synchronize asynchronous function calls within a certain domain with cookie checkpointing
f30d5b307   Cornelia Huck   async: Add some d...
251
252
253
   * @cookie: async_cookie_t to use as checkpoint
   * @running: running list to synchronize on
   *
766ccb9ed   Cornelia Huck   async: Rename _sp...
254
255
256
   * 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...
257
   */
766ccb9ed   Cornelia Huck   async: Rename _sp...
258
259
  void async_synchronize_cookie_domain(async_cookie_t cookie,
  				     struct list_head *running)
22a9d6456   Arjan van de Ven   async: Asynchrono...
260
  {
124ff4e53   Vitaliy Ivanov   async: uninitiali...
261
  	ktime_t uninitialized_var(starttime), delta, endtime;
22a9d6456   Arjan van de Ven   async: Asynchrono...
262

ad160d231   Arjan van de Ven   async: don't do t...
263
  	if (initcall_debug && system_state == SYSTEM_BOOTING) {
84c15027a   Paul McQuade   async: Fixed an i...
264
265
  		printk(KERN_DEBUG "async_waiting @ %i
  ", task_pid_nr(current));
22a9d6456   Arjan van de Ven   async: Asynchrono...
266
267
  		starttime = ktime_get();
  	}
37a76bd4f   Arjan van de Ven   async: fix __lowe...
268
  	wait_event(async_done, lowest_in_progress(running) >= cookie);
22a9d6456   Arjan van de Ven   async: Asynchrono...
269

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

f30d5b307   Cornelia Huck   async: Add some d...
281
282
283
284
285
286
287
  /**
   * 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...
288
289
  void async_synchronize_cookie(async_cookie_t cookie)
  {
766ccb9ed   Cornelia Huck   async: Rename _sp...
290
  	async_synchronize_cookie_domain(cookie, &async_running);
22a9d6456   Arjan van de Ven   async: Asynchrono...
291
292
  }
  EXPORT_SYMBOL_GPL(async_synchronize_cookie);