clang 23.0.0git
CGAtomic.cpp
Go to the documentation of this file.
1//===--- CGAtomic.cpp - Emit LLVM IR for atomic operations ----------------===//
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 contains the code for emitting atomic operations.
10//
11//===----------------------------------------------------------------------===//
12
13#include "CGCall.h"
14#include "CGRecordLayout.h"
15#include "CodeGenFunction.h"
16#include "CodeGenModule.h"
17#include "TargetInfo.h"
21#include "llvm/ADT/DenseMap.h"
22#include "llvm/IR/DataLayout.h"
23#include "llvm/IR/Intrinsics.h"
24
25using namespace clang;
26using namespace CodeGen;
27
28namespace {
29 class AtomicInfo {
30 CodeGenFunction &CGF;
31 QualType AtomicTy;
32 QualType ValueTy;
33 uint64_t AtomicSizeInBits;
34 uint64_t ValueSizeInBits;
35 CharUnits AtomicAlign;
36 CharUnits ValueAlign;
37 TypeEvaluationKind EvaluationKind;
38 bool UseLibcall;
39 LValue LVal;
40 CGBitFieldInfo BFI;
41 public:
42 AtomicInfo(CodeGenFunction &CGF, LValue &lvalue)
43 : CGF(CGF), AtomicSizeInBits(0), ValueSizeInBits(0),
44 EvaluationKind(TEK_Scalar), UseLibcall(true) {
45 assert(!lvalue.isGlobalReg());
46 ASTContext &C = CGF.getContext();
47 if (lvalue.isSimple()) {
48 AtomicTy = lvalue.getType();
49 if (auto *ATy = AtomicTy->getAs<AtomicType>())
50 ValueTy = ATy->getValueType();
51 else
52 ValueTy = AtomicTy;
53 EvaluationKind = CGF.getEvaluationKind(ValueTy);
54
55 uint64_t ValueAlignInBits;
56 uint64_t AtomicAlignInBits;
57 TypeInfo ValueTI = C.getTypeInfo(ValueTy);
58 ValueSizeInBits = ValueTI.Width;
59 ValueAlignInBits = ValueTI.Align;
60
61 TypeInfo AtomicTI = C.getTypeInfo(AtomicTy);
62 AtomicSizeInBits = AtomicTI.Width;
63 AtomicAlignInBits = AtomicTI.Align;
64
65 assert(ValueSizeInBits <= AtomicSizeInBits);
66 assert(ValueAlignInBits <= AtomicAlignInBits);
67
68 AtomicAlign = C.toCharUnitsFromBits(AtomicAlignInBits);
69 ValueAlign = C.toCharUnitsFromBits(ValueAlignInBits);
70 if (lvalue.getAlignment().isZero())
71 lvalue.setAlignment(AtomicAlign);
72
73 LVal = lvalue;
74 } else if (lvalue.isBitField()) {
75 ValueTy = lvalue.getType();
76 ValueSizeInBits = C.getTypeSize(ValueTy);
77 auto &OrigBFI = lvalue.getBitFieldInfo();
78 auto Offset = OrigBFI.Offset % C.toBits(lvalue.getAlignment());
79 AtomicSizeInBits = C.toBits(
80 C.toCharUnitsFromBits(Offset + OrigBFI.Size + C.getCharWidth() - 1)
81 .alignTo(lvalue.getAlignment()));
82 llvm::Value *BitFieldPtr = lvalue.getRawBitFieldPointer(CGF);
83 auto OffsetInChars =
84 (C.toCharUnitsFromBits(OrigBFI.Offset) / lvalue.getAlignment()) *
85 lvalue.getAlignment();
86 llvm::Value *StoragePtr = CGF.Builder.CreateConstGEP1_64(
87 CGF.Int8Ty, BitFieldPtr, OffsetInChars.getQuantity());
88 StoragePtr = CGF.Builder.CreateAddrSpaceCast(
89 StoragePtr, CGF.DefaultPtrTy, "atomic_bitfield_base");
90 BFI = OrigBFI;
91 BFI.Offset = Offset;
92 BFI.StorageSize = AtomicSizeInBits;
93 BFI.StorageOffset += OffsetInChars;
94 llvm::Type *StorageTy = CGF.Builder.getIntNTy(AtomicSizeInBits);
95 LVal = LValue::MakeBitfield(
96 Address(StoragePtr, StorageTy, lvalue.getAlignment()), BFI,
97 lvalue.getType(), lvalue.getBaseInfo(), lvalue.getTBAAInfo());
98 AtomicTy = C.getIntTypeForBitwidth(AtomicSizeInBits, OrigBFI.IsSigned);
99 if (AtomicTy.isNull()) {
100 llvm::APInt Size(
101 /*numBits=*/32,
102 C.toCharUnitsFromBits(AtomicSizeInBits).getQuantity());
103 AtomicTy = C.getConstantArrayType(C.CharTy, Size, nullptr,
104 ArraySizeModifier::Normal,
105 /*IndexTypeQuals=*/0);
106 }
107 AtomicAlign = ValueAlign = lvalue.getAlignment();
108 } else if (lvalue.isVectorElt()) {
109 ValueTy = lvalue.getType()->castAs<VectorType>()->getElementType();
110 ValueSizeInBits = C.getTypeSize(ValueTy);
111 AtomicTy = lvalue.getType();
112 AtomicSizeInBits = C.getTypeSize(AtomicTy);
113 AtomicAlign = ValueAlign = lvalue.getAlignment();
114 LVal = lvalue;
115 } else {
116 assert(lvalue.isExtVectorElt());
117 ValueTy = lvalue.getType();
118 ValueSizeInBits = C.getTypeSize(ValueTy);
119 AtomicTy = ValueTy = CGF.getContext().getExtVectorType(
120 lvalue.getType(), cast<llvm::FixedVectorType>(
121 lvalue.getExtVectorAddress().getElementType())
122 ->getNumElements());
123 AtomicSizeInBits = C.getTypeSize(AtomicTy);
124 AtomicAlign = ValueAlign = lvalue.getAlignment();
125 LVal = lvalue;
126 }
127 UseLibcall = !C.getTargetInfo().hasBuiltinAtomic(
128 AtomicSizeInBits, C.toBits(lvalue.getAlignment()));
129 }
130
131 QualType getAtomicType() const { return AtomicTy; }
132 QualType getValueType() const { return ValueTy; }
133 CharUnits getAtomicAlignment() const { return AtomicAlign; }
134 uint64_t getAtomicSizeInBits() const { return AtomicSizeInBits; }
135 uint64_t getValueSizeInBits() const { return ValueSizeInBits; }
136 TypeEvaluationKind getEvaluationKind() const { return EvaluationKind; }
137 bool shouldUseLibcall() const { return UseLibcall; }
138 const LValue &getAtomicLValue() const { return LVal; }
139 llvm::Value *getAtomicPointer() const {
140 if (LVal.isSimple())
141 return LVal.emitRawPointer(CGF);
142 else if (LVal.isBitField())
143 return LVal.getRawBitFieldPointer(CGF);
144 else if (LVal.isVectorElt())
145 return LVal.getRawVectorPointer(CGF);
146 assert(LVal.isExtVectorElt());
147 return LVal.getRawExtVectorPointer(CGF);
148 }
149 Address getAtomicAddress() const {
150 llvm::Type *ElTy;
151 if (LVal.isSimple())
152 ElTy = LVal.getAddress().getElementType();
153 else if (LVal.isBitField())
154 ElTy = LVal.getBitFieldAddress().getElementType();
155 else if (LVal.isVectorElt())
156 ElTy = LVal.getVectorAddress().getElementType();
157 else
158 ElTy = LVal.getExtVectorAddress().getElementType();
159 return Address(getAtomicPointer(), ElTy, getAtomicAlignment());
160 }
161
162 Address getAtomicAddressAsAtomicIntPointer() const {
163 return castToAtomicIntPointer(getAtomicAddress());
164 }
165
166 /// Is the atomic size larger than the underlying value type?
167 ///
168 /// Note that the absence of padding does not mean that atomic
169 /// objects are completely interchangeable with non-atomic
170 /// objects: we might have promoted the alignment of a type
171 /// without making it bigger.
172 bool hasPadding() const {
173 return (ValueSizeInBits != AtomicSizeInBits);
174 }
175
176 bool emitMemSetZeroIfNecessary() const;
177
178 llvm::Value *getAtomicSizeValue() const {
179 CharUnits size = CGF.getContext().toCharUnitsFromBits(AtomicSizeInBits);
180 return CGF.CGM.getSize(size);
181 }
182
183 /// Cast the given pointer to an integer pointer suitable for atomic
184 /// operations if the source.
185 Address castToAtomicIntPointer(Address Addr) const;
186
187 /// If Addr is compatible with the iN that will be used for an atomic
188 /// operation, bitcast it. Otherwise, create a temporary that is suitable
189 /// and copy the value across.
190 Address convertToAtomicIntPointer(Address Addr) const;
191
192 /// Turn an atomic-layout object into an r-value.
193 RValue convertAtomicTempToRValue(Address addr, AggValueSlot resultSlot,
194 SourceLocation loc, bool AsValue) const;
195
196 llvm::Value *getScalarRValValueOrNull(RValue RVal) const;
197
198 /// Converts an rvalue to integer value if needed.
199 llvm::Value *convertRValueToInt(RValue RVal, bool CmpXchg = false) const;
200
201 RValue ConvertToValueOrAtomic(llvm::Value *IntVal, AggValueSlot ResultSlot,
202 SourceLocation Loc, bool AsValue,
203 bool CmpXchg = false) const;
204
205 /// Copy an atomic r-value into atomic-layout memory.
206 void emitCopyIntoMemory(RValue rvalue) const;
207
208 /// Project an l-value down to the value field.
209 LValue projectValue() const {
210 assert(LVal.isSimple());
211 Address addr = getAtomicAddress();
212 if (hasPadding())
213 addr = CGF.Builder.CreateStructGEP(addr, 0);
214
215 return LValue::MakeAddr(addr, getValueType(), CGF.getContext(),
216 LVal.getBaseInfo(), LVal.getTBAAInfo());
217 }
218
219 /// Emits atomic load.
220 /// \returns Loaded value.
221 RValue EmitAtomicLoad(AggValueSlot ResultSlot, SourceLocation Loc,
222 bool AsValue, llvm::AtomicOrdering AO,
223 bool IsVolatile);
224
225 /// Emits atomic compare-and-exchange sequence.
226 /// \param Expected Expected value.
227 /// \param Desired Desired value.
228 /// \param Success Atomic ordering for success operation.
229 /// \param Failure Atomic ordering for failed operation.
230 /// \param IsWeak true if atomic operation is weak, false otherwise.
231 /// \returns Pair of values: previous value from storage (value type) and
232 /// boolean flag (i1 type) with true if success and false otherwise.
233 std::pair<RValue, llvm::Value *>
234 EmitAtomicCompareExchange(RValue Expected, RValue Desired,
235 llvm::AtomicOrdering Success =
236 llvm::AtomicOrdering::SequentiallyConsistent,
237 llvm::AtomicOrdering Failure =
238 llvm::AtomicOrdering::SequentiallyConsistent,
239 bool IsWeak = false);
240
241 /// Emits atomic update.
242 /// \param AO Atomic ordering.
243 /// \param UpdateOp Update operation for the current lvalue.
244 void EmitAtomicUpdate(llvm::AtomicOrdering AO,
245 const llvm::function_ref<RValue(RValue)> &UpdateOp,
246 bool IsVolatile);
247 /// Emits atomic update.
248 /// \param AO Atomic ordering.
249 void EmitAtomicUpdate(llvm::AtomicOrdering AO, RValue UpdateRVal,
250 bool IsVolatile);
251
252 /// Materialize an atomic r-value in atomic-layout memory.
253 Address materializeRValue(RValue rvalue) const;
254
255 /// Creates temp alloca for intermediate operations on atomic value.
256 Address CreateTempAlloca() const;
257 private:
258 bool requiresMemSetZero(llvm::Type *type) const;
259
260
261 /// Emits atomic load as a libcall.
262 void EmitAtomicLoadLibcall(llvm::Value *AddForLoaded,
263 llvm::AtomicOrdering AO, bool IsVolatile);
264 /// Emits atomic load as LLVM instruction.
265 llvm::Value *EmitAtomicLoadOp(llvm::AtomicOrdering AO, bool IsVolatile,
266 bool CmpXchg = false);
267 /// Emits atomic compare-and-exchange op as a libcall.
268 llvm::Value *EmitAtomicCompareExchangeLibcall(
269 llvm::Value *ExpectedAddr, llvm::Value *DesiredAddr,
270 llvm::AtomicOrdering Success =
271 llvm::AtomicOrdering::SequentiallyConsistent,
272 llvm::AtomicOrdering Failure =
273 llvm::AtomicOrdering::SequentiallyConsistent);
274 /// Emits atomic compare-and-exchange op as LLVM instruction.
275 std::pair<llvm::Value *, llvm::Value *> EmitAtomicCompareExchangeOp(
276 llvm::Value *ExpectedVal, llvm::Value *DesiredVal,
277 llvm::AtomicOrdering Success =
278 llvm::AtomicOrdering::SequentiallyConsistent,
279 llvm::AtomicOrdering Failure =
280 llvm::AtomicOrdering::SequentiallyConsistent,
281 bool IsWeak = false);
282 /// Emit atomic update as libcalls.
283 void
284 EmitAtomicUpdateLibcall(llvm::AtomicOrdering AO,
285 const llvm::function_ref<RValue(RValue)> &UpdateOp,
286 bool IsVolatile);
287 /// Emit atomic update as LLVM instructions.
288 void EmitAtomicUpdateOp(llvm::AtomicOrdering AO,
289 const llvm::function_ref<RValue(RValue)> &UpdateOp,
290 bool IsVolatile);
291 /// Emit atomic update as libcalls.
292 void EmitAtomicUpdateLibcall(llvm::AtomicOrdering AO, RValue UpdateRVal,
293 bool IsVolatile);
294 /// Emit atomic update as LLVM instructions.
295 void EmitAtomicUpdateOp(llvm::AtomicOrdering AO, RValue UpdateRal,
296 bool IsVolatile);
297 };
298}
299
300Address AtomicInfo::CreateTempAlloca() const {
301 // Remove addrspace info from the atomic pointer element when making the
302 // alloca pointer element.
303 QualType TmpTy = (LVal.isBitField() && ValueSizeInBits > AtomicSizeInBits)
304 ? ValueTy
305 : AtomicTy.getUnqualifiedType();
306 Address TempAlloca =
307 CGF.CreateMemTemp(TmpTy, getAtomicAlignment(), "atomic-temp");
308 // Cast to pointer to value type for bitfields.
309 if (LVal.isBitField())
311 TempAlloca, getAtomicAddress().getType(),
312 getAtomicAddress().getElementType());
313 return TempAlloca;
314}
315
317 StringRef fnName,
318 QualType resultType,
319 CallArgList &args) {
320 const CGFunctionInfo &fnInfo =
321 CGF.CGM.getTypes().arrangeBuiltinFunctionCall(resultType, args);
322 llvm::FunctionType *fnTy = CGF.CGM.getTypes().GetFunctionType(fnInfo);
323 llvm::AttrBuilder fnAttrB(CGF.getLLVMContext());
324 fnAttrB.addAttribute(llvm::Attribute::NoUnwind);
325 fnAttrB.addAttribute(llvm::Attribute::WillReturn);
326 llvm::AttributeList fnAttrs = llvm::AttributeList::get(
327 CGF.getLLVMContext(), llvm::AttributeList::FunctionIndex, fnAttrB);
328
329 llvm::FunctionCallee fn =
330 CGF.CGM.CreateRuntimeFunction(fnTy, fnName, fnAttrs);
331 auto callee = CGCallee::forDirect(fn);
332 return CGF.EmitCall(fnInfo, callee, ReturnValueSlot(), args);
333}
334
335/// Does a store of the given IR type modify the full expected width?
336static bool isFullSizeType(CodeGenModule &CGM, llvm::Type *type,
337 uint64_t expectedSize) {
338 return (CGM.getDataLayout().getTypeStoreSize(type) * 8 == expectedSize);
339}
340
341/// Does the atomic type require memsetting to zero before initialization?
342///
343/// The IR type is provided as a way of making certain queries faster.
344bool AtomicInfo::requiresMemSetZero(llvm::Type *type) const {
345 // If the atomic type has size padding, we definitely need a memset.
346 if (hasPadding()) return true;
347
348 // Otherwise, do some simple heuristics to try to avoid it:
349 switch (getEvaluationKind()) {
350 // For scalars and complexes, check whether the store size of the
351 // type uses the full size.
352 case TEK_Scalar:
353 return !isFullSizeType(CGF.CGM, type, AtomicSizeInBits);
354 case TEK_Complex:
355 return !isFullSizeType(CGF.CGM, type->getStructElementType(0),
356 AtomicSizeInBits / 2);
357
358 // Padding in structs has an undefined bit pattern. User beware.
359 case TEK_Aggregate:
360 return false;
361 }
362 llvm_unreachable("bad evaluation kind");
363}
364
365bool AtomicInfo::emitMemSetZeroIfNecessary() const {
366 assert(LVal.isSimple());
367 Address addr = LVal.getAddress();
368 if (!requiresMemSetZero(addr.getElementType()))
369 return false;
370
372 addr.emitRawPointer(CGF), llvm::ConstantInt::get(CGF.Int8Ty, 0),
373 CGF.getContext().toCharUnitsFromBits(AtomicSizeInBits).getQuantity(),
374 LVal.getAlignment().getAsAlign());
375 return true;
376}
377
378static void emitAtomicCmpXchg(CodeGenFunction &CGF, AtomicExpr *E, bool IsWeak,
379 Address Dest, Address Ptr, Address Val1,
380 Address Val2, Address ExpectedResult,
381 uint64_t Size, llvm::AtomicOrdering SuccessOrder,
382 llvm::AtomicOrdering FailureOrder,
383 llvm::SyncScope::ID Scope) {
384 // Note that cmpxchg doesn't support weak cmpxchg, at least at the moment.
385 llvm::Value *Expected = CGF.Builder.CreateLoad(Val1);
386 llvm::Value *Desired = CGF.Builder.CreateLoad(Val2);
387
388 llvm::AtomicCmpXchgInst *Pair = CGF.Builder.CreateAtomicCmpXchg(
389 Ptr, Expected, Desired, SuccessOrder, FailureOrder, Scope);
390 Pair->setVolatile(E->isVolatile());
391 Pair->setWeak(IsWeak);
392 CGF.getTargetHooks().setTargetAtomicMetadata(CGF, *Pair, E);
393
394 // Cmp holds the result of the compare-exchange operation: true on success,
395 // false on failure.
396 llvm::Value *Old = CGF.Builder.CreateExtractValue(Pair, 0);
397 llvm::Value *Cmp = CGF.Builder.CreateExtractValue(Pair, 1);
398
399 // This basic block is used to hold the store instruction if the operation
400 // failed.
401 llvm::BasicBlock *StoreExpectedBB =
402 CGF.createBasicBlock("cmpxchg.store_expected", CGF.CurFn);
403
404 // This basic block is the exit point of the operation, we should end up
405 // here regardless of whether or not the operation succeeded.
406 llvm::BasicBlock *ContinueBB =
407 CGF.createBasicBlock("cmpxchg.continue", CGF.CurFn);
408
409 // Update Expected if Expected isn't equal to Old, otherwise branch to the
410 // exit point.
411 CGF.Builder.CreateCondBr(Cmp, ContinueBB, StoreExpectedBB);
412
413 CGF.Builder.SetInsertPoint(StoreExpectedBB);
414 // Update the memory at Expected with Old's value.
415 llvm::Type *ExpectedType = ExpectedResult.getElementType();
416 const llvm::DataLayout &DL = CGF.CGM.getDataLayout();
417 uint64_t ExpectedSizeInBytes = DL.getTypeStoreSize(ExpectedType);
418
419 if (ExpectedSizeInBytes == Size) {
420 // Sizes match: store directly
421 auto *I = CGF.Builder.CreateStore(Old, ExpectedResult);
422 CGF.addInstToCurrentSourceAtom(I, Old);
423 } else {
424 // store only the first ExpectedSizeInBytes bytes of Old
425 llvm::Type *OldType = Old->getType();
426
427 // Allocate temporary storage for Old value
428 Address OldTmp =
429 CGF.CreateTempAlloca(OldType, Ptr.getAlignment(), "old.tmp");
430
431 // Store Old into this temporary
432 auto *I = CGF.Builder.CreateStore(Old, OldTmp);
433 CGF.addInstToCurrentSourceAtom(I, Old);
434
435 // Perform memcpy for first ExpectedSizeInBytes bytes
436 CGF.Builder.CreateMemCpy(ExpectedResult, OldTmp, ExpectedSizeInBytes,
437 /*isVolatile=*/false);
438 }
439
440 // Finally, branch to the exit point.
441 CGF.Builder.CreateBr(ContinueBB);
442
443 CGF.Builder.SetInsertPoint(ContinueBB);
444 // Update the memory at Dest with Cmp's value.
445 CGF.EmitStoreOfScalar(Cmp, CGF.MakeAddrLValue(Dest, E->getType()));
446}
447
448/// Given an ordering required on success, emit all possible cmpxchg
449/// instructions to cope with the provided (but possibly only dynamically known)
450/// FailureOrder.
452 CodeGenFunction &CGF, AtomicExpr *E, bool IsWeak, Address Dest, Address Ptr,
453 Address Val1, Address Val2, Address ExpectedResult,
454 llvm::Value *FailureOrderVal, uint64_t Size,
455 llvm::AtomicOrdering SuccessOrder, llvm::SyncScope::ID Scope) {
456 llvm::AtomicOrdering FailureOrder;
457 if (llvm::ConstantInt *FO = dyn_cast<llvm::ConstantInt>(FailureOrderVal)) {
458 auto FOS = FO->getSExtValue();
459 if (!llvm::isValidAtomicOrderingCABI(FOS))
460 FailureOrder = llvm::AtomicOrdering::Monotonic;
461 else
462 switch ((llvm::AtomicOrderingCABI)FOS) {
463 case llvm::AtomicOrderingCABI::relaxed:
464 // 31.7.2.18: "The failure argument shall not be memory_order_release
465 // nor memory_order_acq_rel". Fallback to monotonic.
466 case llvm::AtomicOrderingCABI::release:
467 case llvm::AtomicOrderingCABI::acq_rel:
468 FailureOrder = llvm::AtomicOrdering::Monotonic;
469 break;
470 case llvm::AtomicOrderingCABI::consume:
471 case llvm::AtomicOrderingCABI::acquire:
472 FailureOrder = llvm::AtomicOrdering::Acquire;
473 break;
474 case llvm::AtomicOrderingCABI::seq_cst:
475 FailureOrder = llvm::AtomicOrdering::SequentiallyConsistent;
476 break;
477 }
478 // Prior to c++17, "the failure argument shall be no stronger than the
479 // success argument". This condition has been lifted and the only
480 // precondition is 31.7.2.18. Effectively treat this as a DR and skip
481 // language version checks.
482 emitAtomicCmpXchg(CGF, E, IsWeak, Dest, Ptr, Val1, Val2, ExpectedResult,
483 Size, SuccessOrder, FailureOrder, Scope);
484 return;
485 }
486
487 // Create all the relevant BB's
488 auto *MonotonicBB = CGF.createBasicBlock("monotonic_fail", CGF.CurFn);
489 auto *AcquireBB = CGF.createBasicBlock("acquire_fail", CGF.CurFn);
490 auto *SeqCstBB = CGF.createBasicBlock("seqcst_fail", CGF.CurFn);
491 auto *ContBB = CGF.createBasicBlock("atomic.continue", CGF.CurFn);
492
493 // MonotonicBB is arbitrarily chosen as the default case; in practice, this
494 // doesn't matter unless someone is crazy enough to use something that
495 // doesn't fold to a constant for the ordering.
496 llvm::SwitchInst *SI = CGF.Builder.CreateSwitch(FailureOrderVal, MonotonicBB);
497 // Implemented as acquire, since it's the closest in LLVM.
498 SI->addCase(CGF.Builder.getInt32((int)llvm::AtomicOrderingCABI::consume),
499 AcquireBB);
500 SI->addCase(CGF.Builder.getInt32((int)llvm::AtomicOrderingCABI::acquire),
501 AcquireBB);
502 SI->addCase(CGF.Builder.getInt32((int)llvm::AtomicOrderingCABI::seq_cst),
503 SeqCstBB);
504
505 // Emit all the different atomics
506 CGF.Builder.SetInsertPoint(MonotonicBB);
507 emitAtomicCmpXchg(CGF, E, IsWeak, Dest, Ptr, Val1, Val2, ExpectedResult, Size,
508 SuccessOrder, llvm::AtomicOrdering::Monotonic, Scope);
509 CGF.Builder.CreateBr(ContBB);
510
511 CGF.Builder.SetInsertPoint(AcquireBB);
512 emitAtomicCmpXchg(CGF, E, IsWeak, Dest, Ptr, Val1, Val2, ExpectedResult, Size,
513 SuccessOrder, llvm::AtomicOrdering::Acquire, Scope);
514 CGF.Builder.CreateBr(ContBB);
515
516 CGF.Builder.SetInsertPoint(SeqCstBB);
517 emitAtomicCmpXchg(CGF, E, IsWeak, Dest, Ptr, Val1, Val2, ExpectedResult, Size,
518 SuccessOrder, llvm::AtomicOrdering::SequentiallyConsistent,
519 Scope);
520 CGF.Builder.CreateBr(ContBB);
521
522 CGF.Builder.SetInsertPoint(ContBB);
523}
524
525/// Duplicate the atomic min/max operation in conventional IR for the builtin
526/// variants that return the new rather than the original value.
527static llvm::Value *EmitPostAtomicMinMax(CGBuilderTy &Builder,
529 bool IsSigned,
530 llvm::Value *OldVal,
531 llvm::Value *RHS) {
532 const bool IsFP = OldVal->getType()->isFloatingPointTy();
533
534 if (IsFP) {
535 llvm::Intrinsic::ID IID = (Op == AtomicExpr::AO__atomic_max_fetch ||
536 Op == AtomicExpr::AO__scoped_atomic_max_fetch)
537 ? llvm::Intrinsic::maxnum
538 : llvm::Intrinsic::minnum;
539
540 return Builder.CreateBinaryIntrinsic(IID, OldVal, RHS, llvm::FMFSource(),
541 "newval");
542 }
543
544 llvm::CmpInst::Predicate Pred;
545 switch (Op) {
546 default:
547 llvm_unreachable("Unexpected min/max operation");
548 case AtomicExpr::AO__atomic_max_fetch:
549 case AtomicExpr::AO__scoped_atomic_max_fetch:
550 Pred = IsSigned ? llvm::CmpInst::ICMP_SGT : llvm::CmpInst::ICMP_UGT;
551 break;
552 case AtomicExpr::AO__atomic_min_fetch:
553 case AtomicExpr::AO__scoped_atomic_min_fetch:
554 Pred = IsSigned ? llvm::CmpInst::ICMP_SLT : llvm::CmpInst::ICMP_ULT;
555 break;
556 }
557 llvm::Value *Cmp = Builder.CreateICmp(Pred, OldVal, RHS, "tst");
558 return Builder.CreateSelect(Cmp, OldVal, RHS, "newval");
559}
560
562 Address Ptr, Address Val1, Address Val2,
563 Address ExpectedResult, llvm::Value *IsWeak,
564 llvm::Value *FailureOrder, uint64_t Size,
565 llvm::AtomicOrdering Order,
566 llvm::SyncScope::ID Scope) {
567 llvm::AtomicRMWInst::BinOp Op = llvm::AtomicRMWInst::Add;
568 bool PostOpMinMax = false;
569 unsigned PostOp = 0;
570
571 switch (E->getOp()) {
572 case AtomicExpr::AO__c11_atomic_init:
573 case AtomicExpr::AO__opencl_atomic_init:
574 llvm_unreachable("Already handled!");
575
576 case AtomicExpr::AO__c11_atomic_compare_exchange_strong:
577 case AtomicExpr::AO__hip_atomic_compare_exchange_strong:
578 case AtomicExpr::AO__opencl_atomic_compare_exchange_strong:
579 emitAtomicCmpXchgFailureSet(CGF, E, false, Dest, Ptr, Val1, Val2,
580 ExpectedResult, FailureOrder, Size, Order,
581 Scope);
582 return;
583 case AtomicExpr::AO__c11_atomic_compare_exchange_weak:
584 case AtomicExpr::AO__opencl_atomic_compare_exchange_weak:
585 case AtomicExpr::AO__hip_atomic_compare_exchange_weak:
586 emitAtomicCmpXchgFailureSet(CGF, E, true, Dest, Ptr, Val1, Val2,
587 ExpectedResult, FailureOrder, Size, Order,
588 Scope);
589 return;
590 case AtomicExpr::AO__atomic_compare_exchange:
591 case AtomicExpr::AO__atomic_compare_exchange_n:
592 case AtomicExpr::AO__scoped_atomic_compare_exchange:
593 case AtomicExpr::AO__scoped_atomic_compare_exchange_n: {
594 if (llvm::ConstantInt *IsWeakC = dyn_cast<llvm::ConstantInt>(IsWeak)) {
595 emitAtomicCmpXchgFailureSet(CGF, E, IsWeakC->getZExtValue(), Dest, Ptr,
596 Val1, Val2, ExpectedResult, FailureOrder,
597 Size, Order, Scope);
598 } else {
599 // Create all the relevant BB's
600 llvm::BasicBlock *StrongBB =
601 CGF.createBasicBlock("cmpxchg.strong", CGF.CurFn);
602 llvm::BasicBlock *WeakBB = CGF.createBasicBlock("cmxchg.weak", CGF.CurFn);
603 llvm::BasicBlock *ContBB =
604 CGF.createBasicBlock("cmpxchg.continue", CGF.CurFn);
605
606 llvm::SwitchInst *SI = CGF.Builder.CreateSwitch(IsWeak, WeakBB);
607 SI->addCase(CGF.Builder.getInt1(false), StrongBB);
608
609 CGF.Builder.SetInsertPoint(StrongBB);
610 emitAtomicCmpXchgFailureSet(CGF, E, false, Dest, Ptr, Val1, Val2,
611 ExpectedResult, FailureOrder, Size, Order,
612 Scope);
613 CGF.Builder.CreateBr(ContBB);
614
615 CGF.Builder.SetInsertPoint(WeakBB);
616 emitAtomicCmpXchgFailureSet(CGF, E, true, Dest, Ptr, Val1, Val2,
617 ExpectedResult, FailureOrder, Size, Order,
618 Scope);
619 CGF.Builder.CreateBr(ContBB);
620
621 CGF.Builder.SetInsertPoint(ContBB);
622 }
623 return;
624 }
625 case AtomicExpr::AO__c11_atomic_load:
626 case AtomicExpr::AO__opencl_atomic_load:
627 case AtomicExpr::AO__hip_atomic_load:
628 case AtomicExpr::AO__atomic_load_n:
629 case AtomicExpr::AO__atomic_load:
630 case AtomicExpr::AO__scoped_atomic_load_n:
631 case AtomicExpr::AO__scoped_atomic_load: {
632 llvm::LoadInst *Load = CGF.Builder.CreateLoad(Ptr);
633 Load->setAtomic(Order, Scope);
634 Load->setVolatile(E->isVolatile());
635 CGF.maybeAttachRangeForLoad(Load, E->getValueType(), E->getExprLoc());
636 auto *I = CGF.Builder.CreateStore(Load, Dest);
637 CGF.addInstToCurrentSourceAtom(I, Load);
638 return;
639 }
640
641 case AtomicExpr::AO__c11_atomic_store:
642 case AtomicExpr::AO__opencl_atomic_store:
643 case AtomicExpr::AO__hip_atomic_store:
644 case AtomicExpr::AO__atomic_store:
645 case AtomicExpr::AO__atomic_store_n:
646 case AtomicExpr::AO__scoped_atomic_store:
647 case AtomicExpr::AO__scoped_atomic_store_n: {
648 llvm::Value *LoadVal1 = CGF.Builder.CreateLoad(Val1);
649 llvm::StoreInst *Store = CGF.Builder.CreateStore(LoadVal1, Ptr);
650 Store->setAtomic(Order, Scope);
651 Store->setVolatile(E->isVolatile());
652 CGF.addInstToCurrentSourceAtom(Store, LoadVal1);
653 return;
654 }
655
656 case AtomicExpr::AO__c11_atomic_exchange:
657 case AtomicExpr::AO__hip_atomic_exchange:
658 case AtomicExpr::AO__opencl_atomic_exchange:
659 case AtomicExpr::AO__atomic_exchange_n:
660 case AtomicExpr::AO__atomic_exchange:
661 case AtomicExpr::AO__scoped_atomic_exchange_n:
662 case AtomicExpr::AO__scoped_atomic_exchange:
663 Op = llvm::AtomicRMWInst::Xchg;
664 break;
665
666 case AtomicExpr::AO__atomic_add_fetch:
667 case AtomicExpr::AO__scoped_atomic_add_fetch:
668 PostOp = E->getValueType()->isFloatingType() ? llvm::Instruction::FAdd
669 : llvm::Instruction::Add;
670 [[fallthrough]];
671 case AtomicExpr::AO__c11_atomic_fetch_add:
672 case AtomicExpr::AO__hip_atomic_fetch_add:
673 case AtomicExpr::AO__opencl_atomic_fetch_add:
674 case AtomicExpr::AO__atomic_fetch_add:
675 case AtomicExpr::AO__scoped_atomic_fetch_add:
676 Op = E->getValueType()->isFloatingType() ? llvm::AtomicRMWInst::FAdd
677 : llvm::AtomicRMWInst::Add;
678 break;
679
680 case AtomicExpr::AO__atomic_sub_fetch:
681 case AtomicExpr::AO__scoped_atomic_sub_fetch:
682 PostOp = E->getValueType()->isFloatingType() ? llvm::Instruction::FSub
683 : llvm::Instruction::Sub;
684 [[fallthrough]];
685 case AtomicExpr::AO__c11_atomic_fetch_sub:
686 case AtomicExpr::AO__hip_atomic_fetch_sub:
687 case AtomicExpr::AO__opencl_atomic_fetch_sub:
688 case AtomicExpr::AO__atomic_fetch_sub:
689 case AtomicExpr::AO__scoped_atomic_fetch_sub:
690 Op = E->getValueType()->isFloatingType() ? llvm::AtomicRMWInst::FSub
691 : llvm::AtomicRMWInst::Sub;
692 break;
693
694 case AtomicExpr::AO__atomic_min_fetch:
695 case AtomicExpr::AO__scoped_atomic_min_fetch:
696 PostOpMinMax = true;
697 [[fallthrough]];
698 case AtomicExpr::AO__c11_atomic_fetch_min:
699 case AtomicExpr::AO__hip_atomic_fetch_min:
700 case AtomicExpr::AO__opencl_atomic_fetch_min:
701 case AtomicExpr::AO__atomic_fetch_min:
702 case AtomicExpr::AO__scoped_atomic_fetch_min:
703 Op = E->getValueType()->isFloatingType()
704 ? llvm::AtomicRMWInst::FMin
705 : (E->getValueType()->isSignedIntegerType()
706 ? llvm::AtomicRMWInst::Min
707 : llvm::AtomicRMWInst::UMin);
708 break;
709
710 case AtomicExpr::AO__atomic_max_fetch:
711 case AtomicExpr::AO__scoped_atomic_max_fetch:
712 PostOpMinMax = true;
713 [[fallthrough]];
714 case AtomicExpr::AO__c11_atomic_fetch_max:
715 case AtomicExpr::AO__hip_atomic_fetch_max:
716 case AtomicExpr::AO__opencl_atomic_fetch_max:
717 case AtomicExpr::AO__atomic_fetch_max:
718 case AtomicExpr::AO__scoped_atomic_fetch_max:
719 Op = E->getValueType()->isFloatingType()
720 ? llvm::AtomicRMWInst::FMax
721 : (E->getValueType()->isSignedIntegerType()
722 ? llvm::AtomicRMWInst::Max
723 : llvm::AtomicRMWInst::UMax);
724 break;
725
726 case AtomicExpr::AO__atomic_and_fetch:
727 case AtomicExpr::AO__scoped_atomic_and_fetch:
728 PostOp = llvm::Instruction::And;
729 [[fallthrough]];
730 case AtomicExpr::AO__c11_atomic_fetch_and:
731 case AtomicExpr::AO__hip_atomic_fetch_and:
732 case AtomicExpr::AO__opencl_atomic_fetch_and:
733 case AtomicExpr::AO__atomic_fetch_and:
734 case AtomicExpr::AO__scoped_atomic_fetch_and:
735 Op = llvm::AtomicRMWInst::And;
736 break;
737
738 case AtomicExpr::AO__atomic_or_fetch:
739 case AtomicExpr::AO__scoped_atomic_or_fetch:
740 PostOp = llvm::Instruction::Or;
741 [[fallthrough]];
742 case AtomicExpr::AO__c11_atomic_fetch_or:
743 case AtomicExpr::AO__hip_atomic_fetch_or:
744 case AtomicExpr::AO__opencl_atomic_fetch_or:
745 case AtomicExpr::AO__atomic_fetch_or:
746 case AtomicExpr::AO__scoped_atomic_fetch_or:
747 Op = llvm::AtomicRMWInst::Or;
748 break;
749
750 case AtomicExpr::AO__atomic_xor_fetch:
751 case AtomicExpr::AO__scoped_atomic_xor_fetch:
752 PostOp = llvm::Instruction::Xor;
753 [[fallthrough]];
754 case AtomicExpr::AO__c11_atomic_fetch_xor:
755 case AtomicExpr::AO__hip_atomic_fetch_xor:
756 case AtomicExpr::AO__opencl_atomic_fetch_xor:
757 case AtomicExpr::AO__atomic_fetch_xor:
758 case AtomicExpr::AO__scoped_atomic_fetch_xor:
759 Op = llvm::AtomicRMWInst::Xor;
760 break;
761
762 case AtomicExpr::AO__atomic_nand_fetch:
763 case AtomicExpr::AO__scoped_atomic_nand_fetch:
764 PostOp = llvm::Instruction::And; // the NOT is special cased below
765 [[fallthrough]];
766 case AtomicExpr::AO__c11_atomic_fetch_nand:
767 case AtomicExpr::AO__atomic_fetch_nand:
768 case AtomicExpr::AO__scoped_atomic_fetch_nand:
769 Op = llvm::AtomicRMWInst::Nand;
770 break;
771
772 case AtomicExpr::AO__atomic_fetch_uinc:
773 case AtomicExpr::AO__scoped_atomic_fetch_uinc:
774 Op = llvm::AtomicRMWInst::UIncWrap;
775 break;
776 case AtomicExpr::AO__atomic_fetch_udec:
777 case AtomicExpr::AO__scoped_atomic_fetch_udec:
778 Op = llvm::AtomicRMWInst::UDecWrap;
779 break;
780
781 case AtomicExpr::AO__atomic_test_and_set: {
782 llvm::AtomicRMWInst *RMWI =
783 CGF.emitAtomicRMWInst(llvm::AtomicRMWInst::Xchg, Ptr,
784 CGF.Builder.getInt8(1), Order, Scope, E);
785 RMWI->setVolatile(E->isVolatile());
786 llvm::Value *Result = CGF.EmitToMemory(
787 CGF.Builder.CreateIsNotNull(RMWI, "tobool"), E->getType());
788 auto *I = CGF.Builder.CreateStore(Result, Dest);
790 return;
791 }
792
793 case AtomicExpr::AO__atomic_clear: {
794 llvm::StoreInst *Store =
795 CGF.Builder.CreateStore(CGF.Builder.getInt8(0), Ptr);
796 Store->setAtomic(Order, Scope);
797 Store->setVolatile(E->isVolatile());
798 CGF.addInstToCurrentSourceAtom(Store, nullptr);
799 return;
800 }
801 }
802
803 llvm::Value *LoadVal1 = CGF.Builder.CreateLoad(Val1);
804 llvm::AtomicRMWInst *RMWI =
805 CGF.emitAtomicRMWInst(Op, Ptr, LoadVal1, Order, Scope, E);
806 RMWI->setVolatile(E->isVolatile());
807
808 // For __atomic_*_fetch operations, perform the operation again to
809 // determine the value which was written.
810 llvm::Value *Result = RMWI;
811 if (PostOpMinMax)
814 RMWI, LoadVal1);
815 else if (PostOp)
816 Result = CGF.Builder.CreateBinOp((llvm::Instruction::BinaryOps)PostOp, RMWI,
817 LoadVal1);
818 if (E->getOp() == AtomicExpr::AO__atomic_nand_fetch ||
819 E->getOp() == AtomicExpr::AO__scoped_atomic_nand_fetch)
820 Result = CGF.Builder.CreateNot(Result);
821 auto *I = CGF.Builder.CreateStore(Result, Dest);
823}
824
825// This function emits any expression (scalar, complex, or aggregate)
826// into a temporary alloca.
827static Address
829 Address DeclPtr = CGF.CreateMemTemp(E->getType(), ".atomictmp");
830 CGF.EmitAnyExprToMem(E, DeclPtr, E->getType().getQualifiers(),
831 /*Init*/ true);
832 return DeclPtr;
833}
834
835/// Return true if \param ValTy is a type that should be casted to integer
836/// around the atomic memory operation. If \param CmpXchg is true, then the
837/// cast of a floating point type is made as that instruction can not have
838/// floating point operands. TODO: Allow compare-and-exchange and FP - see
839/// comment in AtomicExpandPass.cpp.
840static bool shouldCastToInt(llvm::Type *ValTy, bool CmpXchg) {
841 if (ValTy->isFloatingPointTy())
842 return ValTy->isX86_FP80Ty() || CmpXchg;
843 return !ValTy->isIntegerTy() && !ValTy->isPointerTy();
844}
845
847 Address Ptr, Address Val1, Address Val2,
848 Address OriginalVal1, llvm::Value *IsWeak,
849 llvm::Value *FailureOrder, uint64_t Size,
850 llvm::AtomicOrdering Order, llvm::Value *Scope) {
851 auto ScopeModel = Expr->getScopeModel();
852
853 // LLVM atomic instructions always have sync scope. If clang atomic
854 // expression has no scope operand, use default LLVM sync scope.
855 if (!ScopeModel) {
856 llvm::SyncScope::ID SS;
857 if (CGF.getLangOpts().OpenCL)
858 // OpenCL approach is: "The functions that do not have memory_scope
859 // argument have the same semantics as the corresponding functions with
860 // the memory_scope argument set to memory_scope_device." See ref.:
861 // https://registry.khronos.org/OpenCL/specs/3.0-unified/html/OpenCL_C.html#atomic-functions
864 Order, CGF.getLLVMContext());
865 else
866 SS = llvm::SyncScope::System;
867 EmitAtomicOp(CGF, Expr, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
868 FailureOrder, Size, Order, SS);
869 return;
870 }
871
872 // Handle constant scope.
873 if (auto SC = dyn_cast<llvm::ConstantInt>(Scope)) {
874 auto SCID = CGF.getTargetHooks().getLLVMSyncScopeID(
875 CGF.CGM.getLangOpts(), ScopeModel->map(SC->getZExtValue()),
876 Order, CGF.CGM.getLLVMContext());
877 EmitAtomicOp(CGF, Expr, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
878 FailureOrder, Size, Order, SCID);
879 return;
880 }
881
882 // Handle non-constant scope.
883 auto &Builder = CGF.Builder;
884 auto Scopes = ScopeModel->getRuntimeValues();
885 llvm::DenseMap<unsigned, llvm::BasicBlock *> BB;
886 for (auto S : Scopes)
887 BB[S] = CGF.createBasicBlock(getAsString(ScopeModel->map(S)), CGF.CurFn);
888
889 llvm::BasicBlock *ContBB =
890 CGF.createBasicBlock("atomic.scope.continue", CGF.CurFn);
891
892 auto *SC = Builder.CreateIntCast(Scope, Builder.getInt32Ty(), false);
893 // If unsupported sync scope is encountered at run time, assume a fallback
894 // sync scope value.
895 auto FallBack = ScopeModel->getFallBackValue();
896 llvm::SwitchInst *SI = Builder.CreateSwitch(SC, BB[FallBack]);
897 for (auto S : Scopes) {
898 auto *B = BB[S];
899 if (S != FallBack)
900 SI->addCase(Builder.getInt32(S), B);
901
902 Builder.SetInsertPoint(B);
903 EmitAtomicOp(CGF, Expr, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
904 FailureOrder, Size, Order,
906 CGF.CGM.getLangOpts(), ScopeModel->map(S), Order,
907 CGF.getLLVMContext()));
908 Builder.CreateBr(ContBB);
909 }
910
911 Builder.SetInsertPoint(ContBB);
912}
913
916
917 QualType AtomicTy = E->getPtr()->getType()->getPointeeType();
918 QualType MemTy = AtomicTy;
919 if (const AtomicType *AT = AtomicTy->getAs<AtomicType>())
920 MemTy = AT->getValueType();
921 llvm::Value *IsWeak = nullptr, *OrderFail = nullptr;
922
923 Address Val1 = Address::invalid();
924 Address Val2 = Address::invalid();
925 Address Dest = Address::invalid();
927
928 if (E->getOp() == AtomicExpr::AO__c11_atomic_init ||
929 E->getOp() == AtomicExpr::AO__opencl_atomic_init) {
930 LValue lvalue = MakeAddrLValue(Ptr, AtomicTy);
931 EmitAtomicInit(E->getVal1(), lvalue);
932 return RValue::get(nullptr);
933 }
934
935 auto TInfo = getContext().getTypeInfoInChars(AtomicTy);
936 uint64_t Size = TInfo.Width.getQuantity();
937 unsigned MaxInlineWidthInBits = getTarget().getMaxAtomicInlineWidth();
938
939 CharUnits MaxInlineWidth =
940 getContext().toCharUnitsFromBits(MaxInlineWidthInBits);
941 DiagnosticsEngine &Diags = CGM.getDiags();
942 bool Misaligned = !Ptr.getAlignment().isMultipleOf(TInfo.Width);
943 bool Oversized = getContext().toBits(TInfo.Width) > MaxInlineWidthInBits;
944 if (Misaligned) {
945 Diags.Report(E->getBeginLoc(), diag::warn_atomic_op_misaligned)
946 << (int)TInfo.Width.getQuantity()
947 << (int)Ptr.getAlignment().getQuantity();
948 }
949 if (Oversized) {
950 Diags.Report(E->getBeginLoc(), diag::warn_atomic_op_oversized)
951 << (int)TInfo.Width.getQuantity() << (int)MaxInlineWidth.getQuantity();
952 }
953
954 llvm::Value *Order = EmitScalarExpr(E->getOrder());
955 llvm::Value *Scope =
956 E->getScopeModel() ? EmitScalarExpr(E->getScope()) : nullptr;
957
958 switch (E->getOp()) {
959 case AtomicExpr::AO__c11_atomic_init:
960 case AtomicExpr::AO__opencl_atomic_init:
961 llvm_unreachable("Already handled above with EmitAtomicInit!");
962
963 case AtomicExpr::AO__atomic_load_n:
964 case AtomicExpr::AO__scoped_atomic_load_n:
965 case AtomicExpr::AO__c11_atomic_load:
966 case AtomicExpr::AO__opencl_atomic_load:
967 case AtomicExpr::AO__hip_atomic_load:
968 case AtomicExpr::AO__atomic_test_and_set:
969 case AtomicExpr::AO__atomic_clear:
970 break;
971
972 case AtomicExpr::AO__atomic_load:
973 case AtomicExpr::AO__scoped_atomic_load:
975 break;
976
977 case AtomicExpr::AO__atomic_store:
978 case AtomicExpr::AO__scoped_atomic_store:
980 break;
981
982 case AtomicExpr::AO__atomic_exchange:
983 case AtomicExpr::AO__scoped_atomic_exchange:
986 break;
987
988 case AtomicExpr::AO__atomic_compare_exchange:
989 case AtomicExpr::AO__atomic_compare_exchange_n:
990 case AtomicExpr::AO__c11_atomic_compare_exchange_weak:
991 case AtomicExpr::AO__c11_atomic_compare_exchange_strong:
992 case AtomicExpr::AO__hip_atomic_compare_exchange_weak:
993 case AtomicExpr::AO__hip_atomic_compare_exchange_strong:
994 case AtomicExpr::AO__opencl_atomic_compare_exchange_weak:
995 case AtomicExpr::AO__opencl_atomic_compare_exchange_strong:
996 case AtomicExpr::AO__scoped_atomic_compare_exchange:
997 case AtomicExpr::AO__scoped_atomic_compare_exchange_n:
999 if (E->getOp() == AtomicExpr::AO__atomic_compare_exchange ||
1000 E->getOp() == AtomicExpr::AO__scoped_atomic_compare_exchange)
1001 Val2 = EmitPointerWithAlignment(E->getVal2());
1002 else
1003 Val2 = EmitValToTemp(*this, E->getVal2());
1004 OrderFail = EmitScalarExpr(E->getOrderFail());
1005 if (E->getOp() == AtomicExpr::AO__atomic_compare_exchange_n ||
1006 E->getOp() == AtomicExpr::AO__atomic_compare_exchange ||
1007 E->getOp() == AtomicExpr::AO__scoped_atomic_compare_exchange_n ||
1008 E->getOp() == AtomicExpr::AO__scoped_atomic_compare_exchange)
1009 IsWeak = EmitScalarExpr(E->getWeak());
1010 break;
1011
1012 case AtomicExpr::AO__c11_atomic_fetch_add:
1013 case AtomicExpr::AO__c11_atomic_fetch_sub:
1014 case AtomicExpr::AO__hip_atomic_fetch_add:
1015 case AtomicExpr::AO__hip_atomic_fetch_sub:
1016 case AtomicExpr::AO__opencl_atomic_fetch_add:
1017 case AtomicExpr::AO__opencl_atomic_fetch_sub:
1018 if (MemTy->isPointerType()) {
1019 // For pointer arithmetic, we're required to do a bit of math:
1020 // adding 1 to an int* is not the same as adding 1 to a uintptr_t.
1021 // ... but only for the C11 builtins. The GNU builtins expect the
1022 // user to multiply by sizeof(T).
1023 QualType Val1Ty = E->getVal1()->getType();
1024 llvm::Value *Val1Scalar = EmitScalarExpr(E->getVal1());
1025 CharUnits PointeeIncAmt =
1026 getContext().getTypeSizeInChars(MemTy->getPointeeType());
1027 Val1Scalar = Builder.CreateMul(Val1Scalar, CGM.getSize(PointeeIncAmt));
1028 auto Temp = CreateMemTemp(Val1Ty, ".atomictmp");
1029 Val1 = Temp;
1030 EmitStoreOfScalar(Val1Scalar, MakeAddrLValue(Temp, Val1Ty));
1031 break;
1032 }
1033 [[fallthrough]];
1034 case AtomicExpr::AO__atomic_fetch_add:
1035 case AtomicExpr::AO__atomic_fetch_max:
1036 case AtomicExpr::AO__atomic_fetch_min:
1037 case AtomicExpr::AO__atomic_fetch_sub:
1038 case AtomicExpr::AO__atomic_add_fetch:
1039 case AtomicExpr::AO__atomic_max_fetch:
1040 case AtomicExpr::AO__atomic_min_fetch:
1041 case AtomicExpr::AO__atomic_sub_fetch:
1042 case AtomicExpr::AO__c11_atomic_fetch_max:
1043 case AtomicExpr::AO__c11_atomic_fetch_min:
1044 case AtomicExpr::AO__opencl_atomic_fetch_max:
1045 case AtomicExpr::AO__opencl_atomic_fetch_min:
1046 case AtomicExpr::AO__hip_atomic_fetch_max:
1047 case AtomicExpr::AO__hip_atomic_fetch_min:
1048 case AtomicExpr::AO__scoped_atomic_fetch_add:
1049 case AtomicExpr::AO__scoped_atomic_fetch_max:
1050 case AtomicExpr::AO__scoped_atomic_fetch_min:
1051 case AtomicExpr::AO__scoped_atomic_fetch_sub:
1052 case AtomicExpr::AO__scoped_atomic_add_fetch:
1053 case AtomicExpr::AO__scoped_atomic_max_fetch:
1054 case AtomicExpr::AO__scoped_atomic_min_fetch:
1055 case AtomicExpr::AO__scoped_atomic_sub_fetch:
1056 [[fallthrough]];
1057
1058 case AtomicExpr::AO__atomic_fetch_and:
1059 case AtomicExpr::AO__atomic_fetch_nand:
1060 case AtomicExpr::AO__atomic_fetch_or:
1061 case AtomicExpr::AO__atomic_fetch_xor:
1062 case AtomicExpr::AO__atomic_fetch_uinc:
1063 case AtomicExpr::AO__atomic_fetch_udec:
1064 case AtomicExpr::AO__atomic_and_fetch:
1065 case AtomicExpr::AO__atomic_nand_fetch:
1066 case AtomicExpr::AO__atomic_or_fetch:
1067 case AtomicExpr::AO__atomic_xor_fetch:
1068 case AtomicExpr::AO__atomic_store_n:
1069 case AtomicExpr::AO__atomic_exchange_n:
1070 case AtomicExpr::AO__c11_atomic_fetch_and:
1071 case AtomicExpr::AO__c11_atomic_fetch_nand:
1072 case AtomicExpr::AO__c11_atomic_fetch_or:
1073 case AtomicExpr::AO__c11_atomic_fetch_xor:
1074 case AtomicExpr::AO__c11_atomic_store:
1075 case AtomicExpr::AO__c11_atomic_exchange:
1076 case AtomicExpr::AO__hip_atomic_fetch_and:
1077 case AtomicExpr::AO__hip_atomic_fetch_or:
1078 case AtomicExpr::AO__hip_atomic_fetch_xor:
1079 case AtomicExpr::AO__hip_atomic_store:
1080 case AtomicExpr::AO__hip_atomic_exchange:
1081 case AtomicExpr::AO__opencl_atomic_fetch_and:
1082 case AtomicExpr::AO__opencl_atomic_fetch_or:
1083 case AtomicExpr::AO__opencl_atomic_fetch_xor:
1084 case AtomicExpr::AO__opencl_atomic_store:
1085 case AtomicExpr::AO__opencl_atomic_exchange:
1086 case AtomicExpr::AO__scoped_atomic_fetch_and:
1087 case AtomicExpr::AO__scoped_atomic_fetch_nand:
1088 case AtomicExpr::AO__scoped_atomic_fetch_or:
1089 case AtomicExpr::AO__scoped_atomic_fetch_xor:
1090 case AtomicExpr::AO__scoped_atomic_and_fetch:
1091 case AtomicExpr::AO__scoped_atomic_nand_fetch:
1092 case AtomicExpr::AO__scoped_atomic_or_fetch:
1093 case AtomicExpr::AO__scoped_atomic_xor_fetch:
1094 case AtomicExpr::AO__scoped_atomic_store_n:
1095 case AtomicExpr::AO__scoped_atomic_exchange_n:
1096 case AtomicExpr::AO__scoped_atomic_fetch_uinc:
1097 case AtomicExpr::AO__scoped_atomic_fetch_udec:
1098 Val1 = EmitValToTemp(*this, E->getVal1());
1099 break;
1100 }
1101
1102 QualType RValTy = E->getType().getUnqualifiedType();
1103 bool ShouldCastToIntPtrTy =
1105
1106 // The inlined atomics only function on iN types, where N is a power of 2. We
1107 // need to make sure (via temporaries if necessary) that all incoming values
1108 // are compatible.
1109 LValue AtomicVal = MakeAddrLValue(Ptr, AtomicTy);
1110 AtomicInfo Atomics(*this, AtomicVal);
1111
1112 Address OriginalVal1 = Val1;
1113 if (ShouldCastToIntPtrTy) {
1114 Ptr = Atomics.castToAtomicIntPointer(Ptr);
1115 if (Val1.isValid())
1116 Val1 = Atomics.convertToAtomicIntPointer(Val1);
1117 if (Val2.isValid())
1118 Val2 = Atomics.convertToAtomicIntPointer(Val2);
1119 }
1120 if (Dest.isValid()) {
1121 if (ShouldCastToIntPtrTy)
1122 Dest = Atomics.castToAtomicIntPointer(Dest);
1123 } else if (E->isCmpXChg())
1124 Dest = CreateMemTemp(RValTy, "cmpxchg.bool");
1125 else if (!RValTy->isVoidType()) {
1126 Dest = Atomics.CreateTempAlloca();
1127 if (ShouldCastToIntPtrTy)
1128 Dest = Atomics.castToAtomicIntPointer(Dest);
1129 }
1130
1131 bool PowerOf2Size = (Size & (Size - 1)) == 0;
1132 bool UseLibcall = !PowerOf2Size || (Size > 16);
1133
1134 // For atomics larger than 16 bytes, emit a libcall from the frontend. This
1135 // avoids the overhead of dealing with excessively-large value types in IR.
1136 // Non-power-of-2 values also lower to libcall here, as they are not currently
1137 // permitted in IR instructions (although that constraint could be relaxed in
1138 // the future). For other cases where a libcall is required on a given
1139 // platform, we let the backend handle it (this includes handling for all of
1140 // the size-optimized libcall variants, which are only valid up to 16 bytes.)
1141 //
1142 // See: https://llvm.org/docs/Atomics.html#libcalls-atomic
1143 if (UseLibcall) {
1144 CallArgList Args;
1145 // For non-optimized library calls, the size is the first parameter.
1146 Args.add(RValue::get(llvm::ConstantInt::get(SizeTy, Size)),
1147 getContext().getSizeType());
1148
1149 // The atomic address is the second parameter.
1150 // The OpenCL atomic library functions only accept pointer arguments to
1151 // generic address space.
1152 auto CastToGenericAddrSpace = [&](llvm::Value *V, QualType PT) {
1153 if (!E->isOpenCL())
1154 return V;
1155 auto AS = PT->castAs<PointerType>()->getPointeeType().getAddressSpace();
1156 if (AS == LangAS::opencl_generic)
1157 return V;
1158 auto DestAS = getContext().getTargetAddressSpace(LangAS::opencl_generic);
1159 auto *DestType = llvm::PointerType::get(getLLVMContext(), DestAS);
1160
1161 return performAddrSpaceCast(V, DestType);
1162 };
1163
1164 Args.add(RValue::get(CastToGenericAddrSpace(Ptr.emitRawPointer(*this),
1165 E->getPtr()->getType())),
1167
1168 // The next 1-3 parameters are op-dependent.
1169 std::string LibCallName;
1170 QualType RetTy;
1171 bool HaveRetTy = false;
1172 switch (E->getOp()) {
1173 case AtomicExpr::AO__c11_atomic_init:
1174 case AtomicExpr::AO__opencl_atomic_init:
1175 llvm_unreachable("Already handled!");
1176
1177 // There is only one libcall for compare an exchange, because there is no
1178 // optimisation benefit possible from a libcall version of a weak compare
1179 // and exchange.
1180 // bool __atomic_compare_exchange(size_t size, void *mem, void *expected,
1181 // void *desired, int success, int failure)
1182 case AtomicExpr::AO__atomic_compare_exchange:
1183 case AtomicExpr::AO__atomic_compare_exchange_n:
1184 case AtomicExpr::AO__c11_atomic_compare_exchange_weak:
1185 case AtomicExpr::AO__c11_atomic_compare_exchange_strong:
1186 case AtomicExpr::AO__hip_atomic_compare_exchange_weak:
1187 case AtomicExpr::AO__hip_atomic_compare_exchange_strong:
1188 case AtomicExpr::AO__opencl_atomic_compare_exchange_weak:
1189 case AtomicExpr::AO__opencl_atomic_compare_exchange_strong:
1190 case AtomicExpr::AO__scoped_atomic_compare_exchange:
1191 case AtomicExpr::AO__scoped_atomic_compare_exchange_n:
1192 LibCallName = "__atomic_compare_exchange";
1193 RetTy = getContext().BoolTy;
1194 HaveRetTy = true;
1195 Args.add(RValue::get(CastToGenericAddrSpace(Val1.emitRawPointer(*this),
1196 E->getVal1()->getType())),
1198 Args.add(RValue::get(CastToGenericAddrSpace(Val2.emitRawPointer(*this),
1199 E->getVal2()->getType())),
1201 Args.add(RValue::get(Order), getContext().IntTy);
1202 Order = OrderFail;
1203 break;
1204 // void __atomic_exchange(size_t size, void *mem, void *val, void *return,
1205 // int order)
1206 case AtomicExpr::AO__atomic_exchange:
1207 case AtomicExpr::AO__atomic_exchange_n:
1208 case AtomicExpr::AO__c11_atomic_exchange:
1209 case AtomicExpr::AO__hip_atomic_exchange:
1210 case AtomicExpr::AO__opencl_atomic_exchange:
1211 case AtomicExpr::AO__scoped_atomic_exchange:
1212 case AtomicExpr::AO__scoped_atomic_exchange_n:
1213 LibCallName = "__atomic_exchange";
1214 Args.add(RValue::get(CastToGenericAddrSpace(Val1.emitRawPointer(*this),
1215 E->getVal1()->getType())),
1217 break;
1218 // void __atomic_store(size_t size, void *mem, void *val, int order)
1219 case AtomicExpr::AO__atomic_store:
1220 case AtomicExpr::AO__atomic_store_n:
1221 case AtomicExpr::AO__c11_atomic_store:
1222 case AtomicExpr::AO__hip_atomic_store:
1223 case AtomicExpr::AO__opencl_atomic_store:
1224 case AtomicExpr::AO__scoped_atomic_store:
1225 case AtomicExpr::AO__scoped_atomic_store_n:
1226 LibCallName = "__atomic_store";
1227 RetTy = getContext().VoidTy;
1228 HaveRetTy = true;
1229 Args.add(RValue::get(CastToGenericAddrSpace(Val1.emitRawPointer(*this),
1230 E->getVal1()->getType())),
1232 break;
1233 // void __atomic_load(size_t size, void *mem, void *return, int order)
1234 case AtomicExpr::AO__atomic_load:
1235 case AtomicExpr::AO__atomic_load_n:
1236 case AtomicExpr::AO__c11_atomic_load:
1237 case AtomicExpr::AO__hip_atomic_load:
1238 case AtomicExpr::AO__opencl_atomic_load:
1239 case AtomicExpr::AO__scoped_atomic_load:
1240 case AtomicExpr::AO__scoped_atomic_load_n:
1241 LibCallName = "__atomic_load";
1242 break;
1243 case AtomicExpr::AO__atomic_add_fetch:
1244 case AtomicExpr::AO__scoped_atomic_add_fetch:
1245 case AtomicExpr::AO__atomic_fetch_add:
1246 case AtomicExpr::AO__c11_atomic_fetch_add:
1247 case AtomicExpr::AO__hip_atomic_fetch_add:
1248 case AtomicExpr::AO__opencl_atomic_fetch_add:
1249 case AtomicExpr::AO__scoped_atomic_fetch_add:
1250 case AtomicExpr::AO__atomic_and_fetch:
1251 case AtomicExpr::AO__scoped_atomic_and_fetch:
1252 case AtomicExpr::AO__atomic_fetch_and:
1253 case AtomicExpr::AO__c11_atomic_fetch_and:
1254 case AtomicExpr::AO__hip_atomic_fetch_and:
1255 case AtomicExpr::AO__opencl_atomic_fetch_and:
1256 case AtomicExpr::AO__scoped_atomic_fetch_and:
1257 case AtomicExpr::AO__atomic_or_fetch:
1258 case AtomicExpr::AO__scoped_atomic_or_fetch:
1259 case AtomicExpr::AO__atomic_fetch_or:
1260 case AtomicExpr::AO__c11_atomic_fetch_or:
1261 case AtomicExpr::AO__hip_atomic_fetch_or:
1262 case AtomicExpr::AO__opencl_atomic_fetch_or:
1263 case AtomicExpr::AO__scoped_atomic_fetch_or:
1264 case AtomicExpr::AO__atomic_sub_fetch:
1265 case AtomicExpr::AO__scoped_atomic_sub_fetch:
1266 case AtomicExpr::AO__atomic_fetch_sub:
1267 case AtomicExpr::AO__c11_atomic_fetch_sub:
1268 case AtomicExpr::AO__hip_atomic_fetch_sub:
1269 case AtomicExpr::AO__opencl_atomic_fetch_sub:
1270 case AtomicExpr::AO__scoped_atomic_fetch_sub:
1271 case AtomicExpr::AO__atomic_xor_fetch:
1272 case AtomicExpr::AO__scoped_atomic_xor_fetch:
1273 case AtomicExpr::AO__atomic_fetch_xor:
1274 case AtomicExpr::AO__c11_atomic_fetch_xor:
1275 case AtomicExpr::AO__hip_atomic_fetch_xor:
1276 case AtomicExpr::AO__opencl_atomic_fetch_xor:
1277 case AtomicExpr::AO__scoped_atomic_fetch_xor:
1278 case AtomicExpr::AO__atomic_nand_fetch:
1279 case AtomicExpr::AO__atomic_fetch_nand:
1280 case AtomicExpr::AO__c11_atomic_fetch_nand:
1281 case AtomicExpr::AO__scoped_atomic_fetch_nand:
1282 case AtomicExpr::AO__scoped_atomic_nand_fetch:
1283 case AtomicExpr::AO__atomic_min_fetch:
1284 case AtomicExpr::AO__atomic_fetch_min:
1285 case AtomicExpr::AO__c11_atomic_fetch_min:
1286 case AtomicExpr::AO__hip_atomic_fetch_min:
1287 case AtomicExpr::AO__opencl_atomic_fetch_min:
1288 case AtomicExpr::AO__scoped_atomic_fetch_min:
1289 case AtomicExpr::AO__scoped_atomic_min_fetch:
1290 case AtomicExpr::AO__atomic_max_fetch:
1291 case AtomicExpr::AO__atomic_fetch_max:
1292 case AtomicExpr::AO__c11_atomic_fetch_max:
1293 case AtomicExpr::AO__hip_atomic_fetch_max:
1294 case AtomicExpr::AO__opencl_atomic_fetch_max:
1295 case AtomicExpr::AO__scoped_atomic_fetch_max:
1296 case AtomicExpr::AO__scoped_atomic_max_fetch:
1297 case AtomicExpr::AO__scoped_atomic_fetch_uinc:
1298 case AtomicExpr::AO__scoped_atomic_fetch_udec:
1299 case AtomicExpr::AO__atomic_test_and_set:
1300 case AtomicExpr::AO__atomic_clear:
1301 case AtomicExpr::AO__atomic_fetch_uinc:
1302 case AtomicExpr::AO__atomic_fetch_udec:
1303 llvm_unreachable("Integral atomic operations always become atomicrmw!");
1304 }
1305
1306 if (E->isOpenCL()) {
1307 LibCallName =
1308 std::string("__opencl") + StringRef(LibCallName).drop_front(1).str();
1309 }
1310 // By default, assume we return a value of the atomic type.
1311 if (!HaveRetTy) {
1312 // Value is returned through parameter before the order.
1313 RetTy = getContext().VoidTy;
1314 Args.add(RValue::get(
1315 CastToGenericAddrSpace(Dest.emitRawPointer(*this), RetTy)),
1317 }
1318 // Order is always the last parameter.
1319 Args.add(RValue::get(Order),
1320 getContext().IntTy);
1321 if (E->isOpenCL())
1323
1324 RValue Res = emitAtomicLibcall(*this, LibCallName, RetTy, Args);
1325 // The value is returned directly from the libcall.
1326 if (E->isCmpXChg())
1327 return Res;
1328
1329 if (RValTy->isVoidType())
1330 return RValue::get(nullptr);
1331
1333 RValTy, E->getExprLoc());
1334 }
1335
1336 bool IsStore = E->getOp() == AtomicExpr::AO__c11_atomic_store ||
1337 E->getOp() == AtomicExpr::AO__opencl_atomic_store ||
1338 E->getOp() == AtomicExpr::AO__hip_atomic_store ||
1339 E->getOp() == AtomicExpr::AO__atomic_store ||
1340 E->getOp() == AtomicExpr::AO__atomic_store_n ||
1341 E->getOp() == AtomicExpr::AO__scoped_atomic_store ||
1342 E->getOp() == AtomicExpr::AO__scoped_atomic_store_n ||
1343 E->getOp() == AtomicExpr::AO__atomic_clear;
1344 bool IsLoad = E->getOp() == AtomicExpr::AO__c11_atomic_load ||
1345 E->getOp() == AtomicExpr::AO__opencl_atomic_load ||
1346 E->getOp() == AtomicExpr::AO__hip_atomic_load ||
1347 E->getOp() == AtomicExpr::AO__atomic_load ||
1348 E->getOp() == AtomicExpr::AO__atomic_load_n ||
1349 E->getOp() == AtomicExpr::AO__scoped_atomic_load ||
1350 E->getOp() == AtomicExpr::AO__scoped_atomic_load_n;
1351
1352 if (isa<llvm::ConstantInt>(Order)) {
1353 auto ord = cast<llvm::ConstantInt>(Order)->getZExtValue();
1354 // We should not ever get to a case where the ordering isn't a valid C ABI
1355 // value, but it's hard to enforce that in general.
1356 if (llvm::isValidAtomicOrderingCABI(ord))
1357 switch ((llvm::AtomicOrderingCABI)ord) {
1358 case llvm::AtomicOrderingCABI::relaxed:
1359 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1360 OrderFail, Size, llvm::AtomicOrdering::Monotonic, Scope);
1361 break;
1362 case llvm::AtomicOrderingCABI::consume:
1363 case llvm::AtomicOrderingCABI::acquire:
1364 if (IsStore)
1365 break; // Avoid crashing on code with undefined behavior
1366 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1367 OrderFail, Size, llvm::AtomicOrdering::Acquire, Scope);
1368 break;
1369 case llvm::AtomicOrderingCABI::release:
1370 if (IsLoad)
1371 break; // Avoid crashing on code with undefined behavior
1372 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1373 OrderFail, Size, llvm::AtomicOrdering::Release, Scope);
1374 break;
1375 case llvm::AtomicOrderingCABI::acq_rel:
1376 if (IsLoad || IsStore)
1377 break; // Avoid crashing on code with undefined behavior
1378 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1379 OrderFail, Size, llvm::AtomicOrdering::AcquireRelease,
1380 Scope);
1381 break;
1382 case llvm::AtomicOrderingCABI::seq_cst:
1383 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1384 OrderFail, Size,
1385 llvm::AtomicOrdering::SequentiallyConsistent, Scope);
1386 break;
1387 }
1388 if (RValTy->isVoidType())
1389 return RValue::get(nullptr);
1390
1392 RValTy, E->getExprLoc());
1393 }
1394
1395 // Long case, when Order isn't obviously constant.
1396
1397 // Create all the relevant BB's
1398 llvm::BasicBlock *MonotonicBB = nullptr, *AcquireBB = nullptr,
1399 *ReleaseBB = nullptr, *AcqRelBB = nullptr,
1400 *SeqCstBB = nullptr;
1401 MonotonicBB = createBasicBlock("monotonic", CurFn);
1402 if (!IsStore)
1403 AcquireBB = createBasicBlock("acquire", CurFn);
1404 if (!IsLoad)
1405 ReleaseBB = createBasicBlock("release", CurFn);
1406 if (!IsLoad && !IsStore)
1407 AcqRelBB = createBasicBlock("acqrel", CurFn);
1408 SeqCstBB = createBasicBlock("seqcst", CurFn);
1409 llvm::BasicBlock *ContBB = createBasicBlock("atomic.continue", CurFn);
1410
1411 // Create the switch for the split
1412 // MonotonicBB is arbitrarily chosen as the default case; in practice, this
1413 // doesn't matter unless someone is crazy enough to use something that
1414 // doesn't fold to a constant for the ordering.
1415 Order = Builder.CreateIntCast(Order, Builder.getInt32Ty(), false);
1416 llvm::SwitchInst *SI = Builder.CreateSwitch(Order, MonotonicBB);
1417
1418 // Emit all the different atomics
1419 Builder.SetInsertPoint(MonotonicBB);
1420 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak, OrderFail,
1421 Size, llvm::AtomicOrdering::Monotonic, Scope);
1422 Builder.CreateBr(ContBB);
1423 if (!IsStore) {
1424 Builder.SetInsertPoint(AcquireBB);
1425 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1426 OrderFail, Size, llvm::AtomicOrdering::Acquire, Scope);
1427 Builder.CreateBr(ContBB);
1428 SI->addCase(Builder.getInt32((int)llvm::AtomicOrderingCABI::consume),
1429 AcquireBB);
1430 SI->addCase(Builder.getInt32((int)llvm::AtomicOrderingCABI::acquire),
1431 AcquireBB);
1432 }
1433 if (!IsLoad) {
1434 Builder.SetInsertPoint(ReleaseBB);
1435 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1436 OrderFail, Size, llvm::AtomicOrdering::Release, Scope);
1437 Builder.CreateBr(ContBB);
1438 SI->addCase(Builder.getInt32((int)llvm::AtomicOrderingCABI::release),
1439 ReleaseBB);
1440 }
1441 if (!IsLoad && !IsStore) {
1442 Builder.SetInsertPoint(AcqRelBB);
1443 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak,
1444 OrderFail, Size, llvm::AtomicOrdering::AcquireRelease, Scope);
1445 Builder.CreateBr(ContBB);
1446 SI->addCase(Builder.getInt32((int)llvm::AtomicOrderingCABI::acq_rel),
1447 AcqRelBB);
1448 }
1449 Builder.SetInsertPoint(SeqCstBB);
1450 EmitAtomicOp(*this, E, Dest, Ptr, Val1, Val2, OriginalVal1, IsWeak, OrderFail,
1451 Size, llvm::AtomicOrdering::SequentiallyConsistent, Scope);
1452 Builder.CreateBr(ContBB);
1453 SI->addCase(Builder.getInt32((int)llvm::AtomicOrderingCABI::seq_cst),
1454 SeqCstBB);
1455
1456 // Cleanup and return
1457 Builder.SetInsertPoint(ContBB);
1458 if (RValTy->isVoidType())
1459 return RValue::get(nullptr);
1460
1461 assert(Atomics.getValueSizeInBits() <= Atomics.getAtomicSizeInBits());
1463 RValTy, E->getExprLoc());
1464}
1465
1466Address AtomicInfo::castToAtomicIntPointer(Address addr) const {
1467 llvm::IntegerType *ty =
1468 llvm::IntegerType::get(CGF.getLLVMContext(), AtomicSizeInBits);
1469 return addr.withElementType(ty);
1470}
1471
1472Address AtomicInfo::convertToAtomicIntPointer(Address Addr) const {
1473 llvm::Type *Ty = Addr.getElementType();
1474 uint64_t SourceSizeInBits = CGF.CGM.getDataLayout().getTypeSizeInBits(Ty);
1475 if (SourceSizeInBits != AtomicSizeInBits) {
1476 Address Tmp = CreateTempAlloca();
1478 Tmp.emitRawPointer(CGF), llvm::ConstantInt::get(CGF.Int8Ty, 0),
1479 CGF.getContext().toCharUnitsFromBits(AtomicSizeInBits).getQuantity(),
1480 Tmp.getAlignment().getAsAlign());
1481
1482 CGF.Builder.CreateMemCpy(Tmp, Addr,
1483 std::min(AtomicSizeInBits, SourceSizeInBits) / 8);
1484 Addr = Tmp;
1485 }
1486
1487 return castToAtomicIntPointer(Addr);
1488}
1489
1490RValue AtomicInfo::convertAtomicTempToRValue(Address addr,
1491 AggValueSlot resultSlot,
1492 SourceLocation loc,
1493 bool asValue) const {
1494 if (LVal.isSimple()) {
1495 if (EvaluationKind == TEK_Aggregate)
1496 return resultSlot.asRValue();
1497
1498 // Drill into the padding structure if we have one.
1499 if (hasPadding())
1500 addr = CGF.Builder.CreateStructGEP(addr, 0);
1501
1502 // Otherwise, just convert the temporary to an r-value using the
1503 // normal conversion routine.
1504 return CGF.convertTempToRValue(addr, getValueType(), loc);
1505 }
1506 if (!asValue)
1507 // Get RValue from temp memory as atomic for non-simple lvalues
1508 return RValue::get(CGF.Builder.CreateLoad(addr));
1509 if (LVal.isBitField())
1510 return CGF.EmitLoadOfBitfieldLValue(
1511 LValue::MakeBitfield(addr, LVal.getBitFieldInfo(), LVal.getType(),
1512 LVal.getBaseInfo(), TBAAAccessInfo()), loc);
1513 if (LVal.isVectorElt())
1514 return CGF.EmitLoadOfLValue(
1515 LValue::MakeVectorElt(addr, LVal.getVectorIdx(), LVal.getType(),
1516 LVal.getBaseInfo(), TBAAAccessInfo()), loc);
1517 assert(LVal.isExtVectorElt());
1518 return CGF.EmitLoadOfExtVectorElementLValue(LValue::MakeExtVectorElt(
1519 addr, LVal.getExtVectorElts(), LVal.getType(),
1520 LVal.getBaseInfo(), TBAAAccessInfo()));
1521}
1522
1523RValue AtomicInfo::ConvertToValueOrAtomic(llvm::Value *Val,
1524 AggValueSlot ResultSlot,
1525 SourceLocation Loc, bool AsValue,
1526 bool CmpXchg) const {
1527 // Try not to in some easy cases.
1528 assert((Val->getType()->isIntegerTy() || Val->getType()->isPointerTy() ||
1529 Val->getType()->isIEEELikeFPTy()) &&
1530 "Expected integer, pointer or floating point value when converting "
1531 "result.");
1532 if (getEvaluationKind() == TEK_Scalar &&
1533 (((!LVal.isBitField() ||
1534 LVal.getBitFieldInfo().Size == ValueSizeInBits) &&
1535 !hasPadding()) ||
1536 !AsValue)) {
1537 auto *ValTy = AsValue
1538 ? CGF.ConvertTypeForMem(ValueTy)
1539 : getAtomicAddress().getElementType();
1540 if (!shouldCastToInt(ValTy, CmpXchg)) {
1541 assert((!ValTy->isIntegerTy() || Val->getType() == ValTy) &&
1542 "Different integer types.");
1543 return RValue::get(CGF.EmitFromMemory(Val, ValueTy));
1544 }
1545 if (llvm::CastInst::isBitCastable(Val->getType(), ValTy))
1546 return RValue::get(CGF.Builder.CreateBitCast(Val, ValTy));
1547 }
1548
1549 // Create a temporary. This needs to be big enough to hold the
1550 // atomic integer.
1551 Address Temp = Address::invalid();
1552 bool TempIsVolatile = false;
1553 if (AsValue && getEvaluationKind() == TEK_Aggregate) {
1554 assert(!ResultSlot.isIgnored());
1555 Temp = ResultSlot.getAddress();
1556 TempIsVolatile = ResultSlot.isVolatile();
1557 } else {
1558 Temp = CreateTempAlloca();
1559 }
1560
1561 // Slam the integer into the temporary.
1562 Address CastTemp = castToAtomicIntPointer(Temp);
1563 CGF.Builder.CreateStore(Val, CastTemp)->setVolatile(TempIsVolatile);
1564
1565 return convertAtomicTempToRValue(Temp, ResultSlot, Loc, AsValue);
1566}
1567
1568void AtomicInfo::EmitAtomicLoadLibcall(llvm::Value *AddForLoaded,
1569 llvm::AtomicOrdering AO, bool) {
1570 // void __atomic_load(size_t size, void *mem, void *return, int order);
1571 CallArgList Args;
1572 Args.add(RValue::get(getAtomicSizeValue()), CGF.getContext().getSizeType());
1573 Args.add(RValue::get(getAtomicPointer()), CGF.getContext().VoidPtrTy);
1574 Args.add(RValue::get(AddForLoaded), CGF.getContext().VoidPtrTy);
1575 Args.add(
1576 RValue::get(llvm::ConstantInt::get(CGF.IntTy, (int)llvm::toCABI(AO))),
1577 CGF.getContext().IntTy);
1578 emitAtomicLibcall(CGF, "__atomic_load", CGF.getContext().VoidTy, Args);
1579}
1580
1581llvm::Value *AtomicInfo::EmitAtomicLoadOp(llvm::AtomicOrdering AO,
1582 bool IsVolatile, bool CmpXchg) {
1583 // Okay, we're doing this natively.
1584 Address Addr = getAtomicAddress();
1585 if (shouldCastToInt(Addr.getElementType(), CmpXchg))
1586 Addr = castToAtomicIntPointer(Addr);
1587 llvm::LoadInst *Load = CGF.Builder.CreateLoad(Addr, "atomic-load");
1588 Load->setAtomic(AO);
1589
1590 // Other decoration.
1591 if (IsVolatile)
1592 Load->setVolatile(true);
1593 CGF.CGM.DecorateInstructionWithTBAA(Load, LVal.getTBAAInfo());
1594 return Load;
1595}
1596
1597/// An LValue is a candidate for having its loads and stores be made atomic if
1598/// we are operating under /volatile:ms *and* the LValue itself is volatile and
1599/// performing such an operation can be performed without a libcall.
1601 if (!CGM.getLangOpts().MSVolatile) return false;
1602 AtomicInfo AI(*this, LV);
1603 bool IsVolatile = LV.isVolatile() || hasVolatileMember(LV.getType());
1604 // An atomic is inline if we don't need to use a libcall.
1605 bool AtomicIsInline = !AI.shouldUseLibcall();
1606 // MSVC doesn't seem to do this for types wider than a pointer.
1607 if (getContext().getTypeSize(LV.getType()) >
1608 getContext().getTypeSize(getContext().getIntPtrType()))
1609 return false;
1610 return IsVolatile && AtomicIsInline;
1611}
1612
1614 AggValueSlot Slot) {
1615 llvm::AtomicOrdering AO;
1616 bool IsVolatile = LV.isVolatileQualified();
1617 if (LV.getType()->isAtomicType()) {
1618 AO = llvm::AtomicOrdering::SequentiallyConsistent;
1619 } else {
1620 AO = llvm::AtomicOrdering::Acquire;
1621 IsVolatile = true;
1622 }
1623 return EmitAtomicLoad(LV, SL, AO, IsVolatile, Slot);
1624}
1625
1626RValue AtomicInfo::EmitAtomicLoad(AggValueSlot ResultSlot, SourceLocation Loc,
1627 bool AsValue, llvm::AtomicOrdering AO,
1628 bool IsVolatile) {
1629 // Check whether we should use a library call.
1630 if (shouldUseLibcall()) {
1631 Address TempAddr = Address::invalid();
1632 if (LVal.isSimple() && !ResultSlot.isIgnored()) {
1633 assert(getEvaluationKind() == TEK_Aggregate);
1634 TempAddr = ResultSlot.getAddress();
1635 } else
1636 TempAddr = CreateTempAlloca();
1637
1638 EmitAtomicLoadLibcall(TempAddr.emitRawPointer(CGF), AO, IsVolatile);
1639
1640 // Okay, turn that back into the original value or whole atomic (for
1641 // non-simple lvalues) type.
1642 return convertAtomicTempToRValue(TempAddr, ResultSlot, Loc, AsValue);
1643 }
1644
1645 // Okay, we're doing this natively.
1646 auto *Load = EmitAtomicLoadOp(AO, IsVolatile);
1647
1648 // If we're ignoring an aggregate return, don't do anything.
1649 if (getEvaluationKind() == TEK_Aggregate && ResultSlot.isIgnored())
1650 return RValue::getAggregate(Address::invalid(), false);
1651
1652 // Okay, turn that back into the original value or atomic (for non-simple
1653 // lvalues) type.
1654 return ConvertToValueOrAtomic(Load, ResultSlot, Loc, AsValue);
1655}
1656
1657/// Emit a load from an l-value of atomic type. Note that the r-value
1658/// we produce is an r-value of the atomic *value* type.
1660 llvm::AtomicOrdering AO, bool IsVolatile,
1661 AggValueSlot resultSlot) {
1662 AtomicInfo Atomics(*this, src);
1663 return Atomics.EmitAtomicLoad(resultSlot, loc, /*AsValue=*/true, AO,
1664 IsVolatile);
1665}
1666
1667/// Copy an r-value into memory as part of storing to an atomic type.
1668/// This needs to create a bit-pattern suitable for atomic operations.
1669void AtomicInfo::emitCopyIntoMemory(RValue rvalue) const {
1670 assert(LVal.isSimple());
1671 // If we have an r-value, the rvalue should be of the atomic type,
1672 // which means that the caller is responsible for having zeroed
1673 // any padding. Just do an aggregate copy of that type.
1674 if (rvalue.isAggregate()) {
1675 LValue Dest = CGF.MakeAddrLValue(getAtomicAddress(), getAtomicType());
1676 LValue Src = CGF.MakeAddrLValue(rvalue.getAggregateAddress(),
1677 getAtomicType());
1678 bool IsVolatile = rvalue.isVolatileQualified() ||
1679 LVal.isVolatileQualified();
1680 CGF.EmitAggregateCopy(Dest, Src, getAtomicType(),
1681 AggValueSlot::DoesNotOverlap, IsVolatile);
1682 return;
1683 }
1684
1685 // Okay, otherwise we're copying stuff.
1686
1687 // Zero out the buffer if necessary.
1688 emitMemSetZeroIfNecessary();
1689
1690 // Drill past the padding if present.
1691 LValue TempLVal = projectValue();
1692
1693 // Okay, store the rvalue in.
1694 if (rvalue.isScalar()) {
1695 CGF.EmitStoreOfScalar(rvalue.getScalarVal(), TempLVal, /*init*/ true);
1696 } else {
1697 CGF.EmitStoreOfComplex(rvalue.getComplexVal(), TempLVal, /*init*/ true);
1698 }
1699}
1700
1701
1702/// Materialize an r-value into memory for the purposes of storing it
1703/// to an atomic type.
1704Address AtomicInfo::materializeRValue(RValue rvalue) const {
1705 // Aggregate r-values are already in memory, and EmitAtomicStore
1706 // requires them to be values of the atomic type.
1707 if (rvalue.isAggregate())
1708 return rvalue.getAggregateAddress();
1709
1710 // Otherwise, make a temporary and materialize into it.
1711 LValue TempLV = CGF.MakeAddrLValue(CreateTempAlloca(), getAtomicType());
1712 AtomicInfo Atomics(CGF, TempLV);
1713 Atomics.emitCopyIntoMemory(rvalue);
1714 return TempLV.getAddress();
1715}
1716
1717llvm::Value *AtomicInfo::getScalarRValValueOrNull(RValue RVal) const {
1718 if (RVal.isScalar() && (!hasPadding() || !LVal.isSimple()))
1719 return RVal.getScalarVal();
1720 return nullptr;
1721}
1722
1723llvm::Value *AtomicInfo::convertRValueToInt(RValue RVal, bool CmpXchg) const {
1724 // If we've got a scalar value of the right size, try to avoid going
1725 // through memory. Floats get casted if needed by AtomicExpandPass.
1726 if (llvm::Value *Value = getScalarRValValueOrNull(RVal)) {
1727 if (!shouldCastToInt(Value->getType(), CmpXchg))
1728 return CGF.EmitToMemory(Value, ValueTy);
1729 else {
1730 llvm::IntegerType *InputIntTy = llvm::IntegerType::get(
1731 CGF.getLLVMContext(),
1732 LVal.isSimple() ? getValueSizeInBits() : getAtomicSizeInBits());
1733 if (llvm::BitCastInst::isBitCastable(Value->getType(), InputIntTy))
1734 return CGF.Builder.CreateBitCast(Value, InputIntTy);
1735 }
1736 }
1737 // Otherwise, we need to go through memory.
1738 // Put the r-value in memory.
1739 Address Addr = materializeRValue(RVal);
1740
1741 // Cast the temporary to the atomic int type and pull a value out.
1742 Addr = castToAtomicIntPointer(Addr);
1743 return CGF.Builder.CreateLoad(Addr);
1744}
1745
1746std::pair<llvm::Value *, llvm::Value *> AtomicInfo::EmitAtomicCompareExchangeOp(
1747 llvm::Value *ExpectedVal, llvm::Value *DesiredVal,
1748 llvm::AtomicOrdering Success, llvm::AtomicOrdering Failure, bool IsWeak) {
1749 // Do the atomic store.
1750 Address Addr = getAtomicAddressAsAtomicIntPointer();
1751 auto *Inst = CGF.Builder.CreateAtomicCmpXchg(Addr, ExpectedVal, DesiredVal,
1752 Success, Failure);
1753 // Other decoration.
1754 Inst->setVolatile(LVal.isVolatileQualified());
1755 Inst->setWeak(IsWeak);
1756
1757 // Okay, turn that back into the original value type.
1758 auto *PreviousVal = CGF.Builder.CreateExtractValue(Inst, /*Idxs=*/0);
1759 auto *SuccessFailureVal = CGF.Builder.CreateExtractValue(Inst, /*Idxs=*/1);
1760 return std::make_pair(PreviousVal, SuccessFailureVal);
1761}
1762
1763llvm::Value *
1764AtomicInfo::EmitAtomicCompareExchangeLibcall(llvm::Value *ExpectedAddr,
1765 llvm::Value *DesiredAddr,
1766 llvm::AtomicOrdering Success,
1767 llvm::AtomicOrdering Failure) {
1768 // bool __atomic_compare_exchange(size_t size, void *obj, void *expected,
1769 // void *desired, int success, int failure);
1770 CallArgList Args;
1771 Args.add(RValue::get(getAtomicSizeValue()), CGF.getContext().getSizeType());
1772 Args.add(RValue::get(getAtomicPointer()), CGF.getContext().VoidPtrTy);
1773 Args.add(RValue::get(ExpectedAddr), CGF.getContext().VoidPtrTy);
1774 Args.add(RValue::get(DesiredAddr), CGF.getContext().VoidPtrTy);
1775 Args.add(RValue::get(
1776 llvm::ConstantInt::get(CGF.IntTy, (int)llvm::toCABI(Success))),
1777 CGF.getContext().IntTy);
1778 Args.add(RValue::get(
1779 llvm::ConstantInt::get(CGF.IntTy, (int)llvm::toCABI(Failure))),
1780 CGF.getContext().IntTy);
1781 auto SuccessFailureRVal = emitAtomicLibcall(CGF, "__atomic_compare_exchange",
1782 CGF.getContext().BoolTy, Args);
1783
1784 return SuccessFailureRVal.getScalarVal();
1785}
1786
1787std::pair<RValue, llvm::Value *> AtomicInfo::EmitAtomicCompareExchange(
1788 RValue Expected, RValue Desired, llvm::AtomicOrdering Success,
1789 llvm::AtomicOrdering Failure, bool IsWeak) {
1790 // Check whether we should use a library call.
1791 if (shouldUseLibcall()) {
1792 // Produce a source address.
1793 Address ExpectedAddr = materializeRValue(Expected);
1794 llvm::Value *ExpectedPtr = ExpectedAddr.emitRawPointer(CGF);
1795 llvm::Value *DesiredPtr = materializeRValue(Desired).emitRawPointer(CGF);
1796 auto *Res = EmitAtomicCompareExchangeLibcall(ExpectedPtr, DesiredPtr,
1797 Success, Failure);
1798 return std::make_pair(
1799 convertAtomicTempToRValue(ExpectedAddr, AggValueSlot::ignored(),
1800 SourceLocation(), /*AsValue=*/false),
1801 Res);
1802 }
1803
1804 // If we've got a scalar value of the right size, try to avoid going
1805 // through memory.
1806 auto *ExpectedVal = convertRValueToInt(Expected, /*CmpXchg=*/true);
1807 auto *DesiredVal = convertRValueToInt(Desired, /*CmpXchg=*/true);
1808 auto Res = EmitAtomicCompareExchangeOp(ExpectedVal, DesiredVal, Success,
1809 Failure, IsWeak);
1810 return std::make_pair(
1811 ConvertToValueOrAtomic(Res.first, AggValueSlot::ignored(),
1812 SourceLocation(), /*AsValue=*/false,
1813 /*CmpXchg=*/true),
1814 Res.second);
1815}
1816
1817static void
1818EmitAtomicUpdateValue(CodeGenFunction &CGF, AtomicInfo &Atomics, RValue OldRVal,
1819 const llvm::function_ref<RValue(RValue)> &UpdateOp,
1820 Address DesiredAddr) {
1821 RValue UpRVal;
1822 LValue AtomicLVal = Atomics.getAtomicLValue();
1823 LValue DesiredLVal;
1824 if (AtomicLVal.isSimple()) {
1825 UpRVal = OldRVal;
1826 DesiredLVal = CGF.MakeAddrLValue(DesiredAddr, AtomicLVal.getType());
1827 } else {
1828 // Build new lvalue for temp address.
1829 Address Ptr = Atomics.materializeRValue(OldRVal);
1830 LValue UpdateLVal;
1831 if (AtomicLVal.isBitField()) {
1832 UpdateLVal =
1833 LValue::MakeBitfield(Ptr, AtomicLVal.getBitFieldInfo(),
1834 AtomicLVal.getType(),
1835 AtomicLVal.getBaseInfo(),
1836 AtomicLVal.getTBAAInfo());
1837 DesiredLVal =
1838 LValue::MakeBitfield(DesiredAddr, AtomicLVal.getBitFieldInfo(),
1839 AtomicLVal.getType(), AtomicLVal.getBaseInfo(),
1840 AtomicLVal.getTBAAInfo());
1841 } else if (AtomicLVal.isVectorElt()) {
1842 UpdateLVal = LValue::MakeVectorElt(Ptr, AtomicLVal.getVectorIdx(),
1843 AtomicLVal.getType(),
1844 AtomicLVal.getBaseInfo(),
1845 AtomicLVal.getTBAAInfo());
1846 DesiredLVal = LValue::MakeVectorElt(
1847 DesiredAddr, AtomicLVal.getVectorIdx(), AtomicLVal.getType(),
1848 AtomicLVal.getBaseInfo(), AtomicLVal.getTBAAInfo());
1849 } else {
1850 assert(AtomicLVal.isExtVectorElt());
1851 UpdateLVal = LValue::MakeExtVectorElt(Ptr, AtomicLVal.getExtVectorElts(),
1852 AtomicLVal.getType(),
1853 AtomicLVal.getBaseInfo(),
1854 AtomicLVal.getTBAAInfo());
1855 DesiredLVal = LValue::MakeExtVectorElt(
1856 DesiredAddr, AtomicLVal.getExtVectorElts(), AtomicLVal.getType(),
1857 AtomicLVal.getBaseInfo(), AtomicLVal.getTBAAInfo());
1858 }
1859 UpRVal = CGF.EmitLoadOfLValue(UpdateLVal, SourceLocation());
1860 }
1861 // Store new value in the corresponding memory area.
1862 RValue NewRVal = UpdateOp(UpRVal);
1863 if (NewRVal.isScalar()) {
1864 CGF.EmitStoreThroughLValue(NewRVal, DesiredLVal);
1865 } else {
1866 assert(NewRVal.isComplex());
1867 CGF.EmitStoreOfComplex(NewRVal.getComplexVal(), DesiredLVal,
1868 /*isInit=*/false);
1869 }
1870}
1871
1872void AtomicInfo::EmitAtomicUpdateLibcall(
1873 llvm::AtomicOrdering AO, const llvm::function_ref<RValue(RValue)> &UpdateOp,
1874 bool IsVolatile) {
1875 auto Failure = llvm::AtomicCmpXchgInst::getStrongestFailureOrdering(AO);
1876
1877 Address ExpectedAddr = CreateTempAlloca();
1878
1879 EmitAtomicLoadLibcall(ExpectedAddr.emitRawPointer(CGF), AO, IsVolatile);
1880 auto *ContBB = CGF.createBasicBlock("atomic_cont");
1881 auto *ExitBB = CGF.createBasicBlock("atomic_exit");
1882 CGF.EmitBlock(ContBB);
1883 Address DesiredAddr = CreateTempAlloca();
1884 if ((LVal.isBitField() && BFI.Size != ValueSizeInBits) ||
1885 requiresMemSetZero(getAtomicAddress().getElementType())) {
1886 auto *OldVal = CGF.Builder.CreateLoad(ExpectedAddr);
1887 CGF.Builder.CreateStore(OldVal, DesiredAddr);
1888 }
1889 auto OldRVal = convertAtomicTempToRValue(ExpectedAddr,
1891 SourceLocation(), /*AsValue=*/false);
1892 EmitAtomicUpdateValue(CGF, *this, OldRVal, UpdateOp, DesiredAddr);
1893 llvm::Value *ExpectedPtr = ExpectedAddr.emitRawPointer(CGF);
1894 llvm::Value *DesiredPtr = DesiredAddr.emitRawPointer(CGF);
1895 auto *Res =
1896 EmitAtomicCompareExchangeLibcall(ExpectedPtr, DesiredPtr, AO, Failure);
1897 CGF.Builder.CreateCondBr(Res, ExitBB, ContBB);
1898 CGF.EmitBlock(ExitBB, /*IsFinished=*/true);
1899}
1900
1901void AtomicInfo::EmitAtomicUpdateOp(
1902 llvm::AtomicOrdering AO, const llvm::function_ref<RValue(RValue)> &UpdateOp,
1903 bool IsVolatile) {
1904 auto Failure = llvm::AtomicCmpXchgInst::getStrongestFailureOrdering(AO);
1905
1906 // Do the atomic load.
1907 auto *OldVal = EmitAtomicLoadOp(Failure, IsVolatile, /*CmpXchg=*/true);
1908 // For non-simple lvalues perform compare-and-swap procedure.
1909 auto *ContBB = CGF.createBasicBlock("atomic_cont");
1910 auto *ExitBB = CGF.createBasicBlock("atomic_exit");
1911 auto *CurBB = CGF.Builder.GetInsertBlock();
1912 CGF.EmitBlock(ContBB);
1913 llvm::PHINode *PHI = CGF.Builder.CreatePHI(OldVal->getType(),
1914 /*NumReservedValues=*/2);
1915 PHI->addIncoming(OldVal, CurBB);
1916 Address NewAtomicAddr = CreateTempAlloca();
1917 Address NewAtomicIntAddr =
1918 shouldCastToInt(NewAtomicAddr.getElementType(), /*CmpXchg=*/true)
1919 ? castToAtomicIntPointer(NewAtomicAddr)
1920 : NewAtomicAddr;
1921
1922 if ((LVal.isBitField() && BFI.Size != ValueSizeInBits) ||
1923 requiresMemSetZero(getAtomicAddress().getElementType())) {
1924 CGF.Builder.CreateStore(PHI, NewAtomicIntAddr);
1925 }
1926 auto OldRVal = ConvertToValueOrAtomic(PHI, AggValueSlot::ignored(),
1927 SourceLocation(), /*AsValue=*/false,
1928 /*CmpXchg=*/true);
1929 EmitAtomicUpdateValue(CGF, *this, OldRVal, UpdateOp, NewAtomicAddr);
1930 auto *DesiredVal = CGF.Builder.CreateLoad(NewAtomicIntAddr);
1931 // Try to write new value using cmpxchg operation.
1932 auto Res = EmitAtomicCompareExchangeOp(PHI, DesiredVal, AO, Failure);
1933 PHI->addIncoming(Res.first, CGF.Builder.GetInsertBlock());
1934 CGF.Builder.CreateCondBr(Res.second, ExitBB, ContBB);
1935 CGF.EmitBlock(ExitBB, /*IsFinished=*/true);
1936}
1937
1938static void EmitAtomicUpdateValue(CodeGenFunction &CGF, AtomicInfo &Atomics,
1939 RValue UpdateRVal, Address DesiredAddr) {
1940 LValue AtomicLVal = Atomics.getAtomicLValue();
1941 LValue DesiredLVal;
1942 // Build new lvalue for temp address.
1943 if (AtomicLVal.isBitField()) {
1944 DesiredLVal =
1945 LValue::MakeBitfield(DesiredAddr, AtomicLVal.getBitFieldInfo(),
1946 AtomicLVal.getType(), AtomicLVal.getBaseInfo(),
1947 AtomicLVal.getTBAAInfo());
1948 } else if (AtomicLVal.isVectorElt()) {
1949 DesiredLVal =
1950 LValue::MakeVectorElt(DesiredAddr, AtomicLVal.getVectorIdx(),
1951 AtomicLVal.getType(), AtomicLVal.getBaseInfo(),
1952 AtomicLVal.getTBAAInfo());
1953 } else {
1954 assert(AtomicLVal.isExtVectorElt());
1955 DesiredLVal = LValue::MakeExtVectorElt(
1956 DesiredAddr, AtomicLVal.getExtVectorElts(), AtomicLVal.getType(),
1957 AtomicLVal.getBaseInfo(), AtomicLVal.getTBAAInfo());
1958 }
1959 // Store new value in the corresponding memory area.
1960 assert(UpdateRVal.isScalar());
1961 CGF.EmitStoreThroughLValue(UpdateRVal, DesiredLVal);
1962}
1963
1964void AtomicInfo::EmitAtomicUpdateLibcall(llvm::AtomicOrdering AO,
1965 RValue UpdateRVal, bool IsVolatile) {
1966 auto Failure = llvm::AtomicCmpXchgInst::getStrongestFailureOrdering(AO);
1967
1968 Address ExpectedAddr = CreateTempAlloca();
1969
1970 EmitAtomicLoadLibcall(ExpectedAddr.emitRawPointer(CGF), AO, IsVolatile);
1971 auto *ContBB = CGF.createBasicBlock("atomic_cont");
1972 auto *ExitBB = CGF.createBasicBlock("atomic_exit");
1973 CGF.EmitBlock(ContBB);
1974 Address DesiredAddr = CreateTempAlloca();
1975 if ((LVal.isBitField() && BFI.Size != ValueSizeInBits) ||
1976 requiresMemSetZero(getAtomicAddress().getElementType())) {
1977 auto *OldVal = CGF.Builder.CreateLoad(ExpectedAddr);
1978 CGF.Builder.CreateStore(OldVal, DesiredAddr);
1979 }
1980 EmitAtomicUpdateValue(CGF, *this, UpdateRVal, DesiredAddr);
1981 llvm::Value *ExpectedPtr = ExpectedAddr.emitRawPointer(CGF);
1982 llvm::Value *DesiredPtr = DesiredAddr.emitRawPointer(CGF);
1983 auto *Res =
1984 EmitAtomicCompareExchangeLibcall(ExpectedPtr, DesiredPtr, AO, Failure);
1985 CGF.Builder.CreateCondBr(Res, ExitBB, ContBB);
1986 CGF.EmitBlock(ExitBB, /*IsFinished=*/true);
1987}
1988
1989void AtomicInfo::EmitAtomicUpdateOp(llvm::AtomicOrdering AO, RValue UpdateRVal,
1990 bool IsVolatile) {
1991 auto Failure = llvm::AtomicCmpXchgInst::getStrongestFailureOrdering(AO);
1992
1993 // Do the atomic load.
1994 auto *OldVal = EmitAtomicLoadOp(Failure, IsVolatile, /*CmpXchg=*/true);
1995 // For non-simple lvalues perform compare-and-swap procedure.
1996 auto *ContBB = CGF.createBasicBlock("atomic_cont");
1997 auto *ExitBB = CGF.createBasicBlock("atomic_exit");
1998 auto *CurBB = CGF.Builder.GetInsertBlock();
1999 CGF.EmitBlock(ContBB);
2000 llvm::PHINode *PHI = CGF.Builder.CreatePHI(OldVal->getType(),
2001 /*NumReservedValues=*/2);
2002 PHI->addIncoming(OldVal, CurBB);
2003 Address NewAtomicAddr = CreateTempAlloca();
2004 Address NewAtomicIntAddr = castToAtomicIntPointer(NewAtomicAddr);
2005 if ((LVal.isBitField() && BFI.Size != ValueSizeInBits) ||
2006 requiresMemSetZero(getAtomicAddress().getElementType())) {
2007 CGF.Builder.CreateStore(PHI, NewAtomicIntAddr);
2008 }
2009 EmitAtomicUpdateValue(CGF, *this, UpdateRVal, NewAtomicAddr);
2010 auto *DesiredVal = CGF.Builder.CreateLoad(NewAtomicIntAddr);
2011 // Try to write new value using cmpxchg operation.
2012 auto Res = EmitAtomicCompareExchangeOp(PHI, DesiredVal, AO, Failure);
2013 PHI->addIncoming(Res.first, CGF.Builder.GetInsertBlock());
2014 CGF.Builder.CreateCondBr(Res.second, ExitBB, ContBB);
2015 CGF.EmitBlock(ExitBB, /*IsFinished=*/true);
2016}
2017
2018void AtomicInfo::EmitAtomicUpdate(
2019 llvm::AtomicOrdering AO, const llvm::function_ref<RValue(RValue)> &UpdateOp,
2020 bool IsVolatile) {
2021 if (shouldUseLibcall()) {
2022 EmitAtomicUpdateLibcall(AO, UpdateOp, IsVolatile);
2023 } else {
2024 EmitAtomicUpdateOp(AO, UpdateOp, IsVolatile);
2025 }
2026}
2027
2028void AtomicInfo::EmitAtomicUpdate(llvm::AtomicOrdering AO, RValue UpdateRVal,
2029 bool IsVolatile) {
2030 if (shouldUseLibcall()) {
2031 EmitAtomicUpdateLibcall(AO, UpdateRVal, IsVolatile);
2032 } else {
2033 EmitAtomicUpdateOp(AO, UpdateRVal, IsVolatile);
2034 }
2035}
2036
2038 bool isInit) {
2039 bool IsVolatile = lvalue.isVolatileQualified();
2040 llvm::AtomicOrdering AO;
2041 if (lvalue.getType()->isAtomicType()) {
2042 AO = llvm::AtomicOrdering::SequentiallyConsistent;
2043 } else {
2044 AO = llvm::AtomicOrdering::Release;
2045 IsVolatile = true;
2046 }
2047 return EmitAtomicStore(rvalue, lvalue, AO, IsVolatile, isInit);
2048}
2049
2050/// Emit a store to an l-value of atomic type.
2051///
2052/// Note that the r-value is expected to be an r-value *of the atomic
2053/// type*; this means that for aggregate r-values, it should include
2054/// storage for any padding that was necessary.
2056 llvm::AtomicOrdering AO, bool IsVolatile,
2057 bool isInit) {
2058 // If this is an aggregate r-value, it should agree in type except
2059 // maybe for address-space qualification.
2060 assert(!rvalue.isAggregate() ||
2062 dest.getAddress().getElementType());
2063
2064 AtomicInfo atomics(*this, dest);
2065 LValue LVal = atomics.getAtomicLValue();
2066
2067 // If this is an initialization, just put the value there normally.
2068 if (LVal.isSimple()) {
2069 if (isInit) {
2070 atomics.emitCopyIntoMemory(rvalue);
2071 return;
2072 }
2073
2074 // Check whether we should use a library call.
2075 if (atomics.shouldUseLibcall()) {
2076 // Produce a source address.
2077 Address srcAddr = atomics.materializeRValue(rvalue);
2078
2079 // void __atomic_store(size_t size, void *mem, void *val, int order)
2080 CallArgList args;
2081 args.add(RValue::get(atomics.getAtomicSizeValue()),
2082 getContext().getSizeType());
2083 args.add(RValue::get(atomics.getAtomicPointer()), getContext().VoidPtrTy);
2084 args.add(RValue::get(srcAddr.emitRawPointer(*this)),
2086 args.add(
2087 RValue::get(llvm::ConstantInt::get(IntTy, (int)llvm::toCABI(AO))),
2088 getContext().IntTy);
2089 emitAtomicLibcall(*this, "__atomic_store", getContext().VoidTy, args);
2090 return;
2091 }
2092
2093 // Okay, we're doing this natively.
2094 llvm::Value *ValToStore = atomics.convertRValueToInt(rvalue);
2095
2096 // Do the atomic store.
2097 Address Addr = atomics.getAtomicAddress();
2098 if (llvm::Value *Value = atomics.getScalarRValValueOrNull(rvalue))
2099 if (shouldCastToInt(Value->getType(), /*CmpXchg=*/false)) {
2100 Addr = atomics.castToAtomicIntPointer(Addr);
2101 ValToStore = Builder.CreateIntCast(ValToStore, Addr.getElementType(),
2102 /*isSigned=*/false);
2103 }
2104 llvm::StoreInst *store = Builder.CreateStore(ValToStore, Addr);
2105
2106 if (AO == llvm::AtomicOrdering::Acquire)
2107 AO = llvm::AtomicOrdering::Monotonic;
2108 else if (AO == llvm::AtomicOrdering::AcquireRelease)
2109 AO = llvm::AtomicOrdering::Release;
2110 // Initializations don't need to be atomic.
2111 if (!isInit)
2112 store->setAtomic(AO);
2113
2114 // Other decoration.
2115 if (IsVolatile)
2116 store->setVolatile(true);
2117 CGM.DecorateInstructionWithTBAA(store, dest.getTBAAInfo());
2118 return;
2119 }
2120
2121 // Emit simple atomic update operation.
2122 atomics.EmitAtomicUpdate(AO, rvalue, IsVolatile);
2123}
2124
2125/// Emit a compare-and-exchange op for atomic type.
2126///
2127std::pair<RValue, llvm::Value *> CodeGenFunction::EmitAtomicCompareExchange(
2128 LValue Obj, RValue Expected, RValue Desired, SourceLocation Loc,
2129 llvm::AtomicOrdering Success, llvm::AtomicOrdering Failure, bool IsWeak,
2130 AggValueSlot Slot) {
2131 // If this is an aggregate r-value, it should agree in type except
2132 // maybe for address-space qualification.
2133 assert(!Expected.isAggregate() ||
2134 Expected.getAggregateAddress().getElementType() ==
2135 Obj.getAddress().getElementType());
2136 assert(!Desired.isAggregate() ||
2137 Desired.getAggregateAddress().getElementType() ==
2138 Obj.getAddress().getElementType());
2139 AtomicInfo Atomics(*this, Obj);
2140
2141 return Atomics.EmitAtomicCompareExchange(Expected, Desired, Success, Failure,
2142 IsWeak);
2143}
2144
2145llvm::AtomicRMWInst *
2146CodeGenFunction::emitAtomicRMWInst(llvm::AtomicRMWInst::BinOp Op, Address Addr,
2147 llvm::Value *Val, llvm::AtomicOrdering Order,
2148 llvm::SyncScope::ID SSID,
2149 const AtomicExpr *AE) {
2150 llvm::AtomicRMWInst *RMW =
2151 Builder.CreateAtomicRMW(Op, Addr, Val, Order, SSID);
2152 getTargetHooks().setTargetAtomicMetadata(*this, *RMW, AE);
2153 return RMW;
2154}
2155
2157 LValue LVal, llvm::AtomicOrdering AO,
2158 const llvm::function_ref<RValue(RValue)> &UpdateOp, bool IsVolatile) {
2159 AtomicInfo Atomics(*this, LVal);
2160 Atomics.EmitAtomicUpdate(AO, UpdateOp, IsVolatile);
2161}
2162
2164 AtomicInfo atomics(*this, dest);
2165
2166 switch (atomics.getEvaluationKind()) {
2167 case TEK_Scalar: {
2168 llvm::Value *value = EmitScalarExpr(init);
2169 atomics.emitCopyIntoMemory(RValue::get(value));
2170 return;
2171 }
2172
2173 case TEK_Complex: {
2174 ComplexPairTy value = EmitComplexExpr(init);
2175 atomics.emitCopyIntoMemory(RValue::getComplex(value));
2176 return;
2177 }
2178
2179 case TEK_Aggregate: {
2180 // Fix up the destination if the initializer isn't an expression
2181 // of atomic type.
2182 bool Zeroed = false;
2183 if (!init->getType()->isAtomicType()) {
2184 Zeroed = atomics.emitMemSetZeroIfNecessary();
2185 dest = atomics.projectValue();
2186 }
2187
2188 // Evaluate the expression directly into the destination.
2194
2195 EmitAggExpr(init, slot);
2196 return;
2197 }
2198 }
2199 llvm_unreachable("bad evaluation kind");
2200}
Defines the clang::ASTContext interface.
#define V(N, I)
static llvm::Value * EmitPostAtomicMinMax(CGBuilderTy &Builder, AtomicExpr::AtomicOp Op, bool IsSigned, llvm::Value *OldVal, llvm::Value *RHS)
Duplicate the atomic min/max operation in conventional IR for the builtin variants that return the ne...
Definition CGAtomic.cpp:527
static void EmitAtomicUpdateValue(CodeGenFunction &CGF, AtomicInfo &Atomics, RValue OldRVal, const llvm::function_ref< RValue(RValue)> &UpdateOp, Address DesiredAddr)
static Address EmitValToTemp(CodeGenFunction &CGF, Expr *E)
Definition CGAtomic.cpp:828
static RValue emitAtomicLibcall(CodeGenFunction &CGF, StringRef fnName, QualType resultType, CallArgList &args)
Definition CGAtomic.cpp:316
static void EmitAtomicOp(CodeGenFunction &CGF, AtomicExpr *E, Address Dest, Address Ptr, Address Val1, Address Val2, Address ExpectedResult, llvm::Value *IsWeak, llvm::Value *FailureOrder, uint64_t Size, llvm::AtomicOrdering Order, llvm::SyncScope::ID Scope)
Definition CGAtomic.cpp:561
static Address EmitPointerWithAlignment(const Expr *E, LValueBaseInfo *BaseInfo, TBAAAccessInfo *TBAAInfo, KnownNonNull_t IsKnownNonNull, CodeGenFunction &CGF)
Definition CGExpr.cpp:1458
static bool shouldCastToInt(mlir::Type valueTy, bool cmpxchg)
Return true if.
static void emitAtomicCmpXchg(CIRGenFunction &cgf, AtomicExpr *e, bool isWeak, Address dest, Address ptr, Address val1, Address val2, uint64_t size, cir::MemOrder successOrder, cir::MemOrder failureOrder, cir::SyncScopeKind scope)
static bool isFullSizeType(CIRGenModule &cgm, mlir::Type ty, uint64_t expectedSize)
Does a store of the given IR type modify the full expected width?
static void emitAtomicCmpXchgFailureSet(CIRGenFunction &cgf, AtomicExpr *e, bool isWeak, Address dest, Address ptr, Address val1, Address val2, Expr *failureOrderExpr, uint64_t size, cir::MemOrder successOrder, cir::SyncScopeKind scope)
TokenType getType() const
Returns the token's type, e.g.
Result
Implement __builtin_bit_cast and related operations.
static QualType getPointeeType(const MemRegion *R)
CanQualType VoidPtrTy
CanQualType BoolTy
CanQualType IntTy
CanQualType VoidTy
QualType getSizeType() const
Return the unique type for "size_t" (C99 7.17), defined in <stddef.h>.
QualType getExtVectorType(QualType VectorType, unsigned NumElts) const
Return the unique reference to an extended vector type of the specified element type and size.
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
AtomicExpr - Variadic atomic builtins: __atomic_exchange, __atomic_fetch_*, __atomic_load,...
Definition Expr.h:6928
static std::unique_ptr< AtomicScopeModel > getScopeModel(AtomicOp Op)
Get atomic scope model for the atomic op code.
Definition Expr.h:7077
Expr * getVal2() const
Definition Expr.h:6979
Expr * getOrder() const
Definition Expr.h:6962
QualType getValueType() const
Definition Expr.cpp:5385
Expr * getScope() const
Definition Expr.h:6965
bool isCmpXChg() const
Definition Expr.h:7012
AtomicOp getOp() const
Definition Expr.h:6991
bool isOpenCL() const
Definition Expr.h:7040
Expr * getVal1() const
Definition Expr.h:6969
Expr * getPtr() const
Definition Expr.h:6959
Expr * getWeak() const
Definition Expr.h:6985
SourceLocation getBeginLoc() const LLVM_READONLY
Definition Expr.h:7059
Expr * getOrderFail() const
Definition Expr.h:6975
bool isVolatile() const
Definition Expr.h:7008
CharUnits - This is an opaque type for sizes expressed in character units.
Definition CharUnits.h:38
llvm::Align getAsAlign() const
getAsAlign - Returns Quantity as a valid llvm::Align, Beware llvm::Align assumes power of two 8-bit b...
Definition CharUnits.h:189
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
Definition CharUnits.h:185
bool isMultipleOf(CharUnits N) const
Test whether this is a multiple of the other value.
Definition CharUnits.h:143
Like RawAddress, an abstract representation of an aligned address, but the pointer contained in this ...
Definition Address.h:128
static Address invalid()
Definition Address.h:176
llvm::Value * emitRawPointer(CodeGenFunction &CGF) const
Return the pointer contained in this class after authenticating it and adding offset to it if necessa...
Definition Address.h:253
CharUnits getAlignment() const
Definition Address.h:194
llvm::Type * getElementType() const
Return the type of the values stored in this address.
Definition Address.h:209
Address withElementType(llvm::Type *ElemTy) const
Return address with different element type, but same pointer and alignment.
Definition Address.h:276
bool isValid() const
Definition Address.h:177
An aggregate value slot.
Definition CGValue.h:551
static AggValueSlot ignored()
ignored - Returns an aggregate value slot indicating that the aggregate value is being ignored.
Definition CGValue.h:619
Address getAddress() const
Definition CGValue.h:691
static AggValueSlot forLValue(const LValue &LV, IsDestructed_t isDestructed, NeedsGCBarriers_t needsGC, IsAliased_t isAliased, Overlap_t mayOverlap, IsZeroed_t isZeroed=IsNotZeroed, IsSanitizerChecked_t isChecked=IsNotSanitizerChecked)
Definition CGValue.h:649
RValue asRValue() const
Definition CGValue.h:713
A scoped helper to set the current source atom group for CGDebugInfo::addInstToCurrentSourceAtom.
llvm::StoreInst * CreateStore(llvm::Value *Val, Address Addr, bool IsVolatile=false)
Definition CGBuilder.h:146
Address CreatePointerBitCastOrAddrSpaceCast(Address Addr, llvm::Type *Ty, llvm::Type *ElementTy, const llvm::Twine &Name="")
Definition CGBuilder.h:213
llvm::CallInst * CreateMemSet(Address Dest, llvm::Value *Value, llvm::Value *Size, bool IsVolatile=false)
Definition CGBuilder.h:430
Address CreateStructGEP(Address Addr, unsigned Index, const llvm::Twine &Name="")
Definition CGBuilder.h:229
llvm::AtomicCmpXchgInst * CreateAtomicCmpXchg(Address Addr, llvm::Value *Cmp, llvm::Value *New, llvm::AtomicOrdering SuccessOrdering, llvm::AtomicOrdering FailureOrdering, llvm::SyncScope::ID SSID=llvm::SyncScope::System)
Definition CGBuilder.h:179
llvm::LoadInst * CreateLoad(Address Addr, const llvm::Twine &Name="")
Definition CGBuilder.h:118
llvm::CallInst * CreateMemCpy(Address Dest, Address Src, llvm::Value *Size, bool IsVolatile=false)
Definition CGBuilder.h:397
Address CreateAddrSpaceCast(Address Addr, llvm::Type *Ty, llvm::Type *ElementTy, const llvm::Twine &Name="")
Definition CGBuilder.h:199
static CGCallee forDirect(llvm::Constant *functionPtr, const CGCalleeInfo &abstractInfo=CGCalleeInfo())
Definition CGCall.h:137
CGFunctionInfo - Class to encapsulate the information about a function definition.
CallArgList - Type for representing both the value and type of arguments in a call.
Definition CGCall.h:274
void add(RValue rvalue, QualType type)
Definition CGCall.h:302
CodeGenFunction - This class organizes the per-function state that is used while generating LLVM code...
llvm::Value * performAddrSpaceCast(llvm::Value *Src, llvm::Type *DestTy)
void EmitAtomicInit(Expr *E, LValue lvalue)
RValue convertTempToRValue(Address addr, QualType type, SourceLocation Loc)
Given the address of a temporary variable, produce an r-value of its type.
Definition CGExpr.cpp:7184
bool hasVolatileMember(QualType T)
hasVolatileMember - returns true if aggregate type has a volatile member.
llvm::BasicBlock * createBasicBlock(const Twine &name="", llvm::Function *parent=nullptr, llvm::BasicBlock *before=nullptr)
createBasicBlock - Create an LLVM basic block.
void addInstToCurrentSourceAtom(llvm::Instruction *KeyInstruction, llvm::Value *Backup)
See CGDebugInfo::addInstToCurrentSourceAtom.
const LangOptions & getLangOpts() const
void EmitAtomicUpdate(LValue LVal, llvm::AtomicOrdering AO, const llvm::function_ref< RValue(RValue)> &UpdateOp, bool IsVolatile)
std::pair< RValue, llvm::Value * > EmitAtomicCompareExchange(LValue Obj, RValue Expected, RValue Desired, SourceLocation Loc, llvm::AtomicOrdering Success=llvm::AtomicOrdering::SequentiallyConsistent, llvm::AtomicOrdering Failure=llvm::AtomicOrdering::SequentiallyConsistent, bool IsWeak=false, AggValueSlot Slot=AggValueSlot::ignored())
Emit a compare-and-exchange op for atomic type.
void maybeAttachRangeForLoad(llvm::LoadInst *Load, QualType Ty, SourceLocation Loc)
Definition CGExpr.cpp:2101
void EmitAggregateCopy(LValue Dest, LValue Src, QualType EltTy, AggValueSlot::Overlap_t MayOverlap, bool isVolatile=false)
EmitAggregateCopy - Emit an aggregate copy.
const TargetInfo & getTarget() const
RValue EmitLoadOfLValue(LValue V, SourceLocation Loc)
EmitLoadOfLValue - Given an expression that represents a value lvalue, this method emits the address ...
Definition CGExpr.cpp:2521
RValue EmitAtomicLoad(LValue LV, SourceLocation SL, AggValueSlot Slot=AggValueSlot::ignored())
llvm::Value * getTypeSize(QualType Ty)
Returns calculated size of the specified type.
llvm::Value * EmitToMemory(llvm::Value *Value, QualType Ty)
EmitToMemory - Change a scalar value from its value representation to its in-memory representation.
Definition CGExpr.cpp:2243
llvm::AllocaInst * CreateTempAlloca(llvm::Type *Ty, const Twine &Name="tmp", llvm::Value *ArraySize=nullptr)
CreateTempAlloca - This creates an alloca and inserts it into the entry block if ArraySize is nullptr...
Definition CGExpr.cpp:158
ComplexPairTy EmitComplexExpr(const Expr *E, bool IgnoreReal=false, bool IgnoreImag=false)
EmitComplexExpr - Emit the computation of the specified expression of complex type,...
RValue EmitCall(const CGFunctionInfo &CallInfo, const CGCallee &Callee, ReturnValueSlot ReturnValue, const CallArgList &Args, llvm::CallBase **CallOrInvoke, bool IsMustTail, SourceLocation Loc, bool IsVirtualFunctionPointerThunk=false)
EmitCall - Generate a call of the given function, expecting the given result type,...
Definition CGCall.cpp:5451
const TargetCodeGenInfo & getTargetHooks() const
void EmitStoreOfComplex(ComplexPairTy V, LValue dest, bool isInit)
EmitStoreOfComplex - Store a complex number into the specified l-value.
void EmitStoreThroughLValue(RValue Src, LValue Dst, bool isInit=false)
EmitStoreThroughLValue - Store the specified rvalue into the specified lvalue, where both are guarant...
Definition CGExpr.cpp:2773
llvm::AtomicRMWInst * emitAtomicRMWInst(llvm::AtomicRMWInst::BinOp Op, Address Addr, llvm::Value *Val, llvm::AtomicOrdering Order=llvm::AtomicOrdering::SequentiallyConsistent, llvm::SyncScope::ID SSID=llvm::SyncScope::System, const AtomicExpr *AE=nullptr)
Emit an atomicrmw instruction, and applying relevant metadata when applicable.
void EmitAnyExprToMem(const Expr *E, Address Location, Qualifiers Quals, bool IsInitializer)
EmitAnyExprToMem - Emits the code necessary to evaluate an arbitrary expression into the given memory...
Definition CGExpr.cpp:308
llvm::Type * ConvertTypeForMem(QualType T)
RValue EmitLoadOfBitfieldLValue(LValue LV, SourceLocation Loc)
Definition CGExpr.cpp:2636
RValue EmitAtomicExpr(AtomicExpr *E)
Definition CGAtomic.cpp:914
static TypeEvaluationKind getEvaluationKind(QualType T)
getEvaluationKind - Return the TypeEvaluationKind of QualType T.
bool LValueIsSuitableForInlineAtomic(LValue Src)
An LValue is a candidate for having its loads and stores be made atomic if we are operating under /vo...
RawAddress CreateMemTemp(QualType T, const Twine &Name="tmp", RawAddress *Alloca=nullptr)
CreateMemTemp - Create a temporary memory object of the given type, with appropriate alignmen and cas...
Definition CGExpr.cpp:194
void EmitAggExpr(const Expr *E, AggValueSlot AS)
EmitAggExpr - Emit the computation of the specified expression of aggregate type.
llvm::Value * EmitScalarExpr(const Expr *E, bool IgnoreResultAssign=false)
EmitScalarExpr - Emit the computation of the specified expression of LLVM scalar type,...
RValue EmitLoadOfExtVectorElementLValue(LValue V)
Definition CGExpr.cpp:2673
LValue MakeAddrLValue(Address Addr, QualType T, AlignmentSource Source=AlignmentSource::Type)
void EmitAtomicStore(RValue rvalue, LValue lvalue, bool isInit)
llvm::Value * EmitFromMemory(llvm::Value *Value, QualType Ty)
EmitFromMemory - Change a scalar value from its memory representation to its value representation.
Definition CGExpr.cpp:2277
std::pair< llvm::Value *, llvm::Value * > ComplexPairTy
llvm::LLVMContext & getLLVMContext()
void EmitStoreOfScalar(llvm::Value *Value, Address Addr, bool Volatile, QualType Ty, AlignmentSource Source=AlignmentSource::Type, bool isInit=false, bool isNontemporal=false)
EmitStoreOfScalar - Store a scalar value to an address, taking care to appropriately convert from the...
void EmitBlock(llvm::BasicBlock *BB, bool IsFinished=false)
EmitBlock - Emit the given block.
Definition CGStmt.cpp:643
This class organizes the cross-function state that is used while generating LLVM code.
llvm::FunctionCallee CreateRuntimeFunction(llvm::FunctionType *Ty, StringRef Name, llvm::AttributeList ExtraAttrs=llvm::AttributeList(), bool Local=false, bool AssumeConvergent=false)
Create or return a runtime function declaration with the specified type and name.
const LangOptions & getLangOpts() const
const llvm::DataLayout & getDataLayout() const
void DecorateInstructionWithTBAA(llvm::Instruction *Inst, TBAAAccessInfo TBAAInfo)
DecorateInstructionWithTBAA - Decorate the instruction with a TBAA tag.
llvm::LLVMContext & getLLVMContext()
llvm::ConstantInt * getSize(CharUnits numChars)
Emit the given number of characters as a value of type size_t.
llvm::FunctionType * GetFunctionType(const CGFunctionInfo &Info)
GetFunctionType - Get the LLVM function type for.
Definition CGCall.cpp:1873
const CGFunctionInfo & arrangeBuiltinFunctionCall(QualType resultType, const CallArgList &args)
Definition CGCall.cpp:736
LValue - This represents an lvalue references.
Definition CGValue.h:183
bool isSimple() const
Definition CGValue.h:286
bool isVolatileQualified() const
Definition CGValue.h:297
bool isVolatile() const
Definition CGValue.h:340
Address getAddress() const
Definition CGValue.h:373
QualType getType() const
Definition CGValue.h:303
TBAAAccessInfo getTBAAInfo() const
Definition CGValue.h:347
RValue - This trivial value class is used to represent the result of an expression that is evaluated.
Definition CGValue.h:42
bool isScalar() const
Definition CGValue.h:64
static RValue get(llvm::Value *V)
Definition CGValue.h:99
static RValue getAggregate(Address addr, bool isVolatile=false)
Convert an Address to an RValue.
Definition CGValue.h:126
static RValue getComplex(llvm::Value *V1, llvm::Value *V2)
Definition CGValue.h:109
bool isAggregate() const
Definition CGValue.h:66
Address getAggregateAddress() const
getAggregateAddr() - Return the Value* of the address of the aggregate.
Definition CGValue.h:84
llvm::Value * getScalarVal() const
getScalarVal() - Return the Value* of this scalar value.
Definition CGValue.h:72
bool isComplex() const
Definition CGValue.h:65
bool isVolatileQualified() const
Definition CGValue.h:69
std::pair< llvm::Value *, llvm::Value * > getComplexVal() const
getComplexVal - Return the real/imag components of this complex value.
Definition CGValue.h:79
ReturnValueSlot - Contains the address where the return value of a function can be stored,...
Definition CGCall.h:381
llvm::SyncScope::ID getLLVMSyncScopeID(const LangOptions &LangOpts, SyncScope Scope, llvm::AtomicOrdering Ordering, llvm::LLVMContext &Ctx) const
Get the syncscope used in LLVM IR as a SyncScope ID.
virtual void setTargetAtomicMetadata(CodeGenFunction &CGF, llvm::Instruction &AtomicInst, const AtomicExpr *Expr=nullptr) const
Allow the target to apply other metadata to an atomic instruction.
Definition TargetInfo.h:345
Concrete class used by the front-end to report problems and issues.
Definition Diagnostic.h:233
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
This represents one expression.
Definition Expr.h:112
SourceLocation getExprLoc() const LLVM_READONLY
getExprLoc - Return the preferred location for the arrow when diagnosing a problem with a generic exp...
Definition Expr.cpp:282
QualType getType() const
Definition Expr.h:144
PointerType - C99 6.7.5.1 - Pointer Declarators.
Definition TypeBase.h:3390
A (possibly-)qualified type.
Definition TypeBase.h:937
bool isNull() const
Return true if this QualType doesn't point to a type yet.
Definition TypeBase.h:1004
LangAS getAddressSpace() const
Return the address space of this type.
Definition TypeBase.h:8571
Qualifiers getQualifiers() const
Retrieve the set of qualifiers applied to this type.
Definition TypeBase.h:8485
QualType getUnqualifiedType() const
Retrieve the unqualified variant of the given type, removing as little sugar as possible.
Definition TypeBase.h:8539
Scope - A scope is a transient data structure that is used while parsing the program.
Definition Scope.h:41
Encodes a location in the source.
bool isVoidType() const
Definition TypeBase.h:9048
bool isSignedIntegerType() const
Return true if this is an integer type that is signed, according to C99 6.2.5p4 [char,...
Definition Type.cpp:2266
bool isPointerType() const
Definition TypeBase.h:8682
const T * castAs() const
Member-template castAs<specific type>.
Definition TypeBase.h:9342
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
Definition Type.cpp:789
bool isAtomicType() const
Definition TypeBase.h:8874
bool isFloatingType() const
Definition Type.cpp:2389
const T * getAs() const
Member-template getAs<specific type>'.
Definition TypeBase.h:9275
QualType getType() const
Definition Value.cpp:237
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
@ Address
A pointer to a ValueDecl.
Definition Primitives.h:28
bool Load(InterpState &S, CodePtr OpPC)
Definition Interp.h:2110
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
Definition Address.h:330
@ Success
Annotation was successful.
Definition Parser.h:65
llvm::Expected< QualType > ExpectedType
llvm::StringRef getAsString(SyncScope S)
Definition SyncScope.h:62
U cast(CodeGen::Address addr)
Definition Address.h:327
unsigned long uint64_t
#define true
Definition stdbool.h:25
CharUnits StorageOffset
The offset of the bitfield storage from the start of the struct.
unsigned Offset
The offset within a contiguous run of bitfields that are represented as a single "field" within the L...
unsigned Size
The total size of the bit-field, in bits.
unsigned StorageSize
The storage size in bits which should be used when accessing this bitfield.
llvm::IntegerType * Int8Ty
i8, i16, i32, and i64