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