DynamicLibraryTest.cpp 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  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/Config/config.h"
  10. #include "llvm/Support/DynamicLibrary.h"
  11. #include "llvm/Support/FileSystem.h"
  12. #include "llvm/Support/ManagedStatic.h"
  13. #include "llvm/Support/Path.h"
  14. #include "gtest/gtest.h"
  15. #include "PipSqueak.h"
  16. #include <string>
  17. using namespace llvm;
  18. using namespace llvm::sys;
  19. extern "C" PIPSQUEAK_EXPORT const char *TestA() { return "ProcessCall"; }
  20. std::string LibPath() {
  21. const std::vector<testing::internal::string>& Argvs = testing::internal::GetArgvs();
  22. const char *Argv0 = Argvs.size() > 0 ? Argvs[0].c_str() : "DynamicLibraryTests";
  23. void *Ptr = (void*)(intptr_t)TestA;
  24. std::string Path = fs::getMainExecutable(Argv0, Ptr);
  25. llvm::SmallString<256> Buf(path::parent_path(Path));
  26. path::append(Buf, "PipSqueak.so");
  27. return Buf.str();
  28. }
  29. #if defined(_WIN32) || (defined(HAVE_DLFCN_H) && defined(HAVE_DLOPEN))
  30. typedef void (*SetStrings)(std::string &GStr, std::string &LStr);
  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 const char *OverloadTestA() { return "OverloadCall"; }
  49. std::string StdString(const char *Ptr) { return Ptr ? Ptr : ""; }
  50. TEST(DynamicLibrary, Overload) {
  51. {
  52. std::string Err;
  53. llvm_shutdown_obj Shutdown;
  54. DynamicLibrary DL =
  55. DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err);
  56. EXPECT_TRUE(DL.isValid());
  57. EXPECT_TRUE(Err.empty());
  58. GetString GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA"));
  59. EXPECT_TRUE(GS != nullptr && GS != &TestA);
  60. EXPECT_EQ(StdString(GS()), "LibCall");
  61. GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA"));
  62. EXPECT_TRUE(GS != nullptr && GS != &TestA);
  63. EXPECT_EQ(StdString(GS()), "LibCall");
  64. DL = DynamicLibrary::getPermanentLibrary(nullptr, &Err);
  65. EXPECT_TRUE(DL.isValid());
  66. EXPECT_TRUE(Err.empty());
  67. GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA"));
  68. EXPECT_TRUE(GS != nullptr && GS == &TestA);
  69. EXPECT_EQ(StdString(GS()), "ProcessCall");
  70. GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA"));
  71. EXPECT_TRUE(GS != nullptr && GS == &TestA);
  72. EXPECT_EQ(StdString(GS()), "ProcessCall");
  73. DynamicLibrary::AddSymbol("TestA", PtrFunc(&OverloadTestA));
  74. GS = FuncPtr<GetString>(DL.getAddressOfSymbol("TestA"));
  75. EXPECT_TRUE(GS != nullptr && GS != &OverloadTestA);
  76. GS = FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol("TestA"));
  77. EXPECT_TRUE(GS != nullptr && GS == &OverloadTestA);
  78. EXPECT_EQ(StdString(GS()), "OverloadCall");
  79. }
  80. EXPECT_TRUE(FuncPtr<GetString>(DynamicLibrary::SearchForAddressOfSymbol(
  81. "TestA")) == nullptr);
  82. }
  83. TEST(DynamicLibrary, Shutdown) {
  84. std::string A, B;
  85. {
  86. std::string Err;
  87. llvm_shutdown_obj Shutdown;
  88. DynamicLibrary DL =
  89. DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err);
  90. EXPECT_TRUE(DL.isValid());
  91. EXPECT_TRUE(Err.empty());
  92. SetStrings SS = FuncPtr<SetStrings>(
  93. DynamicLibrary::SearchForAddressOfSymbol("SetStrings"));
  94. EXPECT_TRUE(SS != nullptr);
  95. SS(A, B);
  96. EXPECT_EQ(B, "Local::Local");
  97. }
  98. EXPECT_EQ(A, "Global::~Global");
  99. EXPECT_EQ(B, "Local::~Local");
  100. EXPECT_TRUE(FuncPtr<SetStrings>(DynamicLibrary::SearchForAddressOfSymbol(
  101. "SetStrings")) == nullptr);
  102. }
  103. #else
  104. TEST(DynamicLibrary, Unsupported) {
  105. std::string Err;
  106. DynamicLibrary DL =
  107. DynamicLibrary::getPermanentLibrary(LibPath().c_str(), &Err);
  108. EXPECT_FALSE(DL.isValid());
  109. EXPECT_EQ(Err, "dlopen() not supported on this platform");
  110. }
  111. #endif