This project consists of two (and a half) major parts:
- The frontend (user-facing) HDL, written in Python, living in the top-level folders like
edg_core/
. This elaborates user-written HDL into an internal hierarchical block-diagram representation on a block-by-block, single-level-deep basis.- This also includes the electronics model and libraries build atop the core, in
electronics_model/
,electronics_abstract_parts/
, andelectronics_lib/
.
- This also includes the electronics model and libraries build atop the core, in
- The compiler, written in Scala, living in
compiler/
. This compiles together a whole design, starting at the top-level block and dropping in sub-blocks (recursively), invoking generators, and resolving parameters. - The intermediate representation (defining the core hierarchical block diagram model - blocks, ports, links, and expressions - in a language-independent form), written in protobuf, living in
proto/
.
This project has optional static typing annotations for Python which can be checked using mypy.
If you have mypy installed (you may also need the type stubs for protobuf: pip install mypy-protobuf
), you can typecheck code using:
mypy .
Or faster, with mypy in daemon mode:
dmypy run .
Or, using the Mypy plugin for Intellij.
Note: since mypy currently doesn't infer return types (see mypy issue 4409), some defs might be incomplete, so the type check is leaky and we can't currently use --disallow-incomplete-defs
or --disallow-untyped-defs
.
If that doesn't get resolved, we might go through and manually annotate all return types.
python -m unittest discover
Or, to run tests for a specific package (eg, edg_core
in this command):
python -m unittest discover -s edg_core -t .
Or, to run one specific test:
python -m unittest examples.test_blinky.BlinkyTestCase.test_design_complete
PROTIP: run both static type checking and unit testing by combining the commands with &&
A pre-compiled compiler JAR is included. If you have not modified any of the .scala files, you do not need to recompile the compiler.
- Download and install sbt, a build tool or Scala.
- In the
compiler/
folder, runsbt assembly
to compile the compiler JAR file. The system will automatically prefer the locally built JAR file over the pre-compiled JAR and indicate its use through the console. - Optionally, to commit a new pre-compiled JAR, move the newly compiled JAR from
compiler/target/scala-*/edg-compiler-assembly-*-SNAPSHOT.jar
toedg_core/resources/edg-compiler-precompiled.jar
.
Largely based on this tutorial.
Local:
- Install in editable mode (package refers to this source directory, instead of making a copy):
python -m pip install -e .
Upload:
- These dependencies are necessary:
python -m pip install build twine
- Build the package (creates a
dist
folder with a.whl
(zipped) file):python -m build
- Optionally, upload to TestPyPI:
python -m twine upload -r testpypi dist/*
- Upload to PyPI:
python -m twine upload dist/*
Some documentation may be out of date.
edg_core
is the core package, and sets up base classes for the hierarchy block and links model.
These are domain-neutral (not specific to circuit boards, or even electronics).
The Python implementation of the compiler (backend) also lives here.
- BaseBlock: abstract base class for all block-like constructs, which contain ports (IOs), parameters, and constraints between them. In general, subclasses may override superclass ports with a subtype port. (TODO: is this a good idea?) Provides infrastructure to record the names of ports and parameters (by overriding setattr) with the syntax self.[name] = self.Port([port])
- Link: abstract base class for all links, which are inferred to fit a connection between ports.
- Block: abstract base class for blocks.
- ConcreteBlock: abstract base class for blocks that can be part of a final design.
- PortBridge: block-like construct that defines how an external port of a hierarchy block connects to an internal link, such as by adapting the type and propagating constraints.
- HierarchyBlock: abstract base class for blocks that contain an internal block diagram (with internal blocks and connections), which may also link to external ports.
- GeneratorPart (TODO: needs renaming and implementation): a hierarchy block that contains a function to generate the internal block diagram once the external block diagram is solved. Allows arbitrary Python to control the internal generation, since it is not part of the SMT loop.
- BasePort: abstract mixin for all port-like constructs, which knows its parent (either a block or a container port).
- BaseContainerPort: abstract base class for all ports that contains other ports.
- Port: abstract base class for leaf-level ports, which contains parameters. Defines the link_type it may connect to (TODO: should it support multiple link types, of a common superclass?), and provides access to the connected link which can be used to avoid propagating duplicate parameters. Also defines the bridge_type, if one exists where this port is the external port on a hierarchy block.
- Bundle: a Port and a BaseContainerPort, that defines internal fields (other ports).
- Vector: an unknown-sized vector (array) of ports, which supports a map-extract operation to return an Array of some inner parameter as well as reduction operators that wrap the map-extract and an Array reduction.
- ConstraintExpr: abstract base class for constraint expressions.
- BoolExpr: ConstraintExpr that is a Bool.
- NumLikeExpr: ConstraintExpr abstract base class for numeric expressions, provides arithmetic operations compatible with SMT solvers (add, subtract, comparison).
- FloatExpr: ConstraintExpr that is a real number.
- RangeExpr: ConstraintExpr that is a real-valued interval type, with a min and max. (TODO: may support null-intervals)
- Array: an unknown-sized array (container) of ConstraintExpr, which supports reduction operators (eg, sum, min, max, intersection).
- Driver: provides auto-discovery of block diagram components in a Python library. (TODO: provides conversion to design, including recursive instantiation of library elements)
The fundamental hierarchy blocks and links model is encoded as an intermediate representation in Protocol Buffer, The Protocol Buffers schema is in the edgir folder, and generated Protocol Buffers code (including Python type annotations) is committed to this repository. The intent behind this is to allow cross-language interoperability (eg, solver implemented in another, more efficient, language) and to define a compiled object format.
You can ignore this section, unless you are changing the Protocol Buffer structure.
electronics_model
uses the edg_core
classes to define an electronics model, by adding base classes to support circuit design (pin ports, copper-net links, and footprint blocks) and defining common electronics types.
- VoltageSink/Source/Link/Bridge: represents a single copper net, and models runtime voltage and currents as well as their limits Can be used as a base class for other single-copper-net ports.
- DigitalSink/Source/Link/Bridge: subclass of Voltage*, additionally models logic IO thresholds
- BaseCircuitBlock: a Block with associated footprints (where pins can be mapped to Voltage* ports) and nets (connections between Voltage* ports) This can be used for component-level blocks, links (with nets defining copper connectivity between ports), and hierarchy blocks. Not all blocks need a footprint: abstract blocks can rely on a refinement for a footprint, and hierarchy blocks can rely on internal blocks for footprints.
electronics_abstract_parts
further defines abstract block types for the type hierarchy of components.
electronics_lib
contains a standard library of actual parts (eg, could buy off DigiKey), their models, and supporting reference implementation components / circuits.
examples
contains several example top-level boards, written as test cases and run with the unit test suite.