# PCI & PCIe Enumeration (Bus Scan) Process by

Jitesh Verma



#### PCI & PCIe Enumeration/Scan Process

```
Uint32 vendor id device id:
Uint8 config_header_type;
For (bus=0; bus < 256; bus++)
  For (device=0; device <32; device++)
         For (pci_function=0; pci_function < 8; pci_function++)
                     vendor_id_device_id = read_vendor_id_device_id(bus, device, pci_function);
                      if (vendor id device id == 0xFFFFFFF)
                            break:
                      config_header_type = Read_config_header_type();
                      if (config_header_type == 0) // PCI/PCIe Endpoint device
                            Allocate_system_resources_to_the_pci_function();
                                                                               // Allocate memory/IO region & Intr
                            Program_the_BARs_of_the_pci_function();
                      else if (config_header_type == 1) // PCI/PCIe Root-complex, switch or bridge device
                            Program_the_RC_or_Switch_or_Bridge_and_mark_a_new_bus();
                            Scan the newly detected bus();
```



#### PCI & PCIe Configuration – Base Address Register (BAR)

| Bits | Description      | Values                                                |
|------|------------------|-------------------------------------------------------|
|      | For all PCI BARs |                                                       |
| 0    | Region Type      | 0 = Memory<br>1 = I/O                                 |
|      | For Memory BARs  |                                                       |
| 2-1  | Locatable        | 0 = any 32-bit<br>1 = < 1 <u>MB</u><br>2 = any 64-bit |
| 3    | Prefetchable     | 0 = no<br>1 = yes                                     |
| 31-4 | Base Address     | naturally 16-byte aligned                             |
|      | For I/O BARs     |                                                       |
| 1    | Reserved         |                                                       |
| 31-2 | Base Address     | naturally 4-byte aligned                              |



#### PCI & PCIe Configuration – Device & Register Addressing

| Bit 31     | Bits 30-24 | Bits 23-16 | Bits 15-11       | Bits 10-8          | Bits 7-0           |
|------------|------------|------------|------------------|--------------------|--------------------|
| Enable Bit | Reserved   | Bus Number | Device<br>Number | Function<br>Number | Register<br>Offset |



### PCI & PCIe Configuration Space – Register Offsets

Reg

|         |             |                     | Tation Opac | 9.0                 |                 |  |
|---------|-------------|---------------------|-------------|---------------------|-----------------|--|
| egister | Byte Offset | Bits 31-24          | Bits 23-16  | Bits 15-8           | Bits 7-0        |  |
| 0x0     | 0x0         | Device Id           |             | Vendor Id           |                 |  |
| 0x1     | 0x4         | Sta                 | Status      |                     | mand            |  |
| 0x2     | 0x8         | Class Code          | Sub-class   | ProgIF              | Revision Id     |  |
| 0x3     | 0xC         | BIST                | Header Type | Latency Timer       | Cache Line Size |  |
| 0x4     | 0x10        | BAR0                |             |                     |                 |  |
| 0x5     | 0x14        | BAR1                |             |                     |                 |  |
| 0x6     | 0x18        | BAR2                |             |                     |                 |  |
| 0x7     | 0x1C        | BAR3                |             |                     |                 |  |
| 0x8     | 0x20        | BAR4                |             |                     |                 |  |
| 0x9     | 0x24        | BAR5                |             |                     |                 |  |
| 0xA     | 0x28        | CardBus CIS Pointer |             |                     |                 |  |
| 0xB     | 0x2C        | Subsystem Device Id |             | Subsystem Vendor Id |                 |  |
| 0xC     | 0x30        | Expansion ROM BAR   |             |                     |                 |  |
| 0xD     | 0x34        |                     |             | Capability Ptr      |                 |  |
| 0xE     | 0x38        | Reserved            |             |                     |                 |  |
| 0xF     | 0x3C        | Max Latency         | Min Grant   | Interrupt Pin       | Interrupt Line  |  |

## PCI & PCIe Configuration - Resource Determination & Allocation

```
u32 bar0;
u32 base address;
U32 initial_address = 0xABCDEF0X;
                                         // Chosen by complex resource distribution algorithm
                                         // 4 LSBits are read-only resource (Mem/IO) info bits
u32 info_bits_mask = 0xFFFFFF0;
write bar(BAR0, 0xFFFFFFF);
                                        // Write all ones
bar0 = read_bar(BAR0);
                                         // Read updated value (e.g. 0xFFFFF00X)
bar0 = bar0 & info bits mask;
                                         // Zero out the 4 info bits (0xFFFFF000)
                                         // Invert the bits (0x00000FFF)
bar0 = ~bar0:
                                   // Add 1 to get the resource (mem/IO) length (0x00001000 = 4KB)
resource length = bar0 + 1;
                                        // Base address of the resource (0xABCDE000)
base address = initial address | ~bar0;
write_bar(BAR0, base_address);
                                         // Write the base address of the address range into BAR
```

Note: For 64-bit PCI/PCIe devices / system, pair of two BARs (e.g. BAR0 & BAR1) are combined to form 64-bit BARs.



#### **PCI & PCIe Configuration References**

- https://en.wikipedia.org/wiki/PCI\_configuration\_space
- <a href="https://stackoverflow.com/questions/19006632/how-is-a-pci-pcie-bar-size-determined">https://stackoverflow.com/questions/19006632/how-is-a-pci-pcie-bar-size-determined</a>
- https://pcisig.com/

