1da0c48c4Sopenharmony_ci/* Test program for adding a section to an empty ELF file. 2da0c48c4Sopenharmony_ci Copyright (C) 2016 Red Hat, Inc. 3da0c48c4Sopenharmony_ci This file is part of elfutils. 4da0c48c4Sopenharmony_ci 5da0c48c4Sopenharmony_ci This file is free software; you can redistribute it and/or modify 6da0c48c4Sopenharmony_ci it under the terms of the GNU General Public License as published by 7da0c48c4Sopenharmony_ci the Free Software Foundation; either version 3 of the License, or 8da0c48c4Sopenharmony_ci (at your option) any later version. 9da0c48c4Sopenharmony_ci 10da0c48c4Sopenharmony_ci elfutils is distributed in the hope that it will be useful, but 11da0c48c4Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 12da0c48c4Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13da0c48c4Sopenharmony_ci GNU General Public License for more details. 14da0c48c4Sopenharmony_ci 15da0c48c4Sopenharmony_ci You should have received a copy of the GNU General Public License 16da0c48c4Sopenharmony_ci along with this program. If not, see <http://www.gnu.org/licenses/>. */ 17da0c48c4Sopenharmony_ci 18da0c48c4Sopenharmony_ci 19da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H 20da0c48c4Sopenharmony_ci# include <config.h> 21da0c48c4Sopenharmony_ci#endif 22da0c48c4Sopenharmony_ci 23da0c48c4Sopenharmony_ci#include <errno.h> 24da0c48c4Sopenharmony_ci#include <fcntl.h> 25da0c48c4Sopenharmony_ci#include <inttypes.h> 26da0c48c4Sopenharmony_ci#include <stdio.h> 27da0c48c4Sopenharmony_ci#include <stdlib.h> 28da0c48c4Sopenharmony_ci#include <string.h> 29da0c48c4Sopenharmony_ci#include <unistd.h> 30da0c48c4Sopenharmony_ci#include "system.h" 31da0c48c4Sopenharmony_ci 32da0c48c4Sopenharmony_ci#include ELFUTILS_HEADER(elf) 33da0c48c4Sopenharmony_ci#include <gelf.h> 34da0c48c4Sopenharmony_ci 35da0c48c4Sopenharmony_ci 36da0c48c4Sopenharmony_ci/* Index of last string added. Returned by add_string (). */ 37da0c48c4Sopenharmony_cistatic size_t stridx = 0; 38da0c48c4Sopenharmony_ci 39da0c48c4Sopenharmony_ci/* Adds a string and returns the offset in the section. */ 40da0c48c4Sopenharmony_cistatic size_t 41da0c48c4Sopenharmony_ciadd_string (Elf_Scn *scn, char *str) 42da0c48c4Sopenharmony_ci{ 43da0c48c4Sopenharmony_ci size_t lastidx = stridx; 44da0c48c4Sopenharmony_ci size_t size = strlen (str) + 1; 45da0c48c4Sopenharmony_ci 46da0c48c4Sopenharmony_ci Elf_Data *data = elf_newdata (scn); 47da0c48c4Sopenharmony_ci if (data == NULL) 48da0c48c4Sopenharmony_ci { 49da0c48c4Sopenharmony_ci printf ("cannot create data SHSTRTAB section: %s\n", elf_errmsg (-1)); 50da0c48c4Sopenharmony_ci exit (1); 51da0c48c4Sopenharmony_ci } 52da0c48c4Sopenharmony_ci 53da0c48c4Sopenharmony_ci data->d_buf = str; 54da0c48c4Sopenharmony_ci data->d_type = ELF_T_BYTE; 55da0c48c4Sopenharmony_ci data->d_size = size; 56da0c48c4Sopenharmony_ci data->d_align = 1; 57da0c48c4Sopenharmony_ci data->d_version = EV_CURRENT; 58da0c48c4Sopenharmony_ci 59da0c48c4Sopenharmony_ci stridx += size; 60da0c48c4Sopenharmony_ci printf ("add_string: '%s', stridx: %zd, lastidx: %zd\n", 61da0c48c4Sopenharmony_ci str, stridx, lastidx); 62da0c48c4Sopenharmony_ci return lastidx; 63da0c48c4Sopenharmony_ci} 64da0c48c4Sopenharmony_ci 65da0c48c4Sopenharmony_cistatic void 66da0c48c4Sopenharmony_cicheck_elf (const char *fname, int class, int use_mmap) 67da0c48c4Sopenharmony_ci{ 68da0c48c4Sopenharmony_ci printf ("\nfname: %s\n", fname); 69da0c48c4Sopenharmony_ci stridx = 0; // Reset strtab strings index 70da0c48c4Sopenharmony_ci 71da0c48c4Sopenharmony_ci int fd = open (fname, O_RDWR | O_CREAT | O_TRUNC, DEFFILEMODE); 72da0c48c4Sopenharmony_ci if (fd == -1) 73da0c48c4Sopenharmony_ci { 74da0c48c4Sopenharmony_ci printf ("cannot open `%s': %s\n", fname, strerror (errno)); 75da0c48c4Sopenharmony_ci exit (1); 76da0c48c4Sopenharmony_ci } 77da0c48c4Sopenharmony_ci 78da0c48c4Sopenharmony_ci Elf *elf = elf_begin (fd, use_mmap ? ELF_C_WRITE_MMAP : ELF_C_WRITE, NULL); 79da0c48c4Sopenharmony_ci if (elf == NULL) 80da0c48c4Sopenharmony_ci { 81da0c48c4Sopenharmony_ci printf ("cannot create ELF descriptor: %s\n", elf_errmsg (-1)); 82da0c48c4Sopenharmony_ci exit (1); 83da0c48c4Sopenharmony_ci } 84da0c48c4Sopenharmony_ci 85da0c48c4Sopenharmony_ci // Create an ELF header. 86da0c48c4Sopenharmony_ci if (gelf_newehdr (elf, class) == 0) 87da0c48c4Sopenharmony_ci { 88da0c48c4Sopenharmony_ci printf ("cannot create ELF header: %s\n", elf_errmsg (-1)); 89da0c48c4Sopenharmony_ci exit (1); 90da0c48c4Sopenharmony_ci } 91da0c48c4Sopenharmony_ci 92da0c48c4Sopenharmony_ci GElf_Ehdr ehdr_mem; 93da0c48c4Sopenharmony_ci GElf_Ehdr *ehdr = gelf_getehdr (elf, &ehdr_mem); 94da0c48c4Sopenharmony_ci if (ehdr == NULL) 95da0c48c4Sopenharmony_ci { 96da0c48c4Sopenharmony_ci printf ("cannot get ELF header: %s\n", elf_errmsg (-1)); 97da0c48c4Sopenharmony_ci exit (1); 98da0c48c4Sopenharmony_ci } 99da0c48c4Sopenharmony_ci 100da0c48c4Sopenharmony_ci // Initialize header. 101da0c48c4Sopenharmony_ci ehdr->e_ident[EI_DATA] = class == ELFCLASS64 ? ELFDATA2LSB : ELFDATA2MSB; 102da0c48c4Sopenharmony_ci ehdr->e_ident[EI_OSABI] = ELFOSABI_GNU; 103da0c48c4Sopenharmony_ci ehdr->e_type = ET_NONE; 104da0c48c4Sopenharmony_ci ehdr->e_machine = EM_X86_64; 105da0c48c4Sopenharmony_ci ehdr->e_version = EV_CURRENT; 106da0c48c4Sopenharmony_ci 107da0c48c4Sopenharmony_ci if (gelf_update_ehdr (elf, ehdr) == 0) 108da0c48c4Sopenharmony_ci { 109da0c48c4Sopenharmony_ci printf ("cannot update ELF header: %s\n", elf_errmsg (-1)); 110da0c48c4Sopenharmony_ci exit (1); 111da0c48c4Sopenharmony_ci } 112da0c48c4Sopenharmony_ci 113da0c48c4Sopenharmony_ci // Write everything to disk. 114da0c48c4Sopenharmony_ci if (elf_update (elf, ELF_C_WRITE) < 0) 115da0c48c4Sopenharmony_ci { 116da0c48c4Sopenharmony_ci printf ("failure in elf_update(WRITE): %s\n", elf_errmsg (-1)); 117da0c48c4Sopenharmony_ci exit (1); 118da0c48c4Sopenharmony_ci } 119da0c48c4Sopenharmony_ci 120da0c48c4Sopenharmony_ci if (elf_end (elf) != 0) 121da0c48c4Sopenharmony_ci { 122da0c48c4Sopenharmony_ci printf ("failure in elf_end: %s\n", elf_errmsg (-1)); 123da0c48c4Sopenharmony_ci exit (1); 124da0c48c4Sopenharmony_ci } 125da0c48c4Sopenharmony_ci 126da0c48c4Sopenharmony_ci close (fd); 127da0c48c4Sopenharmony_ci 128da0c48c4Sopenharmony_ci /* Reread the ELF from disk now. */ 129da0c48c4Sopenharmony_ci fd = open (fname, O_RDWR); 130da0c48c4Sopenharmony_ci if (fd == -1) 131da0c48c4Sopenharmony_ci { 132da0c48c4Sopenharmony_ci printf ("cannot (re)open `%s': %s\n", fname, strerror (errno)); 133da0c48c4Sopenharmony_ci exit (1); 134da0c48c4Sopenharmony_ci } 135da0c48c4Sopenharmony_ci 136da0c48c4Sopenharmony_ci elf = elf_begin (fd, use_mmap ? ELF_C_RDWR_MMAP : ELF_C_RDWR, NULL); 137da0c48c4Sopenharmony_ci if (elf == NULL) 138da0c48c4Sopenharmony_ci { 139da0c48c4Sopenharmony_ci printf ("cannot create ELF descriptor read-only: %s\n", elf_errmsg (-1)); 140da0c48c4Sopenharmony_ci exit (1); 141da0c48c4Sopenharmony_ci } 142da0c48c4Sopenharmony_ci 143da0c48c4Sopenharmony_ci // There are no sections yet. 144da0c48c4Sopenharmony_ci if (elf_nextscn (elf, NULL) != NULL) 145da0c48c4Sopenharmony_ci { 146da0c48c4Sopenharmony_ci printf ("Empty elf had a section???\n"); 147da0c48c4Sopenharmony_ci exit (1); 148da0c48c4Sopenharmony_ci } 149da0c48c4Sopenharmony_ci 150da0c48c4Sopenharmony_ci // Create strtab section. 151da0c48c4Sopenharmony_ci Elf_Scn *scn = elf_newscn (elf); 152da0c48c4Sopenharmony_ci if (scn == NULL) 153da0c48c4Sopenharmony_ci { 154da0c48c4Sopenharmony_ci printf ("cannot create strings section: %s\n", elf_errmsg (-1)); 155da0c48c4Sopenharmony_ci exit (1); 156da0c48c4Sopenharmony_ci } 157da0c48c4Sopenharmony_ci 158da0c48c4Sopenharmony_ci // Add an empty string to the table as NUL entry for section zero. 159da0c48c4Sopenharmony_ci add_string (scn, ""); 160da0c48c4Sopenharmony_ci 161da0c48c4Sopenharmony_ci GElf_Shdr shdr_mem; 162da0c48c4Sopenharmony_ci GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem); 163da0c48c4Sopenharmony_ci if (shdr == NULL) 164da0c48c4Sopenharmony_ci { 165da0c48c4Sopenharmony_ci printf ("cannot get header for strings section: %s\n", elf_errmsg (-1)); 166da0c48c4Sopenharmony_ci exit (1); 167da0c48c4Sopenharmony_ci } 168da0c48c4Sopenharmony_ci 169da0c48c4Sopenharmony_ci shdr->sh_type = SHT_STRTAB; 170da0c48c4Sopenharmony_ci shdr->sh_flags = 0; 171da0c48c4Sopenharmony_ci shdr->sh_addr = 0; 172da0c48c4Sopenharmony_ci shdr->sh_link = SHN_UNDEF; 173da0c48c4Sopenharmony_ci shdr->sh_info = SHN_UNDEF; 174da0c48c4Sopenharmony_ci shdr->sh_addralign = 1; 175da0c48c4Sopenharmony_ci shdr->sh_entsize = 0; 176da0c48c4Sopenharmony_ci shdr->sh_name = add_string (scn, ".strtab"); 177da0c48c4Sopenharmony_ci 178da0c48c4Sopenharmony_ci // We have to store the section strtab index in the ELF header. 179da0c48c4Sopenharmony_ci // So sections have actual names. 180da0c48c4Sopenharmony_ci int ndx = elf_ndxscn (scn); 181da0c48c4Sopenharmony_ci ehdr->e_shstrndx = ndx; 182da0c48c4Sopenharmony_ci 183da0c48c4Sopenharmony_ci if (gelf_update_ehdr (elf, ehdr) == 0) 184da0c48c4Sopenharmony_ci { 185da0c48c4Sopenharmony_ci printf ("cannot update ELF header: %s\n", elf_errmsg (-1)); 186da0c48c4Sopenharmony_ci exit (1); 187da0c48c4Sopenharmony_ci } 188da0c48c4Sopenharmony_ci 189da0c48c4Sopenharmony_ci // Finished strtab section, update the header. 190da0c48c4Sopenharmony_ci if (gelf_update_shdr (scn, shdr) == 0) 191da0c48c4Sopenharmony_ci { 192da0c48c4Sopenharmony_ci printf ("cannot update STRTAB section header: %s\n", elf_errmsg (-1)); 193da0c48c4Sopenharmony_ci exit (1); 194da0c48c4Sopenharmony_ci } 195da0c48c4Sopenharmony_ci 196da0c48c4Sopenharmony_ci // Write everything to disk. 197da0c48c4Sopenharmony_ci if (elf_update (elf, ELF_C_WRITE) < 0) 198da0c48c4Sopenharmony_ci { 199da0c48c4Sopenharmony_ci printf ("failure in elf_update(WRITE): %s\n", elf_errmsg (-1)); 200da0c48c4Sopenharmony_ci exit (1); 201da0c48c4Sopenharmony_ci } 202da0c48c4Sopenharmony_ci 203da0c48c4Sopenharmony_ci if (elf_end (elf) != 0) 204da0c48c4Sopenharmony_ci { 205da0c48c4Sopenharmony_ci printf ("failure in elf_end: %s\n", elf_errmsg (-1)); 206da0c48c4Sopenharmony_ci exit (1); 207da0c48c4Sopenharmony_ci } 208da0c48c4Sopenharmony_ci 209da0c48c4Sopenharmony_ci close (fd); 210da0c48c4Sopenharmony_ci 211da0c48c4Sopenharmony_ci // And read it in one last time. 212da0c48c4Sopenharmony_ci fd = open (fname, O_RDONLY); 213da0c48c4Sopenharmony_ci if (fd == -1) 214da0c48c4Sopenharmony_ci { 215da0c48c4Sopenharmony_ci printf ("cannot open `%s' read-only: %s\n", fname, strerror (errno)); 216da0c48c4Sopenharmony_ci exit (1); 217da0c48c4Sopenharmony_ci } 218da0c48c4Sopenharmony_ci 219da0c48c4Sopenharmony_ci elf = elf_begin (fd, use_mmap ? ELF_C_READ_MMAP : ELF_C_READ, NULL); 220da0c48c4Sopenharmony_ci if (elf == NULL) 221da0c48c4Sopenharmony_ci { 222da0c48c4Sopenharmony_ci printf ("cannot create ELF descriptor read-only: %s\n", elf_errmsg (-1)); 223da0c48c4Sopenharmony_ci exit (1); 224da0c48c4Sopenharmony_ci } 225da0c48c4Sopenharmony_ci 226da0c48c4Sopenharmony_ci // Is our new section there? 227da0c48c4Sopenharmony_ci scn = elf_nextscn (elf, NULL); 228da0c48c4Sopenharmony_ci if (scn == NULL) 229da0c48c4Sopenharmony_ci { 230da0c48c4Sopenharmony_ci printf ("cannot get new section: %s\n", elf_errmsg (-1)); 231da0c48c4Sopenharmony_ci exit (1); 232da0c48c4Sopenharmony_ci } 233da0c48c4Sopenharmony_ci 234da0c48c4Sopenharmony_ci shdr = gelf_getshdr (scn, &shdr_mem); 235da0c48c4Sopenharmony_ci if (shdr == NULL) 236da0c48c4Sopenharmony_ci { 237da0c48c4Sopenharmony_ci printf ("cannot get header for new section: %s\n", elf_errmsg (-1)); 238da0c48c4Sopenharmony_ci exit (1); 239da0c48c4Sopenharmony_ci } 240da0c48c4Sopenharmony_ci 241da0c48c4Sopenharmony_ci size_t shstrndx; 242da0c48c4Sopenharmony_ci if (elf_getshdrstrndx (elf, &shstrndx) < 0) 243da0c48c4Sopenharmony_ci { 244da0c48c4Sopenharmony_ci printf ("elf_getshdrstrndx: %s\n", elf_errmsg (-1)); 245da0c48c4Sopenharmony_ci exit (1); 246da0c48c4Sopenharmony_ci } 247da0c48c4Sopenharmony_ci 248da0c48c4Sopenharmony_ci const char *sname = elf_strptr (elf, shstrndx, shdr->sh_name); 249da0c48c4Sopenharmony_ci if (sname == NULL || strcmp (sname, ".strtab") != 0) 250da0c48c4Sopenharmony_ci { 251da0c48c4Sopenharmony_ci printf ("Bad section name: %s\n", sname); 252da0c48c4Sopenharmony_ci exit (1); 253da0c48c4Sopenharmony_ci } 254da0c48c4Sopenharmony_ci 255da0c48c4Sopenharmony_ci if (elf_end (elf) != 0) 256da0c48c4Sopenharmony_ci { 257da0c48c4Sopenharmony_ci printf ("failure in elf_end: %s\n", elf_errmsg (-1)); 258da0c48c4Sopenharmony_ci exit (1); 259da0c48c4Sopenharmony_ci } 260da0c48c4Sopenharmony_ci 261da0c48c4Sopenharmony_ci close (fd); 262da0c48c4Sopenharmony_ci 263da0c48c4Sopenharmony_ci unlink (fname); 264da0c48c4Sopenharmony_ci} 265da0c48c4Sopenharmony_ci 266da0c48c4Sopenharmony_ciint 267da0c48c4Sopenharmony_cimain (int argc __attribute__ ((unused)), 268da0c48c4Sopenharmony_ci char *argv[] __attribute__ ((unused))) 269da0c48c4Sopenharmony_ci{ 270da0c48c4Sopenharmony_ci elf_version (EV_CURRENT); 271da0c48c4Sopenharmony_ci 272da0c48c4Sopenharmony_ci check_elf ("empty.elf.32", ELFCLASS32, 0); 273da0c48c4Sopenharmony_ci check_elf ("empty.elf.32.mmap", ELFCLASS32, 1); 274da0c48c4Sopenharmony_ci check_elf ("empty.elf.64", ELFCLASS64, 0); 275da0c48c4Sopenharmony_ci check_elf ("empty.elf.64.mmap", ELFCLASS64, 1); 276da0c48c4Sopenharmony_ci 277da0c48c4Sopenharmony_ci return 0; 278da0c48c4Sopenharmony_ci} 279