blob: 32f52772f6258f3a87c170f3177a1634e2144640 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
|
/* GNU/Linux/RISC-V native target description support for GDB.
Copyright (C) 2020-2024 Free Software Foundation, Inc.
This file is part of GDB.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>. */
#include "gdbsupport/common-defs.h"
#include "gdb_proc_service.h"
#include "arch/riscv.h"
#include "elf/common.h"
#include "nat/gdb_ptrace.h"
#include "nat/riscv-linux-tdesc.h"
#include <sys/uio.h>
/* Work around glibc header breakage causing ELF_NFPREG not to be usable. */
#ifndef NFPREG
# define NFPREG 33
#endif
/* See nat/riscv-linux-tdesc.h. */
struct riscv_gdbarch_features
riscv_linux_read_features (int tid)
{
struct riscv_gdbarch_features features;
elf_fpregset_t regs;
int flen;
/* Figuring out xlen is easy. */
features.xlen = sizeof (elf_greg_t);
/* Start with no f-registers. */
features.flen = 0;
/* How much worth of f-registers can we fetch if any? */
for (flen = sizeof (regs.__f.__f[0]); ; flen *= 2)
{
size_t regset_size;
struct iovec iov;
/* Regsets have a uniform slot size, so we count FSCR like
an FP data register. */
regset_size = ELF_NFPREG * flen;
if (regset_size > sizeof (regs))
break;
iov.iov_base = ®s;
iov.iov_len = regset_size;
if (ptrace (PTRACE_GETREGSET, tid, NT_FPREGSET,
(PTRACE_TYPE_ARG3) &iov) == -1)
{
switch (errno)
{
case EINVAL:
continue;
case EIO:
break;
default:
perror_with_name (_("Couldn't get registers"));
break;
}
}
else
features.flen = flen;
break;
}
return features;
}
|