2005-07-02 06:44:09 +08:00
|
|
|
//===-- X86IntelAsmPrinter.cpp - Convert X86 LLVM code to Intel assembly --===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-30 04:36:04 +08:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2005-07-02 06:44:09 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains a printer that converts from our internal representation
|
|
|
|
// of machine-dependent LLVM code to Intel format assembly language.
|
|
|
|
// This printer is the output mechanism used by `llc'.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2006-12-20 06:59:26 +08:00
|
|
|
#define DEBUG_TYPE "asm-printer"
|
2005-07-02 06:44:09 +08:00
|
|
|
#include "X86IntelAsmPrinter.h"
|
2008-08-24 20:30:46 +08:00
|
|
|
#include "X86InstrInfo.h"
|
|
|
|
#include "X86TargetAsmInfo.h"
|
|
|
|
#include "X86.h"
|
2006-09-21 06:03:51 +08:00
|
|
|
#include "llvm/CallingConv.h"
|
2006-05-02 09:16:28 +08:00
|
|
|
#include "llvm/Constants.h"
|
2008-06-28 19:07:54 +08:00
|
|
|
#include "llvm/DerivedTypes.h"
|
2005-07-02 06:44:09 +08:00
|
|
|
#include "llvm/Module.h"
|
2008-06-28 19:07:54 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2005-07-02 06:44:09 +08:00
|
|
|
#include "llvm/Assembly/Writer.h"
|
2009-02-19 07:12:06 +08:00
|
|
|
#include "llvm/CodeGen/DwarfWriter.h"
|
2005-07-02 06:44:09 +08:00
|
|
|
#include "llvm/Support/Mangler.h"
|
2006-09-08 06:06:40 +08:00
|
|
|
#include "llvm/Target/TargetAsmInfo.h"
|
2006-02-18 08:15:05 +08:00
|
|
|
#include "llvm/Target/TargetOptions.h"
|
2005-07-02 06:44:09 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
2006-12-20 06:59:26 +08:00
|
|
|
STATISTIC(EmittedInsts, "Number of machine instrs printed");
|
|
|
|
|
2008-06-28 19:07:54 +08:00
|
|
|
static X86MachineFunctionInfo calculateFunctionInfo(const Function *F,
|
|
|
|
const TargetData *TD) {
|
|
|
|
X86MachineFunctionInfo Info;
|
|
|
|
uint64_t Size = 0;
|
|
|
|
|
|
|
|
switch (F->getCallingConv()) {
|
|
|
|
case CallingConv::X86_StdCall:
|
|
|
|
Info.setDecorationStyle(StdCall);
|
|
|
|
break;
|
|
|
|
case CallingConv::X86_FastCall:
|
|
|
|
Info.setDecorationStyle(FastCall);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return Info;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned argNum = 1;
|
|
|
|
for (Function::const_arg_iterator AI = F->arg_begin(), AE = F->arg_end();
|
|
|
|
AI != AE; ++AI, ++argNum) {
|
|
|
|
const Type* Ty = AI->getType();
|
|
|
|
|
|
|
|
// 'Dereference' type in case of byval parameter attribute
|
2008-09-26 05:00:45 +08:00
|
|
|
if (F->paramHasAttr(argNum, Attribute::ByVal))
|
2008-06-28 19:07:54 +08:00
|
|
|
Ty = cast<PointerType>(Ty)->getElementType();
|
|
|
|
|
|
|
|
// Size should be aligned to DWORD boundary
|
2009-01-13 04:38:59 +08:00
|
|
|
Size += ((TD->getTypePaddedSize(Ty) + 3)/4)*4;
|
2008-06-28 19:07:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// We're not supporting tooooo huge arguments :)
|
|
|
|
Info.setBytesToPopOnReturn((unsigned int)Size);
|
|
|
|
return Info;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// decorateName - Query FunctionInfoMap and use this information for various
|
|
|
|
/// name decoration.
|
|
|
|
void X86IntelAsmPrinter::decorateName(std::string &Name,
|
|
|
|
const GlobalValue *GV) {
|
|
|
|
const Function *F = dyn_cast<Function>(GV);
|
|
|
|
if (!F) return;
|
|
|
|
|
|
|
|
// We don't want to decorate non-stdcall or non-fastcall functions right now
|
|
|
|
unsigned CC = F->getCallingConv();
|
|
|
|
if (CC != CallingConv::X86_StdCall && CC != CallingConv::X86_FastCall)
|
|
|
|
return;
|
|
|
|
|
|
|
|
FMFInfoMap::const_iterator info_item = FunctionInfoMap.find(F);
|
|
|
|
|
|
|
|
const X86MachineFunctionInfo *Info;
|
|
|
|
if (info_item == FunctionInfoMap.end()) {
|
|
|
|
// Calculate apropriate function info and populate map
|
|
|
|
FunctionInfoMap[F] = calculateFunctionInfo(F, TM.getTargetData());
|
|
|
|
Info = &FunctionInfoMap[F];
|
|
|
|
} else {
|
|
|
|
Info = &info_item->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
const FunctionType *FT = F->getFunctionType();
|
|
|
|
switch (Info->getDecorationStyle()) {
|
|
|
|
case None:
|
|
|
|
break;
|
|
|
|
case StdCall:
|
|
|
|
// "Pure" variadic functions do not receive @0 suffix.
|
|
|
|
if (!FT->isVarArg() || (FT->getNumParams() == 0) ||
|
|
|
|
(FT->getNumParams() == 1 && F->hasStructRetAttr()))
|
|
|
|
Name += '@' + utostr_32(Info->getBytesToPopOnReturn());
|
|
|
|
break;
|
|
|
|
case FastCall:
|
|
|
|
// "Pure" variadic functions do not receive @0 suffix.
|
|
|
|
if (!FT->isVarArg() || (FT->getNumParams() == 0) ||
|
|
|
|
(FT->getNumParams() == 1 && F->hasStructRetAttr()))
|
|
|
|
Name += '@' + utostr_32(Info->getBytesToPopOnReturn());
|
|
|
|
|
|
|
|
if (Name[0] == '_')
|
|
|
|
Name[0] = '@';
|
|
|
|
else
|
|
|
|
Name = '@' + Name;
|
|
|
|
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(0 && "Unsupported DecorationStyle");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-02 06:44:09 +08:00
|
|
|
/// runOnMachineFunction - This uses the printMachineInstruction()
|
|
|
|
/// method to print assembly for each instruction.
|
|
|
|
///
|
|
|
|
bool X86IntelAsmPrinter::runOnMachineFunction(MachineFunction &MF) {
|
2009-02-24 16:30:20 +08:00
|
|
|
this->MF = &MF;
|
2005-11-21 15:51:23 +08:00
|
|
|
SetupMachineFunction(MF);
|
2005-07-02 06:44:09 +08:00
|
|
|
O << "\n\n";
|
|
|
|
|
|
|
|
// Print out constants referenced by the function
|
2005-11-21 16:32:23 +08:00
|
|
|
EmitConstantPool(MF.getConstantPool());
|
2005-07-02 06:44:09 +08:00
|
|
|
|
|
|
|
// Print out labels for the function.
|
2006-09-26 11:57:53 +08:00
|
|
|
const Function *F = MF.getFunction();
|
2006-09-21 06:03:51 +08:00
|
|
|
unsigned CC = F->getCallingConv();
|
|
|
|
|
|
|
|
// Populate function information map. Actually, We don't want to populate
|
|
|
|
// non-stdcall or non-fastcall functions' information right now.
|
2006-09-26 11:57:53 +08:00
|
|
|
if (CC == CallingConv::X86_StdCall || CC == CallingConv::X86_FastCall)
|
2007-04-18 01:21:52 +08:00
|
|
|
FunctionInfoMap[F] = *MF.getInfo<X86MachineFunctionInfo>();
|
2006-09-21 06:03:51 +08:00
|
|
|
|
2008-06-28 19:07:54 +08:00
|
|
|
decorateName(CurrentFnName, F);
|
2006-09-21 06:03:51 +08:00
|
|
|
|
2008-09-25 06:13:07 +08:00
|
|
|
SwitchToTextSection("_text", F);
|
2006-10-05 10:43:52 +08:00
|
|
|
|
2008-10-02 07:18:38 +08:00
|
|
|
unsigned FnAlign = 4;
|
2008-10-07 01:30:07 +08:00
|
|
|
if (F->hasFnAttr(Attribute::OptimizeForSize))
|
2008-09-05 05:03:41 +08:00
|
|
|
FnAlign = 1;
|
2006-09-15 02:23:27 +08:00
|
|
|
switch (F->getLinkage()) {
|
|
|
|
default: assert(0 && "Unsupported linkage type!");
|
2009-01-16 04:18:42 +08:00
|
|
|
case Function::PrivateLinkage:
|
2006-09-15 02:23:27 +08:00
|
|
|
case Function::InternalLinkage:
|
2008-03-26 06:29:46 +08:00
|
|
|
EmitAlignment(FnAlign);
|
2008-06-28 19:07:18 +08:00
|
|
|
break;
|
2006-09-15 02:23:27 +08:00
|
|
|
case Function::DLLExportLinkage:
|
|
|
|
DLLExportedFns.insert(CurrentFnName);
|
|
|
|
//FALLS THROUGH
|
|
|
|
case Function::ExternalLinkage:
|
2006-05-02 09:16:28 +08:00
|
|
|
O << "\tpublic " << CurrentFnName << "\n";
|
2008-03-26 06:29:46 +08:00
|
|
|
EmitAlignment(FnAlign);
|
2008-06-28 19:07:18 +08:00
|
|
|
break;
|
2006-09-15 02:23:27 +08:00
|
|
|
}
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2006-05-02 09:16:28 +08:00
|
|
|
O << CurrentFnName << "\tproc near\n";
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2005-07-02 06:44:09 +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 there are any predecessors.
|
2007-10-04 03:26:29 +08:00
|
|
|
if (!I->pred_empty()) {
|
2008-02-28 08:43:03 +08:00
|
|
|
printBasicBlockLabel(I, true, true);
|
2006-05-02 13:37:32 +08:00
|
|
|
O << '\n';
|
|
|
|
}
|
2005-07-02 06:44:09 +08:00
|
|
|
for (MachineBasicBlock::const_iterator II = I->begin(), E = I->end();
|
|
|
|
II != E; ++II) {
|
|
|
|
// Print the assembly for the instruction.
|
|
|
|
printMachineInstruction(II);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-10-05 11:01:21 +08:00
|
|
|
// Print out jump tables referenced by the function.
|
|
|
|
EmitJumpTableInfo(MF.getJumpTableInfo(), MF);
|
|
|
|
|
2006-05-02 09:16:28 +08:00
|
|
|
O << CurrentFnName << "\tendp\n";
|
|
|
|
|
2008-11-08 03:49:17 +08:00
|
|
|
O.flush();
|
|
|
|
|
2005-07-02 06:44:09 +08:00
|
|
|
// We didn't modify anything.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2005-12-01 02:54:35 +08:00
|
|
|
void X86IntelAsmPrinter::printSSECC(const MachineInstr *MI, unsigned Op) {
|
2007-12-31 04:49:49 +08:00
|
|
|
unsigned char value = MI->getOperand(Op).getImm();
|
2005-07-15 06:52:25 +08:00
|
|
|
assert(value <= 7 && "Invalid ssecc argument!");
|
|
|
|
switch (value) {
|
|
|
|
case 0: O << "eq"; break;
|
|
|
|
case 1: O << "lt"; break;
|
|
|
|
case 2: O << "le"; break;
|
|
|
|
case 3: O << "unord"; break;
|
|
|
|
case 4: O << "neq"; break;
|
|
|
|
case 5: O << "nlt"; break;
|
|
|
|
case 6: O << "nle"; break;
|
|
|
|
case 7: O << "ord"; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-28 19:07:18 +08:00
|
|
|
void X86IntelAsmPrinter::printOp(const MachineOperand &MO,
|
2006-02-07 07:41:19 +08:00
|
|
|
const char *Modifier) {
|
2005-07-02 06:44:09 +08:00
|
|
|
switch (MO.getType()) {
|
2008-06-28 19:07:18 +08:00
|
|
|
case MachineOperand::MO_Register: {
|
2008-02-11 02:45:23 +08:00
|
|
|
if (TargetRegisterInfo::isPhysicalRegister(MO.getReg())) {
|
2006-05-05 13:40:20 +08:00
|
|
|
unsigned Reg = MO.getReg();
|
2006-06-01 06:34:26 +08:00
|
|
|
if (Modifier && strncmp(Modifier, "subreg", strlen("subreg")) == 0) {
|
2008-06-06 20:08:01 +08:00
|
|
|
MVT VT = (strcmp(Modifier,"subreg64") == 0) ?
|
2006-09-08 14:48:29 +08:00
|
|
|
MVT::i64 : ((strcmp(Modifier, "subreg32") == 0) ? MVT::i32 :
|
|
|
|
((strcmp(Modifier,"subreg16") == 0) ? MVT::i16 :MVT::i8));
|
2006-05-05 13:40:20 +08:00
|
|
|
Reg = getX86SubSuperRegister(Reg, VT);
|
|
|
|
}
|
2008-07-08 06:21:06 +08:00
|
|
|
O << TRI->getName(Reg);
|
2006-05-05 13:40:20 +08:00
|
|
|
} else
|
2006-05-01 13:53:50 +08:00
|
|
|
O << "reg" << MO.getReg();
|
2005-07-02 06:44:09 +08:00
|
|
|
return;
|
2006-12-20 05:04:20 +08:00
|
|
|
}
|
2006-05-05 01:21:20 +08:00
|
|
|
case MachineOperand::MO_Immediate:
|
2007-12-31 04:49:49 +08:00
|
|
|
O << MO.getImm();
|
2005-07-02 06:44:09 +08:00
|
|
|
return;
|
2006-04-23 02:53:45 +08:00
|
|
|
case MachineOperand::MO_MachineBasicBlock:
|
2007-12-31 07:10:15 +08:00
|
|
|
printBasicBlockLabel(MO.getMBB());
|
2005-07-02 06:44:09 +08:00
|
|
|
return;
|
2006-12-20 05:04:20 +08:00
|
|
|
case MachineOperand::MO_JumpTableIndex: {
|
|
|
|
bool isMemOp = Modifier && !strcmp(Modifier, "mem");
|
|
|
|
if (!isMemOp) O << "OFFSET ";
|
2007-10-14 13:57:21 +08:00
|
|
|
O << TAI->getPrivateGlobalPrefix() << "JTI" << getFunctionNumber()
|
2007-12-31 07:10:15 +08:00
|
|
|
<< "_" << MO.getIndex();
|
2006-12-20 05:04:20 +08:00
|
|
|
return;
|
2008-06-28 19:07:18 +08:00
|
|
|
}
|
2006-02-26 16:28:12 +08:00
|
|
|
case MachineOperand::MO_ConstantPoolIndex: {
|
|
|
|
bool isMemOp = Modifier && !strcmp(Modifier, "mem");
|
|
|
|
if (!isMemOp) O << "OFFSET ";
|
2006-09-07 02:34:40 +08:00
|
|
|
O << "[" << TAI->getPrivateGlobalPrefix() << "CPI"
|
2007-12-31 07:10:15 +08:00
|
|
|
<< getFunctionNumber() << "_" << MO.getIndex();
|
2008-11-23 00:15:34 +08:00
|
|
|
printOffset(MO.getOffset());
|
2006-02-26 16:28:12 +08:00
|
|
|
O << "]";
|
|
|
|
return;
|
|
|
|
}
|
2005-07-02 06:44:09 +08:00
|
|
|
case MachineOperand::MO_GlobalAddress: {
|
2006-02-18 08:15:05 +08:00
|
|
|
bool isCallOp = Modifier && !strcmp(Modifier, "call");
|
|
|
|
bool isMemOp = Modifier && !strcmp(Modifier, "mem");
|
2008-06-28 19:07:18 +08:00
|
|
|
GlobalValue *GV = MO.getGlobal();
|
2006-09-21 06:03:51 +08:00
|
|
|
std::string Name = Mang->getValueName(GV);
|
|
|
|
|
2008-06-28 19:07:54 +08:00
|
|
|
decorateName(Name, GV);
|
2006-09-15 02:23:27 +08:00
|
|
|
|
2006-02-18 08:15:05 +08:00
|
|
|
if (!isMemOp && !isCallOp) O << "OFFSET ";
|
2006-09-15 02:23:27 +08:00
|
|
|
if (GV->hasDLLImportLinkage()) {
|
|
|
|
// FIXME: This should be fixed with full support of stdcall & fastcall
|
|
|
|
// CC's
|
2008-06-28 19:07:18 +08:00
|
|
|
O << "__imp_";
|
|
|
|
}
|
2006-09-21 06:03:51 +08:00
|
|
|
O << Name;
|
2008-11-23 00:15:34 +08:00
|
|
|
printOffset(MO.getOffset());
|
2005-07-02 06:44:09 +08:00
|
|
|
return;
|
|
|
|
}
|
2006-02-18 08:15:05 +08:00
|
|
|
case MachineOperand::MO_ExternalSymbol: {
|
|
|
|
bool isCallOp = Modifier && !strcmp(Modifier, "call");
|
2006-02-23 04:19:42 +08:00
|
|
|
if (!isCallOp) O << "OFFSET ";
|
2006-09-07 02:34:40 +08:00
|
|
|
O << TAI->getGlobalPrefix() << MO.getSymbolName();
|
2005-07-02 06:44:09 +08:00
|
|
|
return;
|
2006-02-18 08:15:05 +08:00
|
|
|
}
|
2005-07-02 06:44:09 +08:00
|
|
|
default:
|
|
|
|
O << "<unknown operand type>"; return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-09-08 14:48:29 +08:00
|
|
|
void X86IntelAsmPrinter::printMemReference(const MachineInstr *MI, unsigned Op,
|
|
|
|
const char *Modifier) {
|
2005-07-02 06:44:09 +08:00
|
|
|
assert(isMem(MI, Op) && "Invalid memory reference!");
|
|
|
|
|
|
|
|
const MachineOperand &BaseReg = MI->getOperand(Op);
|
2007-12-31 04:49:49 +08:00
|
|
|
int ScaleVal = MI->getOperand(Op+1).getImm();
|
2005-07-02 06:44:09 +08:00
|
|
|
const MachineOperand &IndexReg = MI->getOperand(Op+2);
|
|
|
|
const MachineOperand &DispSpec = MI->getOperand(Op+3);
|
|
|
|
|
|
|
|
O << "[";
|
|
|
|
bool NeedPlus = false;
|
|
|
|
if (BaseReg.getReg()) {
|
2006-09-08 14:48:29 +08:00
|
|
|
printOp(BaseReg, Modifier);
|
2005-07-02 06:44:09 +08:00
|
|
|
NeedPlus = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IndexReg.getReg()) {
|
|
|
|
if (NeedPlus) O << " + ";
|
|
|
|
if (ScaleVal != 1)
|
|
|
|
O << ScaleVal << "*";
|
2006-09-08 14:48:29 +08:00
|
|
|
printOp(IndexReg, Modifier);
|
2005-07-02 06:44:09 +08:00
|
|
|
NeedPlus = true;
|
|
|
|
}
|
|
|
|
|
2008-10-03 23:45:36 +08:00
|
|
|
if (DispSpec.isGlobal() || DispSpec.isCPI() ||
|
|
|
|
DispSpec.isJTI()) {
|
2005-07-02 06:44:09 +08:00
|
|
|
if (NeedPlus)
|
|
|
|
O << " + ";
|
2006-02-07 16:38:37 +08:00
|
|
|
printOp(DispSpec, "mem");
|
2005-07-02 06:44:09 +08:00
|
|
|
} else {
|
2007-12-31 04:49:49 +08:00
|
|
|
int DispVal = DispSpec.getImm();
|
2005-07-02 06:44:09 +08:00
|
|
|
if (DispVal || (!BaseReg.getReg() && !IndexReg.getReg())) {
|
2008-02-20 19:22:39 +08:00
|
|
|
if (NeedPlus) {
|
2005-07-02 06:44:09 +08:00
|
|
|
if (DispVal > 0)
|
|
|
|
O << " + ";
|
|
|
|
else {
|
|
|
|
O << " - ";
|
|
|
|
DispVal = -DispVal;
|
|
|
|
}
|
2008-02-20 19:22:39 +08:00
|
|
|
}
|
2005-07-02 06:44:09 +08:00
|
|
|
O << DispVal;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
O << "]";
|
|
|
|
}
|
|
|
|
|
2008-06-28 19:07:18 +08:00
|
|
|
void X86IntelAsmPrinter::printPICJumpTableSetLabel(unsigned uid,
|
Much improved pic jumptable codegen:
Then:
call "L1$pb"
"L1$pb":
popl %eax
...
LBB1_1: # entry
imull $4, %ecx, %ecx
leal LJTI1_0-"L1$pb"(%eax), %edx
addl LJTI1_0-"L1$pb"(%ecx,%eax), %edx
jmpl *%edx
.align 2
.set L1_0_set_3,LBB1_3-LJTI1_0
.set L1_0_set_2,LBB1_2-LJTI1_0
.set L1_0_set_5,LBB1_5-LJTI1_0
.set L1_0_set_4,LBB1_4-LJTI1_0
LJTI1_0:
.long L1_0_set_3
.long L1_0_set_2
Now:
call "L1$pb"
"L1$pb":
popl %eax
...
LBB1_1: # entry
addl LJTI1_0-"L1$pb"(%eax,%ecx,4), %eax
jmpl *%eax
.align 2
.set L1_0_set_3,LBB1_3-"L1$pb"
.set L1_0_set_2,LBB1_2-"L1$pb"
.set L1_0_set_5,LBB1_5-"L1$pb"
.set L1_0_set_4,LBB1_4-"L1$pb"
LJTI1_0:
.long L1_0_set_3
.long L1_0_set_2
llvm-svn: 43924
2007-11-09 09:32:10 +08:00
|
|
|
const MachineBasicBlock *MBB) const {
|
|
|
|
if (!TAI->getSetDirective())
|
|
|
|
return;
|
2008-06-28 19:07:18 +08:00
|
|
|
|
Much improved pic jumptable codegen:
Then:
call "L1$pb"
"L1$pb":
popl %eax
...
LBB1_1: # entry
imull $4, %ecx, %ecx
leal LJTI1_0-"L1$pb"(%eax), %edx
addl LJTI1_0-"L1$pb"(%ecx,%eax), %edx
jmpl *%edx
.align 2
.set L1_0_set_3,LBB1_3-LJTI1_0
.set L1_0_set_2,LBB1_2-LJTI1_0
.set L1_0_set_5,LBB1_5-LJTI1_0
.set L1_0_set_4,LBB1_4-LJTI1_0
LJTI1_0:
.long L1_0_set_3
.long L1_0_set_2
Now:
call "L1$pb"
"L1$pb":
popl %eax
...
LBB1_1: # entry
addl LJTI1_0-"L1$pb"(%eax,%ecx,4), %eax
jmpl *%eax
.align 2
.set L1_0_set_3,LBB1_3-"L1$pb"
.set L1_0_set_2,LBB1_2-"L1$pb"
.set L1_0_set_5,LBB1_5-"L1$pb"
.set L1_0_set_4,LBB1_4-"L1$pb"
LJTI1_0:
.long L1_0_set_3
.long L1_0_set_2
llvm-svn: 43924
2007-11-09 09:32:10 +08:00
|
|
|
O << TAI->getSetDirective() << ' ' << TAI->getPrivateGlobalPrefix()
|
|
|
|
<< getFunctionNumber() << '_' << uid << "_set_" << MBB->getNumber() << ',';
|
2008-02-28 08:43:03 +08:00
|
|
|
printBasicBlockLabel(MBB, false, false, false);
|
Much improved pic jumptable codegen:
Then:
call "L1$pb"
"L1$pb":
popl %eax
...
LBB1_1: # entry
imull $4, %ecx, %ecx
leal LJTI1_0-"L1$pb"(%eax), %edx
addl LJTI1_0-"L1$pb"(%ecx,%eax), %edx
jmpl *%edx
.align 2
.set L1_0_set_3,LBB1_3-LJTI1_0
.set L1_0_set_2,LBB1_2-LJTI1_0
.set L1_0_set_5,LBB1_5-LJTI1_0
.set L1_0_set_4,LBB1_4-LJTI1_0
LJTI1_0:
.long L1_0_set_3
.long L1_0_set_2
Now:
call "L1$pb"
"L1$pb":
popl %eax
...
LBB1_1: # entry
addl LJTI1_0-"L1$pb"(%eax,%ecx,4), %eax
jmpl *%eax
.align 2
.set L1_0_set_3,LBB1_3-"L1$pb"
.set L1_0_set_2,LBB1_2-"L1$pb"
.set L1_0_set_5,LBB1_5-"L1$pb"
.set L1_0_set_4,LBB1_4-"L1$pb"
LJTI1_0:
.long L1_0_set_3
.long L1_0_set_2
llvm-svn: 43924
2007-11-09 09:32:10 +08:00
|
|
|
O << '-' << "\"L" << getFunctionNumber() << "$pb\"'\n";
|
|
|
|
}
|
|
|
|
|
2006-02-18 08:15:05 +08:00
|
|
|
void X86IntelAsmPrinter::printPICLabel(const MachineInstr *MI, unsigned Op) {
|
2007-10-14 13:57:21 +08:00
|
|
|
O << "\"L" << getFunctionNumber() << "$pb\"\n";
|
|
|
|
O << "\"L" << getFunctionNumber() << "$pb\":";
|
2006-02-18 08:15:05 +08:00
|
|
|
}
|
2005-07-02 06:44:09 +08:00
|
|
|
|
2006-04-29 07:19:39 +08:00
|
|
|
bool X86IntelAsmPrinter::printAsmMRegister(const MachineOperand &MO,
|
2006-04-29 07:11:40 +08:00
|
|
|
const char Mode) {
|
|
|
|
unsigned Reg = MO.getReg();
|
|
|
|
switch (Mode) {
|
|
|
|
default: return true; // Unknown mode.
|
|
|
|
case 'b': // Print QImode register
|
2006-05-05 13:40:20 +08:00
|
|
|
Reg = getX86SubSuperRegister(Reg, MVT::i8);
|
2006-04-29 07:11:40 +08:00
|
|
|
break;
|
|
|
|
case 'h': // Print QImode high register
|
2006-05-05 13:40:20 +08:00
|
|
|
Reg = getX86SubSuperRegister(Reg, MVT::i8, true);
|
2006-04-29 07:11:40 +08:00
|
|
|
break;
|
|
|
|
case 'w': // Print HImode register
|
2006-05-05 13:40:20 +08:00
|
|
|
Reg = getX86SubSuperRegister(Reg, MVT::i16);
|
2006-04-29 07:11:40 +08:00
|
|
|
break;
|
|
|
|
case 'k': // Print SImode register
|
2006-05-05 13:40:20 +08:00
|
|
|
Reg = getX86SubSuperRegister(Reg, MVT::i32);
|
2006-04-29 07:11:40 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-07-08 06:21:06 +08:00
|
|
|
O << '%' << TRI->getName(Reg);
|
2006-04-29 07:11:40 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2006-04-29 05:19:05 +08:00
|
|
|
/// PrintAsmOperand - Print out an operand for an inline asm expression.
|
|
|
|
///
|
|
|
|
bool X86IntelAsmPrinter::PrintAsmOperand(const MachineInstr *MI, unsigned OpNo,
|
2008-06-28 19:07:18 +08:00
|
|
|
unsigned AsmVariant,
|
2006-04-29 05:19:05 +08:00
|
|
|
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.
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2006-04-29 05:19:05 +08:00
|
|
|
switch (ExtraCode[0]) {
|
|
|
|
default: return true; // Unknown modifier.
|
2006-04-29 07:11:40 +08:00
|
|
|
case 'b': // Print QImode register
|
|
|
|
case 'h': // Print QImode high register
|
|
|
|
case 'w': // Print HImode register
|
|
|
|
case 'k': // Print SImode register
|
2006-04-29 07:19:39 +08:00
|
|
|
return printAsmMRegister(MI->getOperand(OpNo), ExtraCode[0]);
|
2006-04-29 05:19:05 +08:00
|
|
|
}
|
|
|
|
}
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2006-04-29 05:19:05 +08:00
|
|
|
printOperand(MI, OpNo);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool X86IntelAsmPrinter::PrintAsmMemoryOperand(const MachineInstr *MI,
|
|
|
|
unsigned OpNo,
|
2008-06-28 19:07:18 +08:00
|
|
|
unsigned AsmVariant,
|
2006-04-29 05:19:05 +08:00
|
|
|
const char *ExtraCode) {
|
|
|
|
if (ExtraCode && ExtraCode[0])
|
|
|
|
return true; // Unknown modifier.
|
|
|
|
printMemReference(MI, OpNo);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2005-07-02 06:44:09 +08:00
|
|
|
/// printMachineInstruction -- Print out a single X86 LLVM instruction
|
|
|
|
/// MI in Intel syntax to the current output stream.
|
|
|
|
///
|
|
|
|
void X86IntelAsmPrinter::printMachineInstruction(const MachineInstr *MI) {
|
|
|
|
++EmittedInsts;
|
|
|
|
|
|
|
|
// Call the autogenerated instruction printer routines.
|
|
|
|
printInstruction(MI);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool X86IntelAsmPrinter::doInitialization(Module &M) {
|
2008-06-28 19:07:35 +08:00
|
|
|
bool Result = AsmPrinter::doInitialization(M);
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2006-09-07 02:34:40 +08:00
|
|
|
Mang->markCharUnacceptable('.');
|
2006-05-05 00:20:22 +08:00
|
|
|
|
2006-05-02 11:11:50 +08:00
|
|
|
O << "\t.686\n\t.model flat\n\n";
|
|
|
|
|
|
|
|
// Emit declarations for external functions.
|
|
|
|
for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I)
|
2007-01-31 04:08:39 +08:00
|
|
|
if (I->isDeclaration()) {
|
2006-09-21 06:03:51 +08:00
|
|
|
std::string Name = Mang->getValueName(I);
|
2008-06-28 19:07:54 +08:00
|
|
|
decorateName(Name, I);
|
2006-05-02 11:11:50 +08:00
|
|
|
|
2006-09-21 06:03:51 +08:00
|
|
|
O << "\textern " ;
|
|
|
|
if (I->hasDLLImportLinkage()) {
|
|
|
|
O << "__imp_";
|
2008-06-28 19:07:18 +08:00
|
|
|
}
|
2006-09-21 06:03:51 +08:00
|
|
|
O << Name << ":near\n";
|
|
|
|
}
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2006-05-07 05:27:14 +08:00
|
|
|
// Emit declarations for external globals. Note that VC++ always declares
|
|
|
|
// external globals to have type byte, and if that's good enough for VC++...
|
2006-05-02 11:11:50 +08:00
|
|
|
for (Module::const_global_iterator I = M.global_begin(), E = M.global_end();
|
|
|
|
I != E; ++I) {
|
2007-01-31 04:08:39 +08:00
|
|
|
if (I->isDeclaration()) {
|
2006-09-21 06:03:51 +08:00
|
|
|
std::string Name = Mang->getValueName(I);
|
|
|
|
|
|
|
|
O << "\textern " ;
|
|
|
|
if (I->hasDLLImportLinkage()) {
|
|
|
|
O << "__imp_";
|
2008-06-28 19:07:18 +08:00
|
|
|
}
|
2006-09-21 06:03:51 +08:00
|
|
|
O << Name << ":byte\n";
|
|
|
|
}
|
2006-05-02 11:11:50 +08:00
|
|
|
}
|
|
|
|
|
2007-07-26 03:33:14 +08:00
|
|
|
return Result;
|
2006-05-02 11:11:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool X86IntelAsmPrinter::doFinalization(Module &M) {
|
2006-05-07 05:27:14 +08:00
|
|
|
const TargetData *TD = TM.getTargetData();
|
|
|
|
|
|
|
|
// Print out module-level global variables here.
|
|
|
|
for (Module::const_global_iterator I = M.global_begin(), E = M.global_end();
|
|
|
|
I != E; ++I) {
|
2007-01-31 04:08:39 +08:00
|
|
|
if (I->isDeclaration()) continue; // External global require no code
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2006-05-07 05:27:14 +08:00
|
|
|
// Check to see if this is a special global used by LLVM, if so, emit it.
|
|
|
|
if (EmitSpecialLLVMGlobal(I))
|
|
|
|
continue;
|
2008-06-28 19:07:18 +08:00
|
|
|
|
2006-05-07 05:27:14 +08:00
|
|
|
std::string name = Mang->getValueName(I);
|
|
|
|
Constant *C = I->getInitializer();
|
2006-10-25 04:32:14 +08:00
|
|
|
unsigned Align = TD->getPreferredAlignmentLog(I);
|
2006-05-07 05:27:14 +08:00
|
|
|
bool bCustomSegment = false;
|
|
|
|
|
|
|
|
switch (I->getLinkage()) {
|
2009-03-12 04:14:15 +08:00
|
|
|
case GlobalValue::CommonLinkage:
|
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 GlobalValue::LinkOnceAnyLinkage:
|
|
|
|
case GlobalValue::LinkOnceODRLinkage:
|
|
|
|
case GlobalValue::WeakAnyLinkage:
|
|
|
|
case GlobalValue::WeakODRLinkage:
|
2006-10-31 16:31:24 +08:00
|
|
|
SwitchToDataSection("");
|
2006-05-07 05:27:14 +08:00
|
|
|
O << name << "?\tsegment common 'COMMON'\n";
|
|
|
|
bCustomSegment = true;
|
|
|
|
// FIXME: the default alignment is 16 bytes, but 1, 2, 4, and 256
|
|
|
|
// are also available.
|
|
|
|
break;
|
|
|
|
case GlobalValue::AppendingLinkage:
|
2006-10-31 16:31:24 +08:00
|
|
|
SwitchToDataSection("");
|
2006-05-07 05:27:14 +08:00
|
|
|
O << name << "?\tsegment public 'DATA'\n";
|
|
|
|
bCustomSegment = true;
|
|
|
|
// FIXME: the default alignment is 16 bytes, but 1, 2, 4, and 256
|
|
|
|
// are also available.
|
|
|
|
break;
|
2006-09-15 02:23:27 +08:00
|
|
|
case GlobalValue::DLLExportLinkage:
|
|
|
|
DLLExportedGVs.insert(name);
|
|
|
|
// FALL THROUGH
|
2006-05-07 05:27:14 +08:00
|
|
|
case GlobalValue::ExternalLinkage:
|
|
|
|
O << "\tpublic " << name << "\n";
|
|
|
|
// FALL THROUGH
|
|
|
|
case GlobalValue::InternalLinkage:
|
2008-09-25 06:16:16 +08:00
|
|
|
SwitchToSection(TAI->getDataSection());
|
2006-05-07 05:27:14 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(0 && "Unknown linkage type!");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bCustomSegment)
|
|
|
|
EmitAlignment(Align, I);
|
|
|
|
|
2009-03-24 08:17:40 +08:00
|
|
|
O << name << ":";
|
|
|
|
if (VerboseAsm)
|
|
|
|
O << name << "\t\t\t\t" << TAI->getCommentString()
|
|
|
|
<< " " << I->getName();
|
|
|
|
O << '\n';
|
2006-05-07 05:27:14 +08:00
|
|
|
|
|
|
|
EmitGlobalConstant(C);
|
|
|
|
|
|
|
|
if (bCustomSegment)
|
|
|
|
O << name << "?\tends\n";
|
|
|
|
}
|
2006-09-15 02:23:27 +08:00
|
|
|
|
|
|
|
// Output linker support code for dllexported globals
|
2008-06-28 19:07:35 +08:00
|
|
|
if (!DLLExportedGVs.empty() || !DLLExportedFns.empty()) {
|
2006-10-31 16:31:24 +08:00
|
|
|
SwitchToDataSection("");
|
2008-09-20 08:13:08 +08:00
|
|
|
O << "; WARNING: The following code is valid only with MASM v8.x"
|
|
|
|
<< "and (possible) higher\n"
|
|
|
|
<< "; This version of MASM is usually shipped with Microsoft "
|
|
|
|
<< "Visual Studio 2005\n"
|
|
|
|
<< "; or (possible) further versions. Unfortunately, there is no "
|
|
|
|
<< "way to support\n"
|
|
|
|
<< "; dllexported symbols in the earlier versions of MASM in fully "
|
|
|
|
<< "automatic way\n\n";
|
2006-09-15 02:23:27 +08:00
|
|
|
O << "_drectve\t segment info alias('.drectve')\n";
|
|
|
|
}
|
|
|
|
|
2008-06-28 05:22:49 +08:00
|
|
|
for (StringSet<>::iterator i = DLLExportedGVs.begin(),
|
2006-09-15 02:23:27 +08:00
|
|
|
e = DLLExportedGVs.end();
|
2008-06-28 19:07:35 +08:00
|
|
|
i != e; ++i)
|
2008-06-28 05:22:49 +08:00
|
|
|
O << "\t db ' /EXPORT:" << i->getKeyData() << ",data'\n";
|
2006-09-15 02:23:27 +08:00
|
|
|
|
2008-06-28 05:22:49 +08:00
|
|
|
for (StringSet<>::iterator i = DLLExportedFns.begin(),
|
2006-09-15 02:23:27 +08:00
|
|
|
e = DLLExportedFns.end();
|
2008-06-28 19:07:35 +08:00
|
|
|
i != e; ++i)
|
2008-06-28 05:22:49 +08:00
|
|
|
O << "\t db ' /EXPORT:" << i->getKeyData() << "'\n";
|
2006-09-15 02:23:27 +08:00
|
|
|
|
2008-06-28 19:07:35 +08:00
|
|
|
if (!DLLExportedGVs.empty() || !DLLExportedFns.empty())
|
2008-06-28 05:22:49 +08:00
|
|
|
O << "_drectve\t ends\n";
|
|
|
|
|
2006-05-07 05:27:14 +08:00
|
|
|
// Bypass X86SharedAsmPrinter::doFinalization().
|
2007-07-26 03:33:14 +08:00
|
|
|
bool Result = AsmPrinter::doFinalization(M);
|
2006-10-31 16:31:24 +08:00
|
|
|
SwitchToDataSection("");
|
2006-05-02 11:11:50 +08:00
|
|
|
O << "\tend\n";
|
2007-07-26 03:33:14 +08:00
|
|
|
return Result;
|
2005-07-02 06:44:09 +08:00
|
|
|
}
|
|
|
|
|
2006-05-02 09:16:28 +08:00
|
|
|
void X86IntelAsmPrinter::EmitString(const ConstantArray *CVA) const {
|
|
|
|
unsigned NumElts = CVA->getNumOperands();
|
|
|
|
if (NumElts) {
|
|
|
|
// ML does not have escape sequences except '' for '. It also has a maximum
|
|
|
|
// string length of 255.
|
|
|
|
unsigned len = 0;
|
|
|
|
bool inString = false;
|
|
|
|
for (unsigned i = 0; i < NumElts; i++) {
|
2006-10-20 15:07:24 +08:00
|
|
|
int n = cast<ConstantInt>(CVA->getOperand(i))->getZExtValue() & 255;
|
2006-05-02 09:16:28 +08:00
|
|
|
if (len == 0)
|
|
|
|
O << "\tdb ";
|
|
|
|
|
|
|
|
if (n >= 32 && n <= 127) {
|
|
|
|
if (!inString) {
|
|
|
|
if (len > 0) {
|
|
|
|
O << ",'";
|
|
|
|
len += 2;
|
|
|
|
} else {
|
|
|
|
O << "'";
|
|
|
|
len++;
|
|
|
|
}
|
|
|
|
inString = true;
|
|
|
|
}
|
|
|
|
if (n == '\'') {
|
|
|
|
O << "'";
|
|
|
|
len++;
|
|
|
|
}
|
|
|
|
O << char(n);
|
|
|
|
} else {
|
|
|
|
if (inString) {
|
|
|
|
O << "'";
|
|
|
|
len++;
|
|
|
|
inString = false;
|
|
|
|
}
|
|
|
|
if (len > 0) {
|
|
|
|
O << ",";
|
|
|
|
len++;
|
|
|
|
}
|
|
|
|
O << n;
|
|
|
|
len += 1 + (n > 9) + (n > 99);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len > 60) {
|
|
|
|
if (inString) {
|
|
|
|
O << "'";
|
|
|
|
inString = false;
|
|
|
|
}
|
|
|
|
O << "\n";
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len > 0) {
|
|
|
|
if (inString)
|
|
|
|
O << "'";
|
|
|
|
O << "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-02 06:44:09 +08:00
|
|
|
// Include the auto-generated portion of the assembly writer.
|
|
|
|
#include "X86GenAsmWriter1.inc"
|