2008-10-13 19:18:07 +08:00
|
|
|
#ifndef DRIVERS_PCI_H
|
|
|
|
#define DRIVERS_PCI_H
|
|
|
|
|
2009-03-20 11:25:16 +08:00
|
|
|
#include <linux/workqueue.h>
|
|
|
|
|
2008-10-13 19:18:07 +08:00
|
|
|
#define PCI_CFG_SPACE_SIZE 256
|
|
|
|
#define PCI_CFG_SPACE_EXP_SIZE 4096
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Functions internal to the PCI core code */
|
|
|
|
|
2007-08-14 21:15:12 +08:00
|
|
|
extern int pci_uevent(struct device *dev, struct kobj_uevent_env *env);
|
2005-04-17 06:20:36 +08:00
|
|
|
extern int pci_create_sysfs_dev_files(struct pci_dev *pdev);
|
|
|
|
extern void pci_remove_sysfs_dev_files(struct pci_dev *pdev);
|
2011-03-03 01:04:17 +08:00
|
|
|
#if !defined(CONFIG_DMI) && !defined(CONFIG_ACPI)
|
2010-07-26 18:56:50 +08:00
|
|
|
static inline void pci_create_firmware_label_files(struct pci_dev *pdev)
|
2010-08-02 20:44:29 +08:00
|
|
|
{ return; }
|
2010-07-26 18:56:50 +08:00
|
|
|
static inline void pci_remove_firmware_label_files(struct pci_dev *pdev)
|
2010-08-02 20:44:29 +08:00
|
|
|
{ return; }
|
2010-07-26 18:56:50 +08:00
|
|
|
#else
|
|
|
|
extern void pci_create_firmware_label_files(struct pci_dev *pdev);
|
|
|
|
extern void pci_remove_firmware_label_files(struct pci_dev *pdev);
|
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
extern void pci_cleanup_rom(struct pci_dev *dev);
|
2008-10-25 01:32:33 +08:00
|
|
|
#ifdef HAVE_PCI_MMAP
|
2010-11-10 18:03:21 +08:00
|
|
|
enum pci_mmap_api {
|
|
|
|
PCI_MMAP_SYSFS, /* mmap on /sys/bus/pci/devices/<BDF>/resource<N> */
|
|
|
|
PCI_MMAP_PROCFS /* mmap on /proc/bus/pci/<BDF> */
|
|
|
|
};
|
2008-10-25 01:32:33 +08:00
|
|
|
extern int pci_mmap_fits(struct pci_dev *pdev, int resno,
|
2010-11-10 18:03:21 +08:00
|
|
|
struct vm_area_struct *vmai,
|
|
|
|
enum pci_mmap_api mmap_api);
|
2008-10-25 01:32:33 +08:00
|
|
|
#endif
|
2009-07-28 04:37:48 +08:00
|
|
|
int pci_probe_reset_function(struct pci_dev *dev);
|
2007-07-17 12:27:10 +08:00
|
|
|
|
2008-07-07 09:32:02 +08:00
|
|
|
/**
|
2009-01-10 09:04:26 +08:00
|
|
|
* struct pci_platform_pm_ops - Firmware PM callbacks
|
2008-07-07 09:32:02 +08:00
|
|
|
*
|
2009-01-10 09:04:26 +08:00
|
|
|
* @is_manageable: returns 'true' if given device is power manageable by the
|
|
|
|
* platform firmware
|
2008-07-07 09:32:02 +08:00
|
|
|
*
|
2009-01-10 09:04:26 +08:00
|
|
|
* @set_state: invokes the platform firmware to set the device's power state
|
2008-07-07 09:32:02 +08:00
|
|
|
*
|
2009-01-10 09:04:26 +08:00
|
|
|
* @choose_state: returns PCI power state of given device preferred by the
|
|
|
|
* platform; to be used during system-wide transitions from a
|
|
|
|
* sleeping state to the working state and vice versa
|
2008-07-07 09:32:02 +08:00
|
|
|
*
|
2009-01-10 09:04:26 +08:00
|
|
|
* @can_wakeup: returns 'true' if given device is capable of waking up the
|
|
|
|
* system from a sleeping state
|
2008-07-07 09:34:48 +08:00
|
|
|
*
|
2009-01-10 09:04:26 +08:00
|
|
|
* @sleep_wake: enables/disables the system wake up capability of given device
|
2008-07-07 09:34:48 +08:00
|
|
|
*
|
2010-02-18 06:44:09 +08:00
|
|
|
* @run_wake: enables/disables the platform to generate run-time wake-up events
|
|
|
|
* for given device (the device's wake-up capability has to be
|
|
|
|
* enabled by @sleep_wake for this feature to work)
|
|
|
|
*
|
2008-07-07 09:32:02 +08:00
|
|
|
* If given platform is generally capable of power managing PCI devices, all of
|
|
|
|
* these callbacks are mandatory.
|
|
|
|
*/
|
|
|
|
struct pci_platform_pm_ops {
|
|
|
|
bool (*is_manageable)(struct pci_dev *dev);
|
|
|
|
int (*set_state)(struct pci_dev *dev, pci_power_t state);
|
|
|
|
pci_power_t (*choose_state)(struct pci_dev *dev);
|
2008-07-07 09:34:48 +08:00
|
|
|
bool (*can_wakeup)(struct pci_dev *dev);
|
|
|
|
int (*sleep_wake)(struct pci_dev *dev, bool enable);
|
2010-02-18 06:44:09 +08:00
|
|
|
int (*run_wake)(struct pci_dev *dev, bool enable);
|
2008-07-07 09:32:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
extern int pci_set_platform_pm(struct pci_platform_pm_ops *ops);
|
PCI PM: Avoid touching devices behind bridges in unknown state
It generally is better to avoid accessing devices behind bridges that
may not be in the D0 power state, because in that case the bridges'
secondary buses may not be accessible. For this reason, during the
early phase of resume (ie. with interrupts disabled), before
restoring the standard config registers of a device, check the power
state of the bridge the device is behind and postpone the restoration
of the device's config space, as well as any other operations that
would involve accessing the device, if that state is not D0.
In such cases the restoration of the device's config space will be
retried during the "normal" phase of resume (ie. with interrupts
enabled), so that the bridge can be put into D0 before that happens.
Also, save standard configuration registers of PCI devices during the
"normal" phase of suspend (ie. with interrupts enabled), so that the
bridges the devices are behind can be put into low power states (we
don't put bridges into low power states at the moment, but we may
want to do it in the future and it seems reasonable to design for
that).
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Signed-off-by: Jesse Barnes <jbarnes@virtuousgeek.org>
2009-01-07 20:07:15 +08:00
|
|
|
extern void pci_update_current_state(struct pci_dev *dev, pci_power_t state);
|
2009-01-07 20:03:42 +08:00
|
|
|
extern void pci_disable_enabled_device(struct pci_dev *dev);
|
2010-02-18 06:44:58 +08:00
|
|
|
extern int pci_finish_runtime_suspend(struct pci_dev *dev);
|
2010-02-18 06:44:09 +08:00
|
|
|
extern int __pci_pme_wakeup(struct pci_dev *dev, void *ign);
|
2008-07-07 09:34:48 +08:00
|
|
|
extern void pci_pm_init(struct pci_dev *dev);
|
2008-12-18 04:10:05 +08:00
|
|
|
extern void platform_pci_wakeup_init(struct pci_dev *dev);
|
2008-12-08 05:02:58 +08:00
|
|
|
extern void pci_allocate_cap_save_buffers(struct pci_dev *dev);
|
2009-01-17 04:54:43 +08:00
|
|
|
|
2010-12-29 20:21:23 +08:00
|
|
|
static inline void pci_wakeup_event(struct pci_dev *dev)
|
|
|
|
{
|
|
|
|
/* Wait 100 ms before the system can be put into a sleep state. */
|
|
|
|
pm_wakeup_event(&dev->dev, 100);
|
|
|
|
}
|
|
|
|
|
2009-01-17 04:54:43 +08:00
|
|
|
static inline bool pci_is_bridge(struct pci_dev *pci_dev)
|
|
|
|
{
|
|
|
|
return !!(pci_dev->subordinate);
|
|
|
|
}
|
2005-03-19 13:15:48 +08:00
|
|
|
|
2005-09-27 16:21:55 +08:00
|
|
|
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);
|
|
|
|
|
2008-03-06 00:52:39 +08:00
|
|
|
struct pci_vpd_ops {
|
2008-12-19 01:17:16 +08:00
|
|
|
ssize_t (*read)(struct pci_dev *dev, loff_t pos, size_t count, void *buf);
|
|
|
|
ssize_t (*write)(struct pci_dev *dev, loff_t pos, size_t count, const void *buf);
|
2008-03-06 00:52:39 +08:00
|
|
|
void (*release)(struct pci_dev *dev);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct pci_vpd {
|
2008-07-03 01:59:04 +08:00
|
|
|
unsigned int len;
|
2008-12-19 01:17:16 +08:00
|
|
|
const struct pci_vpd_ops *ops;
|
2008-03-06 00:52:39 +08:00
|
|
|
struct bin_attribute *attr; /* descriptor for sysfs VPD entry */
|
|
|
|
};
|
|
|
|
|
|
|
|
extern int pci_vpd_pci22_init(struct pci_dev *dev);
|
|
|
|
static inline void pci_vpd_release(struct pci_dev *dev)
|
|
|
|
{
|
|
|
|
if (dev->vpd)
|
|
|
|
dev->vpd->ops->release(dev);
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* PCI /proc functions */
|
|
|
|
#ifdef CONFIG_PROC_FS
|
|
|
|
extern int pci_proc_attach_device(struct pci_dev *dev);
|
|
|
|
extern int pci_proc_detach_device(struct pci_dev *dev);
|
|
|
|
extern int pci_proc_detach_bus(struct pci_bus *bus);
|
|
|
|
#else
|
|
|
|
static inline int pci_proc_attach_device(struct pci_dev *dev) { return 0; }
|
|
|
|
static inline int pci_proc_detach_device(struct pci_dev *dev) { return 0; }
|
|
|
|
static inline int pci_proc_detach_bus(struct pci_bus *bus) { return 0; }
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Functions for PCI Hotplug drivers to use */
|
|
|
|
extern unsigned int pci_do_scan_bus(struct pci_bus *bus);
|
|
|
|
|
2008-10-03 17:49:32 +08:00
|
|
|
#ifdef HAVE_PCI_LEGACY
|
|
|
|
extern void pci_create_legacy_files(struct pci_bus *bus);
|
2005-04-17 06:20:36 +08:00
|
|
|
extern void pci_remove_legacy_files(struct pci_bus *bus);
|
2008-10-03 17:49:32 +08:00
|
|
|
#else
|
|
|
|
static inline void pci_create_legacy_files(struct pci_bus *bus) { return; }
|
|
|
|
static inline void pci_remove_legacy_files(struct pci_bus *bus) { return; }
|
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Lock for read/write access to pci device and bus lists */
|
2006-06-02 12:35:43 +08:00
|
|
|
extern struct rw_semaphore pci_bus_sem;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-07-12 23:59:00 +08:00
|
|
|
extern unsigned int pci_pm_d3_delay;
|
2007-01-25 16:34:07 +08:00
|
|
|
|
2005-08-17 06:16:05 +08:00
|
|
|
#ifdef CONFIG_PCI_MSI
|
2006-03-06 13:33:34 +08:00
|
|
|
void pci_no_msi(void);
|
2007-04-05 15:19:10 +08:00
|
|
|
extern void pci_msi_init_pci_dev(struct pci_dev *dev);
|
2005-08-17 06:16:05 +08:00
|
|
|
#else
|
2006-03-06 13:33:34 +08:00
|
|
|
static inline void pci_no_msi(void) { }
|
2007-04-05 15:19:10 +08:00
|
|
|
static inline void pci_msi_init_pci_dev(struct pci_dev *dev) { }
|
2005-08-17 06:16:05 +08:00
|
|
|
#endif
|
2007-01-25 16:34:08 +08:00
|
|
|
|
2011-07-08 02:19:10 +08:00
|
|
|
extern void pci_realloc(void);
|
|
|
|
|
2006-07-12 23:59:00 +08:00
|
|
|
static inline int pci_no_d1d2(struct pci_dev *dev)
|
|
|
|
{
|
|
|
|
unsigned int parent_dstates = 0;
|
2005-08-17 06:16:05 +08:00
|
|
|
|
2006-07-12 23:59:00 +08:00
|
|
|
if (dev->bus->self)
|
|
|
|
parent_dstates = dev->bus->self->no_d1d2;
|
|
|
|
return (dev->no_d1d2 || parent_dstates);
|
|
|
|
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
extern struct device_attribute pci_dev_attrs[];
|
2011-05-13 08:11:39 +08:00
|
|
|
extern struct device_attribute pcibus_dev_attrs[];
|
2009-03-21 04:56:31 +08:00
|
|
|
#ifdef CONFIG_HOTPLUG
|
|
|
|
extern struct bus_attribute pci_bus_attrs[];
|
|
|
|
#else
|
|
|
|
#define pci_bus_attrs NULL
|
|
|
|
#endif
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* pci_match_one_device - Tell if a PCI device structure has a matching
|
|
|
|
* PCI device id structure
|
|
|
|
* @id: single PCI device id structure to match
|
|
|
|
* @dev: the PCI device structure to match against
|
2008-01-31 07:21:33 +08:00
|
|
|
*
|
2005-04-17 06:20:36 +08:00
|
|
|
* Returns the matching pci_device_id structure or %NULL if there is no match.
|
|
|
|
*/
|
|
|
|
static inline const struct pci_device_id *
|
|
|
|
pci_match_one_device(const struct pci_device_id *id, const struct pci_dev *dev)
|
|
|
|
{
|
|
|
|
if ((id->vendor == PCI_ANY_ID || id->vendor == dev->vendor) &&
|
|
|
|
(id->device == PCI_ANY_ID || id->device == dev->device) &&
|
|
|
|
(id->subvendor == PCI_ANY_ID || id->subvendor == dev->subsystem_vendor) &&
|
|
|
|
(id->subdevice == PCI_ANY_ID || id->subdevice == dev->subsystem_device) &&
|
|
|
|
!((id->class ^ dev->class) & id->class_mask))
|
|
|
|
return id;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-10-22 07:41:46 +08:00
|
|
|
struct pci_dev *pci_find_upstream_pcie_bridge(struct pci_dev *pdev);
|
2008-06-11 05:28:50 +08:00
|
|
|
|
|
|
|
/* PCI slot sysfs helper code */
|
|
|
|
#define to_pci_slot(s) container_of(s, struct pci_slot, kobj)
|
|
|
|
|
|
|
|
extern struct kset *pci_slots_kset;
|
|
|
|
|
|
|
|
struct pci_slot_attribute {
|
|
|
|
struct attribute attr;
|
|
|
|
ssize_t (*show)(struct pci_slot *, char *);
|
|
|
|
ssize_t (*store)(struct pci_slot *, const char *, size_t);
|
|
|
|
};
|
|
|
|
#define to_pci_slot_attr(s) container_of(s, struct pci_slot_attribute, attr)
|
|
|
|
|
2008-11-22 02:40:40 +08:00
|
|
|
enum pci_bar_type {
|
|
|
|
pci_bar_unknown, /* Standard PCI BAR probe */
|
|
|
|
pci_bar_io, /* An io port BAR */
|
|
|
|
pci_bar_mem32, /* A 32-bit memory BAR */
|
|
|
|
pci_bar_mem64, /* A 64-bit memory BAR */
|
|
|
|
};
|
|
|
|
|
2009-03-20 11:25:14 +08:00
|
|
|
extern int pci_setup_device(struct pci_dev *dev);
|
2008-11-22 02:40:40 +08:00
|
|
|
extern int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type,
|
|
|
|
struct resource *res, unsigned int reg);
|
2008-11-22 02:41:27 +08:00
|
|
|
extern int pci_resource_bar(struct pci_dev *dev, int resno,
|
|
|
|
enum pci_bar_type *type);
|
2008-11-22 02:42:35 +08:00
|
|
|
extern int pci_bus_add_child(struct pci_bus *bus);
|
2008-10-14 14:02:53 +08:00
|
|
|
extern void pci_enable_ari(struct pci_dev *dev);
|
|
|
|
/**
|
|
|
|
* pci_ari_enabled - query ARI forwarding status
|
2008-11-22 02:38:21 +08:00
|
|
|
* @bus: the PCI bus
|
2008-10-14 14:02:53 +08:00
|
|
|
*
|
|
|
|
* Returns 1 if ARI forwarding is enabled, or 0 if not enabled;
|
|
|
|
*/
|
2008-11-22 02:38:21 +08:00
|
|
|
static inline int pci_ari_enabled(struct pci_bus *bus)
|
2008-10-14 14:02:53 +08:00
|
|
|
{
|
2008-11-22 02:38:21 +08:00
|
|
|
return bus->self && bus->self->ari_enabled;
|
2008-10-14 14:02:53 +08:00
|
|
|
}
|
|
|
|
|
2009-03-16 16:13:39 +08:00
|
|
|
#ifdef CONFIG_PCI_QUIRKS
|
|
|
|
extern int pci_is_reassigndev(struct pci_dev *dev);
|
|
|
|
resource_size_t pci_specified_resource_alignment(struct pci_dev *dev);
|
|
|
|
extern void pci_disable_bridge_window(struct pci_dev *dev);
|
|
|
|
#endif
|
|
|
|
|
2009-03-20 11:25:11 +08:00
|
|
|
/* Single Root I/O Virtualization */
|
|
|
|
struct pci_sriov {
|
|
|
|
int pos; /* capability position */
|
|
|
|
int nres; /* number of resources */
|
|
|
|
u32 cap; /* SR-IOV Capabilities */
|
|
|
|
u16 ctrl; /* SR-IOV Control */
|
|
|
|
u16 total; /* total VFs associated with the PF */
|
2009-03-20 11:25:15 +08:00
|
|
|
u16 initial; /* initial VFs associated with the PF */
|
|
|
|
u16 nr_virtfn; /* number of VFs available */
|
2009-03-20 11:25:11 +08:00
|
|
|
u16 offset; /* first VF Routing ID offset */
|
|
|
|
u16 stride; /* following VF stride */
|
|
|
|
u32 pgsz; /* page size for BAR alignment */
|
|
|
|
u8 link; /* Function Dependency Link */
|
|
|
|
struct pci_dev *dev; /* lowest numbered PF */
|
|
|
|
struct pci_dev *self; /* this PF */
|
|
|
|
struct mutex lock; /* lock for VF bus */
|
2009-03-20 11:25:16 +08:00
|
|
|
struct work_struct mtask; /* VF Migration task */
|
|
|
|
u8 __iomem *mstate; /* VF Migration State Array */
|
2009-03-20 11:25:11 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef CONFIG_PCI_IOV
|
|
|
|
extern int pci_iov_init(struct pci_dev *dev);
|
|
|
|
extern void pci_iov_release(struct pci_dev *dev);
|
|
|
|
extern int pci_iov_resource_bar(struct pci_dev *dev, int resno,
|
|
|
|
enum pci_bar_type *type);
|
2010-09-08 08:25:20 +08:00
|
|
|
extern resource_size_t pci_sriov_resource_alignment(struct pci_dev *dev,
|
|
|
|
int resno);
|
2009-03-20 11:25:12 +08:00
|
|
|
extern void pci_restore_iov_state(struct pci_dev *dev);
|
2009-03-20 11:25:13 +08:00
|
|
|
extern int pci_iov_bus_range(struct pci_bus *bus);
|
2009-05-18 13:51:32 +08:00
|
|
|
|
2009-03-20 11:25:11 +08:00
|
|
|
#else
|
|
|
|
static inline int pci_iov_init(struct pci_dev *dev)
|
|
|
|
{
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
static inline void pci_iov_release(struct pci_dev *dev)
|
|
|
|
|
|
|
|
{
|
|
|
|
}
|
|
|
|
static inline int pci_iov_resource_bar(struct pci_dev *dev, int resno,
|
|
|
|
enum pci_bar_type *type)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2009-03-20 11:25:12 +08:00
|
|
|
static inline void pci_restore_iov_state(struct pci_dev *dev)
|
|
|
|
{
|
|
|
|
}
|
2009-03-20 11:25:13 +08:00
|
|
|
static inline int pci_iov_bus_range(struct pci_bus *bus)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2009-05-18 13:51:32 +08:00
|
|
|
|
2009-03-20 11:25:11 +08:00
|
|
|
#endif /* CONFIG_PCI_IOV */
|
|
|
|
|
2010-09-08 08:25:20 +08:00
|
|
|
static inline resource_size_t pci_resource_alignment(struct pci_dev *dev,
|
2009-08-29 04:00:06 +08:00
|
|
|
struct resource *res)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_PCI_IOV
|
|
|
|
int resno = res - dev->resource;
|
|
|
|
|
|
|
|
if (resno >= PCI_IOV_RESOURCES && resno <= PCI_IOV_RESOURCE_END)
|
|
|
|
return pci_sriov_resource_alignment(dev, resno);
|
|
|
|
#endif
|
|
|
|
return resource_alignment(res);
|
|
|
|
}
|
|
|
|
|
2009-10-08 01:27:17 +08:00
|
|
|
extern void pci_enable_acs(struct pci_dev *dev);
|
|
|
|
|
2009-12-07 13:03:21 +08:00
|
|
|
struct pci_dev_reset_methods {
|
|
|
|
u16 vendor;
|
|
|
|
u16 device;
|
|
|
|
int (*reset)(struct pci_dev *dev, int probe);
|
|
|
|
};
|
|
|
|
|
2010-01-03 05:57:24 +08:00
|
|
|
#ifdef CONFIG_PCI_QUIRKS
|
2010-01-01 02:06:35 +08:00
|
|
|
extern int pci_dev_specific_reset(struct pci_dev *dev, int probe);
|
2010-01-03 05:57:24 +08:00
|
|
|
#else
|
|
|
|
static inline int pci_dev_specific_reset(struct pci_dev *dev, int probe)
|
|
|
|
{
|
|
|
|
return -ENOTTY;
|
|
|
|
}
|
|
|
|
#endif
|
2009-12-07 13:03:21 +08:00
|
|
|
|
2008-10-13 19:18:07 +08:00
|
|
|
#endif /* DRIVERS_PCI_H */
|