blob: ea1495ecc5ca47be3a0f59536d9ce6ef51178190 [file] [log] [blame]
Misha Brukman46453792003-09-22 23:44:46 +00001//===- ReaderWrappers.cpp - Parse bytecode from file or buffer -----------===//
Misha Brukman8a96c532005-04-21 21:44:41 +00002//
John Criswellb576c942003-10-20 19:43:21 +00003// The LLVM Compiler Infrastructure
4//
5// This file was developed by the LLVM research group and is distributed under
6// the University of Illinois Open Source License. See LICENSE.TXT for details.
Misha Brukman8a96c532005-04-21 21:44:41 +00007//
John Criswellb576c942003-10-20 19:43:21 +00008//===----------------------------------------------------------------------===//
Misha Brukman46453792003-09-22 23:44:46 +00009//
10// This file implements loading and parsing a bytecode file and parsing a
11// bytecode module from a given buffer.
12//
13//===----------------------------------------------------------------------===//
14
Reid Spencerdf45a542004-06-29 23:24:14 +000015#include "llvm/Bytecode/Analyzer.h"
Chris Lattnerdeab9a72003-11-19 16:06:55 +000016#include "llvm/Bytecode/Reader.h"
Reid Spencerdf45a542004-06-29 23:24:14 +000017#include "Reader.h"
Chris Lattnercb7e2e22003-10-18 05:54:18 +000018#include "llvm/Module.h"
19#include "llvm/Instructions.h"
Reid Spencer551ccae2004-09-01 22:55:40 +000020#include "llvm/ADT/StringExtras.h"
Reid Spencer9153f8f2004-12-13 18:25:27 +000021#include "llvm/System/MappedFile.h"
Reid Spencer32f55532006-06-07 23:18:34 +000022#include "llvm/System/Program.h"
Chris Lattner2d6481c2003-12-29 21:35:05 +000023#include <cerrno>
Duraid Madina0f7bfba2005-12-26 14:23:22 +000024#include <memory>
Chris Lattnerdeab9a72003-11-19 16:06:55 +000025using namespace llvm;
Brian Gaeked0fde302003-11-11 22:41:34 +000026
Chris Lattnercb7e2e22003-10-18 05:54:18 +000027//===----------------------------------------------------------------------===//
28// BytecodeFileReader - Read from an mmap'able file descriptor.
29//
30
Misha Brukman12c29d12003-09-22 23:38:23 +000031namespace {
Misha Brukman12c29d12003-09-22 23:38:23 +000032 /// BytecodeFileReader - parses a bytecode file from a file
33 ///
Reid Spencerdf45a542004-06-29 23:24:14 +000034 class BytecodeFileReader : public BytecodeReader {
Misha Brukman12c29d12003-09-22 23:38:23 +000035 private:
Reid Spencer0b5a5042006-08-25 17:43:11 +000036 std::string fileName;
Chris Lattnerf2e292c2007-02-07 21:41:02 +000037 BCDecompressor_t *Decompressor;
Reid Spencer9153f8f2004-12-13 18:25:27 +000038 sys::MappedFile mapFile;
Misha Brukman12c29d12003-09-22 23:38:23 +000039
40 BytecodeFileReader(const BytecodeFileReader&); // Do not implement
Misha Brukman5c344412003-09-23 15:09:26 +000041 void operator=(const BytecodeFileReader &BFR); // Do not implement
Misha Brukman12c29d12003-09-22 23:38:23 +000042
43 public:
Chris Lattnerf2e292c2007-02-07 21:41:02 +000044 BytecodeFileReader(const std::string &Filename, BCDecompressor_t *BCDC,
45 llvm::BytecodeHandler* H=0);
Reid Spencer0b5a5042006-08-25 17:43:11 +000046 bool read(std::string* ErrMsg);
Chris Lattnerf0edc6c2006-10-12 18:32:30 +000047
48 void freeState() {
49 BytecodeReader::freeState();
50 mapFile.close();
51 }
Misha Brukman12c29d12003-09-22 23:38:23 +000052 };
Misha Brukman12c29d12003-09-22 23:38:23 +000053}
54
Reid Spencerdf45a542004-06-29 23:24:14 +000055BytecodeFileReader::BytecodeFileReader(const std::string &Filename,
Chris Lattnerf2e292c2007-02-07 21:41:02 +000056 BCDecompressor_t *BCDC,
Chris Lattnerf0edc6c2006-10-12 18:32:30 +000057 llvm::BytecodeHandler* H)
Chris Lattnerf2e292c2007-02-07 21:41:02 +000058 : BytecodeReader(H), fileName(Filename), Decompressor(BCDC) {
Reid Spencer0b5a5042006-08-25 17:43:11 +000059}
60
61bool BytecodeFileReader::read(std::string* ErrMsg) {
62 if (mapFile.open(sys::Path(fileName), sys::MappedFile::READ_ACCESS, ErrMsg))
63 return true;
64 if (!mapFile.map(ErrMsg)) {
65 mapFile.close();
66 return true;
Reid Spencerb152fd22006-08-22 16:10:12 +000067 }
Reid Spencer0b5a5042006-08-25 17:43:11 +000068 unsigned char* buffer = reinterpret_cast<unsigned char*>(mapFile.base());
Chris Lattner0d3382a2007-02-07 19:49:01 +000069 return ParseBytecode(buffer, mapFile.size(), fileName,
Chris Lattnerf2e292c2007-02-07 21:41:02 +000070 Decompressor, ErrMsg);
Misha Brukman12c29d12003-09-22 23:38:23 +000071}
72
Chris Lattnercb7e2e22003-10-18 05:54:18 +000073//===----------------------------------------------------------------------===//
74// BytecodeBufferReader - Read from a memory buffer
75//
Misha Brukmand57308a2003-09-23 16:13:28 +000076
77namespace {
78 /// BytecodeBufferReader - parses a bytecode file from a buffer
79 ///
Reid Spencerdf45a542004-06-29 23:24:14 +000080 class BytecodeBufferReader : public BytecodeReader {
Misha Brukmand57308a2003-09-23 16:13:28 +000081 private:
82 const unsigned char *Buffer;
Reid Spencer0b5a5042006-08-25 17:43:11 +000083 const unsigned char *Buf;
84 unsigned Length;
85 std::string ModuleID;
Chris Lattnerf2e292c2007-02-07 21:41:02 +000086 BCDecompressor_t *Decompressor;
Misha Brukmand57308a2003-09-23 16:13:28 +000087 bool MustDelete;
88
89 BytecodeBufferReader(const BytecodeBufferReader&); // Do not implement
90 void operator=(const BytecodeBufferReader &BFR); // Do not implement
91
92 public:
93 BytecodeBufferReader(const unsigned char *Buf, unsigned Length,
Chris Lattnerf2e292c2007-02-07 21:41:02 +000094 const std::string &ModuleID, BCDecompressor_t *BCDC,
Reid Spencer97c7d742004-07-04 11:03:03 +000095 llvm::BytecodeHandler* Handler = 0);
Misha Brukmand57308a2003-09-23 16:13:28 +000096 ~BytecodeBufferReader();
97
Reid Spencer0b5a5042006-08-25 17:43:11 +000098 bool read(std::string* ErrMsg);
99
Misha Brukmand57308a2003-09-23 16:13:28 +0000100 };
101}
102
Reid Spencer0b5a5042006-08-25 17:43:11 +0000103BytecodeBufferReader::BytecodeBufferReader(const unsigned char *buf,
104 unsigned len,
105 const std::string &modID,
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000106 BCDecompressor_t *BCDC,
Chris Lattnerf0edc6c2006-10-12 18:32:30 +0000107 llvm::BytecodeHandler *H)
108 : BytecodeReader(H), Buffer(0), Buf(buf), Length(len), ModuleID(modID)
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000109 , Decompressor(BCDC), MustDelete(false) {
Reid Spencer0b5a5042006-08-25 17:43:11 +0000110}
111
112BytecodeBufferReader::~BytecodeBufferReader() {
113 if (MustDelete) delete [] Buffer;
114}
115
116bool
117BytecodeBufferReader::read(std::string* ErrMsg) {
Misha Brukmand57308a2003-09-23 16:13:28 +0000118 // If not aligned, allocate a new buffer to hold the bytecode...
119 const unsigned char *ParseBegin = 0;
Reid Spencer9a7e0c52004-08-04 22:56:46 +0000120 if (reinterpret_cast<uint64_t>(Buf) & 3) {
Misha Brukman34ce14b2003-09-24 22:04:02 +0000121 Buffer = new unsigned char[Length+4];
Chris Lattner4eed7932003-09-24 22:34:17 +0000122 unsigned Offset = 4 - ((intptr_t)Buffer & 3); // Make sure it's aligned
Misha Brukmand57308a2003-09-23 16:13:28 +0000123 ParseBegin = Buffer + Offset;
Misha Brukman34ce14b2003-09-24 22:04:02 +0000124 memcpy((unsigned char*)ParseBegin, Buf, Length); // Copy it over
Misha Brukmand57308a2003-09-23 16:13:28 +0000125 MustDelete = true;
126 } else {
127 // If we don't need to copy it over, just use the caller's copy
John Criswell4dcbd5e2003-09-23 21:19:11 +0000128 ParseBegin = Buffer = Buf;
Misha Brukmand57308a2003-09-23 16:13:28 +0000129 MustDelete = false;
130 }
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000131 if (ParseBytecode(ParseBegin, Length, ModuleID, Decompressor, ErrMsg)) {
Misha Brukman7f58de22003-10-08 19:55:47 +0000132 if (MustDelete) delete [] Buffer;
Reid Spencer0b5a5042006-08-25 17:43:11 +0000133 return true;
Misha Brukman7f58de22003-10-08 19:55:47 +0000134 }
Reid Spencer0b5a5042006-08-25 17:43:11 +0000135 return false;
Misha Brukmand57308a2003-09-23 16:13:28 +0000136}
137
Chris Lattnercb7e2e22003-10-18 05:54:18 +0000138//===----------------------------------------------------------------------===//
139// BytecodeStdinReader - Read bytecode from Standard Input
140//
Misha Brukmand57308a2003-09-23 16:13:28 +0000141
142namespace {
143 /// BytecodeStdinReader - parses a bytecode file from stdin
Misha Brukman8a96c532005-04-21 21:44:41 +0000144 ///
Reid Spencerdf45a542004-06-29 23:24:14 +0000145 class BytecodeStdinReader : public BytecodeReader {
Misha Brukmand57308a2003-09-23 16:13:28 +0000146 private:
147 std::vector<unsigned char> FileData;
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000148 BCDecompressor_t *Decompressor;
Misha Brukmand57308a2003-09-23 16:13:28 +0000149 unsigned char *FileBuf;
150
151 BytecodeStdinReader(const BytecodeStdinReader&); // Do not implement
152 void operator=(const BytecodeStdinReader &BFR); // Do not implement
153
154 public:
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000155 BytecodeStdinReader(BCDecompressor_t *BCDC, llvm::BytecodeHandler* H = 0);
Reid Spencer0b5a5042006-08-25 17:43:11 +0000156 bool read(std::string* ErrMsg);
Misha Brukmand57308a2003-09-23 16:13:28 +0000157 };
158}
Misha Brukman12c29d12003-09-22 23:38:23 +0000159
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000160BytecodeStdinReader::BytecodeStdinReader(BCDecompressor_t *BCDC,
161 BytecodeHandler* H)
162 : BytecodeReader(H), Decompressor(BCDC) {
Reid Spencer0b5a5042006-08-25 17:43:11 +0000163}
164
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000165bool BytecodeStdinReader::read(std::string* ErrMsg) {
Reid Spencer32f55532006-06-07 23:18:34 +0000166 sys::Program::ChangeStdinToBinary();
Reid Spencer0a834722004-12-21 07:51:33 +0000167 char Buffer[4096*4];
Misha Brukman12c29d12003-09-22 23:38:23 +0000168
169 // Read in all of the data from stdin, we cannot mmap stdin...
Bill Wendlingbcd24982006-12-07 20:28:15 +0000170 while (cin.stream()->good()) {
171 cin.stream()->read(Buffer, 4096*4);
172 int BlockSize = cin.stream()->gcount();
Reid Spencer0a834722004-12-21 07:51:33 +0000173 if (0 >= BlockSize)
174 break;
Misha Brukman12c29d12003-09-22 23:38:23 +0000175 FileData.insert(FileData.end(), Buffer, Buffer+BlockSize);
176 }
177
Reid Spencer0b5a5042006-08-25 17:43:11 +0000178 if (FileData.empty()) {
179 if (ErrMsg)
180 *ErrMsg = "Standard Input is empty!";
181 return true;
182 }
Misha Brukman12c29d12003-09-22 23:38:23 +0000183
Misha Brukman12c29d12003-09-22 23:38:23 +0000184 FileBuf = &FileData[0];
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000185 if (ParseBytecode(FileBuf, FileData.size(), "<stdin>", Decompressor, ErrMsg))
Reid Spencer0b5a5042006-08-25 17:43:11 +0000186 return true;
187 return false;
Misha Brukman12c29d12003-09-22 23:38:23 +0000188}
189
Chris Lattnercb7e2e22003-10-18 05:54:18 +0000190//===----------------------------------------------------------------------===//
Misha Brukmand57308a2003-09-23 16:13:28 +0000191// Wrapper functions
Chris Lattnercb7e2e22003-10-18 05:54:18 +0000192//===----------------------------------------------------------------------===//
Misha Brukmand57308a2003-09-23 16:13:28 +0000193
194/// getBytecodeBufferModuleProvider - lazy function-at-a-time loading from a
195/// buffer
Misha Brukman8a96c532005-04-21 21:44:41 +0000196ModuleProvider*
Chris Lattnerdeab9a72003-11-19 16:06:55 +0000197llvm::getBytecodeBufferModuleProvider(const unsigned char *Buffer,
198 unsigned Length,
Reid Spencerdf45a542004-06-29 23:24:14 +0000199 const std::string &ModuleID,
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000200 BCDecompressor_t *BCDC,
Chris Lattnerf0edc6c2006-10-12 18:32:30 +0000201 std::string *ErrMsg,
202 BytecodeHandler *H) {
Chris Lattnerf3ccb692007-01-07 06:45:57 +0000203 BytecodeBufferReader *rdr =
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000204 new BytecodeBufferReader(Buffer, Length, ModuleID, BCDC, H);
Reid Spencer0b5a5042006-08-25 17:43:11 +0000205 if (rdr->read(ErrMsg))
206 return 0;
Chris Lattnerf3ccb692007-01-07 06:45:57 +0000207 return rdr;
Misha Brukman12c29d12003-09-22 23:38:23 +0000208}
209
Misha Brukmand57308a2003-09-23 16:13:28 +0000210/// ParseBytecodeBuffer - Parse a given bytecode buffer
211///
Chris Lattnerdeab9a72003-11-19 16:06:55 +0000212Module *llvm::ParseBytecodeBuffer(const unsigned char *Buffer, unsigned Length,
213 const std::string &ModuleID,
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000214 BCDecompressor_t *BCDC,
Chris Lattnerf0edc6c2006-10-12 18:32:30 +0000215 std::string *ErrMsg) {
216 ModuleProvider *MP =
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000217 getBytecodeBufferModuleProvider(Buffer, Length, ModuleID, BCDC, ErrMsg, 0);
Chris Lattnerf0edc6c2006-10-12 18:32:30 +0000218 if (!MP) return 0;
Reid Spencer9b84ad12006-12-15 19:49:23 +0000219 Module *M = MP->releaseModule(ErrMsg);
Chris Lattnerf0edc6c2006-10-12 18:32:30 +0000220 delete MP;
221 return M;
Misha Brukman12c29d12003-09-22 23:38:23 +0000222}
223
Misha Brukmand57308a2003-09-23 16:13:28 +0000224/// getBytecodeModuleProvider - lazy function-at-a-time loading from a file
Misha Brukman12c29d12003-09-22 23:38:23 +0000225///
Reid Spencer0b5a5042006-08-25 17:43:11 +0000226ModuleProvider *
227llvm::getBytecodeModuleProvider(const std::string &Filename,
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000228 BCDecompressor_t *BCDC,
Reid Spencer0b5a5042006-08-25 17:43:11 +0000229 std::string* ErrMsg,
230 BytecodeHandler* H) {
231 // Read from a file
232 if (Filename != std::string("-")) {
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000233 BytecodeFileReader *rdr = new BytecodeFileReader(Filename, BCDC, H);
Reid Spencer0b5a5042006-08-25 17:43:11 +0000234 if (rdr->read(ErrMsg))
235 return 0;
Chris Lattnerf3ccb692007-01-07 06:45:57 +0000236 return rdr;
Reid Spencer0b5a5042006-08-25 17:43:11 +0000237 }
238
239 // Read from stdin
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000240 BytecodeStdinReader *rdr = new BytecodeStdinReader(BCDC, H);
Reid Spencer0b5a5042006-08-25 17:43:11 +0000241 if (rdr->read(ErrMsg))
242 return 0;
Chris Lattnerf3ccb692007-01-07 06:45:57 +0000243 return rdr;
Misha Brukman12c29d12003-09-22 23:38:23 +0000244}
245
Misha Brukmand57308a2003-09-23 16:13:28 +0000246/// ParseBytecodeFile - Parse the given bytecode file
247///
Chris Lattnerdeab9a72003-11-19 16:06:55 +0000248Module *llvm::ParseBytecodeFile(const std::string &Filename,
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000249 BCDecompressor_t *BCDC,
Reid Spencer0b5a5042006-08-25 17:43:11 +0000250 std::string *ErrMsg) {
Chris Lattnerf2e292c2007-02-07 21:41:02 +0000251 ModuleProvider* MP = getBytecodeModuleProvider(Filename, BCDC, ErrMsg);
Chris Lattnerf0edc6c2006-10-12 18:32:30 +0000252 if (!MP) return 0;
Reid Spencer9b84ad12006-12-15 19:49:23 +0000253 Module *M = MP->releaseModule(ErrMsg);
Chris Lattnerf0edc6c2006-10-12 18:32:30 +0000254 delete MP;
255 return M;
Misha Brukman12c29d12003-09-22 23:38:23 +0000256}