1 /* aarch64-asm.c -- AArch64 assembler support.
2    Copyright (C) 2012-2014 Free Software Foundation, Inc.
3    Contributed by ARM Ltd.
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; see the file COPYING3. If not,
19    see <http://www.gnu.org/licenses/>.  */
20 
21 #include "sysdep.h"
22 #include <stdarg.h>
23 #include "aarch64-asm.h"
24 
25 /* Utilities.  */
26 
27 /* The unnamed arguments consist of the number of fields and information about
28    these fields where the VALUE will be inserted into CODE.  MASK can be zero or
29    the base mask of the opcode.
30 
31    N.B. the fields are required to be in such an order than the least signficant
32    field for VALUE comes the first, e.g. the <index> in
33     SQDMLAL <Va><d>, <Vb><n>, <Vm>.<Ts>[<index>]
34    is encoded in H:L:M in some cases, the fields H:L:M should be passed in
35    the order of M, L, H.  */
36 
37 static inline void
insert_fields(aarch64_insn * code,aarch64_insn value,aarch64_insn mask,...)38 insert_fields (aarch64_insn *code, aarch64_insn value, aarch64_insn mask, ...)
39 {
40   uint32_t num;
41   const aarch64_field *field;
42   enum aarch64_field_kind kind;
43   va_list va;
44 
45   va_start (va, mask);
46   num = va_arg (va, uint32_t);
47   assert (num <= 5);
48   while (num--)
49     {
50       kind = va_arg (va, enum aarch64_field_kind);
51       field = &fields[kind];
52       insert_field (kind, code, value, mask);
53       value >>= field->width;
54     }
55   va_end (va);
56 }
57 
58 /* Operand inserters.  */
59 
60 /* Insert register number.  */
61 const char *
aarch64_ins_regno(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)62 aarch64_ins_regno (const aarch64_operand *self, const aarch64_opnd_info *info,
63 		   aarch64_insn *code,
64 		   const aarch64_inst *inst ATTRIBUTE_UNUSED)
65 {
66   insert_field (self->fields[0], code, info->reg.regno, 0);
67   return NULL;
68 }
69 
70 /* Insert register number, index and/or other data for SIMD register element
71    operand, e.g. the last source operand in
72      SQDMLAL <Va><d>, <Vb><n>, <Vm>.<Ts>[<index>].  */
73 const char *
aarch64_ins_reglane(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst)74 aarch64_ins_reglane (const aarch64_operand *self, const aarch64_opnd_info *info,
75 		     aarch64_insn *code, const aarch64_inst *inst)
76 {
77   /* regno */
78   insert_field (self->fields[0], code, info->reglane.regno, inst->opcode->mask);
79   /* index and/or type */
80   if (inst->opcode->iclass == asisdone || inst->opcode->iclass == asimdins)
81     {
82       int pos = info->qualifier - AARCH64_OPND_QLF_S_B;
83       if (info->type == AARCH64_OPND_En
84 	  && inst->opcode->operands[0] == AARCH64_OPND_Ed)
85 	{
86 	  /* index2 for e.g. INS <Vd>.<Ts>[<index1>], <Vn>.<Ts>[<index2>].  */
87 	  assert (info->idx == 1);	/* Vn */
88 	  aarch64_insn value = info->reglane.index << pos;
89 	  insert_field (FLD_imm4, code, value, 0);
90 	}
91       else
92 	{
93 	  /* index and type for e.g. DUP <V><d>, <Vn>.<T>[<index>].
94 	     imm5<3:0>	<V>
95 	     0000	RESERVED
96 	     xxx1	B
97 	     xx10	H
98 	     x100	S
99 	     1000	D  */
100 	  aarch64_insn value = ((info->reglane.index << 1) | 1) << pos;
101 	  insert_field (FLD_imm5, code, value, 0);
102 	}
103     }
104   else
105     {
106       /* index for e.g. SQDMLAL <Va><d>, <Vb><n>, <Vm>.<Ts>[<index>]
107          or SQDMLAL <Va><d>, <Vb><n>, <Vm>.<Ts>[<index>].  */
108       switch (info->qualifier)
109 	{
110 	case AARCH64_OPND_QLF_S_H:
111 	  /* H:L:M */
112 	  insert_fields (code, info->reglane.index, 0, 3, FLD_M, FLD_L, FLD_H);
113 	  break;
114 	case AARCH64_OPND_QLF_S_S:
115 	  /* H:L */
116 	  insert_fields (code, info->reglane.index, 0, 2, FLD_L, FLD_H);
117 	  break;
118 	case AARCH64_OPND_QLF_S_D:
119 	  /* H */
120 	  insert_field (FLD_H, code, info->reglane.index, 0);
121 	  break;
122 	default:
123 	  assert (0);
124 	}
125     }
126   return NULL;
127 }
128 
129 /* Insert regno and len field of a register list operand, e.g. Vn in TBL.  */
130 const char *
aarch64_ins_reglist(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)131 aarch64_ins_reglist (const aarch64_operand *self, const aarch64_opnd_info *info,
132 		     aarch64_insn *code,
133 		     const aarch64_inst *inst ATTRIBUTE_UNUSED)
134 {
135   /* R */
136   insert_field (self->fields[0], code, info->reglist.first_regno, 0);
137   /* len */
138   insert_field (FLD_len, code, info->reglist.num_regs - 1, 0);
139   return NULL;
140 }
141 
142 /* Insert Rt and opcode fields for a register list operand, e.g. Vt
143    in AdvSIMD load/store instructions.  */
144 const char *
aarch64_ins_ldst_reglist(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst)145 aarch64_ins_ldst_reglist (const aarch64_operand *self ATTRIBUTE_UNUSED,
146 			  const aarch64_opnd_info *info, aarch64_insn *code,
147 			  const aarch64_inst *inst)
148 {
149   aarch64_insn value = 0;
150   /* Number of elements in each structure to be loaded/stored.  */
151   unsigned num = get_opcode_dependent_value (inst->opcode);
152 
153   /* Rt */
154   insert_field (FLD_Rt, code, info->reglist.first_regno, 0);
155   /* opcode */
156   switch (num)
157     {
158     case 1:
159       switch (info->reglist.num_regs)
160 	{
161 	case 1: value = 0x7; break;
162 	case 2: value = 0xa; break;
163 	case 3: value = 0x6; break;
164 	case 4: value = 0x2; break;
165 	default: assert (0);
166 	}
167       break;
168     case 2:
169       value = info->reglist.num_regs == 4 ? 0x3 : 0x8;
170       break;
171     case 3:
172       value = 0x4;
173       break;
174     case 4:
175       value = 0x0;
176       break;
177     default:
178       assert (0);
179     }
180   insert_field (FLD_opcode, code, value, 0);
181 
182   return NULL;
183 }
184 
185 /* Insert Rt and S fields for a register list operand, e.g. Vt in AdvSIMD load
186    single structure to all lanes instructions.  */
187 const char *
aarch64_ins_ldst_reglist_r(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst)188 aarch64_ins_ldst_reglist_r (const aarch64_operand *self ATTRIBUTE_UNUSED,
189 			    const aarch64_opnd_info *info, aarch64_insn *code,
190 			    const aarch64_inst *inst)
191 {
192   aarch64_insn value;
193   /* The opcode dependent area stores the number of elements in
194      each structure to be loaded/stored.  */
195   int is_ld1r = get_opcode_dependent_value (inst->opcode) == 1;
196 
197   /* Rt */
198   insert_field (FLD_Rt, code, info->reglist.first_regno, 0);
199   /* S */
200   value = (aarch64_insn) 0;
201   if (is_ld1r && info->reglist.num_regs == 2)
202     /* OP_LD1R does not have alternating variant, but have "two consecutive"
203        instead.  */
204     value = (aarch64_insn) 1;
205   insert_field (FLD_S, code, value, 0);
206 
207   return NULL;
208 }
209 
210 /* Insert Q, opcode<2:1>, S, size and Rt fields for a register element list
211    operand e.g. Vt in AdvSIMD load/store single element instructions.  */
212 const char *
aarch64_ins_ldst_elemlist(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)213 aarch64_ins_ldst_elemlist (const aarch64_operand *self ATTRIBUTE_UNUSED,
214 			   const aarch64_opnd_info *info, aarch64_insn *code,
215 			   const aarch64_inst *inst ATTRIBUTE_UNUSED)
216 {
217   aarch64_field field = {0, 0};
218   aarch64_insn QSsize = 0;	/* fields Q:S:size.  */
219   aarch64_insn opcodeh2 = 0;	/* opcode<2:1> */
220 
221   assert (info->reglist.has_index);
222 
223   /* Rt */
224   insert_field (FLD_Rt, code, info->reglist.first_regno, 0);
225   /* Encode the index, opcode<2:1> and size.  */
226   switch (info->qualifier)
227     {
228     case AARCH64_OPND_QLF_S_B:
229       /* Index encoded in "Q:S:size".  */
230       QSsize = info->reglist.index;
231       opcodeh2 = 0x0;
232       break;
233     case AARCH64_OPND_QLF_S_H:
234       /* Index encoded in "Q:S:size<1>".  */
235       QSsize = info->reglist.index << 1;
236       opcodeh2 = 0x1;
237       break;
238     case AARCH64_OPND_QLF_S_S:
239       /* Index encoded in "Q:S".  */
240       QSsize = info->reglist.index << 2;
241       opcodeh2 = 0x2;
242       break;
243     case AARCH64_OPND_QLF_S_D:
244       /* Index encoded in "Q".  */
245       QSsize = info->reglist.index << 3 | 0x1;
246       opcodeh2 = 0x2;
247       break;
248     default:
249       assert (0);
250     }
251   insert_fields (code, QSsize, 0, 3, FLD_vldst_size, FLD_S, FLD_Q);
252   gen_sub_field (FLD_asisdlso_opcode, 1, 2, &field);
253   insert_field_2 (&field, code, opcodeh2, 0);
254 
255   return NULL;
256 }
257 
258 /* Insert fields immh:immb and/or Q for e.g. the shift immediate in
259    SSHR <Vd>.<T>, <Vn>.<T>, #<shift>
260    or SSHR <V><d>, <V><n>, #<shift>.  */
261 const char *
aarch64_ins_advsimd_imm_shift(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst)262 aarch64_ins_advsimd_imm_shift (const aarch64_operand *self ATTRIBUTE_UNUSED,
263 			       const aarch64_opnd_info *info,
264 			       aarch64_insn *code, const aarch64_inst *inst)
265 {
266   unsigned val = aarch64_get_qualifier_standard_value (info->qualifier);
267   aarch64_insn Q, imm;
268 
269   if (inst->opcode->iclass == asimdshf)
270     {
271       /* Q
272 	 immh	Q	<T>
273 	 0000	x	SEE AdvSIMD modified immediate
274 	 0001	0	8B
275 	 0001	1	16B
276 	 001x	0	4H
277 	 001x	1	8H
278 	 01xx	0	2S
279 	 01xx	1	4S
280 	 1xxx	0	RESERVED
281 	 1xxx	1	2D  */
282       Q = (val & 0x1) ? 1 : 0;
283       insert_field (FLD_Q, code, Q, inst->opcode->mask);
284       val >>= 1;
285     }
286 
287   assert (info->type == AARCH64_OPND_IMM_VLSR
288 	  || info->type == AARCH64_OPND_IMM_VLSL);
289 
290   if (info->type == AARCH64_OPND_IMM_VLSR)
291     /* immh:immb
292        immh	<shift>
293        0000	SEE AdvSIMD modified immediate
294        0001	(16-UInt(immh:immb))
295        001x	(32-UInt(immh:immb))
296        01xx	(64-UInt(immh:immb))
297        1xxx	(128-UInt(immh:immb))  */
298     imm = (16 << (unsigned)val) - info->imm.value;
299   else
300     /* immh:immb
301        immh	<shift>
302        0000	SEE AdvSIMD modified immediate
303        0001	(UInt(immh:immb)-8)
304        001x	(UInt(immh:immb)-16)
305        01xx	(UInt(immh:immb)-32)
306        1xxx	(UInt(immh:immb)-64)  */
307     imm = info->imm.value + (8 << (unsigned)val);
308   insert_fields (code, imm, 0, 2, FLD_immb, FLD_immh);
309 
310   return NULL;
311 }
312 
313 /* Insert fields for e.g. the immediate operands in
314    BFM <Wd>, <Wn>, #<immr>, #<imms>.  */
315 const char *
aarch64_ins_imm(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)316 aarch64_ins_imm (const aarch64_operand *self, const aarch64_opnd_info *info,
317 		 aarch64_insn *code,
318 		 const aarch64_inst *inst ATTRIBUTE_UNUSED)
319 {
320   int64_t imm;
321   /* Maximum of two fields to insert.  */
322   assert (self->fields[2] == FLD_NIL);
323 
324   imm = info->imm.value;
325   if (operand_need_shift_by_two (self))
326     imm >>= 2;
327   if (self->fields[1] == FLD_NIL)
328     insert_field (self->fields[0], code, imm, 0);
329   else
330     /* e.g. TBZ b5:b40.  */
331     insert_fields (code, imm, 0, 2, self->fields[1], self->fields[0]);
332   return NULL;
333 }
334 
335 /* Insert immediate and its shift amount for e.g. the last operand in
336      MOVZ <Wd>, #<imm16>{, LSL #<shift>}.  */
337 const char *
aarch64_ins_imm_half(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst)338 aarch64_ins_imm_half (const aarch64_operand *self, const aarch64_opnd_info *info,
339 		      aarch64_insn *code, const aarch64_inst *inst)
340 {
341   /* imm16 */
342   aarch64_ins_imm (self, info, code, inst);
343   /* hw */
344   insert_field (FLD_hw, code, info->shifter.amount >> 4, 0);
345   return NULL;
346 }
347 
348 /* Insert cmode and "a:b:c:d:e:f:g:h" fields for e.g. the last operand in
349      MOVI <Vd>.<T>, #<imm8> {, LSL #<amount>}.  */
350 const char *
aarch64_ins_advsimd_imm_modified(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)351 aarch64_ins_advsimd_imm_modified (const aarch64_operand *self ATTRIBUTE_UNUSED,
352 				  const aarch64_opnd_info *info,
353 				  aarch64_insn *code,
354 				  const aarch64_inst *inst ATTRIBUTE_UNUSED)
355 {
356   enum aarch64_opnd_qualifier opnd0_qualifier = inst->operands[0].qualifier;
357   uint64_t imm = info->imm.value;
358   enum aarch64_modifier_kind kind = info->shifter.kind;
359   int amount = info->shifter.amount;
360   aarch64_field field = {0, 0};
361 
362   /* a:b:c:d:e:f:g:h */
363   if (!info->imm.is_fp && aarch64_get_qualifier_esize (opnd0_qualifier) == 8)
364     {
365       /* Either MOVI <Dd>, #<imm>
366 	 or     MOVI <Vd>.2D, #<imm>.
367 	 <imm> is a 64-bit immediate
368 	 "aaaaaaaabbbbbbbbccccccccddddddddeeeeeeeeffffffffgggggggghhhhhhhh",
369 	 encoded in "a:b:c:d:e:f:g:h".	*/
370       imm = aarch64_shrink_expanded_imm8 (imm);
371       assert ((int)imm >= 0);
372     }
373   insert_fields (code, imm, 0, 2, FLD_defgh, FLD_abc);
374 
375   if (kind == AARCH64_MOD_NONE)
376     return NULL;
377 
378   /* shift amount partially in cmode */
379   assert (kind == AARCH64_MOD_LSL || kind == AARCH64_MOD_MSL);
380   if (kind == AARCH64_MOD_LSL)
381     {
382       /* AARCH64_MOD_LSL: shift zeros.  */
383       int esize = aarch64_get_qualifier_esize (opnd0_qualifier);
384       assert (esize == 4 || esize == 2 || esize == 1);
385       /* For 8-bit move immediate, the optional LSL #0 does not require
386 	 encoding.  */
387       if (esize == 1)
388 	return NULL;
389       amount >>= 3;
390       if (esize == 4)
391 	gen_sub_field (FLD_cmode, 1, 2, &field);	/* per word */
392       else
393 	gen_sub_field (FLD_cmode, 1, 1, &field);	/* per halfword */
394     }
395   else
396     {
397       /* AARCH64_MOD_MSL: shift ones.  */
398       amount >>= 4;
399       gen_sub_field (FLD_cmode, 0, 1, &field);		/* per word */
400     }
401   insert_field_2 (&field, code, amount, 0);
402 
403   return NULL;
404 }
405 
406 /* Insert #<fbits> for the immediate operand in fp fix-point instructions,
407    e.g.  SCVTF <Dd>, <Wn>, #<fbits>.  */
408 const char *
aarch64_ins_fbits(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)409 aarch64_ins_fbits (const aarch64_operand *self, const aarch64_opnd_info *info,
410 		   aarch64_insn *code,
411 		   const aarch64_inst *inst ATTRIBUTE_UNUSED)
412 {
413   insert_field (self->fields[0], code, 64 - info->imm.value, 0);
414   return NULL;
415 }
416 
417 /* Insert arithmetic immediate for e.g. the last operand in
418      SUBS <Wd>, <Wn|WSP>, #<imm> {, <shift>}.  */
419 const char *
aarch64_ins_aimm(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)420 aarch64_ins_aimm (const aarch64_operand *self, const aarch64_opnd_info *info,
421 		  aarch64_insn *code, const aarch64_inst *inst ATTRIBUTE_UNUSED)
422 {
423   /* shift */
424   aarch64_insn value = info->shifter.amount ? 1 : 0;
425   insert_field (self->fields[0], code, value, 0);
426   /* imm12 (unsigned) */
427   insert_field (self->fields[1], code, info->imm.value, 0);
428   return NULL;
429 }
430 
431 /* Insert logical/bitmask immediate for e.g. the last operand in
432      ORR <Wd|WSP>, <Wn>, #<imm>.  */
433 const char *
aarch64_ins_limm(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)434 aarch64_ins_limm (const aarch64_operand *self, const aarch64_opnd_info *info,
435 		  aarch64_insn *code, const aarch64_inst *inst ATTRIBUTE_UNUSED)
436 {
437   aarch64_insn value;
438   uint64_t imm = info->imm.value;
439   int is32 = aarch64_get_qualifier_esize (inst->operands[0].qualifier) == 4;
440 
441   if (inst->opcode->op == OP_BIC)
442     imm = ~imm;
443   if (aarch64_logical_immediate_p (imm, is32, &value) == FALSE)
444     /* The constraint check should have guaranteed this wouldn't happen.  */
445     assert (0);
446 
447   insert_fields (code, value, 0, 3, self->fields[2], self->fields[1],
448 		 self->fields[0]);
449   return NULL;
450 }
451 
452 /* Encode Ft for e.g. STR <Qt>, [<Xn|SP>, <R><m>{, <extend> {<amount>}}]
453    or LDP <Qt1>, <Qt2>, [<Xn|SP>], #<imm>.  */
454 const char *
aarch64_ins_ft(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst)455 aarch64_ins_ft (const aarch64_operand *self, const aarch64_opnd_info *info,
456 		aarch64_insn *code, const aarch64_inst *inst)
457 {
458   aarch64_insn value = 0;
459 
460   assert (info->idx == 0);
461 
462   /* Rt */
463   aarch64_ins_regno (self, info, code, inst);
464   if (inst->opcode->iclass == ldstpair_indexed
465       || inst->opcode->iclass == ldstnapair_offs
466       || inst->opcode->iclass == ldstpair_off
467       || inst->opcode->iclass == loadlit)
468     {
469       /* size */
470       switch (info->qualifier)
471 	{
472 	case AARCH64_OPND_QLF_S_S: value = 0; break;
473 	case AARCH64_OPND_QLF_S_D: value = 1; break;
474 	case AARCH64_OPND_QLF_S_Q: value = 2; break;
475 	default: assert (0);
476 	}
477       insert_field (FLD_ldst_size, code, value, 0);
478     }
479   else
480     {
481       /* opc[1]:size */
482       value = aarch64_get_qualifier_standard_value (info->qualifier);
483       insert_fields (code, value, 0, 2, FLD_ldst_size, FLD_opc1);
484     }
485 
486   return NULL;
487 }
488 
489 /* Encode the address operand for e.g. STXRB <Ws>, <Wt>, [<Xn|SP>{,#0}].  */
490 const char *
aarch64_ins_addr_simple(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)491 aarch64_ins_addr_simple (const aarch64_operand *self ATTRIBUTE_UNUSED,
492 			 const aarch64_opnd_info *info, aarch64_insn *code,
493 			 const aarch64_inst *inst ATTRIBUTE_UNUSED)
494 {
495   /* Rn */
496   insert_field (FLD_Rn, code, info->addr.base_regno, 0);
497   return NULL;
498 }
499 
500 /* Encode the address operand for e.g.
501      STR <Qt>, [<Xn|SP>, <R><m>{, <extend> {<amount>}}].  */
502 const char *
aarch64_ins_addr_regoff(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)503 aarch64_ins_addr_regoff (const aarch64_operand *self ATTRIBUTE_UNUSED,
504 			 const aarch64_opnd_info *info, aarch64_insn *code,
505 			 const aarch64_inst *inst ATTRIBUTE_UNUSED)
506 {
507   aarch64_insn S;
508   enum aarch64_modifier_kind kind = info->shifter.kind;
509 
510   /* Rn */
511   insert_field (FLD_Rn, code, info->addr.base_regno, 0);
512   /* Rm */
513   insert_field (FLD_Rm, code, info->addr.offset.regno, 0);
514   /* option */
515   if (kind == AARCH64_MOD_LSL)
516     kind = AARCH64_MOD_UXTX;	/* Trick to enable the table-driven.  */
517   insert_field (FLD_option, code, aarch64_get_operand_modifier_value (kind), 0);
518   /* S */
519   if (info->qualifier != AARCH64_OPND_QLF_S_B)
520     S = info->shifter.amount != 0;
521   else
522     /* For STR <Bt>, [<Xn|SP>, <R><m>{, <extend> {<amount>}},
523        S	<amount>
524        0	[absent]
525        1	#0
526        Must be #0 if <extend> is explicitly LSL.  */
527     S = info->shifter.operator_present && info->shifter.amount_present;
528   insert_field (FLD_S, code, S, 0);
529 
530   return NULL;
531 }
532 
533 /* Encode the address operand for e.g. LDRSW <Xt>, [<Xn|SP>, #<simm>]!.  */
534 const char *
aarch64_ins_addr_simm(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)535 aarch64_ins_addr_simm (const aarch64_operand *self,
536 		       const aarch64_opnd_info *info,
537 		       aarch64_insn *code,
538 		       const aarch64_inst *inst ATTRIBUTE_UNUSED)
539 {
540   int imm;
541 
542   /* Rn */
543   insert_field (FLD_Rn, code, info->addr.base_regno, 0);
544   /* simm (imm9 or imm7) */
545   imm = info->addr.offset.imm;
546   if (self->fields[0] == FLD_imm7)
547     /* scaled immediate in ld/st pair instructions..  */
548     imm >>= get_logsz (aarch64_get_qualifier_esize (info->qualifier));
549   insert_field (self->fields[0], code, imm, 0);
550   /* pre/post- index */
551   if (info->addr.writeback)
552     {
553       assert (inst->opcode->iclass != ldst_unscaled
554 	      && inst->opcode->iclass != ldstnapair_offs
555 	      && inst->opcode->iclass != ldstpair_off
556 	      && inst->opcode->iclass != ldst_unpriv);
557       assert (info->addr.preind != info->addr.postind);
558       if (info->addr.preind)
559 	insert_field (self->fields[1], code, 1, 0);
560     }
561 
562   return NULL;
563 }
564 
565 /* Encode the address operand for e.g. LDRSW <Xt>, [<Xn|SP>{, #<pimm>}].  */
566 const char *
aarch64_ins_addr_uimm12(const aarch64_operand * self,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)567 aarch64_ins_addr_uimm12 (const aarch64_operand *self,
568 			 const aarch64_opnd_info *info,
569 			 aarch64_insn *code,
570 			 const aarch64_inst *inst ATTRIBUTE_UNUSED)
571 {
572   int shift = get_logsz (aarch64_get_qualifier_esize (info->qualifier));
573 
574   /* Rn */
575   insert_field (self->fields[0], code, info->addr.base_regno, 0);
576   /* uimm12 */
577   insert_field (self->fields[1], code,info->addr.offset.imm >> shift, 0);
578   return NULL;
579 }
580 
581 /* Encode the address operand for e.g.
582      LD1 {<Vt>.<T>, <Vt2>.<T>, <Vt3>.<T>}, [<Xn|SP>], <Xm|#<amount>>.  */
583 const char *
aarch64_ins_simd_addr_post(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)584 aarch64_ins_simd_addr_post (const aarch64_operand *self ATTRIBUTE_UNUSED,
585 			    const aarch64_opnd_info *info, aarch64_insn *code,
586 			    const aarch64_inst *inst ATTRIBUTE_UNUSED)
587 {
588   /* Rn */
589   insert_field (FLD_Rn, code, info->addr.base_regno, 0);
590   /* Rm | #<amount>  */
591   if (info->addr.offset.is_reg)
592     insert_field (FLD_Rm, code, info->addr.offset.regno, 0);
593   else
594     insert_field (FLD_Rm, code, 0x1f, 0);
595   return NULL;
596 }
597 
598 /* Encode the condition operand for e.g. CSEL <Xd>, <Xn>, <Xm>, <cond>.  */
599 const char *
aarch64_ins_cond(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)600 aarch64_ins_cond (const aarch64_operand *self ATTRIBUTE_UNUSED,
601 		  const aarch64_opnd_info *info, aarch64_insn *code,
602 		  const aarch64_inst *inst ATTRIBUTE_UNUSED)
603 {
604   /* cond */
605   insert_field (FLD_cond, code, info->cond->value, 0);
606   return NULL;
607 }
608 
609 /* Encode the system register operand for e.g. MRS <Xt>, <systemreg>.  */
610 const char *
aarch64_ins_sysreg(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)611 aarch64_ins_sysreg (const aarch64_operand *self ATTRIBUTE_UNUSED,
612 		    const aarch64_opnd_info *info, aarch64_insn *code,
613 		    const aarch64_inst *inst ATTRIBUTE_UNUSED)
614 {
615   /* op0:op1:CRn:CRm:op2 */
616   insert_fields (code, info->sysreg, inst->opcode->mask, 5,
617 		 FLD_op2, FLD_CRm, FLD_CRn, FLD_op1, FLD_op0);
618   return NULL;
619 }
620 
621 /* Encode the PSTATE field operand for e.g. MSR <pstatefield>, #<imm>.  */
622 const char *
aarch64_ins_pstatefield(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)623 aarch64_ins_pstatefield (const aarch64_operand *self ATTRIBUTE_UNUSED,
624 			 const aarch64_opnd_info *info, aarch64_insn *code,
625 			 const aarch64_inst *inst ATTRIBUTE_UNUSED)
626 {
627   /* op1:op2 */
628   insert_fields (code, info->pstatefield, inst->opcode->mask, 2,
629 		 FLD_op2, FLD_op1);
630   return NULL;
631 }
632 
633 /* Encode the system instruction op operand for e.g. AT <at_op>, <Xt>.  */
634 const char *
aarch64_ins_sysins_op(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)635 aarch64_ins_sysins_op (const aarch64_operand *self ATTRIBUTE_UNUSED,
636 		       const aarch64_opnd_info *info, aarch64_insn *code,
637 		       const aarch64_inst *inst ATTRIBUTE_UNUSED)
638 {
639   /* op1:CRn:CRm:op2 */
640   insert_fields (code, info->sysins_op->value, inst->opcode->mask, 4,
641 		 FLD_op2, FLD_CRm, FLD_CRn, FLD_op1);
642   return NULL;
643 }
644 
645 /* Encode the memory barrier option operand for e.g. DMB <option>|#<imm>.  */
646 
647 const char *
aarch64_ins_barrier(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)648 aarch64_ins_barrier (const aarch64_operand *self ATTRIBUTE_UNUSED,
649 		     const aarch64_opnd_info *info, aarch64_insn *code,
650 		     const aarch64_inst *inst ATTRIBUTE_UNUSED)
651 {
652   /* CRm */
653   insert_field (FLD_CRm, code, info->barrier->value, 0);
654   return NULL;
655 }
656 
657 /* Encode the prefetch operation option operand for e.g.
658      PRFM <prfop>, [<Xn|SP>{, #<pimm>}].  */
659 
660 const char *
aarch64_ins_prfop(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)661 aarch64_ins_prfop (const aarch64_operand *self ATTRIBUTE_UNUSED,
662 		   const aarch64_opnd_info *info, aarch64_insn *code,
663 		   const aarch64_inst *inst ATTRIBUTE_UNUSED)
664 {
665   /* prfop in Rt */
666   insert_field (FLD_Rt, code, info->prfop->value, 0);
667   return NULL;
668 }
669 
670 /* Encode the extended register operand for e.g.
671      STR <Qt>, [<Xn|SP>, <R><m>{, <extend> {<amount>}}].  */
672 const char *
aarch64_ins_reg_extended(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)673 aarch64_ins_reg_extended (const aarch64_operand *self ATTRIBUTE_UNUSED,
674 			  const aarch64_opnd_info *info, aarch64_insn *code,
675 			  const aarch64_inst *inst ATTRIBUTE_UNUSED)
676 {
677   enum aarch64_modifier_kind kind;
678 
679   /* Rm */
680   insert_field (FLD_Rm, code, info->reg.regno, 0);
681   /* option */
682   kind = info->shifter.kind;
683   if (kind == AARCH64_MOD_LSL)
684     kind = info->qualifier == AARCH64_OPND_QLF_W
685       ? AARCH64_MOD_UXTW : AARCH64_MOD_UXTX;
686   insert_field (FLD_option, code, aarch64_get_operand_modifier_value (kind), 0);
687   /* imm3 */
688   insert_field (FLD_imm3, code, info->shifter.amount, 0);
689 
690   return NULL;
691 }
692 
693 /* Encode the shifted register operand for e.g.
694      SUBS <Xd>, <Xn>, <Xm> {, <shift> #<amount>}.  */
695 const char *
aarch64_ins_reg_shifted(const aarch64_operand * self ATTRIBUTE_UNUSED,const aarch64_opnd_info * info,aarch64_insn * code,const aarch64_inst * inst ATTRIBUTE_UNUSED)696 aarch64_ins_reg_shifted (const aarch64_operand *self ATTRIBUTE_UNUSED,
697 			 const aarch64_opnd_info *info, aarch64_insn *code,
698 			 const aarch64_inst *inst ATTRIBUTE_UNUSED)
699 {
700   /* Rm */
701   insert_field (FLD_Rm, code, info->reg.regno, 0);
702   /* shift */
703   insert_field (FLD_shift, code,
704 		aarch64_get_operand_modifier_value (info->shifter.kind), 0);
705   /* imm6 */
706   insert_field (FLD_imm6, code, info->shifter.amount, 0);
707 
708   return NULL;
709 }
710 
711 /* Miscellaneous encoding functions.  */
712 
713 /* Encode size[0], i.e. bit 22, for
714      e.g. FCVTN<Q> <Vd>.<Tb>, <Vn>.<Ta>.  */
715 
716 static void
encode_asimd_fcvt(aarch64_inst * inst)717 encode_asimd_fcvt (aarch64_inst *inst)
718 {
719   aarch64_insn value;
720   aarch64_field field = {0, 0};
721   enum aarch64_opnd_qualifier qualifier;
722 
723   switch (inst->opcode->op)
724     {
725     case OP_FCVTN:
726     case OP_FCVTN2:
727       /* FCVTN<Q> <Vd>.<Tb>, <Vn>.<Ta>.  */
728       qualifier = inst->operands[1].qualifier;
729       break;
730     case OP_FCVTL:
731     case OP_FCVTL2:
732       /* FCVTL<Q> <Vd>.<Ta>, <Vn>.<Tb>.  */
733       qualifier = inst->operands[0].qualifier;
734       break;
735     default:
736       assert (0);
737     }
738   assert (qualifier == AARCH64_OPND_QLF_V_4S
739 	  || qualifier == AARCH64_OPND_QLF_V_2D);
740   value = (qualifier == AARCH64_OPND_QLF_V_4S) ? 0 : 1;
741   gen_sub_field (FLD_size, 0, 1, &field);
742   insert_field_2 (&field, &inst->value, value, 0);
743 }
744 
745 /* Encode size[0], i.e. bit 22, for
746      e.g. FCVTXN <Vb><d>, <Va><n>.  */
747 
748 static void
encode_asisd_fcvtxn(aarch64_inst * inst)749 encode_asisd_fcvtxn (aarch64_inst *inst)
750 {
751   aarch64_insn val = 1;
752   aarch64_field field = {0, 0};
753   assert (inst->operands[0].qualifier == AARCH64_OPND_QLF_S_S);
754   gen_sub_field (FLD_size, 0, 1, &field);
755   insert_field_2 (&field, &inst->value, val, 0);
756 }
757 
758 /* Encode the 'opc' field for e.g. FCVT <Dd>, <Sn>.  */
759 static void
encode_fcvt(aarch64_inst * inst)760 encode_fcvt (aarch64_inst *inst)
761 {
762   aarch64_insn val;
763   const aarch64_field field = {15, 2};
764 
765   /* opc dstsize */
766   switch (inst->operands[0].qualifier)
767     {
768     case AARCH64_OPND_QLF_S_S: val = 0; break;
769     case AARCH64_OPND_QLF_S_D: val = 1; break;
770     case AARCH64_OPND_QLF_S_H: val = 3; break;
771     default: abort ();
772     }
773   insert_field_2 (&field, &inst->value, val, 0);
774 
775   return;
776 }
777 
778 /* Do miscellaneous encodings that are not common enough to be driven by
779    flags.  */
780 
781 static void
do_misc_encoding(aarch64_inst * inst)782 do_misc_encoding (aarch64_inst *inst)
783 {
784   switch (inst->opcode->op)
785     {
786     case OP_FCVT:
787       encode_fcvt (inst);
788       break;
789     case OP_FCVTN:
790     case OP_FCVTN2:
791     case OP_FCVTL:
792     case OP_FCVTL2:
793       encode_asimd_fcvt (inst);
794       break;
795     case OP_FCVTXN_S:
796       encode_asisd_fcvtxn (inst);
797       break;
798     default: break;
799     }
800 }
801 
802 /* Encode the 'size' and 'Q' field for e.g. SHADD.  */
803 static void
encode_sizeq(aarch64_inst * inst)804 encode_sizeq (aarch64_inst *inst)
805 {
806   aarch64_insn sizeq;
807   enum aarch64_field_kind kind;
808   int idx;
809 
810   /* Get the index of the operand whose information we are going to use
811      to encode the size and Q fields.
812      This is deduced from the possible valid qualifier lists.  */
813   idx = aarch64_select_operand_for_sizeq_field_coding (inst->opcode);
814   DEBUG_TRACE ("idx: %d; qualifier: %s", idx,
815 	       aarch64_get_qualifier_name (inst->operands[idx].qualifier));
816   sizeq = aarch64_get_qualifier_standard_value (inst->operands[idx].qualifier);
817   /* Q */
818   insert_field (FLD_Q, &inst->value, sizeq & 0x1, inst->opcode->mask);
819   /* size */
820   if (inst->opcode->iclass == asisdlse
821      || inst->opcode->iclass == asisdlsep
822      || inst->opcode->iclass == asisdlso
823      || inst->opcode->iclass == asisdlsop)
824     kind = FLD_vldst_size;
825   else
826     kind = FLD_size;
827   insert_field (kind, &inst->value, (sizeq >> 1) & 0x3, inst->opcode->mask);
828 }
829 
830 /* Opcodes that have fields shared by multiple operands are usually flagged
831    with flags.  In this function, we detect such flags and use the
832    information in one of the related operands to do the encoding.  The 'one'
833    operand is not any operand but one of the operands that has the enough
834    information for such an encoding.  */
835 
836 static void
do_special_encoding(struct aarch64_inst * inst)837 do_special_encoding (struct aarch64_inst *inst)
838 {
839   int idx;
840   aarch64_insn value = 0;
841 
842   DEBUG_TRACE ("enter with coding 0x%x", (uint32_t) inst->value);
843 
844   /* Condition for truly conditional executed instructions, e.g. b.cond.  */
845   if (inst->opcode->flags & F_COND)
846     {
847       insert_field (FLD_cond2, &inst->value, inst->cond->value, 0);
848     }
849   if (inst->opcode->flags & F_SF)
850     {
851       idx = select_operand_for_sf_field_coding (inst->opcode);
852       value = (inst->operands[idx].qualifier == AARCH64_OPND_QLF_X
853 	       || inst->operands[idx].qualifier == AARCH64_OPND_QLF_SP)
854 	? 1 : 0;
855       insert_field (FLD_sf, &inst->value, value, 0);
856       if (inst->opcode->flags & F_N)
857 	insert_field (FLD_N, &inst->value, value, inst->opcode->mask);
858     }
859   if (inst->opcode->flags & F_LSE_SZ)
860     {
861       idx = select_operand_for_sf_field_coding (inst->opcode);
862       value = (inst->operands[idx].qualifier == AARCH64_OPND_QLF_X
863 	       || inst->operands[idx].qualifier == AARCH64_OPND_QLF_SP)
864 	? 1 : 0;
865       insert_field (FLD_lse_sz, &inst->value, value, 0);
866     }
867   if (inst->opcode->flags & F_SIZEQ)
868     encode_sizeq (inst);
869   if (inst->opcode->flags & F_FPTYPE)
870     {
871       idx = select_operand_for_fptype_field_coding (inst->opcode);
872       switch (inst->operands[idx].qualifier)
873 	{
874 	case AARCH64_OPND_QLF_S_S: value = 0; break;
875 	case AARCH64_OPND_QLF_S_D: value = 1; break;
876 	case AARCH64_OPND_QLF_S_H: value = 3; break;
877 	default: assert (0);
878 	}
879       insert_field (FLD_type, &inst->value, value, 0);
880     }
881   if (inst->opcode->flags & F_SSIZE)
882     {
883       enum aarch64_opnd_qualifier qualifier;
884       idx = select_operand_for_scalar_size_field_coding (inst->opcode);
885       qualifier = inst->operands[idx].qualifier;
886       assert (qualifier >= AARCH64_OPND_QLF_S_B
887 	      && qualifier <= AARCH64_OPND_QLF_S_Q);
888       value = aarch64_get_qualifier_standard_value (qualifier);
889       insert_field (FLD_size, &inst->value, value, inst->opcode->mask);
890     }
891   if (inst->opcode->flags & F_T)
892     {
893       int num;	/* num of consecutive '0's on the right side of imm5<3:0>.  */
894       aarch64_field field = {0, 0};
895       enum aarch64_opnd_qualifier qualifier;
896 
897       idx = 0;
898       qualifier = inst->operands[idx].qualifier;
899       assert (aarch64_get_operand_class (inst->opcode->operands[0])
900 	      == AARCH64_OPND_CLASS_SIMD_REG
901 	      && qualifier >= AARCH64_OPND_QLF_V_8B
902 	      && qualifier <= AARCH64_OPND_QLF_V_2D);
903       /* imm5<3:0>	q	<t>
904 	 0000		x	reserved
905 	 xxx1		0	8b
906 	 xxx1		1	16b
907 	 xx10		0	4h
908 	 xx10		1	8h
909 	 x100		0	2s
910 	 x100		1	4s
911 	 1000		0	reserved
912 	 1000		1	2d  */
913       value = aarch64_get_qualifier_standard_value (qualifier);
914       insert_field (FLD_Q, &inst->value, value & 0x1, inst->opcode->mask);
915       num = (int) value >> 1;
916       assert (num >= 0 && num <= 3);
917       gen_sub_field (FLD_imm5, 0, num + 1, &field);
918       insert_field_2 (&field, &inst->value, 1 << num, inst->opcode->mask);
919     }
920   if (inst->opcode->flags & F_GPRSIZE_IN_Q)
921     {
922       /* Use Rt to encode in the case of e.g.
923 	 STXP <Ws>, <Xt1>, <Xt2>, [<Xn|SP>{,#0}].  */
924       enum aarch64_opnd_qualifier qualifier;
925       idx = aarch64_operand_index (inst->opcode->operands, AARCH64_OPND_Rt);
926       if (idx == -1)
927 	/* Otherwise use the result operand, which has to be a integer
928 	   register.  */
929 	idx = 0;
930       assert (idx == 0 || idx == 1);
931       assert (aarch64_get_operand_class (inst->opcode->operands[idx])
932 	      == AARCH64_OPND_CLASS_INT_REG);
933       qualifier = inst->operands[idx].qualifier;
934       insert_field (FLD_Q, &inst->value,
935 		    aarch64_get_qualifier_standard_value (qualifier), 0);
936     }
937   if (inst->opcode->flags & F_LDS_SIZE)
938     {
939       /* e.g. LDRSB <Wt>, [<Xn|SP>, <R><m>{, <extend> {<amount>}}].  */
940       enum aarch64_opnd_qualifier qualifier;
941       aarch64_field field = {0, 0};
942       assert (aarch64_get_operand_class (inst->opcode->operands[0])
943 	      == AARCH64_OPND_CLASS_INT_REG);
944       gen_sub_field (FLD_opc, 0, 1, &field);
945       qualifier = inst->operands[0].qualifier;
946       insert_field_2 (&field, &inst->value,
947 		      1 - aarch64_get_qualifier_standard_value (qualifier), 0);
948     }
949   /* Miscellaneous encoding as the last step.  */
950   if (inst->opcode->flags & F_MISC)
951     do_misc_encoding (inst);
952 
953   DEBUG_TRACE ("exit with coding 0x%x", (uint32_t) inst->value);
954 }
955 
956 /* Converters converting an alias opcode instruction to its real form.  */
957 
958 /* ROR <Wd>, <Ws>, #<shift>
959      is equivalent to:
960    EXTR <Wd>, <Ws>, <Ws>, #<shift>.  */
961 static void
convert_ror_to_extr(aarch64_inst * inst)962 convert_ror_to_extr (aarch64_inst *inst)
963 {
964   copy_operand_info (inst, 3, 2);
965   copy_operand_info (inst, 2, 1);
966 }
967 
968 /* UXTL<Q> <Vd>.<Ta>, <Vn>.<Tb>
969      is equivalent to:
970    USHLL<Q> <Vd>.<Ta>, <Vn>.<Tb>, #0.  */
971 static void
convert_xtl_to_shll(aarch64_inst * inst)972 convert_xtl_to_shll (aarch64_inst *inst)
973 {
974   inst->operands[2].qualifier = inst->operands[1].qualifier;
975   inst->operands[2].imm.value = 0;
976 }
977 
978 /* Convert
979      LSR <Xd>, <Xn>, #<shift>
980    to
981      UBFM <Xd>, <Xn>, #<shift>, #63.  */
982 static void
convert_sr_to_bfm(aarch64_inst * inst)983 convert_sr_to_bfm (aarch64_inst *inst)
984 {
985   inst->operands[3].imm.value =
986     inst->operands[2].qualifier == AARCH64_OPND_QLF_imm_0_31 ? 31 : 63;
987 }
988 
989 /* Convert MOV to ORR.  */
990 static void
convert_mov_to_orr(aarch64_inst * inst)991 convert_mov_to_orr (aarch64_inst *inst)
992 {
993   /* MOV <Vd>.<T>, <Vn>.<T>
994      is equivalent to:
995      ORR <Vd>.<T>, <Vn>.<T>, <Vn>.<T>.  */
996   copy_operand_info (inst, 2, 1);
997 }
998 
999 /* When <imms> >= <immr>, the instruction written:
1000      SBFX <Xd>, <Xn>, #<lsb>, #<width>
1001    is equivalent to:
1002      SBFM <Xd>, <Xn>, #<lsb>, #(<lsb>+<width>-1).  */
1003 
1004 static void
convert_bfx_to_bfm(aarch64_inst * inst)1005 convert_bfx_to_bfm (aarch64_inst *inst)
1006 {
1007   int64_t lsb, width;
1008 
1009   /* Convert the operand.  */
1010   lsb = inst->operands[2].imm.value;
1011   width = inst->operands[3].imm.value;
1012   inst->operands[2].imm.value = lsb;
1013   inst->operands[3].imm.value = lsb + width - 1;
1014 }
1015 
1016 /* When <imms> < <immr>, the instruction written:
1017      SBFIZ <Xd>, <Xn>, #<lsb>, #<width>
1018    is equivalent to:
1019      SBFM <Xd>, <Xn>, #((64-<lsb>)&0x3f), #(<width>-1).  */
1020 
1021 static void
convert_bfi_to_bfm(aarch64_inst * inst)1022 convert_bfi_to_bfm (aarch64_inst *inst)
1023 {
1024   int64_t lsb, width;
1025 
1026   /* Convert the operand.  */
1027   lsb = inst->operands[2].imm.value;
1028   width = inst->operands[3].imm.value;
1029   if (inst->operands[2].qualifier == AARCH64_OPND_QLF_imm_0_31)
1030     {
1031       inst->operands[2].imm.value = (32 - lsb) & 0x1f;
1032       inst->operands[3].imm.value = width - 1;
1033     }
1034   else
1035     {
1036       inst->operands[2].imm.value = (64 - lsb) & 0x3f;
1037       inst->operands[3].imm.value = width - 1;
1038     }
1039 }
1040 
1041 /* The instruction written:
1042      LSL <Xd>, <Xn>, #<shift>
1043    is equivalent to:
1044      UBFM <Xd>, <Xn>, #((64-<shift>)&0x3f), #(63-<shift>).  */
1045 
1046 static void
convert_lsl_to_ubfm(aarch64_inst * inst)1047 convert_lsl_to_ubfm (aarch64_inst *inst)
1048 {
1049   int64_t shift = inst->operands[2].imm.value;
1050 
1051   if (inst->operands[2].qualifier == AARCH64_OPND_QLF_imm_0_31)
1052     {
1053       inst->operands[2].imm.value = (32 - shift) & 0x1f;
1054       inst->operands[3].imm.value = 31 - shift;
1055     }
1056   else
1057     {
1058       inst->operands[2].imm.value = (64 - shift) & 0x3f;
1059       inst->operands[3].imm.value = 63 - shift;
1060     }
1061 }
1062 
1063 /* CINC <Wd>, <Wn>, <cond>
1064      is equivalent to:
1065    CSINC <Wd>, <Wn>, <Wn>, invert(<cond>).  */
1066 
1067 static void
convert_to_csel(aarch64_inst * inst)1068 convert_to_csel (aarch64_inst *inst)
1069 {
1070   copy_operand_info (inst, 3, 2);
1071   copy_operand_info (inst, 2, 1);
1072   inst->operands[3].cond = get_inverted_cond (inst->operands[3].cond);
1073 }
1074 
1075 /* CSET <Wd>, <cond>
1076      is equivalent to:
1077    CSINC <Wd>, WZR, WZR, invert(<cond>).  */
1078 
1079 static void
convert_cset_to_csinc(aarch64_inst * inst)1080 convert_cset_to_csinc (aarch64_inst *inst)
1081 {
1082   copy_operand_info (inst, 3, 1);
1083   copy_operand_info (inst, 2, 0);
1084   copy_operand_info (inst, 1, 0);
1085   inst->operands[1].reg.regno = 0x1f;
1086   inst->operands[2].reg.regno = 0x1f;
1087   inst->operands[3].cond = get_inverted_cond (inst->operands[3].cond);
1088 }
1089 
1090 /* MOV <Wd>, #<imm>
1091    is equivalent to:
1092    MOVZ <Wd>, #<imm16>, LSL #<shift>.  */
1093 
1094 static void
convert_mov_to_movewide(aarch64_inst * inst)1095 convert_mov_to_movewide (aarch64_inst *inst)
1096 {
1097   int is32;
1098   uint32_t shift_amount;
1099   uint64_t value;
1100 
1101   switch (inst->opcode->op)
1102     {
1103     case OP_MOV_IMM_WIDE:
1104       value = inst->operands[1].imm.value;
1105       break;
1106     case OP_MOV_IMM_WIDEN:
1107       value = ~inst->operands[1].imm.value;
1108       break;
1109     default:
1110       assert (0);
1111     }
1112   inst->operands[1].type = AARCH64_OPND_HALF;
1113   is32 = inst->operands[0].qualifier == AARCH64_OPND_QLF_W;
1114   if (! aarch64_wide_constant_p (value, is32, &shift_amount))
1115     /* The constraint check should have guaranteed this wouldn't happen.  */
1116     assert (0);
1117   value >>= shift_amount;
1118   value &= 0xffff;
1119   inst->operands[1].imm.value = value;
1120   inst->operands[1].shifter.kind = AARCH64_MOD_LSL;
1121   inst->operands[1].shifter.amount = shift_amount;
1122 }
1123 
1124 /* MOV <Wd>, #<imm>
1125      is equivalent to:
1126    ORR <Wd>, WZR, #<imm>.  */
1127 
1128 static void
convert_mov_to_movebitmask(aarch64_inst * inst)1129 convert_mov_to_movebitmask (aarch64_inst *inst)
1130 {
1131   copy_operand_info (inst, 2, 1);
1132   inst->operands[1].reg.regno = 0x1f;
1133   inst->operands[1].skip = 0;
1134 }
1135 
1136 /* Some alias opcodes are assembled by being converted to their real-form.  */
1137 
1138 static void
convert_to_real(aarch64_inst * inst,const aarch64_opcode * real)1139 convert_to_real (aarch64_inst *inst, const aarch64_opcode *real)
1140 {
1141   const aarch64_opcode *alias = inst->opcode;
1142 
1143   if ((alias->flags & F_CONV) == 0)
1144     goto convert_to_real_return;
1145 
1146   switch (alias->op)
1147     {
1148     case OP_ASR_IMM:
1149     case OP_LSR_IMM:
1150       convert_sr_to_bfm (inst);
1151       break;
1152     case OP_LSL_IMM:
1153       convert_lsl_to_ubfm (inst);
1154       break;
1155     case OP_CINC:
1156     case OP_CINV:
1157     case OP_CNEG:
1158       convert_to_csel (inst);
1159       break;
1160     case OP_CSET:
1161     case OP_CSETM:
1162       convert_cset_to_csinc (inst);
1163       break;
1164     case OP_UBFX:
1165     case OP_BFXIL:
1166     case OP_SBFX:
1167       convert_bfx_to_bfm (inst);
1168       break;
1169     case OP_SBFIZ:
1170     case OP_BFI:
1171     case OP_UBFIZ:
1172       convert_bfi_to_bfm (inst);
1173       break;
1174     case OP_MOV_V:
1175       convert_mov_to_orr (inst);
1176       break;
1177     case OP_MOV_IMM_WIDE:
1178     case OP_MOV_IMM_WIDEN:
1179       convert_mov_to_movewide (inst);
1180       break;
1181     case OP_MOV_IMM_LOG:
1182       convert_mov_to_movebitmask (inst);
1183       break;
1184     case OP_ROR_IMM:
1185       convert_ror_to_extr (inst);
1186       break;
1187     case OP_SXTL:
1188     case OP_SXTL2:
1189     case OP_UXTL:
1190     case OP_UXTL2:
1191       convert_xtl_to_shll (inst);
1192       break;
1193     default:
1194       break;
1195     }
1196 
1197 convert_to_real_return:
1198   aarch64_replace_opcode (inst, real);
1199 }
1200 
1201 /* Encode *INST_ORI of the opcode code OPCODE.
1202    Return the encoded result in *CODE and if QLF_SEQ is not NULL, return the
1203    matched operand qualifier sequence in *QLF_SEQ.  */
1204 
1205 int
aarch64_opcode_encode(const aarch64_opcode * opcode,const aarch64_inst * inst_ori,aarch64_insn * code,aarch64_opnd_qualifier_t * qlf_seq,aarch64_operand_error * mismatch_detail)1206 aarch64_opcode_encode (const aarch64_opcode *opcode,
1207 		       const aarch64_inst *inst_ori, aarch64_insn *code,
1208 		       aarch64_opnd_qualifier_t *qlf_seq,
1209 		       aarch64_operand_error *mismatch_detail)
1210 {
1211   int i;
1212   const aarch64_opcode *aliased;
1213   aarch64_inst copy, *inst;
1214 
1215   DEBUG_TRACE ("enter with %s", opcode->name);
1216 
1217   /* Create a copy of *INST_ORI, so that we can do any change we want.  */
1218   copy = *inst_ori;
1219   inst = &copy;
1220 
1221   assert (inst->opcode == NULL || inst->opcode == opcode);
1222   if (inst->opcode == NULL)
1223     inst->opcode = opcode;
1224 
1225   /* Constrain the operands.
1226      After passing this, the encoding is guaranteed to succeed.  */
1227   if (aarch64_match_operands_constraint (inst, mismatch_detail) == 0)
1228     {
1229       DEBUG_TRACE ("FAIL since operand constraint not met");
1230       return 0;
1231     }
1232 
1233   /* Get the base value.
1234      Note: this has to be before the aliasing handling below in order to
1235      get the base value from the alias opcode before we move on to the
1236      aliased opcode for encoding.  */
1237   inst->value = opcode->opcode;
1238 
1239   /* No need to do anything else if the opcode does not have any operand.  */
1240   if (aarch64_num_of_operands (opcode) == 0)
1241     goto encoding_exit;
1242 
1243   /* Assign operand indexes and check types.  Also put the matched
1244      operand qualifiers in *QLF_SEQ to return.  */
1245   for (i = 0; i < AARCH64_MAX_OPND_NUM; ++i)
1246     {
1247       assert (opcode->operands[i] == inst->operands[i].type);
1248       inst->operands[i].idx = i;
1249       if (qlf_seq != NULL)
1250 	*qlf_seq = inst->operands[i].qualifier;
1251     }
1252 
1253   aliased = aarch64_find_real_opcode (opcode);
1254   /* If the opcode is an alias and it does not ask for direct encoding by
1255      itself, the instruction will be transformed to the form of real opcode
1256      and the encoding will be carried out using the rules for the aliased
1257      opcode.  */
1258   if (aliased != NULL && (opcode->flags & F_CONV))
1259     {
1260       DEBUG_TRACE ("real opcode '%s' has been found for the alias  %s",
1261 		   aliased->name, opcode->name);
1262       /* Convert the operands to the form of the real opcode.  */
1263       convert_to_real (inst, aliased);
1264       opcode = aliased;
1265     }
1266 
1267   aarch64_opnd_info *info = inst->operands;
1268 
1269   /* Call the inserter of each operand.  */
1270   for (i = 0; i < AARCH64_MAX_OPND_NUM; ++i, ++info)
1271     {
1272       const aarch64_operand *opnd;
1273       enum aarch64_opnd type = opcode->operands[i];
1274       if (type == AARCH64_OPND_NIL)
1275 	break;
1276       if (info->skip)
1277 	{
1278 	  DEBUG_TRACE ("skip the incomplete operand %d", i);
1279 	  continue;
1280 	}
1281       opnd = &aarch64_operands[type];
1282       if (operand_has_inserter (opnd))
1283 	aarch64_insert_operand (opnd, info, &inst->value, inst);
1284     }
1285 
1286   /* Call opcode encoders indicated by flags.  */
1287   if (opcode_has_special_coder (opcode))
1288     do_special_encoding (inst);
1289 
1290 encoding_exit:
1291   DEBUG_TRACE ("exit with %s", opcode->name);
1292 
1293   *code = inst->value;
1294 
1295   return 1;
1296 }
1297