| // RUN: %clang_cc1 -Wreturn-type -fsyntax-only -std=c++11 -verify %s | 
 |  | 
 | class A { | 
 | public: | 
 |   A(const A&); | 
 | }; | 
 |  | 
 | struct S { | 
 |   int i; | 
 |   double d; | 
 |  | 
 |   virtual void B() {} | 
 | }; | 
 |  | 
 | union U { | 
 |   struct { | 
 |     int i; | 
 |     virtual void B() {} // Can only do this in C++11 | 
 |   } t; | 
 | }; | 
 |  | 
 | struct S2 { | 
 |   int i; | 
 |   double d; | 
 | }; | 
 |  | 
 | extern "C" U f3( void ); // expected-warning {{'f3' has C-linkage specified, but returns user-defined type 'U' which is incompatible with C}} | 
 | extern "C" S f0(void); // expected-warning {{'f0' has C-linkage specified, but returns user-defined type 'S' which is incompatible with C}} | 
 | extern "C" A f4( void ); // expected-warning {{'f4' has C-linkage specified, but returns user-defined type 'A' which is incompatible with C}} | 
 |  | 
 | // These should all be fine | 
 | extern "C" S2 f5( void ); | 
 | extern "C" void f2( A x ); | 
 | extern "C" void f6( S s ); | 
 | extern "C" void f7( U u ); | 
 | extern "C" double f8(void); | 
 | extern "C" long long f11( void ); | 
 | extern "C" A *f10( void ); | 
 |  | 
 | extern "C" struct mypodstruct f12(); // expected-warning {{'f12' has C-linkage specified, but returns incomplete type 'struct mypodstruct' which could be incompatible with C}} | 
 |  | 
 | namespace test2 { | 
 |   // FIXME: we should probably suppress the first warning as the second one | 
 |   // is more precise. | 
 |   // For now this tests that a second 'extern "C"' is not necessary to trigger | 
 |   // the warning. | 
 |   struct A; | 
 |   extern "C" A f(void); // expected-warning {{'f' has C-linkage specified, but returns incomplete type 'test2::A' which could be incompatible with C}} | 
 |   struct A { | 
 |     A(const A&); | 
 |   }; | 
 |   A f(void);  // no warning. warning is already issued on first declaration. | 
 | } | 
 |  | 
 | namespace test3 { | 
 |   struct A { | 
 |     A(const A&); | 
 |   }; | 
 |   extern "C" { | 
 |     // Don't warn for static functions. | 
 |     static A f(void); | 
 |   } | 
 | } | 
 |  | 
 | // rdar://13364028 | 
 | namespace rdar13364028 { | 
 | class A { | 
 | public: | 
 |     virtual int x(); | 
 | }; | 
 |  | 
 | extern "C" { | 
 | #pragma clang diagnostic push | 
 | #pragma clang diagnostic ignored "-Wreturn-type-c-linkage" | 
 | A xyzzy(); | 
 | #pragma clang diagnostic pop | 
 | A bbb(); // expected-warning {{'bbb' has C-linkage specified, but returns user-defined type 'rdar13364028::A' which is incompatible with C}} | 
 | A ccc() { // expected-warning {{'ccc' has C-linkage specified, but returns user-defined type 'rdar13364028::A' which is incompatible with C}} | 
 |   return A(); | 
 | }; | 
 | } | 
 |  | 
 | A xyzzy(); | 
 |  | 
 | A xyzzy() | 
 | { | 
 |   return A(); | 
 | } | 
 |  | 
 | A bbb() | 
 | { | 
 |   return A(); | 
 | } | 
 |  | 
 | A bbb(); | 
 |  | 
 | A ccc(); | 
 | } |