1da0c48c4Sopenharmony_ci/* Create new ELF program header table. 2da0c48c4Sopenharmony_ci Copyright (C) 1999-2010, 2014, 2015 Red Hat, Inc. 3da0c48c4Sopenharmony_ci This file is part of elfutils. 4da0c48c4Sopenharmony_ci Written by Ulrich Drepper <drepper@redhat.com>, 1998. 5da0c48c4Sopenharmony_ci 6da0c48c4Sopenharmony_ci This file is free software; you can redistribute it and/or modify 7da0c48c4Sopenharmony_ci it under the terms of either 8da0c48c4Sopenharmony_ci 9da0c48c4Sopenharmony_ci * the GNU Lesser General Public License as published by the Free 10da0c48c4Sopenharmony_ci Software Foundation; either version 3 of the License, or (at 11da0c48c4Sopenharmony_ci your option) any later version 12da0c48c4Sopenharmony_ci 13da0c48c4Sopenharmony_ci or 14da0c48c4Sopenharmony_ci 15da0c48c4Sopenharmony_ci * the GNU General Public License as published by the Free 16da0c48c4Sopenharmony_ci Software Foundation; either version 2 of the License, or (at 17da0c48c4Sopenharmony_ci your option) any later version 18da0c48c4Sopenharmony_ci 19da0c48c4Sopenharmony_ci or both in parallel, as here. 20da0c48c4Sopenharmony_ci 21da0c48c4Sopenharmony_ci elfutils is distributed in the hope that it will be useful, but 22da0c48c4Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 23da0c48c4Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 24da0c48c4Sopenharmony_ci General Public License for more details. 25da0c48c4Sopenharmony_ci 26da0c48c4Sopenharmony_ci You should have received copies of the GNU General Public License and 27da0c48c4Sopenharmony_ci the GNU Lesser General Public License along with this program. If 28da0c48c4Sopenharmony_ci not, see <http://www.gnu.org/licenses/>. */ 29da0c48c4Sopenharmony_ci 30da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H 31da0c48c4Sopenharmony_ci# include <config.h> 32da0c48c4Sopenharmony_ci#endif 33da0c48c4Sopenharmony_ci 34da0c48c4Sopenharmony_ci#include <assert.h> 35da0c48c4Sopenharmony_ci#include <stdlib.h> 36da0c48c4Sopenharmony_ci#include <string.h> 37da0c48c4Sopenharmony_ci 38da0c48c4Sopenharmony_ci#include "libelfP.h" 39da0c48c4Sopenharmony_ci 40da0c48c4Sopenharmony_ci#ifndef LIBELFBITS 41da0c48c4Sopenharmony_ci# define LIBELFBITS 32 42da0c48c4Sopenharmony_ci#endif 43da0c48c4Sopenharmony_ci 44da0c48c4Sopenharmony_ci 45da0c48c4Sopenharmony_ciElfW2(LIBELFBITS,Phdr) * 46da0c48c4Sopenharmony_cielfw2(LIBELFBITS,newphdr) (Elf *elf, size_t count) 47da0c48c4Sopenharmony_ci{ 48da0c48c4Sopenharmony_ci ElfW2(LIBELFBITS,Phdr) *result; 49da0c48c4Sopenharmony_ci 50da0c48c4Sopenharmony_ci if (elf == NULL) 51da0c48c4Sopenharmony_ci return NULL; 52da0c48c4Sopenharmony_ci 53da0c48c4Sopenharmony_ci if (unlikely (elf->kind != ELF_K_ELF)) 54da0c48c4Sopenharmony_ci { 55da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_INVALID_HANDLE); 56da0c48c4Sopenharmony_ci return NULL; 57da0c48c4Sopenharmony_ci } 58da0c48c4Sopenharmony_ci 59da0c48c4Sopenharmony_ci /* This check is correct, it is for sh_info, which is either 60da0c48c4Sopenharmony_ci Elf32_Word or Elf64_Word, both being 32 bits. But count is size_t 61da0c48c4Sopenharmony_ci so might not fit on 32bit ELF files. */ 62da0c48c4Sopenharmony_ci if (unlikely ((ElfW2(LIBELFBITS,Word)) count != count)) 63da0c48c4Sopenharmony_ci { 64da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_INVALID_OPERAND); 65da0c48c4Sopenharmony_ci return NULL; 66da0c48c4Sopenharmony_ci } 67da0c48c4Sopenharmony_ci 68da0c48c4Sopenharmony_ci rwlock_wrlock (elf->lock); 69da0c48c4Sopenharmony_ci 70da0c48c4Sopenharmony_ci if (elf->class == 0) 71da0c48c4Sopenharmony_ci elf->class = ELFW(ELFCLASS,LIBELFBITS); 72da0c48c4Sopenharmony_ci else if (unlikely (elf->class != ELFW(ELFCLASS,LIBELFBITS))) 73da0c48c4Sopenharmony_ci { 74da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_INVALID_CLASS); 75da0c48c4Sopenharmony_ci result = NULL; 76da0c48c4Sopenharmony_ci goto out; 77da0c48c4Sopenharmony_ci } 78da0c48c4Sopenharmony_ci 79da0c48c4Sopenharmony_ci if (unlikely (elf->state.ELFW(elf,LIBELFBITS).ehdr == NULL)) 80da0c48c4Sopenharmony_ci { 81da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_WRONG_ORDER_EHDR); 82da0c48c4Sopenharmony_ci result = NULL; 83da0c48c4Sopenharmony_ci goto out; 84da0c48c4Sopenharmony_ci } 85da0c48c4Sopenharmony_ci 86da0c48c4Sopenharmony_ci /* A COUNT of zero means remove existing table. */ 87da0c48c4Sopenharmony_ci if (count == 0) 88da0c48c4Sopenharmony_ci { 89da0c48c4Sopenharmony_ci /* Free the old program header. */ 90da0c48c4Sopenharmony_ci if (elf->state.ELFW(elf,LIBELFBITS).phdr != NULL) 91da0c48c4Sopenharmony_ci { 92da0c48c4Sopenharmony_ci if (elf->state.ELFW(elf,LIBELFBITS).phdr_flags & ELF_F_MALLOCED) 93da0c48c4Sopenharmony_ci free (elf->state.ELFW(elf,LIBELFBITS).phdr); 94da0c48c4Sopenharmony_ci 95da0c48c4Sopenharmony_ci /* Set the pointer to NULL. */ 96da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).phdr = NULL; 97da0c48c4Sopenharmony_ci /* Set the `e_phnum' member to the new value. */ 98da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).ehdr->e_phnum = 0; 99da0c48c4Sopenharmony_ci /* Also clear any old PN_XNUM extended value. */ 100da0c48c4Sopenharmony_ci if (elf->state.ELFW(elf,LIBELFBITS).scns.cnt > 0) 101da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).scns.data[0] 102da0c48c4Sopenharmony_ci .shdr.ELFW(e,LIBELFBITS)->sh_info = 0; 103da0c48c4Sopenharmony_ci /* Also set the size. */ 104da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).ehdr->e_phentsize = 105da0c48c4Sopenharmony_ci sizeof (ElfW2(LIBELFBITS,Phdr)); 106da0c48c4Sopenharmony_ci 107da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).phdr_flags |= ELF_F_DIRTY; 108da0c48c4Sopenharmony_ci elf->flags |= ELF_F_DIRTY; 109da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_NOERROR); 110da0c48c4Sopenharmony_ci } 111da0c48c4Sopenharmony_ci 112da0c48c4Sopenharmony_ci result = NULL; 113da0c48c4Sopenharmony_ci } 114da0c48c4Sopenharmony_ci else if (elf->state.ELFW(elf,LIBELFBITS).ehdr->e_phnum != count 115da0c48c4Sopenharmony_ci || count == PN_XNUM 116da0c48c4Sopenharmony_ci || elf->state.ELFW(elf,LIBELFBITS).phdr == NULL) 117da0c48c4Sopenharmony_ci { 118da0c48c4Sopenharmony_ci if (unlikely (count > SIZE_MAX / sizeof (ElfW2(LIBELFBITS,Phdr)))) 119da0c48c4Sopenharmony_ci { 120da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_INVALID_INDEX); 121da0c48c4Sopenharmony_ci result = NULL; 122da0c48c4Sopenharmony_ci goto out; 123da0c48c4Sopenharmony_ci } 124da0c48c4Sopenharmony_ci 125da0c48c4Sopenharmony_ci Elf_Scn *scn0 = &elf->state.ELFW(elf,LIBELFBITS).scns.data[0]; 126da0c48c4Sopenharmony_ci if (unlikely (count >= PN_XNUM && scn0->shdr.ELFW(e,LIBELFBITS) == NULL)) 127da0c48c4Sopenharmony_ci { 128da0c48c4Sopenharmony_ci /* Something is wrong with section zero, but we need it to write 129da0c48c4Sopenharmony_ci the extended phdr count. */ 130da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_INVALID_SECTION_HEADER); 131da0c48c4Sopenharmony_ci result = NULL; 132da0c48c4Sopenharmony_ci goto out; 133da0c48c4Sopenharmony_ci } 134da0c48c4Sopenharmony_ci 135da0c48c4Sopenharmony_ci /* Allocate a new program header with the appropriate number of 136da0c48c4Sopenharmony_ci elements. */ 137da0c48c4Sopenharmony_ci result = (ElfW2(LIBELFBITS,Phdr) *) 138da0c48c4Sopenharmony_ci realloc (elf->state.ELFW(elf,LIBELFBITS).phdr, 139da0c48c4Sopenharmony_ci count * sizeof (ElfW2(LIBELFBITS,Phdr))); 140da0c48c4Sopenharmony_ci if (result == NULL) 141da0c48c4Sopenharmony_ci __libelf_seterrno (ELF_E_NOMEM); 142da0c48c4Sopenharmony_ci else 143da0c48c4Sopenharmony_ci { 144da0c48c4Sopenharmony_ci /* Now set the result. */ 145da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).phdr = result; 146da0c48c4Sopenharmony_ci if (count >= PN_XNUM) 147da0c48c4Sopenharmony_ci { 148da0c48c4Sopenharmony_ci /* We have to write COUNT into the zeroth section's sh_info. */ 149da0c48c4Sopenharmony_ci if (elf->state.ELFW(elf,LIBELFBITS).scns.cnt == 0) 150da0c48c4Sopenharmony_ci { 151da0c48c4Sopenharmony_ci assert (elf->state.ELFW(elf,LIBELFBITS).scns.max > 0); 152da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).scns.cnt = 1; 153da0c48c4Sopenharmony_ci } 154da0c48c4Sopenharmony_ci scn0->shdr.ELFW(e,LIBELFBITS)->sh_info = count; 155da0c48c4Sopenharmony_ci scn0->shdr_flags |= ELF_F_DIRTY; 156da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).ehdr->e_phnum = PN_XNUM; 157da0c48c4Sopenharmony_ci } 158da0c48c4Sopenharmony_ci else 159da0c48c4Sopenharmony_ci /* Set the `e_phnum' member to the new value. */ 160da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).ehdr->e_phnum = count; 161da0c48c4Sopenharmony_ci /* Clear the whole memory. */ 162da0c48c4Sopenharmony_ci memset (result, '\0', count * sizeof (ElfW2(LIBELFBITS,Phdr))); 163da0c48c4Sopenharmony_ci /* Also set the size. */ 164da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).ehdr->e_phentsize = 165da0c48c4Sopenharmony_ci elf_typesize (LIBELFBITS, ELF_T_PHDR, 1); 166da0c48c4Sopenharmony_ci /* Remember we allocated the array and mark the structure is 167da0c48c4Sopenharmony_ci modified. */ 168da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).phdr_flags |= 169da0c48c4Sopenharmony_ci ELF_F_DIRTY | ELF_F_MALLOCED; 170da0c48c4Sopenharmony_ci /* We have to rewrite the entire file if the size of the 171da0c48c4Sopenharmony_ci program header is changed. */ 172da0c48c4Sopenharmony_ci elf->flags |= ELF_F_DIRTY; 173da0c48c4Sopenharmony_ci } 174da0c48c4Sopenharmony_ci } 175da0c48c4Sopenharmony_ci else 176da0c48c4Sopenharmony_ci { 177da0c48c4Sopenharmony_ci /* We have the same number of entries. Just clear the array. */ 178da0c48c4Sopenharmony_ci assert (elf->state.ELFW(elf,LIBELFBITS).ehdr->e_phentsize 179da0c48c4Sopenharmony_ci == elf_typesize (LIBELFBITS, ELF_T_PHDR, 1)); 180da0c48c4Sopenharmony_ci 181da0c48c4Sopenharmony_ci /* Mark the structure as modified. */ 182da0c48c4Sopenharmony_ci elf->state.ELFW(elf,LIBELFBITS).phdr_flags |= ELF_F_DIRTY; 183da0c48c4Sopenharmony_ci 184da0c48c4Sopenharmony_ci result = elf->state.ELFW(elf,LIBELFBITS).phdr; 185da0c48c4Sopenharmony_ci memset (result, '\0', count * sizeof (ElfW2(LIBELFBITS,Phdr))); 186da0c48c4Sopenharmony_ci } 187da0c48c4Sopenharmony_ci 188da0c48c4Sopenharmony_ci out: 189da0c48c4Sopenharmony_ci rwlock_unlock (elf->lock); 190da0c48c4Sopenharmony_ci 191da0c48c4Sopenharmony_ci return result; 192da0c48c4Sopenharmony_ci} 193da0c48c4Sopenharmony_ciINTDEF(elfw2(LIBELFBITS,newphdr)) 194