2006-05-15 06:18:28 +08:00
|
|
|
//===-- ARMAsmPrinter.cpp - ARM LLVM assembly writer ----------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-30 04:36:04 +08:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
2006-05-15 06:18:28 +08:00
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains a printer that converts from our internal representation
|
|
|
|
// of machine-dependent LLVM code to GAS-format ARM assembly language.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2006-12-20 06:59:26 +08:00
|
|
|
#define DEBUG_TYPE "asm-printer"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include "ARM.h"
|
2009-05-24 03:51:20 +08:00
|
|
|
#include "ARMBuildAttrs.h"
|
2007-01-19 15:51:42 +08:00
|
|
|
#include "ARMTargetMachine.h"
|
|
|
|
#include "ARMAddressingModes.h"
|
|
|
|
#include "ARMConstantPoolValue.h"
|
|
|
|
#include "ARMMachineFunctionInfo.h"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include "llvm/Constants.h"
|
|
|
|
#include "llvm/Module.h"
|
2009-06-25 08:47:42 +08:00
|
|
|
#include "llvm/MDNode.h"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include "llvm/CodeGen/AsmPrinter.h"
|
2007-01-19 15:51:42 +08:00
|
|
|
#include "llvm/CodeGen/DwarfWriter.h"
|
2007-01-27 05:22:28 +08:00
|
|
|
#include "llvm/CodeGen/MachineModuleInfo.h"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
2007-01-19 15:51:42 +08:00
|
|
|
#include "llvm/CodeGen/MachineJumpTableInfo.h"
|
2006-09-07 02:34:40 +08:00
|
|
|
#include "llvm/Target/TargetAsmInfo.h"
|
2006-07-27 19:38:51 +08:00
|
|
|
#include "llvm/Target/TargetData.h"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include "llvm/Target/TargetMachine.h"
|
2007-01-20 03:25:36 +08:00
|
|
|
#include "llvm/Target/TargetOptions.h"
|
2009-07-16 04:24:03 +08:00
|
|
|
#include "llvm/Target/TargetRegistry.h"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2008-12-05 09:06:39 +08:00
|
|
|
#include "llvm/ADT/StringSet.h"
|
2007-01-19 15:51:42 +08:00
|
|
|
#include "llvm/Support/Compiler.h"
|
2009-07-09 04:55:50 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2007-01-19 15:51:42 +08:00
|
|
|
#include "llvm/Support/Mangler.h"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
2009-07-15 04:18:05 +08:00
|
|
|
#include "llvm/Support/FormattedStream.h"
|
2006-05-15 06:18:28 +08:00
|
|
|
#include <cctype>
|
|
|
|
using namespace llvm;
|
|
|
|
|
2006-12-20 06:59:26 +08:00
|
|
|
STATISTIC(EmittedInsts, "Number of machine instrs printed");
|
2006-05-15 06:18:28 +08:00
|
|
|
|
2006-12-20 06:59:26 +08:00
|
|
|
namespace {
|
2009-02-24 16:30:20 +08:00
|
|
|
class VISIBILITY_HIDDEN ARMAsmPrinter : public AsmPrinter {
|
2009-01-09 07:40:34 +08:00
|
|
|
DwarfWriter *DW;
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
/// Subtarget - Keep a pointer to the ARMSubtarget around so that we can
|
|
|
|
/// make the right decision when printing asm code for different targets.
|
|
|
|
const ARMSubtarget *Subtarget;
|
|
|
|
|
|
|
|
/// AFI - Keep a pointer to ARMFunctionInfo for the current
|
2008-09-18 15:27:23 +08:00
|
|
|
/// MachineFunction.
|
2007-01-19 15:51:42 +08:00
|
|
|
ARMFunctionInfo *AFI;
|
|
|
|
|
2008-09-18 15:27:23 +08:00
|
|
|
/// MCP - Keep a pointer to constantpool entries of the current
|
|
|
|
/// MachineFunction.
|
|
|
|
const MachineConstantPool *MCP;
|
|
|
|
|
2006-05-15 06:18:28 +08:00
|
|
|
/// We name each basic block in a Function with a unique number, so
|
|
|
|
/// that we can consistently refer to them later. This is cleared
|
|
|
|
/// at the beginning of each call to runOnMachineFunction().
|
|
|
|
///
|
|
|
|
typedef std::map<const Value *, unsigned> ValueMapTy;
|
|
|
|
ValueMapTy NumberForBB;
|
|
|
|
|
2008-08-08 14:56:16 +08:00
|
|
|
/// GVNonLazyPtrs - Keeps the set of GlobalValues that require
|
|
|
|
/// non-lazy-pointers for indirect access.
|
2009-07-15 11:12:43 +08:00
|
|
|
StringMap<std::string> GVNonLazyPtrs;
|
2008-12-05 09:06:39 +08:00
|
|
|
|
|
|
|
/// HiddenGVNonLazyPtrs - Keeps the set of GlobalValues with hidden
|
|
|
|
/// visibility that require non-lazy-pointers for indirect access.
|
2009-07-15 11:12:43 +08:00
|
|
|
StringMap<std::string> HiddenGVNonLazyPtrs;
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2009-07-15 12:41:01 +08:00
|
|
|
struct FnStubInfo {
|
|
|
|
std::string Stub, LazyPtr, SLP, SCV;
|
|
|
|
|
|
|
|
FnStubInfo() {}
|
|
|
|
|
|
|
|
void Init(const GlobalValue *GV, Mangler *Mang) {
|
|
|
|
// Already initialized.
|
|
|
|
if (!Stub.empty()) return;
|
|
|
|
Stub = Mang->getMangledName(GV, "$stub", true);
|
|
|
|
LazyPtr = Mang->getMangledName(GV, "$lazy_ptr", true);
|
|
|
|
SLP = Mang->getMangledName(GV, "$slp", true);
|
|
|
|
SCV = Mang->getMangledName(GV, "$scv", true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Init(const std::string &GV, Mangler *Mang) {
|
|
|
|
// Already initialized.
|
|
|
|
if (!Stub.empty()) return;
|
2009-07-21 03:41:27 +08:00
|
|
|
Stub = Mang->makeNameProper(GV + "$stub", Mangler::Private);
|
|
|
|
LazyPtr = Mang->makeNameProper(GV + "$lazy_ptr", Mangler::Private);
|
|
|
|
SLP = Mang->makeNameProper(GV + "$slp", Mangler::Private);
|
|
|
|
SCV = Mang->makeNameProper(GV + "$scv", Mangler::Private);
|
2009-07-15 12:41:01 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2008-08-08 14:56:16 +08:00
|
|
|
/// FnStubs - Keeps the set of external function GlobalAddresses that the
|
|
|
|
/// asm printer should generate stubs for.
|
2009-07-15 12:41:01 +08:00
|
|
|
StringMap<FnStubInfo> FnStubs;
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
/// True if asm printer is printing a series of CONSTPOOL_ENTRY.
|
|
|
|
bool InCPMode;
|
2009-02-24 16:30:20 +08:00
|
|
|
public:
|
2009-07-15 04:18:05 +08:00
|
|
|
explicit ARMAsmPrinter(formatted_raw_ostream &O, TargetMachine &TM,
|
2009-07-01 09:48:54 +08:00
|
|
|
const TargetAsmInfo *T, bool V)
|
|
|
|
: AsmPrinter(O, TM, T, V), DW(0), AFI(NULL), MCP(NULL),
|
2009-02-24 16:30:20 +08:00
|
|
|
InCPMode(false) {
|
|
|
|
Subtarget = &TM.getSubtarget<ARMSubtarget>();
|
|
|
|
}
|
|
|
|
|
2006-05-15 06:18:28 +08:00
|
|
|
virtual const char *getPassName() const {
|
|
|
|
return "ARM Assembly Printer";
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void printOperand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
const char *Modifier = 0);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printSOImmOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printSOImm2PartOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printSORegOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printAddrMode2Operand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printAddrMode2OffsetOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printAddrMode3Operand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printAddrMode3OffsetOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printAddrMode4Operand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
const char *Modifier = 0);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printAddrMode5Operand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
const char *Modifier = 0);
|
2009-07-02 07:16:05 +08:00
|
|
|
void printAddrMode6Operand(const MachineInstr *MI, int OpNum);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printAddrModePCOperand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
const char *Modifier = 0);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printBitfieldInvMaskImmOperand (const MachineInstr *MI, int OpNum);
|
2009-06-27 10:26:13 +08:00
|
|
|
|
2009-07-10 07:43:36 +08:00
|
|
|
void printThumbITMask(const MachineInstr *MI, int OpNum);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printThumbAddrModeRROperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printThumbAddrModeRI5Operand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
unsigned Scale);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printThumbAddrModeS1Operand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printThumbAddrModeS2Operand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printThumbAddrModeS4Operand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printThumbAddrModeSPOperand(const MachineInstr *MI, int OpNum);
|
2009-06-27 10:26:13 +08:00
|
|
|
|
|
|
|
void printT2SOOperand(const MachineInstr *MI, int OpNum);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printT2AddrModeImm12Operand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printT2AddrModeImm8Operand(const MachineInstr *MI, int OpNum);
|
2009-07-10 06:21:59 +08:00
|
|
|
void printT2AddrModeImm8s4Operand(const MachineInstr *MI, int OpNum);
|
2009-07-02 15:28:31 +08:00
|
|
|
void printT2AddrModeImm8OffsetOperand(const MachineInstr *MI, int OpNum);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printT2AddrModeSoRegOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
|
|
|
|
void printPredicateOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printSBitModifierOperand(const MachineInstr *MI, int OpNum);
|
|
|
|
void printPCLabel(const MachineInstr *MI, int OpNum);
|
|
|
|
void printRegisterList(const MachineInstr *MI, int OpNum);
|
|
|
|
void printCPInstOperand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
const char *Modifier);
|
2009-06-29 15:51:04 +08:00
|
|
|
void printJTBlockOperand(const MachineInstr *MI, int OpNum);
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
virtual bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
unsigned AsmVariant, const char *ExtraCode);
|
2009-06-29 15:51:04 +08:00
|
|
|
virtual bool PrintAsmMemoryOperand(const MachineInstr *MI, unsigned OpNum,
|
2009-05-19 13:53:42 +08:00
|
|
|
unsigned AsmVariant,
|
|
|
|
const char *ExtraCode);
|
2006-05-15 06:18:28 +08:00
|
|
|
|
2009-07-22 02:38:57 +08:00
|
|
|
void PrintGlobalVariable(const GlobalVariable* GVar);
|
2006-05-15 06:18:28 +08:00
|
|
|
bool printInstruction(const MachineInstr *MI); // autogenerated.
|
2007-01-19 15:51:42 +08:00
|
|
|
void printMachineInstruction(const MachineInstr *MI);
|
2006-05-15 06:18:28 +08:00
|
|
|
bool runOnMachineFunction(MachineFunction &F);
|
|
|
|
bool doInitialization(Module &M);
|
|
|
|
bool doFinalization(Module &M);
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2008-08-08 14:56:16 +08:00
|
|
|
/// EmitMachineConstantPoolValue - Print a machine constantpool value to
|
|
|
|
/// the .s file.
|
2007-01-19 15:51:42 +08:00
|
|
|
virtual void EmitMachineConstantPoolValue(MachineConstantPoolValue *MCPV) {
|
|
|
|
printDataDirective(MCPV->getType());
|
|
|
|
|
2008-08-08 14:56:16 +08:00
|
|
|
ARMConstantPoolValue *ACPV = static_cast<ARMConstantPoolValue*>(MCPV);
|
2007-01-27 03:51:32 +08:00
|
|
|
GlobalValue *GV = ACPV->getGV();
|
2009-07-15 02:17:16 +08:00
|
|
|
std::string Name;
|
2009-07-15 11:12:43 +08:00
|
|
|
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
if (ACPV->isNonLazyPointer()) {
|
2009-07-15 11:12:43 +08:00
|
|
|
std::string SymName = Mang->getMangledName(GV);
|
|
|
|
Name = Mang->getMangledName(GV, "$non_lazy_ptr", true);
|
|
|
|
|
2008-12-05 09:06:39 +08:00
|
|
|
if (GV->hasHiddenVisibility())
|
2009-07-15 11:12:43 +08:00
|
|
|
HiddenGVNonLazyPtrs[SymName] = Name;
|
2008-12-05 09:06:39 +08:00
|
|
|
else
|
2009-07-15 11:12:43 +08:00
|
|
|
GVNonLazyPtrs[SymName] = Name;
|
2007-01-31 04:37:08 +08:00
|
|
|
} else if (ACPV->isStub()) {
|
2009-07-15 12:41:01 +08:00
|
|
|
if (GV) {
|
|
|
|
FnStubInfo &FnInfo = FnStubs[Mang->getMangledName(GV)];
|
|
|
|
FnInfo.Init(GV, Mang);
|
|
|
|
Name = FnInfo.Stub;
|
|
|
|
} else {
|
|
|
|
FnStubInfo &FnInfo = FnStubs[Mang->makeNameProper(ACPV->getSymbol())];
|
|
|
|
FnInfo.Init(ACPV->getSymbol(), Mang);
|
|
|
|
Name = FnInfo.Stub;
|
|
|
|
}
|
2009-07-15 11:12:43 +08:00
|
|
|
} else {
|
|
|
|
if (GV)
|
|
|
|
Name = Mang->getMangledName(GV);
|
|
|
|
else
|
|
|
|
Name = Mang->makeNameProper(ACPV->getSymbol());
|
|
|
|
}
|
2009-07-15 12:41:01 +08:00
|
|
|
O << Name;
|
2009-07-15 11:12:43 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
2007-04-22 08:04:12 +08:00
|
|
|
if (ACPV->hasModifier()) O << "(" << ACPV->getModifier() << ")";
|
2007-04-27 21:54:47 +08:00
|
|
|
if (ACPV->getPCAdjustment() != 0) {
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "-(" << TAI->getPrivateGlobalPrefix() << "PC"
|
|
|
|
<< utostr(ACPV->getLabelId())
|
2007-04-27 21:54:47 +08:00
|
|
|
<< "+" << (unsigned)ACPV->getPCAdjustment();
|
|
|
|
if (ACPV->mustAddCurrentAddress())
|
|
|
|
O << "-.";
|
|
|
|
O << ")";
|
|
|
|
}
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const {
|
2007-09-30 21:39:29 +08:00
|
|
|
AsmPrinter::getAnalysisUsage(AU);
|
2007-01-19 15:51:42 +08:00
|
|
|
AU.setPreservesAll();
|
2007-01-27 05:22:28 +08:00
|
|
|
AU.addRequired<MachineModuleInfo>();
|
2009-01-09 07:40:34 +08:00
|
|
|
AU.addRequired<DwarfWriter>();
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
2006-05-15 06:18:28 +08:00
|
|
|
};
|
|
|
|
} // end of anonymous namespace
|
|
|
|
|
|
|
|
#include "ARMGenAsmWriter.inc"
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
/// runOnMachineFunction - This uses the printInstruction()
|
2006-05-15 06:18:28 +08:00
|
|
|
/// method to print assembly for each instruction.
|
|
|
|
///
|
|
|
|
bool ARMAsmPrinter::runOnMachineFunction(MachineFunction &MF) {
|
2009-02-24 16:30:20 +08:00
|
|
|
this->MF = &MF;
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
AFI = MF.getInfo<ARMFunctionInfo>();
|
2008-09-18 15:27:23 +08:00
|
|
|
MCP = MF.getConstantPool();
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
SetupMachineFunction(MF);
|
|
|
|
O << "\n";
|
|
|
|
|
|
|
|
// NOTE: we don't print out constant pools here, they are handled as
|
|
|
|
// instructions.
|
2006-05-23 10:48:20 +08:00
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "\n";
|
2006-05-23 10:48:20 +08:00
|
|
|
// Print out labels for the function.
|
|
|
|
const Function *F = MF.getFunction();
|
|
|
|
switch (F->getLinkage()) {
|
2009-07-15 00:55:14 +08:00
|
|
|
default: llvm_unreachable("Unknown linkage type!");
|
2009-01-16 04:18:42 +08:00
|
|
|
case Function::PrivateLinkage:
|
2009-07-20 09:03:30 +08:00
|
|
|
case Function::LinkerPrivateLinkage:
|
2006-05-23 10:48:20 +08:00
|
|
|
case Function::InternalLinkage:
|
2007-01-19 15:51:42 +08:00
|
|
|
SwitchToTextSection("\t.text", F);
|
2006-05-23 10:48:20 +08:00
|
|
|
break;
|
|
|
|
case Function::ExternalLinkage:
|
2007-01-19 15:51:42 +08:00
|
|
|
SwitchToTextSection("\t.text", F);
|
2006-05-23 10:48:20 +08:00
|
|
|
O << "\t.globl\t" << CurrentFnName << "\n";
|
|
|
|
break;
|
Introduce new linkage types linkonce_odr, weak_odr, common_odr
and extern_weak_odr. These are the same as the non-odr versions,
except that they indicate that the global will only be overridden
by an *equivalent* global. In C, a function with weak linkage can
be overridden by a function which behaves completely differently.
This means that IP passes have to skip weak functions, since any
deductions made from the function definition might be wrong, since
the definition could be replaced by something completely different
at link time. This is not allowed in C++, thanks to the ODR
(One-Definition-Rule): if a function is replaced by another at
link-time, then the new function must be the same as the original
function. If a language knows that a function or other global can
only be overridden by an equivalent global, it can give it the
weak_odr linkage type, and the optimizers will understand that it
is alright to make deductions based on the function body. The
code generators on the other hand map weak and weak_odr linkage
to the same thing.
llvm-svn: 66339
2009-03-07 23:45:40 +08:00
|
|
|
case Function::WeakAnyLinkage:
|
|
|
|
case Function::WeakODRLinkage:
|
|
|
|
case Function::LinkOnceAnyLinkage:
|
|
|
|
case Function::LinkOnceODRLinkage:
|
2007-01-20 03:25:36 +08:00
|
|
|
if (Subtarget->isTargetDarwin()) {
|
2007-01-19 15:51:42 +08:00
|
|
|
SwitchToTextSection(
|
|
|
|
".section __TEXT,__textcoal_nt,coalesced,pure_instructions", F);
|
|
|
|
O << "\t.globl\t" << CurrentFnName << "\n";
|
|
|
|
O << "\t.weak_definition\t" << CurrentFnName << "\n";
|
|
|
|
} else {
|
|
|
|
O << TAI->getWeakRefDirective() << CurrentFnName << "\n";
|
|
|
|
}
|
2006-05-23 10:48:20 +08:00
|
|
|
break;
|
|
|
|
}
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2008-08-09 02:25:07 +08:00
|
|
|
printVisibility(CurrentFnName, F->getVisibility());
|
2007-03-29 15:49:34 +08:00
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
if (AFI->isThumbFunction()) {
|
2009-07-01 06:38:32 +08:00
|
|
|
EmitAlignment(MF.getAlignment(), F, AFI->getAlign());
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "\t.code\t16\n";
|
2007-02-02 02:25:34 +08:00
|
|
|
O << "\t.thumb_func";
|
|
|
|
if (Subtarget->isTargetDarwin())
|
|
|
|
O << "\t" << CurrentFnName;
|
|
|
|
O << "\n";
|
2007-01-19 15:51:42 +08:00
|
|
|
InCPMode = false;
|
2009-07-01 06:38:32 +08:00
|
|
|
} else {
|
|
|
|
EmitAlignment(MF.getAlignment(), F);
|
|
|
|
}
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2006-05-23 10:48:20 +08:00
|
|
|
O << CurrentFnName << ":\n";
|
2007-05-04 04:28:35 +08:00
|
|
|
// Emit pre-function debug information.
|
2009-01-09 07:40:34 +08:00
|
|
|
DW->BeginFunction(&MF);
|
2006-05-23 10:48:20 +08:00
|
|
|
|
2008-01-28 17:15:03 +08:00
|
|
|
if (Subtarget->isTargetDarwin()) {
|
|
|
|
// If the function is empty, then we need to emit *something*. Otherwise,
|
|
|
|
// the function's label might be associated with something that it wasn't
|
|
|
|
// meant to be associated with. We emit a noop in this situation.
|
|
|
|
MachineFunction::iterator I = MF.begin();
|
|
|
|
|
|
|
|
if (++I == MF.end() && MF.front().empty())
|
|
|
|
O << "\tnop\n";
|
|
|
|
}
|
|
|
|
|
2006-05-23 10:48:20 +08:00
|
|
|
// Print out code for the function.
|
|
|
|
for (MachineFunction::const_iterator I = MF.begin(), E = MF.end();
|
|
|
|
I != E; ++I) {
|
|
|
|
// Print a label for the basic block.
|
|
|
|
if (I != MF.begin()) {
|
2009-03-24 08:17:40 +08:00
|
|
|
printBasicBlockLabel(I, true, true, VerboseAsm);
|
2006-05-23 10:48:20 +08:00
|
|
|
O << '\n';
|
|
|
|
}
|
|
|
|
for (MachineBasicBlock::const_iterator II = I->begin(), E = I->end();
|
|
|
|
II != E; ++II) {
|
|
|
|
// Print the assembly for the instruction.
|
2007-01-19 15:51:42 +08:00
|
|
|
printMachineInstruction(II);
|
2006-09-13 20:09:43 +08:00
|
|
|
}
|
2006-09-12 01:25:40 +08:00
|
|
|
}
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
if (TAI->hasDotTypeDotSizeDirective())
|
|
|
|
O << "\t.size " << CurrentFnName << ", .-" << CurrentFnName << "\n";
|
2006-11-09 01:07:32 +08:00
|
|
|
|
2007-05-04 04:28:35 +08:00
|
|
|
// Emit post-function debug information.
|
2009-01-09 07:40:34 +08:00
|
|
|
DW->EndFunction(&MF);
|
2006-10-18 02:04:53 +08:00
|
|
|
|
2008-11-08 03:49:17 +08:00
|
|
|
O.flush();
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
return false;
|
2006-10-18 02:04:53 +08:00
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printOperand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
const char *Modifier) {
|
2009-06-29 15:51:04 +08:00
|
|
|
const MachineOperand &MO = MI->getOperand(OpNum);
|
2006-05-25 20:57:06 +08:00
|
|
|
switch (MO.getType()) {
|
2009-06-23 07:27:02 +08:00
|
|
|
case MachineOperand::MO_Register: {
|
|
|
|
unsigned Reg = MO.getReg();
|
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(Reg)) {
|
|
|
|
if (Modifier && strcmp(Modifier, "dregpair") == 0) {
|
|
|
|
unsigned DRegLo = TRI->getSubReg(Reg, 5); // arm_dsubreg_0
|
|
|
|
unsigned DRegHi = TRI->getSubReg(Reg, 6); // arm_dsubreg_1
|
|
|
|
O << '{'
|
2009-07-09 02:11:30 +08:00
|
|
|
<< TRI->getAsmName(DRegLo) << ',' << TRI->getAsmName(DRegHi)
|
2009-06-23 07:27:02 +08:00
|
|
|
<< '}';
|
2009-07-09 02:11:30 +08:00
|
|
|
} else if (Modifier && strcmp(Modifier, "dregsingle") == 0) {
|
|
|
|
O << '{' << TRI->getAsmName(Reg) << '}';
|
2009-06-23 07:27:02 +08:00
|
|
|
} else {
|
|
|
|
O << TRI->getAsmName(Reg);
|
|
|
|
}
|
|
|
|
} else
|
2009-07-15 00:55:14 +08:00
|
|
|
llvm_unreachable("not implemented");
|
2006-05-25 20:57:06 +08:00
|
|
|
break;
|
2009-06-23 07:27:02 +08:00
|
|
|
}
|
2007-01-19 15:51:42 +08:00
|
|
|
case MachineOperand::MO_Immediate: {
|
|
|
|
if (!Modifier || strcmp(Modifier, "no_hash") != 0)
|
|
|
|
O << "#";
|
|
|
|
|
2009-05-20 05:27:57 +08:00
|
|
|
O << MO.getImm();
|
2006-05-25 20:57:06 +08:00
|
|
|
break;
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
2006-05-25 20:57:06 +08:00
|
|
|
case MachineOperand::MO_MachineBasicBlock:
|
2007-12-31 07:10:15 +08:00
|
|
|
printBasicBlockLabel(MO.getMBB());
|
2006-05-25 20:57:06 +08:00
|
|
|
return;
|
2006-07-16 09:02:57 +08:00
|
|
|
case MachineOperand::MO_GlobalAddress: {
|
2007-01-19 15:51:42 +08:00
|
|
|
bool isCallOp = Modifier && !strcmp(Modifier, "call");
|
2006-07-16 09:02:57 +08:00
|
|
|
GlobalValue *GV = MO.getGlobal();
|
2009-07-15 12:41:01 +08:00
|
|
|
std::string Name;
|
2009-07-15 12:12:33 +08:00
|
|
|
bool isExt = GV->isDeclaration() || GV->isWeakForLinker();
|
2007-01-20 03:25:36 +08:00
|
|
|
if (isExt && isCallOp && Subtarget->isTargetDarwin() &&
|
2007-01-19 15:51:42 +08:00
|
|
|
TM.getRelocationModel() != Reloc::Static) {
|
2009-07-15 12:41:01 +08:00
|
|
|
FnStubInfo &FnInfo = FnStubs[Mang->getMangledName(GV)];
|
|
|
|
FnInfo.Init(GV, Mang);
|
|
|
|
Name = FnInfo.Stub;
|
|
|
|
} else {
|
|
|
|
Name = Mang->getMangledName(GV);
|
|
|
|
}
|
|
|
|
|
|
|
|
O << Name;
|
2008-11-23 00:15:34 +08:00
|
|
|
|
|
|
|
printOffset(MO.getOffset());
|
|
|
|
|
2007-04-22 08:04:12 +08:00
|
|
|
if (isCallOp && Subtarget->isTargetELF() &&
|
|
|
|
TM.getRelocationModel() == Reloc::PIC_)
|
|
|
|
O << "(PLT)";
|
2006-05-25 20:57:06 +08:00
|
|
|
break;
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
case MachineOperand::MO_ExternalSymbol: {
|
|
|
|
bool isCallOp = Modifier && !strcmp(Modifier, "call");
|
2009-07-15 12:41:01 +08:00
|
|
|
std::string Name;
|
2007-01-20 03:25:36 +08:00
|
|
|
if (isCallOp && Subtarget->isTargetDarwin() &&
|
2007-01-19 15:51:42 +08:00
|
|
|
TM.getRelocationModel() != Reloc::Static) {
|
2009-07-15 12:41:01 +08:00
|
|
|
FnStubInfo &FnInfo = FnStubs[Mang->makeNameProper(MO.getSymbolName())];
|
|
|
|
FnInfo.Init(MO.getSymbolName(), Mang);
|
|
|
|
Name = FnInfo.Stub;
|
2007-01-19 15:51:42 +08:00
|
|
|
} else
|
2009-07-15 12:41:01 +08:00
|
|
|
Name = Mang->makeNameProper(MO.getSymbolName());
|
|
|
|
|
|
|
|
O << Name;
|
2007-04-22 08:04:12 +08:00
|
|
|
if (isCallOp && Subtarget->isTargetELF() &&
|
|
|
|
TM.getRelocationModel() == Reloc::PIC_)
|
|
|
|
O << "(PLT)";
|
2006-05-25 20:57:06 +08:00
|
|
|
break;
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
2006-05-25 20:57:06 +08:00
|
|
|
case MachineOperand::MO_ConstantPoolIndex:
|
2007-10-14 13:57:21 +08:00
|
|
|
O << TAI->getPrivateGlobalPrefix() << "CPI" << getFunctionNumber()
|
2007-12-31 07:10:15 +08:00
|
|
|
<< '_' << MO.getIndex();
|
2006-05-25 20:57:06 +08:00
|
|
|
break;
|
2007-01-19 15:51:42 +08:00
|
|
|
case MachineOperand::MO_JumpTableIndex:
|
2007-10-14 13:57:21 +08:00
|
|
|
O << TAI->getPrivateGlobalPrefix() << "JTI" << getFunctionNumber()
|
2007-12-31 07:10:15 +08:00
|
|
|
<< '_' << MO.getIndex();
|
2007-01-19 15:51:42 +08:00
|
|
|
break;
|
2006-05-25 20:57:06 +08:00
|
|
|
default:
|
|
|
|
O << "<unknown operand type>"; abort (); break;
|
|
|
|
}
|
2006-05-15 06:18:28 +08:00
|
|
|
}
|
|
|
|
|
2009-07-15 04:18:05 +08:00
|
|
|
static void printSOImm(formatted_raw_ostream &O, int64_t V, bool VerboseAsm,
|
2009-03-25 09:47:28 +08:00
|
|
|
const TargetAsmInfo *TAI) {
|
2009-07-09 05:03:57 +08:00
|
|
|
// Break it up into two parts that make up a shifter immediate.
|
|
|
|
V = ARM_AM::getSOImmVal(V);
|
|
|
|
assert(V != -1 && "Not a valid so_imm value!");
|
|
|
|
|
2007-03-20 16:11:30 +08:00
|
|
|
unsigned Imm = ARM_AM::getSOImmValImm(V);
|
|
|
|
unsigned Rot = ARM_AM::getSOImmValRot(V);
|
2008-11-23 00:15:34 +08:00
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
// Print low-level immediate formation info, per
|
|
|
|
// A5.1.3: "Data-processing operands - Immediate".
|
|
|
|
if (Rot) {
|
|
|
|
O << "#" << Imm << ", " << Rot;
|
|
|
|
// Pretty printed version.
|
2009-03-24 08:17:40 +08:00
|
|
|
if (VerboseAsm)
|
|
|
|
O << ' ' << TAI->getCommentString()
|
|
|
|
<< ' ' << (int)ARM_AM::rotr32(Imm, Rot);
|
2007-01-19 15:51:42 +08:00
|
|
|
} else {
|
|
|
|
O << "#" << Imm;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-20 16:11:30 +08:00
|
|
|
/// printSOImmOperand - SOImm is 4-bit rotate amount in bits 8-11 with 8-bit
|
|
|
|
/// immediate in bits 0-7.
|
|
|
|
void ARMAsmPrinter::printSOImmOperand(const MachineInstr *MI, int OpNum) {
|
|
|
|
const MachineOperand &MO = MI->getOperand(OpNum);
|
2008-10-03 23:45:36 +08:00
|
|
|
assert(MO.isImm() && "Not a valid so_imm value!");
|
2009-03-25 09:47:28 +08:00
|
|
|
printSOImm(O, MO.getImm(), VerboseAsm, TAI);
|
2007-03-20 16:11:30 +08:00
|
|
|
}
|
|
|
|
|
2008-11-06 10:25:39 +08:00
|
|
|
/// printSOImm2PartOperand - SOImm is broken into two pieces using a 'mov'
|
|
|
|
/// followed by an 'orr' to materialize.
|
2007-03-20 16:11:30 +08:00
|
|
|
void ARMAsmPrinter::printSOImm2PartOperand(const MachineInstr *MI, int OpNum) {
|
|
|
|
const MachineOperand &MO = MI->getOperand(OpNum);
|
2008-10-03 23:45:36 +08:00
|
|
|
assert(MO.isImm() && "Not a valid so_imm value!");
|
2007-12-31 04:49:49 +08:00
|
|
|
unsigned V1 = ARM_AM::getSOImmTwoPartFirst(MO.getImm());
|
|
|
|
unsigned V2 = ARM_AM::getSOImmTwoPartSecond(MO.getImm());
|
2009-07-09 05:03:57 +08:00
|
|
|
printSOImm(O, V1, VerboseAsm, TAI);
|
2007-06-06 02:55:18 +08:00
|
|
|
O << "\n\torr";
|
|
|
|
printPredicateOperand(MI, 2);
|
|
|
|
O << " ";
|
2007-03-20 16:11:30 +08:00
|
|
|
printOperand(MI, 0);
|
|
|
|
O << ", ";
|
|
|
|
printOperand(MI, 0);
|
|
|
|
O << ", ";
|
2009-07-09 05:03:57 +08:00
|
|
|
printSOImm(O, V2, VerboseAsm, TAI);
|
2007-03-20 16:11:30 +08:00
|
|
|
}
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
// so_reg is a 4-operand unit corresponding to register forms of the A5.1
|
|
|
|
// "Addressing Mode 1 - Data-processing operands" forms. This includes:
|
2009-06-27 10:26:13 +08:00
|
|
|
// REG 0 0 - e.g. R5
|
|
|
|
// REG REG 0,SH_OPC - e.g. R5, ROR R3
|
2007-01-19 15:51:42 +08:00
|
|
|
// REG 0 IMM,SH_OPC - e.g. R5, LSL #3
|
|
|
|
void ARMAsmPrinter::printSORegOperand(const MachineInstr *MI, int Op) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
const MachineOperand &MO3 = MI->getOperand(Op+2);
|
|
|
|
|
2008-02-11 02:45:23 +08:00
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(MO1.getReg()));
|
2009-07-10 08:14:05 +08:00
|
|
|
O << TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
// Print the shift opc.
|
|
|
|
O << ", "
|
2007-12-31 04:49:49 +08:00
|
|
|
<< ARM_AM::getShiftOpcStr(ARM_AM::getSORegShOp(MO3.getImm()))
|
2007-01-19 15:51:42 +08:00
|
|
|
<< " ";
|
|
|
|
|
|
|
|
if (MO2.getReg()) {
|
2008-02-11 02:45:23 +08:00
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(MO2.getReg()));
|
2009-07-10 08:14:05 +08:00
|
|
|
O << TRI->getAsmName(MO2.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
assert(ARM_AM::getSORegOffset(MO3.getImm()) == 0);
|
|
|
|
} else {
|
|
|
|
O << "#" << ARM_AM::getSORegOffset(MO3.getImm());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printAddrMode2Operand(const MachineInstr *MI, int Op) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
const MachineOperand &MO3 = MI->getOperand(Op+2);
|
|
|
|
|
2008-10-03 23:45:36 +08:00
|
|
|
if (!MO1.isReg()) { // FIXME: This is for CP entries, but isn't right.
|
2007-01-19 15:51:42 +08:00
|
|
|
printOperand(MI, Op);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
if (!MO2.getReg()) {
|
|
|
|
if (ARM_AM::getAM2Offset(MO3.getImm())) // Don't print +0.
|
|
|
|
O << ", #"
|
|
|
|
<< (char)ARM_AM::getAM2Op(MO3.getImm())
|
|
|
|
<< ARM_AM::getAM2Offset(MO3.getImm());
|
|
|
|
O << "]";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
O << ", "
|
|
|
|
<< (char)ARM_AM::getAM2Op(MO3.getImm())
|
2009-07-10 08:14:05 +08:00
|
|
|
<< TRI->getAsmName(MO2.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
if (unsigned ShImm = ARM_AM::getAM2Offset(MO3.getImm()))
|
|
|
|
O << ", "
|
2007-12-31 04:49:49 +08:00
|
|
|
<< ARM_AM::getShiftOpcStr(ARM_AM::getAM2ShiftOpc(MO3.getImm()))
|
2007-01-19 15:51:42 +08:00
|
|
|
<< " #" << ShImm;
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printAddrMode2OffsetOperand(const MachineInstr *MI, int Op){
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
|
|
|
|
if (!MO1.getReg()) {
|
2007-05-04 07:30:36 +08:00
|
|
|
unsigned ImmOffs = ARM_AM::getAM2Offset(MO2.getImm());
|
|
|
|
assert(ImmOffs && "Malformed indexed load / store!");
|
|
|
|
O << "#"
|
|
|
|
<< (char)ARM_AM::getAM2Op(MO2.getImm())
|
|
|
|
<< ImmOffs;
|
2007-01-19 15:51:42 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
O << (char)ARM_AM::getAM2Op(MO2.getImm())
|
2009-07-10 08:14:05 +08:00
|
|
|
<< TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
if (unsigned ShImm = ARM_AM::getAM2Offset(MO2.getImm()))
|
|
|
|
O << ", "
|
2007-12-31 04:49:49 +08:00
|
|
|
<< ARM_AM::getShiftOpcStr(ARM_AM::getAM2ShiftOpc(MO2.getImm()))
|
2007-01-19 15:51:42 +08:00
|
|
|
<< " #" << ShImm;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printAddrMode3Operand(const MachineInstr *MI, int Op) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
const MachineOperand &MO3 = MI->getOperand(Op+2);
|
|
|
|
|
2008-02-11 02:45:23 +08:00
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(MO1.getReg()));
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
if (MO2.getReg()) {
|
|
|
|
O << ", "
|
|
|
|
<< (char)ARM_AM::getAM3Op(MO3.getImm())
|
2009-07-10 08:14:05 +08:00
|
|
|
<< TRI->getAsmName(MO2.getReg())
|
2007-01-19 15:51:42 +08:00
|
|
|
<< "]";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unsigned ImmOffs = ARM_AM::getAM3Offset(MO3.getImm()))
|
|
|
|
O << ", #"
|
|
|
|
<< (char)ARM_AM::getAM3Op(MO3.getImm())
|
|
|
|
<< ImmOffs;
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printAddrMode3OffsetOperand(const MachineInstr *MI, int Op){
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
|
|
|
|
if (MO1.getReg()) {
|
|
|
|
O << (char)ARM_AM::getAM3Op(MO2.getImm())
|
2009-07-10 08:14:05 +08:00
|
|
|
<< TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ImmOffs = ARM_AM::getAM3Offset(MO2.getImm());
|
2007-05-04 07:30:36 +08:00
|
|
|
assert(ImmOffs && "Malformed indexed load / store!");
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "#"
|
2007-05-04 07:30:36 +08:00
|
|
|
<< (char)ARM_AM::getAM3Op(MO2.getImm())
|
2007-01-19 15:51:42 +08:00
|
|
|
<< ImmOffs;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printAddrMode4Operand(const MachineInstr *MI, int Op,
|
|
|
|
const char *Modifier) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
ARM_AM::AMSubMode Mode = ARM_AM::getAM4SubMode(MO2.getImm());
|
|
|
|
if (Modifier && strcmp(Modifier, "submode") == 0) {
|
|
|
|
if (MO1.getReg() == ARM::SP) {
|
|
|
|
bool isLDM = (MI->getOpcode() == ARM::LDM ||
|
|
|
|
MI->getOpcode() == ARM::LDM_RET);
|
|
|
|
O << ARM_AM::getAMSubModeAltStr(Mode, isLDM);
|
|
|
|
} else
|
|
|
|
O << ARM_AM::getAMSubModeStr(Mode);
|
|
|
|
} else {
|
|
|
|
printOperand(MI, Op);
|
|
|
|
if (ARM_AM::getAM4WBFlag(MO2.getImm()))
|
|
|
|
O << "!";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printAddrMode5Operand(const MachineInstr *MI, int Op,
|
|
|
|
const char *Modifier) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
|
2008-10-03 23:45:36 +08:00
|
|
|
if (!MO1.isReg()) { // FIXME: This is for CP entries, but isn't right.
|
2007-01-19 15:51:42 +08:00
|
|
|
printOperand(MI, Op);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-02-11 02:45:23 +08:00
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(MO1.getReg()));
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
if (Modifier && strcmp(Modifier, "submode") == 0) {
|
|
|
|
ARM_AM::AMSubMode Mode = ARM_AM::getAM5SubMode(MO2.getImm());
|
|
|
|
if (MO1.getReg() == ARM::SP) {
|
|
|
|
bool isFLDM = (MI->getOpcode() == ARM::FLDMD ||
|
|
|
|
MI->getOpcode() == ARM::FLDMS);
|
|
|
|
O << ARM_AM::getAMSubModeAltStr(Mode, isFLDM);
|
|
|
|
} else
|
|
|
|
O << ARM_AM::getAMSubModeStr(Mode);
|
|
|
|
return;
|
|
|
|
} else if (Modifier && strcmp(Modifier, "base") == 0) {
|
|
|
|
// Used for FSTM{D|S} and LSTM{D|S} operations.
|
2009-07-10 08:14:05 +08:00
|
|
|
O << TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
if (ARM_AM::getAM5WBFlag(MO2.getImm()))
|
|
|
|
O << "!";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
if (unsigned ImmOffs = ARM_AM::getAM5Offset(MO2.getImm())) {
|
|
|
|
O << ", #"
|
|
|
|
<< (char)ARM_AM::getAM5Op(MO2.getImm())
|
|
|
|
<< ImmOffs*4;
|
|
|
|
}
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
2009-07-02 07:16:05 +08:00
|
|
|
void ARMAsmPrinter::printAddrMode6Operand(const MachineInstr *MI, int Op) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
const MachineOperand &MO3 = MI->getOperand(Op+2);
|
|
|
|
|
|
|
|
// FIXME: No support yet for specifying alignment.
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg()) << "]";
|
2009-07-02 07:16:05 +08:00
|
|
|
|
|
|
|
if (ARM_AM::getAM6WBFlag(MO3.getImm())) {
|
|
|
|
if (MO2.getReg() == 0)
|
|
|
|
O << "!";
|
|
|
|
else
|
2009-07-10 08:14:05 +08:00
|
|
|
O << ", " << TRI->getAsmName(MO2.getReg());
|
2009-07-02 07:16:05 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
void ARMAsmPrinter::printAddrModePCOperand(const MachineInstr *MI, int Op,
|
|
|
|
const char *Modifier) {
|
|
|
|
if (Modifier && strcmp(Modifier, "label") == 0) {
|
|
|
|
printPCLabel(MI, Op+1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
2008-02-11 02:45:23 +08:00
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(MO1.getReg()));
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[pc, +" << TRI->getAsmName(MO1.getReg()) << "]";
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
|
2009-06-24 01:48:47 +08:00
|
|
|
void
|
|
|
|
ARMAsmPrinter::printBitfieldInvMaskImmOperand(const MachineInstr *MI, int Op) {
|
|
|
|
const MachineOperand &MO = MI->getOperand(Op);
|
|
|
|
uint32_t v = ~MO.getImm();
|
2009-06-26 06:04:44 +08:00
|
|
|
int32_t lsb = CountTrailingZeros_32(v);
|
2009-06-24 09:08:42 +08:00
|
|
|
int32_t width = (32 - CountLeadingZeros_32 (v)) - lsb;
|
2009-06-24 01:48:47 +08:00
|
|
|
assert(MO.isImm() && "Not a valid bf_inv_mask_imm value!");
|
|
|
|
O << "#" << lsb << ", #" << width;
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
|
2009-07-10 07:43:36 +08:00
|
|
|
void
|
|
|
|
ARMAsmPrinter::printThumbITMask(const MachineInstr *MI, int Op) {
|
|
|
|
// (3 - the number of trailing zeros) is the number of then / else.
|
|
|
|
unsigned Mask = MI->getOperand(Op).getImm();
|
|
|
|
unsigned NumTZ = CountTrailingZeros_32(Mask);
|
|
|
|
assert(NumTZ <= 3 && "Invalid IT mask!");
|
2009-07-10 09:54:42 +08:00
|
|
|
for (unsigned Pos = 3, e = NumTZ; Pos > e; --Pos) {
|
2009-07-10 07:43:36 +08:00
|
|
|
bool T = (Mask & (1 << Pos)) != 0;
|
|
|
|
if (T)
|
|
|
|
O << 't';
|
|
|
|
else
|
|
|
|
O << 'e';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
void
|
|
|
|
ARMAsmPrinter::printThumbAddrModeRROperand(const MachineInstr *MI, int Op) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
|
|
|
O << ", " << TRI->getAsmName(MO2.getReg()) << "]";
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ARMAsmPrinter::printThumbAddrModeRI5Operand(const MachineInstr *MI, int Op,
|
|
|
|
unsigned Scale) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
2007-01-30 10:35:32 +08:00
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
|
|
|
const MachineOperand &MO3 = MI->getOperand(Op+2);
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2008-10-03 23:45:36 +08:00
|
|
|
if (!MO1.isReg()) { // FIXME: This is for CP entries, but isn't right.
|
2007-01-19 15:51:42 +08:00
|
|
|
printOperand(MI, Op);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2007-01-30 10:35:32 +08:00
|
|
|
if (MO3.getReg())
|
2009-07-10 08:14:05 +08:00
|
|
|
O << ", " << TRI->getAsmName(MO3.getReg());
|
2007-01-30 10:35:32 +08:00
|
|
|
else if (unsigned ImmOffs = MO2.getImm()) {
|
2007-01-19 15:51:42 +08:00
|
|
|
O << ", #" << ImmOffs;
|
|
|
|
if (Scale > 1)
|
|
|
|
O << " * " << Scale;
|
|
|
|
}
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2007-01-24 06:59:13 +08:00
|
|
|
ARMAsmPrinter::printThumbAddrModeS1Operand(const MachineInstr *MI, int Op) {
|
2007-01-30 10:35:32 +08:00
|
|
|
printThumbAddrModeRI5Operand(MI, Op, 1);
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
void
|
2007-01-24 06:59:13 +08:00
|
|
|
ARMAsmPrinter::printThumbAddrModeS2Operand(const MachineInstr *MI, int Op) {
|
2007-01-30 10:35:32 +08:00
|
|
|
printThumbAddrModeRI5Operand(MI, Op, 2);
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
void
|
2007-01-24 06:59:13 +08:00
|
|
|
ARMAsmPrinter::printThumbAddrModeS4Operand(const MachineInstr *MI, int Op) {
|
2007-01-30 10:35:32 +08:00
|
|
|
printThumbAddrModeRI5Operand(MI, Op, 4);
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printThumbAddrModeSPOperand(const MachineInstr *MI,int Op) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(Op);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(Op+1);
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2007-01-19 15:51:42 +08:00
|
|
|
if (unsigned ImmOffs = MO2.getImm())
|
|
|
|
O << ", #" << ImmOffs << " * 4";
|
|
|
|
O << "]";
|
2006-05-15 06:18:28 +08:00
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
|
2009-06-27 10:26:13 +08:00
|
|
|
// Constant shifts t2_so_reg is a 2-operand unit corresponding to the Thumb2
|
|
|
|
// register with shift forms.
|
|
|
|
// REG 0 0 - e.g. R5
|
|
|
|
// REG IMM, SH_OPC - e.g. R5, LSL #3
|
|
|
|
void ARMAsmPrinter::printT2SOOperand(const MachineInstr *MI, int OpNum) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(OpNum);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(OpNum+1);
|
|
|
|
|
|
|
|
unsigned Reg = MO1.getReg();
|
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(Reg));
|
2009-07-10 08:14:05 +08:00
|
|
|
O << TRI->getAsmName(Reg);
|
2009-06-27 10:26:13 +08:00
|
|
|
|
|
|
|
// Print the shift opc.
|
|
|
|
O << ", "
|
|
|
|
<< ARM_AM::getShiftOpcStr(ARM_AM::getSORegShOp(MO2.getImm()))
|
|
|
|
<< " ";
|
|
|
|
|
|
|
|
assert(MO2.isImm() && "Not a valid t2_so_reg value!");
|
|
|
|
O << "#" << ARM_AM::getSORegOffset(MO2.getImm());
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printT2AddrModeImm12Operand(const MachineInstr *MI,
|
|
|
|
int OpNum) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(OpNum);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(OpNum+1);
|
|
|
|
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2009-06-29 15:51:04 +08:00
|
|
|
|
|
|
|
unsigned OffImm = MO2.getImm();
|
|
|
|
if (OffImm) // Don't print +0.
|
|
|
|
O << ", #+" << OffImm;
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printT2AddrModeImm8Operand(const MachineInstr *MI,
|
|
|
|
int OpNum) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(OpNum);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(OpNum+1);
|
|
|
|
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2009-06-29 15:51:04 +08:00
|
|
|
|
|
|
|
int32_t OffImm = (int32_t)MO2.getImm();
|
|
|
|
// Don't print +0.
|
|
|
|
if (OffImm < 0)
|
|
|
|
O << ", #-" << -OffImm;
|
|
|
|
else if (OffImm > 0)
|
|
|
|
O << ", #+" << OffImm;
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
2009-07-10 06:21:59 +08:00
|
|
|
void ARMAsmPrinter::printT2AddrModeImm8s4Operand(const MachineInstr *MI,
|
|
|
|
int OpNum) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(OpNum);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(OpNum+1);
|
|
|
|
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2009-07-10 06:21:59 +08:00
|
|
|
|
|
|
|
int32_t OffImm = (int32_t)MO2.getImm() / 4;
|
|
|
|
// Don't print +0.
|
|
|
|
if (OffImm < 0)
|
|
|
|
O << ", #-" << -OffImm << " * 4";
|
|
|
|
else if (OffImm > 0)
|
|
|
|
O << ", #+" << OffImm << " * 4";
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
2009-07-02 15:28:31 +08:00
|
|
|
void ARMAsmPrinter::printT2AddrModeImm8OffsetOperand(const MachineInstr *MI,
|
|
|
|
int OpNum) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(OpNum);
|
|
|
|
int32_t OffImm = (int32_t)MO1.getImm();
|
|
|
|
// Don't print +0.
|
|
|
|
if (OffImm < 0)
|
|
|
|
O << "#-" << -OffImm;
|
|
|
|
else if (OffImm > 0)
|
|
|
|
O << "#+" << OffImm;
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printT2AddrModeSoRegOperand(const MachineInstr *MI,
|
|
|
|
int OpNum) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(OpNum);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(OpNum+1);
|
|
|
|
const MachineOperand &MO3 = MI->getOperand(OpNum+2);
|
|
|
|
|
2009-07-10 08:14:05 +08:00
|
|
|
O << "[" << TRI->getAsmName(MO1.getReg());
|
2009-06-29 15:51:04 +08:00
|
|
|
|
|
|
|
if (MO2.getReg()) {
|
2009-07-10 08:14:05 +08:00
|
|
|
O << ", +" << TRI->getAsmName(MO2.getReg());
|
2009-06-27 10:26:13 +08:00
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
unsigned ShAmt = MO3.getImm();
|
|
|
|
if (ShAmt) {
|
|
|
|
assert(ShAmt <= 3 && "Not a valid Thumb2 addressing mode!");
|
|
|
|
O << ", lsl #" << ShAmt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
void ARMAsmPrinter::printPredicateOperand(const MachineInstr *MI, int OpNum) {
|
|
|
|
ARMCC::CondCodes CC = (ARMCC::CondCodes)MI->getOperand(OpNum).getImm();
|
2007-05-15 09:29:07 +08:00
|
|
|
if (CC != ARMCC::AL)
|
|
|
|
O << ARMCondCodeToString(CC);
|
2006-05-15 06:18:28 +08:00
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printSBitModifierOperand(const MachineInstr *MI, int OpNum){
|
|
|
|
unsigned Reg = MI->getOperand(OpNum).getReg();
|
2007-07-06 09:01:34 +08:00
|
|
|
if (Reg) {
|
|
|
|
assert(Reg == ARM::CPSR && "Expect ARM CPSR register!");
|
|
|
|
O << 's';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printPCLabel(const MachineInstr *MI, int OpNum) {
|
|
|
|
int Id = (int)MI->getOperand(OpNum).getImm();
|
2007-01-19 15:51:42 +08:00
|
|
|
O << TAI->getPrivateGlobalPrefix() << "PC" << Id;
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printRegisterList(const MachineInstr *MI, int OpNum) {
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "{";
|
2009-06-29 15:51:04 +08:00
|
|
|
for (unsigned i = OpNum, e = MI->getNumOperands(); i != e; ++i) {
|
2007-01-19 15:51:42 +08:00
|
|
|
printOperand(MI, i);
|
|
|
|
if (i != e-1) O << ", ";
|
|
|
|
}
|
|
|
|
O << "}";
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printCPInstOperand(const MachineInstr *MI, int OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
const char *Modifier) {
|
|
|
|
assert(Modifier && "This operand only works with a modifier!");
|
|
|
|
// There are two aspects to a CONSTANTPOOL_ENTRY operand, the label and the
|
|
|
|
// data itself.
|
|
|
|
if (!strcmp(Modifier, "label")) {
|
2009-06-29 15:51:04 +08:00
|
|
|
unsigned ID = MI->getOperand(OpNum).getImm();
|
2007-10-14 13:57:21 +08:00
|
|
|
O << TAI->getPrivateGlobalPrefix() << "CPI" << getFunctionNumber()
|
|
|
|
<< '_' << ID << ":\n";
|
2007-01-19 15:51:42 +08:00
|
|
|
} else {
|
|
|
|
assert(!strcmp(Modifier, "cpentry") && "Unknown modifier for CPE");
|
2009-06-29 15:51:04 +08:00
|
|
|
unsigned CPI = MI->getOperand(OpNum).getIndex();
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2008-09-18 15:27:23 +08:00
|
|
|
const MachineConstantPoolEntry &MCPE = MCP->getConstants()[CPI];
|
2007-01-19 15:51:42 +08:00
|
|
|
|
2008-08-08 14:56:16 +08:00
|
|
|
if (MCPE.isMachineConstantPoolEntry()) {
|
2007-01-19 15:51:42 +08:00
|
|
|
EmitMachineConstantPoolValue(MCPE.Val.MachineCPVal);
|
2008-08-08 14:56:16 +08:00
|
|
|
} else {
|
2007-01-19 15:51:42 +08:00
|
|
|
EmitGlobalConstant(MCPE.Val.ConstVal);
|
2007-04-25 22:50:40 +08:00
|
|
|
}
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
void ARMAsmPrinter::printJTBlockOperand(const MachineInstr *MI, int OpNum) {
|
|
|
|
const MachineOperand &MO1 = MI->getOperand(OpNum);
|
|
|
|
const MachineOperand &MO2 = MI->getOperand(OpNum+1); // Unique Id
|
2007-12-31 07:10:15 +08:00
|
|
|
unsigned JTI = MO1.getIndex();
|
2007-10-14 13:57:21 +08:00
|
|
|
O << TAI->getPrivateGlobalPrefix() << "JTI" << getFunctionNumber()
|
2007-12-31 04:49:49 +08:00
|
|
|
<< '_' << JTI << '_' << MO2.getImm() << ":\n";
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
const char *JTEntryDirective = TAI->getJumpTableDirective();
|
|
|
|
if (!JTEntryDirective)
|
|
|
|
JTEntryDirective = TAI->getData32bitsDirective();
|
|
|
|
|
|
|
|
const MachineFunction *MF = MI->getParent()->getParent();
|
2008-07-08 04:06:06 +08:00
|
|
|
const MachineJumpTableInfo *MJTI = MF->getJumpTableInfo();
|
2007-01-19 15:51:42 +08:00
|
|
|
const std::vector<MachineJumpTableEntry> &JT = MJTI->getJumpTables();
|
|
|
|
const std::vector<MachineBasicBlock*> &JTBBs = JT[JTI].MBBs;
|
|
|
|
bool UseSet= TAI->getSetDirective() && TM.getRelocationModel() == Reloc::PIC_;
|
|
|
|
std::set<MachineBasicBlock*> JTSets;
|
|
|
|
for (unsigned i = 0, e = JTBBs.size(); i != e; ++i) {
|
|
|
|
MachineBasicBlock *MBB = JTBBs[i];
|
|
|
|
if (UseSet && JTSets.insert(MBB).second)
|
2007-12-31 04:49:49 +08:00
|
|
|
printPICJumpTableSetLabel(JTI, MO2.getImm(), MBB);
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
O << JTEntryDirective << ' ';
|
|
|
|
if (UseSet)
|
2007-10-14 13:57:21 +08:00
|
|
|
O << TAI->getPrivateGlobalPrefix() << getFunctionNumber()
|
2007-12-31 04:49:49 +08:00
|
|
|
<< '_' << JTI << '_' << MO2.getImm()
|
2007-10-14 13:57:21 +08:00
|
|
|
<< "_set_" << MBB->getNumber();
|
2007-01-19 15:51:42 +08:00
|
|
|
else if (TM.getRelocationModel() == Reloc::PIC_) {
|
2008-02-28 08:43:03 +08:00
|
|
|
printBasicBlockLabel(MBB, false, false, false);
|
2007-01-19 15:51:42 +08:00
|
|
|
// If the arch uses custom Jump Table directives, don't calc relative to JT
|
|
|
|
if (!TAI->getJumpTableDirective())
|
|
|
|
O << '-' << TAI->getPrivateGlobalPrefix() << "JTI"
|
2007-12-31 04:49:49 +08:00
|
|
|
<< getFunctionNumber() << '_' << JTI << '_' << MO2.getImm();
|
2007-01-19 15:51:42 +08:00
|
|
|
} else
|
2008-02-28 08:43:03 +08:00
|
|
|
printBasicBlockLabel(MBB, false, false, false);
|
2007-01-27 10:29:45 +08:00
|
|
|
if (i != e-1)
|
|
|
|
O << '\n';
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
bool ARMAsmPrinter::PrintAsmOperand(const MachineInstr *MI, unsigned OpNum,
|
2007-01-19 15:51:42 +08:00
|
|
|
unsigned AsmVariant, const char *ExtraCode){
|
|
|
|
// Does this asm operand have a single letter operand modifier?
|
|
|
|
if (ExtraCode && ExtraCode[0]) {
|
|
|
|
if (ExtraCode[1] != 0) return true; // Unknown modifier.
|
|
|
|
|
|
|
|
switch (ExtraCode[0]) {
|
|
|
|
default: return true; // Unknown modifier.
|
2009-07-10 07:54:51 +08:00
|
|
|
case 'a': // Print as a memory address.
|
|
|
|
if (MI->getOperand(OpNum).isReg()) {
|
|
|
|
O << "[" << TRI->getAsmName(MI->getOperand(OpNum).getReg()) << "]";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// Fallthrough
|
|
|
|
case 'c': // Don't print "#" before an immediate operand.
|
2009-06-29 15:51:04 +08:00
|
|
|
printOperand(MI, OpNum, "no_hash");
|
2009-04-07 05:46:51 +08:00
|
|
|
return false;
|
2007-04-04 08:13:29 +08:00
|
|
|
case 'P': // Print a VFP double precision register.
|
2009-06-29 15:51:04 +08:00
|
|
|
printOperand(MI, OpNum);
|
2007-03-09 06:42:46 +08:00
|
|
|
return false;
|
2007-01-19 15:51:42 +08:00
|
|
|
case 'Q':
|
|
|
|
if (TM.getTargetData()->isLittleEndian())
|
|
|
|
break;
|
|
|
|
// Fallthrough
|
|
|
|
case 'R':
|
|
|
|
if (TM.getTargetData()->isBigEndian())
|
|
|
|
break;
|
|
|
|
// Fallthrough
|
|
|
|
case 'H': // Write second word of DI / DF reference.
|
|
|
|
// Verify that this operand has two consecutive registers.
|
2009-06-29 15:51:04 +08:00
|
|
|
if (!MI->getOperand(OpNum).isReg() ||
|
|
|
|
OpNum+1 == MI->getNumOperands() ||
|
|
|
|
!MI->getOperand(OpNum+1).isReg())
|
2007-01-19 15:51:42 +08:00
|
|
|
return true;
|
2009-06-29 15:51:04 +08:00
|
|
|
++OpNum; // Return the high-part.
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-29 15:51:04 +08:00
|
|
|
printOperand(MI, OpNum);
|
2007-01-19 15:51:42 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-05-19 13:53:42 +08:00
|
|
|
bool ARMAsmPrinter::PrintAsmMemoryOperand(const MachineInstr *MI,
|
2009-06-29 15:51:04 +08:00
|
|
|
unsigned OpNum, unsigned AsmVariant,
|
2009-05-19 13:53:42 +08:00
|
|
|
const char *ExtraCode) {
|
|
|
|
if (ExtraCode && ExtraCode[0])
|
|
|
|
return true; // Unknown modifier.
|
2009-06-29 15:51:04 +08:00
|
|
|
printAddrMode2Operand(MI, OpNum);
|
2009-05-19 13:53:42 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
void ARMAsmPrinter::printMachineInstruction(const MachineInstr *MI) {
|
|
|
|
++EmittedInsts;
|
|
|
|
|
2007-01-31 04:37:08 +08:00
|
|
|
int Opc = MI->getOpcode();
|
|
|
|
switch (Opc) {
|
|
|
|
case ARM::CONSTPOOL_ENTRY:
|
2007-01-19 15:51:42 +08:00
|
|
|
if (!InCPMode && AFI->isThumbFunction()) {
|
|
|
|
EmitAlignment(2);
|
|
|
|
InCPMode = true;
|
|
|
|
}
|
2007-01-31 04:37:08 +08:00
|
|
|
break;
|
|
|
|
default: {
|
2007-02-01 07:39:39 +08:00
|
|
|
if (InCPMode && AFI->isThumbFunction())
|
2007-01-19 15:51:42 +08:00
|
|
|
InCPMode = false;
|
2007-01-31 04:37:08 +08:00
|
|
|
}}
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
// Call the autogenerated instruction printer routines.
|
|
|
|
printInstruction(MI);
|
|
|
|
}
|
|
|
|
|
2006-05-15 06:18:28 +08:00
|
|
|
bool ARMAsmPrinter::doInitialization(Module &M) {
|
2008-12-11 05:54:21 +08:00
|
|
|
|
2007-07-26 03:33:14 +08:00
|
|
|
bool Result = AsmPrinter::doInitialization(M);
|
2009-01-28 21:14:17 +08:00
|
|
|
DW = getAnalysisIfAvailable<DwarfWriter>();
|
2008-07-10 05:20:54 +08:00
|
|
|
|
2009-07-11 14:43:01 +08:00
|
|
|
// Use unified assembler syntax mode for Thumb.
|
|
|
|
if (Subtarget->isThumb())
|
2009-06-18 07:43:18 +08:00
|
|
|
O << "\t.syntax unified\n";
|
|
|
|
|
2009-05-24 03:51:20 +08:00
|
|
|
// Emit ARM Build Attributes
|
|
|
|
if (Subtarget->isTargetELF()) {
|
|
|
|
// CPU Type
|
2009-06-02 03:03:17 +08:00
|
|
|
std::string CPUString = Subtarget->getCPUString();
|
|
|
|
if (CPUString != "generic")
|
|
|
|
O << "\t.cpu " << CPUString << '\n';
|
2009-05-24 03:51:20 +08:00
|
|
|
|
|
|
|
// FIXME: Emit FPU type
|
|
|
|
if (Subtarget->hasVFP2())
|
|
|
|
O << "\t.eabi_attribute " << ARMBuildAttrs::VFP_arch << ", 2\n";
|
|
|
|
|
|
|
|
// Signal various FP modes.
|
|
|
|
if (!UnsafeFPMath)
|
|
|
|
O << "\t.eabi_attribute " << ARMBuildAttrs::ABI_FP_denormal << ", 1\n"
|
|
|
|
<< "\t.eabi_attribute " << ARMBuildAttrs::ABI_FP_exceptions << ", 1\n";
|
|
|
|
|
|
|
|
if (FiniteOnlyFPMath())
|
|
|
|
O << "\t.eabi_attribute " << ARMBuildAttrs::ABI_FP_number_model << ", 1\n";
|
|
|
|
else
|
|
|
|
O << "\t.eabi_attribute " << ARMBuildAttrs::ABI_FP_number_model << ", 3\n";
|
|
|
|
|
|
|
|
// 8-bytes alignment stuff.
|
|
|
|
O << "\t.eabi_attribute " << ARMBuildAttrs::ABI_align8_needed << ", 1\n"
|
|
|
|
<< "\t.eabi_attribute " << ARMBuildAttrs::ABI_align8_preserved << ", 1\n";
|
|
|
|
|
|
|
|
// FIXME: Should we signal R9 usage?
|
|
|
|
}
|
|
|
|
|
2007-07-26 03:33:14 +08:00
|
|
|
return Result;
|
2006-05-15 06:18:28 +08:00
|
|
|
}
|
|
|
|
|
2008-02-16 03:04:54 +08:00
|
|
|
/// PrintUnmangledNameSafely - Print out the printable characters in the name.
|
2009-03-03 10:55:14 +08:00
|
|
|
/// Don't print things like \\n or \\0.
|
2009-07-15 04:18:05 +08:00
|
|
|
static void PrintUnmangledNameSafely(const Value *V, formatted_raw_ostream &OS) {
|
2008-02-16 03:04:54 +08:00
|
|
|
for (const char *Name = V->getNameStart(), *E = Name+V->getNameLen();
|
|
|
|
Name != E; ++Name)
|
|
|
|
if (isprint(*Name))
|
|
|
|
OS << *Name;
|
|
|
|
}
|
|
|
|
|
2009-07-22 02:38:57 +08:00
|
|
|
void ARMAsmPrinter::PrintGlobalVariable(const GlobalVariable* GVar) {
|
2006-07-27 19:38:51 +08:00
|
|
|
const TargetData *TD = TM.getTargetData();
|
|
|
|
|
2008-08-07 17:54:23 +08:00
|
|
|
if (!GVar->hasInitializer()) // External global require no code
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Check to see if this is a special global used by LLVM, if so, emit it.
|
|
|
|
|
|
|
|
if (EmitSpecialLLVMGlobal(GVar)) {
|
|
|
|
if (Subtarget->isTargetDarwin() &&
|
|
|
|
TM.getRelocationModel() == Reloc::Static) {
|
2009-07-24 11:49:17 +08:00
|
|
|
if (GVar->isName("llvm.global_ctors"))
|
2008-08-07 17:54:23 +08:00
|
|
|
O << ".reference .constructors_used\n";
|
2009-07-24 11:49:17 +08:00
|
|
|
else if (GVar->isName("llvm.global_dtors"))
|
2008-08-07 17:54:23 +08:00
|
|
|
O << ".reference .destructors_used\n";
|
2007-01-30 16:04:53 +08:00
|
|
|
}
|
2008-08-07 17:54:23 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-07-15 02:17:16 +08:00
|
|
|
std::string name = Mang->getMangledName(GVar);
|
2008-08-07 17:54:23 +08:00
|
|
|
Constant *C = GVar->getInitializer();
|
2009-06-26 10:26:12 +08:00
|
|
|
if (isa<MDNode>(C) || isa<MDString>(C))
|
2009-06-25 08:47:42 +08:00
|
|
|
return;
|
2008-08-07 17:54:23 +08:00
|
|
|
const Type *Type = C->getType();
|
2009-05-09 15:06:46 +08:00
|
|
|
unsigned Size = TD->getTypeAllocSize(Type);
|
2008-08-07 17:54:23 +08:00
|
|
|
unsigned Align = TD->getPreferredAlignmentLog(GVar);
|
2008-12-06 10:00:55 +08:00
|
|
|
bool isDarwin = Subtarget->isTargetDarwin();
|
2006-07-27 19:38:51 +08:00
|
|
|
|
2008-08-09 02:25:07 +08:00
|
|
|
printVisibility(name, GVar->getVisibility());
|
2007-04-30 08:23:51 +08:00
|
|
|
|
2008-08-07 17:54:23 +08:00
|
|
|
if (Subtarget->isTargetELF())
|
|
|
|
O << "\t.type " << name << ",%object\n";
|
2009-07-24 11:49:17 +08:00
|
|
|
|
|
|
|
const Section *TheSection = TAI->SectionForGlobal(GVar);
|
|
|
|
SwitchToSection(TheSection);
|
2007-04-30 08:23:51 +08:00
|
|
|
|
2009-02-18 10:19:52 +08:00
|
|
|
if (C->isNullValue() && !GVar->hasSection() && !GVar->isThreadLocal() &&
|
2009-07-24 11:49:17 +08:00
|
|
|
!(isDarwin && TheSection->getFlags() == SectionKind::RODataMergeStr)) {
|
2008-08-07 17:54:23 +08:00
|
|
|
// FIXME: This seems to be pretty darwin-specific
|
|
|
|
|
|
|
|
if (GVar->hasExternalLinkage()) {
|
|
|
|
if (const char *Directive = TAI->getZeroFillDirective()) {
|
|
|
|
O << "\t.globl\t" << name << "\n";
|
|
|
|
O << Directive << "__DATA, __common, " << name << ", "
|
|
|
|
<< Size << ", " << Align << "\n";
|
|
|
|
return;
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
2007-01-20 03:25:36 +08:00
|
|
|
}
|
2006-12-09 05:24:58 +08:00
|
|
|
|
Introduce new linkage types linkonce_odr, weak_odr, common_odr
and extern_weak_odr. These are the same as the non-odr versions,
except that they indicate that the global will only be overridden
by an *equivalent* global. In C, a function with weak linkage can
be overridden by a function which behaves completely differently.
This means that IP passes have to skip weak functions, since any
deductions made from the function definition might be wrong, since
the definition could be replaced by something completely different
at link time. This is not allowed in C++, thanks to the ODR
(One-Definition-Rule): if a function is replaced by another at
link-time, then the new function must be the same as the original
function. If a language knows that a function or other global can
only be overridden by an equivalent global, it can give it the
weak_odr linkage type, and the optimizers will understand that it
is alright to make deductions based on the function body. The
code generators on the other hand map weak and weak_odr linkage
to the same thing.
llvm-svn: 66339
2009-03-07 23:45:40 +08:00
|
|
|
if (GVar->hasLocalLinkage() || GVar->isWeakForLinker()) {
|
2008-08-07 17:54:23 +08:00
|
|
|
if (Size == 0) Size = 1; // .comm Foo, 0 is undefined, avoid it.
|
|
|
|
|
2008-12-06 10:00:55 +08:00
|
|
|
if (isDarwin) {
|
2009-01-16 04:18:42 +08:00
|
|
|
if (GVar->hasLocalLinkage()) {
|
2008-12-06 10:00:55 +08:00
|
|
|
O << TAI->getLCOMMDirective() << name << "," << Size
|
|
|
|
<< ',' << Align;
|
|
|
|
} else if (GVar->hasCommonLinkage()) {
|
|
|
|
O << TAI->getCOMMDirective() << name << "," << Size
|
|
|
|
<< ',' << Align;
|
|
|
|
} else {
|
|
|
|
SwitchToSection(TAI->SectionForGlobal(GVar));
|
|
|
|
O << "\t.globl " << name << '\n'
|
|
|
|
<< TAI->getWeakDefDirective() << name << '\n';
|
|
|
|
EmitAlignment(Align, GVar);
|
2009-03-24 08:17:40 +08:00
|
|
|
O << name << ":";
|
|
|
|
if (VerboseAsm) {
|
|
|
|
O << "\t\t\t\t" << TAI->getCommentString() << ' ';
|
|
|
|
PrintUnmangledNameSafely(GVar, O);
|
|
|
|
}
|
2008-12-06 10:00:55 +08:00
|
|
|
O << '\n';
|
|
|
|
EmitGlobalConstant(C);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if (TAI->getLCOMMDirective() != NULL) {
|
2009-01-16 04:18:42 +08:00
|
|
|
if (GVar->hasLocalLinkage()) {
|
2008-08-07 17:54:23 +08:00
|
|
|
O << TAI->getLCOMMDirective() << name << "," << Size;
|
2008-12-06 10:00:55 +08:00
|
|
|
} else {
|
2008-08-07 17:54:23 +08:00
|
|
|
O << TAI->getCOMMDirective() << name << "," << Size;
|
2008-12-06 10:00:55 +08:00
|
|
|
if (TAI->getCOMMDirectiveTakesAlignment())
|
|
|
|
O << ',' << (TAI->getAlignmentIsInBytes() ? (1 << Align) : Align);
|
|
|
|
}
|
2007-01-20 03:25:36 +08:00
|
|
|
} else {
|
2009-01-16 04:18:42 +08:00
|
|
|
if (GVar->hasLocalLinkage())
|
2008-08-07 17:54:23 +08:00
|
|
|
O << "\t.local\t" << name << "\n";
|
|
|
|
O << TAI->getCOMMDirective() << name << "," << Size;
|
|
|
|
if (TAI->getCOMMDirectiveTakesAlignment())
|
|
|
|
O << "," << (TAI->getAlignmentIsInBytes() ? (1 << Align) : Align);
|
2007-01-20 03:25:36 +08:00
|
|
|
}
|
2009-03-24 08:17:40 +08:00
|
|
|
if (VerboseAsm) {
|
|
|
|
O << "\t\t" << TAI->getCommentString() << " ";
|
|
|
|
PrintUnmangledNameSafely(GVar, O);
|
|
|
|
}
|
2008-08-07 17:54:23 +08:00
|
|
|
O << "\n";
|
|
|
|
return;
|
2007-01-20 03:25:36 +08:00
|
|
|
}
|
2008-08-07 17:54:23 +08:00
|
|
|
}
|
2009-07-24 11:49:17 +08:00
|
|
|
|
2008-08-07 17:54:23 +08:00
|
|
|
switch (GVar->getLinkage()) {
|
2009-07-24 11:49:17 +08:00
|
|
|
case GlobalValue::CommonLinkage:
|
|
|
|
case GlobalValue::LinkOnceAnyLinkage:
|
|
|
|
case GlobalValue::LinkOnceODRLinkage:
|
|
|
|
case GlobalValue::WeakAnyLinkage:
|
|
|
|
case GlobalValue::WeakODRLinkage:
|
2008-12-06 10:00:55 +08:00
|
|
|
if (isDarwin) {
|
2008-08-07 17:54:23 +08:00
|
|
|
O << "\t.globl " << name << "\n"
|
|
|
|
<< "\t.weak_definition " << name << "\n";
|
|
|
|
} else {
|
|
|
|
O << "\t.weak " << name << "\n";
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
2008-08-07 17:54:23 +08:00
|
|
|
break;
|
2009-07-24 11:49:17 +08:00
|
|
|
case GlobalValue::AppendingLinkage:
|
|
|
|
// FIXME: appending linkage variables should go into a section of
|
|
|
|
// their name or something. For now, just emit them as external.
|
|
|
|
case GlobalValue::ExternalLinkage:
|
2008-08-07 17:54:23 +08:00
|
|
|
O << "\t.globl " << name << "\n";
|
|
|
|
break;
|
2009-07-24 11:49:17 +08:00
|
|
|
case GlobalValue::PrivateLinkage:
|
|
|
|
case GlobalValue::LinkerPrivateLinkage:
|
|
|
|
case GlobalValue::InternalLinkage:
|
|
|
|
break;
|
|
|
|
default:
|
2009-07-15 00:55:14 +08:00
|
|
|
llvm_unreachable("Unknown linkage type!");
|
2008-08-07 17:54:23 +08:00
|
|
|
}
|
2006-07-27 19:38:51 +08:00
|
|
|
|
2008-08-07 17:54:23 +08:00
|
|
|
EmitAlignment(Align, GVar);
|
2009-03-24 08:17:40 +08:00
|
|
|
O << name << ":";
|
|
|
|
if (VerboseAsm) {
|
|
|
|
O << "\t\t\t\t" << TAI->getCommentString() << " ";
|
|
|
|
PrintUnmangledNameSafely(GVar, O);
|
|
|
|
}
|
2008-08-07 17:54:23 +08:00
|
|
|
O << "\n";
|
|
|
|
if (TAI->hasDotTypeDotSizeDirective())
|
|
|
|
O << "\t.size " << name << ", " << Size << "\n";
|
|
|
|
|
|
|
|
EmitGlobalConstant(C);
|
|
|
|
O << '\n';
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ARMAsmPrinter::doFinalization(Module &M) {
|
2007-01-20 03:25:36 +08:00
|
|
|
if (Subtarget->isTargetDarwin()) {
|
|
|
|
SwitchToDataSection("");
|
|
|
|
|
2009-07-15 12:12:33 +08:00
|
|
|
O << '\n';
|
2007-01-19 15:51:42 +08:00
|
|
|
// Output stubs for dynamically-linked functions
|
2009-07-15 12:41:01 +08:00
|
|
|
for (StringMap<FnStubInfo>::iterator I = FnStubs.begin(), E = FnStubs.end();
|
2009-07-15 11:12:43 +08:00
|
|
|
I != E; ++I) {
|
2009-07-15 12:41:01 +08:00
|
|
|
const FnStubInfo &Info = I->second;
|
2007-01-19 15:51:42 +08:00
|
|
|
if (TM.getRelocationModel() == Reloc::PIC_)
|
|
|
|
SwitchToTextSection(".section __TEXT,__picsymbolstub4,symbol_stubs,"
|
|
|
|
"none,16", 0);
|
|
|
|
else
|
|
|
|
SwitchToTextSection(".section __TEXT,__symbol_stub4,symbol_stubs,"
|
|
|
|
"none,12", 0);
|
|
|
|
|
|
|
|
EmitAlignment(2);
|
|
|
|
O << "\t.code\t32\n";
|
|
|
|
|
2009-07-15 12:41:01 +08:00
|
|
|
O << Info.Stub << ":\n";
|
|
|
|
O << "\t.indirect_symbol " << I->getKeyData() << '\n';
|
|
|
|
O << "\tldr ip, " << Info.SLP << '\n';
|
2007-01-19 15:51:42 +08:00
|
|
|
if (TM.getRelocationModel() == Reloc::PIC_) {
|
2009-07-15 12:41:01 +08:00
|
|
|
O << Info.SCV << ":\n";
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "\tadd ip, pc, ip\n";
|
|
|
|
}
|
|
|
|
O << "\tldr pc, [ip, #0]\n";
|
2009-07-15 12:41:01 +08:00
|
|
|
O << Info.SLP << ":\n";
|
|
|
|
O << "\t.long\t" << Info.LazyPtr;
|
|
|
|
if (TM.getRelocationModel() == Reloc::PIC_)
|
|
|
|
O << "-(" << Info.SCV << "+8)";
|
|
|
|
O << '\n';
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
SwitchToDataSection(".lazy_symbol_pointer", 0);
|
2009-07-15 12:41:01 +08:00
|
|
|
O << Info.LazyPtr << ":\n";
|
|
|
|
O << "\t.indirect_symbol " << I->getKeyData() << "\n";
|
2007-01-19 15:51:42 +08:00
|
|
|
O << "\t.long\tdyld_stub_binding_helper\n";
|
|
|
|
}
|
2009-07-15 12:12:33 +08:00
|
|
|
O << '\n';
|
2007-01-19 15:51:42 +08:00
|
|
|
|
|
|
|
// Output non-lazy-pointers for external and common global variables.
|
2008-12-05 09:06:39 +08:00
|
|
|
if (!GVNonLazyPtrs.empty()) {
|
2008-12-06 10:00:55 +08:00
|
|
|
SwitchToDataSection("\t.non_lazy_symbol_pointer", 0);
|
2009-07-15 11:12:43 +08:00
|
|
|
for (StringMap<std::string>::iterator I = GVNonLazyPtrs.begin(),
|
|
|
|
E = GVNonLazyPtrs.end(); I != E; ++I) {
|
|
|
|
O << I->second << ":\n";
|
|
|
|
O << "\t.indirect_symbol " << I->getKeyData() << "\n";
|
2008-12-05 09:06:39 +08:00
|
|
|
O << "\t.long\t0\n";
|
|
|
|
}
|
2007-01-19 15:51:42 +08:00
|
|
|
}
|
|
|
|
|
2008-12-05 09:06:39 +08:00
|
|
|
if (!HiddenGVNonLazyPtrs.empty()) {
|
|
|
|
SwitchToSection(TAI->getDataSection());
|
2009-07-15 11:12:43 +08:00
|
|
|
for (StringMap<std::string>::iterator I = HiddenGVNonLazyPtrs.begin(),
|
|
|
|
E = HiddenGVNonLazyPtrs.end(); I != E; ++I) {
|
2008-12-05 09:06:39 +08:00
|
|
|
EmitAlignment(2);
|
2009-07-15 11:12:43 +08:00
|
|
|
O << I->second << ":\n";
|
|
|
|
O << "\t.long " << I->getKeyData() << "\n";
|
2008-12-05 09:06:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-01-19 15:51:42 +08:00
|
|
|
// Funny Darwin hack: This flag tells the linker that no global symbols
|
|
|
|
// contain code that falls through to other global symbols (e.g. the obvious
|
|
|
|
// implementation of multiple entry points). If this doesn't occur, the
|
|
|
|
// linker can safely perform dead code stripping. Since LLVM never
|
|
|
|
// generates code that does this, it is always safe to set.
|
|
|
|
O << "\t.subsections_via_symbols\n";
|
2006-07-27 19:38:51 +08:00
|
|
|
}
|
2006-10-19 21:30:40 +08:00
|
|
|
|
2007-07-26 03:33:14 +08:00
|
|
|
return AsmPrinter::doFinalization(M);
|
2006-05-15 06:18:28 +08:00
|
|
|
}
|
2008-08-17 21:55:10 +08:00
|
|
|
|
|
|
|
/// createARMCodePrinterPass - Returns a pass that prints the ARM
|
|
|
|
/// assembly code for a MachineFunction to the given output stream,
|
|
|
|
/// using the given target machine description. This should work
|
|
|
|
/// regardless of whether the function is in SSA form.
|
|
|
|
///
|
2009-07-15 04:18:05 +08:00
|
|
|
FunctionPass *llvm::createARMCodePrinterPass(formatted_raw_ostream &o,
|
2009-07-16 04:24:03 +08:00
|
|
|
TargetMachine &tm,
|
2009-04-30 07:29:43 +08:00
|
|
|
bool verbose) {
|
2009-07-01 09:48:54 +08:00
|
|
|
return new ARMAsmPrinter(o, tm, tm.getTargetAsmInfo(), verbose);
|
2008-08-17 21:55:10 +08:00
|
|
|
}
|
|
|
|
|
2009-06-24 07:59:40 +08:00
|
|
|
// Force static initialization.
|
2009-07-16 04:24:03 +08:00
|
|
|
extern "C" void LLVMInitializeARMAsmPrinter() {
|
|
|
|
TargetRegistry::RegisterAsmPrinter(TheARMTarget, createARMCodePrinterPass);
|
|
|
|
TargetRegistry::RegisterAsmPrinter(TheThumbTarget, createARMCodePrinterPass);
|
|
|
|
}
|