microReticulumTbeam/tools/constantTFCard/raw_probe/src/main.cpp

463 lines
12 KiB
C++

// 20260401 Codex
#include <Arduino.h>
#include <SD.h>
#include <SPI.h>
#include <Wire.h>
#include <U8g2lib.h>
#include <driver/gpio.h>
#include "tbeam_supreme_adapter.h"
#ifndef NODE_LABEL
#define NODE_LABEL "NODE"
#endif
#ifndef OLED_SDA
#define OLED_SDA 17
#endif
#ifndef OLED_SCL
#define OLED_SCL 18
#endif
static const uint32_t kSerialDelayMs = 1500;
static const uint32_t kPollIntervalMs = 200;
static const uint32_t kSpiHz = 400000;
static const uint32_t kReadyHeartbeatMs = 2000;
enum class RawState : uint8_t {
PMU_FAIL = 0,
RAIL_OFF,
BUS_FLOAT,
BUS_LOW,
CMD0_TIMEOUT,
CMD0_NOT_IDLE,
CMD8_TIMEOUT,
CMD8_BAD_R1,
ACMD41_TIMEOUT,
CMD58_TIMEOUT,
READY
};
struct PinSnapshot {
int cs = -1;
int sck = -1;
int miso = -1;
int mosi = -1;
};
struct ProbeSummary {
uint8_t ffCount = 0;
uint8_t zeroCount = 0;
uint8_t otherCount = 0;
uint8_t bytes[8] = {0};
};
struct RawSnapshot {
RawState state = RawState::PMU_FAIL;
bool pmuOk = false;
bool railOn = false;
PinSnapshot pins{};
ProbeSummary idle{};
uint8_t cmd0 = 0xFF;
uint8_t cmd8r1 = 0xFF;
uint8_t cmd8data[4] = {0xFF, 0xFF, 0xFF, 0xFF};
uint8_t acmd41 = 0xFF;
uint8_t cmd58r1 = 0xFF;
uint8_t ocr[4] = {0xFF, 0xFF, 0xFF, 0xFF};
};
static XPowersLibInterface* g_pmu = nullptr;
static U8G2_SH1106_128X64_NONAME_F_HW_I2C g_oled(U8G2_R0, U8X8_PIN_NONE);
static SPIClass g_spi(HSPI);
static uint32_t g_sampleCount = 0;
static uint32_t g_markCount = 0;
static char g_inputLine[32] = {0};
static uint8_t g_inputLen = 0;
static void forceSpiDeselected() {
pinMode(tbeam_supreme::sdCs(), OUTPUT);
digitalWrite(tbeam_supreme::sdCs(), HIGH);
pinMode(tbeam_supreme::imuCs(), OUTPUT);
digitalWrite(tbeam_supreme::imuCs(), HIGH);
}
static PinSnapshot readPins() {
PinSnapshot s;
s.cs = gpio_get_level((gpio_num_t)tbeam_supreme::sdCs());
s.sck = gpio_get_level((gpio_num_t)tbeam_supreme::sdSck());
s.miso = gpio_get_level((gpio_num_t)tbeam_supreme::sdMiso());
s.mosi = gpio_get_level((gpio_num_t)tbeam_supreme::sdMosi());
return s;
}
static void beginBus() {
SD.end();
g_spi.end();
delay(2);
forceSpiDeselected();
g_spi.begin(
tbeam_supreme::sdSck(),
tbeam_supreme::sdMiso(),
tbeam_supreme::sdMosi(),
tbeam_supreme::sdCs()
);
digitalWrite(tbeam_supreme::sdCs(), HIGH);
}
static ProbeSummary idleProbe() {
ProbeSummary out;
beginBus();
delay(1);
for (int i = 0; i < 8; ++i) {
uint8_t b = g_spi.transfer(0xFF);
out.bytes[i] = b;
if (b == 0xFF) out.ffCount++;
else if (b == 0x00) out.zeroCount++;
else out.otherCount++;
}
return out;
}
static uint8_t waitR1(uint16_t tries = 16) {
for (uint16_t i = 0; i < tries; ++i) {
uint8_t r = g_spi.transfer(0xFF);
if ((r & 0x80) == 0) {
return r;
}
}
return 0xFF;
}
static uint8_t sendCommand(uint8_t cmd, uint32_t arg, uint8_t crc) {
g_spi.transfer(0xFF);
digitalWrite(tbeam_supreme::sdCs(), LOW);
g_spi.transfer(0x40 | cmd);
g_spi.transfer((arg >> 24) & 0xFF);
g_spi.transfer((arg >> 16) & 0xFF);
g_spi.transfer((arg >> 8) & 0xFF);
g_spi.transfer(arg & 0xFF);
g_spi.transfer(crc);
uint8_t r1 = waitR1();
return r1;
}
static void endCommand() {
digitalWrite(tbeam_supreme::sdCs(), HIGH);
g_spi.transfer(0xFF);
}
static const char* stateToString(RawState state) {
switch (state) {
case RawState::PMU_FAIL: return "PMU_FAIL";
case RawState::RAIL_OFF: return "RAIL_OFF";
case RawState::BUS_FLOAT: return "BUS_FLOAT";
case RawState::BUS_LOW: return "BUS_LOW";
case RawState::CMD0_TIMEOUT: return "CMD0_TO";
case RawState::CMD0_NOT_IDLE: return "CMD0_BAD";
case RawState::CMD8_TIMEOUT: return "CMD8_TO";
case RawState::CMD8_BAD_R1: return "CMD8_BAD";
case RawState::ACMD41_TIMEOUT: return "ACMD41_TO";
case RawState::CMD58_TIMEOUT: return "CMD58_TO";
case RawState::READY: return "READY";
default: return "UNKNOWN";
}
}
static RawSnapshot captureSnapshot() {
RawSnapshot snap;
snap.pins = readPins();
snap.pmuOk = (g_pmu != nullptr);
if (!snap.pmuOk) {
snap.state = RawState::PMU_FAIL;
return snap;
}
snap.railOn = g_pmu->isPowerChannelEnable(XPOWERS_BLDO1);
if (!snap.railOn) {
snap.state = RawState::RAIL_OFF;
return snap;
}
snap.idle = idleProbe();
if (snap.idle.ffCount == 8) {
snap.state = RawState::BUS_FLOAT;
} else if (snap.idle.zeroCount == 8) {
snap.state = RawState::BUS_LOW;
}
beginBus();
delay(1);
for (int i = 0; i < 10; ++i) {
g_spi.transfer(0xFF);
}
snap.cmd0 = sendCommand(0, 0, 0x95);
endCommand();
if (snap.cmd0 == 0xFF) {
snap.state = RawState::CMD0_TIMEOUT;
return snap;
}
if (snap.cmd0 != 0x01) {
snap.state = RawState::CMD0_NOT_IDLE;
return snap;
}
snap.cmd8r1 = sendCommand(8, 0x000001AAUL, 0x87);
if (snap.cmd8r1 == 0xFF) {
endCommand();
snap.state = RawState::CMD8_TIMEOUT;
return snap;
}
for (int i = 0; i < 4; ++i) {
snap.cmd8data[i] = g_spi.transfer(0xFF);
}
endCommand();
if (!(snap.cmd8r1 == 0x01 || snap.cmd8r1 == 0x05)) {
snap.state = RawState::CMD8_BAD_R1;
return snap;
}
uint8_t ready = 0xFF;
for (int attempt = 0; attempt < 12; ++attempt) {
uint8_t r1 = sendCommand(55, 0, 0x65);
endCommand();
if (r1 == 0xFF) {
continue;
}
ready = sendCommand(41, 0x40000000UL, 0x77);
endCommand();
if (ready == 0x00) {
break;
}
delay(10);
}
snap.acmd41 = ready;
if (snap.acmd41 != 0x00) {
snap.state = RawState::ACMD41_TIMEOUT;
return snap;
}
snap.cmd58r1 = sendCommand(58, 0, 0xFD);
if (snap.cmd58r1 == 0xFF) {
endCommand();
snap.state = RawState::CMD58_TIMEOUT;
return snap;
}
for (int i = 0; i < 4; ++i) {
snap.ocr[i] = g_spi.transfer(0xFF);
}
endCommand();
snap.state = RawState::READY;
return snap;
}
static void printSnapshot(const RawSnapshot& snap) {
Serial.printf(
"sample=%lu state=%s rail=%s pins=%d/%d/%d/%d "
"idle(ff=%u z=%u o=%u %02X %02X %02X %02X) "
"cmd0=%02X cmd8=%02X [%02X %02X %02X %02X] "
"acmd41=%02X cmd58=%02X [%02X %02X %02X %02X]\r\n",
(unsigned long)g_sampleCount,
stateToString(snap.state),
snap.railOn ? "ON" : "OFF",
snap.pins.cs,
snap.pins.sck,
snap.pins.miso,
snap.pins.mosi,
(unsigned)snap.idle.ffCount,
(unsigned)snap.idle.zeroCount,
(unsigned)snap.idle.otherCount,
snap.idle.bytes[0],
snap.idle.bytes[1],
snap.idle.bytes[2],
snap.idle.bytes[3],
snap.cmd0,
snap.cmd8r1,
snap.cmd8data[0],
snap.cmd8data[1],
snap.cmd8data[2],
snap.cmd8data[3],
snap.acmd41,
snap.cmd58r1,
snap.ocr[0],
snap.ocr[1],
snap.ocr[2],
snap.ocr[3]
);
}
static void printReadyHeartbeat() {
Serial.printf("[%10lu] READY heartbeat\r\n", (unsigned long)millis());
}
static bool sameSnapshot(const RawSnapshot& a, const RawSnapshot& b) {
if (a.state != b.state) return false;
if (a.railOn != b.railOn) return false;
if (a.pins.cs != b.pins.cs || a.pins.sck != b.pins.sck ||
a.pins.miso != b.pins.miso || a.pins.mosi != b.pins.mosi) return false;
if (a.idle.ffCount != b.idle.ffCount ||
a.idle.zeroCount != b.idle.zeroCount ||
a.idle.otherCount != b.idle.otherCount) return false;
if (a.cmd0 != b.cmd0 || a.cmd8r1 != b.cmd8r1 ||
a.acmd41 != b.acmd41 || a.cmd58r1 != b.cmd58r1) return false;
for (int i = 0; i < 4; ++i) {
if (a.cmd8data[i] != b.cmd8data[i]) return false;
if (a.ocr[i] != b.ocr[i]) return false;
}
return true;
}
static void showSnapshot(const RawSnapshot& snap) {
char l1[24];
char l2[24];
char l3[24];
char l4[24];
char l5[24];
snprintf(l1, sizeof(l1), "%s RAW SD", NODE_LABEL);
snprintf(l2, sizeof(l2), "STATE %s", stateToString(snap.state));
snprintf(l3, sizeof(l3), "CMD0 %02X C8 %02X A41 %02X", snap.cmd0, snap.cmd8r1, snap.acmd41);
snprintf(l4, sizeof(l4), "OCR %02X%02X%02X%02X",
snap.ocr[0], snap.ocr[1], snap.ocr[2], snap.ocr[3]);
snprintf(l5, sizeof(l5), "IDL %u/%u/%u #%lu",
(unsigned)snap.idle.ffCount,
(unsigned)snap.idle.zeroCount,
(unsigned)snap.idle.otherCount,
(unsigned long)g_sampleCount);
g_oled.clearBuffer();
g_oled.setFont(u8g2_font_5x8_tf);
g_oled.drawUTF8(0, 12, l1);
g_oled.drawUTF8(0, 24, l2);
g_oled.drawUTF8(0, 36, l3);
g_oled.drawUTF8(0, 48, l4);
g_oled.drawUTF8(0, 60, l5);
g_oled.sendBuffer();
}
static void handleSerialInput() {
auto handleCommand = []() {
if (g_inputLen == 0) {
Serial.println();
return;
}
g_inputLine[g_inputLen] = '\0';
if (strcmp(g_inputLine, "m") == 0 || strcmp(g_inputLine, "M") == 0) {
g_markCount++;
Serial.printf("----- MARK %lu -----\r\n", (unsigned long)g_markCount);
} else if (strcmp(g_inputLine, "ls") == 0 || strcmp(g_inputLine, "LS") == 0) {
Serial.println("----- LS / -----");
beginBus();
for (int i = 0; i < 10; ++i) {
g_spi.transfer(0xFF);
}
if (!SD.begin(tbeam_supreme::sdCs(), g_spi, kSpiHz)) {
Serial.println("ls: SD.begin failed");
g_inputLen = 0;
return;
}
File root = SD.open("/", FILE_READ);
if (!root) {
Serial.println("ls: open / failed");
SD.end();
g_inputLen = 0;
return;
}
File entry = root.openNextFile();
if (!entry) {
Serial.println("ls: root empty or unreadable");
}
while (entry) {
Serial.printf("%s%s %lu\r\n",
entry.name(),
entry.isDirectory() ? "/" : "",
(unsigned long)entry.size());
entry.close();
entry = root.openNextFile();
}
root.close();
SD.end();
Serial.println("----- END LS -----");
} else {
Serial.printf("unknown command: %s\r\n", g_inputLine);
}
g_inputLen = 0;
};
while (Serial.available() > 0) {
int ch = Serial.read();
if (ch == '\r' || ch == '\n') {
handleCommand();
} else if (ch == 0x08 || ch == 0x7F) {
if (g_inputLen > 0) {
g_inputLen--;
}
} else if (g_inputLen + 1 < sizeof(g_inputLine)) {
g_inputLine[g_inputLen++] = (char)ch;
}
}
}
void setup() {
Serial.begin(115200);
delay(kSerialDelayMs);
Wire.begin(OLED_SDA, OLED_SCL);
g_oled.begin();
g_oled.clearDisplay();
tbeam_supreme::initPmuForPeripherals(g_pmu, &Serial);
forceSpiDeselected();
Serial.println();
Serial.println("constantTFCard raw probe");
Serial.printf("Node: %s\r\n", NODE_LABEL);
Serial.printf("Poll interval: %lu ms\r\n", (unsigned long)kPollIntervalMs);
Serial.println("States: PMU_FAIL RAIL_OFF BUS_FLOAT BUS_LOW CMD0_TO CMD0_BAD CMD8_TO CMD8_BAD ACMD41_TO CMD58_TO READY");
Serial.println("Input: Enter=blank line, m=mark, ls=list root");
}
void loop() {
static uint32_t lastPollMs = 0;
static bool haveLastPrinted = false;
static RawSnapshot lastPrinted{};
static uint32_t lastReadyHeartbeatMs = 0;
handleSerialInput();
uint32_t now = millis();
if (now - lastPollMs < kPollIntervalMs) {
delay(5);
return;
}
lastPollMs = now;
g_sampleCount++;
RawSnapshot snap = captureSnapshot();
bool changed = (!haveLastPrinted || !sameSnapshot(snap, lastPrinted));
if (changed) {
printSnapshot(snap);
lastPrinted = snap;
haveLastPrinted = true;
if (snap.state == RawState::READY) {
lastReadyHeartbeatMs = now;
}
} else if (snap.state == RawState::READY && now - lastReadyHeartbeatMs >= kReadyHeartbeatMs) {
printReadyHeartbeat();
lastReadyHeartbeatMs = now;
}
showSnapshot(snap);
}