summaryrefslogtreecommitdiffstats
path: root/clang
diff options
context:
space:
mode:
authorRichard Smith <richard-llvm@metafoo.co.uk>2012-02-08 08:11:33 +0000
committerRichard Smith <richard-llvm@metafoo.co.uk>2012-02-08 08:11:33 +0000
commit90cacbbf3efda3745dd7c10e8f891956bfaf031e (patch)
treea951db452ff6b01ba4f07ac4b7b293d9b26ea8a6 /clang
parent13f118f47b6d1faf9ea41d53b5ebb1e965cd3696 (diff)
downloadbcm5719-llvm-90cacbbf3efda3745dd7c10e8f891956bfaf031e.tar.gz
bcm5719-llvm-90cacbbf3efda3745dd7c10e8f891956bfaf031e.zip
Implement DR1458: Taking the address of an object of incomplete class type is
not a constant expression, because we can't tell whether the complete class type will have an overloaded operator&. llvm-svn: 150066
Diffstat (limited to 'clang')
-rw-r--r--clang/include/clang/Basic/DiagnosticASTKinds.td3
-rw-r--r--clang/lib/AST/ExprConstant.cpp12
-rw-r--r--clang/test/CXX/expr/expr.const/p2-0x.cpp16
-rw-r--r--clang/test/SemaCXX/constant-expression-cxx11.cpp3
4 files changed, 33 insertions, 1 deletions
diff --git a/clang/include/clang/Basic/DiagnosticASTKinds.td b/clang/include/clang/Basic/DiagnosticASTKinds.td
index 9e4b63cfe81..978df36f9f4 100644
--- a/clang/include/clang/Basic/DiagnosticASTKinds.td
+++ b/clang/include/clang/Basic/DiagnosticASTKinds.td
@@ -55,6 +55,9 @@ def note_constexpr_pointer_comparison_differing_access : Note<
"specifiers (%1 vs %3) has unspecified value">;
def note_constexpr_compare_virtual_mem_ptr : Note<
"comparison of pointer to virtual member function %0 has unspecified value">;
+def note_constexpr_addr_of_incomplete : Note<
+ "cannot take address of object of incomplete class type %0 "
+ "in a constant expression">;
def note_constexpr_past_end : Note<
"dereferenced pointer past the end of %select{|subobject of }0"
"%select{temporary|%2}1 is not a constant expression">;
diff --git a/clang/lib/AST/ExprConstant.cpp b/clang/lib/AST/ExprConstant.cpp
index 410406788dc..48e0c6f7daa 100644
--- a/clang/lib/AST/ExprConstant.cpp
+++ b/clang/lib/AST/ExprConstant.cpp
@@ -2937,6 +2937,18 @@ bool PointerExprEvaluator::VisitBinaryOperator(const BinaryOperator *E) {
}
bool PointerExprEvaluator::VisitUnaryAddrOf(const UnaryOperator *E) {
+ QualType SrcTy = E->getSubExpr()->getType();
+ // In C++, taking the address of an object of incomplete class type has
+ // undefined behavior if the complete class type has an overloaded operator&.
+ // DR1458 makes such expressions non-constant.
+ if (Info.getLangOpts().CPlusPlus &&
+ SrcTy->isRecordType() && SrcTy->isIncompleteType()) {
+ const RecordType *RT = SrcTy->getAs<RecordType>();
+ Info.CCEDiag(E->getExprLoc(), diag::note_constexpr_addr_of_incomplete, 1)
+ << SrcTy;
+ Info.Note(RT->getDecl()->getLocation(), diag::note_forward_declaration)
+ << RT->getDecl();
+ }
return EvaluateLValue(E->getSubExpr(), Result, Info);
}
diff --git a/clang/test/CXX/expr/expr.const/p2-0x.cpp b/clang/test/CXX/expr/expr.const/p2-0x.cpp
index 71dc2f7104e..43d683aeca4 100644
--- a/clang/test/CXX/expr/expr.const/p2-0x.cpp
+++ b/clang/test/CXX/expr/expr.const/p2-0x.cpp
@@ -109,6 +109,22 @@ namespace RecursionLimits {
};
}
+// DR1458: taking the address of an object of incomplete class type
+namespace IncompleteClassTypeAddr {
+ struct S; // expected-note {{forward}}
+ extern S s;
+ constexpr S *p = &s; // expected-error {{constant expression}} expected-note {{cannot take address of object of incomplete class type 'IncompleteClassTypeAddr::S' in a constant expression}}
+
+ extern S sArr[];
+ constexpr S (*p2)[] = &sArr; // ok
+
+ struct S {
+ constexpr S *operator&() { return nullptr; }
+ };
+ constexpr S *q = &s;
+ static_assert(!q, "");
+}
+
// - an operation that would have undefined behavior [Note: including, for
// example, signed integer overflow (Clause 5 [expr]), certain pointer
// arithmetic (5.7 [expr.add]), division by zero (5.6 [expr.mul]), or certain
diff --git a/clang/test/SemaCXX/constant-expression-cxx11.cpp b/clang/test/SemaCXX/constant-expression-cxx11.cpp
index 763c41680ce..ccbc8c1abd0 100644
--- a/clang/test/SemaCXX/constant-expression-cxx11.cpp
+++ b/clang/test/SemaCXX/constant-expression-cxx11.cpp
@@ -299,6 +299,7 @@ static_assert(&x > &x, "false"); // expected-error {{false}}
constexpr S* sptr = &s;
constexpr bool dyncast = sptr == dynamic_cast<S*>(sptr); // expected-error {{constant expression}} expected-note {{dynamic_cast}}
+struct U {};
struct Str {
int a : dynamic_cast<S*>(sptr) == dynamic_cast<S*>(sptr); // \
expected-warning {{not an integral constant expression}} \
@@ -315,7 +316,7 @@ struct Str {
int e : (Str*)(sptr) == (Str*)(sptr); // \
expected-warning {{not an integral constant expression}} \
expected-note {{cast which performs the conversions of a reinterpret_cast is not allowed in a constant expression}}
- int f : &(Str&)(*sptr) == &(Str&)(*sptr); // \
+ int f : &(U&)(*sptr) == &(U&)(*sptr); // \
expected-warning {{not an integral constant expression}} \
expected-note {{cast which performs the conversions of a reinterpret_cast is not allowed in a constant expression}}
int g : (S*)(void*)(sptr) == sptr; // \
OpenPOWER on IntegriCloud