irq_work.c 949 B

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950
  1. /*
  2. * x86 specific code for irq_work
  3. *
  4. * Copyright (C) 2010 Red Hat, Inc., Peter Zijlstra <pzijlstr@redhat.com>
  5. */
  6. #include <linux/kernel.h>
  7. #include <linux/irq_work.h>
  8. #include <linux/hardirq.h>
  9. #include <asm/apic.h>
  10. #include <asm/trace/irq_vectors.h>
  11. static inline void irq_work_entering_irq(void)
  12. {
  13. irq_enter();
  14. ack_APIC_irq();
  15. }
  16. static inline void __smp_irq_work_interrupt(void)
  17. {
  18. inc_irq_stat(apic_irq_work_irqs);
  19. irq_work_run();
  20. }
  21. void smp_irq_work_interrupt(struct pt_regs *regs)
  22. {
  23. irq_work_entering_irq();
  24. __smp_irq_work_interrupt();
  25. exiting_irq();
  26. }
  27. void smp_trace_irq_work_interrupt(struct pt_regs *regs)
  28. {
  29. irq_work_entering_irq();
  30. trace_irq_work_entry(IRQ_WORK_VECTOR);
  31. __smp_irq_work_interrupt();
  32. trace_irq_work_exit(IRQ_WORK_VECTOR);
  33. exiting_irq();
  34. }
  35. void arch_irq_work_raise(void)
  36. {
  37. #ifdef CONFIG_X86_LOCAL_APIC
  38. if (!cpu_has_apic)
  39. return;
  40. apic->send_IPI_self(IRQ_WORK_VECTOR);
  41. apic_wait_icr_idle();
  42. #endif
  43. }