2019-04-26 16:09:51 +08:00
|
|
|
// RUN: %clang_cc1 -emit-llvm -triple i686-mingw32 %s -o - | FileCheck %s
|
|
|
|
|
[MinGW] Don't let template instantiation declarations cover nested classes
An explicit template instantiation declaration used to let
callers assume both outer and nested classes instantiations were
defined in a different translation unit.
If the instantiation is marked dllexport, only the outer class
is exported, but the caller will try to reference the instantiation
of both outer and inner classes.
This makes MinGW mode match both MSVC and Windows Itanium, by
having instantations only cover the outer class, and locally emitting
definitions of the nested classes. Windows Itanium was changed to
use this behavious in SVN r300804.
This deviates from what GCC does, but should be safe (and only
inflate the object file size a bit, but MSVC and Windows Itanium
modes do the same), and fixes cases where inner classes aren't
dllexported.
This fixes missing references in combination with dllexported/imported
template intantiations.
GCC suffers from the same issue, reported at [1], but the issue
is still unresolved there. The issue can probably be solved either
by making dllexport cover all nested classes as well, or this
way (matching MSVC).
[1] https://gcc.gnu.org/bugzilla/show_bug.cgi?id=89087
Differential Revision: https://reviews.llvm.org/D61175
llvm-svn: 359342
2019-04-27 03:31:39 +08:00
|
|
|
#define JOIN2(x, y) x##y
|
|
|
|
#define JOIN(x, y) JOIN2(x, y)
|
|
|
|
#define UNIQ(name) JOIN(name, __LINE__)
|
|
|
|
#define USEMEMFUNC(class, func) void (class::*UNIQ(use)())() { return &class::func; }
|
|
|
|
|
2019-04-26 16:09:51 +08:00
|
|
|
template <class T>
|
|
|
|
class c {
|
2019-04-27 03:31:46 +08:00
|
|
|
void f() {}
|
2019-04-26 16:09:51 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template class __declspec(dllexport) c<int>;
|
|
|
|
|
|
|
|
// CHECK: define {{.*}} dllexport {{.*}} @_ZN1cIiE1fEv
|
|
|
|
|
|
|
|
extern template class __declspec(dllexport) c<char>;
|
|
|
|
template class c<char>;
|
|
|
|
|
|
|
|
// CHECK: define {{.*}} dllexport {{.*}} @_ZN1cIcE1fEv
|
|
|
|
|
|
|
|
extern template class c<double>;
|
|
|
|
template class __declspec(dllexport) c<double>;
|
|
|
|
|
|
|
|
// CHECK-NOT: define {{.*}} dllexport {{.*}} @_ZN1cIdE1fEv
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
struct outer {
|
|
|
|
void f();
|
|
|
|
struct inner {
|
|
|
|
void f();
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class T> void outer<T>::f() {}
|
|
|
|
template <class T> void outer<T>::inner::f() {}
|
|
|
|
|
|
|
|
template class __declspec(dllexport) outer<int>;
|
|
|
|
|
|
|
|
// CHECK: define {{.*}} dllexport {{.*}} @_ZN5outerIiE1fEv
|
|
|
|
// CHECK-NOT: define {{.*}} dllexport {{.*}} @_ZN5outerIiE5inner1fEv
|
[MinGW] Don't let template instantiation declarations cover nested classes
An explicit template instantiation declaration used to let
callers assume both outer and nested classes instantiations were
defined in a different translation unit.
If the instantiation is marked dllexport, only the outer class
is exported, but the caller will try to reference the instantiation
of both outer and inner classes.
This makes MinGW mode match both MSVC and Windows Itanium, by
having instantations only cover the outer class, and locally emitting
definitions of the nested classes. Windows Itanium was changed to
use this behavious in SVN r300804.
This deviates from what GCC does, but should be safe (and only
inflate the object file size a bit, but MSVC and Windows Itanium
modes do the same), and fixes cases where inner classes aren't
dllexported.
This fixes missing references in combination with dllexported/imported
template intantiations.
GCC suffers from the same issue, reported at [1], but the issue
is still unresolved there. The issue can probably be solved either
by making dllexport cover all nested classes as well, or this
way (matching MSVC).
[1] https://gcc.gnu.org/bugzilla/show_bug.cgi?id=89087
Differential Revision: https://reviews.llvm.org/D61175
llvm-svn: 359342
2019-04-27 03:31:39 +08:00
|
|
|
|
|
|
|
extern template class __declspec(dllimport) outer<char>;
|
|
|
|
USEMEMFUNC(outer<char>, f)
|
|
|
|
USEMEMFUNC(outer<char>::inner, f)
|
|
|
|
|
|
|
|
// CHECK: declare dllimport {{.*}} @_ZN5outerIcE1fEv
|
|
|
|
// CHECK: define {{.*}} @_ZN5outerIcE5inner1fEv
|