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

spinal.lib.bus.bmb.sim.BmbRegionAllocator.scala Maven / Gradle / Ivy

The newest version!
package spinal.lib.bus.bmb.sim

import spinal.lib.bus.bmb.{Bmb, BmbParameter}
import spinal.lib.bus.misc.SizeMapping

import scala.collection.mutable
import scala.util.Random

case class BmbRegionAllocator(alignmentMinWidth : Int = 0){
  val allocations = mutable.HashSet[SizeMapping]()

  def align(v : Int) = v & ~((1 << alignmentMinWidth) -1)
  def free(region : SizeMapping) = allocations.remove(region)
  def allocate(addressGen : => Int, sizeMax : Int, p : BmbParameter, sizeMin : Int = 1, checkAvailability : Boolean = true) : SizeMapping = {
    val sizeMinAligned = Math.max(sizeMin, 1 << alignmentMinWidth)
    var tryies = 0
    while(tryies < 10){
      var address = align(addressGen)
      val boundaryMax = Bmb.boundarySize - (address & (Bmb.boundarySize-1))
      var size = Math.max(sizeMinAligned, align(Math.min(boundaryMax, Random.nextInt(sizeMax) + 1)))
      if(!p.access.aggregated.alignment.allowByte) {
        address &= ~p.access.wordMask
        if(size > p.access.byteCount){
          size = (size + p.access.byteCount - 1) & ~p.access.wordMask
        }
      }
      if(!p.access.aggregated.alignment.allowWord){
        address &= ~((1 << spinal.core.log2Up(size))-1)
      }
      val region = SizeMapping(address, size)
      if(!checkAvailability || allocations.forall(r => r.base > region.end || r.end < region.base) && size <= boundaryMax) {
        allocations += region
        return region
      }
      tryies += 1
    }
    return null
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy