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