1 //===--- NVPTX.cpp - Implement NVPTX target feature support ---------------===// 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 implements NVPTX TargetInfo objects. 10 // 11 //===----------------------------------------------------------------------===// 12 13 #include "NVPTX.h" 14 #include "Targets.h" 15 #include "clang/Basic/Builtins.h" 16 #include "clang/Basic/MacroBuilder.h" 17 #include "clang/Basic/TargetBuiltins.h" 18 #include "llvm/ADT/StringSwitch.h" 19 20 using namespace clang; 21 using namespace clang::targets; 22 23 const Builtin::Info NVPTXTargetInfo::BuiltinInfo[] = { 24 #define BUILTIN(ID, TYPE, ATTRS) \ 25 {#ID, TYPE, ATTRS, nullptr, ALL_LANGUAGES, nullptr}, 26 #define LIBBUILTIN(ID, TYPE, ATTRS, HEADER) \ 27 {#ID, TYPE, ATTRS, HEADER, ALL_LANGUAGES, nullptr}, 28 #define TARGET_BUILTIN(ID, TYPE, ATTRS, FEATURE) \ 29 {#ID, TYPE, ATTRS, nullptr, ALL_LANGUAGES, FEATURE}, 30 #include "clang/Basic/BuiltinsNVPTX.def" 31 }; 32 33 const char *const NVPTXTargetInfo::GCCRegNames[] = {"r0"}; 34 35 NVPTXTargetInfo::NVPTXTargetInfo(const llvm::Triple &Triple, 36 const TargetOptions &Opts, 37 unsigned TargetPointerWidth) 38 : TargetInfo(Triple) { 39 assert((TargetPointerWidth == 32 || TargetPointerWidth == 64) && 40 "NVPTX only supports 32- and 64-bit modes."); 41 42 PTXVersion = 32; 43 for (const StringRef Feature : Opts.FeaturesAsWritten) { 44 if (!Feature.startswith("+ptx")) 45 continue; 46 PTXVersion = llvm::StringSwitch<unsigned>(Feature) 47 .Case("+ptx75", 75) 48 .Case("+ptx74", 74) 49 .Case("+ptx73", 73) 50 .Case("+ptx72", 72) 51 .Case("+ptx71", 71) 52 .Case("+ptx70", 70) 53 .Case("+ptx65", 65) 54 .Case("+ptx64", 64) 55 .Case("+ptx63", 63) 56 .Case("+ptx61", 61) 57 .Case("+ptx60", 60) 58 .Case("+ptx50", 50) 59 .Case("+ptx43", 43) 60 .Case("+ptx42", 42) 61 .Case("+ptx41", 41) 62 .Case("+ptx40", 40) 63 .Case("+ptx32", 32) 64 .Default(32); 65 } 66 67 TLSSupported = false; 68 VLASupported = false; 69 AddrSpaceMap = &NVPTXAddrSpaceMap; 70 UseAddrSpaceMapMangling = true; 71 72 // Define available target features 73 // These must be defined in sorted order! 74 NoAsmVariants = true; 75 GPU = CudaArch::SM_20; 76 77 if (TargetPointerWidth == 32) 78 resetDataLayout("e-p:32:32-i64:64-i128:128-v16:16-v32:32-n16:32:64"); 79 else if (Opts.NVPTXUseShortPointers) 80 resetDataLayout( 81 "e-p3:32:32-p4:32:32-p5:32:32-i64:64-i128:128-v16:16-v32:32-n16:32:64"); 82 else 83 resetDataLayout("e-i64:64-i128:128-v16:16-v32:32-n16:32:64"); 84 85 // If possible, get a TargetInfo for our host triple, so we can match its 86 // types. 87 llvm::Triple HostTriple(Opts.HostTriple); 88 if (!HostTriple.isNVPTX()) 89 HostTarget.reset(AllocateTarget(llvm::Triple(Opts.HostTriple), Opts)); 90 91 // If no host target, make some guesses about the data layout and return. 92 if (!HostTarget) { 93 LongWidth = LongAlign = TargetPointerWidth; 94 PointerWidth = PointerAlign = TargetPointerWidth; 95 switch (TargetPointerWidth) { 96 case 32: 97 SizeType = TargetInfo::UnsignedInt; 98 PtrDiffType = TargetInfo::SignedInt; 99 IntPtrType = TargetInfo::SignedInt; 100 break; 101 case 64: 102 SizeType = TargetInfo::UnsignedLong; 103 PtrDiffType = TargetInfo::SignedLong; 104 IntPtrType = TargetInfo::SignedLong; 105 break; 106 default: 107 llvm_unreachable("TargetPointerWidth must be 32 or 64"); 108 } 109 return; 110 } 111 112 // Copy properties from host target. 113 PointerWidth = HostTarget->getPointerWidth(/* AddrSpace = */ 0); 114 PointerAlign = HostTarget->getPointerAlign(/* AddrSpace = */ 0); 115 BoolWidth = HostTarget->getBoolWidth(); 116 BoolAlign = HostTarget->getBoolAlign(); 117 IntWidth = HostTarget->getIntWidth(); 118 IntAlign = HostTarget->getIntAlign(); 119 HalfWidth = HostTarget->getHalfWidth(); 120 HalfAlign = HostTarget->getHalfAlign(); 121 FloatWidth = HostTarget->getFloatWidth(); 122 FloatAlign = HostTarget->getFloatAlign(); 123 DoubleWidth = HostTarget->getDoubleWidth(); 124 DoubleAlign = HostTarget->getDoubleAlign(); 125 LongWidth = HostTarget->getLongWidth(); 126 LongAlign = HostTarget->getLongAlign(); 127 LongLongWidth = HostTarget->getLongLongWidth(); 128 LongLongAlign = HostTarget->getLongLongAlign(); 129 MinGlobalAlign = HostTarget->getMinGlobalAlign(/* TypeSize = */ 0); 130 NewAlign = HostTarget->getNewAlign(); 131 DefaultAlignForAttributeAligned = 132 HostTarget->getDefaultAlignForAttributeAligned(); 133 SizeType = HostTarget->getSizeType(); 134 IntMaxType = HostTarget->getIntMaxType(); 135 PtrDiffType = HostTarget->getPtrDiffType(/* AddrSpace = */ 0); 136 IntPtrType = HostTarget->getIntPtrType(); 137 WCharType = HostTarget->getWCharType(); 138 WIntType = HostTarget->getWIntType(); 139 Char16Type = HostTarget->getChar16Type(); 140 Char32Type = HostTarget->getChar32Type(); 141 Int64Type = HostTarget->getInt64Type(); 142 SigAtomicType = HostTarget->getSigAtomicType(); 143 ProcessIDType = HostTarget->getProcessIDType(); 144 145 UseBitFieldTypeAlignment = HostTarget->useBitFieldTypeAlignment(); 146 UseZeroLengthBitfieldAlignment = HostTarget->useZeroLengthBitfieldAlignment(); 147 UseExplicitBitFieldAlignment = HostTarget->useExplicitBitFieldAlignment(); 148 ZeroLengthBitfieldBoundary = HostTarget->getZeroLengthBitfieldBoundary(); 149 150 // This is a bit of a lie, but it controls __GCC_ATOMIC_XXX_LOCK_FREE, and 151 // we need those macros to be identical on host and device, because (among 152 // other things) they affect which standard library classes are defined, and 153 // we need all classes to be defined on both the host and device. 154 MaxAtomicInlineWidth = HostTarget->getMaxAtomicInlineWidth(); 155 156 // Properties intentionally not copied from host: 157 // - LargeArrayMinWidth, LargeArrayAlign: Not visible across the 158 // host/device boundary. 159 // - SuitableAlign: Not visible across the host/device boundary, and may 160 // correctly be different on host/device, e.g. if host has wider vector 161 // types than device. 162 // - LongDoubleWidth, LongDoubleAlign: nvptx's long double type is the same 163 // as its double type, but that's not necessarily true on the host. 164 // TODO: nvcc emits a warning when using long double on device; we should 165 // do the same. 166 } 167 168 ArrayRef<const char *> NVPTXTargetInfo::getGCCRegNames() const { 169 return llvm::makeArrayRef(GCCRegNames); 170 } 171 172 bool NVPTXTargetInfo::hasFeature(StringRef Feature) const { 173 return llvm::StringSwitch<bool>(Feature) 174 .Cases("ptx", "nvptx", true) 175 .Default(false); 176 } 177 178 void NVPTXTargetInfo::getTargetDefines(const LangOptions &Opts, 179 MacroBuilder &Builder) const { 180 Builder.defineMacro("__PTX__"); 181 Builder.defineMacro("__NVPTX__"); 182 if (Opts.CUDAIsDevice) { 183 // Set __CUDA_ARCH__ for the GPU specified. 184 std::string CUDAArchCode = [this] { 185 switch (GPU) { 186 case CudaArch::GFX600: 187 case CudaArch::GFX601: 188 case CudaArch::GFX602: 189 case CudaArch::GFX700: 190 case CudaArch::GFX701: 191 case CudaArch::GFX702: 192 case CudaArch::GFX703: 193 case CudaArch::GFX704: 194 case CudaArch::GFX705: 195 case CudaArch::GFX801: 196 case CudaArch::GFX802: 197 case CudaArch::GFX803: 198 case CudaArch::GFX805: 199 case CudaArch::GFX810: 200 case CudaArch::GFX900: 201 case CudaArch::GFX902: 202 case CudaArch::GFX904: 203 case CudaArch::GFX906: 204 case CudaArch::GFX908: 205 case CudaArch::GFX909: 206 case CudaArch::GFX90a: 207 case CudaArch::GFX90c: 208 case CudaArch::GFX1010: 209 case CudaArch::GFX1011: 210 case CudaArch::GFX1012: 211 case CudaArch::GFX1013: 212 case CudaArch::GFX1030: 213 case CudaArch::GFX1031: 214 case CudaArch::GFX1032: 215 case CudaArch::GFX1033: 216 case CudaArch::GFX1034: 217 case CudaArch::GFX1035: 218 case CudaArch::Generic: 219 case CudaArch::LAST: 220 break; 221 case CudaArch::UNUSED: 222 case CudaArch::UNKNOWN: 223 assert(false && "No GPU arch when compiling CUDA device code."); 224 return ""; 225 case CudaArch::SM_20: 226 return "200"; 227 case CudaArch::SM_21: 228 return "210"; 229 case CudaArch::SM_30: 230 return "300"; 231 case CudaArch::SM_32: 232 return "320"; 233 case CudaArch::SM_35: 234 return "350"; 235 case CudaArch::SM_37: 236 return "370"; 237 case CudaArch::SM_50: 238 return "500"; 239 case CudaArch::SM_52: 240 return "520"; 241 case CudaArch::SM_53: 242 return "530"; 243 case CudaArch::SM_60: 244 return "600"; 245 case CudaArch::SM_61: 246 return "610"; 247 case CudaArch::SM_62: 248 return "620"; 249 case CudaArch::SM_70: 250 return "700"; 251 case CudaArch::SM_72: 252 return "720"; 253 case CudaArch::SM_75: 254 return "750"; 255 case CudaArch::SM_80: 256 return "800"; 257 case CudaArch::SM_86: 258 return "860"; 259 } 260 llvm_unreachable("unhandled CudaArch"); 261 }(); 262 Builder.defineMacro("__CUDA_ARCH__", CUDAArchCode); 263 } 264 } 265 266 ArrayRef<Builtin::Info> NVPTXTargetInfo::getTargetBuiltins() const { 267 return llvm::makeArrayRef(BuiltinInfo, clang::NVPTX::LastTSBuiltin - 268 Builtin::FirstTSBuiltin); 269 } 270