//===--- 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" namespace clang { namespace targets { template 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 class LLVM_LIBRARY_VISIBILITY CloudABITargetInfo : public OSTargetInfo { protected: void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const override { Builder.defineMacro("__CloudABI__"); // CloudABI uses ISO/IEC 10646:2012 for wchar_t, char16_t and char32_t. Builder.defineMacro("__STDC_ISO_10646__", "201206L"); } public: using OSTargetInfo::OSTargetInfo; }; // Ananas target template class LLVM_LIBRARY_VISIBILITY AnanasTargetInfo : public OSTargetInfo { protected: void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const override { // Ananas defines Builder.defineMacro("__Ananas__"); } public: using OSTargetInfo::OSTargetInfo; }; void getDarwinDefines(MacroBuilder &Builder, const LangOptions &Opts, const llvm::Triple &Triple, StringRef &PlatformName, VersionTuple &PlatformMinVersion); template class LLVM_LIBRARY_VISIBILITY DarwinTargetInfo : public OSTargetInfo { 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(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); } else if (Triple.isDriverKit()) { // No TLS on DriverKit. } this->MCountName = "\01mcount"; } 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; } if (T.getOSVersion() < MinVersion) return 64; return OSTargetInfo::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); } bool areDefaultedSMFStillPOD(const LangOptions &) const override { return false; } }; // DragonFlyBSD Target template class LLVM_LIBRARY_VISIBILITY DragonFlyBSDTargetInfo : public OSTargetInfo { 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("__KPRINTF_ATTRIBUTE__"); Builder.defineMacro("__tune_i386__"); DefineStd(Builder, "unix", Opts); if (this->HasFloat128) Builder.defineMacro("__FLOAT128__"); } public: DragonFlyBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(Triple, Opts) { switch (Triple.getArch()) { default: case llvm::Triple::x86: case llvm::Triple::x86_64: this->HasFloat128 = true; this->MCountName = ".mcount"; break; } } }; #ifndef FREEBSD_CC_VERSION #define FREEBSD_CC_VERSION 0U #endif // FreeBSD Target template class LLVM_LIBRARY_VISIBILITY FreeBSDTargetInfo : public OSTargetInfo { 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); if (this->HasFloat128) Builder.defineMacro("__FLOAT128__"); // 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(Triple, Opts) { switch (Triple.getArch()) { case llvm::Triple::x86: case llvm::Triple::x86_64: this->HasFloat128 = true; [[fallthrough]]; default: 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; case llvm::Triple::riscv32: case llvm::Triple::riscv64: break; } } }; // GNU/kFreeBSD Target template class LLVM_LIBRARY_VISIBILITY KFreeBSDTargetInfo : public OSTargetInfo { 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__"); if (Opts.POSIXThreads) Builder.defineMacro("_REENTRANT"); if (Opts.CPlusPlus) Builder.defineMacro("_GNU_SOURCE"); } public: using OSTargetInfo::OSTargetInfo; }; // Haiku Target template class LLVM_LIBRARY_VISIBILITY HaikuTargetInfo : public OSTargetInfo { 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__"); DefineStd(Builder, "unix", Opts); if (this->HasFloat128) Builder.defineMacro("__FLOAT128__"); } public: HaikuTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(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 class LLVM_LIBRARY_VISIBILITY HurdTargetInfo : public OSTargetInfo { 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__"); if (Opts.POSIXThreads) Builder.defineMacro("_REENTRANT"); if (Opts.CPlusPlus) Builder.defineMacro("_GNU_SOURCE"); } public: using OSTargetInfo::OSTargetInfo; }; // Minix Target template class LLVM_LIBRARY_VISIBILITY MinixTargetInfo : public OSTargetInfo { 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"); DefineStd(Builder, "unix", Opts); } public: using OSTargetInfo::OSTargetInfo; }; // Linux target template class LLVM_LIBRARY_VISIBILITY LinuxTargetInfo : public OSTargetInfo { 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); if (Triple.isAndroid()) { Builder.defineMacro("__ANDROID__", "1"); this->PlatformName = "android"; this->PlatformMinVersion = Triple.getEnvironmentVersion(); const unsigned Maj = this->PlatformMinVersion.getMajor(); 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(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 class LLVM_LIBRARY_VISIBILITY NetBSDTargetInfo : public OSTargetInfo { 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__"); if (Opts.POSIXThreads) Builder.defineMacro("_REENTRANT"); if (this->HasFloat128) Builder.defineMacro("__FLOAT128__"); } public: NetBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(Triple, Opts) { this->MCountName = "__mcount"; switch (Triple.getArch()) { default: break; case llvm::Triple::x86: case llvm::Triple::x86_64: this->HasFloat128 = true; break; } } }; // OpenBSD Target template class LLVM_LIBRARY_VISIBILITY OpenBSDTargetInfo : public OSTargetInfo { 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); if (Opts.POSIXThreads) Builder.defineMacro("_REENTRANT"); if (this->HasFloat128) Builder.defineMacro("__FLOAT128__"); if (Opts.C11) Builder.defineMacro("__STDC_NO_THREADS__"); } public: OpenBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(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; [[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; case llvm::Triple::riscv32: case llvm::Triple::riscv64: break; } } }; // PS3 PPU Target template class LLVM_LIBRARY_VISIBILITY PS3PPUTargetInfo : public OSTargetInfo { protected: void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const override { // PS3 PPU defines. Builder.defineMacro("__PPU__"); Builder.defineMacro("__CELLOS_LV2__"); Builder.defineMacro("__LP32__"); Builder.defineMacro("_ARCH_PPC64"); Builder.defineMacro("__powerpc64__"); } public: PS3PPUTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(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-Fi64-i64:64-n32:64"); } }; // Common base class for PS4/PS5 targets. template class LLVM_LIBRARY_VISIBILITY PSOSTargetInfo : public OSTargetInfo { 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("__SCE__"); Builder.defineMacro("__STDC_NO_COMPLEX__"); Builder.defineMacro("__STDC_NO_THREADS__"); } public: PSOSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(Triple, Opts) { this->WCharType = TargetInfo::UnsignedShort; // On PS4/PS5, TLS variable cannot be aligned to more than 32 bytes (256 // bits). this->MaxTLSAlign = 256; // On PS4/PS5, do not honor explicit bit field alignment, // as in "__attribute__((aligned(2))) int b : 1;". this->UseExplicitBitFieldAlignment = false; this->MCountName = ".mcount"; this->NewAlign = 256; this->SuitableAlign = 256; } TargetInfo::CallingConvCheckResult checkCallingConvention(CallingConv CC) const override { return (CC == CC_C) ? TargetInfo::CCCR_OK : TargetInfo::CCCR_Error; } bool areDefaultedSMFStillPOD(const LangOptions &) const override { return false; } }; // PS4 Target template class LLVM_LIBRARY_VISIBILITY PS4OSTargetInfo : public PSOSTargetInfo { protected: void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const override { // Start with base class defines. PSOSTargetInfo::getOSDefines(Opts, Triple, Builder); Builder.defineMacro("__ORBIS__"); } public: using PSOSTargetInfo::PSOSTargetInfo; }; // PS5 Target template class LLVM_LIBRARY_VISIBILITY PS5OSTargetInfo : public PSOSTargetInfo { protected: void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const override { // Start with base class defines. PSOSTargetInfo::getOSDefines(Opts, Triple, Builder); Builder.defineMacro("__PROSPERO__"); } public: using PSOSTargetInfo::PSOSTargetInfo; }; // RTEMS Target template class LLVM_LIBRARY_VISIBILITY RTEMSTargetInfo : public OSTargetInfo { 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__"); if (Opts.CPlusPlus) Builder.defineMacro("_GNU_SOURCE"); } public: RTEMSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(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 class LLVM_LIBRARY_VISIBILITY SolarisTargetInfo : public OSTargetInfo { protected: void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const override { DefineStd(Builder, "sun", Opts); DefineStd(Builder, "unix", Opts); 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(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 class AIXTargetInfo : public OSTargetInfo { 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("__THW_BIG_ENDIAN__"); Builder.defineMacro("_AIX"); Builder.defineMacro("__TOS_AIX__"); Builder.defineMacro("__HOS_AIX__"); if (Opts.C11) { Builder.defineMacro("__STDC_NO_ATOMICS__"); Builder.defineMacro("__STDC_NO_THREADS__"); } if (Opts.EnableAIXExtendedAltivecABI) Builder.defineMacro("__EXTABI__"); VersionTuple OsVersion = Triple.getOSVersion(); // Define AIX OS-Version Macros. // Includes logic for legacy versions of AIX; no specific intent to support. if (OsVersion >= VersionTuple(3, 2)) Builder.defineMacro("_AIX32"); if (OsVersion >= VersionTuple(4, 1)) Builder.defineMacro("_AIX41"); if (OsVersion >= VersionTuple(4, 3)) Builder.defineMacro("_AIX43"); if (OsVersion >= VersionTuple(5, 0)) Builder.defineMacro("_AIX50"); if (OsVersion >= VersionTuple(5, 1)) Builder.defineMacro("_AIX51"); if (OsVersion >= VersionTuple(5, 2)) Builder.defineMacro("_AIX52"); if (OsVersion >= VersionTuple(5, 3)) Builder.defineMacro("_AIX53"); if (OsVersion >= VersionTuple(6, 1)) Builder.defineMacro("_AIX61"); if (OsVersion >= VersionTuple(7, 1)) Builder.defineMacro("_AIX71"); if (OsVersion >= VersionTuple(7, 2)) Builder.defineMacro("_AIX72"); if (OsVersion >= VersionTuple(7, 3)) Builder.defineMacro("_AIX73"); // 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(Triple, Opts) { this->MCountName = "__mcount"; 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. LangOptions::FPEvalMethodKind getFPEvalMethod() const override { return LangOptions::FPEvalMethodKind::FEM_Double; } bool defaultsToAIXPowerAlignment() const override { return true; } bool areDefaultedSMFStillPOD(const LangOptions &) const override { return false; } }; // z/OS target template class LLVM_LIBRARY_VISIBILITY ZOSTargetInfo : public OSTargetInfo { 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("__370__"); Builder.defineMacro("__BFP__"); // FIXME: __BOOL__ should not be defined under -std=c89. Builder.defineMacro("__BOOL__"); Builder.defineMacro("__COMPILER_VER__", "0x50000000"); 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 && 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(Triple, Opts) { this->WCharType = TargetInfo::UnsignedInt; this->MaxAlignedAttribute = 128; this->UseBitFieldTypeAlignment = false; this->UseZeroLengthBitfieldAlignment = true; this->UseLeadingZeroLengthBitfield = false; this->ZeroLengthBitfieldBoundary = 32; this->TheCXXABI.set(TargetCXXABI::XL); } bool areDefaultedSMFStillPOD(const LangOptions &) const override { return false; } }; void addWindowsDefines(const llvm::Triple &Triple, const LangOptions &Opts, MacroBuilder &Builder); // Windows target template class LLVM_LIBRARY_VISIBILITY WindowsTargetInfo : public OSTargetInfo { 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(Triple, Opts) { this->WCharType = TargetInfo::UnsignedShort; this->WIntType = TargetInfo::UnsignedShort; } }; template class LLVM_LIBRARY_VISIBILITY NaClTargetInfo : public OSTargetInfo { 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("__native_client__"); } public: NaClTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(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 class LLVM_LIBRARY_VISIBILITY FuchsiaTargetInfo : public OSTargetInfo { protected: void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const override { Builder.defineMacro("__Fuchsia__"); if (Opts.POSIXThreads) Builder.defineMacro("_REENTRANT"); // Required by the libc++ locale support. if (Opts.CPlusPlus) Builder.defineMacro("_GNU_SOURCE"); Builder.defineMacro("__Fuchsia_API_level__", Twine(Opts.FuchsiaAPILevel)); this->PlatformName = "fuchsia"; this->PlatformMinVersion = VersionTuple(Opts.FuchsiaAPILevel); } public: FuchsiaTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(Triple, Opts) { this->MCountName = "__mcount"; this->TheCXXABI.set(TargetCXXABI::Fuchsia); } }; // WebAssembly target template class LLVM_LIBRARY_VISIBILITY WebAssemblyOSTargetInfo : public OSTargetInfo { 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(Triple, Opts) { this->MCountName = "__mcount"; this->TheCXXABI.set(TargetCXXABI::WebAssembly); this->HasFloat128 = true; } }; // WASI target template class LLVM_LIBRARY_VISIBILITY WASITargetInfo : public WebAssemblyOSTargetInfo { void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const final { WebAssemblyOSTargetInfo::getOSDefines(Opts, Triple, Builder); Builder.defineMacro("__wasi__"); } public: using WebAssemblyOSTargetInfo::WebAssemblyOSTargetInfo; }; // Emscripten target template class LLVM_LIBRARY_VISIBILITY EmscriptenTargetInfo : public WebAssemblyOSTargetInfo { void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, MacroBuilder &Builder) const final { WebAssemblyOSTargetInfo::getOSDefines(Opts, Triple, Builder); DefineStd(Builder, "unix", Opts); Builder.defineMacro("__EMSCRIPTEN__"); if (Opts.POSIXThreads) Builder.defineMacro("__EMSCRIPTEN_PTHREADS__"); } public: explicit EmscriptenTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : WebAssemblyOSTargetInfo(Triple, Opts) { // Keeping the alignment of long double to 8 bytes even though its size is // 16 bytes allows emscripten to have an 8-byte-aligned max_align_t which // in turn gives is a 8-byte aligned malloc. // Emscripten's ABI is unstable and we may change this back to 128 to match // the WebAssembly default in the future. this->LongDoubleAlign = 64; } }; // OHOS target template class LLVM_LIBRARY_VISIBILITY OHOSTargetInfo : public OSTargetInfo { 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); // Generic OHOS target defines if (Triple.isOHOSFamily()) { Builder.defineMacro("__OHOS_FAMILY__", "1"); auto Version = Triple.getEnvironmentVersion(); this->PlatformName = "ohos"; this->PlatformMinVersion = Version; Builder.defineMacro("__OHOS_Major__", Twine(Version.getMajor())); if (auto Minor = Version.getMinor()) Builder.defineMacro("__OHOS_Minor__", Twine(*Minor)); if (auto Subminor = Version.getSubminor()) Builder.defineMacro("__OHOS_Micro__", Twine(*Subminor)); } if (Triple.isOpenHOS()) Builder.defineMacro("__OHOS__"); if (Triple.isOSLinux()) { DefineStd(Builder, "linux", Opts); } else if (Triple.isOSLiteOS()) { Builder.defineMacro("__LITEOS__"); } if (Opts.POSIXThreads) Builder.defineMacro("_REENTRANT"); if (Opts.CPlusPlus) Builder.defineMacro("_GNU_SOURCE"); if (this->HasFloat128) Builder.defineMacro("__FLOAT128__"); } public: OHOSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) : OSTargetInfo(Triple, Opts) { this->WIntType = TargetInfo::UnsignedInt; switch (Triple.getArch()) { default: break; case llvm::Triple::x86: case llvm::Triple::x86_64: this->HasFloat128 = true; break; } } const char *getStaticInitSectionSpecifier() const override { return ".text.startup"; } }; } // namespace targets } // namespace clang #endif // LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H