Skip to content

Latest commit

 

History

History
284 lines (242 loc) · 12 KB

getting_started_advanced.md

File metadata and controls

284 lines (242 loc) · 12 KB

Getting Started Tutorial: Advanced Library Construction

In this section, we will convert the four-LED array into an n-LED array block and go through the advanced constructs of generators and port arrays.

As a refresher, in the first part of the getting started tutorial, we dropped 4 LEDs directly in our top-level design sign with a for loop in:

self.led = ElementDict[IndicatorLed]()
for i in range(4):
  self.led[i] = self.Block(IndicatorLed())
  ...

However, arrays of LEDs are common enough that they would make sense as a library block, and it would condense these three lines of code here into just one.

The IDE does not support graphical operations for programmatic constructing circuits. However, you can continue to use the IDE for visualization.

The Naive (but Broken) Approach

You might be tempted to drop the above into a Block and pipe the int through as a constructor argument:

# DON'T DO THIS - THIS WON'T WORK
class LedArray(Block):
  def __init__(self, count: int) -> None:
    super().__init__()

    self.led = ElementDict[IndicatorLed]()
    for i in range(count):
      self.led[i] = self.Block(IndicatorLed())
      ...

However, there's a few problems with this, which illustrates a few significant ways our HDL model differs from Python:

  • When you instantiate a block, for example with self.Block(LedArray(4)), it only constructs a block stub. As a result, we can't pass parameters into Blocks that are concrete values like int.
    • This structure allows the compiler to resolve the dataflow and fill out the details when the parameter values are ready, instead of forcing the user to take into account dataflow when ordering their code. For example, the IndicatorLed generated resistor depends on its input voltage, which depends on the voltage on its signal pin, which isn't available until it's connected.
  • All IOs must be static, since classes define a single template block. So we can't create different numbers of top-level IOs based on a parameter value.

But, we have ways of achieving the same goals even within the above structure.

Interface

In this section, we'll define the block's interface using advanced constructs for library writers. This will hopefully also illuminate how some of the existing block generators work under the hood.

Start by creating an empty block, as done previously:

class LedArray(Block):
  def __init__(self) -> None:
    super().__init__()

Parameters

First, we'll need to define the block such that it can take a parameter. This is actually only slight twist on the naive approach:

class LedArray(Block):
  @init_in_parent
  def __init__(self, count: IntLike) -> None:
    super().__init__()

Instead of the constructor argument being a Python type that defines a value like int, we use an expression-type. This defines the type of the parameter being passed but not the value. @init_in_parent is needed whenever a Block defines constructor parameters.

The expression type is a way to refer to the parameter but without giving it a concrete value. This is needed since the value is resolved in the compiler and therefore not available in the HDL to the constructor.

Note that the type is not IntExpr, but IntLike, which also includes int. This allows calling the constructor with an int value directly, if we just have a static parameterization. However, because of the required @init_in_parent, the actual value seem by the constructor will be a new IntExpr.

@init_in_parent does some processing on the function itself, for example inspecting the constructor argument list and turning those into parameters, and translating the values passed in from the constructor call into references to the block's own parameters.

Port Arrays - Definition

Next, because we will have n LEDs, we will need n IO pins. While we can't define a separate top-level IO for each LED, we can define a port array IO that is dynamically sized:

class LedArray(Block):
  @init_in_parent
  def __init__(self, count: IntLike) -> None:
    super().__init__()
    self.ios = self.Port(Vector(DigitalSink.empty()), [Input])
    self.gnd = self.Port(Ground.empty(), [Common])

Port arrays are a container port that contain individual ports internally. As we saw from Part 1, viewed externally, they have no defined size and ports can be requested from them with optional names. However, from internally, we will be able to define their size, as well as get the names of incoming connections.

Port arrays require the port type to be undefined. Since we have not defined any ports so far, this is only needed for the type and may not have additional data like parameter values.

Implementation

In this section, we'll actually implement the circuit generator.

Generator

First, we will need a way to get the concrete (int) value for the LED count. Generators are a way to defer the implementation of the block until its parameter values are ready, then get the concrete Python version of that parameter for use in the HDL. We'll change the base class in our prior code from Block to GeneratorBlock which provides this functionality.

class LedArray(GeneratorBlock):
  @init_in_parent
  def __init__(self, count: IntLike) -> None:
    super().__init__()
    self.ios = self.Port(Vector(DigitalSink.empty()), [Input])
    self.gnd = self.Port(Ground.empty(), [Common])
    self.generator(self.generate, count)
    
  def generate(self, count: int) -> None:
    ...

While here we use generators as a way to get a concrete value for circuit generation (the LED count), generators can also be used to do calculations beyond the operations available with the parameters. For example, while we can add two IntExprs (which produces another IntExpr), something more complex like square root is not provided. For those cases, use a generator to get the parameter's value, where you have access to the full power of Python.

Port Arrays - Internal

So far, the port array is still empty, so we must define its elements. With the count available as an int, we can use the for loop structure from before:

class LedArray(GeneratorBlock):
  ...
  def generate(self, count: int) -> None:
    for i in range(count):
      self.ios.append_elt(DigitalSink.empty())
      ...

Port array's .append_elt(...) takes in the same arguments as self.Port(...).

To mark a port array as explicitly having no elements, use .defined():

  class EmptyArrayBlock(Block):
    def __init__(self) -> None:
      super().__init__()
      self.empty_array = self.Port(Vector(DigitalSink.empty()))
      self.empty_array.defined()

Circuit Generation

Instantiate the LEDs and connect them to the IO pin and ground as needed. .append_elt(...) returns the newly created port within the array, which can be used in self.connect(...).

At this point, your HDL might look like...
class LedArray(GeneratorBlock):
  @init_in_parent
  def __init__(self, count: IntLike) -> None:
    super().__init__()
    self.ios = self.Port(Vector(DigitalSink.empty()), [Input])
    self.gnd = self.Port(Ground.empty(), [Common])
    self.generator(self.generate, count)

  def generate(self, count: int) -> None:
      self.led = ElementDict[IndicatorLed]()
      for i in range(count):
        io = self.ios.append_elt(DigitalSink.empty())
        self.led[i] = self.Block(IndicatorLed())
        self.connect(io, self.led[i].signal)
        self.connect(self.gnd, self.led[i].gnd)

Putting it All Together

Replace the for loop in your top-level design with the single parameterized LedArray instantiation, and connect it to the microcontroller:

class BlinkyExample(SimpleBoardTop):
  def contents(self) -> None:
    ...
    with self.implicit_connect(
            ...
    ) as imp:
      ...
      self.led = imp.Block(LedArray(4))
      self.connect(self.mcu.gpio.request_vector('led'), self.led.ios)

Or, since we added the implicit tag Input to the IOs port, we can use chain to combine the instantiation and connection:

class BlinkyExample(SimpleBoardTop):
  def contents(self) -> None:
    ...
    with self.implicit_connect(
            ...
    ) as imp:
      ...
      (self.led, ), _ = self.chain(self.mcu.gpio.request_vector('led'), imp.Block(LedArray(4)))

As shown above, port arrays can be directly connected together to make parallel connections, and we can request sub-arrays from a port array. When connecting port arrays together, exactly one must define the array width, which is automatically propagated to the others in the connection. As a result, the single LED count parameter also drives the connection width and the pins requested from the microcontroller.

There are several ways to connect to arrays from externally:

  • As in the first part of the tutorial, we can request individual sub-ports.
  • As seen here, we can request a sub-array.
  • Or as also seen here, we can connect the array as a whole

For connections, only the types have to match, so you can (as done above) connect a whole array to a requested sub-array, or connect two whole arrays. Just remember that in any array connection, there must be exactly array of defined width, and all other arrays will take their widths from that.

Exception, if connecting an internally-facing array as a whole: it can only be connected to exactly one other externally-facing array.

When request_vector is used, each element's suggested name is the sub-array's suggested name, an underscore (_), then the element index. This is slightly different than the naming we've used for pin assignment so far, so we will need to update the refinements:

class BlinkyExample(SimpleBoardTop):
  ...
  def refinements(self) -> Refinements:
    return super().refinements() + Refinements(
      ...
      instance_values=[
        (['mcu', 'pin_assigns'], [
          'led_0=26',
          'led_1=27',
          'led_2=28',
          'led_3=29',
        ])
      ])
At this point, your HDL might look like...
class BlinkyExample(SimpleBoardTop):
  def contents(self) -> None:
    super().contents()
    self.usb = self.Block(UsbCReceptacle())
    self.buck = self.Block(BuckConverter(3.3*Volt(tol=0.05)))
    self.connect(self.usb.gnd, self.buck.gnd)
    self.connect(self.usb.pwr, self.buck.pwr_in)

    with self.implicit_connect(
        ImplicitConnect(self.buck.pwr_out, [Power]),
        ImplicitConnect(self.buck.gnd, [Common]),
    ) as imp:
      self.mcu = imp.Block(IoController())

      (self.sw, ), _ = self.chain(imp.Block(DigitalSwitch()), self.mcu.gpio.request('sw'))

      (self.led, ), _ = self.chain(self.mcu.gpio.request_vector('led'), imp.Block(LedArray(4)))

      # optionally, you may have also instantiated your magnetic sensor

  def refinements(self) -> Refinements:
    return super().refinements() + Refinements(
    instance_refinements=[
      (['buck'], Tps561201),
      (['mcu'], Esp32_Wroom_32),
    ],
    instance_values=[
      (['mcu', 'pin_assigns'], [
        'led_0=26',
        'led_1=27',
        'led_2=28',
        'led_3=29',
       ])
    ])

Board Optimization

Continue to the next part of the tutorial on optimizing the board using packed components.

Additional Resources

Check out these examples of generators:

  • Charlieplexing LED generator: a much more advanced version of the LED array that minimizes IO pins to drive LEDs, by allowing one IO pin to drive both a column and row of LEDs.
  • Buck and boost converter generators: the power path generator for these switching DC/DC converters encode well-known design equations using generators.