DynamicLibraryTest.cpp 5.8 KB

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