/// 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 /// 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(); }