1da0c48c4Sopenharmony_ci/* Define new symbol for current position in given section.
2da0c48c4Sopenharmony_ci   Copyright (C) 2002, 2005, 2016, 2017 Red Hat, Inc.
3da0c48c4Sopenharmony_ci   This file is part of elfutils.
4da0c48c4Sopenharmony_ci   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
5da0c48c4Sopenharmony_ci
6da0c48c4Sopenharmony_ci   This file is free software; you can redistribute it and/or modify
7da0c48c4Sopenharmony_ci   it under the terms of either
8da0c48c4Sopenharmony_ci
9da0c48c4Sopenharmony_ci     * the GNU Lesser General Public License as published by the Free
10da0c48c4Sopenharmony_ci       Software Foundation; either version 3 of the License, or (at
11da0c48c4Sopenharmony_ci       your option) any later version
12da0c48c4Sopenharmony_ci
13da0c48c4Sopenharmony_ci   or
14da0c48c4Sopenharmony_ci
15da0c48c4Sopenharmony_ci     * the GNU General Public License as published by the Free
16da0c48c4Sopenharmony_ci       Software Foundation; either version 2 of the License, or (at
17da0c48c4Sopenharmony_ci       your option) any later version
18da0c48c4Sopenharmony_ci
19da0c48c4Sopenharmony_ci   or both in parallel, as here.
20da0c48c4Sopenharmony_ci
21da0c48c4Sopenharmony_ci   elfutils is distributed in the hope that it will be useful, but
22da0c48c4Sopenharmony_ci   WITHOUT ANY WARRANTY; without even the implied warranty of
23da0c48c4Sopenharmony_ci   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24da0c48c4Sopenharmony_ci   General Public License for more details.
25da0c48c4Sopenharmony_ci
26da0c48c4Sopenharmony_ci   You should have received copies of the GNU General Public License and
27da0c48c4Sopenharmony_ci   the GNU Lesser General Public License along with this program.  If
28da0c48c4Sopenharmony_ci   not, see <http://www.gnu.org/licenses/>.  */
29da0c48c4Sopenharmony_ci
30da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H
31da0c48c4Sopenharmony_ci# include <config.h>
32da0c48c4Sopenharmony_ci#endif
33da0c48c4Sopenharmony_ci
34da0c48c4Sopenharmony_ci#include <inttypes.h>
35da0c48c4Sopenharmony_ci#include <stdio.h>
36da0c48c4Sopenharmony_ci#include <stdlib.h>
37da0c48c4Sopenharmony_ci#include <string.h>
38da0c48c4Sopenharmony_ci
39da0c48c4Sopenharmony_ci#include <libasmP.h>
40da0c48c4Sopenharmony_ci#include <system.h>
41da0c48c4Sopenharmony_ci
42da0c48c4Sopenharmony_ci
43da0c48c4Sopenharmony_ciAsmSym_t *
44da0c48c4Sopenharmony_ciasm_newsym (AsmScn_t *asmscn, const char *name, GElf_Xword size,
45da0c48c4Sopenharmony_ci	    int type, int binding)
46da0c48c4Sopenharmony_ci{
47da0c48c4Sopenharmony_ci/* We don't really expect labels with many digits, but in theory it could
48da0c48c4Sopenharmony_ci   be 10 digits (plus ".L" and a zero terminator).  */
49da0c48c4Sopenharmony_ci#define TEMPSYMLEN 13
50da0c48c4Sopenharmony_ci  char tempsym[TEMPSYMLEN];
51da0c48c4Sopenharmony_ci  AsmSym_t *result;
52da0c48c4Sopenharmony_ci
53da0c48c4Sopenharmony_ci  if (asmscn == NULL)
54da0c48c4Sopenharmony_ci    /* Something went wrong before.  */
55da0c48c4Sopenharmony_ci    return NULL;
56da0c48c4Sopenharmony_ci
57da0c48c4Sopenharmony_ci  /* Generate a temporary symbol if necessary.  */
58da0c48c4Sopenharmony_ci  if (name == NULL)
59da0c48c4Sopenharmony_ci    {
60da0c48c4Sopenharmony_ci      /* If a local symbol name is created the symbol better have
61da0c48c4Sopenharmony_ci	 local binding.  */
62da0c48c4Sopenharmony_ci      if (binding != STB_LOCAL)
63da0c48c4Sopenharmony_ci	{
64da0c48c4Sopenharmony_ci	  __libasm_seterrno (ASM_E_INVALID);
65da0c48c4Sopenharmony_ci	  return NULL;
66da0c48c4Sopenharmony_ci	}
67da0c48c4Sopenharmony_ci
68da0c48c4Sopenharmony_ci      // XXX This requires getting the format from the machine backend.  */
69da0c48c4Sopenharmony_ci      snprintf (tempsym, TEMPSYMLEN, ".L%07u", asmscn->ctx->tempsym_count++);
70da0c48c4Sopenharmony_ci
71da0c48c4Sopenharmony_ci      name = tempsym;
72da0c48c4Sopenharmony_ci    }
73da0c48c4Sopenharmony_ci
74da0c48c4Sopenharmony_ci  size_t name_len = strlen (name) + 1;
75da0c48c4Sopenharmony_ci
76da0c48c4Sopenharmony_ci  result = malloc (sizeof (AsmSym_t) + name_len);
77da0c48c4Sopenharmony_ci  if (result == NULL)
78da0c48c4Sopenharmony_ci    return NULL;
79da0c48c4Sopenharmony_ci
80da0c48c4Sopenharmony_ci  rwlock_wrlock (asmscn->ctx->lock);
81da0c48c4Sopenharmony_ci
82da0c48c4Sopenharmony_ci  result->scn = asmscn;
83da0c48c4Sopenharmony_ci  result->offset = asmscn->offset;
84da0c48c4Sopenharmony_ci  result->size = size;
85da0c48c4Sopenharmony_ci  result->type = type;
86da0c48c4Sopenharmony_ci  result->binding = binding;
87da0c48c4Sopenharmony_ci  result->symidx = 0;
88da0c48c4Sopenharmony_ci  result->strent = dwelf_strtab_add (asmscn->ctx->symbol_strtab,
89da0c48c4Sopenharmony_ci				     memcpy (result + 1, name, name_len));
90da0c48c4Sopenharmony_ci
91da0c48c4Sopenharmony_ci  if (unlikely (asmscn->ctx->textp))
92da0c48c4Sopenharmony_ci    {
93da0c48c4Sopenharmony_ci      /* We are only interested in the name and don't need to know whether
94da0c48c4Sopenharmony_ci	 it is a local name or not.  */
95da0c48c4Sopenharmony_ci      /* First print the binding pseudo-op.  */
96da0c48c4Sopenharmony_ci      if (binding == STB_GLOBAL)
97da0c48c4Sopenharmony_ci	fprintf (asmscn->ctx->out.file, "\t.globl\t%s\n", name);
98da0c48c4Sopenharmony_ci      else if (binding == STB_WEAK)
99da0c48c4Sopenharmony_ci	fprintf (asmscn->ctx->out.file, "\t.weak\t%s\n", name);
100da0c48c4Sopenharmony_ci
101da0c48c4Sopenharmony_ci      /* Next the symbol type.  */
102da0c48c4Sopenharmony_ci      if (type == STT_OBJECT)
103da0c48c4Sopenharmony_ci	fprintf (asmscn->ctx->out.file, "\t.type\t%s,@object\n", name);
104da0c48c4Sopenharmony_ci      else if (type == STT_FUNC)
105da0c48c4Sopenharmony_ci	fprintf (asmscn->ctx->out.file, "\t.type\t%s,@function\n", name);
106da0c48c4Sopenharmony_ci
107da0c48c4Sopenharmony_ci      /* Finally the size and the label.  */
108da0c48c4Sopenharmony_ci      fprintf (asmscn->ctx->out.file, "\t.size\t%s,%" PRIuMAX "\n%s:\n",
109da0c48c4Sopenharmony_ci	       name, (uintmax_t) size, name);
110da0c48c4Sopenharmony_ci    }
111da0c48c4Sopenharmony_ci  else
112da0c48c4Sopenharmony_ci    {
113da0c48c4Sopenharmony_ci      /* Put the symbol in the hash table so that we can later find it.  */
114da0c48c4Sopenharmony_ci      if (asm_symbol_tab_insert (&asmscn->ctx->symbol_tab, elf_hash (name),
115da0c48c4Sopenharmony_ci				 result) != 0)
116da0c48c4Sopenharmony_ci	{
117da0c48c4Sopenharmony_ci	  /* The symbol already exists.  */
118da0c48c4Sopenharmony_ci	  __libasm_seterrno (ASM_E_DUPLSYM);
119da0c48c4Sopenharmony_ci	  /* Note that we can free the entry since there must be no
120da0c48c4Sopenharmony_ci	     reference in the string table to the string.  We can only
121da0c48c4Sopenharmony_ci	     fail to insert the symbol into the symbol table if there
122da0c48c4Sopenharmony_ci	     is already a symbol with this name.  In this case the
123da0c48c4Sopenharmony_ci	     dwelf_strtab_add function would use the previously provided
124da0c48c4Sopenharmony_ci	     name.  */
125da0c48c4Sopenharmony_ci	  free (result);
126da0c48c4Sopenharmony_ci	  result = NULL;
127da0c48c4Sopenharmony_ci	}
128da0c48c4Sopenharmony_ci      else if (name != tempsym && asm_emit_symbol_p (name))
129da0c48c4Sopenharmony_ci	/* Only count non-private symbols.  */
130da0c48c4Sopenharmony_ci	++asmscn->ctx->nsymbol_tab;
131da0c48c4Sopenharmony_ci    }
132da0c48c4Sopenharmony_ci
133da0c48c4Sopenharmony_ci  rwlock_unlock (asmscn->ctx->lock);
134da0c48c4Sopenharmony_ci
135da0c48c4Sopenharmony_ci  return result;
136da0c48c4Sopenharmony_ci}
137