2008-10-13 15:18:07 +04:00
# ifndef DRIVERS_PCI_H
# define DRIVERS_PCI_H
# define PCI_CFG_SPACE_SIZE 256
# define PCI_CFG_SPACE_EXP_SIZE 4096
2015-06-30 04:16:41 +03:00
# define PCI_FIND_CAP_TTL 48
2013-07-31 10:53:16 +04:00
extern const unsigned char pcie_link_speed [ ] ;
2014-11-11 23:09:46 +03:00
bool pcie_cap_has_lnkctl ( const struct pci_dev * dev ) ;
2005-04-17 02:20:36 +04:00
/* Functions internal to the PCI core code */
2013-04-12 22:02:59 +04:00
int pci_create_sysfs_dev_files ( struct pci_dev * pdev ) ;
void pci_remove_sysfs_dev_files ( struct pci_dev * pdev ) ;
2011-03-02 20:04:17 +03:00
# if !defined(CONFIG_DMI) && !defined(CONFIG_ACPI)
2010-07-26 14:56:50 +04:00
static inline void pci_create_firmware_label_files ( struct pci_dev * pdev )
2010-08-02 16:44:29 +04:00
{ return ; }
2010-07-26 14:56:50 +04:00
static inline void pci_remove_firmware_label_files ( struct pci_dev * pdev )
2010-08-02 16:44:29 +04:00
{ return ; }
2010-07-26 14:56:50 +04:00
# else
2013-04-12 22:02:59 +04:00
void pci_create_firmware_label_files ( struct pci_dev * pdev ) ;
void pci_remove_firmware_label_files ( struct pci_dev * pdev ) ;
2010-07-26 14:56:50 +04:00
# endif
2013-04-12 22:02:59 +04:00
void pci_cleanup_rom ( struct pci_dev * dev ) ;
2008-10-24 21:32:33 +04:00
# ifdef HAVE_PCI_MMAP
2010-11-10 13:03:21 +03: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> */
} ;
2013-04-12 22:02:59 +04:00
int pci_mmap_fits ( struct pci_dev * pdev , int resno , struct vm_area_struct * vmai ,
enum pci_mmap_api mmap_api ) ;
2008-10-24 21:32:33 +04:00
# endif
2009-07-28 00:37:48 +04:00
int pci_probe_reset_function ( struct pci_dev * dev ) ;
2007-07-17 08:27:10 +04:00
2008-07-07 05:32:02 +04:00
/**
2009-01-10 04:04:26 +03:00
* struct pci_platform_pm_ops - Firmware PM callbacks
2008-07-07 05:32:02 +04:00
*
2009-01-10 04:04:26 +03:00
* @ is_manageable : returns ' true ' if given device is power manageable by the
* platform firmware
2008-07-07 05:32:02 +04:00
*
2009-01-10 04:04:26 +03:00
* @ set_state : invokes the platform firmware to set the device ' s power state
2008-07-07 05:32:02 +04:00
*
2009-01-10 04:04:26 +03: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 05:32:02 +04:00
*
2009-01-10 04:04:26 +03:00
* @ sleep_wake : enables / disables the system wake up capability of given device
2008-07-07 05:34:48 +04:00
*
2010-02-18 01:44:09 +03: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 )
*
2015-01-21 04:17:42 +03:00
* @ need_resume : returns ' true ' if the given device ( which is currently
* suspended ) needs to be resumed to be configured for system
* wakeup .
*
2008-07-07 05:32:02 +04: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 05:34:48 +04:00
int ( * sleep_wake ) ( struct pci_dev * dev , bool enable ) ;
2010-02-18 01:44:09 +03:00
int ( * run_wake ) ( struct pci_dev * dev , bool enable ) ;
2015-01-21 04:17:42 +03:00
bool ( * need_resume ) ( struct pci_dev * dev ) ;
2008-07-07 05:32:02 +04:00
} ;
2015-12-06 19:33:45 +03:00
int pci_set_platform_pm ( const struct pci_platform_pm_ops * ops ) ;
2013-04-12 22:02:59 +04:00
void pci_update_current_state ( struct pci_dev * dev , pci_power_t state ) ;
void pci_power_up ( struct pci_dev * dev ) ;
void pci_disable_enabled_device ( struct pci_dev * dev ) ;
int pci_finish_runtime_suspend ( struct pci_dev * dev ) ;
int __pci_pme_wakeup ( struct pci_dev * dev , void * ign ) ;
2015-01-21 04:17:42 +03:00
bool pci_dev_keep_suspended ( struct pci_dev * dev ) ;
2015-09-30 02:10:24 +03:00
void pci_dev_complete_resume ( struct pci_dev * pci_dev ) ;
2013-04-12 22:02:59 +04:00
void pci_config_pm_runtime_get ( struct pci_dev * dev ) ;
void pci_config_pm_runtime_put ( struct pci_dev * dev ) ;
void pci_pm_init ( struct pci_dev * dev ) ;
2015-10-30 01:35:39 +03:00
void pci_ea_init ( struct pci_dev * dev ) ;
2013-04-12 22:02:59 +04:00
void pci_allocate_cap_save_buffers ( struct pci_dev * dev ) ;
2012-02-11 12:18:30 +04:00
void pci_free_cap_save_buffers ( struct pci_dev * dev ) ;
2009-01-16 23:54:43 +03:00
2010-12-29 15:21:23 +03: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 ) ;
}
2014-05-04 08:23:36 +04:00
static inline bool pci_has_subordinate ( struct pci_dev * pci_dev )
2009-01-16 23:54:43 +03:00
{
return ! ! ( pci_dev - > subordinate ) ;
}
2005-03-19 08:15:48 +03:00
2008-03-05 19:52:39 +03:00
struct pci_vpd_ops {
2008-12-18 20:17:16 +03: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-05 19:52:39 +03:00
} ;
struct pci_vpd {
2008-12-18 20:17:16 +03:00
const struct pci_vpd_ops * ops ;
2008-03-05 19:52:39 +03:00
struct bin_attribute * attr ; /* descriptor for sysfs VPD entry */
2016-02-22 23:09:52 +03:00
struct mutex lock ;
unsigned int len ;
u16 flag ;
u8 cap ;
u8 busy : 1 ;
u8 valid : 1 ;
2008-03-05 19:52:39 +03:00
} ;
2016-02-22 22:58:37 +03:00
int pci_vpd_init ( struct pci_dev * dev ) ;
2016-02-22 22:58:06 +03:00
void pci_vpd_release ( struct pci_dev * dev ) ;
2008-03-05 19:52:39 +03:00
2005-04-17 02:20:36 +04:00
/* PCI /proc functions */
# ifdef CONFIG_PROC_FS
2013-04-12 22:02:59 +04:00
int pci_proc_attach_device ( struct pci_dev * dev ) ;
int pci_proc_detach_device ( struct pci_dev * dev ) ;
int pci_proc_detach_bus ( struct pci_bus * bus ) ;
2005-04-17 02:20:36 +04:00
# 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 */
2012-05-18 23:46:34 +04:00
int pci_hp_add_bridge ( struct pci_dev * dev ) ;
2005-04-17 02:20:36 +04:00
2008-10-03 13:49:32 +04:00
# ifdef HAVE_PCI_LEGACY
2013-04-12 22:02:59 +04:00
void pci_create_legacy_files ( struct pci_bus * bus ) ;
void pci_remove_legacy_files ( struct pci_bus * bus ) ;
2008-10-03 13:49:32 +04: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 02:20:36 +04:00
/* Lock for read/write access to pci device and bus lists */
2006-06-02 08:35:43 +04:00
extern struct rw_semaphore pci_bus_sem ;
2005-04-17 02:20:36 +04:00
2011-11-04 12:46:00 +04:00
extern raw_spinlock_t pci_lock ;
2006-07-12 19:59:00 +04:00
extern unsigned int pci_pm_d3_delay ;
2007-01-25 11:34:07 +03:00
2005-08-17 02:16:05 +04:00
# ifdef CONFIG_PCI_MSI
2006-03-06 08:33:34 +03:00
void pci_no_msi ( void ) ;
2005-08-17 02:16:05 +04:00
# else
2006-03-06 08:33:34 +03:00
static inline void pci_no_msi ( void ) { }
2005-08-17 02:16:05 +04:00
# endif
2007-01-25 11:34:08 +03:00
2015-05-07 17:52:21 +03:00
static inline void pci_msi_set_enable ( struct pci_dev * dev , int enable )
{
u16 control ;
pci_read_config_word ( dev , dev - > msi_cap + PCI_MSI_FLAGS , & control ) ;
control & = ~ PCI_MSI_FLAGS_ENABLE ;
if ( enable )
control | = PCI_MSI_FLAGS_ENABLE ;
pci_write_config_word ( dev , dev - > msi_cap + PCI_MSI_FLAGS , control ) ;
}
static inline void pci_msix_clear_and_set_ctrl ( struct pci_dev * dev , u16 clear , u16 set )
{
u16 ctrl ;
pci_read_config_word ( dev , dev - > msix_cap + PCI_MSIX_FLAGS , & ctrl ) ;
ctrl & = ~ clear ;
ctrl | = set ;
pci_write_config_word ( dev , dev - > msix_cap + PCI_MSIX_FLAGS , ctrl ) ;
}
2012-02-24 07:23:30 +04:00
void pci_realloc_get_opt ( char * ) ;
2011-07-07 22:19:10 +04:00
2006-07-12 19:59:00 +04:00
static inline int pci_no_d1d2 ( struct pci_dev * dev )
{
unsigned int parent_dstates = 0 ;
2005-08-17 02:16:05 +04:00
2006-07-12 19:59:00 +04:00
if ( dev - > bus - > self )
parent_dstates = dev - > bus - > self - > no_d1d2 ;
return ( dev - > no_d1d2 | | parent_dstates ) ;
}
2013-10-07 10:55:40 +04:00
extern const struct attribute_group * pci_dev_groups [ ] ;
2013-07-25 02:05:17 +04:00
extern const struct attribute_group * pcibus_groups [ ] ;
2012-11-06 00:20:34 +04:00
extern struct device_type pci_dev_type ;
2013-10-08 00:51:02 +04:00
extern const struct attribute_group * pci_bus_groups [ ] ;
2009-03-20 23:56:31 +03:00
2005-04-17 02:20:36 +04: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 02:21:33 +03:00
*
2005-04-17 02:20:36 +04: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 ;
}
2008-06-11 01:28:50 +04: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-21 21:40:40 +03: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 */
} ;
2012-01-27 22:55:10 +04:00
bool pci_bus_read_dev_vendor_id ( struct pci_bus * bus , int devfn , u32 * pl ,
int crs_timeout ) ;
2013-04-12 22:02:59 +04:00
int pci_setup_device ( struct pci_dev * dev ) ;
int __pci_read_base ( struct pci_dev * dev , enum pci_bar_type type ,
struct resource * res , unsigned int reg ) ;
int pci_resource_bar ( struct pci_dev * dev , int resno , enum pci_bar_type * type ) ;
void pci_configure_ari ( struct pci_dev * dev ) ;
2014-04-15 02:11:40 +04:00
void __pci_bus_size_bridges ( struct pci_bus * bus ,
PCI / ACPI: Use boot-time resource allocation rules during hotplug
On x86 platforms, the kernel respects PCI resource assignments from
the BIOS and only reassigns resources for unassigned BARs at boot
time. However, with the ACPI-based hotplug (acpiphp), it ignores the
BIOS' PCI resource assignments completely and reassigns all resources
by itself. This causes differences in PCI resource allocation
between boot time and runtime hotplug to occur, which is generally
undesirable and sometimes actively breaks things.
Namely, if there are enough resources, reassigning all PCI resources
during runtime hotplug should work, but it may fail if the resources
are constrained. This may happen, for instance, when some PCI
devices with huge MMIO BARs are involved in the runtime hotplug
operations, because the current PCI MMIO alignment algorithm may
waste huge chunks of MMIO address space in those cases.
On the Alexander's Sony VAIO VPCZ23A4R the BIOS allocates limited
MMIO resources for the dock station which contains a device
(graphics adapter) with a 256MB MMIO BAR. An attempt to reassign
that during runtime hotplug causes the dock station MMIO window to be
exhausted and acpiphp fails to allocate resources for the majority
of devices on the dock station as a result.
To prevent that from happening, modify acpiphp to follow the boot
time resources allocation behavior so that the BIOS' resource
assignments are respected during runtime hotplug too.
[rjw: Changelog]
References: https://bugzilla.kernel.org/show_bug.cgi?id=56531
Reported-and-tested-by: Alexander E. Patrakov <patrakov@gmail.com>
Tested-by: Illya Klymov <xanf@xanf.me>
Signed-off-by: Jiang Liu <jiang.liu@huawei.com>
Acked-by: Yinghai Lu <yinghai@kernel.org>
Cc: 3.9+ <stable@vger.kernel.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
2013-06-23 03:01:35 +04:00
struct list_head * realloc_head ) ;
2014-04-15 02:11:40 +04:00
void __pci_bus_assign_resources ( const struct pci_bus * bus ,
struct list_head * realloc_head ,
struct list_head * fail_head ) ;
2015-01-16 01:21:49 +03:00
bool pci_bus_clip_resource ( struct pci_dev * dev , int idx ) ;
2013-01-27 04:35:58 +04:00
2012-02-16 09:40:31 +04:00
void pci_reassigndev_resource_alignment ( struct pci_dev * dev ) ;
2013-04-12 22:02:59 +04:00
void pci_disable_bridge_window ( struct pci_dev * dev ) ;
2009-03-16 11:13:39 +03:00
2009-03-20 06:25:11 +03: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 */
2012-11-10 07:27:53 +04:00
u16 total_VFs ; /* total VFs associated with the PF */
u16 initial_VFs ; /* initial VFs associated with the PF */
u16 num_VFs ; /* number of VFs available */
2009-03-20 06:25:11 +03: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 */
2015-03-25 11:23:47 +03:00
u8 max_VF_buses ; /* max buses consumed by VFs */
2012-11-10 07:27:53 +04:00
u16 driver_max_VFs ; /* max num VFs driver supports */
2009-03-20 06:25:11 +03:00
struct pci_dev * dev ; /* lowest numbered PF */
struct pci_dev * self ; /* this PF */
struct mutex lock ; /* lock for VF bus */
2015-03-25 11:23:44 +03:00
resource_size_t barsz [ PCI_SRIOV_NUM_BARS ] ; /* VF BAR size */
2009-03-20 06:25:11 +03:00
} ;
2011-12-17 17:24:40 +04:00
# ifdef CONFIG_PCI_ATS
2013-04-12 22:02:59 +04:00
void pci_restore_ats_state ( struct pci_dev * dev ) ;
2011-12-17 17:24:40 +04:00
# else
static inline void pci_restore_ats_state ( struct pci_dev * dev )
{
}
# endif /* CONFIG_PCI_ATS */
2009-03-20 06:25:11 +03:00
# ifdef CONFIG_PCI_IOV
2013-04-12 22:02:59 +04:00
int pci_iov_init ( struct pci_dev * dev ) ;
void pci_iov_release ( struct pci_dev * dev ) ;
2014-11-11 18:04:50 +03:00
int pci_iov_resource_bar ( struct pci_dev * dev , int resno ) ;
2013-04-12 22:02:59 +04:00
resource_size_t pci_sriov_resource_alignment ( struct pci_dev * dev , int resno ) ;
void pci_restore_iov_state ( struct pci_dev * dev ) ;
int pci_iov_bus_range ( struct pci_bus * bus ) ;
2009-05-18 09:51:32 +04:00
2009-03-20 06:25:11 +03:00
# else
static inline int pci_iov_init ( struct pci_dev * dev )
{
return - ENODEV ;
}
static inline void pci_iov_release ( struct pci_dev * dev )
{
}
2014-11-11 18:04:50 +03:00
static inline int pci_iov_resource_bar ( struct pci_dev * dev , int resno )
2009-03-20 06:25:11 +03:00
{
return 0 ;
}
2009-03-20 06:25:12 +03:00
static inline void pci_restore_iov_state ( struct pci_dev * dev )
{
}
2009-03-20 06:25:13 +03:00
static inline int pci_iov_bus_range ( struct pci_bus * bus )
{
return 0 ;
}
2009-05-18 09:51:32 +04:00
2009-03-20 06:25:11 +03:00
# endif /* CONFIG_PCI_IOV */
2013-04-12 22:02:59 +04:00
unsigned long pci_cardbus_resource_alignment ( struct resource * ) ;
2011-07-26 00:08:41 +04:00
2010-09-08 04:25:20 +04:00
static inline resource_size_t pci_resource_alignment ( struct pci_dev * dev ,
2013-04-12 22:02:59 +04:00
struct resource * res )
2009-08-29 00:00:06 +04:00
{
# 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
2011-07-26 00:08:41 +04:00
if ( dev - > class > > 8 = = PCI_CLASS_BRIDGE_CARDBUS )
return pci_cardbus_resource_alignment ( res ) ;
2009-08-29 00:00:06 +04:00
return resource_alignment ( res ) ;
}
2013-04-12 22:02:59 +04:00
void pci_enable_acs ( struct pci_dev * dev ) ;
2009-10-07 21:27:17 +04:00
2009-12-07 08:03:21 +03:00
struct pci_dev_reset_methods {
u16 vendor ;
u16 device ;
int ( * reset ) ( struct pci_dev * dev , int probe ) ;
} ;
2010-01-03 00:57:24 +03:00
# ifdef CONFIG_PCI_QUIRKS
2013-04-12 22:02:59 +04:00
int pci_dev_specific_reset ( struct pci_dev * dev , int probe ) ;
2010-01-03 00:57:24 +03:00
# else
static inline int pci_dev_specific_reset ( struct pci_dev * dev , int probe )
{
return - ENOTTY ;
}
# endif
2009-12-07 08:03:21 +03:00
2008-10-13 15:18:07 +04:00
# endif /* DRIVERS_PCI_H */