xref: /aosp_15_r20/external/elfutils/libasm/asm_begin.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Create descriptor for assembling.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2002, 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 <errno.h>
36*7304104dSAndroid Build Coastguard Worker #include <stdio.h>
37*7304104dSAndroid Build Coastguard Worker #include <stdio_ext.h>
38*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
39*7304104dSAndroid Build Coastguard Worker #include <string.h>
40*7304104dSAndroid Build Coastguard Worker 
41*7304104dSAndroid Build Coastguard Worker #include <gelf.h>
42*7304104dSAndroid Build Coastguard Worker #include "libasmP.h"
43*7304104dSAndroid Build Coastguard Worker 
44*7304104dSAndroid Build Coastguard Worker 
45*7304104dSAndroid Build Coastguard Worker static AsmCtx_t *
prepare_text_output(AsmCtx_t * result)46*7304104dSAndroid Build Coastguard Worker prepare_text_output (AsmCtx_t *result)
47*7304104dSAndroid Build Coastguard Worker {
48*7304104dSAndroid Build Coastguard Worker   if (result->fd == -1)
49*7304104dSAndroid Build Coastguard Worker     result->out.file = stdout;
50*7304104dSAndroid Build Coastguard Worker   else
51*7304104dSAndroid Build Coastguard Worker     {
52*7304104dSAndroid Build Coastguard Worker       result->out.file = fdopen (result->fd, "a");
53*7304104dSAndroid Build Coastguard Worker       if (result->out.file == NULL)
54*7304104dSAndroid Build Coastguard Worker 	{
55*7304104dSAndroid Build Coastguard Worker 	  close (result->fd);
56*7304104dSAndroid Build Coastguard Worker 	  free (result);
57*7304104dSAndroid Build Coastguard Worker 	  result = NULL;
58*7304104dSAndroid Build Coastguard Worker 	}
59*7304104dSAndroid Build Coastguard Worker       else
60*7304104dSAndroid Build Coastguard Worker 	__fsetlocking (result->out.file, FSETLOCKING_BYCALLER);
61*7304104dSAndroid Build Coastguard Worker     }
62*7304104dSAndroid Build Coastguard Worker 
63*7304104dSAndroid Build Coastguard Worker   return result;
64*7304104dSAndroid Build Coastguard Worker }
65*7304104dSAndroid Build Coastguard Worker 
66*7304104dSAndroid Build Coastguard Worker 
67*7304104dSAndroid Build Coastguard Worker static AsmCtx_t *
prepare_binary_output(AsmCtx_t * result,Ebl * ebl)68*7304104dSAndroid Build Coastguard Worker prepare_binary_output (AsmCtx_t *result, Ebl *ebl)
69*7304104dSAndroid Build Coastguard Worker {
70*7304104dSAndroid Build Coastguard Worker   GElf_Ehdr *ehdr;
71*7304104dSAndroid Build Coastguard Worker   GElf_Ehdr ehdr_mem;
72*7304104dSAndroid Build Coastguard Worker 
73*7304104dSAndroid Build Coastguard Worker   /* Create the ELF descriptor for the file.  */
74*7304104dSAndroid Build Coastguard Worker   result->out.elf = elf_begin (result->fd, ELF_C_WRITE_MMAP, NULL);
75*7304104dSAndroid Build Coastguard Worker   if (result->out.elf == NULL)
76*7304104dSAndroid Build Coastguard Worker     {
77*7304104dSAndroid Build Coastguard Worker     err_libelf:
78*7304104dSAndroid Build Coastguard Worker       unlink (result->tmp_fname);
79*7304104dSAndroid Build Coastguard Worker       close (result->fd);
80*7304104dSAndroid Build Coastguard Worker       free (result);
81*7304104dSAndroid Build Coastguard Worker       __libasm_seterrno (ASM_E_LIBELF);
82*7304104dSAndroid Build Coastguard Worker       return NULL;
83*7304104dSAndroid Build Coastguard Worker     }
84*7304104dSAndroid Build Coastguard Worker 
85*7304104dSAndroid Build Coastguard Worker   /* Create the ELF header for the output file.  */
86*7304104dSAndroid Build Coastguard Worker   int class = ebl_get_elfclass (ebl);
87*7304104dSAndroid Build Coastguard Worker   if (gelf_newehdr (result->out.elf, class) == 0)
88*7304104dSAndroid Build Coastguard Worker     goto err_libelf;
89*7304104dSAndroid Build Coastguard Worker 
90*7304104dSAndroid Build Coastguard Worker   ehdr = gelf_getehdr (result->out.elf, &ehdr_mem);
91*7304104dSAndroid Build Coastguard Worker   /* If this failed we are in trouble.  */
92*7304104dSAndroid Build Coastguard Worker   assert (ehdr != NULL);
93*7304104dSAndroid Build Coastguard Worker 
94*7304104dSAndroid Build Coastguard Worker   /* We create an object file.  */
95*7304104dSAndroid Build Coastguard Worker   ehdr->e_type = ET_REL;
96*7304104dSAndroid Build Coastguard Worker   /* Set the ELF version.  */
97*7304104dSAndroid Build Coastguard Worker   ehdr->e_version = EV_CURRENT;
98*7304104dSAndroid Build Coastguard Worker 
99*7304104dSAndroid Build Coastguard Worker   /* Use the machine, class, and endianness values from the Ebl descriptor.  */
100*7304104dSAndroid Build Coastguard Worker   ehdr->e_machine = ebl_get_elfmachine (ebl);
101*7304104dSAndroid Build Coastguard Worker   ehdr->e_ident[EI_CLASS] = class;
102*7304104dSAndroid Build Coastguard Worker   ehdr->e_ident[EI_DATA] = ebl_get_elfdata (ebl);
103*7304104dSAndroid Build Coastguard Worker 
104*7304104dSAndroid Build Coastguard Worker   memcpy (&ehdr->e_ident[EI_MAG0], ELFMAG, SELFMAG);
105*7304104dSAndroid Build Coastguard Worker 
106*7304104dSAndroid Build Coastguard Worker   /* Write the ELF header information back.  */
107*7304104dSAndroid Build Coastguard Worker   (void) gelf_update_ehdr (result->out.elf, ehdr);
108*7304104dSAndroid Build Coastguard Worker 
109*7304104dSAndroid Build Coastguard Worker   /* No section so far.  */
110*7304104dSAndroid Build Coastguard Worker   result->section_list = NULL;
111*7304104dSAndroid Build Coastguard Worker 
112*7304104dSAndroid Build Coastguard Worker   /* Initialize the hash table.  */
113*7304104dSAndroid Build Coastguard Worker   asm_symbol_tab_init (&result->symbol_tab, 67);
114*7304104dSAndroid Build Coastguard Worker   result->nsymbol_tab = 0;
115*7304104dSAndroid Build Coastguard Worker   /* And the string tables.  */
116*7304104dSAndroid Build Coastguard Worker   result->section_strtab = dwelf_strtab_init (true);
117*7304104dSAndroid Build Coastguard Worker   result->symbol_strtab = dwelf_strtab_init (true);
118*7304104dSAndroid Build Coastguard Worker 
119*7304104dSAndroid Build Coastguard Worker   /* We have no section groups so far.  */
120*7304104dSAndroid Build Coastguard Worker   result->groups = NULL;
121*7304104dSAndroid Build Coastguard Worker   result->ngroups = 0;
122*7304104dSAndroid Build Coastguard Worker 
123*7304104dSAndroid Build Coastguard Worker   return result;
124*7304104dSAndroid Build Coastguard Worker }
125*7304104dSAndroid Build Coastguard Worker 
126*7304104dSAndroid Build Coastguard Worker 
127*7304104dSAndroid Build Coastguard Worker AsmCtx_t *
asm_begin(const char * fname,Ebl * ebl,bool textp)128*7304104dSAndroid Build Coastguard Worker asm_begin (const char *fname, Ebl *ebl, bool textp)
129*7304104dSAndroid Build Coastguard Worker {
130*7304104dSAndroid Build Coastguard Worker   if (fname == NULL && ! textp)
131*7304104dSAndroid Build Coastguard Worker     return NULL;
132*7304104dSAndroid Build Coastguard Worker 
133*7304104dSAndroid Build Coastguard Worker   size_t fname_len = fname != NULL ? strlen (fname) : 0;
134*7304104dSAndroid Build Coastguard Worker 
135*7304104dSAndroid Build Coastguard Worker   /* Create the file descriptor.  We do not generate the output file
136*7304104dSAndroid Build Coastguard Worker      right away.  Instead we create a temporary file in the same
137*7304104dSAndroid Build Coastguard Worker      directory which, if everything goes alright, will replace a
138*7304104dSAndroid Build Coastguard Worker      possibly existing file with the given name.  */
139*7304104dSAndroid Build Coastguard Worker   AsmCtx_t *result = malloc (sizeof (AsmCtx_t) + 2 * fname_len + 9);
140*7304104dSAndroid Build Coastguard Worker   if (result == NULL)
141*7304104dSAndroid Build Coastguard Worker     return NULL;
142*7304104dSAndroid Build Coastguard Worker 
143*7304104dSAndroid Build Coastguard Worker       /* Initialize the lock.  */
144*7304104dSAndroid Build Coastguard Worker       rwlock_init (result->lock);
145*7304104dSAndroid Build Coastguard Worker 
146*7304104dSAndroid Build Coastguard Worker   if (fname != NULL)
147*7304104dSAndroid Build Coastguard Worker     {
148*7304104dSAndroid Build Coastguard Worker       /* Create the name of the temporary file.  */
149*7304104dSAndroid Build Coastguard Worker       result->fname = stpcpy (mempcpy (result->tmp_fname, fname, fname_len),
150*7304104dSAndroid Build Coastguard Worker 			      ".XXXXXX") + 1;
151*7304104dSAndroid Build Coastguard Worker       memcpy (result->fname, fname, fname_len + 1);
152*7304104dSAndroid Build Coastguard Worker 
153*7304104dSAndroid Build Coastguard Worker       /* Create the temporary file.  */
154*7304104dSAndroid Build Coastguard Worker       result->fd = mkstemp (result->tmp_fname);
155*7304104dSAndroid Build Coastguard Worker       if (result->fd == -1)
156*7304104dSAndroid Build Coastguard Worker 	{
157*7304104dSAndroid Build Coastguard Worker 	  int save_errno = errno;
158*7304104dSAndroid Build Coastguard Worker 	  free (result);
159*7304104dSAndroid Build Coastguard Worker 	  __libasm_seterrno (ASM_E_CANNOT_CREATE);
160*7304104dSAndroid Build Coastguard Worker 	  errno = save_errno;
161*7304104dSAndroid Build Coastguard Worker 	  return NULL;
162*7304104dSAndroid Build Coastguard Worker 	}
163*7304104dSAndroid Build Coastguard Worker     }
164*7304104dSAndroid Build Coastguard Worker   else
165*7304104dSAndroid Build Coastguard Worker     result->fd = -1;
166*7304104dSAndroid Build Coastguard Worker 
167*7304104dSAndroid Build Coastguard Worker   /* Initialize the counter for temporary symbols.  */
168*7304104dSAndroid Build Coastguard Worker   result->tempsym_count = 0;
169*7304104dSAndroid Build Coastguard Worker 
170*7304104dSAndroid Build Coastguard Worker   /* Now we differentiate between textual and binary output.   */
171*7304104dSAndroid Build Coastguard Worker   result->textp = textp;
172*7304104dSAndroid Build Coastguard Worker   if (textp)
173*7304104dSAndroid Build Coastguard Worker     result = prepare_text_output (result);
174*7304104dSAndroid Build Coastguard Worker   else
175*7304104dSAndroid Build Coastguard Worker     result = prepare_binary_output (result, ebl);
176*7304104dSAndroid Build Coastguard Worker 
177*7304104dSAndroid Build Coastguard Worker   return result;
178*7304104dSAndroid Build Coastguard Worker }
179