1 //===--- OSTargets.h - Declare OS target feature support --------*- C++ -*-===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 // 9 // This file declares OS specific TargetInfo types. 10 //===----------------------------------------------------------------------===// 11 12 #ifndef LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H 13 #define LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H 14 15 #include "Targets.h" 16 17 namespace clang { 18 namespace targets { 19 20 template <typename TgtInfo> 21 class LLVM_LIBRARY_VISIBILITY OSTargetInfo : public TgtInfo { 22 protected: 23 virtual void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 24 MacroBuilder &Builder) const = 0; 25 26 public: 27 OSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 28 : TgtInfo(Triple, Opts) {} 29 30 void getTargetDefines(const LangOptions &Opts, 31 MacroBuilder &Builder) const override { 32 TgtInfo::getTargetDefines(Opts, Builder); 33 getOSDefines(Opts, TgtInfo::getTriple(), Builder); 34 } 35 }; 36 37 // CloudABI Target 38 template <typename Target> 39 class LLVM_LIBRARY_VISIBILITY CloudABITargetInfo : public OSTargetInfo<Target> { 40 protected: 41 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 42 MacroBuilder &Builder) const override { 43 Builder.defineMacro("__CloudABI__"); 44 Builder.defineMacro("__ELF__"); 45 46 // CloudABI uses ISO/IEC 10646:2012 for wchar_t, char16_t and char32_t. 47 Builder.defineMacro("__STDC_ISO_10646__", "201206L"); 48 Builder.defineMacro("__STDC_UTF_16__"); 49 Builder.defineMacro("__STDC_UTF_32__"); 50 } 51 52 public: 53 CloudABITargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 54 : OSTargetInfo<Target>(Triple, Opts) {} 55 }; 56 57 // Ananas target 58 template <typename Target> 59 class LLVM_LIBRARY_VISIBILITY AnanasTargetInfo : public OSTargetInfo<Target> { 60 protected: 61 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 62 MacroBuilder &Builder) const override { 63 // Ananas defines 64 Builder.defineMacro("__Ananas__"); 65 Builder.defineMacro("__ELF__"); 66 } 67 68 public: 69 AnanasTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 70 : OSTargetInfo<Target>(Triple, Opts) {} 71 }; 72 73 void getDarwinDefines(MacroBuilder &Builder, const LangOptions &Opts, 74 const llvm::Triple &Triple, StringRef &PlatformName, 75 VersionTuple &PlatformMinVersion); 76 77 template <typename Target> 78 class LLVM_LIBRARY_VISIBILITY DarwinTargetInfo : public OSTargetInfo<Target> { 79 protected: 80 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 81 MacroBuilder &Builder) const override { 82 getDarwinDefines(Builder, Opts, Triple, this->PlatformName, 83 this->PlatformMinVersion); 84 } 85 86 public: 87 DarwinTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 88 : OSTargetInfo<Target>(Triple, Opts) { 89 // By default, no TLS, and we list permitted architecture/OS 90 // combinations. 91 this->TLSSupported = false; 92 93 if (Triple.isMacOSX()) 94 this->TLSSupported = !Triple.isMacOSXVersionLT(10, 7); 95 else if (Triple.isiOS()) { 96 // 64-bit iOS supported it from 8 onwards, 32-bit device from 9 onwards, 97 // 32-bit simulator from 10 onwards. 98 if (Triple.isArch64Bit()) 99 this->TLSSupported = !Triple.isOSVersionLT(8); 100 else if (Triple.isArch32Bit()) { 101 if (!Triple.isSimulatorEnvironment()) 102 this->TLSSupported = !Triple.isOSVersionLT(9); 103 else 104 this->TLSSupported = !Triple.isOSVersionLT(10); 105 } 106 } else if (Triple.isWatchOS()) { 107 if (!Triple.isSimulatorEnvironment()) 108 this->TLSSupported = !Triple.isOSVersionLT(2); 109 else 110 this->TLSSupported = !Triple.isOSVersionLT(3); 111 } 112 113 this->MCountName = "\01mcount"; 114 } 115 116 const char *getStaticInitSectionSpecifier() const override { 117 // FIXME: We should return 0 when building kexts. 118 return "__TEXT,__StaticInit,regular,pure_instructions"; 119 } 120 121 /// Darwin does not support protected visibility. Darwin's "default" 122 /// is very similar to ELF's "protected"; Darwin requires a "weak" 123 /// attribute on declarations that can be dynamically replaced. 124 bool hasProtectedVisibility() const override { return false; } 125 126 unsigned getExnObjectAlignment() const override { 127 // Older versions of libc++abi guarantee an alignment of only 8-bytes for 128 // exception objects because of a bug in __cxa_exception that was 129 // eventually fixed in r319123. 130 llvm::VersionTuple MinVersion; 131 const llvm::Triple &T = this->getTriple(); 132 133 // Compute the earliest OS versions that have the fix to libc++abi. 134 switch (T.getOS()) { 135 case llvm::Triple::Darwin: 136 case llvm::Triple::MacOSX: // Earliest supporting version is 10.14. 137 MinVersion = llvm::VersionTuple(10U, 14U); 138 break; 139 case llvm::Triple::IOS: 140 case llvm::Triple::TvOS: // Earliest supporting version is 12.0.0. 141 MinVersion = llvm::VersionTuple(12U); 142 break; 143 case llvm::Triple::WatchOS: // Earliest supporting version is 5.0.0. 144 MinVersion = llvm::VersionTuple(5U); 145 break; 146 default: 147 // Conservatively return 8 bytes if OS is unknown. 148 return 64; 149 } 150 151 unsigned Major, Minor, Micro; 152 T.getOSVersion(Major, Minor, Micro); 153 if (llvm::VersionTuple(Major, Minor, Micro) < MinVersion) 154 return 64; 155 return OSTargetInfo<Target>::getExnObjectAlignment(); 156 } 157 158 TargetInfo::IntType getLeastIntTypeByWidth(unsigned BitWidth, 159 bool IsSigned) const final { 160 // Darwin uses `long long` for `int_least64_t` and `int_fast64_t`. 161 return BitWidth == 64 162 ? (IsSigned ? TargetInfo::SignedLongLong 163 : TargetInfo::UnsignedLongLong) 164 : TargetInfo::getLeastIntTypeByWidth(BitWidth, IsSigned); 165 } 166 }; 167 168 // DragonFlyBSD Target 169 template <typename Target> 170 class LLVM_LIBRARY_VISIBILITY DragonFlyBSDTargetInfo 171 : public OSTargetInfo<Target> { 172 protected: 173 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 174 MacroBuilder &Builder) const override { 175 // DragonFly defines; list based off of gcc output 176 Builder.defineMacro("__DragonFly__"); 177 Builder.defineMacro("__DragonFly_cc_version", "100001"); 178 Builder.defineMacro("__ELF__"); 179 Builder.defineMacro("__KPRINTF_ATTRIBUTE__"); 180 Builder.defineMacro("__tune_i386__"); 181 DefineStd(Builder, "unix", Opts); 182 } 183 184 public: 185 DragonFlyBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 186 : OSTargetInfo<Target>(Triple, Opts) { 187 switch (Triple.getArch()) { 188 default: 189 case llvm::Triple::x86: 190 case llvm::Triple::x86_64: 191 this->MCountName = ".mcount"; 192 break; 193 } 194 } 195 }; 196 197 #ifndef FREEBSD_CC_VERSION 198 #define FREEBSD_CC_VERSION 0U 199 #endif 200 201 // FreeBSD Target 202 template <typename Target> 203 class LLVM_LIBRARY_VISIBILITY FreeBSDTargetInfo : public OSTargetInfo<Target> { 204 protected: 205 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 206 MacroBuilder &Builder) const override { 207 // FreeBSD defines; list based off of gcc output 208 209 unsigned Release = Triple.getOSMajorVersion(); 210 if (Release == 0U) 211 Release = 8U; 212 unsigned CCVersion = FREEBSD_CC_VERSION; 213 if (CCVersion == 0U) 214 CCVersion = Release * 100000U + 1U; 215 216 Builder.defineMacro("__FreeBSD__", Twine(Release)); 217 Builder.defineMacro("__FreeBSD_cc_version", Twine(CCVersion)); 218 Builder.defineMacro("__KPRINTF_ATTRIBUTE__"); 219 DefineStd(Builder, "unix", Opts); 220 Builder.defineMacro("__ELF__"); 221 222 // On FreeBSD, wchar_t contains the number of the code point as 223 // used by the character set of the locale. These character sets are 224 // not necessarily a superset of ASCII. 225 // 226 // FIXME: This is wrong; the macro refers to the numerical values 227 // of wchar_t *literals*, which are not locale-dependent. However, 228 // FreeBSD systems apparently depend on us getting this wrong, and 229 // setting this to 1 is conforming even if all the basic source 230 // character literals have the same encoding as char and wchar_t. 231 Builder.defineMacro("__STDC_MB_MIGHT_NEQ_WC__", "1"); 232 } 233 234 public: 235 FreeBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 236 : OSTargetInfo<Target>(Triple, Opts) { 237 switch (Triple.getArch()) { 238 default: 239 case llvm::Triple::x86: 240 case llvm::Triple::x86_64: 241 this->MCountName = ".mcount"; 242 break; 243 case llvm::Triple::mips: 244 case llvm::Triple::mipsel: 245 case llvm::Triple::ppc: 246 case llvm::Triple::ppcle: 247 case llvm::Triple::ppc64: 248 case llvm::Triple::ppc64le: 249 this->MCountName = "_mcount"; 250 break; 251 case llvm::Triple::arm: 252 this->MCountName = "__mcount"; 253 break; 254 case llvm::Triple::riscv32: 255 case llvm::Triple::riscv64: 256 break; 257 } 258 } 259 }; 260 261 // GNU/kFreeBSD Target 262 template <typename Target> 263 class LLVM_LIBRARY_VISIBILITY KFreeBSDTargetInfo : public OSTargetInfo<Target> { 264 protected: 265 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 266 MacroBuilder &Builder) const override { 267 // GNU/kFreeBSD defines; list based off of gcc output 268 269 DefineStd(Builder, "unix", Opts); 270 Builder.defineMacro("__FreeBSD_kernel__"); 271 Builder.defineMacro("__GLIBC__"); 272 Builder.defineMacro("__ELF__"); 273 if (Opts.POSIXThreads) 274 Builder.defineMacro("_REENTRANT"); 275 if (Opts.CPlusPlus) 276 Builder.defineMacro("_GNU_SOURCE"); 277 } 278 279 public: 280 KFreeBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 281 : OSTargetInfo<Target>(Triple, Opts) {} 282 }; 283 284 // Haiku Target 285 template <typename Target> 286 class LLVM_LIBRARY_VISIBILITY HaikuTargetInfo : public OSTargetInfo<Target> { 287 protected: 288 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 289 MacroBuilder &Builder) const override { 290 // Haiku defines; list based off of gcc output 291 Builder.defineMacro("__HAIKU__"); 292 Builder.defineMacro("__ELF__"); 293 DefineStd(Builder, "unix", Opts); 294 if (this->HasFloat128) 295 Builder.defineMacro("__FLOAT128__"); 296 } 297 298 public: 299 HaikuTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 300 : OSTargetInfo<Target>(Triple, Opts) { 301 this->SizeType = TargetInfo::UnsignedLong; 302 this->IntPtrType = TargetInfo::SignedLong; 303 this->PtrDiffType = TargetInfo::SignedLong; 304 this->ProcessIDType = TargetInfo::SignedLong; 305 this->TLSSupported = false; 306 switch (Triple.getArch()) { 307 default: 308 break; 309 case llvm::Triple::x86: 310 case llvm::Triple::x86_64: 311 this->HasFloat128 = true; 312 break; 313 } 314 } 315 }; 316 317 // Hurd target 318 template <typename Target> 319 class LLVM_LIBRARY_VISIBILITY HurdTargetInfo : public OSTargetInfo<Target> { 320 protected: 321 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 322 MacroBuilder &Builder) const override { 323 // Hurd defines; list based off of gcc output. 324 DefineStd(Builder, "unix", Opts); 325 Builder.defineMacro("__GNU__"); 326 Builder.defineMacro("__gnu_hurd__"); 327 Builder.defineMacro("__MACH__"); 328 Builder.defineMacro("__GLIBC__"); 329 Builder.defineMacro("__ELF__"); 330 if (Opts.POSIXThreads) 331 Builder.defineMacro("_REENTRANT"); 332 if (Opts.CPlusPlus) 333 Builder.defineMacro("_GNU_SOURCE"); 334 } 335 public: 336 HurdTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 337 : OSTargetInfo<Target>(Triple, Opts) {} 338 }; 339 340 // Minix Target 341 template <typename Target> 342 class LLVM_LIBRARY_VISIBILITY MinixTargetInfo : public OSTargetInfo<Target> { 343 protected: 344 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 345 MacroBuilder &Builder) const override { 346 // Minix defines 347 348 Builder.defineMacro("__minix", "3"); 349 Builder.defineMacro("_EM_WSIZE", "4"); 350 Builder.defineMacro("_EM_PSIZE", "4"); 351 Builder.defineMacro("_EM_SSIZE", "2"); 352 Builder.defineMacro("_EM_LSIZE", "4"); 353 Builder.defineMacro("_EM_FSIZE", "4"); 354 Builder.defineMacro("_EM_DSIZE", "8"); 355 Builder.defineMacro("__ELF__"); 356 DefineStd(Builder, "unix", Opts); 357 } 358 359 public: 360 MinixTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 361 : OSTargetInfo<Target>(Triple, Opts) {} 362 }; 363 364 // Linux target 365 template <typename Target> 366 class LLVM_LIBRARY_VISIBILITY LinuxTargetInfo : public OSTargetInfo<Target> { 367 protected: 368 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 369 MacroBuilder &Builder) const override { 370 // Linux defines; list based off of gcc output 371 DefineStd(Builder, "unix", Opts); 372 DefineStd(Builder, "linux", Opts); 373 Builder.defineMacro("__ELF__"); 374 if (Triple.isAndroid()) { 375 Builder.defineMacro("__ANDROID__", "1"); 376 unsigned Maj, Min, Rev; 377 Triple.getEnvironmentVersion(Maj, Min, Rev); 378 this->PlatformName = "android"; 379 this->PlatformMinVersion = VersionTuple(Maj, Min, Rev); 380 if (Maj) { 381 Builder.defineMacro("__ANDROID_MIN_SDK_VERSION__", Twine(Maj)); 382 // This historical but ambiguous name for the minSdkVersion macro. Keep 383 // defined for compatibility. 384 Builder.defineMacro("__ANDROID_API__", "__ANDROID_MIN_SDK_VERSION__"); 385 } 386 } else { 387 Builder.defineMacro("__gnu_linux__"); 388 } 389 if (Opts.POSIXThreads) 390 Builder.defineMacro("_REENTRANT"); 391 if (Opts.CPlusPlus) 392 Builder.defineMacro("_GNU_SOURCE"); 393 if (this->HasFloat128) 394 Builder.defineMacro("__FLOAT128__"); 395 } 396 397 public: 398 LinuxTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 399 : OSTargetInfo<Target>(Triple, Opts) { 400 this->WIntType = TargetInfo::UnsignedInt; 401 402 switch (Triple.getArch()) { 403 default: 404 break; 405 case llvm::Triple::mips: 406 case llvm::Triple::mipsel: 407 case llvm::Triple::mips64: 408 case llvm::Triple::mips64el: 409 case llvm::Triple::ppc: 410 case llvm::Triple::ppcle: 411 case llvm::Triple::ppc64: 412 case llvm::Triple::ppc64le: 413 this->MCountName = "_mcount"; 414 break; 415 case llvm::Triple::x86: 416 case llvm::Triple::x86_64: 417 this->HasFloat128 = true; 418 break; 419 } 420 } 421 422 const char *getStaticInitSectionSpecifier() const override { 423 return ".text.startup"; 424 } 425 }; 426 427 // NetBSD Target 428 template <typename Target> 429 class LLVM_LIBRARY_VISIBILITY NetBSDTargetInfo : public OSTargetInfo<Target> { 430 protected: 431 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 432 MacroBuilder &Builder) const override { 433 // NetBSD defines; list based off of gcc output 434 Builder.defineMacro("__NetBSD__"); 435 Builder.defineMacro("__unix__"); 436 Builder.defineMacro("__ELF__"); 437 if (Opts.POSIXThreads) 438 Builder.defineMacro("_REENTRANT"); 439 } 440 441 public: 442 NetBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 443 : OSTargetInfo<Target>(Triple, Opts) { 444 this->MCountName = "__mcount"; 445 } 446 }; 447 448 // OpenBSD Target 449 template <typename Target> 450 class LLVM_LIBRARY_VISIBILITY OpenBSDTargetInfo : public OSTargetInfo<Target> { 451 protected: 452 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 453 MacroBuilder &Builder) const override { 454 // OpenBSD defines; list based off of gcc output 455 456 Builder.defineMacro("__OpenBSD__"); 457 DefineStd(Builder, "unix", Opts); 458 Builder.defineMacro("__ELF__"); 459 if (Opts.POSIXThreads) 460 Builder.defineMacro("_REENTRANT"); 461 if (this->HasFloat128) 462 Builder.defineMacro("__FLOAT128__"); 463 464 if (Opts.C11) 465 Builder.defineMacro("__STDC_NO_THREADS__"); 466 } 467 468 public: 469 OpenBSDTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 470 : OSTargetInfo<Target>(Triple, Opts) { 471 this->WCharType = this->WIntType = this->SignedInt; 472 this->IntMaxType = TargetInfo::SignedLongLong; 473 this->Int64Type = TargetInfo::SignedLongLong; 474 switch (Triple.getArch()) { 475 case llvm::Triple::x86: 476 case llvm::Triple::x86_64: 477 this->HasFloat128 = true; 478 LLVM_FALLTHROUGH; 479 default: 480 this->MCountName = "__mcount"; 481 break; 482 case llvm::Triple::mips64: 483 case llvm::Triple::mips64el: 484 case llvm::Triple::ppc: 485 case llvm::Triple::ppc64: 486 case llvm::Triple::ppc64le: 487 case llvm::Triple::sparcv9: 488 this->MCountName = "_mcount"; 489 break; 490 case llvm::Triple::riscv32: 491 case llvm::Triple::riscv64: 492 break; 493 } 494 } 495 }; 496 497 // PSP Target 498 template <typename Target> 499 class LLVM_LIBRARY_VISIBILITY PSPTargetInfo : public OSTargetInfo<Target> { 500 protected: 501 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 502 MacroBuilder &Builder) const override { 503 // PSP defines; list based on the output of the pspdev gcc toolchain. 504 Builder.defineMacro("PSP"); 505 Builder.defineMacro("_PSP"); 506 Builder.defineMacro("__psp__"); 507 Builder.defineMacro("__ELF__"); 508 } 509 510 public: 511 PSPTargetInfo(const llvm::Triple &Triple) : OSTargetInfo<Target>(Triple) {} 512 }; 513 514 // PS3 PPU Target 515 template <typename Target> 516 class LLVM_LIBRARY_VISIBILITY PS3PPUTargetInfo : public OSTargetInfo<Target> { 517 protected: 518 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 519 MacroBuilder &Builder) const override { 520 // PS3 PPU defines. 521 Builder.defineMacro("__PPC__"); 522 Builder.defineMacro("__PPU__"); 523 Builder.defineMacro("__CELLOS_LV2__"); 524 Builder.defineMacro("__ELF__"); 525 Builder.defineMacro("__LP32__"); 526 Builder.defineMacro("_ARCH_PPC64"); 527 Builder.defineMacro("__powerpc64__"); 528 } 529 530 public: 531 PS3PPUTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 532 : OSTargetInfo<Target>(Triple, Opts) { 533 this->LongWidth = this->LongAlign = 32; 534 this->PointerWidth = this->PointerAlign = 32; 535 this->IntMaxType = TargetInfo::SignedLongLong; 536 this->Int64Type = TargetInfo::SignedLongLong; 537 this->SizeType = TargetInfo::UnsignedInt; 538 this->resetDataLayout("E-m:e-p:32:32-i64:64-n32:64"); 539 } 540 }; 541 542 template <typename Target> 543 class LLVM_LIBRARY_VISIBILITY PS4OSTargetInfo : public OSTargetInfo<Target> { 544 protected: 545 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 546 MacroBuilder &Builder) const override { 547 Builder.defineMacro("__FreeBSD__", "9"); 548 Builder.defineMacro("__FreeBSD_cc_version", "900001"); 549 Builder.defineMacro("__KPRINTF_ATTRIBUTE__"); 550 DefineStd(Builder, "unix", Opts); 551 Builder.defineMacro("__ELF__"); 552 Builder.defineMacro("__SCE__"); 553 Builder.defineMacro("__ORBIS__"); 554 } 555 556 public: 557 PS4OSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 558 : OSTargetInfo<Target>(Triple, Opts) { 559 this->WCharType = TargetInfo::UnsignedShort; 560 561 // On PS4, TLS variable cannot be aligned to more than 32 bytes (256 bits). 562 this->MaxTLSAlign = 256; 563 564 // On PS4, do not honor explicit bit field alignment, 565 // as in "__attribute__((aligned(2))) int b : 1;". 566 this->UseExplicitBitFieldAlignment = false; 567 568 switch (Triple.getArch()) { 569 default: 570 case llvm::Triple::x86_64: 571 this->MCountName = ".mcount"; 572 this->NewAlign = 256; 573 break; 574 } 575 } 576 TargetInfo::CallingConvCheckResult 577 checkCallingConvention(CallingConv CC) const override { 578 return (CC == CC_C) ? TargetInfo::CCCR_OK : TargetInfo::CCCR_Error; 579 } 580 }; 581 582 // RTEMS Target 583 template <typename Target> 584 class LLVM_LIBRARY_VISIBILITY RTEMSTargetInfo : public OSTargetInfo<Target> { 585 protected: 586 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 587 MacroBuilder &Builder) const override { 588 // RTEMS defines; list based off of gcc output 589 590 Builder.defineMacro("__rtems__"); 591 Builder.defineMacro("__ELF__"); 592 if (Opts.CPlusPlus) 593 Builder.defineMacro("_GNU_SOURCE"); 594 } 595 596 public: 597 RTEMSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 598 : OSTargetInfo<Target>(Triple, Opts) { 599 switch (Triple.getArch()) { 600 default: 601 case llvm::Triple::x86: 602 // this->MCountName = ".mcount"; 603 break; 604 case llvm::Triple::mips: 605 case llvm::Triple::mipsel: 606 case llvm::Triple::ppc: 607 case llvm::Triple::ppc64: 608 case llvm::Triple::ppc64le: 609 // this->MCountName = "_mcount"; 610 break; 611 case llvm::Triple::arm: 612 // this->MCountName = "__mcount"; 613 break; 614 } 615 } 616 }; 617 618 // Solaris target 619 template <typename Target> 620 class LLVM_LIBRARY_VISIBILITY SolarisTargetInfo : public OSTargetInfo<Target> { 621 protected: 622 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 623 MacroBuilder &Builder) const override { 624 DefineStd(Builder, "sun", Opts); 625 DefineStd(Builder, "unix", Opts); 626 Builder.defineMacro("__ELF__"); 627 Builder.defineMacro("__svr4__"); 628 Builder.defineMacro("__SVR4"); 629 // Solaris headers require _XOPEN_SOURCE to be set to 600 for C99 and 630 // newer, but to 500 for everything else. feature_test.h has a check to 631 // ensure that you are not using C99 with an old version of X/Open or C89 632 // with a new version. 633 if (Opts.C99) 634 Builder.defineMacro("_XOPEN_SOURCE", "600"); 635 else 636 Builder.defineMacro("_XOPEN_SOURCE", "500"); 637 if (Opts.CPlusPlus) { 638 Builder.defineMacro("__C99FEATURES__"); 639 Builder.defineMacro("_FILE_OFFSET_BITS", "64"); 640 } 641 // GCC restricts the next two to C++. 642 Builder.defineMacro("_LARGEFILE_SOURCE"); 643 Builder.defineMacro("_LARGEFILE64_SOURCE"); 644 Builder.defineMacro("__EXTENSIONS__"); 645 if (Opts.POSIXThreads) 646 Builder.defineMacro("_REENTRANT"); 647 if (this->HasFloat128) 648 Builder.defineMacro("__FLOAT128__"); 649 } 650 651 public: 652 SolarisTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 653 : OSTargetInfo<Target>(Triple, Opts) { 654 if (this->PointerWidth == 64) { 655 this->WCharType = this->WIntType = this->SignedInt; 656 } else { 657 this->WCharType = this->WIntType = this->SignedLong; 658 } 659 switch (Triple.getArch()) { 660 default: 661 break; 662 case llvm::Triple::x86: 663 case llvm::Triple::x86_64: 664 this->HasFloat128 = true; 665 break; 666 } 667 } 668 }; 669 670 // AIX Target 671 template <typename Target> 672 class AIXTargetInfo : public OSTargetInfo<Target> { 673 protected: 674 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 675 MacroBuilder &Builder) const override { 676 DefineStd(Builder, "unix", Opts); 677 Builder.defineMacro("_IBMR2"); 678 Builder.defineMacro("_POWER"); 679 680 Builder.defineMacro("_AIX"); 681 Builder.defineMacro("__TOS_AIX__"); 682 683 if (Opts.C11) { 684 Builder.defineMacro("__STDC_NO_ATOMICS__"); 685 Builder.defineMacro("__STDC_NO_THREADS__"); 686 } 687 688 if (Opts.EnableAIXExtendedAltivecABI) 689 Builder.defineMacro("__EXTABI__"); 690 691 unsigned Major, Minor, Micro; 692 Triple.getOSVersion(Major, Minor, Micro); 693 694 // Define AIX OS-Version Macros. 695 // Includes logic for legacy versions of AIX; no specific intent to support. 696 std::pair<int, int> OsVersion = {Major, Minor}; 697 if (OsVersion >= std::make_pair(3, 2)) Builder.defineMacro("_AIX32"); 698 if (OsVersion >= std::make_pair(4, 1)) Builder.defineMacro("_AIX41"); 699 if (OsVersion >= std::make_pair(4, 3)) Builder.defineMacro("_AIX43"); 700 if (OsVersion >= std::make_pair(5, 0)) Builder.defineMacro("_AIX50"); 701 if (OsVersion >= std::make_pair(5, 1)) Builder.defineMacro("_AIX51"); 702 if (OsVersion >= std::make_pair(5, 2)) Builder.defineMacro("_AIX52"); 703 if (OsVersion >= std::make_pair(5, 3)) Builder.defineMacro("_AIX53"); 704 if (OsVersion >= std::make_pair(6, 1)) Builder.defineMacro("_AIX61"); 705 if (OsVersion >= std::make_pair(7, 1)) Builder.defineMacro("_AIX71"); 706 if (OsVersion >= std::make_pair(7, 2)) Builder.defineMacro("_AIX72"); 707 if (OsVersion >= std::make_pair(7, 3)) Builder.defineMacro("_AIX73"); 708 709 // FIXME: Do not define _LONG_LONG when -fno-long-long is specified. 710 Builder.defineMacro("_LONG_LONG"); 711 712 if (Opts.POSIXThreads) { 713 Builder.defineMacro("_THREAD_SAFE"); 714 } 715 716 if (this->PointerWidth == 64) { 717 Builder.defineMacro("__64BIT__"); 718 } 719 720 // Define _WCHAR_T when it is a fundamental type 721 // (i.e., for C++ without -fno-wchar). 722 if (Opts.CPlusPlus && Opts.WChar) { 723 Builder.defineMacro("_WCHAR_T"); 724 } 725 } 726 727 public: 728 AIXTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 729 : OSTargetInfo<Target>(Triple, Opts) { 730 this->TheCXXABI.set(TargetCXXABI::XL); 731 732 if (this->PointerWidth == 64) { 733 this->WCharType = this->UnsignedInt; 734 } else { 735 this->WCharType = this->UnsignedShort; 736 } 737 this->UseZeroLengthBitfieldAlignment = true; 738 } 739 740 // AIX sets FLT_EVAL_METHOD to be 1. 741 unsigned getFloatEvalMethod() const override { return 1; } 742 bool hasInt128Type() const override { return false; } 743 744 bool defaultsToAIXPowerAlignment() const override { return true; } 745 }; 746 747 // z/OS target 748 template <typename Target> 749 class LLVM_LIBRARY_VISIBILITY ZOSTargetInfo : public OSTargetInfo<Target> { 750 protected: 751 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 752 MacroBuilder &Builder) const override { 753 // FIXME: _LONG_LONG should not be defined under -std=c89. 754 Builder.defineMacro("_LONG_LONG"); 755 Builder.defineMacro("_OPEN_DEFAULT"); 756 // _UNIX03_WITHDRAWN is required to build libcxx. 757 Builder.defineMacro("_UNIX03_WITHDRAWN"); 758 Builder.defineMacro("__370__"); 759 Builder.defineMacro("__BFP__"); 760 // FIXME: __BOOL__ should not be defined under -std=c89. 761 Builder.defineMacro("__BOOL__"); 762 Builder.defineMacro("__LONGNAME__"); 763 Builder.defineMacro("__MVS__"); 764 Builder.defineMacro("__THW_370__"); 765 Builder.defineMacro("__THW_BIG_ENDIAN__"); 766 Builder.defineMacro("__TOS_390__"); 767 Builder.defineMacro("__TOS_MVS__"); 768 Builder.defineMacro("__XPLINK__"); 769 770 if (this->PointerWidth == 64) 771 Builder.defineMacro("__64BIT__"); 772 773 if (Opts.CPlusPlus) { 774 Builder.defineMacro("__DLL__"); 775 // _XOPEN_SOURCE=600 is required to build libcxx. 776 Builder.defineMacro("_XOPEN_SOURCE", "600"); 777 } 778 779 if (Opts.GNUMode) { 780 Builder.defineMacro("_MI_BUILTIN"); 781 Builder.defineMacro("_EXT"); 782 } 783 784 if (Opts.CPlusPlus && Opts.WChar) { 785 // Macro __wchar_t is defined so that the wchar_t data 786 // type is not declared as a typedef in system headers. 787 Builder.defineMacro("__wchar_t"); 788 } 789 790 this->PlatformName = llvm::Triple::getOSTypeName(Triple.getOS()); 791 } 792 793 public: 794 ZOSTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 795 : OSTargetInfo<Target>(Triple, Opts) { 796 this->WCharType = TargetInfo::UnsignedInt; 797 this->MaxAlignedAttribute = 128; 798 this->UseBitFieldTypeAlignment = false; 799 this->UseZeroLengthBitfieldAlignment = true; 800 this->UseLeadingZeroLengthBitfield = false; 801 this->ZeroLengthBitfieldBoundary = 32; 802 this->DefaultAlignForAttributeAligned = 128; 803 } 804 }; 805 806 void addWindowsDefines(const llvm::Triple &Triple, const LangOptions &Opts, 807 MacroBuilder &Builder); 808 809 // Windows target 810 template <typename Target> 811 class LLVM_LIBRARY_VISIBILITY WindowsTargetInfo : public OSTargetInfo<Target> { 812 protected: 813 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 814 MacroBuilder &Builder) const override { 815 addWindowsDefines(Triple, Opts, Builder); 816 } 817 818 public: 819 WindowsTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 820 : OSTargetInfo<Target>(Triple, Opts) { 821 this->WCharType = TargetInfo::UnsignedShort; 822 this->WIntType = TargetInfo::UnsignedShort; 823 } 824 }; 825 826 template <typename Target> 827 class LLVM_LIBRARY_VISIBILITY NaClTargetInfo : public OSTargetInfo<Target> { 828 protected: 829 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 830 MacroBuilder &Builder) const override { 831 if (Opts.POSIXThreads) 832 Builder.defineMacro("_REENTRANT"); 833 if (Opts.CPlusPlus) 834 Builder.defineMacro("_GNU_SOURCE"); 835 836 DefineStd(Builder, "unix", Opts); 837 Builder.defineMacro("__ELF__"); 838 Builder.defineMacro("__native_client__"); 839 } 840 841 public: 842 NaClTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 843 : OSTargetInfo<Target>(Triple, Opts) { 844 this->LongAlign = 32; 845 this->LongWidth = 32; 846 this->PointerAlign = 32; 847 this->PointerWidth = 32; 848 this->IntMaxType = TargetInfo::SignedLongLong; 849 this->Int64Type = TargetInfo::SignedLongLong; 850 this->DoubleAlign = 64; 851 this->LongDoubleWidth = 64; 852 this->LongDoubleAlign = 64; 853 this->LongLongWidth = 64; 854 this->LongLongAlign = 64; 855 this->SizeType = TargetInfo::UnsignedInt; 856 this->PtrDiffType = TargetInfo::SignedInt; 857 this->IntPtrType = TargetInfo::SignedInt; 858 // RegParmMax is inherited from the underlying architecture. 859 this->LongDoubleFormat = &llvm::APFloat::IEEEdouble(); 860 if (Triple.getArch() == llvm::Triple::arm) { 861 // Handled in ARM's setABI(). 862 } else if (Triple.getArch() == llvm::Triple::x86) { 863 this->resetDataLayout("e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-" 864 "i64:64-n8:16:32-S128"); 865 } else if (Triple.getArch() == llvm::Triple::x86_64) { 866 this->resetDataLayout("e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-" 867 "i64:64-n8:16:32:64-S128"); 868 } else if (Triple.getArch() == llvm::Triple::mipsel) { 869 // Handled on mips' setDataLayout. 870 } else { 871 assert(Triple.getArch() == llvm::Triple::le32); 872 this->resetDataLayout("e-p:32:32-i64:64"); 873 } 874 } 875 }; 876 877 // Fuchsia Target 878 template <typename Target> 879 class LLVM_LIBRARY_VISIBILITY FuchsiaTargetInfo : public OSTargetInfo<Target> { 880 protected: 881 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 882 MacroBuilder &Builder) const override { 883 Builder.defineMacro("__Fuchsia__"); 884 Builder.defineMacro("__ELF__"); 885 if (Opts.POSIXThreads) 886 Builder.defineMacro("_REENTRANT"); 887 // Required by the libc++ locale support. 888 if (Opts.CPlusPlus) 889 Builder.defineMacro("_GNU_SOURCE"); 890 } 891 892 public: 893 FuchsiaTargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 894 : OSTargetInfo<Target>(Triple, Opts) { 895 this->MCountName = "__mcount"; 896 this->TheCXXABI.set(TargetCXXABI::Fuchsia); 897 } 898 }; 899 900 // WebAssembly target 901 template <typename Target> 902 class LLVM_LIBRARY_VISIBILITY WebAssemblyOSTargetInfo 903 : public OSTargetInfo<Target> { 904 protected: 905 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 906 MacroBuilder &Builder) const override { 907 // A common platform macro. 908 if (Opts.POSIXThreads) 909 Builder.defineMacro("_REENTRANT"); 910 // Follow g++ convention and predefine _GNU_SOURCE for C++. 911 if (Opts.CPlusPlus) 912 Builder.defineMacro("_GNU_SOURCE"); 913 // Indicate that we have __float128. 914 Builder.defineMacro("__FLOAT128__"); 915 } 916 917 public: 918 explicit WebAssemblyOSTargetInfo(const llvm::Triple &Triple, 919 const TargetOptions &Opts) 920 : OSTargetInfo<Target>(Triple, Opts) { 921 this->MCountName = "__mcount"; 922 this->TheCXXABI.set(TargetCXXABI::WebAssembly); 923 this->HasFloat128 = true; 924 } 925 }; 926 927 // WASI target 928 template <typename Target> 929 class LLVM_LIBRARY_VISIBILITY WASITargetInfo 930 : public WebAssemblyOSTargetInfo<Target> { 931 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 932 MacroBuilder &Builder) const final { 933 WebAssemblyOSTargetInfo<Target>::getOSDefines(Opts, Triple, Builder); 934 Builder.defineMacro("__wasi__"); 935 } 936 937 public: 938 explicit WASITargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 939 : WebAssemblyOSTargetInfo<Target>(Triple, Opts) {} 940 }; 941 942 // Emscripten target 943 template <typename Target> 944 class LLVM_LIBRARY_VISIBILITY EmscriptenTargetInfo 945 : public WebAssemblyOSTargetInfo<Target> { 946 void getOSDefines(const LangOptions &Opts, const llvm::Triple &Triple, 947 MacroBuilder &Builder) const final { 948 WebAssemblyOSTargetInfo<Target>::getOSDefines(Opts, Triple, Builder); 949 Builder.defineMacro("__EMSCRIPTEN__"); 950 if (Opts.POSIXThreads) 951 Builder.defineMacro("__EMSCRIPTEN_PTHREADS__"); 952 } 953 954 public: 955 explicit EmscriptenTargetInfo(const llvm::Triple &Triple, 956 const TargetOptions &Opts) 957 : WebAssemblyOSTargetInfo<Target>(Triple, Opts) { 958 // Keeping the alignment of long double to 8 bytes even though its size is 959 // 16 bytes allows emscripten to have an 8-byte-aligned max_align_t which 960 // in turn gives is a 8-byte aligned malloc. 961 // Emscripten's ABI is unstable and we may change this back to 128 to match 962 // the WebAssembly default in the future. 963 this->LongDoubleAlign = 64; 964 } 965 }; 966 967 } // namespace targets 968 } // namespace clang 969 #endif // LLVM_CLANG_LIB_BASIC_TARGETS_OSTARGETS_H 970