xref: /aosp_15_r20/external/elfutils/libelf/elf_newscn.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
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