Release 4.11 drivers/pci/pcie/portdrv_core.c
/*
* File: portdrv_core.c
* Purpose: PCI Express Port Bus Driver's Core Functions
*
* Copyright (C) 2004 Intel
* Copyright (C) Tom Long Nguyen (tom.l.nguyen@intel.com)
*/
#include <linux/module.h>
#include <linux/pci.h>
#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/pm.h>
#include <linux/pm_runtime.h>
#include <linux/string.h>
#include <linux/slab.h>
#include <linux/pcieport_if.h>
#include <linux/aer.h>
#include "../pci.h"
#include "portdrv.h"
bool pciehp_msi_disabled;
static int __init pciehp_setup(char *str)
{
if (!strncmp(str, "nomsi", 5))
pciehp_msi_disabled = true;
return 1;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
MUNEDA Takahiro | 30 | 100.00% | 1 | 100.00% |
Total | 30 | 100.00% | 1 | 100.00% |
__setup("pcie_hp=", pciehp_setup);
/**
* release_pcie_device - free PCI Express port service device structure
* @dev: Port service device to release
*
* Invoked automatically when device is being removed in response to
* device_unregister(dev). Release all resources being claimed.
*/
static void release_pcie_device(struct device *dev)
{
kfree(to_pcie_device(dev));
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 19 | 100.00% | 1 | 100.00% |
Total | 19 | 100.00% | 1 | 100.00% |
/**
* pcie_port_enable_msix - try to set up MSI-X as interrupt mode for given port
* @dev: PCI Express port to handle
* @irqs: Array of interrupt vectors to populate
* @mask: Bitmask of port capabilities returned by get_port_device_capability()
*
* Return value: 0 on success, error code on failure
*/
static int pcie_port_enable_msix(struct pci_dev *dev, int *irqs, int mask)
{
int nr_entries, entry, nvec = 0;
/*
* Allocate as many entries as the port wants, so that we can check
* which of them will be useful. Moreover, if nr_entries is correctly
* equal to the number of entries this port actually uses, we'll happily
* go through without any tricks.
*/
nr_entries = pci_alloc_irq_vectors(dev, 1, PCIE_PORT_MAX_MSIX_ENTRIES,
PCI_IRQ_MSIX);
if (nr_entries < 0)
return nr_entries;
if (mask & (PCIE_PORT_SERVICE_PME | PCIE_PORT_SERVICE_HP)) {
u16 reg16;
/*
* The code below follows the PCI Express Base Specification 2.0
* stating in Section 6.1.6 that "PME and Hot-Plug Event
* interrupts (when both are implemented) always share the same
* MSI or MSI-X vector, as indicated by the Interrupt Message
* Number field in the PCI Express Capabilities register", where
* according to Section 7.8.2 of the specification "For MSI-X,
* the value in this field indicates which MSI-X Table entry is
* used to generate the interrupt message."
*/
pcie_capability_read_word(dev, PCI_EXP_FLAGS, ®16);
entry = (reg16 & PCI_EXP_FLAGS_IRQ) >> 9;
if (entry >= nr_entries)
goto out_free_irqs;
irqs[PCIE_PORT_SERVICE_PME_SHIFT] = pci_irq_vector(dev, entry);
irqs[PCIE_PORT_SERVICE_HP_SHIFT] = pci_irq_vector(dev, entry);
nvec = max(nvec, entry + 1);
}
if (mask & PCIE_PORT_SERVICE_AER) {
u32 reg32, pos;
/*
* The code below follows Section 7.10.10 of the PCI Express
* Base Specification 2.0 stating that bits 31-27 of the Root
* Error Status Register contain a value indicating which of the
* MSI/MSI-X vectors assigned to the port is going to be used
* for AER, where "For MSI-X, the value in this register
* indicates which MSI-X Table entry is used to generate the
* interrupt message."
*/
pos = pci_find_ext_capability(dev, PCI_EXT_CAP_ID_ERR);
pci_read_config_dword(dev, pos + PCI_ERR_ROOT_STATUS, ®32);
entry = reg32 >> 27;
if (entry >= nr_entries)
goto out_free_irqs;
irqs[PCIE_PORT_SERVICE_AER_SHIFT] = pci_irq_vector(dev, entry);
nvec = max(nvec, entry + 1);
}
/*
* If nvec is equal to the allocated number of entries, we can just use
* what we have. Otherwise, the port has some extra entries not for the
* services we know and we need to work around that.
*/
if (nvec != nr_entries) {
/* Drop the temporary MSI-X setup */
pci_free_irq_vectors(dev);
/* Now allocate the MSI-X vectors for real */
nr_entries = pci_alloc_irq_vectors(dev, nvec, nvec,
PCI_IRQ_MSIX);
if (nr_entries < 0)
return nr_entries;
}
return 0;
out_free_irqs:
pci_free_irq_vectors(dev);
return -EIO;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Rafael J. Wysocki | 124 | 48.63% | 2 | 28.57% |
Christoph Hellwig | 77 | 30.20% | 1 | 14.29% |
Tom L. Nguyen | 46 | 18.04% | 1 | 14.29% |
Kenji Kaneshige | 4 | 1.57% | 1 | 14.29% |
Alexander Gordeev | 3 | 1.18% | 1 | 14.29% |
Björn Helgaas | 1 | 0.39% | 1 | 14.29% |
Total | 255 | 100.00% | 7 | 100.00% |
/**
* pcie_init_service_irqs - initialize irqs for PCI Express port services
* @dev: PCI Express port to handle
* @irqs: Array of irqs to populate
* @mask: Bitmask of port capabilities returned by get_port_device_capability()
*
* Return value: Interrupt mode associated with the port
*/
static int pcie_init_service_irqs(struct pci_dev *dev, int *irqs, int mask)
{
unsigned flags = PCI_IRQ_LEGACY | PCI_IRQ_MSI;
int ret, i;
for (i = 0; i < PCIE_PORT_DEVICE_MAXSERVICES; i++)
irqs[i] = -1;
/*
* If MSI cannot be used for PCIe PME or hotplug, we have to use
* INTx or other interrupts, e.g. system shared interrupt.
*/
if (((mask & PCIE_PORT_SERVICE_PME) && pcie_pme_no_msi()) ||
((mask & PCIE_PORT_SERVICE_HP) && pciehp_no_msi())) {
flags &= ~PCI_IRQ_MSI;
} else {
/* Try to use MSI-X if supported */
if (!pcie_port_enable_msix(dev, irqs, mask))
return 0;
}
ret = pci_alloc_irq_vectors(dev, 1, 1, flags);
if (ret < 0)
return -ENODEV;
for (i = 0; i < PCIE_PORT_DEVICE_MAXSERVICES; i++) {
if (i != PCIE_PORT_SERVICE_VC_SHIFT)
irqs[i] = pci_irq_vector(dev, 0);
}
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Christoph Hellwig | 70 | 43.48% | 1 | 12.50% |
Rafael J. Wysocki | 45 | 27.95% | 2 | 25.00% |
Tom L. Nguyen | 22 | 13.66% | 1 | 12.50% |
MUNEDA Takahiro | 13 | 8.07% | 1 | 12.50% |
Kenji Kaneshige | 10 | 6.21% | 2 | 25.00% |
Shengzhou Liu | 1 | 0.62% | 1 | 12.50% |
Total | 161 | 100.00% | 8 | 100.00% |
/**
* get_port_device_capability - discover capabilities of a PCI Express port
* @dev: PCI Express port to examine
*
* The capabilities are read from the port's PCI Express configuration registers
* as described in PCI Express Base Specification 1.0a sections 7.8.2, 7.8.9 and
* 7.9 - 7.11.
*
* Return value: Bitmask of discovered port capabilities
*/
static int get_port_device_capability(struct pci_dev *dev)
{
int services = 0;
int cap_mask = 0;
if (pcie_ports_disabled)
return 0;
cap_mask = PCIE_PORT_SERVICE_PME | PCIE_PORT_SERVICE_HP
| PCIE_PORT_SERVICE_VC | PCIE_PORT_SERVICE_DPC;
if (pci_aer_available())
cap_mask |= PCIE_PORT_SERVICE_AER;
if (pcie_ports_auto)
pcie_port_platform_notify(dev, &cap_mask);
/* Hot-Plug Capable */
if ((cap_mask & PCIE_PORT_SERVICE_HP) && dev->is_hotplug_bridge) {
services |= PCIE_PORT_SERVICE_HP;
/*
* Disable hot-plug interrupts in case they have been enabled
* by the BIOS and the hot-plug service driver is not loaded.
*/
pcie_capability_clear_word(dev, PCI_EXP_SLTCTL,
PCI_EXP_SLTCTL_CCIE | PCI_EXP_SLTCTL_HPIE);
}
/* AER capable */
if ((cap_mask & PCIE_PORT_SERVICE_AER)
&& pci_find_ext_capability(dev, PCI_EXT_CAP_ID_ERR)) {
services |= PCIE_PORT_SERVICE_AER;
/*
* Disable AER on this port in case it's been enabled by the
* BIOS (the AER service driver will enable it when necessary).
*/
pci_disable_pcie_error_reporting(dev);
}
/* VC support */
if (pci_find_ext_capability(dev, PCI_EXT_CAP_ID_VC))
services |= PCIE_PORT_SERVICE_VC;
/* Root ports are capable of generating PME too */
if ((cap_mask & PCIE_PORT_SERVICE_PME)
&& pci_pcie_type(dev) == PCI_EXP_TYPE_ROOT_PORT) {
services |= PCIE_PORT_SERVICE_PME;
/*
* Disable PME interrupt on this port in case it's been enabled
* by the BIOS (the PME service driver will enable it when
* necessary).
*/
pcie_pme_interrupt_enable(dev, false);
}
if (pci_find_ext_capability(dev, PCI_EXT_CAP_ID_DPC))
services |= PCIE_PORT_SERVICE_DPC;
return services;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Rafael J. Wysocki | 76 | 42.70% | 4 | 28.57% |
Tom L. Nguyen | 42 | 23.60% | 1 | 7.14% |
Keith Busch | 15 | 8.43% | 1 | 7.14% |
Kenji Kaneshige | 11 | 6.18% | 1 | 7.14% |
Jesse Barnes | 11 | 6.18% | 1 | 7.14% |
Andrew Murray | 9 | 5.06% | 1 | 7.14% |
Taku Izumi | 4 | 2.25% | 1 | 7.14% |
Yijing Wang | 3 | 1.69% | 1 | 7.14% |
Lukas Wunner | 3 | 1.69% | 1 | 7.14% |
Chunhe Lan | 2 | 1.12% | 1 | 7.14% |
Jiang Liu | 2 | 1.12% | 1 | 7.14% |
Total | 178 | 100.00% | 14 | 100.00% |
/**
* pcie_device_init - allocate and initialize PCI Express port service device
* @pdev: PCI Express port to associate the service device with
* @service: Type of service to associate with the service device
* @irq: Interrupt vector to associate with the service device
*/
static int pcie_device_init(struct pci_dev *pdev, int service, int irq)
{
int retval;
struct pcie_device *pcie;
struct device *device;
pcie = kzalloc(sizeof(*pcie), GFP_KERNEL);
if (!pcie)
return -ENOMEM;
pcie->port = pdev;
pcie->irq = irq;
pcie->service = service;
/* Initialize generic device interface */
device = &pcie->device;
device->bus = &pcie_port_bus_type;
device->release = release_pcie_device; /* callback to free pcie dev */
dev_set_name(device, "%s:pcie%03x",
pci_name(pdev),
get_descriptor_id(pci_pcie_type(pdev), service));
device->parent = &pdev->dev;
device_enable_async_suspend(device);
retval = device_register(device);
if (retval) {
put_device(device);
return retval;
}
pm_runtime_no_callbacks(device);
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 85 | 54.14% | 2 | 15.38% |
Kenji Kaneshige | 45 | 28.66% | 1 | 7.69% |
Rafael J. Wysocki | 9 | 5.73% | 3 | 23.08% |
Sergey Vlasov | 5 | 3.18% | 1 | 7.69% |
Mika Westerberg | 5 | 3.18% | 1 | 7.69% |
Yijing Wang | 3 | 1.91% | 1 | 7.69% |
Björn Helgaas | 2 | 1.27% | 1 | 7.69% |
Kay Sievers | 1 | 0.64% | 1 | 7.69% |
Levente Kurusa | 1 | 0.64% | 1 | 7.69% |
Keith Busch | 1 | 0.64% | 1 | 7.69% |
Total | 157 | 100.00% | 13 | 100.00% |
/**
* pcie_port_device_register - register PCI Express port
* @dev: PCI Express port to register
*
* Allocate the port extension structure and register services associated with
* the port.
*/
int pcie_port_device_register(struct pci_dev *dev)
{
int status, capabilities, i, nr_service;
int irqs[PCIE_PORT_DEVICE_MAXSERVICES];
/* Enable PCI Express port device */
status = pci_enable_device(dev);
if (status)
return status;
/* Get and check PCI Express port services */
capabilities = get_port_device_capability(dev);
if (!capabilities)
return 0;
pci_set_master(dev);
/*
* Initialize service irqs. Don't use service devices that
* require interrupts if there is no way to generate them.
* However, some drivers may have a polling mode (e.g. pciehp_poll_mode)
* that can be used in the absence of irqs. Allow them to determine
* if that is to be used.
*/
status = pcie_init_service_irqs(dev, irqs, capabilities);
if (status) {
capabilities &= PCIE_PORT_SERVICE_VC | PCIE_PORT_SERVICE_HP;
if (!capabilities)
goto error_disable;
}
/* Allocate child services if any */
status = -ENODEV;
nr_service = 0;
for (i = 0; i < PCIE_PORT_DEVICE_MAXSERVICES; i++) {
int service = 1 << i;
if (!(capabilities & service))
continue;
if (!pcie_device_init(dev, service, irqs[i]))
nr_service++;
}
if (!nr_service)
goto error_cleanup_irqs;
return 0;
error_cleanup_irqs:
pci_free_irq_vectors(dev);
error_disable:
pci_disable_device(dev);
return status;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Kenji Kaneshige | 65 | 35.91% | 7 | 53.85% |
Tom L. Nguyen | 63 | 34.81% | 1 | 7.69% |
Rafael J. Wysocki | 48 | 26.52% | 3 | 23.08% |
Rajat Jain | 3 | 1.66% | 1 | 7.69% |
Christoph Hellwig | 2 | 1.10% | 1 | 7.69% |
Total | 181 | 100.00% | 13 | 100.00% |
#ifdef CONFIG_PM
static int suspend_iter(struct device *dev, void *data)
{
struct pcie_port_service_driver *service_driver;
if ((dev->bus == &pcie_port_bus_type) && dev->driver) {
service_driver = to_service_driver(dev->driver);
if (service_driver->suspend)
service_driver->suspend(to_pcie_device(dev));
}
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 65 | 100.00% | 3 | 100.00% |
Total | 65 | 100.00% | 3 | 100.00% |
/**
* pcie_port_device_suspend - suspend port services associated with a PCIe port
* @dev: PCI Express port to handle
*/
int pcie_port_device_suspend(struct device *dev)
{
return device_for_each_child(dev, NULL, suspend_iter);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 17 | 85.00% | 2 | 50.00% |
Rafael J. Wysocki | 2 | 10.00% | 1 | 25.00% |
Greg Kroah-Hartman | 1 | 5.00% | 1 | 25.00% |
Total | 20 | 100.00% | 4 | 100.00% |
static int resume_iter(struct device *dev, void *data)
{
struct pcie_port_service_driver *service_driver;
if ((dev->bus == &pcie_port_bus_type) &&
(dev->driver)) {
service_driver = to_service_driver(dev->driver);
if (service_driver->resume)
service_driver->resume(to_pcie_device(dev));
}
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 67 | 100.00% | 3 | 100.00% |
Total | 67 | 100.00% | 3 | 100.00% |
/**
* pcie_port_device_resume - resume port services associated with a PCIe port
* @dev: PCI Express port to handle
*/
int pcie_port_device_resume(struct device *dev)
{
return device_for_each_child(dev, NULL, resume_iter);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 18 | 90.00% | 2 | 50.00% |
Rafael J. Wysocki | 1 | 5.00% | 1 | 25.00% |
Greg Kroah-Hartman | 1 | 5.00% | 1 | 25.00% |
Total | 20 | 100.00% | 4 | 100.00% |
#endif /* PM */
static int remove_iter(struct device *dev, void *data)
{
if (dev->bus == &pcie_port_bus_type)
device_unregister(dev);
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 31 | 96.88% | 3 | 75.00% |
Eric W. Biedermann | 1 | 3.12% | 1 | 25.00% |
Total | 32 | 100.00% | 4 | 100.00% |
/**
* pcie_port_device_remove - unregister PCI Express port service devices
* @dev: PCI Express port the service devices to unregister are associated with
*
* Remove PCI Express port service devices associated with given port and
* disable MSI-X or MSI for the port.
*/
void pcie_port_device_remove(struct pci_dev *dev)
{
device_for_each_child(&dev->dev, NULL, remove_iter);
pci_free_irq_vectors(dev);
pci_disable_device(dev);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 29 | 90.62% | 2 | 40.00% |
Eric W. Biedermann | 1 | 3.12% | 1 | 20.00% |
Kenji Kaneshige | 1 | 3.12% | 1 | 20.00% |
Christoph Hellwig | 1 | 3.12% | 1 | 20.00% |
Total | 32 | 100.00% | 5 | 100.00% |
/**
* pcie_port_probe_service - probe driver for given PCI Express port service
* @dev: PCI Express port service device to probe against
*
* If PCI Express port service driver is registered with
* pcie_port_service_register(), this function will be called by the driver core
* whenever match is found between the driver and a port service device.
*/
static int pcie_port_probe_service(struct device *dev)
{
struct pcie_device *pciedev;
struct pcie_port_service_driver *driver;
int status;
if (!dev || !dev->driver)
return -ENODEV;
driver = to_service_driver(dev->driver);
if (!driver || !driver->probe)
return -ENODEV;
pciedev = to_pcie_device(dev);
status = driver->probe(pciedev);
if (status)
return status;
get_device(dev);
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Rafael J. Wysocki | 79 | 85.87% | 1 | 25.00% |
Tom L. Nguyen | 6 | 6.52% | 1 | 25.00% |
Björn Helgaas | 4 | 4.35% | 1 | 25.00% |
Randy Dunlap | 3 | 3.26% | 1 | 25.00% |
Total | 92 | 100.00% | 4 | 100.00% |
/**
* pcie_port_remove_service - detach driver from given PCI Express port service
* @dev: PCI Express port service device to handle
*
* If PCI Express port service driver is registered with
* pcie_port_service_register(), this function will be called by the driver core
* when device_unregister() is called for the port service device associated
* with the driver.
*/
static int pcie_port_remove_service(struct device *dev)
{
struct pcie_device *pciedev;
struct pcie_port_service_driver *driver;
if (!dev || !dev->driver)
return 0;
pciedev = to_pcie_device(dev);
driver = to_service_driver(dev->driver);
if (driver && driver->remove) {
driver->remove(pciedev);
put_device(dev);
}
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Rafael J. Wysocki | 68 | 90.67% | 1 | 50.00% |
Tom L. Nguyen | 7 | 9.33% | 1 | 50.00% |
Total | 75 | 100.00% | 2 | 100.00% |
/**
* pcie_port_shutdown_service - shut down given PCI Express port service
* @dev: PCI Express port service device to handle
*
* If PCI Express port service driver is registered with
* pcie_port_service_register(), this function will be called by the driver core
* when device_shutdown() is called for the port service device associated
* with the driver.
*/
static void pcie_port_shutdown_service(struct device *dev) {}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Rafael J. Wysocki | 10 | 100.00% | 1 | 100.00% |
Total | 10 | 100.00% | 1 | 100.00% |
/**
* pcie_port_service_register - register PCI Express port service driver
* @new: PCI Express port service driver to register
*/
int pcie_port_service_register(struct pcie_port_service_driver *new)
{
if (pcie_ports_disabled)
return -ENODEV;
new->driver.name = new->name;
new->driver.bus = &pcie_port_bus_type;
new->driver.probe = pcie_port_probe_service;
new->driver.remove = pcie_port_remove_service;
new->driver.shutdown = pcie_port_shutdown_service;
return driver_register(&new->driver);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 62 | 88.57% | 1 | 50.00% |
Rafael J. Wysocki | 8 | 11.43% | 1 | 50.00% |
Total | 70 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(pcie_port_service_register);
/**
* pcie_port_service_unregister - unregister PCI Express port service driver
* @drv: PCI Express port service driver to unregister
*/
void pcie_port_service_unregister(struct pcie_port_service_driver *drv)
{
driver_unregister(&drv->driver);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 16 | 88.89% | 1 | 50.00% |
Rafael J. Wysocki | 2 | 11.11% | 1 | 50.00% |
Total | 18 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(pcie_port_service_unregister);
Overall Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tom L. Nguyen | 627 | 40.22% | 3 | 5.66% |
Rafael J. Wysocki | 489 | 31.37% | 15 | 28.30% |
Christoph Hellwig | 152 | 9.75% | 1 | 1.89% |
Kenji Kaneshige | 137 | 8.79% | 9 | 16.98% |
MUNEDA Takahiro | 53 | 3.40% | 1 | 1.89% |
Keith Busch | 16 | 1.03% | 2 | 3.77% |
Jesse Barnes | 11 | 0.71% | 1 | 1.89% |
Andrew Murray | 9 | 0.58% | 1 | 1.89% |
Björn Helgaas | 8 | 0.51% | 3 | 5.66% |
Mika Westerberg | 8 | 0.51% | 1 | 1.89% |
Tim Schmielau | 6 | 0.38% | 1 | 1.89% |
Yijing Wang | 6 | 0.38% | 1 | 1.89% |
Hidetoshi Seto | 5 | 0.32% | 1 | 1.89% |
Sergey Vlasov | 5 | 0.32% | 1 | 1.89% |
Taku Izumi | 4 | 0.26% | 1 | 1.89% |
Rajat Jain | 3 | 0.19% | 1 | 1.89% |
Randy Dunlap | 3 | 0.19% | 1 | 1.89% |
Lukas Wunner | 3 | 0.19% | 1 | 1.89% |
Alexander Gordeev | 3 | 0.19% | 1 | 1.89% |
Greg Kroah-Hartman | 2 | 0.13% | 1 | 1.89% |
Chunhe Lan | 2 | 0.13% | 1 | 1.89% |
Jiang Liu | 2 | 0.13% | 1 | 1.89% |
Eric W. Biedermann | 2 | 0.13% | 1 | 1.89% |
Levente Kurusa | 1 | 0.06% | 1 | 1.89% |
Kay Sievers | 1 | 0.06% | 1 | 1.89% |
Shengzhou Liu | 1 | 0.06% | 1 | 1.89% |
Total | 1559 | 100.00% | 53 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.