// SPDX-License-Identifier: Apache-2.0 package chisel3.experimental import chisel3.internal.firrtl.Width import chisel3.internal.sourceinfo.SourceInfo import chisel3.internal._ import chisel3.{ ActualDirection, Bits, CompileOptions, Data, Element, PString, Printable, RawModule, SpecifiedDirection, UInt } import scala.collection.mutable /** Data type for representing bidirectional bitvectors of a given width * * Analog support is limited to allowing wiring up of Verilog BlackBoxes with bidirectional (inout) * pins. There is currently no support for reading or writing of Analog types within Chisel code. * * Given that Analog is bidirectional, it is illegal to assign a direction to any Analog type. It * is legal to "flip" the direction (since Analog can be a member of aggregate types) which has no * effect. * * Analog types are generally connected using the bidirectional [[attach]] mechanism, but also * support limited bulkconnect `<>`. Analog types are only allowed to be bulk connected *once* in a * given module. This is to prevent any surprising consequences of last connect semantics. * * @note This API is experimental and subject to change */ final class Analog private (private[chisel3] val width: Width) extends Element { require(width.known, "Since Analog is only for use in BlackBoxes, width must be known") override def toString: String = stringAccessor(s"Analog$width") private[chisel3] override def typeEquivalent(that: Data): Boolean = that.isInstanceOf[Analog] && this.width == that.width override def litOption: Option[BigInt] = None def cloneType: this.type = new Analog(width).asInstanceOf[this.type] // Used to enforce single bulk connect of Analog types, multi-attach is still okay // Note that this really means 1 bulk connect per Module because a port can // be connected in the parent module as well private[chisel3] val biConnectLocs = mutable.Map.empty[RawModule, SourceInfo] // Define setter/getter pairing // Analog can only be bound to Ports and Wires (and Unbound) private[chisel3] override def bind(target: Binding, parentDirection: SpecifiedDirection): Unit = { _parent.foreach(_.addId(this)) SpecifiedDirection.fromParent(parentDirection, specifiedDirection) match { case SpecifiedDirection.Unspecified | SpecifiedDirection.Flip => case x => throwException(s"Analog may not have explicit direction, got '$x'") } val targetTopBinding = target match { case target: TopBinding => target case ChildBinding(parent) => parent.topBinding // See https://github.com/freechipsproject/chisel3/pull/946 case SampleElementBinding(parent) => parent.topBinding case a: MemTypeBinding[_] => a } targetTopBinding match { case _: WireBinding | _: PortBinding | _: ViewBinding | _: AggregateViewBinding => direction = ActualDirection.Bidirectional(ActualDirection.Default) case x => throwException(s"Analog can only be Ports and Wires, not '$x'") } binding = target } override def do_asUInt(implicit sourceInfo: SourceInfo, compileOptions: CompileOptions): UInt = throwException("Analog does not support asUInt") private[chisel3] override def connectFromBits( that: Bits )( implicit sourceInfo: SourceInfo, compileOptions: CompileOptions ): Unit = { throwException("Analog does not support connectFromBits") } def toPrintable: Printable = PString("Analog") } /** Object that provides factory methods for [[Analog]] objects * * @note This API is experimental and subject to change */ object Analog { def apply(width: Width): Analog = new Analog(width) }