forked from Minki/linux
3304c9c37b
Interrupt entry points are handled with the following code, each 32-byte code block contains seven entry points: ... [push][jump 22] // 4 bytes [push][jump 18] // 4 bytes [push][jump 14] // 4 bytes [push][jump 10] // 4 bytes [push][jump 6] // 4 bytes [push][jump 2] // 4 bytes [push][jump common_interrupt][padding] // 8 bytes [push][jump] [push][jump] [push][jump] [push][jump] [push][jump] [push][jump] [push][jump common_interrupt][padding] [padding_2] common_interrupt: And there is a table which holds pointers to every entry point, IOW: to every push. In cold cache, two jumps are still costlier than one, even though we get the benefit of them residing in the same cacheline. This change replaces short jumps with near ones to 'common_interrupt', and pads every push+jump pair to 8 bytes. This way, each interrupt takes only one jump. This change replaces ".p2align CONFIG_X86_L1_CACHE_SHIFT" before dispatch table with ".align 8" - we do not need anything stronger than that. The table of entry addresses (the interrupt[] array) is no longer necessary, the address of entries can be easily calculated as (irq_entries_start + i*8). text data bss dec hex filename 12546 0 0 12546 3102 entry_64.o.before 11626 0 0 11626 2d6a entry_64.o The size decrease is because 1656 bytes of .init.rodata are gone. That's initdata, though. The resident size does go up a bit. Run-tested (32 and 64 bits). Acked-and-Tested-by: Borislav Petkov <bp@suse.de> Signed-off-by: Denys Vlasenko <dvlasenk@redhat.com> Cc: Alexei Starovoitov <ast@plumgrid.com> Cc: Andy Lutomirski <luto@amacapital.net> Cc: Borislav Petkov <bp@alien8.de> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: H. Peter Anvin <hpa@zytor.com> Cc: Kees Cook <keescook@chromium.org> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Oleg Nesterov <oleg@redhat.com> Cc: Steven Rostedt <rostedt@goodmis.org> Cc: Will Drewry <wad@chromium.org> Link: http://lkml.kernel.org/r/1428090553-7283-1-git-send-email-dvlasenk@redhat.com Signed-off-by: Ingo Molnar <mingo@kernel.org>
198 lines
6.5 KiB
C
198 lines
6.5 KiB
C
#ifndef _ASM_X86_HW_IRQ_H
|
|
#define _ASM_X86_HW_IRQ_H
|
|
|
|
/*
|
|
* (C) 1992, 1993 Linus Torvalds, (C) 1997 Ingo Molnar
|
|
*
|
|
* moved some of the old arch/i386/kernel/irq.h to here. VY
|
|
*
|
|
* IRQ/IPI changes taken from work by Thomas Radke
|
|
* <tomsoft@informatik.tu-chemnitz.de>
|
|
*
|
|
* hacked by Andi Kleen for x86-64.
|
|
* unified by tglx
|
|
*/
|
|
|
|
#include <asm/irq_vectors.h>
|
|
|
|
#ifndef __ASSEMBLY__
|
|
|
|
#include <linux/percpu.h>
|
|
#include <linux/profile.h>
|
|
#include <linux/smp.h>
|
|
|
|
#include <linux/atomic.h>
|
|
#include <asm/irq.h>
|
|
#include <asm/sections.h>
|
|
|
|
/* Interrupt handlers registered during init_IRQ */
|
|
extern asmlinkage void apic_timer_interrupt(void);
|
|
extern asmlinkage void x86_platform_ipi(void);
|
|
extern asmlinkage void kvm_posted_intr_ipi(void);
|
|
extern asmlinkage void error_interrupt(void);
|
|
extern asmlinkage void irq_work_interrupt(void);
|
|
|
|
extern asmlinkage void spurious_interrupt(void);
|
|
extern asmlinkage void thermal_interrupt(void);
|
|
extern asmlinkage void reschedule_interrupt(void);
|
|
|
|
extern asmlinkage void invalidate_interrupt(void);
|
|
extern asmlinkage void invalidate_interrupt0(void);
|
|
extern asmlinkage void invalidate_interrupt1(void);
|
|
extern asmlinkage void invalidate_interrupt2(void);
|
|
extern asmlinkage void invalidate_interrupt3(void);
|
|
extern asmlinkage void invalidate_interrupt4(void);
|
|
extern asmlinkage void invalidate_interrupt5(void);
|
|
extern asmlinkage void invalidate_interrupt6(void);
|
|
extern asmlinkage void invalidate_interrupt7(void);
|
|
extern asmlinkage void invalidate_interrupt8(void);
|
|
extern asmlinkage void invalidate_interrupt9(void);
|
|
extern asmlinkage void invalidate_interrupt10(void);
|
|
extern asmlinkage void invalidate_interrupt11(void);
|
|
extern asmlinkage void invalidate_interrupt12(void);
|
|
extern asmlinkage void invalidate_interrupt13(void);
|
|
extern asmlinkage void invalidate_interrupt14(void);
|
|
extern asmlinkage void invalidate_interrupt15(void);
|
|
extern asmlinkage void invalidate_interrupt16(void);
|
|
extern asmlinkage void invalidate_interrupt17(void);
|
|
extern asmlinkage void invalidate_interrupt18(void);
|
|
extern asmlinkage void invalidate_interrupt19(void);
|
|
extern asmlinkage void invalidate_interrupt20(void);
|
|
extern asmlinkage void invalidate_interrupt21(void);
|
|
extern asmlinkage void invalidate_interrupt22(void);
|
|
extern asmlinkage void invalidate_interrupt23(void);
|
|
extern asmlinkage void invalidate_interrupt24(void);
|
|
extern asmlinkage void invalidate_interrupt25(void);
|
|
extern asmlinkage void invalidate_interrupt26(void);
|
|
extern asmlinkage void invalidate_interrupt27(void);
|
|
extern asmlinkage void invalidate_interrupt28(void);
|
|
extern asmlinkage void invalidate_interrupt29(void);
|
|
extern asmlinkage void invalidate_interrupt30(void);
|
|
extern asmlinkage void invalidate_interrupt31(void);
|
|
|
|
extern asmlinkage void irq_move_cleanup_interrupt(void);
|
|
extern asmlinkage void reboot_interrupt(void);
|
|
extern asmlinkage void threshold_interrupt(void);
|
|
|
|
extern asmlinkage void call_function_interrupt(void);
|
|
extern asmlinkage void call_function_single_interrupt(void);
|
|
|
|
#ifdef CONFIG_TRACING
|
|
/* Interrupt handlers registered during init_IRQ */
|
|
extern void trace_apic_timer_interrupt(void);
|
|
extern void trace_x86_platform_ipi(void);
|
|
extern void trace_error_interrupt(void);
|
|
extern void trace_irq_work_interrupt(void);
|
|
extern void trace_spurious_interrupt(void);
|
|
extern void trace_thermal_interrupt(void);
|
|
extern void trace_reschedule_interrupt(void);
|
|
extern void trace_threshold_interrupt(void);
|
|
extern void trace_call_function_interrupt(void);
|
|
extern void trace_call_function_single_interrupt(void);
|
|
#define trace_irq_move_cleanup_interrupt irq_move_cleanup_interrupt
|
|
#define trace_reboot_interrupt reboot_interrupt
|
|
#define trace_kvm_posted_intr_ipi kvm_posted_intr_ipi
|
|
#endif /* CONFIG_TRACING */
|
|
|
|
#ifdef CONFIG_IRQ_REMAP
|
|
/* Intel specific interrupt remapping information */
|
|
struct irq_2_iommu {
|
|
struct intel_iommu *iommu;
|
|
u16 irte_index;
|
|
u16 sub_handle;
|
|
u8 irte_mask;
|
|
};
|
|
|
|
/* AMD specific interrupt remapping information */
|
|
struct irq_2_irte {
|
|
u16 devid; /* Device ID for IRTE table */
|
|
u16 index; /* Index into IRTE table*/
|
|
};
|
|
#endif /* CONFIG_IRQ_REMAP */
|
|
|
|
#ifdef CONFIG_X86_LOCAL_APIC
|
|
struct irq_data;
|
|
|
|
struct irq_cfg {
|
|
cpumask_var_t domain;
|
|
cpumask_var_t old_domain;
|
|
u8 vector;
|
|
u8 move_in_progress : 1;
|
|
#ifdef CONFIG_IRQ_REMAP
|
|
u8 remapped : 1;
|
|
union {
|
|
struct irq_2_iommu irq_2_iommu;
|
|
struct irq_2_irte irq_2_irte;
|
|
};
|
|
#endif
|
|
union {
|
|
#ifdef CONFIG_X86_IO_APIC
|
|
struct {
|
|
struct list_head irq_2_pin;
|
|
};
|
|
#endif
|
|
};
|
|
};
|
|
|
|
extern struct irq_cfg *irq_cfg(unsigned int irq);
|
|
extern struct irq_cfg *irqd_cfg(struct irq_data *irq_data);
|
|
extern struct irq_cfg *alloc_irq_and_cfg_at(unsigned int at, int node);
|
|
extern void lock_vector_lock(void);
|
|
extern void unlock_vector_lock(void);
|
|
extern int assign_irq_vector(int, struct irq_cfg *, const struct cpumask *);
|
|
extern void clear_irq_vector(int irq, struct irq_cfg *cfg);
|
|
extern void setup_vector_irq(int cpu);
|
|
#ifdef CONFIG_SMP
|
|
extern void send_cleanup_vector(struct irq_cfg *);
|
|
extern void irq_complete_move(struct irq_cfg *cfg);
|
|
#else
|
|
static inline void send_cleanup_vector(struct irq_cfg *c) { }
|
|
static inline void irq_complete_move(struct irq_cfg *c) { }
|
|
#endif
|
|
|
|
extern int apic_retrigger_irq(struct irq_data *data);
|
|
extern void apic_ack_edge(struct irq_data *data);
|
|
extern int apic_set_affinity(struct irq_data *data, const struct cpumask *mask,
|
|
unsigned int *dest_id);
|
|
#else /* CONFIG_X86_LOCAL_APIC */
|
|
static inline void lock_vector_lock(void) {}
|
|
static inline void unlock_vector_lock(void) {}
|
|
#endif /* CONFIG_X86_LOCAL_APIC */
|
|
|
|
/* Statistics */
|
|
extern atomic_t irq_err_count;
|
|
extern atomic_t irq_mis_count;
|
|
|
|
/* EISA */
|
|
extern void eisa_set_level_irq(unsigned int irq);
|
|
|
|
/* SMP */
|
|
extern __visible void smp_apic_timer_interrupt(struct pt_regs *);
|
|
extern __visible void smp_spurious_interrupt(struct pt_regs *);
|
|
extern __visible void smp_x86_platform_ipi(struct pt_regs *);
|
|
extern __visible void smp_error_interrupt(struct pt_regs *);
|
|
#ifdef CONFIG_X86_IO_APIC
|
|
extern asmlinkage void smp_irq_move_cleanup_interrupt(void);
|
|
#endif
|
|
#ifdef CONFIG_SMP
|
|
extern __visible void smp_reschedule_interrupt(struct pt_regs *);
|
|
extern __visible void smp_call_function_interrupt(struct pt_regs *);
|
|
extern __visible void smp_call_function_single_interrupt(struct pt_regs *);
|
|
extern __visible void smp_invalidate_interrupt(struct pt_regs *);
|
|
#endif
|
|
|
|
extern char irq_entries_start[];
|
|
#ifdef CONFIG_TRACING
|
|
#define trace_irq_entries_start irq_entries_start
|
|
#endif
|
|
|
|
#define VECTOR_UNDEFINED (-1)
|
|
#define VECTOR_RETRIGGERED (-2)
|
|
|
|
typedef int vector_irq_t[NR_VECTORS];
|
|
DECLARE_PER_CPU(vector_irq_t, vector_irq);
|
|
|
|
#endif /* !ASSEMBLY_ */
|
|
|
|
#endif /* _ASM_X86_HW_IRQ_H */
|