xref: /aosp_15_r20/external/elfutils/libasm/asm_newsym.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Define new symbol for current position in given section.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2002, 2005, 2016, 2017 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 <inttypes.h>
35*7304104dSAndroid Build Coastguard Worker #include <stdio.h>
36*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
37*7304104dSAndroid Build Coastguard Worker #include <string.h>
38*7304104dSAndroid Build Coastguard Worker 
39*7304104dSAndroid Build Coastguard Worker #include <libasmP.h>
40*7304104dSAndroid Build Coastguard Worker #include <system.h>
41*7304104dSAndroid Build Coastguard Worker 
42*7304104dSAndroid Build Coastguard Worker 
43*7304104dSAndroid Build Coastguard Worker AsmSym_t *
asm_newsym(AsmScn_t * asmscn,const char * name,GElf_Xword size,int type,int binding)44*7304104dSAndroid Build Coastguard Worker asm_newsym (AsmScn_t *asmscn, const char *name, GElf_Xword size,
45*7304104dSAndroid Build Coastguard Worker 	    int type, int binding)
46*7304104dSAndroid Build Coastguard Worker {
47*7304104dSAndroid Build Coastguard Worker /* We don't really expect labels with many digits, but in theory it could
48*7304104dSAndroid Build Coastguard Worker    be 10 digits (plus ".L" and a zero terminator).  */
49*7304104dSAndroid Build Coastguard Worker #define TEMPSYMLEN 13
50*7304104dSAndroid Build Coastguard Worker   char tempsym[TEMPSYMLEN];
51*7304104dSAndroid Build Coastguard Worker   AsmSym_t *result;
52*7304104dSAndroid Build Coastguard Worker 
53*7304104dSAndroid Build Coastguard Worker   if (asmscn == NULL)
54*7304104dSAndroid Build Coastguard Worker     /* Something went wrong before.  */
55*7304104dSAndroid Build Coastguard Worker     return NULL;
56*7304104dSAndroid Build Coastguard Worker 
57*7304104dSAndroid Build Coastguard Worker   /* Generate a temporary symbol if necessary.  */
58*7304104dSAndroid Build Coastguard Worker   if (name == NULL)
59*7304104dSAndroid Build Coastguard Worker     {
60*7304104dSAndroid Build Coastguard Worker       /* If a local symbol name is created the symbol better have
61*7304104dSAndroid Build Coastguard Worker 	 local binding.  */
62*7304104dSAndroid Build Coastguard Worker       if (binding != STB_LOCAL)
63*7304104dSAndroid Build Coastguard Worker 	{
64*7304104dSAndroid Build Coastguard Worker 	  __libasm_seterrno (ASM_E_INVALID);
65*7304104dSAndroid Build Coastguard Worker 	  return NULL;
66*7304104dSAndroid Build Coastguard Worker 	}
67*7304104dSAndroid Build Coastguard Worker 
68*7304104dSAndroid Build Coastguard Worker       // XXX This requires getting the format from the machine backend.  */
69*7304104dSAndroid Build Coastguard Worker       snprintf (tempsym, TEMPSYMLEN, ".L%07u", asmscn->ctx->tempsym_count++);
70*7304104dSAndroid Build Coastguard Worker 
71*7304104dSAndroid Build Coastguard Worker       name = tempsym;
72*7304104dSAndroid Build Coastguard Worker     }
73*7304104dSAndroid Build Coastguard Worker 
74*7304104dSAndroid Build Coastguard Worker   size_t name_len = strlen (name) + 1;
75*7304104dSAndroid Build Coastguard Worker 
76*7304104dSAndroid Build Coastguard Worker   result = malloc (sizeof (AsmSym_t) + name_len);
77*7304104dSAndroid Build Coastguard Worker   if (result == NULL)
78*7304104dSAndroid Build Coastguard Worker     return NULL;
79*7304104dSAndroid Build Coastguard Worker 
80*7304104dSAndroid Build Coastguard Worker   rwlock_wrlock (asmscn->ctx->lock);
81*7304104dSAndroid Build Coastguard Worker 
82*7304104dSAndroid Build Coastguard Worker   result->scn = asmscn;
83*7304104dSAndroid Build Coastguard Worker   result->offset = asmscn->offset;
84*7304104dSAndroid Build Coastguard Worker   result->size = size;
85*7304104dSAndroid Build Coastguard Worker   result->type = type;
86*7304104dSAndroid Build Coastguard Worker   result->binding = binding;
87*7304104dSAndroid Build Coastguard Worker   result->symidx = 0;
88*7304104dSAndroid Build Coastguard Worker   result->strent = dwelf_strtab_add (asmscn->ctx->symbol_strtab,
89*7304104dSAndroid Build Coastguard Worker 				     memcpy (result + 1, name, name_len));
90*7304104dSAndroid Build Coastguard Worker 
91*7304104dSAndroid Build Coastguard Worker   if (unlikely (asmscn->ctx->textp))
92*7304104dSAndroid Build Coastguard Worker     {
93*7304104dSAndroid Build Coastguard Worker       /* We are only interested in the name and don't need to know whether
94*7304104dSAndroid Build Coastguard Worker 	 it is a local name or not.  */
95*7304104dSAndroid Build Coastguard Worker       /* First print the binding pseudo-op.  */
96*7304104dSAndroid Build Coastguard Worker       if (binding == STB_GLOBAL)
97*7304104dSAndroid Build Coastguard Worker 	fprintf (asmscn->ctx->out.file, "\t.globl\t%s\n", name);
98*7304104dSAndroid Build Coastguard Worker       else if (binding == STB_WEAK)
99*7304104dSAndroid Build Coastguard Worker 	fprintf (asmscn->ctx->out.file, "\t.weak\t%s\n", name);
100*7304104dSAndroid Build Coastguard Worker 
101*7304104dSAndroid Build Coastguard Worker       /* Next the symbol type.  */
102*7304104dSAndroid Build Coastguard Worker       if (type == STT_OBJECT)
103*7304104dSAndroid Build Coastguard Worker 	fprintf (asmscn->ctx->out.file, "\t.type\t%s,@object\n", name);
104*7304104dSAndroid Build Coastguard Worker       else if (type == STT_FUNC)
105*7304104dSAndroid Build Coastguard Worker 	fprintf (asmscn->ctx->out.file, "\t.type\t%s,@function\n", name);
106*7304104dSAndroid Build Coastguard Worker 
107*7304104dSAndroid Build Coastguard Worker       /* Finally the size and the label.  */
108*7304104dSAndroid Build Coastguard Worker       fprintf (asmscn->ctx->out.file, "\t.size\t%s,%" PRIuMAX "\n%s:\n",
109*7304104dSAndroid Build Coastguard Worker 	       name, (uintmax_t) size, name);
110*7304104dSAndroid Build Coastguard Worker     }
111*7304104dSAndroid Build Coastguard Worker   else
112*7304104dSAndroid Build Coastguard Worker     {
113*7304104dSAndroid Build Coastguard Worker       /* Put the symbol in the hash table so that we can later find it.  */
114*7304104dSAndroid Build Coastguard Worker       if (asm_symbol_tab_insert (&asmscn->ctx->symbol_tab, elf_hash (name),
115*7304104dSAndroid Build Coastguard Worker 				 result) != 0)
116*7304104dSAndroid Build Coastguard Worker 	{
117*7304104dSAndroid Build Coastguard Worker 	  /* The symbol already exists.  */
118*7304104dSAndroid Build Coastguard Worker 	  __libasm_seterrno (ASM_E_DUPLSYM);
119*7304104dSAndroid Build Coastguard Worker 	  /* Note that we can free the entry since there must be no
120*7304104dSAndroid Build Coastguard Worker 	     reference in the string table to the string.  We can only
121*7304104dSAndroid Build Coastguard Worker 	     fail to insert the symbol into the symbol table if there
122*7304104dSAndroid Build Coastguard Worker 	     is already a symbol with this name.  In this case the
123*7304104dSAndroid Build Coastguard Worker 	     dwelf_strtab_add function would use the previously provided
124*7304104dSAndroid Build Coastguard Worker 	     name.  */
125*7304104dSAndroid Build Coastguard Worker 	  free (result);
126*7304104dSAndroid Build Coastguard Worker 	  result = NULL;
127*7304104dSAndroid Build Coastguard Worker 	}
128*7304104dSAndroid Build Coastguard Worker       else if (name != tempsym && asm_emit_symbol_p (name))
129*7304104dSAndroid Build Coastguard Worker 	/* Only count non-private symbols.  */
130*7304104dSAndroid Build Coastguard Worker 	++asmscn->ctx->nsymbol_tab;
131*7304104dSAndroid Build Coastguard Worker     }
132*7304104dSAndroid Build Coastguard Worker 
133*7304104dSAndroid Build Coastguard Worker   rwlock_unlock (asmscn->ctx->lock);
134*7304104dSAndroid Build Coastguard Worker 
135*7304104dSAndroid Build Coastguard Worker   return result;
136*7304104dSAndroid Build Coastguard Worker }
137