953 lines
32 KiB
C
953 lines
32 KiB
C
|
//===--- OSTargets.h - Declare OS target feature support --------*- C++ -*-===//
|
||
|
//
|
||
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
||
|
// See https://llvm.org/LICENSE.txt for license information.
|
||
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
//
|
||
|
// This file declares OS specific TargetInfo types.
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
|
||
|
#ifndef LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H
|
||
|
#define LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H
|
||
|
|
||
|
#include "Targets.h"
|
||
|
#include "llvm/MC/MCSectionMachO.h"
|
||
|
|
||
|
namespace clang {
|
||
|
namespace targets {
|
||
|
|
||
|
template <typename TgtInfo>
|
||
|
class LLVM_LIBRARY_VISIBILITY OSTargetInfo : public TgtInfo {
|
||
|
protected:
|
||
|
virtual void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const = 0;
|
||
|
|
||
|
public:
|
||
|
OSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: TgtInfo(Triple, Opts) {}
|
||
|
|
||
|
void getTargetDefines(const LangOptions &Opts,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
TgtInfo::getTargetDefines(Opts, Builder);
|
||
|
getOSDefines(Opts, TgtInfo::getTriple(), Builder);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// CloudABI Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY CloudABITargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
Builder.defineMacro("__CloudABI__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
|
||
|
// CloudABI uses ISO/IEC 10646:2012 for wchar_t, char16_t and char32_t.
|
||
|
Builder.defineMacro("__STDC_ISO_10646__", "201206L");
|
||
|
Builder.defineMacro("__STDC_UTF_16__");
|
||
|
Builder.defineMacro("__STDC_UTF_32__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
CloudABITargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {}
|
||
|
};
|
||
|
|
||
|
// Ananas target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY AnanasTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// Ananas defines
|
||
|
Builder.defineMacro("__Ananas__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
AnanasTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {}
|
||
|
};
|
||
|
|
||
|
void getDarwinDefines(MacroBuilder &Builder, const LangOptions &Opts,
|
||
|
const llvm::Triple &Triple, StringRef &PlatformName,
|
||
|
VersionTuple &PlatformMinVersion);
|
||
|
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY DarwinTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
getDarwinDefines(Builder, Opts, Triple, this->PlatformName,
|
||
|
this->PlatformMinVersion);
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
DarwinTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
// By default, no TLS, and we list permitted architecture/OS
|
||
|
// combinations.
|
||
|
this->TLSSupported = false;
|
||
|
|
||
|
if (Triple.isMacOSX())
|
||
|
this->TLSSupported = !Triple.isMacOSXVersionLT(10, 7);
|
||
|
else if (Triple.isiOS()) {
|
||
|
// 64-bit iOS supported it from 8 onwards, 32-bit device from 9 onwards,
|
||
|
// 32-bit simulator from 10 onwards.
|
||
|
if (Triple.isArch64Bit())
|
||
|
this->TLSSupported = !Triple.isOSVersionLT(8);
|
||
|
else if (Triple.isArch32Bit()) {
|
||
|
if (!Triple.isSimulatorEnvironment())
|
||
|
this->TLSSupported = !Triple.isOSVersionLT(9);
|
||
|
else
|
||
|
this->TLSSupported = !Triple.isOSVersionLT(10);
|
||
|
}
|
||
|
} else if (Triple.isWatchOS()) {
|
||
|
if (!Triple.isSimulatorEnvironment())
|
||
|
this->TLSSupported = !Triple.isOSVersionLT(2);
|
||
|
else
|
||
|
this->TLSSupported = !Triple.isOSVersionLT(3);
|
||
|
}
|
||
|
|
||
|
this->MCountName = "\01mcount";
|
||
|
}
|
||
|
|
||
|
std::string isValidSectionSpecifier(StringRef SR) const override {
|
||
|
// Let MCSectionMachO validate this.
|
||
|
StringRef Segment, Section;
|
||
|
unsigned TAA, StubSize;
|
||
|
bool HasTAA;
|
||
|
return llvm::MCSectionMachO::ParseSectionSpecifier(SR, Segment, Section,
|
||
|
TAA, HasTAA, StubSize);
|
||
|
}
|
||
|
|
||
|
const char *getStaticInitSectionSpecifier() const override {
|
||
|
// FIXME: We should return 0 when building kexts.
|
||
|
return "__TEXT,__StaticInit,regular,pure_instructions";
|
||
|
}
|
||
|
|
||
|
/// Darwin does not support protected visibility. Darwin's "default"
|
||
|
/// is very similar to ELF's "protected"; Darwin requires a "weak"
|
||
|
/// attribute on declarations that can be dynamically replaced.
|
||
|
bool hasProtectedVisibility() const override { return false; }
|
||
|
|
||
|
unsigned getExnObjectAlignment() const override {
|
||
|
// Older versions of libc++abi guarantee an alignment of only 8-bytes for
|
||
|
// exception objects because of a bug in __cxa_exception that was
|
||
|
// eventually fixed in r319123.
|
||
|
llvm::VersionTuple MinVersion;
|
||
|
const llvm::Triple &T = this->getTriple();
|
||
|
|
||
|
// Compute the earliest OS versions that have the fix to libc++abi.
|
||
|
switch (T.getOS()) {
|
||
|
case llvm::Triple::Darwin:
|
||
|
case llvm::Triple::MacOSX: // Earliest supporting version is 10.14.
|
||
|
MinVersion = llvm::VersionTuple(10U, 14U);
|
||
|
break;
|
||
|
case llvm::Triple::IOS:
|
||
|
case llvm::Triple::TvOS: // Earliest supporting version is 12.0.0.
|
||
|
MinVersion = llvm::VersionTuple(12U);
|
||
|
break;
|
||
|
case llvm::Triple::WatchOS: // Earliest supporting version is 5.0.0.
|
||
|
MinVersion = llvm::VersionTuple(5U);
|
||
|
break;
|
||
|
default:
|
||
|
// Conservatively return 8 bytes if OS is unknown.
|
||
|
return 64;
|
||
|
}
|
||
|
|
||
|
unsigned Major, Minor, Micro;
|
||
|
T.getOSVersion(Major, Minor, Micro);
|
||
|
if (llvm::VersionTuple(Major, Minor, Micro) < MinVersion)
|
||
|
return 64;
|
||
|
return OSTargetInfo<Target>::getExnObjectAlignment();
|
||
|
}
|
||
|
|
||
|
TargetInfo::IntType getLeastIntTypeByWidth(unsigned BitWidth,
|
||
|
bool IsSigned) const final {
|
||
|
// Darwin uses `long long` for `int_least64_t` and `int_fast64_t`.
|
||
|
return BitWidth == 64
|
||
|
? (IsSigned ? TargetInfo::SignedLongLong
|
||
|
: TargetInfo::UnsignedLongLong)
|
||
|
: TargetInfo::getLeastIntTypeByWidth(BitWidth, IsSigned);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// DragonFlyBSD Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY DragonFlyBSDTargetInfo
|
||
|
: public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// DragonFly defines; list based off of gcc output
|
||
|
Builder.defineMacro("__DragonFly__");
|
||
|
Builder.defineMacro("__DragonFly_cc_version", "100001");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
Builder.defineMacro("__KPRINTF_ATTRIBUTE__");
|
||
|
Builder.defineMacro("__tune_i386__");
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
DragonFlyBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
switch (Triple.getArch()) {
|
||
|
default:
|
||
|
case llvm::Triple::x86:
|
||
|
case llvm::Triple::x86_64:
|
||
|
this->MCountName = ".mcount";
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
#ifndef FREEBSD_CC_VERSION
|
||
|
#define FREEBSD_CC_VERSION 0U
|
||
|
#endif
|
||
|
|
||
|
// FreeBSD Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY FreeBSDTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// FreeBSD defines; list based off of gcc output
|
||
|
|
||
|
unsigned Release = Triple.getOSMajorVersion();
|
||
|
if (Release == 0U)
|
||
|
Release = 8U;
|
||
|
unsigned CCVersion = FREEBSD_CC_VERSION;
|
||
|
if (CCVersion == 0U)
|
||
|
CCVersion = Release * 100000U + 1U;
|
||
|
|
||
|
Builder.defineMacro("__FreeBSD__", Twine(Release));
|
||
|
Builder.defineMacro("__FreeBSD_cc_version", Twine(CCVersion));
|
||
|
Builder.defineMacro("__KPRINTF_ATTRIBUTE__");
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
|
||
|
// On FreeBSD, wchar_t contains the number of the code point as
|
||
|
// used by the character set of the locale. These character sets are
|
||
|
// not necessarily a superset of ASCII.
|
||
|
//
|
||
|
// FIXME: This is wrong; the macro refers to the numerical values
|
||
|
// of wchar_t *literals*, which are not locale-dependent. However,
|
||
|
// FreeBSD systems apparently depend on us getting this wrong, and
|
||
|
// setting this to 1 is conforming even if all the basic source
|
||
|
// character literals have the same encoding as char and wchar_t.
|
||
|
Builder.defineMacro("__STDC_MB_MIGHT_NEQ_WC__", "1");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
FreeBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
switch (Triple.getArch()) {
|
||
|
default:
|
||
|
case llvm::Triple::x86:
|
||
|
case llvm::Triple::x86_64:
|
||
|
this->MCountName = ".mcount";
|
||
|
break;
|
||
|
case llvm::Triple::mips:
|
||
|
case llvm::Triple::mipsel:
|
||
|
case llvm::Triple::ppc:
|
||
|
case llvm::Triple::ppcle:
|
||
|
case llvm::Triple::ppc64:
|
||
|
case llvm::Triple::ppc64le:
|
||
|
this->MCountName = "_mcount";
|
||
|
break;
|
||
|
case llvm::Triple::arm:
|
||
|
this->MCountName = "__mcount";
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// GNU/kFreeBSD Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY KFreeBSDTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// GNU/kFreeBSD defines; list based off of gcc output
|
||
|
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("__FreeBSD_kernel__");
|
||
|
Builder.defineMacro("__GLIBC__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
if (Opts.CPlusPlus)
|
||
|
Builder.defineMacro("_GNU_SOURCE");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
KFreeBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {}
|
||
|
};
|
||
|
|
||
|
// Haiku Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY HaikuTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// Haiku defines; list based off of gcc output
|
||
|
Builder.defineMacro("__HAIKU__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
if (this->HasFloat128)
|
||
|
Builder.defineMacro("__FLOAT128__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
HaikuTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->SizeType = TargetInfo::UnsignedLong;
|
||
|
this->IntPtrType = TargetInfo::SignedLong;
|
||
|
this->PtrDiffType = TargetInfo::SignedLong;
|
||
|
this->ProcessIDType = TargetInfo::SignedLong;
|
||
|
this->TLSSupported = false;
|
||
|
switch (Triple.getArch()) {
|
||
|
default:
|
||
|
break;
|
||
|
case llvm::Triple::x86:
|
||
|
case llvm::Triple::x86_64:
|
||
|
this->HasFloat128 = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// Hurd target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY HurdTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// Hurd defines; list based off of gcc output.
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("__GNU__");
|
||
|
Builder.defineMacro("__gnu_hurd__");
|
||
|
Builder.defineMacro("__MACH__");
|
||
|
Builder.defineMacro("__GLIBC__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
if (Opts.CPlusPlus)
|
||
|
Builder.defineMacro("_GNU_SOURCE");
|
||
|
}
|
||
|
public:
|
||
|
HurdTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {}
|
||
|
};
|
||
|
|
||
|
// Minix Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY MinixTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// Minix defines
|
||
|
|
||
|
Builder.defineMacro("__minix", "3");
|
||
|
Builder.defineMacro("_EM_WSIZE", "4");
|
||
|
Builder.defineMacro("_EM_PSIZE", "4");
|
||
|
Builder.defineMacro("_EM_SSIZE", "2");
|
||
|
Builder.defineMacro("_EM_LSIZE", "4");
|
||
|
Builder.defineMacro("_EM_FSIZE", "4");
|
||
|
Builder.defineMacro("_EM_DSIZE", "8");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
MinixTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {}
|
||
|
};
|
||
|
|
||
|
// Linux target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY LinuxTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// Linux defines; list based off of gcc output
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
DefineStd(Builder, "linux", Opts);
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
if (Triple.isAndroid()) {
|
||
|
Builder.defineMacro("__ANDROID__", "1");
|
||
|
unsigned Maj, Min, Rev;
|
||
|
Triple.getEnvironmentVersion(Maj, Min, Rev);
|
||
|
this->PlatformName = "android";
|
||
|
this->PlatformMinVersion = VersionTuple(Maj, Min, Rev);
|
||
|
if (Maj) {
|
||
|
Builder.defineMacro("__ANDROID_MIN_SDK_VERSION__", Twine(Maj));
|
||
|
// This historical but ambiguous name for the minSdkVersion macro. Keep
|
||
|
// defined for compatibility.
|
||
|
Builder.defineMacro("__ANDROID_API__", "__ANDROID_MIN_SDK_VERSION__");
|
||
|
}
|
||
|
} else {
|
||
|
Builder.defineMacro("__gnu_linux__");
|
||
|
}
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
if (Opts.CPlusPlus)
|
||
|
Builder.defineMacro("_GNU_SOURCE");
|
||
|
if (this->HasFloat128)
|
||
|
Builder.defineMacro("__FLOAT128__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
LinuxTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->WIntType = TargetInfo::UnsignedInt;
|
||
|
|
||
|
switch (Triple.getArch()) {
|
||
|
default:
|
||
|
break;
|
||
|
case llvm::Triple::mips:
|
||
|
case llvm::Triple::mipsel:
|
||
|
case llvm::Triple::mips64:
|
||
|
case llvm::Triple::mips64el:
|
||
|
case llvm::Triple::ppc:
|
||
|
case llvm::Triple::ppcle:
|
||
|
case llvm::Triple::ppc64:
|
||
|
case llvm::Triple::ppc64le:
|
||
|
this->MCountName = "_mcount";
|
||
|
break;
|
||
|
case llvm::Triple::x86:
|
||
|
case llvm::Triple::x86_64:
|
||
|
this->HasFloat128 = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const char *getStaticInitSectionSpecifier() const override {
|
||
|
return ".text.startup";
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// NetBSD Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY NetBSDTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// NetBSD defines; list based off of gcc output
|
||
|
Builder.defineMacro("__NetBSD__");
|
||
|
Builder.defineMacro("__unix__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
NetBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->MCountName = "__mcount";
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// OpenBSD Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY OpenBSDTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// OpenBSD defines; list based off of gcc output
|
||
|
|
||
|
Builder.defineMacro("__OpenBSD__");
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
if (this->HasFloat128)
|
||
|
Builder.defineMacro("__FLOAT128__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
OpenBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->WCharType = this->WIntType = this->SignedInt;
|
||
|
this->IntMaxType = TargetInfo::SignedLongLong;
|
||
|
this->Int64Type = TargetInfo::SignedLongLong;
|
||
|
switch (Triple.getArch()) {
|
||
|
case llvm::Triple::x86:
|
||
|
case llvm::Triple::x86_64:
|
||
|
this->HasFloat128 = true;
|
||
|
LLVM_FALLTHROUGH;
|
||
|
default:
|
||
|
this->MCountName = "__mcount";
|
||
|
break;
|
||
|
case llvm::Triple::mips64:
|
||
|
case llvm::Triple::mips64el:
|
||
|
case llvm::Triple::ppc:
|
||
|
case llvm::Triple::ppc64:
|
||
|
case llvm::Triple::ppc64le:
|
||
|
case llvm::Triple::sparcv9:
|
||
|
this->MCountName = "_mcount";
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// PSP Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY PSPTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// PSP defines; list based on the output of the pspdev gcc toolchain.
|
||
|
Builder.defineMacro("PSP");
|
||
|
Builder.defineMacro("_PSP");
|
||
|
Builder.defineMacro("__psp__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
PSPTargetInfo(const llvm::Triple &Triple) : OSTargetInfo<Target>(Triple) {}
|
||
|
};
|
||
|
|
||
|
// PS3 PPU Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY PS3PPUTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// PS3 PPU defines.
|
||
|
Builder.defineMacro("__PPC__");
|
||
|
Builder.defineMacro("__PPU__");
|
||
|
Builder.defineMacro("__CELLOS_LV2__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
Builder.defineMacro("__LP32__");
|
||
|
Builder.defineMacro("_ARCH_PPC64");
|
||
|
Builder.defineMacro("__powerpc64__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
PS3PPUTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->LongWidth = this->LongAlign = 32;
|
||
|
this->PointerWidth = this->PointerAlign = 32;
|
||
|
this->IntMaxType = TargetInfo::SignedLongLong;
|
||
|
this->Int64Type = TargetInfo::SignedLongLong;
|
||
|
this->SizeType = TargetInfo::UnsignedInt;
|
||
|
this->resetDataLayout("E-m:e-p:32:32-i64:64-n32:64");
|
||
|
}
|
||
|
};
|
||
|
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY PS4OSTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
Builder.defineMacro("__FreeBSD__", "9");
|
||
|
Builder.defineMacro("__FreeBSD_cc_version", "900001");
|
||
|
Builder.defineMacro("__KPRINTF_ATTRIBUTE__");
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
Builder.defineMacro("__SCE__");
|
||
|
Builder.defineMacro("__ORBIS__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
PS4OSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->WCharType = TargetInfo::UnsignedShort;
|
||
|
|
||
|
// On PS4, TLS variable cannot be aligned to more than 32 bytes (256 bits).
|
||
|
this->MaxTLSAlign = 256;
|
||
|
|
||
|
// On PS4, do not honor explicit bit field alignment,
|
||
|
// as in "__attribute__((aligned(2))) int b : 1;".
|
||
|
this->UseExplicitBitFieldAlignment = false;
|
||
|
|
||
|
switch (Triple.getArch()) {
|
||
|
default:
|
||
|
case llvm::Triple::x86_64:
|
||
|
this->MCountName = ".mcount";
|
||
|
this->NewAlign = 256;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
TargetInfo::CallingConvCheckResult
|
||
|
checkCallingConvention(CallingConv CC) const override {
|
||
|
return (CC == CC_C) ? TargetInfo::CCCR_OK : TargetInfo::CCCR_Error;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// RTEMS Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY RTEMSTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// RTEMS defines; list based off of gcc output
|
||
|
|
||
|
Builder.defineMacro("__rtems__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
if (Opts.CPlusPlus)
|
||
|
Builder.defineMacro("_GNU_SOURCE");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
RTEMSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
switch (Triple.getArch()) {
|
||
|
default:
|
||
|
case llvm::Triple::x86:
|
||
|
// this->MCountName = ".mcount";
|
||
|
break;
|
||
|
case llvm::Triple::mips:
|
||
|
case llvm::Triple::mipsel:
|
||
|
case llvm::Triple::ppc:
|
||
|
case llvm::Triple::ppc64:
|
||
|
case llvm::Triple::ppc64le:
|
||
|
// this->MCountName = "_mcount";
|
||
|
break;
|
||
|
case llvm::Triple::arm:
|
||
|
// this->MCountName = "__mcount";
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// Solaris target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY SolarisTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
DefineStd(Builder, "sun", Opts);
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
Builder.defineMacro("__svr4__");
|
||
|
Builder.defineMacro("__SVR4");
|
||
|
// Solaris headers require _XOPEN_SOURCE to be set to 600 for C99 and
|
||
|
// newer, but to 500 for everything else. feature_test.h has a check to
|
||
|
// ensure that you are not using C99 with an old version of X/Open or C89
|
||
|
// with a new version.
|
||
|
if (Opts.C99)
|
||
|
Builder.defineMacro("_XOPEN_SOURCE", "600");
|
||
|
else
|
||
|
Builder.defineMacro("_XOPEN_SOURCE", "500");
|
||
|
if (Opts.CPlusPlus) {
|
||
|
Builder.defineMacro("__C99FEATURES__");
|
||
|
Builder.defineMacro("_FILE_OFFSET_BITS", "64");
|
||
|
}
|
||
|
// GCC restricts the next two to C++.
|
||
|
Builder.defineMacro("_LARGEFILE_SOURCE");
|
||
|
Builder.defineMacro("_LARGEFILE64_SOURCE");
|
||
|
Builder.defineMacro("__EXTENSIONS__");
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
if (this->HasFloat128)
|
||
|
Builder.defineMacro("__FLOAT128__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
SolarisTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
if (this->PointerWidth == 64) {
|
||
|
this->WCharType = this->WIntType = this->SignedInt;
|
||
|
} else {
|
||
|
this->WCharType = this->WIntType = this->SignedLong;
|
||
|
}
|
||
|
switch (Triple.getArch()) {
|
||
|
default:
|
||
|
break;
|
||
|
case llvm::Triple::x86:
|
||
|
case llvm::Triple::x86_64:
|
||
|
this->HasFloat128 = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// AIX Target
|
||
|
template <typename Target>
|
||
|
class AIXTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("_IBMR2");
|
||
|
Builder.defineMacro("_POWER");
|
||
|
|
||
|
Builder.defineMacro("_AIX");
|
||
|
|
||
|
if (Opts.EnableAIXExtendedAltivecABI)
|
||
|
Builder.defineMacro("__EXTABI__");
|
||
|
|
||
|
unsigned Major, Minor, Micro;
|
||
|
Triple.getOSVersion(Major, Minor, Micro);
|
||
|
|
||
|
// Define AIX OS-Version Macros.
|
||
|
// Includes logic for legacy versions of AIX; no specific intent to support.
|
||
|
std::pair<int, int> OsVersion = {Major, Minor};
|
||
|
if (OsVersion >= std::make_pair(3, 2)) Builder.defineMacro("_AIX32");
|
||
|
if (OsVersion >= std::make_pair(4, 1)) Builder.defineMacro("_AIX41");
|
||
|
if (OsVersion >= std::make_pair(4, 3)) Builder.defineMacro("_AIX43");
|
||
|
if (OsVersion >= std::make_pair(5, 0)) Builder.defineMacro("_AIX50");
|
||
|
if (OsVersion >= std::make_pair(5, 1)) Builder.defineMacro("_AIX51");
|
||
|
if (OsVersion >= std::make_pair(5, 2)) Builder.defineMacro("_AIX52");
|
||
|
if (OsVersion >= std::make_pair(5, 3)) Builder.defineMacro("_AIX53");
|
||
|
if (OsVersion >= std::make_pair(6, 1)) Builder.defineMacro("_AIX61");
|
||
|
if (OsVersion >= std::make_pair(7, 1)) Builder.defineMacro("_AIX71");
|
||
|
if (OsVersion >= std::make_pair(7, 2)) Builder.defineMacro("_AIX72");
|
||
|
|
||
|
// FIXME: Do not define _LONG_LONG when -fno-long-long is specified.
|
||
|
Builder.defineMacro("_LONG_LONG");
|
||
|
|
||
|
if (Opts.POSIXThreads) {
|
||
|
Builder.defineMacro("_THREAD_SAFE");
|
||
|
}
|
||
|
|
||
|
if (this->PointerWidth == 64) {
|
||
|
Builder.defineMacro("__64BIT__");
|
||
|
}
|
||
|
|
||
|
// Define _WCHAR_T when it is a fundamental type
|
||
|
// (i.e., for C++ without -fno-wchar).
|
||
|
if (Opts.CPlusPlus && Opts.WChar) {
|
||
|
Builder.defineMacro("_WCHAR_T");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
AIXTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->TheCXXABI.set(TargetCXXABI::XL);
|
||
|
|
||
|
if (this->PointerWidth == 64) {
|
||
|
this->WCharType = this->UnsignedInt;
|
||
|
} else {
|
||
|
this->WCharType = this->UnsignedShort;
|
||
|
}
|
||
|
this->UseZeroLengthBitfieldAlignment = true;
|
||
|
}
|
||
|
|
||
|
// AIX sets FLT_EVAL_METHOD to be 1.
|
||
|
unsigned getFloatEvalMethod() const override { return 1; }
|
||
|
bool hasInt128Type() const override { return false; }
|
||
|
|
||
|
bool defaultsToAIXPowerAlignment() const override { return true; }
|
||
|
};
|
||
|
|
||
|
// z/OS target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY ZOSTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// FIXME: _LONG_LONG should not be defined under -std=c89.
|
||
|
Builder.defineMacro("_LONG_LONG");
|
||
|
Builder.defineMacro("_OPEN_DEFAULT");
|
||
|
// _UNIX03_WITHDRAWN is required to build libcxx.
|
||
|
Builder.defineMacro("_UNIX03_WITHDRAWN");
|
||
|
Builder.defineMacro("__370__");
|
||
|
Builder.defineMacro("__BFP__");
|
||
|
// FIXME: __BOOL__ should not be defined under -std=c89.
|
||
|
Builder.defineMacro("__BOOL__");
|
||
|
Builder.defineMacro("__LONGNAME__");
|
||
|
Builder.defineMacro("__MVS__");
|
||
|
Builder.defineMacro("__THW_370__");
|
||
|
Builder.defineMacro("__THW_BIG_ENDIAN__");
|
||
|
Builder.defineMacro("__TOS_390__");
|
||
|
Builder.defineMacro("__TOS_MVS__");
|
||
|
Builder.defineMacro("__XPLINK__");
|
||
|
|
||
|
if (this->PointerWidth == 64)
|
||
|
Builder.defineMacro("__64BIT__");
|
||
|
|
||
|
if (Opts.CPlusPlus) {
|
||
|
Builder.defineMacro("__DLL__");
|
||
|
// _XOPEN_SOURCE=600 is required to build libcxx.
|
||
|
Builder.defineMacro("_XOPEN_SOURCE", "600");
|
||
|
}
|
||
|
|
||
|
if (Opts.GNUMode) {
|
||
|
Builder.defineMacro("_MI_BUILTIN");
|
||
|
Builder.defineMacro("_EXT");
|
||
|
}
|
||
|
|
||
|
if (Opts.CPlusPlus && Opts.WChar) {
|
||
|
// Macro __wchar_t is defined so that the wchar_t data
|
||
|
// type is not declared as a typedef in system headers.
|
||
|
Builder.defineMacro("__wchar_t");
|
||
|
}
|
||
|
|
||
|
this->PlatformName = llvm::Triple::getOSTypeName(Triple.getOS());
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
ZOSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->WCharType = TargetInfo::UnsignedInt;
|
||
|
this->UseBitFieldTypeAlignment = false;
|
||
|
this->UseZeroLengthBitfieldAlignment = true;
|
||
|
this->ZeroLengthBitfieldBoundary = 32;
|
||
|
this->MinGlobalAlign = 0;
|
||
|
this->DefaultAlignForAttributeAligned = 128;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
void addWindowsDefines(const llvm::Triple &Triple, const LangOptions &Opts,
|
||
|
MacroBuilder &Builder);
|
||
|
|
||
|
// Windows target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY WindowsTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
addWindowsDefines(Triple, Opts, Builder);
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
WindowsTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->WCharType = TargetInfo::UnsignedShort;
|
||
|
this->WIntType = TargetInfo::UnsignedShort;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY NaClTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
if (Opts.CPlusPlus)
|
||
|
Builder.defineMacro("_GNU_SOURCE");
|
||
|
|
||
|
DefineStd(Builder, "unix", Opts);
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
Builder.defineMacro("__native_client__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
NaClTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->LongAlign = 32;
|
||
|
this->LongWidth = 32;
|
||
|
this->PointerAlign = 32;
|
||
|
this->PointerWidth = 32;
|
||
|
this->IntMaxType = TargetInfo::SignedLongLong;
|
||
|
this->Int64Type = TargetInfo::SignedLongLong;
|
||
|
this->DoubleAlign = 64;
|
||
|
this->LongDoubleWidth = 64;
|
||
|
this->LongDoubleAlign = 64;
|
||
|
this->LongLongWidth = 64;
|
||
|
this->LongLongAlign = 64;
|
||
|
this->SizeType = TargetInfo::UnsignedInt;
|
||
|
this->PtrDiffType = TargetInfo::SignedInt;
|
||
|
this->IntPtrType = TargetInfo::SignedInt;
|
||
|
// RegParmMax is inherited from the underlying architecture.
|
||
|
this->LongDoubleFormat = &llvm::APFloat::IEEEdouble();
|
||
|
if (Triple.getArch() == llvm::Triple::arm) {
|
||
|
// Handled in ARM's setABI().
|
||
|
} else if (Triple.getArch() == llvm::Triple::x86) {
|
||
|
this->resetDataLayout("e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-"
|
||
|
"i64:64-n8:16:32-S128");
|
||
|
} else if (Triple.getArch() == llvm::Triple::x86_64) {
|
||
|
this->resetDataLayout("e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-"
|
||
|
"i64:64-n8:16:32:64-S128");
|
||
|
} else if (Triple.getArch() == llvm::Triple::mipsel) {
|
||
|
// Handled on mips' setDataLayout.
|
||
|
} else {
|
||
|
assert(Triple.getArch() == llvm::Triple::le32);
|
||
|
this->resetDataLayout("e-p:32:32-i64:64");
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// Fuchsia Target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY FuchsiaTargetInfo : public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
Builder.defineMacro("__Fuchsia__");
|
||
|
Builder.defineMacro("__ELF__");
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
// Required by the libc++ locale support.
|
||
|
if (Opts.CPlusPlus)
|
||
|
Builder.defineMacro("_GNU_SOURCE");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
FuchsiaTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->MCountName = "__mcount";
|
||
|
this->TheCXXABI.set(TargetCXXABI::Fuchsia);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// WebAssembly target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY WebAssemblyOSTargetInfo
|
||
|
: public OSTargetInfo<Target> {
|
||
|
protected:
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const override {
|
||
|
// A common platform macro.
|
||
|
if (Opts.POSIXThreads)
|
||
|
Builder.defineMacro("_REENTRANT");
|
||
|
// Follow g++ convention and predefine _GNU_SOURCE for C++.
|
||
|
if (Opts.CPlusPlus)
|
||
|
Builder.defineMacro("_GNU_SOURCE");
|
||
|
// Indicate that we have __float128.
|
||
|
Builder.defineMacro("__FLOAT128__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
explicit WebAssemblyOSTargetInfo(const llvm::Triple &Triple,
|
||
|
const TargetOptions &Opts)
|
||
|
: OSTargetInfo<Target>(Triple, Opts) {
|
||
|
this->MCountName = "__mcount";
|
||
|
this->TheCXXABI.set(TargetCXXABI::WebAssembly);
|
||
|
this->HasFloat128 = true;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// WASI target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY WASITargetInfo
|
||
|
: public WebAssemblyOSTargetInfo<Target> {
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const final {
|
||
|
WebAssemblyOSTargetInfo<Target>::getOSDefines(Opts, Triple, Builder);
|
||
|
Builder.defineMacro("__wasi__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
explicit WASITargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: WebAssemblyOSTargetInfo<Target>(Triple, Opts) {}
|
||
|
};
|
||
|
|
||
|
// Emscripten target
|
||
|
template <typename Target>
|
||
|
class LLVM_LIBRARY_VISIBILITY EmscriptenTargetInfo
|
||
|
: public WebAssemblyOSTargetInfo<Target> {
|
||
|
void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple,
|
||
|
MacroBuilder &Builder) const final {
|
||
|
WebAssemblyOSTargetInfo<Target>::getOSDefines(Opts, Triple, Builder);
|
||
|
Builder.defineMacro("__EMSCRIPTEN__");
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
explicit EmscriptenTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts)
|
||
|
: WebAssemblyOSTargetInfo<Target>(Triple, Opts) {}
|
||
|
};
|
||
|
|
||
|
} // namespace targets
|
||
|
} // namespace clang
|
||
|
#endif // LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H
|