ObjectTransformLayerTest.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. //===- ObjectTransformLayerTest.cpp - Unit tests for ObjectTransformLayer -===//
  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/ADT/STLExtras.h"
  10. #include "llvm/ADT/SmallVector.h"
  11. #include "llvm/ExecutionEngine/Orc/CompileUtils.h"
  12. #include "llvm/ExecutionEngine/Orc/IRCompileLayer.h"
  13. #include "llvm/ExecutionEngine/Orc/NullResolver.h"
  14. #include "llvm/ExecutionEngine/Orc/ObjectLinkingLayer.h"
  15. #include "llvm/ExecutionEngine/Orc/ObjectTransformLayer.h"
  16. #include "llvm/Object/ObjectFile.h"
  17. #include "gtest/gtest.h"
  18. using namespace llvm::orc;
  19. namespace {
  20. // Stand-in for RuntimeDyld::MemoryManager
  21. typedef int MockMemoryManager;
  22. // Stand-in for RuntimeDyld::SymbolResolver
  23. typedef int MockSymbolResolver;
  24. // stand-in for object::ObjectFile
  25. typedef int MockObjectFile;
  26. // stand-in for llvm::MemoryBuffer set
  27. typedef int MockMemoryBufferSet;
  28. // Mock transform that operates on unique pointers to object files, and
  29. // allocates new object files rather than mutating the given ones.
  30. struct AllocatingTransform {
  31. std::unique_ptr<MockObjectFile>
  32. operator()(std::unique_ptr<MockObjectFile> Obj) const {
  33. return llvm::make_unique<MockObjectFile>(*Obj + 1);
  34. }
  35. };
  36. // Mock base layer for verifying behavior of transform layer.
  37. // Each method "T foo(args)" is accompanied by two auxiliary methods:
  38. // - "void expectFoo(args)", to be called before calling foo on the transform
  39. // layer; saves values of args, which mock layer foo then verifies against.
  40. // - "void verifyFoo(T)", to be called after foo, which verifies that the
  41. // transform layer called the base layer and forwarded any return value.
  42. class MockBaseLayer {
  43. public:
  44. typedef int ObjSetHandleT;
  45. MockBaseLayer() : MockSymbol(nullptr) { resetExpectations(); }
  46. template <typename ObjSetT, typename MemoryManagerPtrT,
  47. typename SymbolResolverPtrT>
  48. ObjSetHandleT addObjectSet(ObjSetT Objects, MemoryManagerPtrT MemMgr,
  49. SymbolResolverPtrT Resolver) {
  50. EXPECT_EQ(MockManager, *MemMgr) << "MM should pass through";
  51. EXPECT_EQ(MockResolver, *Resolver) << "Resolver should pass through";
  52. size_t I = 0;
  53. for (auto &ObjPtr : Objects) {
  54. EXPECT_EQ(MockObjects[I++] + 1, *ObjPtr) << "Transform should be applied";
  55. }
  56. EXPECT_EQ(MockObjects.size(), I) << "Number of objects should match";
  57. LastCalled = "addObjectSet";
  58. MockObjSetHandle = 111;
  59. return MockObjSetHandle;
  60. }
  61. template <typename ObjSetT>
  62. void expectAddObjectSet(ObjSetT &Objects, MockMemoryManager *MemMgr,
  63. MockSymbolResolver *Resolver) {
  64. MockManager = *MemMgr;
  65. MockResolver = *Resolver;
  66. for (auto &ObjPtr : Objects) {
  67. MockObjects.push_back(*ObjPtr);
  68. }
  69. }
  70. void verifyAddObjectSet(ObjSetHandleT Returned) {
  71. EXPECT_EQ("addObjectSet", LastCalled);
  72. EXPECT_EQ(MockObjSetHandle, Returned) << "Return should pass through";
  73. resetExpectations();
  74. }
  75. void removeObjectSet(ObjSetHandleT H) {
  76. EXPECT_EQ(MockObjSetHandle, H);
  77. LastCalled = "removeObjectSet";
  78. }
  79. void expectRemoveObjectSet(ObjSetHandleT H) { MockObjSetHandle = H; }
  80. void verifyRemoveObjectSet() {
  81. EXPECT_EQ("removeObjectSet", LastCalled);
  82. resetExpectations();
  83. }
  84. JITSymbol findSymbol(const std::string &Name, bool ExportedSymbolsOnly) {
  85. EXPECT_EQ(MockName, Name) << "Name should pass through";
  86. EXPECT_EQ(MockBool, ExportedSymbolsOnly) << "Flag should pass through";
  87. LastCalled = "findSymbol";
  88. MockSymbol = JITSymbol(122, llvm::JITSymbolFlags::None);
  89. return MockSymbol;
  90. }
  91. void expectFindSymbol(const std::string &Name, bool ExportedSymbolsOnly) {
  92. MockName = Name;
  93. MockBool = ExportedSymbolsOnly;
  94. }
  95. void verifyFindSymbol(llvm::orc::JITSymbol Returned) {
  96. EXPECT_EQ("findSymbol", LastCalled);
  97. EXPECT_EQ(MockSymbol.getAddress(), Returned.getAddress())
  98. << "Return should pass through";
  99. resetExpectations();
  100. }
  101. JITSymbol findSymbolIn(ObjSetHandleT H, const std::string &Name,
  102. bool ExportedSymbolsOnly) {
  103. EXPECT_EQ(MockObjSetHandle, H) << "Handle should pass through";
  104. EXPECT_EQ(MockName, Name) << "Name should pass through";
  105. EXPECT_EQ(MockBool, ExportedSymbolsOnly) << "Flag should pass through";
  106. LastCalled = "findSymbolIn";
  107. MockSymbol = JITSymbol(122, llvm::JITSymbolFlags::None);
  108. return MockSymbol;
  109. }
  110. void expectFindSymbolIn(ObjSetHandleT H, const std::string &Name,
  111. bool ExportedSymbolsOnly) {
  112. MockObjSetHandle = H;
  113. MockName = Name;
  114. MockBool = ExportedSymbolsOnly;
  115. }
  116. void verifyFindSymbolIn(llvm::orc::JITSymbol Returned) {
  117. EXPECT_EQ("findSymbolIn", LastCalled);
  118. EXPECT_EQ(MockSymbol.getAddress(), Returned.getAddress())
  119. << "Return should pass through";
  120. resetExpectations();
  121. }
  122. void emitAndFinalize(ObjSetHandleT H) {
  123. EXPECT_EQ(MockObjSetHandle, H) << "Handle should pass through";
  124. LastCalled = "emitAndFinalize";
  125. }
  126. void expectEmitAndFinalize(ObjSetHandleT H) { MockObjSetHandle = H; }
  127. void verifyEmitAndFinalize() {
  128. EXPECT_EQ("emitAndFinalize", LastCalled);
  129. resetExpectations();
  130. }
  131. void mapSectionAddress(ObjSetHandleT H, const void *LocalAddress,
  132. TargetAddress TargetAddr) {
  133. EXPECT_EQ(MockObjSetHandle, H);
  134. EXPECT_EQ(MockLocalAddress, LocalAddress);
  135. EXPECT_EQ(MockTargetAddress, TargetAddr);
  136. LastCalled = "mapSectionAddress";
  137. }
  138. void expectMapSectionAddress(ObjSetHandleT H, const void *LocalAddress,
  139. TargetAddress TargetAddr) {
  140. MockObjSetHandle = H;
  141. MockLocalAddress = LocalAddress;
  142. MockTargetAddress = TargetAddr;
  143. }
  144. void verifyMapSectionAddress() {
  145. EXPECT_EQ("mapSectionAddress", LastCalled);
  146. resetExpectations();
  147. }
  148. private:
  149. // Backing fields for remembering parameter/return values
  150. std::string LastCalled;
  151. MockMemoryManager MockManager;
  152. MockSymbolResolver MockResolver;
  153. std::vector<MockObjectFile> MockObjects;
  154. ObjSetHandleT MockObjSetHandle;
  155. std::string MockName;
  156. bool MockBool;
  157. JITSymbol MockSymbol;
  158. const void *MockLocalAddress;
  159. TargetAddress MockTargetAddress;
  160. MockMemoryBufferSet MockBufferSet;
  161. // Clear remembered parameters between calls
  162. void resetExpectations() {
  163. LastCalled = "nothing";
  164. MockManager = 0;
  165. MockResolver = 0;
  166. MockObjects.clear();
  167. MockObjSetHandle = 0;
  168. MockName = "bogus";
  169. MockSymbol = JITSymbol(nullptr);
  170. MockLocalAddress = nullptr;
  171. MockTargetAddress = 0;
  172. MockBufferSet = 0;
  173. }
  174. };
  175. // Test each operation on ObjectTransformLayer.
  176. TEST(ObjectTransformLayerTest, Main) {
  177. MockBaseLayer M;
  178. // Create one object transform layer using a transform (as a functor)
  179. // that allocates new objects, and deals in unique pointers.
  180. ObjectTransformLayer<MockBaseLayer, AllocatingTransform> T1(M);
  181. // Create a second object transform layer using a transform (as a lambda)
  182. // that mutates objects in place, and deals in naked pointers
  183. ObjectTransformLayer<MockBaseLayer,
  184. std::function<MockObjectFile *(MockObjectFile *)>>
  185. T2(M, [](MockObjectFile *Obj) {
  186. ++(*Obj);
  187. return Obj;
  188. });
  189. // Instantiate some mock objects to use below
  190. MockObjectFile MockObject1 = 211;
  191. MockObjectFile MockObject2 = 222;
  192. MockMemoryManager MockManager = 233;
  193. MockSymbolResolver MockResolver = 244;
  194. // Test addObjectSet with T1 (allocating, unique pointers)
  195. std::vector<std::unique_ptr<MockObjectFile>> Objs1;
  196. Objs1.push_back(llvm::make_unique<MockObjectFile>(MockObject1));
  197. Objs1.push_back(llvm::make_unique<MockObjectFile>(MockObject2));
  198. auto MM = llvm::make_unique<MockMemoryManager>(MockManager);
  199. auto SR = llvm::make_unique<MockSymbolResolver>(MockResolver);
  200. M.expectAddObjectSet(Objs1, MM.get(), SR.get());
  201. auto H = T1.addObjectSet(std::move(Objs1), std::move(MM), std::move(SR));
  202. M.verifyAddObjectSet(H);
  203. // Test addObjectSet with T2 (mutating, naked pointers)
  204. llvm::SmallVector<MockObjectFile *, 2> Objs2Vec;
  205. Objs2Vec.push_back(&MockObject1);
  206. Objs2Vec.push_back(&MockObject2);
  207. llvm::MutableArrayRef<MockObjectFile *> Objs2(Objs2Vec);
  208. M.expectAddObjectSet(Objs2, &MockManager, &MockResolver);
  209. H = T2.addObjectSet(Objs2, &MockManager, &MockResolver);
  210. M.verifyAddObjectSet(H);
  211. EXPECT_EQ(212, MockObject1) << "Expected mutation";
  212. EXPECT_EQ(223, MockObject2) << "Expected mutation";
  213. // Test removeObjectSet
  214. M.expectRemoveObjectSet(H);
  215. T1.removeObjectSet(H);
  216. M.verifyRemoveObjectSet();
  217. // Test findSymbol
  218. std::string Name = "foo";
  219. bool ExportedOnly = true;
  220. M.expectFindSymbol(Name, ExportedOnly);
  221. JITSymbol Symbol = T2.findSymbol(Name, ExportedOnly);
  222. M.verifyFindSymbol(Symbol);
  223. // Test findSymbolIn
  224. Name = "bar";
  225. ExportedOnly = false;
  226. M.expectFindSymbolIn(H, Name, ExportedOnly);
  227. Symbol = T1.findSymbolIn(H, Name, ExportedOnly);
  228. M.verifyFindSymbolIn(Symbol);
  229. // Test emitAndFinalize
  230. M.expectEmitAndFinalize(H);
  231. T2.emitAndFinalize(H);
  232. M.verifyEmitAndFinalize();
  233. // Test mapSectionAddress
  234. char Buffer[24];
  235. TargetAddress MockAddress = 255;
  236. M.expectMapSectionAddress(H, Buffer, MockAddress);
  237. T1.mapSectionAddress(H, Buffer, MockAddress);
  238. M.verifyMapSectionAddress();
  239. // Verify transform getter (non-const)
  240. MockObjectFile Mutatee = 277;
  241. MockObjectFile *Out = T2.getTransform()(&Mutatee);
  242. EXPECT_EQ(&Mutatee, Out) << "Expected in-place transform";
  243. EXPECT_EQ(278, Mutatee) << "Expected incrementing transform";
  244. // Verify transform getter (const)
  245. auto OwnedObj = llvm::make_unique<MockObjectFile>(288);
  246. const auto &T1C = T1;
  247. OwnedObj = T1C.getTransform()(std::move(OwnedObj));
  248. EXPECT_EQ(289, *OwnedObj) << "Expected incrementing transform";
  249. volatile bool RunStaticChecks = false;
  250. if (!RunStaticChecks)
  251. return;
  252. // Make sure that ObjectTransformLayer implements the object layer concept
  253. // correctly by sandwitching one between an ObjectLinkingLayer and an
  254. // IRCompileLayer, verifying that it compiles if we have a call to the
  255. // IRComileLayer's addModuleSet that should call the transform layer's
  256. // addObjectSet, and also calling the other public transform layer methods
  257. // directly to make sure the methods they intend to forward to exist on
  258. // the ObjectLinkingLayer.
  259. // We'll need a concrete MemoryManager class.
  260. class NullManager : public llvm::RuntimeDyld::MemoryManager {
  261. public:
  262. uint8_t *allocateCodeSection(uintptr_t, unsigned, unsigned,
  263. llvm::StringRef) override {
  264. return nullptr;
  265. }
  266. uint8_t *allocateDataSection(uintptr_t, unsigned, unsigned, llvm::StringRef,
  267. bool) override {
  268. return nullptr;
  269. }
  270. void registerEHFrames(uint8_t *, uint64_t, size_t) override {}
  271. void deregisterEHFrames(uint8_t *, uint64_t, size_t) override {}
  272. bool finalizeMemory(std::string *) override { return false; }
  273. };
  274. // Construct the jit layers.
  275. ObjectLinkingLayer<> BaseLayer;
  276. auto IdentityTransform = [](
  277. std::unique_ptr<llvm::object::OwningBinary<llvm::object::ObjectFile>>
  278. Obj) { return Obj; };
  279. ObjectTransformLayer<decltype(BaseLayer), decltype(IdentityTransform)>
  280. TransformLayer(BaseLayer, IdentityTransform);
  281. auto NullCompiler = [](llvm::Module &) {
  282. return llvm::object::OwningBinary<llvm::object::ObjectFile>();
  283. };
  284. IRCompileLayer<decltype(TransformLayer)> CompileLayer(TransformLayer,
  285. NullCompiler);
  286. // Make sure that the calls from IRCompileLayer to ObjectTransformLayer
  287. // compile.
  288. NullResolver Resolver;
  289. NullManager Manager;
  290. CompileLayer.addModuleSet(std::vector<llvm::Module *>(), &Manager, &Resolver);
  291. // Make sure that the calls from ObjectTransformLayer to ObjectLinkingLayer
  292. // compile.
  293. decltype(TransformLayer)::ObjSetHandleT ObjSet;
  294. TransformLayer.emitAndFinalize(ObjSet);
  295. TransformLayer.findSymbolIn(ObjSet, Name, false);
  296. TransformLayer.findSymbol(Name, true);
  297. TransformLayer.mapSectionAddress(ObjSet, nullptr, 0);
  298. TransformLayer.removeObjectSet(ObjSet);
  299. }
  300. }