PCI changes for the v3.10 merge window:
PCI device hotplug - Remove ACPI PCI subdrivers (Jiang Liu, Myron Stowe) - Make acpiphp builtin only, not modular (Jiang Liu) - Add acpiphp mutual exclusion (Jiang Liu) Power management - Skip "PME enabled/disabled" messages when not supported (Rafael Wysocki) - Fix fallback to PCI_D0 (Rafael Wysocki) Miscellaneous - Factor quirk_io_region (Yinghai Lu) - Cache MSI capability offsets & cleanup (Gavin Shan, Bjorn Helgaas) - Clean up EISA resource initialization and logging (Bjorn Helgaas) - Fix prototype warnings (Andy Shevchenko, Bjorn Helgaas) - MIPS: Initialize of_node before scanning bus (Gabor Juhos) - Fix pcibios_get_phb_of_node() declaration "weak" annotation (Gabor Juhos) - Add MSI INTX_DISABLE quirks for AR8161/AR8162/etc (Xiong Huang) - Fix aer_inject return values (Prarit Bhargava) - Remove PME/ACPI dependency (Andrew Murray) - Use shared PCI_BUS_NUM() and PCI_DEVID() (Shuah Khan) -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.11 (GNU/Linux) iQIcBAABAgAGBQJRfhSWAAoJEFmIoMA60/r8GrYQAIHDsyZIuJSf6g+8Td1h+PIC YD3wQhbyrDqQDuKU4+9cz+JsbHmnozUGA4UmlwmOGBxEa/Uauspb6yX1P1+x9Ok1 WD7Ar3BlA5OuYI/1L1mgCiA428MTujwoR4fPnC0+KFy8xk1tBpmhzzeOFohbKyFF hMBO/Xt9tCzPATJ1LhjIH4xAykfDkbnPNHNcUKRoAkRo0CO0lS8gcTk0shXXSNng p9kQ6c4cYZvlRIJTwlawWV09nr7mDsBYa3JClqXYZufUWfEwvIuhisJxCJ57sWi9 t+Ev8dm7VM6Cr5dV+ORArlboBFrq4f/W5U9j9GPFrRplwf+WbNT6tNGSpSDq8XhU Q7JjNgPWVdWXe1vIsMwaO49zi45/bNehuCSFLZiyPZwedMk764tys+iYw+tMRtv1 tBR7lwESSXfagmvWyQAuQOTy6Rj26BPd2T8e2lMsvsuQO9mCyTK6Ey3YyKuqKQK/ l5Gns4vv4eaCjGXqqDGiydUjSes+r/v1bu43XiRnwPQJUKb5kr5SjN5/zSMBuUgm TLT/bnv8qvdFxCpVQJFv4k/uzULARMdbvLtTy8osB14vNHX9jPn+xORjLaZNiO6O 7fFispMU8Om56hNkD6C451r3icRjjGlD7OA8KOlbZ8f876sLzGV9i6P9gwCoRdEB wclDPsN7kAzw/V2sEE60 =bj8i -----END PGP SIGNATURE----- Merge tag 'pci-v3.10-changes' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci Pull PCI updates from Bjorn Helgaas: "PCI changes for the v3.10 merge window: PCI device hotplug - Remove ACPI PCI subdrivers (Jiang Liu, Myron Stowe) - Make acpiphp builtin only, not modular (Jiang Liu) - Add acpiphp mutual exclusion (Jiang Liu) Power management - Skip "PME enabled/disabled" messages when not supported (Rafael Wysocki) - Fix fallback to PCI_D0 (Rafael Wysocki) Miscellaneous - Factor quirk_io_region (Yinghai Lu) - Cache MSI capability offsets & cleanup (Gavin Shan, Bjorn Helgaas) - Clean up EISA resource initialization and logging (Bjorn Helgaas) - Fix prototype warnings (Andy Shevchenko, Bjorn Helgaas) - MIPS: Initialize of_node before scanning bus (Gabor Juhos) - Fix pcibios_get_phb_of_node() declaration "weak" annotation (Gabor Juhos) - Add MSI INTX_DISABLE quirks for AR8161/AR8162/etc (Xiong Huang) - Fix aer_inject return values (Prarit Bhargava) - Remove PME/ACPI dependency (Andrew Murray) - Use shared PCI_BUS_NUM() and PCI_DEVID() (Shuah Khan)" * tag 'pci-v3.10-changes' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci: (63 commits) vfio-pci: Use cached MSI/MSI-X capabilities vfio-pci: Use PCI_MSIX_TABLE_BIR, not PCI_MSIX_FLAGS_BIRMASK PCI: Remove "extern" from function declarations PCI: Use PCI_MSIX_TABLE_BIR, not PCI_MSIX_FLAGS_BIRMASK PCI: Drop msi_mask_reg() and remove drivers/pci/msi.h PCI: Use msix_table_size() directly, drop multi_msix_capable() PCI: Drop msix_table_offset_reg() and msix_pba_offset_reg() macros PCI: Drop is_64bit_address() and is_mask_bit_support() macros PCI: Drop msi_data_reg() macro PCI: Drop msi_lower_address_reg() and msi_upper_address_reg() macros PCI: Drop msi_control_reg() macro and use PCI_MSI_FLAGS directly PCI: Use cached MSI/MSI-X offsets from dev, not from msi_desc PCI: Clean up MSI/MSI-X capability #defines PCI: Use cached MSI-X cap while enabling MSI-X PCI: Use cached MSI cap while enabling MSI interrupts PCI: Remove MSI/MSI-X cap check in pci_msi_check_device() PCI: Cache MSI/MSI-X capability offsets in struct pci_dev PCI: Use u8, not int, for PM capability offset [SCSI] megaraid_sas: Use correct #define for MSI-X capability PCI: Remove "extern" from function declarations ...
This commit is contained in:
commit
96a3e8af5a
|
@ -15,6 +15,7 @@
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
#include <linux/pci-acpi.h>
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/ioport.h>
|
#include <linux/ioport.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
@ -458,6 +459,16 @@ void pcibios_fixup_bus(struct pci_bus *b)
|
||||||
platform_pci_fixup_bus(b);
|
platform_pci_fixup_bus(b);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void pcibios_add_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
acpi_pci_add_bus(bus);
|
||||||
|
}
|
||||||
|
|
||||||
|
void pcibios_remove_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
acpi_pci_remove_bus(bus);
|
||||||
|
}
|
||||||
|
|
||||||
void pcibios_set_master (struct pci_dev *dev)
|
void pcibios_set_master (struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
/* No special bus mastering setup handling */
|
/* No special bus mastering setup handling */
|
||||||
|
|
|
@ -115,7 +115,6 @@ static void pcibios_scanbus(struct pci_controller *hose)
|
||||||
pci_bus_assign_resources(bus);
|
pci_bus_assign_resources(bus);
|
||||||
pci_enable_bridges(bus);
|
pci_enable_bridges(bus);
|
||||||
}
|
}
|
||||||
bus->dev.of_node = hose->of_node;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -169,6 +168,13 @@ void pci_load_of_ranges(struct pci_controller *hose, struct device_node *node)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct device_node *pcibios_get_phb_of_node(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
struct pci_controller *hose = bus->sysdata;
|
||||||
|
|
||||||
|
return of_node_get(hose->of_node);
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static DEFINE_MUTEX(pci_scan_mutex);
|
static DEFINE_MUTEX(pci_scan_mutex);
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
|
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
#include <linux/pci-acpi.h>
|
||||||
#include <linux/ioport.h>
|
#include <linux/ioport.h>
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/dmi.h>
|
#include <linux/dmi.h>
|
||||||
|
@ -170,6 +171,16 @@ void pcibios_fixup_bus(struct pci_bus *b)
|
||||||
pcibios_fixup_device_resources(dev);
|
pcibios_fixup_device_resources(dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void pcibios_add_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
acpi_pci_add_bus(bus);
|
||||||
|
}
|
||||||
|
|
||||||
|
void pcibios_remove_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
acpi_pci_remove_bus(bus);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Only use DMI information to set this if nothing was passed
|
* Only use DMI information to set this if nothing was passed
|
||||||
* on the kernel command line (which was parsed earlier).
|
* on the kernel command line (which was parsed earlier).
|
||||||
|
|
|
@ -65,44 +65,12 @@ static struct acpi_scan_handler pci_root_handler = {
|
||||||
.detach = acpi_pci_root_remove,
|
.detach = acpi_pci_root_remove,
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Lock to protect both acpi_pci_roots and acpi_pci_drivers lists */
|
/* Lock to protect both acpi_pci_roots lists */
|
||||||
static DEFINE_MUTEX(acpi_pci_root_lock);
|
static DEFINE_MUTEX(acpi_pci_root_lock);
|
||||||
static LIST_HEAD(acpi_pci_roots);
|
static LIST_HEAD(acpi_pci_roots);
|
||||||
static LIST_HEAD(acpi_pci_drivers);
|
|
||||||
|
|
||||||
static DEFINE_MUTEX(osc_lock);
|
static DEFINE_MUTEX(osc_lock);
|
||||||
|
|
||||||
int acpi_pci_register_driver(struct acpi_pci_driver *driver)
|
|
||||||
{
|
|
||||||
int n = 0;
|
|
||||||
struct acpi_pci_root *root;
|
|
||||||
|
|
||||||
mutex_lock(&acpi_pci_root_lock);
|
|
||||||
list_add_tail(&driver->node, &acpi_pci_drivers);
|
|
||||||
if (driver->add)
|
|
||||||
list_for_each_entry(root, &acpi_pci_roots, node) {
|
|
||||||
driver->add(root);
|
|
||||||
n++;
|
|
||||||
}
|
|
||||||
mutex_unlock(&acpi_pci_root_lock);
|
|
||||||
|
|
||||||
return n;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(acpi_pci_register_driver);
|
|
||||||
|
|
||||||
void acpi_pci_unregister_driver(struct acpi_pci_driver *driver)
|
|
||||||
{
|
|
||||||
struct acpi_pci_root *root;
|
|
||||||
|
|
||||||
mutex_lock(&acpi_pci_root_lock);
|
|
||||||
list_del(&driver->node);
|
|
||||||
if (driver->remove)
|
|
||||||
list_for_each_entry(root, &acpi_pci_roots, node)
|
|
||||||
driver->remove(root);
|
|
||||||
mutex_unlock(&acpi_pci_root_lock);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(acpi_pci_unregister_driver);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* acpi_is_root_bridge - determine whether an ACPI CA node is a PCI root bridge
|
* acpi_is_root_bridge - determine whether an ACPI CA node is a PCI root bridge
|
||||||
* @handle - the ACPI CA node in question.
|
* @handle - the ACPI CA node in question.
|
||||||
|
@ -413,7 +381,6 @@ static int acpi_pci_root_add(struct acpi_device *device,
|
||||||
acpi_status status;
|
acpi_status status;
|
||||||
int result;
|
int result;
|
||||||
struct acpi_pci_root *root;
|
struct acpi_pci_root *root;
|
||||||
struct acpi_pci_driver *driver;
|
|
||||||
u32 flags, base_flags;
|
u32 flags, base_flags;
|
||||||
|
|
||||||
root = kzalloc(sizeof(struct acpi_pci_root), GFP_KERNEL);
|
root = kzalloc(sizeof(struct acpi_pci_root), GFP_KERNEL);
|
||||||
|
@ -571,12 +538,6 @@ static int acpi_pci_root_add(struct acpi_device *device,
|
||||||
pci_assign_unassigned_bus_resources(root->bus);
|
pci_assign_unassigned_bus_resources(root->bus);
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_lock(&acpi_pci_root_lock);
|
|
||||||
list_for_each_entry(driver, &acpi_pci_drivers, node)
|
|
||||||
if (driver->add)
|
|
||||||
driver->add(root);
|
|
||||||
mutex_unlock(&acpi_pci_root_lock);
|
|
||||||
|
|
||||||
/* need to after hot-added ioapic is registered */
|
/* need to after hot-added ioapic is registered */
|
||||||
if (system_state != SYSTEM_BOOTING)
|
if (system_state != SYSTEM_BOOTING)
|
||||||
pci_enable_bridges(root->bus);
|
pci_enable_bridges(root->bus);
|
||||||
|
@ -597,16 +558,9 @@ end:
|
||||||
static void acpi_pci_root_remove(struct acpi_device *device)
|
static void acpi_pci_root_remove(struct acpi_device *device)
|
||||||
{
|
{
|
||||||
struct acpi_pci_root *root = acpi_driver_data(device);
|
struct acpi_pci_root *root = acpi_driver_data(device);
|
||||||
struct acpi_pci_driver *driver;
|
|
||||||
|
|
||||||
pci_stop_root_bus(root->bus);
|
pci_stop_root_bus(root->bus);
|
||||||
|
|
||||||
mutex_lock(&acpi_pci_root_lock);
|
|
||||||
list_for_each_entry_reverse(driver, &acpi_pci_drivers, node)
|
|
||||||
if (driver->remove)
|
|
||||||
driver->remove(root);
|
|
||||||
mutex_unlock(&acpi_pci_root_lock);
|
|
||||||
|
|
||||||
device_set_run_wake(root->bus->bridge, false);
|
device_set_run_wake(root->bus->bridge, false);
|
||||||
pci_acpi_remove_bus_pm_notifier(device);
|
pci_acpi_remove_bus_pm_notifier(device);
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,9 @@
|
||||||
* Copyright (C) 2007-2008 Hewlett-Packard Development Company, L.P.
|
* Copyright (C) 2007-2008 Hewlett-Packard Development Company, L.P.
|
||||||
* Alex Chiang <achiang@hp.com>
|
* Alex Chiang <achiang@hp.com>
|
||||||
*
|
*
|
||||||
|
* Copyright (C) 2013 Huawei Tech. Co., Ltd.
|
||||||
|
* Jiang Liu <jiang.liu@huawei.com>
|
||||||
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
* under the terms and conditions of the GNU General Public License,
|
* under the terms and conditions of the GNU General Public License,
|
||||||
* version 2, as published by the Free Software Foundation.
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
@ -28,10 +31,9 @@
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
#include <linux/list.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
#include <linux/acpi.h>
|
#include <linux/acpi.h>
|
||||||
#include <acpi/acpi_bus.h>
|
|
||||||
#include <acpi/acpi_drivers.h>
|
|
||||||
#include <linux/dmi.h>
|
#include <linux/dmi.h>
|
||||||
|
|
||||||
static bool debug;
|
static bool debug;
|
||||||
|
@ -61,20 +63,12 @@ ACPI_MODULE_NAME("pci_slot");
|
||||||
#define SLOT_NAME_SIZE 21 /* Inspired by #define in acpiphp.h */
|
#define SLOT_NAME_SIZE 21 /* Inspired by #define in acpiphp.h */
|
||||||
|
|
||||||
struct acpi_pci_slot {
|
struct acpi_pci_slot {
|
||||||
acpi_handle root_handle; /* handle of the root bridge */
|
|
||||||
struct pci_slot *pci_slot; /* corresponding pci_slot */
|
struct pci_slot *pci_slot; /* corresponding pci_slot */
|
||||||
struct list_head list; /* node in the list of slots */
|
struct list_head list; /* node in the list of slots */
|
||||||
};
|
};
|
||||||
|
|
||||||
static int acpi_pci_slot_add(struct acpi_pci_root *root);
|
|
||||||
static void acpi_pci_slot_remove(struct acpi_pci_root *root);
|
|
||||||
|
|
||||||
static LIST_HEAD(slot_list);
|
static LIST_HEAD(slot_list);
|
||||||
static DEFINE_MUTEX(slot_list_lock);
|
static DEFINE_MUTEX(slot_list_lock);
|
||||||
static struct acpi_pci_driver acpi_pci_slot_driver = {
|
|
||||||
.add = acpi_pci_slot_add,
|
|
||||||
.remove = acpi_pci_slot_remove,
|
|
||||||
};
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
check_slot(acpi_handle handle, unsigned long long *sun)
|
check_slot(acpi_handle handle, unsigned long long *sun)
|
||||||
|
@ -113,21 +107,8 @@ out:
|
||||||
return device;
|
return device;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct callback_args {
|
|
||||||
acpi_walk_callback user_function; /* only for walk_p2p_bridge */
|
|
||||||
struct pci_bus *pci_bus;
|
|
||||||
acpi_handle root_handle;
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* register_slot
|
* Check whether handle has an associated slot and create PCI slot if it has.
|
||||||
*
|
|
||||||
* Called once for each SxFy object in the namespace. Don't worry about
|
|
||||||
* calling pci_create_slot multiple times for the same pci_bus:device,
|
|
||||||
* since each subsequent call simply bumps the refcount on the pci_slot.
|
|
||||||
*
|
|
||||||
* The number of calls to pci_destroy_slot from unregister_slot is
|
|
||||||
* symmetrical.
|
|
||||||
*/
|
*/
|
||||||
static acpi_status
|
static acpi_status
|
||||||
register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
|
@ -137,13 +118,22 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
char name[SLOT_NAME_SIZE];
|
char name[SLOT_NAME_SIZE];
|
||||||
struct acpi_pci_slot *slot;
|
struct acpi_pci_slot *slot;
|
||||||
struct pci_slot *pci_slot;
|
struct pci_slot *pci_slot;
|
||||||
struct callback_args *parent_context = context;
|
struct pci_bus *pci_bus = context;
|
||||||
struct pci_bus *pci_bus = parent_context->pci_bus;
|
|
||||||
|
|
||||||
device = check_slot(handle, &sun);
|
device = check_slot(handle, &sun);
|
||||||
if (device < 0)
|
if (device < 0)
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* There may be multiple PCI functions associated with the same slot.
|
||||||
|
* Check whether PCI slot has already been created for this PCI device.
|
||||||
|
*/
|
||||||
|
list_for_each_entry(slot, &slot_list, list) {
|
||||||
|
pci_slot = slot->pci_slot;
|
||||||
|
if (pci_slot->bus == pci_bus && pci_slot->number == device)
|
||||||
|
return AE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
slot = kmalloc(sizeof(*slot), GFP_KERNEL);
|
slot = kmalloc(sizeof(*slot), GFP_KERNEL);
|
||||||
if (!slot) {
|
if (!slot) {
|
||||||
err("%s: cannot allocate memory\n", __func__);
|
err("%s: cannot allocate memory\n", __func__);
|
||||||
|
@ -158,12 +148,8 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
slot->root_handle = parent_context->root_handle;
|
|
||||||
slot->pci_slot = pci_slot;
|
slot->pci_slot = pci_slot;
|
||||||
INIT_LIST_HEAD(&slot->list);
|
|
||||||
mutex_lock(&slot_list_lock);
|
|
||||||
list_add(&slot->list, &slot_list);
|
list_add(&slot->list, &slot_list);
|
||||||
mutex_unlock(&slot_list_lock);
|
|
||||||
|
|
||||||
get_device(&pci_bus->dev);
|
get_device(&pci_bus->dev);
|
||||||
|
|
||||||
|
@ -173,131 +159,24 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
void acpi_pci_slot_enumerate(struct pci_bus *bus, acpi_handle handle)
|
||||||
* walk_p2p_bridge - discover and walk p2p bridges
|
|
||||||
* @handle: points to an acpi_pci_root
|
|
||||||
* @context: p2p_bridge_context pointer
|
|
||||||
*
|
|
||||||
* Note that when we call ourselves recursively, we pass a different
|
|
||||||
* value of pci_bus in the child_context.
|
|
||||||
*/
|
|
||||||
static acpi_status
|
|
||||||
walk_p2p_bridge(acpi_handle handle, u32 lvl, void *context, void **rv)
|
|
||||||
{
|
{
|
||||||
int device, function;
|
mutex_lock(&slot_list_lock);
|
||||||
unsigned long long adr;
|
acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, 1,
|
||||||
acpi_status status;
|
register_slot, NULL, bus, NULL);
|
||||||
acpi_handle dummy_handle;
|
mutex_unlock(&slot_list_lock);
|
||||||
acpi_walk_callback user_function;
|
|
||||||
|
|
||||||
struct pci_dev *dev;
|
|
||||||
struct pci_bus *pci_bus;
|
|
||||||
struct callback_args child_context;
|
|
||||||
struct callback_args *parent_context = context;
|
|
||||||
|
|
||||||
pci_bus = parent_context->pci_bus;
|
|
||||||
user_function = parent_context->user_function;
|
|
||||||
|
|
||||||
status = acpi_get_handle(handle, "_ADR", &dummy_handle);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
return AE_OK;
|
|
||||||
|
|
||||||
status = acpi_evaluate_integer(handle, "_ADR", NULL, &adr);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
return AE_OK;
|
|
||||||
|
|
||||||
device = (adr >> 16) & 0xffff;
|
|
||||||
function = adr & 0xffff;
|
|
||||||
|
|
||||||
dev = pci_get_slot(pci_bus, PCI_DEVFN(device, function));
|
|
||||||
if (!dev || !dev->subordinate)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
child_context.pci_bus = dev->subordinate;
|
|
||||||
child_context.user_function = user_function;
|
|
||||||
child_context.root_handle = parent_context->root_handle;
|
|
||||||
|
|
||||||
dbg("p2p bridge walk, pci_bus = %x\n", dev->subordinate->number);
|
|
||||||
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, (u32)1,
|
|
||||||
user_function, NULL, &child_context, NULL);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, (u32)1,
|
|
||||||
walk_p2p_bridge, NULL, &child_context, NULL);
|
|
||||||
out:
|
|
||||||
pci_dev_put(dev);
|
|
||||||
return AE_OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
void acpi_pci_slot_remove(struct pci_bus *bus)
|
||||||
* walk_root_bridge - generic root bridge walker
|
|
||||||
* @root: poiner of an acpi_pci_root
|
|
||||||
* @user_function: user callback for slot objects
|
|
||||||
*
|
|
||||||
* Call user_function for all objects underneath this root bridge.
|
|
||||||
* Walk p2p bridges underneath us and call user_function on those too.
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
walk_root_bridge(struct acpi_pci_root *root, acpi_walk_callback user_function)
|
|
||||||
{
|
|
||||||
acpi_status status;
|
|
||||||
acpi_handle handle = root->device->handle;
|
|
||||||
struct pci_bus *pci_bus = root->bus;
|
|
||||||
struct callback_args context;
|
|
||||||
|
|
||||||
context.pci_bus = pci_bus;
|
|
||||||
context.user_function = user_function;
|
|
||||||
context.root_handle = handle;
|
|
||||||
|
|
||||||
dbg("root bridge walk, pci_bus = %x\n", pci_bus->number);
|
|
||||||
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, (u32)1,
|
|
||||||
user_function, NULL, &context, NULL);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
return status;
|
|
||||||
|
|
||||||
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, (u32)1,
|
|
||||||
walk_p2p_bridge, NULL, &context, NULL);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
err("%s: walk_p2p_bridge failure - %d\n", __func__, status);
|
|
||||||
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* acpi_pci_slot_add
|
|
||||||
* @handle: points to an acpi_pci_root
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
acpi_pci_slot_add(struct acpi_pci_root *root)
|
|
||||||
{
|
|
||||||
acpi_status status;
|
|
||||||
|
|
||||||
status = walk_root_bridge(root, register_slot);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
err("%s: register_slot failure - %d\n", __func__, status);
|
|
||||||
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* acpi_pci_slot_remove
|
|
||||||
* @handle: points to an acpi_pci_root
|
|
||||||
*/
|
|
||||||
static void
|
|
||||||
acpi_pci_slot_remove(struct acpi_pci_root *root)
|
|
||||||
{
|
{
|
||||||
struct acpi_pci_slot *slot, *tmp;
|
struct acpi_pci_slot *slot, *tmp;
|
||||||
struct pci_bus *pbus;
|
|
||||||
acpi_handle handle = root->device->handle;
|
|
||||||
|
|
||||||
mutex_lock(&slot_list_lock);
|
mutex_lock(&slot_list_lock);
|
||||||
list_for_each_entry_safe(slot, tmp, &slot_list, list) {
|
list_for_each_entry_safe(slot, tmp, &slot_list, list) {
|
||||||
if (slot->root_handle == handle) {
|
if (slot->pci_slot->bus == bus) {
|
||||||
list_del(&slot->list);
|
list_del(&slot->list);
|
||||||
pbus = slot->pci_slot->bus;
|
|
||||||
pci_destroy_slot(slot->pci_slot);
|
pci_destroy_slot(slot->pci_slot);
|
||||||
put_device(&pbus->dev);
|
put_device(&bus->dev);
|
||||||
kfree(slot);
|
kfree(slot);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -332,5 +211,4 @@ static struct dmi_system_id acpi_pci_slot_dmi_table[] __initdata = {
|
||||||
void __init acpi_pci_slot_init(void)
|
void __init acpi_pci_slot_init(void)
|
||||||
{
|
{
|
||||||
dmi_check_system(acpi_pci_slot_dmi_table);
|
dmi_check_system(acpi_pci_slot_dmi_table);
|
||||||
acpi_pci_register_driver(&acpi_pci_slot_driver);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1790,7 +1790,6 @@ int __init acpi_scan_init(void)
|
||||||
acpi_platform_init();
|
acpi_platform_init();
|
||||||
acpi_csrt_init();
|
acpi_csrt_init();
|
||||||
acpi_container_init();
|
acpi_container_init();
|
||||||
acpi_pci_slot_init();
|
|
||||||
|
|
||||||
mutex_lock(&acpi_scan_lock);
|
mutex_lock(&acpi_scan_lock);
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -415,17 +415,17 @@ static const struct pci_device_id ahci_pci_tbl[] = {
|
||||||
/* Marvell */
|
/* Marvell */
|
||||||
{ PCI_VDEVICE(MARVELL, 0x6145), board_ahci_mv }, /* 6145 */
|
{ PCI_VDEVICE(MARVELL, 0x6145), board_ahci_mv }, /* 6145 */
|
||||||
{ PCI_VDEVICE(MARVELL, 0x6121), board_ahci_mv }, /* 6121 */
|
{ PCI_VDEVICE(MARVELL, 0x6121), board_ahci_mv }, /* 6121 */
|
||||||
{ PCI_DEVICE(0x1b4b, 0x9123),
|
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_EXT, 0x9123),
|
||||||
.class = PCI_CLASS_STORAGE_SATA_AHCI,
|
.class = PCI_CLASS_STORAGE_SATA_AHCI,
|
||||||
.class_mask = 0xffffff,
|
.class_mask = 0xffffff,
|
||||||
.driver_data = board_ahci_yes_fbs }, /* 88se9128 */
|
.driver_data = board_ahci_yes_fbs }, /* 88se9128 */
|
||||||
{ PCI_DEVICE(0x1b4b, 0x9125),
|
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_EXT, 0x9125),
|
||||||
.driver_data = board_ahci_yes_fbs }, /* 88se9125 */
|
.driver_data = board_ahci_yes_fbs }, /* 88se9125 */
|
||||||
{ PCI_DEVICE(0x1b4b, 0x917a),
|
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_EXT, 0x917a),
|
||||||
.driver_data = board_ahci_yes_fbs }, /* 88se9172 */
|
.driver_data = board_ahci_yes_fbs }, /* 88se9172 */
|
||||||
{ PCI_DEVICE(0x1b4b, 0x9192),
|
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_EXT, 0x9192),
|
||||||
.driver_data = board_ahci_yes_fbs }, /* 88se9172 on some Gigabyte */
|
.driver_data = board_ahci_yes_fbs }, /* 88se9172 on some Gigabyte */
|
||||||
{ PCI_DEVICE(0x1b4b, 0x91a3),
|
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_EXT, 0x91a3),
|
||||||
.driver_data = board_ahci_yes_fbs },
|
.driver_data = board_ahci_yes_fbs },
|
||||||
|
|
||||||
/* Promise */
|
/* Promise */
|
||||||
|
|
|
@ -275,19 +275,18 @@ static int __init eisa_request_resources(struct eisa_root_device *root,
|
||||||
}
|
}
|
||||||
|
|
||||||
if (slot) {
|
if (slot) {
|
||||||
edev->res[i].name = NULL;
|
|
||||||
edev->res[i].start = SLOT_ADDRESS(root, slot)
|
edev->res[i].start = SLOT_ADDRESS(root, slot)
|
||||||
+ (i * 0x400);
|
+ (i * 0x400);
|
||||||
edev->res[i].end = edev->res[i].start + 0xff;
|
edev->res[i].end = edev->res[i].start + 0xff;
|
||||||
edev->res[i].flags = IORESOURCE_IO;
|
edev->res[i].flags = IORESOURCE_IO;
|
||||||
} else {
|
} else {
|
||||||
edev->res[i].name = NULL;
|
|
||||||
edev->res[i].start = SLOT_ADDRESS(root, slot)
|
edev->res[i].start = SLOT_ADDRESS(root, slot)
|
||||||
+ EISA_VENDOR_ID_OFFSET;
|
+ EISA_VENDOR_ID_OFFSET;
|
||||||
edev->res[i].end = edev->res[i].start + 3;
|
edev->res[i].end = edev->res[i].start + 3;
|
||||||
edev->res[i].flags = IORESOURCE_BUSY;
|
edev->res[i].flags = IORESOURCE_IO | IORESOURCE_BUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dev_printk(KERN_DEBUG, &edev->dev, "%pR\n", &edev->res[i]);
|
||||||
if (request_resource(root->res, &edev->res[i]))
|
if (request_resource(root->res, &edev->res[i]))
|
||||||
goto failed;
|
goto failed;
|
||||||
}
|
}
|
||||||
|
@ -314,41 +313,40 @@ static int __init eisa_probe(struct eisa_root_device *root)
|
||||||
{
|
{
|
||||||
int i, c;
|
int i, c;
|
||||||
struct eisa_device *edev;
|
struct eisa_device *edev;
|
||||||
|
char *enabled_str;
|
||||||
|
|
||||||
printk(KERN_INFO "EISA: Probing bus %d at %s\n",
|
dev_info(root->dev, "Probing EISA bus %d\n", root->bus_nr);
|
||||||
root->bus_nr, dev_name(root->dev));
|
|
||||||
|
|
||||||
/* First try to get hold of slot 0. If there is no device
|
/* First try to get hold of slot 0. If there is no device
|
||||||
* here, simply fail, unless root->force_probe is set. */
|
* here, simply fail, unless root->force_probe is set. */
|
||||||
|
|
||||||
edev = kzalloc(sizeof(*edev), GFP_KERNEL);
|
edev = kzalloc(sizeof(*edev), GFP_KERNEL);
|
||||||
if (!edev) {
|
if (!edev) {
|
||||||
printk(KERN_ERR "EISA: Couldn't allocate mainboard slot\n");
|
dev_err(root->dev, "EISA: Couldn't allocate mainboard slot\n");
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (eisa_request_resources(root, edev, 0)) {
|
|
||||||
printk(KERN_WARNING \
|
|
||||||
"EISA: Cannot allocate resource for mainboard\n");
|
|
||||||
kfree(edev);
|
|
||||||
if (!root->force_probe)
|
|
||||||
return -EBUSY;
|
|
||||||
goto force_probe;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (eisa_init_device(root, edev, 0)) {
|
if (eisa_init_device(root, edev, 0)) {
|
||||||
eisa_release_resources(edev);
|
|
||||||
kfree(edev);
|
kfree(edev);
|
||||||
if (!root->force_probe)
|
if (!root->force_probe)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
goto force_probe;
|
goto force_probe;
|
||||||
}
|
}
|
||||||
|
|
||||||
printk(KERN_INFO "EISA: Mainboard %s detected.\n", edev->id.sig);
|
if (eisa_request_resources(root, edev, 0)) {
|
||||||
|
dev_warn(root->dev,
|
||||||
|
"EISA: Cannot allocate resource for mainboard\n");
|
||||||
|
kfree(edev);
|
||||||
|
if (!root->force_probe)
|
||||||
|
return -EBUSY;
|
||||||
|
goto force_probe;
|
||||||
|
}
|
||||||
|
|
||||||
|
dev_info(&edev->dev, "EISA: Mainboard %s detected\n", edev->id.sig);
|
||||||
|
|
||||||
if (eisa_register_device(edev)) {
|
if (eisa_register_device(edev)) {
|
||||||
printk(KERN_ERR "EISA: Failed to register %s\n",
|
dev_err(&edev->dev, "EISA: Failed to register %s\n",
|
||||||
edev->id.sig);
|
edev->id.sig);
|
||||||
eisa_release_resources(edev);
|
eisa_release_resources(edev);
|
||||||
kfree(edev);
|
kfree(edev);
|
||||||
}
|
}
|
||||||
|
@ -358,55 +356,47 @@ static int __init eisa_probe(struct eisa_root_device *root)
|
||||||
for (c = 0, i = 1; i <= root->slots; i++) {
|
for (c = 0, i = 1; i <= root->slots; i++) {
|
||||||
edev = kzalloc(sizeof(*edev), GFP_KERNEL);
|
edev = kzalloc(sizeof(*edev), GFP_KERNEL);
|
||||||
if (!edev) {
|
if (!edev) {
|
||||||
printk(KERN_ERR "EISA: Out of memory for slot %d\n", i);
|
dev_err(root->dev, "EISA: Out of memory for slot %d\n",
|
||||||
continue;
|
i);
|
||||||
}
|
|
||||||
|
|
||||||
if (eisa_request_resources(root, edev, i)) {
|
|
||||||
printk(KERN_WARNING \
|
|
||||||
"Cannot allocate resource for EISA slot %d\n",
|
|
||||||
i);
|
|
||||||
kfree(edev);
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (eisa_init_device(root, edev, i)) {
|
if (eisa_init_device(root, edev, i)) {
|
||||||
eisa_release_resources(edev);
|
|
||||||
kfree(edev);
|
kfree(edev);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
printk(KERN_INFO "EISA: slot %d : %s detected",
|
|
||||||
i, edev->id.sig);
|
|
||||||
|
|
||||||
switch (edev->state) {
|
|
||||||
case EISA_CONFIG_ENABLED | EISA_CONFIG_FORCED:
|
|
||||||
printk(" (forced enabled)");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case EISA_CONFIG_FORCED:
|
if (eisa_request_resources(root, edev, i)) {
|
||||||
printk(" (forced disabled)");
|
dev_warn(root->dev,
|
||||||
break;
|
"Cannot allocate resource for EISA slot %d\n",
|
||||||
|
i);
|
||||||
case 0:
|
kfree(edev);
|
||||||
printk(" (disabled)");
|
continue;
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
printk (".\n");
|
if (edev->state == (EISA_CONFIG_ENABLED | EISA_CONFIG_FORCED))
|
||||||
|
enabled_str = " (forced enabled)";
|
||||||
|
else if (edev->state == EISA_CONFIG_FORCED)
|
||||||
|
enabled_str = " (forced disabled)";
|
||||||
|
else if (edev->state == 0)
|
||||||
|
enabled_str = " (disabled)";
|
||||||
|
else
|
||||||
|
enabled_str = "";
|
||||||
|
|
||||||
|
dev_info(&edev->dev, "EISA: slot %d: %s detected%s\n", i,
|
||||||
|
edev->id.sig, enabled_str);
|
||||||
|
|
||||||
c++;
|
c++;
|
||||||
|
|
||||||
if (eisa_register_device(edev)) {
|
if (eisa_register_device(edev)) {
|
||||||
printk(KERN_ERR "EISA: Failed to register %s\n",
|
dev_err(&edev->dev, "EISA: Failed to register %s\n",
|
||||||
edev->id.sig);
|
edev->id.sig);
|
||||||
eisa_release_resources(edev);
|
eisa_release_resources(edev);
|
||||||
kfree(edev);
|
kfree(edev);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
printk(KERN_INFO "EISA: Detected %d card%s.\n", c, c == 1 ? "" : "s");
|
dev_info(root->dev, "EISA: Detected %d card%s\n", c, c == 1 ? "" : "s");
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -25,8 +25,7 @@ static int __init pci_eisa_init(struct pci_dev *pdev)
|
||||||
struct resource *res, *bus_res = NULL;
|
struct resource *res, *bus_res = NULL;
|
||||||
|
|
||||||
if ((rc = pci_enable_device (pdev))) {
|
if ((rc = pci_enable_device (pdev))) {
|
||||||
printk (KERN_ERR "pci_eisa : Could not enable device %s\n",
|
dev_err(&pdev->dev, "Could not enable device\n");
|
||||||
pci_name(pdev));
|
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -59,7 +58,7 @@ static int __init pci_eisa_init(struct pci_dev *pdev)
|
||||||
dev_set_drvdata(pci_eisa_root.dev, &pci_eisa_root);
|
dev_set_drvdata(pci_eisa_root.dev, &pci_eisa_root);
|
||||||
|
|
||||||
if (eisa_root_register (&pci_eisa_root)) {
|
if (eisa_root_register (&pci_eisa_root)) {
|
||||||
printk (KERN_ERR "pci_eisa : Could not register EISA root\n");
|
dev_err(&pdev->dev, "Could not register EISA root\n");
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -173,7 +173,7 @@ static inline u16 get_device_id(struct device *dev)
|
||||||
{
|
{
|
||||||
struct pci_dev *pdev = to_pci_dev(dev);
|
struct pci_dev *pdev = to_pci_dev(dev);
|
||||||
|
|
||||||
return calc_devid(pdev->bus->number, pdev->devfn);
|
return PCI_DEVID(pdev->bus->number, pdev->devfn);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct iommu_dev_data *get_dev_data(struct device *dev)
|
static struct iommu_dev_data *get_dev_data(struct device *dev)
|
||||||
|
@ -649,26 +649,26 @@ retry:
|
||||||
case EVENT_TYPE_ILL_DEV:
|
case EVENT_TYPE_ILL_DEV:
|
||||||
printk("ILLEGAL_DEV_TABLE_ENTRY device=%02x:%02x.%x "
|
printk("ILLEGAL_DEV_TABLE_ENTRY device=%02x:%02x.%x "
|
||||||
"address=0x%016llx flags=0x%04x]\n",
|
"address=0x%016llx flags=0x%04x]\n",
|
||||||
PCI_BUS(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
PCI_BUS_NUM(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
||||||
address, flags);
|
address, flags);
|
||||||
dump_dte_entry(devid);
|
dump_dte_entry(devid);
|
||||||
break;
|
break;
|
||||||
case EVENT_TYPE_IO_FAULT:
|
case EVENT_TYPE_IO_FAULT:
|
||||||
printk("IO_PAGE_FAULT device=%02x:%02x.%x "
|
printk("IO_PAGE_FAULT device=%02x:%02x.%x "
|
||||||
"domain=0x%04x address=0x%016llx flags=0x%04x]\n",
|
"domain=0x%04x address=0x%016llx flags=0x%04x]\n",
|
||||||
PCI_BUS(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
PCI_BUS_NUM(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
||||||
domid, address, flags);
|
domid, address, flags);
|
||||||
break;
|
break;
|
||||||
case EVENT_TYPE_DEV_TAB_ERR:
|
case EVENT_TYPE_DEV_TAB_ERR:
|
||||||
printk("DEV_TAB_HARDWARE_ERROR device=%02x:%02x.%x "
|
printk("DEV_TAB_HARDWARE_ERROR device=%02x:%02x.%x "
|
||||||
"address=0x%016llx flags=0x%04x]\n",
|
"address=0x%016llx flags=0x%04x]\n",
|
||||||
PCI_BUS(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
PCI_BUS_NUM(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
||||||
address, flags);
|
address, flags);
|
||||||
break;
|
break;
|
||||||
case EVENT_TYPE_PAGE_TAB_ERR:
|
case EVENT_TYPE_PAGE_TAB_ERR:
|
||||||
printk("PAGE_TAB_HARDWARE_ERROR device=%02x:%02x.%x "
|
printk("PAGE_TAB_HARDWARE_ERROR device=%02x:%02x.%x "
|
||||||
"domain=0x%04x address=0x%016llx flags=0x%04x]\n",
|
"domain=0x%04x address=0x%016llx flags=0x%04x]\n",
|
||||||
PCI_BUS(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
PCI_BUS_NUM(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
||||||
domid, address, flags);
|
domid, address, flags);
|
||||||
break;
|
break;
|
||||||
case EVENT_TYPE_ILL_CMD:
|
case EVENT_TYPE_ILL_CMD:
|
||||||
|
@ -682,13 +682,13 @@ retry:
|
||||||
case EVENT_TYPE_IOTLB_INV_TO:
|
case EVENT_TYPE_IOTLB_INV_TO:
|
||||||
printk("IOTLB_INV_TIMEOUT device=%02x:%02x.%x "
|
printk("IOTLB_INV_TIMEOUT device=%02x:%02x.%x "
|
||||||
"address=0x%016llx]\n",
|
"address=0x%016llx]\n",
|
||||||
PCI_BUS(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
PCI_BUS_NUM(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
||||||
address);
|
address);
|
||||||
break;
|
break;
|
||||||
case EVENT_TYPE_INV_DEV_REQ:
|
case EVENT_TYPE_INV_DEV_REQ:
|
||||||
printk("INVALID_DEVICE_REQUEST device=%02x:%02x.%x "
|
printk("INVALID_DEVICE_REQUEST device=%02x:%02x.%x "
|
||||||
"address=0x%016llx flags=0x%04x]\n",
|
"address=0x%016llx flags=0x%04x]\n",
|
||||||
PCI_BUS(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
PCI_BUS_NUM(devid), PCI_SLOT(devid), PCI_FUNC(devid),
|
||||||
address, flags);
|
address, flags);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
|
|
@ -406,7 +406,7 @@ static int __init find_last_devid_on_pci(int bus, int dev, int fn, int cap_ptr)
|
||||||
u32 cap;
|
u32 cap;
|
||||||
|
|
||||||
cap = read_pci_config(bus, dev, fn, cap_ptr+MMIO_RANGE_OFFSET);
|
cap = read_pci_config(bus, dev, fn, cap_ptr+MMIO_RANGE_OFFSET);
|
||||||
update_last_devid(calc_devid(MMIO_GET_BUS(cap), MMIO_GET_LD(cap)));
|
update_last_devid(PCI_DEVID(MMIO_GET_BUS(cap), MMIO_GET_LD(cap)));
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -423,7 +423,7 @@ static int __init find_last_devid_from_ivhd(struct ivhd_header *h)
|
||||||
p += sizeof(*h);
|
p += sizeof(*h);
|
||||||
end += h->length;
|
end += h->length;
|
||||||
|
|
||||||
find_last_devid_on_pci(PCI_BUS(h->devid),
|
find_last_devid_on_pci(PCI_BUS_NUM(h->devid),
|
||||||
PCI_SLOT(h->devid),
|
PCI_SLOT(h->devid),
|
||||||
PCI_FUNC(h->devid),
|
PCI_FUNC(h->devid),
|
||||||
h->cap_ptr);
|
h->cap_ptr);
|
||||||
|
@ -784,10 +784,10 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
|
|
||||||
DUMP_printk(" DEV_ALL\t\t\t first devid: %02x:%02x.%x"
|
DUMP_printk(" DEV_ALL\t\t\t first devid: %02x:%02x.%x"
|
||||||
" last device %02x:%02x.%x flags: %02x\n",
|
" last device %02x:%02x.%x flags: %02x\n",
|
||||||
PCI_BUS(iommu->first_device),
|
PCI_BUS_NUM(iommu->first_device),
|
||||||
PCI_SLOT(iommu->first_device),
|
PCI_SLOT(iommu->first_device),
|
||||||
PCI_FUNC(iommu->first_device),
|
PCI_FUNC(iommu->first_device),
|
||||||
PCI_BUS(iommu->last_device),
|
PCI_BUS_NUM(iommu->last_device),
|
||||||
PCI_SLOT(iommu->last_device),
|
PCI_SLOT(iommu->last_device),
|
||||||
PCI_FUNC(iommu->last_device),
|
PCI_FUNC(iommu->last_device),
|
||||||
e->flags);
|
e->flags);
|
||||||
|
@ -801,7 +801,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
|
|
||||||
DUMP_printk(" DEV_SELECT\t\t\t devid: %02x:%02x.%x "
|
DUMP_printk(" DEV_SELECT\t\t\t devid: %02x:%02x.%x "
|
||||||
"flags: %02x\n",
|
"flags: %02x\n",
|
||||||
PCI_BUS(e->devid),
|
PCI_BUS_NUM(e->devid),
|
||||||
PCI_SLOT(e->devid),
|
PCI_SLOT(e->devid),
|
||||||
PCI_FUNC(e->devid),
|
PCI_FUNC(e->devid),
|
||||||
e->flags);
|
e->flags);
|
||||||
|
@ -813,7 +813,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
|
|
||||||
DUMP_printk(" DEV_SELECT_RANGE_START\t "
|
DUMP_printk(" DEV_SELECT_RANGE_START\t "
|
||||||
"devid: %02x:%02x.%x flags: %02x\n",
|
"devid: %02x:%02x.%x flags: %02x\n",
|
||||||
PCI_BUS(e->devid),
|
PCI_BUS_NUM(e->devid),
|
||||||
PCI_SLOT(e->devid),
|
PCI_SLOT(e->devid),
|
||||||
PCI_FUNC(e->devid),
|
PCI_FUNC(e->devid),
|
||||||
e->flags);
|
e->flags);
|
||||||
|
@ -827,11 +827,11 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
|
|
||||||
DUMP_printk(" DEV_ALIAS\t\t\t devid: %02x:%02x.%x "
|
DUMP_printk(" DEV_ALIAS\t\t\t devid: %02x:%02x.%x "
|
||||||
"flags: %02x devid_to: %02x:%02x.%x\n",
|
"flags: %02x devid_to: %02x:%02x.%x\n",
|
||||||
PCI_BUS(e->devid),
|
PCI_BUS_NUM(e->devid),
|
||||||
PCI_SLOT(e->devid),
|
PCI_SLOT(e->devid),
|
||||||
PCI_FUNC(e->devid),
|
PCI_FUNC(e->devid),
|
||||||
e->flags,
|
e->flags,
|
||||||
PCI_BUS(e->ext >> 8),
|
PCI_BUS_NUM(e->ext >> 8),
|
||||||
PCI_SLOT(e->ext >> 8),
|
PCI_SLOT(e->ext >> 8),
|
||||||
PCI_FUNC(e->ext >> 8));
|
PCI_FUNC(e->ext >> 8));
|
||||||
|
|
||||||
|
@ -846,11 +846,11 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
DUMP_printk(" DEV_ALIAS_RANGE\t\t "
|
DUMP_printk(" DEV_ALIAS_RANGE\t\t "
|
||||||
"devid: %02x:%02x.%x flags: %02x "
|
"devid: %02x:%02x.%x flags: %02x "
|
||||||
"devid_to: %02x:%02x.%x\n",
|
"devid_to: %02x:%02x.%x\n",
|
||||||
PCI_BUS(e->devid),
|
PCI_BUS_NUM(e->devid),
|
||||||
PCI_SLOT(e->devid),
|
PCI_SLOT(e->devid),
|
||||||
PCI_FUNC(e->devid),
|
PCI_FUNC(e->devid),
|
||||||
e->flags,
|
e->flags,
|
||||||
PCI_BUS(e->ext >> 8),
|
PCI_BUS_NUM(e->ext >> 8),
|
||||||
PCI_SLOT(e->ext >> 8),
|
PCI_SLOT(e->ext >> 8),
|
||||||
PCI_FUNC(e->ext >> 8));
|
PCI_FUNC(e->ext >> 8));
|
||||||
|
|
||||||
|
@ -864,7 +864,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
|
|
||||||
DUMP_printk(" DEV_EXT_SELECT\t\t devid: %02x:%02x.%x "
|
DUMP_printk(" DEV_EXT_SELECT\t\t devid: %02x:%02x.%x "
|
||||||
"flags: %02x ext: %08x\n",
|
"flags: %02x ext: %08x\n",
|
||||||
PCI_BUS(e->devid),
|
PCI_BUS_NUM(e->devid),
|
||||||
PCI_SLOT(e->devid),
|
PCI_SLOT(e->devid),
|
||||||
PCI_FUNC(e->devid),
|
PCI_FUNC(e->devid),
|
||||||
e->flags, e->ext);
|
e->flags, e->ext);
|
||||||
|
@ -877,7 +877,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
|
|
||||||
DUMP_printk(" DEV_EXT_SELECT_RANGE\t devid: "
|
DUMP_printk(" DEV_EXT_SELECT_RANGE\t devid: "
|
||||||
"%02x:%02x.%x flags: %02x ext: %08x\n",
|
"%02x:%02x.%x flags: %02x ext: %08x\n",
|
||||||
PCI_BUS(e->devid),
|
PCI_BUS_NUM(e->devid),
|
||||||
PCI_SLOT(e->devid),
|
PCI_SLOT(e->devid),
|
||||||
PCI_FUNC(e->devid),
|
PCI_FUNC(e->devid),
|
||||||
e->flags, e->ext);
|
e->flags, e->ext);
|
||||||
|
@ -890,7 +890,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
case IVHD_DEV_RANGE_END:
|
case IVHD_DEV_RANGE_END:
|
||||||
|
|
||||||
DUMP_printk(" DEV_RANGE_END\t\t devid: %02x:%02x.%x\n",
|
DUMP_printk(" DEV_RANGE_END\t\t devid: %02x:%02x.%x\n",
|
||||||
PCI_BUS(e->devid),
|
PCI_BUS_NUM(e->devid),
|
||||||
PCI_SLOT(e->devid),
|
PCI_SLOT(e->devid),
|
||||||
PCI_FUNC(e->devid));
|
PCI_FUNC(e->devid));
|
||||||
|
|
||||||
|
@ -924,7 +924,7 @@ static int __init init_iommu_from_acpi(struct amd_iommu *iommu,
|
||||||
|
|
||||||
DUMP_printk(" DEV_SPECIAL(%s[%d])\t\tdevid: %02x:%02x.%x\n",
|
DUMP_printk(" DEV_SPECIAL(%s[%d])\t\tdevid: %02x:%02x.%x\n",
|
||||||
var, (int)handle,
|
var, (int)handle,
|
||||||
PCI_BUS(devid),
|
PCI_BUS_NUM(devid),
|
||||||
PCI_SLOT(devid),
|
PCI_SLOT(devid),
|
||||||
PCI_FUNC(devid));
|
PCI_FUNC(devid));
|
||||||
|
|
||||||
|
@ -1086,7 +1086,7 @@ static int __init init_iommu_all(struct acpi_table_header *table)
|
||||||
|
|
||||||
DUMP_printk("device: %02x:%02x.%01x cap: %04x "
|
DUMP_printk("device: %02x:%02x.%01x cap: %04x "
|
||||||
"seg: %d flags: %01x info %04x\n",
|
"seg: %d flags: %01x info %04x\n",
|
||||||
PCI_BUS(h->devid), PCI_SLOT(h->devid),
|
PCI_BUS_NUM(h->devid), PCI_SLOT(h->devid),
|
||||||
PCI_FUNC(h->devid), h->cap_ptr,
|
PCI_FUNC(h->devid), h->cap_ptr,
|
||||||
h->pci_seg, h->flags, h->info);
|
h->pci_seg, h->flags, h->info);
|
||||||
DUMP_printk(" mmio-addr: %016llx\n",
|
DUMP_printk(" mmio-addr: %016llx\n",
|
||||||
|
@ -1116,7 +1116,7 @@ static int iommu_init_pci(struct amd_iommu *iommu)
|
||||||
int cap_ptr = iommu->cap_ptr;
|
int cap_ptr = iommu->cap_ptr;
|
||||||
u32 range, misc, low, high;
|
u32 range, misc, low, high;
|
||||||
|
|
||||||
iommu->dev = pci_get_bus_and_slot(PCI_BUS(iommu->devid),
|
iommu->dev = pci_get_bus_and_slot(PCI_BUS_NUM(iommu->devid),
|
||||||
iommu->devid & 0xff);
|
iommu->devid & 0xff);
|
||||||
if (!iommu->dev)
|
if (!iommu->dev)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
@ -1128,9 +1128,9 @@ static int iommu_init_pci(struct amd_iommu *iommu)
|
||||||
pci_read_config_dword(iommu->dev, cap_ptr + MMIO_MISC_OFFSET,
|
pci_read_config_dword(iommu->dev, cap_ptr + MMIO_MISC_OFFSET,
|
||||||
&misc);
|
&misc);
|
||||||
|
|
||||||
iommu->first_device = calc_devid(MMIO_GET_BUS(range),
|
iommu->first_device = PCI_DEVID(MMIO_GET_BUS(range),
|
||||||
MMIO_GET_FD(range));
|
MMIO_GET_FD(range));
|
||||||
iommu->last_device = calc_devid(MMIO_GET_BUS(range),
|
iommu->last_device = PCI_DEVID(MMIO_GET_BUS(range),
|
||||||
MMIO_GET_LD(range));
|
MMIO_GET_LD(range));
|
||||||
|
|
||||||
if (!(iommu->cap & (1 << IOMMU_CAP_IOTLB)))
|
if (!(iommu->cap & (1 << IOMMU_CAP_IOTLB)))
|
||||||
|
@ -1388,8 +1388,8 @@ static int __init init_unity_map_range(struct ivmd_header *m)
|
||||||
|
|
||||||
DUMP_printk("%s devid_start: %02x:%02x.%x devid_end: %02x:%02x.%x"
|
DUMP_printk("%s devid_start: %02x:%02x.%x devid_end: %02x:%02x.%x"
|
||||||
" range_start: %016llx range_end: %016llx flags: %x\n", s,
|
" range_start: %016llx range_end: %016llx flags: %x\n", s,
|
||||||
PCI_BUS(e->devid_start), PCI_SLOT(e->devid_start),
|
PCI_BUS_NUM(e->devid_start), PCI_SLOT(e->devid_start),
|
||||||
PCI_FUNC(e->devid_start), PCI_BUS(e->devid_end),
|
PCI_FUNC(e->devid_start), PCI_BUS_NUM(e->devid_end),
|
||||||
PCI_SLOT(e->devid_end), PCI_FUNC(e->devid_end),
|
PCI_SLOT(e->devid_end), PCI_FUNC(e->devid_end),
|
||||||
e->address_start, e->address_end, m->flags);
|
e->address_start, e->address_end, m->flags);
|
||||||
|
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
#include <linux/list.h>
|
#include <linux/list.h>
|
||||||
#include <linux/spinlock.h>
|
#include <linux/spinlock.h>
|
||||||
|
#include <linux/pci.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Maximum number of IOMMUs supported
|
* Maximum number of IOMMUs supported
|
||||||
|
@ -315,9 +316,6 @@
|
||||||
|
|
||||||
#define MAX_DOMAIN_ID 65536
|
#define MAX_DOMAIN_ID 65536
|
||||||
|
|
||||||
/* FIXME: move this macro to <linux/pci.h> */
|
|
||||||
#define PCI_BUS(x) (((x) >> 8) & 0xff)
|
|
||||||
|
|
||||||
/* Protection domain flags */
|
/* Protection domain flags */
|
||||||
#define PD_DMA_OPS_MASK (1UL << 0) /* domain used for dma_ops */
|
#define PD_DMA_OPS_MASK (1UL << 0) /* domain used for dma_ops */
|
||||||
#define PD_DEFAULT_MASK (1UL << 1) /* domain is a default dma_ops
|
#define PD_DEFAULT_MASK (1UL << 1) /* domain is a default dma_ops
|
||||||
|
@ -703,13 +701,6 @@ extern int amd_iommu_max_glx_val;
|
||||||
*/
|
*/
|
||||||
extern void iommu_flush_all_caches(struct amd_iommu *iommu);
|
extern void iommu_flush_all_caches(struct amd_iommu *iommu);
|
||||||
|
|
||||||
/* takes bus and device/function and returns the device id
|
|
||||||
* FIXME: should that be in generic PCI code? */
|
|
||||||
static inline u16 calc_devid(u8 bus, u8 devfn)
|
|
||||||
{
|
|
||||||
return (((u16)bus) << 8) | devfn;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline int get_ioapic_devid(int id)
|
static inline int get_ioapic_devid(int id)
|
||||||
{
|
{
|
||||||
struct devid_map *entry;
|
struct devid_map *entry;
|
||||||
|
|
|
@ -202,20 +202,16 @@ void pci_bus_add_devices(const struct pci_bus *bus)
|
||||||
if (dev->is_added)
|
if (dev->is_added)
|
||||||
continue;
|
continue;
|
||||||
retval = pci_bus_add_device(dev);
|
retval = pci_bus_add_device(dev);
|
||||||
|
if (retval)
|
||||||
|
dev_err(&dev->dev, "Error adding device (%d)\n",
|
||||||
|
retval);
|
||||||
}
|
}
|
||||||
|
|
||||||
list_for_each_entry(dev, &bus->devices, bus_list) {
|
list_for_each_entry(dev, &bus->devices, bus_list) {
|
||||||
BUG_ON(!dev->is_added);
|
BUG_ON(!dev->is_added);
|
||||||
|
|
||||||
child = dev->subordinate;
|
child = dev->subordinate;
|
||||||
|
if (child)
|
||||||
if (!child)
|
pci_bus_add_devices(child);
|
||||||
continue;
|
|
||||||
pci_bus_add_devices(child);
|
|
||||||
|
|
||||||
if (child->is_added)
|
|
||||||
continue;
|
|
||||||
child->is_added = 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -52,15 +52,12 @@ config HOTPLUG_PCI_IBM
|
||||||
When in doubt, say N.
|
When in doubt, say N.
|
||||||
|
|
||||||
config HOTPLUG_PCI_ACPI
|
config HOTPLUG_PCI_ACPI
|
||||||
tristate "ACPI PCI Hotplug driver"
|
bool "ACPI PCI Hotplug driver"
|
||||||
depends on (!ACPI_DOCK && ACPI) || (ACPI_DOCK)
|
depends on HOTPLUG_PCI=y && ((!ACPI_DOCK && ACPI) || (ACPI_DOCK))
|
||||||
help
|
help
|
||||||
Say Y here if you have a system that supports PCI Hotplug using
|
Say Y here if you have a system that supports PCI Hotplug using
|
||||||
ACPI.
|
ACPI.
|
||||||
|
|
||||||
To compile this driver as a module, choose M here: the
|
|
||||||
module will be called acpiphp.
|
|
||||||
|
|
||||||
When in doubt, say N.
|
When in doubt, say N.
|
||||||
|
|
||||||
config HOTPLUG_PCI_ACPI_IBM
|
config HOTPLUG_PCI_ACPI_IBM
|
||||||
|
|
|
@ -73,8 +73,9 @@ static inline const char *slot_name(struct slot *slot)
|
||||||
*/
|
*/
|
||||||
struct acpiphp_bridge {
|
struct acpiphp_bridge {
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
|
struct list_head slots;
|
||||||
|
struct kref ref;
|
||||||
acpi_handle handle;
|
acpi_handle handle;
|
||||||
struct acpiphp_slot *slots;
|
|
||||||
|
|
||||||
/* Ejectable PCI-to-PCI bridge (PCI bridge and PCI function) */
|
/* Ejectable PCI-to-PCI bridge (PCI bridge and PCI function) */
|
||||||
struct acpiphp_func *func;
|
struct acpiphp_func *func;
|
||||||
|
@ -97,7 +98,7 @@ struct acpiphp_bridge {
|
||||||
* PCI slot information for each *physical* PCI slot
|
* PCI slot information for each *physical* PCI slot
|
||||||
*/
|
*/
|
||||||
struct acpiphp_slot {
|
struct acpiphp_slot {
|
||||||
struct acpiphp_slot *next;
|
struct list_head node;
|
||||||
struct acpiphp_bridge *bridge; /* parent */
|
struct acpiphp_bridge *bridge; /* parent */
|
||||||
struct list_head funcs; /* one slot may have different
|
struct list_head funcs; /* one slot may have different
|
||||||
objects (i.e. for each function) */
|
objects (i.e. for each function) */
|
||||||
|
@ -119,7 +120,6 @@ struct acpiphp_slot {
|
||||||
*/
|
*/
|
||||||
struct acpiphp_func {
|
struct acpiphp_func {
|
||||||
struct acpiphp_slot *slot; /* parent */
|
struct acpiphp_slot *slot; /* parent */
|
||||||
struct acpiphp_bridge *bridge; /* Ejectable PCI-to-PCI bridge */
|
|
||||||
|
|
||||||
struct list_head sibling;
|
struct list_head sibling;
|
||||||
struct notifier_block nb;
|
struct notifier_block nb;
|
||||||
|
@ -146,10 +146,6 @@ struct acpiphp_attention_info
|
||||||
#define ACPI_PCI_HOST_HID "PNP0A03"
|
#define ACPI_PCI_HOST_HID "PNP0A03"
|
||||||
|
|
||||||
/* ACPI _STA method value (ignore bit 4; battery present) */
|
/* ACPI _STA method value (ignore bit 4; battery present) */
|
||||||
#define ACPI_STA_PRESENT (0x00000001)
|
|
||||||
#define ACPI_STA_ENABLED (0x00000002)
|
|
||||||
#define ACPI_STA_SHOW_IN_UI (0x00000004)
|
|
||||||
#define ACPI_STA_FUNCTIONING (0x00000008)
|
|
||||||
#define ACPI_STA_ALL (0x0000000f)
|
#define ACPI_STA_ALL (0x0000000f)
|
||||||
|
|
||||||
/* bridge flags */
|
/* bridge flags */
|
||||||
|
@ -174,25 +170,24 @@ struct acpiphp_attention_info
|
||||||
/* function prototypes */
|
/* function prototypes */
|
||||||
|
|
||||||
/* acpiphp_core.c */
|
/* acpiphp_core.c */
|
||||||
extern int acpiphp_register_attention(struct acpiphp_attention_info*info);
|
int acpiphp_register_attention(struct acpiphp_attention_info*info);
|
||||||
extern int acpiphp_unregister_attention(struct acpiphp_attention_info *info);
|
int acpiphp_unregister_attention(struct acpiphp_attention_info *info);
|
||||||
extern int acpiphp_register_hotplug_slot(struct acpiphp_slot *slot);
|
int acpiphp_register_hotplug_slot(struct acpiphp_slot *slot);
|
||||||
extern void acpiphp_unregister_hotplug_slot(struct acpiphp_slot *slot);
|
void acpiphp_unregister_hotplug_slot(struct acpiphp_slot *slot);
|
||||||
|
|
||||||
/* acpiphp_glue.c */
|
/* acpiphp_glue.c */
|
||||||
extern int acpiphp_glue_init (void);
|
|
||||||
extern void acpiphp_glue_exit (void);
|
|
||||||
typedef int (*acpiphp_callback)(struct acpiphp_slot *slot, void *data);
|
typedef int (*acpiphp_callback)(struct acpiphp_slot *slot, void *data);
|
||||||
|
|
||||||
extern int acpiphp_enable_slot (struct acpiphp_slot *slot);
|
int acpiphp_enable_slot(struct acpiphp_slot *slot);
|
||||||
extern int acpiphp_disable_slot (struct acpiphp_slot *slot);
|
int acpiphp_disable_slot(struct acpiphp_slot *slot);
|
||||||
extern int acpiphp_eject_slot (struct acpiphp_slot *slot);
|
int acpiphp_eject_slot(struct acpiphp_slot *slot);
|
||||||
extern u8 acpiphp_get_power_status (struct acpiphp_slot *slot);
|
u8 acpiphp_get_power_status(struct acpiphp_slot *slot);
|
||||||
extern u8 acpiphp_get_attention_status (struct acpiphp_slot *slot);
|
u8 acpiphp_get_attention_status(struct acpiphp_slot *slot);
|
||||||
extern u8 acpiphp_get_latch_status (struct acpiphp_slot *slot);
|
u8 acpiphp_get_latch_status(struct acpiphp_slot *slot);
|
||||||
extern u8 acpiphp_get_adapter_status (struct acpiphp_slot *slot);
|
u8 acpiphp_get_adapter_status(struct acpiphp_slot *slot);
|
||||||
|
|
||||||
/* variables */
|
/* variables */
|
||||||
extern bool acpiphp_debug;
|
extern bool acpiphp_debug;
|
||||||
|
extern bool acpiphp_disabled;
|
||||||
|
|
||||||
#endif /* _ACPIPHP_H */
|
#endif /* _ACPIPHP_H */
|
||||||
|
|
|
@ -37,6 +37,7 @@
|
||||||
|
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
#include <linux/pci-acpi.h>
|
||||||
#include <linux/pci_hotplug.h>
|
#include <linux/pci_hotplug.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/smp.h>
|
#include <linux/smp.h>
|
||||||
|
@ -48,6 +49,7 @@
|
||||||
#define SLOT_NAME_SIZE 21 /* {_SUN} */
|
#define SLOT_NAME_SIZE 21 /* {_SUN} */
|
||||||
|
|
||||||
bool acpiphp_debug;
|
bool acpiphp_debug;
|
||||||
|
bool acpiphp_disabled;
|
||||||
|
|
||||||
/* local variables */
|
/* local variables */
|
||||||
static struct acpiphp_attention_info *attention_info;
|
static struct acpiphp_attention_info *attention_info;
|
||||||
|
@ -60,7 +62,9 @@ MODULE_AUTHOR(DRIVER_AUTHOR);
|
||||||
MODULE_DESCRIPTION(DRIVER_DESC);
|
MODULE_DESCRIPTION(DRIVER_DESC);
|
||||||
MODULE_LICENSE("GPL");
|
MODULE_LICENSE("GPL");
|
||||||
MODULE_PARM_DESC(debug, "Debugging mode enabled or not");
|
MODULE_PARM_DESC(debug, "Debugging mode enabled or not");
|
||||||
|
MODULE_PARM_DESC(disable, "disable acpiphp driver");
|
||||||
module_param_named(debug, acpiphp_debug, bool, 0644);
|
module_param_named(debug, acpiphp_debug, bool, 0644);
|
||||||
|
module_param_named(disable, acpiphp_disabled, bool, 0444);
|
||||||
|
|
||||||
/* export the attention callback registration methods */
|
/* export the attention callback registration methods */
|
||||||
EXPORT_SYMBOL_GPL(acpiphp_register_attention);
|
EXPORT_SYMBOL_GPL(acpiphp_register_attention);
|
||||||
|
@ -351,27 +355,9 @@ void acpiphp_unregister_hotplug_slot(struct acpiphp_slot *acpiphp_slot)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int __init acpiphp_init(void)
|
void __init acpiphp_init(void)
|
||||||
{
|
{
|
||||||
info(DRIVER_DESC " version: " DRIVER_VERSION "\n");
|
info(DRIVER_DESC " version: " DRIVER_VERSION "%s\n",
|
||||||
|
acpiphp_disabled ? ", disabled by user; please report a bug"
|
||||||
if (acpi_pci_disabled)
|
: "");
|
||||||
return 0;
|
|
||||||
|
|
||||||
/* read all the ACPI info from the system */
|
|
||||||
/* initialize internal data structure etc. */
|
|
||||||
return acpiphp_glue_init();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void __exit acpiphp_exit(void)
|
|
||||||
{
|
|
||||||
if (acpi_pci_disabled)
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* deallocate internal data structures etc. */
|
|
||||||
acpiphp_glue_exit();
|
|
||||||
}
|
|
||||||
|
|
||||||
module_init(acpiphp_init);
|
|
||||||
module_exit(acpiphp_exit);
|
|
||||||
|
|
|
@ -54,6 +54,7 @@
|
||||||
#include "acpiphp.h"
|
#include "acpiphp.h"
|
||||||
|
|
||||||
static LIST_HEAD(bridge_list);
|
static LIST_HEAD(bridge_list);
|
||||||
|
static DEFINE_MUTEX(bridge_mutex);
|
||||||
|
|
||||||
#define MY_NAME "acpiphp_glue"
|
#define MY_NAME "acpiphp_glue"
|
||||||
|
|
||||||
|
@ -61,6 +62,7 @@ static void handle_hotplug_event_bridge (acpi_handle, u32, void *);
|
||||||
static void acpiphp_sanitize_bus(struct pci_bus *bus);
|
static void acpiphp_sanitize_bus(struct pci_bus *bus);
|
||||||
static void acpiphp_set_hpp_values(struct pci_bus *bus);
|
static void acpiphp_set_hpp_values(struct pci_bus *bus);
|
||||||
static void handle_hotplug_event_func(acpi_handle handle, u32 type, void *context);
|
static void handle_hotplug_event_func(acpi_handle handle, u32 type, void *context);
|
||||||
|
static void free_bridge(struct kref *kref);
|
||||||
|
|
||||||
/* callback routine to check for the existence of a pci dock device */
|
/* callback routine to check for the existence of a pci dock device */
|
||||||
static acpi_status
|
static acpi_status
|
||||||
|
@ -76,6 +78,39 @@ is_pci_dock_device(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void get_bridge(struct acpiphp_bridge *bridge)
|
||||||
|
{
|
||||||
|
kref_get(&bridge->ref);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void put_bridge(struct acpiphp_bridge *bridge)
|
||||||
|
{
|
||||||
|
kref_put(&bridge->ref, free_bridge);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void free_bridge(struct kref *kref)
|
||||||
|
{
|
||||||
|
struct acpiphp_bridge *bridge;
|
||||||
|
struct acpiphp_slot *slot, *next;
|
||||||
|
struct acpiphp_func *func, *tmp;
|
||||||
|
|
||||||
|
bridge = container_of(kref, struct acpiphp_bridge, ref);
|
||||||
|
|
||||||
|
list_for_each_entry_safe(slot, next, &bridge->slots, node) {
|
||||||
|
list_for_each_entry_safe(func, tmp, &slot->funcs, sibling) {
|
||||||
|
kfree(func);
|
||||||
|
}
|
||||||
|
kfree(slot);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Release reference acquired by acpiphp_bridge_handle_to_function() */
|
||||||
|
if ((bridge->flags & BRIDGE_HAS_EJ0) && bridge->func)
|
||||||
|
put_bridge(bridge->func->slot->bridge);
|
||||||
|
put_device(&bridge->pci_bus->dev);
|
||||||
|
pci_dev_put(bridge->pci_dev);
|
||||||
|
kfree(bridge);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* the _DCK method can do funny things... and sometimes not
|
* the _DCK method can do funny things... and sometimes not
|
||||||
* hah-hah funny.
|
* hah-hah funny.
|
||||||
|
@ -154,9 +189,10 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
acpi_handle tmp;
|
acpi_handle tmp;
|
||||||
acpi_status status = AE_OK;
|
acpi_status status = AE_OK;
|
||||||
unsigned long long adr, sun;
|
unsigned long long adr, sun;
|
||||||
int device, function, retval;
|
int device, function, retval, found = 0;
|
||||||
struct pci_bus *pbus = bridge->pci_bus;
|
struct pci_bus *pbus = bridge->pci_bus;
|
||||||
struct pci_dev *pdev;
|
struct pci_dev *pdev;
|
||||||
|
u32 val;
|
||||||
|
|
||||||
if (!acpi_pci_check_ejectable(pbus, handle) && !is_dock_device(handle))
|
if (!acpi_pci_check_ejectable(pbus, handle) && !is_dock_device(handle))
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
|
@ -170,7 +206,7 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
device = (adr >> 16) & 0xffff;
|
device = (adr >> 16) & 0xffff;
|
||||||
function = adr & 0xffff;
|
function = adr & 0xffff;
|
||||||
|
|
||||||
pdev = pbus->self;
|
pdev = bridge->pci_dev;
|
||||||
if (pdev && device_is_managed_by_native_pciehp(pdev))
|
if (pdev && device_is_managed_by_native_pciehp(pdev))
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
|
|
||||||
|
@ -178,7 +214,6 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
if (!newfunc)
|
if (!newfunc)
|
||||||
return AE_NO_MEMORY;
|
return AE_NO_MEMORY;
|
||||||
|
|
||||||
INIT_LIST_HEAD(&newfunc->sibling);
|
|
||||||
newfunc->handle = handle;
|
newfunc->handle = handle;
|
||||||
newfunc->function = function;
|
newfunc->function = function;
|
||||||
|
|
||||||
|
@ -207,14 +242,15 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* search for objects that share the same slot */
|
/* search for objects that share the same slot */
|
||||||
for (slot = bridge->slots; slot; slot = slot->next)
|
list_for_each_entry(slot, &bridge->slots, node)
|
||||||
if (slot->device == device) {
|
if (slot->device == device) {
|
||||||
if (slot->sun != sun)
|
if (slot->sun != sun)
|
||||||
warn("sibling found, but _SUN doesn't match!\n");
|
warn("sibling found, but _SUN doesn't match!\n");
|
||||||
|
found = 1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!slot) {
|
if (!found) {
|
||||||
slot = kzalloc(sizeof(struct acpiphp_slot), GFP_KERNEL);
|
slot = kzalloc(sizeof(struct acpiphp_slot), GFP_KERNEL);
|
||||||
if (!slot) {
|
if (!slot) {
|
||||||
kfree(newfunc);
|
kfree(newfunc);
|
||||||
|
@ -227,9 +263,9 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
INIT_LIST_HEAD(&slot->funcs);
|
INIT_LIST_HEAD(&slot->funcs);
|
||||||
mutex_init(&slot->crit_sect);
|
mutex_init(&slot->crit_sect);
|
||||||
|
|
||||||
slot->next = bridge->slots;
|
mutex_lock(&bridge_mutex);
|
||||||
bridge->slots = slot;
|
list_add_tail(&slot->node, &bridge->slots);
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
bridge->nr_slots++;
|
bridge->nr_slots++;
|
||||||
|
|
||||||
dbg("found ACPI PCI Hotplug slot %llu at PCI %04x:%02x:%02x\n",
|
dbg("found ACPI PCI Hotplug slot %llu at PCI %04x:%02x:%02x\n",
|
||||||
|
@ -247,13 +283,13 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
}
|
}
|
||||||
|
|
||||||
newfunc->slot = slot;
|
newfunc->slot = slot;
|
||||||
|
mutex_lock(&bridge_mutex);
|
||||||
list_add_tail(&newfunc->sibling, &slot->funcs);
|
list_add_tail(&newfunc->sibling, &slot->funcs);
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
|
|
||||||
pdev = pci_get_slot(pbus, PCI_DEVFN(device, function));
|
if (pci_bus_read_dev_vendor_id(pbus, PCI_DEVFN(device, function),
|
||||||
if (pdev) {
|
&val, 60*1000))
|
||||||
slot->flags |= (SLOT_ENABLED | SLOT_POWEREDON);
|
slot->flags |= (SLOT_ENABLED | SLOT_POWEREDON);
|
||||||
pci_dev_put(pdev);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_dock_device(handle)) {
|
if (is_dock_device(handle)) {
|
||||||
/* we don't want to call this device's _EJ0
|
/* we don't want to call this device's _EJ0
|
||||||
|
@ -290,7 +326,9 @@ register_slot(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
|
|
||||||
err_exit:
|
err_exit:
|
||||||
bridge->nr_slots--;
|
bridge->nr_slots--;
|
||||||
bridge->slots = slot->next;
|
mutex_lock(&bridge_mutex);
|
||||||
|
list_del(&slot->node);
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
kfree(slot);
|
kfree(slot);
|
||||||
kfree(newfunc);
|
kfree(newfunc);
|
||||||
|
|
||||||
|
@ -315,13 +353,17 @@ static void init_bridge_misc(struct acpiphp_bridge *bridge)
|
||||||
acpi_status status;
|
acpi_status status;
|
||||||
|
|
||||||
/* must be added to the list prior to calling register_slot */
|
/* must be added to the list prior to calling register_slot */
|
||||||
|
mutex_lock(&bridge_mutex);
|
||||||
list_add(&bridge->list, &bridge_list);
|
list_add(&bridge->list, &bridge_list);
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
|
|
||||||
/* register all slot objects under this bridge */
|
/* register all slot objects under this bridge */
|
||||||
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, bridge->handle, (u32)1,
|
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, bridge->handle, (u32)1,
|
||||||
register_slot, NULL, bridge, NULL);
|
register_slot, NULL, bridge, NULL);
|
||||||
if (ACPI_FAILURE(status)) {
|
if (ACPI_FAILURE(status)) {
|
||||||
|
mutex_lock(&bridge_mutex);
|
||||||
list_del(&bridge->list);
|
list_del(&bridge->list);
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -351,178 +393,46 @@ static struct acpiphp_func *acpiphp_bridge_handle_to_function(acpi_handle handle
|
||||||
{
|
{
|
||||||
struct acpiphp_bridge *bridge;
|
struct acpiphp_bridge *bridge;
|
||||||
struct acpiphp_slot *slot;
|
struct acpiphp_slot *slot;
|
||||||
struct acpiphp_func *func;
|
struct acpiphp_func *func = NULL;
|
||||||
|
|
||||||
|
mutex_lock(&bridge_mutex);
|
||||||
list_for_each_entry(bridge, &bridge_list, list) {
|
list_for_each_entry(bridge, &bridge_list, list) {
|
||||||
for (slot = bridge->slots; slot; slot = slot->next) {
|
list_for_each_entry(slot, &bridge->slots, node) {
|
||||||
list_for_each_entry(func, &slot->funcs, sibling) {
|
list_for_each_entry(func, &slot->funcs, sibling) {
|
||||||
if (func->handle == handle)
|
if (func->handle == handle) {
|
||||||
|
get_bridge(func->slot->bridge);
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
return func;
|
return func;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static inline void config_p2p_bridge_flags(struct acpiphp_bridge *bridge)
|
|
||||||
{
|
|
||||||
acpi_handle dummy_handle;
|
|
||||||
struct acpiphp_func *func;
|
|
||||||
|
|
||||||
if (ACPI_SUCCESS(acpi_get_handle(bridge->handle,
|
|
||||||
"_EJ0", &dummy_handle))) {
|
|
||||||
bridge->flags |= BRIDGE_HAS_EJ0;
|
|
||||||
|
|
||||||
dbg("found ejectable p2p bridge\n");
|
|
||||||
|
|
||||||
/* make link between PCI bridge and PCI function */
|
|
||||||
func = acpiphp_bridge_handle_to_function(bridge->handle);
|
|
||||||
if (!func)
|
|
||||||
return;
|
|
||||||
bridge->func = func;
|
|
||||||
func->bridge = bridge;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* allocate and initialize host bridge data structure */
|
|
||||||
static void add_host_bridge(struct acpi_pci_root *root)
|
|
||||||
{
|
|
||||||
struct acpiphp_bridge *bridge;
|
|
||||||
acpi_handle handle = root->device->handle;
|
|
||||||
|
|
||||||
bridge = kzalloc(sizeof(struct acpiphp_bridge), GFP_KERNEL);
|
|
||||||
if (bridge == NULL)
|
|
||||||
return;
|
|
||||||
|
|
||||||
bridge->handle = handle;
|
|
||||||
|
|
||||||
bridge->pci_bus = root->bus;
|
|
||||||
|
|
||||||
init_bridge_misc(bridge);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* allocate and initialize PCI-to-PCI bridge data structure */
|
|
||||||
static void add_p2p_bridge(acpi_handle *handle)
|
|
||||||
{
|
|
||||||
struct acpiphp_bridge *bridge;
|
|
||||||
|
|
||||||
bridge = kzalloc(sizeof(struct acpiphp_bridge), GFP_KERNEL);
|
|
||||||
if (bridge == NULL) {
|
|
||||||
err("out of memory\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
bridge->handle = handle;
|
|
||||||
config_p2p_bridge_flags(bridge);
|
|
||||||
|
|
||||||
bridge->pci_dev = acpi_get_pci_dev(handle);
|
|
||||||
bridge->pci_bus = bridge->pci_dev->subordinate;
|
|
||||||
if (!bridge->pci_bus) {
|
|
||||||
err("This is not a PCI-to-PCI bridge!\n");
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Grab a ref to the subordinate PCI bus in case the bus is
|
|
||||||
* removed via PCI core logical hotplug. The ref pins the bus
|
|
||||||
* (which we access during module unload).
|
|
||||||
*/
|
|
||||||
get_device(&bridge->pci_bus->dev);
|
|
||||||
|
|
||||||
init_bridge_misc(bridge);
|
|
||||||
return;
|
|
||||||
err:
|
|
||||||
pci_dev_put(bridge->pci_dev);
|
|
||||||
kfree(bridge);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* callback routine to find P2P bridges */
|
|
||||||
static acpi_status
|
|
||||||
find_p2p_bridge(acpi_handle handle, u32 lvl, void *context, void **rv)
|
|
||||||
{
|
|
||||||
acpi_status status;
|
|
||||||
struct pci_dev *dev;
|
|
||||||
|
|
||||||
dev = acpi_get_pci_dev(handle);
|
|
||||||
if (!dev || !dev->subordinate)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
/* check if this bridge has ejectable slots */
|
|
||||||
if ((detect_ejectable_slots(handle) > 0)) {
|
|
||||||
dbg("found PCI-to-PCI bridge at PCI %s\n", pci_name(dev));
|
|
||||||
add_p2p_bridge(handle);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* search P2P bridges under this p2p bridge */
|
|
||||||
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, (u32)1,
|
|
||||||
find_p2p_bridge, NULL, NULL, NULL);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
warn("find_p2p_bridge failed (error code = 0x%x)\n", status);
|
|
||||||
|
|
||||||
out:
|
|
||||||
pci_dev_put(dev);
|
|
||||||
return AE_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* find hot-pluggable slots, and then find P2P bridge */
|
|
||||||
static int add_bridge(struct acpi_pci_root *root)
|
|
||||||
{
|
|
||||||
acpi_status status;
|
|
||||||
unsigned long long tmp;
|
|
||||||
acpi_handle dummy_handle;
|
|
||||||
acpi_handle handle = root->device->handle;
|
|
||||||
|
|
||||||
/* if the bridge doesn't have _STA, we assume it is always there */
|
|
||||||
status = acpi_get_handle(handle, "_STA", &dummy_handle);
|
|
||||||
if (ACPI_SUCCESS(status)) {
|
|
||||||
status = acpi_evaluate_integer(handle, "_STA", NULL, &tmp);
|
|
||||||
if (ACPI_FAILURE(status)) {
|
|
||||||
dbg("%s: _STA evaluation failure\n", __func__);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
if ((tmp & ACPI_STA_FUNCTIONING) == 0)
|
|
||||||
/* don't register this object */
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* check if this bridge has ejectable slots */
|
|
||||||
if (detect_ejectable_slots(handle) > 0) {
|
|
||||||
dbg("found PCI host-bus bridge with hot-pluggable slots\n");
|
|
||||||
add_host_bridge(root);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* search P2P bridges under this host bridge */
|
|
||||||
status = acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, (u32)1,
|
|
||||||
find_p2p_bridge, NULL, NULL, NULL);
|
|
||||||
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
warn("find_p2p_bridge failed (error code = 0x%x)\n", status);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct acpiphp_bridge *acpiphp_handle_to_bridge(acpi_handle handle)
|
static struct acpiphp_bridge *acpiphp_handle_to_bridge(acpi_handle handle)
|
||||||
{
|
{
|
||||||
struct acpiphp_bridge *bridge;
|
struct acpiphp_bridge *bridge;
|
||||||
|
|
||||||
|
mutex_lock(&bridge_mutex);
|
||||||
list_for_each_entry(bridge, &bridge_list, list)
|
list_for_each_entry(bridge, &bridge_list, list)
|
||||||
if (bridge->handle == handle)
|
if (bridge->handle == handle) {
|
||||||
|
get_bridge(bridge);
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
return bridge;
|
return bridge;
|
||||||
|
}
|
||||||
|
mutex_unlock(&bridge_mutex);
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void cleanup_bridge(struct acpiphp_bridge *bridge)
|
static void cleanup_bridge(struct acpiphp_bridge *bridge)
|
||||||
{
|
{
|
||||||
struct acpiphp_slot *slot, *next;
|
struct acpiphp_slot *slot;
|
||||||
struct acpiphp_func *func, *tmp;
|
struct acpiphp_func *func;
|
||||||
acpi_status status;
|
acpi_status status;
|
||||||
acpi_handle handle = bridge->handle;
|
acpi_handle handle = bridge->handle;
|
||||||
|
|
||||||
|
@ -543,10 +453,8 @@ static void cleanup_bridge(struct acpiphp_bridge *bridge)
|
||||||
err("failed to install interrupt notify handler\n");
|
err("failed to install interrupt notify handler\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
slot = bridge->slots;
|
list_for_each_entry(slot, &bridge->slots, node) {
|
||||||
while (slot) {
|
list_for_each_entry(func, &slot->funcs, sibling) {
|
||||||
next = slot->next;
|
|
||||||
list_for_each_entry_safe(func, tmp, &slot->funcs, sibling) {
|
|
||||||
if (is_dock_device(func->handle)) {
|
if (is_dock_device(func->handle)) {
|
||||||
unregister_hotplug_dock_device(func->handle);
|
unregister_hotplug_dock_device(func->handle);
|
||||||
unregister_dock_notifier(&func->nb);
|
unregister_dock_notifier(&func->nb);
|
||||||
|
@ -558,63 +466,13 @@ static void cleanup_bridge(struct acpiphp_bridge *bridge)
|
||||||
if (ACPI_FAILURE(status))
|
if (ACPI_FAILURE(status))
|
||||||
err("failed to remove notify handler\n");
|
err("failed to remove notify handler\n");
|
||||||
}
|
}
|
||||||
list_del(&func->sibling);
|
|
||||||
kfree(func);
|
|
||||||
}
|
}
|
||||||
acpiphp_unregister_hotplug_slot(slot);
|
acpiphp_unregister_hotplug_slot(slot);
|
||||||
list_del(&slot->funcs);
|
|
||||||
kfree(slot);
|
|
||||||
slot = next;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
mutex_lock(&bridge_mutex);
|
||||||
* Only P2P bridges have a pci_dev
|
|
||||||
*/
|
|
||||||
if (bridge->pci_dev)
|
|
||||||
put_device(&bridge->pci_bus->dev);
|
|
||||||
|
|
||||||
pci_dev_put(bridge->pci_dev);
|
|
||||||
list_del(&bridge->list);
|
list_del(&bridge->list);
|
||||||
kfree(bridge);
|
mutex_unlock(&bridge_mutex);
|
||||||
}
|
|
||||||
|
|
||||||
static acpi_status
|
|
||||||
cleanup_p2p_bridge(acpi_handle handle, u32 lvl, void *context, void **rv)
|
|
||||||
{
|
|
||||||
struct acpiphp_bridge *bridge;
|
|
||||||
|
|
||||||
/* cleanup p2p bridges under this P2P bridge
|
|
||||||
in a depth-first manner */
|
|
||||||
acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, (u32)1,
|
|
||||||
cleanup_p2p_bridge, NULL, NULL, NULL);
|
|
||||||
|
|
||||||
bridge = acpiphp_handle_to_bridge(handle);
|
|
||||||
if (bridge)
|
|
||||||
cleanup_bridge(bridge);
|
|
||||||
|
|
||||||
return AE_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void remove_bridge(struct acpi_pci_root *root)
|
|
||||||
{
|
|
||||||
struct acpiphp_bridge *bridge;
|
|
||||||
acpi_handle handle = root->device->handle;
|
|
||||||
|
|
||||||
/* cleanup p2p bridges under this host bridge
|
|
||||||
in a depth-first manner */
|
|
||||||
acpi_walk_namespace(ACPI_TYPE_DEVICE, handle,
|
|
||||||
(u32)1, cleanup_p2p_bridge, NULL, NULL, NULL);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* On root bridges with hotplug slots directly underneath (ie,
|
|
||||||
* no p2p bridge between), we call cleanup_bridge().
|
|
||||||
*
|
|
||||||
* The else clause cleans up root bridges that either had no
|
|
||||||
* hotplug slots at all, or had a p2p bridge underneath.
|
|
||||||
*/
|
|
||||||
bridge = acpiphp_handle_to_bridge(handle);
|
|
||||||
if (bridge)
|
|
||||||
cleanup_bridge(bridge);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int power_on_slot(struct acpiphp_slot *slot)
|
static int power_on_slot(struct acpiphp_slot *slot)
|
||||||
|
@ -798,6 +656,7 @@ static void check_hotplug_bridge(struct acpiphp_slot *slot, struct pci_dev *dev)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* enable_device - enable, configure a slot
|
* enable_device - enable, configure a slot
|
||||||
* @slot: slot to be enabled
|
* @slot: slot to be enabled
|
||||||
|
@ -810,9 +669,7 @@ static int __ref enable_device(struct acpiphp_slot *slot)
|
||||||
struct pci_dev *dev;
|
struct pci_dev *dev;
|
||||||
struct pci_bus *bus = slot->bridge->pci_bus;
|
struct pci_bus *bus = slot->bridge->pci_bus;
|
||||||
struct acpiphp_func *func;
|
struct acpiphp_func *func;
|
||||||
int retval = 0;
|
|
||||||
int num, max, pass;
|
int num, max, pass;
|
||||||
acpi_status status;
|
|
||||||
|
|
||||||
if (slot->flags & SLOT_ENABLED)
|
if (slot->flags & SLOT_ENABLED)
|
||||||
goto err_exit;
|
goto err_exit;
|
||||||
|
@ -867,23 +724,11 @@ static int __ref enable_device(struct acpiphp_slot *slot)
|
||||||
slot->flags &= (~SLOT_ENABLED);
|
slot->flags &= (~SLOT_ENABLED);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dev->hdr_type != PCI_HEADER_TYPE_BRIDGE &&
|
|
||||||
dev->hdr_type != PCI_HEADER_TYPE_CARDBUS) {
|
|
||||||
pci_dev_put(dev);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
status = find_p2p_bridge(func->handle, (u32)1, bus, NULL);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
warn("find_p2p_bridge failed (error code = 0x%x)\n",
|
|
||||||
status);
|
|
||||||
pci_dev_put(dev);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
err_exit:
|
err_exit:
|
||||||
return retval;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* return first device in slot, acquiring a reference on it */
|
/* return first device in slot, acquiring a reference on it */
|
||||||
|
@ -912,23 +757,6 @@ static int disable_device(struct acpiphp_slot *slot)
|
||||||
{
|
{
|
||||||
struct acpiphp_func *func;
|
struct acpiphp_func *func;
|
||||||
struct pci_dev *pdev;
|
struct pci_dev *pdev;
|
||||||
struct pci_bus *bus = slot->bridge->pci_bus;
|
|
||||||
|
|
||||||
/* The slot will be enabled when func 0 is added, so check
|
|
||||||
func 0 before disable the slot. */
|
|
||||||
pdev = pci_get_slot(bus, PCI_DEVFN(slot->device, 0));
|
|
||||||
if (!pdev)
|
|
||||||
goto err_exit;
|
|
||||||
pci_dev_put(pdev);
|
|
||||||
|
|
||||||
list_for_each_entry(func, &slot->funcs, sibling) {
|
|
||||||
if (func->bridge) {
|
|
||||||
/* cleanup p2p bridges under this P2P bridge */
|
|
||||||
cleanup_p2p_bridge(func->bridge->handle,
|
|
||||||
(u32)1, NULL, NULL);
|
|
||||||
func->bridge = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* enable_device() enumerates all functions in this device via
|
* enable_device() enumerates all functions in this device via
|
||||||
|
@ -947,7 +775,6 @@ static int disable_device(struct acpiphp_slot *slot)
|
||||||
|
|
||||||
slot->flags &= (~SLOT_ENABLED);
|
slot->flags &= (~SLOT_ENABLED);
|
||||||
|
|
||||||
err_exit:
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1037,7 +864,7 @@ static int acpiphp_check_bridge(struct acpiphp_bridge *bridge)
|
||||||
|
|
||||||
enabled = disabled = 0;
|
enabled = disabled = 0;
|
||||||
|
|
||||||
for (slot = bridge->slots; slot; slot = slot->next) {
|
list_for_each_entry(slot, &bridge->slots, node) {
|
||||||
unsigned int status = get_slot_status(slot);
|
unsigned int status = get_slot_status(slot);
|
||||||
if (slot->flags & SLOT_ENABLED) {
|
if (slot->flags & SLOT_ENABLED) {
|
||||||
if (status == ACPI_STA_ALL)
|
if (status == ACPI_STA_ALL)
|
||||||
|
@ -1082,11 +909,11 @@ static void acpiphp_set_hpp_values(struct pci_bus *bus)
|
||||||
*/
|
*/
|
||||||
static void acpiphp_sanitize_bus(struct pci_bus *bus)
|
static void acpiphp_sanitize_bus(struct pci_bus *bus)
|
||||||
{
|
{
|
||||||
struct pci_dev *dev;
|
struct pci_dev *dev, *tmp;
|
||||||
int i;
|
int i;
|
||||||
unsigned long type_mask = IORESOURCE_IO | IORESOURCE_MEM;
|
unsigned long type_mask = IORESOURCE_IO | IORESOURCE_MEM;
|
||||||
|
|
||||||
list_for_each_entry(dev, &bus->devices, bus_list) {
|
list_for_each_entry_safe(dev, tmp, &bus->devices, bus_list) {
|
||||||
for (i=0; i<PCI_BRIDGE_RESOURCES; i++) {
|
for (i=0; i<PCI_BRIDGE_RESOURCES; i++) {
|
||||||
struct resource *res = &dev->resource[i];
|
struct resource *res = &dev->resource[i];
|
||||||
if ((res->flags & type_mask) && !res->start &&
|
if ((res->flags & type_mask) && !res->start &&
|
||||||
|
@ -1118,6 +945,7 @@ check_sub_bridges(acpi_handle handle, u32 lvl, void *context, void **rv)
|
||||||
dbg("%s: re-enumerating slots under %s\n",
|
dbg("%s: re-enumerating slots under %s\n",
|
||||||
__func__, objname);
|
__func__, objname);
|
||||||
acpiphp_check_bridge(bridge);
|
acpiphp_check_bridge(bridge);
|
||||||
|
put_bridge(bridge);
|
||||||
}
|
}
|
||||||
return AE_OK ;
|
return AE_OK ;
|
||||||
}
|
}
|
||||||
|
@ -1195,6 +1023,7 @@ static void _handle_hotplug_event_bridge(struct work_struct *work)
|
||||||
|
|
||||||
acpi_scan_lock_release();
|
acpi_scan_lock_release();
|
||||||
kfree(hp_work); /* allocated in handle_hotplug_event_bridge */
|
kfree(hp_work); /* allocated in handle_hotplug_event_bridge */
|
||||||
|
put_bridge(bridge);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1208,6 +1037,8 @@ static void _handle_hotplug_event_bridge(struct work_struct *work)
|
||||||
static void handle_hotplug_event_bridge(acpi_handle handle, u32 type,
|
static void handle_hotplug_event_bridge(acpi_handle handle, u32 type,
|
||||||
void *context)
|
void *context)
|
||||||
{
|
{
|
||||||
|
struct acpiphp_bridge *bridge = context;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Currently the code adds all hotplug events to the kacpid_wq
|
* Currently the code adds all hotplug events to the kacpid_wq
|
||||||
* queue when it should add hotplug events to the kacpi_hotplug_wq.
|
* queue when it should add hotplug events to the kacpi_hotplug_wq.
|
||||||
|
@ -1216,6 +1047,7 @@ static void handle_hotplug_event_bridge(acpi_handle handle, u32 type,
|
||||||
* For now just re-add this work to the kacpi_hotplug_wq so we
|
* For now just re-add this work to the kacpi_hotplug_wq so we
|
||||||
* don't deadlock on hotplug actions.
|
* don't deadlock on hotplug actions.
|
||||||
*/
|
*/
|
||||||
|
get_bridge(bridge);
|
||||||
alloc_acpi_hp_work(handle, type, context, _handle_hotplug_event_bridge);
|
alloc_acpi_hp_work(handle, type, context, _handle_hotplug_event_bridge);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1270,6 +1102,7 @@ static void _handle_hotplug_event_func(struct work_struct *work)
|
||||||
|
|
||||||
acpi_scan_lock_release();
|
acpi_scan_lock_release();
|
||||||
kfree(hp_work); /* allocated in handle_hotplug_event_func */
|
kfree(hp_work); /* allocated in handle_hotplug_event_func */
|
||||||
|
put_bridge(func->slot->bridge);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1283,6 +1116,8 @@ static void _handle_hotplug_event_func(struct work_struct *work)
|
||||||
static void handle_hotplug_event_func(acpi_handle handle, u32 type,
|
static void handle_hotplug_event_func(acpi_handle handle, u32 type,
|
||||||
void *context)
|
void *context)
|
||||||
{
|
{
|
||||||
|
struct acpiphp_func *func = context;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Currently the code adds all hotplug events to the kacpid_wq
|
* Currently the code adds all hotplug events to the kacpid_wq
|
||||||
* queue when it should add hotplug events to the kacpi_hotplug_wq.
|
* queue when it should add hotplug events to the kacpi_hotplug_wq.
|
||||||
|
@ -1291,33 +1126,69 @@ static void handle_hotplug_event_func(acpi_handle handle, u32 type,
|
||||||
* For now just re-add this work to the kacpi_hotplug_wq so we
|
* For now just re-add this work to the kacpi_hotplug_wq so we
|
||||||
* don't deadlock on hotplug actions.
|
* don't deadlock on hotplug actions.
|
||||||
*/
|
*/
|
||||||
|
get_bridge(func->slot->bridge);
|
||||||
alloc_acpi_hp_work(handle, type, context, _handle_hotplug_event_func);
|
alloc_acpi_hp_work(handle, type, context, _handle_hotplug_event_func);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct acpi_pci_driver acpi_pci_hp_driver = {
|
/*
|
||||||
.add = add_bridge,
|
* Create hotplug slots for the PCI bus.
|
||||||
.remove = remove_bridge,
|
* It should always return 0 to avoid skipping following notifiers.
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* acpiphp_glue_init - initializes all PCI hotplug - ACPI glue data structures
|
|
||||||
*/
|
*/
|
||||||
int __init acpiphp_glue_init(void)
|
void acpiphp_enumerate_slots(struct pci_bus *bus, acpi_handle handle)
|
||||||
{
|
{
|
||||||
acpi_pci_register_driver(&acpi_pci_hp_driver);
|
acpi_handle dummy_handle;
|
||||||
|
struct acpiphp_bridge *bridge;
|
||||||
|
|
||||||
return 0;
|
if (acpiphp_disabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (detect_ejectable_slots(handle) <= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
bridge = kzalloc(sizeof(struct acpiphp_bridge), GFP_KERNEL);
|
||||||
|
if (bridge == NULL) {
|
||||||
|
err("out of memory\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
INIT_LIST_HEAD(&bridge->slots);
|
||||||
|
kref_init(&bridge->ref);
|
||||||
|
bridge->handle = handle;
|
||||||
|
bridge->pci_dev = pci_dev_get(bus->self);
|
||||||
|
bridge->pci_bus = bus;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Grab a ref to the subordinate PCI bus in case the bus is
|
||||||
|
* removed via PCI core logical hotplug. The ref pins the bus
|
||||||
|
* (which we access during module unload).
|
||||||
|
*/
|
||||||
|
get_device(&bus->dev);
|
||||||
|
|
||||||
|
if (!pci_is_root_bus(bridge->pci_bus) &&
|
||||||
|
ACPI_SUCCESS(acpi_get_handle(bridge->handle,
|
||||||
|
"_EJ0", &dummy_handle))) {
|
||||||
|
dbg("found ejectable p2p bridge\n");
|
||||||
|
bridge->flags |= BRIDGE_HAS_EJ0;
|
||||||
|
bridge->func = acpiphp_bridge_handle_to_function(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
init_bridge_misc(bridge);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Destroy hotplug slots associated with the PCI bus */
|
||||||
/**
|
void acpiphp_remove_slots(struct pci_bus *bus)
|
||||||
* acpiphp_glue_exit - terminates all PCI hotplug - ACPI glue data structures
|
|
||||||
*
|
|
||||||
* This function frees all data allocated in acpiphp_glue_init().
|
|
||||||
*/
|
|
||||||
void acpiphp_glue_exit(void)
|
|
||||||
{
|
{
|
||||||
acpi_pci_unregister_driver(&acpi_pci_hp_driver);
|
struct acpiphp_bridge *bridge, *tmp;
|
||||||
|
|
||||||
|
if (acpiphp_disabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
list_for_each_entry_safe(bridge, tmp, &bridge_list, list)
|
||||||
|
if (bridge->pci_bus == bus) {
|
||||||
|
cleanup_bridge(bridge);
|
||||||
|
put_bridge(bridge);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1396,7 +1267,7 @@ u8 acpiphp_get_latch_status(struct acpiphp_slot *slot)
|
||||||
|
|
||||||
sta = get_slot_status(slot);
|
sta = get_slot_status(slot);
|
||||||
|
|
||||||
return (sta & ACPI_STA_SHOW_IN_UI) ? 0 : 1;
|
return (sta & ACPI_STA_DEVICE_UI) ? 0 : 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -75,28 +75,36 @@ static inline const char *slot_name(struct slot *slot)
|
||||||
return hotplug_slot_name(slot->hotplug_slot);
|
return hotplug_slot_name(slot->hotplug_slot);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern int cpci_hp_register_controller(struct cpci_hp_controller *controller);
|
int cpci_hp_register_controller(struct cpci_hp_controller *controller);
|
||||||
extern int cpci_hp_unregister_controller(struct cpci_hp_controller *controller);
|
int cpci_hp_unregister_controller(struct cpci_hp_controller *controller);
|
||||||
extern int cpci_hp_register_bus(struct pci_bus *bus, u8 first, u8 last);
|
int cpci_hp_register_bus(struct pci_bus *bus, u8 first, u8 last);
|
||||||
extern int cpci_hp_unregister_bus(struct pci_bus *bus);
|
int cpci_hp_unregister_bus(struct pci_bus *bus);
|
||||||
extern int cpci_hp_start(void);
|
int cpci_hp_start(void);
|
||||||
extern int cpci_hp_stop(void);
|
int cpci_hp_stop(void);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Internal function prototypes, these functions should not be used by
|
* Internal function prototypes, these functions should not be used by
|
||||||
* board/chassis drivers.
|
* board/chassis drivers.
|
||||||
*/
|
*/
|
||||||
extern u8 cpci_get_attention_status(struct slot *slot);
|
u8 cpci_get_attention_status(struct slot *slot);
|
||||||
extern u8 cpci_get_latch_status(struct slot *slot);
|
u8 cpci_get_latch_status(struct slot *slot);
|
||||||
extern u8 cpci_get_adapter_status(struct slot *slot);
|
u8 cpci_get_adapter_status(struct slot *slot);
|
||||||
extern u16 cpci_get_hs_csr(struct slot * slot);
|
u16 cpci_get_hs_csr(struct slot * slot);
|
||||||
extern int cpci_set_attention_status(struct slot *slot, int status);
|
int cpci_set_attention_status(struct slot *slot, int status);
|
||||||
extern int cpci_check_and_clear_ins(struct slot * slot);
|
int cpci_check_and_clear_ins(struct slot * slot);
|
||||||
extern int cpci_check_ext(struct slot * slot);
|
int cpci_check_ext(struct slot * slot);
|
||||||
extern int cpci_clear_ext(struct slot * slot);
|
int cpci_clear_ext(struct slot * slot);
|
||||||
extern int cpci_led_on(struct slot * slot);
|
int cpci_led_on(struct slot * slot);
|
||||||
extern int cpci_led_off(struct slot * slot);
|
int cpci_led_off(struct slot * slot);
|
||||||
extern int cpci_configure_slot(struct slot *slot);
|
int cpci_configure_slot(struct slot *slot);
|
||||||
extern int cpci_unconfigure_slot(struct slot *slot);
|
int cpci_unconfigure_slot(struct slot *slot);
|
||||||
|
|
||||||
|
#ifdef CONFIG_HOTPLUG_PCI_CPCI
|
||||||
|
int cpci_hotplug_init(int debug);
|
||||||
|
void cpci_hotplug_exit(void);
|
||||||
|
#else
|
||||||
|
static inline int cpci_hotplug_init(int debug) { return 0; }
|
||||||
|
static inline void cpci_hotplug_exit(void) { }
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif /* _CPCI_HOTPLUG_H */
|
#endif /* _CPCI_HOTPLUG_H */
|
||||||
|
|
|
@ -404,50 +404,44 @@ struct resource_lists {
|
||||||
|
|
||||||
|
|
||||||
/* debugfs functions for the hotplug controller info */
|
/* debugfs functions for the hotplug controller info */
|
||||||
extern void cpqhp_initialize_debugfs(void);
|
void cpqhp_initialize_debugfs(void);
|
||||||
extern void cpqhp_shutdown_debugfs(void);
|
void cpqhp_shutdown_debugfs(void);
|
||||||
extern void cpqhp_create_debugfs_files(struct controller *ctrl);
|
void cpqhp_create_debugfs_files(struct controller *ctrl);
|
||||||
extern void cpqhp_remove_debugfs_files(struct controller *ctrl);
|
void cpqhp_remove_debugfs_files(struct controller *ctrl);
|
||||||
|
|
||||||
/* controller functions */
|
/* controller functions */
|
||||||
extern void cpqhp_pushbutton_thread(unsigned long event_pointer);
|
void cpqhp_pushbutton_thread(unsigned long event_pointer);
|
||||||
extern irqreturn_t cpqhp_ctrl_intr(int IRQ, void *data);
|
irqreturn_t cpqhp_ctrl_intr(int IRQ, void *data);
|
||||||
extern int cpqhp_find_available_resources(struct controller *ctrl,
|
int cpqhp_find_available_resources(struct controller *ctrl,
|
||||||
void __iomem *rom_start);
|
void __iomem *rom_start);
|
||||||
extern int cpqhp_event_start_thread(void);
|
int cpqhp_event_start_thread(void);
|
||||||
extern void cpqhp_event_stop_thread(void);
|
void cpqhp_event_stop_thread(void);
|
||||||
extern struct pci_func *cpqhp_slot_create(unsigned char busnumber);
|
struct pci_func *cpqhp_slot_create(unsigned char busnumber);
|
||||||
extern struct pci_func *cpqhp_slot_find(unsigned char bus, unsigned char device,
|
struct pci_func *cpqhp_slot_find(unsigned char bus, unsigned char device,
|
||||||
unsigned char index);
|
unsigned char index);
|
||||||
extern int cpqhp_process_SI(struct controller *ctrl, struct pci_func *func);
|
int cpqhp_process_SI(struct controller *ctrl, struct pci_func *func);
|
||||||
extern int cpqhp_process_SS(struct controller *ctrl, struct pci_func *func);
|
int cpqhp_process_SS(struct controller *ctrl, struct pci_func *func);
|
||||||
extern int cpqhp_hardware_test(struct controller *ctrl, int test_num);
|
int cpqhp_hardware_test(struct controller *ctrl, int test_num);
|
||||||
|
|
||||||
/* resource functions */
|
/* resource functions */
|
||||||
extern int cpqhp_resource_sort_and_combine (struct pci_resource **head);
|
int cpqhp_resource_sort_and_combine (struct pci_resource **head);
|
||||||
|
|
||||||
/* pci functions */
|
/* pci functions */
|
||||||
extern int cpqhp_set_irq(u8 bus_num, u8 dev_num, u8 int_pin, u8 irq_num);
|
int cpqhp_set_irq(u8 bus_num, u8 dev_num, u8 int_pin, u8 irq_num);
|
||||||
extern int cpqhp_get_bus_dev(struct controller *ctrl, u8 *bus_num, u8 *dev_num,
|
int cpqhp_get_bus_dev(struct controller *ctrl, u8 *bus_num, u8 *dev_num,
|
||||||
u8 slot);
|
u8 slot);
|
||||||
extern int cpqhp_save_config(struct controller *ctrl, int busnumber,
|
int cpqhp_save_config(struct controller *ctrl, int busnumber, int is_hot_plug);
|
||||||
int is_hot_plug);
|
int cpqhp_save_base_addr_length(struct controller *ctrl, struct pci_func *func);
|
||||||
extern int cpqhp_save_base_addr_length(struct controller *ctrl,
|
int cpqhp_save_used_resources(struct controller *ctrl, struct pci_func *func);
|
||||||
struct pci_func *func);
|
int cpqhp_configure_board(struct controller *ctrl, struct pci_func *func);
|
||||||
extern int cpqhp_save_used_resources(struct controller *ctrl,
|
int cpqhp_save_slot_config(struct controller *ctrl, struct pci_func *new_slot);
|
||||||
struct pci_func *func);
|
int cpqhp_valid_replace(struct controller *ctrl, struct pci_func *func);
|
||||||
extern int cpqhp_configure_board(struct controller *ctrl,
|
void cpqhp_destroy_board_resources(struct pci_func *func);
|
||||||
struct pci_func *func);
|
int cpqhp_return_board_resources(struct pci_func *func,
|
||||||
extern int cpqhp_save_slot_config(struct controller *ctrl,
|
struct resource_lists *resources);
|
||||||
struct pci_func *new_slot);
|
void cpqhp_destroy_resource_list(struct resource_lists *resources);
|
||||||
extern int cpqhp_valid_replace(struct controller *ctrl, struct pci_func *func);
|
int cpqhp_configure_device(struct controller *ctrl, struct pci_func *func);
|
||||||
extern void cpqhp_destroy_board_resources(struct pci_func *func);
|
int cpqhp_unconfigure_device(struct pci_func *func);
|
||||||
extern int cpqhp_return_board_resources (struct pci_func *func,
|
|
||||||
struct resource_lists *resources);
|
|
||||||
extern void cpqhp_destroy_resource_list(struct resource_lists *resources);
|
|
||||||
extern int cpqhp_configure_device(struct controller *ctrl,
|
|
||||||
struct pci_func *func);
|
|
||||||
extern int cpqhp_unconfigure_device(struct pci_func *func);
|
|
||||||
|
|
||||||
/* Global variables */
|
/* Global variables */
|
||||||
extern int cpqhp_debug;
|
extern int cpqhp_debug;
|
||||||
|
|
|
@ -30,26 +30,26 @@
|
||||||
|
|
||||||
#ifndef CONFIG_HOTPLUG_PCI_COMPAQ_NVRAM
|
#ifndef CONFIG_HOTPLUG_PCI_COMPAQ_NVRAM
|
||||||
|
|
||||||
static inline void compaq_nvram_init (void __iomem *rom_start)
|
static inline void compaq_nvram_init(void __iomem *rom_start)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int compaq_nvram_load (void __iomem *rom_start, struct controller *ctrl)
|
static inline int compaq_nvram_load(void __iomem *rom_start, struct controller *ctrl)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int compaq_nvram_store (void __iomem *rom_start)
|
static inline int compaq_nvram_store(void __iomem *rom_start)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
extern void compaq_nvram_init (void __iomem *rom_start);
|
void compaq_nvram_init(void __iomem *rom_start);
|
||||||
extern int compaq_nvram_load (void __iomem *rom_start, struct controller *ctrl);
|
int compaq_nvram_load(void __iomem *rom_start, struct controller *ctrl);
|
||||||
extern int compaq_nvram_store (void __iomem *rom_start);
|
int compaq_nvram_store(void __iomem *rom_start);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -275,17 +275,17 @@ extern struct list_head ibmphp_slot_head;
|
||||||
* FUNCTION PROTOTYPES *
|
* FUNCTION PROTOTYPES *
|
||||||
***********************************************************/
|
***********************************************************/
|
||||||
|
|
||||||
extern void ibmphp_free_ebda_hpc_queue (void);
|
void ibmphp_free_ebda_hpc_queue(void);
|
||||||
extern int ibmphp_access_ebda (void);
|
int ibmphp_access_ebda(void);
|
||||||
extern struct slot *ibmphp_get_slot_from_physical_num (u8);
|
struct slot *ibmphp_get_slot_from_physical_num(u8);
|
||||||
extern int ibmphp_get_total_hp_slots (void);
|
int ibmphp_get_total_hp_slots(void);
|
||||||
extern void ibmphp_free_ibm_slot (struct slot *);
|
void ibmphp_free_ibm_slot(struct slot *);
|
||||||
extern void ibmphp_free_bus_info_queue (void);
|
void ibmphp_free_bus_info_queue(void);
|
||||||
extern void ibmphp_free_ebda_pci_rsrc_queue (void);
|
void ibmphp_free_ebda_pci_rsrc_queue(void);
|
||||||
extern struct bus_info *ibmphp_find_same_bus_num (u32);
|
struct bus_info *ibmphp_find_same_bus_num(u32);
|
||||||
extern int ibmphp_get_bus_index (u8);
|
int ibmphp_get_bus_index(u8);
|
||||||
extern u16 ibmphp_get_total_controllers (void);
|
u16 ibmphp_get_total_controllers(void);
|
||||||
extern int ibmphp_register_pci (void);
|
int ibmphp_register_pci(void);
|
||||||
|
|
||||||
/* passed parameters */
|
/* passed parameters */
|
||||||
#define MEM 0
|
#define MEM 0
|
||||||
|
@ -381,24 +381,24 @@ struct res_needed {
|
||||||
|
|
||||||
/* functions */
|
/* functions */
|
||||||
|
|
||||||
extern int ibmphp_rsrc_init (void);
|
int ibmphp_rsrc_init(void);
|
||||||
extern int ibmphp_add_resource (struct resource_node *);
|
int ibmphp_add_resource(struct resource_node *);
|
||||||
extern int ibmphp_remove_resource (struct resource_node *);
|
int ibmphp_remove_resource(struct resource_node *);
|
||||||
extern int ibmphp_find_resource (struct bus_node *, u32, struct resource_node **, int);
|
int ibmphp_find_resource(struct bus_node *, u32, struct resource_node **, int);
|
||||||
extern int ibmphp_check_resource (struct resource_node *, u8);
|
int ibmphp_check_resource(struct resource_node *, u8);
|
||||||
extern int ibmphp_remove_bus (struct bus_node *, u8);
|
int ibmphp_remove_bus(struct bus_node *, u8);
|
||||||
extern void ibmphp_free_resources (void);
|
void ibmphp_free_resources(void);
|
||||||
extern int ibmphp_add_pfmem_from_mem (struct resource_node *);
|
int ibmphp_add_pfmem_from_mem(struct resource_node *);
|
||||||
extern struct bus_node *ibmphp_find_res_bus (u8);
|
struct bus_node *ibmphp_find_res_bus(u8);
|
||||||
extern void ibmphp_print_test (void); /* for debugging purposes */
|
void ibmphp_print_test(void); /* for debugging purposes */
|
||||||
|
|
||||||
extern void ibmphp_hpc_initvars (void);
|
void ibmphp_hpc_initvars(void);
|
||||||
extern int ibmphp_hpc_readslot (struct slot *, u8, u8 *);
|
int ibmphp_hpc_readslot(struct slot *, u8, u8 *);
|
||||||
extern int ibmphp_hpc_writeslot (struct slot *, u8);
|
int ibmphp_hpc_writeslot(struct slot *, u8);
|
||||||
extern void ibmphp_lock_operations (void);
|
void ibmphp_lock_operations(void);
|
||||||
extern void ibmphp_unlock_operations (void);
|
void ibmphp_unlock_operations(void);
|
||||||
extern int ibmphp_hpc_start_poll_thread (void);
|
int ibmphp_hpc_start_poll_thread(void);
|
||||||
extern void ibmphp_hpc_stop_poll_thread (void);
|
void ibmphp_hpc_stop_poll_thread(void);
|
||||||
|
|
||||||
//----------------------------------------------------------------------------
|
//----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
@ -749,11 +749,11 @@ struct controller {
|
||||||
|
|
||||||
/* Functions */
|
/* Functions */
|
||||||
|
|
||||||
extern int ibmphp_init_devno (struct slot **); /* This function is called from EBDA, so we need it not be static */
|
int ibmphp_init_devno(struct slot **); /* This function is called from EBDA, so we need it not be static */
|
||||||
extern int ibmphp_do_disable_slot (struct slot *slot_cur);
|
int ibmphp_do_disable_slot(struct slot *slot_cur);
|
||||||
extern int ibmphp_update_slot_info (struct slot *); /* This function is called from HPC, so we need it to not be be static */
|
int ibmphp_update_slot_info(struct slot *); /* This function is called from HPC, so we need it to not be be static */
|
||||||
extern int ibmphp_configure_card (struct pci_func *, u8);
|
int ibmphp_configure_card(struct pci_func *, u8);
|
||||||
extern int ibmphp_unconfigure_card (struct slot **, int);
|
int ibmphp_unconfigure_card(struct slot **, int);
|
||||||
extern struct hotplug_slot_ops ibmphp_hotplug_slot_ops;
|
extern struct hotplug_slot_ops ibmphp_hotplug_slot_ops;
|
||||||
|
|
||||||
#endif //__IBMPHP_H
|
#endif //__IBMPHP_H
|
||||||
|
|
|
@ -41,6 +41,7 @@
|
||||||
#include <linux/pci_hotplug.h>
|
#include <linux/pci_hotplug.h>
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
#include "../pci.h"
|
#include "../pci.h"
|
||||||
|
#include "cpci_hotplug.h"
|
||||||
|
|
||||||
#define MY_NAME "pci_hotplug"
|
#define MY_NAME "pci_hotplug"
|
||||||
|
|
||||||
|
@ -63,14 +64,6 @@ static bool debug;
|
||||||
static LIST_HEAD(pci_hotplug_slot_list);
|
static LIST_HEAD(pci_hotplug_slot_list);
|
||||||
static DEFINE_MUTEX(pci_hp_mutex);
|
static DEFINE_MUTEX(pci_hp_mutex);
|
||||||
|
|
||||||
#ifdef CONFIG_HOTPLUG_PCI_CPCI
|
|
||||||
extern int cpci_hotplug_init(int debug);
|
|
||||||
extern void cpci_hotplug_exit(void);
|
|
||||||
#else
|
|
||||||
static inline int cpci_hotplug_init(int debug) { return 0; }
|
|
||||||
static inline void cpci_hotplug_exit(void) { }
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Weee, fun with macros... */
|
/* Weee, fun with macros... */
|
||||||
#define GET_STATUS(name,type) \
|
#define GET_STATUS(name,type) \
|
||||||
static int get_##name (struct hotplug_slot *slot, type *value) \
|
static int get_##name (struct hotplug_slot *slot, type *value) \
|
||||||
|
@ -524,13 +517,11 @@ int pci_hp_deregister(struct hotplug_slot *hotplug)
|
||||||
*
|
*
|
||||||
* Returns 0 if successful, anything else for an error.
|
* Returns 0 if successful, anything else for an error.
|
||||||
*/
|
*/
|
||||||
int __must_check pci_hp_change_slot_info(struct hotplug_slot *hotplug,
|
int pci_hp_change_slot_info(struct hotplug_slot *hotplug,
|
||||||
struct hotplug_slot_info *info)
|
struct hotplug_slot_info *info)
|
||||||
{
|
{
|
||||||
struct pci_slot *slot;
|
|
||||||
if (!hotplug || !info)
|
if (!hotplug || !info)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
slot = hotplug->pci_slot;
|
|
||||||
|
|
||||||
memcpy(hotplug->info, info, sizeof(struct hotplug_slot_info));
|
memcpy(hotplug->info, info, sizeof(struct hotplug_slot_info));
|
||||||
|
|
||||||
|
|
|
@ -127,15 +127,15 @@ struct controller {
|
||||||
#define NO_CMD_CMPL(ctrl) ((ctrl)->slot_cap & PCI_EXP_SLTCAP_NCCS)
|
#define NO_CMD_CMPL(ctrl) ((ctrl)->slot_cap & PCI_EXP_SLTCAP_NCCS)
|
||||||
#define PSN(ctrl) ((ctrl)->slot_cap >> 19)
|
#define PSN(ctrl) ((ctrl)->slot_cap >> 19)
|
||||||
|
|
||||||
extern int pciehp_sysfs_enable_slot(struct slot *slot);
|
int pciehp_sysfs_enable_slot(struct slot *slot);
|
||||||
extern int pciehp_sysfs_disable_slot(struct slot *slot);
|
int pciehp_sysfs_disable_slot(struct slot *slot);
|
||||||
extern u8 pciehp_handle_attention_button(struct slot *p_slot);
|
u8 pciehp_handle_attention_button(struct slot *p_slot);
|
||||||
extern u8 pciehp_handle_switch_change(struct slot *p_slot);
|
u8 pciehp_handle_switch_change(struct slot *p_slot);
|
||||||
extern u8 pciehp_handle_presence_change(struct slot *p_slot);
|
u8 pciehp_handle_presence_change(struct slot *p_slot);
|
||||||
extern u8 pciehp_handle_power_fault(struct slot *p_slot);
|
u8 pciehp_handle_power_fault(struct slot *p_slot);
|
||||||
extern int pciehp_configure_device(struct slot *p_slot);
|
int pciehp_configure_device(struct slot *p_slot);
|
||||||
extern int pciehp_unconfigure_device(struct slot *p_slot);
|
int pciehp_unconfigure_device(struct slot *p_slot);
|
||||||
extern void pciehp_queue_pushbutton_work(struct work_struct *work);
|
void pciehp_queue_pushbutton_work(struct work_struct *work);
|
||||||
struct controller *pcie_init(struct pcie_device *dev);
|
struct controller *pcie_init(struct pcie_device *dev);
|
||||||
int pcie_init_notification(struct controller *ctrl);
|
int pcie_init_notification(struct controller *ctrl);
|
||||||
int pciehp_enable_slot(struct slot *p_slot);
|
int pciehp_enable_slot(struct slot *p_slot);
|
||||||
|
@ -166,8 +166,8 @@ static inline const char *slot_name(struct slot *slot)
|
||||||
#include <acpi/acpi_bus.h>
|
#include <acpi/acpi_bus.h>
|
||||||
#include <linux/pci-acpi.h>
|
#include <linux/pci-acpi.h>
|
||||||
|
|
||||||
extern void __init pciehp_acpi_slot_detection_init(void);
|
void __init pciehp_acpi_slot_detection_init(void);
|
||||||
extern int pciehp_acpi_slot_detection_check(struct pci_dev *dev);
|
int pciehp_acpi_slot_detection_check(struct pci_dev *dev);
|
||||||
|
|
||||||
static inline void pciehp_firmware_init(void)
|
static inline void pciehp_firmware_init(void)
|
||||||
{
|
{
|
||||||
|
|
|
@ -90,7 +90,7 @@ static int __init dummy_probe(struct pcie_device *dev)
|
||||||
slot = kzalloc(sizeof(*slot), GFP_KERNEL);
|
slot = kzalloc(sizeof(*slot), GFP_KERNEL);
|
||||||
if (!slot)
|
if (!slot)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
slot->number = slot_cap >> 19;
|
slot->number = (slot_cap & PCI_EXP_SLTCAP_PSN) >> 19;
|
||||||
list_for_each_entry(tmp, &dummy_slots, list) {
|
list_for_each_entry(tmp, &dummy_slots, list) {
|
||||||
if (tmp->number == slot->number)
|
if (tmp->number == slot->number)
|
||||||
dup_slot_id++;
|
dup_slot_id++;
|
||||||
|
|
|
@ -15,10 +15,10 @@
|
||||||
#ifndef _RPADLPAR_IO_H_
|
#ifndef _RPADLPAR_IO_H_
|
||||||
#define _RPADLPAR_IO_H_
|
#define _RPADLPAR_IO_H_
|
||||||
|
|
||||||
extern int dlpar_sysfs_init(void);
|
int dlpar_sysfs_init(void);
|
||||||
extern void dlpar_sysfs_exit(void);
|
void dlpar_sysfs_exit(void);
|
||||||
|
|
||||||
extern int dlpar_add_slot(char *drc_name);
|
int dlpar_add_slot(char *drc_name);
|
||||||
extern int dlpar_remove_slot(char *drc_name);
|
int dlpar_remove_slot(char *drc_name);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -86,18 +86,18 @@ extern struct list_head rpaphp_slot_head;
|
||||||
/* function prototypes */
|
/* function prototypes */
|
||||||
|
|
||||||
/* rpaphp_pci.c */
|
/* rpaphp_pci.c */
|
||||||
extern int rpaphp_enable_slot(struct slot *slot);
|
int rpaphp_enable_slot(struct slot *slot);
|
||||||
extern int rpaphp_get_sensor_state(struct slot *slot, int *state);
|
int rpaphp_get_sensor_state(struct slot *slot, int *state);
|
||||||
|
|
||||||
/* rpaphp_core.c */
|
/* rpaphp_core.c */
|
||||||
extern int rpaphp_add_slot(struct device_node *dn);
|
int rpaphp_add_slot(struct device_node *dn);
|
||||||
extern int rpaphp_get_drc_props(struct device_node *dn, int *drc_index,
|
int rpaphp_get_drc_props(struct device_node *dn, int *drc_index,
|
||||||
char **drc_name, char **drc_type, int *drc_power_domain);
|
char **drc_name, char **drc_type, int *drc_power_domain);
|
||||||
|
|
||||||
/* rpaphp_slot.c */
|
/* rpaphp_slot.c */
|
||||||
extern void dealloc_slot_struct(struct slot *slot);
|
void dealloc_slot_struct(struct slot *slot);
|
||||||
extern struct slot *alloc_slot_struct(struct device_node *dn, int drc_index, char *drc_name, int power_domain);
|
struct slot *alloc_slot_struct(struct device_node *dn, int drc_index, char *drc_name, int power_domain);
|
||||||
extern int rpaphp_register_slot(struct slot *slot);
|
int rpaphp_register_slot(struct slot *slot);
|
||||||
extern int rpaphp_deregister_slot(struct slot *slot);
|
int rpaphp_deregister_slot(struct slot *slot);
|
||||||
|
|
||||||
#endif /* _PPC64PHP_H */
|
#endif /* _PPC64PHP_H */
|
||||||
|
|
|
@ -168,19 +168,19 @@ struct controller {
|
||||||
#define WRONG_BUS_FREQUENCY 0x0000000D
|
#define WRONG_BUS_FREQUENCY 0x0000000D
|
||||||
#define POWER_FAILURE 0x0000000E
|
#define POWER_FAILURE 0x0000000E
|
||||||
|
|
||||||
extern int __must_check shpchp_create_ctrl_files(struct controller *ctrl);
|
int __must_check shpchp_create_ctrl_files(struct controller *ctrl);
|
||||||
extern void shpchp_remove_ctrl_files(struct controller *ctrl);
|
void shpchp_remove_ctrl_files(struct controller *ctrl);
|
||||||
extern int shpchp_sysfs_enable_slot(struct slot *slot);
|
int shpchp_sysfs_enable_slot(struct slot *slot);
|
||||||
extern int shpchp_sysfs_disable_slot(struct slot *slot);
|
int shpchp_sysfs_disable_slot(struct slot *slot);
|
||||||
extern u8 shpchp_handle_attention_button(u8 hp_slot, struct controller *ctrl);
|
u8 shpchp_handle_attention_button(u8 hp_slot, struct controller *ctrl);
|
||||||
extern u8 shpchp_handle_switch_change(u8 hp_slot, struct controller *ctrl);
|
u8 shpchp_handle_switch_change(u8 hp_slot, struct controller *ctrl);
|
||||||
extern u8 shpchp_handle_presence_change(u8 hp_slot, struct controller *ctrl);
|
u8 shpchp_handle_presence_change(u8 hp_slot, struct controller *ctrl);
|
||||||
extern u8 shpchp_handle_power_fault(u8 hp_slot, struct controller *ctrl);
|
u8 shpchp_handle_power_fault(u8 hp_slot, struct controller *ctrl);
|
||||||
extern int shpchp_configure_device(struct slot *p_slot);
|
int shpchp_configure_device(struct slot *p_slot);
|
||||||
extern int shpchp_unconfigure_device(struct slot *p_slot);
|
int shpchp_unconfigure_device(struct slot *p_slot);
|
||||||
extern void cleanup_slots(struct controller *ctrl);
|
void cleanup_slots(struct controller *ctrl);
|
||||||
extern void shpchp_queue_pushbutton_work(struct work_struct *work);
|
void shpchp_queue_pushbutton_work(struct work_struct *work);
|
||||||
extern int shpc_init( struct controller *ctrl, struct pci_dev *pdev);
|
int shpc_init( struct controller *ctrl, struct pci_dev *pdev);
|
||||||
|
|
||||||
static inline const char *slot_name(struct slot *slot)
|
static inline const char *slot_name(struct slot *slot)
|
||||||
{
|
{
|
||||||
|
|
|
@ -85,7 +85,7 @@ static ssize_t show_ctrl (struct device *dev, struct device_attribute *attr, cha
|
||||||
}
|
}
|
||||||
static DEVICE_ATTR (ctrl, S_IRUGO, show_ctrl, NULL);
|
static DEVICE_ATTR (ctrl, S_IRUGO, show_ctrl, NULL);
|
||||||
|
|
||||||
int __must_check shpchp_create_ctrl_files (struct controller *ctrl)
|
int shpchp_create_ctrl_files (struct controller *ctrl)
|
||||||
{
|
{
|
||||||
return device_create_file (&ctrl->pci_dev->dev, &dev_attr_ctrl);
|
return device_create_file (&ctrl->pci_dev->dev, &dev_attr_ctrl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,10 +22,12 @@
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
|
||||||
#include "pci.h"
|
#include "pci.h"
|
||||||
#include "msi.h"
|
|
||||||
|
|
||||||
static int pci_msi_enable = 1;
|
static int pci_msi_enable = 1;
|
||||||
|
|
||||||
|
#define msix_table_size(flags) ((flags & PCI_MSIX_FLAGS_QSIZE) + 1)
|
||||||
|
|
||||||
|
|
||||||
/* Arch hooks */
|
/* Arch hooks */
|
||||||
|
|
||||||
#ifndef arch_msi_check_device
|
#ifndef arch_msi_check_device
|
||||||
|
@ -111,32 +113,26 @@ void default_restore_msi_irqs(struct pci_dev *dev, int irq)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static void msi_set_enable(struct pci_dev *dev, int pos, int enable)
|
static void msi_set_enable(struct pci_dev *dev, int enable)
|
||||||
{
|
{
|
||||||
u16 control;
|
u16 control;
|
||||||
|
|
||||||
BUG_ON(!pos);
|
pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &control);
|
||||||
|
|
||||||
pci_read_config_word(dev, pos + PCI_MSI_FLAGS, &control);
|
|
||||||
control &= ~PCI_MSI_FLAGS_ENABLE;
|
control &= ~PCI_MSI_FLAGS_ENABLE;
|
||||||
if (enable)
|
if (enable)
|
||||||
control |= PCI_MSI_FLAGS_ENABLE;
|
control |= PCI_MSI_FLAGS_ENABLE;
|
||||||
pci_write_config_word(dev, pos + PCI_MSI_FLAGS, control);
|
pci_write_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, control);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void msix_set_enable(struct pci_dev *dev, int enable)
|
static void msix_set_enable(struct pci_dev *dev, int enable)
|
||||||
{
|
{
|
||||||
int pos;
|
|
||||||
u16 control;
|
u16 control;
|
||||||
|
|
||||||
pos = pci_find_capability(dev, PCI_CAP_ID_MSIX);
|
pci_read_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, &control);
|
||||||
if (pos) {
|
control &= ~PCI_MSIX_FLAGS_ENABLE;
|
||||||
pci_read_config_word(dev, pos + PCI_MSIX_FLAGS, &control);
|
if (enable)
|
||||||
control &= ~PCI_MSIX_FLAGS_ENABLE;
|
control |= PCI_MSIX_FLAGS_ENABLE;
|
||||||
if (enable)
|
pci_write_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, control);
|
||||||
control |= PCI_MSIX_FLAGS_ENABLE;
|
|
||||||
pci_write_config_word(dev, pos + PCI_MSIX_FLAGS, control);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline __attribute_const__ u32 msi_mask(unsigned x)
|
static inline __attribute_const__ u32 msi_mask(unsigned x)
|
||||||
|
@ -247,18 +243,18 @@ void __read_msi_msg(struct msi_desc *entry, struct msi_msg *msg)
|
||||||
msg->data = readl(base + PCI_MSIX_ENTRY_DATA);
|
msg->data = readl(base + PCI_MSIX_ENTRY_DATA);
|
||||||
} else {
|
} else {
|
||||||
struct pci_dev *dev = entry->dev;
|
struct pci_dev *dev = entry->dev;
|
||||||
int pos = entry->msi_attrib.pos;
|
int pos = dev->msi_cap;
|
||||||
u16 data;
|
u16 data;
|
||||||
|
|
||||||
pci_read_config_dword(dev, msi_lower_address_reg(pos),
|
pci_read_config_dword(dev, pos + PCI_MSI_ADDRESS_LO,
|
||||||
&msg->address_lo);
|
&msg->address_lo);
|
||||||
if (entry->msi_attrib.is_64) {
|
if (entry->msi_attrib.is_64) {
|
||||||
pci_read_config_dword(dev, msi_upper_address_reg(pos),
|
pci_read_config_dword(dev, pos + PCI_MSI_ADDRESS_HI,
|
||||||
&msg->address_hi);
|
&msg->address_hi);
|
||||||
pci_read_config_word(dev, msi_data_reg(pos, 1), &data);
|
pci_read_config_word(dev, pos + PCI_MSI_DATA_64, &data);
|
||||||
} else {
|
} else {
|
||||||
msg->address_hi = 0;
|
msg->address_hi = 0;
|
||||||
pci_read_config_word(dev, msi_data_reg(pos, 0), &data);
|
pci_read_config_word(dev, pos + PCI_MSI_DATA_32, &data);
|
||||||
}
|
}
|
||||||
msg->data = data;
|
msg->data = data;
|
||||||
}
|
}
|
||||||
|
@ -302,24 +298,24 @@ void __write_msi_msg(struct msi_desc *entry, struct msi_msg *msg)
|
||||||
writel(msg->data, base + PCI_MSIX_ENTRY_DATA);
|
writel(msg->data, base + PCI_MSIX_ENTRY_DATA);
|
||||||
} else {
|
} else {
|
||||||
struct pci_dev *dev = entry->dev;
|
struct pci_dev *dev = entry->dev;
|
||||||
int pos = entry->msi_attrib.pos;
|
int pos = dev->msi_cap;
|
||||||
u16 msgctl;
|
u16 msgctl;
|
||||||
|
|
||||||
pci_read_config_word(dev, msi_control_reg(pos), &msgctl);
|
pci_read_config_word(dev, pos + PCI_MSI_FLAGS, &msgctl);
|
||||||
msgctl &= ~PCI_MSI_FLAGS_QSIZE;
|
msgctl &= ~PCI_MSI_FLAGS_QSIZE;
|
||||||
msgctl |= entry->msi_attrib.multiple << 4;
|
msgctl |= entry->msi_attrib.multiple << 4;
|
||||||
pci_write_config_word(dev, msi_control_reg(pos), msgctl);
|
pci_write_config_word(dev, pos + PCI_MSI_FLAGS, msgctl);
|
||||||
|
|
||||||
pci_write_config_dword(dev, msi_lower_address_reg(pos),
|
pci_write_config_dword(dev, pos + PCI_MSI_ADDRESS_LO,
|
||||||
msg->address_lo);
|
msg->address_lo);
|
||||||
if (entry->msi_attrib.is_64) {
|
if (entry->msi_attrib.is_64) {
|
||||||
pci_write_config_dword(dev, msi_upper_address_reg(pos),
|
pci_write_config_dword(dev, pos + PCI_MSI_ADDRESS_HI,
|
||||||
msg->address_hi);
|
msg->address_hi);
|
||||||
pci_write_config_word(dev, msi_data_reg(pos, 1),
|
pci_write_config_word(dev, pos + PCI_MSI_DATA_64,
|
||||||
msg->data);
|
msg->data);
|
||||||
} else {
|
} else {
|
||||||
pci_write_config_word(dev, msi_data_reg(pos, 0),
|
pci_write_config_word(dev, pos + PCI_MSI_DATA_32,
|
||||||
msg->data);
|
msg->data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
entry->msg = *msg;
|
entry->msg = *msg;
|
||||||
|
@ -391,7 +387,6 @@ static void pci_intx_for_msi(struct pci_dev *dev, int enable)
|
||||||
|
|
||||||
static void __pci_restore_msi_state(struct pci_dev *dev)
|
static void __pci_restore_msi_state(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
int pos;
|
|
||||||
u16 control;
|
u16 control;
|
||||||
struct msi_desc *entry;
|
struct msi_desc *entry;
|
||||||
|
|
||||||
|
@ -399,22 +394,20 @@ static void __pci_restore_msi_state(struct pci_dev *dev)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
entry = irq_get_msi_desc(dev->irq);
|
entry = irq_get_msi_desc(dev->irq);
|
||||||
pos = entry->msi_attrib.pos;
|
|
||||||
|
|
||||||
pci_intx_for_msi(dev, 0);
|
pci_intx_for_msi(dev, 0);
|
||||||
msi_set_enable(dev, pos, 0);
|
msi_set_enable(dev, 0);
|
||||||
arch_restore_msi_irqs(dev, dev->irq);
|
arch_restore_msi_irqs(dev, dev->irq);
|
||||||
|
|
||||||
pci_read_config_word(dev, pos + PCI_MSI_FLAGS, &control);
|
pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &control);
|
||||||
msi_mask_irq(entry, msi_capable_mask(control), entry->masked);
|
msi_mask_irq(entry, msi_capable_mask(control), entry->masked);
|
||||||
control &= ~PCI_MSI_FLAGS_QSIZE;
|
control &= ~PCI_MSI_FLAGS_QSIZE;
|
||||||
control |= (entry->msi_attrib.multiple << 4) | PCI_MSI_FLAGS_ENABLE;
|
control |= (entry->msi_attrib.multiple << 4) | PCI_MSI_FLAGS_ENABLE;
|
||||||
pci_write_config_word(dev, pos + PCI_MSI_FLAGS, control);
|
pci_write_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, control);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __pci_restore_msix_state(struct pci_dev *dev)
|
static void __pci_restore_msix_state(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
int pos;
|
|
||||||
struct msi_desc *entry;
|
struct msi_desc *entry;
|
||||||
u16 control;
|
u16 control;
|
||||||
|
|
||||||
|
@ -422,13 +415,12 @@ static void __pci_restore_msix_state(struct pci_dev *dev)
|
||||||
return;
|
return;
|
||||||
BUG_ON(list_empty(&dev->msi_list));
|
BUG_ON(list_empty(&dev->msi_list));
|
||||||
entry = list_first_entry(&dev->msi_list, struct msi_desc, list);
|
entry = list_first_entry(&dev->msi_list, struct msi_desc, list);
|
||||||
pos = entry->msi_attrib.pos;
|
pci_read_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, &control);
|
||||||
pci_read_config_word(dev, pos + PCI_MSIX_FLAGS, &control);
|
|
||||||
|
|
||||||
/* route the table */
|
/* route the table */
|
||||||
pci_intx_for_msi(dev, 0);
|
pci_intx_for_msi(dev, 0);
|
||||||
control |= PCI_MSIX_FLAGS_ENABLE | PCI_MSIX_FLAGS_MASKALL;
|
control |= PCI_MSIX_FLAGS_ENABLE | PCI_MSIX_FLAGS_MASKALL;
|
||||||
pci_write_config_word(dev, pos + PCI_MSIX_FLAGS, control);
|
pci_write_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, control);
|
||||||
|
|
||||||
list_for_each_entry(entry, &dev->msi_list, list) {
|
list_for_each_entry(entry, &dev->msi_list, list) {
|
||||||
arch_restore_msi_irqs(dev, entry->irq);
|
arch_restore_msi_irqs(dev, entry->irq);
|
||||||
|
@ -436,7 +428,7 @@ static void __pci_restore_msix_state(struct pci_dev *dev)
|
||||||
}
|
}
|
||||||
|
|
||||||
control &= ~PCI_MSIX_FLAGS_MASKALL;
|
control &= ~PCI_MSIX_FLAGS_MASKALL;
|
||||||
pci_write_config_word(dev, pos + PCI_MSIX_FLAGS, control);
|
pci_write_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, control);
|
||||||
}
|
}
|
||||||
|
|
||||||
void pci_restore_msi_state(struct pci_dev *dev)
|
void pci_restore_msi_state(struct pci_dev *dev)
|
||||||
|
@ -484,12 +476,12 @@ static struct msi_attribute mode_attribute =
|
||||||
__ATTR(mode, S_IRUGO, show_msi_mode, NULL);
|
__ATTR(mode, S_IRUGO, show_msi_mode, NULL);
|
||||||
|
|
||||||
|
|
||||||
struct attribute *msi_irq_default_attrs[] = {
|
static struct attribute *msi_irq_default_attrs[] = {
|
||||||
&mode_attribute.attr,
|
&mode_attribute.attr,
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
void msi_kobj_release(struct kobject *kobj)
|
static void msi_kobj_release(struct kobject *kobj)
|
||||||
{
|
{
|
||||||
struct msi_desc *entry = to_msi_desc(kobj);
|
struct msi_desc *entry = to_msi_desc(kobj);
|
||||||
|
|
||||||
|
@ -552,27 +544,27 @@ out_unroll:
|
||||||
static int msi_capability_init(struct pci_dev *dev, int nvec)
|
static int msi_capability_init(struct pci_dev *dev, int nvec)
|
||||||
{
|
{
|
||||||
struct msi_desc *entry;
|
struct msi_desc *entry;
|
||||||
int pos, ret;
|
int ret;
|
||||||
u16 control;
|
u16 control;
|
||||||
unsigned mask;
|
unsigned mask;
|
||||||
|
|
||||||
pos = pci_find_capability(dev, PCI_CAP_ID_MSI);
|
msi_set_enable(dev, 0); /* Disable MSI during set up */
|
||||||
msi_set_enable(dev, pos, 0); /* Disable MSI during set up */
|
|
||||||
|
|
||||||
pci_read_config_word(dev, msi_control_reg(pos), &control);
|
pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &control);
|
||||||
/* MSI Entry Initialization */
|
/* MSI Entry Initialization */
|
||||||
entry = alloc_msi_entry(dev);
|
entry = alloc_msi_entry(dev);
|
||||||
if (!entry)
|
if (!entry)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
entry->msi_attrib.is_msix = 0;
|
entry->msi_attrib.is_msix = 0;
|
||||||
entry->msi_attrib.is_64 = is_64bit_address(control);
|
entry->msi_attrib.is_64 = !!(control & PCI_MSI_FLAGS_64BIT);
|
||||||
entry->msi_attrib.entry_nr = 0;
|
entry->msi_attrib.entry_nr = 0;
|
||||||
entry->msi_attrib.maskbit = is_mask_bit_support(control);
|
entry->msi_attrib.maskbit = !!(control & PCI_MSI_FLAGS_MASKBIT);
|
||||||
entry->msi_attrib.default_irq = dev->irq; /* Save IOAPIC IRQ */
|
entry->msi_attrib.default_irq = dev->irq; /* Save IOAPIC IRQ */
|
||||||
entry->msi_attrib.pos = pos;
|
entry->msi_attrib.pos = dev->msi_cap;
|
||||||
|
|
||||||
entry->mask_pos = msi_mask_reg(pos, entry->msi_attrib.is_64);
|
entry->mask_pos = dev->msi_cap + (control & PCI_MSI_FLAGS_64BIT) ?
|
||||||
|
PCI_MSI_MASK_64 : PCI_MSI_MASK_32;
|
||||||
/* All MSIs are unmasked by default, Mask them all */
|
/* All MSIs are unmasked by default, Mask them all */
|
||||||
if (entry->msi_attrib.maskbit)
|
if (entry->msi_attrib.maskbit)
|
||||||
pci_read_config_dword(dev, entry->mask_pos, &entry->masked);
|
pci_read_config_dword(dev, entry->mask_pos, &entry->masked);
|
||||||
|
@ -598,31 +590,30 @@ static int msi_capability_init(struct pci_dev *dev, int nvec)
|
||||||
|
|
||||||
/* Set MSI enabled bits */
|
/* Set MSI enabled bits */
|
||||||
pci_intx_for_msi(dev, 0);
|
pci_intx_for_msi(dev, 0);
|
||||||
msi_set_enable(dev, pos, 1);
|
msi_set_enable(dev, 1);
|
||||||
dev->msi_enabled = 1;
|
dev->msi_enabled = 1;
|
||||||
|
|
||||||
dev->irq = entry->irq;
|
dev->irq = entry->irq;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __iomem *msix_map_region(struct pci_dev *dev, unsigned pos,
|
static void __iomem *msix_map_region(struct pci_dev *dev, unsigned nr_entries)
|
||||||
unsigned nr_entries)
|
|
||||||
{
|
{
|
||||||
resource_size_t phys_addr;
|
resource_size_t phys_addr;
|
||||||
u32 table_offset;
|
u32 table_offset;
|
||||||
u8 bir;
|
u8 bir;
|
||||||
|
|
||||||
pci_read_config_dword(dev, msix_table_offset_reg(pos), &table_offset);
|
pci_read_config_dword(dev, dev->msix_cap + PCI_MSIX_TABLE,
|
||||||
bir = (u8)(table_offset & PCI_MSIX_FLAGS_BIRMASK);
|
&table_offset);
|
||||||
table_offset &= ~PCI_MSIX_FLAGS_BIRMASK;
|
bir = (u8)(table_offset & PCI_MSIX_TABLE_BIR);
|
||||||
|
table_offset &= PCI_MSIX_TABLE_OFFSET;
|
||||||
phys_addr = pci_resource_start(dev, bir) + table_offset;
|
phys_addr = pci_resource_start(dev, bir) + table_offset;
|
||||||
|
|
||||||
return ioremap_nocache(phys_addr, nr_entries * PCI_MSIX_ENTRY_SIZE);
|
return ioremap_nocache(phys_addr, nr_entries * PCI_MSIX_ENTRY_SIZE);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int msix_setup_entries(struct pci_dev *dev, unsigned pos,
|
static int msix_setup_entries(struct pci_dev *dev, void __iomem *base,
|
||||||
void __iomem *base, struct msix_entry *entries,
|
struct msix_entry *entries, int nvec)
|
||||||
int nvec)
|
|
||||||
{
|
{
|
||||||
struct msi_desc *entry;
|
struct msi_desc *entry;
|
||||||
int i;
|
int i;
|
||||||
|
@ -642,7 +633,7 @@ static int msix_setup_entries(struct pci_dev *dev, unsigned pos,
|
||||||
entry->msi_attrib.is_64 = 1;
|
entry->msi_attrib.is_64 = 1;
|
||||||
entry->msi_attrib.entry_nr = entries[i].entry;
|
entry->msi_attrib.entry_nr = entries[i].entry;
|
||||||
entry->msi_attrib.default_irq = dev->irq;
|
entry->msi_attrib.default_irq = dev->irq;
|
||||||
entry->msi_attrib.pos = pos;
|
entry->msi_attrib.pos = dev->msix_cap;
|
||||||
entry->mask_base = base;
|
entry->mask_base = base;
|
||||||
|
|
||||||
list_add_tail(&entry->list, &dev->msi_list);
|
list_add_tail(&entry->list, &dev->msi_list);
|
||||||
|
@ -652,7 +643,7 @@ static int msix_setup_entries(struct pci_dev *dev, unsigned pos,
|
||||||
}
|
}
|
||||||
|
|
||||||
static void msix_program_entries(struct pci_dev *dev,
|
static void msix_program_entries(struct pci_dev *dev,
|
||||||
struct msix_entry *entries)
|
struct msix_entry *entries)
|
||||||
{
|
{
|
||||||
struct msi_desc *entry;
|
struct msi_desc *entry;
|
||||||
int i = 0;
|
int i = 0;
|
||||||
|
@ -682,23 +673,22 @@ static void msix_program_entries(struct pci_dev *dev,
|
||||||
static int msix_capability_init(struct pci_dev *dev,
|
static int msix_capability_init(struct pci_dev *dev,
|
||||||
struct msix_entry *entries, int nvec)
|
struct msix_entry *entries, int nvec)
|
||||||
{
|
{
|
||||||
int pos, ret;
|
int ret;
|
||||||
u16 control;
|
u16 control;
|
||||||
void __iomem *base;
|
void __iomem *base;
|
||||||
|
|
||||||
pos = pci_find_capability(dev, PCI_CAP_ID_MSIX);
|
pci_read_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, &control);
|
||||||
pci_read_config_word(dev, pos + PCI_MSIX_FLAGS, &control);
|
|
||||||
|
|
||||||
/* Ensure MSI-X is disabled while it is set up */
|
/* Ensure MSI-X is disabled while it is set up */
|
||||||
control &= ~PCI_MSIX_FLAGS_ENABLE;
|
control &= ~PCI_MSIX_FLAGS_ENABLE;
|
||||||
pci_write_config_word(dev, pos + PCI_MSIX_FLAGS, control);
|
pci_write_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, control);
|
||||||
|
|
||||||
/* Request & Map MSI-X table region */
|
/* Request & Map MSI-X table region */
|
||||||
base = msix_map_region(dev, pos, multi_msix_capable(control));
|
base = msix_map_region(dev, msix_table_size(control));
|
||||||
if (!base)
|
if (!base)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
ret = msix_setup_entries(dev, pos, base, entries, nvec);
|
ret = msix_setup_entries(dev, base, entries, nvec);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
@ -712,7 +702,7 @@ static int msix_capability_init(struct pci_dev *dev,
|
||||||
* interrupts coming in before they're fully set up.
|
* interrupts coming in before they're fully set up.
|
||||||
*/
|
*/
|
||||||
control |= PCI_MSIX_FLAGS_MASKALL | PCI_MSIX_FLAGS_ENABLE;
|
control |= PCI_MSIX_FLAGS_MASKALL | PCI_MSIX_FLAGS_ENABLE;
|
||||||
pci_write_config_word(dev, pos + PCI_MSIX_FLAGS, control);
|
pci_write_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, control);
|
||||||
|
|
||||||
msix_program_entries(dev, entries);
|
msix_program_entries(dev, entries);
|
||||||
|
|
||||||
|
@ -727,7 +717,7 @@ static int msix_capability_init(struct pci_dev *dev,
|
||||||
dev->msix_enabled = 1;
|
dev->msix_enabled = 1;
|
||||||
|
|
||||||
control &= ~PCI_MSIX_FLAGS_MASKALL;
|
control &= ~PCI_MSIX_FLAGS_MASKALL;
|
||||||
pci_write_config_word(dev, pos + PCI_MSIX_FLAGS, control);
|
pci_write_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, control);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -795,9 +785,6 @@ static int pci_msi_check_device(struct pci_dev *dev, int nvec, int type)
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
if (!pci_find_capability(dev, type))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -816,13 +803,13 @@ static int pci_msi_check_device(struct pci_dev *dev, int nvec, int type)
|
||||||
*/
|
*/
|
||||||
int pci_enable_msi_block(struct pci_dev *dev, unsigned int nvec)
|
int pci_enable_msi_block(struct pci_dev *dev, unsigned int nvec)
|
||||||
{
|
{
|
||||||
int status, pos, maxvec;
|
int status, maxvec;
|
||||||
u16 msgctl;
|
u16 msgctl;
|
||||||
|
|
||||||
pos = pci_find_capability(dev, PCI_CAP_ID_MSI);
|
if (!dev->msi_cap)
|
||||||
if (!pos)
|
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
pci_read_config_word(dev, pos + PCI_MSI_FLAGS, &msgctl);
|
|
||||||
|
pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &msgctl);
|
||||||
maxvec = 1 << ((msgctl & PCI_MSI_FLAGS_QMASK) >> 1);
|
maxvec = 1 << ((msgctl & PCI_MSI_FLAGS_QMASK) >> 1);
|
||||||
if (nvec > maxvec)
|
if (nvec > maxvec)
|
||||||
return maxvec;
|
return maxvec;
|
||||||
|
@ -847,14 +834,13 @@ EXPORT_SYMBOL(pci_enable_msi_block);
|
||||||
|
|
||||||
int pci_enable_msi_block_auto(struct pci_dev *dev, unsigned int *maxvec)
|
int pci_enable_msi_block_auto(struct pci_dev *dev, unsigned int *maxvec)
|
||||||
{
|
{
|
||||||
int ret, pos, nvec;
|
int ret, nvec;
|
||||||
u16 msgctl;
|
u16 msgctl;
|
||||||
|
|
||||||
pos = pci_find_capability(dev, PCI_CAP_ID_MSI);
|
if (!dev->msi_cap)
|
||||||
if (!pos)
|
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
pci_read_config_word(dev, pos + PCI_MSI_FLAGS, &msgctl);
|
pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &msgctl);
|
||||||
ret = 1 << ((msgctl & PCI_MSI_FLAGS_QMASK) >> 1);
|
ret = 1 << ((msgctl & PCI_MSI_FLAGS_QMASK) >> 1);
|
||||||
|
|
||||||
if (maxvec)
|
if (maxvec)
|
||||||
|
@ -876,21 +862,19 @@ void pci_msi_shutdown(struct pci_dev *dev)
|
||||||
struct msi_desc *desc;
|
struct msi_desc *desc;
|
||||||
u32 mask;
|
u32 mask;
|
||||||
u16 ctrl;
|
u16 ctrl;
|
||||||
unsigned pos;
|
|
||||||
|
|
||||||
if (!pci_msi_enable || !dev || !dev->msi_enabled)
|
if (!pci_msi_enable || !dev || !dev->msi_enabled)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
BUG_ON(list_empty(&dev->msi_list));
|
BUG_ON(list_empty(&dev->msi_list));
|
||||||
desc = list_first_entry(&dev->msi_list, struct msi_desc, list);
|
desc = list_first_entry(&dev->msi_list, struct msi_desc, list);
|
||||||
pos = desc->msi_attrib.pos;
|
|
||||||
|
|
||||||
msi_set_enable(dev, pos, 0);
|
msi_set_enable(dev, 0);
|
||||||
pci_intx_for_msi(dev, 1);
|
pci_intx_for_msi(dev, 1);
|
||||||
dev->msi_enabled = 0;
|
dev->msi_enabled = 0;
|
||||||
|
|
||||||
/* Return the device with MSI unmasked as initial states */
|
/* Return the device with MSI unmasked as initial states */
|
||||||
pci_read_config_word(dev, pos + PCI_MSI_FLAGS, &ctrl);
|
pci_read_config_word(dev, dev->msi_cap + PCI_MSI_FLAGS, &ctrl);
|
||||||
mask = msi_capable_mask(ctrl);
|
mask = msi_capable_mask(ctrl);
|
||||||
/* Keep cached state to be restored */
|
/* Keep cached state to be restored */
|
||||||
__msi_mask_irq(desc, mask, ~mask);
|
__msi_mask_irq(desc, mask, ~mask);
|
||||||
|
@ -917,15 +901,13 @@ EXPORT_SYMBOL(pci_disable_msi);
|
||||||
*/
|
*/
|
||||||
int pci_msix_table_size(struct pci_dev *dev)
|
int pci_msix_table_size(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
int pos;
|
|
||||||
u16 control;
|
u16 control;
|
||||||
|
|
||||||
pos = pci_find_capability(dev, PCI_CAP_ID_MSIX);
|
if (!dev->msix_cap)
|
||||||
if (!pos)
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
pci_read_config_word(dev, msi_control_reg(pos), &control);
|
pci_read_config_word(dev, dev->msix_cap + PCI_MSIX_FLAGS, &control);
|
||||||
return multi_msix_capable(control);
|
return msix_table_size(control);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -948,7 +930,7 @@ int pci_enable_msix(struct pci_dev *dev, struct msix_entry *entries, int nvec)
|
||||||
int status, nr_entries;
|
int status, nr_entries;
|
||||||
int i, j;
|
int i, j;
|
||||||
|
|
||||||
if (!entries)
|
if (!entries || !dev->msix_cap)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
status = pci_msi_check_device(dev, nvec, PCI_CAP_ID_MSIX);
|
status = pci_msi_check_device(dev, nvec, PCI_CAP_ID_MSIX);
|
||||||
|
@ -1048,15 +1030,17 @@ EXPORT_SYMBOL(pci_msi_enabled);
|
||||||
|
|
||||||
void pci_msi_init_pci_dev(struct pci_dev *dev)
|
void pci_msi_init_pci_dev(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
int pos;
|
|
||||||
INIT_LIST_HEAD(&dev->msi_list);
|
INIT_LIST_HEAD(&dev->msi_list);
|
||||||
|
|
||||||
/* Disable the msi hardware to avoid screaming interrupts
|
/* Disable the msi hardware to avoid screaming interrupts
|
||||||
* during boot. This is the power on reset default so
|
* during boot. This is the power on reset default so
|
||||||
* usually this should be a noop.
|
* usually this should be a noop.
|
||||||
*/
|
*/
|
||||||
pos = pci_find_capability(dev, PCI_CAP_ID_MSI);
|
dev->msi_cap = pci_find_capability(dev, PCI_CAP_ID_MSI);
|
||||||
if (pos)
|
if (dev->msi_cap)
|
||||||
msi_set_enable(dev, pos, 0);
|
msi_set_enable(dev, 0);
|
||||||
msix_set_enable(dev, 0);
|
|
||||||
|
dev->msix_cap = pci_find_capability(dev, PCI_CAP_ID_MSIX);
|
||||||
|
if (dev->msix_cap)
|
||||||
|
msix_set_enable(dev, 0);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,24 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (C) 2003-2004 Intel
|
|
||||||
* Copyright (C) Tom Long Nguyen (tom.l.nguyen@intel.com)
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef MSI_H
|
|
||||||
#define MSI_H
|
|
||||||
|
|
||||||
#define msi_control_reg(base) (base + PCI_MSI_FLAGS)
|
|
||||||
#define msi_lower_address_reg(base) (base + PCI_MSI_ADDRESS_LO)
|
|
||||||
#define msi_upper_address_reg(base) (base + PCI_MSI_ADDRESS_HI)
|
|
||||||
#define msi_data_reg(base, is64bit) \
|
|
||||||
(base + ((is64bit == 1) ? PCI_MSI_DATA_64 : PCI_MSI_DATA_32))
|
|
||||||
#define msi_mask_reg(base, is64bit) \
|
|
||||||
(base + ((is64bit == 1) ? PCI_MSI_MASK_64 : PCI_MSI_MASK_32))
|
|
||||||
#define is_64bit_address(control) (!!(control & PCI_MSI_FLAGS_64BIT))
|
|
||||||
#define is_mask_bit_support(control) (!!(control & PCI_MSI_FLAGS_MASKBIT))
|
|
||||||
|
|
||||||
#define msix_table_offset_reg(base) (base + PCI_MSIX_TABLE)
|
|
||||||
#define msix_pba_offset_reg(base) (base + PCI_MSIX_PBA)
|
|
||||||
#define msix_table_size(control) ((control & PCI_MSIX_FLAGS_QSIZE)+1)
|
|
||||||
#define multi_msix_capable(control) msix_table_size((control))
|
|
||||||
|
|
||||||
#endif /* MSI_H */
|
|
|
@ -288,6 +288,32 @@ static struct pci_platform_pm_ops acpi_pci_platform_pm = {
|
||||||
.run_wake = acpi_pci_run_wake,
|
.run_wake = acpi_pci_run_wake,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void acpi_pci_add_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
acpi_handle handle = NULL;
|
||||||
|
|
||||||
|
if (bus->bridge)
|
||||||
|
handle = ACPI_HANDLE(bus->bridge);
|
||||||
|
if (acpi_pci_disabled || handle == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
acpi_pci_slot_enumerate(bus, handle);
|
||||||
|
acpiphp_enumerate_slots(bus, handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void acpi_pci_remove_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* bus->bridge->acpi_node.handle has already been reset to NULL
|
||||||
|
* when acpi_pci_remove_bus() is called, so don't check ACPI handle.
|
||||||
|
*/
|
||||||
|
if (acpi_pci_disabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
acpiphp_remove_slots(bus);
|
||||||
|
acpi_pci_slot_remove(bus);
|
||||||
|
}
|
||||||
|
|
||||||
/* ACPI bus type */
|
/* ACPI bus type */
|
||||||
static int acpi_pci_find_device(struct device *dev, acpi_handle *handle)
|
static int acpi_pci_find_device(struct device *dev, acpi_handle *handle)
|
||||||
{
|
{
|
||||||
|
@ -362,7 +388,11 @@ static int __init acpi_pci_init(void)
|
||||||
ret = register_acpi_bus_type(&acpi_pci_bus);
|
ret = register_acpi_bus_type(&acpi_pci_bus);
|
||||||
if (ret)
|
if (ret)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
pci_set_platform_pm(&acpi_pci_platform_pm);
|
pci_set_platform_pm(&acpi_pci_platform_pm);
|
||||||
|
acpi_pci_slot_init();
|
||||||
|
acpiphp_init();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
arch_initcall(acpi_pci_init);
|
arch_initcall(acpi_pci_init);
|
||||||
|
|
|
@ -897,7 +897,7 @@ int pci_mmap_fits(struct pci_dev *pdev, int resno, struct vm_area_struct *vma,
|
||||||
|
|
||||||
if (pci_resource_len(pdev, resno) == 0)
|
if (pci_resource_len(pdev, resno) == 0)
|
||||||
return 0;
|
return 0;
|
||||||
nr = (vma->vm_end - vma->vm_start) >> PAGE_SHIFT;
|
nr = vma_pages(vma);
|
||||||
start = vma->vm_pgoff;
|
start = vma->vm_pgoff;
|
||||||
size = ((pci_resource_len(pdev, resno) - 1) >> PAGE_SHIFT) + 1;
|
size = ((pci_resource_len(pdev, resno) - 1) >> PAGE_SHIFT) + 1;
|
||||||
pci_start = (mmap_api == PCI_MMAP_PROCFS) ?
|
pci_start = (mmap_api == PCI_MMAP_PROCFS) ?
|
||||||
|
|
|
@ -646,15 +646,11 @@ static int pci_platform_power_transition(struct pci_dev *dev, pci_power_t state)
|
||||||
error = platform_pci_set_power_state(dev, state);
|
error = platform_pci_set_power_state(dev, state);
|
||||||
if (!error)
|
if (!error)
|
||||||
pci_update_current_state(dev, state);
|
pci_update_current_state(dev, state);
|
||||||
/* Fall back to PCI_D0 if native PM is not supported */
|
} else
|
||||||
if (!dev->pm_cap)
|
|
||||||
dev->current_state = PCI_D0;
|
|
||||||
} else {
|
|
||||||
error = -ENODEV;
|
error = -ENODEV;
|
||||||
/* Fall back to PCI_D0 if native PM is not supported */
|
|
||||||
if (!dev->pm_cap)
|
if (error && !dev->pm_cap) /* Fall back to PCI_D0 */
|
||||||
dev->current_state = PCI_D0;
|
dev->current_state = PCI_D0;
|
||||||
}
|
|
||||||
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
@ -1575,7 +1571,7 @@ void pci_pme_active(struct pci_dev *dev, bool enable)
|
||||||
{
|
{
|
||||||
u16 pmcsr;
|
u16 pmcsr;
|
||||||
|
|
||||||
if (!dev->pm_cap)
|
if (!dev->pme_support)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
pci_read_config_word(dev, dev->pm_cap + PCI_PM_CTRL, &pmcsr);
|
pci_read_config_word(dev, dev->pm_cap + PCI_PM_CTRL, &pmcsr);
|
||||||
|
@ -1924,6 +1920,7 @@ void pci_pm_init(struct pci_dev *dev)
|
||||||
dev->wakeup_prepared = false;
|
dev->wakeup_prepared = false;
|
||||||
|
|
||||||
dev->pm_cap = 0;
|
dev->pm_cap = 0;
|
||||||
|
dev->pme_support = 0;
|
||||||
|
|
||||||
/* find PCI PM capability in list */
|
/* find PCI PM capability in list */
|
||||||
pm = pci_find_capability(dev, PCI_CAP_ID_PM);
|
pm = pci_find_capability(dev, PCI_CAP_ID_PM);
|
||||||
|
@ -1975,8 +1972,6 @@ void pci_pm_init(struct pci_dev *dev)
|
||||||
device_set_wakeup_capable(&dev->dev, true);
|
device_set_wakeup_capable(&dev->dev, true);
|
||||||
/* Disable the PME# generation functionality */
|
/* Disable the PME# generation functionality */
|
||||||
pci_pme_active(dev, false);
|
pci_pme_active(dev, false);
|
||||||
} else {
|
|
||||||
dev->pme_support = 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2619,7 +2614,7 @@ void pci_release_selected_regions(struct pci_dev *pdev, int bars)
|
||||||
pci_release_region(pdev, i);
|
pci_release_region(pdev, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
int __pci_request_selected_regions(struct pci_dev *pdev, int bars,
|
static int __pci_request_selected_regions(struct pci_dev *pdev, int bars,
|
||||||
const char *res_name, int excl)
|
const char *res_name, int excl)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
@ -3699,7 +3694,7 @@ static DEFINE_SPINLOCK(resource_alignment_lock);
|
||||||
* RETURNS: Resource alignment if it is specified.
|
* RETURNS: Resource alignment if it is specified.
|
||||||
* Zero if it is not specified.
|
* Zero if it is not specified.
|
||||||
*/
|
*/
|
||||||
resource_size_t pci_specified_resource_alignment(struct pci_dev *dev)
|
static resource_size_t pci_specified_resource_alignment(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
int seg, bus, slot, func, align_order, count;
|
int seg, bus, slot, func, align_order, count;
|
||||||
resource_size_t align = 0;
|
resource_size_t align = 0;
|
||||||
|
@ -3812,7 +3807,7 @@ void pci_reassigndev_resource_alignment(struct pci_dev *dev)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t pci_set_resource_alignment_param(const char *buf, size_t count)
|
static ssize_t pci_set_resource_alignment_param(const char *buf, size_t count)
|
||||||
{
|
{
|
||||||
if (count > RESOURCE_ALIGNMENT_PARAM_SIZE - 1)
|
if (count > RESOURCE_ALIGNMENT_PARAM_SIZE - 1)
|
||||||
count = RESOURCE_ALIGNMENT_PARAM_SIZE - 1;
|
count = RESOURCE_ALIGNMENT_PARAM_SIZE - 1;
|
||||||
|
@ -3823,7 +3818,7 @@ ssize_t pci_set_resource_alignment_param(const char *buf, size_t count)
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t pci_get_resource_alignment_param(char *buf, size_t size)
|
static ssize_t pci_get_resource_alignment_param(char *buf, size_t size)
|
||||||
{
|
{
|
||||||
size_t count;
|
size_t count;
|
||||||
spin_lock(&resource_alignment_lock);
|
spin_lock(&resource_alignment_lock);
|
||||||
|
|
|
@ -8,26 +8,25 @@
|
||||||
|
|
||||||
/* Functions internal to the PCI core code */
|
/* Functions internal to the PCI core code */
|
||||||
|
|
||||||
extern int pci_create_sysfs_dev_files(struct pci_dev *pdev);
|
int pci_create_sysfs_dev_files(struct pci_dev *pdev);
|
||||||
extern void pci_remove_sysfs_dev_files(struct pci_dev *pdev);
|
void pci_remove_sysfs_dev_files(struct pci_dev *pdev);
|
||||||
#if !defined(CONFIG_DMI) && !defined(CONFIG_ACPI)
|
#if !defined(CONFIG_DMI) && !defined(CONFIG_ACPI)
|
||||||
static inline void pci_create_firmware_label_files(struct pci_dev *pdev)
|
static inline void pci_create_firmware_label_files(struct pci_dev *pdev)
|
||||||
{ return; }
|
{ return; }
|
||||||
static inline void pci_remove_firmware_label_files(struct pci_dev *pdev)
|
static inline void pci_remove_firmware_label_files(struct pci_dev *pdev)
|
||||||
{ return; }
|
{ return; }
|
||||||
#else
|
#else
|
||||||
extern void pci_create_firmware_label_files(struct pci_dev *pdev);
|
void pci_create_firmware_label_files(struct pci_dev *pdev);
|
||||||
extern void pci_remove_firmware_label_files(struct pci_dev *pdev);
|
void pci_remove_firmware_label_files(struct pci_dev *pdev);
|
||||||
#endif
|
#endif
|
||||||
extern void pci_cleanup_rom(struct pci_dev *dev);
|
void pci_cleanup_rom(struct pci_dev *dev);
|
||||||
#ifdef HAVE_PCI_MMAP
|
#ifdef HAVE_PCI_MMAP
|
||||||
enum pci_mmap_api {
|
enum pci_mmap_api {
|
||||||
PCI_MMAP_SYSFS, /* mmap on /sys/bus/pci/devices/<BDF>/resource<N> */
|
PCI_MMAP_SYSFS, /* mmap on /sys/bus/pci/devices/<BDF>/resource<N> */
|
||||||
PCI_MMAP_PROCFS /* mmap on /proc/bus/pci/<BDF> */
|
PCI_MMAP_PROCFS /* mmap on /proc/bus/pci/<BDF> */
|
||||||
};
|
};
|
||||||
extern int pci_mmap_fits(struct pci_dev *pdev, int resno,
|
int pci_mmap_fits(struct pci_dev *pdev, int resno, struct vm_area_struct *vmai,
|
||||||
struct vm_area_struct *vmai,
|
enum pci_mmap_api mmap_api);
|
||||||
enum pci_mmap_api mmap_api);
|
|
||||||
#endif
|
#endif
|
||||||
int pci_probe_reset_function(struct pci_dev *dev);
|
int pci_probe_reset_function(struct pci_dev *dev);
|
||||||
|
|
||||||
|
@ -60,17 +59,17 @@ struct pci_platform_pm_ops {
|
||||||
int (*run_wake)(struct pci_dev *dev, bool enable);
|
int (*run_wake)(struct pci_dev *dev, bool enable);
|
||||||
};
|
};
|
||||||
|
|
||||||
extern int pci_set_platform_pm(struct pci_platform_pm_ops *ops);
|
int pci_set_platform_pm(struct pci_platform_pm_ops *ops);
|
||||||
extern void pci_update_current_state(struct pci_dev *dev, pci_power_t state);
|
void pci_update_current_state(struct pci_dev *dev, pci_power_t state);
|
||||||
extern void pci_power_up(struct pci_dev *dev);
|
void pci_power_up(struct pci_dev *dev);
|
||||||
extern void pci_disable_enabled_device(struct pci_dev *dev);
|
void pci_disable_enabled_device(struct pci_dev *dev);
|
||||||
extern int pci_finish_runtime_suspend(struct pci_dev *dev);
|
int pci_finish_runtime_suspend(struct pci_dev *dev);
|
||||||
extern int __pci_pme_wakeup(struct pci_dev *dev, void *ign);
|
int __pci_pme_wakeup(struct pci_dev *dev, void *ign);
|
||||||
extern void pci_wakeup_bus(struct pci_bus *bus);
|
void pci_wakeup_bus(struct pci_bus *bus);
|
||||||
extern void pci_config_pm_runtime_get(struct pci_dev *dev);
|
void pci_config_pm_runtime_get(struct pci_dev *dev);
|
||||||
extern void pci_config_pm_runtime_put(struct pci_dev *dev);
|
void pci_config_pm_runtime_put(struct pci_dev *dev);
|
||||||
extern void pci_pm_init(struct pci_dev *dev);
|
void pci_pm_init(struct pci_dev *dev);
|
||||||
extern void pci_allocate_cap_save_buffers(struct pci_dev *dev);
|
void pci_allocate_cap_save_buffers(struct pci_dev *dev);
|
||||||
void pci_free_cap_save_buffers(struct pci_dev *dev);
|
void pci_free_cap_save_buffers(struct pci_dev *dev);
|
||||||
|
|
||||||
static inline void pci_wakeup_event(struct pci_dev *dev)
|
static inline void pci_wakeup_event(struct pci_dev *dev)
|
||||||
|
@ -96,7 +95,7 @@ struct pci_vpd {
|
||||||
struct bin_attribute *attr; /* descriptor for sysfs VPD entry */
|
struct bin_attribute *attr; /* descriptor for sysfs VPD entry */
|
||||||
};
|
};
|
||||||
|
|
||||||
extern int pci_vpd_pci22_init(struct pci_dev *dev);
|
int pci_vpd_pci22_init(struct pci_dev *dev);
|
||||||
static inline void pci_vpd_release(struct pci_dev *dev)
|
static inline void pci_vpd_release(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
if (dev->vpd)
|
if (dev->vpd)
|
||||||
|
@ -105,9 +104,9 @@ static inline void pci_vpd_release(struct pci_dev *dev)
|
||||||
|
|
||||||
/* PCI /proc functions */
|
/* PCI /proc functions */
|
||||||
#ifdef CONFIG_PROC_FS
|
#ifdef CONFIG_PROC_FS
|
||||||
extern int pci_proc_attach_device(struct pci_dev *dev);
|
int pci_proc_attach_device(struct pci_dev *dev);
|
||||||
extern int pci_proc_detach_device(struct pci_dev *dev);
|
int pci_proc_detach_device(struct pci_dev *dev);
|
||||||
extern int pci_proc_detach_bus(struct pci_bus *bus);
|
int pci_proc_detach_bus(struct pci_bus *bus);
|
||||||
#else
|
#else
|
||||||
static inline int pci_proc_attach_device(struct pci_dev *dev) { return 0; }
|
static inline int pci_proc_attach_device(struct pci_dev *dev) { return 0; }
|
||||||
static inline int pci_proc_detach_device(struct pci_dev *dev) { return 0; }
|
static inline int pci_proc_detach_device(struct pci_dev *dev) { return 0; }
|
||||||
|
@ -118,8 +117,8 @@ static inline int pci_proc_detach_bus(struct pci_bus *bus) { return 0; }
|
||||||
int pci_hp_add_bridge(struct pci_dev *dev);
|
int pci_hp_add_bridge(struct pci_dev *dev);
|
||||||
|
|
||||||
#ifdef HAVE_PCI_LEGACY
|
#ifdef HAVE_PCI_LEGACY
|
||||||
extern void pci_create_legacy_files(struct pci_bus *bus);
|
void pci_create_legacy_files(struct pci_bus *bus);
|
||||||
extern void pci_remove_legacy_files(struct pci_bus *bus);
|
void pci_remove_legacy_files(struct pci_bus *bus);
|
||||||
#else
|
#else
|
||||||
static inline void pci_create_legacy_files(struct pci_bus *bus) { return; }
|
static inline void pci_create_legacy_files(struct pci_bus *bus) { return; }
|
||||||
static inline void pci_remove_legacy_files(struct pci_bus *bus) { return; }
|
static inline void pci_remove_legacy_files(struct pci_bus *bus) { return; }
|
||||||
|
@ -134,7 +133,7 @@ extern unsigned int pci_pm_d3_delay;
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_MSI
|
#ifdef CONFIG_PCI_MSI
|
||||||
void pci_no_msi(void);
|
void pci_no_msi(void);
|
||||||
extern void pci_msi_init_pci_dev(struct pci_dev *dev);
|
void pci_msi_init_pci_dev(struct pci_dev *dev);
|
||||||
#else
|
#else
|
||||||
static inline void pci_no_msi(void) { }
|
static inline void pci_no_msi(void) { }
|
||||||
static inline void pci_msi_init_pci_dev(struct pci_dev *dev) { }
|
static inline void pci_msi_init_pci_dev(struct pci_dev *dev) { }
|
||||||
|
@ -198,12 +197,11 @@ enum pci_bar_type {
|
||||||
|
|
||||||
bool pci_bus_read_dev_vendor_id(struct pci_bus *bus, int devfn, u32 *pl,
|
bool pci_bus_read_dev_vendor_id(struct pci_bus *bus, int devfn, u32 *pl,
|
||||||
int crs_timeout);
|
int crs_timeout);
|
||||||
extern int pci_setup_device(struct pci_dev *dev);
|
int pci_setup_device(struct pci_dev *dev);
|
||||||
extern int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type,
|
int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type,
|
||||||
struct resource *res, unsigned int reg);
|
struct resource *res, unsigned int reg);
|
||||||
extern int pci_resource_bar(struct pci_dev *dev, int resno,
|
int pci_resource_bar(struct pci_dev *dev, int resno, enum pci_bar_type *type);
|
||||||
enum pci_bar_type *type);
|
void pci_configure_ari(struct pci_dev *dev);
|
||||||
extern void pci_configure_ari(struct pci_dev *dev);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* pci_ari_enabled - query ARI forwarding status
|
* pci_ari_enabled - query ARI forwarding status
|
||||||
|
@ -217,7 +215,7 @@ static inline int pci_ari_enabled(struct pci_bus *bus)
|
||||||
}
|
}
|
||||||
|
|
||||||
void pci_reassigndev_resource_alignment(struct pci_dev *dev);
|
void pci_reassigndev_resource_alignment(struct pci_dev *dev);
|
||||||
extern void pci_disable_bridge_window(struct pci_dev *dev);
|
void pci_disable_bridge_window(struct pci_dev *dev);
|
||||||
|
|
||||||
/* Single Root I/O Virtualization */
|
/* Single Root I/O Virtualization */
|
||||||
struct pci_sriov {
|
struct pci_sriov {
|
||||||
|
@ -241,7 +239,7 @@ struct pci_sriov {
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_ATS
|
#ifdef CONFIG_PCI_ATS
|
||||||
extern void pci_restore_ats_state(struct pci_dev *dev);
|
void pci_restore_ats_state(struct pci_dev *dev);
|
||||||
#else
|
#else
|
||||||
static inline void pci_restore_ats_state(struct pci_dev *dev)
|
static inline void pci_restore_ats_state(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
|
@ -249,14 +247,13 @@ static inline void pci_restore_ats_state(struct pci_dev *dev)
|
||||||
#endif /* CONFIG_PCI_ATS */
|
#endif /* CONFIG_PCI_ATS */
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_IOV
|
#ifdef CONFIG_PCI_IOV
|
||||||
extern int pci_iov_init(struct pci_dev *dev);
|
int pci_iov_init(struct pci_dev *dev);
|
||||||
extern void pci_iov_release(struct pci_dev *dev);
|
void pci_iov_release(struct pci_dev *dev);
|
||||||
extern int pci_iov_resource_bar(struct pci_dev *dev, int resno,
|
int pci_iov_resource_bar(struct pci_dev *dev, int resno,
|
||||||
enum pci_bar_type *type);
|
enum pci_bar_type *type);
|
||||||
extern resource_size_t pci_sriov_resource_alignment(struct pci_dev *dev,
|
resource_size_t pci_sriov_resource_alignment(struct pci_dev *dev, int resno);
|
||||||
int resno);
|
void pci_restore_iov_state(struct pci_dev *dev);
|
||||||
extern void pci_restore_iov_state(struct pci_dev *dev);
|
int pci_iov_bus_range(struct pci_bus *bus);
|
||||||
extern int pci_iov_bus_range(struct pci_bus *bus);
|
|
||||||
|
|
||||||
#else
|
#else
|
||||||
static inline int pci_iov_init(struct pci_dev *dev)
|
static inline int pci_iov_init(struct pci_dev *dev)
|
||||||
|
@ -282,10 +279,10 @@ static inline int pci_iov_bus_range(struct pci_bus *bus)
|
||||||
|
|
||||||
#endif /* CONFIG_PCI_IOV */
|
#endif /* CONFIG_PCI_IOV */
|
||||||
|
|
||||||
extern unsigned long pci_cardbus_resource_alignment(struct resource *);
|
unsigned long pci_cardbus_resource_alignment(struct resource *);
|
||||||
|
|
||||||
static inline resource_size_t pci_resource_alignment(struct pci_dev *dev,
|
static inline resource_size_t pci_resource_alignment(struct pci_dev *dev,
|
||||||
struct resource *res)
|
struct resource *res)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_PCI_IOV
|
#ifdef CONFIG_PCI_IOV
|
||||||
int resno = res - dev->resource;
|
int resno = res - dev->resource;
|
||||||
|
@ -298,7 +295,7 @@ static inline resource_size_t pci_resource_alignment(struct pci_dev *dev,
|
||||||
return resource_alignment(res);
|
return resource_alignment(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void pci_enable_acs(struct pci_dev *dev);
|
void pci_enable_acs(struct pci_dev *dev);
|
||||||
|
|
||||||
struct pci_dev_reset_methods {
|
struct pci_dev_reset_methods {
|
||||||
u16 vendor;
|
u16 vendor;
|
||||||
|
@ -307,7 +304,7 @@ struct pci_dev_reset_methods {
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_QUIRKS
|
#ifdef CONFIG_PCI_QUIRKS
|
||||||
extern int pci_dev_specific_reset(struct pci_dev *dev, int probe);
|
int pci_dev_specific_reset(struct pci_dev *dev, int probe);
|
||||||
#else
|
#else
|
||||||
static inline int pci_dev_specific_reset(struct pci_dev *dev, int probe)
|
static inline int pci_dev_specific_reset(struct pci_dev *dev, int probe)
|
||||||
{
|
{
|
||||||
|
|
|
@ -82,4 +82,4 @@ endchoice
|
||||||
|
|
||||||
config PCIE_PME
|
config PCIE_PME
|
||||||
def_bool y
|
def_bool y
|
||||||
depends on PCIEPORTBUS && PM_RUNTIME && ACPI
|
depends on PCIEPORTBUS && PM_RUNTIME
|
||||||
|
|
|
@ -212,8 +212,8 @@ out:
|
||||||
return ops->read(bus, devfn, where, size, val);
|
return ops->read(bus, devfn, where, size, val);
|
||||||
}
|
}
|
||||||
|
|
||||||
int pci_write_aer(struct pci_bus *bus, unsigned int devfn, int where, int size,
|
static int pci_write_aer(struct pci_bus *bus, unsigned int devfn, int where,
|
||||||
u32 val)
|
int size, u32 val)
|
||||||
{
|
{
|
||||||
u32 *sim;
|
u32 *sim;
|
||||||
struct aer_error *err;
|
struct aer_error *err;
|
||||||
|
@ -334,13 +334,13 @@ static int aer_inject(struct aer_error_inj *einj)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
rpdev = pcie_find_root_port(dev);
|
rpdev = pcie_find_root_port(dev);
|
||||||
if (!rpdev) {
|
if (!rpdev) {
|
||||||
ret = -ENOTTY;
|
ret = -ENODEV;
|
||||||
goto out_put;
|
goto out_put;
|
||||||
}
|
}
|
||||||
|
|
||||||
pos_cap_err = pci_find_ext_capability(dev, PCI_EXT_CAP_ID_ERR);
|
pos_cap_err = pci_find_ext_capability(dev, PCI_EXT_CAP_ID_ERR);
|
||||||
if (!pos_cap_err) {
|
if (!pos_cap_err) {
|
||||||
ret = -ENOTTY;
|
ret = -EPERM;
|
||||||
goto out_put;
|
goto out_put;
|
||||||
}
|
}
|
||||||
pci_read_config_dword(dev, pos_cap_err + PCI_ERR_UNCOR_SEVER, &sever);
|
pci_read_config_dword(dev, pos_cap_err + PCI_ERR_UNCOR_SEVER, &sever);
|
||||||
|
@ -350,7 +350,7 @@ static int aer_inject(struct aer_error_inj *einj)
|
||||||
|
|
||||||
rp_pos_cap_err = pci_find_ext_capability(rpdev, PCI_EXT_CAP_ID_ERR);
|
rp_pos_cap_err = pci_find_ext_capability(rpdev, PCI_EXT_CAP_ID_ERR);
|
||||||
if (!rp_pos_cap_err) {
|
if (!rp_pos_cap_err) {
|
||||||
ret = -ENOTTY;
|
ret = -EPERM;
|
||||||
goto out_put;
|
goto out_put;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -110,15 +110,15 @@ static inline pci_ers_result_t merge_result(enum pci_ers_result orig,
|
||||||
}
|
}
|
||||||
|
|
||||||
extern struct bus_type pcie_port_bus_type;
|
extern struct bus_type pcie_port_bus_type;
|
||||||
extern void aer_do_secondary_bus_reset(struct pci_dev *dev);
|
void aer_do_secondary_bus_reset(struct pci_dev *dev);
|
||||||
extern int aer_init(struct pcie_device *dev);
|
int aer_init(struct pcie_device *dev);
|
||||||
extern void aer_isr(struct work_struct *work);
|
void aer_isr(struct work_struct *work);
|
||||||
extern void aer_print_error(struct pci_dev *dev, struct aer_err_info *info);
|
void aer_print_error(struct pci_dev *dev, struct aer_err_info *info);
|
||||||
extern void aer_print_port_info(struct pci_dev *dev, struct aer_err_info *info);
|
void aer_print_port_info(struct pci_dev *dev, struct aer_err_info *info);
|
||||||
extern irqreturn_t aer_irq(int irq, void *context);
|
irqreturn_t aer_irq(int irq, void *context);
|
||||||
|
|
||||||
#ifdef CONFIG_ACPI_APEI
|
#ifdef CONFIG_ACPI_APEI
|
||||||
extern int pcie_aer_get_firmware_first(struct pci_dev *pci_dev);
|
int pcie_aer_get_firmware_first(struct pci_dev *pci_dev);
|
||||||
#else
|
#else
|
||||||
static inline int pcie_aer_get_firmware_first(struct pci_dev *pci_dev)
|
static inline int pcie_aer_get_firmware_first(struct pci_dev *pci_dev)
|
||||||
{
|
{
|
||||||
|
|
|
@ -89,8 +89,6 @@ static int add_error_device(struct aer_err_info *e_info, struct pci_dev *dev)
|
||||||
return -ENOSPC;
|
return -ENOSPC;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define PCI_BUS(x) (((x) >> 8) & 0xff)
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* is_error_source - check whether the device is source of reported error
|
* is_error_source - check whether the device is source of reported error
|
||||||
* @dev: pointer to pci_dev to be checked
|
* @dev: pointer to pci_dev to be checked
|
||||||
|
@ -106,7 +104,7 @@ static bool is_error_source(struct pci_dev *dev, struct aer_err_info *e_info)
|
||||||
* When bus id is equal to 0, it might be a bad id
|
* When bus id is equal to 0, it might be a bad id
|
||||||
* reported by root port.
|
* reported by root port.
|
||||||
*/
|
*/
|
||||||
if (!nosourceid && (PCI_BUS(e_info->id) != 0)) {
|
if (!nosourceid && (PCI_BUS_NUM(e_info->id) != 0)) {
|
||||||
/* Device ID match? */
|
/* Device ID match? */
|
||||||
if (e_info->id == ((dev->bus->number << 8) | dev->devfn))
|
if (e_info->id == ((dev->bus->number << 8) | dev->devfn))
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -19,8 +19,6 @@
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
#include <linux/device.h>
|
#include <linux/device.h>
|
||||||
#include <linux/pcieport_if.h>
|
#include <linux/pcieport_if.h>
|
||||||
#include <linux/acpi.h>
|
|
||||||
#include <linux/pci-acpi.h>
|
|
||||||
#include <linux/pm_runtime.h>
|
#include <linux/pm_runtime.h>
|
||||||
|
|
||||||
#include "../pci.h"
|
#include "../pci.h"
|
||||||
|
|
|
@ -21,18 +21,18 @@
|
||||||
#define get_descriptor_id(type, service) (((type - 4) << 4) | service)
|
#define get_descriptor_id(type, service) (((type - 4) << 4) | service)
|
||||||
|
|
||||||
extern struct bus_type pcie_port_bus_type;
|
extern struct bus_type pcie_port_bus_type;
|
||||||
extern int pcie_port_device_register(struct pci_dev *dev);
|
int pcie_port_device_register(struct pci_dev *dev);
|
||||||
#ifdef CONFIG_PM
|
#ifdef CONFIG_PM
|
||||||
extern int pcie_port_device_suspend(struct device *dev);
|
int pcie_port_device_suspend(struct device *dev);
|
||||||
extern int pcie_port_device_resume(struct device *dev);
|
int pcie_port_device_resume(struct device *dev);
|
||||||
#endif
|
#endif
|
||||||
extern void pcie_port_device_remove(struct pci_dev *dev);
|
void pcie_port_device_remove(struct pci_dev *dev);
|
||||||
extern int __must_check pcie_port_bus_register(void);
|
int __must_check pcie_port_bus_register(void);
|
||||||
extern void pcie_port_bus_unregister(void);
|
void pcie_port_bus_unregister(void);
|
||||||
|
|
||||||
struct pci_dev;
|
struct pci_dev;
|
||||||
|
|
||||||
extern void pcie_clear_root_pme_status(struct pci_dev *dev);
|
void pcie_clear_root_pme_status(struct pci_dev *dev);
|
||||||
|
|
||||||
#ifdef CONFIG_HOTPLUG_PCI_PCIE
|
#ifdef CONFIG_HOTPLUG_PCI_PCIE
|
||||||
extern bool pciehp_msi_disabled;
|
extern bool pciehp_msi_disabled;
|
||||||
|
@ -59,7 +59,7 @@ static inline bool pcie_pme_no_msi(void)
|
||||||
return pcie_pme_msi_disabled;
|
return pcie_pme_msi_disabled;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void pcie_pme_interrupt_enable(struct pci_dev *dev, bool enable);
|
void pcie_pme_interrupt_enable(struct pci_dev *dev, bool enable);
|
||||||
#else /* !CONFIG_PCIE_PME */
|
#else /* !CONFIG_PCIE_PME */
|
||||||
static inline void pcie_pme_disable_msi(void) {}
|
static inline void pcie_pme_disable_msi(void) {}
|
||||||
static inline bool pcie_pme_no_msi(void) { return false; }
|
static inline bool pcie_pme_no_msi(void) { return false; }
|
||||||
|
@ -67,7 +67,7 @@ static inline void pcie_pme_interrupt_enable(struct pci_dev *dev, bool en) {}
|
||||||
#endif /* !CONFIG_PCIE_PME */
|
#endif /* !CONFIG_PCIE_PME */
|
||||||
|
|
||||||
#ifdef CONFIG_ACPI
|
#ifdef CONFIG_ACPI
|
||||||
extern int pcie_port_acpi_setup(struct pci_dev *port, int *mask);
|
int pcie_port_acpi_setup(struct pci_dev *port, int *mask);
|
||||||
|
|
||||||
static inline int pcie_port_platform_notify(struct pci_dev *port, int *mask)
|
static inline int pcie_port_platform_notify(struct pci_dev *port, int *mask)
|
||||||
{
|
{
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
|
|
||||||
#include "aer/aerdrv.h"
|
#include "aer/aerdrv.h"
|
||||||
#include "../pci.h"
|
#include "../pci.h"
|
||||||
|
#include "portdrv.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* pcie_port_acpi_setup - Request the BIOS to release control of PCIe services.
|
* pcie_port_acpi_setup - Request the BIOS to release control of PCIe services.
|
||||||
|
|
|
@ -259,11 +259,9 @@ static pci_ers_result_t pcie_portdrv_error_detected(struct pci_dev *dev,
|
||||||
enum pci_channel_state error)
|
enum pci_channel_state error)
|
||||||
{
|
{
|
||||||
struct aer_broadcast_data data = {error, PCI_ERS_RESULT_CAN_RECOVER};
|
struct aer_broadcast_data data = {error, PCI_ERS_RESULT_CAN_RECOVER};
|
||||||
int ret;
|
|
||||||
|
|
||||||
/* can not fail */
|
|
||||||
ret = device_for_each_child(&dev->dev, &data, error_detected_iter);
|
|
||||||
|
|
||||||
|
/* get true return value from &data */
|
||||||
|
device_for_each_child(&dev->dev, &data, error_detected_iter);
|
||||||
return data.result;
|
return data.result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -295,10 +293,9 @@ static int mmio_enabled_iter(struct device *device, void *data)
|
||||||
static pci_ers_result_t pcie_portdrv_mmio_enabled(struct pci_dev *dev)
|
static pci_ers_result_t pcie_portdrv_mmio_enabled(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
pci_ers_result_t status = PCI_ERS_RESULT_RECOVERED;
|
pci_ers_result_t status = PCI_ERS_RESULT_RECOVERED;
|
||||||
int retval;
|
|
||||||
|
|
||||||
/* get true return value from &status */
|
/* get true return value from &status */
|
||||||
retval = device_for_each_child(&dev->dev, &status, mmio_enabled_iter);
|
device_for_each_child(&dev->dev, &status, mmio_enabled_iter);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -330,7 +327,6 @@ static int slot_reset_iter(struct device *device, void *data)
|
||||||
static pci_ers_result_t pcie_portdrv_slot_reset(struct pci_dev *dev)
|
static pci_ers_result_t pcie_portdrv_slot_reset(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
pci_ers_result_t status = PCI_ERS_RESULT_RECOVERED;
|
pci_ers_result_t status = PCI_ERS_RESULT_RECOVERED;
|
||||||
int retval;
|
|
||||||
|
|
||||||
/* If fatal, restore cfg space for possible link reset at upstream */
|
/* If fatal, restore cfg space for possible link reset at upstream */
|
||||||
if (dev->error_state == pci_channel_io_frozen) {
|
if (dev->error_state == pci_channel_io_frozen) {
|
||||||
|
@ -341,8 +337,7 @@ static pci_ers_result_t pcie_portdrv_slot_reset(struct pci_dev *dev)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* get true return value from &status */
|
/* get true return value from &status */
|
||||||
retval = device_for_each_child(&dev->dev, &status, slot_reset_iter);
|
device_for_each_child(&dev->dev, &status, slot_reset_iter);
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -368,9 +363,7 @@ static int resume_iter(struct device *device, void *data)
|
||||||
|
|
||||||
static void pcie_portdrv_err_resume(struct pci_dev *dev)
|
static void pcie_portdrv_err_resume(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
int retval;
|
device_for_each_child(&dev->dev, NULL, resume_iter);
|
||||||
/* nothing to do with error value, if it ever happens */
|
|
||||||
retval = device_for_each_child(&dev->dev, NULL, resume_iter);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -673,6 +673,8 @@ add_dev:
|
||||||
ret = device_register(&child->dev);
|
ret = device_register(&child->dev);
|
||||||
WARN_ON(ret < 0);
|
WARN_ON(ret < 0);
|
||||||
|
|
||||||
|
pcibios_add_bus(child);
|
||||||
|
|
||||||
/* Create legacy_io and legacy_mem files for this bus */
|
/* Create legacy_io and legacy_mem files for this bus */
|
||||||
pci_create_legacy_files(child);
|
pci_create_legacy_files(child);
|
||||||
|
|
||||||
|
@ -1627,8 +1629,7 @@ unsigned int pci_scan_child_bus(struct pci_bus *bus)
|
||||||
if (!bus->is_added) {
|
if (!bus->is_added) {
|
||||||
dev_dbg(&bus->dev, "fixups for bus\n");
|
dev_dbg(&bus->dev, "fixups for bus\n");
|
||||||
pcibios_fixup_bus(bus);
|
pcibios_fixup_bus(bus);
|
||||||
if (pci_is_root_bus(bus))
|
bus->is_added = 1;
|
||||||
bus->is_added = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (pass=0; pass < 2; pass++)
|
for (pass=0; pass < 2; pass++)
|
||||||
|
@ -1661,6 +1662,14 @@ int __weak pcibios_root_bridge_prepare(struct pci_host_bridge *bridge)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void __weak pcibios_add_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void __weak pcibios_remove_bus(struct pci_bus *bus)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
struct pci_bus *pci_create_root_bus(struct device *parent, int bus,
|
struct pci_bus *pci_create_root_bus(struct device *parent, int bus,
|
||||||
struct pci_ops *ops, void *sysdata, struct list_head *resources)
|
struct pci_ops *ops, void *sysdata, struct list_head *resources)
|
||||||
{
|
{
|
||||||
|
@ -1715,6 +1724,8 @@ struct pci_bus *pci_create_root_bus(struct device *parent, int bus,
|
||||||
if (error)
|
if (error)
|
||||||
goto class_dev_reg_err;
|
goto class_dev_reg_err;
|
||||||
|
|
||||||
|
pcibios_add_bus(b);
|
||||||
|
|
||||||
/* Create legacy_io and legacy_mem files for this bus */
|
/* Create legacy_io and legacy_mem files for this bus */
|
||||||
pci_create_legacy_files(b);
|
pci_create_legacy_files(b);
|
||||||
|
|
||||||
|
|
|
@ -324,29 +324,30 @@ static void quirk_cs5536_vsa(struct pci_dev *dev)
|
||||||
}
|
}
|
||||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_CS5536_ISA, quirk_cs5536_vsa);
|
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_CS5536_ISA, quirk_cs5536_vsa);
|
||||||
|
|
||||||
static void quirk_io_region(struct pci_dev *dev, unsigned region,
|
static void quirk_io_region(struct pci_dev *dev, int port,
|
||||||
unsigned size, int nr, const char *name)
|
unsigned size, int nr, const char *name)
|
||||||
{
|
{
|
||||||
region &= ~(size-1);
|
u16 region;
|
||||||
if (region) {
|
struct pci_bus_region bus_region;
|
||||||
struct pci_bus_region bus_region;
|
struct resource *res = dev->resource + nr;
|
||||||
struct resource *res = dev->resource + nr;
|
|
||||||
|
|
||||||
res->name = pci_name(dev);
|
pci_read_config_word(dev, port, ®ion);
|
||||||
res->start = region;
|
region &= ~(size - 1);
|
||||||
res->end = region + size - 1;
|
|
||||||
res->flags = IORESOURCE_IO;
|
|
||||||
|
|
||||||
/* Convert from PCI bus to resource space. */
|
if (!region)
|
||||||
bus_region.start = res->start;
|
return;
|
||||||
bus_region.end = res->end;
|
|
||||||
pcibios_bus_to_resource(dev, res, &bus_region);
|
|
||||||
|
|
||||||
if (pci_claim_resource(dev, nr) == 0)
|
res->name = pci_name(dev);
|
||||||
dev_info(&dev->dev, "quirk: %pR claimed by %s\n",
|
res->flags = IORESOURCE_IO;
|
||||||
res, name);
|
|
||||||
}
|
/* Convert from PCI bus to resource space */
|
||||||
}
|
bus_region.start = region;
|
||||||
|
bus_region.end = region + size - 1;
|
||||||
|
pcibios_bus_to_resource(dev, res, &bus_region);
|
||||||
|
|
||||||
|
if (!pci_claim_resource(dev, nr))
|
||||||
|
dev_info(&dev->dev, "quirk: %pR claimed by %s\n", res, name);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* ATI Northbridge setups MCE the processor if you even
|
* ATI Northbridge setups MCE the processor if you even
|
||||||
|
@ -374,12 +375,8 @@ DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATI, PCI_DEVICE_ID_ATI_RS100, quirk_ati_
|
||||||
*/
|
*/
|
||||||
static void quirk_ali7101_acpi(struct pci_dev *dev)
|
static void quirk_ali7101_acpi(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
u16 region;
|
quirk_io_region(dev, 0xE0, 64, PCI_BRIDGE_RESOURCES, "ali7101 ACPI");
|
||||||
|
quirk_io_region(dev, 0xE2, 32, PCI_BRIDGE_RESOURCES+1, "ali7101 SMB");
|
||||||
pci_read_config_word(dev, 0xE0, ®ion);
|
|
||||||
quirk_io_region(dev, region, 64, PCI_BRIDGE_RESOURCES, "ali7101 ACPI");
|
|
||||||
pci_read_config_word(dev, 0xE2, ®ion);
|
|
||||||
quirk_io_region(dev, region, 32, PCI_BRIDGE_RESOURCES+1, "ali7101 SMB");
|
|
||||||
}
|
}
|
||||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_AL, PCI_DEVICE_ID_AL_M7101, quirk_ali7101_acpi);
|
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_AL, PCI_DEVICE_ID_AL_M7101, quirk_ali7101_acpi);
|
||||||
|
|
||||||
|
@ -442,12 +439,10 @@ static void piix4_mem_quirk(struct pci_dev *dev, const char *name, unsigned int
|
||||||
*/
|
*/
|
||||||
static void quirk_piix4_acpi(struct pci_dev *dev)
|
static void quirk_piix4_acpi(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
u32 region, res_a;
|
u32 res_a;
|
||||||
|
|
||||||
pci_read_config_dword(dev, 0x40, ®ion);
|
quirk_io_region(dev, 0x40, 64, PCI_BRIDGE_RESOURCES, "PIIX4 ACPI");
|
||||||
quirk_io_region(dev, region, 64, PCI_BRIDGE_RESOURCES, "PIIX4 ACPI");
|
quirk_io_region(dev, 0x90, 16, PCI_BRIDGE_RESOURCES+1, "PIIX4 SMB");
|
||||||
pci_read_config_dword(dev, 0x90, ®ion);
|
|
||||||
quirk_io_region(dev, region, 16, PCI_BRIDGE_RESOURCES+1, "PIIX4 SMB");
|
|
||||||
|
|
||||||
/* Device resource A has enables for some of the other ones */
|
/* Device resource A has enables for some of the other ones */
|
||||||
pci_read_config_dword(dev, 0x5c, &res_a);
|
pci_read_config_dword(dev, 0x5c, &res_a);
|
||||||
|
@ -491,7 +486,6 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82443MX_3, qui
|
||||||
*/
|
*/
|
||||||
static void quirk_ich4_lpc_acpi(struct pci_dev *dev)
|
static void quirk_ich4_lpc_acpi(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
u32 region;
|
|
||||||
u8 enable;
|
u8 enable;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -503,22 +497,14 @@ static void quirk_ich4_lpc_acpi(struct pci_dev *dev)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
pci_read_config_byte(dev, ICH_ACPI_CNTL, &enable);
|
pci_read_config_byte(dev, ICH_ACPI_CNTL, &enable);
|
||||||
if (enable & ICH4_ACPI_EN) {
|
if (enable & ICH4_ACPI_EN)
|
||||||
pci_read_config_dword(dev, ICH_PMBASE, ®ion);
|
quirk_io_region(dev, ICH_PMBASE, 128, PCI_BRIDGE_RESOURCES,
|
||||||
region &= PCI_BASE_ADDRESS_IO_MASK;
|
"ICH4 ACPI/GPIO/TCO");
|
||||||
if (region >= PCIBIOS_MIN_IO)
|
|
||||||
quirk_io_region(dev, region, 128, PCI_BRIDGE_RESOURCES,
|
|
||||||
"ICH4 ACPI/GPIO/TCO");
|
|
||||||
}
|
|
||||||
|
|
||||||
pci_read_config_byte(dev, ICH4_GPIO_CNTL, &enable);
|
pci_read_config_byte(dev, ICH4_GPIO_CNTL, &enable);
|
||||||
if (enable & ICH4_GPIO_EN) {
|
if (enable & ICH4_GPIO_EN)
|
||||||
pci_read_config_dword(dev, ICH4_GPIOBASE, ®ion);
|
quirk_io_region(dev, ICH4_GPIOBASE, 64, PCI_BRIDGE_RESOURCES+1,
|
||||||
region &= PCI_BASE_ADDRESS_IO_MASK;
|
"ICH4 GPIO");
|
||||||
if (region >= PCIBIOS_MIN_IO)
|
|
||||||
quirk_io_region(dev, region, 64,
|
|
||||||
PCI_BRIDGE_RESOURCES + 1, "ICH4 GPIO");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801AA_0, quirk_ich4_lpc_acpi);
|
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801AA_0, quirk_ich4_lpc_acpi);
|
||||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801AB_0, quirk_ich4_lpc_acpi);
|
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801AB_0, quirk_ich4_lpc_acpi);
|
||||||
|
@ -533,26 +519,17 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_ESB_1, qui
|
||||||
|
|
||||||
static void ich6_lpc_acpi_gpio(struct pci_dev *dev)
|
static void ich6_lpc_acpi_gpio(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
u32 region;
|
|
||||||
u8 enable;
|
u8 enable;
|
||||||
|
|
||||||
pci_read_config_byte(dev, ICH_ACPI_CNTL, &enable);
|
pci_read_config_byte(dev, ICH_ACPI_CNTL, &enable);
|
||||||
if (enable & ICH6_ACPI_EN) {
|
if (enable & ICH6_ACPI_EN)
|
||||||
pci_read_config_dword(dev, ICH_PMBASE, ®ion);
|
quirk_io_region(dev, ICH_PMBASE, 128, PCI_BRIDGE_RESOURCES,
|
||||||
region &= PCI_BASE_ADDRESS_IO_MASK;
|
"ICH6 ACPI/GPIO/TCO");
|
||||||
if (region >= PCIBIOS_MIN_IO)
|
|
||||||
quirk_io_region(dev, region, 128, PCI_BRIDGE_RESOURCES,
|
|
||||||
"ICH6 ACPI/GPIO/TCO");
|
|
||||||
}
|
|
||||||
|
|
||||||
pci_read_config_byte(dev, ICH6_GPIO_CNTL, &enable);
|
pci_read_config_byte(dev, ICH6_GPIO_CNTL, &enable);
|
||||||
if (enable & ICH6_GPIO_EN) {
|
if (enable & ICH6_GPIO_EN)
|
||||||
pci_read_config_dword(dev, ICH6_GPIOBASE, ®ion);
|
quirk_io_region(dev, ICH6_GPIOBASE, 64, PCI_BRIDGE_RESOURCES+1,
|
||||||
region &= PCI_BASE_ADDRESS_IO_MASK;
|
"ICH6 GPIO");
|
||||||
if (region >= PCIBIOS_MIN_IO)
|
|
||||||
quirk_io_region(dev, region, 64,
|
|
||||||
PCI_BRIDGE_RESOURCES + 1, "ICH6 GPIO");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ich6_lpc_generic_decode(struct pci_dev *dev, unsigned reg, const char *name, int dynsize)
|
static void ich6_lpc_generic_decode(struct pci_dev *dev, unsigned reg, const char *name, int dynsize)
|
||||||
|
@ -650,13 +627,9 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_ICH10_1, qui
|
||||||
*/
|
*/
|
||||||
static void quirk_vt82c586_acpi(struct pci_dev *dev)
|
static void quirk_vt82c586_acpi(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
u32 region;
|
if (dev->revision & 0x10)
|
||||||
|
quirk_io_region(dev, 0x48, 256, PCI_BRIDGE_RESOURCES,
|
||||||
if (dev->revision & 0x10) {
|
"vt82c586 ACPI");
|
||||||
pci_read_config_dword(dev, 0x48, ®ion);
|
|
||||||
region &= PCI_BASE_ADDRESS_IO_MASK;
|
|
||||||
quirk_io_region(dev, region, 256, PCI_BRIDGE_RESOURCES, "vt82c586 ACPI");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C586_3, quirk_vt82c586_acpi);
|
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C586_3, quirk_vt82c586_acpi);
|
||||||
|
|
||||||
|
@ -668,18 +641,12 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C586_3, quirk_vt
|
||||||
*/
|
*/
|
||||||
static void quirk_vt82c686_acpi(struct pci_dev *dev)
|
static void quirk_vt82c686_acpi(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
u16 hm;
|
|
||||||
u32 smb;
|
|
||||||
|
|
||||||
quirk_vt82c586_acpi(dev);
|
quirk_vt82c586_acpi(dev);
|
||||||
|
|
||||||
pci_read_config_word(dev, 0x70, &hm);
|
quirk_io_region(dev, 0x70, 128, PCI_BRIDGE_RESOURCES+1,
|
||||||
hm &= PCI_BASE_ADDRESS_IO_MASK;
|
"vt82c686 HW-mon");
|
||||||
quirk_io_region(dev, hm, 128, PCI_BRIDGE_RESOURCES + 1, "vt82c686 HW-mon");
|
|
||||||
|
|
||||||
pci_read_config_dword(dev, 0x90, &smb);
|
quirk_io_region(dev, 0x90, 16, PCI_BRIDGE_RESOURCES+2, "vt82c686 SMB");
|
||||||
smb &= PCI_BASE_ADDRESS_IO_MASK;
|
|
||||||
quirk_io_region(dev, smb, 16, PCI_BRIDGE_RESOURCES + 2, "vt82c686 SMB");
|
|
||||||
}
|
}
|
||||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C686_4, quirk_vt82c686_acpi);
|
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C686_4, quirk_vt82c686_acpi);
|
||||||
|
|
||||||
|
@ -690,15 +657,8 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C686_4, quirk_vt
|
||||||
*/
|
*/
|
||||||
static void quirk_vt8235_acpi(struct pci_dev *dev)
|
static void quirk_vt8235_acpi(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
u16 pm, smb;
|
quirk_io_region(dev, 0x88, 128, PCI_BRIDGE_RESOURCES, "vt8235 PM");
|
||||||
|
quirk_io_region(dev, 0xd0, 16, PCI_BRIDGE_RESOURCES+1, "vt8235 SMB");
|
||||||
pci_read_config_word(dev, 0x88, &pm);
|
|
||||||
pm &= PCI_BASE_ADDRESS_IO_MASK;
|
|
||||||
quirk_io_region(dev, pm, 128, PCI_BRIDGE_RESOURCES, "vt8235 PM");
|
|
||||||
|
|
||||||
pci_read_config_word(dev, 0xd0, &smb);
|
|
||||||
smb &= PCI_BASE_ADDRESS_IO_MASK;
|
|
||||||
quirk_io_region(dev, smb, 16, PCI_BRIDGE_RESOURCES + 1, "vt8235 SMB");
|
|
||||||
}
|
}
|
||||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8235, quirk_vt8235_acpi);
|
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8235, quirk_vt8235_acpi);
|
||||||
|
|
||||||
|
@ -2594,6 +2554,14 @@ static void quirk_msi_intx_disable_ati_bug(struct pci_dev *dev)
|
||||||
dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG;
|
dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG;
|
||||||
pci_dev_put(p);
|
pci_dev_put(p);
|
||||||
}
|
}
|
||||||
|
static void quirk_msi_intx_disable_qca_bug(struct pci_dev *dev)
|
||||||
|
{
|
||||||
|
/* AR816X/AR817X/E210X MSI is fixed at HW level from revision 0x18 */
|
||||||
|
if (dev->revision < 0x18) {
|
||||||
|
dev_info(&dev->dev, "set MSI_INTX_DISABLE_BUG flag\n");
|
||||||
|
dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG;
|
||||||
|
}
|
||||||
|
}
|
||||||
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_BROADCOM,
|
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_BROADCOM,
|
||||||
PCI_DEVICE_ID_TIGON3_5780,
|
PCI_DEVICE_ID_TIGON3_5780,
|
||||||
quirk_msi_intx_disable_bug);
|
quirk_msi_intx_disable_bug);
|
||||||
|
@ -2643,6 +2611,16 @@ DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0x1073,
|
||||||
quirk_msi_intx_disable_bug);
|
quirk_msi_intx_disable_bug);
|
||||||
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0x1083,
|
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0x1083,
|
||||||
quirk_msi_intx_disable_bug);
|
quirk_msi_intx_disable_bug);
|
||||||
|
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0x1090,
|
||||||
|
quirk_msi_intx_disable_qca_bug);
|
||||||
|
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0x1091,
|
||||||
|
quirk_msi_intx_disable_qca_bug);
|
||||||
|
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0x10a0,
|
||||||
|
quirk_msi_intx_disable_qca_bug);
|
||||||
|
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0x10a1,
|
||||||
|
quirk_msi_intx_disable_qca_bug);
|
||||||
|
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATTANSIC, 0xe091,
|
||||||
|
quirk_msi_intx_disable_qca_bug);
|
||||||
#endif /* CONFIG_PCI_MSI */
|
#endif /* CONFIG_PCI_MSI */
|
||||||
|
|
||||||
/* Allow manual resource allocation for PCI hotplug bridges
|
/* Allow manual resource allocation for PCI hotplug bridges
|
||||||
|
|
|
@ -50,10 +50,8 @@ void pci_remove_bus(struct pci_bus *bus)
|
||||||
list_del(&bus->node);
|
list_del(&bus->node);
|
||||||
pci_bus_release_busn_res(bus);
|
pci_bus_release_busn_res(bus);
|
||||||
up_write(&pci_bus_sem);
|
up_write(&pci_bus_sem);
|
||||||
if (!bus->is_added)
|
|
||||||
return;
|
|
||||||
|
|
||||||
pci_remove_legacy_files(bus);
|
pci_remove_legacy_files(bus);
|
||||||
|
pcibios_remove_bus(bus);
|
||||||
device_unregister(&bus->dev);
|
device_unregister(&bus->dev);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(pci_remove_bus);
|
EXPORT_SYMBOL(pci_remove_bus);
|
||||||
|
|
|
@ -1044,7 +1044,7 @@ handle_done:
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
void __ref __pci_bus_size_bridges(struct pci_bus *bus,
|
static void __ref __pci_bus_size_bridges(struct pci_bus *bus,
|
||||||
struct list_head *realloc_head)
|
struct list_head *realloc_head)
|
||||||
{
|
{
|
||||||
struct pci_dev *dev;
|
struct pci_dev *dev;
|
||||||
|
@ -1545,6 +1545,8 @@ again:
|
||||||
|
|
||||||
enable_all:
|
enable_all:
|
||||||
retval = pci_reenable_device(bridge);
|
retval = pci_reenable_device(bridge);
|
||||||
|
if (retval)
|
||||||
|
dev_err(&bridge->dev, "Error reenabling bridge (%d)\n", retval);
|
||||||
pci_set_master(bridge);
|
pci_set_master(bridge);
|
||||||
pci_enable_bridges(parent);
|
pci_enable_bridges(parent);
|
||||||
}
|
}
|
||||||
|
|
|
@ -261,7 +261,6 @@ int pci_assign_resource(struct pci_dev *dev, int resno)
|
||||||
{
|
{
|
||||||
struct resource *res = dev->resource + resno;
|
struct resource *res = dev->resource + resno;
|
||||||
resource_size_t align, size;
|
resource_size_t align, size;
|
||||||
struct pci_bus *bus;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
align = pci_resource_alignment(dev, res);
|
align = pci_resource_alignment(dev, res);
|
||||||
|
@ -271,7 +270,6 @@ int pci_assign_resource(struct pci_dev *dev, int resno)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
bus = dev->bus;
|
|
||||||
size = resource_size(res);
|
size = resource_size(res);
|
||||||
ret = _pci_assign_resource(dev, resno, size, align);
|
ret = _pci_assign_resource(dev, resno, size, align);
|
||||||
|
|
||||||
|
|
|
@ -377,14 +377,17 @@ void pci_hp_create_module_link(struct pci_slot *pci_slot)
|
||||||
{
|
{
|
||||||
struct hotplug_slot *slot = pci_slot->hotplug;
|
struct hotplug_slot *slot = pci_slot->hotplug;
|
||||||
struct kobject *kobj = NULL;
|
struct kobject *kobj = NULL;
|
||||||
int no_warn;
|
int ret;
|
||||||
|
|
||||||
if (!slot || !slot->ops)
|
if (!slot || !slot->ops)
|
||||||
return;
|
return;
|
||||||
kobj = kset_find_obj(module_kset, slot->ops->mod_name);
|
kobj = kset_find_obj(module_kset, slot->ops->mod_name);
|
||||||
if (!kobj)
|
if (!kobj)
|
||||||
return;
|
return;
|
||||||
no_warn = sysfs_create_link(&pci_slot->kobj, kobj, "module");
|
ret = sysfs_create_link(&pci_slot->kobj, kobj, "module");
|
||||||
|
if (ret)
|
||||||
|
dev_err(&pci_slot->bus->dev, "Error creating sysfs link (%d)\n",
|
||||||
|
ret);
|
||||||
kobject_put(kobj);
|
kobject_put(kobj);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(pci_hp_create_module_link);
|
EXPORT_SYMBOL_GPL(pci_hp_create_module_link);
|
||||||
|
|
|
@ -1488,7 +1488,4 @@ struct megasas_mgmt_info {
|
||||||
int max_index;
|
int max_index;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define msi_control_reg(base) (base + PCI_MSI_FLAGS)
|
|
||||||
#define PCI_MSIX_FLAGS_ENABLE (1 << 15)
|
|
||||||
|
|
||||||
#endif /*LSI_MEGARAID_SAS_H */
|
#endif /*LSI_MEGARAID_SAS_H */
|
||||||
|
|
|
@ -3984,12 +3984,12 @@ static int megasas_probe_one(struct pci_dev *pdev,
|
||||||
if (reset_devices) {
|
if (reset_devices) {
|
||||||
pos = pci_find_capability(pdev, PCI_CAP_ID_MSIX);
|
pos = pci_find_capability(pdev, PCI_CAP_ID_MSIX);
|
||||||
if (pos) {
|
if (pos) {
|
||||||
pci_read_config_word(pdev, msi_control_reg(pos),
|
pci_read_config_word(pdev, pos + PCI_MSIX_FLAGS,
|
||||||
&control);
|
&control);
|
||||||
if (control & PCI_MSIX_FLAGS_ENABLE) {
|
if (control & PCI_MSIX_FLAGS_ENABLE) {
|
||||||
dev_info(&pdev->dev, "resetting MSI-X\n");
|
dev_info(&pdev->dev, "resetting MSI-X\n");
|
||||||
pci_write_config_word(pdev,
|
pci_write_config_word(pdev,
|
||||||
msi_control_reg(pos),
|
pos + PCI_MSIX_FLAGS,
|
||||||
control &
|
control &
|
||||||
~PCI_MSIX_FLAGS_ENABLE);
|
~PCI_MSIX_FLAGS_ENABLE);
|
||||||
}
|
}
|
||||||
|
|
|
@ -703,7 +703,7 @@ static struct pci_device_id mvs_pci_table[] = {
|
||||||
{ PCI_VDEVICE(TTI, 0x2744), chip_9480 },
|
{ PCI_VDEVICE(TTI, 0x2744), chip_9480 },
|
||||||
{ PCI_VDEVICE(TTI, 0x2760), chip_9480 },
|
{ PCI_VDEVICE(TTI, 0x2760), chip_9480 },
|
||||||
{
|
{
|
||||||
.vendor = 0x1b4b,
|
.vendor = PCI_VENDOR_ID_MARVELL_EXT,
|
||||||
.device = 0x9480,
|
.device = 0x9480,
|
||||||
.subvendor = PCI_ANY_ID,
|
.subvendor = PCI_ANY_ID,
|
||||||
.subdevice = 0x9480,
|
.subdevice = 0x9480,
|
||||||
|
@ -712,7 +712,7 @@ static struct pci_device_id mvs_pci_table[] = {
|
||||||
.driver_data = chip_9480,
|
.driver_data = chip_9480,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
.vendor = 0x1b4b,
|
.vendor = PCI_VENDOR_ID_MARVELL_EXT,
|
||||||
.device = 0x9445,
|
.device = 0x9445,
|
||||||
.subvendor = PCI_ANY_ID,
|
.subvendor = PCI_ANY_ID,
|
||||||
.subdevice = 0x9480,
|
.subdevice = 0x9480,
|
||||||
|
@ -721,7 +721,7 @@ static struct pci_device_id mvs_pci_table[] = {
|
||||||
.driver_data = chip_9445,
|
.driver_data = chip_9445,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
.vendor = 0x1b4b,
|
.vendor = PCI_VENDOR_ID_MARVELL_EXT,
|
||||||
.device = 0x9485,
|
.device = 0x9485,
|
||||||
.subvendor = PCI_ANY_ID,
|
.subvendor = PCI_ANY_ID,
|
||||||
.subdevice = 0x9480,
|
.subdevice = 0x9480,
|
||||||
|
|
|
@ -49,8 +49,8 @@ MODULE_AUTHOR("jyli@marvell.com");
|
||||||
MODULE_DESCRIPTION("Marvell UMI Driver");
|
MODULE_DESCRIPTION("Marvell UMI Driver");
|
||||||
|
|
||||||
static DEFINE_PCI_DEVICE_TABLE(mvumi_pci_table) = {
|
static DEFINE_PCI_DEVICE_TABLE(mvumi_pci_table) = {
|
||||||
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_2, PCI_DEVICE_ID_MARVELL_MV9143) },
|
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_EXT, PCI_DEVICE_ID_MARVELL_MV9143) },
|
||||||
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_2, PCI_DEVICE_ID_MARVELL_MV9580) },
|
{ PCI_DEVICE(PCI_VENDOR_ID_MARVELL_EXT, PCI_DEVICE_ID_MARVELL_MV9580) },
|
||||||
{ 0 }
|
{ 0 }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,6 @@
|
||||||
#define VER_BUILD 1500
|
#define VER_BUILD 1500
|
||||||
|
|
||||||
#define MV_DRIVER_NAME "mvumi"
|
#define MV_DRIVER_NAME "mvumi"
|
||||||
#define PCI_VENDOR_ID_MARVELL_2 0x1b4b
|
|
||||||
#define PCI_DEVICE_ID_MARVELL_MV9143 0x9143
|
#define PCI_DEVICE_ID_MARVELL_MV9143 0x9143
|
||||||
#define PCI_DEVICE_ID_MARVELL_MV9580 0x9580
|
#define PCI_DEVICE_ID_MARVELL_MV9580 0x9580
|
||||||
|
|
||||||
|
|
|
@ -70,7 +70,7 @@ static int vfio_pci_enable(struct vfio_pci_device *vdev)
|
||||||
pci_write_config_word(pdev, PCI_COMMAND, cmd);
|
pci_write_config_word(pdev, PCI_COMMAND, cmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
msix_pos = pci_find_capability(pdev, PCI_CAP_ID_MSIX);
|
msix_pos = pdev->msix_cap;
|
||||||
if (msix_pos) {
|
if (msix_pos) {
|
||||||
u16 flags;
|
u16 flags;
|
||||||
u32 table;
|
u32 table;
|
||||||
|
@ -78,8 +78,8 @@ static int vfio_pci_enable(struct vfio_pci_device *vdev)
|
||||||
pci_read_config_word(pdev, msix_pos + PCI_MSIX_FLAGS, &flags);
|
pci_read_config_word(pdev, msix_pos + PCI_MSIX_FLAGS, &flags);
|
||||||
pci_read_config_dword(pdev, msix_pos + PCI_MSIX_TABLE, &table);
|
pci_read_config_dword(pdev, msix_pos + PCI_MSIX_TABLE, &table);
|
||||||
|
|
||||||
vdev->msix_bar = table & PCI_MSIX_FLAGS_BIRMASK;
|
vdev->msix_bar = table & PCI_MSIX_TABLE_BIR;
|
||||||
vdev->msix_offset = table & ~PCI_MSIX_FLAGS_BIRMASK;
|
vdev->msix_offset = table & PCI_MSIX_TABLE_OFFSET;
|
||||||
vdev->msix_size = ((flags & PCI_MSIX_FLAGS_QSIZE) + 1) * 16;
|
vdev->msix_size = ((flags & PCI_MSIX_FLAGS_QSIZE) + 1) * 16;
|
||||||
} else
|
} else
|
||||||
vdev->msix_bar = 0xFF;
|
vdev->msix_bar = 0xFF;
|
||||||
|
@ -183,7 +183,7 @@ static int vfio_pci_get_irq_count(struct vfio_pci_device *vdev, int irq_type)
|
||||||
u8 pos;
|
u8 pos;
|
||||||
u16 flags;
|
u16 flags;
|
||||||
|
|
||||||
pos = pci_find_capability(vdev->pdev, PCI_CAP_ID_MSI);
|
pos = vdev->pdev->msi_cap;
|
||||||
if (pos) {
|
if (pos) {
|
||||||
pci_read_config_word(vdev->pdev,
|
pci_read_config_word(vdev->pdev,
|
||||||
pos + PCI_MSI_FLAGS, &flags);
|
pos + PCI_MSI_FLAGS, &flags);
|
||||||
|
@ -194,7 +194,7 @@ static int vfio_pci_get_irq_count(struct vfio_pci_device *vdev, int irq_type)
|
||||||
u8 pos;
|
u8 pos;
|
||||||
u16 flags;
|
u16 flags;
|
||||||
|
|
||||||
pos = pci_find_capability(vdev->pdev, PCI_CAP_ID_MSIX);
|
pos = vdev->pdev->msix_cap;
|
||||||
if (pos) {
|
if (pos) {
|
||||||
pci_read_config_word(vdev->pdev,
|
pci_read_config_word(vdev->pdev,
|
||||||
pos + PCI_MSIX_FLAGS, &flags);
|
pos + PCI_MSIX_FLAGS, &flags);
|
||||||
|
|
|
@ -152,15 +152,6 @@ void acpi_penalize_isa_irq(int irq, int active);
|
||||||
|
|
||||||
void acpi_pci_irq_disable (struct pci_dev *dev);
|
void acpi_pci_irq_disable (struct pci_dev *dev);
|
||||||
|
|
||||||
struct acpi_pci_driver {
|
|
||||||
struct list_head node;
|
|
||||||
int (*add)(struct acpi_pci_root *root);
|
|
||||||
void (*remove)(struct acpi_pci_root *root);
|
|
||||||
};
|
|
||||||
|
|
||||||
int acpi_pci_register_driver(struct acpi_pci_driver *driver);
|
|
||||||
void acpi_pci_unregister_driver(struct acpi_pci_driver *driver);
|
|
||||||
|
|
||||||
extern int ec_read(u8 addr, u8 *val);
|
extern int ec_read(u8 addr, u8 *val);
|
||||||
extern int ec_write(u8 addr, u8 val);
|
extern int ec_write(u8 addr, u8 val);
|
||||||
extern int ec_transaction(u8 command,
|
extern int ec_transaction(u8 command,
|
||||||
|
|
|
@ -13,14 +13,14 @@ struct msi_msg {
|
||||||
/* Helper functions */
|
/* Helper functions */
|
||||||
struct irq_data;
|
struct irq_data;
|
||||||
struct msi_desc;
|
struct msi_desc;
|
||||||
extern void mask_msi_irq(struct irq_data *data);
|
void mask_msi_irq(struct irq_data *data);
|
||||||
extern void unmask_msi_irq(struct irq_data *data);
|
void unmask_msi_irq(struct irq_data *data);
|
||||||
extern void __read_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
void __read_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
||||||
extern void __get_cached_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
void __get_cached_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
||||||
extern void __write_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
void __write_msi_msg(struct msi_desc *entry, struct msi_msg *msg);
|
||||||
extern void read_msi_msg(unsigned int irq, struct msi_msg *msg);
|
void read_msi_msg(unsigned int irq, struct msi_msg *msg);
|
||||||
extern void get_cached_msi_msg(unsigned int irq, struct msi_msg *msg);
|
void get_cached_msi_msg(unsigned int irq, struct msi_msg *msg);
|
||||||
extern void write_msi_msg(unsigned int irq, struct msi_msg *msg);
|
void write_msi_msg(unsigned int irq, struct msi_msg *msg);
|
||||||
|
|
||||||
struct msi_desc {
|
struct msi_desc {
|
||||||
struct {
|
struct {
|
||||||
|
@ -54,9 +54,8 @@ struct msi_desc {
|
||||||
*/
|
*/
|
||||||
int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc);
|
int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc);
|
||||||
void arch_teardown_msi_irq(unsigned int irq);
|
void arch_teardown_msi_irq(unsigned int irq);
|
||||||
extern int arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type);
|
int arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type);
|
||||||
extern void arch_teardown_msi_irqs(struct pci_dev *dev);
|
void arch_teardown_msi_irqs(struct pci_dev *dev);
|
||||||
extern int arch_msi_check_device(struct pci_dev* dev, int nvec, int type);
|
int arch_msi_check_device(struct pci_dev* dev, int nvec, int type);
|
||||||
|
|
||||||
|
|
||||||
#endif /* LINUX_MSI_H */
|
#endif /* LINUX_MSI_H */
|
||||||
|
|
|
@ -41,8 +41,37 @@ static inline acpi_handle acpi_pci_get_bridge_handle(struct pci_bus *pbus)
|
||||||
|
|
||||||
return DEVICE_ACPI_HANDLE(dev);
|
return DEVICE_ACPI_HANDLE(dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void acpi_pci_add_bus(struct pci_bus *bus);
|
||||||
|
void acpi_pci_remove_bus(struct pci_bus *bus);
|
||||||
|
|
||||||
|
#ifdef CONFIG_ACPI_PCI_SLOT
|
||||||
|
void acpi_pci_slot_init(void);
|
||||||
|
void acpi_pci_slot_enumerate(struct pci_bus *bus, acpi_handle handle);
|
||||||
|
void acpi_pci_slot_remove(struct pci_bus *bus);
|
||||||
|
#else
|
||||||
|
static inline void acpi_pci_slot_init(void) { }
|
||||||
|
static inline void acpi_pci_slot_enumerate(struct pci_bus *bus,
|
||||||
|
acpi_handle handle) { }
|
||||||
|
static inline void acpi_pci_slot_remove(struct pci_bus *bus) { }
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_HOTPLUG_PCI_ACPI
|
||||||
|
void acpiphp_init(void);
|
||||||
|
void acpiphp_enumerate_slots(struct pci_bus *bus, acpi_handle handle);
|
||||||
|
void acpiphp_remove_slots(struct pci_bus *bus);
|
||||||
|
#else
|
||||||
|
static inline void acpiphp_init(void) { }
|
||||||
|
static inline void acpiphp_enumerate_slots(struct pci_bus *bus,
|
||||||
|
acpi_handle handle) { }
|
||||||
|
static inline void acpiphp_remove_slots(struct pci_bus *bus) { }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#else /* CONFIG_ACPI */
|
||||||
|
static inline void acpi_pci_add_bus(struct pci_bus *bus) { }
|
||||||
|
static inline void acpi_pci_remove_bus(struct pci_bus *bus) { }
|
||||||
|
#endif /* CONFIG_ACPI */
|
||||||
|
|
||||||
#ifdef CONFIG_ACPI_APEI
|
#ifdef CONFIG_ACPI_APEI
|
||||||
extern bool aer_acpi_firmware_first(void);
|
extern bool aer_acpi_firmware_first(void);
|
||||||
#else
|
#else
|
||||||
|
|
|
@ -23,14 +23,14 @@
|
||||||
#define PCIE_LINK_STATE_CLKPM 4
|
#define PCIE_LINK_STATE_CLKPM 4
|
||||||
|
|
||||||
#ifdef CONFIG_PCIEASPM
|
#ifdef CONFIG_PCIEASPM
|
||||||
extern void pcie_aspm_init_link_state(struct pci_dev *pdev);
|
void pcie_aspm_init_link_state(struct pci_dev *pdev);
|
||||||
extern void pcie_aspm_exit_link_state(struct pci_dev *pdev);
|
void pcie_aspm_exit_link_state(struct pci_dev *pdev);
|
||||||
extern void pcie_aspm_pm_state_change(struct pci_dev *pdev);
|
void pcie_aspm_pm_state_change(struct pci_dev *pdev);
|
||||||
extern void pcie_aspm_powersave_config_link(struct pci_dev *pdev);
|
void pcie_aspm_powersave_config_link(struct pci_dev *pdev);
|
||||||
extern void pci_disable_link_state(struct pci_dev *pdev, int state);
|
void pci_disable_link_state(struct pci_dev *pdev, int state);
|
||||||
extern void pci_disable_link_state_locked(struct pci_dev *pdev, int state);
|
void pci_disable_link_state_locked(struct pci_dev *pdev, int state);
|
||||||
extern void pcie_clear_aspm(struct pci_bus *bus);
|
void pcie_clear_aspm(struct pci_bus *bus);
|
||||||
extern void pcie_no_aspm(void);
|
void pcie_no_aspm(void);
|
||||||
#else
|
#else
|
||||||
static inline void pcie_aspm_init_link_state(struct pci_dev *pdev)
|
static inline void pcie_aspm_init_link_state(struct pci_dev *pdev)
|
||||||
{
|
{
|
||||||
|
@ -56,8 +56,8 @@ static inline void pcie_no_aspm(void)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_PCIEASPM_DEBUG /* this depends on CONFIG_PCIEASPM */
|
#ifdef CONFIG_PCIEASPM_DEBUG /* this depends on CONFIG_PCIEASPM */
|
||||||
extern void pcie_aspm_create_sysfs_dev_files(struct pci_dev *pdev);
|
void pcie_aspm_create_sysfs_dev_files(struct pci_dev *pdev);
|
||||||
extern void pcie_aspm_remove_sysfs_dev_files(struct pci_dev *pdev);
|
void pcie_aspm_remove_sysfs_dev_files(struct pci_dev *pdev);
|
||||||
#else
|
#else
|
||||||
static inline void pcie_aspm_create_sysfs_dev_files(struct pci_dev *pdev)
|
static inline void pcie_aspm_create_sysfs_dev_files(struct pci_dev *pdev)
|
||||||
{
|
{
|
||||||
|
|
|
@ -14,9 +14,9 @@ struct pci_ats {
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_ATS
|
#ifdef CONFIG_PCI_ATS
|
||||||
|
|
||||||
extern int pci_enable_ats(struct pci_dev *dev, int ps);
|
int pci_enable_ats(struct pci_dev *dev, int ps);
|
||||||
extern void pci_disable_ats(struct pci_dev *dev);
|
void pci_disable_ats(struct pci_dev *dev);
|
||||||
extern int pci_ats_queue_depth(struct pci_dev *dev);
|
int pci_ats_queue_depth(struct pci_dev *dev);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* pci_ats_enabled - query the ATS status
|
* pci_ats_enabled - query the ATS status
|
||||||
|
@ -54,12 +54,12 @@ static inline int pci_ats_enabled(struct pci_dev *dev)
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_PRI
|
#ifdef CONFIG_PCI_PRI
|
||||||
|
|
||||||
extern int pci_enable_pri(struct pci_dev *pdev, u32 reqs);
|
int pci_enable_pri(struct pci_dev *pdev, u32 reqs);
|
||||||
extern void pci_disable_pri(struct pci_dev *pdev);
|
void pci_disable_pri(struct pci_dev *pdev);
|
||||||
extern bool pci_pri_enabled(struct pci_dev *pdev);
|
bool pci_pri_enabled(struct pci_dev *pdev);
|
||||||
extern int pci_reset_pri(struct pci_dev *pdev);
|
int pci_reset_pri(struct pci_dev *pdev);
|
||||||
extern bool pci_pri_stopped(struct pci_dev *pdev);
|
bool pci_pri_stopped(struct pci_dev *pdev);
|
||||||
extern int pci_pri_status(struct pci_dev *pdev);
|
int pci_pri_status(struct pci_dev *pdev);
|
||||||
|
|
||||||
#else /* CONFIG_PCI_PRI */
|
#else /* CONFIG_PCI_PRI */
|
||||||
|
|
||||||
|
@ -95,10 +95,10 @@ static inline int pci_pri_status(struct pci_dev *pdev)
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_PASID
|
#ifdef CONFIG_PCI_PASID
|
||||||
|
|
||||||
extern int pci_enable_pasid(struct pci_dev *pdev, int features);
|
int pci_enable_pasid(struct pci_dev *pdev, int features);
|
||||||
extern void pci_disable_pasid(struct pci_dev *pdev);
|
void pci_disable_pasid(struct pci_dev *pdev);
|
||||||
extern int pci_pasid_features(struct pci_dev *pdev);
|
int pci_pasid_features(struct pci_dev *pdev);
|
||||||
extern int pci_max_pasids(struct pci_dev *pdev);
|
int pci_max_pasids(struct pci_dev *pdev);
|
||||||
|
|
||||||
#else /* CONFIG_PCI_PASID */
|
#else /* CONFIG_PCI_PASID */
|
||||||
|
|
||||||
|
|
|
@ -35,6 +35,21 @@
|
||||||
/* Include the ID list */
|
/* Include the ID list */
|
||||||
#include <linux/pci_ids.h>
|
#include <linux/pci_ids.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The PCI interface treats multi-function devices as independent
|
||||||
|
* devices. The slot/function address of each device is encoded
|
||||||
|
* in a single byte as follows:
|
||||||
|
*
|
||||||
|
* 7:3 = slot
|
||||||
|
* 2:0 = function
|
||||||
|
* PCI_DEVFN(), PCI_SLOT(), and PCI_FUNC() are defined uapi/linux/pci.h
|
||||||
|
* In the interest of not exposing interfaces to user-space unnecessarily,
|
||||||
|
* the following kernel only defines are being added here.
|
||||||
|
*/
|
||||||
|
#define PCI_DEVID(bus, devfn) ((((u16)bus) << 8) | devfn)
|
||||||
|
/* return bus from PCI devid = ((u16)bus_number) << 8) | devfn */
|
||||||
|
#define PCI_BUS_NUM(x) (((x) >> 8) & 0xff)
|
||||||
|
|
||||||
/* pci_slot represents a physical slot */
|
/* pci_slot represents a physical slot */
|
||||||
struct pci_slot {
|
struct pci_slot {
|
||||||
struct pci_bus *bus; /* The bus this slot is on */
|
struct pci_bus *bus; /* The bus this slot is on */
|
||||||
|
@ -232,6 +247,8 @@ struct pci_dev {
|
||||||
u8 revision; /* PCI revision, low byte of class word */
|
u8 revision; /* PCI revision, low byte of class word */
|
||||||
u8 hdr_type; /* PCI header type (`multi' flag masked out) */
|
u8 hdr_type; /* PCI header type (`multi' flag masked out) */
|
||||||
u8 pcie_cap; /* PCI-E capability offset */
|
u8 pcie_cap; /* PCI-E capability offset */
|
||||||
|
u8 msi_cap; /* MSI capability offset */
|
||||||
|
u8 msix_cap; /* MSI-X capability offset */
|
||||||
u8 pcie_mpss:3; /* PCI-E Max Payload Size Supported */
|
u8 pcie_mpss:3; /* PCI-E Max Payload Size Supported */
|
||||||
u8 rom_base_reg; /* which config register controls the ROM */
|
u8 rom_base_reg; /* which config register controls the ROM */
|
||||||
u8 pin; /* which interrupt pin this device uses */
|
u8 pin; /* which interrupt pin this device uses */
|
||||||
|
@ -249,8 +266,7 @@ struct pci_dev {
|
||||||
pci_power_t current_state; /* Current operating state. In ACPI-speak,
|
pci_power_t current_state; /* Current operating state. In ACPI-speak,
|
||||||
this is D0-D3, D0 being fully functional,
|
this is D0-D3, D0 being fully functional,
|
||||||
and D3 being off. */
|
and D3 being off. */
|
||||||
int pm_cap; /* PM capability offset in the
|
u8 pm_cap; /* PM capability offset */
|
||||||
configuration space */
|
|
||||||
unsigned int pme_support:5; /* Bitmask of states from which PME#
|
unsigned int pme_support:5; /* Bitmask of states from which PME#
|
||||||
can be generated */
|
can be generated */
|
||||||
unsigned int pme_interrupt:1;
|
unsigned int pme_interrupt:1;
|
||||||
|
@ -348,7 +364,7 @@ static inline struct pci_dev *pci_physfn(struct pci_dev *dev)
|
||||||
return dev;
|
return dev;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern struct pci_dev *alloc_pci_dev(void);
|
struct pci_dev *alloc_pci_dev(void);
|
||||||
|
|
||||||
#define to_pci_dev(n) container_of(n, struct pci_dev, dev)
|
#define to_pci_dev(n) container_of(n, struct pci_dev, dev)
|
||||||
#define for_each_pci_dev(d) while ((d = pci_get_device(PCI_ANY_ID, PCI_ANY_ID, d)) != NULL)
|
#define for_each_pci_dev(d) while ((d = pci_get_device(PCI_ANY_ID, PCI_ANY_ID, d)) != NULL)
|
||||||
|
@ -504,10 +520,10 @@ struct pci_ops {
|
||||||
* ACPI needs to be able to access PCI config space before we've done a
|
* ACPI needs to be able to access PCI config space before we've done a
|
||||||
* PCI bus scan and created pci_bus structures.
|
* PCI bus scan and created pci_bus structures.
|
||||||
*/
|
*/
|
||||||
extern int raw_pci_read(unsigned int domain, unsigned int bus,
|
int raw_pci_read(unsigned int domain, unsigned int bus, unsigned int devfn,
|
||||||
unsigned int devfn, int reg, int len, u32 *val);
|
int reg, int len, u32 *val);
|
||||||
extern int raw_pci_write(unsigned int domain, unsigned int bus,
|
int raw_pci_write(unsigned int domain, unsigned int bus, unsigned int devfn,
|
||||||
unsigned int devfn, int reg, int len, u32 val);
|
int reg, int len, u32 val);
|
||||||
|
|
||||||
struct pci_bus_region {
|
struct pci_bus_region {
|
||||||
resource_size_t start;
|
resource_size_t start;
|
||||||
|
@ -658,7 +674,7 @@ struct pci_driver {
|
||||||
/* these external functions are only available when PCI support is enabled */
|
/* these external functions are only available when PCI support is enabled */
|
||||||
#ifdef CONFIG_PCI
|
#ifdef CONFIG_PCI
|
||||||
|
|
||||||
extern void pcie_bus_configure_settings(struct pci_bus *bus, u8 smpss);
|
void pcie_bus_configure_settings(struct pci_bus *bus, u8 smpss);
|
||||||
|
|
||||||
enum pcie_bus_config_types {
|
enum pcie_bus_config_types {
|
||||||
PCIE_BUS_TUNE_OFF,
|
PCIE_BUS_TUNE_OFF,
|
||||||
|
@ -675,9 +691,11 @@ extern struct bus_type pci_bus_type;
|
||||||
* code, or pci core code. */
|
* code, or pci core code. */
|
||||||
extern struct list_head pci_root_buses; /* list of all known PCI buses */
|
extern struct list_head pci_root_buses; /* list of all known PCI buses */
|
||||||
/* Some device drivers need know if pci is initiated */
|
/* Some device drivers need know if pci is initiated */
|
||||||
extern int no_pci_devices(void);
|
int no_pci_devices(void);
|
||||||
|
|
||||||
void pcibios_resource_survey_bus(struct pci_bus *bus);
|
void pcibios_resource_survey_bus(struct pci_bus *bus);
|
||||||
|
void pcibios_add_bus(struct pci_bus *bus);
|
||||||
|
void pcibios_remove_bus(struct pci_bus *bus);
|
||||||
void pcibios_fixup_bus(struct pci_bus *);
|
void pcibios_fixup_bus(struct pci_bus *);
|
||||||
int __must_check pcibios_enable_device(struct pci_dev *, int mask);
|
int __must_check pcibios_enable_device(struct pci_dev *, int mask);
|
||||||
/* Architecture specific versions may override this (weak) */
|
/* Architecture specific versions may override this (weak) */
|
||||||
|
@ -699,7 +717,7 @@ void pcibios_resource_to_bus(struct pci_dev *dev, struct pci_bus_region *region,
|
||||||
void pcibios_bus_to_resource(struct pci_dev *dev, struct resource *res,
|
void pcibios_bus_to_resource(struct pci_dev *dev, struct resource *res,
|
||||||
struct pci_bus_region *region);
|
struct pci_bus_region *region);
|
||||||
void pcibios_scan_specific_bus(int busn);
|
void pcibios_scan_specific_bus(int busn);
|
||||||
extern struct pci_bus *pci_find_bus(int domain, int busnr);
|
struct pci_bus *pci_find_bus(int domain, int busnr);
|
||||||
void pci_bus_add_devices(const struct pci_bus *bus);
|
void pci_bus_add_devices(const struct pci_bus *bus);
|
||||||
struct pci_bus *pci_scan_bus_parented(struct device *parent, int bus,
|
struct pci_bus *pci_scan_bus_parented(struct device *parent, int bus,
|
||||||
struct pci_ops *ops, void *sysdata);
|
struct pci_ops *ops, void *sysdata);
|
||||||
|
@ -732,14 +750,14 @@ struct resource *pci_find_parent_resource(const struct pci_dev *dev,
|
||||||
u8 pci_swizzle_interrupt_pin(const struct pci_dev *dev, u8 pin);
|
u8 pci_swizzle_interrupt_pin(const struct pci_dev *dev, u8 pin);
|
||||||
int pci_get_interrupt_pin(struct pci_dev *dev, struct pci_dev **bridge);
|
int pci_get_interrupt_pin(struct pci_dev *dev, struct pci_dev **bridge);
|
||||||
u8 pci_common_swizzle(struct pci_dev *dev, u8 *pinp);
|
u8 pci_common_swizzle(struct pci_dev *dev, u8 *pinp);
|
||||||
extern struct pci_dev *pci_dev_get(struct pci_dev *dev);
|
struct pci_dev *pci_dev_get(struct pci_dev *dev);
|
||||||
extern void pci_dev_put(struct pci_dev *dev);
|
void pci_dev_put(struct pci_dev *dev);
|
||||||
extern void pci_remove_bus(struct pci_bus *b);
|
void pci_remove_bus(struct pci_bus *b);
|
||||||
extern void pci_stop_and_remove_bus_device(struct pci_dev *dev);
|
void pci_stop_and_remove_bus_device(struct pci_dev *dev);
|
||||||
void pci_stop_root_bus(struct pci_bus *bus);
|
void pci_stop_root_bus(struct pci_bus *bus);
|
||||||
void pci_remove_root_bus(struct pci_bus *bus);
|
void pci_remove_root_bus(struct pci_bus *bus);
|
||||||
void pci_setup_cardbus(struct pci_bus *bus);
|
void pci_setup_cardbus(struct pci_bus *bus);
|
||||||
extern void pci_sort_breadthfirst(void);
|
void pci_sort_breadthfirst(void);
|
||||||
#define dev_is_pci(d) ((d)->bus == &pci_bus_type)
|
#define dev_is_pci(d) ((d)->bus == &pci_bus_type)
|
||||||
#define dev_is_pf(d) ((dev_is_pci(d) ? to_pci_dev(d)->is_physfn : false))
|
#define dev_is_pf(d) ((dev_is_pci(d) ? to_pci_dev(d)->is_physfn : false))
|
||||||
#define dev_num_vf(d) ((dev_is_pci(d) ? pci_num_vf(to_pci_dev(d)) : 0))
|
#define dev_num_vf(d) ((dev_is_pci(d) ? pci_num_vf(to_pci_dev(d)) : 0))
|
||||||
|
@ -1142,18 +1160,17 @@ static inline int pci_msi_enabled(void)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
extern int pci_enable_msi_block(struct pci_dev *dev, unsigned int nvec);
|
int pci_enable_msi_block(struct pci_dev *dev, unsigned int nvec);
|
||||||
extern int pci_enable_msi_block_auto(struct pci_dev *dev, unsigned int *maxvec);
|
int pci_enable_msi_block_auto(struct pci_dev *dev, unsigned int *maxvec);
|
||||||
extern void pci_msi_shutdown(struct pci_dev *dev);
|
void pci_msi_shutdown(struct pci_dev *dev);
|
||||||
extern void pci_disable_msi(struct pci_dev *dev);
|
void pci_disable_msi(struct pci_dev *dev);
|
||||||
extern int pci_msix_table_size(struct pci_dev *dev);
|
int pci_msix_table_size(struct pci_dev *dev);
|
||||||
extern int pci_enable_msix(struct pci_dev *dev,
|
int pci_enable_msix(struct pci_dev *dev, struct msix_entry *entries, int nvec);
|
||||||
struct msix_entry *entries, int nvec);
|
void pci_msix_shutdown(struct pci_dev *dev);
|
||||||
extern void pci_msix_shutdown(struct pci_dev *dev);
|
void pci_disable_msix(struct pci_dev *dev);
|
||||||
extern void pci_disable_msix(struct pci_dev *dev);
|
void msi_remove_pci_irq_vectors(struct pci_dev *dev);
|
||||||
extern void msi_remove_pci_irq_vectors(struct pci_dev *dev);
|
void pci_restore_msi_state(struct pci_dev *dev);
|
||||||
extern void pci_restore_msi_state(struct pci_dev *dev);
|
int pci_msi_enabled(void);
|
||||||
extern int pci_msi_enabled(void);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_PCIEPORTBUS
|
#ifdef CONFIG_PCIEPORTBUS
|
||||||
|
@ -1168,8 +1185,8 @@ extern bool pcie_ports_auto;
|
||||||
static inline int pcie_aspm_enabled(void) { return 0; }
|
static inline int pcie_aspm_enabled(void) { return 0; }
|
||||||
static inline bool pcie_aspm_support_enabled(void) { return false; }
|
static inline bool pcie_aspm_support_enabled(void) { return false; }
|
||||||
#else
|
#else
|
||||||
extern int pcie_aspm_enabled(void);
|
int pcie_aspm_enabled(void);
|
||||||
extern bool pcie_aspm_support_enabled(void);
|
bool pcie_aspm_support_enabled(void);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_PCIEAER
|
#ifdef CONFIG_PCIEAER
|
||||||
|
@ -1187,8 +1204,8 @@ static inline void pcie_set_ecrc_checking(struct pci_dev *dev)
|
||||||
}
|
}
|
||||||
static inline void pcie_ecrc_get_policy(char *str) {};
|
static inline void pcie_ecrc_get_policy(char *str) {};
|
||||||
#else
|
#else
|
||||||
extern void pcie_set_ecrc_checking(struct pci_dev *dev);
|
void pcie_set_ecrc_checking(struct pci_dev *dev);
|
||||||
extern void pcie_ecrc_get_policy(char *str);
|
void pcie_ecrc_get_policy(char *str);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define pci_enable_msi(pdev) pci_enable_msi_block(pdev, 1)
|
#define pci_enable_msi(pdev) pci_enable_msi_block(pdev, 1)
|
||||||
|
@ -1199,9 +1216,9 @@ int ht_create_irq(struct pci_dev *dev, int idx);
|
||||||
void ht_destroy_irq(unsigned int irq);
|
void ht_destroy_irq(unsigned int irq);
|
||||||
#endif /* CONFIG_HT_IRQ */
|
#endif /* CONFIG_HT_IRQ */
|
||||||
|
|
||||||
extern void pci_cfg_access_lock(struct pci_dev *dev);
|
void pci_cfg_access_lock(struct pci_dev *dev);
|
||||||
extern bool pci_cfg_access_trylock(struct pci_dev *dev);
|
bool pci_cfg_access_trylock(struct pci_dev *dev);
|
||||||
extern void pci_cfg_access_unlock(struct pci_dev *dev);
|
void pci_cfg_access_unlock(struct pci_dev *dev);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* PCI domain support. Sometimes called PCI segment (eg by ACPI),
|
* PCI domain support. Sometimes called PCI segment (eg by ACPI),
|
||||||
|
@ -1226,7 +1243,7 @@ static inline int pci_proc_domain(struct pci_bus *bus)
|
||||||
/* some architectures require additional setup to direct VGA traffic */
|
/* some architectures require additional setup to direct VGA traffic */
|
||||||
typedef int (*arch_set_vga_state_t)(struct pci_dev *pdev, bool decode,
|
typedef int (*arch_set_vga_state_t)(struct pci_dev *pdev, bool decode,
|
||||||
unsigned int command_bits, u32 flags);
|
unsigned int command_bits, u32 flags);
|
||||||
extern void pci_register_set_vga_state(arch_set_vga_state_t func);
|
void pci_register_set_vga_state(arch_set_vga_state_t func);
|
||||||
|
|
||||||
#else /* CONFIG_PCI is not enabled */
|
#else /* CONFIG_PCI is not enabled */
|
||||||
|
|
||||||
|
@ -1628,8 +1645,8 @@ int pcibios_set_pcie_reset_state(struct pci_dev *dev,
|
||||||
int pcibios_add_device(struct pci_dev *dev);
|
int pcibios_add_device(struct pci_dev *dev);
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_MMCONFIG
|
#ifdef CONFIG_PCI_MMCONFIG
|
||||||
extern void __init pci_mmcfg_early_init(void);
|
void __init pci_mmcfg_early_init(void);
|
||||||
extern void __init pci_mmcfg_late_init(void);
|
void __init pci_mmcfg_late_init(void);
|
||||||
#else
|
#else
|
||||||
static inline void pci_mmcfg_early_init(void) { }
|
static inline void pci_mmcfg_early_init(void) { }
|
||||||
static inline void pci_mmcfg_late_init(void) { }
|
static inline void pci_mmcfg_late_init(void) { }
|
||||||
|
@ -1640,12 +1657,12 @@ int pci_ext_cfg_avail(void);
|
||||||
void __iomem *pci_ioremap_bar(struct pci_dev *pdev, int bar);
|
void __iomem *pci_ioremap_bar(struct pci_dev *pdev, int bar);
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_IOV
|
#ifdef CONFIG_PCI_IOV
|
||||||
extern int pci_enable_sriov(struct pci_dev *dev, int nr_virtfn);
|
int pci_enable_sriov(struct pci_dev *dev, int nr_virtfn);
|
||||||
extern void pci_disable_sriov(struct pci_dev *dev);
|
void pci_disable_sriov(struct pci_dev *dev);
|
||||||
extern irqreturn_t pci_sriov_migration(struct pci_dev *dev);
|
irqreturn_t pci_sriov_migration(struct pci_dev *dev);
|
||||||
extern int pci_num_vf(struct pci_dev *dev);
|
int pci_num_vf(struct pci_dev *dev);
|
||||||
extern int pci_sriov_set_totalvfs(struct pci_dev *dev, u16 numvfs);
|
int pci_sriov_set_totalvfs(struct pci_dev *dev, u16 numvfs);
|
||||||
extern int pci_sriov_get_totalvfs(struct pci_dev *dev);
|
int pci_sriov_get_totalvfs(struct pci_dev *dev);
|
||||||
#else
|
#else
|
||||||
static inline int pci_enable_sriov(struct pci_dev *dev, int nr_virtfn)
|
static inline int pci_enable_sriov(struct pci_dev *dev, int nr_virtfn)
|
||||||
{
|
{
|
||||||
|
@ -1673,8 +1690,8 @@ static inline int pci_sriov_get_totalvfs(struct pci_dev *dev)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if defined(CONFIG_HOTPLUG_PCI) || defined(CONFIG_HOTPLUG_PCI_MODULE)
|
#if defined(CONFIG_HOTPLUG_PCI) || defined(CONFIG_HOTPLUG_PCI_MODULE)
|
||||||
extern void pci_hp_create_module_link(struct pci_slot *pci_slot);
|
void pci_hp_create_module_link(struct pci_slot *pci_slot);
|
||||||
extern void pci_hp_remove_module_link(struct pci_slot *pci_slot);
|
void pci_hp_remove_module_link(struct pci_slot *pci_slot);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1818,13 +1835,13 @@ int pci_vpd_find_info_keyword(const u8 *buf, unsigned int off,
|
||||||
/* PCI <-> OF binding helpers */
|
/* PCI <-> OF binding helpers */
|
||||||
#ifdef CONFIG_OF
|
#ifdef CONFIG_OF
|
||||||
struct device_node;
|
struct device_node;
|
||||||
extern void pci_set_of_node(struct pci_dev *dev);
|
void pci_set_of_node(struct pci_dev *dev);
|
||||||
extern void pci_release_of_node(struct pci_dev *dev);
|
void pci_release_of_node(struct pci_dev *dev);
|
||||||
extern void pci_set_bus_of_node(struct pci_bus *bus);
|
void pci_set_bus_of_node(struct pci_bus *bus);
|
||||||
extern void pci_release_bus_of_node(struct pci_bus *bus);
|
void pci_release_bus_of_node(struct pci_bus *bus);
|
||||||
|
|
||||||
/* Arch may override this (weak) */
|
/* Arch may override this (weak) */
|
||||||
extern struct device_node * __weak pcibios_get_phb_of_node(struct pci_bus *bus);
|
struct device_node *pcibios_get_phb_of_node(struct pci_bus *bus);
|
||||||
|
|
||||||
static inline struct device_node *
|
static inline struct device_node *
|
||||||
pci_device_to_OF_node(const struct pci_dev *pdev)
|
pci_device_to_OF_node(const struct pci_dev *pdev)
|
||||||
|
|
|
@ -125,12 +125,12 @@ static inline const char *hotplug_slot_name(const struct hotplug_slot *slot)
|
||||||
return pci_slot_name(slot->pci_slot);
|
return pci_slot_name(slot->pci_slot);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern int __pci_hp_register(struct hotplug_slot *slot, struct pci_bus *pbus,
|
int __pci_hp_register(struct hotplug_slot *slot, struct pci_bus *pbus, int nr,
|
||||||
int nr, const char *name,
|
const char *name, struct module *owner,
|
||||||
struct module *owner, const char *mod_name);
|
const char *mod_name);
|
||||||
extern int pci_hp_deregister(struct hotplug_slot *slot);
|
int pci_hp_deregister(struct hotplug_slot *slot);
|
||||||
extern int __must_check pci_hp_change_slot_info (struct hotplug_slot *slot,
|
int __must_check pci_hp_change_slot_info(struct hotplug_slot *slot,
|
||||||
struct hotplug_slot_info *info);
|
struct hotplug_slot_info *info);
|
||||||
|
|
||||||
/* use a define to avoid include chaining to get THIS_MODULE & friends */
|
/* use a define to avoid include chaining to get THIS_MODULE & friends */
|
||||||
#define pci_hp_register(slot, pbus, devnr, name) \
|
#define pci_hp_register(slot, pbus, devnr, name) \
|
||||||
|
|
|
@ -1604,6 +1604,7 @@
|
||||||
#define PCI_SUBDEVICE_ID_KEYSPAN_SX2 0x5334
|
#define PCI_SUBDEVICE_ID_KEYSPAN_SX2 0x5334
|
||||||
|
|
||||||
#define PCI_VENDOR_ID_MARVELL 0x11ab
|
#define PCI_VENDOR_ID_MARVELL 0x11ab
|
||||||
|
#define PCI_VENDOR_ID_MARVELL_EXT 0x1b4b
|
||||||
#define PCI_DEVICE_ID_MARVELL_GT64111 0x4146
|
#define PCI_DEVICE_ID_MARVELL_GT64111 0x4146
|
||||||
#define PCI_DEVICE_ID_MARVELL_GT64260 0x6430
|
#define PCI_DEVICE_ID_MARVELL_GT64260 0x6430
|
||||||
#define PCI_DEVICE_ID_MARVELL_MV64360 0x6460
|
#define PCI_DEVICE_ID_MARVELL_MV64360 0x6460
|
||||||
|
|
|
@ -62,7 +62,7 @@ struct pcie_port_service_driver {
|
||||||
#define to_service_driver(d) \
|
#define to_service_driver(d) \
|
||||||
container_of(d, struct pcie_port_service_driver, driver)
|
container_of(d, struct pcie_port_service_driver, driver)
|
||||||
|
|
||||||
extern int pcie_port_service_register(struct pcie_port_service_driver *new);
|
int pcie_port_service_register(struct pcie_port_service_driver *new);
|
||||||
extern void pcie_port_service_unregister(struct pcie_port_service_driver *new);
|
void pcie_port_service_unregister(struct pcie_port_service_driver *new);
|
||||||
|
|
||||||
#endif /* _PCIEPORT_IF_H_ */
|
#endif /* _PCIEPORT_IF_H_ */
|
||||||
|
|
|
@ -292,12 +292,12 @@
|
||||||
|
|
||||||
/* Message Signalled Interrupts registers */
|
/* Message Signalled Interrupts registers */
|
||||||
|
|
||||||
#define PCI_MSI_FLAGS 2 /* Various flags */
|
#define PCI_MSI_FLAGS 2 /* Message Control */
|
||||||
#define PCI_MSI_FLAGS_64BIT 0x80 /* 64-bit addresses allowed */
|
#define PCI_MSI_FLAGS_ENABLE 0x0001 /* MSI feature enabled */
|
||||||
#define PCI_MSI_FLAGS_QSIZE 0x70 /* Message queue size configured */
|
#define PCI_MSI_FLAGS_QMASK 0x000e /* Maximum queue size available */
|
||||||
#define PCI_MSI_FLAGS_QMASK 0x0e /* Maximum queue size available */
|
#define PCI_MSI_FLAGS_QSIZE 0x0070 /* Message queue size configured */
|
||||||
#define PCI_MSI_FLAGS_ENABLE 0x01 /* MSI feature enabled */
|
#define PCI_MSI_FLAGS_64BIT 0x0080 /* 64-bit addresses allowed */
|
||||||
#define PCI_MSI_FLAGS_MASKBIT 0x100 /* 64-bit mask bits allowed */
|
#define PCI_MSI_FLAGS_MASKBIT 0x0100 /* Per-vector masking capable */
|
||||||
#define PCI_MSI_RFU 3 /* Rest of capability flags */
|
#define PCI_MSI_RFU 3 /* Rest of capability flags */
|
||||||
#define PCI_MSI_ADDRESS_LO 4 /* Lower 32 bits */
|
#define PCI_MSI_ADDRESS_LO 4 /* Lower 32 bits */
|
||||||
#define PCI_MSI_ADDRESS_HI 8 /* Upper 32 bits (if PCI_MSI_FLAGS_64BIT set) */
|
#define PCI_MSI_ADDRESS_HI 8 /* Upper 32 bits (if PCI_MSI_FLAGS_64BIT set) */
|
||||||
|
@ -309,13 +309,17 @@
|
||||||
#define PCI_MSI_PENDING_64 20 /* Pending intrs for 64-bit devices */
|
#define PCI_MSI_PENDING_64 20 /* Pending intrs for 64-bit devices */
|
||||||
|
|
||||||
/* MSI-X registers */
|
/* MSI-X registers */
|
||||||
#define PCI_MSIX_FLAGS 2
|
#define PCI_MSIX_FLAGS 2 /* Message Control */
|
||||||
#define PCI_MSIX_FLAGS_QSIZE 0x7FF
|
#define PCI_MSIX_FLAGS_QSIZE 0x07FF /* Table size */
|
||||||
#define PCI_MSIX_FLAGS_ENABLE (1 << 15)
|
#define PCI_MSIX_FLAGS_MASKALL 0x4000 /* Mask all vectors for this function */
|
||||||
#define PCI_MSIX_FLAGS_MASKALL (1 << 14)
|
#define PCI_MSIX_FLAGS_ENABLE 0x8000 /* MSI-X enable */
|
||||||
#define PCI_MSIX_TABLE 4
|
#define PCI_MSIX_TABLE 4 /* Table offset */
|
||||||
#define PCI_MSIX_PBA 8
|
#define PCI_MSIX_TABLE_BIR 0x00000007 /* BAR index */
|
||||||
#define PCI_MSIX_FLAGS_BIRMASK (7 << 0)
|
#define PCI_MSIX_TABLE_OFFSET 0xfffffff8 /* Offset into specified BAR */
|
||||||
|
#define PCI_MSIX_PBA 8 /* Pending Bit Array offset */
|
||||||
|
#define PCI_MSIX_PBA_BIR 0x00000007 /* BAR index */
|
||||||
|
#define PCI_MSIX_PBA_OFFSET 0xfffffff8 /* Offset into specified BAR */
|
||||||
|
#define PCI_MSIX_FLAGS_BIRMASK (7 << 0) /* deprecated */
|
||||||
#define PCI_CAP_MSIX_SIZEOF 12 /* size of MSIX registers */
|
#define PCI_CAP_MSIX_SIZEOF 12 /* size of MSIX registers */
|
||||||
|
|
||||||
/* MSI-X entry's format */
|
/* MSI-X entry's format */
|
||||||
|
|
Loading…
Reference in New Issue