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