From r2g-rtl2gds
Drives an open-source EDA flow from RTL to GDS with signoff checks (DRC, LVS, RCX) using OpenROAD, Yosys, KLayout, and OpenRCX for PPA iteration and flow diagnosis.
How this skill is triggered — by the user, by Claude, or both
Slash command
/r2g-rtl2gds:r2g-rtl2gdsThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Execute a staged, artifact-first open-source EDA flow from specification to GDSII with full signoff checks using OpenROAD-flow-scripts (ORFS). Prefer deterministic scripts for execution, keeping the agent focused on planning, generation, diagnosis, and iteration.
assets/config-template.mkassets/constraint-template.sdcassets/examples/simple-arbiter/constraints/config.mkassets/examples/simple-arbiter/constraints/constraint.sdcassets/examples/simple-arbiter/input/normalized-spec.yamlassets/examples/simple-arbiter/input/raw-spec.mdassets/platforms/nangate45/drc/FreePDK45.lydrcassets/platforms/nangate45/lvs/FreePDK45.lylvsassets/platforms/nangate45/lvs/FreePDK45_symseed.lvsinstall.shknowledge/README.mdknowledge/ab_runner.pyknowledge/analyze_execution.pyknowledge/backfill_fix_events.pyknowledge/escalations.pyknowledge/eval/eval_set.jsonknowledge/eval_heuristics.pyknowledge/eval_set.jsonknowledge/families.jsonknowledge/fix_log_manager.pyExecute a staged, artifact-first open-source EDA flow from specification to GDSII with full signoff checks using OpenROAD-flow-scripts (ORFS). Prefer deterministic scripts for execution, keeping the agent focused on planning, generation, diagnosis, and iteration.
Every flow script sources scripts/flow/_env.sh on entry, which autodetects
ORFS + tool paths and lets the user override any single value. You do not
need to source anything manually.
ORFS_ROOT=... run_orfs.sh ... wins unconditionally.$R2G_ENV_FILE (if set).references/env.local.sh (copy from references/env.local.sh.template).$ORFS_ROOT/env.sh (once ORFS_ROOT is known)./opt/openroad_tools_env.sh (if present).command -v <tool> on $PATH, then a list of well-known install paths (e.g. $ORFS_ROOT/tools/install/OpenROAD/bin/openroad, $HOME/oss-cad-suite/bin/yosys, /usr/local/bin/klayout).bash scripts/flow/check_env.sh
Prints the resolved ORFS_ROOT, every tool binary it picked, and the
platforms it can see. Exits non-zero if a required tool is missing.
# One-off override for a single run
ORFS_ROOT=/opt/ORFS OPENROAD_EXE=/opt/openroad/bin/openroad \
bash scripts/flow/run_orfs.sh design_cases/my_design nangate45
# Or persist overrides in a file
cp references/env.local.sh.template references/env.local.sh
# ...then edit the exports you care about; every subsequent flow picks them up.
nangate45, sky130hd, sky130hs, asap7, gf180, ihp-sg13g2 (default: nangate45).
references/spec-template.md and produce input/normalized-spec.yaml.design_cases/<design-name>/ using scripts/project/init_project.py.references/workflow.md.input/, rtl/, tb/, constraints/, lint/, sim/, synth/, backend/, drc/, lvs/, rcx/, reports/.rtl/design.v.tb/testbench.v.reports/rtl-notes.md.scripts/project/validate_config.py <project-dir> before ORFS backend to catch config/RTL issues early.constraints/config.mk and constraints/constraint.sdc.scripts/flow/run_orfs.sh to invoke the ORFS Makefile.After ORFS completes, extract PPA and run the timing gate:
scripts/extract/extract_ppa.py <project-dir> reports/ppa.json to extract timing metrics.scripts/reports/check_timing.py <project-dir> to classify WNS and TNS and write reports/timing_check.json.reports/timing_check.json and act on the tier:| Tier | Criteria | Agent Action |
|---|---|---|
| clean | WNS >= 0, TNS >= 0 | Proceed to signoff. |
| minor | WNS >= -2.0 AND TNS >= -10.0 | Auto-fix: update clk_period in constraint.sdc to suggested_clock_period from the JSON, then re-run backend. Report the fix to the user after the fact. |
| moderate | WNS >= -5.0 AND TNS >= -100.0 (but not clean/minor) | Stop. Present the numbered options from the JSON to the user. Wait for their choice. |
| severe | WNS < -5.0 OR TNS < -100.0 | Stop. Present options with strong warning. |
| unconstrained | WNS > 1e+30 | Stop. SDC clock port mismatch. Present options. Do NOT proceed. |
wns_tier and tns_tier fields so the agent can explain which metric triggered the tier (e.g., "TNS escalated this from minor to moderate").After a successful backend run, run signoff checks in order:
Two tool options are available:
KLayout DRC (default) — scripts/flow/run_drc.sh <project-dir> [platform]
make drc target with platform .lydrc rulesdrc/6_drc.lyrdb, drc/6_drc_count.rpt, drc/6_drc.logMagic DRC (sky130 only) — scripts/flow/run_magic_drc.sh <project-dir> [platform]
$PDK_ROOT/sky130A/libs.tech/magic/sky130A.tech
(set PDK_ROOT via references/env.local.sh — /opt/pdks is only the fallback default).drc/magic_drc.rpt, drc/magic_drc_count.rpt, drc/magic_drc_result.jsonTwo tool options are available:
KLayout LVS (default) — scripts/flow/run_lvs.sh <project-dir> [platform]
make lvs target with platform .lylvs rules + CDL netlistlvs/lvs_result.json with status "skipped")lvs/6_lvs.lvsdb, lvs/6_lvs.log, lvs/6_final.cdlconnect_implicit("VDD"/"VSS") for bulk merging and schematic.purge for unused cell pins (e.g., QN on DFFR_X1)LVS_TIMEOUT=7200 for these designs. The default 3600s may not be enough.Netgen LVS (sky130 only) — scripts/flow/run_netgen_lvs.sh <project-dir> [platform]
$PDK_ROOT/sky130A/libs.tech/netgen/sky130A_setup.tcl).
Set PDK_ROOT via references/env.local.sh; /opt/pdks is only the fallback default.lvs/extracted.spice, lvs/netgen_lvs.rpt, lvs/netgen_lvs_result.jsonreferences/failure-patterns.md,
"sky130 LVS").X-subcircuit instances to D devices (perim=→pj=) and runs netgen with
MAGIC_EXT_USE_GDS=1, so sky130_fd_sc_hd__diode_2 matches instead of flattening.assign out_port = in_port) need
export POST_GLOBAL_PLACE_TCL = <skill>/scripts/flow/orfs_hooks/buffer_port_feedthroughs.tcl
in config.mk before the backend run — SPICE cannot express two top-level ports on one
net, so without the hook LVS fails "Top level cell failed pin matching". The hook is a
no-op for designs without feedthroughs (safe to set everywhere); a backend re-run is
required when adding it. See references/failure-patterns.md, "sky130 LVS" cause 5.scripts/flow/run_rcx.sh <project-dir> [platform]
rcx/run_rcx.tcl) with define_process_corner, extract_parasitics, write_spef6_final.odb from ORFS results, writes SPEF outputrcx/6_final.spef, rcx/rcx.log, rcx/run_rcx.tclExtract results into JSON for reporting and dashboard:
scripts/extract/extract_drc.py <project-root> reports/drc.jsonscripts/extract/extract_lvs.py <project-root> reports/lvs.jsonscripts/extract/extract_rcx.py <project-root> reports/rcx.jsonfail, attempt automated real-layout fixes:
scripts/flow/fix_signoff.sh <project-dir> [platform] [--check drc|lvs|both]
(See references/signoff-fixing.md.)The skill learns from every fix attempt so candidate strategies are proposed in evidence-ranked order on the next similar violation.
fix_signoff.sh and check_timing.py --journal append lossless,
session-keyed rows to reports/fix_log.jsonl (one per iteration: strategy, before/after
counts, pre-fix violation class, verdict). fix_signoff.sh uses an adaptive budget (base
3 iters, hard cap 8, early-stop after 2 non-improving iters past the base).knowledge/ingest_run.py) reads fix_log.jsonl into the
Tier-1 fix_events table and writes a run_violations snapshot for every run — clean
or not (the full violation landscape). It then auto-runs fix_log_manager.manage()
(toggle R2G_FIX_AUTOLEARN, default on).learn_heuristics.py derives Tier-2 fix_trajectories (per-episode path,
including abandoned episodes and failed strategies — negative learning) and folds them
into Tier-3 fix_recipes inside heuristics.json.diagnose_signoff_fix.py reorders the strategy list by empirical clearance — there is no
hard gate, all real-fix strategies are always proposed, priority-ordered.
diagnose_signoff_fix.py <proj> --check drc --list prints the evidence-ranked candidate
set as JSON. Hard safety clamps are unchanged.knowledge/symptom.py: {check, class, predicates} → a stable symptom_id), NOT the
design-family name. learn_heuristics.py emits a top-level symptoms[symptom_id]
projection in heuristics.json (pooled across families/platforms, with by_platform +
evidence_designs provenance); diagnose_signoff_fix.py looks recipes up by symptom and
seeds an informed cross-platform prior for untried strategies (so a fix learned on
nangate45 transfers to e.g. sky130hd). It also surfaces the matching active prose lesson
(via search_failures.lessons_for_symptom) at the fix-decision point. monitor_health.py
(degradation alerts) and analyze_execution.py (fix-proposal triage) are operator-invoked
CLIs over the same store.See references/signoff-fixing.md ("Fix-Learning Loop") and knowledge/README.md.
Use campaign mode when you need to run the full flow unattended across many designs — or
when you want the A/B-gated recipe-learning cycle to run autonomously. The campaign
orchestrator (scripts/loop/engineer_loop.py) drives the flow scripts, ingests results,
triggers learning, and manages A/B trials without human gates.
# Add a project to the campaign ledger
python3 scripts/loop/engineer_loop.py add \
--ledger design_cases/_batch/campaign.jsonl \
--project design_cases/my_design [--platform nangate45]
# Run the campaign (optionally limit to N designs)
python3 scripts/loop/engineer_loop.py run \
--ledger design_cases/_batch/campaign.jsonl [--max N]
# Inspect per-design state
python3 scripts/loop/engineer_loop.py status \
--ledger design_cases/_batch/campaign.jsonl
The ledger is JSONL (last-state-wins); kill/restart is safe — the campaign resumes where it
left off. States: pending → flow → signoff → fixing → clean | escalated | abandoned.
Hard rules for campaign mode:
DESIGN_NAME
concurrently.DESIGN_NAME + FLOW_VARIANT concurrently.promoted recipes affect live strategy ranking; shadow and candidate recipes are
inert until their A/B trial completes.When the loop opens an escalation (unknown symptom, exhausted catalog, unseen crash, or
repeated regression), drain it following the agent runbook in
references/engineer-loop.md ("Escalation Drain"). That document also covers provenance
queries (trace_provenance.py) and the full safety-invariant list.
| Platform | KLayout DRC | KLayout LVS | Magic DRC | Netgen LVS | RCX |
|---|---|---|---|---|---|
| nangate45 | Yes | Yes | No | No | Yes |
| sky130hd | Yes | Yes | Yes | Yes | Yes |
| sky130hs | Yes | Yes | Yes | Yes | Yes |
| asap7 | Yes | No | No | No | Yes |
| gf180 | Yes | Yes | No | No | Yes |
| ihp-sg13g2 | Yes | Yes | No | No | Yes |
references/failure-patterns.md.After every flow — successful, failed, or partial — run:
python3 r2g-rtl2gds/knowledge/ingest_run.py design_cases/<project>
This reads the structured JSON artifacts produced by the extraction scripts
and appends one row to r2g-rtl2gds/knowledge/knowledge.sqlite. It never
parses raw ORFS logs.
Then rebuild derived artifacts:
python3 r2g-rtl2gds/knowledge/learn_heuristics.py
python3 r2g-rtl2gds/knowledge/mine_rules.py
knowledge/heuristics.json is consumed automatically by
suggest_config.py on the next project — no CLI changes required.knowledge/failure_candidates.json is a review queue, not a rule
source. Surface new signatures to the user and, if confirmed, edit
references/failure-patterns.md by hand.A family/platform pair appears in heuristics.json only after at least
3 successful runs under that configuration.
check_timing.py after every backend run. It checks both WNS and TNS. For minor violations (WNS >= -2.0 AND TNS >= -10.0), auto-fix by increasing clock period and re-running. For moderate/severe/unconstrained, stop and present numbered fix options — do not proceed without the user's decision.scripts/ for repeatable operations instead of re-inventing shell commands each time.scripts/flow/_env.sh, which autodetects ORFS and tool paths (see "Environment Setup"). /opt/openroad_tools_env.sh is only one optional source in that chain and may be absent.references/failure-patterns.md (see "Batch-Campaign Failure Patterns") and apply the repo-level batch fixer before any code changes. That fixer (tools/fix_orfs_failures.py in the agent-r2g repository — not shipped with the installed skill) handles the six dominant failure modes (memory inference, IO-pin perimeter overflow, place density >1, PDN straps, missing include dirs, stage timeouts) by rewriting config.mk. When running standalone, apply the same patterns by hand using references/failure-patterns.md. Do not hand-edit configs case-by-case in batch — extend the fix tool so future batches self-heal.CORE_UTILIZATION when in doubt.DIE_AREA = 0 0 S S with S = ceil((required_perim / 4) * 1.3) rounded up to 10um.SYNTH_MEMORY_MAX_BITS = 131072 (default 4096 is too tight for register files and FIFOs).design_cases/<design-name>/
├── input/
│ ├── raw-spec.md
│ └── normalized-spec.yaml
├── rtl/
│ └── design.v
├── tb/
│ └── testbench.v
├── constraints/
│ ├── config.mk
│ └── constraint.sdc
├── lint/
│ └── lint.log
├── sim/
│ ├── sim.log
│ └── output.vcd
├── synth/
│ ├── synth.ys
│ ├── synth.log
│ └── synth_output.v
├── backend/
│ └── RUN_<timestamp>/
│ ├── final/ # GDS, DEF, ODB
│ ├── logs/ # Per-stage logs
│ ├── reports/ # Timing, area, power
│ ├── drc/ # DRC results (copied)
│ ├── lvs/ # LVS results (copied)
│ └── rcx/ # RCX results (copied)
├── drc/
│ ├── 6_drc.lyrdb # KLayout DRC violation database (XML)
│ ├── 6_drc_count.rpt # Violation count
│ ├── 6_drc.log # DRC log
│ └── drc_run.log # Full make output
├── lvs/
│ ├── 6_lvs.lvsdb # KLayout LVS comparison database (XML)
│ ├── 6_lvs.log # LVS log
│ ├── 6_final.cdl # CDL netlist
│ ├── lvs_run.log # Full make output
│ └── lvs_result.json # Only if skipped (no rules)
├── rcx/
│ ├── 6_final.spef # SPEF parasitic data
│ ├── rcx.log # OpenRCX extraction log
│ └── run_rcx.tcl # Generated Tcl extraction script
├── labels/ # Dataset labels (Y): congestion/wirelength/timing/irdrop CSVs (run_labels.sh)
├── features/ # Dataset features (X): nodes/edges/metadata CSVs (run_features.sh)
├── reports/
│ ├── ppa.json # PPA metrics + geometry
│ ├── progress.json # ORFS stage completion
│ ├── run-history.json # Multi-run comparison
│ ├── run-compare.json # Baseline vs current delta
│ ├── diagnosis.json # Issue detection & suggestions
│ ├── drc.json # DRC summary (violations, categories)
│ ├── lvs.json # LVS summary (match/mismatch/skipped)
│ ├── rcx.json # RCX summary (net count, cap, res)
│ └── demo-summary.md # Human-readable summary
└── metadata.json
references/spec-template.md when the specification is incomplete or ambiguous.references/workflow.md when you need the phase-by-phase execution order.references/orfs-playbook.md before setting up or debugging the ORFS backend.references/failure-patterns.md when a run fails and you need a triage path.references/ppa-report-guide.md when summarizing synthesis/backend reports.references/label-extraction.md when building the physical-design dataset (per-cell/per-net labels + stats).references/feature-extraction.md when building the graph-feature (X) side of the dataset (per-node/per-edge/metadata CSVs + stats).scripts/extract/techlib/ for the shared per-platform tech layer consumed by both stages: profile.py (supply voltage, tap patterns, cell-type strategy per ORFS platform), resolve.py (the Python backend for resolve_platform_paths.sh — same KEY=VALUE contract), def_parse.py (single DEF/SDC parser), lef.py (routing-layer names, pitch/direction, regex matcher), liberty.py (cell/pin/net classifiers), cell_types.py (cell_type_id map — curated for nangate45, runtime-built for all others).scripts/ for initialization, spec normalization, environment checks, lint, simulation, synthesis, ORFS backend, DRC, LVS, RCX extraction, result collection, GDS preview rendering, dashboard generation, and run summaries.scripts/flow/orfs_hooks/ for Tcl sourced into ORFS stages via the generic
PRE_<STAGE>_TCL / POST_<STAGE>_TCL env hooks (set them in config.mk, not the shell —
ORFS scrubs exported variables). Currently: buffer_port_feedthroughs.tcl
(POST_GLOBAL_PLACE_TCL) splits port-to-port feedthrough nets behind real buffers so
Netgen LVS top-level pins match; a no-op for designs without feedthroughs.assets/examples/simple-arbiter/ as the first smoke-test case.assets/config-template.mk and assets/constraint-template.sdc as default backend configuration templates.Required Tools (all installed on this machine):
python3 (3.13+)yosys (synthesis)iverilog + vvp (simulation)openroad (place & route, OpenRCX)Optional Tools (also available):
verilator (faster lint/simulation)klayout (GDS visualization, DRC, LVS)magic (DRC, SPICE extraction for sky130)netgen-lvs (LVS comparison for sky130)gtkwave (waveform viewing)sta / opensta (static timing analysis)No manual env setup needed — flow scripts autodetect ORFS/tools via scripts/flow/_env.sh. Run scripts/flow/check_env.sh to confirm what was found.
Initialize a run directory with scripts/project/init_project.py <design-name>.
Save user requirements to input/raw-spec.md.
Normalize them into input/normalized-spec.yaml using scripts/project/normalize_spec.py.
Write or copy rtl/design.v and tb/testbench.v.
Run scripts/flow/check_env.sh to verify tool availability.
Run scripts/flow/run_lint.sh, then scripts/flow/run_sim.sh, then scripts/flow/run_synth.sh.
After those pass, prepare constraints/config.mk and constraints/constraint.sdc.
Run scripts/flow/run_orfs.sh <project-dir> for the backend.
Extract PPA: scripts/extract/extract_ppa.py <project-dir> reports/ppa.json
Run timing gate: scripts/reports/check_timing.py <project-dir> — reads reports/timing_check.json:
tier=clean: proceed to step 12.tier=minor: auto-fix clock period per suggested_clock_period, re-run step 9, then re-check.tier=moderate/severe/unconstrained: stop, present options to user, wait for decision.wns_tier and tns_tier to explain which metric drove the tier.Run signoff checks (only after timing gate passes or user approves):
scripts/flow/run_drc.sh <project-dir> [platform] (KLayout DRC)scripts/flow/run_magic_drc.sh <project-dir> [platform] (Magic DRC, sky130 only)scripts/flow/run_lvs.sh <project-dir> [platform] (KLayout LVS)scripts/flow/run_netgen_lvs.sh <project-dir> [platform] (Netgen LVS, sky130 only)scripts/flow/run_rcx.sh <project-dir> [platform]Extract remaining results:
scripts/extract/extract_drc.py <project-root> reports/drc.jsonscripts/extract/extract_lvs.py <project-root> reports/lvs.jsonscripts/extract/extract_rcx.py <project-root> reports/rcx.jsonfail, attempt real layout fixes (antenna diode insertion +
repair iters, route effort, density relief; LVS triage):
scripts/flow/fix_signoff.sh <project-dir> [platform] [--check drc|lvs|both] [--max-iters 3]
Real-fixes-only — never relaxes the rule deck. Residual stuck/timeout/KLayout-crash
cases are reported honestly. See references/signoff-fixing.md.
13b. Extract dataset labels (optional, for dataset building):scripts/flow/run_labels.sh <project-dir> [platform]<project-dir>/labels/ (congestion,
wirelength, timing, IR drop) and a per-design reports/labels_stats.json.references/label-extraction.md.tools/run_labels_batch.sh.
13c. Extract dataset features (optional, for dataset building — the X side of 13b's Y):scripts/flow/run_features.sh <project-dir> [platform]<project-dir>/features/: metadata.csv (graph-level),
nodes_{gate,net,iopin,pin}.csv, edges_{gate_pin,pin_net,iopin_net}.csv, plus a
per-design reports/features_stats.json.6_final.def (+ optional 6_final.spef) as the labels, so feature
rows join the label rows on graph_id+inst_name/net_name. Fail-soft; SPEF
absence degrades cap columns to 0.references/feature-extraction.md.tools/run_features_batch.sh.Diagnose issues: scripts/reports/build_diagnosis.py <project-root> reports/diagnosis.json
Get config suggestions: knowledge/suggest_config.py <project-dir> (optional, useful for tuning)
Collect artifacts with scripts/reports/collect_reports.py and summarize with scripts/reports/summarize_run.py.
Generate the dashboard with scripts/dashboard/generate_multi_project_dashboard.py.
The index page carries two read-only knowledge-store panels at the top (above the
project grid), computed by scripts/reports/build_lineage_view.py:
knowledge_db.is_success), signoff
positives, and whether heuristics.json is populated (the EMPTY case renders red — the
screaming diagnostic that learning is inert).config_lineage diff chain per design/platform
(changed/added/removed config keys + orfs/drc/lvs outcome delta prev→cur).INVARIANT: these panels are a strictly descriptive, READ-ONLY projection over
knowledge.sqlite / config_lineage / heuristics.json (opened mode=ro; the projection
writes only JSON). They are NEVER wired into suggest_config as an auto-tuner. The
config-variant lineage is a loose single-parent diff chain, not a true DAG.
Serve it with scripts/dashboard/serve_multi_project_dashboard.py 8765.
Default to an MVP flow that supports:
Escalate to the user before attempting CDC, multi-clock constraints, DFT, or signoff-quality closure.
Macro designs using fakeram45 on nangate45 are supported and validated (riscv32i, tinyRocket, swerv, bp_multi_top — all produce GDS + pass RCX). LVS passes for designs <150K cells; designs >150K cells may need extended LVS timeout.
Some designs instantiate hard memory macros (fakeram45 on nangate45, SRAM on sky130). These require extra config:
Verilog blackbox stubs — Provide module definitions for all macros referenced in the RTL. For designs using SYNTH_HIERARCHICAL=1, stubs must be actual module implementations (not (* blackbox *) attributes), because Yosys CELLMATCH pass needs cost info. Wrap the fakeram macros inside the BSG-style wrapper modules with real port connections.
ADDITIONAL_LEFS / ADDITIONAL_LIBS — Point to the platform's LEF and LIB files for each macro type.
CDL_FILE for LVS — The platform config.mk sets a default CDL_FILE that only includes standard cells. Macro designs need a combined CDL with both standard cells and fakeram subcircuit definitions. Use override export CDL_FILE (the override keyword is critical — without it, the platform config.mk, which is included after the design config, will silently overwrite your CDL_FILE).
MACRO_PLACEMENT_TCL — Provide a Tcl script for macro placement. Use find_macros (not all_macros) to check for macros, and call global_placement before macro_placement:
if {[find_macros] != ""} {
global_placement -density [place_density_with_lb_addon] -pad_left 2 -pad_right 2
macro_placement -halo {10 10} -style corner_max_wl
}
GDS_ALLOW_EMPTY — Set export GDS_ALLOW_EMPTY = fakeram.* so ORFS allows empty GDS cells for the macro stubs.
Safety flags — Large macro designs need SKIP_CTS_REPAIR_TIMING=1 and SKIP_LAST_GASP=1 to avoid OpenROAD crashes.
When the design's SRAM macros are undefined in the RTL set (e.g., Chipyard
BOOM's freepdk45_sram_*, generic foundry stubs without LEF/LIB) AND the
total memory bits are modest (<~256K bits), substitute behavioral
flop-array implementations instead of mapping to fakeram45.
Use tools/gen_openram_behavioral_stubs.py <wrapper.v> <stubs.v> (a repo-level helper in the agent-r2g repository — not bundled with the installed skill) to
auto-generate behavioral Verilog for every freepdk45_sram_<ports>_<rows>x<cols>[_<gran>]
referenced. The generator handles both 1rw0r (single-port) and 1w1r
(write-port + read-port, independent clocks) styles, with optional
write-mask granularity.
Why behavioral instead of fakeram45:
1w1r macros.When NOT to use behavioral stubs:
memory_map pass turns each
read port into a wide mux tree. At BOOM SmallSEBoom's 168K total bits,
the post-mapping cell count exceeds ~1M gates and ABC's speed
script grinds beyond the 4 h ORFS_TIMEOUT. The earlier guidance
(512K) was too optimistic; lowered after a 2h28m ABC failure.SYNTH_MEMORY_MAX_BITS per memory at 65536.
Single memories larger than that should use real macros.When behavioral stubs hit the ABC ceiling but you still want to avoid
real macros, try SYNTH_HIERARCHICAL=1 before switching to fakeram45.
With hierarchical synth, ABC is invoked separately per Yosys module —
each freepdk45_sram_* stub becomes its own small ABC run (≤32K gates
per memory) instead of one giant 1M-gate ABC pass.
Validated:
tsyncram_* sizes, 87K total memory bits,
79 RTL files, dual SYSCLK/BUSCLK): synth passes in 213 s under
SYNTH_HIERARCHICAL=1 + ABC_AREA=1. Yosys peak 553 MB; ABC ran
per kept module (5 ABC invocations, 57 s combined). Confirms
hierarchical mode lifts the behavioral ceiling well past the 50K-bit
flat-mode limit when single memories stay <16K bits each.SYNTH_HIERARCHICAL=1 is the recommended next step; if that still
fails, fall back to mapping the four 1rw0r macros (61K bits total)
to fakeram45 and keep the thirteen 1w1r macros behavioral.docs/faraday_viability.md for the per-design SRAM scale audit.export DESIGN_NAME = <top_module>
export PLATFORM = nangate45
export VERILOG_FILES = <absolute_path_to_rtl>
export SDC_FILE = <absolute_path_to_sdc>
export CORE_UTILIZATION = 30
export PLACE_DENSITY_LB_ADDON = 0.20
For macro designs, add (note the override on CDL_FILE):
export ADDITIONAL_LEFS = $(PLATFORM_DIR)/lef/fakeram45_64x32.lef
export ADDITIONAL_LIBS = $(PLATFORM_DIR)/lib/fakeram45_64x32.lib
export GDS_ALLOW_EMPTY = fakeram.*
export MACRO_PLACEMENT_TCL = /absolute/path/to/macro_placement.tcl
override export CDL_FILE = /absolute/path/to/combined_with_fakerams.cdl
The override keyword on CDL_FILE is essential: ORFS includes the platform config.mk after the design config.mk, so a bare export CDL_FILE gets silently overwritten by the platform default (which only has standard cells).
For sky130 (and any design whose RTL contains port-to-port assign out = in feedthroughs),
also add the stage hook that keeps Netgen LVS top-level pins matchable:
# Split port-to-port feedthrough nets so Netgen LVS top-level pins match
export POST_GLOBAL_PLACE_TCL = <skill>/scripts/flow/orfs_hooks/buffer_port_feedthroughs.tcl
Set it in config.mk (not the shell — ORFS scrubs exported variables). It is a no-op for designs without feedthroughs, so it is safe to set unconditionally.
Before running ORFS, validate every config.mk against these rules:
export SKIP_CTS_REPAIR_TIMING = 1
export SKIP_LAST_GASP = 1
Without these, OpenROAD may SIGSEGV during CTS timing repair.current_design <top_module>
set clk_name core_clock
set clk_port_name clk
set clk_period 10.0
set clk_io_pct 0.2
set clk_port [get_ports $clk_port_name]
create_clock -name $clk_name -period $clk_period $clk_port
set non_clock_inputs [all_inputs -no_clocks]
set_input_delay [expr $clk_period * $clk_io_pct] -clock $clk_name $non_clock_inputs
set_output_delay [expr $clk_period * $clk_io_pct] -clock $clk_name [all_outputs]
The scripts/flow/run_orfs.sh script:
FLOW_VARIANT from the project directory name (prevents collisions)make DESIGN_CONFIG=<config.mk> FLOW_VARIANT=<variant> with optional timeoutResource control via environment variables:
ORFS_TIMEOUT=7200 # Per-stage max runtime in seconds (default: 2 hours)
ORFS_MAX_CPUS=4 # Limit CPU cores via taskset (default: all)
PLACE_FAST=1 # Disable GPL_TIMING_DRIVEN/ROUTABILITY_DRIVEN — use for
# BOOM-class designs (>1M nets) where the timing-repair
# loop in gpl spins for hours after Nesterov has already
# converged. CTS/route still apply timing closure.
ROUTE_FAST=1 # Cap GRT iterations + skip optional GRT-stage repair
# passes. Sets SKIP_INCREMENTAL_REPAIR=1,
# SKIP_ANTENNA_REPAIR=1, DETAILED_ROUTE_END_ITERATION=10,
# GLOBAL_ROUTE_ARGS='-congestion_iterations 5
# -allow_congestion -verbose ...'. Use when GRT extra
# iterations cycle 1→30 forever on >1M-net designs
# (BOOM ChipTop). Result: GDS may have congestion/DRC
# but produces final ODB.
ROUTE_FAST_SKIP_DRT=1 # Combined with ROUTE_FAST: also SKIP_DETAILED_ROUTE=1.
# Last-resort fallback — produces DEF + GRT solution
# but no GDS. Use only when ROUTE_FAST still hangs in
# detailed routing.
FROM_STAGE=place # Resume from a specific stage (synth|floorplan|place|cts|route|finish)
These rules are enforced in all 6 execution scripts (run_orfs.sh, run_lvs.sh, run_rcx.sh, run_drc.sh, run_magic_drc.sh, run_netgen_lvs.sh):
PIPESTATUS capture — When piping timeout ... | tee, temporarily disable set -e and pipefail to capture the correct exit code:
set +e +o pipefail
setsid timeout ... make ... 2>&1 | tee logfile
STATUS=${PIPESTATUS[0]}
set -e -o pipefail
Never use || true after a pipeline — it resets PIPESTATUS to 0 and masks all failures.
Environment isolation — Always unset SCRIPTS_DIR before calling ORFS make. ORFS Makefile uses this variable internally; an external value causes "No rule to make target synth.sh" errors.
Process group kills — Use setsid timeout (not bare timeout) so the entire process tree is killed on timeout. Without setsid, grandchild processes (klayout, openroad) survive as zombies, consuming memory and holding file locks.
LVS timeout scaling — Default LVS_TIMEOUT=3600 works for designs <100K cells. For swerv-class (~145K), use 4200s. For bp_multi_top-class (~200K), use 7200s. ORFS and RCX always complete within default timeouts.
After a successful ORFS run, the signoff scripts operate on the ORFS results in-place:
run_drc.sh — runs make DESIGN_CONFIG=<config.mk> drc in ORFS flow directory (KLayout)run_magic_drc.sh — runs Magic DRC with sky130A tech file (sky130 only)run_lvs.sh — runs make DESIGN_CONFIG=<config.mk> lvs in ORFS flow directory (KLayout)run_netgen_lvs.sh — extracts SPICE via Magic, compares with Netgen (sky130 only)run_rcx.sh — runs OpenROAD directly with a generated Tcl script reading 6_final.odbAll scripts collect results back to the project's drc/, lvs/, rcx/ directories and also copy them into the latest backend/RUN_*/ subdirectory.
npx claudepluginhub shenshan123/agent-r2gInternal routing reference for the RTL design pipeline. Maps natural language tasks to Action Skills and defines the agent delegation matrix for HDL/FPGA/ASIC workflows.
Orchestrates end-to-end nanodevice design from user query through flake detection, material analysis, geometry creation, routing, evaluation, and save. Device-agnostic methodology derives physics rules from device type and materials.