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/Config/config.h" 11c1db8cf9SFrederich Munch #include "llvm/Support/DynamicLibrary.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 #include <string> 19c1db8cf9SFrederich Munch 20c1db8cf9SFrederich Munch using namespace llvm; 21c1db8cf9SFrederich Munch using namespace llvm::sys; 22c1db8cf9SFrederich Munch 23c1db8cf9SFrederich Munch extern "C" PIPSQUEAK_EXPORT const char *TestA() { return "ProcessCall"; } 24c1db8cf9SFrederich Munch 25c1db8cf9SFrederich Munch std::string LibPath() { 26*287a9ea0SDimitry Andric const std::vector<testing::internal::string>& Argvs = testing::internal::GetArgvs(); 27*287a9ea0SDimitry Andric const char *Argv0 = Argvs.size() > 0 ? Argvs[0].c_str() : "DynamicLibraryTests"; 28eab432a1SFrederich Munch void *Ptr = (void*)(intptr_t)TestA; 29*287a9ea0SDimitry Andric std::string Path = fs::getMainExecutable(Argv0, Ptr); 30c1db8cf9SFrederich Munch llvm::SmallString<256> Buf(path::parent_path(Path)); 31c1db8cf9SFrederich Munch path::append(Buf, "PipSqueak.so"); 32c1db8cf9SFrederich Munch return Buf.str(); 33c1db8cf9SFrederich Munch } 34c1db8cf9SFrederich Munch 35c1db8cf9SFrederich Munch #if defined(_WIN32) || (defined(HAVE_DLFCN_H) && defined(HAVE_DLOPEN)) 36c1db8cf9SFrederich Munch 37c1db8cf9SFrederich Munch typedef void (*SetStrings)(std::string &GStr, std::string &LStr); 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) { 103c1db8cf9SFrederich Munch std::string A, B; 104c1db8cf9SFrederich Munch { 105c1db8cf9SFrederich Munch std::string Err; 106c1db8cf9SFrederich Munch llvm_shutdown_obj Shutdown; 107c1db8cf9SFrederich Munch DynamicLibrary DL = 108c1db8cf9SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err); 109c1db8cf9SFrederich Munch EXPECT_TRUE(DL.isValid()); 110c1db8cf9SFrederich Munch EXPECT_TRUE(Err.empty()); 111c1db8cf9SFrederich Munch 112c1db8cf9SFrederich Munch SetStrings SS = FuncPtr<SetStrings>( 113c1db8cf9SFrederich Munch DynamicLibrary::SearchForAddressOfSymbol("SetStrings")); 114c1db8cf9SFrederich Munch EXPECT_TRUE(SS != nullptr); 115c1db8cf9SFrederich Munch 116c1db8cf9SFrederich Munch SS(A, B); 117c1db8cf9SFrederich Munch EXPECT_EQ(B, "Local::Local"); 118c1db8cf9SFrederich Munch } 119c1db8cf9SFrederich Munch EXPECT_EQ(A, "Global::~Global"); 120c1db8cf9SFrederich Munch EXPECT_EQ(B, "Local::~Local"); 121c1db8cf9SFrederich Munch EXPECT_TRUE(FuncPtr<SetStrings>(DynamicLibrary::SearchForAddressOfSymbol( 122c1db8cf9SFrederich Munch "SetStrings")) == nullptr); 123c1db8cf9SFrederich Munch } 124c1db8cf9SFrederich Munch 125c1db8cf9SFrederich Munch #else 126c1db8cf9SFrederich Munch 127c1db8cf9SFrederich Munch TEST(DynamicLibrary, Unsupported) { 128c1db8cf9SFrederich Munch std::string Err; 129c1db8cf9SFrederich Munch DynamicLibrary DL = 130c1db8cf9SFrederich Munch DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err); 131c1db8cf9SFrederich Munch EXPECT_FALSE(DL.isValid()); 132c1db8cf9SFrederich Munch EXPECT_EQ(Err, "dlopen() not supported on this platform"); 133c1db8cf9SFrederich Munch } 134c1db8cf9SFrederich Munch 135c1db8cf9SFrederich Munch #endif 136