 4788501606
			
		
	
	
	4788501606
	
	
	
		
			
			irq work currently only supports local callbacks. However its code is mostly ready to run remote callbacks and we have some potential user. The full nohz subsystem currently open codes its own remote irq work on top of the scheduler ipi when it wants a CPU to reevaluate its next tick. However this ad hoc solution bloats the scheduler IPI. Lets just extend the irq work subsystem to support remote queuing on top of the generic SMP IPI to handle this kind of user. This shouldn't add noticeable overhead. Suggested-by: Peter Zijlstra <peterz@infradead.org> Acked-by: Peter Zijlstra <peterz@infradead.org> Cc: Andrew Morton <akpm@linux-foundation.org> Cc: Eric Dumazet <eric.dumazet@gmail.com> Cc: Ingo Molnar <mingo@kernel.org> Cc: Kevin Hilman <khilman@linaro.org> Cc: Paul E. McKenney <paulmck@linux.vnet.ibm.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Viresh Kumar <viresh.kumar@linaro.org> Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com>
		
			
				
	
	
		
			50 lines
		
	
	
	
		
			1.2 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			50 lines
		
	
	
	
		
			1.2 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef _LINUX_IRQ_WORK_H
 | |
| #define _LINUX_IRQ_WORK_H
 | |
| 
 | |
| #include <linux/llist.h>
 | |
| 
 | |
| /*
 | |
|  * 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
 | |
|  */
 | |
| 
 | |
| #define IRQ_WORK_PENDING	1UL
 | |
| #define IRQ_WORK_BUSY		2UL
 | |
| #define IRQ_WORK_FLAGS		3UL
 | |
| #define IRQ_WORK_LAZY		4UL /* Doesn't want IPI, wait for tick */
 | |
| 
 | |
| struct irq_work {
 | |
| 	unsigned long flags;
 | |
| 	struct llist_node llnode;
 | |
| 	void (*func)(struct irq_work *);
 | |
| };
 | |
| 
 | |
| static inline
 | |
| void init_irq_work(struct irq_work *work, void (*func)(struct irq_work *))
 | |
| {
 | |
| 	work->flags = 0;
 | |
| 	work->func = func;
 | |
| }
 | |
| 
 | |
| #define DEFINE_IRQ_WORK(name, _f) struct irq_work name = { .func = (_f), }
 | |
| 
 | |
| bool irq_work_queue(struct irq_work *work);
 | |
| 
 | |
| #ifdef CONFIG_SMP
 | |
| bool irq_work_queue_on(struct irq_work *work, int cpu);
 | |
| #endif
 | |
| 
 | |
| void irq_work_run(void);
 | |
| void irq_work_sync(struct irq_work *work);
 | |
| 
 | |
| #ifdef CONFIG_IRQ_WORK
 | |
| bool irq_work_needs_cpu(void);
 | |
| #else
 | |
| static inline bool irq_work_needs_cpu(void) { return false; }
 | |
| #endif
 | |
| 
 | |
| #endif /* _LINUX_IRQ_WORK_H */
 |