diff options
| author | jackbackrack | 2015-04-27 15:58:20 -0700 |
|---|---|---|
| committer | jackbackrack | 2015-04-27 15:58:20 -0700 |
| commit | 136352557ba0c6b9c97621e9dfefb4009faef7cc (patch) | |
| tree | 0d4b04d90e5efd95b04e02bfd7c8c5e1cc210b8c /src/main/scala/utils.scala | |
first commit
Diffstat (limited to 'src/main/scala/utils.scala')
| -rw-r--r-- | src/main/scala/utils.scala | 253 |
1 files changed, 253 insertions, 0 deletions
diff --git a/src/main/scala/utils.scala b/src/main/scala/utils.scala new file mode 100644 index 00000000..346ce657 --- /dev/null +++ b/src/main/scala/utils.scala @@ -0,0 +1,253 @@ +package Chisel +import Builder._ +import scala.math._ + +object log2Up +{ + def apply(in: Int): Int = if(in == 1) 1 else ceil(log(in)/log(2)).toInt +} + +object log2Ceil +{ + def apply(in: Int): Int = ceil(log(in)/log(2)).toInt +} + + +object log2Down +{ + def apply(x : Int): Int = if (x == 1) 1 else floor(log(x)/log(2.0)).toInt +} + +object log2Floor +{ + def apply(x : Int): Int = floor(log(x)/log(2.0)).toInt +} + + +object isPow2 +{ + def apply(in: Int): Boolean = in > 0 && ((in & (in-1)) == 0) +} + +object FillInterleaved +{ + def apply(n: Int, in: Bits): Bits = apply(n, in.toBools) + def apply(n: Int, in: Seq[Bool]): Bits = Vec(in.map(Fill(n, _))).toBits +} + +/** Returns the number of bits set (i.e value is 1) in the input signal. + */ +object PopCount +{ + def apply(in: Iterable[Bool]): UInt = { + if (in.size == 0) { + UInt(0) + } else if (in.size == 1) { + in.head + } else { + apply(in.slice(0, in.size/2)) + Cat(UInt(0), apply(in.slice(in.size/2, in.size))) + } + } + def apply(in: Bits): UInt = apply((0 until in.getWidth).map(in(_))) +} + +object RegNext { + + def apply[T <: Data](next: T): T = Reg[T](next, next, null.asInstanceOf[T]) + + def apply[T <: Data](next: T, init: T): T = Reg[T](next, next, init) + +} + +object RegInit { + + def apply[T <: Data](init: T): T = Reg[T](init, null.asInstanceOf[T], init) + +} + +object RegEnable +{ + def apply[T <: Data](updateData: T, enable: Bool) = { + val r = Reg(updateData) + when (enable) { r := updateData } + r + } + def apply[T <: Data](updateData: T, resetData: T, enable: Bool) = { + val r = RegInit(resetData) + when (enable) { r := updateData } + r + } +} + +/** Builds a Mux tree out of the input signal vector using a one hot encoded + select signal. Returns the output of the Mux tree. + */ +object Mux1H +{ + def apply[T <: Data](sel: Iterable[Bool], in: Iterable[T]): T = { + if (in.tail.isEmpty) in.head + else { + val masked = (sel, in).zipped map ((s, i) => Mux(s, i.toBits, Bits(0))) + in.head.fromBits(masked.reduceLeft(_|_)) + } + } + def apply[T <: Data](in: Iterable[(Bool, T)]): T = { + val (sel, data) = in.unzip + apply(sel, data) + } + def apply[T <: Data](sel: Bits, in: Iterable[T]): T = + apply((0 until in.size).map(sel(_)), in) + def apply(sel: Bits, in: Bits): Bool = (sel & in).orR +} + +/** Builds a Mux tree under the assumption that multiple select signals + can be enabled. Priority is given to the first select signal. + + Returns the output of the Mux tree. + */ +object PriorityMux +{ + def apply[T <: Bits](in: Iterable[(Bool, T)]): T = { + if (in.size == 1) { + in.head._2 + } else { + Mux(in.head._1, in.head._2, apply(in.tail)) + } + } + def apply[T <: Bits](sel: Iterable[Bool], in: Iterable[T]): T = apply(sel zip in) + def apply[T <: Bits](sel: Bits, in: Iterable[T]): T = apply((0 until in.size).map(sel(_)), in) +} + +object unless { + def apply(c: Bool)(block: => Unit) { + when (!c) { block } + } +} + +object switch { + def apply(c: Bits)(block: => Unit) { + switchKeys.push(c) + block + switchKeys.pop() + } +} + +object is { + def apply(v: Bits)(block: => Unit): Unit = + apply(Seq(v))(block) + def apply(v: Bits, vr: Bits*)(block: => Unit): Unit = + apply(v :: vr.toList)(block) + def apply(v: Iterable[Bits])(block: => Unit): Unit = { + val keys = switchKeys + if (keys.isEmpty) ChiselError.error("The 'is' keyword may not be used outside of a switch.") + else if (!v.isEmpty) when (v.map(_ === keys.top).reduce(_||_)) { block } + } +} + +object MuxLookup { + def apply[S <: UInt, T <: Bits] (key: S, default: T, mapping: Seq[(S, T)]): T = { + var res = default; + for ((k, v) <- mapping.reverse) + res = Mux(key === k, v, res); + res + } + +} + +object Fill { + def apply(n: Int, x: Bool): UInt = n match { + case 0 => UInt(width=0) + case 1 => x + case x if n > 1 => UInt(0,n) - UInt(x) + case _ => throw new IllegalArgumentException(s"n (=$n) must be nonnegative integer.") + } + def apply(n: Int, y: UInt): UInt = { + n match { + case 0 => UInt(width=0) + case 1 => y + case x if n > 1 => + val p2 = Array.ofDim[UInt](log2Up(n+1)) + p2(0) = y + for (i <- 1 until p2.length) + p2(i) = Cat(p2(i-1), p2(i-1)) + Cat((0 until log2Up(x+1)).filter(i => (x & (1 << i)) != 0).map(p2(_))) + case _ => throw new IllegalArgumentException(s"n (=$n) must be nonnegative integer.") + } + } +} + +object MuxCase { + def apply[T <: Bits] (default: T, mapping: Seq[(Bool, T)]): T = { + var res = default; + for ((t, v) <- mapping.reverse){ + res = Mux(t, v, res); + } + res + } +} + +object ListLookup { + def apply[T <: Data](addr: UInt, default: List[T], mapping: Array[(UInt, List[T])]): List[T] = { + val map = mapping.map(m => (m._1 === addr, m._2)) + default.zipWithIndex map { case (d, i) => + map.foldRight(d)((m, n) => Mux(m._1, m._2(i), n)) + } + } +} + +object Lookup { + def apply[T <: Bits](addr: UInt, default: T, mapping: Seq[(UInt, T)]): T = + ListLookup(addr, List(default), mapping.map(m => (m._1, List(m._2))).toArray).head +} + +/** Litte/big bit endian convertion: reverse the order of the bits in a UInt. +*/ +object Reverse +{ + private def doit(in: UInt, length: Int): UInt = { + if (length == 1) { + in + } else if (isPow2(length) && length >= 8 && length <= 64) { + // Do it in logarithmic time to speed up C++. Neutral for real HW. + var res = in + var shift = length >> 1 + var mask = UInt((BigInt(1) << length) - 1, length) + do { + mask = mask ^ (mask(length-shift-1,0) << UInt(shift)) + res = ((res >> UInt(shift)) & mask) | (res(length-shift-1,0) << UInt(shift) & ~mask) + shift = shift >> 1 + } while (shift > 0) + res + } else { + val half = (1 << log2Up(length))/2 + Cat(doit(in(half-1,0), half), doit(in(length-1,half), length-half)) + } + } + def apply(in: UInt): UInt = doit(in, in.getWidth) +} + +/** Returns the n-cycle delayed version of the input signal. + */ +object ShiftRegister +{ + def apply[T <: Data](in: T, n: Int, en: Bool = Bool(true)): T = + { + // The order of tests reflects the expected use cases. + if (n == 1) { + RegEnable(in, en) + } else if (n != 0) { + RegNext(apply(in, n-1, en)) + } else { + in + } + } +} + +/** Returns the one hot encoding of the input UInt. + */ +object UIntToOH +{ + def apply(in: UInt, width: Int = -1): UInt = + if (width == -1) UInt(1) << in + else (UInt(1) << in(log2Up(width)-1,0))(width-1,0) +} |
