| // RUN: %clang_cc1 -fsyntax-only -verify -fexceptions %s |
| |
| // Straight from the standard: |
| // Plain function with spec |
| void f() throw(int); |
| // Pointer to function with spec |
| void (*fp)() throw (int); |
| // Function taking reference to function with spec |
| void g(void pfa() throw(int)); |
| // Typedef for pointer to function with spec |
| typedef int (*pf)() throw(int); // expected-error {{specifications are not allowed in typedefs}} |
| |
| // Some more: |
| // Function returning function with spec |
| void (*h())() throw(int); |
| // Ultimate parser thrill: function with spec returning function with spec and |
| // taking pointer to function with spec. |
| // The actual function throws int, the return type double, the argument float. |
| void (*i() throw(int))(void (*)() throw(float)) throw(double); |
| // Pointer to pointer to function taking function with spec |
| void (**k)(void pfa() throw(int)); // no-error |
| // Pointer to pointer to function with spec |
| void (**j)() throw(int); // expected-error {{not allowed beyond a single}} |
| // Pointer to function returning pointer to pointer to function with spec |
| void (**(*h())())() throw(int); // expected-error {{not allowed beyond a single}} |
| |
| struct Incomplete; // expected-note 3 {{forward declaration}} |
| |
| // Exception spec must not have incomplete types, or pointers to them, except |
| // void. |
| void ic1() throw(void); // expected-error {{incomplete type 'void' is not allowed in exception specification}} |
| void ic2() throw(Incomplete); // expected-error {{incomplete type 'Incomplete' is not allowed in exception specification}} |
| void ic3() throw(void*); |
| void ic4() throw(Incomplete*); // expected-error {{pointer to incomplete type 'Incomplete' is not allowed in exception specification}} |
| void ic5() throw(Incomplete&); // expected-error {{reference to incomplete type 'Incomplete' is not allowed in exception specification}} |
| |
| // Redeclarations |
| typedef int INT; |
| void r1() throw(int); |
| void r1() throw(int); |
| |
| void r2() throw(int); |
| void r2() throw(INT); |
| |
| // throw-any spec and no spec at all are semantically equivalent |
| void r4() throw(int, float); |
| void r4() throw(float, int); |
| |
| void r5() throw(int); // expected-note {{previous declaration}} |
| void r5(); // expected-warning {{missing exception specification}} |
| |
| void r7() throw(int); // expected-note {{previous declaration}} |
| void r7() throw(float); // expected-error {{exception specification in declaration does not match}} |
| |
| // Top-level const doesn't matter. |
| void r8() throw(int); |
| void r8() throw(const int); |
| |
| // Multiple appearances don't matter. |
| void r9() throw(int, int); |
| void r9() throw(int, int); |
| |
| struct A |
| { |
| }; |
| |
| struct B1 : A |
| { |
| }; |
| |
| struct B2 : A |
| { |
| }; |
| |
| struct D : B1, B2 |
| { |
| }; |
| |
| struct P : private A |
| { |
| }; |
| |
| struct Base |
| { |
| virtual void f1() throw(); |
| virtual void f4() throw(int, float); |
| |
| virtual void f5() throw(int, float); |
| virtual void f6() throw(A); |
| virtual void f7() throw(A, int, float); |
| virtual void f8(); |
| |
| virtual void g1() throw(); // expected-note {{overridden virtual function is here}} |
| virtual void g2() throw(int); // expected-note {{overridden virtual function is here}} |
| virtual void g3() throw(A); // expected-note {{overridden virtual function is here}} |
| virtual void g4() throw(B1); // expected-note {{overridden virtual function is here}} |
| virtual void g5() throw(A); // expected-note {{overridden virtual function is here}} |
| }; |
| struct Derived : Base |
| { |
| virtual void f1() throw(); |
| virtual void f4() throw(float, int); |
| |
| virtual void f5() throw(float); |
| virtual void f6() throw(B1); |
| virtual void f7() throw(B1, B2, int); |
| virtual void f8() throw(B2, B2, int, float, char, double, bool); |
| |
| virtual void g1() throw(int); // expected-error {{exception specification of overriding function is more lax}} |
| virtual void g2(); // expected-error {{exception specification of overriding function is more lax}} |
| virtual void g3() throw(D); // expected-error {{exception specification of overriding function is more lax}} |
| virtual void g4() throw(A); // expected-error {{exception specification of overriding function is more lax}} |
| virtual void g5() throw(P); // expected-error {{exception specification of overriding function is more lax}} |
| }; |
| |
| // Some functions to play with below. |
| void s1() throw(); |
| void s2() throw(int); |
| void s3() throw(A); |
| void s4() throw(B1); |
| void s5() throw(D); |
| void s6(); |
| void s7() throw(int, float); |
| void (*s8())() throw(B1); // s8 returns a pointer to function with spec |
| void s9(void (*)() throw(B1)); // s9 takes pointer to function with spec |
| |
| void fnptrs() |
| { |
| // Assignment and initialization of function pointers. |
| void (*t1)() throw() = &s1; // valid |
| t1 = &s2; // expected-error {{not superset}} expected-error {{incompatible type}} |
| t1 = &s3; // expected-error {{not superset}} expected-error {{incompatible type}} |
| void (&t2)() throw() = s2; // expected-error {{not superset}} |
| void (*t3)() throw(int) = &s2; // valid |
| void (*t4)() throw(A) = &s1; // valid |
| t4 = &s3; // valid |
| t4 = &s4; // valid |
| t4 = &s5; // expected-error {{not superset}} expected-error {{incompatible type}} |
| void (*t5)() = &s1; // valid |
| t5 = &s2; // valid |
| t5 = &s6; // valid |
| t5 = &s7; // valid |
| t1 = t3; // expected-error {{not superset}} expected-error {{incompatible type}} |
| t3 = t1; // valid |
| void (*t6)() throw(B1); |
| t6 = t4; // expected-error {{not superset}} expected-error {{incompatible type}} |
| t4 = t6; // valid |
| t5 = t1; // valid |
| t1 = t5; // expected-error {{not superset}} expected-error {{incompatible type}} |
| |
| // return types and arguments must match exactly, no inheritance allowed |
| void (*(*t7)())() throw(B1) = &s8; // valid |
| void (*(*t8)())() throw(A) = &s8; // expected-error {{return types differ}} |
| void (*(*t9)())() throw(D) = &s8; // expected-error {{return types differ}} |
| void (*t10)(void (*)() throw(B1)) = &s9; // valid expected-warning{{disambiguated}} |
| void (*t11)(void (*)() throw(A)) = &s9; // expected-error {{argument types differ}} expected-warning{{disambiguated}} |
| void (*t12)(void (*)() throw(D)) = &s9; // expected-error {{argument types differ}} expected-warning{{disambiguated}} |
| } |
| |
| // Member function stuff |
| |
| struct Str1 { void f() throw(int); }; // expected-note {{previous declaration}} |
| void Str1::f() // expected-warning {{missing exception specification}} |
| { |
| } |
| |
| void mfnptr() |
| { |
| void (Str1::*pfn1)() throw(int) = &Str1::f; // valid |
| void (Str1::*pfn2)() = &Str1::f; // valid |
| void (Str1::*pfn3)() throw() = &Str1::f; // expected-error {{not superset}} |
| } |
| |
| // Don't suppress errors in template instantiation. |
| template <typename T> struct TEx; // expected-note {{template is declared here}} |
| |
| void tf() throw(TEx<int>); // expected-error {{implicit instantiation of undefined template}} |
| |
| // DR 437, class throws itself. |
| struct DR437 { |
| void f() throw(DR437); |
| void g() throw(DR437*); |
| void h() throw(DR437&); |
| }; |
| |
| // DR 437 within a nested class |
| struct DR437_out { |
| struct DR437_in { |
| void f() throw(DR437_out); |
| void g() throw(DR437_out*); |
| void h() throw(DR437_out&); |
| }; |
| }; |