2010-09-18 02:46:17 +08:00
|
|
|
//===-- ARM/ARMMCCodeEmitter.cpp - Convert ARM code to machine code -------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2010-09-18 02:46:17 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the ARMMCCodeEmitter class.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-07-21 07:34:39 +08:00
|
|
|
#include "MCTargetDesc/ARMAddressingModes.h"
|
2011-07-23 08:00:19 +08:00
|
|
|
#include "MCTargetDesc/ARMBaseInfo.h"
|
|
|
|
#include "MCTargetDesc/ARMFixupKinds.h"
|
2011-07-21 07:34:39 +08:00
|
|
|
#include "MCTargetDesc/ARMMCExpr.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/APFloat.h"
|
2017-01-28 07:58:02 +08:00
|
|
|
#include "llvm/ADT/APInt.h"
|
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
2017-01-28 07:58:02 +08:00
|
|
|
#include "llvm/ADT/Triple.h"
|
2010-09-18 02:46:17 +08:00
|
|
|
#include "llvm/MC/MCCodeEmitter.h"
|
2012-08-10 06:10:21 +08:00
|
|
|
#include "llvm/MC/MCContext.h"
|
2010-09-18 02:46:17 +08:00
|
|
|
#include "llvm/MC/MCExpr.h"
|
2017-01-28 07:58:02 +08:00
|
|
|
#include "llvm/MC/MCFixup.h"
|
2010-09-18 02:46:17 +08:00
|
|
|
#include "llvm/MC/MCInst.h"
|
2017-01-28 07:58:02 +08:00
|
|
|
#include "llvm/MC/MCInstrDesc.h"
|
2011-07-11 11:57:24 +08:00
|
|
|
#include "llvm/MC/MCInstrInfo.h"
|
2011-07-23 08:00:19 +08:00
|
|
|
#include "llvm/MC/MCRegisterInfo.h"
|
2011-07-11 11:57:24 +08:00
|
|
|
#include "llvm/MC/MCSubtargetInfo.h"
|
2017-01-28 07:58:02 +08:00
|
|
|
#include "llvm/Support/Casting.h"
|
|
|
|
#include "llvm/Support/Compiler.h"
|
2014-01-12 07:03:48 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2017-01-28 07:58:02 +08:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
2010-09-18 02:46:17 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2017-01-28 07:58:02 +08:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cassert>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <cstdlib>
|
2011-07-11 11:57:24 +08:00
|
|
|
|
2010-09-18 02:46:17 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
2014-04-22 10:41:26 +08:00
|
|
|
#define DEBUG_TYPE "mccodeemitter"
|
|
|
|
|
2010-11-04 09:12:30 +08:00
|
|
|
STATISTIC(MCNumEmitted, "Number of MC instructions emitted.");
|
|
|
|
STATISTIC(MCNumCPRelocations, "Number of constant pool relocations created.");
|
2010-10-08 06:12:50 +08:00
|
|
|
|
2010-09-18 02:46:17 +08:00
|
|
|
namespace {
|
2017-01-28 07:58:02 +08:00
|
|
|
|
2010-09-18 02:46:17 +08:00
|
|
|
class ARMMCCodeEmitter : public MCCodeEmitter {
|
2011-07-11 11:57:24 +08:00
|
|
|
const MCInstrInfo &MCII;
|
2012-08-10 06:10:21 +08:00
|
|
|
const MCContext &CTX;
|
2014-03-28 22:35:30 +08:00
|
|
|
bool IsLittleEndian;
|
2010-09-18 02:46:17 +08:00
|
|
|
|
|
|
|
public:
|
2014-03-28 22:35:30 +08:00
|
|
|
ARMMCCodeEmitter(const MCInstrInfo &mcii, MCContext &ctx, bool IsLittle)
|
|
|
|
: MCII(mcii), CTX(ctx), IsLittleEndian(IsLittle) {
|
2010-09-18 02:46:17 +08:00
|
|
|
}
|
2017-01-28 07:58:02 +08:00
|
|
|
ARMMCCodeEmitter(const ARMMCCodeEmitter &) = delete;
|
|
|
|
ARMMCCodeEmitter &operator=(const ARMMCCodeEmitter &) = delete;
|
|
|
|
~ARMMCCodeEmitter() override = default;
|
2010-09-18 02:46:17 +08:00
|
|
|
|
2014-01-29 07:13:25 +08:00
|
|
|
bool isThumb(const MCSubtargetInfo &STI) const {
|
2015-05-26 18:47:10 +08:00
|
|
|
return STI.getFeatureBits()[ARM::ModeThumb];
|
2011-07-11 11:57:24 +08:00
|
|
|
}
|
2017-01-28 07:58:02 +08:00
|
|
|
|
2014-01-29 07:13:25 +08:00
|
|
|
bool isThumb2(const MCSubtargetInfo &STI) const {
|
2015-05-26 18:47:10 +08:00
|
|
|
return isThumb(STI) && STI.getFeatureBits()[ARM::FeatureThumb2];
|
2011-07-11 11:57:24 +08:00
|
|
|
}
|
2017-01-28 07:58:02 +08:00
|
|
|
|
2014-01-29 07:13:25 +08:00
|
|
|
bool isTargetMachO(const MCSubtargetInfo &STI) const {
|
2015-09-16 00:17:27 +08:00
|
|
|
const Triple &TT = STI.getTargetTriple();
|
2014-01-06 22:28:05 +08:00
|
|
|
return TT.isOSBinFormatMachO();
|
2011-07-11 11:57:24 +08:00
|
|
|
}
|
|
|
|
|
2010-10-13 01:11:26 +08:00
|
|
|
unsigned getMachineSoImmOpValue(unsigned SoImm) const;
|
|
|
|
|
2010-10-08 05:57:55 +08:00
|
|
|
// getBinaryCodeForInstr - TableGen'erated function for getting the
|
|
|
|
// binary encoding for an instruction.
|
2012-01-25 02:37:29 +08:00
|
|
|
uint64_t getBinaryCodeForInstr(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-10-08 05:57:55 +08:00
|
|
|
|
|
|
|
/// getMachineOpValue - Return binary encoding of operand. If the machine
|
|
|
|
/// operand requires relocation, record the relocation and return zero.
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned getMachineOpValue(const MCInst &MI,const MCOperand &MO,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-10-08 05:57:55 +08:00
|
|
|
|
2011-01-13 15:58:56 +08:00
|
|
|
/// getHiLo16ImmOpValue - Return the encoding for the hi / low 16-bit of
|
2011-02-09 06:39:40 +08:00
|
|
|
/// the specified operand. This is used for operands with :lower16: and
|
2011-01-13 15:58:56 +08:00
|
|
|
/// :upper16: prefixes.
|
|
|
|
uint32_t getHiLo16ImmOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-19 07:37:15 +08:00
|
|
|
|
2010-11-03 09:49:29 +08:00
|
|
|
bool EncodeAddrModeOpValues(const MCInst &MI, unsigned OpIdx,
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned &Reg, unsigned &Imm,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-03 09:49:29 +08:00
|
|
|
|
2010-12-07 07:57:07 +08:00
|
|
|
/// getThumbBLTargetOpValue - Return encoding info for Thumb immediate
|
2010-12-09 08:39:08 +08:00
|
|
|
/// BL branch target.
|
2010-12-07 07:57:07 +08:00
|
|
|
uint32_t getThumbBLTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-07 07:57:07 +08:00
|
|
|
|
2010-12-09 08:39:08 +08:00
|
|
|
/// getThumbBLXTargetOpValue - Return encoding info for Thumb immediate
|
|
|
|
/// BLX branch target.
|
|
|
|
uint32_t getThumbBLXTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-09 08:39:08 +08:00
|
|
|
|
2010-12-11 02:21:33 +08:00
|
|
|
/// getThumbBRTargetOpValue - Return encoding info for Thumb branch target.
|
|
|
|
uint32_t getThumbBRTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-11 02:21:33 +08:00
|
|
|
|
2010-12-11 01:13:40 +08:00
|
|
|
/// getThumbBCCTargetOpValue - Return encoding info for Thumb branch target.
|
|
|
|
uint32_t getThumbBCCTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-11 01:13:40 +08:00
|
|
|
|
2010-12-10 03:04:53 +08:00
|
|
|
/// getThumbCBTargetOpValue - Return encoding info for Thumb branch target.
|
|
|
|
uint32_t getThumbCBTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-09 07:01:43 +08:00
|
|
|
|
2010-11-12 02:04:49 +08:00
|
|
|
/// getBranchTargetOpValue - Return encoding info for 24-bit immediate
|
|
|
|
/// branch target.
|
|
|
|
uint32_t getBranchTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-12 02:04:49 +08:00
|
|
|
|
2016-07-12 06:29:37 +08:00
|
|
|
/// getThumbBranchTargetOpValue - Return encoding info for 24-bit
|
2010-12-14 03:31:11 +08:00
|
|
|
/// immediate Thumb2 direct branch target.
|
2016-07-12 06:29:37 +08:00
|
|
|
uint32_t getThumbBranchTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-08-31 05:58:18 +08:00
|
|
|
|
2011-02-05 03:47:15 +08:00
|
|
|
/// getARMBranchTargetOpValue - Return encoding info for 24-bit immediate
|
|
|
|
/// branch target.
|
|
|
|
uint32_t getARMBranchTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2012-02-28 05:36:23 +08:00
|
|
|
uint32_t getARMBLTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-08-27 07:32:08 +08:00
|
|
|
uint32_t getARMBLXTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-14 03:31:11 +08:00
|
|
|
|
2010-12-02 03:47:31 +08:00
|
|
|
/// getAdrLabelOpValue - Return encoding info for 12-bit immediate
|
|
|
|
/// ADR label target.
|
|
|
|
uint32_t getAdrLabelOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-15 06:28:03 +08:00
|
|
|
uint32_t getThumbAdrLabelOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-14 08:36:49 +08:00
|
|
|
uint32_t getT2AdrLabelOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-02-09 06:39:40 +08:00
|
|
|
|
2010-12-02 03:47:31 +08:00
|
|
|
|
2010-11-03 09:49:29 +08:00
|
|
|
/// getAddrModeImm12OpValue - Return encoding info for 'reg +/- imm12'
|
|
|
|
/// operand.
|
2010-11-04 07:52:49 +08:00
|
|
|
uint32_t getAddrModeImm12OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-03 09:49:29 +08:00
|
|
|
|
2010-12-14 11:36:38 +08:00
|
|
|
/// getThumbAddrModeRegRegOpValue - Return encoding for 'reg + reg' operand.
|
|
|
|
uint32_t getThumbAddrModeRegRegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-11 06:11:13 +08:00
|
|
|
|
2010-12-02 03:18:46 +08:00
|
|
|
/// getT2AddrModeImm8s4OpValue - Return encoding info for 'reg +/- imm8<<2'
|
|
|
|
/// operand.
|
|
|
|
uint32_t getT2AddrModeImm8s4OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-09-10 02:37:27 +08:00
|
|
|
|
|
|
|
/// getT2AddrModeImm0_1020s4OpValue - Return encoding info for 'reg + imm8<<2'
|
|
|
|
/// operand.
|
|
|
|
uint32_t getT2AddrModeImm0_1020s4OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-09-10 02:37:27 +08:00
|
|
|
|
2011-09-09 06:07:06 +08:00
|
|
|
/// getT2Imm8s4OpValue - Return encoding info for '+/- imm8<<2'
|
|
|
|
/// operand.
|
|
|
|
uint32_t getT2Imm8s4OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-02 03:18:46 +08:00
|
|
|
|
|
|
|
|
2010-11-10 01:20:53 +08:00
|
|
|
/// getLdStSORegOpValue - Return encoding info for 'reg +/- reg shop imm'
|
|
|
|
/// operand as needed by load/store instructions.
|
|
|
|
uint32_t getLdStSORegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-10 01:20:53 +08:00
|
|
|
|
2010-11-11 07:38:36 +08:00
|
|
|
/// getLdStmModeOpValue - Return encoding for load/store multiple mode.
|
|
|
|
uint32_t getLdStmModeOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-11 07:38:36 +08:00
|
|
|
ARM_AM::AMSubMode Mode = (ARM_AM::AMSubMode)MI.getOperand(OpIdx).getImm();
|
|
|
|
switch (Mode) {
|
2012-02-07 10:50:20 +08:00
|
|
|
default: llvm_unreachable("Unknown addressing sub-mode!");
|
2010-11-11 07:38:36 +08:00
|
|
|
case ARM_AM::da: return 0;
|
|
|
|
case ARM_AM::ia: return 1;
|
|
|
|
case ARM_AM::db: return 2;
|
|
|
|
case ARM_AM::ib: return 3;
|
|
|
|
}
|
|
|
|
}
|
2017-01-28 07:58:02 +08:00
|
|
|
|
2010-11-16 04:47:07 +08:00
|
|
|
/// getShiftOp - Return the shift opcode (bit[6:5]) of the immediate value.
|
|
|
|
///
|
|
|
|
unsigned getShiftOp(ARM_AM::ShiftOpc ShOpc) const {
|
|
|
|
switch (ShOpc) {
|
|
|
|
case ARM_AM::no_shift:
|
|
|
|
case ARM_AM::lsl: return 0;
|
|
|
|
case ARM_AM::lsr: return 1;
|
|
|
|
case ARM_AM::asr: return 2;
|
|
|
|
case ARM_AM::ror:
|
|
|
|
case ARM_AM::rrx: return 3;
|
|
|
|
}
|
2012-01-21 05:51:11 +08:00
|
|
|
llvm_unreachable("Invalid ShiftOpc!");
|
2010-11-16 04:47:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// getAddrMode2OffsetOpValue - Return encoding for am2offset operands.
|
|
|
|
uint32_t getAddrMode2OffsetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-16 04:47:07 +08:00
|
|
|
|
2011-08-04 07:50:40 +08:00
|
|
|
/// getPostIdxRegOpValue - Return encoding for postidx_reg operands.
|
|
|
|
uint32_t getPostIdxRegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-08-04 07:50:40 +08:00
|
|
|
|
2010-11-12 00:55:29 +08:00
|
|
|
/// getAddrMode3OffsetOpValue - Return encoding for am3offset operands.
|
|
|
|
uint32_t getAddrMode3OffsetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-12 00:55:29 +08:00
|
|
|
|
2010-11-11 09:09:40 +08:00
|
|
|
/// getAddrMode3OpValue - Return encoding for addrmode3 operands.
|
|
|
|
uint32_t getAddrMode3OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-11 07:38:36 +08:00
|
|
|
|
2010-12-08 05:50:47 +08:00
|
|
|
/// getAddrModeThumbSPOpValue - Return encoding info for 'reg +/- imm12'
|
|
|
|
/// operand.
|
|
|
|
uint32_t getAddrModeThumbSPOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-08 05:50:47 +08:00
|
|
|
|
2010-12-14 11:36:38 +08:00
|
|
|
/// getAddrModeISOpValue - Encode the t_addrmode_is# operands.
|
|
|
|
uint32_t getAddrModeISOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-01 06:57:21 +08:00
|
|
|
|
2010-12-08 09:57:09 +08:00
|
|
|
/// getAddrModePCOpValue - Return encoding for t_addrmode_pc operands.
|
|
|
|
uint32_t getAddrModePCOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-08 09:57:09 +08:00
|
|
|
|
2016-01-25 18:26:26 +08:00
|
|
|
/// getAddrMode5OpValue - Return encoding info for 'reg +/- (imm8 << 2)' operand.
|
2010-11-04 07:52:49 +08:00
|
|
|
uint32_t getAddrMode5OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-10-27 06:37:02 +08:00
|
|
|
|
2016-01-25 18:26:26 +08:00
|
|
|
/// getAddrMode5FP16OpValue - Return encoding info for 'reg +/- (imm8 << 1)' operand.
|
|
|
|
uint32_t getAddrMode5FP16OpValue(const MCInst &MI, unsigned OpIdx,
|
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
|
|
|
|
2010-10-13 07:00:24 +08:00
|
|
|
/// getCCOutOpValue - Return encoding of the 's' bit.
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned getCCOutOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-10-13 07:00:24 +08:00
|
|
|
// The operand is either reg0 or CPSR. The 's' bit is encoded as '0' or
|
|
|
|
// '1' respectively.
|
|
|
|
return MI.getOperand(Op).getReg() == ARM::CPSR;
|
|
|
|
}
|
2010-10-13 07:53:58 +08:00
|
|
|
|
2014-12-02 18:53:20 +08:00
|
|
|
unsigned getModImmOpValue(const MCInst &MI, unsigned Op,
|
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &ST) const {
|
|
|
|
const MCOperand &MO = MI.getOperand(Op);
|
|
|
|
|
|
|
|
// Support for fixups (MCFixup)
|
|
|
|
if (MO.isExpr()) {
|
|
|
|
const MCExpr *Expr = MO.getExpr();
|
Fix for pr24346: arm asm label calculation error in sub
Some ARM instructions encode 32-bit immediates as a 8-bit integer (0-255)
and a 4-bit rotation (0-30, even) in its least significant 12 bits. The
original fixup, FK_Data_4, patches the instruction by the value bit-to-bit,
regardless of the encoding. For example, assuming the label L1 and L2 are
0x0 and 0x104 respectively, the following instruction:
add r0, r0, #(L2 - L1) ; expects 0x104, i.e., 260
would be assembled to the following, which adds 1 to r0, instead of 260:
e2800104 add r0, r0, #4, 2 ; equivalently 1
The new fixup kind fixup_arm_mod_imm takes care of the encoding:
e2800f41 add r0, r0, #260
Patch by Ting-Yuan Huang!
llvm-svn: 265122
2016-04-01 17:40:47 +08:00
|
|
|
// Fixups resolve to plain values that need to be encoded.
|
|
|
|
MCFixupKind Kind = MCFixupKind(ARM::fixup_arm_mod_imm);
|
2015-05-16 03:13:05 +08:00
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
2014-12-02 18:53:20 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Immediate is already in its encoded format
|
|
|
|
return MO.getImm();
|
|
|
|
}
|
|
|
|
|
2010-11-13 05:12:40 +08:00
|
|
|
/// getT2SOImmOpValue - Return an encoded 12-bit shifted-immediate value.
|
|
|
|
unsigned getT2SOImmOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2017-06-05 17:37:12 +08:00
|
|
|
const MCOperand &MO = MI.getOperand(Op);
|
|
|
|
|
|
|
|
// Support for fixups (MCFixup)
|
|
|
|
if (MO.isExpr()) {
|
|
|
|
const MCExpr *Expr = MO.getExpr();
|
|
|
|
// Fixups resolve to plain values that need to be encoded.
|
|
|
|
MCFixupKind Kind = MCFixupKind(ARM::fixup_t2_so_imm);
|
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
unsigned SoImm = MO.getImm();
|
2010-11-13 05:12:40 +08:00
|
|
|
unsigned Encoded = ARM_AM::getT2SOImmVal(SoImm);
|
|
|
|
assert(Encoded != ~0U && "Not a Thumb2 so_imm value?");
|
|
|
|
return Encoded;
|
|
|
|
}
|
2010-10-13 07:00:24 +08:00
|
|
|
|
2010-11-30 06:44:32 +08:00
|
|
|
unsigned getT2AddrModeSORegOpValue(const MCInst &MI, unsigned OpNum,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-30 06:44:32 +08:00
|
|
|
unsigned getT2AddrModeImm8OpValue(const MCInst &MI, unsigned OpNum,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-30 08:14:31 +08:00
|
|
|
unsigned getT2AddrModeImm8OffsetOpValue(const MCInst &MI, unsigned OpNum,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-30 06:44:32 +08:00
|
|
|
|
2010-10-13 07:53:58 +08:00
|
|
|
/// getSORegOpValue - Return an encoded so_reg shifted register value.
|
2011-07-22 07:38:37 +08:00
|
|
|
unsigned getSORegRegOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-07-22 07:38:37 +08:00
|
|
|
unsigned getSORegImmOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-13 05:12:40 +08:00
|
|
|
unsigned getT2SORegOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-10-13 07:53:58 +08:00
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned getNEONVcvtImm32OpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-10-28 06:49:00 +08:00
|
|
|
return 64 - MI.getOperand(Op).getImm();
|
|
|
|
}
|
2010-10-16 01:15:16 +08:00
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned getBitfieldInvertedMaskOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-10-22 06:03:21 +08:00
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned getRegisterListOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned getAddrMode6AddressOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-05-10 01:47:27 +08:00
|
|
|
unsigned getAddrMode6OneLane32AddressOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-30 08:00:42 +08:00
|
|
|
unsigned getAddrMode6DupAddressOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned getAddrMode6OffsetOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-10-30 08:37:59 +08:00
|
|
|
|
2011-03-08 07:38:41 +08:00
|
|
|
unsigned getShiftRight8Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-03-08 07:38:41 +08:00
|
|
|
unsigned getShiftRight16Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-03-08 07:38:41 +08:00
|
|
|
unsigned getShiftRight32Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-03-08 07:38:41 +08:00
|
|
|
unsigned getShiftRight64Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-03-01 09:00:59 +08:00
|
|
|
|
2011-08-09 04:42:17 +08:00
|
|
|
unsigned getThumbSRImmOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2011-08-09 04:42:17 +08:00
|
|
|
|
2010-11-12 03:07:48 +08:00
|
|
|
unsigned NEONThumb2DataIPostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-12 05:36:43 +08:00
|
|
|
unsigned NEONThumb2LoadStorePostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-12 07:12:55 +08:00
|
|
|
unsigned NEONThumb2DupPostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2013-07-17 21:59:38 +08:00
|
|
|
unsigned NEONThumb2V8PostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-12-02 05:54:50 +08:00
|
|
|
|
|
|
|
unsigned VFPThumb2PostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const;
|
2010-11-12 03:07:48 +08:00
|
|
|
|
2010-11-04 09:12:30 +08:00
|
|
|
void EmitByte(unsigned char C, raw_ostream &OS) const {
|
2010-09-18 02:46:17 +08:00
|
|
|
OS << (char)C;
|
|
|
|
}
|
|
|
|
|
2010-11-04 09:12:30 +08:00
|
|
|
void EmitConstant(uint64_t Val, unsigned Size, raw_ostream &OS) const {
|
2010-09-18 02:46:17 +08:00
|
|
|
// Output the constant in little endian byte order.
|
|
|
|
for (unsigned i = 0; i != Size; ++i) {
|
2014-03-28 22:35:30 +08:00
|
|
|
unsigned Shift = IsLittleEndian ? i * 8 : (Size - 1 - i) * 8;
|
|
|
|
EmitByte((Val >> Shift) & 0xff, OS);
|
2010-09-18 02:46:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-16 03:13:16 +08:00
|
|
|
void encodeInstruction(const MCInst &MI, raw_ostream &OS,
|
2014-01-29 07:13:07 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
2014-03-10 11:19:03 +08:00
|
|
|
const MCSubtargetInfo &STI) const override;
|
2010-09-18 02:46:17 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2010-12-11 05:57:34 +08:00
|
|
|
/// NEONThumb2DataIPostEncoder - Post-process encoded NEON data-processing
|
|
|
|
/// instructions, and rewrite them to their Thumb2 form if we are currently in
|
2010-11-12 03:07:48 +08:00
|
|
|
/// Thumb2 mode.
|
|
|
|
unsigned ARMMCCodeEmitter::NEONThumb2DataIPostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI)) {
|
2010-12-11 05:57:34 +08:00
|
|
|
// NEON Thumb2 data-processsing encodings are very simple: bit 24 is moved
|
2010-11-12 03:07:48 +08:00
|
|
|
// to bit 12 of the high half-word (i.e. bit 28), and bits 27-24 are
|
|
|
|
// set to 1111.
|
|
|
|
unsigned Bit24 = EncodedValue & 0x01000000;
|
|
|
|
unsigned Bit28 = Bit24 << 4;
|
|
|
|
EncodedValue &= 0xEFFFFFFF;
|
|
|
|
EncodedValue |= Bit28;
|
|
|
|
EncodedValue |= 0x0F000000;
|
|
|
|
}
|
2010-12-11 05:57:34 +08:00
|
|
|
|
2010-11-12 03:07:48 +08:00
|
|
|
return EncodedValue;
|
|
|
|
}
|
|
|
|
|
2010-11-12 05:36:43 +08:00
|
|
|
/// NEONThumb2LoadStorePostEncoder - Post-process encoded NEON load/store
|
2010-12-11 05:57:34 +08:00
|
|
|
/// instructions, and rewrite them to their Thumb2 form if we are currently in
|
2010-11-12 05:36:43 +08:00
|
|
|
/// Thumb2 mode.
|
|
|
|
unsigned ARMMCCodeEmitter::NEONThumb2LoadStorePostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI)) {
|
2010-11-12 05:36:43 +08:00
|
|
|
EncodedValue &= 0xF0FFFFFF;
|
|
|
|
EncodedValue |= 0x09000000;
|
|
|
|
}
|
2010-12-11 05:57:34 +08:00
|
|
|
|
2010-11-12 05:36:43 +08:00
|
|
|
return EncodedValue;
|
|
|
|
}
|
|
|
|
|
2010-11-12 07:12:55 +08:00
|
|
|
/// NEONThumb2DupPostEncoder - Post-process encoded NEON vdup
|
2010-12-11 05:57:34 +08:00
|
|
|
/// instructions, and rewrite them to their Thumb2 form if we are currently in
|
2010-11-12 07:12:55 +08:00
|
|
|
/// Thumb2 mode.
|
|
|
|
unsigned ARMMCCodeEmitter::NEONThumb2DupPostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI)) {
|
2010-11-12 07:12:55 +08:00
|
|
|
EncodedValue &= 0x00FFFFFF;
|
|
|
|
EncodedValue |= 0xEE000000;
|
|
|
|
}
|
2010-12-11 05:57:34 +08:00
|
|
|
|
2010-11-12 07:12:55 +08:00
|
|
|
return EncodedValue;
|
|
|
|
}
|
|
|
|
|
2013-07-17 21:59:38 +08:00
|
|
|
/// Post-process encoded NEON v8 instructions, and rewrite them to Thumb2 form
|
|
|
|
/// if we are in Thumb2.
|
|
|
|
unsigned ARMMCCodeEmitter::NEONThumb2V8PostEncoder(const MCInst &MI,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI)) {
|
2013-07-17 21:59:38 +08:00
|
|
|
EncodedValue |= 0xC000000; // Set bits 27-26
|
|
|
|
}
|
|
|
|
|
|
|
|
return EncodedValue;
|
|
|
|
}
|
|
|
|
|
2010-12-02 05:54:50 +08:00
|
|
|
/// VFPThumb2PostEncoder - Post-process encoded VFP instructions and rewrite
|
|
|
|
/// them to their Thumb2 form if we are currently in Thumb2 mode.
|
|
|
|
unsigned ARMMCCodeEmitter::
|
2014-01-29 07:13:18 +08:00
|
|
|
VFPThumb2PostEncoder(const MCInst &MI, unsigned EncodedValue,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI)) {
|
2010-12-02 05:54:50 +08:00
|
|
|
EncodedValue &= 0x0FFFFFFF;
|
|
|
|
EncodedValue |= 0xE0000000;
|
|
|
|
}
|
|
|
|
return EncodedValue;
|
|
|
|
}
|
2010-11-12 05:36:43 +08:00
|
|
|
|
2010-10-09 05:45:55 +08:00
|
|
|
/// getMachineOpValue - Return binary encoding of operand. If the machine
|
|
|
|
/// operand requires relocation, record the relocation and return zero.
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getMachineOpValue(const MCInst &MI, const MCOperand &MO,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-10-14 10:33:26 +08:00
|
|
|
if (MO.isReg()) {
|
2010-11-03 06:53:11 +08:00
|
|
|
unsigned Reg = MO.getReg();
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned RegNo = CTX.getRegisterInfo()->getEncodingValue(Reg);
|
2010-10-30 07:21:03 +08:00
|
|
|
|
2010-12-01 07:51:41 +08:00
|
|
|
// Q registers are encoded as 2x their register number.
|
2010-11-03 06:53:11 +08:00
|
|
|
switch (Reg) {
|
|
|
|
default:
|
|
|
|
return RegNo;
|
|
|
|
case ARM::Q0: case ARM::Q1: case ARM::Q2: case ARM::Q3:
|
|
|
|
case ARM::Q4: case ARM::Q5: case ARM::Q6: case ARM::Q7:
|
|
|
|
case ARM::Q8: case ARM::Q9: case ARM::Q10: case ARM::Q11:
|
|
|
|
case ARM::Q12: case ARM::Q13: case ARM::Q14: case ARM::Q15:
|
|
|
|
return 2 * RegNo;
|
2010-10-22 04:49:13 +08:00
|
|
|
}
|
2010-10-14 10:33:26 +08:00
|
|
|
} else if (MO.isImm()) {
|
2010-10-09 05:45:55 +08:00
|
|
|
return static_cast<unsigned>(MO.getImm());
|
2010-10-14 10:33:26 +08:00
|
|
|
} else if (MO.isFPImm()) {
|
|
|
|
return static_cast<unsigned>(APFloat(MO.getFPImm())
|
|
|
|
.bitcastToAPInt().getHiBits(32).getLimitedValue());
|
2010-11-03 06:53:11 +08:00
|
|
|
}
|
|
|
|
|
2010-11-19 08:27:09 +08:00
|
|
|
llvm_unreachable("Unable to encode MCOperand!");
|
2010-10-09 05:45:55 +08:00
|
|
|
}
|
|
|
|
|
2010-11-03 06:31:46 +08:00
|
|
|
/// getAddrModeImmOpValue - Return encoding info for 'reg +/- imm' operand.
|
2010-11-04 07:52:49 +08:00
|
|
|
bool ARMMCCodeEmitter::
|
|
|
|
EncodeAddrModeOpValues(const MCInst &MI, unsigned OpIdx, unsigned &Reg,
|
2014-01-29 07:13:18 +08:00
|
|
|
unsigned &Imm, SmallVectorImpl<MCFixup> &Fixups,
|
2017-01-28 07:58:02 +08:00
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-10-27 06:37:02 +08:00
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx + 1);
|
2010-11-02 07:45:50 +08:00
|
|
|
|
2013-06-18 15:20:20 +08:00
|
|
|
Reg = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2010-11-03 09:49:29 +08:00
|
|
|
|
|
|
|
int32_t SImm = MO1.getImm();
|
|
|
|
bool isAdd = true;
|
2010-11-03 06:31:46 +08:00
|
|
|
|
2010-10-29 02:34:10 +08:00
|
|
|
// Special value for #-0
|
2011-08-30 03:36:44 +08:00
|
|
|
if (SImm == INT32_MIN) {
|
2010-11-03 09:49:29 +08:00
|
|
|
SImm = 0;
|
2011-08-30 03:36:44 +08:00
|
|
|
isAdd = false;
|
|
|
|
}
|
2010-11-03 06:31:46 +08:00
|
|
|
|
2010-10-29 02:34:10 +08:00
|
|
|
// Immediate is always encoded as positive. The 'U' bit controls add vs sub.
|
2010-11-03 09:49:29 +08:00
|
|
|
if (SImm < 0) {
|
|
|
|
SImm = -SImm;
|
|
|
|
isAdd = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Imm = SImm;
|
|
|
|
return isAdd;
|
|
|
|
}
|
2010-11-03 06:31:46 +08:00
|
|
|
|
2010-12-09 07:01:43 +08:00
|
|
|
/// getBranchTargetOpValue - Helper function to get the branch target operand,
|
|
|
|
/// which is either an immediate or requires a fixup.
|
|
|
|
static uint32_t getBranchTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
|
|
|
unsigned FixupKind,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) {
|
2010-12-07 07:57:07 +08:00
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
|
|
|
|
// If the destination is an immediate, we have nothing to do.
|
|
|
|
if (MO.isImm()) return MO.getImm();
|
2010-12-09 07:01:43 +08:00
|
|
|
assert(MO.isExpr() && "Unexpected branch target type!");
|
2010-12-07 07:57:07 +08:00
|
|
|
const MCExpr *Expr = MO.getExpr();
|
2010-12-09 07:01:43 +08:00
|
|
|
MCFixupKind Kind = MCFixupKind(FixupKind);
|
2015-05-16 03:13:05 +08:00
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
2010-12-07 07:57:07 +08:00
|
|
|
|
|
|
|
// All of the information is in the fixup.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-09-01 02:30:20 +08:00
|
|
|
// Thumb BL and BLX use a strange offset encoding where bits 22 and 21 are
|
|
|
|
// determined by negating them and XOR'ing them with bit 23.
|
|
|
|
static int32_t encodeThumbBLOffset(int32_t offset) {
|
|
|
|
offset >>= 1;
|
|
|
|
uint32_t S = (offset & 0x800000) >> 23;
|
|
|
|
uint32_t J1 = (offset & 0x400000) >> 22;
|
|
|
|
uint32_t J2 = (offset & 0x200000) >> 21;
|
|
|
|
J1 = (~J1 & 0x1);
|
|
|
|
J2 = (~J2 & 0x1);
|
|
|
|
J1 ^= S;
|
|
|
|
J2 ^= S;
|
|
|
|
|
|
|
|
offset &= ~0x600000;
|
|
|
|
offset |= J1 << 22;
|
|
|
|
offset |= J2 << 21;
|
|
|
|
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
2010-12-09 07:01:43 +08:00
|
|
|
/// getThumbBLTargetOpValue - Return encoding info for immediate branch target.
|
2010-11-12 02:04:49 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
2010-12-09 07:01:43 +08:00
|
|
|
getThumbBLTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-09-01 02:30:20 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_thumb_bl,
|
2014-01-29 07:13:18 +08:00
|
|
|
Fixups, STI);
|
2011-09-01 02:30:20 +08:00
|
|
|
return encodeThumbBLOffset(MO.getImm());
|
2010-12-09 07:01:43 +08:00
|
|
|
}
|
2010-11-12 02:04:49 +08:00
|
|
|
|
2010-12-09 08:39:08 +08:00
|
|
|
/// getThumbBLXTargetOpValue - Return encoding info for Thumb immediate
|
|
|
|
/// BLX branch target.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getThumbBLXTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-09-01 02:30:20 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_thumb_blx,
|
2014-01-29 07:13:18 +08:00
|
|
|
Fixups, STI);
|
2011-09-01 02:30:20 +08:00
|
|
|
return encodeThumbBLOffset(MO.getImm());
|
2010-12-09 08:39:08 +08:00
|
|
|
}
|
|
|
|
|
2010-12-11 02:21:33 +08:00
|
|
|
/// getThumbBRTargetOpValue - Return encoding info for Thumb branch target.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getThumbBRTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-31 06:03:20 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
2011-09-01 02:30:20 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_thumb_br,
|
2014-01-29 07:13:18 +08:00
|
|
|
Fixups, STI);
|
2011-08-31 06:03:20 +08:00
|
|
|
return (MO.getImm() >> 1);
|
2010-12-11 02:21:33 +08:00
|
|
|
}
|
|
|
|
|
2010-12-11 01:13:40 +08:00
|
|
|
/// getThumbBCCTargetOpValue - Return encoding info for Thumb branch target.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getThumbBCCTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-09-01 04:26:14 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_thumb_bcc,
|
2014-01-29 07:13:18 +08:00
|
|
|
Fixups, STI);
|
2011-09-01 04:26:14 +08:00
|
|
|
return (MO.getImm() >> 1);
|
2010-12-11 01:13:40 +08:00
|
|
|
}
|
|
|
|
|
2010-12-10 03:04:53 +08:00
|
|
|
/// getThumbCBTargetOpValue - Return encoding info for Thumb branch target.
|
2010-12-09 07:01:43 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
2010-12-10 03:04:53 +08:00
|
|
|
getThumbCBTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-31 06:15:17 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
2014-01-29 07:13:18 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_thumb_cb, Fixups, STI);
|
2011-08-31 06:15:17 +08:00
|
|
|
return (MO.getImm() >> 1);
|
2010-12-09 07:01:43 +08:00
|
|
|
}
|
2010-11-12 02:04:49 +08:00
|
|
|
|
2011-02-05 03:47:15 +08:00
|
|
|
/// Return true if this branch has a non-always predication
|
|
|
|
static bool HasConditionalBranch(const MCInst &MI) {
|
|
|
|
int NumOp = MI.getNumOperands();
|
|
|
|
if (NumOp >= 2) {
|
|
|
|
for (int i = 0; i < NumOp-1; ++i) {
|
|
|
|
const MCOperand &MCOp1 = MI.getOperand(i);
|
|
|
|
const MCOperand &MCOp2 = MI.getOperand(i + 1);
|
2011-08-31 05:58:18 +08:00
|
|
|
if (MCOp1.isImm() && MCOp2.isReg() &&
|
2011-02-05 03:47:15 +08:00
|
|
|
(MCOp2.getReg() == 0 || MCOp2.getReg() == ARM::CPSR)) {
|
2011-08-31 05:58:18 +08:00
|
|
|
if (ARMCC::CondCodes(MCOp1.getImm()) != ARMCC::AL)
|
2011-02-05 03:47:15 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-12-09 07:01:43 +08:00
|
|
|
/// getBranchTargetOpValue - Return encoding info for 24-bit immediate branch
|
|
|
|
/// target.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getBranchTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-12-11 07:41:10 +08:00
|
|
|
// FIXME: This really, really shouldn't use TargetMachine. We don't want
|
|
|
|
// coupling between MC and TM anywhere we can help it.
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI))
|
2010-12-14 03:31:11 +08:00
|
|
|
return
|
2014-01-29 07:13:18 +08:00
|
|
|
::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_t2_condbranch, Fixups, STI);
|
|
|
|
return getARMBranchTargetOpValue(MI, OpIdx, Fixups, STI);
|
2011-02-05 03:47:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// getBranchTargetOpValue - Return encoding info for 24-bit immediate branch
|
|
|
|
/// target.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getARMBranchTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-27 06:54:51 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr()) {
|
2011-08-31 05:58:18 +08:00
|
|
|
if (HasConditionalBranch(MI))
|
2011-08-27 06:54:51 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
ARM::fixup_arm_condbranch, Fixups, STI);
|
2011-08-31 05:58:18 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
ARM::fixup_arm_uncondbranch, Fixups, STI);
|
2011-08-27 06:54:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return MO.getImm() >> 2;
|
2010-11-12 02:04:49 +08:00
|
|
|
}
|
|
|
|
|
2012-02-28 05:36:23 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getARMBLTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2012-02-28 05:36:23 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
2012-03-30 17:15:32 +08:00
|
|
|
if (MO.isExpr()) {
|
|
|
|
if (HasConditionalBranch(MI))
|
2018-07-31 03:41:25 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
ARM::fixup_arm_condbl, Fixups, STI);
|
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_uncondbl, Fixups, STI);
|
2012-03-30 17:15:32 +08:00
|
|
|
}
|
2012-02-28 05:36:23 +08:00
|
|
|
|
|
|
|
return MO.getImm() >> 2;
|
|
|
|
}
|
|
|
|
|
2011-08-27 07:32:08 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getARMBLXTargetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-27 07:32:08 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
2012-02-28 05:36:23 +08:00
|
|
|
if (MO.isExpr())
|
2014-01-29 07:13:18 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_blx, Fixups, STI);
|
2011-02-05 03:47:15 +08:00
|
|
|
|
2011-08-27 07:32:08 +08:00
|
|
|
return MO.getImm() >> 1;
|
|
|
|
}
|
2011-02-05 03:47:15 +08:00
|
|
|
|
2010-12-14 03:31:11 +08:00
|
|
|
/// getUnconditionalBranchTargetOpValue - Return encoding info for 24-bit
|
|
|
|
/// immediate branch target.
|
2016-07-12 06:29:37 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::getThumbBranchTargetOpValue(
|
|
|
|
const MCInst &MI, unsigned OpIdx, SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2013-08-09 18:38:32 +08:00
|
|
|
unsigned Val = 0;
|
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
2018-07-31 03:41:25 +08:00
|
|
|
|
2013-08-09 18:38:32 +08:00
|
|
|
if(MO.isExpr())
|
2014-01-29 07:13:18 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_t2_uncondbranch, Fixups, STI);
|
2018-07-31 03:41:25 +08:00
|
|
|
else
|
2013-08-09 18:38:32 +08:00
|
|
|
Val = MO.getImm() >> 1;
|
|
|
|
|
2010-12-14 03:31:11 +08:00
|
|
|
bool I = (Val & 0x800000);
|
|
|
|
bool J1 = (Val & 0x400000);
|
|
|
|
bool J2 = (Val & 0x200000);
|
|
|
|
if (I ^ J1)
|
|
|
|
Val &= ~0x400000;
|
|
|
|
else
|
|
|
|
Val |= 0x400000;
|
2011-02-09 06:39:40 +08:00
|
|
|
|
2010-12-14 03:31:11 +08:00
|
|
|
if (I ^ J2)
|
|
|
|
Val &= ~0x200000;
|
|
|
|
else
|
|
|
|
Val |= 0x200000;
|
2011-02-09 06:39:40 +08:00
|
|
|
|
2010-12-14 03:31:11 +08:00
|
|
|
return Val;
|
|
|
|
}
|
|
|
|
|
2012-08-02 16:13:13 +08:00
|
|
|
/// getAdrLabelOpValue - Return encoding info for 12-bit shifted-immediate
|
|
|
|
/// ADR label target.
|
2010-12-02 03:47:31 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAdrLabelOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-27 02:09:22 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_adr_pcrel_12,
|
2014-01-29 07:13:18 +08:00
|
|
|
Fixups, STI);
|
2013-08-13 22:02:13 +08:00
|
|
|
int64_t offset = MO.getImm();
|
2011-08-27 02:09:22 +08:00
|
|
|
uint32_t Val = 0x2000;
|
2012-08-02 16:13:13 +08:00
|
|
|
|
2013-02-28 00:43:09 +08:00
|
|
|
int SoImmVal;
|
2012-08-02 16:13:13 +08:00
|
|
|
if (offset == INT32_MIN) {
|
|
|
|
Val = 0x1000;
|
2013-02-28 00:43:09 +08:00
|
|
|
SoImmVal = 0;
|
2012-08-02 16:13:13 +08:00
|
|
|
} else if (offset < 0) {
|
2011-08-27 02:09:22 +08:00
|
|
|
Val = 0x1000;
|
|
|
|
offset *= -1;
|
2013-02-28 00:43:09 +08:00
|
|
|
SoImmVal = ARM_AM::getSOImmVal(offset);
|
|
|
|
if(SoImmVal == -1) {
|
|
|
|
Val = 0x2000;
|
|
|
|
offset *= -1;
|
|
|
|
SoImmVal = ARM_AM::getSOImmVal(offset);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SoImmVal = ARM_AM::getSOImmVal(offset);
|
|
|
|
if(SoImmVal == -1) {
|
|
|
|
Val = 0x1000;
|
|
|
|
offset *= -1;
|
|
|
|
SoImmVal = ARM_AM::getSOImmVal(offset);
|
|
|
|
}
|
2011-08-27 02:09:22 +08:00
|
|
|
}
|
2012-08-02 16:13:13 +08:00
|
|
|
|
|
|
|
assert(SoImmVal != -1 && "Not a valid so_imm value!");
|
|
|
|
|
|
|
|
Val |= SoImmVal;
|
2011-08-27 02:09:22 +08:00
|
|
|
return Val;
|
2010-12-02 03:47:31 +08:00
|
|
|
}
|
|
|
|
|
2012-08-02 16:13:13 +08:00
|
|
|
/// getT2AdrLabelOpValue - Return encoding info for 12-bit immediate ADR label
|
2010-12-14 08:36:49 +08:00
|
|
|
/// target.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getT2AdrLabelOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-27 02:09:22 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_t2_adr_pcrel_12,
|
2014-01-29 07:13:18 +08:00
|
|
|
Fixups, STI);
|
2011-09-10 06:24:36 +08:00
|
|
|
int32_t Val = MO.getImm();
|
2012-08-02 16:13:13 +08:00
|
|
|
if (Val == INT32_MIN)
|
|
|
|
Val = 0x1000;
|
|
|
|
else if (Val < 0) {
|
2011-09-10 06:24:36 +08:00
|
|
|
Val *= -1;
|
|
|
|
Val |= 0x1000;
|
|
|
|
}
|
|
|
|
return Val;
|
2010-12-14 08:36:49 +08:00
|
|
|
}
|
|
|
|
|
2012-08-02 16:13:13 +08:00
|
|
|
/// getThumbAdrLabelOpValue - Return encoding info for 8-bit immediate ADR label
|
2010-12-15 06:28:03 +08:00
|
|
|
/// target.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getThumbAdrLabelOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-27 02:09:22 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_thumb_adr_pcrel_10,
|
2014-01-29 07:13:18 +08:00
|
|
|
Fixups, STI);
|
2011-08-27 02:09:22 +08:00
|
|
|
return MO.getImm();
|
2010-12-15 06:28:03 +08:00
|
|
|
}
|
|
|
|
|
2010-12-14 11:36:38 +08:00
|
|
|
/// getThumbAddrModeRegRegOpValue - Return encoding info for 'reg + reg'
|
|
|
|
/// operand.
|
2010-12-11 06:11:13 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
2010-12-14 11:36:38 +08:00
|
|
|
getThumbAddrModeRegRegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-12-14 11:36:38 +08:00
|
|
|
// [Rn, Rm]
|
|
|
|
// {5-3} = Rm
|
|
|
|
// {2-0} = Rn
|
2010-12-11 06:11:13 +08:00
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx);
|
2010-12-14 11:36:38 +08:00
|
|
|
const MCOperand &MO2 = MI.getOperand(OpIdx + 1);
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Rn = CTX.getRegisterInfo()->getEncodingValue(MO1.getReg());
|
|
|
|
unsigned Rm = CTX.getRegisterInfo()->getEncodingValue(MO2.getReg());
|
2010-12-11 06:11:13 +08:00
|
|
|
return (Rm << 3) | Rn;
|
|
|
|
}
|
|
|
|
|
2010-11-03 09:49:29 +08:00
|
|
|
/// getAddrModeImm12OpValue - Return encoding info for 'reg +/- imm12' operand.
|
2010-11-04 07:52:49 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrModeImm12OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-03 09:49:29 +08:00
|
|
|
// {17-13} = reg
|
|
|
|
// {12} = (U)nsigned (add == '1', sub == '0')
|
|
|
|
// {11-0} = imm12
|
|
|
|
unsigned Reg, Imm12;
|
2010-11-04 09:12:30 +08:00
|
|
|
bool isAdd = true;
|
|
|
|
// If The first operand isn't a register, we have a label reference.
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
2011-02-09 06:39:40 +08:00
|
|
|
if (!MO.isReg()) {
|
2013-06-18 15:20:20 +08:00
|
|
|
Reg = CTX.getRegisterInfo()->getEncodingValue(ARM::PC); // Rn is PC.
|
2010-11-04 09:12:30 +08:00
|
|
|
Imm12 = 0;
|
|
|
|
|
2011-09-13 04:36:51 +08:00
|
|
|
if (MO.isExpr()) {
|
|
|
|
const MCExpr *Expr = MO.getExpr();
|
2013-06-18 16:13:05 +08:00
|
|
|
isAdd = false ; // 'U' bit is set as part of the fixup.
|
2011-09-13 04:36:51 +08:00
|
|
|
|
|
|
|
MCFixupKind Kind;
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI))
|
2011-09-13 04:36:51 +08:00
|
|
|
Kind = MCFixupKind(ARM::fixup_t2_ldst_pcrel_12);
|
|
|
|
else
|
|
|
|
Kind = MCFixupKind(ARM::fixup_arm_ldst_pcrel_12);
|
2015-05-16 03:13:05 +08:00
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
2011-09-13 04:36:51 +08:00
|
|
|
|
|
|
|
++MCNumCPRelocations;
|
|
|
|
} else {
|
|
|
|
Reg = ARM::PC;
|
|
|
|
int32_t Offset = MO.getImm();
|
2013-08-16 20:03:00 +08:00
|
|
|
if (Offset == INT32_MIN) {
|
|
|
|
Offset = 0;
|
|
|
|
isAdd = false;
|
|
|
|
} else if (Offset < 0) {
|
2011-09-13 04:36:51 +08:00
|
|
|
Offset *= -1;
|
|
|
|
isAdd = false;
|
|
|
|
}
|
|
|
|
Imm12 = Offset;
|
|
|
|
}
|
2010-11-04 09:12:30 +08:00
|
|
|
} else
|
2014-01-29 07:13:18 +08:00
|
|
|
isAdd = EncodeAddrModeOpValues(MI, OpIdx, Reg, Imm12, Fixups, STI);
|
2010-11-03 09:49:29 +08:00
|
|
|
|
|
|
|
uint32_t Binary = Imm12 & 0xfff;
|
|
|
|
// Immediate is always encoded as positive. The 'U' bit controls add vs sub.
|
2010-10-29 02:34:10 +08:00
|
|
|
if (isAdd)
|
2010-11-03 09:49:29 +08:00
|
|
|
Binary |= (1 << 12);
|
|
|
|
Binary |= (Reg << 13);
|
|
|
|
return Binary;
|
|
|
|
}
|
|
|
|
|
2011-09-09 06:07:06 +08:00
|
|
|
/// getT2Imm8s4OpValue - Return encoding info for
|
|
|
|
/// '+/- imm8<<2' operand.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getT2Imm8s4OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-09-09 06:07:06 +08:00
|
|
|
// FIXME: The immediate operand should have already been encoded like this
|
|
|
|
// before ever getting here. The encoder method should just need to combine
|
|
|
|
// the MI operands for the register and the offset into a single
|
|
|
|
// representation for the complex operand in the .td file. This isn't just
|
|
|
|
// style, unfortunately. As-is, we can't represent the distinct encoding
|
|
|
|
// for #-0.
|
|
|
|
|
|
|
|
// {8} = (U)nsigned (add == '1', sub == '0')
|
|
|
|
// {7-0} = imm8
|
|
|
|
int32_t Imm8 = MI.getOperand(OpIdx).getImm();
|
|
|
|
bool isAdd = Imm8 >= 0;
|
|
|
|
|
|
|
|
// Immediate is always encoded as positive. The 'U' bit controls add vs sub.
|
|
|
|
if (Imm8 < 0)
|
2012-08-24 08:35:46 +08:00
|
|
|
Imm8 = -(uint32_t)Imm8;
|
2011-09-09 06:07:06 +08:00
|
|
|
|
|
|
|
// Scaled by 4.
|
|
|
|
Imm8 /= 4;
|
|
|
|
|
|
|
|
uint32_t Binary = Imm8 & 0xff;
|
|
|
|
// Immediate is always encoded as positive. The 'U' bit controls add vs sub.
|
|
|
|
if (isAdd)
|
|
|
|
Binary |= (1 << 8);
|
|
|
|
return Binary;
|
|
|
|
}
|
|
|
|
|
2010-12-02 03:18:46 +08:00
|
|
|
/// getT2AddrModeImm8s4OpValue - Return encoding info for
|
|
|
|
/// 'reg +/- imm8<<2' operand.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getT2AddrModeImm8s4OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-12-11 05:05:07 +08:00
|
|
|
// {12-9} = reg
|
|
|
|
// {8} = (U)nsigned (add == '1', sub == '0')
|
|
|
|
// {7-0} = imm8
|
2010-12-02 03:18:46 +08:00
|
|
|
unsigned Reg, Imm8;
|
|
|
|
bool isAdd = true;
|
|
|
|
// If The first operand isn't a register, we have a label reference.
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
if (!MO.isReg()) {
|
2013-06-18 15:20:20 +08:00
|
|
|
Reg = CTX.getRegisterInfo()->getEncodingValue(ARM::PC); // Rn is PC.
|
2010-12-02 03:18:46 +08:00
|
|
|
Imm8 = 0;
|
|
|
|
isAdd = false ; // 'U' bit is set as part of the fixup.
|
|
|
|
|
|
|
|
assert(MO.isExpr() && "Unexpected machine operand type!");
|
|
|
|
const MCExpr *Expr = MO.getExpr();
|
2011-12-20 07:06:24 +08:00
|
|
|
MCFixupKind Kind = MCFixupKind(ARM::fixup_t2_pcrel_10);
|
2015-05-16 03:13:05 +08:00
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
2010-12-02 03:18:46 +08:00
|
|
|
|
|
|
|
++MCNumCPRelocations;
|
|
|
|
} else
|
2014-01-29 07:13:18 +08:00
|
|
|
isAdd = EncodeAddrModeOpValues(MI, OpIdx, Reg, Imm8, Fixups, STI);
|
2010-12-02 03:18:46 +08:00
|
|
|
|
2011-09-09 06:07:06 +08:00
|
|
|
// FIXME: The immediate operand should have already been encoded like this
|
|
|
|
// before ever getting here. The encoder method should just need to combine
|
|
|
|
// the MI operands for the register and the offset into a single
|
|
|
|
// representation for the complex operand in the .td file. This isn't just
|
|
|
|
// style, unfortunately. As-is, we can't represent the distinct encoding
|
|
|
|
// for #-0.
|
2010-12-02 03:18:46 +08:00
|
|
|
uint32_t Binary = (Imm8 >> 2) & 0xff;
|
|
|
|
// Immediate is always encoded as positive. The 'U' bit controls add vs sub.
|
|
|
|
if (isAdd)
|
2010-12-11 05:05:07 +08:00
|
|
|
Binary |= (1 << 8);
|
2010-12-02 03:18:46 +08:00
|
|
|
Binary |= (Reg << 9);
|
|
|
|
return Binary;
|
|
|
|
}
|
|
|
|
|
2011-09-10 02:37:27 +08:00
|
|
|
/// getT2AddrModeImm0_1020s4OpValue - Return encoding info for
|
|
|
|
/// 'reg + imm8<<2' operand.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getT2AddrModeImm0_1020s4OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-09-10 02:37:27 +08:00
|
|
|
// {11-8} = reg
|
|
|
|
// {7-0} = imm8
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx + 1);
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Reg = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2011-09-10 02:37:27 +08:00
|
|
|
unsigned Imm8 = MO1.getImm();
|
|
|
|
return (Reg << 8) | Imm8;
|
|
|
|
}
|
|
|
|
|
2011-01-13 15:58:56 +08:00
|
|
|
uint32_t
|
|
|
|
ARMMCCodeEmitter::getHiLo16ImmOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-19 07:37:15 +08:00
|
|
|
// {20-16} = imm{15-12}
|
|
|
|
// {11-0} = imm{11-0}
|
2010-12-11 05:57:34 +08:00
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
2011-01-13 15:58:56 +08:00
|
|
|
if (MO.isImm())
|
|
|
|
// Hi / lo 16 bits already extracted during earlier passes.
|
2010-11-19 07:37:15 +08:00
|
|
|
return static_cast<unsigned>(MO.getImm());
|
2011-01-13 15:58:56 +08:00
|
|
|
|
|
|
|
// Handle :upper16: and :lower16: assembly prefixes.
|
|
|
|
const MCExpr *E = MO.getExpr();
|
2012-05-02 04:43:21 +08:00
|
|
|
MCFixupKind Kind;
|
2011-01-13 15:58:56 +08:00
|
|
|
if (E->getKind() == MCExpr::Target) {
|
|
|
|
const ARMMCExpr *ARM16Expr = cast<ARMMCExpr>(E);
|
|
|
|
E = ARM16Expr->getSubExpr();
|
|
|
|
|
2014-01-12 07:03:48 +08:00
|
|
|
if (const MCConstantExpr *MCE = dyn_cast<MCConstantExpr>(E)) {
|
|
|
|
const int64_t Value = MCE->getValue();
|
|
|
|
if (Value > UINT32_MAX)
|
|
|
|
report_fatal_error("constant value truncated (limited to 32-bit)");
|
|
|
|
|
|
|
|
switch (ARM16Expr->getKind()) {
|
|
|
|
case ARMMCExpr::VK_ARM_HI16:
|
|
|
|
return (int32_t(Value) & 0xffff0000) >> 16;
|
|
|
|
case ARMMCExpr::VK_ARM_LO16:
|
|
|
|
return (int32_t(Value) & 0x0000ffff);
|
|
|
|
default: llvm_unreachable("Unsupported ARMFixup");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-13 15:58:56 +08:00
|
|
|
switch (ARM16Expr->getKind()) {
|
2012-02-07 10:50:20 +08:00
|
|
|
default: llvm_unreachable("Unsupported ARMFixup");
|
2011-01-13 15:58:56 +08:00
|
|
|
case ARMMCExpr::VK_ARM_HI16:
|
2016-01-15 18:25:14 +08:00
|
|
|
Kind = MCFixupKind(isThumb(STI) ? ARM::fixup_t2_movt_hi16
|
|
|
|
: ARM::fixup_arm_movt_hi16);
|
2010-11-19 07:37:15 +08:00
|
|
|
break;
|
2011-01-13 15:58:56 +08:00
|
|
|
case ARMMCExpr::VK_ARM_LO16:
|
2016-01-15 18:25:14 +08:00
|
|
|
Kind = MCFixupKind(isThumb(STI) ? ARM::fixup_t2_movw_lo16
|
|
|
|
: ARM::fixup_arm_movw_lo16);
|
2010-11-19 07:37:15 +08:00
|
|
|
break;
|
|
|
|
}
|
2014-04-27 11:48:22 +08:00
|
|
|
|
2015-05-16 03:13:05 +08:00
|
|
|
Fixups.push_back(MCFixup::create(0, E, Kind, MI.getLoc()));
|
2010-11-19 07:37:15 +08:00
|
|
|
return 0;
|
2012-05-02 04:43:21 +08:00
|
|
|
}
|
|
|
|
// If the expression doesn't have :upper16: or :lower16: on it,
|
2014-04-19 07:06:39 +08:00
|
|
|
// it's just a plain immediate expression, previously those evaluated to
|
2012-05-02 04:43:21 +08:00
|
|
|
// the lower 16 bits of the expression regardless of whether
|
2014-04-19 07:06:39 +08:00
|
|
|
// we have a movt or a movw, but that led to misleadingly results.
|
2015-06-19 09:53:21 +08:00
|
|
|
// This is disallowed in the AsmParser in validateInstruction()
|
2014-04-19 07:06:39 +08:00
|
|
|
// so this should never happen.
|
2014-06-19 14:10:58 +08:00
|
|
|
llvm_unreachable("expression without :upper16: or :lower16:");
|
2010-11-19 07:37:15 +08:00
|
|
|
}
|
|
|
|
|
2010-11-10 01:20:53 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getLdStSORegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-10 01:20:53 +08:00
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx+1);
|
|
|
|
const MCOperand &MO2 = MI.getOperand(OpIdx+2);
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Rn = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
|
|
|
unsigned Rm = CTX.getRegisterInfo()->getEncodingValue(MO1.getReg());
|
2010-11-10 01:20:53 +08:00
|
|
|
unsigned ShImm = ARM_AM::getAM2Offset(MO2.getImm());
|
|
|
|
bool isAdd = ARM_AM::getAM2Op(MO2.getImm()) == ARM_AM::add;
|
2010-11-16 04:47:07 +08:00
|
|
|
ARM_AM::ShiftOpc ShOp = ARM_AM::getAM2ShiftOpc(MO2.getImm());
|
|
|
|
unsigned SBits = getShiftOp(ShOp);
|
2010-11-10 01:20:53 +08:00
|
|
|
|
2012-09-22 19:18:12 +08:00
|
|
|
// While "lsr #32" and "asr #32" exist, they are encoded with a 0 in the shift
|
|
|
|
// amount. However, it would be an easy mistake to make so check here.
|
|
|
|
assert((ShImm & ~0x1f) == 0 && "Out of range shift amount");
|
|
|
|
|
2010-11-10 01:20:53 +08:00
|
|
|
// {16-13} = Rn
|
|
|
|
// {12} = isAdd
|
|
|
|
// {11-0} = shifter
|
|
|
|
// {3-0} = Rm
|
|
|
|
// {4} = 0
|
|
|
|
// {6-5} = type
|
|
|
|
// {11-7} = imm
|
2010-11-11 09:09:40 +08:00
|
|
|
uint32_t Binary = Rm;
|
2010-11-10 01:20:53 +08:00
|
|
|
Binary |= Rn << 13;
|
|
|
|
Binary |= SBits << 5;
|
|
|
|
Binary |= ShImm << 7;
|
|
|
|
if (isAdd)
|
|
|
|
Binary |= 1 << 12;
|
|
|
|
return Binary;
|
|
|
|
}
|
|
|
|
|
2010-11-16 04:47:07 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrMode2OffsetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-16 04:47:07 +08:00
|
|
|
// {13} 1 == imm12, 0 == Rm
|
|
|
|
// {12} isAdd
|
|
|
|
// {11-0} imm12/Rm
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx+1);
|
|
|
|
unsigned Imm = MO1.getImm();
|
|
|
|
bool isAdd = ARM_AM::getAM2Op(Imm) == ARM_AM::add;
|
|
|
|
bool isReg = MO.getReg() != 0;
|
|
|
|
uint32_t Binary = ARM_AM::getAM2Offset(Imm);
|
|
|
|
// if reg +/- reg, Rm will be non-zero. Otherwise, we have reg +/- imm12
|
|
|
|
if (isReg) {
|
|
|
|
ARM_AM::ShiftOpc ShOp = ARM_AM::getAM2ShiftOpc(Imm);
|
|
|
|
Binary <<= 7; // Shift amount is bits [11:7]
|
|
|
|
Binary |= getShiftOp(ShOp) << 5; // Shift type is bits [6:5]
|
2013-06-18 15:20:20 +08:00
|
|
|
Binary |= CTX.getRegisterInfo()->getEncodingValue(MO.getReg()); // Rm is bits [3:0]
|
2010-11-16 04:47:07 +08:00
|
|
|
}
|
|
|
|
return Binary | (isAdd << 12) | (isReg << 13);
|
|
|
|
}
|
|
|
|
|
2011-08-04 07:50:40 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getPostIdxRegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-04 07:50:40 +08:00
|
|
|
// {4} isAdd
|
|
|
|
// {3-0} Rm
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx+1);
|
2011-08-06 00:11:38 +08:00
|
|
|
bool isAdd = MO1.getImm() != 0;
|
2013-06-18 15:20:20 +08:00
|
|
|
return CTX.getRegisterInfo()->getEncodingValue(MO.getReg()) | (isAdd << 4);
|
2011-08-04 07:50:40 +08:00
|
|
|
}
|
|
|
|
|
2010-11-12 00:55:29 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrMode3OffsetOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-12 00:55:29 +08:00
|
|
|
// {9} 1 == imm8, 0 == Rm
|
|
|
|
// {8} isAdd
|
|
|
|
// {7-4} imm7_4/zero
|
|
|
|
// {3-0} imm3_0/Rm
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx+1);
|
|
|
|
unsigned Imm = MO1.getImm();
|
|
|
|
bool isAdd = ARM_AM::getAM3Op(Imm) == ARM_AM::add;
|
|
|
|
bool isImm = MO.getReg() == 0;
|
|
|
|
uint32_t Imm8 = ARM_AM::getAM3Offset(Imm);
|
|
|
|
// if reg +/- reg, Rm will be non-zero. Otherwise, we have reg +/- imm8
|
|
|
|
if (!isImm)
|
2013-06-18 15:20:20 +08:00
|
|
|
Imm8 = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2010-11-12 00:55:29 +08:00
|
|
|
return Imm8 | (isAdd << 8) | (isImm << 9);
|
|
|
|
}
|
|
|
|
|
2010-11-11 09:09:40 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrMode3OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-11 09:09:40 +08:00
|
|
|
// {13} 1 == imm8, 0 == Rm
|
|
|
|
// {12-9} Rn
|
|
|
|
// {8} isAdd
|
|
|
|
// {7-4} imm7_4/zero
|
|
|
|
// {3-0} imm3_0/Rm
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx+1);
|
|
|
|
const MCOperand &MO2 = MI.getOperand(OpIdx+2);
|
2011-12-20 07:06:24 +08:00
|
|
|
|
|
|
|
// If The first operand isn't a register, we have a label reference.
|
|
|
|
if (!MO.isReg()) {
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Rn = CTX.getRegisterInfo()->getEncodingValue(ARM::PC); // Rn is PC.
|
2011-12-20 07:06:24 +08:00
|
|
|
|
|
|
|
assert(MO.isExpr() && "Unexpected machine operand type!");
|
|
|
|
const MCExpr *Expr = MO.getExpr();
|
|
|
|
MCFixupKind Kind = MCFixupKind(ARM::fixup_arm_pcrel_10_unscaled);
|
2015-05-16 03:13:05 +08:00
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
2011-12-20 07:06:24 +08:00
|
|
|
|
|
|
|
++MCNumCPRelocations;
|
|
|
|
return (Rn << 9) | (1 << 13);
|
|
|
|
}
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Rn = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2010-11-11 09:09:40 +08:00
|
|
|
unsigned Imm = MO2.getImm();
|
|
|
|
bool isAdd = ARM_AM::getAM3Op(Imm) == ARM_AM::add;
|
|
|
|
bool isImm = MO1.getReg() == 0;
|
|
|
|
uint32_t Imm8 = ARM_AM::getAM3Offset(Imm);
|
|
|
|
// if reg +/- reg, Rm will be non-zero. Otherwise, we have reg +/- imm8
|
|
|
|
if (!isImm)
|
2013-06-18 15:20:20 +08:00
|
|
|
Imm8 = CTX.getRegisterInfo()->getEncodingValue(MO1.getReg());
|
2010-11-11 09:09:40 +08:00
|
|
|
return (Rn << 9) | Imm8 | (isAdd << 8) | (isImm << 13);
|
|
|
|
}
|
|
|
|
|
2010-12-08 09:57:09 +08:00
|
|
|
/// getAddrModeThumbSPOpValue - Encode the t_addrmode_sp operands.
|
2010-12-08 05:50:47 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrModeThumbSPOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-12-08 05:50:47 +08:00
|
|
|
// [SP, #imm]
|
|
|
|
// {7-0} = imm8
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx + 1);
|
2010-12-08 09:57:09 +08:00
|
|
|
assert(MI.getOperand(OpIdx).getReg() == ARM::SP &&
|
|
|
|
"Unexpected base register!");
|
2010-12-16 07:32:27 +08:00
|
|
|
|
2010-12-08 05:50:47 +08:00
|
|
|
// The immediate is already shifted for the implicit zeroes, so no change
|
|
|
|
// here.
|
|
|
|
return MO1.getImm() & 0xff;
|
|
|
|
}
|
|
|
|
|
2010-12-14 11:36:38 +08:00
|
|
|
/// getAddrModeISOpValue - Encode the t_addrmode_is# operands.
|
2010-12-10 05:49:07 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
2010-12-14 11:36:38 +08:00
|
|
|
getAddrModeISOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-30 15:44:32 +08:00
|
|
|
// [Rn, #imm]
|
|
|
|
// {7-3} = imm5
|
|
|
|
// {2-0} = Rn
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx + 1);
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Rn = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2010-12-16 09:34:26 +08:00
|
|
|
unsigned Imm5 = MO1.getImm();
|
2010-12-10 05:49:07 +08:00
|
|
|
return ((Imm5 & 0x1f) << 3) | Rn;
|
2010-12-01 06:57:21 +08:00
|
|
|
}
|
|
|
|
|
2010-12-08 09:57:09 +08:00
|
|
|
/// getAddrModePCOpValue - Return encoding for t_addrmode_pc operands.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrModePCOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-08-31 06:10:03 +08:00
|
|
|
const MCOperand MO = MI.getOperand(OpIdx);
|
|
|
|
if (MO.isExpr())
|
2014-01-29 07:13:18 +08:00
|
|
|
return ::getBranchTargetOpValue(MI, OpIdx, ARM::fixup_arm_thumb_cp, Fixups, STI);
|
2011-08-31 06:10:03 +08:00
|
|
|
return (MO.getImm() >> 2);
|
2010-12-08 09:57:09 +08:00
|
|
|
}
|
|
|
|
|
2016-01-25 18:26:26 +08:00
|
|
|
/// getAddrMode5OpValue - Return encoding info for 'reg +/- (imm8 << 2)' operand.
|
2010-11-04 07:52:49 +08:00
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrMode5OpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-03 09:49:29 +08:00
|
|
|
// {12-9} = reg
|
|
|
|
// {8} = (U)nsigned (add == '1', sub == '0')
|
|
|
|
// {7-0} = imm8
|
|
|
|
unsigned Reg, Imm8;
|
2010-12-01 06:40:36 +08:00
|
|
|
bool isAdd;
|
2010-11-04 09:12:30 +08:00
|
|
|
// If The first operand isn't a register, we have a label reference.
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
if (!MO.isReg()) {
|
2013-06-18 15:20:20 +08:00
|
|
|
Reg = CTX.getRegisterInfo()->getEncodingValue(ARM::PC); // Rn is PC.
|
2010-11-04 09:12:30 +08:00
|
|
|
Imm8 = 0;
|
2010-12-01 06:40:36 +08:00
|
|
|
isAdd = false; // 'U' bit is handled as part of the fixup.
|
2010-11-04 09:12:30 +08:00
|
|
|
|
|
|
|
assert(MO.isExpr() && "Unexpected machine operand type!");
|
|
|
|
const MCExpr *Expr = MO.getExpr();
|
2010-12-08 08:18:36 +08:00
|
|
|
MCFixupKind Kind;
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb2(STI))
|
2010-12-08 08:18:36 +08:00
|
|
|
Kind = MCFixupKind(ARM::fixup_t2_pcrel_10);
|
|
|
|
else
|
|
|
|
Kind = MCFixupKind(ARM::fixup_arm_pcrel_10);
|
2015-05-16 03:13:05 +08:00
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
2010-11-04 09:12:30 +08:00
|
|
|
|
|
|
|
++MCNumCPRelocations;
|
2010-12-01 06:40:36 +08:00
|
|
|
} else {
|
2014-01-29 07:13:18 +08:00
|
|
|
EncodeAddrModeOpValues(MI, OpIdx, Reg, Imm8, Fixups, STI);
|
2010-12-01 06:40:36 +08:00
|
|
|
isAdd = ARM_AM::getAM5Op(Imm8) == ARM_AM::add;
|
|
|
|
}
|
2010-11-03 09:49:29 +08:00
|
|
|
|
|
|
|
uint32_t Binary = ARM_AM::getAM5Offset(Imm8);
|
|
|
|
// Immediate is always encoded as positive. The 'U' bit controls add vs sub.
|
2010-12-01 06:40:36 +08:00
|
|
|
if (isAdd)
|
2010-11-03 09:49:29 +08:00
|
|
|
Binary |= (1 << 8);
|
|
|
|
Binary |= (Reg << 9);
|
2010-10-27 06:37:02 +08:00
|
|
|
return Binary;
|
|
|
|
}
|
|
|
|
|
2016-01-25 18:26:26 +08:00
|
|
|
/// getAddrMode5FP16OpValue - Return encoding info for 'reg +/- (imm8 << 1)' operand.
|
|
|
|
uint32_t ARMMCCodeEmitter::
|
|
|
|
getAddrMode5FP16OpValue(const MCInst &MI, unsigned OpIdx,
|
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
|
|
|
// {12-9} = reg
|
|
|
|
// {8} = (U)nsigned (add == '1', sub == '0')
|
|
|
|
// {7-0} = imm8
|
|
|
|
unsigned Reg, Imm8;
|
|
|
|
bool isAdd;
|
|
|
|
// If The first operand isn't a register, we have a label reference.
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
if (!MO.isReg()) {
|
|
|
|
Reg = CTX.getRegisterInfo()->getEncodingValue(ARM::PC); // Rn is PC.
|
|
|
|
Imm8 = 0;
|
|
|
|
isAdd = false; // 'U' bit is handled as part of the fixup.
|
|
|
|
|
|
|
|
assert(MO.isExpr() && "Unexpected machine operand type!");
|
|
|
|
const MCExpr *Expr = MO.getExpr();
|
|
|
|
MCFixupKind Kind;
|
|
|
|
if (isThumb2(STI))
|
|
|
|
Kind = MCFixupKind(ARM::fixup_t2_pcrel_9);
|
|
|
|
else
|
|
|
|
Kind = MCFixupKind(ARM::fixup_arm_pcrel_9);
|
|
|
|
Fixups.push_back(MCFixup::create(0, Expr, Kind, MI.getLoc()));
|
|
|
|
|
|
|
|
++MCNumCPRelocations;
|
|
|
|
} else {
|
|
|
|
EncodeAddrModeOpValues(MI, OpIdx, Reg, Imm8, Fixups, STI);
|
|
|
|
isAdd = ARM_AM::getAM5Op(Imm8) == ARM_AM::add;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t Binary = ARM_AM::getAM5Offset(Imm8);
|
|
|
|
// Immediate is always encoded as positive. The 'U' bit controls add vs sub.
|
|
|
|
if (isAdd)
|
|
|
|
Binary |= (1 << 8);
|
|
|
|
Binary |= (Reg << 9);
|
|
|
|
return Binary;
|
|
|
|
}
|
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
2011-07-22 07:38:37 +08:00
|
|
|
getSORegRegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-03 06:53:11 +08:00
|
|
|
// Sub-operands are [reg, reg, imm]. The first register is Rm, the reg to be
|
2011-07-29 01:56:55 +08:00
|
|
|
// shifted. The second is Rs, the amount to shift by, and the third specifies
|
|
|
|
// the type of the shift.
|
2010-11-04 06:03:20 +08:00
|
|
|
//
|
2010-10-13 07:53:58 +08:00
|
|
|
// {3-0} = Rm.
|
2011-07-29 01:56:55 +08:00
|
|
|
// {4} = 1
|
2010-10-13 07:53:58 +08:00
|
|
|
// {6-5} = type
|
2011-07-29 01:56:55 +08:00
|
|
|
// {11-8} = Rs
|
|
|
|
// {7} = 0
|
2010-10-13 07:53:58 +08:00
|
|
|
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx + 1);
|
|
|
|
const MCOperand &MO2 = MI.getOperand(OpIdx + 2);
|
|
|
|
ARM_AM::ShiftOpc SOpc = ARM_AM::getSORegShOp(MO2.getImm());
|
|
|
|
|
|
|
|
// Encode Rm.
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Binary = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2010-10-13 07:53:58 +08:00
|
|
|
|
|
|
|
// Encode the shift opcode.
|
|
|
|
unsigned SBits = 0;
|
|
|
|
unsigned Rs = MO1.getReg();
|
|
|
|
if (Rs) {
|
|
|
|
// Set shift operand (bit[7:4]).
|
|
|
|
// LSL - 0001
|
|
|
|
// LSR - 0011
|
|
|
|
// ASR - 0101
|
|
|
|
// ROR - 0111
|
|
|
|
switch (SOpc) {
|
|
|
|
default: llvm_unreachable("Unknown shift opc!");
|
|
|
|
case ARM_AM::lsl: SBits = 0x1; break;
|
|
|
|
case ARM_AM::lsr: SBits = 0x3; break;
|
|
|
|
case ARM_AM::asr: SBits = 0x5; break;
|
|
|
|
case ARM_AM::ror: SBits = 0x7; break;
|
|
|
|
}
|
|
|
|
}
|
2010-11-03 06:53:11 +08:00
|
|
|
|
2010-10-13 07:53:58 +08:00
|
|
|
Binary |= SBits << 4;
|
|
|
|
|
2011-07-29 01:56:55 +08:00
|
|
|
// Encode the shift operation Rs.
|
2011-07-22 07:38:37 +08:00
|
|
|
// Encode Rs bit[11:8].
|
|
|
|
assert(ARM_AM::getSORegOffset(MO2.getImm()) == 0);
|
2013-06-18 15:20:20 +08:00
|
|
|
return Binary | (CTX.getRegisterInfo()->getEncodingValue(Rs) << ARMII::RegRsShift);
|
2011-07-22 07:38:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getSORegImmOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-07-29 01:56:55 +08:00
|
|
|
// Sub-operands are [reg, imm]. The first register is Rm, the reg to be
|
|
|
|
// shifted. The second is the amount to shift by.
|
2011-07-22 07:38:37 +08:00
|
|
|
//
|
|
|
|
// {3-0} = Rm.
|
2011-07-29 01:56:55 +08:00
|
|
|
// {4} = 0
|
2011-07-22 07:38:37 +08:00
|
|
|
// {6-5} = type
|
2011-07-29 01:56:55 +08:00
|
|
|
// {11-7} = imm
|
2011-07-22 07:38:37 +08:00
|
|
|
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx + 1);
|
|
|
|
ARM_AM::ShiftOpc SOpc = ARM_AM::getSORegShOp(MO1.getImm());
|
|
|
|
|
|
|
|
// Encode Rm.
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Binary = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2011-07-22 07:38:37 +08:00
|
|
|
|
|
|
|
// Encode the shift opcode.
|
|
|
|
unsigned SBits = 0;
|
|
|
|
|
|
|
|
// Set shift operand (bit[6:4]).
|
|
|
|
// LSL - 000
|
|
|
|
// LSR - 010
|
|
|
|
// ASR - 100
|
|
|
|
// ROR - 110
|
|
|
|
// RRX - 110 and bit[11:8] clear.
|
|
|
|
switch (SOpc) {
|
|
|
|
default: llvm_unreachable("Unknown shift opc!");
|
|
|
|
case ARM_AM::lsl: SBits = 0x0; break;
|
|
|
|
case ARM_AM::lsr: SBits = 0x2; break;
|
|
|
|
case ARM_AM::asr: SBits = 0x4; break;
|
|
|
|
case ARM_AM::ror: SBits = 0x6; break;
|
|
|
|
case ARM_AM::rrx:
|
|
|
|
Binary |= 0x60;
|
|
|
|
return Binary;
|
2010-10-13 07:53:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Encode shift_imm bit[11:7].
|
2011-07-22 07:38:37 +08:00
|
|
|
Binary |= SBits << 4;
|
2011-08-12 02:41:59 +08:00
|
|
|
unsigned Offset = ARM_AM::getSORegOffset(MO1.getImm());
|
2012-04-26 02:00:18 +08:00
|
|
|
assert(Offset < 32 && "Offset must be in range 0-31!");
|
2011-08-12 02:41:59 +08:00
|
|
|
return Binary | (Offset << 7);
|
2010-10-13 07:53:58 +08:00
|
|
|
}
|
|
|
|
|
2011-07-22 07:38:37 +08:00
|
|
|
|
2010-11-30 06:44:32 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getT2AddrModeSORegOpValue(const MCInst &MI, unsigned OpNum,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-30 06:44:32 +08:00
|
|
|
const MCOperand &MO1 = MI.getOperand(OpNum);
|
|
|
|
const MCOperand &MO2 = MI.getOperand(OpNum+1);
|
2010-12-11 05:57:34 +08:00
|
|
|
const MCOperand &MO3 = MI.getOperand(OpNum+2);
|
|
|
|
|
2010-11-30 06:44:32 +08:00
|
|
|
// Encoded as [Rn, Rm, imm].
|
|
|
|
// FIXME: Needs fixup support.
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Value = CTX.getRegisterInfo()->getEncodingValue(MO1.getReg());
|
2010-11-30 06:44:32 +08:00
|
|
|
Value <<= 4;
|
2013-06-18 15:20:20 +08:00
|
|
|
Value |= CTX.getRegisterInfo()->getEncodingValue(MO2.getReg());
|
2010-11-30 06:44:32 +08:00
|
|
|
Value <<= 2;
|
|
|
|
Value |= MO3.getImm();
|
2010-12-11 05:57:34 +08:00
|
|
|
|
2010-11-30 06:44:32 +08:00
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getT2AddrModeImm8OpValue(const MCInst &MI, unsigned OpNum,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-30 06:44:32 +08:00
|
|
|
const MCOperand &MO1 = MI.getOperand(OpNum);
|
|
|
|
const MCOperand &MO2 = MI.getOperand(OpNum+1);
|
|
|
|
|
|
|
|
// FIXME: Needs fixup support.
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Value = CTX.getRegisterInfo()->getEncodingValue(MO1.getReg());
|
2010-12-11 05:57:34 +08:00
|
|
|
|
2010-11-30 06:44:32 +08:00
|
|
|
// Even though the immediate is 8 bits long, we need 9 bits in order
|
|
|
|
// to represent the (inverse of the) sign bit.
|
|
|
|
Value <<= 9;
|
2010-11-30 08:14:31 +08:00
|
|
|
int32_t tmp = (int32_t)MO2.getImm();
|
|
|
|
if (tmp < 0)
|
|
|
|
tmp = abs(tmp);
|
|
|
|
else
|
|
|
|
Value |= 256; // Set the ADD bit
|
|
|
|
Value |= tmp & 255;
|
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getT2AddrModeImm8OffsetOpValue(const MCInst &MI, unsigned OpNum,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-30 08:14:31 +08:00
|
|
|
const MCOperand &MO1 = MI.getOperand(OpNum);
|
|
|
|
|
|
|
|
// FIXME: Needs fixup support.
|
|
|
|
unsigned Value = 0;
|
|
|
|
int32_t tmp = (int32_t)MO1.getImm();
|
|
|
|
if (tmp < 0)
|
|
|
|
tmp = abs(tmp);
|
|
|
|
else
|
|
|
|
Value |= 256; // Set the ADD bit
|
|
|
|
Value |= tmp & 255;
|
2010-11-30 06:44:32 +08:00
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
|
2010-11-13 05:12:40 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getT2SORegOpValue(const MCInst &MI, unsigned OpIdx,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-13 05:12:40 +08:00
|
|
|
// Sub-operands are [reg, imm]. The first register is Rm, the reg to be
|
|
|
|
// shifted. The second is the amount to shift by.
|
|
|
|
//
|
|
|
|
// {3-0} = Rm.
|
|
|
|
// {4} = 0
|
|
|
|
// {6-5} = type
|
|
|
|
// {11-7} = imm
|
|
|
|
|
|
|
|
const MCOperand &MO = MI.getOperand(OpIdx);
|
|
|
|
const MCOperand &MO1 = MI.getOperand(OpIdx + 1);
|
|
|
|
ARM_AM::ShiftOpc SOpc = ARM_AM::getSORegShOp(MO1.getImm());
|
|
|
|
|
|
|
|
// Encode Rm.
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned Binary = CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2010-11-13 05:12:40 +08:00
|
|
|
|
|
|
|
// Encode the shift opcode.
|
|
|
|
unsigned SBits = 0;
|
|
|
|
// Set shift operand (bit[6:4]).
|
|
|
|
// LSL - 000
|
|
|
|
// LSR - 010
|
|
|
|
// ASR - 100
|
|
|
|
// ROR - 110
|
|
|
|
switch (SOpc) {
|
|
|
|
default: llvm_unreachable("Unknown shift opc!");
|
|
|
|
case ARM_AM::lsl: SBits = 0x0; break;
|
|
|
|
case ARM_AM::lsr: SBits = 0x2; break;
|
|
|
|
case ARM_AM::asr: SBits = 0x4; break;
|
2016-08-17 13:10:15 +08:00
|
|
|
case ARM_AM::rrx: LLVM_FALLTHROUGH;
|
2010-11-13 05:12:40 +08:00
|
|
|
case ARM_AM::ror: SBits = 0x6; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
Binary |= SBits << 4;
|
|
|
|
if (SOpc == ARM_AM::rrx)
|
|
|
|
return Binary;
|
|
|
|
|
|
|
|
// Encode shift_imm bit[11:7].
|
|
|
|
return Binary | ARM_AM::getSORegOffset(MO1.getImm()) << 7;
|
|
|
|
}
|
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getBitfieldInvertedMaskOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2018-04-09 12:37:53 +08:00
|
|
|
// 10 bits. lower 5 bits are the lsb of the mask, high five bits are the
|
2010-10-22 06:03:21 +08:00
|
|
|
// msb of the mask.
|
|
|
|
const MCOperand &MO = MI.getOperand(Op);
|
|
|
|
uint32_t v = ~MO.getImm();
|
2013-05-25 06:23:49 +08:00
|
|
|
uint32_t lsb = countTrailingZeros(v);
|
|
|
|
uint32_t msb = (32 - countLeadingZeros (v)) - 1;
|
2017-01-28 07:58:02 +08:00
|
|
|
assert(v != 0 && lsb < 32 && msb < 32 && "Illegal bitfield mask!");
|
2010-10-22 06:03:21 +08:00
|
|
|
return lsb | (msb << 5);
|
|
|
|
}
|
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getRegisterListOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-17 08:45:23 +08:00
|
|
|
// VLDM/VSTM:
|
|
|
|
// {12-8} = Vd
|
|
|
|
// {7-0} = Number of registers
|
|
|
|
//
|
|
|
|
// LDM/STM:
|
|
|
|
// {15-0} = Bitfield of GPRs.
|
|
|
|
unsigned Reg = MI.getOperand(Op).getReg();
|
2012-03-27 15:21:54 +08:00
|
|
|
bool SPRRegs = ARMMCRegisterClasses[ARM::SPRRegClassID].contains(Reg);
|
|
|
|
bool DPRRegs = ARMMCRegisterClasses[ARM::DPRRegClassID].contains(Reg);
|
2010-11-17 08:45:23 +08:00
|
|
|
|
2010-11-09 08:30:18 +08:00
|
|
|
unsigned Binary = 0;
|
2010-11-17 08:45:23 +08:00
|
|
|
|
|
|
|
if (SPRRegs || DPRRegs) {
|
|
|
|
// VLDM/VSTM
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned RegNo = CTX.getRegisterInfo()->getEncodingValue(Reg);
|
2010-11-17 08:45:23 +08:00
|
|
|
unsigned NumRegs = (MI.getNumOperands() - Op) & 0xff;
|
|
|
|
Binary |= (RegNo & 0x1f) << 8;
|
|
|
|
if (SPRRegs)
|
|
|
|
Binary |= NumRegs;
|
|
|
|
else
|
|
|
|
Binary |= NumRegs * 2;
|
|
|
|
} else {
|
Recommit: ARM: sort register lists by encoding in push/pop instructions.
For example we were producing
push {r8, r10, r11, r4, r5, r7, lr}
This is misleading (r4, r5 and r7 are actually pushed before the rest), and
other components (stack folding recently) often forget to deal with the extra
complexity coming from the different order, leading to miscompiles. Finally, we
warn about our own code in -no-integrated-as mode without this, which is really
not a good idea.
Fixed usage of std::sort so that we (hopefully) use instantiations that
actually exist in GCC 4.8.
llvm-svn: 286881
2016-11-15 04:28:24 +08:00
|
|
|
const MCRegisterInfo &MRI = *CTX.getRegisterInfo();
|
|
|
|
assert(std::is_sorted(MI.begin() + Op, MI.end(),
|
|
|
|
[&](const MCOperand &LHS, const MCOperand &RHS) {
|
|
|
|
return MRI.getEncodingValue(LHS.getReg()) <
|
|
|
|
MRI.getEncodingValue(RHS.getReg());
|
|
|
|
}));
|
|
|
|
|
2010-11-17 08:45:23 +08:00
|
|
|
for (unsigned I = Op, E = MI.getNumOperands(); I < E; ++I) {
|
Recommit: ARM: sort register lists by encoding in push/pop instructions.
For example we were producing
push {r8, r10, r11, r4, r5, r7, lr}
This is misleading (r4, r5 and r7 are actually pushed before the rest), and
other components (stack folding recently) often forget to deal with the extra
complexity coming from the different order, leading to miscompiles. Finally, we
warn about our own code in -no-integrated-as mode without this, which is really
not a good idea.
Fixed usage of std::sort so that we (hopefully) use instantiations that
actually exist in GCC 4.8.
llvm-svn: 286881
2016-11-15 04:28:24 +08:00
|
|
|
unsigned RegNo = MRI.getEncodingValue(MI.getOperand(I).getReg());
|
2010-11-17 08:45:23 +08:00
|
|
|
Binary |= 1 << RegNo;
|
|
|
|
}
|
2010-11-09 08:30:18 +08:00
|
|
|
}
|
2010-11-17 08:45:23 +08:00
|
|
|
|
2010-10-30 08:37:59 +08:00
|
|
|
return Binary;
|
|
|
|
}
|
|
|
|
|
2010-11-30 08:00:42 +08:00
|
|
|
/// getAddrMode6AddressOpValue - Encode an addrmode6 register number along
|
|
|
|
/// with the alignment operand.
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getAddrMode6AddressOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-02 08:05:05 +08:00
|
|
|
const MCOperand &Reg = MI.getOperand(Op);
|
2010-11-03 06:53:11 +08:00
|
|
|
const MCOperand &Imm = MI.getOperand(Op + 1);
|
2010-11-04 06:03:20 +08:00
|
|
|
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned RegNo = CTX.getRegisterInfo()->getEncodingValue(Reg.getReg());
|
2010-11-03 06:53:11 +08:00
|
|
|
unsigned Align = 0;
|
|
|
|
|
|
|
|
switch (Imm.getImm()) {
|
|
|
|
default: break;
|
|
|
|
case 2:
|
|
|
|
case 4:
|
|
|
|
case 8: Align = 0x01; break;
|
|
|
|
case 16: Align = 0x02; break;
|
|
|
|
case 32: Align = 0x03; break;
|
2010-11-02 08:05:05 +08:00
|
|
|
}
|
2010-11-03 06:53:11 +08:00
|
|
|
|
2010-11-02 08:05:05 +08:00
|
|
|
return RegNo | (Align << 4);
|
|
|
|
}
|
|
|
|
|
2011-05-10 01:47:27 +08:00
|
|
|
/// getAddrMode6OneLane32AddressOpValue - Encode an addrmode6 register number
|
|
|
|
/// along with the alignment operand for use in VST1 and VLD1 with size 32.
|
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getAddrMode6OneLane32AddressOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-05-10 01:47:27 +08:00
|
|
|
const MCOperand &Reg = MI.getOperand(Op);
|
|
|
|
const MCOperand &Imm = MI.getOperand(Op + 1);
|
|
|
|
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned RegNo = CTX.getRegisterInfo()->getEncodingValue(Reg.getReg());
|
2011-05-10 01:47:27 +08:00
|
|
|
unsigned Align = 0;
|
|
|
|
|
|
|
|
switch (Imm.getImm()) {
|
|
|
|
default: break;
|
|
|
|
case 8:
|
2011-12-20 02:31:43 +08:00
|
|
|
case 16:
|
|
|
|
case 32: // Default '0' value for invalid alignments of 8, 16, 32 bytes.
|
|
|
|
case 2: Align = 0x00; break;
|
|
|
|
case 4: Align = 0x03; break;
|
2011-05-10 01:47:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return RegNo | (Align << 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-30 08:00:42 +08:00
|
|
|
/// getAddrMode6DupAddressOpValue - Encode an addrmode6 register number and
|
|
|
|
/// alignment operand for use in VLD-dup instructions. This is the same as
|
|
|
|
/// getAddrMode6AddressOpValue except for the alignment encoding, which is
|
|
|
|
/// different for VLD4-dup.
|
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getAddrMode6DupAddressOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-30 08:00:42 +08:00
|
|
|
const MCOperand &Reg = MI.getOperand(Op);
|
|
|
|
const MCOperand &Imm = MI.getOperand(Op + 1);
|
|
|
|
|
2013-06-18 15:20:20 +08:00
|
|
|
unsigned RegNo = CTX.getRegisterInfo()->getEncodingValue(Reg.getReg());
|
2010-11-30 08:00:42 +08:00
|
|
|
unsigned Align = 0;
|
|
|
|
|
|
|
|
switch (Imm.getImm()) {
|
|
|
|
default: break;
|
|
|
|
case 2:
|
|
|
|
case 4:
|
|
|
|
case 8: Align = 0x01; break;
|
|
|
|
case 16: Align = 0x03; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return RegNo | (Align << 4);
|
|
|
|
}
|
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getAddrMode6OffsetOpValue(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-11-03 06:53:11 +08:00
|
|
|
const MCOperand &MO = MI.getOperand(Op);
|
|
|
|
if (MO.getReg() == 0) return 0x0D;
|
2013-06-18 15:20:20 +08:00
|
|
|
return CTX.getRegisterInfo()->getEncodingValue(MO.getReg());
|
2011-03-01 09:00:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ARMMCCodeEmitter::
|
2011-03-08 07:38:41 +08:00
|
|
|
getShiftRight8Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-03-01 09:00:59 +08:00
|
|
|
return 8 - MI.getOperand(Op).getImm();
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ARMMCCodeEmitter::
|
2011-03-08 07:38:41 +08:00
|
|
|
getShiftRight16Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-03-01 09:00:59 +08:00
|
|
|
return 16 - MI.getOperand(Op).getImm();
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ARMMCCodeEmitter::
|
2011-03-08 07:38:41 +08:00
|
|
|
getShiftRight32Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-03-01 09:00:59 +08:00
|
|
|
return 32 - MI.getOperand(Op).getImm();
|
2011-03-08 07:38:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ARMMCCodeEmitter::
|
|
|
|
getShiftRight64Imm(const MCInst &MI, unsigned Op,
|
2014-01-29 07:13:18 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2011-03-08 07:38:41 +08:00
|
|
|
return 64 - MI.getOperand(Op).getImm();
|
2010-11-02 09:24:55 +08:00
|
|
|
}
|
|
|
|
|
2010-09-18 02:46:17 +08:00
|
|
|
void ARMMCCodeEmitter::
|
2015-05-16 03:13:16 +08:00
|
|
|
encodeInstruction(const MCInst &MI, raw_ostream &OS,
|
2014-01-29 07:13:07 +08:00
|
|
|
SmallVectorImpl<MCFixup> &Fixups,
|
|
|
|
const MCSubtargetInfo &STI) const {
|
2010-10-08 06:12:50 +08:00
|
|
|
// Pseudo instructions don't get encoded.
|
2011-07-11 11:57:24 +08:00
|
|
|
const MCInstrDesc &Desc = MCII.get(MI.getOpcode());
|
2010-11-12 07:41:09 +08:00
|
|
|
uint64_t TSFlags = Desc.TSFlags;
|
|
|
|
if ((TSFlags & ARMII::FormMask) == ARMII::Pseudo)
|
2010-10-08 06:12:50 +08:00
|
|
|
return;
|
2011-07-14 07:22:26 +08:00
|
|
|
|
2010-11-12 07:41:09 +08:00
|
|
|
int Size;
|
2011-07-14 07:22:26 +08:00
|
|
|
if (Desc.getSize() == 2 || Desc.getSize() == 4)
|
|
|
|
Size = Desc.getSize();
|
|
|
|
else
|
|
|
|
llvm_unreachable("Unexpected instruction size!");
|
2011-08-31 05:58:18 +08:00
|
|
|
|
2014-01-29 07:13:18 +08:00
|
|
|
uint32_t Binary = getBinaryCodeForInstr(MI, Fixups, STI);
|
2011-01-13 15:58:56 +08:00
|
|
|
// Thumb 32-bit wide instructions need to emit the high order halfword
|
|
|
|
// first.
|
2014-01-29 07:13:25 +08:00
|
|
|
if (isThumb(STI) && Size == 4) {
|
2010-12-04 06:31:40 +08:00
|
|
|
EmitConstant(Binary >> 16, 2, OS);
|
|
|
|
EmitConstant(Binary & 0xffff, 2, OS);
|
|
|
|
} else
|
|
|
|
EmitConstant(Binary, Size, OS);
|
2010-11-03 06:44:12 +08:00
|
|
|
++MCNumEmitted; // Keep track of the # of mi's emitted.
|
2010-09-18 02:46:17 +08:00
|
|
|
}
|
2010-10-08 05:57:55 +08:00
|
|
|
|
2010-11-04 07:52:49 +08:00
|
|
|
#include "ARMGenMCCodeEmitter.inc"
|
2017-01-28 07:58:02 +08:00
|
|
|
|
|
|
|
MCCodeEmitter *llvm::createARMLEMCCodeEmitter(const MCInstrInfo &MCII,
|
|
|
|
const MCRegisterInfo &MRI,
|
|
|
|
MCContext &Ctx) {
|
|
|
|
return new ARMMCCodeEmitter(MCII, Ctx, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
MCCodeEmitter *llvm::createARMBEMCCodeEmitter(const MCInstrInfo &MCII,
|
|
|
|
const MCRegisterInfo &MRI,
|
|
|
|
MCContext &Ctx) {
|
|
|
|
return new ARMMCCodeEmitter(MCII, Ctx, false);
|
|
|
|
}
|