1c1db8cf9SFrederich Munch //===- llvm/unittest/Support/DynamicLibrary/DynamicLibraryTest.cpp --------===// 2c1db8cf9SFrederich Munch // 3c1db8cf9SFrederich Munch // The LLVM Compiler Infrastructure 4c1db8cf9SFrederich Munch // 5c1db8cf9SFrederich Munch // This file is distributed under the University of Illinois Open Source 6c1db8cf9SFrederich Munch // License. See LICENSE.TXT for details. 7c1db8cf9SFrederich Munch // 8c1db8cf9SFrederich Munch //===----------------------------------------------------------------------===// 9c1db8cf9SFrederich Munch 10c1db8cf9SFrederich Munch #include "llvm/Support/DynamicLibrary.h" 11*9a67b073SChandler Carruth #include "llvm/Config/config.h" 12c1db8cf9SFrederich Munch #include "llvm/Support/FileSystem.h" 13c1db8cf9SFrederich Munch #include "llvm/Support/ManagedStatic.h" 14c1db8cf9SFrederich Munch #include "llvm/Support/Path.h" 15c1db8cf9SFrederich Munch #include "gtest/gtest.h" 16c1db8cf9SFrederich Munch 17c1db8cf9SFrederich Munch #include "PipSqueak.h" 18c1db8cf9SFrederich Munch 19c1db8cf9SFrederich Munch using namespace llvm; 20c1db8cf9SFrederich Munch using namespace llvm::sys; 21c1db8cf9SFrederich Munch 22c1db8cf9SFrederich Munch extern "C" PIPSQUEAK_EXPORT const char *TestA() { return "ProcessCall"; } 23c1db8cf9SFrederich Munch 24ad125800SFrederich Munch std::string LibPath(const std::string Name = "PipSqueak") { 25287a9ea0SDimitry Andric const std::vector<testing::internal::string>& Argvs = testing::internal::GetArgvs(); 26287a9ea0SDimitry Andric const char *Argv0 = Argvs.size() > 0 ? Argvs[0].c_str() : "DynamicLibraryTests"; 27eab432a1SFrederich Munch void *Ptr = (void*)(intptr_t)TestA; 28287a9ea0SDimitry Andric std::string Path = fs::getMainExecutable(Argv0, Ptr); 29c1db8cf9SFrederich Munch llvm::SmallString<256> Buf(path::parent_path(Path)); 30ad125800SFrederich Munch path::append(Buf, (Name+".so").c_str()); 31c1db8cf9SFrederich Munch return Buf.str(); 32c1db8cf9SFrederich Munch } 33c1db8cf9SFrederich Munch 34c1db8cf9SFrederich Munch #if defined(_WIN32) || (defined(HAVE_DLFCN_H) && defined(HAVE_DLOPEN)) 35c1db8cf9SFrederich Munch 36c1db8cf9SFrederich Munch typedef void (*SetStrings)(std::string &GStr, std::string &LStr); 37ad125800SFrederich Munch typedef void (*TestOrder)(std::vector<std::string> &V); 38c1db8cf9SFrederich Munch typedef const char *(*GetString)(); 39c1db8cf9SFrederich Munch 40c1db8cf9SFrederich Munch template <class T> static T FuncPtr(void *Ptr) { 41c1db8cf9SFrederich Munch union { 42c1db8cf9SFrederich Munch T F; 43c1db8cf9SFrederich Munch void *P; 44c1db8cf9SFrederich Munch } Tmp; 45c1db8cf9SFrederich Munch Tmp.P = Ptr; 46c1db8cf9SFrederich Munch return Tmp.F; 47c1db8cf9SFrederich Munch } 48c1db8cf9SFrederich Munch template <class T> static void* PtrFunc(T *Func) { 49c1db8cf9SFrederich Munch union { 50c1db8cf9SFrederich Munch T *F; 51c1db8cf9SFrederich Munch void *P; 52c1db8cf9SFrederich Munch } Tmp; 53c1db8cf9SFrederich Munch Tmp.F = Func; 54c1db8cf9SFrederich Munch return Tmp.P; 55c1db8cf9SFrederich Munch } 56c1db8cf9SFrederich Munch 57c1db8cf9SFrederich Munch static const char *OverloadTestA() { return "OverloadCall"; } 58c1db8cf9SFrederich Munch 59c1db8cf9SFrederich Munch std::string StdString(const char *Ptr) { return Ptr ? Ptr : ""; } 60c1db8cf9SFrederich Munch 61c1db8cf9SFrederich Munch TEST(DynamicLibrary, Overload) { 62c1db8cf9SFrederich Munch { 63c1db8cf9SFrederich Munch std::string Err; 64c1db8cf9SFrederich Munch llvm_shutdown_obj Shutdown; 65c1db8cf9SFrederich Munch DynamicLibrary DL = 66c1db8cf9SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err); 67c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid()); 68c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty()); 69c1db8cf9SFrederich Munch 70c1db8cf9SFrederich Munch GetString GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA")); 71c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS != &TestA); 72c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "LibCall"); 73c1db8cf9SFrederich Munch 74c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA")); 75c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && 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 82c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA")); 83c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS == &TestA); 84c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "ProcessCall"); 85c1db8cf9SFrederich Munch 86c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA")); 87c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS == &TestA); 88c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "ProcessCall"); 89c1db8cf9SFrederich Munch 90c1db8cf9SFrederich Munch DynamicLibrary::AddSymbol("TestA", PtrFunc(&OverloadTestA)); 91c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA")); 92c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS != &OverloadTestA); 93c1db8cf9SFrederich Munch 94c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA")); 95c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS == &OverloadTestA); 96c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "OverloadCall"); 97c1db8cf9SFrederich Munch } 98c1db8cf9SFrederich Munch EXPECT_TRUE(FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol( 99c1db8cf9SFrederich Munch "TestA")) == nullptr); 100c1db8cf9SFrederich Munch } 101c1db8cf9SFrederich Munch 102c1db8cf9SFrederich Munch TEST(DynamicLibrary, Shutdown) { 103ad125800SFrederich Munch std::string A("PipSqueak"), B, C("SecondLib"); 104ad125800SFrederich Munch std::vector<std::string> Order; 105c1db8cf9SFrederich Munch { 106c1db8cf9SFrederich Munch std::string Err; 107c1db8cf9SFrederich Munch llvm_shutdown_obj Shutdown; 108c1db8cf9SFrederich Munch DynamicLibrary DL = 109ad125800SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath(A).c_str(), &Err); 110c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid()); 111c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty()); 112c1db8cf9SFrederich Munch 113ad125800SFrederich Munch SetStrings SS_0 = FuncPtr<SetStrings>( 114c1db8cf9SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("SetStrings")); 115ad125800SFrederich Munch EXPECT_TRUE(SS_0 != nullptr); 116c1db8cf9SFrederich Munch 117ad125800SFrederich Munch SS_0(A, B); 118ad125800SFrederich Munch EXPECT_EQ(B, "Local::Local(PipSqueak)"); 119ad125800SFrederich Munch 120ad125800SFrederich Munch TestOrder TO_0 = FuncPtr<TestOrder>( 121ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("TestOrder")); 122ad125800SFrederich Munch EXPECT_TRUE(TO_0 != nullptr); 123ad125800SFrederich Munch 124ad125800SFrederich Munch DynamicLibrary DL2 = 125ad125800SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath(C).c_str(), &Err); 126ad125800SFrederich Munch EXPECT_TRUE(DL2.isValid()); 127ad125800SFrederich Munch EXPECT_TRUE(Err.empty()); 128ad125800SFrederich Munch 129ad125800SFrederich Munch // Should find latest version of symbols in SecondLib 130ad125800SFrederich Munch SetStrings SS_1 = FuncPtr<SetStrings>( 131ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("SetStrings")); 132ad125800SFrederich Munch EXPECT_TRUE(SS_1 != nullptr); 133ad125800SFrederich Munch EXPECT_TRUE(SS_0 != SS_1); 134ad125800SFrederich Munch 135ad125800SFrederich Munch TestOrder TO_1 = FuncPtr<TestOrder>( 136ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("TestOrder")); 137ad125800SFrederich Munch EXPECT_TRUE(TO_1 != nullptr); 138ad125800SFrederich Munch EXPECT_TRUE(TO_0 != TO_1); 139ad125800SFrederich Munch 140ad125800SFrederich Munch B.clear(); 141ad125800SFrederich Munch SS_1(C, B); 142ad125800SFrederich Munch EXPECT_EQ(B, "Local::Local(SecondLib)"); 143ad125800SFrederich Munch 144ad125800SFrederich Munch TO_0(Order); 145ad125800SFrederich Munch TO_1(Order); 146c1db8cf9SFrederich Munch } 147c1db8cf9SFrederich Munch EXPECT_EQ(A, "Global::~Global"); 148c1db8cf9SFrederich Munch EXPECT_EQ(B, "Local::~Local"); 149c1db8cf9SFrederich Munch EXPECT_TRUE(FuncPtr<SetStrings>(DynamicLibrary::SearchForAddressOfSymbol( 150c1db8cf9SFrederich Munch "SetStrings")) == nullptr); 151ad125800SFrederich Munch 152ad125800SFrederich Munch // Test unload/destruction ordering 153ad125800SFrederich Munch EXPECT_EQ(Order.size(), 2UL); 154ad125800SFrederich Munch EXPECT_EQ(Order.front(), "SecondLib"); 155ad125800SFrederich Munch EXPECT_EQ(Order.back(), "PipSqueak"); 156c1db8cf9SFrederich Munch } 157c1db8cf9SFrederich Munch 158c1db8cf9SFrederich Munch #else 159c1db8cf9SFrederich Munch 160c1db8cf9SFrederich Munch TEST(DynamicLibrary, Unsupported) { 161c1db8cf9SFrederich Munch std::string Err; 162c1db8cf9SFrederich Munch DynamicLibrary DL = 163c1db8cf9SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err); 164c1db8cf9SFrederich Munch EXPECT_FALSE(DL.isValid()); 165c1db8cf9SFrederich Munch EXPECT_EQ(Err, "dlopen() not supported on this platform"); 166c1db8cf9SFrederich Munch } 167c1db8cf9SFrederich Munch 168c1db8cf9SFrederich Munch #endif 169