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

spinal.lib.com.spi.xdr.Apb3SpiXdrMasterCtrl.scala Maven / Gradle / Ivy

The newest version!
package spinal.lib.com.spi.ddr

import spinal.core._
import spinal.lib.bus.amba3.apb.{Apb3, Apb3Config, Apb3SlaveFactory}
import spinal.lib.com.spi.ddr.SpiXdrMasterCtrl.{Cmd, Config, Rsp}
import spinal.lib.io.InOutWrapper
import spinal.lib.{Flow, Stream, master, slave}



object Apb3SpiXdrMasterCtrl{
  def getApb3Config = Apb3Config(
    addressWidth = 8,
    dataWidth = 32,
    selWidth = 1,
    useSlaveError = false
  )

  def main(args: Array[String]): Unit = {
    SpinalConfig(mergeAsyncProcess = false).generateVerilog {
      val c = Apb3SpiXdrMasterCtrl(
        SpiXdrMasterCtrl.MemoryMappingParameters(
          SpiXdrMasterCtrl.Parameters(
            dataWidth = 8, // Each transfer will be 8 bits
            timerWidth = 12, // The timer is used to slow down the transmition
            spi = SpiXdrParameter( //Specify the physical SPI interface
              dataWidth = 4, //Number of physical SPI data pins
              ioRate = 1, //Specify the number of transfer that each spi pin can do per clock 1 => SDR, 2 => DDR
              ssWidth = 1 //Number of chip selects
            )
          )
          .addFullDuplex(id = 0) // Add support for regular SPI (MISO / MOSI) using the mode id 0
          .addHalfDuplex( // Add another mode
            id = 1,  // mapped on mode id 1
            rate = 1, // When rate is 1, the clock will do up to one toggle per cycle, divided by the (timer+1)
                      // When rate bigger (ex 2), the controller will ignore the timer, and use the SpiXdrParameter.ioRate
                      // capabilities to emit up to "rate" transition per clock cycle.
            ddr = false, // sdr => 1 bit per SPI clock, DDR => 2 bits per SPI clock
            spiWidth = 4 //Number of physical SPI data pin used for serialisation
          ),
          cmdFifoDepth = 32,
          rspFifoDepth = 32,
          xip = SpiXdrMasterCtrl.XipBusParameters(addressWidth = 24, lengthWidth = 2)
        )
      )
//      c.rework{
//        val sclkShift = in(Bool).setName("clkEarly")
//        c.io.spi.sclk.setAsDirectionLess.unsetName().allowDirectionLessIo
//        c.io.spi.data.setAsDirectionLess.unsetName().allowDirectionLessIo
//        ClockDomain(sclkShift)(master(c.io.spi.sclk.addTag(crossClockDomain).toTriState()).setName("io_spi_sclk"))
//        for((d, idx) <- c.io.spi.data.zipWithIndex)
//          master(d.toTriState()).setName(s"io_spi_data$idx")
//      }
      InOutWrapper(c)
    }
  }
}


case class Apb3SpiXdrMasterCtrl(p : SpiXdrMasterCtrl.MemoryMappingParameters) extends Component{
  val io = new Bundle {
    val apb = slave(Apb3(Apb3SpiXdrMasterCtrl.getApb3Config))
    val xip = ifGen(p.xip != null) (slave(SpiXdrMasterCtrl.XipBus(p.xip)))
    val spi = master(SpiXdrMaster(p.ctrl.spi))
    val interrupt = out Bool()
  }

  val ctrl = SpiXdrMasterCtrl(p.ctrl)
  val mapping = SpiXdrMasterCtrl.driveFrom(ctrl, Apb3SlaveFactory(io.apb, 0))(p)
  if(p.xip != null) io.xip <> mapping.xip.xipBus
  io.spi <> ctrl.io.spi
  io.interrupt <> mapping.interruptCtrl.interrupt
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy