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

spinal.lib.misc.pipeline.S2MLink.scala Maven / Gradle / Ivy

package spinal.lib.misc.pipeline

import spinal.core._


object S2MLink{
  def apply(up : Node, down : Node) = new S2MLink(up, down)
}

class S2MLink(val up : Node, val down : Node) extends Link {
  down.up = this
  up.down = this

  override def ups: Seq[Node] = List(up)
  override def downs: Seq[Node] = List(down)

  override def propagateDown(): Unit = {
    propagateDownAll()
    if(up.ctrl.valid.nonEmpty) down.valid
    down.ctrl.forgetOneSupported = true
  }
  override def propagateUp(): Unit = {
    propagateUpAll()
    if (down.ctrl.forgetOne.nonEmpty) up.ctrl.forgetOneCreate()
    if (down.ctrl.cancel.nonEmpty) up.cancel
    up.ready
  }

  override def build(): Unit = {
    val matches = down.fromUp.payload.intersect(up.fromDown.payload)

    val rValid = RegInit(False) setWhen (up.isValid) clearWhen (down.ready) setCompositeName(this, "rValid")
    val rData = matches.map(e => RegNextWhen(up(e), up.ready).setCompositeName(this, "s2mBuffer"))

    up.ready := !rValid

    down.valid := up.isValid || rValid
    when(rValid) {
      (matches, rData).zipped.foreach(down(_) := _)
    } otherwise {
      matches.foreach(e => down(e) := up(e))
    }

    down.ctrl.forgetOne.foreach { cond =>
      rValid clearWhen(cond)
      up.ctrl.forgetOne.get := cond && !rValid
    }
    down.ctrl.cancel.foreach { cond =>
      up.ctrl.cancel.get := cond && !rValid
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy