511 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			511 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
// RUN: %clang_cc1 -fcxx-exceptions -fexceptions -fsyntax-only -verify %s
 | 
						|
 | 
						|
// C++0x [class.access]p4:
 | 
						|
 | 
						|
//   Access control is applied uniformly to all names, whether the
 | 
						|
//   names are referred to from declarations or expressions.  In the
 | 
						|
//   case of overloaded function names, access control is applied to
 | 
						|
//   the function selected by overload resolution.
 | 
						|
 | 
						|
class Public {} PublicInst;
 | 
						|
class Protected {} ProtectedInst;
 | 
						|
class Private {} PrivateInst;
 | 
						|
 | 
						|
namespace test0 {
 | 
						|
  class A {
 | 
						|
  public:
 | 
						|
    void foo(Public&);
 | 
						|
  protected:
 | 
						|
    void foo(Protected&); // expected-note 2 {{declared protected here}}
 | 
						|
  private:
 | 
						|
    void foo(Private&); // expected-note 2 {{declared private here}}
 | 
						|
  };
 | 
						|
 | 
						|
  void test(A *op) {
 | 
						|
    op->foo(PublicInst);
 | 
						|
    op->foo(ProtectedInst); // expected-error {{'foo' is a protected member}}
 | 
						|
    op->foo(PrivateInst); // expected-error {{'foo' is a private member}}
 | 
						|
 | 
						|
    void (A::*a)(Public&) = &A::foo;
 | 
						|
    void (A::*b)(Protected&) = &A::foo; // expected-error {{'foo' is a protected member}}
 | 
						|
    void (A::*c)(Private&) = &A::foo; // expected-error {{'foo' is a private member}}
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// Member operators.
 | 
						|
namespace test1 {
 | 
						|
  class A {
 | 
						|
  public:
 | 
						|
    void operator+(Public&);
 | 
						|
    void operator[](Public&);
 | 
						|
    void operator()(Public&);
 | 
						|
    typedef void (*PublicSurrogate)(Public&);
 | 
						|
    operator PublicSurrogate() const;
 | 
						|
  protected:
 | 
						|
    void operator+(Protected&); // expected-note {{declared protected here}}
 | 
						|
    void operator[](Protected&); // expected-note {{declared protected here}}
 | 
						|
    void operator()(Protected&); // expected-note {{declared protected here}}
 | 
						|
    typedef void (*ProtectedSurrogate)(Protected&);
 | 
						|
    operator ProtectedSurrogate() const; // expected-note {{declared protected here}}
 | 
						|
  private:
 | 
						|
    void operator+(Private&); // expected-note {{declared private here}}
 | 
						|
    void operator[](Private&); // expected-note {{declared private here}}
 | 
						|
    void operator()(Private&); // expected-note {{declared private here}}
 | 
						|
    void operator-(); // expected-note {{declared private here}}
 | 
						|
    typedef void (*PrivateSurrogate)(Private&);
 | 
						|
    operator PrivateSurrogate() const; // expected-note {{declared private here}}
 | 
						|
  };
 | 
						|
  void operator+(const A &, Public&);
 | 
						|
  void operator+(const A &, Protected&);
 | 
						|
  void operator+(const A &, Private&);
 | 
						|
  void operator-(const A &);
 | 
						|
 | 
						|
  void test(A &a, Public &pub, Protected &prot, Private &priv) {
 | 
						|
    a + pub;
 | 
						|
    a + prot; // expected-error {{'operator+' is a protected member}}
 | 
						|
    a + priv; // expected-error {{'operator+' is a private member}}
 | 
						|
    a[pub];
 | 
						|
    a[prot]; // expected-error {{'operator[]' is a protected member}}
 | 
						|
    a[priv]; // expected-error {{'operator[]' is a private member}}
 | 
						|
    a(pub);
 | 
						|
    a(prot); // expected-error {{'operator()' is a protected member}}
 | 
						|
    a(priv); // expected-error {{'operator()' is a private member}}
 | 
						|
    -a;       // expected-error {{'operator-' is a private member}}
 | 
						|
 | 
						|
    const A &ca = a;
 | 
						|
    ca + pub;
 | 
						|
    ca + prot;
 | 
						|
    ca + priv;
 | 
						|
    -ca;
 | 
						|
    // These are all surrogate calls
 | 
						|
    ca(pub);
 | 
						|
    ca(prot); // expected-error {{'operator void (*)(class Protected &)' is a protected member}}
 | 
						|
    ca(priv); // expected-error {{'operator void (*)(class Private &)' is a private member}}
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// Implicit constructor calls.
 | 
						|
namespace test2 {
 | 
						|
  class A {
 | 
						|
  private:
 | 
						|
    A(); // expected-note 3 {{declared private here}}
 | 
						|
 | 
						|
    static A foo;
 | 
						|
  };
 | 
						|
 | 
						|
  A a; // expected-error {{calling a private constructor}}
 | 
						|
  A A::foo; // okay
 | 
						|
  
 | 
						|
  class B : A { }; // expected-error {{base class 'test2::A' has private default constructor}}
 | 
						|
  B b; // expected-note{{implicit default constructor}}
 | 
						|
  
 | 
						|
  class C : virtual A { 
 | 
						|
  public:
 | 
						|
    C();
 | 
						|
  };
 | 
						|
 | 
						|
  class D : C { }; // expected-error {{inherited virtual base class 'test2::A' has private default constructor}}
 | 
						|
  D d; // expected-note{{implicit default constructor}}
 | 
						|
}
 | 
						|
 | 
						|
// Implicit destructor calls.
 | 
						|
namespace test3 {
 | 
						|
  class A {
 | 
						|
  private:
 | 
						|
    ~A(); // expected-note 2 {{declared private here}}
 | 
						|
    static A foo;
 | 
						|
  };
 | 
						|
 | 
						|
  A a; // expected-error {{variable of type 'test3::A' has private destructor}}
 | 
						|
  A A::foo;
 | 
						|
 | 
						|
  void foo(A param) { // okay
 | 
						|
    A local; // expected-error {{variable of type 'test3::A' has private destructor}}
 | 
						|
  }
 | 
						|
 | 
						|
  template <unsigned N> class Base { ~Base(); }; // expected-note 14 {{declared private here}}
 | 
						|
  class Base2 : virtual Base<2> { ~Base2(); }; // expected-note 3 {{declared private here}} \
 | 
						|
                                               // expected-error {{base class 'Base<2>' has private destructor}}
 | 
						|
  class Base3 : virtual Base<3> { public: ~Base3(); }; // expected-error {{base class 'Base<3>' has private destructor}}
 | 
						|
 | 
						|
  // These don't cause diagnostics because we don't need the destructor.
 | 
						|
  class Derived0 : Base<0> { ~Derived0(); };
 | 
						|
  class Derived1 : Base<1> { };
 | 
						|
 | 
						|
  class Derived2 : // expected-error {{inherited virtual base class 'Base<2>' has private destructor}} \
 | 
						|
                   // expected-error {{inherited virtual base class 'Base<3>' has private destructor}}
 | 
						|
    Base<0>,  // expected-error {{base class 'Base<0>' has private destructor}}
 | 
						|
    virtual Base<1>, // expected-error {{base class 'Base<1>' has private destructor}}
 | 
						|
    Base2, // expected-error {{base class 'test3::Base2' has private destructor}}
 | 
						|
    virtual Base3
 | 
						|
  {
 | 
						|
    ~Derived2() {}
 | 
						|
  };
 | 
						|
 | 
						|
  class Derived3 : // expected-error 2 {{inherited virtual base class 'Base<2>' has private destructor}} \
 | 
						|
                   // expected-error 2 {{inherited virtual base class 'Base<3>' has private destructor}} \
 | 
						|
    // expected-note 2{{implicit default constructor}}
 | 
						|
    Base<0>,  // expected-error 2 {{base class 'Base<0>' has private destructor}}
 | 
						|
    virtual Base<1>, // expected-error 2 {{base class 'Base<1>' has private destructor}}
 | 
						|
    Base2, // expected-error 2 {{base class 'test3::Base2' has private destructor}}
 | 
						|
    virtual Base3
 | 
						|
  {}; 
 | 
						|
  Derived3 d3; // expected-note {{implicit default constructor}}\
 | 
						|
               // expected-note{{implicit default destructor}}}
 | 
						|
}
 | 
						|
 | 
						|
// Conversion functions.
 | 
						|
namespace test4 {
 | 
						|
  class Base {
 | 
						|
  private:
 | 
						|
    operator Private(); // expected-note 4 {{declared private here}}
 | 
						|
  public:
 | 
						|
    operator Public(); // expected-note 2{{member is declared here}}
 | 
						|
  };
 | 
						|
 | 
						|
  class Derived1 : private Base { // expected-note 2 {{declared private here}} \
 | 
						|
                                  // expected-note {{constrained by private inheritance}}
 | 
						|
    Private test1() { return *this; } // expected-error {{'operator Private' is a private member}}
 | 
						|
    Public test2() { return *this; }
 | 
						|
  };
 | 
						|
  Private test1(Derived1 &d) { return d; } // expected-error {{'operator Private' is a private member}} \
 | 
						|
                                           // expected-error {{cannot cast 'test4::Derived1' to its private base class}}
 | 
						|
  Public test2(Derived1 &d) { return d; } // expected-error {{cannot cast 'test4::Derived1' to its private base class}} \
 | 
						|
                                          // expected-error {{'operator Public' is a private member}}
 | 
						|
 | 
						|
 | 
						|
  class Derived2 : public Base {
 | 
						|
    Private test1() { return *this; } // expected-error {{'operator Private' is a private member}}
 | 
						|
    Public test2() { return *this; }
 | 
						|
  };
 | 
						|
  Private test1(Derived2 &d) { return d; } // expected-error {{'operator Private' is a private member}}
 | 
						|
  Public test2(Derived2 &d) { return d; }
 | 
						|
 | 
						|
  class Derived3 : private Base { // expected-note {{constrained by private inheritance here}} \
 | 
						|
                                  // expected-note {{declared private here}}
 | 
						|
  public:
 | 
						|
    operator Private();
 | 
						|
  };
 | 
						|
  Private test1(Derived3 &d) { return d; }
 | 
						|
  Public test2(Derived3 &d) { return d; } // expected-error {{'operator Public' is a private member of 'test4::Base'}} \
 | 
						|
                                          // expected-error {{cannot cast 'test4::Derived3' to its private base class}}
 | 
						|
 | 
						|
  class Derived4 : public Base {
 | 
						|
  public:
 | 
						|
    operator Private();
 | 
						|
  };
 | 
						|
  Private test1(Derived4 &d) { return d; }
 | 
						|
  Public test2(Derived4 &d) { return d; }
 | 
						|
}
 | 
						|
 | 
						|
// Implicit copy assignment operator uses.
 | 
						|
namespace test5 {
 | 
						|
  class A {
 | 
						|
    void operator=(const A &); // expected-note 2 {{implicitly declared private here}}
 | 
						|
  };
 | 
						|
 | 
						|
  class Test1 { A a; }; // expected-error {{private member}}
 | 
						|
  void test1() {
 | 
						|
    Test1 a; 
 | 
						|
    a = Test1(); // expected-note{{implicit default copy}}
 | 
						|
  }
 | 
						|
 | 
						|
  class Test2 : A {}; // expected-error {{private member}}
 | 
						|
  void test2() {
 | 
						|
    Test2 a;
 | 
						|
    a = Test2(); // expected-note{{implicit default copy}}
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// Implicit copy constructor uses.
 | 
						|
namespace test6 {
 | 
						|
  class A {
 | 
						|
    public: A();
 | 
						|
    private: A(const A &); // expected-note 2 {{declared private here}}
 | 
						|
  };
 | 
						|
 | 
						|
  class Test1 { A a; }; // expected-error {{field of type 'test6::A' has private copy constructor}}
 | 
						|
  void test1(const Test1 &t) {
 | 
						|
    Test1 a = t; // expected-note{{implicit default copy}}
 | 
						|
  }
 | 
						|
 | 
						|
  class Test2 : A {}; // expected-error {{base class 'test6::A' has private copy constructor}}
 | 
						|
  void test2(const Test2 &t) {
 | 
						|
    Test2 a = t; // expected-note{{implicit default copy}}
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// Redeclaration lookups are not accesses.
 | 
						|
namespace test7 {
 | 
						|
  class A {
 | 
						|
    int private_member;
 | 
						|
  };
 | 
						|
  class B : A {
 | 
						|
    int foo(int private_member) {
 | 
						|
      return 0;
 | 
						|
    }
 | 
						|
  };
 | 
						|
}
 | 
						|
 | 
						|
// Ignored operator new and delete overloads are not 
 | 
						|
namespace test8 {
 | 
						|
  typedef __typeof__(sizeof(int)) size_t;
 | 
						|
 | 
						|
  class A {
 | 
						|
    void *operator new(size_t s);
 | 
						|
    void operator delete(void *p);
 | 
						|
  public:
 | 
						|
    void *operator new(size_t s, int n);
 | 
						|
    void operator delete(void *p, int n);
 | 
						|
  };
 | 
						|
 | 
						|
  void test() {
 | 
						|
    new (2) A();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// Don't silently upgrade forbidden-access paths to private.
 | 
						|
namespace test9 {
 | 
						|
  class A {
 | 
						|
  public: static int x; // expected-note {{member is declared here}}
 | 
						|
  };
 | 
						|
  class B : private A { // expected-note {{constrained by private inheritance here}}
 | 
						|
  };
 | 
						|
  class C : public B {
 | 
						|
    static int getX() { return x; } // expected-error {{'x' is a private member of 'test9::A'}}
 | 
						|
  };
 | 
						|
}
 | 
						|
 | 
						|
namespace test10 {
 | 
						|
  class A {
 | 
						|
    enum {
 | 
						|
      value = 10 // expected-note {{declared private here}}
 | 
						|
    };
 | 
						|
    friend class C;
 | 
						|
  };
 | 
						|
 | 
						|
  class B {
 | 
						|
    enum {
 | 
						|
      value = A::value // expected-error {{'value' is a private member of 'test10::A'}}
 | 
						|
    };
 | 
						|
  };
 | 
						|
 | 
						|
  class C {
 | 
						|
    enum {
 | 
						|
      value = A::value
 | 
						|
    };
 | 
						|
  };
 | 
						|
}
 | 
						|
 | 
						|
namespace test11 {
 | 
						|
  class A {
 | 
						|
    protected: virtual ~A();
 | 
						|
  };
 | 
						|
 | 
						|
  class B : public A {
 | 
						|
    ~B();
 | 
						|
  };
 | 
						|
 | 
						|
  B::~B() {};
 | 
						|
}
 | 
						|
 | 
						|
namespace test12 {
 | 
						|
  class A {
 | 
						|
    int x;
 | 
						|
 | 
						|
    void foo() {
 | 
						|
      class Local {
 | 
						|
        int foo(A *a) {
 | 
						|
          return a->x;
 | 
						|
        }
 | 
						|
      };
 | 
						|
    }
 | 
						|
  };
 | 
						|
}
 | 
						|
 | 
						|
namespace test13 {
 | 
						|
  struct A {
 | 
						|
    int x;
 | 
						|
    unsigned foo() const;
 | 
						|
  };
 | 
						|
 | 
						|
  struct B : protected A {
 | 
						|
    using A::foo;
 | 
						|
    using A::x;
 | 
						|
  };
 | 
						|
 | 
						|
  void test() {
 | 
						|
    A *d;
 | 
						|
    d->foo();
 | 
						|
    (void) d->x;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// Destructors for temporaries.
 | 
						|
namespace test14 {
 | 
						|
  class A {
 | 
						|
  private: ~A(); // expected-note {{declared private here}}
 | 
						|
  };
 | 
						|
  A foo();
 | 
						|
 | 
						|
  void test() {
 | 
						|
    foo(); // expected-error {{temporary of type 'test14::A' has private destructor}}
 | 
						|
  }
 | 
						|
 | 
						|
  class X {
 | 
						|
    ~X(); // expected-note {{declared private here}}
 | 
						|
  };
 | 
						|
  
 | 
						|
  struct Y1 {
 | 
						|
    operator X();
 | 
						|
  };
 | 
						|
  
 | 
						|
  void g() {
 | 
						|
    const X &xr = Y1(); // expected-error{{temporary of type 'test14::X' has private destructor}}
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// PR 7024
 | 
						|
namespace test15 {
 | 
						|
  template <class T> class A {
 | 
						|
  private:
 | 
						|
    int private_foo; // expected-note {{declared private here}}
 | 
						|
    static int private_sfoo; // expected-note {{declared private here}}
 | 
						|
  protected:
 | 
						|
    int protected_foo; // expected-note 3 {{declared protected here}} // expected-note {{can only access this member on an object of type 'test15::B<int>'}}
 | 
						|
    static int protected_sfoo; // expected-note 3 {{declared protected here}}
 | 
						|
 | 
						|
    int test1(A<int> &a) {
 | 
						|
      return a.private_foo; // expected-error {{private member}}
 | 
						|
    }
 | 
						|
 | 
						|
    int test2(A<int> &a) {
 | 
						|
      return a.private_sfoo; // expected-error {{private member}}
 | 
						|
    }
 | 
						|
 | 
						|
    int test3(A<int> &a) {
 | 
						|
      return a.protected_foo; // expected-error {{protected member}}
 | 
						|
    }
 | 
						|
 | 
						|
    int test4(A<int> &a) {
 | 
						|
      return a.protected_sfoo; // expected-error {{protected member}}
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  template class A<int>;
 | 
						|
  template class A<long>; // expected-note 4 {{in instantiation}} 
 | 
						|
 | 
						|
  template <class T> class B : public A<T> {
 | 
						|
    // TODO: These first two accesses can be detected as ill-formed at
 | 
						|
    // definition time because they're member accesses and A<int> can't
 | 
						|
    // be a subclass of B<T> for any T.
 | 
						|
 | 
						|
    int test1(A<int> &a) {
 | 
						|
      return a.protected_foo; // expected-error 2 {{protected member}}
 | 
						|
    }
 | 
						|
 | 
						|
    int test2(A<int> &a) {
 | 
						|
      return a.protected_sfoo; // expected-error {{protected member}}
 | 
						|
    }
 | 
						|
 | 
						|
    int test3(B<int> &b) {
 | 
						|
      return b.protected_foo; // expected-error {{protected member}}
 | 
						|
    }
 | 
						|
 | 
						|
    int test4(B<int> &b) {
 | 
						|
      return b.protected_sfoo; // expected-error {{protected member}}
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  template class B<int>;  // expected-note {{in instantiation}}
 | 
						|
  template class B<long>; // expected-note 4 {{in instantiation}}
 | 
						|
}
 | 
						|
 | 
						|
// PR7281
 | 
						|
namespace test16 {
 | 
						|
  class A { ~A(); }; // expected-note 2{{declared private here}}
 | 
						|
  void b() { throw A(); } // expected-error{{temporary of type 'test16::A' has private destructor}} \
 | 
						|
  // expected-error{{exception object of type 'test16::A' has private destructor}}
 | 
						|
}
 | 
						|
 | 
						|
// rdar://problem/8146294
 | 
						|
namespace test17 {
 | 
						|
  class A {
 | 
						|
    template <typename T> class Inner { }; // expected-note {{declared private here}}
 | 
						|
  };
 | 
						|
 | 
						|
  A::Inner<int> s; // expected-error {{'Inner' is a private member of 'test17::A'}}
 | 
						|
}
 | 
						|
 | 
						|
namespace test18 {
 | 
						|
  template <class T> class A {};
 | 
						|
  class B : A<int> {
 | 
						|
    A<int> member;
 | 
						|
  };
 | 
						|
 | 
						|
  // FIXME: this access to A should be forbidden (because C++ is dumb),
 | 
						|
  // but LookupResult can't express the necessary information to do
 | 
						|
  // the check, so we aggressively suppress access control.
 | 
						|
  class C : B {
 | 
						|
    A<int> member;
 | 
						|
  };
 | 
						|
}
 | 
						|
 | 
						|
// PR8325
 | 
						|
namespace test19 {
 | 
						|
  class A { ~A(); };
 | 
						|
  // The destructor is not implicitly referenced here.  Contrast to test16, 
 | 
						|
  // testing PR7281, earlier in this file.
 | 
						|
  void b(A* x) { throw x; }
 | 
						|
}
 | 
						|
 | 
						|
// PR7930
 | 
						|
namespace test20 {
 | 
						|
  class Foo {
 | 
						|
    Foo(); // expected-note {{implicitly declared private here}}
 | 
						|
  };
 | 
						|
  Foo::Foo() {}
 | 
						|
 | 
						|
  void test() {
 | 
						|
    Foo a; // expected-error {{calling a private constructor}}
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
namespace test21 {
 | 
						|
  template <class T> class A {
 | 
						|
    void foo();
 | 
						|
    void bar();
 | 
						|
    class Inner; // expected-note {{implicitly declared private here}}
 | 
						|
  public:
 | 
						|
    void baz();
 | 
						|
  };
 | 
						|
  template <class T> class A<T>::Inner {};
 | 
						|
  class B {
 | 
						|
    template <class T> class A<T>::Inner; // expected-error{{non-friend class member 'Inner' cannot have a qualified name}}
 | 
						|
  };
 | 
						|
 | 
						|
  void test() {
 | 
						|
    A<int>::Inner i; // expected-error {{'Inner' is a private member}}
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
namespace rdar8876150 {
 | 
						|
  struct A { operator bool(); };
 | 
						|
  struct B : private A { using A::operator bool; };
 | 
						|
 | 
						|
  bool f() {
 | 
						|
    B b;
 | 
						|
    return !b;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
namespace test23 {
 | 
						|
  template <typename T> class A {
 | 
						|
    A();
 | 
						|
    static A instance;
 | 
						|
  };
 | 
						|
 | 
						|
  template <typename T> A<T> A<T>::instance;
 | 
						|
  template class A<int>;
 | 
						|
}
 |