# HelioX Mech翻译教程 - AI版本

你好，我现在有一个兼容NEURON的神经元模拟器，需要你把NEURON模拟器中的Mech给翻译为HelioX可接受的格式。因此，我会详细地告诉你编写和翻译的要点。

## 教程概述

在你学习完如何编写HelioX的Mech后，我会给你NEURON中的.mod格式的Mech，以及其编译成CPP后的代码。然后你将要给我一个符合HelioX格式的Mech。

## 目录
1. [基础概念](#基础概念)
2. [MechTrait结构体定义](#mechtrait结构体定义)
3. [Mech标志位设置](#mech标志位设置)
4. [变量注册](#变量注册)
5. [POINTER变量（NEURON: POINTER）](#pointer变量neuron-pointer)
6. [Mech函数编写](#mech函数编写)
7. [特殊注意事项](#特殊注意事项)
8. [完整示例](#完整示例)

## 基础概念

Mech是基于C++/CUDA实现的，有以下几个关键特点：
- **命名空间隔离**：为了防止代码污染，需要包装在namespace中，推荐将namespace取名与Mech相同
- **枚举变量系统**：Mech中需要的变量均需要定义在一个MechTrait结构体中，里面可以定义四种不同的枚举类型，每种枚举类型对应一种类型的变量
- **核心特点**：**在HelioX框架中，最大的特点就是把枚举类型当变量来使用**

## MechTrait结构体定义

MechTrait结构体中可以定义四种枚举类型的变量：

### 1. VarNames（必需）
对应NEURON中的RANGE和STATE变量，每个节点都有独立的变量实例。

### 2. GlobalVarNames（可选）
对应NEURON中的GLOBAL变量，整个机制的所有节点共享一份变量。

### 3. IonVarNames（可选）
对应NEURON中的离子通道相关变量，这些变量实际挂载到全局的离子浓度变量上。

### 4. PointerVarNames（可选）
对应NEURON中的`POINTER`变量：它不是本机制自己的RANGE/STATE数据，而是“引用”到别处变量（例如别的机制变量、离子变量或`v`）。
在 HelioX/HELIOX 中，这些 POINTER 会在模型导入/初始化阶段解析为 `double*`，运行时通过 `vars.Ptr(...)` 直接读写目标值。

### 示例：

```cuda
struct  MechTrait{

enum  class  VarNames{//可以自由添加或修改变量

m, h, n,

ina, ik, il,

gnabar, gkbar, gl, gna, gk,

minf, hinf, ninf, mtau, htau, ntau,

ena, ek, el

};

enum  class  GlobalVarNames{

celsius

};

enum  class  IonVarNames{

_ion_ena,_ion_ina,_ion_ek,_ion_ik

};


};
```

## Mech标志位设置

编写好MechTrait定义后，第一件事是设置对应的flags。
需要在Mech类中定义：
```cpp
constexpr static MechFlags flags = ENABLE_INIT | ENABLE_CURRENT | POINT_PROCESS | ELECTRODE_CURRENT;
```
可用的flags有：
```
enum  class  MechFlags{
NONE  =  0,
ENABLE_INIT  =  1  <<  0,
ENABLE_CURRENT  =  1  <<  1,
ENABLE_STATE  =  1  <<  2,
POINT_PROCESS  =  1  <<  3,
ELECTRODE_CURRENT  =  1  <<  4,
WRITE_EION_IN_STATE  =  1  <<  5
};
```
### 标志位含义说明

#### 核心功能标志
- **ENABLE_INIT**：需要实现`init_single_node`函数（对应.mod中的INITIAL块）
- **ENABLE_CURRENT**：需要实现`current_single_node`函数（对应.mod中的BREAKPOINT块）  
- **ENABLE_STATE**：需要实现`state_single_node`函数（对应.mod中的STATE块）

#### 机制类型标志
- **POINT_PROCESS**：点过程机制（从.mod文件中`POINT_PROCESS`可以看出）
- **ELECTRODE_CURRENT**：外部电极注入电流（从.mod文件中`ELECTRODE_CURRENT`可以看出）

##### 特殊处理标志
- **WRITE_EION_IN_STATE**：如果STATE函数中修改了全局离子浓度变量，需要打上此标志

### 重要：关于ENABLE_INIT标志
**任何包含STATE变量的机制都必须设置ENABLE_INIT标志**，即使.mod文件中没有显式的INITIAL块。这是因为：

1. **NEURON编译器的默认行为**：当.mod文件包含STATE变量但没有INITIAL块时，NEURON会自动生成初始化代码，将所有STATE变量初始化为0.0
2. **HelioX的要求**：为了与NEURON保持一致，必须实现相应的`init_single_node`函数来进行初始化

### 如何确定标志位
可以从.mod文件的NEURON块中看出：
```NEURON {

POINT_PROCESS IClamp

RANGE del, dur, amp, i

ELECTRODE_CURRENT i

}
```
这代表是否为点过程，以及是否为外部电极注入的电流。父类模板中，会根据这个Flag来调整电流。

## 变量注册

接着，需要在类的构造函数中，实现变量与NEURON数据结构的关联。
需要注册的数据结构有：
```
map<VarNames, double> init_values;
map<VarNames, CoreIdxInfo> var_in_coredata_idx;

IonVarInfoMap<MechTrait> ion_var_map;
GlobalVarInfoMap<MechTrait> global_info_map;
```
其中，ion_var和global_var相关的Map是由宏定义生成的：
```
//定义一下可选枚举类，以及对应的信息map
DEFINE_HAS_ENUM(GlobalVarNames);
DEFINE_ENUM_MAP_ALIAS(GlobalVarNames,CoreGlobalVarInfo,GlobalVarInfoMap);
using  IonVarMapInfo  =  std::tuple<std::string, EionVarNames>;
DEFINE_HAS_ENUM(IonVarNames);
DEFINE_ENUM_MAP_ALIAS(IonVarNames,IonVarMapInfo, IonVarInfoMap);
```

这里的变量可分为三类，一类是普通变量（对应着NEURON中的Range和State变量），特点是Mech中的每个node都有一个独立的变量。 State变量和Range有一点不同的是，只要出现了State变量，必须带有init函数，State函数每次finitialize的时候需要初始化一次。而Range则不需要。
第二个是Global变量，这个变量的特点是，一个mech所有的node都共享一份变量
第三种是ion类的变量，这些变量的特点是，这实际上是挂载到全局的离子浓度变量上的，而不是专属于某一个mech的变量。
### 普通变量（RANGE或STATE）
这种变量在NEURON的CPP里面有如下特征：
在NEURON编译出的Mech CPP里，会出现这样的语句：
```cpp
#define  del  _ml->template  fpfield<0>(_iml)
#define  del_columnindex  0
#define  dur  _ml->template  fpfield<1>(_iml)
#define  dur_columnindex  1
#define  amp  _ml->template  fpfield<2>(_iml)
#define  amp_columnindex  2
```
例如这里就是声明del,dur,amp这三个变量

同时，对应的你也可以看到这代码：
```cpp
_nrn_mechanism_register_data_fields(_mechtype,
_nrn_mechanism_field<double>{"del"} /* 0 */,
_nrn_mechanism_field<double>{"dur"} /* 1 */,
_nrn_mechanism_field<double>{"amp"} /* 2 */,....后面省略);
```
这指明了各个变量在读入的模型文件中的下标，你可以看到，和上面的`dur  _ml->template  fpfield<1>(_iml)`中的数字1是一致的。
那么，在HelioX中，需要相应的写上这些下标：
```cpp
var_in_coredata_idx.insert({del,0});
var_in_coredata_idx.insert({dur,1});
var_in_coredata_idx.insert({amp,2});
```
特别的，有变量他是一个数组，此时可以看到
```cpp
#define  gbar  _ml->template  data_array<0, 2>(_iml)
#define  gbar_columnindex  0
...
_nrn_mechanism_register_data_fields(_mechtype,

_nrn_mechanism_field<double>{"gbar", 2} /* 0 */,...);
```
在HelioX中，需要这样来注册：
```cpp
var_in_coredata_idx.insert({gbar, {0,2}});//模型文件中的下标是0，数组长度是2
```

以及变量如果需要初始值，那么可以写这样的语句：
```
init_values.insert({gnabar,0.25});
init_values.insert({gl,0.00016666});
init_values.insert({el,-60.0});
```
### Global类型变量
在NEURON的MOD中，你可以看到：GLOBAL depth,cainf,taur
而在NEURON的CPP中，可以看到这是直接声明的一个变量，例如这的`cainf`（被宏扩展成了  `cainf_cad2_la`），随后，将这个变量与字符串`“cainf_cad2_la”`绑定
```cpp
#define  cainf  cainf_cad2_la
double  cainf  =  0.0001;
#define  depth  depth_cad2_la
double  depth  =  0.1;
#define  taur  taur_cad2_la
double  taur  =  80;
....
static DoubScal hoc_scdoub[] = {
{"depth_cad2_la", &depth_cad2_la},
{"taur_cad2_la", &taur_cad2_la},
{"cainf_cad2_la", &cainf_cad2_la},
{0, 0}
};
```
这种变量，需要这样注册，其中，字符串内容必须和NEURON中完全一致：
特别的，温度变量的名称是celsius，在NEURON中并没有声明，但是HelioX中需要手动声明
```cpp

        // 注册全局变量 - 后面的string需要和与NEURON完全一致
        global_info_map.insert({depth, {"depth_cad2_la"}});
        global_info_map.insert({tau, {"taur_cad2_la"}});
        //特别的，温度的名称是：
        global_info_map.insert({celsius,{"celsius"}});
```
### ion类型变量

有一种特殊的变量，他是从离子浓度机制中获取的，在NEURON的CPP中明显能看出差异：
```cpp
#define  _ion_ica  *(_ml->dptr_field<0>(_iml))
#define  _p_ion_ica static_cast<neuron::container::data_handle<double>>(_ppvar[0])
#define  _ion_cao  *(_ml->dptr_field<1>(_iml))
#define  _p_ion_cao static_cast<neuron::container::data_handle<double>>(_ppvar[1])
#define  _ion_cai  *(_ml->dptr_field<2>(_iml))
#define  _p_ion_cai static_cast<neuron::container::data_handle<double>>(_ppvar[2])
```
而后续的代码也有
```cpp
_nrn_mechanism_register_data_fields(_mechtype,
_nrn_mechanism_field<double>{"gamma"} /* 0 */,
....中间省略...
_nrn_mechanism_field<double*>{"_ion_ica", "ca_ion"} /* 0 */,
_nrn_mechanism_field<double*>{"_ion_cao", "ca_ion"} /* 1 */,
_nrn_mechanism_field<double*>{"_ion_cai", "ca_ion"} /* 2 */,
_nrn_mechanism_field<double*>{"_ion_ca_erev", "ca_ion"} /* 3 */,
```
在HelioX中这种变量需要这么注册：
```cpp
ion_var_map.insert({_ion_ica, {"ca_ion", EionVarNames::cur}});
ion_var_map.insert({_ion_cai, {"ca_ion", EionVarNames::conci}});
```
其中，EionVarNames有这些选项：
```
enum  class  EionVarNames
{//以na离子为例，对应着：
erev, //e[na]
conci,//[na]i
conco,//[na]o
cur,//i[na]
dcurdv//di[na]_dv
};
```

<a id="pointer变量neuron-pointer"></a>
## POINTER变量（NEURON: POINTER）

NEURON 的 `.mod` 机制里如果出现：

```NEURON
NEURON {
    POINT_PROCESS gapjunction_lr
    POINTER vpre
}
```

那么 `vpre` **不是** RANGE/STATE 变量（它通常不在 `_nrn_mechanism_register_data_fields` 的 `double` 列表里），而是“指向别处变量的引用”。

在 `pc.nrnbbcore_write(...)` 导出模型时，NEURON/CoreNEURON 通常会把 POINTER 的目标写进 `pdata/dparam`（整数索引形式），并在 `pointer2type` 里记录它指向的“目标类型”。HelioX/HELIOX 读入时会在两个阶段处理它：

1. **permute 阶段修正 pdata**：当 node/mech 实例被 permute（例如 permute_type=3）后，需要把 `pdata` 里存的“索引”同步更新，否则 POINTER 会指到错误对象。
2. **初始化后解析为真实地址**：当所有 node/离子变量/其他机制变量都分配好内存后，才把 POINTER 解析为 `double*`，并存入 VarStruct 的 pointer 表中，运行时直接解引用使用。

### 1) 在 MechTrait 里声明 PointerVarNames（可选）

把所有 `.mod` 里的 `POINTER` 变量列到 `PointerVarNames` 里（顺序要一致）：

```cpp
struct MechTrait {
    enum class VarNames { g, i };
    enum class PointerVarNames { vpre };
};
```

### 2) 注册 POINTER 对应的 dparam/pdata slot（非常重要）

HelioX/HELIOX 需要知道：这个机制的哪几个 `dparam/pdata` 槽位存的是 POINTER 目标（并且它们分别对应哪个 POINTER 变量）。

写法是：

```cpp
REGISTER_POINTER_DPARAM_SLOTS("gapjunction_lr", 2);
```

表示：这个机制的第 1 个 POINTER 变量（`PointerVarNames` 的第 0 个枚举值）对应 `dparam[2]`。

如果有两个 POINTER，比如 `POINTER ipre, gpre`，那么：

```cpp
REGISTER_POINTER_DPARAM_SLOTS("ptrdst2_ng", 2, 3);
```

#### 如何从 NEURON 生成的 C++ 找到 slot 编号？

优先看 **dparam semantics**（最稳健，布局变化也不怕）。在 nrnivmodl 生成的 C++ 里通常能看到类似的注册/数组：

- `*_dparam_semantics*` 数组/注册调用（里面会标出哪些 slot 是 `pointer`）
- 或者类似 `_nrn_mechanism_register_dparam_semantics(...)` 的地方

在本项目里建议你也同步注册 semantics（便于自动推断/调试），例如：

```cpp
#define DPSEM(x) dpsem(DparamSemantics::x)
REGISTER_DPARAM_SEMANTICS("gapjunction_lr", DPSEM(area), DPSEM(pntproc), DPSEM(pointer));
#undef DPSEM
```

经验值（仅供参考）：很多 `POINT_PROCESS` 的 `dparam` 结构是：

- `dparam[0] = area`
- `dparam[1] = pntproc`
- `dparam[2..] = POINTER...`

但不要盲猜，最好以生成的 C++ 为准。

### 3) 在 Mech 代码里使用 POINTER

运行时访问方式：

- `vars.Ptr(x)`：得到 `double&`（直接当作“被指向的变量值”使用）
- `vars.PtrPtr(x)`：得到 `double*`（需要指针本身时用）

例如（gap junction）：

```cpp
DUAL_EXEC double current_single_node(MechTempCurParam& param, VarAccessor<MechTrait>& vars) {
    const double vpre_val = vars.Ptr(vpre);
    vars(i) = vars(g) * (param.volt - vpre_val);
    return vars(i);
}
```

### 4) POINTER 可能指向什么？

在 bbcore 导入路径里，一个 POINTER 常见会指向：

- **node 级变量**：最常见是电压 `v`（通过 `pointer2type == -1` 表示）
- **其他机制变量**：例如指向另一个 POINT_PROCESS 的 `i`，或 density 机制的 `g_pas`
- **离子变量**：例如 `ena / ina / nai / nao / ...`

写机制时一般不需要区分“它指向的是哪种变量”，你只需要用 `vars.Ptr(...)` 当作一个 `double` 来读写即可；真正的语义解析和地址解析发生在导入/初始化阶段。

### 5) 重要：BBCOREPOINTER 不是普通 POINTER（遇到要停下来）

NEURON 里还有一种相关但更“强自定义”的声明：`BBCOREPOINTER`（dparam semantics 通常为 `bbcorepointer`）。它通常意味着：

- 机制作者在做某种 **Hack/强自定义对象引用**（例如保存 HOC 对象指针、内部自建结构体、非 `double` 数组指针等）
- 这类东西**无法**靠通用的 “(pointer2type + pdata index) -> double*” 方式解析
- 正确导入/导出往往依赖机制自己实现 `bbcore_write/bbcore_read`（把自定义状态序列化/反序列化到 `iArray/dArray`）

本项目里常见的 `VecStim/vecevent` 已经做了特化处理；但如果你在其它 `.mod` 里看到 `BBCOREPOINTER`，基本应当判定为“需要特例分析”，而不是通用翻译能自动搞定：请停止并向用户报告该机制需要手工处理/特化实现。

### 变量的使用

在Mech的函数中，会传入一个特殊的类`VarAccessor<MechTrait> &vars`，配合`using enum`语句后，就可以用`vars(name)`的方式来访问对应的变量，例如：
```c
vars(m) =  vars(m) + (1.0  -  exp(dt * (-1.0  / _mtau))) * ((_minf / _mtau) / (1.0  / _mtau) -  vars(m));
vars(h) =  vars(h) + (1.0  -  exp(dt * (-1.0  / _htau))) * ((_hinf / _htau) / (1.0  / _htau) -  vars(h));
vars(n) =  vars(n) + (1.0  -  exp(dt * (-1.0  / _ntau))) * ((_ninf / _ntau) / (1.0  / _ntau) -  vars(n));
```
不过，有一些其他的特殊变量，例如dt,t,volt等，是通过`MechTempStateParam &param`等变量传入的，使用的时候是`param.t`这样的使用方法，具体变量见mechtemplate.cuh实现。

以及，在访问**离子浓度**类型的变量时，如果需要给全局离子浓度进行加法，需要像这样调用原子加法`mechAtomAdd(&vars(_ion_ina), vars(ina));`。在后续部分会继续介绍

## Mech函数编写

根据设置的标志位，需要实现对应的函数。HelioX有三个核心函数：
 ```cpp
 DUAL_EXEC  void  init_single_node(MechTempInitParam &param, VarAccessor<MechTrait>  &vars);
 DUAL_EXEC  double  current_single_node(MechTempCurParam &param, VarAccessor<MechTrait>  &vars);
DUAL_EXEC  void  state_single_node(MechTempStateParam &param, VarAccessor<MechTrait>  &vars);
```

### 函数实现要点

**单节点处理**：与NEURON不同的是，NEURON中的代码往往是一个for循环对每个node进行计算，而在HelioX中只需实现单个Node的计算，for循环在父类模板中已实现。

### init_single_node函数的实现

对于包含STATE变量的机制，`init_single_node`函数的实现需要遵循以下规则：

#### 1. 默认初始化规则
如果.mod文件中**没有INITIAL块**，则：
- 所有STATE变量初始化为 **0.0**
- 如果有稳态值计算（如mInf, hInf），则STATE变量可能需要初始化为稳态值

#### 2. 实现示例

**情况A：无INITIAL块，简单初始化**
```cpp
DUAL_EXEC void init_single_node(MechTempInitParam &param, VarAccessor<MechTrait> &vars) {
    // 将所有STATE变量初始化为0（对应NEURON编译后的默认行为）
    vars(m) = 0.0;
    vars(h) = 0.0;
    vars(n) = 0.0;
}
```

**情况B：有稳态值计算的初始化**
```cpp
DUAL_EXEC void init_single_node(MechTempInitParam &param, VarAccessor<MechTrait> &vars) {
    // 先初始化为0（对应NEURON的默认行为）
    vars(m) = 0.0;
    vars(h) = 0.0;
    
    // 如果NEURON的initmodel中计算了稳态值并赋值给STATE变量
    // 则需要调用相应的计算函数并赋值
    // 例如：计算rates，然后设置为稳态值
    // rates_function_equivalent(param.volt, vars);
    // vars(m) = vars(mInf);
    // vars(h) = vars(hInf);
}
```

#### 3. 如何确定初始化策略
查看NEURON编译后的CPP文件中的`initmodel`函数：
- 如果只有 `m = m0; h = h0;` 这样的语句，且m0、h0定义为0，则用情况A
- 如果还有额外的计算和赋值（如`m = mInf;`），则用情况B

只要在类中声明了 `using  enum  VarNames;`后，便可在函数内直接访问枚举类型。
如果出现了函数传入的变量名称，或者是中间结果变量名称和枚举类中声明的变量类型重复了的情况，需要修改一下变量名称。枚举的变量名永远有最高优先级。

与NEURON不同的是，例如t,v(volt)等某些变量，是函数调用的时候通过对应的`MechTempXXParam`传进来的，因此不需要在VarNames中声明，这部分代码可以简化。

同时，HelioX中，只有这三个关键的函数需要实现。而NEURON中可能会实现一些无关的函数或变量。用不着的变量也可以简化。

所有的变量均可用`vars(enum_name)`来访问

```cpp
DUAL_EXEC  double  current_single_node(MechTempCurParam &param, VarAccessor<MechTrait>  &vars)
{
vars(eca) =  vars(_ion_eca);
double current =  ca_hva_cal_current(param.volt,
vars(gbar), vars(m), vars(h), vars(eca),
vars(g), vars(ica), vars(i_mech));
if (param.updateIon)
{
mechAtomAdd(&vars(_ion_ica), vars(ica));
}
return current;
}
```

## 特殊注意事项

### 离子浓度变量处理
对于离子浓度类型的变量，有几个重要注意点：

1. **原子操作**：如果要给离子浓度变量加值，必须使用`mechAtomAdd`函数：
   ```cpp
   mechAtomAdd(&vars(_ion_ina), vars(ina));
   ```

2. **Current函数的两次调用**：在current函数中，只有当`param.updateIon == true`时才能更新全局离子浓度。因为current函数会被调用两次来计算导数，只有第二次才是真正的计算。

3. **STATE函数中的离子浓度修改**：如果STATE函数中修改了全局离子浓度（通常是赋值操作），需要在MechFlags中加上`WRITE_EION_IN_STATE`标志。

## 完整示例

以下是一个完整的HH类型离子通道机制的HelioX实现示例：

```cpp
#pragma once
#include "mech_template.cuh"
#include <cstdio>
#include <cmath>

namespace HH_Example {

struct MechTrait {
    enum class VarNames {
        // STATE变量
        m, h, n,
        // 电流
        ina, ik, il,
        // 参数
        gnabar, gkbar, gl, el,
        // 离子反转电位（本地副本）
        ena, ek
    };
    
    enum class IonVarNames {
        _ion_ena, _ion_ina, _ion_ek, _ion_ik
    };
};

class HH_Channel : public MechTemp<HH_Channel, MechTrait> {
public:
    using enum MechTrait::VarNames;
    using enum MechTrait::IonVarNames;
    
    // 根据机制特点设置标志
    constexpr static MechFlags flags = ENABLE_INIT | ENABLE_CURRENT | ENABLE_STATE;
    
    HH_Channel(MechInitParams &param) : MechTemp(param) {
        // 设置默认值
        init_values.insert({gnabar, 0.12});
        init_values.insert({gkbar, 0.036});
        init_values.insert({gl, 0.0003});
        init_values.insert({el, -54.3});
        
        // 注册变量索引（按NEURON CPP中的顺序）
        var_in_coredata_idx.insert({gnabar, 0});
        var_in_coredata_idx.insert({gkbar, 1});
        var_in_coredata_idx.insert({gl, 2});
        var_in_coredata_idx.insert({el, 3});
        var_in_coredata_idx.insert({m, 4});
        var_in_coredata_idx.insert({h, 5});
        var_in_coredata_idx.insert({n, 6});
        
        // 注册离子通道变量
        ion_var_map.insert({_ion_ena, {"na_ion", EionVarNames::erev}});
        ion_var_map.insert({_ion_ina, {"na_ion", EionVarNames::cur}});
        ion_var_map.insert({_ion_ek, {"k_ion", EionVarNames::erev}});
        ion_var_map.insert({_ion_ik, {"k_ion", EionVarNames::cur}});
    }
    
    // 初始化函数：将STATE变量初始化为0
    DUAL_EXEC void init_single_node(MechTempInitParam &param, VarAccessor<MechTrait> &vars) {
        vars(m) = 0.0;
        vars(h) = 0.0;
        vars(n) = 0.0;
    }
    
    // 电流计算函数
    DUAL_EXEC double current_single_node(MechTempCurParam &param, VarAccessor<MechTrait> &vars) {
        // 读取离子反转电位
        vars(ena) = vars(_ion_ena);
        vars(ek) = vars(_ion_ek);
        
        // 计算电流
        vars(ina) = vars(gnabar) * vars(m)*vars(m)*vars(m) * vars(h) * (param.volt - vars(ena));
        vars(ik) = vars(gkbar) * vars(n)*vars(n)*vars(n)*vars(n) * (param.volt - vars(ek));
        vars(il) = vars(gl) * (param.volt - vars(el));
        
        // 更新全局离子电流（仅在updateIon=true时）
        if (param.updateIon) {
            mechAtomAdd(&vars(_ion_ina), vars(ina));
            mechAtomAdd(&vars(_ion_ik), vars(ik));
        }
        
        return vars(ina) + vars(ik) + vars(il);
    }
    
    // 状态更新函数
    DUAL_EXEC void state_single_node(MechTempStateParam &param, VarAccessor<MechTrait> &vars) {
        // 计算速率函数（按照原始mod文件的公式）
        double alpha_m = 0.1 * (param.volt + 40.0) / (1.0 - exp(-(param.volt + 40.0) / 10.0));
        double beta_m = 4.0 * exp(-(param.volt + 65.0) / 18.0);
        double alpha_h = 0.07 * exp(-(param.volt + 65.0) / 20.0);
        double beta_h = 1.0 / (1.0 + exp(-(param.volt + 35.0) / 10.0));
        double alpha_n = 0.01 * (param.volt + 55.0) / (1.0 - exp(-(param.volt + 55.0) / 10.0));
        double beta_n = 0.125 * exp(-(param.volt + 65.0) / 80.0);
        
        // 更新状态变量
        vars(m) += param.dt * (alpha_m * (1.0 - vars(m)) - beta_m * vars(m));
        vars(h) += param.dt * (alpha_h * (1.0 - vars(h)) - beta_h * vars(h));
        vars(n) += param.dt * (alpha_n * (1.0 - vars(n)) - beta_n * vars(n));
    }
};

REGISTER_MECHANISM("hh", HH_Channel);

} // namespace HH_Example
```

## 重要提醒

### 突触机制的限制
**通用的 Mech 模板（MechTemp）不支持突触/事件机制**，即包含 `NET_RECEIVE/net_receive` 的机制——这种机制不要按本文的 “RANGE/STATE + current/state” 通用翻译路径来做。

但需要注意：HELIOX 已经有一套**独立的 spike 子系统**（见 `src/spike/`）可以覆盖“部分”突触/事件机制，因此这里不是“一概不支持”，而是**需要分流**：

1) **只收不发（receiver-only）的突触**（最常见，例如 `ExpSyn/Exp2Syn` 这种 `NET_RECEIVE(w)` 只更新状态，不 `net_send`）：
   - 优先复用 `src/spike/postsyn_mech/` 里已有实现（如 `exp2syn`）。
   - 或者用 `src/spike/postsyn_template.cuh` 的 `PostSynTemplate` 方式手写一个“简化版翻译”（写 `net_receive_single_node`，框架负责按事件列表调用）。

2) **会在 NET_RECEIVE 里继续调度事件/发新事件的机制**（例如 `VecStim/NetStim` 这类 ARTIFICIAL_CELL）：
   - 这类机制通常需要特化实现（本项目里 `VecStim`/`NetStim` 已内置在 `src/spike/`，不是通用翻译能自动覆盖的）。
   - 虽然 HELIOX 的模拟循环对人工细胞支持“在 net_receive 中继续 net_send 并在同一时间步内反复处理”，但这属于 spike 子系统的特化行为，不代表任意包含复杂 `NET_RECEIVE` 的 mod 都能通用翻译。

3) **遇到复杂 NET_RECEIVE 语义时应当停止**：如果 `NET_RECEIVE` 里包含 `net_send/net_event/WATCH/RANDOM`、依赖复杂 `flag` 分支、或强依赖 HOC/对象指针等自定义行为，请直接告知用户需要手工实现/特例分析（并且机制代码应放到 `src/spike/` 体系下），而不是继续按通用教程翻译。

### 翻译原则
1. **以CPP为准**：翻译时应以NEURON编译生成的CPP代码为准，mod文件仅供理解机制行为
2. **保持一致性**：目标是让HelioX的运行结果与NEURON完全一致
3. **仔细检查变量索引**：变量在coredata中的索引必须与NEURON CPP中的完全一致
4. **机制注册名称必须一致**：在`REGISTER_MECHANISM`中使用的字符串必须与mod文件中`SUFFIX`指定的名称完全一致，否则会导致模型解析失败

#### 机制名称一致性示例
```mod
NEURON {
    SUFFIX egl19    // mod文件中的SUFFIX
    ...
}
```
对应的注册代码必须是：
```cpp
REGISTER_MECHANISM("egl19", EGL19_Channel);  // 字符串必须与SUFFIX一致
```
**错误示例**：
```cpp
REGISTER_MECHANISM("EGL19", EGL19_Channel);     // 大小写不一致 - 错误！
REGISTER_MECHANISM("egl19_worm", EGL19_Channel); // 名称不一致 - 错误！
```

现在你已经掌握了HelioX Mech翻译的所有要点。在后续翻译过程中，请严格按照以上规则进行。
