LCOV - code coverage report
Current view: top level - drivers/acpi - resource.c (source / functions) Hit Total Coverage
Test: combined.info Lines: 182 293 62.1 %
Date: 2022-03-28 16:04:14 Functions: 14 21 66.7 %
Branches: 64 169 37.9 %

           Branch data     Line data    Source code
       1                 :            : // SPDX-License-Identifier: GPL-2.0-only
       2                 :            : /*
       3                 :            :  * drivers/acpi/resource.c - ACPI device resources interpretation.
       4                 :            :  *
       5                 :            :  * Copyright (C) 2012, Intel Corp.
       6                 :            :  * Author: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
       7                 :            :  *
       8                 :            :  * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
       9                 :            :  *
      10                 :            :  * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
      11                 :            :  */
      12                 :            : 
      13                 :            : #include <linux/acpi.h>
      14                 :            : #include <linux/device.h>
      15                 :            : #include <linux/export.h>
      16                 :            : #include <linux/ioport.h>
      17                 :            : #include <linux/slab.h>
      18                 :            : #include <linux/irq.h>
      19                 :            : 
      20                 :            : #ifdef CONFIG_X86
      21                 :            : #define valid_IRQ(i) (((i) != 0) && ((i) != 2))
      22                 :        156 : static inline bool acpi_iospace_resource_valid(struct resource *res)
      23                 :            : {
      24                 :            :         /* On X86 IO space is limited to the [0 - 64K] IO port range */
      25                 :        156 :         return res->end < 0x10003;
      26                 :            : }
      27                 :            : #else
      28                 :            : #define valid_IRQ(i) (true)
      29                 :            : /*
      30                 :            :  * ACPI IO descriptors on arches other than X86 contain MMIO CPU physical
      31                 :            :  * addresses mapping IO space in CPU physical address space, IO space
      32                 :            :  * resources can be placed anywhere in the 64-bit physical address space.
      33                 :            :  */
      34                 :            : static inline bool
      35                 :            : acpi_iospace_resource_valid(struct resource *res) { return true; }
      36                 :            : #endif
      37                 :            : 
      38                 :            : #if IS_ENABLED(CONFIG_ACPI_GENERIC_GSI)
      39                 :            : static inline bool is_gsi(struct acpi_resource_extended_irq *ext_irq)
      40                 :            : {
      41                 :            :         return ext_irq->resource_source.string_length == 0 &&
      42                 :            :                ext_irq->producer_consumer == ACPI_CONSUMER;
      43                 :            : }
      44                 :            : #else
      45                 :          0 : static inline bool is_gsi(struct acpi_resource_extended_irq *ext_irq)
      46                 :            : {
      47                 :          0 :         return true;
      48                 :            : }
      49                 :            : #endif
      50                 :            : 
      51                 :        195 : static bool acpi_dev_resource_len_valid(u64 start, u64 end, u64 len, bool io)
      52                 :            : {
      53                 :        195 :         u64 reslen = end - start + 1;
      54                 :            : 
      55                 :            :         /*
      56                 :            :          * CHECKME: len might be required to check versus a minimum
      57                 :            :          * length as well. 1 for io is fine, but for memory it does
      58                 :            :          * not make any sense at all.
      59                 :            :          * Note: some BIOSes report incorrect length for ACPI address space
      60                 :            :          * descriptor, so remove check of 'reslen == len' to avoid regression.
      61                 :            :          */
      62   [ -  +  -  +  :        195 :         if (len && reslen && start <= end)
          -  -  -  -  -  
                      + ]
      63                 :            :                 return true;
      64                 :            : 
      65                 :            :         pr_debug("ACPI: invalid or unassigned resource %s [%016llx - %016llx] length [%016llx]\n",
      66                 :            :                 io ? "io" : "mem", start, end, len);
      67                 :            : 
      68                 :            :         return false;
      69                 :            : }
      70                 :            : 
      71                 :         39 : static void acpi_dev_memresource_flags(struct resource *res, u64 len,
      72                 :            :                                        u8 write_protect)
      73                 :            : {
      74                 :         39 :         res->flags = IORESOURCE_MEM;
      75                 :            : 
      76                 :         39 :         if (!acpi_dev_resource_len_valid(res->start, res->end, len, false))
      77                 :          0 :                 res->flags |= IORESOURCE_DISABLED | IORESOURCE_UNSET;
      78                 :            : 
      79   [ +  -  -  -  :         39 :         if (write_protect == ACPI_READ_WRITE_MEMORY)
             -  -  -  + ]
      80                 :         26 :                 res->flags |= IORESOURCE_MEM_WRITEABLE;
      81                 :            : }
      82                 :            : 
      83                 :         13 : static void acpi_dev_get_memresource(struct resource *res, u64 start, u64 len,
      84                 :            :                                      u8 write_protect)
      85                 :            : {
      86                 :         13 :         res->start = start;
      87                 :         13 :         res->end = start + len - 1;
      88                 :         13 :         acpi_dev_memresource_flags(res, len, write_protect);
      89                 :            : }
      90                 :            : 
      91                 :            : /**
      92                 :            :  * acpi_dev_resource_memory - Extract ACPI memory resource information.
      93                 :            :  * @ares: Input ACPI resource object.
      94                 :            :  * @res: Output generic resource object.
      95                 :            :  *
      96                 :            :  * Check if the given ACPI resource object represents a memory resource and
      97                 :            :  * if that's the case, use the information in it to populate the generic
      98                 :            :  * resource object pointed to by @res.
      99                 :            :  *
     100                 :            :  * Return:
     101                 :            :  * 1) false with res->flags setting to zero: not the expected resource type
     102                 :            :  * 2) false with IORESOURCE_DISABLED in res->flags: valid unassigned resource
     103                 :            :  * 3) true: valid assigned resource
     104                 :            :  */
     105                 :        117 : bool acpi_dev_resource_memory(struct acpi_resource *ares, struct resource *res)
     106                 :            : {
     107                 :        117 :         struct acpi_resource_memory24 *memory24;
     108                 :        117 :         struct acpi_resource_memory32 *memory32;
     109                 :        117 :         struct acpi_resource_fixed_memory32 *fixed_memory32;
     110                 :            : 
     111   [ -  -  +  + ]:        117 :         switch (ares->type) {
     112                 :          0 :         case ACPI_RESOURCE_TYPE_MEMORY24:
     113                 :          0 :                 memory24 = &ares->data.memory24;
     114                 :          0 :                 acpi_dev_get_memresource(res, memory24->minimum << 8,
     115                 :          0 :                                          memory24->address_length << 8,
     116         [ #  # ]:          0 :                                          memory24->write_protect);
     117                 :            :                 break;
     118                 :          0 :         case ACPI_RESOURCE_TYPE_MEMORY32:
     119                 :          0 :                 memory32 = &ares->data.memory32;
     120                 :          0 :                 acpi_dev_get_memresource(res, memory32->minimum,
     121                 :          0 :                                          memory32->address_length,
     122         [ #  # ]:          0 :                                          memory32->write_protect);
     123                 :            :                 break;
     124                 :         13 :         case ACPI_RESOURCE_TYPE_FIXED_MEMORY32:
     125                 :         13 :                 fixed_memory32 = &ares->data.fixed_memory32;
     126                 :         13 :                 acpi_dev_get_memresource(res, fixed_memory32->address,
     127                 :         13 :                                          fixed_memory32->address_length,
     128         [ +  - ]:         13 :                                          fixed_memory32->write_protect);
     129                 :            :                 break;
     130                 :        104 :         default:
     131                 :        104 :                 res->flags = 0;
     132                 :        104 :                 return false;
     133                 :            :         }
     134                 :            : 
     135                 :         13 :         return !(res->flags & IORESOURCE_DISABLED);
     136                 :            : }
     137                 :            : EXPORT_SYMBOL_GPL(acpi_dev_resource_memory);
     138                 :            : 
     139                 :        156 : static void acpi_dev_ioresource_flags(struct resource *res, u64 len,
     140                 :            :                                       u8 io_decode, u8 translation_type)
     141                 :            : {
     142                 :        156 :         res->flags = IORESOURCE_IO;
     143                 :            : 
     144         [ +  - ]:        156 :         if (!acpi_dev_resource_len_valid(res->start, res->end, len, true))
     145                 :          0 :                 res->flags |= IORESOURCE_DISABLED | IORESOURCE_UNSET;
     146                 :            : 
     147         [ -  + ]:        156 :         if (!acpi_iospace_resource_valid(res))
     148                 :          0 :                 res->flags |= IORESOURCE_DISABLED | IORESOURCE_UNSET;
     149                 :            : 
     150         [ +  - ]:        156 :         if (io_decode == ACPI_DECODE_16)
     151                 :        156 :                 res->flags |= IORESOURCE_IO_16BIT_ADDR;
     152         [ -  + ]:        156 :         if (translation_type == ACPI_SPARSE_TRANSLATION)
     153                 :          0 :                 res->flags |= IORESOURCE_IO_SPARSE;
     154                 :        156 : }
     155                 :            : 
     156                 :        130 : static void acpi_dev_get_ioresource(struct resource *res, u64 start, u64 len,
     157                 :            :                                     u8 io_decode)
     158                 :            : {
     159                 :        130 :         res->start = start;
     160                 :        130 :         res->end = start + len - 1;
     161                 :        130 :         acpi_dev_ioresource_flags(res, len, io_decode, 0);
     162                 :        130 : }
     163                 :            : 
     164                 :            : /**
     165                 :            :  * acpi_dev_resource_io - Extract ACPI I/O resource information.
     166                 :            :  * @ares: Input ACPI resource object.
     167                 :            :  * @res: Output generic resource object.
     168                 :            :  *
     169                 :            :  * Check if the given ACPI resource object represents an I/O resource and
     170                 :            :  * if that's the case, use the information in it to populate the generic
     171                 :            :  * resource object pointed to by @res.
     172                 :            :  *
     173                 :            :  * Return:
     174                 :            :  * 1) false with res->flags setting to zero: not the expected resource type
     175                 :            :  * 2) false with IORESOURCE_DISABLED in res->flags: valid unassigned resource
     176                 :            :  * 3) true: valid assigned resource
     177                 :            :  */
     178                 :        208 : bool acpi_dev_resource_io(struct acpi_resource *ares, struct resource *res)
     179                 :            : {
     180                 :        208 :         struct acpi_resource_io *io;
     181                 :        208 :         struct acpi_resource_fixed_io *fixed_io;
     182                 :            : 
     183      [ +  -  + ]:        208 :         switch (ares->type) {
     184                 :        130 :         case ACPI_RESOURCE_TYPE_IO:
     185                 :        130 :                 io = &ares->data.io;
     186                 :        130 :                 acpi_dev_get_ioresource(res, io->minimum,
     187                 :        130 :                                         io->address_length,
     188                 :        130 :                                         io->io_decode);
     189                 :            :                 break;
     190                 :          0 :         case ACPI_RESOURCE_TYPE_FIXED_IO:
     191                 :          0 :                 fixed_io = &ares->data.fixed_io;
     192                 :          0 :                 acpi_dev_get_ioresource(res, fixed_io->address,
     193                 :          0 :                                         fixed_io->address_length,
     194                 :            :                                         ACPI_DECODE_10);
     195                 :            :                 break;
     196                 :         78 :         default:
     197                 :         78 :                 res->flags = 0;
     198                 :         78 :                 return false;
     199                 :            :         }
     200                 :            : 
     201                 :        130 :         return !(res->flags & IORESOURCE_DISABLED);
     202                 :            : }
     203                 :            : EXPORT_SYMBOL_GPL(acpi_dev_resource_io);
     204                 :            : 
     205                 :         52 : static bool acpi_decode_space(struct resource_win *win,
     206                 :            :                               struct acpi_resource_address *addr,
     207                 :            :                               struct acpi_address64_attribute *attr)
     208                 :            : {
     209                 :         52 :         u8 iodec = attr->granularity == 0xfff ? ACPI_DECODE_10 : ACPI_DECODE_16;
     210                 :         52 :         bool wp = addr->info.mem.write_protect;
     211                 :         52 :         u64 len = attr->address_length;
     212                 :         52 :         u64 start, end, offset = 0;
     213                 :         52 :         struct resource *res = &win->res;
     214                 :            : 
     215                 :            :         /*
     216                 :            :          * Filter out invalid descriptor according to ACPI Spec 5.0, section
     217                 :            :          * 6.4.3.5 Address Space Resource Descriptors.
     218                 :            :          */
     219                 :         52 :         if ((addr->min_address_fixed != addr->max_address_fixed && len) ||
     220                 :            :             (addr->min_address_fixed && addr->max_address_fixed && !len))
     221                 :            :                 pr_debug("ACPI: Invalid address space min_addr_fix %d, max_addr_fix %d, len %llx\n",
     222                 :            :                          addr->min_address_fixed, addr->max_address_fixed, len);
     223                 :            : 
     224                 :            :         /*
     225                 :            :          * For bridges that translate addresses across the bridge,
     226                 :            :          * translation_offset is the offset that must be added to the
     227                 :            :          * address on the secondary side to obtain the address on the
     228                 :            :          * primary side. Non-bridge devices must list 0 for all Address
     229                 :            :          * Translation offset bits.
     230                 :            :          */
     231         [ +  - ]:         52 :         if (addr->producer_consumer == ACPI_PRODUCER)
     232                 :         52 :                 offset = attr->translation_offset;
     233                 :            :         else if (attr->translation_offset)
     234                 :            :                 pr_debug("ACPI: translation_offset(%lld) is invalid for non-bridge device.\n",
     235                 :            :                          attr->translation_offset);
     236                 :         52 :         start = attr->minimum + offset;
     237                 :         52 :         end = attr->maximum + offset;
     238                 :            : 
     239                 :         52 :         win->offset = offset;
     240                 :         52 :         res->start = start;
     241                 :         52 :         res->end = end;
     242                 :         52 :         if (sizeof(resource_size_t) < sizeof(u64) &&
     243                 :            :             (offset != win->offset || start != res->start || end != res->end)) {
     244                 :            :                 pr_warn("acpi resource window ([%#llx-%#llx] ignored, not CPU addressable)\n",
     245                 :            :                         attr->minimum, attr->maximum);
     246                 :            :                 return false;
     247                 :            :         }
     248                 :            : 
     249   [ +  +  -  - ]:         52 :         switch (addr->resource_type) {
     250                 :         26 :         case ACPI_MEMORY_RANGE:
     251         [ +  - ]:         26 :                 acpi_dev_memresource_flags(res, len, wp);
     252                 :            :                 break;
     253                 :         26 :         case ACPI_IO_RANGE:
     254                 :         26 :                 acpi_dev_ioresource_flags(res, len, iodec,
     255                 :         26 :                                           addr->info.io.translation_type);
     256                 :         26 :                 break;
     257                 :          0 :         case ACPI_BUS_NUMBER_RANGE:
     258                 :          0 :                 res->flags = IORESOURCE_BUS;
     259                 :          0 :                 break;
     260                 :            :         default:
     261                 :            :                 return false;
     262                 :            :         }
     263                 :            : 
     264         [ +  - ]:         52 :         if (addr->producer_consumer == ACPI_PRODUCER)
     265                 :         52 :                 res->flags |= IORESOURCE_WINDOW;
     266                 :            : 
     267         [ -  + ]:         52 :         if (addr->info.mem.caching == ACPI_PREFETCHABLE_MEMORY)
     268                 :          0 :                 res->flags |= IORESOURCE_PREFETCH;
     269                 :            : 
     270                 :         52 :         return !(res->flags & IORESOURCE_DISABLED);
     271                 :            : }
     272                 :            : 
     273                 :            : /**
     274                 :            :  * acpi_dev_resource_address_space - Extract ACPI address space information.
     275                 :            :  * @ares: Input ACPI resource object.
     276                 :            :  * @win: Output generic resource object.
     277                 :            :  *
     278                 :            :  * Check if the given ACPI resource object represents an address space resource
     279                 :            :  * and if that's the case, use the information in it to populate the generic
     280                 :            :  * resource object pointed to by @win.
     281                 :            :  *
     282                 :            :  * Return:
     283                 :            :  * 1) false with win->res.flags setting to zero: not the expected resource type
     284                 :            :  * 2) false with IORESOURCE_DISABLED in win->res.flags: valid unassigned
     285                 :            :  *    resource
     286                 :            :  * 3) true: valid assigned resource
     287                 :            :  */
     288                 :        351 : bool acpi_dev_resource_address_space(struct acpi_resource *ares,
     289                 :            :                                      struct resource_win *win)
     290                 :            : {
     291                 :        351 :         struct acpi_resource_address64 addr;
     292                 :            : 
     293                 :        351 :         win->res.flags = 0;
     294         [ +  + ]:        351 :         if (ACPI_FAILURE(acpi_resource_to_address64(ares, &addr)))
     295                 :            :                 return false;
     296                 :            : 
     297                 :         52 :         return acpi_decode_space(win, (struct acpi_resource_address *)&addr,
     298                 :            :                                  &addr.address);
     299                 :            : }
     300                 :            : EXPORT_SYMBOL_GPL(acpi_dev_resource_address_space);
     301                 :            : 
     302                 :            : /**
     303                 :            :  * acpi_dev_resource_ext_address_space - Extract ACPI address space information.
     304                 :            :  * @ares: Input ACPI resource object.
     305                 :            :  * @win: Output generic resource object.
     306                 :            :  *
     307                 :            :  * Check if the given ACPI resource object represents an extended address space
     308                 :            :  * resource and if that's the case, use the information in it to populate the
     309                 :            :  * generic resource object pointed to by @win.
     310                 :            :  *
     311                 :            :  * Return:
     312                 :            :  * 1) false with win->res.flags setting to zero: not the expected resource type
     313                 :            :  * 2) false with IORESOURCE_DISABLED in win->res.flags: valid unassigned
     314                 :            :  *    resource
     315                 :            :  * 3) true: valid assigned resource
     316                 :            :  */
     317                 :        299 : bool acpi_dev_resource_ext_address_space(struct acpi_resource *ares,
     318                 :            :                                          struct resource_win *win)
     319                 :            : {
     320                 :        299 :         struct acpi_resource_extended_address64 *ext_addr;
     321                 :            : 
     322                 :        299 :         win->res.flags = 0;
     323   [ -  -  +  -  :        299 :         if (ares->type != ACPI_RESOURCE_TYPE_EXTENDED_ADDRESS64)
                   -  + ]
     324                 :            :                 return false;
     325                 :            : 
     326                 :          0 :         ext_addr = &ares->data.ext_address64;
     327                 :            : 
     328                 :          0 :         return acpi_decode_space(win, (struct acpi_resource_address *)ext_addr,
     329                 :            :                                  &ext_addr->address);
     330                 :            : }
     331                 :            : EXPORT_SYMBOL_GPL(acpi_dev_resource_ext_address_space);
     332                 :            : 
     333                 :            : /**
     334                 :            :  * acpi_dev_irq_flags - Determine IRQ resource flags.
     335                 :            :  * @triggering: Triggering type as provided by ACPI.
     336                 :            :  * @polarity: Interrupt polarity as provided by ACPI.
     337                 :            :  * @shareable: Whether or not the interrupt is shareable.
     338                 :            :  */
     339                 :         78 : unsigned long acpi_dev_irq_flags(u8 triggering, u8 polarity, u8 shareable)
     340                 :            : {
     341                 :         78 :         unsigned long flags;
     342                 :            : 
     343         [ #  # ]:          0 :         if (triggering == ACPI_LEVEL_SENSITIVE)
     344                 :          0 :                 flags = polarity == ACPI_ACTIVE_LOW ?
     345   [ #  #  #  # ]:          0 :                         IORESOURCE_IRQ_LOWLEVEL : IORESOURCE_IRQ_HIGHLEVEL;
     346                 :            :         else
     347                 :         78 :                 flags = polarity == ACPI_ACTIVE_LOW ?
     348   [ +  -  -  - ]:         78 :                         IORESOURCE_IRQ_LOWEDGE : IORESOURCE_IRQ_HIGHEDGE;
     349                 :            : 
     350   [ -  +  -  - ]:         78 :         if (shareable == ACPI_SHARED)
     351                 :          0 :                 flags |= IORESOURCE_IRQ_SHAREABLE;
     352                 :            : 
     353                 :         78 :         return flags | IORESOURCE_IRQ;
     354                 :            : }
     355                 :            : EXPORT_SYMBOL_GPL(acpi_dev_irq_flags);
     356                 :            : 
     357                 :            : /**
     358                 :            :  * acpi_dev_get_irq_type - Determine irq type.
     359                 :            :  * @triggering: Triggering type as provided by ACPI.
     360                 :            :  * @polarity: Interrupt polarity as provided by ACPI.
     361                 :            :  */
     362                 :          0 : unsigned int acpi_dev_get_irq_type(int triggering, int polarity)
     363                 :            : {
     364   [ #  #  #  # ]:          0 :         switch (polarity) {
     365                 :          0 :         case ACPI_ACTIVE_LOW:
     366                 :          0 :                 return triggering == ACPI_EDGE_SENSITIVE ?
     367         [ #  # ]:          0 :                        IRQ_TYPE_EDGE_FALLING :
     368                 :            :                        IRQ_TYPE_LEVEL_LOW;
     369                 :          0 :         case ACPI_ACTIVE_HIGH:
     370                 :          0 :                 return triggering == ACPI_EDGE_SENSITIVE ?
     371         [ #  # ]:          0 :                        IRQ_TYPE_EDGE_RISING :
     372                 :            :                        IRQ_TYPE_LEVEL_HIGH;
     373                 :          0 :         case ACPI_ACTIVE_BOTH:
     374         [ #  # ]:          0 :                 if (triggering == ACPI_EDGE_SENSITIVE)
     375                 :          0 :                         return IRQ_TYPE_EDGE_BOTH;
     376                 :            :                 /* fall through */
     377                 :            :         default:
     378                 :            :                 return IRQ_TYPE_NONE;
     379                 :            :         }
     380                 :            : }
     381                 :            : EXPORT_SYMBOL_GPL(acpi_dev_get_irq_type);
     382                 :            : 
     383                 :         78 : static void acpi_dev_irqresource_disabled(struct resource *res, u32 gsi)
     384                 :            : {
     385                 :         78 :         res->start = gsi;
     386                 :         78 :         res->end = gsi;
     387                 :         78 :         res->flags = IORESOURCE_IRQ | IORESOURCE_DISABLED | IORESOURCE_UNSET;
     388                 :          0 : }
     389                 :            : 
     390                 :         78 : static void acpi_dev_get_irqresource(struct resource *res, u32 gsi,
     391                 :            :                                      u8 triggering, u8 polarity, u8 shareable,
     392                 :            :                                      bool legacy)
     393                 :            : {
     394                 :         78 :         int irq, p, t;
     395                 :            : 
     396         [ -  + ]:         78 :         if (!valid_IRQ(gsi)) {
     397                 :          0 :                 acpi_dev_irqresource_disabled(res, gsi);
     398                 :          0 :                 return;
     399                 :            :         }
     400                 :            : 
     401                 :            :         /*
     402                 :            :          * In IO-APIC mode, use overridden attribute. Two reasons:
     403                 :            :          * 1. BIOS bug in DSDT
     404                 :            :          * 2. BIOS uses IO-APIC mode Interrupt Source Override
     405                 :            :          *
     406                 :            :          * We do this only if we are dealing with IRQ() or IRQNoFlags()
     407                 :            :          * resource (the legacy ISA resources). With modern ACPI 5 devices
     408                 :            :          * using extended IRQ descriptors we take the IRQ configuration
     409                 :            :          * from _CRS directly.
     410                 :            :          */
     411   [ +  -  +  - ]:         78 :         if (legacy && !acpi_get_override_irq(gsi, &t, &p)) {
     412                 :         78 :                 u8 trig = t ? ACPI_LEVEL_SENSITIVE : ACPI_EDGE_SENSITIVE;
     413                 :         78 :                 u8 pol = p ? ACPI_ACTIVE_LOW : ACPI_ACTIVE_HIGH;
     414                 :            : 
     415         [ -  + ]:         78 :                 if (triggering != trig || polarity != pol) {
     416   [ #  #  #  # ]:          0 :                         pr_warn("ACPI: IRQ %d override to %s, %s\n", gsi,
     417                 :            :                                 t ? "level" : "edge", p ? "low" : "high");
     418                 :          0 :                         triggering = trig;
     419                 :          0 :                         polarity = pol;
     420                 :            :                 }
     421                 :            :         }
     422                 :            : 
     423         [ -  + ]:         78 :         res->flags = acpi_dev_irq_flags(triggering, polarity, shareable);
     424                 :         78 :         irq = acpi_register_gsi(NULL, gsi, triggering, polarity);
     425         [ +  - ]:         78 :         if (irq >= 0) {
     426                 :         78 :                 res->start = irq;
     427                 :         78 :                 res->end = irq;
     428                 :            :         } else {
     429                 :          0 :                 acpi_dev_irqresource_disabled(res, gsi);
     430                 :            :         }
     431                 :            : }
     432                 :            : 
     433                 :            : /**
     434                 :            :  * acpi_dev_resource_interrupt - Extract ACPI interrupt resource information.
     435                 :            :  * @ares: Input ACPI resource object.
     436                 :            :  * @index: Index into the array of GSIs represented by the resource.
     437                 :            :  * @res: Output generic resource object.
     438                 :            :  *
     439                 :            :  * Check if the given ACPI resource object represents an interrupt resource
     440                 :            :  * and @index does not exceed the resource's interrupt count (true is returned
     441                 :            :  * in that case regardless of the results of the other checks)).  If that's the
     442                 :            :  * case, register the GSI corresponding to @index from the array of interrupts
     443                 :            :  * represented by the resource and populate the generic resource object pointed
     444                 :            :  * to by @res accordingly.  If the registration of the GSI is not successful,
     445                 :            :  * IORESOURCE_DISABLED will be set it that object's flags.
     446                 :            :  *
     447                 :            :  * Return:
     448                 :            :  * 1) false with res->flags setting to zero: not the expected resource type
     449                 :            :  * 2) false with IORESOURCE_DISABLED in res->flags: valid unassigned resource
     450                 :            :  * 3) true: valid assigned resource
     451                 :            :  */
     452                 :        377 : bool acpi_dev_resource_interrupt(struct acpi_resource *ares, int index,
     453                 :            :                                  struct resource *res)
     454                 :            : {
     455                 :        377 :         struct acpi_resource_irq *irq;
     456                 :        377 :         struct acpi_resource_extended_irq *ext_irq;
     457                 :            : 
     458      [ +  -  + ]:        377 :         switch (ares->type) {
     459                 :        156 :         case ACPI_RESOURCE_TYPE_IRQ:
     460                 :            :                 /*
     461                 :            :                  * Per spec, only one interrupt per descriptor is allowed in
     462                 :            :                  * _CRS, but some firmware violates this, so parse them all.
     463                 :            :                  */
     464                 :        156 :                 irq = &ares->data.irq;
     465         [ +  + ]:        156 :                 if (index >= irq->interrupt_count) {
     466                 :         78 :                         acpi_dev_irqresource_disabled(res, 0);
     467                 :         78 :                         return false;
     468                 :            :                 }
     469                 :         78 :                 acpi_dev_get_irqresource(res, irq->interrupts[index],
     470                 :         78 :                                          irq->triggering, irq->polarity,
     471                 :         78 :                                          irq->shareable, true);
     472                 :         78 :                 break;
     473                 :          0 :         case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
     474                 :          0 :                 ext_irq = &ares->data.extended_irq;
     475         [ #  # ]:          0 :                 if (index >= ext_irq->interrupt_count) {
     476                 :          0 :                         acpi_dev_irqresource_disabled(res, 0);
     477                 :          0 :                         return false;
     478                 :            :                 }
     479                 :          0 :                 if (is_gsi(ext_irq))
     480                 :          0 :                         acpi_dev_get_irqresource(res, ext_irq->interrupts[index],
     481                 :          0 :                                          ext_irq->triggering, ext_irq->polarity,
     482                 :          0 :                                          ext_irq->shareable, false);
     483                 :            :                 else
     484                 :            :                         acpi_dev_irqresource_disabled(res, 0);
     485                 :            :                 break;
     486                 :        221 :         default:
     487                 :        221 :                 res->flags = 0;
     488                 :        221 :                 return false;
     489                 :            :         }
     490                 :            : 
     491                 :            :         return true;
     492                 :            : }
     493                 :            : EXPORT_SYMBOL_GPL(acpi_dev_resource_interrupt);
     494                 :            : 
     495                 :            : /**
     496                 :            :  * acpi_dev_free_resource_list - Free resource from %acpi_dev_get_resources().
     497                 :            :  * @list: The head of the resource list to free.
     498                 :            :  */
     499                 :        780 : void acpi_dev_free_resource_list(struct list_head *list)
     500                 :            : {
     501                 :        780 :         resource_list_free(list);
     502                 :        780 : }
     503                 :            : EXPORT_SYMBOL_GPL(acpi_dev_free_resource_list);
     504                 :            : 
     505                 :            : struct res_proc_context {
     506                 :            :         struct list_head *list;
     507                 :            :         int (*preproc)(struct acpi_resource *, void *);
     508                 :            :         void *preproc_data;
     509                 :            :         int count;
     510                 :            :         int error;
     511                 :            : };
     512                 :            : 
     513                 :         91 : static acpi_status acpi_dev_new_resource_entry(struct resource_win *win,
     514                 :            :                                                struct res_proc_context *c)
     515                 :            : {
     516                 :         91 :         struct resource_entry *rentry;
     517                 :            : 
     518                 :         91 :         rentry = resource_list_create_entry(NULL, 0);
     519         [ -  + ]:         91 :         if (!rentry) {
     520                 :          0 :                 c->error = -ENOMEM;
     521                 :          0 :                 return AE_NO_MEMORY;
     522                 :            :         }
     523                 :         91 :         *rentry->res = win->res;
     524                 :         91 :         rentry->offset = win->offset;
     525                 :         91 :         resource_list_add_tail(rentry, c->list);
     526                 :         91 :         c->count++;
     527                 :         91 :         return AE_OK;
     528                 :            : }
     529                 :            : 
     530                 :        806 : static acpi_status acpi_dev_process_resource(struct acpi_resource *ares,
     531                 :            :                                              void *context)
     532                 :            : {
     533                 :        806 :         struct res_proc_context *c = context;
     534                 :        806 :         struct resource_win win;
     535                 :        806 :         struct resource *res = &win.res;
     536                 :        806 :         int i;
     537                 :            : 
     538         [ +  + ]:        806 :         if (c->preproc) {
     539                 :        754 :                 int ret;
     540                 :            : 
     541                 :        754 :                 ret = c->preproc(ares, c->preproc_data);
     542         [ -  + ]:        754 :                 if (ret < 0) {
     543                 :          0 :                         c->error = ret;
     544                 :          0 :                         return AE_CTRL_TERMINATE;
     545         [ +  + ]:        754 :                 } else if (ret > 0) {
     546                 :            :                         return AE_OK;
     547                 :            :                 }
     548                 :            :         }
     549                 :            : 
     550                 :        117 :         memset(&win, 0, sizeof(win));
     551                 :            : 
     552         [ +  + ]:        117 :         if (acpi_dev_resource_memory(ares, res)
     553         [ +  + ]:        104 :             || acpi_dev_resource_io(ares, res)
     554         [ +  + ]:         78 :             || acpi_dev_resource_address_space(ares, &win)
     555         [ #  # ]:          0 :             || acpi_dev_resource_ext_address_space(ares, &win))
     556                 :         91 :                 return acpi_dev_new_resource_entry(&win, c);
     557                 :            : 
     558         [ -  + ]:         26 :         for (i = 0; acpi_dev_resource_interrupt(ares, i, res); i++) {
     559                 :          0 :                 acpi_status status;
     560                 :            : 
     561                 :          0 :                 status = acpi_dev_new_resource_entry(&win, c);
     562         [ #  # ]:          0 :                 if (ACPI_FAILURE(status))
     563                 :          0 :                         return status;
     564                 :            :         }
     565                 :            : 
     566                 :            :         return AE_OK;
     567                 :            : }
     568                 :            : 
     569                 :        793 : static int __acpi_dev_get_resources(struct acpi_device *adev,
     570                 :            :                                     struct list_head *list,
     571                 :            :                                     int (*preproc)(struct acpi_resource *, void *),
     572                 :            :                                     void *preproc_data, char *method)
     573                 :            : {
     574                 :        793 :         struct res_proc_context c;
     575                 :        793 :         acpi_status status;
     576                 :            : 
     577   [ +  -  +  +  :        793 :         if (!adev || !adev->handle || !list_empty(list))
                   +  - ]
     578                 :            :                 return -EINVAL;
     579                 :            : 
     580         [ +  + ]:        780 :         if (!acpi_has_method(adev->handle, method))
     581                 :            :                 return 0;
     582                 :            : 
     583                 :        273 :         c.list = list;
     584                 :        273 :         c.preproc = preproc;
     585                 :        273 :         c.preproc_data = preproc_data;
     586                 :        273 :         c.count = 0;
     587                 :        273 :         c.error = 0;
     588                 :        273 :         status = acpi_walk_resources(adev->handle, method,
     589                 :            :                                      acpi_dev_process_resource, &c);
     590         [ -  + ]:        273 :         if (ACPI_FAILURE(status)) {
     591                 :          0 :                 acpi_dev_free_resource_list(list);
     592         [ #  # ]:          0 :                 return c.error ? c.error : -EIO;
     593                 :            :         }
     594                 :            : 
     595                 :        273 :         return c.count;
     596                 :            : }
     597                 :            : 
     598                 :            : /**
     599                 :            :  * acpi_dev_get_resources - Get current resources of a device.
     600                 :            :  * @adev: ACPI device node to get the resources for.
     601                 :            :  * @list: Head of the resultant list of resources (must be empty).
     602                 :            :  * @preproc: The caller's preprocessing routine.
     603                 :            :  * @preproc_data: Pointer passed to the caller's preprocessing routine.
     604                 :            :  *
     605                 :            :  * Evaluate the _CRS method for the given device node and process its output by
     606                 :            :  * (1) executing the @preproc() rountine provided by the caller, passing the
     607                 :            :  * resource pointer and @preproc_data to it as arguments, for each ACPI resource
     608                 :            :  * returned and (2) converting all of the returned ACPI resources into struct
     609                 :            :  * resource objects if possible.  If the return value of @preproc() in step (1)
     610                 :            :  * is different from 0, step (2) is not applied to the given ACPI resource and
     611                 :            :  * if that value is negative, the whole processing is aborted and that value is
     612                 :            :  * returned as the final error code.
     613                 :            :  *
     614                 :            :  * The resultant struct resource objects are put on the list pointed to by
     615                 :            :  * @list, that must be empty initially, as members of struct resource_entry
     616                 :            :  * objects.  Callers of this routine should use %acpi_dev_free_resource_list() to
     617                 :            :  * free that list.
     618                 :            :  *
     619                 :            :  * The number of resources in the output list is returned on success, an error
     620                 :            :  * code reflecting the error condition is returned otherwise.
     621                 :            :  */
     622                 :        793 : int acpi_dev_get_resources(struct acpi_device *adev, struct list_head *list,
     623                 :            :                            int (*preproc)(struct acpi_resource *, void *),
     624                 :            :                            void *preproc_data)
     625                 :            : {
     626                 :        793 :         return __acpi_dev_get_resources(adev, list, preproc, preproc_data,
     627                 :            :                                         METHOD_NAME__CRS);
     628                 :            : }
     629                 :            : EXPORT_SYMBOL_GPL(acpi_dev_get_resources);
     630                 :            : 
     631                 :          0 : static int is_memory(struct acpi_resource *ares, void *not_used)
     632                 :            : {
     633                 :          0 :         struct resource_win win;
     634                 :          0 :         struct resource *res = &win.res;
     635                 :            : 
     636                 :          0 :         memset(&win, 0, sizeof(win));
     637                 :            : 
     638   [ #  #  #  # ]:          0 :         return !(acpi_dev_resource_memory(ares, res)
     639         [ #  # ]:          0 :                || acpi_dev_resource_address_space(ares, &win)
     640                 :            :                || acpi_dev_resource_ext_address_space(ares, &win));
     641                 :            : }
     642                 :            : 
     643                 :            : /**
     644                 :            :  * acpi_dev_get_dma_resources - Get current DMA resources of a device.
     645                 :            :  * @adev: ACPI device node to get the resources for.
     646                 :            :  * @list: Head of the resultant list of resources (must be empty).
     647                 :            :  *
     648                 :            :  * Evaluate the _DMA method for the given device node and process its
     649                 :            :  * output.
     650                 :            :  *
     651                 :            :  * The resultant struct resource objects are put on the list pointed to
     652                 :            :  * by @list, that must be empty initially, as members of struct
     653                 :            :  * resource_entry objects.  Callers of this routine should use
     654                 :            :  * %acpi_dev_free_resource_list() to free that list.
     655                 :            :  *
     656                 :            :  * The number of resources in the output list is returned on success,
     657                 :            :  * an error code reflecting the error condition is returned otherwise.
     658                 :            :  */
     659                 :          0 : int acpi_dev_get_dma_resources(struct acpi_device *adev, struct list_head *list)
     660                 :            : {
     661                 :          0 :         return __acpi_dev_get_resources(adev, list, is_memory, NULL,
     662                 :            :                                         METHOD_NAME__DMA);
     663                 :            : }
     664                 :            : EXPORT_SYMBOL_GPL(acpi_dev_get_dma_resources);
     665                 :            : 
     666                 :            : /**
     667                 :            :  * acpi_dev_filter_resource_type - Filter ACPI resource according to resource
     668                 :            :  *                                 types
     669                 :            :  * @ares: Input ACPI resource object.
     670                 :            :  * @types: Valid resource types of IORESOURCE_XXX
     671                 :            :  *
     672                 :            :  * This is a helper function to support acpi_dev_get_resources(), which filters
     673                 :            :  * ACPI resource objects according to resource types.
     674                 :            :  */
     675                 :         91 : int acpi_dev_filter_resource_type(struct acpi_resource *ares,
     676                 :            :                                   unsigned long types)
     677                 :            : {
     678                 :         91 :         unsigned long type = 0;
     679                 :            : 
     680   [ -  +  -  -  :         91 :         switch (ares->type) {
                -  +  + ]
     681                 :          0 :         case ACPI_RESOURCE_TYPE_MEMORY24:
     682                 :            :         case ACPI_RESOURCE_TYPE_MEMORY32:
     683                 :            :         case ACPI_RESOURCE_TYPE_FIXED_MEMORY32:
     684                 :          0 :                 type = IORESOURCE_MEM;
     685                 :          0 :                 break;
     686                 :         13 :         case ACPI_RESOURCE_TYPE_IO:
     687                 :            :         case ACPI_RESOURCE_TYPE_FIXED_IO:
     688                 :         13 :                 type = IORESOURCE_IO;
     689                 :         13 :                 break;
     690                 :          0 :         case ACPI_RESOURCE_TYPE_IRQ:
     691                 :            :         case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
     692                 :          0 :                 type = IORESOURCE_IRQ;
     693                 :          0 :                 break;
     694                 :          0 :         case ACPI_RESOURCE_TYPE_DMA:
     695                 :            :         case ACPI_RESOURCE_TYPE_FIXED_DMA:
     696                 :          0 :                 type = IORESOURCE_DMA;
     697                 :          0 :                 break;
     698                 :          0 :         case ACPI_RESOURCE_TYPE_GENERIC_REGISTER:
     699                 :          0 :                 type = IORESOURCE_REG;
     700                 :          0 :                 break;
     701                 :         65 :         case ACPI_RESOURCE_TYPE_ADDRESS16:
     702                 :            :         case ACPI_RESOURCE_TYPE_ADDRESS32:
     703                 :            :         case ACPI_RESOURCE_TYPE_ADDRESS64:
     704                 :            :         case ACPI_RESOURCE_TYPE_EXTENDED_ADDRESS64:
     705         [ +  + ]:         65 :                 if (ares->data.address.resource_type == ACPI_MEMORY_RANGE)
     706                 :            :                         type = IORESOURCE_MEM;
     707         [ +  + ]:         39 :                 else if (ares->data.address.resource_type == ACPI_IO_RANGE)
     708                 :            :                         type = IORESOURCE_IO;
     709         [ +  - ]:         13 :                 else if (ares->data.address.resource_type ==
     710                 :            :                          ACPI_BUS_NUMBER_RANGE)
     711                 :         13 :                         type = IORESOURCE_BUS;
     712                 :            :                 break;
     713                 :            :         default:
     714                 :            :                 break;
     715                 :            :         }
     716                 :            : 
     717                 :         91 :         return (type & types) ? 0 : 1;
     718                 :            : }
     719                 :            : EXPORT_SYMBOL_GPL(acpi_dev_filter_resource_type);
     720                 :            : 
     721                 :          0 : static int acpi_dev_consumes_res(struct acpi_device *adev, struct resource *res)
     722                 :            : {
     723                 :          0 :         struct list_head resource_list;
     724                 :          0 :         struct resource_entry *rentry;
     725                 :          0 :         int ret, found = 0;
     726                 :            : 
     727                 :          0 :         INIT_LIST_HEAD(&resource_list);
     728                 :          0 :         ret = acpi_dev_get_resources(adev, &resource_list, NULL, NULL);
     729         [ #  # ]:          0 :         if (ret < 0)
     730                 :            :                 return 0;
     731                 :            : 
     732         [ #  # ]:          0 :         list_for_each_entry(rentry, &resource_list, node) {
     733   [ #  #  #  # ]:          0 :                 if (resource_contains(rentry->res, res)) {
     734                 :            :                         found = 1;
     735                 :            :                         break;
     736                 :            :                 }
     737                 :            : 
     738                 :            :         }
     739                 :            : 
     740                 :          0 :         acpi_dev_free_resource_list(&resource_list);
     741                 :          0 :         return found;
     742                 :            : }
     743                 :            : 
     744                 :          0 : static acpi_status acpi_res_consumer_cb(acpi_handle handle, u32 depth,
     745                 :            :                                          void *context, void **ret)
     746                 :            : {
     747                 :          0 :         struct resource *res = context;
     748                 :          0 :         struct acpi_device **consumer = (struct acpi_device **) ret;
     749                 :          0 :         struct acpi_device *adev;
     750                 :            : 
     751         [ #  # ]:          0 :         if (acpi_bus_get_device(handle, &adev))
     752                 :            :                 return AE_OK;
     753                 :            : 
     754         [ #  # ]:          0 :         if (acpi_dev_consumes_res(adev, res)) {
     755                 :          0 :                 *consumer = adev;
     756                 :          0 :                 return AE_CTRL_TERMINATE;
     757                 :            :         }
     758                 :            : 
     759                 :            :         return AE_OK;
     760                 :            : }
     761                 :            : 
     762                 :            : /**
     763                 :            :  * acpi_resource_consumer - Find the ACPI device that consumes @res.
     764                 :            :  * @res: Resource to search for.
     765                 :            :  *
     766                 :            :  * Search the current resource settings (_CRS) of every ACPI device node
     767                 :            :  * for @res.  If we find an ACPI device whose _CRS includes @res, return
     768                 :            :  * it.  Otherwise, return NULL.
     769                 :            :  */
     770                 :          0 : struct acpi_device *acpi_resource_consumer(struct resource *res)
     771                 :            : {
     772                 :          0 :         struct acpi_device *consumer = NULL;
     773                 :            : 
     774                 :          0 :         acpi_get_devices(NULL, acpi_res_consumer_cb, res, (void **) &consumer);
     775                 :          0 :         return consumer;
     776                 :            : }

Generated by: LCOV version 1.14