Blame view

kernel/irq_work.c 3.08 KB
e360adbe2   Peter Zijlstra   irq_work: Add gen...
1
2
3
4
5
6
7
8
  /*
   * Copyright (C) 2010 Red Hat, Inc., Peter Zijlstra <pzijlstr@redhat.com>
   *
   * Provides a framework for enqueueing and running callbacks from hardirq
   * context. The enqueueing is NMI-safe.
   */
  
  #include <linux/kernel.h>
9984de1a5   Paul Gortmaker   kernel: Map most ...
9
  #include <linux/export.h>
e360adbe2   Peter Zijlstra   irq_work: Add gen...
10
  #include <linux/irq_work.h>
967d1f906   Paul Gortmaker   kernel: fix two i...
11
  #include <linux/percpu.h>
e360adbe2   Peter Zijlstra   irq_work: Add gen...
12
  #include <linux/hardirq.h>
967d1f906   Paul Gortmaker   kernel: fix two i...
13
  #include <asm/processor.h>
e360adbe2   Peter Zijlstra   irq_work: Add gen...
14
15
16
17
18
19
20
21
  
  /*
   * An entry can be in one of four states:
   *
   * free	     NULL, 0 -> {claimed}       : free to be used
   * claimed   NULL, 3 -> {pending}       : claimed to be enqueued
   * pending   next, 3 -> {busy}          : queued, pending callback
   * busy      NULL, 2 -> {free, claimed} : callback in progress, can be claimed
e360adbe2   Peter Zijlstra   irq_work: Add gen...
22
23
24
25
26
   */
  
  #define IRQ_WORK_PENDING	1UL
  #define IRQ_WORK_BUSY		2UL
  #define IRQ_WORK_FLAGS		3UL
38aaf8090   Huang Ying   irq_work: Use lli...
27
  static DEFINE_PER_CPU(struct llist_head, irq_work_list);
e360adbe2   Peter Zijlstra   irq_work: Add gen...
28
29
30
31
  
  /*
   * Claim the entry so that no one else will poke at it.
   */
38aaf8090   Huang Ying   irq_work: Use lli...
32
  static bool irq_work_claim(struct irq_work *work)
e360adbe2   Peter Zijlstra   irq_work: Add gen...
33
  {
38aaf8090   Huang Ying   irq_work: Use lli...
34
  	unsigned long flags, nflags;
e360adbe2   Peter Zijlstra   irq_work: Add gen...
35

38aaf8090   Huang Ying   irq_work: Use lli...
36
37
38
  	for (;;) {
  		flags = work->flags;
  		if (flags & IRQ_WORK_PENDING)
e360adbe2   Peter Zijlstra   irq_work: Add gen...
39
  			return false;
38aaf8090   Huang Ying   irq_work: Use lli...
40
41
42
43
44
  		nflags = flags | IRQ_WORK_FLAGS;
  		if (cmpxchg(&work->flags, flags, nflags) == flags)
  			break;
  		cpu_relax();
  	}
e360adbe2   Peter Zijlstra   irq_work: Add gen...
45
46
47
  
  	return true;
  }
e360adbe2   Peter Zijlstra   irq_work: Add gen...
48
49
50
51
52
53
54
55
56
57
  void __weak arch_irq_work_raise(void)
  {
  	/*
  	 * Lame architectures will get the timer tick callback
  	 */
  }
  
  /*
   * Queue the entry and raise the IPI if needed.
   */
38aaf8090   Huang Ying   irq_work: Use lli...
58
  static void __irq_work_queue(struct irq_work *work)
e360adbe2   Peter Zijlstra   irq_work: Add gen...
59
  {
38aaf8090   Huang Ying   irq_work: Use lli...
60
  	bool empty;
e360adbe2   Peter Zijlstra   irq_work: Add gen...
61

20b876918   Christoph Lameter   irq_work: Use per...
62
  	preempt_disable();
e360adbe2   Peter Zijlstra   irq_work: Add gen...
63

38aaf8090   Huang Ying   irq_work: Use lli...
64
  	empty = llist_add(&work->llnode, &__get_cpu_var(irq_work_list));
e360adbe2   Peter Zijlstra   irq_work: Add gen...
65
  	/* The list was empty, raise self-interrupt to start processing. */
38aaf8090   Huang Ying   irq_work: Use lli...
66
  	if (empty)
e360adbe2   Peter Zijlstra   irq_work: Add gen...
67
  		arch_irq_work_raise();
20b876918   Christoph Lameter   irq_work: Use per...
68
  	preempt_enable();
e360adbe2   Peter Zijlstra   irq_work: Add gen...
69
70
71
72
73
74
75
76
  }
  
  /*
   * Enqueue the irq_work @entry, returns true on success, failure when the
   * @entry was already enqueued by someone else.
   *
   * Can be re-enqueued while the callback is still in progress.
   */
38aaf8090   Huang Ying   irq_work: Use lli...
77
  bool irq_work_queue(struct irq_work *work)
e360adbe2   Peter Zijlstra   irq_work: Add gen...
78
  {
38aaf8090   Huang Ying   irq_work: Use lli...
79
  	if (!irq_work_claim(work)) {
e360adbe2   Peter Zijlstra   irq_work: Add gen...
80
81
82
83
84
  		/*
  		 * Already enqueued, can't do!
  		 */
  		return false;
  	}
38aaf8090   Huang Ying   irq_work: Use lli...
85
  	__irq_work_queue(work);
e360adbe2   Peter Zijlstra   irq_work: Add gen...
86
87
88
89
90
91
92
93
94
95
  	return true;
  }
  EXPORT_SYMBOL_GPL(irq_work_queue);
  
  /*
   * Run the irq_work entries on this cpu. Requires to be ran from hardirq
   * context with local IRQs disabled.
   */
  void irq_work_run(void)
  {
38aaf8090   Huang Ying   irq_work: Use lli...
96
97
98
  	struct irq_work *work;
  	struct llist_head *this_list;
  	struct llist_node *llnode;
e360adbe2   Peter Zijlstra   irq_work: Add gen...
99

38aaf8090   Huang Ying   irq_work: Use lli...
100
101
  	this_list = &__get_cpu_var(irq_work_list);
  	if (llist_empty(this_list))
e360adbe2   Peter Zijlstra   irq_work: Add gen...
102
103
104
105
  		return;
  
  	BUG_ON(!in_irq());
  	BUG_ON(!irqs_disabled());
38aaf8090   Huang Ying   irq_work: Use lli...
106
107
108
  	llnode = llist_del_all(this_list);
  	while (llnode != NULL) {
  		work = llist_entry(llnode, struct irq_work, llnode);
e360adbe2   Peter Zijlstra   irq_work: Add gen...
109

924f8f5af   Peter Zijlstra   llist: Add llist_...
110
  		llnode = llist_next(llnode);
e360adbe2   Peter Zijlstra   irq_work: Add gen...
111
112
  
  		/*
38aaf8090   Huang Ying   irq_work: Use lli...
113
  		 * Clear the PENDING bit, after this point the @work
e360adbe2   Peter Zijlstra   irq_work: Add gen...
114
115
  		 * can be re-used.
  		 */
38aaf8090   Huang Ying   irq_work: Use lli...
116
117
  		work->flags = IRQ_WORK_BUSY;
  		work->func(work);
e360adbe2   Peter Zijlstra   irq_work: Add gen...
118
119
120
121
  		/*
  		 * Clear the BUSY bit and return to the free state if
  		 * no-one else claimed it meanwhile.
  		 */
38aaf8090   Huang Ying   irq_work: Use lli...
122
  		(void)cmpxchg(&work->flags, IRQ_WORK_BUSY, 0);
e360adbe2   Peter Zijlstra   irq_work: Add gen...
123
124
125
126
127
128
129
130
  	}
  }
  EXPORT_SYMBOL_GPL(irq_work_run);
  
  /*
   * Synchronize against the irq_work @entry, ensures the entry is not
   * currently in use.
   */
38aaf8090   Huang Ying   irq_work: Use lli...
131
  void irq_work_sync(struct irq_work *work)
e360adbe2   Peter Zijlstra   irq_work: Add gen...
132
133
  {
  	WARN_ON_ONCE(irqs_disabled());
38aaf8090   Huang Ying   irq_work: Use lli...
134
  	while (work->flags & IRQ_WORK_BUSY)
e360adbe2   Peter Zijlstra   irq_work: Add gen...
135
136
137
  		cpu_relax();
  }
  EXPORT_SYMBOL_GPL(irq_work_sync);