// Copyright 2016, VIXL authors
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
//   * Redistributions of source code must retain the above copyright notice,
//     this list of conditions and the following disclaimer.
//   * Redistributions in binary form must reproduce the above copyright notice,
//     this list of conditions and the following disclaimer in the documentation
//     and/or other materials provided with the distribution.
//   * Neither the name of ARM Limited nor the names of its contributors may be
//     used to endorse or promote products derived from this software without
//     specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS CONTRIBUTORS "AS IS" AND
// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

#include <sstream>
#include <string>
#include <list>

#include "test-runner.h"
#include "test-utils.h"

#include "aarch32/macro-assembler-aarch32.h"
#include "aarch32/disasm-aarch32.h"

#ifdef VIXL_NEGATIVE_TESTING
#include <stdexcept>
#endif

namespace vixl {
namespace aarch32 {

#define __ masm.
#define TEST(name) TEST_(AARCH32_DISASM_##name)

#ifdef VIXL_INCLUDE_TARGET_T32
#define TEST_T32(name) TEST_(AARCH32_DISASM_##name)
#else
#define TEST_T32(name) void Test##name()
#endif

#ifdef VIXL_INCLUDE_TARGET_A32
#define TEST_A32(name) TEST_(AARCH32_DISASM_##name)
#else
#define TEST_A32(name) void Test##name()
#endif

#define BUF_SIZE (4096)

#define SETUP() MacroAssembler masm(BUF_SIZE);

#define CLEANUP()

#ifdef VIXL_NEGATIVE_TESTING
#define START_COMPARE() \
  {                     \
    try {               \
      int32_t start = masm.GetCursorOffset();

#define END_COMPARE_CHECK_SIZE(EXP, SIZE)                       \
  int32_t end = masm.GetCursorOffset();                         \
  masm.FinalizeCode();                                          \
  std::ostringstream ss;                                        \
  TestDisassembler disassembler(ss, 0);                         \
  if (masm.IsUsingT32()) {                                      \
    disassembler.DisassembleT32(*masm.GetBuffer(), start, end); \
  } else {                                                      \
    disassembler.DisassembleA32(*masm.GetBuffer(), start, end); \
  }                                                             \
  masm.GetBuffer()->Reset();                                    \
  if (Test::disassemble()) {                                    \
    printf("----\n");                                           \
    printf("%s", ss.str().c_str());                             \
  }                                                             \
  if (std::string(EXP) != ss.str()) {                           \
    printf("\n%s:%d:%s\nFound:\n%sExpected:\n%s",               \
           __FILE__,                                            \
           __LINE__,                                            \
           masm.IsUsingT32() ? "T32" : "A32",                   \
           ss.str().c_str(),                                    \
           EXP);                                                \
    abort();                                                    \
  }                                                             \
  if ((SIZE) != -1 && ((end - start) != (SIZE))) {              \
    printf("\nExpected %d bits, found %d bits\n",               \
           8 * (SIZE),                                          \
           8 * (end - start));                                  \
    abort();                                                    \
  }                                                             \
  }                                                             \
  catch (std::runtime_error e) {                                \
    const char* msg = e.what();                                 \
    printf("\n%s:%d:%s\nFound:\n%sExpected:\n%s",               \
           __FILE__,                                            \
           __LINE__,                                            \
           masm.IsUsingT32() ? "T32" : "A32",                   \
           msg,                                                 \
           EXP);                                                \
    abort();                                                    \
  }                                                             \
  }
#else
#define START_COMPARE() \
  {                     \
    int32_t start = masm.GetCursorOffset();

#define END_COMPARE_CHECK_SIZE(EXP, SIZE)                       \
  int32_t end = masm.GetCursorOffset();                         \
  masm.FinalizeCode();                                          \
  std::ostringstream ss;                                        \
  TestDisassembler disassembler(ss, 0);                         \
  if (masm.IsUsingT32()) {                                      \
    disassembler.DisassembleT32(*masm.GetBuffer(), start, end); \
  } else {                                                      \
    disassembler.DisassembleA32(*masm.GetBuffer(), start, end); \
  }                                                             \
  masm.GetBuffer()->Reset();                                    \
  if (Test::disassemble()) {                                    \
    printf("----\n");                                           \
    printf("%s", ss.str().c_str());                             \
  }                                                             \
  if (std::string(EXP) != ss.str()) {                           \
    printf("\n%s:%d:%s\nFound:\n%sExpected:\n%s",               \
           __FILE__,                                            \
           __LINE__,                                            \
           masm.IsUsingT32() ? "T32" : "A32",                   \
           ss.str().c_str(),                                    \
           EXP);                                                \
    abort();                                                    \
  }                                                             \
  if ((SIZE) != -1 && ((end - start) != (SIZE))) {              \
    printf("\nExpected %d bits, found %d bits\n",               \
           8 * (SIZE),                                          \
           8 * (end - start));                                  \
    abort();                                                    \
  }                                                             \
  }
#endif

#define END_COMPARE(EXP) END_COMPARE_CHECK_SIZE(EXP, -1)

#ifdef VIXL_INCLUDE_TARGET_A32
#define COMPARE_A32(ASM, EXP) \
  masm.UseA32();              \
  START_COMPARE()             \
  masm.ASM;                   \
  END_COMPARE(EXP)
#else
#define COMPARE_A32(ASM, EXP)
#endif

#ifdef VIXL_INCLUDE_TARGET_T32
#define COMPARE_T32(ASM, EXP) \
  masm.UseT32();              \
  START_COMPARE()             \
  masm.ASM;                   \
  END_COMPARE(EXP)
#else
#define COMPARE_T32(ASM, EXP)
#endif

#ifdef VIXL_INCLUDE_TARGET_T32
#define COMPARE_T32_CHECK_SIZE(ASM, EXP, SIZE) \
  masm.UseT32();                               \
  START_COMPARE()                              \
  masm.ASM;                                    \
  END_COMPARE_CHECK_SIZE(EXP, SIZE)
#else
#define COMPARE_T32_CHECK_SIZE(ASM, EXP, SIZE)
#endif

#define COMPARE_BOTH(ASM, EXP) \
  COMPARE_A32(ASM, EXP)        \
  COMPARE_T32(ASM, EXP)

#ifdef VIXL_NEGATIVE_TESTING
#define NEGATIVE_TEST(ASM, EXP, TEMPORARILY_ACCEPTED)                          \
  {                                                                            \
    try {                                                                      \
      int32_t start = masm.GetCursorOffset();                                  \
      ASM int32_t end = masm.GetCursorOffset();                                \
      masm.FinalizeCode();                                                     \
      if (!TEMPORARILY_ACCEPTED) {                                             \
        std::ostringstream ss;                                                 \
        PrintDisassembler disassembler(ss, 0);                                 \
        if (masm.IsUsingT32()) {                                               \
          disassembler.DisassembleT32Buffer(masm.GetBuffer()                   \
                                                ->GetOffsetAddress<uint16_t*>( \
                                                    start),                    \
                                            end);                              \
        } else {                                                               \
          disassembler.DisassembleA32Buffer(masm.GetBuffer()                   \
                                                ->GetOffsetAddress<uint32_t*>( \
                                                    start),                    \
                                            end);                              \
        }                                                                      \
        printf("\n%s:%d:%s\nNo exception raised.\n",                           \
               __FILE__,                                                       \
               __LINE__,                                                       \
               masm.IsUsingT32() ? "T32" : "A32");                             \
        printf("Found:\n%sExpected:\n%s", ss.str().c_str(), EXP);              \
        abort();                                                               \
      }                                                                        \
    } catch (std::runtime_error e) {                                           \
      const char* msg = e.what();                                              \
      size_t exp_len = strlen(EXP);                                            \
      if (TEMPORARILY_ACCEPTED) {                                              \
        printf(                                                                \
            "\nNegative MacroAssembler test that was temporarily "             \
            "assembling a deprecated or unpredictable instruction is now "     \
            "correctly raising an exception. Please update the "               \
            "test to reflect this.\n");                                        \
        printf("at: %s:%d:%s\n",                                               \
               __FILE__,                                                       \
               __LINE__,                                                       \
               masm.IsUsingT32() ? "T32" : "A32");                             \
        abort();                                                               \
      } else if (std::strncmp(EXP, msg, exp_len) != 0) {                       \
        printf("\n%s:%d:%s\nFound:\n%sExpected:\n%s...",                       \
               __FILE__,                                                       \
               __LINE__,                                                       \
               masm.IsUsingT32() ? "T32" : "A32",                              \
               msg,                                                            \
               EXP);                                                           \
        abort();                                                               \
      }                                                                        \
    }                                                                          \
  }

#ifdef VIXL_INCLUDE_TARGET_A32
#define MUST_FAIL_TEST_A32(ASM, EXP)       \
  masm.UseA32();                           \
  NEGATIVE_TEST({ masm.ASM; }, EXP, false) \
  masm.GetBuffer()->Reset();
#else
#define MUST_FAIL_TEST_A32(ASM, EXP)
#endif

#ifdef VIXL_INCLUDE_TARGET_T32
#define MUST_FAIL_TEST_T32(ASM, EXP)       \
  masm.UseT32();                           \
  NEGATIVE_TEST({ masm.ASM; }, EXP, false) \
  masm.GetBuffer()->Reset();
#else
#define MUST_FAIL_TEST_T32(ASM, EXP)
#endif

#define MUST_FAIL_TEST_BOTH(ASM, EXP) \
  MUST_FAIL_TEST_A32(ASM, EXP)        \
  MUST_FAIL_TEST_T32(ASM, EXP)

#ifdef VIXL_INCLUDE_TARGET_A32
#define MUST_FAIL_TEST_A32_BLOCK(ASM, EXP) \
  masm.UseA32();                           \
  NEGATIVE_TEST(ASM, EXP, false)           \
  masm.GetBuffer()->Reset();
#else
#define MUST_FAIL_TEST_A32_BLOCK(ASM, EXP)
#endif

#ifdef VIXL_INCLUDE_TARGET_T32
#define MUST_FAIL_TEST_T32_BLOCK(ASM, EXP) \
  masm.UseT32();                           \
  NEGATIVE_TEST(ASM, EXP, false)           \
  masm.GetBuffer()->Reset();
#else
#define MUST_FAIL_TEST_T32_BLOCK(ASM, EXP)
#endif

#define MUST_FAIL_TEST_BOTH_BLOCK(ASM, EXP) \
  MUST_FAIL_TEST_A32_BLOCK(ASM, EXP)        \
  MUST_FAIL_TEST_T32_BLOCK(ASM, EXP)
#else
// Skip negative tests.
#define MUST_FAIL_TEST_A32(ASM, EXP)                         \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#define MUST_FAIL_TEST_T32(ASM, EXP)                         \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#define MUST_FAIL_TEST_BOTH(ASM, EXP)                        \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#define MUST_FAIL_TEST_A32_BLOCK(ASM, EXP)                   \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#define MUST_FAIL_TEST_T32_BLOCK(ASM, EXP)                   \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#define MUST_FAIL_TEST_BOTH_BLOCK(ASM, EXP)                  \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#endif

#ifdef VIXL_NEGATIVE_TESTING
#ifdef VIXL_INCLUDE_TARGET_A32
#define SHOULD_FAIL_TEST_A32(ASM)        \
  masm.UseA32();                         \
  NEGATIVE_TEST({ masm.ASM; }, "", true) \
  masm.GetBuffer()->Reset();
#else
#define SHOULD_FAIL_TEST_A32(ASM)
#endif

#ifdef VIXL_INCLUDE_TARGET_T32
#define SHOULD_FAIL_TEST_T32(ASM)        \
  masm.UseT32();                         \
  NEGATIVE_TEST({ masm.ASM; }, "", true) \
  masm.GetBuffer()->Reset();
#else
#define SHOULD_FAIL_TEST_T32(ASM)
#endif

#define SHOULD_FAIL_TEST_BOTH(ASM) \
  SHOULD_FAIL_TEST_A32(ASM)        \
  SHOULD_FAIL_TEST_T32(ASM)
#else
#define SHOULD_FAIL_TEST_A32(ASM)                            \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#define SHOULD_FAIL_TEST_T32(ASM)                            \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#define SHOULD_FAIL_TEST_BOTH(ASM)                           \
  printf(                                                    \
      "Skipping negative tests. To enable them, build with " \
      "'negative_testing=on'.\n");
#endif

class TestDisassembler : public PrintDisassembler {
 public:
  TestDisassembler(std::ostream& os, uint32_t pc)  // NOLINT(runtime/references)
      : PrintDisassembler(os, pc) {}

  virtual void PrintCodeAddress(uint32_t code_address) VIXL_OVERRIDE {
    USE(code_address);
  }

  virtual void PrintOpcode16(uint32_t opcode) VIXL_OVERRIDE { USE(opcode); }

  virtual void PrintOpcode32(uint32_t opcode) VIXL_OVERRIDE { USE(opcode); }

  void DisassembleA32(const CodeBuffer& buffer,
                      ptrdiff_t start,
                      ptrdiff_t end) {
    DisassembleA32Buffer(buffer.GetOffsetAddress<const uint32_t*>(start),
                         end - start);
  }

  void DisassembleT32(const CodeBuffer& buffer,
                      ptrdiff_t start,
                      ptrdiff_t end) {
    DisassembleT32Buffer(buffer.GetOffsetAddress<const uint16_t*>(start),
                         end - start);
  }
};


TEST_T32(t32_disassembler_limit1) {
  SETUP();

  masm.UseT32();
  START_COMPARE()
  masm.Add(r9, r10, r11);
  masm.GetBuffer()->Emit16(kLowestT32_32Opcode >> 16);
  END_COMPARE(
      "add r9, r10, r11\n"
      "?\n");

  CLEANUP();
}


TEST_T32(t32_disassembler_limit2) {
  SETUP();

  masm.UseT32();
  START_COMPARE()
  masm.Add(r9, r10, r11);
  masm.Add(r0, r0, r1);
  END_COMPARE(
      "add r9, r10, r11\n"
      "add r0, r1\n");

  CLEANUP();
}


TEST(macro_assembler_orn) {
  SETUP();

  // - Identities.

  COMPARE_BOTH(Orn(r0, r1, 0), "mvn r0, #0\n");
  COMPARE_BOTH(Orn(r0, r0, 0xffffffff), "");

  // - Immediate form. This form does not need macro-assembler support
  //   for T32.

  // Use r0 as the temporary register.
  COMPARE_A32(Orn(r0, r1, 1),
              "mvn r0, #1\n"
              "orr r0, r1, r0\n");
  // Use ip as the temporary register.
  COMPARE_A32(Orns(r0, r0, 1),
              "mvn ip, #1\n"
              "orrs r0, ip\n");

  //  - Too large immediate form.
  COMPARE_BOTH(Orn(r0, r1, 0x00ffffff), "orr r0, r1, #0xff000000\n");
  COMPARE_BOTH(Orn(r0, r1, 0xff00ffff), "orr r0, r1, #0xff0000\n");
  COMPARE_BOTH(Orns(r0, r1, 0x00ffffff), "orrs r0, r1, #0xff000000\n");

  COMPARE_A32(Orns(r0, r1, 0xabcd2345),
              "mov ip, #9029\n"
              "movt ip, #43981\n"
              "mvn r0, ip\n"
              "orrs r0, r1, r0\n");
  COMPARE_T32(Orn(r0, r1, 0xabcd2345),
              "mov r0, #9029\n"
              "movt r0, #43981\n"
              "orn r0, r1, r0\n");

  // - Plain register form. This form does not need macro-assembler
  //   support for T32.

  // Use r0 as the temporary register.
  COMPARE_A32(Orn(r0, r1, r2),
              "mvn r0, r2\n"
              "orr r0, r1, r0\n");
  // Use ip as the temporary register.
  COMPARE_A32(Orn(r0, r0, r1),
              "mvn ip, r1\n"
              "orr r0, ip\n");
  // Use r0 as the temporary register.
  COMPARE_A32(Orn(r0, r1, r0),
              "mvn r0, r0\n"
              "orr r0, r1, r0\n");
  // Use ip as the temporary register.
  COMPARE_A32(Orn(r0, r0, r0),
              "mvn ip, r0\n"
              "orr r0, ip\n");

  // - Shifted register form. This form does not need macro-assembler
  //   support for T32.

  // Use r0 as the temporary register.
  COMPARE_A32(Orn(r0, r1, Operand(r2, LSL, 1)),
              "mvn r0, r2, lsl #1\n"
              "orr r0, r1, r0\n");
  // Use ip as the temporary register.
  COMPARE_A32(Orns(r0, r0, Operand(r2, LSR, 2)),
              "mvn ip, r2, lsr #2\n"
              "orrs r0, ip\n");

  // - Register shifted register form.

  // Use r0 as the temporary register.
  COMPARE_A32(Orn(r0, r1, Operand(r2, LSL, r3)),
              "mvn r0, r2, lsl r3\n"
              "orr r0, r1, r0\n");
  COMPARE_T32(Orn(r0, r1, Operand(r2, LSL, r3)),
              "lsl r0, r2, r3\n"
              "orn r0, r1, r0\n");
  // Use ip as the temporary register.
  COMPARE_A32(Orns(r0, r0, Operand(r2, LSR, r3)),
              "mvn ip, r2, lsr r3\n"
              "orrs r0, ip\n");
  COMPARE_T32(Orns(r0, r0, Operand(r2, LSR, r3)),
              "lsr ip, r2, r3\n"
              "orns r0, ip\n");
  // Use ip as the temporary register.
  COMPARE_A32(Orn(r0, r0, Operand(r0, ASR, r3)),
              "mvn ip, r0, asr r3\n"
              "orr r0, ip\n");
  COMPARE_T32(Orn(r0, r0, Operand(r0, ASR, r3)),
              "asr ip, r0, r3\n"
              "orn r0, ip\n");
  CLEANUP();
}


TEST(macro_assembler_t32_rsc) {
  SETUP();

  // - Immediate form. We can always re-use `rn`.

  // No need for temporay registers.
  COMPARE_T32(Rsc(r0, r1, 1),
              "mvn r0, r1\n"
              "adc r0, #1\n");
  // No need for temporay registers.
  COMPARE_T32(Rscs(r0, r0, 2),
              "mvn r0, r0\n"
              "adcs r0, #2\n");

  //  - Too large immediate form.

  // TODO: optimize this.
  COMPARE_A32(Rsc(r0, r1, 0x00ffffff),
              "mvn r0, #4278190080\n"
              "rsc r0, r1, r0\n");
  COMPARE_T32(Rscs(r0, r1, 0x00ffffff),
              "mvn r0, r1\n"
              "mvn ip, #4278190080\n"
              "adcs r0, ip\n");
  COMPARE_A32(Rsc(r0, r0, 0x00ffffff),
              "mvn ip, #4278190080\n"
              "rsc r0, ip\n");
  COMPARE_T32(Rscs(r0, r0, 0x00ffffff),
              "mvn r0, r0\n"
              "mvn ip, #4278190080\n"
              "adcs r0, ip\n");

  COMPARE_A32(Rsc(r0, r1, 0xabcd2345),
              "mov r0, #9029\n"
              "movt r0, #43981\n"
              "rsc r0, r1, r0\n");
  COMPARE_T32(Rscs(r0, r1, 0xabcd2345),
              "mvn r0, r1\n"
              "mov ip, #56506\n"
              "movt ip, #21554\n"
              "sbcs r0, ip\n");
  COMPARE_A32(Rsc(r0, r0, 0xabcd2345),
              "mov ip, #9029\n"
              "movt ip, #43981\n"
              "rsc r0, ip\n");
  COMPARE_T32(Rscs(r0, r0, 0xabcd2345),
              "mvn r0, r0\n"
              "mov ip, #56506\n"
              "movt ip, #21554\n"
              "sbcs r0, ip\n");

  // - Plain register form.

  // No need for temporary registers.
  COMPARE_T32(Rscs(r0, r1, r2),
              "mvn r0, r1\n"
              "adcs r0, r2\n");
  // Use r0 as the temporary register.
  COMPARE_T32(Rscs(r0, r1, r1),
              "mvn r0, r1\n"
              "adcs r0, r1\n");
  // Use ip as the temporary register.
  COMPARE_T32(Rscs(r0, r0, r0),
              "mvn ip, r0\n"
              "adcs r0, ip, r0\n");

  // - Shifted register form.

  // No need for temporay registers.
  COMPARE_T32(Rsc(r0, r1, Operand(r2, LSL, 1)),
              "mvn r0, r1\n"
              "adc r0, r2, lsl #1\n");
  // Use ip as the temporary register.
  COMPARE_T32(Rscs(r0, r1, Operand(r0, LSR, 2)),
              "mvn ip, r1\n"
              "adcs r0, ip, r0, lsr #2\n");
  // Use r0 as the temporary register.
  COMPARE_T32(Rsc(r0, r1, Operand(r1, ASR, 3)),
              "mvn r0, r1\n"
              "adc r0, r1, asr #3\n");
  // Use ip as the temporary register.
  COMPARE_T32(Rscs(r0, r0, Operand(r0, ROR, 4)),
              "mvn ip, r0\n"
              "adcs r0, ip, r0, ror #4\n");

  // - Register shifted register form. The macro-assembler handles this form in
  //   two steps. First, a shift instruction is generated from the operand. And
  //   finally the operation is reduced to its plain register form.

  COMPARE_T32(Rsc(r0, r1, Operand(r2, LSL, r3)),
              "lsl r0, r2, r3\n"
              "mvn ip, r1\n"
              "adc r0, ip, r0\n");
  // Use r0 and ip as the temporary register.
  COMPARE_T32(Rscs(r0, r1, Operand(r1, LSR, r3)),
              "lsr r0, r1, r3\n"
              "mvn ip, r1\n"
              "adcs r0, ip, r0\n");
  // Use ip and r0 as the temporary register.
  COMPARE_T32(Rsc(r0, r0, Operand(r2, ASR, r3)),
              "asr ip, r2, r3\n"
              "mvn r0, r0\n"
              "adc r0, ip\n");
  // Use ip and r0 as the temporary register.
  COMPARE_T32(Rscs(r0, r0, Operand(r0, ROR, r3)),
              "ror ip, r0, r3\n"
              "mvn r0, r0\n"
              "adcs r0, ip\n");
  // Use ip and r0 as the temporary register.
  COMPARE_T32(Rsc(r0, r0, Operand(r0, LSL, r0)),
              "lsl ip, r0, r0\n"
              "mvn r0, r0\n"
              "adc r0, ip\n");

  CLEANUP();
}


TEST(macro_assembler_t32_register_shift_register) {
  SETUP();

  COMPARE_T32(Adc(r0, r1, Operand(r2, LSL, r3)),
              "lsl r0, r2, r3\n"
              "adc r0, r1, r0\n");
  COMPARE_T32(Adcs(r0, r0, Operand(r2, LSR, r3)),
              "lsr ip, r2, r3\n"
              "adcs r0, ip\n");
  COMPARE_T32(Add(r0, r0, Operand(r0, ASR, r3)),
              "asr ip, r0, r3\n"
              "add r0, ip\n");
  COMPARE_T32(Adds(r0, r0, Operand(r0, ROR, r0)),
              "ror ip, r0, r0\n"
              "adds r0, ip\n");

  CLEANUP();
}


TEST(macro_assembler_big_offset) {
  SETUP();

  COMPARE_BOTH(Ldr(r0, MemOperand(r1, 0xfff123)),
               "add r0, r1, #1044480\n"  // #0xff000
               "add r0, #15728640\n"     // #0x00f00000
               "ldr r0, [r0, #291]\n");  // #0x123
  COMPARE_BOTH(Ldr(r0, MemOperand(r1, 0xff123)),
               "add r0, r1, #1044480\n"  // #0xff000
               "ldr r0, [r0, #291]\n");  // #0x123
  COMPARE_BOTH(Ldr(r0, MemOperand(r1, -0xff123)),
               "sub r0, r1, #1048576\n"   // #0x100000
               "ldr r0, [r0, #3805]\n");  // #0xedd

  COMPARE_A32(Ldr(r0, MemOperand(r1, 0xfff123, PreIndex)),
              "add r1, #1044480\n"       // #0xff000
              "add r1, #15728640\n"      // #0x00f00000
              "ldr r0, [r1, #291]!\n");  // #0x123
  COMPARE_A32(Ldr(r0, MemOperand(r1, 0xff123, PreIndex)),
              "add r1, #1044480\n"       // #0xff000
              "ldr r0, [r1, #291]!\n");  // #0x123
  COMPARE_A32(Ldr(r0, MemOperand(r1, -0xff123, PreIndex)),
              "sub r1, #1048576\n"        // #0x100000
              "ldr r0, [r1, #3805]!\n");  // #0xedd

  COMPARE_T32(Ldr(r0, MemOperand(r1, 0xfff12, PreIndex)),
              "add r1, #65280\n"        // #0xff00
              "add r1, #983040\n"       // #0x000f0000
              "ldr r0, [r1, #18]!\n");  // #0x12
  COMPARE_T32(Ldr(r0, MemOperand(r1, 0xff12, PreIndex)),
              "add r1, #65280\n"        // #0xff00
              "ldr r0, [r1, #18]!\n");  // #0x12
  COMPARE_T32(Ldr(r0, MemOperand(r1, -0xff12, PreIndex)),
              "sub r1, #65536\n"         // #0x10000
              "ldr r0, [r1, #238]!\n");  // #0xee

  COMPARE_A32(Ldr(r0, MemOperand(r1, 0xfff123, PostIndex)),
              "ldr r0, [r1], #291\n"   // #0x123
              "add r1, #1044480\n"     // #0xff000
              "add r1, #15728640\n");  // #0x00f00000
  COMPARE_A32(Ldr(r0, MemOperand(r1, 0xff123, PostIndex)),
              "ldr r0, [r1], #291\n"  // #0x123
              "add r1, #1044480\n");  // #0xff000
  COMPARE_A32(Ldr(r0, MemOperand(r1, -0xff123, PostIndex)),
              "ldr r0, [r1], #3805\n"  // #0xedd
              "sub r1, #1048576\n");   // #0x100000

  COMPARE_T32(Ldr(r0, MemOperand(r1, 0xfff12, PostIndex)),
              "ldr r0, [r1], #18\n"  // #0x12
              "add r1, #65280\n"     // #0xff00
              "add r1, #983040\n");  // #0x000f0000
  COMPARE_T32(Ldr(r0, MemOperand(r1, 0xff12, PostIndex)),
              "ldr r0, [r1], #18\n"  // #0x12
              "add r1, #65280\n");   // #0xff00
  COMPARE_T32(Ldr(r0, MemOperand(r1, -0xff12, PostIndex)),
              "ldr r0, [r1], #238\n"  // #0xee
              "sub r1, #65536\n");    // #0x10000

  COMPARE_A32(Ldrh(r0, MemOperand(r1, 0xfff123)),
              "add r0, r1, #61696\n"    // #0xf100
              "add r0, #16711680\n"     // #0x00ff0000
              "ldrh r0, [r0, #35]\n");  // #0x23
  COMPARE_T32(Ldrh(r0, MemOperand(r1, 0xfff123)),
              "add r0, r1, #1044480\n"   // #0xff000
              "add r0, #15728640\n"      // #0x00f00000
              "ldrh r0, [r0, #291]\n");  // #0x123

  COMPARE_A32(Ldrh(r0, MemOperand(r1, 0xff123)),
              "add r0, r1, #61696\n"    // #0xf100
              "add r0, #983040\n"       // #0x000f0000
              "ldrh r0, [r0, #35]\n");  // #0x23
  COMPARE_T32(Ldrh(r0, MemOperand(r1, 0xff123)),
              "add r0, r1, #1044480\n"   // #0xff000
              "ldrh r0, [r0, #291]\n");  // #0x123
  COMPARE_A32(Ldrh(r0, MemOperand(r1, -0xff123)),
              "sub r0, r1, #61952\n"     // #0xf200
              "sub r0, #983040\n"        // #0x000f0000
              "ldrh r0, [r0, #221]\n");  // #0xdd
  COMPARE_T32(Ldrh(r0, MemOperand(r1, -0xff123)),
              "sub r0, r1, #1048576\n"    // #0x100000
              "ldrh r0, [r0, #3805]\n");  // #0xedd

  MUST_FAIL_TEST_BOTH(Ldr(r0, MemOperand(r0, 0xfff12, PreIndex)),
                      "Ill-formed 'ldr' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldr(r0, MemOperand(r0, 0xfff12, PostIndex)),
                      "Ill-formed 'ldr' instruction.\n");
  CLEANUP();
}


TEST(macro_assembler_load) {
  SETUP();

  // Register base and offset that we can encode in both A1 and T1.
  COMPARE_BOTH(Ldr(r0, MemOperand(r1, r8, Offset)), "ldr r0, [r1, r8]\n");

  // Negative register offset. Use the destination as a scratch register,
  // regardless of the values of the base and offset register.
  COMPARE_T32(Ldr(r0, MemOperand(r0, minus, r0, Offset)),
              "sub r0, r0\n"
              "ldr r0, [r0]\n");

  COMPARE_T32(Ldr(r0, MemOperand(r0, minus, r1, Offset)),
              "sub r0, r1\n"
              "ldr r0, [r0]\n");

  COMPARE_T32(Ldr(r0, MemOperand(r1, minus, r0, Offset)),
              "sub r0, r1, r0\n"
              "ldr r0, [r0]\n");

  COMPARE_T32(Ldr(r0, MemOperand(r1, minus, r2, Offset)),
              "sub r0, r1, r2\n"
              "ldr r0, [r0]\n");

  // Pre-index negative offset.
  COMPARE_T32(Ldr(r0, MemOperand(r1, minus, r2, PreIndex)),
              "sub r1, r2\n"
              "ldr r0, [r1]\n");

  // Post-index negative offset.
  COMPARE_T32(Ldr(r0, MemOperand(r1, minus, r2, PostIndex)),
              "ldr r0, [r1]\n"
              "sub r1, r2\n");

  // SP is allowed as base, offset and destination.
  COMPARE_BOTH(Ldr(sp, MemOperand(sp, sp, Offset)), "ldr sp, [sp, sp]\n");

  // PC is allowed as destination - make sure it is not used as a temporary
  // register.
  COMPARE_BOTH(Ldr(pc, MemOperand(r0, r0, Offset)), "ldr pc, [r0, r0]\n");
  COMPARE_A32(Ldr(pc, MemOperand(r0, r0, PreIndex)), "ldr pc, [r0, r0]!\n");
  COMPARE_T32(Ldr(pc, MemOperand(r0, r0, PreIndex)),
              "add r0, r0\n"
              "ldr pc, [r0]\n");
  COMPARE_A32(Ldr(pc, MemOperand(r0, r0, PostIndex)), "ldr pc, [r0], r0\n");
  COMPARE_T32(Ldr(pc, MemOperand(r0, r0, PostIndex)),
              "ldr pc, [r0]\n"
              "add r0, r0\n");

  // PC is allowed as register base in the offset variant only for A32.
  COMPARE_A32(Ldr(r0, MemOperand(pc, r0, Offset)), "ldr r0, [pc, r0]\n");
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(pc, r0, Offset)),
                     "The MacroAssembler does not convert loads and stores with"
                     " a PC base register for T32.\n");

  // PC is not allowed as register base in the pre-index and post-index
  // variants.
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(pc, r0, PreIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC base register in pre-index or post-index "
                     "mode.\n");
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(pc, r0, PostIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC base register in pre-index or post-index "
                     "mode.\n");

  // We don't convert loads with PC as the register offset.
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(r0, minus, pc, Offset)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC offset register.\n");
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(r0, pc, PreIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC offset register.\n");
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(r0, pc, PostIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC offset register.\n");

  // TODO: PC should not be allowed as register offset (unpredictable).
  SHOULD_FAIL_TEST_BOTH(Ldr(r0, MemOperand(r0, Sign(plus), pc, Offset)));

  // TODO: PC should not be allowed as register base in A32 with pre-index
  //       and post-index (unpredictable).
  SHOULD_FAIL_TEST_A32(Ldr(r0, MemOperand(pc, r0, PreIndex)));
  SHOULD_FAIL_TEST_A32(Ldr(r0, MemOperand(pc, r0, PostIndex)));

  // TODO: load with the same register used as base and as destination
  //       should fail to assemble (unpredictable).
  SHOULD_FAIL_TEST_A32(Ldr(r0, MemOperand(r0, r1, PreIndex)));
  SHOULD_FAIL_TEST_A32(Ldr(r0, MemOperand(r0, r1, PostIndex)));
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(r0, r1, PreIndex)),
                     "Ill-formed 'ldr' instruction.\n");
  MUST_FAIL_TEST_T32(Ldr(r0, MemOperand(r0, r1, PostIndex)),
                     "Ill-formed 'ldr' instruction.\n");

  CLEANUP();
}


TEST(macro_assembler_store) {
  SETUP();

  // Register base and offset that we can encode in both A1 and T1.
  COMPARE_BOTH(Str(r0, MemOperand(r1, r8, Offset)), "str r0, [r1, r8]\n");

  // Negative register offset.
  COMPARE_T32(Str(r0, MemOperand(r0, minus, r0, Offset)),
              "sub ip, r0, r0\n"
              "str r0, [ip]\n");

  COMPARE_T32(Str(r0, MemOperand(r0, minus, r1, Offset)),
              "sub ip, r0, r1\n"
              "str r0, [ip]\n");

  COMPARE_T32(Str(r0, MemOperand(r1, minus, r0, Offset)),
              "sub ip, r1, r0\n"
              "str r0, [ip]\n");

  COMPARE_T32(Str(r0, MemOperand(r1, minus, r2, Offset)),
              "sub ip, r1, r2\n"
              "str r0, [ip]\n");

  // Pre-index negative offset.
  COMPARE_T32(Str(r0, MemOperand(r1, minus, r2, PreIndex)),
              "sub r1, r2\n"
              "str r0, [r1]\n");

  // Post-index negative offset.
  COMPARE_T32(Str(r0, MemOperand(r1, minus, r2, PostIndex)),
              "str r0, [r1]\n"
              "sub r1, r2\n");

  // SP is allowed as base, offset and source.
  COMPARE_BOTH(Str(sp, MemOperand(sp, sp, Offset)), "str sp, [sp, sp]\n");

  // TODO: PC is allowed as the value we are storing for A32, but
  //       should not be allowed for T32 (unpredictable).
  COMPARE_A32(Str(pc, MemOperand(r0, r0, Offset)), "str pc, [r0, r0]\n");
  COMPARE_A32(Str(pc, MemOperand(r0, r0, PreIndex)), "str pc, [r0, r0]!\n");
  COMPARE_A32(Str(pc, MemOperand(r0, r0, PostIndex)), "str pc, [r0], r0\n");
  SHOULD_FAIL_TEST_T32(Str(pc, MemOperand(r0, r0, Offset)));
  SHOULD_FAIL_TEST_T32(Str(pc, MemOperand(r0, r0, PreIndex)));
  SHOULD_FAIL_TEST_T32(Str(pc, MemOperand(r0, r0, PostIndex)));

  // PC is allowed as register base in the offset variant only for A32.
  COMPARE_A32(Str(r0, MemOperand(pc, r0, Offset)), "str r0, [pc, r0]\n");
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(pc, r0, Offset)),
                     "The MacroAssembler does not convert loads and stores with"
                     " a PC base register for T32.\n");

  // PC is not allowed as register base in the pre-index and post-index
  // variants.
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(pc, r0, PreIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC base register in pre-index or post-index "
                     "mode.\n");
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(pc, r0, PostIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC base register in pre-index or post-index "
                     "mode.\n");

  // We don't convert loads with PC as the register offset.
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(r0, minus, pc, Offset)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC offset register.\n");
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(r0, pc, PreIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC offset register.\n");
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(r0, pc, PostIndex)),
                     "The MacroAssembler does not convert loads and stores "
                     "with a PC offset register.\n");

  // TODO: PC should not be allowed as register offset (unpredictable).
  SHOULD_FAIL_TEST_BOTH(Str(r0, MemOperand(r0, Sign(plus), pc, Offset)));

  // TODO: PC should not be allowed as register base in A32 with pre-index
  //       and post-index (unpredictable).
  SHOULD_FAIL_TEST_A32(Str(r0, MemOperand(pc, r0, PreIndex)));
  SHOULD_FAIL_TEST_A32(Str(r0, MemOperand(pc, r0, PostIndex)));

  // TODO: store with the same register used as base and as source
  //       should fail to assemble (unpredictable).
  SHOULD_FAIL_TEST_A32(Str(r0, MemOperand(r0, r1, PreIndex)));
  SHOULD_FAIL_TEST_A32(Str(r0, MemOperand(r0, r1, PostIndex)));
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(r0, r1, PreIndex)),
                     "Ill-formed 'str' instruction.\n");
  MUST_FAIL_TEST_T32(Str(r0, MemOperand(r0, r1, PostIndex)),
                     "Ill-formed 'str' instruction.\n");

  CLEANUP();
}


TEST(macro_assembler_ldrd) {
  SETUP();

  // - Tests with no offset.

  COMPARE_BOTH(Ldrd(r0, r1, MemOperand(r3)), "ldrd r0, r1, [r3]\n");
  // Destination registers need to start with a even numbered register on A32.
  MUST_FAIL_TEST_A32(Ldrd(r1, r2, MemOperand(r3)),
                     "Unpredictable instruction.\n");
  COMPARE_T32(Ldrd(r1, r2, MemOperand(r3)), "ldrd r1, r2, [r3]\n");
  // Registers need to be adjacent on A32.
  MUST_FAIL_TEST_A32(Ldrd(r0, r2, MemOperand(r1)),
                     "Ill-formed 'ldrd' instruction.\n");
  COMPARE_T32(Ldrd(r0, r2, MemOperand(r1)), "ldrd r0, r2, [r1]\n");

  COMPARE_BOTH(Ldrd(r0, r1, MemOperand(r2)), "ldrd r0, r1, [r2]\n");

  // - Tests with immediate offsets.

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, 1020)),
              "add r0, r2, #1020\n"
              "ldrd r0, r1, [r0]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, 1020)), "ldrd r0, r1, [r2, #1020]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, -1020)),
              "sub r0, r2, #1020\n"
              "ldrd r0, r1, [r0]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, -1020)),
              "ldrd r0, r1, [r2, #-1020]\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, 0xabcc)),
              "add r0, r2, #43776\n"
              "ldrd r0, r1, [r0, #204]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, 0xabcc)),
              "add r0, r2, #43008\n"
              "ldrd r0, r1, [r0, #972]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, -0xabcc)),
              "sub r0, r2, #44032\n"
              "ldrd r0, r1, [r0, #52]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, -0xabcc)),
              "sub r0, r2, #44032\n"
              "ldrd r0, r1, [r0, #52]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, 0xabcdec)),
              "add r0, r2, #52480\n"
              "add r0, #11206656\n"
              "ldrd r0, r1, [r0, #236]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, 0xabcdec)),
              "add r0, r2, #248832\n"
              "add r0, #11010048\n"
              "ldrd r0, r1, [r0, #492]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, -0xabcdec)),
              "sub r0, r2, #52736\n"
              "sub r0, #11206656\n"
              "ldrd r0, r1, [r0, #20]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, -0xabcdec)),
              "sub r0, r2, #774144\n"
              "sub r0, #10485760\n"
              "ldrd r0, r1, [r0, #532]\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r0, 0xabcc)),
              "add r1, r0, #43776\n"
              "ldrd r0, r1, [r1, #204]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r0, 0xabcc)),
              "add r1, r0, #43008\n"
              "ldrd r0, r1, [r1, #972]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r0, -0xabcc)),
              "sub r1, r0, #44032\n"
              "ldrd r0, r1, [r1, #52]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r0, -0xabcc)),
              "sub r1, r0, #44032\n"
              "ldrd r0, r1, [r1, #52]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r0, 0xabcdec)),
              "add r1, r0, #52480\n"
              "add r1, #11206656\n"
              "ldrd r0, r1, [r1, #236]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r0, 0xabcdec)),
              "add r1, r0, #248832\n"
              "add r1, #11010048\n"
              "ldrd r0, r1, [r1, #492]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r0, -0xabcdec)),
              "sub r1, r0, #52736\n"
              "sub r1, #11206656\n"
              "ldrd r0, r1, [r1, #20]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r0, -0xabcdec)),
              "sub r1, r0, #774144\n"
              "sub r1, #10485760\n"
              "ldrd r0, r1, [r1, #532]\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r1, 0xabcc)),
              "add r0, r1, #43776\n"
              "ldrd r0, r1, [r0, #204]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r1, 0xabcc)),
              "add r0, r1, #43008\n"
              "ldrd r0, r1, [r0, #972]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r1, -0xabcc)),
              "sub r0, r1, #44032\n"
              "ldrd r0, r1, [r0, #52]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r1, -0xabcc)),
              "sub r0, r1, #44032\n"
              "ldrd r0, r1, [r0, #52]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r1, 0xabcdec)),
              "add r0, r1, #52480\n"
              "add r0, #11206656\n"
              "ldrd r0, r1, [r0, #236]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r1, 0xabcdec)),
              "add r0, r1, #248832\n"
              "add r0, #11010048\n"
              "ldrd r0, r1, [r0, #492]\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r1, -0xabcdec)),
              "sub r0, r1, #52736\n"
              "sub r0, #11206656\n"
              "ldrd r0, r1, [r0, #20]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r1, -0xabcdec)),
              "sub r0, r1, #774144\n"
              "sub r0, #10485760\n"
              "ldrd r0, r1, [r0, #532]\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, 0xabcc, PostIndex)),
              "ldrd r0, r1, [r2], #204\n"
              "add r2, #43776\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, 0xabcc, PostIndex)),
              "ldrd r0, r1, [r2], #972\n"
              "add r2, #43008\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, -0xabcc, PostIndex)),
              "ldrd r0, r1, [r2], #52\n"
              "sub r2, #44032\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, -0xabcc, PostIndex)),
              "ldrd r0, r1, [r2], #52\n"
              "sub r2, #44032\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, 0xabcdec, PostIndex)),
              "ldrd r0, r1, [r2], #236\n"
              "add r2, #52480\n"
              "add r2, #11206656\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, 0xabcdec, PostIndex)),
              "ldrd r0, r1, [r2], #492\n"
              "add r2, #248832\n"
              "add r2, #11010048\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, -0xabcdec, PostIndex)),
              "ldrd r0, r1, [r2], #20\n"
              "sub r2, #52736\n"
              "sub r2, #11206656\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, -0xabcdec, PostIndex)),
              "ldrd r0, r1, [r2], #532\n"
              "sub r2, #774144\n"
              "sub r2, #10485760\n");

  // PostIndex with the same register as base and destination is invalid.
  MUST_FAIL_TEST_BOTH(Ldrd(r0, r1, MemOperand(r0, 0xabcd, PostIndex)),
                      "Ill-formed 'ldrd' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldrd(r0, r1, MemOperand(r1, 0xabcdef, PostIndex)),
                      "Ill-formed 'ldrd' instruction.\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, 0xabcc, PreIndex)),
              "add r2, #43776\n"
              "ldrd r0, r1, [r2, #204]!\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, 0xabcc, PreIndex)),
              "add r2, #43008\n"
              "ldrd r0, r1, [r2, #972]!\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, -0xabcc, PreIndex)),
              "sub r2, #44032\n"
              "ldrd r0, r1, [r2, #52]!\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, -0xabcc, PreIndex)),
              "sub r2, #44032\n"
              "ldrd r0, r1, [r2, #52]!\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, 0xabcdec, PreIndex)),
              "add r2, #52480\n"
              "add r2, #11206656\n"
              "ldrd r0, r1, [r2, #236]!\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, 0xabcdec, PreIndex)),
              "add r2, #248832\n"
              "add r2, #11010048\n"
              "ldrd r0, r1, [r2, #492]!\n");
  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, -0xabcdec, PreIndex)),
              "sub r2, #52736\n"
              "sub r2, #11206656\n"
              "ldrd r0, r1, [r2, #20]!\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, -0xabcdec, PreIndex)),
              "sub r2, #774144\n"
              "sub r2, #10485760\n"
              "ldrd r0, r1, [r2, #532]!\n");

  // - Tests with register offsets.

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, r3)), "ldrd r0, r1, [r2, r3]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, r3)),
              "add r0, r2, r3\n"
              "ldrd r0, r1, [r0]\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, minus, r3)),
              "ldrd r0, r1, [r2, -r3]\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, minus, r3)),
              "sub r0, r2, r3\n"
              "ldrd r0, r1, [r0]\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, r3, PostIndex)),
              "ldrd r0, r1, [r2], r3\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, r3, PostIndex)),
              "ldrd r0, r1, [r2]\n"
              "add r2, r3\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, minus, r3, PostIndex)),
              "ldrd r0, r1, [r2], -r3\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, minus, r3, PostIndex)),
              "ldrd r0, r1, [r2]\n"
              "sub r2, r3\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, r3, PreIndex)),
              "ldrd r0, r1, [r2, r3]!\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, r3, PreIndex)),
              "add r2, r3\n"
              "ldrd r0, r1, [r2]\n");

  COMPARE_A32(Ldrd(r0, r1, MemOperand(r2, minus, r3, PreIndex)),
              "ldrd r0, r1, [r2, -r3]!\n");
  COMPARE_T32(Ldrd(r0, r1, MemOperand(r2, minus, r3, PreIndex)),
              "sub r2, r3\n"
              "ldrd r0, r1, [r2]\n");

  // - We do not support register shifted base register operands with LDRD.

  MUST_FAIL_TEST_BOTH(Ldrd(r0, r1, MemOperand(r2, r3, LSL, 4)),
                      "Ill-formed 'ldrd' instruction.\n");

  // First register is odd - rejected by the Assembler.
  MUST_FAIL_TEST_A32(Ldrd(r1, r2, MemOperand(r0)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Ldrd(r1, r2, MemOperand(r0, r0, PreIndex)),
                     "Unpredictable instruction.\n");
  // First register is odd - rejected by the MacroAssembler.
  MUST_FAIL_TEST_A32(Ldrd(r1, r2, MemOperand(r0, 0xabcd, PreIndex)),
                     "Ill-formed 'ldrd' instruction.\n");

  // First register is lr - rejected by the Assembler.
  MUST_FAIL_TEST_A32(Ldrd(lr, pc, MemOperand(r0)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Ldrd(lr, pc, MemOperand(r0, r0, PreIndex)),
                     "Unpredictable instruction.\n");
  // First register is lr - rejected by the MacroAssembler.
  MUST_FAIL_TEST_A32(Ldrd(lr, pc, MemOperand(r0, 0xabcd, PreIndex)),
                     "Ill-formed 'ldrd' instruction.\n");

  // Non-adjacent registers.
  MUST_FAIL_TEST_A32(Ldrd(r0, r2, MemOperand(r0)),
                     "Ill-formed 'ldrd' instruction.\n");

  CLEANUP();
}

TEST(macro_assembler_strd) {
  SETUP();

  // - Tests with no offset.

  COMPARE_BOTH(Strd(r0, r1, MemOperand(r3)), "strd r0, r1, [r3]\n");
  // Destination registers need to start with a even numbered register on A32.
  MUST_FAIL_TEST_A32(Strd(r1, r2, MemOperand(r3)),
                     "Unpredictable instruction.\n");
  COMPARE_T32(Strd(r1, r2, MemOperand(r3)), "strd r1, r2, [r3]\n");
  // Registers need to be adjacent on A32.
  MUST_FAIL_TEST_A32(Strd(r0, r2, MemOperand(r1)),
                     "Ill-formed 'strd' instruction.\n");
  COMPARE_T32(Strd(r0, r2, MemOperand(r1)), "strd r0, r2, [r1]\n");

  COMPARE_BOTH(Strd(r0, r1, MemOperand(r2)), "strd r0, r1, [r2]\n");

  // - Tests with immediate offsets.

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, 1020)),
              "add ip, r2, #1020\n"
              "strd r0, r1, [ip]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, 1020)), "strd r0, r1, [r2, #1020]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, -1020)),
              "sub ip, r2, #1020\n"
              "strd r0, r1, [ip]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, -1020)),
              "strd r0, r1, [r2, #-1020]\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, 0xabcc)),
              "add ip, r2, #43776\n"
              "strd r0, r1, [ip, #204]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, 0xabcc)),
              "add ip, r2, #43008\n"
              "strd r0, r1, [ip, #972]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, -0xabcc)),
              "sub ip, r2, #44032\n"
              "strd r0, r1, [ip, #52]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, -0xabcc)),
              "sub ip, r2, #44032\n"
              "strd r0, r1, [ip, #52]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, 0xabcdec)),
              "add ip, r2, #52480\n"
              "add ip, #11206656\n"
              "strd r0, r1, [ip, #236]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, 0xabcdec)),
              "add ip, r2, #248832\n"
              "add ip, #11010048\n"
              "strd r0, r1, [ip, #492]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, -0xabcdec)),
              "sub ip, r2, #52736\n"
              "sub ip, #11206656\n"
              "strd r0, r1, [ip, #20]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, -0xabcdec)),
              "sub ip, r2, #774144\n"
              "sub ip, #10485760\n"
              "strd r0, r1, [ip, #532]\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r0, 0xabcc)),
              "add ip, r0, #43776\n"
              "strd r0, r1, [ip, #204]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r0, 0xabcc)),
              "add ip, r0, #43008\n"
              "strd r0, r1, [ip, #972]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r0, -0xabcc)),
              "sub ip, r0, #44032\n"
              "strd r0, r1, [ip, #52]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r0, -0xabcc)),
              "sub ip, r0, #44032\n"
              "strd r0, r1, [ip, #52]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r0, 0xabcdec)),
              "add ip, r0, #52480\n"
              "add ip, #11206656\n"
              "strd r0, r1, [ip, #236]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r0, 0xabcdec)),
              "add ip, r0, #248832\n"
              "add ip, #11010048\n"
              "strd r0, r1, [ip, #492]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r0, -0xabcdec)),
              "sub ip, r0, #52736\n"
              "sub ip, #11206656\n"
              "strd r0, r1, [ip, #20]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r0, -0xabcdec)),
              "sub ip, r0, #774144\n"
              "sub ip, #10485760\n"
              "strd r0, r1, [ip, #532]\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r1, 0xabcc)),
              "add ip, r1, #43776\n"
              "strd r0, r1, [ip, #204]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r1, 0xabcc)),
              "add ip, r1, #43008\n"
              "strd r0, r1, [ip, #972]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r1, -0xabcc)),
              "sub ip, r1, #44032\n"
              "strd r0, r1, [ip, #52]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r1, -0xabcc)),
              "sub ip, r1, #44032\n"
              "strd r0, r1, [ip, #52]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r1, 0xabcdec)),
              "add ip, r1, #52480\n"
              "add ip, #11206656\n"
              "strd r0, r1, [ip, #236]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r1, 0xabcdec)),
              "add ip, r1, #248832\n"
              "add ip, #11010048\n"
              "strd r0, r1, [ip, #492]\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r1, -0xabcdec)),
              "sub ip, r1, #52736\n"
              "sub ip, #11206656\n"
              "strd r0, r1, [ip, #20]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r1, -0xabcdec)),
              "sub ip, r1, #774144\n"
              "sub ip, #10485760\n"
              "strd r0, r1, [ip, #532]\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, 0xabcc, PostIndex)),
              "strd r0, r1, [r2], #204\n"
              "add r2, #43776\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, 0xabcc, PostIndex)),
              "strd r0, r1, [r2], #972\n"
              "add r2, #43008\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, -0xabcc, PostIndex)),
              "strd r0, r1, [r2], #52\n"
              "sub r2, #44032\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, -0xabcc, PostIndex)),
              "strd r0, r1, [r2], #52\n"
              "sub r2, #44032\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, 0xabcdec, PostIndex)),
              "strd r0, r1, [r2], #236\n"
              "add r2, #52480\n"
              "add r2, #11206656\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, 0xabcdec, PostIndex)),
              "strd r0, r1, [r2], #492\n"
              "add r2, #248832\n"
              "add r2, #11010048\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, -0xabcdec, PostIndex)),
              "strd r0, r1, [r2], #20\n"
              "sub r2, #52736\n"
              "sub r2, #11206656\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, -0xabcdec, PostIndex)),
              "strd r0, r1, [r2], #532\n"
              "sub r2, #774144\n"
              "sub r2, #10485760\n");

  // PostIndex with the same register as base and source is invalid.
  MUST_FAIL_TEST_BOTH(Strd(r0, r1, MemOperand(r0, 0xabcd, PostIndex)),
                      "Ill-formed 'strd' instruction.\n");
  MUST_FAIL_TEST_BOTH(Strd(r0, r1, MemOperand(r1, 0xabcdef, PostIndex)),
                      "Ill-formed 'strd' instruction.\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, 0xabcc, PreIndex)),
              "add r2, #43776\n"
              "strd r0, r1, [r2, #204]!\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, 0xabcc, PreIndex)),
              "add r2, #43008\n"
              "strd r0, r1, [r2, #972]!\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, -0xabcc, PreIndex)),
              "sub r2, #44032\n"
              "strd r0, r1, [r2, #52]!\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, -0xabcc, PreIndex)),
              "sub r2, #44032\n"
              "strd r0, r1, [r2, #52]!\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, 0xabcdec, PreIndex)),
              "add r2, #52480\n"
              "add r2, #11206656\n"
              "strd r0, r1, [r2, #236]!\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, 0xabcdec, PreIndex)),
              "add r2, #248832\n"
              "add r2, #11010048\n"
              "strd r0, r1, [r2, #492]!\n");
  COMPARE_A32(Strd(r0, r1, MemOperand(r2, -0xabcdec, PreIndex)),
              "sub r2, #52736\n"
              "sub r2, #11206656\n"
              "strd r0, r1, [r2, #20]!\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, -0xabcdec, PreIndex)),
              "sub r2, #774144\n"
              "sub r2, #10485760\n"
              "strd r0, r1, [r2, #532]!\n");

  // - Tests with register offsets.

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, r3)), "strd r0, r1, [r2, r3]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, r3)),
              "add ip, r2, r3\n"
              "strd r0, r1, [ip]\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, minus, r3)),
              "strd r0, r1, [r2, -r3]\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, minus, r3)),
              "sub ip, r2, r3\n"
              "strd r0, r1, [ip]\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, r3, PostIndex)),
              "strd r0, r1, [r2], r3\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, r3, PostIndex)),
              "strd r0, r1, [r2]\n"
              "add r2, r3\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, minus, r3, PostIndex)),
              "strd r0, r1, [r2], -r3\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, minus, r3, PostIndex)),
              "strd r0, r1, [r2]\n"
              "sub r2, r3\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, r3, PreIndex)),
              "strd r0, r1, [r2, r3]!\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, r3, PreIndex)),
              "add r2, r3\n"
              "strd r0, r1, [r2]\n");

  COMPARE_A32(Strd(r0, r1, MemOperand(r2, minus, r3, PreIndex)),
              "strd r0, r1, [r2, -r3]!\n");
  COMPARE_T32(Strd(r0, r1, MemOperand(r2, minus, r3, PreIndex)),
              "sub r2, r3\n"
              "strd r0, r1, [r2]\n");

  // - We do not support register shifted base register operands with LDRD.

  MUST_FAIL_TEST_BOTH(Strd(r0, r1, MemOperand(r2, r3, LSL, 4)),
                      "Ill-formed 'strd' instruction.\n");

  // First register is odd - rejected by the Assembler.
  MUST_FAIL_TEST_A32(Strd(r1, r2, MemOperand(r0)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Strd(r1, r2, MemOperand(r0, r0, PreIndex)),
                     "Unpredictable instruction.\n");
  // First register is odd - rejected by the MacroAssembler.
  MUST_FAIL_TEST_A32(Strd(r1, r2, MemOperand(r0, 0xabcd, PreIndex)),
                     "Ill-formed 'strd' instruction.\n");

  // First register is lr - rejected by the Assembler.
  MUST_FAIL_TEST_A32(Strd(lr, pc, MemOperand(r0)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Strd(lr, pc, MemOperand(r0, r0, PreIndex)),
                     "Unpredictable instruction.\n");
  // First register is lr - rejected by the MacroAssembler.
  MUST_FAIL_TEST_A32(Strd(lr, pc, MemOperand(r0, 0xabcd, PreIndex)),
                     "Ill-formed 'strd' instruction.\n");

  // Non-adjacent registers.
  MUST_FAIL_TEST_A32(Strd(r0, r2, MemOperand(r0)),
                     "Ill-formed 'strd' instruction.\n");

  CLEANUP();
}


TEST(macro_assembler_wide_immediate) {
  SETUP();

  COMPARE_BOTH(Adc(r0, r1, 0xbadbeef),
               "mov r0, #48879\n"
               "movt r0, #2989\n"
               "adc r0, r1, r0\n");

  COMPARE_BOTH(Add(r0, r0, 0xbadbeef),
               "mov ip, #48879\n"
               "movt ip, #2989\n"
               "add r0, ip\n");

  COMPARE_BOTH(Mov(r0, 0xbadbeef),
               "mov r0, #48879\n"
               "movt r0, #2989\n");
  COMPARE_A32(Mov(eq, r0, 0xbadbeef),
              "moveq r0, #48879\n"
              "movteq r0, #2989\n");
  COMPARE_T32(Mov(eq, r0, 0xbadbeef),
              "bne 0x0000000a\n"
              "mov r0, #48879\n"
              "movt r0, #2989\n");

  COMPARE_BOTH(Movs(r0, 0xbadbeef),
               "mov r0, #48879\n"
               "movt r0, #2989\n"
               "tst r0, r0\n");
  COMPARE_A32(Movs(eq, r0, 0xbadbeef),
              "moveq r0, #48879\n"
              "movteq r0, #2989\n"
              "tsteq r0, r0\n");
  COMPARE_T32(Movs(eq, r0, 0xbadbeef),
              "bne 0x0000000c\n"
              "mov r0, #48879\n"
              "movt r0, #2989\n"
              "tst r0, r0\n");
  COMPARE_A32(Movs(pc, 0x1), "movs pc, #1\n");
  MUST_FAIL_TEST_T32(Movs(pc, 0x1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Movs(pc, 0xbadbeed), "Ill-formed 'movs' instruction.\n");

  COMPARE_BOTH(Mov(pc, 0xbadbeef),
               "mov ip, #48879\n"
               "movt ip, #2989\n"
               "bx ip\n");
  COMPARE_A32(Mov(eq, pc, 0xbadbeef),
              "mov ip, #48879\n"
              "movt ip, #2989\n"
              "bxeq ip\n");
  COMPARE_T32(Mov(eq, pc, 0xbadbeef),
              "bne 0x0000000c\n"
              "mov ip, #48879\n"
              "movt ip, #2989\n"
              "bx ip\n");

  CLEANUP();
}


TEST(macro_assembler_And) {
  SETUP();

  // Identities.
  COMPARE_BOTH(And(r0, r1, 0), "mov r0, #0\n");
  COMPARE_BOTH(And(r0, r0, 0xffffffff), "");
  CLEANUP();
}


TEST(macro_assembler_Bic) {
  SETUP();

  // Identities.
  COMPARE_BOTH(Bic(r0, r1, 0xffffffff), "mov r0, #0\n");
  COMPARE_BOTH(Bic(r0, r0, 0), "");
  CLEANUP();
}


TEST(macro_assembler_Orr) {
  SETUP();

  // Identities.
  COMPARE_BOTH(Orr(r0, r1, 0xffffffff), "mvn r0, #0\n");
  COMPARE_BOTH(Orr(r0, r0, 0), "");
  CLEANUP();
}


TEST(macro_assembler_InstructionCondSizeRROp) {
  SETUP();

  // Special case for Orr <-> Orn correspondance.

  COMPARE_T32(Orr(r0, r1, 0x00ffffff), "orn r0, r1, #0xff000000\n");
  COMPARE_T32(Orrs(r0, r1, 0x00ffffff), "orns r0, r1, #0xff000000\n");

  // Encodable immediates.

  COMPARE_A32(Add(r0, r1, -1), "sub r0, r1, #1\n");
  COMPARE_A32(Adds(r0, r1, -1), "subs r0, r1, #1\n");
  // 0xffffffff is encodable in a T32 ADD.
  COMPARE_T32(Add(r0, r1, -1), "add r0, r1, #4294967295\n");
  COMPARE_T32(Adds(r0, r1, -1), "adds r0, r1, #4294967295\n");

  COMPARE_BOTH(Add(r0, r1, -4), "sub r0, r1, #4\n");
  COMPARE_BOTH(Adds(r0, r1, -4), "subs r0, r1, #4\n");

  COMPARE_BOTH(Adc(r0, r1, -2), "sbc r0, r1, #1\n");
  COMPARE_BOTH(Adcs(r0, r1, -2), "sbcs r0, r1, #1\n");

  COMPARE_A32(Sub(r0, r1, -1), "add r0, r1, #1\n");
  COMPARE_A32(Subs(r0, r1, -1), "adds r0, r1, #1\n");
  // 0xffffffff is encodable in a T32 SUB.
  COMPARE_T32(Sub(r0, r1, -1), "sub r0, r1, #4294967295\n");
  COMPARE_T32(Subs(r0, r1, -1), "subs r0, r1, #4294967295\n");

  COMPARE_BOTH(Sub(r0, r1, -4), "add r0, r1, #4\n");
  COMPARE_BOTH(Subs(r0, r1, -4), "adds r0, r1, #4\n");

  COMPARE_BOTH(Sbc(r0, r1, -5), "adc r0, r1, #4\n");
  COMPARE_BOTH(Sbcs(r0, r1, -5), "adcs r0, r1, #4\n");

  // Non-encodable immediates

  COMPARE_BOTH(Adc(r0, r1, 0xabcd),
               "mov r0, #43981\n"
               "adc r0, r1, r0\n");

  COMPARE_BOTH(Adc(r0, r1, -0xabcd),
               "mov r0, #43980\n"  // This represents #0xabcd - 1.
               "sbc r0, r1, r0\n");

  COMPARE_BOTH(Adc(r0, r1, 0x1234abcd),
               "mov r0, #43981\n"
               "movt r0, #4660\n"
               "adc r0, r1, r0\n");

  COMPARE_BOTH(Adc(r0, r1, -0x1234abcd),
               "mov r0, #43980\n"  // This represents #0x1234abcd - 1.
               "movt r0, #4660\n"
               "sbc r0, r1, r0\n");

  // Non-encodable immediates with the same source and destination registers.

  COMPARE_BOTH(Sbc(r0, r0, 0xabcd),
               "mov ip, #43981\n"
               "sbc r0, ip\n");

  COMPARE_BOTH(Sbc(r0, r0, -0xabcd),
               "mov ip, #43980\n"  // This represents #0xabcd - 1.
               "adc r0, ip\n");

  COMPARE_BOTH(Sbc(r0, r0, 0x1234abcd),
               "mov ip, #43981\n"
               "movt ip, #4660\n"
               "sbc r0, ip\n");

  COMPARE_BOTH(Sbc(r0, r0, -0x1234abcd),
               "mov ip, #43980\n"  // This represents #0x1234abcd - 1.
               "movt ip, #4660\n"
               "adc r0, ip\n");


  // Test that we can pass a register shifted register operand in T32.

  COMPARE_T32(Adc(r0, r1, Operand(r2, LSL, r3)),
              "lsl r0, r2, r3\n"
              "adc r0, r1, r0\n");

  COMPARE_T32(Add(r3, r2, Operand(r2, ASR, r3)),
              "asr r3, r2, r3\n"
              "add r3, r2, r3\n");

  COMPARE_T32(Ands(r3, r2, Operand(r2, LSR, r2)),
              "lsr r3, r2, r2\n"
              "ands r3, r2, r3\n");

  COMPARE_T32(Asr(r2, r2, Operand(r2, ROR, r2)),
              "ror ip, r2, r2\n"
              "asr r2, ip\n");

  COMPARE_T32(Asr(r2, r2, Operand(r2, ROR, r2)),
              "ror ip, r2, r2\n"
              "asr r2, ip\n");


  CLEANUP();
}


TEST(macro_assembler_InstructionCondRO) {
  SETUP();

  COMPARE_BOTH(Teq(r0, 0xbad),
               "mov ip, #2989\n"
               "teq r0, ip\n");
  COMPARE_BOTH(Teq(r0, 0xbadbeef),
               "mov ip, #48879\n"
               "movt ip, #2989\n"
               "teq r0, ip\n");
  MUST_FAIL_TEST_T32(Teq(r0, Operand(r1, LSL, r2)),
                     "Ill-formed 'teq' instruction.\n");

  CLEANUP();
}


TEST(macro_assembler_too_large_immediate) {
  SETUP();

  // Attempting to use a 17-bit immediate with movt.
  MUST_FAIL_TEST_BOTH(Movt(r0, 0x10000), "`Movt` expects a 16-bit immediate.");

  CLEANUP();
}

TEST(macro_assembler_Cbz) {
  SETUP();

#ifdef VIXL_INCLUDE_TARGET_A32
  // Cbz/Cbnz are not available in A32 mode.
  // Make sure GetArchitectureStatePCOffset() returns the correct value.
  __ UseA32();
  Label label_64(__ GetCursorOffset() + __ GetArchitectureStatePCOffset(), 64);
  MUST_FAIL_TEST_A32(Cbz(r0, &label_64), "Cbz is only available for T32.\n");
  MUST_FAIL_TEST_A32(Cbnz(r0, &label_64), "Cbnz is only available for T32.\n");
#endif

#ifdef VIXL_INCLUDE_TARGET_T32
  // Make sure GetArchitectureStatePCOffset() returns the correct value.
  __ UseT32();
  // Largest encodable offset.
  Label label_126(__ GetCursorOffset() + __ GetArchitectureStatePCOffset(),
                  126);
  COMPARE_T32(Cbz(r0, &label_126), "cbz r0, 0x00000082\n");
  COMPARE_T32(Cbnz(r0, &label_126), "cbnz r0, 0x00000082\n");

  // Offset cannot be encoded.
  Label label_128(__ GetCursorOffset() + __ GetArchitectureStatePCOffset(),
                  128);
  COMPARE_T32(Cbz(r0, &label_128),
              "cbnz r0, 0x00000004\n"
              "b 0x00000084\n");
  COMPARE_T32(Cbnz(r0, &label_128),
              "cbz r0, 0x00000004\n"
              "b 0x00000084\n");

  // Offset that cannot be encoded and needs 32-bit branch instruction.
  Label label_8192(__ GetCursorOffset() + __ GetArchitectureStatePCOffset(),
                   8192);
  COMPARE_T32(Cbz(r0, &label_8192),
              "cbnz r0, 0x00000006\n"
              "b 0x00002004\n");
  COMPARE_T32(Cbnz(r0, &label_8192),
              "cbz r0, 0x00000006\n"
              "b 0x00002004\n");

  // Negative offset.
  Label label_neg(__ GetCursorOffset() + __ GetArchitectureStatePCOffset(), -8);
  COMPARE_T32(Cbz(r0, &label_neg),
              "cbnz r0, 0x00000004\n"
              "b 0xfffffffc\n");
  COMPARE_T32(Cbnz(r0, &label_neg),
              "cbz r0, 0x00000004\n"
              "b 0xfffffffc\n");

  // Large negative offset.
  Label label_neg128(__ GetCursorOffset() + __ GetArchitectureStatePCOffset(),
                     -128);
  COMPARE_T32(Cbz(r0, &label_neg128),
              "cbnz r0, 0x00000004\n"
              "b 0xffffff84\n");
  COMPARE_T32(Cbnz(r0, &label_neg128),
              "cbz r0, 0x00000004\n"
              "b 0xffffff84\n");
#endif

  CLEANUP();
}


#define TEST_VMEMOP(MACRO_OP, STRING_OP, DST_REG)                    \
  SETUP();                                                           \
                                                                     \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, 1024)),               \
              "add ip, r8, #1024\n" STRING_OP #DST_REG ", [ip]\n");  \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, 1371)),               \
              "add ip, r8, #1371\n" STRING_OP #DST_REG ", [ip]\n");  \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, 4113)),               \
              "add ip, r8, #17\n"                                    \
              "add ip, #4096\n" STRING_OP #DST_REG ", [ip]\n");      \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, 65808)),              \
              "add ip, r8, #272\n"                                   \
              "add ip, #65536\n" STRING_OP #DST_REG ", [ip]\n");     \
                                                                     \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, -1024)),              \
              "sub ip, r8, #1024\n" STRING_OP #DST_REG ", [ip]\n");  \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, -1371)),              \
              "sub ip, r8, #1371\n" STRING_OP #DST_REG ", [ip]\n");  \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, -4113)),              \
              "sub ip, r8, #17\n"                                    \
              "sub ip, #4096\n" STRING_OP #DST_REG ", [ip]\n");      \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r8, -65808)),             \
              "sub ip, r8, #272\n"                                   \
              "sub ip, #65536\n" STRING_OP #DST_REG ", [ip]\n");     \
                                                                     \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r9, 0, PreIndex)),        \
              STRING_OP #DST_REG ", [r9]\n");                        \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r9, 137, PreIndex)),      \
              "add r9, #137\n" STRING_OP #DST_REG ", [r9]\n");       \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r9, 4110, PreIndex)),     \
              "add r9, #14\n"                                        \
              "add r9, #4096\n" STRING_OP #DST_REG ", [r9]\n");      \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r9, 65623, PreIndex)),    \
              "add r9, #87\n"                                        \
              "add r9, #65536\n" STRING_OP #DST_REG ", [r9]\n");     \
                                                                     \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r9, -137, PreIndex)),     \
              "sub r9, #137\n" STRING_OP #DST_REG ", [r9]\n");       \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r9, -4110, PreIndex)),    \
              "sub r9, #14\n"                                        \
              "sub r9, #4096\n" STRING_OP #DST_REG ", [r9]\n");      \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r9, -65623, PreIndex)),   \
              "sub r9, #87\n"                                        \
              "sub r9, #65536\n" STRING_OP #DST_REG ", [r9]\n");     \
                                                                     \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r10, 0, PostIndex)),      \
              STRING_OP #DST_REG ", [r10]\n");                       \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r10, 137, PostIndex)),    \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "add r10, #137\n");                                    \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r10, 4110, PostIndex)),   \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "add r10, #14\n"                                       \
              "add r10, #4096\n");                                   \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r10, 65623, PostIndex)),  \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "add r10, #87\n"                                       \
              "add r10, #65536\n");                                  \
                                                                     \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r10, -137, PostIndex)),   \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "sub r10, #137\n");                                    \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r10, -4110, PostIndex)),  \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "sub r10, #14\n"                                       \
              "sub r10, #4096\n");                                   \
  COMPARE_T32(MACRO_OP(DST_REG, MemOperand(r10, -65623, PostIndex)), \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "sub r10, #87\n"                                       \
              "sub r10, #65536\n");                                  \
  CLEANUP();

TEST(macro_assembler_T32_Vldr_d) { TEST_VMEMOP(Vldr, "vldr ", d0); }

TEST(macro_assembler_T32_Vstr_d) { TEST_VMEMOP(Vstr, "vstr ", d1); }

TEST(macro_assembler_T32_Vldr_s) { TEST_VMEMOP(Vldr, "vldr ", s2); }

TEST(macro_assembler_T32_Vstr_s) { TEST_VMEMOP(Vstr, "vstr ", s3); }

#undef TEST_VMEMOP

#define TEST_VMEMOP(MACRO_OP, STRING_OP, DST_REG)                    \
  SETUP();                                                           \
                                                                     \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r8, 137)),                \
              "add ip, r8, #137\n" STRING_OP #DST_REG ", [ip]\n");   \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r8, 274)),                \
              "add ip, r8, #18\n"                                    \
              "add ip, #256\n" STRING_OP #DST_REG ", [ip]\n");       \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r8, 65623)),              \
              "add ip, r8, #87\n"                                    \
              "add ip, #65536\n" STRING_OP #DST_REG ", [ip]\n");     \
                                                                     \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r8, -137)),               \
              "sub ip, r8, #137\n" STRING_OP #DST_REG ", [ip]\n");   \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r8, -274)),               \
              "sub ip, r8, #18\n"                                    \
              "sub ip, #256\n" STRING_OP #DST_REG ", [ip]\n");       \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r8, -65623)),             \
              "sub ip, r8, #87\n"                                    \
              "sub ip, #65536\n" STRING_OP #DST_REG ", [ip]\n");     \
                                                                     \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r9, 0, PreIndex)),        \
              STRING_OP #DST_REG ", [r9]\n");                        \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r9, 137, PreIndex)),      \
              "add r9, #137\n" STRING_OP #DST_REG ", [r9]\n");       \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r9, 274, PreIndex)),      \
              "add r9, #18\n"                                        \
              "add r9, #256\n" STRING_OP #DST_REG ", [r9]\n");       \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r9, 65623, PreIndex)),    \
              "add r9, #87\n"                                        \
              "add r9, #65536\n" STRING_OP #DST_REG ", [r9]\n");     \
                                                                     \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r9, -137, PreIndex)),     \
              "sub r9, #137\n" STRING_OP #DST_REG ", [r9]\n");       \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r9, -274, PreIndex)),     \
              "sub r9, #18\n"                                        \
              "sub r9, #256\n" STRING_OP #DST_REG ", [r9]\n");       \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r9, -65623, PreIndex)),   \
              "sub r9, #87\n"                                        \
              "sub r9, #65536\n" STRING_OP #DST_REG ", [r9]\n");     \
                                                                     \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r10, 0, PostIndex)),      \
              STRING_OP #DST_REG ", [r10]\n");                       \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r10, 137, PostIndex)),    \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "add r10, #137\n");                                    \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r10, 274, PostIndex)),    \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "add r10, #18\n"                                       \
              "add r10, #256\n");                                    \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r10, 65623, PostIndex)),  \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "add r10, #87\n"                                       \
              "add r10, #65536\n");                                  \
                                                                     \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r10, -137, PostIndex)),   \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "sub r10, #137\n");                                    \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r10, -274, PostIndex)),   \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "sub r10, #18\n"                                       \
              "sub r10, #256\n");                                    \
  COMPARE_A32(MACRO_OP(DST_REG, MemOperand(r10, -65623, PostIndex)), \
              STRING_OP #DST_REG                                     \
              ", [r10]\n"                                            \
              "sub r10, #87\n"                                       \
              "sub r10, #65536\n");                                  \
  CLEANUP();


TEST(macro_assembler_A32_Vldr_d) { TEST_VMEMOP(Vldr, "vldr ", d0); }

TEST(macro_assembler_A32_Vstr_d) { TEST_VMEMOP(Vstr, "vstr ", d1); }

TEST(macro_assembler_A32_Vldr_s) { TEST_VMEMOP(Vldr, "vldr ", s2); }

TEST(macro_assembler_A32_Vstr_s) { TEST_VMEMOP(Vstr, "vstr ", s3); }

#undef TEST_VMEMOP

TEST(macro_assembler_Vldr_Vstr_negative) {
  SETUP();

  MUST_FAIL_TEST_BOTH(Vldr(s0, MemOperand(pc, 1, PreIndex)),
                      "The MacroAssembler does not convert vldr or vstr"
                      " with a PC base register.\n");

  MUST_FAIL_TEST_BOTH(Vldr(s0, MemOperand(pc, r0, PreIndex)),
                      "Ill-formed 'vldr' instruction.\n");

  MUST_FAIL_TEST_BOTH(Vstr(s0, MemOperand(pc, 1, PreIndex)),
                      "The MacroAssembler does not convert vldr or vstr"
                      " with a PC base register.\n");

  MUST_FAIL_TEST_BOTH(Vstr(s0, MemOperand(pc, r0, PreIndex)),
                      "Ill-formed 'vstr' instruction.\n");

  MUST_FAIL_TEST_BOTH(Vldr(d0, MemOperand(pc, 1, PreIndex)),
                      "The MacroAssembler does not convert vldr or vstr"
                      " with a PC base register.\n");

  MUST_FAIL_TEST_BOTH(Vldr(d0, MemOperand(pc, r0, PreIndex)),
                      "Ill-formed 'vldr' instruction.\n");

  MUST_FAIL_TEST_BOTH(Vstr(d0, MemOperand(pc, 1, PreIndex)),
                      "The MacroAssembler does not convert vldr or vstr"
                      " with a PC base register.\n");

  MUST_FAIL_TEST_BOTH(Vstr(d0, MemOperand(pc, r0, PreIndex)),
                      "Ill-formed 'vstr' instruction.\n");
  CLEANUP();
}

#define TEST_SHIFT_T32(Inst, name, offset)          \
  COMPARE_T32(Inst(r0, Operand(r1, LSL, r2)),       \
              "lsl ip, r1, r2\n" name " r0, ip\n"); \
  COMPARE_T32(Inst(r0, Operand(r1, LSR, r2)),       \
              "lsr ip, r1, r2\n" name " r0, ip\n"); \
  COMPARE_T32(Inst(r0, Operand(r1, ASR, r2)),       \
              "asr ip, r1, r2\n" name " r0, ip\n"); \
  COMPARE_T32(Inst(r0, Operand(r1, ROR, r2)),       \
              "ror ip, r1, r2\n" name " r0, ip\n"); \
  COMPARE_T32(Inst(eq, r0, Operand(r1, LSL, r2)),   \
              "bne " #offset                        \
              "\n"                                  \
              "lsl ip, r1, r2\n" name " r0, ip\n"); \
  COMPARE_T32(Inst(le, r0, Operand(r1, LSL, r2)),   \
              "bgt " #offset                        \
              "\n"                                  \
              "lsl ip, r1, r2\n" name " r0, ip\n");

#define TEST_MOV_SHIFT_T32(Inst, s, offset)                             \
  COMPARE_T32(Inst(r0, Operand(r1, LSL, r2)), "lsl" s " r0, r1, r2\n"); \
  COMPARE_T32(Inst(r0, Operand(r1, LSR, r2)), "lsr" s " r0, r1, r2\n"); \
  COMPARE_T32(Inst(r0, Operand(r1, ASR, r2)), "asr" s " r0, r1, r2\n"); \
  COMPARE_T32(Inst(r0, Operand(r1, ROR, r2)), "ror" s " r0, r1, r2\n"); \
  COMPARE_T32(Inst(eq, r0, Operand(r1, LSL, r2)),                       \
              "bne " #offset                                            \
              "\n"                                                      \
              "lsl" s " r0, r1, r2\n");                                 \
  COMPARE_T32(Inst(le, r0, Operand(r1, LSL, r2)),                       \
              "bgt " #offset                                            \
              "\n"                                                      \
              "lsl" s " r0, r1, r2\n");

#define TEST_WIDE_IMMEDIATE(Inst, name, offset)         \
  COMPARE_BOTH(Inst(r0, 0xbadbeef),                     \
               "mov ip, #48879\n"                       \
               "movt ip, #2989\n" name " r0, ip\n");    \
  COMPARE_A32(Inst(eq, r0, 0xbadbeef),                  \
              "moveq ip, #48879\n"                      \
              "movteq ip, #2989\n" name "eq r0, ip\n"); \
  COMPARE_T32(Inst(eq, r0, 0xbadbeef),                  \
              "bne " #offset                            \
              "\n"                                      \
              "mov ip, #48879\n"                        \
              "movt ip, #2989\n" name " r0, ip\n");

#define TEST_WIDE_IMMEDIATE_PC(Inst, name, offset)            \
  COMPARE_A32(Inst(pc, 0xbadbeef),                            \
              "mov ip, #48879\n"                              \
              "movt ip, #2989\n" name " pc, ip\n");           \
  COMPARE_A32(Inst(eq, pc, 0xbadbeef),                        \
              "moveq ip, #48879\n"                            \
              "movteq ip, #2989\n" name "eq pc, ip\n");       \
  MUST_FAIL_TEST_T32(Inst(pc, 0xbadbeef),                     \
                     "Ill-formed '" name "' instruction.\n"); \
  MUST_FAIL_TEST_T32(Inst(eq, pc, 0xbadbeef),                 \
                     "Ill-formed '" name "' instruction.\n");

TEST(macro_assembler_InstructionCondSizeROp) {
  SETUP();

  // T32 register shifted register.
  TEST_SHIFT_T32(Cmn, "cmn", 0x0000000a)
  TEST_SHIFT_T32(Cmp, "cmp", 0x00000008)
  TEST_SHIFT_T32(Mvn, "mvn", 0x0000000a)
  TEST_SHIFT_T32(Mvns, "mvns", 0x0000000a)
  TEST_SHIFT_T32(Sxtb, "sxtb", 0x0000000a)
  TEST_SHIFT_T32(Sxth, "sxth", 0x0000000a)
  TEST_SHIFT_T32(Tst, "tst", 0x0000000a)
  TEST_SHIFT_T32(Uxtb, "uxtb", 0x0000000a)
  TEST_SHIFT_T32(Uxth, "uxth", 0x0000000a)

  TEST_MOV_SHIFT_T32(Mov, "", 0x00000006)
  TEST_MOV_SHIFT_T32(Movs, "s", 0x00000006)

  MUST_FAIL_TEST_BOTH(Movs(pc, r0), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Movs(pc, Operand(r0, LSL, 0x4)),
                      "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Movs(pc, Operand(r0, ASR, r2)),
                      "Unpredictable instruction.\n");

  // Wide immediates (Mov and Movs are tested in
  // "macro_assembler_wide_immediate").
  TEST_WIDE_IMMEDIATE(Cmp, "cmp", 0x0000000c);
  TEST_WIDE_IMMEDIATE(Cmn, "cmn", 0x0000000e);
  TEST_WIDE_IMMEDIATE(Tst, "tst", 0x0000000e);
  TEST_WIDE_IMMEDIATE_PC(Cmp, "cmp", 0x0000000c);
  TEST_WIDE_IMMEDIATE_PC(Cmn, "cmn", 0x0000000e);
  TEST_WIDE_IMMEDIATE_PC(Tst, "tst", 0x0000000e);

  // For Mvn and Mvns, we don't allow PC as a destination.
  TEST_WIDE_IMMEDIATE(Mvn, "mvn", 0x0000000e);
  TEST_WIDE_IMMEDIATE(Mvns, "mvns", 0x0000000e);
  MUST_FAIL_TEST_BOTH(Mvn(pc, 0xbadbeef), "Ill-formed 'mvn' instruction.\n");
  MUST_FAIL_TEST_BOTH(Mvn(eq, pc, 0xbadbeef),
                      "Ill-formed 'mvn' instruction.\n");
  MUST_FAIL_TEST_BOTH(Mvns(pc, 0xbadbeef), "Ill-formed 'mvns' instruction.\n");
  MUST_FAIL_TEST_BOTH(Mvns(eq, pc, 0xbadbeef),
                      "Ill-formed 'mvns' instruction.\n");

  MUST_FAIL_TEST_BOTH(Sxtb(r0, 0x1), "Ill-formed 'sxtb' instruction.\n");
  MUST_FAIL_TEST_BOTH(Sxth(r0, 0x1), "Ill-formed 'sxth' instruction.\n");
  MUST_FAIL_TEST_BOTH(Uxtb(r0, 0x1), "Ill-formed 'uxtb' instruction.\n");
  MUST_FAIL_TEST_BOTH(Uxth(r0, 0x1), "Ill-formed 'uxth' instruction.\n");

  CLEANUP();
}

#undef TEST_SHIFT_T32
#undef TEST_MOV_SHIFT_T32
#undef TEST_WIDE_IMMEDIATE
#undef TEST_WIDE_IMMEDIATE_PC

TEST(macro_assembler_Msr) {
  SETUP();

  // Msr with immediate for T32.
  COMPARE_T32(Msr(APSR_nzcvq, 0x0),
              "mov ip, #0\n"
              "msr APSR_nzcvq, ip\n");

  // Wide immediate.
  COMPARE_BOTH(Msr(APSR_nzcvq, 0xbadbeef),
               "mov ip, #48879\n"
               "movt ip, #2989\n"
               "msr APSR_nzcvq, ip\n");

  // Other types of operands are not handled.
  MUST_FAIL_TEST_BOTH(Msr(APSR_nzcvq, Operand(r0, LSR, r1)),
                      "Ill-formed 'msr' instruction.\n");
  CLEANUP();
}


TEST(macro_assembler_Vmov_imm) {
  SETUP();

  COMPARE_BOTH(Vmov(s0, 0.0f),
               "mov ip, #0\n"
               "vmov s0, ip\n");
  COMPARE_BOTH(Vmov(s1, 1.0f), "vmov.f32 s1, #1\n");
  COMPARE_BOTH(Vmov(s2, RawbitsToFloat(0x0000db6c)),
               "mov ip, #56172\n"
               "vmov s2, ip\n");
  COMPARE_BOTH(Vmov(s3, RawbitsToFloat(0x327b23c6)),
               "mov ip, #9158\n"
               "movt ip, #12923\n"
               "vmov s3, ip\n");
  COMPARE_BOTH(Vmov(s4, RawbitsToFloat(0xffcc7fff)),
               "mvn ip, #3375104\n"
               "vmov s4, ip\n");
  COMPARE_BOTH(Vmov(s5, RawbitsToFloat(0xb72df575)),
               "mov ip, #62837\n"
               "movt ip, #46893\n"
               "vmov s5, ip\n");

  COMPARE_BOTH(Vmov(d6, 0.0), "vmov.i64 d6, #0x0000000000000000\n");
  COMPARE_BOTH(Vmov(d7, 1.0), "vmov.f64 d7, #1\n");
  COMPARE_BOTH(Vmov(d8, RawbitsToDouble(0x000000000000af8e)),
               "mov ip, #44942\n"
               "vdup.32 d8, ip\n"
               "mov ip, #0\n"
               "vmov.32 d8[1], ip\n");
  COMPARE_BOTH(Vmov(d9, RawbitsToDouble(0x000070210000af8e)),
               "mov ip, #44942\n"
               "vdup.32 d9, ip\n"
               "mov ip, #28705\n"
               "vmov.32 d9[1], ip\n");
  COMPARE_BOTH(Vmov(d10, RawbitsToDouble(0x7021000000000000)),
               "mov ip, #0\n"
               "vdup.32 d10, ip\n"
               "mov ip, #0\n"
               "movt ip, #28705\n"
               "vmov.32 d10[1], ip\n");
  COMPARE_BOTH(Vmov(d11, RawbitsToDouble(0x7021da4b0000af8e)),
               "mov ip, #44942\n"
               "vdup.32 d11, ip\n"
               "mov ip, #55883\n"
               "movt ip, #28705\n"
               "vmov.32 d11[1], ip\n");
  COMPARE_BOTH(Vmov(d12, RawbitsToDouble(0x0cff553204ec4a3f)),
               "mov ip, #19007\n"
               "movt ip, #1260\n"
               "vdup.32 d12, ip\n"
               "mov ip, #21810\n"
               "movt ip, #3327\n"
               "vmov.32 d12[1], ip\n");
  COMPARE_BOTH(Vmov(d13, RawbitsToDouble(0xa2037ad20000f592)),
               "mov ip, #62866\n"
               "vdup.32 d13, ip\n"
               "mov ip, #31442\n"
               "movt ip, #41475\n"
               "vmov.32 d13[1], ip\n");
  COMPARE_BOTH(Vmov(d14, RawbitsToDouble(0xe62556c325a59470)),
               "mov ip, #38000\n"
               "movt ip, #9637\n"
               "vdup.32 d14, ip\n"
               "mov ip, #22211\n"
               "movt ip, #58917\n"
               "vmov.32 d14[1], ip\n");
  CLEANUP();
}

TEST(macro_assembler_PushRegisterList) {
  SETUP();

  // Allow the test to use all registers.
  UseScratchRegisterScope temps(&masm);
  temps.ExcludeAll();

  COMPARE_BOTH(Push(RegisterList(0x1111)), "push {r0,r4,r8,ip}\n");

  COMPARE_BOTH(Push(RegisterList(0x1fff)),
               "push {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip}\n");

  COMPARE_BOTH(Push(RegisterList(0x5fff)),
               "push {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip,lr}\n");

  COMPARE_A32(Push(ne, RegisterList(0x1fff)),
              "pushne {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip}\n");

  COMPARE_T32(Push(ne, RegisterList(0x1fff)),
              "beq 0x00000006\n"
              "push {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip}\n");

  COMPARE_A32(Push(RegisterList(sp)), "stmdb sp!, {sp}\n");

  // TODO: Clarify behaviour of MacroAssembler vs Assembler with respect to
  //       deprecated and unpredictable instructions. The tests reflect the
  //       current behaviour and will need to be updated.

  // Deprecated, but accepted:
  SHOULD_FAIL_TEST_A32(Push(RegisterList(pc)));
  // Whereas we don't accept the single-register version:
  MUST_FAIL_TEST_A32(Push(pc), "Unpredictable instruction.\n");

  // For T32, pushing the PC is allowed:
  COMPARE_T32(Push(pc), "push {pc}\n");

  // Accepted, but stores UNKNOWN value for the SP:
  SHOULD_FAIL_TEST_A32(Push(RegisterList(r0, sp)));

  // The following use the T1 and A1 encodings for T32 and A32 respectively, and
  // hence have different preferred disassembly.
  COMPARE_T32(Push(RegisterList(r0)), "push {r0}\n");
  COMPARE_A32(Push(RegisterList(r0)), "stmdb sp!, {r0}\n");
  COMPARE_T32(Push(RegisterList(r7)), "push {r7}\n");
  COMPARE_A32(Push(RegisterList(r7)), "stmdb sp!, {r7}\n");
  COMPARE_T32(Push(RegisterList(lr)), "push {lr}\n");
  COMPARE_A32(Push(RegisterList(lr)), "stmdb sp!, {lr}\n");

  // T2 and A1 encodings, with the same preferred disassembly:
  COMPARE_BOTH(Push(RegisterList(r8)), "stmdb sp!, {r8}\n");

  // Cannot push the sp and pc in T32 when using a register list.
  MUST_FAIL_TEST_T32(Push(RegisterList(sp)),
                     "Ill-formed 'push' instruction.\n");
  MUST_FAIL_TEST_T32(Push(RegisterList(pc)),
                     "Ill-formed 'push' instruction.\n");

  CLEANUP();
}

TEST(macro_assembler_PopRegisterList) {
  SETUP();

  // Allow the test to use all registers.
  UseScratchRegisterScope temps(&masm);
  temps.ExcludeAll();

  COMPARE_BOTH(Pop(RegisterList(0x1111)), "pop {r0,r4,r8,ip}\n");

  COMPARE_BOTH(Pop(RegisterList(0x1fff)),
               "pop {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip}\n");

  COMPARE_BOTH(Pop(RegisterList(0x5fff)),
               "pop {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip,lr}\n");

  COMPARE_A32(Pop(ne, RegisterList(0x1fff)),
              "popne {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip}\n");

  COMPARE_T32(Pop(ne, RegisterList(0x1fff)),
              "beq 0x00000006\n"
              "pop {r0,r1,r2,r3,r4,r5,r6,r7,r8,r9,r10,r11,ip}\n");

  // TODO: Accepted, but value of SP after the instruction is UNKNOWN:
  SHOULD_FAIL_TEST_A32(Pop(RegisterList(sp)));

  // Cannot pop the sp in T32 when using a register list.
  MUST_FAIL_TEST_T32(Pop(RegisterList(sp)), "Ill-formed 'pop' instruction.\n");

  // The following use the T1 and A1 encodings for T32 and A32 respectively, and
  // hence have different preferred disassembly.
  COMPARE_T32(Pop(RegisterList(pc)), "pop {pc}\n");
  COMPARE_A32(Pop(RegisterList(pc)), "ldm sp!, {pc}\n");
  COMPARE_T32(Pop(RegisterList(r0)), "pop {r0}\n");
  COMPARE_A32(Pop(RegisterList(r0)), "ldm sp!, {r0}\n");
  COMPARE_T32(Pop(RegisterList(r7)), "pop {r7}\n");
  COMPARE_A32(Pop(RegisterList(r7)), "ldm sp!, {r7}\n");

  // T2 and A1 encodings, with the same preferred disassembly:
  COMPARE_BOTH(Pop(RegisterList(r8)), "ldm sp!, {r8}\n");
  COMPARE_BOTH(Pop(RegisterList(lr)), "ldm sp!, {lr}\n");

  // TODO: Pushing both the lr and pc should not be allowed by the
  //       MacroAssembler (deprecated for A32, for T32 they shouldn't both
  //       be in the list).
  SHOULD_FAIL_TEST_BOTH(Pop(RegisterList(lr, pc)));

  CLEANUP();
}


TEST(macro_assembler_unpredictable) {
  SETUP();

  // ADC, ADCS (immediate).
  COMPARE_A32(Adc(pc, r0, 1), "adc pc, r0, #1\n");
  COMPARE_A32(Adc(r0, pc, 1), "adc r0, pc, #1\n");
  MUST_FAIL_TEST_T32(Adc(pc, r0, 1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adc(r0, pc, 1), "Unpredictable instruction.\n");
  COMPARE_A32(Adcs(pc, r0, 1), "adcs pc, r0, #1\n");
  COMPARE_A32(Adcs(r0, pc, 1), "adcs r0, pc, #1\n");
  MUST_FAIL_TEST_T32(Adcs(pc, r0, 1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adcs(r0, pc, 1), "Unpredictable instruction.\n");

  // ADC, ADCS (register).
  COMPARE_A32(Adc(pc, r0, r1), "adc pc, r0, r1\n");
  COMPARE_A32(Adc(r0, pc, r1), "adc r0, pc, r1\n");
  COMPARE_A32(Adc(r0, r1, pc), "adc r0, r1, pc\n");
  MUST_FAIL_TEST_T32(Adc(pc, r0, r1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adc(r0, pc, r1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adc(r0, r1, pc), "Unpredictable instruction.\n");
  COMPARE_A32(Adcs(pc, r0, r1), "adcs pc, r0, r1\n");
  COMPARE_A32(Adcs(r0, pc, r1), "adcs r0, pc, r1\n");
  COMPARE_A32(Adcs(r0, r1, pc), "adcs r0, r1, pc\n");
  MUST_FAIL_TEST_T32(Adcs(pc, r0, r1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adcs(r0, pc, r1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adcs(r0, r1, pc), "Unpredictable instruction.\n");

  // ADC, ADCS (register-shifted register).
  MUST_FAIL_TEST_A32(Adc(pc, r0, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adc(r0, pc, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adc(r0, r1, Operand(pc, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adc(r0, r1, Operand(r2, LSL, pc)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adcs(pc, r0, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adcs(r0, pc, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adcs(r0, r1, Operand(pc, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adcs(r0, r1, Operand(r2, LSL, pc)),
                     "Unpredictable instruction.\n");

  // ADD (immediate, to PC).
  COMPARE_A32(Add(r0, pc, 1), "adr r0, 0x00000009\n");
  COMPARE_T32(Add(r0, pc, 1), "adr r0, 0x00000005\n");
  COMPARE_A32(Add(pc, pc, 1), "adr pc, 0x00000009\n");
  MUST_FAIL_TEST_T32(Add(pc, pc, 1), "Unpredictable instruction.\n");

  // ADD, ADDS (immediate).
  COMPARE_A32(Add(pc, r0, 1), "add pc, r0, #1\n");
  MUST_FAIL_TEST_T32(Add(pc, r0, 1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Add(pc, r0, 0x123), "Unpredictable instruction.\n");
  COMPARE_A32(Adds(pc, r0, 1), "adds pc, r0, #1\n");
  COMPARE_A32(Adds(r0, pc, 1), "adds r0, pc, #1\n");
  // TODO: Try to make these error messages more consistent.
  MUST_FAIL_TEST_T32(Adds(r0, pc, 1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adds(r0, pc, 0x123), "Ill-formed 'adds' instruction.\n");

  // ADD, ADDS (register).
  COMPARE_A32(Add(pc, r0, r1), "add pc, r0, r1\n");
  COMPARE_A32(Add(r0, pc, r1), "add r0, pc, r1\n");
  COMPARE_A32(Add(r0, r1, pc), "add r0, r1, pc\n");
  COMPARE_T32(Add(r0, r0, pc), "add r0, pc\n");
  COMPARE_T32(Add(pc, pc, r0), "add pc, r0\n");
  MUST_FAIL_TEST_T32(Add(pc, pc, pc), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Add(pc, r0, r1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Add(r0, pc, r1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Add(r0, r1, pc), "Unpredictable instruction.\n");
  COMPARE_A32(Adds(pc, r0, r1), "adds pc, r0, r1\n");
  COMPARE_A32(Adds(r0, pc, r1), "adds r0, pc, r1\n");
  COMPARE_A32(Adds(r0, r1, pc), "adds r0, r1, pc\n");
  MUST_FAIL_TEST_T32(Adds(r0, pc, r1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Adds(r0, r1, pc), "Unpredictable instruction.\n");

  // ADD, ADDS (register-shifted register)
  MUST_FAIL_TEST_A32(Add(pc, r0, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Add(r0, pc, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Add(r0, r1, Operand(pc, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Add(r0, r1, Operand(r2, LSL, pc)),
                     "Unpredictable instruction.\n");
  COMPARE_A32(Add(pc, sp, 1), "add pc, sp, #1\n");
  MUST_FAIL_TEST_T32(Add(pc, sp, 1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adds(pc, r0, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adds(r0, pc, Operand(r1, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adds(r0, r1, Operand(pc, LSL, r2)),
                     "Unpredictable instruction.\n");
  MUST_FAIL_TEST_A32(Adds(r0, r1, Operand(r2, LSL, pc)),
                     "Unpredictable instruction.\n");

  // ADD, ADDS (SP plus immediate).
  COMPARE_A32(Add(pc, sp, 1), "add pc, sp, #1\n");
  MUST_FAIL_TEST_T32(Add(pc, sp, 1), "Unpredictable instruction.\n");
  COMPARE_A32(Adds(pc, sp, 1), "adds pc, sp, #1\n");
  MUST_FAIL_TEST_T32(Adds(pc, sp, 1), "Ill-formed 'adds' instruction.\n");

  // ADD, ADDS (SP plus register).
  COMPARE_A32(Add(pc, sp, r0), "add pc, sp, r0\n");
  MUST_FAIL_TEST_T32(Add(pc, sp, r0), "Unpredictable instruction.\n");
  COMPARE_A32(Add(r0, sp, pc), "add r0, sp, pc\n");
  MUST_FAIL_TEST_T32(Add(r0, sp, pc), "Unpredictable instruction.\n");
  COMPARE_BOTH(Add(pc, sp, pc), "add pc, sp, pc\n");
  COMPARE_BOTH(Add(sp, sp, pc), "add sp, pc\n");
  COMPARE_A32(Adds(pc, sp, r0), "adds pc, sp, r0\n");
  MUST_FAIL_TEST_T32(Adds(pc, sp, r0), "Ill-formed 'adds' instruction.\n");
  COMPARE_A32(Adds(r0, sp, pc), "adds r0, sp, pc\n");
  MUST_FAIL_TEST_T32(Adds(r0, sp, pc), "Unpredictable instruction.\n");
  COMPARE_A32(Adds(pc, sp, pc), "adds pc, sp, pc\n");
  MUST_FAIL_TEST_T32(Adds(pc, sp, pc), "Ill-formed 'adds' instruction.\n");
  COMPARE_A32(Adds(sp, sp, pc), "adds sp, pc\n");
  MUST_FAIL_TEST_T32(Adds(sp, sp, pc), "Unpredictable instruction.\n");

  // ADR.
  {
    Literal<uint32_t> literal(0x12345678);
    // The address is 0x4 and not 0x0 because of the branch over the literal.
    // TODO: Consider disallowing this instruction.
    COMPARE_A32(Adr(pc, &literal), "adr pc, 0x00000004\n");
    MUST_FAIL_TEST_T32(Adr(pc, &literal), "Unpredictable instruction.\n");
  }

  // CLZ.
  MUST_FAIL_TEST_BOTH(Clz(pc, r0), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Clz(r0, pc), "Unpredictable instruction.\n");

  // MOV, MOVS (immediate).
  COMPARE_A32(Mov(pc, 1), "mov pc, #1\n");
  MUST_FAIL_TEST_T32(Mov(pc, 1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_T32(Mov(pc, 0xfff), "Unpredictable instruction.\n");
  COMPARE_A32(Mov(pc, 0xf000), "mov pc, #61440\n");
  MUST_FAIL_TEST_T32(Mov(pc, 0xf000), "Unpredictable instruction.\n");
  COMPARE_A32(Movs(pc, 1), "movs pc, #1\n");
  MUST_FAIL_TEST_T32(Movs(pc, 1), "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Movs(pc, 0xfff), "Ill-formed 'movs' instruction.\n");
  COMPARE_A32(Movs(pc, 0xf000), "movs pc, #61440\n");
  MUST_FAIL_TEST_T32(Movs(pc, 0xf000), "Unpredictable instruction.\n");

  // MOV, MOVS (register).
  COMPARE_BOTH(Mov(pc, r0), "mov pc, r0\n");
  COMPARE_BOTH(Mov(r0, pc), "mov r0, pc\n");
  MUST_FAIL_TEST_BOTH(Movs(pc, r0), "Unpredictable instruction.\n");
  COMPARE_A32(Movs(r0, pc), "movs r0, pc\n");
  MUST_FAIL_TEST_T32(Movs(r0, pc), "Unpredictable instruction.\n");

  // MOV, MOVS (register-shifted register).
  MUST_FAIL_TEST_BOTH(Mov(pc, Operand(r0, ASR, r1)),
                      "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Mov(r0, Operand(pc, ASR, r1)),
                      "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Mov(r0, Operand(r1, ASR, pc)),
                      "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Movs(pc, Operand(r0, ASR, r1)),
                      "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Movs(r0, Operand(pc, ASR, r1)),
                      "Unpredictable instruction.\n");
  MUST_FAIL_TEST_BOTH(Movs(r0, Operand(r1, ASR, pc)),
                      "Unpredictable instruction.\n");

  CLEANUP();
}


TEST(macro_assembler_pc_rel_A32) {
  SETUP();
  // Simple cases alias adr.
  COMPARE_A32(Add(r0, pc, -8), "adr r0, 0x00000000\n");
  COMPARE_A32(Add(r0, pc, 255), "adr r0, 0x00000107\n");
  COMPARE_A32(Add(r0, pc, 256), "adr r0, 0x00000108\n");
  COMPARE_A32(Add(r0, pc, 1024), "adr r0, 0x00000408\n");
  COMPARE_A32(Add(r0, pc, -9), "adr r0, 0xffffffff\n");
  COMPARE_A32(Add(r0, pc, -1024), "adr r0, 0xfffffc08\n");
  COMPARE_A32(Add(r0, pc, UINT32_C(0x80000000)), "adr r0, 0x80000008\n");
  COMPARE_A32(Add(r0, pc, -0x7fffffff), "adr r0, 0x80000009\n");

  COMPARE_A32(Sub(r0, pc, 8), "adr r0, 0x00000000\n");
  COMPARE_A32(Sub(r0, pc, -255), "adr r0, 0x00000107\n");
  COMPARE_A32(Sub(r0, pc, -256), "adr r0, 0x00000108\n");
  COMPARE_A32(Sub(r0, pc, -1024), "adr r0, 0x00000408\n");
  COMPARE_A32(Sub(r0, pc, 9), "adr r0, 0xffffffff\n");
  COMPARE_A32(Sub(r0, pc, 1024), "adr r0, 0xfffffc08\n");
  COMPARE_A32(Sub(r0, pc, UINT32_C(0x80000000)), "adr r0, 0x80000008\n");
  COMPARE_A32(Sub(r0, pc, -0x7fffffff), "adr r0, 0x80000007\n");

  // Cases out of range.
  // Only negative offsets are supported, because the proper behaviour for
  // positive offsets is not clear.
  MUST_FAIL_TEST_A32(Add(r0, pc, 1025), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_A32(Add(r0, pc, 0xffff), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_A32(Add(r0, pc, 0x10001), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_A32(Add(r0, pc, 0x12345678),
                     "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_A32(Add(r0, pc, 0x7fffffff),
                     "Ill-formed 'add' instruction.\n");
  COMPARE_A32(Add(r0, pc, -1025),
              "adr r0, 0x00000007\n"
              "sub r0, #1024\n");
  COMPARE_A32(Add(r0, pc, -0xffff),
              "adr r0, 0xffffff09\n"
              "sub r0, #65280\n");
  COMPARE_A32(Add(r0, pc, -0x10001),
              "adr r0, 0x00000007\n"
              "sub r0, #65536\n");
  COMPARE_A32(Add(r0, pc, -0x2345678),
              "adr r0, 0xfffffd90\n"
              "sub r0, #21504\n"
              "sub r0, #36962304\n");
  COMPARE_A32(Add(r0, pc, -0x12345678),
              "adr r0, 0xfffffd90\n"
              "mov ip, #21504\n"
              "movt ip, #4660\n"
              "sub r0, ip\n");

  MUST_FAIL_TEST_A32(Sub(r0, pc, -1025), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_A32(Sub(r0, pc, -0xffff), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_A32(Sub(r0, pc, -0x10001), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_A32(Sub(r0, pc, -0x12345678),
                     "Ill-formed 'add' instruction.\n");
  COMPARE_A32(Sub(r0, pc, 1025),
              "adr r0, 0x00000007\n"
              "sub r0, #1024\n");
  COMPARE_A32(Sub(r0, pc, 0xffff),
              "adr r0, 0xffffff09\n"
              "sub r0, #65280\n");
  COMPARE_A32(Sub(r0, pc, 0x10001),
              "adr r0, 0x00000007\n"
              "sub r0, #65536\n");
  COMPARE_A32(Sub(r0, pc, 0x2345678),
              "adr r0, 0xfffffd90\n"
              "sub r0, #21504\n"
              "sub r0, #36962304\n");
  COMPARE_A32(Sub(r0, pc, 0x12345678),
              "adr r0, 0xfffffd90\n"
              "mov ip, #21504\n"
              "movt ip, #4660\n"
              "sub r0, ip\n");
  COMPARE_A32(Sub(r0, pc, 0x7fffffff),
              "adr r0, 0xffffff09\n"
              "add r0, #256\n"
              "add r0, #2147483648\n");

  CLEANUP();
}


TEST(macro_assembler_pc_rel_T32) {
  SETUP();
  // Simple cases alias adr.
  COMPARE_T32(Add(r0, pc, -4), "adr r0, 0x00000000\n");     // T1
  COMPARE_T32(Add(r0, pc, 1020), "adr r0, 0x00000400\n");   // T1
  COMPARE_T32(Add(r0, pc, -5), "adr r0, 0xffffffff\n");     // T2
  COMPARE_T32(Add(r0, pc, -4095), "adr r0, 0xfffff005\n");  // T2
  COMPARE_T32(Add(r0, pc, -3), "adr r0, 0x00000001\n");     // T3
  COMPARE_T32(Add(r0, pc, 1021), "adr r0, 0x00000401\n");   // T3
  COMPARE_T32(Add(r0, pc, 1019), "adr r0, 0x000003ff\n");   // T3
  COMPARE_T32(Add(r0, pc, 4095), "adr r0, 0x00001003\n");   // T3

  COMPARE_T32(Sub(r0, pc, 4), "adr r0, 0x00000000\n");      // T1
  COMPARE_T32(Sub(r0, pc, -1020), "adr r0, 0x00000400\n");  // T1
  COMPARE_T32(Sub(r0, pc, 5), "adr r0, 0xffffffff\n");      // T2
  COMPARE_T32(Sub(r0, pc, 4095), "adr r0, 0xfffff005\n");   // T2
  COMPARE_T32(Sub(r0, pc, 3), "adr r0, 0x00000001\n");      // T3
  COMPARE_T32(Sub(r0, pc, -1021), "adr r0, 0x00000401\n");  // T3
  COMPARE_T32(Sub(r0, pc, -1019), "adr r0, 0x000003ff\n");  // T3
  COMPARE_T32(Sub(r0, pc, -4095), "adr r0, 0x00001003\n");  // T3

  // Cases out of range.
  // Only negative offsets are supported, because the proper behaviour for
  // positive offsets is not clear.

  MUST_FAIL_TEST_T32(Add(r0, pc, 4096), "Unpredictable instruction.\n");

  // TODO: This case is incorrect; the instruction is unpredictable. The test
  // must be updated once the bug is fixed.
  COMPARE_T32(Add(r0, pc, -4096), "sub r0, pc, #4096\n");

  MUST_FAIL_TEST_T32(Add(r0, pc, 0xffff), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_T32(Add(r0, pc, 0x10002), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_T32(Add(r0, pc, 0x12345678),
                     "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_T32(Add(r0, pc, 0x7fffffff),
                     "Ill-formed 'add' instruction.\n");
  COMPARE_T32(Add(r0, pc, -0x12345678),
              "mov r0, pc\n"
              "mov ip, #22136\n"
              "movt ip, #4660\n"
              "sub r0, ip\n");
  COMPARE_T32(Add(r0, pc, -0x7fffffff),
              "mov r0, pc\n"
              "add r0, #1\n"
              "add r0, #2147483648\n");

  // TODO: This test aborts in the Assembler (with unpredictable instruction
  // errors) before the MacroAssembler gets a chance to do something
  // predictable.
  // COMPARE_T32(Sub(r0, pc, -4096), "mov r0, pc\n"
  //                                 "add r0, #4096\n");

  // TODO: This case is incorrect; the instruction is unpredictable. The test
  // must be updated once the bug is fixed.
  COMPARE_T32(Sub(r0, pc, 4096), "sub r0, pc, #4096\n");

  MUST_FAIL_TEST_T32(Sub(r0, pc, -0xffff), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_T32(Sub(r0, pc, -0x10002), "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_T32(Sub(r0, pc, -0x12345678),
                     "Ill-formed 'add' instruction.\n");
  MUST_FAIL_TEST_T32(Sub(r0, pc, -0x7fffffff),
                     "Ill-formed 'add' instruction.\n");
  COMPARE_T32(Sub(r0, pc, 0x12345678),
              "mov r0, pc\n"
              "mov ip, #22136\n"
              "movt ip, #4660\n"
              "sub r0, ip\n");
  COMPARE_T32(Sub(r0, pc, 0x7fffffff),
              "mov r0, pc\n"
              "add r0, #1\n"
              "add r0, #2147483648\n");
  CLEANUP();
}


TEST(macro_assembler_unsupported) {
  SETUP();

  MUST_FAIL_TEST_BOTH(Sxtab(r0, r1, Operand(r2, ROR, 1)),
                      "Ill-formed 'sxtab' instruction.\n");
  MUST_FAIL_TEST_BOTH(Sxtab16(r0, r1, Operand(r0, ASR, 2)),
                      "Ill-formed 'sxtab16' instruction.\n");
  MUST_FAIL_TEST_BOTH(Sxtah(r0, r1, Operand(r0, LSL, r1)),
                      "Ill-formed 'sxtah' instruction.\n");
  MUST_FAIL_TEST_BOTH(Uxtab(r0, r1, Operand(r0, LSR, r2)),
                      "Ill-formed 'uxtab' instruction.\n");
  MUST_FAIL_TEST_BOTH(Uxtab16(r0, r1, Operand(r0, ROR, 1)),
                      "Ill-formed 'uxtab16' instruction.\n");
  MUST_FAIL_TEST_BOTH(Uxtah(r0, r1, Operand(r0, ASR, 2)),
                      "Ill-formed 'uxtah' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pkhbt(r0, r1, Operand(r0, LSL, r1)),
                      "Ill-formed 'pkhbt' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pkhtb(r0, r1, Operand(r0, LSR, r2)),
                      "Ill-formed 'pkhtb' instruction.\n");

  MUST_FAIL_TEST_BOTH(Pld(MemOperand(r0, 1, PreIndex)),
                      "Ill-formed 'pld' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pldw(MemOperand(r0, 1, PostIndex)),
                      "Ill-formed 'pldw' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pli(MemOperand(r0, 1, PreIndex)),
                      "Ill-formed 'pli' instruction.\n");

  MUST_FAIL_TEST_BOTH(Pld(MemOperand(r0, r0, PreIndex)),
                      "Ill-formed 'pld' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pldw(MemOperand(r0, r1, PostIndex)),
                      "Ill-formed 'pldw' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pli(MemOperand(r0, r2, PreIndex)),
                      "Ill-formed 'pli' instruction.\n");

  MUST_FAIL_TEST_BOTH(Pld(MemOperand(r0, r0, LSL, 1, PreIndex)),
                      "Ill-formed 'pld' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pldw(MemOperand(r0, r1, LSR, 2, PostIndex)),
                      "Ill-formed 'pldw' instruction.\n");
  MUST_FAIL_TEST_BOTH(Pli(MemOperand(r0, r2, ASR, 3, PreIndex)),
                      "Ill-formed 'pli' instruction.\n");

  MUST_FAIL_TEST_BOTH(Lda(r0, MemOperand(r0, 1)),
                      "Ill-formed 'lda' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldab(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldab' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldaex(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldaex' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldaexb(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldaexb' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldaexh(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldaexh' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldah(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldah' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldrex(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldrex' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldrexb(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldrexb' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldrexh(r0, MemOperand(r0, 1)),
                      "Ill-formed 'ldrexh' instruction.\n");
  MUST_FAIL_TEST_BOTH(Stl(r0, MemOperand(r0, 1)),
                      "Ill-formed 'stl' instruction.\n");
  MUST_FAIL_TEST_BOTH(Stlb(r0, MemOperand(r0, 1)),
                      "Ill-formed 'stlb' instruction.\n");
  MUST_FAIL_TEST_BOTH(Stlh(r0, MemOperand(r0, 1)),
                      "Ill-formed 'stlh' instruction.\n");

  MUST_FAIL_TEST_BOTH(Ldaexd(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'ldaexd' instruction.\n");
  MUST_FAIL_TEST_BOTH(Ldrexd(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'ldrexd' instruction.\n");
  MUST_FAIL_TEST_BOTH(Stlex(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'stlex' instruction.\n");
  MUST_FAIL_TEST_BOTH(Stlexb(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'stlexb' instruction.\n");
  MUST_FAIL_TEST_BOTH(Stlexh(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'stlexh' instruction.\n");
  MUST_FAIL_TEST_BOTH(Strex(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'strex' instruction.\n");
  MUST_FAIL_TEST_BOTH(Strexb(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'strexb' instruction.\n");
  MUST_FAIL_TEST_BOTH(Strexh(r0, r1, MemOperand(r0, 1)),
                      "Ill-formed 'strexh' instruction.\n");

  MUST_FAIL_TEST_BOTH(Stlexd(r0, r1, r2, MemOperand(r0, 1)),
                      "Ill-formed 'stlexd' instruction.\n");
  MUST_FAIL_TEST_BOTH(Strexd(r0, r1, r2, MemOperand(r0, 1)),
                      "Ill-formed 'strexd' instruction.\n");

  CLEANUP();
}

TEST(macro_assembler_Vmov_neon_immediate) {
  SETUP();

  // Move 8, 16 and 32-bit immediates into D registers, duplicated across the
  // destination.
  COMPARE_BOTH(Vmov(I8, d0, 0xac), "vmov.i8 d0, #172\n");

  COMPARE_BOTH(Vmov(I16, d0, 0xa4), "vmov.i16 d0, #164\n");
  COMPARE_BOTH(Vmov(I16, d0, 0x9797), "vmov.i8 d0, #151\n");
  COMPARE_BOTH(Vmov(I16, d0, 0x9ef6),
               "mov ip, #40694\n"
               "vdup.16 d0, ip\n");

  COMPARE_BOTH(Vmov(I32, d0, 0x6d0000), "vmov.i32 d0, #7143424\n");
  COMPARE_BOTH(Vmov(I32, d0, 0x15ffffff), "vmvn.i32 d0, #3925868544\n");
  COMPARE_BOTH(Vmov(I32, d0, 0x74747474), "vmov.i8 d0, #116\n");
  COMPARE_BOTH(Vmov(I32, d0, 0xff0000ff), "vmov.i64 d0, #0xff0000ffff0000ff\n");
  COMPARE_BOTH(Vmov(I32, d0, 0x1ecb9ef6),
               "mov ip, #40694\n"
               "movt ip, #7883\n"
               "vdup.32 d0, ip\n");
  COMPARE_BOTH(Vmov(I32, d0, 0x006d0000), "vmov.i32 d0, #7143424\n");
  COMPARE_BOTH(Vmov(I32, d0, 0x00004da6),
               "mov ip, #19878\n"
               "vdup.32 d0, ip\n");
  COMPARE_BOTH(Vmov(I32, d0, 0xffffff55), "vmvn.i32 d0, #170\n");
  COMPARE_BOTH(Vmov(I32, d0, 0xffff55ff), "vmvn.i32 d0, #43520\n");
  COMPARE_BOTH(Vmov(I32, d0, 0xff55ffff), "vmvn.i32 d0, #11141120\n");

  COMPARE_BOTH(Vmov(I64, d0, UINT64_C(0xa5a5a5a5a5a5a5a5)),
               "vmov.i8 d0, #165\n");
  COMPARE_BOTH(Vmov(I64, d0, UINT64_C(0x0a01248315ffffff)),
               "mvn ip, #3925868544\n"
               "vdup.32 d0, ip\n"
               "mov ip, #9347\n"
               "movt ip, #2561\n"
               "vmov.32 d0[1], ip\n");
  COMPARE_BOTH(Vmov(I64, d0, UINT64_C(0x6fe1a7a779e33af2)),
               "mov ip, #15090\n"
               "movt ip, #31203\n"
               "vdup.32 d0, ip\n"
               "mov ip, #42919\n"
               "movt ip, #28641\n"
               "vmov.32 d0[1], ip\n");
  COMPARE_BOTH(Vmov(I64, d0, UINT64_C(0x2efa8b440000c1da)),
               "mov ip, #49626\n"
               "vdup.32 d0, ip\n"
               "mov ip, #35652\n"
               "movt ip, #12026\n"
               "vmov.32 d0[1], ip\n");
  COMPARE_BOTH(Vmov(I64, d0, UINT64_C(0x00008bb75c3036fd)),
               "mov ip, #14077\n"
               "movt ip, #23600\n"
               "vdup.32 d0, ip\n"
               "mov ip, #35767\n"
               "vmov.32 d0[1], ip\n");

  COMPARE_BOTH(Vmov(F32, d0, 0.5), "vmov.f32 d0, #0.5\n");
  COMPARE_BOTH(Vmov(F32, d0, 1.1),
               "mov ip, #52429\n"
               "movt ip, #16268\n"
               "vdup.32 d0, ip\n");

  COMPARE_T32(Vmov(I64, d0, UINT64_C(0x2fff2fff3e2869e7)),
              "mov ip, #27111\n"
              "movt ip, #15912\n"
              "vdup.32 d0, ip\n"
              "mvn ip, #3489714176\n"
              "vmov.32 d0[1], ip\n");

  COMPARE_A32(Vmov(I32, d0, 0x0ffffffa),
              "mvn ip, #4026531845\n"
              "vdup.32 d0, ip\n");
  COMPARE_A32(Vmov(I64, d0, UINT64_C(0x65ffffff16a0ef46)),
              "mov ip, #61254\n"
              "movt ip, #5792\n"
              "vdup.32 d0, ip\n"
              "mvn ip, #2583691264\n"
              "vmov.32 d0[1], ip\n");

  // Move 8, 16 and 32-bit immediates into Q registers, duplicated across the
  // destination.
  COMPARE_BOTH(Vmov(I8, q0, 0xac), "vmov.i8 q0, #172\n");

  COMPARE_BOTH(Vmov(I16, q0, 0xa4), "vmov.i16 q0, #164\n");
  COMPARE_BOTH(Vmov(I16, q0, 0x9797), "vmov.i8 q0, #151\n");
  COMPARE_BOTH(Vmov(I16, q0, 0x9ef6),
               "mov ip, #40694\n"
               "vdup.16 q0, ip\n");

  COMPARE_BOTH(Vmov(I32, q0, 0x6d0000), "vmov.i32 q0, #7143424\n");
  COMPARE_BOTH(Vmov(I32, q0, 0x15ffffff), "vmvn.i32 q0, #3925868544\n");
  COMPARE_BOTH(Vmov(I32, q0, 0x74747474), "vmov.i8 q0, #116\n");
  COMPARE_BOTH(Vmov(I32, q0, 0xff0000ff), "vmov.i64 q0, #0xff0000ffff0000ff\n");
  COMPARE_BOTH(Vmov(I32, q0, 0x1ecb9ef6),
               "mov ip, #40694\n"
               "movt ip, #7883\n"
               "vdup.32 q0, ip\n");
  COMPARE_BOTH(Vmov(I32, q0, 0x006d0000), "vmov.i32 q0, #7143424\n");
  COMPARE_BOTH(Vmov(I32, q0, 0x00004da6),
               "mov ip, #19878\n"
               "vdup.32 q0, ip\n");

  COMPARE_BOTH(Vmov(I64, q0, UINT64_C(0xa5a5a5a5a5a5a5a5)),
               "vmov.i8 q0, #165\n");
  COMPARE_BOTH(Vmov(I64, q0, UINT64_C(0x0a01248315ffffff)),
               "mvn ip, #3925868544\n"
               "vdup.32 q0, ip\n"
               "mov ip, #9347\n"
               "movt ip, #2561\n"
               "vmov.32 d0[1], ip\n"
               "vmov.f64 d1, d0\n");
  COMPARE_BOTH(Vmov(I64, q0, UINT64_C(0x6fe1a7a779e33af2)),
               "mov ip, #15090\n"
               "movt ip, #31203\n"
               "vdup.32 q0, ip\n"
               "mov ip, #42919\n"
               "movt ip, #28641\n"
               "vmov.32 d0[1], ip\n"
               "vmov.f64 d1, d0\n");
  COMPARE_BOTH(Vmov(I64, q0, UINT64_C(0x2efa8b440000c1da)),
               "mov ip, #49626\n"
               "vdup.32 q0, ip\n"
               "mov ip, #35652\n"
               "movt ip, #12026\n"
               "vmov.32 d0[1], ip\n"
               "vmov.f64 d1, d0\n");
  COMPARE_BOTH(Vmov(I64, q0, UINT64_C(0x00008bb75c3036fd)),
               "mov ip, #14077\n"
               "movt ip, #23600\n"
               "vdup.32 q0, ip\n"
               "mov ip, #35767\n"
               "vmov.32 d0[1], ip\n"
               "vmov.f64 d1, d0\n");

  COMPARE_BOTH(Vmov(F32, q0, 0.5), "vmov.f32 q0, #0.5\n");
  COMPARE_BOTH(Vmov(F32, q0, 1.1),
               "mov ip, #52429\n"
               "movt ip, #16268\n"
               "vdup.32 q0, ip\n");
  COMPARE_BOTH(Vmov(F64, q0, 0.5),
               "vmov.f64 d0, #0.5\n"
               "vmov.f64 d1, d0\n");
  COMPARE_BOTH(Vmov(F64, q0, 1.1),
               "mov ip, #39322\n"
               "movt ip, #39321\n"
               "vdup.32 d0, ip\n"
               "mov ip, #39321\n"
               "movt ip, #16369\n"
               "vmov.32 d0[1], ip\n"
               "vmov.f64 d1, d0\n");

  COMPARE_T32(Vmov(I64, q0, UINT64_C(0x2fff2fff3e2869e7)),
              "mov ip, #27111\n"
              "movt ip, #15912\n"
              "vdup.32 q0, ip\n"
              "mvn ip, #3489714176\n"
              "vmov.32 d0[1], ip\n"
              "vmov.f64 d1, d0\n");

  COMPARE_A32(Vmov(I32, q0, 0x0ffffffa),
              "mvn ip, #4026531845\n"
              "vdup.32 q0, ip\n");
  COMPARE_A32(Vmov(I64, q0, UINT64_C(0x65ffffff16a0ef46)),
              "mov ip, #61254\n"
              "movt ip, #5792\n"
              "vdup.32 q0, ip\n"
              "mvn ip, #2583691264\n"
              "vmov.32 d0[1], ip\n"
              "vmov.f64 d1, d0\n");
  CLEANUP();
}

TEST(macro_assembler_T32_IT) {
  SETUP();

  // ADC (register) T1
  COMPARE_T32(Adc(eq, r0, r0, r1),
              "it eq\n"
              "adceq r0, r1\n");

  COMPARE_T32(Adc(eq, r0, r1, r2),
              "bne 0x00000006\n"
              "adc r0, r1, r2\n");

  // ADD (immediate) T1
  COMPARE_T32(Add(eq, r0, r1, 0x1),
              "it eq\n"
              "addeq r0, r1, #1\n");

  COMPARE_T32(Add(eq, r0, r1, 0x8),
              "bne 0x00000006\n"
              "add r0, r1, #8\n");

  // ADD (immediate) T2
  COMPARE_T32(Add(eq, r0, r0, 0xff),
              "it eq\n"
              "addeq r0, #255\n");

  // ADD (register) T1
  COMPARE_T32(Add(eq, r0, r1, r7),
              "it eq\n"
              "addeq r0, r1, r7\n");

  // ADD (register) T2
  COMPARE_T32(Add(eq, r5, r5, r8),
              "it eq\n"
              "addeq r5, r8\n");

  // ADD (SP plus immediate) T1
  COMPARE_T32(Add(eq, r7, sp, 1020),
              "it eq\n"
              "addeq r7, sp, #1020\n");

  COMPARE_T32(Add(eq, r7, sp, 1),
              "bne 0x00000006\n"
              "add r7, sp, #1\n");

  COMPARE_T32(Add(eq, r7, sp, 1024),
              "bne 0x00000006\n"
              "add r7, sp, #1024\n");

  COMPARE_T32(Add(eq, sp, sp, 32),
              "bne 0x00000004\n"
              "add sp, #32\n");

  // ADD (SP plus register) T1
  COMPARE_T32(Add(eq, r7, sp, r7),
              "it eq\n"
              "addeq r7, sp, r7\n");

  // ADD (SP plus register) T2
  COMPARE_T32(Add(eq, sp, sp, r10),
              "it eq\n"
              "addeq sp, r10\n");

  COMPARE_T32(Add(eq, r5, r5, sp),
              "bne 0x00000006\n"
              "add.w r5, sp\n");

  // AND (register) T1
  COMPARE_T32(And(eq, r7, r7, r0),
              "it eq\n"
              "andeq r7, r0\n");

  COMPARE_T32(And(eq, r8, r8, r0),
              "bne 0x00000006\n"
              "and r8, r0\n");

  // ASR (immediate) T2
  COMPARE_T32(Asr(eq, r0, r1, 16),
              "it eq\n"
              "asreq r0, r1, #16\n");

  COMPARE_T32(Asr(eq, r0, r1, 32),
              "it eq\n"
              "asreq r0, r1, #32\n");

  COMPARE_T32(Asr(eq, r0, r1, 0),
              "bne 0x0000000a\n"
              "mov r0, #0\n"
              "asr r0, r1, r0\n");

  // ASR (register) T1
  COMPARE_T32(Asr(eq, r7, r7, r3),
              "it eq\n"
              "asreq r7, r3\n");

  COMPARE_T32(Asr(eq, r8, r8, r3),
              "bne 0x00000006\n"
              "asr r8, r3\n");

  // BIC (register) T1
  COMPARE_T32(Bic(eq, r7, r7, r6),
              "it eq\n"
              "biceq r7, r6\n");

  COMPARE_T32(Bic(eq, r8, r8, r6),
              "bne 0x00000006\n"
              "bic r8, r6\n");

  Label l;
  __ Bind(&l);

  // BLX (register) T1
  COMPARE_T32(Blx(eq, lr),
              "it eq\n"
              "blxeq lr\n");
  COMPARE_T32(Blx(eq, &l),
              "bne 0x00000006\n"
              "blx 0x00000000\n");

  // BX (register) T1
  COMPARE_T32(Bx(eq, lr),
              "it eq\n"
              "bxeq lr\n");

  // CMN (register) T1
  COMPARE_T32(Cmn(eq, r0, r1),
              "it eq\n"
              "cmneq r0, r1\n");

  COMPARE_T32(Cmn(eq, r0, r8),
              "bne 0x00000006\n"
              "cmn r0, r8\n");

  // CMP (immediate) T1
  COMPARE_T32(Cmp(eq, r7, 0xff),
              "it eq\n"
              "cmpeq r7, #255\n");

  // CMP (register) T1
  COMPARE_T32(Cmp(eq, r6, r7),
              "it eq\n"
              "cmpeq r6, r7\n");

  // CMP (register) T2
  COMPARE_T32(Cmp(eq, r9, r10),
              "it eq\n"
              "cmpeq r9, r10\n");

  COMPARE_T32(Cmp(eq, r0, 0x100),
              "bne 0x00000006\n"
              "cmp r0, #256\n");

  // EOR (register) T1
  COMPARE_T32(Eor(eq, r0, r0, r7),
              "it eq\n"
              "eoreq r0, r7\n");

  COMPARE_T32(Eor(eq, r0, r0, 0x1),
              "bne 0x00000006\n"
              "eor r0, #0x1\n");

  // LDR (immediate) T1
  COMPARE_T32(Ldr(eq, r4, MemOperand(r7, 124)),
              "it eq\n"
              "ldreq r4, [r7, #124]\n");

  COMPARE_T32(Ldr(eq, r4, MemOperand(r7, 1)),
              "bne 0x00000006\n"
              "ldr r4, [r7, #1]\n");

  COMPARE_T32(Ldr(eq, r4, MemOperand(r7, 128)),
              "bne 0x00000006\n"
              "ldr r4, [r7, #128]\n");

  // LDR (immediate) T2
  COMPARE_T32(Ldr(eq, r4, MemOperand(sp, 1020)),
              "it eq\n"
              "ldreq r4, [sp, #1020]\n");

  COMPARE_T32(Ldr(eq, r4, MemOperand(sp, 1)),
              "bne 0x00000006\n"
              "ldr r4, [sp, #1]\n");

  COMPARE_T32(Ldr(eq, r4, MemOperand(sp, 1024)),
              "bne 0x00000006\n"
              "ldr r4, [sp, #1024]\n");

  // LDR (register) T1
  COMPARE_T32(Ldr(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "ldreq r5, [r6, r7]\n");

  COMPARE_T32(Ldr(eq, r5, MemOperand(r6, r8)),
              "bne 0x00000006\n"
              "ldr r5, [r6, r8]\n");

  // LDRB (immediate) T1
  COMPARE_T32(Ldrb(eq, r6, MemOperand(r7, 31)),
              "it eq\n"
              "ldrbeq r6, [r7, #31]\n");

  COMPARE_T32(Ldrb(eq, r6, MemOperand(r7, 32)),
              "bne 0x00000006\n"
              "ldrb r6, [r7, #32]\n");

  // LDRB (register) T1
  COMPARE_T32(Ldrb(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "ldrbeq r5, [r6, r7]\n");

  COMPARE_T32(Ldrb(eq, r6, MemOperand(r9)),
              "bne 0x00000006\n"
              "ldrb r6, [r9]\n");

  // LDRH (immediate) T1
  COMPARE_T32(Ldrh(eq, r6, MemOperand(r7, 62)),
              "it eq\n"
              "ldrheq r6, [r7, #62]\n");

  COMPARE_T32(Ldrh(eq, r6, MemOperand(r7, 64)),
              "bne 0x00000006\n"
              "ldrh r6, [r7, #64]\n");

  COMPARE_T32(Ldrh(eq, r6, MemOperand(r7, 1)),
              "bne 0x00000006\n"
              "ldrh r6, [r7, #1]\n");

  // LDRH (register) T1
  COMPARE_T32(Ldrh(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "ldrheq r5, [r6, r7]\n");

  COMPARE_T32(Ldrh(eq, r6, MemOperand(r9)),
              "bne 0x00000006\n"
              "ldrh r6, [r9]\n");

  // LDRSB (register) T1
  COMPARE_T32(Ldrsb(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "ldrsbeq r5, [r6, r7]\n");

  COMPARE_T32(Ldrsb(eq, r6, MemOperand(r9)),
              "bne 0x00000006\n"
              "ldrsb r6, [r9]\n");

  // LDRSH (register) T1
  COMPARE_T32(Ldrsh(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "ldrsheq r5, [r6, r7]\n");

  COMPARE_T32(Ldrsh(eq, r6, MemOperand(r9)),
              "bne 0x00000006\n"
              "ldrsh r6, [r9]\n");

  // LSL (immediate) T2
  COMPARE_T32(Lsl(eq, r0, r1, 16),
              "it eq\n"
              "lsleq r0, r1, #16\n");

  COMPARE_T32(Lsl(eq, r0, r1, 0),
              "bne 0x0000000a\n"
              "mov r0, #0\n"
              "lsl r0, r1, r0\n");

  COMPARE_T32(Lsl(eq, r0, r1, 32),
              "bne 0x0000000a\n"
              "mov r0, #32\n"
              "lsl r0, r1, r0\n");

  // LSL (register) T1
  COMPARE_T32(Lsl(eq, r7, r7, r3),
              "it eq\n"
              "lsleq r7, r3\n");

  COMPARE_T32(Lsl(eq, r8, r8, r3),
              "bne 0x00000006\n"
              "lsl r8, r3\n");

  // LSR (immediate) T2
  COMPARE_T32(Lsr(eq, r0, r1, 16),
              "it eq\n"
              "lsreq r0, r1, #16\n");

  COMPARE_T32(Lsr(eq, r0, r1, 32),
              "it eq\n"
              "lsreq r0, r1, #32\n");

  COMPARE_T32(Lsr(eq, r0, r1, 0),
              "bne 0x0000000a\n"
              "mov r0, #0\n"
              "lsr r0, r1, r0\n");

  // LSR (register) T1
  COMPARE_T32(Lsr(eq, r7, r7, r3),
              "it eq\n"
              "lsreq r7, r3\n");

  COMPARE_T32(Lsr(eq, r8, r8, r3),
              "bne 0x00000006\n"
              "lsr r8, r3\n");

  // MOV (immediate) T1
  COMPARE_T32(Mov(eq, r7, 0xff),
              "it eq\n"
              "moveq r7, #255\n");

  // MOV (register) T1
  COMPARE_T32(Mov(eq, r9, r8),
              "it eq\n"
              "moveq r9, r8\n");

  // MOV (register) T2
  COMPARE_T32(Mov(eq, r0, Operand(r1, LSR, 16)),
              "it eq\n"
              "lsreq r0, r1, #16\n");

  COMPARE_T32(Mov(eq, r0, Operand(r1, ROR, 16)),
              "bne 0x00000006\n"
              "ror r0, r1, #16\n");

  // MOV (register-shifted register) T1
  COMPARE_T32(Mov(eq, r0, Operand(r0, LSR, r1)),
              "it eq\n"
              "lsreq r0, r1\n");

  COMPARE_T32(Mov(eq, r0, Operand(r1, LSR, r2)),
              "bne 0x00000006\n"
              "lsr r0, r1, r2\n");

  // MUL (T1)
  COMPARE_T32(Mul(eq, r0, r1, r0),
              "it eq\n"
              "muleq r0, r1, r0\n");

  COMPARE_T32(Mul(eq, r0, r1, r2),
              "bne 0x00000006\n"
              "mul r0, r1, r2\n");

  // MVN (register) T1
  COMPARE_T32(Mvn(eq, r4, r6),
              "it eq\n"
              "mvneq r4, r6\n");

  COMPARE_T32(Mvn(eq, r8, r6),
              "bne 0x00000006\n"
              "mvn r8, r6\n");

  // ORR (register) T1
  COMPARE_T32(Orr(eq, r0, r0, r1),
              "it eq\n"
              "orreq r0, r1\n");

  COMPARE_T32(Orr(eq, r0, r1, r2),
              "bne 0x00000006\n"
              "orr r0, r1, r2\n");

  // ROR (register) T1
  COMPARE_T32(Ror(eq, r7, r7, r3),
              "it eq\n"
              "roreq r7, r3\n");

  COMPARE_T32(Ror(eq, r8, r8, r3),
              "bne 0x00000006\n"
              "ror r8, r3\n");

  COMPARE_T32(Ror(eq, r0, r1, 16),
              "bne 0x00000006\n"
              "ror r0, r1, #16\n");

  // RSB (immediate) T1
  COMPARE_T32(Rsb(eq, r0, r1, 0),
              "it eq\n"
              "rsbeq r0, r1, #0\n");

  COMPARE_T32(Rsb(eq, r0, r1, 1),
              "bne 0x00000006\n"
              "rsb r0, r1, #1\n");

  // SBC (register) T1
  COMPARE_T32(Sbc(eq, r0, r0, r1),
              "it eq\n"
              "sbceq r0, r1\n");

  COMPARE_T32(Sbc(eq, r0, r1, r2),
              "bne 0x00000006\n"
              "sbc r0, r1, r2\n");

  // STR (immediate) T1
  COMPARE_T32(Str(eq, r4, MemOperand(r7, 124)),
              "it eq\n"
              "streq r4, [r7, #124]\n");

  COMPARE_T32(Str(eq, r4, MemOperand(r7, 1)),
              "bne 0x00000006\n"
              "str r4, [r7, #1]\n");

  COMPARE_T32(Str(eq, r4, MemOperand(r7, 128)),
              "bne 0x00000006\n"
              "str r4, [r7, #128]\n");

  // STR (immediate) T2
  COMPARE_T32(Str(eq, r4, MemOperand(sp, 1020)),
              "it eq\n"
              "streq r4, [sp, #1020]\n");

  COMPARE_T32(Str(eq, r4, MemOperand(sp, 1)),
              "bne 0x00000006\n"
              "str r4, [sp, #1]\n");

  COMPARE_T32(Str(eq, r4, MemOperand(sp, 1024)),
              "bne 0x00000006\n"
              "str r4, [sp, #1024]\n");

  // STR (register) T1
  COMPARE_T32(Str(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "streq r5, [r6, r7]\n");

  COMPARE_T32(Str(eq, r5, MemOperand(r6, r8)),
              "bne 0x00000006\n"
              "str r5, [r6, r8]\n");

  // STRB (immediate) T1
  COMPARE_T32(Strb(eq, r6, MemOperand(r7, 31)),
              "it eq\n"
              "strbeq r6, [r7, #31]\n");

  COMPARE_T32(Strb(eq, r6, MemOperand(r7, 32)),
              "bne 0x00000006\n"
              "strb r6, [r7, #32]\n");

  // STRB (register) T1
  COMPARE_T32(Strb(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "strbeq r5, [r6, r7]\n");

  COMPARE_T32(Strb(eq, r6, MemOperand(r9)),
              "bne 0x00000006\n"
              "strb r6, [r9]\n");

  // STRH (immediate) T1
  COMPARE_T32(Strh(eq, r6, MemOperand(r7, 62)),
              "it eq\n"
              "strheq r6, [r7, #62]\n");

  COMPARE_T32(Strh(eq, r6, MemOperand(r7, 64)),
              "bne 0x00000006\n"
              "strh r6, [r7, #64]\n");

  COMPARE_T32(Strh(eq, r6, MemOperand(r7, 1)),
              "bne 0x00000006\n"
              "strh r6, [r7, #1]\n");

  // STRH (register) T1
  COMPARE_T32(Strh(eq, r5, MemOperand(r6, r7)),
              "it eq\n"
              "strheq r5, [r6, r7]\n");

  COMPARE_T32(Strh(eq, r6, MemOperand(r9)),
              "bne 0x00000006\n"
              "strh r6, [r9]\n");

  // SUB (immediate) T1
  COMPARE_T32(Sub(eq, r0, r1, 0x1),
              "it eq\n"
              "subeq r0, r1, #1\n");

  COMPARE_T32(Sub(eq, r0, r1, 0x8),
              "bne 0x00000006\n"
              "sub r0, r1, #8\n");

  // SUB (immediate) T2
  COMPARE_T32(Sub(eq, r0, r0, 0xff),
              "it eq\n"
              "subeq r0, #255\n");

  // SUB (register) T1
  COMPARE_T32(Sub(eq, r0, r1, r7),
              "it eq\n"
              "subeq r0, r1, r7\n");

  COMPARE_T32(Sub(eq, r5, r5, r8),
              "bne 0x00000006\n"
              "sub r5, r8\n");

  COMPARE_T32(Sub(eq, r7, sp, 1),
              "bne 0x00000006\n"
              "sub r7, sp, #1\n");

  COMPARE_T32(Sub(eq, pc, pc, 0),
              "bne 0x00000006\n"
              "sub pc, #0\n");

  // TST (register) T1
  COMPARE_T32(Tst(eq, r0, r1),
              "it eq\n"
              "tsteq r0, r1\n");

  COMPARE_T32(Tst(eq, r8, r9),
              "bne 0x00000006\n"
              "tst r8, r9\n");

  CLEANUP();
}


TEST(unbound_label) {
  SETUP();

#ifdef VIXL_DEBUG
  MUST_FAIL_TEST_BOTH_BLOCK({
    Label label;
    masm.B(&label);
  }, "Label used but not bound.\n")

  MUST_FAIL_TEST_BOTH_BLOCK({
    Label label;
    masm.B(eq, &label);
  }, "Label used but not bound.\n")

  MUST_FAIL_TEST_T32_BLOCK({
    Label label;
    masm.Cbz(r0, &label);
  }, "Label used but not bound.\n")

  MUST_FAIL_TEST_T32_BLOCK({
    Label label;
    masm.Cbnz(r1, &label);
  }, "Label used but not bound.\n")
#endif

  CLEANUP();
}


TEST(macro_assembler_AddressComputationHelper) {
  SETUP();

  // Simple cases: the address fits in the mask.
  COMPARE_A32(Ldr(r0, masm.MemOperandComputationHelper(r1, r1, 0xfff, 0xfff)),
              "ldr r0, [r1, #4095]\n");
  COMPARE_A32(Ldr(r0, masm.MemOperandComputationHelper(r1, r1, 1, 0xfff)),
              "ldr r0, [r1, #1]\n");
  COMPARE_A32(Ldr(r0, masm.MemOperandComputationHelper(r1, r1, 0, 0xfff)),
              "ldr r0, [r1]\n");

  // Similar, but the base register must be preserved. (This has no effect for
  // encodable cases.)
  COMPARE_A32(Ldr(r0, masm.MemOperandComputationHelper(r2, r1, 0xfff, 0xfff)),
              "ldr r0, [r1, #4095]\n");

  // Cases where the extra offset has to be aligned.
  COMPARE_A32(Vldr(d0, masm.MemOperandComputationHelper(r1, r1, 0x3fc, 0x3fc)),
              "vldr d0, [r1, #1020]\n");

  // Out-of-range offsets.
  COMPARE_A32(Ldr(r0, masm.MemOperandComputationHelper(r1, r1, 0x1000, 0xfff)),
              "add r1, #4096\n"
              "ldr r0, [r1]\n");
  COMPARE_A32(Ldr(r0, masm.MemOperandComputationHelper(r2, r1, 0x1000, 0xfff)),
              "add r2, r1, #4096\n"
              "ldr r0, [r2]\n");
  COMPARE_A32(Ldr(r0,
                  masm.MemOperandComputationHelper(r2, r1, 0xffffffff, 0xfff)),
              "sub r2, r1, #1\n"
              "ldr r0, [r2]\n");

  // TODO: Improve the code generation for these cases.

  COMPARE_A32(Ldr(r0,
                  masm.MemOperandComputationHelper(r2, r1, 0x12345678, 0xfff)),
              "mov r2, #20480\n"
              "movt r2, #4660\n"
              "add r2, r1, r2\n"
              "ldr r0, [r2, #1656]\n");
  COMPARE_A32(Ldr(r0,
                  masm.MemOperandComputationHelper(r2, r1, 0x7fffffff, 0xfff)),
              "sub r2, r1, #1\n"
              "sub r2, #2147483648\n"
              "ldr r0, [r2]\n");
  COMPARE_A32(Ldr(r0,
                  masm.MemOperandComputationHelper(r2, r1, 0xffcba000, 0xfff)),
              "sub r2, r1, #286720\n"
              "sub r2, #3145728\n"
              "ldr r0, [r2]\n");

  CLEANUP();
}


TEST(barriers) {
  SETUP();

  // DMB
  COMPARE_BOTH(Dmb(SY), "dmb sy\n");
  COMPARE_BOTH(Dmb(ST), "dmb st\n");
  COMPARE_BOTH(Dmb(ISH), "dmb ish\n");
  COMPARE_BOTH(Dmb(ISHST), "dmb ishst\n");
  COMPARE_BOTH(Dmb(NSH), "dmb nsh\n");
  COMPARE_BOTH(Dmb(NSHST), "dmb nshst\n");
  COMPARE_BOTH(Dmb(OSH), "dmb osh\n");
  COMPARE_BOTH(Dmb(OSHST), "dmb oshst\n");

  // DSB
  COMPARE_BOTH(Dsb(SY), "dsb sy\n");
  COMPARE_BOTH(Dsb(ST), "dsb st\n");
  COMPARE_BOTH(Dsb(ISH), "dsb ish\n");
  COMPARE_BOTH(Dsb(ISHST), "dsb ishst\n");
  COMPARE_BOTH(Dsb(NSH), "dsb nsh\n");
  COMPARE_BOTH(Dsb(NSHST), "dsb nshst\n");
  COMPARE_BOTH(Dsb(OSH), "dsb osh\n");
  COMPARE_BOTH(Dsb(OSHST), "dsb oshst\n");

  // ISB
  COMPARE_BOTH(Isb(SY), "isb sy\n");

  CLEANUP();
}


TEST(preloads) {
  // Smoke test for various pld/pli forms.
  SETUP();

  // PLD immediate
  COMPARE_BOTH(Pld(MemOperand(r0, 0)), "pld [r0]\n");
  COMPARE_BOTH(Pld(MemOperand(r1, 123)), "pld [r1, #123]\n");
  COMPARE_BOTH(Pld(MemOperand(r4, -123)), "pld [r4, #-123]\n");

  COMPARE_A32(Pld(MemOperand(r7, -4095)), "pld [r7, #-4095]\n");

  // PLDW immediate
  COMPARE_BOTH(Pldw(MemOperand(r0, 0)), "pldw [r0]\n");
  COMPARE_BOTH(Pldw(MemOperand(r1, 123)), "pldw [r1, #123]\n");
  COMPARE_BOTH(Pldw(MemOperand(r4, -123)), "pldw [r4, #-123]\n");

  COMPARE_A32(Pldw(MemOperand(r7, -4095)), "pldw [r7, #-4095]\n");

  // PLD register
  COMPARE_BOTH(Pld(MemOperand(r0, r1)), "pld [r0, r1]\n");
  COMPARE_BOTH(Pld(MemOperand(r0, r1, LSL, 1)), "pld [r0, r1, lsl #1]\n");

  COMPARE_A32(Pld(MemOperand(r0, r1, LSL, 20)), "pld [r0, r1, lsl #20]\n");

  // PLDW register
  COMPARE_BOTH(Pldw(MemOperand(r0, r1)), "pldw [r0, r1]\n");
  COMPARE_BOTH(Pldw(MemOperand(r0, r1, LSL, 1)), "pldw [r0, r1, lsl #1]\n");

  COMPARE_A32(Pldw(MemOperand(r0, r1, LSL, 20)), "pldw [r0, r1, lsl #20]\n");

  // PLD literal
  Label pld_label;
  COMPARE_A32(Pld(&pld_label);, "pld [pc, #-0]\n");
  COMPARE_T32(Pld(&pld_label);, "pld [pc, #-0]\n");
  __ Bind(&pld_label);

  // PLI immediate
  COMPARE_BOTH(Pli(MemOperand(r0, 0)), "pli [r0]\n");
  COMPARE_BOTH(Pli(MemOperand(r1, 123)), "pli [r1, #123]\n");
  COMPARE_BOTH(Pli(MemOperand(r4, -123)), "pli [r4, #-123]\n");

  COMPARE_A32(Pli(MemOperand(r7, -4095)), "pli [r7, #-4095]\n");

  // PLI register
  COMPARE_BOTH(Pli(MemOperand(r0, r1)), "pli [r0, r1]\n");
  COMPARE_BOTH(Pli(MemOperand(r0, r1, LSL, 1)), "pli [r0, r1, lsl #1]\n");

  COMPARE_A32(Pli(MemOperand(r0, r1, LSL, 20)), "pli [r0, r1, lsl #20]\n");

  // PLI literal
  Label pli_label;
  COMPARE_A32(Pli(&pli_label);, "pli [pc, #-0]\n");
  COMPARE_T32(Pli(&pli_label);, "pli [pc, #-0]\n");
  __ Bind(&pli_label);

  CLEANUP();
}


TEST(vmrs_vmsr) {
  SETUP();

  COMPARE_BOTH(Vmsr(FPSCR, r0), "vmsr FPSCR, r0\n");

  COMPARE_BOTH(Vmrs(RegisterOrAPSR_nzcv(r1.GetCode()), FPSCR),
               "vmrs r1, FPSCR\n");

  COMPARE_BOTH(Vmrs(RegisterOrAPSR_nzcv(pc.GetCode()), FPSCR),
               "vmrs APSR_nzcv, FPSCR\n");

  CLEANUP();
}


TEST(ldm_stm) {
  SETUP();

  // ldm/stm
  COMPARE_BOTH(Ldm(r0, NO_WRITE_BACK, RegisterList(r1)), "ldm r0, {r1}\n");

  COMPARE_BOTH(Ldm(r1, NO_WRITE_BACK, RegisterList(r2, r5, r9, r10)),
               "ldm r1, {r2,r5,r9,r10}\n");

  COMPARE_BOTH(Ldm(r0, WRITE_BACK, RegisterList(r1, r2)), "ldm r0!, {r1,r2}\n");

  COMPARE_BOTH(Stm(r1, NO_WRITE_BACK, RegisterList(r2, r5, r9, r10)),
               "stm r1, {r2,r5,r9,r10}\n");

  COMPARE_BOTH(Stm(r0, WRITE_BACK, RegisterList(r1, r2)), "stm r0!, {r1,r2}\n");

  // ldmda/stmda
  COMPARE_A32(Ldmda(r11, WRITE_BACK, RegisterList(r0, r1)),
              "ldmda r11!, {r0,r1}\n");

  COMPARE_A32(Ldmda(r11, NO_WRITE_BACK, RegisterList(r2, r3)),
              "ldmda r11, {r2,r3}\n");

  COMPARE_A32(Stmda(r11, WRITE_BACK, RegisterList(r0, r1)),
              "stmda r11!, {r0,r1}\n");

  COMPARE_A32(Stmda(r11, NO_WRITE_BACK, RegisterList(r2, r3)),
              "stmda r11, {r2,r3}\n");

  // ldmib/stmib
  COMPARE_A32(Ldmib(r11, WRITE_BACK, RegisterList(r0, r1)),
              "ldmib r11!, {r0,r1}\n");

  COMPARE_A32(Ldmib(r11, NO_WRITE_BACK, RegisterList(r2, r3)),
              "ldmib r11, {r2,r3}\n");

  COMPARE_A32(Stmib(r11, WRITE_BACK, RegisterList(r0, r1)),
              "stmib r11!, {r0,r1}\n");

  COMPARE_A32(Stmib(r11, NO_WRITE_BACK, RegisterList(r2, r3)),
              "stmib r11, {r2,r3}\n");

  // ldmdb/stmdb
  COMPARE_BOTH(Ldmdb(r11, WRITE_BACK, RegisterList(r0, r1)),
               "ldmdb r11!, {r0,r1}\n");

  COMPARE_BOTH(Ldmdb(r11, NO_WRITE_BACK, RegisterList(r2, r3)),
               "ldmdb r11, {r2,r3}\n");

  COMPARE_BOTH(Stmdb(r11, WRITE_BACK, RegisterList(r0, r1)),
               "stmdb r11!, {r0,r1}\n");

  COMPARE_BOTH(Stmdb(r11, NO_WRITE_BACK, RegisterList(r2, r3)),
               "stmdb r11, {r2,r3}\n");

  CLEANUP();
}


#define CHECK_T32_16(ASM, EXP) COMPARE_T32_CHECK_SIZE(ASM, EXP, 2)
// For instructions inside an IT block, we need to account for the IT
// instruction as well (another 16 bits).
#define CHECK_T32_16_IT_BLOCK(ASM, EXP) COMPARE_T32_CHECK_SIZE(ASM, EXP, 4)

TEST(macro_assembler_T32_16bit) {
  SETUP();

  // Allow the test to use all registers.
  UseScratchRegisterScope temps(&masm);
  temps.ExcludeAll();

  CHECK_T32_16(Adc(DontCare, r7, r7, r6), "adcs r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Adc(DontCare, eq, r7, r7, r6),
                        "it eq\n"
                        "adceq r7, r6\n");

  CHECK_T32_16(Add(DontCare, r6, r7, 7), "adds r6, r7, #7\n");

  CHECK_T32_16_IT_BLOCK(Add(DontCare, lt, r6, r7, 7),
                        "it lt\n"
                        "addlt r6, r7, #7\n");

  CHECK_T32_16(Add(DontCare, r5, r5, 255), "adds r5, #255\n");

  CHECK_T32_16_IT_BLOCK(Add(DontCare, lt, r5, r5, 255),
                        "it lt\n"
                        "addlt r5, #255\n");

  // Make sure we select the non flag-setting version here, since
  // this can have two potential encodings.
  CHECK_T32_16(Add(DontCare, r1, r1, r2), "add r1, r2\n");

  CHECK_T32_16(Add(DontCare, r1, r2, r7), "adds r1, r2, r7\n");

  CHECK_T32_16_IT_BLOCK(Add(DontCare, lt, r1, r2, r7),
                        "it lt\n"
                        "addlt r1, r2, r7\n");

  CHECK_T32_16(Add(DontCare, r4, r4, r12), "add r4, ip\n");

  CHECK_T32_16_IT_BLOCK(Add(DontCare, eq, r4, r4, r12),
                        "it eq\n"
                        "addeq r4, ip\n");

  CHECK_T32_16(Add(DontCare, r0, sp, 1020), "add r0, sp, #1020\n");

  CHECK_T32_16_IT_BLOCK(Add(DontCare, ge, r0, sp, 1020),
                        "it ge\n"
                        "addge r0, sp, #1020\n");

  // The equivalent inside an IT block is deprecated.
  CHECK_T32_16(Add(DontCare, sp, sp, 508), "add sp, #508\n");

  CHECK_T32_16(Add(DontCare, r7, sp, r7), "add r7, sp, r7\n");

  CHECK_T32_16_IT_BLOCK(Add(DontCare, eq, r7, sp, r7),
                        "it eq\n"
                        "addeq r7, sp, r7\n");

  CHECK_T32_16(Add(DontCare, sp, sp, r10), "add sp, r10\n");

  CHECK_T32_16_IT_BLOCK(Add(DontCare, eq, sp, sp, r10),
                        "it eq\n"
                        "addeq sp, r10\n");

  CHECK_T32_16(And(DontCare, r7, r7, r6), "ands r7, r6\n");

  CHECK_T32_16_IT_BLOCK(And(DontCare, eq, r7, r7, r6),
                        "it eq\n"
                        "andeq r7, r6\n");

  CHECK_T32_16(Asr(DontCare, r0, r1, 32), "asrs r0, r1, #32\n");

  CHECK_T32_16_IT_BLOCK(Asr(DontCare, eq, r0, r1, 32),
                        "it eq\n"
                        "asreq r0, r1, #32\n");

  CHECK_T32_16(Asr(DontCare, r0, r0, r1), "asrs r0, r1\n");

  CHECK_T32_16_IT_BLOCK(Asr(DontCare, eq, r0, r0, r1),
                        "it eq\n"
                        "asreq r0, r1\n");

  CHECK_T32_16(Bic(DontCare, r7, r7, r6), "bics r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Bic(DontCare, eq, r7, r7, r6),
                        "it eq\n"
                        "biceq r7, r6\n");

  CHECK_T32_16(Eor(DontCare, r7, r7, r6), "eors r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Eor(DontCare, eq, r7, r7, r6),
                        "it eq\n"
                        "eoreq r7, r6\n");

  CHECK_T32_16(Lsl(DontCare, r0, r1, 31), "lsls r0, r1, #31\n");

  CHECK_T32_16_IT_BLOCK(Lsl(DontCare, eq, r0, r1, 31),
                        "it eq\n"
                        "lsleq r0, r1, #31\n");

  CHECK_T32_16(Lsl(DontCare, r0, r0, r1), "lsls r0, r1\n");

  CHECK_T32_16_IT_BLOCK(Lsl(DontCare, eq, r0, r0, r1),
                        "it eq\n"
                        "lsleq r0, r1\n");

  CHECK_T32_16(Lsr(DontCare, r0, r1, 32), "lsrs r0, r1, #32\n");

  CHECK_T32_16_IT_BLOCK(Lsr(DontCare, eq, r0, r1, 32),
                        "it eq\n"
                        "lsreq r0, r1, #32\n");

  CHECK_T32_16(Lsr(DontCare, r0, r0, r1), "lsrs r0, r1\n");

  CHECK_T32_16_IT_BLOCK(Lsr(DontCare, eq, r0, r0, r1),
                        "it eq\n"
                        "lsreq r0, r1\n");

  CHECK_T32_16(Mov(DontCare, r7, 255), "movs r7, #255\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r7, 255),
                        "it eq\n"
                        "moveq r7, #255\n");

  CHECK_T32_16(Mov(DontCare, r9, r8), "mov r9, r8\n");

  // Check that we don't try to pick the MOVS register-shifted register variant.
  CHECK_T32_16(Mov(DontCare, r5, r6), "mov r5, r6\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r9, r8),
                        "it eq\n"
                        "moveq r9, r8\n");

  CHECK_T32_16(Mov(DontCare, r5, Operand(r6, ASR, 1)), "asrs r5, r6, #1\n");

  CHECK_T32_16(Mov(DontCare, r5, Operand(r6, ASR, 32)), "asrs r5, r6, #32\n");

  CHECK_T32_16(Mov(DontCare, r5, Operand(r6, LSR, 1)), "lsrs r5, r6, #1\n");

  CHECK_T32_16(Mov(DontCare, r5, Operand(r6, LSR, 32)), "lsrs r5, r6, #32\n");

  CHECK_T32_16(Mov(DontCare, r5, Operand(r6, LSL, 1)), "lsls r5, r6, #1\n");

  CHECK_T32_16(Mov(DontCare, r5, Operand(r6, LSL, 31)), "lsls r5, r6, #31\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r5, Operand(r6, ASR, 1)),
                        "it eq\n"
                        "asreq r5, r6, #1\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r5, Operand(r6, ASR, 32)),
                        "it eq\n"
                        "asreq r5, r6, #32\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r5, Operand(r6, LSR, 1)),
                        "it eq\n"
                        "lsreq r5, r6, #1\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r5, Operand(r6, LSR, 32)),
                        "it eq\n"
                        "lsreq r5, r6, #32\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r5, Operand(r6, LSL, 1)),
                        "it eq\n"
                        "lsleq r5, r6, #1\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r5, Operand(r6, LSL, 31)),
                        "it eq\n"
                        "lsleq r5, r6, #31\n");

  CHECK_T32_16(Mov(DontCare, r7, Operand(r7, ASR, r6)), "asrs r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r7, Operand(r7, ASR, r6)),
                        "it eq\n"
                        "asreq r7, r6\n");

  CHECK_T32_16(Mov(DontCare, r7, Operand(r7, LSR, r6)), "lsrs r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r7, Operand(r7, LSR, r6)),
                        "it eq\n"
                        "lsreq r7, r6\n");

  CHECK_T32_16(Mov(DontCare, r7, Operand(r7, LSL, r6)), "lsls r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r7, Operand(r7, LSL, r6)),
                        "it eq\n"
                        "lsleq r7, r6\n");

  CHECK_T32_16(Mov(DontCare, r7, Operand(r7, ROR, r6)), "rors r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Mov(DontCare, eq, r7, Operand(r7, ROR, r6)),
                        "it eq\n"
                        "roreq r7, r6\n");

  CHECK_T32_16(Mul(DontCare, r0, r1, r0), "muls r0, r1, r0\n");

  CHECK_T32_16_IT_BLOCK(Mul(DontCare, eq, r0, r1, r0),
                        "it eq\n"
                        "muleq r0, r1, r0\n");

  CHECK_T32_16(Mvn(DontCare, r6, r7), "mvns r6, r7\n");

  CHECK_T32_16_IT_BLOCK(Mvn(DontCare, eq, r6, r7),
                        "it eq\n"
                        "mvneq r6, r7\n");

  CHECK_T32_16(Orr(DontCare, r7, r7, r6), "orrs r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Orr(DontCare, eq, r7, r7, r6),
                        "it eq\n"
                        "orreq r7, r6\n");

  CHECK_T32_16(Ror(DontCare, r0, r0, r1), "rors r0, r1\n");

  CHECK_T32_16_IT_BLOCK(Ror(DontCare, eq, r0, r0, r1),
                        "it eq\n"
                        "roreq r0, r1\n");

  CHECK_T32_16(Rsb(DontCare, r7, r6, 0), "rsbs r7, r6, #0\n");

  CHECK_T32_16_IT_BLOCK(Rsb(DontCare, eq, r7, r6, 0),
                        "it eq\n"
                        "rsbeq r7, r6, #0\n");

  CHECK_T32_16(Sbc(DontCare, r7, r7, r6), "sbcs r7, r6\n");

  CHECK_T32_16_IT_BLOCK(Sbc(DontCare, eq, r7, r7, r6),
                        "it eq\n"
                        "sbceq r7, r6\n");

  CHECK_T32_16(Sub(DontCare, r6, r7, 7), "subs r6, r7, #7\n");

  CHECK_T32_16_IT_BLOCK(Sub(DontCare, lt, r6, r7, 7),
                        "it lt\n"
                        "sublt r6, r7, #7\n");

  CHECK_T32_16(Sub(DontCare, r5, r5, 255), "subs r5, #255\n");

  CHECK_T32_16_IT_BLOCK(Sub(DontCare, lt, r5, r5, 255),
                        "it lt\n"
                        "sublt r5, #255\n");

  CHECK_T32_16(Sub(DontCare, r1, r2, r7), "subs r1, r2, r7\n");

  CHECK_T32_16_IT_BLOCK(Sub(DontCare, lt, r1, r2, r7),
                        "it lt\n"
                        "sublt r1, r2, r7\n");

  // The equivalent inside an IT block is deprecated.
  CHECK_T32_16(Sub(DontCare, sp, sp, 508), "sub sp, #508\n");

  // Generate SUBS for ADD.
  CHECK_T32_16(Add(DontCare, r0, r1, -1), "subs r0, r1, #1\n");

  CHECK_T32_16(Add(DontCare, r0, r1, -7), "subs r0, r1, #7\n");

  CHECK_T32_16(Add(DontCare, r6, r6, -1), "subs r6, #1\n");

  CHECK_T32_16(Add(DontCare, r6, r6, -255), "subs r6, #255\n");

  // Generate ADDS for SUB.
  CHECK_T32_16(Sub(DontCare, r0, r1, -1), "adds r0, r1, #1\n");

  CHECK_T32_16(Sub(DontCare, r0, r1, -7), "adds r0, r1, #7\n");

  CHECK_T32_16(Sub(DontCare, r6, r6, -1), "adds r6, #1\n");

  CHECK_T32_16(Sub(DontCare, r6, r6, -255), "adds r6, #255\n");

  // Check that we don't change the opcode for INT_MIN.
  COMPARE_T32(Add(DontCare, r6, r6, 0x80000000), "add r6, #2147483648\n");

  COMPARE_T32(Sub(DontCare, r6, r6, 0x80000000), "sub r6, #2147483648\n");

  CLEANUP();
}
#undef CHECK_T32_16
#undef CHECK_T32_16_IT_BLOCK

TEST(nop_code) {
  SETUP();

  COMPARE_BOTH(Nop(), "nop\n");

  COMPARE_BOTH(And(r0, r0, r0), "");
  COMPARE_BOTH(And(DontCare, r0, r0, r0), "");

  COMPARE_BOTH(Mov(r0, r0), "");
  COMPARE_BOTH(Mov(DontCare, r0, r0), "");

  COMPARE_BOTH(Orr(r0, r0, r0), "");
  COMPARE_BOTH(Orr(DontCare, r0, r0, r0), "");

  CLEANUP();
}


TEST(big_add_sub) {
  SETUP();

  COMPARE_A32(Add(r0, r1, 0x4321),
              "add r0, r1, #33\n"
              "add r0, #17152\n");
  COMPARE_T32(Add(r0, r1, 0x4321),
              "add r0, r1, #801\n"
              "add r0, #16384\n");
  COMPARE_BOTH(Add(r0, r1, 0x432100),
               "add r0, r1, #8448\n"
               "add r0, #4390912\n");
  COMPARE_BOTH(Add(r0, r1, 0x43000210),
               "add r0, r1, #528\n"
               "add r0, #1124073472\n");
  COMPARE_BOTH(Add(r0, r1, 0x30c00210),
               "add r0, r1, #528\n"
               "add r0, #817889280\n");
  COMPARE_BOTH(Add(r0, r1, 0x43000021),
               "add r0, r1, #33\n"
               "add r0, #1124073472\n");
  COMPARE_T32(Add(r0, r1, 0x54321),
              "add r0, r1, #801\n"
              "add r0, #344064\n");
  COMPARE_T32(Add(r0, r1, 0x54000321),
              "add r0, r1, #801\n"
              "add r0, #1409286144\n");

  COMPARE_A32(Sub(r0, r1, 0x4321),
              "sub r0, r1, #33\n"
              "sub r0, #17152\n");
  COMPARE_T32(Sub(r0, r1, 0x4321),
              "sub r0, r1, #801\n"
              "sub r0, #16384\n");
  COMPARE_BOTH(Sub(r0, r1, 0x432100),
               "sub r0, r1, #8448\n"
               "sub r0, #4390912\n");
  COMPARE_BOTH(Sub(r0, r1, 0x43000210),
               "sub r0, r1, #528\n"
               "sub r0, #1124073472\n");
  COMPARE_BOTH(Sub(r0, r1, 0x30c00210),
               "sub r0, r1, #528\n"
               "sub r0, #817889280\n");
  COMPARE_BOTH(Sub(r0, r1, 0x43000021),
               "sub r0, r1, #33\n"
               "sub r0, #1124073472\n");
  COMPARE_T32(Sub(r0, r1, 0x54321),
              "sub r0, r1, #801\n"
              "sub r0, #344064\n");
  COMPARE_T32(Sub(r0, r1, 0x54000321),
              "sub r0, r1, #801\n"
              "sub r0, #1409286144\n");

  CLEANUP();
}

}  // namespace aarch32
}  // namespace vixl