# HelioX WORM 暂态训练 Demo（可打包分享版）

这个目录是一个“最小可运行”的训练 demo：

- **NEURON**：负责建模/导出（加载 MOD 机制、构建网络、导出 bbcore）
- **HelioX**：负责仿真与学习后端（forward 仿真 + replay 计算 `dw/dx` + 可选后端优化器）
- **Python**：负责目标函数（corr loss / `dL/dv`），便于用户替换 loss

本目录不包含历史变体（legacy cupy、旧脚本、杂乱运行输出等），只保留可运行所需的最小文件集与一份最佳训练记录。

## 目录结构

- `train.py`：Python 主入口
- `run.sh`：推荐 CLI 入口（更适合跑训练/续跑）
- `worm_*.py`：训练实现与后端绑定
- `x86_64/`：NEURON MOD 机制编译产物（运行时自动生成；不提交到 git）
- `components/`, `network/`, `utils/`：建模依赖数据与前端代码
- `data/trial10/`：基础 trial 输入（config/pkl/K）
- `seeds/`：0.125 预训练对（作为默认初始化，不依赖外部路径）
- `records/best_0p029828/`：历史最佳训练记录（包含 log/npy/ckpt）

## 运行前置条件

1) Python 环境

- 默认脚本会用 `PYTHON=python3`
- 你也可以自己指定：`PYTHON=/path/to/python`

建议在你要用的那个 Python 环境里安装本 demo 所需的常规依赖（不含 NEURON/HelioX 本体）：

```bash
$PYTHON -m pip install -r requirements.txt
```

说明：

- `opt_einsum` 主要用于加速部分张量收缩；如果没装也能跑（会自动回退到 `torch.einsum`，可能更慢）。
- `neuron`（NEURON Python 绑定）通常需要按你本机的 NEURON 安装方式来装，可能不是简单 `pip install` 就能搞定。

2) HelioX Python 库路径

你需要设置 `HELIOX_PYTHON_LIB` 指向 HelioX 的 `python_lib`（例如：`/path/to/heliox/python_lib`）：

```bash
export HELIOX_PYTHON_LIB=/path/to/heliox/python_lib
```

3) NEURON Python 绑定路径（可选）

如果你的 `python -c "import neuron"` 不能直接工作，但你本机有一个 NEURON 本地安装（例如 `$HOME/nrn/install`），可以设置：

```bash
export NRN_PYTHON_LIB=$HOME/nrn/install/lib/python
```

`run.sh` 会自动把它加到 `PYTHONPATH`（如果该目录存在）。

## 快速 smoke（推荐先跑这个）

```bash
cd /path/to/worm_demo_release
PYTHON=python3 HELIOX_PYTHON_LIB=/path/to/heliox/python_lib ./run.sh --epochs 1 --tstop-ms 200 --replay 1 --out ./runs/smoke --suffix smoke
```

成功标志：

- 运行结束生成 `./runs/smoke/log_eworm_smoke.txt`
- 输出目录下生成 `ckpt_*.npz` / `weights_train_*.npy` / `x_train_*.npy` / `error_*.npy`

## 从 0.125 初始化跑训练

### 新开训练（不 resume）

```bash
cd /path/to/worm_demo_release
PYTHON=python3 HELIOX_PYTHON_LIB=/path/to/heliox/python_lib ./run.sh --epochs 50 --tstop-ms 5000 --replay 1
```

### 续跑（resume）

```bash
cd /path/to/worm_demo_release
PYTHON=python3 HELIOX_PYTHON_LIB=/path/to/heliox/python_lib ./run.sh --epochs 50 --tstop-ms 5000 --replay 1 --out ./runs/your_run_dir --suffix from0125 --resume
```

### 续跑并重置 Adam（用于突破平台）

```bash
cd /path/to/worm_demo_release
PYTHON=python3 HELIOX_PYTHON_LIB=/path/to/heliox/python_lib ./run.sh --epochs 20 --tstop-ms 5000 --replay 1 --out ./runs/train_from0125 --suffix from0125 --resume --resume-reset-adam both
```

## 采样倍率与 K 历史窗口（可调）

本 demo 训练中有两类“时间尺度”：

- `dt`：NEURON/HelioX 仿真的基础步长（来自 `trial10/000_circuit_search_config.json`）
- `K_mul`：训练时的 LR 降采样倍率（每 `K_mul` 个 `dt` step 才采样/更新一次）

同时，`K` 是转移阻抗的历史窗口长度（决定“看多长的历史”）。三者关系是：

```
K_len_lr = floor(K_max_t_ms / (dt_ms * K_mul))
```

你可以通过两个参数控制 K（任选其一）：

- `--k-len <int>`：直接指定 `K_len_lr`（LR tick 数）。当你要对齐“原始模型的窗口长度”时，推荐用这个。
- `--k-max-t-ms <float>`：指定历史窗口的物理时间长度（ms）。当你要对齐某个生理时间尺度时，推荐用这个。

采样倍率参数：

- `--k-mul <int>`：LR 降采样倍率，`1` 表示“不降采样”（每个 `dt` step 都更新一次）。

提示：`--k-mul 1` 会显著增加 LR tick 数与部分中间缓存规模；如果遇到显存/内存不足，优先同时调小 `--k-len` 或 `--k-max-t-ms`（缩短历史窗口）。

### 与“原始 eworm_learn（无降采样版本）”对齐的建议参数

在原始 eworm_learn 脚本中，历史版本使用：

- `K_max_t = 80`（ms）
- 且没有 `K_mul`（等价于本 demo 的 `--k-mul 1`）

如果你的 `dt=0.5ms`（本 demo 自带的 `trial10` 默认就是 0.5ms），那么：

- `K_len = K_max_t / dt = 80 / 0.5 = 160`

因此，想对齐原始无降采样窗口时，推荐直接用：

- `--k-mul 1 --k-len 160`（最稳：固定 LR tick 数）

或等价地用：

- `--k-mul 1 --k-max-t-ms 80`

### K 缓存文件命名（用于避免冲突）

- 默认（与历史最佳设置一致、且不改 K 窗口）：
  - `K_eworm_v4_x{K_mul}.npz`
- 当你显式改变了 `K_max_t_ms` / `--k-len` 导致窗口不同时：
  - `K_eworm_v4_x{K_mul}_t{round(K_max_t_ms)}.npz`

说明：如果目标 K 文件不存在，代码会尝试从已有的 K 文件“重采样时间轴”快速合成（仅在历史窗口匹配时启用）；否则会回退到 NEURON 计算 K 并缓存。

## 最佳训练记录（参考）

历史最佳记录已打包在：

- `records/best_0p029828/`

其中包含：

- `log_eworm_from0125_heliox_plateau08.txt`
- `weights_optimal_*.npy`, `x_optimal_*.npy`
- `ckpt_*.npz`

你可以用它做回归对照或继续续跑，但**不要直接把 `--out` 指向 `records/`**（records 目录在 release 包里视为“只读存档”）。

推荐方式是：从 records 读取 checkpoint，但把新训练输出写到 `runs/`：

```bash
cd /path/to/worm_demo_release
PYTHON=python3 HELIOX_PYTHON_LIB=/path/to/heliox/python_lib ./run.sh --epochs 20 --tstop-ms 5000 --replay 1 --prefix eworm --suffix from0125_heliox_plateau08 --resume --resume-from ./records/best_0p029828 --out ./runs/resume_from_best_0p029828
```
