Pass target triple string in to TargetMachine constructor.

This is not just a matter of passing in the target triple from the module;
currently backends are making decisions based on the build and host
architecture. The goal is to migrate to making these decisions based off of the
triple (in conjunction with the feature string). Thus most clients pass in the
target triple, or the host triple if that is empty.

This has one important change in the way behavior of the JIT and llc.

For the JIT, it was previously selecting the Target based on the host
(naturally), but it was setting the target machine features based on the triple
from the module. Now it is setting the target machine features based on the
triple of the host.

For LLC, -march was previously only used to select the target, the target
machine features were initialized from the module's triple (which may have been
empty). Now the target triple is taken from the module, or the host's triple is
used if that is empty. Then the triple is adjusted to match -march.

The take away is that -march for llc is now used in conjunction with the host
triple to initialize the subtarget. If users want more deterministic behavior
from llc, they should use -mtriple, or set the triple in the input module.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@77946 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Daniel Dunbar
2009-08-03 04:03:51 +00:00
parent 0c794b8725
commit 3c2d4bf97f
11 changed files with 148 additions and 70 deletions

View File

@@ -95,7 +95,7 @@ public:
/// @{ /// @{
Triple() : Data(""), Arch(InvalidArch) {} Triple() : Data(""), Arch(InvalidArch) {}
explicit Triple(const char *Str) : Data(Str), Arch(InvalidArch) {} explicit Triple(const StringRef &Str) : Data(Str), Arch(InvalidArch) {}
explicit Triple(const char *ArchStr, const char *VendorStr, const char *OSStr) explicit Triple(const char *ArchStr, const char *VendorStr, const char *OSStr)
: Data(ArchStr), Arch(InvalidArch) { : Data(ArchStr), Arch(InvalidArch) {
Data += '-'; Data += '-';
@@ -212,6 +212,10 @@ public:
/// getOSTypeName - Get the canonical name for the \arg Kind vendor. /// getOSTypeName - Get the canonical name for the \arg Kind vendor.
static const char *getOSTypeName(OSType Kind); static const char *getOSTypeName(OSType Kind);
/// getArchTypeForLLVMName - The canonical type for the given LLVM
/// architecture name (e.g., "x86").
static ArchType getArchTypeForLLVMName(const StringRef &Str);
/// @} /// @}
}; };

View File

@@ -51,6 +51,7 @@ namespace llvm {
typedef TargetMachine *(*TargetMachineCtorTy)(const Target &, typedef TargetMachine *(*TargetMachineCtorTy)(const Target &,
const Module &, const Module &,
const std::string &,
const std::string &); const std::string &);
typedef FunctionPass *(*AsmPrinterCtorTy)(formatted_raw_ostream &, typedef FunctionPass *(*AsmPrinterCtorTy)(formatted_raw_ostream &,
TargetMachine &, TargetMachine &,
@@ -110,12 +111,21 @@ namespace llvm {
/// hasAsmParser - Check if this target supports .s parsing. /// hasAsmParser - Check if this target supports .s parsing.
bool hasAsmParser() const { return AsmParserCtorFn != 0; } bool hasAsmParser() const { return AsmParserCtorFn != 0; }
/// createTargetMachine - Create a target specific machine implementation. /// createTargetMachine - Create a target specific machine implementation
/// for the module \arg M and \arg Triple.
///
/// \arg M - This argument is used for some machines to access the target
/// data.
/// \arg Triple - This argument is used to determine the target machine
/// feature set; it should always be provided. Generally this should be
/// either the target triple from the module, or the target triple of the
/// host if that does not exist.
TargetMachine *createTargetMachine(const Module &M, TargetMachine *createTargetMachine(const Module &M,
const std::string &Triple,
const std::string &Features) const { const std::string &Features) const {
if (!TargetMachineCtorFn) if (!TargetMachineCtorFn)
return 0; return 0;
return TargetMachineCtorFn(*this, M, Features); return TargetMachineCtorFn(*this, M, Triple, Features);
} }
/// createAsmPrinter - Create a target specific assembly printer pass. /// createAsmPrinter - Create a target specific assembly printer pass.
@@ -325,8 +335,9 @@ namespace llvm {
private: private:
static TargetMachine *Allocator(const Target &T, const Module &M, static TargetMachine *Allocator(const Target &T, const Module &M,
const std::string &TT,
const std::string &FS) { const std::string &FS) {
return new TargetMachineImpl(T, M.getTargetTriple(), FS); return new TargetMachineImpl(T, TT, FS);
} }
}; };
@@ -338,6 +349,7 @@ namespace llvm {
private: private:
static TargetMachine *Allocator(const Target &T, const Module &M, static TargetMachine *Allocator(const Target &T, const Module &M,
const std::string &TT,
const std::string &FS) { const std::string &FS) {
return new TargetMachineImpl(T, M, FS); return new TargetMachineImpl(T, M, FS);
} }

View File

@@ -16,8 +16,10 @@
#include "JIT.h" #include "JIT.h"
#include "llvm/Module.h" #include "llvm/Module.h"
#include "llvm/ModuleProvider.h" #include "llvm/ModuleProvider.h"
#include "llvm/ADT/Triple.h"
#include "llvm/Support/CommandLine.h" #include "llvm/Support/CommandLine.h"
#include "llvm/Support/Streams.h" #include "llvm/Support/raw_ostream.h"
#include "llvm/System/Host.h"
#include "llvm/Target/SubtargetFeature.h" #include "llvm/Target/SubtargetFeature.h"
#include "llvm/Target/TargetMachine.h" #include "llvm/Target/TargetMachine.h"
#include "llvm/Target/TargetRegistry.h" #include "llvm/Target/TargetRegistry.h"
@@ -41,36 +43,29 @@ MAttrs("mattr",
/// selectTarget - Pick a target either via -march or by guessing the native /// selectTarget - Pick a target either via -march or by guessing the native
/// arch. Add any CPU features specified via -mcpu or -mattr. /// arch. Add any CPU features specified via -mcpu or -mattr.
TargetMachine *JIT::selectTarget(ModuleProvider *MP, std::string *ErrorStr) { TargetMachine *JIT::selectTarget(ModuleProvider *MP, std::string *ErrorStr) {
const Target *TheTarget = 0; Triple TheTriple(sys::getHostTriple());
if (MArch.empty()) {
// Adjust the triple to match what the user requested.
if (!MArch.empty())
TheTriple.setArch(Triple::getArchTypeForLLVMName(MArch));
std::string Error; std::string Error;
TheTarget = TargetRegistry::getClosestTargetForJIT(Error); const Target *TheTarget =
TargetRegistry::lookupTarget(TheTriple.getTriple(),
/*FallbackToHost=*/false,
/*RequireJIT=*/false,
Error);
if (TheTarget == 0) { if (TheTarget == 0) {
if (ErrorStr) if (ErrorStr)
*ErrorStr = Error; *ErrorStr = Error;
return 0; return 0;
} }
} else {
for (TargetRegistry::iterator it = TargetRegistry::begin(),
ie = TargetRegistry::end(); it != ie; ++it) {
if (MArch == it->getName()) {
TheTarget = &*it;
break;
}
}
if (TheTarget == 0) {
if (ErrorStr)
*ErrorStr = std::string("invalid target '" + MArch + "'.\n");
return 0;
}
if (!TheTarget->hasJIT()) { if (!TheTarget->hasJIT()) {
cerr << "WARNING: This target JIT is not designed for the host you are" errs() << "WARNING: This target JIT is not designed for the host you are"
<< " running. If bad things happen, please choose a different " << " running. If bad things happen, please choose a different "
<< "-march switch.\n"; << "-march switch.\n";
} }
}
// Package up features to be passed to target/subtarget // Package up features to be passed to target/subtarget
std::string FeaturesStr; std::string FeaturesStr;
@@ -84,7 +79,8 @@ TargetMachine *JIT::selectTarget(ModuleProvider *MP, std::string *ErrorStr) {
// Allocate a target... // Allocate a target...
TargetMachine *Target = TargetMachine *Target =
TheTarget->createTargetMachine(*MP->getModule(), FeaturesStr); TheTarget->createTargetMachine(*MP->getModule(), TheTriple.getTriple(),
FeaturesStr);
assert(Target && "Could not allocate target machine!"); assert(Target && "Could not allocate target machine!");
return Target; return Target;
} }

View File

@@ -71,6 +71,41 @@ const char *Triple::getOSTypeName(OSType Kind) {
return "<invalid>"; return "<invalid>";
} }
Triple::ArchType Triple::getArchTypeForLLVMName(const StringRef &Name) {
if (Name == "alpha")
return alpha;
if (Name == "arm")
return arm;
if (Name == "bfin")
return bfin;
if (Name == "cellspu")
return cellspu;
if (Name == "mips")
return mips;
if (Name == "mipsel")
return mipsel;
if (Name == "msp430")
return msp430;
if (Name == "ppc64")
return ppc64;
if (Name == "ppc")
return ppc;
if (Name == "sparc")
return sparc;
if (Name == "systemz")
return systemz;
if (Name == "thumb")
return thumb;
if (Name == "x86")
return x86;
if (Name == "x86_64")
return x86_64;
if (Name == "xcore")
return xcore;
return UnknownArch;
}
// //
void Triple::Parse() const { void Triple::Parse() const {

View File

@@ -24,6 +24,8 @@
#include "llvm/Intrinsics.h" #include "llvm/Intrinsics.h"
#include "llvm/IntrinsicInst.h" #include "llvm/IntrinsicInst.h"
#include "llvm/InlineAsm.h" #include "llvm/InlineAsm.h"
#include "llvm/ADT/StringExtras.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/Analysis/ConstantsScanner.h" #include "llvm/Analysis/ConstantsScanner.h"
#include "llvm/Analysis/FindUsedTypes.h" #include "llvm/Analysis/FindUsedTypes.h"
#include "llvm/Analysis/LoopInfo.h" #include "llvm/Analysis/LoopInfo.h"
@@ -41,9 +43,7 @@
#include "llvm/Support/InstVisitor.h" #include "llvm/Support/InstVisitor.h"
#include "llvm/Support/Mangler.h" #include "llvm/Support/Mangler.h"
#include "llvm/Support/MathExtras.h" #include "llvm/Support/MathExtras.h"
#include "llvm/ADT/StringExtras.h" #include "llvm/System/Host.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/Support/MathExtras.h"
#include "llvm/Config/config.h" #include "llvm/Config/config.h"
#include <algorithm> #include <algorithm>
#include <sstream> #include <sstream>
@@ -3181,16 +3181,21 @@ std::string CWriter::InterpretASMConstraint(InlineAsm::ConstraintInfo& c) {
// Grab the translation table from TargetAsmInfo if it exists. // Grab the translation table from TargetAsmInfo if it exists.
if (!TAsm) { if (!TAsm) {
std::string Triple = TheModule->getTargetTriple();
if (Triple.empty())
Triple = llvm::sys::getHostTriple();
std::string E; std::string E;
const Target *Match = const Target *Match =
TargetRegistry::lookupTarget(TheModule->getTargetTriple(), TargetRegistry::lookupTarget(Triple,
/*FallbackToHost=*/true, /*FallbackToHost=*/false,
/*RequireJIT=*/false, /*RequireJIT=*/false,
E); E);
if (Match) { if (Match) {
// Per platform Target Machines don't exist, so create it; // Per platform Target Machines don't exist, so create it;
// this must be done only once. // this must be done only once.
const TargetMachine* TM = Match->createTargetMachine(*TheModule, ""); const TargetMachine* TM = Match->createTargetMachine(*TheModule, Triple,
"");
TAsm = TM->getTargetAsmInfo(); TAsm = TM->getTargetAsmInfo();
} }
} }

View File

@@ -1,4 +1,4 @@
; RUN: llvm-as < %s | llc -march=mips | grep wsbw | count 1 ; RUN: llvm-as < %s | llc | grep wsbw | count 1
target datalayout = "e-p:32:32:32-i1:8:8-i8:8:32-i16:16:32-i32:32:32-i64:32:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64" target datalayout = "e-p:32:32:32-i1:8:8-i8:8:32-i16:16:32-i32:32:32-i64:32:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64"
target triple = "psp" target triple = "psp"

View File

@@ -1,4 +1,4 @@
; RUN: llvm-as < %s | llc -march=x86 -mtriple=mingw32 | \ ; RUN: llvm-as < %s | llc -mtriple=i386-unknown-mingw32 | \
; RUN: grep {@12} ; RUN: grep {@12}
; Check that a fastcall function gets correct mangling ; Check that a fastcall function gets correct mangling

View File

@@ -1,5 +1,7 @@
; RUN: llvm-as < %s | llc -march=x86 | not grep rep ; RUN: llvm-as < %s | llc | not grep rep
; RUN: llvm-as < %s | llc -march=x86 | grep memset ; RUN: llvm-as < %s | llc | grep memset
target triple = "i386"
declare void @llvm.memset.i32(i8*, i8, i32, i32) nounwind declare void @llvm.memset.i32(i8*, i8, i32, i32) nounwind

View File

@@ -13,21 +13,20 @@
// //
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
#include "llvm/Bitcode/ReaderWriter.h"
#include "llvm/CodeGen/FileWriters.h"
#include "llvm/CodeGen/LinkAllCodegenComponents.h"
#include "llvm/CodeGen/LinkAllAsmWriterComponents.h"
#include "llvm/CodeGen/ObjectCodeEmitter.h"
#include "llvm/Target/SubtargetFeature.h"
#include "llvm/Target/TargetData.h"
#include "llvm/Target/TargetMachine.h"
#include "llvm/Target/TargetRegistry.h"
#include "llvm/Transforms/Scalar.h"
#include "llvm/LLVMContext.h" #include "llvm/LLVMContext.h"
#include "llvm/Module.h" #include "llvm/Module.h"
#include "llvm/ModuleProvider.h" #include "llvm/ModuleProvider.h"
#include "llvm/PassManager.h" #include "llvm/PassManager.h"
#include "llvm/Pass.h" #include "llvm/Pass.h"
#include "llvm/ADT/Triple.h"
#include "llvm/Analysis/Verifier.h"
#include "llvm/Bitcode/ReaderWriter.h"
#include "llvm/CodeGen/FileWriters.h"
#include "llvm/CodeGen/LinkAllAsmWriterComponents.h"
#include "llvm/CodeGen/LinkAllCodegenComponents.h"
#include "llvm/CodeGen/ObjectCodeEmitter.h"
#include "llvm/Config/config.h"
#include "llvm/LinkAllVMCore.h"
#include "llvm/Support/CommandLine.h" #include "llvm/Support/CommandLine.h"
#include "llvm/Support/FileUtilities.h" #include "llvm/Support/FileUtilities.h"
#include "llvm/Support/FormattedStream.h" #include "llvm/Support/FormattedStream.h"
@@ -35,11 +34,14 @@
#include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/MemoryBuffer.h"
#include "llvm/Support/PluginLoader.h" #include "llvm/Support/PluginLoader.h"
#include "llvm/Support/PrettyStackTrace.h" #include "llvm/Support/PrettyStackTrace.h"
#include "llvm/Analysis/Verifier.h" #include "llvm/System/Host.h"
#include "llvm/System/Signals.h" #include "llvm/System/Signals.h"
#include "llvm/Config/config.h" #include "llvm/Target/SubtargetFeature.h"
#include "llvm/LinkAllVMCore.h" #include "llvm/Target/TargetData.h"
#include "llvm/Target/TargetMachine.h"
#include "llvm/Target/TargetRegistry.h"
#include "llvm/Target/TargetSelect.h" #include "llvm/Target/TargetSelect.h"
#include "llvm/Transforms/Scalar.h"
#include <memory> #include <memory>
using namespace llvm; using namespace llvm;
@@ -234,8 +236,13 @@ int main(int argc, char **argv) {
if (!TargetTriple.empty()) if (!TargetTriple.empty())
mod.setTargetTriple(TargetTriple); mod.setTargetTriple(TargetTriple);
// Allocate target machine. First, check whether the user has Triple TheTriple(mod.getTargetTriple());
// explicitly specified an architecture to compile for. if (TheTriple.getTriple().empty())
TheTriple.setTriple(sys::getHostTriple());
// Allocate target machine. First, check whether the user has explicitly
// specified an architecture to compile for. If so we have to look it up by
// name, because it might be a backend that has no mapping to a target triple.
const Target *TheTarget = 0; const Target *TheTarget = 0;
if (!MArch.empty()) { if (!MArch.empty()) {
for (TargetRegistry::iterator it = TargetRegistry::begin(), for (TargetRegistry::iterator it = TargetRegistry::begin(),
@@ -250,10 +257,16 @@ int main(int argc, char **argv) {
errs() << argv[0] << ": error: invalid target '" << MArch << "'.\n"; errs() << argv[0] << ": error: invalid target '" << MArch << "'.\n";
return 1; return 1;
} }
// Adjust the triple to match (if known), otherwise stick with the
// module/host triple.
Triple::ArchType Type = Triple::getArchTypeForLLVMName(MArch);
if (Type != Triple::UnknownArch)
TheTriple.setArch(Type);
} else { } else {
std::string Err; std::string Err;
TheTarget = TargetRegistry::lookupTarget(mod.getTargetTriple(), TheTarget = TargetRegistry::lookupTarget(TheTriple.getTriple(),
/*FallbackToHost=*/true, /*FallbackToHost=*/false,
/*RequireJIT=*/false, /*RequireJIT=*/false,
Err); Err);
if (TheTarget == 0) { if (TheTarget == 0) {
@@ -275,7 +288,8 @@ int main(int argc, char **argv) {
} }
std::auto_ptr<TargetMachine> std::auto_ptr<TargetMachine>
target(TheTarget->createTargetMachine(mod, FeaturesStr)); target(TheTarget->createTargetMachine(mod, TheTriple.getTriple(),
FeaturesStr));
assert(target.get() && "Could not allocate target machine!"); assert(target.get() && "Could not allocate target machine!");
TargetMachine &Target = *target.get(); TargetMachine &Target = *target.get();

View File

@@ -35,6 +35,7 @@
#include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/MemoryBuffer.h"
#include "llvm/Support/StandardPasses.h" #include "llvm/Support/StandardPasses.h"
#include "llvm/Support/SystemUtils.h" #include "llvm/Support/SystemUtils.h"
#include "llvm/System/Host.h"
#include "llvm/System/Signals.h" #include "llvm/System/Signals.h"
#include "llvm/Target/SubtargetFeature.h" #include "llvm/Target/SubtargetFeature.h"
#include "llvm/Target/TargetOptions.h" #include "llvm/Target/TargetOptions.h"
@@ -326,11 +327,15 @@ bool LTOCodeGenerator::assemble(const std::string& asmPath,
bool LTOCodeGenerator::determineTarget(std::string& errMsg) bool LTOCodeGenerator::determineTarget(std::string& errMsg)
{ {
if ( _target == NULL ) { if ( _target == NULL ) {
std::string Triple = _linker.getModule()->getTargetTriple();
if (Triple.empty())
Triple = sys::getHostTriple();
// create target machine from info for merged modules // create target machine from info for merged modules
Module* mergedModule = _linker.getModule(); Module* mergedModule = _linker.getModule();
const Target *march = const Target *march =
TargetRegistry::lookupTarget(mergedModule->getTargetTriple(), TargetRegistry::lookupTarget(Triple,
/*FallbackToHost=*/true, /*FallbackToHost=*/false,
/*RequireJIT=*/false, /*RequireJIT=*/false,
errMsg); errMsg);
if ( march == NULL ) if ( march == NULL )
@@ -351,9 +356,8 @@ bool LTOCodeGenerator::determineTarget(std::string& errMsg)
} }
// construct LTModule, hand over ownership of module and target // construct LTModule, hand over ownership of module and target
std::string FeatureStr = std::string FeatureStr = getFeatureString(Triple.c_str());
getFeatureString(_linker.getModule()->getTargetTriple().c_str()); _target = march->createTargetMachine(*mergedModule, Triple, FeatureStr);
_target = march->createTargetMachine(*mergedModule, FeatureStr.c_str());
} }
return false; return false;
} }

View File

@@ -24,6 +24,7 @@
#include "llvm/Support/Mangler.h" #include "llvm/Support/Mangler.h"
#include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/MemoryBuffer.h"
#include "llvm/Support/MathExtras.h" #include "llvm/Support/MathExtras.h"
#include "llvm/System/Host.h"
#include "llvm/System/Path.h" #include "llvm/System/Path.h"
#include "llvm/System/Process.h" #include "llvm/System/Process.h"
#include "llvm/Target/SubtargetFeature.h" #include "llvm/Target/SubtargetFeature.h"
@@ -149,17 +150,22 @@ LTOModule* LTOModule::makeLTOModule(MemoryBuffer* buffer,
OwningPtr<Module> m(ParseBitcodeFile(buffer, getGlobalContext(), &errMsg)); OwningPtr<Module> m(ParseBitcodeFile(buffer, getGlobalContext(), &errMsg));
if ( !m ) if ( !m )
return NULL; return NULL;
std::string Triple = m->getTargetTriple();
if (Triple.empty())
Triple = sys::getHostTriple();
// find machine architecture for this module // find machine architecture for this module
const Target* march = TargetRegistry::lookupTarget(m->getTargetTriple(), const Target* march = TargetRegistry::lookupTarget(Triple,
/*FallbackToHost=*/true, /*FallbackToHost=*/false,
/*RequireJIT=*/false, /*RequireJIT=*/false,
errMsg); errMsg);
if ( march == NULL ) if ( march == NULL )
return NULL; return NULL;
// construct LTModule, hand over ownership of module and target // construct LTModule, hand over ownership of module and target
std::string FeatureStr = getFeatureString(m->getTargetTriple().c_str()); std::string FeatureStr = getFeatureString(Triple.c_str());
TargetMachine* target = march->createTargetMachine(*m, FeatureStr); TargetMachine* target = march->createTargetMachine(*m, Triple, FeatureStr);
return new LTOModule(m.take(), target); return new LTOModule(m.take(), target);
} }