Coverage Report

Created: 2025-04-11 06:16

/src/elfutils/libelf/elf_getphdrnum.c
Line
Count
Source (jump to first uncovered line)
1
/* Return number of program headers in the ELF file.
2
   Copyright (C) 2010, 2014, 2015, 2016 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 <assert.h>
34
#include <gelf.h>
35
#include <stddef.h>
36
37
#include "libelfP.h"
38
39
40
int
41
internal_function
42
__elf_getphdrnum_rdlock (Elf *elf, size_t *dst)
43
8.83M
{
44
8.83M
 if (unlikely (elf->state.elf64.ehdr == NULL))
45
0
   {
46
     /* Maybe no ELF header was created yet.  */
47
0
     *dst = 0;
48
0
     __libelf_seterrno (ELF_E_WRONG_ORDER_EHDR);
49
0
     return -1;
50
0
   }
51
52
8.83M
 *dst = (elf->class == ELFCLASS32
53
8.83M
   ? elf->state.elf32.ehdr->e_phnum
54
8.83M
   : elf->state.elf64.ehdr->e_phnum);
55
56
8.83M
 if (*dst == PN_XNUM)
57
4.75M
   {
58
4.75M
     const Elf_ScnList *const scns = (elf->class == ELFCLASS32
59
4.75M
              ? &elf->state.elf32.scns
60
4.75M
              : &elf->state.elf64.scns);
61
62
     /* If there are no section headers, perhaps this is really just 65536
63
  written without PN_XNUM support.  Either that or it's bad data.  */
64
65
4.75M
     if (elf->class == ELFCLASS32)
66
3.58M
       {
67
3.58M
   if (likely (scns->cnt > 0))
68
1.91M
     {
69
1.91M
       Elf_Scn *scn = &elf->state.elf32.scns.data[0];
70
1.91M
       Elf32_Shdr *shdr = scn->shdr.e32 ?: __elf32_getshdr_rdlock (scn);
71
1.91M
       if (shdr)
72
1.91M
         *dst = shdr->sh_info;
73
79
     }
74
3.58M
       }
75
1.17M
     else
76
1.17M
       {
77
1.17M
   if (likely (scns->cnt > 0))
78
587k
     {
79
587k
       Elf_Scn *scn = &elf->state.elf64.scns.data[0];
80
587k
       Elf64_Shdr *shdr = scn->shdr.e64 ?: __elf64_getshdr_rdlock (scn);
81
587k
       if (shdr)
82
587k
         *dst = shdr->sh_info;
83
128
     }
84
1.17M
       }
85
4.75M
   }
86
87
8.83M
 return 0;
88
8.83M
}
89
90
int
91
internal_function
92
__elf_getphdrnum_chk_rdlock (Elf *elf, size_t *dst)
93
8.83M
{
94
8.83M
  int result = __elf_getphdrnum_rdlock (elf, dst);
95
96
  /* If the phdrs haven't been created or read in yet then do some
97
     sanity checking to make sure phnum and phoff are consistent.  */
98
8.83M
  if (elf->state.elf.phdr == NULL)
99
12.0k
    {
100
12.0k
      Elf64_Off off = (elf->class == ELFCLASS32
101
12.0k
           ? elf->state.elf32.ehdr->e_phoff
102
12.0k
           : elf->state.elf64.ehdr->e_phoff);
103
12.0k
      if (unlikely (off == 0))
104
7.87k
  {
105
7.87k
    *dst = 0;
106
7.87k
    return result;
107
7.87k
  }
108
109
4.17k
      if (unlikely (off >= elf->maximum_size))
110
483
  {
111
483
    __libelf_seterrno (ELF_E_INVALID_DATA);
112
483
    return -1;
113
483
  }
114
115
      /* Check for too many sections.  */
116
3.69k
      size_t phdr_size = (elf->class == ELFCLASS32
117
3.69k
        ? sizeof (Elf32_Phdr) : sizeof (Elf64_Phdr));
118
3.69k
      if (unlikely (*dst > SIZE_MAX / phdr_size))
119
0
  {
120
0
    __libelf_seterrno (ELF_E_INVALID_DATA);
121
0
    return -1;
122
0
  }
123
124
      /* Truncated file?  Don't return more than can be indexed.  */
125
3.69k
      if (unlikely (elf->maximum_size - off < *dst * phdr_size))
126
767
  *dst = (elf->maximum_size - off) / phdr_size;
127
3.69k
    }
128
129
8.82M
  return result;
130
8.83M
}
131
132
int
133
elf_getphdrnum (Elf *elf, size_t *dst)
134
12.9k
{
135
12.9k
  int result;
136
137
12.9k
  if (elf == NULL)
138
0
    return -1;
139
140
12.9k
  if (unlikely (elf->kind != ELF_K_ELF))
141
0
    {
142
0
      __libelf_seterrno (ELF_E_INVALID_HANDLE);
143
0
      return -1;
144
0
    }
145
146
12.9k
  rwlock_rdlock (elf->lock);
147
12.9k
  result = __elf_getphdrnum_chk_rdlock (elf, dst);
148
12.9k
  rwlock_unlock (elf->lock);
149
150
12.9k
  return result;
151
12.9k
}