Coverage Report

Created: 2025-08-03 07:05

/src/elfutils/libelf/elf_getshdrstrndx.c
Line
Count
Source (jump to first uncovered line)
1
/* Return section index of section header string table.
2
   Copyright (C) 2002, 2005, 2009, 2014, 2015 Red Hat, Inc.
3
   This file is part of elfutils.
4
   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
5
6
   This file is free software; you can redistribute it and/or modify
7
   it under the terms of either
8
9
     * the GNU Lesser General Public License as published by the Free
10
       Software Foundation; either version 3 of the License, or (at
11
       your option) any later version
12
13
   or
14
15
     * the GNU General Public License as published by the Free
16
       Software Foundation; either version 2 of the License, or (at
17
       your option) any later version
18
19
   or both in parallel, as here.
20
21
   elfutils is distributed in the hope that it will be useful, but
22
   WITHOUT ANY WARRANTY; without even the implied warranty of
23
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24
   General Public License for more details.
25
26
   You should have received copies of the GNU General Public License and
27
   the GNU Lesser General Public License along with this program.  If
28
   not, see <http://www.gnu.org/licenses/>.  */
29
30
#ifdef HAVE_CONFIG_H
31
# include <config.h>
32
#endif
33
34
#include <assert.h>
35
#include <errno.h>
36
#include <gelf.h>
37
#include <stddef.h>
38
39
#include "libelfP.h"
40
#include "common.h"
41
42
43
int
44
elf_getshdrstrndx (Elf *elf, size_t *dst)
45
11.8k
{
46
11.8k
  int result = 0;
47
48
11.8k
  if (elf == NULL)
49
0
    return -1;
50
51
11.8k
  if (unlikely (elf->kind != ELF_K_ELF))
52
292
    {
53
292
      __libelf_seterrno (ELF_E_INVALID_HANDLE);
54
292
      return -1;
55
292
    }
56
57
11.5k
  rwlock_rdlock (elf->lock);
58
59
  /* We rely here on the fact that the `elf' element is a common prefix
60
     of `elf32' and `elf64'.  */
61
11.5k
  assert (offsetof (struct Elf, state.elf.ehdr)
62
11.5k
    == offsetof (struct Elf, state.elf32.ehdr));
63
11.5k
  assert (sizeof (elf->state.elf.ehdr)
64
11.5k
    == sizeof (elf->state.elf32.ehdr));
65
11.5k
  assert (offsetof (struct Elf, state.elf.ehdr)
66
11.5k
    == offsetof (struct Elf, state.elf64.ehdr));
67
11.5k
  assert (sizeof (elf->state.elf.ehdr)
68
11.5k
    == sizeof (elf->state.elf64.ehdr));
69
70
11.5k
  if (unlikely (elf->state.elf.ehdr == NULL))
71
0
    {
72
0
      __libelf_seterrno (ELF_E_WRONG_ORDER_EHDR);
73
0
      result = -1;
74
0
    }
75
11.5k
  else
76
11.5k
    {
77
11.5k
      Elf32_Word num;
78
79
11.5k
      num = (elf->class == ELFCLASS32
80
11.5k
       ? elf->state.elf32.ehdr->e_shstrndx
81
11.5k
       : elf->state.elf64.ehdr->e_shstrndx);
82
83
      /* Determine whether the index is too big to fit in the ELF
84
   header.  */
85
11.5k
      if (unlikely (num == SHN_XINDEX))
86
262
  {
87
    /* Yes.  Search the zeroth section header.  */
88
262
    if (elf->class == ELFCLASS32)
89
156
      {
90
156
        size_t offset;
91
156
        if (unlikely (elf->state.elf32.scns.cnt == 0))
92
28
    {
93
      /* Cannot use SHN_XINDEX without section headers.  */
94
28
      __libelf_seterrno (ELF_E_INVALID_SECTION_HEADER);
95
28
      result = -1;
96
28
      goto out;
97
28
    }
98
99
128
        if (elf->state.elf32.scns.data[0].shdr.e32 != NULL)
100
0
    {
101
0
      num = elf->state.elf32.scns.data[0].shdr.e32->sh_link;
102
0
      goto success;
103
0
    }
104
105
128
        offset = elf->state.elf32.ehdr->e_shoff;
106
107
128
        if (elf->map_address != NULL
108
128
      && elf->state.elf32.ehdr->e_ident[EI_DATA] == MY_ELFDATA
109
128
      && (ALLOW_UNALIGNED
110
0
          || (((size_t) ((char *) elf->map_address
111
0
         + elf->start_offset + offset))
112
0
        & (__alignof__ (Elf32_Shdr) - 1)) == 0))
113
0
    {
114
      /* First see whether the information in the ELF header is
115
         valid and it does not ask for too much.  */
116
0
      if (unlikely (elf->maximum_size - offset
117
0
        < sizeof (Elf32_Shdr)))
118
0
        {
119
          /* Something is wrong.  */
120
0
          __libelf_seterrno (ELF_E_INVALID_SECTION_HEADER);
121
0
          result = -1;
122
0
          goto out;
123
0
        }
124
125
      /* We can directly access the memory.  */
126
0
      num = ((Elf32_Shdr *) (elf->map_address + elf->start_offset
127
0
           + offset))->sh_link;
128
0
    }
129
128
        else
130
128
    {
131
      /* We avoid reading in all the section headers.  Just read
132
         the first one.  */
133
128
      Elf32_Shdr shdr_mem;
134
128
      ssize_t r;
135
136
128
      if (unlikely ((r = pread_retry (elf->fildes, &shdr_mem,
137
128
              sizeof (Elf32_Shdr), offset))
138
128
        != sizeof (Elf32_Shdr)))
139
0
        {
140
          /* We must be able to read this ELF section header.  */
141
0
          if (r < 0)
142
0
      __libelf_seterrno (ELF_E_INVALID_FILE);
143
0
          else
144
0
      __libelf_seterrno (ELF_E_INVALID_ELF);
145
0
          result = -1;
146
0
          goto out;
147
0
        }
148
149
128
      if (elf->state.elf32.ehdr->e_ident[EI_DATA] != MY_ELFDATA)
150
66
        CONVERT (shdr_mem.sh_link);
151
128
      num = shdr_mem.sh_link;
152
128
    }
153
128
      }
154
106
    else
155
106
      {
156
106
        if (unlikely (elf->state.elf64.scns.cnt == 0))
157
56
    {
158
      /* Cannot use SHN_XINDEX without section headers.  */
159
56
      __libelf_seterrno (ELF_E_INVALID_SECTION_HEADER);
160
56
      result = -1;
161
56
      goto out;
162
56
    }
163
164
50
        if (elf->state.elf64.scns.data[0].shdr.e64 != NULL)
165
0
    {
166
0
      num = elf->state.elf64.scns.data[0].shdr.e64->sh_link;
167
0
      goto success;
168
0
    }
169
170
50
        size_t offset = elf->state.elf64.ehdr->e_shoff;
171
172
50
        if (elf->map_address != NULL
173
50
      && elf->state.elf64.ehdr->e_ident[EI_DATA] == MY_ELFDATA
174
50
      && (ALLOW_UNALIGNED
175
0
          || (((size_t) ((char *) elf->map_address
176
0
         + elf->start_offset + offset))
177
0
        & (__alignof__ (Elf64_Shdr) - 1)) == 0))
178
0
    {
179
      /* First see whether the information in the ELF header is
180
         valid and it does not ask for too much.  */
181
0
      if (unlikely (elf->maximum_size - offset
182
0
        < sizeof (Elf64_Shdr)))
183
0
        {
184
          /* Something is wrong.  */
185
0
          __libelf_seterrno (ELF_E_INVALID_SECTION_HEADER);
186
0
          result = -1;
187
0
          goto out;
188
0
        }
189
190
      /* We can directly access the memory.  */
191
0
      num = ((Elf64_Shdr *) (elf->map_address + elf->start_offset
192
0
           + offset))->sh_link;
193
0
    }
194
50
        else
195
50
    {
196
      /* We avoid reading in all the section headers.  Just read
197
         the first one.  */
198
50
      Elf64_Shdr shdr_mem;
199
50
      ssize_t r;
200
201
50
      if (unlikely ((r = pread_retry (elf->fildes, &shdr_mem,
202
50
              sizeof (Elf64_Shdr), offset))
203
50
        != sizeof (Elf64_Shdr)))
204
0
        {
205
          /* We must be able to read this ELF section header.  */
206
0
          if (r < 0)
207
0
      __libelf_seterrno (ELF_E_INVALID_FILE);
208
0
          else
209
0
      __libelf_seterrno (ELF_E_INVALID_ELF);
210
0
          result = -1;
211
0
          goto out;
212
0
        }
213
214
50
      if (elf->state.elf64.ehdr->e_ident[EI_DATA] != MY_ELFDATA)
215
12
        CONVERT (shdr_mem.sh_link);
216
50
      num = shdr_mem.sh_link;
217
50
    }
218
50
      }
219
262
  }
220
221
      /* Store the result.  */
222
11.4k
    success:
223
11.4k
      *dst = num;
224
11.4k
    }
225
226
11.5k
 out:
227
11.5k
  rwlock_unlock (elf->lock);
228
229
11.5k
  return result;
230
11.5k
}
231
INTDEF(elf_getshdrstrndx)
232
/* Alias for the deprecated name.  */
233
strong_alias (elf_getshdrstrndx, elf_getshstrndx)