From: Anders Carlsson Date: Sat, 28 Mar 2009 01:09:05 +0000 (+0000) Subject: Implement access checking for protected base classes. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=e5194ff24c224fa8ee83064dff73f62f745a4469;p=clang Implement access checking for protected base classes. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@67887 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Sema/SemaAccess.cpp b/lib/Sema/SemaAccess.cpp index cc212434b7..e65b050500 100644 --- a/lib/Sema/SemaAccess.cpp +++ b/lib/Sema/SemaAccess.cpp @@ -55,7 +55,7 @@ bool Sema::CheckBaseClassAccess(QualType Derived, QualType Base, const CXXBaseSpecifier *InacessibleBase = 0; - const CXXRecordDecl* CurrentClassDecl = 0; + CXXRecordDecl* CurrentClassDecl = 0; if (CXXMethodDecl *MD = dyn_cast_or_null(getCurFunctionDecl())) CurrentClassDecl = MD->getParent(); @@ -79,9 +79,21 @@ bool Sema::CheckBaseClassAccess(QualType Derived, QualType Base, if (CurrentClassDecl != Element->Class) FoundInaccessibleBase = true; break; - case AS_protected: - // FIXME: Implement - break; + case AS_protected: + // FIXME: Check if the current function/class is a friend. + if (!CurrentClassDecl) { + FoundInaccessibleBase = true; + break; + } + + if (CurrentClassDecl != Element->Class) { + QualType CurrentClassType = Context.getTypeDeclType(CurrentClassDecl); + QualType ClassType = Context.getTypeDeclType(Element->Class); + + if (!IsDerivedFrom(CurrentClassType, ClassType)) + FoundInaccessibleBase = true; + break; + } } if (FoundInaccessibleBase) { diff --git a/lib/Sema/SemaInherit.h b/lib/Sema/SemaInherit.h index 6138685b26..a164d43e8a 100644 --- a/lib/Sema/SemaInherit.h +++ b/lib/Sema/SemaInherit.h @@ -40,7 +40,7 @@ namespace clang { const CXXBaseSpecifier *Base; /// Class - The record decl of the class that the base is a base of. - const CXXRecordDecl *Class; + CXXRecordDecl *Class; /// SubobjectNumber - Identifies which base class subobject (of type /// @c Base->getType()) this base path element refers to. This diff --git a/test/SemaCXX/access-base-class.cpp b/test/SemaCXX/access-base-class.cpp index f98437695e..3e50b26e5a 100644 --- a/test/SemaCXX/access-base-class.cpp +++ b/test/SemaCXX/access-base-class.cpp @@ -80,3 +80,35 @@ namespace T6 { A *a = c; } } + +namespace T7 { + +class C; +class A { }; +class B : protected A { // expected-note {{'protected' inheritance specifier here}} + void f(C *); +}; + +class C : protected B { // expected-note {{'protected' inheritance specifier here}} + void f(C *c) { + A* a = c; + } +}; + +void B::f(C *c) { + A *a = c; // expected-error {{conversion from 'class T7::C' to inaccessible base class 'class T7::A'}} \ + expected-error {{incompatible type initializing 'class T7::C *', expected 'class T7::A *'}} +} + +class D : private C { + void f(D *d) { + A *a = d; + } +}; + +void f(B* b) { + A *a = b; // expected-error {{conversion from 'class T7::B' to inaccessible base class 'class T7::A'}} \ + expected-error {{incompatible type initializing 'class T7::B *', expected 'class T7::A *'}} +} + +}