All Downloads are FREE. Search and download functionalities are using the official Maven repository.

scala.scalanative.nscplugin.NirGenExpr.scala Maven / Gradle / Ivy

There is a newer version: 0.5.6
Show newest version
package scala.scalanative
package nscplugin

import scala.annotation.{tailrec, switch}
import scala.collection.mutable
import scala.tools.nsc
import scalanative.util.{StringUtils, unsupported}
import scalanative.util.ScopedVar.scoped
import scalanative.nscplugin.NirPrimitives._

trait NirGenExpr[G <: nsc.Global with Singleton] { self: NirGenPhase[G] =>
  import global.{definitions => defn, _}
  import defn._
  import treeInfo.{hasSynthCaseSymbol, StripCast}
  import nirAddons._
  import nirDefinitions._
  import SimpleType.{fromType, fromSymbol}

  sealed case class ValTree(value: nir.Val)(
      pos: global.Position = global.NoPosition
  ) extends Tree {
    super.setPos(pos)
  }
  object ValTree {
    def apply(from: Tree)(value: nir.Val) =
      new ValTree(value = value)(pos = from.pos)
  }

  sealed case class ContTree(f: ExprBuffer => nir.Val)(
      pos: global.Position
  ) extends Tree { super.setPos(pos) }
  object ContTree {
    def apply(from: Tree)(build: ExprBuffer => nir.Val) =
      new ContTree(f = build)(
        pos = from.pos
      )
  }

  class FixupBuffer(implicit fresh: nir.Fresh) extends nir.InstructionBuilder {
    private var labeled = false

    override def +=(inst: nir.Inst): Unit = {
      implicit val pos: nir.SourcePosition = inst.pos
      inst match {
        case inst: nir.Inst.Label =>
          if (labeled) {
            unreachable(unwind)
          }
          labeled = true
        case _ =>
          if (!labeled) {
            label(fresh())
          }
          labeled = !inst.isInstanceOf[nir.Inst.Cf]
      }
      super.+=(inst)
      inst match {
        case nir.Inst.Let(_, op, _) if op.resty == nir.Type.Nothing =>
          unreachable(unwind)
          label(fresh())
        case _ =>
          ()
      }
    }

    override def ++=(insts: Seq[nir.Inst]): Unit =
      insts.foreach { inst => this += inst }

    override def ++=(other: nir.InstructionBuilder): Unit =
      this ++= other.toSeq
  }

  class ExprBuffer(implicit fresh: nir.Fresh) extends FixupBuffer { buf =>
    def genExpr(tree: Tree): nir.Val = tree match {
      case EmptyTree =>
        nir.Val.Unit
      case ValTree(value) =>
        value
      case ContTree(f) =>
        f(this)
      case tree: Block =>
        genBlock(tree)
      case tree: LabelDef =>
        genLabelDef(tree)
      case tree: ValDef =>
        genValDef(tree)
      case tree: If =>
        genIf(tree)
      case tree: Match =>
        genMatch(tree)
      case tree: Try =>
        genTry(tree)
      case tree: Throw =>
        genThrow(tree)
      case tree: Return =>
        genReturn(tree)
      case tree: Literal =>
        genLiteral(tree)
      case tree: ArrayValue =>
        genArrayValue(tree)
      case tree: This =>
        genThis(tree)
      case tree: Ident =>
        genIdent(tree)
      case tree: Select =>
        genSelect(tree)
      case tree: Assign =>
        genAssign(tree)
      case tree: Typed =>
        genTyped(tree)
      case tree: Function =>
        genFunction(tree)
      case tree: ApplyDynamic =>
        genApplyDynamic(tree)
      case tree: Apply =>
        genApply(tree)
      case _ =>
        abort(
          "Unexpected tree in genExpr: " + tree + "/" + tree.getClass +
            " at: " + tree.pos
        )
    }

    def genBlock(block: Block): nir.Val = {
      val Block(stats, last) = block

      def isCaseLabelDef(tree: Tree) =
        tree.isInstanceOf[LabelDef] && hasSynthCaseSymbol(tree)

      def translateMatch(last: LabelDef) = {
        val (prologue, cases) = stats.span(s => !isCaseLabelDef(s))
        val labels = cases.map { case label: LabelDef => label }
        genMatch(prologue, labels :+ last)
      }

      withFreshBlockScope(block.pos) { parentScope =>
        last match {
          case label: LabelDef if isCaseLabelDef(label) =>
            translateMatch(label)

          case Apply(
                TypeApply(Select(label: LabelDef, nme.asInstanceOf_Ob), _),
                _
              ) if isCaseLabelDef(label) =>
            translateMatch(label)

          case _ =>
            stats.foreach(genExpr(_))
            genExpr(last)
        }
      }
    }

    def genLabelDef(label: LabelDef): nir.Val = {
      assert(label.params.isEmpty, "empty LabelDef params")
      buf.jump(nir.Next(curMethodEnv.enterLabel(label)))(label.pos)
      genLabel(label)
    }

    def genLabel(label: LabelDef): nir.Val = {
      val local = curMethodEnv.resolveLabel(label)
      val params = label.params.map { id =>
        val local = nir.Val.Local(fresh(), genType(id.tpe))
        curMethodEnv.enter(id.symbol, local)
        local
      }

      buf.label(local, params)(label.pos)
      genExpr(label.rhs)
    }

    def genTailRecLabel(
        dd: DefDef,
        isStatic: Boolean,
        label: LabelDef
    ): nir.Val = {
      val local = curMethodEnv.resolveLabel(label)
      val params = label.params.zip(genParamSyms(dd, isStatic)).map {
        case (lparam, mparamopt) =>
          val local = nir.Val.Local(fresh(), genType(lparam.tpe))
          curMethodEnv.enter(lparam.symbol, local)
          mparamopt.foreach(curMethodEnv.enter(_, local))
          local
      }

      buf.label(local, params)(label.pos)
      if (isStatic) {
        genExpr(label.rhs)
      } else
        withFreshBlockScope(label.rhs.pos) { _ =>
          scoped(
            curMethodThis := Some(params.head)
          )(genExpr(label.rhs))
        }
    }

    def genValDef(vd: ValDef): nir.Val = {
      implicit val pos: nir.SourcePosition = vd.pos
      val localNames = curMethodLocalNames.get
      val isMutable = curMethodInfo.mutableVars.contains(vd.symbol)
      def name = genLocalName(vd.symbol)

      val rhs = genExpr(vd.rhs) match {
        case v @ nir.Val.Local(id, _) =>
          if (localNames.contains(id) || isMutable) ()
          else localNames.update(id, name)
          vd.rhs match {
            // When rhs is a block patch the scopeId of it's result to match the current scopeId
            // This allows us to reflect that ValDef is accessible in this scope
            case _: Block | Typed(_: Block, _) | Try(_: Block, _, _) |
                Try(Typed(_: Block, _), _, _) =>
              buf.updateLetInst(id)(i => i.copy()(i.pos, curScopeId.get))
            case _ => ()
          }
          v
        case nir.Val.Unit => nir.Val.Unit
        case v =>
          if (isMutable) v
          else buf.let(namedId(fresh)(name), nir.Op.Copy(v), unwind)
      }
      if (isMutable) {
        val slot = curMethodEnv.resolve(vd.symbol)
        buf.varstore(slot, rhs, unwind)
      } else {
        curMethodEnv.enter(vd.symbol, rhs)
        nir.Val.Unit
      }
    }

    def genIf(tree: If): nir.Val = {
      val If(cond, thenp, elsep) = tree
      def isUnitType(tpe: Type) =
        defn.isUnitType(tpe) || tpe =:= defn.BoxedUnitTpe
      val retty =
        if (isUnitType(thenp.tpe) || isUnitType(elsep.tpe)) nir.Type.Unit
        else genType(tree.tpe)
      genIf(retty, cond, thenp, elsep)(tree.pos.orElse(fallbackSourcePosition))
    }

    def genIf(
        retty: nir.Type,
        condp: Tree,
        thenp: Tree,
        elsep: Tree,
        ensureLinktime: Boolean = false
    )(implicit ifPos: nir.SourcePosition): nir.Val = {
      val thenn, elsen, mergen = fresh()
      val mergev = nir.Val.Local(fresh(), retty)

      getLinktimeCondition(condp).fold {
        if (ensureLinktime) {
          globalError(
            condp.pos,
            "Cannot resolve given condition in linktime, it might be depending on runtime value"
          )
        }
        val cond = genExpr(condp)
        buf.branch(cond, nir.Next(thenn), nir.Next(elsen))(
          condp.pos.orElse(ifPos)
        )
      } { cond =>
        curMethodEnv.get.isUsingLinktimeResolvedValue = true
        buf.branchLinktime(cond, nir.Next(thenn), nir.Next(elsen))(
          condp.pos.orElse(ifPos)
        )
      }

      locally {
        buf.label(thenn)(thenp.pos.orElse(ifPos))
        val thenv = genExpr(thenp)
        buf.jumpExcludeUnitValue(retty)(mergen, thenv)
      }
      locally {
        buf.label(elsen)(elsep.pos.orElse(ifPos))
        val elsev = genExpr(elsep)
        buf.jumpExcludeUnitValue(retty)(mergen, elsev)
      }
      buf.labelExcludeUnitValue(mergen, mergev)
    }

    def genMatch(m: Match): nir.Val = {
      val Match(scrutp, allcaseps) = m
      type Case = (nir.Local, nir.Val, Tree, global.Position)

      // Extract switch cases and assign unique names to them.
      val caseps: Seq[Case] = allcaseps.flatMap {
        case CaseDef(Ident(nme.WILDCARD), _, _) =>
          Seq.empty
        case cd @ CaseDef(pat, guard, body) =>
          assert(guard.isEmpty, "CaseDef guard was not empty")
          val vals: Seq[nir.Val] = pat match {
            case lit: Literal =>
              List(genLiteralValue(lit))
            case Alternative(alts) =>
              alts.map {
                case lit: Literal => genLiteralValue(lit)
              }
            case _ =>
              Nil
          }
          vals.map((fresh(), _, body, cd.pos))
      }

      // Extract default case.
      val defaultp: Tree = allcaseps.collectFirst {
        case c @ CaseDef(Ident(nme.WILDCARD), _, body) => body
      }.get

      val retty = genType(m.tpe)
      val scrut = genExpr(scrutp)

      // Generate code for the switch and its cases.
      def genSwitch(): nir.Val = {
        // Generate some more fresh names and types.
        val casenexts = caseps.map { case (n, v, _, _) => nir.Next.Case(v, n) }
        val defaultnext = nir.Next(fresh())
        val merge = fresh()
        val mergev = nir.Val.Local(fresh(), retty)

        implicit val pos: nir.SourcePosition = m.pos

        // Generate code for the switch and its cases.
        val scrut = genExpr(scrutp)
        buf.switch(scrut, defaultnext, casenexts)
        buf.label(defaultnext.id)(defaultp.pos)
        buf.jumpExcludeUnitValue(retty)(merge, genExpr(defaultp))(
          defaultp.pos
        )
        caseps.foreach {
          case (n, _, expr, pos) =>
            buf.label(n)(pos)
            val caseres = genExpr(expr)
            buf.jumpExcludeUnitValue(retty)(merge, caseres)(pos)
        }
        buf.labelExcludeUnitValue(merge, mergev)
      }

      def genIfsChain(): nir.Val = {
        /* Default label needs to be generated before any others and then added to
         * current MethodEnv. It's label might be referenced in any of them in
         * case of match with guards, eg.:
         *
         * "Hello, World!" match {
         *  case "Hello" if cond1 => "foo"
         *  case "World" if cond2 => "bar"
         *  case _ if cond3 => "bar-baz"
         *  case _ => "baz-bar"
         * }
         *
         * might be translated to something like:
         *
         * val x1 = "Hello, World!"
         * if(x1 == "Hello"){ if(cond1) "foo" else default4() }
         * else if (x1 == "World"){ if(cond2) "bar" else default4() }
         * else default4()
         *
         * def default4() = if(cond3) "bar-baz" else "baz-bar
         *
         * We need to make sure to only generate LabelDef at this stage.
         * Generating other ASTs and mutating state might lead to unexpected
         * runtime errors.
         */
        val optDefaultLabel = defaultp match {
          case label: LabelDef => Some(genLabelDef(label))
          case _               => None
        }

        def loop(cases: List[Case]): nir.Val = {
          cases match {
            case (_, caze, body, p) :: elsep =>
              implicit val pos: nir.SourcePosition = p

              val cond =
                buf.genClassEquality(
                  leftp = ValTree(scrut)(p),
                  rightp = ValTree(caze)(p),
                  ref = false,
                  negated = false
                )
              buf.genIf(
                retty = retty,
                condp = ValTree(cond)(p),
                thenp = ContTree(body)(_.genExpr(body)),
                elsep = ContTree(_ => loop(elsep))(p)
              )

            case Nil => optDefaultLabel.getOrElse(genExpr(defaultp))
          }
        }
        loop(caseps.toList)
      }

      /* Since 2.13 we need to enforce that only Int switch cases reach backend
       * For all other cases we're generating If-else chain */
      val isIntMatch = scrut.ty == nir.Type.Int &&
        caseps.forall(_._2.ty == nir.Type.Int)

      if (isIntMatch) genSwitch()
      else genIfsChain()
    }

    def genMatch(prologue: List[Tree], lds: List[LabelDef]): nir.Val = {
      // Generate prologue expressions.
      prologue.foreach(genExpr(_))

      // Enter symbols for all labels and jump to the first one.
      lds.foreach(curMethodEnv.enterLabel)
      val firstLd = lds.head
      buf.jump(nir.Next(curMethodEnv.resolveLabel(firstLd)))(firstLd.pos)

      // Generate code for all labels and return value of the last one.
      lds.map(genLabel(_)).last
    }

    def genTry(tree: Try): nir.Val = tree match {
      case Try(expr, catches, finalizer)
          if catches.isEmpty && finalizer.isEmpty =>
        genExpr(expr)
      case Try(expr, catches, finalizer) =>
        val retty = genType(tree.tpe)
        genTry(retty, expr, catches, finalizer)(tree.pos)
    }

    def genTry(
        retty: nir.Type,
        expr: Tree,
        catches: List[Tree],
        finallyp: Tree
    )(enclosingPos: nir.SourcePosition): nir.Val = {
      val handler = fresh()
      val excn = fresh()
      val normaln = fresh()
      val mergen = fresh()
      val excv = nir.Val.Local(fresh(), nir.Rt.Object)
      val mergev = nir.Val.Local(fresh(), retty)

      implicit val pos: nir.SourcePosition = expr.pos.orElse(enclosingPos)
      // Nested code gen to separate out try/catch-related instructions.
      val nested = new ExprBuffer
      withFreshBlockScope(pos) { _ =>
        scoped(
          curUnwindHandler := Some(handler)
        ) {
          nested.label(normaln)
          val res = nested.genExpr(expr)
          nested.jumpExcludeUnitValue(retty)(mergen, res)
        }
      }
      withFreshBlockScope(pos) { _ =>
        nested.label(handler, Seq(excv))
        val res = nested.genTryCatch(retty, excv, mergen, catches)
        nested.jumpExcludeUnitValue(retty)(mergen, res)
      }

      // Append finally to the try/catch instructions and merge them back.
      val insts =
        if (finallyp.isEmpty) nested.toSeq
        else genTryFinally(finallyp, nested.toSeq)

      // Append try/catch instructions to the outher instruction buffer.
      buf.jump(nir.Next(normaln))
      buf ++= insts
      buf.labelExcludeUnitValue(mergen, mergev)
    }

    def genTryCatch(
        retty: nir.Type,
        exc: nir.Val,
        mergen: nir.Local,
        catches: List[Tree]
    )(implicit exprPos: nir.SourcePosition): nir.Val = {
      val cases = catches.map {
        case CaseDef(pat, _, body) =>
          val (excty, symopt) = pat match {
            case Typed(Ident(nme.WILDCARD), tpt) =>
              (genType(tpt.tpe), None)
            case Ident(nme.WILDCARD) =>
              (genType(ThrowableClass.tpe), None)
            case Bind(_, _) =>
              (genType(pat.symbol.tpe), Some(pat.symbol))
          }
          val f = ContTree(body) { (buf: ExprBuffer) =>
            withFreshBlockScope(body.pos) { _ =>
              symopt.foreach { sym =>
                val cast = buf.as(excty, exc, unwind)
                curMethodLocalNames.get.update(cast.id, genLocalName(sym))
                curMethodEnv.enter(sym, cast)
              }
              val res = genExpr(body)
              buf.jumpExcludeUnitValue(retty)(mergen, res)
            }
            nir.Val.Unit
          }
          (excty, f, exprPos)
      }

      def wrap(
          cases: Seq[(nir.Type, ContTree, nir.SourcePosition)]
      ): nir.Val =
        cases match {
          case Seq() =>
            buf.raise(exc, unwind)
            nir.Val.Unit
          case (excty, f, pos) +: rest =>
            val cond = buf.is(excty, exc, unwind)(pos, getScopeId)
            genIf(
              retty,
              ValTree(f)(cond),
              f,
              ContTree(f)(_ => wrap(rest))
            )(pos)
        }

      wrap(cases)
    }

    def genTryFinally(finallyp: Tree, insts: Seq[nir.Inst]): Seq[nir.Inst] = {
      val labels =
        insts.collect {
          case nir.Inst.Label(n, _) => n
        }.toSet
      def internal(cf: nir.Inst.Cf) = cf match {
        case inst @ nir.Inst.Jump(n) =>
          labels.contains(n.id)
        case inst @ nir.Inst.If(_, n1, n2) =>
          labels.contains(n1.id) && labels.contains(n2.id)
        case inst @ nir.Inst.LinktimeIf(_, n1, n2) =>
          labels.contains(n1.id) && labels.contains(n2.id)
        case inst @ nir.Inst.Switch(_, n, ns) =>
          labels.contains(n.id) && ns.forall(n => labels.contains(n.id))
        case inst @ nir.Inst.Throw(_, n) =>
          (n ne nir.Next.None) && labels.contains(n.id)
        case _ =>
          false
      }

      val finalies = new ExprBuffer
      val transformed = insts.map {
        case cf: nir.Inst.Cf if internal(cf) =>
          // We don't touch control-flow within try/catch block.
          cf
        case cf: nir.Inst.Cf =>
          // All control-flow edges that jump outside the try/catch block
          // must first go through finally block if it's present. We generate
          // a new copy of the finally handler for every edge.
          val finallyn = fresh()
          withFreshBlockScope(cf.pos) { _ =>
            finalies.label(finallyn)(cf.pos)
            val res = finalies.genExpr(finallyp)
          }
          finalies += cf
          // The original jump outside goes through finally block first.
          nir.Inst.Jump(nir.Next(finallyn))(cf.pos)
        case inst =>
          inst
      }
      transformed ++ finalies.toSeq
    }

    def genThrow(tree: Throw): nir.Val = {
      val Throw(exprp) = tree
      val res = genExpr(exprp)
      buf.raise(res, unwind)(tree.pos)
      nir.Val.Unit
    }

    def genReturn(tree: Return): nir.Val = {
      val Return(exprp) = tree
      genReturn(genExpr(exprp))(exprp.pos)
    }

    def genReturn(value: nir.Val)(implicit pos: nir.SourcePosition): nir.Val = {
      val retv =
        if (curMethodIsExtern.get) {
          val nir.Type.Function(_, retty) = genExternMethodSig(curMethodSym)
          toExtern(retty, value)
        } else {
          value
        }
      buf.ret(retv)
      nir.Val.Unit
    }

    def genLiteral(lit: Literal): nir.Val = {
      val value = lit.value
      implicit val pos: nir.SourcePosition =
        lit.pos.orElse(fallbackSourcePosition)
      value.tag match {
        case UnitTag | NullTag | BooleanTag | ByteTag | ShortTag | CharTag |
            IntTag | LongTag | FloatTag | DoubleTag | StringTag =>
          genLiteralValue(lit)

        case ClazzTag =>
          genTypeValue(value.typeValue)

        case EnumTag =>
          genStaticMember(EmptyTree, value.symbolValue)
      }
    }

    def genLiteralValue(lit: Literal): nir.Val = {
      val value = lit.value
      value.tag match {
        case UnitTag =>
          nir.Val.Unit
        case NullTag =>
          nir.Val.Null
        case BooleanTag =>
          if (value.booleanValue) nir.Val.True else nir.Val.False
        case ByteTag =>
          nir.Val.Byte(value.intValue.toByte)
        case ShortTag =>
          nir.Val.Short(value.intValue.toShort)
        case CharTag =>
          nir.Val.Char(value.intValue.toChar)
        case IntTag =>
          nir.Val.Int(value.intValue)
        case LongTag =>
          nir.Val.Long(value.longValue)
        case FloatTag =>
          nir.Val.Float(value.floatValue)
        case DoubleTag =>
          nir.Val.Double(value.doubleValue)
        case StringTag =>
          nir.Val.String(value.stringValue)
      }
    }

    def genArrayValue(av: ArrayValue): nir.Val = {
      val ArrayValue(tpt, elems) = av
      implicit val pos: nir.SourcePosition =
        av.pos.orElse(fallbackSourcePosition)
      genArrayValue(tpt, elems)
    }

    def genArrayValue(tpt: Tree, elems: Seq[Tree])(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      val elemty = genType(tpt.tpe)
      val values = genSimpleArgs(elems)

      if (values.forall(_.isCanonical) && values.exists(v => !v.isZero)) {
        buf.arrayalloc(elemty, nir.Val.ArrayValue(elemty, values), unwind)
      } else {
        val alloc = buf.arrayalloc(elemty, nir.Val.Int(elems.length), unwind)
        values.zip(elems).zipWithIndex.foreach {
          case ((v, elem), i) =>
            if (!v.isZero) {
              buf.arraystore(elemty, alloc, nir.Val.Int(i), v, unwind)(
                elem.pos.orElse(pos),
                getScopeId
              )
            }
        }
        alloc
      }
    }

    def genThis(tree: This): nir.Val =
      if (curMethodThis.nonEmpty && tree.symbol == curClassSym.get) {
        curMethodThis.get.get
      } else {
        genModule(tree.symbol)(tree.pos)
      }

    def genModule(sym: Symbol)(implicit pos: nir.SourcePosition): nir.Val = {
      if (sym.isModule && sym.isScala3Defined &&
          sym.hasAttachment[DottyEnumSingletonCompat.type]) {
        /* #2983 This is a reference to a singleton `case` from a Scala 3 `enum`.
         * It is not a module. Instead, it is a static field (accessed through
         * a static getter) in the `enum` class.
         * We use `originalOwner` and `rawname` because that's what the JVM back-end uses.
         */
        val className = genTypeName(sym.originalOwner.companionClass)
        val getterMethodName = nir.Sig.Method(
          sym.rawname.toString(),
          Seq(genType(sym.tpe)),
          nir.Sig.Scope.PublicStatic
        )
        val name = className.member(getterMethodName)
        buf.call(
          ty = genMethodSig(sym),
          ptr = nir.Val.Global(name, nir.Type.Ptr),
          args = Nil,
          unwind = unwind
        )
      } else {
        buf.module(genModuleName(sym), unwind)
      }
    }

    def genIdent(tree: Ident): nir.Val = {
      val sym = tree.symbol
      implicit val pos: nir.SourcePosition =
        tree.pos.orElse(fallbackSourcePosition)
      if (curMethodInfo.mutableVars.contains(sym)) {
        buf.varload(curMethodEnv.resolve(sym), unwind)
      } else if (sym.isModule) {
        genModule(sym)
      } else {
        curMethodEnv.resolve(sym)
      }
    }

    def genSelect(tree: Select): nir.Val = {
      val Select(qualp, selp) = tree

      val sym = tree.symbol
      val owner = sym.owner
      implicit val pos: nir.SourcePosition = tree.pos.orElse(curMethodSym.pos)

      if (sym.isModule) {
        genModule(sym)
      } else if (sym.isStaticMember) {
        genStaticMember(qualp, sym)
      } else if (sym.isMethod) {
        genApplyMethod(sym, statically = false, qualp, Seq.empty)
      } else if (owner.isStruct) {
        val index = owner.info.decls.filter(_.isField).toList.indexOf(sym)
        val qual = genExpr(qualp)
        buf.extract(qual, Seq(index), unwind)
      } else {
        val ty = genType(tree.symbol.tpe)
        val name = genFieldName(tree.symbol)
        if (sym.isExtern) {
          val externTy = genExternType(tree.symbol.tpe)
          genLoadExtern(ty, externTy, tree.symbol)
        } else {
          val qual = genExpr(qualp)
          buf.fieldload(ty, qual, name, unwind)
        }
      }
    }

    def genStaticMember(receiver: Tree, sym: Symbol)(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      if (sym == BoxedUnit_UNIT) nir.Val.Unit
      else genApplyStaticMethod(sym, receiver.symbol, Seq.empty)
    }

    def genAssign(tree: Assign): nir.Val = {
      val Assign(lhsp, rhsp) = tree
      implicit val pos: nir.SourcePosition =
        tree.pos.orElse(lhsp.pos).orElse(rhsp.pos).orElse(curMethodSym.pos)

      lhsp match {
        case sel @ Select(qualp, _) =>
          val sym = sel.symbol
          val qual = genExpr(qualp)
          val rhs = genExpr(rhsp)
          val name = genFieldName(sym)
          if (sym.isExtern) {
            val externTy = genExternType(sym.tpe)
            genStoreExtern(externTy, sym, rhs)
          } else {
            val ty = genType(sym.tpe)
            val qual = genExpr(qualp)
            buf.fieldstore(ty, qual, name, rhs, unwind)
          }

        case id: Ident =>
          val rhs = genExpr(rhsp)
          val slot = curMethodEnv.resolve(id.symbol)
          buf.varstore(slot, rhs, unwind)
      }
    }

    def genTyped(tree: Typed): nir.Val = tree match {
      case Typed(Super(_, _), _) =>
        curMethodThis.get.get
      case Typed(expr, _) =>
        genExpr(expr)
    }

    // On Scala 2.12 (and on 2.11 with -Ydelambdafy:method), closures are preserved
    // until the backend to be compiled using LambdaMetaFactory.
    //
    // Scala Native does not have any special treatment for closures.
    // We reimplement the anonymous class generation which was originally used on
    // Scala 2.11 and earlier.
    //
    // Each anonymous function gets a generated class for it,
    // similarly to closure encoding on Scala 2.11 and earlier:
    //
    //   class AnonFunX extends java.lang.Object with FunctionalInterface {
    //     var capture1: T1
    //     ...
    //     var captureN: TN
    //     def (this, v1: T1, ..., vN: TN): Unit = {
    //       java.lang.Object.(this)
    //       this.capture1 = v1
    //       ...
    //       this.captureN = vN
    //     }
    //     def samMethod(param1: Tp1, ..., paramK: TpK): Tret = {
    //       EnclsoingClass.this.staticMethod(param1, ..., paramK, this.capture1, ..., this.captureN)
    //     }
    //   }
    //
    // Bridges might require multiple samMethod variants to be created.
    def genFunction(tree: Function): nir.Val = {
      val Function(
        paramTrees,
        callTree @ Apply(targetTree @ Select(_, _), functionArgs)
      ) =
        tree
      implicit val pos: nir.SourcePosition = tree.pos

      val funSym = tree.tpe.typeSymbolDirect
      val paramSyms = paramTrees.map(_.symbol)
      val captureSyms =
        global.delambdafy.FreeVarTraverser.freeVarsOf(tree).toSeq

      val statBuf = curStatBuffer.get

      // Generate an anonymous class definition.

      val suffix = "$$Lambda$" + curClassFresh.get.apply().id
      val anonName = nir.Global.Top(genName(curClassSym).top.id + suffix)
      val traitName = genTypeName(funSym)

      statBuf += nir.Defn.Class(
        nir.Attrs.None,
        anonName,
        Some(nir.Rt.Object.name),
        Seq(traitName)
      )

      // Generate fields to store the captures.

      // enclosing class `this` reference + capture symbols
      val captureSymsWithEnclThis = curClassSym.get +: captureSyms

      val captureTypes = captureSymsWithEnclThis.map(sym => genType(sym.tpe))
      val captureNames =
        captureSymsWithEnclThis.zipWithIndex.map {
          case (sym, idx) =>
            val name = anonName.member(nir.Sig.Field("capture" + idx))
            val ty = genType(sym.tpe)
            statBuf += nir.Defn.Var(nir.Attrs.None, name, ty, nir.Val.Zero(ty))
            name
        }

      // Generate an anonymous class constructor that initializes all the fields.

      val ctorName = anonName.member(nir.Sig.Ctor(captureTypes))
      val ctorTy =
        nir.Type.Function(nir.Type.Ref(anonName) +: captureTypes, nir.Type.Unit)
      val ctorBody = scoped(curScopeId := nir.ScopeId.TopLevel) {
        val fresh = nir.Fresh()
        val buf = new nir.InstructionBuilder()(fresh)
        val self = nir.Val.Local(fresh(), nir.Type.Ref(anonName))
        val captureFormals = captureTypes.map { ty =>
          nir.Val.Local(fresh(), ty)
        }
        buf.label(fresh(), self +: captureFormals)
        val superTy = nir.Type.Function(Seq(nir.Rt.Object), nir.Type.Unit)
        val superName = nir.Rt.Object.name.member(nir.Sig.Ctor(Seq.empty))
        val superCtor = nir.Val.Global(superName, nir.Type.Ptr)
        buf.call(superTy, superCtor, Seq(self), nir.Next.None)
        captureNames.zip(captureFormals).foreach {
          case (name, capture) =>
            buf.fieldstore(capture.ty, self, name, capture, nir.Next.None)
        }
        buf.ret(nir.Val.Unit)
        buf.toSeq
      }

      statBuf += new nir.Defn.Define(nir.Attrs.None, ctorName, ctorTy, ctorBody)

      // Generate methods that implement SAM interface each of the required signatures.

      functionMethodSymbols(tree).foreach { funSym =>
        val funSig = genName(funSym).asInstanceOf[nir.Global.Member].sig
        val funName = anonName.member(funSig)

        val selfType = nir.Type.Ref(anonName)
        val nir.Sig.Method(_, sigTypes :+ retType, _) = funSig.unmangled
        val paramTypes = selfType +: sigTypes

        val bodyFresh = nir.Fresh()
        val bodyEnv = new MethodEnv(fresh)

        val body = scoped(
          curMethodEnv := bodyEnv,
          curMethodInfo := (new CollectMethodInfo).collect(EmptyTree),
          curFresh := bodyFresh,
          curScopeId := nir.ScopeId.TopLevel,
          curUnwindHandler := None
        ) {
          val fresh = nir.Fresh()
          val buf = new ExprBuffer()(fresh)
          val self = nir.Val.Local(fresh(), selfType)
          val params = sigTypes.map { ty => nir.Val.Local(fresh(), ty) }
          buf.label(fresh(), self +: params)

          // At this point, the type parameter symbols are all Objects.
          // We need to transform them, so that their type conforms to
          // what the apply method expects:
          // - values that can be unboxed, are unboxed
          // - otherwise, the value is cast to the appropriate type
          paramSyms
            .zip(functionArgs.takeRight(sigTypes.length))
            .zip(params)
            .foreach {
              case ((sym, arg), value) =>
                implicit val pos: nir.SourcePosition = arg.pos

                val result =
                  enteringPhase(currentRun.posterasurePhase)(sym.tpe) match {
                    case tpe if tpe.sym.isPrimitiveValueClass =>
                      val targetTpe = genType(tpe)
                      if (targetTpe == value.ty) value
                      else buf.unbox(genBoxType(tpe), value, nir.Next.None)

                    case ErasedValueType(valueClazz, underlying) =>
                      val unboxMethod = valueClazz.derivedValueClassUnbox
                      val casted =
                        buf.genCastOp(value.ty, genType(valueClazz), value)
                      val unboxed = buf.genApplyMethod(
                        sym = unboxMethod,
                        statically = false,
                        self = casted,
                        argsp = Nil
                      )
                      if (unboxMethod.tpe.resultType == underlying)
                        unboxed
                      else
                        buf.genCastOp(unboxed.ty, genType(underlying), unboxed)

                    case _ =>
                      val unboxed =
                        buf.unboxValue(sym.tpe, partial = true, value)
                      if (unboxed == value) // no need to or cannot unbox, we should cast
                        buf.genCastOp(genType(sym.tpe), genType(arg.tpe), value)
                      else unboxed
                  }
                curMethodEnv.enter(sym, result)
            }

          captureSymsWithEnclThis.zip(captureNames).foreach {
            case (sym, name) =>
              val value =
                buf.fieldload(genType(sym.tpe), self, name, nir.Next.None)
              curMethodEnv.enter(sym, value)
          }

          val sym = targetTree.symbol
          val method = nir.Val.Global(genMethodName(sym), nir.Type.Ptr)
          val values =
            buf.genMethodArgs(sym, Ident(curClassSym.get) +: functionArgs)
          val sig = genMethodSig(sym)
          val res = buf.call(sig, method, values, nir.Next.None)

          // Get the result type of the lambda after erasure, when entering posterasure.
          // This allows to recover the correct type in case value classes are involved.
          // In that case, the type will be an ErasedValueType.
          val resTyEnteringPosterasure =
            enteringPhase(currentRun.posterasurePhase) {
              targetTree.symbol.tpe.resultType
            }
          buf.ret(
            if (retType == res.ty && resTyEnteringPosterasure == sym.tpe.resultType)
              res
            else
              ensureBoxed(res, resTyEnteringPosterasure, callTree.tpe)(
                buf,
                callTree.pos
              )
          )
          buf.toSeq
        }

        statBuf += new nir.Defn.Define(
          nir.Attrs.None,
          funName,
          nir.Type.Function(paramTypes, retType),
          body
        )
      }

      // Generate call site of the closure allocation to
      // instantiante the anonymous class and call its constructor
      // passing all of the captures as arguments.

      val alloc = buf.classalloc(anonName, unwind)
      val captureVals = curMethodThis.get.get +: captureSyms.map { sym =>
        genExpr(Ident(sym))
      }
      buf.call(
        ctorTy,
        nir.Val.Global(ctorName, nir.Type.Ptr),
        alloc +: captureVals,
        unwind
      )
      alloc
    }

    def ensureBoxed(
        value: nir.Val,
        tpeEnteringPosterasure: Type,
        targetTpe: Type
    )(implicit buf: ExprBuffer, pos: nir.SourcePosition): nir.Val = {
      tpeEnteringPosterasure match {
        case tpe if isPrimitiveValueType(tpe) =>
          buf.boxValue(targetTpe, value)

        case tpe: ErasedValueType =>
          val boxedClass = tpe.valueClazz
          val ctorName = genMethodName(boxedClass.primaryConstructor)
          val ctorSig = genMethodSig(boxedClass.primaryConstructor)

          val alloc =
            buf.classalloc(nir.Global.Top(boxedClass.fullName), unwind)
          val ctor = buf.method(
            alloc,
            ctorName.asInstanceOf[nir.Global.Member].sig,
            unwind
          )
          buf.call(ctorSig, ctor, Seq(alloc, value), unwind)

          alloc

        case _ =>
          value
      }
    }

    private def ensureUnboxed(
        value: nir.Val,
        tpeEnteringPosterasure: Type
    )(implicit buf: ExprBuffer, pos: nir.SourcePosition): nir.Val = {
      tpeEnteringPosterasure match {
        case tpe if isPrimitiveValueType(tpe) =>
          val targetTpe = genType(tpeEnteringPosterasure)
          if (targetTpe == value.ty) value
          else buf.unbox(genBoxType(tpe), value, nir.Next.None)

        case tpe: ErasedValueType =>
          val valueClass = tpe.valueClazz
          val unboxMethod = treeInfo.ValueClass.valueUnbox(tpe)
          val castedValue = buf.genCastOp(value.ty, genType(valueClass), value)
          buf.genApplyMethod(
            sym = unboxMethod,
            statically = false,
            self = castedValue,
            argsp = Nil
          )

        case tpe =>
          val unboxed = buf.unboxValue(tpe, partial = true, value)
          if (unboxed == value) // no need to or cannot unbox, we should cast
            buf.genCastOp(genType(tpeEnteringPosterasure), genType(tpe), value)
          else unboxed
      }
    }

    // Compute a set of method symbols that SAM-generated class needs to implement.
    def functionMethodSymbols(tree: Function): Seq[Symbol] = {
      val funSym = tree.tpe.typeSymbolDirect
      if (isFunctionSymbol(funSym)) {
        unspecializedSymbol(funSym).info.members
          .filter(_.name.toString == "apply")
          .toSeq
      } else {
        val samInfo = tree.attachments.get[SAMFunction].getOrElse {
          abort(
            s"Cannot find the SAMFunction attachment on $tree at ${tree.pos}"
          )
        }

        val samsBuilder = List.newBuilder[Symbol]
        val seenSignatures = mutable.Set.empty[nir.Sig]

        val synthCls = samInfo.synthCls
        // On Scala < 2.12.5, `synthCls` is polyfilled to `NoSymbol`
        // and hence `samBridges` will always be empty (scala/bug#10512).
        // Since we only support Scala 2.12.12 and up,
        // we assert that this is not the case.
        assert(synthCls != NoSymbol, "Unexpected NoSymbol")
        val samBridges = {
          import scala.reflect.internal.Flags.BRIDGE
          synthCls.info
            .findMembers(excludedFlags = 0L, requiredFlags = BRIDGE)
            .toList
        }

        for (sam <- samInfo.sam :: samBridges) {
          // Remove duplicates, e.g., if we override the same method declared
          // in two super traits.
          val sig = genName(sam).asInstanceOf[nir.Global.Member].sig
          if (seenSignatures.add(sig))
            samsBuilder += sam
        }

        samsBuilder.result()
      }
    }

    def genApplyDynamic(app: ApplyDynamic): nir.Val = {
      val ApplyDynamic(obj, args) = app
      val sym = app.symbol
      implicit val pos: nir.SourcePosition = app.pos

      val params = sym.tpe.params

      val isEqEqOrBangEq =
        (sym.name == EqEqMethodName || sym.name == NotEqMethodName) &&
          params.size == 1

      // If the method is '=='or '!=' generate class equality instead of dyn-call
      if (isEqEqOrBangEq) {
        val neg = sym.name == nme.ne || sym.name == NotEqMethodName
        val last = genClassEquality(obj, args.head, ref = false, negated = neg)
        buf.box(nir.Type.Ref(nir.Global.Top("java.lang.Boolean")), last, unwind)
      } else {
        genApplyDynamic(sym, obj, args)
      }
    }

    def genApplyDynamic(sym: Symbol, obj: Tree, argsp: Seq[Tree])(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      val self = genExpr(obj)
      val methodSig = genMethodSig(sym).asInstanceOf[nir.Type.Function]
      val params = sym.tpe.params

      def isArrayLikeOp = {
        sym.name == nme.update &&
        params.size == 2 && params.head.tpe.typeSymbol == IntClass
      }

      def genDynCall(arrayUpdate: Boolean)(buf: ExprBuffer) = {

        // In the case of an array update we need to manually erase the return type.
        val methodName: nir.Sig =
          if (arrayUpdate) {
            nir.Sig.Proxy("update", Seq(nir.Type.Int, nir.Rt.Object))
          } else {
            val nir.Global.Member(_, sig) = genMethodName(sym)
            sig.toProxy
          }

        val sig =
          nir.Type.Function(
            methodSig.args.head ::
              methodSig.args.tail
                .map(ty => nir.Type.box.getOrElse(ty, ty))
                .toList,
            nir.Type.Ref(nir.Global.Top("java.lang.Object"))
          )

        val callerType = methodSig.args.head
        val boxedArgTypes =
          methodSig.args.tail.map(ty => nir.Type.box.getOrElse(ty, ty)).toList

        val retType = nir.Type.Ref(nir.Global.Top("java.lang.Object"))
        val signature = nir.Type.Function(callerType :: boxedArgTypes, retType)
        val args = genMethodArgs(sym, argsp)

        val method = buf.dynmethod(self, methodName, unwind)
        val values = self +: args

        val call = buf.call(signature, method, values, unwind)
        buf.as(
          nir.Type.box.getOrElse(methodSig.ret, methodSig.ret),
          call,
          unwind
        )
      }

      // If the signature matches an array update, tests at runtime if it really is an array update.
      if (isArrayLikeOp) {
        val cond = ContTree(obj) { (buf: ExprBuffer) =>
          buf.is(
            nir.Type.Ref(
              nir.Global.Top("scala.scalanative.runtime.ObjectArray")
            ),
            self,
            unwind
          )
        }
        val thenp = ContTree(obj)(genDynCall(arrayUpdate = true))
        val elsep = ContTree(obj)(genDynCall(arrayUpdate = false))
        genIf(
          nir.Type.Ref(nir.Global.Top("java.lang.Object")),
          cond,
          thenp,
          elsep
        )

      } else {
        genDynCall(arrayUpdate = false)(this)
      }
    }

    def genApply(app: Apply): nir.Val = {
      def tree = app
      val Apply(fun, args) = app

      implicit val pos: nir.SourcePosition =
        app.pos.orElse(fallbackSourcePosition)
      def fail(msg: String) = {
        reporter.error(app.pos, msg)
        nir.Val.Null
      }
      tree match {
        case _ if fun.symbol == ExternMethod =>
          fail(s"extern can be used only from non-inlined extern methods")
        case Apply(_: TypeApply, _) =>
          genApplyTypeApply(app)
        case Apply(Select(Super(_, _), _), _) =>
          genApplyMethod(
            fun.symbol,
            statically = true,
            curMethodThis.get.get,
            args
          )
        case Apply(Select(New(_), nme.CONSTRUCTOR), _) =>
          genApplyNew(app)

        // Based on Scala2 Cleanup phase, and WrapArray extractor defined in Scala.js variant
        // Replaces `Array(.wrapArray(ArrayValue(...).$asInstanceOf[...]), )`
        // with just `ArrayValue(...).$asInstanceOf[...]`
        //
        // See scala/bug#6611; we must *only* do this for literal vararg arrays.
        // format: off
        case Apply(appMeth @ Select(appMethQual, _), Apply(wrapRefArrayMeth, StripCast(arrValue @ ArrayValue(elemtpt, elems)) :: Nil) :: classTagEvidence :: Nil)
        if WrapArray.isClassTagBasedWrapArrayMethod(wrapRefArrayMeth.symbol) &&
                appMeth.symbol == ArrayModule_genericApply &&
                !elemtpt.tpe.typeSymbol.isBottomClass &&
                !elemtpt.tpe.typeSymbol.isPrimitiveValueClass /* can happen via specialization.*/ =>
        
          classTagEvidence.attachments.get[analyzer.MacroExpansionAttachment] match {
            case Some(att) 
            if att.expandee.symbol.name == nme.materializeClassTag && 
               tree.isInstanceOf[ApplyToImplicitArgs] =>
                 genArrayValue(arrValue)
            case _ =>
              val arrValue = genApplyMethod(
                  ClassTagClass.info.decl(nme.newArray),
                  statically = false,
                  classTagEvidence,
                  ValTree(tree)(nir.Val.Int(elems.size)) :: Nil
                )
              val scalaRuntimeTimeModule = genModule(ScalaRunTimeModule)
              elems.zipWithIndex.foreach { case (elem, i) =>
                genApplyModuleMethod(
                  ScalaRunTimeModule,
                  currentRun.runDefinitions.arrayUpdateMethod,
                  ValTree(tree)(arrValue) :: ValTree(tree)(nir.Val.Int(i)) :: elem :: Nil
                )
              }
              arrValue
          }

        case Apply(appMeth @ Select(appMethQual, _), elem0 :: WrapArray(rest @ ArrayValue(elemtpt, elems)) :: Nil)
            if appMeth.symbol == ArrayModule_apply(elemtpt.tpe) && 
               treeInfo.isQualifierSafeToElide(appMethQual) =>
          genArrayValue(elemtpt, elem0 +: elems)

        // See scala/bug#12201, should be rewrite as Primitive Array.
        // Match Array
        case Apply(appMeth @ Select(appMethQual, _), WrapArray(arrValue: ArrayValue) :: _ :: Nil) 
        if appMeth.symbol == ArrayModule_genericApply && 
           treeInfo.isQualifierSafeToElide(appMethQual) =>
          genArrayValue(arrValue)

        case Apply(appMeth @ Select(appMethQual, _), elem :: (nil: RefTree) :: Nil)
        if nil.symbol == NilModule && appMeth.symbol == ArrayModule_apply(elem.tpe.widen) && 
           treeInfo.isExprSafeToInline(nil) && 
           treeInfo.isQualifierSafeToElide(appMethQual) =>
          genArrayValue(TypeTree(elem.tpe.widen), elem :: Nil)
        
        // format: on
        case _ =>
          val sym = fun.symbol
          if (sym.isLabel) {
            genApplyLabel(app)
          } else if (scalaPrimitives.isPrimitive(sym)) {
            genApplyPrimitive(app)
          } else if (currentRun.runDefinitions.isBox(sym)) {
            val arg = args.head
            genApplyBox(arg.tpe, arg)
          } else if (currentRun.runDefinitions.isUnbox(sym)) {
            genApplyUnbox(app.tpe, args.head)
          } else {
            val Select(receiverp, _) = fun
            genApplyMethod(fun.symbol, statically = false, receiverp, args)
          }
      }
    }

    def genApplyLabel(tree: Tree): nir.Val = {
      val Apply(fun, argsp) = tree
      val nir.Val.Local(label, _) = curMethodEnv.resolve(fun.symbol)
      val args = genSimpleArgs(argsp)
      buf.jump(label, args)(tree.pos)
      nir.Val.Unit
    }

    def genApplyBox(st: SimpleType, argp: Tree)(implicit
        enclosingPos: nir.SourcePosition
    ): nir.Val = {
      val value = genExpr(argp)

      buf.box(genBoxType(st), value, unwind)(
        argp.pos.orElse(enclosingPos),
        getScopeId
      )
    }

    def genApplyUnbox(st: SimpleType, argp: Tree)(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      val value = genExpr(argp)
      value.ty match {
        case _: nir.Type.I | _: nir.Type.F =>
          // No need for unboxing, fixing some slack generated by the general
          // purpose Scala compiler.
          value
        case _ =>
          buf.unbox(genBoxType(st), value, unwind)
      }
    }

    def genApplyPrimitive(app: Apply): nir.Val = {
      import scalaPrimitives._

      val Apply(fun @ Select(receiver, _), args) = app
      implicit val pos: nir.SourcePosition = app.pos

      val sym = app.symbol
      val code = scalaPrimitives.getPrimitive(sym, receiver.tpe)
      (code: @switch) match {
        case CONCAT                 => genStringConcat(app)
        case HASH                   => genHashCode(args.head)
        case CFUNCPTR_APPLY         => genCFuncPtrApply(app, code)
        case CFUNCPTR_FROM_FUNCTION => genCFuncFromScalaFunction(app)
        case SYNCHRONIZED =>
          val Apply(Select(receiverp, _), List(argp)) = app
          genSynchronized(receiverp, argp)(app.pos)
        case STACKALLOC              => genStackalloc(app)
        case CLASS_FIELD_RAWPTR      => genClassFieldRawPtr(app)
        case SIZE_OF                 => genSizeOf(app)
        case ALIGNMENT_OF            => genAlignmentOf(app)
        case CQUOTE                  => genCQuoteOp(app)
        case BOXED_UNIT              => nir.Val.Unit
        case USES_LINKTIME_INTRINSIC => genLinktimeIntrinsicApply(app)
        case code =>
          if (isArithmeticOp(code) || isLogicalOp(code) || isComparisonOp(code))
            genSimpleOp(app, receiver :: args, code)
          else if (isArrayOp(code) || code == ARRAY_CLONE) genArrayOp(app, code)
          else if (nirPrimitives.isRawPtrOp(code)) genRawPtrOp(app, code)
          else if (nirPrimitives.isRawPtrCastOp(code))
            genRawPtrCastOp(app, code)
          else if (nirPrimitives.isRawSizeCastOp(code))
            genRawSizeCastOp(app, args.head, code)
          else if (isCoercion(code)) genCoercion(app, receiver, code)
          else if (code >= DIV_UINT && code <= ULONG_TO_DOUBLE)
            genUnsignedOp(app, code)
          else {
            abort(
              "Unknown primitive operation: " + sym.fullName + "(" +
                fun.symbol.simpleName + ") " + " at: " + (app.pos)
            )
          }
      }
    }

    private def genLinktimeIntrinsicApply(app: Apply): nir.Val = {
      import nirDefinitions._
      implicit def pos: nir.SourcePosition = app.pos
      val Apply(fun, args) = app

      val sym = fun.symbol
      val Select(receiverp, _) = fun
      val isStatic = sym.owner.isStaticOwner

      sym match {
        case _
            if JavaUtilServiceLoaderLoad.contains(sym) ||
              JavaUtilServiceLoaderLoadInstalled == sym =>
          args.head match {
            case Literal(c: Constant) => () // ok
            case _ =>
              reporter.error(
                app.pos,
                s"Limitation of ScalaNative runtime: first argument of ${sym} needs to be literal constant of class type, use `classOf[T]` instead."
              )
          }
        case _ =>
          reporter.error(
            app.pos,
            s"Unhandled intrinsic function call for $sym"
          )
      }

      curMethodEnv.get.isUsingIntrinsics = true
      genApplyMethod(sym, statically = isStatic, receiverp, args)
    }

    private final val ExternForwarderSig =
      nir.Sig.Generated("$extern$forwarder")

    def getLinktimeCondition(condp: Tree): Option[nir.LinktimeCondition] = {
      import nir.LinktimeCondition._
      def genComparsion(name: Name, value: nir.Val): nir.Comp = {
        def intOrFloatComparison(onInt: nir.Comp, onFloat: nir.Comp)(implicit
            tpe: nir.Type
        ) =
          if (tpe.isInstanceOf[nir.Type.F]) onFloat else onInt

        import nir.Comp._
        implicit val tpe: nir.Type = value.ty
        name match {
          case nme.EQ => intOrFloatComparison(Ieq, Feq)
          case nme.NE => intOrFloatComparison(Ine, Fne)
          case nme.GT => intOrFloatComparison(Sgt, Fgt)
          case nme.GE => intOrFloatComparison(Sge, Fge)
          case nme.LT => intOrFloatComparison(Slt, Flt)
          case nme.LE => intOrFloatComparison(Sle, Fle)
          case nme =>
            globalError(condp.pos, s"Unsupported condition '$nme'");
            nir.Comp.Ine
        }
      }

      condp match {
        // if(bool) (...)
        case Apply(LinktimeProperty(name, _, position), Nil) =>
          Some {
            SimpleCondition(
              propertyName = name,
              comparison = nir.Comp.Ieq,
              value = nir.Val.True
            )(position)
          }

        // if(!bool) (...)
        case Apply(
              Select(
                Apply(LinktimeProperty(name, _, position), Nil),
                nme.UNARY_!
              ),
              Nil
            ) =>
          Some {
            SimpleCondition(
              propertyName = name,
              comparison = nir.Comp.Ieq,
              value = nir.Val.False
            )(position)
          }

        // if(property  x) (...)
        case Apply(
              Select(LinktimeProperty(name, _, position), comp),
              List(arg @ Literal(Constant(_)))
            ) =>
          Some {
            val argValue = genLiteralValue(arg)
            SimpleCondition(
              propertyName = name,
              comparison = genComparsion(comp, argValue),
              value = argValue
            )(position)
          }

        // if(cond1 {&&,||} cond2) (...)
        case Apply(Select(cond1, op), List(cond2)) =>
          (getLinktimeCondition(cond1), getLinktimeCondition(cond2)) match {
            case (Some(c1), Some(c2)) =>
              val bin = op match {
                case nme.ZAND => nir.Bin.And
                case nme.ZOR  => nir.Bin.Or
              }
              Some(ComplexCondition(bin, c1, c2)(condp.pos))
            case (None, None) => None
            case _ =>
              globalError(
                condp.pos,
                "Mixing link-time and runtime conditions is not allowed"
              )
              None
          }

        case _ => None
      }
    }

    def genFuncExternForwarder(funcName: nir.Global, treeSym: Symbol)(implicit
        pos: nir.SourcePosition
    ): nir.Defn = {
      val attrs = nir.Attrs(isExtern = true)

      val sig = genMethodSig(treeSym)
      val externSig = genExternMethodSig(treeSym)

      val nir.Type.Function(origtys, _) = sig
      val nir.Type.Function(paramtys, retty) = externSig

      val methodName = genMethodName(treeSym)
      val method = nir.Val.Global(methodName, nir.Type.Ptr)
      val methodRef = nir.Val.Global(methodName, origtys.head)

      val forwarderName = funcName.member(ExternForwarderSig)
      val forwarderBody = scoped(
        curUnwindHandler := None,
        curScopeId := nir.ScopeId.TopLevel
      ) {
        val fresh = nir.Fresh()
        val buf = new ExprBuffer()(fresh)

        val params = paramtys.map(ty => nir.Val.Local(fresh(), ty))
        buf.label(fresh(), params)
        val boxedParams = params.zip(origtys.tail).map {
          case (param, ty) => buf.fromExtern(ty, param)
        }

        val res = buf.call(sig, method, methodRef +: boxedParams, nir.Next.None)
        val unboxedRes = buf.toExtern(retty, res)
        buf.ret(unboxedRes)

        buf.toSeq
      }
      new nir.Defn.Define(attrs, forwarderName, externSig, forwarderBody)
    }

    def genCFuncFromScalaFunction(app: Apply): nir.Val = {
      implicit val pos: nir.SourcePosition = app.pos
      val fn = app.args.head

      def withGeneratedForwarder(fnRef: nir.Val)(sym: Symbol): nir.Val = {
        val nir.Type.Ref(className, _, _) = fnRef.ty
        curStatBuffer += genFuncExternForwarder(className, sym)
        fnRef
      }

      def reportClosingOverLocalState(args: Seq[Tree]): Unit =
        reporter.error(
          fn.pos,
          s"Closing over local state of ${args.map(v => show(v.symbol)).mkString(", ")} in function transformed to CFuncPtr results in undefined behaviour."
        )

      @tailrec
      def resolveFunction(tree: Tree): nir.Val = tree match {
        case Typed(expr, _) => resolveFunction(expr)
        case Block(_, expr) => resolveFunction(expr)
        case fn @ Function(
              params,
              Apply(targetTree, targetArgs)
            ) => // Scala 2.12+
          val paramTermNames = params.map(_.name)
          val localStateParams = targetArgs
            .filter(arg => !paramTermNames.contains(arg.symbol.name))
          if (localStateParams.nonEmpty)
            reportClosingOverLocalState(localStateParams)

          withGeneratedForwarder {
            genFunction(fn)
          }(targetTree.symbol)

        case _ =>
          unsupported(
            "Failed to resolve function ref for extern forwarder "
              + s"in ${showRaw(fn)} [$pos]"
          )
      }

      val fnRef = resolveFunction(fn)
      val className = genTypeName(app.tpe.sym)

      val ctorTy = nir.Type.Function(
        Seq(nir.Type.Ref(className), nir.Type.Ptr),
        nir.Type.Unit
      )
      val ctorName = className.member(nir.Sig.Ctor(Seq(nir.Type.Ptr)))
      val rawptr = buf.method(fnRef, ExternForwarderSig, unwind)

      val alloc = buf.classalloc(className, unwind)
      buf.call(
        ctorTy,
        nir.Val.Global(ctorName, nir.Type.Ptr),
        Seq(alloc, rawptr),
        unwind
      )
      alloc
    }

    def numOfType(num: Int, ty: nir.Type): nir.Val = ty match {
      case nir.Type.Byte                  => nir.Val.Byte(num.toByte)
      case nir.Type.Short | nir.Type.Char => nir.Val.Short(num.toShort)
      case nir.Type.Int                   => nir.Val.Int(num)
      case nir.Type.Long                  => nir.Val.Long(num.toLong)
      case nir.Type.Size                  => nir.Val.Size(num.toLong)
      case nir.Type.Float                 => nir.Val.Float(num.toFloat)
      case nir.Type.Double                => nir.Val.Double(num.toDouble)
      case _ => unsupported(s"num = $num, ty = ${ty.show}")
    }

    def genSimpleOp(app: Apply, args: List[Tree], code: Int): nir.Val = {
      val retty = genType(app.tpe)

      implicit val pos: nir.SourcePosition =
        app.pos.orElse(fallbackSourcePosition)

      args match {
        case List(right)       => genUnaryOp(code, right, retty)
        case List(left, right) => genBinaryOp(code, left, right, retty)
        case _ => abort("Too many arguments for primitive function: " + app)
      }
    }

    def negateInt(value: nir.Val)(implicit pos: nir.SourcePosition): nir.Val =
      buf.bin(nir.Bin.Isub, value.ty, numOfType(0, value.ty), value, unwind)
    def negateFloat(value: nir.Val)(implicit pos: nir.SourcePosition): nir.Val =
      buf.bin(nir.Bin.Fmul, value.ty, numOfType(-1, value.ty), value, unwind)
    def negateBits(value: nir.Val)(implicit pos: nir.SourcePosition): nir.Val =
      buf.bin(nir.Bin.Xor, value.ty, numOfType(-1, value.ty), value, unwind)
    def negateBool(value: nir.Val)(implicit pos: nir.SourcePosition): nir.Val =
      buf.bin(nir.Bin.Xor, nir.Type.Bool, nir.Val.True, value, unwind)

    def genUnaryOp(code: Int, rightp: Tree, opty: nir.Type)(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      import scalaPrimitives._

      val right = genExpr(rightp)
      val coerced = genCoercion(right, right.ty, opty)

      (opty, code) match {
        case (_: nir.Type.I | _: nir.Type.F, POS) => coerced
        case (_: nir.Type.I, NOT)                 => negateBits(coerced)
        case (_: nir.Type.F, NEG)                 => negateFloat(coerced)
        case (_: nir.Type.I, NEG)                 => negateInt(coerced)
        case (nir.Type.Bool, ZNOT)                => negateBool(coerced)
        case _ => abort("Unknown unary operation code: " + code)
      }
    }

    def genBinaryOp(code: Int, left: Tree, right: Tree, retty: nir.Type)(
        implicit exprPos: nir.SourcePosition
    ): nir.Val = {
      import scalaPrimitives._

      val lty = genType(left.tpe)
      val rty = genType(right.tpe)
      val opty =
        if (isShiftOp(code)) {
          if (lty == nir.Type.Long) {
            nir.Type.Long
          } else {
            nir.Type.Int
          }
        } else {
          binaryOperationType(lty, rty)
        }

      val binres = opty match {
        case _: nir.Type.F =>
          code match {
            case ADD =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Fadd, _, _, _), left, right, opty)
            case SUB =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Fsub, _, _, _), left, right, opty)
            case MUL =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Fmul, _, _, _), left, right, opty)
            case DIV =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Fdiv, _, _, _), left, right, opty)
            case MOD =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Frem, _, _, _), left, right, opty)

            case EQ =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Feq, _, _, _), left, right, opty)
            case NE =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Fne, _, _, _), left, right, opty)
            case LT =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Flt, _, _, _), left, right, opty)
            case LE =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Fle, _, _, _), left, right, opty)
            case GT =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Fgt, _, _, _), left, right, opty)
            case GE =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Fge, _, _, _), left, right, opty)

            case _ =>
              abort(
                "Unknown floating point type binary operation code: " + code
              )
          }

        case nir.Type.Bool | _: nir.Type.I =>
          code match {
            case ADD =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Iadd, _, _, _), left, right, opty)
            case SUB =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Isub, _, _, _), left, right, opty)
            case MUL =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Imul, _, _, _), left, right, opty)
            case DIV =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Sdiv, _, _, _), left, right, opty)
            case MOD =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Srem, _, _, _), left, right, opty)

            case OR =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Or, _, _, _), left, right, opty)
            case XOR =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Xor, _, _, _), left, right, opty)
            case AND =>
              genBinaryOp(nir.Op.Bin(nir.Bin.And, _, _, _), left, right, opty)
            case LSL =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Shl, _, _, _), left, right, opty)
            case LSR =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Lshr, _, _, _), left, right, opty)
            case ASR =>
              genBinaryOp(nir.Op.Bin(nir.Bin.Ashr, _, _, _), left, right, opty)

            case EQ =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Ieq, _, _, _), left, right, opty)
            case NE =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Ine, _, _, _), left, right, opty)
            case LT =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Slt, _, _, _), left, right, opty)
            case LE =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Sle, _, _, _), left, right, opty)
            case GT =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Sgt, _, _, _), left, right, opty)
            case GE =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Sge, _, _, _), left, right, opty)

            case ZOR =>
              genIf(retty, left, Literal(Constant(true)), right)
            case ZAND =>
              genIf(retty, left, right, Literal(Constant(false)))

            case _ =>
              abort("Unknown integer type binary operation code: " + code)
          }

        case _: nir.Type.RefKind =>
          def genEquals(ref: Boolean, negated: Boolean) = (left, right) match {
            // If null is present on either side, we must always
            // generate reference equality, regardless of where it
            // was called with == or eq. This shortcut is not optional.
            case (Literal(Constant(null)), _) | (_, Literal(Constant(null))) =>
              genClassEquality(left, right, ref = true, negated = negated)
            case _ =>
              genClassEquality(left, right, ref = ref, negated = negated)
          }

          code match {
            case EQ =>
              genEquals(ref = false, negated = false)
            case NE =>
              genEquals(ref = false, negated = true)
            case ID =>
              genEquals(ref = true, negated = false)
            case NI =>
              genEquals(ref = true, negated = true)

            case _ =>
              abort("Unknown reference type operation code: " + code)
          }

        case nir.Type.Ptr =>
          code match {
            case EQ | ID =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Ieq, _, _, _), left, right, opty)
            case NE | NI =>
              genBinaryOp(nir.Op.Comp(nir.Comp.Ine, _, _, _), left, right, opty)
          }

        case ty =>
          abort("Unknown binary operation type: " + ty)
      }

      genCoercion(binres, binres.ty, retty)(right.pos)
    }

    def genBinaryOp(
        op: (nir.Type, nir.Val, nir.Val) => nir.Op,
        leftp: Tree,
        rightp: Tree,
        opty: nir.Type
    )(implicit enclosingPos: nir.SourcePosition): nir.Val = {
      val leftPos: nir.SourcePosition = leftp.pos.orElse(enclosingPos)
      val leftty = genType(leftp.tpe)
      val left = genExpr(leftp)
      val leftcoerced = genCoercion(left, leftty, opty)(leftPos)
      val rightty = genType(rightp.tpe)
      val rightPos: nir.SourcePosition = rightp.pos.orElse(enclosingPos)
      val right = genExpr(rightp)
      val rightcoerced = genCoercion(right, rightty, opty)

      buf.let(op(opty, leftcoerced, rightcoerced), unwind)
    }

    private def genClassEquality(
        leftp: Tree,
        rightp: Tree,
        ref: Boolean,
        negated: Boolean
    )(implicit pos: nir.SourcePosition): nir.Val = {

      if (ref) {
        // referencial equality
        val left = genExpr(leftp)
        val right = genExpr(rightp)
        val comp = if (negated) nir.Comp.Ine else nir.Comp.Ieq
        buf.comp(comp, nir.Rt.Object, left, right, unwind)
      } else genClassUniversalEquality(leftp, rightp, negated)
    }

    private def genClassUniversalEquality(l: Tree, r: Tree, negated: Boolean)(
        implicit pos: nir.SourcePosition
    ): nir.Val = {

      /* True if the equality comparison is between values that require the use of the rich equality
       * comparator (scala.runtime.BoxesRunTime.equals). This is the case when either side of the
       * comparison might have a run-time type subtype of java.lang.Number or java.lang.Character.
       * When it is statically known that both sides are equal and subtypes of Number of Character,
       * not using the rich equality is possible (their own equals method will do ok.)
       */
      val mustUseAnyComparator: Boolean = {
        // Exclude custom trees introduced by Scala Natvie from checks
        def isScalaTree(tree: Tree) = tree match {
          case _: ValTree  => false
          case _: ContTree => false
          case _           => true
        }
        val usesOnlyScalaTrees = isScalaTree(l) && isScalaTree(r)
        def areSameFinals =
          l.tpe.isFinalType && r.tpe.isFinalType && (l.tpe =:= r.tpe) && {
            val sym = l.tpe.typeSymbol
            sym != BoxedFloatClass && sym != BoxedDoubleClass
          }
        usesOnlyScalaTrees && !areSameFinals &&
          platform.isMaybeBoxed(l.tpe.typeSymbol) &&
          platform.isMaybeBoxed(r.tpe.typeSymbol)
      }
      def isNull(t: Tree) = PartialFunction.cond(t) {
        case Literal(Constant(null)) => true
      }
      def isLiteral(t: Tree) = PartialFunction.cond(t) {
        case Literal(_) => true
      }
      def isNonNullExpr(t: Tree) =
        isLiteral(t) || ((t.symbol ne null) && t.symbol.isModule)
      def maybeNegate(v: nir.Val): nir.Val = if (negated) negateBool(v) else v
      def comparator = if (negated) nir.Comp.Ine else nir.Comp.Ieq
      if (mustUseAnyComparator) maybeNegate {
        val equalsMethod: Symbol = {
          if (l.tpe <:< BoxedNumberClass.tpe) {
            if (r.tpe <:< BoxedNumberClass.tpe) platform.externalEqualsNumNum
            else if (r.tpe <:< BoxedCharacterClass.tpe)
              platform.externalEqualsNumChar
            else platform.externalEqualsNumObject
          } else platform.externalEquals
        }
        genApplyStaticMethod(equalsMethod, defn.BoxesRunTimeModule, Seq(l, r))
      }
      else if (isNull(l)) {
        // null == expr -> expr eq null
        buf.comp(comparator, nir.Rt.Object, genExpr(r), nir.Val.Null, unwind)
      } else if (isNull(r)) {
        // expr == null -> expr eq null
        buf.comp(comparator, nir.Rt.Object, genExpr(l), nir.Val.Null, unwind)
      } else if (isNonNullExpr(l)) maybeNegate {
        // SI-7852 Avoid null check if L is statically non-null.
        genApplyMethod(
          sym = defn.Any_equals,
          statically = false,
          selfp = l,
          argsp = Seq(r)
        )
      }
      else
        maybeNegate {
          // l == r -> if (l eq null) r eq null else l.equals(r)
          val thenn, elsen, mergen = fresh()
          val mergev = nir.Val.Local(fresh(), nir.Type.Bool)
          val left = genExpr(l)
          val isnull =
            buf.comp(nir.Comp.Ieq, nir.Rt.Object, left, nir.Val.Null, unwind)
          buf.branch(isnull, nir.Next(thenn), nir.Next(elsen))
          locally {
            buf.label(thenn)
            val right = genExpr(r)
            val thenv =
              buf.comp(nir.Comp.Ieq, nir.Rt.Object, right, nir.Val.Null, unwind)
            buf.jump(mergen, Seq(thenv))
          }
          locally {
            buf.label(elsen)
            val elsev = genApplyMethod(
              defn.Any_equals,
              statically = false,
              left,
              Seq(r)
            )
            buf.jump(mergen, Seq(elsev))
          }
          buf.label(mergen, Seq(mergev))
          mergev
        }
    }

    def binaryOperationType(lty: nir.Type, rty: nir.Type) = (lty, rty) match {
      // Bug compatibility with scala/bug/issues/11253
      case (nir.Type.Long, nir.Type.Float) =>
        nir.Type.Double

      case (nir.Type.Ptr, _: nir.Type.RefKind) =>
        lty

      case (_: nir.Type.RefKind, nir.Type.Ptr) =>
        rty

      case (nir.Type.Bool, nir.Type.Bool) =>
        nir.Type.Bool

      case (lhs: nir.Type.FixedSizeI, rhs: nir.Type.FixedSizeI) =>
        if (lhs.width < 32 && rhs.width < 32) {
          nir.Type.Int
        } else if (lhs.width >= rhs.width) {
          lhs
        } else {
          rhs
        }

      case (_: nir.Type.FixedSizeI, _: nir.Type.F) =>
        rty

      case (_: nir.Type.F, _: nir.Type.FixedSizeI) =>
        lty

      case (lhs: nir.Type.F, rhs: nir.Type.F) =>
        if (lhs.width >= rhs.width) lhs else rhs

      case (_: nir.Type.RefKind, _: nir.Type.RefKind) =>
        nir.Rt.Object

      case (ty1, ty2) if ty1 == ty2 =>
        ty1

      case (nir.Type.Nothing, ty) =>
        ty

      case (ty, nir.Type.Nothing) =>
        ty

      case _ =>
        abort(s"can't perform binary operation between $lty and $rty")
    }

    /*
     * Returns a list of trees that each should be concatenated, from left to right.
     * It turns a chained call like "a".+("b").+("c") into a list of arguments.
     */
    def liftStringConcat(tree: Tree): List[Tree] = {
      val result = collection.mutable.ListBuffer[Tree]()
      def loop(tree: Tree): Unit = {
        tree match {
          case Apply(fun @ Select(larg, method), rarg :: Nil)
              if (scalaPrimitives.isPrimitive(fun.symbol) &&
                scalaPrimitives.getPrimitive(fun.symbol) ==
                scalaPrimitives.CONCAT) =>
            loop(larg)
            loop(rarg)
          case _ =>
            result += tree
        }
      }
      loop(tree)
      result.toList
    }

    /* Issue a call to `StringBuilder#append` for the right element type     */
    private final def genStringBuilderAppend(
        stringBuilder: nir.Val.Local,
        tree: Tree
    ): Unit = {
      implicit val nirPos: nir.SourcePosition =
        tree.pos.orElse(fallbackSourcePosition)

      val tpe = tree.tpe
      val argType =
        if (tpe <:< defn.StringTpe) nir.Rt.String
        else if (tpe <:< nirDefinitions.jlStringBufferType)
          genType(nirDefinitions.jlStringBufferRef)
        else if (tpe <:< nirDefinitions.jlCharSequenceType)
          genType(nirDefinitions.jlCharSequenceRef)
        // Don't match for `Array(Char)`, even though StringBuilder has such an overload:
        // `"a" + Array('b')` should NOT be "ab", but "a[C@...".
        else if (tpe <:< defn.ObjectTpe) nir.Rt.Object
        else genType(tpe)

      val value = genExpr(tree)
      val (adaptedValue, targetType) = argType match {
        // jlStringBuilder does not have overloads for byte and short, but we can just use the int version
        case nir.Type.Byte | nir.Type.Short =>
          genCoercion(value, value.ty, nir.Type.Int) -> nir.Type.Int
        case nirType => value -> nirType
      }

      val (appendFunction, appendSig) =
        jlStringBuilderAppendForSymbol(targetType)
      buf.call(
        appendSig,
        appendFunction,
        Seq(stringBuilder, adaptedValue),
        unwind
      )
    }

    private lazy val jlStringBuilderRef =
      nir.Type.Ref(genTypeName(nirDefinitions.jlStringBuilderRef))
    private lazy val jlStringBuilderCtor =
      jlStringBuilderRef.name.member(nir.Sig.Ctor(Seq(nir.Type.Int)))
    private lazy val jlStringBuilderCtorSig = nir.Type.Function(
      Seq(jlStringBuilderRef, nir.Type.Int),
      nir.Type.Unit
    )
    private lazy val jlStringBuilderToString =
      jlStringBuilderRef.name.member(
        nir.Sig.Method("toString", Seq(nir.Rt.String))
      )
    private lazy val jlStringBuilderToStringSig = nir.Type.Function(
      Seq(jlStringBuilderRef),
      nir.Rt.String
    )

    private def genStringConcat(tree: Apply): nir.Val = {
      implicit val nirPos: nir.SourcePosition = tree.pos
      liftStringConcat(tree) match {
        // Optimization for expressions of the form "" + x
        case List(Literal(Constant("")), arg) =>
          genApplyStaticMethod(
            nirDefinitions.String_valueOf_Object,
            defn.StringClass,
            Seq(arg)
          )

        case concatenations =>
          val concatArguments = concatenations.view
            .filter {
              // empty strings are no-ops in concatenation
              case Literal(Constant("")) => false
              case _                     => true
            }
            .map {
              // Eliminate boxing of primitive values. Boxing is introduced by erasure because
              // there's only a single synthetic `+` method "added" to the string class.
              case Apply(boxOp, value :: Nil)
                  // TODO: SN specific boxing
                  if currentRun.runDefinitions.isBox(boxOp.symbol) =>
                value
              case other => other
            }
            .toList
          // Estimate capacity needed for the string builder
          val approxBuilderSize = concatArguments.view.map {
            case Literal(Constant(s: String)) => s.length
            case Literal(c @ Constant(_)) if c.isNonUnitAnyVal =>
              String.valueOf(c).length
            case _ => 0
          }.sum

          // new StringBuidler(approxBuilderSize)
          val stringBuilder =
            buf.classalloc(jlStringBuilderRef.name, unwind, None)
          buf.call(
            jlStringBuilderCtorSig,
            nir.Val.Global(jlStringBuilderCtor, nir.Type.Ptr),
            Seq(stringBuilder, nir.Val.Int(approxBuilderSize)),
            unwind
          )
          // concat substrings
          concatArguments.foreach(genStringBuilderAppend(stringBuilder, _))
          // stringBuilder.toString
          buf.call(
            jlStringBuilderToStringSig,
            nir.Val.Global(jlStringBuilderToString, nir.Type.Ptr),
            Seq(stringBuilder),
            unwind
          )
      }
    }

    def genHashCode(argp: Tree)(implicit pos: nir.SourcePosition): nir.Val = {
      genApplyStaticMethod(
        getMemberMethod(RuntimeStaticsModule, nme.anyHash),
        defn.RuntimeStaticsModule,
        Seq(argp)
      )
    }

    def genArrayOp(app: Apply, code: Int): nir.Val = {
      import scalaPrimitives._

      val Apply(Select(arrayp, _), argsp) = app

      val nir.Type.Array(elemty, _) = genType(arrayp.tpe)

      def elemcode = genArrayCode(arrayp.tpe)
      val array = genExpr(arrayp)

      implicit val pos: nir.SourcePosition = app.pos

      if (code == ARRAY_CLONE) {
        val method = RuntimeArrayCloneMethod(elemcode)
        genApplyMethod(method, statically = true, array, argsp)
      } else if (scalaPrimitives.isArrayGet(code)) {
        val idx = genExpr(argsp(0))
        buf.arrayload(elemty, array, idx, unwind)
      } else if (scalaPrimitives.isArraySet(code)) {
        val idx = genExpr(argsp(0))
        val value = genExpr(argsp(1))
        buf.arraystore(elemty, array, idx, value, unwind)
      } else {
        buf.arraylength(array, unwind)
      }
    }

    def boxValue(st: SimpleType, value: nir.Val)(implicit
        pos: nir.SourcePosition
    ): nir.Val =
      st.sym match {
        case UByteClass | UShortClass | UIntClass | ULongClass | USizeClass =>
          genApplyModuleMethod(
            RuntimeBoxesModule,
            BoxUnsignedMethod(st.sym),
            Seq(ValTree(value)())
          )
        case _ =>
          if (genPrimCode(st) == 'O') {
            value
          } else {
            genApplyBox(st, ValTree(value)())
          }
      }

    def unboxValue(st: SimpleType, partial: Boolean, value: nir.Val)(implicit
        pos: nir.SourcePosition
    ): nir.Val = st.sym match {
      case UByteClass | UShortClass | UIntClass | ULongClass | USizeClass =>
        // Results of asInstanceOfs are partially unboxed, meaning
        // that non-standard value types remain to be boxed.
        if (partial) {
          value
        } else {
          genApplyModuleMethod(
            RuntimeBoxesModule,
            UnboxUnsignedMethod(st.sym),
            Seq(ValTree(value)())
          )
        }
      case _ =>
        if (genPrimCode(st) == 'O') {
          value
        } else {
          genApplyUnbox(st, ValTree(value)())
        }
    }

    def genRawPtrOp(app: Apply, code: Int): nir.Val = code match {
      case _ if nirPrimitives.isRawPtrLoadOp(code) =>
        genRawPtrLoadOp(app, code)
      case _ if nirPrimitives.isRawPtrStoreOp(code) =>
        genRawPtrStoreOp(app, code)
      case ELEM_RAW_PTR =>
        genRawPtrElemOp(app, code)
      case _ =>
        abort(
          s"Unknown pointer operation #$code : " + app +
            " at: " + app.pos
        )
    }

    def genRawPtrLoadOp(app: Apply, code: Int): nir.Val = {
      val Apply(_, Seq(ptrp)) = app
      implicit def pos: nir.SourcePosition = app.pos

      val ptr = genExpr(ptrp)

      val ty = code match {
        case LOAD_BOOL     => nir.Type.Bool
        case LOAD_CHAR     => nir.Type.Char
        case LOAD_BYTE     => nir.Type.Byte
        case LOAD_SHORT    => nir.Type.Short
        case LOAD_INT      => nir.Type.Int
        case LOAD_LONG     => nir.Type.Long
        case LOAD_FLOAT    => nir.Type.Float
        case LOAD_DOUBLE   => nir.Type.Double
        case LOAD_RAW_PTR  => nir.Type.Ptr
        case LOAD_RAW_SIZE => nir.Type.Size
        case LOAD_OBJECT   => nir.Rt.Object
      }
      val memoryOrder =
        if (!ptrp.symbol.isVolatile) None
        else Some(nir.MemoryOrder.Acquire)
      buf.load(ty, ptr, unwind, memoryOrder)
    }

    def genRawPtrStoreOp(app: Apply, code: Int): nir.Val = {
      val Apply(_, Seq(ptrp, valuep)) = app
      implicit def pos: nir.SourcePosition = app.pos

      val ptr = genExpr(ptrp)
      val value = genExpr(valuep)

      val ty = code match {
        case STORE_BOOL     => nir.Type.Bool
        case STORE_CHAR     => nir.Type.Char
        case STORE_BYTE     => nir.Type.Byte
        case STORE_SHORT    => nir.Type.Short
        case STORE_INT      => nir.Type.Int
        case STORE_LONG     => nir.Type.Long
        case STORE_FLOAT    => nir.Type.Float
        case STORE_DOUBLE   => nir.Type.Double
        case STORE_RAW_PTR  => nir.Type.Ptr
        case STORE_RAW_SIZE => nir.Type.Size
        case STORE_OBJECT   => nir.Rt.Object
      }
      val memoryOrder =
        if (!ptrp.symbol.isVolatile) None
        else Some(nir.MemoryOrder.Release)
      buf.store(ty, ptr, value, unwind, memoryOrder)
    }

    def genRawPtrElemOp(app: Apply, code: Int): nir.Val = {
      val Apply(_, Seq(ptrp, offsetp)) = app
      implicit def pos: nir.SourcePosition = app.pos

      val ptr = genExpr(ptrp)
      val offset = genExpr(offsetp)

      buf.elem(nir.Type.Byte, ptr, Seq(offset), unwind)
    }

    def genRawPtrCastOp(app: Apply, code: Int): nir.Val = {
      val Apply(_, Seq(argp)) = app
      implicit def pos: nir.SourcePosition = app.pos

      val fromty = genType(argp.tpe)
      val toty = genType(app.tpe)
      val value = genExpr(argp)

      genCastOp(fromty, toty, value)
    }

    def genRawSizeCastOp(app: Apply, receiver: Tree, code: Int): nir.Val = {
      implicit def pos: nir.SourcePosition = app.pos
      val rec = genExpr(receiver)
      val (fromty, toty, conv) = code match {
        case CAST_RAWSIZE_TO_INT =>
          (nir.Type.Size, nir.Type.Int, nir.Conv.SSizeCast)
        case CAST_RAWSIZE_TO_LONG =>
          (nir.Type.Size, nir.Type.Long, nir.Conv.SSizeCast)
        case CAST_RAWSIZE_TO_LONG_UNSIGNED =>
          (nir.Type.Size, nir.Type.Long, nir.Conv.ZSizeCast)
        case CAST_INT_TO_RAWSIZE =>
          (nir.Type.Int, nir.Type.Size, nir.Conv.SSizeCast)
        case CAST_INT_TO_RAWSIZE_UNSIGNED =>
          (nir.Type.Int, nir.Type.Size, nir.Conv.ZSizeCast)
        case CAST_LONG_TO_RAWSIZE =>
          (nir.Type.Long, nir.Type.Size, nir.Conv.SSizeCast)
      }

      buf.conv(conv, toty, rec, unwind)
    }

    def castConv(fromty: nir.Type, toty: nir.Type): Option[nir.Conv] =
      (fromty, toty) match {
        case (_: nir.Type.I, nir.Type.Ptr)       => Some(nir.Conv.Inttoptr)
        case (nir.Type.Ptr, _: nir.Type.I)       => Some(nir.Conv.Ptrtoint)
        case (_: nir.Type.RefKind, nir.Type.Ptr) => Some(nir.Conv.Bitcast)
        case (nir.Type.Ptr, _: nir.Type.RefKind) => Some(nir.Conv.Bitcast)
        case (_: nir.Type.RefKind, _: nir.Type.RefKind) =>
          Some(nir.Conv.Bitcast)
        case (_: nir.Type.RefKind, _: nir.Type.I) => Some(nir.Conv.Ptrtoint)
        case (_: nir.Type.I, _: nir.Type.RefKind) => Some(nir.Conv.Inttoptr)
        case (l: nir.Type.FixedSizeI, r: nir.Type.F) if l.width == r.width =>
          Some(nir.Conv.Bitcast)
        case (l: nir.Type.F, r: nir.Type.FixedSizeI) if l.width == r.width =>
          Some(nir.Conv.Bitcast)
        case _ if fromty == toty               => None
        case (nir.Type.Float, nir.Type.Double) => Some(nir.Conv.Fpext)
        case (nir.Type.Double, nir.Type.Float) => Some(nir.Conv.Fptrunc)
        case _ =>
          unsupported(s"cast from $fromty to $toty")
      }

    /** Generates direct call to function ptr with optional unboxing arguments
     *  and boxing result Apply.args can contain different number of arguments
     *  depending on usage, however they are passed in constant order:
     *    - 0..N args
     *    - return type evidence
     */
    def genCFuncPtrApply(app: Apply, code: Int): nir.Val = {
      val Apply(appRec @ Select(receiverp, _), aargs) = app

      val paramTypes = app.attachments.get[NonErasedTypes] match {
        case None =>
          reporter.error(
            app.pos,
            s"Failed to generate exact NIR types for $app, something is wrong with scala-native internal."
          )
          Nil
        case Some(NonErasedTypes(paramTys)) => paramTys
      }

      implicit val pos: nir.SourcePosition = app.pos

      val self = genExpr(receiverp)
      val retType = genType(paramTypes.last)
      val unboxedRetType = nir.Type.unbox.getOrElse(retType, retType)

      val args = aargs
        .zip(paramTypes)
        .map {
          case (arg, ty) =>
            val tpe = genType(ty)
            val obj = genExpr(arg)

            /* buf.unboxValue does not handle Ref( Ptr | CArray | ... ) unboxing
             * That's why we're doing it directly */
            if (nir.Type.unbox.isDefinedAt(tpe)) {
              buf.unbox(tpe, obj, unwind)(arg.pos, getScopeId)
            } else {
              buf.unboxValue(fromType(ty), partial = false, obj)(arg.pos)
            }
        }
      val argTypes = args.map(_.ty)
      val funcSig = nir.Type.Function(argTypes, unboxedRetType)

      val selfName = genTypeName(CFuncPtrClass)
      val getRawPtrName = selfName
        .member(nir.Sig.Field("rawptr", nir.Sig.Scope.Private(selfName)))

      val target = buf.fieldload(nir.Type.Ptr, self, getRawPtrName, unwind)
      val result = buf.call(funcSig, target, args, unwind)
      if (retType != unboxedRetType)
        buf.box(retType, result, unwind)
      else {
        boxValue(paramTypes.last, result)
      }
    }

    def genCastOp(fromty: nir.Type, toty: nir.Type, value: nir.Val)(implicit
        pos: nir.SourcePosition
    ): nir.Val =
      castConv(fromty, toty).fold(value)(buf.conv(_, toty, value, unwind))

    private lazy val optimizedFunctions = {
      // Included functions should be pure, and should not not narrow the result type
      Set[Symbol](
        CastIntToRawSize,
        CastIntToRawSizeUnsigned,
        CastLongToRawSize,
        CastRawSizeToInt,
        CastRawSizeToLong,
        CastRawSizeToLongUnsigned,
        Size_fromByte,
        Size_fromShort,
        Size_fromInt,
        USize_fromUByte,
        USize_fromUShort,
        USize_fromUInt,
        RuntimePackage_fromRawSize,
        RuntimePackage_fromRawUSize
      ) ++ UnsignedOfMethods ++ RuntimePackage_toRawSizeAlts
    }

    private def getUnboxedSize(
        sizep: Tree
    )(implicit pos: nir.SourcePosition): nir.Val =
      sizep match {
        // Optimize call, strip numeric conversions
        case Literal(Constant(size: Int)) => nir.Val.Size(size)
        case Block(Nil, expr)             => getUnboxedSize(expr)
        case Apply(fun, List(arg))
            if optimizedFunctions.contains(fun.symbol) ||
              arg.symbol.exists && optimizedFunctions.contains(arg.symbol) =>
          getUnboxedSize(arg)
        case Typed(expr, _) => getUnboxedSize(expr)
        case _              =>
          // actual unboxing
          val size = genExpr(sizep)
          val sizeTy = nir.Type.normalize(size.ty)
          val unboxed =
            if (nir.Type.unbox.contains(sizeTy)) buf.unbox(sizeTy, size, unwind)
            else if (nir.Type.box.contains(sizeTy)) size
            else {
              reporter.error(
                sizep.pos,
                s"Invalid usage of Intrinsic.stackalloc, argument is not an integer type: ${sizeTy}"
              )
              nir.Val.Size(0)
            }

          if (unboxed.ty == nir.Type.Size) unboxed
          else buf.conv(nir.Conv.SSizeCast, nir.Type.Size, unboxed, unwind)
      }

    private def genStackalloc(app: Apply): nir.Val = app match {
      case Apply(_, args) => {
        implicit val pos: nir.SourcePosition = app.pos
        val tpe = app.attachments
          .get[NonErasedType]
          .map(v => genType(v.tpe, deconstructValueTypes = true))
          .getOrElse {
            reporter.error(
              app.pos,
              "Not found type attachment for stackalloc operation, report it as a bug."
            )
            nir.Type.Nothing
          }

        val size = args match {
          case Seq()         => nir.Val.Size(1)
          case Seq(sizep)    => getUnboxedSize(sizep)
          case Seq(_, sizep) => getUnboxedSize(sizep)
        }
        buf.stackalloc(tpe, size, unwind)
      }
    }

    def genCQuoteOp(app: Apply): nir.Val = {
      app match {
        // Sometimes I really miss quasiquotes.
        //
        // case q"""
        //   scala.scalanative.unsafe.`package`.CQuote(
        //     new StringContext(scala.this.Predef.wrapRefArray(
        //       Array[String]{${str: String}}.$asInstanceOf[Array[Object]]()
        //     ))
        //   ).c()
        // """ =>
        case Apply(
              Select(
                Apply(
                  _,
                  List(
                    Apply(
                      _,
                      List(
                        Apply(
                          _,
                          List(
                            Apply(
                              TypeApply(
                                Select(
                                  ArrayValue(
                                    _,
                                    List(Literal(Constant(str: String)))
                                  ),
                                  _
                                ),
                                _
                              ),
                              _
                            )
                          )
                        )
                      )
                    )
                  )
                ),
                _
              ),
              _
            ) =>
          val bytes = nir.Val.ByteString(StringUtils.processEscapes(str))
          val const = nir.Val.Const(bytes)
          buf.box(nir.Rt.BoxedPtr, const, unwind)(app.pos, getScopeId)

        case _ =>
          unsupported(app)
      }
    }

    def genUnsignedOp(app: Tree, code: Int): nir.Val = {
      implicit val pos: nir.SourcePosition = app.pos
      app match {
        case Apply(_, Seq(argp)) if code == UNSIGNED_OF =>
          val ty = genType(app.tpe.resultType)
          val arg = genExpr(argp)

          buf.box(ty, arg, unwind)

        case Apply(_, Seq(argp))
            if code >= BYTE_TO_UINT && code <= INT_TO_ULONG =>
          val ty = genType(app.tpe)
          val arg = genExpr(argp)

          buf.conv(nir.Conv.Zext, ty, arg, unwind)

        case Apply(_, Seq(argp))
            if code >= UINT_TO_FLOAT && code <= ULONG_TO_DOUBLE =>
          val ty = genType(app.tpe)
          val arg = genExpr(argp)

          buf.conv(nir.Conv.Uitofp, ty, arg, unwind)

        case Apply(_, Seq(leftp, rightp)) =>
          val bin = code match {
            case DIV_UINT | DIV_ULONG => nir.Bin.Udiv
            case REM_UINT | REM_ULONG => nir.Bin.Urem
          }
          val ty = genType(leftp.tpe)
          val left = genExpr(leftp)
          val right = genExpr(rightp)

          buf.bin(bin, ty, left, right, unwind)
      }
    }

    def genClassFieldRawPtr(
        app: Apply
    )(implicit pos: nir.SourcePosition): nir.Val = {
      val Apply(_, List(target, fieldName: Literal)) = app
      val fieldNameId = fieldName.value.stringValue
      val classInfo = target.tpe.finalResultType
      val classInfoSym = classInfo.typeSymbol.asClass
      def matchesName(f: Symbol) = {
        f.nameString == TermName(fieldNameId).toString()
      }

      val candidates =
        (classInfo.decls ++ classInfoSym.parentSymbols.flatMap(_.info.decls))
          .filter(f => f.isField && matchesName(f))

      candidates.find(!_.isVar).foreach { f =>
        reporter.error(
          app.pos,
          s"Resolving pointer of immutable field ${fieldNameId} in ${f.owner} is not allowed"
        )
      }

      candidates
        .collectFirst {
          case f if matchesName(f) && f.isVariable =>
            // Don't allow to get pointer to immutable field, as it might allow for mutation
            buf.field(genExpr(target), genFieldName(f), unwind)
        }
        .getOrElse {
          reporter.error(
            app.pos,
            s"${classInfoSym} does not contain field ${fieldNameId}"
          )
          nir.Val.Int(-1)
        }

    }

    def genSizeOf(app: Apply)(implicit pos: nir.SourcePosition): nir.Val =
      genLayoutValueOf("sizeOf", buf.sizeOf(_, unwind))(app)

    def genAlignmentOf(app: Apply)(implicit pos: nir.SourcePosition): nir.Val =
      genLayoutValueOf("alignmentOf", buf.alignmentOf(_, unwind))(app)

    // used as internal implementation of sizeOf / alignmentOf
    private def genLayoutValueOf(opType: String, toVal: nir.Type => nir.Val)(
        app: Apply
    )(implicit pos: nir.SourcePosition): nir.Val = {
      def fail(msg: => String) = {
        reporter.error(app.pos, msg)
        nir.Val.Zero(nir.Type.Size)
      }
      app.attachments.get[NonErasedType] match {
        case None =>
          app.args match {
            case Seq(Literal(cls: Constant)) =>
              val nirTpe = genType(cls.typeValue, deconstructValueTypes = false)
              toVal(nirTpe)
            case _ =>
              fail(
                s"Method $opType(Class[_]) requires single class literal argument, if you used $opType[T] report it as a bug"
              )
          }
        case Some(NonErasedType(tpe)) if tpe.sym.isTraitOrInterface =>
          fail(
            s"Type ${tpe} is a trait or interface, $opType cannot be calculated"
          )
        case Some(NonErasedType(tpe)) =>
          try {
            val nirTpe = genType(tpe, deconstructValueTypes = true)
            toVal(nirTpe)
          } catch {
            case ex: Throwable =>
              fail(
                s"Failed to generate exact NIR type of $tpe - ${ex.getMessage}"
              )
          }
      }
    }

    def genSynchronized(receiverp: Tree, bodyp: Tree)(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      genSynchronized(receiverp)(_.genExpr(bodyp))
    }

    def genSynchronized(
        receiverp: Tree
    )(
        bodyGen: ExprBuffer => nir.Val
    )(implicit pos: nir.SourcePosition): nir.Val = {
      // Here we wrap the synchronized call into the try-finally block
      // to ensure that monitor would be released even in case of the exception
      // or in case of non-local returns
      val nested = new ExprBuffer()
      val normaln = fresh()
      val handler = fresh()
      val mergen = fresh()

      // scalanative.runtime.`package`.enterMonitor(receiver)
      genApplyStaticMethod(
        sym = RuntimeEnterMonitorMethod,
        receiver = RuntimePackageClass,
        argsp = List(receiverp)
      )
      // synchronized block
      val retValue = scoped(curUnwindHandler := Some(handler)) {
        nested.label(normaln)
        bodyGen(nested)
      }
      val retty = retValue.ty
      nested.jumpExcludeUnitValue(retty)(mergen, retValue)

      // dummy exception handler,
      // monitor$.exit() call would be added to it in genTryFinally transformer
      locally {
        val excv = nir.Val.Local(fresh(), nir.Rt.Object)
        nested.label(handler, Seq(excv))
        nested.raise(excv, unwind)
        nested.jumpExcludeUnitValue(retty)(mergen, nir.Val.Zero(retty))
      }

      // Append try/catch instructions to the outher instruction buffer.
      buf.jump(nir.Next(normaln))
      buf ++= genTryFinally(
        // scalanative.runtime.`package`.exitMonitor(receiver)
        finallyp = ContTree(receiverp)(
          _.genApplyStaticMethod(
            sym = RuntimeExitMonitorMethod,
            receiver = RuntimePackageClass,
            argsp = List(receiverp)
          )
        ),
        insts = nested.toSeq
      )
      val mergev = nir.Val.Local(fresh(), retty)
      buf.labelExcludeUnitValue(mergen, mergev)
    }

    def genCoercion(app: Apply, receiver: Tree, code: Int): nir.Val = {
      val rec = genExpr(receiver)
      val (fromty, toty) = coercionTypes(code)

      genCoercion(rec, fromty, toty)(app.pos)
    }

    def genCoercion(value: nir.Val, fromty: nir.Type, toty: nir.Type)(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      if (fromty == toty) {
        value
      } else {
        val conv = (fromty, toty) match {
          case (nir.Type.Ptr, _: nir.Type.RefKind) =>
            nir.Conv.Bitcast
          case (_: nir.Type.RefKind, nir.Type.Ptr) =>
            nir.Conv.Bitcast
          case (l: nir.Type.FixedSizeI, r: nir.Type.FixedSizeI) =>
            if (l.width < r.width) {
              if (l.signed) {
                nir.Conv.Sext
              } else {
                nir.Conv.Zext
              }
            } else if (l.width > r.width) {
              nir.Conv.Trunc
            } else {
              nir.Conv.Bitcast
            }
          case (i: nir.Type.I, _: nir.Type.F) if i.signed =>
            nir.Conv.Sitofp
          case (i: nir.Type.I, _: nir.Type.F) if !i.signed =>
            nir.Conv.Uitofp
          case (_: nir.Type.F, i: nir.Type.FixedSizeI) if i.signed =>
            if (i.width < 32) {
              val ivalue = genCoercion(value, fromty, nir.Type.Int)
              return genCoercion(ivalue, nir.Type.Int, toty)
            }
            nir.Conv.Fptosi
          case (_: nir.Type.F, i: nir.Type.FixedSizeI) if !i.signed =>
            if (i.width < 32) {
              val ivalue = genCoercion(value, fromty, nir.Type.Int)
              return genCoercion(ivalue, nir.Type.Int, toty)
            }
            nir.Conv.Fptoui
          case (nir.Type.Double, nir.Type.Float) =>
            nir.Conv.Fptrunc
          case (nir.Type.Float, nir.Type.Double) =>
            nir.Conv.Fpext
        }
        buf.conv(conv, toty, value, unwind)
      }
    }

    def coercionTypes(code: Int): (nir.Type, nir.Type) = {
      import scalaPrimitives._

      code match {
        case B2B => (nir.Type.Byte, nir.Type.Byte)
        case B2S => (nir.Type.Byte, nir.Type.Short)
        case B2C => (nir.Type.Byte, nir.Type.Char)
        case B2I => (nir.Type.Byte, nir.Type.Int)
        case B2L => (nir.Type.Byte, nir.Type.Long)
        case B2F => (nir.Type.Byte, nir.Type.Float)
        case B2D => (nir.Type.Byte, nir.Type.Double)

        case S2B => (nir.Type.Short, nir.Type.Byte)
        case S2S => (nir.Type.Short, nir.Type.Short)
        case S2C => (nir.Type.Short, nir.Type.Char)
        case S2I => (nir.Type.Short, nir.Type.Int)
        case S2L => (nir.Type.Short, nir.Type.Long)
        case S2F => (nir.Type.Short, nir.Type.Float)
        case S2D => (nir.Type.Short, nir.Type.Double)

        case C2B => (nir.Type.Char, nir.Type.Byte)
        case C2S => (nir.Type.Char, nir.Type.Short)
        case C2C => (nir.Type.Char, nir.Type.Char)
        case C2I => (nir.Type.Char, nir.Type.Int)
        case C2L => (nir.Type.Char, nir.Type.Long)
        case C2F => (nir.Type.Char, nir.Type.Float)
        case C2D => (nir.Type.Char, nir.Type.Double)

        case I2B => (nir.Type.Int, nir.Type.Byte)
        case I2S => (nir.Type.Int, nir.Type.Short)
        case I2C => (nir.Type.Int, nir.Type.Char)
        case I2I => (nir.Type.Int, nir.Type.Int)
        case I2L => (nir.Type.Int, nir.Type.Long)
        case I2F => (nir.Type.Int, nir.Type.Float)
        case I2D => (nir.Type.Int, nir.Type.Double)

        case L2B => (nir.Type.Long, nir.Type.Byte)
        case L2S => (nir.Type.Long, nir.Type.Short)
        case L2C => (nir.Type.Long, nir.Type.Char)
        case L2I => (nir.Type.Long, nir.Type.Int)
        case L2L => (nir.Type.Long, nir.Type.Long)
        case L2F => (nir.Type.Long, nir.Type.Float)
        case L2D => (nir.Type.Long, nir.Type.Double)

        case F2B => (nir.Type.Float, nir.Type.Byte)
        case F2S => (nir.Type.Float, nir.Type.Short)
        case F2C => (nir.Type.Float, nir.Type.Char)
        case F2I => (nir.Type.Float, nir.Type.Int)
        case F2L => (nir.Type.Float, nir.Type.Long)
        case F2F => (nir.Type.Float, nir.Type.Float)
        case F2D => (nir.Type.Float, nir.Type.Double)

        case D2B => (nir.Type.Double, nir.Type.Byte)
        case D2S => (nir.Type.Double, nir.Type.Short)
        case D2C => (nir.Type.Double, nir.Type.Char)
        case D2I => (nir.Type.Double, nir.Type.Int)
        case D2L => (nir.Type.Double, nir.Type.Long)
        case D2F => (nir.Type.Double, nir.Type.Float)
        case D2D => (nir.Type.Double, nir.Type.Double)
      }
    }

    def genApplyTypeApply(app: Apply): nir.Val = {
      val Apply(tapp @ TypeApply(fun @ Select(receiverp, _), targs), argsp) =
        app

      val fromty = genType(receiverp.tpe)
      val toty = genType(targs.head.tpe)
      def boxty = genBoxType(targs.head.tpe)
      val value = genExpr(receiverp)
      implicit val pos: nir.SourcePosition =
        tapp.pos.orElse(app.pos).orElse(fallbackSourcePosition)
      lazy val boxed = boxValue(receiverp.tpe, value)(receiverp.pos.orElse(pos))

      fun.symbol match {
        case Object_isInstanceOf =>
          buf.is(boxty, boxed, unwind)

        case Object_asInstanceOf =>
          (fromty, toty) match {
            case (_: nir.Type.PrimitiveKind, _: nir.Type.PrimitiveKind) =>
              genCoercion(value, fromty, toty)
            case _ if boxed.ty =?= boxty => boxed
            case (_, nir.Type.Nothing) =>
              val runtimeNothing = genType(RuntimeNothingClass)
              val isNullL, notNullL = fresh()
              val isNull =
                buf.comp(nir.Comp.Ieq, boxed.ty, boxed, nir.Val.Null, unwind)
              buf.branch(isNull, nir.Next(isNullL), nir.Next(notNullL))
              buf.label(isNullL)
              buf.raise(nir.Val.Null, unwind)
              buf.label(notNullL)
              buf.as(runtimeNothing, boxed, unwind)
              buf.unreachable(unwind)
              buf.label(fresh())
              nir.Val.Zero(nir.Type.Nothing)
            case _ =>
              val cast = buf.as(boxty, boxed, unwind)
              unboxValue(app.tpe, partial = true, cast)(app.pos)
          }

        case Object_synchronized =>
          assert(argsp.size == 1, "synchronized with wrong number of args")
          genSynchronized(ValTree(receiverp)(boxed), argsp.head)
      }
    }

    def genApplyNew(app: Apply): nir.Val = {
      val Apply(fun @ Select(New(tpt), nme.CONSTRUCTOR), args) = app
      implicit val pos: nir.SourcePosition =
        app.pos.orElse(fallbackSourcePosition)

      SimpleType.fromType(tpt.tpe) match {
        case SimpleType(ArrayClass, Seq(targ)) =>
          genApplyNewArray(targ, args)

        case st if st.isStruct =>
          genApplyNewStruct(st, args)

        case SimpleType(cls, Seq()) =>
          genApplyNew(cls, fun.symbol, args)

        case SimpleType(sym, targs) =>
          unsupported(s"unexpected new: $sym with targs $targs")
      }
    }

    def genApplyNewStruct(st: SimpleType, argsp: Seq[Tree]): nir.Val = {
      val ty = genType(st)
      val args = genSimpleArgs(argsp)
      var res: nir.Val = nir.Val.Zero(ty)

      args.zip(argsp).zipWithIndex.foreach {
        case ((arg, argp), idx) =>
          res = buf.insert(res, arg, Seq(idx), unwind)(argp.pos, getScopeId)
      }

      res
    }

    def genApplyNewArray(targ: SimpleType, argsp: Seq[Tree])(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      val Seq(lengthp) = argsp
      val length = genExpr(lengthp)

      buf.arrayalloc(genType(targ), length, unwind)
    }

    def genApplyNew(clssym: Symbol, ctorsym: Symbol, args: List[Tree])(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      val alloc = buf.classalloc(genTypeName(clssym), unwind)
      val call = genApplyMethod(ctorsym, statically = true, alloc, args)
      alloc
    }

    def genApplyModuleMethod(module: Symbol, method: Symbol, args: Seq[Tree])(
        implicit pos: nir.SourcePosition
    ): nir.Val = {
      val self = genModule(module)
      genApplyMethod(method, statically = true, self, args)
    }

    def genApplyMethod(
        sym: Symbol,
        statically: Boolean,
        selfp: Tree,
        argsp: Seq[Tree]
    )(implicit pos: nir.SourcePosition): nir.Val = {
      if (sym.isExtern && sym.isAccessor) {
        genApplyExternAccessor(sym, argsp)
      } else if (sym.isStaticMember) {
        genApplyStaticMethod(sym, selfp.symbol, argsp)
      } else {
        val self = genExpr(selfp)
        genApplyMethod(sym, statically, self, argsp)
      }
    }

    private def genApplyStaticMethod(
        sym: Symbol,
        receiver: Symbol,
        argsp: Seq[Tree]
    )(implicit pos: nir.SourcePosition): nir.Val = {
      require(!sym.isExtern, sym.owner)
      val name = genStaticMemberName(sym, receiver)
      val method = nir.Val.Global(name, nir.Type.Ptr)
      val sig = genMethodSig(sym, statically = true)
      val args = genMethodArgs(sym, argsp)
      buf.call(sig, method, args, unwind)
    }

    def genApplyExternAccessor(sym: Symbol, argsp: Seq[Tree])(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      argsp match {
        case Seq() =>
          val ty = genMethodSig(sym).ret
          val externTy = genExternMethodSig(sym).ret
          genLoadExtern(ty, externTy, sym)
        case Seq(valuep) =>
          val externTy = genExternType(sym.tpe.paramss.flatten.last.tpe)
          genStoreExtern(externTy, sym, genExpr(valuep))
      }
    }

    def genLoadExtern(ty: nir.Type, externTy: nir.Type, sym: Symbol)(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      assert(sym.isExtern, "loadExtern was not extern")

      val name = nir.Val.Global(genName(sym), nir.Type.Ptr)
      val memoryOrder =
        if (!sym.isVolatile) None
        else Some(nir.MemoryOrder.Acquire)

      fromExtern(
        ty,
        buf.load(externTy, name, unwind, memoryOrder)
      )
    }

    def genStoreExtern(externTy: nir.Type, sym: Symbol, value: nir.Val)(implicit
        pos: nir.SourcePosition
    ): nir.Val = {
      assert(sym.isExtern, "storeExtern was not extern")
      val name = nir.Val.Global(genName(sym), nir.Type.Ptr)
      val externValue = toExtern(externTy, value)
      val memoryOrder =
        if (!sym.isVolatile) None
        else Some(nir.MemoryOrder.Release)

      buf.store(externTy, name, externValue, unwind, memoryOrder)
    }

    def toExtern(expectedTy: nir.Type, value: nir.Val)(implicit
        pos: nir.SourcePosition
    ): nir.Val =
      (expectedTy, value.ty) match {
        case (_, refty: nir.Type.Ref)
            if nir.Type.boxClasses.contains(refty.name)
              && nir.Type.unbox(nir.Type.Ref(refty.name)) == expectedTy =>
          buf.unbox(nir.Type.Ref(refty.name), value, unwind)
        case _ =>
          value
      }

    def fromExtern(expectedTy: nir.Type, value: nir.Val)(implicit
        pos: nir.SourcePosition
    ): nir.Val =
      (expectedTy, value.ty) match {
        case (refty: nir.Type.Ref, ty)
            if nir.Type.boxClasses.contains(refty.name)
              && nir.Type.unbox(nir.Type.Ref(refty.name)) == ty =>
          buf.box(nir.Type.Ref(refty.name), value, unwind)
        case _ =>
          value
      }

    def genApplyMethod(
        sym: Symbol,
        statically: Boolean,
        self: nir.Val,
        argsp: Seq[Tree]
    )(implicit pos: nir.SourcePosition): nir.Val = {
      val owner = sym.owner
      val name = genMethodName(sym)
      val origSig = genMethodSig(sym)
      val isExtern = sym.isExtern
      val sig =
        if (isExtern) genExternMethodSig(sym)
        else origSig
      val args = genMethodArgs(sym, argsp)
      val method =
        if (statically || owner.isStruct || isExtern) {
          nir.Val.Global(name, nir.Type.Ptr)
        } else {
          val nir.Global.Member(_, sig) = name
          buf.method(self, sig, unwind)
        }
      val values =
        if (sym.isStaticInNIR) args
        else self +: args

      val res = buf.call(sig, method, values, unwind)

      if (!isExtern) {
        res
      } else {
        val nir.Type.Function(_, retty) = origSig
        fromExtern(retty, res)
      }
    }

    def genMethodArgs(sym: Symbol, argsp: Seq[Tree]): Seq[nir.Val] =
      if (sym.isExtern) genExternMethodArgs(sym, argsp)
      else genSimpleArgs(argsp)

    private def genSimpleArgs(argsp: Seq[Tree]): Seq[nir.Val] =
      argsp.map(genExpr)

    private def genExternMethodArgs(
        sym: Symbol,
        argsp: Seq[Tree]
    ): Seq[nir.Val] = {
      val res = Seq.newBuilder[nir.Val]
      val nir.Type.Function(argTypes, _) = genExternMethodSig(sym)
      val paramSyms = sym.tpe.params
      assert(
        argTypes.size == argsp.size && argTypes.size == paramSyms.size,
        "Different number of arguments passed to method signature and apply method"
      )

      def genArg(
          argp: Tree,
          paramTpe: global.Type,
          isVarArg: Boolean = false
      ): nir.Val = {
        implicit def pos: nir.SourcePosition =
          argp.pos.orElse(fallbackSourcePosition)
        implicit def exprBuf: ExprBuffer = buf
        val rawValue = genExpr(argp)
        val maybeUnboxed =
          if (isVarArg) ensureUnboxed(rawValue, paramTpe.finalResultType)
          else rawValue
        val externType = genExternType(paramTpe.finalResultType)
        val value = (maybeUnboxed, nir.Type.box.get(externType)) match {
          case (value @ nir.Val.Null, Some(unboxedType)) =>
            externType match {
              case nir.Type.Ptr | _: nir.Type.RefKind => value
              case _ =>
                reporter.warning(
                  argp.pos,
                  s"Passing null as argument of type ${paramTpe} to the extern method is unsafe. " +
                    s"The argument would be unboxed to primitive value of type $externType."
                )
                nir.Val.Zero(unboxedType)
            }
          case (value, _) => value
        }
        toExtern(externType, value)
      }

      for (((argp, sigType), paramSym) <- argsp zip argTypes zip paramSyms) {
        sigType match {
          case nir.Type.Vararg =>
            argp match {
              case Apply(_, List(ArrayValue(_, args))) =>
                for (tree <- args) {
                  implicit def pos: nir.SourcePosition =
                    tree.pos.orElse(fallbackSourcePosition)
                  val sym = tree.symbol
                  val tpe =
                    tree.attachments
                      .get[NonErasedType]
                      .map(_.tpe)
                      .getOrElse {
                        if (tree.symbol != null && tree.symbol.exists)
                          tree.symbol.tpe.finalResultType
                        else tree.tpe
                      }
                  val arg = genArg(tree, tpe, isVarArg = true)
                  def isUnsigned = nir.Type.isUnsignedType(genType(tpe))
                  // Decimal varargs needs to be promoted to at least Int, and float needs to be promoted to Double
                  val promotedArg = arg.ty match {
                    case nir.Type.Float =>
                      this.genCastOp(nir.Type.Float, nir.Type.Double, arg)
                    case i: nir.Type.FixedSizeI
                        if i.width < nir.Type.Int.width =>
                      val conv =
                        if (isUnsigned) nir.Conv.Zext
                        else nir.Conv.Sext
                      buf.conv(conv, nir.Type.Int, arg, unwind)
                    case _ => arg
                  }
                  res += promotedArg
                }
              // Scala 2.13 only
              case Select(_, name) if name == defn.NilModule.name => ()
              case _ =>
                reporter.error(
                  argp.pos,
                  "Unable to extract vararg arguments, varargs to extern methods must be passed directly to the applied function"
                )
            }
          case _ => res += genArg(argp, paramSym.tpe)
        }
      }
      res.result()
    }

    private def labelExcludeUnitValue(label: nir.Local, value: nir.Val.Local)(
        implicit pos: nir.SourcePosition
    ): nir.Val =
      value.ty match {
        case nir.Type.Unit =>
          buf.label(label); nir.Val.Unit
        case _ =>
          buf.label(label, Seq(value)); value
      }

    private def jumpExcludeUnitValue(
        mergeType: nir.Type
    )(label: nir.Local, value: nir.Val)(implicit
        pos: nir.SourcePosition
    ): Unit =
      mergeType match {
        case nir.Type.Unit =>
          buf.jump(label, Nil)
        case _ =>
          buf.jump(label, Seq(value))
      }
  }

  object WrapArray {
    private lazy val hasNewCollections =
      !scala.util.Properties.versionNumberString.startsWith("2.12.")

    private val wrapArrayModule =
      if (hasNewCollections) ScalaRunTimeModule
      else PredefModule

    val wrapRefArrayMethod: Symbol =
      getMemberMethod(wrapArrayModule, nme.wrapRefArray)

    val genericWrapArrayMethod: Symbol =
      getMemberMethod(wrapArrayModule, nme.genericWrapArray)

    def isClassTagBasedWrapArrayMethod(sym: Symbol): Boolean =
      sym == wrapRefArrayMethod || sym == genericWrapArrayMethod

    private val isWrapArray: Set[Symbol] = {
      Seq(
        nme.wrapRefArray,
        nme.wrapByteArray,
        nme.wrapShortArray,
        nme.wrapCharArray,
        nme.wrapIntArray,
        nme.wrapLongArray,
        nme.wrapFloatArray,
        nme.wrapDoubleArray,
        nme.wrapBooleanArray,
        nme.wrapUnitArray,
        nme.genericWrapArray
      ).map(getMemberMethod(wrapArrayModule, _)).toSet
    }

    def unapply(tree: Apply): Option[Tree] = tree match {
      case Apply(wrapArray_?, List(wrapped))
          if isWrapArray(wrapArray_?.symbol) =>
        Some(wrapped)
      case _ =>
        None
    }
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy