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

spinal.lib.bus.bsb.BsbInterconnectGenerator.scala Maven / Gradle / Ivy

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

import spinal.core._
import spinal.core.fiber._
import spinal.lib._
import spinal.lib.generator._

import scala.collection.mutable
import scala.collection.mutable.ArrayBuffer


//case class BspParameterOption(byteCount : Option[Int],
//                         sourceWidth : Option[Int],
//                         sinkWidth : Option[Int],
//                         withMask : Option[Boolean])

case class BsbInterconnectGenerator() extends Area{
  val masters = mutable.LinkedHashMap[Handle[Bsb], MasterModel]()
  val slaves = mutable.LinkedHashMap[Handle[Bsb], SlaveModel]()

  def getMaster(key : Handle[Bsb]) = masters.getOrElseUpdate(key, MasterModel(key).setCompositeName(key, "masterModel"))
  def getSlave(key : Handle[Bsb]) = slaves.getOrElseUpdate(key, SlaveModel(key).setCompositeName(key, "slaveModel"))

  case class MasterModel(bsb : Handle[Bsb]) extends Area {
    val generatorClockDomain = ClockDomain.currentHandle
    val byteCount = Handle[Int]
    val sourceWidth = Handle[Int]
    val sinkWidth = Handle[Int]
    val withMask = Handle[Boolean]
    val connections = ArrayBuffer[ConnectionModel]()

    sinkWidth.loadAsync(connections.map(_.s.sinkWidth.get).max + log2Up(connections.size))
  }

  case class SlaveModel(bsb : Handle[Bsb]) extends Area {
    val generatorClockDomain = ClockDomain.currentHandle
    val byteCount = Handle[Int]
    val sourceWidth = Handle[Int]
    val sinkWidth = Handle[Int]
    val withMask = Handle[Boolean]

    val connections = ArrayBuffer[ConnectionModel]()

    byteCount.loadAsync(connections.map(_.m.byteCount.get).max)
    sourceWidth.loadAsync(connections.map(_.m.sourceWidth.get).max + log2Up(connections.size))
    withMask.loadAsync(connections.map(_.m.withMask.get).reduce(_ || _))
  }

  case class ConnectionModel(m : MasterModel, s : SlaveModel) extends Area {
    m.connections += this
    s.connections += this

    val rtl = Handle{
      if(m.generatorClockDomain.get != s.generatorClockDomain.get) { //TODO better sync check
        m.bsb.queue(128, m.generatorClockDomain, s.generatorClockDomain) >> s.bsb
      } else {
        m.bsb >> s.bsb
      }
    }
  }

  def addMaster(bsb : Handle[Bsb]/*,
                byteCount : Handle[Int],
                sourceWidth : Handle[Int],
                sinkWidth : Handle[Int],
                withMask : Handle[Boolean]*/): MasterModel ={
    getMaster(bsb)
  }

  def addSlave(bsb : Handle[Bsb]/*,
               byteCount : Handle[Int],
               sourceWidth : Handle[Int],
               sinkWidth : Handle[Int],
               withMask : Handle[Boolean]*/): SlaveModel ={

    getSlave(bsb)
  }

  def connect(m : Handle[Bsb], s : Handle[Bsb], id : Handle[Int] = Handle(0)): Unit ={
    val c = new ConnectionModel(getMaster(m), getSlave(s))
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy