
spinal.lib.bus.amba4.axi.sim.Axi4Master.scala Maven / Gradle / Ivy
package spinal.lib.bus.amba4.axi.sim
import spinal.core._
import spinal.core.sim._
import spinal.lib._
import spinal.lib.bus.amba4.axi._
import spinal.lib.sim._
import scala.collection.mutable
object Axi4Bursts extends Enumeration {
type Axi4Burst = Value
val Fixed, Incr, Wrap, Reserved = Value
}
object Axi4Resps extends Enumeration {
type Axi4Resp = Value
val Okay, ExOkay, SlvErr, DecErr = Value
}
import Axi4Bursts._
import Axi4Resps._
/**
* Simulation master for the Axi4 bus protocol [[spinal.lib.bus.amba4.axi.Axi4]].
*
* @constructor create a new simulation master with the given bus instance and clock domain.
* @param axi bus master to drive
* @param clockDomain clock domain to sample data on
* @example
* {{{
* SimConfig.compile(new Component {
* val io = new Bundle {
* val axiSlave = slave(Axi4(Axi4Config(32, 32)))
* }
* io.axiSlave.assignDontCare
* }).doSim("sample") { dut =>
* val master = Axi4Master(dut.io.axiSlave, dut.clockDomain)
* val data = master.read(0x1000, 4)
* }
* }}}
*/
case class Axi4Master(axi: Axi4, clockDomain: ClockDomain, name: String = "unnamed") {
private val busConfig = axi.config
private val arQueue = mutable.Queue[Axi4Ar => Unit]()
private val awQueue = mutable.Queue[Axi4Aw => Unit]()
private val idCount = if (busConfig.useId) (1 << busConfig.idWidth) else 1
private val rQueue = Array.fill(idCount)(mutable.Queue[Axi4R => Unit]())
private val wQueue = mutable.Queue[Axi4W => Unit]()
private val bQueue = Array.fill(idCount)(mutable.Queue[Axi4B => Unit]())
/** check if all read channels are idle (no read transactions active) */
def readIdle = arQueue.isEmpty && rQueue.map(_.isEmpty).reduce(_ && _)
/** check if all write channels are idle (no write transactions active) */
def writeIdle = awQueue.isEmpty && wQueue.isEmpty && bQueue.map(_.isEmpty).reduce(_ && _)
/** check if bus master is idle (readIdle && writeIdle) */
def idle = readIdle && writeIdle
private val maxSize = log2Up(busConfig.bytePerWord)
private def log(chan: String, msg: String): Unit = {
println(s"Axi4Master ($name) [$chan]\t: $msg")
}
/**
* Read synchronously multiple bytes from the specified address.
*
* @param address address to read from; does not need to be aligned (data will be truncated automatically)
* @param totalBytes total number of bytes in the result;
* if longer than a single transaction (as specified by `burst`, `len`, and `size`),
* the bus master will issue multiple transactions
* @param id AxID to use in the request; xID in the response will be checked against this (cf. AXI specification)
* @param burst burst mode to issue (cf. AXI specification)
* @param len number of beats in a single transaction minus one (cf. AXI specification)
* @param size number of bytes in one beat, log encoded (cf. AXI specification)
*/
def read(address: BigInt, totalBytes: BigInt, id: Int = 0, burst: Axi4Burst = Incr, len: Int = 0, size: Int = maxSize) = {
var result: List[Byte] = null
val mtx = SimMutex().lock()
readCB(address, totalBytes, id, burst, len, size) { data =>
result = data
mtx.unlock()
}
mtx.await()
result
}
/** Read asynchronously; same as {@link read}, but result is delivered in a callback
*
* @param callback callback function on finish
*/
def readCB(address: BigInt, totalBytes: BigInt, id: Int = 0, burst: Axi4Burst = Incr, len: Int = 0, size: Int = maxSize)(callback: List[Byte] => Unit): Unit = {
val bytePerBeat = 1 << size
val bytes = (len + 1) * bytePerBeat // FIXME: 4K limitation?
val numTransactions = (totalBytes.toDouble / bytes).ceil.toInt
val builder = new mutable.ArrayBuilder.ofByte
numTransactions match {
case 0 => callback(List()); return
case x if x > 1 => log("..", f"read $address%#x in $x transactions")
case _ =>
}
def run(addr: BigInt, totBytes: Int, numTransactions: Int) = {
val transactionSize = if (totBytes > bytes) bytes else totBytes
readSingle(addr, transactionSize, id, burst, len, size)(handleTransaction(addr, totBytes, numTransactions))
}
def handleTransaction(addr: BigInt, tot: Int, numTransactions: Int)(data: List[Byte]): Unit = {
builder ++= data
if (numTransactions == 1) {
// we are the last one
callback(builder.result().toList)
} else {
run(addr + data.length, tot - data.length, numTransactions - 1)
}
}
run(address, totalBytes.toInt, numTransactions)
}
/** Read asynchronously with only one transaction */
def readSingle(address: BigInt, totalBytes: Int, id: Int = 0, burst: Axi4Burst = Incr, len: Int = 0, size: Int = maxSize)(callback: List[Byte] => Unit): Unit = {
assert(size <= maxSize, s"requested beat size too big: $size vs $maxSize")
if (burst != Incr) {
assert(len <= 15, s"max fixed/wrap burst in one transaction is 16")
}
assert(len <= 255, s"max burst in one transaction is 256")
val bytePerBeat = 1 << size
val bytes = (len + 1) * bytePerBeat // FIXME: 4K limitation?
assert(totalBytes <= bytes, s"requested length $totalBytes could not be completed in one transaction")
val bytePerBus = 1 << log2Up(busConfig.dataWidth / 8)
val roundedAddress = address - (address & (busConfig.bytePerWord - 1))
val dropFront = (address - roundedAddress).toInt
arQueue += { ar =>
ar.addr #= address
if (busConfig.useId) ar.id #= id
if (busConfig.useBurst) ar.burst #= burst.id
if (busConfig.useLen) ar.len #= len
if (busConfig.useSize) ar.size #= size
log("AR", f"addr $address%#x size $size len $len burst $burst")
val builder = new mutable.ArrayBuilder.ofByte
for (beat <- 0 to len) {
rQueue(id) += { r =>
if (busConfig.useLast) assert(r.last.toBoolean == (beat == len), "bad last beat")
if (busConfig.useResp) assert(r.resp.toInt == Okay.id, s"bad resp ${r.resp.toInt}")
val data = r.data.toBigInt
val beatAddress = burst match {
case Fixed => address
case Incr => address + bytePerBeat * beat
case Wrap =>
val base = address & ~BigInt(bytes - 1)
base + ((address + bytePerBeat * beat) & BigInt(bytes - 1))
}
val accessAddress = beatAddress & ~BigInt(busConfig.bytePerWord - 1)
val start = ((beatAddress & ~BigInt(bytePerBeat - 1)) - accessAddress).toInt
val end = start + bytePerBeat
for (i <- 0 until bytePerBus) {
val _byte = ((data >> (8 * i)).toInt & 0xFF).toByte
if (start <= i && i < end) {
builder += _byte
}
}
if (beat == len) {
val response = builder.result().slice(dropFront, dropFront + totalBytes).toList
log("R", f"got data ${response.bytesToHex}")
callback(response)
}
}
}
}
}
private val arDriver = StreamDriver(axi.ar, clockDomain) { ar =>
if (arQueue.isEmpty) false else {
arQueue.dequeue()(ar)
true
}
}
StreamReadyRandomizer(axi.r, clockDomain)
StreamMonitor(axi.r, clockDomain) { r =>
val id = if (busConfig.useId) r.id.toInt else 0
if (rQueue(id).nonEmpty) {
rQueue(id).dequeue()(r)
}
}
private def padData(address: BigInt, data: List[Byte]) = {
val roundedAddress = address - (address & (busConfig.bytePerWord - 1))
val padFront = (address - roundedAddress).toInt
val totalLen = roundUp(padFront + data.length, busConfig.bytePerWord).toInt
val paddedData = (List.fill(padFront)(0.toByte) ++ data).padTo(totalLen, 0.toByte)
val padBack = totalLen - padFront - data.length
(roundedAddress, padFront, padBack, paddedData)
}
/**
* Write synchronously multiple bytes to the specified address.
*
* @param address address to write to; does not need to be aligned (data will be truncated automatically)
* @param data list of bytes to write to address;
* if longer than a single transaction (as specified by `burst`, `len`, and `size`),
* the bus master will issue multiple transactions
* @param id AxID to use in the request; xID in the response will be checked against this (cf. AXI specification)
* @param burst burst mode to issue (cf. AXI specification)
* @param len number of beats in a single transaction minus one (cf. AXI specification)
* @param size number of bytes in one beat, log encoded (cf. AXI specification)
*/
def write(address: BigInt, data: List[Byte], id: Int = 0, burst: Axi4Burst = Incr, len: Int = 0, size: Int = maxSize): Unit = {
val mtx = SimMutex().lock()
writeCB(address, data, id, burst, len, size) {
mtx.unlock()
}
mtx.await()
}
/** Write asynchronously; same as {@link write}, but completion is delivered in a callback
*
* @param callback callback function on finish
*/
def writeCB(address: BigInt, data: List[Byte], id: Int = 0, burst: Axi4Burst = Incr, len: Int = 0, size: Int = maxSize)(callback: => Unit): Unit = {
val bytePerBeat = 1 << size
val bytes = (len + 1) * bytePerBeat
val (_, padFront, _, paddedData) = padData(address, data)
val numTransactions = paddedData.length / bytes
if (numTransactions > 1) {
log("..", f"write $address%#x in $numTransactions transactions")
}
def run(addr: BigInt, data: List[Byte], transactionId: Int): Unit = {
val slice = transactionId match {
case 0 => data.take(bytes - padFront)
case tid if tid == numTransactions - 1 => data
case _ => data.take(bytes)
}
val remaining = data.drop(slice.length)
writeSingle(addr, slice, id, burst, len, size)(handleTransaction(addr, transactionId, remaining))
}
def handleTransaction(addr: BigInt, transactionId: Int, remaining: List[Byte])() = {
if (transactionId == numTransactions - 1) {
// we are the last one
assert(remaining.isEmpty, s"left over ${remaining.length} bytes unsent!")
callback
} else {
val addrInc = if (transactionId == 0) bytes - padFront else bytes
run(addr + addrInc, remaining, transactionId + 1)
}
}
run(address, data, 0)
}
/** Write asynchronously with only one transaction */
def writeSingle(address: BigInt, data: List[Byte], id: Int = 0, burst: Axi4Burst = Incr, len: Int = 0, size: Int = maxSize)(callback: => Unit): Unit = {
assert(size <= maxSize, s"requested beat size too big: $size vs $maxSize")
if (burst != Incr) {
assert(len <= 15, s"max fixed/wrap burst in one transaction is 16")
}
assert(len <= 255, s"max burst in one transaction is 256")
val bytePerBeat = 1 << size
val bytes = (len + 1) * bytePerBeat
val bytePerBus = 1 << log2Up(busConfig.dataWidth / 8)
val (roundedAddress, padFront, padBack, paddedData) = padData(address, data)
val realLen = data.length
assert(paddedData.length <= bytes, s"requested length ${data.length} (${paddedData.length} with padding) could not be completed in one transaction")
awQueue += { aw =>
aw.addr #= roundedAddress
if (busConfig.useId) aw.id #= id
if (busConfig.useLen) aw.len #= len
if (busConfig.useSize) aw.size #= size
if (busConfig.useBurst) aw.burst #= burst.id
log("AW", f"addr $roundedAddress%#x size $size len $len burst $burst")
for (beat <- 0 to len) {
wQueue += { w =>
val data = paddedData.slice(beat * bytePerBeat, (beat + 1) * bytePerBeat)
w.data #= data.toArray
val strb = if (len == 0) {
((BigInt(1) << realLen) - 1) << padFront
} else beat match {
case 0 => (BigInt(1) << (bytePerBeat - padFront)) - 1
case `len` => ~((BigInt(1) << padBack) - 1)
case _ => BigInt(1) << busConfig.bytePerWord
}
if (busConfig.useStrb) w.strb #= strb
if (busConfig.useLast) w.last #= beat == len
log("W", f"data ${data.bytesToHex} strb $strb%#x last ${beat == len}")
}
}
bQueue(id) += { b =>
if (busConfig.useResp) assert(b.resp.toInt == Okay.id, s"bad resp ${b.resp.toInt}")
log("B", s"transaction finished resp ${b.resp.toInt}")
callback
}
}
}
private val awDriver = StreamDriver(axi.aw, clockDomain) { aw =>
if (awQueue.isEmpty) false else {
awQueue.dequeue()(aw)
true
}
}
private val wDriver = StreamDriver(axi.w, clockDomain) { w =>
if (wQueue.isEmpty) false else {
wQueue.dequeue()(w)
true
}
}
StreamReadyRandomizer(axi.b, clockDomain)
StreamMonitor(axi.b, clockDomain) { b =>
val id = if (busConfig.useId) b.id.toInt else 0
if (bQueue(id).nonEmpty) {
bQueue(id).dequeue()(b)
}
}
/** Reset bus master (dropping all pending transactions) */
def reset(): Unit = {
arQueue.clear
rQueue.foreach(_.clear)
awQueue.clear
wQueue.clear
bQueue.foreach(_.clear)
arDriver.reset
awDriver.reset
wDriver.reset
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy