From 6957108e50b9d99cfcce151f34b8725f6cc8fe13 Mon Sep 17 00:00:00 2001 From: Evan Ramos Date: Wed, 3 Jul 2024 13:12:17 -0500 Subject: [PATCH] Add Jason's tests to repository Co-authored-by: Jason Cohen --- tests/.gitignore | 1 + tests/Attributes.cpp | 204 +++++++++++ tests/CMakeLists.txt | 138 +++++++ tests/CMakeSettings.json | 40 ++ tests/Calls.cpp | 413 +++++++++++++++++++++ tests/Coverage.cpp | 8 + tests/CoverageC.c | 73 ++++ tests/CoverageCounter.c | 30 ++ tests/CoverageCuda.cu | 8 + tests/CoverageMem.c | 26 ++ tests/CoverageMemCudaRt.cu | 25 ++ tests/CoveragePayload.c | 27 ++ tests/DllHelper.h | 38 ++ tests/Domains.cpp | 97 +++++ tests/InjectionHelper.h | 569 +++++++++++++++++++++++++++++ tests/LinkerDupesFileA.cpp | 6 + tests/LinkerDupesFileB.cpp | 6 + tests/LinkerDupesMain.cpp | 13 + tests/NamedCategories.cpp | 128 +++++++ tests/PrettyPrintersNvtxC.h | 140 +++++++ tests/PrettyPrintersNvtxCpp.h | 20 + tests/PrintInjection.cpp | 198 ++++++++++ tests/README.txt | 37 ++ tests/RegisteredStrings.cpp | 80 ++++ tests/RunTest.cpp | 189 ++++++++++ tests/Same.h | 166 +++++++++ tests/SelfInjection.cpp | 178 +++++++++ tests/SelfInjection.h | 668 ++++++++++++++++++++++++++++++++++ tests/TestCoverage.h | 478 ++++++++++++++++++++++++ tests/TestSelfInjection.cpp | 251 +++++++++++++ 30 files changed, 4255 insertions(+) create mode 100644 tests/.gitignore create mode 100644 tests/Attributes.cpp create mode 100644 tests/CMakeLists.txt create mode 100644 tests/CMakeSettings.json create mode 100644 tests/Calls.cpp create mode 100644 tests/Coverage.cpp create mode 100644 tests/CoverageC.c create mode 100644 tests/CoverageCounter.c create mode 100644 tests/CoverageCuda.cu create mode 100644 tests/CoverageMem.c create mode 100644 tests/CoverageMemCudaRt.cu create mode 100644 tests/CoveragePayload.c create mode 100644 tests/DllHelper.h create mode 100644 tests/Domains.cpp create mode 100644 tests/InjectionHelper.h create mode 100644 tests/LinkerDupesFileA.cpp create mode 100644 tests/LinkerDupesFileB.cpp create mode 100644 tests/LinkerDupesMain.cpp create mode 100644 tests/NamedCategories.cpp create mode 100644 tests/PrettyPrintersNvtxC.h create mode 100644 tests/PrettyPrintersNvtxCpp.h create mode 100644 tests/PrintInjection.cpp create mode 100644 tests/README.txt create mode 100644 tests/RegisteredStrings.cpp create mode 100644 tests/RunTest.cpp create mode 100644 tests/Same.h create mode 100644 tests/SelfInjection.cpp create mode 100644 tests/SelfInjection.h create mode 100644 tests/TestCoverage.h create mode 100644 tests/TestSelfInjection.cpp diff --git a/tests/.gitignore b/tests/.gitignore new file mode 100644 index 0000000..796b96d --- /dev/null +++ b/tests/.gitignore @@ -0,0 +1 @@ +/build diff --git a/tests/Attributes.cpp b/tests/Attributes.cpp new file mode 100644 index 0000000..6ca86b0 --- /dev/null +++ b/tests/Attributes.cpp @@ -0,0 +1,204 @@ +#include +// Include again to catch bad guards +#include + +#include +#include + +#include "PrettyPrintersNvtxCpp.h" + +struct a_lib +{ + static constexpr const char* name{"Library A"}; +}; + +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + { + std::cout << "Default attributes:\n"; + nvtx3::event_attributes attr; + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a payload:\n"; + nvtx3::event_attributes attr{nvtx3::payload{5.0f}}; + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a color with RGB hex code 0xFF7F00:\n"; + nvtx3::event_attributes attr{nvtx3::color{0xFFFF7F00}}; + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + + { + std::cout << "Set a color with RGB=255,127,0:\n"; + nvtx3::event_attributes attr{nvtx3::rgb{255,127,0}}; + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + + { + std::cout << "Set a color & payload:\n"; + nvtx3::event_attributes attr{nvtx3::rgb{255,127,0}, nvtx3::payload{5.0f}}; + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a color (red), payload, color again (green)... first color wins:\n"; + + nvtx3::event_attributes attr{ + nvtx3::rgb{255,0,0}, + nvtx3::payload{5.0f}, + nvtx3::rgb{0, 255, 0}}; + + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a message (ascii), payload, color, and category:\n"; + + nvtx3::event_attributes attr{ + nvtx3::message{"Hello"}, + nvtx3::category{11}, + nvtx3::payload{5.0f}, + nvtx3::rgb{0,255,0}}; + + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a message with different string types:\n"; + + nvtx3::event_attributes a{nvtx3::message{"Hello"}}; + std::cout << a; + + nvtx3::event_attributes wa{nvtx3::message{L"Hello"}}; + std::cout << wa; + + std::string hello{"Hello"}; + nvtx3::event_attributes b{nvtx3::message{hello}}; + std::cout << b; + + std::wstring whello{L"Hello"}; + nvtx3::event_attributes wb{nvtx3::message{whello}}; + std::cout << wb; + + // Important! Neither of following will compile: + // + // nvtx3::event_attributes c{nvtx3::message{std::string{"foo"}}}; + // std::cout << c; + // + // std::string foo{"foo"}; + // nvtx3::event_attributes d{nvtx3::message{hello + "bar"}}; + // std::cout << d; + // + // Both of those usages fail with: + // "error C2280: 'nvtx3::message::message(std::string &&)': + // attempting to reference a deleted function" + // + // nvtx3::message is a "view" class, not an owning class. + // It cannot take ownership of a temporary string and + // destroy it when it goes out of scope. Similarly, + // nvtx3::event_attributes is not an owning class, so it cannot take + // ownership of an nvtx3::message either. + // + // TODO: Could we add implicit support for this? + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a message (registered):\n"; + auto hTacobell = reinterpret_cast(0x7ac0be11); + nvtx3::event_attributes attr{nvtx3::message{hTacobell}}; + std::cout << attr; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Set category/message/payload/color, with \"using\":\n"; + + using namespace nvtx3; + + event_attributes a{ + category{11}, + message{"Hello"}, + payload{5.0f}, + rgb{1,2,3}}; + + std::cout << a; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Convenience: Set a message without the helper type:\n"; + + nvtx3::event_attributes a{"Hello"}; + std::cout << a; + + std::string hello{"Hello"}; + nvtx3::event_attributes b{hello}; + std::cout << b; + } + std::cout << "-------------------------------------\n"; + + { + std::cout << "Examples: \"using\", skip helper type for msg, set other fields:\n"; + + using namespace nvtx3; + + event_attributes a{"Hello", payload{7.0}}; + std::cout << a; + + event_attributes b{"Hello", rgb{255,255,0}}; + std::cout << b; + + event_attributes c{"Hello", category{4}}; + std::cout << c; + + // Order doesn't matter + event_attributes d{"Hello", rgb{255,255,0}, payload{7.0}, category{4}}; + std::cout << d; + + event_attributes e{payload{7.0}, "Hello", category{4}, rgb{255,255,0}}; + std::cout << e; + + event_attributes f{category{4}, rgb{255,255,0}, payload{7.0}, "Hello"}; + std::cout << f; + + // Vertical formatting is nice too: + event_attributes g{ + "Hello", + category{4}, + rgb{255,255,0}, + payload{7.0}}; + std::cout << g; + + event_attributes h + { + "Hello", + category{4}, + rgb{255,255,0}, + payload{7.0} + }; + std::cout << h; + } + std::cout << "-------------------------------------\n"; + + return 0; +} diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt new file mode 100644 index 0000000..f79ab09 --- /dev/null +++ b/tests/CMakeLists.txt @@ -0,0 +1,138 @@ +cmake_minimum_required (VERSION 3.19) + +project ("NvtxTests" LANGUAGES C CXX CUDA) + +# Enforce standard C/C++ with sensible warnings and minimal compiler output on all platforms +set(CMAKE_C_STANDARD 90) +set(CMAKE_C_EXTENSIONS OFF) +set(CMAKE_CXX_STANDARD 11) +set(CMAKE_CXX_EXTENSIONS OFF) +if(${MSVC}) + # Must use - instead of / for option prefix when using NVCC, because it forwards args + # it doesn't know to the host compiler, but being a UNIX-heritage program, it thinks + # an argument like "/nologo" is an input file path. Luckily, MSVC accepts - prefixes. + if(${CMAKE_C_COMPILER_VERSION} VERSION_LESS "19.14.0.0") + # Enable options to behave closer to standard + else() + add_compile_options(-permissive-) + endif() + # The following line can be uncommented to test with WIN32_LEAN_AND_MEAN + #add_compile_definitions(WIN32_LEAN_AND_MEAN) +endif() + +# Build with minimal or no dependencies on installed C/C++ runtime libraries +if (${MSVC}) + # For Non-debug, change /MD (MultiThreadedDLL) to /MT (MultiThreaded) + # For Debug, change /MDd (MultiThreadedDebugDLL) to /MTd ((MultiThreadedDebug) + set(CMAKE_MSVC_RUNTIME_LIBRARY "MultiThreaded$<$:Debug>") +else() + # Statically link libstdc++ and libgcc. Do not statically link libc, though. + # Use an old sysroot if compatibility with old GLIBC versions is required. + # In non-DEBUG builds, use -s to strip unneeded symbols + add_link_options( + -static-libstdc++ + -static-libgcc + $<$>:-s> + ) +endif() + +# Compiler-specific idiosyncracies +if(${MSVC}) + # Must use - instead of / for option prefix when using NVCC, because it forwards args + # it doesn't know to the host compiler, but being a UNIX-heritage program, it thinks + # an argument like "/nologo" is an input file path. Luckily, MSVC accepts - prefixes. + add_compile_options(-nologo) + #add_compile_options(-wd26812) # Disable warning: prefer enum class over unscoped enum + add_link_options(-NOLOGO -INCREMENTAL:NO) + # On some platforms, CMake doesn't automatically add C++ flags to enable RTTI (/GR) or + # configure C++ exceptions to the commonly preferred value (/EHsc or /GX). Add these + # if they are missing. + if(NOT CMAKE_CXX_FLAGS MATCHES "(/|-)GR( |$)") + string(APPEND CMAKE_CXX_FLAGS " -GR") + endif() + if(NOT CMAKE_CXX_FLAGS MATCHES "(/|-)(EHsc|GX)( |$)") + string(APPEND CMAKE_CXX_FLAGS " -EHsc") + endif() + # Improve debugging + if (${CMAKE_BUILD_TYPE} STREQUAL "Debug") + # This for some reason also adds "MDd" even though above we asked for MTd, + # so add the /JMC option manually + #set(CMAKE_VS_JUST_MY_CODE_DEBUGGING ON) + add_compile_options(-JMC) + endif() +else() + # Stop compiling immediately after first error + add_compile_options(-Wfatal-errors) + # Check for initializing unions without required braces + add_compile_options(-Wmissing-braces) +endif() + + +add_subdirectory("../c" "ImportNvtx") + +#if(${DOMAINS_ERROR_TEST_NAME_IS_MISSING}) +# target_compile_definitions(domains PRIVATE ERROR_TEST_NAME_IS_MISSING) +#endif() + +add_executable(runtest "RunTest.cpp") +target_link_libraries(runtest PRIVATE nvtx3-cpp) + +add_library(inj SHARED "PrintInjection.cpp") +target_link_libraries(inj PRIVATE nvtx3-cpp) +set_target_properties(inj PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(self SHARED "TestSelfInjection.cpp" "SelfInjection.cpp") +target_link_libraries(self PRIVATE nvtx3-cpp) +set_target_properties(self PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(calls SHARED "Calls.cpp" "SelfInjection.cpp") +target_link_libraries(calls PRIVATE nvtx3-cpp) +set_target_properties(calls PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(coverage SHARED "Coverage.cpp") +target_link_libraries(coverage PRIVATE nvtx3-cpp) +set_target_properties(coverage PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(coveragec SHARED "CoverageC.c") +target_link_libraries(coveragec PRIVATE nvtx3-c) +set_target_properties(coveragec PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(coverage-cu SHARED "CoverageCuda.cu") +target_link_libraries(coverage-cu PRIVATE nvtx3-cpp) +set_target_properties(coverage-cu PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(coverage-mem SHARED "CoverageMem.c") +target_link_libraries(coverage-mem PRIVATE nvtx3-c) +set_target_properties(coverage-mem PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(coverage-memcudart SHARED "CoverageMemCudaRt.cu") +target_link_libraries(coverage-memcudart PRIVATE nvtx3-c) +set_target_properties(coverage-memcudart PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(coverage-payload SHARED "CoveragePayload.c") +target_link_libraries(coverage-payload PRIVATE nvtx3-c) +set_target_properties(coverage-payload PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(coverage-counter SHARED "CoverageCounter.c") +target_link_libraries(coverage-counter PRIVATE nvtx3-c) +set_target_properties(coverage-counter PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(attributes SHARED "Attributes.cpp") +target_link_libraries(attributes PRIVATE nvtx3-cpp) +set_target_properties(attributes PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(domains SHARED "Domains.cpp") +target_link_libraries(domains PRIVATE nvtx3-cpp) +set_target_properties(domains PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(categories SHARED "NamedCategories.cpp") +target_link_libraries(categories PRIVATE nvtx3-cpp) +set_target_properties(categories PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(regstrings SHARED "RegisteredStrings.cpp") +target_link_libraries(regstrings PRIVATE nvtx3-cpp) +set_target_properties(regstrings PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + +add_library(linkerdupes SHARED "LinkerDupesMain.cpp" "LinkerDupesFileA.cpp" "LinkerDupesFileB.cpp") +target_link_libraries(linkerdupes PRIVATE nvtx3-cpp) +set_target_properties(linkerdupes PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) diff --git a/tests/CMakeSettings.json b/tests/CMakeSettings.json new file mode 100644 index 0000000..33690da --- /dev/null +++ b/tests/CMakeSettings.json @@ -0,0 +1,40 @@ +{ + "configurations": [ + { + "name": "x64-Debug", + "generator": "Ninja", + "configurationType": "Debug", + "inheritEnvironments": [ "msvc_x64_x64" ], + "buildRoot": "${projectDir}\\out\\build\\${name}", + "installRoot": "${projectDir}\\out\\install\\${name}", + "cmakeCommandArgs": "", + "buildCommandArgs": "-v", + "ctestCommandArgs": "" + }, + { + "name": "x64-Release", + "generator": "Ninja", + "configurationType": "RelWithDebInfo", + "buildRoot": "${projectDir}\\out\\build\\${name}", + "installRoot": "${projectDir}\\out\\install\\${name}", + "cmakeCommandArgs": "", + "buildCommandArgs": "-v", + "ctestCommandArgs": "", + "inheritEnvironments": [ "msvc_x64_x64" ] + }, + { + "name": "WSL-GCC-Debug", + "generator": "Ninja", + "configurationType": "Debug", + "buildRoot": "${projectDir}\\out\\build\\${name}", + "installRoot": "${projectDir}\\out\\install\\${name}", + "cmakeExecutable": "cmake", + "cmakeCommandArgs": "", + "buildCommandArgs": "", + "ctestCommandArgs": "", + "inheritEnvironments": [ "linux_x64" ], + "wslPath": "${defaultWSLPath}", + "variables": [] + } + ] +} \ No newline at end of file diff --git a/tests/Calls.cpp b/tests/Calls.cpp new file mode 100644 index 0000000..0849a80 --- /dev/null +++ b/tests/Calls.cpp @@ -0,0 +1,413 @@ +#include + +#include +#include + +#include +#include +#include + +#include "SelfInjection.h" +#include "PrettyPrintersNvtxC.h" + +#include "DllHelper.h" + +class CallbackTester +{ + Callbacks stored; + std::vector calls; +public: + +public: + void Record(Call const& call) { calls.push_back(call); } + + CallbackTester() : stored(g_callbacks) + { + g_callbacks.Default = [&](Call const& call) { Record(call); }; + } + ~CallbackTester() { g_callbacks = stored; } + + bool CallsMatch(std::vector expCalls, bool verbose = false) const + { + auto cmp = [&](Call const& lhs, Call const& rhs) + { + return Same(lhs, rhs, true, verbose, "NVTX call"); + }; + + bool match = std::equal(calls.begin(), calls.end(), expCalls.begin(), cmp); + if (verbose && !match) + { + auto printCall = [](Call const& c) { std::cout << " " << *c << "\n"; }; + std::cout << "Did not get expected NVTX C API call sequence! Expected:\n"; + for (auto& c : expCalls) printCall(c); + std::cout << "Recorded:\n"; + for (auto& c : calls) printCall(c); + } + + return match; + } +}; + +template struct a_lib { static constexpr const char* name = "LibA"; }; +template struct b_lib { static constexpr const char* name = "LibB"; }; +template struct c_lib { static constexpr const char* name = "LibC"; }; + +template struct cat1 { static constexpr const char* name = "Cat1"; static constexpr const uint32_t id = 1; }; +template struct cat2 { static constexpr const char* name = "Cat2"; static constexpr const uint32_t id = 2; }; +template struct cat3 { static constexpr const char* name = "Cat3"; static constexpr const uint32_t id = 3; }; + +template struct reg1 { static constexpr const char* message = "Reg1"; }; +template struct reg2 { static constexpr const char* message = "Reg2"; }; +template struct reg3 { static constexpr const char* message = "Reg3"; }; + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + bool verbose = false; + const std::string verboseArg = "-v"; + for (; *argv; ++argv) + { + if (*argv == verboseArg) verbose = true; + } + + using namespace nvtx3; + + //---------------------------- Tests -------------------------------------- + + if (verbose) std::cout << "--------- Testing injection loader\n"; + + { + CallbackTester t; + + nvtxInitialize(nullptr); + nvtxInitialize(nullptr); + + if (!t.CallsMatch({ + CALL_LOAD(1), + CALL(CORE2, Initialize, nullptr), + CALL(CORE2, Initialize, nullptr) + }, verbose)) return 1; + } + + if (verbose) std::cout << "--------- Testing C API\n"; + + { + CallbackTester t; + + const char* teststr = "Testing 1 2 3!"; + nvtxMarkA(teststr); + + if (!t.CallsMatch({ + CALL(CORE, MarkA, teststr) + }, verbose)) return 1; + } + + { + CallbackTester t; + + char teststr[] = "Testing 1 2 3!"; + nvtxMarkA(teststr); + memcpy(teststr, "Overwritten!!!", sizeof(teststr)); + + if (!t.CallsMatch({ + CALL(CORE, MarkA, "Testing 1 2 3!") + }, verbose)) return 1; + } + + { + CallbackTester t; + + wchar_t teststr[] = L"Testing 1 2 3!"; + nvtxMarkW(teststr); + memcpy(teststr, L"Overwritten!!!", sizeof(teststr)); + + if (!t.CallsMatch({ + CALL(CORE, MarkW, L"Testing 1 2 3!") + }, verbose)) return 1; + } + + { + CallbackTester t; + + nvtxEventAttributes_t attr{NVTX_VERSION, sizeof(nvtxEventAttributes_t)}; + attr.category = 123; + attr.colorType = NVTX_COLOR_ARGB; + attr.color = 0xFF4466BB; + attr.messageType = NVTX_MESSAGE_TYPE_ASCII; + attr.message = MakeMessage("Test MarkEX"); + attr.category = 123; + attr.payloadType = NVTX_PAYLOAD_TYPE_DOUBLE; + attr.payload = MakePayload(3.14159); + nvtxMarkEx(&attr); + + nvtxEventAttributes_t attr2 = attr; + memset(&attr, 0, sizeof(attr)); + + if (!t.CallsMatch({ + CALL(CORE, MarkEx, &attr2) + }, verbose)) return 1; + } + + if (verbose) std::cout << "--------- Testing C++ API\n"; + + { + CallbackTester t; + + mark("Testing 1 2 3!"); + mark(L"Testing 1 2 3!"); + + if (!t.CallsMatch({ + CALL(CORE2, DomainMarkEx, nullptr, event_attributes{"Testing 1 2 3!"}.get()), + CALL(CORE2, DomainMarkEx, nullptr, event_attributes{L"Testing 1 2 3!"}.get()) + }, verbose)) return 1; + } + + { + CallbackTester t; + + nvtxEventAttributes_t attrExpected{NVTX_VERSION, sizeof(nvtxEventAttributes_t), + 123, // category + NVTX_COLOR_ARGB, 0xFF4466BB, + NVTX_PAYLOAD_TYPE_DOUBLE, 0, MakePayload(3.14159), + NVTX_MESSAGE_TYPE_ASCII, MakeMessage("Test msg") + }; + + // Same args, different order + mark("Test msg", rgb(0x44, 0x66, 0xBB), category(123), payload(3.14159)); + mark(payload(3.14159), "Test msg", rgb(0x44, 0x66, 0xBB), category(123)); + mark(category(123), payload(3.14159), "Test msg", rgb(0x44, 0x66, 0xBB)); + mark(rgb(0x44, 0x66, 0xBB), category(123), payload(3.14159), "Test msg"); + + // Same args with duplicates, test first-one-wins behavior (including union type changes) + mark("Test msg", rgb(0x44, 0x66, 0xBB), category(123), payload(3.14159), + "Bad msg", rgb(0x10, 0x20, 0x30), category(321), payload(3.0)); + mark("Test msg", rgb(0x44, 0x66, 0xBB), category(123), payload(3.14159), + L"Bad message"); + mark("Test msg", rgb(0x44, 0x66, 0xBB), category(123), payload(3.14159), + payload(3.14159f)); + + if (!t.CallsMatch({ + 7, CALL(CORE2, DomainMarkEx, nullptr, &attrExpected) + }, verbose)) return 1; + } + + { + CallbackTester t; + constexpr int N = 1; + auto hA = (nvtxDomainHandle_t)1; + + mark_in>("First call"); + mark_in>("Second call"); + mark_in>("Third call"); + + if (!t.CallsMatch({ + CALL(CORE2, DomainCreateA, "LibA"), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"First call"}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"Second call"}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"Third call"}.get()) + }, verbose)) return 1; + } + + { + CallbackTester t; + constexpr int N = 2; + auto hA = (nvtxDomainHandle_t)1; + auto hB = (nvtxDomainHandle_t)2; + + mark_in>("First call"); + mark_in>("Second call"); + mark_in>("First call"); + mark_in>("Second call"); + + if (!t.CallsMatch({ + CALL(CORE2, DomainCreateA, "LibA"), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"First call"}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"Second call"}.get()), + CALL(CORE2, DomainCreateA, "LibB"), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"First call"}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"Second call"}.get()) + }, verbose)) return 1; + } + + { + CallbackTester t; + constexpr int N = 3; + auto hA = (nvtxDomainHandle_t)1; + auto hB = (nvtxDomainHandle_t)2; + + mark_in>("DA, Cat 1, call 1", named_category_in>::get>()); + mark_in>("DA, Cat 1, call 2", named_category_in>::get>()); + mark_in>("DA, Cat 2, call 1", named_category_in>::get>()); + mark_in>("DA, Cat 2, call 2", named_category_in>::get>()); + mark_in>("DB, Cat 1, call 1", named_category_in>::get>()); + mark_in>("DB, Cat 1, call 2", named_category_in>::get>()); + mark_in>("DB, Cat 2, call 1", named_category_in>::get>()); + mark_in>("DB, Cat 2, call 2", named_category_in>::get>()); + + if (!t.CallsMatch({ + CALL(CORE2, DomainCreateA, "LibA"), + CALL(CORE2, DomainNameCategoryA, hA, 1, "Cat1"), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"DA, Cat 1, call 1", category(1)}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"DA, Cat 1, call 2", category(1)}.get()), + CALL(CORE2, DomainNameCategoryA, hA, 2, "Cat2"), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"DA, Cat 2, call 1", category(2)}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"DA, Cat 2, call 2", category(2)}.get()), + CALL(CORE2, DomainCreateA, "LibB"), + CALL(CORE2, DomainNameCategoryA, hB, 1, "Cat1"), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"DB, Cat 1, call 1", category(1)}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"DB, Cat 1, call 2", category(1)}.get()), + CALL(CORE2, DomainNameCategoryA, hB, 2, "Cat2"), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"DB, Cat 2, call 1", category(2)}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"DB, Cat 2, call 2", category(2)}.get()), + }, verbose)) return 1; + } + + { + CallbackTester t; + constexpr int N = 4; + auto hA = (nvtxDomainHandle_t)1; + auto hB = (nvtxDomainHandle_t)2; + auto hReg1 = (nvtxStringHandle_t)1; + auto hReg2 = (nvtxStringHandle_t)2; + + mark_in>(registered_string_in>::get>()); + mark_in>(registered_string_in>::get>()); + mark_in>(registered_string_in>::get>()); + mark_in>(registered_string_in>::get>()); + mark_in>(registered_string_in>::get>()); + mark_in>(registered_string_in>::get>()); + mark_in>(registered_string_in>::get>()); + mark_in>(registered_string_in>::get>()); + + if (!t.CallsMatch({ + CALL(CORE2, DomainCreateA, "LibA"), + CALL(CORE2, DomainRegisterStringA, hA, "Reg1"), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg1}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg1}.get()), + CALL(CORE2, DomainRegisterStringA, hA, "Reg2"), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg2}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg2}.get()), + CALL(CORE2, DomainCreateA, "LibB"), + CALL(CORE2, DomainRegisterStringA, hB, "Reg1"), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg1}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg1}.get()), + CALL(CORE2, DomainRegisterStringA, hB, "Reg2"), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg2}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg2}.get()), + }, verbose)) return 1; + } + + { + CallbackTester t; + constexpr int N = 5; + auto hA = (nvtxDomainHandle_t)1; + auto hB = (nvtxDomainHandle_t)2; + auto hReg1 = (nvtxStringHandle_t)1; + auto hReg2 = (nvtxStringHandle_t)2; + + auto& a_regstr1 = registered_string_in>::get>(); + auto& a_regstr2 = registered_string_in>::get>(); + auto& b_regstr1 = registered_string_in>::get>(); + auto& b_regstr2 = registered_string_in>::get>(); + + auto& a_cat1 = named_category_in>::get>(); + auto& a_cat2 = named_category_in>::get>(); + auto& b_cat1 = named_category_in>::get>(); + auto& b_cat2 = named_category_in>::get>(); + + mark_in>(a_cat1, a_regstr1); + mark_in>(a_cat1, a_regstr1); + mark_in>(a_cat2, a_regstr2); + mark_in>(a_cat2, a_regstr2); + mark_in>(b_cat1, b_regstr1); + mark_in>(b_cat1, b_regstr1); + mark_in>(b_cat2, b_regstr2); + mark_in>(b_cat2, b_regstr2); + + if (!t.CallsMatch({ + CALL(CORE2, DomainCreateA, "LibA"), + CALL(CORE2, DomainRegisterStringA, hA, "Reg1"), + CALL(CORE2, DomainRegisterStringA, hA, "Reg2"), + CALL(CORE2, DomainCreateA, "LibB"), + CALL(CORE2, DomainRegisterStringA, hB, "Reg1"), + CALL(CORE2, DomainRegisterStringA, hB, "Reg2"), + CALL(CORE2, DomainNameCategoryA, hA, 1, "Cat1"), + CALL(CORE2, DomainNameCategoryA, hA, 2, "Cat2"), + CALL(CORE2, DomainNameCategoryA, hB, 1, "Cat1"), + CALL(CORE2, DomainNameCategoryA, hB, 2, "Cat2"), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg1, category(1)}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg1, category(1)}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg2, category(2)}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{hReg2, category(2)}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg1, category(1)}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg1, category(1)}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg2, category(2)}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{hReg2, category(2)}.get()), + }, verbose)) return 1; + } + + { + CallbackTester t; + constexpr int N = 6; + auto hA = (nvtxDomainHandle_t)1; + auto hB = (nvtxDomainHandle_t)2; + + { + scoped_range_in> r1("Sequential range 1"); + mark_in>("Mark in range"); + } + { + scoped_range_in> r2("Sequential range 2"); + mark_in>("Mark in range"); + } + { + scoped_range_in> r1("Nested range 1"); + scoped_range_in> r2("Nested range 2"); + mark_in>("Mark in range"); + } + + { + scoped_range_in> r1("Sequential range 1"); + mark_in>("Mark in range"); + } + { + scoped_range_in> r2("Sequential range 2"); + mark_in>("Mark in range"); + } + { + scoped_range_in> r1("Nested range 1"); + scoped_range_in> r2("Nested range 2"); + mark_in>("Mark in range"); + } + + if (!t.CallsMatch({ + CALL(CORE2, DomainCreateA, "LibA"), + CALL(CORE2, DomainRangePushEx, hA, event_attributes{"Sequential range 1"}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"Mark in range"}.get()), + CALL(CORE2, DomainRangePop, hA), + CALL(CORE2, DomainRangePushEx, hA, event_attributes{"Sequential range 2"}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"Mark in range"}.get()), + CALL(CORE2, DomainRangePop, hA), + CALL(CORE2, DomainRangePushEx, hA, event_attributes{"Nested range 1"}.get()), + CALL(CORE2, DomainRangePushEx, hA, event_attributes{"Nested range 2"}.get()), + CALL(CORE2, DomainMarkEx, hA, event_attributes{"Mark in range"}.get()), + CALL(CORE2, DomainRangePop, hA), + CALL(CORE2, DomainRangePop, hA), + CALL(CORE2, DomainCreateA, "LibB"), + CALL(CORE2, DomainRangePushEx, hB, event_attributes{"Sequential range 1"}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"Mark in range"}.get()), + CALL(CORE2, DomainRangePop, hB), + CALL(CORE2, DomainRangePushEx, hB, event_attributes{"Sequential range 2"}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"Mark in range"}.get()), + CALL(CORE2, DomainRangePop, hB), + CALL(CORE2, DomainRangePushEx, hB, event_attributes{"Nested range 1"}.get()), + CALL(CORE2, DomainRangePushEx, hB, event_attributes{"Nested range 2"}.get()), + CALL(CORE2, DomainMarkEx, hB, event_attributes{"Mark in range"}.get()), + CALL(CORE2, DomainRangePop, hB), + CALL(CORE2, DomainRangePop, hB), + }, verbose)) return 1; + } + + if (verbose) std::cout << "--------- Success!\n"; + return 0; +} diff --git a/tests/Coverage.cpp b/tests/Coverage.cpp new file mode 100644 index 0000000..440d80c --- /dev/null +++ b/tests/Coverage.cpp @@ -0,0 +1,8 @@ +#include "TestCoverage.h" +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + return RunTestCommon(argc, argv); +} diff --git a/tests/CoverageC.c b/tests/CoverageC.c new file mode 100644 index 0000000..0fc3059 --- /dev/null +++ b/tests/CoverageC.c @@ -0,0 +1,73 @@ +#include + +void TestCore() +{ + nvtxEventAttributes_t attributes; + nvtxRangeId_t rangeId; + + attributes.version = NVTX_VERSION; + attributes.size = sizeof(attributes); + attributes.category = 0; + attributes.colorType = NVTX_COLOR_ARGB; + attributes.color = 0xFF1133FF; + attributes.payloadType = NVTX_PAYLOAD_UNKNOWN; + attributes.payload.llValue = 0; + attributes.messageType = NVTX_MESSAGE_TYPE_ASCII; + attributes.message.ascii = "Test message"; + + nvtxMarkEx(&attributes); + nvtxMarkA("MarkA"); + nvtxMarkW(L"MarkW"); + rangeId = nvtxRangeStartEx(&attributes); + nvtxRangeEnd(rangeId); + rangeId = nvtxRangeStartA("RangeStartA"); + nvtxRangeEnd(rangeId); + rangeId = nvtxRangeStartW(L"RangeStartW"); + nvtxRangeEnd(rangeId); + nvtxRangePushEx(&attributes); + nvtxRangePop(); + nvtxRangePushA("RangePushA"); + nvtxRangePop(); + nvtxRangePushW(L"RangePushW"); + nvtxRangePop(); +} + +void TestCore2() +{ + nvtxEventAttributes_t attributes; + nvtxRangeId_t rangeId; + nvtxDomainHandle_t domain, domainW; + + attributes.version = NVTX_VERSION; + attributes.size = sizeof(attributes); + attributes.category = 0; + attributes.colorType = NVTX_COLOR_ARGB; + attributes.color = 0xFF1133FF; + attributes.payloadType = NVTX_PAYLOAD_UNKNOWN; + attributes.payload.llValue = 0; + attributes.messageType = NVTX_MESSAGE_TYPE_ASCII; + attributes.message.ascii = "Test message"; + + domain = nvtxDomainCreateA("DomainA"); + domainW = nvtxDomainCreateW(L"DomainW"); + + nvtxDomainMarkEx(domain, &attributes); + rangeId = nvtxDomainRangeStartEx(domain, &attributes); + nvtxDomainRangeEnd(domain, rangeId); + nvtxDomainRangePushEx(domain, &attributes); + nvtxDomainRangePop(domain); +} + +#include "DllHelper.h" + +DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + TestCore(); + TestCore2(); + + return 0; +} diff --git a/tests/CoverageCounter.c b/tests/CoverageCounter.c new file mode 100644 index 0000000..72ede95 --- /dev/null +++ b/tests/CoverageCounter.c @@ -0,0 +1,30 @@ +#include + +void TestMem() +{ + nvtxDomainHandle_t domain; + nvtxCountersHandle_t counter; + nvtxCountersAttr_t attr; + int64_t i64 = 0; + double f64 = 0.0; + + domain = nvtxDomainCreateA("Domain"); + + counter = nvtxCountersRegister(domain, &attr); + nvtxCountersSampleInt64(domain, counter, i64); + nvtxCountersSampleFloat64(domain, counter, f64); + nvtxCountersSampleNoValue(domain, counter, NVTX_COUNTERS_SAMPLE_UNCHANGED); +} + +#include "DllHelper.h" + +DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + TestMem(); + + return 0; +} diff --git a/tests/CoverageCuda.cu b/tests/CoverageCuda.cu new file mode 100644 index 0000000..440d80c --- /dev/null +++ b/tests/CoverageCuda.cu @@ -0,0 +1,8 @@ +#include "TestCoverage.h" +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + return RunTestCommon(argc, argv); +} diff --git a/tests/CoverageMem.c b/tests/CoverageMem.c new file mode 100644 index 0000000..8266806 --- /dev/null +++ b/tests/CoverageMem.c @@ -0,0 +1,26 @@ +#include + +void TestMem() +{ + nvtxDomainHandle_t domain; + nvtxMemHeapHandle_t heap; + nvtxMemHeapDesc_t heapDesc; + + domain = nvtxDomainCreateA("Domain"); + + heap = nvtxMemHeapRegister(domain, &heapDesc); + nvtxMemPermissionsUnbind(domain, 0); +} + +#include "DllHelper.h" + +DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + TestMem(); + + return 0; +} diff --git a/tests/CoverageMemCudaRt.cu b/tests/CoverageMemCudaRt.cu new file mode 100644 index 0000000..adb3902 --- /dev/null +++ b/tests/CoverageMemCudaRt.cu @@ -0,0 +1,25 @@ +#include + +void TestMemCudaRt() +{ + nvtxDomainHandle_t domain; + nvtxMemPermissionsHandle_t perm; + + domain = nvtxDomainCreateA("Domain"); + + perm = nvtxMemCudaGetProcessWidePermissions(domain); + nvtxMemCudaSetPeerAccess(domain, perm, 0, 0); +} + +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + TestMemCudaRt(); + + return 0; +} diff --git a/tests/CoveragePayload.c b/tests/CoveragePayload.c new file mode 100644 index 0000000..6b222c9 --- /dev/null +++ b/tests/CoveragePayload.c @@ -0,0 +1,27 @@ +#include +#include + +void TestMem() +{ + nvtxDomainHandle_t domain; + uint8_t enabled; + uint64_t handle; + nvtxPayloadSchemaAttr_t attr; + + domain = nvtxDomainCreateA("Domain"); + enabled = nvtxDomainIsEnabled(domain); + handle = nvtxPayloadSchemaRegister(domain, &attr); +} + +#include "DllHelper.h" + +DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + TestMem(); + + return 0; +} diff --git a/tests/DllHelper.h b/tests/DllHelper.h new file mode 100644 index 0000000..ddfb569 --- /dev/null +++ b/tests/DllHelper.h @@ -0,0 +1,38 @@ +/* Adapted from Niall Douglas's explanation of visibility in the GCC docs */ +#if defined(_WIN32) || defined(__CYGWIN__) +#ifdef __GNUC__ +#define DLL_EXPORT __attribute__((dllexport)) +#else +#define DLL_EXPORT __declspec(dllexport) +#endif +#else +#if __GNUC__ >= 4 +#define DLL_EXPORT __attribute__((visibility ("default"))) +#else +#define DLL_EXPORT +#endif +#endif + +/* It's best to build with -fvisibility=hidden. In CMake, that can be done with: + set_target_properties(MyTarget PROPERTIES C_VISIBILITY_PRESET hidden CXX_VISIBILITY_PRESET hidden) + + If you can't build with that flag, then push visibility=hidden and never pop it: + + #ifdef __GNUC__ + #pragma GCC visibility push(hidden) + #endif +*/ + +#if defined(_WIN32) +#include +#define LOAD_DLL(x) LoadLibraryA(x) +#define GET_DLL_FUNC GetProcAddress +#define DLL_PREFIX "" +#define DLL_SUFFIX ".dll" +#elif defined(__GNUC__) +#include +#define LOAD_DLL(x) dlopen(x, RTLD_LAZY) +#define GET_DLL_FUNC dlsym +#define DLL_PREFIX "lib" +#define DLL_SUFFIX ".so" +#endif diff --git a/tests/Domains.cpp b/tests/Domains.cpp new file mode 100644 index 0000000..6b1005b --- /dev/null +++ b/tests/Domains.cpp @@ -0,0 +1,97 @@ +#if defined(_MSC_VER) && _MSC_VER < 1914 +#define STATIC_ASSERT_TESTING 0 +#else +#define STATIC_ASSERT_TESTING 1 +#endif + +#if defined(STATIC_ASSERT_TESTING) +#include +#define NVTX3_STATIC_ASSERT(c, m) do { if (!(c)) printf("static_assert would fail: %s\n", m); } while (0) +#endif + +#include + +#include + +// Domain description types +struct char_test { static constexpr const char* name{"Test char"}; }; +struct wchar_test { static constexpr const wchar_t* name{L"Test wchar_t"}; }; +struct error_name_missing { static constexpr const char* not_name{"Test name is missing"}; }; +struct error_name_is_bad_type { static constexpr const int name{5}; }; + +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + using namespace nvtx3; + + if (0) + { + std::cout << std::boolalpha; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string= " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string= " << detail::is_c_string::value << '\n'; + + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string= " << detail::is_c_string::value << '\n'; + std::cout << "is_c_string = " << detail::is_c_string::value << '\n'; + + std::cout << "-------------\n"; + } + + std::cout << "- Global domain (mark alias):\n"; + mark("Mark in global domain (implicit)"); + + std::cout << "- Global domain implicit:\n"; + auto& gi = domain::get<>(); + mark_in<>("Mark in global domain (implicit)"); + + std::cout << "- Global domain explicit:\n"; + auto& ge = domain::get(); + mark_in("Mark in global domain (explicit)"); + + std::cout << "- Test domain (char):\n"; + auto& d1 = domain::get(); + mark_in("Mark in char_test domain"); + + std::cout << "- Test domain (wchar_t):\n"; + auto& d2 = domain::get(); + mark_in("Mark in wchar_test domain"); + +#if STATIC_ASSERT_TESTING + +#if 1 // defined(ERROR_TEST_NAME_IS_MISSING) + { + std::cout << "- Error test - domain is missing name member:\n"; + auto& d3 = domain::get(); + mark_in("Mark in error_name_missing domain"); + scoped_range_in r3("Mark in error_name_missing domain"); + } +#endif + +#if 1 // defined(ERROR_TEST_NAME_IS_BAD_TYPE) + { + std::cout << "- Error test - domain name member isn't narrow or wide char array:\n"; + auto& d4 = domain::get(); + mark_in("Mark in error_name_is_bad_type domain"); + scoped_range_in r4("Mark in error_name_is_bad_type domain"); + } +#endif + +#endif // STATIC_ASSERT_TESTING + + return 0; +} diff --git a/tests/InjectionHelper.h b/tests/InjectionHelper.h new file mode 100644 index 0000000..427b54d --- /dev/null +++ b/tests/InjectionHelper.h @@ -0,0 +1,569 @@ +#pragma once + +// [Best practices for injection implementions] +// Set NVTX_NO_IMPL to make the NVTX headers define the API types and function +// prototypes only, not the inline impls. Be sure on GCC to use -Wno-unused-function +// to avoid warnings for undefined static prototypes. +#define NVTX_NO_IMPL + +// [Best practices for injection implementions] +// Microsoft's compiler issues warning 26812 when compiling a C-style enum in C++ +// instead of using the new "enum class" style. Since the NVTX headers are written in +// C, the enums defined there will trigger this warning. Use this code to disable it. +#if defined(_MSC_VER) +#pragma warning (disable : 26812) +#endif +#include + +#include +#include +#include +#include + +// On Windows, use __declspec(dllexport) with extern "C" to produce an export +// with the correct unmangled name, or use a .def file. Everywhere else, the +// extern "C" symbol will be exported by default. +#if defined(_MSC_VER) +#define NVTX_INJECTION_DLL_EXPORT __declspec(dllexport) +#else +#define NVTX_INJECTION_DLL_EXPORT +#endif + +namespace NvtxInjectionHelper { + +//============ Generic utility functions ====================================== + +inline namespace detail_generic { + +//--- maxVal --- + +// Variadic alternative to std::max that doesn't need an initializer list, +// doesn't conflict with MSVC's #define for max, and has no trouble with +// constexpr usage. Handles having zero parameters passed, returning +// std::numeric_limits::min in that case, as long as the template +// parameter T is explicitly specified. Takes arguments by value, which +// avoids the issue of returning a reference to something when called +// with no parameters. Example uses: +// + +template +constexpr inline T maxVal() { return std::numeric_limits::min(); } + +template +constexpr inline T maxVal(T first, Rest... rest) +{ + T restMax = maxVal(rest...); + return (first > restMax) ? first : restMax; +} + +//--- tuple size helper --- + +// Generic utility for getting the size of a std::tuple, using its value +// as opposed to std::tuple_size<> which takes the tuple's type. In a +// generic lambda where the parameter's type is "auto", it's extra work +// to figure out the type +template +constexpr inline size_t size_of_tuple(std::tuple const&) +{ + return sizeof...(Ts); +} + +//--- tuple helpers to loop over items --- + +// We need a way to call a function f on each element of a tuple, like this: +// +// f(std::get<0>(t)); +// f(std::get<1>(t)); +// f(std::get<2>(t)); etc. +// +// We want something like this, where "Is" is a parameter pack of 0,1,2,etc.: +// +// f(std::get(t))...; +// +// ...but parameter pack expansion is only allowed within the context of args +// to a function call or a braced init list. We also must handle the case +// where the tuple is empty, we should discard the results of all the calls +// to f, even if it returns different types for each call. Easiest way to +// do this is by forwarding the elements of the tuple as args to a helper +// function that calls f on each arg, like this: +// +// for_each_in_parameter_pack(f, std::get(t)...); +// +// But we also want perfect forwarding of the function and the tuple. +// The following utilites "for_each_in_tuple", "for_each_in_tuple_helper", +// and "for_each_in_parameter_pack" are provided to allow code such as this +// "loop" over tuple elements. Note that "thing" in each iteration can be +// a different type, because a tuple's elements may be different types, so +// generic lambdas are very convenient here: +// +// for_each_in_tuple(tuple_of_things, +// [](auto const& thing) +// { +// std::cout << thing << std::endl; +// } +// ); + +template +inline void for_each_in_parameter_pack(F&& f) {} + +template +inline void for_each_in_parameter_pack(F&& f, First const& first, Rest const&... rest) +{ + // Call f on the first argument, and explicitly discard the result by casting to void + static_cast(std::forward(f)(first)); + + // Recurse to call f on the rest of the arguments + for_each_in_parameter_pack(std::forward(f), rest...); +} + +// Generic utility for calling a function f for each element of a tuple t +template +inline void for_each_in_tuple_helper(T const& t, F&& f, std::index_sequence) +{ + for_each_in_parameter_pack( + std::forward(f), + std::get(t)... + ); +} +template +inline void for_each_in_tuple(std::tuple const& t, F&& f) +{ + for_each_in_tuple_helper(t, std::forward(f), std::make_index_sequence()); +} + +} // namespace detail_generic + +//============ NVTX injection helper internal utilities ======================= + +inline namespace detail_nvtx { + +//--- id_t --- +// Define generic integer type for holding all modules' callback id enum values. +// These are used as indexes into the handler arrays for each module. +using id_t = unsigned int; + +//--- id_v --- +// Nickname for std::integral_constant, which is used for all callback enum values. +// Using an integral constant allows performing correctness checks at compile time, +// which is not possible in C++ with function parameter values, only their types. +// Including the value in the type works around this problem. +template +using id_v = std::integral_constant; + +//--- NVTX_CBID --- +// Macro to succinctly turn an NVTX_CBID_* enum value into a compile-time constant, +// using std::integral_constant. This makes it possible to perform correctness +// checks at compile time, for example ensuring a handler's signature is compatible +// with the NVTX API call it is being installed to handle. Syntax is meant to look +// familiar. For example, replace: +// NVTX_CBID_CORE_MarkA +// with: +// NVTX_CBID(CORE_MarkA) +// when passing CBID values to NvtxInjectionHelper::MakeHandlerTable. +#define NVTX_CBID(func) NvtxInjectionHelper::id_v{} + +//--- EnumTypeToModuleId --- +// Template variable to map from call id enum types to module id values (see nvtxTypes.h) +// For example, EnumTypeToModuleId == NVTX_CB_MODULE_CORE. +template +constexpr static NvtxCallbackModule EnumTypeToModuleId = NVTX_CB_MODULE_INVALID; + +template<> constexpr NvtxCallbackModule EnumTypeToModuleId = NVTX_CB_MODULE_CORE; +template<> constexpr NvtxCallbackModule EnumTypeToModuleId = NVTX_CB_MODULE_CUDA; +template<> constexpr NvtxCallbackModule EnumTypeToModuleId = NVTX_CB_MODULE_OPENCL; +template<> constexpr NvtxCallbackModule EnumTypeToModuleId = NVTX_CB_MODULE_CUDART; +template<> constexpr NvtxCallbackModule EnumTypeToModuleId = NVTX_CB_MODULE_CORE2; +template<> constexpr NvtxCallbackModule EnumTypeToModuleId = NVTX_CB_MODULE_SYNC; + +//--- IdToModuleId --- +// Helper for EnumTypeToModuleId to convert directly from an integral_constant of a call id enum +// to its module id. For example, since NVTX_CBID(CORE_MarkA) is an integral_constant, it cannot +// be used directly as in EnumTypeToModuleId, since NVTX_CBID(CORE_MarkA)'s +// type is std::integral_constant. This helper extracts +// the enum's type from the integral_constant, allowing EnumConstToModuleId. +template +constexpr static NvtxCallbackModule IdToModuleId = EnumTypeToModuleId; + + +//--- IdToHandlerType +// Template using to map from call id values to matching function pointer types. +template struct IdToHandlerType { using type = nullptr_t; }; + +// Macro for defining IdToHandlerType specializations for each id. +// mod = module, i.e. CORE, CORE2 +// func = prefixless function name, i.e. MarkEx, DomainCreateA +// impl = impl or fakeimpl, depending on whether or not to use real types or the +// nvtxTypes.h "fakeimpl" types, which don't depend on CUDA/OpenCL headers. +#define NVTX_ID_TO_TYPE(mod, func, impl) \ +template <> struct IdToHandlerType { using type = nvtx##func##_##impl##_fntype; } + +NVTX_ID_TO_TYPE(CORE, MarkEx , impl); +NVTX_ID_TO_TYPE(CORE, MarkA , impl); +NVTX_ID_TO_TYPE(CORE, MarkW , impl); +NVTX_ID_TO_TYPE(CORE, RangeStartEx , impl); +NVTX_ID_TO_TYPE(CORE, RangeStartA , impl); +NVTX_ID_TO_TYPE(CORE, RangeStartW , impl); +NVTX_ID_TO_TYPE(CORE, RangeEnd , impl); +NVTX_ID_TO_TYPE(CORE, RangePushEx , impl); +NVTX_ID_TO_TYPE(CORE, RangePushA , impl); +NVTX_ID_TO_TYPE(CORE, RangePushW , impl); +NVTX_ID_TO_TYPE(CORE, RangePop , impl); +NVTX_ID_TO_TYPE(CORE, NameCategoryA, impl); +NVTX_ID_TO_TYPE(CORE, NameCategoryW, impl); +NVTX_ID_TO_TYPE(CORE, NameOsThreadA, impl); +NVTX_ID_TO_TYPE(CORE, NameOsThreadW, impl); + +NVTX_ID_TO_TYPE(CORE2, DomainMarkEx , impl); +NVTX_ID_TO_TYPE(CORE2, DomainRangeStartEx , impl); +NVTX_ID_TO_TYPE(CORE2, DomainRangeEnd , impl); +NVTX_ID_TO_TYPE(CORE2, DomainRangePushEx , impl); +NVTX_ID_TO_TYPE(CORE2, DomainRangePop , impl); +NVTX_ID_TO_TYPE(CORE2, DomainResourceCreate , impl); +NVTX_ID_TO_TYPE(CORE2, DomainResourceDestroy, impl); +NVTX_ID_TO_TYPE(CORE2, DomainNameCategoryA , impl); +NVTX_ID_TO_TYPE(CORE2, DomainNameCategoryW , impl); +NVTX_ID_TO_TYPE(CORE2, DomainRegisterStringA, impl); +NVTX_ID_TO_TYPE(CORE2, DomainRegisterStringW, impl); +NVTX_ID_TO_TYPE(CORE2, DomainCreateA , impl); +NVTX_ID_TO_TYPE(CORE2, DomainCreateW , impl); +NVTX_ID_TO_TYPE(CORE2, DomainDestroy , impl); +NVTX_ID_TO_TYPE(CORE2, Initialize , impl); + +#undef NVTX_ID_TO_TYPE + +//--- CheckHandlerTypeMatchesId --- +// Compile-time check provides easy-to-read error if FuncT isn't compatible with EnumT +template +constexpr inline void CheckHandlerTypeMatchesId() +{ + using ExpectedFuncT = typename IdToHandlerType::type; + + static_assert(std::is_same(), + "NVTX Injection Helper: The provided handler function's signature does not match the NVTX API for the given call id."); +} + +//--- Handler --- +// Represents id/handler pair for an NVTX call. Provides: +// - the call's id (NVTX_CBID_* enum values) +// - handler function pointer +// Preserves the type of the function as a template parameter. +// Erases the type of the enum, so it's not module-specific anymore. +// Allows being constructed and placed into a container at compile time, then +// later at run time doing the run-time-only cast of the function pointer. +// This enables processing of ids to occur at compile time. +template +class Handler +{ +public: + id_t id; + FuncT pfn; + + template + constexpr Handler(id_v e, FuncT pfn_) + : id(static_cast(EnumVal)) // Erase enum's type + , pfn(pfn_) + {} + + NvtxFunctionPointer Address() const noexcept + { + return reinterpret_cast(pfn); + } +}; + +//--- MakeHandler --- +// "Make" function for Handler to automatically deduce types from parameters +template +constexpr inline Handler MakeHandler(IdT id_, FuncT func) +{ + CheckHandlerTypeMatchesId(); + return Handler(id_, func); +} + +//--- ModuleHandlerTable --- +// Represents the set of Handlers for one module. Provides: +// - the module's id (NVTX_CB_MODULE_* enum values) +// - iterable container of id/handler pairs (empty means skip getting etbl for module) +// - highest call id value of handler in module (to confirm client has sufficient size) +// - a method to assign all the stored handlers into a client's handler table +// These objects can be constructed at compile time, including the highest call id used. +template +class ModuleHandlerTable +{ +public: + using tuple_t = std::tuple...>; + + static constexpr NvtxCallbackModule moduleId = mod; + tuple_t handlers; + id_t highestIdUsed; + + constexpr ModuleHandlerTable(tuple_t t) + : handlers(t) + , highestIdUsed(FindHighestId(t)) + {} + + void AssignToClient(NvtxFunctionTable clientTable) const noexcept + { + for_each_in_tuple(handlers, + [clientTable](auto const& handler) + { + if (handler.id != 0 && handler.pfn != nullptr) + { + *clientTable[handler.id] = handler.Address(); + } + } + ); + } + +private: + template + static constexpr id_t FindHighestIdHelper(tuple_t t, std::index_sequence) + { + return maxVal(std::get(t).id...); + } + + static constexpr id_t FindHighestId(tuple_t t) + { + return FindHighestIdHelper(t, std::make_index_sequence()); + } +}; + + +//--- MakeModuleHandlerTuple --- +// MakeModuleHandlerTuple takes NvtxCallbackModule "mod" as a template parameter, +// and loops over pairs of arguments (an enum and a handler function), building a +// tuple of Handler objects for the enums that are in module "mod", and ignoring +// ones that aren't. This lets the user pass in handlers for for all modules in +// one simple call, and we can build up separate handler tables for each module. +// MakeModuleHandlerTuple is recursive, peeling off two arguments in each recursive +// case, and having no args be the base case. The recursive case has a pair of +// overloads for whether or not the enum's type matches "mod" or not. Since these +// overloads are separate functions, it's mutual recursion, so both are declared +// first before the definitions. + +// Base case: no more arguments +template +constexpr inline auto MakeModuleHandlerTuple() +{ + return std::tuple<>{}; +} + +// Prototypes of recursive cases -- needed since they can call each other +template == mod, int> = 0, + typename... Args> +constexpr inline auto MakeModuleHandlerTuple(IdT, FuncT, Args...); + +template != mod, int> = 0, + typename... Args> +constexpr inline auto MakeModuleHandlerTuple(IdT, FuncT, Args...); + +// Recursive case 1: enum's type matches mod, so add it to the tuple +template == mod, int>, + typename... Args> +constexpr inline auto MakeModuleHandlerTuple(IdT id, FuncT f, Args... rest) +{ + // Verify types of id and function, using static_assert to provide a + // clear compile error if the types don't meet the requirements. + static_assert(IdToModuleId != NVTX_CB_MODULE_INVALID, + "MakeHandlerTable arguments must be pairs of IDs and handler functions. IDs must be enums starting with NVTX_CBID_. An invalid ID value was provided."); + + // Before adding this id/handler pair to the tuple, check to make sure + // there's not already an entry in the tuple with the same id. If so, + // provide a clear compile-time error message. + auto restTuple = MakeModuleHandlerTuple(rest...); + + return std::tuple_cat( + std::make_tuple(MakeHandler(id, f)), + restTuple); +} + +// Recursive case 2: id is not in module, so fwd result from remaining args +template != mod, int>, + typename... Args> +constexpr inline auto MakeModuleHandlerTuple(IdT id, FuncT f, Args... rest) +{ + return MakeModuleHandlerTuple(rest...); +} + +//--- MakeModuleHandlerFromTuple --- +// Helper function for MakeModuleHandlerTable. Coverts type of Handlers into +// a ModuleHandlerTable object. This approach was simpler than building up the +// ModuleHandlerTable incrementally, since std::tuple_cat makes it so easy to +// build up a tuple. +template +constexpr inline auto MakeModuleHandlerFromTuple(std::tuple...> t) +{ + return ModuleHandlerTable(t); +} + +//--- "Make" function for ModuleHandlerTable to automatically deduce type --- +// First, create a tuple of just the handlers in the argument list in module "mod". +// Uses the mutually-recursive MakeModuleHandlerTuple overloads, which only add +// handlers into the tuple if the module matches. Then, MakeModuleHandlerFromTuple +// converts the tuple into a properly-typed ModuleHandlerTable object. +template +constexpr inline auto MakeModuleHandlerTable(Args... args) +{ + const auto handlerTuple = MakeModuleHandlerTuple(args...); + return MakeModuleHandlerFromTuple(handlerTuple); +} + +} // namespace detail_nvtx + +//============ NVTX injection helper public interface ========================= + +// Define sentinel-value constants for use in handler implementations +namespace ReturnCodes { + constexpr auto NVTX_TOOL_ATTACHED_UNUSED_RANGE_ID = static_cast(-1LL); + constexpr int NVTX_TOOL_ATTACHED_UNUSED_PUSH_POP_ID = -1; + const auto NVTX_TOOL_ATTACHED_UNUSED_DOMAIN_HANDLE = reinterpret_cast(-1LL); + const auto NVTX_TOOL_ATTACHED_UNUSED_STRING_HANDLE = reinterpret_cast(-1LL); + // Note: In C++20, use bit_cast instead of reinterpret_cast, so the handles + // (which are pointer types) can also be made constexpr. +} + +template +constexpr inline auto MakeHandlerTable(Args... args) +{ + return std::make_tuple( + MakeModuleHandlerTable(args...), + MakeModuleHandlerTable(args...), + MakeModuleHandlerTable(args...), + MakeModuleHandlerTable(args...), + MakeModuleHandlerTable(args...), + MakeModuleHandlerTable(args...) + ); +} + +enum class InstallResult +{ + Success, + ExportTableVersionInfoMissing, + ExportTableVersionInfoTooSmall, + ClientVersionTooOld, + ExportTableCallbacksMissing, + ExportTableCallbacksTooSmall, + ModuleNotSupported, + ModuleTableTooSmall +}; + +template +inline InstallResult InstallHandlers( + NvtxGetExportTableFunc_t getExportTable, + HandlerTableT const& injectionHandlerTable, + std::ostringstream* errStream = nullptr, + uint32_t* pVersion = nullptr) +{ + uint32_t version = 0; + auto pVersionInfo = + reinterpret_cast(getExportTable(NVTX_ETID_VERSIONINFO)); + if (!pVersionInfo) + { + if (errStream) *errStream + << "Client NVTX instance doesn't support NVTX_ETID_VERSIONINFO"; + return InstallResult::ExportTableVersionInfoMissing; + } + + if (pVersionInfo->struct_size < sizeof(*pVersionInfo)) + { + if (errStream) *errStream + << "NvtxExportTableVersionInfo structure size is " << pVersionInfo->struct_size + << ", expected " << sizeof(*pVersionInfo) << "!"; + return InstallResult::ExportTableVersionInfoTooSmall; + } + + version = pVersionInfo->version; + if (version < 2) + { + if (errStream) *errStream + << "client's NVTX version is " << version << ", expected 2+"; + return InstallResult::ClientVersionTooOld; + } + + if (pVersion) *pVersion = version; + + auto pCallbacks = + reinterpret_cast(getExportTable(NVTX_ETID_CALLBACKS)); + if (!pCallbacks) + { + if (errStream) *errStream + << "Client NVTX instance doesn't support NVTX_ETID_CALLBACKS"; + return InstallResult::ExportTableCallbacksMissing; + } + + if (pCallbacks->struct_size < sizeof(*pCallbacks)) + { + if (errStream) *errStream + << "NvtxExportTableCallbacks structure size is " << pCallbacks->struct_size + << ", expected " << sizeof(*pCallbacks) << "!"; + return InstallResult::ExportTableCallbacksTooSmall; + } + +#if defined(DEBUG) || true + // Simple loop to print handler table internal details + for_each_in_tuple(injectionHandlerTable, + [](auto const& handlerModule) + { + auto count = size_of_tuple(handlerModule.handlers); + printf("Module: %d Count: %d Highest: %d\n", + (int)handlerModule.moduleId, (int)count, (int)handlerModule.highestIdUsed); + + if (count > 0) + { + for_each_in_tuple(handlerModule.handlers, + [](auto const& handler) + { + auto addr = (long long)handler.Address(); + printf(" Id: %d Address: 0x%llx\n", + (int)handler.id, addr); + } + ); + } + } + ); +#endif + + // Loop over module handler tables and install handlers into client + bool errors = false; + for_each_in_tuple(injectionHandlerTable, + [&](auto const& handlerModule) + { + NvtxFunctionTable clientTable = 0; + unsigned int clientTableSize = 0; + int success; + + if (handlerModule.moduleId == NVTX_CB_MODULE_INVALID) return; + + success = pCallbacks->GetModuleFunctionTable(handlerModule.moduleId, &clientTable, &clientTableSize); + if (!success || !clientTable) + { + if (errStream) *errStream + << "Client NVTX instance doesn't support callback module with id " << handlerModule.moduleId; + // TODO: return InstallResult::ModuleNotSupported; + errors = true; + } + + // Ensure client's table is new enough to support the function pointers we want to register + if (clientTableSize <= handlerModule.highestIdUsed) + { + if (errStream) *errStream + << "Size of client NVTX instance's handler table with module id " << handlerModule.moduleId + << " too small. Size is " << clientTableSize + << ", but injection needs to assign table[" << handlerModule.highestIdUsed << "]"; + // TODO: return InstallResult::ModuleTableTooSmall; + errors = true; + } + + handlerModule.AssignToClient(clientTable); + } + ); + + if (errors) return InstallResult::ModuleNotSupported; + + return InstallResult::Success; +} + +} // namespace NvtxInjectionHelper diff --git a/tests/LinkerDupesFileA.cpp b/tests/LinkerDupesFileA.cpp new file mode 100644 index 0000000..b43deba --- /dev/null +++ b/tests/LinkerDupesFileA.cpp @@ -0,0 +1,6 @@ +#include "TestCoverage.h" + +void FileA(int argc, const char** argv) +{ + RunTestCommon(argc, argv); +} diff --git a/tests/LinkerDupesFileB.cpp b/tests/LinkerDupesFileB.cpp new file mode 100644 index 0000000..f6b7268 --- /dev/null +++ b/tests/LinkerDupesFileB.cpp @@ -0,0 +1,6 @@ +#include "TestCoverage.h" + +void FileB(int argc, const char** argv) +{ + RunTestCommon(argc, argv); +} diff --git a/tests/LinkerDupesMain.cpp b/tests/LinkerDupesMain.cpp new file mode 100644 index 0000000..c938533 --- /dev/null +++ b/tests/LinkerDupesMain.cpp @@ -0,0 +1,13 @@ +void FileA(int argc, const char** argv); +void FileB(int argc, const char** argv); + +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + FileA(argc, argv); + FileB(argc, argv); + + return 0; +} diff --git a/tests/NamedCategories.cpp b/tests/NamedCategories.cpp new file mode 100644 index 0000000..202c526 --- /dev/null +++ b/tests/NamedCategories.cpp @@ -0,0 +1,128 @@ +#if defined(_MSC_VER) && _MSC_VER < 1914 +#define STATIC_ASSERT_TESTING 0 +#else +#define STATIC_ASSERT_TESTING 1 +#endif + +#if defined(STATIC_ASSERT_TESTING) +#include +#define NVTX3_STATIC_ASSERT(c, m) do { if (!(c)) printf("static_assert would fail: %s\n", m); } while (0) +#endif + +#include + +#include + +// Domain description types +struct d { static constexpr const char* name{"Test domain"}; }; + +// Named category types +struct cat_char_test { static constexpr const char* name{"Cat char"}; static constexpr uint32_t id{1}; }; +struct cat_wchar_test { static constexpr const wchar_t* name{L"Cat wchar_t"}; static constexpr uint32_t id{2}; }; +struct error_name_missing { static constexpr const char* x {"Name"}; static constexpr uint32_t id{3}; }; +struct error_name_is_bad_type { static constexpr const int name{5}; static constexpr uint32_t id{4}; }; +struct error_id_missing { static constexpr const char* name{"Name"}; static constexpr uint32_t y {5}; }; +struct error_id_is_bad_type { static constexpr const char* name{"Name"}; static constexpr float id{6}; }; +struct error_both_missing { static constexpr const char* x {"Name"}; static constexpr uint32_t y {7}; }; +struct error_both_bad_type { static constexpr const int name{5}; static constexpr float id{8}; }; +struct error_no_name_bad_id { static constexpr const char* x {"Name"}; static constexpr float id{9}; }; +struct error_bad_name_no_id { static constexpr const int name{5}; static constexpr uint32_t y {10}; }; +struct cat_global_domain1 { static constexpr const char* name{"Global1"}; static constexpr uint32_t id{11}; }; +struct cat_global_domain2 { static constexpr const char* name{"Global2"}; static constexpr uint32_t id{12}; }; +struct cat_global_domain3 { static constexpr const char* name{"Global3"}; static constexpr uint32_t id{13}; }; + +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + using namespace nvtx3; + + auto& d1 = domain::get(); + +#if 1 + std::cout << "- Named category (char):\n"; + auto& c1 = named_category_in::get(); + mark_in("Mark in cat_char_test category", named_category_in::get()); + + std::cout << "- Named category (wchar_t):\n"; + auto& c2 = named_category_in::get(); + mark_in("Mark in cat_wchar_test category", named_category_in::get()); +#endif + +#if 1 + std::cout << "- Named category in global domain (alias):\n"; + auto& cd1 = named_category::get(); + + std::cout << "- Named category in global domain (implicit):\n"; + auto& cd2 = named_category_in<>::get(); + + std::cout << "- Named category in global domain (explicit):\n"; + auto& cd3 = named_category_in::get(); +#endif + +#if STATIC_ASSERT_TESTING + +#if 1 // defined(ERROR_TEST_NAME_IS_MISSING) + { + std::cout << "- Error test - category is missing name member:\n"; + auto& c3 = named_category_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_NAME_IS_BAD_TYPE) + { + std::cout << "- Error test - category name member isn't narrow or wide char array:\n"; + auto& c4 = named_category_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_ID_IS_MISSING) + { + std::cout << "- Error test - category is missing id member:\n"; + auto& c5 = named_category_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_ID_IS_BAD_TYPE) + { + std::cout << "- Error test - category id member isn't uint32_t:\n"; + auto& c6 = named_category_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_BOTH_MISSING) + { + std::cout << "- Error test - category is missing both members:\n"; + auto& c7 = named_category_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_BOTH_BAD_TYPE) + { + std::cout << "- Error test - category members are both bad types:\n"; + auto& c8 = named_category_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_NO_NAME_BAD_ID) + { + std::cout << "- Error test - category has no name and bad id type:\n"; + auto& c9 = named_category_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_BAD_NAME_NO_ID) + { + std::cout << "- Error test - category has bad name type and no id:\n"; + auto& c10 = named_category_in::get(); + } +#endif + +#endif // STATIC_ASSERT_TESTING + + return 0; +} diff --git a/tests/PrettyPrintersNvtxC.h b/tests/PrettyPrintersNvtxC.h new file mode 100644 index 0000000..dc99498 --- /dev/null +++ b/tests/PrettyPrintersNvtxC.h @@ -0,0 +1,140 @@ +#pragma once +#include +#include + +// Pretty-printers for color, payload, and message discriminated-union types + +inline void WriteColorType(std::ostream& os, nvtxColorType_t t) +{ + switch (t) + { + case NVTX_COLOR_ARGB : os << "NVTX_COLOR_ARGB"; break; + case NVTX_COLOR_UNKNOWN: os << ""; break; + default : os << ""; + } +} + +inline std::ostream& operator<<(std::ostream& os, nvtxColorType_t t) +{ + WriteColorType(os, t); + return os; +} + +inline void WritePayloadType(std::ostream& os, nvtxPayloadType_t t) +{ + switch (t) + { + case NVTX_PAYLOAD_TYPE_UNSIGNED_INT64: os << "NVTX_PAYLOAD_TYPE_UNSIGNED_INT64"; break; + case NVTX_PAYLOAD_TYPE_INT64 : os << "NVTX_PAYLOAD_TYPE_INT64 "; break; + case NVTX_PAYLOAD_TYPE_DOUBLE : os << "NVTX_PAYLOAD_TYPE_DOUBLE "; break; + case NVTX_PAYLOAD_TYPE_UNSIGNED_INT32: os << "NVTX_PAYLOAD_TYPE_UNSIGNED_INT32"; break; + case NVTX_PAYLOAD_TYPE_INT32 : os << "NVTX_PAYLOAD_TYPE_INT32 "; break; + case NVTX_PAYLOAD_TYPE_FLOAT : os << "NVTX_PAYLOAD_TYPE_FLOAT "; break; + case NVTX_PAYLOAD_UNKNOWN : os << ""; break; + default : os << ""; + } +} + +inline void WritePayloadValue(std::ostream& os, nvtxPayloadType_t t, nvtxEventAttributes_v2::payload_t val) +{ + switch (t) + { + case NVTX_PAYLOAD_TYPE_UNSIGNED_INT64: os << val.ullValue; break; + case NVTX_PAYLOAD_TYPE_INT64 : os << val.llValue; break; + case NVTX_PAYLOAD_TYPE_DOUBLE : os << val.dValue; break; + case NVTX_PAYLOAD_TYPE_UNSIGNED_INT32: os << val.uiValue; break; + case NVTX_PAYLOAD_TYPE_INT32 : os << val.iValue; break; + case NVTX_PAYLOAD_TYPE_FLOAT : os << val.fValue; break; + case NVTX_PAYLOAD_UNKNOWN : os << ""; break; + default : os << ""; + } +} + +inline void WritePayload(std::ostream& os, nvtxPayloadType_t t, nvtxEventAttributes_v2::payload_t val) +{ + WritePayloadType(os, t); + os << " = "; + WritePayloadValue(os, t, val); +} + +inline std::ostream& operator<<(std::ostream& os, nvtxPayloadType_t t) +{ + WritePayloadType(os, t); + return os; +} + +inline void WriteMessageType(std::ostream& os, nvtxMessageType_t t) +{ + switch (t) + { + case NVTX_MESSAGE_TYPE_ASCII : os << "NVTX_MESSAGE_TYPE_ASCII"; break; + case NVTX_MESSAGE_TYPE_UNICODE : os << "NVTX_MESSAGE_TYPE_UNICODE"; break; + case NVTX_MESSAGE_TYPE_REGISTERED: os << "NVTX_MESSAGE_TYPE_REGISTERED"; break; + case NVTX_MESSAGE_UNKNOWN : os << ""; break; + default : os << ""; + } +} + +inline void WriteMessageValue(std::ostream& os, nvtxMessageType_t t, nvtxMessageValue_t val) +{ + switch (t) + { + case NVTX_MESSAGE_TYPE_ASCII : os << val.ascii; break; + case NVTX_MESSAGE_TYPE_UNICODE : os << ""; break; + case NVTX_MESSAGE_TYPE_REGISTERED: os << "Registered handle: " << (void*)val.registered; break; + case NVTX_MESSAGE_UNKNOWN : os << ""; break; + default : os << ""; + } +} + +inline void WriteMessage(std::ostream& os, nvtxMessageType_t t, nvtxMessageValue_t val) +{ + WriteMessageType(os, t); + os << " = "; + WriteMessageValue(os, t, val); +} + +inline std::ostream& operator<<(std::ostream& os, nvtxMessageType_t t) +{ + WriteMessageType(os, t); + return os; +} + +// Pretty-printer for attributes struct + +#if 1 +inline std::ostream& operator<<(std::ostream& os, nvtxEventAttributes_t const& a) +{ + os << "{ver: " << a.version + << ", size: " << a.size + << ", category: " << a.category + << ", color: " << (nvtxColorType_t)a.colorType << " 0x" << std::hex << a.color << std::dec + << ", payload: " << (nvtxPayloadType_t)a.payloadType << " "; + WritePayloadValue(os, (nvtxPayloadType_t)a.payloadType, a.payload); + os << ", message: " << (nvtxMessageType_t)a.messageType << " \""; + WriteMessageValue(os, (nvtxMessageType_t)a.messageType, a.message); + os << "\"}"; + + return os; +} +#else +inline std::ostream& operator<<(std::ostream& os, nvtxEventAttributes_t const& a) +{ + os + << "uint16_t version = " << a.version << "\n" + << "uint16_t size = " << a.size << "\n" + << "uint32_t category = " << a.category << "\n" + << "int32_t colorType = " << (nvtxColorType_t)a.colorType << "\n" + << "uint32_t color = 0x" << std::hex << a.color << std::dec << "\n" + << "int32_t payloadType = " << (nvtxPayloadType_t)a.payloadType << "\n" + << "(union) payload = "; + WritePayloadValue(os, (nvtxPayloadType_t)a.payloadType, a.payload); + os << "\n" + << "int32_t messageType = " << (nvtxMessageType_t)a.messageType << "\n" + << "(union) message = "; + WriteMessageValue(os, (nvtxMessageType_t)a.messageType, a.message); + os << "\n"; + + return os; +} +#endif diff --git a/tests/PrettyPrintersNvtxCpp.h b/tests/PrettyPrintersNvtxCpp.h new file mode 100644 index 0000000..6ab8d87 --- /dev/null +++ b/tests/PrettyPrintersNvtxCpp.h @@ -0,0 +1,20 @@ +#pragma once +#include "PrettyPrintersNvtxC.h" +#include + +inline std::ostream& operator<<(std::ostream& os, nvtx3::event_attributes const& attr) +{ + return os << *attr.get(); +} + +inline std::ostream& operator<<(std::ostream& os, nvtx3::payload const& p) +{ + WritePayload(os, p.get_type(), p.get_value()); + return os; +} + +inline std::ostream& operator<<(std::ostream& os, nvtx3::message const& m) +{ + WriteMessage(os, m.get_type(), m.get_value()); + return os; +} diff --git a/tests/PrintInjection.cpp b/tests/PrintInjection.cpp new file mode 100644 index 0000000..4cc87a8 --- /dev/null +++ b/tests/PrintInjection.cpp @@ -0,0 +1,198 @@ +#define NVTX_NO_IMPL +#include "nvtx3/nvToolsExt.h" + +#include "DllHelper.h" +#include + + +#if defined(NVTX_INJECTION_TEST_QUIET) +#define LOG_INFO(...) +#define LOG_ERROR(...) +#else +#define LOG_INFO(...) fprintf(stderr, "[inj] " __VA_ARGS__) +#define LOG_ERROR(...) fprintf(stderr, "[inj] ERROR: " __VA_ARGS__) +#endif + +/* Implementations of NVTX functions to attach to client */ + +#define NVTX_TOOL_ATTACHED_UNUSED_RANGE_ID (nvtxRangeId_t)(-1LL) +#define NVTX_TOOL_ATTACHED_UNUSED_PUSH_POP_ID (int)(-1) +#define NVTX_TOOL_ATTACHED_UNUSED_DOMAIN_HANDLE (nvtxDomainHandle_t)(-1LL) +#define NVTX_TOOL_ATTACHED_UNUSED_STRING_HANDLE (nvtxStringHandle_t)(-1LL) + +/* NVTX_CB_MODULE_CORE */ + +static void NVTX_API HandleMarkA(const char* str) +{ + LOG_INFO("%s\n", "nvtxMarkA"); +} + +static int NVTX_API HandleRangePushA(const char* str) +{ + LOG_INFO("%s\n", "nvtxRangePushA"); + return NVTX_TOOL_ATTACHED_UNUSED_PUSH_POP_ID; +} + +static int NVTX_API HandleRangePop() +{ + LOG_INFO("%s\n", "nvtxRangePop"); + return NVTX_TOOL_ATTACHED_UNUSED_PUSH_POP_ID; +} + +/* NVTX_CB_MODULE_CORE2 */ + +static void NVTX_API HandleDomainMarkEx(nvtxDomainHandle_t domain, const nvtxEventAttributes_t* eventAttrib) +{ + LOG_INFO("%s\n", "nvtxDomainMarkEx"); +} + +static nvtxRangeId_t NVTX_API HandleDomainRangeStartEx(nvtxDomainHandle_t domain, const nvtxEventAttributes_t* eventAttrib) +{ + LOG_INFO("%s\n", "nvtxDomainRangeStartEx"); + return NVTX_TOOL_ATTACHED_UNUSED_RANGE_ID; +} + +static void NVTX_API HandleDomainRangeEnd(nvtxDomainHandle_t domain, nvtxRangeId_t id) +{ + LOG_INFO("%s\n", "nvtxDomainRangeEnd"); +} + +static int NVTX_API HandleDomainRangePushEx(nvtxDomainHandle_t domain, const nvtxEventAttributes_t* eventAttrib) +{ + LOG_INFO("%s\n", "nvtxDomainRangePushEx"); + return NVTX_TOOL_ATTACHED_UNUSED_PUSH_POP_ID; +} + +static int NVTX_API HandleDomainRangePop(nvtxDomainHandle_t domain) +{ + LOG_INFO("%s\n", "nvtxDomainRangePop"); + return NVTX_TOOL_ATTACHED_UNUSED_PUSH_POP_ID; +} + +static nvtxStringHandle_t NVTX_API HandleDomainRegisterStringA(nvtxDomainHandle_t domain, const char* string) +{ + LOG_INFO("%s\n", "nvtxDomainRegisterStringA"); + return NVTX_TOOL_ATTACHED_UNUSED_STRING_HANDLE; +} + +static nvtxDomainHandle_t NVTX_API HandleDomainCreateA(const char* name) +{ + LOG_INFO("%s\n", "nvtxDomainCreateA"); + return NVTX_TOOL_ATTACHED_UNUSED_DOMAIN_HANDLE; +} + +static void NVTX_API HandleDomainDestroy(nvtxDomainHandle_t domain) +{ + LOG_INFO("%s\n", "nvtxDomainDestroy"); +} + +static void NVTX_API HandleInitialize(const void* reserved) +{ + LOG_INFO("%s\n", "nvtxInitialize"); +} + +#define ADD_TO_TABLE(mod, name) *table[NVTX_CBID_##mod##_##name] = (NvtxFunctionPointer)Handle##name; + +extern "C" DLL_EXPORT +int NVTX_API InitializeInjectionNvtx2(NvtxGetExportTableFunc_t getExportTable) +{ + uint32_t version = 0; + const NvtxExportTableVersionInfo* pVersionInfo = + (const NvtxExportTableVersionInfo*)getExportTable(NVTX_ETID_VERSIONINFO); + if (pVersionInfo) + { + if (pVersionInfo->struct_size < sizeof(*pVersionInfo)) + { + LOG_ERROR( + "(init v2) NvtxExportTableVersionInfo structure size is %d, expected %d!\n", + (int)pVersionInfo->struct_size, + (int)sizeof(*pVersionInfo)); + return 0; + } + + version = pVersionInfo->version; + if (version < 2) + { + LOG_ERROR( + "(init v2) client's NVTX version is %d, expected 2+\n", + (int)version); + return 0; + } + } + + LOG_INFO("---- InitializeInjectionNvtx2 called from client's NVTX v%d\n", version); + + const NvtxExportTableCallbacks* pCallbacks = + (const NvtxExportTableCallbacks*)getExportTable(NVTX_ETID_CALLBACKS); + if (!pCallbacks) + { + LOG_ERROR("(init v2) NVTX_ETID_CALLBACKS is not supported.\n"); + return 0; + } + + if (pCallbacks->struct_size < sizeof(*pCallbacks)) + { + LOG_ERROR("(init v2) NvtxExportTableCallbacks structure size is %d, expected %d!\n", + (int)pCallbacks->struct_size, + (int)sizeof(*pCallbacks)); + return 0; + } + + { + NvtxFunctionTable table = 0; + unsigned int size = 0; + int success = pCallbacks->GetModuleFunctionTable(NVTX_CB_MODULE_CORE, &table, &size); + if (!success || !table) + { + LOG_ERROR("(init v2) NVTX_CB_MODULE_CORE is not supported.\n"); + return 0; + } + + /* Ensure client's table is new enough to support the function pointers we want to register */ + unsigned int highestIdUsed = NVTX_CBID_CORE_RangePop; /* Can auto-detect this in C++ */ + if (size <= highestIdUsed) + { + LOG_ERROR("(init v2) Client's function pointer table size is %d, and we need to assign to table[%d].\n", + (int)size, + (int)highestIdUsed); + return 0; + } + + *table[NVTX_CBID_CORE_MarkA ] = (NvtxFunctionPointer)HandleMarkA ; + *table[NVTX_CBID_CORE_RangePushA] = (NvtxFunctionPointer)HandleRangePushA; + *table[NVTX_CBID_CORE_RangePop ] = (NvtxFunctionPointer)HandleRangePop ; + } + + { + NvtxFunctionTable table = 0; + unsigned int size = 0; + int success = pCallbacks->GetModuleFunctionTable(NVTX_CB_MODULE_CORE2, &table, &size); + if (!success || !table) + { + LOG_ERROR("(init v2) NVTX_CB_MODULE_CORE2 is not supported.\n"); + return 0; + } + + /* Ensure client's table is new enough to support the function pointers we want to register */ + unsigned int highestIdUsed = NVTX_CBID_CORE2_Initialize; /* Can auto-detect this in C++ */ + if (size <= highestIdUsed) + { + LOG_ERROR("(init v2) Client's function pointer table size is %d, and we need to assign to table[%d].\n", + (int)size, + (int)highestIdUsed); + return 0; + } + + *table[NVTX_CBID_CORE2_DomainMarkEx ] = (NvtxFunctionPointer)HandleDomainMarkEx ; + *table[NVTX_CBID_CORE2_DomainRangeStartEx ] = (NvtxFunctionPointer)HandleDomainRangeStartEx ; + *table[NVTX_CBID_CORE2_DomainRangeEnd ] = (NvtxFunctionPointer)HandleDomainRangeEnd ; + *table[NVTX_CBID_CORE2_DomainRangePushEx ] = (NvtxFunctionPointer)HandleDomainRangePushEx ; + *table[NVTX_CBID_CORE2_DomainRangePop ] = (NvtxFunctionPointer)HandleDomainRangePop ; + *table[NVTX_CBID_CORE2_DomainRegisterStringA] = (NvtxFunctionPointer)HandleDomainRegisterStringA; + *table[NVTX_CBID_CORE2_DomainCreateA ] = (NvtxFunctionPointer)HandleDomainCreateA ; + *table[NVTX_CBID_CORE2_DomainDestroy ] = (NvtxFunctionPointer)HandleDomainDestroy ; + *table[NVTX_CBID_CORE2_Initialize ] = (NvtxFunctionPointer)HandleInitialize ; + } + + return 1; +} diff --git a/tests/README.txt b/tests/README.txt new file mode 100644 index 0000000..119c90f --- /dev/null +++ b/tests/README.txt @@ -0,0 +1,37 @@ +This test suite builds an executable called "runtest" and several dynamic libraries. +The dynamic libraries serve as tests, NVTX injections, or both. To invoke "runtest", +specify the desired test library using -t and the desired injection library using -i. +For example: + runtest -t coverage -i inj +...will run the "coverage" test using "inj" for the injection. The tests can be run +without any injection to see what they do without NVTX. Some test libraries are self- +injecting, i.e. they serve as both the test and the injection. Specifying "-i -" for +the injection option instructs runtest to use the test library as the injection also. +Additional arguments are forwarded to the test. For example: + runtest -t calls -i - -v +...will use the "calls" library for both the test and the injection, and will pass +the "-v" argument to the calls test. + +Some tests include compile-time negative tests, guarded by #ifs. By defining macros +like ERROR_TEST_NAME_IS_MISSING, the tests should fail to compile with the expected +error message. In these cases, successful compilation or emitting the wrong error +message should be considered a failure of the test. + +Here are the dynamic libraries, and X in columns to show which usage they support: + + Test? Injection? Description +attributes X - Use NVTX C++ API for setting event attributes +calls X X - Use self-injection to test C/C++ APIs call handlers with all parameters correctly +categories X - Use NVTX C++ API for naming categories +coverage X - Use all features of NVTX C++ API +coveragec X - Use all features of NVTX C API +coverage-counter X - Use all features of NVTX C API Extension for Counters +coverage-cu X - Use all features of NVTX C++ API from a .cu file (i.e. use nvcc instead of host cc) +coverage-mem X - Use all features of NVTX C API Extension for Memory Naming +coverage-memcudart X - Use all features of NVTX C API Extension for Memory Naming (using CUDART types) +coverage-payload X - Use all features of NVTX C API Extension for Payloads +domains X - Use NVTX C++ API for creating domains +inj X - A simple injection that prints messages when NVTX functions are called +linkerdupes X - Compile-time tests to ensure multiple libraries using NVTX don't conflict +regstrings X - Use NVTX C++ API for registering strings +self X X - Use self-injection to demonstrate the self-injection mechanism is working diff --git a/tests/RegisteredStrings.cpp b/tests/RegisteredStrings.cpp new file mode 100644 index 0000000..63867cc --- /dev/null +++ b/tests/RegisteredStrings.cpp @@ -0,0 +1,80 @@ +#if defined(_MSC_VER) && _MSC_VER < 1914 +#define STATIC_ASSERT_TESTING 0 +#else +#define STATIC_ASSERT_TESTING 1 +#endif + +#if defined(STATIC_ASSERT_TESTING) +#include +#define NVTX3_STATIC_ASSERT(c, m) do { if (!(c)) printf("static_assert would fail: %s\n", m); } while (0) +#endif + +#include + +#include + +// Domain description types +struct d { static constexpr const char* name{"Test domain"}; }; + +// Registered string types +struct regstr_char_test { static constexpr const char* message{"Reg str char"}; }; +struct regstr_wchar_test { static constexpr const wchar_t* message{L"Reg str wchar_t"}; }; +struct error_msg_missing { static constexpr const char* x {"Name"}; }; +struct error_msg_is_bad_type { static constexpr const int message{5}; }; +struct regstr_global_domain1 { static constexpr const char* message{"Global1"}; }; +struct regstr_global_domain2 { static constexpr const char* message{"Global2"}; }; +struct regstr_global_domain3 { static constexpr const char* message{"Global3"}; }; + +#include "DllHelper.h" + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + (void)argc; + (void)argv; + + using namespace nvtx3; + + auto& d1 = domain::get(); + +#if 1 + std::cout << "- Registered string (char):\n"; + auto& r1 = registered_string_in::get(); + mark_in("Mark in regstr_char_test category", registered_string_in::get()); + + std::cout << "- Registered string (wchar_t):\n"; + auto& r2 = registered_string_in::get(); + mark_in("Mark in regstr_wchar_test category", registered_string_in::get()); +#endif + +#if 1 + std::cout << "- Registered string in global domain (alias):\n"; + auto& rd1 = registered_string::get(); + + std::cout << "- Registered string in global domain (implicit):\n"; + auto& rd2 = registered_string_in<>::get(); + + std::cout << "- Registered string in global domain (explicit):\n"; + auto& rd3 = registered_string_in::get(); +#endif + +#if STATIC_ASSERT_TESTING + +#if 1 // defined(ERROR_TEST_MSG_IS_MISSING) + { + std::cout << "- Error test - registered string is missing name member:\n"; + auto& r3 = registered_string_in::get(); + } +#endif + +#if 1 // defined(ERROR_TEST_MSG_IS_BAD_TYPE) + { + std::cout << "- Error test - registered string message member isn't narrow or wide char array:\n"; + auto& r4 = registered_string_in::get(); + } +#endif + +#endif // STATIC_ASSERT_TESTING + + return 0; +} diff --git a/tests/RunTest.cpp b/tests/RunTest.cpp new file mode 100644 index 0000000..9c565b7 --- /dev/null +++ b/tests/RunTest.cpp @@ -0,0 +1,189 @@ +#include "DllHelper.h" +#include +#include +#include +#include +#ifndef _WIN32 +#include +#endif + +#if defined(_WIN32) +constexpr char pathsep = '\\'; +#else +constexpr char pathsep = '/'; +#endif + +bool SetEnvVar(const char* name, const char* value) +{ +#if defined(_WIN32) + auto result = _putenv_s(name, value); +#else + auto result = setenv(name, value, 1); +#endif + return result == 0; +} + +// Adapted from C function in NVTXW implementation +std::string GetCurrentProcessPath() +{ + char* buf; +#if defined(_WIN32) + { + DWORD size = MAX_PATH; + DWORD newSize; + buf = NULL; + while (1) + { + buf = (char*)realloc(buf, size); + newSize = GetModuleFileNameA(NULL, buf, size); + if (newSize < size) break; + size *= 2; + } + } +#elif defined(__QNX__) + { + size_t size = fpathconf(0, _PC_MAX_INPUT); + if (size <= 0) + { + size = 4096; + } + ++size; + buf = (char*)malloc(size); + _cmdname(buf); + } +#else + { + size_t size = 1024; + ssize_t bytesReadSigned; + size_t bytesRead; + const char* linkName = "/proc/self/exe"; + buf = NULL; + while (1) + { + buf = (char*)realloc(buf, size); + bytesReadSigned = readlink(linkName, buf, size); + if (bytesReadSigned < 0) { free(buf); return NULL; } + bytesRead = (size_t)bytesReadSigned; + if (bytesRead < size) break; + size *= 2; + } + buf[bytesRead] = '\0'; + } +#endif + + std::string result; + if (buf) result = buf; + free(buf); + return result; +} + +int MainInternal(int argc, const char** argv) +{ + const std::string testArg("-t"); + const std::string injectionArg("-i"); + std::string test; + std::string injection; + + auto oldArgv = argv; + ++argv; + while (*argv) + { + if (*argv == testArg ) { ++argv; if (*argv) test = *argv; else return 100; } + else if (*argv == injectionArg) { ++argv; if (*argv) injection = *argv; else return 101; } + else break; + ++argv; + } + argc -= (int)(argv - oldArgv); + + printf("RunTest:\n"); + + auto runTestDir = GetCurrentProcessPath(); + runTestDir.resize(runTestDir.find_last_of(pathsep)); + runTestDir += pathsep; + + if (test.empty()) + { + return 103; + } + else + { + test = runTestDir + DLL_PREFIX + test + DLL_SUFFIX; + } + + printf(" - Using test: %s\n", test.c_str()); + + if (!injection.empty()) + { + const char* injectionVar = (sizeof(void*) == 8) + ? "NVTX_INJECTION64_PATH" + : "NVTX_INJECTION32_PATH"; + + if (injection == "-") + { + injection = test; + } + else + { + injection = runTestDir + DLL_PREFIX + injection + DLL_SUFFIX; + } + bool success = SetEnvVar(injectionVar, injection.c_str()); + if (!success) return 102; + } + + printf(" - Using injection: %s\n", injection.empty() ? "" : injection.c_str()); + + auto hDll = LOAD_DLL(test.c_str()); + if (!hDll) return 104; + + using pfnRunTest_t = int(*)(int, const char**); + + auto pfnRunTest = (pfnRunTest_t)GET_DLL_FUNC(hDll, "RunTest"); + if (!pfnRunTest) return 105; + + int result = pfnRunTest(argc, argv); // Forward remaining args + if (result) return result; + + return 0; +} + +int main(int argc, const char** argv) +{ + int result = MainInternal(argc, argv); + if (result == 0) + { + printf("RunTest PASSED\n"); + } + else + { + // For error codes known to this test driver, print useful error descriptions. + // Otherwise, rely on test to print information about errors. + switch (result) + { + case 100: + puts("RunTest: -t requires an argument, the base name of the library to use as a test"); + break; + case 101: + puts("RunTest: -i requires an argument, the base name of the library to use as an injection"); + break; + case 102: + puts("RunTest: Failed to set NVTX injection environment variable"); + break; + case 103: + puts("RunTest: Missing required argument: -t "); + break; + case 104: + puts("RunTest: Test library failed to load"); +#ifndef _WIN32 + printf(" dlerror: %s\n", dlerror()); +#endif + break; + case 105: + puts("RunTest: Test library loaded, but does not export required entry point RunTest"); + break; + } + + printf("RunTest FAILED with return code: %d\n", result); + } + + return result; +} \ No newline at end of file diff --git a/tests/Same.h b/tests/Same.h new file mode 100644 index 0000000..fe47c21 --- /dev/null +++ b/tests/Same.h @@ -0,0 +1,166 @@ +#include +#include +#include +#include +#include +#include + +//----------------------------------------------------------------------------------------------- +// Implementations of "Same" function for various types +// Provides better comparison capabilities than operator== +// - Option for shallow or deep comparision (i.e. pointers vs. what they point at) +// - Option for verbose mode, with a custom ostream to write to +// - Option to specify name string for what is being compared +// - Option for indent depth, so nested comparisons can print unwinding mismatch messages +//----------------------------------------------------------------------------------------------- + +// C++11-compatible SFINAE helpers to choose overloads based on whether a type is complete or not +template struct make_void { typedef void type; }; +template using void_t = typename make_void::type; +template using enable_if = typename std::enable_if::type; +template struct is_complete { static constexpr bool value = false; }; +template struct is_complete> { static constexpr bool value = true; }; + +#define SAME_COMMON_ARGS \ + bool deep = false, bool verbose = false, const char* name = "", std::ostream& oss = std::cout, int depth = 0 + +// Test if two objects are the same. When 'deep' is true, ignore pointer values and only +// compare pointed-to contents, otherwise behave as operator==. When 'verbose' is true, +// print information about differences to 'oss'. The generic overload only works if there's +// an operator== and operator<< defined. +template +inline auto Same(T const& lhs, T const& rhs, SAME_COMMON_ARGS) + -> decltype(lhs == rhs, oss << lhs, bool()) +{ + bool objSame = lhs == rhs; + if (verbose && !objSame) + { + oss << std::string(depth, ' ') << "'" << name << "' different: values are " + << lhs << " and " << rhs + // << " (type is " << typeid(lhs).name() << ")" + << '\n'; + } + return objSame; +} + +// Generic pointer overload for complete types +template ::value> = 0> +inline bool Same(T* lhs, T* rhs, SAME_COMMON_ARGS) +{ + if (deep) + { + return Same(*lhs, *rhs, deep, verbose, name, oss, depth); + } + else + { + bool ptrSame = lhs == rhs; + if (verbose && !ptrSame) + { + oss << std::string(depth, ' ') << "'" << name << "' different: pointer values are 0x" + << static_cast(lhs) << " and 0x" << static_cast(rhs) << '\n'; + } + return ptrSame; + } +} + +// Generic pointer overload for incomplete types +template ::value> = 0> +inline bool Same(T* lhs, T* rhs, SAME_COMMON_ARGS) +{ + // Don't know how to deep-copy incomplete types, so always compare pointers + bool ptrSame = lhs == rhs; + if (verbose && !ptrSame) + { + oss << std::string(depth, ' ') << "'" << name << "' different: pointer values (to incomplete type) are 0x" + << static_cast(lhs) << " and 0x" << static_cast(rhs) << '\n'; + } + return ptrSame; +} + +// Overloads for smart pointers -- in all cases, forward to contained raw pointer. +// In deep mode the comparison will be on the pointed-at objects, and in non-deep +// mode the comparison will be on the raw pointer values. +template +inline bool Same(std::shared_ptr const& lhs, std::shared_ptr const& rhs, SAME_COMMON_ARGS) +{ + return Same(lhs.get(), rhs.get(), deep, verbose, name, oss, depth); +} + +template +inline bool Same(std::unique_ptr const& lhs, std::unique_ptr const& rhs, SAME_COMMON_ARGS) +{ + return Same(lhs.get(), rhs.get(), deep, verbose, name, oss, depth); +} + +// Overloads for C-style strings (narrow and wide) +inline bool Same(char const* lhs, char const* rhs, SAME_COMMON_ARGS) +{ + if (deep) + { + bool strSame = strcmp(lhs, rhs) == 0; + if (verbose && !strSame) + { + oss << std::string(depth, ' ') << "'" << name << "' different: char strings are \"" + << lhs << "\" and \"" << rhs << "\"\n"; + } + return strSame; + } + else + { + bool ptrSame = lhs == rhs; + if (verbose && !ptrSame) + { + oss << std::string(depth, ' ') << "'" << name << "' different: pointer values are " + << static_cast(lhs) << " and " << static_cast(rhs) << '\n'; + } + return ptrSame; + } +} + +inline bool Same(wchar_t const* lhs, wchar_t const* rhs, SAME_COMMON_ARGS) +{ + if (deep) + { + bool strSame = wcscmp(lhs, rhs) == 0; + if (verbose && !strSame) + { + oss << std::string(depth, ' ') << "'" << name << "' different: wchar_t strings are L\"" + << "" << "\" and L\"" << "" << "\"\n"; + } + return strSame; + } + else + { + bool ptrSame = lhs == rhs; + if (verbose && !ptrSame) + { + oss << std::string(depth, ' ') << "'" << name << "' different: pointer values are " + << static_cast(lhs) << " and " << static_cast(rhs) << '\n'; + } + return ptrSame; + } +} + +// Helper macros to define Same() overloads (and operators == and !=) for struct and tagged union types + +#define MEMBER_SAME(member) Same(lhs.member, rhs.member, deep, verbose, #member, oss, depth + 1) +#define UNION_MEMBER_SAME(tagField, tagValue, member) (lhs.tagField == tagValue && MEMBER_SAME(member)) + +#define VERBOSE_PRINT() if (verbose && !same) oss << std::string(depth, ' ') << "'" << name << "' members different\n" + +#define EQ_SIG(T) inline bool operator==(T const& lhs, T const& rhs) +#define NE_FROM_EQ(T) inline bool operator!=(T const& lhs, T const& rhs) { return !(lhs == rhs); } + +#define DEFINE_EQ_NE_DEEP(T) EQ_SIG(T) { return Same(lhs, rhs, true ); } NE_FROM_EQ(T) +#define DEFINE_EQ_NE_SHALLOW(T) EQ_SIG(T) { return Same(lhs, rhs, false); } NE_FROM_EQ(T) + +#define DEFINE_MEMBER_SAME_1(a) MEMBER_SAME(a) +#define DEFINE_MEMBER_SAME_2(a, b) MEMBER_SAME(a) && DEFINE_MEMBER_SAME_1(b) +#define DEFINE_MEMBER_SAME_3(a, b, c) MEMBER_SAME(a) && DEFINE_MEMBER_SAME_2(b, c) + +#define SAME_SIG(T) inline bool Same(T const& lhs, T const& rhs, SAME_COMMON_ARGS) + +#define DEFINE_SAME_0(T) SAME_SIG(T) { return true; } DEFINE_EQ_NE_DEEP(T) +#define DEFINE_SAME_1(T, a) SAME_SIG(T) { bool same = DEFINE_MEMBER_SAME_1(a); VERBOSE_PRINT(); return same; } DEFINE_EQ_NE_DEEP(T) +#define DEFINE_SAME_2(T, a, b) SAME_SIG(T) { bool same = DEFINE_MEMBER_SAME_2(a, b); VERBOSE_PRINT(); return same; } DEFINE_EQ_NE_DEEP(T) +#define DEFINE_SAME_3(T, a, b, c) SAME_SIG(T) { bool same = DEFINE_MEMBER_SAME_3(a, b, c); VERBOSE_PRINT(); return same; } DEFINE_EQ_NE_DEEP(T) diff --git a/tests/SelfInjection.cpp b/tests/SelfInjection.cpp new file mode 100644 index 0000000..3279d7e --- /dev/null +++ b/tests/SelfInjection.cpp @@ -0,0 +1,178 @@ +#include "SelfInjection.h" +#include "DllHelper.h" +#include + +#if defined(NVTX_INJECTION_TEST_QUIET) +#define LOG_ERROR(...) +#else +#define LOG_ERROR(...) do { fprintf(stderr, " [inj] ERROR: " __VA_ARGS__); } while (0) +#endif + +Callbacks g_callbacks; + +namespace { + +/* NVTX_CB_MODULE_CORE */ +void NVTX_API HandleMarkEx (const nvtxEventAttributes_t* eventAttrib) { g_callbacks.MarkEx (eventAttrib); } +void NVTX_API HandleMarkA (const char* str ) { g_callbacks.MarkA (str ); } +void NVTX_API HandleMarkW (const wchar_t* str ) { g_callbacks.MarkW (str ); } +nvtxRangeId_t NVTX_API HandleRangeStartEx (const nvtxEventAttributes_t* eventAttrib) { return g_callbacks.RangeStartEx (eventAttrib); } +nvtxRangeId_t NVTX_API HandleRangeStartA (const char* str ) { return g_callbacks.RangeStartA (str ); } +nvtxRangeId_t NVTX_API HandleRangeStartW (const wchar_t* str ) { return g_callbacks.RangeStartW (str ); } +void NVTX_API HandleRangeEnd (nvtxRangeId_t id ) { g_callbacks.RangeEnd (id ); } +int NVTX_API HandleRangePushEx (const nvtxEventAttributes_t* eventAttrib) { return g_callbacks.RangePushEx (eventAttrib); } +int NVTX_API HandleRangePushA (const char* str ) { return g_callbacks.RangePushA (str ); } +int NVTX_API HandleRangePushW (const wchar_t* str ) { return g_callbacks.RangePushW (str ); } +int NVTX_API HandleRangePop ( ) { return g_callbacks.RangePop ( ); } +void NVTX_API HandleNameCategoryA(uint32_t id, const char* str ) { g_callbacks.NameCategoryA(id, str ); } +void NVTX_API HandleNameCategoryW(uint32_t id, const wchar_t* str ) { g_callbacks.NameCategoryW(id, str ); } +void NVTX_API HandleNameOsThreadA(uint32_t id, const char* str ) { g_callbacks.NameOsThreadA(id, str ); } +void NVTX_API HandleNameOsThreadW(uint32_t id, const wchar_t* str ) { g_callbacks.NameOsThreadW(id, str ); } + +/* NVTX_CB_MODULE_CORE2 */ +void NVTX_API HandleDomainMarkEx (nvtxDomainHandle_t domain, const nvtxEventAttributes_t* eventAttrib) { g_callbacks.DomainMarkEx (domain, eventAttrib); } +nvtxRangeId_t NVTX_API HandleDomainRangeStartEx (nvtxDomainHandle_t domain, const nvtxEventAttributes_t* eventAttrib) { return g_callbacks.DomainRangeStartEx (domain, eventAttrib); } +void NVTX_API HandleDomainRangeEnd (nvtxDomainHandle_t domain, nvtxRangeId_t id ) { g_callbacks.DomainRangeEnd (domain, id ); } +int NVTX_API HandleDomainRangePushEx (nvtxDomainHandle_t domain, const nvtxEventAttributes_t* eventAttrib) { return g_callbacks.DomainRangePushEx (domain, eventAttrib); } +int NVTX_API HandleDomainRangePop (nvtxDomainHandle_t domain ) { return g_callbacks.DomainRangePop (domain ); } +nvtxResourceHandle_t NVTX_API HandleDomainResourceCreate (nvtxDomainHandle_t domain, nvtxResourceAttributes_t* attr ) { return g_callbacks.DomainResourceCreate (domain, attr ); } +void NVTX_API HandleDomainResourceDestroy(nvtxResourceHandle_t attr ) { g_callbacks.DomainResourceDestroy(attr ); } +void NVTX_API HandleDomainNameCategoryA (nvtxDomainHandle_t domain, uint32_t id, const char* str ) { g_callbacks.DomainNameCategoryA (domain, id, str ); } +void NVTX_API HandleDomainNameCategoryW (nvtxDomainHandle_t domain, uint32_t id, const wchar_t* str ) { g_callbacks.DomainNameCategoryW (domain, id, str ); } +nvtxStringHandle_t NVTX_API HandleDomainRegisterStringA(nvtxDomainHandle_t domain, const char* str ) { return g_callbacks.DomainRegisterStringA(domain, str ); } +nvtxStringHandle_t NVTX_API HandleDomainRegisterStringW(nvtxDomainHandle_t domain, const wchar_t* str ) { return g_callbacks.DomainRegisterStringW(domain, str ); } +nvtxDomainHandle_t NVTX_API HandleDomainCreateA (const char* name ) { return g_callbacks.DomainCreateA (name ); } +nvtxDomainHandle_t NVTX_API HandleDomainCreateW (const wchar_t* name ) { return g_callbacks.DomainCreateW (name ); } +void NVTX_API HandleDomainDestroy (nvtxDomainHandle_t domain ) { g_callbacks.DomainDestroy (domain ); } +void NVTX_API HandleInitialize (const void* reserved ) { g_callbacks.Initialize (reserved ); } + +} + +extern "C" DLL_EXPORT +int NVTX_API InitializeInjectionNvtx2(NvtxGetExportTableFunc_t getExportTable) +{ + uint32_t version = 0; + const NvtxExportTableVersionInfo* pVersionInfo = + (const NvtxExportTableVersionInfo*)getExportTable(NVTX_ETID_VERSIONINFO); + if (pVersionInfo) + { + if (pVersionInfo->struct_size < sizeof(*pVersionInfo)) + { + LOG_ERROR( + "(init v2) NvtxExportTableVersionInfo structure size is %d, expected %d!\n", + (int)pVersionInfo->struct_size, + (int)sizeof(*pVersionInfo)); + g_callbacks.Load(0); + return 0; + } + + version = pVersionInfo->version; + if (version < 2) + { + LOG_ERROR( + "(init v2) client's NVTX version is %d, expected 2+\n", + (int)version); + g_callbacks.Load(0); + return 0; + } + } + + const NvtxExportTableCallbacks* pCallbacks = + (const NvtxExportTableCallbacks*)getExportTable(NVTX_ETID_CALLBACKS); + if (!pCallbacks) + { + LOG_ERROR("(init v2) NVTX_ETID_CALLBACKS is not supported.\n"); + g_callbacks.Load(0); + return 0; + } + + if (pCallbacks->struct_size < sizeof(*pCallbacks)) + { + LOG_ERROR("(init v2) NvtxExportTableCallbacks structure size is %d, expected %d!\n", + (int)pCallbacks->struct_size, + (int)sizeof(*pCallbacks)); + g_callbacks.Load(0); + return 0; + } + + { + NvtxFunctionTable table = 0; + unsigned int size = 0; + int success = pCallbacks->GetModuleFunctionTable(NVTX_CB_MODULE_CORE, &table, &size); + if (!success || !table) + { + LOG_ERROR("(init v2) NVTX_CB_MODULE_CORE is not supported.\n"); + g_callbacks.Load(0); + return 0; + } + + /* Ensure client's table is new enough to support the function pointers we want to register */ + unsigned int highestIdUsed = NVTX_CBID_CORE_RangePop; /* Can auto-detect this in C++ */ + if (size <= highestIdUsed) + { + LOG_ERROR("(init v2) Client's function pointer table size is %d, and we need to assign to table[%d].\n", + (int)size, + (int)highestIdUsed); + g_callbacks.Load(0); + return 0; + } + + *table[NVTX_CBID_CORE_MarkEx ] = (NvtxFunctionPointer)HandleMarkEx ; + *table[NVTX_CBID_CORE_MarkA ] = (NvtxFunctionPointer)HandleMarkA ; + *table[NVTX_CBID_CORE_MarkW ] = (NvtxFunctionPointer)HandleMarkW ; + *table[NVTX_CBID_CORE_RangeStartEx ] = (NvtxFunctionPointer)HandleRangeStartEx ; + *table[NVTX_CBID_CORE_RangeStartA ] = (NvtxFunctionPointer)HandleRangeStartA ; + *table[NVTX_CBID_CORE_RangeStartW ] = (NvtxFunctionPointer)HandleRangeStartW ; + *table[NVTX_CBID_CORE_RangeEnd ] = (NvtxFunctionPointer)HandleRangeEnd ; + *table[NVTX_CBID_CORE_RangePushEx ] = (NvtxFunctionPointer)HandleRangePushEx ; + *table[NVTX_CBID_CORE_RangePushA ] = (NvtxFunctionPointer)HandleRangePushA ; + *table[NVTX_CBID_CORE_RangePushW ] = (NvtxFunctionPointer)HandleRangePushW ; + *table[NVTX_CBID_CORE_RangePop ] = (NvtxFunctionPointer)HandleRangePop ; + *table[NVTX_CBID_CORE_NameCategoryA] = (NvtxFunctionPointer)HandleNameCategoryA; + *table[NVTX_CBID_CORE_NameCategoryW] = (NvtxFunctionPointer)HandleNameCategoryW; + *table[NVTX_CBID_CORE_NameOsThreadA] = (NvtxFunctionPointer)HandleNameOsThreadA; + *table[NVTX_CBID_CORE_NameOsThreadW] = (NvtxFunctionPointer)HandleNameOsThreadW; + } + + { + NvtxFunctionTable table = 0; + unsigned int size = 0; + int success = pCallbacks->GetModuleFunctionTable(NVTX_CB_MODULE_CORE2, &table, &size); + if (!success || !table) + { + LOG_ERROR("(init v2) NVTX_CB_MODULE_CORE2 is not supported.\n"); + g_callbacks.Load(0); + return 0; + } + + /* Ensure client's table is new enough to support the function pointers we want to register */ + unsigned int highestIdUsed = NVTX_CBID_CORE2_Initialize; /* Can auto-detect this in C++ */ + if (size <= highestIdUsed) + { + LOG_ERROR("(init v2) Client's function pointer table size is %d, and we need to assign to table[%d].\n", + (int)size, + (int)highestIdUsed); + g_callbacks.Load(0); + return 0; + } + + *table[NVTX_CBID_CORE2_DomainMarkEx ] = (NvtxFunctionPointer)HandleDomainMarkEx ; + *table[NVTX_CBID_CORE2_DomainRangeStartEx ] = (NvtxFunctionPointer)HandleDomainRangeStartEx ; + *table[NVTX_CBID_CORE2_DomainRangeEnd ] = (NvtxFunctionPointer)HandleDomainRangeEnd ; + *table[NVTX_CBID_CORE2_DomainRangePushEx ] = (NvtxFunctionPointer)HandleDomainRangePushEx ; + *table[NVTX_CBID_CORE2_DomainRangePop ] = (NvtxFunctionPointer)HandleDomainRangePop ; + *table[NVTX_CBID_CORE2_DomainResourceCreate ] = (NvtxFunctionPointer)HandleDomainResourceCreate ; + *table[NVTX_CBID_CORE2_DomainResourceDestroy] = (NvtxFunctionPointer)HandleDomainResourceDestroy; + *table[NVTX_CBID_CORE2_DomainNameCategoryA ] = (NvtxFunctionPointer)HandleDomainNameCategoryA ; + *table[NVTX_CBID_CORE2_DomainNameCategoryW ] = (NvtxFunctionPointer)HandleDomainNameCategoryW ; + *table[NVTX_CBID_CORE2_DomainRegisterStringA] = (NvtxFunctionPointer)HandleDomainRegisterStringA; + *table[NVTX_CBID_CORE2_DomainRegisterStringW] = (NvtxFunctionPointer)HandleDomainRegisterStringW; + *table[NVTX_CBID_CORE2_DomainCreateA ] = (NvtxFunctionPointer)HandleDomainCreateA ; + *table[NVTX_CBID_CORE2_DomainCreateW ] = (NvtxFunctionPointer)HandleDomainCreateW ; + *table[NVTX_CBID_CORE2_DomainDestroy ] = (NvtxFunctionPointer)HandleDomainDestroy ; + *table[NVTX_CBID_CORE2_Initialize ] = (NvtxFunctionPointer)HandleInitialize ; + } + + g_callbacks.Load(1); + return 1; +} diff --git a/tests/SelfInjection.h b/tests/SelfInjection.h new file mode 100644 index 0000000..949b690 --- /dev/null +++ b/tests/SelfInjection.h @@ -0,0 +1,668 @@ +#define NVTX_NO_IMPL +#include "nvtx3/nvToolsExt.h" + +#include "Same.h" +#include "PrettyPrintersNvtxC.h" + +#include +#include +#include +#include +#include +#include + +constexpr auto NVTX_TOOL_ATTACHED_UNUSED_RANGE_ID = static_cast(-1LL); +constexpr int NVTX_TOOL_ATTACHED_UNUSED_PUSH_POP_ID = -1; +const auto NVTX_TOOL_ATTACHED_UNUSED_DOMAIN_HANDLE = reinterpret_cast(-1LL); +const auto NVTX_TOOL_ATTACHED_UNUSED_STRING_HANDLE = reinterpret_cast(-1LL); +const auto NVTX_TOOL_ATTACHED_UNUSED_RESOURCE_HANDLE = reinterpret_cast(-1LL); + +struct ArgsLoad { int success; }; + +struct ArgsMarkEx { const nvtxEventAttributes_t* eventAttrib; }; +struct ArgsMarkA { const char* str ; }; +struct ArgsMarkW { const wchar_t* str ; }; +struct ArgsRangeStartEx { const nvtxEventAttributes_t* eventAttrib; }; +struct ArgsRangeStartA { const char* str ; }; +struct ArgsRangeStartW { const wchar_t* str ; }; +struct ArgsRangeEnd { nvtxRangeId_t id ; }; +struct ArgsRangePushEx { const nvtxEventAttributes_t* eventAttrib; }; +struct ArgsRangePushA { const char* str ; }; +struct ArgsRangePushW { const wchar_t* str ; }; +struct ArgsRangePop { ; }; +struct ArgsNameCategoryA { uint32_t id; const char* str ; }; +struct ArgsNameCategoryW { uint32_t id; const wchar_t* str ; }; +struct ArgsNameOsThreadA { uint32_t id; const char* str ; }; +struct ArgsNameOsThreadW { uint32_t id; const wchar_t* str ; }; + +struct ArgsDomainMarkEx { nvtxDomainHandle_t domain; const nvtxEventAttributes_t* eventAttrib; }; +struct ArgsDomainRangeStartEx { nvtxDomainHandle_t domain; const nvtxEventAttributes_t* eventAttrib; }; +struct ArgsDomainRangeEnd { nvtxDomainHandle_t domain; nvtxRangeId_t id ; }; +struct ArgsDomainRangePushEx { nvtxDomainHandle_t domain; const nvtxEventAttributes_t* eventAttrib; }; +struct ArgsDomainRangePop { nvtxDomainHandle_t domain ; }; +struct ArgsDomainResourceCreate { nvtxDomainHandle_t domain; nvtxResourceAttributes_t* attr ; }; +struct ArgsDomainResourceDestroy { nvtxResourceHandle_t attr ; }; +struct ArgsDomainNameCategoryA { nvtxDomainHandle_t domain; uint32_t id; const char* str ; }; +struct ArgsDomainNameCategoryW { nvtxDomainHandle_t domain; uint32_t id; const wchar_t* str ; }; +struct ArgsDomainRegisterStringA { nvtxDomainHandle_t domain; const char* str ; }; +struct ArgsDomainRegisterStringW { nvtxDomainHandle_t domain; const wchar_t* str ; }; +struct ArgsDomainCreateA { const char* name ; }; +struct ArgsDomainCreateW { const wchar_t* name ; }; +struct ArgsDomainDestroy { nvtxDomainHandle_t domain ; }; +struct ArgsInitialize { const void* reserved ; }; + +struct CallId +{ + NvtxCallbackModule mod; + int32_t cb; +}; +DEFINE_SAME_2(CallId, mod, cb) + +// Helper to write CALLID(CORE, MarkEx) as shorthand for CallId{NVTX_CB_MODULE_CORE, NVTX_CBID_CORE_MarkEx} +#define CALLID(m,c) CallId{NVTX_CB_MODULE_##m, (int32_t)NVTX_CBID_##m##_##c} + +#define CALLID_LOAD() CallId{NVTX_CB_MODULE_INVALID, (int32_t)0x7ac0be11} + +inline const char* CallName(CallId const& id) +{ + if (id == CALLID_LOAD()) return "InitializeInjectionNvtx2"; + switch (id.mod) + { + case NVTX_CB_MODULE_CORE: + switch (id.cb) + { + case NVTX_CBID_CORE_MarkEx : return "MarkEx"; + case NVTX_CBID_CORE_MarkA : return "MarkA"; + case NVTX_CBID_CORE_MarkW : return "MarkW"; + case NVTX_CBID_CORE_RangeStartEx : return "RangeStartEx"; + case NVTX_CBID_CORE_RangeStartA : return "RangeStartA"; + case NVTX_CBID_CORE_RangeStartW : return "RangeStartW"; + case NVTX_CBID_CORE_RangeEnd : return "RangeEnd"; + case NVTX_CBID_CORE_RangePushEx : return "RangePushEx"; + case NVTX_CBID_CORE_RangePushA : return "RangePushA"; + case NVTX_CBID_CORE_RangePushW : return "RangePushW"; + case NVTX_CBID_CORE_RangePop : return "RangePop"; + case NVTX_CBID_CORE_NameCategoryA: return "NameCategoryA"; + case NVTX_CBID_CORE_NameCategoryW: return "NameCategoryW"; + case NVTX_CBID_CORE_NameOsThreadA: return "NameOsThreadA"; + case NVTX_CBID_CORE_NameOsThreadW: return "NameOsThreadW"; + default: return ""; + } + case NVTX_CB_MODULE_CORE2: + switch (id.cb) + { + case NVTX_CBID_CORE2_DomainMarkEx : return "DomainMarkEx"; + case NVTX_CBID_CORE2_DomainRangeStartEx : return "DomainRangeStartEx"; + case NVTX_CBID_CORE2_DomainRangeEnd : return "DomainRangeEnd"; + case NVTX_CBID_CORE2_DomainRangePushEx : return "DomainRangePushEx"; + case NVTX_CBID_CORE2_DomainRangePop : return "DomainRangePop"; + case NVTX_CBID_CORE2_DomainResourceCreate : return "DomainResourceCreate"; + case NVTX_CBID_CORE2_DomainResourceDestroy: return "DomainResourceDestroy"; + case NVTX_CBID_CORE2_DomainNameCategoryA : return "DomainNameCategoryA"; + case NVTX_CBID_CORE2_DomainNameCategoryW : return "DomainNameCategoryW"; + case NVTX_CBID_CORE2_DomainRegisterStringA: return "DomainRegisterStringA"; + case NVTX_CBID_CORE2_DomainRegisterStringW: return "DomainRegisterStringW"; + case NVTX_CBID_CORE2_DomainCreateA : return "DomainCreateA"; + case NVTX_CBID_CORE2_DomainCreateW : return "DomainCreateW"; + case NVTX_CBID_CORE2_DomainDestroy : return "DomainDestroy"; + case NVTX_CBID_CORE2_Initialize : return "Initialize"; + default: return ""; + } + default: return ""; + } +} + +inline std::ostream& operator<<(std::ostream& os, CallId const& id) +{ + return os << CallName(id); +}; + +union Args +{ + ArgsLoad Load; + + ArgsMarkEx MarkEx ; + ArgsMarkA MarkA ; + ArgsMarkW MarkW ; + ArgsRangeStartEx RangeStartEx ; + ArgsRangeStartA RangeStartA ; + ArgsRangeStartW RangeStartW ; + ArgsRangeEnd RangeEnd ; + ArgsRangePushEx RangePushEx ; + ArgsRangePushA RangePushA ; + ArgsRangePushW RangePushW ; + ArgsRangePop RangePop ; + ArgsNameCategoryA NameCategoryA; + ArgsNameCategoryW NameCategoryW; + ArgsNameOsThreadA NameOsThreadA; + ArgsNameOsThreadW NameOsThreadW; + + ArgsDomainMarkEx DomainMarkEx ; + ArgsDomainRangeStartEx DomainRangeStartEx ; + ArgsDomainRangeEnd DomainRangeEnd ; + ArgsDomainRangePushEx DomainRangePushEx ; + ArgsDomainRangePop DomainRangePop ; + ArgsDomainResourceCreate DomainResourceCreate ; + ArgsDomainResourceDestroy DomainResourceDestroy; + ArgsDomainNameCategoryA DomainNameCategoryA ; + ArgsDomainNameCategoryW DomainNameCategoryW ; + ArgsDomainRegisterStringA DomainRegisterStringA; + ArgsDomainRegisterStringW DomainRegisterStringW; + ArgsDomainCreateA DomainCreateA ; + ArgsDomainCreateW DomainCreateW ; + ArgsDomainDestroy DomainDestroy ; + ArgsInitialize Initialize ; +}; + +// Free functions to emulate copy constructors and destructors for the NVTX C API types using pointers +inline void CopyCstring(const char*& lhs, const char* rhs) +{ + size_t len = strlen(rhs) + 1; + auto* tmp = new char[len]; + std::copy(rhs, rhs + len, tmp); + lhs = tmp; +} +inline void CopyCstring(const char*& s) { CopyCstring(s, s); } +inline void DestroyCstring(const char* s) { delete[] s; } + +inline void CopyCstring(const wchar_t*& lhs, const wchar_t* rhs) +{ + size_t len = wcslen(rhs) + 1; + auto* tmp = new wchar_t[len]; + std::copy(rhs, rhs + len, tmp); + lhs = tmp; +} +inline void CopyCstring(const wchar_t*& s) { CopyCstring(s, s); } +inline void DestroyCstring(const wchar_t* s) { delete[] s; } + +inline void CopyEventAttributes(const nvtxEventAttributes_t*& lhs, const nvtxEventAttributes_t* rhs) +{ + auto* tmp = new nvtxEventAttributes_t; + memcpy(tmp, rhs, sizeof(*tmp)); + switch (tmp->messageType) + { + case NVTX_MESSAGE_TYPE_ASCII: CopyCstring(tmp->message.ascii); break; + case NVTX_MESSAGE_TYPE_UNICODE: CopyCstring(tmp->message.unicode); break; + } + lhs = tmp; +} +inline void CopyEventAttributes(const nvtxEventAttributes_t*& a) { CopyEventAttributes(a, a); } +inline void DestroyEventAttributes(const nvtxEventAttributes_t* a) +{ + switch (a->messageType) + { + case NVTX_MESSAGE_TYPE_ASCII: DestroyCstring(a->message.ascii); break; + case NVTX_MESSAGE_TYPE_UNICODE: DestroyCstring(a->message.unicode); break; + } + delete a; +} + +inline void CopyResourceAttributes(nvtxResourceAttributes_t*& lhs, const nvtxResourceAttributes_t* rhs) +{ + auto* tmp = new nvtxResourceAttributes_t; + memcpy(tmp, rhs, sizeof(*tmp)); + switch (tmp->messageType) + { + case NVTX_MESSAGE_TYPE_ASCII: CopyCstring(tmp->message.ascii); break; + case NVTX_MESSAGE_TYPE_UNICODE: CopyCstring(tmp->message.unicode); break; + } + lhs = tmp; +} +inline void CopyResourceAttributes(nvtxResourceAttributes_t*& a) { CopyResourceAttributes(a, a); } +inline void DestroyResourceAttributes(nvtxResourceAttributes_t* a) +{ + switch (a->messageType) + { + case NVTX_MESSAGE_TYPE_ASCII: DestroyCstring(a->message.ascii); break; + case NVTX_MESSAGE_TYPE_UNICODE: DestroyCstring(a->message.unicode); break; + } + delete a; +} + +template inline void DeepCopyAssign(ArgsT& lhs, ArgsT const& rhs) { lhs = rhs; } + +template <> inline void DeepCopyAssign(ArgsMarkEx & lhs, ArgsMarkEx const& rhs) { lhs = rhs; CopyEventAttributes(lhs.eventAttrib); } +template <> inline void DeepCopyAssign(ArgsMarkA & lhs, ArgsMarkA const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsMarkW & lhs, ArgsMarkW const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsRangeStartEx & lhs, ArgsRangeStartEx const& rhs) { lhs = rhs; CopyEventAttributes(lhs.eventAttrib); } +template <> inline void DeepCopyAssign(ArgsRangeStartA & lhs, ArgsRangeStartA const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsRangeStartW & lhs, ArgsRangeStartW const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsRangeEnd & lhs, ArgsRangeEnd const& rhs) { lhs = rhs; } +template <> inline void DeepCopyAssign(ArgsRangePushEx & lhs, ArgsRangePushEx const& rhs) { lhs = rhs; CopyEventAttributes(lhs.eventAttrib); } +template <> inline void DeepCopyAssign(ArgsRangePushA & lhs, ArgsRangePushA const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsRangePushW & lhs, ArgsRangePushW const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsRangePop & lhs, ArgsRangePop const& rhs) { lhs = rhs; } +template <> inline void DeepCopyAssign(ArgsNameCategoryA& lhs, ArgsNameCategoryA const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsNameCategoryW& lhs, ArgsNameCategoryW const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsNameOsThreadA& lhs, ArgsNameOsThreadA const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsNameOsThreadW& lhs, ArgsNameOsThreadW const& rhs) { lhs = rhs; CopyCstring(lhs.str); } + +template <> inline void DeepCopyAssign(ArgsDomainMarkEx & lhs, ArgsDomainMarkEx const& rhs) { lhs = rhs; CopyEventAttributes(lhs.eventAttrib); } +template <> inline void DeepCopyAssign(ArgsDomainRangeStartEx & lhs, ArgsDomainRangeStartEx const& rhs) { lhs = rhs; CopyEventAttributes(lhs.eventAttrib); } +template <> inline void DeepCopyAssign(ArgsDomainRangeEnd & lhs, ArgsDomainRangeEnd const& rhs) { lhs = rhs; } +template <> inline void DeepCopyAssign(ArgsDomainRangePushEx & lhs, ArgsDomainRangePushEx const& rhs) { lhs = rhs; CopyEventAttributes(lhs.eventAttrib); } +template <> inline void DeepCopyAssign(ArgsDomainRangePop & lhs, ArgsDomainRangePop const& rhs) { lhs = rhs; } +template <> inline void DeepCopyAssign(ArgsDomainResourceCreate & lhs, ArgsDomainResourceCreate const& rhs) { lhs = rhs; CopyResourceAttributes(lhs.attr); } +template <> inline void DeepCopyAssign(ArgsDomainResourceDestroy& lhs, ArgsDomainResourceDestroy const& rhs) { lhs = rhs; } +template <> inline void DeepCopyAssign(ArgsDomainNameCategoryA & lhs, ArgsDomainNameCategoryA const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsDomainNameCategoryW & lhs, ArgsDomainNameCategoryW const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsDomainRegisterStringA& lhs, ArgsDomainRegisterStringA const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsDomainRegisterStringW& lhs, ArgsDomainRegisterStringW const& rhs) { lhs = rhs; CopyCstring(lhs.str); } +template <> inline void DeepCopyAssign(ArgsDomainCreateA & lhs, ArgsDomainCreateA const& rhs) { lhs = rhs; CopyCstring(lhs.name); } +template <> inline void DeepCopyAssign(ArgsDomainCreateW & lhs, ArgsDomainCreateW const& rhs) { lhs = rhs; CopyCstring(lhs.name); } +template <> inline void DeepCopyAssign(ArgsDomainDestroy & lhs, ArgsDomainDestroy const& rhs) { lhs = rhs; } +template <> inline void DeepCopyAssign(ArgsInitialize & lhs, ArgsInitialize const& rhs) { lhs = rhs; } + +template inline void DeepCopyDestroy(ArgsT&) {} + +template <> inline void DeepCopyDestroy(ArgsMarkEx & args) { DestroyEventAttributes(args.eventAttrib); } +template <> inline void DeepCopyDestroy(ArgsMarkA & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsMarkW & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsRangeStartEx & args) { DestroyEventAttributes(args.eventAttrib); } +template <> inline void DeepCopyDestroy(ArgsRangeStartA & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsRangeStartW & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsRangeEnd & args) { } +template <> inline void DeepCopyDestroy(ArgsRangePushEx & args) { DestroyEventAttributes(args.eventAttrib); } +template <> inline void DeepCopyDestroy(ArgsRangePushA & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsRangePushW & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsRangePop & args) { } +template <> inline void DeepCopyDestroy(ArgsNameCategoryA& args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsNameCategoryW& args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsNameOsThreadA& args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsNameOsThreadW& args) { DestroyCstring(args.str); } + +template <> inline void DeepCopyDestroy(ArgsDomainMarkEx & args) { DestroyEventAttributes(args.eventAttrib); } +template <> inline void DeepCopyDestroy(ArgsDomainRangeStartEx & args) { DestroyEventAttributes(args.eventAttrib); } +template <> inline void DeepCopyDestroy(ArgsDomainRangeEnd & args) { } +template <> inline void DeepCopyDestroy(ArgsDomainRangePushEx & args) { DestroyEventAttributes(args.eventAttrib); } +template <> inline void DeepCopyDestroy(ArgsDomainRangePop & args) { } +template <> inline void DeepCopyDestroy(ArgsDomainResourceCreate & args) { DestroyResourceAttributes(args.attr); } +template <> inline void DeepCopyDestroy(ArgsDomainResourceDestroy& args) { } +template <> inline void DeepCopyDestroy(ArgsDomainNameCategoryA & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsDomainNameCategoryW & args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsDomainRegisterStringA& args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsDomainRegisterStringW& args) { DestroyCstring(args.str); } +template <> inline void DeepCopyDestroy(ArgsDomainCreateA & args) { DestroyCstring(args.name); } +template <> inline void DeepCopyDestroy(ArgsDomainCreateW & args) { DestroyCstring(args.name); } +template <> inline void DeepCopyDestroy(ArgsDomainDestroy & args) { } +template <> inline void DeepCopyDestroy(ArgsInitialize & args) { } + +struct CallData +{ + CallId id{NVTX_CB_MODULE_INVALID, 0}; + Args args; + + ~CallData() + { + switch (id.mod) + { + case NVTX_CB_MODULE_CORE: + switch (id.cb) + { + case NVTX_CBID_CORE_MarkEx : DeepCopyDestroy(args.MarkEx ); break; + case NVTX_CBID_CORE_MarkA : DeepCopyDestroy(args.MarkA ); break; + case NVTX_CBID_CORE_MarkW : DeepCopyDestroy(args.MarkW ); break; + case NVTX_CBID_CORE_RangeStartEx : DeepCopyDestroy(args.RangeStartEx ); break; + case NVTX_CBID_CORE_RangeStartA : DeepCopyDestroy(args.RangeStartA ); break; + case NVTX_CBID_CORE_RangeStartW : DeepCopyDestroy(args.RangeStartW ); break; + case NVTX_CBID_CORE_RangeEnd : DeepCopyDestroy(args.RangeEnd ); break; + case NVTX_CBID_CORE_RangePushEx : DeepCopyDestroy(args.RangePushEx ); break; + case NVTX_CBID_CORE_RangePushA : DeepCopyDestroy(args.RangePushA ); break; + case NVTX_CBID_CORE_RangePushW : DeepCopyDestroy(args.RangePushW ); break; + case NVTX_CBID_CORE_RangePop : DeepCopyDestroy(args.RangePop ); break; + case NVTX_CBID_CORE_NameCategoryA: DeepCopyDestroy(args.NameCategoryA); break; + case NVTX_CBID_CORE_NameCategoryW: DeepCopyDestroy(args.NameCategoryW); break; + case NVTX_CBID_CORE_NameOsThreadA: DeepCopyDestroy(args.NameOsThreadA); break; + case NVTX_CBID_CORE_NameOsThreadW: DeepCopyDestroy(args.NameOsThreadW); break; + default: break; + } + break; + case NVTX_CB_MODULE_CORE2: + switch (id.cb) + { + case NVTX_CBID_CORE2_DomainMarkEx : DeepCopyDestroy(args.DomainMarkEx ); break; + case NVTX_CBID_CORE2_DomainRangeStartEx : DeepCopyDestroy(args.DomainRangeStartEx ); break; + case NVTX_CBID_CORE2_DomainRangeEnd : DeepCopyDestroy(args.DomainRangeEnd ); break; + case NVTX_CBID_CORE2_DomainRangePushEx : DeepCopyDestroy(args.DomainRangePushEx ); break; + case NVTX_CBID_CORE2_DomainRangePop : DeepCopyDestroy(args.DomainRangePop ); break; + case NVTX_CBID_CORE2_DomainResourceCreate : DeepCopyDestroy(args.DomainResourceCreate ); break; + case NVTX_CBID_CORE2_DomainResourceDestroy: DeepCopyDestroy(args.DomainResourceDestroy); break; + case NVTX_CBID_CORE2_DomainNameCategoryA : DeepCopyDestroy(args.DomainNameCategoryA ); break; + case NVTX_CBID_CORE2_DomainNameCategoryW : DeepCopyDestroy(args.DomainNameCategoryW ); break; + case NVTX_CBID_CORE2_DomainRegisterStringA: DeepCopyDestroy(args.DomainRegisterStringA); break; + case NVTX_CBID_CORE2_DomainRegisterStringW: DeepCopyDestroy(args.DomainRegisterStringW); break; + case NVTX_CBID_CORE2_DomainCreateA : DeepCopyDestroy(args.DomainCreateA ); break; + case NVTX_CBID_CORE2_DomainCreateW : DeepCopyDestroy(args.DomainCreateW ); break; + case NVTX_CBID_CORE2_DomainDestroy : DeepCopyDestroy(args.DomainDestroy ); break; + case NVTX_CBID_CORE2_Initialize : DeepCopyDestroy(args.Initialize ); break; + default: break; + } + break; + default: break; + } + } +}; + +inline std::ostream& operator<<(std::ostream& os, CallData const& data) +{ + if (data.id == CALLID_LOAD()) + { + return os << CallName(data.id) << " returned " << data.args.Load.success; + } + + os << "[" << data.id.mod << "," << std::setw(2) << data.id.cb << "] "; + os << CallName(data.id) << '('; + switch (data.id.mod) + { + case NVTX_CB_MODULE_CORE: + switch (data.id.cb) + { + case NVTX_CBID_CORE_MarkEx : {auto& a = data.args.MarkEx ; os << *a.eventAttrib; } break; + case NVTX_CBID_CORE_MarkA : {auto& a = data.args.MarkA ; os << '"' << a.str << '"'; } break; + case NVTX_CBID_CORE_MarkW : {auto& a = data.args.MarkW ; os << "WIDE"; } break; + case NVTX_CBID_CORE_RangeStartEx : {auto& a = data.args.RangeStartEx ; os << *a.eventAttrib; } break; + case NVTX_CBID_CORE_RangeStartA : {auto& a = data.args.RangeStartA ; os << '"' << a.str << '"'; } break; + case NVTX_CBID_CORE_RangeStartW : {auto& a = data.args.RangeStartW ; os << "WIDE"; } break; + case NVTX_CBID_CORE_RangeEnd : {auto& a = data.args.RangeEnd ; os << a.id; } break; + case NVTX_CBID_CORE_RangePushEx : {auto& a = data.args.RangePushEx ; os << *a.eventAttrib; } break; + case NVTX_CBID_CORE_RangePushA : {auto& a = data.args.RangePushA ; os << '"' << a.str << '"'; } break; + case NVTX_CBID_CORE_RangePushW : {auto& a = data.args.RangePushW ; os << "WIDE"; } break; + case NVTX_CBID_CORE_RangePop : {auto& a = data.args.RangePop ; } break; + case NVTX_CBID_CORE_NameCategoryA: {auto& a = data.args.NameCategoryA; os << a.id << ", \"" << a.str << '"'; } break; + case NVTX_CBID_CORE_NameCategoryW: {auto& a = data.args.NameCategoryW; os << a.id << ", " << "WIDE"; } break; + case NVTX_CBID_CORE_NameOsThreadA: {auto& a = data.args.NameOsThreadA; os << a.id << ", \"" << a.str << '"'; } break; + case NVTX_CBID_CORE_NameOsThreadW: {auto& a = data.args.NameOsThreadW; os << a.id << ", " << "WIDE"; } break; + default: break; + } + break; + case NVTX_CB_MODULE_CORE2: + switch (data.id.cb) + { + case NVTX_CBID_CORE2_DomainMarkEx : {auto& a = data.args.DomainMarkEx ; os << a.domain << ", " << *a.eventAttrib; } break; + case NVTX_CBID_CORE2_DomainRangeStartEx : {auto& a = data.args.DomainRangeStartEx ; os << a.domain << ", " << *a.eventAttrib; } break; + case NVTX_CBID_CORE2_DomainRangeEnd : {auto& a = data.args.DomainRangeEnd ; os << a.domain << ", " << a.id; } break; + case NVTX_CBID_CORE2_DomainRangePushEx : {auto& a = data.args.DomainRangePushEx ; os << a.domain << ", " << *a.eventAttrib; } break; + case NVTX_CBID_CORE2_DomainRangePop : {auto& a = data.args.DomainRangePop ; os << a.domain; } break; + case NVTX_CBID_CORE2_DomainResourceCreate : {auto& a = data.args.DomainResourceCreate ; os << a.domain << ", " << a.attr; } break; // TODO + case NVTX_CBID_CORE2_DomainResourceDestroy: {auto& a = data.args.DomainResourceDestroy; os << a.attr; } break; + case NVTX_CBID_CORE2_DomainNameCategoryA : {auto& a = data.args.DomainNameCategoryA ; os << a.domain << ", " << a.id << ", \"" << a.str << '"';} break; + case NVTX_CBID_CORE2_DomainNameCategoryW : {auto& a = data.args.DomainNameCategoryW ; os << a.domain << ", " << a.id << ", " << "WIDE"; } break; + case NVTX_CBID_CORE2_DomainRegisterStringA: {auto& a = data.args.DomainRegisterStringA; os << a.domain << ", \"" << a.str << '"'; } break; + case NVTX_CBID_CORE2_DomainRegisterStringW: {auto& a = data.args.DomainRegisterStringW; os << a.domain << ", " << "WIDE"; } break; + case NVTX_CBID_CORE2_DomainCreateA : {auto& a = data.args.DomainCreateA ; os << '"' << a.name << '"'; } break; + case NVTX_CBID_CORE2_DomainCreateW : {auto& a = data.args.DomainCreateW ; os << "WIDE"; } break; + case NVTX_CBID_CORE2_DomainDestroy : {auto& a = data.args.DomainDestroy ; os << a.domain; } break; + case NVTX_CBID_CORE2_Initialize : {auto& a = data.args.Initialize ; os << a.reserved; } break; + default: break; + } + break; + default: break; + } + os << ')'; + return os; +}; + +using Call = std::shared_ptr; + +// Helper to write CALL(CORE, NameCategoryA, id, str) to construct a Call with arg values +#define CALL(m,c,...) [=]{ Call v(new CallData); v->id = CALLID(m,c); DeepCopyAssign(v->args.c, Args##c{__VA_ARGS__}); return v; }() + +#define CALL_LOAD(s) [=]{ Call v(new CallData); v->id = CALLID_LOAD(); v->args.Load = ArgsLoad{s}; return v; }() + +// Helpers to construct unions from NVTX C API types +inline nvtxMessageValue_t MakeMessage(const char* msg) { nvtxMessageValue_t v; v.ascii = msg; return v; } +inline nvtxMessageValue_t MakeMessage(const wchar_t* msg) { nvtxMessageValue_t v; v.unicode = msg; return v; } +inline nvtxMessageValue_t MakeMessage(nvtxStringHandle_t msg) { nvtxMessageValue_t v; v.registered = msg; return v; } + +inline nvtxEventAttributes_t::payload_t MakePayload(uint64_t v) { nvtxEventAttributes_t::payload_t p; p.ullValue = v; return p; } +inline nvtxEventAttributes_t::payload_t MakePayload(int64_t v) { nvtxEventAttributes_t::payload_t p; p.llValue = v; return p; } +inline nvtxEventAttributes_t::payload_t MakePayload(double v) { nvtxEventAttributes_t::payload_t p; p.dValue = v; return p; } +inline nvtxEventAttributes_t::payload_t MakePayload(uint32_t v) { nvtxEventAttributes_t::payload_t p; p.uiValue = v; return p; } +inline nvtxEventAttributes_t::payload_t MakePayload(int32_t v) { nvtxEventAttributes_t::payload_t p; p.iValue = v; return p; } +inline nvtxEventAttributes_t::payload_t MakePayload(float v) { nvtxEventAttributes_t::payload_t p; p.fValue = v; return p; } + +// Define Same() overloads for NVTX API types +inline bool Same(nvtxEventAttributes_t const& lhs, nvtxEventAttributes_t const& rhs, SAME_COMMON_ARGS) +{ + bool same = true + && MEMBER_SAME(version) + && MEMBER_SAME(size) + && MEMBER_SAME(category) + && MEMBER_SAME(colorType) + && MEMBER_SAME(color) + && MEMBER_SAME(payloadType) + && (false + || lhs.payloadType == NVTX_PAYLOAD_UNKNOWN + || lhs.payloadType == NVTX_PAYLOAD_TYPE_UNSIGNED_INT64 && MEMBER_SAME(payload.ullValue) + || lhs.payloadType == NVTX_PAYLOAD_TYPE_INT64 && MEMBER_SAME(payload.llValue) + || lhs.payloadType == NVTX_PAYLOAD_TYPE_DOUBLE && MEMBER_SAME(payload.dValue) + || lhs.payloadType == NVTX_PAYLOAD_TYPE_UNSIGNED_INT32 && MEMBER_SAME(payload.uiValue) + || lhs.payloadType == NVTX_PAYLOAD_TYPE_INT32 && MEMBER_SAME(payload.iValue) + || lhs.payloadType == NVTX_PAYLOAD_TYPE_FLOAT && MEMBER_SAME(payload.fValue) + ) + && MEMBER_SAME(messageType) + && (false + || lhs.messageType == NVTX_MESSAGE_UNKNOWN + || lhs.messageType == NVTX_MESSAGE_TYPE_ASCII && MEMBER_SAME(message.ascii) + || lhs.messageType == NVTX_MESSAGE_TYPE_UNICODE && MEMBER_SAME(message.unicode) + || lhs.messageType == NVTX_MESSAGE_TYPE_REGISTERED && MEMBER_SAME(message.registered) + ) + ; + VERBOSE_PRINT() + << std::string(depth, ' ') << "Expected: " << rhs << "\n" + << std::string(depth, ' ') << "Provided: " << lhs << "\n"; + return same; +} +DEFINE_EQ_NE_DEEP(nvtxEventAttributes_t) + +inline bool Same(nvtxResourceAttributes_t const& lhs, nvtxResourceAttributes_t const& rhs, SAME_COMMON_ARGS) +{ + bool same = true + && MEMBER_SAME(version) + && MEMBER_SAME(size) + && MEMBER_SAME(identifierType) + && (false + || lhs.identifierType == NVTX_RESOURCE_TYPE_UNKNOWN + || lhs.identifierType == NVTX_RESOURCE_TYPE_GENERIC_POINTER && MEMBER_SAME(identifier.pValue) + || lhs.identifierType == NVTX_RESOURCE_TYPE_GENERIC_HANDLE && MEMBER_SAME(identifier.ullValue) + || lhs.identifierType == NVTX_RESOURCE_TYPE_GENERIC_THREAD_NATIVE && MEMBER_SAME(identifier.ullValue) + || lhs.identifierType == NVTX_RESOURCE_TYPE_GENERIC_THREAD_POSIX && MEMBER_SAME(identifier.ullValue) + ) + && MEMBER_SAME(messageType) + && (false + || lhs.messageType == NVTX_MESSAGE_UNKNOWN + || lhs.messageType == NVTX_MESSAGE_TYPE_ASCII && MEMBER_SAME(message.ascii) + || lhs.messageType == NVTX_MESSAGE_TYPE_UNICODE && MEMBER_SAME(message.unicode) + || lhs.messageType == NVTX_MESSAGE_TYPE_REGISTERED && MEMBER_SAME(message.registered) + ) + ; + VERBOSE_PRINT(); + return same; +} +DEFINE_EQ_NE_DEEP(nvtxResourceAttributes_t) + +// Define Same() overloads (and operators == and !=) for NVTX arg pack types & Args union + +#define DEFINE_ARGS_SAME_0(cb) DEFINE_SAME_0(Args##cb) +#define DEFINE_ARGS_SAME_1(cb, a) DEFINE_SAME_1(Args##cb, a) +#define DEFINE_ARGS_SAME_2(cb, a, b) DEFINE_SAME_2(Args##cb, a, b) +#define DEFINE_ARGS_SAME_3(cb, a, b, c) DEFINE_SAME_3(Args##cb, a, b, c) + +DEFINE_ARGS_SAME_1(Load, success) +// CORE +DEFINE_ARGS_SAME_1(MarkEx, eventAttrib) +DEFINE_ARGS_SAME_1(MarkA, str) +DEFINE_ARGS_SAME_1(MarkW, str) +DEFINE_ARGS_SAME_1(RangeStartEx, eventAttrib) +DEFINE_ARGS_SAME_1(RangeStartA, str) +DEFINE_ARGS_SAME_1(RangeStartW, str) +DEFINE_ARGS_SAME_0(RangeEnd) +DEFINE_ARGS_SAME_1(RangePushEx, eventAttrib) +DEFINE_ARGS_SAME_1(RangePushA, str) +DEFINE_ARGS_SAME_1(RangePushW, str) +DEFINE_ARGS_SAME_0(RangePop) +DEFINE_ARGS_SAME_2(NameCategoryA, id, str) +DEFINE_ARGS_SAME_2(NameCategoryW, id, str) +DEFINE_ARGS_SAME_2(NameOsThreadA, id, str) +DEFINE_ARGS_SAME_2(NameOsThreadW, id, str) +// CORE2 +DEFINE_ARGS_SAME_2(DomainMarkEx, domain, eventAttrib) +DEFINE_ARGS_SAME_2(DomainRangeStartEx, domain, eventAttrib) +DEFINE_ARGS_SAME_2(DomainRangeEnd, domain, id) +DEFINE_ARGS_SAME_2(DomainRangePushEx, domain, eventAttrib) +DEFINE_ARGS_SAME_1(DomainRangePop, domain) +DEFINE_ARGS_SAME_2(DomainResourceCreate, domain, attr) +DEFINE_ARGS_SAME_1(DomainResourceDestroy, attr) +DEFINE_ARGS_SAME_3(DomainNameCategoryA, domain, id, str) +DEFINE_ARGS_SAME_3(DomainNameCategoryW, domain, id, str) +DEFINE_ARGS_SAME_2(DomainRegisterStringA, domain, str) +DEFINE_ARGS_SAME_2(DomainRegisterStringW, domain, str) +DEFINE_ARGS_SAME_1(DomainCreateA, name) +DEFINE_ARGS_SAME_1(DomainCreateW, name) +DEFINE_ARGS_SAME_1(DomainDestroy, domain) +DEFINE_ARGS_SAME_1(Initialize, reserved) + +inline bool Same(CallData const& lhs, CallData const& rhs, SAME_COMMON_ARGS) +{ + bool same = true + && MEMBER_SAME(id) + && (false + || UNION_MEMBER_SAME(id, CALLID_LOAD(), args.Load) + || UNION_MEMBER_SAME(id, CALLID(CORE, MarkEx), args.MarkEx) + || UNION_MEMBER_SAME(id, CALLID(CORE, MarkA), args.MarkA) + || UNION_MEMBER_SAME(id, CALLID(CORE, MarkW), args.MarkW) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangeStartEx), args.RangeStartEx) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangeStartA), args.RangeStartA) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangeStartW), args.RangeStartW) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangeEnd), args.RangeEnd) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangePushEx), args.RangePushEx) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangePushA), args.RangePushA) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangePushW), args.RangePushW) + || UNION_MEMBER_SAME(id, CALLID(CORE, RangePop), args.RangePop) + || UNION_MEMBER_SAME(id, CALLID(CORE, NameCategoryA), args.NameCategoryA) + || UNION_MEMBER_SAME(id, CALLID(CORE, NameCategoryW), args.NameCategoryW) + || UNION_MEMBER_SAME(id, CALLID(CORE, NameOsThreadA), args.NameOsThreadA) + || UNION_MEMBER_SAME(id, CALLID(CORE, NameOsThreadW), args.NameOsThreadW) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainMarkEx), args.DomainMarkEx) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainRangeStartEx), args.DomainRangeStartEx) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainRangeEnd), args.DomainRangeEnd) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainRangePushEx), args.DomainRangePushEx) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainRangePop), args.DomainRangePop) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainResourceCreate), args.DomainResourceCreate) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainResourceDestroy), args.DomainResourceDestroy) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainNameCategoryA), args.DomainNameCategoryA) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainNameCategoryW), args.DomainNameCategoryW) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainRegisterStringA), args.DomainRegisterStringA) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainRegisterStringW), args.DomainRegisterStringW) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainCreateA), args.DomainCreateA) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainCreateW), args.DomainCreateW) + || UNION_MEMBER_SAME(id, CALLID(CORE2, DomainDestroy), args.DomainDestroy) + || UNION_MEMBER_SAME(id, CALLID(CORE2, Initialize), args.Initialize) + ) + ; + + VERBOSE_PRINT(); + return same; +} +DEFINE_EQ_NE_DEEP(CallData) + +inline nvtxDomainHandle_t PostInc(nvtxDomainHandle_t & h) { auto v = h; ++(intptr_t&)h; return v; } +inline nvtxStringHandle_t PostInc(nvtxStringHandle_t & h) { auto v = h; ++(intptr_t&)h; return v; } +inline nvtxResourceHandle_t PostInc(nvtxResourceHandle_t& h) { auto v = h; ++(intptr_t&)h; return v; } +inline nvtxRangeId_t PostInc(nvtxRangeId_t & h) { return h++; } + +struct Callbacks +{ + std::function Default; + std::function Load; + + std::function MarkEx; + std::function MarkA; + std::function MarkW; + std::function RangeStartEx; + std::function RangeStartA; + std::function RangeStartW; + std::function RangeEnd; + std::function RangePushEx; + std::function RangePushA; + std::function RangePushW; + std::function RangePop; + std::function NameCategoryA; + std::function NameCategoryW; + std::function NameOsThreadA; + std::function NameOsThreadW; + + std::function DomainMarkEx; + std::function DomainRangeStartEx; + std::function DomainRangeEnd; + std::function DomainRangePushEx; + std::function DomainRangePop; + std::function DomainResourceCreate; + std::function DomainResourceDestroy; + std::function DomainNameCategoryA; + std::function DomainNameCategoryW; + std::function DomainRegisterStringA; + std::function DomainRegisterStringW; + std::function DomainCreateA; + std::function DomainCreateW; + std::function DomainDestroy; + std::function Initialize; + + + Callbacks(Callbacks const&) = default; + Callbacks& operator=(Callbacks const&) = default; + Callbacks(Callbacks&&) = default; + Callbacks& operator=(Callbacks&&) = default; + + nvtxDomainHandle_t nextDomainHandle = (nvtxDomainHandle_t)1; + struct DomainData + { + int pushPopDepth = 0; + nvtxRangeId_t nextRangeId = (nvtxRangeId_t)1; + nvtxStringHandle_t nextStringHandle = (nvtxStringHandle_t)1; + nvtxResourceHandle_t nextResourceHandle = (nvtxResourceHandle_t)1; + }; + std::map domainData; + + Callbacks() + : Default([](Call const&) {}) + , Load ([&](int success) { Default(CALL_LOAD(success)); }) + // CORE + , MarkEx ([&](const nvtxEventAttributes_t* a) { Default(CALL(CORE, MarkEx , a )); }) + , MarkA ([&](const char* a) { Default(CALL(CORE, MarkA , a )); }) + , MarkW ([&](const wchar_t* a) { Default(CALL(CORE, MarkW , a )); }) + , RangeStartEx ([&](const nvtxEventAttributes_t* a) { Default(CALL(CORE, RangeStartEx , a )); return PostInc(domainData[nullptr].nextRangeId); }) + , RangeStartA ([&](const char* a) { Default(CALL(CORE, RangeStartA , a )); return PostInc(domainData[nullptr].nextRangeId); }) + , RangeStartW ([&](const wchar_t* a) { Default(CALL(CORE, RangeStartW , a )); return PostInc(domainData[nullptr].nextRangeId); }) + , RangeEnd ([&](nvtxRangeId_t a) { Default(CALL(CORE, RangeEnd , a )); }) + , RangePushEx ([&](const nvtxEventAttributes_t* a) { Default(CALL(CORE, RangePushEx , a )); return ++domainData[nullptr].pushPopDepth; }) + , RangePushA ([&](const char* a) { Default(CALL(CORE, RangePushA , a )); return ++domainData[nullptr].pushPopDepth; }) + , RangePushW ([&](const wchar_t* a) { Default(CALL(CORE, RangePushW , a )); return ++domainData[nullptr].pushPopDepth; }) + , RangePop ([&]( ) { Default(CALL(CORE, RangePop )); return domainData[nullptr].pushPopDepth--; }) + , NameCategoryA([&](uint32_t a, const char* b) { Default(CALL(CORE, NameCategoryA, a, b)); }) + , NameCategoryW([&](uint32_t a, const wchar_t* b) { Default(CALL(CORE, NameCategoryW, a, b)); }) + , NameOsThreadA([&](uint32_t a, const char* b) { Default(CALL(CORE, NameOsThreadA, a, b)); }) + , NameOsThreadW([&](uint32_t a, const wchar_t* b) { Default(CALL(CORE, NameOsThreadW, a, b)); }) + // CORE2 + , DomainMarkEx ([&](nvtxDomainHandle_t a, const nvtxEventAttributes_t* b) { Default(CALL(CORE2, DomainMarkEx , a, b )); }) + , DomainRangeStartEx ([&](nvtxDomainHandle_t a, const nvtxEventAttributes_t* b) { Default(CALL(CORE2, DomainRangeStartEx , a, b )); return PostInc(domainData[a].nextRangeId); }) + , DomainRangeEnd ([&](nvtxDomainHandle_t a, nvtxRangeId_t b) { Default(CALL(CORE2, DomainRangeEnd , a, b )); }) + , DomainRangePushEx ([&](nvtxDomainHandle_t a, const nvtxEventAttributes_t* b) { Default(CALL(CORE2, DomainRangePushEx , a, b )); return ++domainData[a].pushPopDepth; }) + , DomainRangePop ([&](nvtxDomainHandle_t a) { Default(CALL(CORE2, DomainRangePop , a )); return domainData[a].pushPopDepth--; }) + , DomainResourceCreate ([&](nvtxDomainHandle_t a, nvtxResourceAttributes_t* b) { Default(CALL(CORE2, DomainResourceCreate , a, b )); return PostInc(domainData[a].nextResourceHandle); }) + , DomainResourceDestroy([&](nvtxResourceHandle_t a) { Default(CALL(CORE2, DomainResourceDestroy, a )); }) + , DomainNameCategoryA ([&](nvtxDomainHandle_t a, uint32_t b, const char* c) { Default(CALL(CORE2, DomainNameCategoryA , a, b, c)); }) + , DomainNameCategoryW ([&](nvtxDomainHandle_t a, uint32_t b, const wchar_t* c) { Default(CALL(CORE2, DomainNameCategoryW , a, b, c)); }) + , DomainRegisterStringA([&](nvtxDomainHandle_t a, const char* b) { Default(CALL(CORE2, DomainRegisterStringA, a, b )); return PostInc(domainData[a].nextStringHandle); }) + , DomainRegisterStringW([&](nvtxDomainHandle_t a, const wchar_t* b) { Default(CALL(CORE2, DomainRegisterStringW, a, b )); return PostInc(domainData[a].nextStringHandle); }) + , DomainCreateA ([&](const char* a) { Default(CALL(CORE2, DomainCreateA , a )); return PostInc(nextDomainHandle); }) + , DomainCreateW ([&](const wchar_t* a) { Default(CALL(CORE2, DomainCreateW , a )); return PostInc(nextDomainHandle); }) + , DomainDestroy ([&](nvtxDomainHandle_t a) { Default(CALL(CORE2, DomainDestroy , a )); }) + , Initialize ([&](const void* a) { Default(CALL(CORE2, Initialize , a )); }) + { + } +}; + +extern Callbacks g_callbacks; + + diff --git a/tests/TestCoverage.h b/tests/TestCoverage.h new file mode 100644 index 0000000..efc9347 --- /dev/null +++ b/tests/TestCoverage.h @@ -0,0 +1,478 @@ +#include + +#include +#include + +#include "PrettyPrintersNvtxCpp.h" + +struct a_lib +{ + static constexpr const char* name{"Library A"}; + //static constexpr const float name{3.14f}; +}; + +struct cat_x +{ + static constexpr const char* name{"Category X"}; + static constexpr uint32_t id{42}; +}; + +struct cat_y +{ + static constexpr const char* name{"Category Y"}; + //static constexpr const float name{3.14f}; + static constexpr uint32_t id{43}; +}; + +struct regstr_hello +{ + static constexpr const char* message{"Hello"}; +}; + +static void TestFuncRange() +{ + NVTX3_FUNC_RANGE(); + nvtx3::mark("Marker in TestFuncRange"); +} + +static void TestFuncRangeV() +{ + NVTX3_V1_FUNC_RANGE(); + nvtx3::mark("Marker in TestFuncRangeV"); +} + +static void TestFuncRangeIfDyn(bool cond) +{ + NVTX3_FUNC_RANGE_IF(cond); + nvtx3::mark("Marker in TestFuncRangeIfDyn"); +} + +static void TestFuncRangeIfDynV(bool cond) +{ + NVTX3_V1_FUNC_RANGE_IF(cond); + nvtx3::mark("Marker in TestFuncRangeIfDynV"); +} + +static void TestFuncRangeIfStat(bool cond) +{ + NVTX3_FUNC_RANGE_IF(cond); + nvtx3::mark("Marker in TestFuncRangeIfStat"); +} + +static void TestFuncRangeIfStatV(bool cond) +{ + NVTX3_V1_FUNC_RANGE_IF(cond); + nvtx3::mark("Marker in TestFuncRangeIfStatV"); +} + +static void TestFuncRangeIn() +{ + NVTX3_FUNC_RANGE_IN(a_lib); + nvtx3::mark("Marker in TestFuncRangeIn"); +} + +static void TestFuncRangeInV() +{ + NVTX3_V1_FUNC_RANGE_IN(a_lib); + nvtx3::mark("Marker in TestFuncRangeInV"); +} + +static void TestFuncRangeIfInDyn(bool cond) +{ + NVTX3_FUNC_RANGE_IF_IN(a_lib, cond); + nvtx3::mark("Marker in TestFuncRangeIfInDyn"); +} + +static void TestFuncRangeIfInDynV(bool cond) +{ + NVTX3_V1_FUNC_RANGE_IF_IN(a_lib, cond); + nvtx3::mark("Marker in TestFuncRangeIfInDynV"); +} + +static void TestFuncRangeIfInStat(bool cond) +{ + NVTX3_FUNC_RANGE_IF_IN(a_lib, cond); + nvtx3::mark("Marker in TestFuncRangeIfInStat"); +} + +static void TestFuncRangeIfInStatV(bool cond) +{ + NVTX3_V1_FUNC_RANGE_IF_IN(a_lib, cond); + nvtx3::mark("Marker in TestFuncRangeIfInStatV"); +} + +static int RunTestCommon(int argc, const char** argv) +{ + bool verbose = false; + const std::string verboseArg = "-v"; + for (; *argv; ++argv) + { + if (*argv == verboseArg) verbose = true; + } + + using namespace nvtx3; + + { + std::cout << "Default attributes:\n"; + event_attributes attr; + if (verbose) std::cout << attr << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a message (ascii), payload, color, and category:\n"; + event_attributes attr{ + message{"Hello"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}; + if (verbose) std::cout << attr << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a message with different string types:\n"; + + event_attributes a{message{"Hello"}}; + if (verbose) std::cout << a << '\n'; + + event_attributes wa{message{L"Hello"}}; + if (verbose) std::cout << wa << '\n'; + + std::string hello{"Hello"}; + event_attributes b{message{hello}}; + if (verbose) std::cout << b << '\n'; + + std::wstring whello{L"Hello"}; + event_attributes wb{message{whello}}; + if (verbose) std::cout << wb << '\n'; + + // Important! Neither of following will compile: + // + // event_attributes c{message{std::string{"foo"}}}; + // std::cout << c; + // + // std::string foo{"foo"}; + // event_attributes d{message{hello + "bar"}}; + // std::cout << d; + // + // Both of those usages fail with: + // "error C2280: 'message::message(std::string &&)': + // attempting to reference a deleted function" + // + // message is a "view" class, not an owning class. + // It cannot take ownership of a temporary string and + // destroy it when it goes out of scope. Similarly, + // event_attributes is not an owning class, so it cannot take + // ownership of an message either. + // + // TODO: Could we add implicit support for this? + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a message (registered):\n"; + auto hTacobell = reinterpret_cast(0x7ac0be11); + event_attributes attr{message{hTacobell}}; + if (verbose) std::cout << attr << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Convenience: Set a message without the helper type:\n"; + + event_attributes a{"Hello"}; + if (verbose) std::cout << a << '\n'; + + std::string hello{"Hello"}; + event_attributes b{hello}; + if (verbose) std::cout << b << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a payload twice (first should win):\n"; + event_attributes attr{"test", payload{1.0f}, payload{2}}; + if (verbose) std::cout << attr << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a color twice (first should win):\n"; + event_attributes attr{"test", argb{127,0,0,255}, rgb{0,255,0}}; + if (verbose) std::cout << attr << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a message twice (first should win):\n"; + event_attributes attr{L"wide", "narrow"}; + if (verbose) std::cout << attr << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Set a category twice (first should win):\n"; + event_attributes attr{"test", category{1}, category{2}}; + if (verbose) std::cout << attr << '\n'; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Markers\n"; + + // Global domain + event_attributes attr{ + message{"Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + mark(attr); + + mark(event_attributes{ + message{"Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}); + + mark( + message{"Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}); + + // a_lib domain + event_attributes a_attr{ + message{"a: Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + mark_in(attr); + + mark_in(event_attributes{ + message{"a: Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}); + + mark_in( + message{"a: Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}); + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Range start/end and range_handle\n"; + + // Global domain + event_attributes attr{ + message{"Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + auto h1 = start_range(attr); + + auto h2 = start_range(event_attributes{ + message{"Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}); + + auto h3 = start_range( + message{"Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}); + + // a_lib domain + event_attributes a_attr{ + message{"a: Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + auto h4 = start_range_in(attr); + + auto h5 = start_range_in(event_attributes{ + message{"a: Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}); + + auto h6 = start_range_in( + message{"a: Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}); + + // range_handle operator ==, !=, and cast overloads + bool testEq = h1 == h2; + bool testNe = h3 != h4; + bool testCast = bool(h5); + if (verbose) std::cout << std::boolalpha + << testEq << "\n" + << testNe << "\n" + << testCast << "\n"; + + end_range(h1); + end_range(h2); + end_range(h3); + + end_range_in(h4); + end_range_in(h5); + end_range_in(h6); + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "unique_range\n"; + + // Global domain + event_attributes attr{ + message{"Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + unique_range u1{attr}; + + unique_range u2{event_attributes{ + message{"Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}}; + + unique_range u3{ + message{"Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}; + + // a_lib domain + event_attributes a_attr{ + message{"a: Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + unique_range_in u4{attr}; + + unique_range_in u5{event_attributes{ + message{"a: Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}}; + + unique_range_in u6{ + message{"a: Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}; + + // movability + auto move_in_out_global = [](unique_range u) { return u; }; + auto move_in_out_domain = [](unique_range_in u) { return u; }; + + auto u1moved = move_in_out_global(std::move(u1)); + auto u4moved = move_in_out_domain(std::move(u4)); + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "scoped_range\n"; + + // Global domain + event_attributes attr{ + message{"Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + scoped_range s1{attr}; + + scoped_range s2{event_attributes{ + message{"Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}}; + + scoped_range s3{ + message{"Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}; + + // a_lib domain + event_attributes a_attr{ + message{"a: Hello1"}, + category{11}, + payload{5.0f}, + rgb{1,2,3}}; + scoped_range_in s4{attr}; + + scoped_range_in s5{event_attributes{ + message{"a: Hello2"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}}; + + scoped_range_in s6{ + message{"a: Hello3"}, + category{11}, + payload{5.0f}, + rgb{0,255,0}}; + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "named_category\n"; + + // Global domain + mark ("Cat", named_category::get ()); + mark_in<> ("Cat", named_category_in<>::get ()); + mark_in("Cat", named_category_in::get()); + + // a_lib domain + mark_in("Cat", named_category_in::get()); + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "registered_string\n"; + + // Global domain + mark ("RegStr", registered_string::get ()); + mark_in<> ("RegStr", registered_string_in<>::get ()); + mark_in("RegStr", registered_string_in::get()); + + // a_lib domain + mark_in("RegStr", registered_string_in::get()); + } + if (verbose) std::cout << "-------------------------------------\n"; + + { + std::cout << "Macros:\n"; + TestFuncRange(); + TestFuncRangeV(); + TestFuncRangeIfDyn(argc == 1001); + TestFuncRangeIfDyn(argc != 1001); + TestFuncRangeIfDynV(argc == 1002); + TestFuncRangeIfDynV(argc != 1002); + TestFuncRangeIfStat(true); + TestFuncRangeIfStat(false); + TestFuncRangeIfStatV(true); + TestFuncRangeIfStatV(false); + + TestFuncRangeIn(); + TestFuncRangeInV(); + TestFuncRangeIfInDyn(argc == 1003); + TestFuncRangeIfInDyn(argc != 1003); + TestFuncRangeIfInDynV(argc == 1004); + TestFuncRangeIfInDynV(argc != 1004); + TestFuncRangeIfInStat(true); + TestFuncRangeIfInStat(false); + TestFuncRangeIfInStatV(true); + TestFuncRangeIfInStatV(false); + } + if (verbose) std::cout << "-------------------------------------\n"; + + return 0; +} diff --git a/tests/TestSelfInjection.cpp b/tests/TestSelfInjection.cpp new file mode 100644 index 0000000..fb44e46 --- /dev/null +++ b/tests/TestSelfInjection.cpp @@ -0,0 +1,251 @@ +#include + +#include +#include + +#include +#include +#include + +#include "SelfInjection.h" + +#include "DllHelper.h" + +struct S1 +{ + int i; + float f; +}; +bool operator==(S1 const& lhs, S1 const& rhs) +{ + return lhs.i == rhs.i && lhs.f == rhs.f; +} +std::ostream& operator<<(std::ostream& lhs, S1 const& rhs) +{ + return lhs << '{' << rhs.i << ',' << rhs.f << '}'; +} + +struct S2 +{ + int i; + float f; + const char* s; +}; + +bool Same(S2 const& lhs, S2 const& rhs, SAME_COMMON_ARGS) +{ + bool same = + Same(lhs.i, rhs.i, deep, verbose, "i", oss, depth + 1) && + Same(lhs.f, rhs.f, deep, verbose, "f", oss, depth + 1) && + Same(lhs.s, rhs.s, deep, verbose, "s", oss, depth + 1); + if (verbose && !same) oss << std::string(depth, ' ') << "'" << name << "' members different\n"; + return same; +} + +bool TestSame(bool verbose, bool deep) +{ + std::cout << std::boolalpha; + + std::cout << "--- Simple ints:\n"; + { + int xL = 5, xR = 5; + bool result = Same(xL, xR, deep, verbose, "x"); + std::cout << "> == ints: " << result << '\n'; + } + { + int xL = 5, xR = 6; + bool result = Same(xL, xR, deep, verbose, "x"); + std::cout << "> != ints: " << result << '\n'; + } + + std::cout << "--- C-style strings:\n"; + { + const char* str = "String"; + bool result = Same(str, str, deep, verbose, "str"); + std::cout << "> char string w/itself: " << result << '\n'; + } + { + const char* strL = "String"; + const char* strR = "String"; + bool result = Same(strL, strR, deep, verbose, "str"); + std::cout << "> == char strings: " << result << '\n'; + } + { + const char* strL = "StringA"; + const char* strR = "StringB"; + bool result = Same(strL, strR, deep, verbose, "str"); + std::cout << "> != char strings: " << result << '\n'; + } + + std::cout << "--- Structs with == and << operators:\n"; + { + S1 sL{5, 3.125f}; + S1 sR{5, 3.125f}; + bool result = Same(sL, sR, deep, verbose, "S1"); + std::cout << "> == S1s: " << result << '\n'; + } + { + S1 sL{5, 3.125f}; + S1 sR{5, 3.14159f}; + bool result = Same(sL, sR, deep, verbose, "S1"); + std::cout << "> != S1s: " << result << '\n'; + } + + std::cout << "--- Pointers to structs with == and << operators:\n"; + { + S1 sL{5, 3.125f}; + S1* psL = &sL; + bool result = Same(psL, psL, deep, verbose, "S1 ptr"); + std::cout << "> same ptr to an S1: " << result << '\n'; + } + { + S1 sL{5, 3.125f}; + S1 sR{5, 3.125f}; + S1* psL = &sL; + S1* psR = &sR; + bool result = Same(psL, psR, deep, verbose, "S1 ptr"); + std::cout << "> different ptrs to == S1s: " << result << '\n'; + } + { + S1 sL{5, 3.125f}; + S1 sR{5, 3.14159f}; + S1* psL = &sL; + S1* psR = &sR; + bool result = Same(psL, psR, deep, verbose, "S1 ptr"); + std::cout << "> different ptrs to != S1s: " << result << '\n'; + } + + std::cout << "--- Structs with Same function defined:\n"; + { + S2 sL{5, 3.125f, "An S2"}; + S2 sR{5, 3.125f, "An S2"}; + bool result = Same(sL, sR, deep, verbose, "S2"); + std::cout << "> == S2s: " << result << '\n'; + } + { + S2 sL{5, 3.125f, "An S2"}; + S2 sR{5, 3.14159f, "An S2"}; + bool result = Same(sL, sR, deep, verbose, "S2"); + std::cout << "> !=f in S2s: " << result << '\n'; + } + { + S2 sL{5, 3.125f, "An S2"}; + S2 sR{5, 3.125f, "Another S2"}; + bool result = Same(sL, sR, deep, verbose, "S2"); + std::cout << "> !=s in S2s: " << result << '\n'; + } + + std::cout << "--- NVTX handles - pointers to incomplete types:\n"; + { + auto hL = reinterpret_cast(1024); + auto hR = reinterpret_cast(1024); + bool result = Same(hL, hR, deep, verbose, "nvtxDomainHandle_t"); + std::cout << "> == domain handles: " << result << '\n'; + } + { + auto hL = reinterpret_cast(1024); + auto hR = reinterpret_cast(2048); + bool result = Same(hL, hR, deep, verbose, "nvtxDomainHandle_t"); + std::cout << "> != domain handles: " << result << '\n'; + } + + std::cout << "--- NVTX event attributes - struct with tagged union:\n"; + { + char buf1[]{"Test message"}; + char buf2[]{"Test message"}; + + nvtxEventAttributes_t aL{}; + aL.version = NVTX_VERSION; + aL.size = sizeof(nvtxEventAttributes_t); + aL.category = 5; + aL.colorType = NVTX_COLOR_ARGB; + aL.color = 0xFF446688; + aL.payloadType = NVTX_PAYLOAD_TYPE_DOUBLE; + aL.payload.dValue = 3.125; + aL.messageType = NVTX_MESSAGE_TYPE_ASCII; + aL.message.ascii = buf1; + aL.reserved0 = 1; + + auto aR = aL; + + auto* paL = &aL; + auto* paR = &aR; + + bool result = Same(aL, aR, deep, verbose, "nvtxEventAttributes_t"); + std::cout << "> == attrs: " << result << '\n'; + + aR = aL; + aR.reserved0 = 2; + result = Same(aL, aR, deep, verbose, "nvtxEventAttributes_t"); + std::cout << "> == attrs with different padding: " << result << '\n'; + + aR = aL; + aR.category = 6; + result = Same(aL, aR, deep, verbose, "nvtxEventAttributes_t"); + std::cout << "> != attrs, category: " << result << '\n'; + + aR = aL; + aR.message.ascii = buf2; + result = Same(aL, aR, deep, verbose, "nvtxEventAttributes_t"); + std::cout << "> == attrs with same message in different buffers: " << result << '\n'; + + aR = aL; + aR.message.ascii = "Different message"; + result = Same(aL, aR, deep, verbose, "nvtxEventAttributes_t"); + std::cout << "> != attrs, message: " << result << '\n'; + + aR = aL; + aR.payloadType = NVTX_PAYLOAD_TYPE_FLOAT; + result = Same(aL, aR, deep, verbose, "nvtxEventAttributes_t"); + std::cout << "> != attrs, payloadType: " << result << '\n'; + + aR = aL; + aR.payload.dValue = -3.125; + result = Same(aL, aR, deep, verbose, "nvtxEventAttributes_t"); + std::cout << "> != attrs, payload union value: " << result << '\n'; + + aR = aL; + result = Same(paL, paL, deep, verbose, "nvtxEventAttributes_t by pointer"); + std::cout << "> == attr pointers: " << result << '\n'; + + result = Same(paL, paR, deep, verbose, "nvtxEventAttributes_t by pointer"); + std::cout << "> == attr values, different pointers: " << result << '\n'; + + aR.payload.dValue = -3.125; + result = Same(paL, paR, deep, verbose, "nvtxEventAttributes_t by pointer"); + std::cout << "> != attr values, payload union value: " << result << '\n'; + } + + return true; +} + +extern "C" DLL_EXPORT +int RunTest(int argc, const char** argv) +{ + // Always verbose -- tests both verbose and non-verbose modes + + { + std::cout << "\n------- Non-verbose, non-deep:\n"; + bool success = TestSame(false, false); + if (!success) { std::cout << "TestSame returned false!\n"; return 1; } + } + { + std::cout << "\n------- Non-verbose, deep:\n"; + bool success = TestSame(false, true); + if (!success) { std::cout << "TestSame returned false!\n"; return 1; } + } + + { + std::cout << "\n------- Verbose, non-deep:\n"; + bool success = TestSame(true, false); + if (!success) { std::cout << "TestSame returned false!\n"; return 1; } + } + { + std::cout << "\n------- Verbose, deep:\n"; + bool success = TestSame(true, true); + if (!success) { std::cout << "TestSame returned false!\n"; return 1; } + } + + std::cout << "\n--------- Success!\n"; + return 0; +}