Coverage Report

Created: 2025-07-08 11:15

/src/binutils-gdb/opcodes/mips-dis.c
Line
Count
Source (jump to first uncovered line)
1
/* Print mips instructions for GDB, the GNU debugger, or for objdump.
2
   Copyright (C) 1989-2025 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
6.23M
#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
13.5k
{
768
13.5k
  const struct mips_abi_choice *c;
769
13.5k
  unsigned int i;
770
771
104k
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_abi_choices) && c == NULL; i++)
772
90.9k
    if (strncmp (mips_abi_choices[i].name, name, namelen) == 0
773
90.9k
  && strlen (mips_abi_choices[i].name) == namelen)
774
1.10k
      c = &mips_abi_choices[i];
775
776
13.5k
  return c;
777
13.5k
}
778
779
static const struct mips_arch_choice *
780
choose_arch_by_name (const char *name, unsigned int namelen)
781
45.5k
{
782
45.5k
  const struct mips_arch_choice *c = NULL;
783
45.5k
  unsigned int i;
784
785
2.34M
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_arch_choices) && c == NULL; i++)
786
2.29M
    if (strncmp (mips_arch_choices[i].name, name, namelen) == 0
787
2.29M
  && strlen (mips_arch_choices[i].name) == namelen)
788
5.88k
      c = &mips_arch_choices[i];
789
790
45.5k
  return c;
791
45.5k
}
792
793
static const struct mips_arch_choice *
794
choose_arch_by_number (unsigned long mach)
795
3.17M
{
796
3.17M
  static unsigned long hint_bfd_mach;
797
3.17M
  static const struct mips_arch_choice *hint_arch_choice;
798
3.17M
  const struct mips_arch_choice *c;
799
3.17M
  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
3.17M
  if (hint_bfd_mach == mach
804
3.17M
      && hint_arch_choice != NULL
805
3.17M
      && hint_arch_choice->bfd_mach == hint_bfd_mach)
806
2.87M
    return hint_arch_choice;
807
808
15.4M
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_arch_choices) && c == NULL; i++)
809
15.1M
    {
810
15.1M
      if (mips_arch_choices[i].bfd_mach_valid
811
15.1M
    && mips_arch_choices[i].bfd_mach == mach)
812
2.06k
  {
813
2.06k
    c = &mips_arch_choices[i];
814
2.06k
    hint_bfd_mach = mach;
815
2.06k
    hint_arch_choice = c;
816
2.06k
  }
817
15.1M
    }
818
298k
  return c;
819
3.17M
}
820
821
/* Check if the object uses NewABI conventions.  */
822
823
static int
824
is_newabi (Elf_Internal_Ehdr *header)
825
2.14M
{
826
  /* There are no old-style ABIs which use 64-bit ELF.  */
827
2.14M
  if (header->e_ident[EI_CLASS] == ELFCLASS64)
828
206k
    return 1;
829
830
  /* If a 32-bit ELF file, n32 is a new-style ABI.  */
831
1.93M
  if ((header->e_flags & EF_MIPS_ABI2) != 0)
832
1.12k
    return 1;
833
834
1.93M
  return 0;
835
1.93M
}
836
837
/* Check if the object has microMIPS ASE code.  */
838
839
static int
840
is_micromips (Elf_Internal_Ehdr *header)
841
2.14M
{
842
2.14M
  if ((header->e_flags & EF_MIPS_ARCH_ASE_MICROMIPS) != 0)
843
297k
    return 1;
844
845
1.84M
  return 0;
846
2.14M
}
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
3.19M
{
889
3.19M
  unsigned long combination_ases = 0;
890
891
3.19M
  if ((opcode_ases & (ASE_XPA | ASE_VIRT)) == (ASE_XPA | ASE_VIRT))
892
133k
    combination_ases |= ASE_XPA_VIRT;
893
3.19M
  if ((opcode_ases & (ASE_MIPS16E2 | ASE_MT)) == (ASE_MIPS16E2 | ASE_MT))
894
6.34k
    combination_ases |= ASE_MIPS16E2_MT;
895
3.19M
  if ((opcode_ases & ASE_EVA)
896
3.19M
      && ((opcode_isa & INSN_ISA_MASK) == ISA_MIPS64R6
897
140k
    || (opcode_isa & INSN_ISA_MASK) == ISA_MIPS32R6))
898
116k
    combination_ases |= ASE_EVA_R6;
899
3.19M
  return combination_ases;
900
3.19M
}
901
902
static void
903
set_default_mips_dis_options (struct disassemble_info *info)
904
3.17M
{
905
3.17M
  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
3.17M
  mips_isa = ISA_MIPS3;
911
3.17M
  mips_processor = CPU_R3000;
912
3.17M
  micromips_ase = 0;
913
3.17M
  mips_ase = 0;
914
3.17M
  mips_gpr_names = mips_gpr_names_oldabi;
915
3.17M
  mips_fpr_names = mips_fpr_names_numeric;
916
3.17M
  mips_cp0_names = mips_cp0_names_numeric;
917
3.17M
  mips_cp0sel_names = NULL;
918
3.17M
  mips_cp0sel_names_len = 0;
919
3.17M
  mips_cp1_names = mips_cp1_names_numeric;
920
3.17M
  mips_hwr_names = mips_hwr_names_numeric;
921
3.17M
  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
3.17M
  chosen_arch = choose_arch_by_number (info->mach);
932
3.17M
  if (chosen_arch != NULL)
933
2.87M
    {
934
2.87M
      mips_processor = chosen_arch->processor;
935
2.87M
      mips_isa = chosen_arch->isa;
936
2.87M
      mips_ase = chosen_arch->ase;
937
2.87M
      mips_cp0_names = chosen_arch->cp0_names;
938
2.87M
      mips_cp0sel_names = chosen_arch->cp0sel_names;
939
2.87M
      mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
940
2.87M
      mips_cp1_names = chosen_arch->cp1_names;
941
2.87M
      mips_hwr_names = chosen_arch->hwr_names;
942
2.87M
    }
943
944
  /* Update settings according to the ELF file header flags.  */
945
3.17M
  if (info->flavour == bfd_target_elf_flavour && info->section != NULL)
946
2.14M
    {
947
2.14M
      struct bfd *abfd = info->section->owner;
948
2.14M
      Elf_Internal_Ehdr *header = elf_elfheader (abfd);
949
2.14M
      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
2.14M
#ifdef HAVE_BFD_MIPS_ELF_GET_ABIFLAGS
957
2.14M
      abiflags = bfd_mips_elf_get_abiflags (abfd);
958
2.14M
#endif
959
      /* If an ELF "newabi" binary, use the n32/(n)64 GPR names.  */
960
2.14M
      if (is_newabi (header))
961
207k
  mips_gpr_names = mips_gpr_names_newabi;
962
      /* If a microMIPS binary, then don't use MIPS16 bindings.  */
963
2.14M
      micromips_ase = is_micromips (header);
964
      /* OR in any extra ASE flags set in ELF file structures.  */
965
2.14M
      if (abiflags)
966
0
  mips_ase |= mips_convert_abiflags_ases (abiflags->ases);
967
2.14M
      else if (header->e_flags & EF_MIPS_ARCH_ASE_MDMX)
968
267k
  mips_ase |= ASE_MDMX;
969
2.14M
    }
970
3.17M
#endif
971
3.17M
  mips_ase |= mips_calculate_combination_ases (mips_isa, mips_ase);
972
3.17M
}
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
430k
{
980
430k
  if (startswith (option, "msa"))
981
3.52k
    {
982
3.52k
      mips_ase |= ASE_MSA;
983
3.52k
      if ((mips_isa & INSN_ISA_MASK) == ISA_MIPS64R2
984
3.52k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R3
985
3.52k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R5
986
3.52k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R6)
987
1.01k
    mips_ase |= ASE_MSA64;
988
3.52k
      return true;
989
3.52k
    }
990
991
427k
  if (startswith (option, "virt"))
992
6.52k
    {
993
6.52k
      mips_ase |= ASE_VIRT;
994
6.52k
      if (mips_isa & ISA_MIPS64R2
995
6.52k
    || mips_isa & ISA_MIPS64R3
996
6.52k
    || mips_isa & ISA_MIPS64R5
997
6.52k
    || mips_isa & ISA_MIPS64R6)
998
6.52k
  mips_ase |= ASE_VIRT64;
999
6.52k
      return true;
1000
6.52k
    }
1001
1002
420k
  if (startswith (option, "xpa"))
1003
644
    {
1004
644
      mips_ase |= ASE_XPA;
1005
644
      return true;
1006
644
    }
1007
1008
420k
  if (startswith (option, "ginv"))
1009
1.66k
    {
1010
1.66k
      mips_ase |= ASE_GINV;
1011
1.66k
      return true;
1012
1.66k
    }
1013
1014
418k
  if (startswith (option, "loongson-mmi"))
1015
438
    {
1016
438
      mips_ase |= ASE_LOONGSON_MMI;
1017
438
      return true;
1018
438
    }
1019
1020
418k
  if (startswith (option, "loongson-cam"))
1021
848
    {
1022
848
      mips_ase |= ASE_LOONGSON_CAM;
1023
848
      return true;
1024
848
    }
1025
  
1026
  /* Put here for match ext2 frist */
1027
417k
  if (startswith (option, "loongson-ext2"))
1028
420
    {
1029
420
      mips_ase |= ASE_LOONGSON_EXT2;
1030
420
      return true;
1031
420
    }
1032
1033
416k
  if (startswith (option, "loongson-ext"))
1034
5.14k
    {
1035
5.14k
      mips_ase |= ASE_LOONGSON_EXT;
1036
5.14k
      return true;
1037
5.14k
    }
1038
1039
411k
  return false;
1040
416k
}
1041
1042
static void
1043
parse_mips_dis_option (const char *option, unsigned int len)
1044
436k
{
1045
436k
  unsigned int i, optionlen, vallen;
1046
436k
  const char *val;
1047
436k
  const struct mips_abi_choice *chosen_abi;
1048
436k
  const struct mips_arch_choice *chosen_arch;
1049
1050
  /* Try to match options that are simple flags */
1051
436k
  if (startswith (option, "no-aliases"))
1052
6.04k
    {
1053
6.04k
      no_aliases = 1;
1054
6.04k
      return;
1055
6.04k
    }
1056
1057
430k
  if (parse_mips_ase_option (option))
1058
19.2k
    {
1059
19.2k
      mips_ase |= mips_calculate_combination_ases (mips_isa, mips_ase);
1060
19.2k
      return;
1061
19.2k
    }
1062
1063
  /* Look for the = that delimits the end of the option name.  */
1064
7.98M
  for (i = 0; i < len; i++)
1065
7.72M
    if (option[i] == '=')
1066
153k
      break;
1067
1068
411k
  if (i == 0)    /* Invalid option: no name before '='.  */
1069
914
    return;
1070
410k
  if (i == len)    /* Invalid option: no '='.  */
1071
257k
    return;
1072
152k
  if (i == (len - 1))  /* Invalid option: no value after '='.  */
1073
972
    return;
1074
1075
151k
  optionlen = i;
1076
151k
  val = option + (optionlen + 1);
1077
151k
  vallen = len - (optionlen + 1);
1078
1079
151k
  if (strncmp ("gpr-names", option, optionlen) == 0
1080
151k
      && strlen ("gpr-names") == optionlen)
1081
1.37k
    {
1082
1.37k
      chosen_abi = choose_abi_by_name (val, vallen);
1083
1.37k
      if (chosen_abi != NULL)
1084
322
  mips_gpr_names = chosen_abi->gpr_names;
1085
1.37k
      return;
1086
1.37k
    }
1087
1088
150k
  if (strncmp ("fpr-names", option, optionlen) == 0
1089
150k
      && strlen ("fpr-names") == optionlen)
1090
2.47k
    {
1091
2.47k
      chosen_abi = choose_abi_by_name (val, vallen);
1092
2.47k
      if (chosen_abi != NULL)
1093
302
  mips_fpr_names = chosen_abi->fpr_names;
1094
2.47k
      return;
1095
2.47k
    }
1096
1097
148k
  if (strncmp ("cp0-names", option, optionlen) == 0
1098
148k
      && strlen ("cp0-names") == optionlen)
1099
5.30k
    {
1100
5.30k
      chosen_arch = choose_arch_by_name (val, vallen);
1101
5.30k
      if (chosen_arch != NULL)
1102
4.32k
  {
1103
4.32k
    mips_cp0_names = chosen_arch->cp0_names;
1104
4.32k
    mips_cp0sel_names = chosen_arch->cp0sel_names;
1105
4.32k
    mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
1106
4.32k
  }
1107
5.30k
      return;
1108
5.30k
    }
1109
1110
142k
  if (strncmp ("cp1-names", option, optionlen) == 0
1111
142k
      && strlen ("cp1-names") == optionlen)
1112
29.2k
    {
1113
29.2k
      chosen_arch = choose_arch_by_name (val, vallen);
1114
29.2k
      if (chosen_arch != NULL)
1115
350
  mips_cp1_names = chosen_arch->cp1_names;
1116
29.2k
      return;
1117
29.2k
    }
1118
1119
113k
  if (strncmp ("hwr-names", option, optionlen) == 0
1120
113k
      && strlen ("hwr-names") == optionlen)
1121
1.29k
    {
1122
1.29k
      chosen_arch = choose_arch_by_name (val, vallen);
1123
1.29k
      if (chosen_arch != NULL)
1124
442
  mips_hwr_names = chosen_arch->hwr_names;
1125
1.29k
      return;
1126
1.29k
    }
1127
1128
112k
  if (strncmp ("reg-names", option, optionlen) == 0
1129
112k
      && strlen ("reg-names") == optionlen)
1130
9.69k
    {
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
9.69k
      chosen_abi = choose_abi_by_name (val, vallen);
1136
9.69k
      if (chosen_abi != NULL)
1137
476
  {
1138
476
    mips_gpr_names = chosen_abi->gpr_names;
1139
476
    mips_fpr_names = chosen_abi->fpr_names;
1140
476
  }
1141
9.69k
      chosen_arch = choose_arch_by_name (val, vallen);
1142
9.69k
      if (chosen_arch != NULL)
1143
766
  {
1144
766
    mips_cp0_names = chosen_arch->cp0_names;
1145
766
    mips_cp0sel_names = chosen_arch->cp0sel_names;
1146
766
    mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
1147
766
    mips_cp1_names = chosen_arch->cp1_names;
1148
766
    mips_hwr_names = chosen_arch->hwr_names;
1149
766
  }
1150
9.69k
      return;
1151
9.69k
    }
1152
1153
  /* Invalid option.  */
1154
112k
}
1155
1156
static void
1157
parse_mips_dis_options (const char *options)
1158
3.17M
{
1159
3.17M
  const char *option_end;
1160
1161
3.17M
  if (options == NULL)
1162
2.86M
    return;
1163
1164
899k
  while (*options != '\0')
1165
587k
    {
1166
      /* Skip empty options.  */
1167
587k
      if (*options == ',')
1168
150k
  {
1169
150k
    options++;
1170
150k
    continue;
1171
150k
  }
1172
1173
      /* We know that *options is neither NUL or a comma.  */
1174
436k
      option_end = options + 1;
1175
9.36M
      while (*option_end != ',' && *option_end != '\0')
1176
8.92M
  option_end++;
1177
1178
436k
      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
436k
      options = option_end;
1183
436k
    }
1184
312k
}
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.60k
{
1192
2.60k
  unsigned int i;
1193
1194
47.5k
  for (i = 0; i < len; i++)
1195
45.6k
    if (names[i].cp0reg == cp0reg && names[i].sel == sel)
1196
691
      return &names[i];
1197
1.91k
  return NULL;
1198
2.60k
}
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
4.44M
{
1206
4.44M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1207
1208
4.44M
  switch (type)
1209
4.44M
    {
1210
4.33M
    case OP_REG_GP:
1211
4.33M
      infprintf (info->stream, dis_style_register, "%s",
1212
4.33M
     mips_gpr_names[regno]);
1213
4.33M
      break;
1214
1215
40.8k
    case OP_REG_FP:
1216
40.8k
      infprintf (info->stream, dis_style_register, "%s",
1217
40.8k
     mips_fpr_names[regno]);
1218
40.8k
      break;
1219
1220
8.36k
    case OP_REG_CCC:
1221
8.36k
      if (opcode->pinfo & (FP_D | FP_S))
1222
8.06k
  infprintf (info->stream, dis_style_register, "$fcc%d", regno);
1223
303
      else
1224
303
  infprintf (info->stream, dis_style_register, "$cc%d", regno);
1225
8.36k
      break;
1226
1227
3.93k
    case OP_REG_VEC:
1228
3.93k
      if (opcode->membership & INSN_5400)
1229
533
  infprintf (info->stream, dis_style_register, "$f%d", regno);
1230
3.40k
      else
1231
3.40k
  infprintf (info->stream, dis_style_register, "$v%d", regno);
1232
3.93k
      break;
1233
1234
2.41k
    case OP_REG_ACC:
1235
2.41k
      infprintf (info->stream, dis_style_register, "$ac%d", regno);
1236
2.41k
      break;
1237
1238
36.1k
    case OP_REG_COPRO:
1239
36.1k
      if (opcode->name[strlen (opcode->name) - 1] == '0')
1240
6.39k
  infprintf (info->stream, dis_style_register, "%s", mips_cp0_names[regno]);
1241
29.7k
      else
1242
29.7k
  infprintf (info->stream, dis_style_register, "$%d", regno);
1243
36.1k
      break;
1244
1245
959
    case OP_REG_CONTROL:
1246
959
      if (opcode->name[strlen (opcode->name) - 1] == '1')
1247
417
  infprintf (info->stream, dis_style_register, "%s", mips_cp1_names[regno]);
1248
542
      else
1249
542
  infprintf (info->stream, dis_style_register, "$%d", regno);
1250
959
      break;
1251
1252
511
    case OP_REG_HW:
1253
511
      infprintf (info->stream, dis_style_register, "%s", mips_hwr_names[regno]);
1254
511
      break;
1255
1256
4.68k
    case OP_REG_VF:
1257
4.68k
      infprintf (info->stream, dis_style_register, "$vf%d", regno);
1258
4.68k
      break;
1259
1260
450
    case OP_REG_VI:
1261
450
      infprintf (info->stream, dis_style_register, "$vi%d", regno);
1262
450
      break;
1263
1264
233
    case OP_REG_R5900_I:
1265
233
      infprintf (info->stream, dis_style_register, "$I");
1266
233
      break;
1267
1268
284
    case OP_REG_R5900_Q:
1269
284
      infprintf (info->stream, dis_style_register, "$Q");
1270
284
      break;
1271
1272
194
    case OP_REG_R5900_R:
1273
194
      infprintf (info->stream, dis_style_register, "$R");
1274
194
      break;
1275
1276
240
    case OP_REG_R5900_ACC:
1277
240
      infprintf (info->stream, dis_style_register, "$ACC");
1278
240
      break;
1279
1280
12.5k
    case OP_REG_MSA:
1281
12.5k
      infprintf (info->stream, dis_style_register, "$w%d", regno);
1282
12.5k
      break;
1283
1284
371
    case OP_REG_MSA_CTRL:
1285
371
      infprintf (info->stream, dis_style_register, "%s",
1286
371
     msa_control_names[regno]);
1287
371
      break;
1288
1289
4.44M
    }
1290
4.44M
}
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
4.62M
{
1312
4.62M
  memset (state, 0, sizeof (*state));
1313
4.62M
}
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
5.95k
{
1323
5.95k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1324
1325
5.95k
  if (operand->size == 4)
1326
4.25k
    infprintf (info->stream, style, "%s%s%s%s",
1327
4.25k
      uval & 8 ? "x" : "",
1328
4.25k
      uval & 4 ? "y" : "",
1329
4.25k
      uval & 2 ? "z" : "",
1330
4.25k
      uval & 1 ? "w" : "");
1331
1.69k
  else if (operand->size == 2)
1332
1.69k
    infprintf (info->stream, style, "%c", "xyzw"[uval]);
1333
0
  else
1334
0
    abort ();
1335
5.95k
}
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
7.82M
{
1344
7.82M
  state->last_reg_type = reg_type;
1345
7.82M
  state->last_regno = regno;
1346
1347
7.82M
  if (!state->seen_dest)
1348
4.18M
    {
1349
4.18M
      state->seen_dest = 1;
1350
4.18M
      state->dest_regno = regno;
1351
4.18M
    }
1352
7.82M
}
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
9.60k
{
1365
9.60k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1366
9.60k
  unsigned int nargs, nstatics, smask, i, j;
1367
9.60k
  void *is = info->stream;
1368
9.60k
  const char *sep;
1369
1370
9.60k
  if (amask == MIPS_SVRS_ALL_ARGS)
1371
259
    {
1372
259
      nargs = 4;
1373
259
      nstatics = 0;
1374
259
    }
1375
9.34k
  else if (amask == MIPS_SVRS_ALL_STATICS)
1376
2.00k
    {
1377
2.00k
      nargs = 0;
1378
2.00k
      nstatics = 4;
1379
2.00k
    }
1380
7.34k
  else
1381
7.34k
    {
1382
7.34k
      nargs = amask >> 2;
1383
7.34k
      nstatics = amask & 3;
1384
7.34k
    }
1385
1386
9.60k
  sep = "";
1387
9.60k
  if (nargs > 0)
1388
1.58k
    {
1389
1.58k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[4]);
1390
1.58k
      if (nargs > 1)
1391
765
  infprintf (is, dis_style_register, "-%s", mips_gpr_names[4 + nargs - 1]);
1392
1.58k
      sep = ",";
1393
1.58k
    }
1394
1395
9.60k
  infprintf (is, dis_style_text, "%s", sep);
1396
9.60k
  infprintf (is, dis_style_immediate, "%d", frame_size);
1397
1398
9.60k
  if (ra)      /* $ra */
1399
7.53k
    {
1400
7.53k
      infprintf (is, dis_style_text, ",");
1401
7.53k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1402
7.53k
    }
1403
1404
9.60k
  smask = 0;
1405
9.60k
  if (s0)      /* $s0 */
1406
7.68k
    smask |= 1 << 0;
1407
9.60k
  if (s1)      /* $s1 */
1408
6.41k
    smask |= 1 << 1;
1409
9.60k
  if (nsreg > 0)    /* $s2-$s8 */
1410
2.37k
    smask |= ((1 << nsreg) - 1) << 2;
1411
1412
71.6k
  for (i = 0; i < 9; i++)
1413
62.0k
    if (smask & (1 << i))
1414
8.59k
      {
1415
8.59k
  infprintf (is, dis_style_text, ",");
1416
8.59k
  infprintf (is, dis_style_register, "%s",
1417
8.59k
       mips_gpr_names[i == 8 ? 30 : (16 + i)]);
1418
  /* Skip over string of set bits.  */
1419
24.6k
  for (j = i; smask & (2 << j); j++)
1420
16.0k
    continue;
1421
8.59k
  if (j > i)
1422
6.74k
    {
1423
6.74k
      infprintf (is, dis_style_text, "-");
1424
6.74k
      infprintf (is, dis_style_register, "%s",
1425
6.74k
           mips_gpr_names[j == 8 ? 30 : (16 + j)]);
1426
6.74k
    }
1427
8.59k
  i = j + 1;
1428
8.59k
      }
1429
  /* Statics $ax - $a3.  */
1430
9.60k
  if (nstatics == 1)
1431
1.15k
    {
1432
1.15k
      infprintf (is, dis_style_text, ",");
1433
1.15k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[7]);
1434
1.15k
    }
1435
8.44k
  else if (nstatics > 0)
1436
2.61k
    {
1437
2.61k
      infprintf (is, dis_style_text, ",");
1438
2.61k
      infprintf (is, dis_style_register, "%s",
1439
2.61k
     mips_gpr_names[7 - nstatics + 1]);
1440
2.61k
      infprintf (is, dis_style_text, "-");
1441
2.61k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[7]);
1442
2.61k
    }
1443
9.60k
}
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
6.68M
{
1458
6.68M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1459
6.68M
  void *is = info->stream;
1460
1461
6.68M
  switch (operand->type)
1462
6.68M
    {
1463
1.82M
    case OP_INT:
1464
1.82M
      {
1465
1.82M
  const struct mips_int_operand *int_op;
1466
1467
1.82M
  int_op = (const struct mips_int_operand *) operand;
1468
1.82M
  uval = mips_decode_int_operand (int_op, uval);
1469
1.82M
  state->last_int = uval;
1470
1.82M
  if (int_op->print_hex)
1471
231k
    infprintf (is, dis_style_immediate, "0x%x", uval);
1472
1.59M
  else
1473
1.59M
    infprintf (is, dis_style_immediate, "%d", uval);
1474
1.82M
      }
1475
1.82M
      break;
1476
1477
13.5k
    case OP_MAPPED_INT:
1478
13.5k
      {
1479
13.5k
  const struct mips_mapped_int_operand *mint_op;
1480
1481
13.5k
  mint_op = (const struct mips_mapped_int_operand *) operand;
1482
13.5k
  uval = mint_op->int_map[uval];
1483
13.5k
  state->last_int = uval;
1484
13.5k
  if (mint_op->print_hex)
1485
7.46k
    infprintf (is, dis_style_immediate, "0x%x", uval);
1486
6.10k
  else
1487
6.10k
    infprintf (is, dis_style_immediate, "%d", uval);
1488
13.5k
      }
1489
13.5k
      break;
1490
1491
4.01k
    case OP_MSB:
1492
4.01k
      {
1493
4.01k
  const struct mips_msb_operand *msb_op;
1494
1495
4.01k
  msb_op = (const struct mips_msb_operand *) operand;
1496
4.01k
  uval += msb_op->bias;
1497
4.01k
  if (msb_op->add_lsb)
1498
2.22k
    uval -= state->last_int;
1499
4.01k
  infprintf (is, dis_style_immediate, "0x%x", uval);
1500
4.01k
      }
1501
4.01k
      break;
1502
1503
3.94M
    case OP_REG:
1504
4.39M
    case OP_OPTIONAL_REG:
1505
4.39M
      {
1506
4.39M
  const struct mips_reg_operand *reg_op;
1507
1508
4.39M
  reg_op = (const struct mips_reg_operand *) operand;
1509
4.39M
  uval = mips_decode_reg_operand (reg_op, uval);
1510
4.39M
  print_reg (info, opcode, reg_op->reg_type, uval);
1511
1512
4.39M
  mips_seen_register (state, uval, reg_op->reg_type);
1513
4.39M
      }
1514
4.39M
      break;
1515
1516
2.92k
    case OP_REG_PAIR:
1517
2.92k
      {
1518
2.92k
  const struct mips_reg_pair_operand *pair_op;
1519
1520
2.92k
  pair_op = (const struct mips_reg_pair_operand *) operand;
1521
2.92k
  print_reg (info, opcode, pair_op->reg_type,
1522
2.92k
       pair_op->reg1_map[uval]);
1523
2.92k
  infprintf (is, dis_style_text, ",");
1524
2.92k
  print_reg (info, opcode, pair_op->reg_type,
1525
2.92k
       pair_op->reg2_map[uval]);
1526
2.92k
      }
1527
2.92k
      break;
1528
1529
366k
    case OP_PCREL:
1530
366k
      {
1531
366k
  const struct mips_pcrel_operand *pcrel_op;
1532
1533
366k
  pcrel_op = (const struct mips_pcrel_operand *) operand;
1534
366k
  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
366k
  if (pcrel_op->include_isa_bit
1539
366k
      && info->flavour != bfd_target_unknown_flavour)
1540
311k
    info->target &= -2;
1541
1542
366k
  (*info->print_address_func) (info->target, info);
1543
366k
      }
1544
366k
      break;
1545
1546
258
    case OP_PERF_REG:
1547
258
      infprintf (is, dis_style_register, "%d", uval);
1548
258
      break;
1549
1550
11.5k
    case OP_ADDIUSP_INT:
1551
11.5k
      {
1552
11.5k
  int sval;
1553
1554
11.5k
  sval = mips_signed_operand (operand, uval) * 4;
1555
11.5k
  if (sval >= -8 && sval < 8)
1556
691
    sval ^= 0x400;
1557
11.5k
  infprintf (is, dis_style_immediate, "%d", sval);
1558
11.5k
  break;
1559
3.94M
      }
1560
1561
1.19k
    case OP_CLO_CLZ_DEST:
1562
1.19k
      {
1563
1.19k
  unsigned int reg1, reg2;
1564
1565
1.19k
  reg1 = uval & 31;
1566
1.19k
  reg2 = uval >> 5;
1567
  /* If one is zero use the other.  */
1568
1.19k
  if (reg1 == reg2 || reg2 == 0)
1569
538
    infprintf (is, dis_style_register, "%s", mips_gpr_names[reg1]);
1570
657
  else if (reg1 == 0)
1571
239
    infprintf (is, dis_style_register, "%s", mips_gpr_names[reg2]);
1572
418
  else
1573
418
    {
1574
      /* Bogus, result depends on processor.  */
1575
418
      infprintf (is, dis_style_register, "%s", mips_gpr_names[reg1]);
1576
418
      infprintf (is, dis_style_text, " or ");
1577
418
      infprintf (is, dis_style_register, "%s", mips_gpr_names[reg2]);
1578
418
    }
1579
1.19k
      }
1580
1.19k
      break;
1581
1582
290
    case OP_SAME_RS_RT:
1583
11.7k
    case OP_CHECK_PREV:
1584
22.7k
    case OP_NON_ZERO_REG:
1585
22.7k
      {
1586
22.7k
  print_reg (info, opcode, OP_REG_GP, uval & 31);
1587
22.7k
  mips_seen_register (state, uval, OP_REG_GP);
1588
22.7k
      }
1589
22.7k
      break;
1590
1591
7.32k
    case OP_LWM_SWM_LIST:
1592
7.32k
      if (operand->size == 2)
1593
1.95k
  {
1594
1.95k
    if (uval == 0)
1595
1.24k
      {
1596
1.24k
        infprintf (is, dis_style_register, "%s",
1597
1.24k
       mips_gpr_names[16]);
1598
1.24k
        infprintf (is, dis_style_text, ",");
1599
1.24k
        infprintf (is, dis_style_register, "%s",
1600
1.24k
       mips_gpr_names[31]);
1601
1.24k
      }
1602
711
    else
1603
711
      {
1604
711
        infprintf (is, dis_style_register, "%s",
1605
711
       mips_gpr_names[16]);
1606
711
        infprintf (is, dis_style_text, "-");
1607
711
        infprintf (is, dis_style_register, "%s",
1608
711
       mips_gpr_names[16 + uval]);
1609
711
        infprintf (is, dis_style_text, ",");
1610
711
        infprintf (is, dis_style_register, "%s",
1611
711
       mips_gpr_names[31]);
1612
711
      }
1613
1.95k
  }
1614
5.37k
      else
1615
5.37k
  {
1616
5.37k
    int s_reg_encode;
1617
1618
5.37k
    s_reg_encode = uval & 0xf;
1619
5.37k
    if (s_reg_encode != 0)
1620
4.88k
      {
1621
4.88k
        if (s_reg_encode == 1)
1622
2.12k
    infprintf (is, dis_style_register, "%s", mips_gpr_names[16]);
1623
2.76k
        else if (s_reg_encode < 9)
1624
1.68k
    {
1625
1.68k
      infprintf (is, dis_style_register, "%s",
1626
1.68k
           mips_gpr_names[16]);
1627
1.68k
      infprintf (is, dis_style_text, "-");
1628
1.68k
      infprintf (is, dis_style_register, "%s",
1629
1.68k
           mips_gpr_names[15 + s_reg_encode]);
1630
1.68k
    }
1631
1.08k
        else if (s_reg_encode == 9)
1632
543
    {
1633
543
      infprintf (is, dis_style_register, "%s",
1634
543
           mips_gpr_names[16]);
1635
543
      infprintf (is, dis_style_text, "-");
1636
543
      infprintf (is, dis_style_register, "%s",
1637
543
           mips_gpr_names[23]);
1638
543
      infprintf (is, dis_style_text, ",");
1639
543
      infprintf (is, dis_style_register, "%s",
1640
543
           mips_gpr_names[30]);
1641
543
    }
1642
539
        else
1643
539
    infprintf (is, dis_style_text, "UNKNOWN");
1644
4.88k
      }
1645
1646
5.37k
    if (uval & 0x10) /* For ra.  */
1647
3.24k
      {
1648
3.24k
        if (s_reg_encode == 0)
1649
364
    infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1650
2.87k
        else
1651
2.87k
    {
1652
2.87k
      infprintf (is, dis_style_text, ",");
1653
2.87k
      infprintf (is, dis_style_register, "%s",
1654
2.87k
           mips_gpr_names[31]);
1655
2.87k
    }
1656
3.24k
      }
1657
5.37k
  }
1658
7.32k
      break;
1659
1660
4.57k
    case OP_ENTRY_EXIT_LIST:
1661
4.57k
      {
1662
4.57k
  const char *sep;
1663
4.57k
  unsigned int amask, smask;
1664
1665
4.57k
  sep = "";
1666
4.57k
  amask = (uval >> 3) & 7;
1667
4.57k
  if (amask > 0 && amask < 5)
1668
3.22k
    {
1669
3.22k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[4]);
1670
3.22k
      if (amask > 1)
1671
431
        {
1672
431
    infprintf (is, dis_style_text, "-");
1673
431
    infprintf (is, dis_style_register, "%s",
1674
431
         mips_gpr_names[amask + 3]);
1675
431
        }
1676
3.22k
      sep = ",";
1677
3.22k
    }
1678
1679
4.57k
  smask = (uval >> 1) & 3;
1680
4.57k
  if (smask == 3)
1681
3.27k
    {
1682
3.27k
      infprintf (is, dis_style_text, "%s??", sep);
1683
3.27k
      sep = ",";
1684
3.27k
    }
1685
1.30k
  else if (smask > 0)
1686
1.00k
    {
1687
1.00k
      infprintf (is, dis_style_text, "%s", sep);
1688
1.00k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[16]);
1689
1.00k
      if (smask > 1)
1690
477
        {
1691
477
    infprintf (is, dis_style_text, "-");
1692
477
    infprintf (is, dis_style_register, "%s",
1693
477
         mips_gpr_names[smask + 15]);
1694
477
        }
1695
1.00k
      sep = ",";
1696
1.00k
    }
1697
1698
4.57k
  if (uval & 1)
1699
3.93k
    {
1700
3.93k
      infprintf (is, dis_style_text, "%s", sep);
1701
3.93k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1702
3.93k
      sep = ",";
1703
3.93k
    }
1704
1705
4.57k
  if (amask == 5 || amask == 6)
1706
616
    {
1707
616
      infprintf (is, dis_style_text, "%s", sep);
1708
616
      infprintf (is, dis_style_register, "%s", mips_fpr_names[0]);
1709
616
      if (amask == 6)
1710
328
        {
1711
328
    infprintf (is, dis_style_text, "-");
1712
328
    infprintf (is, dis_style_register, "%s", mips_fpr_names[1]);
1713
328
        }
1714
616
    }
1715
4.57k
      }
1716
4.57k
      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.90k
    case OP_MDMX_IMM_REG:
1723
1.90k
      {
1724
1.90k
  unsigned int vsel;
1725
1726
1.90k
  vsel = uval >> 5;
1727
1.90k
  uval &= 31;
1728
1.90k
  if ((vsel & 0x10) == 0)
1729
794
    {
1730
794
      int fmt;
1731
1732
794
      vsel &= 0x0f;
1733
2.07k
      for (fmt = 0; fmt < 3; fmt++, vsel >>= 1)
1734
1.76k
        if ((vsel & 1) == 0)
1735
478
    break;
1736
794
      print_reg (info, opcode, OP_REG_VEC, uval);
1737
794
      infprintf (is, dis_style_text, "[");
1738
794
      infprintf (is, dis_style_immediate, "%d", vsel >> 1);
1739
794
      infprintf (is, dis_style_text, "]");
1740
794
    }
1741
1.10k
  else if ((vsel & 0x08) == 0)
1742
620
    print_reg (info, opcode, OP_REG_VEC, uval);
1743
486
  else
1744
486
    infprintf (is, dis_style_immediate, "0x%x", uval);
1745
1.90k
      }
1746
1.90k
      break;
1747
1748
15.4k
    case OP_REPEAT_PREV_REG:
1749
15.4k
      print_reg (info, opcode, state->last_reg_type, state->last_regno);
1750
15.4k
      break;
1751
1752
624
    case OP_REPEAT_DEST_REG:
1753
624
      print_reg (info, opcode, state->last_reg_type, state->dest_regno);
1754
624
      break;
1755
1756
4.21k
    case OP_PC:
1757
4.21k
      infprintf (is, dis_style_register, "$pc");
1758
4.21k
      break;
1759
1760
852
    case OP_REG28:
1761
852
      print_reg (info, opcode, OP_REG_GP, 28);
1762
852
      break;
1763
1764
715
    case OP_VU0_SUFFIX:
1765
4.46k
    case OP_VU0_MATCH_SUFFIX:
1766
4.46k
      print_vu0_channel (info, operand, uval, dis_style_register);
1767
4.46k
      break;
1768
1769
1.05k
    case OP_IMM_INDEX:
1770
1.05k
      infprintf (is, dis_style_text, "[");
1771
1.05k
      infprintf (is, dis_style_immediate, "%d", uval);
1772
1.05k
      infprintf (is, dis_style_text, "]");
1773
1.05k
      break;
1774
1775
293
    case OP_REG_INDEX:
1776
293
      infprintf (is, dis_style_text, "[");
1777
293
      print_reg (info, opcode, OP_REG_GP, uval);
1778
293
      infprintf (is, dis_style_text, "]");
1779
293
      break;
1780
6.68M
    }
1781
6.68M
}
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
2.04M
{
1791
2.04M
  struct mips_print_arg_state state;
1792
2.04M
  const struct mips_operand *operand;
1793
2.04M
  const char *s;
1794
2.04M
  unsigned int uval;
1795
1796
2.04M
  init_print_arg_state (&state);
1797
11.1M
  for (s = opcode->args; *s; ++s)
1798
9.06M
    {
1799
9.06M
      switch (*s)
1800
9.06M
  {
1801
2.32M
  case ',':
1802
3.14M
  case '(':
1803
3.96M
  case ')':
1804
3.96M
    break;
1805
1806
218
  case '#':
1807
218
    ++s;
1808
218
    break;
1809
1810
5.10M
  default:
1811
5.10M
    operand = decode_operand (s);
1812
1813
5.10M
    if (operand)
1814
5.10M
      {
1815
5.10M
        uval = mips_extract_operand (operand, insn);
1816
5.10M
        switch (operand->type)
1817
5.10M
    {
1818
2.96M
    case OP_REG:
1819
3.40M
    case OP_OPTIONAL_REG:
1820
3.40M
      {
1821
3.40M
        const struct mips_reg_operand *reg_op;
1822
1823
3.40M
        reg_op = (const struct mips_reg_operand *) operand;
1824
3.40M
        uval = mips_decode_reg_operand (reg_op, uval);
1825
3.40M
        mips_seen_register (&state, uval, reg_op->reg_type);
1826
3.40M
      }
1827
3.40M
    break;
1828
1829
5.08k
    case OP_SAME_RS_RT:
1830
5.08k
      {
1831
5.08k
        unsigned int reg1, reg2;
1832
1833
5.08k
        reg1 = uval & 31;
1834
5.08k
        reg2 = uval >> 5;
1835
1836
5.08k
        if (reg1 != reg2 || reg1 == 0)
1837
4.79k
          return false;
1838
5.08k
      }
1839
290
    break;
1840
1841
16.6k
    case OP_CHECK_PREV:
1842
16.6k
      {
1843
16.6k
        const struct mips_check_prev_operand *prev_op;
1844
1845
16.6k
        prev_op = (const struct mips_check_prev_operand *) operand;
1846
1847
16.6k
        if (!prev_op->zero_ok && uval == 0)
1848
763
          return false;
1849
1850
15.9k
        if (((prev_op->less_than_ok && uval < state.last_regno)
1851
15.9k
      || (prev_op->greater_than_ok && uval > state.last_regno)
1852
15.9k
      || (prev_op->equal_ok && uval == state.last_regno)))
1853
11.4k
          break;
1854
1855
4.42k
        return false;
1856
15.9k
      }
1857
1858
13.1k
    case OP_NON_ZERO_REG:
1859
13.1k
      {
1860
13.1k
        if (uval == 0)
1861
1.62k
          return false;
1862
13.1k
      }
1863
11.5k
    break;
1864
1865
1.32M
    case OP_INT:
1866
1.33M
    case OP_MAPPED_INT:
1867
1.34M
    case OP_MSB:
1868
1.34M
    case OP_REG_PAIR:
1869
1.61M
    case OP_PCREL:
1870
1.61M
    case OP_PERF_REG:
1871
1.62M
    case OP_ADDIUSP_INT:
1872
1.63M
    case OP_CLO_CLZ_DEST:
1873
1.63M
    case OP_LWM_SWM_LIST:
1874
1.63M
    case OP_ENTRY_EXIT_LIST:
1875
1.64M
    case OP_MDMX_IMM_REG:
1876
1.65M
    case OP_REPEAT_PREV_REG:
1877
1.65M
    case OP_REPEAT_DEST_REG:
1878
1.66M
    case OP_PC:
1879
1.66M
    case OP_REG28:
1880
1.66M
    case OP_VU0_SUFFIX:
1881
1.66M
    case OP_VU0_MATCH_SUFFIX:
1882
1.66M
    case OP_IMM_INDEX:
1883
1.66M
    case OP_REG_INDEX:
1884
1.66M
    case OP_SAVE_RESTORE_LIST:
1885
1.66M
      break;
1886
5.10M
    }
1887
5.10M
      }
1888
5.09M
    if (*s == 'm' || *s == '+' || *s == '-')
1889
440k
      ++s;
1890
9.06M
  }
1891
9.06M
    }
1892
2.03M
  return true;
1893
2.04M
}
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
1.94M
{
1906
1.94M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1907
1.94M
  void *is = info->stream;
1908
1.94M
  struct mips_print_arg_state state;
1909
1.94M
  const struct mips_operand *operand;
1910
1.94M
  const char *s;
1911
1912
1.94M
  init_print_arg_state (&state);
1913
10.9M
  for (s = opcode->args; *s; ++s)
1914
9.04M
    {
1915
9.04M
      switch (*s)
1916
9.04M
  {
1917
2.31M
  case ',':
1918
3.13M
  case '(':
1919
3.95M
  case ')':
1920
3.95M
    infprintf (is, dis_style_text, "%c", *s);
1921
3.95M
    break;
1922
1923
218
  case '#':
1924
218
    ++s;
1925
218
    infprintf (is, dis_style_text, "%c%c", *s, *s);
1926
218
    break;
1927
1928
5.08M
  default:
1929
5.08M
    operand = decode_operand (s);
1930
5.08M
    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
5.08M
    if (operand->type == OP_SAVE_RESTORE_LIST)
1940
295
      {
1941
        /* Handle this case here because of the complex behavior.  */
1942
295
        unsigned int amask = (insn >> 15) & 0xf;
1943
295
        unsigned int nsreg = (insn >> 23) & 0x7;
1944
295
        unsigned int ra = insn & 0x1000;      /* $ra */
1945
295
        unsigned int s0 = insn & 0x800;     /* $s0 */
1946
295
        unsigned int s1 = insn & 0x400;     /* $s1 */
1947
295
        unsigned int frame_size = (((insn >> 15) & 0xf0)
1948
295
           | ((insn >> 6) & 0x0f)) * 8;
1949
295
        mips_print_save_restore (info, amask, nsreg, ra, s0, s1,
1950
295
               frame_size);
1951
295
      }
1952
5.08M
    else if (operand->type == OP_REG
1953
5.08M
       && s[1] == ','
1954
5.08M
       && (s[2] == 'H' || s[2] == 'J')
1955
5.08M
       && opcode->name[strlen (opcode->name) - 1] == '0')
1956
1.73k
      {
1957
        /* Coprocessor register 0 with sel field.  */
1958
1.73k
        const struct mips_cp0sel_name *n;
1959
1.73k
        unsigned int reg, sel;
1960
1961
1.73k
        reg = mips_extract_operand (operand, insn);
1962
1.73k
        s += 2;
1963
1.73k
        operand = decode_operand (s);
1964
1.73k
        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.73k
        n = lookup_mips_cp0sel_name (mips_cp0sel_names,
1972
1.73k
             mips_cp0sel_names_len,
1973
1.73k
             reg, sel);
1974
1.73k
        if (n != NULL)
1975
315
    infprintf (is, dis_style_register, "%s", n->name);
1976
1.42k
        else
1977
1.42k
    {
1978
1.42k
      infprintf (is, dis_style_register, "$%d", reg);
1979
1.42k
      infprintf (is, dis_style_text, ",");
1980
1.42k
      infprintf (is, dis_style_immediate, "%d", sel);
1981
1.42k
    }
1982
1.73k
      }
1983
5.08M
    else
1984
5.08M
      {
1985
5.08M
        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
5.08M
        if (operand->type == OP_PCREL)
1991
274k
    {
1992
274k
      const struct mips_pcrel_operand *pcrel_op;
1993
1994
274k
      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
274k
      if (pcrel_op->include_isa_bit)
1998
272k
        base_pc += length;
1999
274k
    }
2000
2001
5.08M
        print_insn_arg (info, &state, opcode, operand, base_pc,
2002
5.08M
            mips_extract_operand (operand, insn));
2003
5.08M
      }
2004
5.08M
    if (*s == 'm' || *s == '+' || *s == '-')
2005
439k
      ++s;
2006
5.08M
    break;
2007
9.04M
  }
2008
9.04M
    }
2009
1.94M
}
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
2.07M
{
2021
2.07M
#define GET_OP(insn, field)     \
2022
2.18M
  (((insn) >> OP_SH_##field) & OP_MASK_##field)
2023
2.07M
  static const struct mips_opcode *mips_hash[OP_MASK_OP + 1];
2024
2.07M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2025
2.07M
  const struct mips_opcode *op;
2026
2.07M
  static bool init = 0;
2027
2.07M
  void *is = info->stream;
2028
2029
  /* Build a hash table to shorten the search time.  */
2030
2.07M
  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
126k
      || (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
2.07M
  info->bytes_per_chunk = INSNLEN;
2053
2.07M
  info->display_endian = info->endian;
2054
2.07M
  info->insn_info_valid = 1;
2055
2.07M
  info->branch_delay_insns = 0;
2056
2.07M
  info->data_size = 0;
2057
2.07M
  info->insn_type = dis_nonbranch;
2058
2.07M
  info->target = 0;
2059
2.07M
  info->target2 = 0;
2060
2061
2.07M
  op = mips_hash[GET_OP (word, OP)];
2062
2.07M
  if (op != NULL)
2063
2.07M
    {
2064
1.41G
      for (; op < &mips_opcodes[NUMOPCODES]; op++)
2065
1.41G
  {
2066
1.41G
    if (op->pinfo != INSN_MACRO
2067
1.41G
        && !(no_aliases && (op->pinfo2 & INSN2_ALIAS))
2068
1.41G
        && (word & op->mask) == op->match)
2069
2.20M
      {
2070
        /* We always disassemble the jalx instruction, except for MIPS r6.  */
2071
2.20M
        if (!opcode_is_member (op, mips_isa, mips_ase, mips_processor)
2072
2.20M
     && (strcmp (op->name, "jalx")
2073
490k
         || (mips_isa & INSN_ISA_MASK) == ISA_MIPS32R6
2074
490k
         || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R6))
2075
490k
    continue;
2076
2077
        /* Figure out instruction type and branch delay information.  */
2078
1.71M
        if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0)
2079
164k
          {
2080
164k
      if ((op->pinfo & (INSN_WRITE_GPR_31 | INSN_WRITE_1)) != 0)
2081
99.1k
        info->insn_type = dis_jsr;
2082
65.8k
      else
2083
65.8k
        info->insn_type = dis_branch;
2084
164k
      info->branch_delay_insns = 1;
2085
164k
    }
2086
1.54M
        else if ((op->pinfo & (INSN_COND_BRANCH_DELAY
2087
1.54M
             | INSN_COND_BRANCH_LIKELY)) != 0)
2088
141k
    {
2089
141k
      if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2090
1.05k
        info->insn_type = dis_condjsr;
2091
140k
      else
2092
140k
        info->insn_type = dis_condbranch;
2093
141k
      info->branch_delay_insns = 1;
2094
141k
    }
2095
1.40M
        else if ((op->pinfo & (INSN_STORE_MEMORY
2096
1.40M
             | INSN_LOAD_MEMORY)) != 0)
2097
675k
    info->insn_type = dis_dref;
2098
2099
1.71M
        if (!validate_insn_args (op, decode_mips_operand, word))
2100
11.6k
    continue;
2101
2102
1.70M
        infprintf (is, dis_style_mnemonic, "%s", op->name);
2103
1.70M
        if (op->pinfo2 & INSN2_VU0_CHANNEL_SUFFIX)
2104
1.48k
    {
2105
1.48k
      unsigned int uval;
2106
2107
1.48k
      infprintf (is, dis_style_mnemonic, ".");
2108
1.48k
      uval = mips_extract_operand (&mips_vu0_channel_mask, word);
2109
1.48k
      print_vu0_channel (info, &mips_vu0_channel_mask, uval,
2110
1.48k
             dis_style_mnemonic);
2111
1.48k
    }
2112
2113
1.70M
        if (op->args[0])
2114
1.62M
    {
2115
1.62M
      infprintf (is, dis_style_text, "\t");
2116
1.62M
      print_insn_args (info, op, decode_mips_operand, word,
2117
1.62M
           memaddr, 4);
2118
1.62M
    }
2119
2120
1.70M
        return INSNLEN;
2121
1.71M
      }
2122
1.41G
  }
2123
2.07M
    }
2124
377k
#undef GET_OP
2125
2126
  /* Handle undefined instructions.  */
2127
377k
  info->insn_type = dis_noninsn;
2128
377k
  infprintf (is, dis_style_assembler_directive, ".word");
2129
377k
  infprintf (is, dis_style_text, "\t");
2130
377k
  infprintf (is, dis_style_immediate, "0x%x", word);
2131
377k
  return INSNLEN;
2132
2.07M
}
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
2.74M
{
2144
2.74M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2145
2.74M
  void *is = info->stream;
2146
2.74M
  const struct mips_operand *operand, *ext_operand;
2147
2.74M
  unsigned short ext_size;
2148
2.74M
  unsigned int uval;
2149
2.74M
  bfd_vma baseaddr;
2150
2151
2.74M
  if (!use_extend)
2152
2.57M
    extend = 0;
2153
2154
2.74M
  switch (type)
2155
2.74M
    {
2156
822k
    case ',':
2157
975k
    case '(':
2158
1.12M
    case ')':
2159
1.12M
      infprintf (is, dis_style_text, "%c", type);
2160
1.12M
      break;
2161
2162
1.61M
    default:
2163
1.61M
      operand = decode_mips16_operand (type, false);
2164
1.61M
      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.61M
      if (operand->type == OP_SAVE_RESTORE_LIST)
2173
9.30k
  {
2174
    /* Handle this case here because of the complex interaction
2175
       with the EXTEND opcode.  */
2176
9.30k
    unsigned int amask = extend & 0xf;
2177
9.30k
    unsigned int nsreg = (extend >> 8) & 0x7;
2178
9.30k
    unsigned int ra = insn & 0x40;      /* $ra */
2179
9.30k
    unsigned int s0 = insn & 0x20;      /* $s0 */
2180
9.30k
    unsigned int s1 = insn & 0x10;      /* $s1 */
2181
9.30k
    unsigned int frame_size = ((extend & 0xf0) | (insn & 0x0f)) * 8;
2182
9.30k
    if (frame_size == 0 && !use_extend)
2183
2.14k
      frame_size = 128;
2184
9.30k
    mips_print_save_restore (info, amask, nsreg, ra, s0, s1, frame_size);
2185
9.30k
    break;
2186
9.30k
  }
2187
2188
1.60M
      if (is_offset && operand->type == OP_INT)
2189
153k
  {
2190
153k
    const struct mips_int_operand *int_op;
2191
2192
153k
    int_op = (const struct mips_int_operand *) operand;
2193
153k
    info->insn_type = dis_dref;
2194
153k
    info->data_size = 1 << int_op->shift;
2195
153k
  }
2196
2197
1.60M
      ext_size = 0;
2198
1.60M
      if (use_extend)
2199
97.5k
  {
2200
97.5k
    ext_operand = decode_mips16_operand (type, true);
2201
97.5k
    if (ext_operand != operand
2202
97.5k
        || (operand->type == OP_INT && operand->lsb == 0
2203
69.6k
      && mips_opcode_32bit_p (opcode)))
2204
31.2k
      {
2205
31.2k
        ext_size = ext_operand->size;
2206
31.2k
        operand = ext_operand;
2207
31.2k
      }
2208
97.5k
  }
2209
1.60M
      if (operand->size == 26)
2210
11.5k
  uval = ((extend & 0x1f) << 21) | ((extend & 0x3e0) << 11) | insn;
2211
1.59M
      else if (ext_size == 16 || ext_size == 9)
2212
27.8k
  uval = ((extend & 0x1f) << 11) | (extend & 0x7e0) | (insn & 0x1f);
2213
1.56M
      else if (ext_size == 15)
2214
1.47k
  uval = ((extend & 0xf) << 11) | (extend & 0x7f0) | (insn & 0xf);
2215
1.56M
      else if (ext_size == 6)
2216
598
  uval = ((extend >> 6) & 0x1f) | (extend & 0x20);
2217
1.56M
      else
2218
1.56M
  uval = mips_extract_operand (operand, (extend << 16) | insn);
2219
1.60M
      if (ext_size == 9)
2220
2.19k
  uval &= (1U << ext_size) - 1;
2221
2222
1.60M
      baseaddr = memaddr + 2;
2223
1.60M
      if (operand->type == OP_PCREL)
2224
91.9k
  {
2225
91.9k
    const struct mips_pcrel_operand *pcrel_op;
2226
2227
91.9k
    pcrel_op = (const struct mips_pcrel_operand *) operand;
2228
91.9k
    if (!pcrel_op->include_isa_bit && use_extend)
2229
2.24k
      baseaddr = memaddr - 2;
2230
89.6k
    else if (!pcrel_op->include_isa_bit)
2231
27.2k
      {
2232
27.2k
        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
27.2k
        if (info->read_memory_func (memaddr - 4, buffer, 2, info) == 0
2242
27.2k
      && (((info->endian == BFD_ENDIAN_BIG
2243
16.8k
      ? bfd_getb16 (buffer)
2244
16.8k
      : bfd_getl16 (buffer))
2245
16.8k
           & 0xf800) == 0x1800))
2246
288
    baseaddr = memaddr - 4;
2247
26.9k
        else if (info->read_memory_func (memaddr - 2, buffer, 2,
2248
26.9k
                 info) == 0
2249
26.9k
           && (((info->endian == BFD_ENDIAN_BIG
2250
16.5k
           ? bfd_getb16 (buffer)
2251
16.5k
           : bfd_getl16 (buffer))
2252
16.5k
          & 0xf89f) == 0xe800)
2253
26.9k
           && (((info->endian == BFD_ENDIAN_BIG
2254
96
           ? bfd_getb16 (buffer)
2255
96
           : bfd_getl16 (buffer))
2256
96
          & 0x0060) != 0x0060))
2257
96
    baseaddr = memaddr - 2;
2258
26.8k
        else
2259
26.8k
    baseaddr = memaddr;
2260
27.2k
      }
2261
91.9k
  }
2262
2263
1.60M
      print_insn_arg (info, state, opcode, operand, baseaddr + 1, uval);
2264
1.60M
      break;
2265
2.74M
    }
2266
2.74M
}
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
683k
{
2276
683k
  if (info->symbols
2277
683k
      && info->symbols[0]
2278
683k
      && (info->symbols[0]->flags & BSF_SYNTHETIC)
2279
683k
      && addr == bfd_asymbol_value (info->symbols[0]) + 12)
2280
0
    return true;
2281
2282
683k
  return false;
2283
683k
}
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
683k
{
2299
683k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2300
683k
  int status;
2301
683k
  bfd_byte buffer[4];
2302
683k
  const struct mips_opcode *op, *opend;
2303
683k
  struct mips_print_arg_state state;
2304
683k
  void *is = info->stream;
2305
683k
  bool have_second;
2306
683k
  bool extend_only;
2307
683k
  unsigned int second;
2308
683k
  unsigned int first;
2309
683k
  unsigned int full;
2310
2311
683k
  info->bytes_per_chunk = 2;
2312
683k
  info->display_endian = info->endian;
2313
683k
  info->insn_info_valid = 1;
2314
683k
  info->branch_delay_insns = 0;
2315
683k
  info->data_size = 0;
2316
683k
  info->target = 0;
2317
683k
  info->target2 = 0;
2318
2319
683k
#define GET_OP(insn, field) \
2320
683k
  (((insn) >> MIPS16OP_SH_##field) & MIPS16OP_MASK_##field)
2321
  /* Decode PLT entry's GOT slot address word.  */
2322
683k
  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
683k
  else
2342
683k
    {
2343
683k
      info->insn_type = dis_nonbranch;
2344
683k
      status = (*info->read_memory_func) (memaddr, buffer, 2, info);
2345
683k
    }
2346
683k
  if (status != 0)
2347
1.01k
    {
2348
1.01k
      (*info->memory_error_func) (status, memaddr, info);
2349
1.01k
      return -1;
2350
1.01k
    }
2351
2352
682k
  extend_only = false;
2353
2354
682k
  if (info->endian == BFD_ENDIAN_BIG)
2355
93.3k
    first = bfd_getb16 (buffer);
2356
589k
  else
2357
589k
    first = bfd_getl16 (buffer);
2358
2359
682k
  status = (*info->read_memory_func) (memaddr + 2, buffer, 2, info);
2360
682k
  if (status == 0)
2361
681k
    {
2362
681k
      have_second = true;
2363
681k
      if (info->endian == BFD_ENDIAN_BIG)
2364
93.3k
  second = bfd_getb16 (buffer);
2365
588k
      else
2366
588k
  second = bfd_getl16 (buffer);
2367
681k
      full = (first << 16) | second;
2368
681k
    }
2369
1.26k
  else
2370
1.26k
    {
2371
1.26k
      have_second = false;
2372
1.26k
      second = 0;
2373
1.26k
      full = first;
2374
1.26k
    }
2375
2376
  /* FIXME: Should probably use a hash table on the major opcode here.  */
2377
2378
682k
  opend = mips16_opcodes + bfd_mips16_num_opcodes;
2379
68.6M
  for (op = mips16_opcodes; op < opend; op++)
2380
68.6M
    {
2381
68.6M
      enum match_kind match;
2382
2383
68.6M
      if (!opcode_is_member (op, mips_isa, mips_ase, mips_processor))
2384
14.2M
  continue;
2385
2386
54.3M
      if (op->pinfo == INSN_MACRO
2387
54.3M
    || (no_aliases && (op->pinfo2 & INSN2_ALIAS)))
2388
12.2M
  match = MATCH_NONE;
2389
42.1M
      else if (mips_opcode_32bit_p (op))
2390
6.98M
  {
2391
6.98M
    if (have_second
2392
6.98M
        && (full & op->mask) == op->match)
2393
19.4k
      match = MATCH_FULL;
2394
6.96M
    else
2395
6.96M
      match = MATCH_NONE;
2396
6.98M
  }
2397
35.1M
      else if ((first & op->mask) == op->match)
2398
592k
  {
2399
592k
    match = MATCH_SHORT;
2400
592k
    second = 0;
2401
592k
    full = first;
2402
592k
  }
2403
34.5M
      else if ((first & 0xf800) == 0xf000
2404
34.5M
         && have_second
2405
34.5M
         && !extend_only
2406
34.5M
         && (second & op->mask) == op->match)
2407
37.1k
  {
2408
37.1k
    if (op->pinfo2 & INSN2_SHORT_ONLY)
2409
7.29k
      {
2410
7.29k
        match = MATCH_NONE;
2411
7.29k
        extend_only = true;
2412
7.29k
      }
2413
29.8k
    else
2414
29.8k
      match = MATCH_FULL;
2415
37.1k
  }
2416
34.5M
      else
2417
34.5M
  match = MATCH_NONE;
2418
2419
54.3M
      if (match != MATCH_NONE)
2420
641k
  {
2421
641k
    const char *s;
2422
2423
641k
    infprintf (is, dis_style_mnemonic, "%s", op->name);
2424
641k
    if (op->args[0] != '\0')
2425
640k
      infprintf (is, dis_style_text, "\t");
2426
2427
641k
    init_print_arg_state (&state);
2428
3.39M
    for (s = op->args; *s != '\0'; s++)
2429
2.75M
      {
2430
2.75M
        if (*s == ','
2431
2.75M
      && s[1] == 'w'
2432
2.75M
      && GET_OP (full, RX) == GET_OP (full, RY))
2433
2.65k
    {
2434
      /* Skip the register and the comma.  */
2435
2.65k
      ++s;
2436
2.65k
      continue;
2437
2.65k
    }
2438
2.74M
        if (*s == ','
2439
2.74M
      && s[1] == 'v'
2440
2.74M
      && GET_OP (full, RZ) == GET_OP (full, RX))
2441
1.24k
    {
2442
      /* Skip the register and the comma.  */
2443
1.24k
      ++s;
2444
1.24k
      continue;
2445
1.24k
    }
2446
2.74M
        if (s[0] == 'N'
2447
2.74M
      && s[1] == ','
2448
2.74M
      && s[2] == 'O'
2449
2.74M
      && op->name[strlen (op->name) - 1] == '0')
2450
870
    {
2451
      /* Coprocessor register 0 with sel field.  */
2452
870
      const struct mips_cp0sel_name *n;
2453
870
      const struct mips_operand *operand;
2454
870
      unsigned int reg, sel;
2455
2456
870
      operand = decode_mips16_operand (*s, true);
2457
870
      reg = mips_extract_operand (operand, (first << 16) | second);
2458
870
      s += 2;
2459
870
      operand = decode_mips16_operand (*s, true);
2460
870
      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
870
      n = lookup_mips_cp0sel_name (mips_cp0sel_names,
2468
870
                 mips_cp0sel_names_len,
2469
870
                 reg, sel);
2470
870
      if (n != NULL)
2471
376
        infprintf (is, dis_style_register, "%s", n->name);
2472
494
      else
2473
494
        {
2474
494
          infprintf (is, dis_style_register, "$%d", reg);
2475
494
          infprintf (is, dis_style_text, ",");
2476
494
          infprintf (is, dis_style_immediate, "%d", sel);
2477
494
        }
2478
870
    }
2479
2.74M
        else
2480
2.74M
    switch (match)
2481
2.74M
      {
2482
166k
        case MATCH_FULL:
2483
166k
          print_mips16_insn_arg (info, &state, op, *s, memaddr + 2,
2484
166k
               second, true, first, s[1] == '(');
2485
166k
          break;
2486
2.57M
        case MATCH_SHORT:
2487
2.57M
          print_mips16_insn_arg (info, &state, op, *s, memaddr,
2488
2.57M
               first, false, 0, s[1] == '(');
2489
2.57M
          break;
2490
0
        case MATCH_NONE:  /* Stop the compiler complaining.  */
2491
0
          break;
2492
2.74M
      }
2493
2.74M
      }
2494
2495
    /* Figure out branch instruction type and delay slot information.  */
2496
641k
    if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0)
2497
13.2k
      info->branch_delay_insns = 1;
2498
641k
    if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0
2499
641k
        || (op->pinfo2 & INSN2_UNCOND_BRANCH) != 0)
2500
30.9k
      {
2501
30.9k
        if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2502
11.7k
    info->insn_type = dis_jsr;
2503
19.1k
        else
2504
19.1k
    info->insn_type = dis_branch;
2505
30.9k
      }
2506
610k
    else if ((op->pinfo2 & INSN2_COND_BRANCH) != 0)
2507
33.2k
      info->insn_type = dis_condbranch;
2508
2509
641k
    return match == MATCH_FULL ? 4 : 2;
2510
641k
  }
2511
54.3M
    }
2512
41.3k
#undef GET_OP
2513
2514
41.3k
  infprintf (is, dis_style_assembler_directive, ".short");
2515
41.3k
  infprintf (is, dis_style_text, "\t");
2516
41.3k
  infprintf (is, dis_style_immediate, "0x%x", first);
2517
41.3k
  info->insn_type = dis_noninsn;
2518
2519
41.3k
  return 2;
2520
682k
}
2521
2522
/* Disassemble microMIPS instructions.  */
2523
2524
static int
2525
print_insn_micromips (bfd_vma memaddr, struct disassemble_info *info)
2526
410k
{
2527
410k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2528
410k
  const struct mips_opcode *op, *opend;
2529
410k
  void *is = info->stream;
2530
410k
  bfd_byte buffer[2];
2531
410k
  unsigned int higher;
2532
410k
  unsigned int length;
2533
410k
  int status;
2534
410k
  unsigned int insn;
2535
2536
410k
  info->bytes_per_chunk = 2;
2537
410k
  info->display_endian = info->endian;
2538
410k
  info->insn_info_valid = 1;
2539
410k
  info->branch_delay_insns = 0;
2540
410k
  info->data_size = 0;
2541
410k
  info->insn_type = dis_nonbranch;
2542
410k
  info->target = 0;
2543
410k
  info->target2 = 0;
2544
2545
410k
  status = (*info->read_memory_func) (memaddr, buffer, 2, info);
2546
410k
  if (status != 0)
2547
391
    {
2548
391
      (*info->memory_error_func) (status, memaddr, info);
2549
391
      return -1;
2550
391
    }
2551
2552
410k
  length = 2;
2553
2554
410k
  if (info->endian == BFD_ENDIAN_BIG)
2555
119k
    insn = bfd_getb16 (buffer);
2556
291k
  else
2557
291k
    insn = bfd_getl16 (buffer);
2558
2559
410k
  if ((insn & 0x1c00) == 0x0000 || (insn & 0x1000) == 0x1000)
2560
279k
    {
2561
      /* This is a 32-bit microMIPS instruction.  */
2562
279k
      higher = insn;
2563
2564
279k
      status = (*info->read_memory_func) (memaddr + 2, buffer, 2, info);
2565
279k
      if (status != 0)
2566
233
  {
2567
233
    infprintf (is, dis_style_text, "micromips 0x%x", higher);
2568
233
    (*info->memory_error_func) (status, memaddr + 2, info);
2569
233
    return -1;
2570
233
  }
2571
2572
279k
      if (info->endian == BFD_ENDIAN_BIG)
2573
85.6k
  insn = bfd_getb16 (buffer);
2574
193k
      else
2575
193k
  insn = bfd_getl16 (buffer);
2576
2577
279k
      insn = insn | (higher << 16);
2578
2579
279k
      length += 2;
2580
279k
    }
2581
2582
  /* FIXME: Should probably use a hash table on the major opcode here.  */
2583
2584
410k
  opend = micromips_opcodes + bfd_micromips_num_opcodes;
2585
267M
  for (op = micromips_opcodes; op < opend; op++)
2586
267M
    {
2587
267M
      if (op->pinfo != INSN_MACRO
2588
267M
    && !(no_aliases && (op->pinfo2 & INSN2_ALIAS))
2589
267M
    && (insn & op->mask) == op->match
2590
267M
    && ((length == 2 && (op->mask & 0xffff0000) == 0)
2591
420k
        || (length == 4 && (op->mask & 0xffff0000) != 0)))
2592
334k
  {
2593
334k
    if (!validate_insn_args (op, decode_micromips_operand, insn))
2594
0
      continue;
2595
2596
334k
    infprintf (is, dis_style_mnemonic, "%s", op->name);
2597
2598
334k
    if (op->args[0])
2599
313k
      {
2600
313k
        infprintf (is, dis_style_text, "\t");
2601
313k
        print_insn_args (info, op, decode_micromips_operand, insn,
2602
313k
             memaddr + 1, length);
2603
313k
      }
2604
2605
    /* Figure out instruction type and branch delay information.  */
2606
334k
    if ((op->pinfo
2607
334k
         & (INSN_UNCOND_BRANCH_DELAY | INSN_COND_BRANCH_DELAY)) != 0)
2608
32.7k
      info->branch_delay_insns = 1;
2609
334k
    if (((op->pinfo & INSN_UNCOND_BRANCH_DELAY)
2610
334k
         | (op->pinfo2 & INSN2_UNCOND_BRANCH)) != 0)
2611
20.4k
      {
2612
20.4k
        if ((op->pinfo & (INSN_WRITE_GPR_31 | INSN_WRITE_1)) != 0)
2613
13.0k
    info->insn_type = dis_jsr;
2614
7.38k
        else
2615
7.38k
    info->insn_type = dis_branch;
2616
20.4k
      }
2617
313k
    else if (((op->pinfo & INSN_COND_BRANCH_DELAY)
2618
313k
        | (op->pinfo2 & INSN2_COND_BRANCH)) != 0)
2619
12.5k
      {
2620
12.5k
        if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2621
663
    info->insn_type = dis_condjsr;
2622
11.8k
        else
2623
11.8k
    info->insn_type = dis_condbranch;
2624
12.5k
      }
2625
301k
    else if ((op->pinfo
2626
301k
        & (INSN_STORE_MEMORY | INSN_LOAD_MEMORY)) != 0)
2627
140k
      info->insn_type = dis_dref;
2628
2629
334k
    return length;
2630
334k
  }
2631
267M
    }
2632
2633
76.1k
  infprintf (is, dis_style_assembler_directive, ".short");
2634
76.1k
  infprintf (is, dis_style_text, "\t");
2635
76.1k
  if (length != 2)
2636
64.1k
    {
2637
64.1k
      infprintf (is, dis_style_immediate, "0x%x", (insn >> 16) & 0xffff);
2638
64.1k
      infprintf (is, dis_style_text, ", ");
2639
64.1k
    }
2640
76.1k
  infprintf (is, dis_style_immediate, "0x%x", (insn & 0xffff));
2641
2642
76.1k
  info->insn_type = dis_noninsn;
2643
2644
76.1k
  return length;
2645
410k
}
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
4.15M
{
2658
4.15M
  int i;
2659
4.15M
  int l;
2660
2661
6.68M
  for (i = info->symtab_pos, l = i + info->num_symbols; i < l; i++)
2662
2.52M
    if (((info->symtab[i])->flags & BSF_SYNTHETIC) != 0
2663
2.52M
  && ((!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
2.52M
    else if (bfd_asymbol_flavour (info->symtab[i]) == bfd_target_elf_flavour
2669
2.52M
        && info->symtab[i]->section == info->section)
2670
2.52M
      {
2671
2.52M
  elf_symbol_type *symbol = (elf_symbol_type *) info->symtab[i];
2672
2.52M
  if ((!micromips_p
2673
2.52M
       && ELF_ST_IS_MIPS16 (symbol->internal_elf_sym.st_other))
2674
2.52M
      || (micromips_p
2675
2.52M
    && ELF_ST_IS_MICROMIPS (symbol->internal_elf_sym.st_other)))
2676
129
    return 1;
2677
2.52M
      }
2678
2679
4.15M
  return 0;
2680
4.15M
}
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
3.17M
{
2693
3.17M
  bfd_byte buffer[INSNLEN];
2694
3.17M
  int status;
2695
2696
3.17M
  set_default_mips_dis_options (info);
2697
3.17M
  parse_mips_dis_options (info->disassembler_options);
2698
2699
3.17M
  if (info->mach == bfd_mach_mips16)
2700
319k
    return print_insn_mips16 (memaddr, info);
2701
2.85M
  if (info->mach == bfd_mach_mips_micromips)
2702
259k
    return print_insn_micromips (memaddr, info);
2703
2704
2.59M
#if 1
2705
  /* FIXME: If odd address, this is CLEARLY a compressed instruction.  */
2706
  /* Only a few tools will work this way.  */
2707
2.59M
  if (memaddr & 0x01)
2708
514k
    {
2709
514k
      if (micromips_ase)
2710
151k
  return print_insn_micromips (memaddr, info);
2711
363k
      else
2712
363k
  return print_insn_mips16 (memaddr, info);
2713
514k
    }
2714
2.07M
#endif
2715
2716
2.07M
#if SYMTAB_AVAILABLE
2717
2.07M
  if (is_compressed_mode_p (info, true))
2718
0
    return print_insn_micromips (memaddr, info);
2719
2.07M
  if (is_compressed_mode_p (info, false))
2720
129
    return print_insn_mips16 (memaddr, info);
2721
2.07M
#endif
2722
2723
2.07M
  status = (*info->read_memory_func) (memaddr, buffer, INSNLEN, info);
2724
2.07M
  if (status == 0)
2725
2.07M
    {
2726
2.07M
      int insn;
2727
2728
2.07M
      if (endianness == BFD_ENDIAN_BIG)
2729
1.75M
  insn = bfd_getb32 (buffer);
2730
324k
      else
2731
324k
  insn = bfd_getl32 (buffer);
2732
2733
2.07M
      return print_insn_mips (memaddr, insn, info);
2734
2.07M
    }
2735
1.55k
  else
2736
1.55k
    {
2737
1.55k
      (*info->memory_error_func) (status, memaddr, info);
2738
1.55k
      return -1;
2739
1.55k
    }
2740
2.07M
}
2741
2742
int
2743
print_insn_big_mips (bfd_vma memaddr, struct disassemble_info *info)
2744
2.09M
{
2745
2.09M
  return _print_insn_mips (memaddr, info, BFD_ENDIAN_BIG);
2746
2.09M
}
2747
2748
int
2749
print_insn_little_mips (bfd_vma memaddr, struct disassemble_info *info)
2750
1.08M
{
2751
1.08M
  return _print_insn_mips (memaddr, info, BFD_ENDIAN_LITTLE);
2752
1.08M
}
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
0
  fprintf (stream, _("\n"));
2946
0
}