Objective-C properties. Fixes a crash in Sema where RHS of

the property assignment is an lvalue for an incomplete type.
// rdar://15118128. Reviewed offline by John McCall.

llvm-svn: 203043
This commit is contained in:
Fariborz Jahanian 2014-03-06 00:34:05 +00:00
parent bb29e518c8
commit 15dde89357
2 changed files with 60 additions and 4 deletions

View File

@ -235,7 +235,10 @@ namespace {
} }
/// Return true if assignments have a non-void result. /// Return true if assignments have a non-void result.
bool CanCaptureValueOfType(QualType ty) { bool CanCaptureValue(Expr *exp) {
if (exp->isGLValue())
return true;
QualType ty = exp->getType();
assert(!ty->isIncompleteType()); assert(!ty->isIncompleteType());
assert(!ty->isDependentType()); assert(!ty->isDependentType());
@ -461,7 +464,7 @@ PseudoOpBuilder::buildIncDecOperation(Scope *Sc, SourceLocation opcLoc,
// That's the postfix result. // That's the postfix result.
if (UnaryOperator::isPostfix(opcode) && if (UnaryOperator::isPostfix(opcode) &&
(result.get()->isTypeDependent() || CanCaptureValueOfType(resultType))) { (result.get()->isTypeDependent() || CanCaptureValue(result.get()))) {
result = capture(result.take()); result = capture(result.take());
setResultToLastSemantic(); setResultToLastSemantic();
} }
@ -762,7 +765,7 @@ ExprResult ObjCPropertyOpBuilder::buildSet(Expr *op, SourceLocation opcLoc,
ObjCMessageExpr *msgExpr = ObjCMessageExpr *msgExpr =
cast<ObjCMessageExpr>(msg.get()->IgnoreImplicit()); cast<ObjCMessageExpr>(msg.get()->IgnoreImplicit());
Expr *arg = msgExpr->getArg(0); Expr *arg = msgExpr->getArg(0);
if (CanCaptureValueOfType(arg->getType())) if (CanCaptureValue(arg))
msgExpr->setArg(0, captureValueAsResult(arg)); msgExpr->setArg(0, captureValueAsResult(arg));
} }
@ -1368,7 +1371,7 @@ ExprResult ObjCSubscriptOpBuilder::buildSet(Expr *op, SourceLocation opcLoc,
ObjCMessageExpr *msgExpr = ObjCMessageExpr *msgExpr =
cast<ObjCMessageExpr>(msg.get()->IgnoreImplicit()); cast<ObjCMessageExpr>(msg.get()->IgnoreImplicit());
Expr *arg = msgExpr->getArg(0); Expr *arg = msgExpr->getArg(0);
if (CanCaptureValueOfType(arg->getType())) if (CanCaptureValue(arg))
msgExpr->setArg(0, captureValueAsResult(arg)); msgExpr->setArg(0, captureValueAsResult(arg));
} }

View File

@ -0,0 +1,53 @@
// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm -o - %s | FileCheck %s
// rdar://15118128
template <typename T> struct Quad2 {
Quad2() {}
};
typedef Quad2<double> Quad2d;
@interface Root @end
@interface PAGeometryFrame
- (const Quad2d &)quad;
- (void)setQuad:(const Quad2d &)quad;
@end
@interface PA2DScaleTransform : Root
@end
@implementation PA2DScaleTransform
- (void)transformFrame:(PAGeometryFrame *)frame {
PAGeometryFrame *result;
result.quad = frame.quad;
}
@end
// CHECK: [[TWO:%.*]] = load i8** @"\01L_OBJC_SELECTOR_REFERENCES_", !invariant.load !5
// CHECK: [[THREE:%.*]] = bitcast [[ONET:%.*]]* [[ONE:%.*]] to i8*
// CHECK: [[CALL:%.*]] = call %struct.Quad2* bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to %struct.Quad2* (i8*, i8*)*)(i8* [[THREE]], i8* [[TWO]])
// CHECK: [[FOUR:%.*]] = load i8** @"\01L_OBJC_SELECTOR_REFERENCES_2", !invariant.load !5
// CHECK: [[FIVE:%.*]] = bitcast [[ONET]]* [[ZERO:%.*]] to i8*
// CHECK: call void bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to void (i8*, i8*, %struct.Quad2*)*)(i8* [[FIVE]], i8* [[FOUR]], %struct.Quad2* [[CALL]])
struct A {
void *ptr;
A();
A(const A &);
~A();
};
@interface C
- (void) setProp: (const A&) value;
@end
void test(C *c, const A &a) {
const A &result = c.prop = a;
}
// CHECK: [[ONE1:%.*]] = load %struct.A** [[AADDR:%.*]], align 8
// CHECK: [[TWO1:%.*]] = load i8** @"\01L_OBJC_SELECTOR_REFERENCES_5", !invariant.load !5
// CHECK: [[THREE1:%.*]] = bitcast [[TWOT:%.*]]* [[ZERO1:%.*]] to i8*
// CHECK: call void bitcast (i8* (i8*, i8*, ...)* @objc_msgSend to void (i8*, i8*, %struct.A*)*)(i8* [[THREE1]], i8* [[TWO1]], %struct.A* [[ONE1]])
// CHECK: store %struct.A* [[ONE1]], %struct.A** [[RESULT:%.*]], align 8