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

spinal.core.Literal.scala Maven / Gradle / Ivy

The newest version!
/*                                                                           *\
**        _____ ____  _____   _____    __                                    **
**       / ___// __ \/  _/ | / /   |  / /   HDL Core                         **
**       \__ \/ /_/ // //  |/ / /| | / /    (c) Dolu, All rights reserved    **
**      ___/ / ____// // /|  / ___ |/ /___                                   **
**     /____/_/   /___/_/ |_/_/  |_/_____/                                   **
**                                                                           **
**      This library is free software; you can redistribute it and/or        **
**    modify it under the terms of the GNU Lesser General Public             **
**    License as published by the Free Software Foundation; either           **
**    version 3.0 of the License, or (at your option) any later version.     **
**                                                                           **
**      This library is distributed in the hope that it will be useful,      **
**    but WITHOUT ANY WARRANTY; without even the implied warranty of         **
**    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU      **
**    Lesser General Public License for more details.                        **
**                                                                           **
**      You should have received a copy of the GNU Lesser General Public     **
**    License along with this library.                                       **
\*                                                                           */
package spinal.core

import spinal.core.internals._
import scala.collection.Seq

/**
  * Base class to create Bit Vector from literal
  */
abstract class BitVectorLiteralFactory[T <: BitVector] {

  def apply(): T

  def apply(value: Int): T = this(BigInt(value))
  def apply(value: Int, width: BitCount): T = this(BigInt(value),width)
  def apply(value: Long): T = this(BigInt(value))
  def apply(value: Long, width: BitCount): T = this(BigInt(value),width)
  def apply(value: BigInt): T = getFactory(value, -1, this().setAsTypeNode())
  def apply(value: BigInt, width: BitCount): T = getFactory(value, width.value, this().setWidth(width.value).setAsTypeNode())
  def apply(value: String): T = bitVectorStringParser(this, value,  isSigned)

  def apply(bitCount: BitCount, rangesValue: (Any, Any), _rangesValues: (Any, Any)*): T = this.aggregate(bitCount, rangesValue +: _rangesValues)

  def apply(rangesValue: (Any, Any), _rangesValues: (Any ,Any)*): T = {
    val rangesValues = rangesValue +: _rangesValues
    var hig = -1

    rangesValues.foreach {
      case (pos: Int, _) =>
        hig = Math.max(hig, pos)
      case (range: Range, _) =>
        hig = Math.max(hig, range.high)
      case (`default`, _) =>
        val trace = ScalaLocated.long
        GlobalData.get.pendingErrors += (() => s"UNSUPPORTED DEFAULT. You can't use the default statement to create a literal without explicitly specifying how long this literal should be. Example : U(10 bits, 4 -> false, default -> true)\n$trace")
      case _ =>
    }

    this.aggregate(hig + 1 bit, rangesValues)
  }

  def getFactory: (BigInt, Int, T) => T

  def isSigned: Boolean

  private[core] def newInstance(bitCount: BitCount) : T

  private def aggregate(bitCount: BitCount, rangesValues: Seq[(Any, Any)]): T = {
    val ret: T = this.newInstance(bitCount.value bit)
    applyTuples(ret, rangesValues)
    ret
  }

  def applyTuples(on: T, rangesValues: Seq[(Any, Any)]): Unit = {
    //Apply default
    rangesValues.foreach {
      case (`default`, value: Boolean) => on.setAllTo(value)
      case (`default`, value: Bool) => on.setAllTo(value)
      case _ =>
    }

    //Apply specific access in order
    for(e <- rangesValues) e match{
      case (pos: Int, value: Boolean)   => on(pos) := Bool(value)
      case (pos: Int, value: Bool)      => on(pos) := value //literal extraction
      case (range: Range, value: Bool) =>
        for(i <- range)
          on(i) := value
      case (range: Range, value: Boolean) => on(range) := apply(
        if(! value)
          BigInt(0)
        else {
          if(!on.isInstanceOf[SInt])
            (BigInt(1) << range.size) - 1
          else
            BigInt(-1)
        }
      )
      case (range: Range, value: String) => on(range) := apply(value)
      case (range: Range, value: BitVector) if value.getClass == on.getClass => on(range) := value.asInstanceOf[T]
      case (`default`, value: Boolean) =>
      case (`default`, value: Bool)    =>
    }
  }
}


/**
  * Used to create a new Bits or cast to Bits
  */
object B extends BitVectorLiteralFactory[Bits] {
  def apply(): Bits = new Bits()
  def apply(that: Data): Bits = that.asBits
  def apply(that: Data, width : BitCount): Bits = that.asBits.resize(width)
  def apply(head: Data, tail: Data*) : Bits = Cat((head +: tail).reverse)
  def apply(value: Data, times: Int) : Bits = value #* times
  def apply(value : Seq[Data]) : Bits = Cat(value)
  def apply[T <: Data](value : Vec[T]) : Bits = B(value.asInstanceOf[Data])
  def apply(value : MaskedLiteral, filling : Boolean = false): Bits = value.asBits(filling)

  override private[core] def newInstance(bitCount: BitCount): Bits = Bits(bitCount)
  override def isSigned: Boolean = false

  override def getFactory: (BigInt, Int, Bits) => Bits = {
    BitsLiteral.apply[Bits]
  }
}


/**
  * Used to create a new UInt or cast to UInt
  */
object U extends BitVectorLiteralFactory[UInt] {
  def apply(): UInt = new UInt()
  def apply(that: Bool): UInt = that.asUInt
  def apply(that: Bits): UInt = that.asUInt
  def apply(that: SInt): UInt = that.asUInt
  def apply(that: UFix): UInt = that.toUInt
  def apply(that: AFix): UInt = that.asUInt()

  def apply(that: Bool, width : BitCount): UInt = that.asUInt.resize(width)
  def apply(that: Bits, width : BitCount): UInt = that.asUInt.resize(width)
  def apply(that: SInt, width : BitCount): UInt = that.asUInt.resize(width)

  def apply(value : MaskedLiteral, filling : Boolean = false): UInt = value.asUInt(filling)

  override private[core] def newInstance(bitCount: BitCount): UInt = UInt(bitCount)
  override def isSigned: Boolean = false
  override def getFactory: (BigInt, Int, UInt) => UInt = UIntLiteral.apply[UInt]
}


/**
  * Used to create a new SInt or cast to SInt
  */
object S extends BitVectorLiteralFactory[SInt] {
  def apply(): SInt = new SInt()
  def apply(that: Bool): SInt = that.asSInt
  def apply(that: Bits): SInt = that.asSInt
  def apply(that: UInt): SInt = that.asSInt
  def apply(that: SFix): SInt = that.toSInt

  def apply(that: Bool, width : BitCount): SInt = that.asSInt.resize(width)
  def apply(that: Bits, width : BitCount): SInt = that.asSInt.resize(width)
  def apply(that: UInt, width : BitCount): SInt = that.asSInt.resize(width)

  def apply(value : MaskedLiteral, filling : Boolean = false): SInt = value.asSInt(filling)

  override private[core] def newInstance(bitCount: BitCount): SInt = SInt(bitCount)
  override def isSigned: Boolean = true
  override def getFactory: (BigInt, Int, SInt) => SInt = SIntLiteral.apply[SInt]
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy