Open main menu
Home
Random
Recent changes
Special pages
Community portal
Preferences
About Wikipedia
Disclaimers
Incubator escapee wiki
Search
User menu
Talk
Dark mode
Contributions
Create account
Log in
Editing
PCI configuration space
Warning:
You are not logged in. Your IP address will be publicly visible if you make any edits. If you
log in
or
create an account
, your edits will be attributed to your username, along with other benefits.
Anti-spam check. Do
not
fill this in!
{{Short description|Auto-configuration mechanism used by PCI}} {{technical|date=February 2025}} {{citations needed|date=February 2025}} '''PCI configuration space''' is the underlying way that the [[Conventional PCI]], [[PCI-X]] and [[PCI Express]] perform auto configuration of the cards inserted into their bus. == Overview == PCI devices have a set of registers referred to as ''configuration space'' and PCI Express introduces ''extended configuration space'' for devices. Configuration space registers are mapped to memory locations. Device drivers and diagnostic software must have access to the configuration space, and operating systems typically use [[API|APIs]] to allow access to device configuration space. When the operating system does not have access methods defined or APIs for memory mapped configuration space requests, the driver or diagnostic software has the burden to access the configuration space in a manner that is compatible with the operating system's underlying access rules. In all systems, device drivers are encouraged to use APIs provided by the operating system to access the configuration space of the device. == Technical information <span class="anchor" id="DEVICE-FUNCTIONS"></span><span class="anchor" id="BDF"></span>== {{See also|Memory-mapped I/O and port-mapped I/O}} One of the major improvements the [[PCI Local Bus]] had over other I/O architectures was its configuration mechanism. In addition to the normal memory-mapped and I/O port spaces, each device function on the bus has a ''configuration space'', which is 256 bytes long, addressable by knowing the eight-bit PCI [[computer bus|bus]], five-bit device, and three-bit function numbers for the device (commonly referred to as the ''BDF'' or ''B/D/F'', as abbreviated from ''bus/device/function''). This allows up to 256 buses, each with up to 32 devices, each supporting eight functions. A single PCI expansion card can respond as a device and must implement at least function number zero. The first 64 bytes of configuration space are standardized; the remainder are available for vendor-defined purposes. Some high-end computers support more than one PCI domain (or PCI segment); each PCI segment supports up to 256 buses. A PCI segment may be referred to a ''PCI root bridge''. In order to allow more parts of configuration space to be standardized without conflicting with existing uses, there can be a list of ''capabilities'' defined within the remaining 192 bytes of PCI configuration space. Each capability has one byte that describes which capability it is, and one byte to point to the next capability. The number of additional bytes depends on the capability ID. If capabilities are being used, a bit in the ''Status'' register is set, and a pointer to the first in a linked list of capabilities is provided in the ''Cap. pointer'' register defined in the Standardized Registers. [[PCI-X]] 2.0 and [[PCI Express]] introduced an extended configuration space, up to 4096 bytes. The only standardized part of extended configuration space is the first four bytes at {{mono|0x100}} which are the start of an extended capability list. Extended capabilities are very much like normal capabilities except that they can refer to any byte in the extended configuration space (by using 12 bits instead of eight), have a four-bit version number and a 16-bit capability ID. Extended capability IDs overlap with normal capability IDs, but there is no chance of confusion as they are in separate lists. == Standardized registers == [[Image:Pci-config-space.svg|thumb|300px|right|Standard registers of PCI Type 0 (Non-Bridge) Configuration Space Header]] The ''Device ID (DID)'' and ''Vendor ID (VID)'' registers identify the device (such as an [[integrated circuit|IC]]), and are commonly called the ''PCI ID''. The 16-bit vendor ID is allocated by the [[PCI-SIG]]. The 16-bit device ID is then assigned by the vendor. There is an inactive project to collect all known Vendor and Device IDs. (See the [[#External links|external links]] below.) The ''Status'' register is used to report which features are supported and whether certain kinds of errors have occurred. The ''Command'' register contains a bitmask of features that can be individually enabled and disabled. The ''Header Type'' register values determine the different layouts of remaining 48 bytes (64-16) of the header, depending on the function of the device. That is, Type 1 headers for Root Complex, switches, and bridges. Then Type 0 for endpoints. The ''Cache Line Size'' register must be programmed before the device is told it can use the memory-write-and-invalidate transaction. This should normally match the [[CPU cache|CPU's cache]] line size, but the correct setting is system dependent. This register does not apply to PCI Express. The ''Subsystem ID (SSID)'' and the ''Subsystem Vendor ID (SVID)'' differentiate specific model (such as an add-in card). While the Vendor ID is that of the [[chipset]] manufacturer, the Subsystem Vendor ID is that of the card manufacturer. The Subsystem ID is assigned by the subsystem vendor, the Device ID is assigned by the chipset manufacturer. As an example, in the case of [[Wireless network interface controller|wireless network cards]], the chip manufacturer might be [[Intel Corp|Intel]] or [[Broadcom]] or [[Atheros]], and the card manufacturer might be [[Netgear]] or [[Hewlett-Packard]]. Generally, the Vendor ID–Device ID combination designates which [[Device driver|driver]] the host should load in order to handle the device, as all cards with the same VID:DID combination can be handled by the same driver. The Subsystem Vendor ID–Subsystem ID combination identifies the card, which is the kind of information the driver may use to apply a minor card-specific change in its operation. == Bus enumeration <span class="anchor" id="BUS-ENUM"></span><span class="anchor" id="IDSEL"></span>== To address a PCI device, it must be enabled by being mapped into the system's I/O port address space or [[Memory-mapped I/O and port-mapped I/O|memory-mapped address space]]. The system's firmware (e.g. [[BIOS]]) or the [[operating system]] program the ''Base Address Registers'' (commonly called BARs) to inform the device of its [[system resource|resource]]s configuration by writing configuration commands to the PCI controller. Because all PCI devices are in an ''inactive'' state upon system reset, they will have no addresses assigned to them by which the operating system or [[device driver]]s can communicate with them. Either the [[BIOS]] or the operating system geographically addresses the PCI devices (for example, the first PCI slot, the second PCI slot, the third PCI slot, or the integrated PCI devices, etc., on the [[motherboard]]) through the PCI controller using the per slot or per device IDSEL (Initialization Device Select) signals. {| class="wikitable floatright" style="margin-left: 1.5em; margin-right: 0;" |+ PCI BAR Bits |- ! Bits !! Description !! Values |- | colspan="3" style="text-align: center;" | ''For all PCI BARs'' |- style="vertical-align: top" | style="text-align: right; padding: 0em 0.5em;" | 0 || style="padding: 0em 0.5em;" | Region Type || style="padding: 0em 0.5em;" | 0 = Memory<br>1 = I/O |- | colspan="3" style="text-align: center;" | ''For Memory BARs'' |- style="vertical-align: top" | style="text-align: right; padding: 0em 0.5em;" | 2-1 || style="padding: 0em 0.5em;" | Locatable || style="padding: 0em 0.5em;" | 0 = any 32-bit<br>1 = < 1 [[megabyte|MB]]<br>2 = any 64-bit |- style="vertical-align: top" | style="text-align: right; padding: 0em 0.5em;" | 3 || style="padding: 0em 0.5em;" | Prefetchable || style="padding: 0em 0.5em;" | 0 = no<br>1 = yes |- style="vertical-align: top" | style="text-align: right; padding: 0em 0.5em;" | 31-4 || style="padding: 0em 0.5em;" | Base Address || style="padding: 0em 0.5em;" | naturally 16-byte aligned |- | colspan="3" style="text-align: center;" | ''For I/O BARs'' |- style="vertical-align: top" | style="text-align: right; padding: 0em 0.5em;" | 1 || style="padding: 0em 0.5em;" | Reserved || style="padding: 0em 0.5em;" | |- style="vertical-align: top" | style="text-align: right; padding: 0em 0.5em;" | 31-2 || style="padding: 0em 0.5em;" | Base Address || style="padding: 0em 0.5em;" | naturally 4-byte aligned |} When the computer is powered on, the PCI bus(es) and device(s) must be ''enumerated'' by BIOS or operating system. Bus enumeration is performed by attempting to access the PCI configuration space registers for each buses, devices and functions. Note that device number, different from VID and DID, is merely a device's sequential number on that bus. Moreover, after a new bridge is detected, a new bus number is defined, and device enumeration restarts at device number zero. If no response is received from the device's function #0, the [[Bus mastering|bus master]] performs an abort and returns an all-bits-on value ({{mono|FFFFFFFF}} in hexadecimal), which is an invalid VID/DID value, thus the BIOS or operating system can tell that the specified combination bus/device_number/function (B/D/F) is not present. So, when a read to a function ID of zero for a given bus/device causes the master (initiator) to abort, it must then be presumed that no working device exists on that bus because devices are required to implement function number zero. In this case, reads to the remaining functions numbers (1–7) are not necessary as they also will not exist. When a read to a specified B/D/F combination for the vendor ID register succeeds, the system firmware or operating system knows that it exists; it writes all ones to its BARs and reads back the device's requested memory size in an encoded form. The design implies that all address space sizes are a power of two and are naturally aligned.<ref name="pci-bars" /> At this point, the BIOS or operating system will program the memory-mapped addresses and I/O port addresses into the device's BAR configuration registers. These addresses stay valid as long as the system remains turned on. Upon power-off, these settings are lost and the procedure is repeated next time the system is powered back on. The BIOS or operating system will also program some other registers of the PCI configuration space for each PCI device, e.g. [[interrupt request]]. Since this entire process is fully automated, the user is spared the task of configuring any newly added hardware manually by changing [[DIP switch]]es on the cards themselves. This automatic device discovery and address space assignment is how [[plug and play]] is implemented. If a PCI-to-PCI bridge is found, the system must assign the secondary PCI bus beyond the bridge a bus number other than zero, and then enumerate the devices on that secondary bus. If more PCI bridges are found, the discovery continues recursively until all possible domain/bus/device combinations are scanned. Each non-bridge PCI device function can implement up to 6 BARs, each of which can respond to different addresses in I/O port and memory-mapped address space. Each BAR describes a region <ref>{{Cite web | url = http://www.read.seas.harvard.edu/~kohler/class/04f-aos/ref/hardware/vgadoc/PCI.TXT | title = PCI configuration methods | date = 2011-11-22 | accessdate = 2021-09-27 | publisher = read.seas.harvard.edu }}</ref><ref name="pci-bars">{{Cite web | url = http://wiki.osdev.org/PCI#Base_Address_Registers | title = Base Address Registers | work = PCI | date = 2013-12-24 | accessdate = 2014-04-17 | publisher = osdev.org }}</ref> that is between 16 bytes and 2 gigabytes in size, located below the 4 gigabyte address space limit. If a platform supports the "Above 4G" option in system firmware, 64 bit BARs can be used. '''Resizable BAR''' (also known as ''Re-Size BAR'', [[AMD]] ''Smart Access Memory'' (SAM),<ref name="rps-rebar">{{cite news |first=James |last=Archer |date=2021-12-07 |title=What is Resizable BAR, and should you use it? |url=https://www.rockpapershotgun.com/what-is-resizable-bar-and-should-you-use-it |work=[[Rock Paper Shotgun]] |access-date=2024-03-26}}</ref> or [[ASRock]] ''Clever Access Memory'' (CAM))<ref>{{cite web |url=https://www.techpowerup.com/275578/asrock-implements-cam-clever-access-memory-on-intel-z490-taichi-motherboard |title=ASRock Implements CAM (Clever Access Memory) on Intel Z490 Taichi Motherboard |author=Raevenlord |date=2020-12-04 |website=TechPowerUp |access-date=2024-03-26}}</ref> is a capability which a PCIe device can use to negotiate a larger BAR size.<ref name="intel-rebar">{{cite web |url=https://www.intel.com/content/www/us/en/support/articles/000090831/graphics.html |title=What Is Resizable BAR and How Do I Enable It? |date=2023-04-18 |website=intel.com |publisher=Intel Corporation |access-date=2024-03-26 }}</ref> Classically, BARs were limited to a size of 256MB, but modern [[graphics card]]s have [[framebuffer]]s much larger than that.<ref name="rps-rebar"/> This mismatch led to inefficiencies when the CPU accessed the framebuffer.<ref name="rps-rebar"/> Resizable BAR lets a CPU access the whole framebuffer at once, thus improving performance.<ref name="rps-rebar"/> A PCI device may also have an ''[[option ROM]]''. == Hardware implementation == When performing a ''Configuration Space'' access, a PCI device does not decode the address to determine if it should respond, but instead looks at the ''Initialization Device Select'' signal (IDSEL). There is a system-wide unique activation method for each IDSEL signal. The PCI device is required to decode only the lowest order 11 bits of the address space (AD[10] to AD[0]) address/data signals, and can ignore decoding the 21 high order A/D signals (AD[31] to AD[11]) because a ''Configuration Space'' access implementation has each slot's IDSEL pin connected to a different high order address/data line AD[11] through AD[31]. The IDSEL signal is a different pin for each PCI device/adapter/slot. To configure the card in slot ''n'', the PCI bus bridge performs a configuration-space access cycle with the PCI device's register to be addressed on lines AD[7:2] (AD[1:0] are always zero since registers are double words (32-bits)), and the PCI function number specified on bits AD[10:8], with all higher-order bits zeros except for AD[''n''+11] being used as the IDSEL signal on a given slot/device. To reduce electrically loading down the timing critical (and thus electrically loading sensitive) AD[] bus, the IDSEL signal on the PCI slot connector is usually connected to its assigned AD[''n''+11] pin through a resistor. This causes the PCI's IDSEL signal to reach its active condition more slowly than other PCI bus signals (due to the RC time constant of both the resistor and the IDSEL pin's input capacitance). Thus ''Configuration Space'' accesses are performed more slowly to allow time for the IDSEL signal to reach a valid level. The scanning on the bus is performed on the Intel platform by accessing two defined standardized ports. These ports are the ''Configuration Space Address'' ({{mono|0xCF8}}) I/O port and ''Configuration Space Data'' ({{mono|0xCFC}}) I/O port. The value written to the ''Configuration Space Address'' I/O port is created by combining B/D/F values and the registers address value into a 32-bit word. ==Software implementation== Configuration reads and writes can be initiated from the CPU in two ways: one legacy method via I/O addresses {{mono|0xCF8}} and {{mono|0xCFC}}, and another called memory-mapped configuration.<ref>{{cite web |title=Accessing PCI Express* Configuration Registers Using Intel Chipsets |url=http://www.csit-sun.pub.ro/~cpop/Documentatie_SMP/Intel_Microprocessor_Systems/Intel_ProcessorNew/Intel%20White%20Paper/Accessing%20PCI%20Express%20Configuration%20Registers%20Using%20Intel%20Chipsets.pdf |publisher=Intel Corporation |accessdate=27 September 2018}}</ref> The legacy method was present in the original PCI, and it is called Configuration Access Mechanism (CAM). It allows for 256 bytes of a device's address space to be reached indirectly via two 32-bit registers called PCI CONFIG_ADDRESS and PCI CONFIG_DATA. These registers are at addresses {{mono|0xCF8}} and {{mono|0xCFC}} in the x86 I/O address space.<ref>{{Cite web | url = http://wiki.osdev.org/PCI#Configuration_Mechanism_.231 | title = PCI Configuration Mechanism #1 | date = 2015-01-01 | accessdate = 2015-01-01 | website = osdev.org }}</ref> For example, a software driver (firmware, OS kernel or kernel driver) can use these registers to configure a PCI device by writing the address of the device's register into CONFIG_ADDRESS, and by putting the data that is supposed to be written to the device into CONFIG_DATA. Since this process requires a write to a register in order to write the device's register, it is referred to as "indirection". The format of CONFIG_ADDRESS is the following: <syntaxhighlight lang="c"> 0x80000000 | bus << 16 | device << 11 | function << 8 | offset </syntaxhighlight> As explained previously, addressing a device via Bus, Device, and Function (BDF) is also referred to as "addressing a device geographically." See <code>arch/x86/pci/early.c</code> in the [[Linux kernel]] code for an example of code that uses geographical addressing.<ref>{{cite web | url = https://git.kernel.org/cgit/linux/kernel/git/stable/linux-stable.git/tree/arch/x86/pci/early.c?id=refs/tags/v3.12.7 | title = kernel/git/stable/linux-stable.git: arch/x86/pci/early.c (Linux kernel stable tree, version 3.12.7) | accessdate = 2014-01-10 | publisher = [[kernel.org]] }}</ref> When extended configuration space is used on some AMD CPUs, the extra bits 11:8 of the offset are written to bits 27:24 of the CONFIG_ADDRESS register:<ref>{{cite web | url = https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux-stable.git/tree/arch/x86/pci/direct.c?id=refs/tags/v3.12.7#n16 | title = kernel/git/stable/linux-stable.git: arch/x86/pci/direct.c (Linux kernel stable tree, version 3.12.7) | accessdate = 2017-09-11 | publisher = [[kernel.org]] }}</ref><ref>{{cite web |last1=Richter |first1=Robert |title=x86: add PCI extended config space access for AMD Barcelona |url=https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=831d991821daedd4839073dbca55514432ef1768 |publisher=kernel.org |accessdate=26 September 2018}}</ref> <syntaxhighlight lang="c"> 0x80000000 | (offset & 0xf00) << 16 | bus << 16 | device << 11 | function << 8 | (offset & 0xff) </syntaxhighlight> The second method was created for PCI Express. It is called Enhanced Configuration Access Mechanism (ECAM). It extends device's configuration space to 4 KB, with the bottom 256 bytes overlapping the original (legacy) configuration space in PCI. The section of the addressable space is "stolen" so that the accesses from the CPU don't go to memory but rather reach a given device in the PCI Express fabric. During system initialization, BIOS determines the base address for this "stolen" address region and communicates it to the root complex and to the operating system. Each device has its own 4 KB space and each device's info is accessible through a simple array <code>dev[bus][device][function]</code> so that 256 MB of physical contiguous space is "stolen" for this use (256 buses Γ 32 devices Γ 8 functions Γ 4 KB = 256 MB). The base physical address of this array is not specified. For example, on modern x86 systems the ACPI tables contain the necessary information.<ref>{{cite web | url = http://wiki.osdev.org/XSDT | title = XSDT - OSDev Wiki | accessdate = 2017-04-30 }}</ref> == See also == {{Portal|Electronics}} * [[PC card]] * [[Root complex]] == References == {{Reflist}} == External links == * [https://web.archive.org/web/20080209182956/http://www.pcidatabase.com/index.php PCI Vendor and Device Lists] * [http://pci-ids.ucw.cz/ The PCI ID Repository], a project to collect all known IDs * [https://pdos.csail.mit.edu/6.828/2005/readings/hardware/vgadoc/PCI.TXT Description of IO Port usage for PCI configuration] * [https://elixir.bootlin.com/linux/latest/source/include/uapi/linux/pci_regs.h Linux kernel header file with configuration space register definitions] [[Category:Peripheral Component Interconnect]]
Edit summary
(Briefly describe your changes)
By publishing changes, you agree to the
Terms of Use
, and you irrevocably agree to release your contribution under the
CC BY-SA 4.0 License
and the
GFDL
. You agree that a hyperlink or URL is sufficient attribution under the Creative Commons license.
Cancel
Editing help
(opens in new window)
Pages transcluded onto the current version of this page
(
help
)
:
Template:Ambox
(
edit
)
Template:Citations needed
(
edit
)
Template:Cite news
(
edit
)
Template:Cite web
(
edit
)
Template:Mono
(
edit
)
Template:Portal
(
edit
)
Template:Reflist
(
edit
)
Template:See also
(
edit
)
Template:Short description
(
edit
)
Template:Technical
(
edit
)