From 38bf472a15210c222bb3885820e763c47760a704 Mon Sep 17 00:00:00 2001 From: "Maciej W. Rozycki" Date: Wed, 28 Jun 2017 02:07:36 +0100 Subject: MIPS: Add Imagination interAptiv MR2 MIPS32r3 processor support Add support for the Imagination interAptiv MR2 MIPS32r3 processor with the MIPS16e2 ASE as per documentation, including in particular: 1. Support for implementation-specific interAptiv MR2 COPYW and UCOPYW MIPS16e2 instructions[1], for assembly and disassembly, 2. Support for implementation-specific interAptiv MR2 SAVE and RESTORE regular MIPS instructions[2], for assembly and disassembly, 3. ELF binary file annotation for the interAptiv MR2 MIPS architecture extension. 4. Support for interAptiv MR2 architecture selection for assembly, in the form of the `-march=interaptiv-mr2' command-line option and its corresponding `arch=interaptiv-mr2' setting for the `.set' and `.module' pseudo-ops. 5. Support for interAptiv MR2 architecture selection for disassembly, in the form of the `mips:interaptiv-mr2' target architecture, for use e.g. with the `-m' command-line option for `objdump'. Parts of this change by Matthew Fortune and Andrew Bennett. References: [1] "MIPS32 interAptiv Multiprocessing System Software User's Manual", Imagination Technologies Ltd., Document Number: MD00904, Revision 02.01, June 15, 2016, Section 24.3 "MIPS16e2 Implementation Specific Instructions", pp. 878-883 [2] same, Chapter 25 "Implementation-specific Instructions", pp. 911-917 include/ * elf/mips.h (E_MIPS_MACH_IAMR2): New macro. (AFL_EXT_INTERAPTIV_MR2): Likewise. * opcode/mips.h: Document new operand codes defined. (INSN_INTERAPTIV_MR2): New macro. (INSN_CHIP_MASK): Adjust accordingly. (CPU_INTERAPTIV_MR2): New macro. (cpu_is_member) : New case. (MIPS16_ALL_ARGS): Rename to... (MIPS_SVRS_ALL_ARGS): ... this. (MIPS16_ALL_STATICS): Rename to... (MIPS_SVRS_ALL_STATICS): ... this. bfd/ * archures.c (bfd_mach_mips_interaptiv_mr2): New macro. * cpu-mips.c (I_interaptiv_mr2): New enum value. (arch_info_struct): Add "mips:interaptiv-mr2" entry. * elfxx-mips.c (_bfd_elf_mips_mach) : New case. (mips_set_isa_flags) : Likewise. (bfd_mips_isa_ext) : Likewise. (print_mips_isa_ext) : Likewise. (mips_mach_extensions): Add `bfd_mach_mipsisa32r3' and `bfd_mach_mips_interaptiv_mr2' entries. * bfd-in2.h: Regenerate. opcodes/ * mips-formats.h (INT_BIAS): New macro. (INT_ADJ): Redefine in INT_BIAS terms. * mips-dis.c (mips_arch_choices): Add "interaptiv-mr2" entry. (mips_print_save_restore): New function. (print_insn_arg) : Update comment. (validate_insn_args) : Remove `abort' call. (print_insn_args): Handle OP_SAVE_RESTORE_LIST. (print_mips16_insn_arg): Call `mips_print_save_restore' for OP_SAVE_RESTORE_LIST handling, factored out from here. * mips-opc.c (decode_mips_operand) <'-'> <'m'>: New case. (RD_31, RD_SP, WR_SP, MOD_SP, IAMR2): New macros. (mips_builtin_opcodes): Add "restore" and "save" entries. * mips16-opc.c (decode_mips16_operand) <'n', 'o'>: New cases. (IAMR2): New macro. (mips16_opcodes): Add "copyw" and "ucopyw" entries. binutils/ * readelf.c (get_machine_flags) : New case. (print_mips_isa_ext) : Likewise. * NEWS: Mention Imagination interAptiv MR2 processor support. gas/ * config/tc-mips.c (validate_mips_insn): Handle OP_SAVE_RESTORE_LIST specially. (mips_encode_save_restore, mips16_encode_save_restore): New functions. (match_save_restore_list_operand): Factor out SAVE/RESTORE operand insertion into the instruction word or halfword to these new functions. (mips_cpu_info_table): Add "interaptiv-mr2" entry. * doc/c-mips.texi (MIPS Options): Add `interaptiv-mr2' to the `-march=' argument list. --- gas/ChangeLog | 15 +++++++++++ gas/config/tc-mips.c | 75 +++++++++++++++++++++++++++++++++++++++++----------- gas/doc/c-mips.texi | 1 + 3 files changed, 76 insertions(+), 15 deletions(-) (limited to 'gas') diff --git a/gas/ChangeLog b/gas/ChangeLog index 6a3437b..69e6547 100644 --- a/gas/ChangeLog +++ b/gas/ChangeLog @@ -1,3 +1,18 @@ +2017-06-28 Maciej W. Rozycki + Matthew Fortune + + * config/tc-mips.c (validate_mips_insn): Handle + OP_SAVE_RESTORE_LIST specially. + (mips_encode_save_restore, mips16_encode_save_restore): New + functions. + (match_save_restore_list_operand): Factor out SAVE/RESTORE + operand insertion into the instruction word or halfword to these + new functions. + (mips_cpu_info_table): Add "interaptiv-mr2" entry. + + * doc/c-mips.texi (MIPS Options): Add `interaptiv-mr2' to the + `-march=' argument list. + 2017-06-27 Maciej W. Rozycki * testsuite/gas/mips/mips16e-save.d: Rename to... diff --git a/gas/config/tc-mips.c b/gas/config/tc-mips.c index ecd3c8e..d7a1ff3 100644 --- a/gas/config/tc-mips.c +++ b/gas/config/tc-mips.c @@ -3463,6 +3463,10 @@ validate_mips_insn (const struct mips_opcode *opcode, used_bits &= ~(1 << (operand->lsb + 5)); if (operand->type == OP_ENTRY_EXIT_LIST) used_bits &= ~(mask & 0x700); + /* interAptiv MR2 SAVE/RESTORE instructions have a discontiguous + operand field that cannot be fully described with LSB/SIZE. */ + if (operand->type == OP_SAVE_RESTORE_LIST && operand->lsb == 6) + used_bits &= ~0x6000; } /* Skip prefix characters. */ if (decode_operand && (*s == '+' || *s == 'm' || *s == '-')) @@ -5555,6 +5559,39 @@ match_entry_exit_operand (struct mips_arg_info *arg, return TRUE; } +/* Encode regular MIPS SAVE/RESTORE instruction operands according to + the argument register mask AMASK, the number of static registers + saved NSREG, the $ra, $s0 and $s1 register specifiers RA, S0 and S1 + respectively, and the frame size FRAME_SIZE. */ + +static unsigned int +mips_encode_save_restore (unsigned int amask, unsigned int nsreg, + unsigned int ra, unsigned int s0, unsigned int s1, + unsigned int frame_size) +{ + return ((nsreg << 23) | ((frame_size & 0xf0) << 15) | (amask << 15) + | (ra << 12) | (s0 << 11) | (s1 << 10) | ((frame_size & 0xf) << 6)); +} + +/* Encode MIPS16 SAVE/RESTORE instruction operands according to the + argument register mask AMASK, the number of static registers saved + NSREG, the $ra, $s0 and $s1 register specifiers RA, S0 and S1 + respectively, and the frame size FRAME_SIZE. */ + +static unsigned int +mips16_encode_save_restore (unsigned int amask, unsigned int nsreg, + unsigned int ra, unsigned int s0, unsigned int s1, + unsigned int frame_size) +{ + unsigned int args; + + args = (ra << 6) | (s0 << 5) | (s1 << 4) | (frame_size & 0xf); + if (nsreg || amask || frame_size == 0 || frame_size > 16) + args |= (MIPS16_EXTEND | (nsreg << 24) | (amask << 16) + | ((frame_size & 0xf0) << 16)); + return args; +} + /* OP_SAVE_RESTORE_LIST matcher. */ static bfd_boolean @@ -5562,6 +5599,7 @@ match_save_restore_list_operand (struct mips_arg_info *arg) { unsigned int opcode, args, statics, sregs; unsigned int num_frame_sizes, num_args, num_statics, num_sregs; + unsigned int arg_mask, ra, s0, s1; offsetT frame_size; opcode = arg->insn->insn_opcode; @@ -5570,6 +5608,9 @@ match_save_restore_list_operand (struct mips_arg_info *arg) args = 0; statics = 0; sregs = 0; + ra = 0; + s0 = 0; + s1 = 0; do { unsigned int regno1, regno2; @@ -5605,7 +5646,7 @@ match_save_restore_list_operand (struct mips_arg_info *arg) sregs |= 1 << 8; else if (regno1 == 31) /* Add $ra to insn. */ - opcode |= 0x40; + ra = 1; else return FALSE; regno1 += 1; @@ -5621,10 +5662,10 @@ match_save_restore_list_operand (struct mips_arg_info *arg) return FALSE; else if (args == 0xf) /* All $a0-$a3 are args. */ - opcode |= MIPS16_ALL_ARGS << 16; + arg_mask = MIPS_SVRS_ALL_ARGS; else if (statics == 0xf) /* All $a0-$a3 are statics. */ - opcode |= MIPS16_ALL_STATICS << 16; + arg_mask = MIPS_SVRS_ALL_STATICS; else { /* Count arg registers. */ @@ -5648,14 +5689,14 @@ match_save_restore_list_operand (struct mips_arg_info *arg) return FALSE; /* Encode args/statics. */ - opcode |= ((num_args << 2) | num_statics) << 16; + arg_mask = (num_args << 2) | num_statics; } /* Encode $s0/$s1. */ if (sregs & (1 << 0)) /* $s0 */ - opcode |= 0x20; + s0 = 1; if (sregs & (1 << 1)) /* $s1 */ - opcode |= 0x10; + s1 = 1; sregs >>= 2; /* Encode $s2-$s8. */ @@ -5667,7 +5708,6 @@ match_save_restore_list_operand (struct mips_arg_info *arg) } if (sregs != 0) return FALSE; - opcode |= num_sregs << 24; /* Encode frame size. */ if (num_frame_sizes == 0) @@ -5685,16 +5725,18 @@ match_save_restore_list_operand (struct mips_arg_info *arg) set_insn_error (arg->argnum, _("invalid frame size")); return FALSE; } - if (frame_size != 128 || (opcode >> 16) != 0) - { - frame_size /= 8; - opcode |= (((frame_size & 0xf0) << 16) - | (frame_size & 0x0f)); - } + frame_size /= 8; /* Finally build the instruction. */ - if ((opcode >> 16) != 0 || frame_size == 0) - opcode |= MIPS16_EXTEND; + if (mips_opts.mips16) + opcode |= mips16_encode_save_restore (arg_mask, num_sregs, ra, s0, s1, + frame_size); + else if (!mips_opts.micromips) + opcode |= mips_encode_save_restore (arg_mask, num_sregs, ra, s0, s1, + frame_size); + else + abort (); + arg->insn->insn_opcode = opcode; return TRUE; } @@ -19665,6 +19707,9 @@ static const struct mips_cpu_info mips_cpu_info_table[] = { "1004kf1_1", 0, ASE_DSP | ASE_MT, ISA_MIPS32R2, CPU_MIPS32R2 }, /* interaptiv is the new name for 1004kf */ { "interaptiv", 0, ASE_DSP | ASE_MT, ISA_MIPS32R2, CPU_MIPS32R2 }, + { "interaptiv-mr2", 0, + ASE_DSP | ASE_EVA | ASE_MT | ASE_MIPS16E2 | ASE_MIPS16E2_MT, + ISA_MIPS32R3, CPU_INTERAPTIV_MR2 }, /* M5100 family */ { "m5100", 0, ASE_MCU, ISA_MIPS32R5, CPU_MIPS32R5 }, { "m5101", 0, ASE_MCU, ISA_MIPS32R5, CPU_MIPS32R5 }, diff --git a/gas/doc/c-mips.texi b/gas/doc/c-mips.texi index c2d9b7a..a430b0d 100644 --- a/gas/doc/c-mips.texi +++ b/gas/doc/c-mips.texi @@ -383,6 +383,7 @@ m14kec, 1004kf, 1004kf1_1, interaptiv, +interaptiv-mr2, m5100, m5101, p5600, -- cgit v1.1