| //===--- CIRGenExprAgg.cpp - Emit CIR Code from Aggregate Expressions -----===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This contains code to emit Aggregate Expr nodes as CIR code. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "CIRGenBuilder.h" |
| #include "CIRGenFunction.h" |
| #include "CIRGenValue.h" |
| #include "clang/CIR/Dialect/IR/CIRAttrs.h" |
| |
| #include "clang/AST/Expr.h" |
| #include "clang/AST/StmtVisitor.h" |
| #include <cstdint> |
| |
| using namespace clang; |
| using namespace clang::CIRGen; |
| |
| namespace { |
| class AggExprEmitter : public StmtVisitor<AggExprEmitter> { |
| |
| CIRGenFunction &cgf; |
| AggValueSlot dest; |
| |
| AggValueSlot ensureSlot(mlir::Location loc, QualType t) { |
| if (!dest.isIgnored()) |
| return dest; |
| |
| cgf.cgm.errorNYI(loc, "Slot for ignored address"); |
| return dest; |
| } |
| |
| public: |
| AggExprEmitter(CIRGenFunction &cgf, AggValueSlot dest) |
| : cgf(cgf), dest(dest) {} |
| |
| void emitArrayInit(Address destPtr, cir::ArrayType arrayTy, QualType arrayQTy, |
| Expr *exprToVisit, ArrayRef<Expr *> args, |
| Expr *arrayFiller); |
| |
| void emitInitializationToLValue(Expr *e, LValue lv); |
| |
| void emitNullInitializationToLValue(mlir::Location loc, LValue lv); |
| |
| void Visit(Expr *e) { StmtVisitor<AggExprEmitter>::Visit(e); } |
| |
| void VisitInitListExpr(InitListExpr *e); |
| |
| void visitCXXParenListOrInitListExpr(Expr *e, ArrayRef<Expr *> args, |
| FieldDecl *initializedFieldInUnion, |
| Expr *arrayFiller); |
| }; |
| |
| } // namespace |
| |
| static bool isTrivialFiller(Expr *e) { |
| if (!e) |
| return true; |
| |
| if (isa<ImplicitValueInitExpr>(e)) |
| return true; |
| |
| if (auto *ile = dyn_cast<InitListExpr>(e)) { |
| if (ile->getNumInits()) |
| return false; |
| return isTrivialFiller(ile->getArrayFiller()); |
| } |
| |
| if (const auto *cons = dyn_cast_or_null<CXXConstructExpr>(e)) |
| return cons->getConstructor()->isDefaultConstructor() && |
| cons->getConstructor()->isTrivial(); |
| |
| return false; |
| } |
| |
| void AggExprEmitter::emitArrayInit(Address destPtr, cir::ArrayType arrayTy, |
| QualType arrayQTy, Expr *e, |
| ArrayRef<Expr *> args, Expr *arrayFiller) { |
| CIRGenBuilderTy &builder = cgf.getBuilder(); |
| const mlir::Location loc = cgf.getLoc(e->getSourceRange()); |
| |
| const uint64_t numInitElements = args.size(); |
| |
| const QualType elementType = |
| cgf.getContext().getAsArrayType(arrayQTy)->getElementType(); |
| |
| if (elementType.isDestructedType()) { |
| cgf.cgm.errorNYI(loc, "dtorKind NYI"); |
| return; |
| } |
| |
| const QualType elementPtrType = cgf.getContext().getPointerType(elementType); |
| |
| const mlir::Type cirElementType = cgf.convertType(elementType); |
| const cir::PointerType cirElementPtrType = |
| builder.getPointerTo(cirElementType); |
| |
| auto begin = builder.create<cir::CastOp>(loc, cirElementPtrType, |
| cir::CastKind::array_to_ptrdecay, |
| destPtr.getPointer()); |
| |
| const CharUnits elementSize = |
| cgf.getContext().getTypeSizeInChars(elementType); |
| const CharUnits elementAlign = |
| destPtr.getAlignment().alignmentOfArrayElement(elementSize); |
| |
| // The 'current element to initialize'. The invariants on this |
| // variable are complicated. Essentially, after each iteration of |
| // the loop, it points to the last initialized element, except |
| // that it points to the beginning of the array before any |
| // elements have been initialized. |
| mlir::Value element = begin; |
| |
| // Don't build the 'one' before the cycle to avoid |
| // emmiting the redundant `cir.const 1` instrs. |
| mlir::Value one; |
| |
| // Emit the explicit initializers. |
| for (uint64_t i = 0; i != numInitElements; ++i) { |
| // Advance to the next element. |
| if (i > 0) { |
| one = builder.getConstantInt(loc, cgf.PtrDiffTy, i); |
| element = builder.createPtrStride(loc, begin, one); |
| } |
| |
| const Address address = Address(element, cirElementType, elementAlign); |
| const LValue elementLV = LValue::makeAddr(address, elementType); |
| emitInitializationToLValue(args[i], elementLV); |
| } |
| |
| const uint64_t numArrayElements = arrayTy.getSize(); |
| |
| // Check whether there's a non-trivial array-fill expression. |
| const bool hasTrivialFiller = isTrivialFiller(arrayFiller); |
| |
| // Any remaining elements need to be zero-initialized, possibly |
| // using the filler expression. We can skip this if the we're |
| // emitting to zeroed memory. |
| if (numInitElements != numArrayElements && |
| !(dest.isZeroed() && hasTrivialFiller && |
| cgf.getTypes().isZeroInitializable(elementType))) { |
| // Advance to the start of the rest of the array. |
| if (numInitElements) { |
| one = builder.getConstantInt(loc, cgf.PtrDiffTy, 1); |
| element = builder.create<cir::PtrStrideOp>(loc, cirElementPtrType, |
| element, one); |
| } |
| |
| // Allocate the temporary variable |
| // to store the pointer to first unitialized element |
| const Address tmpAddr = cgf.createTempAlloca( |
| cirElementPtrType, cgf.getPointerAlign(), loc, "arrayinit.temp", |
| /*insertIntoFnEntryBlock=*/false); |
| LValue tmpLV = LValue::makeAddr(tmpAddr, elementPtrType); |
| cgf.emitStoreThroughLValue(RValue::get(element), tmpLV); |
| |
| // TODO(CIR): Replace this part later with cir::DoWhileOp |
| for (unsigned i = numInitElements; i != numArrayElements; ++i) { |
| cir::LoadOp currentElement = |
| builder.createLoad(loc, tmpAddr.getPointer()); |
| |
| // Emit the actual filler expression. |
| const LValue elementLV = LValue::makeAddr( |
| Address(currentElement, cirElementType, elementAlign), elementType); |
| |
| if (arrayFiller) |
| emitInitializationToLValue(arrayFiller, elementLV); |
| else |
| emitNullInitializationToLValue(loc, elementLV); |
| |
| // Advance pointer and store them to temporary variable |
| one = builder.getConstantInt(loc, cgf.PtrDiffTy, 1); |
| cir::PtrStrideOp nextElement = |
| builder.createPtrStride(loc, currentElement, one); |
| cgf.emitStoreThroughLValue(RValue::get(nextElement), tmpLV); |
| } |
| } |
| } |
| |
| void AggExprEmitter::emitInitializationToLValue(Expr *e, LValue lv) { |
| const QualType type = lv.getType(); |
| |
| if (isa<ImplicitValueInitExpr, CXXScalarValueInitExpr>(e)) { |
| const mlir::Location loc = e->getSourceRange().isValid() |
| ? cgf.getLoc(e->getSourceRange()) |
| : *cgf.currSrcLoc; |
| return emitNullInitializationToLValue(loc, lv); |
| } |
| |
| if (isa<NoInitExpr>(e)) |
| return; |
| |
| if (type->isReferenceType()) |
| cgf.cgm.errorNYI("emitInitializationToLValue ReferenceType"); |
| |
| switch (cgf.getEvaluationKind(type)) { |
| case cir::TEK_Complex: |
| cgf.cgm.errorNYI("emitInitializationToLValue TEK_Complex"); |
| break; |
| case cir::TEK_Aggregate: |
| cgf.emitAggExpr(e, AggValueSlot::forLValue(lv)); |
| return; |
| case cir::TEK_Scalar: |
| if (lv.isSimple()) |
| cgf.emitScalarInit(e, cgf.getLoc(e->getSourceRange()), lv); |
| else |
| cgf.emitStoreThroughLValue(RValue::get(cgf.emitScalarExpr(e)), lv); |
| return; |
| } |
| } |
| |
| void AggExprEmitter::emitNullInitializationToLValue(mlir::Location loc, |
| LValue lv) { |
| const QualType type = lv.getType(); |
| |
| // If the destination slot is already zeroed out before the aggregate is |
| // copied into it, we don't have to emit any zeros here. |
| if (dest.isZeroed() && cgf.getTypes().isZeroInitializable(type)) |
| return; |
| |
| if (cgf.hasScalarEvaluationKind(type)) { |
| // For non-aggregates, we can store the appropriate null constant. |
| mlir::Value null = cgf.cgm.emitNullConstant(type, loc); |
| if (lv.isSimple()) { |
| cgf.emitStoreOfScalar(null, lv, /* isInitialization */ true); |
| return; |
| } |
| |
| cgf.cgm.errorNYI("emitStoreThroughBitfieldLValue"); |
| return; |
| } |
| |
| // There's a potential optimization opportunity in combining |
| // memsets; that would be easy for arrays, but relatively |
| // difficult for structures with the current code. |
| cgf.emitNullInitialization(loc, lv.getAddress(), lv.getType()); |
| } |
| |
| void AggExprEmitter::VisitInitListExpr(InitListExpr *e) { |
| if (e->hadArrayRangeDesignator()) |
| llvm_unreachable("GNU array range designator extension"); |
| |
| if (e->isTransparent()) |
| return Visit(e->getInit(0)); |
| |
| visitCXXParenListOrInitListExpr( |
| e, e->inits(), e->getInitializedFieldInUnion(), e->getArrayFiller()); |
| } |
| |
| void AggExprEmitter::visitCXXParenListOrInitListExpr( |
| Expr *e, ArrayRef<Expr *> args, FieldDecl *initializedFieldInUnion, |
| Expr *arrayFiller) { |
| |
| const AggValueSlot dest = |
| ensureSlot(cgf.getLoc(e->getSourceRange()), e->getType()); |
| |
| if (e->getType()->isConstantArrayType()) { |
| cir::ArrayType arrayTy = |
| cast<cir::ArrayType>(dest.getAddress().getElementType()); |
| emitArrayInit(dest.getAddress(), arrayTy, e->getType(), e, args, |
| arrayFiller); |
| return; |
| } |
| |
| cgf.cgm.errorNYI( |
| "visitCXXParenListOrInitListExpr Record or VariableSizeArray type"); |
| } |
| |
| void CIRGenFunction::emitAggExpr(const Expr *e, AggValueSlot slot) { |
| AggExprEmitter(*this, slot).Visit(const_cast<Expr *>(e)); |
| } |