2023-02-07 01:24:32 +08:00
|
|
|
/* Common target dependent routines for AArch64 Scalable Extensions
|
|
|
|
(SVE/SME).
|
2018-05-31 21:36:48 +08:00
|
|
|
|
2023-01-01 20:49:04 +08:00
|
|
|
Copyright (C) 2018-2023 Free Software Foundation, Inc.
|
2018-05-31 21:36:48 +08:00
|
|
|
|
|
|
|
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 <sys/utsname.h>
|
|
|
|
#include <sys/uio.h>
|
Rename common to gdbsupport
This is the next patch in the ongoing series to move gdbsever to the
top level.
This patch just renames the "common" directory. The idea is to do
this move in two parts: first rename the directory (this patch), then
move the directory to the top. This approach makes the patches a bit
more tractable.
I chose the name "gdbsupport" for the directory. However, as this
patch was largely written by sed, we could pick a new name without too
much difficulty.
Tested by the buildbot.
gdb/ChangeLog
2019-07-09 Tom Tromey <tom@tromey.com>
* contrib/ari/gdb_ari.sh: Change common to gdbsupport.
* configure: Rebuild.
* configure.ac: Change common to gdbsupport.
* gdbsupport: Rename from common.
* acinclude.m4: Change common to gdbsupport.
* Makefile.in (CONFIG_SRC_SUBDIR, COMMON_SFILES)
(HFILES_NO_SRCDIR, stamp-version, ALLDEPFILES): Change common to
gdbsupport.
* aarch64-tdep.c, ada-lang.c, ada-lang.h, agent.c, alloc.c,
amd64-darwin-tdep.c, amd64-dicos-tdep.c, amd64-fbsd-nat.c,
amd64-fbsd-tdep.c, amd64-linux-nat.c, amd64-linux-tdep.c,
amd64-nbsd-tdep.c, amd64-obsd-tdep.c, amd64-sol2-tdep.c,
amd64-tdep.c, amd64-windows-tdep.c, arch-utils.c,
arch/aarch64-insn.c, arch/aarch64.c, arch/aarch64.h, arch/amd64.c,
arch/amd64.h, arch/arm-get-next-pcs.c, arch/arm-linux.c,
arch/arm.c, arch/i386.c, arch/i386.h, arch/ppc-linux-common.c,
arch/riscv.c, arch/riscv.h, arch/tic6x.c, arm-tdep.c, auto-load.c,
auxv.c, ax-gdb.c, ax-general.c, ax.h, breakpoint.c, breakpoint.h,
btrace.c, btrace.h, build-id.c, build-id.h, c-lang.h, charset.c,
charset.h, cli/cli-cmds.c, cli/cli-cmds.h, cli/cli-decode.c,
cli/cli-dump.c, cli/cli-option.h, cli/cli-script.c,
coff-pe-read.c, command.h, compile/compile-c-support.c,
compile/compile-c.h, compile/compile-cplus-symbols.c,
compile/compile-cplus-types.c, compile/compile-cplus.h,
compile/compile-loc2c.c, compile/compile.c, completer.c,
completer.h, contrib/ari/gdb_ari.sh, corefile.c, corelow.c,
cp-support.c, cp-support.h, cp-valprint.c, csky-tdep.c, ctf.c,
darwin-nat.c, debug.c, defs.h, disasm-selftests.c, disasm.c,
disasm.h, dtrace-probe.c, dwarf-index-cache.c,
dwarf-index-cache.h, dwarf-index-write.c, dwarf2-frame.c,
dwarf2expr.c, dwarf2loc.c, dwarf2read.c, event-loop.c,
event-top.c, exceptions.c, exec.c, extension.h, fbsd-nat.c,
features/aarch64-core.c, features/aarch64-fpu.c,
features/aarch64-pauth.c, features/aarch64-sve.c,
features/i386/32bit-avx.c, features/i386/32bit-avx512.c,
features/i386/32bit-core.c, features/i386/32bit-linux.c,
features/i386/32bit-mpx.c, features/i386/32bit-pkeys.c,
features/i386/32bit-segments.c, features/i386/32bit-sse.c,
features/i386/64bit-avx.c, features/i386/64bit-avx512.c,
features/i386/64bit-core.c, features/i386/64bit-linux.c,
features/i386/64bit-mpx.c, features/i386/64bit-pkeys.c,
features/i386/64bit-segments.c, features/i386/64bit-sse.c,
features/i386/x32-core.c, features/riscv/32bit-cpu.c,
features/riscv/32bit-csr.c, features/riscv/32bit-fpu.c,
features/riscv/64bit-cpu.c, features/riscv/64bit-csr.c,
features/riscv/64bit-fpu.c, features/tic6x-c6xp.c,
features/tic6x-core.c, features/tic6x-gp.c, filename-seen-cache.h,
findcmd.c, findvar.c, fork-child.c, gcore.c, gdb_bfd.c, gdb_bfd.h,
gdb_proc_service.h, gdb_regex.c, gdb_select.h, gdb_usleep.c,
gdbarch-selftests.c, gdbthread.h, gdbtypes.h, gnu-nat.c,
go32-nat.c, guile/guile.c, guile/scm-ports.c,
guile/scm-safe-call.c, guile/scm-type.c, i386-fbsd-nat.c,
i386-fbsd-tdep.c, i386-go32-tdep.c, i386-linux-nat.c,
i386-linux-tdep.c, i386-tdep.c, i387-tdep.c,
ia64-libunwind-tdep.c, ia64-linux-nat.c, inf-child.c,
inf-ptrace.c, infcall.c, infcall.h, infcmd.c, inferior-iter.h,
inferior.c, inferior.h, inflow.c, inflow.h, infrun.c, infrun.h,
inline-frame.c, language.h, linespec.c, linux-fork.c, linux-nat.c,
linux-tdep.c, linux-thread-db.c, location.c, machoread.c,
macrotab.h, main.c, maint.c, maint.h, memattr.c, memrange.h,
mi/mi-cmd-break.h, mi/mi-cmd-env.c, mi/mi-cmd-stack.c,
mi/mi-cmd-var.c, mi/mi-interp.c, mi/mi-main.c, mi/mi-parse.h,
minsyms.c, mips-linux-tdep.c, namespace.h,
nat/aarch64-linux-hw-point.c, nat/aarch64-linux-hw-point.h,
nat/aarch64-linux.c, nat/aarch64-sve-linux-ptrace.c,
nat/amd64-linux-siginfo.c, nat/fork-inferior.c,
nat/linux-btrace.c, nat/linux-btrace.h, nat/linux-namespaces.c,
nat/linux-nat.h, nat/linux-osdata.c, nat/linux-personality.c,
nat/linux-procfs.c, nat/linux-ptrace.c, nat/linux-ptrace.h,
nat/linux-waitpid.c, nat/mips-linux-watch.c,
nat/mips-linux-watch.h, nat/ppc-linux.c, nat/x86-dregs.c,
nat/x86-dregs.h, nat/x86-linux-dregs.c, nat/x86-linux.c,
nto-procfs.c, nto-tdep.c, objfile-flags.h, objfiles.c, objfiles.h,
obsd-nat.c, observable.h, osdata.c, p-valprint.c, parse.c,
parser-defs.h, ppc-linux-nat.c, printcmd.c, probe.c, proc-api.c,
procfs.c, producer.c, progspace.h, psymtab.h,
python/py-framefilter.c, python/py-inferior.c, python/py-ref.h,
python/py-type.c, python/python.c, record-btrace.c, record-full.c,
record.c, record.h, regcache-dump.c, regcache.c, regcache.h,
remote-fileio.c, remote-fileio.h, remote-sim.c, remote.c,
riscv-tdep.c, rs6000-aix-tdep.c, rust-exp.y, s12z-tdep.c,
selftest-arch.c, ser-base.c, ser-event.c, ser-pipe.c, ser-tcp.c,
ser-unix.c, skip.c, solib-aix.c, solib-target.c, solib.c,
source-cache.c, source.c, source.h, sparc-nat.c, spu-linux-nat.c,
stack.c, stap-probe.c, symfile-add-flags.h, symfile.c, symfile.h,
symtab.c, symtab.h, target-descriptions.c, target-descriptions.h,
target-memory.c, target.c, target.h, target/waitstatus.c,
target/waitstatus.h, thread-iter.h, thread.c, tilegx-tdep.c,
top.c, top.h, tracefile-tfile.c, tracefile.c, tracepoint.c,
tracepoint.h, tui/tui-io.c, ui-file.c, ui-out.h,
unittests/array-view-selftests.c,
unittests/child-path-selftests.c, unittests/cli-utils-selftests.c,
unittests/common-utils-selftests.c,
unittests/copy_bitwise-selftests.c, unittests/environ-selftests.c,
unittests/format_pieces-selftests.c,
unittests/function-view-selftests.c,
unittests/lookup_name_info-selftests.c,
unittests/memory-map-selftests.c, unittests/memrange-selftests.c,
unittests/mkdir-recursive-selftests.c,
unittests/observable-selftests.c,
unittests/offset-type-selftests.c, unittests/optional-selftests.c,
unittests/parse-connection-spec-selftests.c,
unittests/ptid-selftests.c, unittests/rsp-low-selftests.c,
unittests/scoped_fd-selftests.c,
unittests/scoped_mmap-selftests.c,
unittests/scoped_restore-selftests.c,
unittests/string_view-selftests.c, unittests/style-selftests.c,
unittests/tracepoint-selftests.c, unittests/unpack-selftests.c,
unittests/utils-selftests.c, unittests/xml-utils-selftests.c,
utils.c, utils.h, valarith.c, valops.c, valprint.c, value.c,
value.h, varobj.c, varobj.h, windows-nat.c, x86-linux-nat.c,
xml-support.c, xml-support.h, xml-tdesc.h, xstormy16-tdep.c,
xtensa-linux-nat.c, dwarf2read.h: Change common to gdbsupport.
gdb/gdbserver/ChangeLog
2019-07-09 Tom Tromey <tom@tromey.com>
* configure: Rebuild.
* configure.ac: Change common to gdbsupport.
* acinclude.m4: Change common to gdbsupport.
* Makefile.in (SFILES, OBS, GDBREPLAY_OBS, IPA_OBJS)
(version-generated.c, gdbsupport/%-ipa.o, gdbsupport/%.o): Change
common to gdbsupport.
* ax.c, event-loop.c, fork-child.c, gdb_proc_service.h,
gdbreplay.c, gdbthread.h, hostio-errno.c, hostio.c, i387-fp.c,
inferiors.c, inferiors.h, linux-aarch64-tdesc-selftest.c,
linux-amd64-ipa.c, linux-i386-ipa.c, linux-low.c,
linux-tic6x-low.c, linux-x86-low.c, linux-x86-tdesc-selftest.c,
linux-x86-tdesc.c, lynx-i386-low.c, lynx-low.c, mem-break.h,
nto-x86-low.c, regcache.c, regcache.h, remote-utils.c, server.c,
server.h, spu-low.c, symbol.c, target.h, tdesc.c, tdesc.h,
thread-db.c, tracepoint.c, win32-i386-low.c, win32-low.c: Change
common to gdbsupport.
2019-05-06 10:29:24 +08:00
|
|
|
#include "gdbsupport/common-defs.h"
|
2018-05-31 21:36:48 +08:00
|
|
|
#include "elf/external.h"
|
|
|
|
#include "elf/common.h"
|
2023-02-07 01:24:32 +08:00
|
|
|
#include "aarch64-scalable-linux-ptrace.h"
|
2018-05-31 21:36:48 +08:00
|
|
|
#include "arch/aarch64.h"
|
Rename common to gdbsupport
This is the next patch in the ongoing series to move gdbsever to the
top level.
This patch just renames the "common" directory. The idea is to do
this move in two parts: first rename the directory (this patch), then
move the directory to the top. This approach makes the patches a bit
more tractable.
I chose the name "gdbsupport" for the directory. However, as this
patch was largely written by sed, we could pick a new name without too
much difficulty.
Tested by the buildbot.
gdb/ChangeLog
2019-07-09 Tom Tromey <tom@tromey.com>
* contrib/ari/gdb_ari.sh: Change common to gdbsupport.
* configure: Rebuild.
* configure.ac: Change common to gdbsupport.
* gdbsupport: Rename from common.
* acinclude.m4: Change common to gdbsupport.
* Makefile.in (CONFIG_SRC_SUBDIR, COMMON_SFILES)
(HFILES_NO_SRCDIR, stamp-version, ALLDEPFILES): Change common to
gdbsupport.
* aarch64-tdep.c, ada-lang.c, ada-lang.h, agent.c, alloc.c,
amd64-darwin-tdep.c, amd64-dicos-tdep.c, amd64-fbsd-nat.c,
amd64-fbsd-tdep.c, amd64-linux-nat.c, amd64-linux-tdep.c,
amd64-nbsd-tdep.c, amd64-obsd-tdep.c, amd64-sol2-tdep.c,
amd64-tdep.c, amd64-windows-tdep.c, arch-utils.c,
arch/aarch64-insn.c, arch/aarch64.c, arch/aarch64.h, arch/amd64.c,
arch/amd64.h, arch/arm-get-next-pcs.c, arch/arm-linux.c,
arch/arm.c, arch/i386.c, arch/i386.h, arch/ppc-linux-common.c,
arch/riscv.c, arch/riscv.h, arch/tic6x.c, arm-tdep.c, auto-load.c,
auxv.c, ax-gdb.c, ax-general.c, ax.h, breakpoint.c, breakpoint.h,
btrace.c, btrace.h, build-id.c, build-id.h, c-lang.h, charset.c,
charset.h, cli/cli-cmds.c, cli/cli-cmds.h, cli/cli-decode.c,
cli/cli-dump.c, cli/cli-option.h, cli/cli-script.c,
coff-pe-read.c, command.h, compile/compile-c-support.c,
compile/compile-c.h, compile/compile-cplus-symbols.c,
compile/compile-cplus-types.c, compile/compile-cplus.h,
compile/compile-loc2c.c, compile/compile.c, completer.c,
completer.h, contrib/ari/gdb_ari.sh, corefile.c, corelow.c,
cp-support.c, cp-support.h, cp-valprint.c, csky-tdep.c, ctf.c,
darwin-nat.c, debug.c, defs.h, disasm-selftests.c, disasm.c,
disasm.h, dtrace-probe.c, dwarf-index-cache.c,
dwarf-index-cache.h, dwarf-index-write.c, dwarf2-frame.c,
dwarf2expr.c, dwarf2loc.c, dwarf2read.c, event-loop.c,
event-top.c, exceptions.c, exec.c, extension.h, fbsd-nat.c,
features/aarch64-core.c, features/aarch64-fpu.c,
features/aarch64-pauth.c, features/aarch64-sve.c,
features/i386/32bit-avx.c, features/i386/32bit-avx512.c,
features/i386/32bit-core.c, features/i386/32bit-linux.c,
features/i386/32bit-mpx.c, features/i386/32bit-pkeys.c,
features/i386/32bit-segments.c, features/i386/32bit-sse.c,
features/i386/64bit-avx.c, features/i386/64bit-avx512.c,
features/i386/64bit-core.c, features/i386/64bit-linux.c,
features/i386/64bit-mpx.c, features/i386/64bit-pkeys.c,
features/i386/64bit-segments.c, features/i386/64bit-sse.c,
features/i386/x32-core.c, features/riscv/32bit-cpu.c,
features/riscv/32bit-csr.c, features/riscv/32bit-fpu.c,
features/riscv/64bit-cpu.c, features/riscv/64bit-csr.c,
features/riscv/64bit-fpu.c, features/tic6x-c6xp.c,
features/tic6x-core.c, features/tic6x-gp.c, filename-seen-cache.h,
findcmd.c, findvar.c, fork-child.c, gcore.c, gdb_bfd.c, gdb_bfd.h,
gdb_proc_service.h, gdb_regex.c, gdb_select.h, gdb_usleep.c,
gdbarch-selftests.c, gdbthread.h, gdbtypes.h, gnu-nat.c,
go32-nat.c, guile/guile.c, guile/scm-ports.c,
guile/scm-safe-call.c, guile/scm-type.c, i386-fbsd-nat.c,
i386-fbsd-tdep.c, i386-go32-tdep.c, i386-linux-nat.c,
i386-linux-tdep.c, i386-tdep.c, i387-tdep.c,
ia64-libunwind-tdep.c, ia64-linux-nat.c, inf-child.c,
inf-ptrace.c, infcall.c, infcall.h, infcmd.c, inferior-iter.h,
inferior.c, inferior.h, inflow.c, inflow.h, infrun.c, infrun.h,
inline-frame.c, language.h, linespec.c, linux-fork.c, linux-nat.c,
linux-tdep.c, linux-thread-db.c, location.c, machoread.c,
macrotab.h, main.c, maint.c, maint.h, memattr.c, memrange.h,
mi/mi-cmd-break.h, mi/mi-cmd-env.c, mi/mi-cmd-stack.c,
mi/mi-cmd-var.c, mi/mi-interp.c, mi/mi-main.c, mi/mi-parse.h,
minsyms.c, mips-linux-tdep.c, namespace.h,
nat/aarch64-linux-hw-point.c, nat/aarch64-linux-hw-point.h,
nat/aarch64-linux.c, nat/aarch64-sve-linux-ptrace.c,
nat/amd64-linux-siginfo.c, nat/fork-inferior.c,
nat/linux-btrace.c, nat/linux-btrace.h, nat/linux-namespaces.c,
nat/linux-nat.h, nat/linux-osdata.c, nat/linux-personality.c,
nat/linux-procfs.c, nat/linux-ptrace.c, nat/linux-ptrace.h,
nat/linux-waitpid.c, nat/mips-linux-watch.c,
nat/mips-linux-watch.h, nat/ppc-linux.c, nat/x86-dregs.c,
nat/x86-dregs.h, nat/x86-linux-dregs.c, nat/x86-linux.c,
nto-procfs.c, nto-tdep.c, objfile-flags.h, objfiles.c, objfiles.h,
obsd-nat.c, observable.h, osdata.c, p-valprint.c, parse.c,
parser-defs.h, ppc-linux-nat.c, printcmd.c, probe.c, proc-api.c,
procfs.c, producer.c, progspace.h, psymtab.h,
python/py-framefilter.c, python/py-inferior.c, python/py-ref.h,
python/py-type.c, python/python.c, record-btrace.c, record-full.c,
record.c, record.h, regcache-dump.c, regcache.c, regcache.h,
remote-fileio.c, remote-fileio.h, remote-sim.c, remote.c,
riscv-tdep.c, rs6000-aix-tdep.c, rust-exp.y, s12z-tdep.c,
selftest-arch.c, ser-base.c, ser-event.c, ser-pipe.c, ser-tcp.c,
ser-unix.c, skip.c, solib-aix.c, solib-target.c, solib.c,
source-cache.c, source.c, source.h, sparc-nat.c, spu-linux-nat.c,
stack.c, stap-probe.c, symfile-add-flags.h, symfile.c, symfile.h,
symtab.c, symtab.h, target-descriptions.c, target-descriptions.h,
target-memory.c, target.c, target.h, target/waitstatus.c,
target/waitstatus.h, thread-iter.h, thread.c, tilegx-tdep.c,
top.c, top.h, tracefile-tfile.c, tracefile.c, tracepoint.c,
tracepoint.h, tui/tui-io.c, ui-file.c, ui-out.h,
unittests/array-view-selftests.c,
unittests/child-path-selftests.c, unittests/cli-utils-selftests.c,
unittests/common-utils-selftests.c,
unittests/copy_bitwise-selftests.c, unittests/environ-selftests.c,
unittests/format_pieces-selftests.c,
unittests/function-view-selftests.c,
unittests/lookup_name_info-selftests.c,
unittests/memory-map-selftests.c, unittests/memrange-selftests.c,
unittests/mkdir-recursive-selftests.c,
unittests/observable-selftests.c,
unittests/offset-type-selftests.c, unittests/optional-selftests.c,
unittests/parse-connection-spec-selftests.c,
unittests/ptid-selftests.c, unittests/rsp-low-selftests.c,
unittests/scoped_fd-selftests.c,
unittests/scoped_mmap-selftests.c,
unittests/scoped_restore-selftests.c,
unittests/string_view-selftests.c, unittests/style-selftests.c,
unittests/tracepoint-selftests.c, unittests/unpack-selftests.c,
unittests/utils-selftests.c, unittests/xml-utils-selftests.c,
utils.c, utils.h, valarith.c, valops.c, valprint.c, value.c,
value.h, varobj.c, varobj.h, windows-nat.c, x86-linux-nat.c,
xml-support.c, xml-support.h, xml-tdesc.h, xstormy16-tdep.c,
xtensa-linux-nat.c, dwarf2read.h: Change common to gdbsupport.
gdb/gdbserver/ChangeLog
2019-07-09 Tom Tromey <tom@tromey.com>
* configure: Rebuild.
* configure.ac: Change common to gdbsupport.
* acinclude.m4: Change common to gdbsupport.
* Makefile.in (SFILES, OBS, GDBREPLAY_OBS, IPA_OBJS)
(version-generated.c, gdbsupport/%-ipa.o, gdbsupport/%.o): Change
common to gdbsupport.
* ax.c, event-loop.c, fork-child.c, gdb_proc_service.h,
gdbreplay.c, gdbthread.h, hostio-errno.c, hostio.c, i387-fp.c,
inferiors.c, inferiors.h, linux-aarch64-tdesc-selftest.c,
linux-amd64-ipa.c, linux-i386-ipa.c, linux-low.c,
linux-tic6x-low.c, linux-x86-low.c, linux-x86-tdesc-selftest.c,
linux-x86-tdesc.c, lynx-i386-low.c, lynx-low.c, mem-break.h,
nto-x86-low.c, regcache.c, regcache.h, remote-utils.c, server.c,
server.h, spu-low.c, symbol.c, target.h, tdesc.c, tdesc.h,
thread-db.c, tracepoint.c, win32-i386-low.c, win32-low.c: Change
common to gdbsupport.
2019-05-06 10:29:24 +08:00
|
|
|
#include "gdbsupport/common-regcache.h"
|
|
|
|
#include "gdbsupport/byte-vector.h"
|
2020-03-19 00:06:05 +08:00
|
|
|
#include <endian.h>
|
2018-06-15 19:21:31 +08:00
|
|
|
|
2023-02-07 01:24:32 +08:00
|
|
|
/* See nat/aarch64-scalable-linux-ptrace.h. */
|
2018-05-31 21:36:48 +08:00
|
|
|
|
2018-06-01 23:37:45 +08:00
|
|
|
uint64_t
|
2018-05-31 21:36:48 +08:00
|
|
|
aarch64_sve_get_vq (int tid)
|
|
|
|
{
|
|
|
|
struct iovec iovec;
|
|
|
|
struct user_sve_header header;
|
|
|
|
|
|
|
|
iovec.iov_len = sizeof (header);
|
|
|
|
iovec.iov_base = &header;
|
|
|
|
|
|
|
|
/* Ptrace gives the vector length in bytes. Convert it to VQ, the number of
|
|
|
|
128bit chunks in a Z register. We use VQ because 128bits is the minimum
|
|
|
|
a Z register can increase in size. */
|
|
|
|
|
|
|
|
if (ptrace (PTRACE_GETREGSET, tid, NT_ARM_SVE, &iovec) < 0)
|
|
|
|
{
|
|
|
|
/* SVE is not supported. */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-06-15 19:21:31 +08:00
|
|
|
uint64_t vq = sve_vq_from_vl (header.vl);
|
2018-05-31 21:36:48 +08:00
|
|
|
|
|
|
|
if (!sve_vl_valid (header.vl))
|
|
|
|
{
|
|
|
|
warning (_("Invalid SVE state from kernel; SVE disabled."));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return vq;
|
|
|
|
}
|
2018-06-15 19:21:31 +08:00
|
|
|
|
2023-02-07 01:24:32 +08:00
|
|
|
/* See nat/aarch64-scalable-linux-ptrace.h. */
|
2018-06-15 19:21:31 +08:00
|
|
|
|
2019-04-15 19:31:21 +08:00
|
|
|
bool
|
|
|
|
aarch64_sve_set_vq (int tid, uint64_t vq)
|
|
|
|
{
|
|
|
|
struct iovec iovec;
|
|
|
|
struct user_sve_header header;
|
|
|
|
|
|
|
|
iovec.iov_len = sizeof (header);
|
|
|
|
iovec.iov_base = &header;
|
|
|
|
|
|
|
|
if (ptrace (PTRACE_GETREGSET, tid, NT_ARM_SVE, &iovec) < 0)
|
|
|
|
{
|
|
|
|
/* SVE is not supported. */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
header.vl = sve_vl_from_vq (vq);
|
|
|
|
|
|
|
|
if (ptrace (PTRACE_SETREGSET, tid, NT_ARM_SVE, &iovec) < 0)
|
|
|
|
{
|
|
|
|
/* Vector length change failed. */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-02-07 01:24:32 +08:00
|
|
|
/* See nat/aarch64-scalable-linux-ptrace.h. */
|
2019-04-15 19:31:21 +08:00
|
|
|
|
|
|
|
bool
|
|
|
|
aarch64_sve_set_vq (int tid, struct reg_buffer_common *reg_buf)
|
|
|
|
{
|
[AArch64] When unavailable, fetch VG from ptrace.
I was doing some SVE tests on system QEMU and noticed quite a few failures
related to inferior function calls. Any attempt to do an inferior function
call would result in the following:
Unable to set VG register.: Success.
This happens because, after an inferior function call, GDB attempts to restore
the regcache state and updates the SVE register in order. Since the Z registers
show up before the VG register, VG is still INVALID by the time the first Z
register is being updated. So when executing the following code in
aarch64_sve_set_vq:
if (reg_buf->get_register_status (AARCH64_SVE_VG_REGNUM) != REG_VALID)
return false;
By returning false, we signal something is wrong, then we get to this:
/* First store vector length to the thread. This is done first to ensure the
ptrace buffers read from the kernel are the correct size. */
if (!aarch64_sve_set_vq (tid, regcache))
perror_with_name (_("Unable to set VG register."));
Ideally we'd always have a valid VG before attempting to set the Z registers,
but in this case the ordering of registers doesn't make that possible.
I considered reordering the registers to put VG before the Z registers, like
the DWARF numbering, but that would break backwards compatibility with
existing implementations. Also, the Z register numbering is pinned to the V
registers, and adding VG before Z would create a gap for non-SVE targets,
since we wouldn't be able to undefine VG for non-SVE targets.
As a compromise, it seems we can safely fetch the VG register value from
ptrace. The value in the kernel is likely the updated value anyway.
This patch fixed all the failures i saw in the testsuite and caused no further
regressions.
gdb/ChangeLog:
2020-03-19 Luis Machado <luis.machado@linaro.org>
* nat/aarch64-sve-linux-ptrace.c (aarch64_sve_set_vq): If vg is not
valid, fetch vg value from ptrace.
2020-03-17 01:09:25 +08:00
|
|
|
uint64_t reg_vg = 0;
|
|
|
|
|
|
|
|
/* The VG register may not be valid if we've not collected any value yet.
|
|
|
|
This can happen, for example, if we're restoring the regcache after an
|
|
|
|
inferior function call, and the VG register comes after the Z
|
|
|
|
registers. */
|
2019-04-15 19:31:21 +08:00
|
|
|
if (reg_buf->get_register_status (AARCH64_SVE_VG_REGNUM) != REG_VALID)
|
2021-05-28 03:01:28 +08:00
|
|
|
{
|
|
|
|
/* If vg is not available yet, fetch it from ptrace. The VG value from
|
|
|
|
ptrace is likely the correct one. */
|
|
|
|
uint64_t vq = aarch64_sve_get_vq (tid);
|
2019-04-15 19:31:21 +08:00
|
|
|
|
2021-05-28 03:01:28 +08:00
|
|
|
/* If something went wrong, just bail out. */
|
|
|
|
if (vq == 0)
|
|
|
|
return false;
|
[AArch64] When unavailable, fetch VG from ptrace.
I was doing some SVE tests on system QEMU and noticed quite a few failures
related to inferior function calls. Any attempt to do an inferior function
call would result in the following:
Unable to set VG register.: Success.
This happens because, after an inferior function call, GDB attempts to restore
the regcache state and updates the SVE register in order. Since the Z registers
show up before the VG register, VG is still INVALID by the time the first Z
register is being updated. So when executing the following code in
aarch64_sve_set_vq:
if (reg_buf->get_register_status (AARCH64_SVE_VG_REGNUM) != REG_VALID)
return false;
By returning false, we signal something is wrong, then we get to this:
/* First store vector length to the thread. This is done first to ensure the
ptrace buffers read from the kernel are the correct size. */
if (!aarch64_sve_set_vq (tid, regcache))
perror_with_name (_("Unable to set VG register."));
Ideally we'd always have a valid VG before attempting to set the Z registers,
but in this case the ordering of registers doesn't make that possible.
I considered reordering the registers to put VG before the Z registers, like
the DWARF numbering, but that would break backwards compatibility with
existing implementations. Also, the Z register numbering is pinned to the V
registers, and adding VG before Z would create a gap for non-SVE targets,
since we wouldn't be able to undefine VG for non-SVE targets.
As a compromise, it seems we can safely fetch the VG register value from
ptrace. The value in the kernel is likely the updated value anyway.
This patch fixed all the failures i saw in the testsuite and caused no further
regressions.
gdb/ChangeLog:
2020-03-19 Luis Machado <luis.machado@linaro.org>
* nat/aarch64-sve-linux-ptrace.c (aarch64_sve_set_vq): If vg is not
valid, fetch vg value from ptrace.
2020-03-17 01:09:25 +08:00
|
|
|
|
2021-05-28 03:01:28 +08:00
|
|
|
reg_vg = sve_vg_from_vq (vq);
|
|
|
|
}
|
[AArch64] When unavailable, fetch VG from ptrace.
I was doing some SVE tests on system QEMU and noticed quite a few failures
related to inferior function calls. Any attempt to do an inferior function
call would result in the following:
Unable to set VG register.: Success.
This happens because, after an inferior function call, GDB attempts to restore
the regcache state and updates the SVE register in order. Since the Z registers
show up before the VG register, VG is still INVALID by the time the first Z
register is being updated. So when executing the following code in
aarch64_sve_set_vq:
if (reg_buf->get_register_status (AARCH64_SVE_VG_REGNUM) != REG_VALID)
return false;
By returning false, we signal something is wrong, then we get to this:
/* First store vector length to the thread. This is done first to ensure the
ptrace buffers read from the kernel are the correct size. */
if (!aarch64_sve_set_vq (tid, regcache))
perror_with_name (_("Unable to set VG register."));
Ideally we'd always have a valid VG before attempting to set the Z registers,
but in this case the ordering of registers doesn't make that possible.
I considered reordering the registers to put VG before the Z registers, like
the DWARF numbering, but that would break backwards compatibility with
existing implementations. Also, the Z register numbering is pinned to the V
registers, and adding VG before Z would create a gap for non-SVE targets,
since we wouldn't be able to undefine VG for non-SVE targets.
As a compromise, it seems we can safely fetch the VG register value from
ptrace. The value in the kernel is likely the updated value anyway.
This patch fixed all the failures i saw in the testsuite and caused no further
regressions.
gdb/ChangeLog:
2020-03-19 Luis Machado <luis.machado@linaro.org>
* nat/aarch64-sve-linux-ptrace.c (aarch64_sve_set_vq): If vg is not
valid, fetch vg value from ptrace.
2020-03-17 01:09:25 +08:00
|
|
|
else
|
|
|
|
reg_buf->raw_collect (AARCH64_SVE_VG_REGNUM, ®_vg);
|
2019-04-15 19:31:21 +08:00
|
|
|
|
|
|
|
return aarch64_sve_set_vq (tid, sve_vq_from_vg (reg_vg));
|
|
|
|
}
|
|
|
|
|
2023-02-07 01:24:32 +08:00
|
|
|
/* See nat/aarch64-scalable-linux-ptrace.h. */
|
2019-04-15 19:31:21 +08:00
|
|
|
|
2018-06-15 19:21:31 +08:00
|
|
|
std::unique_ptr<gdb_byte[]>
|
|
|
|
aarch64_sve_get_sveregs (int tid)
|
|
|
|
{
|
|
|
|
struct iovec iovec;
|
|
|
|
uint64_t vq = aarch64_sve_get_vq (tid);
|
|
|
|
|
|
|
|
if (vq == 0)
|
|
|
|
perror_with_name (_("Unable to fetch SVE register header"));
|
|
|
|
|
|
|
|
/* A ptrace call with NT_ARM_SVE will return a header followed by either a
|
|
|
|
dump of all the SVE and FP registers, or an fpsimd structure (identical to
|
|
|
|
the one returned by NT_FPREGSET) if the kernel has not yet executed any
|
|
|
|
SVE code. Make sure we allocate enough space for a full SVE dump. */
|
|
|
|
|
|
|
|
iovec.iov_len = SVE_PT_SIZE (vq, SVE_PT_REGS_SVE);
|
|
|
|
std::unique_ptr<gdb_byte[]> buf (new gdb_byte[iovec.iov_len]);
|
|
|
|
iovec.iov_base = buf.get ();
|
|
|
|
|
|
|
|
if (ptrace (PTRACE_GETREGSET, tid, NT_ARM_SVE, &iovec) < 0)
|
|
|
|
perror_with_name (_("Unable to fetch SVE registers"));
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2020-03-19 00:06:05 +08:00
|
|
|
/* If we are running in BE mode, byteswap the contents
|
|
|
|
of SRC to DST for SIZE bytes. Other, just copy the contents
|
|
|
|
from SRC to DST. */
|
|
|
|
|
|
|
|
static void
|
|
|
|
aarch64_maybe_swab128 (gdb_byte *dst, const gdb_byte *src, size_t size)
|
|
|
|
{
|
|
|
|
gdb_assert (src != nullptr && dst != nullptr);
|
|
|
|
gdb_assert (size > 1);
|
|
|
|
|
|
|
|
#if (__BYTE_ORDER == __BIG_ENDIAN)
|
|
|
|
for (int i = 0; i < size - 1; i++)
|
|
|
|
dst[i] = src[size - i];
|
|
|
|
#else
|
|
|
|
memcpy (dst, src, size);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-02-07 01:24:32 +08:00
|
|
|
/* See nat/aarch64-scalable-linux-ptrace.h. */
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
void
|
|
|
|
aarch64_sve_regs_copy_to_reg_buf (struct reg_buffer_common *reg_buf,
|
|
|
|
const void *buf)
|
|
|
|
{
|
|
|
|
char *base = (char *) buf;
|
|
|
|
struct user_sve_header *header = (struct user_sve_header *) buf;
|
|
|
|
|
2019-04-15 19:31:21 +08:00
|
|
|
uint64_t vq = sve_vq_from_vl (header->vl);
|
|
|
|
uint64_t vg = sve_vg_from_vl (header->vl);
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
/* Sanity check the data in the header. */
|
|
|
|
if (!sve_vl_valid (header->vl)
|
|
|
|
|| SVE_PT_SIZE (vq, header->flags) != header->size)
|
|
|
|
error (_("Invalid SVE header from kernel."));
|
|
|
|
|
2019-04-15 19:31:21 +08:00
|
|
|
/* Update VG. Note, the registers in the regcache will already be of the
|
|
|
|
correct length. */
|
|
|
|
reg_buf->raw_supply (AARCH64_SVE_VG_REGNUM, &vg);
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
if (HAS_SVE_STATE (*header))
|
|
|
|
{
|
|
|
|
/* The register dump contains a set of SVE registers. */
|
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_Z_REGS_NUM; i++)
|
|
|
|
reg_buf->raw_supply (AARCH64_SVE_Z0_REGNUM + i,
|
|
|
|
base + SVE_PT_SVE_ZREG_OFFSET (vq, i));
|
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_P_REGS_NUM; i++)
|
|
|
|
reg_buf->raw_supply (AARCH64_SVE_P0_REGNUM + i,
|
|
|
|
base + SVE_PT_SVE_PREG_OFFSET (vq, i));
|
|
|
|
|
|
|
|
reg_buf->raw_supply (AARCH64_SVE_FFR_REGNUM,
|
|
|
|
base + SVE_PT_SVE_FFR_OFFSET (vq));
|
|
|
|
reg_buf->raw_supply (AARCH64_FPSR_REGNUM,
|
|
|
|
base + SVE_PT_SVE_FPSR_OFFSET (vq));
|
|
|
|
reg_buf->raw_supply (AARCH64_FPCR_REGNUM,
|
|
|
|
base + SVE_PT_SVE_FPCR_OFFSET (vq));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-03-19 00:06:05 +08:00
|
|
|
/* WARNING: SIMD state is laid out in memory in target-endian format,
|
|
|
|
while SVE state is laid out in an endianness-independent format (LE).
|
|
|
|
|
|
|
|
So we have a couple cases to consider:
|
|
|
|
|
|
|
|
1 - If the target is big endian, then SIMD state is big endian,
|
|
|
|
requiring a byteswap.
|
|
|
|
|
|
|
|
2 - If the target is little endian, then SIMD state is little endian,
|
|
|
|
which matches the SVE format, so no byteswap is needed. */
|
|
|
|
|
2018-06-15 19:21:31 +08:00
|
|
|
/* There is no SVE state yet - the register dump contains a fpsimd
|
|
|
|
structure instead. These registers still exist in the hardware, but
|
|
|
|
the kernel has not yet initialised them, and so they will be null. */
|
|
|
|
|
2020-03-19 00:06:05 +08:00
|
|
|
gdb_byte *reg = (gdb_byte *) alloca (SVE_PT_SVE_ZREG_SIZE (vq));
|
2018-06-15 19:21:31 +08:00
|
|
|
struct user_fpsimd_state *fpsimd
|
|
|
|
= (struct user_fpsimd_state *)(base + SVE_PT_FPSIMD_OFFSET);
|
|
|
|
|
2020-03-19 00:06:05 +08:00
|
|
|
/* Make sure we have a zeroed register buffer. We will need the zero
|
|
|
|
padding below. */
|
|
|
|
memset (reg, 0, SVE_PT_SVE_ZREG_SIZE (vq));
|
|
|
|
|
2018-06-15 19:21:31 +08:00
|
|
|
/* Copy across the V registers from fpsimd structure to the Z registers,
|
|
|
|
ensuring the non overlapping state is set to null. */
|
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_Z_REGS_NUM; i++)
|
|
|
|
{
|
2020-03-19 00:06:05 +08:00
|
|
|
/* Handle big endian/little endian SIMD/SVE conversion. */
|
|
|
|
aarch64_maybe_swab128 (reg, (const gdb_byte *) &fpsimd->vregs[i],
|
|
|
|
V_REGISTER_SIZE);
|
|
|
|
reg_buf->raw_supply (AARCH64_SVE_Z0_REGNUM + i, reg);
|
2018-06-15 19:21:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
reg_buf->raw_supply (AARCH64_FPSR_REGNUM, &fpsimd->fpsr);
|
|
|
|
reg_buf->raw_supply (AARCH64_FPCR_REGNUM, &fpsimd->fpcr);
|
|
|
|
|
|
|
|
/* Clear the SVE only registers. */
|
2020-03-19 00:06:05 +08:00
|
|
|
memset (reg, 0, SVE_PT_SVE_ZREG_SIZE (vq));
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_P_REGS_NUM; i++)
|
2020-03-19 00:06:05 +08:00
|
|
|
reg_buf->raw_supply (AARCH64_SVE_P0_REGNUM + i, reg);
|
2018-06-15 19:21:31 +08:00
|
|
|
|
2020-03-19 00:06:05 +08:00
|
|
|
reg_buf->raw_supply (AARCH64_SVE_FFR_REGNUM, reg);
|
2018-06-15 19:21:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-07 01:24:32 +08:00
|
|
|
/* See nat/aarch64-scalable-linux-ptrace.h. */
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
void
|
|
|
|
aarch64_sve_regs_copy_from_reg_buf (const struct reg_buffer_common *reg_buf,
|
|
|
|
void *buf)
|
|
|
|
{
|
|
|
|
struct user_sve_header *header = (struct user_sve_header *) buf;
|
|
|
|
char *base = (char *) buf;
|
2019-04-15 19:31:21 +08:00
|
|
|
uint64_t vq = sve_vq_from_vl (header->vl);
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
/* Sanity check the data in the header. */
|
|
|
|
if (!sve_vl_valid (header->vl)
|
|
|
|
|| SVE_PT_SIZE (vq, header->flags) != header->size)
|
|
|
|
error (_("Invalid SVE header from kernel."));
|
|
|
|
|
|
|
|
if (!HAS_SVE_STATE (*header))
|
|
|
|
{
|
|
|
|
/* There is no SVE state yet - the register dump contains a fpsimd
|
|
|
|
structure instead. Where possible we want to write the reg_buf data
|
|
|
|
back to the kernel using the fpsimd structure. However, if we cannot
|
|
|
|
then we'll need to reformat the fpsimd into a full SVE structure,
|
|
|
|
resulting in the initialization of SVE state written back to the
|
|
|
|
kernel, which is why we try to avoid it. */
|
|
|
|
|
|
|
|
bool has_sve_state = false;
|
2020-03-19 00:06:05 +08:00
|
|
|
gdb_byte *reg = (gdb_byte *) alloca (SVE_PT_SVE_ZREG_SIZE (vq));
|
2018-06-15 19:21:31 +08:00
|
|
|
struct user_fpsimd_state *fpsimd
|
|
|
|
= (struct user_fpsimd_state *)(base + SVE_PT_FPSIMD_OFFSET);
|
|
|
|
|
2020-03-19 00:06:05 +08:00
|
|
|
memset (reg, 0, SVE_PT_SVE_ZREG_SIZE (vq));
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
/* Check in the reg_buf if any of the Z registers are set after the
|
|
|
|
first 128 bits, or if any of the other SVE registers are set. */
|
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_Z_REGS_NUM; i++)
|
|
|
|
{
|
|
|
|
has_sve_state |= reg_buf->raw_compare (AARCH64_SVE_Z0_REGNUM + i,
|
2020-03-19 00:06:05 +08:00
|
|
|
reg, sizeof (__int128_t));
|
2018-06-15 19:21:31 +08:00
|
|
|
if (has_sve_state)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_sve_state)
|
|
|
|
for (int i = 0; i < AARCH64_SVE_P_REGS_NUM; i++)
|
|
|
|
{
|
|
|
|
has_sve_state |= reg_buf->raw_compare (AARCH64_SVE_P0_REGNUM + i,
|
2020-03-19 00:06:05 +08:00
|
|
|
reg, 0);
|
2018-06-15 19:21:31 +08:00
|
|
|
if (has_sve_state)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_sve_state)
|
|
|
|
has_sve_state |= reg_buf->raw_compare (AARCH64_SVE_FFR_REGNUM,
|
2020-03-19 00:06:05 +08:00
|
|
|
reg, 0);
|
2018-06-15 19:21:31 +08:00
|
|
|
|
|
|
|
/* If no SVE state exists, then use the existing fpsimd structure to
|
|
|
|
write out state and return. */
|
|
|
|
if (!has_sve_state)
|
|
|
|
{
|
2020-03-19 00:06:05 +08:00
|
|
|
/* WARNING: SIMD state is laid out in memory in target-endian format,
|
|
|
|
while SVE state is laid out in an endianness-independent format
|
|
|
|
(LE).
|
|
|
|
|
|
|
|
So we have a couple cases to consider:
|
|
|
|
|
|
|
|
1 - If the target is big endian, then SIMD state is big endian,
|
|
|
|
requiring a byteswap.
|
|
|
|
|
|
|
|
2 - If the target is little endian, then SIMD state is little
|
|
|
|
endian, which matches the SVE format, so no byteswap is needed. */
|
|
|
|
|
2018-06-15 19:21:31 +08:00
|
|
|
/* The collects of the Z registers will overflow the size of a vreg.
|
|
|
|
There is enough space in the structure to allow for this, but we
|
|
|
|
cannot overflow into the next register as we might not be
|
|
|
|
collecting every register. */
|
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_Z_REGS_NUM; i++)
|
|
|
|
{
|
|
|
|
if (REG_VALID
|
|
|
|
== reg_buf->get_register_status (AARCH64_SVE_Z0_REGNUM + i))
|
|
|
|
{
|
2020-03-19 00:06:05 +08:00
|
|
|
reg_buf->raw_collect (AARCH64_SVE_Z0_REGNUM + i, reg);
|
|
|
|
/* Handle big endian/little endian SIMD/SVE conversion. */
|
|
|
|
aarch64_maybe_swab128 ((gdb_byte *) &fpsimd->vregs[i], reg,
|
|
|
|
V_REGISTER_SIZE);
|
2018-06-15 19:21:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (REG_VALID == reg_buf->get_register_status (AARCH64_FPSR_REGNUM))
|
|
|
|
reg_buf->raw_collect (AARCH64_FPSR_REGNUM, &fpsimd->fpsr);
|
|
|
|
if (REG_VALID == reg_buf->get_register_status (AARCH64_FPCR_REGNUM))
|
|
|
|
reg_buf->raw_collect (AARCH64_FPCR_REGNUM, &fpsimd->fpcr);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise, reformat the fpsimd structure into a full SVE set, by
|
|
|
|
expanding the V registers (working backwards so we don't splat
|
|
|
|
registers before they are copied) and using null for everything else.
|
|
|
|
Note that enough space for a full SVE dump was originally allocated
|
|
|
|
for base. */
|
|
|
|
|
|
|
|
header->flags |= SVE_PT_REGS_SVE;
|
|
|
|
header->size = SVE_PT_SIZE (vq, SVE_PT_REGS_SVE);
|
|
|
|
|
|
|
|
memcpy (base + SVE_PT_SVE_FPSR_OFFSET (vq), &fpsimd->fpsr,
|
|
|
|
sizeof (uint32_t));
|
|
|
|
memcpy (base + SVE_PT_SVE_FPCR_OFFSET (vq), &fpsimd->fpcr,
|
|
|
|
sizeof (uint32_t));
|
|
|
|
|
|
|
|
for (int i = AARCH64_SVE_Z_REGS_NUM; i >= 0 ; i--)
|
|
|
|
{
|
|
|
|
memcpy (base + SVE_PT_SVE_ZREG_OFFSET (vq, i), &fpsimd->vregs[i],
|
|
|
|
sizeof (__int128_t));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Replace the kernel values with those from reg_buf. */
|
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_Z_REGS_NUM; i++)
|
|
|
|
if (REG_VALID == reg_buf->get_register_status (AARCH64_SVE_Z0_REGNUM + i))
|
|
|
|
reg_buf->raw_collect (AARCH64_SVE_Z0_REGNUM + i,
|
|
|
|
base + SVE_PT_SVE_ZREG_OFFSET (vq, i));
|
|
|
|
|
|
|
|
for (int i = 0; i < AARCH64_SVE_P_REGS_NUM; i++)
|
|
|
|
if (REG_VALID == reg_buf->get_register_status (AARCH64_SVE_P0_REGNUM + i))
|
|
|
|
reg_buf->raw_collect (AARCH64_SVE_P0_REGNUM + i,
|
|
|
|
base + SVE_PT_SVE_PREG_OFFSET (vq, i));
|
|
|
|
|
|
|
|
if (REG_VALID == reg_buf->get_register_status (AARCH64_SVE_FFR_REGNUM))
|
|
|
|
reg_buf->raw_collect (AARCH64_SVE_FFR_REGNUM,
|
|
|
|
base + SVE_PT_SVE_FFR_OFFSET (vq));
|
|
|
|
if (REG_VALID == reg_buf->get_register_status (AARCH64_FPSR_REGNUM))
|
|
|
|
reg_buf->raw_collect (AARCH64_FPSR_REGNUM,
|
|
|
|
base + SVE_PT_SVE_FPSR_OFFSET (vq));
|
|
|
|
if (REG_VALID == reg_buf->get_register_status (AARCH64_FPCR_REGNUM))
|
|
|
|
reg_buf->raw_collect (AARCH64_FPCR_REGNUM,
|
|
|
|
base + SVE_PT_SVE_FPCR_OFFSET (vq));
|
|
|
|
|
|
|
|
}
|