xref: /aosp_15_r20/external/elfutils/libelf/elf32_getphdr.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Get ELF program header table.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 1998-2010, 2014, 2015 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker    This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker    Written by Ulrich Drepper <[email protected]>, 1998.
5*7304104dSAndroid Build Coastguard Worker 
6*7304104dSAndroid Build Coastguard Worker    This file is free software; you can redistribute it and/or modify
7*7304104dSAndroid Build Coastguard Worker    it under the terms of either
8*7304104dSAndroid Build Coastguard Worker 
9*7304104dSAndroid Build Coastguard Worker      * the GNU Lesser General Public License as published by the Free
10*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 3 of the License, or (at
11*7304104dSAndroid Build Coastguard Worker        your option) any later version
12*7304104dSAndroid Build Coastguard Worker 
13*7304104dSAndroid Build Coastguard Worker    or
14*7304104dSAndroid Build Coastguard Worker 
15*7304104dSAndroid Build Coastguard Worker      * the GNU General Public License as published by the Free
16*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 2 of the License, or (at
17*7304104dSAndroid Build Coastguard Worker        your option) any later version
18*7304104dSAndroid Build Coastguard Worker 
19*7304104dSAndroid Build Coastguard Worker    or both in parallel, as here.
20*7304104dSAndroid Build Coastguard Worker 
21*7304104dSAndroid Build Coastguard Worker    elfutils is distributed in the hope that it will be useful, but
22*7304104dSAndroid Build Coastguard Worker    WITHOUT ANY WARRANTY; without even the implied warranty of
23*7304104dSAndroid Build Coastguard Worker    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24*7304104dSAndroid Build Coastguard Worker    General Public License for more details.
25*7304104dSAndroid Build Coastguard Worker 
26*7304104dSAndroid Build Coastguard Worker    You should have received copies of the GNU General Public License and
27*7304104dSAndroid Build Coastguard Worker    the GNU Lesser General Public License along with this program.  If
28*7304104dSAndroid Build Coastguard Worker    not, see <http://www.gnu.org/licenses/>.  */
29*7304104dSAndroid Build Coastguard Worker 
30*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
31*7304104dSAndroid Build Coastguard Worker # include <config.h>
32*7304104dSAndroid Build Coastguard Worker #endif
33*7304104dSAndroid Build Coastguard Worker 
34*7304104dSAndroid Build Coastguard Worker #include <errno.h>
35*7304104dSAndroid Build Coastguard Worker #include <stdbool.h>
36*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
37*7304104dSAndroid Build Coastguard Worker #include <assert.h>
38*7304104dSAndroid Build Coastguard Worker 
39*7304104dSAndroid Build Coastguard Worker #include "libelfP.h"
40*7304104dSAndroid Build Coastguard Worker #include "common.h"
41*7304104dSAndroid Build Coastguard Worker 
42*7304104dSAndroid Build Coastguard Worker #ifndef LIBELFBITS
43*7304104dSAndroid Build Coastguard Worker # define LIBELFBITS 32
44*7304104dSAndroid Build Coastguard Worker #endif
45*7304104dSAndroid Build Coastguard Worker 
ElfW2(LIBELFBITS,Phdr)46*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *
47*7304104dSAndroid Build Coastguard Worker __elfw2(LIBELFBITS,getphdr_wrlock) (Elf *elf)
48*7304104dSAndroid Build Coastguard Worker {
49*7304104dSAndroid Build Coastguard Worker   ElfW2(LIBELFBITS,Phdr) *result;
50*7304104dSAndroid Build Coastguard Worker 
51*7304104dSAndroid Build Coastguard Worker   /* If the program header entry has already been filled in the code
52*7304104dSAndroid Build Coastguard Worker      below must already have been run.  So the class is set, too.  No
53*7304104dSAndroid Build Coastguard Worker      need to waste any more time here.  */
54*7304104dSAndroid Build Coastguard Worker   result = elf->state.ELFW(elf,LIBELFBITS).phdr;
55*7304104dSAndroid Build Coastguard Worker   if (likely (result != NULL))
56*7304104dSAndroid Build Coastguard Worker     return result;
57*7304104dSAndroid Build Coastguard Worker 
58*7304104dSAndroid Build Coastguard Worker   if (elf->class == 0)
59*7304104dSAndroid Build Coastguard Worker     elf->class = ELFW(ELFCLASS,LIBELFBITS);
60*7304104dSAndroid Build Coastguard Worker   else if (elf->class != ELFW(ELFCLASS,LIBELFBITS))
61*7304104dSAndroid Build Coastguard Worker     {
62*7304104dSAndroid Build Coastguard Worker       __libelf_seterrno (ELF_E_INVALID_CLASS);
63*7304104dSAndroid Build Coastguard Worker       result = NULL;
64*7304104dSAndroid Build Coastguard Worker       goto out;
65*7304104dSAndroid Build Coastguard Worker     }
66*7304104dSAndroid Build Coastguard Worker 
67*7304104dSAndroid Build Coastguard Worker   if (likely (result == NULL))
68*7304104dSAndroid Build Coastguard Worker     {
69*7304104dSAndroid Build Coastguard Worker       /* Read the section header table.  */
70*7304104dSAndroid Build Coastguard Worker       ElfW2(LIBELFBITS,Ehdr) *ehdr = elf->state.ELFW(elf,LIBELFBITS).ehdr;
71*7304104dSAndroid Build Coastguard Worker 
72*7304104dSAndroid Build Coastguard Worker       /* If no program header exists return NULL.  */
73*7304104dSAndroid Build Coastguard Worker       size_t phnum;
74*7304104dSAndroid Build Coastguard Worker       if (__elf_getphdrnum_rdlock (elf, &phnum) != 0)
75*7304104dSAndroid Build Coastguard Worker 	goto out;
76*7304104dSAndroid Build Coastguard Worker       if (phnum == 0 || ehdr->e_phoff == 0)
77*7304104dSAndroid Build Coastguard Worker 	{
78*7304104dSAndroid Build Coastguard Worker 	  __libelf_seterrno (ELF_E_NO_PHDR);
79*7304104dSAndroid Build Coastguard Worker 	  goto out;
80*7304104dSAndroid Build Coastguard Worker 	}
81*7304104dSAndroid Build Coastguard Worker 
82*7304104dSAndroid Build Coastguard Worker       /* Check this doesn't overflow.  */
83*7304104dSAndroid Build Coastguard Worker       size_t size = phnum * sizeof (ElfW2(LIBELFBITS,Phdr));
84*7304104dSAndroid Build Coastguard Worker 
85*7304104dSAndroid Build Coastguard Worker       if (phnum > SIZE_MAX / sizeof (ElfW2(LIBELFBITS,Phdr))
86*7304104dSAndroid Build Coastguard Worker 	  || ehdr->e_phoff > elf->maximum_size
87*7304104dSAndroid Build Coastguard Worker 	  || elf->maximum_size - ehdr->e_phoff < size)
88*7304104dSAndroid Build Coastguard Worker 	{
89*7304104dSAndroid Build Coastguard Worker 	  __libelf_seterrno (ELF_E_INVALID_DATA);
90*7304104dSAndroid Build Coastguard Worker 	  goto out;
91*7304104dSAndroid Build Coastguard Worker 	}
92*7304104dSAndroid Build Coastguard Worker 
93*7304104dSAndroid Build Coastguard Worker       if (elf->map_address != NULL)
94*7304104dSAndroid Build Coastguard Worker 	{
95*7304104dSAndroid Build Coastguard Worker 	  /* First see whether the information in the ELF header is
96*7304104dSAndroid Build Coastguard Worker 	     valid and it does not ask for too much.  */
97*7304104dSAndroid Build Coastguard Worker 	  if (unlikely (ehdr->e_phoff >= elf->maximum_size)
98*7304104dSAndroid Build Coastguard Worker 	      || unlikely (elf->maximum_size - ehdr->e_phoff < size))
99*7304104dSAndroid Build Coastguard Worker 	    {
100*7304104dSAndroid Build Coastguard Worker 	      /* Something is wrong.  */
101*7304104dSAndroid Build Coastguard Worker 	      __libelf_seterrno (ELF_E_INVALID_PHDR);
102*7304104dSAndroid Build Coastguard Worker 	      goto out;
103*7304104dSAndroid Build Coastguard Worker 	    }
104*7304104dSAndroid Build Coastguard Worker 
105*7304104dSAndroid Build Coastguard Worker 	  /* All the data is already mapped.  Use it.  */
106*7304104dSAndroid Build Coastguard Worker 	  void *file_phdr = ((char *) elf->map_address
107*7304104dSAndroid Build Coastguard Worker 			     + elf->start_offset + ehdr->e_phoff);
108*7304104dSAndroid Build Coastguard Worker 	  if (ehdr->e_ident[EI_DATA] == MY_ELFDATA
109*7304104dSAndroid Build Coastguard Worker 	      && (ALLOW_UNALIGNED
110*7304104dSAndroid Build Coastguard Worker 		  || ((uintptr_t) file_phdr
111*7304104dSAndroid Build Coastguard Worker 		      & (__alignof__ (ElfW2(LIBELFBITS,Phdr)) - 1)) == 0))
112*7304104dSAndroid Build Coastguard Worker 	    /* Simply use the mapped data.  */
113*7304104dSAndroid Build Coastguard Worker 	    elf->state.ELFW(elf,LIBELFBITS).phdr = file_phdr;
114*7304104dSAndroid Build Coastguard Worker 	  else
115*7304104dSAndroid Build Coastguard Worker 	    {
116*7304104dSAndroid Build Coastguard Worker 	      ElfW2(LIBELFBITS,Phdr) *notcvt;
117*7304104dSAndroid Build Coastguard Worker 	      ElfW2(LIBELFBITS,Phdr) *phdr;
118*7304104dSAndroid Build Coastguard Worker 
119*7304104dSAndroid Build Coastguard Worker 	      /* Allocate memory for the program headers.  We know the number
120*7304104dSAndroid Build Coastguard Worker 		 of entries from the ELF header.  */
121*7304104dSAndroid Build Coastguard Worker 	      phdr = elf->state.ELFW(elf,LIBELFBITS).phdr =
122*7304104dSAndroid Build Coastguard Worker 		(ElfW2(LIBELFBITS,Phdr) *) malloc (size);
123*7304104dSAndroid Build Coastguard Worker 	      if (elf->state.ELFW(elf,LIBELFBITS).phdr == NULL)
124*7304104dSAndroid Build Coastguard Worker 		{
125*7304104dSAndroid Build Coastguard Worker 		  __libelf_seterrno (ELF_E_NOMEM);
126*7304104dSAndroid Build Coastguard Worker 		  goto out;
127*7304104dSAndroid Build Coastguard Worker 		}
128*7304104dSAndroid Build Coastguard Worker 	      elf->state.ELFW(elf,LIBELFBITS).phdr_flags |=
129*7304104dSAndroid Build Coastguard Worker 		ELF_F_MALLOCED | ELF_F_DIRTY;
130*7304104dSAndroid Build Coastguard Worker 
131*7304104dSAndroid Build Coastguard Worker 	      /* Now copy the data and at the same time convert the
132*7304104dSAndroid Build Coastguard Worker 		 byte order.  */
133*7304104dSAndroid Build Coastguard Worker 
134*7304104dSAndroid Build Coastguard Worker 	      if (ehdr->e_ident[EI_DATA] == MY_ELFDATA)
135*7304104dSAndroid Build Coastguard Worker 		{
136*7304104dSAndroid Build Coastguard Worker 		  assert (! ALLOW_UNALIGNED);
137*7304104dSAndroid Build Coastguard Worker 		  memcpy (phdr, file_phdr, size);
138*7304104dSAndroid Build Coastguard Worker 		}
139*7304104dSAndroid Build Coastguard Worker 	      else
140*7304104dSAndroid Build Coastguard Worker 		{
141*7304104dSAndroid Build Coastguard Worker 		  bool copy = ! (ALLOW_UNALIGNED
142*7304104dSAndroid Build Coastguard Worker 				 || ((uintptr_t) file_phdr
143*7304104dSAndroid Build Coastguard Worker 				     & (__alignof__ (ElfW2(LIBELFBITS,Phdr))
144*7304104dSAndroid Build Coastguard Worker 					- 1)) == 0);
145*7304104dSAndroid Build Coastguard Worker 		  if (! copy)
146*7304104dSAndroid Build Coastguard Worker 		    notcvt = file_phdr;
147*7304104dSAndroid Build Coastguard Worker 		  else
148*7304104dSAndroid Build Coastguard Worker 		    {
149*7304104dSAndroid Build Coastguard Worker 		      notcvt = (ElfW2(LIBELFBITS,Phdr) *) malloc (size);
150*7304104dSAndroid Build Coastguard Worker 		      if (unlikely (notcvt == NULL))
151*7304104dSAndroid Build Coastguard Worker 			{
152*7304104dSAndroid Build Coastguard Worker 			  __libelf_seterrno (ELF_E_NOMEM);
153*7304104dSAndroid Build Coastguard Worker 			  goto out;
154*7304104dSAndroid Build Coastguard Worker 			}
155*7304104dSAndroid Build Coastguard Worker 		      memcpy (notcvt, file_phdr, size);
156*7304104dSAndroid Build Coastguard Worker 		    }
157*7304104dSAndroid Build Coastguard Worker 
158*7304104dSAndroid Build Coastguard Worker 		  for (size_t cnt = 0; cnt < phnum; ++cnt)
159*7304104dSAndroid Build Coastguard Worker 		    {
160*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_type, notcvt[cnt].p_type);
161*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_offset, notcvt[cnt].p_offset);
162*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_vaddr, notcvt[cnt].p_vaddr);
163*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_paddr, notcvt[cnt].p_paddr);
164*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_filesz, notcvt[cnt].p_filesz);
165*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_memsz, notcvt[cnt].p_memsz);
166*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_flags, notcvt[cnt].p_flags);
167*7304104dSAndroid Build Coastguard Worker 		      CONVERT_TO (phdr[cnt].p_align, notcvt[cnt].p_align);
168*7304104dSAndroid Build Coastguard Worker 		    }
169*7304104dSAndroid Build Coastguard Worker 
170*7304104dSAndroid Build Coastguard Worker 		  if (copy)
171*7304104dSAndroid Build Coastguard Worker 		    free (notcvt);
172*7304104dSAndroid Build Coastguard Worker 		}
173*7304104dSAndroid Build Coastguard Worker 	    }
174*7304104dSAndroid Build Coastguard Worker 	}
175*7304104dSAndroid Build Coastguard Worker       else if (likely (elf->fildes != -1))
176*7304104dSAndroid Build Coastguard Worker 	{
177*7304104dSAndroid Build Coastguard Worker 	  /* Allocate memory for the program headers.  We know the number
178*7304104dSAndroid Build Coastguard Worker 	     of entries from the ELF header.  */
179*7304104dSAndroid Build Coastguard Worker 	  elf->state.ELFW(elf,LIBELFBITS).phdr =
180*7304104dSAndroid Build Coastguard Worker 	    (ElfW2(LIBELFBITS,Phdr) *) malloc (size);
181*7304104dSAndroid Build Coastguard Worker 	  if (elf->state.ELFW(elf,LIBELFBITS).phdr == NULL)
182*7304104dSAndroid Build Coastguard Worker 	    {
183*7304104dSAndroid Build Coastguard Worker 	      __libelf_seterrno (ELF_E_NOMEM);
184*7304104dSAndroid Build Coastguard Worker 	      goto out;
185*7304104dSAndroid Build Coastguard Worker 	    }
186*7304104dSAndroid Build Coastguard Worker 	  elf->state.ELFW(elf,LIBELFBITS).phdr_flags |= ELF_F_MALLOCED;
187*7304104dSAndroid Build Coastguard Worker 
188*7304104dSAndroid Build Coastguard Worker 	  /* Read the header.  */
189*7304104dSAndroid Build Coastguard Worker 	  ssize_t n = pread_retry (elf->fildes,
190*7304104dSAndroid Build Coastguard Worker 				   elf->state.ELFW(elf,LIBELFBITS).phdr, size,
191*7304104dSAndroid Build Coastguard Worker 				   elf->start_offset + ehdr->e_phoff);
192*7304104dSAndroid Build Coastguard Worker 	  if (unlikely ((size_t) n != size))
193*7304104dSAndroid Build Coastguard Worker 	    {
194*7304104dSAndroid Build Coastguard Worker 	      /* Severe problems.  We cannot read the data.  */
195*7304104dSAndroid Build Coastguard Worker 	      __libelf_seterrno (ELF_E_READ_ERROR);
196*7304104dSAndroid Build Coastguard Worker 	      free (elf->state.ELFW(elf,LIBELFBITS).phdr);
197*7304104dSAndroid Build Coastguard Worker 	      elf->state.ELFW(elf,LIBELFBITS).phdr = NULL;
198*7304104dSAndroid Build Coastguard Worker 	      goto out;
199*7304104dSAndroid Build Coastguard Worker 	    }
200*7304104dSAndroid Build Coastguard Worker 
201*7304104dSAndroid Build Coastguard Worker 	  /* If the byte order of the file is not the same as the one
202*7304104dSAndroid Build Coastguard Worker 	     of the host convert the data now.  */
203*7304104dSAndroid Build Coastguard Worker 	  if (ehdr->e_ident[EI_DATA] != MY_ELFDATA)
204*7304104dSAndroid Build Coastguard Worker 	    {
205*7304104dSAndroid Build Coastguard Worker 	      ElfW2(LIBELFBITS,Phdr) *phdr
206*7304104dSAndroid Build Coastguard Worker 		= elf->state.ELFW(elf,LIBELFBITS).phdr;
207*7304104dSAndroid Build Coastguard Worker 
208*7304104dSAndroid Build Coastguard Worker 	      for (size_t cnt = 0; cnt < phnum; ++cnt)
209*7304104dSAndroid Build Coastguard Worker 		{
210*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_type);
211*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_offset);
212*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_vaddr);
213*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_paddr);
214*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_filesz);
215*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_memsz);
216*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_flags);
217*7304104dSAndroid Build Coastguard Worker 		  CONVERT (phdr[cnt].p_align);
218*7304104dSAndroid Build Coastguard Worker 		}
219*7304104dSAndroid Build Coastguard Worker 	    }
220*7304104dSAndroid Build Coastguard Worker 	}
221*7304104dSAndroid Build Coastguard Worker       else
222*7304104dSAndroid Build Coastguard Worker 	{
223*7304104dSAndroid Build Coastguard Worker 	  /* The file descriptor was already enabled and not all data was
224*7304104dSAndroid Build Coastguard Worker 	     read.  */
225*7304104dSAndroid Build Coastguard Worker 	  __libelf_seterrno (ELF_E_FD_DISABLED);
226*7304104dSAndroid Build Coastguard Worker 	  goto out;
227*7304104dSAndroid Build Coastguard Worker 	}
228*7304104dSAndroid Build Coastguard Worker 
229*7304104dSAndroid Build Coastguard Worker       result = elf->state.ELFW(elf,LIBELFBITS).phdr;
230*7304104dSAndroid Build Coastguard Worker     }
231*7304104dSAndroid Build Coastguard Worker 
232*7304104dSAndroid Build Coastguard Worker  out:
233*7304104dSAndroid Build Coastguard Worker   return result;
234*7304104dSAndroid Build Coastguard Worker }
235*7304104dSAndroid Build Coastguard Worker 
ElfW2(LIBELFBITS,Phdr)236*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *
237*7304104dSAndroid Build Coastguard Worker elfw2(LIBELFBITS,getphdr) (Elf *elf)
238*7304104dSAndroid Build Coastguard Worker {
239*7304104dSAndroid Build Coastguard Worker   ElfW2(LIBELFBITS,Phdr) *result;
240*7304104dSAndroid Build Coastguard Worker 
241*7304104dSAndroid Build Coastguard Worker   if (elf == NULL)
242*7304104dSAndroid Build Coastguard Worker     return NULL;
243*7304104dSAndroid Build Coastguard Worker 
244*7304104dSAndroid Build Coastguard Worker   if (unlikely (elf->kind != ELF_K_ELF))
245*7304104dSAndroid Build Coastguard Worker     {
246*7304104dSAndroid Build Coastguard Worker       __libelf_seterrno (ELF_E_INVALID_HANDLE);
247*7304104dSAndroid Build Coastguard Worker       return NULL;
248*7304104dSAndroid Build Coastguard Worker     }
249*7304104dSAndroid Build Coastguard Worker 
250*7304104dSAndroid Build Coastguard Worker   /* If the program header entry has already been filled in the code
251*7304104dSAndroid Build Coastguard Worker    * in getphdr_wrlock must already have been run.  So the class is
252*7304104dSAndroid Build Coastguard Worker    * set, too.  No need to waste any more time here.  */
253*7304104dSAndroid Build Coastguard Worker   result = elf->state.ELFW(elf,LIBELFBITS).phdr;
254*7304104dSAndroid Build Coastguard Worker   if (likely (result != NULL))
255*7304104dSAndroid Build Coastguard Worker     return result;
256*7304104dSAndroid Build Coastguard Worker 
257*7304104dSAndroid Build Coastguard Worker   rwlock_wrlock (elf->lock);
258*7304104dSAndroid Build Coastguard Worker   result = __elfw2(LIBELFBITS,getphdr_wrlock) (elf);
259*7304104dSAndroid Build Coastguard Worker   rwlock_unlock (elf->lock);
260*7304104dSAndroid Build Coastguard Worker 
261*7304104dSAndroid Build Coastguard Worker   return result;
262*7304104dSAndroid Build Coastguard Worker }
263*7304104dSAndroid Build Coastguard Worker INTDEF(elfw2(LIBELFBITS,getphdr))
264