1 //===--- X86.h - Declare X86 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 X86 TargetInfo objects. 10 // 11 //===----------------------------------------------------------------------===// 12 13 #ifndef LLVM_CLANG_LIB_BASIC_TARGETS_X86_H 14 #define LLVM_CLANG_LIB_BASIC_TARGETS_X86_H 15 16 #include "OSTargets.h" 17 #include "clang/Basic/BitmaskEnum.h" 18 #include "clang/Basic/TargetInfo.h" 19 #include "clang/Basic/TargetOptions.h" 20 #include "llvm/Support/Compiler.h" 21 #include "llvm/TargetParser/Triple.h" 22 #include "llvm/TargetParser/X86TargetParser.h" 23 #include <optional> 24 25 namespace clang { 26 namespace targets { 27 28 static const unsigned X86AddrSpaceMap[] = { 29 0, // Default 30 0, // opencl_global 31 0, // opencl_local 32 0, // opencl_constant 33 0, // opencl_private 34 0, // opencl_generic 35 0, // opencl_global_device 36 0, // opencl_global_host 37 0, // cuda_device 38 0, // cuda_constant 39 0, // cuda_shared 40 0, // sycl_global 41 0, // sycl_global_device 42 0, // sycl_global_host 43 0, // sycl_local 44 0, // sycl_private 45 270, // ptr32_sptr 46 271, // ptr32_uptr 47 272, // ptr64 48 0, // hlsl_groupshared 49 // Wasm address space values for this target are dummy values, 50 // as it is only enabled for Wasm targets. 51 20, // wasm_funcref 52 }; 53 54 // X86 target abstract base class; x86-32 and x86-64 are very close, so 55 // most of the implementation can be shared. 56 class LLVM_LIBRARY_VISIBILITY X86TargetInfo : public TargetInfo { 57 58 enum X86SSEEnum { 59 NoSSE, 60 SSE1, 61 SSE2, 62 SSE3, 63 SSSE3, 64 SSE41, 65 SSE42, 66 AVX, 67 AVX2, 68 AVX512F 69 } SSELevel = NoSSE; 70 bool HasMMX = false; 71 enum XOPEnum { NoXOP, SSE4A, FMA4, XOP } XOPLevel = NoXOP; 72 enum AddrSpace { ptr32_sptr = 270, ptr32_uptr = 271, ptr64 = 272 }; 73 74 bool HasAES = false; 75 bool HasVAES = false; 76 bool HasPCLMUL = false; 77 bool HasVPCLMULQDQ = false; 78 bool HasGFNI = false; 79 bool HasLZCNT = false; 80 bool HasRDRND = false; 81 bool HasFSGSBASE = false; 82 bool HasBMI = false; 83 bool HasBMI2 = false; 84 bool HasPOPCNT = false; 85 bool HasRTM = false; 86 bool HasPRFCHW = false; 87 bool HasRDSEED = false; 88 bool HasADX = false; 89 bool HasTBM = false; 90 bool HasLWP = false; 91 bool HasFMA = false; 92 bool HasF16C = false; 93 bool HasAVX10_1 = false; 94 bool HasAVX10_1_512 = false; 95 bool HasEVEX512 = false; 96 bool HasAVX512CD = false; 97 bool HasAVX512VPOPCNTDQ = false; 98 bool HasAVX512VNNI = false; 99 bool HasAVX512FP16 = false; 100 bool HasAVX512BF16 = false; 101 bool HasAVX512DQ = false; 102 bool HasAVX512BITALG = false; 103 bool HasAVX512BW = false; 104 bool HasAVX512VL = false; 105 bool HasAVX512VBMI = false; 106 bool HasAVX512VBMI2 = false; 107 bool HasAVXIFMA = false; 108 bool HasAVX512IFMA = false; 109 bool HasAVX512VP2INTERSECT = false; 110 bool HasSHA = false; 111 bool HasSHA512 = false; 112 bool HasSHSTK = false; 113 bool HasSM3 = false; 114 bool HasSGX = false; 115 bool HasSM4 = false; 116 bool HasCX8 = false; 117 bool HasCX16 = false; 118 bool HasFXSR = false; 119 bool HasXSAVE = false; 120 bool HasXSAVEOPT = false; 121 bool HasXSAVEC = false; 122 bool HasXSAVES = false; 123 bool HasMWAITX = false; 124 bool HasCLZERO = false; 125 bool HasCLDEMOTE = false; 126 bool HasPCONFIG = false; 127 bool HasPKU = false; 128 bool HasCLFLUSHOPT = false; 129 bool HasCLWB = false; 130 bool HasMOVBE = false; 131 bool HasPREFETCHI = false; 132 bool HasRDPID = false; 133 bool HasRDPRU = false; 134 bool HasRetpolineExternalThunk = false; 135 bool HasLAHFSAHF = false; 136 bool HasWBNOINVD = false; 137 bool HasWAITPKG = false; 138 bool HasMOVDIRI = false; 139 bool HasMOVDIR64B = false; 140 bool HasPTWRITE = false; 141 bool HasINVPCID = false; 142 bool HasENQCMD = false; 143 bool HasAVXVNNIINT16 = false; 144 bool HasAMXFP16 = false; 145 bool HasCMPCCXADD = false; 146 bool HasRAOINT = false; 147 bool HasAVXVNNIINT8 = false; 148 bool HasAVXNECONVERT = false; 149 bool HasKL = false; // For key locker 150 bool HasWIDEKL = false; // For wide key locker 151 bool HasHRESET = false; 152 bool HasAVXVNNI = false; 153 bool HasAMXTILE = false; 154 bool HasAMXINT8 = false; 155 bool HasAMXBF16 = false; 156 bool HasAMXCOMPLEX = false; 157 bool HasSERIALIZE = false; 158 bool HasTSXLDTRK = false; 159 bool HasUSERMSR = false; 160 bool HasUINTR = false; 161 bool HasCRC32 = false; 162 bool HasX87 = false; 163 bool HasEGPR = false; 164 bool HasPush2Pop2 = false; 165 bool HasPPX = false; 166 bool HasNDD = false; 167 bool HasCCMP = false; 168 bool HasNF = false; 169 bool HasCF = false; 170 bool HasZU = false; 171 bool HasInlineAsmUseGPR32 = false; 172 bool HasBranchHint = false; 173 174 protected: 175 llvm::X86::CPUKind CPU = llvm::X86::CK_None; 176 177 enum FPMathKind { FP_Default, FP_SSE, FP_387 } FPMath = FP_Default; 178 179 public: 180 X86TargetInfo(const llvm::Triple &Triple, const TargetOptions &) 181 : TargetInfo(Triple) { 182 BFloat16Width = BFloat16Align = 16; 183 BFloat16Format = &llvm::APFloat::BFloat(); 184 LongDoubleFormat = &llvm::APFloat::x87DoubleExtended(); 185 AddrSpaceMap = &X86AddrSpaceMap; 186 HasStrictFP = true; 187 HasUnalignedAccess = true; 188 189 bool IsWinCOFF = 190 getTriple().isOSWindows() && getTriple().isOSBinFormatCOFF(); 191 if (IsWinCOFF) 192 MaxVectorAlign = MaxTLSAlign = 8192u * getCharWidth(); 193 } 194 195 const char *getLongDoubleMangling() const override { 196 return LongDoubleFormat == &llvm::APFloat::IEEEquad() ? "g" : "e"; 197 } 198 199 LangOptions::FPEvalMethodKind getFPEvalMethod() const override { 200 // X87 evaluates with 80 bits "long double" precision. 201 return SSELevel == NoSSE ? LangOptions::FPEvalMethodKind::FEM_Extended 202 : LangOptions::FPEvalMethodKind::FEM_Source; 203 } 204 205 // EvalMethod `source` is not supported for targets with `NoSSE` feature. 206 bool supportSourceEvalMethod() const override { return SSELevel > NoSSE; } 207 208 ArrayRef<const char *> getGCCRegNames() const override; 209 210 ArrayRef<TargetInfo::GCCRegAlias> getGCCRegAliases() const override { 211 return std::nullopt; 212 } 213 214 ArrayRef<TargetInfo::AddlRegName> getGCCAddlRegNames() const override; 215 216 bool isSPRegName(StringRef RegName) const override { 217 return RegName == "esp" || RegName == "rsp"; 218 } 219 220 bool supportsCpuSupports() const override { return true; } 221 bool supportsCpuIs() const override { return true; } 222 bool supportsCpuInit() const override { return true; } 223 224 bool validateCpuSupports(StringRef FeatureStr) const override; 225 226 bool validateCpuIs(StringRef FeatureStr) const override; 227 228 bool validateCPUSpecificCPUDispatch(StringRef Name) const override; 229 230 char CPUSpecificManglingCharacter(StringRef Name) const override; 231 232 void getCPUSpecificCPUDispatchFeatures( 233 StringRef Name, 234 llvm::SmallVectorImpl<StringRef> &Features) const override; 235 236 std::optional<unsigned> getCPUCacheLineSize() const override; 237 238 bool validateAsmConstraint(const char *&Name, 239 TargetInfo::ConstraintInfo &info) const override; 240 241 bool validateGlobalRegisterVariable(StringRef RegName, unsigned RegSize, 242 bool &HasSizeMismatch) const override { 243 // esp and ebp are the only 32-bit registers the x86 backend can currently 244 // handle. 245 if (RegName == "esp" || RegName == "ebp") { 246 // Check that the register size is 32-bit. 247 HasSizeMismatch = RegSize != 32; 248 return true; 249 } 250 251 return false; 252 } 253 254 bool validateOutputSize(const llvm::StringMap<bool> &FeatureMap, 255 StringRef Constraint, unsigned Size) const override; 256 257 bool validateInputSize(const llvm::StringMap<bool> &FeatureMap, 258 StringRef Constraint, unsigned Size) const override; 259 260 bool 261 checkCFProtectionReturnSupported(DiagnosticsEngine &Diags) const override { 262 if (CPU == llvm::X86::CK_None || CPU >= llvm::X86::CK_PentiumPro) 263 return true; 264 return TargetInfo::checkCFProtectionReturnSupported(Diags); 265 }; 266 267 bool 268 checkCFProtectionBranchSupported(DiagnosticsEngine &Diags) const override { 269 if (CPU == llvm::X86::CK_None || CPU >= llvm::X86::CK_PentiumPro) 270 return true; 271 return TargetInfo::checkCFProtectionBranchSupported(Diags); 272 }; 273 274 virtual bool validateOperandSize(const llvm::StringMap<bool> &FeatureMap, 275 StringRef Constraint, unsigned Size) const; 276 277 std::string convertConstraint(const char *&Constraint) const override; 278 std::string_view getClobbers() const override { 279 return "~{dirflag},~{fpsr},~{flags}"; 280 } 281 282 StringRef getConstraintRegister(StringRef Constraint, 283 StringRef Expression) const override { 284 StringRef::iterator I, E; 285 for (I = Constraint.begin(), E = Constraint.end(); I != E; ++I) { 286 if (isalpha(*I) || *I == '@') 287 break; 288 } 289 if (I == E) 290 return ""; 291 switch (*I) { 292 // For the register constraints, return the matching register name 293 case 'a': 294 return "ax"; 295 case 'b': 296 return "bx"; 297 case 'c': 298 return "cx"; 299 case 'd': 300 return "dx"; 301 case 'S': 302 return "si"; 303 case 'D': 304 return "di"; 305 // In case the constraint is 'r' we need to return Expression 306 case 'r': 307 return Expression; 308 // Double letters Y<x> constraints 309 case 'Y': 310 if ((++I != E) && ((*I == '0') || (*I == 'z'))) 311 return "xmm0"; 312 break; 313 default: 314 break; 315 } 316 return ""; 317 } 318 319 bool useFP16ConversionIntrinsics() const override { 320 return false; 321 } 322 323 void getTargetDefines(const LangOptions &Opts, 324 MacroBuilder &Builder) const override; 325 326 void setFeatureEnabled(llvm::StringMap<bool> &Features, StringRef Name, 327 bool Enabled) const final; 328 329 bool 330 initFeatureMap(llvm::StringMap<bool> &Features, DiagnosticsEngine &Diags, 331 StringRef CPU, 332 const std::vector<std::string> &FeaturesVec) const override; 333 334 bool isValidFeatureName(StringRef Name) const override; 335 336 bool hasFeature(StringRef Feature) const final; 337 338 bool handleTargetFeatures(std::vector<std::string> &Features, 339 DiagnosticsEngine &Diags) override; 340 341 StringRef getABI() const override { 342 if (getTriple().getArch() == llvm::Triple::x86_64 && SSELevel >= AVX512F) 343 return "avx512"; 344 if (getTriple().getArch() == llvm::Triple::x86_64 && SSELevel >= AVX) 345 return "avx"; 346 if (getTriple().getArch() == llvm::Triple::x86 && !HasMMX) 347 return "no-mmx"; 348 return ""; 349 } 350 351 bool supportsTargetAttributeTune() const override { 352 return true; 353 } 354 355 bool isValidCPUName(StringRef Name) const override { 356 bool Only64Bit = getTriple().getArch() != llvm::Triple::x86; 357 return llvm::X86::parseArchX86(Name, Only64Bit) != llvm::X86::CK_None; 358 } 359 360 bool isValidTuneCPUName(StringRef Name) const override { 361 if (Name == "generic") 362 return true; 363 364 // Allow 32-bit only CPUs regardless of 64-bit mode unlike isValidCPUName. 365 // NOTE: gcc rejects 32-bit mtune CPUs in 64-bit mode. But being lenient 366 // since mtune was ignored by clang for so long. 367 return llvm::X86::parseTuneCPU(Name) != llvm::X86::CK_None; 368 } 369 370 void fillValidCPUList(SmallVectorImpl<StringRef> &Values) const override; 371 void fillValidTuneCPUList(SmallVectorImpl<StringRef> &Values) const override; 372 373 bool setCPU(const std::string &Name) override { 374 bool Only64Bit = getTriple().getArch() != llvm::Triple::x86; 375 CPU = llvm::X86::parseArchX86(Name, Only64Bit); 376 return CPU != llvm::X86::CK_None; 377 } 378 379 unsigned multiVersionSortPriority(StringRef Name) const override; 380 381 bool setFPMath(StringRef Name) override; 382 383 bool supportsExtendIntArgs() const override { 384 return getTriple().getArch() != llvm::Triple::x86; 385 } 386 387 CallingConvCheckResult checkCallingConvention(CallingConv CC) const override { 388 // Most of the non-ARM calling conventions are i386 conventions. 389 switch (CC) { 390 case CC_X86ThisCall: 391 case CC_X86FastCall: 392 case CC_X86StdCall: 393 case CC_X86VectorCall: 394 case CC_X86RegCall: 395 case CC_C: 396 case CC_PreserveMost: 397 case CC_Swift: 398 case CC_X86Pascal: 399 case CC_IntelOclBicc: 400 case CC_OpenCLKernel: 401 return CCCR_OK; 402 case CC_SwiftAsync: 403 return CCCR_Error; 404 default: 405 return CCCR_Warning; 406 } 407 } 408 409 bool checkArithmeticFenceSupported() const override { return true; } 410 411 CallingConv getDefaultCallingConv() const override { 412 return CC_C; 413 } 414 415 bool hasSjLjLowering() const override { return true; } 416 417 void setSupportedOpenCLOpts() override { supportAllOpenCLOpts(); } 418 419 uint64_t getPointerWidthV(LangAS AS) const override { 420 unsigned TargetAddrSpace = getTargetAddressSpace(AS); 421 if (TargetAddrSpace == ptr32_sptr || TargetAddrSpace == ptr32_uptr) 422 return 32; 423 if (TargetAddrSpace == ptr64) 424 return 64; 425 return PointerWidth; 426 } 427 428 uint64_t getPointerAlignV(LangAS AddrSpace) const override { 429 return getPointerWidthV(AddrSpace); 430 } 431 432 }; 433 434 // X86-32 generic target 435 class LLVM_LIBRARY_VISIBILITY X86_32TargetInfo : public X86TargetInfo { 436 public: 437 X86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 438 : X86TargetInfo(Triple, Opts) { 439 DoubleAlign = LongLongAlign = 32; 440 LongDoubleWidth = 96; 441 LongDoubleAlign = 32; 442 SuitableAlign = 128; 443 resetDataLayout(Triple.isOSBinFormatMachO() 444 ? "e-m:o-p:32:32-p270:32:32-p271:32:32-p272:64:64-i128:" 445 "128-f64:32:64-f80:32-n8:16:32-S128" 446 : "e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-i128:" 447 "128-f64:32:64-f80:32-n8:16:32-S128", 448 Triple.isOSBinFormatMachO() ? "_" : ""); 449 SizeType = UnsignedInt; 450 PtrDiffType = SignedInt; 451 IntPtrType = SignedInt; 452 RegParmMax = 3; 453 454 // Use fpret for all types. 455 RealTypeUsesObjCFPRetMask = 456 (unsigned)(FloatModeKind::Float | FloatModeKind::Double | 457 FloatModeKind::LongDouble); 458 459 // x86-32 has atomics up to 8 bytes 460 MaxAtomicPromoteWidth = 64; 461 MaxAtomicInlineWidth = 32; 462 } 463 464 BuiltinVaListKind getBuiltinVaListKind() const override { 465 return TargetInfo::CharPtrBuiltinVaList; 466 } 467 468 int getEHDataRegisterNumber(unsigned RegNo) const override { 469 if (RegNo == 0) 470 return 0; 471 if (RegNo == 1) 472 return 2; 473 return -1; 474 } 475 476 bool validateOperandSize(const llvm::StringMap<bool> &FeatureMap, 477 StringRef Constraint, unsigned Size) const override { 478 switch (Constraint[0]) { 479 default: 480 break; 481 case 'R': 482 case 'q': 483 case 'Q': 484 case 'a': 485 case 'b': 486 case 'c': 487 case 'd': 488 case 'S': 489 case 'D': 490 return Size <= 32; 491 case 'A': 492 return Size <= 64; 493 } 494 495 return X86TargetInfo::validateOperandSize(FeatureMap, Constraint, Size); 496 } 497 498 void setMaxAtomicWidth() override { 499 if (hasFeature("cx8")) 500 MaxAtomicInlineWidth = 64; 501 } 502 503 ArrayRef<Builtin::Info> getTargetBuiltins() const override; 504 505 bool hasBitIntType() const override { return true; } 506 size_t getMaxBitIntWidth() const override { 507 return llvm::IntegerType::MAX_INT_BITS; 508 } 509 }; 510 511 class LLVM_LIBRARY_VISIBILITY NetBSDI386TargetInfo 512 : public NetBSDTargetInfo<X86_32TargetInfo> { 513 public: 514 NetBSDI386TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 515 : NetBSDTargetInfo<X86_32TargetInfo>(Triple, Opts) {} 516 }; 517 518 class LLVM_LIBRARY_VISIBILITY OpenBSDI386TargetInfo 519 : public OpenBSDTargetInfo<X86_32TargetInfo> { 520 public: 521 OpenBSDI386TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 522 : OpenBSDTargetInfo<X86_32TargetInfo>(Triple, Opts) { 523 SizeType = UnsignedLong; 524 IntPtrType = SignedLong; 525 PtrDiffType = SignedLong; 526 } 527 }; 528 529 class LLVM_LIBRARY_VISIBILITY DarwinI386TargetInfo 530 : public DarwinTargetInfo<X86_32TargetInfo> { 531 public: 532 DarwinI386TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 533 : DarwinTargetInfo<X86_32TargetInfo>(Triple, Opts) { 534 LongDoubleWidth = 128; 535 LongDoubleAlign = 128; 536 SuitableAlign = 128; 537 MaxVectorAlign = 256; 538 // The watchOS simulator uses the builtin bool type for Objective-C. 539 llvm::Triple T = llvm::Triple(Triple); 540 if (T.isWatchOS()) 541 UseSignedCharForObjCBool = false; 542 SizeType = UnsignedLong; 543 IntPtrType = SignedLong; 544 resetDataLayout("e-m:o-p:32:32-p270:32:32-p271:32:32-p272:64:64-i128:128-" 545 "f64:32:64-f80:128-n8:16:32-S128", 546 "_"); 547 HasAlignMac68kSupport = true; 548 } 549 550 bool handleTargetFeatures(std::vector<std::string> &Features, 551 DiagnosticsEngine &Diags) override { 552 if (!DarwinTargetInfo<X86_32TargetInfo>::handleTargetFeatures(Features, 553 Diags)) 554 return false; 555 // We now know the features we have: we can decide how to align vectors. 556 MaxVectorAlign = 557 hasFeature("avx512f") ? 512 : hasFeature("avx") ? 256 : 128; 558 return true; 559 } 560 }; 561 562 // x86-32 Windows target 563 class LLVM_LIBRARY_VISIBILITY WindowsX86_32TargetInfo 564 : public WindowsTargetInfo<X86_32TargetInfo> { 565 public: 566 WindowsX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 567 : WindowsTargetInfo<X86_32TargetInfo>(Triple, Opts) { 568 DoubleAlign = LongLongAlign = 64; 569 bool IsWinCOFF = 570 getTriple().isOSWindows() && getTriple().isOSBinFormatCOFF(); 571 bool IsMSVC = getTriple().isWindowsMSVCEnvironment(); 572 std::string Layout = IsWinCOFF ? "e-m:x" : "e-m:e"; 573 Layout += "-p:32:32-p270:32:32-p271:32:32-p272:64:64-i64:64-i128:128-"; 574 Layout += IsMSVC ? "f80:128" : "f80:32"; 575 Layout += "-n8:16:32-a:0:32-S32"; 576 resetDataLayout(Layout, IsWinCOFF ? "_" : ""); 577 } 578 }; 579 580 // x86-32 Windows Visual Studio target 581 class LLVM_LIBRARY_VISIBILITY MicrosoftX86_32TargetInfo 582 : public WindowsX86_32TargetInfo { 583 public: 584 MicrosoftX86_32TargetInfo(const llvm::Triple &Triple, 585 const TargetOptions &Opts) 586 : WindowsX86_32TargetInfo(Triple, Opts) { 587 LongDoubleWidth = LongDoubleAlign = 64; 588 LongDoubleFormat = &llvm::APFloat::IEEEdouble(); 589 } 590 591 void getTargetDefines(const LangOptions &Opts, 592 MacroBuilder &Builder) const override { 593 WindowsX86_32TargetInfo::getTargetDefines(Opts, Builder); 594 // The value of the following reflects processor type. 595 // 300=386, 400=486, 500=Pentium, 600=Blend (default) 596 // We lost the original triple, so we use the default. 597 Builder.defineMacro("_M_IX86", "600"); 598 } 599 }; 600 601 // x86-32 MinGW target 602 class LLVM_LIBRARY_VISIBILITY MinGWX86_32TargetInfo 603 : public WindowsX86_32TargetInfo { 604 public: 605 MinGWX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 606 : WindowsX86_32TargetInfo(Triple, Opts) { 607 HasFloat128 = true; 608 } 609 610 void getTargetDefines(const LangOptions &Opts, 611 MacroBuilder &Builder) const override { 612 WindowsX86_32TargetInfo::getTargetDefines(Opts, Builder); 613 Builder.defineMacro("_X86_"); 614 } 615 }; 616 617 // x86-32 Cygwin target 618 class LLVM_LIBRARY_VISIBILITY CygwinX86_32TargetInfo : public X86_32TargetInfo { 619 public: 620 CygwinX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 621 : X86_32TargetInfo(Triple, Opts) { 622 this->WCharType = TargetInfo::UnsignedShort; 623 DoubleAlign = LongLongAlign = 64; 624 resetDataLayout("e-m:x-p:32:32-p270:32:32-p271:32:32-p272:64:64-i64:64-" 625 "i128:128-f80:32-n8:16:32-a:0:32-S32", 626 "_"); 627 } 628 629 void getTargetDefines(const LangOptions &Opts, 630 MacroBuilder &Builder) const override { 631 X86_32TargetInfo::getTargetDefines(Opts, Builder); 632 Builder.defineMacro("_X86_"); 633 Builder.defineMacro("__CYGWIN__"); 634 Builder.defineMacro("__CYGWIN32__"); 635 addCygMingDefines(Opts, Builder); 636 DefineStd(Builder, "unix", Opts); 637 if (Opts.CPlusPlus) 638 Builder.defineMacro("_GNU_SOURCE"); 639 } 640 }; 641 642 // x86-32 Haiku target 643 class LLVM_LIBRARY_VISIBILITY HaikuX86_32TargetInfo 644 : public HaikuTargetInfo<X86_32TargetInfo> { 645 public: 646 HaikuX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 647 : HaikuTargetInfo<X86_32TargetInfo>(Triple, Opts) {} 648 649 void getTargetDefines(const LangOptions &Opts, 650 MacroBuilder &Builder) const override { 651 HaikuTargetInfo<X86_32TargetInfo>::getTargetDefines(Opts, Builder); 652 Builder.defineMacro("__INTEL__"); 653 } 654 }; 655 656 // X86-32 MCU target 657 class LLVM_LIBRARY_VISIBILITY MCUX86_32TargetInfo : public X86_32TargetInfo { 658 public: 659 MCUX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 660 : X86_32TargetInfo(Triple, Opts) { 661 LongDoubleWidth = 64; 662 DefaultAlignForAttributeAligned = 32; 663 LongDoubleFormat = &llvm::APFloat::IEEEdouble(); 664 resetDataLayout("e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-i64:32-" 665 "f64:32-f128:32-n8:16:32-a:0:32-S32"); 666 WIntType = UnsignedInt; 667 } 668 669 CallingConvCheckResult checkCallingConvention(CallingConv CC) const override { 670 // On MCU we support only C calling convention. 671 return CC == CC_C ? CCCR_OK : CCCR_Warning; 672 } 673 674 void getTargetDefines(const LangOptions &Opts, 675 MacroBuilder &Builder) const override { 676 X86_32TargetInfo::getTargetDefines(Opts, Builder); 677 Builder.defineMacro("__iamcu"); 678 Builder.defineMacro("__iamcu__"); 679 } 680 681 bool allowsLargerPreferedTypeAlignment() const override { return false; } 682 }; 683 684 // x86-32 RTEMS target 685 class LLVM_LIBRARY_VISIBILITY RTEMSX86_32TargetInfo : public X86_32TargetInfo { 686 public: 687 RTEMSX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 688 : X86_32TargetInfo(Triple, Opts) { 689 SizeType = UnsignedLong; 690 IntPtrType = SignedLong; 691 PtrDiffType = SignedLong; 692 } 693 694 void getTargetDefines(const LangOptions &Opts, 695 MacroBuilder &Builder) const override { 696 X86_32TargetInfo::getTargetDefines(Opts, Builder); 697 Builder.defineMacro("__INTEL__"); 698 Builder.defineMacro("__rtems__"); 699 } 700 }; 701 702 // x86-64 generic target 703 class LLVM_LIBRARY_VISIBILITY X86_64TargetInfo : public X86TargetInfo { 704 public: 705 X86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 706 : X86TargetInfo(Triple, Opts) { 707 const bool IsX32 = getTriple().isX32(); 708 bool IsWinCOFF = 709 getTriple().isOSWindows() && getTriple().isOSBinFormatCOFF(); 710 LongWidth = LongAlign = PointerWidth = PointerAlign = IsX32 ? 32 : 64; 711 LongDoubleWidth = 128; 712 LongDoubleAlign = 128; 713 LargeArrayMinWidth = 128; 714 LargeArrayAlign = 128; 715 SuitableAlign = 128; 716 SizeType = IsX32 ? UnsignedInt : UnsignedLong; 717 PtrDiffType = IsX32 ? SignedInt : SignedLong; 718 IntPtrType = IsX32 ? SignedInt : SignedLong; 719 IntMaxType = IsX32 ? SignedLongLong : SignedLong; 720 Int64Type = IsX32 ? SignedLongLong : SignedLong; 721 RegParmMax = 6; 722 723 // Pointers are 32-bit in x32. 724 resetDataLayout(IsX32 ? "e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-" 725 "i64:64-i128:128-f80:128-n8:16:32:64-S128" 726 : IsWinCOFF ? "e-m:w-p270:32:32-p271:32:32-p272:64:64-i64:" 727 "64-i128:128-f80:128-n8:16:32:64-S128" 728 : "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:" 729 "64-i128:128-f80:128-n8:16:32:64-S128"); 730 731 // Use fpret only for long double. 732 RealTypeUsesObjCFPRetMask = (unsigned)FloatModeKind::LongDouble; 733 734 // Use fp2ret for _Complex long double. 735 ComplexLongDoubleUsesFP2Ret = true; 736 737 // Make __builtin_ms_va_list available. 738 HasBuiltinMSVaList = true; 739 740 // x86-64 has atomics up to 16 bytes. 741 MaxAtomicPromoteWidth = 128; 742 MaxAtomicInlineWidth = 64; 743 } 744 745 BuiltinVaListKind getBuiltinVaListKind() const override { 746 return TargetInfo::X86_64ABIBuiltinVaList; 747 } 748 749 int getEHDataRegisterNumber(unsigned RegNo) const override { 750 if (RegNo == 0) 751 return 0; 752 if (RegNo == 1) 753 return 1; 754 return -1; 755 } 756 757 CallingConvCheckResult checkCallingConvention(CallingConv CC) const override { 758 switch (CC) { 759 case CC_C: 760 case CC_Swift: 761 case CC_SwiftAsync: 762 case CC_X86VectorCall: 763 case CC_IntelOclBicc: 764 case CC_Win64: 765 case CC_PreserveMost: 766 case CC_PreserveAll: 767 case CC_PreserveNone: 768 case CC_X86RegCall: 769 case CC_OpenCLKernel: 770 return CCCR_OK; 771 default: 772 return CCCR_Warning; 773 } 774 } 775 776 CallingConv getDefaultCallingConv() const override { 777 return CC_C; 778 } 779 780 // for x32 we need it here explicitly 781 bool hasInt128Type() const override { return true; } 782 783 unsigned getUnwindWordWidth() const override { return 64; } 784 785 unsigned getRegisterWidth() const override { return 64; } 786 787 bool validateGlobalRegisterVariable(StringRef RegName, unsigned RegSize, 788 bool &HasSizeMismatch) const override { 789 // rsp and rbp are the only 64-bit registers the x86 backend can currently 790 // handle. 791 if (RegName == "rsp" || RegName == "rbp") { 792 // Check that the register size is 64-bit. 793 HasSizeMismatch = RegSize != 64; 794 return true; 795 } 796 797 // Check if the register is a 32-bit register the backend can handle. 798 return X86TargetInfo::validateGlobalRegisterVariable(RegName, RegSize, 799 HasSizeMismatch); 800 } 801 802 void setMaxAtomicWidth() override { 803 if (hasFeature("cx16")) 804 MaxAtomicInlineWidth = 128; 805 } 806 807 ArrayRef<Builtin::Info> getTargetBuiltins() const override; 808 809 bool hasBitIntType() const override { return true; } 810 size_t getMaxBitIntWidth() const override { 811 return llvm::IntegerType::MAX_INT_BITS; 812 } 813 }; 814 815 // x86-64 Windows target 816 class LLVM_LIBRARY_VISIBILITY WindowsX86_64TargetInfo 817 : public WindowsTargetInfo<X86_64TargetInfo> { 818 public: 819 WindowsX86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 820 : WindowsTargetInfo<X86_64TargetInfo>(Triple, Opts) { 821 LongWidth = LongAlign = 32; 822 DoubleAlign = LongLongAlign = 64; 823 IntMaxType = SignedLongLong; 824 Int64Type = SignedLongLong; 825 SizeType = UnsignedLongLong; 826 PtrDiffType = SignedLongLong; 827 IntPtrType = SignedLongLong; 828 } 829 830 BuiltinVaListKind getBuiltinVaListKind() const override { 831 return TargetInfo::CharPtrBuiltinVaList; 832 } 833 834 CallingConvCheckResult checkCallingConvention(CallingConv CC) const override { 835 switch (CC) { 836 case CC_X86StdCall: 837 case CC_X86ThisCall: 838 case CC_X86FastCall: 839 return CCCR_Ignore; 840 case CC_C: 841 case CC_X86VectorCall: 842 case CC_IntelOclBicc: 843 case CC_PreserveMost: 844 case CC_PreserveAll: 845 case CC_PreserveNone: 846 case CC_X86_64SysV: 847 case CC_Swift: 848 case CC_SwiftAsync: 849 case CC_X86RegCall: 850 case CC_OpenCLKernel: 851 return CCCR_OK; 852 default: 853 return CCCR_Warning; 854 } 855 } 856 }; 857 858 // x86-64 Windows Visual Studio target 859 class LLVM_LIBRARY_VISIBILITY MicrosoftX86_64TargetInfo 860 : public WindowsX86_64TargetInfo { 861 public: 862 MicrosoftX86_64TargetInfo(const llvm::Triple &Triple, 863 const TargetOptions &Opts) 864 : WindowsX86_64TargetInfo(Triple, Opts) { 865 LongDoubleWidth = LongDoubleAlign = 64; 866 LongDoubleFormat = &llvm::APFloat::IEEEdouble(); 867 } 868 869 void getTargetDefines(const LangOptions &Opts, 870 MacroBuilder &Builder) const override { 871 WindowsX86_64TargetInfo::getTargetDefines(Opts, Builder); 872 Builder.defineMacro("_M_X64", "100"); 873 Builder.defineMacro("_M_AMD64", "100"); 874 } 875 876 TargetInfo::CallingConvKind 877 getCallingConvKind(bool ClangABICompat4) const override { 878 return CCK_MicrosoftWin64; 879 } 880 }; 881 882 // x86-64 MinGW target 883 class LLVM_LIBRARY_VISIBILITY MinGWX86_64TargetInfo 884 : public WindowsX86_64TargetInfo { 885 public: 886 MinGWX86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 887 : WindowsX86_64TargetInfo(Triple, Opts) { 888 // Mingw64 rounds long double size and alignment up to 16 bytes, but sticks 889 // with x86 FP ops. Weird. 890 LongDoubleWidth = LongDoubleAlign = 128; 891 LongDoubleFormat = &llvm::APFloat::x87DoubleExtended(); 892 HasFloat128 = true; 893 } 894 }; 895 896 // x86-64 Cygwin target 897 class LLVM_LIBRARY_VISIBILITY CygwinX86_64TargetInfo : public X86_64TargetInfo { 898 public: 899 CygwinX86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 900 : X86_64TargetInfo(Triple, Opts) { 901 this->WCharType = TargetInfo::UnsignedShort; 902 TLSSupported = false; 903 } 904 905 void getTargetDefines(const LangOptions &Opts, 906 MacroBuilder &Builder) const override { 907 X86_64TargetInfo::getTargetDefines(Opts, Builder); 908 Builder.defineMacro("__x86_64__"); 909 Builder.defineMacro("__CYGWIN__"); 910 Builder.defineMacro("__CYGWIN64__"); 911 addCygMingDefines(Opts, Builder); 912 DefineStd(Builder, "unix", Opts); 913 if (Opts.CPlusPlus) 914 Builder.defineMacro("_GNU_SOURCE"); 915 } 916 }; 917 918 class LLVM_LIBRARY_VISIBILITY DarwinX86_64TargetInfo 919 : public DarwinTargetInfo<X86_64TargetInfo> { 920 public: 921 DarwinX86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 922 : DarwinTargetInfo<X86_64TargetInfo>(Triple, Opts) { 923 Int64Type = SignedLongLong; 924 // The 64-bit iOS simulator uses the builtin bool type for Objective-C. 925 llvm::Triple T = llvm::Triple(Triple); 926 if (T.isiOS()) 927 UseSignedCharForObjCBool = false; 928 resetDataLayout("e-m:o-p270:32:32-p271:32:32-p272:64:64-i64:64-i128:128-" 929 "f80:128-n8:16:32:64-S128", 930 "_"); 931 } 932 933 bool handleTargetFeatures(std::vector<std::string> &Features, 934 DiagnosticsEngine &Diags) override { 935 if (!DarwinTargetInfo<X86_64TargetInfo>::handleTargetFeatures(Features, 936 Diags)) 937 return false; 938 // We now know the features we have: we can decide how to align vectors. 939 MaxVectorAlign = 940 hasFeature("avx512f") ? 512 : hasFeature("avx") ? 256 : 128; 941 return true; 942 } 943 }; 944 945 class LLVM_LIBRARY_VISIBILITY OpenBSDX86_64TargetInfo 946 : public OpenBSDTargetInfo<X86_64TargetInfo> { 947 public: 948 OpenBSDX86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 949 : OpenBSDTargetInfo<X86_64TargetInfo>(Triple, Opts) { 950 IntMaxType = SignedLongLong; 951 Int64Type = SignedLongLong; 952 } 953 }; 954 955 // x86_32 Android target 956 class LLVM_LIBRARY_VISIBILITY AndroidX86_32TargetInfo 957 : public LinuxTargetInfo<X86_32TargetInfo> { 958 public: 959 AndroidX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 960 : LinuxTargetInfo<X86_32TargetInfo>(Triple, Opts) { 961 SuitableAlign = 32; 962 LongDoubleWidth = 64; 963 LongDoubleFormat = &llvm::APFloat::IEEEdouble(); 964 } 965 }; 966 967 // x86_64 Android target 968 class LLVM_LIBRARY_VISIBILITY AndroidX86_64TargetInfo 969 : public LinuxTargetInfo<X86_64TargetInfo> { 970 public: 971 AndroidX86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 972 : LinuxTargetInfo<X86_64TargetInfo>(Triple, Opts) { 973 LongDoubleFormat = &llvm::APFloat::IEEEquad(); 974 } 975 }; 976 977 // x86_32 OHOS target 978 class LLVM_LIBRARY_VISIBILITY OHOSX86_32TargetInfo 979 : public OHOSTargetInfo<X86_32TargetInfo> { 980 public: 981 OHOSX86_32TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 982 : OHOSTargetInfo<X86_32TargetInfo>(Triple, Opts) { 983 SuitableAlign = 32; 984 LongDoubleWidth = 64; 985 LongDoubleFormat = &llvm::APFloat::IEEEdouble(); 986 } 987 }; 988 989 // x86_64 OHOS target 990 class LLVM_LIBRARY_VISIBILITY OHOSX86_64TargetInfo 991 : public OHOSTargetInfo<X86_64TargetInfo> { 992 public: 993 OHOSX86_64TargetInfo(const llvm::Triple &Triple, const TargetOptions &Opts) 994 : OHOSTargetInfo<X86_64TargetInfo>(Triple, Opts) { 995 LongDoubleFormat = &llvm::APFloat::IEEEquad(); 996 } 997 }; 998 } // namespace targets 999 } // namespace clang 1000 #endif // LLVM_CLANG_LIB_BASIC_TARGETS_X86_H 1001