1 //===--- TargetCXXABI.h - C++ ABI Target Configuration ----------*- 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 /// \file 10 /// Defines the TargetCXXABI class, which abstracts details of the 11 /// C++ ABI that we're targeting. 12 /// 13 //===----------------------------------------------------------------------===// 14 15 #ifndef LLVM_CLANG_BASIC_TARGETCXXABI_H 16 #define LLVM_CLANG_BASIC_TARGETCXXABI_H 17 18 #include "clang/Basic/LLVM.h" 19 #include "llvm/Support/ErrorHandling.h" 20 21 namespace clang { 22 23 /// The basic abstraction for the target C++ ABI. 24 class TargetCXXABI { 25 public: 26 /// The basic C++ ABI kind. 27 enum Kind { 28 /// The generic Itanium ABI is the standard ABI of most open-source 29 /// and Unix-like platforms. It is the primary ABI targeted by 30 /// many compilers, including Clang and GCC. 31 /// 32 /// It is documented here: 33 /// http://www.codesourcery.com/public/cxx-abi/ 34 GenericItanium, 35 36 /// The generic ARM ABI is a modified version of the Itanium ABI 37 /// proposed by ARM for use on ARM-based platforms. 38 /// 39 /// These changes include: 40 /// - the representation of member function pointers is adjusted 41 /// to not conflict with the 'thumb' bit of ARM function pointers; 42 /// - constructors and destructors return 'this'; 43 /// - guard variables are smaller; 44 /// - inline functions are never key functions; 45 /// - array cookies have a slightly different layout; 46 /// - additional convenience functions are specified; 47 /// - and more! 48 /// 49 /// It is documented here: 50 /// http://infocenter.arm.com 51 /// /help/topic/com.arm.doc.ihi0041c/IHI0041C_cppabi.pdf 52 GenericARM, 53 54 /// The iOS ABI is a partial implementation of the ARM ABI. 55 /// Several of the features of the ARM ABI were not fully implemented 56 /// in the compilers that iOS was launched with. 57 /// 58 /// Essentially, the iOS ABI includes the ARM changes to: 59 /// - member function pointers, 60 /// - guard variables, 61 /// - array cookies, and 62 /// - constructor/destructor signatures. 63 iOS, 64 65 /// The iOS 64-bit and macOS 64-bit ARM ABI follows ARM's published 64-bit 66 /// ABI more closely, but we don't guarantee to follow it perfectly. 67 /// 68 /// It is documented here: 69 /// http://infocenter.arm.com 70 /// /help/topic/com.arm.doc.ihi0059a/IHI0059A_cppabi64.pdf 71 AppleARM64, 72 73 /// WatchOS is a modernisation of the iOS ABI, which roughly means it's 74 /// the AppleARM64 ABI ported to 32-bits. The primary difference from 75 /// AppleARM64 is that RTTI objects must still be unique at the moment. 76 WatchOS, 77 78 /// The generic AArch64 ABI is also a modified version of the Itanium ABI, 79 /// but it has fewer divergences than the 32-bit ARM ABI. 80 /// 81 /// The relevant changes from the generic ABI in this case are: 82 /// - representation of member function pointers adjusted as in ARM. 83 /// - guard variables are smaller. 84 GenericAArch64, 85 86 /// The generic Mips ABI is a modified version of the Itanium ABI. 87 /// 88 /// At the moment, only change from the generic ABI in this case is: 89 /// - representation of member function pointers adjusted as in ARM. 90 GenericMIPS, 91 92 /// The WebAssembly ABI is a modified version of the Itanium ABI. 93 /// 94 /// The changes from the Itanium ABI are: 95 /// - representation of member function pointers is adjusted, as in ARM; 96 /// - member functions are not specially aligned; 97 /// - constructors and destructors return 'this', as in ARM; 98 /// - guard variables are 32-bit on wasm32, as in ARM; 99 /// - unused bits of guard variables are reserved, as in ARM; 100 /// - inline functions are never key functions, as in ARM; 101 /// - C++11 POD rules are used for tail padding, as in AppleARM64. 102 /// 103 /// TODO: At present the WebAssembly ABI is not considered stable, so none 104 /// of these details is necessarily final yet. 105 WebAssembly, 106 107 /// The Fuchsia ABI is a modified version of the Itanium ABI. 108 /// 109 /// The relevant changes from the Itanium ABI are: 110 /// - constructors and destructors return 'this', as in ARM. 111 Fuchsia, 112 113 /// The XL ABI is the ABI used by IBM xlclang compiler and is a modified 114 /// version of the Itanium ABI. 115 /// 116 /// The relevant changes from the Itanium ABI are: 117 /// - static initialization is adjusted to use sinit and sterm functions; 118 XL, 119 120 /// The Microsoft ABI is the ABI used by Microsoft Visual Studio (and 121 /// compatible compilers). 122 /// 123 /// FIXME: should this be split into Win32 and Win64 variants? 124 /// 125 /// Only scattered and incomplete official documentation exists. 126 Microsoft 127 }; 128 129 private: 130 // Right now, this class is passed around as a cheap value type. 131 // If you add more members, especially non-POD members, please 132 // audit the users to pass it by reference instead. 133 Kind TheKind; 134 135 public: 136 /// A bogus initialization of the platform ABI. TargetCXXABI()137 TargetCXXABI() : TheKind(GenericItanium) {} 138 TargetCXXABI(Kind kind)139 TargetCXXABI(Kind kind) : TheKind(kind) {} 140 set(Kind kind)141 void set(Kind kind) { 142 TheKind = kind; 143 } 144 getKind()145 Kind getKind() const { return TheKind; } 146 147 /// Does this ABI generally fall into the Itanium family of ABIs? isItaniumFamily()148 bool isItaniumFamily() const { 149 switch (getKind()) { 150 case AppleARM64: 151 case Fuchsia: 152 case GenericAArch64: 153 case GenericItanium: 154 case GenericARM: 155 case iOS: 156 case WatchOS: 157 case GenericMIPS: 158 case WebAssembly: 159 case XL: 160 return true; 161 162 case Microsoft: 163 return false; 164 } 165 llvm_unreachable("bad ABI kind"); 166 } 167 168 /// Is this ABI an MSVC-compatible ABI? isMicrosoft()169 bool isMicrosoft() const { 170 switch (getKind()) { 171 case AppleARM64: 172 case Fuchsia: 173 case GenericAArch64: 174 case GenericItanium: 175 case GenericARM: 176 case iOS: 177 case WatchOS: 178 case GenericMIPS: 179 case WebAssembly: 180 case XL: 181 return false; 182 183 case Microsoft: 184 return true; 185 } 186 llvm_unreachable("bad ABI kind"); 187 } 188 189 /// Are member functions differently aligned? 190 /// 191 /// Many Itanium-style C++ ABIs require member functions to be aligned, so 192 /// that a pointer to such a function is guaranteed to have a zero in the 193 /// least significant bit, so that pointers to member functions can use that 194 /// bit to distinguish between virtual and non-virtual functions. However, 195 /// some Itanium-style C++ ABIs differentiate between virtual and non-virtual 196 /// functions via other means, and consequently don't require that member 197 /// functions be aligned. areMemberFunctionsAligned()198 bool areMemberFunctionsAligned() const { 199 switch (getKind()) { 200 case WebAssembly: 201 // WebAssembly doesn't require any special alignment for member functions. 202 return false; 203 case AppleARM64: 204 case Fuchsia: 205 case GenericARM: 206 case GenericAArch64: 207 case GenericMIPS: 208 // TODO: ARM-style pointers to member functions put the discriminator in 209 // the this adjustment, so they don't require functions to have any 210 // special alignment and could therefore also return false. 211 case GenericItanium: 212 case iOS: 213 case WatchOS: 214 case Microsoft: 215 case XL: 216 return true; 217 } 218 llvm_unreachable("bad ABI kind"); 219 } 220 221 /// Are arguments to a call destroyed left to right in the callee? 222 /// This is a fundamental language change, since it implies that objects 223 /// passed by value do *not* live to the end of the full expression. 224 /// Temporaries passed to a function taking a const reference live to the end 225 /// of the full expression as usual. Both the caller and the callee must 226 /// have access to the destructor, while only the caller needs the 227 /// destructor if this is false. areArgsDestroyedLeftToRightInCallee()228 bool areArgsDestroyedLeftToRightInCallee() const { 229 return isMicrosoft(); 230 } 231 232 /// Does this ABI have different entrypoints for complete-object 233 /// and base-subobject constructors? hasConstructorVariants()234 bool hasConstructorVariants() const { 235 return isItaniumFamily(); 236 } 237 238 /// Does this ABI allow virtual bases to be primary base classes? hasPrimaryVBases()239 bool hasPrimaryVBases() const { 240 return isItaniumFamily(); 241 } 242 243 /// Does this ABI use key functions? If so, class data such as the 244 /// vtable is emitted with strong linkage by the TU containing the key 245 /// function. hasKeyFunctions()246 bool hasKeyFunctions() const { 247 return isItaniumFamily(); 248 } 249 250 /// Can an out-of-line inline function serve as a key function? 251 /// 252 /// This flag is only useful in ABIs where type data (for example, 253 /// vtables and type_info objects) are emitted only after processing 254 /// the definition of a special "key" virtual function. (This is safe 255 /// because the ODR requires that every virtual function be defined 256 /// somewhere in a program.) This usually permits such data to be 257 /// emitted in only a single object file, as opposed to redundantly 258 /// in every object file that requires it. 259 /// 260 /// One simple and common definition of "key function" is the first 261 /// virtual function in the class definition which is not defined there. 262 /// This rule works very well when that function has a non-inline 263 /// definition in some non-header file. Unfortunately, when that 264 /// function is defined inline, this rule requires the type data 265 /// to be emitted weakly, as if there were no key function. 266 /// 267 /// The ARM ABI observes that the ODR provides an additional guarantee: 268 /// a virtual function is always ODR-used, so if it is defined inline, 269 /// that definition must appear in every translation unit that defines 270 /// the class. Therefore, there is no reason to allow such functions 271 /// to serve as key functions. 272 /// 273 /// Because this changes the rules for emitting type data, 274 /// it can cause type data to be emitted with both weak and strong 275 /// linkage, which is not allowed on all platforms. Therefore, 276 /// exploiting this observation requires an ABI break and cannot be 277 /// done on a generic Itanium platform. canKeyFunctionBeInline()278 bool canKeyFunctionBeInline() const { 279 switch (getKind()) { 280 case AppleARM64: 281 case Fuchsia: 282 case GenericARM: 283 case WebAssembly: 284 case WatchOS: 285 return false; 286 287 case GenericAArch64: 288 case GenericItanium: 289 case iOS: // old iOS compilers did not follow this rule 290 case Microsoft: 291 case GenericMIPS: 292 case XL: 293 return true; 294 } 295 llvm_unreachable("bad ABI kind"); 296 } 297 298 /// When is record layout allowed to allocate objects in the tail 299 /// padding of a base class? 300 /// 301 /// This decision cannot be changed without breaking platform ABI 302 /// compatibility. In ISO C++98, tail padding reuse was only permitted for 303 /// non-POD base classes, but that restriction was removed retroactively by 304 /// DR 43, and tail padding reuse is always permitted in all de facto C++ 305 /// language modes. However, many platforms use a variant of the old C++98 306 /// rule for compatibility. 307 enum TailPaddingUseRules { 308 /// The tail-padding of a base class is always theoretically 309 /// available, even if it's POD. 310 AlwaysUseTailPadding, 311 312 /// Only allocate objects in the tail padding of a base class if 313 /// the base class is not POD according to the rules of C++ TR1. 314 UseTailPaddingUnlessPOD03, 315 316 /// Only allocate objects in the tail padding of a base class if 317 /// the base class is not POD according to the rules of C++11. 318 UseTailPaddingUnlessPOD11 319 }; getTailPaddingUseRules()320 TailPaddingUseRules getTailPaddingUseRules() const { 321 switch (getKind()) { 322 // To preserve binary compatibility, the generic Itanium ABI has 323 // permanently locked the definition of POD to the rules of C++ TR1, 324 // and that trickles down to derived ABIs. 325 case GenericItanium: 326 case GenericAArch64: 327 case GenericARM: 328 case iOS: 329 case GenericMIPS: 330 case XL: 331 return UseTailPaddingUnlessPOD03; 332 333 // AppleARM64 and WebAssembly use the C++11 POD rules. They do not honor 334 // the Itanium exception about classes with over-large bitfields. 335 case AppleARM64: 336 case Fuchsia: 337 case WebAssembly: 338 case WatchOS: 339 return UseTailPaddingUnlessPOD11; 340 341 // MSVC always allocates fields in the tail-padding of a base class 342 // subobject, even if they're POD. 343 case Microsoft: 344 return AlwaysUseTailPadding; 345 } 346 llvm_unreachable("bad ABI kind"); 347 } 348 349 friend bool operator==(const TargetCXXABI &left, const TargetCXXABI &right) { 350 return left.getKind() == right.getKind(); 351 } 352 353 friend bool operator!=(const TargetCXXABI &left, const TargetCXXABI &right) { 354 return !(left == right); 355 } 356 }; 357 358 } // end namespace clang 359 360 #endif 361