xref: /aosp_15_r20/external/elfutils/tests/emptyfile.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Test program for adding a section to an empty 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 
36*7304104dSAndroid Build Coastguard Worker /* Index of last string added.  Returned by add_string ().  */
37*7304104dSAndroid Build Coastguard Worker static size_t stridx = 0;
38*7304104dSAndroid Build Coastguard Worker 
39*7304104dSAndroid Build Coastguard Worker /* Adds a string and returns the offset in the section.  */
40*7304104dSAndroid Build Coastguard Worker static size_t
add_string(Elf_Scn * scn,char * str)41*7304104dSAndroid Build Coastguard Worker add_string (Elf_Scn *scn, char *str)
42*7304104dSAndroid Build Coastguard Worker {
43*7304104dSAndroid Build Coastguard Worker   size_t lastidx = stridx;
44*7304104dSAndroid Build Coastguard Worker   size_t size = strlen (str) + 1;
45*7304104dSAndroid Build Coastguard Worker 
46*7304104dSAndroid Build Coastguard Worker   Elf_Data *data = elf_newdata (scn);
47*7304104dSAndroid Build Coastguard Worker   if (data == NULL)
48*7304104dSAndroid Build Coastguard Worker     {
49*7304104dSAndroid Build Coastguard Worker       printf ("cannot create data SHSTRTAB section: %s\n", elf_errmsg (-1));
50*7304104dSAndroid Build Coastguard Worker       exit (1);
51*7304104dSAndroid Build Coastguard Worker     }
52*7304104dSAndroid Build Coastguard Worker 
53*7304104dSAndroid Build Coastguard Worker   data->d_buf = str;
54*7304104dSAndroid Build Coastguard Worker   data->d_type = ELF_T_BYTE;
55*7304104dSAndroid Build Coastguard Worker   data->d_size = size;
56*7304104dSAndroid Build Coastguard Worker   data->d_align = 1;
57*7304104dSAndroid Build Coastguard Worker   data->d_version = EV_CURRENT;
58*7304104dSAndroid Build Coastguard Worker 
59*7304104dSAndroid Build Coastguard Worker   stridx += size;
60*7304104dSAndroid Build Coastguard Worker   printf ("add_string: '%s', stridx: %zd, lastidx: %zd\n",
61*7304104dSAndroid Build Coastguard Worker 	  str, stridx, lastidx);
62*7304104dSAndroid Build Coastguard Worker   return lastidx;
63*7304104dSAndroid Build Coastguard Worker }
64*7304104dSAndroid Build Coastguard Worker 
65*7304104dSAndroid Build Coastguard Worker static void
check_elf(const char * fname,int class,int use_mmap)66*7304104dSAndroid Build Coastguard Worker check_elf (const char *fname, int class, int use_mmap)
67*7304104dSAndroid Build Coastguard Worker {
68*7304104dSAndroid Build Coastguard Worker   printf ("\nfname: %s\n", fname);
69*7304104dSAndroid Build Coastguard Worker   stridx = 0; // Reset strtab strings index
70*7304104dSAndroid Build Coastguard Worker 
71*7304104dSAndroid Build Coastguard Worker   int fd = open (fname, O_RDWR | O_CREAT | O_TRUNC, DEFFILEMODE);
72*7304104dSAndroid Build Coastguard Worker   if (fd == -1)
73*7304104dSAndroid Build Coastguard Worker     {
74*7304104dSAndroid Build Coastguard Worker       printf ("cannot open `%s': %s\n", fname, strerror (errno));
75*7304104dSAndroid Build Coastguard Worker       exit (1);
76*7304104dSAndroid Build Coastguard Worker     }
77*7304104dSAndroid Build Coastguard Worker 
78*7304104dSAndroid Build Coastguard Worker   Elf *elf = elf_begin (fd, use_mmap ? ELF_C_WRITE_MMAP : ELF_C_WRITE, NULL);
79*7304104dSAndroid Build Coastguard Worker   if (elf == NULL)
80*7304104dSAndroid Build Coastguard Worker     {
81*7304104dSAndroid Build Coastguard Worker       printf ("cannot create ELF descriptor: %s\n", elf_errmsg (-1));
82*7304104dSAndroid Build Coastguard Worker       exit (1);
83*7304104dSAndroid Build Coastguard Worker     }
84*7304104dSAndroid Build Coastguard Worker 
85*7304104dSAndroid Build Coastguard Worker   // Create an ELF header.
86*7304104dSAndroid Build Coastguard Worker   if (gelf_newehdr (elf, class) == 0)
87*7304104dSAndroid Build Coastguard Worker     {
88*7304104dSAndroid Build Coastguard Worker       printf ("cannot create ELF header: %s\n", elf_errmsg (-1));
89*7304104dSAndroid Build Coastguard Worker       exit (1);
90*7304104dSAndroid Build Coastguard Worker     }
91*7304104dSAndroid Build Coastguard Worker 
92*7304104dSAndroid Build Coastguard Worker   GElf_Ehdr ehdr_mem;
93*7304104dSAndroid Build Coastguard Worker   GElf_Ehdr *ehdr = gelf_getehdr (elf, &ehdr_mem);
94*7304104dSAndroid Build Coastguard Worker   if (ehdr == NULL)
95*7304104dSAndroid Build Coastguard Worker     {
96*7304104dSAndroid Build Coastguard Worker       printf ("cannot get ELF header: %s\n", elf_errmsg (-1));
97*7304104dSAndroid Build Coastguard Worker       exit (1);
98*7304104dSAndroid Build Coastguard Worker     }
99*7304104dSAndroid Build Coastguard Worker 
100*7304104dSAndroid Build Coastguard Worker   // Initialize header.
101*7304104dSAndroid Build Coastguard Worker   ehdr->e_ident[EI_DATA] = class == ELFCLASS64 ? ELFDATA2LSB : ELFDATA2MSB;
102*7304104dSAndroid Build Coastguard Worker   ehdr->e_ident[EI_OSABI] = ELFOSABI_GNU;
103*7304104dSAndroid Build Coastguard Worker   ehdr->e_type = ET_NONE;
104*7304104dSAndroid Build Coastguard Worker   ehdr->e_machine = EM_X86_64;
105*7304104dSAndroid Build Coastguard Worker   ehdr->e_version = EV_CURRENT;
106*7304104dSAndroid Build Coastguard Worker 
107*7304104dSAndroid Build Coastguard Worker   if (gelf_update_ehdr (elf, ehdr) == 0)
108*7304104dSAndroid Build Coastguard Worker     {
109*7304104dSAndroid Build Coastguard Worker       printf ("cannot update ELF header: %s\n", elf_errmsg (-1));
110*7304104dSAndroid Build Coastguard Worker       exit (1);
111*7304104dSAndroid Build Coastguard Worker     }
112*7304104dSAndroid Build Coastguard Worker 
113*7304104dSAndroid Build Coastguard Worker   // Write everything to disk.
114*7304104dSAndroid Build Coastguard Worker   if (elf_update (elf, ELF_C_WRITE) < 0)
115*7304104dSAndroid Build Coastguard Worker     {
116*7304104dSAndroid Build Coastguard Worker       printf ("failure in elf_update(WRITE): %s\n", elf_errmsg (-1));
117*7304104dSAndroid Build Coastguard Worker       exit (1);
118*7304104dSAndroid Build Coastguard Worker     }
119*7304104dSAndroid Build Coastguard Worker 
120*7304104dSAndroid Build Coastguard Worker   if (elf_end (elf) != 0)
121*7304104dSAndroid Build Coastguard Worker     {
122*7304104dSAndroid Build Coastguard Worker       printf ("failure in elf_end: %s\n", elf_errmsg (-1));
123*7304104dSAndroid Build Coastguard Worker       exit (1);
124*7304104dSAndroid Build Coastguard Worker     }
125*7304104dSAndroid Build Coastguard Worker 
126*7304104dSAndroid Build Coastguard Worker   close (fd);
127*7304104dSAndroid Build Coastguard Worker 
128*7304104dSAndroid Build Coastguard Worker   /* Reread the ELF from disk now.  */
129*7304104dSAndroid Build Coastguard Worker   fd = open (fname, O_RDWR);
130*7304104dSAndroid Build Coastguard Worker   if (fd == -1)
131*7304104dSAndroid Build Coastguard Worker     {
132*7304104dSAndroid Build Coastguard Worker       printf ("cannot (re)open `%s': %s\n", fname, strerror (errno));
133*7304104dSAndroid Build Coastguard Worker       exit (1);
134*7304104dSAndroid Build Coastguard Worker     }
135*7304104dSAndroid Build Coastguard Worker 
136*7304104dSAndroid Build Coastguard Worker   elf = elf_begin (fd, use_mmap ? ELF_C_RDWR_MMAP : ELF_C_RDWR, NULL);
137*7304104dSAndroid Build Coastguard Worker   if (elf == NULL)
138*7304104dSAndroid Build Coastguard Worker     {
139*7304104dSAndroid Build Coastguard Worker       printf ("cannot create ELF descriptor read-only: %s\n", elf_errmsg (-1));
140*7304104dSAndroid Build Coastguard Worker       exit (1);
141*7304104dSAndroid Build Coastguard Worker     }
142*7304104dSAndroid Build Coastguard Worker 
143*7304104dSAndroid Build Coastguard Worker   // There are no sections yet.
144*7304104dSAndroid Build Coastguard Worker   if (elf_nextscn (elf, NULL) != NULL)
145*7304104dSAndroid Build Coastguard Worker     {
146*7304104dSAndroid Build Coastguard Worker       printf ("Empty elf had a section???\n");
147*7304104dSAndroid Build Coastguard Worker       exit (1);
148*7304104dSAndroid Build Coastguard Worker     }
149*7304104dSAndroid Build Coastguard Worker 
150*7304104dSAndroid Build Coastguard Worker   // Create strtab section.
151*7304104dSAndroid Build Coastguard Worker   Elf_Scn *scn = elf_newscn (elf);
152*7304104dSAndroid Build Coastguard Worker   if (scn == NULL)
153*7304104dSAndroid Build Coastguard Worker     {
154*7304104dSAndroid Build Coastguard Worker       printf ("cannot create strings section: %s\n", elf_errmsg (-1));
155*7304104dSAndroid Build Coastguard Worker       exit (1);
156*7304104dSAndroid Build Coastguard Worker     }
157*7304104dSAndroid Build Coastguard Worker 
158*7304104dSAndroid Build Coastguard Worker   // Add an empty string to the table as NUL entry for section zero.
159*7304104dSAndroid Build Coastguard Worker   add_string (scn, "");
160*7304104dSAndroid Build Coastguard Worker 
161*7304104dSAndroid Build Coastguard Worker   GElf_Shdr shdr_mem;
162*7304104dSAndroid Build Coastguard Worker   GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
163*7304104dSAndroid Build Coastguard Worker   if (shdr == NULL)
164*7304104dSAndroid Build Coastguard Worker     {
165*7304104dSAndroid Build Coastguard Worker       printf ("cannot get header for strings section: %s\n", elf_errmsg (-1));
166*7304104dSAndroid Build Coastguard Worker       exit (1);
167*7304104dSAndroid Build Coastguard Worker     }
168*7304104dSAndroid Build Coastguard Worker 
169*7304104dSAndroid Build Coastguard Worker   shdr->sh_type = SHT_STRTAB;
170*7304104dSAndroid Build Coastguard Worker   shdr->sh_flags = 0;
171*7304104dSAndroid Build Coastguard Worker   shdr->sh_addr = 0;
172*7304104dSAndroid Build Coastguard Worker   shdr->sh_link = SHN_UNDEF;
173*7304104dSAndroid Build Coastguard Worker   shdr->sh_info = SHN_UNDEF;
174*7304104dSAndroid Build Coastguard Worker   shdr->sh_addralign = 1;
175*7304104dSAndroid Build Coastguard Worker   shdr->sh_entsize = 0;
176*7304104dSAndroid Build Coastguard Worker   shdr->sh_name = add_string (scn, ".strtab");
177*7304104dSAndroid Build Coastguard Worker 
178*7304104dSAndroid Build Coastguard Worker   // We have to store the section strtab index in the ELF header.
179*7304104dSAndroid Build Coastguard Worker   // So sections have actual names.
180*7304104dSAndroid Build Coastguard Worker   int ndx = elf_ndxscn (scn);
181*7304104dSAndroid Build Coastguard Worker   ehdr->e_shstrndx = ndx;
182*7304104dSAndroid Build Coastguard Worker 
183*7304104dSAndroid Build Coastguard Worker   if (gelf_update_ehdr (elf, ehdr) == 0)
184*7304104dSAndroid Build Coastguard Worker     {
185*7304104dSAndroid Build Coastguard Worker       printf ("cannot update ELF header: %s\n", elf_errmsg (-1));
186*7304104dSAndroid Build Coastguard Worker       exit (1);
187*7304104dSAndroid Build Coastguard Worker     }
188*7304104dSAndroid Build Coastguard Worker 
189*7304104dSAndroid Build Coastguard Worker   // Finished strtab section, update the header.
190*7304104dSAndroid Build Coastguard Worker   if (gelf_update_shdr (scn, shdr) == 0)
191*7304104dSAndroid Build Coastguard Worker     {
192*7304104dSAndroid Build Coastguard Worker       printf ("cannot update STRTAB section header: %s\n", elf_errmsg (-1));
193*7304104dSAndroid Build Coastguard Worker       exit (1);
194*7304104dSAndroid Build Coastguard Worker     }
195*7304104dSAndroid Build Coastguard Worker 
196*7304104dSAndroid Build Coastguard Worker   // Write everything to disk.
197*7304104dSAndroid Build Coastguard Worker   if (elf_update (elf, ELF_C_WRITE) < 0)
198*7304104dSAndroid Build Coastguard Worker     {
199*7304104dSAndroid Build Coastguard Worker       printf ("failure in elf_update(WRITE): %s\n", elf_errmsg (-1));
200*7304104dSAndroid Build Coastguard Worker       exit (1);
201*7304104dSAndroid Build Coastguard Worker     }
202*7304104dSAndroid Build Coastguard Worker 
203*7304104dSAndroid Build Coastguard Worker   if (elf_end (elf) != 0)
204*7304104dSAndroid Build Coastguard Worker     {
205*7304104dSAndroid Build Coastguard Worker       printf ("failure in elf_end: %s\n", elf_errmsg (-1));
206*7304104dSAndroid Build Coastguard Worker       exit (1);
207*7304104dSAndroid Build Coastguard Worker     }
208*7304104dSAndroid Build Coastguard Worker 
209*7304104dSAndroid Build Coastguard Worker   close (fd);
210*7304104dSAndroid Build Coastguard Worker 
211*7304104dSAndroid Build Coastguard Worker   // And read it in one last time.
212*7304104dSAndroid Build Coastguard Worker   fd = open (fname, O_RDONLY);
213*7304104dSAndroid Build Coastguard Worker   if (fd == -1)
214*7304104dSAndroid Build Coastguard Worker     {
215*7304104dSAndroid Build Coastguard Worker       printf ("cannot open `%s' read-only: %s\n", fname, strerror (errno));
216*7304104dSAndroid Build Coastguard Worker       exit (1);
217*7304104dSAndroid Build Coastguard Worker     }
218*7304104dSAndroid Build Coastguard Worker 
219*7304104dSAndroid Build Coastguard Worker   elf = elf_begin (fd, use_mmap ? ELF_C_READ_MMAP : ELF_C_READ, NULL);
220*7304104dSAndroid Build Coastguard Worker   if (elf == NULL)
221*7304104dSAndroid Build Coastguard Worker     {
222*7304104dSAndroid Build Coastguard Worker       printf ("cannot create ELF descriptor read-only: %s\n", elf_errmsg (-1));
223*7304104dSAndroid Build Coastguard Worker       exit (1);
224*7304104dSAndroid Build Coastguard Worker     }
225*7304104dSAndroid Build Coastguard Worker 
226*7304104dSAndroid Build Coastguard Worker   // Is our new section there?
227*7304104dSAndroid Build Coastguard Worker   scn = elf_nextscn (elf, NULL);
228*7304104dSAndroid Build Coastguard Worker   if (scn == NULL)
229*7304104dSAndroid Build Coastguard Worker     {
230*7304104dSAndroid Build Coastguard Worker       printf ("cannot get new section: %s\n", elf_errmsg (-1));
231*7304104dSAndroid Build Coastguard Worker       exit (1);
232*7304104dSAndroid Build Coastguard Worker     }
233*7304104dSAndroid Build Coastguard Worker 
234*7304104dSAndroid Build Coastguard Worker   shdr = gelf_getshdr (scn, &shdr_mem);
235*7304104dSAndroid Build Coastguard Worker   if (shdr == NULL)
236*7304104dSAndroid Build Coastguard Worker     {
237*7304104dSAndroid Build Coastguard Worker       printf ("cannot get header for new section: %s\n", elf_errmsg (-1));
238*7304104dSAndroid Build Coastguard Worker       exit (1);
239*7304104dSAndroid Build Coastguard Worker     }
240*7304104dSAndroid Build Coastguard Worker 
241*7304104dSAndroid Build Coastguard Worker   size_t shstrndx;
242*7304104dSAndroid Build Coastguard Worker   if (elf_getshdrstrndx (elf, &shstrndx) < 0)
243*7304104dSAndroid Build Coastguard Worker     {
244*7304104dSAndroid Build Coastguard Worker       printf ("elf_getshdrstrndx: %s\n", elf_errmsg (-1));
245*7304104dSAndroid Build Coastguard Worker       exit (1);
246*7304104dSAndroid Build Coastguard Worker     }
247*7304104dSAndroid Build Coastguard Worker 
248*7304104dSAndroid Build Coastguard Worker   const char *sname = elf_strptr (elf, shstrndx, shdr->sh_name);
249*7304104dSAndroid Build Coastguard Worker   if (sname == NULL || strcmp (sname, ".strtab") != 0)
250*7304104dSAndroid Build Coastguard Worker     {
251*7304104dSAndroid Build Coastguard Worker       printf ("Bad section name: %s\n", sname);
252*7304104dSAndroid Build Coastguard Worker       exit (1);
253*7304104dSAndroid Build Coastguard Worker     }
254*7304104dSAndroid Build Coastguard Worker 
255*7304104dSAndroid Build Coastguard Worker   if (elf_end (elf) != 0)
256*7304104dSAndroid Build Coastguard Worker     {
257*7304104dSAndroid Build Coastguard Worker       printf ("failure in elf_end: %s\n", elf_errmsg (-1));
258*7304104dSAndroid Build Coastguard Worker       exit (1);
259*7304104dSAndroid Build Coastguard Worker     }
260*7304104dSAndroid Build Coastguard Worker 
261*7304104dSAndroid Build Coastguard Worker   close (fd);
262*7304104dSAndroid Build Coastguard Worker 
263*7304104dSAndroid Build Coastguard Worker   unlink (fname);
264*7304104dSAndroid Build Coastguard Worker }
265*7304104dSAndroid Build Coastguard Worker 
266*7304104dSAndroid Build Coastguard Worker int
main(int argc,char * argv[])267*7304104dSAndroid Build Coastguard Worker main (int argc __attribute__ ((unused)),
268*7304104dSAndroid Build Coastguard Worker       char *argv[] __attribute__ ((unused)))
269*7304104dSAndroid Build Coastguard Worker {
270*7304104dSAndroid Build Coastguard Worker   elf_version (EV_CURRENT);
271*7304104dSAndroid Build Coastguard Worker 
272*7304104dSAndroid Build Coastguard Worker   check_elf ("empty.elf.32", ELFCLASS32, 0);
273*7304104dSAndroid Build Coastguard Worker   check_elf ("empty.elf.32.mmap", ELFCLASS32, 1);
274*7304104dSAndroid Build Coastguard Worker   check_elf ("empty.elf.64", ELFCLASS64, 0);
275*7304104dSAndroid Build Coastguard Worker   check_elf ("empty.elf.64.mmap", ELFCLASS64, 1);
276*7304104dSAndroid Build Coastguard Worker 
277*7304104dSAndroid Build Coastguard Worker   return 0;
278*7304104dSAndroid Build Coastguard Worker }
279