forked from OSchip/llvm-project
141 lines
5.0 KiB
C++
141 lines
5.0 KiB
C++
//===- 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/Support/SourceMgr.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);
|
|
}
|
|
}
|