291 lines
5.8 KiB
C++
291 lines
5.8 KiB
C++
|
// RUN: %clang_cc1 -fsyntax-only -verify %s
|
||
|
|
||
|
friend class A; // expected-error {{'friend' used outside of class}}
|
||
|
void f() { friend class A; } // expected-error {{'friend' used outside of class}}
|
||
|
class C { friend class A; };
|
||
|
class D { void f() { friend class A; } }; // expected-error {{'friend' used outside of class}}
|
||
|
|
||
|
// PR5760
|
||
|
namespace test0 {
|
||
|
namespace ns {
|
||
|
void f(int);
|
||
|
}
|
||
|
|
||
|
struct A {
|
||
|
friend void ns::f(int a);
|
||
|
};
|
||
|
}
|
||
|
|
||
|
// Test derived from LLVM's Registry.h
|
||
|
namespace test1 {
|
||
|
template <class T> struct Outer {
|
||
|
void foo(T);
|
||
|
struct Inner {
|
||
|
friend void Outer::foo(T);
|
||
|
};
|
||
|
};
|
||
|
|
||
|
void test() {
|
||
|
(void) Outer<int>::Inner();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// PR5476
|
||
|
namespace test2 {
|
||
|
namespace foo {
|
||
|
void Func(int x);
|
||
|
}
|
||
|
|
||
|
class Bar {
|
||
|
friend void ::test2::foo::Func(int x);
|
||
|
};
|
||
|
}
|
||
|
|
||
|
// PR5134
|
||
|
namespace test3 {
|
||
|
class Foo {
|
||
|
friend const int getInt(int inInt = 0) {}
|
||
|
|
||
|
};
|
||
|
}
|
||
|
|
||
|
namespace test4 {
|
||
|
class T4A {
|
||
|
friend class T4B;
|
||
|
|
||
|
public:
|
||
|
T4A(class T4B *);
|
||
|
|
||
|
protected:
|
||
|
T4B *mB; // error here
|
||
|
};
|
||
|
|
||
|
class T4B {};
|
||
|
}
|
||
|
|
||
|
namespace rdar8529993 {
|
||
|
struct A { ~A(); };
|
||
|
|
||
|
struct B : A
|
||
|
{
|
||
|
template<int> friend A::~A(); // expected-error {{destructor cannot be declared as a template}}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
// PR7915
|
||
|
namespace test5 {
|
||
|
struct A;
|
||
|
struct A1 { friend void A(); };
|
||
|
|
||
|
struct B { friend void B(); };
|
||
|
}
|
||
|
|
||
|
// PR8479
|
||
|
namespace test6_1 {
|
||
|
class A {
|
||
|
public:
|
||
|
private:
|
||
|
friend class vectorA;
|
||
|
A() {}
|
||
|
};
|
||
|
class vectorA {
|
||
|
public:
|
||
|
vectorA(int i, const A& t = A()) {}
|
||
|
};
|
||
|
void f() {
|
||
|
vectorA v(1);
|
||
|
}
|
||
|
}
|
||
|
namespace test6_2 {
|
||
|
template<class T>
|
||
|
class vector {
|
||
|
public:
|
||
|
vector(int i, const T& t = T()) {}
|
||
|
};
|
||
|
class A {
|
||
|
public:
|
||
|
private:
|
||
|
friend class vector<A>;
|
||
|
A() {}
|
||
|
};
|
||
|
void f() {
|
||
|
vector<A> v(1);
|
||
|
}
|
||
|
}
|
||
|
namespace test6_3 {
|
||
|
template<class T>
|
||
|
class vector {
|
||
|
public:
|
||
|
vector(int i) {}
|
||
|
void f(const T& t = T()) {}
|
||
|
};
|
||
|
class A {
|
||
|
public:
|
||
|
private:
|
||
|
friend void vector<A>::f(const A&);
|
||
|
A() {}
|
||
|
};
|
||
|
void f() {
|
||
|
vector<A> v(1);
|
||
|
v.f();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
namespace test7 {
|
||
|
extern "C" {
|
||
|
class X {
|
||
|
friend int test7_f() { return 42; }
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// PR15485
|
||
|
namespace test8 {
|
||
|
namespace ns1 {
|
||
|
namespace ns2 {
|
||
|
template<class T> void f(T t); // expected-note {{target of using declaration}}
|
||
|
}
|
||
|
using ns2::f; // expected-note {{using declaration}}
|
||
|
}
|
||
|
struct A { void f(); }; // expected-note {{target of using declaration}}
|
||
|
struct B : public A { using A::f; }; // expected-note {{using declaration}}
|
||
|
struct X {
|
||
|
template<class T> friend void ns1::f(T t); // expected-error {{cannot befriend target of using declaration}}
|
||
|
friend void B::f(); // expected-error {{cannot befriend target of using declaration}}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
// PR16423
|
||
|
namespace test9 {
|
||
|
class C {
|
||
|
};
|
||
|
struct A {
|
||
|
friend void C::f(int, int, int) {} // expected-error {{no function named 'f' with type 'void (int, int, int)' was found in the specified scope}}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
namespace test10 {
|
||
|
struct X {};
|
||
|
extern void f10_a();
|
||
|
extern void f10_a(X);
|
||
|
struct A {
|
||
|
friend void f10_a();
|
||
|
friend void f10_b();
|
||
|
friend void f10_c();
|
||
|
friend void f10_d();
|
||
|
friend void f10_a(X);
|
||
|
friend void f10_b(X);
|
||
|
friend void f10_c(X);
|
||
|
friend void f10_d(X);
|
||
|
};
|
||
|
extern void f10_b();
|
||
|
extern void f10_b(X);
|
||
|
struct B {
|
||
|
friend void f10_a();
|
||
|
friend void f10_b();
|
||
|
friend void f10_c();
|
||
|
friend void f10_d();
|
||
|
friend void f10_a(X);
|
||
|
friend void f10_b(X);
|
||
|
friend void f10_c(X);
|
||
|
friend void f10_d(X);
|
||
|
};
|
||
|
extern void f10_c();
|
||
|
extern void f10_c(X);
|
||
|
|
||
|
// FIXME: Give a better diagnostic for the case where a function exists but is
|
||
|
// not visible.
|
||
|
void g(X x) {
|
||
|
f10_a();
|
||
|
f10_b();
|
||
|
f10_c();
|
||
|
f10_d(); // expected-error {{undeclared identifier}}
|
||
|
|
||
|
::test10::f10_a();
|
||
|
::test10::f10_b();
|
||
|
::test10::f10_c();
|
||
|
::test10::f10_d(); // expected-error {{no member named 'f10_d'}}
|
||
|
|
||
|
f10_a(x);
|
||
|
f10_b(x);
|
||
|
f10_c(x);
|
||
|
f10_d(x); // PR16597: expected-error {{undeclared identifier}}
|
||
|
|
||
|
::test10::f10_a(x);
|
||
|
::test10::f10_b(x);
|
||
|
::test10::f10_c(x);
|
||
|
::test10::f10_d(x); // expected-error {{no type named 'f10_d'}}
|
||
|
}
|
||
|
|
||
|
struct Y : X {
|
||
|
friend void f10_d();
|
||
|
friend void f10_d(X);
|
||
|
};
|
||
|
|
||
|
struct Z {
|
||
|
operator X();
|
||
|
friend void f10_d();
|
||
|
friend void f10_d(X);
|
||
|
};
|
||
|
|
||
|
void g(X x, Y y, Z z) {
|
||
|
f10_d(); // expected-error {{undeclared identifier}}
|
||
|
::test10::f10_d(); // expected-error {{no member named 'f10_d'}}
|
||
|
|
||
|
// f10_d is visible to ADL in the second and third cases.
|
||
|
f10_d(x); // expected-error {{undeclared identifier}}
|
||
|
f10_d(y);
|
||
|
f10_d(z);
|
||
|
|
||
|
// No ADL here.
|
||
|
::test10::f10_d(x); // expected-error {{no type named 'f10_d'}}
|
||
|
::test10::f10_d(y); // expected-error {{no type named 'f10_d'}}
|
||
|
::test10::f10_d(z); // expected-error {{no type named 'f10_d'}}
|
||
|
}
|
||
|
|
||
|
void local_externs(X x, Y y) {
|
||
|
extern void f10_d();
|
||
|
extern void f10_d(X);
|
||
|
f10_d();
|
||
|
f10_d(x);
|
||
|
// FIXME: This lookup should fail, because the local extern declaration
|
||
|
// should suppress ADL.
|
||
|
f10_d(y);
|
||
|
{
|
||
|
int f10_d;
|
||
|
f10_d(); // expected-error {{not a function}}
|
||
|
f10_d(x); // expected-error {{not a function}}
|
||
|
f10_d(y); // expected-error {{not a function}}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void i(X x, Y y) {
|
||
|
f10_d(); // expected-error {{undeclared identifier}}
|
||
|
f10_d(x); // expected-error {{undeclared identifier}}
|
||
|
f10_d(y);
|
||
|
}
|
||
|
|
||
|
struct C {
|
||
|
friend void f10_d();
|
||
|
friend void f10_d(X);
|
||
|
};
|
||
|
|
||
|
void j(X x, Y y) {
|
||
|
f10_d(); // expected-error {{undeclared identifier}}
|
||
|
f10_d(x); // expected-error {{undeclared identifier}}
|
||
|
f10_d(y);
|
||
|
}
|
||
|
|
||
|
extern void f10_d();
|
||
|
extern void f10_d(X);
|
||
|
void k(X x, Y y, Z z) {
|
||
|
// All OK now.
|
||
|
f10_d();
|
||
|
f10_d(x);
|
||
|
::test10::f10_d();
|
||
|
::test10::f10_d(x);
|
||
|
::test10::f10_d(y);
|
||
|
::test10::f10_d(z);
|
||
|
}
|
||
|
}
|