diff options
author | Douglas Gregor <dgregor@apple.com> | 2011-02-01 15:50:11 +0000 |
---|---|---|
committer | Douglas Gregor <dgregor@apple.com> | 2011-02-01 15:50:11 +0000 |
commit | d880f52be3e4a8ccad92ac31932eeae5e0870a93 (patch) | |
tree | 97af0d083661709907c9ab8374ce6c1d92d877b9 | |
parent | f9997a0834e0e0298b04ef044ad2699c727a7979 (diff) |
Implement access checking for the "delete" operator. Fixes PR9050,
from Alex Miller!
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@124663 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | include/clang/Basic/DiagnosticSemaKinds.td | 3 | ||||
-rw-r--r-- | lib/Sema/SemaExprCXX.cpp | 10 | ||||
-rw-r--r-- | test/CXX/dcl.dcl/basic.namespace/namespace.udecl/p1.cpp | 1 | ||||
-rw-r--r-- | test/CXX/expr/expr.unary/expr.delete/p5.cpp | 22 | ||||
-rw-r--r-- | test/SemaCXX/new-delete.cpp | 11 |
5 files changed, 41 insertions, 6 deletions
diff --git a/include/clang/Basic/DiagnosticSemaKinds.td b/include/clang/Basic/DiagnosticSemaKinds.td index 3e7f48cf81..316db75215 100644 --- a/include/clang/Basic/DiagnosticSemaKinds.td +++ b/include/clang/Basic/DiagnosticSemaKinds.td @@ -564,6 +564,9 @@ def err_access_field: Error< def err_access_ctor_field : Error<"field of type %1 has %select{private|protected}2 constructor">, AccessControl; +def err_access_dtor : Error< + "calling a %select{private|protected}1 destructor of class %0">, + AccessControl; def err_access_dtor_base : Error<"base class %0 has %select{private|protected}1 destructor">, AccessControl; diff --git a/lib/Sema/SemaExprCXX.cpp b/lib/Sema/SemaExprCXX.cpp index 82e818fba0..7bc9af1191 100644 --- a/lib/Sema/SemaExprCXX.cpp +++ b/lib/Sema/SemaExprCXX.cpp @@ -1687,7 +1687,15 @@ Sema::ActOnCXXDelete(SourceLocation StartLoc, bool UseGlobal, MarkDeclarationReferenced(StartLoc, OperatorDelete); - // FIXME: Check access and ambiguity of operator delete and destructor. + // Check access and ambiguity of operator delete and destructor. + if (const RecordType *RT = PointeeElem->getAs<RecordType>()) { + CXXRecordDecl *RD = cast<CXXRecordDecl>(RT->getDecl()); + if (CXXDestructorDecl *Dtor = LookupDestructor(RD)) { + CheckDestructorAccess(Ex->getExprLoc(), Dtor, + PDiag(diag::err_access_dtor) << PointeeElem); + } + } + } return Owned(new (Context) CXXDeleteExpr(Context.VoidTy, UseGlobal, ArrayForm, diff --git a/test/CXX/dcl.dcl/basic.namespace/namespace.udecl/p1.cpp b/test/CXX/dcl.dcl/basic.namespace/namespace.udecl/p1.cpp index bb1d67f51a..634369dd52 100644 --- a/test/CXX/dcl.dcl/basic.namespace/namespace.udecl/p1.cpp +++ b/test/CXX/dcl.dcl/basic.namespace/namespace.udecl/p1.cpp @@ -89,6 +89,7 @@ namespace test2 { namespace test3 { class A { + public: ~A(); }; diff --git a/test/CXX/expr/expr.unary/expr.delete/p5.cpp b/test/CXX/expr/expr.unary/expr.delete/p5.cpp index 2fa30e59dc..ecb29189af 100644 --- a/test/CXX/expr/expr.unary/expr.delete/p5.cpp +++ b/test/CXX/expr/expr.unary/expr.delete/p5.cpp @@ -24,11 +24,23 @@ void f0(T2_A *a) { T2_C x; x.f0(a); } class T2_A { }; // An alternate version of the same. -// -// FIXME: Revisit this case when we have access control. class T3_A; template<typename T> -struct T3_B { void f0(T *a) { delete a; } }; -struct T3_C { T3_B<T3_A> x; void f0(T3_A *a) { x.f0(a); } }; +struct T3_B { + void f0(T *a) { + delete a; // expected-error{{calling a private destructor of class 'T3_A'}} + } +}; + +struct T3_C { + T3_B<T3_A> x; + void f0(T3_A *a) { + x.f0(a); // expected-note{{in instantiation of member function 'T3_B<T3_A>::f0' requested here}} + } +}; + void f0(T3_A *a) { T3_C x; x.f0(a); } -class T3_A { private: ~T3_A(); }; +class T3_A { +private: + ~T3_A(); // expected-note{{declared private here}} +}; diff --git a/test/SemaCXX/new-delete.cpp b/test/SemaCXX/new-delete.cpp index 19d5df6673..13ef461e7c 100644 --- a/test/SemaCXX/new-delete.cpp +++ b/test/SemaCXX/new-delete.cpp @@ -234,6 +234,17 @@ void f(X14 *x14a, X14 *x14b) { delete x14a; } +class X15 { +private: + X15(); // expected-note {{declared private here}} + ~X15(); // expected-note {{declared private here}} +}; + +void f(X15* x) { + new X15(); // expected-error {{calling a private constructor}} + delete x; // expected-error {{calling a private destructor}} +} + namespace PR5918 { // Look for template operator new overloads. struct S { template<typename T> static void* operator new(size_t, T); }; void test() { |