]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blame - gdb/ns32k-tdep.c
import gdb-1999-07-07 post reformat
[thirdparty/binutils-gdb.git] / gdb / ns32k-tdep.c
CommitLineData
c906108c
SS
1/* Print NS 32000 instructions for GDB, the GNU debugger.
2 Copyright 1986, 1988, 1991, 1992, 1994, 1995
3 Free Software Foundation, Inc.
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
24void
25_initialize_ns32k_tdep ()
26{
27 tm_print_insn = print_insn_ns32k;
28}
b83266a0
SS
29
30/* Advance PC across any function entry prologue instructions
31 to reach some "real" code. */
32
33CORE_ADDR
34merlin_skip_prologue (pc)
35 CORE_ADDR pc;
36{
37 register int op = read_memory_integer (pc, 1);
38 if (op == 0x82)
39 {
c5aa993b 40 op = read_memory_integer (pc + 2, 1);
b83266a0
SS
41 if ((op & 0x80) == 0)
42 pc += 3;
43 else if ((op & 0xc0) == 0x80)
44 pc += 4;
c5aa993b
JM
45 else
46 pc += 6;
b83266a0
SS
47 }
48 return pc;
49}
50
51CORE_ADDR
52umax_skip_prologue (pc)
53 CORE_ADDR pc;
54{
55 register unsigned char op = read_memory_integer (pc, 1);
56 if (op == 0x82)
57 {
c5aa993b 58 op = read_memory_integer (pc + 2, 1);
b83266a0
SS
59 if ((op & 0x80) == 0)
60 pc += 3;
61 else if ((op & 0xc0) == 0x80)
62 pc += 4;
63 else
64 pc += 6;
c5aa993b 65 }
b83266a0
SS
66 return pc;
67}
68
392a587b
JM
69/* Return number of args passed to a frame.
70 Can return -1, meaning no way to tell. */
71
72int
73merlin_frame_num_args (fi)
74 struct frame_info *fi;
75{
76 int numargs;
77 CORE_ADDR pc;
78 int insn;
79 int addr_mode;
80 int width;
81
82 pc = FRAME_SAVED_PC (fi);
c5aa993b 83 insn = read_memory_integer (pc, 2);
392a587b
JM
84 addr_mode = (insn >> 11) & 0x1f;
85 insn = insn & 0x7ff;
86 if ((insn & 0x7fc) == 0x57c
c5aa993b 87 && addr_mode == 0x14) /* immediate */
392a587b 88 {
c5aa993b 89 if (insn == 0x57c) /* adjspb */
392a587b 90 width = 1;
c5aa993b 91 else if (insn == 0x57d) /* adjspw */
392a587b 92 width = 2;
c5aa993b 93 else if (insn == 0x57f) /* adjspd */
392a587b 94 width = 4;
c5aa993b 95 numargs = read_memory_integer (pc + 2, width);
392a587b
JM
96 if (width > 1)
97 flip_bytes (&numargs, width);
c5aa993b 98 numargs = -sign_extend (numargs, width * 8) / 4;
392a587b
JM
99 }
100 else
101 numargs = -1;
102 return numargs;
103}
104
cce74817
JM
105
106/* Return number of args passed to a frame.
107 Can return -1, meaning no way to tell.
108 Encore's C compiler often reuses same area on stack for args,
109 so this will often not work properly. If the arg names
110 are known, it's likely most of them will be printed. */
392a587b
JM
111int
112umax_frame_num_args (fi)
113 struct frame_info *fi;
114{
115 int numargs;
116 CORE_ADDR pc;
117 CORE_ADDR enter_addr;
118 unsigned int insn;
119 unsigned int addr_mode;
120 int width;
121
122 numargs = -1;
123 enter_addr = ns32k_get_enter_addr ((fi)->pc);
124 if (enter_addr > 0)
125 {
126 pc = ((enter_addr == 1)
127 ? SAVED_PC_AFTER_CALL (fi)
128 : FRAME_SAVED_PC (fi));
c5aa993b 129 insn = read_memory_integer (pc, 2);
392a587b
JM
130 addr_mode = (insn >> 11) & 0x1f;
131 insn = insn & 0x7ff;
132 if ((insn & 0x7fc) == 0x57c
c5aa993b 133 && addr_mode == 0x14) /* immediate */
392a587b 134 {
c5aa993b 135 if (insn == 0x57c) /* adjspb */
392a587b 136 width = 1;
c5aa993b 137 else if (insn == 0x57d) /* adjspw */
392a587b 138 width = 2;
c5aa993b 139 else if (insn == 0x57f) /* adjspd */
392a587b 140 width = 4;
c5aa993b 141 numargs = read_memory_integer (pc + 2, width);
392a587b
JM
142 if (width > 1)
143 flip_bytes (&numargs, width);
c5aa993b 144 numargs = -sign_extend (numargs, width * 8) / 4;
392a587b
JM
145 }
146 }
147 return numargs;
148}
b83266a0 149
c906108c
SS
150
151sign_extend (value, bits)
152{
153 value = value & ((1 << bits) - 1);
c5aa993b 154 return (value & (1 << (bits - 1))
c906108c
SS
155 ? value | (~((1 << bits) - 1))
156 : value);
157}
158
159void
160flip_bytes (ptr, count)
161 char *ptr;
162 int count;
163{
164 char tmp;
165
166 while (count > 0)
167 {
168 tmp = *ptr;
c5aa993b
JM
169 ptr[0] = ptr[count - 1];
170 ptr[count - 1] = tmp;
c906108c
SS
171 ptr++;
172 count -= 2;
173 }
174}
175
176/* Return the number of locals in the current frame given a pc
177 pointing to the enter instruction. This is used in the macro
178 FRAME_FIND_SAVED_REGS. */
179
180int
181ns32k_localcount (enter_pc)
182 CORE_ADDR enter_pc;
183{
184 unsigned char localtype;
185 int localcount;
186
c5aa993b 187 localtype = read_memory_integer (enter_pc + 2, 1);
c906108c
SS
188 if ((localtype & 0x80) == 0)
189 localcount = localtype;
190 else if ((localtype & 0xc0) == 0x80)
191 localcount = (((localtype & 0x3f) << 8)
c5aa993b 192 | (read_memory_integer (enter_pc + 3, 1) & 0xff));
c906108c
SS
193 else
194 localcount = (((localtype & 0x3f) << 24)
c5aa993b
JM
195 | ((read_memory_integer (enter_pc + 3, 1) & 0xff) << 16)
196 | ((read_memory_integer (enter_pc + 4, 1) & 0xff) << 8)
197 | (read_memory_integer (enter_pc + 5, 1) & 0xff));
c906108c
SS
198 return localcount;
199}
200
201
202/* Nonzero if instruction at PC is a return instruction. */
203
204static int
205ns32k_about_to_return (pc)
206 CORE_ADDR pc;
207{
208 return (read_memory_integer (pc, 1) == 0x12);
209}
210
211
212/*
213 * Get the address of the enter opcode for the function
214 * containing PC, if there is an enter for the function,
215 * and if the pc is between the enter and exit.
216 * Returns positive address if pc is between enter/exit,
217 * 1 if pc before enter or after exit, 0 otherwise.
218 */
219
220CORE_ADDR
221ns32k_get_enter_addr (pc)
222 CORE_ADDR pc;
223{
224 CORE_ADDR enter_addr;
225 unsigned char op;
226
227 if (pc == 0)
228 return 0;
229
230 if (ns32k_about_to_return (pc))
c5aa993b 231 return 1; /* after exit */
c906108c
SS
232
233 enter_addr = get_pc_function_start (pc);
234
c5aa993b
JM
235 if (pc == enter_addr)
236 return 1; /* before enter */
c906108c
SS
237
238 op = read_memory_integer (enter_addr, 1);
239
240 if (op != 0x82)
c5aa993b 241 return 0; /* function has no enter/exit */
c906108c 242
c5aa993b 243 return enter_addr; /* pc is between enter and exit */
c906108c 244}