LCOV - code coverage report
Current view: top level - drivers/pci - host-bridge.c (source / functions) Hit Total Coverage
Test: combined.info Lines: 47 47 100.0 %
Date: 2022-04-01 14:17:54 Functions: 6 6 100.0 %
Branches: 17 24 70.8 %

           Branch data     Line data    Source code
       1                 :            : // SPDX-License-Identifier: GPL-2.0
       2                 :            : /*
       3                 :            :  * Host bridge related code
       4                 :            :  */
       5                 :            : 
       6                 :            : #include <linux/kernel.h>
       7                 :            : #include <linux/pci.h>
       8                 :            : #include <linux/module.h>
       9                 :            : 
      10                 :            : #include "pci.h"
      11                 :            : 
      12                 :        495 : static struct pci_bus *find_pci_root_bus(struct pci_bus *bus)
      13                 :            : {
      14   [ -  +  -  +  :        495 :         while (bus->parent)
             -  +  -  + ]
      15                 :            :                 bus = bus->parent;
      16                 :            : 
      17                 :        495 :         return bus;
      18                 :            : }
      19                 :            : 
      20                 :        440 : struct pci_host_bridge *pci_find_host_bridge(struct pci_bus *bus)
      21                 :            : {
      22                 :        198 :         struct pci_bus *root_bus = find_pci_root_bus(bus);
      23                 :            : 
      24                 :        440 :         return to_pci_host_bridge(root_bus->bridge);
      25                 :            : }
      26                 :            : 
      27                 :         55 : struct device *pci_get_host_bridge_device(struct pci_dev *dev)
      28                 :            : {
      29                 :         55 :         struct pci_bus *root_bus = find_pci_root_bus(dev->bus);
      30                 :         55 :         struct device *bridge = root_bus->bridge;
      31                 :            : 
      32                 :         55 :         kobject_get(&bridge->kobj);
      33                 :         55 :         return bridge;
      34                 :            : }
      35                 :            : 
      36                 :         55 : void  pci_put_host_bridge_device(struct device *dev)
      37                 :            : {
      38                 :         55 :         kobject_put(&dev->kobj);
      39                 :         55 : }
      40                 :            : 
      41                 :         11 : void pci_set_host_bridge_release(struct pci_host_bridge *bridge,
      42                 :            :                                  void (*release_fn)(struct pci_host_bridge *),
      43                 :            :                                  void *release_data)
      44                 :            : {
      45                 :         11 :         bridge->release_fn = release_fn;
      46                 :         11 :         bridge->release_data = release_data;
      47                 :         11 : }
      48                 :            : EXPORT_SYMBOL_GPL(pci_set_host_bridge_release);
      49                 :            : 
      50                 :         88 : void pcibios_resource_to_bus(struct pci_bus *bus, struct pci_bus_region *region,
      51                 :            :                              struct resource *res)
      52                 :            : {
      53                 :         88 :         struct pci_host_bridge *bridge = pci_find_host_bridge(bus);
      54                 :         88 :         struct resource_entry *window;
      55                 :         88 :         resource_size_t offset = 0;
      56                 :            : 
      57         [ +  - ]:        330 :         resource_list_for_each_entry(window, &bridge->windows) {
      58   [ +  +  +  + ]:        506 :                 if (resource_contains(window->res, res)) {
      59                 :         88 :                         offset = window->offset;
      60                 :         88 :                         break;
      61                 :            :                 }
      62                 :            :         }
      63                 :            : 
      64                 :         88 :         region->start = res->start - offset;
      65                 :         88 :         region->end = res->end - offset;
      66                 :         88 : }
      67                 :            : EXPORT_SYMBOL(pcibios_resource_to_bus);
      68                 :            : 
      69                 :        242 : static bool region_contains(struct pci_bus_region *region1,
      70                 :            :                             struct pci_bus_region *region2)
      71                 :            : {
      72         [ +  + ]:        242 :         return region1->start <= region2->start && region1->end >= region2->end;
      73                 :            : }
      74                 :            : 
      75                 :        154 : void pcibios_bus_to_resource(struct pci_bus *bus, struct resource *res,
      76                 :            :                              struct pci_bus_region *region)
      77                 :            : {
      78                 :        154 :         struct pci_host_bridge *bridge = pci_find_host_bridge(bus);
      79                 :        154 :         struct resource_entry *window;
      80                 :        154 :         resource_size_t offset = 0;
      81                 :            : 
      82         [ +  - ]:        396 :         resource_list_for_each_entry(window, &bridge->windows) {
      83                 :        396 :                 struct pci_bus_region bus_region;
      84                 :            : 
      85         [ +  + ]:        396 :                 if (resource_type(res) != resource_type(window->res))
      86                 :        154 :                         continue;
      87                 :            : 
      88                 :        242 :                 bus_region.start = window->res->start - window->offset;
      89                 :        242 :                 bus_region.end = window->res->end - window->offset;
      90                 :            : 
      91   [ +  -  +  + ]:        484 :                 if (region_contains(&bus_region, region)) {
      92                 :        154 :                         offset = window->offset;
      93                 :        154 :                         break;
      94                 :            :                 }
      95                 :            :         }
      96                 :            : 
      97                 :        154 :         res->start = region->start + offset;
      98                 :        154 :         res->end = region->end + offset;
      99                 :        154 : }
     100                 :            : EXPORT_SYMBOL(pcibios_bus_to_resource);

Generated by: LCOV version 1.14