blob: f8d2f5a9bd8caf946695e350ca408892613ee3f4 [file] [log] [blame]
Chandler Carruth7fc162f2013-03-26 02:25:37 +00001//===---- IRReader.cpp - Reader for LLVM IR files -------------------------===//
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
10#include "llvm/IRReader/IRReader.h"
Stephen Hines36b56882014-04-23 16:57:46 -070011#include "llvm-c/Core.h"
12#include "llvm-c/IRReader.h"
13#include "llvm/AsmParser/Parser.h"
Chandler Carruth7fc162f2013-03-26 02:25:37 +000014#include "llvm/Bitcode/ReaderWriter.h"
Peter Zotovc6099db2013-11-06 09:21:15 +000015#include "llvm/IR/LLVMContext.h"
16#include "llvm/IR/Module.h"
Chandler Carruth7fc162f2013-03-26 02:25:37 +000017#include "llvm/Support/MemoryBuffer.h"
18#include "llvm/Support/SourceMgr.h"
Eli Bendersky16299652013-04-03 15:33:45 +000019#include "llvm/Support/Timer.h"
Peter Zotovc6099db2013-11-06 09:21:15 +000020#include "llvm/Support/raw_ostream.h"
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070021#include <system_error>
Chandler Carruth7fc162f2013-03-26 02:25:37 +000022
23using namespace llvm;
24
Eli Bendersky16299652013-04-03 15:33:45 +000025namespace llvm {
26 extern bool TimePassesIsEnabled;
27}
28
Craig Topper4172a8a2013-07-16 01:17:10 +000029static const char *const TimeIRParsingGroupName = "LLVM IR Parsing";
30static const char *const TimeIRParsingName = "Parse IR";
Eli Bendersky16299652013-04-03 15:33:45 +000031
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070032static Module *getLazyIRModule(MemoryBuffer *Buffer, SMDiagnostic &Err,
33 LLVMContext &Context) {
Chandler Carruth7fc162f2013-03-26 02:25:37 +000034 if (isBitcode((const unsigned char *)Buffer->getBufferStart(),
35 (const unsigned char *)Buffer->getBufferEnd())) {
36 std::string ErrMsg;
Stephen Hines36b56882014-04-23 16:57:46 -070037 ErrorOr<Module *> ModuleOrErr = getLazyBitcodeModule(Buffer, Context);
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070038 if (std::error_code EC = ModuleOrErr.getError()) {
Chandler Carruth7fc162f2013-03-26 02:25:37 +000039 Err = SMDiagnostic(Buffer->getBufferIdentifier(), SourceMgr::DK_Error,
Stephen Hines36b56882014-04-23 16:57:46 -070040 EC.message());
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070041 // getLazyBitcodeModule does not take ownership of the Buffer in the
Chandler Carruth7fc162f2013-03-26 02:25:37 +000042 // case of an error.
43 delete Buffer;
Stephen Hinesdce4a402014-05-29 02:49:00 -070044 return nullptr;
Chandler Carruth7fc162f2013-03-26 02:25:37 +000045 }
Stephen Hines36b56882014-04-23 16:57:46 -070046 return ModuleOrErr.get();
Chandler Carruth7fc162f2013-03-26 02:25:37 +000047 }
48
Stephen Hinesdce4a402014-05-29 02:49:00 -070049 return ParseAssembly(Buffer, nullptr, Err, Context);
Chandler Carruth7fc162f2013-03-26 02:25:37 +000050}
51
52Module *llvm::getLazyIRFileModule(const std::string &Filename, SMDiagnostic &Err,
Eli Bendersky19801a62013-04-01 19:47:56 +000053 LLVMContext &Context) {
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070054 ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
55 MemoryBuffer::getFileOrSTDIN(Filename);
56 if (std::error_code EC = FileOrErr.getError()) {
Chandler Carruth7fc162f2013-03-26 02:25:37 +000057 Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070058 "Could not open input file: " + EC.message());
Stephen Hinesdce4a402014-05-29 02:49:00 -070059 return nullptr;
Chandler Carruth7fc162f2013-03-26 02:25:37 +000060 }
61
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070062 return getLazyIRModule(FileOrErr.get().release(), Err, Context);
Chandler Carruth7fc162f2013-03-26 02:25:37 +000063}
64
65Module *llvm::ParseIR(MemoryBuffer *Buffer, SMDiagnostic &Err,
66 LLVMContext &Context) {
Eli Bendersky16299652013-04-03 15:33:45 +000067 NamedRegionTimer T(TimeIRParsingName, TimeIRParsingGroupName,
68 TimePassesIsEnabled);
Chandler Carruth7fc162f2013-03-26 02:25:37 +000069 if (isBitcode((const unsigned char *)Buffer->getBufferStart(),
70 (const unsigned char *)Buffer->getBufferEnd())) {
Stephen Hines36b56882014-04-23 16:57:46 -070071 ErrorOr<Module *> ModuleOrErr = parseBitcodeFile(Buffer, Context);
Stephen Hinesdce4a402014-05-29 02:49:00 -070072 Module *M = nullptr;
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070073 if (std::error_code EC = ModuleOrErr.getError())
Chandler Carruth7fc162f2013-03-26 02:25:37 +000074 Err = SMDiagnostic(Buffer->getBufferIdentifier(), SourceMgr::DK_Error,
Stephen Hines36b56882014-04-23 16:57:46 -070075 EC.message());
76 else
77 M = ModuleOrErr.get();
78 // parseBitcodeFile does not take ownership of the Buffer.
Chandler Carruth7fc162f2013-03-26 02:25:37 +000079 return M;
80 }
81
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070082 return ParseAssembly(MemoryBuffer::getMemBuffer(
83 Buffer->getBuffer(), Buffer->getBufferIdentifier()),
84 nullptr, Err, Context);
Chandler Carruth7fc162f2013-03-26 02:25:37 +000085}
86
87Module *llvm::ParseIRFile(const std::string &Filename, SMDiagnostic &Err,
88 LLVMContext &Context) {
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070089 ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
90 MemoryBuffer::getFileOrSTDIN(Filename);
91 if (std::error_code EC = FileOrErr.getError()) {
Chandler Carruth7fc162f2013-03-26 02:25:37 +000092 Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070093 "Could not open input file: " + EC.message());
Stephen Hinesdce4a402014-05-29 02:49:00 -070094 return nullptr;
Chandler Carruth7fc162f2013-03-26 02:25:37 +000095 }
96
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -070097 return ParseIR(FileOrErr.get().get(), Err, Context);
Chandler Carruth7fc162f2013-03-26 02:25:37 +000098}
Peter Zotovc6099db2013-11-06 09:21:15 +000099
100//===----------------------------------------------------------------------===//
101// C API.
102//===----------------------------------------------------------------------===//
103
104LLVMBool LLVMParseIRInContext(LLVMContextRef ContextRef,
105 LLVMMemoryBufferRef MemBuf, LLVMModuleRef *OutM,
106 char **OutMessage) {
107 SMDiagnostic Diag;
108
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -0700109 std::unique_ptr<MemoryBuffer> MB(unwrap(MemBuf));
110 *OutM = wrap(ParseIR(MB.get(), Diag, *unwrap(ContextRef)));
Peter Zotovc6099db2013-11-06 09:21:15 +0000111
112 if(!*OutM) {
113 if (OutMessage) {
114 std::string buf;
115 raw_string_ostream os(buf);
116
Stephen Hinesdce4a402014-05-29 02:49:00 -0700117 Diag.print(nullptr, os, false);
Peter Zotovc6099db2013-11-06 09:21:15 +0000118 os.flush();
119
120 *OutMessage = strdup(buf.c_str());
121 }
122 return 1;
123 }
124
125 return 0;
126}