80 lines
3.4 KiB
C++
80 lines
3.4 KiB
C++
|
// RUN: %clang_cc1 -std=c++20 -verify %s
|
||
|
|
||
|
// Templates and partial and explicit specializations can't have C linkage.
|
||
|
namespace extern_c_templates {
|
||
|
|
||
|
template<typename T> struct A {
|
||
|
static int a;
|
||
|
struct b;
|
||
|
void c();
|
||
|
enum class d;
|
||
|
|
||
|
template<typename U> static int e;
|
||
|
template<typename U> struct f;
|
||
|
template<typename U> void g();
|
||
|
};
|
||
|
|
||
|
template<typename T> int B;
|
||
|
template<typename T> void C();
|
||
|
|
||
|
extern "C" { // expected-note 1+{{begins here}}
|
||
|
// templates
|
||
|
template<typename T> struct A; // expected-error {{templates must have C++ linkage}}
|
||
|
template<typename T> int B; // expected-error {{templates must have C++ linkage}}
|
||
|
template<typename T> void C(); // expected-error {{templates must have C++ linkage}}
|
||
|
|
||
|
// non-template members of a template
|
||
|
// FIXME: Should these really be valid?
|
||
|
template<typename T> int A<T>::a;
|
||
|
template<typename T> struct A<T>::b {};
|
||
|
template<typename T> void A<T>::c() {}
|
||
|
template<typename T> enum class A<T>::d {};
|
||
|
|
||
|
// templates
|
||
|
template<typename T> template<typename U> int A<T>::e; // expected-error {{templates must have C++ linkage}}
|
||
|
template<typename T> template<typename U> struct A<T>::f {}; // expected-error {{templates must have C++ linkage}}
|
||
|
template<typename T> template<typename U> void A<T>::g() {} // expected-error {{templates must have C++ linkage}}
|
||
|
|
||
|
// partial specializations
|
||
|
template<typename T> struct A<int*>; // expected-error {{templates must have C++ linkage}}
|
||
|
template<typename T> int B<int*>; // expected-error {{templates must have C++ linkage}}
|
||
|
template<typename T> template<typename U> int A<T>::e<U*>; // expected-error {{templates must have C++ linkage}}
|
||
|
template<typename T> template<typename U> struct A<T>::f<U*> {}; // expected-error {{templates must have C++ linkage}}
|
||
|
|
||
|
// explicit specializations of templates
|
||
|
template<> struct A<char> {}; // expected-error {{templates must have C++ linkage}}
|
||
|
template<> int B<char>; // expected-error {{templates must have C++ linkage}}
|
||
|
template<> void C<char>() {} // expected-error {{templates must have C++ linkage}}
|
||
|
|
||
|
// explicit specializations of members of a template
|
||
|
template<> int A<int>::a; // expected-error {{templates must have C++ linkage}}
|
||
|
template<> struct A<int>::b {}; // expected-error {{templates must have C++ linkage}}
|
||
|
template<> void A<int>::c() {} // expected-error {{templates must have C++ linkage}}
|
||
|
template<> enum class A<int>::d {}; // expected-error {{templates must have C++ linkage}}
|
||
|
|
||
|
// explicit specializations of member templates
|
||
|
template<> template<typename U> int A<int>::e; // expected-error {{templates must have C++ linkage}}
|
||
|
template<> template<typename U> struct A<int>::f {}; // expected-error {{templates must have C++ linkage}}
|
||
|
template<> template<typename U> void A<int>::g() {} // expected-error {{templates must have C++ linkage}}
|
||
|
}
|
||
|
|
||
|
// Provide valid definitions for the explicit instantiations below.
|
||
|
// FIXME: Our recovery from the invalid definitions above isn't very good.
|
||
|
template<typename T> template<typename U> int A<T>::e;
|
||
|
template<typename T> template<typename U> struct A<T>::f {};
|
||
|
template<typename T> template<typename U> void A<T>::g() {}
|
||
|
|
||
|
extern "C" {
|
||
|
// explicit instantiations
|
||
|
// FIXME: Should these really be valid?
|
||
|
template struct A<double>;
|
||
|
template int A<float>::a;
|
||
|
template struct A<float>::b;
|
||
|
template void A<float>::c();
|
||
|
template int A<float>::e<float>;
|
||
|
template struct A<float>::f<float>;
|
||
|
template void A<float>::g<float>();
|
||
|
}
|
||
|
|
||
|
}
|