xref: /aosp_15_r20/external/elfutils/libasm/asm_newscn.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Create new section in output file.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2002-2011, 2016 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]>, 2002.
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 <stdlib.h>
36*7304104dSAndroid Build Coastguard Worker #include <string.h>
37*7304104dSAndroid Build Coastguard Worker 
38*7304104dSAndroid Build Coastguard Worker #include <libasmP.h>
39*7304104dSAndroid Build Coastguard Worker #include <libelf.h>
40*7304104dSAndroid Build Coastguard Worker #include <system.h>
41*7304104dSAndroid Build Coastguard Worker 
42*7304104dSAndroid Build Coastguard Worker 
43*7304104dSAndroid Build Coastguard Worker /* Memory for the default pattern.  The type uses a flexible array
44*7304104dSAndroid Build Coastguard Worker    which does work well with a static initializer.  Work around this by
45*7304104dSAndroid Build Coastguard Worker    wrapping it in a union, whose second member is a char array 1 byte larger
46*7304104dSAndroid Build Coastguard Worker    than struct FillPattern.  According to 6.7.9, this does what we need:
47*7304104dSAndroid Build Coastguard Worker 
48*7304104dSAndroid Build Coastguard Worker         If an object that has static or thread storage duration is not
49*7304104dSAndroid Build Coastguard Worker         initialized explicitly, then ... if it is a union, the first named
50*7304104dSAndroid Build Coastguard Worker         member is initialized (recursively) according to these rules, and
51*7304104dSAndroid Build Coastguard Worker         any padding is initialized to zero bits.  */
52*7304104dSAndroid Build Coastguard Worker 
53*7304104dSAndroid Build Coastguard Worker static const union
54*7304104dSAndroid Build Coastguard Worker {
55*7304104dSAndroid Build Coastguard Worker   struct FillPattern pattern;
56*7304104dSAndroid Build Coastguard Worker   char zeroes[sizeof(struct FillPattern) + 1];
57*7304104dSAndroid Build Coastguard Worker } xdefault_pattern =
58*7304104dSAndroid Build Coastguard Worker   {
59*7304104dSAndroid Build Coastguard Worker     .pattern =
60*7304104dSAndroid Build Coastguard Worker     {
61*7304104dSAndroid Build Coastguard Worker       .len = 1
62*7304104dSAndroid Build Coastguard Worker     },
63*7304104dSAndroid Build Coastguard Worker   };
64*7304104dSAndroid Build Coastguard Worker const struct FillPattern *__libasm_default_pattern = &xdefault_pattern.pattern;
65*7304104dSAndroid Build Coastguard Worker 
66*7304104dSAndroid Build Coastguard Worker 
67*7304104dSAndroid Build Coastguard Worker static AsmScn_t *
text_newscn(AsmScn_t * result,GElf_Word type,GElf_Xword flags)68*7304104dSAndroid Build Coastguard Worker text_newscn (AsmScn_t *result, GElf_Word type, GElf_Xword flags)
69*7304104dSAndroid Build Coastguard Worker {
70*7304104dSAndroid Build Coastguard Worker   /* Buffer where we construct the flag string.  */
71*7304104dSAndroid Build Coastguard Worker   char flagstr[sizeof (GElf_Xword) * 8 + 5];
72*7304104dSAndroid Build Coastguard Worker   char *wp = flagstr;
73*7304104dSAndroid Build Coastguard Worker   const char *typestr = "";
74*7304104dSAndroid Build Coastguard Worker 
75*7304104dSAndroid Build Coastguard Worker   /* Only write out the flag string if this is the first time the
76*7304104dSAndroid Build Coastguard Worker      section is selected.  Some assemblers cannot cope with the
77*7304104dSAndroid Build Coastguard Worker      .section pseudo-op otherwise.  */
78*7304104dSAndroid Build Coastguard Worker   wp = stpcpy (wp, ", \"");
79*7304104dSAndroid Build Coastguard Worker 
80*7304104dSAndroid Build Coastguard Worker   if (flags & SHF_WRITE)
81*7304104dSAndroid Build Coastguard Worker     *wp++ = 'w';
82*7304104dSAndroid Build Coastguard Worker   if (flags & SHF_ALLOC)
83*7304104dSAndroid Build Coastguard Worker     *wp++ = 'a';
84*7304104dSAndroid Build Coastguard Worker   if (flags & SHF_EXECINSTR)
85*7304104dSAndroid Build Coastguard Worker     *wp++ = 'x';
86*7304104dSAndroid Build Coastguard Worker   if (flags & SHF_MERGE)
87*7304104dSAndroid Build Coastguard Worker     *wp++ = 'M';
88*7304104dSAndroid Build Coastguard Worker   if (flags & SHF_STRINGS)
89*7304104dSAndroid Build Coastguard Worker     *wp++ = 'S';
90*7304104dSAndroid Build Coastguard Worker   if (flags & SHF_LINK_ORDER)
91*7304104dSAndroid Build Coastguard Worker     *wp++ = 'L';
92*7304104dSAndroid Build Coastguard Worker 
93*7304104dSAndroid Build Coastguard Worker   *wp++ = '"';
94*7304104dSAndroid Build Coastguard Worker 
95*7304104dSAndroid Build Coastguard Worker   if (type == SHT_PROGBITS)
96*7304104dSAndroid Build Coastguard Worker     typestr = ",@progbits";
97*7304104dSAndroid Build Coastguard Worker   else if (type == SHT_NOBITS)
98*7304104dSAndroid Build Coastguard Worker     typestr = ",@nobits";
99*7304104dSAndroid Build Coastguard Worker 
100*7304104dSAndroid Build Coastguard Worker   /* Terminate the string.  */
101*7304104dSAndroid Build Coastguard Worker   *wp = '\0';
102*7304104dSAndroid Build Coastguard Worker 
103*7304104dSAndroid Build Coastguard Worker   fprintf (result->ctx->out.file, "\t.section \"%s\"%s%s\n",
104*7304104dSAndroid Build Coastguard Worker 	   result->name, flagstr, typestr);
105*7304104dSAndroid Build Coastguard Worker 
106*7304104dSAndroid Build Coastguard Worker   return result;
107*7304104dSAndroid Build Coastguard Worker }
108*7304104dSAndroid Build Coastguard Worker 
109*7304104dSAndroid Build Coastguard Worker 
110*7304104dSAndroid Build Coastguard Worker static AsmScn_t *
binary_newscn(AsmScn_t * result,GElf_Word type,GElf_Xword flags,size_t scnname_len)111*7304104dSAndroid Build Coastguard Worker binary_newscn (AsmScn_t *result, GElf_Word type, GElf_Xword flags,
112*7304104dSAndroid Build Coastguard Worker 	       size_t scnname_len)
113*7304104dSAndroid Build Coastguard Worker {
114*7304104dSAndroid Build Coastguard Worker   GElf_Shdr shdr_mem;
115*7304104dSAndroid Build Coastguard Worker   GElf_Shdr *shdr;
116*7304104dSAndroid Build Coastguard Worker   Elf_Scn *scn;
117*7304104dSAndroid Build Coastguard Worker 
118*7304104dSAndroid Build Coastguard Worker   /* The initial subsection has the number zero.  */
119*7304104dSAndroid Build Coastguard Worker   result->subsection_id = 0;
120*7304104dSAndroid Build Coastguard Worker 
121*7304104dSAndroid Build Coastguard Worker   /* We start at offset zero.  */
122*7304104dSAndroid Build Coastguard Worker   result->offset = 0;
123*7304104dSAndroid Build Coastguard Worker   /* And generic alignment.  */
124*7304104dSAndroid Build Coastguard Worker   result->max_align = 1;
125*7304104dSAndroid Build Coastguard Worker 
126*7304104dSAndroid Build Coastguard Worker   /* No output yet.  */
127*7304104dSAndroid Build Coastguard Worker   result->content = NULL;
128*7304104dSAndroid Build Coastguard Worker 
129*7304104dSAndroid Build Coastguard Worker   /* Put the default fill pattern in place.  */
130*7304104dSAndroid Build Coastguard Worker   result->pattern = (struct FillPattern *) __libasm_default_pattern;
131*7304104dSAndroid Build Coastguard Worker 
132*7304104dSAndroid Build Coastguard Worker   /* There are no subsections so far.  */
133*7304104dSAndroid Build Coastguard Worker   result->subnext = NULL;
134*7304104dSAndroid Build Coastguard Worker 
135*7304104dSAndroid Build Coastguard Worker   /* Add the name to the section header string table.  */
136*7304104dSAndroid Build Coastguard Worker   result->data.main.strent = dwelf_strtab_add_len (result->ctx->section_strtab,
137*7304104dSAndroid Build Coastguard Worker 						   result->name, scnname_len);
138*7304104dSAndroid Build Coastguard Worker   assert (result->data.main.strent != NULL);
139*7304104dSAndroid Build Coastguard Worker 
140*7304104dSAndroid Build Coastguard Worker   /* Create the new ELF section.  */
141*7304104dSAndroid Build Coastguard Worker   result->data.main.scn = scn = elf_newscn (result->ctx->out.elf);
142*7304104dSAndroid Build Coastguard Worker   if (scn == NULL)
143*7304104dSAndroid Build Coastguard Worker     {
144*7304104dSAndroid Build Coastguard Worker       free (result);
145*7304104dSAndroid Build Coastguard Worker       __libasm_seterrno (ASM_E_LIBELF);
146*7304104dSAndroid Build Coastguard Worker       return NULL;
147*7304104dSAndroid Build Coastguard Worker     }
148*7304104dSAndroid Build Coastguard Worker 
149*7304104dSAndroid Build Coastguard Worker   /* Not part of a section group (yet).  */
150*7304104dSAndroid Build Coastguard Worker   result->data.main.next_in_group = NULL;
151*7304104dSAndroid Build Coastguard Worker 
152*7304104dSAndroid Build Coastguard Worker   /* Remember the flags.  */
153*7304104dSAndroid Build Coastguard Worker   shdr = gelf_getshdr (scn, &shdr_mem);
154*7304104dSAndroid Build Coastguard Worker 
155*7304104dSAndroid Build Coastguard Worker   shdr->sh_flags = flags;
156*7304104dSAndroid Build Coastguard Worker   result->type = shdr->sh_type = type;
157*7304104dSAndroid Build Coastguard Worker 
158*7304104dSAndroid Build Coastguard Worker   (void) gelf_update_shdr (scn, shdr);
159*7304104dSAndroid Build Coastguard Worker 
160*7304104dSAndroid Build Coastguard Worker   return result;
161*7304104dSAndroid Build Coastguard Worker }
162*7304104dSAndroid Build Coastguard Worker 
163*7304104dSAndroid Build Coastguard Worker 
164*7304104dSAndroid Build Coastguard Worker AsmScn_t *
asm_newscn(AsmCtx_t * ctx,const char * scnname,GElf_Word type,GElf_Xword flags)165*7304104dSAndroid Build Coastguard Worker asm_newscn (AsmCtx_t *ctx, const char *scnname, GElf_Word type,
166*7304104dSAndroid Build Coastguard Worker 	    GElf_Xword flags)
167*7304104dSAndroid Build Coastguard Worker {
168*7304104dSAndroid Build Coastguard Worker   size_t scnname_len = strlen (scnname) + 1;
169*7304104dSAndroid Build Coastguard Worker   AsmScn_t *result;
170*7304104dSAndroid Build Coastguard Worker 
171*7304104dSAndroid Build Coastguard Worker   /* If no context is given there might be an earlier error.  */
172*7304104dSAndroid Build Coastguard Worker   if (ctx == NULL)
173*7304104dSAndroid Build Coastguard Worker     return NULL;
174*7304104dSAndroid Build Coastguard Worker 
175*7304104dSAndroid Build Coastguard Worker   /* Check whether only flags are set which areselectable by the user.  */
176*7304104dSAndroid Build Coastguard Worker   if (unlikely ((flags & ~(SHF_WRITE | SHF_ALLOC | SHF_EXECINSTR | SHF_MERGE
177*7304104dSAndroid Build Coastguard Worker 			   | SHF_STRINGS | SHF_LINK_ORDER)) != 0)
178*7304104dSAndroid Build Coastguard Worker       /* We allow only two section types: data and data without file
179*7304104dSAndroid Build Coastguard Worker 	 representation.  */
180*7304104dSAndroid Build Coastguard Worker       || (type != SHT_PROGBITS && unlikely (type != SHT_NOBITS)))
181*7304104dSAndroid Build Coastguard Worker     {
182*7304104dSAndroid Build Coastguard Worker       __libasm_seterrno (ASM_E_INVALID);
183*7304104dSAndroid Build Coastguard Worker       return NULL;
184*7304104dSAndroid Build Coastguard Worker     }
185*7304104dSAndroid Build Coastguard Worker 
186*7304104dSAndroid Build Coastguard Worker   rwlock_wrlock (ctx->lock);
187*7304104dSAndroid Build Coastguard Worker 
188*7304104dSAndroid Build Coastguard Worker   /* This is a new section.  */
189*7304104dSAndroid Build Coastguard Worker   result = malloc (sizeof (AsmScn_t) + scnname_len);
190*7304104dSAndroid Build Coastguard Worker   if (result != NULL)
191*7304104dSAndroid Build Coastguard Worker     {
192*7304104dSAndroid Build Coastguard Worker       /* Add the name.  */
193*7304104dSAndroid Build Coastguard Worker       memcpy (result->name, scnname, scnname_len);
194*7304104dSAndroid Build Coastguard Worker 
195*7304104dSAndroid Build Coastguard Worker       /* Add the reference to the context.  */
196*7304104dSAndroid Build Coastguard Worker       result->ctx = ctx;
197*7304104dSAndroid Build Coastguard Worker 
198*7304104dSAndroid Build Coastguard Worker       /* Perform operations according to output mode.  */
199*7304104dSAndroid Build Coastguard Worker       result = (unlikely (ctx->textp)
200*7304104dSAndroid Build Coastguard Worker 		? text_newscn (result, type, flags)
201*7304104dSAndroid Build Coastguard Worker 		: binary_newscn (result, type, flags, scnname_len));
202*7304104dSAndroid Build Coastguard Worker 
203*7304104dSAndroid Build Coastguard Worker       /* If everything went well finally add the new section to the hash
204*7304104dSAndroid Build Coastguard Worker 	 table.  */
205*7304104dSAndroid Build Coastguard Worker       if (result != NULL)
206*7304104dSAndroid Build Coastguard Worker 	{
207*7304104dSAndroid Build Coastguard Worker 	  result->allnext = ctx->section_list;
208*7304104dSAndroid Build Coastguard Worker 	  ctx->section_list = result;
209*7304104dSAndroid Build Coastguard Worker 	}
210*7304104dSAndroid Build Coastguard Worker     }
211*7304104dSAndroid Build Coastguard Worker 
212*7304104dSAndroid Build Coastguard Worker   rwlock_unlock (ctx->lock);
213*7304104dSAndroid Build Coastguard Worker 
214*7304104dSAndroid Build Coastguard Worker   return result;
215*7304104dSAndroid Build Coastguard Worker }
216*7304104dSAndroid Build Coastguard Worker INTDEF(asm_newscn)
217