1e60e57beSChandler Carruth //===---- IRReader.cpp - Reader for LLVM IR files -------------------------===//
2e60e57beSChandler Carruth //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e60e57beSChandler Carruth //
7e60e57beSChandler Carruth //===----------------------------------------------------------------------===//
8e60e57beSChandler Carruth 
9e60e57beSChandler Carruth #include "llvm/IRReader/IRReader.h"
108a8cd2baSChandler Carruth #include "llvm-c/IRReader.h"
119aca918dSChandler Carruth #include "llvm/AsmParser/Parser.h"
12ad17679aSTeresa Johnson #include "llvm/Bitcode/BitcodeReader.h"
13285eed60SPeter Zotov #include "llvm/IR/LLVMContext.h"
14285eed60SPeter Zotov #include "llvm/IR/Module.h"
15e60e57beSChandler Carruth #include "llvm/Support/MemoryBuffer.h"
16e60e57beSChandler Carruth #include "llvm/Support/SourceMgr.h"
17b35a211fSEli Bendersky #include "llvm/Support/Timer.h"
18285eed60SPeter Zotov #include "llvm/Support/raw_ostream.h"
19a6e9c3e4SRafael Espindola #include <system_error>
20e60e57beSChandler Carruth 
21e60e57beSChandler Carruth using namespace llvm;
22e60e57beSChandler Carruth 
23b35a211fSEli Bendersky namespace llvm {
24b35a211fSEli Bendersky   extern bool TimePassesIsEnabled;
25b35a211fSEli Bendersky }
26b35a211fSEli Bendersky 
27a5309438SFangrui Song const char TimeIRParsingGroupName[] = "irparse";
28a5309438SFangrui Song const char TimeIRParsingGroupDescription[] = "LLVM IR Parsing";
29a5309438SFangrui Song const char TimeIRParsingName[] = "parse";
30a5309438SFangrui Song const char TimeIRParsingDescription[] = "Parse IR";
31b35a211fSEli Bendersky 
3272a0f4e8SScott Linder std::unique_ptr<Module>
getLazyIRModule(std::unique_ptr<MemoryBuffer> Buffer,SMDiagnostic & Err,LLVMContext & Context,bool ShouldLazyLoadMetadata)3372a0f4e8SScott Linder llvm::getLazyIRModule(std::unique_ptr<MemoryBuffer> Buffer, SMDiagnostic &Err,
34e5a61917STeresa Johnson                       LLVMContext &Context, bool ShouldLazyLoadMetadata) {
35e60e57beSChandler Carruth   if (isBitcode((const unsigned char *)Buffer->getBufferStart(),
36e60e57beSChandler Carruth                 (const unsigned char *)Buffer->getBufferEnd())) {
37d9445c49SPeter Collingbourne     Expected<std::unique_ptr<Module>> ModuleOrErr = getOwningLazyBitcodeModule(
38e5a61917STeresa Johnson         std::move(Buffer), Context, ShouldLazyLoadMetadata);
39d9445c49SPeter Collingbourne     if (Error E = ModuleOrErr.takeError()) {
40d9445c49SPeter Collingbourne       handleAllErrors(std::move(E), [&](ErrorInfoBase &EIB) {
41e60e57beSChandler Carruth         Err = SMDiagnostic(Buffer->getBufferIdentifier(), SourceMgr::DK_Error,
42d9445c49SPeter Collingbourne                            EIB.message());
43d9445c49SPeter Collingbourne       });
442617dcceSCraig Topper       return nullptr;
45e60e57beSChandler Carruth     }
46dcd1dca2SRafael Espindola     return std::move(ModuleOrErr.get());
47e60e57beSChandler Carruth   }
48e60e57beSChandler Carruth 
49d96d553dSRafael Espindola   return parseAssembly(Buffer->getMemBufferRef(), Err, Context);
50e60e57beSChandler Carruth }
51e60e57beSChandler Carruth 
getLazyIRFileModule(StringRef Filename,SMDiagnostic & Err,LLVMContext & Context,bool ShouldLazyLoadMetadata)52d233b06aSRafael Espindola std::unique_ptr<Module> llvm::getLazyIRFileModule(StringRef Filename,
53d233b06aSRafael Espindola                                                   SMDiagnostic &Err,
54e5a61917STeresa Johnson                                                   LLVMContext &Context,
55e5a61917STeresa Johnson                                                   bool ShouldLazyLoadMetadata) {
56adf21f2aSRafael Espindola   ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
57adf21f2aSRafael Espindola       MemoryBuffer::getFileOrSTDIN(Filename);
58adf21f2aSRafael Espindola   if (std::error_code EC = FileOrErr.getError()) {
59e60e57beSChandler Carruth     Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
60adf21f2aSRafael Espindola                        "Could not open input file: " + EC.message());
612617dcceSCraig Topper     return nullptr;
62e60e57beSChandler Carruth   }
63e60e57beSChandler Carruth 
64e5a61917STeresa Johnson   return getLazyIRModule(std::move(FileOrErr.get()), Err, Context,
65e5a61917STeresa Johnson                          ShouldLazyLoadMetadata);
66e60e57beSChandler Carruth }
67e60e57beSChandler Carruth 
parseIR(MemoryBufferRef Buffer,SMDiagnostic & Err,LLVMContext & Context,DataLayoutCallbackTy DataLayoutCallback)68d96d553dSRafael Espindola std::unique_ptr<Module> llvm::parseIR(MemoryBufferRef Buffer, SMDiagnostic &Err,
69a8b2ddbdSAdrian Prantl                                       LLVMContext &Context,
704532a508SEli Friedman                                       DataLayoutCallbackTy DataLayoutCallback) {
719f15a79eSMatthias Braun   NamedRegionTimer T(TimeIRParsingName, TimeIRParsingDescription,
729f15a79eSMatthias Braun                      TimeIRParsingGroupName, TimeIRParsingGroupDescription,
73b35a211fSEli Bendersky                      TimePassesIsEnabled);
74d96d553dSRafael Espindola   if (isBitcode((const unsigned char *)Buffer.getBufferStart(),
75d96d553dSRafael Espindola                 (const unsigned char *)Buffer.getBufferEnd())) {
76d9445c49SPeter Collingbourne     Expected<std::unique_ptr<Module>> ModuleOrErr =
774532a508SEli Friedman         parseBitcodeFile(Buffer, Context, DataLayoutCallback);
78d9445c49SPeter Collingbourne     if (Error E = ModuleOrErr.takeError()) {
79d9445c49SPeter Collingbourne       handleAllErrors(std::move(E), [&](ErrorInfoBase &EIB) {
80d96d553dSRafael Espindola         Err = SMDiagnostic(Buffer.getBufferIdentifier(), SourceMgr::DK_Error,
81d9445c49SPeter Collingbourne                            EIB.message());
82d9445c49SPeter Collingbourne       });
83d233b06aSRafael Espindola       return nullptr;
84d233b06aSRafael Espindola     }
85dcd1dca2SRafael Espindola     return std::move(ModuleOrErr.get());
86e60e57beSChandler Carruth   }
87e60e57beSChandler Carruth 
884532a508SEli Friedman   return parseAssembly(Buffer, Err, Context, nullptr, DataLayoutCallback);
89e60e57beSChandler Carruth }
90e60e57beSChandler Carruth 
914532a508SEli Friedman std::unique_ptr<Module>
parseIRFile(StringRef Filename,SMDiagnostic & Err,LLVMContext & Context,DataLayoutCallbackTy DataLayoutCallback)924532a508SEli Friedman llvm::parseIRFile(StringRef Filename, SMDiagnostic &Err, LLVMContext &Context,
934532a508SEli Friedman                   DataLayoutCallbackTy DataLayoutCallback) {
94adf21f2aSRafael Espindola   ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr =
95*c83cd8feSAbhina Sreeskantharajan       MemoryBuffer::getFileOrSTDIN(Filename, /*IsText=*/true);
96adf21f2aSRafael Espindola   if (std::error_code EC = FileOrErr.getError()) {
97e60e57beSChandler Carruth     Err = SMDiagnostic(Filename, SourceMgr::DK_Error,
98adf21f2aSRafael Espindola                        "Could not open input file: " + EC.message());
992617dcceSCraig Topper     return nullptr;
100e60e57beSChandler Carruth   }
101e60e57beSChandler Carruth 
102a8b2ddbdSAdrian Prantl   return parseIR(FileOrErr.get()->getMemBufferRef(), Err, Context,
1034532a508SEli Friedman                  DataLayoutCallback);
104e60e57beSChandler Carruth }
105285eed60SPeter Zotov 
106285eed60SPeter Zotov //===----------------------------------------------------------------------===//
107285eed60SPeter Zotov // C API.
108285eed60SPeter Zotov //===----------------------------------------------------------------------===//
109285eed60SPeter Zotov 
LLVMParseIRInContext(LLVMContextRef ContextRef,LLVMMemoryBufferRef MemBuf,LLVMModuleRef * OutM,char ** OutMessage)110285eed60SPeter Zotov LLVMBool LLVMParseIRInContext(LLVMContextRef ContextRef,
111285eed60SPeter Zotov                               LLVMMemoryBufferRef MemBuf, LLVMModuleRef *OutM,
112285eed60SPeter Zotov                               char **OutMessage) {
113285eed60SPeter Zotov   SMDiagnostic Diag;
114285eed60SPeter Zotov 
1155ebb7b31SAlp Toker   std::unique_ptr<MemoryBuffer> MB(unwrap(MemBuf));
116d96d553dSRafael Espindola   *OutM =
117d96d553dSRafael Espindola       wrap(parseIR(MB->getMemBufferRef(), Diag, *unwrap(ContextRef)).release());
118285eed60SPeter Zotov 
119285eed60SPeter Zotov   if(!*OutM) {
120285eed60SPeter Zotov     if (OutMessage) {
121e69170a1SAlp Toker       std::string buf;
122e69170a1SAlp Toker       raw_string_ostream os(buf);
123e69170a1SAlp Toker 
1242617dcceSCraig Topper       Diag.print(nullptr, os, false);
125e69170a1SAlp Toker       os.flush();
126e69170a1SAlp Toker 
127e69170a1SAlp Toker       *OutMessage = strdup(buf.c_str());
128285eed60SPeter Zotov     }
129285eed60SPeter Zotov     return 1;
130285eed60SPeter Zotov   }
131285eed60SPeter Zotov 
132285eed60SPeter Zotov   return 0;
133285eed60SPeter Zotov }
134