1c1db8cf9SFrederich Munch //===- llvm/unittest/Support/DynamicLibrary/DynamicLibraryTest.cpp --------===//
2c1db8cf9SFrederich Munch //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6c1db8cf9SFrederich Munch //
7c1db8cf9SFrederich Munch //===----------------------------------------------------------------------===//
8c1db8cf9SFrederich Munch
9c1db8cf9SFrederich Munch #include "llvm/Support/DynamicLibrary.h"
109a67b073SChandler Carruth #include "llvm/Config/config.h"
11c1db8cf9SFrederich Munch #include "llvm/Support/FileSystem.h"
12c1db8cf9SFrederich Munch #include "llvm/Support/ManagedStatic.h"
13c1db8cf9SFrederich Munch #include "llvm/Support/Path.h"
14c1db8cf9SFrederich Munch #include "gtest/gtest.h"
15c1db8cf9SFrederich Munch
16c1db8cf9SFrederich Munch #include "PipSqueak.h"
17c1db8cf9SFrederich Munch
18c1db8cf9SFrederich Munch using namespace llvm;
19c1db8cf9SFrederich Munch using namespace llvm::sys;
20c1db8cf9SFrederich Munch
LibPath(const std::string Name="PipSqueak")21ad125800SFrederich Munch std::string LibPath(const std::string Name = "PipSqueak") {
22d4d80a29SBenjamin Kramer const auto &Argvs = testing::internal::GetArgvs();
2312389819SNico Weber const char *Argv0 =
2412389819SNico Weber Argvs.size() > 0 ? Argvs[0].c_str() : "DynamicLibraryTests";
25eab432a1SFrederich Munch void *Ptr = (void*)(intptr_t)TestA;
26287a9ea0SDimitry Andric std::string Path = fs::getMainExecutable(Argv0, Ptr);
27c1db8cf9SFrederich Munch llvm::SmallString<256> Buf(path::parent_path(Path));
28e9f9ec83SXiangling Liao path::append(Buf, (Name + LLVM_PLUGIN_EXT).c_str());
29adcd0268SBenjamin Kramer return std::string(Buf.str());
30c1db8cf9SFrederich Munch }
31c1db8cf9SFrederich Munch
32c1db8cf9SFrederich Munch #if defined(_WIN32) || (defined(HAVE_DLFCN_H) && defined(HAVE_DLOPEN))
33c1db8cf9SFrederich Munch
34c1db8cf9SFrederich Munch typedef void (*SetStrings)(std::string &GStr, std::string &LStr);
35ad125800SFrederich Munch typedef void (*TestOrder)(std::vector<std::string> &V);
36c1db8cf9SFrederich Munch typedef const char *(*GetString)();
37c1db8cf9SFrederich Munch
FuncPtr(void * Ptr)38c1db8cf9SFrederich Munch template <class T> static T FuncPtr(void *Ptr) {
39c1db8cf9SFrederich Munch union {
40c1db8cf9SFrederich Munch T F;
41c1db8cf9SFrederich Munch void *P;
42c1db8cf9SFrederich Munch } Tmp;
43c1db8cf9SFrederich Munch Tmp.P = Ptr;
44c1db8cf9SFrederich Munch return Tmp.F;
45c1db8cf9SFrederich Munch }
PtrFunc(T * Func)46c1db8cf9SFrederich Munch template <class T> static void* PtrFunc(T *Func) {
47c1db8cf9SFrederich Munch union {
48c1db8cf9SFrederich Munch T *F;
49c1db8cf9SFrederich Munch void *P;
50c1db8cf9SFrederich Munch } Tmp;
51c1db8cf9SFrederich Munch Tmp.F = Func;
52c1db8cf9SFrederich Munch return Tmp.P;
53c1db8cf9SFrederich Munch }
54c1db8cf9SFrederich Munch
OverloadTestA()55c1db8cf9SFrederich Munch static const char *OverloadTestA() { return "OverloadCall"; }
56c1db8cf9SFrederich Munch
StdString(const char * Ptr)57c1db8cf9SFrederich Munch std::string StdString(const char *Ptr) { return Ptr ? Ptr : ""; }
58c1db8cf9SFrederich Munch
TEST(DynamicLibrary,Overload)59c1db8cf9SFrederich Munch TEST(DynamicLibrary, Overload) {
60c1db8cf9SFrederich Munch {
61c1db8cf9SFrederich Munch std::string Err;
62c1db8cf9SFrederich Munch llvm_shutdown_obj Shutdown;
63c1db8cf9SFrederich Munch DynamicLibrary DL =
64c1db8cf9SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err);
65c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid());
66c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty());
67c1db8cf9SFrederich Munch
68c1db8cf9SFrederich Munch GetString GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA"));
69*38ac4093SArchibald Elliott EXPECT_NE(GS, nullptr);
70*38ac4093SArchibald Elliott EXPECT_NE(GS, &TestA);
71c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "LibCall");
72c1db8cf9SFrederich Munch
73c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA"));
74*38ac4093SArchibald Elliott EXPECT_NE(GS, nullptr);
75*38ac4093SArchibald Elliott EXPECT_NE(GS, &TestA);
76c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "LibCall");
77c1db8cf9SFrederich Munch
78c1db8cf9SFrederich Munch DL = DynamicLibrary::getPermanentLibrary(nullptr, &Err);
79c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid());
80c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty());
81c1db8cf9SFrederich Munch
825fdd2cbaSFrederich Munch // Test overloading local symbols does not occur by default
83c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA"));
84*38ac4093SArchibald Elliott EXPECT_NE(GS, nullptr);
85*38ac4093SArchibald Elliott EXPECT_EQ(GS, &TestA);
86c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "ProcessCall");
87c1db8cf9SFrederich Munch
88c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA"));
89*38ac4093SArchibald Elliott EXPECT_NE(GS, nullptr);
90*38ac4093SArchibald Elliott EXPECT_EQ(GS, &TestA);
91c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "ProcessCall");
92c1db8cf9SFrederich Munch
935fdd2cbaSFrederich Munch // Test overloading by forcing library priority when searching for a symbol
945fdd2cbaSFrederich Munch DynamicLibrary::SearchOrder = DynamicLibrary::SO_LoadedFirst;
955fdd2cbaSFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA"));
96*38ac4093SArchibald Elliott EXPECT_NE(GS, nullptr);
97*38ac4093SArchibald Elliott EXPECT_NE(GS, &TestA);
985fdd2cbaSFrederich Munch EXPECT_EQ(StdString(GS()), "LibCall");
995fdd2cbaSFrederich Munch
100c1db8cf9SFrederich Munch DynamicLibrary::AddSymbol("TestA", PtrFunc(&OverloadTestA));
101c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA"));
102*38ac4093SArchibald Elliott EXPECT_NE(GS, nullptr);
103*38ac4093SArchibald Elliott EXPECT_NE(GS, &OverloadTestA);
104c1db8cf9SFrederich Munch
105c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA"));
106*38ac4093SArchibald Elliott EXPECT_NE(GS, nullptr);
107*38ac4093SArchibald Elliott EXPECT_EQ(GS, &OverloadTestA);
108c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "OverloadCall");
109c1db8cf9SFrederich Munch }
110c1db8cf9SFrederich Munch EXPECT_TRUE(FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol(
111c1db8cf9SFrederich Munch "TestA")) == nullptr);
1125fdd2cbaSFrederich Munch
1135fdd2cbaSFrederich Munch // Check serach ordering is reset to default after call to llvm_shutdown
114*38ac4093SArchibald Elliott EXPECT_EQ(DynamicLibrary::SearchOrder, DynamicLibrary::SO_Linker);
115c1db8cf9SFrederich Munch }
116c1db8cf9SFrederich Munch
TEST(DynamicLibrary,Shutdown)117c1db8cf9SFrederich Munch TEST(DynamicLibrary, Shutdown) {
118ad125800SFrederich Munch std::string A("PipSqueak"), B, C("SecondLib");
119ad125800SFrederich Munch std::vector<std::string> Order;
120c1db8cf9SFrederich Munch {
121c1db8cf9SFrederich Munch std::string Err;
122c1db8cf9SFrederich Munch llvm_shutdown_obj Shutdown;
123c1db8cf9SFrederich Munch DynamicLibrary DL =
124ad125800SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath(A).c_str(), &Err);
125c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid());
126c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty());
127c1db8cf9SFrederich Munch
128ad125800SFrederich Munch SetStrings SS_0 = FuncPtr<SetStrings>(
129c1db8cf9SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("SetStrings"));
130*38ac4093SArchibald Elliott EXPECT_NE(SS_0, nullptr);
131c1db8cf9SFrederich Munch
132ad125800SFrederich Munch SS_0(A, B);
133ad125800SFrederich Munch EXPECT_EQ(B, "Local::Local(PipSqueak)");
134ad125800SFrederich Munch
135ad125800SFrederich Munch TestOrder TO_0 = FuncPtr<TestOrder>(
136ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("TestOrder"));
137*38ac4093SArchibald Elliott EXPECT_NE(TO_0, nullptr);
138ad125800SFrederich Munch
139ad125800SFrederich Munch DynamicLibrary DL2 =
140ad125800SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath(C).c_str(), &Err);
141ad125800SFrederich Munch EXPECT_TRUE(DL2.isValid());
142ad125800SFrederich Munch EXPECT_TRUE(Err.empty());
143ad125800SFrederich Munch
144ad125800SFrederich Munch // Should find latest version of symbols in SecondLib
145ad125800SFrederich Munch SetStrings SS_1 = FuncPtr<SetStrings>(
146ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("SetStrings"));
147*38ac4093SArchibald Elliott EXPECT_NE(SS_1, nullptr);
148*38ac4093SArchibald Elliott EXPECT_NE(SS_0, SS_1);
149ad125800SFrederich Munch
150ad125800SFrederich Munch TestOrder TO_1 = FuncPtr<TestOrder>(
151ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("TestOrder"));
152*38ac4093SArchibald Elliott EXPECT_NE(TO_1, nullptr);
153*38ac4093SArchibald Elliott EXPECT_NE(TO_0, TO_1);
154ad125800SFrederich Munch
155ad125800SFrederich Munch B.clear();
156ad125800SFrederich Munch SS_1(C, B);
157ad125800SFrederich Munch EXPECT_EQ(B, "Local::Local(SecondLib)");
158ad125800SFrederich Munch
159ad125800SFrederich Munch TO_0(Order);
160ad125800SFrederich Munch TO_1(Order);
161c1db8cf9SFrederich Munch }
162c1db8cf9SFrederich Munch EXPECT_EQ(A, "Global::~Global");
163c1db8cf9SFrederich Munch EXPECT_EQ(B, "Local::~Local");
164*38ac4093SArchibald Elliott EXPECT_EQ(FuncPtr<SetStrings>(
165*38ac4093SArchibald Elliott DynamicLibrary::SearchForAddressOfSymbol("SetStrings")),
166*38ac4093SArchibald Elliott nullptr);
167ad125800SFrederich Munch
168ad125800SFrederich Munch // Test unload/destruction ordering
169ad125800SFrederich Munch EXPECT_EQ(Order.size(), 2UL);
170ad125800SFrederich Munch EXPECT_EQ(Order.front(), "SecondLib");
171ad125800SFrederich Munch EXPECT_EQ(Order.back(), "PipSqueak");
172c1db8cf9SFrederich Munch }
173c1db8cf9SFrederich Munch
174c1db8cf9SFrederich Munch #else
175c1db8cf9SFrederich Munch
TEST(DynamicLibrary,Unsupported)176c1db8cf9SFrederich Munch TEST(DynamicLibrary, Unsupported) {
177c1db8cf9SFrederich Munch std::string Err;
178c1db8cf9SFrederich Munch DynamicLibrary DL =
179c1db8cf9SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err);
180c1db8cf9SFrederich Munch EXPECT_FALSE(DL.isValid());
181c1db8cf9SFrederich Munch EXPECT_EQ(Err, "dlopen() not supported on this platform");
182c1db8cf9SFrederich Munch }
183c1db8cf9SFrederich Munch
184c1db8cf9SFrederich Munch #endif
185