blob: 259b41af24fe4d0cc5e3af638e859a557f6117cd [file] [log] [blame]
Chandler Carruthe60e57b2013-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"
Chandler Carruth8a8cd2b2014-01-07 11:48:04 +000011#include "llvm-c/Core.h"
12#include "llvm-c/IRReader.h"
Chandler Carruth9aca9182014-01-07 12:34:26 +000013#include "llvm/AsmParser/Parser.h"
Chandler Carruthe60e57b2013-03-26 02:25:37 +000014#include "llvm/Bitcode/ReaderWriter.h"
Peter Zotov285eed62013-11-06 09:21:15 +000015#include "llvm/IR/LLVMContext.h"
16#include "llvm/IR/Module.h"
Chandler Carruthe60e57b2013-03-26 02:25:37 +000017#include "llvm/Support/MemoryBuffer.h"
18#include "llvm/Support/SourceMgr.h"
Eli Benderskyb35a2112013-04-03 15:33:45 +000019#include "llvm/Support/Timer.h"
Peter Zotov285eed62013-11-06 09:21:15 +000020#include "llvm/Support/raw_ostream.h"
Rafael Espindolaa6e9c3e2014-06-12 17:38:55 +000021#include <system_error>
Chandler Carruthe60e57b2013-03-26 02:25:37 +000022
23using namespace llvm;
24
Eli Benderskyb35a2112013-04-03 15:33:45 +000025namespace llvm {
26 extern bool TimePassesIsEnabled;
27}
28
Craig Topperd3a34f82013-07-16 01:17:10 +000029static const char *const TimeIRParsingGroupName = "LLVM IR Parsing";
30static const char *const TimeIRParsingName = "Parse IR";
Eli Benderskyb35a2112013-04-03 15:33:45 +000031
Rafael Espindola11c07d72014-08-19 16:58:54 +000032static std::unique_ptr<Module>
33getLazyIRModule(std::unique_ptr<MemoryBuffer> Buffer, SMDiagnostic &Err,
34 LLVMContext &Context) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000035 if (isBitcode((const unsigned char *)Buffer->getBufferStart(),
36 (const unsigned char *)Buffer->getBufferEnd())) {
37 std::string ErrMsg;
Rafael Espindola2bb0c942014-08-17 21:22:19 +000038 ErrorOr<Module *> ModuleOrErr = getLazyBitcodeModule(Buffer.get(), Context);
Rafael Espindoladb4ed0b2014-06-13 02:24:39 +000039 if (std::error_code EC = ModuleOrErr.getError()) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000040 Err = SMDiagnostic(Buffer->getBufferIdentifier(), SourceMgr::DK_Error,
Rafael Espindola5b6c1e82014-01-13 18:31:04 +000041 EC.message());
Craig Topper2617dcc2014-04-15 06:32:26 +000042 return nullptr;
Chandler Carruthe60e57b2013-03-26 02:25:37 +000043 }
Rafael Espindola2bb0c942014-08-17 21:22:19 +000044 // getLazyBitcodeModule takes ownership of the Buffer when successful.
45 Buffer.release();
Rafael Espindola11c07d72014-08-19 16:58:54 +000046 return std::unique_ptr<Module>(ModuleOrErr.get());
Chandler Carruthe60e57b2013-03-26 02:25:37 +000047 }
48
Rafael Espindola11c07d72014-08-19 16:58:54 +000049 return parseAssembly(std::move(Buffer), Err, Context);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000050}
51
Rafael Espindolad233b062014-08-26 17:29:46 +000052std::unique_ptr<Module> llvm::getLazyIRFileModule(StringRef Filename,
53 SMDiagnostic &Err,
54 LLVMContext &Context) {
Rafael Espindolaadf21f22014-07-06 17:43:13 +000055 ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
56 MemoryBuffer::getFileOrSTDIN(Filename);
57 if (std::error_code EC = FileOrErr.getError()) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000058 Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
Rafael Espindolaadf21f22014-07-06 17:43:13 +000059 "Could not open input file: " + EC.message());
Craig Topper2617dcc2014-04-15 06:32:26 +000060 return nullptr;
Chandler Carruthe60e57b2013-03-26 02:25:37 +000061 }
62
Rafael Espindolad233b062014-08-26 17:29:46 +000063 return getLazyIRModule(std::move(FileOrErr.get()), Err, Context);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000064}
65
Rafael Espindolad233b062014-08-26 17:29:46 +000066std::unique_ptr<Module> llvm::parseIR(MemoryBuffer *Buffer, SMDiagnostic &Err,
67 LLVMContext &Context) {
Eli Benderskyb35a2112013-04-03 15:33:45 +000068 NamedRegionTimer T(TimeIRParsingName, TimeIRParsingGroupName,
69 TimePassesIsEnabled);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000070 if (isBitcode((const unsigned char *)Buffer->getBufferStart(),
71 (const unsigned char *)Buffer->getBufferEnd())) {
Alp Tokerf6ae8442014-06-27 04:48:32 +000072 ErrorOr<Module *> ModuleOrErr = parseBitcodeFile(Buffer, Context);
Rafael Espindolad233b062014-08-26 17:29:46 +000073 if (std::error_code EC = ModuleOrErr.getError()) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000074 Err = SMDiagnostic(Buffer->getBufferIdentifier(), SourceMgr::DK_Error,
Rafael Espindola8f31e212014-01-15 01:08:23 +000075 EC.message());
Rafael Espindolad233b062014-08-26 17:29:46 +000076 return nullptr;
77 }
78 return std::unique_ptr<Module>(ModuleOrErr.get());
Chandler Carruthe60e57b2013-03-26 02:25:37 +000079 }
80
Rafael Espindola11c07d72014-08-19 16:58:54 +000081 return parseAssembly(std::unique_ptr<MemoryBuffer>(MemoryBuffer::getMemBuffer(
Rafael Espindolaf7aed802014-08-17 21:36:47 +000082 Buffer->getBuffer(), Buffer->getBufferIdentifier())),
Rafael Espindolad233b062014-08-26 17:29:46 +000083 Err, Context);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000084}
85
Rafael Espindolad233b062014-08-26 17:29:46 +000086std::unique_ptr<Module> llvm::parseIRFile(StringRef Filename, SMDiagnostic &Err,
87 LLVMContext &Context) {
Rafael Espindolaadf21f22014-07-06 17:43:13 +000088 ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
89 MemoryBuffer::getFileOrSTDIN(Filename);
90 if (std::error_code EC = FileOrErr.getError()) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000091 Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
Rafael Espindolaadf21f22014-07-06 17:43:13 +000092 "Could not open input file: " + EC.message());
Craig Topper2617dcc2014-04-15 06:32:26 +000093 return nullptr;
Chandler Carruthe60e57b2013-03-26 02:25:37 +000094 }
95
Rafael Espindolad233b062014-08-26 17:29:46 +000096 return parseIR(FileOrErr.get().get(), Err, Context);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000097}
Peter Zotov285eed62013-11-06 09:21:15 +000098
99//===----------------------------------------------------------------------===//
100// C API.
101//===----------------------------------------------------------------------===//
102
103LLVMBool LLVMParseIRInContext(LLVMContextRef ContextRef,
104 LLVMMemoryBufferRef MemBuf, LLVMModuleRef *OutM,
105 char **OutMessage) {
106 SMDiagnostic Diag;
107
Alp Toker5ebb7b32014-06-27 04:33:58 +0000108 std::unique_ptr<MemoryBuffer> MB(unwrap(MemBuf));
Rafael Espindolad233b062014-08-26 17:29:46 +0000109 *OutM = wrap(parseIR(MB.get(), Diag, *unwrap(ContextRef)).release());
Peter Zotov285eed62013-11-06 09:21:15 +0000110
111 if(!*OutM) {
112 if (OutMessage) {
Alp Tokere69170a2014-06-26 22:52:05 +0000113 std::string buf;
114 raw_string_ostream os(buf);
115
Craig Topper2617dcc2014-04-15 06:32:26 +0000116 Diag.print(nullptr, os, false);
Alp Tokere69170a2014-06-26 22:52:05 +0000117 os.flush();
118
119 *OutMessage = strdup(buf.c_str());
Peter Zotov285eed62013-11-06 09:21:15 +0000120 }
121 return 1;
122 }
123
124 return 0;
125}