Index: clang/lib/AST/Interp/ByteCodeExprGen.h =================================================================== --- clang/lib/AST/Interp/ByteCodeExprGen.h +++ clang/lib/AST/Interp/ByteCodeExprGen.h @@ -85,6 +85,8 @@ bool VisitConstantExpr(const ConstantExpr *E); bool VisitUnaryExprOrTypeTraitExpr(const UnaryExprOrTypeTraitExpr *E); bool VisitMemberExpr(const MemberExpr *E); + bool VisitArrayInitIndexExpr(const ArrayInitIndexExpr *E); + bool VisitOpaqueValueExpr(const OpaqueValueExpr *E); protected: bool visitExpr(const Expr *E) override; @@ -260,7 +262,7 @@ /// Current scope. VariableScope *VarScope = nullptr; - /// Current argument index. + /// Current argument index. Needed to emit ArrayInitIndexExpr. llvm::Optional ArrayIndex; /// Flag indicating if return value is to be discarded. Index: clang/lib/AST/Interp/ByteCodeExprGen.cpp =================================================================== --- clang/lib/AST/Interp/ByteCodeExprGen.cpp +++ clang/lib/AST/Interp/ByteCodeExprGen.cpp @@ -325,6 +325,18 @@ return false; } +template +bool ByteCodeExprGen::VisitArrayInitIndexExpr( + const ArrayInitIndexExpr *E) { + assert(ArrayIndex); + return this->emitConstUint64(*ArrayIndex, E); +} + +template +bool ByteCodeExprGen::VisitOpaqueValueExpr(const OpaqueValueExpr *E) { + return this->visit(E->getSourceExpr()); +} + template bool ByteCodeExprGen::discard(const Expr *E) { OptionScope Scope(this, /*NewDiscardResult=*/true); return this->Visit(E); @@ -626,6 +638,33 @@ return true; } else if (const auto *DIE = dyn_cast(Initializer)) { return this->visitInitializer(DIE->getExpr()); + } else if (const auto AILE = dyn_cast(Initializer)) { + // TODO: This compiles to quite a lot of bytecode if the array is larger. + // Investigate compiling this to a loop, or at least try to use + // the AILE's Common expr. + const Expr *SubExpr = AILE->getSubExpr(); + size_t Size = AILE->getArraySize().getZExtValue(); + Optional ElemT = classify(SubExpr->getType()); + + if (!ElemT) + return false; + + for (size_t I = 0; I != Size; ++I) { + ArrayIndex = I; + if (!this->emitDupPtr(SubExpr)) + return false; + + if (!this->visit(SubExpr)) + return false; + + if (!this->emitInitElem(*ElemT, I, Initializer)) + return false; + + if (!this->emitPopPtr(Initializer)) + return false; + } + ArrayIndex = None; + return true; } assert(false && "Unknown expression for array initialization"); @@ -640,10 +679,20 @@ if (const auto CtorExpr = dyn_cast(Initializer)) { const Function *Func = getFunction(CtorExpr->getConstructor()); + if (!Func->isConstexpr()) + return false; + // The This pointer is already on the stack because this is an initializer, // but we need to dup() so the call() below has its own copy. if (!this->emitDupPtr(Initializer)) return false; + + // Constructor arguments. + for (const auto *Arg : CtorExpr->arguments()) { + if (!this->visit(Arg)) + return false; + } + return this->emitCallVoid(Func, Initializer); } else if (const auto *InitList = dyn_cast(Initializer)) { const Record *R = getRecord(InitList->getType()); @@ -652,15 +701,26 @@ for (const Expr *Init : InitList->inits()) { const Record::Field *FieldToInit = R->getField(InitIndex); - if (Optional T = classify(Init->getType())) { - if (!this->emitDupPtr(Initializer)) - return false; + if (!this->emitDupPtr(Initializer)) + return false; + if (Optional T = classify(Init->getType())) { if (!this->visit(Init)) return false; if (!this->emitInitField(*T, FieldToInit->Offset, Initializer)) return false; + } else { + // Non-primitive case. Get a pointer to the field-to-initialize + // on the stack and recurse into visitInitializer(). + if (!this->emitGetPtrField(FieldToInit->Offset, Init)) + return false; + + if (!this->visitInitializer(Init)) + return false; + + if (!this->emitPopPtr(Initializer)) + return false; } ++InitIndex; } Index: clang/lib/AST/Interp/ByteCodeStmtGen.cpp =================================================================== --- clang/lib/AST/Interp/ByteCodeStmtGen.cpp +++ clang/lib/AST/Interp/ByteCodeStmtGen.cpp @@ -105,7 +105,7 @@ const Record::Field *F = R->getField(Member); if (Optional T = this->classify(InitExpr->getType())) { - if (!this->emitDupPtr(InitExpr)) + if (!this->emitThis(InitExpr)) return false; if (!this->visit(InitExpr)) @@ -116,7 +116,7 @@ } else { // Non-primitive case. Get a pointer to the field-to-initialize // on the stack and call visitInitialzer() for it. - if (!this->emitDupPtr(InitExpr)) + if (!this->emitThis(InitExpr)) return false; if (!this->emitGetPtrField(F->Offset, InitExpr)) Index: clang/test/AST/Interp/records.cpp =================================================================== --- clang/test/AST/Interp/records.cpp +++ clang/test/AST/Interp/records.cpp @@ -37,6 +37,9 @@ static_assert(ints.b == 30, ""); static_assert(ints.c, ""); static_assert(ints.getTen() == 10, ""); +static_assert(ints.numbers[0] == 1, ""); +static_assert(ints.numbers[1] == 2, ""); +static_assert(ints.numbers[2] == 3, ""); constexpr const BoolPair &BP = ints.bp; static_assert(BP.first, ""); @@ -64,11 +67,17 @@ static_assert(ints4.a == (40 * 50), ""); static_assert(ints4.b == 0, ""); static_assert(ints4.c, ""); - - -// FIXME: Implement initialization by DeclRefExpr. -//constexpr Ints ints4 = ints3; TODO - +static_assert(ints4.numbers[0] == 1, ""); +static_assert(ints4.numbers[1] == 2, ""); +static_assert(ints4.numbers[2] == 3, ""); + +constexpr Ints ints5 = ints4; +static_assert(ints5.a == (40 * 50), ""); +static_assert(ints5.b == 0, ""); +static_assert(ints5.c, ""); +static_assert(ints5.numbers[0] == 1, ""); +static_assert(ints5.numbers[1] == 2, ""); +static_assert(ints5.numbers[2] == 3, ""); struct Ints2 {