You've already forked linux-packaging-mono
acceptance-tests
data
debian
docs
external
Newtonsoft.Json
api-doc-tools
api-snapshot
aspnetwebstack
bdwgc
binary-reference-assemblies
bockbuild
boringssl
cecil
cecil-legacy
corefx
corert
helix-binaries
ikdasm
ikvm
illinker-test-assets
linker
llvm-project
clang
clang-tools-extra
compiler-rt
libcxx
libcxxabi
libunwind
lld
lldb
llvm
bindings
cmake
docs
examples
include
lib
Analysis
AsmParser
BinaryFormat
Bitcode
CodeGen
DebugInfo
Demangle
ExecutionEngine
FuzzMutate
Fuzzer
IR
AsmWriter.cpp.REMOVED.git-id
AttributeImpl.h
Attributes.cpp
AttributesCompatFunc.td
AutoUpgrade.cpp.REMOVED.git-id
BasicBlock.cpp
CMakeLists.txt
Comdat.cpp
ConstantFold.cpp
ConstantFold.h
ConstantRange.cpp
Constants.cpp.REMOVED.git-id
ConstantsContext.h
Core.cpp.REMOVED.git-id
DIBuilder.cpp
DataLayout.cpp
DebugInfo.cpp
DebugInfoMetadata.cpp
DebugLoc.cpp
DiagnosticHandler.cpp
DiagnosticInfo.cpp
DiagnosticPrinter.cpp
Dominators.cpp
Function.cpp
GVMaterializer.cpp
Globals.cpp
IRBuilder.cpp
IRPrintingPasses.cpp
InlineAsm.cpp
Instruction.cpp
Instructions.cpp.REMOVED.git-id
IntrinsicInst.cpp
LLVMBuild.txt
LLVMContext.cpp
LLVMContextImpl.cpp
LLVMContextImpl.h
LegacyPassManager.cpp
MDBuilder.cpp
Mangler.cpp
Metadata.cpp
MetadataImpl.h
Module.cpp
ModuleSummaryIndex.cpp
Operator.cpp
OptBisect.cpp
Pass.cpp
PassManager.cpp
PassRegistry.cpp
ProfileSummary.cpp
SafepointIRVerifier.cpp
Statepoint.cpp
SymbolTableListTraitsImpl.h
Type.cpp
TypeFinder.cpp
Use.cpp
User.cpp
Value.cpp
ValueSymbolTable.cpp
ValueTypes.cpp
Verifier.cpp.REMOVED.git-id
IRReader
LTO
LineEditor
Linker
MC
Object
ObjectYAML
Option
Passes
ProfileData
Support
TableGen
Target
Testing
ToolDrivers
Transforms
WindowsManifest
XRay
CMakeLists.txt
LLVMBuild.txt
projects
resources
runtimes
scripts
test
tools
unittests
utils
.arcconfig
.clang-format
.clang-tidy
.gitattributes
.gitignore
CMakeLists.txt
CODE_OWNERS.TXT
CREDITS.TXT
LICENSE.TXT
LLVMBuild.txt
README.txt
RELEASE_TESTERS.TXT
configure
llvm.spec.in
openmp
polly
nuget-buildtasks
nunit-lite
roslyn-binaries
rx
xunit-binaries
how-to-bump-roslyn-binaries.md
ikvm-native
llvm
m4
man
mcs
mk
mono
msvc
netcore
po
runtime
samples
scripts
support
tools
COPYING.LIB
LICENSE
Makefile.am
Makefile.in
NEWS
README.md
acinclude.m4
aclocal.m4
autogen.sh
code_of_conduct.md
compile
config.guess
config.h.in
config.rpath
config.sub
configure.REMOVED.git-id
configure.ac.REMOVED.git-id
depcomp
install-sh
ltmain.sh.REMOVED.git-id
missing
mkinstalldirs
mono-uninstalled.pc.in
test-driver
winconfig.h
207 lines
6.8 KiB
C++
207 lines
6.8 KiB
C++
![]() |
//===-- Mangler.cpp - Self-contained c/asm llvm name mangler --------------===//
|
||
|
//
|
||
|
// The LLVM Compiler Infrastructure
|
||
|
//
|
||
|
// This file is distributed under the University of Illinois Open Source
|
||
|
// License. See LICENSE.TXT for details.
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
//
|
||
|
// Unified name mangler for assembly backends.
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
|
||
|
#include "llvm/IR/Mangler.h"
|
||
|
#include "llvm/ADT/SmallString.h"
|
||
|
#include "llvm/ADT/Triple.h"
|
||
|
#include "llvm/ADT/Twine.h"
|
||
|
#include "llvm/IR/DataLayout.h"
|
||
|
#include "llvm/IR/DerivedTypes.h"
|
||
|
#include "llvm/IR/Function.h"
|
||
|
#include "llvm/IR/Module.h"
|
||
|
#include "llvm/Support/raw_ostream.h"
|
||
|
using namespace llvm;
|
||
|
|
||
|
namespace {
|
||
|
enum ManglerPrefixTy {
|
||
|
Default, ///< Emit default string before each symbol.
|
||
|
Private, ///< Emit "private" prefix before each symbol.
|
||
|
LinkerPrivate ///< Emit "linker private" prefix before each symbol.
|
||
|
};
|
||
|
}
|
||
|
|
||
|
static void getNameWithPrefixImpl(raw_ostream &OS, const Twine &GVName,
|
||
|
ManglerPrefixTy PrefixTy,
|
||
|
const DataLayout &DL, char Prefix) {
|
||
|
SmallString<256> TmpData;
|
||
|
StringRef Name = GVName.toStringRef(TmpData);
|
||
|
assert(!Name.empty() && "getNameWithPrefix requires non-empty name");
|
||
|
|
||
|
// No need to do anything special if the global has the special "do not
|
||
|
// mangle" flag in the name.
|
||
|
if (Name[0] == '\1') {
|
||
|
OS << Name.substr(1);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (PrefixTy == Private)
|
||
|
OS << DL.getPrivateGlobalPrefix();
|
||
|
else if (PrefixTy == LinkerPrivate)
|
||
|
OS << DL.getLinkerPrivateGlobalPrefix();
|
||
|
|
||
|
if (Prefix != '\0')
|
||
|
OS << Prefix;
|
||
|
|
||
|
// If this is a simple string that doesn't need escaping, just append it.
|
||
|
OS << Name;
|
||
|
}
|
||
|
|
||
|
static void getNameWithPrefixImpl(raw_ostream &OS, const Twine &GVName,
|
||
|
const DataLayout &DL,
|
||
|
ManglerPrefixTy PrefixTy) {
|
||
|
char Prefix = DL.getGlobalPrefix();
|
||
|
return getNameWithPrefixImpl(OS, GVName, PrefixTy, DL, Prefix);
|
||
|
}
|
||
|
|
||
|
void Mangler::getNameWithPrefix(raw_ostream &OS, const Twine &GVName,
|
||
|
const DataLayout &DL) {
|
||
|
return getNameWithPrefixImpl(OS, GVName, DL, Default);
|
||
|
}
|
||
|
|
||
|
void Mangler::getNameWithPrefix(SmallVectorImpl<char> &OutName,
|
||
|
const Twine &GVName, const DataLayout &DL) {
|
||
|
raw_svector_ostream OS(OutName);
|
||
|
char Prefix = DL.getGlobalPrefix();
|
||
|
return getNameWithPrefixImpl(OS, GVName, Default, DL, Prefix);
|
||
|
}
|
||
|
|
||
|
static bool hasByteCountSuffix(CallingConv::ID CC) {
|
||
|
switch (CC) {
|
||
|
case CallingConv::X86_FastCall:
|
||
|
case CallingConv::X86_StdCall:
|
||
|
case CallingConv::X86_VectorCall:
|
||
|
return true;
|
||
|
default:
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Microsoft fastcall and stdcall functions require a suffix on their name
|
||
|
/// indicating the number of words of arguments they take.
|
||
|
static void addByteCountSuffix(raw_ostream &OS, const Function *F,
|
||
|
const DataLayout &DL) {
|
||
|
// Calculate arguments size total.
|
||
|
unsigned ArgWords = 0;
|
||
|
for (Function::const_arg_iterator AI = F->arg_begin(), AE = F->arg_end();
|
||
|
AI != AE; ++AI) {
|
||
|
Type *Ty = AI->getType();
|
||
|
// 'Dereference' type in case of byval or inalloca parameter attribute.
|
||
|
if (AI->hasByValOrInAllocaAttr())
|
||
|
Ty = cast<PointerType>(Ty)->getElementType();
|
||
|
// Size should be aligned to pointer size.
|
||
|
unsigned PtrSize = DL.getPointerSize();
|
||
|
ArgWords += alignTo(DL.getTypeAllocSize(Ty), PtrSize);
|
||
|
}
|
||
|
|
||
|
OS << '@' << ArgWords;
|
||
|
}
|
||
|
|
||
|
void Mangler::getNameWithPrefix(raw_ostream &OS, const GlobalValue *GV,
|
||
|
bool CannotUsePrivateLabel) const {
|
||
|
ManglerPrefixTy PrefixTy = Default;
|
||
|
if (GV->hasPrivateLinkage()) {
|
||
|
if (CannotUsePrivateLabel)
|
||
|
PrefixTy = LinkerPrivate;
|
||
|
else
|
||
|
PrefixTy = Private;
|
||
|
}
|
||
|
|
||
|
const DataLayout &DL = GV->getParent()->getDataLayout();
|
||
|
if (!GV->hasName()) {
|
||
|
// Get the ID for the global, assigning a new one if we haven't got one
|
||
|
// already.
|
||
|
unsigned &ID = AnonGlobalIDs[GV];
|
||
|
if (ID == 0)
|
||
|
ID = AnonGlobalIDs.size();
|
||
|
|
||
|
// Must mangle the global into a unique ID.
|
||
|
getNameWithPrefixImpl(OS, "__unnamed_" + Twine(ID), DL, PrefixTy);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
StringRef Name = GV->getName();
|
||
|
char Prefix = DL.getGlobalPrefix();
|
||
|
|
||
|
// Mangle functions with Microsoft calling conventions specially. Only do
|
||
|
// this mangling for x86_64 vectorcall and 32-bit x86.
|
||
|
const Function *MSFunc = dyn_cast<Function>(GV);
|
||
|
if (Name.startswith("\01"))
|
||
|
MSFunc = nullptr; // Don't mangle when \01 is present.
|
||
|
CallingConv::ID CC =
|
||
|
MSFunc ? MSFunc->getCallingConv() : (unsigned)CallingConv::C;
|
||
|
if (!DL.hasMicrosoftFastStdCallMangling() &&
|
||
|
CC != CallingConv::X86_VectorCall)
|
||
|
MSFunc = nullptr;
|
||
|
if (MSFunc) {
|
||
|
if (CC == CallingConv::X86_FastCall)
|
||
|
Prefix = '@'; // fastcall functions have an @ prefix instead of _.
|
||
|
else if (CC == CallingConv::X86_VectorCall)
|
||
|
Prefix = '\0'; // vectorcall functions have no prefix.
|
||
|
}
|
||
|
|
||
|
getNameWithPrefixImpl(OS, Name, PrefixTy, DL, Prefix);
|
||
|
|
||
|
if (!MSFunc)
|
||
|
return;
|
||
|
|
||
|
// If we are supposed to add a microsoft-style suffix for stdcall, fastcall,
|
||
|
// or vectorcall, add it. These functions have a suffix of @N where N is the
|
||
|
// cumulative byte size of all of the parameters to the function in decimal.
|
||
|
if (CC == CallingConv::X86_VectorCall)
|
||
|
OS << '@'; // vectorcall functions use a double @ suffix.
|
||
|
FunctionType *FT = MSFunc->getFunctionType();
|
||
|
if (hasByteCountSuffix(CC) &&
|
||
|
// "Pure" variadic functions do not receive @0 suffix.
|
||
|
(!FT->isVarArg() || FT->getNumParams() == 0 ||
|
||
|
(FT->getNumParams() == 1 && MSFunc->hasStructRetAttr())))
|
||
|
addByteCountSuffix(OS, MSFunc, DL);
|
||
|
}
|
||
|
|
||
|
void Mangler::getNameWithPrefix(SmallVectorImpl<char> &OutName,
|
||
|
const GlobalValue *GV,
|
||
|
bool CannotUsePrivateLabel) const {
|
||
|
raw_svector_ostream OS(OutName);
|
||
|
getNameWithPrefix(OS, GV, CannotUsePrivateLabel);
|
||
|
}
|
||
|
|
||
|
void llvm::emitLinkerFlagsForGlobalCOFF(raw_ostream &OS, const GlobalValue *GV,
|
||
|
const Triple &TT, Mangler &Mangler) {
|
||
|
if (!GV->hasDLLExportStorageClass() || GV->isDeclaration())
|
||
|
return;
|
||
|
|
||
|
if (TT.isKnownWindowsMSVCEnvironment())
|
||
|
OS << " /EXPORT:";
|
||
|
else
|
||
|
OS << " -export:";
|
||
|
|
||
|
if (TT.isWindowsGNUEnvironment() || TT.isWindowsCygwinEnvironment()) {
|
||
|
std::string Flag;
|
||
|
raw_string_ostream FlagOS(Flag);
|
||
|
Mangler.getNameWithPrefix(FlagOS, GV, false);
|
||
|
FlagOS.flush();
|
||
|
if (Flag[0] == GV->getParent()->getDataLayout().getGlobalPrefix())
|
||
|
OS << Flag.substr(1);
|
||
|
else
|
||
|
OS << Flag;
|
||
|
} else {
|
||
|
Mangler.getNameWithPrefix(OS, GV, false);
|
||
|
}
|
||
|
|
||
|
if (!GV->getValueType()->isFunctionTy()) {
|
||
|
if (TT.isKnownWindowsMSVCEnvironment())
|
||
|
OS << ",DATA";
|
||
|
else
|
||
|
OS << ",data";
|
||
|
}
|
||
|
}
|