Switch the license of all .c files to GPLv3.
[deliverable/binutils-gdb.git] / gdb / mips64obsd-tdep.c
CommitLineData
abc08782
MK
1/* Target-dependent code for OpenBSD/mips64.
2
6aba47ca 3 Copyright (C) 2004, 2007 Free Software Foundation, Inc.
abc08782
MK
4
5 This file is part of GDB.
6
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
a9762ec7 9 the Free Software Foundation; either version 3 of the License, or
abc08782
MK
10 (at your option) any later version.
11
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
a9762ec7 18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
abc08782
MK
19
20#include "defs.h"
21#include "osabi.h"
22#include "regcache.h"
23#include "regset.h"
8e763687
MK
24#include "trad-frame.h"
25#include "tramp-frame.h"
abc08782 26
8e763687 27#include "gdb_assert.h"
abc08782
MK
28#include "gdb_string.h"
29
30#include "mips-tdep.h"
31#include "solib-svr4.h"
32
33#define MIPS64OBSD_NUM_REGS 73
34
35/* Core file support. */
36
37/* Supply register REGNUM from the buffer specified by GREGS and LEN
38 in the general-purpose register set REGSET to register cache
39 REGCACHE. If REGNUM is -1, do this for all registers in REGSET. */
40
41static void
42mips64obsd_supply_gregset (const struct regset *regset,
43 struct regcache *regcache, int regnum,
44 const void *gregs, size_t len)
45{
46 const char *regs = gregs;
47 int i;
48
49 for (i = 0; i < MIPS64OBSD_NUM_REGS; i++)
50 {
51 if (regnum == i || regnum == -1)
52 regcache_raw_supply (regcache, i, regs + i * 8);
53 }
54}
55
56/* OpenBSD/mips64 register set. */
57
58static struct regset mips64obsd_gregset =
59{
60 NULL,
61 mips64obsd_supply_gregset
62};
63
64/* Return the appropriate register set for the core section identified
65 by SECT_NAME and SECT_SIZE. */
66
67static const struct regset *
68mips64obsd_regset_from_core_section (struct gdbarch *gdbarch,
69 const char *sect_name, size_t sect_size)
70{
71 if (strcmp (sect_name, ".reg") == 0 && sect_size >= MIPS64OBSD_NUM_REGS * 8)
72 return &mips64obsd_gregset;
73
74 return NULL;
75}
76\f
77
8e763687
MK
78/* Signal trampolines. */
79
80static void
81mips64obsd_sigframe_init (const struct tramp_frame *self,
82 struct frame_info *next_frame,
83 struct trad_frame_cache *cache,
84 CORE_ADDR func)
85{
86 struct gdbarch *gdbarch = get_frame_arch (next_frame);
87 CORE_ADDR sp, sigcontext_addr, addr;
88 int regnum;
89
90 /* We find the appropriate instance of `struct sigcontext' at a
91 fixed offset in the signal frame. */
f57d151a
UW
92 sp = frame_unwind_register_signed (next_frame,
93 MIPS_SP_REGNUM
94 + gdbarch_num_regs (current_gdbarch));
8e763687
MK
95 sigcontext_addr = sp + 32;
96
97 /* PC. */
98 regnum = mips_regnum (gdbarch)->pc;
f57d151a
UW
99 trad_frame_set_reg_addr (cache,
100 regnum + gdbarch_num_regs (current_gdbarch),
101 sigcontext_addr + 16);
8e763687
MK
102
103 /* GPRs. */
104 for (regnum = MIPS_AT_REGNUM, addr = sigcontext_addr + 32;
105 regnum <= MIPS_RA_REGNUM; regnum++, addr += 8)
f57d151a
UW
106 trad_frame_set_reg_addr (cache,
107 regnum + gdbarch_num_regs (current_gdbarch),
108 addr);
8e763687
MK
109
110 /* HI and LO. */
111 regnum = mips_regnum (gdbarch)->lo;
f57d151a
UW
112 trad_frame_set_reg_addr (cache,
113 regnum + gdbarch_num_regs (current_gdbarch),
114 sigcontext_addr + 280);
8e763687 115 regnum = mips_regnum (gdbarch)->hi;
f57d151a
UW
116 trad_frame_set_reg_addr (cache,
117 regnum + gdbarch_num_regs (current_gdbarch),
118 sigcontext_addr + 288);
8e763687
MK
119
120 /* TODO: Handle the floating-point registers. */
121
122 trad_frame_set_id (cache, frame_id_build (sp, func));
123}
124
125static const struct tramp_frame mips64obsd_sigframe =
126{
127 SIGTRAMP_FRAME,
128 MIPS_INSN32_SIZE,
129 {
130 { 0x67a40020, -1 }, /* daddiu a0,sp,32 */
131 { 0x24020067, -1 }, /* li v0,103 */
132 { 0x0000000c, -1 }, /* syscall */
133 { 0x0000000d, -1 }, /* break */
134 { TRAMP_SENTINEL_INSN, -1 }
135 },
136 mips64obsd_sigframe_init
137};
138
139\f
abc08782
MK
140static void
141mips64obsd_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
142{
8e763687
MK
143 /* OpenBSD/mips64 only supports the n64 ABI, but the braindamaged
144 way GDB works, forces us to pretend we can handle them all. */
145
abc08782
MK
146 set_gdbarch_regset_from_core_section
147 (gdbarch, mips64obsd_regset_from_core_section);
148
8e763687
MK
149 tramp_frame_prepend_unwinder (gdbarch, &mips64obsd_sigframe);
150
8e763687 151 /* OpenBSD/mips64 has SVR4-style shared libraries. */
abc08782
MK
152 set_solib_svr4_fetch_link_map_offsets
153 (gdbarch, svr4_lp64_fetch_link_map_offsets);
154}
155
156\f
157/* Provide a prototype to silence -Wmissing-prototypes. */
158void _initialize_mips64obsd_tdep (void);
159
160void
161_initialize_mips64obsd_tdep (void)
162{
163 gdbarch_register_osabi (bfd_arch_mips, 0, GDB_OSABI_OPENBSD_ELF,
164 mips64obsd_init_abi);
165}
This page took 0.206722 seconds and 4 git commands to generate.