]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blame - gdb/vax-tdep.c
2004-01-17 Andrew Cagney <cagney@redhat.com>
[thirdparty/binutils-gdb.git] / gdb / vax-tdep.c
CommitLineData
c906108c 1/* Print VAX instructions for GDB, the GNU debugger.
4be87837 2 Copyright 1986, 1989, 1991, 1992, 1995, 1996, 1998, 1999, 2000, 2002, 2003
b6ba6518 3 Free Software Foundation, Inc.
c906108c 4
c5aa993b 5 This file is part of GDB.
c906108c 6
c5aa993b
JM
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
9 the Free Software Foundation; either version 2 of the License, or
10 (at your option) any later version.
c906108c 11
c5aa993b
JM
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.
c906108c 16
c5aa993b
JM
17 You should have received a copy of the GNU General Public License
18 along with this program; if not, write to the Free Software
19 Foundation, Inc., 59 Temple Place - Suite 330,
20 Boston, MA 02111-1307, USA. */
c906108c
SS
21
22#include "defs.h"
23#include "symtab.h"
24#include "opcode/vax.h"
c11c3a98 25#include "gdbcore.h"
f267bd6a 26#include "inferior.h"
a33f7558 27#include "regcache.h"
c11c3a98
AC
28#include "frame.h"
29#include "value.h"
f267bd6a 30#include "arch-utils.h"
9bbe19fb 31#include "gdb_string.h"
4be87837 32#include "osabi.h"
a89aa300 33#include "dis-asm.h"
f267bd6a
JT
34
35#include "vax-tdep.h"
36
37static gdbarch_register_name_ftype vax_register_name;
f267bd6a
JT
38
39static gdbarch_skip_prologue_ftype vax_skip_prologue;
f267bd6a 40static gdbarch_frame_num_args_ftype vax_frame_num_args;
618ce49f 41static gdbarch_deprecated_frame_chain_ftype vax_frame_chain;
f267bd6a 42
26e9b323 43static gdbarch_deprecated_extract_return_value_ftype vax_extract_return_value;
26e9b323 44static gdbarch_deprecated_extract_struct_value_address_ftype
f267bd6a
JT
45 vax_extract_struct_value_address;
46
f3824013 47static gdbarch_deprecated_push_dummy_frame_ftype vax_push_dummy_frame;
c906108c 48\f
fa88f677 49static const char *
51eb8b08
JT
50vax_register_name (int regno)
51{
52 static char *register_names[] =
53 {
54 "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
55 "r8", "r9", "r10", "r11", "ap", "fp", "sp", "pc",
56 "ps",
57 };
58
59 if (regno < 0)
60 return (NULL);
61 if (regno >= (sizeof(register_names) / sizeof(*register_names)))
62 return (NULL);
63 return (register_names[regno]);
64}
65
f267bd6a 66static int
51eb8b08
JT
67vax_register_byte (int regno)
68{
69 return (regno * 4);
70}
71
f267bd6a 72static int
51eb8b08
JT
73vax_register_raw_size (int regno)
74{
75 return (4);
76}
77
f267bd6a 78static int
51eb8b08
JT
79vax_register_virtual_size (int regno)
80{
81 return (4);
82}
83
f267bd6a 84static struct type *
51eb8b08
JT
85vax_register_virtual_type (int regno)
86{
87 return (builtin_type_int);
88}
89\f
f267bd6a 90static void
ab62c900
JT
91vax_frame_init_saved_regs (struct frame_info *frame)
92{
93 int regnum, regmask;
94 CORE_ADDR next_addr;
95
1b1d3794 96 if (deprecated_get_frame_saved_regs (frame))
ab62c900
JT
97 return;
98
99 frame_saved_regs_zalloc (frame);
100
1e2330ba 101 regmask = read_memory_integer (get_frame_base (frame) + 4, 4) >> 16;
ab62c900 102
1e2330ba 103 next_addr = get_frame_base (frame) + 16;
ab62c900
JT
104
105 /* regmask's low bit is for register 0, which is the first one
106 what would be pushed. */
1d049c5e 107 for (regnum = 0; regnum < VAX_AP_REGNUM; regnum++)
ab62c900
JT
108 {
109 if (regmask & (1 << regnum))
1b1d3794 110 deprecated_get_frame_saved_regs (frame)[regnum] = next_addr += 4;
ab62c900
JT
111 }
112
1b1d3794 113 deprecated_get_frame_saved_regs (frame)[SP_REGNUM] = next_addr + 4;
0ba6dca9 114 if (regmask & (1 << DEPRECATED_FP_REGNUM))
1b1d3794 115 deprecated_get_frame_saved_regs (frame)[SP_REGNUM] +=
ab62c900
JT
116 4 + (4 * read_memory_integer (next_addr + 4, 4));
117
1b1d3794
AC
118 deprecated_get_frame_saved_regs (frame)[PC_REGNUM] = get_frame_base (frame) + 16;
119 deprecated_get_frame_saved_regs (frame)[DEPRECATED_FP_REGNUM] = get_frame_base (frame) + 12;
120 deprecated_get_frame_saved_regs (frame)[VAX_AP_REGNUM] = get_frame_base (frame) + 8;
121 deprecated_get_frame_saved_regs (frame)[PS_REGNUM] = get_frame_base (frame) + 4;
ab62c900 122}
5516aa92 123
f407986f
AC
124/* Get saved user PC for sigtramp from sigcontext for BSD style sigtramp. */
125
126static CORE_ADDR
127vax_sigtramp_saved_pc (struct frame_info *frame)
128{
129 CORE_ADDR sigcontext_addr;
130 char *buf;
131 int ptrbytes = TYPE_LENGTH (builtin_type_void_func_ptr);
132 int sigcontext_offs = (2 * TARGET_INT_BIT) / TARGET_CHAR_BIT;
133
134 buf = alloca (ptrbytes);
135 /* Get sigcontext address, it is the third parameter on the stack. */
11c02a10 136 if (get_next_frame (frame))
f407986f 137 sigcontext_addr = read_memory_typed_address
42efa47a 138 (DEPRECATED_FRAME_ARGS_ADDRESS (get_next_frame (frame))
11c02a10 139 + FRAME_ARGS_SKIP + sigcontext_offs,
f407986f
AC
140 builtin_type_void_data_ptr);
141 else
142 sigcontext_addr = read_memory_typed_address
143 (read_register (SP_REGNUM) + sigcontext_offs, builtin_type_void_data_ptr);
144
145 /* Don't cause a memory_error when accessing sigcontext in case the stack
146 layout has changed or the stack is corrupt. */
147 target_read_memory (sigcontext_addr + SIGCONTEXT_PC_OFFSET, buf, ptrbytes);
148 return extract_typed_address (buf, builtin_type_void_func_ptr);
149}
150
f267bd6a 151static CORE_ADDR
5516aa92
JT
152vax_frame_saved_pc (struct frame_info *frame)
153{
5a203e44 154 if ((get_frame_type (frame) == SIGTRAMP_FRAME))
f407986f 155 return (vax_sigtramp_saved_pc (frame)); /* XXXJRT */
5516aa92 156
1e2330ba 157 return (read_memory_integer (get_frame_base (frame) + 16, 4));
5516aa92
JT
158}
159
f267bd6a 160static CORE_ADDR
5516aa92
JT
161vax_frame_args_address (struct frame_info *frame)
162{
25e3a86b
AC
163 /* In most of GDB, getting the args address is too important to just
164 say "I don't know". This is sometimes wrong for functions that
165 aren't on top of the stack, but c'est la vie. */
11c02a10
AC
166 if (get_next_frame (frame))
167 return (read_memory_integer (get_frame_base (get_next_frame (frame)) + 8, 4));
25e3a86b
AC
168 /* Cannot find the AP register value directly from the FP value.
169 Must find it saved in the frame called by this one, or in the AP
170 register for the innermost frame. However, there is no way to
171 tell the difference between the innermost frame and a frame for
172 which we just don't know the frame that it called (e.g. "info
173 frame 0x7ffec789"). For the sake of argument, suppose that the
174 stack is somewhat trashed (which is one reason that "info frame"
175 exists). So, return 0 (indicating we don't know the address of
176 the arglist) if we don't know what frame this frame calls. */
177 return 0;
5516aa92
JT
178}
179
f267bd6a 180static int
5516aa92
JT
181vax_frame_num_args (struct frame_info *fi)
182{
42efa47a 183 return (0xff & read_memory_integer (DEPRECATED_FRAME_ARGS_ADDRESS (fi), 1));
5516aa92 184}
52efde73 185
f267bd6a 186static CORE_ADDR
52efde73
JT
187vax_frame_chain (struct frame_info *frame)
188{
189 /* In the case of the VAX, the frame's nominal address is the FP value,
190 and 12 bytes later comes the saved previous FP value as a 4-byte word. */
627b3ba2 191 if (deprecated_inside_entry_file (get_frame_pc (frame)))
52efde73
JT
192 return (0);
193
1e2330ba 194 return (read_memory_integer (get_frame_base (frame) + 12, 4));
52efde73
JT
195}
196\f
f267bd6a 197static void
52efde73
JT
198vax_push_dummy_frame (void)
199{
200 CORE_ADDR sp = read_register (SP_REGNUM);
201 int regnum;
202
203 sp = push_word (sp, 0); /* arglist */
204 for (regnum = 11; regnum >= 0; regnum--)
205 sp = push_word (sp, read_register (regnum));
206 sp = push_word (sp, read_register (PC_REGNUM));
0ba6dca9 207 sp = push_word (sp, read_register (DEPRECATED_FP_REGNUM));
1d049c5e 208 sp = push_word (sp, read_register (VAX_AP_REGNUM));
52efde73
JT
209 sp = push_word (sp, (read_register (PS_REGNUM) & 0xffef) + 0x2fff0000);
210 sp = push_word (sp, 0);
211 write_register (SP_REGNUM, sp);
0ba6dca9 212 write_register (DEPRECATED_FP_REGNUM, sp);
1d049c5e 213 write_register (VAX_AP_REGNUM, sp + (17 * 4));
52efde73
JT
214}
215
f267bd6a 216static void
52efde73
JT
217vax_pop_frame (void)
218{
0ba6dca9 219 CORE_ADDR fp = read_register (DEPRECATED_FP_REGNUM);
52efde73
JT
220 int regnum;
221 int regmask = read_memory_integer (fp + 4, 4);
222
223 write_register (PS_REGNUM,
224 (regmask & 0xffff)
225 | (read_register (PS_REGNUM) & 0xffff0000));
226 write_register (PC_REGNUM, read_memory_integer (fp + 16, 4));
0ba6dca9 227 write_register (DEPRECATED_FP_REGNUM, read_memory_integer (fp + 12, 4));
1d049c5e 228 write_register (VAX_AP_REGNUM, read_memory_integer (fp + 8, 4));
52efde73
JT
229 fp += 16;
230 for (regnum = 0; regnum < 12; regnum++)
231 if (regmask & (0x10000 << regnum))
232 write_register (regnum, read_memory_integer (fp += 4, 4));
233 fp = fp + 4 + ((regmask >> 30) & 3);
234 if (regmask & 0x20000000)
235 {
236 regnum = read_memory_integer (fp, 4);
237 fp += (regnum + 1) * 4;
238 }
239 write_register (SP_REGNUM, fp);
240 flush_cached_frames ();
241}
a33f7558
JT
242
243/* The VAX call dummy sequence:
244
245 calls #69, @#32323232
246 bpt
247
248 It is 8 bytes long. The address and argc are patched by
249 vax_fix_call_dummy(). */
f267bd6a
JT
250static LONGEST vax_call_dummy_words[] = { 0x329f69fb, 0x03323232 };
251static int sizeof_vax_call_dummy_words = sizeof(vax_call_dummy_words);
a33f7558 252
f267bd6a 253static void
a33f7558
JT
254vax_fix_call_dummy (char *dummy, CORE_ADDR pc, CORE_ADDR fun, int nargs,
255 struct value **args, struct type *type, int gcc_p)
256{
257 dummy[1] = nargs;
258 store_unsigned_integer (dummy + 3, 4, fun);
259}
ab62c900 260\f
f267bd6a 261static void
ea74468c
JT
262vax_store_struct_return (CORE_ADDR addr, CORE_ADDR sp)
263{
264 write_register (1, addr);
265}
266
f267bd6a 267static void
ea74468c
JT
268vax_extract_return_value (struct type *valtype, char *regbuf, char *valbuf)
269{
62700349 270 memcpy (valbuf, regbuf + DEPRECATED_REGISTER_BYTE (0), TYPE_LENGTH (valtype));
ea74468c
JT
271}
272
f267bd6a 273static void
ea74468c
JT
274vax_store_return_value (struct type *valtype, char *valbuf)
275{
73937e03 276 deprecated_write_register_bytes (0, valbuf, TYPE_LENGTH (valtype));
ea74468c
JT
277}
278
f267bd6a 279static CORE_ADDR
ea74468c
JT
280vax_extract_struct_value_address (char *regbuf)
281{
62700349 282 return (extract_unsigned_integer (regbuf + DEPRECATED_REGISTER_BYTE (0),
12c266ea 283 DEPRECATED_REGISTER_RAW_SIZE (0)));
ea74468c
JT
284}
285\f
1d049c5e
JT
286static const unsigned char *
287vax_breakpoint_from_pc (CORE_ADDR *pcptr, int *lenptr)
288{
289 static const unsigned char vax_breakpoint[] = { 3 };
290
291 *lenptr = sizeof(vax_breakpoint);
292 return (vax_breakpoint);
293}
294\f
b83266a0
SS
295/* Advance PC across any function entry prologue instructions
296 to reach some "real" code. */
297
f267bd6a 298static CORE_ADDR
fba45db2 299vax_skip_prologue (CORE_ADDR pc)
b83266a0 300{
52f0bd74 301 int op = (unsigned char) read_memory_integer (pc, 1);
b83266a0 302 if (op == 0x11)
c5aa993b 303 pc += 2; /* skip brb */
b83266a0 304 if (op == 0x31)
c5aa993b 305 pc += 3; /* skip brw */
b83266a0 306 if (op == 0xC2
c5aa993b
JM
307 && ((unsigned char) read_memory_integer (pc + 2, 1)) == 0x5E)
308 pc += 3; /* skip subl2 */
b83266a0 309 if (op == 0x9E
c5aa993b
JM
310 && ((unsigned char) read_memory_integer (pc + 1, 1)) == 0xAE
311 && ((unsigned char) read_memory_integer (pc + 3, 1)) == 0x5E)
312 pc += 4; /* skip movab */
b83266a0 313 if (op == 0x9E
c5aa993b
JM
314 && ((unsigned char) read_memory_integer (pc + 1, 1)) == 0xCE
315 && ((unsigned char) read_memory_integer (pc + 4, 1)) == 0x5E)
316 pc += 5; /* skip movab */
b83266a0 317 if (op == 0x9E
c5aa993b
JM
318 && ((unsigned char) read_memory_integer (pc + 1, 1)) == 0xEE
319 && ((unsigned char) read_memory_integer (pc + 6, 1)) == 0x5E)
320 pc += 7; /* skip movab */
b83266a0
SS
321 return pc;
322}
323
f267bd6a 324static CORE_ADDR
a33f7558
JT
325vax_saved_pc_after_call (struct frame_info *frame)
326{
8bedc050 327 return (DEPRECATED_FRAME_SAVED_PC(frame));
a33f7558
JT
328}
329\f
f267bd6a
JT
330/* Initialize the current architecture based on INFO. If possible, re-use an
331 architecture from ARCHES, which is a list of architectures already created
332 during this debugging session.
333
334 Called e.g. at program startup, when reading a core file, and when reading
335 a binary file. */
336
337static struct gdbarch *
338vax_gdbarch_init (struct gdbarch_info info, struct gdbarch_list *arches)
339{
340 struct gdbarch *gdbarch;
341
4be87837
DJ
342 /* If there is already a candidate, use it. */
343 arches = gdbarch_list_lookup_by_info (arches, &info);
344 if (arches != NULL)
345 return arches->gdbarch;
f267bd6a 346
4be87837 347 gdbarch = gdbarch_alloc (&info, NULL);
4791e091 348
a5afb99f
AC
349 /* NOTE: cagney/2002-12-06: This can be deleted when this arch is
350 ready to unwind the PC first (see frame.c:get_prev_frame()). */
0968aa8c 351 set_gdbarch_deprecated_init_frame_pc (gdbarch, deprecated_init_frame_pc_default);
a5afb99f 352
f267bd6a
JT
353 /* Register info */
354 set_gdbarch_num_regs (gdbarch, VAX_NUM_REGS);
355 set_gdbarch_sp_regnum (gdbarch, VAX_SP_REGNUM);
0ba6dca9 356 set_gdbarch_deprecated_fp_regnum (gdbarch, VAX_FP_REGNUM);
f267bd6a
JT
357 set_gdbarch_pc_regnum (gdbarch, VAX_PC_REGNUM);
358 set_gdbarch_ps_regnum (gdbarch, VAX_PS_REGNUM);
359
360 set_gdbarch_register_name (gdbarch, vax_register_name);
b1e29e33 361 set_gdbarch_deprecated_register_size (gdbarch, VAX_REGISTER_SIZE);
b8b527c5 362 set_gdbarch_deprecated_register_bytes (gdbarch, VAX_REGISTER_BYTES);
9c04cab7
AC
363 set_gdbarch_deprecated_register_byte (gdbarch, vax_register_byte);
364 set_gdbarch_deprecated_register_raw_size (gdbarch, vax_register_raw_size);
a0ed5532 365 set_gdbarch_deprecated_max_register_raw_size (gdbarch, VAX_MAX_REGISTER_RAW_SIZE);
9c04cab7 366 set_gdbarch_deprecated_register_virtual_size (gdbarch, vax_register_virtual_size);
a0ed5532 367 set_gdbarch_deprecated_max_register_virtual_size (gdbarch,
f267bd6a 368 VAX_MAX_REGISTER_VIRTUAL_SIZE);
9c04cab7 369 set_gdbarch_deprecated_register_virtual_type (gdbarch, vax_register_virtual_type);
f267bd6a
JT
370
371 /* Frame and stack info */
372 set_gdbarch_skip_prologue (gdbarch, vax_skip_prologue);
6913c89a 373 set_gdbarch_deprecated_saved_pc_after_call (gdbarch, vax_saved_pc_after_call);
f267bd6a
JT
374
375 set_gdbarch_frame_num_args (gdbarch, vax_frame_num_args);
376 set_gdbarch_frameless_function_invocation (gdbarch,
377 generic_frameless_function_invocation_not);
378
618ce49f 379 set_gdbarch_deprecated_frame_chain (gdbarch, vax_frame_chain);
8bedc050 380 set_gdbarch_deprecated_frame_saved_pc (gdbarch, vax_frame_saved_pc);
f267bd6a 381
42efa47a 382 set_gdbarch_deprecated_frame_args_address (gdbarch, vax_frame_args_address);
f267bd6a 383
f30ee0bc 384 set_gdbarch_deprecated_frame_init_saved_regs (gdbarch, vax_frame_init_saved_regs);
f267bd6a
JT
385
386 set_gdbarch_frame_args_skip (gdbarch, 4);
387
f267bd6a
JT
388 set_gdbarch_inner_than (gdbarch, core_addr_lessthan);
389
390 /* Return value info */
4183d812 391 set_gdbarch_deprecated_store_struct_return (gdbarch, vax_store_struct_return);
26e9b323 392 set_gdbarch_deprecated_extract_return_value (gdbarch, vax_extract_return_value);
ebba8386 393 set_gdbarch_deprecated_store_return_value (gdbarch, vax_store_return_value);
26e9b323 394 set_gdbarch_deprecated_extract_struct_value_address (gdbarch, vax_extract_struct_value_address);
f267bd6a
JT
395
396 /* Call dummy info */
f3824013 397 set_gdbarch_deprecated_push_dummy_frame (gdbarch, vax_push_dummy_frame);
749b82f6 398 set_gdbarch_deprecated_pop_frame (gdbarch, vax_pop_frame);
f267bd6a 399 set_gdbarch_call_dummy_location (gdbarch, ON_STACK);
b1e29e33
AC
400 set_gdbarch_deprecated_call_dummy_words (gdbarch, vax_call_dummy_words);
401 set_gdbarch_deprecated_sizeof_call_dummy_words (gdbarch, sizeof_vax_call_dummy_words);
402 set_gdbarch_deprecated_fix_call_dummy (gdbarch, vax_fix_call_dummy);
403 set_gdbarch_deprecated_call_dummy_breakpoint_offset (gdbarch, 7);
07555a72 404 set_gdbarch_deprecated_use_generic_dummy_frames (gdbarch, 0);
ae45cd16 405 set_gdbarch_deprecated_pc_in_call_dummy (gdbarch, deprecated_pc_in_call_dummy_on_stack);
f267bd6a
JT
406
407 /* Breakpoint info */
1d049c5e 408 set_gdbarch_breakpoint_from_pc (gdbarch, vax_breakpoint_from_pc);
f267bd6a
JT
409
410 /* Misc info */
411 set_gdbarch_function_start_offset (gdbarch, 2);
1d049c5e 412 set_gdbarch_believe_pcc_promotion (gdbarch, 1);
f267bd6a 413
6c0e89ed 414 /* Should be using push_dummy_call. */
b46e02f6 415 set_gdbarch_deprecated_dummy_write_sp (gdbarch, deprecated_write_sp);
6c0e89ed 416
4791e091 417 /* Hook in ABI-specific overrides, if they have been registered. */
4be87837 418 gdbarch_init_osabi (info, gdbarch);
4791e091 419
ee2842e2
ILT
420 set_gdbarch_print_insn (gdbarch, print_insn_vax);
421
f267bd6a
JT
422 return (gdbarch);
423}
c906108c 424
a78f21af
AC
425extern initialize_file_ftype _initialize_vax_tdep; /* -Wmissing-prototypes */
426
c906108c 427void
fba45db2 428_initialize_vax_tdep (void)
c906108c 429{
4be87837 430 gdbarch_register (bfd_arch_vax, vax_gdbarch_init, NULL);
c906108c 431}