GNU Binutils with patches for OS216
修訂 | 136982a0a15e6c33d915cd9d48e6928bf8dab45b (tree) |
---|---|
時間 | 2016-08-23 17:41:03 |
作者 | Richard Sandiford <richard.sandiford@arm....> |
Commiter | Richard Sandiford |
[AArch64][SVE 19/32] Refactor address-printing code
SVE adds addresses in which the base or offset are vector registers.
The addresses otherwise have the same kind of form as normal AArch64
addresses, including things like SXTW with or without a shift, UXTW
with or without a shift, and LSL.
This patch therefore refactors the address-printing code so that it
can cope with both scalar and vector registers.
opcodes/
* aarch64-opc.c (get_offset_int_reg_name): New function.
(print_immediate_offset_address): Likewise.
(print_register_offset_address): Take the base and offset
registers as parameters.
(aarch64_print_operand): Update caller accordingly. Use
print_immediate_offset_address.
Change-Id: Ib975218f68fd866a50184d82ed3ea9bc792bbf02
@@ -2189,6 +2189,27 @@ get_64bit_int_reg_name (int regno, int sp_reg_p) | ||
2189 | 2189 | return int_reg[has_zr][1][regno]; |
2190 | 2190 | } |
2191 | 2191 | |
2192 | +/* Get the name of the integer offset register in OPND, using the shift type | |
2193 | + to decide whether it's a word or doubleword. */ | |
2194 | + | |
2195 | +static inline const char * | |
2196 | +get_offset_int_reg_name (const aarch64_opnd_info *opnd) | |
2197 | +{ | |
2198 | + switch (opnd->shifter.kind) | |
2199 | + { | |
2200 | + case AARCH64_MOD_UXTW: | |
2201 | + case AARCH64_MOD_SXTW: | |
2202 | + return get_int_reg_name (opnd->addr.offset.regno, AARCH64_OPND_QLF_W, 0); | |
2203 | + | |
2204 | + case AARCH64_MOD_LSL: | |
2205 | + case AARCH64_MOD_SXTX: | |
2206 | + return get_int_reg_name (opnd->addr.offset.regno, AARCH64_OPND_QLF_X, 0); | |
2207 | + | |
2208 | + default: | |
2209 | + abort (); | |
2210 | + } | |
2211 | +} | |
2212 | + | |
2192 | 2213 | /* Types for expanding an encoded 8-bit value to a floating-point value. */ |
2193 | 2214 | |
2194 | 2215 | typedef union |
@@ -2311,28 +2332,43 @@ print_register_list (char *buf, size_t size, const aarch64_opnd_info *opnd, | ||
2311 | 2332 | } |
2312 | 2333 | } |
2313 | 2334 | |
2335 | +/* Print the register+immediate address in OPND to BUF, which has SIZE | |
2336 | + characters. BASE is the name of the base register. */ | |
2337 | + | |
2338 | +static void | |
2339 | +print_immediate_offset_address (char *buf, size_t size, | |
2340 | + const aarch64_opnd_info *opnd, | |
2341 | + const char *base) | |
2342 | +{ | |
2343 | + if (opnd->addr.writeback) | |
2344 | + { | |
2345 | + if (opnd->addr.preind) | |
2346 | + snprintf (buf, size, "[%s,#%d]!", base, opnd->addr.offset.imm); | |
2347 | + else | |
2348 | + snprintf (buf, size, "[%s],#%d", base, opnd->addr.offset.imm); | |
2349 | + } | |
2350 | + else | |
2351 | + { | |
2352 | + if (opnd->addr.offset.imm) | |
2353 | + snprintf (buf, size, "[%s,#%d]", base, opnd->addr.offset.imm); | |
2354 | + else | |
2355 | + snprintf (buf, size, "[%s]", base); | |
2356 | + } | |
2357 | +} | |
2358 | + | |
2314 | 2359 | /* Produce the string representation of the register offset address operand |
2315 | - *OPND in the buffer pointed by BUF of size SIZE. */ | |
2360 | + *OPND in the buffer pointed by BUF of size SIZE. BASE and OFFSET are | |
2361 | + the names of the base and offset registers. */ | |
2316 | 2362 | static void |
2317 | 2363 | print_register_offset_address (char *buf, size_t size, |
2318 | - const aarch64_opnd_info *opnd) | |
2364 | + const aarch64_opnd_info *opnd, | |
2365 | + const char *base, const char *offset) | |
2319 | 2366 | { |
2320 | 2367 | char tb[16]; /* Temporary buffer. */ |
2321 | - bfd_boolean lsl_p = FALSE; /* Is LSL shift operator? */ | |
2322 | - bfd_boolean wm_p = FALSE; /* Should Rm be Wm? */ | |
2323 | 2368 | bfd_boolean print_extend_p = TRUE; |
2324 | 2369 | bfd_boolean print_amount_p = TRUE; |
2325 | 2370 | const char *shift_name = aarch64_operand_modifiers[opnd->shifter.kind].name; |
2326 | 2371 | |
2327 | - switch (opnd->shifter.kind) | |
2328 | - { | |
2329 | - case AARCH64_MOD_UXTW: wm_p = TRUE; break; | |
2330 | - case AARCH64_MOD_LSL : lsl_p = TRUE; break; | |
2331 | - case AARCH64_MOD_SXTW: wm_p = TRUE; break; | |
2332 | - case AARCH64_MOD_SXTX: break; | |
2333 | - default: assert (0); | |
2334 | - } | |
2335 | - | |
2336 | 2372 | if (!opnd->shifter.amount && (opnd->qualifier != AARCH64_OPND_QLF_S_B |
2337 | 2373 | || !opnd->shifter.amount_present)) |
2338 | 2374 | { |
@@ -2341,7 +2377,7 @@ print_register_offset_address (char *buf, size_t size, | ||
2341 | 2377 | print_amount_p = FALSE; |
2342 | 2378 | /* Likewise, no need to print the shift operator LSL in such a |
2343 | 2379 | situation. */ |
2344 | - if (lsl_p) | |
2380 | + if (opnd->shifter.kind == AARCH64_MOD_LSL) | |
2345 | 2381 | print_extend_p = FALSE; |
2346 | 2382 | } |
2347 | 2383 |
@@ -2356,12 +2392,7 @@ print_register_offset_address (char *buf, size_t size, | ||
2356 | 2392 | else |
2357 | 2393 | tb[0] = '\0'; |
2358 | 2394 | |
2359 | - snprintf (buf, size, "[%s,%s%s]", | |
2360 | - get_64bit_int_reg_name (opnd->addr.base_regno, 1), | |
2361 | - get_int_reg_name (opnd->addr.offset.regno, | |
2362 | - wm_p ? AARCH64_OPND_QLF_W : AARCH64_OPND_QLF_X, | |
2363 | - 0 /* sp_reg_p */), | |
2364 | - tb); | |
2395 | + snprintf (buf, size, "[%s,%s%s]", base, offset, tb); | |
2365 | 2396 | } |
2366 | 2397 | |
2367 | 2398 | /* Generate the string representation of the operand OPNDS[IDX] for OPCODE |
@@ -2668,27 +2699,16 @@ aarch64_print_operand (char *buf, size_t size, bfd_vma pc, | ||
2668 | 2699 | break; |
2669 | 2700 | |
2670 | 2701 | case AARCH64_OPND_ADDR_REGOFF: |
2671 | - print_register_offset_address (buf, size, opnd); | |
2702 | + print_register_offset_address | |
2703 | + (buf, size, opnd, get_64bit_int_reg_name (opnd->addr.base_regno, 1), | |
2704 | + get_offset_int_reg_name (opnd)); | |
2672 | 2705 | break; |
2673 | 2706 | |
2674 | 2707 | case AARCH64_OPND_ADDR_SIMM7: |
2675 | 2708 | case AARCH64_OPND_ADDR_SIMM9: |
2676 | 2709 | case AARCH64_OPND_ADDR_SIMM9_2: |
2677 | - name = get_64bit_int_reg_name (opnd->addr.base_regno, 1); | |
2678 | - if (opnd->addr.writeback) | |
2679 | - { | |
2680 | - if (opnd->addr.preind) | |
2681 | - snprintf (buf, size, "[%s,#%d]!", name, opnd->addr.offset.imm); | |
2682 | - else | |
2683 | - snprintf (buf, size, "[%s],#%d", name, opnd->addr.offset.imm); | |
2684 | - } | |
2685 | - else | |
2686 | - { | |
2687 | - if (opnd->addr.offset.imm) | |
2688 | - snprintf (buf, size, "[%s,#%d]", name, opnd->addr.offset.imm); | |
2689 | - else | |
2690 | - snprintf (buf, size, "[%s]", name); | |
2691 | - } | |
2710 | + print_immediate_offset_address | |
2711 | + (buf, size, opnd, get_64bit_int_reg_name (opnd->addr.base_regno, 1)); | |
2692 | 2712 | break; |
2693 | 2713 | |
2694 | 2714 | case AARCH64_OPND_ADDR_UIMM12: |