blob: a39811fd90edcd70b249cc614089945e0e65e552 [file] [log] [blame]
Ted Kremenekbfa82c42007-10-16 23:37:27 +00001//===--- SerializationTest.cpp - Experimental Object Serialization --------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner0bc735f2007-12-29 19:59:25 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Ted Kremenekbfa82c42007-10-16 23:37:27 +00007//
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 Kremenekbfa82c42007-10-16 23:37:27 +000017#include "clang/AST/ASTConsumer.h"
Ted Kremenek5656b782007-11-29 01:24:25 +000018#include "clang/AST/CFG.h"
Daniel Dunbare91593e2008-08-11 04:54:23 +000019#include "clang/AST/Decl.h"
Ted Kremenekc4ca2882007-12-03 22:48:14 +000020#include "clang.h"
Ted Kremenek35537662007-12-05 22:08:43 +000021#include "ASTConsumers.h"
Ted Kremenek525fdbc2007-12-18 21:36:21 +000022#include "clang/AST/TranslationUnit.h"
Ted Kremenekc637e6b2007-10-23 22:18:37 +000023#include "llvm/Support/MemoryBuffer.h"
Ted Kremenekee533642007-12-20 19:47:16 +000024#include "llvm/ADT/OwningPtr.h"
Ted Kremenek35537662007-12-05 22:08:43 +000025#include "llvm/Support/Streams.h"
Ted Kremenekea75c552007-11-28 21:32:21 +000026#include <fstream>
Chris Lattner87cf5ac2008-03-10 17:04:53 +000027#include <cstring>
Ted Kremenekbfa82c42007-10-16 23:37:27 +000028using namespace clang;
Ted Kremenekbfa82c42007-10-16 23:37:27 +000029
Ted Kremenek79a2a262007-11-28 19:21:47 +000030//===----------------------------------------------------------------------===//
Ted Kremenek79a2a262007-11-28 19:21:47 +000031// Driver code.
32//===----------------------------------------------------------------------===//
33
Ted Kremenek35537662007-12-05 22:08:43 +000034namespace {
Ted Kremenek4ac81212007-11-05 21:39:35 +000035
Ted Kremenek35537662007-12-05 22:08:43 +000036class SerializationTest : public ASTConsumer {
Ted Kremenek35537662007-12-05 22:08:43 +000037 Diagnostic &Diags;
38 FileManager &FMgr;
Ted Kremenek018b3952007-11-06 19:50:53 +000039public:
Ted Kremeneke7d07d12008-06-04 15:55:15 +000040 SerializationTest(Diagnostic &d, FileManager& fmgr)
Ted Kremeneke452e0f2008-08-08 04:15:52 +000041 : Diags(d), FMgr(fmgr) {}
Ted Kremenekdca24662007-12-05 00:26:13 +000042
Ted Kremeneke452e0f2008-08-08 04:15:52 +000043 ~SerializationTest() {}
44
Chris Lattnerdacbc5d2009-03-28 04:11:33 +000045 virtual void HandleTranslationUnit(ASTContext &C);
Ted Kremenek63ea8632007-12-19 19:27:38 +000046
Ted Kremenek018b3952007-11-06 19:50:53 +000047private:
Ted Kremeneke452e0f2008-08-08 04:15:52 +000048 bool Serialize(llvm::sys::Path& Filename, llvm::sys::Path& FNameDeclPrint,
Chris Lattnerd2fa6752009-03-28 03:56:54 +000049 ASTContext &Ctx);
Ted Kremeneke452e0f2008-08-08 04:15:52 +000050
Ted Kremenek35537662007-12-05 22:08:43 +000051 bool Deserialize(llvm::sys::Path& Filename, llvm::sys::Path& FNameDeclPrint);
Ted Kremenek4ac81212007-11-05 21:39:35 +000052};
53
Ted Kremenekc637e6b2007-10-23 22:18:37 +000054} // end anonymous namespace
Ted Kremenekbfa82c42007-10-16 23:37:27 +000055
Ted Kremenekdca24662007-12-05 00:26:13 +000056ASTConsumer*
Ted Kremeneke7d07d12008-06-04 15:55:15 +000057clang::CreateSerializationTest(Diagnostic &Diags, FileManager& FMgr) {
58 return new SerializationTest(Diags, FMgr);
Ted Kremenek018b3952007-11-06 19:50:53 +000059}
60
Ted Kremenek018b3952007-11-06 19:50:53 +000061
Ted Kremenek35537662007-12-05 22:08:43 +000062bool SerializationTest::Serialize(llvm::sys::Path& Filename,
Ted Kremeneke452e0f2008-08-08 04:15:52 +000063 llvm::sys::Path& FNameDeclPrint,
Chris Lattnerd2fa6752009-03-28 03:56:54 +000064 ASTContext &Ctx) {
Ted Kremenek35537662007-12-05 22:08:43 +000065 {
66 // Pretty-print the decls to a temp file.
Ted Kremeneka95d3752008-09-13 05:16:45 +000067 std::string Err;
Daniel Dunbar26fb2722008-11-13 05:09:21 +000068 llvm::raw_fd_ostream DeclPP(FNameDeclPrint.c_str(), true, Err);
Ted Kremeneka95d3752008-09-13 05:16:45 +000069 assert (Err.empty() && "Could not open file for printing out decls.");
Ted Kremenekee533642007-12-20 19:47:16 +000070 llvm::OwningPtr<ASTConsumer> FilePrinter(CreateASTPrinter(&DeclPP));
Ted Kremeneke7201982007-11-13 22:56:10 +000071
Chris Lattnerd2fa6752009-03-28 03:56:54 +000072 TranslationUnitDecl *TUD = Ctx.getTranslationUnitDecl();
73 for (DeclContext::decl_iterator I = TUD->decls_begin(), E =TUD->decls_end();
74 I != E; ++I)
Ted Kremenek35537662007-12-05 22:08:43 +000075 FilePrinter->HandleTopLevelDecl(*I);
Ted Kremenek7a1f4db2007-11-10 02:07:12 +000076 }
77
Ted Kremenek35537662007-12-05 22:08:43 +000078 // Serialize the translation unit.
Chris Lattner80a03332009-03-28 03:53:02 +000079
80 // Reserve 256K for bitstream buffer.
81 std::vector<unsigned char> Buffer;
82 Buffer.reserve(256*1024);
83
Chris Lattner557c5b12009-03-28 04:27:18 +000084 Ctx.EmitASTBitcodeBuffer(Buffer);
Chris Lattner80a03332009-03-28 03:53:02 +000085
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 Kremenekbfa82c42007-10-16 23:37:27 +000094}
95
Ted Kremenek35537662007-12-05 22:08:43 +000096bool SerializationTest::Deserialize(llvm::sys::Path& Filename,
Ted Kremenek79a2a262007-11-28 19:21:47 +000097 llvm::sys::Path& FNameDeclPrint) {
Ted Kremenek018b3952007-11-06 19:50:53 +000098
Ted Kremenek35537662007-12-05 22:08:43 +000099 // Deserialize the translation unit.
Chris Lattner557c5b12009-03-28 04:27:18 +0000100 ASTContext *NewCtx;
Chris Lattner5f737cc2009-03-28 03:49:26 +0000101
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 Lattner557c5b12009-03-28 04:27:18 +0000110 NewCtx = ASTContext::ReadASTBitcodeBuffer(*MBuffer, FMgr);
Chris Lattner5f737cc2009-03-28 03:49:26 +0000111 }
Ted Kremenekb899ad22007-11-14 17:46:35 +0000112
Chris Lattner557c5b12009-03-28 04:27:18 +0000113 if (!NewCtx)
Ted Kremenek35537662007-12-05 22:08:43 +0000114 return false;
Ted Kremenekb899ad22007-11-14 17:46:35 +0000115
Ted Kremeneke452e0f2008-08-08 04:15:52 +0000116 {
Ted Kremenek35537662007-12-05 22:08:43 +0000117 // Pretty-print the deserialized decls to a temp file.
Ted Kremeneka95d3752008-09-13 05:16:45 +0000118 std::string Err;
Daniel Dunbar26fb2722008-11-13 05:09:21 +0000119 llvm::raw_fd_ostream DeclPP(FNameDeclPrint.c_str(), true, Err);
Ted Kremeneka95d3752008-09-13 05:16:45 +0000120 assert (Err.empty() && "Could not open file for printing out decls.");
Ted Kremenekee533642007-12-20 19:47:16 +0000121 llvm::OwningPtr<ASTConsumer> FilePrinter(CreateASTPrinter(&DeclPP));
Ted Kremenek35537662007-12-05 22:08:43 +0000122
Chris Lattner557c5b12009-03-28 04:27:18 +0000123 TranslationUnitDecl *TUD = NewCtx->getTranslationUnitDecl();
124 for (DeclContext::decl_iterator I = TUD->decls_begin(), E = TUD->decls_end();
125 I != E; ++I)
Ted Kremenek35537662007-12-05 22:08:43 +0000126 FilePrinter->HandleTopLevelDecl(*I);
Ted Kremenekc4ca2882007-12-03 22:48:14 +0000127 }
Sam Bishop4ccea1a2008-04-03 05:03:34 +0000128
Chris Lattner557c5b12009-03-28 04:27:18 +0000129 delete NewCtx;
Ted Kremenek018b3952007-11-06 19:50:53 +0000130
Ted Kremenek35537662007-12-05 22:08:43 +0000131 return true;
Ted Kremenek018b3952007-11-06 19:50:53 +0000132}
133
Ted Kremenek79a2a262007-11-28 19:21:47 +0000134namespace {
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 Kremenek018b3952007-11-06 19:50:53 +0000146
Chris Lattnerdacbc5d2009-03-28 04:11:33 +0000147void SerializationTest::HandleTranslationUnit(ASTContext &Ctx) {
Ted Kremenek79a2a262007-11-28 19:21:47 +0000148
Ted Kremenek018b3952007-11-06 19:50:53 +0000149 std::string ErrMsg;
Ted Kremenek79a2a262007-11-28 19:21:47 +0000150 llvm::sys::Path Dir = llvm::sys::Path::GetTemporaryDirectory(&ErrMsg);
Ted Kremenek018b3952007-11-06 19:50:53 +0000151
Ted Kremenek79a2a262007-11-28 19:21:47 +0000152 if (Dir.isEmpty()) {
Ted Kremenek018b3952007-11-06 19:50:53 +0000153 llvm::cerr << "Error: " << ErrMsg << "\n";
154 return;
Ted Kremenekc637e6b2007-10-23 22:18:37 +0000155 }
156
Ted Kremenek79a2a262007-11-28 19:21:47 +0000157 TmpDirJanitor RemoveTmpOnExit(Dir);
158
159 llvm::sys::Path FNameDeclBefore = Dir;
160 FNameDeclBefore.appendComponent("test.decl_before.txt");
161
Chris Lattnerdacbc5d2009-03-28 04:11:33 +0000162 if (FNameDeclBefore.makeUnique(true, &ErrMsg)) {
Ted Kremenek018b3952007-11-06 19:50:53 +0000163 llvm::cerr << "Error: " << ErrMsg << "\n";
164 return;
165 }
Ted Kremenek4ac81212007-11-05 21:39:35 +0000166
Ted Kremenek79a2a262007-11-28 19:21:47 +0000167 llvm::sys::Path FNameDeclAfter = Dir;
168 FNameDeclAfter.appendComponent("test.decl_after.txt");
169
Chris Lattnerdacbc5d2009-03-28 04:11:33 +0000170 if (FNameDeclAfter.makeUnique(true, &ErrMsg)) {
Ted Kremenek79a2a262007-11-28 19:21:47 +0000171 llvm::cerr << "Error: " << ErrMsg << "\n";
172 return;
173 }
174
175 llvm::sys::Path ASTFilename = Dir;
176 ASTFilename.appendComponent("test.ast");
177
Chris Lattnerdacbc5d2009-03-28 04:11:33 +0000178 if (ASTFilename.makeUnique(true, &ErrMsg)) {
Ted Kremenek79a2a262007-11-28 19:21:47 +0000179 llvm::cerr << "Error: " << ErrMsg << "\n";
180 return;
181 }
182
183 // Serialize and then deserialize the ASTs.
Chris Lattnerdacbc5d2009-03-28 04:11:33 +0000184 bool status = Serialize(ASTFilename, FNameDeclBefore, Ctx);
Ted Kremenek35537662007-12-05 22:08:43 +0000185 assert (status && "Serialization failed.");
186 status = Deserialize(ASTFilename, FNameDeclAfter);
187 assert (status && "Deserialization failed.");
Ted Kremenek79a2a262007-11-28 19:21:47 +0000188
189 // Read both pretty-printed files and compare them.
190
191 using llvm::MemoryBuffer;
192
Ted Kremenekee533642007-12-20 19:47:16 +0000193 llvm::OwningPtr<MemoryBuffer>
Chris Lattner35de5122008-04-01 18:04:30 +0000194 MBufferSer(MemoryBuffer::getFile(FNameDeclBefore.c_str()));
Ted Kremenek79a2a262007-11-28 19:21:47 +0000195
196 if(!MBufferSer) {
197 llvm::cerr << "ERROR: Cannot read pretty-printed file (pre-pickle).\n";
198 return;
199 }
200
Ted Kremenekee533642007-12-20 19:47:16 +0000201 llvm::OwningPtr<MemoryBuffer>
Chris Lattner35de5122008-04-01 18:04:30 +0000202 MBufferDSer(MemoryBuffer::getFile(FNameDeclAfter.c_str()));
Ted Kremenek79a2a262007-11-28 19:21:47 +0000203
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 Kremenekc637e6b2007-10-23 22:18:37 +0000222}