SKILL.md
$27
- A
.zenfile is either a normal Starlark module loaded withload()or an instantiable schematic module loaded withModule().
load("./foo.zen", "helper")imports Starlark symbols.Foo = Module("./Foo.zen")orFoo = Module("github.com/org/repo/path/Foo.zen")loads a subcircuit.
./paths are relative to the current file and resolve within the same package. Cross-packageload()andModule()require the full package URL.
- Instantiation always passes
name=...first, then anyio()/config()inputs. Useful extras:properties,dnp,schematic.
Nets and interfaces:
Net(name=None, voltage=None, impedance=None)is the base connection type.
Power,Ground, andNotConnectedare specialized net types; more specialized net types live in stdlib.
- Across
io()boundaries:NotConnectedcan promote to any net type; specialized nets can demote to plainNet; plainNetdoes not auto-promote to specialized types. Use explicit casts likePower(net, voltage=...)orNet(power_net)when needed.
Components and sourcing:
Component(...)is the primitive physical-part constructor. Required fields are effectivelyname,symbol, andpins.
- The symbol is the source of truth for footprint and part metadata. Make the symbol properties correct; do not repeat
footprint=orpart=inComponent()when they are already provided by the symbol.
- Prefer
part=Part(mpn=..., manufacturer=...)over legacy scalarmpnandmanufacturerwhen part metadata is not already in the symbol.
Symbol(library, name=None)points at a.kicad_sym;nameis required for multi-symbol libraries.
- Omit
no_connectpins frompins;Component()wiresNotConnected()automatically.
io():
- Preferred form:
NAME = io(template, ...)wheretemplateis a net/interface type or instance, e.g.Power(voltage="3.3V").
- Name is inferred from the assignment target or struct field.
optional=Truemeans omitted inputs get auto-generated nets or interfaces.
config():
- Preferred form:
name = config(typ, default=..., ...); name is inferred from the assignment target.
typcan be primitive types, enums, records, or physical values such asVoltageorResistance.
allowed=constrains accepted values to a discrete set. Strings auto-convert when possible, e.g."10k"toResistance("10k").
- For discrete physical values, prefer a physical type with
allowed=[...]over an ad hoc enum.
- Use physical types from
@stdlib/units.zenfor physical-value configs. Useenum()only for non-physical design choices.
Utilities:
Layout(name, path)associates reusable layout metadata to a module.
check(condition, message),warn(message), anderror(message)are the validation and diagnostic primitives.
Authoring Idioms
Power, Interfaces, And Checks
- Keep rails explicit with prelude
Power(voltage=...)andGround; each publicPowerio()declares its voltage range unless the local API intentionally keeps it generic.
- Use
@stdlib/interfaces.zeninterfaces for buses and grouped signals that are not in the prelude.
- Use typed values and validation primitives (
check(...),warn(...),error(...),@stdlib/checks.zen) for electrical constraints instead of comments when possible.
- Connect
PowerandGroundios directly to pins and passives.
VDD = io(Power(voltage="3.0V to 5.5V"))
GND = io(Ground)
EN = io(Net, help="High to enable the regulator")
Configs And Computation
- Expose meaningful design choices, not incidental implementation details. Good configs include output voltage, gain, cutoff frequency, address, mode, or optional feature enablement. Avoid configs for fixed decoupling values, passive package sizes, and test-point style unless local code already makes them public API.
- Prefer one meaningful physical config over raw R/C/L strings. For example, expose a cutoff
Frequencyand compute snapped passives internally.
- Put non-trivial calculations in named functions with datasheet section or equation references when available. Snap results to E-series values with
e96(),e24(), or the appropriate stdlib utility.
def load_r(v_out, v_sense):
"""Datasheet §8.1.1 / Eq 4: V_OUT = V_SENSE × gm × R_L"""
GM = Current("200uA") / Voltage("1V")
return e96(v_out / (v_sense * GM))
DNP And Optional Circuitry
- Configs may change component values and
dnp=state, but they should not change which instances or nets exist in the schematic.
- Never use conditional instantiation to add, remove, or reconnect circuitry. Always instantiate the relevant components and use
dnp=for population state.
- When a config selects a value on the same two nets, prefer one component with a computed value.
- When a config selects between mutually exclusive net straps, instantiate each strap option and DNP the inactive ones so topology stays stable.
- Leverage an IC's internal pull-up or pull-down when the default mode uses it; use external bias components with
dnp=only for populated alternatives.
load("@stdlib/units.zen", "Voltage", "Resistance")
load("@stdlib/utils.zen", "e96")
Resistor = Module("@stdlib/generics/Resistor.zen")
Mode = enum("PFM", "PWM")
mode = config(Mode, default="PFM")
voltage_out = config(Voltage, default="5V", allowed=["3.3V", "5V"])
VOUT = io(Power(voltage=voltage_out))
GND = io(Ground())
VFB_REF = Voltage("0.8V")
R_FB_TOP_VAL = Resistance("100kohm")
def fb_bottom(vout):
"""Datasheet Table 1: R2 = R1 × VFB / (VOUT − VFB)"""
return e96(R_FB_TOP_VAL * VFB_REF / (vout - VFB_REF))
VCC = Power()
FB = Net()
MSYNC = Net()
# Same feedback divider instances and nets for every output voltage; only value changes.
Resistor(name="R_FB_TOP", value=R_FB_TOP_VAL.with_tolerance("1%"), package="0402", P1=VOUT, P2=FB)
Resistor(name="R_FB_BOT", value=fb_bottom(voltage_out).with_tolerance("1%"), package="0402", P1=FB, P2=GND)
# Same strap options and nets for every mode; only population changes.
Resistor(name="R_MSYNC_GND", value="0ohm", package="0402", P1=MSYNC, P2=GND, dnp=mode != Mode("PFM"))
Resistor(name="R_MSYNC_VCC", value="0ohm", package="0402", P1=MSYNC, P2=VCC, dnp=mode != Mode("PWM"))
Style
- Prefer concise one-line
io()andconfig()declarations when readable.
- Avoid overly verbose
help=text. Usehelp=only when it adds integrator-visible meaning that is not already obvious from the name, type, or default.
- Omit comments and help text that merely restate the code.
- Do not use decorative section-divider comments such as
# ===== Config =====,# ----- IOs -----, or multi-line banner blocks. They add no value.
Naming
Element
Convention
Example
io() names
UPPERCASE
VDD, GND, I2C
config() names
lowercase
input_filter, output_voltage
Components
Uppercase functional prefix
R_LOAD, C_VDD, U_LDO
Differential pairs
_P / _N suffixes
IN_P, IN_N
Packages And Manifests
Imports and dependencies:
@stdlib/...is implicit and toolchain-managed; do not declare it in[dependencies].
pcb.toml per repository/package type:
- Board repository root:
[workspace]metadata,[board]withname,path, anddescription, and board[dependencies].
- Registry repository root:
[workspace]metadata and top-levelcomponents/**/modules/*members; no[board].
- Reusable packages (modules, components):
[dependencies]and optional defaultparts.
Stdlib
Prelude symbols available in .zen files without load(): Net, Power, Ground, NotConnected, Board, Layout, Part. Local definitions can shadow them.
@stdlib/board_config.zen:
-
Board is a prelude helper backed by @stdlib/board_config.zen. For standard boards, prefer the layers= helper instead of manually writing stackups and design rules:
Board(name="MainBoard", layout_path="layout/MainBoard", layers=4)
-
layers selects default stackup, netclasses, constraints, and predefined sizes for common 2/4/6/8/10-layer boards.
-
outer_copper_weight, copper_finish, solder_mask_color, track_widths, and via_dimensions customize those defaults. Extra track widths and vias are appended, deduplicated, and sorted.
-
Use explicit BoardConfig, Stackup, DesignRules, NetClass, and related records only when the standard defaults are insufficient; if both layers and config are provided, config is merged over the layers-derived defaults.
@stdlib/interfaces.zen:
- Common interfaces:
DiffPair,I2c,I3c,Spi,Qspi,Uart,Usart,Swd,Jtag,Usb2,Usb3, and others.
UartPair()andUsartPair()generate cross-connected point-to-point links.
@stdlib/units.zen:
-
Physical types: Voltage, Current, Resistance, Capacitance, Inductance, Impedance, Frequency, Temperature, Time, Power.
-
Constructors accept point values and ranges:
Voltage("3.3V") # point value
Resistance("4k7") # 4.7kΩ resistor notation
Capacitance("100nF")
Voltage("1.1–3.6V") # range
Voltage("11–26V (12V)") # range with explicit nominal
-
Arithmetic tracks units automatically: Voltage("3.3V") * Current("0.5A") → 1.65W; Voltage("5V") / Current("100mA") → 50Ω.
-
Properties: .value (alias for .nominal), .nominal, .min, .max, .tolerance, .unit.
-
Methods: .with_tolerance(t), .with_value(v), .with_unit(u), .abs(), .diff(other), .within(other), .matches(other).
-
Operators: +, -, *, / (with unit tracking), <, >, <=, >=, == (strict equality against another PhysicalValue), unary -. Use .matches(other) for coercive comparisons against strings or scalars, e.g. Voltage("5V").matches("5V").
-
String formatting: point → "3.3V"; symmetric tolerance → "10k 5%"; range → "11–26V (16V nom.)".
@stdlib/checks.zen:
voltage_within(...)is the main reusableio()-boundary power-rail check.
@stdlib/utils.zen:
e3,e6,e12,e24,e48,e96,e192snap physical values to standard E-series.
@stdlib/generics/*:
- Prefer generics for common parts:
Resistor,Capacitor,Inductor,FerriteBead,Led,Rectifier,Zener,Tvs,Crystal,TestPoint,PinHeader,NetTie,SolderJumper,MountingHole,Fiducial,Version.
Diodeis deprecated; useRectifier(standard/Schottky),Zener(breakdown/reference), orTvs(transient suppressor).