1
2#include <linux/kdebug.h>
3#include <linux/kprobes.h>
4#include <linux/export.h>
5#include <linux/notifier.h>
6#include <linux/rcupdate.h>
7#include <linux/vmalloc.h>
8#include <linux/reboot.h>
9
10
11
12
13
14
15BLOCKING_NOTIFIER_HEAD(reboot_notifier_list);
16
17
18
19
20
21
22static int notifier_chain_register(struct notifier_block **nl,
23 struct notifier_block *n)
24{
25 while ((*nl) != NULL) {
26 if (unlikely((*nl) == n)) {
27 WARN(1, "double register detected");
28 return 0;
29 }
30 if (n->priority > (*nl)->priority)
31 break;
32 nl = &((*nl)->next);
33 }
34 n->next = *nl;
35 rcu_assign_pointer(*nl, n);
36 return 0;
37}
38
39static int notifier_chain_unregister(struct notifier_block **nl,
40 struct notifier_block *n)
41{
42 while ((*nl) != NULL) {
43 if ((*nl) == n) {
44 rcu_assign_pointer(*nl, n->next);
45 return 0;
46 }
47 nl = &((*nl)->next);
48 }
49 return -ENOENT;
50}
51
52
53
54
55
56
57
58
59
60
61
62
63
64static int notifier_call_chain(struct notifier_block **nl,
65 unsigned long val, void *v,
66 int nr_to_call, int *nr_calls)
67{
68 int ret = NOTIFY_DONE;
69 struct notifier_block *nb, *next_nb;
70
71 nb = rcu_dereference_raw(*nl);
72
73 while (nb && nr_to_call) {
74 next_nb = rcu_dereference_raw(nb->next);
75
76#ifdef CONFIG_DEBUG_NOTIFIERS
77 if (unlikely(!func_ptr_is_kernel_text(nb->notifier_call))) {
78 WARN(1, "Invalid notifier called!");
79 nb = next_nb;
80 continue;
81 }
82#endif
83 ret = nb->notifier_call(nb, val, v);
84
85 if (nr_calls)
86 (*nr_calls)++;
87
88 if (ret & NOTIFY_STOP_MASK)
89 break;
90 nb = next_nb;
91 nr_to_call--;
92 }
93 return ret;
94}
95NOKPROBE_SYMBOL(notifier_call_chain);
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111int atomic_notifier_chain_register(struct atomic_notifier_head *nh,
112 struct notifier_block *n)
113{
114 unsigned long flags;
115 int ret;
116
117 spin_lock_irqsave(&nh->lock, flags);
118 ret = notifier_chain_register(&nh->head, n);
119 spin_unlock_irqrestore(&nh->lock, flags);
120 return ret;
121}
122EXPORT_SYMBOL_GPL(atomic_notifier_chain_register);
123
124
125
126
127
128
129
130
131
132
133int atomic_notifier_chain_unregister(struct atomic_notifier_head *nh,
134 struct notifier_block *n)
135{
136 unsigned long flags;
137 int ret;
138
139 spin_lock_irqsave(&nh->lock, flags);
140 ret = notifier_chain_unregister(&nh->head, n);
141 spin_unlock_irqrestore(&nh->lock, flags);
142 synchronize_rcu();
143 return ret;
144}
145EXPORT_SYMBOL_GPL(atomic_notifier_chain_unregister);
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166int __atomic_notifier_call_chain(struct atomic_notifier_head *nh,
167 unsigned long val, void *v,
168 int nr_to_call, int *nr_calls)
169{
170 int ret;
171
172 rcu_read_lock();
173 ret = notifier_call_chain(&nh->head, val, v, nr_to_call, nr_calls);
174 rcu_read_unlock();
175 return ret;
176}
177EXPORT_SYMBOL_GPL(__atomic_notifier_call_chain);
178NOKPROBE_SYMBOL(__atomic_notifier_call_chain);
179
180int atomic_notifier_call_chain(struct atomic_notifier_head *nh,
181 unsigned long val, void *v)
182{
183 return __atomic_notifier_call_chain(nh, val, v, -1, NULL);
184}
185EXPORT_SYMBOL_GPL(atomic_notifier_call_chain);
186NOKPROBE_SYMBOL(atomic_notifier_call_chain);
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203int blocking_notifier_chain_register(struct blocking_notifier_head *nh,
204 struct notifier_block *n)
205{
206 int ret;
207
208
209
210
211
212
213 if (unlikely(system_state == SYSTEM_BOOTING))
214 return notifier_chain_register(&nh->head, n);
215
216 down_write(&nh->rwsem);
217 ret = notifier_chain_register(&nh->head, n);
218 up_write(&nh->rwsem);
219 return ret;
220}
221EXPORT_SYMBOL_GPL(blocking_notifier_chain_register);
222
223
224
225
226
227
228
229
230
231
232
233int blocking_notifier_chain_unregister(struct blocking_notifier_head *nh,
234 struct notifier_block *n)
235{
236 int ret;
237
238
239
240
241
242
243 if (unlikely(system_state == SYSTEM_BOOTING))
244 return notifier_chain_unregister(&nh->head, n);
245
246 down_write(&nh->rwsem);
247 ret = notifier_chain_unregister(&nh->head, n);
248 up_write(&nh->rwsem);
249 return ret;
250}
251EXPORT_SYMBOL_GPL(blocking_notifier_chain_unregister);
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271int __blocking_notifier_call_chain(struct blocking_notifier_head *nh,
272 unsigned long val, void *v,
273 int nr_to_call, int *nr_calls)
274{
275 int ret = NOTIFY_DONE;
276
277
278
279
280
281
282 if (rcu_access_pointer(nh->head)) {
283 down_read(&nh->rwsem);
284 ret = notifier_call_chain(&nh->head, val, v, nr_to_call,
285 nr_calls);
286 up_read(&nh->rwsem);
287 }
288 return ret;
289}
290EXPORT_SYMBOL_GPL(__blocking_notifier_call_chain);
291
292int blocking_notifier_call_chain(struct blocking_notifier_head *nh,
293 unsigned long val, void *v)
294{
295 return __blocking_notifier_call_chain(nh, val, v, -1, NULL);
296}
297EXPORT_SYMBOL_GPL(blocking_notifier_call_chain);
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314int raw_notifier_chain_register(struct raw_notifier_head *nh,
315 struct notifier_block *n)
316{
317 return notifier_chain_register(&nh->head, n);
318}
319EXPORT_SYMBOL_GPL(raw_notifier_chain_register);
320
321
322
323
324
325
326
327
328
329
330
331int raw_notifier_chain_unregister(struct raw_notifier_head *nh,
332 struct notifier_block *n)
333{
334 return notifier_chain_unregister(&nh->head, n);
335}
336EXPORT_SYMBOL_GPL(raw_notifier_chain_unregister);
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357int __raw_notifier_call_chain(struct raw_notifier_head *nh,
358 unsigned long val, void *v,
359 int nr_to_call, int *nr_calls)
360{
361 return notifier_call_chain(&nh->head, val, v, nr_to_call, nr_calls);
362}
363EXPORT_SYMBOL_GPL(__raw_notifier_call_chain);
364
365int raw_notifier_call_chain(struct raw_notifier_head *nh,
366 unsigned long val, void *v)
367{
368 return __raw_notifier_call_chain(nh, val, v, -1, NULL);
369}
370EXPORT_SYMBOL_GPL(raw_notifier_call_chain);
371
372#ifdef CONFIG_SRCU
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388int srcu_notifier_chain_register(struct srcu_notifier_head *nh,
389 struct notifier_block *n)
390{
391 int ret;
392
393
394
395
396
397
398 if (unlikely(system_state == SYSTEM_BOOTING))
399 return notifier_chain_register(&nh->head, n);
400
401 mutex_lock(&nh->mutex);
402 ret = notifier_chain_register(&nh->head, n);
403 mutex_unlock(&nh->mutex);
404 return ret;
405}
406EXPORT_SYMBOL_GPL(srcu_notifier_chain_register);
407
408
409
410
411
412
413
414
415
416
417
418int srcu_notifier_chain_unregister(struct srcu_notifier_head *nh,
419 struct notifier_block *n)
420{
421 int ret;
422
423
424
425
426
427
428 if (unlikely(system_state == SYSTEM_BOOTING))
429 return notifier_chain_unregister(&nh->head, n);
430
431 mutex_lock(&nh->mutex);
432 ret = notifier_chain_unregister(&nh->head, n);
433 mutex_unlock(&nh->mutex);
434 synchronize_srcu(&nh->srcu);
435 return ret;
436}
437EXPORT_SYMBOL_GPL(srcu_notifier_chain_unregister);
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457int __srcu_notifier_call_chain(struct srcu_notifier_head *nh,
458 unsigned long val, void *v,
459 int nr_to_call, int *nr_calls)
460{
461 int ret;
462 int idx;
463
464 idx = srcu_read_lock(&nh->srcu);
465 ret = notifier_call_chain(&nh->head, val, v, nr_to_call, nr_calls);
466 srcu_read_unlock(&nh->srcu, idx);
467 return ret;
468}
469EXPORT_SYMBOL_GPL(__srcu_notifier_call_chain);
470
471int srcu_notifier_call_chain(struct srcu_notifier_head *nh,
472 unsigned long val, void *v)
473{
474 return __srcu_notifier_call_chain(nh, val, v, -1, NULL);
475}
476EXPORT_SYMBOL_GPL(srcu_notifier_call_chain);
477
478
479
480
481
482
483
484
485
486
487
488
489
490void srcu_init_notifier_head(struct srcu_notifier_head *nh)
491{
492 mutex_init(&nh->mutex);
493 if (init_srcu_struct(&nh->srcu) < 0)
494 BUG();
495 nh->head = NULL;
496}
497EXPORT_SYMBOL_GPL(srcu_init_notifier_head);
498
499#endif
500
501static ATOMIC_NOTIFIER_HEAD(die_chain);
502
503int notrace notify_die(enum die_val val, const char *str,
504 struct pt_regs *regs, long err, int trap, int sig)
505{
506 struct die_args args = {
507 .regs = regs,
508 .str = str,
509 .err = err,
510 .trapnr = trap,
511 .signr = sig,
512
513 };
514 RCU_LOCKDEP_WARN(!rcu_is_watching(),
515 "notify_die called but RCU thinks we're quiescent");
516 return atomic_notifier_call_chain(&die_chain, val, &args);
517}
518NOKPROBE_SYMBOL(notify_die);
519
520int register_die_notifier(struct notifier_block *nb)
521{
522 vmalloc_sync_mappings();
523 return atomic_notifier_chain_register(&die_chain, nb);
524}
525EXPORT_SYMBOL_GPL(register_die_notifier);
526
527int unregister_die_notifier(struct notifier_block *nb)
528{
529 return atomic_notifier_chain_unregister(&die_chain, nb);
530}
531EXPORT_SYMBOL_GPL(unregister_die_notifier);
532