Coverage Report

Created: 2026-05-11 07:54

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/opcodes/mips-dis.c
Line
Count
Source
1
/* Print mips instructions for GDB, the GNU debugger, or for objdump.
2
   Copyright (C) 1989-2026 Free Software Foundation, Inc.
3
   Contributed by Nobuyuki Hikichi(hikichi@sra.co.jp).
4
5
   This file is part of the GNU opcodes library.
6
7
   This library is free software; you can redistribute it and/or modify
8
   it under the terms of the GNU General Public License as published by
9
   the Free Software Foundation; either version 3, or (at your option)
10
   any later version.
11
12
   It is distributed in the hope that it will be useful, but WITHOUT
13
   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
14
   or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
15
   License for more details.
16
17
   You should have received a copy of the GNU General Public License
18
   along with this program; if not, write to the Free Software
19
   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
20
   MA 02110-1301, USA.  */
21
22
#include "sysdep.h"
23
#include "disassemble.h"
24
#include "libiberty.h"
25
#include "opcode/mips.h"
26
#include "opintl.h"
27
#include "elf-bfd.h"
28
#include "elf/mips.h"
29
#include "elfxx-mips.h"
30
31
/* FIXME: These are needed to figure out if the code is mips16 or
32
   not. The low bit of the address is often a good indicator.  No
33
   symbol table is available when this code runs out in an embedded
34
   system as when it is used for disassembler support in a monitor.  */
35
36
#if !defined(EMBEDDED_ENV)
37
#define SYMTAB_AVAILABLE 1
38
#endif
39
40
/* Mips instructions are at maximum this many bytes long.  */
41
1.39M
#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
9.58k
{
768
9.58k
  const struct mips_abi_choice *c;
769
9.58k
  unsigned int i;
770
771
73.1k
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_abi_choices) && c == NULL; i++)
772
63.5k
    if (strncmp (mips_abi_choices[i].name, name, namelen) == 0
773
4.40k
  && strlen (mips_abi_choices[i].name) == namelen)
774
952
      c = &mips_abi_choices[i];
775
776
9.58k
  return c;
777
9.58k
}
778
779
static const struct mips_arch_choice *
780
choose_arch_by_name (const char *name, unsigned int namelen)
781
16.3k
{
782
16.3k
  const struct mips_arch_choice *c = NULL;
783
16.3k
  unsigned int i;
784
785
757k
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_arch_choices) && c == NULL; i++)
786
741k
    if (strncmp (mips_arch_choices[i].name, name, namelen) == 0
787
14.4k
  && strlen (mips_arch_choices[i].name) == namelen)
788
7.50k
      c = &mips_arch_choices[i];
789
790
16.3k
  return c;
791
16.3k
}
792
793
static const struct mips_arch_choice *
794
choose_arch_by_number (unsigned long mach)
795
1.43M
{
796
1.43M
  static unsigned long hint_bfd_mach;
797
1.43M
  static const struct mips_arch_choice *hint_arch_choice;
798
1.43M
  const struct mips_arch_choice *c;
799
1.43M
  unsigned int i;
800
801
  /* We optimize this because even if the user specifies no
802
     flags, this will be done for every instruction!  */
803
1.43M
  if (hint_bfd_mach == mach
804
1.20M
      && hint_arch_choice != NULL
805
1.20M
      && hint_arch_choice->bfd_mach == hint_bfd_mach)
806
1.20M
    return hint_arch_choice;
807
808
12.3M
  for (i = 0, c = NULL; i < ARRAY_SIZE (mips_arch_choices) && c == NULL; i++)
809
12.1M
    {
810
12.1M
      if (mips_arch_choices[i].bfd_mach_valid
811
11.8M
    && mips_arch_choices[i].bfd_mach == mach)
812
2.21k
  {
813
2.21k
    c = &mips_arch_choices[i];
814
2.21k
    hint_bfd_mach = mach;
815
2.21k
    hint_arch_choice = c;
816
2.21k
  }
817
12.1M
    }
818
238k
  return c;
819
1.43M
}
820
821
/* Check if the object uses NewABI conventions.  */
822
823
static int
824
is_newabi (Elf_Internal_Ehdr *header)
825
629k
{
826
  /* There are no old-style ABIs which use 64-bit ELF.  */
827
629k
  if (header->e_ident[EI_CLASS] == ELFCLASS64)
828
178k
    return 1;
829
830
  /* If a 32-bit ELF file, n32 is a new-style ABI.  */
831
451k
  if ((header->e_flags & EF_MIPS_ABI2) != 0)
832
7.32k
    return 1;
833
834
444k
  return 0;
835
451k
}
836
837
/* Check if the object has microMIPS ASE code.  */
838
839
static int
840
is_micromips (Elf_Internal_Ehdr *header)
841
629k
{
842
629k
  if ((header->e_flags & EF_MIPS_ARCH_ASE_MICROMIPS) != 0)
843
462k
    return 1;
844
845
166k
  return 0;
846
629k
}
847
848
/* Convert ASE flags from .MIPS.abiflags to internal values.  */
849
850
static unsigned long
851
mips_convert_abiflags_ases (unsigned long afl_ases)
852
0
{
853
0
  unsigned long opcode_ases = 0;
854
855
0
  if (afl_ases & AFL_ASE_DSP)
856
0
    opcode_ases |= ASE_DSP;
857
0
  if (afl_ases & AFL_ASE_DSPR2)
858
0
    opcode_ases |= ASE_DSPR2;
859
0
  if (afl_ases & AFL_ASE_EVA)
860
0
    opcode_ases |= ASE_EVA;
861
0
  if (afl_ases & AFL_ASE_MCU)
862
0
    opcode_ases |= ASE_MCU;
863
0
  if (afl_ases & AFL_ASE_MDMX)
864
0
    opcode_ases |= ASE_MDMX;
865
0
  if (afl_ases & AFL_ASE_MIPS3D)
866
0
    opcode_ases |= ASE_MIPS3D;
867
0
  if (afl_ases & AFL_ASE_MT)
868
0
    opcode_ases |= ASE_MT;
869
0
  if (afl_ases & AFL_ASE_SMARTMIPS)
870
0
    opcode_ases |= ASE_SMARTMIPS;
871
0
  if (afl_ases & AFL_ASE_VIRT)
872
0
    opcode_ases |= ASE_VIRT;
873
0
  if (afl_ases & AFL_ASE_MSA)
874
0
    opcode_ases |= ASE_MSA;
875
0
  if (afl_ases & AFL_ASE_XPA)
876
0
    opcode_ases |= ASE_XPA;
877
0
  if (afl_ases & AFL_ASE_DSPR3)
878
0
    opcode_ases |= ASE_DSPR3;
879
0
  if (afl_ases & AFL_ASE_MIPS16E2)
880
0
    opcode_ases |= ASE_MIPS16E2;
881
0
  return opcode_ases;
882
0
}
883
884
/* Calculate combination ASE flags from regular ASE flags.  */
885
886
static unsigned long
887
mips_calculate_combination_ases (int opcode_isa, unsigned long opcode_ases)
888
1.46M
{
889
1.46M
  unsigned long combination_ases = 0;
890
891
1.46M
  if ((opcode_ases & (ASE_XPA | ASE_VIRT)) == (ASE_XPA | ASE_VIRT))
892
201k
    combination_ases |= ASE_XPA_VIRT;
893
1.46M
  if ((opcode_ases & (ASE_MIPS16E2 | ASE_MT)) == (ASE_MIPS16E2 | ASE_MT))
894
3.43k
    combination_ases |= ASE_MIPS16E2_MT;
895
1.46M
  if ((opcode_ases & ASE_EVA)
896
205k
      && ((opcode_isa & INSN_ISA_MASK) == ISA_MIPS64R6
897
138k
    || (opcode_isa & INSN_ISA_MASK) == ISA_MIPS32R6))
898
176k
    combination_ases |= ASE_EVA_R6;
899
1.46M
  return combination_ases;
900
1.46M
}
901
902
static void
903
set_default_mips_dis_options (struct disassemble_info *info)
904
1.43M
{
905
1.43M
  const struct mips_arch_choice *chosen_arch;
906
907
  /* Defaults: mipsIII/r3000 (?!), no microMIPS ASE (any compressed code
908
     is MIPS16 ASE) (o)32-style ("oldabi") GPR names, and numeric FPR,
909
     CP0 register, and HWR names.  */
910
1.43M
  mips_isa = ISA_MIPS3;
911
1.43M
  mips_processor = CPU_R3000;
912
1.43M
  micromips_ase = 0;
913
1.43M
  mips_ase = 0;
914
1.43M
  mips_gpr_names = mips_gpr_names_oldabi;
915
1.43M
  mips_fpr_names = mips_fpr_names_numeric;
916
1.43M
  mips_cp0_names = mips_cp0_names_numeric;
917
1.43M
  mips_cp0sel_names = NULL;
918
1.43M
  mips_cp0sel_names_len = 0;
919
1.43M
  mips_cp1_names = mips_cp1_names_numeric;
920
1.43M
  mips_hwr_names = mips_hwr_names_numeric;
921
1.43M
  no_aliases = 0;
922
923
  /* Set ISA, architecture, and cp0 register names as best we can.  */
924
#if ! SYMTAB_AVAILABLE
925
  /* This is running out on a target machine, not in a host tool.
926
     FIXME: Where does mips_target_info come from?  */
927
  target_processor = mips_target_info.processor;
928
  mips_isa = mips_target_info.isa;
929
  mips_ase = mips_target_info.ase;
930
#else
931
1.43M
  chosen_arch = choose_arch_by_number (info->mach);
932
1.43M
  if (chosen_arch != NULL)
933
1.20M
    {
934
1.20M
      mips_processor = chosen_arch->processor;
935
1.20M
      mips_isa = chosen_arch->isa;
936
1.20M
      mips_ase = chosen_arch->ase;
937
1.20M
      mips_cp0_names = chosen_arch->cp0_names;
938
1.20M
      mips_cp0sel_names = chosen_arch->cp0sel_names;
939
1.20M
      mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
940
1.20M
      mips_cp1_names = chosen_arch->cp1_names;
941
1.20M
      mips_hwr_names = chosen_arch->hwr_names;
942
1.20M
    }
943
944
  /* Update settings according to the ELF file header flags.  */
945
1.43M
  if (info->flavour == bfd_target_elf_flavour && info->section != NULL)
946
629k
    {
947
629k
      struct bfd *abfd = info->section->owner;
948
629k
      Elf_Internal_Ehdr *header = elf_elfheader (abfd);
949
629k
      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
629k
#ifdef HAVE_BFD_MIPS_ELF_GET_ABIFLAGS
957
629k
      abiflags = bfd_mips_elf_get_abiflags (abfd);
958
629k
#endif
959
      /* If an ELF "newabi" binary, use the n32/(n)64 GPR names.  */
960
629k
      if (is_newabi (header))
961
185k
  mips_gpr_names = mips_gpr_names_newabi;
962
      /* If a microMIPS binary, then don't use MIPS16 bindings.  */
963
629k
      micromips_ase = is_micromips (header);
964
      /* OR in any extra ASE flags set in ELF file structures.  */
965
629k
      if (abiflags)
966
0
  mips_ase |= mips_convert_abiflags_ases (abiflags->ases);
967
629k
      else if (header->e_flags & EF_MIPS_ARCH_ASE_MDMX)
968
171k
  mips_ase |= ASE_MDMX;
969
629k
    }
970
1.43M
#endif
971
1.43M
  mips_ase |= mips_calculate_combination_ases (mips_isa, mips_ase);
972
1.43M
}
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
233k
{
980
233k
  if (startswith (option, "msa"))
981
7.13k
    {
982
7.13k
      mips_ase |= ASE_MSA;
983
7.13k
      if ((mips_isa & INSN_ISA_MASK) == ISA_MIPS64R2
984
6.87k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R3
985
6.67k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R5
986
6.46k
     || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R6)
987
1.84k
    mips_ase |= ASE_MSA64;
988
7.13k
      return true;
989
7.13k
    }
990
991
226k
  if (startswith (option, "virt"))
992
2.31k
    {
993
2.31k
      mips_ase |= ASE_VIRT;
994
2.31k
      if (mips_isa & ISA_MIPS64R2
995
1.73k
    || mips_isa & ISA_MIPS64R3
996
1.31k
    || mips_isa & ISA_MIPS64R5
997
0
    || mips_isa & ISA_MIPS64R6)
998
2.31k
  mips_ase |= ASE_VIRT64;
999
2.31k
      return true;
1000
2.31k
    }
1001
1002
224k
  if (startswith (option, "xpa"))
1003
1.15k
    {
1004
1.15k
      mips_ase |= ASE_XPA;
1005
1.15k
      return true;
1006
1.15k
    }
1007
1008
223k
  if (startswith (option, "ginv"))
1009
1.51k
    {
1010
1.51k
      mips_ase |= ASE_GINV;
1011
1.51k
      return true;
1012
1.51k
    }
1013
1014
221k
  if (startswith (option, "loongson-mmi"))
1015
5.75k
    {
1016
5.75k
      mips_ase |= ASE_LOONGSON_MMI;
1017
5.75k
      return true;
1018
5.75k
    }
1019
1020
215k
  if (startswith (option, "loongson-cam"))
1021
564
    {
1022
564
      mips_ase |= ASE_LOONGSON_CAM;
1023
564
      return true;
1024
564
    }
1025
  
1026
  /* Put here for match ext2 frist */
1027
215k
  if (startswith (option, "loongson-ext2"))
1028
500
    {
1029
500
      mips_ase |= ASE_LOONGSON_EXT2;
1030
500
      return true;
1031
500
    }
1032
1033
214k
  if (startswith (option, "loongson-ext"))
1034
4.85k
    {
1035
4.85k
      mips_ase |= ASE_LOONGSON_EXT;
1036
4.85k
      return true;
1037
4.85k
    }
1038
1039
210k
  return false;
1040
214k
}
1041
1042
static void
1043
parse_mips_dis_option (const char *option, unsigned int len)
1044
236k
{
1045
236k
  unsigned int i, optionlen, vallen;
1046
236k
  const char *val;
1047
236k
  const struct mips_abi_choice *chosen_abi;
1048
236k
  const struct mips_arch_choice *chosen_arch;
1049
1050
  /* Try to match options that are simple flags */
1051
236k
  if (startswith (option, "no-aliases"))
1052
2.25k
    {
1053
2.25k
      no_aliases = 1;
1054
2.25k
      return;
1055
2.25k
    }
1056
1057
233k
  if (parse_mips_ase_option (option))
1058
23.7k
    {
1059
23.7k
      mips_ase |= mips_calculate_combination_ases (mips_isa, mips_ase);
1060
23.7k
      return;
1061
23.7k
    }
1062
1063
  /* Look for the = that delimits the end of the option name.  */
1064
3.92M
  for (i = 0; i < len; i++)
1065
3.77M
    if (option[i] == '=')
1066
62.4k
      break;
1067
1068
210k
  if (i == 0)    /* Invalid option: no name before '='.  */
1069
1.67k
    return;
1070
208k
  if (i == len)    /* Invalid option: no '='.  */
1071
147k
    return;
1072
60.8k
  if (i == (len - 1))  /* Invalid option: no value after '='.  */
1073
1.92k
    return;
1074
1075
58.8k
  optionlen = i;
1076
58.8k
  val = option + (optionlen + 1);
1077
58.8k
  vallen = len - (optionlen + 1);
1078
1079
58.8k
  if (strncmp ("gpr-names", option, optionlen) == 0
1080
1.78k
      && strlen ("gpr-names") == optionlen)
1081
1.48k
    {
1082
1.48k
      chosen_abi = choose_abi_by_name (val, vallen);
1083
1.48k
      if (chosen_abi != NULL)
1084
208
  mips_gpr_names = chosen_abi->gpr_names;
1085
1.48k
      return;
1086
1.48k
    }
1087
1088
57.3k
  if (strncmp ("fpr-names", option, optionlen) == 0
1089
2.40k
      && strlen ("fpr-names") == optionlen)
1090
1.90k
    {
1091
1.90k
      chosen_abi = choose_abi_by_name (val, vallen);
1092
1.90k
      if (chosen_abi != NULL)
1093
276
  mips_fpr_names = chosen_abi->fpr_names;
1094
1.90k
      return;
1095
1.90k
    }
1096
1097
55.4k
  if (strncmp ("cp0-names", option, optionlen) == 0
1098
4.21k
      && strlen ("cp0-names") == optionlen)
1099
3.71k
    {
1100
3.71k
      chosen_arch = choose_arch_by_name (val, vallen);
1101
3.71k
      if (chosen_arch != NULL)
1102
846
  {
1103
846
    mips_cp0_names = chosen_arch->cp0_names;
1104
846
    mips_cp0sel_names = chosen_arch->cp0sel_names;
1105
846
    mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
1106
846
  }
1107
3.71k
      return;
1108
3.71k
    }
1109
1110
51.7k
  if (strncmp ("cp1-names", option, optionlen) == 0
1111
6.03k
      && strlen ("cp1-names") == optionlen)
1112
5.48k
    {
1113
5.48k
      chosen_arch = choose_arch_by_name (val, vallen);
1114
5.48k
      if (chosen_arch != NULL)
1115
1.60k
  mips_cp1_names = chosen_arch->cp1_names;
1116
5.48k
      return;
1117
5.48k
    }
1118
1119
46.2k
  if (strncmp ("hwr-names", option, optionlen) == 0
1120
1.17k
      && strlen ("hwr-names") == optionlen)
1121
926
    {
1122
926
      chosen_arch = choose_arch_by_name (val, vallen);
1123
926
      if (chosen_arch != NULL)
1124
286
  mips_hwr_names = chosen_arch->hwr_names;
1125
926
      return;
1126
926
    }
1127
1128
45.3k
  if (strncmp ("reg-names", option, optionlen) == 0
1129
6.50k
      && strlen ("reg-names") == optionlen)
1130
6.20k
    {
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
6.20k
      chosen_abi = choose_abi_by_name (val, vallen);
1136
6.20k
      if (chosen_abi != NULL)
1137
468
  {
1138
468
    mips_gpr_names = chosen_abi->gpr_names;
1139
468
    mips_fpr_names = chosen_abi->fpr_names;
1140
468
  }
1141
6.20k
      chosen_arch = choose_arch_by_name (val, vallen);
1142
6.20k
      if (chosen_arch != NULL)
1143
4.77k
  {
1144
4.77k
    mips_cp0_names = chosen_arch->cp0_names;
1145
4.77k
    mips_cp0sel_names = chosen_arch->cp0sel_names;
1146
4.77k
    mips_cp0sel_names_len = chosen_arch->cp0sel_names_len;
1147
4.77k
    mips_cp1_names = chosen_arch->cp1_names;
1148
4.77k
    mips_hwr_names = chosen_arch->hwr_names;
1149
4.77k
  }
1150
6.20k
      return;
1151
6.20k
    }
1152
1153
  /* Invalid option.  */
1154
45.3k
}
1155
1156
static void
1157
parse_mips_dis_options (const char *options)
1158
1.43M
{
1159
1.43M
  const char *option_end;
1160
1161
1.43M
  if (options == NULL)
1162
1.30M
    return;
1163
1164
483k
  while (*options != '\0')
1165
354k
    {
1166
      /* Skip empty options.  */
1167
354k
      if (*options == ',')
1168
118k
  {
1169
118k
    options++;
1170
118k
    continue;
1171
118k
  }
1172
1173
      /* We know that *options is neither NUL or a comma.  */
1174
236k
      option_end = options + 1;
1175
4.83M
      while (*option_end != ',' && *option_end != '\0')
1176
4.60M
  option_end++;
1177
1178
236k
      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
236k
      options = option_end;
1183
236k
    }
1184
129k
}
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
3.24k
{
1192
3.24k
  unsigned int i;
1193
1194
51.3k
  for (i = 0; i < len; i++)
1195
48.6k
    if (names[i].cp0reg == cp0reg && names[i].sel == sel)
1196
565
      return &names[i];
1197
2.67k
  return NULL;
1198
3.24k
}
1199
1200
/* Print register REGNO, of type TYPE, for instruction OPCODE.  */
1201
1202
static void
1203
print_reg (struct disassemble_info *info, const struct mips_opcode *opcode,
1204
     enum mips_reg_operand_type type, int regno)
1205
1.82M
{
1206
1.82M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1207
1208
1.82M
  switch (type)
1209
1.82M
    {
1210
1.72M
    case OP_REG_GP:
1211
1.72M
      infprintf (info->stream, dis_style_register, "%s",
1212
1.72M
     mips_gpr_names[regno]);
1213
1.72M
      break;
1214
1215
33.4k
    case OP_REG_FP:
1216
33.4k
      infprintf (info->stream, dis_style_register, "%s",
1217
33.4k
     mips_fpr_names[regno]);
1218
33.4k
      break;
1219
1220
1.63k
    case OP_REG_CCC:
1221
1.63k
      if (opcode->pinfo & (FP_D | FP_S))
1222
1.32k
  infprintf (info->stream, dis_style_register, "$fcc%d", regno);
1223
318
      else
1224
318
  infprintf (info->stream, dis_style_register, "$cc%d", regno);
1225
1.63k
      break;
1226
1227
5.25k
    case OP_REG_VEC:
1228
5.25k
      if (opcode->membership & INSN_5400)
1229
1.14k
  infprintf (info->stream, dis_style_register, "$f%d", regno);
1230
4.10k
      else
1231
4.10k
  infprintf (info->stream, dis_style_register, "$v%d", regno);
1232
5.25k
      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
21.0k
    case OP_REG_COPRO:
1239
21.0k
      if (opcode->name[strlen (opcode->name) - 1] == '0')
1240
2.95k
  infprintf (info->stream, dis_style_register, "%s", mips_cp0_names[regno]);
1241
18.1k
      else
1242
18.1k
  infprintf (info->stream, dis_style_register, "$%d", regno);
1243
21.0k
      break;
1244
1245
1.40k
    case OP_REG_CONTROL:
1246
1.40k
      if (opcode->name[strlen (opcode->name) - 1] == '1')
1247
324
  infprintf (info->stream, dis_style_register, "%s", mips_cp1_names[regno]);
1248
1.07k
      else
1249
1.07k
  infprintf (info->stream, dis_style_register, "$%d", regno);
1250
1.40k
      break;
1251
1252
562
    case OP_REG_HW:
1253
562
      infprintf (info->stream, dis_style_register, "%s", mips_hwr_names[regno]);
1254
562
      break;
1255
1256
8.86k
    case OP_REG_VF:
1257
8.86k
      infprintf (info->stream, dis_style_register, "$vf%d", regno);
1258
8.86k
      break;
1259
1260
542
    case OP_REG_VI:
1261
542
      infprintf (info->stream, dis_style_register, "$vi%d", regno);
1262
542
      break;
1263
1264
343
    case OP_REG_R5900_I:
1265
343
      infprintf (info->stream, dis_style_register, "$I");
1266
343
      break;
1267
1268
375
    case OP_REG_R5900_Q:
1269
375
      infprintf (info->stream, dis_style_register, "$Q");
1270
375
      break;
1271
1272
195
    case OP_REG_R5900_R:
1273
195
      infprintf (info->stream, dis_style_register, "$R");
1274
195
      break;
1275
1276
347
    case OP_REG_R5900_ACC:
1277
347
      infprintf (info->stream, dis_style_register, "$ACC");
1278
347
      break;
1279
1280
19.5k
    case OP_REG_MSA:
1281
19.5k
      infprintf (info->stream, dis_style_register, "$w%d", regno);
1282
19.5k
      break;
1283
1284
442
    case OP_REG_MSA_CTRL:
1285
442
      infprintf (info->stream, dis_style_register, "%s",
1286
442
     msa_control_names[regno]);
1287
442
      break;
1288
1289
1.82M
    }
1290
1.82M
}
1291

1292
/* Used to track the state carried over from previous operands in
1293
   an instruction.  */
1294
struct mips_print_arg_state {
1295
  /* The value of the last OP_INT seen.  We only use this for OP_MSB,
1296
     where the value is known to be unsigned and small.  */
1297
  unsigned int last_int;
1298
1299
  /* The type and number of the last OP_REG seen.  We only use this for
1300
     OP_REPEAT_DEST_REG and OP_REPEAT_PREV_REG.  */
1301
  enum mips_reg_operand_type last_reg_type;
1302
  unsigned int last_regno;
1303
  unsigned int dest_regno;
1304
  unsigned int seen_dest;
1305
};
1306
1307
/* Initialize STATE for the start of an instruction.  */
1308
1309
static inline void
1310
init_print_arg_state (struct mips_print_arg_state *state)
1311
1.77M
{
1312
1.77M
  memset (state, 0, sizeof (*state));
1313
1.77M
}
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
11.3k
{
1323
11.3k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1324
1325
11.3k
  if (operand->size == 4)
1326
8.48k
    infprintf (info->stream, style, "%s%s%s%s",
1327
8.48k
      uval & 8 ? "x" : "",
1328
8.48k
      uval & 4 ? "y" : "",
1329
8.48k
      uval & 2 ? "z" : "",
1330
8.48k
      uval & 1 ? "w" : "");
1331
2.81k
  else if (operand->size == 2)
1332
2.81k
    infprintf (info->stream, style, "%c", "xyzw"[uval]);
1333
0
  else
1334
0
    abort ();
1335
11.3k
}
1336
1337
/* Record information about a register operand.  */
1338
1339
static void
1340
mips_seen_register (struct mips_print_arg_state *state,
1341
        unsigned int regno,
1342
        enum mips_reg_operand_type reg_type)
1343
2.85M
{
1344
2.85M
  state->last_reg_type = reg_type;
1345
2.85M
  state->last_regno = regno;
1346
1347
2.85M
  if (!state->seen_dest)
1348
1.51M
    {
1349
1.51M
      state->seen_dest = 1;
1350
1.51M
      state->dest_regno = regno;
1351
1.51M
    }
1352
2.85M
}
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
8.26k
{
1365
8.26k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1366
8.26k
  unsigned int nargs, nstatics, smask, i, j;
1367
8.26k
  void *is = info->stream;
1368
8.26k
  const char *sep;
1369
1370
8.26k
  if (amask == MIPS_SVRS_ALL_ARGS)
1371
247
    {
1372
247
      nargs = 4;
1373
247
      nstatics = 0;
1374
247
    }
1375
8.01k
  else if (amask == MIPS_SVRS_ALL_STATICS)
1376
480
    {
1377
480
      nargs = 0;
1378
480
      nstatics = 4;
1379
480
    }
1380
7.53k
  else
1381
7.53k
    {
1382
7.53k
      nargs = amask >> 2;
1383
7.53k
      nstatics = amask & 3;
1384
7.53k
    }
1385
1386
8.26k
  sep = "";
1387
8.26k
  if (nargs > 0)
1388
1.50k
    {
1389
1.50k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[4]);
1390
1.50k
      if (nargs > 1)
1391
746
  infprintf (is, dis_style_register, "-%s", mips_gpr_names[4 + nargs - 1]);
1392
1.50k
      sep = ",";
1393
1.50k
    }
1394
1395
8.26k
  infprintf (is, dis_style_text, "%s", sep);
1396
8.26k
  infprintf (is, dis_style_immediate, "%d", frame_size);
1397
1398
8.26k
  if (ra)      /* $ra */
1399
5.37k
    {
1400
5.37k
      infprintf (is, dis_style_text, ",");
1401
5.37k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1402
5.37k
    }
1403
1404
8.26k
  smask = 0;
1405
8.26k
  if (s0)      /* $s0 */
1406
5.85k
    smask |= 1 << 0;
1407
8.26k
  if (s1)      /* $s1 */
1408
4.25k
    smask |= 1 << 1;
1409
8.26k
  if (nsreg > 0)    /* $s2-$s8 */
1410
2.35k
    smask |= ((1 << nsreg) - 1) << 2;
1411
1412
61.6k
  for (i = 0; i < 9; i++)
1413
53.3k
    if (smask & (1 << i))
1414
6.69k
      {
1415
6.69k
  infprintf (is, dis_style_text, ",");
1416
6.69k
  infprintf (is, dis_style_register, "%s",
1417
6.69k
       mips_gpr_names[i == 8 ? 30 : (16 + i)]);
1418
  /* Skip over string of set bits.  */
1419
21.4k
  for (j = i; smask & (2 << j); j++)
1420
14.7k
    continue;
1421
6.69k
  if (j > i)
1422
4.59k
    {
1423
4.59k
      infprintf (is, dis_style_text, "-");
1424
4.59k
      infprintf (is, dis_style_register, "%s",
1425
4.59k
           mips_gpr_names[j == 8 ? 30 : (16 + j)]);
1426
4.59k
    }
1427
6.69k
  i = j + 1;
1428
6.69k
      }
1429
  /* Statics $ax - $a3.  */
1430
8.26k
  if (nstatics == 1)
1431
838
    {
1432
838
      infprintf (is, dis_style_text, ",");
1433
838
      infprintf (is, dis_style_register, "%s", mips_gpr_names[7]);
1434
838
    }
1435
7.42k
  else if (nstatics > 0)
1436
1.34k
    {
1437
1.34k
      infprintf (is, dis_style_text, ",");
1438
1.34k
      infprintf (is, dis_style_register, "%s",
1439
1.34k
     mips_gpr_names[7 - nstatics + 1]);
1440
1.34k
      infprintf (is, dis_style_text, "-");
1441
1.34k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[7]);
1442
1.34k
    }
1443
8.26k
}
1444
1445
1446
/* Print operand OPERAND of OPCODE, using STATE to track inter-operand state.
1447
   UVAL is the encoding of the operand (shifted into bit 0) and BASE_PC is
1448
   the base address for OP_PCREL operands.  */
1449
1450
static void
1451
print_insn_arg (struct disassemble_info *info,
1452
    struct mips_print_arg_state *state,
1453
    const struct mips_opcode *opcode,
1454
    const struct mips_operand *operand,
1455
    bfd_vma base_pc,
1456
    unsigned int uval)
1457
2.83M
{
1458
2.83M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1459
2.83M
  void *is = info->stream;
1460
1461
2.83M
  switch (operand->type)
1462
2.83M
    {
1463
795k
    case OP_INT:
1464
795k
      {
1465
795k
  const struct mips_int_operand *int_op;
1466
1467
795k
  int_op = (const struct mips_int_operand *) operand;
1468
795k
  uval = mips_decode_int_operand (int_op, uval);
1469
795k
  state->last_int = uval;
1470
795k
  if (int_op->print_hex)
1471
141k
    infprintf (is, dis_style_immediate, "0x%x", uval);
1472
653k
  else
1473
653k
    infprintf (is, dis_style_immediate, "%d", uval);
1474
795k
      }
1475
795k
      break;
1476
1477
11.7k
    case OP_MAPPED_INT:
1478
11.7k
      {
1479
11.7k
  const struct mips_mapped_int_operand *mint_op;
1480
1481
11.7k
  mint_op = (const struct mips_mapped_int_operand *) operand;
1482
11.7k
  uval = mint_op->int_map[uval];
1483
11.7k
  state->last_int = uval;
1484
11.7k
  if (mint_op->print_hex)
1485
6.48k
    infprintf (is, dis_style_immediate, "0x%x", uval);
1486
5.26k
  else
1487
5.26k
    infprintf (is, dis_style_immediate, "%d", uval);
1488
11.7k
      }
1489
11.7k
      break;
1490
1491
4.50k
    case OP_MSB:
1492
4.50k
      {
1493
4.50k
  const struct mips_msb_operand *msb_op;
1494
1495
4.50k
  msb_op = (const struct mips_msb_operand *) operand;
1496
4.50k
  uval += msb_op->bias;
1497
4.50k
  if (msb_op->add_lsb)
1498
2.21k
    uval -= state->last_int;
1499
4.50k
  infprintf (is, dis_style_immediate, "0x%x", uval);
1500
4.50k
      }
1501
4.50k
      break;
1502
1503
1.57M
    case OP_REG:
1504
1.77M
    case OP_OPTIONAL_REG:
1505
1.77M
      {
1506
1.77M
  const struct mips_reg_operand *reg_op;
1507
1508
1.77M
  reg_op = (const struct mips_reg_operand *) operand;
1509
1.77M
  uval = mips_decode_reg_operand (reg_op, uval);
1510
1.77M
  print_reg (info, opcode, reg_op->reg_type, uval);
1511
1512
1.77M
  mips_seen_register (state, uval, reg_op->reg_type);
1513
1.77M
      }
1514
1.77M
      break;
1515
1516
4.46k
    case OP_REG_PAIR:
1517
4.46k
      {
1518
4.46k
  const struct mips_reg_pair_operand *pair_op;
1519
1520
4.46k
  pair_op = (const struct mips_reg_pair_operand *) operand;
1521
4.46k
  print_reg (info, opcode, pair_op->reg_type,
1522
4.46k
       pair_op->reg1_map[uval]);
1523
4.46k
  infprintf (is, dis_style_text, ",");
1524
4.46k
  print_reg (info, opcode, pair_op->reg_type,
1525
4.46k
       pair_op->reg2_map[uval]);
1526
4.46k
      }
1527
4.46k
      break;
1528
1529
178k
    case OP_PCREL:
1530
178k
      {
1531
178k
  const struct mips_pcrel_operand *pcrel_op;
1532
1533
178k
  pcrel_op = (const struct mips_pcrel_operand *) operand;
1534
178k
  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
178k
  if (pcrel_op->include_isa_bit
1539
153k
      && info->flavour != bfd_target_unknown_flavour)
1540
135k
    info->target &= -2;
1541
1542
178k
  (*info->print_address_func) (info->target, info);
1543
178k
      }
1544
178k
      break;
1545
1546
290
    case OP_PERF_REG:
1547
290
      infprintf (is, dis_style_register, "%d", uval);
1548
290
      break;
1549
1550
2.81k
    case OP_ADDIUSP_INT:
1551
2.81k
      {
1552
2.81k
  int sval;
1553
1554
2.81k
  sval = mips_signed_operand (operand, uval) * 4;
1555
2.81k
  if (sval >= -8 && sval < 8)
1556
320
    sval ^= 0x400;
1557
2.81k
  infprintf (is, dis_style_immediate, "%d", sval);
1558
2.81k
  break;
1559
1.57M
      }
1560
1561
1.10k
    case OP_CLO_CLZ_DEST:
1562
1.10k
      {
1563
1.10k
  unsigned int reg1, reg2;
1564
1565
1.10k
  reg1 = uval & 31;
1566
1.10k
  reg2 = uval >> 5;
1567
  /* If one is zero use the other.  */
1568
1.10k
  if (reg1 == reg2 || reg2 == 0)
1569
490
    infprintf (is, dis_style_register, "%s", mips_gpr_names[reg1]);
1570
615
  else if (reg1 == 0)
1571
248
    infprintf (is, dis_style_register, "%s", mips_gpr_names[reg2]);
1572
367
  else
1573
367
    {
1574
      /* Bogus, result depends on processor.  */
1575
367
      infprintf (is, dis_style_register, "%s", mips_gpr_names[reg1]);
1576
367
      infprintf (is, dis_style_text, " or ");
1577
367
      infprintf (is, dis_style_register, "%s", mips_gpr_names[reg2]);
1578
367
    }
1579
1.10k
      }
1580
1.10k
      break;
1581
1582
670
    case OP_SAME_RS_RT:
1583
14.4k
    case OP_CHECK_PREV:
1584
27.8k
    case OP_NON_ZERO_REG:
1585
27.8k
      {
1586
27.8k
  print_reg (info, opcode, OP_REG_GP, uval & 31);
1587
27.8k
  mips_seen_register (state, uval, OP_REG_GP);
1588
27.8k
      }
1589
27.8k
      break;
1590
1591
7.58k
    case OP_LWM_SWM_LIST:
1592
7.58k
      if (operand->size == 2)
1593
2.20k
  {
1594
2.20k
    if (uval == 0)
1595
1.60k
      {
1596
1.60k
        infprintf (is, dis_style_register, "%s",
1597
1.60k
       mips_gpr_names[16]);
1598
1.60k
        infprintf (is, dis_style_text, ",");
1599
1.60k
        infprintf (is, dis_style_register, "%s",
1600
1.60k
       mips_gpr_names[31]);
1601
1.60k
      }
1602
606
    else
1603
606
      {
1604
606
        infprintf (is, dis_style_register, "%s",
1605
606
       mips_gpr_names[16]);
1606
606
        infprintf (is, dis_style_text, "-");
1607
606
        infprintf (is, dis_style_register, "%s",
1608
606
       mips_gpr_names[16 + uval]);
1609
606
        infprintf (is, dis_style_text, ",");
1610
606
        infprintf (is, dis_style_register, "%s",
1611
606
       mips_gpr_names[31]);
1612
606
      }
1613
2.20k
  }
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.28k
      {
1621
4.28k
        if (s_reg_encode == 1)
1622
810
    infprintf (is, dis_style_register, "%s", mips_gpr_names[16]);
1623
3.47k
        else if (s_reg_encode < 9)
1624
1.79k
    {
1625
1.79k
      infprintf (is, dis_style_register, "%s",
1626
1.79k
           mips_gpr_names[16]);
1627
1.79k
      infprintf (is, dis_style_text, "-");
1628
1.79k
      infprintf (is, dis_style_register, "%s",
1629
1.79k
           mips_gpr_names[15 + s_reg_encode]);
1630
1.79k
    }
1631
1.67k
        else if (s_reg_encode == 9)
1632
1.08k
    {
1633
1.08k
      infprintf (is, dis_style_register, "%s",
1634
1.08k
           mips_gpr_names[16]);
1635
1.08k
      infprintf (is, dis_style_text, "-");
1636
1.08k
      infprintf (is, dis_style_register, "%s",
1637
1.08k
           mips_gpr_names[23]);
1638
1.08k
      infprintf (is, dis_style_text, ",");
1639
1.08k
      infprintf (is, dis_style_register, "%s",
1640
1.08k
           mips_gpr_names[30]);
1641
1.08k
    }
1642
591
        else
1643
591
    infprintf (is, dis_style_text, "UNKNOWN");
1644
4.28k
      }
1645
1646
5.37k
    if (uval & 0x10) /* For ra.  */
1647
3.30k
      {
1648
3.30k
        if (s_reg_encode == 0)
1649
952
    infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1650
2.35k
        else
1651
2.35k
    {
1652
2.35k
      infprintf (is, dis_style_text, ",");
1653
2.35k
      infprintf (is, dis_style_register, "%s",
1654
2.35k
           mips_gpr_names[31]);
1655
2.35k
    }
1656
3.30k
      }
1657
5.37k
  }
1658
7.58k
      break;
1659
1660
3.52k
    case OP_ENTRY_EXIT_LIST:
1661
3.52k
      {
1662
3.52k
  const char *sep;
1663
3.52k
  unsigned int amask, smask;
1664
1665
3.52k
  sep = "";
1666
3.52k
  amask = (uval >> 3) & 7;
1667
3.52k
  if (amask > 0 && amask < 5)
1668
1.55k
    {
1669
1.55k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[4]);
1670
1.55k
      if (amask > 1)
1671
376
        {
1672
376
    infprintf (is, dis_style_text, "-");
1673
376
    infprintf (is, dis_style_register, "%s",
1674
376
         mips_gpr_names[amask + 3]);
1675
376
        }
1676
1.55k
      sep = ",";
1677
1.55k
    }
1678
1679
3.52k
  smask = (uval >> 1) & 3;
1680
3.52k
  if (smask == 3)
1681
1.87k
    {
1682
1.87k
      infprintf (is, dis_style_text, "%s??", sep);
1683
1.87k
      sep = ",";
1684
1.87k
    }
1685
1.64k
  else if (smask > 0)
1686
1.04k
    {
1687
1.04k
      infprintf (is, dis_style_text, "%s", sep);
1688
1.04k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[16]);
1689
1.04k
      if (smask > 1)
1690
375
        {
1691
375
    infprintf (is, dis_style_text, "-");
1692
375
    infprintf (is, dis_style_register, "%s",
1693
375
         mips_gpr_names[smask + 15]);
1694
375
        }
1695
1.04k
      sep = ",";
1696
1.04k
    }
1697
1698
3.52k
  if (uval & 1)
1699
2.71k
    {
1700
2.71k
      infprintf (is, dis_style_text, "%s", sep);
1701
2.71k
      infprintf (is, dis_style_register, "%s", mips_gpr_names[31]);
1702
2.71k
      sep = ",";
1703
2.71k
    }
1704
1705
3.52k
  if (amask == 5 || amask == 6)
1706
640
    {
1707
640
      infprintf (is, dis_style_text, "%s", sep);
1708
640
      infprintf (is, dis_style_register, "%s", mips_fpr_names[0]);
1709
640
      if (amask == 6)
1710
330
        {
1711
330
    infprintf (is, dis_style_text, "-");
1712
330
    infprintf (is, dis_style_register, "%s", mips_fpr_names[1]);
1713
330
        }
1714
640
    }
1715
3.52k
      }
1716
3.52k
      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
2.60k
    case OP_MDMX_IMM_REG:
1723
2.60k
      {
1724
2.60k
  unsigned int vsel;
1725
1726
2.60k
  vsel = uval >> 5;
1727
2.60k
  uval &= 31;
1728
2.60k
  if ((vsel & 0x10) == 0)
1729
1.04k
    {
1730
1.04k
      int fmt;
1731
1732
1.04k
      vsel &= 0x0f;
1733
2.94k
      for (fmt = 0; fmt < 3; fmt++, vsel >>= 1)
1734
2.47k
        if ((vsel & 1) == 0)
1735
579
    break;
1736
1.04k
      print_reg (info, opcode, OP_REG_VEC, uval);
1737
1.04k
      infprintf (is, dis_style_text, "[");
1738
1.04k
      infprintf (is, dis_style_immediate, "%d", vsel >> 1);
1739
1.04k
      infprintf (is, dis_style_text, "]");
1740
1.04k
    }
1741
1.56k
  else if ((vsel & 0x08) == 0)
1742
1.20k
    print_reg (info, opcode, OP_REG_VEC, uval);
1743
357
  else
1744
357
    infprintf (is, dis_style_immediate, "0x%x", uval);
1745
2.60k
      }
1746
2.60k
      break;
1747
1748
6.51k
    case OP_REPEAT_PREV_REG:
1749
6.51k
      print_reg (info, opcode, state->last_reg_type, state->last_regno);
1750
6.51k
      break;
1751
1752
813
    case OP_REPEAT_DEST_REG:
1753
813
      print_reg (info, opcode, state->last_reg_type, state->dest_regno);
1754
813
      break;
1755
1756
4.58k
    case OP_PC:
1757
4.58k
      infprintf (is, dis_style_register, "$pc");
1758
4.58k
      break;
1759
1760
952
    case OP_REG28:
1761
952
      print_reg (info, opcode, OP_REG_GP, 28);
1762
952
      break;
1763
1764
724
    case OP_VU0_SUFFIX:
1765
8.43k
    case OP_VU0_MATCH_SUFFIX:
1766
8.43k
      print_vu0_channel (info, operand, uval, dis_style_register);
1767
8.43k
      break;
1768
1769
1.48k
    case OP_IMM_INDEX:
1770
1.48k
      infprintf (is, dis_style_text, "[");
1771
1.48k
      infprintf (is, dis_style_immediate, "%d", uval);
1772
1.48k
      infprintf (is, dis_style_text, "]");
1773
1.48k
      break;
1774
1775
375
    case OP_REG_INDEX:
1776
375
      infprintf (is, dis_style_text, "[");
1777
375
      print_reg (info, opcode, OP_REG_GP, uval);
1778
375
      infprintf (is, dis_style_text, "]");
1779
375
      break;
1780
2.83M
    }
1781
2.83M
}
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
690k
{
1791
690k
  struct mips_print_arg_state state;
1792
690k
  const struct mips_operand *operand;
1793
690k
  const char *s;
1794
690k
  unsigned int uval;
1795
1796
690k
  init_print_arg_state (&state);
1797
3.69M
  for (s = opcode->args; *s; ++s)
1798
3.01M
    {
1799
3.01M
      switch (*s)
1800
3.01M
  {
1801
822k
  case ',':
1802
1.06M
  case '(':
1803
1.30M
  case ')':
1804
1.30M
    break;
1805
1806
210
  case '#':
1807
210
    ++s;
1808
210
    break;
1809
1810
1.70M
  default:
1811
1.70M
    operand = decode_operand (s);
1812
1813
1.70M
    if (operand)
1814
1.70M
      {
1815
1.70M
        uval = mips_extract_operand (operand, insn);
1816
1.70M
        switch (operand->type)
1817
1.70M
    {
1818
885k
    case OP_REG:
1819
1.05M
    case OP_OPTIONAL_REG:
1820
1.05M
      {
1821
1.05M
        const struct mips_reg_operand *reg_op;
1822
1823
1.05M
        reg_op = (const struct mips_reg_operand *) operand;
1824
1.05M
        uval = mips_decode_reg_operand (reg_op, uval);
1825
1.05M
        mips_seen_register (&state, uval, reg_op->reg_type);
1826
1.05M
      }
1827
1.05M
    break;
1828
1829
5.75k
    case OP_SAME_RS_RT:
1830
5.75k
      {
1831
5.75k
        unsigned int reg1, reg2;
1832
1833
5.75k
        reg1 = uval & 31;
1834
5.75k
        reg2 = uval >> 5;
1835
1836
5.75k
        if (reg1 != reg2 || reg1 == 0)
1837
5.08k
          return false;
1838
5.75k
      }
1839
670
    break;
1840
1841
20.0k
    case OP_CHECK_PREV:
1842
20.0k
      {
1843
20.0k
        const struct mips_check_prev_operand *prev_op;
1844
1845
20.0k
        prev_op = (const struct mips_check_prev_operand *) operand;
1846
1847
20.0k
        if (!prev_op->zero_ok && uval == 0)
1848
899
          return false;
1849
1850
19.1k
        if (((prev_op->less_than_ok && uval < state.last_regno)
1851
13.0k
      || (prev_op->greater_than_ok && uval > state.last_regno)
1852
6.45k
      || (prev_op->equal_ok && uval == state.last_regno)))
1853
13.7k
          break;
1854
1855
5.40k
        return false;
1856
19.1k
      }
1857
1858
15.3k
    case OP_NON_ZERO_REG:
1859
15.3k
      {
1860
15.3k
        if (uval == 0)
1861
1.39k
          return false;
1862
15.3k
      }
1863
13.9k
    break;
1864
1865
446k
    case OP_INT:
1866
458k
    case OP_MAPPED_INT:
1867
462k
    case OP_MSB:
1868
466k
    case OP_REG_PAIR:
1869
572k
    case OP_PCREL:
1870
572k
    case OP_PERF_REG:
1871
575k
    case OP_ADDIUSP_INT:
1872
576k
    case OP_CLO_CLZ_DEST:
1873
584k
    case OP_LWM_SWM_LIST:
1874
584k
    case OP_ENTRY_EXIT_LIST:
1875
587k
    case OP_MDMX_IMM_REG:
1876
593k
    case OP_REPEAT_PREV_REG:
1877
594k
    case OP_REPEAT_DEST_REG:
1878
598k
    case OP_PC:
1879
598k
    case OP_REG28:
1880
598k
    case OP_VU0_SUFFIX:
1881
606k
    case OP_VU0_MATCH_SUFFIX:
1882
608k
    case OP_IMM_INDEX:
1883
608k
    case OP_REG_INDEX:
1884
608k
    case OP_SAVE_RESTORE_LIST:
1885
608k
      break;
1886
1.70M
    }
1887
1.70M
      }
1888
1.69M
    if (*s == 'm' || *s == '+' || *s == '-')
1889
435k
      ++s;
1890
3.01M
  }
1891
3.01M
    }
1892
677k
  return true;
1893
690k
}
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
618k
{
1906
618k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
1907
618k
  void *is = info->stream;
1908
618k
  struct mips_print_arg_state state;
1909
618k
  const struct mips_operand *operand;
1910
618k
  const char *s;
1911
1912
618k
  init_print_arg_state (&state);
1913
3.60M
  for (s = opcode->args; *s; ++s)
1914
2.98M
    {
1915
2.98M
      switch (*s)
1916
2.98M
  {
1917
814k
  case ',':
1918
1.05M
  case '(':
1919
1.30M
  case ')':
1920
1.30M
    infprintf (is, dis_style_text, "%c", *s);
1921
1.30M
    break;
1922
1923
210
  case '#':
1924
210
    ++s;
1925
210
    infprintf (is, dis_style_text, "%c%c", *s, *s);
1926
210
    break;
1927
1928
1.68M
  default:
1929
1.68M
    operand = decode_operand (s);
1930
1.68M
    if (!operand)
1931
0
      {
1932
        /* xgettext:c-format */
1933
0
        infprintf (is, dis_style_text,
1934
0
       _("# internal error, undefined operand in `%s %s'"),
1935
0
       opcode->name, opcode->args);
1936
0
        return;
1937
0
      }
1938
1939
1.68M
    if (operand->type == OP_SAVE_RESTORE_LIST)
1940
292
      {
1941
        /* Handle this case here because of the complex behavior.  */
1942
292
        unsigned int amask = (insn >> 15) & 0xf;
1943
292
        unsigned int nsreg = (insn >> 23) & 0x7;
1944
292
        unsigned int ra = insn & 0x1000;      /* $ra */
1945
292
        unsigned int s0 = insn & 0x800;     /* $s0 */
1946
292
        unsigned int s1 = insn & 0x400;     /* $s1 */
1947
292
        unsigned int frame_size = (((insn >> 15) & 0xf0)
1948
292
           | ((insn >> 6) & 0x0f)) * 8;
1949
292
        mips_print_save_restore (info, amask, nsreg, ra, s0, s1,
1950
292
               frame_size);
1951
292
      }
1952
1.68M
    else if (operand->type == OP_REG
1953
879k
       && s[1] == ','
1954
442k
       && (s[2] == 'H' || s[2] == 'J')
1955
2.30k
       && opcode->name[strlen (opcode->name) - 1] == '0')
1956
2.01k
      {
1957
        /* Coprocessor register 0 with sel field.  */
1958
2.01k
        const struct mips_cp0sel_name *n;
1959
2.01k
        unsigned int reg, sel;
1960
1961
2.01k
        reg = mips_extract_operand (operand, insn);
1962
2.01k
        s += 2;
1963
2.01k
        operand = decode_operand (s);
1964
2.01k
        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
2.01k
        n = lookup_mips_cp0sel_name (mips_cp0sel_names,
1972
2.01k
             mips_cp0sel_names_len,
1973
2.01k
             reg, sel);
1974
2.01k
        if (n != NULL)
1975
335
    infprintf (is, dis_style_register, "%s", n->name);
1976
1.68k
        else
1977
1.68k
    {
1978
1.68k
      infprintf (is, dis_style_register, "$%d", reg);
1979
1.68k
      infprintf (is, dis_style_text, ",");
1980
1.68k
      infprintf (is, dis_style_immediate, "%d", sel);
1981
1.68k
    }
1982
2.01k
      }
1983
1.68M
    else
1984
1.68M
      {
1985
1.68M
        bfd_vma base_pc = insn_pc;
1986
1987
        /* Adjust the PC relative base so that branch/jump insns use
1988
     the following PC as the base but genuinely PC relative
1989
     operands use the current PC.  */
1990
1.68M
        if (operand->type == OP_PCREL)
1991
106k
    {
1992
106k
      const struct mips_pcrel_operand *pcrel_op;
1993
1994
106k
      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
106k
      if (pcrel_op->include_isa_bit)
1998
104k
        base_pc += length;
1999
106k
    }
2000
2001
1.68M
        print_insn_arg (info, &state, opcode, operand, base_pc,
2002
1.68M
            mips_extract_operand (operand, insn));
2003
1.68M
      }
2004
1.68M
    if (*s == 'm' || *s == '+' || *s == '-')
2005
435k
      ++s;
2006
1.68M
    break;
2007
2.98M
  }
2008
2.98M
    }
2009
618k
}
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
463k
{
2021
463k
#define GET_OP(insn, field)     \
2022
575k
  (((insn) >> OP_SH_##field) & OP_MASK_##field)
2023
463k
  static const struct mips_opcode *mips_hash[OP_MASK_OP + 1];
2024
463k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2025
463k
  const struct mips_opcode *op;
2026
463k
  static bool init = 0;
2027
463k
  void *is = info->stream;
2028
2029
  /* Build a hash table to shorten the search time.  */
2030
463k
  if (! init)
2031
3
    {
2032
3
      unsigned int i;
2033
2034
195
      for (i = 0; i <= OP_MASK_OP; i++)
2035
192
  {
2036
126k
    for (op = mips_opcodes; op < &mips_opcodes[NUMOPCODES]; op++)
2037
126k
      {
2038
126k
        if (op->pinfo == INSN_MACRO
2039
111k
      || (no_aliases && (op->pinfo2 & INSN2_ALIAS)))
2040
14.6k
    continue;
2041
111k
        if (i == GET_OP (op->match, OP))
2042
192
    {
2043
192
      mips_hash[i] = op;
2044
192
      break;
2045
192
    }
2046
111k
      }
2047
192
  }
2048
2049
3
      init = 1;
2050
3
    }
2051
2052
463k
  info->bytes_per_chunk = INSNLEN;
2053
463k
  info->display_endian = info->endian;
2054
463k
  info->insn_info_valid = 1;
2055
463k
  info->branch_delay_insns = 0;
2056
463k
  info->data_size = 0;
2057
463k
  info->insn_type = dis_nonbranch;
2058
463k
  info->target = 0;
2059
463k
  info->target2 = 0;
2060
2061
463k
  op = mips_hash[GET_OP (word, OP)];
2062
463k
  if (op != NULL)
2063
463k
    {
2064
573M
      for (; op < &mips_opcodes[NUMOPCODES]; op++)
2065
573M
  {
2066
573M
    if (op->pinfo != INSN_MACRO
2067
525M
        && !(no_aliases && (op->pinfo2 & INSN2_ALIAS))
2068
525M
        && (word & op->mask) == op->match)
2069
567k
      {
2070
        /* We always disassemble the jalx instruction, except for MIPS r6.  */
2071
567k
        if (!opcode_is_member (op, mips_isa, mips_ase, mips_processor)
2072
252k
     && (strcmp (op->name, "jalx")
2073
2.71k
         || (mips_isa & INSN_ISA_MASK) == ISA_MIPS32R6
2074
1.24k
         || (mips_isa & INSN_ISA_MASK) == ISA_MIPS64R6))
2075
252k
    continue;
2076
2077
        /* Figure out instruction type and branch delay information.  */
2078
315k
        if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0)
2079
19.7k
          {
2080
19.7k
      if ((op->pinfo & (INSN_WRITE_GPR_31 | INSN_WRITE_1)) != 0)
2081
11.8k
        info->insn_type = dis_jsr;
2082
7.95k
      else
2083
7.95k
        info->insn_type = dis_branch;
2084
19.7k
      info->branch_delay_insns = 1;
2085
19.7k
    }
2086
295k
        else if ((op->pinfo & (INSN_COND_BRANCH_DELAY
2087
295k
             | INSN_COND_BRANCH_LIKELY)) != 0)
2088
24.3k
    {
2089
24.3k
      if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2090
620
        info->insn_type = dis_condjsr;
2091
23.7k
      else
2092
23.7k
        info->insn_type = dis_condbranch;
2093
24.3k
      info->branch_delay_insns = 1;
2094
24.3k
    }
2095
271k
        else if ((op->pinfo & (INSN_STORE_MEMORY
2096
271k
             | INSN_LOAD_MEMORY)) != 0)
2097
91.9k
    info->insn_type = dis_dref;
2098
2099
315k
        if (!validate_insn_args (op, decode_mips_operand, word))
2100
12.7k
    continue;
2101
2102
302k
        infprintf (is, dis_style_mnemonic, "%s", op->name);
2103
302k
        if (op->pinfo2 & INSN2_VU0_CHANNEL_SUFFIX)
2104
2.86k
    {
2105
2.86k
      unsigned int uval;
2106
2107
2.86k
      infprintf (is, dis_style_mnemonic, ".");
2108
2.86k
      uval = mips_extract_operand (&mips_vu0_channel_mask, word);
2109
2.86k
      print_vu0_channel (info, &mips_vu0_channel_mask, uval,
2110
2.86k
             dis_style_mnemonic);
2111
2.86k
    }
2112
2113
302k
        if (op->args[0])
2114
269k
    {
2115
269k
      infprintf (is, dis_style_text, "\t");
2116
269k
      print_insn_args (info, op, decode_mips_operand, word,
2117
269k
           memaddr, 4);
2118
269k
    }
2119
2120
302k
        return INSNLEN;
2121
315k
      }
2122
573M
  }
2123
463k
    }
2124
161k
#undef GET_OP
2125
2126
  /* Handle undefined instructions.  */
2127
161k
  info->insn_type = dis_noninsn;
2128
161k
  infprintf (is, dis_style_assembler_directive, ".word");
2129
161k
  infprintf (is, dis_style_text, "\t");
2130
161k
  infprintf (is, dis_style_immediate, "0x%x", word);
2131
161k
  return INSNLEN;
2132
463k
}
2133

2134
/* Disassemble an operand for a mips16 instruction.  */
2135
2136
static void
2137
print_mips16_insn_arg (struct disassemble_info *info,
2138
           struct mips_print_arg_state *state,
2139
           const struct mips_opcode *opcode,
2140
           char type, bfd_vma memaddr,
2141
           unsigned insn, bool use_extend,
2142
           unsigned extend, bool is_offset)
2143
1.97M
{
2144
1.97M
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2145
1.97M
  void *is = info->stream;
2146
1.97M
  const struct mips_operand *operand, *ext_operand;
2147
1.97M
  unsigned short ext_size;
2148
1.97M
  unsigned int uval;
2149
1.97M
  bfd_vma baseaddr;
2150
2151
1.97M
  if (!use_extend)
2152
1.84M
    extend = 0;
2153
2154
1.97M
  switch (type)
2155
1.97M
    {
2156
592k
    case ',':
2157
700k
    case '(':
2158
808k
    case ')':
2159
808k
      infprintf (is, dis_style_text, "%c", type);
2160
808k
      break;
2161
2162
1.16M
    default:
2163
1.16M
      operand = decode_mips16_operand (type, false);
2164
1.16M
      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.16M
      if (operand->type == OP_SAVE_RESTORE_LIST)
2173
7.97k
  {
2174
    /* Handle this case here because of the complex interaction
2175
       with the EXTEND opcode.  */
2176
7.97k
    unsigned int amask = extend & 0xf;
2177
7.97k
    unsigned int nsreg = (extend >> 8) & 0x7;
2178
7.97k
    unsigned int ra = insn & 0x40;      /* $ra */
2179
7.97k
    unsigned int s0 = insn & 0x20;      /* $s0 */
2180
7.97k
    unsigned int s1 = insn & 0x10;      /* $s1 */
2181
7.97k
    unsigned int frame_size = ((extend & 0xf0) | (insn & 0x0f)) * 8;
2182
7.97k
    if (frame_size == 0 && !use_extend)
2183
1.88k
      frame_size = 128;
2184
7.97k
    mips_print_save_restore (info, amask, nsreg, ra, s0, s1, frame_size);
2185
7.97k
    break;
2186
7.97k
  }
2187
2188
1.15M
      if (is_offset && operand->type == OP_INT)
2189
108k
  {
2190
108k
    const struct mips_int_operand *int_op;
2191
2192
108k
    int_op = (const struct mips_int_operand *) operand;
2193
108k
    info->insn_type = dis_dref;
2194
108k
    info->data_size = 1 << int_op->shift;
2195
108k
  }
2196
2197
1.15M
      ext_size = 0;
2198
1.15M
      if (use_extend)
2199
74.6k
  {
2200
74.6k
    ext_operand = decode_mips16_operand (type, true);
2201
74.6k
    if (ext_operand != operand
2202
52.3k
        || (operand->type == OP_INT && operand->lsb == 0
2203
2.15k
      && mips_opcode_32bit_p (opcode)))
2204
24.3k
      {
2205
24.3k
        ext_size = ext_operand->size;
2206
24.3k
        operand = ext_operand;
2207
24.3k
      }
2208
74.6k
  }
2209
1.15M
      if (operand->size == 26)
2210
6.61k
  uval = ((extend & 0x1f) << 21) | ((extend & 0x3e0) << 11) | insn;
2211
1.14M
      else if (ext_size == 16 || ext_size == 9)
2212
21.0k
  uval = ((extend & 0x1f) << 11) | (extend & 0x7e0) | (insn & 0x1f);
2213
1.12M
      else if (ext_size == 15)
2214
1.28k
  uval = ((extend & 0xf) << 11) | (extend & 0x7f0) | (insn & 0xf);
2215
1.12M
      else if (ext_size == 6)
2216
1.12k
  uval = ((extend >> 6) & 0x1f) | (extend & 0x20);
2217
1.12M
      else
2218
1.12M
  uval = mips_extract_operand (operand, (extend << 16) | insn);
2219
1.15M
      if (ext_size == 9)
2220
1.34k
  uval &= (1U << ext_size) - 1;
2221
2222
1.15M
      baseaddr = memaddr + 2;
2223
1.15M
      if (operand->type == OP_PCREL)
2224
72.4k
  {
2225
72.4k
    const struct mips_pcrel_operand *pcrel_op;
2226
2227
72.4k
    pcrel_op = (const struct mips_pcrel_operand *) operand;
2228
72.4k
    if (!pcrel_op->include_isa_bit && use_extend)
2229
2.55k
      baseaddr = memaddr - 2;
2230
69.8k
    else if (!pcrel_op->include_isa_bit)
2231
21.4k
      {
2232
21.4k
        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
21.4k
        if (info->read_memory_func (memaddr - 4, buffer, 2, info) == 0
2242
7.05k
      && (((info->endian == BFD_ENDIAN_BIG
2243
7.05k
      ? bfd_getb16 (buffer)
2244
7.05k
      : bfd_getl16 (buffer))
2245
7.05k
           & 0xf800) == 0x1800))
2246
161
    baseaddr = memaddr - 4;
2247
21.2k
        else if (info->read_memory_func (memaddr - 2, buffer, 2,
2248
21.2k
                 info) == 0
2249
6.89k
           && (((info->endian == BFD_ENDIAN_BIG
2250
6.89k
           ? bfd_getb16 (buffer)
2251
6.89k
           : bfd_getl16 (buffer))
2252
6.89k
          & 0xf89f) == 0xe800)
2253
36
           && (((info->endian == BFD_ENDIAN_BIG
2254
36
           ? bfd_getb16 (buffer)
2255
36
           : bfd_getl16 (buffer))
2256
36
          & 0x0060) != 0x0060))
2257
9
    baseaddr = memaddr - 2;
2258
21.2k
        else
2259
21.2k
    baseaddr = memaddr;
2260
21.4k
      }
2261
72.4k
  }
2262
2263
1.15M
      print_insn_arg (info, state, opcode, operand, baseaddr + 1, uval);
2264
1.15M
      break;
2265
1.97M
    }
2266
1.97M
}
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
489k
{
2276
489k
  if (info->symbols
2277
39
      && info->symbols[0]
2278
39
      && (info->symbols[0]->flags & BSF_SYNTHETIC)
2279
0
      && addr == bfd_asymbol_value (info->symbols[0]) + 12)
2280
0
    return true;
2281
2282
489k
  return false;
2283
489k
}
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
489k
{
2299
489k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2300
489k
  int status;
2301
489k
  bfd_byte buffer[4];
2302
489k
  const struct mips_opcode *op, *opend;
2303
489k
  struct mips_print_arg_state state;
2304
489k
  void *is = info->stream;
2305
489k
  bool have_second;
2306
489k
  bool extend_only;
2307
489k
  unsigned int second;
2308
489k
  unsigned int first;
2309
489k
  unsigned int full;
2310
2311
489k
  info->bytes_per_chunk = 2;
2312
489k
  info->display_endian = info->endian;
2313
489k
  info->insn_info_valid = 1;
2314
489k
  info->branch_delay_insns = 0;
2315
489k
  info->data_size = 0;
2316
489k
  info->target = 0;
2317
489k
  info->target2 = 0;
2318
2319
489k
#define GET_OP(insn, field) \
2320
489k
  (((insn) >> MIPS16OP_SH_##field) & MIPS16OP_MASK_##field)
2321
  /* Decode PLT entry's GOT slot address word.  */
2322
489k
  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
489k
  else
2342
489k
    {
2343
489k
      info->insn_type = dis_nonbranch;
2344
489k
      status = (*info->read_memory_func) (memaddr, buffer, 2, info);
2345
489k
    }
2346
489k
  if (status != 0)
2347
1.32k
    {
2348
1.32k
      (*info->memory_error_func) (status, memaddr, info);
2349
1.32k
      return -1;
2350
1.32k
    }
2351
2352
487k
  extend_only = false;
2353
2354
487k
  if (info->endian == BFD_ENDIAN_BIG)
2355
111k
    first = bfd_getb16 (buffer);
2356
375k
  else
2357
375k
    first = bfd_getl16 (buffer);
2358
2359
487k
  status = (*info->read_memory_func) (memaddr + 2, buffer, 2, info);
2360
487k
  if (status == 0)
2361
486k
    {
2362
486k
      have_second = true;
2363
486k
      if (info->endian == BFD_ENDIAN_BIG)
2364
111k
  second = bfd_getb16 (buffer);
2365
374k
      else
2366
374k
  second = bfd_getl16 (buffer);
2367
486k
      full = (first << 16) | second;
2368
486k
    }
2369
1.20k
  else
2370
1.20k
    {
2371
1.20k
      have_second = false;
2372
1.20k
      second = 0;
2373
1.20k
      full = first;
2374
1.20k
    }
2375
2376
  /* FIXME: Should probably use a hash table on the major opcode here.  */
2377
2378
487k
  opend = mips16_opcodes + bfd_mips16_num_opcodes;
2379
48.2M
  for (op = mips16_opcodes; op < opend; op++)
2380
48.2M
    {
2381
48.2M
      enum match_kind match;
2382
2383
48.2M
      if (!opcode_is_member (op, mips_isa, mips_ase, mips_processor))
2384
7.90M
  continue;
2385
2386
40.3M
      if (op->pinfo == INSN_MACRO
2387
31.5M
    || (no_aliases && (op->pinfo2 & INSN2_ALIAS)))
2388
8.78M
  match = MATCH_NONE;
2389
31.5M
      else if (mips_opcode_32bit_p (op))
2390
6.34M
  {
2391
6.34M
    if (have_second
2392
6.32M
        && (full & op->mask) == op->match)
2393
13.7k
      match = MATCH_FULL;
2394
6.33M
    else
2395
6.33M
      match = MATCH_NONE;
2396
6.34M
  }
2397
25.1M
      else if ((first & op->mask) == op->match)
2398
426k
  {
2399
426k
    match = MATCH_SHORT;
2400
426k
    second = 0;
2401
426k
    full = first;
2402
426k
  }
2403
24.7M
      else if ((first & 0xf800) == 0xf000
2404
3.44M
         && have_second
2405
3.43M
         && !extend_only
2406
3.24M
         && (second & op->mask) == op->match)
2407
28.5k
  {
2408
28.5k
    if (op->pinfo2 & INSN2_SHORT_ONLY)
2409
6.10k
      {
2410
6.10k
        match = MATCH_NONE;
2411
6.10k
        extend_only = true;
2412
6.10k
      }
2413
22.4k
    else
2414
22.4k
      match = MATCH_FULL;
2415
28.5k
  }
2416
24.7M
      else
2417
24.7M
  match = MATCH_NONE;
2418
2419
40.3M
      if (match != MATCH_NONE)
2420
462k
  {
2421
462k
    const char *s;
2422
2423
462k
    infprintf (is, dis_style_mnemonic, "%s", op->name);
2424
462k
    if (op->args[0] != '\0')
2425
462k
      infprintf (is, dis_style_text, "\t");
2426
2427
462k
    init_print_arg_state (&state);
2428
2.43M
    for (s = op->args; *s != '\0'; s++)
2429
1.97M
      {
2430
1.97M
        if (*s == ','
2431
595k
      && s[1] == 'w'
2432
16.7k
      && GET_OP (full, RX) == GET_OP (full, RY))
2433
2.24k
    {
2434
      /* Skip the register and the comma.  */
2435
2.24k
      ++s;
2436
2.24k
      continue;
2437
2.24k
    }
2438
1.97M
        if (*s == ','
2439
592k
      && s[1] == 'v'
2440
8.73k
      && GET_OP (full, RZ) == GET_OP (full, RX))
2441
746
    {
2442
      /* Skip the register and the comma.  */
2443
746
      ++s;
2444
746
      continue;
2445
746
    }
2446
1.97M
        if (s[0] == 'N'
2447
1.52k
      && s[1] == ','
2448
1.22k
      && s[2] == 'O'
2449
1.22k
      && op->name[strlen (op->name) - 1] == '0')
2450
1.22k
    {
2451
      /* Coprocessor register 0 with sel field.  */
2452
1.22k
      const struct mips_cp0sel_name *n;
2453
1.22k
      const struct mips_operand *operand;
2454
1.22k
      unsigned int reg, sel;
2455
2456
1.22k
      operand = decode_mips16_operand (*s, true);
2457
1.22k
      reg = mips_extract_operand (operand, (first << 16) | second);
2458
1.22k
      s += 2;
2459
1.22k
      operand = decode_mips16_operand (*s, true);
2460
1.22k
      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
1.22k
      n = lookup_mips_cp0sel_name (mips_cp0sel_names,
2468
1.22k
                 mips_cp0sel_names_len,
2469
1.22k
                 reg, sel);
2470
1.22k
      if (n != NULL)
2471
230
        infprintf (is, dis_style_register, "%s", n->name);
2472
995
      else
2473
995
        {
2474
995
          infprintf (is, dis_style_register, "$%d", reg);
2475
995
          infprintf (is, dis_style_text, ",");
2476
995
          infprintf (is, dis_style_immediate, "%d", sel);
2477
995
        }
2478
1.22k
    }
2479
1.97M
        else
2480
1.97M
    switch (match)
2481
1.97M
      {
2482
128k
        case MATCH_FULL:
2483
128k
          print_mips16_insn_arg (info, &state, op, *s, memaddr + 2,
2484
128k
               second, true, first, s[1] == '(');
2485
128k
          break;
2486
1.84M
        case MATCH_SHORT:
2487
1.84M
          print_mips16_insn_arg (info, &state, op, *s, memaddr,
2488
1.84M
               first, false, 0, s[1] == '(');
2489
1.84M
          break;
2490
0
        case MATCH_NONE:  /* Stop the compiler complaining.  */
2491
0
          break;
2492
1.97M
      }
2493
1.97M
      }
2494
2495
    /* Figure out branch instruction type and delay slot information.  */
2496
462k
    if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0)
2497
7.64k
      info->branch_delay_insns = 1;
2498
462k
    if ((op->pinfo & INSN_UNCOND_BRANCH_DELAY) != 0
2499
455k
        || (op->pinfo2 & INSN2_UNCOND_BRANCH) != 0)
2500
23.5k
      {
2501
23.5k
        if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2502
6.68k
    info->insn_type = dis_jsr;
2503
16.8k
        else
2504
16.8k
    info->insn_type = dis_branch;
2505
23.5k
      }
2506
439k
    else if ((op->pinfo2 & INSN2_COND_BRANCH) != 0)
2507
26.2k
      info->insn_type = dis_condbranch;
2508
2509
462k
    return match == MATCH_FULL ? 4 : 2;
2510
462k
  }
2511
40.3M
    }
2512
24.8k
#undef GET_OP
2513
2514
24.8k
  infprintf (is, dis_style_assembler_directive, ".short");
2515
24.8k
  infprintf (is, dis_style_text, "\t");
2516
24.8k
  infprintf (is, dis_style_immediate, "0x%x", first);
2517
24.8k
  info->insn_type = dis_noninsn;
2518
2519
24.8k
  return 2;
2520
487k
}
2521
2522
/* Disassemble microMIPS instructions.  */
2523
2524
static int
2525
print_insn_micromips (bfd_vma memaddr, struct disassemble_info *info)
2526
484k
{
2527
484k
  const fprintf_styled_ftype infprintf = info->fprintf_styled_func;
2528
484k
  const struct mips_opcode *op, *opend;
2529
484k
  void *is = info->stream;
2530
484k
  bfd_byte buffer[2];
2531
484k
  unsigned int higher;
2532
484k
  unsigned int length;
2533
484k
  int status;
2534
484k
  unsigned int insn;
2535
2536
484k
  info->bytes_per_chunk = 2;
2537
484k
  info->display_endian = info->endian;
2538
484k
  info->insn_info_valid = 1;
2539
484k
  info->branch_delay_insns = 0;
2540
484k
  info->data_size = 0;
2541
484k
  info->insn_type = dis_nonbranch;
2542
484k
  info->target = 0;
2543
484k
  info->target2 = 0;
2544
2545
484k
  status = (*info->read_memory_func) (memaddr, buffer, 2, info);
2546
484k
  if (status != 0)
2547
350
    {
2548
350
      (*info->memory_error_func) (status, memaddr, info);
2549
350
      return -1;
2550
350
    }
2551
2552
483k
  length = 2;
2553
2554
483k
  if (info->endian == BFD_ENDIAN_BIG)
2555
196k
    insn = bfd_getb16 (buffer);
2556
287k
  else
2557
287k
    insn = bfd_getl16 (buffer);
2558
2559
483k
  if ((insn & 0x1c00) == 0x0000 || (insn & 0x1000) == 0x1000)
2560
348k
    {
2561
      /* This is a 32-bit microMIPS instruction.  */
2562
348k
      higher = insn;
2563
2564
348k
      status = (*info->read_memory_func) (memaddr + 2, buffer, 2, info);
2565
348k
      if (status != 0)
2566
239
  {
2567
239
    infprintf (is, dis_style_text, "micromips 0x%x", higher);
2568
239
    (*info->memory_error_func) (status, memaddr + 2, info);
2569
239
    return -1;
2570
239
  }
2571
2572
347k
      if (info->endian == BFD_ENDIAN_BIG)
2573
149k
  insn = bfd_getb16 (buffer);
2574
198k
      else
2575
198k
  insn = bfd_getl16 (buffer);
2576
2577
347k
      insn = insn | (higher << 16);
2578
2579
347k
      length += 2;
2580
347k
    }
2581
2582
  /* FIXME: Should probably use a hash table on the major opcode here.  */
2583
2584
483k
  opend = micromips_opcodes + bfd_micromips_num_opcodes;
2585
341M
  for (op = micromips_opcodes; op < opend; op++)
2586
341M
    {
2587
341M
      if (op->pinfo != INSN_MACRO
2588
274M
    && !(no_aliases && (op->pinfo2 & INSN2_ALIAS))
2589
274M
    && (insn & op->mask) == op->match
2590
472k
    && ((length == 2 && (op->mask & 0xffff0000) == 0)
2591
354k
        || (length == 4 && (op->mask & 0xffff0000) != 0)))
2592
374k
  {
2593
374k
    if (!validate_insn_args (op, decode_micromips_operand, insn))
2594
0
      continue;
2595
2596
374k
    infprintf (is, dis_style_mnemonic, "%s", op->name);
2597
2598
374k
    if (op->args[0])
2599
348k
      {
2600
348k
        infprintf (is, dis_style_text, "\t");
2601
348k
        print_insn_args (info, op, decode_micromips_operand, insn,
2602
348k
             memaddr + 1, length);
2603
348k
      }
2604
2605
    /* Figure out instruction type and branch delay information.  */
2606
374k
    if ((op->pinfo
2607
374k
         & (INSN_UNCOND_BRANCH_DELAY | INSN_COND_BRANCH_DELAY)) != 0)
2608
40.8k
      info->branch_delay_insns = 1;
2609
374k
    if (((op->pinfo & INSN_UNCOND_BRANCH_DELAY)
2610
374k
         | (op->pinfo2 & INSN2_UNCOND_BRANCH)) != 0)
2611
26.1k
      {
2612
26.1k
        if ((op->pinfo & (INSN_WRITE_GPR_31 | INSN_WRITE_1)) != 0)
2613
16.6k
    info->insn_type = dis_jsr;
2614
9.50k
        else
2615
9.50k
    info->insn_type = dis_branch;
2616
26.1k
      }
2617
348k
    else if (((op->pinfo & INSN_COND_BRANCH_DELAY)
2618
348k
        | (op->pinfo2 & INSN2_COND_BRANCH)) != 0)
2619
15.2k
      {
2620
15.2k
        if ((op->pinfo & INSN_WRITE_GPR_31) != 0)
2621
602
    info->insn_type = dis_condjsr;
2622
14.6k
        else
2623
14.6k
    info->insn_type = dis_condbranch;
2624
15.2k
      }
2625
333k
    else if ((op->pinfo
2626
333k
        & (INSN_STORE_MEMORY | INSN_LOAD_MEMORY)) != 0)
2627
147k
      info->insn_type = dis_dref;
2628
2629
374k
    return length;
2630
374k
  }
2631
341M
    }
2632
2633
109k
  infprintf (is, dis_style_assembler_directive, ".short");
2634
109k
  infprintf (is, dis_style_text, "\t");
2635
109k
  if (length != 2)
2636
91.1k
    {
2637
91.1k
      infprintf (is, dis_style_immediate, "0x%x", (insn >> 16) & 0xffff);
2638
91.1k
      infprintf (is, dis_style_text, ", ");
2639
91.1k
    }
2640
109k
  infprintf (is, dis_style_immediate, "0x%x", (insn & 0xffff));
2641
2642
109k
  info->insn_type = dis_noninsn;
2643
2644
109k
  return length;
2645
483k
}
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
930k
{
2658
930k
  int i;
2659
930k
  int l;
2660
2661
930k
  for (i = info->symtab_pos, l = i + info->num_symbols; i < l; i++)
2662
188
    if (((info->symtab[i])->flags & BSF_SYNTHETIC) != 0
2663
0
  && ((!micromips_p
2664
0
       && ELF_ST_IS_MIPS16 ((*info->symbols)->udata.i))
2665
0
      || (micromips_p
2666
0
    && ELF_ST_IS_MICROMIPS ((*info->symbols)->udata.i))))
2667
0
      return 1;
2668
188
    else if (bfd_asymbol_flavour (info->symtab[i]) == bfd_target_elf_flavour
2669
188
        && info->symtab[i]->section == info->section)
2670
136
      {
2671
136
  elf_symbol_type *symbol = (elf_symbol_type *) info->symtab[i];
2672
136
  if ((!micromips_p
2673
68
       && ELF_ST_IS_MIPS16 (symbol->internal_elf_sym.st_other))
2674
97
      || (micromips_p
2675
68
    && ELF_ST_IS_MICROMIPS (symbol->internal_elf_sym.st_other)))
2676
39
    return 1;
2677
136
      }
2678
2679
930k
  return 0;
2680
930k
}
2681
2682
/* In an environment where we do not know the symbol type of the
2683
   instruction we are forced to assume that the low order bit of the
2684
   instructions' address may mark it as a mips16 instruction.  If we
2685
   are single stepping, or the pc is within the disassembled function,
2686
   this works.  Otherwise, we need a clue.  Sometimes.  */
2687
2688
static int
2689
_print_insn_mips (bfd_vma memaddr,
2690
      struct disassemble_info *info,
2691
      enum bfd_endian endianness)
2692
1.43M
{
2693
1.43M
  bfd_byte buffer[INSNLEN];
2694
1.43M
  int status;
2695
2696
1.43M
  set_default_mips_dis_options (info);
2697
1.43M
  parse_mips_dis_options (info->disassembler_options);
2698
2699
1.43M
  if (info->mach == bfd_mach_mips16)
2700
319k
    return print_insn_mips16 (memaddr, info);
2701
1.11M
  if (info->mach == bfd_mach_mips_micromips)
2702
227k
    return print_insn_micromips (memaddr, info);
2703
2704
892k
#if 1
2705
  /* FIXME: If odd address, this is CLEARLY a compressed instruction.  */
2706
  /* Only a few tools will work this way.  */
2707
892k
  if (memaddr & 0x01)
2708
426k
    {
2709
426k
      if (micromips_ase)
2710
256k
  return print_insn_micromips (memaddr, info);
2711
169k
      else
2712
169k
  return print_insn_mips16 (memaddr, info);
2713
426k
    }
2714
465k
#endif
2715
2716
465k
#if SYMTAB_AVAILABLE
2717
465k
  if (is_compressed_mode_p (info, true))
2718
0
    return print_insn_micromips (memaddr, info);
2719
465k
  if (is_compressed_mode_p (info, false))
2720
39
    return print_insn_mips16 (memaddr, info);
2721
465k
#endif
2722
2723
465k
  status = (*info->read_memory_func) (memaddr, buffer, INSNLEN, info);
2724
465k
  if (status == 0)
2725
463k
    {
2726
463k
      int insn;
2727
2728
463k
      if (endianness == BFD_ENDIAN_BIG)
2729
157k
  insn = bfd_getb32 (buffer);
2730
305k
      else
2731
305k
  insn = bfd_getl32 (buffer);
2732
2733
463k
      return print_insn_mips (memaddr, insn, info);
2734
463k
    }
2735
1.69k
  else
2736
1.69k
    {
2737
1.69k
      (*info->memory_error_func) (status, memaddr, info);
2738
1.69k
      return -1;
2739
1.69k
    }
2740
465k
}
2741
2742
int
2743
print_insn_big_mips (bfd_vma memaddr, struct disassemble_info *info)
2744
515k
{
2745
515k
  return _print_insn_mips (memaddr, info, BFD_ENDIAN_BIG);
2746
515k
}
2747
2748
int
2749
print_insn_little_mips (bfd_vma memaddr, struct disassemble_info *info)
2750
922k
{
2751
922k
  return _print_insn_mips (memaddr, info, BFD_ENDIAN_LITTLE);
2752
922k
}
2753

2754
/* Indices into option argument vector for options accepting an argument.
2755
   Use MIPS_OPTION_ARG_NONE for options accepting no argument.  */
2756
typedef enum
2757
{
2758
  MIPS_OPTION_ARG_NONE = -1,
2759
  MIPS_OPTION_ARG_ABI,
2760
  MIPS_OPTION_ARG_ARCH,
2761
  MIPS_OPTION_ARG_SIZE
2762
} mips_option_arg_t;
2763
2764
/* Valid MIPS disassembler options.  */
2765
static struct
2766
{
2767
  const char *name;
2768
  const char *description;
2769
  mips_option_arg_t arg;
2770
} mips_options[] =
2771
{
2772
  { "no-aliases", N_("Use canonical instruction forms.\n"),
2773
      MIPS_OPTION_ARG_NONE },
2774
  { "msa",        N_("Recognize MSA instructions.\n"),
2775
      MIPS_OPTION_ARG_NONE },
2776
  { "virt",       N_("Recognize the virtualization ASE instructions.\n"),
2777
      MIPS_OPTION_ARG_NONE },
2778
  { "xpa",        N_("Recognize the eXtended Physical Address (XPA) ASE\n\
2779
                  instructions.\n"),
2780
      MIPS_OPTION_ARG_NONE },
2781
  { "ginv",       N_("Recognize the Global INValidate (GINV) ASE "
2782
         "instructions.\n"),
2783
      MIPS_OPTION_ARG_NONE },
2784
  { "loongson-mmi",
2785
      N_("Recognize the Loongson MultiMedia extensions "
2786
         "Instructions (MMI) ASE instructions.\n"),
2787
      MIPS_OPTION_ARG_NONE },
2788
  { "loongson-cam",
2789
      N_("Recognize the Loongson Content Address Memory (CAM) "
2790
         " instructions.\n"),
2791
      MIPS_OPTION_ARG_NONE },
2792
  { "loongson-ext",
2793
      N_("Recognize the Loongson EXTensions (EXT) "
2794
         " instructions.\n"),
2795
      MIPS_OPTION_ARG_NONE },
2796
  { "loongson-ext2",
2797
      N_("Recognize the Loongson EXTensions R2 (EXT2) "
2798
         " instructions.\n"),
2799
      MIPS_OPTION_ARG_NONE },
2800
  { "gpr-names=", N_("Print GPR names according to specified ABI.\n\
2801
                  Default: based on binary being disassembled.\n"),
2802
      MIPS_OPTION_ARG_ABI },
2803
  { "fpr-names=", N_("Print FPR names according to specified ABI.\n\
2804
                  Default: numeric.\n"),
2805
      MIPS_OPTION_ARG_ABI },
2806
  { "cp0-names=", N_("Print CP0 register names according to specified "
2807
         "architecture.\n\
2808
                  Default: based on binary being disassembled.\n"),
2809
      MIPS_OPTION_ARG_ARCH },
2810
  { "hwr-names=", N_("Print HWR names according to specified architecture.\n\
2811
                  Default: based on binary being disassembled.\n"),
2812
      MIPS_OPTION_ARG_ARCH },
2813
  { "reg-names=", N_("Print GPR and FPR names according to specified ABI.\n"),
2814
      MIPS_OPTION_ARG_ABI },
2815
  { "reg-names=", N_("Print CP0 register and HWR names according to "
2816
         "specified\n\
2817
                  architecture."),
2818
      MIPS_OPTION_ARG_ARCH }
2819
};
2820
2821
/* Build the structure representing valid MIPS disassembler options.
2822
   This is done dynamically for maintenance ease purpose; a static
2823
   initializer would be unreadable.  */
2824
2825
const disasm_options_and_args_t *
2826
disassembler_options_mips (void)
2827
0
{
2828
0
  static disasm_options_and_args_t *opts_and_args;
2829
2830
0
  if (opts_and_args == NULL)
2831
0
    {
2832
0
      size_t num_options = ARRAY_SIZE (mips_options);
2833
0
      size_t num_args = MIPS_OPTION_ARG_SIZE;
2834
0
      disasm_option_arg_t *args;
2835
0
      disasm_options_t *opts;
2836
0
      size_t i;
2837
0
      size_t j;
2838
2839
0
      args = XNEWVEC (disasm_option_arg_t, num_args + 1);
2840
2841
0
      args[MIPS_OPTION_ARG_ABI].name = "ABI";
2842
0
      args[MIPS_OPTION_ARG_ABI].values
2843
0
  = XNEWVEC (const char *, ARRAY_SIZE (mips_abi_choices) + 1);
2844
0
      for (i = 0; i < ARRAY_SIZE (mips_abi_choices); i++)
2845
0
  args[MIPS_OPTION_ARG_ABI].values[i] = mips_abi_choices[i].name;
2846
      /* The array we return must be NULL terminated.  */
2847
0
      args[MIPS_OPTION_ARG_ABI].values[i] = NULL;
2848
2849
0
      args[MIPS_OPTION_ARG_ARCH].name = "ARCH";
2850
0
      args[MIPS_OPTION_ARG_ARCH].values
2851
0
  = XNEWVEC (const char *, ARRAY_SIZE (mips_arch_choices) + 1);
2852
0
      for (i = 0, j = 0; i < ARRAY_SIZE (mips_arch_choices); i++)
2853
0
  if (*mips_arch_choices[i].name != '\0')
2854
0
    args[MIPS_OPTION_ARG_ARCH].values[j++] = mips_arch_choices[i].name;
2855
      /* The array we return must be NULL terminated.  */
2856
0
      args[MIPS_OPTION_ARG_ARCH].values[j] = NULL;
2857
2858
      /* The array we return must be NULL terminated.  */
2859
0
      args[MIPS_OPTION_ARG_SIZE].name = NULL;
2860
0
      args[MIPS_OPTION_ARG_SIZE].values = NULL;
2861
2862
0
      opts_and_args = XNEW (disasm_options_and_args_t);
2863
0
      opts_and_args->args = args;
2864
2865
0
      opts = &opts_and_args->options;
2866
0
      opts->name = XNEWVEC (const char *, num_options + 1);
2867
0
      opts->description = XNEWVEC (const char *, num_options + 1);
2868
0
      opts->arg = XNEWVEC (const disasm_option_arg_t *, num_options + 1);
2869
0
      for (i = 0; i < num_options; i++)
2870
0
  {
2871
0
    opts->name[i] = mips_options[i].name;
2872
0
    opts->description[i] = _(mips_options[i].description);
2873
0
    if (mips_options[i].arg != MIPS_OPTION_ARG_NONE)
2874
0
      opts->arg[i] = &args[mips_options[i].arg];
2875
0
    else
2876
0
      opts->arg[i] = NULL;
2877
0
  }
2878
      /* The array we return must be NULL terminated.  */
2879
0
      opts->name[i] = NULL;
2880
0
      opts->description[i] = NULL;
2881
0
      opts->arg[i] = NULL;
2882
0
    }
2883
2884
0
  return opts_and_args;
2885
0
}
2886
2887
void
2888
print_mips_disassembler_options (FILE *stream)
2889
0
{
2890
0
  const disasm_options_and_args_t *opts_and_args;
2891
0
  const disasm_option_arg_t *args;
2892
0
  const disasm_options_t *opts;
2893
0
  size_t max_len = 0;
2894
0
  size_t i;
2895
0
  size_t j;
2896
2897
0
  opts_and_args = disassembler_options_mips ();
2898
0
  opts = &opts_and_args->options;
2899
0
  args = opts_and_args->args;
2900
2901
0
  fprintf (stream, _("\n\
2902
0
The following MIPS specific disassembler options are supported for use\n\
2903
0
with the -M switch (multiple options should be separated by commas):\n\n"));
2904
2905
  /* Compute the length of the longest option name.  */
2906
0
  for (i = 0; opts->name[i] != NULL; i++)
2907
0
    {
2908
0
      size_t len = strlen (opts->name[i]);
2909
2910
0
      if (opts->arg[i] != NULL)
2911
0
  len += strlen (opts->arg[i]->name);
2912
0
      if (max_len < len)
2913
0
  max_len = len;
2914
0
    }
2915
2916
0
  for (i = 0, max_len++; opts->name[i] != NULL; i++)
2917
0
    {
2918
0
      fprintf (stream, "  %s", opts->name[i]);
2919
0
      if (opts->arg[i] != NULL)
2920
0
  fprintf (stream, "%s", opts->arg[i]->name);
2921
0
      if (opts->description[i] != NULL)
2922
0
  {
2923
0
    size_t len = strlen (opts->name[i]);
2924
2925
0
    if (opts->arg[i] != NULL)
2926
0
      len += strlen (opts->arg[i]->name);
2927
0
    fprintf (stream,
2928
0
       "%*c %s", (int) (max_len - len), ' ', opts->description[i]);
2929
0
  }
2930
0
      fprintf (stream, _("\n"));
2931
0
    }
2932
2933
0
  for (i = 0; args[i].name != NULL; i++)
2934
0
    {
2935
0
      if (args[i].values == NULL)
2936
0
  continue;
2937
0
      fprintf (stream, _("\n\
2938
0
  For the options above, the following values are supported for \"%s\":\n   "),
2939
0
         args[i].name);
2940
0
      for (j = 0; args[i].values[j] != NULL; j++)
2941
0
  fprintf (stream, " %s", args[i].values[j]);
2942
0
      fprintf (stream, _("\n"));
2943
0
    }
2944
2945
  fprintf (stream, _("\n"));
2946
0
}