2006-09-08 06:05:02 +08:00
|
|
|
//===-- X86TargetAsmInfo.cpp - X86 asm properties ---------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// 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.
|
2006-09-08 06:05:02 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains the declarations of the X86TargetAsmInfo properties.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "X86TargetAsmInfo.h"
|
|
|
|
#include "X86TargetMachine.h"
|
|
|
|
#include "X86Subtarget.h"
|
2007-01-13 07:22:14 +08:00
|
|
|
#include "llvm/DerivedTypes.h"
|
2006-11-29 09:14:06 +08:00
|
|
|
#include "llvm/InlineAsm.h"
|
|
|
|
#include "llvm/Instructions.h"
|
2007-04-02 04:49:36 +08:00
|
|
|
#include "llvm/Intrinsics.h"
|
2006-11-29 09:14:06 +08:00
|
|
|
#include "llvm/Module.h"
|
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2008-02-28 07:33:50 +08:00
|
|
|
#include "llvm/Support/Dwarf.h"
|
|
|
|
|
2006-09-08 06:05:02 +08:00
|
|
|
using namespace llvm;
|
2008-02-28 07:33:50 +08:00
|
|
|
using namespace llvm::dwarf;
|
2006-09-08 06:05:02 +08:00
|
|
|
|
2008-03-26 05:45:14 +08:00
|
|
|
static const char *const x86_asm_table[] = {
|
|
|
|
"{si}", "S",
|
2006-11-29 03:52:49 +08:00
|
|
|
"{di}", "D",
|
|
|
|
"{ax}", "a",
|
|
|
|
"{cx}", "c",
|
|
|
|
"{memory}", "memory",
|
|
|
|
"{flags}", "",
|
|
|
|
"{dirflag}", "",
|
|
|
|
"{fpsr}", "",
|
|
|
|
"{cc}", "cc",
|
|
|
|
0,0};
|
|
|
|
|
2006-09-08 06:05:02 +08:00
|
|
|
X86TargetAsmInfo::X86TargetAsmInfo(const X86TargetMachine &TM) {
|
|
|
|
const X86Subtarget *Subtarget = &TM.getSubtarget<X86Subtarget>();
|
2008-02-28 07:33:50 +08:00
|
|
|
X86TM = &TM;
|
|
|
|
|
2006-11-29 03:52:49 +08:00
|
|
|
AsmTransCBE = x86_asm_table;
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2007-01-16 17:29:17 +08:00
|
|
|
AssemblerDialect = Subtarget->getAsmFlavor();
|
2006-09-08 06:05:02 +08:00
|
|
|
}
|
2006-10-05 10:43:52 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
bool X86TargetAsmInfo::LowerToBSwap(CallInst *CI) const {
|
|
|
|
// FIXME: this should verify that we are targetting a 486 or better. If not,
|
|
|
|
// we will turn this bswap into something that will be lowered to logical ops
|
|
|
|
// instead of emitting the bswap asm. For now, we don't support 486 or lower
|
|
|
|
// so don't worry about this.
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
// Verify this is a simple bswap.
|
|
|
|
if (CI->getNumOperands() != 2 ||
|
|
|
|
CI->getType() != CI->getOperand(1)->getType() ||
|
2007-01-15 10:27:26 +08:00
|
|
|
!CI->getType()->isInteger())
|
2006-11-29 09:14:06 +08:00
|
|
|
return false;
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2007-04-02 04:49:36 +08:00
|
|
|
const IntegerType *Ty = dyn_cast<IntegerType>(CI->getType());
|
|
|
|
if (!Ty || Ty->getBitWidth() % 16 != 0)
|
For PR1064:
Implement the arbitrary bit-width integer feature. The feature allows
integers of any bitwidth (up to 64) to be defined instead of just 1, 8,
16, 32, and 64 bit integers.
This change does several things:
1. Introduces a new Derived Type, IntegerType, to represent the number of
bits in an integer. The Type classes SubclassData field is used to
store the number of bits. This allows 2^23 bits in an integer type.
2. Removes the five integer Type::TypeID values for the 1, 8, 16, 32 and
64-bit integers. These are replaced with just IntegerType which is not
a primitive any more.
3. Adjust the rest of LLVM to account for this change.
Note that while this incremental change lays the foundation for arbitrary
bit-width integers, LLVM has not yet been converted to actually deal with
them in any significant way. Most optimization passes, for example, will
still only deal with the byte-width integer types. Future increments
will rectify this situation.
llvm-svn: 33113
2007-01-12 15:05:14 +08:00
|
|
|
return false;
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
// Okay, we can do this xform, do so now.
|
2007-08-04 09:51:18 +08:00
|
|
|
const Type *Tys[] = { Ty };
|
2006-11-29 09:14:06 +08:00
|
|
|
Module *M = CI->getParent()->getParent()->getParent();
|
2007-08-04 09:51:18 +08:00
|
|
|
Constant *Int = Intrinsic::getDeclaration(M, Intrinsic::bswap, Tys, 1);
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
Value *Op = CI->getOperand(1);
|
2008-04-07 04:25:17 +08:00
|
|
|
Op = CallInst::Create(Int, Op, CI->getName(), CI);
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
CI->replaceAllUsesWith(Op);
|
|
|
|
CI->eraseFromParent();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool X86TargetAsmInfo::ExpandInlineAsm(CallInst *CI) const {
|
|
|
|
InlineAsm *IA = cast<InlineAsm>(CI->getCalledValue());
|
Upgrade the ugly darwin 64-bit bswap idiom (bswap %eax / bswap %edx /
xchgl %eax, %edx) to llvm.bswap.i64. This compiles:
long long test2(long long A) {
return _OSSwapInt64(A);
}
to:
_test2:
movl 8(%esp), %eax
movl 4(%esp), %edx
bswapl %eax
bswapl %edx
ret
instead of:
_test2:
movl 8(%esp), %edx
movl 4(%esp), %eax
bswap %eax
bswap %edx
xchgl %eax, %edx
ret
GCC manages (with -fomit-frame-pointer) the uglier:
_test2:
subl $4, %esp
movl 8(%esp), %eax
movl 12(%esp), %edx
bswap %eax
bswap %edx
xchgl %eax, %edx
addl $4, %esp
ret
llvm-svn: 32001
2006-11-29 09:48:01 +08:00
|
|
|
std::vector<InlineAsm::ConstraintInfo> Constraints = IA->ParseConstraints();
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
std::string AsmStr = IA->getAsmString();
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
// TODO: should remove alternatives from the asmstring: "foo {a|b}" -> "foo a"
|
|
|
|
std::vector<std::string> AsmPieces;
|
|
|
|
SplitString(AsmStr, AsmPieces, "\n"); // ; as separator?
|
2008-07-09 21:20:27 +08:00
|
|
|
|
2006-11-29 09:14:06 +08:00
|
|
|
switch (AsmPieces.size()) {
|
2008-07-09 21:20:27 +08:00
|
|
|
default: return false;
|
2006-11-29 09:14:06 +08:00
|
|
|
case 1:
|
|
|
|
AsmStr = AsmPieces[0];
|
|
|
|
AsmPieces.clear();
|
|
|
|
SplitString(AsmStr, AsmPieces, " \t"); // Split with whitespace.
|
2008-07-09 21:20:27 +08:00
|
|
|
|
Upgrade the ugly darwin 64-bit bswap idiom (bswap %eax / bswap %edx /
xchgl %eax, %edx) to llvm.bswap.i64. This compiles:
long long test2(long long A) {
return _OSSwapInt64(A);
}
to:
_test2:
movl 8(%esp), %eax
movl 4(%esp), %edx
bswapl %eax
bswapl %edx
ret
instead of:
_test2:
movl 8(%esp), %edx
movl 4(%esp), %eax
bswap %eax
bswap %edx
xchgl %eax, %edx
ret
GCC manages (with -fomit-frame-pointer) the uglier:
_test2:
subl $4, %esp
movl 8(%esp), %eax
movl 12(%esp), %edx
bswap %eax
bswap %edx
xchgl %eax, %edx
addl $4, %esp
ret
llvm-svn: 32001
2006-11-29 09:48:01 +08:00
|
|
|
// bswap $0
|
2008-07-09 21:20:27 +08:00
|
|
|
if (AsmPieces.size() == 2 &&
|
2006-11-29 09:14:06 +08:00
|
|
|
AsmPieces[0] == "bswap" && AsmPieces[1] == "$0") {
|
|
|
|
// No need to check constraints, nothing other than the equivalent of
|
|
|
|
// "=r,0" would be valid here.
|
|
|
|
return LowerToBSwap(CI);
|
|
|
|
}
|
|
|
|
break;
|
Upgrade the ugly darwin 64-bit bswap idiom (bswap %eax / bswap %edx /
xchgl %eax, %edx) to llvm.bswap.i64. This compiles:
long long test2(long long A) {
return _OSSwapInt64(A);
}
to:
_test2:
movl 8(%esp), %eax
movl 4(%esp), %edx
bswapl %eax
bswapl %edx
ret
instead of:
_test2:
movl 8(%esp), %edx
movl 4(%esp), %eax
bswap %eax
bswap %edx
xchgl %eax, %edx
ret
GCC manages (with -fomit-frame-pointer) the uglier:
_test2:
subl $4, %esp
movl 8(%esp), %eax
movl 12(%esp), %edx
bswap %eax
bswap %edx
xchgl %eax, %edx
addl $4, %esp
ret
llvm-svn: 32001
2006-11-29 09:48:01 +08:00
|
|
|
case 3:
|
2006-12-31 13:55:36 +08:00
|
|
|
if (CI->getType() == Type::Int64Ty && Constraints.size() >= 2 &&
|
Upgrade the ugly darwin 64-bit bswap idiom (bswap %eax / bswap %edx /
xchgl %eax, %edx) to llvm.bswap.i64. This compiles:
long long test2(long long A) {
return _OSSwapInt64(A);
}
to:
_test2:
movl 8(%esp), %eax
movl 4(%esp), %edx
bswapl %eax
bswapl %edx
ret
instead of:
_test2:
movl 8(%esp), %edx
movl 4(%esp), %eax
bswap %eax
bswap %edx
xchgl %eax, %edx
ret
GCC manages (with -fomit-frame-pointer) the uglier:
_test2:
subl $4, %esp
movl 8(%esp), %eax
movl 12(%esp), %edx
bswap %eax
bswap %edx
xchgl %eax, %edx
addl $4, %esp
ret
llvm-svn: 32001
2006-11-29 09:48:01 +08:00
|
|
|
Constraints[0].Codes.size() == 1 && Constraints[0].Codes[0] == "A" &&
|
|
|
|
Constraints[1].Codes.size() == 1 && Constraints[1].Codes[0] == "0") {
|
|
|
|
// bswap %eax / bswap %edx / xchgl %eax, %edx -> llvm.bswap.i64
|
|
|
|
std::vector<std::string> Words;
|
|
|
|
SplitString(AsmPieces[0], Words, " \t");
|
|
|
|
if (Words.size() == 2 && Words[0] == "bswap" && Words[1] == "%eax") {
|
|
|
|
Words.clear();
|
|
|
|
SplitString(AsmPieces[1], Words, " \t");
|
|
|
|
if (Words.size() == 2 && Words[0] == "bswap" && Words[1] == "%edx") {
|
|
|
|
Words.clear();
|
|
|
|
SplitString(AsmPieces[2], Words, " \t,");
|
|
|
|
if (Words.size() == 3 && Words[0] == "xchgl" && Words[1] == "%eax" &&
|
|
|
|
Words[2] == "%edx") {
|
|
|
|
return LowerToBSwap(CI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2006-11-29 09:14:06 +08:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2008-02-28 07:33:50 +08:00
|
|
|
|
2008-07-09 21:20:48 +08:00
|
|
|
X86DarwinTargetAsmInfo::X86DarwinTargetAsmInfo(const X86TargetMachine &TM):
|
|
|
|
X86TargetAsmInfo(TM) {
|
|
|
|
bool is64Bit = X86TM->getSubtarget<X86Subtarget>().is64Bit();
|
|
|
|
|
|
|
|
AlignmentIsInBytes = false;
|
|
|
|
TextAlignFillValue = 0x90;
|
|
|
|
GlobalPrefix = "_";
|
|
|
|
if (!is64Bit)
|
|
|
|
Data64bitsDirective = 0; // we can't emit a 64-bit unit
|
|
|
|
ZeroDirective = "\t.space\t"; // ".space N" emits N zeros.
|
|
|
|
PrivateGlobalPrefix = "L"; // Marker for constant pool idxs
|
|
|
|
BSSSection = 0; // no BSS section.
|
|
|
|
ZeroFillDirective = "\t.zerofill\t"; // Uses .zerofill
|
2008-07-10 05:54:26 +08:00
|
|
|
if (X86TM->getRelocationModel() != Reloc::Static)
|
|
|
|
ConstantPoolSection = "\t.const_data";
|
|
|
|
else
|
|
|
|
ConstantPoolSection = "\t.const\n";
|
2008-07-09 21:20:48 +08:00
|
|
|
JumpTableDataSection = "\t.const\n";
|
|
|
|
CStringSection = "\t.cstring";
|
2008-07-09 21:28:49 +08:00
|
|
|
CStringSection_ = getUnnamedSection("\t.cstring",
|
|
|
|
SectionFlags::Mergeable | SectionFlags::Strings);
|
2008-07-09 21:20:48 +08:00
|
|
|
FourByteConstantSection = "\t.literal4\n";
|
2008-07-09 21:28:49 +08:00
|
|
|
FourByteConstantSection_ = getUnnamedSection("\t.literal4\n",
|
|
|
|
SectionFlags::Mergeable);
|
2008-07-10 04:47:55 +08:00
|
|
|
EightByteConstantSection = "\t.literal8\n";
|
2008-07-09 21:28:49 +08:00
|
|
|
EightByteConstantSection_ = getUnnamedSection("\t.literal8\n",
|
|
|
|
SectionFlags::Mergeable);
|
2008-07-09 21:20:48 +08:00
|
|
|
// FIXME: Why don't always use this section?
|
2008-07-09 21:28:49 +08:00
|
|
|
if (is64Bit) {
|
2008-07-09 21:20:48 +08:00
|
|
|
SixteenByteConstantSection = "\t.literal16\n";
|
2008-07-09 21:28:49 +08:00
|
|
|
SixteenByteConstantSection_ = getUnnamedSection("\t.literal16\n",
|
|
|
|
SectionFlags::Mergeable);
|
|
|
|
}
|
2008-07-09 21:20:48 +08:00
|
|
|
ReadOnlySection = "\t.const\n";
|
2008-07-09 21:28:49 +08:00
|
|
|
ReadOnlySection_ = getUnnamedSection("\t.const\n", SectionFlags::None);
|
2008-07-09 21:30:02 +08:00
|
|
|
// FIXME: These should be named sections, really.
|
2008-07-09 21:28:49 +08:00
|
|
|
TextCoalSection =
|
|
|
|
getUnnamedSection(".section __TEXT,__textcoal_nt,coalesced,pure_instructions",
|
|
|
|
SectionFlags::Code);
|
|
|
|
ConstDataCoalSection =
|
|
|
|
getUnnamedSection(".section __DATA,__const_coal,coalesced",
|
|
|
|
SectionFlags::None);
|
|
|
|
ConstDataSection = getUnnamedSection(".const_data", SectionFlags::None);
|
|
|
|
DataCoalSection = getUnnamedSection(".section __DATA,__datacoal_nt,coalesced",
|
|
|
|
SectionFlags::Writeable);
|
|
|
|
|
2008-07-09 21:20:48 +08:00
|
|
|
LCOMMDirective = "\t.lcomm\t";
|
|
|
|
SwitchToSectionDirective = "\t.section ";
|
|
|
|
StringConstantPrefix = "\1LC";
|
|
|
|
COMMDirectiveTakesAlignment = false;
|
|
|
|
HasDotTypeDotSizeDirective = false;
|
|
|
|
if (TM.getRelocationModel() == Reloc::Static) {
|
|
|
|
StaticCtorsSection = ".constructor";
|
|
|
|
StaticDtorsSection = ".destructor";
|
|
|
|
} else {
|
|
|
|
StaticCtorsSection = ".mod_init_func";
|
|
|
|
StaticDtorsSection = ".mod_term_func";
|
|
|
|
}
|
|
|
|
if (is64Bit) {
|
|
|
|
PersonalityPrefix = "";
|
|
|
|
PersonalitySuffix = "+4@GOTPCREL";
|
|
|
|
} else {
|
|
|
|
PersonalityPrefix = "L";
|
|
|
|
PersonalitySuffix = "$non_lazy_ptr";
|
|
|
|
}
|
|
|
|
NeedsIndirectEncoding = true;
|
|
|
|
InlineAsmStart = "## InlineAsm Start";
|
|
|
|
InlineAsmEnd = "## InlineAsm End";
|
|
|
|
CommentString = "##";
|
|
|
|
SetDirective = "\t.set";
|
|
|
|
PCSymbol = ".";
|
|
|
|
UsedDirective = "\t.no_dead_strip\t";
|
|
|
|
WeakDefDirective = "\t.weak_definition ";
|
|
|
|
WeakRefDirective = "\t.weak_reference ";
|
|
|
|
HiddenDirective = "\t.private_extern ";
|
|
|
|
ProtectedDirective = "\t.globl\t";
|
|
|
|
|
|
|
|
// In non-PIC modes, emit a special label before jump tables so that the
|
|
|
|
// linker can perform more accurate dead code stripping.
|
|
|
|
if (TM.getRelocationModel() != Reloc::PIC_) {
|
|
|
|
// Emit a local label that is preserved until the linker runs.
|
|
|
|
JumpTableSpecialLabelPrefix = "l";
|
|
|
|
}
|
|
|
|
|
|
|
|
SupportsDebugInformation = true;
|
|
|
|
NeedsSet = true;
|
|
|
|
DwarfAbbrevSection = ".section __DWARF,__debug_abbrev,regular,debug";
|
|
|
|
DwarfInfoSection = ".section __DWARF,__debug_info,regular,debug";
|
|
|
|
DwarfLineSection = ".section __DWARF,__debug_line,regular,debug";
|
|
|
|
DwarfFrameSection = ".section __DWARF,__debug_frame,regular,debug";
|
|
|
|
DwarfPubNamesSection = ".section __DWARF,__debug_pubnames,regular,debug";
|
|
|
|
DwarfPubTypesSection = ".section __DWARF,__debug_pubtypes,regular,debug";
|
|
|
|
DwarfStrSection = ".section __DWARF,__debug_str,regular,debug";
|
|
|
|
DwarfLocSection = ".section __DWARF,__debug_loc,regular,debug";
|
|
|
|
DwarfARangesSection = ".section __DWARF,__debug_aranges,regular,debug";
|
|
|
|
DwarfRangesSection = ".section __DWARF,__debug_ranges,regular,debug";
|
|
|
|
DwarfMacInfoSection = ".section __DWARF,__debug_macinfo,regular,debug";
|
|
|
|
|
|
|
|
// Exceptions handling
|
|
|
|
SupportsExceptionHandling = true;
|
|
|
|
GlobalEHDirective = "\t.globl\t";
|
|
|
|
SupportsWeakOmittedEHFrame = false;
|
|
|
|
AbsoluteEHSectionOffsets = false;
|
|
|
|
DwarfEHFrameSection =
|
|
|
|
".section __TEXT,__eh_frame,coalesced,no_toc+strip_static_syms+live_support";
|
|
|
|
DwarfExceptionSection = ".section __DATA,__gcc_except_tab";
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:08 +08:00
|
|
|
unsigned
|
|
|
|
X86DarwinTargetAsmInfo::PreferredEHDataFormat(DwarfEncoding::Target Reason,
|
|
|
|
bool Global) const {
|
|
|
|
if (Reason == DwarfEncoding::Functions && Global)
|
|
|
|
return (DW_EH_PE_pcrel | DW_EH_PE_indirect | DW_EH_PE_sdata4);
|
|
|
|
else if (Reason == DwarfEncoding::CodeLabels || !Global)
|
|
|
|
return DW_EH_PE_pcrel;
|
|
|
|
else
|
|
|
|
return DW_EH_PE_absptr;
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
const Section*
|
2008-07-09 21:23:57 +08:00
|
|
|
X86DarwinTargetAsmInfo::SelectSectionForGlobal(const GlobalValue *GV) const {
|
|
|
|
SectionKind::Kind Kind = SectionKindForGlobal(GV);
|
2008-07-09 21:27:59 +08:00
|
|
|
bool isWeak = GV->isWeakForLinker();
|
2008-07-10 04:01:42 +08:00
|
|
|
bool isNonStatic = (X86TM->getRelocationModel() != Reloc::Static);
|
2008-07-09 21:23:57 +08:00
|
|
|
|
|
|
|
switch (Kind) {
|
|
|
|
case SectionKind::Text:
|
|
|
|
if (isWeak)
|
2008-07-09 21:28:49 +08:00
|
|
|
return TextCoalSection;
|
2008-07-09 21:23:57 +08:00
|
|
|
else
|
2008-07-09 21:28:49 +08:00
|
|
|
return getTextSection_();
|
2008-07-09 21:23:57 +08:00
|
|
|
case SectionKind::Data:
|
|
|
|
case SectionKind::ThreadData:
|
|
|
|
case SectionKind::BSS:
|
|
|
|
case SectionKind::ThreadBSS:
|
2008-07-09 21:28:49 +08:00
|
|
|
if (cast<GlobalVariable>(GV)->isConstant())
|
|
|
|
return (isWeak ? ConstDataCoalSection : ConstDataSection);
|
2008-07-09 21:23:57 +08:00
|
|
|
else
|
2008-07-09 21:28:49 +08:00
|
|
|
return (isWeak ? DataCoalSection : getDataSection_());
|
|
|
|
case SectionKind::ROData:
|
2008-07-10 04:01:42 +08:00
|
|
|
return (isWeak ? ConstDataCoalSection :
|
|
|
|
(isNonStatic ? ConstDataSection : getReadOnlySection_()));
|
2008-07-09 21:23:57 +08:00
|
|
|
case SectionKind::RODataMergeStr:
|
2008-07-10 03:06:02 +08:00
|
|
|
return (isWeak ?
|
|
|
|
ConstDataCoalSection :
|
|
|
|
MergeableStringSection(cast<GlobalVariable>(GV)));
|
2008-07-09 21:23:57 +08:00
|
|
|
case SectionKind::RODataMergeConst:
|
2008-07-10 03:06:02 +08:00
|
|
|
return (isWeak ?
|
|
|
|
ConstDataCoalSection:
|
|
|
|
MergeableConstSection(cast<GlobalVariable>(GV)));
|
2008-07-09 21:23:57 +08:00
|
|
|
default:
|
|
|
|
assert(0 && "Unsuported section kind for global");
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Do we have any extra special weird cases?
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
const Section*
|
2008-07-09 21:23:57 +08:00
|
|
|
X86DarwinTargetAsmInfo::MergeableStringSection(const GlobalVariable *GV) const {
|
2008-07-09 21:28:49 +08:00
|
|
|
const TargetData *TD = X86TM->getTargetData();
|
|
|
|
Constant *C = cast<GlobalVariable>(GV)->getInitializer();
|
|
|
|
const Type *Type = cast<ConstantArray>(C)->getType()->getElementType();
|
2008-07-09 21:23:57 +08:00
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
unsigned Size = TD->getABITypeSize(Type);
|
2008-07-09 21:23:57 +08:00
|
|
|
if (Size) {
|
|
|
|
const TargetData *TD = X86TM->getTargetData();
|
|
|
|
unsigned Align = TD->getPreferredAlignment(GV);
|
|
|
|
if (Align <= 32)
|
2008-07-09 21:28:49 +08:00
|
|
|
return getCStringSection_();
|
2008-07-09 21:23:57 +08:00
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
return getReadOnlySection_();
|
2008-07-09 21:23:57 +08:00
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
const Section*
|
2008-07-09 21:23:57 +08:00
|
|
|
X86DarwinTargetAsmInfo::MergeableConstSection(const GlobalVariable *GV) const {
|
2008-07-09 21:28:49 +08:00
|
|
|
const TargetData *TD = X86TM->getTargetData();
|
|
|
|
Constant *C = cast<GlobalVariable>(GV)->getInitializer();
|
2008-07-09 21:23:57 +08:00
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
unsigned Size = TD->getABITypeSize(C->getType());
|
2008-07-09 21:23:57 +08:00
|
|
|
if (Size == 4)
|
2008-07-09 21:28:49 +08:00
|
|
|
return FourByteConstantSection_;
|
2008-07-09 21:23:57 +08:00
|
|
|
else if (Size == 8)
|
2008-07-09 21:28:49 +08:00
|
|
|
return EightByteConstantSection_;
|
2008-07-09 21:23:57 +08:00
|
|
|
else if (Size == 16 && X86TM->getSubtarget<X86Subtarget>().is64Bit())
|
2008-07-09 21:28:49 +08:00
|
|
|
return SixteenByteConstantSection_;
|
2008-07-09 21:23:57 +08:00
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
return getReadOnlySection_();
|
2008-07-09 21:23:57 +08:00
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:29 +08:00
|
|
|
std::string
|
|
|
|
X86DarwinTargetAsmInfo::UniqueSectionForGlobal(const GlobalValue* GV,
|
|
|
|
SectionKind::Kind kind) const {
|
2008-07-09 21:23:57 +08:00
|
|
|
assert(0 && "Darwin does not use unique sections");
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:20:48 +08:00
|
|
|
X86ELFTargetAsmInfo::X86ELFTargetAsmInfo(const X86TargetMachine &TM):
|
|
|
|
X86TargetAsmInfo(TM) {
|
|
|
|
bool is64Bit = X86TM->getSubtarget<X86Subtarget>().is64Bit();
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
TextSection_ = getUnnamedSection("\t.text", SectionFlags::Code);
|
|
|
|
DataSection_ = getUnnamedSection("\t.data", SectionFlags::Writeable);
|
|
|
|
BSSSection_ = getUnnamedSection("\t.bss",
|
|
|
|
SectionFlags::Writeable | SectionFlags::BSS);
|
2008-07-09 21:29:27 +08:00
|
|
|
ReadOnlySection_ = getNamedSection("\t.rodata", SectionFlags::None);
|
2008-07-09 21:28:49 +08:00
|
|
|
TLSDataSection_ = getNamedSection("\t.tdata",
|
|
|
|
SectionFlags::Writeable | SectionFlags::TLS);
|
|
|
|
TLSBSSSection_ = getNamedSection("\t.tbss",
|
|
|
|
SectionFlags::Writeable | SectionFlags::TLS | SectionFlags::BSS);
|
|
|
|
|
2008-07-09 21:25:26 +08:00
|
|
|
ReadOnlySection = ".rodata";
|
2008-07-09 21:25:46 +08:00
|
|
|
FourByteConstantSection = "\t.section\t.rodata.cst4,\"aM\",@progbits,4";
|
|
|
|
EightByteConstantSection = "\t.section\t.rodata.cst8,\"aM\",@progbits,8";
|
|
|
|
SixteenByteConstantSection = "\t.section\t.rodata.cst16,\"aM\",@progbits,16";
|
2008-07-09 21:25:26 +08:00
|
|
|
CStringSection = ".rodata.str";
|
2008-07-09 21:20:48 +08:00
|
|
|
PrivateGlobalPrefix = ".L";
|
|
|
|
WeakRefDirective = "\t.weak\t";
|
|
|
|
SetDirective = "\t.set\t";
|
|
|
|
PCSymbol = ".";
|
|
|
|
|
|
|
|
// Set up DWARF directives
|
|
|
|
HasLEB128 = true; // Target asm supports leb128 directives (little-endian)
|
|
|
|
|
|
|
|
// Debug Information
|
|
|
|
AbsoluteDebugSectionOffsets = true;
|
|
|
|
SupportsDebugInformation = true;
|
|
|
|
DwarfAbbrevSection = "\t.section\t.debug_abbrev,\"\",@progbits";
|
|
|
|
DwarfInfoSection = "\t.section\t.debug_info,\"\",@progbits";
|
|
|
|
DwarfLineSection = "\t.section\t.debug_line,\"\",@progbits";
|
|
|
|
DwarfFrameSection = "\t.section\t.debug_frame,\"\",@progbits";
|
|
|
|
DwarfPubNamesSection ="\t.section\t.debug_pubnames,\"\",@progbits";
|
|
|
|
DwarfPubTypesSection ="\t.section\t.debug_pubtypes,\"\",@progbits";
|
|
|
|
DwarfStrSection = "\t.section\t.debug_str,\"\",@progbits";
|
|
|
|
DwarfLocSection = "\t.section\t.debug_loc,\"\",@progbits";
|
|
|
|
DwarfARangesSection = "\t.section\t.debug_aranges,\"\",@progbits";
|
|
|
|
DwarfRangesSection = "\t.section\t.debug_ranges,\"\",@progbits";
|
|
|
|
DwarfMacInfoSection = "\t.section\t.debug_macinfo,\"\",@progbits";
|
|
|
|
|
|
|
|
// Exceptions handling
|
|
|
|
if (!is64Bit)
|
|
|
|
SupportsExceptionHandling = true;
|
|
|
|
AbsoluteEHSectionOffsets = false;
|
|
|
|
DwarfEHFrameSection = "\t.section\t.eh_frame,\"aw\",@progbits";
|
|
|
|
DwarfExceptionSection = "\t.section\t.gcc_except_table,\"a\",@progbits";
|
|
|
|
|
|
|
|
// On Linux we must declare when we can use a non-executable stack.
|
|
|
|
if (X86TM->getSubtarget<X86Subtarget>().isLinux())
|
|
|
|
NonexecutableStackDirective = "\t.section\t.note.GNU-stack,\"\",@progbits";
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:08 +08:00
|
|
|
unsigned
|
|
|
|
X86ELFTargetAsmInfo::PreferredEHDataFormat(DwarfEncoding::Target Reason,
|
|
|
|
bool Global) const {
|
|
|
|
CodeModel::Model CM = X86TM->getCodeModel();
|
|
|
|
bool is64Bit = X86TM->getSubtarget<X86Subtarget>().is64Bit();
|
|
|
|
|
|
|
|
if (X86TM->getRelocationModel() == Reloc::PIC_) {
|
|
|
|
unsigned Format = 0;
|
|
|
|
|
|
|
|
if (!is64Bit)
|
|
|
|
// 32 bit targets always encode pointers as 4 bytes
|
|
|
|
Format = DW_EH_PE_sdata4;
|
|
|
|
else {
|
|
|
|
// 64 bit targets encode pointers in 4 bytes iff:
|
|
|
|
// - code model is small OR
|
|
|
|
// - code model is medium and we're emitting externally visible symbols
|
|
|
|
// or any code symbols
|
|
|
|
if (CM == CodeModel::Small ||
|
|
|
|
(CM == CodeModel::Medium && (Global ||
|
|
|
|
Reason != DwarfEncoding::Data)))
|
|
|
|
Format = DW_EH_PE_sdata4;
|
|
|
|
else
|
|
|
|
Format = DW_EH_PE_sdata8;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Global)
|
|
|
|
Format |= DW_EH_PE_indirect;
|
|
|
|
|
|
|
|
return (Format | DW_EH_PE_pcrel);
|
|
|
|
} else {
|
|
|
|
if (is64Bit &&
|
|
|
|
(CM == CodeModel::Small ||
|
|
|
|
(CM == CodeModel::Medium && Reason != DwarfEncoding::Data)))
|
|
|
|
return DW_EH_PE_udata4;
|
|
|
|
else
|
|
|
|
return DW_EH_PE_absptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
const Section*
|
2008-07-09 21:23:08 +08:00
|
|
|
X86ELFTargetAsmInfo::SelectSectionForGlobal(const GlobalValue *GV) const {
|
2008-07-09 21:28:49 +08:00
|
|
|
SectionKind::Kind Kind = SectionKindForGlobal(GV);
|
2008-07-09 21:23:08 +08:00
|
|
|
|
|
|
|
if (const Function *F = dyn_cast<Function>(GV)) {
|
|
|
|
switch (F->getLinkage()) {
|
|
|
|
default: assert(0 && "Unknown linkage type!");
|
|
|
|
case Function::InternalLinkage:
|
|
|
|
case Function::DLLExportLinkage:
|
|
|
|
case Function::ExternalLinkage:
|
2008-07-09 21:28:49 +08:00
|
|
|
return getTextSection_();
|
2008-07-09 21:23:08 +08:00
|
|
|
case Function::WeakLinkage:
|
|
|
|
case Function::LinkOnceLinkage:
|
2008-07-09 21:28:49 +08:00
|
|
|
std::string Name = UniqueSectionForGlobal(GV, Kind);
|
|
|
|
unsigned Flags = SectionFlagsForGlobal(GV, Name.c_str());
|
|
|
|
return getNamedSection(Name.c_str(), Flags);
|
2008-07-09 21:23:08 +08:00
|
|
|
}
|
|
|
|
} else if (const GlobalVariable *GVar = dyn_cast<GlobalVariable>(GV)) {
|
2008-07-09 21:28:49 +08:00
|
|
|
if (GVar->isWeakForLinker()) {
|
|
|
|
std::string Name = UniqueSectionForGlobal(GVar, Kind);
|
|
|
|
unsigned Flags = SectionFlagsForGlobal(GVar, Name.c_str());
|
|
|
|
return getNamedSection(Name.c_str(), Flags);
|
|
|
|
} else {
|
|
|
|
switch (Kind) {
|
2008-07-09 21:23:08 +08:00
|
|
|
case SectionKind::Data:
|
2008-07-09 21:28:49 +08:00
|
|
|
return getDataSection_();
|
2008-07-09 21:23:08 +08:00
|
|
|
case SectionKind::BSS:
|
|
|
|
// ELF targets usually have BSS sections
|
2008-07-09 21:28:49 +08:00
|
|
|
return getBSSSection_();
|
2008-07-09 21:23:08 +08:00
|
|
|
case SectionKind::ROData:
|
2008-07-09 21:28:49 +08:00
|
|
|
return getReadOnlySection_();
|
2008-07-09 21:23:08 +08:00
|
|
|
case SectionKind::RODataMergeStr:
|
2008-07-09 21:23:37 +08:00
|
|
|
return MergeableStringSection(GVar);
|
2008-07-09 21:23:08 +08:00
|
|
|
case SectionKind::RODataMergeConst:
|
2008-07-09 21:23:37 +08:00
|
|
|
return MergeableConstSection(GVar);
|
2008-07-09 21:23:08 +08:00
|
|
|
case SectionKind::ThreadData:
|
|
|
|
// ELF targets usually support TLS stuff
|
2008-07-09 21:28:49 +08:00
|
|
|
return getTLSDataSection_();
|
2008-07-09 21:23:08 +08:00
|
|
|
case SectionKind::ThreadBSS:
|
2008-07-09 21:28:49 +08:00
|
|
|
return getTLSBSSSection_();
|
2008-07-09 21:23:08 +08:00
|
|
|
default:
|
|
|
|
assert(0 && "Unsuported section kind for global");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
assert(0 && "Unsupported global");
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
const Section*
|
2008-07-09 21:23:37 +08:00
|
|
|
X86ELFTargetAsmInfo::MergeableConstSection(const GlobalVariable *GV) const {
|
2008-07-09 21:28:49 +08:00
|
|
|
const TargetData *TD = X86TM->getTargetData();
|
|
|
|
Constant *C = cast<GlobalVariable>(GV)->getInitializer();
|
|
|
|
const Type *Type = C->getType();
|
2008-07-09 21:23:37 +08:00
|
|
|
|
|
|
|
// FIXME: string here is temporary, until stuff will fully land in.
|
2008-07-09 21:25:46 +08:00
|
|
|
// We cannot use {Four,Eight,Sixteen}ByteConstantSection here, since it's
|
|
|
|
// currently directly used by asmprinter.
|
2008-07-09 21:28:49 +08:00
|
|
|
unsigned Size = TD->getABITypeSize(Type);
|
|
|
|
if (Size == 4 || Size == 8 || Size == 16) {
|
|
|
|
std::string Name = ".rodata.cst" + utostr(Size);
|
2008-07-09 21:23:57 +08:00
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
return getNamedSection(Name.c_str(),
|
|
|
|
SectionFlags::setEntitySize(SectionFlags::Mergeable,
|
|
|
|
Size));
|
|
|
|
}
|
|
|
|
|
|
|
|
return getReadOnlySection_();
|
2008-07-09 21:23:37 +08:00
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
const Section*
|
2008-07-09 21:23:37 +08:00
|
|
|
X86ELFTargetAsmInfo::MergeableStringSection(const GlobalVariable *GV) const {
|
2008-07-09 21:28:49 +08:00
|
|
|
const TargetData *TD = X86TM->getTargetData();
|
|
|
|
Constant *C = cast<GlobalVariable>(GV)->getInitializer();
|
|
|
|
const ConstantArray *CVA = cast<ConstantArray>(C);
|
|
|
|
const Type *Type = CVA->getType()->getElementType();
|
2008-07-09 21:23:37 +08:00
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
unsigned Size = TD->getABITypeSize(Type);
|
|
|
|
if (Size <= 16) {
|
2008-07-09 21:23:37 +08:00
|
|
|
// We also need alignment here
|
|
|
|
const TargetData *TD = X86TM->getTargetData();
|
|
|
|
unsigned Align = TD->getPreferredAlignment(GV);
|
|
|
|
if (Align < Size)
|
|
|
|
Align = Size;
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
std::string Name = getCStringSection() + utostr(Size) + '.' + utostr(Align);
|
|
|
|
unsigned Flags = SectionFlags::setEntitySize(SectionFlags::Mergeable |
|
|
|
|
SectionFlags::Strings,
|
|
|
|
Size);
|
|
|
|
return getNamedSection(Name.c_str(), Flags);
|
2008-07-09 21:23:57 +08:00
|
|
|
}
|
|
|
|
|
2008-07-09 21:28:49 +08:00
|
|
|
return getReadOnlySection_();
|
2008-07-09 21:23:37 +08:00
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:49 +08:00
|
|
|
std::string X86ELFTargetAsmInfo::PrintSectionFlags(unsigned flags) const {
|
|
|
|
std::string Flags = ",\"";
|
|
|
|
|
|
|
|
if (!(flags & SectionFlags::Debug))
|
|
|
|
Flags += 'a';
|
|
|
|
if (flags & SectionFlags::Code)
|
|
|
|
Flags += 'x';
|
|
|
|
if (flags & SectionFlags::Writeable)
|
|
|
|
Flags += 'w';
|
|
|
|
if (flags & SectionFlags::Mergeable)
|
|
|
|
Flags += 'M';
|
|
|
|
if (flags & SectionFlags::Strings)
|
|
|
|
Flags += 'S';
|
|
|
|
if (flags & SectionFlags::TLS)
|
|
|
|
Flags += 'T';
|
|
|
|
|
|
|
|
Flags += "\"";
|
|
|
|
|
|
|
|
// FIXME: There can be exceptions here
|
|
|
|
if (flags & SectionFlags::BSS)
|
|
|
|
Flags += ",@nobits";
|
|
|
|
else
|
|
|
|
Flags += ",@progbits";
|
|
|
|
|
2008-07-09 21:22:17 +08:00
|
|
|
if (unsigned entitySize = SectionFlags::getEntitySize(flags))
|
|
|
|
Flags += "," + utostr(entitySize);
|
2008-07-09 21:21:49 +08:00
|
|
|
|
|
|
|
return Flags;
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:20:48 +08:00
|
|
|
X86COFFTargetAsmInfo::X86COFFTargetAsmInfo(const X86TargetMachine &TM):
|
|
|
|
X86TargetAsmInfo(TM) {
|
|
|
|
GlobalPrefix = "_";
|
|
|
|
LCOMMDirective = "\t.lcomm\t";
|
|
|
|
COMMDirectiveTakesAlignment = false;
|
|
|
|
HasDotTypeDotSizeDirective = false;
|
|
|
|
StaticCtorsSection = "\t.section .ctors,\"aw\"";
|
|
|
|
StaticDtorsSection = "\t.section .dtors,\"aw\"";
|
|
|
|
HiddenDirective = NULL;
|
|
|
|
PrivateGlobalPrefix = "L"; // Prefix for private global symbols
|
|
|
|
WeakRefDirective = "\t.weak\t";
|
|
|
|
SetDirective = "\t.set\t";
|
|
|
|
|
|
|
|
// Set up DWARF directives
|
|
|
|
HasLEB128 = true; // Target asm supports leb128 directives (little-endian)
|
|
|
|
AbsoluteDebugSectionOffsets = true;
|
|
|
|
AbsoluteEHSectionOffsets = false;
|
|
|
|
SupportsDebugInformation = true;
|
|
|
|
DwarfSectionOffsetDirective = "\t.secrel32\t";
|
|
|
|
DwarfAbbrevSection = "\t.section\t.debug_abbrev,\"dr\"";
|
|
|
|
DwarfInfoSection = "\t.section\t.debug_info,\"dr\"";
|
|
|
|
DwarfLineSection = "\t.section\t.debug_line,\"dr\"";
|
|
|
|
DwarfFrameSection = "\t.section\t.debug_frame,\"dr\"";
|
|
|
|
DwarfPubNamesSection ="\t.section\t.debug_pubnames,\"dr\"";
|
|
|
|
DwarfPubTypesSection ="\t.section\t.debug_pubtypes,\"dr\"";
|
|
|
|
DwarfStrSection = "\t.section\t.debug_str,\"dr\"";
|
|
|
|
DwarfLocSection = "\t.section\t.debug_loc,\"dr\"";
|
|
|
|
DwarfARangesSection = "\t.section\t.debug_aranges,\"dr\"";
|
|
|
|
DwarfRangesSection = "\t.section\t.debug_ranges,\"dr\"";
|
|
|
|
DwarfMacInfoSection = "\t.section\t.debug_macinfo,\"dr\"";
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:08 +08:00
|
|
|
unsigned
|
|
|
|
X86COFFTargetAsmInfo::PreferredEHDataFormat(DwarfEncoding::Target Reason,
|
|
|
|
bool Global) const {
|
|
|
|
CodeModel::Model CM = X86TM->getCodeModel();
|
|
|
|
bool is64Bit = X86TM->getSubtarget<X86Subtarget>().is64Bit();
|
2008-02-28 07:33:50 +08:00
|
|
|
|
2008-07-09 21:21:08 +08:00
|
|
|
if (X86TM->getRelocationModel() == Reloc::PIC_) {
|
|
|
|
unsigned Format = 0;
|
|
|
|
|
|
|
|
if (!is64Bit)
|
|
|
|
// 32 bit targets always encode pointers as 4 bytes
|
|
|
|
Format = DW_EH_PE_sdata4;
|
|
|
|
else {
|
|
|
|
// 64 bit targets encode pointers in 4 bytes iff:
|
|
|
|
// - code model is small OR
|
|
|
|
// - code model is medium and we're emitting externally visible symbols
|
|
|
|
// or any code symbols
|
|
|
|
if (CM == CodeModel::Small ||
|
|
|
|
(CM == CodeModel::Medium && (Global ||
|
|
|
|
Reason != DwarfEncoding::Data)))
|
2008-02-28 07:33:50 +08:00
|
|
|
Format = DW_EH_PE_sdata4;
|
|
|
|
else
|
2008-07-09 21:21:08 +08:00
|
|
|
Format = DW_EH_PE_sdata8;
|
2008-02-28 07:33:50 +08:00
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:08 +08:00
|
|
|
if (Global)
|
|
|
|
Format |= DW_EH_PE_indirect;
|
|
|
|
|
|
|
|
return (Format | DW_EH_PE_pcrel);
|
|
|
|
} else {
|
|
|
|
if (is64Bit &&
|
|
|
|
(CM == CodeModel::Small ||
|
|
|
|
(CM == CodeModel::Medium && Reason != DwarfEncoding::Data)))
|
|
|
|
return DW_EH_PE_udata4;
|
|
|
|
else
|
|
|
|
return DW_EH_PE_absptr;
|
2008-02-28 07:33:50 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:29 +08:00
|
|
|
std::string
|
|
|
|
X86COFFTargetAsmInfo::UniqueSectionForGlobal(const GlobalValue* GV,
|
|
|
|
SectionKind::Kind kind) const {
|
|
|
|
switch (kind) {
|
|
|
|
case SectionKind::Text:
|
|
|
|
return ".text$linkonce" + GV->getName();
|
|
|
|
case SectionKind::Data:
|
|
|
|
case SectionKind::BSS:
|
|
|
|
case SectionKind::ThreadData:
|
|
|
|
case SectionKind::ThreadBSS:
|
|
|
|
return ".data$linkonce" + GV->getName();
|
|
|
|
case SectionKind::ROData:
|
|
|
|
case SectionKind::RODataMergeConst:
|
|
|
|
case SectionKind::RODataMergeStr:
|
|
|
|
return ".rdata$linkonce" + GV->getName();
|
2008-07-09 21:19:38 +08:00
|
|
|
default:
|
2008-07-09 21:21:29 +08:00
|
|
|
assert(0 && "Unknown section kind");
|
2008-07-09 21:19:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-09 21:21:49 +08:00
|
|
|
std::string X86COFFTargetAsmInfo::PrintSectionFlags(unsigned flags) const {
|
|
|
|
std::string Flags = ",\"";
|
|
|
|
|
|
|
|
if (flags & SectionFlags::Code)
|
|
|
|
Flags += 'x';
|
|
|
|
if (flags & SectionFlags::Writeable)
|
|
|
|
Flags += 'w';
|
|
|
|
|
|
|
|
Flags += "\"";
|
|
|
|
|
|
|
|
return Flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
X86WinTargetAsmInfo::X86WinTargetAsmInfo(const X86TargetMachine &TM):
|
|
|
|
X86TargetAsmInfo(TM) {
|
|
|
|
GlobalPrefix = "_";
|
|
|
|
CommentString = ";";
|
|
|
|
|
|
|
|
PrivateGlobalPrefix = "$";
|
|
|
|
AlignDirective = "\talign\t";
|
|
|
|
ZeroDirective = "\tdb\t";
|
|
|
|
ZeroDirectiveSuffix = " dup(0)";
|
|
|
|
AsciiDirective = "\tdb\t";
|
|
|
|
AscizDirective = 0;
|
|
|
|
Data8bitsDirective = "\tdb\t";
|
|
|
|
Data16bitsDirective = "\tdw\t";
|
|
|
|
Data32bitsDirective = "\tdd\t";
|
|
|
|
Data64bitsDirective = "\tdq\t";
|
|
|
|
HasDotTypeDotSizeDirective = false;
|
|
|
|
|
|
|
|
TextSection = "_text";
|
|
|
|
DataSection = "_data";
|
|
|
|
JumpTableDataSection = NULL;
|
|
|
|
SwitchToSectionDirective = "";
|
|
|
|
TextSectionStartSuffix = "\tsegment 'CODE'";
|
|
|
|
DataSectionStartSuffix = "\tsegment 'DATA'";
|
|
|
|
SectionEndDirectiveSuffix = "\tends\n";
|
|
|
|
}
|