| //===- unittests/Passes/Plugins/PluginsTest.cpp ---------------------------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "llvm/Analysis/CGSCCPassManager.h" |
| #include "llvm/AsmParser/Parser.h" |
| #include "llvm/Config/config.h" |
| #include "llvm/IR/GlobalVariable.h" |
| #include "llvm/IR/PassManager.h" |
| #include "llvm/Passes/PassBuilder.h" |
| #include "llvm/Passes/PassPlugin.h" |
| #include "llvm/Support/FileSystem.h" |
| #include "llvm/Support/ManagedStatic.h" |
| #include "llvm/Support/Path.h" |
| #include "llvm/Testing/Support/Error.h" |
| #include "llvm/Transforms/Scalar/LoopPassManager.h" |
| #include "gtest/gtest.h" |
| |
| #include "TestPlugin.h" |
| |
| #include <cstdint> |
| |
| using namespace llvm; |
| |
| void anchor() {} |
| |
| static std::string LibPath(const std::string Name = "TestPlugin") { |
| const auto &Argvs = testing::internal::GetArgvs(); |
| const char *Argv0 = Argvs.size() > 0 ? Argvs[0].c_str() : "PluginsTests"; |
| void *Ptr = (void *)(intptr_t)anchor; |
| std::string Path = sys::fs::getMainExecutable(Argv0, Ptr); |
| llvm::SmallString<256> Buf{sys::path::parent_path(Path)}; |
| sys::path::append(Buf, (Name + LLVM_PLUGIN_EXT).c_str()); |
| return std::string(Buf.str()); |
| } |
| |
| TEST(PluginsTests, LoadPlugin) { |
| #if !defined(LLVM_ENABLE_PLUGINS) |
| // Disable the test if plugins are disabled. |
| return; |
| #endif |
| |
| auto PluginPath = LibPath(); |
| ASSERT_NE("", PluginPath); |
| |
| Expected<PassPlugin> Plugin = PassPlugin::Load(PluginPath); |
| ASSERT_TRUE(!!Plugin) << "Plugin path: " << PluginPath; |
| |
| ASSERT_EQ(TEST_PLUGIN_NAME, Plugin->getPluginName()); |
| ASSERT_EQ(TEST_PLUGIN_VERSION, Plugin->getPluginVersion()); |
| |
| PassBuilder PB; |
| ModulePassManager PM; |
| ASSERT_THAT_ERROR(PB.parsePassPipeline(PM, "plugin-pass"), Failed()); |
| |
| Plugin->registerPassBuilderCallbacks(PB); |
| ASSERT_THAT_ERROR(PB.parsePassPipeline(PM, "plugin-pass"), Succeeded()); |
| } |
| |
| // Test that llvmGetPassPluginInfo from DoublerPlugin is called twice with |
| // -fpass-plugin=DoublerPlugin -fpass-plugin=TestPlugin |
| // -fpass-plugin=DoublerPlugin. |
| TEST(PluginsTests, LoadMultiplePlugins) { |
| #if !defined(LLVM_ENABLE_PLUGINS) |
| // Disable the test if plugins are disabled. |
| return; |
| #endif |
| |
| auto DoublerPluginPath = LibPath("DoublerPlugin"); |
| auto TestPluginPath = LibPath("TestPlugin"); |
| ASSERT_NE("", DoublerPluginPath); |
| ASSERT_NE("", TestPluginPath); |
| |
| Expected<PassPlugin> DoublerPlugin1 = PassPlugin::Load(DoublerPluginPath); |
| ASSERT_TRUE(!!DoublerPlugin1) |
| << "Plugin path: " << DoublerPlugin1->getFilename(); |
| |
| Expected<PassPlugin> TestPlugin = PassPlugin::Load(TestPluginPath); |
| ASSERT_TRUE(!!TestPlugin) << "Plugin path: " << TestPlugin->getFilename(); |
| |
| // If llvmGetPassPluginInfo is resolved as a weak symbol taking into account |
| // all loaded symbols, the second call to PassPlugin::Load will actually |
| // return the llvmGetPassPluginInfo from the most recently loaded plugin, in |
| // this case TestPlugin. |
| Expected<PassPlugin> DoublerPlugin2 = PassPlugin::Load(DoublerPluginPath); |
| ASSERT_TRUE(!!DoublerPlugin2) |
| << "Plugin path: " << DoublerPlugin2->getFilename(); |
| |
| ASSERT_EQ("DoublerPlugin", DoublerPlugin1->getPluginName()); |
| ASSERT_EQ("2.2-unit", DoublerPlugin1->getPluginVersion()); |
| ASSERT_EQ(TEST_PLUGIN_NAME, TestPlugin->getPluginName()); |
| ASSERT_EQ(TEST_PLUGIN_VERSION, TestPlugin->getPluginVersion()); |
| // Check that the plugin name/version is set correctly when loaded a second |
| // time |
| ASSERT_EQ("DoublerPlugin", DoublerPlugin2->getPluginName()); |
| ASSERT_EQ("2.2-unit", DoublerPlugin2->getPluginVersion()); |
| |
| PassBuilder PB; |
| ModulePassManager PM; |
| const char *PipelineText = "module(doubler-pass,plugin-pass,doubler-pass)"; |
| ASSERT_THAT_ERROR(PB.parsePassPipeline(PM, PipelineText), Failed()); |
| TestPlugin->registerPassBuilderCallbacks(PB); |
| DoublerPlugin1->registerPassBuilderCallbacks(PB); |
| DoublerPlugin2->registerPassBuilderCallbacks(PB); |
| ASSERT_THAT_ERROR(PB.parsePassPipeline(PM, PipelineText), Succeeded()); |
| |
| LLVMContext C; |
| SMDiagnostic Err; |
| std::unique_ptr<Module> M = |
| parseAssemblyString(R"IR(@doubleme = constant i32 7)IR", Err, C); |
| |
| // Check that the initial value is 7 |
| { |
| auto *GV = M->getNamedValue("doubleme"); |
| auto *Init = cast<GlobalVariable>(GV)->getInitializer(); |
| auto *CI = cast<ConstantInt>(Init); |
| ASSERT_EQ(CI->getSExtValue(), 7); |
| } |
| |
| ModuleAnalysisManager MAM; |
| // Register required pass instrumentation analysis. |
| MAM.registerPass([&] { return PassInstrumentationAnalysis(); }); |
| PM.run(*M, MAM); |
| |
| // Check that the final value is 28 because DoublerPlugin::run was called |
| // twice, indicating that the llvmGetPassPluginInfo and registerCallbacks |
| // were correctly called. |
| { |
| // Check the value was doubled twice |
| auto *GV = M->getNamedValue("doubleme"); |
| auto *Init = cast<GlobalVariable>(GV)->getInitializer(); |
| auto *CI = cast<ConstantInt>(Init); |
| ASSERT_EQ(CI->getSExtValue(), 28); |
| } |
| } |