LCOV - code coverage report
Current view: top level - libdwfl - cu.c (source / functions) Coverage Total Hit
Test: elfutils-0.192 Lines: 86.5 % 126 109
Test Date: 2024-10-18 15:14:37 Functions: 87.5 % 8 7
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed Branches: 63.1 % 84 53

             Branch data     Line data    Source code
       1                 :             : /* Keeping track of DWARF compilation units in libdwfl.
       2                 :             :    Copyright (C) 2005-2010, 2015, 2016, 2017 Red Hat, Inc.
       3                 :             :    This file is part of elfutils.
       4                 :             : 
       5                 :             :    This file is free software; you can redistribute it and/or modify
       6                 :             :    it under the terms of either
       7                 :             : 
       8                 :             :      * the GNU Lesser General Public License as published by the Free
       9                 :             :        Software Foundation; either version 3 of the License, or (at
      10                 :             :        your option) any later version
      11                 :             : 
      12                 :             :    or
      13                 :             : 
      14                 :             :      * the GNU General Public License as published by the Free
      15                 :             :        Software Foundation; either version 2 of the License, or (at
      16                 :             :        your option) any later version
      17                 :             : 
      18                 :             :    or both in parallel, as here.
      19                 :             : 
      20                 :             :    elfutils is distributed in the hope that it will be useful, but
      21                 :             :    WITHOUT ANY WARRANTY; without even the implied warranty of
      22                 :             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
      23                 :             :    General Public License for more details.
      24                 :             : 
      25                 :             :    You should have received copies of the GNU General Public License and
      26                 :             :    the GNU Lesser General Public License along with this program.  If
      27                 :             :    not, see <http://www.gnu.org/licenses/>.  */
      28                 :             : 
      29                 :             : #ifdef HAVE_CONFIG_H
      30                 :             : # include <config.h>
      31                 :             : #endif
      32                 :             : 
      33                 :             : #include "eu-search.h"
      34                 :             : #include "libdwflP.h"
      35                 :             : #include "libdwP.h"
      36                 :             : #include "memory-access.h"
      37                 :             : 
      38                 :             : 
      39                 :             : static inline Dwarf_Arange *
      40                 :        1056 : dwar (Dwfl_Module *mod, unsigned int idx)
      41                 :             : {
      42                 :        1056 :   return &mod->dw->dieranges->info[mod->aranges[idx].arange];
      43                 :             : }
      44                 :             : 
      45                 :             : 
      46                 :             : static Dwfl_Error
      47                 :         904 : addrarange (Dwfl_Module *mod, Dwarf_Addr addr, struct dwfl_arange **arange)
      48                 :             : {
      49         [ +  + ]:         904 :   if (mod->aranges == NULL)
      50                 :             :     {
      51                 :         126 :       struct dwfl_arange *aranges = NULL;
      52                 :         126 :       Dwarf_Aranges *dwaranges = NULL;
      53                 :         126 :       size_t naranges;
      54         [ +  - ]:         126 :       if (__libdw_getdieranges (mod->dw, &dwaranges, &naranges) != 0)
      55                 :           0 :         return DWFL_E_LIBDW;
      56                 :             : 
      57                 :             :       /* If the module has no aranges (when no code is included) we
      58                 :             :          allocate nothing.  */
      59         [ +  - ]:         126 :       if (naranges != 0)
      60                 :             :         {
      61                 :         126 :           aranges = malloc (naranges * sizeof *aranges);
      62         [ +  - ]:         126 :           if (unlikely (aranges == NULL))
      63                 :             :             return DWFL_E_NOMEM;
      64                 :             : 
      65                 :             :           /* libdw has sorted its list by address, which is how we want it.
      66                 :             :              But the sorted list is full of not-quite-contiguous runs pointing
      67                 :             :              to the same CU.  We don't care about the little gaps inside the
      68                 :             :              module, we'll consider them part of the surrounding CU anyway.
      69                 :             :              Collect our own array with just one record for each run of ranges
      70                 :             :              pointing to one CU.  */
      71                 :             : 
      72                 :         126 :           naranges = 0;
      73                 :         126 :           Dwarf_Off lastcu = 0;
      74         [ +  + ]:         298 :           for (size_t i = 0; i < dwaranges->naranges; ++i)
      75   [ +  +  +  + ]:         172 :             if (i == 0 || dwaranges->info[i].offset != lastcu)
      76                 :             :               {
      77                 :         154 :                 aranges[naranges].arange = i;
      78                 :         154 :                 aranges[naranges].cu = NULL;
      79                 :         154 :                 ++naranges;
      80                 :         154 :                 lastcu = dwaranges->info[i].offset;
      81                 :             :               }
      82                 :             :         }
      83                 :             : 
      84                 :             :       /* Store the final array, which is probably much smaller than before.  */
      85                 :         126 :       mod->naranges = naranges;
      86         [ +  - ]:         126 :       if (naranges > 0)
      87                 :         126 :         mod->aranges = (realloc (aranges, naranges * sizeof aranges[0])
      88         [ -  + ]:         126 :                         ?: aranges);
      89         [ #  # ]:           0 :       else if (aranges != NULL)
      90                 :           0 :         free (aranges);
      91                 :         126 :       mod->lazycu += naranges;
      92                 :             :     }
      93                 :             : 
      94                 :             :   /* The address must be inside the module to begin with.  */
      95                 :         904 :   addr = dwfl_deadjust_dwarf_addr (mod, addr);
      96                 :             : 
      97                 :             :   /* The ranges are sorted by address, so we can use binary search.  */
      98                 :         904 :   size_t l = 0, u = mod->naranges;
      99         [ +  - ]:         980 :   while (l < u)
     100                 :             :     {
     101                 :         980 :       size_t idx = (l + u) / 2;
     102                 :         980 :       Dwarf_Addr start = dwar (mod, idx)->addr;
     103         [ -  + ]:         980 :       if (addr < start)
     104                 :             :         {
     105                 :           0 :           u = idx;
     106                 :           0 :           continue;
     107                 :             :         }
     108         [ +  + ]:         980 :       else if (addr > start)
     109                 :             :         {
     110         [ +  + ]:         744 :           if (idx + 1 < mod->naranges)
     111                 :             :             {
     112         [ +  - ]:          76 :               if (addr >= dwar (mod, idx + 1)->addr)
     113                 :             :                 {
     114                 :          76 :                   l = idx + 1;
     115                 :          76 :                   continue;
     116                 :             :                 }
     117                 :             :             }
     118                 :             :           else
     119                 :             :             {
     120                 :             :               /* It might be in the last range.  */
     121                 :         668 :               const Dwarf_Arange *last
     122                 :         668 :                 = &mod->dw->dieranges->info[mod->dw->dieranges->naranges - 1];
     123         [ +  + ]:         668 :               if (addr > last->addr + last->length)
     124                 :             :                 break;
     125                 :             :             }
     126                 :             :         }
     127                 :             : 
     128                 :         892 :       *arange = &mod->aranges[idx];
     129                 :         892 :       return DWFL_E_NOERROR;
     130                 :             :     }
     131                 :             : 
     132                 :             :   return DWFL_E_ADDR_OUTOFRANGE;
     133                 :             : }
     134                 :             : 
     135                 :             : 
     136                 :             : static void
     137                 :           0 : nofree (void *arg)
     138                 :             : {
     139                 :           0 :   struct dwfl_cu *cu = arg;
     140         [ #  # ]:           0 :   if (cu == (void *) -1l)
     141                 :             :     return;
     142                 :             : 
     143         [ #  # ]:           0 :   assert (cu->mod->lazycu == 0);
     144                 :             : }
     145                 :             : 
     146                 :             : /* One reason fewer to keep the lazy lookup table for CUs.  */
     147                 :             : static inline void
     148                 :         132 : less_lazy (Dwfl_Module *mod)
     149                 :             : {
     150         [ -  + ]:         132 :   if (--mod->lazycu > 0)
     151                 :             :     return;
     152                 :             : 
     153                 :             :   /* We know about all the CUs now, we don't need this table.  */
     154                 :           0 :   eu_tdestroy (&mod->lazy_cu_tree, nofree);
     155                 :             : }
     156                 :             : 
     157                 :             : static inline Dwarf_Off
     158                 :      120624 : cudie_offset (const struct dwfl_cu *cu)
     159                 :             : {
     160                 :      120624 :   return __libdw_first_die_off_from_cu (cu->die.cu);
     161                 :             : }
     162                 :             : 
     163                 :             : static int
     164                 :      120624 : compare_cukey (const void *a, const void *b)
     165                 :             : {
     166                 :      120624 :   Dwarf_Off a_off = cudie_offset (a);
     167                 :      120624 :   Dwarf_Off b_off = cudie_offset (b);
     168         [ +  + ]:      120624 :   return (a_off < b_off) ? -1 : ((a_off > b_off) ? 1 : 0);
     169                 :             : }
     170                 :             : 
     171                 :             : /* Intern the CU if necessary.  */
     172                 :             : static Dwfl_Error
     173                 :       12454 : intern_cu (Dwfl_Module *mod, Dwarf_Off cuoff, struct dwfl_cu **result)
     174                 :             : {
     175         [ -  + ]:       12454 :   if (unlikely (cuoff + 4 >= mod->dw->sectiondata[IDX_debug_info]->d_size))
     176                 :             :     {
     177         [ #  # ]:           0 :       if (likely (mod->lazycu == 1))
     178                 :             :         {
     179                 :             :           /* This is the EOF marker.  Now we have interned all the CUs.
     180                 :             :              One increment in MOD->lazycu counts not having hit EOF yet.  */
     181                 :           0 :           *result = (void *) -1;
     182                 :           0 :           less_lazy (mod);
     183                 :           0 :           return DWFL_E_NOERROR;
     184                 :             :         }
     185                 :             :       else
     186                 :             :         {
     187                 :             :           /* Unexpected EOF, most likely a bogus aranges.  */
     188                 :             :           return (DWFL_E (LIBDW, DWARF_E_INVALID_DWARF));
     189                 :             :         }
     190                 :             :     }
     191                 :             : 
     192                 :             :   /* Make sure the cuoff points to a real DIE.  */
     193                 :       12454 :   Dwarf_Die cudie;
     194                 :       12454 :   Dwarf_Die *die = INTUSE(dwarf_offdie) (mod->dw, cuoff, &cudie);
     195         [ +  - ]:       12454 :   if (die == NULL)
     196                 :             :     return DWFL_E_LIBDW;
     197                 :             : 
     198                 :       12454 :   struct dwfl_cu key;
     199                 :       12454 :   key.die.cu = die->cu;
     200                 :       12454 :   struct dwfl_cu **found = eu_tsearch (&key, &mod->lazy_cu_tree,
     201                 :             :                                        &compare_cukey);
     202         [ +  - ]:       12454 :   if (unlikely (found == NULL))
     203                 :             :     return DWFL_E_NOMEM;
     204                 :             : 
     205   [ +  +  -  + ]:       12454 :   if (*found == &key || *found == NULL)
     206                 :             :     {
     207                 :             :       /* This is a new entry, meaning we haven't looked at this CU.  */
     208                 :             : 
     209                 :       12452 :       *found = NULL;
     210                 :             : 
     211                 :       12452 :       struct dwfl_cu *cu = malloc (sizeof *cu);
     212         [ +  - ]:       12452 :       if (unlikely (cu == NULL))
     213                 :             :         return DWFL_E_NOMEM;
     214                 :             : 
     215                 :       12452 :       cu->mod = mod;
     216                 :       12452 :       cu->next = NULL;
     217                 :       12452 :       cu->lines = NULL;
     218                 :       12452 :       cu->die = cudie;
     219                 :             : 
     220                 :       12452 :       struct dwfl_cu **newvec = realloc (mod->cu, ((mod->ncu + 1)
     221                 :             :                                                    * sizeof (mod->cu[0])));
     222         [ -  + ]:       12452 :       if (newvec == NULL)
     223                 :             :         {
     224                 :           0 :           free (cu);
     225                 :           0 :           return DWFL_E_NOMEM;
     226                 :             :         }
     227                 :       12452 :       mod->cu = newvec;
     228                 :             : 
     229                 :       12452 :       mod->cu[mod->ncu++] = cu;
     230         [ +  + ]:       12452 :       if (cu->die.cu->start == 0)
     231                 :         286 :         mod->first_cu = cu;
     232                 :             : 
     233                 :       12452 :       *found = cu;
     234                 :             :     }
     235                 :             : 
     236                 :       12454 :   *result = *found;
     237                 :       12454 :   return DWFL_E_NOERROR;
     238                 :             : }
     239                 :             : 
     240                 :             : 
     241                 :             : /* Traverse all the CUs in the module.  */
     242                 :             : 
     243                 :             : Dwfl_Error
     244                 :             : internal_function
     245                 :       12536 : __libdwfl_nextcu (Dwfl_Module *mod, struct dwfl_cu *lastcu,
     246                 :             :                   struct dwfl_cu **cu)
     247                 :             : {
     248                 :       12536 :   Dwarf_Off cuoff;
     249                 :       12536 :   struct dwfl_cu **nextp;
     250                 :             : 
     251         [ +  + ]:       12536 :   if (lastcu == NULL)
     252                 :             :     {
     253                 :             :       /* Start the traversal.  */
     254                 :         192 :       cuoff = 0;
     255                 :         192 :       nextp = &mod->first_cu;
     256                 :             :     }
     257                 :             :   else
     258                 :             :     {
     259                 :             :       /* Continue following LASTCU.  */
     260                 :       12344 :       cuoff = lastcu->die.cu->end;
     261                 :       12344 :       nextp = &lastcu->next;
     262                 :             :     }
     263                 :             : 
     264         [ +  + ]:       12536 :   if (*nextp == NULL)
     265                 :             :     {
     266                 :       12514 :       size_t cuhdrsz;
     267                 :       12514 :       Dwarf_Off nextoff;
     268                 :       12514 :       int end = INTUSE(dwarf_nextcu) (mod->dw, cuoff, &nextoff, &cuhdrsz,
     269                 :             :                                       NULL, NULL, NULL);
     270         [ +  - ]:       12514 :       if (end < 0)
     271                 :         192 :         return DWFL_E_LIBDW;
     272         [ +  + ]:       12514 :       if (end > 0)
     273                 :             :         {
     274                 :         192 :           *cu = NULL;
     275                 :         192 :           return DWFL_E_NOERROR;
     276                 :             :         }
     277                 :             : 
     278                 :       12322 :       Dwfl_Error result = intern_cu (mod, cuoff + cuhdrsz, nextp);
     279         [ +  - ]:       12322 :       if (result != DWFL_E_NOERROR)
     280                 :             :         return result;
     281                 :             : 
     282         [ +  - ]:       12322 :       if (*nextp != (void *) -1
     283   [ +  -  -  + ]:       12322 :           && (*nextp)->next == NULL && nextoff == (Dwarf_Off) -1l)
     284                 :           0 :         (*nextp)->next = (void *) -1l;
     285                 :             :     }
     286                 :             : 
     287         [ +  - ]:       12344 :   *cu = *nextp == (void *) -1l ? NULL : *nextp;
     288                 :       12344 :   return DWFL_E_NOERROR;
     289                 :             : }
     290                 :             : 
     291                 :             : 
     292                 :             : /* Intern the CU arange points to, if necessary.  */
     293                 :             : 
     294                 :             : static Dwfl_Error
     295                 :         892 : arangecu (Dwfl_Module *mod, struct dwfl_arange *arange, struct dwfl_cu **cu)
     296                 :             : {
     297         [ +  + ]:         892 :   if (arange->cu == NULL)
     298                 :             :     {
     299                 :         132 :       const Dwarf_Arange *dwarange = &mod->dw->dieranges->info[arange->arange];
     300                 :         132 :       Dwfl_Error result = intern_cu (mod, dwarange->offset, &arange->cu);
     301         [ +  - ]:         132 :       if (result != DWFL_E_NOERROR)
     302                 :             :         return result;
     303         [ -  + ]:         132 :       assert (arange->cu != NULL && arange->cu != (void *) -1l);
     304                 :         132 :       less_lazy (mod);          /* Each arange with null ->cu counts once.  */
     305                 :             :     }
     306                 :             : 
     307                 :         892 :   *cu = arange->cu;
     308                 :         892 :   return DWFL_E_NOERROR;
     309                 :             : }
     310                 :             : 
     311                 :             : Dwfl_Error
     312                 :             : internal_function
     313                 :         904 : __libdwfl_addrcu (Dwfl_Module *mod, Dwarf_Addr addr, struct dwfl_cu **cu)
     314                 :             : {
     315                 :         904 :   struct dwfl_arange *arange;
     316         [ +  + ]:         904 :   return addrarange (mod, addr, &arange) ?: arangecu (mod, arange, cu);
     317                 :             : }
        

Generated by: LCOV version 2.0-1