Dashboard + palette + analyze screen; consolidated categorization rules.
UI restructuring:
- Dashboard (`screen_dashboard.cpp`, `screen_idx = 6`) is the new home
screen at boot. Reads Overview / Health / Analysis / Modules from
the current System every frame; per-module rows list parts grouped
by `connector_type` and a Power/Gnd inference summary (yellow when
any name-Power signal is refuted). Scrollable via PgUp/PgDn/Home/End.
Letter shortcuts: `c`=console, `s`=search, `p`=plug (alias of
connect), `t`=set-type, `e`=explore, `n`=net, `a`=analyze, `q`=quit.
- Global Ctrl-P palette (`screen_palette.cpp`) — fuzzy-finds over
registered commands + module / signal names. Activation runs the
bare command or jumps to the matching screen with state seeded.
- Unified analyze screen (`screen_analyze.cpp`, `screen_idx = 7`):
tabbed layout (`Issues / Groups / Types`), Tab or ←→ to switch
tabs, ↑/↓ to navigate the focused list. Replaces the previous
shell-bouncing `[v]erify` shortcut — `verify` content is now in
the Issues tab. Types tab attaches the decision rationale to each
signal row (fan-out / voltage / hard floor).
- Context help panel: `RenderHelpPanel(title, entries)` in
`tui_helpers.{hpp,cpp}` rendered on the right of every screen.
- Console (former "log") rename: screen 0 is `[c]onsole` in the UI
and "console" in its help-panel title. The underlying screen and
the shell prompt are unchanged.
- Esc from any non-home screen returns to the dashboard. The
dashboard itself swallows Esc; quit via `q` / the `quit` command.
`quit` now calls `screen_ptr->Exit()` directly so it works from
any screen including via the palette.
Signal type inference:
- `Signal::type` defaults to `Other` — auto-inference no longer
happens at construction.
- `infer_signal_types(System*)` is called at the end of every load.
Three rules: GndShield from name alone; Power requires name match
+ a hard fan-out floor (< 3 pins = always Other, regardless of
name or voltage) + at least one positive structural signal
(fan-out ≥ 4 OR voltage pattern in the name like `3V3`, `5V`).
- Thresholds exposed in `analysis.hpp` (`POWER_FANOUT_HARD_FLOOR`,
`POWER_FANOUT_CONFIRM_MIN`, `has_voltage_pattern`) so the analyze
screen can render the same rationale without duplicating logic.
- `set-signal-type` still wins; save/restore round-trips the type.
Analysis groups & anomalies:
- New `GroupKind::DiffBus` — ≥ 2 diff pairs sharing the same
outer-stem with consecutive integer indices are aggregated into a
single bus (`MDI[0..3]_P/N`). `MDI0` and `PCIE_TX_0` index forms
both accepted. Solo pairs under a bus-able stem fall back to
`DiffPair`.
- New `AnomalyKind::DiffBusGap` for missing lanes.
Documentation:
- `DESIGN.md`: dedicated "Categorization rules (normative)" section
consolidating signal type, NC origin, signal groups, anomalies,
component kind, and connector wiring rules with exact thresholds
and decision order.
- `doc/user/analysis.md` (new): user-facing version of the same
rules in plain language. Linked from `doc/user/index.md`.
Tests: +6 new cases (62 total). Adjusted `test_persist.cpp` to set
the signal type explicitly in the fixture (no more auto-inference).
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
@@ -50,6 +50,67 @@ TEST_CASE("analyze detects diff pairs and reports `_P` orphans only") {
|
||||
CHECK(orphans == 1); // only PCIE_RX_P
|
||||
}
|
||||
|
||||
TEST_CASE("analyze aggregates diff pairs into a diff bus by trailing index") {
|
||||
auto sys = std::make_unique<System>();
|
||||
Module *m = sys->modules()->merge("M");
|
||||
Part *p = new Part("U1"); m->add(p);
|
||||
|
||||
add_signal(m, p, "MDI0_P"); add_signal(m, p, "MDI0_N");
|
||||
add_signal(m, p, "MDI1_P"); add_signal(m, p, "MDI1_N");
|
||||
add_signal(m, p, "MDI2_P"); add_signal(m, p, "MDI2_N");
|
||||
add_signal(m, p, "MDI3_P"); add_signal(m, p, "MDI3_N");
|
||||
|
||||
// Underscore-separated index variant.
|
||||
add_signal(m, p, "PCIE_TX_0_P"); add_signal(m, p, "PCIE_TX_0_N");
|
||||
add_signal(m, p, "PCIE_TX_1_P"); add_signal(m, p, "PCIE_TX_1_N");
|
||||
|
||||
// A lonely pair under a bus-able stem (only one index) must stay DiffPair.
|
||||
add_signal(m, p, "USB3_TX_P"); add_signal(m, p, "USB3_TX_N");
|
||||
|
||||
AnalysisReport r = analyze_system(sys.get());
|
||||
|
||||
int dp = 0, db = 0;
|
||||
bool mdi_found = false, pcie_found = false;
|
||||
for (const auto &g : r.groups) {
|
||||
if (g.kind == GroupKind::DiffPair) ++dp;
|
||||
if (g.kind == GroupKind::DiffBus) ++db;
|
||||
if (g.kind == GroupKind::DiffBus && g.label.find("MDI[") == 0) {
|
||||
mdi_found = true;
|
||||
CHECK(g.lo == 0);
|
||||
CHECK(g.hi == 3);
|
||||
CHECK(g.members.size() == 8); // 4 pairs × 2 signals
|
||||
}
|
||||
if (g.kind == GroupKind::DiffBus && g.label.find("PCIE_TX_[") == 0) {
|
||||
pcie_found = true;
|
||||
CHECK(g.lo == 0);
|
||||
CHECK(g.hi == 1);
|
||||
CHECK(g.members.size() == 4);
|
||||
}
|
||||
}
|
||||
CHECK(db == 2);
|
||||
CHECK(dp == 1); // USB3_TX kept as solo DiffPair (single index — degenerate bus)
|
||||
CHECK(mdi_found);
|
||||
CHECK(pcie_found);
|
||||
}
|
||||
|
||||
TEST_CASE("analyze flags a diff bus with a missing lane") {
|
||||
auto sys = std::make_unique<System>();
|
||||
Module *m = sys->modules()->merge("M");
|
||||
Part *p = new Part("U1"); m->add(p);
|
||||
|
||||
add_signal(m, p, "LANE0_P"); add_signal(m, p, "LANE0_N");
|
||||
add_signal(m, p, "LANE1_P"); add_signal(m, p, "LANE1_N");
|
||||
// LANE2 missing
|
||||
add_signal(m, p, "LANE3_P"); add_signal(m, p, "LANE3_N");
|
||||
|
||||
AnalysisReport r = analyze_system(sys.get());
|
||||
|
||||
int gaps = 0;
|
||||
for (const auto &a : r.anomalies)
|
||||
if (a.kind == AnomalyKind::DiffBusGap) ++gaps;
|
||||
CHECK(gaps == 1);
|
||||
}
|
||||
|
||||
TEST_CASE("analyze detects buses with bracketed and underscore forms") {
|
||||
auto sys = std::make_unique<System>();
|
||||
Module *m = sys->modules()->merge("M");
|
||||
@@ -160,6 +221,60 @@ TEST_CASE("analyze on empty / null system") {
|
||||
CHECK(r.anomalies.empty());
|
||||
}
|
||||
|
||||
TEST_CASE("infer_signal_types: Power requires name+structural agreement") {
|
||||
auto sys = std::make_unique<System>();
|
||||
Module *m = sys->modules()->merge("M");
|
||||
Part *p = new Part("U1"); m->add(p);
|
||||
|
||||
auto fan_out = [&](Signal *s, int n) {
|
||||
for (int i = 0; i < n; ++i) {
|
||||
Pin *pin = new Pin(s->name + "_" + std::to_string(i));
|
||||
p->add(pin); s->add(pin); pin->connect(s);
|
||||
}
|
||||
};
|
||||
|
||||
Signal *p_3v3 = m->signals->merge("PWR_3V3"); fan_out(p_3v3, 3); // voltage + ≥ floor → Power
|
||||
Signal *vcc = m->signals->merge("VCC"); fan_out(vcc, 5); // fan-out ≥ 4 → Power
|
||||
Signal *pwr_ok = m->signals->merge("PWR_OK"); fan_out(pwr_ok, 1); // < 3 → hard floor → Other
|
||||
Signal *pwr_2 = m->signals->merge("PWR_2"); fan_out(pwr_2, 2); // < 3 → hard floor → Other
|
||||
Signal *gnd = m->signals->merge("GND"); fan_out(gnd, 1); // gnd: name alone
|
||||
Signal *clk = m->signals->merge("CLK_50MHZ"); fan_out(clk, 3); // not power-ish → Other
|
||||
|
||||
auto st = infer_signal_types(sys.get());
|
||||
CHECK(st.power == 2); // PWR_3V3, VCC
|
||||
CHECK(st.gnd == 1); // GND (name alone)
|
||||
CHECK(st.kept_other == 2); // PWR_OK, PWR_2 below the hard floor
|
||||
|
||||
CHECK(p_3v3->type == SignalType::Power);
|
||||
CHECK(vcc->type == SignalType::Power);
|
||||
CHECK(gnd->type == SignalType::GndShield);
|
||||
CHECK(pwr_ok->type == SignalType::Other);
|
||||
CHECK(pwr_2->type == SignalType::Other);
|
||||
CHECK(clk->type == SignalType::Other);
|
||||
}
|
||||
|
||||
TEST_CASE("infer_signal_types: fan-out hard floor overrides voltage in name") {
|
||||
auto sys = std::make_unique<System>();
|
||||
Module *m = sys->modules()->merge("M");
|
||||
Part *p = new Part("U1"); m->add(p);
|
||||
|
||||
// VS_3V3 has a voltage pattern that would normally confirm Power, but
|
||||
// with only 2 pins it must still drop to Other because of the hard floor.
|
||||
Signal *s = m->signals->merge("VS_3V3");
|
||||
Pin *p1 = new Pin("p1"); p->add(p1); s->add(p1); p1->connect(s);
|
||||
Pin *p2 = new Pin("p2"); p->add(p2); s->add(p2); p2->connect(s);
|
||||
|
||||
auto st = infer_signal_types(sys.get());
|
||||
CHECK(st.power == 0);
|
||||
CHECK(st.kept_other == 1);
|
||||
CHECK(s->type == SignalType::Other);
|
||||
}
|
||||
|
||||
TEST_CASE("Signal ctor defaults type to Other (no auto-inference)") {
|
||||
Signal s("VCC");
|
||||
CHECK(s.type == SignalType::Other);
|
||||
}
|
||||
|
||||
TEST_CASE("analyze scopes detection per module (no cross-module merge)") {
|
||||
auto sys = std::make_unique<System>();
|
||||
Module *m1 = sys->modules()->merge("M1");
|
||||
|
||||
Reference in New Issue
Block a user