summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRafael Espindola <rafael.espindola@gmail.com>2012-05-01 20:58:29 +0000
committerRafael Espindola <rafael.espindola@gmail.com>2012-05-01 20:58:29 +0000
commita01ff786ed48ebc310615caa7ad6ea6d216f0552 (patch)
treefcfce62c597f4ee067abc4ad2299d4f0377946d9
parent882e14c8022b504a0e142b9d64f77810196f70f1 (diff)
downloadbcm5719-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.cpp8
-rw-r--r--clang/test/Sema/attr-visibility.c1
-rw-r--r--clang/test/SemaCXX/attr-visibility.cpp16
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>() {
+}
OpenPOWER on IntegriCloud