diff options
author | Rafael Espindola <rafael.espindola@gmail.com> | 2012-05-01 20:58:29 +0000 |
---|---|---|
committer | Rafael Espindola <rafael.espindola@gmail.com> | 2012-05-01 20:58:29 +0000 |
commit | a01ff786ed48ebc310615caa7ad6ea6d216f0552 (patch) | |
tree | fcfce62c597f4ee067abc4ad2299d4f0377946d9 | |
parent | 882e14c8022b504a0e142b9d64f77810196f70f1 (diff) | |
download | bcm5719-llvm-a01ff786ed48ebc310615caa7ad6ea6d216f0552.tar.gz bcm5719-llvm-a01ff786ed48ebc310615caa7ad6ea6d216f0552.zip |
Extend the error about incompatible visibility attributes in different
decls to work on function templates specializations.
llvm-svn: 155943
-rw-r--r-- | clang/lib/Sema/SemaDeclAttr.cpp | 8 | ||||
-rw-r--r-- | clang/test/Sema/attr-visibility.c | 1 | ||||
-rw-r--r-- | clang/test/SemaCXX/attr-visibility.cpp | 16 |
3 files changed, 24 insertions, 1 deletions
diff --git a/clang/lib/Sema/SemaDeclAttr.cpp b/clang/lib/Sema/SemaDeclAttr.cpp index 47e393c18dd..c8af3a430a2 100644 --- a/clang/lib/Sema/SemaDeclAttr.cpp +++ b/clang/lib/Sema/SemaDeclAttr.cpp @@ -1752,7 +1752,13 @@ static void handleVisibilityAttr(Sema &S, Decl *D, const AttributeList &Attr) { return; } - VisibilityAttr *PrevAttr = D->getCanonicalDecl()->getAttr<VisibilityAttr>(); + Decl *PrevDecl; + if (isa<FunctionDecl>(D)) + PrevDecl = D->getMostRecentDecl()->getPreviousDecl(); + else + PrevDecl = D->getCanonicalDecl(); + + VisibilityAttr *PrevAttr = PrevDecl ? PrevDecl->getAttr<VisibilityAttr>() : 0; if (PrevAttr) { VisibilityAttr::VisibilityType PrevVisibility = PrevAttr->getVisibility(); if (PrevVisibility != type) { diff --git a/clang/test/Sema/attr-visibility.c b/clang/test/Sema/attr-visibility.c index afeb4f1546f..499111f86cc 100644 --- a/clang/test/Sema/attr-visibility.c +++ b/clang/test/Sema/attr-visibility.c @@ -8,4 +8,5 @@ void test2() __attribute__((visibility("internal"))); void test3() __attribute__((visibility("protected"))); // expected-warning {{target does not support 'protected' visibility; using 'default'}} struct __attribute__((visibility("hidden"))) test4; // expected-note {{previous attribute is here}} +struct test4; struct __attribute__((visibility("default"))) test4; // expected-error {{visibility does not match previous declaration}} diff --git a/clang/test/SemaCXX/attr-visibility.cpp b/clang/test/SemaCXX/attr-visibility.cpp new file mode 100644 index 00000000000..26dc67c6de2 --- /dev/null +++ b/clang/test/SemaCXX/attr-visibility.cpp @@ -0,0 +1,16 @@ +// RUN: %clang_cc1 -fsyntax-only -verify %s + +template <class Element> +void foo() { +} +template <> + __attribute__((visibility("hidden"))) // expected-note {{previous attribute is here}} +void foo<int>(); + +template <> +void foo<int>(); + +template <> + __attribute__((visibility("default"))) // expected-error {{visibility does not match previous declaration}} +void foo<int>() { +} |