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