23#define DEBUG_TYPE "cg-data"
26using namespace cgdata;
30 cl::desc(
"Emit CodeGen Data into custom sections"));
33 cl::desc(
"File path to where .cgdata file is read"));
36 cl::desc(
"Enable two-round ThinLTO code generation. The first round "
37 "emits codegen data, while the second round uses the emitted "
38 "codegen data for further optimizations."));
41 const std::string &ErrMsg =
"") {
46 case cgdata_error::success:
49 case cgdata_error::eof:
52 case cgdata_error::bad_magic:
53 OS <<
"invalid codegen data (bad magic)";
55 case cgdata_error::bad_header:
56 OS <<
"invalid codegen data (file header is corrupt)";
58 case cgdata_error::empty_cgdata:
59 OS <<
"empty codegen data";
61 case cgdata_error::malformed:
62 OS <<
"malformed codegen data";
64 case cgdata_error::unsupported_version:
65 OS <<
"unsupported codegen data version";
81class CGDataErrorCategoryType :
public std::error_category {
82 const char *
name()
const noexcept
override {
return "llvm.cgdata"; }
84 std::string message(
int IE)
const override {
92 static CGDataErrorCategoryType ErrorCategory;
104const char *CodeGenDataSectNameCommon[] = {
105#define CG_DATA_SECT_ENTRY(Kind, SectNameCommon, SectNameCoff, Prefix) \
110const char *CodeGenDataSectNameCoff[] = {
111#define CG_DATA_SECT_ENTRY(Kind, SectNameCommon, SectNameCoff, Prefix) \
116const char *CodeGenDataSectNamePrefix[] = {
117#define CG_DATA_SECT_ENTRY(Kind, SectNameCommon, SectNameCoff, Prefix) Prefix,
127 bool AddSegmentInfo) {
128 std::string SectName;
131 SectName = CodeGenDataSectNamePrefix[CGSK];
134 SectName += CodeGenDataSectNameCoff[CGSK];
136 SectName += CodeGenDataSectNameCommon[CGSK];
141std::unique_ptr<CodeGenData> CodeGenData::Instance =
nullptr;
142std::once_flag CodeGenData::OnceFlag;
145 std::call_once(CodeGenData::OnceFlag, []() {
146 Instance = std::unique_ptr<CodeGenData>(
new CodeGenData());
149 Instance->EmitCGData =
true;
157 if (
Error E = ReaderOrErr.takeError()) {
162 auto Reader = ReaderOrErr->get();
163 if (Reader->hasOutlinedHashTree())
164 Instance->publishOutlinedHashTree(Reader->releaseOutlinedHashTree());
165 if (Reader->hasStableFunctionMap())
166 Instance->publishStableFunctionMap(Reader->releaseStableFunctionMap());
172namespace IndexedCGData {
175 using namespace support;
177 static_assert(std::is_standard_layout_v<llvm::IndexedCGData::Header>,
178 "The header should be standard layout type since we use offset "
179 "of fields to read.");
181 H.Magic = endian::readNext<uint64_t, endianness::little, unaligned>(Curr);
184 H.Version = endian::readNext<uint32_t, endianness::little, unaligned>(Curr);
187 H.DataKind = endian::readNext<uint32_t, endianness::little, unaligned>(Curr);
190 "Please update the offset computation below if a new field has "
191 "been added to the header.");
192 H.OutlinedHashTreeOffset =
193 endian::readNext<uint64_t, endianness::little, unaligned>(Curr);
195 H.StableFunctionMapOffset =
196 endian::readNext<uint64_t, endianness::little, unaligned>(Curr);
208 errs() << Whence <<
": ";
209 errs() << Message <<
"\n";
225 <<
" in Task " << Task <<
"\n");
230 std::unique_ptr<CachedFileStream> &Stream = *StreamOrErr;
235 if (
Error Err = Stream->commit())
244 <<
" in Task " << Task <<
"\n");
246 IRFiles[Task],
"in-memory IR file",
false);
250 Twine(
"Failed to parse optimized bitcode loaded for Task: ") +
255 return std::move(*RestoredModule);
262 for (
auto File : ObjFiles) {
266 File,
"in-memory object file",
false);
272 std::unique_ptr<object::ObjectFile> &Obj = BinOrErr.
get();
274 Obj.get(), GlobalOutlineRecord, GlobalStableFunctionMapRecord,
279 GlobalStableFunctionMapRecord.
finalize();
281 if (!GlobalOutlineRecord.
empty())
283 if (!GlobalStableFunctionMapRecord.
empty())
285 std::move(GlobalStableFunctionMapRecord.
FunctionMap));
static std::string getCGDataErrString(cgdata_error Err, const std::string &ErrMsg="")
cl::opt< bool > CodeGenDataThinLTOTwoRounds("codegen-data-thinlto-two-rounds", cl::init(false), cl::Hidden, cl::desc("Enable two-round ThinLTO code generation. The first round " "emits codegen data, while the second round uses the emitted " "codegen data for further optimizations."))
static cl::opt< std::string > CodeGenDataUsePath("codegen-data-use-path", cl::init(""), cl::Hidden, cl::desc("File path to where .cgdata file is read"))
static cl::opt< bool > CodeGenDataGenerate("codegen-data-generate", cl::init(false), cl::Hidden, cl::desc("Emit CodeGen Data into custom sections"))
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
Represents a module in a bitcode file.
StringRef getModuleIdentifier() const
std::string message() const override
Return the error message as a string.
static LLVM_ABI Expected< std::unique_ptr< CodeGenDataReader > > create(const Twine &Path, vfs::FileSystem &FS)
Factory method to create an appropriately typed reader for the given codegen data file path and file ...
static LLVM_ABI Error mergeFromObjectFile(const object::ObjectFile *Obj, OutlinedHashTreeRecord &GlobalOutlineRecord, StableFunctionMapRecord &GlobalFunctionMapRecord, stable_hash *CombinedHash=nullptr)
Extract the cgdata embedded in sections from the given object file and merge them into the GlobalOutl...
static LLVM_ABI CodeGenData & getInstance()
Lightweight error class with error context and mandatory checking.
bool isA() const
Check whether one error is a subclass of another.
Tagged union holding either a T or a Error.
Error takeError()
Take ownership of the stored error.
reference get()
Returns a reference to the stored T value.
This is an important class for using LLVM in a threaded context.
static std::unique_ptr< MemoryBuffer > getMemBuffer(StringRef InputData, StringRef BufferName="", bool RequiresNullTerminator=true)
Open the specified memory range as a MemoryBuffer.
A Module instance is used to store all the information related to an LLVM module.
const std::string & getModuleIdentifier() const
Get the module identifier which is, essentially, the name of the module.
StringRef - Represent a constant reference to a string, i.e.
constexpr bool empty() const
empty - Check if the string is empty.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
static LLVM_ABI raw_ostream & warning()
Convenience method for printing "warning: " to stderr.
static LLVM_ABI raw_ostream & note()
Convenience method for printing "note: " to stderr.
static Expected< OwningBinary< ObjectFile > > createObjectFile(StringRef ObjectPath)
A raw_ostream that writes to an std::string.
LLVM_ABI Expected< stable_hash > mergeCodeGenData(ArrayRef< StringRef > ObjectFiles)
Merge the codegen data from the scratch objects ObjectFiles from the first codegen round.
void publishOutlinedHashTree(std::unique_ptr< OutlinedHashTree > HashTree)
LLVM_ABI void warn(Error E, StringRef Whence="")
void publishStableFunctionMap(std::unique_ptr< StableFunctionMap > FunctionMap)
LLVM_ABI void saveModuleForTwoRounds(const Module &TheModule, unsigned Task, AddStreamFn AddStream)
Save TheModule before the first codegen round.
LLVM_ABI std::unique_ptr< Module > loadModuleForTwoRounds(BitcodeModule &OrigModule, unsigned Task, LLVMContext &Context, ArrayRef< StringRef > IRFiles)
Load the optimized bitcode module for the second codegen round.
initializer< Ty > init(const Ty &Val)
LLVM_ABI IntrusiveRefCntPtr< FileSystem > getRealFileSystem()
Gets an vfs::FileSystem for the 'real' file system, as seen by the operating system.
This is an optimization pass for GlobalISel generic memory operations.
LLVM_ABI Expected< std::unique_ptr< Module > > parseBitcodeFile(MemoryBufferRef Buffer, LLVMContext &Context, ParserCallbacks Callbacks={})
Read the specified bitcode file, returning the module.
LLVM_ABI void WriteBitcodeToFile(const Module &M, raw_ostream &Out, bool ShouldPreserveUseListOrder=false, const ModuleSummaryIndex *Index=nullptr, bool GenerateHash=false, ModuleHash *ModHash=nullptr)
Write the specified module to the specified raw output stream.
void handleAllErrors(Error E, HandlerTs &&... Handlers)
Behaves the same as handleErrors, except that by contract all errors must be handled by the given han...
std::function< Expected< std::unique_ptr< CachedFileStream > >(unsigned Task, const Twine &ModuleName)> AddStreamFn
This type defines the callback to add a file that is generated on the fly.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
LLVM_ABI const std::error_category & cgdata_category()
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
LLVM_ABI std::string getCodeGenDataSectionName(CGDataSectKind CGSK, Triple::ObjectFormatType OF, bool AddSegmentInfo=true)
std::unique_ptr< OutlinedHashTree > HashTree
The structure of the serialized stable function map is as follows:
void finalize(bool SkipTrim=false)
Finalize the stable function map by trimming content.
std::unique_ptr< StableFunctionMap > FunctionMap