mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-28 11:18:45 +07:00
0e4ccb1505
Certain platforms do not allow writes in the MSI-X BARs to setup or tear down vector values. To combat against the generic code trying to write to that and either silently being ignored or crashing due to the pagetables being marked R/O this patch introduces a platform override. Note that we keep two separate, non-weak, functions default_mask_msi_irqs() and default_mask_msix_irqs() for the behavior of the arch_mask_msi_irqs() and arch_mask_msix_irqs(), as the default behavior is needed by x86 PCI code. For Xen, which does not allow the guest to write to MSI-X tables - as the hypervisor is solely responsible for setting the vector values - we implement two nops. This fixes a Xen guest crash when passing a PCI device with MSI-X to the guest. See the bugzilla for more details. [bhelgaas: add bugzilla info] Reference: https://bugzilla.kernel.org/show_bug.cgi?id=64581 Signed-off-by: Konrad Rzeszutek Wilk <konrad.wilk@oracle.com> Signed-off-by: Bjorn Helgaas <bhelgaas@google.com> CC: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com> CC: Zhenzhong Duan <zhenzhong.duan@oracle.com>
84 lines
2.6 KiB
C
84 lines
2.6 KiB
C
#ifndef LINUX_MSI_H
|
|
#define LINUX_MSI_H
|
|
|
|
#include <linux/kobject.h>
|
|
#include <linux/list.h>
|
|
|
|
struct msi_msg {
|
|
u32 address_lo; /* low 32 bits of msi message address */
|
|
u32 address_hi; /* high 32 bits of msi message address */
|
|
u32 data; /* 16 bits of msi message data */
|
|
};
|
|
|
|
/* Helper functions */
|
|
struct irq_data;
|
|
struct msi_desc;
|
|
void mask_msi_irq(struct irq_data *data);
|
|
void unmask_msi_irq(struct irq_data *data);
|
|
void __read_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
|
void __get_cached_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
|
void __write_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
|
void read_msi_msg(unsigned int irq, struct msi_msg *msg);
|
|
void get_cached_msi_msg(unsigned int irq, struct msi_msg *msg);
|
|
void write_msi_msg(unsigned int irq, struct msi_msg *msg);
|
|
|
|
struct msi_desc {
|
|
struct {
|
|
__u8 is_msix : 1;
|
|
__u8 multiple: 3; /* log2 number of messages */
|
|
__u8 maskbit : 1; /* mask-pending bit supported ? */
|
|
__u8 is_64 : 1; /* Address size: 0=32bit 1=64bit */
|
|
__u8 pos; /* Location of the msi capability */
|
|
__u16 entry_nr; /* specific enabled entry */
|
|
unsigned default_irq; /* default pre-assigned irq */
|
|
} msi_attrib;
|
|
|
|
u32 masked; /* mask bits */
|
|
unsigned int irq;
|
|
unsigned int nvec_used; /* number of messages */
|
|
struct list_head list;
|
|
|
|
union {
|
|
void __iomem *mask_base;
|
|
u8 mask_pos;
|
|
};
|
|
struct pci_dev *dev;
|
|
|
|
/* Last set MSI message */
|
|
struct msi_msg msg;
|
|
|
|
struct kobject kobj;
|
|
};
|
|
|
|
/*
|
|
* The arch hooks to setup up msi irqs. Those functions are
|
|
* implemented as weak symbols so that they /can/ be overriden by
|
|
* architecture specific code if needed.
|
|
*/
|
|
int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc);
|
|
void arch_teardown_msi_irq(unsigned int irq);
|
|
int arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type);
|
|
void arch_teardown_msi_irqs(struct pci_dev *dev);
|
|
int arch_msi_check_device(struct pci_dev* dev, int nvec, int type);
|
|
void arch_restore_msi_irqs(struct pci_dev *dev, int irq);
|
|
|
|
void default_teardown_msi_irqs(struct pci_dev *dev);
|
|
void default_restore_msi_irqs(struct pci_dev *dev, int irq);
|
|
u32 default_msi_mask_irq(struct msi_desc *desc, u32 mask, u32 flag);
|
|
u32 default_msix_mask_irq(struct msi_desc *desc, u32 flag);
|
|
|
|
struct msi_chip {
|
|
struct module *owner;
|
|
struct device *dev;
|
|
struct device_node *of_node;
|
|
struct list_head list;
|
|
|
|
int (*setup_irq)(struct msi_chip *chip, struct pci_dev *dev,
|
|
struct msi_desc *desc);
|
|
void (*teardown_irq)(struct msi_chip *chip, unsigned int irq);
|
|
int (*check_device)(struct msi_chip *chip, struct pci_dev *dev,
|
|
int nvec, int type);
|
|
};
|
|
|
|
#endif /* LINUX_MSI_H */
|