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