diff options
| author | Anders Carlsson <andersca@mac.com> | 2009-09-10 20:48:14 +0000 |
|---|---|---|
| committer | Anders Carlsson <andersca@mac.com> | 2009-09-10 20:48:14 +0000 |
| commit | 04e1e22fe7a5d2728714fb2e5fd031f341d63640 (patch) | |
| tree | fd6dcc821cb259f71c7cd35321686b97b3e91b68 | |
| parent | 7b98b9291c4b25478393738e61b2d828919de2e9 (diff) | |
| download | bcm5719-llvm-04e1e22fe7a5d2728714fb2e5fd031f341d63640.tar.gz bcm5719-llvm-04e1e22fe7a5d2728714fb2e5fd031f341d63640.zip | |
Don't check use of a member function declaration used if the member function is virtual and the member reference expression doesn't explicitly qualify it. Fixes PR4878.
llvm-svn: 81460
| -rw-r--r-- | clang/lib/Sema/SemaExpr.cpp | 10 | ||||
| -rw-r--r-- | clang/test/SemaCXX/attr-deprecated.cpp | 17 |
2 files changed, 26 insertions, 1 deletions
diff --git a/clang/lib/Sema/SemaExpr.cpp b/clang/lib/Sema/SemaExpr.cpp index 723d6d7dfad..53f3dde2831 100644 --- a/clang/lib/Sema/SemaExpr.cpp +++ b/clang/lib/Sema/SemaExpr.cpp @@ -2142,8 +2142,16 @@ Sema::BuildMemberReferenceExpr(Scope *S, ExprArg Base, SourceLocation OpLoc, if (MemberDecl->isInvalidDecl()) return ExprError(); + bool ShouldCheckUse = true; + if (CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(MemberDecl)) { + // Don't diagnose the use of a virtual member function unless it's + // explicitly qualified. + if (MD->isVirtual() && (!SS || !SS->isSet())) + ShouldCheckUse = false; + } + // Check the use of this field - if (DiagnoseUseOfDecl(MemberDecl, MemberLoc)) + if (ShouldCheckUse && DiagnoseUseOfDecl(MemberDecl, MemberLoc)) return ExprError(); if (FieldDecl *FD = dyn_cast<FieldDecl>(MemberDecl)) { diff --git a/clang/test/SemaCXX/attr-deprecated.cpp b/clang/test/SemaCXX/attr-deprecated.cpp index 99a249e10d9..10b4a55198c 100644 --- a/clang/test/SemaCXX/attr-deprecated.cpp +++ b/clang/test/SemaCXX/attr-deprecated.cpp @@ -24,3 +24,20 @@ void A::h(A* a) (void)b; (void)a->b; } + +struct B { + virtual void f() __attribute__((deprecated)); +}; + +struct C : B { + virtual void f(); +}; + +void f(B* b, C *c) { + b->f(); + b->B::f(); // expected-warning{{'f' is deprecated}} + + c->f(); + c->C::f(); + c->B::f(); // expected-warning{{'f' is deprecated}} +} |

