Files
DFPlayer-carrier-board/GEMINI.md
2025-09-27 12:44:25 +02:00

20 KiB
Raw Permalink Blame History

GEMINI.md

ato is a declarative DSL to design electronics (PCBs) with. It is part of the atopile project. Atopile is run by the vscode/cursor/windsurf extension. The CLI (which is invoked by the extension) actually builds the project.

Not available in ato

  • if statements
  • while loops
  • functions (calls or definitions)
  • classes
  • objects
  • exceptions
  • generators

Ato Syntax

ato sytax is heavily inspired by Python, but fully declarative. ato thus has no procedural code, and no side effects.

Examples of syntax

#pragma text
#pragma func("X")
# enable for loop syntax feature:
#pragma experiment("FOR_LOOP)

# --- Imports ---
# Standard import (newline terminated)
import ModuleName

# Import with multiple modules (newline terminated)
import Module1, Module2.Submodule

# Import from a specific file/source (newline terminated)
from "path/to/source.ato" import SpecificModule

# Multiple imports on one line (semicolon separated)
import AnotherModule; from "another/source.ato" import AnotherSpecific

# Deprecated import form (newline terminated)
# TODO: remove when unsupported
import DeprecatedModule from "other/source.ato"

# --- Top-level Definitions and Statements ---

pass
pass;

"docstring-like statement"
"docstring-like statement";

top_level_var = 123

# Compound statement
pass; another_var = 456; "another docstring"

# Block definitions
component MyComponent:
    # Simple statement inside block (newline terminated)
    pass

    # Multiple simple statements on one line (semicolon separated)
    pass; internal_flag = True

module AnotherBaseModule:
    pin base_pin
    base_param = 10

interface MyInterface:
    pin io

module DemoModule from AnotherBaseModule:
    # --- Declarations ---
    pin p1              # Pin declaration with name
    pin 1               # Pin declaration with number
    pin "GND"           # Pin declaration with string
    signal my_signal    # Signal definition
    a_field: AnotherBaseModule      # Field declaration with type hint

    # --- Assignments ---
    # Newline terminated:
    internal_variable = 123

    # Semicolon separated on one line:
    var_a = 1; var_b = "string"

    # Cumulative assignment (+=, -=) - Newline terminated
    value = 1
    value += 1; value -= 1

    # Set assignment (|=, &=) - Newline terminated
    flags |= 1; flags &= 2

    # --- Connections ---
    p1 ~ base_pin
    mif ~> bridge
    mif ~> bridge ~> bridge
    mif ~> bridge ~> bridge ~> mif
    bridge ~> mif
    mif <~ bridge
    mif <~ bridge <~ bridge
    mif <~ bridge <~ bridge <~ mif
    bridge <~ mif

    # Semicolon separated on one line:
    p_multi1 ~ my_signal; p_multi2 ~ sig_multi1

    # --- Retyping ---
    instance.x -> AnotherBaseModule

    # --- Instantiation ---
    instance = new MyComponent
    container = new MyComponent[10]
    templated_instance_a = new MyComponent
    templated_instance_b = new MyComponent<int_=1>
    templated_instance_c = new MyComponent<float_=2.5>
    templated_instance_d = new MyComponent<string_="hello">
    templated_instance_e = new MyComponent<int_=1, float_=2.5, string_="hello">
    templated_instance_f = new MyComponent<int_=1, float_=2.5, string_="hello", bool_=True>

    # Semicolon separated instantiations (via assignment):
    inst_a = new MyComponent; inst_b = new AnotherBaseModule

    # --- Traits ---
    trait trait_name
    trait trait_name<int_=1>
    trait trait_name<float_=2.5>
    trait trait_name<string_="hello">
    trait trait_name<bool_=True>
    trait trait_name::constructor
    trait trait_name::constructor<int_=1>

    # Semicolon separated on one line:
    trait TraitA; trait TraitB::constructor; trait TraitC<arg_=1>

    # --- Assertions ---
    assert x > 5V
    assert x < 10V
    assert 5V < x < 10V
    assert x >= 5V
    assert x <= 10V
    assert current within 1A +/- 10mA
    assert voltage within 1V +/- 10%
    assert resistance is 1kohm to 1.1kohm

    # Semicolon separated on one line:
    assert x is 1V; assert another_param is 2V

    # --- Loops ---
    for item in container:
        item ~ p1

    # For loop iterating over a slice
    for item in container[0:4]:
        pass
        item.value = 1; pass

    # For loop iterating over a list literal of field references
    for ref in [p1, x.1, x.GND]:
        pass

    # --- References and Indexing ---
    # Reference with array index assignment
    array_element = container[3]

    # --- Literals and Expressions ---
    # Integer
    int_val = 100
    neg_int_val = -50
    hex_val = 0xF1
    bin_val = 0b10
    oct_val = 0o10
    # Float
    float_val = 3.14
    # Physical quantities
    voltage: V = 5V
    resistance: ohm = 10kohm
    capacitance: F = 100nF
    # Bilateral tolerance
    tolerance_val = 1kohm +/- 10%
    tolerance_abs = 5V +/- 500mV
    tolerance_explicit_unit = 10A +/- 1A
    # Bounded quantity (range)
    voltage_range = 3V to 3.6V
    # Boolean
    is_enabled = True
    is_active = False
    # String
    message = "Hello inside module"

    # Arithmetic expressions
    sum_val = 1 + 2
    diff_val = 10 - 3ohm
    prod_val = 5 * 2mA
    div_val = 10V / 2kohm # Results in current
    power_val = 2**3
    complex_expr = (5 + 3) * 2 - 1
    flag_check = state | MASK_VALUE

    # Comparisons
    assert voltage within voltage_range
    assert length <= 5mm
    assert height >= 2mm



# --- Multi-line variations ---
pass; nested_var=1; another=2

complex_assignment = (
    voltage + resistance
    * capacitance
)


G4 Grammar

parser grammar AtoParser;

options {
	superClass = AtoParserBase;
	tokenVocab = AtoLexer;
}

file_input: (NEWLINE | stmt)* EOF;

pragma_stmt: PRAGMA;

stmt: simple_stmts | compound_stmt | pragma_stmt;
simple_stmts:
	simple_stmt (SEMI_COLON simple_stmt)* SEMI_COLON? NEWLINE;
simple_stmt:
	import_stmt
	| dep_import_stmt
	| assign_stmt
	| cum_assign_stmt
	| set_assign_stmt
	| connect_stmt
	| directed_connect_stmt
	| retype_stmt
	| pin_declaration
	| signaldef_stmt
	| assert_stmt
	| declaration_stmt
	| string_stmt
	| pass_stmt
	| trait_stmt;

compound_stmt: blockdef | for_stmt;

blockdef: blocktype name blockdef_super? COLON block;
// TODO @v0.4 consider ()
blockdef_super: FROM type_reference;
// TODO @v0.4 consider removing component (or more explicit code-as-data)
blocktype: (COMPONENT | MODULE | INTERFACE);
block: simple_stmts | NEWLINE INDENT stmt+ DEDENT;

// TODO: @v0.4 remove the deprecated import form
dep_import_stmt: IMPORT type_reference FROM string;
import_stmt: (FROM string)? IMPORT type_reference (
		COMMA type_reference
	)*;

declaration_stmt: field_reference type_info;
field_reference_or_declaration:
	field_reference
	| declaration_stmt;
assign_stmt: field_reference_or_declaration '=' assignable;
cum_assign_stmt:
	field_reference_or_declaration cum_operator cum_assignable;
// TODO: consider sets cum operator
set_assign_stmt:
	field_reference_or_declaration (OR_ASSIGN | AND_ASSIGN) cum_assignable;
cum_operator: ADD_ASSIGN | SUB_ASSIGN;
cum_assignable: literal_physical | arithmetic_expression;

assignable:
	string
	| new_stmt
	| literal_physical
	| arithmetic_expression
	| boolean_;

retype_stmt: field_reference ARROW type_reference;

directed_connect_stmt
	: bridgeable ((SPERM | LSPERM) bridgeable)+; // only one type of SPERM per stmt allowed. both here for better error messages
connect_stmt: mif WIRE mif;
bridgeable: connectable;
mif: connectable;
connectable: field_reference | signaldef_stmt | pindef_stmt;

signaldef_stmt: SIGNAL name;
pindef_stmt: pin_stmt;
pin_declaration: pin_stmt;
pin_stmt: PIN (name | number_hint_natural | string);

new_stmt: NEW type_reference ('[' new_count ']')? template?;
new_count: number_hint_natural;

string_stmt:
	string; // the unbound string is a statement used to add doc-strings

pass_stmt:
	PASS; // the unbound string is a statement used to add doc-strings

list_literal_of_field_references:
	'[' (field_reference (COMMA field_reference)* COMMA?)? ']';

iterable_references:
	field_reference slice?
	| list_literal_of_field_references;

for_stmt: FOR name IN iterable_references COLON block;

assert_stmt: ASSERT comparison;

trait_stmt
	: TRAIT type_reference (DOUBLE_COLON constructor)? template?; // TODO: move namespacing to type_reference
constructor: name;
template: '<' (template_arg (COMMA template_arg)* COMMA?)? '>';
template_arg: name ASSIGN literal;

// Comparison operators --------------------
comparison: arithmetic_expression compare_op_pair+;

compare_op_pair:
	lt_arithmetic_or
	| gt_arithmetic_or
	| lt_eq_arithmetic_or
	| gt_eq_arithmetic_or
	| in_arithmetic_or
	| is_arithmetic_or;

lt_arithmetic_or: LESS_THAN arithmetic_expression;
gt_arithmetic_or: GREATER_THAN arithmetic_expression;
lt_eq_arithmetic_or: LT_EQ arithmetic_expression;
gt_eq_arithmetic_or: GT_EQ arithmetic_expression;
in_arithmetic_or: WITHIN arithmetic_expression;
is_arithmetic_or: IS arithmetic_expression;

// Arithmetic operators --------------------

arithmetic_expression:
	arithmetic_expression (OR_OP | AND_OP) sum
	| sum;

sum: sum (PLUS | MINUS) term | term;

term: term (STAR | DIV) power | power;

power: functional (POWER functional)?;

functional: bound | name '(' bound+ ')';

bound: atom;

// Primary elements ----------------

slice:
	'[' (slice_start? COLON slice_stop? (COLON slice_step?)?)? ']'
	// else [::step] wouldn't match
	| '[' ( DOUBLE_COLON slice_step?) ']';
slice_start: number_hint_integer;
slice_stop: number_hint_integer;
slice_step: number_hint_integer;

atom: field_reference | literal_physical | arithmetic_group;

arithmetic_group: '(' arithmetic_expression ')';

literal_physical:
	bound_quantity
	| bilateral_quantity
	| quantity;

bound_quantity: quantity TO quantity;
bilateral_quantity: quantity PLUS_OR_MINUS bilateral_tolerance;
quantity: number name?;
bilateral_tolerance: number_signless (PERCENT | name)?;

key: number_hint_integer;
array_index: '[' key ']';

// backwards compatibility for A.1
pin_reference_end: DOT number_hint_natural;
field_reference_part: name array_index?;
field_reference:
	field_reference_part (DOT field_reference_part)* pin_reference_end?;
type_reference: name (DOT name)*;
// TODO better unit
unit: name;
type_info: COLON unit;
name: NAME;

// Literals
literal: string | boolean_ | number;

string: STRING;
boolean_: TRUE | FALSE;
number_hint_natural: number_signless;
number_hint_integer: number;
number: (PLUS | MINUS)? number_signless;
number_signless: NUMBER;

Most used library modules/interfaces (api of them)

interface Electrical:
    pass

interface ElectricPower:
    hv = new Electrical
    lv = new Electrical

module Resistor:
    resistance: ohm
    max_power: W
    max_voltage: V
    unnamed = new Electrical[2]

module Capacitor:
    capacitance: F
    max_voltage: V
    unnamed = new Electrical[2]

interface I2C:
    scl = new ElectricLogic
    sda = new ElectricLogic
    frequency: Hz
    address: dimensionless

interface ElectricLogic:
    line = new Electrical
    reference = new ElectricPower

For the rest use the atopile MCP server

  • get_library_interfaces to list interfaces
  • get_library_modules to list modules
  • inspect_library_module_or_interface to inspect the code

Ato language features

experimental features

Enable with #pragma experiment("BRIDGE_CONNECT") BRIDGE_CONNECT: enables p1 ~> resistor ~> p2 syntax FOR_LOOP: enables for item in container: pass syntax TRAITS: enables trait trait_name syntax MODULE_TEMPLATING: enables new MyComponent<param=literal> syntax

modules, interfaces, parameters, traits

A block is either a module, interface or component. Components are just modules for code-as-data. Interfaces describe a connectable interface (e.g Electrical, ElectricPower, I2C, etc). A module is a block that can be instantiated. Think of it as the ato equivalent of a class. Parameters are variables for numbers and they work with constraints. E.g resistance: ohm is a parameter. Constrain with assert resistance within 10kohm +/- 10%. It's very important to use toleranced values for parameters. If you constrain a resistor.resistance to 10kohm there won't be a single part found because that's a tolerance of 0%.

Traits mark a module to have some kind of functionality that can be used in other modules. E.g trait has_designator_prefix is the way to mark a module to have a specific designator prefix that will be used in the designator field in the footprint.

connecting

You can only connect interfaces of the same type. resistor0.unnamed[0] ~ resistor0.unnamed[0] is the way to connect two resistors in series. If a module has the can_bridge trait you can use the sperm operator ~> to bridge the module. led.anode ~> resistor ~> power.hv connects the anode in series with the resistor and then the resistor in series with the high voltage power supply.

for loop syntax

for item in container: pass is the way to iterate over a container.

Ato CLI

How to run

You run ato commands through the MCP tool.

Packages

Packages can be found on the ato registry. To install a package you need to run ato add <PACKAGE_NAME>. e.g ato install atopile/addressable-leds And then can be imported with from "atopile/addressable-leds/sk6805-ec20.ato" import SK6805_EC20_driver. And used like this:

module MyModule:
    led = new SK6805_EC20_driver

Footprints & Part picking

Footprint selection is done through the part choice (ato create part auto-generates ato code for the part). The pin keyword is used to build footprint pinmaps so avoid using it outside of component blocks. Preferrably use Electrical interface for electrical interfaces. A lot of times it's actually ElectricLogic for things like GPIOs etc or ElectricPower for power supplies.

Passive modules (Resistors, Capacitors) are picked automatically by the constraints on their parameters. To constrain the package do e.g package = "0402". To explictly pick a part for a module use lcsc = "<LCSC_PART_NUMBER>".

Creating a package

Package generation process:

Review structure of other pacakges.

  1. Create new Directory in 'packages/packages' with naming convention '-' eg 'adi-adau145x'
  2. create an ato.yaml file in the new directory with the following content:
requires-atopile: '^0.9.0'

paths:
    src: '.'
    layout: ./layouts

builds:
    default:
        entry: <device>.ato:<device>_driver
    example:
        entry: <device>.ato:Example
  1. Create part using tool call 'search_and_install_jlcpcb_part'

  2. Import the part into the .ato file

  3. Read the datasheet for the device

  4. Find common interfaces in the part eg I2C, I2S, SPI, Power

  5. Create interfaces and connect them

power interfaces: power* = new ElectricPower power*.required = True # If critical to the device assert power*.voltage within <minimumoperating_voltage>V to <maximum_operating_voltage>V power.vcc ~ . power_.gnd ~ .

i2c interfaces: i2c = new I2C i2c.scl.line ~ . i2c.sda.line ~ .

spi interfaces: spi = new SPI spi.sclk.line ~ . spi.mosi.line ~ . spi.miso.line ~ .

  1. Add decoupling capacitors

looking at the datasheet, determine the required decoupling capacitors

eg: 2x 100nF 0402:

power_3v3 = new ElectricPower

Decoupling power_3v3

power_3v3_caps = new Capacitor[2] for capacitor in power_3v3_caps: capacitor.capacitance = 100nF +/- 20% capacitor.package = "0402" power_3v3.hv ~> capacitor ~> power_3v3.lv

  1. If device has pin configurable i2c addresses

If format is: use addressor module:

  • Use Addressor<address_bits=N> where N = number of address pins.
  • Connect each address_lines[i].line to the corresponding pin, and its .reference to a local power rail.
  • Set addressor.base to the lowest possible address and assert addressor.address is i2c.address.
  1. Create a README.md

Usage

<copy in example>

Contributing

Contributions to this package are welcome via pull requests on the GitHub repository.

License

This atopile package is provided under the MIT License.

  1. Connect high level interfaces directly in example:

eg:

i2c = new I2C power = new ElectricPower sensor = new Sensor

i2c ~ sensor.i2c power ~ sensor.power_3v3

Additional Notes & Gotchas (generic)

  • Multi-rail devices (VDD / VDDIO, AVDD / DVDD, etc.)

    • Model separate ElectricPower interfaces for each rail (e.g. power_core, power_io).
    • Mark each .required = True if the device cannot function without it, and add voltage assertions per datasheet.
  • Optional interfaces (SPI vs I²C)

    • If the device supports multiple buses, pick one for the initial driver. Leave unused bus pins as ElectricLogic lines or expose a second interface module later.
  • Decoupling guidance

    • If the datasheet shows multiple caps, model the minimum required set so the build passes; you can refine values/packages later.
  • File / directory layout recap

    • <vendor>-<device>/ package root
    • ato.yaml build manifest (include default and example targets)
    • <device>.ato driver + optional example module
    • parts/<MANUFACTURER_PARTNO>/ atomic part + footprint/symbol/step files

These tips should prevent common "footprint not found", "pin X missing", and build-time path errors when you add new devices.

Vibe coding a project

If the user gives you high level description of the project, use the following guide:

How LLMs can design electronics:

#1 Rule: USE THE TOOLS. If the tools dont work, dont freak out, you are probably using them wrong. Ask for help if you get stuck.

Top level design

  1. Research available packages relevant to the user requests using 'find_packages'
  2. Inspect promising packages using 'inspect_package'
  3. Propose packages to use for project and architucture to user, revise if needed
  4. Install needed packages using 'install_package'
  5. Import packages into main file
  6. Create instances of packages in main module

Power

  1. Review for each package the required voltage and current (current may not be provided, use judement if nessesary)
  2. Determine the power rails that need to be generated and a suitable tollerance (typically ~3-5% is acceptable)
  3. Determine the input power source, typically a battery, USB connector or other power connector (eg XT30) and install relevant package
  4. Find suitable regulators: a) if input voltage > required voltage and current is low, use an LDO package b) if input voltage > required voltage and current is high, use buck converter c) if input votlage < required voltage, use a boost converter d) if input voltage can be both less than or greater than input voltage, use buck boost (eg battery powered device that needs 3v3)
  5. If battery powered, add charger package

Typical power architucture example with LDO:

  • USB input power
  • Low current output (eg microcontroller)

from "atopile/ti-tlv75901/ti-tlv75901.ato" import TLV75901_driver from "atopile/usb-connectors/usb-connectors.ato" import USBCConn

module App:

# Rails
power_5v = new Power
power_3v3 = new Power

# Components
ldo = new TLV75901_driver
usb_connector = new USBCConn

# Connections
usb_connector.power ~ power_vbus
power_vbus ~> ldo ~> power_3v3

Communicaions

  1. Review packages required interfaces, typically i2c, spi or ElectricLogics
  2. Find suitable pins on the controller, typically a microcontroller or Linux SOC
  3. Connect interfaces eg micro.i2c[0] ~ sensor.i2c

Development process notes

  • After making changes, be sure to use 'build_project' to update the PCB
  • Builds will often generate errors/warnings, these should be reviewed and fixed
  • Prioritize pacakges from 'atopile' over other packages