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

spinal.lib.bus.amba3.ahblite.AhbLite3Interconnect.scala Maven / Gradle / Ivy

There is a newer version: 1.10.2a
Show newest version
/*                                                                           *\
**        _____ ____  _____   _____    __                                    **
**       / ___// __ \/  _/ | / /   |  / /   HDL Lib                          **
**       \__ \/ /_/ // //  |/ / /| | / /    (c) Dolu, All rights reserved    **
**      ___/ / ____// // /|  / ___ |/ /___                                   **
**     /____/_/   /___/_/ |_/_/  |_/_____/  MIT Licence                      **
**                                                                           **
** Permission is hereby granted, free of charge, to any person obtaining a   **
** copy of this software and associated documentation files (the "Software"),**
** to deal in the Software without restriction, including without limitation **
** the rights to use, copy, modify, merge, publish, distribute, sublicense,  **
** and/or sell copies of the Software, and to permit persons to whom the     **
** Software is furnished to do so, subject to the following conditions:      **
**                                                                           **
** The above copyright notice and this permission notice shall be included   **
** in all copies or substantial portions of the Software.                    **
**                                                                           **
** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS   **
** OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF                **
** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.    **
** IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY      **
** CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT **
** OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR  **
** THE USE OR OTHER DEALINGS IN THE SOFTWARE.                                **
\*                                                                           */
package spinal.lib.bus.amba3.ahblite

import spinal.core._
import spinal.lib.bus.misc.SizeMapping

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


case class AhbLite3CrossbarSlaveConnection(master: AhbLite3 /*,priority : Int*/)


case class AhbLite3CrossbarSlaveConfig(mapping: SizeMapping, index: Int){
  val masters = ArrayBuffer[AhbLite3CrossbarSlaveConnection]()
}


/**
  * AhbLite3 Crossbar Factory
  *
  * @example {{{
  *     class TopLevel extends Component {
  *         val ahbConfig = AhbLite3Config(addressWidth = 16, dataWidth = 32)
  *
  *         val io = new Bundle{
  *             val ahbMasters = Vec(slave(AhbLite3Master(ahbConfig)), 3)
  *             val ahbSlaves  = Vec(master(AhbLite3(ahbConfig)), 4)
  *         }
  *
  *         val crossbar = AhbLite3CrossbarFactory(ahbConfig)
  *             .addSlaves(
  *                 io.ahbSlaves(0) -> (0x1000,0x1000),
  *                 io.ahbSlaves(1) -> (0x3000,0x1000),
  *                 io.ahbSlaves(2) -> (0x4000,0x1000),
  *                 io.ahbSlaves(3) -> (0x5000,0x1000)
  *              )
  *              .addConnections(
  *                 io.ahbMasters(0).toAhbLite3() -> List(ahbSlaves(0), ahbSlaves(1)),
  *                 io.ahbMasters(1).toAhbLite3() -> List(ahbSlaves(1), ahbSlaves(2), ahbSlaves(3)),
  *                 io.ahbMasters(2).toAhbLite3() -> List(ahbSlaves(0), ahbSlaves(3))
  *              )
  *              // ** OPTIONAL **
  *              //.addGlobalDefaultSlave(io.defaultSalve)
  *              //.addDefaultSalves(
  *              //   io.ahbMaster(0) -> io.defaultSlaveM0,
  *              //   io.ahbMaster(1) -> io.defaultSalveM1
  *              //)
  *              .build()
  *             }
  *          }}}
  */
case class AhbLite3CrossbarFactory(ahbLite3Config: AhbLite3Config){
  var roundRobinArbiter = true
  def noRoundRobinArbiter(): Unit ={
    roundRobinArbiter = false
  }
  val slavesConfigs = mutable.LinkedHashMap[AhbLite3, AhbLite3CrossbarSlaveConfig]()

  private def getNextSlaveIndex = if(slavesConfigs.isEmpty) 0 else (slavesConfigs.values.map(_.index).max + 1)

  def addSlave(ahb: AhbLite3, mapping: SizeMapping): this.type = {
    slavesConfigs(ahb) = AhbLite3CrossbarSlaveConfig(mapping, getNextSlaveIndex)
    this
  }

  def addSlaves(orders: (AhbLite3, SizeMapping)*): this.type = {
    orders.foreach(order => addSlave(order._1, order._2))
    this
  }

  def addConnection(ahb: AhbLite3, ahbLite3Slave: Seq[AhbLite3]): this.type = {
    ahbLite3Slave.foreach(slavesConfigs(_).masters += AhbLite3CrossbarSlaveConnection(ahb))
    this
  }

  def addConnection(order: (AhbLite3, Seq[AhbLite3])): this.type = addConnection(order._1, order._2)

  def addConnections(orders: (AhbLite3, Seq[AhbLite3])*): this.type = {
    orders.foreach(addConnection(_))
    this
  }

  /**
    * Add a global default slaves
    */
  def addGlobalDefaultSlave(slave: AhbLite3): this.type = {
    assert(slavesConfigs.count(_._2.mapping == null) == 0, "AhbLite3CrossbarFactory : default slave(s) has already been added")
    slavesConfigs(slave) = AhbLite3CrossbarSlaveConfig(null, getNextSlaveIndex)
    masters.foreach(m => addConnection(m, List(slave)))
    this
  }

  /**
    * Add a custom default slave for each Master (decoder)
    */
  def addDefaultSlaves(order: (AhbLite3, AhbLite3)*): this.type = {
    assert(slavesConfigs.count(_._2.mapping == null) == 0,  "AhbLite3CrossbarFactory : default slave(s) has already been added")
    order.map(_._2).foreach(slave => slavesConfigs(slave) = AhbLite3CrossbarSlaveConfig(null, getNextSlaveIndex))
    order.foreach(order => addConnection(order._1, List(order._2)))
    this
  }

  /** Get a list of all masters */
  def masters = slavesConfigs.values.map(_.masters.map(_.master)).flatten.toSeq.distinct


  /** Build the crossbar */
  def build() = new Area {

    val masterToDecodedSlave = mutable.LinkedHashMap[AhbLite3, Map[AhbLite3, AhbLite3]]()

    /**
      * Create a decoder for each master
      */
    val decoders = for(master <- masters) yield new Area {

      val slaves = slavesConfigs.filter {
        case (_, config) => config.masters.exists(connection => connection.master == master)
      }.toSeq.sortBy{s => s._2.index}

      val hasDefaultSlave = slaves.map(s => s._2.mapping == null).reduce(_ || _)

      val decoder = new AhbLite3Decoder(
        ahbLite3Config           = ahbLite3Config,
        decodings                = slaves.filter(p => p._2.mapping != null).map(_._2.mapping),
        addDefaultSlaveInterface = hasDefaultSlave
      )

      val outputs = if(hasDefaultSlave) decoder.io.outputs :+ decoder.io.defaultSlave else decoder.io.outputs
      masterToDecodedSlave(master) = (slaves.map(_._1), outputs).zipped.toMap
      decoder.io.input << master

      decoder.setPartialName(master, "decoder")
    }


    /**
      * Create an arbiter for each slaves if needed
      */
    val arbiters = for((slave, config) <- slavesConfigs) yield new Area {

        val arbiter = AhbLite3Arbiter(ahbLite3Config = ahbLite3Config, inputsCount = config.masters.length, roundRobinArbiter)

        for((input, master) <- (arbiter.io.inputs, config.masters).zipped){
          input <> masterToDecodedSlave(master.master)(slave)
        }

        arbiter.io.output >> slave
        arbiter.setPartialName(slave, "arbiter")
    }
  }
}






© 2015 - 2024 Weber Informatics LLC | Privacy Policy