]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blob - gdb/loongarch-linux-tdep.c
configure: require libzstd >= 1.4.0
[thirdparty/binutils-gdb.git] / gdb / loongarch-linux-tdep.c
1 /* Target-dependent code for GNU/Linux on LoongArch processors.
2
3 Copyright (C) 2022 Free Software Foundation, Inc.
4 Contributed by Loongson Ltd.
5
6 This file is part of GDB.
7
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
12
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>. */
20
21 #include "defs.h"
22 #include "glibc-tdep.h"
23 #include "inferior.h"
24 #include "linux-tdep.h"
25 #include "loongarch-tdep.h"
26 #include "solib-svr4.h"
27 #include "target-descriptions.h"
28 #include "trad-frame.h"
29 #include "tramp-frame.h"
30
31 /* Unpack an elf_gregset_t into GDB's register cache. */
32
33 static void
34 loongarch_supply_gregset (const struct regset *regset,
35 struct regcache *regcache, int regnum,
36 const void *gprs, size_t len)
37 {
38 int regsize = register_size (regcache->arch (), 0);
39 const gdb_byte *buf = nullptr;
40
41 if (regnum == -1)
42 {
43 regcache->raw_supply_zeroed (0);
44
45 for (int i = 1; i < 32; i++)
46 {
47 buf = (const gdb_byte*) gprs + regsize * i;
48 regcache->raw_supply (i, (const void *) buf);
49 }
50
51 buf = (const gdb_byte*) gprs + regsize * LOONGARCH_ORIG_A0_REGNUM;
52 regcache->raw_supply (LOONGARCH_ORIG_A0_REGNUM, (const void *) buf);
53
54 buf = (const gdb_byte*) gprs + regsize * LOONGARCH_PC_REGNUM;
55 regcache->raw_supply (LOONGARCH_PC_REGNUM, (const void *) buf);
56
57 buf = (const gdb_byte*) gprs + regsize * LOONGARCH_BADV_REGNUM;
58 regcache->raw_supply (LOONGARCH_BADV_REGNUM, (const void *) buf);
59 }
60 else if (regnum == 0)
61 regcache->raw_supply_zeroed (0);
62 else if ((regnum > 0 && regnum < 32)
63 || regnum == LOONGARCH_ORIG_A0_REGNUM
64 || regnum == LOONGARCH_PC_REGNUM
65 || regnum == LOONGARCH_BADV_REGNUM)
66 {
67 buf = (const gdb_byte*) gprs + regsize * regnum;
68 regcache->raw_supply (regnum, (const void *) buf);
69 }
70 }
71
72 /* Pack the GDB's register cache value into an elf_gregset_t. */
73
74 static void
75 loongarch_fill_gregset (const struct regset *regset,
76 const struct regcache *regcache, int regnum,
77 void *gprs, size_t len)
78 {
79 int regsize = register_size (regcache->arch (), 0);
80 gdb_byte *buf = nullptr;
81
82 if (regnum == -1)
83 {
84 for (int i = 0; i < 32; i++)
85 {
86 buf = (gdb_byte *) gprs + regsize * i;
87 regcache->raw_collect (i, (void *) buf);
88 }
89
90 buf = (gdb_byte *) gprs + regsize * LOONGARCH_ORIG_A0_REGNUM;
91 regcache->raw_collect (LOONGARCH_ORIG_A0_REGNUM, (void *) buf);
92
93 buf = (gdb_byte *) gprs + regsize * LOONGARCH_PC_REGNUM;
94 regcache->raw_collect (LOONGARCH_PC_REGNUM, (void *) buf);
95
96 buf = (gdb_byte *) gprs + regsize * LOONGARCH_BADV_REGNUM;
97 regcache->raw_collect (LOONGARCH_BADV_REGNUM, (void *) buf);
98 }
99 else if ((regnum >= 0 && regnum < 32)
100 || regnum == LOONGARCH_ORIG_A0_REGNUM
101 || regnum == LOONGARCH_PC_REGNUM
102 || regnum == LOONGARCH_BADV_REGNUM)
103 {
104 buf = (gdb_byte *) gprs + regsize * regnum;
105 regcache->raw_collect (regnum, (void *) buf);
106 }
107 }
108
109 /* Define the general register regset. */
110
111 const struct regset loongarch_gregset =
112 {
113 nullptr,
114 loongarch_supply_gregset,
115 loongarch_fill_gregset,
116 };
117
118 /* Unpack an elf_fpregset_t into GDB's register cache. */
119 static void
120 loongarch_supply_fpregset (const struct regset *r,
121 struct regcache *regcache, int regnum,
122 const void *fprs, size_t len)
123 {
124 const gdb_byte *buf = nullptr;
125 int fprsize = register_size (regcache->arch (), LOONGARCH_FIRST_FP_REGNUM);
126 int fccsize = register_size (regcache->arch (), LOONGARCH_FIRST_FCC_REGNUM);
127
128 if (regnum == -1)
129 {
130 for (int i = 0; i < LOONGARCH_LINUX_NUM_FPREGSET; i++)
131 {
132 buf = (const gdb_byte *)fprs + fprsize * i;
133 regcache->raw_supply (LOONGARCH_FIRST_FP_REGNUM + i, (const void *)buf);
134 }
135 for (int i = 0; i < LOONGARCH_LINUX_NUM_FCC; i++)
136 {
137 buf = (const gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
138 fccsize * i;
139 regcache->raw_supply (LOONGARCH_FIRST_FCC_REGNUM + i, (const void *)buf);
140 }
141 buf = (const gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
142 fccsize * LOONGARCH_LINUX_NUM_FCC;
143 regcache->raw_supply (LOONGARCH_FCSR_REGNUM, (const void *)buf);
144 }
145 else if (regnum >= LOONGARCH_FIRST_FP_REGNUM && regnum < LOONGARCH_FIRST_FCC_REGNUM)
146 {
147 buf = (const gdb_byte *)fprs + fprsize * (regnum - LOONGARCH_FIRST_FP_REGNUM);
148 regcache->raw_supply (regnum, (const void *)buf);
149 }
150 else if (regnum >= LOONGARCH_FIRST_FCC_REGNUM && regnum < LOONGARCH_FCSR_REGNUM)
151 {
152 buf = (const gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
153 fccsize * (regnum - LOONGARCH_FIRST_FCC_REGNUM);
154 regcache->raw_supply (regnum, (const void *)buf);
155 }
156 else if (regnum == LOONGARCH_FCSR_REGNUM)
157 {
158 buf = (const gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
159 fccsize * LOONGARCH_LINUX_NUM_FCC;
160 regcache->raw_supply (regnum, (const void *)buf);
161 }
162 }
163
164 /* Pack the GDB's register cache value into an elf_fpregset_t. */
165 static void
166 loongarch_fill_fpregset (const struct regset *r,
167 const struct regcache *regcache, int regnum,
168 void *fprs, size_t len)
169 {
170 gdb_byte *buf = nullptr;
171 int fprsize = register_size (regcache->arch (), LOONGARCH_FIRST_FP_REGNUM);
172 int fccsize = register_size (regcache->arch (), LOONGARCH_FIRST_FCC_REGNUM);
173
174 if (regnum == -1)
175 {
176 for (int i = 0; i < LOONGARCH_LINUX_NUM_FPREGSET; i++)
177 {
178 buf = (gdb_byte *)fprs + fprsize * i;
179 regcache->raw_collect (LOONGARCH_FIRST_FP_REGNUM + i, (void *)buf);
180 }
181 for (int i = 0; i < LOONGARCH_LINUX_NUM_FCC; i++)
182 {
183 buf = (gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
184 fccsize * i;
185 regcache->raw_collect (LOONGARCH_FIRST_FCC_REGNUM + i, (void *)buf);
186 }
187 buf = (gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
188 fccsize * LOONGARCH_LINUX_NUM_FCC;
189 regcache->raw_collect (LOONGARCH_FCSR_REGNUM, (void *)buf);
190 }
191 else if (regnum >= LOONGARCH_FIRST_FP_REGNUM && regnum < LOONGARCH_FIRST_FCC_REGNUM)
192 {
193 buf = (gdb_byte *)fprs + fprsize * (regnum - LOONGARCH_FIRST_FP_REGNUM);
194 regcache->raw_collect (regnum, (void *)buf);
195 }
196 else if (regnum >= LOONGARCH_FIRST_FCC_REGNUM && regnum < LOONGARCH_FCSR_REGNUM)
197 {
198 buf = (gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
199 fccsize * (regnum - LOONGARCH_FIRST_FCC_REGNUM);
200 regcache->raw_collect (regnum, (void *)buf);
201 }
202 else if (regnum == LOONGARCH_FCSR_REGNUM)
203 {
204 buf = (gdb_byte *)fprs + fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
205 fccsize * LOONGARCH_LINUX_NUM_FCC;
206 regcache->raw_collect (regnum, (void *)buf);
207 }
208 }
209
210 /* Define the FP register regset. */
211 const struct regset loongarch_fpregset =
212 {
213 nullptr,
214 loongarch_supply_fpregset,
215 loongarch_fill_fpregset,
216 };
217
218 /* Implement the "init" method of struct tramp_frame. */
219
220 #define LOONGARCH_RT_SIGFRAME_UCONTEXT_OFFSET 128
221 #define LOONGARCH_UCONTEXT_SIGCONTEXT_OFFSET 176
222
223 static void
224 loongarch_linux_rt_sigframe_init (const struct tramp_frame *self,
225 frame_info_ptr this_frame,
226 struct trad_frame_cache *this_cache,
227 CORE_ADDR func)
228 {
229 CORE_ADDR frame_sp = get_frame_sp (this_frame);
230 CORE_ADDR sigcontext_base = (frame_sp + LOONGARCH_RT_SIGFRAME_UCONTEXT_OFFSET
231 + LOONGARCH_UCONTEXT_SIGCONTEXT_OFFSET);
232
233 trad_frame_set_reg_addr (this_cache, LOONGARCH_PC_REGNUM, sigcontext_base);
234 for (int i = 0; i < 32; i++)
235 trad_frame_set_reg_addr (this_cache, i, sigcontext_base + 8 + i * 8);
236
237 trad_frame_set_id (this_cache, frame_id_build (frame_sp, func));
238 }
239
240 /* li.w a7, __NR_rt_sigreturn */
241 #define LOONGARCH_INST_LIW_A7_RT_SIGRETURN 0x03822c0b
242 /* syscall 0 */
243 #define LOONGARCH_INST_SYSCALL 0x002b0000
244
245 static const struct tramp_frame loongarch_linux_rt_sigframe =
246 {
247 SIGTRAMP_FRAME,
248 4,
249 {
250 { LOONGARCH_INST_LIW_A7_RT_SIGRETURN, ULONGEST_MAX },
251 { LOONGARCH_INST_SYSCALL, ULONGEST_MAX },
252 { TRAMP_SENTINEL_INSN, ULONGEST_MAX }
253 },
254 loongarch_linux_rt_sigframe_init,
255 nullptr
256 };
257
258 /* Implement the "iterate_over_regset_sections" gdbarch method. */
259
260 static void
261 loongarch_iterate_over_regset_sections (struct gdbarch *gdbarch,
262 iterate_over_regset_sections_cb *cb,
263 void *cb_data,
264 const struct regcache *regcache)
265 {
266 int gprsize = register_size (gdbarch, 0);
267 int fprsize = register_size (gdbarch, LOONGARCH_FIRST_FP_REGNUM);
268 int fccsize = register_size (gdbarch, LOONGARCH_FIRST_FCC_REGNUM);
269 int fcsrsize = register_size (gdbarch, LOONGARCH_FCSR_REGNUM);
270 int fpsize = fprsize * LOONGARCH_LINUX_NUM_FPREGSET +
271 fccsize * LOONGARCH_LINUX_NUM_FCC + fcsrsize;
272
273 cb (".reg", LOONGARCH_LINUX_NUM_GREGSET * gprsize,
274 LOONGARCH_LINUX_NUM_GREGSET * gprsize, &loongarch_gregset, nullptr, cb_data);
275 cb (".reg2", fpsize, fpsize, &loongarch_fpregset, nullptr, cb_data);
276 }
277
278 /* The following value is derived from __NR_rt_sigreturn in
279 <include/uapi/asm-generic/unistd.h> from the Linux source tree. */
280
281 #define LOONGARCH_NR_rt_sigreturn 139
282
283 /* When FRAME is at a syscall instruction, return the PC of the next
284 instruction to be executed. */
285
286 static CORE_ADDR
287 loongarch_linux_syscall_next_pc (frame_info_ptr frame)
288 {
289 const CORE_ADDR pc = get_frame_pc (frame);
290 ULONGEST a7 = get_frame_register_unsigned (frame, LOONGARCH_A7_REGNUM);
291
292 /* If we are about to make a sigreturn syscall, use the unwinder to
293 decode the signal frame. */
294 if (a7 == LOONGARCH_NR_rt_sigreturn)
295 return frame_unwind_caller_pc (frame);
296
297 return pc + 4;
298 }
299
300 /* Initialize LoongArch Linux ABI info. */
301
302 static void
303 loongarch_linux_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
304 {
305 loongarch_gdbarch_tdep *tdep = gdbarch_tdep<loongarch_gdbarch_tdep> (gdbarch);
306
307 linux_init_abi (info, gdbarch, 0);
308
309 set_solib_svr4_fetch_link_map_offsets (gdbarch,
310 info.bfd_arch_info->bits_per_address == 32
311 ? linux_ilp32_fetch_link_map_offsets
312 : linux_lp64_fetch_link_map_offsets);
313
314 /* GNU/Linux uses SVR4-style shared libraries. */
315 set_gdbarch_skip_trampoline_code (gdbarch, find_solib_trampoline_target);
316
317 /* GNU/Linux uses the dynamic linker included in the GNU C Library. */
318 set_gdbarch_skip_solib_resolver (gdbarch, glibc_skip_solib_resolver);
319
320 /* Enable TLS support. */
321 set_gdbarch_fetch_tls_load_module_address (gdbarch, svr4_fetch_objfile_link_map);
322
323 /* Prepend tramp frame unwinder for signal. */
324 tramp_frame_prepend_unwinder (gdbarch, &loongarch_linux_rt_sigframe);
325
326 /* Core file support. */
327 set_gdbarch_iterate_over_regset_sections (gdbarch, loongarch_iterate_over_regset_sections);
328
329 tdep->syscall_next_pc = loongarch_linux_syscall_next_pc;
330 }
331
332 /* Initialize LoongArch Linux target support. */
333
334 void _initialize_loongarch_linux_tdep ();
335 void
336 _initialize_loongarch_linux_tdep ()
337 {
338 gdbarch_register_osabi (bfd_arch_loongarch, bfd_mach_loongarch32,
339 GDB_OSABI_LINUX, loongarch_linux_init_abi);
340 gdbarch_register_osabi (bfd_arch_loongarch, bfd_mach_loongarch64,
341 GDB_OSABI_LINUX, loongarch_linux_init_abi);
342 }