Home | History | Annotate | Line # | Download | only in hardware
hwpci.c revision 1.1.1.3.2.4
      1          1.1    jruoho /*******************************************************************************
      2          1.1    jruoho  *
      3          1.1    jruoho  * Module Name: hwpci - Obtain PCI bus, device, and function numbers
      4          1.1    jruoho  *
      5          1.1    jruoho  ******************************************************************************/
      6          1.1    jruoho 
      7          1.1    jruoho /*
      8  1.1.1.3.2.4     skrll  * Copyright (C) 2000 - 2017, Intel Corp.
      9          1.1    jruoho  * All rights reserved.
     10          1.1    jruoho  *
     11          1.1    jruoho  * Redistribution and use in source and binary forms, with or without
     12          1.1    jruoho  * modification, are permitted provided that the following conditions
     13          1.1    jruoho  * are met:
     14          1.1    jruoho  * 1. Redistributions of source code must retain the above copyright
     15          1.1    jruoho  *    notice, this list of conditions, and the following disclaimer,
     16          1.1    jruoho  *    without modification.
     17          1.1    jruoho  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
     18          1.1    jruoho  *    substantially similar to the "NO WARRANTY" disclaimer below
     19          1.1    jruoho  *    ("Disclaimer") and any redistribution must be conditioned upon
     20          1.1    jruoho  *    including a substantially similar Disclaimer requirement for further
     21          1.1    jruoho  *    binary redistribution.
     22          1.1    jruoho  * 3. Neither the names of the above-listed copyright holders nor the names
     23          1.1    jruoho  *    of any contributors may be used to endorse or promote products derived
     24          1.1    jruoho  *    from this software without specific prior written permission.
     25          1.1    jruoho  *
     26          1.1    jruoho  * Alternatively, this software may be distributed under the terms of the
     27          1.1    jruoho  * GNU General Public License ("GPL") version 2 as published by the Free
     28          1.1    jruoho  * Software Foundation.
     29          1.1    jruoho  *
     30          1.1    jruoho  * NO WARRANTY
     31          1.1    jruoho  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
     32          1.1    jruoho  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
     33          1.1    jruoho  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
     34          1.1    jruoho  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
     35          1.1    jruoho  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     36          1.1    jruoho  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     37          1.1    jruoho  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     38          1.1    jruoho  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
     39          1.1    jruoho  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
     40          1.1    jruoho  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     41          1.1    jruoho  * POSSIBILITY OF SUCH DAMAGES.
     42          1.1    jruoho  */
     43          1.1    jruoho 
     44          1.1    jruoho #include "acpi.h"
     45          1.1    jruoho #include "accommon.h"
     46          1.1    jruoho 
     47          1.1    jruoho 
     48          1.1    jruoho #define _COMPONENT          ACPI_NAMESPACE
     49          1.1    jruoho         ACPI_MODULE_NAME    ("hwpci")
     50          1.1    jruoho 
     51          1.1    jruoho 
     52          1.1    jruoho /* PCI configuration space values */
     53          1.1    jruoho 
     54          1.1    jruoho #define PCI_CFG_HEADER_TYPE_REG             0x0E
     55          1.1    jruoho #define PCI_CFG_PRIMARY_BUS_NUMBER_REG      0x18
     56          1.1    jruoho #define PCI_CFG_SECONDARY_BUS_NUMBER_REG    0x19
     57          1.1    jruoho 
     58          1.1    jruoho /* PCI header values */
     59          1.1    jruoho 
     60          1.1    jruoho #define PCI_HEADER_TYPE_MASK                0x7F
     61          1.1    jruoho #define PCI_TYPE_BRIDGE                     0x01
     62          1.1    jruoho #define PCI_TYPE_CARDBUS_BRIDGE             0x02
     63          1.1    jruoho 
     64          1.1    jruoho typedef struct acpi_pci_device
     65          1.1    jruoho {
     66          1.1    jruoho     ACPI_HANDLE             Device;
     67          1.1    jruoho     struct acpi_pci_device  *Next;
     68          1.1    jruoho 
     69          1.1    jruoho } ACPI_PCI_DEVICE;
     70          1.1    jruoho 
     71          1.1    jruoho 
     72          1.1    jruoho /* Local prototypes */
     73          1.1    jruoho 
     74          1.1    jruoho static ACPI_STATUS
     75          1.1    jruoho AcpiHwBuildPciList (
     76          1.1    jruoho     ACPI_HANDLE             RootPciDevice,
     77          1.1    jruoho     ACPI_HANDLE             PciRegion,
     78          1.1    jruoho     ACPI_PCI_DEVICE         **ReturnListHead);
     79          1.1    jruoho 
     80          1.1    jruoho static ACPI_STATUS
     81          1.1    jruoho AcpiHwProcessPciList (
     82          1.1    jruoho     ACPI_PCI_ID             *PciId,
     83          1.1    jruoho     ACPI_PCI_DEVICE         *ListHead);
     84          1.1    jruoho 
     85          1.1    jruoho static void
     86          1.1    jruoho AcpiHwDeletePciList (
     87          1.1    jruoho     ACPI_PCI_DEVICE         *ListHead);
     88          1.1    jruoho 
     89          1.1    jruoho static ACPI_STATUS
     90          1.1    jruoho AcpiHwGetPciDeviceInfo (
     91          1.1    jruoho     ACPI_PCI_ID             *PciId,
     92          1.1    jruoho     ACPI_HANDLE             PciDevice,
     93          1.1    jruoho     UINT16                  *BusNumber,
     94          1.1    jruoho     BOOLEAN                 *IsBridge);
     95          1.1    jruoho 
     96          1.1    jruoho 
     97          1.1    jruoho /*******************************************************************************
     98          1.1    jruoho  *
     99          1.1    jruoho  * FUNCTION:    AcpiHwDerivePciId
    100          1.1    jruoho  *
    101          1.1    jruoho  * PARAMETERS:  PciId               - Initial values for the PCI ID. May be
    102          1.1    jruoho  *                                    modified by this function.
    103          1.1    jruoho  *              RootPciDevice       - A handle to a PCI device object. This
    104          1.1    jruoho  *                                    object must be a PCI Root Bridge having a
    105          1.1    jruoho  *                                    _HID value of either PNP0A03 or PNP0A08
    106          1.1    jruoho  *              PciRegion           - A handle to a PCI configuration space
    107          1.1    jruoho  *                                    Operation Region being initialized
    108          1.1    jruoho  *
    109          1.1    jruoho  * RETURN:      Status
    110          1.1    jruoho  *
    111          1.1    jruoho  * DESCRIPTION: This function derives a full PCI ID for a PCI device,
    112          1.1    jruoho  *              consisting of a Segment number, Bus number, Device number,
    113          1.1    jruoho  *              and function code.
    114          1.1    jruoho  *
    115          1.1    jruoho  *              The PCI hardware dynamically configures PCI bus numbers
    116          1.1    jruoho  *              depending on the bus topology discovered during system
    117          1.1    jruoho  *              initialization. This function is invoked during configuration
    118          1.1    jruoho  *              of a PCI_Config Operation Region in order to (possibly) update
    119          1.1    jruoho  *              the Bus/Device/Function numbers in the PciId with the actual
    120          1.1    jruoho  *              values as determined by the hardware and operating system
    121          1.1    jruoho  *              configuration.
    122          1.1    jruoho  *
    123          1.1    jruoho  *              The PciId parameter is initially populated during the Operation
    124          1.1    jruoho  *              Region initialization. This function is then called, and is
    125          1.1    jruoho  *              will make any necessary modifications to the Bus, Device, or
    126          1.1    jruoho  *              Function number PCI ID subfields as appropriate for the
    127          1.1    jruoho  *              current hardware and OS configuration.
    128          1.1    jruoho  *
    129          1.1    jruoho  * NOTE:        Created 08/2010. Replaces the previous OSL AcpiOsDerivePciId
    130          1.1    jruoho  *              interface since this feature is OS-independent. This module
    131          1.1    jruoho  *              specifically avoids any use of recursion by building a local
    132          1.1    jruoho  *              temporary device list.
    133          1.1    jruoho  *
    134          1.1    jruoho  ******************************************************************************/
    135          1.1    jruoho 
    136          1.1    jruoho ACPI_STATUS
    137          1.1    jruoho AcpiHwDerivePciId (
    138          1.1    jruoho     ACPI_PCI_ID             *PciId,
    139          1.1    jruoho     ACPI_HANDLE             RootPciDevice,
    140          1.1    jruoho     ACPI_HANDLE             PciRegion)
    141          1.1    jruoho {
    142          1.1    jruoho     ACPI_STATUS             Status;
    143  1.1.1.3.2.2     skrll     ACPI_PCI_DEVICE         *ListHead;
    144          1.1    jruoho 
    145          1.1    jruoho 
    146          1.1    jruoho     ACPI_FUNCTION_TRACE (HwDerivePciId);
    147          1.1    jruoho 
    148          1.1    jruoho 
    149          1.1    jruoho     if (!PciId)
    150          1.1    jruoho     {
    151          1.1    jruoho         return_ACPI_STATUS (AE_BAD_PARAMETER);
    152          1.1    jruoho     }
    153          1.1    jruoho 
    154          1.1    jruoho     /* Build a list of PCI devices, from PciRegion up to RootPciDevice */
    155          1.1    jruoho 
    156          1.1    jruoho     Status = AcpiHwBuildPciList (RootPciDevice, PciRegion, &ListHead);
    157          1.1    jruoho     if (ACPI_SUCCESS (Status))
    158          1.1    jruoho     {
    159          1.1    jruoho         /* Walk the list, updating the PCI device/function/bus numbers */
    160          1.1    jruoho 
    161          1.1    jruoho         Status = AcpiHwProcessPciList (PciId, ListHead);
    162          1.1    jruoho 
    163      1.1.1.3  christos         /* Delete the list */
    164      1.1.1.3  christos 
    165      1.1.1.3  christos         AcpiHwDeletePciList (ListHead);
    166      1.1.1.3  christos     }
    167          1.1    jruoho 
    168          1.1    jruoho     return_ACPI_STATUS (Status);
    169          1.1    jruoho }
    170          1.1    jruoho 
    171          1.1    jruoho 
    172          1.1    jruoho /*******************************************************************************
    173          1.1    jruoho  *
    174          1.1    jruoho  * FUNCTION:    AcpiHwBuildPciList
    175          1.1    jruoho  *
    176          1.1    jruoho  * PARAMETERS:  RootPciDevice       - A handle to a PCI device object. This
    177          1.1    jruoho  *                                    object is guaranteed to be a PCI Root
    178          1.1    jruoho  *                                    Bridge having a _HID value of either
    179          1.1    jruoho  *                                    PNP0A03 or PNP0A08
    180          1.1    jruoho  *              PciRegion           - A handle to the PCI configuration space
    181          1.1    jruoho  *                                    Operation Region
    182          1.1    jruoho  *              ReturnListHead      - Where the PCI device list is returned
    183          1.1    jruoho  *
    184          1.1    jruoho  * RETURN:      Status
    185          1.1    jruoho  *
    186          1.1    jruoho  * DESCRIPTION: Builds a list of devices from the input PCI region up to the
    187          1.1    jruoho  *              Root PCI device for this namespace subtree.
    188          1.1    jruoho  *
    189          1.1    jruoho  ******************************************************************************/
    190          1.1    jruoho 
    191          1.1    jruoho static ACPI_STATUS
    192          1.1    jruoho AcpiHwBuildPciList (
    193          1.1    jruoho     ACPI_HANDLE             RootPciDevice,
    194          1.1    jruoho     ACPI_HANDLE             PciRegion,
    195          1.1    jruoho     ACPI_PCI_DEVICE         **ReturnListHead)
    196          1.1    jruoho {
    197          1.1    jruoho     ACPI_HANDLE             CurrentDevice;
    198          1.1    jruoho     ACPI_HANDLE             ParentDevice;
    199          1.1    jruoho     ACPI_STATUS             Status;
    200          1.1    jruoho     ACPI_PCI_DEVICE         *ListElement;
    201          1.1    jruoho 
    202          1.1    jruoho 
    203          1.1    jruoho     /*
    204          1.1    jruoho      * Ascend namespace branch until the RootPciDevice is reached, building
    205          1.1    jruoho      * a list of device nodes. Loop will exit when either the PCI device is
    206          1.1    jruoho      * found, or the root of the namespace is reached.
    207          1.1    jruoho      */
    208  1.1.1.3.2.2     skrll     *ReturnListHead = NULL;
    209          1.1    jruoho     CurrentDevice = PciRegion;
    210          1.1    jruoho     while (1)
    211          1.1    jruoho     {
    212          1.1    jruoho         Status = AcpiGetParent (CurrentDevice, &ParentDevice);
    213          1.1    jruoho         if (ACPI_FAILURE (Status))
    214          1.1    jruoho         {
    215      1.1.1.3  christos             /* Must delete the list before exit */
    216      1.1.1.3  christos 
    217      1.1.1.3  christos             AcpiHwDeletePciList (*ReturnListHead);
    218          1.1    jruoho             return (Status);
    219          1.1    jruoho         }
    220          1.1    jruoho 
    221          1.1    jruoho         /* Finished when we reach the PCI root device (PNP0A03 or PNP0A08) */
    222          1.1    jruoho 
    223          1.1    jruoho         if (ParentDevice == RootPciDevice)
    224          1.1    jruoho         {
    225          1.1    jruoho             return (AE_OK);
    226          1.1    jruoho         }
    227          1.1    jruoho 
    228          1.1    jruoho         ListElement = ACPI_ALLOCATE (sizeof (ACPI_PCI_DEVICE));
    229          1.1    jruoho         if (!ListElement)
    230          1.1    jruoho         {
    231      1.1.1.3  christos             /* Must delete the list before exit */
    232      1.1.1.3  christos 
    233      1.1.1.3  christos             AcpiHwDeletePciList (*ReturnListHead);
    234          1.1    jruoho             return (AE_NO_MEMORY);
    235          1.1    jruoho         }
    236          1.1    jruoho 
    237          1.1    jruoho         /* Put new element at the head of the list */
    238          1.1    jruoho 
    239  1.1.1.3.2.2     skrll         ListElement->Next = *ReturnListHead;
    240          1.1    jruoho         ListElement->Device = ParentDevice;
    241  1.1.1.3.2.2     skrll         *ReturnListHead = ListElement;
    242          1.1    jruoho 
    243          1.1    jruoho         CurrentDevice = ParentDevice;
    244          1.1    jruoho     }
    245          1.1    jruoho }
    246          1.1    jruoho 
    247          1.1    jruoho 
    248          1.1    jruoho /*******************************************************************************
    249          1.1    jruoho  *
    250          1.1    jruoho  * FUNCTION:    AcpiHwProcessPciList
    251          1.1    jruoho  *
    252          1.1    jruoho  * PARAMETERS:  PciId               - Initial values for the PCI ID. May be
    253          1.1    jruoho  *                                    modified by this function.
    254          1.1    jruoho  *              ListHead            - Device list created by
    255          1.1    jruoho  *                                    AcpiHwBuildPciList
    256          1.1    jruoho  *
    257          1.1    jruoho  * RETURN:      Status
    258          1.1    jruoho  *
    259          1.1    jruoho  * DESCRIPTION: Walk downward through the PCI device list, getting the device
    260          1.1    jruoho  *              info for each, via the PCI configuration space and updating
    261          1.1    jruoho  *              the PCI ID as necessary. Deletes the list during traversal.
    262          1.1    jruoho  *
    263          1.1    jruoho  ******************************************************************************/
    264          1.1    jruoho 
    265          1.1    jruoho static ACPI_STATUS
    266          1.1    jruoho AcpiHwProcessPciList (
    267          1.1    jruoho     ACPI_PCI_ID             *PciId,
    268          1.1    jruoho     ACPI_PCI_DEVICE         *ListHead)
    269          1.1    jruoho {
    270          1.1    jruoho     ACPI_STATUS             Status = AE_OK;
    271          1.1    jruoho     ACPI_PCI_DEVICE         *Info;
    272          1.1    jruoho     UINT16                  BusNumber;
    273          1.1    jruoho     BOOLEAN                 IsBridge = TRUE;
    274          1.1    jruoho 
    275          1.1    jruoho 
    276          1.1    jruoho     ACPI_FUNCTION_NAME (HwProcessPciList);
    277          1.1    jruoho 
    278          1.1    jruoho 
    279          1.1    jruoho     ACPI_DEBUG_PRINT ((ACPI_DB_OPREGION,
    280          1.1    jruoho         "Input PciId:  Seg %4.4X Bus %4.4X Dev %4.4X Func %4.4X\n",
    281          1.1    jruoho         PciId->Segment, PciId->Bus, PciId->Device, PciId->Function));
    282          1.1    jruoho 
    283          1.1    jruoho     BusNumber = PciId->Bus;
    284          1.1    jruoho 
    285          1.1    jruoho     /*
    286          1.1    jruoho      * Descend down the namespace tree, collecting PCI device, function,
    287          1.1    jruoho      * and bus numbers. BusNumber is only important for PCI bridges.
    288          1.1    jruoho      * Algorithm: As we descend the tree, use the last valid PCI device,
    289          1.1    jruoho      * function, and bus numbers that are discovered, and assign them
    290          1.1    jruoho      * to the PCI ID for the target device.
    291          1.1    jruoho      */
    292          1.1    jruoho     Info = ListHead;
    293          1.1    jruoho     while (Info)
    294          1.1    jruoho     {
    295          1.1    jruoho         Status = AcpiHwGetPciDeviceInfo (PciId, Info->Device,
    296          1.1    jruoho             &BusNumber, &IsBridge);
    297          1.1    jruoho         if (ACPI_FAILURE (Status))
    298          1.1    jruoho         {
    299      1.1.1.2  christos             return (Status);
    300          1.1    jruoho         }
    301          1.1    jruoho 
    302          1.1    jruoho         Info = Info->Next;
    303          1.1    jruoho     }
    304          1.1    jruoho 
    305          1.1    jruoho     ACPI_DEBUG_PRINT ((ACPI_DB_OPREGION,
    306          1.1    jruoho         "Output PciId: Seg %4.4X Bus %4.4X Dev %4.4X Func %4.4X "
    307          1.1    jruoho         "Status %X BusNumber %X IsBridge %X\n",
    308          1.1    jruoho         PciId->Segment, PciId->Bus, PciId->Device, PciId->Function,
    309          1.1    jruoho         Status, BusNumber, IsBridge));
    310          1.1    jruoho 
    311      1.1.1.2  christos     return (AE_OK);
    312          1.1    jruoho }
    313          1.1    jruoho 
    314          1.1    jruoho 
    315          1.1    jruoho /*******************************************************************************
    316          1.1    jruoho  *
    317          1.1    jruoho  * FUNCTION:    AcpiHwDeletePciList
    318          1.1    jruoho  *
    319          1.1    jruoho  * PARAMETERS:  ListHead            - Device list created by
    320          1.1    jruoho  *                                    AcpiHwBuildPciList
    321          1.1    jruoho  *
    322          1.1    jruoho  * RETURN:      None
    323          1.1    jruoho  *
    324          1.1    jruoho  * DESCRIPTION: Free the entire PCI list.
    325          1.1    jruoho  *
    326          1.1    jruoho  ******************************************************************************/
    327          1.1    jruoho 
    328          1.1    jruoho static void
    329          1.1    jruoho AcpiHwDeletePciList (
    330          1.1    jruoho     ACPI_PCI_DEVICE         *ListHead)
    331          1.1    jruoho {
    332          1.1    jruoho     ACPI_PCI_DEVICE         *Next;
    333          1.1    jruoho     ACPI_PCI_DEVICE         *Previous;
    334          1.1    jruoho 
    335          1.1    jruoho 
    336          1.1    jruoho     Next = ListHead;
    337          1.1    jruoho     while (Next)
    338          1.1    jruoho     {
    339          1.1    jruoho         Previous = Next;
    340          1.1    jruoho         Next = Previous->Next;
    341          1.1    jruoho         ACPI_FREE (Previous);
    342          1.1    jruoho     }
    343          1.1    jruoho }
    344          1.1    jruoho 
    345          1.1    jruoho 
    346          1.1    jruoho /*******************************************************************************
    347          1.1    jruoho  *
    348          1.1    jruoho  * FUNCTION:    AcpiHwGetPciDeviceInfo
    349          1.1    jruoho  *
    350          1.1    jruoho  * PARAMETERS:  PciId               - Initial values for the PCI ID. May be
    351          1.1    jruoho  *                                    modified by this function.
    352          1.1    jruoho  *              PciDevice           - Handle for the PCI device object
    353          1.1    jruoho  *              BusNumber           - Where a PCI bridge bus number is returned
    354          1.1    jruoho  *              IsBridge            - Return value, indicates if this PCI
    355          1.1    jruoho  *                                    device is a PCI bridge
    356          1.1    jruoho  *
    357          1.1    jruoho  * RETURN:      Status
    358          1.1    jruoho  *
    359          1.1    jruoho  * DESCRIPTION: Get the device info for a single PCI device object. Get the
    360          1.1    jruoho  *              _ADR (contains PCI device and function numbers), and for PCI
    361          1.1    jruoho  *              bridge devices, get the bus number from PCI configuration
    362          1.1    jruoho  *              space.
    363          1.1    jruoho  *
    364          1.1    jruoho  ******************************************************************************/
    365          1.1    jruoho 
    366          1.1    jruoho static ACPI_STATUS
    367          1.1    jruoho AcpiHwGetPciDeviceInfo (
    368          1.1    jruoho     ACPI_PCI_ID             *PciId,
    369          1.1    jruoho     ACPI_HANDLE             PciDevice,
    370          1.1    jruoho     UINT16                  *BusNumber,
    371          1.1    jruoho     BOOLEAN                 *IsBridge)
    372          1.1    jruoho {
    373          1.1    jruoho     ACPI_STATUS             Status;
    374          1.1    jruoho     ACPI_OBJECT_TYPE        ObjectType;
    375          1.1    jruoho     UINT64                  ReturnValue;
    376          1.1    jruoho     UINT64                  PciValue;
    377          1.1    jruoho 
    378          1.1    jruoho 
    379          1.1    jruoho     /* We only care about objects of type Device */
    380          1.1    jruoho 
    381          1.1    jruoho     Status = AcpiGetType (PciDevice, &ObjectType);
    382          1.1    jruoho     if (ACPI_FAILURE (Status))
    383          1.1    jruoho     {
    384          1.1    jruoho         return (Status);
    385          1.1    jruoho     }
    386          1.1    jruoho 
    387          1.1    jruoho     if (ObjectType != ACPI_TYPE_DEVICE)
    388          1.1    jruoho     {
    389          1.1    jruoho         return (AE_OK);
    390          1.1    jruoho     }
    391          1.1    jruoho 
    392          1.1    jruoho     /* We need an _ADR. Ignore device if not present */
    393          1.1    jruoho 
    394          1.1    jruoho     Status = AcpiUtEvaluateNumericObject (METHOD_NAME__ADR,
    395          1.1    jruoho         PciDevice, &ReturnValue);
    396          1.1    jruoho     if (ACPI_FAILURE (Status))
    397          1.1    jruoho     {
    398          1.1    jruoho         return (AE_OK);
    399          1.1    jruoho     }
    400          1.1    jruoho 
    401          1.1    jruoho     /*
    402          1.1    jruoho      * From _ADR, get the PCI Device and Function and
    403          1.1    jruoho      * update the PCI ID.
    404          1.1    jruoho      */
    405          1.1    jruoho     PciId->Device = ACPI_HIWORD (ACPI_LODWORD (ReturnValue));
    406          1.1    jruoho     PciId->Function = ACPI_LOWORD (ACPI_LODWORD (ReturnValue));
    407          1.1    jruoho 
    408          1.1    jruoho     /*
    409          1.1    jruoho      * If the previous device was a bridge, use the previous
    410          1.1    jruoho      * device bus number
    411          1.1    jruoho      */
    412          1.1    jruoho     if (*IsBridge)
    413          1.1    jruoho     {
    414          1.1    jruoho         PciId->Bus = *BusNumber;
    415          1.1    jruoho     }
    416          1.1    jruoho 
    417          1.1    jruoho     /*
    418          1.1    jruoho      * Get the bus numbers from PCI Config space:
    419          1.1    jruoho      *
    420          1.1    jruoho      * First, get the PCI HeaderType
    421          1.1    jruoho      */
    422          1.1    jruoho     *IsBridge = FALSE;
    423          1.1    jruoho     Status = AcpiOsReadPciConfiguration (PciId,
    424          1.1    jruoho         PCI_CFG_HEADER_TYPE_REG, &PciValue, 8);
    425          1.1    jruoho     if (ACPI_FAILURE (Status))
    426          1.1    jruoho     {
    427          1.1    jruoho         return (Status);
    428          1.1    jruoho     }
    429          1.1    jruoho 
    430          1.1    jruoho     /* We only care about bridges (1=PciBridge, 2=CardBusBridge) */
    431          1.1    jruoho 
    432          1.1    jruoho     PciValue &= PCI_HEADER_TYPE_MASK;
    433          1.1    jruoho 
    434          1.1    jruoho     if ((PciValue != PCI_TYPE_BRIDGE) &&
    435          1.1    jruoho         (PciValue != PCI_TYPE_CARDBUS_BRIDGE))
    436          1.1    jruoho     {
    437          1.1    jruoho         return (AE_OK);
    438          1.1    jruoho     }
    439          1.1    jruoho 
    440          1.1    jruoho     /* Bridge: Get the Primary BusNumber */
    441          1.1    jruoho 
    442          1.1    jruoho     Status = AcpiOsReadPciConfiguration (PciId,
    443          1.1    jruoho         PCI_CFG_PRIMARY_BUS_NUMBER_REG, &PciValue, 8);
    444          1.1    jruoho     if (ACPI_FAILURE (Status))
    445          1.1    jruoho     {
    446          1.1    jruoho         return (Status);
    447          1.1    jruoho     }
    448          1.1    jruoho 
    449          1.1    jruoho     *IsBridge = TRUE;
    450          1.1    jruoho     PciId->Bus = (UINT16) PciValue;
    451          1.1    jruoho 
    452          1.1    jruoho     /* Bridge: Get the Secondary BusNumber */
    453          1.1    jruoho 
    454          1.1    jruoho     Status = AcpiOsReadPciConfiguration (PciId,
    455          1.1    jruoho         PCI_CFG_SECONDARY_BUS_NUMBER_REG, &PciValue, 8);
    456          1.1    jruoho     if (ACPI_FAILURE (Status))
    457          1.1    jruoho     {
    458          1.1    jruoho         return (Status);
    459          1.1    jruoho     }
    460          1.1    jruoho 
    461          1.1    jruoho     *BusNumber = (UINT16) PciValue;
    462          1.1    jruoho     return (AE_OK);
    463          1.1    jruoho }
    464