When variadic return stmt doesn't have a return value, invalidate the calculation of the last expr for evaluation.
In the same cases we still need to synthesize the setValueNoAlloc, to initialize the cling::Value to void.
This commit is contained in:
parent
529088b816
commit
d1a416ddc0
@ -87,16 +87,29 @@ namespace cling {
|
|||||||
I != E; ++I) {
|
I != E; ++I) {
|
||||||
ReturnStmt* RS = dyn_cast<ReturnStmt>(**I);
|
ReturnStmt* RS = dyn_cast<ReturnStmt>(**I);
|
||||||
if (RS) {
|
if (RS) {
|
||||||
if (Expr* RetV = RS->getRetValue()) {
|
// When we are handling a return stmt, the last expression must be the
|
||||||
assert (RetV->getType()->isVoidType() && "Must be void type.");
|
// return stmt value. Ignore the calculation of the lastStmt because it
|
||||||
|
// might be wrong, in cases where the return is not in the end of the
|
||||||
|
// function.
|
||||||
|
lastExpr = RS->getRetValue();
|
||||||
|
if (lastExpr) {
|
||||||
|
assert (lastExpr->getType()->isVoidType() && "Must be void type.");
|
||||||
// Any return statement will have been "healed" by Sema
|
// Any return statement will have been "healed" by Sema
|
||||||
// to correspond to the original void return type of the
|
// to correspond to the original void return type of the
|
||||||
// wrapper, using a ImplicitCastExpr 'void' <ToVoid>.
|
// wrapper, using a ImplicitCastExpr 'void' <ToVoid>.
|
||||||
// Remove that.
|
// Remove that.
|
||||||
if (ImplicitCastExpr* VoidCast = dyn_cast<ImplicitCastExpr>(RetV)) {
|
if (ImplicitCastExpr* VoidCast
|
||||||
|
= dyn_cast<ImplicitCastExpr>(lastExpr)) {
|
||||||
lastExpr = VoidCast->getSubExpr();
|
lastExpr = VoidCast->getSubExpr();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// if no value assume void
|
||||||
|
else {
|
||||||
|
// We can't PushDeclContext, because we don't have scope.
|
||||||
|
Sema::ContextRAII pushedDC(*m_Sema, FD);
|
||||||
|
RS->setRetValue(SynthesizeSVRInit(0));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
lastExpr = cast<Expr>(**I);
|
lastExpr = cast<Expr>(**I);
|
||||||
@ -151,11 +164,8 @@ namespace cling {
|
|||||||
// if we had return stmt update to execute the SVR init, even if the
|
// if we had return stmt update to execute the SVR init, even if the
|
||||||
// wrapper returns void.
|
// wrapper returns void.
|
||||||
if (RS) {
|
if (RS) {
|
||||||
Expr* retValue = RS->getRetValue();
|
if (ImplicitCastExpr* VoidCast
|
||||||
if (!retValue)
|
= dyn_cast<ImplicitCastExpr>(RS->getRetValue()))
|
||||||
RS->setRetValue(SVRInit);
|
|
||||||
else if (ImplicitCastExpr* VoidCast
|
|
||||||
= dyn_cast<ImplicitCastExpr>(retValue))
|
|
||||||
VoidCast->setSubExpr(SVRInit);
|
VoidCast->setSubExpr(SVRInit);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -174,10 +184,12 @@ namespace cling {
|
|||||||
|
|
||||||
// Build a reference to Value* in the wrapper, should be
|
// Build a reference to Value* in the wrapper, should be
|
||||||
// the only argument of the wrapper.
|
// the only argument of the wrapper.
|
||||||
|
SourceLocation locStart = (E) ? E->getLocStart() : FD->getLocStart();
|
||||||
|
SourceLocation locEnd = (E) ? E->getLocEnd() : FD->getLocEnd();
|
||||||
ExprResult wrapperSVRDRE
|
ExprResult wrapperSVRDRE
|
||||||
= m_Sema->BuildDeclRefExpr(FD->getParamDecl(0), m_Context->VoidPtrTy,
|
= m_Sema->BuildDeclRefExpr(FD->getParamDecl(0), m_Context->VoidPtrTy,
|
||||||
VK_RValue, E->getLocStart());
|
VK_RValue, locStart);
|
||||||
QualType ETy = E->getType();
|
QualType ETy = (E) ? E->getType() : m_Context->VoidTy;
|
||||||
QualType desugaredTy = ETy.getDesugaredType(*m_Context);
|
QualType desugaredTy = ETy.getDesugaredType(*m_Context);
|
||||||
|
|
||||||
// The expr result is transported as reference, pointer, array, float etc
|
// The expr result is transported as reference, pointer, array, float etc
|
||||||
@ -206,18 +218,16 @@ namespace cling {
|
|||||||
// We need to synthesize setValueNoAlloc(...), E, because we still need
|
// We need to synthesize setValueNoAlloc(...), E, because we still need
|
||||||
// to run E.
|
// to run E.
|
||||||
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedNoAlloc,
|
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedNoAlloc,
|
||||||
E->getLocStart(), CallArgs,
|
locStart, CallArgs, locEnd);
|
||||||
E->getLocEnd());
|
if (E)
|
||||||
Call = m_Sema->CreateBuiltinBinOp(Call.get()->getLocStart(), BO_Comma,
|
Call = m_Sema->CreateBuiltinBinOp(locStart, BO_Comma, Call.take(), E);
|
||||||
Call.take(), E);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
else if (desugaredTy->isRecordType() || desugaredTy->isConstantArrayType()){
|
else if (desugaredTy->isRecordType() || desugaredTy->isConstantArrayType()){
|
||||||
// 2) object types :
|
// 2) object types :
|
||||||
// call new (setValueWithAlloc(gCling, &SVR, ETy)) (E)
|
// call new (setValueWithAlloc(gCling, &SVR, ETy)) (E)
|
||||||
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedWithAlloc,
|
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedWithAlloc,
|
||||||
E->getLocStart(), CallArgs,
|
locStart, CallArgs, locEnd);
|
||||||
E->getLocEnd());
|
|
||||||
Expr* placement = Call.take();
|
Expr* placement = Call.take();
|
||||||
if (const ConstantArrayType* constArray
|
if (const ConstantArrayType* constArray
|
||||||
= dyn_cast<ConstantArrayType>(desugaredTy.getTypePtr())) {
|
= dyn_cast<ConstantArrayType>(desugaredTy.getTypePtr())) {
|
||||||
@ -233,8 +243,7 @@ namespace cling {
|
|||||||
// 2.1) arrays:
|
// 2.1) arrays:
|
||||||
// call copyArray(T* src, void* placement, int size)
|
// call copyArray(T* src, void* placement, int size)
|
||||||
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedCopyArray,
|
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedCopyArray,
|
||||||
E->getLocStart(), CallArgs,
|
locStart, CallArgs, locEnd);
|
||||||
E->getLocEnd());
|
|
||||||
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -296,8 +305,7 @@ namespace cling {
|
|||||||
CallArgs.push_back(E);
|
CallArgs.push_back(E);
|
||||||
}
|
}
|
||||||
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedNoAlloc,
|
Call = m_Sema->ActOnCallExpr(/*Scope*/0, m_UnresolvedNoAlloc,
|
||||||
E->getLocStart(), CallArgs,
|
locStart, CallArgs, locEnd);
|
||||||
E->getLocEnd());
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
assert(0 && "Unhandled code path?");
|
assert(0 && "Unhandled code path?");
|
||||||
|
@ -23,9 +23,13 @@ V // CHECK-NEXT: (cling::Value) boxes [(void) @0x{{.*}}]
|
|||||||
// Returns must put the result in the Value.
|
// Returns must put the result in the Value.
|
||||||
bool cond = true;
|
bool cond = true;
|
||||||
gCling->evaluate("if (cond) return \"true\"; else return 0;", V);
|
gCling->evaluate("if (cond) return \"true\"; else return 0;", V);
|
||||||
V // CHECK: (cling::Value) boxes [(const char [5]) "true"]
|
V // CHECK-NEXT: (cling::Value) boxes [(const char [5]) "true"]
|
||||||
|
gCling->evaluate("if (cond) return; else return 12;", V);
|
||||||
|
V // CHECK-NEXT: (cling::Value) boxes [(void) @0x{{.*}}]
|
||||||
|
gCling->evaluate("if (cond) return; int aa = 12;", V);
|
||||||
|
V // CHECK-NEXT: (cling::Value) boxes [(void) @0x{{.*}}]
|
||||||
gCling->evaluate("cond = false; if (cond) return \"true\"; else return 0;", V);
|
gCling->evaluate("cond = false; if (cond) return \"true\"; else return 0;", V);
|
||||||
V // CHECK: (cling::Value) boxes [(int) 0]
|
V // CHECK-NEXT: (cling::Value) boxes [(int) 0]
|
||||||
|
|
||||||
gCling->evaluate("auto a = 12.3; a;", V);
|
gCling->evaluate("auto a = 12.3; a;", V);
|
||||||
V // CHECK: (cling::Value) boxes [(double) 1.230000e+01]
|
V // CHECK: (cling::Value) boxes [(double) 1.230000e+01]
|
||||||
|
Loading…
x
Reference in New Issue
Block a user