1*7304104dSAndroid Build Coastguard Worker /* Test program for adding a program header to a vendor specific ELF file.
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 2016 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker
5*7304104dSAndroid Build Coastguard Worker This file is free software; you can redistribute it and/or modify
6*7304104dSAndroid Build Coastguard Worker it under the terms of the GNU General Public License as published by
7*7304104dSAndroid Build Coastguard Worker the Free Software Foundation; either version 3 of the License, or
8*7304104dSAndroid Build Coastguard Worker (at your option) any later version.
9*7304104dSAndroid Build Coastguard Worker
10*7304104dSAndroid Build Coastguard Worker elfutils is distributed in the hope that it will be useful, but
11*7304104dSAndroid Build Coastguard Worker WITHOUT ANY WARRANTY; without even the implied warranty of
12*7304104dSAndroid Build Coastguard Worker MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13*7304104dSAndroid Build Coastguard Worker GNU General Public License for more details.
14*7304104dSAndroid Build Coastguard Worker
15*7304104dSAndroid Build Coastguard Worker You should have received a copy of the GNU General Public License
16*7304104dSAndroid Build Coastguard Worker along with this program. If not, see <http://www.gnu.org/licenses/>. */
17*7304104dSAndroid Build Coastguard Worker
18*7304104dSAndroid Build Coastguard Worker
19*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
20*7304104dSAndroid Build Coastguard Worker # include <config.h>
21*7304104dSAndroid Build Coastguard Worker #endif
22*7304104dSAndroid Build Coastguard Worker
23*7304104dSAndroid Build Coastguard Worker #include <errno.h>
24*7304104dSAndroid Build Coastguard Worker #include <fcntl.h>
25*7304104dSAndroid Build Coastguard Worker #include <inttypes.h>
26*7304104dSAndroid Build Coastguard Worker #include <stdio.h>
27*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
28*7304104dSAndroid Build Coastguard Worker #include <string.h>
29*7304104dSAndroid Build Coastguard Worker #include <unistd.h>
30*7304104dSAndroid Build Coastguard Worker #include "system.h"
31*7304104dSAndroid Build Coastguard Worker
32*7304104dSAndroid Build Coastguard Worker #include ELFUTILS_HEADER(elf)
33*7304104dSAndroid Build Coastguard Worker #include <gelf.h>
34*7304104dSAndroid Build Coastguard Worker
35*7304104dSAndroid Build Coastguard Worker void
check_elf(const char * fname,int class,int use_mmap)36*7304104dSAndroid Build Coastguard Worker check_elf (const char *fname, int class, int use_mmap)
37*7304104dSAndroid Build Coastguard Worker {
38*7304104dSAndroid Build Coastguard Worker printf ("\nfname: %s\n", fname);
39*7304104dSAndroid Build Coastguard Worker
40*7304104dSAndroid Build Coastguard Worker int fd = open (fname, O_RDWR | O_CREAT | O_TRUNC, DEFFILEMODE);
41*7304104dSAndroid Build Coastguard Worker if (fd == -1)
42*7304104dSAndroid Build Coastguard Worker {
43*7304104dSAndroid Build Coastguard Worker printf ("cannot open `%s': %s\n", fname, strerror (errno));
44*7304104dSAndroid Build Coastguard Worker exit (1);
45*7304104dSAndroid Build Coastguard Worker }
46*7304104dSAndroid Build Coastguard Worker
47*7304104dSAndroid Build Coastguard Worker Elf *elf = elf_begin (fd, use_mmap ? ELF_C_WRITE_MMAP : ELF_C_WRITE, NULL);
48*7304104dSAndroid Build Coastguard Worker if (elf == NULL)
49*7304104dSAndroid Build Coastguard Worker {
50*7304104dSAndroid Build Coastguard Worker printf ("cannot create ELF descriptor: %s\n", elf_errmsg (-1));
51*7304104dSAndroid Build Coastguard Worker exit (1);
52*7304104dSAndroid Build Coastguard Worker }
53*7304104dSAndroid Build Coastguard Worker
54*7304104dSAndroid Build Coastguard Worker // Create an ELF header.
55*7304104dSAndroid Build Coastguard Worker if (gelf_newehdr (elf, class) == 0)
56*7304104dSAndroid Build Coastguard Worker {
57*7304104dSAndroid Build Coastguard Worker printf ("cannot create ELF header: %s\n", elf_errmsg (-1));
58*7304104dSAndroid Build Coastguard Worker exit (1);
59*7304104dSAndroid Build Coastguard Worker }
60*7304104dSAndroid Build Coastguard Worker
61*7304104dSAndroid Build Coastguard Worker GElf_Ehdr ehdr_mem;
62*7304104dSAndroid Build Coastguard Worker GElf_Ehdr *ehdr = gelf_getehdr (elf, &ehdr_mem);
63*7304104dSAndroid Build Coastguard Worker if (ehdr == NULL)
64*7304104dSAndroid Build Coastguard Worker {
65*7304104dSAndroid Build Coastguard Worker printf ("cannot get ELF header: %s\n", elf_errmsg (-1));
66*7304104dSAndroid Build Coastguard Worker exit (1);
67*7304104dSAndroid Build Coastguard Worker }
68*7304104dSAndroid Build Coastguard Worker
69*7304104dSAndroid Build Coastguard Worker // Initialize header.
70*7304104dSAndroid Build Coastguard Worker ehdr->e_ident[EI_DATA] = class == ELFCLASS64 ? ELFDATA2LSB : ELFDATA2MSB;
71*7304104dSAndroid Build Coastguard Worker ehdr->e_ident[EI_OSABI] = ELFOSABI_GNU;
72*7304104dSAndroid Build Coastguard Worker ehdr->e_type = ET_LOOS + 1;
73*7304104dSAndroid Build Coastguard Worker ehdr->e_machine = EM_X86_64;
74*7304104dSAndroid Build Coastguard Worker ehdr->e_version = EV_CURRENT;
75*7304104dSAndroid Build Coastguard Worker
76*7304104dSAndroid Build Coastguard Worker if (gelf_update_ehdr (elf, ehdr) == 0)
77*7304104dSAndroid Build Coastguard Worker {
78*7304104dSAndroid Build Coastguard Worker printf ("cannot update ELF header: %s\n", elf_errmsg (-1));
79*7304104dSAndroid Build Coastguard Worker exit (1);
80*7304104dSAndroid Build Coastguard Worker }
81*7304104dSAndroid Build Coastguard Worker
82*7304104dSAndroid Build Coastguard Worker // Create a program header.
83*7304104dSAndroid Build Coastguard Worker if (gelf_newphdr (elf, 1) == 0)
84*7304104dSAndroid Build Coastguard Worker {
85*7304104dSAndroid Build Coastguard Worker printf ("cannot create program header: %s\n", elf_errmsg (-1));
86*7304104dSAndroid Build Coastguard Worker exit (1);
87*7304104dSAndroid Build Coastguard Worker }
88*7304104dSAndroid Build Coastguard Worker
89*7304104dSAndroid Build Coastguard Worker GElf_Phdr phdr;
90*7304104dSAndroid Build Coastguard Worker if (gelf_getphdr (elf, 0, &phdr) == NULL)
91*7304104dSAndroid Build Coastguard Worker {
92*7304104dSAndroid Build Coastguard Worker printf ("cannot get program header: %s\n", elf_errmsg (-1));
93*7304104dSAndroid Build Coastguard Worker exit (1);
94*7304104dSAndroid Build Coastguard Worker }
95*7304104dSAndroid Build Coastguard Worker
96*7304104dSAndroid Build Coastguard Worker // Some random values to check later.
97*7304104dSAndroid Build Coastguard Worker phdr.p_type = PT_NULL;
98*7304104dSAndroid Build Coastguard Worker phdr.p_offset = 0;
99*7304104dSAndroid Build Coastguard Worker phdr.p_vaddr = 0;
100*7304104dSAndroid Build Coastguard Worker phdr.p_paddr = 1;
101*7304104dSAndroid Build Coastguard Worker phdr.p_filesz = 0;
102*7304104dSAndroid Build Coastguard Worker phdr.p_memsz = 1024;
103*7304104dSAndroid Build Coastguard Worker phdr.p_flags = PF_R;
104*7304104dSAndroid Build Coastguard Worker phdr.p_align = 16;
105*7304104dSAndroid Build Coastguard Worker
106*7304104dSAndroid Build Coastguard Worker if (gelf_update_phdr (elf, 0, &phdr) == 0)
107*7304104dSAndroid Build Coastguard Worker {
108*7304104dSAndroid Build Coastguard Worker printf ("cannot update program header: %s\n", elf_errmsg (-1));
109*7304104dSAndroid Build Coastguard Worker exit (1);
110*7304104dSAndroid Build Coastguard Worker }
111*7304104dSAndroid Build Coastguard Worker
112*7304104dSAndroid Build Coastguard Worker // Write everything to disk.
113*7304104dSAndroid Build Coastguard Worker if (elf_update (elf, ELF_C_WRITE) < 0)
114*7304104dSAndroid Build Coastguard Worker {
115*7304104dSAndroid Build Coastguard Worker printf ("failure in elf_update(WRITE): %s\n", elf_errmsg (-1));
116*7304104dSAndroid Build Coastguard Worker exit (1);
117*7304104dSAndroid Build Coastguard Worker }
118*7304104dSAndroid Build Coastguard Worker
119*7304104dSAndroid Build Coastguard Worker if (elf_end (elf) != 0)
120*7304104dSAndroid Build Coastguard Worker {
121*7304104dSAndroid Build Coastguard Worker printf ("failure in elf_end: %s\n", elf_errmsg (-1));
122*7304104dSAndroid Build Coastguard Worker exit (1);
123*7304104dSAndroid Build Coastguard Worker }
124*7304104dSAndroid Build Coastguard Worker
125*7304104dSAndroid Build Coastguard Worker close (fd);
126*7304104dSAndroid Build Coastguard Worker
127*7304104dSAndroid Build Coastguard Worker /* Reread the ELF from disk now. */
128*7304104dSAndroid Build Coastguard Worker fd = open (fname, O_RDONLY);
129*7304104dSAndroid Build Coastguard Worker if (fd == -1)
130*7304104dSAndroid Build Coastguard Worker {
131*7304104dSAndroid Build Coastguard Worker printf ("cannot open `%s' read-only: %s\n", fname, strerror (errno));
132*7304104dSAndroid Build Coastguard Worker exit (1);
133*7304104dSAndroid Build Coastguard Worker }
134*7304104dSAndroid Build Coastguard Worker
135*7304104dSAndroid Build Coastguard Worker elf = elf_begin (fd, use_mmap ? ELF_C_READ_MMAP : ELF_C_READ, NULL);
136*7304104dSAndroid Build Coastguard Worker if (elf == NULL)
137*7304104dSAndroid Build Coastguard Worker {
138*7304104dSAndroid Build Coastguard Worker printf ("cannot create ELF descriptor read-only: %s\n", elf_errmsg (-1));
139*7304104dSAndroid Build Coastguard Worker exit (1);
140*7304104dSAndroid Build Coastguard Worker }
141*7304104dSAndroid Build Coastguard Worker
142*7304104dSAndroid Build Coastguard Worker // Is our phdr there?
143*7304104dSAndroid Build Coastguard Worker size_t phnum;
144*7304104dSAndroid Build Coastguard Worker if (elf_getphdrnum (elf, &phnum) != 0)
145*7304104dSAndroid Build Coastguard Worker {
146*7304104dSAndroid Build Coastguard Worker printf ("cannot get phdr num: %s\n", elf_errmsg (-1));
147*7304104dSAndroid Build Coastguard Worker exit (1);
148*7304104dSAndroid Build Coastguard Worker }
149*7304104dSAndroid Build Coastguard Worker
150*7304104dSAndroid Build Coastguard Worker if (phnum != 1)
151*7304104dSAndroid Build Coastguard Worker {
152*7304104dSAndroid Build Coastguard Worker printf ("Expected just 1 phdr, got: %zd\n", phnum);
153*7304104dSAndroid Build Coastguard Worker exit (1);
154*7304104dSAndroid Build Coastguard Worker }
155*7304104dSAndroid Build Coastguard Worker
156*7304104dSAndroid Build Coastguard Worker if (gelf_getphdr (elf, 0, &phdr) == NULL)
157*7304104dSAndroid Build Coastguard Worker {
158*7304104dSAndroid Build Coastguard Worker printf ("cannot get program header from file: %s\n", elf_errmsg (-1));
159*7304104dSAndroid Build Coastguard Worker exit (1);
160*7304104dSAndroid Build Coastguard Worker }
161*7304104dSAndroid Build Coastguard Worker
162*7304104dSAndroid Build Coastguard Worker if (phdr.p_type != PT_NULL
163*7304104dSAndroid Build Coastguard Worker || phdr.p_offset != 0
164*7304104dSAndroid Build Coastguard Worker || phdr.p_vaddr != 0
165*7304104dSAndroid Build Coastguard Worker || phdr.p_paddr != 1
166*7304104dSAndroid Build Coastguard Worker || phdr.p_filesz != 0
167*7304104dSAndroid Build Coastguard Worker || phdr.p_memsz != 1024
168*7304104dSAndroid Build Coastguard Worker || phdr.p_flags != PF_R
169*7304104dSAndroid Build Coastguard Worker || phdr.p_align != 16)
170*7304104dSAndroid Build Coastguard Worker {
171*7304104dSAndroid Build Coastguard Worker printf ("Unexpected phdr values\n");
172*7304104dSAndroid Build Coastguard Worker exit (1);
173*7304104dSAndroid Build Coastguard Worker }
174*7304104dSAndroid Build Coastguard Worker
175*7304104dSAndroid Build Coastguard Worker if (elf_end (elf) != 0)
176*7304104dSAndroid Build Coastguard Worker {
177*7304104dSAndroid Build Coastguard Worker printf ("failure in elf_end: %s\n", elf_errmsg (-1));
178*7304104dSAndroid Build Coastguard Worker exit (1);
179*7304104dSAndroid Build Coastguard Worker }
180*7304104dSAndroid Build Coastguard Worker
181*7304104dSAndroid Build Coastguard Worker close (fd);
182*7304104dSAndroid Build Coastguard Worker
183*7304104dSAndroid Build Coastguard Worker unlink (fname);
184*7304104dSAndroid Build Coastguard Worker }
185*7304104dSAndroid Build Coastguard Worker
186*7304104dSAndroid Build Coastguard Worker int
main(int argc,char * argv[])187*7304104dSAndroid Build Coastguard Worker main (int argc __attribute__ ((unused)),
188*7304104dSAndroid Build Coastguard Worker char *argv[] __attribute__ ((unused)))
189*7304104dSAndroid Build Coastguard Worker {
190*7304104dSAndroid Build Coastguard Worker elf_version (EV_CURRENT);
191*7304104dSAndroid Build Coastguard Worker
192*7304104dSAndroid Build Coastguard Worker check_elf ("vendor.elf.32", ELFCLASS32, 0);
193*7304104dSAndroid Build Coastguard Worker check_elf ("vendor.elf.32.mmap", ELFCLASS32, 1);
194*7304104dSAndroid Build Coastguard Worker check_elf ("vendor.elf.64", ELFCLASS64, 0);
195*7304104dSAndroid Build Coastguard Worker check_elf ("vendor.elf.64.mmap", ELFCLASS64, 1);
196*7304104dSAndroid Build Coastguard Worker
197*7304104dSAndroid Build Coastguard Worker return 0;
198*7304104dSAndroid Build Coastguard Worker }
199