Home | History | Annotate | Download | only in SemaCXX
      1 // RUN: %clang_cc1 -triple i686-win32     -fsyntax-only -fms-extensions -verify -std=c++11 -Wunsupported-dll-base-class-template -DMS %s
      2 // RUN: %clang_cc1 -triple x86_64-win32   -fsyntax-only -fms-extensions -verify -std=c++1y -Wunsupported-dll-base-class-template -DMS %s
      3 // RUN: %clang_cc1 -triple i686-mingw32   -fsyntax-only -fms-extensions -verify -std=c++1y -Wunsupported-dll-base-class-template %s
      4 // RUN: %clang_cc1 -triple x86_64-mingw32 -fsyntax-only -fms-extensions -verify -std=c++11 -Wunsupported-dll-base-class-template %s
      5 
      6 // Helper structs to make templates more expressive.
      7 struct ImplicitInst_Exported {};
      8 struct ExplicitDecl_Exported {};
      9 struct ExplicitInst_Exported {};
     10 struct ExplicitSpec_Exported {};
     11 struct ExplicitSpec_Def_Exported {};
     12 struct ExplicitSpec_InlineDef_Exported {};
     13 struct ExplicitSpec_NotExported {};
     14 namespace { struct Internal {}; }
     15 struct External { int v; };
     16 
     17 
     18 // Invalid usage.
     19 __declspec(dllexport) typedef int typedef1; // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
     20 typedef __declspec(dllexport) int typedef2; // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
     21 typedef int __declspec(dllexport) typedef3; // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
     22 typedef __declspec(dllexport) void (*FunTy)(); // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
     23 enum __declspec(dllexport) Enum {}; // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
     24 #if __has_feature(cxx_strong_enums)
     25   enum class __declspec(dllexport) EnumClass {}; // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
     26 #endif
     27 
     28 
     29 
     30 //===----------------------------------------------------------------------===//
     31 // Globals
     32 //===----------------------------------------------------------------------===//
     33 
     34 // Export declaration.
     35 __declspec(dllexport) extern int ExternGlobalDecl;
     36 
     37 // dllexport implies a definition.
     38 __declspec(dllexport) int GlobalDef;
     39 
     40 // Export definition.
     41 __declspec(dllexport) int GlobalInit1 = 1;
     42 int __declspec(dllexport) GlobalInit2 = 1;
     43 
     44 // Declare, then export definition.
     45 __declspec(dllexport) extern int GlobalDeclInit;
     46 int GlobalDeclInit = 1;
     47 
     48 // Redeclarations
     49 __declspec(dllexport) extern int GlobalRedecl1;
     50 __declspec(dllexport)        int GlobalRedecl1;
     51 
     52 __declspec(dllexport) extern int GlobalRedecl2;
     53                              int GlobalRedecl2;
     54 
     55                       extern int GlobalRedecl3; // expected-note{{previous declaration is here}}
     56 __declspec(dllexport) extern int GlobalRedecl3; // expected-warning{{redeclaration of 'GlobalRedecl3' should not add 'dllexport' attribute}}
     57 
     58 extern "C" {
     59                       extern int GlobalRedecl4; // expected-note{{previous declaration is here}}
     60 __declspec(dllexport) extern int GlobalRedecl4; // expected-warning{{redeclaration of 'GlobalRedecl4' should not add 'dllexport' attribute}}
     61 }
     62 
     63 // External linkage is required.
     64 __declspec(dllexport) static int StaticGlobal; // expected-error{{'StaticGlobal' must have external linkage when declared 'dllexport'}}
     65 __declspec(dllexport) Internal InternalTypeGlobal; // expected-error{{'InternalTypeGlobal' must have external linkage when declared 'dllexport'}}
     66 namespace    { __declspec(dllexport) int InternalGlobal; } // expected-error{{'(anonymous namespace)::InternalGlobal' must have external linkage when declared 'dllexport'}}
     67 namespace ns { __declspec(dllexport) int ExternalGlobal; }
     68 
     69 __declspec(dllexport) auto InternalAutoTypeGlobal = Internal(); // expected-error{{'InternalAutoTypeGlobal' must have external linkage when declared 'dllexport'}}
     70 __declspec(dllexport) auto ExternalAutoTypeGlobal = External();
     71 
     72 // Thread local variables are invalid.
     73 __declspec(dllexport) __thread int ThreadLocalGlobal; // expected-error{{'ThreadLocalGlobal' cannot be thread local when declared 'dllexport'}}
     74 // But a static local TLS var in an export function is OK.
     75 inline void __declspec(dllexport) ExportedInlineWithThreadLocal() {
     76   static __thread int OK; // no-error
     77 }
     78 
     79 // Export in local scope.
     80 void functionScope() {
     81   __declspec(dllexport)        int LocalVarDecl; // expected-error{{'LocalVarDecl' must have external linkage when declared 'dllexport'}}
     82   __declspec(dllexport)        int LocalVarDef = 1; // expected-error{{'LocalVarDef' must have external linkage when declared 'dllexport'}}
     83   __declspec(dllexport) extern int ExternLocalVarDecl;
     84   __declspec(dllexport) static int StaticLocalVar; // expected-error{{'StaticLocalVar' must have external linkage when declared 'dllexport'}}
     85 }
     86 
     87 
     88 
     89 //===----------------------------------------------------------------------===//
     90 // Variable templates
     91 //===----------------------------------------------------------------------===//
     92 #if __has_feature(cxx_variable_templates)
     93 
     94 // Export declaration.
     95 template<typename T> __declspec(dllexport) extern int ExternVarTmplDecl;
     96 
     97 // dllexport implies a definition.
     98 template<typename T> __declspec(dllexport) int VarTmplDef;
     99 
    100 // Export definition.
    101 template<typename T> __declspec(dllexport) int VarTmplInit1 = 1;
    102 template<typename T> int __declspec(dllexport) VarTmplInit2 = 1;
    103 
    104 // Declare, then export definition.
    105 template<typename T> __declspec(dllexport) extern int VarTmplDeclInit;
    106 template<typename T>                              int VarTmplDeclInit = 1;
    107 
    108 // Redeclarations
    109 template<typename T> __declspec(dllexport) extern int VarTmplRedecl1;
    110 template<typename T> __declspec(dllexport)        int VarTmplRedecl1 = 1;
    111 
    112 template<typename T> __declspec(dllexport) extern int VarTmplRedecl2;
    113 template<typename T>                              int VarTmplRedecl2 = 1;
    114 
    115 template<typename T>                       extern int VarTmplRedecl3; // expected-note{{previous declaration is here}}
    116 template<typename T> __declspec(dllexport) extern int VarTmplRedecl3; // expected-error{{redeclaration of 'VarTmplRedecl3' cannot add 'dllexport' attribute}}
    117 
    118 // External linkage is required.
    119 template<typename T> __declspec(dllexport) static int StaticVarTmpl; // expected-error{{'StaticVarTmpl' must have external linkage when declared 'dllexport'}}
    120 template<typename T> __declspec(dllexport) Internal InternalTypeVarTmpl; // expected-error{{'InternalTypeVarTmpl' must have external linkage when declared 'dllexport'}}
    121 namespace    { template<typename T> __declspec(dllexport) int InternalVarTmpl; } // expected-error{{'(anonymous namespace)::InternalVarTmpl' must have external linkage when declared 'dllexport'}}
    122 namespace ns { template<typename T> __declspec(dllexport) int ExternalVarTmpl = 1; }
    123 
    124 template<typename T> __declspec(dllexport) auto InternalAutoTypeVarTmpl = Internal(); // expected-error{{'InternalAutoTypeVarTmpl' must have external linkage when declared 'dllexport'}}
    125 template<typename T> __declspec(dllexport) auto ExternalAutoTypeVarTmpl = External();
    126 template External ExternalAutoTypeVarTmpl<ExplicitInst_Exported>;
    127 
    128 
    129 template<typename T> int VarTmpl = 1;
    130 template<typename T> __declspec(dllexport) int ExportedVarTmpl = 1;
    131 
    132 // Export implicit instantiation of an exported variable template.
    133 int useVarTmpl() { return ExportedVarTmpl<ImplicitInst_Exported>; }
    134 
    135 // Export explicit instantiation declaration of an exported variable template.
    136 extern template int ExportedVarTmpl<ExplicitDecl_Exported>;
    137        template int ExportedVarTmpl<ExplicitDecl_Exported>;
    138 
    139 // Export explicit instantiation definition of an exported variable template.
    140 template __declspec(dllexport) int ExportedVarTmpl<ExplicitInst_Exported>;
    141 
    142 // Export specialization of an exported variable template.
    143 template<> __declspec(dllexport) int ExportedVarTmpl<ExplicitSpec_Exported>;
    144 template<> __declspec(dllexport) int ExportedVarTmpl<ExplicitSpec_Def_Exported> = 1;
    145 
    146 // Not exporting specialization of an exported variable template without
    147 // explicit dllexport.
    148 template<> int ExportedVarTmpl<ExplicitSpec_NotExported>;
    149 
    150 
    151 // Export explicit instantiation declaration of a non-exported variable template.
    152 extern template __declspec(dllexport) int VarTmpl<ExplicitDecl_Exported>;
    153        template __declspec(dllexport) int VarTmpl<ExplicitDecl_Exported>;
    154 
    155 // Export explicit instantiation definition of a non-exported variable template.
    156 template __declspec(dllexport) int VarTmpl<ExplicitInst_Exported>;
    157 
    158 // Export specialization of a non-exported variable template.
    159 template<> __declspec(dllexport) int VarTmpl<ExplicitSpec_Exported>;
    160 template<> __declspec(dllexport) int VarTmpl<ExplicitSpec_Def_Exported> = 1;
    161 
    162 #endif // __has_feature(cxx_variable_templates)
    163 
    164 
    165 
    166 //===----------------------------------------------------------------------===//
    167 // Functions
    168 //===----------------------------------------------------------------------===//
    169 
    170 // Export function declaration. Check different placements.
    171 __attribute__((dllexport)) void decl1A(); // Sanity check with __attribute__
    172 __declspec(dllexport)      void decl1B();
    173 
    174 void __attribute__((dllexport)) decl2A();
    175 void __declspec(dllexport)      decl2B();
    176 
    177 // Export function definition.
    178 __declspec(dllexport) void def() {}
    179 
    180 // extern "C"
    181 extern "C" __declspec(dllexport) void externC() {}
    182 
    183 // Export inline function.
    184 __declspec(dllexport) inline void inlineFunc1() {}
    185 inline void __attribute__((dllexport)) inlineFunc2() {}
    186 
    187 __declspec(dllexport) inline void inlineDecl();
    188                              void inlineDecl() {}
    189 
    190 __declspec(dllexport) void inlineDef();
    191                inline void inlineDef() {}
    192 
    193 // Redeclarations
    194 __declspec(dllexport) void redecl1();
    195 __declspec(dllexport) void redecl1() {}
    196 
    197 __declspec(dllexport) void redecl2();
    198                       void redecl2() {}
    199 
    200                       void redecl3(); // expected-note{{previous declaration is here}}
    201 __declspec(dllexport) void redecl3(); // expected-warning{{redeclaration of 'redecl3' should not add 'dllexport' attribute}}
    202 
    203 extern "C" {
    204                       void redecl4(); // expected-note{{previous declaration is here}}
    205 __declspec(dllexport) void redecl4(); // expected-warning{{redeclaration of 'redecl4' should not add 'dllexport' attribute}}
    206 }
    207 
    208                       void redecl5(); // expected-note{{previous declaration is here}}
    209 __declspec(dllexport) inline void redecl5() {} // expected-warning{{redeclaration of 'redecl5' should not add 'dllexport' attribute}}
    210 
    211 // Friend functions
    212 struct FuncFriend {
    213   friend __declspec(dllexport) void friend1();
    214   friend __declspec(dllexport) void friend2();
    215   friend                       void friend3(); // expected-note{{previous declaration is here}}
    216   friend                       void friend4(); // expected-note{{previous declaration is here}}
    217 };
    218 __declspec(dllexport) void friend1() {}
    219                       void friend2() {}
    220 __declspec(dllexport) void friend3() {} // expected-warning{{redeclaration of 'friend3' should not add 'dllexport' attribute}}
    221 __declspec(dllexport) inline void friend4() {} // expected-warning{{redeclaration of 'friend4' should not add 'dllexport' attribute}}
    222 
    223 // Implicit declarations can be redeclared with dllexport.
    224 __declspec(dllexport) void* operator new(__SIZE_TYPE__ n);
    225 
    226 // External linkage is required.
    227 __declspec(dllexport) static int staticFunc(); // expected-error{{'staticFunc' must have external linkage when declared 'dllexport'}}
    228 __declspec(dllexport) Internal internalRetFunc(); // expected-error{{'internalRetFunc' must have external linkage when declared 'dllexport'}}
    229 namespace    { __declspec(dllexport) void internalFunc() {} } // expected-error{{'(anonymous namespace)::internalFunc' must have external linkage when declared 'dllexport'}}
    230 namespace ns { __declspec(dllexport) void externalFunc() {} }
    231 
    232 // Export deleted function.
    233 __declspec(dllexport) void deletedFunc() = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    234 __declspec(dllexport) inline void deletedInlineFunc() = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    235 
    236 
    237 
    238 //===----------------------------------------------------------------------===//
    239 // Function templates
    240 //===----------------------------------------------------------------------===//
    241 
    242 // Export function template declaration. Check different placements.
    243 template<typename T> __declspec(dllexport) void funcTmplDecl1();
    244 template<typename T> void __declspec(dllexport) funcTmplDecl2();
    245 
    246 // Export function template definition.
    247 template<typename T> __declspec(dllexport) void funcTmplDef() {}
    248 
    249 // Export inline function template.
    250 template<typename T> __declspec(dllexport) inline void inlineFuncTmpl1() {}
    251 template<typename T> inline void __attribute__((dllexport)) inlineFuncTmpl2() {}
    252 
    253 template<typename T> __declspec(dllexport) inline void inlineFuncTmplDecl();
    254 template<typename T>                              void inlineFuncTmplDecl() {}
    255 
    256 template<typename T> __declspec(dllexport) void inlineFuncTmplDef();
    257 template<typename T>                inline void inlineFuncTmplDef() {}
    258 
    259 // Redeclarations
    260 template<typename T> __declspec(dllexport) void funcTmplRedecl1();
    261 template<typename T> __declspec(dllexport) void funcTmplRedecl1() {}
    262 
    263 template<typename T> __declspec(dllexport) void funcTmplRedecl2();
    264 template<typename T>                       void funcTmplRedecl2() {}
    265 
    266 template<typename T>                       void funcTmplRedecl3(); // expected-note{{previous declaration is here}}
    267 template<typename T> __declspec(dllexport) void funcTmplRedecl3(); // expected-error{{redeclaration of 'funcTmplRedecl3' cannot add 'dllexport' attribute}}
    268 
    269 template<typename T>                       void funcTmplRedecl4(); // expected-note{{previous declaration is here}}
    270 template<typename T> __declspec(dllexport) inline void funcTmplRedecl4() {} // expected-error{{redeclaration of 'funcTmplRedecl4' cannot add 'dllexport' attribute}}
    271 
    272 // Function template friends
    273 struct FuncTmplFriend {
    274   template<typename T> friend __declspec(dllexport) void funcTmplFriend1();
    275   template<typename T> friend __declspec(dllexport) void funcTmplFriend2();
    276   template<typename T> friend                       void funcTmplFriend3(); // expected-note{{previous declaration is here}}
    277   template<typename T> friend                       void funcTmplFriend4(); // expected-note{{previous declaration is here}}
    278 };
    279 template<typename T> __declspec(dllexport) void funcTmplFriend1() {}
    280 template<typename T>                       void funcTmplFriend2() {}
    281 template<typename T> __declspec(dllexport) void funcTmplFriend3() {} // expected-error{{redeclaration of 'funcTmplFriend3' cannot add 'dllexport' attribute}}
    282 template<typename T> __declspec(dllexport) inline void funcTmplFriend4() {} // expected-error{{redeclaration of 'funcTmplFriend4' cannot add 'dllexport' attribute}}
    283 
    284 // External linkage is required.
    285 template<typename T> __declspec(dllexport) static int staticFuncTmpl(); // expected-error{{'staticFuncTmpl' must have external linkage when declared 'dllexport'}}
    286 template<typename T> __declspec(dllexport) Internal internalRetFuncTmpl(); // expected-error{{'internalRetFuncTmpl' must have external linkage when declared 'dllexport'}}
    287 namespace    { template<typename T> __declspec(dllexport) void internalFuncTmpl(); } // expected-error{{'(anonymous namespace)::internalFuncTmpl' must have external linkage when declared 'dllexport'}}
    288 namespace ns { template<typename T> __declspec(dllexport) void externalFuncTmpl(); }
    289 
    290 
    291 template<typename T> void funcTmpl() {}
    292 template<typename T> __declspec(dllexport) void exportedFuncTmplDecl();
    293 template<typename T> __declspec(dllexport) void exportedFuncTmpl() {}
    294 
    295 // Export implicit instantiation of an exported function template.
    296 void useFunTmplDecl() { exportedFuncTmplDecl<ImplicitInst_Exported>(); }
    297 void useFunTmplDef() { exportedFuncTmpl<ImplicitInst_Exported>(); }
    298 
    299 // Export explicit instantiation declaration of an exported function template.
    300 extern template void exportedFuncTmpl<ExplicitDecl_Exported>();
    301        template void exportedFuncTmpl<ExplicitDecl_Exported>();
    302 
    303 // Export explicit instantiation definition of an exported function template.
    304 template void exportedFuncTmpl<ExplicitInst_Exported>();
    305 
    306 // Export specialization of an exported function template.
    307 template<> __declspec(dllexport) void exportedFuncTmpl<ExplicitSpec_Exported>();
    308 template<> __declspec(dllexport) void exportedFuncTmpl<ExplicitSpec_Def_Exported>() {}
    309 template<> __declspec(dllexport) inline void exportedFuncTmpl<ExplicitSpec_InlineDef_Exported>() {}
    310 
    311 // Not exporting specialization of an exported function template without
    312 // explicit dllexport.
    313 template<> void exportedFuncTmpl<ExplicitSpec_NotExported>() {}
    314 
    315 
    316 // Export explicit instantiation declaration of a non-exported function template.
    317 extern template __declspec(dllexport) void funcTmpl<ExplicitDecl_Exported>();
    318        template __declspec(dllexport) void funcTmpl<ExplicitDecl_Exported>();
    319 
    320 // Export explicit instantiation definition of a non-exported function template.
    321 template __declspec(dllexport) void funcTmpl<ExplicitInst_Exported>();
    322 
    323 // Export specialization of a non-exported function template.
    324 template<> __declspec(dllexport) void funcTmpl<ExplicitSpec_Exported>();
    325 template<> __declspec(dllexport) void funcTmpl<ExplicitSpec_Def_Exported>() {}
    326 template<> __declspec(dllexport) inline void funcTmpl<ExplicitSpec_InlineDef_Exported>() {}
    327 
    328 
    329 
    330 //===----------------------------------------------------------------------===//
    331 // Classes
    332 //===----------------------------------------------------------------------===//
    333 
    334 namespace {
    335   struct __declspec(dllexport) AnonymousClass {}; // expected-error{{(anonymous namespace)::AnonymousClass' must have external linkage when declared 'dllexport'}}
    336 }
    337 
    338 class __declspec(dllexport) ClassDecl;
    339 
    340 class __declspec(dllexport) ClassDef {};
    341 
    342 #ifdef MS
    343 // expected-warning@+3{{'dllexport' attribute ignored}}
    344 #endif
    345 template <typename T> struct PartiallySpecializedClassTemplate {};
    346 template <typename T> struct __declspec(dllexport) PartiallySpecializedClassTemplate<T*> { void f() {} };
    347 
    348 template <typename T> struct ExpliciallySpecializedClassTemplate {};
    349 template <> struct __declspec(dllexport) ExpliciallySpecializedClassTemplate<int> { void f() {} };
    350 
    351 // Don't instantiate class members of implicitly instantiated templates, even if they are exported.
    352 struct IncompleteType;
    353 template <typename T> struct __declspec(dllexport) ImplicitlyInstantiatedExportedTemplate {
    354   int f() { return sizeof(T); } // no-error
    355 };
    356 ImplicitlyInstantiatedExportedTemplate<IncompleteType> implicitlyInstantiatedExportedTemplate;
    357 
    358 // Don't instantiate class members of templates with explicit instantiation declarations, even if they are exported.
    359 struct IncompleteType2;
    360 template <typename T> struct __declspec(dllexport) ExportedTemplateWithExplicitInstantiationDecl { // expected-note{{attribute is here}}
    361   int f() { return sizeof(T); } // no-error
    362 };
    363 extern template struct ExportedTemplateWithExplicitInstantiationDecl<IncompleteType2>; // expected-warning{{explicit instantiation declaration should not be 'dllexport'}}
    364 
    365 // Instantiate class members for explicitly instantiated exported templates.
    366 struct IncompleteType3; // expected-note{{forward declaration of 'IncompleteType3'}}
    367 template <typename T> struct __declspec(dllexport) ExplicitlyInstantiatedExportedTemplate {
    368   int f() { return sizeof(T); } // expected-error{{invalid application of 'sizeof' to an incomplete type 'IncompleteType3'}}
    369 };
    370 template struct ExplicitlyInstantiatedExportedTemplate<IncompleteType3>; // expected-note{{in instantiation of member function 'ExplicitlyInstantiatedExportedTemplate<IncompleteType3>::f' requested here}}
    371 
    372 // In MS mode, instantiate members of class templates that are base classes of exported classes.
    373 #ifdef MS
    374   // expected-note@+3{{forward declaration of 'IncompleteType4'}}
    375   // expected-note@+3{{in instantiation of member function 'BaseClassTemplateOfExportedClass<IncompleteType4>::f' requested here}}
    376 #endif
    377 struct IncompleteType4;
    378 template <typename T> struct BaseClassTemplateOfExportedClass {
    379 #ifdef MS
    380   // expected-error@+2{{invalid application of 'sizeof' to an incomplete type 'IncompleteType4'}}
    381 #endif
    382   int f() { return sizeof(T); };
    383 };
    384 struct __declspec(dllexport) ExportedBaseClass : public BaseClassTemplateOfExportedClass<IncompleteType4> {};
    385 
    386 // Don't instantiate members of explicitly exported class templates that are base classes of exported classes.
    387 struct IncompleteType5;
    388 template <typename T> struct __declspec(dllexport) ExportedBaseClassTemplateOfExportedClass {
    389   int f() { return sizeof(T); }; // no-error
    390 };
    391 struct __declspec(dllexport) ExportedBaseClass2 : public ExportedBaseClassTemplateOfExportedClass<IncompleteType5> {};
    392 
    393 // Warn about explicit instantiation declarations of dllexport classes.
    394 template <typename T> struct ExplicitInstantiationDeclTemplate {};
    395 extern template struct __declspec(dllexport) ExplicitInstantiationDeclTemplate<int>; // expected-warning{{explicit instantiation declaration should not be 'dllexport'}} expected-note{{attribute is here}}
    396 
    397 template <typename T> struct __declspec(dllexport) ExplicitInstantiationDeclExportedTemplate {}; // expected-note{{attribute is here}}
    398 extern template struct ExplicitInstantiationDeclExportedTemplate<int>; // expected-warning{{explicit instantiation declaration should not be 'dllexport'}}
    399 
    400 namespace { struct InternalLinkageType {}; }
    401 struct __declspec(dllexport) PR23308 {
    402   void f(InternalLinkageType*);
    403 };
    404 void PR23308::f(InternalLinkageType*) {} // No error; we don't try to export f because it has internal linkage.
    405 
    406 //===----------------------------------------------------------------------===//
    407 // Classes with template base classes
    408 //===----------------------------------------------------------------------===//
    409 
    410 template <typename T> class ClassTemplate {};
    411 template <typename T> class __declspec(dllexport) ExportedClassTemplate {};
    412 template <typename T> class __declspec(dllimport) ImportedClassTemplate {};
    413 
    414 template <typename T> struct ExplicitlySpecializedTemplate { void func() {} };
    415 #ifdef MS
    416 // expected-note@+2{{class template 'ExplicitlySpecializedTemplate<int>' was explicitly specialized here}}
    417 #endif
    418 template <> struct ExplicitlySpecializedTemplate<int> { void func() {} };
    419 template <typename T> struct ExplicitlyExportSpecializedTemplate { void func() {} };
    420 template <> struct __declspec(dllexport) ExplicitlyExportSpecializedTemplate<int> { void func() {} };
    421 template <typename T> struct ExplicitlyImportSpecializedTemplate { void func() {} };
    422 template <> struct __declspec(dllimport) ExplicitlyImportSpecializedTemplate<int> { void func() {} };
    423 
    424 template <typename T> struct ExplicitlyInstantiatedTemplate { void func() {} };
    425 #ifdef MS
    426 // expected-note@+2{{class template 'ExplicitlyInstantiatedTemplate<int>' was instantiated here}}
    427 #endif
    428 template struct ExplicitlyInstantiatedTemplate<int>;
    429 template <typename T> struct ExplicitlyExportInstantiatedTemplate { void func() {} };
    430 template struct __declspec(dllexport) ExplicitlyExportInstantiatedTemplate<int>;
    431 template <typename T> struct ExplicitlyImportInstantiatedTemplate { void func() {} };
    432 template struct __declspec(dllimport) ExplicitlyImportInstantiatedTemplate<int>;
    433 
    434 // ClassTemplate<int> gets exported.
    435 class __declspec(dllexport) DerivedFromTemplate : public ClassTemplate<int> {};
    436 
    437 // ClassTemplate<int> is already exported.
    438 class __declspec(dllexport) DerivedFromTemplate2 : public ClassTemplate<int> {};
    439 
    440 // ExportedTemplate is explicitly exported.
    441 class __declspec(dllexport) DerivedFromExportedTemplate : public ExportedClassTemplate<int> {};
    442 
    443 // ImportedTemplate is explicitly imported.
    444 class __declspec(dllexport) DerivedFromImportedTemplate : public ImportedClassTemplate<int> {};
    445 
    446 class DerivedFromTemplateD : public ClassTemplate<double> {};
    447 // Base class previously implicitly instantiated without attribute; it will get propagated.
    448 class __declspec(dllexport) DerivedFromTemplateD2 : public ClassTemplate<double> {};
    449 
    450 // Base class has explicit instantiation declaration; the attribute will get propagated.
    451 extern template class ClassTemplate<float>;
    452 class __declspec(dllexport) DerivedFromTemplateF : public ClassTemplate<float> {};
    453 
    454 class __declspec(dllexport) DerivedFromTemplateB : public ClassTemplate<bool> {};
    455 // The second derived class doesn't change anything, the attribute that was propagated first wins.
    456 class __declspec(dllimport) DerivedFromTemplateB2 : public ClassTemplate<bool> {};
    457 
    458 #ifdef MS
    459 // expected-warning@+3{{propagating dll attribute to explicitly specialized base class template without dll attribute is not supported}}
    460 // expected-note@+2{{attribute is here}}
    461 #endif
    462 struct __declspec(dllexport) DerivedFromExplicitlySpecializedTemplate : public ExplicitlySpecializedTemplate<int> {};
    463 
    464 // Base class alredy specialized with export attribute.
    465 struct __declspec(dllexport) DerivedFromExplicitlyExportSpecializedTemplate : public ExplicitlyExportSpecializedTemplate<int> {};
    466 
    467 // Base class already specialized with import attribute.
    468 struct __declspec(dllexport) DerivedFromExplicitlyImportSpecializedTemplate : public ExplicitlyImportSpecializedTemplate<int> {};
    469 
    470 #ifdef MS
    471 // expected-warning@+3{{propagating dll attribute to already instantiated base class template without dll attribute is not supported}}
    472 // expected-note@+2{{attribute is here}}
    473 #endif
    474 struct __declspec(dllexport) DerivedFromExplicitlyInstantiatedTemplate : public ExplicitlyInstantiatedTemplate<int> {};
    475 
    476 // Base class already instantiated with export attribute.
    477 struct __declspec(dllexport) DerivedFromExplicitlyExportInstantiatedTemplate : public ExplicitlyExportInstantiatedTemplate<int> {};
    478 
    479 // Base class already instantiated with import attribute.
    480 struct __declspec(dllexport) DerivedFromExplicitlyImportInstantiatedTemplate : public ExplicitlyImportInstantiatedTemplate<int> {};
    481 
    482 template <typename T> struct ExplicitInstantiationDeclTemplateBase { void func() {} };
    483 extern template struct ExplicitInstantiationDeclTemplateBase<int>;
    484 struct __declspec(dllexport) DerivedFromExplicitInstantiationDeclTemplateBase : public ExplicitInstantiationDeclTemplateBase<int> {};
    485 
    486 
    487 //===----------------------------------------------------------------------===//
    488 // Precedence
    489 //===----------------------------------------------------------------------===//
    490 
    491 // dllexport takes precedence over dllimport if both are specified.
    492 __attribute__((dllimport, dllexport))       extern int PrecedenceExternGlobal1A; // expected-warning{{'dllimport' attribute ignored}}
    493 __declspec(dllimport) __declspec(dllexport) extern int PrecedenceExternGlobal1B; // expected-warning{{'dllimport' attribute ignored}}
    494 
    495 __attribute__((dllexport, dllimport))       extern int PrecedenceExternGlobal2A; // expected-warning{{'dllimport' attribute ignored}}
    496 __declspec(dllexport) __declspec(dllimport) extern int PrecedenceExternGlobal2B; // expected-warning{{'dllimport' attribute ignored}}
    497 
    498 __attribute__((dllimport, dllexport))       int PrecedenceGlobal1A; // expected-warning{{'dllimport' attribute ignored}}
    499 __declspec(dllimport) __declspec(dllexport) int PrecedenceGlobal1B; // expected-warning{{'dllimport' attribute ignored}}
    500 
    501 __attribute__((dllexport, dllimport))       int PrecedenceGlobal2A; // expected-warning{{'dllimport' attribute ignored}}
    502 __declspec(dllexport) __declspec(dllimport) int PrecedenceGlobal2B; // expected-warning{{'dllimport' attribute ignored}}
    503 
    504 __declspec(dllexport) extern int PrecedenceExternGlobalRedecl1;
    505 __declspec(dllimport) extern int PrecedenceExternGlobalRedecl1; // expected-warning{{'dllimport' attribute ignored}}
    506 
    507 __declspec(dllimport) extern int PrecedenceExternGlobalRedecl2; // expected-warning{{'dllimport' attribute ignored}}
    508 __declspec(dllexport) extern int PrecedenceExternGlobalRedecl2;
    509 
    510 __declspec(dllexport) extern int PrecedenceGlobalRedecl1;
    511 __declspec(dllimport)        int PrecedenceGlobalRedecl1; // expected-warning{{'dllimport' attribute ignored}}
    512 
    513 __declspec(dllimport) extern int PrecedenceGlobalRedecl2; // expected-warning{{'dllimport' attribute ignored}}
    514 __declspec(dllexport)        int PrecedenceGlobalRedecl2;
    515 
    516 void __attribute__((dllimport, dllexport))       precedence1A() {} // expected-warning{{'dllimport' attribute ignored}}
    517 void __declspec(dllimport) __declspec(dllexport) precedence1B() {} // expected-warning{{'dllimport' attribute ignored}}
    518 
    519 void __attribute__((dllexport, dllimport))       precedence2A() {} // expected-warning{{'dllimport' attribute ignored}}
    520 void __declspec(dllexport) __declspec(dllimport) precedence2B() {} // expected-warning{{'dllimport' attribute ignored}}
    521 
    522 void __declspec(dllimport) precedenceRedecl1(); // expected-warning{{'dllimport' attribute ignored}}
    523 void __declspec(dllexport) precedenceRedecl1() {}
    524 
    525 void __declspec(dllexport) precedenceRedecl2();
    526 void __declspec(dllimport) precedenceRedecl2() {} // expected-warning{{'dllimport' attribute ignored}}
    527 
    528 
    529 
    530 //===----------------------------------------------------------------------===//
    531 // Class members
    532 //===----------------------------------------------------------------------===//
    533 
    534 // Export individual members of a class.
    535 struct ExportMembers {
    536   struct Nested {
    537     __declspec(dllexport) void normalDef();
    538   };
    539 
    540   __declspec(dllexport)                void normalDecl();
    541   __declspec(dllexport)                void normalDef();
    542   __declspec(dllexport)                void normalInclass() {}
    543   __declspec(dllexport)                void normalInlineDef();
    544   __declspec(dllexport)         inline void normalInlineDecl();
    545   __declspec(dllexport) virtual        void virtualDecl();
    546   __declspec(dllexport) virtual        void virtualDef();
    547   __declspec(dllexport) virtual        void virtualInclass() {}
    548   __declspec(dllexport) virtual        void virtualInlineDef();
    549   __declspec(dllexport) virtual inline void virtualInlineDecl();
    550   __declspec(dllexport) static         void staticDecl();
    551   __declspec(dllexport) static         void staticDef();
    552   __declspec(dllexport) static         void staticInclass() {}
    553   __declspec(dllexport) static         void staticInlineDef();
    554   __declspec(dllexport) static  inline void staticInlineDecl();
    555 
    556 protected:
    557   __declspec(dllexport)                void protectedDef();
    558 private:
    559   __declspec(dllexport)                void privateDef();
    560 public:
    561 
    562   __declspec(dllexport)                int  Field; // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
    563   __declspec(dllexport) static         int  StaticField;
    564   __declspec(dllexport) static         int  StaticFieldDef;
    565   __declspec(dllexport) static  const  int  StaticConstField;
    566   __declspec(dllexport) static  const  int  StaticConstFieldDef;
    567   __declspec(dllexport) static  const  int  StaticConstFieldEqualInit = 1;
    568   __declspec(dllexport) static  const  int  StaticConstFieldBraceInit{1};
    569   __declspec(dllexport) constexpr static int ConstexprField = 1;
    570   __declspec(dllexport) constexpr static int ConstexprFieldDef = 1;
    571 };
    572 
    573        void ExportMembers::Nested::normalDef() {}
    574        void ExportMembers::normalDef() {}
    575 inline void ExportMembers::normalInlineDef() {}
    576        void ExportMembers::normalInlineDecl() {}
    577        void ExportMembers::virtualDef() {}
    578 inline void ExportMembers::virtualInlineDef() {}
    579        void ExportMembers::virtualInlineDecl() {}
    580        void ExportMembers::staticDef() {}
    581 inline void ExportMembers::staticInlineDef() {}
    582        void ExportMembers::staticInlineDecl() {}
    583        void ExportMembers::protectedDef() {}
    584        void ExportMembers::privateDef() {}
    585 
    586        int  ExportMembers::StaticFieldDef;
    587 const  int  ExportMembers::StaticConstFieldDef = 1;
    588 constexpr int ExportMembers::ConstexprFieldDef;
    589 
    590 
    591 // Export on member definitions.
    592 struct ExportMemberDefs {
    593   __declspec(dllexport)                void normalDef();
    594   __declspec(dllexport)                void normalInlineDef();
    595   __declspec(dllexport)         inline void normalInlineDecl();
    596   __declspec(dllexport) virtual        void virtualDef();
    597   __declspec(dllexport) virtual        void virtualInlineDef();
    598   __declspec(dllexport) virtual inline void virtualInlineDecl();
    599   __declspec(dllexport) static         void staticDef();
    600   __declspec(dllexport) static         void staticInlineDef();
    601   __declspec(dllexport) static  inline void staticInlineDecl();
    602 
    603   __declspec(dllexport) static         int  StaticField;
    604   __declspec(dllexport) static  const  int  StaticConstField;
    605   __declspec(dllexport) constexpr static int ConstexprField = 1;
    606 };
    607 
    608 __declspec(dllexport)        void ExportMemberDefs::normalDef() {}
    609 __declspec(dllexport) inline void ExportMemberDefs::normalInlineDef() {}
    610 __declspec(dllexport)        void ExportMemberDefs::normalInlineDecl() {}
    611 __declspec(dllexport)        void ExportMemberDefs::virtualDef() {}
    612 __declspec(dllexport) inline void ExportMemberDefs::virtualInlineDef() {}
    613 __declspec(dllexport)        void ExportMemberDefs::virtualInlineDecl() {}
    614 __declspec(dllexport)        void ExportMemberDefs::staticDef() {}
    615 __declspec(dllexport) inline void ExportMemberDefs::staticInlineDef() {}
    616 __declspec(dllexport)        void ExportMemberDefs::staticInlineDecl() {}
    617 
    618 __declspec(dllexport)        int  ExportMemberDefs::StaticField;
    619 __declspec(dllexport) const  int  ExportMemberDefs::StaticConstField = 1;
    620 __declspec(dllexport) constexpr int ExportMemberDefs::ConstexprField;
    621 
    622 
    623 // Export special member functions.
    624 struct ExportSpecials {
    625   __declspec(dllexport) ExportSpecials() {}
    626   __declspec(dllexport) ~ExportSpecials();
    627   __declspec(dllexport) inline ExportSpecials(const ExportSpecials&);
    628   __declspec(dllexport) ExportSpecials& operator=(const ExportSpecials&);
    629   __declspec(dllexport) ExportSpecials(ExportSpecials&&);
    630   __declspec(dllexport) ExportSpecials& operator=(ExportSpecials&&);
    631 };
    632 
    633 ExportSpecials::~ExportSpecials() {}
    634 ExportSpecials::ExportSpecials(const ExportSpecials&) {}
    635 inline ExportSpecials& ExportSpecials::operator=(const ExportSpecials&) { return *this; }
    636 ExportSpecials::ExportSpecials(ExportSpecials&&) {}
    637 ExportSpecials& ExportSpecials::operator=(ExportSpecials&&) { return *this; }
    638 
    639 
    640 // Export allocation functions.
    641 extern "C" void* malloc(__SIZE_TYPE__ size);
    642 extern "C" void free(void* p);
    643 struct ExportAlloc {
    644   __declspec(dllexport) void* operator new(__SIZE_TYPE__);
    645   __declspec(dllexport) void* operator new[](__SIZE_TYPE__);
    646   __declspec(dllexport) void operator delete(void*);
    647   __declspec(dllexport) void operator delete[](void*);
    648 };
    649 void* ExportAlloc::operator new(__SIZE_TYPE__ n) { return malloc(n); }
    650 void* ExportAlloc::operator new[](__SIZE_TYPE__ n) { return malloc(n); }
    651 void ExportAlloc::operator delete(void* p) { free(p); }
    652 void ExportAlloc::operator delete[](void* p) { free(p); }
    653 
    654 
    655 // Export deleted member functions.
    656 struct ExportDeleted {
    657   __declspec(dllexport) ExportDeleted() = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    658   __declspec(dllexport) ~ExportDeleted() = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    659   __declspec(dllexport) ExportDeleted(const ExportDeleted&) = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    660   __declspec(dllexport) ExportDeleted& operator=(const ExportDeleted&) = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    661   __declspec(dllexport) ExportDeleted(ExportDeleted&&) = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    662   __declspec(dllexport) ExportDeleted& operator=(ExportDeleted&&) = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    663   __declspec(dllexport) void deleted() = delete; // expected-error{{attribute 'dllexport' cannot be applied to a deleted function}}
    664 };
    665 
    666 
    667 // Export defaulted member functions.
    668 struct ExportDefaulted {
    669   __declspec(dllexport) ExportDefaulted() = default;
    670   __declspec(dllexport) ~ExportDefaulted() = default;
    671   __declspec(dllexport) ExportDefaulted(const ExportDefaulted&) = default;
    672   __declspec(dllexport) ExportDefaulted& operator=(const ExportDefaulted&) = default;
    673   __declspec(dllexport) ExportDefaulted(ExportDefaulted&&) = default;
    674   __declspec(dllexport) ExportDefaulted& operator=(ExportDefaulted&&) = default;
    675 };
    676 
    677 
    678 // Export defaulted member function definitions.
    679 struct ExportDefaultedDefs {
    680   __declspec(dllexport) ExportDefaultedDefs();
    681   __declspec(dllexport) ~ExportDefaultedDefs();
    682 
    683   __declspec(dllexport) inline ExportDefaultedDefs(const ExportDefaultedDefs&);
    684   __declspec(dllexport) ExportDefaultedDefs& operator=(const ExportDefaultedDefs&);
    685 
    686   __declspec(dllexport) ExportDefaultedDefs(ExportDefaultedDefs&&);
    687   __declspec(dllexport) ExportDefaultedDefs& operator=(ExportDefaultedDefs&&);
    688 };
    689 
    690 // Export definitions.
    691 __declspec(dllexport) ExportDefaultedDefs::ExportDefaultedDefs() = default;
    692 ExportDefaultedDefs::~ExportDefaultedDefs() = default;
    693 
    694 // Export inline declaration and definition.
    695 __declspec(dllexport) ExportDefaultedDefs::ExportDefaultedDefs(const ExportDefaultedDefs&) = default;
    696 inline ExportDefaultedDefs& ExportDefaultedDefs::operator=(const ExportDefaultedDefs&) = default;
    697 
    698 __declspec(dllexport) ExportDefaultedDefs::ExportDefaultedDefs(ExportDefaultedDefs&&) = default;
    699 ExportDefaultedDefs& ExportDefaultedDefs::operator=(ExportDefaultedDefs&&) = default;
    700 
    701 
    702 // Redeclarations cannot add dllexport.
    703 struct MemberRedecl {
    704                  void normalDef();         // expected-note{{previous declaration is here}}
    705                  void normalInlineDef();   // expected-note{{previous declaration is here}}
    706           inline void normalInlineDecl();  // expected-note{{previous declaration is here}}
    707   virtual        void virtualDef();        // expected-note{{previous declaration is here}}
    708   virtual        void virtualInlineDef();  // expected-note{{previous declaration is here}}
    709   virtual inline void virtualInlineDecl(); // expected-note{{previous declaration is here}}
    710   static         void staticDef();         // expected-note{{previous declaration is here}}
    711   static         void staticInlineDef();   // expected-note{{previous declaration is here}}
    712   static  inline void staticInlineDecl();  // expected-note{{previous declaration is here}}
    713 
    714   static         int  StaticField;         // expected-note{{previous declaration is here}}
    715   static  const  int  StaticConstField;    // expected-note{{previous declaration is here}}
    716   constexpr static int ConstexprField = 1; // expected-note{{previous declaration is here}}
    717 };
    718 
    719 __declspec(dllexport)        void MemberRedecl::normalDef() {}         // expected-error{{redeclaration of 'MemberRedecl::normalDef' cannot add 'dllexport' attribute}}
    720 __declspec(dllexport) inline void MemberRedecl::normalInlineDef() {}   // expected-error{{redeclaration of 'MemberRedecl::normalInlineDef' cannot add 'dllexport' attribute}}
    721 __declspec(dllexport)        void MemberRedecl::normalInlineDecl() {}  // expected-error{{redeclaration of 'MemberRedecl::normalInlineDecl' cannot add 'dllexport' attribute}}
    722 __declspec(dllexport)        void MemberRedecl::virtualDef() {}        // expected-error{{redeclaration of 'MemberRedecl::virtualDef' cannot add 'dllexport' attribute}}
    723 __declspec(dllexport) inline void MemberRedecl::virtualInlineDef() {}  // expected-error{{redeclaration of 'MemberRedecl::virtualInlineDef' cannot add 'dllexport' attribute}}
    724 __declspec(dllexport)        void MemberRedecl::virtualInlineDecl() {} // expected-error{{redeclaration of 'MemberRedecl::virtualInlineDecl' cannot add 'dllexport' attribute}}
    725 __declspec(dllexport)        void MemberRedecl::staticDef() {}         // expected-error{{redeclaration of 'MemberRedecl::staticDef' cannot add 'dllexport' attribute}}
    726 __declspec(dllexport) inline void MemberRedecl::staticInlineDef() {}   // expected-error{{redeclaration of 'MemberRedecl::staticInlineDef' cannot add 'dllexport' attribute}}
    727 __declspec(dllexport)        void MemberRedecl::staticInlineDecl() {}  // expected-error{{redeclaration of 'MemberRedecl::staticInlineDecl' cannot add 'dllexport' attribute}}
    728 
    729 __declspec(dllexport)        int  MemberRedecl::StaticField = 1;       // expected-error{{redeclaration of 'MemberRedecl::StaticField' cannot add 'dllexport' attribute}}
    730 __declspec(dllexport) const  int  MemberRedecl::StaticConstField = 1;  // expected-error{{redeclaration of 'MemberRedecl::StaticConstField' cannot add 'dllexport' attribute}}
    731 __declspec(dllexport) constexpr int MemberRedecl::ConstexprField;      // expected-error{{redeclaration of 'MemberRedecl::ConstexprField' cannot add 'dllexport' attribute}}
    732 
    733 
    734 
    735 //===----------------------------------------------------------------------===//
    736 // Class member templates
    737 //===----------------------------------------------------------------------===//
    738 
    739 struct ExportMemberTmpl {
    740   template<typename T> __declspec(dllexport)               void normalDecl();
    741   template<typename T> __declspec(dllexport)               void normalDef();
    742   template<typename T> __declspec(dllexport)               void normalInclass() {}
    743   template<typename T> __declspec(dllexport)               void normalInlineDef();
    744   template<typename T> __declspec(dllexport)        inline void normalInlineDecl();
    745   template<typename T> __declspec(dllexport) static        void staticDecl();
    746   template<typename T> __declspec(dllexport) static        void staticDef();
    747   template<typename T> __declspec(dllexport) static        void staticInclass() {}
    748   template<typename T> __declspec(dllexport) static        void staticInlineDef();
    749   template<typename T> __declspec(dllexport) static inline void staticInlineDecl();
    750 
    751 #if __has_feature(cxx_variable_templates)
    752   template<typename T> __declspec(dllexport) static        int  StaticField;
    753   template<typename T> __declspec(dllexport) static        int  StaticFieldDef;
    754   template<typename T> __declspec(dllexport) static const  int  StaticConstField;
    755   template<typename T> __declspec(dllexport) static const  int  StaticConstFieldDef;
    756   template<typename T> __declspec(dllexport) static const  int  StaticConstFieldEqualInit = 1;
    757   template<typename T> __declspec(dllexport) static const  int  StaticConstFieldBraceInit{1};
    758   template<typename T> __declspec(dllexport) constexpr static int ConstexprField = 1;
    759   template<typename T> __declspec(dllexport) constexpr static int ConstexprFieldDef = 1;
    760 #endif // __has_feature(cxx_variable_templates)
    761 };
    762 
    763 template<typename T>        void ExportMemberTmpl::normalDef() {}
    764 template<typename T> inline void ExportMemberTmpl::normalInlineDef() {}
    765 template<typename T>        void ExportMemberTmpl::normalInlineDecl() {}
    766 template<typename T>        void ExportMemberTmpl::staticDef() {}
    767 template<typename T> inline void ExportMemberTmpl::staticInlineDef() {}
    768 template<typename T>        void ExportMemberTmpl::staticInlineDecl() {}
    769 
    770 #if __has_feature(cxx_variable_templates)
    771 template<typename T>        int  ExportMemberTmpl::StaticFieldDef;
    772 template<typename T> const  int  ExportMemberTmpl::StaticConstFieldDef = 1;
    773 template<typename T> constexpr int ExportMemberTmpl::ConstexprFieldDef;
    774 #endif // __has_feature(cxx_variable_templates)
    775 
    776 
    777 // Redeclarations cannot add dllexport.
    778 struct MemTmplRedecl {
    779   template<typename T>               void normalDef();         // expected-note{{previous declaration is here}}
    780   template<typename T>               void normalInlineDef();   // expected-note{{previous declaration is here}}
    781   template<typename T>        inline void normalInlineDecl();  // expected-note{{previous declaration is here}}
    782   template<typename T> static        void staticDef();         // expected-note{{previous declaration is here}}
    783   template<typename T> static        void staticInlineDef();   // expected-note{{previous declaration is here}}
    784   template<typename T> static inline void staticInlineDecl();  // expected-note{{previous declaration is here}}
    785 
    786 #if __has_feature(cxx_variable_templates)
    787   template<typename T> static        int  StaticField;         // expected-note{{previous declaration is here}}
    788   template<typename T> static const  int  StaticConstField;    // expected-note{{previous declaration is here}}
    789   template<typename T> constexpr static int ConstexprField = 1; // expected-note{{previous declaration is here}}
    790 #endif // __has_feature(cxx_variable_templates)
    791 };
    792 
    793 template<typename T> __declspec(dllexport)        void MemTmplRedecl::normalDef() {}        // expected-error{{redeclaration of 'MemTmplRedecl::normalDef' cannot add 'dllexport' attribute}}
    794 template<typename T> __declspec(dllexport) inline void MemTmplRedecl::normalInlineDef() {}  // expected-error{{redeclaration of 'MemTmplRedecl::normalInlineDef' cannot add 'dllexport' attribute}}
    795 template<typename T> __declspec(dllexport)        void MemTmplRedecl::normalInlineDecl() {} // expected-error{{redeclaration of 'MemTmplRedecl::normalInlineDecl' cannot add 'dllexport' attribute}}
    796 template<typename T> __declspec(dllexport)        void MemTmplRedecl::staticDef() {}        // expected-error{{redeclaration of 'MemTmplRedecl::staticDef' cannot add 'dllexport' attribute}}
    797 template<typename T> __declspec(dllexport) inline void MemTmplRedecl::staticInlineDef() {}  // expected-error{{redeclaration of 'MemTmplRedecl::staticInlineDef' cannot add 'dllexport' attribute}}
    798 template<typename T> __declspec(dllexport)        void MemTmplRedecl::staticInlineDecl() {} // expected-error{{redeclaration of 'MemTmplRedecl::staticInlineDecl' cannot add 'dllexport' attribute}}
    799 
    800 #if __has_feature(cxx_variable_templates)
    801 template<typename T> __declspec(dllexport)        int  MemTmplRedecl::StaticField = 1;      // expected-error{{redeclaration of 'MemTmplRedecl::StaticField' cannot add 'dllexport' attribute}}
    802 template<typename T> __declspec(dllexport) const  int  MemTmplRedecl::StaticConstField = 1; // expected-error{{redeclaration of 'MemTmplRedecl::StaticConstField' cannot add 'dllexport' attribute}}
    803 template<typename T> __declspec(dllexport) constexpr int MemTmplRedecl::ConstexprField;     // expected-error{{redeclaration of 'MemTmplRedecl::ConstexprField' cannot add 'dllexport' attribute}}
    804 #endif // __has_feature(cxx_variable_templates)
    805 
    806 
    807 
    808 struct MemFunTmpl {
    809   template<typename T>                              void normalDef() {}
    810   template<typename T> __declspec(dllexport)        void exportedNormal() {}
    811   template<typename T>                       static void staticDef() {}
    812   template<typename T> __declspec(dllexport) static void exportedStatic() {}
    813 };
    814 
    815 // Export implicit instantiation of an exported member function template.
    816 void useMemFunTmpl() {
    817   MemFunTmpl().exportedNormal<ImplicitInst_Exported>();
    818   MemFunTmpl().exportedStatic<ImplicitInst_Exported>();
    819 }
    820 
    821 // Export explicit instantiation declaration of an exported member function
    822 // template.
    823 extern template void MemFunTmpl::exportedNormal<ExplicitDecl_Exported>();
    824        template void MemFunTmpl::exportedNormal<ExplicitDecl_Exported>();
    825 
    826 extern template void MemFunTmpl::exportedStatic<ExplicitDecl_Exported>();
    827        template void MemFunTmpl::exportedStatic<ExplicitDecl_Exported>();
    828 
    829 // Export explicit instantiation definition of an exported member function
    830 // template.
    831 template void MemFunTmpl::exportedNormal<ExplicitInst_Exported>();
    832 template void MemFunTmpl::exportedStatic<ExplicitInst_Exported>();
    833 
    834 // Export specialization of an exported member function template.
    835 template<> __declspec(dllexport) void MemFunTmpl::exportedNormal<ExplicitSpec_Exported>();
    836 template<> __declspec(dllexport) void MemFunTmpl::exportedNormal<ExplicitSpec_Def_Exported>() {}
    837 template<> __declspec(dllexport) inline void MemFunTmpl::exportedNormal<ExplicitSpec_InlineDef_Exported>() {}
    838 
    839 template<> __declspec(dllexport) void MemFunTmpl::exportedStatic<ExplicitSpec_Exported>();
    840 template<> __declspec(dllexport) void MemFunTmpl::exportedStatic<ExplicitSpec_Def_Exported>() {}
    841 template<> __declspec(dllexport) inline void MemFunTmpl::exportedStatic<ExplicitSpec_InlineDef_Exported>() {}
    842 
    843 // Not exporting specialization of an exported member function template without
    844 // explicit dllexport.
    845 template<> void MemFunTmpl::exportedNormal<ExplicitSpec_NotExported>() {}
    846 template<> void MemFunTmpl::exportedStatic<ExplicitSpec_NotExported>() {}
    847 
    848 
    849 // Export explicit instantiation declaration of a non-exported member function
    850 // template.
    851 extern template __declspec(dllexport) void MemFunTmpl::normalDef<ExplicitDecl_Exported>();
    852        template __declspec(dllexport) void MemFunTmpl::normalDef<ExplicitDecl_Exported>();
    853 
    854 extern template __declspec(dllexport) void MemFunTmpl::staticDef<ExplicitDecl_Exported>();
    855        template __declspec(dllexport) void MemFunTmpl::staticDef<ExplicitDecl_Exported>();
    856 
    857 // Export explicit instantiation definition of a non-exported member function
    858 // template.
    859 template __declspec(dllexport) void MemFunTmpl::normalDef<ExplicitInst_Exported>();
    860 template __declspec(dllexport) void MemFunTmpl::staticDef<ExplicitInst_Exported>();
    861 
    862 // Export specialization of a non-exported member function template.
    863 template<> __declspec(dllexport) void MemFunTmpl::normalDef<ExplicitSpec_Exported>();
    864 template<> __declspec(dllexport) void MemFunTmpl::normalDef<ExplicitSpec_Def_Exported>() {}
    865 template<> __declspec(dllexport) inline void MemFunTmpl::normalDef<ExplicitSpec_InlineDef_Exported>() {}
    866 
    867 template<> __declspec(dllexport) void MemFunTmpl::staticDef<ExplicitSpec_Exported>();
    868 template<> __declspec(dllexport) void MemFunTmpl::staticDef<ExplicitSpec_Def_Exported>() {}
    869 template<> __declspec(dllexport) inline void MemFunTmpl::staticDef<ExplicitSpec_InlineDef_Exported>() {}
    870 
    871 
    872 
    873 #if __has_feature(cxx_variable_templates)
    874 struct MemVarTmpl {
    875   template<typename T>                       static const int StaticVar = 1;
    876   template<typename T> __declspec(dllexport) static const int ExportedStaticVar = 1;
    877 };
    878 template<typename T> const int MemVarTmpl::StaticVar;
    879 template<typename T> const int MemVarTmpl::ExportedStaticVar;
    880 
    881 // Export implicit instantiation of an exported member variable template.
    882 int useMemVarTmpl() { return MemVarTmpl::ExportedStaticVar<ImplicitInst_Exported>; }
    883 
    884 // Export explicit instantiation declaration of an exported member variable
    885 // template.
    886 extern template const int MemVarTmpl::ExportedStaticVar<ExplicitDecl_Exported>;
    887        template const int MemVarTmpl::ExportedStaticVar<ExplicitDecl_Exported>;
    888 
    889 // Export explicit instantiation definition of an exported member variable
    890 // template.
    891 template const int MemVarTmpl::ExportedStaticVar<ExplicitInst_Exported>;
    892 
    893 // Export specialization of an exported member variable template.
    894 template<> __declspec(dllexport) const int MemVarTmpl::ExportedStaticVar<ExplicitSpec_Exported>;
    895 template<> __declspec(dllexport) const int MemVarTmpl::ExportedStaticVar<ExplicitSpec_Def_Exported> = 1;
    896 
    897 // Not exporting specialization of an exported member variable template without
    898 // explicit dllexport.
    899 template<> const int MemVarTmpl::ExportedStaticVar<ExplicitSpec_NotExported>;
    900 
    901 
    902 // Export explicit instantiation declaration of a non-exported member variable
    903 // template.
    904 extern template __declspec(dllexport) const int MemVarTmpl::StaticVar<ExplicitDecl_Exported>;
    905        template __declspec(dllexport) const int MemVarTmpl::StaticVar<ExplicitDecl_Exported>;
    906 
    907 // Export explicit instantiation definition of a non-exported member variable
    908 // template.
    909 template __declspec(dllexport) const int MemVarTmpl::StaticVar<ExplicitInst_Exported>;
    910 
    911 // Export specialization of a non-exported member variable template.
    912 template<> __declspec(dllexport) const int MemVarTmpl::StaticVar<ExplicitSpec_Exported>;
    913 template<> __declspec(dllexport) const int MemVarTmpl::StaticVar<ExplicitSpec_Def_Exported> = 1;
    914 
    915 #endif // __has_feature(cxx_variable_templates)
    916 
    917 
    918 
    919 //===----------------------------------------------------------------------===//
    920 // Class template members
    921 //===----------------------------------------------------------------------===//
    922 
    923 // Export individual members of a class template.
    924 template<typename T>
    925 struct ExportClassTmplMembers {
    926   __declspec(dllexport)                void normalDecl();
    927   __declspec(dllexport)                void normalDef();
    928   __declspec(dllexport)                void normalInclass() {}
    929   __declspec(dllexport)                void normalInlineDef();
    930   __declspec(dllexport)         inline void normalInlineDecl();
    931   __declspec(dllexport) virtual        void virtualDecl();
    932   __declspec(dllexport) virtual        void virtualDef();
    933   __declspec(dllexport) virtual        void virtualInclass() {}
    934   __declspec(dllexport) virtual        void virtualInlineDef();
    935   __declspec(dllexport) virtual inline void virtualInlineDecl();
    936   __declspec(dllexport) static         void staticDecl();
    937   __declspec(dllexport) static         void staticDef();
    938   __declspec(dllexport) static         void staticInclass() {}
    939   __declspec(dllexport) static         void staticInlineDef();
    940   __declspec(dllexport) static  inline void staticInlineDecl();
    941 
    942 protected:
    943   __declspec(dllexport)                void protectedDef();
    944 private:
    945   __declspec(dllexport)                void privateDef();
    946 public:
    947 
    948   __declspec(dllexport)                int  Field; // expected-warning{{'dllexport' attribute only applies to variables, functions and classes}}
    949   __declspec(dllexport) static         int  StaticField;
    950   __declspec(dllexport) static         int  StaticFieldDef;
    951   __declspec(dllexport) static  const  int  StaticConstField;
    952   __declspec(dllexport) static  const  int  StaticConstFieldDef;
    953   __declspec(dllexport) static  const  int  StaticConstFieldEqualInit = 1;
    954   __declspec(dllexport) static  const  int  StaticConstFieldBraceInit{1};
    955   __declspec(dllexport) constexpr static int ConstexprField = 1;
    956   __declspec(dllexport) constexpr static int ConstexprFieldDef = 1;
    957 };
    958 
    959 template<typename T>        void ExportClassTmplMembers<T>::normalDef() {}
    960 template<typename T> inline void ExportClassTmplMembers<T>::normalInlineDef() {}
    961 template<typename T>        void ExportClassTmplMembers<T>::normalInlineDecl() {}
    962 template<typename T>        void ExportClassTmplMembers<T>::virtualDef() {}
    963 template<typename T> inline void ExportClassTmplMembers<T>::virtualInlineDef() {}
    964 template<typename T>        void ExportClassTmplMembers<T>::virtualInlineDecl() {}
    965 template<typename T>        void ExportClassTmplMembers<T>::staticDef() {}
    966 template<typename T> inline void ExportClassTmplMembers<T>::staticInlineDef() {}
    967 template<typename T>        void ExportClassTmplMembers<T>::staticInlineDecl() {}
    968 template<typename T>        void ExportClassTmplMembers<T>::protectedDef() {}
    969 template<typename T>        void ExportClassTmplMembers<T>::privateDef() {}
    970 
    971 template<typename T>        int  ExportClassTmplMembers<T>::StaticFieldDef;
    972 template<typename T> const  int  ExportClassTmplMembers<T>::StaticConstFieldDef = 1;
    973 template<typename T> constexpr int ExportClassTmplMembers<T>::ConstexprFieldDef;
    974 
    975 template struct ExportClassTmplMembers<ImplicitInst_Exported>;
    976 
    977 
    978 // Redeclarations cannot add dllexport.
    979 template<typename T>
    980 struct CTMR /*ClassTmplMemberRedecl*/ {
    981                  void normalDef();         // expected-note{{previous declaration is here}}
    982                  void normalInlineDef();   // expected-note{{previous declaration is here}}
    983           inline void normalInlineDecl();  // expected-note{{previous declaration is here}}
    984   virtual        void virtualDef();        // expected-note{{previous declaration is here}}
    985   virtual        void virtualInlineDef();  // expected-note{{previous declaration is here}}
    986   virtual inline void virtualInlineDecl(); // expected-note{{previous declaration is here}}
    987   static         void staticDef();         // expected-note{{previous declaration is here}}
    988   static         void staticInlineDef();   // expected-note{{previous declaration is here}}
    989   static  inline void staticInlineDecl();  // expected-note{{previous declaration is here}}
    990 
    991   static         int  StaticField;         // expected-note{{previous declaration is here}}
    992   static  const  int  StaticConstField;    // expected-note{{previous declaration is here}}
    993   constexpr static int ConstexprField = 1; // expected-note{{previous declaration is here}}
    994 };
    995 
    996 template<typename T> __declspec(dllexport)        void CTMR<T>::normalDef() {}         // expected-error{{redeclaration of 'CTMR::normalDef' cannot add 'dllexport' attribute}}
    997 template<typename T> __declspec(dllexport) inline void CTMR<T>::normalInlineDef() {}   // expected-error{{redeclaration of 'CTMR::normalInlineDef' cannot add 'dllexport' attribute}}
    998 template<typename T> __declspec(dllexport)        void CTMR<T>::normalInlineDecl() {}  // expected-error{{redeclaration of 'CTMR::normalInlineDecl' cannot add 'dllexport' attribute}}
    999 template<typename T> __declspec(dllexport)        void CTMR<T>::virtualDef() {}        // expected-error{{redeclaration of 'CTMR::virtualDef' cannot add 'dllexport' attribute}}
   1000 template<typename T> __declspec(dllexport) inline void CTMR<T>::virtualInlineDef() {}  // expected-error{{redeclaration of 'CTMR::virtualInlineDef' cannot add 'dllexport' attribute}}
   1001 template<typename T> __declspec(dllexport)        void CTMR<T>::virtualInlineDecl() {} // expected-error{{redeclaration of 'CTMR::virtualInlineDecl' cannot add 'dllexport' attribute}}
   1002 template<typename T> __declspec(dllexport)        void CTMR<T>::staticDef() {}         // expected-error{{redeclaration of 'CTMR::staticDef' cannot add 'dllexport' attribute}}
   1003 template<typename T> __declspec(dllexport) inline void CTMR<T>::staticInlineDef() {}   // expected-error{{redeclaration of 'CTMR::staticInlineDef' cannot add 'dllexport' attribute}}
   1004 template<typename T> __declspec(dllexport)        void CTMR<T>::staticInlineDecl() {}  // expected-error{{redeclaration of 'CTMR::staticInlineDecl' cannot add 'dllexport' attribute}}
   1005 
   1006 template<typename T> __declspec(dllexport)        int  CTMR<T>::StaticField = 1;       // expected-error{{redeclaration of 'CTMR::StaticField' cannot add 'dllexport' attribute}}
   1007 template<typename T> __declspec(dllexport) const  int  CTMR<T>::StaticConstField = 1;  // expected-error{{redeclaration of 'CTMR::StaticConstField' cannot add 'dllexport' attribute}}
   1008 template<typename T> __declspec(dllexport) constexpr int CTMR<T>::ConstexprField;      // expected-error{{redeclaration of 'CTMR::ConstexprField' cannot add 'dllexport' attribute}}
   1009 
   1010 
   1011 
   1012 //===----------------------------------------------------------------------===//
   1013 // Class template member templates
   1014 //===----------------------------------------------------------------------===//
   1015 
   1016 template<typename T>
   1017 struct ExportClsTmplMemTmpl {
   1018   template<typename U> __declspec(dllexport)               void normalDecl();
   1019   template<typename U> __declspec(dllexport)               void normalDef();
   1020   template<typename U> __declspec(dllexport)               void normalInclass() {}
   1021   template<typename U> __declspec(dllexport)               void normalInlineDef();
   1022   template<typename U> __declspec(dllexport)        inline void normalInlineDecl();
   1023   template<typename U> __declspec(dllexport) static        void staticDecl();
   1024   template<typename U> __declspec(dllexport) static        void staticDef();
   1025   template<typename U> __declspec(dllexport) static        void staticInclass() {}
   1026   template<typename U> __declspec(dllexport) static        void staticInlineDef();
   1027   template<typename U> __declspec(dllexport) static inline void staticInlineDecl();
   1028 
   1029 #if __has_feature(cxx_variable_templates)
   1030   template<typename U> __declspec(dllexport) static        int  StaticField;
   1031   template<typename U> __declspec(dllexport) static        int  StaticFieldDef;
   1032   template<typename U> __declspec(dllexport) static const  int  StaticConstField;
   1033   template<typename U> __declspec(dllexport) static const  int  StaticConstFieldDef;
   1034   template<typename U> __declspec(dllexport) static const  int  StaticConstFieldEqualInit = 1;
   1035   template<typename U> __declspec(dllexport) static const  int  StaticConstFieldBraceInit{1};
   1036   template<typename U> __declspec(dllexport) constexpr static int ConstexprField = 1;
   1037   template<typename U> __declspec(dllexport) constexpr static int ConstexprFieldDef = 1;
   1038 #endif // __has_feature(cxx_variable_templates)
   1039 };
   1040 
   1041 template<typename T> template<typename U>        void ExportClsTmplMemTmpl<T>::normalDef() {}
   1042 template<typename T> template<typename U> inline void ExportClsTmplMemTmpl<T>::normalInlineDef() {}
   1043 template<typename T> template<typename U>        void ExportClsTmplMemTmpl<T>::normalInlineDecl() {}
   1044 template<typename T> template<typename U>        void ExportClsTmplMemTmpl<T>::staticDef() {}
   1045 template<typename T> template<typename U> inline void ExportClsTmplMemTmpl<T>::staticInlineDef() {}
   1046 template<typename T> template<typename U>        void ExportClsTmplMemTmpl<T>::staticInlineDecl() {}
   1047 
   1048 #if __has_feature(cxx_variable_templates)
   1049 template<typename T> template<typename U>        int  ExportClsTmplMemTmpl<T>::StaticFieldDef;
   1050 template<typename T> template<typename U> const  int  ExportClsTmplMemTmpl<T>::StaticConstFieldDef = 1;
   1051 template<typename T> template<typename U> constexpr int ExportClsTmplMemTmpl<T>::ConstexprFieldDef;
   1052 #endif // __has_feature(cxx_variable_templates)
   1053 
   1054 
   1055 // Redeclarations cannot add dllexport.
   1056 template<typename T>
   1057 struct CTMTR /*ClassTmplMemberTmplRedecl*/ {
   1058   template<typename U>               void normalDef();         // expected-note{{previous declaration is here}}
   1059   template<typename U>               void normalInlineDef();   // expected-note{{previous declaration is here}}
   1060   template<typename U>        inline void normalInlineDecl();  // expected-note{{previous declaration is here}}
   1061   template<typename U> static        void staticDef();         // expected-note{{previous declaration is here}}
   1062   template<typename U> static        void staticInlineDef();   // expected-note{{previous declaration is here}}
   1063   template<typename U> static inline void staticInlineDecl();  // expected-note{{previous declaration is here}}
   1064 
   1065 #if __has_feature(cxx_variable_templates)
   1066   template<typename U> static        int  StaticField;         // expected-note{{previous declaration is here}}
   1067   template<typename U> static const  int  StaticConstField;    // expected-note{{previous declaration is here}}
   1068   template<typename U> constexpr static int ConstexprField = 1; // expected-note{{previous declaration is here}}
   1069 #endif // __has_feature(cxx_variable_templates)
   1070 };
   1071 
   1072 template<typename T> template<typename U> __declspec(dllexport)        void CTMTR<T>::normalDef() {}         // expected-error{{redeclaration of 'CTMTR::normalDef' cannot add 'dllexport' attribute}}
   1073 template<typename T> template<typename U> __declspec(dllexport) inline void CTMTR<T>::normalInlineDef() {}   // expected-error{{redeclaration of 'CTMTR::normalInlineDef' cannot add 'dllexport' attribute}}
   1074 template<typename T> template<typename U> __declspec(dllexport)        void CTMTR<T>::normalInlineDecl() {}  // expected-error{{redeclaration of 'CTMTR::normalInlineDecl' cannot add 'dllexport' attribute}}
   1075 template<typename T> template<typename U> __declspec(dllexport)        void CTMTR<T>::staticDef() {}         // expected-error{{redeclaration of 'CTMTR::staticDef' cannot add 'dllexport' attribute}}
   1076 template<typename T> template<typename U> __declspec(dllexport) inline void CTMTR<T>::staticInlineDef() {}   // expected-error{{redeclaration of 'CTMTR::staticInlineDef' cannot add 'dllexport' attribute}}
   1077 template<typename T> template<typename U> __declspec(dllexport)        void CTMTR<T>::staticInlineDecl() {}  // expected-error{{redeclaration of 'CTMTR::staticInlineDecl' cannot add 'dllexport' attribute}}
   1078 
   1079 #if __has_feature(cxx_variable_templates)
   1080 template<typename T> template<typename U> __declspec(dllexport)        int  CTMTR<T>::StaticField = 1;       // expected-error{{redeclaration of 'CTMTR::StaticField' cannot add 'dllexport' attribute}}
   1081 template<typename T> template<typename U> __declspec(dllexport) const  int  CTMTR<T>::StaticConstField = 1;  // expected-error{{redeclaration of 'CTMTR::StaticConstField' cannot add 'dllexport' attribute}}
   1082 template<typename T> template<typename U> __declspec(dllexport) constexpr int CTMTR<T>::ConstexprField;      // expected-error{{redeclaration of 'CTMTR::ConstexprField' cannot add 'dllexport' attribute}}
   1083 #endif // __has_feature(cxx_variable_templates)
   1084 
   1085 // FIXME: Precedence rules seem to be different for classes.
   1086 
   1087 //===----------------------------------------------------------------------===//
   1088 // Lambdas
   1089 //===----------------------------------------------------------------------===//
   1090 // The MS ABI doesn't provide a stable mangling for lambdas, so they can't be imported or exported.
   1091 #ifdef MS
   1092 // expected-error@+2{{lambda cannot be declared 'dllexport'}}
   1093 #endif
   1094 auto Lambda = []() __declspec(dllexport) -> bool { return true; };
   1095