56#define DEBUG_TYPE "asm-printer"
67 for (
MVT T : {MVT::i32, MVT::i64, MVT::f32, MVT::f64, MVT::v16i8, MVT::v8i16,
68 MVT::v4i32, MVT::v2i64, MVT::v4f32, MVT::v2f64, MVT::v8f16})
69 if (
TRI->isTypeLegalForClass(*TRC,
T))
71 LLVM_DEBUG(
errs() <<
"Unknown type for register number: " << RegNo);
79 "Unlowered physical register encountered during assembly printing");
80 assert(!MFI->isVRegStackified(RegNo));
81 unsigned WAReg = MFI->getWAReg(RegNo);
83 return '$' +
utostr(WAReg);
108 if (Name.front() ==
'"' && Name.back() ==
'"')
109 Name = Name.substr(1, Name.size() - 2);
110 return Name.starts_with(
"__invoke_");
142 std::string Ret =
"invoke_";
162 const bool EnableEmEH =
166 InvokeDetected =
true;
169 "Emscripten EH/SjLj does not support multivalue returns: " +
170 std::string(
F->getName()) +
": " +
185 "common symbols are not yet implemented for Wasm: " +
197 if (!Sym->getType()) {
227 if (WasmSym->getType())
236 if (Name ==
"__stack_pointer" || Name ==
"__tls_base" ||
237 Name ==
"__memory_base" || Name ==
"__table_base" ||
238 Name ==
"__tls_size" || Name ==
"__tls_align") {
240 Name ==
"__stack_pointer" || Name ==
"__tls_base";
249 if (Name.starts_with(
"GCC_except_table")) {
256 if (Name ==
"__cpp_exception" || Name ==
"__c_longjmp") {
258 WasmSym->setExternal(
true);
272 auto Signature =
OutContext.createWasmSignature();
273 Signature->Returns = std::move(Returns);
274 Signature->Params = std::move(Params);
275 WasmSym->setSignature(Signature);
281 std::optional<wasm::WasmSymbolType> WasmTy = Sym->
getType();
301 if (signaturesEmitted)
303 signaturesEmitted =
true;
312 if (WasmSym->isFunction()) {
323 auto Sym =
static_cast<MCSymbolWasm *
>(It.getValue().Symbol);
324 if (Sym && !Sym->isDefined())
329 for (
const auto &
F : M) {
345 bool InvokeDetected =
false;
352 if (InvokeDetected && !InvokeSymbols.
insert(Sym).second)
356 if (!Sym->getSignature()) {
357 Sym->setSignature(Signature);
362 if (
F.hasFnAttribute(
"wasm-import-module")) {
364 F.getFnAttribute(
"wasm-import-module").getValueAsString();
365 Sym->setImportModule(
OutContext.allocateString(Name));
368 if (
F.hasFnAttribute(
"wasm-import-name")) {
374 :
F.getFnAttribute(
"wasm-import-name").getValueAsString();
375 Sym->setImportName(
OutContext.allocateString(Name));
379 if (
F.hasFnAttribute(
"wasm-export-name")) {
381 StringRef Name =
F.getFnAttribute(
"wasm-export-name").getValueAsString();
382 Sym->setExportName(
OutContext.allocateString(Name));
400 for (
const auto &
F : M) {
401 if (!
F.isIntrinsic() &&
F.hasAddressTaken()) {
409 for (
const auto &
G : M.globals()) {
410 if (!
G.hasInitializer() &&
G.hasExternalLinkage() &&
412 G.getValueType()->isSized()) {
419 if (
const NamedMDNode *Named = M.getNamedMetadata(
"wasm.custom_sections")) {
420 for (
const Metadata *MD : Named->operands()) {
422 if (!Tuple || Tuple->getNumOperands() != 2)
426 if (!Name || !Contents)
430 std::string
SectionName = (
".custom_section." + Name->getString()).str();
448 for (
size_t I = 0, E =
Debug->getNumOperands();
I < E; ++
I) {
453 Language.consume_front(
"DW_LANG_");
454 if (SeenLanguages.
insert(Language).second)
460 if (
const NamedMDNode *Ident = M.getNamedMetadata(
"llvm.ident")) {
462 for (
size_t I = 0, E = Ident->getNumOperands();
I < E; ++
I) {
464 std::pair<StringRef, StringRef>
Field = S->getString().split(
"version");
467 if (SeenTools.
insert(Name).second)
472 int FieldCount = int(!Languages.
empty()) + int(!Tools.
empty());
473 if (FieldCount != 0) {
479 for (
auto &Producers : {std::make_pair(
"language", &Languages),
480 std::make_pair(
"processed-by", &Tools)}) {
481 if (Producers.second->empty())
483 OutStreamer->emitULEB128IntValue(strlen(Producers.first));
485 OutStreamer->emitULEB128IntValue(Producers.second->size());
486 for (
auto &Producer : *Producers.second) {
487 OutStreamer->emitULEB128IntValue(Producer.first.size());
489 OutStreamer->emitULEB128IntValue(Producer.second.size());
498 struct FeatureEntry {
505 auto EmitFeature = [&](std::string Feature) {
506 std::string MDKey = (
StringRef(
"wasm-feature-") + Feature).str();
507 Metadata *Policy = M.getModuleFlag(MDKey);
508 if (Policy ==
nullptr)
513 Entry.Name = Feature;
517 Entry.Prefix =
I->getZExtValue();
531 EmitFeature(
"shared-mem");
536 if (M.getDataLayout().getPointerSize() == 8) {
542 if (EmittedFeatures.
size() == 0)
552 for (
auto &
F : EmittedFeatures) {
562 auto V = M.getNamedGlobal(
"llvm.global.annotations");
583 CustomSections[AnnotationString].push_back(Sym);
587 for (
const auto &[Name, Symbols] : CustomSections) {
593 for (
auto &Sym : Symbols) {
604 assert(
MF->getConstantPool()->getConstants().empty() &&
605 "WebAssembly disables constant pools");
626 if (
MDNode *Idx =
F.getMetadata(
"wasm.index")) {
627 assert(Idx->getNumOperands() == 1);
642 WebAssembly_MC::verifyInstructionPredicates(
MI->getOpcode(),
643 Subtarget->getFeatureBits());
645 switch (
MI->getOpcode()) {
646 case WebAssembly::ARGUMENT_i32:
647 case WebAssembly::ARGUMENT_i32_S:
648 case WebAssembly::ARGUMENT_i64:
649 case WebAssembly::ARGUMENT_i64_S:
650 case WebAssembly::ARGUMENT_f32:
651 case WebAssembly::ARGUMENT_f32_S:
652 case WebAssembly::ARGUMENT_f64:
653 case WebAssembly::ARGUMENT_f64_S:
654 case WebAssembly::ARGUMENT_v16i8:
655 case WebAssembly::ARGUMENT_v16i8_S:
656 case WebAssembly::ARGUMENT_v8i16:
657 case WebAssembly::ARGUMENT_v8i16_S:
658 case WebAssembly::ARGUMENT_v4i32:
659 case WebAssembly::ARGUMENT_v4i32_S:
660 case WebAssembly::ARGUMENT_v2i64:
661 case WebAssembly::ARGUMENT_v2i64_S:
662 case WebAssembly::ARGUMENT_v4f32:
663 case WebAssembly::ARGUMENT_v4f32_S:
664 case WebAssembly::ARGUMENT_v2f64:
665 case WebAssembly::ARGUMENT_v2f64_S:
666 case WebAssembly::ARGUMENT_v8f16:
667 case WebAssembly::ARGUMENT_v8f16_S:
668 case WebAssembly::ARGUMENT_externref:
669 case WebAssembly::ARGUMENT_externref_S:
670 case WebAssembly::ARGUMENT_funcref:
671 case WebAssembly::ARGUMENT_funcref_S:
672 case WebAssembly::ARGUMENT_exnref:
673 case WebAssembly::ARGUMENT_exnref_S:
677 case WebAssembly::FALLTHROUGH_RETURN: {
686 case WebAssembly::COMPILER_FENCE:
690 case WebAssembly::CATCH:
691 case WebAssembly::CATCH_S:
692 case WebAssembly::CATCH_REF:
693 case WebAssembly::CATCH_REF_S:
694 case WebAssembly::CATCH_ALL:
695 case WebAssembly::CATCH_ALL_S:
696 case WebAssembly::CATCH_ALL_REF:
697 case WebAssembly::CATCH_ALL_REF_S:
704 MCInstLowering.
lower(
MI, TmpInst);
713 const char *ExtraCode,
728 assert(
MI->getOpcode() == WebAssembly::INLINEASM);
751 const char *ExtraCode,
765 "WebAssembly Assmebly Printer",
false,
false)
769LLVMInitializeWebAssemblyAsmPrinter() {
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Function Alias Analysis Results
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
#define LLVM_EXTERNAL_VISIBILITY
Module.h This file contains the declarations for the Module class.
This file declares the MachineConstantPool class which is an abstract constant pool to keep track of ...
Register const TargetRegisterInfo * TRI
This file implements a map that provides insertion order iteration.
OptimizedStructLayoutField Field
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
static StringRef getName(Value *V)
This file defines the SmallSet class.
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
static std::string getEmscriptenInvokeSymbolName(wasm::WasmSignature *Sig)
static bool isEmscriptenInvokeName(StringRef Name)
static char getInvokeSig(wasm::ValType VT)
cl::opt< bool > WasmKeepRegisters
This file contains the declaration of the WebAssemblyMCAsmInfo class.
This file declares the class to lower WebAssembly MachineInstrs to their corresponding MCInst records...
This file provides WebAssembly-specific target descriptions.
This file declares WebAssembly-specific per-machine-function information.
This file contains the WebAssembly implementation of the WebAssemblyRegisterInfo class.
This file provides signature information for runtime libcalls.
This file registers the WebAssembly target.
This file declares the WebAssembly-specific subclass of TargetMachine.
This file declares WebAssembly-specific target streamer classes.
This file contains the declaration of the WebAssembly-specific type parsing utility functions.
This file contains the declaration of the WebAssembly-specific utility functions.
This file contains the entry points for global functions defined in the LLVM WebAssembly back-end.
MCSymbol * getSymbol(const GlobalValue *GV) const
void EmitToStreamer(MCStreamer &S, const MCInst &Inst)
virtual void emitGlobalVariable(const GlobalVariable *GV)
Emit the specified global variable to the .s file.
TargetMachine & TM
Target machine description.
virtual void PrintSymbolOperand(const MachineOperand &MO, raw_ostream &OS)
Print the MachineOperand as a symbol.
const MCAsmInfo * MAI
Target Asm Printer information.
MachineFunction * MF
The current machine function.
virtual const MCExpr * lowerConstant(const Constant *CV, const Constant *BaseCV=nullptr, uint64_t Offset=0)
Lower the specified LLVM Constant to an MCExpr.
virtual void emitFunctionBodyStart()
Targets can override this to emit stuff before the first basic block in the function.
virtual void emitLinkage(const GlobalValue *GV, MCSymbol *GVSym) const
This emits linkage information about GVSym based on GV, if this is supported by the target.
void printOffset(int64_t Offset, raw_ostream &OS) const
This is just convenient handler for printing offsets.
MCSymbol * getSymbolPreferLocal(const GlobalValue &GV) const
Similar to getSymbol() but preferred for references.
MCSymbol * CurrentFnSym
The symbol for the current function.
MachineModuleInfo * MMI
This is a pointer to the current MachineModuleInfo.
MCContext & OutContext
This is the context for the output file that we are streaming.
void emitVisibility(MCSymbol *Sym, unsigned Visibility, bool IsDefinition=true) const
This emits visibility information about symbol, if this is supported by the target.
std::unique_ptr< MCStreamer > OutStreamer
This is the MCStreamer object for the file we are generating.
virtual bool PrintAsmMemoryOperand(const MachineInstr *MI, unsigned OpNo, const char *ExtraCode, raw_ostream &OS)
Print the specified operand of MI, an INLINEASM instruction, using the specified assembler variant as...
bool isVerbose() const
Return true if assembly output should contain comments.
MCSymbol * GetExternalSymbolSymbol(const Twine &Sym) const
Return the MCSymbol for the specified ExternalSymbol.
@ Debug
Emit .debug_frame.
virtual bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNo, const char *ExtraCode, raw_ostream &OS)
Print the specified operand of MI, an INLINEASM instruction, using the specified assembler variant.
ConstantArray - Constant Array Declarations.
Implements a dense probed hash-table based set.
bool isThreadLocal() const
If the value is "Thread Local", its value isn't shared by the threads.
VisibilityTypes getVisibility() const
LLVM_ABI bool isDeclaration() const
Return true if the primary definition of this global value is outside of the current translation unit...
unsigned getAddressSpace() const
Module * getParent()
Get the module that this global value is contained inside of...
LLVM_ABI const DataLayout & getDataLayout() const
Get the data layout of the module this global belongs to.
bool hasCommonLinkage() const
Type * getValueType() const
bool hasInitializer() const
Definitions have initializers, declarations don't.
static LLVM_ABI const MCConstantExpr * create(int64_t Value, MCContext &Ctx, bool PrintInHex=false, unsigned SizeInBytes=0)
Instances of this class represent a single low-level machine instruction.
This represents a section on wasm.
static const MCSymbolRefExpr * create(const MCSymbol *Symbol, MCContext &Ctx, SMLoc Loc=SMLoc())
void setSignature(wasm::WasmSignature *Sig)
std::optional< wasm::WasmSymbolType > getType() const
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
LLVM_ABI void print(raw_ostream &OS, const MCAsmInfo *MAI) const
print - Print the value to the stream OS.
Target specific streamer interface.
LLVM_ABI StringRef getString() const
LLVM_ABI MCSymbol * getSymbol() const
Return the MCSymbol for this basic block.
Representation of each machine instruction.
MachineModuleInfoWasm - This is a MachineModuleInfoImpl implementation for Wasm targets.
SetVector< StringRef > MachineSymbolsUsed
MachineOperand class - Representation of each machine instruction operand.
MachineBasicBlock * getMBB() const
MachineOperandType getType() const
getType - Returns the MachineOperandType for this operand.
const char * getSymbolName() const
Register getReg() const
getReg - Returns the register number.
@ MO_Immediate
Immediate operand.
@ MO_GlobalAddress
Address of a global value.
@ MO_MachineBasicBlock
MachineBasicBlock reference.
@ MO_Register
Register operand.
@ MO_ExternalSymbol
Name of external global symbol.
int64_t getOffset() const
Return the offset from the symbol in this operand.
This class implements a map that also provides access to all stored values in a deterministic order.
A Module instance is used to store all the information related to an LLVM module.
LLVMContext & getContext() const
Get the global data context.
Wrapper class representing virtual and physical registers.
constexpr bool isVirtual() const
Return true if the specified register number is in the virtual register namespace.
Represents a location in source code.
static SectionKind getMetadata()
SmallSet - This maintains a set of unique values, optimizing for the case when the set is small (less...
std::pair< const_iterator, bool > insert(const T &V)
insert - Insert an element into the set if it isn't already there.
reference emplace_back(ArgTypes &&... Args)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringRef - Represent a constant reference to a string, i.e.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
The instances of the Type class are immutable: once they are created, they are never changed.
LLVM Value Representation.
LLVM_ABI const Value * stripPointerCasts() const
Strip off pointer casts, all-zero GEPs and address space casts.
void emitJumpTableInfo() override
Print assembly representations of the jump tables used by the current function to the current output ...
void EmitProducerInfo(Module &M)
void emitGlobalVariable(const GlobalVariable *GV) override
Emit the specified global variable to the .s file.
void EmitTargetFeatures(Module &M)
bool PrintAsmMemoryOperand(const MachineInstr *MI, unsigned OpNo, const char *ExtraCode, raw_ostream &OS) override
Print the specified operand of MI, an INLINEASM instruction, using the specified assembler variant as...
const WebAssemblySubtarget & getSubtarget() const
WebAssemblyTargetStreamer * getTargetStreamer()
void emitInstruction(const MachineInstr *MI) override
Targets should implement this to emit instructions.
std::string regToString(const MachineOperand &MO)
void emitSymbolType(const MCSymbolWasm *Sym)
MCSymbol * getOrCreateWasmSymbol(StringRef Name)
void emitConstantPool() override
Print to the current output stream assembly representations of the constants in the constant pool MCP...
MVT getRegType(unsigned RegNo) const
void emitDecls(const Module &M)
void emitFunctionBodyStart() override
Targets can override this to emit stuff before the first basic block in the function.
bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNo, const char *ExtraCode, raw_ostream &OS) override
Print the specified operand of MI, an INLINEASM instruction, using the specified assembler variant.
void EmitFunctionAttributes(Module &M)
void emitEndOfAsmFile(Module &M) override
This virtual method can be overridden by targets that want to emit something at the end of their file...
MCSymbolWasm * getMCSymbolForFunction(const Function *F, wasm::WasmSignature *Sig, bool &InvokeDetected)
This class is used to lower an MachineInstr into an MCInst.
void lower(const MachineInstr *MI, MCInst &OutMI) const
WebAssembly-specific streamer interface, to implement support WebAssembly-specific assembly directive...
virtual void emitFunctionType(const MCSymbolWasm *Sym)=0
.functype
virtual void emitLocal(ArrayRef< wasm::ValType > Types)=0
.local
virtual void emitTagType(const MCSymbolWasm *Sym)=0
.tagtype
virtual void emitExportName(const MCSymbolWasm *Sym, StringRef ExportName)=0
.export_name
virtual void emitGlobalType(const MCSymbolWasm *Sym)=0
.globaltype
virtual void emitImportModule(const MCSymbolWasm *Sym, StringRef ImportModule)=0
.import_module
virtual void emitTableType(const MCSymbolWasm *Sym)=0
.tabletype
virtual void emitImportName(const MCSymbolWasm *Sym, StringRef ImportName)=0
.import_name
virtual void emitIndIdx(const MCExpr *Value)=0
.indidx
std::pair< iterator, bool > insert(const ValueT &V)
This class implements an extremely fast bulk output stream that can only output to a stream.
LLVM_ABI StringRef LanguageString(unsigned Language)
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
MCSymbolWasm * getOrCreateFunctionTableSymbol(MCContext &Ctx, const WebAssemblySubtarget *Subtarget)
Returns the __indirect_function_table, for use in call_indirect and in function bitcasts.
void wasmSymbolSetType(MCSymbolWasm *Sym, const Type *GlobalVT, ArrayRef< MVT > VTs)
Sets a Wasm Symbol Type.
static const unsigned UnusedReg
cl::opt< bool > WasmEnableEmEH
cl::opt< bool > WasmEnableEmSjLj
std::string signatureToString(const wasm::WasmSignature *Sig)
void getLibcallSignature(const WebAssemblySubtarget &Subtarget, RTLIB::Libcall LC, SmallVectorImpl< wasm::ValType > &Rets, SmallVectorImpl< wasm::ValType > &Params)
bool isWasmVarAddressSpace(unsigned AS)
@ WASM_FEATURE_PREFIX_USED
@ WASM_FEATURE_PREFIX_DISALLOWED
@ WASM_SYMBOL_TYPE_GLOBAL
@ WASM_SYMBOL_TYPE_FUNCTION
This is an optimization pass for GlobalISel generic memory operations.
void computeSignatureVTs(const FunctionType *Ty, const Function *TargetFunc, const Function &ContextFunc, const TargetMachine &TM, SmallVectorImpl< MVT > &Params, SmallVectorImpl< MVT > &Results)
const SubtargetFeatureKV WebAssemblyFeatureKV[WebAssembly::NumSubtargetFeatures]
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI bool getConstantStringInfo(const Value *V, StringRef &Str, bool TrimAtNul=true)
This function computes the length of a null-terminated C string pointed to by V.
std::string utostr(uint64_t X, bool isNeg=false)
FunctionAddr VTableAddr uintptr_t uintptr_t Version
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Target & getTheWebAssemblyTarget32()
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
Target & getTheWebAssemblyTarget64()
DWARFExpression::Operation Op
void valTypesFromMVTs(ArrayRef< MVT > In, SmallVectorImpl< wasm::ValType > &Out)
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
wasm::WasmSignature * signatureFromMVTs(MCContext &Ctx, const SmallVectorImpl< MVT > &Results, const SmallVectorImpl< MVT > &Params)
void computeLegalValueVTs(const WebAssemblyTargetLowering &TLI, LLVMContext &Ctx, const DataLayout &DL, Type *Ty, SmallVectorImpl< MVT > &ValueVTs)
@ MCSA_NoDeadStrip
.no_dead_strip (MachO)
RegisterAsmPrinter - Helper template for registering a target specific assembly printer,...
Used to provide key value pairs for feature and CPU bit flags.
SmallVector< ValType, 1 > Returns
SmallVector< ValType, 4 > Params