Coverage Report

Created: 2026-04-04 08:16

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/opcodes/mips-dis.c
Line
Count
Source
1
/* Print mips instructions for GDB, the GNU debugger, or for objdump.
2
   Copyright (C) 1989-2026 Free Software Foundation, Inc.
3
   Contributed by Nobuyuki Hikichi(hikichi@sra.co.jp).
4
5
   This file is part of the GNU opcodes library.
6
7
   This library 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 3, or (at your option)
10
   any later version.
11
12
   It is distributed in the hope that it will be useful, but WITHOUT
13
   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
14
   or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
15
   License for more details.
16
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., 51 Franklin Street - Fifth Floor, Boston,
20
   MA 02110-1301, USA.  */
21
22
#include "sysdep.h"
23
#include "disassemble.h"
24
#include "libiberty.h"
25
#include "opcode/mips.h"
26
#include "opintl.h"
27
#include "elf-bfd.h"
28
#include "elf/mips.h"
29
#include "elfxx-mips.h"
30
31
/* FIXME: These are needed to figure out if the code is mips16 or
32
   not. The low bit of the address is often a good indicator.  No
33
   symbol table is available when this code runs out in an embedded
34
   system as when it is used for disassembler support in a monitor.  */
35
36
#if !defined(EMBEDDED_ENV)
37
#define SYMTAB_AVAILABLE 1
38
#endif
39
40
/* Mips instructions are at maximum this many bytes long.  */
41
1.33M
#define INSNLEN 4
42
43

44
/* FIXME: These should be shared with gdb somehow.  */
45
46
struct mips_cp0sel_name
47
{
48
  unsigned int cp0reg;
49
  unsigned int sel;
50
  const char * const name;
51
};
52
53
static const char * const mips_gpr_names_numeric[32] =
54
{
55
  "$0",   "$1",   "$2",   "$3",   "$4",   "$5",   "$6",   "$7",
56
  "$8",   "$9",   "$10",  "$11",  "$12",  "$13",  "$14",  "$15",
57
  "$16",  "$17",  "$18",  "$19",  "$20",  "$21",  "$22",  "$23",
58
  "$24",  "$25",  "$26",  "$27",  "$28",  "$29",  "$30",  "$31"
59
};
60
61
static const char * const mips_gpr_names_oldabi[32] =
62
{
63
  "zero", "at",   "v0",   "v1",   "a0",   "a1",   "a2",   "a3",
64
  "t0",   "t1",   "t2",   "t3",   "t4",   "t5",   "t6",   "t7",
65
  "s0",   "s1",   "s2",   "s3",   "s4",   "s5",   "s6",   "s7",
66
  "t8",   "t9",   "k0",   "k1",   "gp",   "sp",   "s8",   "ra"
67
};
68
69
static const char * const mips_gpr_names_newabi[32] =
70
{
71
  "zero", "at",   "v0",   "v1",   "a0",   "a1",   "a2",   "a3",
72
  "a4",   "a5",   "a6",   "a7",   "t0",   "t1",   "t2",   "t3",
73
  "s0",   "s1",   "s2",   "s3",   "s4",   "s5",   "s6",   "s7",
74
  "t8",   "t9",   "k0",   "k1",   "gp",   "sp",   "s8",   "ra"
75
};
76
77
static const char * const mips_fpr_names_numeric[32] =
78
{
79
  "$f0",  "$f1",  "$f2",  "$f3",  "$f4",  "$f5",  "$f6",  "$f7",
80
  "$f8",  "$f9",  "$f10", "$f11", "$f12", "$f13", "$f14", "$f15",
81
  "$f16", "$f17", "$f18", "$f19", "$f20", "$f21", "$f22", "$f23",
82
  "$f24", "$f25", "$f26", "$f27", "$f28", "$f29", "$f30", "$f31"
83
};
84
85
static const char * const mips_fpr_names_32[32] =
86
{
87
  "fv0",  "fv0f", "fv1",  "fv1f", "ft0",  "ft0f", "ft1",  "ft1f",
88
  "ft2",  "ft2f", "ft3",  "ft3f", "fa0",  "fa0f", "fa1",  "fa1f",
89
  "ft4",  "ft4f", "ft5",  "ft5f", "fs0",  "fs0f", "fs1",  "fs1f",
90
  "fs2",  "fs2f", "fs3",  "fs3f", "fs4",  "fs4f", "fs5",  "fs5f"
91
};
92
93
static const char * const mips_fpr_names_o64[32] =
94
{
95
  "fv0",  "fv1",  "ft0",  "ft1",  "ft2",  "ft3",  "ft4",  "ft5",
96
  "ft6",  "ft7",  "ft8",  "ft9",  "fa0",  "fa1",  "ft10", "ft11",
97
  "ft12", "ft13", "ft14", "ft15", "fs0",  "fs1",  "fs2",  "fs3",
98
  "fs4",  "fs5",  "fs6",  "fs7",  "fs8",  "fs9",  "fs10", "fs11",
99
};
100
101
static const char * const mips_fpr_names_n32[32] =
102
{
103
  "fv0",  "ft14", "fv1",  "ft15", "ft0",  "ft1",  "ft2",  "ft3",
104
  "ft4",  "ft5",  "ft6",  "ft7",  "fa0",  "fa1",  "fa2",  "fa3",
105
  "fa4",  "fa5",  "fa6",  "fa7",  "fs0",  "ft8",  "fs1",  "ft9",
106
  "fs2",  "ft10", "fs3",  "ft11", "fs4",  "ft12", "fs5",  "ft13"
107
};
108
109
static const char * const mips_fpr_names_64[32] =
110
{
111
  "fv0",  "ft12", "fv1",  "ft13", "ft0",  "ft1",  "ft2",  "ft3",
112
  "ft4",  "ft5",  "ft6",  "ft7",  "fa0",  "fa1",  "fa2",  "fa3",
113
  "fa4",  "fa5",  "fa6",  "fa7",  "ft8",  "ft9",  "ft10", "ft11",
114
  "fs0",  "fs1",  "fs2",  "fs3",  "fs4",  "fs5",  "fs6",  "fs7"
115
};
116
117
static const char * const mips_fpr_names_eabi32[32] =
118
{
119
  "fv0",  "fv0f", "fv1",  "fv1f", "ft0",  "ft0f",  "ft1", "ft1f",
120
  "ft2",  "ft2f", "ft3",  "ft3f", "fa0",  "fa0f",  "fa1", "fa1f",
121
  "fa2",  "fa2f", "fa3",  "fa3f", "fs0",  "fs0f",  "fs1", "fs1f",
122
  "fs2",  "fs2f", "fs3",  "fs3f", "fs4",  "fs4f",  "fs5", "fs5f",
123
};
124
125
static const char * const mips_fpr_names_eabi64[32] =
126
{
127
  "fv0",  "fv1",  "ft0",  "ft1",  "ft2",  "ft3",  "ft4",  "ft5",
128
  "ft6",  "ft7",  "ft8",  "ft9",  "fa0",  "fa1",  "fa2",  "fa3",
129
  "fa4",  "fa5",  "fa6",  "fa7",  "fs0",  "fs1",  "fs2",  "fs3",
130
  "fs4",  "fs5",  "fs6",  "fs7",  "fs8",  "fs9",  "fs10", "fs11",
131
};
132
133
static const char * const mips_cp0_names_numeric[32] =
134
{
135
  "$0",   "$1",   "$2",   "$3",   "$4",   "$5",   "$6",   "$7",
136
  "$8",   "$9",   "$10",  "$11",  "$12",  "$13",  "$14",  "$15",
137
  "$16",  "$17",  "$18",  "$19",  "$20",  "$21",  "$22",  "$23",
138
  "$24",  "$25",  "$26",  "$27",  "$28",  "$29",  "$30",  "$31"
139
};
140
141
static const char * const mips_cp1_names_numeric[32] =
142
{
143
  "$0",   "$1",   "$2",   "$3",   "$4",   "$5",   "$6",   "$7",
144
  "$8",   "$9",   "$10",  "$11",  "$12",  "$13",  "$14",  "$15",
145
  "$16",  "$17",  "$18",  "$19",  "$20",  "$21",  "$22",  "$23",
146
  "$24",  "$25",  "$26",  "$27",  "$28",  "$29",  "$30",  "$31"
147
};
148
149
static const char * const mips_cp0_names_r3900[32] =
150
{
151
  "$0",           "$1",           "$2",           "c0_config",
152
  "$4",           "$5",           "$6",           "c0_cache",
153
  "c0_badvaddr",  "$9",           "$10",          "$11",
154
  "c0_sr",        "c0_cause",     "c0_epc",       "c0_prid",
155
  "c0_debug",     "c0_depc",      "$18",          "$19",
156
  "$20",          "$21",          "$22",          "$23",
157
  "$24",          "$25",          "$26",          "$27",
158
  "$28",          "$29",          "$30",          "$31",
159
};
160
161
static const char * const mips_cp0_names_r3000[32] =
162
{
163
  "c0_index",     "c0_random",    "c0_entrylo",   "$3",
164
  "c0_context",   "$5",           "$6",           "$7",
165
  "c0_badvaddr",  "$9",           "c0_entryhi",   "$11",
166
  "c0_sr",        "c0_cause",     "c0_epc",       "c0_prid",
167
  "$16",          "$17",          "$18",          "$19",
168
  "$20",          "$21",          "$22",          "$23",
169
  "$24",          "$25",          "$26",          "$27",
170
  "$28",          "$29",          "$30",          "$31",
171
};
172
173
static const char * const mips_cp0_names_r4000[32] =
174
{
175
  "c0_index",     "c0_random",    "c0_entrylo0",  "c0_entrylo1",
176
  "c0_context",   "c0_pagemask",  "c0_wired",     "$7",
177
  "c0_badvaddr",  "c0_count",     "c0_entryhi",   "c0_compare",
178
  "c0_sr",        "c0_cause",     "c0_epc",       "c0_prid",
179
  "c0_config",    "c0_lladdr",    "c0_watchlo",   "c0_watchhi",
180
  "c0_xcontext",  "$21",          "$22",          "$23",
181
  "$24",          "$25",          "c0_ecc",       "c0_cacheerr",
182
  "c0_taglo",     "c0_taghi",     "c0_errorepc",  "$31",
183
};
184
185
static const char * const mips_cp0_names_r5900[32] =
186
{
187
  "c0_index",     "c0_random",    "c0_entrylo0",  "c0_entrylo1",
188
  "c0_context",   "c0_pagemask",  "c0_wired",     "$7",
189
  "c0_badvaddr",  "c0_count",     "c0_entryhi",   "c0_compare",
190
  "c0_sr",        "c0_cause",     "c0_epc",       "c0_prid",
191
  "c0_config",    "$17",          "$18",          "$19",
192
  "$20",          "$21",          "$22",          "c0_badpaddr",
193
  "c0_depc",      "c0_perfcnt",   "$26",          "$27",
194
  "c0_taglo",     "c0_taghi",     "c0_errorepc",  "$31"
195
};
196
197
static const char * const mips_cp0_names_mips3264[32] =
198
{
199
  "c0_index",     "c0_random",    "c0_entrylo0",  "c0_entrylo1",
200
  "c0_context",   "c0_pagemask",  "c0_wired",     "$7",
201
  "c0_badvaddr",  "c0_count",     "c0_entryhi",   "c0_compare",
202
  "c0_status",    "c0_cause",     "c0_epc",       "c0_prid",
203
  "c0_config",    "c0_lladdr",    "c0_watchlo",   "c0_watchhi",
204
  "c0_xcontext",  "$21",          "$22",          "c0_debug",
205
  "c0_depc",      "c0_perfcnt",   "c0_errctl",    "c0_cacheerr",
206
  "c0_taglo",     "c0_taghi",     "c0_errorepc",  "c0_desave",
207
};
208
209
static const char * const mips_cp1_names_mips[32] =
210
{
211
  "c1_fir",       "$1",           "$2",           "$3",
212
  "$4",           "$5",           "$6",           "$7",
213
  "$8",           "$9",           "$10",          "$11",
214
  "$12",          "$13",          "$14",          "$15",
215
  "$16",          "$17",          "$18",          "$19",
216
  "$20",          "$21",          "$22",          "$23",
217
  "$24",          "$25",          "$26",          "$27",
218
  "$28",          "$29",          "$30",          "c1_fcsr"
219
};
220
221
static const char * const mips_cp1_names_mips3264[32] =
222
{
223
  "c1_fir",       "c1_ufr",       "$2",           "$3",
224
  "c1_unfr",      "$5",           "$6",           "$7",
225
  "$8",           "$9",           "$10",          "$11",
226
  "$12",          "$13",          "$14",          "$15",
227
  "$16",          "$17",          "$18",          "$19",
228
  "$20",          "$21",          "$22",          "$23",
229
  "$24",          "c1_fccr",      "c1_fexr",      "$27",
230
  "c1_fenr",      "$29",          "$30",          "c1_fcsr"
231
};
232
233
static const struct mips_cp0sel_name mips_cp0sel_names_mips3264[] =
234
{
235
  { 16, 1, "c0_config1"   },
236
  { 16, 2, "c0_config2"   },
237
  { 16, 3, "c0_config3"   },
238
  { 18, 1, "c0_watchlo,1" },
239
  { 18, 2, "c0_watchlo,2" },
240
  { 18, 3, "c0_watchlo,3" },
241
  { 18, 4, "c0_watchlo,4" },
242
  { 18, 5, "c0_watchlo,5" },
243
  { 18, 6, "c0_watchlo,6" },
244
  { 18, 7, "c0_watchlo,7" },
245
  { 19, 1, "c0_watchhi,1" },
246
  { 19, 2, "c0_watchhi,2" },
247
  { 19, 3, "c0_watchhi,3" },
248
  { 19, 4, "c0_watchhi,4" },
249
  { 19, 5, "c0_watchhi,5" },
250
  { 19, 6, "c0_watchhi,6" },
251
  { 19, 7, "c0_watchhi,7" },
252
  { 25, 1, "c0_perfcnt,1" },
253
  { 25, 2, "c0_perfcnt,2" },
254
  { 25, 3, "c0_perfcnt,3" },
255
  { 25, 4, "c0_perfcnt,4" },
256
  { 25, 5, "c0_perfcnt,5" },
257
  { 25, 6, "c0_perfcnt,6" },
258
  { 25, 7, "c0_perfcnt,7" },
259
  { 27, 1, "c0_cacheerr,1"  },
260
  { 27, 2, "c0_cacheerr,2"  },
261
  { 27, 3, "c0_cacheerr,3"  },
262
  { 28, 1, "c0_datalo"    },
263
  { 29, 1, "c0_datahi"    }
264
};
265
266
static const char * const mips_cp0_names_mips3264r2[32] =
267
{
268
  "c0_index",     "c0_random",    "c0_entrylo0",  "c0_entrylo1",
269
  "c0_context",   "c0_pagemask",  "c0_wired",     "c0_hwrena",
270
  "c0_badvaddr",  "c0_count",     "c0_entryhi",   "c0_compare",
271
  "c0_status",    "c0_cause",     "c0_epc",       "c0_prid",
272
  "c0_config",    "c0_lladdr",    "c0_watchlo",   "c0_watchhi",
273
  "c0_xcontext",  "$21",          "$22",          "c0_debug",
274
  "c0_depc",      "c0_perfcnt",   "c0_errctl",    "c0_cacheerr",
275
  "c0_taglo",     "c0_taghi",     "c0_errorepc",  "c0_desave",
276
};
277
278
static const struct mips_cp0sel_name mips_cp0sel_names_mips3264r2[] =
279
{
280
  {  4, 1, "c0_contextconfig" },
281
  {  0, 1, "c0_mvpcontrol"  },
282
  {  0, 2, "c0_mvpconf0"  },
283
  {  0, 3, "c0_mvpconf1"  },
284
  {  1, 1, "c0_vpecontrol"  },
285
  {  1, 2, "c0_vpeconf0"  },
286
  {  1, 3, "c0_vpeconf1"  },
287
  {  1, 4, "c0_yqmask"    },
288
  {  1, 5, "c0_vpeschedule" },
289
  {  1, 6, "c0_vpeschefback"  },
290
  {  2, 1, "c0_tcstatus"  },
291
  {  2, 2, "c0_tcbind"    },
292
  {  2, 3, "c0_tcrestart" },
293
  {  2, 4, "c0_tchalt"    },
294
  {  2, 5, "c0_tccontext" },
295
  {  2, 6, "c0_tcschedule"  },
296
  {  2, 7, "c0_tcschefback" },
297
  {  5, 1, "c0_pagegrain" },
298
  {  6, 1, "c0_srsconf0"  },
299
  {  6, 2, "c0_srsconf1"  },
300
  {  6, 3, "c0_srsconf2"  },
301
  {  6, 4, "c0_srsconf3"  },
302
  {  6, 5, "c0_srsconf4"  },
303
  { 12, 1, "c0_intctl"    },
304
  { 12, 2, "c0_srsctl"    },
305
  { 12, 3, "c0_srsmap"    },
306
  { 15, 1, "c0_ebase"   },
307
  { 16, 1, "c0_config1"   },
308
  { 16, 2, "c0_config2"   },
309
  { 16, 3, "c0_config3"   },
310
  { 18, 1, "c0_watchlo,1" },
311
  { 18, 2, "c0_watchlo,2" },
312
  { 18, 3, "c0_watchlo,3" },
313
  { 18, 4, "c0_watchlo,4" },
314
  { 18, 5, "c0_watchlo,5" },
315
  { 18, 6, "c0_watchlo,6" },
316
  { 18, 7, "c0_watchlo,7" },
317
  { 19, 1, "c0_watchhi,1" },
318
  { 19, 2, "c0_watchhi,2" },
319
  { 19, 3, "c0_watchhi,3" },
320
  { 19, 4, "c0_watchhi,4" },
321
  { 19, 5, "c0_watchhi,5" },
322
  { 19, 6, "c0_watchhi,6" },
323
  { 19, 7, "c0_watchhi,7" },
324
  { 23, 1, "c0_tracecontrol"  },
325
  { 23, 2, "c0_tracecontrol2" },
326
  { 23, 3, "c0_usertracedata" },
327
  { 23, 4, "c0_tracebpc"  },
328
  { 25, 1, "c0_perfcnt,1" },
329
  { 25, 2, "c0_perfcnt,2" },
330
  { 25, 3, "c0_perfcnt,3" },
331
  { 25, 4, "c0_perfcnt,4" },
332
  { 25, 5, "c0_perfcnt,5" },
333
  { 25, 6, "c0_perfcnt,6" },
334
  { 25, 7, "c0_perfcnt,7" },
335
  { 27, 1, "c0_cacheerr,1"  },
336
  { 27, 2, "c0_cacheerr,2"  },
337
  { 27, 3, "c0_cacheerr,3"  },
338
  { 28, 1, "c0_datalo"    },
339
  { 28, 2, "c0_taglo1"    },
340
  { 28, 3, "c0_datalo1"   },
341
  { 28, 4, "c0_taglo2"    },
342
  { 28, 5, "c0_datalo2"   },
343
  { 28, 6, "c0_taglo3"    },
344
  { 28, 7, "c0_datalo3"   },
345
  { 29, 1, "c0_datahi"    },
346
  { 29, 2, "c0_taghi1"    },
347
  { 29, 3, "c0_datahi1"   },
348
  { 29, 4, "c0_taghi2"    },
349
  { 29, 5, "c0_datahi2"   },
350
  { 29, 6, "c0_taghi3"    },
351
  { 29, 7, "c0_datahi3"   },
352
};
353
354
/* SB-1: MIPS64 (mips_cp0_names_mips3264) with minor mods.  */
355
static const char * const mips_cp0_names_sb1[32] =
356
{
357
  "c0_index",     "c0_random",    "c0_entrylo0",  "c0_entrylo1",
358
  "c0_context",   "c0_pagemask",  "c0_wired",     "$7",
359
  "c0_badvaddr",  "c0_count",     "c0_entryhi",   "c0_compare",
360
  "c0_status",    "c0_cause",     "c0_epc",       "c0_prid",
361
  "c0_config",    "c0_lladdr",    "c0_watchlo",   "c0_watchhi",
362
  "c0_xcontext",  "$21",          "$22",          "c0_debug",
363
  "c0_depc",      "c0_perfcnt",   "c0_errctl",    "c0_cacheerr_i",
364
  "c0_taglo_i",   "c0_taghi_i",   "c0_errorepc",  "c0_desave",
365
};
366
367
static const struct mips_cp0sel_name mips_cp0sel_names_sb1[] =
368
{
369
  { 16, 1, "c0_config1"   },
370
  { 18, 1, "c0_watchlo,1" },
371
  { 19, 1, "c0_watchhi,1" },
372
  { 22, 0, "c0_perftrace" },
373
  { 23, 3, "c0_edebug"    },
374
  { 25, 1, "c0_perfcnt,1" },
375
  { 25, 2, "c0_perfcnt,2" },
376
  { 25, 3, "c0_perfcnt,3" },
377
  { 25, 4, "c0_perfcnt,4" },
378
  { 25, 5, "c0_perfcnt,5" },
379
  { 25, 6, "c0_perfcnt,6" },
380
  { 25, 7, "c0_perfcnt,7" },
381
  { 26, 1, "c0_buserr_pa" },
382
  { 27, 1, "c0_cacheerr_d"  },
383
  { 27, 3, "c0_cacheerr_d_pa" },
384
  { 28, 1, "c0_datalo_i"  },
385
  { 28, 2, "c0_taglo_d"   },
386
  { 28, 3, "c0_datalo_d"  },
387
  { 29, 1, "c0_datahi_i"  },
388
  { 29, 2, "c0_taghi_d"   },
389
  { 29, 3, "c0_datahi_d"  },
390
};
391
392
/* Xlr cop0 register names.  */
393
static const char * const mips_cp0_names_xlr[32] = {
394
  "c0_index",     "c0_random",    "c0_entrylo0",  "c0_entrylo1",
395
  "c0_context",   "c0_pagemask",  "c0_wired",     "$7",
396
  "c0_badvaddr",  "c0_count",     "c0_entryhi",   "c0_compare",
397
  "c0_status",    "c0_cause",     "c0_epc",       "c0_prid",
398
  "c0_config",    "c0_lladdr",    "c0_watchlo",   "c0_watchhi",
399
  "c0_xcontext",  "$21",          "$22",          "c0_debug",
400
  "c0_depc",      "c0_perfcnt",   "c0_errctl",    "c0_cacheerr_i",
401
  "c0_taglo_i",   "c0_taghi_i",   "c0_errorepc",  "c0_desave",
402
};
403
404
/* XLR's CP0 Select Registers.  */
405
406
static const struct mips_cp0sel_name mips_cp0sel_names_xlr[] = {
407
  {  9, 6, "c0_extintreq"       },
408
  {  9, 7, "c0_extintmask"      },
409
  { 15, 1, "c0_ebase"           },
410
  { 16, 1, "c0_config1"         },
411
  { 16, 2, "c0_config2"         },
412
  { 16, 3, "c0_config3"         },
413
  { 16, 7, "c0_procid2"         },
414
  { 18, 1, "c0_watchlo,1"       },
415
  { 18, 2, "c0_watchlo,2"       },
416
  { 18, 3, "c0_watchlo,3"       },
417
  { 18, 4, "c0_watchlo,4"       },
418
  { 18, 5, "c0_watchlo,5"       },
419
  { 18, 6, "c0_watchlo,6"       },
420
  { 18, 7, "c0_watchlo,7"       },
421
  { 19, 1, "c0_watchhi,1"       },
422
  { 19, 2, "c0_watchhi,2"       },
423
  { 19, 3, "c0_watchhi,3"       },
424
  { 19, 4, "c0_watchhi,4"       },
425
  { 19, 5, "c0_watchhi,5"       },
426
  { 19, 6, "c0_watchhi,6"       },
427
  { 19, 7, "c0_watchhi,7"       },
428
  { 25, 1, "c0_perfcnt,1"       },
429
  { 25, 2, "c0_perfcnt,2"       },
430
  { 25, 3, "c0_perfcnt,3"       },
431
  { 25, 4, "c0_perfcnt,4"       },
432
  { 25, 5, "c0_perfcnt,5"       },
433
  { 25, 6, "c0_perfcnt,6"       },
434
  { 25, 7, "c0_perfcnt,7"       },
435
  { 27, 1, "c0_cacheerr,1"      },
436
  { 27, 2, "c0_cacheerr,2"      },
437
  { 27, 3, "c0_cacheerr,3"      },
438
  { 28, 1, "c0_datalo"          },
439
  { 29, 1, "c0_datahi"          }
440
};
441
442
static const char * const mips_hwr_names_numeric[32] =
443
{
444
  "$0",   "$1",   "$2",   "$3",   "$4",   "$5",   "$6",   "$7",
445
  "$8",   "$9",   "$10",  "$11",  "$12",  "$13",  "$14",  "$15",
446
  "$16",  "$17",  "$18",  "$19",  "$20",  "$21",  "$22",  "$23",
447
  "$24",  "$25",  "$26",  "$27",  "$28",  "$29",  "$30",  "$31"
448
};
449
450
static const char * const mips_hwr_names_mips3264r2[32] =
451
{
452
  "hwr_cpunum",   "hwr_synci_step", "hwr_cc",     "hwr_ccres",
453
  "$4",          "$5",            "$6",           "$7",
454
  "$8",   "$9",   "$10",  "$11",  "$12",  "$13",  "$14",  "$15",
455
  "$16",  "$17",  "$18",  "$19",  "$20",  "$21",  "$22",  "$23",
456
  "$24",  "$25",  "$26",  "$27",  "$28",  "$29",  "$30",  "$31"
457
};
458
459
static const char * const msa_control_names[32] =
460
{
461
  "msa_ir", "msa_csr",  "msa_access", "msa_save",
462
  "msa_modify", "msa_request",  "msa_map",  "msa_unmap",
463
  "$8",   "$9",   "$10",  "$11",  "$12",  "$13",  "$14",  "$15",
464
  "$16",  "$17",  "$18",  "$19",  "$20",  "$21",  "$22",  "$23",
465
  "$24",  "$25",  "$26",  "$27",  "$28",  "$29",  "$30",  "$31"
466
};
467
468
struct mips_abi_choice
469
{
470
  const char * name;
471
  const char * const *gpr_names;
472
  const char * const *fpr_names;
473
};
474
475
struct mips_abi_choice mips_abi_choices[] =
476
{
477
  { "numeric", mips_gpr_names_numeric, mips_fpr_names_numeric },
478
  { "32", mips_gpr_names_oldabi, mips_fpr_names_32 },
479
  { "o64", mips_gpr_names_oldabi, mips_fpr_names_o64 },
480
  { "n32", mips_gpr_names_newabi, mips_fpr_names_n32 },
481
  { "64", mips_gpr_names_newabi, mips_fpr_names_64 },
482
  { "eabi32", mips_gpr_names_newabi, mips_fpr_names_eabi32 },
483
  { "eabi64", mips_gpr_names_newabi, mips_fpr_names_eabi64 },
484
};
485
486
struct mips_arch_choice
487
{
488
  const char *name;
489
  int bfd_mach_valid;
490
  unsigned long bfd_mach;
491
  int processor;
492
  int isa;
493
  int ase;
494
  const char * const *cp0_names;
495
  const struct mips_cp0sel_name *cp0sel_names;
496
  unsigned int cp0sel_names_len;
497
  const char * const *cp1_names;
498
  const char * const *hwr_names;
499
};
500
501
const struct mips_arch_choice mips_arch_choices[] =
502
{
503
  { "numeric",  0, 0, 0, 0, 0,
504
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_numeric,
505
    mips_hwr_names_numeric },
506
507
  { "r3000",  1, bfd_mach_mips3000, CPU_R3000, ISA_MIPS1, 0,
508
    mips_cp0_names_r3000, NULL, 0, mips_cp1_names_mips,
509
    mips_hwr_names_numeric },
510
  { "r3900",  1, bfd_mach_mips3900, CPU_R3900, ISA_MIPS1, 0,
511
    mips_cp0_names_r3900, NULL, 0, mips_cp1_names_numeric,
512
    mips_hwr_names_numeric },
513
  { "r4000",  1, bfd_mach_mips4000, CPU_R4000, ISA_MIPS3, 0,
514
    mips_cp0_names_r4000, NULL, 0, mips_cp1_names_mips,
515
    mips_hwr_names_numeric },
516
  { "r4010",  1, bfd_mach_mips4010, CPU_R4010, ISA_MIPS2, 0,
517
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
518
    mips_hwr_names_numeric },
519
  { "allegrex", 1, bfd_mach_mips_allegrex, CPU_ALLEGREX, ISA_MIPS2, 0,
520
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_numeric,
521
    mips_hwr_names_numeric },
522
  { "vr4100", 1, bfd_mach_mips4100, CPU_VR4100, ISA_MIPS3, 0,
523
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
524
    mips_hwr_names_numeric },
525
  { "vr4111", 1, bfd_mach_mips4111, CPU_R4111, ISA_MIPS3, 0,
526
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
527
    mips_hwr_names_numeric },
528
  { "vr4120", 1, bfd_mach_mips4120, CPU_VR4120, ISA_MIPS3, 0,
529
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
530
    mips_hwr_names_numeric },
531
  { "r4300",  1, bfd_mach_mips4300, CPU_R4300, ISA_MIPS3, 0,
532
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
533
    mips_hwr_names_numeric },
534
  { "r4400",  1, bfd_mach_mips4400, CPU_R4400, ISA_MIPS3, 0,
535
    mips_cp0_names_r4000, NULL, 0, mips_cp1_names_mips,
536
    mips_hwr_names_numeric },
537
  { "r4600",  1, bfd_mach_mips4600, CPU_R4600, ISA_MIPS3, 0,
538
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
539
    mips_hwr_names_numeric },
540
  { "r4650",  1, bfd_mach_mips4650, CPU_R4650, ISA_MIPS3, 0,
541
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
542
    mips_hwr_names_numeric },
543
  { "r5000",  1, bfd_mach_mips5000, CPU_R5000, ISA_MIPS4, 0,
544
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
545
    mips_hwr_names_numeric },
546
  { "vr5400", 1, bfd_mach_mips5400, CPU_VR5400, ISA_MIPS4, 0,
547
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
548
    mips_hwr_names_numeric },
549
  { "vr5500", 1, bfd_mach_mips5500, CPU_VR5500, ISA_MIPS4, 0,
550
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
551
    mips_hwr_names_numeric },
552
  { "r5900",  1, bfd_mach_mips5900, CPU_R5900, ISA_MIPS3, 0,
553
    mips_cp0_names_r5900, NULL, 0, mips_cp1_names_mips,
554
    mips_hwr_names_numeric },
555
  { "r6000",  1, bfd_mach_mips6000, CPU_R6000, ISA_MIPS2, 0,
556
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
557
    mips_hwr_names_numeric },
558
  { "rm7000", 1, bfd_mach_mips7000, CPU_RM7000, ISA_MIPS4, 0,
559
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
560
    mips_hwr_names_numeric },
561
  { "rm9000", 1, bfd_mach_mips7000, CPU_RM7000, ISA_MIPS4, 0,
562
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
563
    mips_hwr_names_numeric },
564
  { "r8000",  1, bfd_mach_mips8000, CPU_R8000, ISA_MIPS4, 0,
565
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
566
    mips_hwr_names_numeric },
567
  { "r10000", 1, bfd_mach_mips10000, CPU_R10000, ISA_MIPS4, 0,
568
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
569
    mips_hwr_names_numeric },
570
  { "r12000", 1, bfd_mach_mips12000, CPU_R12000, ISA_MIPS4, 0,
571
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
572
    mips_hwr_names_numeric },
573
  { "r14000", 1, bfd_mach_mips14000, CPU_R14000, ISA_MIPS4, 0,
574
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
575
    mips_hwr_names_numeric },
576
  { "r16000", 1, bfd_mach_mips16000, CPU_R16000, ISA_MIPS4, 0,
577
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
578
    mips_hwr_names_numeric },
579
  { "mips5",  1, bfd_mach_mips5, CPU_MIPS5, ISA_MIPS5, 0,
580
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips,
581
    mips_hwr_names_numeric },
582
583
  /* For stock MIPS32, disassemble all applicable MIPS-specified ASEs.
584
     Note that MIPS-3D and MDMX are not applicable to MIPS32.  (See
585
     _MIPS32 Architecture For Programmers Volume I: Introduction to the
586
     MIPS32 Architecture_ (MIPS Document Number MD00082, Revision 0.95),
587
     page 1.  */
588
  { "mips32", 1, bfd_mach_mipsisa32, CPU_MIPS32,
589
    ISA_MIPS32,  ASE_SMARTMIPS,
590
    mips_cp0_names_mips3264,
591
    mips_cp0sel_names_mips3264, ARRAY_SIZE (mips_cp0sel_names_mips3264),
592
    mips_cp1_names_mips3264, mips_hwr_names_numeric },
593
594
  { "mips32r2", 1, bfd_mach_mipsisa32r2, CPU_MIPS32R2,
595
    ISA_MIPS32R2,
596
    (ASE_SMARTMIPS | ASE_DSP | ASE_DSPR2 | ASE_EVA | ASE_MIPS3D
597
     | ASE_MT | ASE_MCU | ASE_VIRT | ASE_MSA | ASE_XPA),
598
    mips_cp0_names_mips3264r2,
599
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
600
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
601
602
  { "mips32r3", 1, bfd_mach_mipsisa32r3, CPU_MIPS32R3,
603
    ISA_MIPS32R3,
604
    (ASE_SMARTMIPS | ASE_DSP | ASE_DSPR2 | ASE_EVA | ASE_MIPS3D
605
     | ASE_MT | ASE_MCU | ASE_VIRT | ASE_MSA | ASE_XPA),
606
    mips_cp0_names_mips3264r2,
607
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
608
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
609
610
  { "mips32r5", 1, bfd_mach_mipsisa32r5, CPU_MIPS32R5,
611
    ISA_MIPS32R5,
612
    (ASE_SMARTMIPS | ASE_DSP | ASE_DSPR2 | ASE_EVA | ASE_MIPS3D
613
     | ASE_MT | ASE_MCU | ASE_VIRT | ASE_MSA | ASE_XPA),
614
    mips_cp0_names_mips3264r2,
615
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
616
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
617
618
  { "mips32r6", 1, bfd_mach_mipsisa32r6, CPU_MIPS32R6,
619
    ISA_MIPS32R6,
620
    (ASE_EVA | ASE_MSA | ASE_VIRT | ASE_XPA | ASE_MCU | ASE_MT | ASE_DSP
621
     | ASE_DSPR2 | ASE_DSPR3 | ASE_CRC | ASE_GINV),
622
    mips_cp0_names_mips3264r2,
623
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
624
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
625
626
  /* For stock MIPS64, disassemble all applicable MIPS-specified ASEs.  */
627
  { "mips64", 1, bfd_mach_mipsisa64, CPU_MIPS64,
628
    ISA_MIPS64,  ASE_MIPS3D | ASE_MDMX,
629
    mips_cp0_names_mips3264,
630
    mips_cp0sel_names_mips3264, ARRAY_SIZE (mips_cp0sel_names_mips3264),
631
    mips_cp1_names_mips3264, mips_hwr_names_numeric },
632
633
  { "mips64r2", 1, bfd_mach_mipsisa64r2, CPU_MIPS64R2,
634
    ISA_MIPS64R2,
635
    (ASE_MIPS3D | ASE_DSP | ASE_DSPR2 | ASE_DSP64 | ASE_EVA | ASE_MT
636
     | ASE_MCU | ASE_VIRT | ASE_VIRT64 | ASE_MSA | ASE_MSA64 | ASE_XPA),
637
    mips_cp0_names_mips3264r2,
638
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
639
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
640
641
  { "mips64r3", 1, bfd_mach_mipsisa64r3, CPU_MIPS64R3,
642
    ISA_MIPS64R3,
643
    (ASE_MIPS3D | ASE_DSP | ASE_DSPR2 | ASE_DSP64 | ASE_EVA | ASE_MT
644
     | ASE_MCU | ASE_VIRT | ASE_VIRT64 | ASE_MSA | ASE_MSA64 | ASE_XPA),
645
    mips_cp0_names_mips3264r2,
646
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
647
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
648
649
  { "mips64r5", 1, bfd_mach_mipsisa64r5, CPU_MIPS64R5,
650
    ISA_MIPS64R5,
651
    (ASE_MIPS3D | ASE_DSP | ASE_DSPR2 | ASE_DSP64 | ASE_EVA | ASE_MT
652
     | ASE_MCU | ASE_VIRT | ASE_VIRT64 | ASE_MSA | ASE_MSA64 | ASE_XPA),
653
    mips_cp0_names_mips3264r2,
654
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
655
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
656
657
  { "mips64r6", 1, bfd_mach_mipsisa64r6, CPU_MIPS64R6,
658
    ISA_MIPS64R6,
659
    (ASE_EVA | ASE_MSA | ASE_MSA64 | ASE_XPA | ASE_VIRT | ASE_VIRT64
660
     | ASE_MCU | ASE_MT | ASE_DSP | ASE_DSPR2 | ASE_DSPR3 | ASE_CRC
661
     | ASE_CRC64 | ASE_GINV),
662
    mips_cp0_names_mips3264r2,
663
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
664
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
665
666
  { "interaptiv-mr2", 1, bfd_mach_mips_interaptiv_mr2, CPU_INTERAPTIV_MR2,
667
    ISA_MIPS32R3,
668
    ASE_MT | ASE_EVA | ASE_DSP | ASE_DSPR2 | ASE_MIPS16E2 | ASE_MIPS16E2_MT,
669
    mips_cp0_names_mips3264r2,
670
    mips_cp0sel_names_mips3264r2, ARRAY_SIZE (mips_cp0sel_names_mips3264r2),
671
    mips_cp1_names_mips3264, mips_hwr_names_mips3264r2 },
672
673
  { "sb1",  1, bfd_mach_mips_sb1, CPU_SB1,
674
    ISA_MIPS64 | INSN_SB1,  ASE_MIPS3D,
675
    mips_cp0_names_sb1,
676
    mips_cp0sel_names_sb1, ARRAY_SIZE (mips_cp0sel_names_sb1),
677
    mips_cp1_names_mips3264, mips_hwr_names_numeric },
678
679
  { "loongson2e",   1, bfd_mach_mips_loongson_2e, CPU_LOONGSON_2E,
680
    ISA_MIPS3 | INSN_LOONGSON_2E, 0, mips_cp0_names_numeric,
681
    NULL, 0, mips_cp1_names_mips, mips_hwr_names_numeric },
682
683
  { "loongson2f",   1, bfd_mach_mips_loongson_2f, CPU_LOONGSON_2F,
684
    ISA_MIPS3 | INSN_LOONGSON_2F, ASE_LOONGSON_MMI, mips_cp0_names_numeric,
685
    NULL, 0, mips_cp1_names_mips, mips_hwr_names_numeric },
686
687
  /* The loongson3a is an alias of gs464 for compatibility */
688
  { "loongson3a",   1, bfd_mach_mips_gs464, CPU_GS464,
689
    ISA_MIPS64R2, ASE_LOONGSON_MMI | ASE_LOONGSON_CAM | ASE_LOONGSON_EXT,
690
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips3264,
691
    mips_hwr_names_numeric },
692
693
  { "gs464",   1, bfd_mach_mips_gs464, CPU_GS464,
694
    ISA_MIPS64R2, ASE_LOONGSON_MMI | ASE_LOONGSON_CAM | ASE_LOONGSON_EXT,
695
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips3264,
696
    mips_hwr_names_numeric },
697
698
  { "gs464e",   1, bfd_mach_mips_gs464e, CPU_GS464E,
699
    ISA_MIPS64R2, ASE_LOONGSON_MMI | ASE_LOONGSON_CAM | ASE_LOONGSON_EXT
700
    | ASE_LOONGSON_EXT2, mips_cp0_names_numeric, NULL, 0, mips_cp1_names_mips3264,
701
    mips_hwr_names_numeric },
702
703
  { "gs264e",   1, bfd_mach_mips_gs264e, CPU_GS264E,
704
    ISA_MIPS64R2, ASE_LOONGSON_MMI | ASE_LOONGSON_CAM | ASE_LOONGSON_EXT
705
    | ASE_LOONGSON_EXT2 | ASE_MSA | ASE_MSA64, mips_cp0_names_numeric, NULL,
706
    0, mips_cp1_names_mips3264, mips_hwr_names_numeric },
707
708
  { "octeon",   1, bfd_mach_mips_octeon, CPU_OCTEON,
709
    ISA_MIPS64R2 | INSN_OCTEON, 0, mips_cp0_names_numeric, NULL, 0,
710
    mips_cp1_names_mips3264, mips_hwr_names_numeric },
711
712
  { "octeon+",   1, bfd_mach_mips_octeonp, CPU_OCTEONP,
713
    ISA_MIPS64R2 | INSN_OCTEONP, 0, mips_cp0_names_numeric,
714
    NULL, 0, mips_cp1_names_mips3264, mips_hwr_names_numeric },
715
716
  { "octeon2",   1, bfd_mach_mips_octeon2, CPU_OCTEON2,
717
    ISA_MIPS64R2 | INSN_OCTEON2, 0, mips_cp0_names_numeric,
718
    NULL, 0, mips_cp1_names_mips3264, mips_hwr_names_numeric },
719
720
  { "octeon3",   1, bfd_mach_mips_octeon3, CPU_OCTEON3,
721
    ISA_MIPS64R5 | INSN_OCTEON3, ASE_VIRT | ASE_VIRT64,
722
    mips_cp0_names_numeric,
723
    NULL, 0, mips_cp1_names_mips3264, mips_hwr_names_numeric },
724
725
  { "xlr", 1, bfd_mach_mips_xlr, CPU_XLR,
726
    ISA_MIPS64 | INSN_XLR, 0,
727
    mips_cp0_names_xlr,
728
    mips_cp0sel_names_xlr, ARRAY_SIZE (mips_cp0sel_names_xlr),
729
    mips_cp1_names_mips3264, mips_hwr_names_numeric },
730
731
  /* XLP is mostly like XLR, with the prominent exception it is being
732
     MIPS64R2.  */
733
  { "xlp", 1, bfd_mach_mips_xlr, CPU_XLR,
734
    ISA_MIPS64R2 | INSN_XLR, 0,
735
    mips_cp0_names_xlr,
736
    mips_cp0sel_names_xlr, ARRAY_SIZE (mips_cp0sel_names_xlr),
737
    mips_cp1_names_mips3264, mips_hwr_names_numeric },
738
739
  /* This entry, mips16, is here only for ISA/processor selection; do
740
     not print its name.  */
741
  { "",   1, bfd_mach_mips16, CPU_MIPS16, ISA_MIPS64,
742
    ASE_MIPS16E2 | ASE_MIPS16E2_MT,
743
    mips_cp0_names_numeric, NULL, 0, mips_cp1_names_numeric,
744
    mips_hwr_names_numeric },
745
};
746
747
/* ISA and processor type to disassemble for, and register names to use.
748
   set_default_mips_dis_options and parse_mips_dis_options fill in these
749
   values.  */
750
static int mips_processor;
751
static int mips_isa;
752
static int mips_ase;
753
static int micromips_ase;
754
static const char * const *mips_gpr_names;
755
static const char * const *mips_fpr_names;
756
static const char * const *mips_cp0_names;
757
static const struct mips_cp0sel_name *mips_cp0sel_names;
758
static int mips_cp0sel_names_len;
759
static const char * const *mips_cp1_names;
760
static const char * const *mips_hwr_names;
761
762
/* Other options */
763
static int no_aliases;  /* If set disassemble as most general inst.  */
764

765
static const struct mips_abi_choice *
766
choose_abi_by_name (const char *name, unsigned int namelen)
767
16.6k
{
768
16.6k
  const struct mips_abi_choice *c;
769
16.6k
  unsigned int i;
770
771
129k
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_abi_choices) && c == NULL; i++)
772
112k
    if (strncmp (mips_abi_choices[i].name, name, namelen) == 0
773
4.50k
  && strlen (mips_abi_choices[i].name) == namelen)
774
992
      c = &mips_abi_choices[i];
775
776
16.6k
  return c;
777
16.6k
}
778
779
static const struct mips_arch_choice *
780
choose_arch_by_name (const char *name, unsigned int namelen)
781
38.6k
{
782
38.6k
  const struct mips_arch_choice *c = NULL;
783
38.6k
  unsigned int i;
784
785
1.98M
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_arch_choices) && c == NULL; i++)
786
1.94M
    if (strncmp (mips_arch_choices[i].name, name, namelen) == 0
787
9.13k
  && strlen (mips_arch_choices[i].name) == namelen)
788
6.43k
      c = &mips_arch_choices[i];
789
790
38.6k
  return c;
791
38.6k
}
792
793
static const struct mips_arch_choice *
794
choose_arch_by_number (unsigned long mach)
795
1.22M
{
796
1.22M
  static unsigned long hint_bfd_mach;
797
1.22M
  static const struct mips_arch_choice *hint_arch_choice;
798
1.22M
  const struct mips_arch_choice *c;
799
1.22M
  unsigned int i;
800
801
  /* We optimize this because even if the user specifies no
802
     flags, this will be done for every instruction!  */
803
1.22M
  if (hint_bfd_mach == mach
804
1.00M
      && hint_arch_choice != NULL
805
1.00M
      && hint_arch_choice->bfd_mach == hint_bfd_mach)
806
1.00M
    return hint_arch_choice;
807
808
11.4M
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_arch_choices) && c == NULL; i++)
809
11.1M
    {
810
11.1M
      if (mips_arch_choices[i].bfd_mach_valid
811
10.9M
    && mips_arch_choices[i].bfd_mach == mach)
812
2.24k
  {
813
2.24k
    c = &mips_arch_choices[i];
814
2.24k
    hint_bfd_mach = mach;
815
2.24k
    hint_arch_choice = c;
816
2.24k
  }
817
11.1M
    }
818
220k
  return c;
819
1.22M
}
820
821
/* Check if the object uses NewABI conventions.  */
822
823
static int
824
is_newabi (Elf_Internal_Ehdr *header)
825
474k
{
826
  /* There are no old-style ABIs which use 64-bit ELF.  */
827
474k
  if (header->e_ident[EI_CLASS] == ELFCLASS64)
828
228k
    return 1;
829
830
  /* If a 32-bit ELF file, n32 is a new-style ABI.  */
831
246k
  if ((header->e_flags & EF_MIPS_ABI2) != 0)
832
7.94k
    return 1;
833
834
238k
  return 0;
835
246k
}
836
837
/* Check if the object has microMIPS ASE code.  */
838
839
static int
840
is_micromips (Elf_Internal_Ehdr *header)
841
474k
{
842
474k
  if ((header->e_flags & EF_MIPS_ARCH_ASE_MICROMIPS) != 0)
843
283k
    return 1;
844
845
191k
  return 0;
846
474k
}
847
848
/* Convert ASE flags from .MIPS.abiflags to internal values.  */
849
850
static unsigned long
851
mips_convert_abiflags_ases (unsigned long afl_ases)
852
0
{
853
0
  unsigned long opcode_ases = 0;
854
855
0
  if (afl_ases & AFL_ASE_DSP)
856
0
    opcode_ases |= ASE_DSP;
857
0
  if (afl_ases & AFL_ASE_DSPR2)
858
0
    opcode_ases |= ASE_DSPR2;
859
0
  if (afl_ases & AFL_ASE_EVA)
860
0
    opcode_ases |= ASE_EVA;
861
0
  if (afl_ases & AFL_ASE_MCU)
862
0
    opcode_ases |= ASE_MCU;
863
0
  if (afl_ases & AFL_ASE_MDMX)
864
0
    opcode_ases |= ASE_MDMX;
865
0
  if (afl_ases & AFL_ASE_MIPS3D)
866
0
    opcode_ases |= ASE_MIPS3D;
867
0
  if (afl_ases & AFL_ASE_MT)
868
0
    opcode_ases |= ASE_MT;
869
0
  if (afl_ases & AFL_ASE_SMARTMIPS)
870
0
    opcode_ases |= ASE_SMARTMIPS;
871
0
  if (afl_ases & AFL_ASE_VIRT)
872
0
    opcode_ases |= ASE_VIRT;
873
0
  if (afl_ases & AFL_ASE_MSA)
874
0
    opcode_ases |= ASE_MSA;
875
0
  if (afl_ases & AFL_ASE_XPA)
876
0
    opcode_ases |= ASE_XPA;
877
0
  if (afl_ases & AFL_ASE_DSPR3)
878
0
    opcode_ases |= ASE_DSPR3;
879
0
  if (afl_ases & AFL_ASE_MIPS16E2)
880
0
    opcode_ases |= ASE_MIPS16E2;
881
0
  return opcode_ases;
882
0
}
883
884
/* Calculate combination ASE flags from regular ASE flags.  */
885
886
static unsigned long
887
mips_calculate_combination_ases (int opcode_isa, unsigned long opcode_ases)
888
1.42M
{
889
1.42M
  unsigned long combination_ases = 0;
890
891
1.42M
  if ((opcode_ases & (ASE_XPA | ASE_VIRT)) == (ASE_XPA | ASE_VIRT))
892
250k
    combination_ases |= ASE_XPA_VIRT;
893
1.42M
  if ((opcode_ases & (ASE_MIPS16E2 | ASE_MT)) == (ASE_MIPS16E2 | ASE_MT))
894
3.25k
    combination_ases |= ASE_MIPS16E2_MT;
895
1.42M
  if ((opcode_ases & ASE_EVA)
896
253k
      && ((opcode_isa & INSN_ISA_MASK) == ISA_MIPS64R6
897
174k
    || (opcode_isa & INSN_ISA_MASK) == ISA_MIPS32R6))
898
211k
    combination_ases |= ASE_EVA_R6;
899
1.42M
  return combination_ases;
900
1.42M
}
901
902
static void
903
set_default_mips_dis_options (struct disassemble_info *info)
904
1.22M
{
905
1.22M
  const struct mips_arch_choice *chosen_arch;
906
907
  /* Defaults: mipsIII/r3000 (?!), no microMIPS ASE (any compressed code
908
     is MIPS16 ASE) (o)32-style ("oldabi") GPR names, and numeric FPR,
909
     CP0 register, and HWR names.  */
910
1.22M
  mips_isa = ISA_MIPS3;
911
1.22M
  mips_processor = CPU_R3000;
912
1.22M
  micromips_ase = 0;
913
1.22M
  mips_ase = 0;
914
1.22M
  mips_gpr_names = mips_gpr_names_oldabi;
915
1.22M
  mips_fpr_names = mips_fpr_names_numeric;
916
1.22M
  mips_cp0_names = mips_cp0_names_numeric;
917
1.22M
  mips_cp0sel_names = NULL;
918
1.22M
  mips_cp0sel_names_len = 0;
919
1.22M
  mips_cp1_names = mips_cp1_names_numeric;
920
1.22M
  mips_hwr_names = mips_hwr_names_numeric;
921
1.22M
  no_aliases = 0;
922
923
  /* Set ISA, architecture, and cp0 register names as best we can.  */
924
#if ! SYMTAB_AVAILABLE
925
  /* This is running out on a target machine, not in a host tool.
926
     FIXME: Where does mips_target_info come from?  */
927
  target_processor = mips_target_info.processor;
928
  mips_isa = mips_target_info.isa;
929
  mips_ase = mips_target_info.ase;
930
#else
931
1.22M
  chosen_arch = choose_arch_by_number (info->mach);
932
1.22M
  if (chosen_arch != NULL)
933
1.00M
    {
934
1.00M
      mips_processor = chosen_arch->processor;
935
1.00M
      mips_isa = chosen_arch->isa;
936
1.00M
      mips_ase = chosen_arch->ase;
937
1.00M
      mips_cp0_names = chosen_arch->cp0_names;
938
1.00M
      mips_cp0sel_names = chosen_arch->cp0sel_names;
939
1.00M
      mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
940
1.00M
      mips_cp1_names = chosen_arch->cp1_names;
941
1.00M
      mips_hwr_names = chosen_arch->hwr_names;
942
1.00M
    }
943
944
  /* Update settings according to the ELF file header flags.  */
945
1.22M
  if (info->flavour == bfd_target_elf_flavour && info->section != NULL)
946
474k
    {
947
474k
      struct bfd *abfd = info->section->owner;
948
474k
      Elf_Internal_Ehdr *header = elf_elfheader (abfd);
949
474k
      Elf_Internal_ABIFlags_v0 *abiflags = NULL;
950
951
      /* We won't ever get here if !HAVE_BFD_MIPS_ELF_GET_ABIFLAGS,
952
   because we won't then have a MIPS/ELF BFD, however we need
953
   to guard against a link error in a `--enable-targets=...'
954
   configuration with a 32-bit host where the MIPS target is
955
   a secondary, or with MIPS/ECOFF configurations.  */
956
474k
#ifdef HAVE_BFD_MIPS_ELF_GET_ABIFLAGS
957
474k
      abiflags = bfd_mips_elf_get_abiflags (abfd);
958
474k
#endif
959
      /* If an ELF "newabi" binary, use the n32/(n)64 GPR names.  */
960
474k
      if (is_newabi (header))
961
236k
  mips_gpr_names = mips_gpr_names_newabi;
962
      /* If a microMIPS binary, then don't use MIPS16 bindings.  */
963
474k
      micromips_ase = is_micromips (header);
964
      /* OR in any extra ASE flags set in ELF file structures.  */
965
474k
      if (abiflags)
966
0
  mips_ase |= mips_convert_abiflags_ases (abiflags->ases);
967
474k
      else if (header->e_flags & EF_MIPS_ARCH_ASE_MDMX)
968
181k
  mips_ase |= ASE_MDMX;
969
474k
    }
970
1.22M
#endif
971
1.22M
  mips_ase |= mips_calculate_combination_ases (mips_isa, mips_ase);
972
1.22M
}
973
974
/* Parse an ASE disassembler option and set the corresponding global
975
   ASE flag(s).  Return TRUE if successful, FALSE otherwise.  */
976
977
static bool
978
parse_mips_ase_option (const char *option)
979
642k
{
980
642k
  if (startswith (option, "msa"))
981
187k
    {
982
187k
      mips_ase |= ASE_MSA;
983
187k
      if ((mips_isa & INSN_ISA_MASK) == ISA_MIPS64R2
984
187k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R3
985
187k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R5
986
185k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R6)
987
2.96k
    mips_ase |= ASE_MSA64;
988
187k
      return true;
989
187k
    }
990
991
454k
  if (startswith (option, "virt"))
992
1.24k
    {
993
1.24k
      mips_ase |= ASE_VIRT;
994
1.24k
      if (mips_isa & ISA_MIPS64R2
995
754
    || mips_isa & ISA_MIPS64R3
996
410
    || mips_isa & ISA_MIPS64R5
997
0
    || mips_isa & ISA_MIPS64R6)
998
1.24k
  mips_ase |= ASE_VIRT64;
999
1.24k
      return true;
1000
1.24k
    }
1001
1002
453k
  if (startswith (option, "xpa"))
1003
992
    {
1004
992
      mips_ase |= ASE_XPA;
1005
992
      return true;
1006
992
    }
1007
1008
452k
  if (startswith (option, "ginv"))
1009
2.40k
    {
1010
2.40k
      mips_ase |= ASE_GINV;
1011
2.40k
      return true;
1012
2.40k
    }
1013
1014
450k
  if (startswith (option, "loongson-mmi"))
1015
4.97k
    {
1016
4.97k
      mips_ase |= ASE_LOONGSON_MMI;
1017
4.97k
      return true;
1018
4.97k
    }
1019
1020
445k
  if (startswith (option, "loongson-cam"))
1021
710
    {
1022
710
      mips_ase |= ASE_LOONGSON_CAM;
1023
710
      return true;
1024
710
    }
1025
  
1026
  /* Put here for match ext2 frist */
1027
444k
  if (startswith (option, "loongson-ext2"))
1028
576
    {
1029
576
      mips_ase |= ASE_LOONGSON_EXT2;
1030
576
      return true;
1031
576
    }
1032
1033
443k
  if (startswith (option, "loongson-ext"))
1034
1.07k
    {
1035
1.07k
      mips_ase |= ASE_LOONGSON_EXT;
1036
1.07k
      return true;
1037
1.07k
    }
1038
1039
442k
  return false;
1040
443k
}
1041
1042
static void
1043
parse_mips_dis_option (const char *option, unsigned int len)
1044
654k
{
1045
654k
  unsigned int i, optionlen, vallen;
1046
654k
  const char *val;
1047
654k
  const struct mips_abi_choice *chosen_abi;
1048
654k
  const struct mips_arch_choice *chosen_arch;
1049
1050
  /* Try to match options that are simple flags */
1051
654k
  if (startswith (option, "no-aliases"))
1052
11.5k
    {
1053
11.5k
      no_aliases = 1;
1054
11.5k
      return;
1055
11.5k
    }
1056
1057
642k
  if (parse_mips_ase_option (option))
1058
199k
    {
1059
199k
      mips_ase |= mips_calculate_combination_ases (mips_isa, mips_ase);
1060
199k
      return;
1061
199k
    }
1062
1063
  /* Look for the = that delimits the end of the option name.  */
1064
4.73M
  for (i = 0; i < len; i++)
1065
4.42M
    if (option[i] == '=')
1066
131k
      break;
1067
1068
442k
  if (i == 0)    /* Invalid option: no name before '='.  */
1069
1.36k
    return;
1070
441k
  if (i == len)    /* Invalid option: no '='.  */
1071
310k
    return;
1072
130k
  if (i == (len - 1))  /* Invalid option: no value after '='.  */
1073
1.18k
    return;
1074
1075
129k
  optionlen = i;
1076
129k
  val = option + (optionlen + 1);
1077
129k
  vallen = len - (optionlen + 1);
1078
1079
129k
  if (strncmp ("gpr-names", option, optionlen) == 0
1080
1.27k
      && strlen ("gpr-names") == optionlen)
1081
1.00k
    {
1082
1.00k
      chosen_abi = choose_abi_by_name (val, vallen);
1083
1.00k
      if (chosen_abi != NULL)
1084
228
  mips_gpr_names = chosen_abi->gpr_names;
1085
1.00k
      return;
1086
1.00k
    }
1087
1088
128k
  if (strncmp ("fpr-names", option, optionlen) == 0
1089
2.24k
      && strlen ("fpr-names") == optionlen)
1090
1.95k
    {
1091
1.95k
      chosen_abi = choose_abi_by_name (val, vallen);
1092
1.95k
      if (chosen_abi != NULL)
1093
296
  mips_fpr_names = chosen_abi->fpr_names;
1094
1.95k
      return;
1095
1.95k
    }
1096
1097
126k
  if (strncmp ("cp0-names", option, optionlen) == 0
1098
6.43k
      && strlen ("cp0-names") == optionlen)
1099
6.07k
    {
1100
6.07k
      chosen_arch = choose_arch_by_name (val, vallen);
1101
6.07k
      if (chosen_arch != NULL)
1102
5.10k
  {
1103
5.10k
    mips_cp0_names = chosen_arch->cp0_names;
1104
5.10k
    mips_cp0sel_names = chosen_arch->cp0sel_names;
1105
5.10k
    mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
1106
5.10k
  }
1107
6.07k
      return;
1108
6.07k
    }
1109
1110
120k
  if (strncmp ("cp1-names", option, optionlen) == 0
1111
17.8k
      && strlen ("cp1-names") == optionlen)
1112
17.2k
    {
1113
17.2k
      chosen_arch = choose_arch_by_name (val, vallen);
1114
17.2k
      if (chosen_arch != NULL)
1115
508
  mips_cp1_names = chosen_arch->cp1_names;
1116
17.2k
      return;
1117
17.2k
    }
1118
1119
103k
  if (strncmp ("hwr-names", option, optionlen) == 0
1120
3.81k
      && strlen ("hwr-names") == optionlen)
1121
1.59k
    {
1122
1.59k
      chosen_arch = choose_arch_by_name (val, vallen);
1123
1.59k
      if (chosen_arch != NULL)
1124
212
  mips_hwr_names = chosen_arch->hwr_names;
1125
1.59k
      return;
1126
1.59k
    }
1127
1128
101k
  if (strncmp ("reg-names", option, optionlen) == 0
1129
29.5k
      && strlen ("reg-names") == optionlen)
1130
13.6k
    {
1131
      /* We check both ABI and ARCH here unconditionally, so
1132
   that "numeric" will do the desirable thing: select
1133
   numeric register names for all registers.  Other than
1134
   that, a given name probably won't match both.  */
1135
13.6k
      chosen_abi = choose_abi_by_name (val, vallen);
1136
13.6k
      if (chosen_abi != NULL)
1137
468
  {
1138
468
    mips_gpr_names = chosen_abi->gpr_names;
1139
468
    mips_fpr_names = chosen_abi->fpr_names;
1140
468
  }
1141
13.6k
      chosen_arch = choose_arch_by_name (val, vallen);
1142
13.6k
      if (chosen_arch != NULL)
1143
614
  {
1144
614
    mips_cp0_names = chosen_arch->cp0_names;
1145
614
    mips_cp0sel_names = chosen_arch->cp0sel_names;
1146
614
    mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
1147
614
    mips_cp1_names = chosen_arch->cp1_names;
1148
614
    mips_hwr_names = chosen_arch->hwr_names;
1149
614
  }
1150
13.6k
      return;
1151
13.6k
    }
1152
1153
  /* Invalid option.  */
1154
101k
}
1155
1156
static void
1157
parse_mips_dis_options (const char *options)
1158
1.22M
{
1159
1.22M
  const char *option_end;
1160
1161
1.22M
  if (options == NULL)
1162
968k
    return;
1163
1164
1.32M
  while (*options != '\0')
1165
1.06M
    {
1166
      /* Skip empty options.  */
1167
1.06M
      if (*options == ',')
1168
415k
  {
1169
415k
    options++;
1170
415k
    continue;
1171
415k
  }
1172
1173
      /* We know that *options is neither NUL or a comma.  */
1174
654k
      option_end = options + 1;
1175
6.97M
      while (*option_end != ',' && *option_end != '\0')
1176
6.32M
  option_end++;
1177
1178
654k
      parse_mips_dis_option (options, option_end - options);
1179
1180
      /* Go on to the next one.  If option_end points to a comma, it
1181
   will be skipped above.  */
1182
654k
      options = option_end;
1183
654k
    }
1184
258k
}
1185
1186
static const struct mips_cp0sel_name *
1187
lookup_mips_cp0sel_name (const struct mips_cp0sel_name *names,
1188
       unsigned int len,
1189
       unsigned int cp0reg,
1190
       unsigned int sel)
1191
2.30k
{
1192
2.30k
  unsigned int i;
1193
1194
44.5k
  for (i = 0; i < len; i++)
1195
42.8k
    if (names[i].cp0reg == cp0reg && names[i].sel == sel)
1196
599
      return &names[i];
1197
1.70k
  return NULL;
1198
2.30k
}
1199
1200
/* Print register REGNO, of type TYPE, for instruction OPCODE.  */
1201
1202
static void
1203
print_reg (struct disassemble_info *info, const struct mips_opcode *opcode,
1204
     enum mips_reg_operand_type type, int regno)
1205
1.53M
{
1206
1.53M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1207
1208
1.53M
  switch (type)
1209
1.53M
    {
1210
1.45M
    case OP_REG_GP:
1211
1.45M
      infprintf (info->stream, dis_style_register, "%s",
1212
1.45M
     mips_gpr_names[regno]);
1213
1.45M
      break;
1214
1215
23.7k
    case OP_REG_FP:
1216
23.7k
      infprintf (info->stream, dis_style_register, "%s",
1217
23.7k
     mips_fpr_names[regno]);
1218
23.7k
      break;
1219
1220
1.55k
    case OP_REG_CCC:
1221
1.55k
      if (opcode->pinfo & (FP_D | FP_S))
1222
1.13k
  infprintf (info->stream, dis_style_register, "$fcc%d", regno);
1223
418
      else
1224
418
  infprintf (info->stream, dis_style_register, "$cc%d", regno);
1225
1.55k
      break;
1226
1227
4.18k
    case OP_REG_VEC:
1228
4.18k
      if (opcode->membership & INSN_5400)
1229
816
  infprintf (info->stream, dis_style_register, "$f%d", regno);
1230
3.37k
      else
1231
3.37k
  infprintf (info->stream, dis_style_register, "$v%d", regno);
1232
4.18k
      break;
1233
1234
2.36k
    case OP_REG_ACC:
1235
2.36k
      infprintf (info->stream, dis_style_register, "$ac%d", regno);
1236
2.36k
      break;
1237
1238
18.1k
    case OP_REG_COPRO:
1239
18.1k
      if (opcode->name[strlen (opcode->name) - 1] == '0')
1240
2.41k
  infprintf (info->stream, dis_style_register, "%s", mips_cp0_names[regno]);
1241
15.7k
      else
1242
15.7k
  infprintf (info->stream, dis_style_register, "$%d", regno);
1243
18.1k
      break;
1244
1245
878
    case OP_REG_CONTROL:
1246
878
      if (opcode->name[strlen (opcode->name) - 1] == '1')
1247
398
  infprintf (info->stream, dis_style_register, "%s", mips_cp1_names[regno]);
1248
480
      else
1249
480
  infprintf (info->stream, dis_style_register, "$%d", regno);
1250
878
      break;
1251
1252
509
    case OP_REG_HW:
1253
509
      infprintf (info->stream, dis_style_register, "%s", mips_hwr_names[regno]);
1254
509
      break;
1255
1256
5.93k
    case OP_REG_VF:
1257
5.93k
      infprintf (info->stream, dis_style_register, "$vf%d", regno);
1258
5.93k
      break;
1259
1260
504
    case OP_REG_VI:
1261
504
      infprintf (info->stream, dis_style_register, "$vi%d", regno);
1262
504
      break;
1263
1264
390
    case OP_REG_R5900_I:
1265
390
      infprintf (info->stream, dis_style_register, "$I");
1266
390
      break;
1267
1268
304
    case OP_REG_R5900_Q:
1269
304
      infprintf (info->stream, dis_style_register, "$Q");
1270
304
      break;
1271
1272
196
    case OP_REG_R5900_R:
1273
196
      infprintf (info->stream, dis_style_register, "$R");
1274
196
      break;
1275
1276
396
    case OP_REG_R5900_ACC:
1277
396
      infprintf (info->stream, dis_style_register, "$ACC");
1278
396
      break;
1279
1280
18.3k
    case OP_REG_MSA:
1281
18.3k
      infprintf (info->stream, dis_style_register, "$w%d", regno);
1282
18.3k
      break;
1283
1284
399
    case OP_REG_MSA_CTRL:
1285
399
      infprintf (info->stream, dis_style_register, "%s",
1286
399
     msa_control_names[regno]);
1287
399
      break;
1288
1289
1.53M
    }
1290
1.53M
}
1291

1292
/* Used to track the state carried over from previous operands in
1293
   an instruction.  */
1294
struct mips_print_arg_state {
1295
  /* The value of the last OP_INT seen.  We only use this for OP_MSB,
1296
     where the value is known to be unsigned and small.  */
1297
  unsigned int last_int;
1298
1299
  /* The type and number of the last OP_REG seen.  We only use this for
1300
     OP_REPEAT_DEST_REG and OP_REPEAT_PREV_REG.  */
1301
  enum mips_reg_operand_type last_reg_type;
1302
  unsigned int last_regno;
1303
  unsigned int dest_regno;
1304
  unsigned int seen_dest;
1305
};
1306
1307
/* Initialize STATE for the start of an instruction.  */
1308
1309
static inline void
1310
init_print_arg_state (struct mips_print_arg_state *state)
1311
1.45M
{
1312
1.45M
  memset (state, 0, sizeof (*state));
1313
1.45M
}
1314
1315
/* Print OP_VU0_SUFFIX or OP_VU0_MATCH_SUFFIX operand OPERAND,
1316
   whose value is given by UVAL.  */
1317
1318
static void
1319
print_vu0_channel (struct disassemble_info *info,
1320
       const struct mips_operand *operand, unsigned int uval,
1321
       enum disassembler_style style)
1322
7.38k
{
1323
7.38k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1324
1325
7.38k
  if (operand->size == 4)
1326
5.48k
    infprintf (info->stream, style, "%s%s%s%s",
1327
5.48k
      uval & 8 ? "x" : "",
1328
5.48k
      uval & 4 ? "y" : "",
1329
5.48k
      uval & 2 ? "z" : "",
1330
5.48k
      uval & 1 ? "w" : "");
1331
1.89k
  else if (operand->size == 2)
1332
1.89k
    infprintf (info->stream, style, "%c", "xyzw"[uval]);
1333
0
  else
1334
0
    abort ();
1335
7.38k
}
1336
1337
/* Record information about a register operand.  */
1338
1339
static void
1340
mips_seen_register (struct mips_print_arg_state *state,
1341
        unsigned int regno,
1342
        enum mips_reg_operand_type reg_type)
1343
2.30M
{
1344
2.30M
  state->last_reg_type = reg_type;
1345
2.30M
  state->last_regno = regno;
1346
1347
2.30M
  if (!state->seen_dest)
1348
1.23M
    {
1349
1.23M
      state->seen_dest = 1;
1350
1.23M
      state->dest_regno = regno;
1351
1.23M
    }
1352
2.30M
}
1353
1354
/* Print SAVE/RESTORE instruction operands according to the argument
1355
   register mask AMASK, the number of static registers saved NSREG,
1356
   the $ra, $s0 and $s1 register specifiers RA, S0 and S1 respectively,
1357
   and the frame size FRAME_SIZE.  */
1358
1359
static void
1360
mips_print_save_restore (struct disassemble_info *info, unsigned int amask,
1361
       unsigned int nsreg, unsigned int ra,
1362
       unsigned int s0, unsigned int s1,
1363
       unsigned int frame_size)
1364
12.3k
{
1365
12.3k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1366
12.3k
  unsigned int nargs, nstatics, smask, i, j;
1367
12.3k
  void *is = info->stream;
1368
12.3k
  const char *sep;
1369
1370
12.3k
  if (amask == MIPS_SVRS_ALL_ARGS)
1371
367
    {
1372
367
      nargs = 4;
1373
367
      nstatics = 0;
1374
367
    }
1375
11.9k
  else if (amask == MIPS_SVRS_ALL_STATICS)
1376
2.32k
    {
1377
2.32k
      nargs = 0;
1378
2.32k
      nstatics = 4;
1379
2.32k
    }
1380
9.61k
  else
1381
9.61k
    {
1382
9.61k
      nargs = amask >> 2;
1383
9.61k
      nstatics = amask & 3;
1384
9.61k
    }
1385
1386
12.3k
  sep = "";
1387
12.3k
  if (nargs > 0)
1388
1.98k
    {
1389
1.98k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[4]);
1390
1.98k
      if (nargs > 1)
1391
1.03k
  infprintf (is, dis_style_register, "-%s", mips_gpr_names[4 + nargs - 1]);
1392
1.98k
      sep = ",";
1393
1.98k
    }
1394
1395
12.3k
  infprintf (is, dis_style_text, "%s", sep);
1396
12.3k
  infprintf (is, dis_style_immediate, "%d", frame_size);
1397
1398
12.3k
  if (ra)      /* $ra */
1399
9.20k
    {
1400
9.20k
      infprintf (is, dis_style_text, ",");
1401
9.20k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1402
9.20k
    }
1403
1404
12.3k
  smask = 0;
1405
12.3k
  if (s0)      /* $s0 */
1406
9.84k
    smask |= 1 << 0;
1407
12.3k
  if (s1)      /* $s1 */
1408
7.80k
    smask |= 1 << 1;
1409
12.3k
  if (nsreg > 0)    /* $s2-$s8 */
1410
2.45k
    smask |= ((1 << nsreg) - 1) << 2;
1411
1412
94.0k
  for (i = 0; i < 9; i++)
1413
81.7k
    if (smask & (1 << i))
1414
10.8k
      {
1415
10.8k
  infprintf (is, dis_style_text, ",");
1416
10.8k
  infprintf (is, dis_style_register, "%s",
1417
10.8k
       mips_gpr_names[i == 8 ? 30 : (16 + i)]);
1418
  /* Skip over string of set bits.  */
1419
29.4k
  for (j = i; smask & (2 << j); j++)
1420
18.5k
    continue;
1421
10.8k
  if (j > i)
1422
8.08k
    {
1423
8.08k
      infprintf (is, dis_style_text, "-");
1424
8.08k
      infprintf (is, dis_style_register, "%s",
1425
8.08k
           mips_gpr_names[j == 8 ? 30 : (16 + j)]);
1426
8.08k
    }
1427
10.8k
  i = j + 1;
1428
10.8k
      }
1429
  /* Statics $ax - $a3.  */
1430
12.3k
  if (nstatics == 1)
1431
1.04k
    {
1432
1.04k
      infprintf (is, dis_style_text, ",");
1433
1.04k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[7]);
1434
1.04k
    }
1435
11.2k
  else if (nstatics > 0)
1436
3.22k
    {
1437
3.22k
      infprintf (is, dis_style_text, ",");
1438
3.22k
      infprintf (is, dis_style_register, "%s",
1439
3.22k
     mips_gpr_names[7 - nstatics + 1]);
1440
3.22k
      infprintf (is, dis_style_text, "-");
1441
3.22k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[7]);
1442
3.22k
    }
1443
12.3k
}
1444
1445
1446
/* Print operand OPERAND of OPCODE, using STATE to track inter-operand state.
1447
   UVAL is the encoding of the operand (shifted into bit 0) and BASE_PC is
1448
   the base address for OP_PCREL operands.  */
1449
1450
static void
1451
print_insn_arg (struct disassemble_info *info,
1452
    struct mips_print_arg_state *state,
1453
    const struct mips_opcode *opcode,
1454
    const struct mips_operand *operand,
1455
    bfd_vma base_pc,
1456
    unsigned int uval)
1457
2.40M
{
1458
2.40M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1459
2.40M
  void *is = info->stream;
1460
1461
2.40M
  switch (operand->type)
1462
2.40M
    {
1463
689k
    case OP_INT:
1464
689k
      {
1465
689k
  const struct mips_int_operand *int_op;
1466
1467
689k
  int_op = (const struct mips_int_operand *) operand;
1468
689k
  uval = mips_decode_int_operand (int_op, uval);
1469
689k
  state->last_int = uval;
1470
689k
  if (int_op->print_hex)
1471
119k
    infprintf (is, dis_style_immediate, "0x%x", uval);
1472
570k
  else
1473
570k
    infprintf (is, dis_style_immediate, "%d", uval);
1474
689k
      }
1475
689k
      break;
1476
1477
8.77k
    case OP_MAPPED_INT:
1478
8.77k
      {
1479
8.77k
  const struct mips_mapped_int_operand *mint_op;
1480
1481
8.77k
  mint_op = (const struct mips_mapped_int_operand *) operand;
1482
8.77k
  uval = mint_op->int_map[uval];
1483
8.77k
  state->last_int = uval;
1484
8.77k
  if (mint_op->print_hex)
1485
4.75k
    infprintf (is, dis_style_immediate, "0x%x", uval);
1486
4.02k
  else
1487
4.02k
    infprintf (is, dis_style_immediate, "%d", uval);
1488
8.77k
      }
1489
8.77k
      break;
1490
1491
3.55k
    case OP_MSB:
1492
3.55k
      {
1493
3.55k
  const struct mips_msb_operand *msb_op;
1494
1495
3.55k
  msb_op = (const struct mips_msb_operand *) operand;
1496
3.55k
  uval += msb_op->bias;
1497
3.55k
  if (msb_op->add_lsb)
1498
1.95k
    uval -= state->last_int;
1499
3.55k
  infprintf (is, dis_style_immediate, "0x%x", uval);
1500
3.55k
      }
1501
3.55k
      break;
1502
1503
1.33M
    case OP_REG:
1504
1.48M
    case OP_OPTIONAL_REG:
1505
1.48M
      {
1506
1.48M
  const struct mips_reg_operand *reg_op;
1507
1508
1.48M
  reg_op = (const struct mips_reg_operand *) operand;
1509
1.48M
  uval = mips_decode_reg_operand (reg_op, uval);
1510
1.48M
  print_reg (info, opcode, reg_op->reg_type, uval);
1511
1512
1.48M
  mips_seen_register (state, uval, reg_op->reg_type);
1513
1.48M
      }
1514
1.48M
      break;
1515
1516
3.20k
    case OP_REG_PAIR:
1517
3.20k
      {
1518
3.20k
  const struct mips_reg_pair_operand *pair_op;
1519
1520
3.20k
  pair_op = (const struct mips_reg_pair_operand *) operand;
1521
3.20k
  print_reg (info, opcode, pair_op->reg_type,
1522
3.20k
       pair_op->reg1_map[uval]);
1523
3.20k
  infprintf (is, dis_style_text, ",");
1524
3.20k
  print_reg (info, opcode, pair_op->reg_type,
1525
3.20k
       pair_op->reg2_map[uval]);
1526
3.20k
      }
1527
3.20k
      break;
1528
1529
149k
    case OP_PCREL:
1530
149k
      {
1531
149k
  const struct mips_pcrel_operand *pcrel_op;
1532
1533
149k
  pcrel_op = (const struct mips_pcrel_operand *) operand;
1534
149k
  info->target = mips_decode_pcrel_operand (pcrel_op, base_pc, uval);
1535
1536
  /* For jumps and branches clear the ISA bit except for
1537
     the GDB disassembler.  */
1538
149k
  if (pcrel_op->include_isa_bit
1539
126k
      && info->flavour != bfd_target_unknown_flavour)
1540
109k
    info->target &= -2;
1541
1542
149k
  (*info->print_address_func) (info->target, info);
1543
149k
      }
1544
149k
      break;
1545
1546
270
    case OP_PERF_REG:
1547
270
      infprintf (is, dis_style_register, "%d", uval);
1548
270
      break;
1549
1550
5.20k
    case OP_ADDIUSP_INT:
1551
5.20k
      {
1552
5.20k
  int sval;
1553
1554
5.20k
  sval = mips_signed_operand (operand, uval) * 4;
1555
5.20k
  if (sval >= -8 && sval < 8)
1556
426
    sval ^= 0x400;
1557
5.20k
  infprintf (is, dis_style_immediate, "%d", sval);
1558
5.20k
  break;
1559
1.33M
      }
1560
1561
1.14k
    case OP_CLO_CLZ_DEST:
1562
1.14k
      {
1563
1.14k
  unsigned int reg1, reg2;
1564
1565
1.14k
  reg1 = uval & 31;
1566
1.14k
  reg2 = uval >> 5;
1567
  /* If one is zero use the other.  */
1568
1.14k
  if (reg1 == reg2 || reg2 == 0)
1569
492
    infprintf (is, dis_style_register, "%s", mips_gpr_names[reg1]);
1570
650
  else if (reg1 == 0)
1571
278
    infprintf (is, dis_style_register, "%s", mips_gpr_names[reg2]);
1572
372
  else
1573
372
    {
1574
      /* Bogus, result depends on processor.  */
1575
372
      infprintf (is, dis_style_register, "%s", mips_gpr_names[reg1]);
1576
372
      infprintf (is, dis_style_text, " or ");
1577
372
      infprintf (is, dis_style_register, "%s", mips_gpr_names[reg2]);
1578
372
    }
1579
1.14k
      }
1580
1.14k
      break;
1581
1582
673
    case OP_SAME_RS_RT:
1583
14.4k
    case OP_CHECK_PREV:
1584
29.2k
    case OP_NON_ZERO_REG:
1585
29.2k
      {
1586
29.2k
  print_reg (info, opcode, OP_REG_GP, uval & 31);
1587
29.2k
  mips_seen_register (state, uval, OP_REG_GP);
1588
29.2k
      }
1589
29.2k
      break;
1590
1591
5.48k
    case OP_LWM_SWM_LIST:
1592
5.48k
      if (operand->size == 2)
1593
1.48k
  {
1594
1.48k
    if (uval == 0)
1595
909
      {
1596
909
        infprintf (is, dis_style_register, "%s",
1597
909
       mips_gpr_names[16]);
1598
909
        infprintf (is, dis_style_text, ",");
1599
909
        infprintf (is, dis_style_register, "%s",
1600
909
       mips_gpr_names[31]);
1601
909
      }
1602
574
    else
1603
574
      {
1604
574
        infprintf (is, dis_style_register, "%s",
1605
574
       mips_gpr_names[16]);
1606
574
        infprintf (is, dis_style_text, "-");
1607
574
        infprintf (is, dis_style_register, "%s",
1608
574
       mips_gpr_names[16 + uval]);
1609
574
        infprintf (is, dis_style_text, ",");
1610
574
        infprintf (is, dis_style_register, "%s",
1611
574
       mips_gpr_names[31]);
1612
574
      }
1613
1.48k
  }
1614
3.99k
      else
1615
3.99k
  {
1616
3.99k
    int s_reg_encode;
1617
1618
3.99k
    s_reg_encode = uval & 0xf;
1619
3.99k
    if (s_reg_encode != 0)
1620
3.22k
      {
1621
3.22k
        if (s_reg_encode == 1)
1622
782
    infprintf (is, dis_style_register, "%s", mips_gpr_names[16]);
1623
2.44k
        else if (s_reg_encode < 9)
1624
1.38k
    {
1625
1.38k
      infprintf (is, dis_style_register, "%s",
1626
1.38k
           mips_gpr_names[16]);
1627
1.38k
      infprintf (is, dis_style_text, "-");
1628
1.38k
      infprintf (is, dis_style_register, "%s",
1629
1.38k
           mips_gpr_names[15 + s_reg_encode]);
1630
1.38k
    }
1631
1.05k
        else if (s_reg_encode == 9)
1632
657
    {
1633
657
      infprintf (is, dis_style_register, "%s",
1634
657
           mips_gpr_names[16]);
1635
657
      infprintf (is, dis_style_text, "-");
1636
657
      infprintf (is, dis_style_register, "%s",
1637
657
           mips_gpr_names[23]);
1638
657
      infprintf (is, dis_style_text, ",");
1639
657
      infprintf (is, dis_style_register, "%s",
1640
657
           mips_gpr_names[30]);
1641
657
    }
1642
401
        else
1643
401
    infprintf (is, dis_style_text, "UNKNOWN");
1644
3.22k
      }
1645
1646
3.99k
    if (uval & 0x10) /* For ra.  */
1647
2.74k
      {
1648
2.74k
        if (s_reg_encode == 0)
1649
730
    infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1650
2.01k
        else
1651
2.01k
    {
1652
2.01k
      infprintf (is, dis_style_text, ",");
1653
2.01k
      infprintf (is, dis_style_register, "%s",
1654
2.01k
           mips_gpr_names[31]);
1655
2.01k
    }
1656
2.74k
      }
1657
3.99k
  }
1658
5.48k
      break;
1659
1660
2.75k
    case OP_ENTRY_EXIT_LIST:
1661
2.75k
      {
1662
2.75k
  const char *sep;
1663
2.75k
  unsigned int amask, smask;
1664
1665
2.75k
  sep = "";
1666
2.75k
  amask = (uval >> 3) & 7;
1667
2.75k
  if (amask > 0 && amask < 5)
1668
1.46k
    {
1669
1.46k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[4]);
1670
1.46k
      if (amask > 1)
1671
457
        {
1672
457
    infprintf (is, dis_style_text, "-");
1673
457
    infprintf (is, dis_style_register, "%s",
1674
457
         mips_gpr_names[amask + 3]);
1675
457
        }
1676
1.46k
      sep = ",";
1677
1.46k
    }
1678
1679
2.75k
  smask = (uval >> 1) & 3;
1680
2.75k
  if (smask == 3)
1681
1.60k
    {
1682
1.60k
      infprintf (is, dis_style_text, "%s??", sep);
1683
1.60k
      sep = ",";
1684
1.60k
    }
1685
1.15k
  else if (smask > 0)
1686
766
    {
1687
766
      infprintf (is, dis_style_text, "%s", sep);
1688
766
      infprintf (is, dis_style_register, "%s", mips_gpr_names[16]);
1689
766
      if (smask > 1)
1690
299
        {
1691
299
    infprintf (is, dis_style_text, "-");
1692
299
    infprintf (is, dis_style_register, "%s",
1693
299
         mips_gpr_names[smask + 15]);
1694
299
        }
1695
766
      sep = ",";
1696
766
    }
1697
1698
2.75k
  if (uval & 1)
1699
2.09k
    {
1700
2.09k
      infprintf (is, dis_style_text, "%s", sep);
1701
2.09k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1702
2.09k
      sep = ",";
1703
2.09k
    }
1704
1705
2.75k
  if (amask == 5 || amask == 6)
1706
536
    {
1707
536
      infprintf (is, dis_style_text, "%s", sep);
1708
536
      infprintf (is, dis_style_register, "%s", mips_fpr_names[0]);
1709
536
      if (amask == 6)
1710
243
        {
1711
243
    infprintf (is, dis_style_text, "-");
1712
243
    infprintf (is, dis_style_register, "%s", mips_fpr_names[1]);
1713
243
        }
1714
536
    }
1715
2.75k
      }
1716
2.75k
      break;
1717
1718
0
    case OP_SAVE_RESTORE_LIST:
1719
      /* Should be handled by the caller due to complex behavior.  */
1720
0
      abort ();
1721
1722
1.97k
    case OP_MDMX_IMM_REG:
1723
1.97k
      {
1724
1.97k
  unsigned int vsel;
1725
1726
1.97k
  vsel = uval >> 5;
1727
1.97k
  uval &= 31;
1728
1.97k
  if ((vsel & 0x10) == 0)
1729
772
    {
1730
772
      int fmt;
1731
1732
772
      vsel &= 0x0f;
1733
2.14k
      for (fmt = 0; fmt < 3; fmt++, vsel >>= 1)
1734
1.80k
        if ((vsel & 1) == 0)
1735
430
    break;
1736
772
      print_reg (info, opcode, OP_REG_VEC, uval);
1737
772
      infprintf (is, dis_style_text, "[");
1738
772
      infprintf (is, dis_style_immediate, "%d", vsel >> 1);
1739
772
      infprintf (is, dis_style_text, "]");
1740
772
    }
1741
1.20k
  else if ((vsel & 0x08) == 0)
1742
890
    print_reg (info, opcode, OP_REG_VEC, uval);
1743
310
  else
1744
310
    infprintf (is, dis_style_immediate, "0x%x", uval);
1745
1.97k
      }
1746
1.97k
      break;
1747
1748
7.48k
    case OP_REPEAT_PREV_REG:
1749
7.48k
      print_reg (info, opcode, state->last_reg_type, state->last_regno);
1750
7.48k
      break;
1751
1752
778
    case OP_REPEAT_DEST_REG:
1753
778
      print_reg (info, opcode, state->last_reg_type, state->dest_regno);
1754
778
      break;
1755
1756
2.96k
    case OP_PC:
1757
2.96k
      infprintf (is, dis_style_register, "$pc");
1758
2.96k
      break;
1759
1760
1.02k
    case OP_REG28:
1761
1.02k
      print_reg (info, opcode, OP_REG_GP, 28);
1762
1.02k
      break;
1763
1764
722
    case OP_VU0_SUFFIX:
1765
5.49k
    case OP_VU0_MATCH_SUFFIX:
1766
5.49k
      print_vu0_channel (info, operand, uval, dis_style_register);
1767
5.49k
      break;
1768
1769
904
    case OP_IMM_INDEX:
1770
904
      infprintf (is, dis_style_text, "[");
1771
904
      infprintf (is, dis_style_immediate, "%d", uval);
1772
904
      infprintf (is, dis_style_text, "]");
1773
904
      break;
1774
1775
294
    case OP_REG_INDEX:
1776
294
      infprintf (is, dis_style_text, "[");
1777
294
      print_reg (info, opcode, OP_REG_GP, uval);
1778
294
      infprintf (is, dis_style_text, "]");
1779
294
      break;
1780
2.40M
    }
1781
2.40M
}
1782
1783
/* Validate the arguments for INSN, which is described by OPCODE.
1784
   Use DECODE_OPERAND to get the encoding of each operand.  */
1785
1786
static bool
1787
validate_insn_args (const struct mips_opcode *opcode,
1788
        const struct mips_operand *(*decode_operand) (const char *),
1789
        unsigned int insn)
1790
530k
{
1791
530k
  struct mips_print_arg_state state;
1792
530k
  const struct mips_operand *operand;
1793
530k
  const char *s;
1794
530k
  unsigned int uval;
1795
1796
530k
  init_print_arg_state (&state);
1797
2.81M
  for (s = opcode->args; *s; ++s)
1798
2.29M
    {
1799
2.29M
      switch (*s)
1800
2.29M
  {
1801
633k
  case ',':
1802
813k
  case '(':
1803
994k
  case ')':
1804
994k
    break;
1805
1806
220
  case '#':
1807
220
    ++s;
1808
220
    break;
1809
1810
1.30M
  default:
1811
1.30M
    operand = decode_operand (s);
1812
1813
1.30M
    if (operand)
1814
1.30M
      {
1815
1.30M
        uval = mips_extract_operand (operand, insn);
1816
1.30M
        switch (operand->type)
1817
1.30M
    {
1818
662k
    case OP_REG:
1819
791k
    case OP_OPTIONAL_REG:
1820
791k
      {
1821
791k
        const struct mips_reg_operand *reg_op;
1822
1823
791k
        reg_op = (const struct mips_reg_operand *) operand;
1824
791k
        uval = mips_decode_reg_operand (reg_op, uval);
1825
791k
        mips_seen_register (&state, uval, reg_op->reg_type);
1826
791k
      }
1827
791k
    break;
1828
1829
7.24k
    case OP_SAME_RS_RT:
1830
7.24k
      {
1831
7.24k
        unsigned int reg1, reg2;
1832
1833
7.24k
        reg1 = uval & 31;
1834
7.24k
        reg2 = uval >> 5;
1835
1836
7.24k
        if (reg1 != reg2 || reg1 == 0)
1837
6.57k
          return false;
1838
7.24k
      }
1839
673
    break;
1840
1841
19.6k
    case OP_CHECK_PREV:
1842
19.6k
      {
1843
19.6k
        const struct mips_check_prev_operand *prev_op;
1844
1845
19.6k
        prev_op = (const struct mips_check_prev_operand *) operand;
1846
1847
19.6k
        if (!prev_op->zero_ok && uval == 0)
1848
841
          return false;
1849
1850
18.8k
        if (((prev_op->less_than_ok && uval < state.last_regno)
1851
13.9k
      || (prev_op->greater_than_ok && uval > state.last_regno)
1852
6.15k
      || (prev_op->equal_ok && uval == state.last_regno)))
1853
13.7k
          break;
1854
1855
5.08k
        return false;
1856
18.8k
      }
1857
1858
17.0k
    case OP_NON_ZERO_REG:
1859
17.0k
      {
1860
17.0k
        if (uval == 0)
1861
1.63k
          return false;
1862
17.0k
      }
1863
15.3k
    break;
1864
1865
338k
    case OP_INT:
1866
347k
    case OP_MAPPED_INT:
1867
350k
    case OP_MSB:
1868
353k
    case OP_REG_PAIR:
1869
436k
    case OP_PCREL:
1870
436k
    case OP_PERF_REG:
1871
442k
    case OP_ADDIUSP_INT:
1872
443k
    case OP_CLO_CLZ_DEST:
1873
448k
    case OP_LWM_SWM_LIST:
1874
448k
    case OP_ENTRY_EXIT_LIST:
1875
450k
    case OP_MDMX_IMM_REG:
1876
458k
    case OP_REPEAT_PREV_REG:
1877
459k
    case OP_REPEAT_DEST_REG:
1878
461k
    case OP_PC:
1879
461k
    case OP_REG28:
1880
462k
    case OP_VU0_SUFFIX:
1881
466k
    case OP_VU0_MATCH_SUFFIX:
1882
467k
    case OP_IMM_INDEX:
1883
468k
    case OP_REG_INDEX:
1884
468k
    case OP_SAVE_RESTORE_LIST:
1885
468k
      break;
1886
1.30M
    }
1887
1.30M
      }
1888
1.29M
    if (*s == 'm' || *s == '+' || *s == '-')
1889
312k
      ++s;
1890
2.29M
  }
1891
2.29M
    }
1892
516k
  return true;
1893
530k
}
1894
1895
/* Print the arguments for INSN, which is described by OPCODE.
1896
   Use DECODE_OPERAND to get the encoding of each operand.  Use BASE_PC
1897
   as the base of OP_PCREL operands, adjusting by LENGTH if the OP_PCREL
1898
   operand is for a branch or jump.  */
1899
1900
static void
1901
print_insn_args (struct disassemble_info *info,
1902
     const struct mips_opcode *opcode,
1903
     const struct mips_operand *(*decode_operand) (const char *),
1904
     unsigned int insn, bfd_vma insn_pc, unsigned int length)
1905
470k
{
1906
470k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1907
470k
  void *is = info->stream;
1908
470k
  struct mips_print_arg_state state;
1909
470k
  const struct mips_operand *operand;
1910
470k
  const char *s;
1911
1912
470k
  init_print_arg_state (&state);
1913
2.73M
  for (s = opcode->args; *s; ++s)
1914
2.26M
    {
1915
2.26M
      switch (*s)
1916
2.26M
  {
1917
625k
  case ',':
1918
806k
  case '(':
1919
986k
  case ')':
1920
986k
    infprintf (is, dis_style_text, "%c", *s);
1921
986k
    break;
1922
1923
220
  case '#':
1924
220
    ++s;
1925
220
    infprintf (is, dis_style_text, "%c%c", *s, *s);
1926
220
    break;
1927
1928
1.28M
  default:
1929
1.28M
    operand = decode_operand (s);
1930
1.28M
    if (!operand)
1931
0
      {
1932
        /* xgettext:c-format */
1933
0
        infprintf (is, dis_style_text,
1934
0
       _("# internal error, undefined operand in `%s %s'"),
1935
0
       opcode->name, opcode->args);
1936
0
        return;
1937
0
      }
1938
1939
1.28M
    if (operand->type == OP_SAVE_RESTORE_LIST)
1940
293
      {
1941
        /* Handle this case here because of the complex behavior.  */
1942
293
        unsigned int amask = (insn >> 15) & 0xf;
1943
293
        unsigned int nsreg = (insn >> 23) & 0x7;
1944
293
        unsigned int ra = insn & 0x1000;      /* $ra */
1945
293
        unsigned int s0 = insn & 0x800;     /* $s0 */
1946
293
        unsigned int s1 = insn & 0x400;     /* $s1 */
1947
293
        unsigned int frame_size = (((insn >> 15) & 0xf0)
1948
293
           | ((insn >> 6) & 0x0f)) * 8;
1949
293
        mips_print_save_restore (info, amask, nsreg, ra, s0, s1,
1950
293
               frame_size);
1951
293
      }
1952
1.28M
    else if (operand->type == OP_REG
1953
657k
       && s[1] == ','
1954
340k
       && (s[2] == 'H' || s[2] == 'J')
1955
1.67k
       && opcode->name[strlen (opcode->name) - 1] == '0')
1956
1.45k
      {
1957
        /* Coprocessor register 0 with sel field.  */
1958
1.45k
        const struct mips_cp0sel_name *n;
1959
1.45k
        unsigned int reg, sel;
1960
1961
1.45k
        reg = mips_extract_operand (operand, insn);
1962
1.45k
        s += 2;
1963
1.45k
        operand = decode_operand (s);
1964
1.45k
        sel = mips_extract_operand (operand, insn);
1965
1966
        /* CP0 register including 'sel' code for mftc0, to be
1967
     printed textually if known.  If not known, print both
1968
     CP0 register name and sel numerically since CP0 register
1969
     with sel 0 may have a name unrelated to register being
1970
     printed.  */
1971
1.45k
        n = lookup_mips_cp0sel_name (mips_cp0sel_names,
1972
1.45k
             mips_cp0sel_names_len,
1973
1.45k
             reg, sel);
1974
1.45k
        if (n != NULL)
1975
351
    infprintf (is, dis_style_register, "%s", n->name);
1976
1.10k
        else
1977
1.10k
    {
1978
1.10k
      infprintf (is, dis_style_register, "$%d", reg);
1979
1.10k
      infprintf (is, dis_style_text, ",");
1980
1.10k
      infprintf (is, dis_style_immediate, "%d", sel);
1981
1.10k
    }
1982
1.45k
      }
1983
1.28M
    else
1984
1.28M
      {
1985
1.28M
        bfd_vma base_pc = insn_pc;
1986
1987
        /* Adjust the PC relative base so that branch/jump insns use
1988
     the following PC as the base but genuinely PC relative
1989
     operands use the current PC.  */
1990
1.28M
        if (operand->type == OP_PCREL)
1991
83.1k
    {
1992
83.1k
      const struct mips_pcrel_operand *pcrel_op;
1993
1994
83.1k
      pcrel_op = (const struct mips_pcrel_operand *) operand;
1995
      /* The include_isa_bit flag is sufficient to distinguish
1996
         branch/jump from other PC relative operands.  */
1997
83.1k
      if (pcrel_op->include_isa_bit)
1998
82.1k
        base_pc += length;
1999
83.1k
    }
2000
2001
1.28M
        print_insn_arg (info, &state, opcode, operand, base_pc,
2002
1.28M
            mips_extract_operand (operand, insn));
2003
1.28M
      }
2004
1.28M
    if (*s == 'm' || *s == '+' || *s == '-')
2005
312k
      ++s;
2006
1.28M
    break;
2007
2.26M
  }
2008
2.26M
    }
2009
470k
}
2010

2011
/* Print the mips instruction at address MEMADDR in debugged memory,
2012
   on using INFO.  Returns length of the instruction, in bytes, which is
2013
   always INSNLEN.  BIGENDIAN must be 1 if this is big-endian code, 0 if
2014
   this is little-endian code.  */
2015
2016
static int
2017
print_insn_mips (bfd_vma memaddr,
2018
     int word,
2019
     struct disassemble_info *info)
2020
445k
{
2021
445k
#define GET_OP(insn, field)     \
2022
557k
  (((insn) >> OP_SH_##field) & OP_MASK_##field)
2023
445k
  static const struct mips_opcode *mips_hash[OP_MASK_OP + 1];
2024
445k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2025
445k
  const struct mips_opcode *op;
2026
445k
  static bool init = 0;
2027
445k
  void *is = info->stream;
2028
2029
  /* Build a hash table to shorten the search time.  */
2030
445k
  if (! init)
2031
3
    {
2032
3
      unsigned int i;
2033
2034
195
      for (i = 0; i <= OP_MASK_OP; i++)
2035
192
  {
2036
126k
    for (op = mips_opcodes; op < &mips_opcodes[NUMOPCODES]; op++)
2037
126k
      {
2038
126k
        if (op->pinfo == INSN_MACRO
2039
111k
      || (no_aliases && (op->pinfo2 & INSN2_ALIAS)))
2040
14.6k
    continue;
2041
111k
        if (i == GET_OP (op->match, OP))
2042
192
    {
2043
192
      mips_hash[i] = op;
2044
192
      break;
2045
192
    }
2046
111k
      }
2047
192
  }
2048
2049
3
      init = 1;
2050
3
    }
2051
2052
445k
  info->bytes_per_chunk = INSNLEN;
2053
445k
  info->display_endian = info->endian;
2054
445k
  info->insn_info_valid = 1;
2055
445k
  info->branch_delay_insns = 0;
2056
445k
  info->data_size = 0;
2057
445k
  info->insn_type = dis_nonbranch;
2058
445k
  info->target = 0;
2059
445k
  info->target2 = 0;
2060
2061
445k
  op = mips_hash[GET_OP (word, OP)];
2062
445k
  if (op != NULL)
2063
445k
    {
2064
570M
      for (; op < &mips_opcodes[NUMOPCODES]; op++)
2065
569M
  {
2066
569M
    if (op->pinfo != INSN_MACRO
2067
522M
        && !(no_aliases && (op->pinfo2 & INSN2_ALIAS))
2068
522M
        && (word & op->mask) == op->match)
2069
559k
      {
2070
        /* We always disassemble the jalx instruction, except for MIPS r6.  */
2071
559k
        if (!opcode_is_member (op, mips_isa, mips_ase, mips_processor)
2072
265k
     && (strcmp (op->name, "jalx")
2073
3.36k
         || (mips_isa & INSN_ISA_MASK) == ISA_MIPS32R6
2074
1.68k
         || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R6))
2075
265k
    continue;
2076
2077
        /* Figure out instruction type and branch delay information.  */
2078
294k
        if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0)
2079
19.4k
          {
2080
19.4k
      if ((op->pinfo & (INSN_WRITE_GPR_31 | INSN_WRITE_1)) != 0)
2081
11.7k
        info->insn_type = dis_jsr;
2082
7.71k
      else
2083
7.71k
        info->insn_type = dis_branch;
2084
19.4k
      info->branch_delay_insns = 1;
2085
19.4k
    }
2086
275k
        else if ((op->pinfo & (INSN_COND_BRANCH_DELAY
2087
275k
             | INSN_COND_BRANCH_LIKELY)) != 0)
2088
19.7k
    {
2089
19.7k
      if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2090
466
        info->insn_type = dis_condjsr;
2091
19.3k
      else
2092
19.3k
        info->insn_type = dis_condbranch;
2093
19.7k
      info->branch_delay_insns = 1;
2094
19.7k
    }
2095
255k
        else if ((op->pinfo & (INSN_STORE_MEMORY
2096
255k
             | INSN_LOAD_MEMORY)) != 0)
2097
84.0k
    info->insn_type = dis_dref;
2098
2099
294k
        if (!validate_insn_args (op, decode_mips_operand, word))
2100
14.1k
    continue;
2101
2102
280k
        infprintf (is, dis_style_mnemonic, "%s", op->name);
2103
280k
        if (op->pinfo2 & INSN2_VU0_CHANNEL_SUFFIX)
2104
1.88k
    {
2105
1.88k
      unsigned int uval;
2106
2107
1.88k
      infprintf (is, dis_style_mnemonic, ".");
2108
1.88k
      uval = mips_extract_operand (&mips_vu0_channel_mask, word);
2109
1.88k
      print_vu0_channel (info, &mips_vu0_channel_mask, uval,
2110
1.88k
             dis_style_mnemonic);
2111
1.88k
    }
2112
2113
280k
        if (op->args[0])
2114
248k
    {
2115
248k
      infprintf (is, dis_style_text, "\t");
2116
248k
      print_insn_args (info, op, decode_mips_operand, word,
2117
248k
           memaddr, 4);
2118
248k
    }
2119
2120
280k
        return INSNLEN;
2121
294k
      }
2122
569M
  }
2123
445k
    }
2124
165k
#undef GET_OP
2125
2126
  /* Handle undefined instructions.  */
2127
165k
  info->insn_type = dis_noninsn;
2128
165k
  infprintf (is, dis_style_assembler_directive, ".word");
2129
165k
  infprintf (is, dis_style_text, "\t");
2130
165k
  infprintf (is, dis_style_immediate, "0x%x", word);
2131
165k
  return INSNLEN;
2132
445k
}
2133

2134
/* Disassemble an operand for a mips16 instruction.  */
2135
2136
static void
2137
print_mips16_insn_arg (struct disassemble_info *info,
2138
           struct mips_print_arg_state *state,
2139
           const struct mips_opcode *opcode,
2140
           char type, bfd_vma memaddr,
2141
           unsigned insn, bool use_extend,
2142
           unsigned extend, bool is_offset)
2143
1.93M
{
2144
1.93M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2145
1.93M
  void *is = info->stream;
2146
1.93M
  const struct mips_operand *operand, *ext_operand;
2147
1.93M
  unsigned short ext_size;
2148
1.93M
  unsigned int uval;
2149
1.93M
  bfd_vma baseaddr;
2150
2151
1.93M
  if (!use_extend)
2152
1.77M
    extend = 0;
2153
2154
1.93M
  switch (type)
2155
1.93M
    {
2156
571k
    case ',':
2157
683k
    case '(':
2158
795k
    case ')':
2159
795k
      infprintf (is, dis_style_text, "%c", type);
2160
795k
      break;
2161
2162
1.13M
    default:
2163
1.13M
      operand = decode_mips16_operand (type, false);
2164
1.13M
      if (!operand)
2165
0
  {
2166
    /* xgettext:c-format */
2167
0
    infprintf (is, dis_style_text, _("# internal error, undefined operand in `%s %s'"),
2168
0
         opcode->name, opcode->args);
2169
0
    return;
2170
0
  }
2171
2172
1.13M
      if (operand->type == OP_SAVE_RESTORE_LIST)
2173
12.0k
  {
2174
    /* Handle this case here because of the complex interaction
2175
       with the EXTEND opcode.  */
2176
12.0k
    unsigned int amask = extend & 0xf;
2177
12.0k
    unsigned int nsreg = (extend >> 8) & 0x7;
2178
12.0k
    unsigned int ra = insn & 0x40;      /* $ra */
2179
12.0k
    unsigned int s0 = insn & 0x20;      /* $s0 */
2180
12.0k
    unsigned int s1 = insn & 0x10;      /* $s1 */
2181
12.0k
    unsigned int frame_size = ((extend & 0xf0) | (insn & 0x0f)) * 8;
2182
12.0k
    if (frame_size == 0 && !use_extend)
2183
2.89k
      frame_size = 128;
2184
12.0k
    mips_print_save_restore (info, amask, nsreg, ra, s0, s1, frame_size);
2185
12.0k
    break;
2186
12.0k
  }
2187
2188
1.12M
      if (is_offset && operand->type == OP_INT)
2189
111k
  {
2190
111k
    const struct mips_int_operand *int_op;
2191
2192
111k
    int_op = (const struct mips_int_operand *) operand;
2193
111k
    info->insn_type = dis_dref;
2194
111k
    info->data_size = 1 << int_op->shift;
2195
111k
  }
2196
2197
1.12M
      ext_size = 0;
2198
1.12M
      if (use_extend)
2199
88.0k
  {
2200
88.0k
    ext_operand = decode_mips16_operand (type, true);
2201
88.0k
    if (ext_operand != operand
2202
61.1k
        || (operand->type == OP_INT && operand->lsb == 0
2203
3.61k
      && mips_opcode_32bit_p (opcode)))
2204
30.4k
      {
2205
30.4k
        ext_size = ext_operand->size;
2206
30.4k
        operand = ext_operand;
2207
30.4k
      }
2208
88.0k
  }
2209
1.12M
      if (operand->size == 26)
2210
5.71k
  uval = ((extend & 0x1f) << 21) | ((extend & 0x3e0) << 11) | insn;
2211
1.11M
      else if (ext_size == 16 || ext_size == 9)
2212
27.0k
  uval = ((extend & 0x1f) << 11) | (extend & 0x7e0) | (insn & 0x1f);
2213
1.09M
      else if (ext_size == 15)
2214
1.04k
  uval = ((extend & 0xf) << 11) | (extend & 0x7f0) | (insn & 0xf);
2215
1.09M
      else if (ext_size == 6)
2216
1.44k
  uval = ((extend >> 6) & 0x1f) | (extend & 0x20);
2217
1.08M
      else
2218
1.08M
  uval = mips_extract_operand (operand, (extend << 16) | insn);
2219
1.12M
      if (ext_size == 9)
2220
2.52k
  uval &= (1U << ext_size) - 1;
2221
2222
1.12M
      baseaddr = memaddr + 2;
2223
1.12M
      if (operand->type == OP_PCREL)
2224
65.8k
  {
2225
65.8k
    const struct mips_pcrel_operand *pcrel_op;
2226
2227
65.8k
    pcrel_op = (const struct mips_pcrel_operand *) operand;
2228
65.8k
    if (!pcrel_op->include_isa_bit && use_extend)
2229
2.27k
      baseaddr = memaddr - 2;
2230
63.6k
    else if (!pcrel_op->include_isa_bit)
2231
19.5k
      {
2232
19.5k
        bfd_byte buffer[2];
2233
2234
        /* If this instruction is in the delay slot of a JAL/JALX
2235
     instruction, the base address is the address of the
2236
     JAL/JALX instruction.  If it is in the delay slot of
2237
     a JR/JALR instruction, the base address is the address
2238
     of the JR/JALR instruction.  This test is unreliable:
2239
     we have no way of knowing whether the previous word is
2240
     instruction or data.  */
2241
19.5k
        if (info->read_memory_func (memaddr - 4, buffer, 2, info) == 0
2242
7.05k
      && (((info->endian == BFD_ENDIAN_BIG
2243
7.05k
      ? bfd_getb16 (buffer)
2244
7.05k
      : bfd_getl16 (buffer))
2245
7.05k
           & 0xf800) == 0x1800))
2246
159
    baseaddr = memaddr - 4;
2247
19.4k
        else if (info->read_memory_func (memaddr - 2, buffer, 2,
2248
19.4k
                 info) == 0
2249
6.89k
           && (((info->endian == BFD_ENDIAN_BIG
2250
6.89k
           ? bfd_getb16 (buffer)
2251
6.89k
           : bfd_getl16 (buffer))
2252
6.89k
          & 0xf89f) == 0xe800)
2253
55
           && (((info->endian == BFD_ENDIAN_BIG
2254
55
           ? bfd_getb16 (buffer)
2255
55
           : bfd_getl16 (buffer))
2256
55
          & 0x0060) != 0x0060))
2257
18
    baseaddr = memaddr - 2;
2258
19.4k
        else
2259
19.4k
    baseaddr = memaddr;
2260
19.5k
      }
2261
65.8k
  }
2262
2263
1.12M
      print_insn_arg (info, state, opcode, operand, baseaddr + 1, uval);
2264
1.12M
      break;
2265
1.93M
    }
2266
1.93M
}
2267
2268
2269
/* Check if the given address is the last word of a MIPS16 PLT entry.
2270
   This word is data and depending on the value it may interfere with
2271
   disassembly of further PLT entries.  We make use of the fact PLT
2272
   symbols are marked BSF_SYNTHETIC.  */
2273
static bool
2274
is_mips16_plt_tail (struct disassemble_info *info, bfd_vma addr)
2275
479k
{
2276
479k
  if (info->symbols
2277
52
      && info->symbols[0]
2278
52
      && (info->symbols[0]->flags & BSF_SYNTHETIC)
2279
0
      && addr == bfd_asymbol_value (info->symbols[0]) + 12)
2280
0
    return true;
2281
2282
479k
  return false;
2283
479k
}
2284
2285
/* Whether none, a 32-bit or a 16-bit instruction match has been done.  */
2286
2287
enum match_kind
2288
{
2289
  MATCH_NONE,
2290
  MATCH_FULL,
2291
  MATCH_SHORT
2292
};
2293
2294
/* Disassemble mips16 instructions.  */
2295
2296
static int
2297
print_insn_mips16 (bfd_vma memaddr, struct disassemble_info *info)
2298
479k
{
2299
479k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2300
479k
  int status;
2301
479k
  bfd_byte buffer[4];
2302
479k
  const struct mips_opcode *op, *opend;
2303
479k
  struct mips_print_arg_state state;
2304
479k
  void *is = info->stream;
2305
479k
  bool have_second;
2306
479k
  bool extend_only;
2307
479k
  unsigned int second;
2308
479k
  unsigned int first;
2309
479k
  unsigned int full;
2310
2311
479k
  info->bytes_per_chunk = 2;
2312
479k
  info->display_endian = info->endian;
2313
479k
  info->insn_info_valid = 1;
2314
479k
  info->branch_delay_insns = 0;
2315
479k
  info->data_size = 0;
2316
479k
  info->target = 0;
2317
479k
  info->target2 = 0;
2318
2319
479k
#define GET_OP(insn, field) \
2320
479k
  (((insn) >> MIPS16OP_SH_##field) & MIPS16OP_MASK_##field)
2321
  /* Decode PLT entry's GOT slot address word.  */
2322
479k
  if (is_mips16_plt_tail (info, memaddr))
2323
0
    {
2324
0
      info->insn_type = dis_noninsn;
2325
0
      status = (*info->read_memory_func) (memaddr, buffer, 4, info);
2326
0
      if (status == 0)
2327
0
  {
2328
0
    unsigned int gotslot;
2329
2330
0
    if (info->endian == BFD_ENDIAN_BIG)
2331
0
      gotslot = bfd_getb32 (buffer);
2332
0
    else
2333
0
      gotslot = bfd_getl32 (buffer);
2334
0
    infprintf (is, dis_style_assembler_directive, ".word");
2335
0
    infprintf (is, dis_style_text, "\t");
2336
0
    infprintf (is, dis_style_immediate, "0x%x", gotslot);
2337
2338
0
    return 4;
2339
0
  }
2340
0
    }
2341
479k
  else
2342
479k
    {
2343
479k
      info->insn_type = dis_nonbranch;
2344
479k
      status = (*info->read_memory_func) (memaddr, buffer, 2, info);
2345
479k
    }
2346
479k
  if (status != 0)
2347
1.16k
    {
2348
1.16k
      (*info->memory_error_func) (status, memaddr, info);
2349
1.16k
      return -1;
2350
1.16k
    }
2351
2352
477k
  extend_only = false;
2353
2354
477k
  if (info->endian == BFD_ENDIAN_BIG)
2355
104k
    first = bfd_getb16 (buffer);
2356
373k
  else
2357
373k
    first = bfd_getl16 (buffer);
2358
2359
477k
  status = (*info->read_memory_func) (memaddr + 2, buffer, 2, info);
2360
477k
  if (status == 0)
2361
476k
    {
2362
476k
      have_second = true;
2363
476k
      if (info->endian == BFD_ENDIAN_BIG)
2364
104k
  second = bfd_getb16 (buffer);
2365
372k
      else
2366
372k
  second = bfd_getl16 (buffer);
2367
476k
      full = (first << 16) | second;
2368
476k
    }
2369
1.16k
  else
2370
1.16k
    {
2371
1.16k
      have_second = false;
2372
1.16k
      second = 0;
2373
1.16k
      full = first;
2374
1.16k
    }
2375
2376
  /* FIXME: Should probably use a hash table on the major opcode here.  */
2377
2378
477k
  opend = mips16_opcodes + bfd_mips16_num_opcodes;
2379
49.0M
  for (op = mips16_opcodes; op < opend; op++)
2380
49.0M
    {
2381
49.0M
      enum match_kind match;
2382
2383
49.0M
      if (!opcode_is_member (op, mips_isa, mips_ase, mips_processor))
2384
7.39M
  continue;
2385
2386
41.6M
      if (op->pinfo == INSN_MACRO
2387
32.7M
    || (no_aliases && (op->pinfo2 & INSN2_ALIAS)))
2388
8.90M
  match = MATCH_NONE;
2389
32.7M
      else if (mips_opcode_32bit_p (op))
2390
6.74M
  {
2391
6.74M
    if (have_second
2392
6.71M
        && (full & op->mask) == op->match)
2393
14.1k
      match = MATCH_FULL;
2394
6.72M
    else
2395
6.72M
      match = MATCH_NONE;
2396
6.74M
  }
2397
25.9M
      else if ((first & op->mask) == op->match)
2398
411k
  {
2399
411k
    match = MATCH_SHORT;
2400
411k
    second = 0;
2401
411k
    full = first;
2402
411k
  }
2403
25.5M
      else if ((first & 0xf800) == 0xf000
2404
4.52M
         && have_second
2405
4.51M
         && !extend_only
2406
4.27M
         && (second & op->mask) == op->match)
2407
36.6k
  {
2408
36.6k
    if (op->pinfo2 & INSN2_SHORT_ONLY)
2409
7.45k
      {
2410
7.45k
        match = MATCH_NONE;
2411
7.45k
        extend_only = true;
2412
7.45k
      }
2413
29.1k
    else
2414
29.1k
      match = MATCH_FULL;
2415
36.6k
  }
2416
25.5M
      else
2417
25.5M
  match = MATCH_NONE;
2418
2419
41.6M
      if (match != MATCH_NONE)
2420
454k
  {
2421
454k
    const char *s;
2422
2423
454k
    infprintf (is, dis_style_mnemonic, "%s", op->name);
2424
454k
    if (op->args[0] != '\0')
2425
453k
      infprintf (is, dis_style_text, "\t");
2426
2427
454k
    init_print_arg_state (&state);
2428
2.38M
    for (s = op->args; *s != '\0'; s++)
2429
1.93M
      {
2430
1.93M
        if (*s == ','
2431
574k
      && s[1] == 'w'
2432
15.0k
      && GET_OP (full, RX) == GET_OP (full, RY))
2433
1.75k
    {
2434
      /* Skip the register and the comma.  */
2435
1.75k
      ++s;
2436
1.75k
      continue;
2437
1.75k
    }
2438
1.93M
        if (*s == ','
2439
572k
      && s[1] == 'v'
2440
7.12k
      && GET_OP (full, RZ) == GET_OP (full, RX))
2441
1.04k
    {
2442
      /* Skip the register and the comma.  */
2443
1.04k
      ++s;
2444
1.04k
      continue;
2445
1.04k
    }
2446
1.93M
        if (s[0] == 'N'
2447
1.15k
      && s[1] == ','
2448
845
      && s[2] == 'O'
2449
845
      && op->name[strlen (op->name) - 1] == '0')
2450
845
    {
2451
      /* Coprocessor register 0 with sel field.  */
2452
845
      const struct mips_cp0sel_name *n;
2453
845
      const struct mips_operand *operand;
2454
845
      unsigned int reg, sel;
2455
2456
845
      operand = decode_mips16_operand (*s, true);
2457
845
      reg = mips_extract_operand (operand, (first << 16) | second);
2458
845
      s += 2;
2459
845
      operand = decode_mips16_operand (*s, true);
2460
845
      sel = mips_extract_operand (operand, (first << 16) | second);
2461
2462
      /* CP0 register including 'sel' code for mftc0, to be
2463
         printed textually if known.  If not known, print both
2464
         CP0 register name and sel numerically since CP0 register
2465
         with sel 0 may have a name unrelated to register being
2466
         printed.  */
2467
845
      n = lookup_mips_cp0sel_name (mips_cp0sel_names,
2468
845
                 mips_cp0sel_names_len,
2469
845
                 reg, sel);
2470
845
      if (n != NULL)
2471
248
        infprintf (is, dis_style_register, "%s", n->name);
2472
597
      else
2473
597
        {
2474
597
          infprintf (is, dis_style_register, "$%d", reg);
2475
597
          infprintf (is, dis_style_text, ",");
2476
597
          infprintf (is, dis_style_immediate, "%d", sel);
2477
597
        }
2478
845
    }
2479
1.93M
        else
2480
1.93M
    switch (match)
2481
1.93M
      {
2482
153k
        case MATCH_FULL:
2483
153k
          print_mips16_insn_arg (info, &state, op, *s, memaddr + 2,
2484
153k
               second, true, first, s[1] == '(');
2485
153k
          break;
2486
1.77M
        case MATCH_SHORT:
2487
1.77M
          print_mips16_insn_arg (info, &state, op, *s, memaddr,
2488
1.77M
               first, false, 0, s[1] == '(');
2489
1.77M
          break;
2490
0
        case MATCH_NONE:  /* Stop the compiler complaining.  */
2491
0
          break;
2492
1.93M
      }
2493
1.93M
      }
2494
2495
    /* Figure out branch instruction type and delay slot information.  */
2496
454k
    if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0)
2497
6.79k
      info->branch_delay_insns = 1;
2498
454k
    if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0
2499
447k
        || (op->pinfo2 & INSN2_UNCOND_BRANCH) != 0)
2500
21.4k
      {
2501
21.4k
        if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2502
5.77k
    info->insn_type = dis_jsr;
2503
15.6k
        else
2504
15.6k
    info->insn_type = dis_branch;
2505
21.4k
      }
2506
432k
    else if ((op->pinfo2 & INSN2_COND_BRANCH) != 0)
2507
23.8k
      info->insn_type = dis_condbranch;
2508
2509
454k
    return match == MATCH_FULL ? 4 : 2;
2510
454k
  }
2511
41.6M
    }
2512
23.5k
#undef GET_OP
2513
2514
23.5k
  infprintf (is, dis_style_assembler_directive, ".short");
2515
23.5k
  infprintf (is, dis_style_text, "\t");
2516
23.5k
  infprintf (is, dis_style_immediate, "0x%x", first);
2517
23.5k
  info->insn_type = dis_noninsn;
2518
2519
23.5k
  return 2;
2520
477k
}
2521
2522
/* Disassemble microMIPS instructions.  */
2523
2524
static int
2525
print_insn_micromips (bfd_vma memaddr, struct disassemble_info *info)
2526
300k
{
2527
300k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2528
300k
  const struct mips_opcode *op, *opend;
2529
300k
  void *is = info->stream;
2530
300k
  bfd_byte buffer[2];
2531
300k
  unsigned int higher;
2532
300k
  unsigned int length;
2533
300k
  int status;
2534
300k
  unsigned int insn;
2535
2536
300k
  info->bytes_per_chunk = 2;
2537
300k
  info->display_endian = info->endian;
2538
300k
  info->insn_info_valid = 1;
2539
300k
  info->branch_delay_insns = 0;
2540
300k
  info->data_size = 0;
2541
300k
  info->insn_type = dis_nonbranch;
2542
300k
  info->target = 0;
2543
300k
  info->target2 = 0;
2544
2545
300k
  status = (*info->read_memory_func) (memaddr, buffer, 2, info);
2546
300k
  if (status != 0)
2547
330
    {
2548
330
      (*info->memory_error_func) (status, memaddr, info);
2549
330
      return -1;
2550
330
    }
2551
2552
300k
  length = 2;
2553
2554
300k
  if (info->endian == BFD_ENDIAN_BIG)
2555
32.2k
    insn = bfd_getb16 (buffer);
2556
267k
  else
2557
267k
    insn = bfd_getl16 (buffer);
2558
2559
300k
  if ((insn & 0x1c00) == 0x0000 || (insn & 0x1000) == 0x1000)
2560
208k
    {
2561
      /* This is a 32-bit microMIPS instruction.  */
2562
208k
      higher = insn;
2563
2564
208k
      status = (*info->read_memory_func) (memaddr + 2, buffer, 2, info);
2565
208k
      if (status != 0)
2566
204
  {
2567
204
    infprintf (is, dis_style_text, "micromips 0x%x", higher);
2568
204
    (*info->memory_error_func) (status, memaddr + 2, info);
2569
204
    return -1;
2570
204
  }
2571
2572
207k
      if (info->endian == BFD_ENDIAN_BIG)
2573
24.5k
  insn = bfd_getb16 (buffer);
2574
183k
      else
2575
183k
  insn = bfd_getl16 (buffer);
2576
2577
207k
      insn = insn | (higher << 16);
2578
2579
207k
      length += 2;
2580
207k
    }
2581
2582
  /* FIXME: Should probably use a hash table on the major opcode here.  */
2583
2584
299k
  opend = micromips_opcodes + bfd_micromips_num_opcodes;
2585
209M
  for (op = micromips_opcodes; op < opend; op++)
2586
209M
    {
2587
209M
      if (op->pinfo != INSN_MACRO
2588
168M
    && !(no_aliases && (op->pinfo2 & INSN2_ALIAS))
2589
168M
    && (insn & op->mask) == op->match
2590
302k
    && ((length == 2 && (op->mask & 0xffff0000) == 0)
2591
225k
        || (length == 4 && (op->mask & 0xffff0000) != 0)))
2592
235k
  {
2593
235k
    if (!validate_insn_args (op, decode_micromips_operand, insn))
2594
0
      continue;
2595
2596
235k
    infprintf (is, dis_style_mnemonic, "%s", op->name);
2597
2598
235k
    if (op->args[0])
2599
221k
      {
2600
221k
        infprintf (is, dis_style_text, "\t");
2601
221k
        print_insn_args (info, op, decode_micromips_operand, insn,
2602
221k
             memaddr + 1, length);
2603
221k
      }
2604
2605
    /* Figure out instruction type and branch delay information.  */
2606
235k
    if ((op->pinfo
2607
235k
         & (INSN_UNCOND_BRANCH_DELAY | INSN_COND_BRANCH_DELAY)) != 0)
2608
23.2k
      info->branch_delay_insns = 1;
2609
235k
    if (((op->pinfo & INSN_UNCOND_BRANCH_DELAY)
2610
235k
         | (op->pinfo2 & INSN2_UNCOND_BRANCH)) != 0)
2611
15.3k
      {
2612
15.3k
        if ((op->pinfo & (INSN_WRITE_GPR_31 | INSN_WRITE_1)) != 0)
2613
10.1k
    info->insn_type = dis_jsr;
2614
5.21k
        else
2615
5.21k
    info->insn_type = dis_branch;
2616
15.3k
      }
2617
220k
    else if (((op->pinfo & INSN_COND_BRANCH_DELAY)
2618
220k
        | (op->pinfo2 & INSN2_COND_BRANCH)) != 0)
2619
8.08k
      {
2620
8.08k
        if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2621
521
    info->insn_type = dis_condjsr;
2622
7.56k
        else
2623
7.56k
    info->insn_type = dis_condbranch;
2624
8.08k
      }
2625
212k
    else if ((op->pinfo
2626
212k
        & (INSN_STORE_MEMORY | INSN_LOAD_MEMORY)) != 0)
2627
93.0k
      info->insn_type = dis_dref;
2628
2629
235k
    return length;
2630
235k
  }
2631
209M
    }
2632
2633
64.0k
  infprintf (is, dis_style_assembler_directive, ".short");
2634
64.0k
  infprintf (is, dis_style_text, "\t");
2635
64.0k
  if (length != 2)
2636
49.6k
    {
2637
49.6k
      infprintf (is, dis_style_immediate, "0x%x", (insn >> 16) & 0xffff);
2638
49.6k
      infprintf (is, dis_style_text, ", ");
2639
49.6k
    }
2640
64.0k
  infprintf (is, dis_style_immediate, "0x%x", (insn & 0xffff));
2641
2642
64.0k
  info->insn_type = dis_noninsn;
2643
2644
64.0k
  return length;
2645
299k
}
2646
2647
/* Return 1 if a symbol associated with the location being disassembled
2648
   indicates a compressed mode, either MIPS16 or microMIPS, according to
2649
   MICROMIPS_P.  We iterate over all the symbols at the address being
2650
   considered assuming if at least one of them indicates code compression,
2651
   then such code has been genuinely produced here (other symbols could
2652
   have been derived from function symbols defined elsewhere or could
2653
   define data).  Otherwise, return 0.  */
2654
2655
static bool
2656
is_compressed_mode_p (struct disassemble_info *info, bool micromips_p)
2657
895k
{
2658
895k
  int i;
2659
895k
  int l;
2660
2661
895k
  for (i = info->symtab_pos, l = i + info->num_symbols; i < l; i++)
2662
164
    if (((info->symtab[i])->flags & BSF_SYNTHETIC) != 0
2663
0
  && ((!micromips_p
2664
0
       && ELF_ST_IS_MIPS16 ((*info->symbols)->udata.i))
2665
0
      || (micromips_p
2666
0
    && ELF_ST_IS_MICROMIPS ((*info->symbols)->udata.i))))
2667
0
      return 1;
2668
164
    else if (bfd_asymbol_flavour (info->symtab[i]) == bfd_target_elf_flavour
2669
164
        && info->symtab[i]->section == info->section)
2670
134
      {
2671
134
  elf_symbol_type *symbol = (elf_symbol_type *) info->symtab[i];
2672
134
  if ((!micromips_p
2673
67
       && ELF_ST_IS_MIPS16 (symbol->internal_elf_sym.st_other))
2674
82
      || (micromips_p
2675
67
    && ELF_ST_IS_MICROMIPS (symbol->internal_elf_sym.st_other)))
2676
52
    return 1;
2677
134
      }
2678
2679
895k
  return 0;
2680
895k
}
2681
2682
/* In an environment where we do not know the symbol type of the
2683
   instruction we are forced to assume that the low order bit of the
2684
   instructions' address may mark it as a mips16 instruction.  If we
2685
   are single stepping, or the pc is within the disassembled function,
2686
   this works.  Otherwise, we need a clue.  Sometimes.  */
2687
2688
static int
2689
_print_insn_mips (bfd_vma memaddr,
2690
      struct disassemble_info *info,
2691
      enum bfd_endian endianness)
2692
1.22M
{
2693
1.22M
  bfd_byte buffer[INSNLEN];
2694
1.22M
  int status;
2695
2696
1.22M
  set_default_mips_dis_options (info);
2697
1.22M
  parse_mips_dis_options (info->disassembler_options);
2698
2699
1.22M
  if (info->mach == bfd_mach_mips16)
2700
308k
    return print_insn_mips16 (memaddr, info);
2701
918k
  if (info->mach == bfd_mach_mips_micromips)
2702
192k
    return print_insn_micromips (memaddr, info);
2703
2704
726k
#if 1
2705
  /* FIXME: If odd address, this is CLEARLY a compressed instruction.  */
2706
  /* Only a few tools will work this way.  */
2707
726k
  if (memaddr & 0x01)
2708
278k
    {
2709
278k
      if (micromips_ase)
2710
108k
  return print_insn_micromips (memaddr, info);
2711
170k
      else
2712
170k
  return print_insn_mips16 (memaddr, info);
2713
278k
    }
2714
447k
#endif
2715
2716
447k
#if SYMTAB_AVAILABLE
2717
447k
  if (is_compressed_mode_p (info, true))
2718
0
    return print_insn_micromips (memaddr, info);
2719
447k
  if (is_compressed_mode_p (info, false))
2720
52
    return print_insn_mips16 (memaddr, info);
2721
447k
#endif
2722
2723
447k
  status = (*info->read_memory_func) (memaddr, buffer, INSNLEN, info);
2724
447k
  if (status == 0)
2725
445k
    {
2726
445k
      int insn;
2727
2728
445k
      if (endianness == BFD_ENDIAN_BIG)
2729
144k
  insn = bfd_getb32 (buffer);
2730
301k
      else
2731
301k
  insn = bfd_getl32 (buffer);
2732
2733
445k
      return print_insn_mips (memaddr, insn, info);
2734
445k
    }
2735
1.75k
  else
2736
1.75k
    {
2737
1.75k
      (*info->memory_error_func) (status, memaddr, info);
2738
1.75k
      return -1;
2739
1.75k
    }
2740
447k
}
2741
2742
int
2743
print_insn_big_mips (bfd_vma memaddr, struct disassemble_info *info)
2744
370k
{
2745
370k
  return _print_insn_mips (memaddr, info, BFD_ENDIAN_BIG);
2746
370k
}
2747
2748
int
2749
print_insn_little_mips (bfd_vma memaddr, struct disassemble_info *info)
2750
856k
{
2751
856k
  return _print_insn_mips (memaddr, info, BFD_ENDIAN_LITTLE);
2752
856k
}
2753

2754
/* Indices into option argument vector for options accepting an argument.
2755
   Use MIPS_OPTION_ARG_NONE for options accepting no argument.  */
2756
typedef enum
2757
{
2758
  MIPS_OPTION_ARG_NONE = -1,
2759
  MIPS_OPTION_ARG_ABI,
2760
  MIPS_OPTION_ARG_ARCH,
2761
  MIPS_OPTION_ARG_SIZE
2762
} mips_option_arg_t;
2763
2764
/* Valid MIPS disassembler options.  */
2765
static struct
2766
{
2767
  const char *name;
2768
  const char *description;
2769
  mips_option_arg_t arg;
2770
} mips_options[] =
2771
{
2772
  { "no-aliases", N_("Use canonical instruction forms.\n"),
2773
      MIPS_OPTION_ARG_NONE },
2774
  { "msa",        N_("Recognize MSA instructions.\n"),
2775
      MIPS_OPTION_ARG_NONE },
2776
  { "virt",       N_("Recognize the virtualization ASE instructions.\n"),
2777
      MIPS_OPTION_ARG_NONE },
2778
  { "xpa",        N_("Recognize the eXtended Physical Address (XPA) ASE\n\
2779
                  instructions.\n"),
2780
      MIPS_OPTION_ARG_NONE },
2781
  { "ginv",       N_("Recognize the Global INValidate (GINV) ASE "
2782
         "instructions.\n"),
2783
      MIPS_OPTION_ARG_NONE },
2784
  { "loongson-mmi",
2785
      N_("Recognize the Loongson MultiMedia extensions "
2786
         "Instructions (MMI) ASE instructions.\n"),
2787
      MIPS_OPTION_ARG_NONE },
2788
  { "loongson-cam",
2789
      N_("Recognize the Loongson Content Address Memory (CAM) "
2790
         " instructions.\n"),
2791
      MIPS_OPTION_ARG_NONE },
2792
  { "loongson-ext",
2793
      N_("Recognize the Loongson EXTensions (EXT) "
2794
         " instructions.\n"),
2795
      MIPS_OPTION_ARG_NONE },
2796
  { "loongson-ext2",
2797
      N_("Recognize the Loongson EXTensions R2 (EXT2) "
2798
         " instructions.\n"),
2799
      MIPS_OPTION_ARG_NONE },
2800
  { "gpr-names=", N_("Print GPR names according to specified ABI.\n\
2801
                  Default: based on binary being disassembled.\n"),
2802
      MIPS_OPTION_ARG_ABI },
2803
  { "fpr-names=", N_("Print FPR names according to specified ABI.\n\
2804
                  Default: numeric.\n"),
2805
      MIPS_OPTION_ARG_ABI },
2806
  { "cp0-names=", N_("Print CP0 register names according to specified "
2807
         "architecture.\n\
2808
                  Default: based on binary being disassembled.\n"),
2809
      MIPS_OPTION_ARG_ARCH },
2810
  { "hwr-names=", N_("Print HWR names according to specified architecture.\n\
2811
                  Default: based on binary being disassembled.\n"),
2812
      MIPS_OPTION_ARG_ARCH },
2813
  { "reg-names=", N_("Print GPR and FPR names according to specified ABI.\n"),
2814
      MIPS_OPTION_ARG_ABI },
2815
  { "reg-names=", N_("Print CP0 register and HWR names according to "
2816
         "specified\n\
2817
                  architecture."),
2818
      MIPS_OPTION_ARG_ARCH }
2819
};
2820
2821
/* Build the structure representing valid MIPS disassembler options.
2822
   This is done dynamically for maintenance ease purpose; a static
2823
   initializer would be unreadable.  */
2824
2825
const disasm_options_and_args_t *
2826
disassembler_options_mips (void)
2827
0
{
2828
0
  static disasm_options_and_args_t *opts_and_args;
2829
2830
0
  if (opts_and_args == NULL)
2831
0
    {
2832
0
      size_t num_options = ARRAY_SIZE (mips_options);
2833
0
      size_t num_args = MIPS_OPTION_ARG_SIZE;
2834
0
      disasm_option_arg_t *args;
2835
0
      disasm_options_t *opts;
2836
0
      size_t i;
2837
0
      size_t j;
2838
2839
0
      args = XNEWVEC (disasm_option_arg_t, num_args + 1);
2840
2841
0
      args[MIPS_OPTION_ARG_ABI].name = "ABI";
2842
0
      args[MIPS_OPTION_ARG_ABI].values
2843
0
  = XNEWVEC (const char *, ARRAY_SIZE (mips_abi_choices) + 1);
2844
0
      for (i = 0; i < ARRAY_SIZE (mips_abi_choices); i++)
2845
0
  args[MIPS_OPTION_ARG_ABI].values[i] = mips_abi_choices[i].name;
2846
      /* The array we return must be NULL terminated.  */
2847
0
      args[MIPS_OPTION_ARG_ABI].values[i] = NULL;
2848
2849
0
      args[MIPS_OPTION_ARG_ARCH].name = "ARCH";
2850
0
      args[MIPS_OPTION_ARG_ARCH].values
2851
0
  = XNEWVEC (const char *, ARRAY_SIZE (mips_arch_choices) + 1);
2852
0
      for (i = 0, j = 0; i < ARRAY_SIZE (mips_arch_choices); i++)
2853
0
  if (*mips_arch_choices[i].name != '\0')
2854
0
    args[MIPS_OPTION_ARG_ARCH].values[j++] = mips_arch_choices[i].name;
2855
      /* The array we return must be NULL terminated.  */
2856
0
      args[MIPS_OPTION_ARG_ARCH].values[j] = NULL;
2857
2858
      /* The array we return must be NULL terminated.  */
2859
0
      args[MIPS_OPTION_ARG_SIZE].name = NULL;
2860
0
      args[MIPS_OPTION_ARG_SIZE].values = NULL;
2861
2862
0
      opts_and_args = XNEW (disasm_options_and_args_t);
2863
0
      opts_and_args->args = args;
2864
2865
0
      opts = &opts_and_args->options;
2866
0
      opts->name = XNEWVEC (const char *, num_options + 1);
2867
0
      opts->description = XNEWVEC (const char *, num_options + 1);
2868
0
      opts->arg = XNEWVEC (const disasm_option_arg_t *, num_options + 1);
2869
0
      for (i = 0; i < num_options; i++)
2870
0
  {
2871
0
    opts->name[i] = mips_options[i].name;
2872
0
    opts->description[i] = _(mips_options[i].description);
2873
0
    if (mips_options[i].arg != MIPS_OPTION_ARG_NONE)
2874
0
      opts->arg[i] = &args[mips_options[i].arg];
2875
0
    else
2876
0
      opts->arg[i] = NULL;
2877
0
  }
2878
      /* The array we return must be NULL terminated.  */
2879
0
      opts->name[i] = NULL;
2880
0
      opts->description[i] = NULL;
2881
0
      opts->arg[i] = NULL;
2882
0
    }
2883
2884
0
  return opts_and_args;
2885
0
}
2886
2887
void
2888
print_mips_disassembler_options (FILE *stream)
2889
0
{
2890
0
  const disasm_options_and_args_t *opts_and_args;
2891
0
  const disasm_option_arg_t *args;
2892
0
  const disasm_options_t *opts;
2893
0
  size_t max_len = 0;
2894
0
  size_t i;
2895
0
  size_t j;
2896
2897
0
  opts_and_args = disassembler_options_mips ();
2898
0
  opts = &opts_and_args->options;
2899
0
  args = opts_and_args->args;
2900
2901
0
  fprintf (stream, _("\n\
2902
0
The following MIPS specific disassembler options are supported for use\n\
2903
0
with the -M switch (multiple options should be separated by commas):\n\n"));
2904
2905
  /* Compute the length of the longest option name.  */
2906
0
  for (i = 0; opts->name[i] != NULL; i++)
2907
0
    {
2908
0
      size_t len = strlen (opts->name[i]);
2909
2910
0
      if (opts->arg[i] != NULL)
2911
0
  len += strlen (opts->arg[i]->name);
2912
0
      if (max_len < len)
2913
0
  max_len = len;
2914
0
    }
2915
2916
0
  for (i = 0, max_len++; opts->name[i] != NULL; i++)
2917
0
    {
2918
0
      fprintf (stream, "  %s", opts->name[i]);
2919
0
      if (opts->arg[i] != NULL)
2920
0
  fprintf (stream, "%s", opts->arg[i]->name);
2921
0
      if (opts->description[i] != NULL)
2922
0
  {
2923
0
    size_t len = strlen (opts->name[i]);
2924
2925
0
    if (opts->arg[i] != NULL)
2926
0
      len += strlen (opts->arg[i]->name);
2927
0
    fprintf (stream,
2928
0
       "%*c %s", (int) (max_len - len), ' ', opts->description[i]);
2929
0
  }
2930
0
      fprintf (stream, _("\n"));
2931
0
    }
2932
2933
0
  for (i = 0; args[i].name != NULL; i++)
2934
0
    {
2935
0
      if (args[i].values == NULL)
2936
0
  continue;
2937
0
      fprintf (stream, _("\n\
2938
0
  For the options above, the following values are supported for \"%s\":\n   "),
2939
0
         args[i].name);
2940
0
      for (j = 0; args[i].values[j] != NULL; j++)
2941
0
  fprintf (stream, " %s", args[i].values[j]);
2942
0
      fprintf (stream, _("\n"));
2943
0
    }
2944
2945
  fprintf (stream, _("\n"));
2946
0
}