1*7304104dSAndroid Build Coastguard Worker /* Append new section.
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 1998,1999,2000,2001,2002,2005,2009,2014,2015 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker Written by Ulrich Drepper <[email protected]>, 1998.
5*7304104dSAndroid Build Coastguard Worker
6*7304104dSAndroid Build Coastguard Worker This file is free software; you can redistribute it and/or modify
7*7304104dSAndroid Build Coastguard Worker it under the terms of either
8*7304104dSAndroid Build Coastguard Worker
9*7304104dSAndroid Build Coastguard Worker * the GNU Lesser General Public License as published by the Free
10*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 3 of the License, or (at
11*7304104dSAndroid Build Coastguard Worker your option) any later version
12*7304104dSAndroid Build Coastguard Worker
13*7304104dSAndroid Build Coastguard Worker or
14*7304104dSAndroid Build Coastguard Worker
15*7304104dSAndroid Build Coastguard Worker * the GNU General Public License as published by the Free
16*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 2 of the License, or (at
17*7304104dSAndroid Build Coastguard Worker your option) any later version
18*7304104dSAndroid Build Coastguard Worker
19*7304104dSAndroid Build Coastguard Worker or both in parallel, as here.
20*7304104dSAndroid Build Coastguard Worker
21*7304104dSAndroid Build Coastguard Worker elfutils is distributed in the hope that it will be useful, but
22*7304104dSAndroid Build Coastguard Worker WITHOUT ANY WARRANTY; without even the implied warranty of
23*7304104dSAndroid Build Coastguard Worker MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
24*7304104dSAndroid Build Coastguard Worker General Public License for more details.
25*7304104dSAndroid Build Coastguard Worker
26*7304104dSAndroid Build Coastguard Worker You should have received copies of the GNU General Public License and
27*7304104dSAndroid Build Coastguard Worker the GNU Lesser General Public License along with this program. If
28*7304104dSAndroid Build Coastguard Worker not, see <http://www.gnu.org/licenses/>. */
29*7304104dSAndroid Build Coastguard Worker
30*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
31*7304104dSAndroid Build Coastguard Worker # include <config.h>
32*7304104dSAndroid Build Coastguard Worker #endif
33*7304104dSAndroid Build Coastguard Worker
34*7304104dSAndroid Build Coastguard Worker #include <assert.h>
35*7304104dSAndroid Build Coastguard Worker #include <stdbool.h>
36*7304104dSAndroid Build Coastguard Worker #include <stddef.h>
37*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
38*7304104dSAndroid Build Coastguard Worker #include <string.h>
39*7304104dSAndroid Build Coastguard Worker
40*7304104dSAndroid Build Coastguard Worker #include "libelfP.h"
41*7304104dSAndroid Build Coastguard Worker
42*7304104dSAndroid Build Coastguard Worker
43*7304104dSAndroid Build Coastguard Worker Elf_Scn *
elf_newscn(Elf * elf)44*7304104dSAndroid Build Coastguard Worker elf_newscn (Elf *elf)
45*7304104dSAndroid Build Coastguard Worker {
46*7304104dSAndroid Build Coastguard Worker Elf_Scn *result = NULL;
47*7304104dSAndroid Build Coastguard Worker bool first = false;
48*7304104dSAndroid Build Coastguard Worker
49*7304104dSAndroid Build Coastguard Worker if (elf == NULL)
50*7304104dSAndroid Build Coastguard Worker return NULL;
51*7304104dSAndroid Build Coastguard Worker
52*7304104dSAndroid Build Coastguard Worker /* We rely on the prefix of the `elf', `elf32', and `elf64' element
53*7304104dSAndroid Build Coastguard Worker being the same. */
54*7304104dSAndroid Build Coastguard Worker assert (offsetof (Elf, state.elf.scns_last)
55*7304104dSAndroid Build Coastguard Worker == offsetof (Elf, state.elf32.scns_last));
56*7304104dSAndroid Build Coastguard Worker assert (offsetof (Elf, state.elf.scns_last)
57*7304104dSAndroid Build Coastguard Worker == offsetof (Elf, state.elf64.scns_last));
58*7304104dSAndroid Build Coastguard Worker assert (offsetof (Elf, state.elf32.scns)
59*7304104dSAndroid Build Coastguard Worker == offsetof (Elf, state.elf64.scns));
60*7304104dSAndroid Build Coastguard Worker
61*7304104dSAndroid Build Coastguard Worker rwlock_wrlock (elf->lock);
62*7304104dSAndroid Build Coastguard Worker
63*7304104dSAndroid Build Coastguard Worker again:
64*7304104dSAndroid Build Coastguard Worker if (elf->state.elf.scns_last->cnt < elf->state.elf.scns_last->max)
65*7304104dSAndroid Build Coastguard Worker {
66*7304104dSAndroid Build Coastguard Worker result = &elf->state.elf.scns_last->data[elf->state.elf.scns_last->cnt];
67*7304104dSAndroid Build Coastguard Worker
68*7304104dSAndroid Build Coastguard Worker if (++elf->state.elf.scns_last->cnt == 1
69*7304104dSAndroid Build Coastguard Worker && (elf->state.elf.scns_last
70*7304104dSAndroid Build Coastguard Worker == (elf->class == ELFCLASS32
71*7304104dSAndroid Build Coastguard Worker || (offsetof (Elf, state.elf32.scns)
72*7304104dSAndroid Build Coastguard Worker == offsetof (Elf, state.elf64.scns))
73*7304104dSAndroid Build Coastguard Worker ? &elf->state.elf32.scns : &elf->state.elf64.scns)))
74*7304104dSAndroid Build Coastguard Worker /* This is zeroth section. */
75*7304104dSAndroid Build Coastguard Worker first = true;
76*7304104dSAndroid Build Coastguard Worker else
77*7304104dSAndroid Build Coastguard Worker {
78*7304104dSAndroid Build Coastguard Worker assert (elf->state.elf.scns_last->cnt > 1);
79*7304104dSAndroid Build Coastguard Worker result->index = result[-1].index + 1;
80*7304104dSAndroid Build Coastguard Worker }
81*7304104dSAndroid Build Coastguard Worker }
82*7304104dSAndroid Build Coastguard Worker else
83*7304104dSAndroid Build Coastguard Worker {
84*7304104dSAndroid Build Coastguard Worker /* We must allocate a new element. */
85*7304104dSAndroid Build Coastguard Worker Elf_ScnList *newp = NULL;
86*7304104dSAndroid Build Coastguard Worker
87*7304104dSAndroid Build Coastguard Worker assert (elf->state.elf.scnincr > 0);
88*7304104dSAndroid Build Coastguard Worker
89*7304104dSAndroid Build Coastguard Worker if (
90*7304104dSAndroid Build Coastguard Worker #if SIZE_MAX <= 4294967295U
91*7304104dSAndroid Build Coastguard Worker likely (elf->state.elf.scnincr
92*7304104dSAndroid Build Coastguard Worker < SIZE_MAX / 2 / sizeof (Elf_Scn) - sizeof (Elf_ScnList))
93*7304104dSAndroid Build Coastguard Worker #else
94*7304104dSAndroid Build Coastguard Worker 1
95*7304104dSAndroid Build Coastguard Worker #endif
96*7304104dSAndroid Build Coastguard Worker )
97*7304104dSAndroid Build Coastguard Worker newp = calloc (1, sizeof (Elf_ScnList)
98*7304104dSAndroid Build Coastguard Worker + ((elf->state.elf.scnincr *= 2)
99*7304104dSAndroid Build Coastguard Worker * sizeof (Elf_Scn)));
100*7304104dSAndroid Build Coastguard Worker if (newp == NULL)
101*7304104dSAndroid Build Coastguard Worker {
102*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_NOMEM);
103*7304104dSAndroid Build Coastguard Worker goto out;
104*7304104dSAndroid Build Coastguard Worker }
105*7304104dSAndroid Build Coastguard Worker
106*7304104dSAndroid Build Coastguard Worker result = &newp->data[0];
107*7304104dSAndroid Build Coastguard Worker
108*7304104dSAndroid Build Coastguard Worker /* One section used. */
109*7304104dSAndroid Build Coastguard Worker ++newp->cnt;
110*7304104dSAndroid Build Coastguard Worker
111*7304104dSAndroid Build Coastguard Worker /* This is the number of sections we allocated. */
112*7304104dSAndroid Build Coastguard Worker newp->max = elf->state.elf.scnincr;
113*7304104dSAndroid Build Coastguard Worker
114*7304104dSAndroid Build Coastguard Worker /* Remember the index for the first section in this block. */
115*7304104dSAndroid Build Coastguard Worker newp->data[0].index
116*7304104dSAndroid Build Coastguard Worker = 1 + elf->state.elf.scns_last->data[elf->state.elf.scns_last->max - 1].index;
117*7304104dSAndroid Build Coastguard Worker
118*7304104dSAndroid Build Coastguard Worker /* Enqueue the new list element. */
119*7304104dSAndroid Build Coastguard Worker elf->state.elf.scns_last = elf->state.elf.scns_last->next = newp;
120*7304104dSAndroid Build Coastguard Worker }
121*7304104dSAndroid Build Coastguard Worker
122*7304104dSAndroid Build Coastguard Worker /* Create a section header for this section. */
123*7304104dSAndroid Build Coastguard Worker if (elf->class == ELFCLASS32)
124*7304104dSAndroid Build Coastguard Worker {
125*7304104dSAndroid Build Coastguard Worker result->shdr.e32 = calloc (1, sizeof (Elf32_Shdr));
126*7304104dSAndroid Build Coastguard Worker if (result->shdr.e32 == NULL)
127*7304104dSAndroid Build Coastguard Worker {
128*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_NOMEM);
129*7304104dSAndroid Build Coastguard Worker goto out;
130*7304104dSAndroid Build Coastguard Worker }
131*7304104dSAndroid Build Coastguard Worker }
132*7304104dSAndroid Build Coastguard Worker else
133*7304104dSAndroid Build Coastguard Worker {
134*7304104dSAndroid Build Coastguard Worker result->shdr.e64 = calloc (1, sizeof (Elf64_Shdr));
135*7304104dSAndroid Build Coastguard Worker if (result->shdr.e64 == NULL)
136*7304104dSAndroid Build Coastguard Worker {
137*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_NOMEM);
138*7304104dSAndroid Build Coastguard Worker goto out;
139*7304104dSAndroid Build Coastguard Worker }
140*7304104dSAndroid Build Coastguard Worker }
141*7304104dSAndroid Build Coastguard Worker
142*7304104dSAndroid Build Coastguard Worker result->elf = elf;
143*7304104dSAndroid Build Coastguard Worker result->shdr_flags = ELF_F_DIRTY | ELF_F_MALLOCED;
144*7304104dSAndroid Build Coastguard Worker result->list = elf->state.elf.scns_last;
145*7304104dSAndroid Build Coastguard Worker
146*7304104dSAndroid Build Coastguard Worker /* Initialize the data part. */
147*7304104dSAndroid Build Coastguard Worker result->data_read = 1;
148*7304104dSAndroid Build Coastguard Worker if (unlikely (first))
149*7304104dSAndroid Build Coastguard Worker {
150*7304104dSAndroid Build Coastguard Worker /* For the first section we mark the data as already available. */
151*7304104dSAndroid Build Coastguard Worker //result->data_list_rear = &result->data_list;
152*7304104dSAndroid Build Coastguard Worker first = false;
153*7304104dSAndroid Build Coastguard Worker goto again;
154*7304104dSAndroid Build Coastguard Worker }
155*7304104dSAndroid Build Coastguard Worker
156*7304104dSAndroid Build Coastguard Worker result->flags |= ELF_F_DIRTY;
157*7304104dSAndroid Build Coastguard Worker
158*7304104dSAndroid Build Coastguard Worker out:
159*7304104dSAndroid Build Coastguard Worker rwlock_unlock (elf->lock);
160*7304104dSAndroid Build Coastguard Worker
161*7304104dSAndroid Build Coastguard Worker return result;
162*7304104dSAndroid Build Coastguard Worker }
163