1da0c48c4Sopenharmony_ci/* Append new section.
2da0c48c4Sopenharmony_ci   Copyright (C) 1998,1999,2000,2001,2002,2005,2009,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 <stdbool.h>
36da0c48c4Sopenharmony_ci#include <stddef.h>
37da0c48c4Sopenharmony_ci#include <stdlib.h>
38da0c48c4Sopenharmony_ci#include <string.h>
39da0c48c4Sopenharmony_ci
40da0c48c4Sopenharmony_ci#include "libelfP.h"
41da0c48c4Sopenharmony_ci
42da0c48c4Sopenharmony_ci
43da0c48c4Sopenharmony_ciElf_Scn *
44da0c48c4Sopenharmony_cielf_newscn (Elf *elf)
45da0c48c4Sopenharmony_ci{
46da0c48c4Sopenharmony_ci  Elf_Scn *result = NULL;
47da0c48c4Sopenharmony_ci  bool first = false;
48da0c48c4Sopenharmony_ci
49da0c48c4Sopenharmony_ci  if (elf == NULL)
50da0c48c4Sopenharmony_ci    return NULL;
51da0c48c4Sopenharmony_ci
52da0c48c4Sopenharmony_ci  /* We rely on the prefix of the `elf', `elf32', and `elf64' element
53da0c48c4Sopenharmony_ci     being the same.  */
54da0c48c4Sopenharmony_ci  assert (offsetof (Elf, state.elf.scns_last)
55da0c48c4Sopenharmony_ci	  == offsetof (Elf, state.elf32.scns_last));
56da0c48c4Sopenharmony_ci  assert (offsetof (Elf, state.elf.scns_last)
57da0c48c4Sopenharmony_ci	  == offsetof (Elf, state.elf64.scns_last));
58da0c48c4Sopenharmony_ci  assert (offsetof (Elf, state.elf32.scns)
59da0c48c4Sopenharmony_ci	  == offsetof (Elf, state.elf64.scns));
60da0c48c4Sopenharmony_ci
61da0c48c4Sopenharmony_ci  rwlock_wrlock (elf->lock);
62da0c48c4Sopenharmony_ci
63da0c48c4Sopenharmony_ci again:
64da0c48c4Sopenharmony_ci  if (elf->state.elf.scns_last->cnt < elf->state.elf.scns_last->max)
65da0c48c4Sopenharmony_ci    {
66da0c48c4Sopenharmony_ci      result = &elf->state.elf.scns_last->data[elf->state.elf.scns_last->cnt];
67da0c48c4Sopenharmony_ci
68da0c48c4Sopenharmony_ci      if (++elf->state.elf.scns_last->cnt == 1
69da0c48c4Sopenharmony_ci	  && (elf->state.elf.scns_last
70da0c48c4Sopenharmony_ci	      == (elf->class == ELFCLASS32
71da0c48c4Sopenharmony_ci		  || (offsetof (Elf, state.elf32.scns)
72da0c48c4Sopenharmony_ci		      == offsetof (Elf, state.elf64.scns))
73da0c48c4Sopenharmony_ci		  ? &elf->state.elf32.scns : &elf->state.elf64.scns)))
74da0c48c4Sopenharmony_ci	/* This is zeroth section.  */
75da0c48c4Sopenharmony_ci	first = true;
76da0c48c4Sopenharmony_ci      else
77da0c48c4Sopenharmony_ci	{
78da0c48c4Sopenharmony_ci	  assert (elf->state.elf.scns_last->cnt > 1);
79da0c48c4Sopenharmony_ci	  result->index = result[-1].index + 1;
80da0c48c4Sopenharmony_ci	}
81da0c48c4Sopenharmony_ci    }
82da0c48c4Sopenharmony_ci  else
83da0c48c4Sopenharmony_ci    {
84da0c48c4Sopenharmony_ci      /* We must allocate a new element.  */
85da0c48c4Sopenharmony_ci      Elf_ScnList *newp = NULL;
86da0c48c4Sopenharmony_ci
87da0c48c4Sopenharmony_ci      assert (elf->state.elf.scnincr > 0);
88da0c48c4Sopenharmony_ci
89da0c48c4Sopenharmony_ci      if (
90da0c48c4Sopenharmony_ci#if SIZE_MAX <= 4294967295U
91da0c48c4Sopenharmony_ci	  likely (elf->state.elf.scnincr
92da0c48c4Sopenharmony_ci		  < SIZE_MAX / 2 / sizeof (Elf_Scn) - sizeof (Elf_ScnList))
93da0c48c4Sopenharmony_ci#else
94da0c48c4Sopenharmony_ci	  1
95da0c48c4Sopenharmony_ci#endif
96da0c48c4Sopenharmony_ci	  )
97da0c48c4Sopenharmony_ci      newp = calloc (sizeof (Elf_ScnList)
98da0c48c4Sopenharmony_ci		     + ((elf->state.elf.scnincr *= 2)
99da0c48c4Sopenharmony_ci			* sizeof (Elf_Scn)), 1);
100da0c48c4Sopenharmony_ci      if (newp == NULL)
101da0c48c4Sopenharmony_ci	{
102da0c48c4Sopenharmony_ci	  __libelf_seterrno (ELF_E_NOMEM);
103da0c48c4Sopenharmony_ci	  goto out;
104da0c48c4Sopenharmony_ci	}
105da0c48c4Sopenharmony_ci
106da0c48c4Sopenharmony_ci      result = &newp->data[0];
107da0c48c4Sopenharmony_ci
108da0c48c4Sopenharmony_ci      /* One section used.  */
109da0c48c4Sopenharmony_ci      ++newp->cnt;
110da0c48c4Sopenharmony_ci
111da0c48c4Sopenharmony_ci      /* This is the number of sections we allocated.  */
112da0c48c4Sopenharmony_ci      newp->max = elf->state.elf.scnincr;
113da0c48c4Sopenharmony_ci
114da0c48c4Sopenharmony_ci      /* Remember the index for the first section in this block.  */
115da0c48c4Sopenharmony_ci      newp->data[0].index
116da0c48c4Sopenharmony_ci	= 1 + elf->state.elf.scns_last->data[elf->state.elf.scns_last->max - 1].index;
117da0c48c4Sopenharmony_ci
118da0c48c4Sopenharmony_ci      /* Enqueue the new list element.  */
119da0c48c4Sopenharmony_ci      elf->state.elf.scns_last = elf->state.elf.scns_last->next = newp;
120da0c48c4Sopenharmony_ci    }
121da0c48c4Sopenharmony_ci
122da0c48c4Sopenharmony_ci  /* Create a section header for this section.  */
123da0c48c4Sopenharmony_ci  if (elf->class == ELFCLASS32)
124da0c48c4Sopenharmony_ci    {
125da0c48c4Sopenharmony_ci      result->shdr.e32 = calloc (1, sizeof (Elf32_Shdr));
126da0c48c4Sopenharmony_ci      if (result->shdr.e32 == NULL)
127da0c48c4Sopenharmony_ci	{
128da0c48c4Sopenharmony_ci	  __libelf_seterrno (ELF_E_NOMEM);
129da0c48c4Sopenharmony_ci	  goto out;
130da0c48c4Sopenharmony_ci	}
131da0c48c4Sopenharmony_ci    }
132da0c48c4Sopenharmony_ci  else
133da0c48c4Sopenharmony_ci    {
134da0c48c4Sopenharmony_ci      result->shdr.e64 = calloc (1, sizeof (Elf64_Shdr));
135da0c48c4Sopenharmony_ci      if (result->shdr.e64 == NULL)
136da0c48c4Sopenharmony_ci	{
137da0c48c4Sopenharmony_ci	  __libelf_seterrno (ELF_E_NOMEM);
138da0c48c4Sopenharmony_ci	  goto out;
139da0c48c4Sopenharmony_ci	}
140da0c48c4Sopenharmony_ci    }
141da0c48c4Sopenharmony_ci
142da0c48c4Sopenharmony_ci  result->elf = elf;
143da0c48c4Sopenharmony_ci  result->shdr_flags = ELF_F_DIRTY | ELF_F_MALLOCED;
144da0c48c4Sopenharmony_ci  result->list = elf->state.elf.scns_last;
145da0c48c4Sopenharmony_ci
146da0c48c4Sopenharmony_ci  /* Initialize the data part.  */
147da0c48c4Sopenharmony_ci  result->data_read = 1;
148da0c48c4Sopenharmony_ci  if (unlikely (first))
149da0c48c4Sopenharmony_ci    {
150da0c48c4Sopenharmony_ci      /* For the first section we mark the data as already available.  */
151da0c48c4Sopenharmony_ci      //result->data_list_rear = &result->data_list;
152da0c48c4Sopenharmony_ci      first = false;
153da0c48c4Sopenharmony_ci      goto again;
154da0c48c4Sopenharmony_ci    }
155da0c48c4Sopenharmony_ci
156da0c48c4Sopenharmony_ci  result->flags |= ELF_F_DIRTY;
157da0c48c4Sopenharmony_ci
158da0c48c4Sopenharmony_ci out:
159da0c48c4Sopenharmony_ci  rwlock_unlock (elf->lock);
160da0c48c4Sopenharmony_ci
161da0c48c4Sopenharmony_ci  return result;
162da0c48c4Sopenharmony_ci}
163