Merge branches 'pci/aer', 'pci/misc' and 'pci/virtualization' into next
* pci/aer: PCI/AER: Use list_first_entry_or_null() to simplify code PCI/AER: Restore pci_ops pointer while calling original pci_ops PCI/AER: Rename pci_ops_aer to aer_inj_pci_ops * pci/misc: PCI: Remove includes of asm/pci-bridge.h PCI: Remove empty asm-generic/pci-bridge.h ARM64: PCI: Remove generated include of asm-generic/pci-bridge.h PCI: Remove includes of empty asm-generic/pci-bridge.h PCI: Move pci_set_flags() from asm-generic/pci-bridge.h to linux/pci.h PCI/PME: Restructure pcie_pme_suspend() to prevent compiler warning PCI/PME: Remove redundant port lookup PCI: Check device_attach() return value always * pci/virtualization: PCI: Add ACS quirk for all Cavium devices
This commit is contained in:
@@ -124,16 +124,13 @@ static struct pci_ops *__find_pci_bus_ops(struct pci_bus *bus)
|
||||
static struct pci_bus_ops *pci_bus_ops_pop(void)
|
||||
{
|
||||
unsigned long flags;
|
||||
struct pci_bus_ops *bus_ops = NULL;
|
||||
struct pci_bus_ops *bus_ops;
|
||||
|
||||
spin_lock_irqsave(&inject_lock, flags);
|
||||
if (list_empty(&pci_bus_ops_list))
|
||||
bus_ops = NULL;
|
||||
else {
|
||||
struct list_head *lh = pci_bus_ops_list.next;
|
||||
list_del(lh);
|
||||
bus_ops = list_entry(lh, struct pci_bus_ops, list);
|
||||
}
|
||||
bus_ops = list_first_entry_or_null(&pci_bus_ops_list,
|
||||
struct pci_bus_ops, list);
|
||||
if (bus_ops)
|
||||
list_del(&bus_ops->list);
|
||||
spin_unlock_irqrestore(&inject_lock, flags);
|
||||
return bus_ops;
|
||||
}
|
||||
@@ -181,14 +178,16 @@ static u32 *find_pci_config_dword(struct aer_error *err, int where,
|
||||
return target;
|
||||
}
|
||||
|
||||
static int pci_read_aer(struct pci_bus *bus, unsigned int devfn, int where,
|
||||
int size, u32 *val)
|
||||
static int aer_inj_read_config(struct pci_bus *bus, unsigned int devfn,
|
||||
int where, int size, u32 *val)
|
||||
{
|
||||
u32 *sim;
|
||||
struct aer_error *err;
|
||||
unsigned long flags;
|
||||
struct pci_ops *ops;
|
||||
struct pci_ops *my_ops;
|
||||
int domain;
|
||||
int rv;
|
||||
|
||||
spin_lock_irqsave(&inject_lock, flags);
|
||||
if (size != sizeof(u32))
|
||||
@@ -208,19 +207,32 @@ static int pci_read_aer(struct pci_bus *bus, unsigned int devfn, int where,
|
||||
}
|
||||
out:
|
||||
ops = __find_pci_bus_ops(bus);
|
||||
/*
|
||||
* pci_lock must already be held, so we can directly
|
||||
* manipulate bus->ops. Many config access functions,
|
||||
* including pci_generic_config_read() require the original
|
||||
* bus->ops be installed to function, so temporarily put them
|
||||
* back.
|
||||
*/
|
||||
my_ops = bus->ops;
|
||||
bus->ops = ops;
|
||||
rv = ops->read(bus, devfn, where, size, val);
|
||||
bus->ops = my_ops;
|
||||
spin_unlock_irqrestore(&inject_lock, flags);
|
||||
return ops->read(bus, devfn, where, size, val);
|
||||
return rv;
|
||||
}
|
||||
|
||||
static int pci_write_aer(struct pci_bus *bus, unsigned int devfn, int where,
|
||||
int size, u32 val)
|
||||
static int aer_inj_write_config(struct pci_bus *bus, unsigned int devfn,
|
||||
int where, int size, u32 val)
|
||||
{
|
||||
u32 *sim;
|
||||
struct aer_error *err;
|
||||
unsigned long flags;
|
||||
int rw1cs;
|
||||
struct pci_ops *ops;
|
||||
struct pci_ops *my_ops;
|
||||
int domain;
|
||||
int rv;
|
||||
|
||||
spin_lock_irqsave(&inject_lock, flags);
|
||||
if (size != sizeof(u32))
|
||||
@@ -243,13 +255,24 @@ static int pci_write_aer(struct pci_bus *bus, unsigned int devfn, int where,
|
||||
}
|
||||
out:
|
||||
ops = __find_pci_bus_ops(bus);
|
||||
/*
|
||||
* pci_lock must already be held, so we can directly
|
||||
* manipulate bus->ops. Many config access functions,
|
||||
* including pci_generic_config_write() require the original
|
||||
* bus->ops be installed to function, so temporarily put them
|
||||
* back.
|
||||
*/
|
||||
my_ops = bus->ops;
|
||||
bus->ops = ops;
|
||||
rv = ops->write(bus, devfn, where, size, val);
|
||||
bus->ops = my_ops;
|
||||
spin_unlock_irqrestore(&inject_lock, flags);
|
||||
return ops->write(bus, devfn, where, size, val);
|
||||
return rv;
|
||||
}
|
||||
|
||||
static struct pci_ops pci_ops_aer = {
|
||||
.read = pci_read_aer,
|
||||
.write = pci_write_aer,
|
||||
static struct pci_ops aer_inj_pci_ops = {
|
||||
.read = aer_inj_read_config,
|
||||
.write = aer_inj_write_config,
|
||||
};
|
||||
|
||||
static void pci_bus_ops_init(struct pci_bus_ops *bus_ops,
|
||||
@@ -270,9 +293,9 @@ static int pci_bus_set_aer_ops(struct pci_bus *bus)
|
||||
bus_ops = kmalloc(sizeof(*bus_ops), GFP_KERNEL);
|
||||
if (!bus_ops)
|
||||
return -ENOMEM;
|
||||
ops = pci_bus_set_ops(bus, &pci_ops_aer);
|
||||
ops = pci_bus_set_ops(bus, &aer_inj_pci_ops);
|
||||
spin_lock_irqsave(&inject_lock, flags);
|
||||
if (ops == &pci_ops_aer)
|
||||
if (ops == &aer_inj_pci_ops)
|
||||
goto out;
|
||||
pci_bus_ops_init(bus_ops, bus, ops);
|
||||
list_add(&bus_ops->list, &pci_bus_ops_list);
|
||||
|
@@ -396,7 +396,7 @@ static int pcie_pme_suspend(struct pcie_device *srv)
|
||||
{
|
||||
struct pcie_pme_service_data *data = get_service_data(srv);
|
||||
struct pci_dev *port = srv->port;
|
||||
bool wakeup;
|
||||
bool wakeup, wake_irq_enabled = false;
|
||||
int ret;
|
||||
|
||||
if (device_may_wakeup(&port->dev)) {
|
||||
@@ -409,11 +409,12 @@ static int pcie_pme_suspend(struct pcie_device *srv)
|
||||
spin_lock_irq(&data->lock);
|
||||
if (wakeup) {
|
||||
ret = enable_irq_wake(srv->irq);
|
||||
data->suspend_level = PME_SUSPEND_WAKEUP;
|
||||
if (ret == 0) {
|
||||
data->suspend_level = PME_SUSPEND_WAKEUP;
|
||||
wake_irq_enabled = true;
|
||||
}
|
||||
}
|
||||
if (!wakeup || ret) {
|
||||
struct pci_dev *port = srv->port;
|
||||
|
||||
if (!wake_irq_enabled) {
|
||||
pcie_pme_interrupt_enable(port, false);
|
||||
pcie_clear_root_pme_status(port);
|
||||
data->suspend_level = PME_SUSPEND_NOIRQ;
|
||||
|
Reference in New Issue
Block a user