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

spinal.lib.Misc.scala Maven / Gradle / Ivy

The newest version!
package spinal.lib

import spinal.core._

import java.io.File
import java.net.Socket
import scala.collection.mutable
import scala.sys.process.{Process, ProcessLogger}

class BoolPimped(pimped: Bool){

  //Warning, there is no overflow protection
  def genEvent : Event = {
    val e = Event
    val reg = RegInit(False)

    reg := (reg | pimped) & !e.ready
    e.valid := pimped | reg

    e
  }
}

object KeepAttribute{
  object syn_keep_verilog extends AttributeFlag("synthesis syn_keep = 1", COMMENT_ATTRIBUTE){
    override def isLanguageReady(language: Language) : Boolean = language == Language.VERILOG || language == Language.SYSTEM_VERILOG
  }

  object syn_keep_vhdl extends AttributeFlag("syn_keep"){
    override def isLanguageReady(language: Language) : Boolean = language == Language.VHDL
  }
  object keep extends AttributeFlag("keep")

  def apply[T <: Data](that : T) : T = that.addAttribute(keep).addAttribute(syn_keep_verilog).addAttribute(syn_keep_vhdl)

  def apply[T <: Data](that: T, tail : T*) : Unit = {
    apply(that)
    tail.foreach(apply)
  }
  val all = List(keep, syn_keep_verilog ,syn_keep_vhdl)
}


object CheckSocketPort{
  val reserved = mutable.LinkedHashSet[Int]()
  def reserve(port : Int): Unit = {
    while(true) {
      synchronized{
        if(!reserved.contains(port)){
          reserved += port
          return
        }
      }
      Thread.sleep(100) //I know, all of this is dirty
    }
  }
  def release(port : Int): Unit = {
    synchronized{
      reserved -= port
    }
  }
  def apply(port : Int) : Boolean = {
    var s: Socket = null
    try {
      s = new Socket("localhost", port)
      // If the code makes it this far without an exception it means
      // something is using the port and has responded.
      return false
    } catch {
      case e: Throwable =>
        return true
    } finally {
      if (s != null) s.close
    }
  }
}

/**
 * Run command
 */
object DoCmd {
  val isWindows: Boolean = System.getProperty("os.name").toLowerCase().contains("win")

  /**
   * Run command
   * @param cmd command to run
   */
  def doCmd(cmd: String): Unit = {
    println(cmd)
    if (isWindows)
      Process("cmd /C " + cmd) !
    else
      Process(cmd) !
  }

  def startCmd(cmd: String): Process = {
    println(cmd)
    if (isWindows)
      Process("cmd /C " + cmd).run()
    else
      Process(cmd).run()
  }

  /**
   * Run command with custom CWD
   * @param cmd Command to run
   * @param path CWD of new process
   */
  def doCmd(cmd: String, path: String): Unit = {
    println(cmd)
    if (isWindows)
      Process("cmd /C " + cmd, new java.io.File(path)) !
    else
      Process(cmd, new java.io.File(path)) !

  }

  def doCmdWithLog(cmd: String, path: String): String = {
    val stdOut = new StringBuilder()
    class Logger extends ProcessLogger {
      override def err(s: => String): Unit = {
        //if (!s.startsWith("ar: creating ")) println(s)
      }
      override def out(s: => String): Unit = {
        //println(s)
        stdOut ++= s
      }
      override def buffer[T](f: => T) = f
    }
    Process(cmd, new java.io.File(path)).!(new Logger)
    stdOut.toString()
  }

  def doCmd(cmd : Seq[String], location : File) = {
    println(cmd.mkString(" "))
    class Logger extends ProcessLogger {
      override def err(s: => String): Unit = { if(!s.startsWith("ar: creating ")) println(s) }
      override def out(s: => String): Unit = {}
      override def buffer[T](f: => T) = f
    }

    assert(Process(cmd, location).! (new Logger()) == 0)
  }
}


object Repeat{
  def apply[T <: Data](value : T, times : Int) = value #* times
}



object FlowCmdRsp{
  def apply() : FlowCmdRsp[NoData, NoData] = FlowCmdRsp(NoData(), NoData())
  def apply[T <: Data, T2 <: Data](cmdType : HardType[T], rspType : HardType[T2]) : FlowCmdRsp[T, T2] = new FlowCmdRsp(cmdType, rspType)
}

class FlowCmdRsp[T <: Data, T2 <: Data](cmdType : HardType[T], rspType : HardType[T2]) extends Bundle with IMasterSlave {

  val cmd = Flow(cmdType())
  val rsp = Flow(rspType())


  override def asMaster() = {
    master(cmd)
    slave(rsp)
  }

  def setIdleAll(): this.type ={
    cmd.setIdle()
    rsp.setIdle()
    this
  }

  def setIdle(): this.type ={
    cmd.setIdle()
    this
  }

  def isPending(pendingMax : Int) : Bool = pendingMax match{
    case 1 => RegInit(False) setWhen(cmd.valid) clearWhen(rsp.valid)
  }
}


/**
 * Will use the BaseType.clockDomain to figure out how to connect 2 signals together (allowed use StreamCCByToggle)
 */
object DataCc{
  def apply[T <: BaseType](to : T, from : T): Unit = {
    ClockDomain.areSynchronous(to.clockDomain, from.clockDomain) match {
      case true => to := from
      case false => {
        to := signalCache((from, to.clockDomain, "DataCc")) {
          val cc = new StreamCCByToggle(to, from.clockDomain, to.clockDomain).setCompositeName(to, "cc_driver")
          cc.io.input.valid := True
          cc.io.input.payload := from
          cc.io.output.ready := True
          cc.io.output.payload
        }
      }
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy