1 /* CPU data for lm32.
2
3 THIS FILE IS MACHINE GENERATED WITH CGEN.
4
5 Copyright (C) 1996-2014 Free Software Foundation, Inc.
6
7 This file is part of the GNU Binutils and/or GDB, the GNU debugger.
8
9 This file is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 3, or (at your option)
12 any later version.
13
14 It is distributed in the hope that it will be useful, but WITHOUT
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
16 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
17 License for more details.
18
19 You should have received a copy of the GNU General Public License along
20 with this program; if not, write to the Free Software Foundation, Inc.,
21 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.
22
23 */
24
25 #include "sysdep.h"
26 #include <stdio.h>
27 #include <stdarg.h>
28 #include "ansidecl.h"
29 #include "bfd.h"
30 #include "symcat.h"
31 #include "lm32-desc.h"
32 #include "lm32-opc.h"
33 #include "opintl.h"
34 #include "libiberty.h"
35 #include "xregex.h"
36
37 /* Attributes. */
38
39 static const CGEN_ATTR_ENTRY bool_attr[] =
40 {
41 { "#f", 0 },
42 { "#t", 1 },
43 { 0, 0 }
44 };
45
46 static const CGEN_ATTR_ENTRY MACH_attr[] ATTRIBUTE_UNUSED =
47 {
48 { "base", MACH_BASE },
49 { "lm32", MACH_LM32 },
50 { "max", MACH_MAX },
51 { 0, 0 }
52 };
53
54 static const CGEN_ATTR_ENTRY ISA_attr[] ATTRIBUTE_UNUSED =
55 {
56 { "lm32", ISA_LM32 },
57 { "max", ISA_MAX },
58 { 0, 0 }
59 };
60
61 const CGEN_ATTR_TABLE lm32_cgen_ifield_attr_table[] =
62 {
63 { "MACH", & MACH_attr[0], & MACH_attr[0] },
64 { "VIRTUAL", &bool_attr[0], &bool_attr[0] },
65 { "PCREL-ADDR", &bool_attr[0], &bool_attr[0] },
66 { "ABS-ADDR", &bool_attr[0], &bool_attr[0] },
67 { "RESERVED", &bool_attr[0], &bool_attr[0] },
68 { "SIGN-OPT", &bool_attr[0], &bool_attr[0] },
69 { "SIGNED", &bool_attr[0], &bool_attr[0] },
70 { 0, 0, 0 }
71 };
72
73 const CGEN_ATTR_TABLE lm32_cgen_hardware_attr_table[] =
74 {
75 { "MACH", & MACH_attr[0], & MACH_attr[0] },
76 { "VIRTUAL", &bool_attr[0], &bool_attr[0] },
77 { "CACHE-ADDR", &bool_attr[0], &bool_attr[0] },
78 { "PC", &bool_attr[0], &bool_attr[0] },
79 { "PROFILE", &bool_attr[0], &bool_attr[0] },
80 { 0, 0, 0 }
81 };
82
83 const CGEN_ATTR_TABLE lm32_cgen_operand_attr_table[] =
84 {
85 { "MACH", & MACH_attr[0], & MACH_attr[0] },
86 { "VIRTUAL", &bool_attr[0], &bool_attr[0] },
87 { "PCREL-ADDR", &bool_attr[0], &bool_attr[0] },
88 { "ABS-ADDR", &bool_attr[0], &bool_attr[0] },
89 { "SIGN-OPT", &bool_attr[0], &bool_attr[0] },
90 { "SIGNED", &bool_attr[0], &bool_attr[0] },
91 { "NEGATIVE", &bool_attr[0], &bool_attr[0] },
92 { "RELAX", &bool_attr[0], &bool_attr[0] },
93 { "SEM-ONLY", &bool_attr[0], &bool_attr[0] },
94 { 0, 0, 0 }
95 };
96
97 const CGEN_ATTR_TABLE lm32_cgen_insn_attr_table[] =
98 {
99 { "MACH", & MACH_attr[0], & MACH_attr[0] },
100 { "ALIAS", &bool_attr[0], &bool_attr[0] },
101 { "VIRTUAL", &bool_attr[0], &bool_attr[0] },
102 { "UNCOND-CTI", &bool_attr[0], &bool_attr[0] },
103 { "COND-CTI", &bool_attr[0], &bool_attr[0] },
104 { "SKIP-CTI", &bool_attr[0], &bool_attr[0] },
105 { "DELAY-SLOT", &bool_attr[0], &bool_attr[0] },
106 { "RELAXABLE", &bool_attr[0], &bool_attr[0] },
107 { "RELAXED", &bool_attr[0], &bool_attr[0] },
108 { "NO-DIS", &bool_attr[0], &bool_attr[0] },
109 { "PBB", &bool_attr[0], &bool_attr[0] },
110 { 0, 0, 0 }
111 };
112
113 /* Instruction set variants. */
114
115 static const CGEN_ISA lm32_cgen_isa_table[] = {
116 { "lm32", 32, 32, 32, 32 },
117 { 0, 0, 0, 0, 0 }
118 };
119
120 /* Machine variants. */
121
122 static const CGEN_MACH lm32_cgen_mach_table[] = {
123 { "lm32", "lm32", MACH_LM32, 0 },
124 { 0, 0, 0, 0 }
125 };
126
127 static CGEN_KEYWORD_ENTRY lm32_cgen_opval_h_gr_entries[] =
128 {
129 { "gp", 26, {0, {{{0, 0}}}}, 0, 0 },
130 { "fp", 27, {0, {{{0, 0}}}}, 0, 0 },
131 { "sp", 28, {0, {{{0, 0}}}}, 0, 0 },
132 { "ra", 29, {0, {{{0, 0}}}}, 0, 0 },
133 { "ea", 30, {0, {{{0, 0}}}}, 0, 0 },
134 { "ba", 31, {0, {{{0, 0}}}}, 0, 0 },
135 { "r0", 0, {0, {{{0, 0}}}}, 0, 0 },
136 { "r1", 1, {0, {{{0, 0}}}}, 0, 0 },
137 { "r2", 2, {0, {{{0, 0}}}}, 0, 0 },
138 { "r3", 3, {0, {{{0, 0}}}}, 0, 0 },
139 { "r4", 4, {0, {{{0, 0}}}}, 0, 0 },
140 { "r5", 5, {0, {{{0, 0}}}}, 0, 0 },
141 { "r6", 6, {0, {{{0, 0}}}}, 0, 0 },
142 { "r7", 7, {0, {{{0, 0}}}}, 0, 0 },
143 { "r8", 8, {0, {{{0, 0}}}}, 0, 0 },
144 { "r9", 9, {0, {{{0, 0}}}}, 0, 0 },
145 { "r10", 10, {0, {{{0, 0}}}}, 0, 0 },
146 { "r11", 11, {0, {{{0, 0}}}}, 0, 0 },
147 { "r12", 12, {0, {{{0, 0}}}}, 0, 0 },
148 { "r13", 13, {0, {{{0, 0}}}}, 0, 0 },
149 { "r14", 14, {0, {{{0, 0}}}}, 0, 0 },
150 { "r15", 15, {0, {{{0, 0}}}}, 0, 0 },
151 { "r16", 16, {0, {{{0, 0}}}}, 0, 0 },
152 { "r17", 17, {0, {{{0, 0}}}}, 0, 0 },
153 { "r18", 18, {0, {{{0, 0}}}}, 0, 0 },
154 { "r19", 19, {0, {{{0, 0}}}}, 0, 0 },
155 { "r20", 20, {0, {{{0, 0}}}}, 0, 0 },
156 { "r21", 21, {0, {{{0, 0}}}}, 0, 0 },
157 { "r22", 22, {0, {{{0, 0}}}}, 0, 0 },
158 { "r23", 23, {0, {{{0, 0}}}}, 0, 0 },
159 { "r24", 24, {0, {{{0, 0}}}}, 0, 0 },
160 { "r25", 25, {0, {{{0, 0}}}}, 0, 0 },
161 { "r26", 26, {0, {{{0, 0}}}}, 0, 0 },
162 { "r27", 27, {0, {{{0, 0}}}}, 0, 0 },
163 { "r28", 28, {0, {{{0, 0}}}}, 0, 0 },
164 { "r29", 29, {0, {{{0, 0}}}}, 0, 0 },
165 { "r30", 30, {0, {{{0, 0}}}}, 0, 0 },
166 { "r31", 31, {0, {{{0, 0}}}}, 0, 0 }
167 };
168
169 CGEN_KEYWORD lm32_cgen_opval_h_gr =
170 {
171 & lm32_cgen_opval_h_gr_entries[0],
172 38,
173 0, 0, 0, 0, ""
174 };
175
176 static CGEN_KEYWORD_ENTRY lm32_cgen_opval_h_csr_entries[] =
177 {
178 { "IE", 0, {0, {{{0, 0}}}}, 0, 0 },
179 { "IM", 1, {0, {{{0, 0}}}}, 0, 0 },
180 { "IP", 2, {0, {{{0, 0}}}}, 0, 0 },
181 { "ICC", 3, {0, {{{0, 0}}}}, 0, 0 },
182 { "DCC", 4, {0, {{{0, 0}}}}, 0, 0 },
183 { "CC", 5, {0, {{{0, 0}}}}, 0, 0 },
184 { "CFG", 6, {0, {{{0, 0}}}}, 0, 0 },
185 { "EBA", 7, {0, {{{0, 0}}}}, 0, 0 },
186 { "DC", 8, {0, {{{0, 0}}}}, 0, 0 },
187 { "DEBA", 9, {0, {{{0, 0}}}}, 0, 0 },
188 { "CFG2", 10, {0, {{{0, 0}}}}, 0, 0 },
189 { "JTX", 14, {0, {{{0, 0}}}}, 0, 0 },
190 { "JRX", 15, {0, {{{0, 0}}}}, 0, 0 },
191 { "BP0", 16, {0, {{{0, 0}}}}, 0, 0 },
192 { "BP1", 17, {0, {{{0, 0}}}}, 0, 0 },
193 { "BP2", 18, {0, {{{0, 0}}}}, 0, 0 },
194 { "BP3", 19, {0, {{{0, 0}}}}, 0, 0 },
195 { "WP0", 24, {0, {{{0, 0}}}}, 0, 0 },
196 { "WP1", 25, {0, {{{0, 0}}}}, 0, 0 },
197 { "WP2", 26, {0, {{{0, 0}}}}, 0, 0 },
198 { "WP3", 27, {0, {{{0, 0}}}}, 0, 0 },
199 { "PSW", 29, {0, {{{0, 0}}}}, 0, 0 },
200 { "TLBVADDR", 30, {0, {{{0, 0}}}}, 0, 0 },
201 { "TLBPADDR", 31, {0, {{{0, 0}}}}, 0, 0 },
202 { "TLBBADVADDR", 31, {0, {{{0, 0}}}}, 0, 0 }
203 };
204
205 CGEN_KEYWORD lm32_cgen_opval_h_csr =
206 {
207 & lm32_cgen_opval_h_csr_entries[0],
208 25,
209 0, 0, 0, 0, ""
210 };
211
212
213 /* The hardware table. */
214
215 #define A(a) (1 << CGEN_HW_##a)
216
217 const CGEN_HW_ENTRY lm32_cgen_hw_table[] =
218 {
219 { "h-memory", HW_H_MEMORY, CGEN_ASM_NONE, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
220 { "h-sint", HW_H_SINT, CGEN_ASM_NONE, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
221 { "h-uint", HW_H_UINT, CGEN_ASM_NONE, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
222 { "h-addr", HW_H_ADDR, CGEN_ASM_NONE, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
223 { "h-iaddr", HW_H_IADDR, CGEN_ASM_NONE, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
224 { "h-pc", HW_H_PC, CGEN_ASM_NONE, 0, { 0|A(PC), { { { (1<<MACH_BASE), 0 } } } } },
225 { "h-gr", HW_H_GR, CGEN_ASM_KEYWORD, (PTR) & lm32_cgen_opval_h_gr, { 0, { { { (1<<MACH_BASE), 0 } } } } },
226 { "h-csr", HW_H_CSR, CGEN_ASM_KEYWORD, (PTR) & lm32_cgen_opval_h_csr, { 0, { { { (1<<MACH_BASE), 0 } } } } },
227 { 0, 0, CGEN_ASM_NONE, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } }
228 };
229
230 #undef A
231
232
233 /* The instruction field table. */
234
235 #define A(a) (1 << CGEN_IFLD_##a)
236
237 const CGEN_IFLD lm32_cgen_ifld_table[] =
238 {
239 { LM32_F_NIL, "f-nil", 0, 0, 0, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
240 { LM32_F_ANYOF, "f-anyof", 0, 0, 0, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
241 { LM32_F_OPCODE, "f-opcode", 0, 32, 31, 6, { 0, { { { (1<<MACH_BASE), 0 } } } } },
242 { LM32_F_R0, "f-r0", 0, 32, 25, 5, { 0, { { { (1<<MACH_BASE), 0 } } } } },
243 { LM32_F_R1, "f-r1", 0, 32, 20, 5, { 0, { { { (1<<MACH_BASE), 0 } } } } },
244 { LM32_F_R2, "f-r2", 0, 32, 15, 5, { 0, { { { (1<<MACH_BASE), 0 } } } } },
245 { LM32_F_RESV0, "f-resv0", 0, 32, 10, 11, { 0|A(RESERVED), { { { (1<<MACH_BASE), 0 } } } } },
246 { LM32_F_SHIFT, "f-shift", 0, 32, 4, 5, { 0, { { { (1<<MACH_BASE), 0 } } } } },
247 { LM32_F_IMM, "f-imm", 0, 32, 15, 16, { 0, { { { (1<<MACH_BASE), 0 } } } } },
248 { LM32_F_UIMM, "f-uimm", 0, 32, 15, 16, { 0, { { { (1<<MACH_BASE), 0 } } } } },
249 { LM32_F_CSR, "f-csr", 0, 32, 25, 5, { 0, { { { (1<<MACH_BASE), 0 } } } } },
250 { LM32_F_USER, "f-user", 0, 32, 10, 11, { 0, { { { (1<<MACH_BASE), 0 } } } } },
251 { LM32_F_EXCEPTION, "f-exception", 0, 32, 25, 26, { 0, { { { (1<<MACH_BASE), 0 } } } } },
252 { LM32_F_BRANCH, "f-branch", 0, 32, 15, 16, { 0|A(PCREL_ADDR), { { { (1<<MACH_BASE), 0 } } } } },
253 { LM32_F_CALL, "f-call", 0, 32, 25, 26, { 0|A(PCREL_ADDR), { { { (1<<MACH_BASE), 0 } } } } },
254 { 0, 0, 0, 0, 0, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } }
255 };
256
257 #undef A
258
259
260
261 /* multi ifield declarations */
262
263
264
265 /* multi ifield definitions */
266
267
268 /* The operand table. */
269
270 #define A(a) (1 << CGEN_OPERAND_##a)
271 #define OPERAND(op) LM32_OPERAND_##op
272
273 const CGEN_OPERAND lm32_cgen_operand_table[] =
274 {
275 /* pc: program counter */
276 { "pc", LM32_OPERAND_PC, HW_H_PC, 0, 0,
277 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_NIL] } },
278 { 0|A(SEM_ONLY), { { { (1<<MACH_BASE), 0 } } } } },
279 /* r0: register 0 */
280 { "r0", LM32_OPERAND_R0, HW_H_GR, 25, 5,
281 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_R0] } },
282 { 0, { { { (1<<MACH_BASE), 0 } } } } },
283 /* r1: register 1 */
284 { "r1", LM32_OPERAND_R1, HW_H_GR, 20, 5,
285 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_R1] } },
286 { 0, { { { (1<<MACH_BASE), 0 } } } } },
287 /* r2: register 2 */
288 { "r2", LM32_OPERAND_R2, HW_H_GR, 15, 5,
289 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_R2] } },
290 { 0, { { { (1<<MACH_BASE), 0 } } } } },
291 /* shift: shift amout */
292 { "shift", LM32_OPERAND_SHIFT, HW_H_UINT, 4, 5,
293 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_SHIFT] } },
294 { 0, { { { (1<<MACH_BASE), 0 } } } } },
295 /* imm: signed immediate */
296 { "imm", LM32_OPERAND_IMM, HW_H_SINT, 15, 16,
297 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_IMM] } },
298 { 0, { { { (1<<MACH_BASE), 0 } } } } },
299 /* uimm: unsigned immediate */
300 { "uimm", LM32_OPERAND_UIMM, HW_H_UINT, 15, 16,
301 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_UIMM] } },
302 { 0, { { { (1<<MACH_BASE), 0 } } } } },
303 /* branch: branch offset */
304 { "branch", LM32_OPERAND_BRANCH, HW_H_IADDR, 15, 16,
305 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_BRANCH] } },
306 { 0|A(PCREL_ADDR), { { { (1<<MACH_BASE), 0 } } } } },
307 /* call: call offset */
308 { "call", LM32_OPERAND_CALL, HW_H_IADDR, 25, 26,
309 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_CALL] } },
310 { 0|A(PCREL_ADDR), { { { (1<<MACH_BASE), 0 } } } } },
311 /* csr: csr */
312 { "csr", LM32_OPERAND_CSR, HW_H_CSR, 25, 5,
313 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_CSR] } },
314 { 0, { { { (1<<MACH_BASE), 0 } } } } },
315 /* user: user */
316 { "user", LM32_OPERAND_USER, HW_H_UINT, 10, 11,
317 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_USER] } },
318 { 0, { { { (1<<MACH_BASE), 0 } } } } },
319 /* exception: exception */
320 { "exception", LM32_OPERAND_EXCEPTION, HW_H_UINT, 25, 26,
321 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_EXCEPTION] } },
322 { 0, { { { (1<<MACH_BASE), 0 } } } } },
323 /* hi16: high 16-bit immediate */
324 { "hi16", LM32_OPERAND_HI16, HW_H_UINT, 15, 16,
325 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_UIMM] } },
326 { 0, { { { (1<<MACH_BASE), 0 } } } } },
327 /* lo16: low 16-bit immediate */
328 { "lo16", LM32_OPERAND_LO16, HW_H_UINT, 15, 16,
329 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_UIMM] } },
330 { 0, { { { (1<<MACH_BASE), 0 } } } } },
331 /* gp16: gp relative 16-bit immediate */
332 { "gp16", LM32_OPERAND_GP16, HW_H_SINT, 15, 16,
333 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_IMM] } },
334 { 0, { { { (1<<MACH_BASE), 0 } } } } },
335 /* got16: got 16-bit immediate */
336 { "got16", LM32_OPERAND_GOT16, HW_H_SINT, 15, 16,
337 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_IMM] } },
338 { 0, { { { (1<<MACH_BASE), 0 } } } } },
339 /* gotoffhi16: got offset high 16-bit immediate */
340 { "gotoffhi16", LM32_OPERAND_GOTOFFHI16, HW_H_SINT, 15, 16,
341 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_IMM] } },
342 { 0, { { { (1<<MACH_BASE), 0 } } } } },
343 /* gotofflo16: got offset low 16-bit immediate */
344 { "gotofflo16", LM32_OPERAND_GOTOFFLO16, HW_H_SINT, 15, 16,
345 { 0, { (const PTR) &lm32_cgen_ifld_table[LM32_F_IMM] } },
346 { 0, { { { (1<<MACH_BASE), 0 } } } } },
347 /* sentinel */
348 { 0, 0, 0, 0, 0,
349 { 0, { (const PTR) 0 } },
350 { 0, { { { (1<<MACH_BASE), 0 } } } } }
351 };
352
353 #undef A
354
355
356 /* The instruction table. */
357
358 #define OP(field) CGEN_SYNTAX_MAKE_FIELD (OPERAND (field))
359 #define A(a) (1 << CGEN_INSN_##a)
360
361 static const CGEN_IBASE lm32_cgen_insn_table[MAX_INSNS] =
362 {
363 /* Special null first entry.
364 A `num' value of zero is thus invalid.
365 Also, the special `invalid' insn resides here. */
366 { 0, 0, 0, 0, { 0, { { { (1<<MACH_BASE), 0 } } } } },
367 /* add $r2,$r0,$r1 */
368 {
369 LM32_INSN_ADD, "add", "add", 32,
370 { 0, { { { (1<<MACH_BASE), 0 } } } }
371 },
372 /* addi $r1,$r0,$imm */
373 {
374 LM32_INSN_ADDI, "addi", "addi", 32,
375 { 0, { { { (1<<MACH_BASE), 0 } } } }
376 },
377 /* and $r2,$r0,$r1 */
378 {
379 LM32_INSN_AND, "and", "and", 32,
380 { 0, { { { (1<<MACH_BASE), 0 } } } }
381 },
382 /* andi $r1,$r0,$uimm */
383 {
384 LM32_INSN_ANDI, "andi", "andi", 32,
385 { 0, { { { (1<<MACH_BASE), 0 } } } }
386 },
387 /* andhi $r1,$r0,$hi16 */
388 {
389 LM32_INSN_ANDHII, "andhii", "andhi", 32,
390 { 0, { { { (1<<MACH_BASE), 0 } } } }
391 },
392 /* b $r0 */
393 {
394 LM32_INSN_B, "b", "b", 32,
395 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
396 },
397 /* bi $call */
398 {
399 LM32_INSN_BI, "bi", "bi", 32,
400 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
401 },
402 /* be $r0,$r1,$branch */
403 {
404 LM32_INSN_BE, "be", "be", 32,
405 { 0|A(COND_CTI), { { { (1<<MACH_BASE), 0 } } } }
406 },
407 /* bg $r0,$r1,$branch */
408 {
409 LM32_INSN_BG, "bg", "bg", 32,
410 { 0|A(COND_CTI), { { { (1<<MACH_BASE), 0 } } } }
411 },
412 /* bge $r0,$r1,$branch */
413 {
414 LM32_INSN_BGE, "bge", "bge", 32,
415 { 0|A(COND_CTI), { { { (1<<MACH_BASE), 0 } } } }
416 },
417 /* bgeu $r0,$r1,$branch */
418 {
419 LM32_INSN_BGEU, "bgeu", "bgeu", 32,
420 { 0|A(COND_CTI), { { { (1<<MACH_BASE), 0 } } } }
421 },
422 /* bgu $r0,$r1,$branch */
423 {
424 LM32_INSN_BGU, "bgu", "bgu", 32,
425 { 0|A(COND_CTI), { { { (1<<MACH_BASE), 0 } } } }
426 },
427 /* bne $r0,$r1,$branch */
428 {
429 LM32_INSN_BNE, "bne", "bne", 32,
430 { 0|A(COND_CTI), { { { (1<<MACH_BASE), 0 } } } }
431 },
432 /* call $r0 */
433 {
434 LM32_INSN_CALL, "call", "call", 32,
435 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
436 },
437 /* calli $call */
438 {
439 LM32_INSN_CALLI, "calli", "calli", 32,
440 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
441 },
442 /* cmpe $r2,$r0,$r1 */
443 {
444 LM32_INSN_CMPE, "cmpe", "cmpe", 32,
445 { 0, { { { (1<<MACH_BASE), 0 } } } }
446 },
447 /* cmpei $r1,$r0,$imm */
448 {
449 LM32_INSN_CMPEI, "cmpei", "cmpei", 32,
450 { 0, { { { (1<<MACH_BASE), 0 } } } }
451 },
452 /* cmpg $r2,$r0,$r1 */
453 {
454 LM32_INSN_CMPG, "cmpg", "cmpg", 32,
455 { 0, { { { (1<<MACH_BASE), 0 } } } }
456 },
457 /* cmpgi $r1,$r0,$imm */
458 {
459 LM32_INSN_CMPGI, "cmpgi", "cmpgi", 32,
460 { 0, { { { (1<<MACH_BASE), 0 } } } }
461 },
462 /* cmpge $r2,$r0,$r1 */
463 {
464 LM32_INSN_CMPGE, "cmpge", "cmpge", 32,
465 { 0, { { { (1<<MACH_BASE), 0 } } } }
466 },
467 /* cmpgei $r1,$r0,$imm */
468 {
469 LM32_INSN_CMPGEI, "cmpgei", "cmpgei", 32,
470 { 0, { { { (1<<MACH_BASE), 0 } } } }
471 },
472 /* cmpgeu $r2,$r0,$r1 */
473 {
474 LM32_INSN_CMPGEU, "cmpgeu", "cmpgeu", 32,
475 { 0, { { { (1<<MACH_BASE), 0 } } } }
476 },
477 /* cmpgeui $r1,$r0,$uimm */
478 {
479 LM32_INSN_CMPGEUI, "cmpgeui", "cmpgeui", 32,
480 { 0, { { { (1<<MACH_BASE), 0 } } } }
481 },
482 /* cmpgu $r2,$r0,$r1 */
483 {
484 LM32_INSN_CMPGU, "cmpgu", "cmpgu", 32,
485 { 0, { { { (1<<MACH_BASE), 0 } } } }
486 },
487 /* cmpgui $r1,$r0,$uimm */
488 {
489 LM32_INSN_CMPGUI, "cmpgui", "cmpgui", 32,
490 { 0, { { { (1<<MACH_BASE), 0 } } } }
491 },
492 /* cmpne $r2,$r0,$r1 */
493 {
494 LM32_INSN_CMPNE, "cmpne", "cmpne", 32,
495 { 0, { { { (1<<MACH_BASE), 0 } } } }
496 },
497 /* cmpnei $r1,$r0,$imm */
498 {
499 LM32_INSN_CMPNEI, "cmpnei", "cmpnei", 32,
500 { 0, { { { (1<<MACH_BASE), 0 } } } }
501 },
502 /* divu $r2,$r0,$r1 */
503 {
504 LM32_INSN_DIVU, "divu", "divu", 32,
505 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
506 },
507 /* lb $r1,($r0+$imm) */
508 {
509 LM32_INSN_LB, "lb", "lb", 32,
510 { 0, { { { (1<<MACH_BASE), 0 } } } }
511 },
512 /* lbu $r1,($r0+$imm) */
513 {
514 LM32_INSN_LBU, "lbu", "lbu", 32,
515 { 0, { { { (1<<MACH_BASE), 0 } } } }
516 },
517 /* lh $r1,($r0+$imm) */
518 {
519 LM32_INSN_LH, "lh", "lh", 32,
520 { 0, { { { (1<<MACH_BASE), 0 } } } }
521 },
522 /* lhu $r1,($r0+$imm) */
523 {
524 LM32_INSN_LHU, "lhu", "lhu", 32,
525 { 0, { { { (1<<MACH_BASE), 0 } } } }
526 },
527 /* lw $r1,($r0+$imm) */
528 {
529 LM32_INSN_LW, "lw", "lw", 32,
530 { 0, { { { (1<<MACH_BASE), 0 } } } }
531 },
532 /* modu $r2,$r0,$r1 */
533 {
534 LM32_INSN_MODU, "modu", "modu", 32,
535 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
536 },
537 /* mul $r2,$r0,$r1 */
538 {
539 LM32_INSN_MUL, "mul", "mul", 32,
540 { 0, { { { (1<<MACH_BASE), 0 } } } }
541 },
542 /* muli $r1,$r0,$imm */
543 {
544 LM32_INSN_MULI, "muli", "muli", 32,
545 { 0, { { { (1<<MACH_BASE), 0 } } } }
546 },
547 /* nor $r2,$r0,$r1 */
548 {
549 LM32_INSN_NOR, "nor", "nor", 32,
550 { 0, { { { (1<<MACH_BASE), 0 } } } }
551 },
552 /* nori $r1,$r0,$uimm */
553 {
554 LM32_INSN_NORI, "nori", "nori", 32,
555 { 0, { { { (1<<MACH_BASE), 0 } } } }
556 },
557 /* or $r2,$r0,$r1 */
558 {
559 LM32_INSN_OR, "or", "or", 32,
560 { 0, { { { (1<<MACH_BASE), 0 } } } }
561 },
562 /* ori $r1,$r0,$lo16 */
563 {
564 LM32_INSN_ORI, "ori", "ori", 32,
565 { 0, { { { (1<<MACH_BASE), 0 } } } }
566 },
567 /* orhi $r1,$r0,$hi16 */
568 {
569 LM32_INSN_ORHII, "orhii", "orhi", 32,
570 { 0, { { { (1<<MACH_BASE), 0 } } } }
571 },
572 /* rcsr $r2,$csr */
573 {
574 LM32_INSN_RCSR, "rcsr", "rcsr", 32,
575 { 0, { { { (1<<MACH_BASE), 0 } } } }
576 },
577 /* sb ($r0+$imm),$r1 */
578 {
579 LM32_INSN_SB, "sb", "sb", 32,
580 { 0, { { { (1<<MACH_BASE), 0 } } } }
581 },
582 /* sextb $r2,$r0 */
583 {
584 LM32_INSN_SEXTB, "sextb", "sextb", 32,
585 { 0, { { { (1<<MACH_BASE), 0 } } } }
586 },
587 /* sexth $r2,$r0 */
588 {
589 LM32_INSN_SEXTH, "sexth", "sexth", 32,
590 { 0, { { { (1<<MACH_BASE), 0 } } } }
591 },
592 /* sh ($r0+$imm),$r1 */
593 {
594 LM32_INSN_SH, "sh", "sh", 32,
595 { 0, { { { (1<<MACH_BASE), 0 } } } }
596 },
597 /* sl $r2,$r0,$r1 */
598 {
599 LM32_INSN_SL, "sl", "sl", 32,
600 { 0, { { { (1<<MACH_BASE), 0 } } } }
601 },
602 /* sli $r1,$r0,$imm */
603 {
604 LM32_INSN_SLI, "sli", "sli", 32,
605 { 0, { { { (1<<MACH_BASE), 0 } } } }
606 },
607 /* sr $r2,$r0,$r1 */
608 {
609 LM32_INSN_SR, "sr", "sr", 32,
610 { 0, { { { (1<<MACH_BASE), 0 } } } }
611 },
612 /* sri $r1,$r0,$imm */
613 {
614 LM32_INSN_SRI, "sri", "sri", 32,
615 { 0, { { { (1<<MACH_BASE), 0 } } } }
616 },
617 /* sru $r2,$r0,$r1 */
618 {
619 LM32_INSN_SRU, "sru", "sru", 32,
620 { 0, { { { (1<<MACH_BASE), 0 } } } }
621 },
622 /* srui $r1,$r0,$imm */
623 {
624 LM32_INSN_SRUI, "srui", "srui", 32,
625 { 0, { { { (1<<MACH_BASE), 0 } } } }
626 },
627 /* sub $r2,$r0,$r1 */
628 {
629 LM32_INSN_SUB, "sub", "sub", 32,
630 { 0, { { { (1<<MACH_BASE), 0 } } } }
631 },
632 /* sw ($r0+$imm),$r1 */
633 {
634 LM32_INSN_SW, "sw", "sw", 32,
635 { 0, { { { (1<<MACH_BASE), 0 } } } }
636 },
637 /* user $r2,$r0,$r1,$user */
638 {
639 LM32_INSN_USER, "user", "user", 32,
640 { 0, { { { (1<<MACH_BASE), 0 } } } }
641 },
642 /* wcsr $csr,$r1 */
643 {
644 LM32_INSN_WCSR, "wcsr", "wcsr", 32,
645 { 0, { { { (1<<MACH_BASE), 0 } } } }
646 },
647 /* xor $r2,$r0,$r1 */
648 {
649 LM32_INSN_XOR, "xor", "xor", 32,
650 { 0, { { { (1<<MACH_BASE), 0 } } } }
651 },
652 /* xori $r1,$r0,$uimm */
653 {
654 LM32_INSN_XORI, "xori", "xori", 32,
655 { 0, { { { (1<<MACH_BASE), 0 } } } }
656 },
657 /* xnor $r2,$r0,$r1 */
658 {
659 LM32_INSN_XNOR, "xnor", "xnor", 32,
660 { 0, { { { (1<<MACH_BASE), 0 } } } }
661 },
662 /* xnori $r1,$r0,$uimm */
663 {
664 LM32_INSN_XNORI, "xnori", "xnori", 32,
665 { 0, { { { (1<<MACH_BASE), 0 } } } }
666 },
667 /* break */
668 {
669 LM32_INSN_BREAK, "break", "break", 32,
670 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
671 },
672 /* scall */
673 {
674 LM32_INSN_SCALL, "scall", "scall", 32,
675 { 0|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
676 },
677 /* bret */
678 {
679 -1, "bret", "bret", 32,
680 { 0|A(ALIAS)|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
681 },
682 /* eret */
683 {
684 -1, "eret", "eret", 32,
685 { 0|A(ALIAS)|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
686 },
687 /* ret */
688 {
689 -1, "ret", "ret", 32,
690 { 0|A(ALIAS)|A(UNCOND_CTI), { { { (1<<MACH_BASE), 0 } } } }
691 },
692 /* mv $r2,$r0 */
693 {
694 -1, "mv", "mv", 32,
695 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
696 },
697 /* mvi $r1,$imm */
698 {
699 -1, "mvi", "mvi", 32,
700 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
701 },
702 /* mvu $r1,$lo16 */
703 {
704 -1, "mvui", "mvu", 32,
705 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
706 },
707 /* mvhi $r1,$hi16 */
708 {
709 -1, "mvhi", "mvhi", 32,
710 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
711 },
712 /* mva $r1,$gp16 */
713 {
714 -1, "mva", "mva", 32,
715 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
716 },
717 /* not $r2,$r0 */
718 {
719 -1, "not", "not", 32,
720 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
721 },
722 /* nop */
723 {
724 -1, "nop", "nop", 32,
725 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
726 },
727 /* lb $r1,$gp16 */
728 {
729 -1, "lbgprel", "lb", 32,
730 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
731 },
732 /* lbu $r1,$gp16 */
733 {
734 -1, "lbugprel", "lbu", 32,
735 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
736 },
737 /* lh $r1,$gp16 */
738 {
739 -1, "lhgprel", "lh", 32,
740 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
741 },
742 /* lhu $r1,$gp16 */
743 {
744 -1, "lhugprel", "lhu", 32,
745 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
746 },
747 /* lw $r1,$gp16 */
748 {
749 -1, "lwgprel", "lw", 32,
750 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
751 },
752 /* sb $gp16,$r1 */
753 {
754 -1, "sbgprel", "sb", 32,
755 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
756 },
757 /* sh $gp16,$r1 */
758 {
759 -1, "shgprel", "sh", 32,
760 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
761 },
762 /* sw $gp16,$r1 */
763 {
764 -1, "swgprel", "sw", 32,
765 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
766 },
767 /* lw $r1,(gp+$got16) */
768 {
769 -1, "lwgotrel", "lw", 32,
770 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
771 },
772 /* orhi $r1,$r0,$gotoffhi16 */
773 {
774 -1, "orhigotoffi", "orhi", 32,
775 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
776 },
777 /* addi $r1,$r0,$gotofflo16 */
778 {
779 -1, "addgotoff", "addi", 32,
780 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
781 },
782 /* sw ($r0+$gotofflo16),$r1 */
783 {
784 -1, "swgotoff", "sw", 32,
785 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
786 },
787 /* lw $r1,($r0+$gotofflo16) */
788 {
789 -1, "lwgotoff", "lw", 32,
790 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
791 },
792 /* sh ($r0+$gotofflo16),$r1 */
793 {
794 -1, "shgotoff", "sh", 32,
795 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
796 },
797 /* lh $r1,($r0+$gotofflo16) */
798 {
799 -1, "lhgotoff", "lh", 32,
800 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
801 },
802 /* lhu $r1,($r0+$gotofflo16) */
803 {
804 -1, "lhugotoff", "lhu", 32,
805 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
806 },
807 /* sb ($r0+$gotofflo16),$r1 */
808 {
809 -1, "sbgotoff", "sb", 32,
810 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
811 },
812 /* lb $r1,($r0+$gotofflo16) */
813 {
814 -1, "lbgotoff", "lb", 32,
815 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
816 },
817 /* lbu $r1,($r0+$gotofflo16) */
818 {
819 -1, "lbugotoff", "lbu", 32,
820 { 0|A(ALIAS), { { { (1<<MACH_BASE), 0 } } } }
821 },
822 };
823
824 #undef OP
825 #undef A
826
827 /* Initialize anything needed to be done once, before any cpu_open call. */
828
829 static void
init_tables(void)830 init_tables (void)
831 {
832 }
833
834 static const CGEN_MACH * lookup_mach_via_bfd_name (const CGEN_MACH *, const char *);
835 static void build_hw_table (CGEN_CPU_TABLE *);
836 static void build_ifield_table (CGEN_CPU_TABLE *);
837 static void build_operand_table (CGEN_CPU_TABLE *);
838 static void build_insn_table (CGEN_CPU_TABLE *);
839 static void lm32_cgen_rebuild_tables (CGEN_CPU_TABLE *);
840
841 /* Subroutine of lm32_cgen_cpu_open to look up a mach via its bfd name. */
842
843 static const CGEN_MACH *
lookup_mach_via_bfd_name(const CGEN_MACH * table,const char * name)844 lookup_mach_via_bfd_name (const CGEN_MACH *table, const char *name)
845 {
846 while (table->name)
847 {
848 if (strcmp (name, table->bfd_name) == 0)
849 return table;
850 ++table;
851 }
852 abort ();
853 }
854
855 /* Subroutine of lm32_cgen_cpu_open to build the hardware table. */
856
857 static void
build_hw_table(CGEN_CPU_TABLE * cd)858 build_hw_table (CGEN_CPU_TABLE *cd)
859 {
860 int i;
861 int machs = cd->machs;
862 const CGEN_HW_ENTRY *init = & lm32_cgen_hw_table[0];
863 /* MAX_HW is only an upper bound on the number of selected entries.
864 However each entry is indexed by it's enum so there can be holes in
865 the table. */
866 const CGEN_HW_ENTRY **selected =
867 (const CGEN_HW_ENTRY **) xmalloc (MAX_HW * sizeof (CGEN_HW_ENTRY *));
868
869 cd->hw_table.init_entries = init;
870 cd->hw_table.entry_size = sizeof (CGEN_HW_ENTRY);
871 memset (selected, 0, MAX_HW * sizeof (CGEN_HW_ENTRY *));
872 /* ??? For now we just use machs to determine which ones we want. */
873 for (i = 0; init[i].name != NULL; ++i)
874 if (CGEN_HW_ATTR_VALUE (&init[i], CGEN_HW_MACH)
875 & machs)
876 selected[init[i].type] = &init[i];
877 cd->hw_table.entries = selected;
878 cd->hw_table.num_entries = MAX_HW;
879 }
880
881 /* Subroutine of lm32_cgen_cpu_open to build the hardware table. */
882
883 static void
build_ifield_table(CGEN_CPU_TABLE * cd)884 build_ifield_table (CGEN_CPU_TABLE *cd)
885 {
886 cd->ifld_table = & lm32_cgen_ifld_table[0];
887 }
888
889 /* Subroutine of lm32_cgen_cpu_open to build the hardware table. */
890
891 static void
build_operand_table(CGEN_CPU_TABLE * cd)892 build_operand_table (CGEN_CPU_TABLE *cd)
893 {
894 int i;
895 int machs = cd->machs;
896 const CGEN_OPERAND *init = & lm32_cgen_operand_table[0];
897 /* MAX_OPERANDS is only an upper bound on the number of selected entries.
898 However each entry is indexed by it's enum so there can be holes in
899 the table. */
900 const CGEN_OPERAND **selected = xmalloc (MAX_OPERANDS * sizeof (* selected));
901
902 cd->operand_table.init_entries = init;
903 cd->operand_table.entry_size = sizeof (CGEN_OPERAND);
904 memset (selected, 0, MAX_OPERANDS * sizeof (CGEN_OPERAND *));
905 /* ??? For now we just use mach to determine which ones we want. */
906 for (i = 0; init[i].name != NULL; ++i)
907 if (CGEN_OPERAND_ATTR_VALUE (&init[i], CGEN_OPERAND_MACH)
908 & machs)
909 selected[init[i].type] = &init[i];
910 cd->operand_table.entries = selected;
911 cd->operand_table.num_entries = MAX_OPERANDS;
912 }
913
914 /* Subroutine of lm32_cgen_cpu_open to build the hardware table.
915 ??? This could leave out insns not supported by the specified mach/isa,
916 but that would cause errors like "foo only supported by bar" to become
917 "unknown insn", so for now we include all insns and require the app to
918 do the checking later.
919 ??? On the other hand, parsing of such insns may require their hardware or
920 operand elements to be in the table [which they mightn't be]. */
921
922 static void
build_insn_table(CGEN_CPU_TABLE * cd)923 build_insn_table (CGEN_CPU_TABLE *cd)
924 {
925 int i;
926 const CGEN_IBASE *ib = & lm32_cgen_insn_table[0];
927 CGEN_INSN *insns = xmalloc (MAX_INSNS * sizeof (CGEN_INSN));
928
929 memset (insns, 0, MAX_INSNS * sizeof (CGEN_INSN));
930 for (i = 0; i < MAX_INSNS; ++i)
931 insns[i].base = &ib[i];
932 cd->insn_table.init_entries = insns;
933 cd->insn_table.entry_size = sizeof (CGEN_IBASE);
934 cd->insn_table.num_init_entries = MAX_INSNS;
935 }
936
937 /* Subroutine of lm32_cgen_cpu_open to rebuild the tables. */
938
939 static void
lm32_cgen_rebuild_tables(CGEN_CPU_TABLE * cd)940 lm32_cgen_rebuild_tables (CGEN_CPU_TABLE *cd)
941 {
942 int i;
943 CGEN_BITSET *isas = cd->isas;
944 unsigned int machs = cd->machs;
945
946 cd->int_insn_p = CGEN_INT_INSN_P;
947
948 /* Data derived from the isa spec. */
949 #define UNSET (CGEN_SIZE_UNKNOWN + 1)
950 cd->default_insn_bitsize = UNSET;
951 cd->base_insn_bitsize = UNSET;
952 cd->min_insn_bitsize = 65535; /* Some ridiculously big number. */
953 cd->max_insn_bitsize = 0;
954 for (i = 0; i < MAX_ISAS; ++i)
955 if (cgen_bitset_contains (isas, i))
956 {
957 const CGEN_ISA *isa = & lm32_cgen_isa_table[i];
958
959 /* Default insn sizes of all selected isas must be
960 equal or we set the result to 0, meaning "unknown". */
961 if (cd->default_insn_bitsize == UNSET)
962 cd->default_insn_bitsize = isa->default_insn_bitsize;
963 else if (isa->default_insn_bitsize == cd->default_insn_bitsize)
964 ; /* This is ok. */
965 else
966 cd->default_insn_bitsize = CGEN_SIZE_UNKNOWN;
967
968 /* Base insn sizes of all selected isas must be equal
969 or we set the result to 0, meaning "unknown". */
970 if (cd->base_insn_bitsize == UNSET)
971 cd->base_insn_bitsize = isa->base_insn_bitsize;
972 else if (isa->base_insn_bitsize == cd->base_insn_bitsize)
973 ; /* This is ok. */
974 else
975 cd->base_insn_bitsize = CGEN_SIZE_UNKNOWN;
976
977 /* Set min,max insn sizes. */
978 if (isa->min_insn_bitsize < cd->min_insn_bitsize)
979 cd->min_insn_bitsize = isa->min_insn_bitsize;
980 if (isa->max_insn_bitsize > cd->max_insn_bitsize)
981 cd->max_insn_bitsize = isa->max_insn_bitsize;
982 }
983
984 /* Data derived from the mach spec. */
985 for (i = 0; i < MAX_MACHS; ++i)
986 if (((1 << i) & machs) != 0)
987 {
988 const CGEN_MACH *mach = & lm32_cgen_mach_table[i];
989
990 if (mach->insn_chunk_bitsize != 0)
991 {
992 if (cd->insn_chunk_bitsize != 0 && cd->insn_chunk_bitsize != mach->insn_chunk_bitsize)
993 {
994 fprintf (stderr, "lm32_cgen_rebuild_tables: conflicting insn-chunk-bitsize values: `%d' vs. `%d'\n",
995 cd->insn_chunk_bitsize, mach->insn_chunk_bitsize);
996 abort ();
997 }
998
999 cd->insn_chunk_bitsize = mach->insn_chunk_bitsize;
1000 }
1001 }
1002
1003 /* Determine which hw elements are used by MACH. */
1004 build_hw_table (cd);
1005
1006 /* Build the ifield table. */
1007 build_ifield_table (cd);
1008
1009 /* Determine which operands are used by MACH/ISA. */
1010 build_operand_table (cd);
1011
1012 /* Build the instruction table. */
1013 build_insn_table (cd);
1014 }
1015
1016 /* Initialize a cpu table and return a descriptor.
1017 It's much like opening a file, and must be the first function called.
1018 The arguments are a set of (type/value) pairs, terminated with
1019 CGEN_CPU_OPEN_END.
1020
1021 Currently supported values:
1022 CGEN_CPU_OPEN_ISAS: bitmap of values in enum isa_attr
1023 CGEN_CPU_OPEN_MACHS: bitmap of values in enum mach_attr
1024 CGEN_CPU_OPEN_BFDMACH: specify 1 mach using bfd name
1025 CGEN_CPU_OPEN_ENDIAN: specify endian choice
1026 CGEN_CPU_OPEN_END: terminates arguments
1027
1028 ??? Simultaneous multiple isas might not make sense, but it's not (yet)
1029 precluded. */
1030
1031 CGEN_CPU_DESC
lm32_cgen_cpu_open(enum cgen_cpu_open_arg arg_type,...)1032 lm32_cgen_cpu_open (enum cgen_cpu_open_arg arg_type, ...)
1033 {
1034 CGEN_CPU_TABLE *cd = (CGEN_CPU_TABLE *) xmalloc (sizeof (CGEN_CPU_TABLE));
1035 static int init_p;
1036 CGEN_BITSET *isas = 0; /* 0 = "unspecified" */
1037 unsigned int machs = 0; /* 0 = "unspecified" */
1038 enum cgen_endian endian = CGEN_ENDIAN_UNKNOWN;
1039 va_list ap;
1040
1041 if (! init_p)
1042 {
1043 init_tables ();
1044 init_p = 1;
1045 }
1046
1047 memset (cd, 0, sizeof (*cd));
1048
1049 va_start (ap, arg_type);
1050 while (arg_type != CGEN_CPU_OPEN_END)
1051 {
1052 switch (arg_type)
1053 {
1054 case CGEN_CPU_OPEN_ISAS :
1055 isas = va_arg (ap, CGEN_BITSET *);
1056 break;
1057 case CGEN_CPU_OPEN_MACHS :
1058 machs = va_arg (ap, unsigned int);
1059 break;
1060 case CGEN_CPU_OPEN_BFDMACH :
1061 {
1062 const char *name = va_arg (ap, const char *);
1063 const CGEN_MACH *mach =
1064 lookup_mach_via_bfd_name (lm32_cgen_mach_table, name);
1065
1066 machs |= 1 << mach->num;
1067 break;
1068 }
1069 case CGEN_CPU_OPEN_ENDIAN :
1070 endian = va_arg (ap, enum cgen_endian);
1071 break;
1072 default :
1073 fprintf (stderr, "lm32_cgen_cpu_open: unsupported argument `%d'\n",
1074 arg_type);
1075 abort (); /* ??? return NULL? */
1076 }
1077 arg_type = va_arg (ap, enum cgen_cpu_open_arg);
1078 }
1079 va_end (ap);
1080
1081 /* Mach unspecified means "all". */
1082 if (machs == 0)
1083 machs = (1 << MAX_MACHS) - 1;
1084 /* Base mach is always selected. */
1085 machs |= 1;
1086 if (endian == CGEN_ENDIAN_UNKNOWN)
1087 {
1088 /* ??? If target has only one, could have a default. */
1089 fprintf (stderr, "lm32_cgen_cpu_open: no endianness specified\n");
1090 abort ();
1091 }
1092
1093 cd->isas = cgen_bitset_copy (isas);
1094 cd->machs = machs;
1095 cd->endian = endian;
1096 /* FIXME: for the sparc case we can determine insn-endianness statically.
1097 The worry here is where both data and insn endian can be independently
1098 chosen, in which case this function will need another argument.
1099 Actually, will want to allow for more arguments in the future anyway. */
1100 cd->insn_endian = endian;
1101
1102 /* Table (re)builder. */
1103 cd->rebuild_tables = lm32_cgen_rebuild_tables;
1104 lm32_cgen_rebuild_tables (cd);
1105
1106 /* Default to not allowing signed overflow. */
1107 cd->signed_overflow_ok_p = 0;
1108
1109 return (CGEN_CPU_DESC) cd;
1110 }
1111
1112 /* Cover fn to lm32_cgen_cpu_open to handle the simple case of 1 isa, 1 mach.
1113 MACH_NAME is the bfd name of the mach. */
1114
1115 CGEN_CPU_DESC
lm32_cgen_cpu_open_1(const char * mach_name,enum cgen_endian endian)1116 lm32_cgen_cpu_open_1 (const char *mach_name, enum cgen_endian endian)
1117 {
1118 return lm32_cgen_cpu_open (CGEN_CPU_OPEN_BFDMACH, mach_name,
1119 CGEN_CPU_OPEN_ENDIAN, endian,
1120 CGEN_CPU_OPEN_END);
1121 }
1122
1123 /* Close a cpu table.
1124 ??? This can live in a machine independent file, but there's currently
1125 no place to put this file (there's no libcgen). libopcodes is the wrong
1126 place as some simulator ports use this but they don't use libopcodes. */
1127
1128 void
lm32_cgen_cpu_close(CGEN_CPU_DESC cd)1129 lm32_cgen_cpu_close (CGEN_CPU_DESC cd)
1130 {
1131 unsigned int i;
1132 const CGEN_INSN *insns;
1133
1134 if (cd->macro_insn_table.init_entries)
1135 {
1136 insns = cd->macro_insn_table.init_entries;
1137 for (i = 0; i < cd->macro_insn_table.num_init_entries; ++i, ++insns)
1138 if (CGEN_INSN_RX ((insns)))
1139 regfree (CGEN_INSN_RX (insns));
1140 }
1141
1142 if (cd->insn_table.init_entries)
1143 {
1144 insns = cd->insn_table.init_entries;
1145 for (i = 0; i < cd->insn_table.num_init_entries; ++i, ++insns)
1146 if (CGEN_INSN_RX (insns))
1147 regfree (CGEN_INSN_RX (insns));
1148 }
1149
1150 if (cd->macro_insn_table.init_entries)
1151 free ((CGEN_INSN *) cd->macro_insn_table.init_entries);
1152
1153 if (cd->insn_table.init_entries)
1154 free ((CGEN_INSN *) cd->insn_table.init_entries);
1155
1156 if (cd->hw_table.entries)
1157 free ((CGEN_HW_ENTRY *) cd->hw_table.entries);
1158
1159 if (cd->operand_table.entries)
1160 free ((CGEN_HW_ENTRY *) cd->operand_table.entries);
1161
1162 free (cd);
1163 }
1164
1165