Ted Kremenek | bfa82c4 | 2007-10-16 23:37:27 +0000 | [diff] [blame] | 1 | //===--- SerializationTest.cpp - Experimental Object Serialization --------===// |
| 2 | // |
| 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
Chris Lattner | 0bc735f | 2007-12-29 19:59:25 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
Ted Kremenek | bfa82c4 | 2007-10-16 23:37:27 +0000 | [diff] [blame] | 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | // |
| 10 | // This file implements prototype code for serialization of objects in clang. |
| 11 | // It is not intended yet for public use, but simply is a placeholder to |
| 12 | // experiment with new serialization features. Serialization will eventually |
| 13 | // be integrated as a proper component of the clang libraries. |
| 14 | // |
| 15 | //===----------------------------------------------------------------------===// |
| 16 | |
Ted Kremenek | bfa82c4 | 2007-10-16 23:37:27 +0000 | [diff] [blame] | 17 | #include "clang/AST/ASTConsumer.h" |
Ted Kremenek | 5656b78 | 2007-11-29 01:24:25 +0000 | [diff] [blame] | 18 | #include "clang/AST/CFG.h" |
Daniel Dunbar | e91593e | 2008-08-11 04:54:23 +0000 | [diff] [blame] | 19 | #include "clang/AST/Decl.h" |
Ted Kremenek | c4ca288 | 2007-12-03 22:48:14 +0000 | [diff] [blame] | 20 | #include "clang.h" |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 21 | #include "ASTConsumers.h" |
Ted Kremenek | 525fdbc | 2007-12-18 21:36:21 +0000 | [diff] [blame] | 22 | #include "clang/AST/TranslationUnit.h" |
Ted Kremenek | c637e6b | 2007-10-23 22:18:37 +0000 | [diff] [blame] | 23 | #include "llvm/Support/MemoryBuffer.h" |
Ted Kremenek | ee53364 | 2007-12-20 19:47:16 +0000 | [diff] [blame] | 24 | #include "llvm/ADT/OwningPtr.h" |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 25 | #include "llvm/Support/Streams.h" |
Ted Kremenek | ea75c55 | 2007-11-28 21:32:21 +0000 | [diff] [blame] | 26 | #include <fstream> |
Chris Lattner | 87cf5ac | 2008-03-10 17:04:53 +0000 | [diff] [blame] | 27 | #include <cstring> |
Ted Kremenek | bfa82c4 | 2007-10-16 23:37:27 +0000 | [diff] [blame] | 28 | using namespace clang; |
Ted Kremenek | bfa82c4 | 2007-10-16 23:37:27 +0000 | [diff] [blame] | 29 | |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 30 | //===----------------------------------------------------------------------===// |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 31 | // Driver code. |
| 32 | //===----------------------------------------------------------------------===// |
| 33 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 34 | namespace { |
Ted Kremenek | 4ac8121 | 2007-11-05 21:39:35 +0000 | [diff] [blame] | 35 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 36 | class SerializationTest : public ASTConsumer { |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 37 | Diagnostic &Diags; |
| 38 | FileManager &FMgr; |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 39 | public: |
Ted Kremenek | e7d07d1 | 2008-06-04 15:55:15 +0000 | [diff] [blame] | 40 | SerializationTest(Diagnostic &d, FileManager& fmgr) |
Ted Kremenek | e452e0f | 2008-08-08 04:15:52 +0000 | [diff] [blame] | 41 | : Diags(d), FMgr(fmgr) {} |
Ted Kremenek | dca2466 | 2007-12-05 00:26:13 +0000 | [diff] [blame] | 42 | |
Ted Kremenek | e452e0f | 2008-08-08 04:15:52 +0000 | [diff] [blame] | 43 | ~SerializationTest() {} |
| 44 | |
Chris Lattner | dacbc5d | 2009-03-28 04:11:33 +0000 | [diff] [blame] | 45 | virtual void HandleTranslationUnit(ASTContext &C); |
Ted Kremenek | 63ea863 | 2007-12-19 19:27:38 +0000 | [diff] [blame] | 46 | |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 47 | private: |
Ted Kremenek | e452e0f | 2008-08-08 04:15:52 +0000 | [diff] [blame] | 48 | bool Serialize(llvm::sys::Path& Filename, llvm::sys::Path& FNameDeclPrint, |
Chris Lattner | d2fa675 | 2009-03-28 03:56:54 +0000 | [diff] [blame] | 49 | ASTContext &Ctx); |
Ted Kremenek | e452e0f | 2008-08-08 04:15:52 +0000 | [diff] [blame] | 50 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 51 | bool Deserialize(llvm::sys::Path& Filename, llvm::sys::Path& FNameDeclPrint); |
Ted Kremenek | 4ac8121 | 2007-11-05 21:39:35 +0000 | [diff] [blame] | 52 | }; |
| 53 | |
Ted Kremenek | c637e6b | 2007-10-23 22:18:37 +0000 | [diff] [blame] | 54 | } // end anonymous namespace |
Ted Kremenek | bfa82c4 | 2007-10-16 23:37:27 +0000 | [diff] [blame] | 55 | |
Ted Kremenek | dca2466 | 2007-12-05 00:26:13 +0000 | [diff] [blame] | 56 | ASTConsumer* |
Ted Kremenek | e7d07d1 | 2008-06-04 15:55:15 +0000 | [diff] [blame] | 57 | clang::CreateSerializationTest(Diagnostic &Diags, FileManager& FMgr) { |
| 58 | return new SerializationTest(Diags, FMgr); |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 59 | } |
| 60 | |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 61 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 62 | bool SerializationTest::Serialize(llvm::sys::Path& Filename, |
Ted Kremenek | e452e0f | 2008-08-08 04:15:52 +0000 | [diff] [blame] | 63 | llvm::sys::Path& FNameDeclPrint, |
Chris Lattner | d2fa675 | 2009-03-28 03:56:54 +0000 | [diff] [blame] | 64 | ASTContext &Ctx) { |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 65 | { |
| 66 | // Pretty-print the decls to a temp file. |
Ted Kremenek | a95d375 | 2008-09-13 05:16:45 +0000 | [diff] [blame] | 67 | std::string Err; |
Daniel Dunbar | 26fb272 | 2008-11-13 05:09:21 +0000 | [diff] [blame] | 68 | llvm::raw_fd_ostream DeclPP(FNameDeclPrint.c_str(), true, Err); |
Ted Kremenek | a95d375 | 2008-09-13 05:16:45 +0000 | [diff] [blame] | 69 | assert (Err.empty() && "Could not open file for printing out decls."); |
Ted Kremenek | ee53364 | 2007-12-20 19:47:16 +0000 | [diff] [blame] | 70 | llvm::OwningPtr<ASTConsumer> FilePrinter(CreateASTPrinter(&DeclPP)); |
Ted Kremenek | e720198 | 2007-11-13 22:56:10 +0000 | [diff] [blame] | 71 | |
Chris Lattner | d2fa675 | 2009-03-28 03:56:54 +0000 | [diff] [blame] | 72 | TranslationUnitDecl *TUD = Ctx.getTranslationUnitDecl(); |
| 73 | for (DeclContext::decl_iterator I = TUD->decls_begin(), E =TUD->decls_end(); |
| 74 | I != E; ++I) |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 75 | FilePrinter->HandleTopLevelDecl(*I); |
Ted Kremenek | 7a1f4db | 2007-11-10 02:07:12 +0000 | [diff] [blame] | 76 | } |
| 77 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 78 | // Serialize the translation unit. |
Chris Lattner | 80a0333 | 2009-03-28 03:53:02 +0000 | [diff] [blame] | 79 | |
| 80 | // Reserve 256K for bitstream buffer. |
| 81 | std::vector<unsigned char> Buffer; |
| 82 | Buffer.reserve(256*1024); |
| 83 | |
Chris Lattner | 557c5b1 | 2009-03-28 04:27:18 +0000 | [diff] [blame^] | 84 | Ctx.EmitASTBitcodeBuffer(Buffer); |
Chris Lattner | 80a0333 | 2009-03-28 03:53:02 +0000 | [diff] [blame] | 85 | |
| 86 | // Write the bits to disk. |
| 87 | if (FILE* fp = fopen(Filename.c_str(),"wb")) { |
| 88 | fwrite((char*)&Buffer.front(), sizeof(char), Buffer.size(), fp); |
| 89 | fclose(fp); |
| 90 | return true; |
| 91 | } |
| 92 | |
| 93 | return false; |
Ted Kremenek | bfa82c4 | 2007-10-16 23:37:27 +0000 | [diff] [blame] | 94 | } |
| 95 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 96 | bool SerializationTest::Deserialize(llvm::sys::Path& Filename, |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 97 | llvm::sys::Path& FNameDeclPrint) { |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 98 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 99 | // Deserialize the translation unit. |
Chris Lattner | 557c5b1 | 2009-03-28 04:27:18 +0000 | [diff] [blame^] | 100 | ASTContext *NewCtx; |
Chris Lattner | 5f737cc | 2009-03-28 03:49:26 +0000 | [diff] [blame] | 101 | |
| 102 | { |
| 103 | // Create the memory buffer that contains the contents of the file. |
| 104 | llvm::OwningPtr<llvm::MemoryBuffer> |
| 105 | MBuffer(llvm::MemoryBuffer::getFile(Filename.c_str())); |
| 106 | |
| 107 | if (!MBuffer) |
| 108 | return false; |
| 109 | |
Chris Lattner | 557c5b1 | 2009-03-28 04:27:18 +0000 | [diff] [blame^] | 110 | NewCtx = ASTContext::ReadASTBitcodeBuffer(*MBuffer, FMgr); |
Chris Lattner | 5f737cc | 2009-03-28 03:49:26 +0000 | [diff] [blame] | 111 | } |
Ted Kremenek | b899ad2 | 2007-11-14 17:46:35 +0000 | [diff] [blame] | 112 | |
Chris Lattner | 557c5b1 | 2009-03-28 04:27:18 +0000 | [diff] [blame^] | 113 | if (!NewCtx) |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 114 | return false; |
Ted Kremenek | b899ad2 | 2007-11-14 17:46:35 +0000 | [diff] [blame] | 115 | |
Ted Kremenek | e452e0f | 2008-08-08 04:15:52 +0000 | [diff] [blame] | 116 | { |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 117 | // Pretty-print the deserialized decls to a temp file. |
Ted Kremenek | a95d375 | 2008-09-13 05:16:45 +0000 | [diff] [blame] | 118 | std::string Err; |
Daniel Dunbar | 26fb272 | 2008-11-13 05:09:21 +0000 | [diff] [blame] | 119 | llvm::raw_fd_ostream DeclPP(FNameDeclPrint.c_str(), true, Err); |
Ted Kremenek | a95d375 | 2008-09-13 05:16:45 +0000 | [diff] [blame] | 120 | assert (Err.empty() && "Could not open file for printing out decls."); |
Ted Kremenek | ee53364 | 2007-12-20 19:47:16 +0000 | [diff] [blame] | 121 | llvm::OwningPtr<ASTConsumer> FilePrinter(CreateASTPrinter(&DeclPP)); |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 122 | |
Chris Lattner | 557c5b1 | 2009-03-28 04:27:18 +0000 | [diff] [blame^] | 123 | TranslationUnitDecl *TUD = NewCtx->getTranslationUnitDecl(); |
| 124 | for (DeclContext::decl_iterator I = TUD->decls_begin(), E = TUD->decls_end(); |
| 125 | I != E; ++I) |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 126 | FilePrinter->HandleTopLevelDecl(*I); |
Ted Kremenek | c4ca288 | 2007-12-03 22:48:14 +0000 | [diff] [blame] | 127 | } |
Sam Bishop | 4ccea1a | 2008-04-03 05:03:34 +0000 | [diff] [blame] | 128 | |
Chris Lattner | 557c5b1 | 2009-03-28 04:27:18 +0000 | [diff] [blame^] | 129 | delete NewCtx; |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 130 | |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 131 | return true; |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 132 | } |
| 133 | |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 134 | namespace { |
| 135 | class TmpDirJanitor { |
| 136 | llvm::sys::Path& Dir; |
| 137 | public: |
| 138 | explicit TmpDirJanitor(llvm::sys::Path& dir) : Dir(dir) {} |
| 139 | |
| 140 | ~TmpDirJanitor() { |
| 141 | llvm::cerr << "Removing: " << Dir.c_str() << '\n'; |
| 142 | Dir.eraseFromDisk(true); |
| 143 | } |
| 144 | }; |
| 145 | } |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 146 | |
Chris Lattner | dacbc5d | 2009-03-28 04:11:33 +0000 | [diff] [blame] | 147 | void SerializationTest::HandleTranslationUnit(ASTContext &Ctx) { |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 148 | |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 149 | std::string ErrMsg; |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 150 | llvm::sys::Path Dir = llvm::sys::Path::GetTemporaryDirectory(&ErrMsg); |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 151 | |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 152 | if (Dir.isEmpty()) { |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 153 | llvm::cerr << "Error: " << ErrMsg << "\n"; |
| 154 | return; |
Ted Kremenek | c637e6b | 2007-10-23 22:18:37 +0000 | [diff] [blame] | 155 | } |
| 156 | |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 157 | TmpDirJanitor RemoveTmpOnExit(Dir); |
| 158 | |
| 159 | llvm::sys::Path FNameDeclBefore = Dir; |
| 160 | FNameDeclBefore.appendComponent("test.decl_before.txt"); |
| 161 | |
Chris Lattner | dacbc5d | 2009-03-28 04:11:33 +0000 | [diff] [blame] | 162 | if (FNameDeclBefore.makeUnique(true, &ErrMsg)) { |
Ted Kremenek | 018b395 | 2007-11-06 19:50:53 +0000 | [diff] [blame] | 163 | llvm::cerr << "Error: " << ErrMsg << "\n"; |
| 164 | return; |
| 165 | } |
Ted Kremenek | 4ac8121 | 2007-11-05 21:39:35 +0000 | [diff] [blame] | 166 | |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 167 | llvm::sys::Path FNameDeclAfter = Dir; |
| 168 | FNameDeclAfter.appendComponent("test.decl_after.txt"); |
| 169 | |
Chris Lattner | dacbc5d | 2009-03-28 04:11:33 +0000 | [diff] [blame] | 170 | if (FNameDeclAfter.makeUnique(true, &ErrMsg)) { |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 171 | llvm::cerr << "Error: " << ErrMsg << "\n"; |
| 172 | return; |
| 173 | } |
| 174 | |
| 175 | llvm::sys::Path ASTFilename = Dir; |
| 176 | ASTFilename.appendComponent("test.ast"); |
| 177 | |
Chris Lattner | dacbc5d | 2009-03-28 04:11:33 +0000 | [diff] [blame] | 178 | if (ASTFilename.makeUnique(true, &ErrMsg)) { |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 179 | llvm::cerr << "Error: " << ErrMsg << "\n"; |
| 180 | return; |
| 181 | } |
| 182 | |
| 183 | // Serialize and then deserialize the ASTs. |
Chris Lattner | dacbc5d | 2009-03-28 04:11:33 +0000 | [diff] [blame] | 184 | bool status = Serialize(ASTFilename, FNameDeclBefore, Ctx); |
Ted Kremenek | 3553766 | 2007-12-05 22:08:43 +0000 | [diff] [blame] | 185 | assert (status && "Serialization failed."); |
| 186 | status = Deserialize(ASTFilename, FNameDeclAfter); |
| 187 | assert (status && "Deserialization failed."); |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 188 | |
| 189 | // Read both pretty-printed files and compare them. |
| 190 | |
| 191 | using llvm::MemoryBuffer; |
| 192 | |
Ted Kremenek | ee53364 | 2007-12-20 19:47:16 +0000 | [diff] [blame] | 193 | llvm::OwningPtr<MemoryBuffer> |
Chris Lattner | 35de512 | 2008-04-01 18:04:30 +0000 | [diff] [blame] | 194 | MBufferSer(MemoryBuffer::getFile(FNameDeclBefore.c_str())); |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 195 | |
| 196 | if(!MBufferSer) { |
| 197 | llvm::cerr << "ERROR: Cannot read pretty-printed file (pre-pickle).\n"; |
| 198 | return; |
| 199 | } |
| 200 | |
Ted Kremenek | ee53364 | 2007-12-20 19:47:16 +0000 | [diff] [blame] | 201 | llvm::OwningPtr<MemoryBuffer> |
Chris Lattner | 35de512 | 2008-04-01 18:04:30 +0000 | [diff] [blame] | 202 | MBufferDSer(MemoryBuffer::getFile(FNameDeclAfter.c_str())); |
Ted Kremenek | 79a2a26 | 2007-11-28 19:21:47 +0000 | [diff] [blame] | 203 | |
| 204 | if(!MBufferDSer) { |
| 205 | llvm::cerr << "ERROR: Cannot read pretty-printed file (post-pickle).\n"; |
| 206 | return; |
| 207 | } |
| 208 | |
| 209 | const char *p1 = MBufferSer->getBufferStart(); |
| 210 | const char *e1 = MBufferSer->getBufferEnd(); |
| 211 | const char *p2 = MBufferDSer->getBufferStart(); |
| 212 | const char *e2 = MBufferDSer->getBufferEnd(); |
| 213 | |
| 214 | if (MBufferSer->getBufferSize() == MBufferDSer->getBufferSize()) |
| 215 | for ( ; p1 != e1 ; ++p1, ++p2 ) |
| 216 | if (*p1 != *p2) break; |
| 217 | |
| 218 | if (p1 != e1 || p2 != e2 ) |
| 219 | llvm::cerr << "ERROR: Pretty-printed files are not the same.\n"; |
| 220 | else |
| 221 | llvm::cerr << "SUCCESS: Pretty-printed files are the same.\n"; |
Ted Kremenek | c637e6b | 2007-10-23 22:18:37 +0000 | [diff] [blame] | 222 | } |