/// For a class that has a vtable and typeinfo symbol for RTTI, if a user marks
/// either:
///
/// (a) The entire class as dllexport (dllimport)
/// (b) Any non-inline method of the class as dllexport (dllimport)
///
/// then Clang must export the vtable and typeinfo symbol from the TU where they
/// are defined (the TU containing the definition of the Itanium C++ ABI "key
/// function") and must import them in other modules where they are referenced.
// RUN: %clang_cc1 -I%S -fdeclspec -triple x86_64-unknown-windows-itanium -emit-llvm -o - %s -fhalf-no-semantic-interposition \
// RUN: | FileCheck %s -check-prefix=WI
// RUN: %clang_cc1 -I%S -fdeclspec -triple x86_64-scei-windows-itanium -emit-llvm -o - %s -fhalf-no-semantic-interposition \
// RUN: | FileCheck %s --check-prefixes=PS
// RUN: %clang_cc1 -I%S -fdeclspec -triple x86_64-scei-ps4 -emit-llvm -o - %s -fhalf-no-semantic-interposition \
// RUN: | FileCheck %s --check-prefixes=PS
// RUN: %clang_cc1 -I%S -fdeclspec -triple x86_64-sie-ps5 -emit-llvm -o - %s -fhalf-no-semantic-interposition \
// RUN: | FileCheck %s --check-prefixes=PS
#include <typeinfo>
/// Case (a) -- Import Aspect
/// The entire class is imported. The typeinfo symbol must also be imported, but
/// the vtable will not be referenced, and so does not need to be imported.
// PS-DAG: @_ZTI10FullImport = {{.*}}dllimport
// WI-DAG: @_ZTI10FullImport = external dllimport constant ptr
struct __declspec(dllimport) FullImport {
virtual void inlineFunc() const {}
virtual void key();
virtual void func();
};
/// 'FullImport::key()' is the key function, so the vtable and typeinfo symbol
/// of 'FullImport' will be defined in the TU that contains the definition of
/// 'key()' (and they must be exported from there).
void FullImportTest() { typeid(FullImport).name(); }
/// Case (a) -- Export Aspect
/// The entire class is exported. The vtable and typeinfo symbols must also be
/// exported.
// PS-DAG: @_ZTV10FullExport = {{.*}}dllexport
// WI-DAG: @_ZTV10FullExport = {{.*}}dllexport
// PS-DAG: @_ZTI10FullExport = {{.*}}dllexport
// WI-DAG: @_ZTI10FullExport = dso_local dllexport constant {
struct __declspec(dllexport) FullExport {
virtual void inlineFunc() const {}
virtual void key();
virtual void func();
};
/// This is the key function of the class 'FullExport', so the vtable and
/// typeinfo symbols of 'FullExport' will be defined in this TU, and so they
/// must be exported from this TU.
void FullExport::key() { typeid(FullExport).name(); }
/// Case (b) -- Import Aspect
/// The class as a whole is not imported, but a non-inline method of the class
/// is, so the vtable and typeinfo symbol must be imported.
// PS-DAG: @_ZTV10PartImport = {{.*}}dllimport
// WI-DAG: @_ZTV10PartImport = external dso_local unnamed_addr constant {
// PS-DAG: @_ZTI10PartImport = {{.*}}dllimport
// WI-DAG: @_ZTI10PartImport = external dso_local constant ptr
struct PartImport {
virtual void inlineFunc() const {}
virtual void key();
__declspec(dllimport) virtual void func();
};
/// 'PartImport::key()' is the key function, so the vtable and typeinfo symbol
/// of 'PartImport' will be defined in the TU that contains the definition of
/// 'key()' (and they must be exported from there). Here, we will reference the
/// vtable and typeinfo symbol, so we must also import them.
void PartImportTest() {
PartImport f;
typeid(PartImport).name();
}
/// Case (b) -- Export Aspect
/// The class as a whole is not exported, but a non-inline method of the class
/// is, so the vtable and typeinfo symbol must be exported.
// PS-DAG: @_ZTV10PartExport = {{.*}}dllexport
// WI-DAG: @_ZTV10PartExport = dso_local unnamed_addr constant {
// PS-DAG: @_ZTI10PartExport = {{.*}}dllexport
// WI-DAG: @_ZTI10PartExport = dso_local constant {
struct PartExport {
virtual void inlineFunc() const {}
virtual void key();
__declspec(dllexport) virtual void func();
};
/// This is the key function of the class 'PartExport', so the vtable and
/// typeinfo symbol of 'PartExport' will be defined in this TU, and so they must
/// be exported from this TU.
void PartExport::key() { typeid(PartExport).name(); }
/// Case (b) -- Export Aspect
/// The class as a whole is not exported, but the constructor of the class
/// is, so the vtable and typeinfo symbol must be exported.
// PS-DAG: @_ZTV10ConsExport = {{.*}}dllexport
// WI-DAG: @_ZTV10ConsExport = dso_local unnamed_addr constant {
// PS-DAG: @_ZTI10ConsExport = {{.*}}dllexport
// WI-DAG: @_ZTI10ConsExport = dso_local constant {
struct ConsExport {
__declspec(dllexport) ConsExport();
virtual void key();
};
ConsExport::ConsExport() {}
void ConsExport::key() { typeid(ConsExport).name(); }