blob: ba587ced71821170fb8710625d8404e2e33e9a51 [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"
Teresa Johnsonad176792016-11-11 05:34:58 +000014#include "llvm/Bitcode/BitcodeReader.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
Matthias Braun9f15a792016-11-18 19:43:18 +000029static const char *const TimeIRParsingGroupName = "irparse";
30static const char *const TimeIRParsingGroupDescription = "LLVM IR Parsing";
31static const char *const TimeIRParsingName = "parse";
32static const char *const TimeIRParsingDescription = "Parse IR";
Eli Benderskyb35a2112013-04-03 15:33:45 +000033
Rafael Espindola11c07d72014-08-19 16:58:54 +000034static std::unique_ptr<Module>
35getLazyIRModule(std::unique_ptr<MemoryBuffer> Buffer, SMDiagnostic &Err,
Teresa Johnsone5a61912015-12-17 17:14:09 +000036 LLVMContext &Context, bool ShouldLazyLoadMetadata) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000037 if (isBitcode((const unsigned char *)Buffer->getBufferStart(),
38 (const unsigned char *)Buffer->getBufferEnd())) {
Peter Collingbourned9445c42016-11-13 07:00:17 +000039 Expected<std::unique_ptr<Module>> ModuleOrErr = getOwningLazyBitcodeModule(
Teresa Johnsone5a61912015-12-17 17:14:09 +000040 std::move(Buffer), Context, ShouldLazyLoadMetadata);
Peter Collingbourned9445c42016-11-13 07:00:17 +000041 if (Error E = ModuleOrErr.takeError()) {
42 handleAllErrors(std::move(E), [&](ErrorInfoBase &EIB) {
43 Err = SMDiagnostic(Buffer->getBufferIdentifier(), SourceMgr::DK_Error,
44 EIB.message());
45 });
Craig Topper2617dcc2014-04-15 06:32:26 +000046 return nullptr;
Chandler Carruthe60e57b2013-03-26 02:25:37 +000047 }
Rafael Espindoladcd1dca2015-06-16 22:27:55 +000048 return std::move(ModuleOrErr.get());
Chandler Carruthe60e57b2013-03-26 02:25:37 +000049 }
50
Rafael Espindolad96d5532014-08-26 21:49:01 +000051 return parseAssembly(Buffer->getMemBufferRef(), Err, Context);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000052}
53
Rafael Espindolad233b062014-08-26 17:29:46 +000054std::unique_ptr<Module> llvm::getLazyIRFileModule(StringRef Filename,
55 SMDiagnostic &Err,
Teresa Johnsone5a61912015-12-17 17:14:09 +000056 LLVMContext &Context,
57 bool ShouldLazyLoadMetadata) {
Rafael Espindolaadf21f22014-07-06 17:43:13 +000058 ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
59 MemoryBuffer::getFileOrSTDIN(Filename);
60 if (std::error_code EC = FileOrErr.getError()) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000061 Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
Rafael Espindolaadf21f22014-07-06 17:43:13 +000062 "Could not open input file: " + EC.message());
Craig Topper2617dcc2014-04-15 06:32:26 +000063 return nullptr;
Chandler Carruthe60e57b2013-03-26 02:25:37 +000064 }
65
Teresa Johnsone5a61912015-12-17 17:14:09 +000066 return getLazyIRModule(std::move(FileOrErr.get()), Err, Context,
67 ShouldLazyLoadMetadata);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000068}
69
Rafael Espindolad96d5532014-08-26 21:49:01 +000070std::unique_ptr<Module> llvm::parseIR(MemoryBufferRef Buffer, SMDiagnostic &Err,
Rafael Espindolad233b062014-08-26 17:29:46 +000071 LLVMContext &Context) {
Matthias Braun9f15a792016-11-18 19:43:18 +000072 NamedRegionTimer T(TimeIRParsingName, TimeIRParsingDescription,
73 TimeIRParsingGroupName, TimeIRParsingGroupDescription,
Eli Benderskyb35a2112013-04-03 15:33:45 +000074 TimePassesIsEnabled);
Rafael Espindolad96d5532014-08-26 21:49:01 +000075 if (isBitcode((const unsigned char *)Buffer.getBufferStart(),
76 (const unsigned char *)Buffer.getBufferEnd())) {
Peter Collingbourned9445c42016-11-13 07:00:17 +000077 Expected<std::unique_ptr<Module>> ModuleOrErr =
Rafael Espindoladcd1dca2015-06-16 22:27:55 +000078 parseBitcodeFile(Buffer, Context);
Peter Collingbourned9445c42016-11-13 07:00:17 +000079 if (Error E = ModuleOrErr.takeError()) {
80 handleAllErrors(std::move(E), [&](ErrorInfoBase &EIB) {
81 Err = SMDiagnostic(Buffer.getBufferIdentifier(), SourceMgr::DK_Error,
82 EIB.message());
83 });
Rafael Espindolad233b062014-08-26 17:29:46 +000084 return nullptr;
85 }
Rafael Espindoladcd1dca2015-06-16 22:27:55 +000086 return std::move(ModuleOrErr.get());
Chandler Carruthe60e57b2013-03-26 02:25:37 +000087 }
88
Rafael Espindolad96d5532014-08-26 21:49:01 +000089 return parseAssembly(Buffer, Err, Context);
Chandler Carruthe60e57b2013-03-26 02:25:37 +000090}
91
Rafael Espindolad233b062014-08-26 17:29:46 +000092std::unique_ptr<Module> llvm::parseIRFile(StringRef Filename, SMDiagnostic &Err,
93 LLVMContext &Context) {
Rafael Espindolaadf21f22014-07-06 17:43:13 +000094 ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
95 MemoryBuffer::getFileOrSTDIN(Filename);
96 if (std::error_code EC = FileOrErr.getError()) {
Chandler Carruthe60e57b2013-03-26 02:25:37 +000097 Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
Rafael Espindolaadf21f22014-07-06 17:43:13 +000098 "Could not open input file: " + EC.message());
Craig Topper2617dcc2014-04-15 06:32:26 +000099 return nullptr;
Chandler Carruthe60e57b2013-03-26 02:25:37 +0000100 }
101
Rafael Espindolad96d5532014-08-26 21:49:01 +0000102 return parseIR(FileOrErr.get()->getMemBufferRef(), Err, Context);
Chandler Carruthe60e57b2013-03-26 02:25:37 +0000103}
Peter Zotov285eed62013-11-06 09:21:15 +0000104
105//===----------------------------------------------------------------------===//
106// C API.
107//===----------------------------------------------------------------------===//
108
109LLVMBool LLVMParseIRInContext(LLVMContextRef ContextRef,
110 LLVMMemoryBufferRef MemBuf, LLVMModuleRef *OutM,
111 char **OutMessage) {
112 SMDiagnostic Diag;
113
Alp Toker5ebb7b32014-06-27 04:33:58 +0000114 std::unique_ptr<MemoryBuffer> MB(unwrap(MemBuf));
Rafael Espindolad96d5532014-08-26 21:49:01 +0000115 *OutM =
116 wrap(parseIR(MB->getMemBufferRef(), Diag, *unwrap(ContextRef)).release());
Peter Zotov285eed62013-11-06 09:21:15 +0000117
118 if(!*OutM) {
119 if (OutMessage) {
Alp Tokere69170a2014-06-26 22:52:05 +0000120 std::string buf;
121 raw_string_ostream os(buf);
122
Craig Topper2617dcc2014-04-15 06:32:26 +0000123 Diag.print(nullptr, os, false);
Alp Tokere69170a2014-06-26 22:52:05 +0000124 os.flush();
125
126 *OutMessage = strdup(buf.c_str());
Peter Zotov285eed62013-11-06 09:21:15 +0000127 }
128 return 1;
129 }
130
131 return 0;
132}