Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 350185
b: refs/heads/master
c: 1c4248c
h: refs/heads/master
i:
  350183: 228da32
v: v3
  • Loading branch information
Joerg Roedel committed Jan 28, 2013
1 parent 5bcc3a1 commit 1566974
Show file tree
Hide file tree
Showing 6 changed files with 62 additions and 32 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 336224ba5e4fb42a95d02ab0aa0fdff21649bb38
refs/heads/master: 1c4248ca4e783e47cc34e313d9f82b4ea52774cc
2 changes: 2 additions & 0 deletions trunk/arch/x86/include/asm/io_apic.h
Original file line number Diff line number Diff line change
Expand Up @@ -179,6 +179,7 @@ extern void __init native_io_apic_init_mappings(void);
extern unsigned int native_io_apic_read(unsigned int apic, unsigned int reg);
extern void native_io_apic_write(unsigned int apic, unsigned int reg, unsigned int val);
extern void native_io_apic_modify(unsigned int apic, unsigned int reg, unsigned int val);
extern void native_disable_io_apic(void);

static inline unsigned int io_apic_read(unsigned int apic, unsigned int reg)
{
Expand Down Expand Up @@ -223,6 +224,7 @@ static inline void disable_ioapic_support(void) { }
#define native_io_apic_read NULL
#define native_io_apic_write NULL
#define native_io_apic_modify NULL
#define native_disable_io_apic NULL
#endif

#endif /* _ASM_X86_IO_APIC_H */
9 changes: 5 additions & 4 deletions trunk/arch/x86/include/asm/x86_init.h
Original file line number Diff line number Diff line change
Expand Up @@ -190,10 +190,11 @@ struct x86_msi_ops {
};

struct x86_io_apic_ops {
void (*init) (void);
unsigned int (*read) (unsigned int apic, unsigned int reg);
void (*write) (unsigned int apic, unsigned int reg, unsigned int value);
void (*modify)(unsigned int apic, unsigned int reg, unsigned int value);
void (*init) (void);
unsigned int (*read) (unsigned int apic, unsigned int reg);
void (*write) (unsigned int apic, unsigned int reg, unsigned int value);
void (*modify) (unsigned int apic, unsigned int reg, unsigned int value);
void (*disable)(void);
};

extern struct x86_init_ops x86_init;
Expand Down
41 changes: 19 additions & 22 deletions trunk/arch/x86/kernel/apic/io_apic.c
Original file line number Diff line number Diff line change
Expand Up @@ -1921,30 +1921,14 @@ void __init enable_IO_APIC(void)
clear_IO_APIC();
}

/*
* Not an __init, needed by the reboot code
*/
void disable_IO_APIC(void)
void native_disable_io_apic(void)
{
/*
* Clear the IO-APIC before rebooting:
*/
clear_IO_APIC();

if (!legacy_pic->nr_legacy_irqs)
return;

/*
* If the i8259 is routed through an IOAPIC
* Put that IOAPIC in virtual wire mode
* so legacy interrupts can be delivered.
*
* With interrupt-remapping, for now we will use virtual wire A mode,
* as virtual wire B is little complex (need to configure both
* IOAPIC RTE as well as interrupt-remapping table entry).
* As this gets called during crash dump, keep this simple for now.
*/
if (ioapic_i8259.pin != -1 && !irq_remapping_enabled) {
if (ioapic_i8259.pin != -1) {
struct IO_APIC_route_entry entry;

memset(&entry, 0, sizeof(entry));
Expand All @@ -1964,12 +1948,25 @@ void disable_IO_APIC(void)
ioapic_write_entry(ioapic_i8259.apic, ioapic_i8259.pin, entry);
}

if (cpu_has_apic || apic_from_smp_config())
disconnect_bsp_APIC(ioapic_i8259.pin != -1);

}

/*
* Not an __init, needed by the reboot code
*/
void disable_IO_APIC(void)
{
/*
* Use virtual wire A mode when interrupt remapping is enabled.
* Clear the IO-APIC before rebooting:
*/
if (cpu_has_apic || apic_from_smp_config())
disconnect_bsp_APIC(!irq_remapping_enabled &&
ioapic_i8259.pin != -1);
clear_IO_APIC();

if (!legacy_pic->nr_legacy_irqs)
return;

x86_io_apic_ops.disable();
}

#ifdef CONFIG_X86_32
Expand Down
9 changes: 5 additions & 4 deletions trunk/arch/x86/kernel/x86_init.c
Original file line number Diff line number Diff line change
Expand Up @@ -118,8 +118,9 @@ struct x86_msi_ops x86_msi = {
};

struct x86_io_apic_ops x86_io_apic_ops = {
.init = native_io_apic_init_mappings,
.read = native_io_apic_read,
.write = native_io_apic_write,
.modify = native_io_apic_modify,
.init = native_io_apic_init_mappings,
.read = native_io_apic_read,
.write = native_io_apic_write,
.modify = native_io_apic_modify,
.disable = native_disable_io_apic,
};
31 changes: 30 additions & 1 deletion trunk/drivers/iommu/irq_remapping.c
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
#include <linux/cpumask.h>
#include <linux/kernel.h>
#include <linux/string.h>
#include <linux/cpumask.h>
Expand All @@ -6,6 +7,9 @@

#include <asm/hw_irq.h>
#include <asm/irq_remapping.h>
#include <asm/processor.h>
#include <asm/x86_init.h>
#include <asm/apic.h>

#include "irq_remapping.h"

Expand All @@ -17,6 +21,24 @@ int no_x2apic_optout;

static struct irq_remap_ops *remap_ops;

static void irq_remapping_disable_io_apic(void)
{
/*
* With interrupt-remapping, for now we will use virtual wire A
* mode, as virtual wire B is little complex (need to configure
* both IOAPIC RTE as well as interrupt-remapping table entry).
* As this gets called during crash dump, keep this simple for
* now.
*/
if (cpu_has_apic || apic_from_smp_config())
disconnect_bsp_APIC(0);
}

static void __init irq_remapping_modify_x86_ops(void)
{
x86_io_apic_ops.disable = irq_remapping_disable_io_apic;
}

static __init int setup_nointremap(char *str)
{
disable_irq_remap = 1;
Expand Down Expand Up @@ -79,10 +101,17 @@ int __init irq_remapping_prepare(void)

int __init irq_remapping_enable(void)
{
int ret;

if (!remap_ops || !remap_ops->enable)
return -ENODEV;

return remap_ops->enable();
ret = remap_ops->enable();

if (irq_remapping_enabled)
irq_remapping_modify_x86_ops();

return ret;
}

void irq_remapping_disable(void)
Expand Down

0 comments on commit 1566974

Please sign in to comment.