32#define DEBUG_TYPE "wasm-explicit-locals" 
   37    return "WebAssembly Explicit Locals";
 
   54char WebAssemblyExplicitLocals::ID = 0;
 
   56                "Convert registers to WebAssembly locals", 
false, 
false)
 
   59  return new WebAssemblyExplicitLocals();
 
 
   67      dbgs() << 
"Allocating local " << 
Local << 
"for VReg " 
 
   79  auto P = Reg2Local.
insert(std::make_pair(
Reg, CurLocal));
 
   84  return P.first->second;
 
 
   89  if (RC == &WebAssembly::I32RegClass)
 
   90    return WebAssembly::DROP_I32;
 
   91  if (RC == &WebAssembly::I64RegClass)
 
   92    return WebAssembly::DROP_I64;
 
   93  if (RC == &WebAssembly::F32RegClass)
 
   94    return WebAssembly::DROP_F32;
 
   95  if (RC == &WebAssembly::F64RegClass)
 
   96    return WebAssembly::DROP_F64;
 
   97  if (RC == &WebAssembly::V128RegClass)
 
   98    return WebAssembly::DROP_V128;
 
   99  if (RC == &WebAssembly::FUNCREFRegClass)
 
  100    return WebAssembly::DROP_FUNCREF;
 
  101  if (RC == &WebAssembly::EXTERNREFRegClass)
 
  102    return WebAssembly::DROP_EXTERNREF;
 
  103  if (RC == &WebAssembly::EXNREFRegClass)
 
  104    return WebAssembly::DROP_EXNREF;
 
 
  110  if (RC == &WebAssembly::I32RegClass)
 
  111    return WebAssembly::LOCAL_GET_I32;
 
  112  if (RC == &WebAssembly::I64RegClass)
 
  113    return WebAssembly::LOCAL_GET_I64;
 
  114  if (RC == &WebAssembly::F32RegClass)
 
  115    return WebAssembly::LOCAL_GET_F32;
 
  116  if (RC == &WebAssembly::F64RegClass)
 
  117    return WebAssembly::LOCAL_GET_F64;
 
  118  if (RC == &WebAssembly::V128RegClass)
 
  119    return WebAssembly::LOCAL_GET_V128;
 
  120  if (RC == &WebAssembly::FUNCREFRegClass)
 
  121    return WebAssembly::LOCAL_GET_FUNCREF;
 
  122  if (RC == &WebAssembly::EXTERNREFRegClass)
 
  123    return WebAssembly::LOCAL_GET_EXTERNREF;
 
  124  if (RC == &WebAssembly::EXNREFRegClass)
 
  125    return WebAssembly::LOCAL_GET_EXNREF;
 
 
  131  if (RC == &WebAssembly::I32RegClass)
 
  132    return WebAssembly::LOCAL_SET_I32;
 
  133  if (RC == &WebAssembly::I64RegClass)
 
  134    return WebAssembly::LOCAL_SET_I64;
 
  135  if (RC == &WebAssembly::F32RegClass)
 
  136    return WebAssembly::LOCAL_SET_F32;
 
  137  if (RC == &WebAssembly::F64RegClass)
 
  138    return WebAssembly::LOCAL_SET_F64;
 
  139  if (RC == &WebAssembly::V128RegClass)
 
  140    return WebAssembly::LOCAL_SET_V128;
 
  141  if (RC == &WebAssembly::FUNCREFRegClass)
 
  142    return WebAssembly::LOCAL_SET_FUNCREF;
 
  143  if (RC == &WebAssembly::EXTERNREFRegClass)
 
  144    return WebAssembly::LOCAL_SET_EXTERNREF;
 
  145  if (RC == &WebAssembly::EXNREFRegClass)
 
  146    return WebAssembly::LOCAL_SET_EXNREF;
 
 
  152  if (RC == &WebAssembly::I32RegClass)
 
  153    return WebAssembly::LOCAL_TEE_I32;
 
  154  if (RC == &WebAssembly::I64RegClass)
 
  155    return WebAssembly::LOCAL_TEE_I64;
 
  156  if (RC == &WebAssembly::F32RegClass)
 
  157    return WebAssembly::LOCAL_TEE_F32;
 
  158  if (RC == &WebAssembly::F64RegClass)
 
  159    return WebAssembly::LOCAL_TEE_F64;
 
  160  if (RC == &WebAssembly::V128RegClass)
 
  161    return WebAssembly::LOCAL_TEE_V128;
 
  162  if (RC == &WebAssembly::FUNCREFRegClass)
 
  163    return WebAssembly::LOCAL_TEE_FUNCREF;
 
  164  if (RC == &WebAssembly::EXTERNREFRegClass)
 
  165    return WebAssembly::LOCAL_TEE_EXTERNREF;
 
  166  if (RC == &WebAssembly::EXNREFRegClass)
 
  167    return WebAssembly::LOCAL_TEE_EXNREF;
 
 
  173  if (RC == &WebAssembly::I32RegClass)
 
  175  if (RC == &WebAssembly::I64RegClass)
 
  177  if (RC == &WebAssembly::F32RegClass)
 
  179  if (RC == &WebAssembly::F64RegClass)
 
  181  if (RC == &WebAssembly::V128RegClass)
 
  183  if (RC == &WebAssembly::FUNCREFRegClass)
 
  185  if (RC == &WebAssembly::EXTERNREFRegClass)
 
  186    return MVT::externref;
 
  187  if (RC == &WebAssembly::EXNREFRegClass)
 
 
  202  for (
auto DefReg : Def->defs()) {
 
 
  227  for (
auto *
MI : ToDelete)
 
  228    MI->eraseFromParent();
 
 
  231bool WebAssemblyExplicitLocals::runOnMachineFunction(
MachineFunction &MF) {
 
  232  LLVM_DEBUG(
dbgs() << 
"********** Make Locals Explicit **********\n" 
  233                       "********** Function: " 
  238  WebAssemblyFunctionInfo &MFI = *MF.
getInfo<WebAssemblyFunctionInfo>();
 
  239  const auto *
TII = MF.
getSubtarget<WebAssemblySubtarget>().getInstrInfo();
 
  244  DenseMap<unsigned, unsigned> Reg2Local;
 
  250    MachineInstr &
MI = *
I++;
 
  255    auto Local = 
static_cast<unsigned>(
MI.getOperand(1).
getImm());
 
  260    WebAssemblyDebugValueManager(&
MI).replaceWithLocal(
Local);
 
  262    MI.eraseFromParent();
 
  268  unsigned CurLocal = 
static_cast<unsigned>(MFI.
getParams().
size());
 
  276  BitVector UseEmpty(
MRI.getNumVirtRegs());
 
  277  for (
unsigned I = 0, 
E = 
MRI.getNumVirtRegs(); 
I < 
E; ++
I) {
 
  286  for (MachineBasicBlock &
MBB : MF) {
 
  290      if (
MI.isDebugInstr() || 
MI.isLabel())
 
  293      if (
MI.getOpcode() == WebAssembly::IMPLICIT_DEF) {
 
  294        MI.eraseFromParent();
 
  333        const TargetRegisterClass *RC = 
MRI.getRegClass(DefReg);
 
  337          unsigned LocalId = 
getLocalId(Reg2Local, MFI, CurLocal, DefReg);
 
  342          MI.getOperand(2).setReg(NewReg);
 
  348            getLocalId(Reg2Local, MFI, CurLocal, 
MI.getOperand(1).getReg());
 
  351                MI.getOperand(0).getReg())
 
  355        WebAssemblyDebugValueManager(&
MI).replaceWithLocal(LocalId);
 
  357        MI.eraseFromParent();
 
  363      for (
auto &Def : 
MI.defs()) {
 
  366          const TargetRegisterClass *RC = 
MRI.getRegClass(OldReg);
 
  368          auto InsertPt = std::next(
MI.getIterator());
 
  375            Drop->getOperand(0).setIsKill();
 
  379            unsigned LocalId = 
getLocalId(Reg2Local, MFI, CurLocal, OldReg);
 
  382            WebAssemblyDebugValueManager(&
MI).replaceWithLocal(LocalId);
 
  392          Def.setIsDead(
false);
 
  399      MachineInstr *InsertPt = &
MI;
 
  400      for (MachineOperand &MO : 
reverse(
MI.explicit_uses())) {
 
  411          unsigned LocalId = 
getLocalId(Reg2Local, MFI, CurLocal, OldReg);
 
  414          MI.untieRegOperand(MO.getOperandNo());
 
  415          MO.ChangeToImmediate(LocalId);
 
  428        if (
MI.isInlineAsm()) {
 
  429          unsigned LocalId = 
getLocalId(Reg2Local, MFI, CurLocal, OldReg);
 
  431          MI.untieRegOperand(MO.getOperandNo());
 
  432          MO.ChangeToImmediate(LocalId);
 
  437        unsigned LocalId = 
getLocalId(Reg2Local, MFI, CurLocal, OldReg);
 
  438        const TargetRegisterClass *RC = 
MRI.getRegClass(OldReg);
 
  456        MRI.replaceRegWith(
MI.getOperand(1).getReg(),
 
  457                           MI.getOperand(0).getReg());
 
  458        MI.eraseFromParent();
 
  467  for (
unsigned I = 0, 
E = 
MRI.getNumVirtRegs(); 
I < 
E; ++
I) {
 
  470    if (RL == Reg2Local.
end() || RL->second < MFI.
getParams().size())
 
  480  for (
const MachineBasicBlock &
MBB : MF) {
 
  481    for (
const MachineInstr &
MI : 
MBB) {
 
  482      if (
MI.isDebugInstr() || 
MI.isLabel())
 
  484      for (
const MachineOperand &MO : 
MI.explicit_operands()) {
 
  486            (!MO.isReg() || 
MRI.use_empty(MO.getReg()) ||
 
  488            "WebAssemblyExplicitLocals failed to stackify a register operand");
 
unsigned const MachineRegisterInfo * MRI
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
const HexagonInstrInfo * TII
Promote Memory to Register
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
This file contains the declaration of the WebAssembly-specific manager for DebugValues associated wit...
static unsigned getLocalGetOpcode(const TargetRegisterClass *RC)
Get the appropriate local.get opcode for the given register class.
static unsigned getLocalId(DenseMap< unsigned, unsigned > &Reg2Local, WebAssemblyFunctionInfo &MFI, unsigned &CurLocal, unsigned Reg)
Return a local id number for the given register, assigning it a new one if it doesn't yet have one.
static MachineInstr * findStartOfTree(MachineOperand &MO, MachineRegisterInfo &MRI, const WebAssemblyFunctionInfo &MFI)
Given a MachineOperand of a stackified vreg, return the instruction at the start of the expression tr...
static MVT typeForRegClass(const TargetRegisterClass *RC)
Get the type associated with the given register class.
static unsigned getLocalTeeOpcode(const TargetRegisterClass *RC)
Get the appropriate local.tee opcode for the given register class.
static void checkFrameBase(WebAssemblyFunctionInfo &MFI, unsigned Local, unsigned Reg)
static unsigned getLocalSetOpcode(const TargetRegisterClass *RC)
Get the appropriate local.set opcode for the given register class.
static unsigned getDropOpcode(const TargetRegisterClass *RC)
Get the appropriate drop opcode for the given register class.
static void removeFakeUses(MachineFunction &MF)
This file provides WebAssembly-specific target descriptions.
This file declares WebAssembly-specific per-machine-function information.
This file declares the WebAssembly-specific subclass of TargetSubtarget.
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.
Represent the analysis usage information of a pass.
AnalysisUsage & addPreserved()
Add the specified Pass class to the set of analyses preserved by this pass.
LLVM_ABI void setPreservesCFG()
This function should be called by the pass, iff they do not:
iterator find(const_arg_type_t< KeyT > Val)
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
FunctionPass class - This class is used to implement most global optimizations.
MachineInstrBundleIterator< MachineInstr > iterator
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
StringRef getName() const
getName - Return the name of the corresponding LLVM function.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
const MachineInstrBuilder & addReg(Register RegNo, unsigned flags=0, unsigned SubReg=0) const
Add a new virtual register operand.
Representation of each machine instruction.
const DebugLoc & getDebugLoc() const
Returns the debug location id of this MachineInstr.
MachineOperand class - Representation of each machine instruction operand.
Register getReg() const
getReg - Returns the register number.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
Wrapper class representing virtual and physical registers.
unsigned virtRegIndex() const
Convert a virtual register number to a 0-based index.
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.
This class is derived from MachineFunctionInfo and contains private WebAssembly-specific information ...
void stackifyVReg(MachineRegisterInfo &MRI, Register VReg)
void setLocal(size_t i, MVT VT)
void setNumLocals(size_t NumLocals)
unsigned getFrameBaseVreg() const
void setFrameBaseLocal(unsigned Local)
bool isVRegStackified(Register VReg) const
const std::vector< MVT > & getLocals() const
void clearFrameBaseVreg()
bool isFrameBaseVirtual() const
void unstackifyVReg(Register VReg)
const std::vector< MVT > & getParams() const
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
bool isArgument(unsigned Opc)
bool isCopy(unsigned Opc)
NodeAddr< DefNode * > Def
This is an optimization pass for GlobalISel generic memory operations.
auto size(R &&Range, std::enable_if_t< std::is_base_of< std::random_access_iterator_tag, typename std::iterator_traits< decltype(Range.begin())>::iterator_category >::value, void > *=nullptr)
Get the size of a range.
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
iterator_range< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
auto reverse(ContainerTy &&C)
MachineInstr * getImm(const MachineOperand &MO, const MachineRegisterInfo *MRI)
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
FunctionPass * createWebAssemblyExplicitLocals()