diff options
Diffstat (limited to 'llvm/lib/Target/WebAssembly/MCTargetDesc')
12 files changed, 1927 insertions, 0 deletions
| diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyAsmBackend.cpp b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyAsmBackend.cpp new file mode 100644 index 000000000000..8314de41021f --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyAsmBackend.cpp @@ -0,0 +1,135 @@ +//===-- WebAssemblyAsmBackend.cpp - WebAssembly Assembler Backend ---------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file implements the WebAssemblyAsmBackend class. +/// +//===----------------------------------------------------------------------===// + +#include "MCTargetDesc/WebAssemblyFixupKinds.h" +#include "MCTargetDesc/WebAssemblyMCTargetDesc.h" +#include "llvm/MC/MCAsmBackend.h" +#include "llvm/MC/MCAssembler.h" +#include "llvm/MC/MCDirectives.h" +#include "llvm/MC/MCExpr.h" +#include "llvm/MC/MCFixupKindInfo.h" +#include "llvm/MC/MCObjectWriter.h" +#include "llvm/MC/MCSubtargetInfo.h" +#include "llvm/MC/MCSymbol.h" +#include "llvm/MC/MCWasmObjectWriter.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/raw_ostream.h" + +using namespace llvm; + +namespace { + +class WebAssemblyAsmBackend final : public MCAsmBackend { +  bool Is64Bit; +  bool IsEmscripten; + +public: +  explicit WebAssemblyAsmBackend(bool Is64Bit, bool IsEmscripten) +      : MCAsmBackend(support::little), Is64Bit(Is64Bit), +        IsEmscripten(IsEmscripten) {} + +  unsigned getNumFixupKinds() const override { +    return WebAssembly::NumTargetFixupKinds; +  } + +  const MCFixupKindInfo &getFixupKindInfo(MCFixupKind Kind) const override; + +  void applyFixup(const MCAssembler &Asm, const MCFixup &Fixup, +                  const MCValue &Target, MutableArrayRef<char> Data, +                  uint64_t Value, bool IsPCRel, +                  const MCSubtargetInfo *STI) const override; + +  std::unique_ptr<MCObjectTargetWriter> +  createObjectTargetWriter() const override; + +  // No instruction requires relaxation +  bool fixupNeedsRelaxation(const MCFixup &Fixup, uint64_t Value, +                            const MCRelaxableFragment *DF, +                            const MCAsmLayout &Layout) const override { +    return false; +  } + +  bool mayNeedRelaxation(const MCInst &Inst, +                         const MCSubtargetInfo &STI) const override { +    return false; +  } + +  void relaxInstruction(const MCInst &Inst, const MCSubtargetInfo &STI, +                        MCInst &Res) const override {} + +  bool writeNopData(raw_ostream &OS, uint64_t Count) const override; +}; + +const MCFixupKindInfo & +WebAssemblyAsmBackend::getFixupKindInfo(MCFixupKind Kind) const { +  const static MCFixupKindInfo Infos[WebAssembly::NumTargetFixupKinds] = { +      // This table *must* be in the order that the fixup_* kinds are defined in +      // WebAssemblyFixupKinds.h. +      // +      // Name                     Offset (bits) Size (bits)     Flags +      {"fixup_sleb128_i32", 0, 5 * 8, 0}, +      {"fixup_sleb128_i64", 0, 10 * 8, 0}, +      {"fixup_uleb128_i32", 0, 5 * 8, 0}, +  }; + +  if (Kind < FirstTargetFixupKind) +    return MCAsmBackend::getFixupKindInfo(Kind); + +  assert(unsigned(Kind - FirstTargetFixupKind) < getNumFixupKinds() && +         "Invalid kind!"); +  return Infos[Kind - FirstTargetFixupKind]; +} + +bool WebAssemblyAsmBackend::writeNopData(raw_ostream &OS, +                                         uint64_t Count) const { +  for (uint64_t I = 0; I < Count; ++I) +    OS << char(WebAssembly::Nop); + +  return true; +} + +void WebAssemblyAsmBackend::applyFixup(const MCAssembler &Asm, +                                       const MCFixup &Fixup, +                                       const MCValue &Target, +                                       MutableArrayRef<char> Data, +                                       uint64_t Value, bool IsPCRel, +                                       const MCSubtargetInfo *STI) const { +  const MCFixupKindInfo &Info = getFixupKindInfo(Fixup.getKind()); +  assert(Info.Flags == 0 && "WebAssembly does not use MCFixupKindInfo flags"); + +  unsigned NumBytes = alignTo(Info.TargetSize, 8) / 8; +  if (Value == 0) +    return; // Doesn't change encoding. + +  // Shift the value into position. +  Value <<= Info.TargetOffset; + +  unsigned Offset = Fixup.getOffset(); +  assert(Offset + NumBytes <= Data.size() && "Invalid fixup offset!"); + +  // For each byte of the fragment that the fixup touches, mask in the +  // bits from the fixup value. +  for (unsigned I = 0; I != NumBytes; ++I) +    Data[Offset + I] |= uint8_t((Value >> (I * 8)) & 0xff); +} + +std::unique_ptr<MCObjectTargetWriter> +WebAssemblyAsmBackend::createObjectTargetWriter() const { +  return createWebAssemblyWasmObjectWriter(Is64Bit, IsEmscripten); +} + +} // end anonymous namespace + +MCAsmBackend *llvm::createWebAssemblyAsmBackend(const Triple &TT) { +  return new WebAssemblyAsmBackend(TT.isArch64Bit(), TT.isOSEmscripten()); +} diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyFixupKinds.h b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyFixupKinds.h new file mode 100644 index 000000000000..33e8de282955 --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyFixupKinds.h @@ -0,0 +1,28 @@ +//=- WebAssemblyFixupKinds.h - WebAssembly Specific Fixup Entries -*- C++ -*-=// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYFIXUPKINDS_H +#define LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYFIXUPKINDS_H + +#include "llvm/MC/MCFixup.h" + +namespace llvm { +namespace WebAssembly { +enum Fixups { +  fixup_sleb128_i32 = FirstTargetFixupKind, // 32-bit signed +  fixup_sleb128_i64,                        // 64-bit signed +  fixup_uleb128_i32,                        // 32-bit unsigned + +  // Marker +  LastTargetFixupKind, +  NumTargetFixupKinds = LastTargetFixupKind - FirstTargetFixupKind +}; +} // end namespace WebAssembly +} // end namespace llvm + +#endif diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyInstPrinter.cpp b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyInstPrinter.cpp new file mode 100644 index 000000000000..221ac17b8336 --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyInstPrinter.cpp @@ -0,0 +1,339 @@ +//=- WebAssemblyInstPrinter.cpp - WebAssembly assembly instruction printing -=// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// Print MCInst instructions to wasm format. +/// +//===----------------------------------------------------------------------===// + +#include "MCTargetDesc/WebAssemblyInstPrinter.h" +#include "MCTargetDesc/WebAssemblyMCTargetDesc.h" +#include "WebAssembly.h" +#include "WebAssemblyMachineFunctionInfo.h" +#include "WebAssemblyUtilities.h" +#include "llvm/ADT/SmallSet.h" +#include "llvm/ADT/StringExtras.h" +#include "llvm/CodeGen/TargetRegisterInfo.h" +#include "llvm/MC/MCExpr.h" +#include "llvm/MC/MCInst.h" +#include "llvm/MC/MCInstrInfo.h" +#include "llvm/MC/MCSubtargetInfo.h" +#include "llvm/MC/MCSymbol.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/FormattedStream.h" +using namespace llvm; + +#define DEBUG_TYPE "asm-printer" + +#include "WebAssemblyGenAsmWriter.inc" + +WebAssemblyInstPrinter::WebAssemblyInstPrinter(const MCAsmInfo &MAI, +                                               const MCInstrInfo &MII, +                                               const MCRegisterInfo &MRI) +    : MCInstPrinter(MAI, MII, MRI) {} + +void WebAssemblyInstPrinter::printRegName(raw_ostream &OS, +                                          unsigned RegNo) const { +  assert(RegNo != WebAssemblyFunctionInfo::UnusedReg); +  // Note that there's an implicit local.get/local.set here! +  OS << "$" << RegNo; +} + +void WebAssemblyInstPrinter::printInst(const MCInst *MI, raw_ostream &OS, +                                       StringRef Annot, +                                       const MCSubtargetInfo &STI) { +  // Print the instruction (this uses the AsmStrings from the .td files). +  printInstruction(MI, OS); + +  // Print any additional variadic operands. +  const MCInstrDesc &Desc = MII.get(MI->getOpcode()); +  if (Desc.isVariadic()) { +    if (Desc.getNumOperands() == 0 && MI->getNumOperands() > 0) +      OS << "\t"; +    for (auto I = Desc.getNumOperands(), E = MI->getNumOperands(); I < E; ++I) { +      // FIXME: For CALL_INDIRECT_VOID, don't print a leading comma, because +      // we have an extra flags operand which is not currently printed, for +      // compatiblity reasons. +      if (I != 0 && ((MI->getOpcode() != WebAssembly::CALL_INDIRECT_VOID && +                      MI->getOpcode() != WebAssembly::CALL_INDIRECT_VOID_S) || +                     I != Desc.getNumOperands())) +        OS << ", "; +      printOperand(MI, I, OS); +    } +  } + +  // Print any added annotation. +  printAnnotation(OS, Annot); + +  if (CommentStream) { +    // Observe any effects on the control flow stack, for use in annotating +    // control flow label references. +    unsigned Opc = MI->getOpcode(); +    switch (Opc) { +    default: +      break; + +    case WebAssembly::LOOP: +    case WebAssembly::LOOP_S: +      printAnnotation(OS, "label" + utostr(ControlFlowCounter) + ':'); +      ControlFlowStack.push_back(std::make_pair(ControlFlowCounter++, true)); +      break; + +    case WebAssembly::BLOCK: +    case WebAssembly::BLOCK_S: +      ControlFlowStack.push_back(std::make_pair(ControlFlowCounter++, false)); +      break; + +    case WebAssembly::TRY: +    case WebAssembly::TRY_S: +      ControlFlowStack.push_back(std::make_pair(ControlFlowCounter++, false)); +      EHPadStack.push_back(EHPadStackCounter++); +      LastSeenEHInst = TRY; +      break; + +    case WebAssembly::END_LOOP: +    case WebAssembly::END_LOOP_S: +      if (ControlFlowStack.empty()) { +        printAnnotation(OS, "End marker mismatch!"); +      } else { +        ControlFlowStack.pop_back(); +      } +      break; + +    case WebAssembly::END_BLOCK: +    case WebAssembly::END_BLOCK_S: +      if (ControlFlowStack.empty()) { +        printAnnotation(OS, "End marker mismatch!"); +      } else { +        printAnnotation( +            OS, "label" + utostr(ControlFlowStack.pop_back_val().first) + ':'); +      } +      break; + +    case WebAssembly::END_TRY: +    case WebAssembly::END_TRY_S: +      if (ControlFlowStack.empty()) { +        printAnnotation(OS, "End marker mismatch!"); +      } else { +        printAnnotation( +            OS, "label" + utostr(ControlFlowStack.pop_back_val().first) + ':'); +        LastSeenEHInst = END_TRY; +      } +      break; + +    case WebAssembly::CATCH: +    case WebAssembly::CATCH_S: +      if (EHPadStack.empty()) { +        printAnnotation(OS, "try-catch mismatch!"); +      } else { +        printAnnotation(OS, "catch" + utostr(EHPadStack.pop_back_val()) + ':'); +      } +      break; +    } + +    // Annotate any control flow label references. + +    // rethrow instruction does not take any depth argument and rethrows to the +    // nearest enclosing catch scope, if any. If there's no enclosing catch +    // scope, it throws up to the caller. +    if (Opc == WebAssembly::RETHROW || Opc == WebAssembly::RETHROW_S) { +      if (EHPadStack.empty()) { +        printAnnotation(OS, "to caller"); +      } else { +        printAnnotation(OS, "down to catch" + utostr(EHPadStack.back())); +      } + +    } else { +      unsigned NumFixedOperands = Desc.NumOperands; +      SmallSet<uint64_t, 8> Printed; +      for (unsigned I = 0, E = MI->getNumOperands(); I < E; ++I) { +        // See if this operand denotes a basic block target. +        if (I < NumFixedOperands) { +          // A non-variable_ops operand, check its type. +          if (Desc.OpInfo[I].OperandType != WebAssembly::OPERAND_BASIC_BLOCK) +            continue; +        } else { +          // A variable_ops operand, which currently can be immediates (used in +          // br_table) which are basic block targets, or for call instructions +          // when using -wasm-keep-registers (in which case they are registers, +          // and should not be processed). +          if (!MI->getOperand(I).isImm()) +            continue; +        } +        uint64_t Depth = MI->getOperand(I).getImm(); +        if (!Printed.insert(Depth).second) +          continue; +        if (Depth >= ControlFlowStack.size()) { +          printAnnotation(OS, "Invalid depth argument!"); +        } else { +          const auto &Pair = ControlFlowStack.rbegin()[Depth]; +          printAnnotation(OS, utostr(Depth) + ": " + +                                  (Pair.second ? "up" : "down") + " to label" + +                                  utostr(Pair.first)); +        } +      } +    } +  } +} + +static std::string toString(const APFloat &FP) { +  // Print NaNs with custom payloads specially. +  if (FP.isNaN() && !FP.bitwiseIsEqual(APFloat::getQNaN(FP.getSemantics())) && +      !FP.bitwiseIsEqual( +          APFloat::getQNaN(FP.getSemantics(), /*Negative=*/true))) { +    APInt AI = FP.bitcastToAPInt(); +    return std::string(AI.isNegative() ? "-" : "") + "nan:0x" + +           utohexstr(AI.getZExtValue() & +                         (AI.getBitWidth() == 32 ? INT64_C(0x007fffff) +                                                 : INT64_C(0x000fffffffffffff)), +                     /*LowerCase=*/true); +  } + +  // Use C99's hexadecimal floating-point representation. +  static const size_t BufBytes = 128; +  char Buf[BufBytes]; +  auto Written = FP.convertToHexString( +      Buf, /*HexDigits=*/0, /*UpperCase=*/false, APFloat::rmNearestTiesToEven); +  (void)Written; +  assert(Written != 0); +  assert(Written < BufBytes); +  return Buf; +} + +void WebAssemblyInstPrinter::printOperand(const MCInst *MI, unsigned OpNo, +                                          raw_ostream &O) { +  const MCOperand &Op = MI->getOperand(OpNo); +  if (Op.isReg()) { +    unsigned WAReg = Op.getReg(); +    if (int(WAReg) >= 0) +      printRegName(O, WAReg); +    else if (OpNo >= MII.get(MI->getOpcode()).getNumDefs()) +      O << "$pop" << WebAssemblyFunctionInfo::getWARegStackId(WAReg); +    else if (WAReg != WebAssemblyFunctionInfo::UnusedReg) +      O << "$push" << WebAssemblyFunctionInfo::getWARegStackId(WAReg); +    else +      O << "$drop"; +    // Add a '=' suffix if this is a def. +    if (OpNo < MII.get(MI->getOpcode()).getNumDefs()) +      O << '='; +  } else if (Op.isImm()) { +    O << Op.getImm(); +  } else if (Op.isFPImm()) { +    const MCInstrDesc &Desc = MII.get(MI->getOpcode()); +    const MCOperandInfo &Info = Desc.OpInfo[OpNo]; +    if (Info.OperandType == WebAssembly::OPERAND_F32IMM) { +      // TODO: MC converts all floating point immediate operands to double. +      // This is fine for numeric values, but may cause NaNs to change bits. +      O << ::toString(APFloat(float(Op.getFPImm()))); +    } else { +      assert(Info.OperandType == WebAssembly::OPERAND_F64IMM); +      O << ::toString(APFloat(Op.getFPImm())); +    } +  } else { +    assert(Op.isExpr() && "unknown operand kind in printOperand"); +    // call_indirect instructions have a TYPEINDEX operand that we print +    // as a signature here, such that the assembler can recover this +    // information. +    auto SRE = static_cast<const MCSymbolRefExpr *>(Op.getExpr()); +    if (SRE->getKind() == MCSymbolRefExpr::VK_WASM_TYPEINDEX) { +      auto &Sym = static_cast<const MCSymbolWasm &>(SRE->getSymbol()); +      O << WebAssembly::signatureToString(Sym.getSignature()); +    } else { +      Op.getExpr()->print(O, &MAI); +    } +  } +} + +void WebAssemblyInstPrinter::printBrList(const MCInst *MI, unsigned OpNo, +                                         raw_ostream &O) { +  O << "{"; +  for (unsigned I = OpNo, E = MI->getNumOperands(); I != E; ++I) { +    if (I != OpNo) +      O << ", "; +    O << MI->getOperand(I).getImm(); +  } +  O << "}"; +} + +void WebAssemblyInstPrinter::printWebAssemblyP2AlignOperand(const MCInst *MI, +                                                            unsigned OpNo, +                                                            raw_ostream &O) { +  int64_t Imm = MI->getOperand(OpNo).getImm(); +  if (Imm == WebAssembly::GetDefaultP2Align(MI->getOpcode())) +    return; +  O << ":p2align=" << Imm; +} + +void WebAssemblyInstPrinter::printWebAssemblySignatureOperand(const MCInst *MI, +                                                              unsigned OpNo, +                                                              raw_ostream &O) { +  const MCOperand &Op = MI->getOperand(OpNo); +  if (Op.isImm()) { +    auto Imm = static_cast<unsigned>(Op.getImm()); +    if (Imm != wasm::WASM_TYPE_NORESULT) +      O << WebAssembly::anyTypeToString(Imm); +  } else { +    auto Expr = cast<MCSymbolRefExpr>(Op.getExpr()); +    auto *Sym = cast<MCSymbolWasm>(&Expr->getSymbol()); +    if (Sym->getSignature()) { +      O << WebAssembly::signatureToString(Sym->getSignature()); +    } else { +      // Disassembler does not currently produce a signature +      O << "unknown_type"; +    } +  } +} + +// We have various enums representing a subset of these types, use this +// function to convert any of them to text. +const char *WebAssembly::anyTypeToString(unsigned Ty) { +  switch (Ty) { +  case wasm::WASM_TYPE_I32: +    return "i32"; +  case wasm::WASM_TYPE_I64: +    return "i64"; +  case wasm::WASM_TYPE_F32: +    return "f32"; +  case wasm::WASM_TYPE_F64: +    return "f64"; +  case wasm::WASM_TYPE_V128: +    return "v128"; +  case wasm::WASM_TYPE_FUNCREF: +    return "funcref"; +  case wasm::WASM_TYPE_FUNC: +    return "func"; +  case wasm::WASM_TYPE_EXNREF: +    return "exnref"; +  case wasm::WASM_TYPE_NORESULT: +    return "void"; +  default: +    return "invalid_type"; +  } +} + +const char *WebAssembly::typeToString(wasm::ValType Ty) { +  return anyTypeToString(static_cast<unsigned>(Ty)); +} + +std::string WebAssembly::typeListToString(ArrayRef<wasm::ValType> List) { +  std::string S; +  for (auto &Ty : List) { +    if (&Ty != &List[0]) S += ", "; +    S += WebAssembly::typeToString(Ty); +  } +  return S; +} + +std::string WebAssembly::signatureToString(const wasm::WasmSignature *Sig) { +  std::string S("("); +  S += typeListToString(Sig->Params); +  S += ") -> ("; +  S += typeListToString(Sig->Returns); +  S += ")"; +  return S; +} diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyInstPrinter.h b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyInstPrinter.h new file mode 100644 index 000000000000..cf37778099a0 --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyInstPrinter.h @@ -0,0 +1,68 @@ +// WebAssemblyInstPrinter.h - Print wasm MCInst to assembly syntax -*- C++ -*-// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This class prints an WebAssembly MCInst to wasm file syntax. +/// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_LIB_TARGET_WEBASSEMBLY_INSTPRINTER_WEBASSEMBLYINSTPRINTER_H +#define LLVM_LIB_TARGET_WEBASSEMBLY_INSTPRINTER_WEBASSEMBLYINSTPRINTER_H + +#include "llvm/ADT/SmallVector.h" +#include "llvm/BinaryFormat/Wasm.h" +#include "llvm/MC/MCInstPrinter.h" +#include "llvm/Support/MachineValueType.h" + +namespace llvm { + +class MCSubtargetInfo; + +class WebAssemblyInstPrinter final : public MCInstPrinter { +  uint64_t ControlFlowCounter = 0; +  uint64_t EHPadStackCounter = 0; +  SmallVector<std::pair<uint64_t, bool>, 4> ControlFlowStack; +  SmallVector<uint64_t, 4> EHPadStack; + +  enum EHInstKind { TRY, CATCH, END_TRY }; +  EHInstKind LastSeenEHInst = END_TRY; + +public: +  WebAssemblyInstPrinter(const MCAsmInfo &MAI, const MCInstrInfo &MII, +                         const MCRegisterInfo &MRI); + +  void printRegName(raw_ostream &OS, unsigned RegNo) const override; +  void printInst(const MCInst *MI, raw_ostream &OS, StringRef Annot, +                 const MCSubtargetInfo &STI) override; + +  // Used by tblegen code. +  void printOperand(const MCInst *MI, unsigned OpNo, raw_ostream &O); +  void printBrList(const MCInst *MI, unsigned OpNo, raw_ostream &O); +  void printWebAssemblyP2AlignOperand(const MCInst *MI, unsigned OpNo, +                                      raw_ostream &O); +  void printWebAssemblySignatureOperand(const MCInst *MI, unsigned OpNo, +                                        raw_ostream &O); + +  // Autogenerated by tblgen. +  void printInstruction(const MCInst *MI, raw_ostream &O); +  static const char *getRegisterName(unsigned RegNo); +}; + +namespace WebAssembly { + +const char *typeToString(wasm::ValType Ty); +const char *anyTypeToString(unsigned Ty); + +std::string typeListToString(ArrayRef<wasm::ValType> List); +std::string signatureToString(const wasm::WasmSignature *Sig); + +} // end namespace WebAssembly + +} // end namespace llvm + +#endif diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCAsmInfo.cpp b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCAsmInfo.cpp new file mode 100644 index 000000000000..8f6531563e1b --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCAsmInfo.cpp @@ -0,0 +1,47 @@ +//===-- WebAssemblyMCAsmInfo.cpp - WebAssembly asm properties -------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file contains the declarations of the WebAssemblyMCAsmInfo +/// properties. +/// +//===----------------------------------------------------------------------===// + +#include "WebAssemblyMCAsmInfo.h" +#include "llvm/ADT/Triple.h" + +using namespace llvm; + +#define DEBUG_TYPE "wasm-mc-asm-info" + +WebAssemblyMCAsmInfo::~WebAssemblyMCAsmInfo() = default; // anchor. + +WebAssemblyMCAsmInfo::WebAssemblyMCAsmInfo(const Triple &T) { +  CodePointerSize = CalleeSaveStackSlotSize = T.isArch64Bit() ? 8 : 4; + +  // TODO: What should MaxInstLength be? + +  UseDataRegionDirectives = true; + +  // Use .skip instead of .zero because .zero is confusing when used with two +  // arguments (it doesn't actually zero things out). +  ZeroDirective = "\t.skip\t"; + +  Data8bitsDirective = "\t.int8\t"; +  Data16bitsDirective = "\t.int16\t"; +  Data32bitsDirective = "\t.int32\t"; +  Data64bitsDirective = "\t.int64\t"; + +  AlignmentIsInBytes = false; +  COMMDirectiveAlignmentIsInBytes = false; +  LCOMMDirectiveAlignmentType = LCOMM::Log2Alignment; + +  SupportsDebugInformation = true; + +  // TODO: UseIntegratedAssembler? +} diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCAsmInfo.h b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCAsmInfo.h new file mode 100644 index 000000000000..9efbbf881f59 --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCAsmInfo.h @@ -0,0 +1,31 @@ +//===-- WebAssemblyMCAsmInfo.h - WebAssembly asm properties -----*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file contains the declaration of the WebAssemblyMCAsmInfo class. +/// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYMCASMINFO_H +#define LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYMCASMINFO_H + +#include "llvm/MC/MCAsmInfoWasm.h" + +namespace llvm { + +class Triple; + +class WebAssemblyMCAsmInfo final : public MCAsmInfoWasm { +public: +  explicit WebAssemblyMCAsmInfo(const Triple &T); +  ~WebAssemblyMCAsmInfo() override; +}; + +} // end namespace llvm + +#endif diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCCodeEmitter.cpp b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCCodeEmitter.cpp new file mode 100644 index 000000000000..1a4c57e66d2f --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCCodeEmitter.cpp @@ -0,0 +1,176 @@ +//=- WebAssemblyMCCodeEmitter.cpp - Convert WebAssembly code to machine code -// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file implements the WebAssemblyMCCodeEmitter class. +/// +//===----------------------------------------------------------------------===// + +#include "MCTargetDesc/WebAssemblyFixupKinds.h" +#include "MCTargetDesc/WebAssemblyMCTargetDesc.h" +#include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/Statistic.h" +#include "llvm/MC/MCCodeEmitter.h" +#include "llvm/MC/MCFixup.h" +#include "llvm/MC/MCInst.h" +#include "llvm/MC/MCInstrInfo.h" +#include "llvm/MC/MCRegisterInfo.h" +#include "llvm/MC/MCSubtargetInfo.h" +#include "llvm/MC/MCSymbol.h" +#include "llvm/Support/Debug.h" +#include "llvm/Support/EndianStream.h" +#include "llvm/Support/LEB128.h" +#include "llvm/Support/raw_ostream.h" + +using namespace llvm; + +#define DEBUG_TYPE "mccodeemitter" + +STATISTIC(MCNumEmitted, "Number of MC instructions emitted."); +STATISTIC(MCNumFixups, "Number of MC fixups created."); + +namespace { +class WebAssemblyMCCodeEmitter final : public MCCodeEmitter { +  const MCInstrInfo &MCII; + +  // Implementation generated by tablegen. +  uint64_t getBinaryCodeForInstr(const MCInst &MI, +                                 SmallVectorImpl<MCFixup> &Fixups, +                                 const MCSubtargetInfo &STI) const; + +  void encodeInstruction(const MCInst &MI, raw_ostream &OS, +                         SmallVectorImpl<MCFixup> &Fixups, +                         const MCSubtargetInfo &STI) const override; + +public: +  WebAssemblyMCCodeEmitter(const MCInstrInfo &MCII) : MCII(MCII) {} +}; +} // end anonymous namespace + +MCCodeEmitter *llvm::createWebAssemblyMCCodeEmitter(const MCInstrInfo &MCII) { +  return new WebAssemblyMCCodeEmitter(MCII); +} + +void WebAssemblyMCCodeEmitter::encodeInstruction( +    const MCInst &MI, raw_ostream &OS, SmallVectorImpl<MCFixup> &Fixups, +    const MCSubtargetInfo &STI) const { +  uint64_t Start = OS.tell(); + +  uint64_t Binary = getBinaryCodeForInstr(MI, Fixups, STI); +  if (Binary <= UINT8_MAX) { +    OS << uint8_t(Binary); +  } else { +    assert(Binary <= UINT16_MAX && "Several-byte opcodes not supported yet"); +    OS << uint8_t(Binary >> 8); +    encodeULEB128(uint8_t(Binary), OS); +  } + +  // For br_table instructions, encode the size of the table. In the MCInst, +  // there's an index operand (if not a stack instruction), one operand for +  // each table entry, and the default operand. +  if (MI.getOpcode() == WebAssembly::BR_TABLE_I32_S || +      MI.getOpcode() == WebAssembly::BR_TABLE_I64_S) +    encodeULEB128(MI.getNumOperands() - 1, OS); +  if (MI.getOpcode() == WebAssembly::BR_TABLE_I32 || +      MI.getOpcode() == WebAssembly::BR_TABLE_I64) +    encodeULEB128(MI.getNumOperands() - 2, OS); + +  const MCInstrDesc &Desc = MCII.get(MI.getOpcode()); +  for (unsigned I = 0, E = MI.getNumOperands(); I < E; ++I) { +    const MCOperand &MO = MI.getOperand(I); +    if (MO.isReg()) { +      /* nothing to encode */ + +    } else if (MO.isImm()) { +      if (I < Desc.getNumOperands()) { +        const MCOperandInfo &Info = Desc.OpInfo[I]; +        LLVM_DEBUG(dbgs() << "Encoding immediate: type=" +                          << int(Info.OperandType) << "\n"); +        switch (Info.OperandType) { +        case WebAssembly::OPERAND_I32IMM: +          encodeSLEB128(int32_t(MO.getImm()), OS); +          break; +        case WebAssembly::OPERAND_OFFSET32: +          encodeULEB128(uint32_t(MO.getImm()), OS); +          break; +        case WebAssembly::OPERAND_I64IMM: +          encodeSLEB128(int64_t(MO.getImm()), OS); +          break; +        case WebAssembly::OPERAND_SIGNATURE: +          OS << uint8_t(MO.getImm()); +          break; +        case WebAssembly::OPERAND_VEC_I8IMM: +          support::endian::write<uint8_t>(OS, MO.getImm(), support::little); +          break; +        case WebAssembly::OPERAND_VEC_I16IMM: +          support::endian::write<uint16_t>(OS, MO.getImm(), support::little); +          break; +        case WebAssembly::OPERAND_VEC_I32IMM: +          support::endian::write<uint32_t>(OS, MO.getImm(), support::little); +          break; +        case WebAssembly::OPERAND_VEC_I64IMM: +          support::endian::write<uint64_t>(OS, MO.getImm(), support::little); +          break; +        case WebAssembly::OPERAND_GLOBAL: +          llvm_unreachable("wasm globals should only be accessed symbolicly"); +        default: +          encodeULEB128(uint64_t(MO.getImm()), OS); +        } +      } else { +        encodeULEB128(uint64_t(MO.getImm()), OS); +      } + +    } else if (MO.isFPImm()) { +      const MCOperandInfo &Info = Desc.OpInfo[I]; +      if (Info.OperandType == WebAssembly::OPERAND_F32IMM) { +        // TODO: MC converts all floating point immediate operands to double. +        // This is fine for numeric values, but may cause NaNs to change bits. +        auto F = float(MO.getFPImm()); +        support::endian::write<float>(OS, F, support::little); +      } else { +        assert(Info.OperandType == WebAssembly::OPERAND_F64IMM); +        double D = MO.getFPImm(); +        support::endian::write<double>(OS, D, support::little); +      } + +    } else if (MO.isExpr()) { +      const MCOperandInfo &Info = Desc.OpInfo[I]; +      llvm::MCFixupKind FixupKind; +      size_t PaddedSize = 5; +      switch (Info.OperandType) { +      case WebAssembly::OPERAND_I32IMM: +        FixupKind = MCFixupKind(WebAssembly::fixup_sleb128_i32); +        break; +      case WebAssembly::OPERAND_I64IMM: +        FixupKind = MCFixupKind(WebAssembly::fixup_sleb128_i64); +        PaddedSize = 10; +        break; +      case WebAssembly::OPERAND_FUNCTION32: +      case WebAssembly::OPERAND_OFFSET32: +      case WebAssembly::OPERAND_SIGNATURE: +      case WebAssembly::OPERAND_TYPEINDEX: +      case WebAssembly::OPERAND_GLOBAL: +      case WebAssembly::OPERAND_EVENT: +        FixupKind = MCFixupKind(WebAssembly::fixup_uleb128_i32); +        break; +      default: +        llvm_unreachable("unexpected symbolic operand kind"); +      } +      Fixups.push_back(MCFixup::create(OS.tell() - Start, MO.getExpr(), +                                       FixupKind, MI.getLoc())); +      ++MCNumFixups; +      encodeULEB128(0, OS, PaddedSize); +    } else { +      llvm_unreachable("unexpected operand kind"); +    } +  } + +  ++MCNumEmitted; // Keep track of the # of mi's emitted. +} + +#include "WebAssemblyGenMCCodeEmitter.inc" diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCTargetDesc.cpp b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCTargetDesc.cpp new file mode 100644 index 000000000000..9c8ca1f13b18 --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCTargetDesc.cpp @@ -0,0 +1,154 @@ +//===-- WebAssemblyMCTargetDesc.cpp - WebAssembly Target Descriptions -----===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file provides WebAssembly-specific target descriptions. +/// +//===----------------------------------------------------------------------===// + +#include "MCTargetDesc/WebAssemblyMCTargetDesc.h" +#include "MCTargetDesc/WebAssemblyInstPrinter.h" +#include "MCTargetDesc/WebAssemblyMCAsmInfo.h" +#include "MCTargetDesc/WebAssemblyTargetStreamer.h" +#include "TargetInfo/WebAssemblyTargetInfo.h" +#include "llvm/MC/MCInstrInfo.h" +#include "llvm/MC/MCRegisterInfo.h" +#include "llvm/MC/MCSubtargetInfo.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/TargetRegistry.h" +using namespace llvm; + +#define DEBUG_TYPE "wasm-mc-target-desc" + +#define GET_INSTRINFO_MC_DESC +#include "WebAssemblyGenInstrInfo.inc" + +#define GET_SUBTARGETINFO_MC_DESC +#include "WebAssemblyGenSubtargetInfo.inc" + +#define GET_REGINFO_MC_DESC +#include "WebAssemblyGenRegisterInfo.inc" + +static MCAsmInfo *createMCAsmInfo(const MCRegisterInfo & /*MRI*/, +                                  const Triple &TT) { +  return new WebAssemblyMCAsmInfo(TT); +} + +static MCInstrInfo *createMCInstrInfo() { +  auto *X = new MCInstrInfo(); +  InitWebAssemblyMCInstrInfo(X); +  return X; +} + +static MCRegisterInfo *createMCRegisterInfo(const Triple & /*T*/) { +  auto *X = new MCRegisterInfo(); +  InitWebAssemblyMCRegisterInfo(X, 0); +  return X; +} + +static MCInstPrinter *createMCInstPrinter(const Triple & /*T*/, +                                          unsigned SyntaxVariant, +                                          const MCAsmInfo &MAI, +                                          const MCInstrInfo &MII, +                                          const MCRegisterInfo &MRI) { +  assert(SyntaxVariant == 0 && "WebAssembly only has one syntax variant"); +  return new WebAssemblyInstPrinter(MAI, MII, MRI); +} + +static MCCodeEmitter *createCodeEmitter(const MCInstrInfo &MCII, +                                        const MCRegisterInfo & /*MRI*/, +                                        MCContext &Ctx) { +  return createWebAssemblyMCCodeEmitter(MCII); +} + +static MCAsmBackend *createAsmBackend(const Target & /*T*/, +                                      const MCSubtargetInfo &STI, +                                      const MCRegisterInfo & /*MRI*/, +                                      const MCTargetOptions & /*Options*/) { +  return createWebAssemblyAsmBackend(STI.getTargetTriple()); +} + +static MCSubtargetInfo *createMCSubtargetInfo(const Triple &TT, StringRef CPU, +                                              StringRef FS) { +  return createWebAssemblyMCSubtargetInfoImpl(TT, CPU, FS); +} + +static MCTargetStreamer * +createObjectTargetStreamer(MCStreamer &S, const MCSubtargetInfo &STI) { +  return new WebAssemblyTargetWasmStreamer(S); +} + +static MCTargetStreamer *createAsmTargetStreamer(MCStreamer &S, +                                                 formatted_raw_ostream &OS, +                                                 MCInstPrinter * /*InstPrint*/, +                                                 bool /*isVerboseAsm*/) { +  return new WebAssemblyTargetAsmStreamer(S, OS); +} + +static MCTargetStreamer *createNullTargetStreamer(MCStreamer &S) { +  return new WebAssemblyTargetNullStreamer(S); +} + +// Force static initialization. +extern "C" void LLVMInitializeWebAssemblyTargetMC() { +  for (Target *T : +       {&getTheWebAssemblyTarget32(), &getTheWebAssemblyTarget64()}) { +    // Register the MC asm info. +    RegisterMCAsmInfoFn X(*T, createMCAsmInfo); + +    // Register the MC instruction info. +    TargetRegistry::RegisterMCInstrInfo(*T, createMCInstrInfo); + +    // Register the MC register info. +    TargetRegistry::RegisterMCRegInfo(*T, createMCRegisterInfo); + +    // Register the MCInstPrinter. +    TargetRegistry::RegisterMCInstPrinter(*T, createMCInstPrinter); + +    // Register the MC code emitter. +    TargetRegistry::RegisterMCCodeEmitter(*T, createCodeEmitter); + +    // Register the ASM Backend. +    TargetRegistry::RegisterMCAsmBackend(*T, createAsmBackend); + +    // Register the MC subtarget info. +    TargetRegistry::RegisterMCSubtargetInfo(*T, createMCSubtargetInfo); + +    // Register the object target streamer. +    TargetRegistry::RegisterObjectTargetStreamer(*T, +                                                 createObjectTargetStreamer); +    // Register the asm target streamer. +    TargetRegistry::RegisterAsmTargetStreamer(*T, createAsmTargetStreamer); +    // Register the null target streamer. +    TargetRegistry::RegisterNullTargetStreamer(*T, createNullTargetStreamer); +  } +} + +wasm::ValType WebAssembly::toValType(const MVT &Ty) { +  switch (Ty.SimpleTy) { +  case MVT::i32: +    return wasm::ValType::I32; +  case MVT::i64: +    return wasm::ValType::I64; +  case MVT::f32: +    return wasm::ValType::F32; +  case MVT::f64: +    return wasm::ValType::F64; +  case MVT::v16i8: +  case MVT::v8i16: +  case MVT::v4i32: +  case MVT::v2i64: +  case MVT::v4f32: +  case MVT::v2f64: +    return wasm::ValType::V128; +  case MVT::exnref: +    return wasm::ValType::EXNREF; +  default: +    llvm_unreachable("unexpected type"); +  } +} diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCTargetDesc.h b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCTargetDesc.h new file mode 100644 index 000000000000..b339860a381d --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyMCTargetDesc.h @@ -0,0 +1,596 @@ +//==- WebAssemblyMCTargetDesc.h - WebAssembly Target Descriptions -*- C++ -*-=// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file provides WebAssembly-specific target descriptions. +/// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYMCTARGETDESC_H +#define LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYMCTARGETDESC_H + +#include "../WebAssemblySubtarget.h" +#include "llvm/BinaryFormat/Wasm.h" +#include "llvm/MC/MCInstrDesc.h" +#include "llvm/Support/DataTypes.h" +#include <memory> + +namespace llvm { + +class MCAsmBackend; +class MCCodeEmitter; +class MCContext; +class MCInstrInfo; +class MCObjectTargetWriter; +class MCSubtargetInfo; +class MVT; +class Target; +class Triple; +class raw_pwrite_stream; + +MCCodeEmitter *createWebAssemblyMCCodeEmitter(const MCInstrInfo &MCII); + +MCAsmBackend *createWebAssemblyAsmBackend(const Triple &TT); + +std::unique_ptr<MCObjectTargetWriter> +createWebAssemblyWasmObjectWriter(bool Is64Bit, bool IsEmscripten); + +namespace WebAssembly { +enum OperandType { +  /// Basic block label in a branch construct. +  OPERAND_BASIC_BLOCK = MCOI::OPERAND_FIRST_TARGET, +  /// Local index. +  OPERAND_LOCAL, +  /// Global index. +  OPERAND_GLOBAL, +  /// 32-bit integer immediates. +  OPERAND_I32IMM, +  /// 64-bit integer immediates. +  OPERAND_I64IMM, +  /// 32-bit floating-point immediates. +  OPERAND_F32IMM, +  /// 64-bit floating-point immediates. +  OPERAND_F64IMM, +  /// 8-bit vector lane immediate +  OPERAND_VEC_I8IMM, +  /// 16-bit vector lane immediate +  OPERAND_VEC_I16IMM, +  /// 32-bit vector lane immediate +  OPERAND_VEC_I32IMM, +  /// 64-bit vector lane immediate +  OPERAND_VEC_I64IMM, +  /// 32-bit unsigned function indices. +  OPERAND_FUNCTION32, +  /// 32-bit unsigned memory offsets. +  OPERAND_OFFSET32, +  /// p2align immediate for load and store address alignment. +  OPERAND_P2ALIGN, +  /// signature immediate for block/loop. +  OPERAND_SIGNATURE, +  /// type signature immediate for call_indirect. +  OPERAND_TYPEINDEX, +  /// Event index. +  OPERAND_EVENT, +  /// A list of branch targets for br_list. +  OPERAND_BRLIST, +}; +} // end namespace WebAssembly + +namespace WebAssemblyII { + +/// Target Operand Flag enum. +enum TOF { +  MO_NO_FLAG = 0, + +  // On a symbol operand this indicates that the immediate is a wasm global +  // index.  The value of the wasm global will be set to the symbol address at +  // runtime.  This adds a level of indirection similar to the GOT on native +  // platforms. +  MO_GOT, + +  // On a symbol operand this indicates that the immediate is the symbol +  // address relative the __memory_base wasm global. +  // Only applicable to data symbols. +  MO_MEMORY_BASE_REL, + +  // On a symbol operand this indicates that the immediate is the symbol +  // address relative the __table_base wasm global. +  // Only applicable to function symbols. +  MO_TABLE_BASE_REL, +}; + +} // end namespace WebAssemblyII + +} // end namespace llvm + +// Defines symbolic names for WebAssembly registers. This defines a mapping from +// register name to register number. +// +#define GET_REGINFO_ENUM +#include "WebAssemblyGenRegisterInfo.inc" + +// Defines symbolic names for the WebAssembly instructions. +// +#define GET_INSTRINFO_ENUM +#include "WebAssemblyGenInstrInfo.inc" + +namespace llvm { +namespace WebAssembly { + +/// Used as immediate MachineOperands for block signatures +enum class BlockType : unsigned { +  Invalid = 0x00, +  Void = 0x40, +  I32 = unsigned(wasm::ValType::I32), +  I64 = unsigned(wasm::ValType::I64), +  F32 = unsigned(wasm::ValType::F32), +  F64 = unsigned(wasm::ValType::F64), +  V128 = unsigned(wasm::ValType::V128), +  Exnref = unsigned(wasm::ValType::EXNREF), +  // Multivalue blocks (and other non-void blocks) are only emitted when the +  // blocks will never be exited and are at the ends of functions (see +  // WebAssemblyCFGStackify::fixEndsAtEndOfFunction). They also are never made +  // to pop values off the stack, so the exact multivalue signature can always +  // be inferred from the return type of the parent function in MCInstLower. +  Multivalue = 0xffff, +}; + +/// Instruction opcodes emitted via means other than CodeGen. +static const unsigned Nop = 0x01; +static const unsigned End = 0x0b; + +wasm::ValType toValType(const MVT &Ty); + +/// Return the default p2align value for a load or store with the given opcode. +inline unsigned GetDefaultP2AlignAny(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::LOAD8_S_I32: +  case WebAssembly::LOAD8_S_I32_S: +  case WebAssembly::LOAD8_U_I32: +  case WebAssembly::LOAD8_U_I32_S: +  case WebAssembly::LOAD8_S_I64: +  case WebAssembly::LOAD8_S_I64_S: +  case WebAssembly::LOAD8_U_I64: +  case WebAssembly::LOAD8_U_I64_S: +  case WebAssembly::ATOMIC_LOAD8_U_I32: +  case WebAssembly::ATOMIC_LOAD8_U_I32_S: +  case WebAssembly::ATOMIC_LOAD8_U_I64: +  case WebAssembly::ATOMIC_LOAD8_U_I64_S: +  case WebAssembly::STORE8_I32: +  case WebAssembly::STORE8_I32_S: +  case WebAssembly::STORE8_I64: +  case WebAssembly::STORE8_I64_S: +  case WebAssembly::ATOMIC_STORE8_I32: +  case WebAssembly::ATOMIC_STORE8_I32_S: +  case WebAssembly::ATOMIC_STORE8_I64: +  case WebAssembly::ATOMIC_STORE8_I64_S: +  case WebAssembly::ATOMIC_RMW8_U_ADD_I32: +  case WebAssembly::ATOMIC_RMW8_U_ADD_I32_S: +  case WebAssembly::ATOMIC_RMW8_U_ADD_I64: +  case WebAssembly::ATOMIC_RMW8_U_ADD_I64_S: +  case WebAssembly::ATOMIC_RMW8_U_SUB_I32: +  case WebAssembly::ATOMIC_RMW8_U_SUB_I32_S: +  case WebAssembly::ATOMIC_RMW8_U_SUB_I64: +  case WebAssembly::ATOMIC_RMW8_U_SUB_I64_S: +  case WebAssembly::ATOMIC_RMW8_U_AND_I32: +  case WebAssembly::ATOMIC_RMW8_U_AND_I32_S: +  case WebAssembly::ATOMIC_RMW8_U_AND_I64: +  case WebAssembly::ATOMIC_RMW8_U_AND_I64_S: +  case WebAssembly::ATOMIC_RMW8_U_OR_I32: +  case WebAssembly::ATOMIC_RMW8_U_OR_I32_S: +  case WebAssembly::ATOMIC_RMW8_U_OR_I64: +  case WebAssembly::ATOMIC_RMW8_U_OR_I64_S: +  case WebAssembly::ATOMIC_RMW8_U_XOR_I32: +  case WebAssembly::ATOMIC_RMW8_U_XOR_I32_S: +  case WebAssembly::ATOMIC_RMW8_U_XOR_I64: +  case WebAssembly::ATOMIC_RMW8_U_XOR_I64_S: +  case WebAssembly::ATOMIC_RMW8_U_XCHG_I32: +  case WebAssembly::ATOMIC_RMW8_U_XCHG_I32_S: +  case WebAssembly::ATOMIC_RMW8_U_XCHG_I64: +  case WebAssembly::ATOMIC_RMW8_U_XCHG_I64_S: +  case WebAssembly::ATOMIC_RMW8_U_CMPXCHG_I32: +  case WebAssembly::ATOMIC_RMW8_U_CMPXCHG_I32_S: +  case WebAssembly::ATOMIC_RMW8_U_CMPXCHG_I64: +  case WebAssembly::ATOMIC_RMW8_U_CMPXCHG_I64_S: +  case WebAssembly::LOAD_SPLAT_v8x16: +  case WebAssembly::LOAD_SPLAT_v8x16_S: +    return 0; +  case WebAssembly::LOAD16_S_I32: +  case WebAssembly::LOAD16_S_I32_S: +  case WebAssembly::LOAD16_U_I32: +  case WebAssembly::LOAD16_U_I32_S: +  case WebAssembly::LOAD16_S_I64: +  case WebAssembly::LOAD16_S_I64_S: +  case WebAssembly::LOAD16_U_I64: +  case WebAssembly::LOAD16_U_I64_S: +  case WebAssembly::ATOMIC_LOAD16_U_I32: +  case WebAssembly::ATOMIC_LOAD16_U_I32_S: +  case WebAssembly::ATOMIC_LOAD16_U_I64: +  case WebAssembly::ATOMIC_LOAD16_U_I64_S: +  case WebAssembly::STORE16_I32: +  case WebAssembly::STORE16_I32_S: +  case WebAssembly::STORE16_I64: +  case WebAssembly::STORE16_I64_S: +  case WebAssembly::ATOMIC_STORE16_I32: +  case WebAssembly::ATOMIC_STORE16_I32_S: +  case WebAssembly::ATOMIC_STORE16_I64: +  case WebAssembly::ATOMIC_STORE16_I64_S: +  case WebAssembly::ATOMIC_RMW16_U_ADD_I32: +  case WebAssembly::ATOMIC_RMW16_U_ADD_I32_S: +  case WebAssembly::ATOMIC_RMW16_U_ADD_I64: +  case WebAssembly::ATOMIC_RMW16_U_ADD_I64_S: +  case WebAssembly::ATOMIC_RMW16_U_SUB_I32: +  case WebAssembly::ATOMIC_RMW16_U_SUB_I32_S: +  case WebAssembly::ATOMIC_RMW16_U_SUB_I64: +  case WebAssembly::ATOMIC_RMW16_U_SUB_I64_S: +  case WebAssembly::ATOMIC_RMW16_U_AND_I32: +  case WebAssembly::ATOMIC_RMW16_U_AND_I32_S: +  case WebAssembly::ATOMIC_RMW16_U_AND_I64: +  case WebAssembly::ATOMIC_RMW16_U_AND_I64_S: +  case WebAssembly::ATOMIC_RMW16_U_OR_I32: +  case WebAssembly::ATOMIC_RMW16_U_OR_I32_S: +  case WebAssembly::ATOMIC_RMW16_U_OR_I64: +  case WebAssembly::ATOMIC_RMW16_U_OR_I64_S: +  case WebAssembly::ATOMIC_RMW16_U_XOR_I32: +  case WebAssembly::ATOMIC_RMW16_U_XOR_I32_S: +  case WebAssembly::ATOMIC_RMW16_U_XOR_I64: +  case WebAssembly::ATOMIC_RMW16_U_XOR_I64_S: +  case WebAssembly::ATOMIC_RMW16_U_XCHG_I32: +  case WebAssembly::ATOMIC_RMW16_U_XCHG_I32_S: +  case WebAssembly::ATOMIC_RMW16_U_XCHG_I64: +  case WebAssembly::ATOMIC_RMW16_U_XCHG_I64_S: +  case WebAssembly::ATOMIC_RMW16_U_CMPXCHG_I32: +  case WebAssembly::ATOMIC_RMW16_U_CMPXCHG_I32_S: +  case WebAssembly::ATOMIC_RMW16_U_CMPXCHG_I64: +  case WebAssembly::ATOMIC_RMW16_U_CMPXCHG_I64_S: +  case WebAssembly::LOAD_SPLAT_v16x8: +  case WebAssembly::LOAD_SPLAT_v16x8_S: +    return 1; +  case WebAssembly::LOAD_I32: +  case WebAssembly::LOAD_I32_S: +  case WebAssembly::LOAD_F32: +  case WebAssembly::LOAD_F32_S: +  case WebAssembly::STORE_I32: +  case WebAssembly::STORE_I32_S: +  case WebAssembly::STORE_F32: +  case WebAssembly::STORE_F32_S: +  case WebAssembly::LOAD32_S_I64: +  case WebAssembly::LOAD32_S_I64_S: +  case WebAssembly::LOAD32_U_I64: +  case WebAssembly::LOAD32_U_I64_S: +  case WebAssembly::STORE32_I64: +  case WebAssembly::STORE32_I64_S: +  case WebAssembly::ATOMIC_LOAD_I32: +  case WebAssembly::ATOMIC_LOAD_I32_S: +  case WebAssembly::ATOMIC_LOAD32_U_I64: +  case WebAssembly::ATOMIC_LOAD32_U_I64_S: +  case WebAssembly::ATOMIC_STORE_I32: +  case WebAssembly::ATOMIC_STORE_I32_S: +  case WebAssembly::ATOMIC_STORE32_I64: +  case WebAssembly::ATOMIC_STORE32_I64_S: +  case WebAssembly::ATOMIC_RMW_ADD_I32: +  case WebAssembly::ATOMIC_RMW_ADD_I32_S: +  case WebAssembly::ATOMIC_RMW32_U_ADD_I64: +  case WebAssembly::ATOMIC_RMW32_U_ADD_I64_S: +  case WebAssembly::ATOMIC_RMW_SUB_I32: +  case WebAssembly::ATOMIC_RMW_SUB_I32_S: +  case WebAssembly::ATOMIC_RMW32_U_SUB_I64: +  case WebAssembly::ATOMIC_RMW32_U_SUB_I64_S: +  case WebAssembly::ATOMIC_RMW_AND_I32: +  case WebAssembly::ATOMIC_RMW_AND_I32_S: +  case WebAssembly::ATOMIC_RMW32_U_AND_I64: +  case WebAssembly::ATOMIC_RMW32_U_AND_I64_S: +  case WebAssembly::ATOMIC_RMW_OR_I32: +  case WebAssembly::ATOMIC_RMW_OR_I32_S: +  case WebAssembly::ATOMIC_RMW32_U_OR_I64: +  case WebAssembly::ATOMIC_RMW32_U_OR_I64_S: +  case WebAssembly::ATOMIC_RMW_XOR_I32: +  case WebAssembly::ATOMIC_RMW_XOR_I32_S: +  case WebAssembly::ATOMIC_RMW32_U_XOR_I64: +  case WebAssembly::ATOMIC_RMW32_U_XOR_I64_S: +  case WebAssembly::ATOMIC_RMW_XCHG_I32: +  case WebAssembly::ATOMIC_RMW_XCHG_I32_S: +  case WebAssembly::ATOMIC_RMW32_U_XCHG_I64: +  case WebAssembly::ATOMIC_RMW32_U_XCHG_I64_S: +  case WebAssembly::ATOMIC_RMW_CMPXCHG_I32: +  case WebAssembly::ATOMIC_RMW_CMPXCHG_I32_S: +  case WebAssembly::ATOMIC_RMW32_U_CMPXCHG_I64: +  case WebAssembly::ATOMIC_RMW32_U_CMPXCHG_I64_S: +  case WebAssembly::ATOMIC_NOTIFY: +  case WebAssembly::ATOMIC_NOTIFY_S: +  case WebAssembly::ATOMIC_WAIT_I32: +  case WebAssembly::ATOMIC_WAIT_I32_S: +  case WebAssembly::LOAD_SPLAT_v32x4: +  case WebAssembly::LOAD_SPLAT_v32x4_S: +    return 2; +  case WebAssembly::LOAD_I64: +  case WebAssembly::LOAD_I64_S: +  case WebAssembly::LOAD_F64: +  case WebAssembly::LOAD_F64_S: +  case WebAssembly::STORE_I64: +  case WebAssembly::STORE_I64_S: +  case WebAssembly::STORE_F64: +  case WebAssembly::STORE_F64_S: +  case WebAssembly::ATOMIC_LOAD_I64: +  case WebAssembly::ATOMIC_LOAD_I64_S: +  case WebAssembly::ATOMIC_STORE_I64: +  case WebAssembly::ATOMIC_STORE_I64_S: +  case WebAssembly::ATOMIC_RMW_ADD_I64: +  case WebAssembly::ATOMIC_RMW_ADD_I64_S: +  case WebAssembly::ATOMIC_RMW_SUB_I64: +  case WebAssembly::ATOMIC_RMW_SUB_I64_S: +  case WebAssembly::ATOMIC_RMW_AND_I64: +  case WebAssembly::ATOMIC_RMW_AND_I64_S: +  case WebAssembly::ATOMIC_RMW_OR_I64: +  case WebAssembly::ATOMIC_RMW_OR_I64_S: +  case WebAssembly::ATOMIC_RMW_XOR_I64: +  case WebAssembly::ATOMIC_RMW_XOR_I64_S: +  case WebAssembly::ATOMIC_RMW_XCHG_I64: +  case WebAssembly::ATOMIC_RMW_XCHG_I64_S: +  case WebAssembly::ATOMIC_RMW_CMPXCHG_I64: +  case WebAssembly::ATOMIC_RMW_CMPXCHG_I64_S: +  case WebAssembly::ATOMIC_WAIT_I64: +  case WebAssembly::ATOMIC_WAIT_I64_S: +  case WebAssembly::LOAD_SPLAT_v64x2: +  case WebAssembly::LOAD_SPLAT_v64x2_S: +  case WebAssembly::LOAD_EXTEND_S_v8i16: +  case WebAssembly::LOAD_EXTEND_S_v8i16_S: +  case WebAssembly::LOAD_EXTEND_U_v8i16: +  case WebAssembly::LOAD_EXTEND_U_v8i16_S: +  case WebAssembly::LOAD_EXTEND_S_v4i32: +  case WebAssembly::LOAD_EXTEND_S_v4i32_S: +  case WebAssembly::LOAD_EXTEND_U_v4i32: +  case WebAssembly::LOAD_EXTEND_U_v4i32_S: +  case WebAssembly::LOAD_EXTEND_S_v2i64: +  case WebAssembly::LOAD_EXTEND_S_v2i64_S: +  case WebAssembly::LOAD_EXTEND_U_v2i64: +  case WebAssembly::LOAD_EXTEND_U_v2i64_S: +    return 3; +  case WebAssembly::LOAD_V128: +  case WebAssembly::LOAD_V128_S: +  case WebAssembly::STORE_V128: +  case WebAssembly::STORE_V128_S: +    return 4; +  default: +    return -1; +  } +} + +inline unsigned GetDefaultP2Align(unsigned Opc) { +  auto Align = GetDefaultP2AlignAny(Opc); +  if (Align == -1U) { +    llvm_unreachable("Only loads and stores have p2align values"); +  } +  return Align; +} + +inline bool isArgument(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::ARGUMENT_i32: +  case WebAssembly::ARGUMENT_i32_S: +  case WebAssembly::ARGUMENT_i64: +  case WebAssembly::ARGUMENT_i64_S: +  case WebAssembly::ARGUMENT_f32: +  case WebAssembly::ARGUMENT_f32_S: +  case WebAssembly::ARGUMENT_f64: +  case WebAssembly::ARGUMENT_f64_S: +  case WebAssembly::ARGUMENT_v16i8: +  case WebAssembly::ARGUMENT_v16i8_S: +  case WebAssembly::ARGUMENT_v8i16: +  case WebAssembly::ARGUMENT_v8i16_S: +  case WebAssembly::ARGUMENT_v4i32: +  case WebAssembly::ARGUMENT_v4i32_S: +  case WebAssembly::ARGUMENT_v2i64: +  case WebAssembly::ARGUMENT_v2i64_S: +  case WebAssembly::ARGUMENT_v4f32: +  case WebAssembly::ARGUMENT_v4f32_S: +  case WebAssembly::ARGUMENT_v2f64: +  case WebAssembly::ARGUMENT_v2f64_S: +  case WebAssembly::ARGUMENT_exnref: +  case WebAssembly::ARGUMENT_exnref_S: +    return true; +  default: +    return false; +  } +} + +inline bool isCopy(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::COPY_I32: +  case WebAssembly::COPY_I32_S: +  case WebAssembly::COPY_I64: +  case WebAssembly::COPY_I64_S: +  case WebAssembly::COPY_F32: +  case WebAssembly::COPY_F32_S: +  case WebAssembly::COPY_F64: +  case WebAssembly::COPY_F64_S: +  case WebAssembly::COPY_V128: +  case WebAssembly::COPY_V128_S: +  case WebAssembly::COPY_EXNREF: +  case WebAssembly::COPY_EXNREF_S: +    return true; +  default: +    return false; +  } +} + +inline bool isTee(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::TEE_I32: +  case WebAssembly::TEE_I32_S: +  case WebAssembly::TEE_I64: +  case WebAssembly::TEE_I64_S: +  case WebAssembly::TEE_F32: +  case WebAssembly::TEE_F32_S: +  case WebAssembly::TEE_F64: +  case WebAssembly::TEE_F64_S: +  case WebAssembly::TEE_V128: +  case WebAssembly::TEE_V128_S: +  case WebAssembly::TEE_EXNREF: +  case WebAssembly::TEE_EXNREF_S: +    return true; +  default: +    return false; +  } +} + +inline bool isCallDirect(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::CALL_VOID: +  case WebAssembly::CALL_VOID_S: +  case WebAssembly::CALL_i32: +  case WebAssembly::CALL_i32_S: +  case WebAssembly::CALL_i64: +  case WebAssembly::CALL_i64_S: +  case WebAssembly::CALL_f32: +  case WebAssembly::CALL_f32_S: +  case WebAssembly::CALL_f64: +  case WebAssembly::CALL_f64_S: +  case WebAssembly::CALL_v16i8: +  case WebAssembly::CALL_v16i8_S: +  case WebAssembly::CALL_v8i16: +  case WebAssembly::CALL_v8i16_S: +  case WebAssembly::CALL_v4i32: +  case WebAssembly::CALL_v4i32_S: +  case WebAssembly::CALL_v2i64: +  case WebAssembly::CALL_v2i64_S: +  case WebAssembly::CALL_v4f32: +  case WebAssembly::CALL_v4f32_S: +  case WebAssembly::CALL_v2f64: +  case WebAssembly::CALL_v2f64_S: +  case WebAssembly::CALL_exnref: +  case WebAssembly::CALL_exnref_S: +  case WebAssembly::RET_CALL: +  case WebAssembly::RET_CALL_S: +    return true; +  default: +    return false; +  } +} + +inline bool isCallIndirect(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::CALL_INDIRECT_VOID: +  case WebAssembly::CALL_INDIRECT_VOID_S: +  case WebAssembly::CALL_INDIRECT_i32: +  case WebAssembly::CALL_INDIRECT_i32_S: +  case WebAssembly::CALL_INDIRECT_i64: +  case WebAssembly::CALL_INDIRECT_i64_S: +  case WebAssembly::CALL_INDIRECT_f32: +  case WebAssembly::CALL_INDIRECT_f32_S: +  case WebAssembly::CALL_INDIRECT_f64: +  case WebAssembly::CALL_INDIRECT_f64_S: +  case WebAssembly::CALL_INDIRECT_v16i8: +  case WebAssembly::CALL_INDIRECT_v16i8_S: +  case WebAssembly::CALL_INDIRECT_v8i16: +  case WebAssembly::CALL_INDIRECT_v8i16_S: +  case WebAssembly::CALL_INDIRECT_v4i32: +  case WebAssembly::CALL_INDIRECT_v4i32_S: +  case WebAssembly::CALL_INDIRECT_v2i64: +  case WebAssembly::CALL_INDIRECT_v2i64_S: +  case WebAssembly::CALL_INDIRECT_v4f32: +  case WebAssembly::CALL_INDIRECT_v4f32_S: +  case WebAssembly::CALL_INDIRECT_v2f64: +  case WebAssembly::CALL_INDIRECT_v2f64_S: +  case WebAssembly::CALL_INDIRECT_exnref: +  case WebAssembly::CALL_INDIRECT_exnref_S: +  case WebAssembly::RET_CALL_INDIRECT: +  case WebAssembly::RET_CALL_INDIRECT_S: +    return true; +  default: +    return false; +  } +} + +/// Returns the operand number of a callee, assuming the argument is a call +/// instruction. +inline unsigned getCalleeOpNo(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::CALL_VOID: +  case WebAssembly::CALL_VOID_S: +  case WebAssembly::CALL_INDIRECT_VOID: +  case WebAssembly::CALL_INDIRECT_VOID_S: +  case WebAssembly::RET_CALL: +  case WebAssembly::RET_CALL_S: +  case WebAssembly::RET_CALL_INDIRECT: +  case WebAssembly::RET_CALL_INDIRECT_S: +    return 0; +  case WebAssembly::CALL_i32: +  case WebAssembly::CALL_i32_S: +  case WebAssembly::CALL_i64: +  case WebAssembly::CALL_i64_S: +  case WebAssembly::CALL_f32: +  case WebAssembly::CALL_f32_S: +  case WebAssembly::CALL_f64: +  case WebAssembly::CALL_f64_S: +  case WebAssembly::CALL_v16i8: +  case WebAssembly::CALL_v16i8_S: +  case WebAssembly::CALL_v8i16: +  case WebAssembly::CALL_v8i16_S: +  case WebAssembly::CALL_v4i32: +  case WebAssembly::CALL_v4i32_S: +  case WebAssembly::CALL_v2i64: +  case WebAssembly::CALL_v2i64_S: +  case WebAssembly::CALL_v4f32: +  case WebAssembly::CALL_v4f32_S: +  case WebAssembly::CALL_v2f64: +  case WebAssembly::CALL_v2f64_S: +  case WebAssembly::CALL_exnref: +  case WebAssembly::CALL_exnref_S: +  case WebAssembly::CALL_INDIRECT_i32: +  case WebAssembly::CALL_INDIRECT_i32_S: +  case WebAssembly::CALL_INDIRECT_i64: +  case WebAssembly::CALL_INDIRECT_i64_S: +  case WebAssembly::CALL_INDIRECT_f32: +  case WebAssembly::CALL_INDIRECT_f32_S: +  case WebAssembly::CALL_INDIRECT_f64: +  case WebAssembly::CALL_INDIRECT_f64_S: +  case WebAssembly::CALL_INDIRECT_v16i8: +  case WebAssembly::CALL_INDIRECT_v16i8_S: +  case WebAssembly::CALL_INDIRECT_v8i16: +  case WebAssembly::CALL_INDIRECT_v8i16_S: +  case WebAssembly::CALL_INDIRECT_v4i32: +  case WebAssembly::CALL_INDIRECT_v4i32_S: +  case WebAssembly::CALL_INDIRECT_v2i64: +  case WebAssembly::CALL_INDIRECT_v2i64_S: +  case WebAssembly::CALL_INDIRECT_v4f32: +  case WebAssembly::CALL_INDIRECT_v4f32_S: +  case WebAssembly::CALL_INDIRECT_v2f64: +  case WebAssembly::CALL_INDIRECT_v2f64_S: +  case WebAssembly::CALL_INDIRECT_exnref: +  case WebAssembly::CALL_INDIRECT_exnref_S: +    return 1; +  default: +    llvm_unreachable("Not a call instruction"); +  } +} + +inline bool isMarker(unsigned Opc) { +  switch (Opc) { +  case WebAssembly::BLOCK: +  case WebAssembly::BLOCK_S: +  case WebAssembly::END_BLOCK: +  case WebAssembly::END_BLOCK_S: +  case WebAssembly::LOOP: +  case WebAssembly::LOOP_S: +  case WebAssembly::END_LOOP: +  case WebAssembly::END_LOOP_S: +  case WebAssembly::TRY: +  case WebAssembly::TRY_S: +  case WebAssembly::END_TRY: +  case WebAssembly::END_TRY_S: +    return true; +  default: +    return false; +  } +} + +} // end namespace WebAssembly +} // end namespace llvm + +#endif diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyTargetStreamer.cpp b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyTargetStreamer.cpp new file mode 100644 index 000000000000..40926201931a --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyTargetStreamer.cpp @@ -0,0 +1,123 @@ +//==-- WebAssemblyTargetStreamer.cpp - WebAssembly Target Streamer Methods --=// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file defines WebAssembly-specific target streamer classes. +/// These are for implementing support for target-specific assembly directives. +/// +//===----------------------------------------------------------------------===// + +#include "MCTargetDesc/WebAssemblyTargetStreamer.h" +#include "MCTargetDesc/WebAssemblyInstPrinter.h" +#include "MCTargetDesc/WebAssemblyMCTargetDesc.h" +#include "llvm/MC/MCContext.h" +#include "llvm/MC/MCSectionWasm.h" +#include "llvm/MC/MCSubtargetInfo.h" +#include "llvm/MC/MCSymbolWasm.h" +#include "llvm/Support/Casting.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/FormattedStream.h" +using namespace llvm; + +WebAssemblyTargetStreamer::WebAssemblyTargetStreamer(MCStreamer &S) +    : MCTargetStreamer(S) {} + +void WebAssemblyTargetStreamer::emitValueType(wasm::ValType Type) { +  Streamer.EmitIntValue(uint8_t(Type), 1); +} + +WebAssemblyTargetAsmStreamer::WebAssemblyTargetAsmStreamer( +    MCStreamer &S, formatted_raw_ostream &OS) +    : WebAssemblyTargetStreamer(S), OS(OS) {} + +WebAssemblyTargetWasmStreamer::WebAssemblyTargetWasmStreamer(MCStreamer &S) +    : WebAssemblyTargetStreamer(S) {} + +static void printTypes(formatted_raw_ostream &OS, +                       ArrayRef<wasm::ValType> Types) { +  bool First = true; +  for (auto Type : Types) { +    if (First) +      First = false; +    else +      OS << ", "; +    OS << WebAssembly::typeToString(Type); +  } +  OS << '\n'; +} + +void WebAssemblyTargetAsmStreamer::emitLocal(ArrayRef<wasm::ValType> Types) { +  if (!Types.empty()) { +    OS << "\t.local  \t"; +    printTypes(OS, Types); +  } +} + +void WebAssemblyTargetAsmStreamer::emitEndFunc() { OS << "\t.endfunc\n"; } + +void WebAssemblyTargetAsmStreamer::emitFunctionType(const MCSymbolWasm *Sym) { +  assert(Sym->isFunction()); +  OS << "\t.functype\t" << Sym->getName() << " "; +  OS << WebAssembly::signatureToString(Sym->getSignature()); +  OS << "\n"; +} + +void WebAssemblyTargetAsmStreamer::emitGlobalType(const MCSymbolWasm *Sym) { +  assert(Sym->isGlobal()); +  OS << "\t.globaltype\t" << Sym->getName() << ", " +     << WebAssembly::typeToString( +            static_cast<wasm::ValType>(Sym->getGlobalType().Type)) +     << '\n'; +} + +void WebAssemblyTargetAsmStreamer::emitEventType(const MCSymbolWasm *Sym) { +  assert(Sym->isEvent()); +  OS << "\t.eventtype\t" << Sym->getName() << " "; +  OS << WebAssembly::typeListToString(Sym->getSignature()->Params); +  OS << "\n"; +} + +void WebAssemblyTargetAsmStreamer::emitImportModule(const MCSymbolWasm *Sym, +                                                    StringRef ImportModule) { +  OS << "\t.import_module\t" << Sym->getName() << ", " +                             << ImportModule << '\n'; +} + +void WebAssemblyTargetAsmStreamer::emitImportName(const MCSymbolWasm *Sym, +                                                  StringRef ImportName) { +  OS << "\t.import_name\t" << Sym->getName() << ", " +                           << ImportName << '\n'; +} + +void WebAssemblyTargetAsmStreamer::emitIndIdx(const MCExpr *Value) { +  OS << "\t.indidx  \t" << *Value << '\n'; +} + +void WebAssemblyTargetWasmStreamer::emitLocal(ArrayRef<wasm::ValType> Types) { +  SmallVector<std::pair<wasm::ValType, uint32_t>, 4> Grouped; +  for (auto Type : Types) { +    if (Grouped.empty() || Grouped.back().first != Type) +      Grouped.push_back(std::make_pair(Type, 1)); +    else +      ++Grouped.back().second; +  } + +  Streamer.EmitULEB128IntValue(Grouped.size()); +  for (auto Pair : Grouped) { +    Streamer.EmitULEB128IntValue(Pair.second); +    emitValueType(Pair.first); +  } +} + +void WebAssemblyTargetWasmStreamer::emitEndFunc() { +  llvm_unreachable(".end_func is not needed for direct wasm output"); +} + +void WebAssemblyTargetWasmStreamer::emitIndIdx(const MCExpr *Value) { +  llvm_unreachable(".indidx encoding not yet implemented"); +} diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyTargetStreamer.h b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyTargetStreamer.h new file mode 100644 index 000000000000..0164f8e572ef --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyTargetStreamer.h @@ -0,0 +1,108 @@ +//==-- WebAssemblyTargetStreamer.h - WebAssembly Target Streamer -*- C++ -*-==// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file declares WebAssembly-specific target streamer classes. +/// These are for implementing support for target-specific assembly directives. +/// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYTARGETSTREAMER_H +#define LLVM_LIB_TARGET_WEBASSEMBLY_MCTARGETDESC_WEBASSEMBLYTARGETSTREAMER_H + +#include "llvm/BinaryFormat/Wasm.h" +#include "llvm/MC/MCStreamer.h" +#include "llvm/Support/MachineValueType.h" + +namespace llvm { + +class MCWasmStreamer; +class MCSymbolWasm; + +/// WebAssembly-specific streamer interface, to implement support +/// WebAssembly-specific assembly directives. +class WebAssemblyTargetStreamer : public MCTargetStreamer { +public: +  explicit WebAssemblyTargetStreamer(MCStreamer &S); + +  /// .local +  virtual void emitLocal(ArrayRef<wasm::ValType> Types) = 0; +  /// .endfunc +  virtual void emitEndFunc() = 0; +  /// .functype +  virtual void emitFunctionType(const MCSymbolWasm *Sym) = 0; +  /// .indidx +  virtual void emitIndIdx(const MCExpr *Value) = 0; +  /// .globaltype +  virtual void emitGlobalType(const MCSymbolWasm *Sym) = 0; +  /// .eventtype +  virtual void emitEventType(const MCSymbolWasm *Sym) = 0; +  /// .import_module +  virtual void emitImportModule(const MCSymbolWasm *Sym, +                                StringRef ImportModule) = 0; +  /// .import_name +  virtual void emitImportName(const MCSymbolWasm *Sym, +                              StringRef ImportName) = 0; + +protected: +  void emitValueType(wasm::ValType Type); +}; + +/// This part is for ascii assembly output +class WebAssemblyTargetAsmStreamer final : public WebAssemblyTargetStreamer { +  formatted_raw_ostream &OS; + +public: +  WebAssemblyTargetAsmStreamer(MCStreamer &S, formatted_raw_ostream &OS); + +  void emitLocal(ArrayRef<wasm::ValType> Types) override; +  void emitEndFunc() override; +  void emitFunctionType(const MCSymbolWasm *Sym) override; +  void emitIndIdx(const MCExpr *Value) override; +  void emitGlobalType(const MCSymbolWasm *Sym) override; +  void emitEventType(const MCSymbolWasm *Sym) override; +  void emitImportModule(const MCSymbolWasm *Sym, StringRef ImportModule) override; +  void emitImportName(const MCSymbolWasm *Sym, StringRef ImportName) override; +}; + +/// This part is for Wasm object output +class WebAssemblyTargetWasmStreamer final : public WebAssemblyTargetStreamer { +public: +  explicit WebAssemblyTargetWasmStreamer(MCStreamer &S); + +  void emitLocal(ArrayRef<wasm::ValType> Types) override; +  void emitEndFunc() override; +  void emitFunctionType(const MCSymbolWasm *Sym) override {} +  void emitIndIdx(const MCExpr *Value) override; +  void emitGlobalType(const MCSymbolWasm *Sym) override {} +  void emitEventType(const MCSymbolWasm *Sym) override {} +  void emitImportModule(const MCSymbolWasm *Sym, +                        StringRef ImportModule) override {} +  void emitImportName(const MCSymbolWasm *Sym, +                      StringRef ImportName) override {} +}; + +/// This part is for null output +class WebAssemblyTargetNullStreamer final : public WebAssemblyTargetStreamer { +public: +  explicit WebAssemblyTargetNullStreamer(MCStreamer &S) +      : WebAssemblyTargetStreamer(S) {} + +  void emitLocal(ArrayRef<wasm::ValType>) override {} +  void emitEndFunc() override {} +  void emitFunctionType(const MCSymbolWasm *) override {} +  void emitIndIdx(const MCExpr *) override {} +  void emitGlobalType(const MCSymbolWasm *) override {} +  void emitEventType(const MCSymbolWasm *) override {} +  void emitImportModule(const MCSymbolWasm *, StringRef) override {} +  void emitImportName(const MCSymbolWasm *, StringRef) override {} +}; + +} // end namespace llvm + +#endif diff --git a/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyWasmObjectWriter.cpp b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyWasmObjectWriter.cpp new file mode 100644 index 000000000000..e7a599e3e175 --- /dev/null +++ b/llvm/lib/Target/WebAssembly/MCTargetDesc/WebAssemblyWasmObjectWriter.cpp @@ -0,0 +1,122 @@ +//===-- WebAssemblyWasmObjectWriter.cpp - WebAssembly Wasm Writer ---------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +/// +/// \file +/// This file handles Wasm-specific object emission, converting LLVM's +/// internal fixups into the appropriate relocations. +/// +//===----------------------------------------------------------------------===// + +#include "MCTargetDesc/WebAssemblyFixupKinds.h" +#include "MCTargetDesc/WebAssemblyMCTargetDesc.h" +#include "llvm/BinaryFormat/Wasm.h" +#include "llvm/MC/MCAsmBackend.h" +#include "llvm/MC/MCFixup.h" +#include "llvm/MC/MCFixupKindInfo.h" +#include "llvm/MC/MCObjectWriter.h" +#include "llvm/MC/MCSectionWasm.h" +#include "llvm/MC/MCSymbolWasm.h" +#include "llvm/MC/MCValue.h" +#include "llvm/MC/MCWasmObjectWriter.h" +#include "llvm/Support/Casting.h" +#include "llvm/Support/ErrorHandling.h" + +using namespace llvm; + +namespace { +class WebAssemblyWasmObjectWriter final : public MCWasmObjectTargetWriter { +public: +  explicit WebAssemblyWasmObjectWriter(bool Is64Bit, bool IsEmscripten); + +private: +  unsigned getRelocType(const MCValue &Target, +                        const MCFixup &Fixup) const override; +}; +} // end anonymous namespace + +WebAssemblyWasmObjectWriter::WebAssemblyWasmObjectWriter(bool Is64Bit, +                                                         bool IsEmscripten) +    : MCWasmObjectTargetWriter(Is64Bit, IsEmscripten) {} + +static const MCSection *getFixupSection(const MCExpr *Expr) { +  if (auto SyExp = dyn_cast<MCSymbolRefExpr>(Expr)) { +    if (SyExp->getSymbol().isInSection()) +      return &SyExp->getSymbol().getSection(); +    return nullptr; +  } + +  if (auto BinOp = dyn_cast<MCBinaryExpr>(Expr)) { +    auto SectionLHS = getFixupSection(BinOp->getLHS()); +    auto SectionRHS = getFixupSection(BinOp->getRHS()); +    return SectionLHS == SectionRHS ? nullptr : SectionLHS; +  } + +  if (auto UnOp = dyn_cast<MCUnaryExpr>(Expr)) +    return getFixupSection(UnOp->getSubExpr()); + +  return nullptr; +} + +unsigned WebAssemblyWasmObjectWriter::getRelocType(const MCValue &Target, +                                                   const MCFixup &Fixup) const { +  const MCSymbolRefExpr *RefA = Target.getSymA(); +  assert(RefA); +  auto& SymA = cast<MCSymbolWasm>(RefA->getSymbol()); + +  MCSymbolRefExpr::VariantKind Modifier = Target.getAccessVariant(); + +  switch (Modifier) { +    case MCSymbolRefExpr::VK_GOT: +      return wasm::R_WASM_GLOBAL_INDEX_LEB; +    case MCSymbolRefExpr::VK_WASM_TBREL: +      assert(SymA.isFunction()); +      return wasm::R_WASM_TABLE_INDEX_REL_SLEB; +    case MCSymbolRefExpr::VK_WASM_MBREL: +      assert(SymA.isData()); +      return wasm::R_WASM_MEMORY_ADDR_REL_SLEB; +    case MCSymbolRefExpr::VK_WASM_TYPEINDEX: +      return wasm::R_WASM_TYPE_INDEX_LEB; +    default: +      break; +  } + +  switch (unsigned(Fixup.getKind())) { +  case WebAssembly::fixup_sleb128_i32: +    if (SymA.isFunction()) +      return wasm::R_WASM_TABLE_INDEX_SLEB; +    return wasm::R_WASM_MEMORY_ADDR_SLEB; +  case WebAssembly::fixup_sleb128_i64: +    llvm_unreachable("fixup_sleb128_i64 not implemented yet"); +  case WebAssembly::fixup_uleb128_i32: +    if (SymA.isGlobal()) +      return wasm::R_WASM_GLOBAL_INDEX_LEB; +    if (SymA.isFunction()) +      return wasm::R_WASM_FUNCTION_INDEX_LEB; +    if (SymA.isEvent()) +      return wasm::R_WASM_EVENT_INDEX_LEB; +    return wasm::R_WASM_MEMORY_ADDR_LEB; +  case FK_Data_4: +    if (SymA.isFunction()) +      return wasm::R_WASM_TABLE_INDEX_I32; +    if (auto Section = static_cast<const MCSectionWasm *>( +            getFixupSection(Fixup.getValue()))) { +      if (Section->getKind().isText()) +        return wasm::R_WASM_FUNCTION_OFFSET_I32; +      else if (!Section->isWasmData()) +        return wasm::R_WASM_SECTION_OFFSET_I32; +    } +    return wasm::R_WASM_MEMORY_ADDR_I32; +  default: +    llvm_unreachable("unimplemented fixup kind"); +  } +} + +std::unique_ptr<MCObjectTargetWriter> +llvm::createWebAssemblyWasmObjectWriter(bool Is64Bit, bool IsEmscripten) { +  return std::make_unique<WebAssemblyWasmObjectWriter>(Is64Bit, IsEmscripten); +} | 
