mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-16 08:44:21 +08:00
Merge master.kernel.org:/pub/scm/linux/kernel/git/gregkh/pci-2.6
This commit is contained in:
commit
7efe5d7c3e
@ -286,7 +286,9 @@ X!Edrivers/pci/search.c
|
||||
-->
|
||||
!Edrivers/pci/msi.c
|
||||
!Edrivers/pci/bus.c
|
||||
!Edrivers/pci/hotplug.c
|
||||
<!-- FIXME: Removed for now since no structured comments in source
|
||||
X!Edrivers/pci/hotplug.c
|
||||
-->
|
||||
!Edrivers/pci/probe.c
|
||||
!Edrivers/pci/rom.c
|
||||
</sect1>
|
||||
|
@ -2,6 +2,8 @@
|
||||
* Exceptions for specific devices. Usually work-arounds for fatal design flaws.
|
||||
*/
|
||||
|
||||
#include <linux/delay.h>
|
||||
#include <linux/dmi.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/init.h>
|
||||
#include "pci.h"
|
||||
@ -384,3 +386,60 @@ static void __devinit pci_fixup_video(struct pci_dev *pdev)
|
||||
}
|
||||
}
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_ANY_ID, PCI_ANY_ID, pci_fixup_video);
|
||||
|
||||
/*
|
||||
* Some Toshiba laptops need extra code to enable their TI TSB43AB22/A.
|
||||
*
|
||||
* We pretend to bring them out of full D3 state, and restore the proper
|
||||
* IRQ, PCI cache line size, and BARs, otherwise the device won't function
|
||||
* properly. In some cases, the device will generate an interrupt on
|
||||
* the wrong IRQ line, causing any devices sharing the the line it's
|
||||
* *supposed* to use to be disabled by the kernel's IRQ debug code.
|
||||
*/
|
||||
static u16 toshiba_line_size;
|
||||
|
||||
static struct dmi_system_id __devinit toshiba_ohci1394_dmi_table[] = {
|
||||
{
|
||||
.ident = "Toshiba PS5 based laptop",
|
||||
.matches = {
|
||||
DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"),
|
||||
DMI_MATCH(DMI_PRODUCT_VERSION, "PS5"),
|
||||
},
|
||||
},
|
||||
{
|
||||
.ident = "Toshiba PSM4 based laptop",
|
||||
.matches = {
|
||||
DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"),
|
||||
DMI_MATCH(DMI_PRODUCT_VERSION, "PSM4"),
|
||||
},
|
||||
},
|
||||
{ }
|
||||
};
|
||||
|
||||
static void __devinit pci_pre_fixup_toshiba_ohci1394(struct pci_dev *dev)
|
||||
{
|
||||
if (!dmi_check_system(toshiba_ohci1394_dmi_table))
|
||||
return; /* only applies to certain Toshibas (so far) */
|
||||
|
||||
dev->current_state = PCI_D3cold;
|
||||
pci_read_config_word(dev, PCI_CACHE_LINE_SIZE, &toshiba_line_size);
|
||||
}
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_TI, 0x8032,
|
||||
pci_pre_fixup_toshiba_ohci1394);
|
||||
|
||||
static void __devinit pci_post_fixup_toshiba_ohci1394(struct pci_dev *dev)
|
||||
{
|
||||
if (!dmi_check_system(toshiba_ohci1394_dmi_table))
|
||||
return; /* only applies to certain Toshibas (so far) */
|
||||
|
||||
/* Restore config space on Toshiba laptops */
|
||||
mdelay(10);
|
||||
pci_write_config_word(dev, PCI_CACHE_LINE_SIZE, toshiba_line_size);
|
||||
pci_write_config_word(dev, PCI_INTERRUPT_LINE, dev->irq);
|
||||
pci_write_config_dword(dev, PCI_BASE_ADDRESS_0,
|
||||
pci_resource_start(dev, 0));
|
||||
pci_write_config_dword(dev, PCI_BASE_ADDRESS_1,
|
||||
pci_resource_start(dev, 1));
|
||||
}
|
||||
DECLARE_PCI_FIXUP_ENABLE(PCI_VENDOR_ID_TI, 0x8032,
|
||||
pci_post_fixup_toshiba_ohci1394);
|
||||
|
@ -60,3 +60,92 @@ EXPORT_SYMBOL(pci_bus_read_config_dword);
|
||||
EXPORT_SYMBOL(pci_bus_write_config_byte);
|
||||
EXPORT_SYMBOL(pci_bus_write_config_word);
|
||||
EXPORT_SYMBOL(pci_bus_write_config_dword);
|
||||
|
||||
static u32 pci_user_cached_config(struct pci_dev *dev, int pos)
|
||||
{
|
||||
u32 data;
|
||||
|
||||
data = dev->saved_config_space[pos/sizeof(dev->saved_config_space[0])];
|
||||
data >>= (pos % sizeof(dev->saved_config_space[0])) * 8;
|
||||
return data;
|
||||
}
|
||||
|
||||
#define PCI_USER_READ_CONFIG(size,type) \
|
||||
int pci_user_read_config_##size \
|
||||
(struct pci_dev *dev, int pos, type *val) \
|
||||
{ \
|
||||
unsigned long flags; \
|
||||
int ret = 0; \
|
||||
u32 data = -1; \
|
||||
if (PCI_##size##_BAD) return PCIBIOS_BAD_REGISTER_NUMBER; \
|
||||
spin_lock_irqsave(&pci_lock, flags); \
|
||||
if (likely(!dev->block_ucfg_access)) \
|
||||
ret = dev->bus->ops->read(dev->bus, dev->devfn, \
|
||||
pos, sizeof(type), &data); \
|
||||
else if (pos < sizeof(dev->saved_config_space)) \
|
||||
data = pci_user_cached_config(dev, pos); \
|
||||
spin_unlock_irqrestore(&pci_lock, flags); \
|
||||
*val = (type)data; \
|
||||
return ret; \
|
||||
}
|
||||
|
||||
#define PCI_USER_WRITE_CONFIG(size,type) \
|
||||
int pci_user_write_config_##size \
|
||||
(struct pci_dev *dev, int pos, type val) \
|
||||
{ \
|
||||
unsigned long flags; \
|
||||
int ret = -EIO; \
|
||||
if (PCI_##size##_BAD) return PCIBIOS_BAD_REGISTER_NUMBER; \
|
||||
spin_lock_irqsave(&pci_lock, flags); \
|
||||
if (likely(!dev->block_ucfg_access)) \
|
||||
ret = dev->bus->ops->write(dev->bus, dev->devfn, \
|
||||
pos, sizeof(type), val); \
|
||||
spin_unlock_irqrestore(&pci_lock, flags); \
|
||||
return ret; \
|
||||
}
|
||||
|
||||
PCI_USER_READ_CONFIG(byte, u8)
|
||||
PCI_USER_READ_CONFIG(word, u16)
|
||||
PCI_USER_READ_CONFIG(dword, u32)
|
||||
PCI_USER_WRITE_CONFIG(byte, u8)
|
||||
PCI_USER_WRITE_CONFIG(word, u16)
|
||||
PCI_USER_WRITE_CONFIG(dword, u32)
|
||||
|
||||
/**
|
||||
* pci_block_user_cfg_access - Block userspace PCI config reads/writes
|
||||
* @dev: pci device struct
|
||||
*
|
||||
* This function blocks any userspace PCI config accesses from occurring.
|
||||
* When blocked, any writes will be bit bucketed and reads will return the
|
||||
* data saved using pci_save_state for the first 64 bytes of config
|
||||
* space and return 0xff for all other config reads.
|
||||
**/
|
||||
void pci_block_user_cfg_access(struct pci_dev *dev)
|
||||
{
|
||||
unsigned long flags;
|
||||
|
||||
pci_save_state(dev);
|
||||
|
||||
/* spinlock to synchronize with anyone reading config space now */
|
||||
spin_lock_irqsave(&pci_lock, flags);
|
||||
dev->block_ucfg_access = 1;
|
||||
spin_unlock_irqrestore(&pci_lock, flags);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(pci_block_user_cfg_access);
|
||||
|
||||
/**
|
||||
* pci_unblock_user_cfg_access - Unblock userspace PCI config reads/writes
|
||||
* @dev: pci device struct
|
||||
*
|
||||
* This function allows userspace PCI config accesses to resume.
|
||||
**/
|
||||
void pci_unblock_user_cfg_access(struct pci_dev *dev)
|
||||
{
|
||||
unsigned long flags;
|
||||
|
||||
/* spinlock to synchronize with anyone reading saved config space */
|
||||
spin_lock_irqsave(&pci_lock, flags);
|
||||
dev->block_ucfg_access = 0;
|
||||
spin_unlock_irqrestore(&pci_lock, flags);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(pci_unblock_user_cfg_access);
|
||||
|
@ -58,6 +58,9 @@ static LIST_HEAD(bridge_list);
|
||||
|
||||
static void handle_hotplug_event_bridge (acpi_handle, u32, void *);
|
||||
static void handle_hotplug_event_func (acpi_handle, u32, void *);
|
||||
static void acpiphp_sanitize_bus(struct pci_bus *bus);
|
||||
static void acpiphp_set_hpp_values(acpi_handle handle, struct pci_bus *bus);
|
||||
|
||||
|
||||
/*
|
||||
* initialization & terminatation routines
|
||||
@ -796,8 +799,13 @@ static int enable_device(struct acpiphp_slot *slot)
|
||||
}
|
||||
}
|
||||
|
||||
pci_bus_size_bridges(bus);
|
||||
pci_bus_assign_resources(bus);
|
||||
acpiphp_sanitize_bus(bus);
|
||||
pci_enable_bridges(bus);
|
||||
pci_bus_add_devices(bus);
|
||||
acpiphp_set_hpp_values(DEVICE_ACPI_HANDLE(&bus->self->dev), bus);
|
||||
acpiphp_configure_ioapics(DEVICE_ACPI_HANDLE(&bus->self->dev));
|
||||
|
||||
/* associate pci_dev to our representation */
|
||||
list_for_each (l, &slot->funcs) {
|
||||
|
@ -78,11 +78,20 @@ static void __iomem *csr_int_mask;
|
||||
|
||||
static int zt5550_hc_config(struct pci_dev *pdev)
|
||||
{
|
||||
int ret;
|
||||
|
||||
/* Since we know that no boards exist with two HC chips, treat it as an error */
|
||||
if(hc_dev) {
|
||||
err("too many host controller devices?");
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
ret = pci_enable_device(pdev);
|
||||
if(ret) {
|
||||
err("cannot enable %s\n", pci_name(pdev));
|
||||
return ret;
|
||||
}
|
||||
|
||||
hc_dev = pdev;
|
||||
dbg("hc_dev = %p", hc_dev);
|
||||
dbg("pci resource start %lx", pci_resource_start(hc_dev, 1));
|
||||
@ -91,7 +100,8 @@ static int zt5550_hc_config(struct pci_dev *pdev)
|
||||
if(!request_mem_region(pci_resource_start(hc_dev, 1),
|
||||
pci_resource_len(hc_dev, 1), MY_NAME)) {
|
||||
err("cannot reserve MMIO region");
|
||||
return -ENOMEM;
|
||||
ret = -ENOMEM;
|
||||
goto exit_disable_device;
|
||||
}
|
||||
|
||||
hc_registers =
|
||||
@ -99,9 +109,8 @@ static int zt5550_hc_config(struct pci_dev *pdev)
|
||||
if(!hc_registers) {
|
||||
err("cannot remap MMIO region %lx @ %lx",
|
||||
pci_resource_len(hc_dev, 1), pci_resource_start(hc_dev, 1));
|
||||
release_mem_region(pci_resource_start(hc_dev, 1),
|
||||
pci_resource_len(hc_dev, 1));
|
||||
return -ENODEV;
|
||||
ret = -ENODEV;
|
||||
goto exit_release_region;
|
||||
}
|
||||
|
||||
csr_hc_index = hc_registers + CSR_HCINDEX;
|
||||
@ -124,6 +133,13 @@ static int zt5550_hc_config(struct pci_dev *pdev)
|
||||
writeb((u8) ALL_DIRECT_INTS_MASK, csr_int_mask);
|
||||
dbg("disabled timer0, timer1 and ENUM interrupts");
|
||||
return 0;
|
||||
|
||||
exit_release_region:
|
||||
release_mem_region(pci_resource_start(hc_dev, 1),
|
||||
pci_resource_len(hc_dev, 1));
|
||||
exit_disable_device:
|
||||
pci_disable_device(hc_dev);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int zt5550_hc_cleanup(void)
|
||||
@ -134,6 +150,7 @@ static int zt5550_hc_cleanup(void)
|
||||
iounmap(hc_registers);
|
||||
release_mem_region(pci_resource_start(hc_dev, 1),
|
||||
pci_resource_len(hc_dev, 1));
|
||||
pci_disable_device(hc_dev);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -794,12 +794,21 @@ static int cpqhpc_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
u32 rc;
|
||||
struct controller *ctrl;
|
||||
struct pci_func *func;
|
||||
int err;
|
||||
|
||||
err = pci_enable_device(pdev);
|
||||
if (err) {
|
||||
printk(KERN_ERR MY_NAME ": cannot enable PCI device %s (%d)\n",
|
||||
pci_name(pdev), err);
|
||||
return err;
|
||||
}
|
||||
|
||||
// Need to read VID early b/c it's used to differentiate CPQ and INTC discovery
|
||||
rc = pci_read_config_word(pdev, PCI_VENDOR_ID, &vendor_id);
|
||||
if (rc || ((vendor_id != PCI_VENDOR_ID_COMPAQ) && (vendor_id != PCI_VENDOR_ID_INTEL))) {
|
||||
err(msg_HPC_non_compaq_or_intel);
|
||||
return -ENODEV;
|
||||
rc = -ENODEV;
|
||||
goto err_disable_device;
|
||||
}
|
||||
dbg("Vendor ID: %x\n", vendor_id);
|
||||
|
||||
@ -807,7 +816,8 @@ static int cpqhpc_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
dbg("revision: %d\n", rev);
|
||||
if (rc || ((vendor_id == PCI_VENDOR_ID_COMPAQ) && (!rev))) {
|
||||
err(msg_HPC_rev_error);
|
||||
return -ENODEV;
|
||||
rc = -ENODEV;
|
||||
goto err_disable_device;
|
||||
}
|
||||
|
||||
/* Check for the proper subsytem ID's
|
||||
@ -820,18 +830,20 @@ static int cpqhpc_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
rc = pci_read_config_word(pdev, PCI_SUBSYSTEM_VENDOR_ID, &subsystem_vid);
|
||||
if (rc) {
|
||||
err("%s : pci_read_config_word failed\n", __FUNCTION__);
|
||||
return rc;
|
||||
goto err_disable_device;
|
||||
}
|
||||
dbg("Subsystem Vendor ID: %x\n", subsystem_vid);
|
||||
if ((subsystem_vid != PCI_VENDOR_ID_COMPAQ) && (subsystem_vid != PCI_VENDOR_ID_INTEL)) {
|
||||
err(msg_HPC_non_compaq_or_intel);
|
||||
return -ENODEV;
|
||||
rc = -ENODEV;
|
||||
goto err_disable_device;
|
||||
}
|
||||
|
||||
ctrl = (struct controller *) kmalloc(sizeof(struct controller), GFP_KERNEL);
|
||||
if (!ctrl) {
|
||||
err("%s : out of memory\n", __FUNCTION__);
|
||||
return -ENOMEM;
|
||||
rc = -ENOMEM;
|
||||
goto err_disable_device;
|
||||
}
|
||||
memset(ctrl, 0, sizeof(struct controller));
|
||||
|
||||
@ -1264,6 +1276,8 @@ err_free_bus:
|
||||
kfree(ctrl->pci_bus);
|
||||
err_free_ctrl:
|
||||
kfree(ctrl);
|
||||
err_disable_device:
|
||||
pci_disable_device(pdev);
|
||||
return rc;
|
||||
}
|
||||
|
||||
|
@ -92,9 +92,10 @@ extern struct pci_bus *rpaphp_find_pci_bus(struct device_node *dn);
|
||||
extern int rpaphp_claim_resource(struct pci_dev *dev, int resource);
|
||||
extern int rpaphp_enable_pci_slot(struct slot *slot);
|
||||
extern int register_pci_slot(struct slot *slot);
|
||||
extern int rpaphp_unconfig_pci_adapter(struct slot *slot);
|
||||
extern int rpaphp_get_pci_adapter_status(struct slot *slot, int is_init, u8 * value);
|
||||
|
||||
extern int rpaphp_config_pci_adapter(struct pci_bus *bus);
|
||||
extern int rpaphp_unconfig_pci_adapter(struct pci_bus *bus);
|
||||
|
||||
/* rpaphp_core.c */
|
||||
extern int rpaphp_add_slot(struct device_node *dn);
|
||||
|
@ -426,8 +426,11 @@ static int disable_slot(struct hotplug_slot *hotplug_slot)
|
||||
|
||||
dbg("DISABLING SLOT %s\n", slot->name);
|
||||
down(&rpaphp_sem);
|
||||
retval = rpaphp_unconfig_pci_adapter(slot);
|
||||
retval = rpaphp_unconfig_pci_adapter(slot->bus);
|
||||
up(&rpaphp_sem);
|
||||
slot->state = NOT_CONFIGURED;
|
||||
info("%s: devices in slot[%s] unconfigured.\n", __FUNCTION__,
|
||||
slot->name);
|
||||
exit:
|
||||
dbg("%s - Exit: rc[%d]\n", __FUNCTION__, retval);
|
||||
return retval;
|
||||
|
@ -319,20 +319,15 @@ static void rpaphp_eeh_remove_bus_device(struct pci_dev *dev)
|
||||
return;
|
||||
}
|
||||
|
||||
int rpaphp_unconfig_pci_adapter(struct slot *slot)
|
||||
int rpaphp_unconfig_pci_adapter(struct pci_bus *bus)
|
||||
{
|
||||
struct pci_dev *dev, *tmp;
|
||||
int retval = 0;
|
||||
|
||||
list_for_each_entry_safe(dev, tmp, slot->pci_devs, bus_list) {
|
||||
list_for_each_entry_safe(dev, tmp, &bus->devices, bus_list) {
|
||||
rpaphp_eeh_remove_bus_device(dev);
|
||||
pci_remove_bus_device(dev);
|
||||
}
|
||||
|
||||
slot->state = NOT_CONFIGURED;
|
||||
info("%s: devices in slot[%s] unconfigured.\n", __FUNCTION__,
|
||||
slot->name);
|
||||
return retval;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int setup_pci_hotplug_slot_info(struct slot *slot)
|
||||
|
@ -32,8 +32,6 @@
|
||||
#include <linux/types.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/delay.h>
|
||||
#include <asm/semaphore.h>
|
||||
#include <asm/io.h>
|
||||
#include "pci_hotplug.h"
|
||||
|
||||
#if !defined(MODULE)
|
||||
@ -52,42 +50,18 @@ extern int shpchp_debug;
|
||||
#define info(format, arg...) printk(KERN_INFO "%s: " format, MY_NAME , ## arg)
|
||||
#define warn(format, arg...) printk(KERN_WARNING "%s: " format, MY_NAME , ## arg)
|
||||
|
||||
struct pci_func {
|
||||
struct pci_func *next;
|
||||
u8 bus;
|
||||
u8 device;
|
||||
u8 function;
|
||||
u8 is_a_board;
|
||||
u16 status;
|
||||
u8 configured;
|
||||
u8 switch_save;
|
||||
u8 presence_save;
|
||||
u8 pwr_save;
|
||||
u32 base_length[0x06];
|
||||
u8 base_type[0x06];
|
||||
u16 reserved2;
|
||||
u32 config_space[0x20];
|
||||
struct pci_resource *mem_head;
|
||||
struct pci_resource *p_mem_head;
|
||||
struct pci_resource *io_head;
|
||||
struct pci_resource *bus_head;
|
||||
struct pci_dev* pci_dev;
|
||||
};
|
||||
|
||||
#define SLOT_MAGIC 0x67267321
|
||||
struct slot {
|
||||
u32 magic;
|
||||
struct slot *next;
|
||||
u8 bus;
|
||||
u8 device;
|
||||
u16 status;
|
||||
u32 number;
|
||||
u8 is_a_board;
|
||||
u8 configured;
|
||||
u8 state;
|
||||
u8 switch_save;
|
||||
u8 presence_save;
|
||||
u32 capabilities;
|
||||
u16 reserved2;
|
||||
u8 pwr_save;
|
||||
struct timer_list task_event;
|
||||
u8 hp_slot;
|
||||
struct controller *ctrl;
|
||||
@ -96,12 +70,6 @@ struct slot {
|
||||
struct list_head slot_list;
|
||||
};
|
||||
|
||||
struct pci_resource {
|
||||
struct pci_resource * next;
|
||||
u32 base;
|
||||
u32 length;
|
||||
};
|
||||
|
||||
struct event_info {
|
||||
u32 event_type;
|
||||
u8 hp_slot;
|
||||
@ -110,13 +78,9 @@ struct event_info {
|
||||
struct controller {
|
||||
struct controller *next;
|
||||
struct semaphore crit_sect; /* critical section semaphore */
|
||||
void * hpc_ctlr_handle; /* HPC controller handle */
|
||||
struct php_ctlr_state_s *hpc_ctlr_handle; /* HPC controller handle */
|
||||
int num_slots; /* Number of slots on ctlr */
|
||||
int slot_num_inc; /* 1 or -1 */
|
||||
struct pci_resource *mem_head;
|
||||
struct pci_resource *p_mem_head;
|
||||
struct pci_resource *io_head;
|
||||
struct pci_resource *bus_head;
|
||||
struct pci_dev *pci_dev;
|
||||
struct pci_bus *pci_bus;
|
||||
struct event_info event_queue[10];
|
||||
@ -124,33 +88,21 @@ struct controller {
|
||||
struct hpc_ops *hpc_ops;
|
||||
wait_queue_head_t queue; /* sleep & wake process */
|
||||
u8 next_event;
|
||||
u8 seg;
|
||||
u8 bus;
|
||||
u8 device;
|
||||
u8 function;
|
||||
u8 rev;
|
||||
u8 slot_device_offset;
|
||||
u8 add_support;
|
||||
enum pci_bus_speed speed;
|
||||
u32 first_slot; /* First physical slot number */
|
||||
u8 slot_bus; /* Bus where the slots handled by this controller sit */
|
||||
u8 push_flag;
|
||||
u16 ctlrcap;
|
||||
u16 vendor_id;
|
||||
};
|
||||
|
||||
struct irq_mapping {
|
||||
u8 barber_pole;
|
||||
u8 valid_INT;
|
||||
u8 interrupt[4];
|
||||
};
|
||||
|
||||
struct resource_lists {
|
||||
struct pci_resource *mem_head;
|
||||
struct pci_resource *p_mem_head;
|
||||
struct pci_resource *io_head;
|
||||
struct pci_resource *bus_head;
|
||||
struct irq_mapping *irqs;
|
||||
struct hotplug_params {
|
||||
u8 cache_line_size;
|
||||
u8 latency_timer;
|
||||
u8 enable_serr;
|
||||
u8 enable_perr;
|
||||
};
|
||||
|
||||
/* Define AMD SHPC ID */
|
||||
@ -194,24 +146,16 @@ struct resource_lists {
|
||||
* error Messages
|
||||
*/
|
||||
#define msg_initialization_err "Initialization failure, error=%d\n"
|
||||
#define msg_HPC_rev_error "Unsupported revision of the PCI hot plug controller found.\n"
|
||||
#define msg_HPC_non_shpc "The PCI hot plug controller is not supported by this driver.\n"
|
||||
#define msg_HPC_not_supported "This system is not supported by this version of shpcphd mdoule. Upgrade to a newer version of shpchpd\n"
|
||||
#define msg_unable_to_save "Unable to store PCI hot plug add resource information. This system must be rebooted before adding any PCI devices.\n"
|
||||
#define msg_button_on "PCI slot #%d - powering on due to button press.\n"
|
||||
#define msg_button_off "PCI slot #%d - powering off due to button press.\n"
|
||||
#define msg_button_cancel "PCI slot #%d - action canceled due to button press.\n"
|
||||
#define msg_button_ignore "PCI slot #%d - button press ignored. (action in progress...)\n"
|
||||
|
||||
/* sysfs functions for the hotplug controller info */
|
||||
extern void shpchp_create_ctrl_files (struct controller *ctrl);
|
||||
|
||||
/* controller functions */
|
||||
extern int shpchprm_find_available_resources(struct controller *ctrl);
|
||||
extern int shpchp_event_start_thread(void);
|
||||
extern void shpchp_event_stop_thread(void);
|
||||
extern struct pci_func *shpchp_slot_create(unsigned char busnumber);
|
||||
extern struct pci_func *shpchp_slot_find(unsigned char bus, unsigned char device, unsigned char index);
|
||||
extern int shpchp_enable_slot(struct slot *slot);
|
||||
extern int shpchp_disable_slot(struct slot *slot);
|
||||
|
||||
@ -220,29 +164,20 @@ extern u8 shpchp_handle_switch_change(u8 hp_slot, void *inst_id);
|
||||
extern u8 shpchp_handle_presence_change(u8 hp_slot, void *inst_id);
|
||||
extern u8 shpchp_handle_power_fault(u8 hp_slot, void *inst_id);
|
||||
|
||||
/* resource functions */
|
||||
extern int shpchp_resource_sort_and_combine(struct pci_resource **head);
|
||||
|
||||
/* pci functions */
|
||||
extern int shpchp_set_irq(u8 bus_num, u8 dev_num, u8 int_pin, u8 irq_num);
|
||||
/*extern int shpchp_get_bus_dev(struct controller *ctrl, u8 *bus_num, u8 *dev_num, struct slot *slot);*/
|
||||
extern int shpchp_save_config(struct controller *ctrl, int busnumber, int num_ctlr_slots, int first_device_num);
|
||||
extern int shpchp_save_used_resources(struct controller *ctrl, struct pci_func * func, int flag);
|
||||
extern int shpchp_save_slot_config(struct controller *ctrl, struct pci_func * new_slot);
|
||||
extern void shpchp_destroy_board_resources(struct pci_func * func);
|
||||
extern int shpchp_return_board_resources(struct pci_func * func, struct resource_lists * resources);
|
||||
extern void shpchp_destroy_resource_list(struct resource_lists * resources);
|
||||
extern int shpchp_configure_device(struct controller* ctrl, struct pci_func* func);
|
||||
extern int shpchp_unconfigure_device(struct pci_func* func);
|
||||
extern int shpchp_configure_device(struct slot *p_slot);
|
||||
extern int shpchp_unconfigure_device(struct slot *p_slot);
|
||||
extern void get_hp_hw_control_from_firmware(struct pci_dev *dev);
|
||||
extern void get_hp_params_from_firmware(struct pci_dev *dev,
|
||||
struct hotplug_params *hpp);
|
||||
extern int shpchprm_get_physical_slot_number(struct controller *ctrl,
|
||||
u32 *sun, u8 busnum, u8 devnum);
|
||||
extern void shpchp_remove_ctrl_files(struct controller *ctrl);
|
||||
|
||||
|
||||
/* Global variables */
|
||||
extern struct controller *shpchp_ctrl_list;
|
||||
extern struct pci_func *shpchp_slot_list[256];
|
||||
|
||||
/* These are added to support AMD shpc */
|
||||
extern u8 shpchp_nic_irq;
|
||||
extern u8 shpchp_disk_irq;
|
||||
|
||||
struct ctrl_reg {
|
||||
volatile u32 base_offset;
|
||||
@ -298,7 +233,7 @@ enum ctrl_offsets {
|
||||
SLOT11 = offsetof(struct ctrl_reg, slot11),
|
||||
SLOT12 = offsetof(struct ctrl_reg, slot12),
|
||||
};
|
||||
typedef u8(*php_intr_callback_t) (unsigned int change_id, void *instance_id);
|
||||
typedef u8(*php_intr_callback_t) (u8 hp_slot, void *instance_id);
|
||||
struct php_ctlr_state_s {
|
||||
struct php_ctlr_state_s *pnext;
|
||||
struct pci_dev *pci_dev;
|
||||
@ -359,12 +294,9 @@ static inline struct slot *shpchp_find_slot (struct controller *ctrl, u8 device)
|
||||
|
||||
p_slot = ctrl->slot;
|
||||
|
||||
dbg("p_slot = %p\n", p_slot);
|
||||
|
||||
while (p_slot && (p_slot->device != device)) {
|
||||
tmp_slot = p_slot;
|
||||
p_slot = p_slot->next;
|
||||
dbg("In while loop, p_slot = %p\n", p_slot);
|
||||
}
|
||||
if (p_slot == NULL) {
|
||||
err("ERROR: shpchp_find_slot device=0x%x\n", device);
|
||||
@ -379,8 +311,6 @@ static inline int wait_for_ctrl_irq (struct controller *ctrl)
|
||||
DECLARE_WAITQUEUE(wait, current);
|
||||
int retval = 0;
|
||||
|
||||
dbg("%s : start\n",__FUNCTION__);
|
||||
|
||||
add_wait_queue(&ctrl->queue, &wait);
|
||||
|
||||
if (!shpchp_poll_mode) {
|
||||
@ -394,19 +324,9 @@ static inline int wait_for_ctrl_irq (struct controller *ctrl)
|
||||
if (signal_pending(current))
|
||||
retval = -EINTR;
|
||||
|
||||
dbg("%s : end\n", __FUNCTION__);
|
||||
return retval;
|
||||
}
|
||||
|
||||
/* Puts node back in the resource list pointed to by head */
|
||||
static inline void return_resource(struct pci_resource **head, struct pci_resource *node)
|
||||
{
|
||||
if (!node || !head)
|
||||
return;
|
||||
node->next = *head;
|
||||
*head = node;
|
||||
}
|
||||
|
||||
#define SLOT_NAME_SIZE 10
|
||||
|
||||
static inline void make_slot_name(char *buffer, int buffer_size, struct slot *slot)
|
||||
@ -420,11 +340,7 @@ enum php_ctlr_type {
|
||||
ACPI
|
||||
};
|
||||
|
||||
int shpc_init( struct controller *ctrl, struct pci_dev *pdev,
|
||||
php_intr_callback_t attention_button_callback,
|
||||
php_intr_callback_t switch_change_callback,
|
||||
php_intr_callback_t presence_change_callback,
|
||||
php_intr_callback_t power_fault_callback);
|
||||
int shpc_init( struct controller *ctrl, struct pci_dev *pdev);
|
||||
|
||||
int shpc_get_ctlr_slot_config( struct controller *ctrl,
|
||||
int *num_ctlr_slots,
|
||||
@ -437,8 +353,6 @@ struct hpc_ops {
|
||||
int (*power_on_slot ) (struct slot *slot);
|
||||
int (*slot_enable ) (struct slot *slot);
|
||||
int (*slot_disable ) (struct slot *slot);
|
||||
int (*enable_all_slots) (struct slot *slot);
|
||||
int (*pwr_on_all_slots) (struct slot *slot);
|
||||
int (*set_bus_speed_mode) (struct slot *slot, enum pci_bus_speed speed);
|
||||
int (*get_power_status) (struct slot *slot, u8 *status);
|
||||
int (*get_attention_status) (struct slot *slot, u8 *status);
|
||||
|
@ -27,26 +27,18 @@
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/moduleparam.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/proc_fs.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/workqueue.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/init.h>
|
||||
#include <asm/uaccess.h>
|
||||
#include "shpchp.h"
|
||||
#include "shpchprm.h"
|
||||
|
||||
/* Global variables */
|
||||
int shpchp_debug;
|
||||
int shpchp_poll_mode;
|
||||
int shpchp_poll_time;
|
||||
struct controller *shpchp_ctrl_list; /* = NULL */
|
||||
struct pci_func *shpchp_slot_list[256];
|
||||
|
||||
#define DRIVER_VERSION "0.4"
|
||||
#define DRIVER_AUTHOR "Dan Zink <dan.zink@compaq.com>, Greg Kroah-Hartman <greg@kroah.com>, Dely Sy <dely.l.sy@intel.com>"
|
||||
@ -113,8 +105,6 @@ static int init_slots(struct controller *ctrl)
|
||||
u32 slot_number, sun;
|
||||
int result = -ENOMEM;
|
||||
|
||||
dbg("%s\n",__FUNCTION__);
|
||||
|
||||
number_of_slots = ctrl->num_slots;
|
||||
slot_device = ctrl->slot_device_offset;
|
||||
slot_number = ctrl->first_slot;
|
||||
@ -352,6 +342,17 @@ static int get_cur_bus_speed (struct hotplug_slot *hotplug_slot, enum pci_bus_sp
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int is_shpc_capable(struct pci_dev *dev)
|
||||
{
|
||||
if ((dev->vendor == PCI_VENDOR_ID_AMD) || (dev->device ==
|
||||
PCI_DEVICE_ID_AMD_GOLAM_7450))
|
||||
return 1;
|
||||
if (pci_find_capability(dev, PCI_CAP_ID_SHPC))
|
||||
return 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int shpc_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
{
|
||||
int rc;
|
||||
@ -360,6 +361,9 @@ static int shpc_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
int first_device_num; /* first PCI device number supported by this SHPC */
|
||||
int num_ctlr_slots; /* number of slots supported by this SHPC */
|
||||
|
||||
if (!is_shpc_capable(pdev))
|
||||
return -ENODEV;
|
||||
|
||||
ctrl = (struct controller *) kmalloc(sizeof(struct controller), GFP_KERNEL);
|
||||
if (!ctrl) {
|
||||
err("%s : out of memory\n", __FUNCTION__);
|
||||
@ -367,19 +371,12 @@ static int shpc_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
}
|
||||
memset(ctrl, 0, sizeof(struct controller));
|
||||
|
||||
dbg("DRV_thread pid = %d\n", current->pid);
|
||||
|
||||
rc = shpc_init(ctrl, pdev,
|
||||
(php_intr_callback_t) shpchp_handle_attention_button,
|
||||
(php_intr_callback_t) shpchp_handle_switch_change,
|
||||
(php_intr_callback_t) shpchp_handle_presence_change,
|
||||
(php_intr_callback_t) shpchp_handle_power_fault);
|
||||
rc = shpc_init(ctrl, pdev);
|
||||
if (rc) {
|
||||
dbg("%s: controller initialization failed\n", SHPC_MODULE_NAME);
|
||||
goto err_out_free_ctrl;
|
||||
}
|
||||
|
||||
dbg("%s: controller initialization success\n", __FUNCTION__);
|
||||
ctrl->pci_dev = pdev; /* pci_dev of the P2P bridge */
|
||||
|
||||
pci_set_drvdata(pdev, ctrl);
|
||||
@ -411,22 +408,7 @@ static int shpc_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
first_device_num = ctrl->slot_device_offset;
|
||||
num_ctlr_slots = ctrl->num_slots;
|
||||
|
||||
/* Store PCI Config Space for all devices on this bus */
|
||||
rc = shpchp_save_config(ctrl, ctrl->slot_bus, num_ctlr_slots, first_device_num);
|
||||
if (rc) {
|
||||
err("%s: unable to save PCI configuration data, error %d\n", __FUNCTION__, rc);
|
||||
goto err_out_free_ctrl_bus;
|
||||
}
|
||||
|
||||
/* Get IO, memory, and IRQ resources for new devices */
|
||||
rc = shpchprm_find_available_resources(ctrl);
|
||||
ctrl->add_support = !rc;
|
||||
|
||||
if (rc) {
|
||||
dbg("shpchprm_find_available_resources = %#x\n", rc);
|
||||
err("unable to locate PCI configuration resources for hot plug add.\n");
|
||||
goto err_out_free_ctrl_bus;
|
||||
}
|
||||
ctrl->add_support = 1;
|
||||
|
||||
/* Setup the slot information structures */
|
||||
rc = init_slots(ctrl);
|
||||
@ -477,7 +459,6 @@ err_out_none:
|
||||
|
||||
static int shpc_start_thread(void)
|
||||
{
|
||||
int loop;
|
||||
int retval = 0;
|
||||
|
||||
dbg("Initialize + Start the notification/polling mechanism \n");
|
||||
@ -488,48 +469,21 @@ static int shpc_start_thread(void)
|
||||
return retval;
|
||||
}
|
||||
|
||||
dbg("Initialize slot lists\n");
|
||||
/* One slot list for each bus in the system */
|
||||
for (loop = 0; loop < 256; loop++) {
|
||||
shpchp_slot_list[loop] = NULL;
|
||||
}
|
||||
|
||||
return retval;
|
||||
}
|
||||
|
||||
static inline void __exit
|
||||
free_shpchp_res(struct pci_resource *res)
|
||||
{
|
||||
struct pci_resource *tres;
|
||||
|
||||
while (res) {
|
||||
tres = res;
|
||||
res = res->next;
|
||||
kfree(tres);
|
||||
}
|
||||
}
|
||||
|
||||
static void __exit unload_shpchpd(void)
|
||||
{
|
||||
struct pci_func *next;
|
||||
struct pci_func *TempSlot;
|
||||
int loop;
|
||||
struct controller *ctrl;
|
||||
struct controller *tctrl;
|
||||
|
||||
ctrl = shpchp_ctrl_list;
|
||||
|
||||
while (ctrl) {
|
||||
shpchp_remove_ctrl_files(ctrl);
|
||||
cleanup_slots(ctrl);
|
||||
|
||||
free_shpchp_res(ctrl->io_head);
|
||||
free_shpchp_res(ctrl->mem_head);
|
||||
free_shpchp_res(ctrl->p_mem_head);
|
||||
free_shpchp_res(ctrl->bus_head);
|
||||
|
||||
kfree (ctrl->pci_bus);
|
||||
|
||||
dbg("%s: calling release_ctlr\n", __FUNCTION__);
|
||||
ctrl->hpc_ops->release_ctlr(ctrl);
|
||||
|
||||
tctrl = ctrl;
|
||||
@ -538,20 +492,6 @@ static void __exit unload_shpchpd(void)
|
||||
kfree(tctrl);
|
||||
}
|
||||
|
||||
for (loop = 0; loop < 256; loop++) {
|
||||
next = shpchp_slot_list[loop];
|
||||
while (next != NULL) {
|
||||
free_shpchp_res(next->io_head);
|
||||
free_shpchp_res(next->mem_head);
|
||||
free_shpchp_res(next->p_mem_head);
|
||||
free_shpchp_res(next->bus_head);
|
||||
|
||||
TempSlot = next;
|
||||
next = next->next;
|
||||
kfree(TempSlot);
|
||||
}
|
||||
}
|
||||
|
||||
/* Stop the notification mechanism */
|
||||
shpchp_event_stop_thread();
|
||||
|
||||
@ -596,20 +536,14 @@ static int __init shpcd_init(void)
|
||||
if (retval)
|
||||
goto error_hpc_init;
|
||||
|
||||
retval = shpchprm_init(PCI);
|
||||
if (!retval) {
|
||||
retval = pci_register_driver(&shpc_driver);
|
||||
dbg("%s: pci_register_driver = %d\n", __FUNCTION__, retval);
|
||||
info(DRIVER_DESC " version: " DRIVER_VERSION "\n");
|
||||
}
|
||||
|
||||
error_hpc_init:
|
||||
if (retval) {
|
||||
shpchprm_cleanup();
|
||||
shpchp_event_stop_thread();
|
||||
} else
|
||||
shpchprm_print_pirt();
|
||||
|
||||
}
|
||||
return retval;
|
||||
}
|
||||
|
||||
@ -618,9 +552,6 @@ static void __exit shpcd_cleanup(void)
|
||||
dbg("unload_shpchpd()\n");
|
||||
unload_shpchpd();
|
||||
|
||||
shpchprm_cleanup();
|
||||
|
||||
dbg("pci_unregister_driver\n");
|
||||
pci_unregister_driver(&shpc_driver);
|
||||
|
||||
info(DRIVER_DESC " version: " DRIVER_VERSION " unloaded\n");
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -27,17 +27,10 @@
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/vmalloc.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/spinlock.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/pci.h>
|
||||
#include <asm/system.h>
|
||||
#include "shpchp.h"
|
||||
|
||||
#ifdef DEBUG
|
||||
@ -282,7 +275,7 @@ static void start_int_poll_timer(struct php_ctlr_state_s *php_ctlr, int seconds)
|
||||
|
||||
static int shpc_write_cmd(struct slot *slot, u8 t_slot, u8 cmd)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u16 cmd_status;
|
||||
int retval = 0;
|
||||
u16 temp_word;
|
||||
@ -320,7 +313,6 @@ static int shpc_write_cmd(struct slot *slot, u8 t_slot, u8 cmd)
|
||||
* command.
|
||||
*/
|
||||
writew(temp_word, php_ctlr->creg + CMD);
|
||||
dbg("%s: temp_word written %x\n", __FUNCTION__, temp_word);
|
||||
|
||||
DBG_LEAVE_ROUTINE
|
||||
return retval;
|
||||
@ -328,7 +320,7 @@ static int shpc_write_cmd(struct slot *slot, u8 t_slot, u8 cmd)
|
||||
|
||||
static int hpc_check_cmd_status(struct controller *ctrl)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = ctrl->hpc_ctlr_handle;
|
||||
u16 cmd_status;
|
||||
int retval = 0;
|
||||
|
||||
@ -368,7 +360,7 @@ static int hpc_check_cmd_status(struct controller *ctrl)
|
||||
|
||||
static int hpc_get_attention_status(struct slot *slot, u8 *status)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u32 slot_reg;
|
||||
u16 slot_status;
|
||||
u8 atten_led_state;
|
||||
@ -408,7 +400,7 @@ static int hpc_get_attention_status(struct slot *slot, u8 *status)
|
||||
|
||||
static int hpc_get_power_status(struct slot * slot, u8 *status)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u32 slot_reg;
|
||||
u16 slot_status;
|
||||
u8 slot_state;
|
||||
@ -450,7 +442,7 @@ static int hpc_get_power_status(struct slot * slot, u8 *status)
|
||||
|
||||
static int hpc_get_latch_status(struct slot *slot, u8 *status)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u32 slot_reg;
|
||||
u16 slot_status;
|
||||
|
||||
@ -473,7 +465,7 @@ static int hpc_get_latch_status(struct slot *slot, u8 *status)
|
||||
|
||||
static int hpc_get_adapter_status(struct slot *slot, u8 *status)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u32 slot_reg;
|
||||
u16 slot_status;
|
||||
u8 card_state;
|
||||
@ -496,7 +488,7 @@ static int hpc_get_adapter_status(struct slot *slot, u8 *status)
|
||||
|
||||
static int hpc_get_prog_int(struct slot *slot, u8 *prog_int)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
|
||||
DBG_ENTER_ROUTINE
|
||||
|
||||
@ -513,7 +505,7 @@ static int hpc_get_prog_int(struct slot *slot, u8 *prog_int)
|
||||
|
||||
static int hpc_get_adapter_speed(struct slot *slot, enum pci_bus_speed *value)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u32 slot_reg;
|
||||
u16 slot_status, sec_bus_status;
|
||||
u8 m66_cap, pcix_cap, pi;
|
||||
@ -594,7 +586,7 @@ static int hpc_get_adapter_speed(struct slot *slot, enum pci_bus_speed *value)
|
||||
|
||||
static int hpc_get_mode1_ECC_cap(struct slot *slot, u8 *mode)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u16 sec_bus_status;
|
||||
u8 pi;
|
||||
int retval = 0;
|
||||
@ -623,7 +615,7 @@ static int hpc_get_mode1_ECC_cap(struct slot *slot, u8 *mode)
|
||||
|
||||
static int hpc_query_power_fault(struct slot * slot)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u32 slot_reg;
|
||||
u16 slot_status;
|
||||
u8 pwr_fault_state, status;
|
||||
@ -647,7 +639,7 @@ static int hpc_query_power_fault(struct slot * slot)
|
||||
|
||||
static int hpc_set_attention_status(struct slot *slot, u8 value)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr =(struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u8 slot_cmd = 0;
|
||||
int rc = 0;
|
||||
|
||||
@ -683,7 +675,7 @@ static int hpc_set_attention_status(struct slot *slot, u8 value)
|
||||
|
||||
static void hpc_set_green_led_on(struct slot *slot)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr =(struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u8 slot_cmd;
|
||||
|
||||
if (!slot->ctrl->hpc_ctlr_handle) {
|
||||
@ -705,7 +697,7 @@ static void hpc_set_green_led_on(struct slot *slot)
|
||||
|
||||
static void hpc_set_green_led_off(struct slot *slot)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr =(struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u8 slot_cmd;
|
||||
|
||||
if (!slot->ctrl->hpc_ctlr_handle) {
|
||||
@ -727,7 +719,7 @@ static void hpc_set_green_led_off(struct slot *slot)
|
||||
|
||||
static void hpc_set_green_led_blink(struct slot *slot)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr =(struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u8 slot_cmd;
|
||||
|
||||
if (!slot->ctrl->hpc_ctlr_handle) {
|
||||
@ -754,7 +746,7 @@ int shpc_get_ctlr_slot_config(struct controller *ctrl,
|
||||
int *updown, /* physical_slot_num increament: 1 or -1 */
|
||||
int *flags)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = ctrl->hpc_ctlr_handle;
|
||||
|
||||
DBG_ENTER_ROUTINE
|
||||
|
||||
@ -776,7 +768,7 @@ int shpc_get_ctlr_slot_config(struct controller *ctrl,
|
||||
|
||||
static void hpc_release_ctlr(struct controller *ctrl)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *p, *p_prev;
|
||||
|
||||
DBG_ENTER_ROUTINE
|
||||
@ -796,10 +788,8 @@ static void hpc_release_ctlr(struct controller *ctrl)
|
||||
}
|
||||
}
|
||||
if (php_ctlr->pci_dev) {
|
||||
dbg("%s: before calling iounmap & release_mem_region\n", __FUNCTION__);
|
||||
iounmap(php_ctlr->creg);
|
||||
release_mem_region(pci_resource_start(php_ctlr->pci_dev, 0), pci_resource_len(php_ctlr->pci_dev, 0));
|
||||
dbg("%s: before calling iounmap & release_mem_region\n", __FUNCTION__);
|
||||
php_ctlr->pci_dev = NULL;
|
||||
}
|
||||
|
||||
@ -828,7 +818,7 @@ DBG_LEAVE_ROUTINE
|
||||
|
||||
static int hpc_power_on_slot(struct slot * slot)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u8 slot_cmd;
|
||||
int retval = 0;
|
||||
|
||||
@ -859,7 +849,7 @@ static int hpc_power_on_slot(struct slot * slot)
|
||||
|
||||
static int hpc_slot_enable(struct slot * slot)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u8 slot_cmd;
|
||||
int retval = 0;
|
||||
|
||||
@ -890,7 +880,7 @@ static int hpc_slot_enable(struct slot * slot)
|
||||
|
||||
static int hpc_slot_disable(struct slot * slot)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
u8 slot_cmd;
|
||||
int retval = 0;
|
||||
|
||||
@ -920,51 +910,12 @@ static int hpc_slot_disable(struct slot * slot)
|
||||
return retval;
|
||||
}
|
||||
|
||||
static int hpc_enable_all_slots( struct slot *slot )
|
||||
{
|
||||
int retval = 0;
|
||||
|
||||
DBG_ENTER_ROUTINE
|
||||
|
||||
if (!slot->ctrl->hpc_ctlr_handle) {
|
||||
err("%s: Invalid HPC controller handle!\n", __FUNCTION__);
|
||||
return -1;
|
||||
}
|
||||
|
||||
retval = shpc_write_cmd(slot, 0, SET_ENABLE_ALL);
|
||||
if (retval) {
|
||||
err("%s: Write command failed!\n", __FUNCTION__);
|
||||
return -1;
|
||||
}
|
||||
|
||||
DBG_LEAVE_ROUTINE
|
||||
|
||||
return retval;
|
||||
}
|
||||
|
||||
static int hpc_pwr_on_all_slots(struct slot *slot)
|
||||
{
|
||||
int retval = 0;
|
||||
|
||||
DBG_ENTER_ROUTINE
|
||||
|
||||
retval = shpc_write_cmd(slot, 0, SET_PWR_ON_ALL);
|
||||
|
||||
if (retval) {
|
||||
err("%s: Write command failed!\n", __FUNCTION__);
|
||||
return -1;
|
||||
}
|
||||
|
||||
DBG_LEAVE_ROUTINE
|
||||
return retval;
|
||||
}
|
||||
|
||||
static int hpc_set_bus_speed_mode(struct slot * slot, enum pci_bus_speed value)
|
||||
{
|
||||
u8 slot_cmd;
|
||||
u8 pi;
|
||||
int retval = 0;
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
|
||||
DBG_ENTER_ROUTINE
|
||||
|
||||
@ -1089,18 +1040,13 @@ static irqreturn_t shpc_isr(int IRQ, void *dev_id, struct pt_regs *regs)
|
||||
|
||||
if (!intr_loc)
|
||||
return IRQ_NONE;
|
||||
dbg("%s: shpc_isr proceeds\n", __FUNCTION__);
|
||||
dbg("%s: intr_loc = %x\n",__FUNCTION__, intr_loc);
|
||||
|
||||
if(!shpchp_poll_mode) {
|
||||
/* Mask Global Interrupt Mask - see implementation note on p. 139 */
|
||||
/* of SHPC spec rev 1.0*/
|
||||
temp_dword = readl(php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
dbg("%s: Before masking global interrupt, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
temp_dword |= 0x00000001;
|
||||
dbg("%s: After masking global interrupt, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
writel(temp_dword, php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
|
||||
intr_loc2 = readl(php_ctlr->creg + INTR_LOC);
|
||||
@ -1114,11 +1060,7 @@ static irqreturn_t shpc_isr(int IRQ, void *dev_id, struct pt_regs *regs)
|
||||
* Detect bit in Controller SERR-INT register
|
||||
*/
|
||||
temp_dword = readl(php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
dbg("%s: Before clearing CCIP, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
temp_dword &= 0xfffeffff;
|
||||
dbg("%s: After clearing CCIP, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
writel(temp_dword, php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
wake_up_interruptible(&ctrl->queue);
|
||||
}
|
||||
@ -1126,11 +1068,7 @@ static irqreturn_t shpc_isr(int IRQ, void *dev_id, struct pt_regs *regs)
|
||||
if ((intr_loc = (intr_loc >> 1)) == 0) {
|
||||
/* Unmask Global Interrupt Mask */
|
||||
temp_dword = readl(php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
dbg("%s: 1-Before unmasking global interrupt, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
temp_dword &= 0xfffffffe;
|
||||
dbg("%s: 1-After unmasking global interrupt, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
writel(temp_dword, php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
|
||||
return IRQ_NONE;
|
||||
@ -1140,11 +1078,9 @@ static irqreturn_t shpc_isr(int IRQ, void *dev_id, struct pt_regs *regs)
|
||||
/* To find out which slot has interrupt pending */
|
||||
if ((intr_loc >> hp_slot) & 0x01) {
|
||||
temp_dword = readl(php_ctlr->creg + SLOT1 + (4*hp_slot));
|
||||
dbg("%s: Slot %x with intr, temp_dword = %x\n",
|
||||
dbg("%s: Slot %x with intr, slot register = %x\n",
|
||||
__FUNCTION__, hp_slot, temp_dword);
|
||||
temp_byte = (temp_dword >> 16) & 0xFF;
|
||||
dbg("%s: Slot with intr, temp_byte = %x\n",
|
||||
__FUNCTION__, temp_byte);
|
||||
if ((php_ctlr->switch_change_callback) && (temp_byte & 0x08))
|
||||
schedule_flag += php_ctlr->switch_change_callback(
|
||||
hp_slot, php_ctlr->callback_instance_id);
|
||||
@ -1160,8 +1096,6 @@ static irqreturn_t shpc_isr(int IRQ, void *dev_id, struct pt_regs *regs)
|
||||
|
||||
/* Clear all slot events */
|
||||
temp_dword = 0xe01f3fff;
|
||||
dbg("%s: Clearing slot events, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
writel(temp_dword, php_ctlr->creg + SLOT1 + (4*hp_slot));
|
||||
|
||||
intr_loc2 = readl(php_ctlr->creg + INTR_LOC);
|
||||
@ -1171,11 +1105,7 @@ static irqreturn_t shpc_isr(int IRQ, void *dev_id, struct pt_regs *regs)
|
||||
if (!shpchp_poll_mode) {
|
||||
/* Unmask Global Interrupt Mask */
|
||||
temp_dword = readl(php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
dbg("%s: 2-Before unmasking global interrupt, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
temp_dword &= 0xfffffffe;
|
||||
dbg("%s: 2-After unmasking global interrupt, temp_dword = %x\n",
|
||||
__FUNCTION__, temp_dword);
|
||||
writel(temp_dword, php_ctlr->creg + SERR_INTR_ENABLE);
|
||||
}
|
||||
|
||||
@ -1184,7 +1114,7 @@ static irqreturn_t shpc_isr(int IRQ, void *dev_id, struct pt_regs *regs)
|
||||
|
||||
static int hpc_get_max_bus_speed (struct slot *slot, enum pci_bus_speed *value)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
enum pci_bus_speed bus_speed = PCI_SPEED_UNKNOWN;
|
||||
int retval = 0;
|
||||
u8 pi;
|
||||
@ -1253,7 +1183,7 @@ static int hpc_get_max_bus_speed (struct slot *slot, enum pci_bus_speed *value)
|
||||
|
||||
static int hpc_get_cur_bus_speed (struct slot *slot, enum pci_bus_speed *value)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr = (struct php_ctlr_state_s *) slot->ctrl->hpc_ctlr_handle;
|
||||
struct php_ctlr_state_s *php_ctlr = slot->ctrl->hpc_ctlr_handle;
|
||||
enum pci_bus_speed bus_speed = PCI_SPEED_UNKNOWN;
|
||||
u16 sec_bus_status;
|
||||
int retval = 0;
|
||||
@ -1367,8 +1297,6 @@ static struct hpc_ops shpchp_hpc_ops = {
|
||||
.power_on_slot = hpc_power_on_slot,
|
||||
.slot_enable = hpc_slot_enable,
|
||||
.slot_disable = hpc_slot_disable,
|
||||
.enable_all_slots = hpc_enable_all_slots,
|
||||
.pwr_on_all_slots = hpc_pwr_on_all_slots,
|
||||
.set_bus_speed_mode = hpc_set_bus_speed_mode,
|
||||
.set_attention_status = hpc_set_attention_status,
|
||||
.get_power_status = hpc_get_power_status,
|
||||
@ -1391,12 +1319,7 @@ static struct hpc_ops shpchp_hpc_ops = {
|
||||
.check_cmd_status = hpc_check_cmd_status,
|
||||
};
|
||||
|
||||
int shpc_init(struct controller * ctrl,
|
||||
struct pci_dev * pdev,
|
||||
php_intr_callback_t attention_button_callback,
|
||||
php_intr_callback_t switch_change_callback,
|
||||
php_intr_callback_t presence_change_callback,
|
||||
php_intr_callback_t power_fault_callback)
|
||||
int shpc_init(struct controller * ctrl, struct pci_dev * pdev)
|
||||
{
|
||||
struct php_ctlr_state_s *php_ctlr, *p;
|
||||
void *instance_id = ctrl;
|
||||
@ -1405,7 +1328,6 @@ int shpc_init(struct controller * ctrl,
|
||||
static int first = 1;
|
||||
u32 shpc_cap_offset, shpc_base_offset;
|
||||
u32 tempdword, slot_reg;
|
||||
u16 vendor_id, device_id;
|
||||
u8 i;
|
||||
|
||||
DBG_ENTER_ROUTINE
|
||||
@ -1422,21 +1344,8 @@ int shpc_init(struct controller * ctrl,
|
||||
|
||||
php_ctlr->pci_dev = pdev; /* save pci_dev in context */
|
||||
|
||||
rc = pci_read_config_word(pdev, PCI_VENDOR_ID, &vendor_id);
|
||||
dbg("%s: Vendor ID: %x\n",__FUNCTION__, vendor_id);
|
||||
if (rc) {
|
||||
err("%s: unable to read PCI configuration data\n", __FUNCTION__);
|
||||
goto abort_free_ctlr;
|
||||
}
|
||||
|
||||
rc = pci_read_config_word(pdev, PCI_DEVICE_ID, &device_id);
|
||||
dbg("%s: Device ID: %x\n",__FUNCTION__, device_id);
|
||||
if (rc) {
|
||||
err("%s: unable to read PCI configuration data\n", __FUNCTION__);
|
||||
goto abort_free_ctlr;
|
||||
}
|
||||
|
||||
if ((vendor_id == PCI_VENDOR_ID_AMD) || (device_id == PCI_DEVICE_ID_AMD_GOLAM_7450)) {
|
||||
if ((pdev->vendor == PCI_VENDOR_ID_AMD) || (pdev->device ==
|
||||
PCI_DEVICE_ID_AMD_GOLAM_7450)) {
|
||||
shpc_base_offset = 0; /* amd shpc driver doesn't use this; assume 0 */
|
||||
} else {
|
||||
if ((shpc_cap_offset = pci_find_capability(pdev, PCI_CAP_ID_SHPC)) == 0) {
|
||||
@ -1469,7 +1378,8 @@ int shpc_init(struct controller * ctrl,
|
||||
err("%s : pci_read_config_dword failed\n", __FUNCTION__);
|
||||
goto abort_free_ctlr;
|
||||
}
|
||||
dbg("%s: offset %d: tempdword %x\n", __FUNCTION__,i, tempdword);
|
||||
dbg("%s: offset %d: value %x\n", __FUNCTION__,i,
|
||||
tempdword);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1478,13 +1388,6 @@ int shpc_init(struct controller * ctrl,
|
||||
first = 0;
|
||||
}
|
||||
|
||||
dbg("pdev = %p: b:d:f:irq=0x%x:%x:%x:%x\n", pdev, pdev->bus->number, PCI_SLOT(pdev->devfn),
|
||||
PCI_FUNC(pdev->devfn), pdev->irq);
|
||||
for ( rc = 0; rc < DEVICE_COUNT_RESOURCE; rc++)
|
||||
if (pci_resource_len(pdev, rc) > 0)
|
||||
dbg("pci resource[%d] start=0x%lx(len=0x%lx), shpc_base_offset %x\n", rc,
|
||||
pci_resource_start(pdev, rc), pci_resource_len(pdev, rc), shpc_base_offset);
|
||||
|
||||
info("HPC vendor_id %x device_id %x ss_vid %x ss_did %x\n", pdev->vendor, pdev->device, pdev->subsystem_vendor,
|
||||
pdev->subsystem_device);
|
||||
|
||||
@ -1504,7 +1407,6 @@ int shpc_init(struct controller * ctrl,
|
||||
goto abort_free_ctlr;
|
||||
}
|
||||
dbg("%s: php_ctlr->creg %p\n", __FUNCTION__, php_ctlr->creg);
|
||||
dbg("%s: physical addr %p\n", __FUNCTION__, (void*)pci_resource_start(pdev, 0));
|
||||
|
||||
init_MUTEX(&ctrl->crit_sect);
|
||||
/* Setup wait queue */
|
||||
@ -1512,13 +1414,10 @@ int shpc_init(struct controller * ctrl,
|
||||
|
||||
/* Find the IRQ */
|
||||
php_ctlr->irq = pdev->irq;
|
||||
dbg("HPC interrupt = %d\n", php_ctlr->irq);
|
||||
|
||||
/* Save interrupt callback info */
|
||||
php_ctlr->attention_button_callback = attention_button_callback;
|
||||
php_ctlr->switch_change_callback = switch_change_callback;
|
||||
php_ctlr->presence_change_callback = presence_change_callback;
|
||||
php_ctlr->power_fault_callback = power_fault_callback;
|
||||
php_ctlr->attention_button_callback = shpchp_handle_attention_button,
|
||||
php_ctlr->switch_change_callback = shpchp_handle_switch_change;
|
||||
php_ctlr->presence_change_callback = shpchp_handle_presence_change;
|
||||
php_ctlr->power_fault_callback = shpchp_handle_power_fault;
|
||||
php_ctlr->callback_instance_id = instance_id;
|
||||
|
||||
/* Return PCI Controller Info */
|
||||
@ -1556,7 +1455,6 @@ int shpc_init(struct controller * ctrl,
|
||||
if (rc) {
|
||||
info("Can't get msi for the hotplug controller\n");
|
||||
info("Use INTx for the hotplug controller\n");
|
||||
dbg("%s: rc = %x\n", __FUNCTION__, rc);
|
||||
} else
|
||||
php_ctlr->irq = pdev->irq;
|
||||
|
||||
@ -1566,9 +1464,11 @@ int shpc_init(struct controller * ctrl,
|
||||
err("Can't get irq %d for the hotplug controller\n", php_ctlr->irq);
|
||||
goto abort_free_ctlr;
|
||||
}
|
||||
/* Execute OSHP method here */
|
||||
}
|
||||
dbg("%s: Before adding HPC to HPC list\n", __FUNCTION__);
|
||||
dbg("%s: HPC at b:d:f:irq=0x%x:%x:%x:%x\n", __FUNCTION__,
|
||||
pdev->bus->number, PCI_SLOT(pdev->devfn),
|
||||
PCI_FUNC(pdev->devfn), pdev->irq);
|
||||
get_hp_hw_control_from_firmware(pdev);
|
||||
|
||||
/* Add this HPC instance into the HPC list */
|
||||
spin_lock(&list_lock);
|
||||
@ -1607,7 +1507,6 @@ int shpc_init(struct controller * ctrl,
|
||||
dbg("%s: SERR_INTR_ENABLE = %x\n", __FUNCTION__, tempdword);
|
||||
}
|
||||
|
||||
dbg("%s: Leaving shpc_init\n", __FUNCTION__);
|
||||
DBG_LEAVE_ROUTINE
|
||||
return 0;
|
||||
|
||||
|
@ -27,784 +27,151 @@
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/workqueue.h>
|
||||
#include <linux/proc_fs.h>
|
||||
#include <linux/pci.h>
|
||||
#include "../pci.h"
|
||||
#include "shpchp.h"
|
||||
#ifndef CONFIG_IA64
|
||||
#include "../../../arch/i386/pci/pci.h" /* horrible hack showing how processor dependant we are... */
|
||||
#endif
|
||||
|
||||
int shpchp_configure_device (struct controller* ctrl, struct pci_func* func)
|
||||
void program_fw_provided_values(struct pci_dev *dev)
|
||||
{
|
||||
unsigned char bus;
|
||||
u16 pci_cmd, pci_bctl;
|
||||
struct pci_dev *cdev;
|
||||
struct hotplug_params hpp = {0x8, 0x40, 0, 0}; /* defaults */
|
||||
|
||||
/* Program hpp values for this device */
|
||||
if (!(dev->hdr_type == PCI_HEADER_TYPE_NORMAL ||
|
||||
(dev->hdr_type == PCI_HEADER_TYPE_BRIDGE &&
|
||||
(dev->class >> 8) == PCI_CLASS_BRIDGE_PCI)))
|
||||
return;
|
||||
|
||||
get_hp_params_from_firmware(dev, &hpp);
|
||||
|
||||
pci_write_config_byte(dev, PCI_CACHE_LINE_SIZE, hpp.cache_line_size);
|
||||
pci_write_config_byte(dev, PCI_LATENCY_TIMER, hpp.latency_timer);
|
||||
pci_read_config_word(dev, PCI_COMMAND, &pci_cmd);
|
||||
if (hpp.enable_serr)
|
||||
pci_cmd |= PCI_COMMAND_SERR;
|
||||
else
|
||||
pci_cmd &= ~PCI_COMMAND_SERR;
|
||||
if (hpp.enable_perr)
|
||||
pci_cmd |= PCI_COMMAND_PARITY;
|
||||
else
|
||||
pci_cmd &= ~PCI_COMMAND_PARITY;
|
||||
pci_write_config_word(dev, PCI_COMMAND, pci_cmd);
|
||||
|
||||
/* Program bridge control value and child devices */
|
||||
if ((dev->class >> 8) == PCI_CLASS_BRIDGE_PCI) {
|
||||
pci_write_config_byte(dev, PCI_SEC_LATENCY_TIMER,
|
||||
hpp.latency_timer);
|
||||
pci_read_config_word(dev, PCI_BRIDGE_CONTROL, &pci_bctl);
|
||||
if (hpp.enable_serr)
|
||||
pci_bctl |= PCI_BRIDGE_CTL_SERR;
|
||||
else
|
||||
pci_bctl &= ~PCI_BRIDGE_CTL_SERR;
|
||||
if (hpp.enable_perr)
|
||||
pci_bctl |= PCI_BRIDGE_CTL_PARITY;
|
||||
else
|
||||
pci_bctl &= ~PCI_BRIDGE_CTL_PARITY;
|
||||
pci_write_config_word(dev, PCI_BRIDGE_CONTROL, pci_bctl);
|
||||
if (dev->subordinate) {
|
||||
list_for_each_entry(cdev, &dev->subordinate->devices,
|
||||
bus_list)
|
||||
program_fw_provided_values(cdev);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int shpchp_configure_device(struct slot *p_slot)
|
||||
{
|
||||
struct pci_dev *dev;
|
||||
struct pci_bus *parent = p_slot->ctrl->pci_dev->subordinate;
|
||||
int num, fn;
|
||||
|
||||
dev = pci_find_slot(p_slot->bus, PCI_DEVFN(p_slot->device, 0));
|
||||
if (dev) {
|
||||
err("Device %s already exists at %x:%x, cannot hot-add\n",
|
||||
pci_name(dev), p_slot->bus, p_slot->device);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
num = pci_scan_slot(parent, PCI_DEVFN(p_slot->device, 0));
|
||||
if (num == 0) {
|
||||
err("No new device found\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
for (fn = 0; fn < 8; fn++) {
|
||||
if (!(dev = pci_find_slot(p_slot->bus,
|
||||
PCI_DEVFN(p_slot->device, fn))))
|
||||
continue;
|
||||
if ((dev->class >> 16) == PCI_BASE_CLASS_DISPLAY) {
|
||||
err("Cannot hot-add display device %s\n",
|
||||
pci_name(dev));
|
||||
continue;
|
||||
}
|
||||
if ((dev->hdr_type == PCI_HEADER_TYPE_BRIDGE) ||
|
||||
(dev->hdr_type == PCI_HEADER_TYPE_CARDBUS)) {
|
||||
/* Find an unused bus number for the new bridge */
|
||||
struct pci_bus *child;
|
||||
int num;
|
||||
|
||||
if (func->pci_dev == NULL)
|
||||
func->pci_dev = pci_find_slot(func->bus, PCI_DEVFN(func->device, func->function));
|
||||
|
||||
/* Still NULL ? Well then scan for it ! */
|
||||
if (func->pci_dev == NULL) {
|
||||
num = pci_scan_slot(ctrl->pci_dev->subordinate, PCI_DEVFN(func->device, func->function));
|
||||
if (num) {
|
||||
dbg("%s: subordiante %p number %x\n", __FUNCTION__, ctrl->pci_dev->subordinate,
|
||||
ctrl->pci_dev->subordinate->number);
|
||||
pci_bus_add_devices(ctrl->pci_dev->subordinate);
|
||||
}
|
||||
|
||||
func->pci_dev = pci_find_slot(func->bus, PCI_DEVFN(func->device, func->function));
|
||||
if (func->pci_dev == NULL) {
|
||||
dbg("ERROR: pci_dev still null\n");
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
if (func->pci_dev->hdr_type == PCI_HEADER_TYPE_BRIDGE) {
|
||||
pci_read_config_byte(func->pci_dev, PCI_SECONDARY_BUS, &bus);
|
||||
child = pci_add_new_bus(func->pci_dev->bus, (func->pci_dev), bus);
|
||||
pci_do_scan_bus(child);
|
||||
|
||||
unsigned char busnr, start = parent->secondary;
|
||||
unsigned char end = parent->subordinate;
|
||||
for (busnr = start; busnr <= end; busnr++) {
|
||||
if (!pci_find_bus(pci_domain_nr(parent),
|
||||
busnr))
|
||||
break;
|
||||
}
|
||||
if (busnr >= end) {
|
||||
err("No free bus for hot-added bridge\n");
|
||||
continue;
|
||||
}
|
||||
child = pci_add_new_bus(parent, dev, busnr);
|
||||
if (!child) {
|
||||
err("Cannot add new bus for %s\n",
|
||||
pci_name(dev));
|
||||
continue;
|
||||
}
|
||||
child->subordinate = pci_do_scan_bus(child);
|
||||
pci_bus_size_bridges(child);
|
||||
}
|
||||
program_fw_provided_values(dev);
|
||||
}
|
||||
|
||||
pci_bus_assign_resources(parent);
|
||||
pci_bus_add_devices(parent);
|
||||
pci_enable_bridges(parent);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
int shpchp_unconfigure_device(struct pci_func* func)
|
||||
int shpchp_unconfigure_device(struct slot *p_slot)
|
||||
{
|
||||
int rc = 0;
|
||||
int j;
|
||||
u8 bctl = 0;
|
||||
|
||||
dbg("%s: bus/dev/func = %x/%x/%x\n", __FUNCTION__, func->bus,
|
||||
func->device, func->function);
|
||||
dbg("%s: bus/dev = %x/%x\n", __FUNCTION__, p_slot->bus, p_slot->device);
|
||||
|
||||
for (j=0; j<8 ; j++) {
|
||||
struct pci_dev* temp = pci_find_slot(func->bus,
|
||||
(func->device << 3) | j);
|
||||
if (temp) {
|
||||
struct pci_dev* temp = pci_find_slot(p_slot->bus,
|
||||
(p_slot->device << 3) | j);
|
||||
if (!temp)
|
||||
continue;
|
||||
if ((temp->class >> 16) == PCI_BASE_CLASS_DISPLAY) {
|
||||
err("Cannot remove display device %s\n",
|
||||
pci_name(temp));
|
||||
continue;
|
||||
}
|
||||
if (temp->hdr_type == PCI_HEADER_TYPE_BRIDGE) {
|
||||
pci_read_config_byte(temp, PCI_BRIDGE_CONTROL, &bctl);
|
||||
if (bctl & PCI_BRIDGE_CTL_VGA) {
|
||||
err("Cannot remove display device %s\n",
|
||||
pci_name(temp));
|
||||
continue;
|
||||
}
|
||||
}
|
||||
pci_remove_bus_device(temp);
|
||||
}
|
||||
}
|
||||
return rc;
|
||||
}
|
||||
|
||||
/*
|
||||
* shpchp_set_irq
|
||||
*
|
||||
* @bus_num: bus number of PCI device
|
||||
* @dev_num: device number of PCI device
|
||||
* @slot: pointer to u8 where slot number will be returned
|
||||
*/
|
||||
int shpchp_set_irq (u8 bus_num, u8 dev_num, u8 int_pin, u8 irq_num)
|
||||
{
|
||||
#if defined(CONFIG_X86) && !defined(CONFIG_X86_IO_APIC) && !defined(CONFIG_X86_64)
|
||||
int rc;
|
||||
u16 temp_word;
|
||||
struct pci_dev fakedev;
|
||||
struct pci_bus fakebus;
|
||||
|
||||
fakedev.devfn = dev_num << 3;
|
||||
fakedev.bus = &fakebus;
|
||||
fakebus.number = bus_num;
|
||||
dbg("%s: dev %d, bus %d, pin %d, num %d\n",
|
||||
__FUNCTION__, dev_num, bus_num, int_pin, irq_num);
|
||||
rc = pcibios_set_irq_routing(&fakedev, int_pin - 0x0a, irq_num);
|
||||
dbg("%s: rc %d\n", __FUNCTION__, rc);
|
||||
if (!rc)
|
||||
return !rc;
|
||||
|
||||
/* set the Edge Level Control Register (ELCR) */
|
||||
temp_word = inb(0x4d0);
|
||||
temp_word |= inb(0x4d1) << 8;
|
||||
|
||||
temp_word |= 0x01 << irq_num;
|
||||
|
||||
/* This should only be for x86 as it sets the Edge Level Control Register */
|
||||
outb((u8) (temp_word & 0xFF), 0x4d0);
|
||||
outb((u8) ((temp_word & 0xFF00) >> 8), 0x4d1);
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* More PCI configuration routines; this time centered around hotplug controller */
|
||||
|
||||
|
||||
/*
|
||||
* shpchp_save_config
|
||||
*
|
||||
* Reads configuration for all slots in a PCI bus and saves info.
|
||||
*
|
||||
* Note: For non-hot plug busses, the slot # saved is the device #
|
||||
*
|
||||
* returns 0 if success
|
||||
*/
|
||||
int shpchp_save_config(struct controller *ctrl, int busnumber, int num_ctlr_slots, int first_device_num)
|
||||
{
|
||||
int rc;
|
||||
u8 class_code;
|
||||
u8 header_type;
|
||||
u32 ID;
|
||||
u8 secondary_bus;
|
||||
struct pci_func *new_slot;
|
||||
int sub_bus;
|
||||
int FirstSupported;
|
||||
int LastSupported;
|
||||
int max_functions;
|
||||
int function;
|
||||
u8 DevError;
|
||||
int device = 0;
|
||||
int cloop = 0;
|
||||
int stop_it;
|
||||
int index;
|
||||
int is_hot_plug = num_ctlr_slots || first_device_num;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
|
||||
dbg("%s: num_ctlr_slots = %d, first_device_num = %d\n", __FUNCTION__,
|
||||
num_ctlr_slots, first_device_num);
|
||||
|
||||
memcpy(&lpci_bus, ctrl->pci_dev->subordinate, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
|
||||
dbg("%s: num_ctlr_slots = %d, first_device_num = %d\n", __FUNCTION__,
|
||||
num_ctlr_slots, first_device_num);
|
||||
|
||||
/* Decide which slots are supported */
|
||||
if (is_hot_plug) {
|
||||
/*********************************
|
||||
* is_hot_plug is the slot mask
|
||||
*********************************/
|
||||
FirstSupported = first_device_num;
|
||||
LastSupported = FirstSupported + num_ctlr_slots - 1;
|
||||
} else {
|
||||
FirstSupported = 0;
|
||||
LastSupported = 0x1F;
|
||||
}
|
||||
|
||||
dbg("FirstSupported = %d, LastSupported = %d\n", FirstSupported,
|
||||
LastSupported);
|
||||
|
||||
/* Save PCI configuration space for all devices in supported slots */
|
||||
pci_bus->number = busnumber;
|
||||
for (device = FirstSupported; device <= LastSupported; device++) {
|
||||
ID = 0xFFFFFFFF;
|
||||
rc = pci_bus_read_config_dword(pci_bus, PCI_DEVFN(device, 0),
|
||||
PCI_VENDOR_ID, &ID);
|
||||
|
||||
if (ID != 0xFFFFFFFF) { /* device in slot */
|
||||
rc = pci_bus_read_config_byte(pci_bus, PCI_DEVFN(device, 0),
|
||||
0x0B, &class_code);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
rc = pci_bus_read_config_byte(pci_bus, PCI_DEVFN(device, 0),
|
||||
PCI_HEADER_TYPE, &header_type);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
dbg("class_code = %x, header_type = %x\n", class_code, header_type);
|
||||
|
||||
/* If multi-function device, set max_functions to 8 */
|
||||
if (header_type & 0x80)
|
||||
max_functions = 8;
|
||||
else
|
||||
max_functions = 1;
|
||||
|
||||
function = 0;
|
||||
|
||||
do {
|
||||
DevError = 0;
|
||||
|
||||
if ((header_type & 0x7F) == PCI_HEADER_TYPE_BRIDGE) { /* P-P Bridge */
|
||||
/* Recurse the subordinate bus
|
||||
* get the subordinate bus number
|
||||
*/
|
||||
rc = pci_bus_read_config_byte(pci_bus,
|
||||
PCI_DEVFN(device, function),
|
||||
PCI_SECONDARY_BUS, &secondary_bus);
|
||||
if (rc) {
|
||||
return rc;
|
||||
} else {
|
||||
sub_bus = (int) secondary_bus;
|
||||
|
||||
/* Save secondary bus cfg spc with this recursive call. */
|
||||
rc = shpchp_save_config(ctrl, sub_bus, 0, 0);
|
||||
if (rc)
|
||||
return rc;
|
||||
}
|
||||
}
|
||||
|
||||
index = 0;
|
||||
new_slot = shpchp_slot_find(busnumber, device, index++);
|
||||
|
||||
dbg("new_slot = %p\n", new_slot);
|
||||
|
||||
while (new_slot && (new_slot->function != (u8) function)) {
|
||||
new_slot = shpchp_slot_find(busnumber, device, index++);
|
||||
dbg("new_slot = %p\n", new_slot);
|
||||
}
|
||||
if (!new_slot) {
|
||||
/* Setup slot structure. */
|
||||
new_slot = shpchp_slot_create(busnumber);
|
||||
dbg("new_slot = %p\n", new_slot);
|
||||
|
||||
if (new_slot == NULL)
|
||||
return(1);
|
||||
}
|
||||
|
||||
new_slot->bus = (u8) busnumber;
|
||||
new_slot->device = (u8) device;
|
||||
new_slot->function = (u8) function;
|
||||
new_slot->is_a_board = 1;
|
||||
new_slot->switch_save = 0x10;
|
||||
new_slot->pwr_save = 1;
|
||||
/* In case of unsupported board */
|
||||
new_slot->status = DevError;
|
||||
new_slot->pci_dev = pci_find_slot(new_slot->bus,
|
||||
(new_slot->device << 3) | new_slot->function);
|
||||
dbg("new_slot->pci_dev = %p\n", new_slot->pci_dev);
|
||||
|
||||
for (cloop = 0; cloop < 0x20; cloop++) {
|
||||
rc = pci_bus_read_config_dword(pci_bus,
|
||||
PCI_DEVFN(device, function),
|
||||
cloop << 2,
|
||||
(u32 *) &(new_slot->config_space [cloop]));
|
||||
/* dbg("new_slot->config_space[%x] = %x\n",
|
||||
cloop, new_slot->config_space[cloop]); */
|
||||
if (rc)
|
||||
return rc;
|
||||
}
|
||||
|
||||
function++;
|
||||
|
||||
stop_it = 0;
|
||||
|
||||
/* this loop skips to the next present function
|
||||
* reading in Class Code and Header type.
|
||||
*/
|
||||
|
||||
while ((function < max_functions)&&(!stop_it)) {
|
||||
rc = pci_bus_read_config_dword(pci_bus,
|
||||
PCI_DEVFN(device, function),
|
||||
PCI_VENDOR_ID, &ID);
|
||||
|
||||
if (ID == 0xFFFFFFFF) { /* nothing there. */
|
||||
function++;
|
||||
dbg("Nothing there\n");
|
||||
} else { /* Something there */
|
||||
rc = pci_bus_read_config_byte(pci_bus,
|
||||
PCI_DEVFN(device, function),
|
||||
0x0B, &class_code);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
rc = pci_bus_read_config_byte(pci_bus,
|
||||
PCI_DEVFN(device, function),
|
||||
PCI_HEADER_TYPE, &header_type);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
dbg("class_code = %x, header_type = %x\n",
|
||||
class_code, header_type);
|
||||
stop_it++;
|
||||
}
|
||||
}
|
||||
|
||||
} while (function < max_functions);
|
||||
/* End of IF (device in slot?) */
|
||||
} else if (is_hot_plug) {
|
||||
/* Setup slot structure with entry for empty slot */
|
||||
new_slot = shpchp_slot_create(busnumber);
|
||||
|
||||
if (new_slot == NULL) {
|
||||
return(1);
|
||||
}
|
||||
dbg("new_slot = %p\n", new_slot);
|
||||
|
||||
new_slot->bus = (u8) busnumber;
|
||||
new_slot->device = (u8) device;
|
||||
new_slot->function = 0;
|
||||
new_slot->is_a_board = 0;
|
||||
new_slot->presence_save = 0;
|
||||
new_slot->switch_save = 0;
|
||||
}
|
||||
} /* End of FOR loop */
|
||||
|
||||
return(0);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* shpchp_save_slot_config
|
||||
*
|
||||
* Saves configuration info for all PCI devices in a given slot
|
||||
* including subordinate busses.
|
||||
*
|
||||
* returns 0 if success
|
||||
*/
|
||||
int shpchp_save_slot_config(struct controller *ctrl, struct pci_func * new_slot)
|
||||
{
|
||||
int rc;
|
||||
u8 class_code;
|
||||
u8 header_type;
|
||||
u32 ID;
|
||||
u8 secondary_bus;
|
||||
int sub_bus;
|
||||
int max_functions;
|
||||
int function;
|
||||
int cloop = 0;
|
||||
int stop_it;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
memcpy(&lpci_bus, ctrl->pci_dev->subordinate, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
pci_bus->number = new_slot->bus;
|
||||
|
||||
ID = 0xFFFFFFFF;
|
||||
|
||||
pci_bus_read_config_dword(pci_bus, PCI_DEVFN(new_slot->device, 0),
|
||||
PCI_VENDOR_ID, &ID);
|
||||
|
||||
if (ID != 0xFFFFFFFF) { /* device in slot */
|
||||
pci_bus_read_config_byte(pci_bus, PCI_DEVFN(new_slot->device, 0),
|
||||
0x0B, &class_code);
|
||||
|
||||
pci_bus_read_config_byte(pci_bus, PCI_DEVFN(new_slot->device, 0),
|
||||
PCI_HEADER_TYPE, &header_type);
|
||||
|
||||
if (header_type & 0x80) /* Multi-function device */
|
||||
max_functions = 8;
|
||||
else
|
||||
max_functions = 1;
|
||||
|
||||
function = 0;
|
||||
|
||||
do {
|
||||
if ((header_type & 0x7F) == PCI_HEADER_TYPE_BRIDGE) { /* PCI-PCI Bridge */
|
||||
/* Recurse the subordinate bus */
|
||||
pci_bus_read_config_byte(pci_bus,
|
||||
PCI_DEVFN(new_slot->device, function),
|
||||
PCI_SECONDARY_BUS, &secondary_bus);
|
||||
|
||||
sub_bus = (int) secondary_bus;
|
||||
|
||||
/* Save the config headers for the secondary bus. */
|
||||
rc = shpchp_save_config(ctrl, sub_bus, 0, 0);
|
||||
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
} /* End of IF */
|
||||
|
||||
new_slot->status = 0;
|
||||
|
||||
for (cloop = 0; cloop < 0x20; cloop++) {
|
||||
pci_bus_read_config_dword(pci_bus,
|
||||
PCI_DEVFN(new_slot->device, function),
|
||||
cloop << 2,
|
||||
(u32 *) &(new_slot->config_space [cloop]));
|
||||
}
|
||||
|
||||
function++;
|
||||
|
||||
stop_it = 0;
|
||||
|
||||
/* this loop skips to the next present function
|
||||
* reading in the Class Code and the Header type.
|
||||
*/
|
||||
|
||||
while ((function < max_functions) && (!stop_it)) {
|
||||
pci_bus_read_config_dword(pci_bus,
|
||||
PCI_DEVFN(new_slot->device, function),
|
||||
PCI_VENDOR_ID, &ID);
|
||||
|
||||
if (ID == 0xFFFFFFFF) { /* nothing there. */
|
||||
function++;
|
||||
} else { /* Something there */
|
||||
pci_bus_read_config_byte(pci_bus,
|
||||
PCI_DEVFN(new_slot->device, function),
|
||||
0x0B, &class_code);
|
||||
|
||||
pci_bus_read_config_byte(pci_bus,
|
||||
PCI_DEVFN(new_slot->device, function),
|
||||
PCI_HEADER_TYPE, &header_type);
|
||||
|
||||
stop_it++;
|
||||
}
|
||||
}
|
||||
|
||||
} while (function < max_functions);
|
||||
} /* End of IF (device in slot?) */
|
||||
else {
|
||||
return 2;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* shpchp_save_used_resources
|
||||
*
|
||||
* Stores used resource information for existing boards. this is
|
||||
* for boards that were in the system when this driver was loaded.
|
||||
* this function is for hot plug ADD
|
||||
*
|
||||
* returns 0 if success
|
||||
* if disable == 1(DISABLE_CARD),
|
||||
* it loops for all functions of the slot and disables them.
|
||||
* else, it just get resources of the function and return.
|
||||
*/
|
||||
int shpchp_save_used_resources(struct controller *ctrl, struct pci_func *func, int disable)
|
||||
{
|
||||
u8 cloop;
|
||||
u8 header_type;
|
||||
u8 secondary_bus;
|
||||
u8 temp_byte;
|
||||
u16 command;
|
||||
u16 save_command;
|
||||
u16 w_base, w_length;
|
||||
u32 temp_register;
|
||||
u32 save_base;
|
||||
u32 base, length;
|
||||
u64 base64 = 0;
|
||||
int index = 0;
|
||||
unsigned int devfn;
|
||||
struct pci_resource *mem_node = NULL;
|
||||
struct pci_resource *p_mem_node = NULL;
|
||||
struct pci_resource *t_mem_node;
|
||||
struct pci_resource *io_node;
|
||||
struct pci_resource *bus_node;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
memcpy(&lpci_bus, ctrl->pci_dev->subordinate, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
|
||||
if (disable)
|
||||
func = shpchp_slot_find(func->bus, func->device, index++);
|
||||
|
||||
while ((func != NULL) && func->is_a_board) {
|
||||
pci_bus->number = func->bus;
|
||||
devfn = PCI_DEVFN(func->device, func->function);
|
||||
|
||||
/* Save the command register */
|
||||
pci_bus_read_config_word(pci_bus, devfn, PCI_COMMAND, &save_command);
|
||||
|
||||
if (disable) {
|
||||
/* disable card */
|
||||
command = 0x00;
|
||||
pci_bus_write_config_word(pci_bus, devfn, PCI_COMMAND, command);
|
||||
}
|
||||
|
||||
/* Check for Bridge */
|
||||
pci_bus_read_config_byte(pci_bus, devfn, PCI_HEADER_TYPE, &header_type);
|
||||
|
||||
if ((header_type & 0x7F) == PCI_HEADER_TYPE_BRIDGE) { /* PCI-PCI Bridge */
|
||||
dbg("Save_used_res of PCI bridge b:d=0x%x:%x, sc=0x%x\n",
|
||||
func->bus, func->device, save_command);
|
||||
if (disable) {
|
||||
/* Clear Bridge Control Register */
|
||||
command = 0x00;
|
||||
pci_bus_write_config_word(pci_bus, devfn, PCI_BRIDGE_CONTROL, command);
|
||||
}
|
||||
|
||||
pci_bus_read_config_byte(pci_bus, devfn, PCI_SECONDARY_BUS, &secondary_bus);
|
||||
pci_bus_read_config_byte(pci_bus, devfn, PCI_SUBORDINATE_BUS, &temp_byte);
|
||||
|
||||
bus_node = kmalloc(sizeof(struct pci_resource),
|
||||
GFP_KERNEL);
|
||||
if (!bus_node)
|
||||
return -ENOMEM;
|
||||
|
||||
bus_node->base = (ulong)secondary_bus;
|
||||
bus_node->length = (ulong)(temp_byte - secondary_bus + 1);
|
||||
|
||||
bus_node->next = func->bus_head;
|
||||
func->bus_head = bus_node;
|
||||
|
||||
/* Save IO base and Limit registers */
|
||||
pci_bus_read_config_byte(pci_bus, devfn, PCI_IO_BASE, &temp_byte);
|
||||
base = temp_byte;
|
||||
pci_bus_read_config_byte(pci_bus, devfn, PCI_IO_LIMIT, &temp_byte);
|
||||
length = temp_byte;
|
||||
|
||||
if ((base <= length) && (!disable || (save_command & PCI_COMMAND_IO))) {
|
||||
io_node = kmalloc(sizeof(struct pci_resource),
|
||||
GFP_KERNEL);
|
||||
if (!io_node)
|
||||
return -ENOMEM;
|
||||
|
||||
io_node->base = (ulong)(base & PCI_IO_RANGE_MASK) << 8;
|
||||
io_node->length = (ulong)(length - base + 0x10) << 8;
|
||||
|
||||
io_node->next = func->io_head;
|
||||
func->io_head = io_node;
|
||||
}
|
||||
|
||||
/* Save memory base and Limit registers */
|
||||
pci_bus_read_config_word(pci_bus, devfn, PCI_MEMORY_BASE, &w_base);
|
||||
pci_bus_read_config_word(pci_bus, devfn, PCI_MEMORY_LIMIT, &w_length);
|
||||
|
||||
if ((w_base <= w_length) && (!disable || (save_command & PCI_COMMAND_MEMORY))) {
|
||||
mem_node = kmalloc(sizeof(struct pci_resource),
|
||||
GFP_KERNEL);
|
||||
if (!mem_node)
|
||||
return -ENOMEM;
|
||||
|
||||
mem_node->base = (ulong)w_base << 16;
|
||||
mem_node->length = (ulong)(w_length - w_base + 0x10) << 16;
|
||||
|
||||
mem_node->next = func->mem_head;
|
||||
func->mem_head = mem_node;
|
||||
}
|
||||
/* Save prefetchable memory base and Limit registers */
|
||||
pci_bus_read_config_word(pci_bus, devfn, PCI_PREF_MEMORY_BASE, &w_base);
|
||||
pci_bus_read_config_word(pci_bus, devfn, PCI_PREF_MEMORY_LIMIT, &w_length);
|
||||
|
||||
if ((w_base <= w_length) && (!disable || (save_command & PCI_COMMAND_MEMORY))) {
|
||||
p_mem_node = kmalloc(sizeof(struct pci_resource),
|
||||
GFP_KERNEL);
|
||||
if (!p_mem_node)
|
||||
return -ENOMEM;
|
||||
|
||||
p_mem_node->base = (ulong)w_base << 16;
|
||||
p_mem_node->length = (ulong)(w_length - w_base + 0x10) << 16;
|
||||
|
||||
p_mem_node->next = func->p_mem_head;
|
||||
func->p_mem_head = p_mem_node;
|
||||
}
|
||||
} else if ((header_type & 0x7F) == PCI_HEADER_TYPE_NORMAL) {
|
||||
dbg("Save_used_res of PCI adapter b:d=0x%x:%x, sc=0x%x\n",
|
||||
func->bus, func->device, save_command);
|
||||
|
||||
/* Figure out IO and memory base lengths */
|
||||
for (cloop = PCI_BASE_ADDRESS_0; cloop <= PCI_BASE_ADDRESS_5; cloop += 4) {
|
||||
pci_bus_read_config_dword(pci_bus, devfn, cloop, &save_base);
|
||||
|
||||
temp_register = 0xFFFFFFFF;
|
||||
pci_bus_write_config_dword(pci_bus, devfn, cloop, temp_register);
|
||||
pci_bus_read_config_dword(pci_bus, devfn, cloop, &temp_register);
|
||||
|
||||
if (!disable)
|
||||
pci_bus_write_config_dword(pci_bus, devfn, cloop, save_base);
|
||||
|
||||
if (!temp_register)
|
||||
continue;
|
||||
|
||||
base = temp_register;
|
||||
|
||||
if ((base & PCI_BASE_ADDRESS_SPACE_IO) &&
|
||||
(!disable || (save_command & PCI_COMMAND_IO))) {
|
||||
/* IO base */
|
||||
/* set temp_register = amount of IO space requested */
|
||||
base = base & 0xFFFFFFFCL;
|
||||
base = (~base) + 1;
|
||||
|
||||
io_node = kmalloc(sizeof (struct pci_resource),
|
||||
GFP_KERNEL);
|
||||
if (!io_node)
|
||||
return -ENOMEM;
|
||||
|
||||
io_node->base = (ulong)save_base & PCI_BASE_ADDRESS_IO_MASK;
|
||||
io_node->length = (ulong)base;
|
||||
dbg("sur adapter: IO bar=0x%x(length=0x%x)\n",
|
||||
io_node->base, io_node->length);
|
||||
|
||||
io_node->next = func->io_head;
|
||||
func->io_head = io_node;
|
||||
} else { /* map Memory */
|
||||
int prefetchable = 1;
|
||||
/* struct pci_resources **res_node; */
|
||||
char *res_type_str = "PMEM";
|
||||
u32 temp_register2;
|
||||
|
||||
t_mem_node = kmalloc(sizeof (struct pci_resource),
|
||||
GFP_KERNEL);
|
||||
if (!t_mem_node)
|
||||
return -ENOMEM;
|
||||
|
||||
if (!(base & PCI_BASE_ADDRESS_MEM_PREFETCH) &&
|
||||
(!disable || (save_command & PCI_COMMAND_MEMORY))) {
|
||||
prefetchable = 0;
|
||||
mem_node = t_mem_node;
|
||||
res_type_str++;
|
||||
} else
|
||||
p_mem_node = t_mem_node;
|
||||
|
||||
base = base & 0xFFFFFFF0L;
|
||||
base = (~base) + 1;
|
||||
|
||||
switch (temp_register & PCI_BASE_ADDRESS_MEM_TYPE_MASK) {
|
||||
case PCI_BASE_ADDRESS_MEM_TYPE_32:
|
||||
if (prefetchable) {
|
||||
p_mem_node->base = (ulong)save_base & PCI_BASE_ADDRESS_MEM_MASK;
|
||||
p_mem_node->length = (ulong)base;
|
||||
dbg("sur adapter: 32 %s bar=0x%x(length=0x%x)\n",
|
||||
res_type_str,
|
||||
p_mem_node->base,
|
||||
p_mem_node->length);
|
||||
|
||||
p_mem_node->next = func->p_mem_head;
|
||||
func->p_mem_head = p_mem_node;
|
||||
} else {
|
||||
mem_node->base = (ulong)save_base & PCI_BASE_ADDRESS_MEM_MASK;
|
||||
mem_node->length = (ulong)base;
|
||||
dbg("sur adapter: 32 %s bar=0x%x(length=0x%x)\n",
|
||||
res_type_str,
|
||||
mem_node->base,
|
||||
mem_node->length);
|
||||
|
||||
mem_node->next = func->mem_head;
|
||||
func->mem_head = mem_node;
|
||||
}
|
||||
break;
|
||||
case PCI_BASE_ADDRESS_MEM_TYPE_64:
|
||||
pci_bus_read_config_dword(pci_bus, devfn, cloop+4, &temp_register2);
|
||||
base64 = temp_register2;
|
||||
base64 = (base64 << 32) | save_base;
|
||||
|
||||
if (temp_register2) {
|
||||
dbg("sur adapter: 64 %s high dword of base64(0x%x:%x) masked to 0\n",
|
||||
res_type_str, temp_register2, (u32)base64);
|
||||
base64 &= 0x00000000FFFFFFFFL;
|
||||
}
|
||||
|
||||
if (prefetchable) {
|
||||
p_mem_node->base = base64 & PCI_BASE_ADDRESS_MEM_MASK;
|
||||
p_mem_node->length = base;
|
||||
dbg("sur adapter: 64 %s base=0x%x(len=0x%x)\n",
|
||||
res_type_str,
|
||||
p_mem_node->base,
|
||||
p_mem_node->length);
|
||||
|
||||
p_mem_node->next = func->p_mem_head;
|
||||
func->p_mem_head = p_mem_node;
|
||||
} else {
|
||||
mem_node->base = base64 & PCI_BASE_ADDRESS_MEM_MASK;
|
||||
mem_node->length = base;
|
||||
dbg("sur adapter: 64 %s base=0x%x(len=0x%x)\n",
|
||||
res_type_str,
|
||||
mem_node->base,
|
||||
mem_node->length);
|
||||
|
||||
mem_node->next = func->mem_head;
|
||||
func->mem_head = mem_node;
|
||||
}
|
||||
cloop += 4;
|
||||
break;
|
||||
default:
|
||||
dbg("asur: reserved BAR type=0x%x\n",
|
||||
temp_register);
|
||||
break;
|
||||
}
|
||||
}
|
||||
} /* End of base register loop */
|
||||
} else { /* Some other unknown header type */
|
||||
dbg("Save_used_res of PCI unknown type b:d=0x%x:%x. skip.\n",
|
||||
func->bus, func->device);
|
||||
}
|
||||
|
||||
/* find the next device in this slot */
|
||||
if (!disable)
|
||||
break;
|
||||
func = shpchp_slot_find(func->bus, func->device, index++);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* kfree_resource_list: release memory of all list members
|
||||
* @res: resource list to free
|
||||
*/
|
||||
static inline void
|
||||
return_resource_list(struct pci_resource **func, struct pci_resource **res)
|
||||
{
|
||||
struct pci_resource *node;
|
||||
struct pci_resource *t_node;
|
||||
|
||||
node = *func;
|
||||
*func = NULL;
|
||||
while (node) {
|
||||
t_node = node->next;
|
||||
return_resource(res, node);
|
||||
node = t_node;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* shpchp_return_board_resources
|
||||
*
|
||||
* this routine returns all resources allocated to a board to
|
||||
* the available pool.
|
||||
*
|
||||
* returns 0 if success
|
||||
*/
|
||||
int shpchp_return_board_resources(struct pci_func * func,
|
||||
struct resource_lists * resources)
|
||||
{
|
||||
int rc;
|
||||
dbg("%s\n", __FUNCTION__);
|
||||
|
||||
if (!func)
|
||||
return 1;
|
||||
|
||||
return_resource_list(&(func->io_head),&(resources->io_head));
|
||||
return_resource_list(&(func->mem_head),&(resources->mem_head));
|
||||
return_resource_list(&(func->p_mem_head),&(resources->p_mem_head));
|
||||
return_resource_list(&(func->bus_head),&(resources->bus_head));
|
||||
|
||||
rc = shpchp_resource_sort_and_combine(&(resources->mem_head));
|
||||
rc |= shpchp_resource_sort_and_combine(&(resources->p_mem_head));
|
||||
rc |= shpchp_resource_sort_and_combine(&(resources->io_head));
|
||||
rc |= shpchp_resource_sort_and_combine(&(resources->bus_head));
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
/**
|
||||
* kfree_resource_list: release memory of all list members
|
||||
* @res: resource list to free
|
||||
*/
|
||||
static inline void
|
||||
kfree_resource_list(struct pci_resource **r)
|
||||
{
|
||||
struct pci_resource *res, *tres;
|
||||
|
||||
res = *r;
|
||||
*r = NULL;
|
||||
|
||||
while (res) {
|
||||
tres = res;
|
||||
res = res->next;
|
||||
kfree(tres);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* shpchp_destroy_resource_list: put node back in the resource list
|
||||
* @resources: list to put nodes back
|
||||
*/
|
||||
void shpchp_destroy_resource_list(struct resource_lists *resources)
|
||||
{
|
||||
kfree_resource_list(&(resources->io_head));
|
||||
kfree_resource_list(&(resources->mem_head));
|
||||
kfree_resource_list(&(resources->p_mem_head));
|
||||
kfree_resource_list(&(resources->bus_head));
|
||||
}
|
||||
|
||||
/**
|
||||
* shpchp_destroy_board_resources: put node back in the resource list
|
||||
* @resources: list to put nodes back
|
||||
*/
|
||||
void shpchp_destroy_board_resources(struct pci_func * func)
|
||||
{
|
||||
kfree_resource_list(&(func->io_head));
|
||||
kfree_resource_list(&(func->mem_head));
|
||||
kfree_resource_list(&(func->p_mem_head));
|
||||
kfree_resource_list(&(func->bus_head));
|
||||
}
|
||||
|
@ -26,12 +26,9 @@
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/proc_fs.h>
|
||||
#include <linux/workqueue.h>
|
||||
#include <linux/pci.h>
|
||||
#include "shpchp.h"
|
||||
|
||||
@ -40,104 +37,60 @@
|
||||
|
||||
static ssize_t show_ctrl (struct device *dev, struct device_attribute *attr, char *buf)
|
||||
{
|
||||
struct pci_dev *pci_dev;
|
||||
struct controller *ctrl;
|
||||
struct pci_dev *pdev;
|
||||
char * out = buf;
|
||||
int index;
|
||||
struct pci_resource *res;
|
||||
int index, busnr;
|
||||
struct resource *res;
|
||||
struct pci_bus *bus;
|
||||
|
||||
pci_dev = container_of (dev, struct pci_dev, dev);
|
||||
ctrl = pci_get_drvdata(pci_dev);
|
||||
pdev = container_of (dev, struct pci_dev, dev);
|
||||
bus = pdev->subordinate;
|
||||
|
||||
out += sprintf(buf, "Free resources: memory\n");
|
||||
index = 11;
|
||||
res = ctrl->mem_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
for (index = 0; index < PCI_BUS_NUM_RESOURCES; index++) {
|
||||
res = bus->resource[index];
|
||||
if (res && (res->flags & IORESOURCE_MEM) &&
|
||||
!(res->flags & IORESOURCE_PREFETCH)) {
|
||||
out += sprintf(out, "start = %8.8lx, length = %8.8lx\n",
|
||||
res->start, (res->end - res->start));
|
||||
}
|
||||
}
|
||||
out += sprintf(out, "Free resources: prefetchable memory\n");
|
||||
index = 11;
|
||||
res = ctrl->p_mem_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
for (index = 0; index < PCI_BUS_NUM_RESOURCES; index++) {
|
||||
res = bus->resource[index];
|
||||
if (res && (res->flags & IORESOURCE_MEM) &&
|
||||
(res->flags & IORESOURCE_PREFETCH)) {
|
||||
out += sprintf(out, "start = %8.8lx, length = %8.8lx\n",
|
||||
res->start, (res->end - res->start));
|
||||
}
|
||||
}
|
||||
out += sprintf(out, "Free resources: IO\n");
|
||||
index = 11;
|
||||
res = ctrl->io_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
for (index = 0; index < PCI_BUS_NUM_RESOURCES; index++) {
|
||||
res = bus->resource[index];
|
||||
if (res && (res->flags & IORESOURCE_IO)) {
|
||||
out += sprintf(out, "start = %8.8lx, length = %8.8lx\n",
|
||||
res->start, (res->end - res->start));
|
||||
}
|
||||
}
|
||||
out += sprintf(out, "Free resources: bus numbers\n");
|
||||
index = 11;
|
||||
res = ctrl->bus_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
for (busnr = bus->secondary; busnr <= bus->subordinate; busnr++) {
|
||||
if (!pci_find_bus(pci_domain_nr(bus), busnr))
|
||||
break;
|
||||
}
|
||||
if (busnr < bus->subordinate)
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n",
|
||||
busnr, (bus->subordinate - busnr));
|
||||
|
||||
return out - buf;
|
||||
}
|
||||
static DEVICE_ATTR (ctrl, S_IRUGO, show_ctrl, NULL);
|
||||
|
||||
static ssize_t show_dev (struct device *dev, struct device_attribute *attr, char *buf)
|
||||
{
|
||||
struct pci_dev *pci_dev;
|
||||
struct controller *ctrl;
|
||||
char * out = buf;
|
||||
int index;
|
||||
struct pci_resource *res;
|
||||
struct pci_func *new_slot;
|
||||
struct slot *slot;
|
||||
|
||||
pci_dev = container_of (dev, struct pci_dev, dev);
|
||||
ctrl = pci_get_drvdata(pci_dev);
|
||||
|
||||
slot=ctrl->slot;
|
||||
|
||||
while (slot) {
|
||||
new_slot = shpchp_slot_find(slot->bus, slot->device, 0);
|
||||
if (!new_slot)
|
||||
break;
|
||||
out += sprintf(out, "assigned resources: memory\n");
|
||||
index = 11;
|
||||
res = new_slot->mem_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
}
|
||||
out += sprintf(out, "assigned resources: prefetchable memory\n");
|
||||
index = 11;
|
||||
res = new_slot->p_mem_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
}
|
||||
out += sprintf(out, "assigned resources: IO\n");
|
||||
index = 11;
|
||||
res = new_slot->io_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
}
|
||||
out += sprintf(out, "assigned resources: bus numbers\n");
|
||||
index = 11;
|
||||
res = new_slot->bus_head;
|
||||
while (res && index--) {
|
||||
out += sprintf(out, "start = %8.8x, length = %8.8x\n", res->base, res->length);
|
||||
res = res->next;
|
||||
}
|
||||
slot=slot->next;
|
||||
}
|
||||
|
||||
return out - buf;
|
||||
}
|
||||
static DEVICE_ATTR (dev, S_IRUGO, show_dev, NULL);
|
||||
|
||||
void shpchp_create_ctrl_files (struct controller *ctrl)
|
||||
{
|
||||
device_create_file (&ctrl->pci_dev->dev, &dev_attr_ctrl);
|
||||
device_create_file (&ctrl->pci_dev->dev, &dev_attr_dev);
|
||||
}
|
||||
|
||||
void shpchp_remove_ctrl_files(struct controller *ctrl)
|
||||
{
|
||||
device_remove_file(&ctrl->pci_dev->dev, &dev_attr_ctrl);
|
||||
}
|
||||
|
@ -1,55 +0,0 @@
|
||||
/*
|
||||
* SHPCHPRM : SHPCHP Resource Manager for ACPI/non-ACPI platform
|
||||
*
|
||||
* Copyright (C) 1995,2001 Compaq Computer Corporation
|
||||
* Copyright (C) 2001 Greg Kroah-Hartman (greg@kroah.com)
|
||||
* Copyright (C) 2001 IBM Corp.
|
||||
* Copyright (C) 2003-2004 Intel Corporation
|
||||
*
|
||||
* All rights reserved.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or (at
|
||||
* your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, GOOD TITLE or
|
||||
* NON INFRINGEMENT. See the GNU General Public License for more
|
||||
* details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*
|
||||
* Send feedback to <greg@kroah.com>, <kristen.c.accardi@intel.com>
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef _SHPCHPRM_H_
|
||||
#define _SHPCHPRM_H_
|
||||
|
||||
#ifdef CONFIG_HOTPLUG_PCI_SHPC_PHPRM_LEGACY
|
||||
#include "shpchprm_legacy.h"
|
||||
#else
|
||||
#include "shpchprm_nonacpi.h"
|
||||
#endif
|
||||
|
||||
int shpchprm_init(enum php_ctlr_type ct);
|
||||
void shpchprm_cleanup(void);
|
||||
int shpchprm_print_pirt(void);
|
||||
int shpchprm_find_available_resources(struct controller *ctrl);
|
||||
int shpchprm_set_hpp(struct controller *ctrl, struct pci_func *func, u8 card_type);
|
||||
void shpchprm_enable_card(struct controller *ctrl, struct pci_func *func, u8 card_type);
|
||||
int shpchprm_get_physical_slot_number(struct controller *ctrl, u32 *sun, u8 busnum, u8 devnum);
|
||||
|
||||
#ifdef DEBUG
|
||||
#define RES_CHECK(this, bits) \
|
||||
{ if (((this) & (bits - 1))) \
|
||||
printk("%s:%d ERR: potential res loss!\n", __FUNCTION__, __LINE__); }
|
||||
#else
|
||||
#define RES_CHECK(this, bits)
|
||||
#endif
|
||||
|
||||
#endif /* _SHPCHPRM_H_ */
|
File diff suppressed because it is too large
Load Diff
@ -27,33 +27,11 @@
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/init.h>
|
||||
#include <asm/uaccess.h>
|
||||
#ifdef CONFIG_IA64
|
||||
#include <asm/iosapic.h>
|
||||
#endif
|
||||
#include "shpchp.h"
|
||||
#include "shpchprm.h"
|
||||
#include "shpchprm_legacy.h"
|
||||
|
||||
static void __iomem *shpchp_rom_start;
|
||||
static u16 unused_IRQ;
|
||||
|
||||
void shpchprm_cleanup(void)
|
||||
{
|
||||
if (shpchp_rom_start)
|
||||
iounmap(shpchp_rom_start);
|
||||
}
|
||||
|
||||
int shpchprm_print_pirt(void)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
int shpchprm_get_physical_slot_number(struct controller *ctrl, u32 *sun, u8 busnum, u8 devnum)
|
||||
{
|
||||
@ -63,377 +41,14 @@ int shpchprm_get_physical_slot_number(struct controller *ctrl, u32 *sun, u8 busn
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Find the Hot Plug Resource Table in the specified region of memory */
|
||||
static void __iomem *detect_HRT_floating_pointer(void __iomem *begin, void __iomem *end)
|
||||
void get_hp_params_from_firmware(struct pci_dev *dev,
|
||||
struct hotplug_params *hpp)
|
||||
{
|
||||
void __iomem *fp;
|
||||
void __iomem *endp;
|
||||
u8 temp1, temp2, temp3, temp4;
|
||||
int status = 0;
|
||||
|
||||
endp = (end - sizeof(struct hrt) + 1);
|
||||
|
||||
for (fp = begin; fp <= endp; fp += 16) {
|
||||
temp1 = readb(fp + SIG0);
|
||||
temp2 = readb(fp + SIG1);
|
||||
temp3 = readb(fp + SIG2);
|
||||
temp4 = readb(fp + SIG3);
|
||||
if (temp1 == '$' && temp2 == 'H' && temp3 == 'R' && temp4 == 'T') {
|
||||
status = 1;
|
||||
break;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (!status)
|
||||
fp = NULL;
|
||||
|
||||
dbg("Discovered Hotplug Resource Table at %p\n", fp);
|
||||
return fp;
|
||||
}
|
||||
|
||||
/*
|
||||
* shpchprm_find_available_resources
|
||||
*
|
||||
* Finds available memory, IO, and IRQ resources for programming
|
||||
* devices which may be added to the system
|
||||
* this function is for hot plug ADD!
|
||||
*
|
||||
* returns 0 if success
|
||||
*/
|
||||
int shpchprm_find_available_resources(struct controller *ctrl)
|
||||
void get_hp_hw_control_from_firmware(struct pci_dev *dev)
|
||||
{
|
||||
u8 populated_slot;
|
||||
u8 bridged_slot;
|
||||
void __iomem *one_slot;
|
||||
struct pci_func *func = NULL;
|
||||
int i = 10, index = 0;
|
||||
u32 temp_dword, rc;
|
||||
ulong temp_ulong;
|
||||
struct pci_resource *mem_node;
|
||||
struct pci_resource *p_mem_node;
|
||||
struct pci_resource *io_node;
|
||||
struct pci_resource *bus_node;
|
||||
void __iomem *rom_resource_table;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
u8 cfgspc_irq, temp;
|
||||
|
||||
memcpy(&lpci_bus, ctrl->pci_bus, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
rom_resource_table = detect_HRT_floating_pointer(shpchp_rom_start, shpchp_rom_start + 0xffff);
|
||||
dbg("rom_resource_table = %p\n", rom_resource_table);
|
||||
if (rom_resource_table == NULL)
|
||||
return -ENODEV;
|
||||
|
||||
/* Sum all resources and setup resource maps */
|
||||
unused_IRQ = readl(rom_resource_table + UNUSED_IRQ);
|
||||
dbg("unused_IRQ = %x\n", unused_IRQ);
|
||||
|
||||
temp = 0;
|
||||
while (unused_IRQ) {
|
||||
if (unused_IRQ & 1) {
|
||||
shpchp_disk_irq = temp;
|
||||
break;
|
||||
}
|
||||
unused_IRQ = unused_IRQ >> 1;
|
||||
temp++;
|
||||
return;
|
||||
}
|
||||
|
||||
dbg("shpchp_disk_irq= %d\n", shpchp_disk_irq);
|
||||
unused_IRQ = unused_IRQ >> 1;
|
||||
temp++;
|
||||
|
||||
while (unused_IRQ) {
|
||||
if (unused_IRQ & 1) {
|
||||
shpchp_nic_irq = temp;
|
||||
break;
|
||||
}
|
||||
unused_IRQ = unused_IRQ >> 1;
|
||||
temp++;
|
||||
}
|
||||
|
||||
dbg("shpchp_nic_irq= %d\n", shpchp_nic_irq);
|
||||
unused_IRQ = readl(rom_resource_table + PCIIRQ);
|
||||
|
||||
temp = 0;
|
||||
|
||||
pci_read_config_byte(ctrl->pci_dev, PCI_INTERRUPT_LINE, &cfgspc_irq);
|
||||
|
||||
if (!shpchp_nic_irq) {
|
||||
shpchp_nic_irq = cfgspc_irq;
|
||||
}
|
||||
|
||||
if (!shpchp_disk_irq) {
|
||||
shpchp_disk_irq = cfgspc_irq;
|
||||
}
|
||||
|
||||
dbg("shpchp_disk_irq, shpchp_nic_irq= %d, %d\n", shpchp_disk_irq, shpchp_nic_irq);
|
||||
|
||||
one_slot = rom_resource_table + sizeof(struct hrt);
|
||||
|
||||
i = readb(rom_resource_table + NUMBER_OF_ENTRIES);
|
||||
dbg("number_of_entries = %d\n", i);
|
||||
|
||||
if (!readb(one_slot + SECONDARY_BUS))
|
||||
return (1);
|
||||
|
||||
dbg("dev|IO base|length|MEMbase|length|PM base|length|PB SB MB\n");
|
||||
|
||||
while (i && readb(one_slot + SECONDARY_BUS)) {
|
||||
u8 dev_func = readb(one_slot + DEV_FUNC);
|
||||
u8 primary_bus = readb(one_slot + PRIMARY_BUS);
|
||||
u8 secondary_bus = readb(one_slot + SECONDARY_BUS);
|
||||
u8 max_bus = readb(one_slot + MAX_BUS);
|
||||
u16 io_base = readw(one_slot + IO_BASE);
|
||||
u16 io_length = readw(one_slot + IO_LENGTH);
|
||||
u16 mem_base = readw(one_slot + MEM_BASE);
|
||||
u16 mem_length = readw(one_slot + MEM_LENGTH);
|
||||
u16 pre_mem_base = readw(one_slot + PRE_MEM_BASE);
|
||||
u16 pre_mem_length = readw(one_slot + PRE_MEM_LENGTH);
|
||||
|
||||
dbg("%2.2x | %4.4x | %4.4x | %4.4x | %4.4x | %4.4x | %4.4x |%2.2x %2.2x %2.2x\n",
|
||||
dev_func, io_base, io_length, mem_base, mem_length, pre_mem_base, pre_mem_length,
|
||||
primary_bus, secondary_bus, max_bus);
|
||||
|
||||
/* If this entry isn't for our controller's bus, ignore it */
|
||||
if (primary_bus != ctrl->slot_bus) {
|
||||
i--;
|
||||
one_slot += sizeof(struct slot_rt);
|
||||
continue;
|
||||
}
|
||||
/* find out if this entry is for an occupied slot */
|
||||
temp_dword = 0xFFFFFFFF;
|
||||
pci_bus->number = primary_bus;
|
||||
pci_bus_read_config_dword(pci_bus, dev_func, PCI_VENDOR_ID, &temp_dword);
|
||||
|
||||
dbg("temp_D_word = %x\n", temp_dword);
|
||||
|
||||
if (temp_dword != 0xFFFFFFFF) {
|
||||
index = 0;
|
||||
func = shpchp_slot_find(primary_bus, dev_func >> 3, 0);
|
||||
|
||||
while (func && (func->function != (dev_func & 0x07))) {
|
||||
dbg("func = %p b:d:f(%x:%x:%x)\n", func, primary_bus, dev_func >> 3, index);
|
||||
func = shpchp_slot_find(primary_bus, dev_func >> 3, index++);
|
||||
}
|
||||
|
||||
/* If we can't find a match, skip this table entry */
|
||||
if (!func) {
|
||||
i--;
|
||||
one_slot += sizeof(struct slot_rt);
|
||||
continue;
|
||||
}
|
||||
/* this may not work and shouldn't be used */
|
||||
if (secondary_bus != primary_bus)
|
||||
bridged_slot = 1;
|
||||
else
|
||||
bridged_slot = 0;
|
||||
|
||||
populated_slot = 1;
|
||||
} else {
|
||||
populated_slot = 0;
|
||||
bridged_slot = 0;
|
||||
}
|
||||
dbg("slot populated =%s \n", populated_slot?"yes":"no");
|
||||
|
||||
/* If we've got a valid IO base, use it */
|
||||
|
||||
temp_ulong = io_base + io_length;
|
||||
|
||||
if ((io_base) && (temp_ulong <= 0x10000)) {
|
||||
io_node = (struct pci_resource *) kmalloc(sizeof(struct pci_resource), GFP_KERNEL);
|
||||
if (!io_node)
|
||||
return -ENOMEM;
|
||||
|
||||
io_node->base = (ulong)io_base;
|
||||
io_node->length = (ulong)io_length;
|
||||
dbg("found io_node(base, length) = %x, %x\n", io_node->base, io_node->length);
|
||||
|
||||
if (!populated_slot) {
|
||||
io_node->next = ctrl->io_head;
|
||||
ctrl->io_head = io_node;
|
||||
} else {
|
||||
io_node->next = func->io_head;
|
||||
func->io_head = io_node;
|
||||
}
|
||||
}
|
||||
|
||||
/* If we've got a valid memory base, use it */
|
||||
temp_ulong = mem_base + mem_length;
|
||||
if ((mem_base) && (temp_ulong <= 0x10000)) {
|
||||
mem_node = (struct pci_resource *) kmalloc(sizeof(struct pci_resource), GFP_KERNEL);
|
||||
if (!mem_node)
|
||||
return -ENOMEM;
|
||||
|
||||
mem_node->base = (ulong)mem_base << 16;
|
||||
mem_node->length = (ulong)(mem_length << 16);
|
||||
dbg("found mem_node(base, length) = %x, %x\n", mem_node->base, mem_node->length);
|
||||
|
||||
if (!populated_slot) {
|
||||
mem_node->next = ctrl->mem_head;
|
||||
ctrl->mem_head = mem_node;
|
||||
} else {
|
||||
mem_node->next = func->mem_head;
|
||||
func->mem_head = mem_node;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* If we've got a valid prefetchable memory base, and
|
||||
* the base + length isn't greater than 0xFFFF
|
||||
*/
|
||||
temp_ulong = pre_mem_base + pre_mem_length;
|
||||
if ((pre_mem_base) && (temp_ulong <= 0x10000)) {
|
||||
p_mem_node = (struct pci_resource *) kmalloc(sizeof(struct pci_resource), GFP_KERNEL);
|
||||
if (!p_mem_node)
|
||||
return -ENOMEM;
|
||||
|
||||
p_mem_node->base = (ulong)pre_mem_base << 16;
|
||||
p_mem_node->length = (ulong)pre_mem_length << 16;
|
||||
dbg("found p_mem_node(base, length) = %x, %x\n", p_mem_node->base, p_mem_node->length);
|
||||
|
||||
if (!populated_slot) {
|
||||
p_mem_node->next = ctrl->p_mem_head;
|
||||
ctrl->p_mem_head = p_mem_node;
|
||||
} else {
|
||||
p_mem_node->next = func->p_mem_head;
|
||||
func->p_mem_head = p_mem_node;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* If we've got a valid bus number, use it
|
||||
* The second condition is to ignore bus numbers on
|
||||
* populated slots that don't have PCI-PCI bridges
|
||||
*/
|
||||
if (secondary_bus && (secondary_bus != primary_bus)) {
|
||||
bus_node = (struct pci_resource *) kmalloc(sizeof(struct pci_resource), GFP_KERNEL);
|
||||
if (!bus_node)
|
||||
return -ENOMEM;
|
||||
|
||||
bus_node->base = (ulong)secondary_bus;
|
||||
bus_node->length = (ulong)(max_bus - secondary_bus + 1);
|
||||
dbg("found bus_node(base, length) = %x, %x\n", bus_node->base, bus_node->length);
|
||||
|
||||
if (!populated_slot) {
|
||||
bus_node->next = ctrl->bus_head;
|
||||
ctrl->bus_head = bus_node;
|
||||
} else {
|
||||
bus_node->next = func->bus_head;
|
||||
func->bus_head = bus_node;
|
||||
}
|
||||
}
|
||||
|
||||
i--;
|
||||
one_slot += sizeof(struct slot_rt);
|
||||
}
|
||||
|
||||
/* If all of the following fail, we don't have any resources for hot plug add */
|
||||
rc = 1;
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->mem_head));
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->p_mem_head));
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->io_head));
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->bus_head));
|
||||
|
||||
return (rc);
|
||||
}
|
||||
|
||||
int shpchprm_set_hpp(
|
||||
struct controller *ctrl,
|
||||
struct pci_func *func,
|
||||
u8 card_type)
|
||||
{
|
||||
u32 rc;
|
||||
u8 temp_byte;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
unsigned int devfn;
|
||||
memcpy(&lpci_bus, ctrl->pci_bus, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
pci_bus->number = func->bus;
|
||||
devfn = PCI_DEVFN(func->device, func->function);
|
||||
|
||||
temp_byte = 0x40; /* hard coded value for LT */
|
||||
if (card_type == PCI_HEADER_TYPE_BRIDGE) {
|
||||
/* set subordinate Latency Timer */
|
||||
rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_SEC_LATENCY_TIMER, temp_byte);
|
||||
if (rc) {
|
||||
dbg("%s: set secondary LT error. b:d:f(%02x:%02x:%02x)\n", __FUNCTION__, func->bus,
|
||||
func->device, func->function);
|
||||
return rc;
|
||||
}
|
||||
}
|
||||
|
||||
/* set base Latency Timer */
|
||||
rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_LATENCY_TIMER, temp_byte);
|
||||
if (rc) {
|
||||
dbg("%s: set LT error. b:d:f(%02x:%02x:%02x)\n", __FUNCTION__, func->bus, func->device, func->function);
|
||||
return rc;
|
||||
}
|
||||
|
||||
/* set Cache Line size */
|
||||
temp_byte = 0x08; /* hard coded value for CLS */
|
||||
rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_CACHE_LINE_SIZE, temp_byte);
|
||||
if (rc) {
|
||||
dbg("%s: set CLS error. b:d:f(%02x:%02x:%02x)\n", __FUNCTION__, func->bus, func->device, func->function);
|
||||
}
|
||||
|
||||
/* set enable_perr */
|
||||
/* set enable_serr */
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
void shpchprm_enable_card(
|
||||
struct controller *ctrl,
|
||||
struct pci_func *func,
|
||||
u8 card_type)
|
||||
{
|
||||
u16 command, bcommand;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
unsigned int devfn;
|
||||
int rc;
|
||||
|
||||
memcpy(&lpci_bus, ctrl->pci_bus, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
pci_bus->number = func->bus;
|
||||
devfn = PCI_DEVFN(func->device, func->function);
|
||||
|
||||
rc = pci_bus_read_config_word(pci_bus, devfn, PCI_COMMAND, &command);
|
||||
command |= PCI_COMMAND_PARITY | PCI_COMMAND_SERR
|
||||
| PCI_COMMAND_MASTER | PCI_COMMAND_INVALIDATE
|
||||
| PCI_COMMAND_IO | PCI_COMMAND_MEMORY;
|
||||
rc = pci_bus_write_config_word(pci_bus, devfn, PCI_COMMAND, command);
|
||||
|
||||
if (card_type == PCI_HEADER_TYPE_BRIDGE) {
|
||||
rc = pci_bus_read_config_word(pci_bus, devfn, PCI_BRIDGE_CONTROL, &bcommand);
|
||||
bcommand |= PCI_BRIDGE_CTL_PARITY | PCI_BRIDGE_CTL_SERR
|
||||
| PCI_BRIDGE_CTL_NO_ISA;
|
||||
rc = pci_bus_write_config_word(pci_bus, devfn, PCI_BRIDGE_CONTROL, bcommand);
|
||||
}
|
||||
}
|
||||
|
||||
static int legacy_shpchprm_init_pci(void)
|
||||
{
|
||||
shpchp_rom_start = ioremap(ROM_PHY_ADDR, ROM_PHY_LEN);
|
||||
if (!shpchp_rom_start) {
|
||||
err("Could not ioremap memory region for ROM\n");
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int shpchprm_init(enum php_ctlr_type ctrl_type)
|
||||
{
|
||||
int retval;
|
||||
|
||||
switch (ctrl_type) {
|
||||
case PCI:
|
||||
retval = legacy_shpchprm_init_pci();
|
||||
break;
|
||||
default:
|
||||
retval = -ENODEV;
|
||||
break;
|
||||
}
|
||||
|
||||
return retval;
|
||||
}
|
||||
|
@ -1,113 +0,0 @@
|
||||
/*
|
||||
* SHPCHPRM Legacy: PHP Resource Manager for Non-ACPI/Legacy platform using HRT
|
||||
*
|
||||
* Copyright (C) 1995,2001 Compaq Computer Corporation
|
||||
* Copyright (C) 2001 Greg Kroah-Hartman (greg@kroah.com)
|
||||
* Copyright (C) 2001 IBM Corp.
|
||||
* Copyright (C) 2003-2004 Intel Corporation
|
||||
*
|
||||
* All rights reserved.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or (at
|
||||
* your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, GOOD TITLE or
|
||||
* NON INFRINGEMENT. See the GNU General Public License for more
|
||||
* details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*
|
||||
* Send feedback to <greg@kroah.com>, <kristen.c.accardi@intel.com>
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef _SHPCHPRM_LEGACY_H_
|
||||
#define _SHPCHPRM_LEGACY_H_
|
||||
|
||||
#define ROM_PHY_ADDR 0x0F0000
|
||||
#define ROM_PHY_LEN 0x00FFFF
|
||||
|
||||
struct slot_rt {
|
||||
u8 dev_func;
|
||||
u8 primary_bus;
|
||||
u8 secondary_bus;
|
||||
u8 max_bus;
|
||||
u16 io_base;
|
||||
u16 io_length;
|
||||
u16 mem_base;
|
||||
u16 mem_length;
|
||||
u16 pre_mem_base;
|
||||
u16 pre_mem_length;
|
||||
} __attribute__ ((packed));
|
||||
|
||||
/* offsets to the hotplug slot resource table registers based on the above structure layout */
|
||||
enum slot_rt_offsets {
|
||||
DEV_FUNC = offsetof(struct slot_rt, dev_func),
|
||||
PRIMARY_BUS = offsetof(struct slot_rt, primary_bus),
|
||||
SECONDARY_BUS = offsetof(struct slot_rt, secondary_bus),
|
||||
MAX_BUS = offsetof(struct slot_rt, max_bus),
|
||||
IO_BASE = offsetof(struct slot_rt, io_base),
|
||||
IO_LENGTH = offsetof(struct slot_rt, io_length),
|
||||
MEM_BASE = offsetof(struct slot_rt, mem_base),
|
||||
MEM_LENGTH = offsetof(struct slot_rt, mem_length),
|
||||
PRE_MEM_BASE = offsetof(struct slot_rt, pre_mem_base),
|
||||
PRE_MEM_LENGTH = offsetof(struct slot_rt, pre_mem_length),
|
||||
};
|
||||
|
||||
struct hrt {
|
||||
char sig0;
|
||||
char sig1;
|
||||
char sig2;
|
||||
char sig3;
|
||||
u16 unused_IRQ;
|
||||
u16 PCIIRQ;
|
||||
u8 number_of_entries;
|
||||
u8 revision;
|
||||
u16 reserved1;
|
||||
u32 reserved2;
|
||||
} __attribute__ ((packed));
|
||||
|
||||
/* offsets to the hotplug resource table registers based on the above structure layout */
|
||||
enum hrt_offsets {
|
||||
SIG0 = offsetof(struct hrt, sig0),
|
||||
SIG1 = offsetof(struct hrt, sig1),
|
||||
SIG2 = offsetof(struct hrt, sig2),
|
||||
SIG3 = offsetof(struct hrt, sig3),
|
||||
UNUSED_IRQ = offsetof(struct hrt, unused_IRQ),
|
||||
PCIIRQ = offsetof(struct hrt, PCIIRQ),
|
||||
NUMBER_OF_ENTRIES = offsetof(struct hrt, number_of_entries),
|
||||
REVISION = offsetof(struct hrt, revision),
|
||||
HRT_RESERVED1 = offsetof(struct hrt, reserved1),
|
||||
HRT_RESERVED2 = offsetof(struct hrt, reserved2),
|
||||
};
|
||||
|
||||
struct irq_info {
|
||||
u8 bus, devfn; /* bus, device and function */
|
||||
struct {
|
||||
u8 link; /* IRQ line ID, chipset dependent, 0=not routed */
|
||||
u16 bitmap; /* Available IRQs */
|
||||
} __attribute__ ((packed)) irq[4];
|
||||
u8 slot; /* slot number, 0=onboard */
|
||||
u8 rfu;
|
||||
} __attribute__ ((packed));
|
||||
|
||||
struct irq_routing_table {
|
||||
u32 signature; /* PIRQ_SIGNATURE should be here */
|
||||
u16 version; /* PIRQ_VERSION */
|
||||
u16 size; /* Table size in bytes */
|
||||
u8 rtr_bus, rtr_devfn; /* Where the interrupt router lies */
|
||||
u16 exclusive_irqs; /* IRQs devoted exclusively to PCI usage */
|
||||
u16 rtr_vendor, rtr_device; /* Vendor and device ID of interrupt router */
|
||||
u32 miniport_data; /* Crap */
|
||||
u8 rfu[11];
|
||||
u8 checksum; /* Modulo 256 checksum must give zero */
|
||||
struct irq_info slots[0];
|
||||
} __attribute__ ((packed));
|
||||
|
||||
#endif /* _SHPCHPRM_LEGACY_H_ */
|
@ -32,24 +32,7 @@
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/init.h>
|
||||
#include <asm/uaccess.h>
|
||||
#ifdef CONFIG_IA64
|
||||
#include <asm/iosapic.h>
|
||||
#endif
|
||||
#include "shpchp.h"
|
||||
#include "shpchprm.h"
|
||||
#include "shpchprm_nonacpi.h"
|
||||
|
||||
void shpchprm_cleanup(void)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
int shpchprm_print_pirt(void)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
int shpchprm_get_physical_slot_number(struct controller *ctrl, u32 *sun, u8 busnum, u8 devnum)
|
||||
{
|
||||
@ -60,375 +43,13 @@ int shpchprm_get_physical_slot_number(struct controller *ctrl, u32 *sun, u8 busn
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void print_pci_resource ( struct pci_resource *aprh)
|
||||
void get_hp_params_from_firmware(struct pci_dev *dev,
|
||||
struct hotplug_params *hpp)
|
||||
{
|
||||
struct pci_resource *res;
|
||||
|
||||
for (res = aprh; res; res = res->next)
|
||||
dbg(" base= 0x%x length= 0x%x\n", res->base, res->length);
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
static void phprm_dump_func_res( struct pci_func *fun)
|
||||
void get_hp_hw_control_from_firmware(struct pci_dev *dev)
|
||||
{
|
||||
struct pci_func *func = fun;
|
||||
|
||||
if (func->bus_head) {
|
||||
dbg(": BUS Resources:\n");
|
||||
print_pci_resource (func->bus_head);
|
||||
}
|
||||
if (func->io_head) {
|
||||
dbg(": IO Resources:\n");
|
||||
print_pci_resource (func->io_head);
|
||||
}
|
||||
if (func->mem_head) {
|
||||
dbg(": MEM Resources:\n");
|
||||
print_pci_resource (func->mem_head);
|
||||
}
|
||||
if (func->p_mem_head) {
|
||||
dbg(": PMEM Resources:\n");
|
||||
print_pci_resource (func->p_mem_head);
|
||||
}
|
||||
}
|
||||
|
||||
static int phprm_get_used_resources (
|
||||
struct controller *ctrl,
|
||||
struct pci_func *func
|
||||
)
|
||||
{
|
||||
return shpchp_save_used_resources (ctrl, func, !DISABLE_CARD);
|
||||
}
|
||||
|
||||
static int phprm_delete_resource(
|
||||
struct pci_resource **aprh,
|
||||
ulong base,
|
||||
ulong size)
|
||||
{
|
||||
struct pci_resource *res;
|
||||
struct pci_resource *prevnode;
|
||||
struct pci_resource *split_node;
|
||||
ulong tbase;
|
||||
|
||||
shpchp_resource_sort_and_combine(aprh);
|
||||
|
||||
for (res = *aprh; res; res = res->next) {
|
||||
if (res->base > base)
|
||||
continue;
|
||||
|
||||
if ((res->base + res->length) < (base + size))
|
||||
continue;
|
||||
|
||||
if (res->base < base) {
|
||||
tbase = base;
|
||||
|
||||
if ((res->length - (tbase - res->base)) < size)
|
||||
continue;
|
||||
|
||||
split_node = (struct pci_resource *) kmalloc(sizeof(struct pci_resource), GFP_KERNEL);
|
||||
if (!split_node)
|
||||
return -ENOMEM;
|
||||
|
||||
split_node->base = res->base;
|
||||
split_node->length = tbase - res->base;
|
||||
res->base = tbase;
|
||||
res->length -= split_node->length;
|
||||
|
||||
split_node->next = res->next;
|
||||
res->next = split_node;
|
||||
}
|
||||
|
||||
if (res->length >= size) {
|
||||
split_node = (struct pci_resource*) kmalloc(sizeof(struct pci_resource), GFP_KERNEL);
|
||||
if (!split_node)
|
||||
return -ENOMEM;
|
||||
|
||||
split_node->base = res->base + size;
|
||||
split_node->length = res->length - size;
|
||||
res->length = size;
|
||||
|
||||
split_node->next = res->next;
|
||||
res->next = split_node;
|
||||
}
|
||||
|
||||
if (*aprh == res) {
|
||||
*aprh = res->next;
|
||||
} else {
|
||||
prevnode = *aprh;
|
||||
while (prevnode->next != res)
|
||||
prevnode = prevnode->next;
|
||||
|
||||
prevnode->next = res->next;
|
||||
}
|
||||
res->next = NULL;
|
||||
kfree(res);
|
||||
break;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static int phprm_delete_resources(
|
||||
struct pci_resource **aprh,
|
||||
struct pci_resource *this
|
||||
)
|
||||
{
|
||||
struct pci_resource *res;
|
||||
|
||||
for (res = this; res; res = res->next)
|
||||
phprm_delete_resource(aprh, res->base, res->length);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static int configure_existing_function(
|
||||
struct controller *ctrl,
|
||||
struct pci_func *func
|
||||
)
|
||||
{
|
||||
int rc;
|
||||
|
||||
/* see how much resources the func has used. */
|
||||
rc = phprm_get_used_resources (ctrl, func);
|
||||
|
||||
if (!rc) {
|
||||
/* subtract the resources used by the func from ctrl resources */
|
||||
rc = phprm_delete_resources (&ctrl->bus_head, func->bus_head);
|
||||
rc |= phprm_delete_resources (&ctrl->io_head, func->io_head);
|
||||
rc |= phprm_delete_resources (&ctrl->mem_head, func->mem_head);
|
||||
rc |= phprm_delete_resources (&ctrl->p_mem_head, func->p_mem_head);
|
||||
if (rc)
|
||||
warn("aCEF: cannot del used resources\n");
|
||||
} else
|
||||
err("aCEF: cannot get used resources\n");
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
static int bind_pci_resources_to_slots ( struct controller *ctrl)
|
||||
{
|
||||
struct pci_func *func, new_func;
|
||||
int busn = ctrl->slot_bus;
|
||||
int devn, funn;
|
||||
u32 vid;
|
||||
|
||||
for (devn = 0; devn < 32; devn++) {
|
||||
for (funn = 0; funn < 8; funn++) {
|
||||
/*
|
||||
if (devn == ctrl->device && funn == ctrl->function)
|
||||
continue;
|
||||
*/
|
||||
/* find out if this entry is for an occupied slot */
|
||||
vid = 0xFFFFFFFF;
|
||||
|
||||
pci_bus_read_config_dword(ctrl->pci_dev->subordinate, PCI_DEVFN(devn, funn), PCI_VENDOR_ID, &vid);
|
||||
|
||||
if (vid != 0xFFFFFFFF) {
|
||||
func = shpchp_slot_find(busn, devn, funn);
|
||||
if (!func) {
|
||||
memset(&new_func, 0, sizeof(struct pci_func));
|
||||
new_func.bus = busn;
|
||||
new_func.device = devn;
|
||||
new_func.function = funn;
|
||||
new_func.is_a_board = 1;
|
||||
configure_existing_function(ctrl, &new_func);
|
||||
phprm_dump_func_res(&new_func);
|
||||
} else {
|
||||
configure_existing_function(ctrl, func);
|
||||
phprm_dump_func_res(func);
|
||||
}
|
||||
dbg("aCCF:existing PCI 0x%x Func ResourceDump\n", ctrl->bus);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void phprm_dump_ctrl_res( struct controller *ctlr)
|
||||
{
|
||||
struct controller *ctrl = ctlr;
|
||||
|
||||
if (ctrl->bus_head) {
|
||||
dbg(": BUS Resources:\n");
|
||||
print_pci_resource (ctrl->bus_head);
|
||||
}
|
||||
if (ctrl->io_head) {
|
||||
dbg(": IO Resources:\n");
|
||||
print_pci_resource (ctrl->io_head);
|
||||
}
|
||||
if (ctrl->mem_head) {
|
||||
dbg(": MEM Resources:\n");
|
||||
print_pci_resource (ctrl->mem_head);
|
||||
}
|
||||
if (ctrl->p_mem_head) {
|
||||
dbg(": PMEM Resources:\n");
|
||||
print_pci_resource (ctrl->p_mem_head);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* phprm_find_available_resources
|
||||
*
|
||||
* Finds available memory, IO, and IRQ resources for programming
|
||||
* devices which may be added to the system
|
||||
* this function is for hot plug ADD!
|
||||
*
|
||||
* returns 0 if success
|
||||
*/
|
||||
int shpchprm_find_available_resources(struct controller *ctrl)
|
||||
{
|
||||
struct pci_func func;
|
||||
u32 rc;
|
||||
|
||||
memset(&func, 0, sizeof(struct pci_func));
|
||||
|
||||
func.bus = ctrl->bus;
|
||||
func.device = ctrl->device;
|
||||
func.function = ctrl->function;
|
||||
func.is_a_board = 1;
|
||||
|
||||
/* Get resources for this PCI bridge */
|
||||
rc = shpchp_save_used_resources (ctrl, &func, !DISABLE_CARD);
|
||||
dbg("%s: shpchp_save_used_resources rc = %d\n", __FUNCTION__, rc);
|
||||
|
||||
if (func.mem_head)
|
||||
func.mem_head->next = ctrl->mem_head;
|
||||
ctrl->mem_head = func.mem_head;
|
||||
|
||||
if (func.p_mem_head)
|
||||
func.p_mem_head->next = ctrl->p_mem_head;
|
||||
ctrl->p_mem_head = func.p_mem_head;
|
||||
|
||||
if (func.io_head)
|
||||
func.io_head->next = ctrl->io_head;
|
||||
ctrl->io_head = func.io_head;
|
||||
|
||||
if(func.bus_head)
|
||||
func.bus_head->next = ctrl->bus_head;
|
||||
ctrl->bus_head = func.bus_head;
|
||||
if (ctrl->bus_head)
|
||||
phprm_delete_resource(&ctrl->bus_head, ctrl->pci_dev->subordinate->number, 1);
|
||||
|
||||
dbg("%s:pre-Bind PCI 0x%x Ctrl Resource Dump\n", __FUNCTION__, ctrl->bus);
|
||||
phprm_dump_ctrl_res(ctrl);
|
||||
bind_pci_resources_to_slots (ctrl);
|
||||
|
||||
dbg("%s:post-Bind PCI 0x%x Ctrl Resource Dump\n", __FUNCTION__, ctrl->bus);
|
||||
phprm_dump_ctrl_res(ctrl);
|
||||
|
||||
|
||||
/* If all of the following fail, we don't have any resources for hot plug add */
|
||||
rc = 1;
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->mem_head));
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->p_mem_head));
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->io_head));
|
||||
rc &= shpchp_resource_sort_and_combine(&(ctrl->bus_head));
|
||||
|
||||
return (rc);
|
||||
}
|
||||
|
||||
int shpchprm_set_hpp(
|
||||
struct controller *ctrl,
|
||||
struct pci_func *func,
|
||||
u8 card_type)
|
||||
{
|
||||
u32 rc;
|
||||
u8 temp_byte;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
unsigned int devfn;
|
||||
memcpy(&lpci_bus, ctrl->pci_bus, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
pci_bus->number = func->bus;
|
||||
devfn = PCI_DEVFN(func->device, func->function);
|
||||
|
||||
temp_byte = 0x40; /* hard coded value for LT */
|
||||
if (card_type == PCI_HEADER_TYPE_BRIDGE) {
|
||||
/* set subordinate Latency Timer */
|
||||
rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_SEC_LATENCY_TIMER, temp_byte);
|
||||
|
||||
if (rc) {
|
||||
dbg("%s: set secondary LT error. b:d:f(%02x:%02x:%02x)\n", __FUNCTION__, func->bus,
|
||||
func->device, func->function);
|
||||
return rc;
|
||||
}
|
||||
}
|
||||
|
||||
/* set base Latency Timer */
|
||||
rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_LATENCY_TIMER, temp_byte);
|
||||
|
||||
if (rc) {
|
||||
dbg("%s: set LT error. b:d:f(%02x:%02x:%02x)\n", __FUNCTION__, func->bus, func->device, func->function);
|
||||
return rc;
|
||||
}
|
||||
|
||||
/* set Cache Line size */
|
||||
temp_byte = 0x08; /* hard coded value for CLS */
|
||||
|
||||
rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_CACHE_LINE_SIZE, temp_byte);
|
||||
|
||||
if (rc) {
|
||||
dbg("%s: set CLS error. b:d:f(%02x:%02x:%02x)\n", __FUNCTION__, func->bus, func->device, func->function);
|
||||
}
|
||||
|
||||
/* set enable_perr */
|
||||
/* set enable_serr */
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
void shpchprm_enable_card(
|
||||
struct controller *ctrl,
|
||||
struct pci_func *func,
|
||||
u8 card_type)
|
||||
{
|
||||
u16 command, bcommand;
|
||||
struct pci_bus lpci_bus, *pci_bus;
|
||||
unsigned int devfn;
|
||||
int rc;
|
||||
|
||||
memcpy(&lpci_bus, ctrl->pci_bus, sizeof(lpci_bus));
|
||||
pci_bus = &lpci_bus;
|
||||
pci_bus->number = func->bus;
|
||||
devfn = PCI_DEVFN(func->device, func->function);
|
||||
|
||||
rc = pci_bus_read_config_word(pci_bus, devfn, PCI_COMMAND, &command);
|
||||
|
||||
command |= PCI_COMMAND_PARITY | PCI_COMMAND_SERR
|
||||
| PCI_COMMAND_MASTER | PCI_COMMAND_INVALIDATE
|
||||
| PCI_COMMAND_IO | PCI_COMMAND_MEMORY;
|
||||
|
||||
rc = pci_bus_write_config_word(pci_bus, devfn, PCI_COMMAND, command);
|
||||
|
||||
if (card_type == PCI_HEADER_TYPE_BRIDGE) {
|
||||
|
||||
rc = pci_bus_read_config_word(pci_bus, devfn, PCI_BRIDGE_CONTROL, &bcommand);
|
||||
|
||||
bcommand |= PCI_BRIDGE_CTL_PARITY | PCI_BRIDGE_CTL_SERR
|
||||
| PCI_BRIDGE_CTL_NO_ISA;
|
||||
|
||||
rc = pci_bus_write_config_word(pci_bus, devfn, PCI_BRIDGE_CONTROL, bcommand);
|
||||
}
|
||||
}
|
||||
|
||||
static int legacy_shpchprm_init_pci(void)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
int shpchprm_init(enum php_ctlr_type ctrl_type)
|
||||
{
|
||||
int retval;
|
||||
|
||||
switch (ctrl_type) {
|
||||
case PCI:
|
||||
retval = legacy_shpchprm_init_pci();
|
||||
break;
|
||||
default:
|
||||
retval = -ENODEV;
|
||||
break;
|
||||
}
|
||||
|
||||
return retval;
|
||||
return;
|
||||
}
|
||||
|
@ -1,56 +0,0 @@
|
||||
/*
|
||||
* SHPCHPRM NONACPI: PHP Resource Manager for Non-ACPI/Legacy platform
|
||||
*
|
||||
* Copyright (C) 1995,2001 Compaq Computer Corporation
|
||||
* Copyright (C) 2001 Greg Kroah-Hartman (greg@kroah.com)
|
||||
* Copyright (C) 2001 IBM Corp.
|
||||
* Copyright (C) 2003-2004 Intel Corporation
|
||||
*
|
||||
* All rights reserved.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or (at
|
||||
* your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, GOOD TITLE or
|
||||
* NON INFRINGEMENT. See the GNU General Public License for more
|
||||
* details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*
|
||||
* Send feedback to <greg@kroah.com>, <kristen.c.accardi@intel.com>
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef _SHPCHPRM_NONACPI_H_
|
||||
#define _SHPCHPRM_NONACPI_H_
|
||||
|
||||
struct irq_info {
|
||||
u8 bus, devfn; /* bus, device and function */
|
||||
struct {
|
||||
u8 link; /* IRQ line ID, chipset dependent, 0=not routed */
|
||||
u16 bitmap; /* Available IRQs */
|
||||
} __attribute__ ((packed)) irq[4];
|
||||
u8 slot; /* slot number, 0=onboard */
|
||||
u8 rfu;
|
||||
} __attribute__ ((packed));
|
||||
|
||||
struct irq_routing_table {
|
||||
u32 signature; /* PIRQ_SIGNATURE should be here */
|
||||
u16 version; /* PIRQ_VERSION */
|
||||
u16 size; /* Table size in bytes */
|
||||
u8 rtr_bus, rtr_devfn; /* Where the interrupt router lies */
|
||||
u16 exclusive_irqs; /* IRQs devoted exclusively to PCI usage */
|
||||
u16 rtr_vendor, rtr_device; /* Vendor and device ID of interrupt router */
|
||||
u32 miniport_data; /* Crap */
|
||||
u8 rfu[11];
|
||||
u8 checksum; /* Modulo 256 checksum must give zero */
|
||||
struct irq_info slots[0];
|
||||
} __attribute__ ((packed));
|
||||
|
||||
#endif /* _SHPCHPRM_NONACPI_H_ */
|
@ -575,6 +575,8 @@ static int msi_capability_init(struct pci_dev *dev)
|
||||
/**
|
||||
* msix_capability_init - configure device's MSI-X capability
|
||||
* @dev: pointer to the pci_dev data structure of MSI-X device function
|
||||
* @entries: pointer to an array of struct msix_entry entries
|
||||
* @nvec: number of @entries
|
||||
*
|
||||
* Setup the MSI-X capability structure of device function with a
|
||||
* single MSI-X vector. A return of zero indicates the successful setup of
|
||||
|
@ -26,7 +26,10 @@ struct pci_dynid {
|
||||
#ifdef CONFIG_HOTPLUG
|
||||
|
||||
/**
|
||||
* store_new_id
|
||||
* store_new_id - add a new PCI device ID to this driver and re-probe devices
|
||||
* @driver: target device driver
|
||||
* @buf: buffer for scanning device ID data
|
||||
* @count: input size
|
||||
*
|
||||
* Adds a new dynamic pci device ID to this driver,
|
||||
* and causes the driver to probe for all devices again.
|
||||
@ -194,6 +197,8 @@ static int pci_call_probe(struct pci_driver *drv, struct pci_dev *dev,
|
||||
|
||||
/**
|
||||
* __pci_device_probe()
|
||||
* @drv: driver to call to check if it wants the PCI device
|
||||
* @pci_dev: PCI device being probed
|
||||
*
|
||||
* returns 0 on success, else error.
|
||||
* side-effect: pci_dev->driver is set to drv when drv claims pci_dev.
|
||||
@ -377,6 +382,10 @@ int pci_register_driver(struct pci_driver *drv)
|
||||
* the pci shutdown function, this test can go away. */
|
||||
if (!drv->driver.shutdown)
|
||||
drv->driver.shutdown = pci_device_shutdown;
|
||||
else
|
||||
printk(KERN_WARNING "Warning: PCI driver %s has a struct "
|
||||
"device_driver shutdown method, please update!\n",
|
||||
drv->name);
|
||||
drv->driver.owner = drv->owner;
|
||||
drv->driver.kobj.ktype = &pci_driver_kobj_type;
|
||||
|
||||
@ -436,8 +445,8 @@ pci_dev_driver(const struct pci_dev *dev)
|
||||
|
||||
/**
|
||||
* pci_bus_match - Tell if a PCI device structure has a matching PCI device id structure
|
||||
* @ids: array of PCI device id structures to search in
|
||||
* @dev: the PCI device structure to match against
|
||||
* @drv: the device driver to search for matching PCI device id structures
|
||||
*
|
||||
* Used by a driver to check whether a PCI device present in the
|
||||
* system is in its list of supported devices. Returns the matching
|
||||
|
@ -130,7 +130,7 @@ pci_read_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
|
||||
if ((off & 1) && size) {
|
||||
u8 val;
|
||||
pci_read_config_byte(dev, off, &val);
|
||||
pci_user_read_config_byte(dev, off, &val);
|
||||
data[off - init_off] = val;
|
||||
off++;
|
||||
size--;
|
||||
@ -138,7 +138,7 @@ pci_read_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
|
||||
if ((off & 3) && size > 2) {
|
||||
u16 val;
|
||||
pci_read_config_word(dev, off, &val);
|
||||
pci_user_read_config_word(dev, off, &val);
|
||||
data[off - init_off] = val & 0xff;
|
||||
data[off - init_off + 1] = (val >> 8) & 0xff;
|
||||
off += 2;
|
||||
@ -147,7 +147,7 @@ pci_read_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
|
||||
while (size > 3) {
|
||||
u32 val;
|
||||
pci_read_config_dword(dev, off, &val);
|
||||
pci_user_read_config_dword(dev, off, &val);
|
||||
data[off - init_off] = val & 0xff;
|
||||
data[off - init_off + 1] = (val >> 8) & 0xff;
|
||||
data[off - init_off + 2] = (val >> 16) & 0xff;
|
||||
@ -158,7 +158,7 @@ pci_read_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
|
||||
if (size >= 2) {
|
||||
u16 val;
|
||||
pci_read_config_word(dev, off, &val);
|
||||
pci_user_read_config_word(dev, off, &val);
|
||||
data[off - init_off] = val & 0xff;
|
||||
data[off - init_off + 1] = (val >> 8) & 0xff;
|
||||
off += 2;
|
||||
@ -167,7 +167,7 @@ pci_read_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
|
||||
if (size > 0) {
|
||||
u8 val;
|
||||
pci_read_config_byte(dev, off, &val);
|
||||
pci_user_read_config_byte(dev, off, &val);
|
||||
data[off - init_off] = val;
|
||||
off++;
|
||||
--size;
|
||||
@ -192,7 +192,7 @@ pci_write_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
}
|
||||
|
||||
if ((off & 1) && size) {
|
||||
pci_write_config_byte(dev, off, data[off - init_off]);
|
||||
pci_user_write_config_byte(dev, off, data[off - init_off]);
|
||||
off++;
|
||||
size--;
|
||||
}
|
||||
@ -200,7 +200,7 @@ pci_write_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
if ((off & 3) && size > 2) {
|
||||
u16 val = data[off - init_off];
|
||||
val |= (u16) data[off - init_off + 1] << 8;
|
||||
pci_write_config_word(dev, off, val);
|
||||
pci_user_write_config_word(dev, off, val);
|
||||
off += 2;
|
||||
size -= 2;
|
||||
}
|
||||
@ -210,7 +210,7 @@ pci_write_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
val |= (u32) data[off - init_off + 1] << 8;
|
||||
val |= (u32) data[off - init_off + 2] << 16;
|
||||
val |= (u32) data[off - init_off + 3] << 24;
|
||||
pci_write_config_dword(dev, off, val);
|
||||
pci_user_write_config_dword(dev, off, val);
|
||||
off += 4;
|
||||
size -= 4;
|
||||
}
|
||||
@ -218,13 +218,13 @@ pci_write_config(struct kobject *kobj, char *buf, loff_t off, size_t count)
|
||||
if (size >= 2) {
|
||||
u16 val = data[off - init_off];
|
||||
val |= (u16) data[off - init_off + 1] << 8;
|
||||
pci_write_config_word(dev, off, val);
|
||||
pci_user_write_config_word(dev, off, val);
|
||||
off += 2;
|
||||
size -= 2;
|
||||
}
|
||||
|
||||
if (size) {
|
||||
pci_write_config_byte(dev, off, data[off - init_off]);
|
||||
pci_user_write_config_byte(dev, off, data[off - init_off]);
|
||||
off++;
|
||||
--size;
|
||||
}
|
||||
|
@ -252,6 +252,8 @@ pci_restore_bars(struct pci_dev *dev)
|
||||
pci_update_resource(dev, &dev->resource[i], i);
|
||||
}
|
||||
|
||||
int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t t);
|
||||
|
||||
/**
|
||||
* pci_set_power_state - Set the power state of a PCI device
|
||||
* @dev: PCI device to be suspended
|
||||
@ -266,7 +268,6 @@ pci_restore_bars(struct pci_dev *dev)
|
||||
* -EIO if device does not support PCI PM.
|
||||
* 0 if we can successfully change the power state.
|
||||
*/
|
||||
int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t t);
|
||||
int
|
||||
pci_set_power_state(struct pci_dev *dev, pci_power_t state)
|
||||
{
|
||||
@ -314,19 +315,19 @@ pci_set_power_state(struct pci_dev *dev, pci_power_t state)
|
||||
* sets PowerState to 0.
|
||||
*/
|
||||
switch (dev->current_state) {
|
||||
case PCI_D0:
|
||||
case PCI_D1:
|
||||
case PCI_D2:
|
||||
pmcsr &= ~PCI_PM_CTRL_STATE_MASK;
|
||||
pmcsr |= state;
|
||||
break;
|
||||
case PCI_UNKNOWN: /* Boot-up */
|
||||
if ((pmcsr & PCI_PM_CTRL_STATE_MASK) == PCI_D3hot
|
||||
&& !(pmcsr & PCI_PM_CTRL_NO_SOFT_RESET))
|
||||
need_restore = 1;
|
||||
/* Fall-through: force to D0 */
|
||||
case PCI_D3hot:
|
||||
case PCI_D3cold:
|
||||
case PCI_POWER_ERROR:
|
||||
pmcsr = 0;
|
||||
break;
|
||||
default:
|
||||
pmcsr &= ~PCI_PM_CTRL_STATE_MASK;
|
||||
pmcsr |= state;
|
||||
pmcsr = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
@ -808,8 +809,8 @@ pci_clear_mwi(struct pci_dev *dev)
|
||||
|
||||
/**
|
||||
* pci_intx - enables/disables PCI INTx for device dev
|
||||
* @dev: the PCI device to operate on
|
||||
* @enable: boolean
|
||||
* @pdev: the PCI device to operate on
|
||||
* @enable: boolean: whether to enable or disable PCI INTx
|
||||
*
|
||||
* Enables/disables PCI INTx for device dev
|
||||
*/
|
||||
|
@ -15,6 +15,13 @@ extern int pci_bus_alloc_resource(struct pci_bus *bus, struct resource *res,
|
||||
extern int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state);
|
||||
extern int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t state);
|
||||
|
||||
extern int pci_user_read_config_byte(struct pci_dev *dev, int where, u8 *val);
|
||||
extern int pci_user_read_config_word(struct pci_dev *dev, int where, u16 *val);
|
||||
extern int pci_user_read_config_dword(struct pci_dev *dev, int where, u32 *val);
|
||||
extern int pci_user_write_config_byte(struct pci_dev *dev, int where, u8 val);
|
||||
extern int pci_user_write_config_word(struct pci_dev *dev, int where, u16 val);
|
||||
extern int pci_user_write_config_dword(struct pci_dev *dev, int where, u32 val);
|
||||
|
||||
/* PCI /proc functions */
|
||||
#ifdef CONFIG_PROC_FS
|
||||
extern int pci_proc_attach_device(struct pci_dev *dev);
|
||||
|
@ -669,6 +669,7 @@ static void pci_release_dev(struct device *dev)
|
||||
|
||||
/**
|
||||
* pci_cfg_space_size - get the configuration space size of the PCI device.
|
||||
* @dev: PCI device
|
||||
*
|
||||
* Regular PCI devices have 256 bytes, but PCI-X 2 and PCI Express devices
|
||||
* have 4096 bytes. Even if the device is capable, that doesn't mean we can
|
||||
|
@ -80,7 +80,7 @@ proc_bus_pci_read(struct file *file, char __user *buf, size_t nbytes, loff_t *pp
|
||||
|
||||
if ((pos & 1) && cnt) {
|
||||
unsigned char val;
|
||||
pci_read_config_byte(dev, pos, &val);
|
||||
pci_user_read_config_byte(dev, pos, &val);
|
||||
__put_user(val, buf);
|
||||
buf++;
|
||||
pos++;
|
||||
@ -89,7 +89,7 @@ proc_bus_pci_read(struct file *file, char __user *buf, size_t nbytes, loff_t *pp
|
||||
|
||||
if ((pos & 3) && cnt > 2) {
|
||||
unsigned short val;
|
||||
pci_read_config_word(dev, pos, &val);
|
||||
pci_user_read_config_word(dev, pos, &val);
|
||||
__put_user(cpu_to_le16(val), (unsigned short __user *) buf);
|
||||
buf += 2;
|
||||
pos += 2;
|
||||
@ -98,7 +98,7 @@ proc_bus_pci_read(struct file *file, char __user *buf, size_t nbytes, loff_t *pp
|
||||
|
||||
while (cnt >= 4) {
|
||||
unsigned int val;
|
||||
pci_read_config_dword(dev, pos, &val);
|
||||
pci_user_read_config_dword(dev, pos, &val);
|
||||
__put_user(cpu_to_le32(val), (unsigned int __user *) buf);
|
||||
buf += 4;
|
||||
pos += 4;
|
||||
@ -107,7 +107,7 @@ proc_bus_pci_read(struct file *file, char __user *buf, size_t nbytes, loff_t *pp
|
||||
|
||||
if (cnt >= 2) {
|
||||
unsigned short val;
|
||||
pci_read_config_word(dev, pos, &val);
|
||||
pci_user_read_config_word(dev, pos, &val);
|
||||
__put_user(cpu_to_le16(val), (unsigned short __user *) buf);
|
||||
buf += 2;
|
||||
pos += 2;
|
||||
@ -116,7 +116,7 @@ proc_bus_pci_read(struct file *file, char __user *buf, size_t nbytes, loff_t *pp
|
||||
|
||||
if (cnt) {
|
||||
unsigned char val;
|
||||
pci_read_config_byte(dev, pos, &val);
|
||||
pci_user_read_config_byte(dev, pos, &val);
|
||||
__put_user(val, buf);
|
||||
buf++;
|
||||
pos++;
|
||||
@ -151,7 +151,7 @@ proc_bus_pci_write(struct file *file, const char __user *buf, size_t nbytes, lof
|
||||
if ((pos & 1) && cnt) {
|
||||
unsigned char val;
|
||||
__get_user(val, buf);
|
||||
pci_write_config_byte(dev, pos, val);
|
||||
pci_user_write_config_byte(dev, pos, val);
|
||||
buf++;
|
||||
pos++;
|
||||
cnt--;
|
||||
@ -160,7 +160,7 @@ proc_bus_pci_write(struct file *file, const char __user *buf, size_t nbytes, lof
|
||||
if ((pos & 3) && cnt > 2) {
|
||||
unsigned short val;
|
||||
__get_user(val, (unsigned short __user *) buf);
|
||||
pci_write_config_word(dev, pos, le16_to_cpu(val));
|
||||
pci_user_write_config_word(dev, pos, le16_to_cpu(val));
|
||||
buf += 2;
|
||||
pos += 2;
|
||||
cnt -= 2;
|
||||
@ -169,7 +169,7 @@ proc_bus_pci_write(struct file *file, const char __user *buf, size_t nbytes, lof
|
||||
while (cnt >= 4) {
|
||||
unsigned int val;
|
||||
__get_user(val, (unsigned int __user *) buf);
|
||||
pci_write_config_dword(dev, pos, le32_to_cpu(val));
|
||||
pci_user_write_config_dword(dev, pos, le32_to_cpu(val));
|
||||
buf += 4;
|
||||
pos += 4;
|
||||
cnt -= 4;
|
||||
@ -178,7 +178,7 @@ proc_bus_pci_write(struct file *file, const char __user *buf, size_t nbytes, lof
|
||||
if (cnt >= 2) {
|
||||
unsigned short val;
|
||||
__get_user(val, (unsigned short __user *) buf);
|
||||
pci_write_config_word(dev, pos, le16_to_cpu(val));
|
||||
pci_user_write_config_word(dev, pos, le16_to_cpu(val));
|
||||
buf += 2;
|
||||
pos += 2;
|
||||
cnt -= 2;
|
||||
@ -187,7 +187,7 @@ proc_bus_pci_write(struct file *file, const char __user *buf, size_t nbytes, lof
|
||||
if (cnt) {
|
||||
unsigned char val;
|
||||
__get_user(val, buf);
|
||||
pci_write_config_byte(dev, pos, val);
|
||||
pci_user_write_config_byte(dev, pos, val);
|
||||
buf++;
|
||||
pos++;
|
||||
cnt--;
|
||||
@ -484,10 +484,10 @@ static int show_dev_config(struct seq_file *m, void *v)
|
||||
|
||||
drv = pci_dev_driver(dev);
|
||||
|
||||
pci_read_config_dword(dev, PCI_CLASS_REVISION, &class_rev);
|
||||
pci_read_config_byte (dev, PCI_LATENCY_TIMER, &latency);
|
||||
pci_read_config_byte (dev, PCI_MIN_GNT, &min_gnt);
|
||||
pci_read_config_byte (dev, PCI_MAX_LAT, &max_lat);
|
||||
pci_user_read_config_dword(dev, PCI_CLASS_REVISION, &class_rev);
|
||||
pci_user_read_config_byte (dev, PCI_LATENCY_TIMER, &latency);
|
||||
pci_user_read_config_byte (dev, PCI_MIN_GNT, &min_gnt);
|
||||
pci_user_read_config_byte (dev, PCI_MAX_LAT, &max_lat);
|
||||
seq_printf(m, " Bus %2d, device %3d, function %2d:\n",
|
||||
dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn));
|
||||
seq_printf(m, " Class %04x", class_rev >> 16);
|
||||
|
@ -414,6 +414,18 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801DB_12,
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801EB_0, quirk_ich4_lpc_acpi );
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_ESB_1, quirk_ich4_lpc_acpi );
|
||||
|
||||
static void __devinit quirk_ich6_lpc_acpi(struct pci_dev *dev)
|
||||
{
|
||||
u32 region;
|
||||
|
||||
pci_read_config_dword(dev, 0x40, ®ion);
|
||||
quirk_io_region(dev, region, 128, PCI_BRIDGE_RESOURCES, "ICH6 ACPI/GPIO/TCO");
|
||||
|
||||
pci_read_config_dword(dev, 0x48, ®ion);
|
||||
quirk_io_region(dev, region, 64, PCI_BRIDGE_RESOURCES+1, "ICH6 GPIO");
|
||||
}
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_ICH6_1, quirk_ich6_lpc_acpi );
|
||||
|
||||
/*
|
||||
* VIA ACPI: One IO region pointed to by longword at
|
||||
* 0x48 or 0x20 (256 bytes of ACPI registers)
|
||||
@ -922,6 +934,12 @@ static void __init asus_hides_smbus_hostbridge(struct pci_dev *dev)
|
||||
case 0x186a: /* M6Ne notebook */
|
||||
asus_hides_smbus = 1;
|
||||
}
|
||||
if (dev->device == PCI_DEVICE_ID_INTEL_82915GM_HB) {
|
||||
switch (dev->subsystem_device) {
|
||||
case 0x1882: /* M6V notebook */
|
||||
asus_hides_smbus = 1;
|
||||
}
|
||||
}
|
||||
} else if (unlikely(dev->subsystem_vendor == PCI_VENDOR_ID_HP)) {
|
||||
if (dev->device == PCI_DEVICE_ID_INTEL_82855PM_HB)
|
||||
switch(dev->subsystem_device) {
|
||||
@ -932,6 +950,7 @@ static void __init asus_hides_smbus_hostbridge(struct pci_dev *dev)
|
||||
if (dev->device == PCI_DEVICE_ID_INTEL_82865_HB)
|
||||
switch (dev->subsystem_device) {
|
||||
case 0x12bc: /* HP D330L */
|
||||
case 0x12bd: /* HP D530 */
|
||||
asus_hides_smbus = 1;
|
||||
}
|
||||
} else if (unlikely(dev->subsystem_vendor == PCI_VENDOR_ID_TOSHIBA)) {
|
||||
@ -966,6 +985,7 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82865_HB, asus
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_7205_0, asus_hides_smbus_hostbridge );
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82855PM_HB, asus_hides_smbus_hostbridge );
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82855GM_HB, asus_hides_smbus_hostbridge );
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82915GM_HB, asus_hides_smbus_hostbridge );
|
||||
|
||||
static void __init asus_hides_smbus_lpc(struct pci_dev *dev)
|
||||
{
|
||||
@ -990,6 +1010,23 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801CA_12, as
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801DB_12, asus_hides_smbus_lpc );
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801EB_0, asus_hides_smbus_lpc );
|
||||
|
||||
static void __init asus_hides_smbus_lpc_ich6(struct pci_dev *dev)
|
||||
{
|
||||
u32 val, rcba;
|
||||
void __iomem *base;
|
||||
|
||||
if (likely(!asus_hides_smbus))
|
||||
return;
|
||||
pci_read_config_dword(dev, 0xF0, &rcba);
|
||||
base = ioremap_nocache(rcba & 0xFFFFC000, 0x4000); /* use bits 31:14, 16 kB aligned */
|
||||
if (base == NULL) return;
|
||||
val=readl(base + 0x3418); /* read the Function Disable register, dword mode only */
|
||||
writel(val & 0xFFFFFFF7, base + 0x3418); /* enable the SMBus device */
|
||||
iounmap(base);
|
||||
printk(KERN_INFO "PCI: Enabled ICH6/i801 SMBus device\n");
|
||||
}
|
||||
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_ICH6_1, asus_hides_smbus_lpc_ich6 );
|
||||
|
||||
/*
|
||||
* SiS 96x south bridge: BIOS typically hides SMBus device...
|
||||
*/
|
||||
|
@ -13,7 +13,7 @@
|
||||
#include <linux/smp_lock.h>
|
||||
#include <linux/syscalls.h>
|
||||
#include <asm/uaccess.h>
|
||||
|
||||
#include "pci.h"
|
||||
|
||||
asmlinkage long
|
||||
sys_pciconfig_read(unsigned long bus, unsigned long dfn,
|
||||
@ -38,13 +38,13 @@ sys_pciconfig_read(unsigned long bus, unsigned long dfn,
|
||||
lock_kernel();
|
||||
switch (len) {
|
||||
case 1:
|
||||
cfg_ret = pci_read_config_byte(dev, off, &byte);
|
||||
cfg_ret = pci_user_read_config_byte(dev, off, &byte);
|
||||
break;
|
||||
case 2:
|
||||
cfg_ret = pci_read_config_word(dev, off, &word);
|
||||
cfg_ret = pci_user_read_config_word(dev, off, &word);
|
||||
break;
|
||||
case 4:
|
||||
cfg_ret = pci_read_config_dword(dev, off, &dword);
|
||||
cfg_ret = pci_user_read_config_dword(dev, off, &dword);
|
||||
break;
|
||||
default:
|
||||
err = -EINVAL;
|
||||
@ -112,7 +112,7 @@ sys_pciconfig_write(unsigned long bus, unsigned long dfn,
|
||||
err = get_user(byte, (u8 __user *)buf);
|
||||
if (err)
|
||||
break;
|
||||
err = pci_write_config_byte(dev, off, byte);
|
||||
err = pci_user_write_config_byte(dev, off, byte);
|
||||
if (err != PCIBIOS_SUCCESSFUL)
|
||||
err = -EIO;
|
||||
break;
|
||||
@ -121,7 +121,7 @@ sys_pciconfig_write(unsigned long bus, unsigned long dfn,
|
||||
err = get_user(word, (u16 __user *)buf);
|
||||
if (err)
|
||||
break;
|
||||
err = pci_write_config_word(dev, off, word);
|
||||
err = pci_user_write_config_word(dev, off, word);
|
||||
if (err != PCIBIOS_SUCCESSFUL)
|
||||
err = -EIO;
|
||||
break;
|
||||
@ -130,7 +130,7 @@ sys_pciconfig_write(unsigned long bus, unsigned long dfn,
|
||||
err = get_user(dword, (u32 __user *)buf);
|
||||
if (err)
|
||||
break;
|
||||
err = pci_write_config_dword(dev, off, dword);
|
||||
err = pci_user_write_config_dword(dev, off, dword);
|
||||
if (err != PCIBIOS_SUCCESSFUL)
|
||||
err = -EIO;
|
||||
break;
|
||||
|
@ -4944,6 +4944,7 @@ static int ipr_reset_restore_cfg_space(struct ipr_cmnd *ipr_cmd)
|
||||
int rc;
|
||||
|
||||
ENTER;
|
||||
pci_unblock_user_cfg_access(ioa_cfg->pdev);
|
||||
rc = pci_restore_state(ioa_cfg->pdev);
|
||||
|
||||
if (rc != PCIBIOS_SUCCESSFUL) {
|
||||
@ -4998,6 +4999,7 @@ static int ipr_reset_start_bist(struct ipr_cmnd *ipr_cmd)
|
||||
int rc;
|
||||
|
||||
ENTER;
|
||||
pci_block_user_cfg_access(ioa_cfg->pdev);
|
||||
rc = pci_write_config_byte(ioa_cfg->pdev, PCI_BIST, PCI_BIST_START);
|
||||
|
||||
if (rc != PCIBIOS_SUCCESSFUL) {
|
||||
|
@ -76,7 +76,7 @@ static void megaraid_exit(void);
|
||||
|
||||
static int megaraid_probe_one(struct pci_dev*, const struct pci_device_id *);
|
||||
static void megaraid_detach_one(struct pci_dev *);
|
||||
static void megaraid_mbox_shutdown(struct device *);
|
||||
static void megaraid_mbox_shutdown(struct pci_dev *);
|
||||
|
||||
static int megaraid_io_attach(adapter_t *);
|
||||
static void megaraid_io_detach(adapter_t *);
|
||||
@ -369,9 +369,7 @@ static struct pci_driver megaraid_pci_driver_g = {
|
||||
.id_table = pci_id_table_g,
|
||||
.probe = megaraid_probe_one,
|
||||
.remove = __devexit_p(megaraid_detach_one),
|
||||
.driver = {
|
||||
.shutdown = megaraid_mbox_shutdown,
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
@ -673,9 +671,9 @@ megaraid_detach_one(struct pci_dev *pdev)
|
||||
* Shutdown notification, perform flush cache
|
||||
*/
|
||||
static void
|
||||
megaraid_mbox_shutdown(struct device *device)
|
||||
megaraid_mbox_shutdown(struct pci_dev *pdev)
|
||||
{
|
||||
adapter_t *adapter = pci_get_drvdata(to_pci_dev(device));
|
||||
adapter_t *adapter = pci_get_drvdata(pdev);
|
||||
static int counter;
|
||||
|
||||
if (!adapter) {
|
||||
|
@ -275,20 +275,20 @@ static const struct cirrusfb_board_info_rec {
|
||||
|
||||
#ifdef CONFIG_PCI
|
||||
#define CHIP(id, btype) \
|
||||
{ PCI_VENDOR_ID_CIRRUS, PCI_DEVICE_ID_##id, PCI_ANY_ID, PCI_ANY_ID, 0, 0, (btype) }
|
||||
{ PCI_VENDOR_ID_CIRRUS, id, PCI_ANY_ID, PCI_ANY_ID, 0, 0, (btype) }
|
||||
|
||||
static struct pci_device_id cirrusfb_pci_table[] = {
|
||||
CHIP( CIRRUS_5436, BT_ALPINE ),
|
||||
CHIP( CIRRUS_5434_8, BT_ALPINE ),
|
||||
CHIP( CIRRUS_5434_4, BT_ALPINE ),
|
||||
CHIP( CIRRUS_5430, BT_ALPINE ), /* GD-5440 has identical id */
|
||||
CHIP( CIRRUS_7543, BT_ALPINE ),
|
||||
CHIP( CIRRUS_7548, BT_ALPINE ),
|
||||
CHIP( CIRRUS_5480, BT_GD5480 ), /* MacPicasso probably */
|
||||
CHIP( CIRRUS_5446, BT_PICASSO4 ), /* Picasso 4 is a GD5446 */
|
||||
CHIP( CIRRUS_5462, BT_LAGUNA ), /* CL Laguna */
|
||||
CHIP( CIRRUS_5464, BT_LAGUNA ), /* CL Laguna 3D */
|
||||
CHIP( CIRRUS_5465, BT_LAGUNA ), /* CL Laguna 3DA*/
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5436, BT_ALPINE ),
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5434_8, BT_ALPINE ),
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5434_4, BT_ALPINE ),
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5430, BT_ALPINE ), /* GD-5440 is same id */
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_7543, BT_ALPINE ),
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_7548, BT_ALPINE ),
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5480, BT_GD5480 ), /* MacPicasso likely */
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5446, BT_PICASSO4 ), /* Picasso 4 is 5446 */
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5462, BT_LAGUNA ), /* CL Laguna */
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5464, BT_LAGUNA ), /* CL Laguna 3D */
|
||||
CHIP( PCI_DEVICE_ID_CIRRUS_5465, BT_LAGUNA ), /* CL Laguna 3DA*/
|
||||
{ 0, }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(pci, cirrusfb_pci_table);
|
||||
|
@ -132,6 +132,7 @@ struct pci_dev {
|
||||
unsigned int is_enabled:1; /* pci_enable_device has been called */
|
||||
unsigned int is_busmaster:1; /* device is busmaster */
|
||||
unsigned int no_msi:1; /* device may not use msi */
|
||||
unsigned int block_ucfg_access:1; /* userspace config space access is blocked */
|
||||
|
||||
u32 saved_config_space[16]; /* config space saved at suspend time */
|
||||
struct bin_attribute *rom_attr; /* attribute descriptor for sysfs ROM entry */
|
||||
@ -490,6 +491,9 @@ extern void pci_disable_msix(struct pci_dev *dev);
|
||||
extern void msi_remove_pci_irq_vectors(struct pci_dev *dev);
|
||||
#endif
|
||||
|
||||
extern void pci_block_user_cfg_access(struct pci_dev *dev);
|
||||
extern void pci_unblock_user_cfg_access(struct pci_dev *dev);
|
||||
|
||||
/*
|
||||
* PCI domain support. Sometimes called PCI segment (eg by ACPI),
|
||||
* a PCI domain is defined to be a set of PCI busses which share
|
||||
@ -560,6 +564,9 @@ static inline int pci_enable_wake(struct pci_dev *dev, pci_power_t state, int en
|
||||
|
||||
#define pci_dma_burst_advice(pdev, strat, strategy_parameter) do { } while (0)
|
||||
|
||||
static inline void pci_block_user_cfg_access(struct pci_dev *dev) { }
|
||||
static inline void pci_unblock_user_cfg_access(struct pci_dev *dev) { }
|
||||
|
||||
#endif /* CONFIG_PCI */
|
||||
|
||||
/* Include architecture-dependent settings and functions */
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -107,14 +107,15 @@ static LIST_HEAD(ymf_devs);
|
||||
*/
|
||||
|
||||
static struct pci_device_id ymf_id_tbl[] = {
|
||||
#define DEV(v, d, data) \
|
||||
{ PCI_VENDOR_ID_##v, PCI_DEVICE_ID_##v##_##d, PCI_ANY_ID, PCI_ANY_ID, 0, 0, (unsigned long)data }
|
||||
DEV (YAMAHA, 724, "YMF724"),
|
||||
DEV (YAMAHA, 724F, "YMF724F"),
|
||||
DEV (YAMAHA, 740, "YMF740"),
|
||||
DEV (YAMAHA, 740C, "YMF740C"),
|
||||
DEV (YAMAHA, 744, "YMF744"),
|
||||
DEV (YAMAHA, 754, "YMF754"),
|
||||
#define DEV(dev, data) \
|
||||
{ PCI_VENDOR_ID_YAMAHA, dev, PCI_ANY_ID, PCI_ANY_ID, 0, 0, \
|
||||
(unsigned long)data }
|
||||
DEV (PCI_DEVICE_ID_YAMAHA_724, "YMF724"),
|
||||
DEV (PCI_DEVICE_ID_YAMAHA_724F, "YMF724F"),
|
||||
DEV (PCI_DEVICE_ID_YAMAHA_740, "YMF740"),
|
||||
DEV (PCI_DEVICE_ID_YAMAHA_740C, "YMF740C"),
|
||||
DEV (PCI_DEVICE_ID_YAMAHA_744, "YMF744"),
|
||||
DEV (PCI_DEVICE_ID_YAMAHA_754, "YMF754"),
|
||||
#undef DEV
|
||||
{ }
|
||||
};
|
||||
|
@ -761,15 +761,18 @@ static int __devinit snd_bt87x_create(snd_card_t *card,
|
||||
|
||||
#define BT_DEVICE(chip, subvend, subdev, rate) \
|
||||
{ .vendor = PCI_VENDOR_ID_BROOKTREE, \
|
||||
.device = PCI_DEVICE_ID_BROOKTREE_##chip, \
|
||||
.device = chip, \
|
||||
.subvendor = subvend, .subdevice = subdev, \
|
||||
.driver_data = rate }
|
||||
|
||||
/* driver_data is the default digital_rate value for that device */
|
||||
static struct pci_device_id snd_bt87x_ids[] = {
|
||||
BT_DEVICE(878, 0x0070, 0x13eb, 32000), /* Hauppauge WinTV series */
|
||||
BT_DEVICE(879, 0x0070, 0x13eb, 32000), /* Hauppauge WinTV series */
|
||||
BT_DEVICE(878, 0x0070, 0xff01, 44100), /* Viewcast Osprey 200 */
|
||||
/* Hauppauge WinTV series */
|
||||
BT_DEVICE(PCI_DEVICE_ID_BROOKTREE_878, 0x0070, 0x13eb, 32000),
|
||||
/* Hauppauge WinTV series */
|
||||
BT_DEVICE(PCI_DEVICE_ID_BROOKTREE_879, 0x0070, 0x13eb, 32000),
|
||||
/* Viewcast Osprey 200 */
|
||||
BT_DEVICE(PCI_DEVICE_ID_BROOKTREE_878, 0x0070, 0xff01, 44100),
|
||||
{ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(pci, snd_bt87x_ids);
|
||||
|
Loading…
Reference in New Issue
Block a user