xref: /aosp_15_r20/external/elfutils/libelf/gelf_update_phdr.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Update program header program header table entry.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2000-2010 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]>, 2000.
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 <gelf.h>
35*7304104dSAndroid Build Coastguard Worker #include <string.h>
36*7304104dSAndroid Build Coastguard Worker 
37*7304104dSAndroid Build Coastguard Worker #include "libelfP.h"
38*7304104dSAndroid Build Coastguard Worker 
39*7304104dSAndroid Build Coastguard Worker 
40*7304104dSAndroid Build Coastguard Worker int
gelf_update_phdr(Elf * elf,int ndx,GElf_Phdr * src)41*7304104dSAndroid Build Coastguard Worker gelf_update_phdr (Elf *elf, int ndx, GElf_Phdr *src)
42*7304104dSAndroid Build Coastguard Worker {
43*7304104dSAndroid Build Coastguard Worker   int result = 0;
44*7304104dSAndroid Build Coastguard Worker 
45*7304104dSAndroid Build Coastguard Worker   if (elf == NULL)
46*7304104dSAndroid Build Coastguard Worker     return 0;
47*7304104dSAndroid Build Coastguard Worker 
48*7304104dSAndroid Build Coastguard Worker   if (unlikely (elf->kind != ELF_K_ELF))
49*7304104dSAndroid Build Coastguard Worker     {
50*7304104dSAndroid Build Coastguard Worker       __libelf_seterrno (ELF_E_INVALID_HANDLE);
51*7304104dSAndroid Build Coastguard Worker       return 0;
52*7304104dSAndroid Build Coastguard Worker     }
53*7304104dSAndroid Build Coastguard Worker 
54*7304104dSAndroid Build Coastguard Worker   rwlock_wrlock (elf->lock);
55*7304104dSAndroid Build Coastguard Worker 
56*7304104dSAndroid Build Coastguard Worker   if (elf->class == ELFCLASS32)
57*7304104dSAndroid Build Coastguard Worker     {
58*7304104dSAndroid Build Coastguard Worker       Elf32_Phdr *phdr = elf->state.elf32.phdr;
59*7304104dSAndroid Build Coastguard Worker 
60*7304104dSAndroid Build Coastguard Worker       /* We have to convert the data to the 32 bit format.  This might
61*7304104dSAndroid Build Coastguard Worker 	 overflow some fields so we have to test for this case before
62*7304104dSAndroid Build Coastguard Worker 	 copying.  */
63*7304104dSAndroid Build Coastguard Worker       if (unlikely (src->p_offset > 0xffffffffull)
64*7304104dSAndroid Build Coastguard Worker 	  || unlikely (src->p_vaddr > 0xffffffffull)
65*7304104dSAndroid Build Coastguard Worker 	  || unlikely (src->p_paddr > 0xffffffffull)
66*7304104dSAndroid Build Coastguard Worker 	  || unlikely (src->p_filesz > 0xffffffffull)
67*7304104dSAndroid Build Coastguard Worker 	  || unlikely (src->p_memsz > 0xffffffffull)
68*7304104dSAndroid Build Coastguard Worker 	  || unlikely (src->p_align > 0xffffffffull))
69*7304104dSAndroid Build Coastguard Worker 	{
70*7304104dSAndroid Build Coastguard Worker 	  __libelf_seterrno (ELF_E_INVALID_DATA);
71*7304104dSAndroid Build Coastguard Worker 	  goto out;
72*7304104dSAndroid Build Coastguard Worker 	}
73*7304104dSAndroid Build Coastguard Worker 
74*7304104dSAndroid Build Coastguard Worker       if (phdr == NULL)
75*7304104dSAndroid Build Coastguard Worker 	{
76*7304104dSAndroid Build Coastguard Worker 	  phdr = __elf32_getphdr_wrlock (elf);
77*7304104dSAndroid Build Coastguard Worker 	  if (phdr == NULL)
78*7304104dSAndroid Build Coastguard Worker 	    /* The error number is already set.  */
79*7304104dSAndroid Build Coastguard Worker 	    goto out;
80*7304104dSAndroid Build Coastguard Worker 	}
81*7304104dSAndroid Build Coastguard Worker 
82*7304104dSAndroid Build Coastguard Worker       /* Test whether the index is ok.  */
83*7304104dSAndroid Build Coastguard Worker       size_t phnum;
84*7304104dSAndroid Build Coastguard Worker       if (ndx >= elf->state.elf32.ehdr->e_phnum
85*7304104dSAndroid Build Coastguard Worker 	  && (elf->state.elf32.ehdr->e_phnum != PN_XNUM
86*7304104dSAndroid Build Coastguard Worker 	      || __elf_getphdrnum_rdlock (elf, &phnum) != 0
87*7304104dSAndroid Build Coastguard Worker 	      || (size_t) ndx >= phnum))
88*7304104dSAndroid Build Coastguard Worker 	{
89*7304104dSAndroid Build Coastguard Worker 	  __libelf_seterrno (ELF_E_INVALID_INDEX);
90*7304104dSAndroid Build Coastguard Worker 	  goto out;
91*7304104dSAndroid Build Coastguard Worker 	}
92*7304104dSAndroid Build Coastguard Worker 
93*7304104dSAndroid Build Coastguard Worker       /* Now correct the pointer to point to the correct element.  */
94*7304104dSAndroid Build Coastguard Worker       phdr += ndx;
95*7304104dSAndroid Build Coastguard Worker 
96*7304104dSAndroid Build Coastguard Worker #define COPY(name) \
97*7304104dSAndroid Build Coastguard Worker       phdr->name = src->name
98*7304104dSAndroid Build Coastguard Worker       COPY (p_type);
99*7304104dSAndroid Build Coastguard Worker       COPY (p_offset);
100*7304104dSAndroid Build Coastguard Worker       COPY (p_vaddr);
101*7304104dSAndroid Build Coastguard Worker       COPY (p_paddr);
102*7304104dSAndroid Build Coastguard Worker       COPY (p_filesz);
103*7304104dSAndroid Build Coastguard Worker       COPY (p_memsz);
104*7304104dSAndroid Build Coastguard Worker       COPY (p_flags);
105*7304104dSAndroid Build Coastguard Worker       COPY (p_align);
106*7304104dSAndroid Build Coastguard Worker     }
107*7304104dSAndroid Build Coastguard Worker   else
108*7304104dSAndroid Build Coastguard Worker     {
109*7304104dSAndroid Build Coastguard Worker       Elf64_Phdr *phdr = elf->state.elf64.phdr;
110*7304104dSAndroid Build Coastguard Worker 
111*7304104dSAndroid Build Coastguard Worker       if (phdr == NULL)
112*7304104dSAndroid Build Coastguard Worker 	{
113*7304104dSAndroid Build Coastguard Worker 	  phdr = __elf64_getphdr_wrlock (elf);
114*7304104dSAndroid Build Coastguard Worker 	  if (phdr == NULL)
115*7304104dSAndroid Build Coastguard Worker 	    /* The error number is already set.  */
116*7304104dSAndroid Build Coastguard Worker 	    goto out;
117*7304104dSAndroid Build Coastguard Worker 	}
118*7304104dSAndroid Build Coastguard Worker 
119*7304104dSAndroid Build Coastguard Worker       /* Test whether the index is ok.  */
120*7304104dSAndroid Build Coastguard Worker       size_t phnum;
121*7304104dSAndroid Build Coastguard Worker       if (ndx >= elf->state.elf64.ehdr->e_phnum
122*7304104dSAndroid Build Coastguard Worker 	  && (elf->state.elf64.ehdr->e_phnum != PN_XNUM
123*7304104dSAndroid Build Coastguard Worker 	      || __elf_getphdrnum_rdlock (elf, &phnum) != 0
124*7304104dSAndroid Build Coastguard Worker 	      || (size_t) ndx >= phnum))
125*7304104dSAndroid Build Coastguard Worker 	{
126*7304104dSAndroid Build Coastguard Worker 	  __libelf_seterrno (ELF_E_INVALID_INDEX);
127*7304104dSAndroid Build Coastguard Worker 	  goto out;
128*7304104dSAndroid Build Coastguard Worker 	}
129*7304104dSAndroid Build Coastguard Worker 
130*7304104dSAndroid Build Coastguard Worker       /* Just copy the data.  */
131*7304104dSAndroid Build Coastguard Worker       memcpy (phdr + ndx, src, sizeof (Elf64_Phdr));
132*7304104dSAndroid Build Coastguard Worker     }
133*7304104dSAndroid Build Coastguard Worker 
134*7304104dSAndroid Build Coastguard Worker   /* Mark the program header as modified.  */
135*7304104dSAndroid Build Coastguard Worker   elf->state.elf.phdr_flags |= ELF_F_DIRTY;
136*7304104dSAndroid Build Coastguard Worker 
137*7304104dSAndroid Build Coastguard Worker   result = 1;
138*7304104dSAndroid Build Coastguard Worker 
139*7304104dSAndroid Build Coastguard Worker  out:
140*7304104dSAndroid Build Coastguard Worker   rwlock_unlock (elf->lock);
141*7304104dSAndroid Build Coastguard Worker 
142*7304104dSAndroid Build Coastguard Worker   return result;
143*7304104dSAndroid Build Coastguard Worker }
144