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" 119a67b073SChandler 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 22ad125800SFrederich Munch std::string LibPath(const std::string Name = "PipSqueak") { 23287a9ea0SDimitry Andric const std::vector<testing::internal::string>& Argvs = testing::internal::GetArgvs(); 24287a9ea0SDimitry Andric const char *Argv0 = 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)); 28ad125800SFrederich Munch path::append(Buf, (Name+".so").c_str()); 29c1db8cf9SFrederich Munch return 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 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 } 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 55c1db8cf9SFrederich Munch static const char *OverloadTestA() { return "OverloadCall"; } 56c1db8cf9SFrederich Munch 57c1db8cf9SFrederich Munch std::string StdString(const char *Ptr) { return Ptr ? Ptr : ""; } 58c1db8cf9SFrederich Munch 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")); 69c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS != &TestA); 70c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "LibCall"); 71c1db8cf9SFrederich Munch 72c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA")); 73c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS != &TestA); 74c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "LibCall"); 75c1db8cf9SFrederich Munch 76c1db8cf9SFrederich Munch DL = DynamicLibrary::getPermanentLibrary(nullptr, &Err); 77c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid()); 78c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty()); 79c1db8cf9SFrederich Munch 80*5fdd2cbaSFrederich Munch // Test overloading local symbols does not occur by default 81c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA")); 82c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS == &TestA); 83c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "ProcessCall"); 84c1db8cf9SFrederich Munch 85c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA")); 86c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS == &TestA); 87c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "ProcessCall"); 88c1db8cf9SFrederich Munch 89*5fdd2cbaSFrederich Munch // Test overloading by forcing library priority when searching for a symbol 90*5fdd2cbaSFrederich Munch DynamicLibrary::SearchOrder = DynamicLibrary::SO_LoadedFirst; 91*5fdd2cbaSFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA")); 92*5fdd2cbaSFrederich Munch EXPECT_TRUE(GS != nullptr && GS != &TestA); 93*5fdd2cbaSFrederich Munch EXPECT_EQ(StdString(GS()), "LibCall"); 94*5fdd2cbaSFrederich Munch 95c1db8cf9SFrederich Munch DynamicLibrary::AddSymbol("TestA", PtrFunc(&OverloadTestA)); 96c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA")); 97c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS != &OverloadTestA); 98c1db8cf9SFrederich Munch 99c1db8cf9SFrederich Munch GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA")); 100c1db8cf9SFrederich Munch EXPECT_TRUE(GS != nullptr && GS == &OverloadTestA); 101c1db8cf9SFrederich Munch EXPECT_EQ(StdString(GS()), "OverloadCall"); 102c1db8cf9SFrederich Munch } 103c1db8cf9SFrederich Munch EXPECT_TRUE(FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol( 104c1db8cf9SFrederich Munch "TestA")) == nullptr); 105*5fdd2cbaSFrederich Munch 106*5fdd2cbaSFrederich Munch // Check serach ordering is reset to default after call to llvm_shutdown 107*5fdd2cbaSFrederich Munch EXPECT_TRUE(DynamicLibrary::SearchOrder == DynamicLibrary::SO_Linker); 108c1db8cf9SFrederich Munch } 109c1db8cf9SFrederich Munch 110c1db8cf9SFrederich Munch TEST(DynamicLibrary, Shutdown) { 111ad125800SFrederich Munch std::string A("PipSqueak"), B, C("SecondLib"); 112ad125800SFrederich Munch std::vector<std::string> Order; 113c1db8cf9SFrederich Munch { 114c1db8cf9SFrederich Munch std::string Err; 115c1db8cf9SFrederich Munch llvm_shutdown_obj Shutdown; 116c1db8cf9SFrederich Munch DynamicLibrary DL = 117ad125800SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath(A).c_str(), &Err); 118c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid()); 119c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty()); 120c1db8cf9SFrederich Munch 121ad125800SFrederich Munch SetStrings SS_0 = FuncPtr<SetStrings>( 122c1db8cf9SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("SetStrings")); 123ad125800SFrederich Munch EXPECT_TRUE(SS_0 != nullptr); 124c1db8cf9SFrederich Munch 125ad125800SFrederich Munch SS_0(A, B); 126ad125800SFrederich Munch EXPECT_EQ(B, "Local::Local(PipSqueak)"); 127ad125800SFrederich Munch 128ad125800SFrederich Munch TestOrder TO_0 = FuncPtr<TestOrder>( 129ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("TestOrder")); 130ad125800SFrederich Munch EXPECT_TRUE(TO_0 != nullptr); 131ad125800SFrederich Munch 132ad125800SFrederich Munch DynamicLibrary DL2 = 133ad125800SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath(C).c_str(), &Err); 134ad125800SFrederich Munch EXPECT_TRUE(DL2.isValid()); 135ad125800SFrederich Munch EXPECT_TRUE(Err.empty()); 136ad125800SFrederich Munch 137ad125800SFrederich Munch // Should find latest version of symbols in SecondLib 138ad125800SFrederich Munch SetStrings SS_1 = FuncPtr<SetStrings>( 139ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("SetStrings")); 140ad125800SFrederich Munch EXPECT_TRUE(SS_1 != nullptr); 141ad125800SFrederich Munch EXPECT_TRUE(SS_0 != SS_1); 142ad125800SFrederich Munch 143ad125800SFrederich Munch TestOrder TO_1 = FuncPtr<TestOrder>( 144ad125800SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("TestOrder")); 145ad125800SFrederich Munch EXPECT_TRUE(TO_1 != nullptr); 146ad125800SFrederich Munch EXPECT_TRUE(TO_0 != TO_1); 147ad125800SFrederich Munch 148ad125800SFrederich Munch B.clear(); 149ad125800SFrederich Munch SS_1(C, B); 150ad125800SFrederich Munch EXPECT_EQ(B, "Local::Local(SecondLib)"); 151ad125800SFrederich Munch 152ad125800SFrederich Munch TO_0(Order); 153ad125800SFrederich Munch TO_1(Order); 154c1db8cf9SFrederich Munch } 155c1db8cf9SFrederich Munch EXPECT_EQ(A, "Global::~Global"); 156c1db8cf9SFrederich Munch EXPECT_EQ(B, "Local::~Local"); 157c1db8cf9SFrederich Munch EXPECT_TRUE(FuncPtr<SetStrings>(DynamicLibrary::SearchForAddressOfSymbol( 158c1db8cf9SFrederich Munch "SetStrings")) == nullptr); 159ad125800SFrederich Munch 160ad125800SFrederich Munch // Test unload/destruction ordering 161ad125800SFrederich Munch EXPECT_EQ(Order.size(), 2UL); 162ad125800SFrederich Munch EXPECT_EQ(Order.front(), "SecondLib"); 163ad125800SFrederich Munch EXPECT_EQ(Order.back(), "PipSqueak"); 164c1db8cf9SFrederich Munch } 165c1db8cf9SFrederich Munch 166c1db8cf9SFrederich Munch #else 167c1db8cf9SFrederich Munch 168c1db8cf9SFrederich Munch TEST(DynamicLibrary, Unsupported) { 169c1db8cf9SFrederich Munch std::string Err; 170c1db8cf9SFrederich Munch DynamicLibrary DL = 171c1db8cf9SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err); 172c1db8cf9SFrederich Munch EXPECT_FALSE(DL.isValid()); 173c1db8cf9SFrederich Munch EXPECT_EQ(Err, "dlopen() not supported on this platform"); 174c1db8cf9SFrederich Munch } 175c1db8cf9SFrederich Munch 176c1db8cf9SFrederich Munch #endif 177