Welcome cvs to the big time.
[deliverable/binutils-gdb.git] / gdb / am29k-pinsn.c
CommitLineData
dd3b648e
RP
1/* Instruction printing code for the AMD 29000
2 Copyright (C) 1990 Free Software Foundation, Inc.
3 Contributed by Cygnus Support. Written by Jim Kingdon.
4
5This file is part of GDB.
6
7This program is free software; you can redistribute it and/or modify
8it under the terms of the GNU General Public License as published by
99a7de40
JG
9the Free Software Foundation; either version 2 of the License, or
10(at your option) any later version.
dd3b648e
RP
11
12This program is distributed in the hope that it will be useful,
13but WITHOUT ANY WARRANTY; without even the implied warranty of
14MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15GNU General Public License for more details.
16
17You should have received a copy of the GNU General Public License
99a7de40
JG
18along with this program; if not, write to the Free Software
19Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
dd3b648e
RP
20
21#include <stdio.h>
22
23#include "defs.h"
24#include "target.h"
25#include "am29k-opcode.h"
26
27/* Print a symbolic representation of a general-purpose
28 register number NUM on STREAM.
29 NUM is a number as found in the instruction, not as found in
30 debugging symbols; it must be in the range 0-255. */
31static void
32print_general (num, stream)
33 int num;
34 FILE *stream;
35{
36 if (num < 128)
37 fprintf_filtered (stream, "gr%d", num);
38 else
39 fprintf_filtered (stream, "lr%d", num - 128);
40}
41
42/* Like print_general but a special-purpose register.
43
44 The mnemonics used by the AMD assembler are not quite the same
45 as the ones in the User's Manual. We use the ones that the
46 assembler uses. */
47static void
48print_special (num, stream)
49 int num;
50 FILE *stream;
51{
52 /* Register names of registers 0-SPEC0_NUM-1. */
53 static char *spec0_names[] = {
54 "vab", "ops", "cps", "cfg", "cha", "chd", "chc", "rbp", "tmc", "tmr",
55 "pc0", "pc1", "pc2", "mmu", "lru"
56 };
57#define SPEC0_NUM ((sizeof spec0_names) / (sizeof spec0_names[0]))
58
59 /* Register names of registers 128-128+SPEC128_NUM-1. */
60 static char *spec128_names[] = {
61 "ipc", "ipa", "ipb", "q", "alu", "bp", "fc", "cr"
62 };
63#define SPEC128_NUM ((sizeof spec128_names) / (sizeof spec128_names[0]))
64
65 /* Register names of registers 160-160+SPEC160_NUM-1. */
66 static char *spec160_names[] = {
67 "fpe", "inte", "fps", "sr163", "exop"
68 };
69#define SPEC160_NUM ((sizeof spec160_names) / (sizeof spec160_names[0]))
70
71 if (num < SPEC0_NUM)
72 fprintf_filtered (stream, spec0_names[num]);
73 else if (num >= 128 && num < 128 + SPEC128_NUM)
74 fprintf_filtered (stream, spec128_names[num-128]);
75 else if (num >= 160 && num < 160 + SPEC160_NUM)
76 fprintf_filtered (stream, spec160_names[num-160]);
77 else
78 fprintf_filtered (stream, "sr%d", num);
79}
80
81/* Is an instruction with OPCODE a delayed branch? */
82static int
83is_delayed_branch (opcode)
84 int opcode;
85{
86 return (opcode == 0xa8 || opcode == 0xa9 || opcode == 0xa0 || opcode == 0xa1
87 || opcode == 0xa4 || opcode == 0xa5
88 || opcode == 0xb4 || opcode == 0xb5
89 || opcode == 0xc4 || opcode == 0xc0
90 || opcode == 0xac || opcode == 0xad
91 || opcode == 0xcc);
92}
93
94/* Now find the four bytes of INSN and put them in *INSN{0,8,16,24}.
95 Note that the amd can be set up as either
96 big or little-endian (the tm file says which) and we can't assume
97 the host machine is the same. */
98static void
99find_bytes (insn, insn0, insn8, insn16, insn24)
100 char *insn;
101 unsigned char *insn0;
102 unsigned char *insn8;
103 unsigned char *insn16;
104 unsigned char *insn24;
105{
106#if TARGET_BYTE_ORDER == BIG_ENDIAN
107 *insn24 = insn[0];
108 *insn16 = insn[1];
109 *insn8 = insn[2];
110 *insn0 = insn[3];
111#else /* Little-endian. */
112 *insn24 = insn[3];
113 *insn16 = insn[2];
114 *insn8 = insn[1];
115 *insn0 = insn[0];
116#endif /* Little-endian. */
117}
118
119/* Print one instruction from MEMADDR on STREAM.
120 Return the size of the instruction (always 4 on am29k). */
121int
122print_insn (memaddr, stream)
123 CORE_ADDR memaddr;
124 FILE *stream;
125{
126 /* The raw instruction. */
127 char insn[4];
128
129 /* The four bytes of the instruction. */
130 unsigned char insn24, insn16, insn8, insn0;
131
132 struct am29k_opcode *opcode;
133
134 read_memory (memaddr, &insn[0], 4);
135
136 find_bytes (insn, &insn0, &insn8, &insn16, &insn24);
137
138 /* The opcode is always in insn24. */
139 for (opcode = &am29k_opcodes[0];
140 opcode < &am29k_opcodes[NUM_OPCODES];
141 ++opcode)
142 {
143 if (insn24 == opcode->opcode)
144 {
145 char *s;
146
147 fprintf_filtered (stream, "%s ", opcode->name);
148 for (s = opcode->args; *s != '\0'; ++s)
149 {
150 switch (*s)
151 {
152 case 'a':
153 print_general (insn8, stream);
154 break;
155
156 case 'b':
157 print_general (insn0, stream);
158 break;
159
160 case 'c':
161 print_general (insn16, stream);
162 break;
163
164 case 'i':
165 fprintf_filtered (stream, "%d", insn0);
166 break;
167
168 case 'x':
169 fprintf_filtered (stream, "%d", (insn16 << 8) + insn0);
170 break;
171
172 case 'h':
173 fprintf_filtered (stream, "0x%x",
174 (insn16 << 24) + (insn0 << 16));
175 break;
176
177 case 'X':
178 fprintf_filtered (stream, "%d",
179 ((insn16 << 8) + insn0) | 0xffff0000);
180 break;
181
182 case 'P':
183 /* This output looks just like absolute addressing, but
184 maybe that's OK (it's what the GDB 68k and EBMON
185 29k disassemblers do). */
186 /* All the shifting is to sign-extend it. p*/
187 print_address
188 (memaddr +
189 (((int)((insn16 << 10) + (insn0 << 2)) << 14) >> 14),
190 stream);
191 break;
192
193 case 'A':
194 print_address ((insn16 << 10) + (insn0 << 2), stream);
195 break;
196
197 case 'e':
198 fprintf_filtered (stream, "%d", insn16 >> 7);
199 break;
200
201 case 'n':
202 fprintf_filtered (stream, "0x%x", insn16 & 0x7f);
203 break;
204
205 case 'v':
206 fprintf_filtered (stream, "%#x", insn16);
207 break;
208
209 case 's':
210 print_special (insn8, stream);
211 break;
212
213 case 'u':
214 fprintf_filtered (stream, "%d", insn0 >> 7);
215 break;
216
217 case 'r':
218 fprintf_filtered (stream, "%d", (insn0 >> 4) & 7);
219 break;
220
221 case 'd':
222 fprintf_filtered (stream, "%d", (insn0 >> 2) & 3);
223 break;
224
225 case 'f':
226 fprintf_filtered (stream, "%d", insn0 & 3);
227 break;
228
229 case 'F':
230 fprintf_filtered (stream, "%d", (insn0 >> 18) & 15);
231 break;
232
233 case 'C':
234 fprintf_filtered (stream, "%d", (insn0 >> 16) & 3);
235 break;
236
237 default:
238 fprintf_filtered (stream, "%c", *s);
239 }
240 }
241
242 /* Now we look for a const,consth pair of instructions,
243 in which case we try to print the symbolic address. */
244 if (insn24 == 2) /* consth */
245 {
246 int errcode;
247 char prev_insn[4];
248 unsigned char prev_insn0, prev_insn8, prev_insn16, prev_insn24;
249
250 errcode = target_read_memory (memaddr - 4,
251 &prev_insn[0],
252 4);
253 if (errcode == 0)
254 {
255 /* If it is a delayed branch, we need to look at the
256 instruction before the delayed brach to handle
257 things like
258
259 const _foo
260 call _printf
261 consth _foo
262 */
263 find_bytes (prev_insn, &prev_insn0, &prev_insn8,
264 &prev_insn16, &prev_insn24);
265 if (is_delayed_branch (prev_insn24))
266 {
267 errcode = target_read_memory
268 (memaddr - 8, &prev_insn[0], 4);
269 find_bytes (prev_insn, &prev_insn0, &prev_insn8,
270 &prev_insn16, &prev_insn24);
271 }
272 }
273
274 /* If there was a problem reading memory, then assume
275 the previous instruction was not const. */
276 if (errcode == 0)
277 {
278 /* Is it const to the same register? */
279 if (prev_insn24 == 3
280 && prev_insn8 == insn8)
281 {
282 fprintf_filtered (stream, "\t; ");
283 print_address (((insn16 << 24) + (insn0 << 16)
284 + (prev_insn16 << 8) + (prev_insn0)),
285 stream);
286 }
287 }
288 }
289
290 return 4;
291 }
292 }
293 fprintf_filtered (stream, ".word %#8x",
294 (insn24 << 24) + (insn16 << 16) + (insn8 << 8) + insn0);
295 return 4;
296}
This page took 0.068828 seconds and 4 git commands to generate.