You've already forked linux-packaging-mono
Imported Upstream version 5.18.0.182
Former-commit-id: f9d55cf82631bfd710c387739687e5845296aea1
This commit is contained in:
parent
8625704ad8
commit
b716dc8d12
226
external/llvm/tools/llvm-readobj/WasmDumper.cpp
vendored
226
external/llvm/tools/llvm-readobj/WasmDumper.cpp
vendored
@ -1,226 +0,0 @@
|
||||
//===-- WasmDumper.cpp - Wasm-specific object file dumper -----------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file implements the Wasm-specific dumper for llvm-readobj.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "Error.h"
|
||||
#include "ObjDumper.h"
|
||||
#include "llvm-readobj.h"
|
||||
#include "llvm/Object/Wasm.h"
|
||||
#include "llvm/Support/ScopedPrinter.h"
|
||||
|
||||
using namespace llvm;
|
||||
using namespace object;
|
||||
|
||||
namespace {
|
||||
|
||||
static const EnumEntry<unsigned> WasmSymbolTypes[] = {
|
||||
#define ENUM_ENTRY(X) { #X, static_cast<unsigned>(WasmSymbol::SymbolType::X) }
|
||||
ENUM_ENTRY(FUNCTION_IMPORT),
|
||||
ENUM_ENTRY(FUNCTION_EXPORT),
|
||||
ENUM_ENTRY(GLOBAL_IMPORT),
|
||||
ENUM_ENTRY(GLOBAL_EXPORT),
|
||||
ENUM_ENTRY(DEBUG_FUNCTION_NAME),
|
||||
#undef ENUM_ENTRY
|
||||
};
|
||||
|
||||
static const EnumEntry<uint32_t> WasmSectionTypes[] = {
|
||||
#define ENUM_ENTRY(X) { #X, wasm::WASM_SEC_##X }
|
||||
ENUM_ENTRY(CUSTOM),
|
||||
ENUM_ENTRY(TYPE),
|
||||
ENUM_ENTRY(IMPORT),
|
||||
ENUM_ENTRY(FUNCTION),
|
||||
ENUM_ENTRY(TABLE),
|
||||
ENUM_ENTRY(MEMORY),
|
||||
ENUM_ENTRY(GLOBAL),
|
||||
ENUM_ENTRY(EXPORT),
|
||||
ENUM_ENTRY(START),
|
||||
ENUM_ENTRY(ELEM),
|
||||
ENUM_ENTRY(CODE),
|
||||
ENUM_ENTRY(DATA),
|
||||
#undef ENUM_ENTRY
|
||||
};
|
||||
|
||||
class WasmDumper : public ObjDumper {
|
||||
public:
|
||||
WasmDumper(const WasmObjectFile *Obj, ScopedPrinter &Writer)
|
||||
: ObjDumper(Writer), Obj(Obj) {}
|
||||
|
||||
void printFileHeaders() override;
|
||||
void printSections() override;
|
||||
void printRelocations() override;
|
||||
void printSymbols() override;
|
||||
void printDynamicSymbols() override { llvm_unreachable("unimplemented"); }
|
||||
void printUnwindInfo() override { llvm_unreachable("unimplemented"); }
|
||||
void printStackMap() const override { llvm_unreachable("unimplemented"); }
|
||||
|
||||
protected:
|
||||
void printSymbol(const SymbolRef &Sym);
|
||||
void printRelocation(const SectionRef &Section, const RelocationRef &Reloc);
|
||||
|
||||
private:
|
||||
const WasmObjectFile *Obj;
|
||||
};
|
||||
|
||||
void WasmDumper::printFileHeaders() {
|
||||
W.printHex("Version", Obj->getHeader().Version);
|
||||
}
|
||||
|
||||
void WasmDumper::printRelocation(const SectionRef &Section,
|
||||
const RelocationRef &Reloc) {
|
||||
SmallString<64> RelocTypeName;
|
||||
uint64_t RelocType = Reloc.getType();
|
||||
Reloc.getTypeName(RelocTypeName);
|
||||
const wasm::WasmRelocation &WasmReloc = Obj->getWasmRelocation(Reloc);
|
||||
|
||||
bool HasAddend = false;
|
||||
switch (RelocType) {
|
||||
case wasm::R_WEBASSEMBLY_MEMORY_ADDR_LEB:
|
||||
case wasm::R_WEBASSEMBLY_MEMORY_ADDR_SLEB:
|
||||
case wasm::R_WEBASSEMBLY_MEMORY_ADDR_I32:
|
||||
HasAddend = true;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
if (opts::ExpandRelocs) {
|
||||
DictScope Group(W, "Relocation");
|
||||
W.printNumber("Type", RelocTypeName, RelocType);
|
||||
W.printHex("Offset", Reloc.getOffset());
|
||||
W.printHex("Index", WasmReloc.Index);
|
||||
if (HasAddend)
|
||||
W.printNumber("Addend", WasmReloc.Addend);
|
||||
} else {
|
||||
raw_ostream& OS = W.startLine();
|
||||
OS << W.hex(Reloc.getOffset()) << " " << RelocTypeName << "["
|
||||
<< WasmReloc.Index << "]";
|
||||
if (HasAddend)
|
||||
OS << " " << WasmReloc.Addend;
|
||||
OS << "\n";
|
||||
}
|
||||
}
|
||||
|
||||
void WasmDumper::printRelocations() {
|
||||
ListScope D(W, "Relocations");
|
||||
|
||||
int SectionNumber = 0;
|
||||
for (const SectionRef &Section : Obj->sections()) {
|
||||
bool PrintedGroup = false;
|
||||
StringRef Name;
|
||||
error(Section.getName(Name));
|
||||
++SectionNumber;
|
||||
|
||||
for (const RelocationRef &Reloc : Section.relocations()) {
|
||||
if (!PrintedGroup) {
|
||||
W.startLine() << "Section (" << SectionNumber << ") " << Name << " {\n";
|
||||
W.indent();
|
||||
PrintedGroup = true;
|
||||
}
|
||||
|
||||
printRelocation(Section, Reloc);
|
||||
}
|
||||
|
||||
if (PrintedGroup) {
|
||||
W.unindent();
|
||||
W.startLine() << "}\n";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void WasmDumper::printSymbols() {
|
||||
ListScope Group(W, "Symbols");
|
||||
|
||||
for (const SymbolRef &Symbol : Obj->symbols())
|
||||
printSymbol(Symbol);
|
||||
}
|
||||
|
||||
void WasmDumper::printSections() {
|
||||
ListScope Group(W, "Sections");
|
||||
for (const SectionRef &Section : Obj->sections()) {
|
||||
const WasmSection &WasmSec = Obj->getWasmSection(Section);
|
||||
DictScope SectionD(W, "Section");
|
||||
W.printEnum("Type", WasmSec.Type, makeArrayRef(WasmSectionTypes));
|
||||
W.printNumber("Size", static_cast<uint64_t>(WasmSec.Content.size()));
|
||||
W.printNumber("Offset", WasmSec.Offset);
|
||||
switch (WasmSec.Type) {
|
||||
case wasm::WASM_SEC_CUSTOM:
|
||||
W.printString("Name", WasmSec.Name);
|
||||
if (WasmSec.Name == "linking") {
|
||||
const wasm::WasmLinkingData &LinkingData = Obj->linkingData();
|
||||
W.printNumber("DataSize", LinkingData.DataSize);
|
||||
if (!LinkingData.InitFunctions.empty()) {
|
||||
ListScope Group(W, "InitFunctions");
|
||||
for (const wasm::WasmInitFunc &F: LinkingData.InitFunctions)
|
||||
W.startLine() << F.FunctionIndex << " (priority=" << F.Priority
|
||||
<< ")\n";
|
||||
}
|
||||
}
|
||||
break;
|
||||
case wasm::WASM_SEC_DATA: {
|
||||
ListScope Group(W, "Segments");
|
||||
for (const WasmSegment &Segment : Obj->dataSegments()) {
|
||||
const wasm::WasmDataSegment& Seg = Segment.Data;
|
||||
DictScope Group(W, "Segment");
|
||||
if (!Seg.Name.empty())
|
||||
W.printString("Name", Seg.Name);
|
||||
W.printNumber("Size", static_cast<uint64_t>(Seg.Content.size()));
|
||||
if (Seg.Offset.Opcode == wasm::WASM_OPCODE_I32_CONST)
|
||||
W.printNumber("Offset", Seg.Offset.Value.Int32);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case wasm::WASM_SEC_MEMORY:
|
||||
ListScope Group(W, "Memories");
|
||||
for (const wasm::WasmLimits &Memory : Obj->memories()) {
|
||||
DictScope Group(W, "Memory");
|
||||
W.printNumber("InitialPages", Memory.Initial);
|
||||
if (Memory.Flags & wasm::WASM_LIMITS_FLAG_HAS_MAX) {
|
||||
W.printNumber("MaxPages", WasmSec.Offset);
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
if (opts::SectionRelocations) {
|
||||
ListScope D(W, "Relocations");
|
||||
for (const RelocationRef &Reloc : Section.relocations())
|
||||
printRelocation(Section, Reloc);
|
||||
}
|
||||
|
||||
if (opts::SectionData) {
|
||||
W.printBinaryBlock("SectionData", WasmSec.Content);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void WasmDumper::printSymbol(const SymbolRef &Sym) {
|
||||
DictScope D(W, "Symbol");
|
||||
WasmSymbol Symbol = Obj->getWasmSymbol(Sym.getRawDataRefImpl());
|
||||
W.printString("Name", Symbol.Name);
|
||||
W.printEnum("Type", static_cast<unsigned>(Symbol.Type), makeArrayRef(WasmSymbolTypes));
|
||||
W.printHex("Flags", Symbol.Flags);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
namespace llvm {
|
||||
|
||||
std::error_code createWasmDumper(const object::ObjectFile *Obj,
|
||||
ScopedPrinter &Writer,
|
||||
std::unique_ptr<ObjDumper> &Result) {
|
||||
const WasmObjectFile *WasmObj = dyn_cast<WasmObjectFile>(Obj);
|
||||
assert(WasmObj && "createWasmDumper called with non-wasm object");
|
||||
|
||||
Result.reset(new WasmDumper(WasmObj, Writer));
|
||||
return readobj_error::success;
|
||||
}
|
||||
|
||||
} // namespace llvm
|
Reference in New Issue
Block a user