xref: /aosp_15_r20/external/elfutils/backends/riscv_symbol.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* RISC-V specific symbolic name handling.
2*7304104dSAndroid Build Coastguard Worker    This file is part of elfutils.
3*7304104dSAndroid Build Coastguard Worker 
4*7304104dSAndroid Build Coastguard Worker    This file is free software; you can redistribute it and/or modify
5*7304104dSAndroid Build Coastguard Worker    it under the terms of either
6*7304104dSAndroid Build Coastguard Worker 
7*7304104dSAndroid Build Coastguard Worker      * the GNU Lesser General Public License as published by the Free
8*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 3 of the License, or (at
9*7304104dSAndroid Build Coastguard Worker        your option) any later version
10*7304104dSAndroid Build Coastguard Worker 
11*7304104dSAndroid Build Coastguard Worker    or
12*7304104dSAndroid Build Coastguard Worker 
13*7304104dSAndroid Build Coastguard Worker      * the GNU General Public License as published by the Free
14*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 2 of the License, or (at
15*7304104dSAndroid Build Coastguard Worker        your option) any later version
16*7304104dSAndroid Build Coastguard Worker 
17*7304104dSAndroid Build Coastguard Worker    or both in parallel, as here.
18*7304104dSAndroid Build Coastguard Worker 
19*7304104dSAndroid Build Coastguard Worker    elfutils is distributed in the hope that it will be useful, but
20*7304104dSAndroid Build Coastguard Worker    WITHOUT ANY WARRANTY; without even the implied warranty of
21*7304104dSAndroid Build Coastguard Worker    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
22*7304104dSAndroid Build Coastguard Worker    General Public License for more details.
23*7304104dSAndroid Build Coastguard Worker 
24*7304104dSAndroid Build Coastguard Worker    You should have received copies of the GNU General Public License and
25*7304104dSAndroid Build Coastguard Worker    the GNU Lesser General Public License along with this program.  If
26*7304104dSAndroid Build Coastguard Worker    not, see <http://www.gnu.org/licenses/>.  */
27*7304104dSAndroid Build Coastguard Worker 
28*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
29*7304104dSAndroid Build Coastguard Worker # include <config.h>
30*7304104dSAndroid Build Coastguard Worker #endif
31*7304104dSAndroid Build Coastguard Worker 
32*7304104dSAndroid Build Coastguard Worker #include <assert.h>
33*7304104dSAndroid Build Coastguard Worker #include <elf.h>
34*7304104dSAndroid Build Coastguard Worker #include <stddef.h>
35*7304104dSAndroid Build Coastguard Worker #include <string.h>
36*7304104dSAndroid Build Coastguard Worker 
37*7304104dSAndroid Build Coastguard Worker #define BACKEND riscv_
38*7304104dSAndroid Build Coastguard Worker #include "libebl_CPU.h"
39*7304104dSAndroid Build Coastguard Worker 
40*7304104dSAndroid Build Coastguard Worker 
41*7304104dSAndroid Build Coastguard Worker /* Check for the simple reloc types.  */
42*7304104dSAndroid Build Coastguard Worker Elf_Type
riscv_reloc_simple_type(Ebl * ebl,int type,int * addsub)43*7304104dSAndroid Build Coastguard Worker riscv_reloc_simple_type (Ebl *ebl __attribute__ ((unused)), int type,
44*7304104dSAndroid Build Coastguard Worker 			 int *addsub)
45*7304104dSAndroid Build Coastguard Worker {
46*7304104dSAndroid Build Coastguard Worker   switch (type)
47*7304104dSAndroid Build Coastguard Worker     {
48*7304104dSAndroid Build Coastguard Worker     case R_RISCV_SET8:
49*7304104dSAndroid Build Coastguard Worker       return ELF_T_BYTE;
50*7304104dSAndroid Build Coastguard Worker     case R_RISCV_SET16:
51*7304104dSAndroid Build Coastguard Worker       return ELF_T_HALF;
52*7304104dSAndroid Build Coastguard Worker     case R_RISCV_32:
53*7304104dSAndroid Build Coastguard Worker     case R_RISCV_SET32:
54*7304104dSAndroid Build Coastguard Worker       return ELF_T_WORD;
55*7304104dSAndroid Build Coastguard Worker     case R_RISCV_64:
56*7304104dSAndroid Build Coastguard Worker       return ELF_T_XWORD;
57*7304104dSAndroid Build Coastguard Worker     case R_RISCV_ADD16:
58*7304104dSAndroid Build Coastguard Worker       *addsub = 1;
59*7304104dSAndroid Build Coastguard Worker       return ELF_T_HALF;
60*7304104dSAndroid Build Coastguard Worker     case R_RISCV_SUB16:
61*7304104dSAndroid Build Coastguard Worker       *addsub = -1;
62*7304104dSAndroid Build Coastguard Worker       return ELF_T_HALF;
63*7304104dSAndroid Build Coastguard Worker     case R_RISCV_ADD32:
64*7304104dSAndroid Build Coastguard Worker       *addsub = 1;
65*7304104dSAndroid Build Coastguard Worker       return ELF_T_WORD;
66*7304104dSAndroid Build Coastguard Worker     case R_RISCV_SUB32:
67*7304104dSAndroid Build Coastguard Worker       *addsub = -1;
68*7304104dSAndroid Build Coastguard Worker       return ELF_T_WORD;
69*7304104dSAndroid Build Coastguard Worker     case R_RISCV_ADD64:
70*7304104dSAndroid Build Coastguard Worker       *addsub = 1;
71*7304104dSAndroid Build Coastguard Worker       return ELF_T_XWORD;
72*7304104dSAndroid Build Coastguard Worker     case R_RISCV_SUB64:
73*7304104dSAndroid Build Coastguard Worker       *addsub = -1;
74*7304104dSAndroid Build Coastguard Worker       return ELF_T_XWORD;
75*7304104dSAndroid Build Coastguard Worker     default:
76*7304104dSAndroid Build Coastguard Worker       return ELF_T_NUM;
77*7304104dSAndroid Build Coastguard Worker     }
78*7304104dSAndroid Build Coastguard Worker }
79*7304104dSAndroid Build Coastguard Worker 
80*7304104dSAndroid Build Coastguard Worker /* Check whether machine flags are valid.  */
81*7304104dSAndroid Build Coastguard Worker bool
riscv_machine_flag_check(GElf_Word flags)82*7304104dSAndroid Build Coastguard Worker riscv_machine_flag_check (GElf_Word flags)
83*7304104dSAndroid Build Coastguard Worker {
84*7304104dSAndroid Build Coastguard Worker   return ((flags &~ (EF_RISCV_RVC
85*7304104dSAndroid Build Coastguard Worker 		     | EF_RISCV_FLOAT_ABI)) == 0);
86*7304104dSAndroid Build Coastguard Worker }
87*7304104dSAndroid Build Coastguard Worker 
88*7304104dSAndroid Build Coastguard Worker /* Check whether given symbol's st_value and st_size are OK despite failing
89*7304104dSAndroid Build Coastguard Worker    normal checks.  */
90*7304104dSAndroid Build Coastguard Worker bool
riscv_check_special_symbol(Elf * elf,const GElf_Sym * sym,const char * name,const GElf_Shdr * destshdr)91*7304104dSAndroid Build Coastguard Worker riscv_check_special_symbol (Elf *elf, const GElf_Sym *sym,
92*7304104dSAndroid Build Coastguard Worker 			    const char *name, const GElf_Shdr *destshdr)
93*7304104dSAndroid Build Coastguard Worker {
94*7304104dSAndroid Build Coastguard Worker   if (name == NULL)
95*7304104dSAndroid Build Coastguard Worker     return false;
96*7304104dSAndroid Build Coastguard Worker 
97*7304104dSAndroid Build Coastguard Worker   size_t shstrndx;
98*7304104dSAndroid Build Coastguard Worker   if (elf_getshdrstrndx (elf, &shstrndx) != 0)
99*7304104dSAndroid Build Coastguard Worker     return false;
100*7304104dSAndroid Build Coastguard Worker   const char *sname = elf_strptr (elf, shstrndx, destshdr->sh_name);
101*7304104dSAndroid Build Coastguard Worker   if (sname == NULL)
102*7304104dSAndroid Build Coastguard Worker     return false;
103*7304104dSAndroid Build Coastguard Worker 
104*7304104dSAndroid Build Coastguard Worker   /* _GLOBAL_OFFSET_TABLE_ points to the start of the .got section, but it
105*7304104dSAndroid Build Coastguard Worker      is preceded by the .got.plt section in the output .got section.  */
106*7304104dSAndroid Build Coastguard Worker   if (strcmp (name, "_GLOBAL_OFFSET_TABLE_") == 0)
107*7304104dSAndroid Build Coastguard Worker     return (strcmp (sname, ".got") == 0
108*7304104dSAndroid Build Coastguard Worker 	    && sym->st_value >= destshdr->sh_addr
109*7304104dSAndroid Build Coastguard Worker 	    && sym->st_value < destshdr->sh_addr + destshdr->sh_size);
110*7304104dSAndroid Build Coastguard Worker 
111*7304104dSAndroid Build Coastguard Worker   /* __global_pointer$ points to the .sdata section with an offset of
112*7304104dSAndroid Build Coastguard Worker      0x800.  It might however fall in the .got section, in which case we
113*7304104dSAndroid Build Coastguard Worker      cannot check the offset.  The size always should be zero.  */
114*7304104dSAndroid Build Coastguard Worker   if (strcmp (name, "__global_pointer$") == 0)
115*7304104dSAndroid Build Coastguard Worker     return (((strcmp (sname, ".sdata") == 0
116*7304104dSAndroid Build Coastguard Worker 	      && sym->st_value == destshdr->sh_addr + 0x800)
117*7304104dSAndroid Build Coastguard Worker 	     || strcmp (sname, ".got") == 0)
118*7304104dSAndroid Build Coastguard Worker 	    && sym->st_size == 0);
119*7304104dSAndroid Build Coastguard Worker 
120*7304104dSAndroid Build Coastguard Worker   return false;
121*7304104dSAndroid Build Coastguard Worker }
122*7304104dSAndroid Build Coastguard Worker 
123*7304104dSAndroid Build Coastguard Worker const char *
riscv_segment_type_name(int segment,char * buf,size_t len)124*7304104dSAndroid Build Coastguard Worker riscv_segment_type_name (int segment, char *buf __attribute__ ((unused)),
125*7304104dSAndroid Build Coastguard Worker 			 size_t len __attribute__ ((unused)))
126*7304104dSAndroid Build Coastguard Worker {
127*7304104dSAndroid Build Coastguard Worker   switch (segment)
128*7304104dSAndroid Build Coastguard Worker     {
129*7304104dSAndroid Build Coastguard Worker     case PT_RISCV_ATTRIBUTES:
130*7304104dSAndroid Build Coastguard Worker       return "RISCV_ATTRIBUTES";
131*7304104dSAndroid Build Coastguard Worker     }
132*7304104dSAndroid Build Coastguard Worker   return NULL;
133*7304104dSAndroid Build Coastguard Worker }
134*7304104dSAndroid Build Coastguard Worker 
135*7304104dSAndroid Build Coastguard Worker /* Return symbolic representation of section type.  */
136*7304104dSAndroid Build Coastguard Worker const char *
riscv_section_type_name(int type,char * buf,size_t len)137*7304104dSAndroid Build Coastguard Worker riscv_section_type_name (int type,
138*7304104dSAndroid Build Coastguard Worker 			 char *buf __attribute__ ((unused)),
139*7304104dSAndroid Build Coastguard Worker 			 size_t len __attribute__ ((unused)))
140*7304104dSAndroid Build Coastguard Worker {
141*7304104dSAndroid Build Coastguard Worker   switch (type)
142*7304104dSAndroid Build Coastguard Worker     {
143*7304104dSAndroid Build Coastguard Worker     case SHT_RISCV_ATTRIBUTES:
144*7304104dSAndroid Build Coastguard Worker       return "RISCV_ATTRIBUTES";
145*7304104dSAndroid Build Coastguard Worker     }
146*7304104dSAndroid Build Coastguard Worker 
147*7304104dSAndroid Build Coastguard Worker   return NULL;
148*7304104dSAndroid Build Coastguard Worker }
149*7304104dSAndroid Build Coastguard Worker 
150*7304104dSAndroid Build Coastguard Worker const char *
riscv_dynamic_tag_name(int64_t tag,char * buf,size_t len)151*7304104dSAndroid Build Coastguard Worker riscv_dynamic_tag_name (int64_t tag, char *buf __attribute__ ((unused)),
152*7304104dSAndroid Build Coastguard Worker 			size_t len __attribute__ ((unused)))
153*7304104dSAndroid Build Coastguard Worker {
154*7304104dSAndroid Build Coastguard Worker   switch (tag)
155*7304104dSAndroid Build Coastguard Worker     {
156*7304104dSAndroid Build Coastguard Worker     case DT_RISCV_VARIANT_CC:
157*7304104dSAndroid Build Coastguard Worker       return "RISCV_VARIANT_CC";
158*7304104dSAndroid Build Coastguard Worker     }
159*7304104dSAndroid Build Coastguard Worker   return NULL;
160*7304104dSAndroid Build Coastguard Worker }
161*7304104dSAndroid Build Coastguard Worker 
162*7304104dSAndroid Build Coastguard Worker bool
riscv_dynamic_tag_check(int64_t tag)163*7304104dSAndroid Build Coastguard Worker riscv_dynamic_tag_check (int64_t tag)
164*7304104dSAndroid Build Coastguard Worker {
165*7304104dSAndroid Build Coastguard Worker   return tag == DT_RISCV_VARIANT_CC;
166*7304104dSAndroid Build Coastguard Worker }
167