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

spinal.lib.bus.amba4.axis.Axi4Stream.scala Maven / Gradle / Ivy

There is a newer version: 1.10.2a
Show newest version
package spinal.lib.bus.amba4.axis

import spinal.core._
import spinal.lib._

/**
 * Axi4-Stream configuration
 * @param dataWidth Width of the bus in BYTES
 * @param idWidth Width of the ID field in bits
 * @param destWidth Width of the Destination field in bits
 * @param userWidth Width of the User field in bits
 * @param useStrb Use byte strobe bits
 * @param useKeep Use byte keep bits
 * @param useLast Use last bit
 * @param useId Use ID field, must specify idWidth
 * @param useDest Use Destination field, must specify destWidth
 * @param useUser Use User field, must specify userWidth
 */
case class Axi4StreamConfig(dataWidth: Int,
                            idWidth:   Int = -1,
                            destWidth: Int = -1,
                            userWidth: Int = -1,
                            useStrb:   Boolean = false,
                            useKeep:   Boolean = false,
                            useLast:   Boolean = false,
                            useId:     Boolean = false,
                            useDest:   Boolean = false,
                            useUser:   Boolean = false)

object Axi4Stream {

  case class Axi4StreamBundle(val config: Axi4StreamConfig) extends Bundle {
    val data = Bits(8*config.dataWidth bit)
    val id =   (config.useId)   generate UInt(config.idWidth bit)
    val strb = (config.useStrb) generate Bits(config.dataWidth bit)
    val keep = (config.useKeep) generate Bits(config.dataWidth bit)
    val last = (config.useLast) generate Bool()
    val dest = (config.useDest) generate UInt(config.destWidth bit)
    val user = (config.useUser) generate Bits(config.userWidth*config.dataWidth bit)

    def isLast = if (this.last != null) this.last else False

    override def clone: Axi4StreamBundle = Axi4StreamBundle(config)

    override def bundleAssign(that: Bundle)(f: (Data, Data) => Unit): Unit = {
      that match {
        case that : Axi4StreamBundle => {
          assert(that.config.dataWidth <= this.config.dataWidth, s"Axi4Stream $that directly drives stream $this with smaller data width! (${that.config.dataWidth} > ${this.config.dataWidth})")
          if (that.config.useId)
            assert(that.config.idWidth <= this.config.idWidth, s"Axi4Stream $that directly drives stream $this with smaller ID width! (${that.config.idWidth} > ${this.config.idWidth})")
          if (that.config.useDest)
            assert(that.config.destWidth <= this.config.destWidth, s"Axi4Stream $that directly drives stream $this with smaller destination width! (${that.config.destWidth} > ${this.config.destWidth})")
          if (that.config.useUser)
            assert(that.config.userWidth <= this.config.userWidth, s"Axi4Stream $that directly drives stream $this with smaller user width! (${that.config.userWidth} > ${this.config.userWidth})")

          this.data := that.data.resized
          Axi4StreamBundlePriv.driveWeak(that,this,that.id,this.id, () => U(this.id.bitsRange -> false), allowResize = true, allowDrop = false)
          Axi4StreamBundlePriv.driveWeak(that,this,that.strb,this.strb, () => B(this.strb.bitsRange -> true), allowResize = true, allowDrop = false)
          Axi4StreamBundlePriv.driveWeak(that,this,that.keep,this.keep, () => B(this.keep.bitsRange -> true), allowResize = true, allowDrop = false)
          Axi4StreamBundlePriv.driveWeak(that,this,that.last,this.last, () => False, allowResize = false,allowDrop = false)
          Axi4StreamBundlePriv.driveWeak(that,this,that.dest,this.dest, () => B(this.dest.bitsRange -> true), allowResize = true, allowDrop = false)

          (this.user != null, that.user != null) match {
            case (false, false) =>
            case (true, false) => B(this.user.bitsRange -> false)
            case (false, true) => LocatedPendingError(s"${that.user} can't drive $this because the corresponding sink signal does not exist")
            case (true, true) => {
              val sourceSlices = that.user.subdivideIn(that.config.dataWidth slices)
              val sinkSlices = this.user.subdivideIn(this.config.dataWidth slices)
              val sourceSlicesPad = sourceSlices.padTo(sinkSlices.length, null)
              sourceSlicesPad.zip(sinkSlices).foreach(pair => {
                val (sourceSlice, sinkSlice) = pair
                if (sourceSlice != null) {
                  sinkSlice := sourceSlice.resized
                } else
                  sinkSlice := B(sinkSlice.bitsRange -> false)
              })
            }
          }
        }
      }
    }
  }

  private object Axi4StreamBundlePriv {
    def driveWeak[T <: Data](source : Bundle,sink : Bundle, by : T,to : T,defaultValue : () => T,allowResize : Boolean,allowDrop : Boolean) : Unit = {
      (to != null,by != null) match {
        case (false,false) =>
        case (true,false) => if(defaultValue != null) to := defaultValue() else LocatedPendingError(s"$source can't drive $to because the corresponding source signal does not exist")
        case (false,true) => if(!allowDrop) LocatedPendingError(s"$by can't drive $sink because the corresponding sink signal does not exist")
        case (true,true) => to := (if(allowResize) by.resized else by)
      }
    }
  }

  type Axi4Stream = Stream[Axi4StreamBundle]

  def apply(config: Axi4StreamConfig): Stream[Axi4StreamBundle] = Stream(Axi4StreamBundle(config))

  def apply[T <: Data](source: Stream[T]): Axi4Stream = {
    source.payload match {
      case fragment: Fragment[_] =>
        val payloadBytes = (fragment.fragment.getBitsWidth/8.0).ceil.intValue()
        val axisStream = Axi4Stream(Axi4StreamConfig(dataWidth = payloadBytes, useLast = true))
        axisStream.data := fragment.fragment.asBits
        axisStream.arbitrationFrom(source)
        axisStream.last := fragment.last
        axisStream
      case data: Data =>
        val payloadBytes = (data.getBitsWidth/8.0).ceil.intValue()
        val axisStream = Axi4Stream(Axi4StreamConfig(dataWidth = payloadBytes))
        axisStream.data := data.asBits
        axisStream.arbitrationFrom(source)
        axisStream
    }
  }

  implicit class Axi4StreamRich(stream: Stream[Axi4StreamBundle]) {

    def lastFire: Bool = stream.isLast && stream.fire

    /**
      * Converts the Axi4Stream into a Stream of bits.
      * Does not support TSTRB or TKEEP. Will only convert continuous and aligned streams.
      * If TLAST was present it will be dropped. Consider toStreamFragment
      * @return Stream of bits
      */
    def toBitStream(): Stream[Bits] = {
      val that = Stream(Bits(stream.config.dataWidth*8 bit))
      assert(!stream.config.useStrb, s"Can't convert Axi4Stream $this to Stream of ${that.payloadType} because the Axi4Stream supports TSTRB.")
      if (stream.config.useLast)
        SpinalWarning(s"Axi4Stream $this converted to Stream of ${that.payloadType} discards TLAST. Consider using toFragmentStream instead.")

      that.arbitrationFrom(stream)
      that.payload.assignFromBits(stream.data)

      that
    }

    /**
      * Converts the Axi4Stream into a Stream Fragment of bits.
      * Does not support TSTRB or TKEEP. Will only convert continuous and aligned streams.
      * @return Stream of bits
      */
    def toBitStreamFragment(): Stream[Fragment[Bits]] = {
      val that = Stream(Fragment(Bits(stream.config.dataWidth*8 bit)))
      assert(!stream.config.useStrb, s"Can't convert Axi4Stream $this to Stream of ${that.payloadType} because the Axi4Stream supports TSTRB.")
      assert(stream.config.useLast, "Can't convert Axi4Stream $this to Fragment Stream of ${that.payloadType} because the Axi4Stream doesn't support TLAST")

      that.arbitrationFrom(stream)
      that.fragment.assignFromBits(stream.data)
      that.last := stream.last

      that
    }

    /**
      * Results a Flow of TID. To be used with toStream functions.
      * @return Flow of Axi4Stream TID field
      */
    def getIdFlow(): Flow[UInt] = {
      val idFlow = Flow(if (stream.id != null) stream.id else UInt(0 bit))
      idFlow.valid := stream.valid
      idFlow.payload := stream.id
      idFlow
    }

    /**
      * Results a Flow of TDEST. To be used with toStream functions.
      * @return Flow of Axi4Stream TDEST field
      */
    def getDestFlow(): Flow[UInt] = {
      val destFlow = Flow(if (stream.dest != null) stream.dest else UInt(0 bit))
      destFlow.valid := stream.valid
      destFlow.payload := stream.dest
      destFlow
    }

    /**
      * Results a Flow of TUSER. To be used with toStream functions.
      * @return Flow of Axi4Stream TUSER field
      */
    def getUserFlow(): Flow[Bits] = {
      val userFlow = Flow(if (stream.user != null) stream.user else Bits(0 bit))
      userFlow.valid := stream.valid
      userFlow.payload := stream.user
      userFlow
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy