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