36 KiB
Executable File
1. 背景与现象
- 设备:Google Kaisa(Acer CXI4 Chromebox,基于 Puff baseboard)
- 固件:coreboot + Chrome EC
- 主要问题:
- Windows 下曾出现热事件关机;
- EC 控扇仅依赖一颗板载热敏电阻,温度响应滞后于 CPU,风扇启动与拉满偏晚。
本文整理 Puff/Kaisa 上 温度传感器、EC 温控、DPTF 控扇 的整体路径,并记录调试 DPTF FAND/_FSL 时的关键发现。
2. 温度与 EC 的关系概览
-
CPU/主机侧温度:
- Linux:
/sys/class/thermal/thermal_zone*中的TCPU、x86_pkg_temp等,由 CPU/PCH 提供,不经过 EC。 - Windows:DPTF 通过 ACPI 热区(如
\_SB.PCI0.TCPU)取 CPU 温度。
- Linux:
-
EC 侧温度(Puff/Kaisa):
- 只有 1 路板载 thermistor:
- ADC:
ADC_TEMP_SENSOR_1(NPCX_ADC_CH0) - 逻辑 ID:
TEMP_SENSOR_CORE - 读函数:
get_temp_3v3_30k9_47k_4050b
- ADC:
- EC 每秒读这路 ADC,写入 EC memory map,对外通过 ACPI 暴露为
TSR0。
- 只有 1 路板载 thermistor:
结论:
目前板上可用的温感来源只有两类:CPU 自身温度 + EC 的一颗板载 thermistor,没有额外硬件温感。
3. EC 控扇链路(Puff/Kaisa)
3.1 板级配置(ec/board/puff/board.c & board.h)
- 温度枚举:
enum temp_sensor_id {
TEMP_SENSOR_CORE,
TEMP_SENSOR_COUNT
};
- ADC 与
temp_sensors[]:
[ADC_TEMP_SENSOR_1] = {
.name = "TEMP_SENSOR_1",
.input_ch = NPCX_ADC_CH0,
.factor_mul = ADC_MAX_VOLT,
.factor_div = ADC_READ_MAX + 1,
},
const struct temp_sensor_t temp_sensors[] = {
[TEMP_SENSOR_CORE] = {
.name = "Core",
.type = TEMP_SENSOR_TYPE_BOARD,
.read = get_temp_3v3_30k9_47k_4050b,
.idx = ADC_TEMP_SENSOR_1,
},
};
- EC 的温阈值与风扇曲线(
thermal_params[]):
const static struct ec_thermal_config thermal_a = {
.temp_host = {
[EC_TEMP_THRESH_WARN] = 0,
[EC_TEMP_THRESH_HIGH] = C_TO_K(68),
[EC_TEMP_THRESH_HALT] = C_TO_K(78),
},
.temp_host_release = {
[EC_TEMP_THRESH_WARN] = 0,
[EC_TEMP_THRESH_HIGH] = C_TO_K(58),
[EC_TEMP_THRESH_HALT] = 0,
},
.temp_fan_off = C_TO_K(40), // <40℃ 关风扇
.temp_fan_max = C_TO_K(60), // ≥60℃ 满速
};
struct ec_thermal_config thermal_params[] = {
[TEMP_SENSOR_CORE] = thermal_a,
};
3.2 通用 thermal 引擎(ec/common/thermal.c)
-
每秒执行一次
thermal_control():- 遍历所有温度传感器(这里只读
TEMP_SENSOR_CORE):for (i = 0; i < TEMP_SENSOR_COUNT; ++i) rv = temp_sensor_read(i, &t); - 根据
thermal_params[i].temp_host[]与temp_host_release[]统计 WARN/HIGH/HALT,触发:chipset_force_shutdown(CHIPSET_SHUTDOWN_THERMAL);throttle_ap(THROTTLE_ON, THROTTLE_HARD/SOFT, THROTTLE_SRC_THERMAL);
- 根据
temp_fan_off和temp_fan_max计算该传感器需要的风扇百分比:f = thermal_fan_percent(thermal_params[i].temp_fan_off, thermal_params[i].temp_fan_max, t); fmax = max(fmax, f); - 对所有风扇设置统一需求:
for (i = 0; i < fan_get_count(); i++) fan_set_percent_needed(i, fmax);
- 遍历所有温度传感器(这里只读
结论:
在 “EC 自动控扇” 模式下,风扇完全由 EC 根据这路 thermistor + thermal_a 曲线驱动。
4. DPTF 与 EC 的桥接(ACPI/DPTF 侧)
4.1 温度:TSR0..TSR4
- coreboot 的
ec_dptf_helpers.c为 DPTF temp participants 生成_TMP:
/* TSRx._TMP = EC0.TSRD(x) */
acpigen_write_method_serialized("_TMP", 0);
acpigen_emit_byte(RETURN_OP);
acpigen_emit_namestring(acpi_device_path_join(ec, "TSRD"));
acpigen_write_integer(tsr_index); // 0..4
ec.asl中:
Method (TINS, 1, Serialized)
{
Switch (ToInteger (Arg0)) {
Case (0) { Return (TIN0) }
Case (1) { Return (TIN1) }
...
Default { Return (TIN0) }
}
}
Method (TSRD, 1, Serialized)
{
Local0 = \_SB.PCI0.LPCB.EC0.TINS (Arg0)
/* 检查 TNCA/TNPR/TNOP/TBAD 省略 */
Local0 += \_SB.PCI0.LPCB.EC0.TOFS
Local0 *= 10 // 转为 1/10 K
Return (Local0)
}
- Puff 上:只有
TEMP_SENSOR_CORE→TIN0→TSR0,其它 TSR1..TSR4 基本为空。
4.2 风扇:TFN1 + FST/FSL + FAND
根据 DPTF 规范,FAN participant(TFN1)需要提供:
_FST:返回{ Revision, Control, Speed };_FSL(level):设置 fan level/duty。
Chrome EC 路径(ec_dptf_helpers.c):
- 写回包
TFST:
/* TFST = { Revision, Control, Speed } */
acpigen_write_name("TFST");
acpigen_write_package(3);
acpigen_write_integer(0); /* Revision */
acpigen_write_integer(0); /* Control */
acpigen_write_integer(0); /* Speed */
acpigen_pop_len(); /* Package */
_FST实现大致为:
/* _FST: TFST[1] = EC0.FAND; TFST[2] = FANx_RPM; return TFST */
acpigen_write_method_serialized("_FST", 0);
acpigen_write_store();
acpigen_emit_namestring(acpi_device_path_join(ec, "FAND"));
acpigen_emit_byte(INDEX_OP);
acpigen_emit_namestring("TFST");
acpigen_write_integer(1); // Control
...
acpigen_write_store();
acpigen_emit_namestring(acpi_device_path_join(ec, "FAN0")); // 实际 RPM 来源
acpigen_emit_byte(INDEX_OP);
acpigen_emit_namestring("TFST");
acpigen_write_integer(2); // Speed
...
acpigen_emit_byte(RETURN_OP);
acpigen_emit_namestring("TFST");
acpigen_pop_len();
_FSL(level)实现:
/* _FSL(Arg0): Store(Arg0, EC0.FAND) */
acpigen_write_method_serialized("_FSL", 1);
acpigen_write_store();
acpigen_emit_byte(ARG0_OP);
acpigen_emit_namestring(acpi_device_path_join(ec, "FAND"));
acpigen_pop_len();
其中 FAND 对应 EC ACPI RAM 中地址 0x04(EC_ACPI_MEM_FAN_DUTY),在 ec_commands.h 定义为:
/* DPTF Target Fan Duty (0-100, 0xff for auto/none) */
#define EC_ACPI_MEM_FAN_DUTY 0x04
ACPI 通过 EC_ACPI_MEM_FAN_DUTY 写入后,EC C 侧(ec/common/acpi.c)处理:
case EC_ACPI_MEM_FAN_DUTY:
dptf_set_fan_duty_target(data);
break;
dptf_set_fan_duty_target() 实现(ec/common/fan.c):
/* 0-100% sets duty, out of range means let the EC drive */
void dptf_set_fan_duty_target(int pct)
{
int fan;
if (pct < 0 || pct > 100) {
/* 超出范围:交回 EC 自动控扇 */
for (fan = 0; fan < fan_count; fan++)
set_thermal_control_enabled(fan, 1);
} else {
/* 合法 0-100:按 duty 控扇,关闭 thermal 自动控扇 */
for (fan = 0; fan < fan_count; fan++)
set_duty_cycle(fan, pct);
}
}
读取时,若处于 duty 模式,dptf_get_fan_duty_target() 返回 0–100,否则返回 -1,对应 FAND=0xff。
结论(非常重要):
- 写入 0–100:DPTF 接管风扇,EC 用该 duty 控扇,thermal 自动模式被关闭;
- 写入 0xff 或越界值(<0 或 >100,例如 0x80):退回 EC 自动控扇模式,FAND 对外显示为 0xff。
5. 模式选择:EC 控扇 vs Windows DPTF 控扇
5.1 EC 控扇模式(稳定默认)
coreboot defconfig 中:
CONFIG_MINIPC_EC_FULL_FAN_CONTROL=y
CONFIG_DRIVERS_INTEL_DPTF_DISABLE_ACTIVE_FAN=y
- DPTF 风扇参与者被隐藏/禁用 Active Policy;
- EC 在
_REG时写FAND=0xFF,明确进入自动模式; - 风扇完全由 EC thermal 引擎(第 3 节)驱动。
适用:
- 无 Windows DPTF 驱动或不希望依赖 DPTF;
- 简化行为,只调一条 EC 曲线。
5.2 Windows DPTF 控扇模式(实验/优化)
coreboot defconfig 中:
# CONFIG_MINIPC_EC_FULL_FAN_CONTROL is not set
# CONFIG_DRIVERS_INTEL_DPTF_DISABLE_ACTIVE_FAN is not set
CONFIG_DRIVERS_INTEL_DPTF_ECTST_FAND_DEBUG=y
- DPTF FAN(TFN1)暴露给 Windows;
- Active Policy 启用,DPTF 根据 CPU 温度 + TSR0 等决定何时
_FSL(level); - EC 通过
dptf_set_fan_duty_target(level)接收 duty:- 0–100:DPTF 覆盖;
- 0xff/越界:交回 EC 自动。
前提:
Windows 中必须安装并启用 Intel DPTF 驱动(INT3400/INT3404 等设备正常工作),否则不会按策略调用 _FSL()。
6. ECT0 调试:为什么 _FSL(0x80) 后 FAND 一直是 0xff?
在 Windows 下,使用 ECT0._FST/_FSL 调试 FAND(logs.txt):
[1] Current FAND from _FST:
FAND = 0x46 # 70%,说明曾经处于 duty 模式
[2] Writing FAND via _FSL(0x80) ...
[3] Read-back FAND after _FSL:
FAND = 0xff
[4] 继续监视 _FST:
FAND 始终是 0xff
结合第 4 节的实现:
- 0x80 = 128 > 100 → 按
dptf_set_fan_duty_target()的语义,这是“越界值”:- 进入
pct < 0 || pct > 100分支; - 调用
set_thermal_control_enabled(fan, 1),恢复 EC 自动控扇; dptf_get_fan_duty_target()返回 -1,对 ACPI 显示为 FAND=0xff。
- 进入
正确结论:
- FSL/FST → EC 的链路是通的,只是你写入的是一个“退回自动”的值(>100),EC 正确地退出了 DPTF duty 模式;
_FST一直看到 FAND=0xff,说明当前处于 “auto/none” 状态,而不是链路失效。
正确的测试方式:
-
要验证 DPTF/ECT0 控扇是否生效,应使用 0–100 的值:
ectest -acpi \_SB.ECT0._FSL 50 # 50% ectest -acpi \_SB.ECT0._FST # FAND 应为 0x32,且风扇转速有变化 -
要主动退回 EC 自动控扇,可以写 0xff 或 >100(例如 0x80):
13.7 用其他方法监控 DPTF 是否发送风扇控制信号
除轮询 \_SB.ECT0._FST 看 FAND 是否随温度变化外,可用脚本 Monitor_DPTF_Fan.ps1 做两类监控(需管理员、ectest 可用):
| 模式 | 说明 |
|---|---|
| check | 检查本机是否有 DPTF/ESIF 服务、ACPI 设备 INT3404/INT3400、相关驱动,判断 DPTF 栈是否加载。 |
| poll | 高频率轮询 _FST(默认每 200ms),将时间戳与 FAND 写入 CSV,持续一段时间(默认 60s)。对 CPU 加压时若 CSV 中 FAND 始终不变,可判断 DPTF 未通过 _FSL 写 FAND。 |
| watch | 专用于监控 INT3404 是否发送控扇信号:轮询 _FST,仅在 FAND 发生变化时在控制台打印(并写 CSV)。若加压过程中出现 “FAND changed: 0xff -> 0x50” 等,说明有对象(多为 INT3404/DPTF)写入了 _FSL;若始终无输出,则 INT3404 未写 _FSL 或 EC 覆盖了写入。 |
| etw | 用 logman 采集 DPTF/ESIF 的 ETW(EsifLfEtwProvider 等),记录一段时间后 tracerpt 转 CSV。若已装 Intel DPTF 驱动,可查看是否有风扇/ACPI 相关事件。 |
| all | 先 check,再运行 etw(按提示加压后回车停止),最后做 poll 并落盘。 |
用法示例:
.\Monitor_DPTF_Fan.ps1 -Mode check
.\Monitor_DPTF_Fan.ps1 -Mode watch -PollSeconds 120 -PollIntervalMs 150 # 监控 INT3404 是否发控扇
.\Monitor_DPTF_Fan.ps1 -Mode poll -PollSeconds 120 -PollIntervalMs 150
.\Monitor_DPTF_Fan.ps1 -Mode etw
.\Monitor_DPTF_Fan.ps1 -Mode all
结论解读:若 poll 阶段在负载下 FAND 始终不变,且 check 未见 DPTF 设备/服务,则当前系统未通过 DPTF 发送风扇控制信号;若 ETW 中有相关事件,可进一步对照 DPTF 文档看是否包含 _FSL 调用。
- 要主动退回 EC 自动控扇,可以写 0xff 或 >100(例如 0x80):
ectest -acpi \_SB.ECT0._FSL 0xff # 或 0x80 等 >100 ectest -acpi \_SB.ECT0._FST # FAND 应回到 0xff
7. 总结
- Puff/Kaisa 的 EC 只有一颗板载 thermistor(
TEMP_SENSOR_CORE),所有 EC 自动温控与关机/限频都基于这一路。 - coreboot + Chrome EC 已经在 ACPI/DPTF 层实现了完整的 FAN 接口(TFN1._FST/_FSL):
_FSL(level)→ 写入EC_ACPI_MEM_FAN_DUTY→dptf_set_fan_duty_target(level);_FST()→ 回读当前 FAND 与实际 RPM。
- EC 明确定义了 fan duty 的语义:
- 0–100:DPTF 直接指定 duty,EC 关闭 thermal 自动控扇;
- 0xff 或越界:退出 DPTF duty 模式,交回 EC 自动控扇,FAND 显示为 0xff。
- Windows 下已枚举到 INT3400/INT3404:在设备管理器中分别显示为 Intel(R) Dynamic Tuning Manager(INT3400)与 Intel(R) Dynamic Tuning Fan Participant(INT3404),设备实例路径为
ACPI\INT3400\0、ACPI\INT3404\0,说明 DPTF 风扇控制器栈与 ACPI 风扇设备都是存在的。 - 实测轮询
\_SB.ECT0._FST时,FAND 只在手动_FSL(x)时改变,负载/温度变化本身并不会让 DPTF 自动写入 FAND;结合 INT3404 存在但 FAND 不随温度变化,可以判断:当前固件/策略仍以 EC 自动曲线为主控,DPTF 风扇参与者更多处于观察/可选控制状态。 - “
_FSL(0x80)后 FAND 一直 0xff” 是因为 0x80>100,被 EC 解读为 “放弃 DPTF 覆盖、恢复 EC 自动”,而非链路未接通。 - 选择 EC 控扇还是 DPTF 控扇,只需在 coreboot defconfig 中切换:
- EC 控扇:
CONFIG_MINIPC_EC_FULL_FAN_CONTROL=y+CONFIG_DRIVERS_INTEL_DPTF_DISABLE_ACTIVE_FAN=y; - DPTF 控扇:两者都不设,并确保 Windows 中 DPTF 驱动工作正常。
- EC 控扇:
7.1 固件 20260224034907 配置确认(DPTF 控扇固件)
- 固件:
20260224034907目录下为coreboot_edk2-kaisa-mrchromebox_20260224.rom的构建产物,含.config、build.log。 - .config 中与风扇控制相关的项:
# CONFIG_MINIPC_EC_FULL_FAN_CONTROL is not set→ EC 非全权控扇,主机可写 FAND;# CONFIG_DRIVERS_INTEL_DPTF_DISABLE_ACTIVE_FAN is not set→ DPTF 主动风扇策略开启;CONFIG_DRIVERS_INTEL_DPTF_ECTST_FAND_DEBUG=y→ 暴露 ECT0 调试设备,可读/写 FAND。
- 结论:该固件为 DPTF 控扇 配置,风扇应由 DPTF 通过 _FSL 写 FAND,EC 按 FAND 执行。
- 与实测的差异:在 Windows 下用
Monitor_DPTF_Fan.ps1 -Mode watch观察,CPU 升温时 FAND 从未变化(无 “FAND changed”),说明 DPTF 驱动/策略没有在按温度调用 _FSL。问题在 Windows/DPTF 侧,不在固件是否为 DPTF 控扇。 - 建议:在 Windows 中检查电源/热管理是否有 “Intel DPTF” 或 “平台热管理” 并确认已启用;从 Microsoft Update Catalog 或 OEM 安装/更新 “Intel Dynamic Platform and Thermal Framework” 驱动后,再以 watch 模式观察负载下 FAND 是否变化。
7.2 Windows 电源/热管理检查步骤(驱动已确认时)
硬件与 DPTF 驱动(INT3400/INT3404、esifsvc)已确认正常时,若 FAND 仍不随温度变化,可逐项检查电源与热管理是否“关掉”了 DPTF 控扇:
| 步骤 | 位置 | 检查内容 |
|---|---|---|
| 1 | 设置 → 系统 → 电源 | 查看当前“电源模式”(平衡/最佳性能等);部分机型在此有“热管理”或“Intel 图形/平台”相关项,确认未选“静音/省电”导致不拉高风扇。 |
| 2 | 控制面板 → 电源选项 → 更改计划设置 → 更改高级电源设置 | 在列表中查找 处理器电源管理、Intel、Thermal、平台热管理 或 OEM(如 Acer)提供的“风扇/热管理”子项。若有“最大处理器状态”“散热方式”等,可先选“主动/积极散热”或提高最大状态,排除因电源计划限制导致 DPTF 不写 _FSL。 |
| 3 | 设备管理器 → 系统设备 → Intel(R) Dynamic Tuning Fan Participant | 属性 → 驱动程序:确认设备已启用、无感叹号;若有“电源管理”选项卡,可取消勾选“允许计算机关闭此设备以节约电源”(若存在),避免被省电关闭。 |
| 4 | OEM 自带软件 | 若装有 Acer Care、Lenovo Vantage、Dell Power Manager 等,在“性能/散热/风扇”类设置中查看是否有“静音模式”“禁用 DPTF”“自定义风扇曲线”等;若有且设为静音或禁用 DPTF,改为平衡/性能或启用 DPTF 后再用 watch 观察。 |
| 5 | 当前电源方案 | 管理员 PowerShell:powercfg /getactivescheme 查看当前方案;必要时切换为“高性能”或“卓越性能”后再次运行 Monitor_DPTF_Fan.ps1 -Mode watch,看负载下是否出现 “FAND changed”。 |
脚本 Monitor_DPTF_Fan.ps1 -Mode check 会输出当前电源方案及上述 1~4 的手动检查提示,便于逐项核对。
这份笔记即为当前 Puff/Kaisa 上温控与 EC/DPTF 行为的整理,可作为后续调整 EC 曲线或 DPTF 策略时的参考。
温控与 EC 传感器分析笔记
本文档记录 Chromebox 刷 coreboot 后,Windows 热关机、Linux 温控、EC 温度源及 I2C 设备识别相关的有用信息。
1. 问题背景
- 现象:Windows 下出现热事件关机;EC 控扇仅依赖板载温度传感器,延迟与偏差大,导致风扇响应滞后。
- 根因:EC 只有一路 ADC 热敏电阻(“Core”),且该点滞后于 CPU 实际温度,Windows 负载/功耗较高时易触发过热关机。
2. 热区与 EC 的关系(Linux 视角)
- thermal_zone(如 TCPU、x86_pkg_temp):内核/CPU 热区,数据不经过 EC。
- EC 获取 CPU trip 的方式:
- 硬件:THERMTRIP#、PROCHOT# 等引脚由 CPU 拉低,EC 直接检测。
- 软件:ACPI Notify(thermal zone critical 等)可通知 EC。
- 直接连到 EC 的温度:cros_ec 上报的 “Core” = EC 自己读的 ADC 热敏电阻;INT3400 与 EC 策略/关机链路相关。
3. Linux 温控 vs Windows DPTF
| 项目 | Linux 温控 | Windows DPTF |
|---|---|---|
| 策略 | 内核 governor / thermald | OEM 策略引擎 + ACPI |
| 应用感知 | 一般无 | 有(游戏/办公等不同策略) |
| 风扇 | 内核 cooling 或 fancontrol | 常由 EC 独管,DPTF 不控扇 |
DPTF 未真正控扇常见原因:风扇权在 EC;或 OEM 未把风扇暴露给 DPTF;或 Windows 无 EC 风扇驱动。
4. EC 温度源(以 Puff 板为例)
- 当前:仅 1 路 温度 —— ADC 热敏电阻(
ADC_TEMP_SENSOR_1→TEMP_SENSOR_CORE),读函数get_temp_3v3_30k9_47k_4050b。 - 无 I2C 温度传感器:board.c 里 temp_sensors[] 只有上述 ADC 项;EC 的 I2C 上接的是 INA3221、PPC、TCPC、POWER、EEPROM 等,无温感芯片。
- 增加温度源的可能:若硬件有预留(空余 ADC 通道或 EC I2C 上接温感),可在 EC 源码里加 temp_sensor 配置;否则需改硬件。
5. 主机 I2C 与 EC I2C 是两套总线
- 主机侧(
i2cdetect -l的 i2c-0~i2c-11):CPU/PCH 的 SMBus、i915 等,由 Linux 枚举;扫描用i2c_scan.sh(调用 i2cdetect)。 - EC 侧:EC 固件里
i2c_ports[]定义的 GPIO_I2C0~I2C7,设备在 board.c 中定义。 - 结论:主机 I2C 扫描到的地址(如 0x08、0x44、0x52…)不会出现在 EC 的 board.c 里;EC 能扫的只是自己那几条总线。
6. 本机 EC I2C 扫描结果(ectool-cec i2cxfer)
| EC Port | 扫到地址 | 用途/推断 |
|---|---|---|
| 1 | 0x40 | 电流/功率类,非标准 INA3221 |
| 3 | 0x2c 0x38 0x39 0x3d 0x3f 0x42 | 见下表 |
| 5 | 0x20 | EEPROM (24cxx) |
Port 3 各地址识别摘要
| Addr | reg 0x00 / 0x80 等 | 可能型号/类型 |
|---|---|---|
| 0x2c | 0x77 0x79…;index/data 访问 | Nuvoton Super I/O / I2C 多功能(如 NCT5532D、NCT5577D、NCT38xx);可能带温度,需 index/data 读 |
| 0x38 | 0x30 0x30 (ASCII) 等 | GPIO/IO 扩展或显示 |
| 0x39 | 全 0 | 未初始化或空 |
| 0x3d | 全 0 | 可能 SSD1306 或未初始化 |
| 0x3f | 0x80 = 0x5A 0x5E | 某 PMIC/传感器厂商 Device ID;可能带温度,需查 datasheet |
| 0x42 | 全 0 | 可能 INA3221 未使能或它用 |
7. 常见芯片 ID 参考(用于 ec_i2c_identify)
- 0x80 = 0x54 0x49:TI INA3221/INA219(电流/功率)。
- 0x80 = 0x5A 0x5E:某厂商 Device ID,多见于 PMIC/传感器,具体型号需查该厂商手册。
- 0x2c 总线、首字节 0x77 等:Nuvoton NCT55xx/NCT5577D/NCT38xx 等;访问多为 index/data,直接读“reg 0”未必对应真实寄存器。
- 0x20 + 数据区大段 0xff:常见 EEPROM (24c02 等)。
- 0x38 读得 0x30 0x30:常见 GPIO/IO 扩展(如 PCA9555)。
- 0x3c/0x3d 全 0:常见 SSD1306 等 OLED 未初始化。
8. 哪个可能是我们要找的温度传感器?
-
首选候选:Port 3, 0x2c
Nuvoton NCT 系常带硬件监控温度;需在 EC 内按 index/data 协议和该芯片 datasheet 的温度寄存器 index 读取。 -
次选候选:Port 3, 0x3f (0x5A5E)
可能是带温度通道的 PMIC/传感器;需找到对应 datasheet 确认是否有温度寄存器及地址/格式。 -
其余(0x40/0x42 INA 类、0x38 GPIO、0x39/0x3d、0x20 EEPROM)不视为温度传感器。
9. 脚本与工具
| 脚本/命令 | 作用 |
|---|---|
linux_collect_sensors.sh |
收集 Linux 下 thermal zone、hwmon、lm-sensors、ectool temps、dmesg、CPU 信息;报告存当前目录或 /tmp。 |
i2c_scan.sh |
主机侧 I2C 扫描(i2cdetect),扫 i2c-0~i2c-N。 |
ec_i2c_scan.sh |
EC 侧 I2C 扫描,用 ectool-cec 的 i2cxfer 对指定 port 扫 0x08~0x77。须用 ectool-cec,系统 ectool 无 i2cxfer。 |
ec_i2c_identify.sh |
对指定 port:addr 读 reg 0x00/0x80/0xFE/0xFF/0x0F,并给出可能型号推断。 |
使用注意
- 脚本行尾须为 LF;若出现
$'\r': 未找到命令等,执行:sed -i 's/\r$//' 脚本名.sh。 - EC 扫描/识别需 sudo,且 ECTOOL 指向 ectool-cec,例如:
ECTOOL=/home/jack/bin/ectool-cec sudo ./ec_i2c_scan.sh
10. 后续可做
- 在 EC 源码中查 0x2c 对应芯片的 datasheet,用 index/data 读温度寄存器,并在 temp_sensor/thermal 里增加一路。
- 若有 0x5A5E 对应型号的 datasheet,确认是否带温度通道及寄存器,再在 EC 中实现读取。
- Windows 侧可先通过限 CPU 功耗/电源计划降低热负载,减轻 EC 单路温控滞后带来的热关机。
11. Puff/Kaisa 上增加一路温度传感器的可行性
11.1 地址与硬件对应关系(Puff)
- I2C Port 3(TCPC0)上的 0x2c:在 Puff 板为 AN7447 TCPC(
AN7447_TCPC0_I2C_ADDR_FLAGS = 0x2C),不是独立温感芯片,不能当第二路温度源使用。 - 0x3f:同颗 AN7447 的 SPI 转 I2C 接口(
AN7447_SPI0_I2C_ADDR_FLAGS = 0x3F),读到的 0x5A5E 等为 AN7447 相关,不是独立 I2C 温度传感器。 - 结论:当前 Puff 板 EC 可见的 I2C 总线上,没有空闲的、可用的 I2C 温度传感器芯片;笔记中“Port 3 的 0x2c/0x3f 可能是温感”的推断在本板不成立。
11.2 可行方向
| 方向 | 说明 |
|---|---|
| A. 第二路 ADC 热敏电阻 | 若原理图有未用的 ADC 通道 + 热敏电阻,在 board.h 增加 ADC_TEMP_SENSOR_2 与对应 enum,在 board.c 的 adc_channels[]、temp_sensors[]、thermal_params[] 各加一项,并用现有 get_temp_3v3_30k9_47k_4050b 或对应 thermistor 读函数。 |
| B. 新增 I2C 温感芯片 | 若硬件在某一 I2C 总线上新接温感(如 TMP112/TMP432 等),需:该总线在 i2c_ports[] 中、地址不与现有设备冲突;在 EC 中启用对应 driver(如 CONFIG_TEMP_SENSOR_TMP112)、在 board.h 增加 TEMP_SENSOR_xxx 与 TEMP_SENSOR_COUNT,在 board.c 中增加 temp_sensors[] 与 thermal_params[]。 |
| C. 确认 0x5A5E 芯片 | 若某板型在其它 I2C 端口上有独立 0x5A5E 芯片且 datasheet 标明带温度寄存器,可为该芯片写专用 driver(index/data 或普通 I2C 读),再按 B 的方式挂到 temp_sensors[]。 |
11.3 EC 侧增加一路温度传感器的实现步骤(通用)
-
board.h
- 增加
enum adc_channel或 I2C 温感用的enum(若用现有 driver,通常只需temp_sensor_id)。 - 在
enum temp_sensor_id中增加新 ID(如TEMP_SENSOR_BOARD2),并保证TEMP_SENSOR_COUNT正确。
- 增加
-
board.c
- ADC 热敏电阻:在
adc_channels[]增加一路 ADC;在temp_sensors[]增加一项(name、type、read、idx);在thermal_params[]增加对应ec_thermal_config。 - I2C 温感:若为现有 driver(如 TMP112),在
temp_sensors[]增加一项,read指向 driver 的get_val,idx为 channel/index;同样在thermal_params[]增加一项。 - 若需参与控扇:在
setup_thermal()等逻辑中为新 sensor 选择/绑定 thermal table。
- ADC 热敏电阻:在
-
Kconfig / 构建
- 若用 I2C 温感 driver,在对应 board 的
Kconfig或board.h中启用CONFIG_TEMP_SENSOR_xxx及该 driver 所需的I2C_PORT_xxx。
- 若用 I2C 温感 driver,在对应 board 的
-
thermal 策略
- 若希望新传感器参与风扇控制,需在 thermal 逻辑中把新 sensor 的读数纳入(例如与现有
TEMP_SENSOR_CORE取 max,或单独阈值),并保证thermal_params[新ID]的temp_fan_off/temp_fan_max等已配置。
- 若希望新传感器参与风扇控制,需在 thermal 逻辑中把新 sensor 的读数纳入(例如与现有
11.4 小结
- 不改硬件:Puff 上仅能依赖现有 1 路 ADC 热敏电阻(Core);无法在现有 I2C 设备(0x2c/0x3f 均为 AN7447)上“多出一路”温度传感器。
- 改硬件:增加第二路 ADC 热敏电阻,或新接 I2C 温感芯片并接在已有或新增 I2C 总线上,再按 11.3 在 EC 中增加一项 temp_sensor 与 thermal 配置即可。
12. DPTF 控制风扇的可行性(Puff/Kaisa)
12.1 固件侧已支持 DPTF 控扇
- Puff 的 DPTF 设计本来就是“DPTF 控扇”:
baseboard/acpi/dptf.asl中定义了DPTF_ENABLE_FAN_CONTROL、DFPS(Fan Performance States)、DART(风扇对 CPU/TSR 的影响)。- Kaisa 的
overridetree.cb里配置了 Active Policy:policies.active[0]= DPTF_CPU,policies.active[1]= DPTF_TEMP_SENSOR_0,即 CPU 温度 + 板载温感(EC 上报)共同驱动风扇;另有controls.fan_perf[]、fine_grained_control、step_size等。 - 注释写明:"Active Policy: CPU drives fan (DPTF controls via FAND)"。
- 风扇档位由 DPTF 通过 ACPI 写 FAND 下发给 EC,EC 按 FAND 驱动 PWM;固件表和 devicetree 已具备完整策略,从固件角度看 DPTF 控扇是可行的。
12.2 当前 Kaisa 配置为何是 EC 控扇
- 为规避“Windows 下无人写 FAND、风扇卡在固定转速”的问题,当前 defconfig 改为 EC 完全控扇:
CONFIG_MINIPC_EC_FULL_FAN_CONTROL=y:隐藏 DPTF 风扇设备,EC 在 _REG 时写 FAND=0xFF(EC 自动控扇)。CONFIG_DRIVERS_INTEL_DPTF_DISABLE_ACTIVE_FAN=y:关闭 DPTF Active Policy,DPTF 不再写 FAND。
- 这样 Windows 即使没有 DPTF 驱动,EC 也会按自身 thermistor 控扇,避免“既不用 DPTF 也不写 0xFF”导致的卡死。
12.3 若改为 DPTF 控扇
| 项目 | 说明 |
|---|---|
| 固件配置 | 取消 EC 全权控扇、恢复 DPTF 控扇:CONFIG_MINIPC_EC_FULL_FAN_CONTROL=n(或不设),CONFIG_DRIVERS_INTEL_DPTF_DISABLE_ACTIVE_FAN=n。重新编译刷写后,DPTF 风扇设备可见,Active Policy 生效。 |
| 温度源 | DPTF 使用 CPU 温度(_SB.PCI0.TCPU)+ EC 上报的板载温感(TSR0 = TEMP_SENSOR_0,即 Core)。CPU 温度由 OS/驱动提供,响应比单路 EC thermistor 快,有利于提前拉高转速。 |
| Windows 依赖 | 必须安装 Intel DPTF 相关驱动(如 INT3400、INT3404 等),否则 OS 不会根据 DPTF 表写 FAND,风扇仍可能不随负载变化。驱动可从 Microsoft Update Catalog 或 OEM 包获取。 |
| Linux | 若使用内核 thermal + DPTF ACPI,需相应驱动/用户态配合写 cooling_device;常见做法仍是 EC 控扇或 fancontrol。 |
12.4 小结
- DPTF 控扇在 Puff/Kaisa 上固件可行:表与策略已就绪,只需改 defconfig 并确保 Windows 有 DPTF 驱动。
- EC 控扇:不依赖 Windows 驱动,适合“通用 Windows/无 DPTF”场景,但仅有一路板载温感,响应偏慢。
- 二选一:要么 EC 全权控扇(当前配置),要么 DPTF 控扇(改配置 + 装 Windows DPTF 驱动),不可同时“DPTF 可见但不写 FAND、EC 也不写 0xFF”,否则会再现风扇卡死。
13. 监视 FAND 与“DPTF 未改变 FAND”的排查
13.1 概念:DPTF 如何写风扇
- DPTF 控扇时,OS 驱动调用 ACPI 方法 _FSL(Fine-grained Set Level),ACPI 里把参数写入 EC 的 FAND 字段;EC 固件读 FAND 驱动 PWM。
- FAND:EC ACPI 中 8 位字段(Fan Duty),在
ec.asl的 ERAM 里;0xFF 表示 EC 自动控扇。 - _FSL(Arg0):DPTF 风扇设备(如 TFN1)上的方法,实现为
Store(Arg0, EC0.FAND)。 - 无法直接“hook _FSL”,监视 _FSL 的效果 = 轮询读取 FAND,看其是否随负载/温度变化。
13.2 如何监视 FAND
- 固件开启 CONFIG_DRIVERS_INTEL_DPTF_ECTST_FAND_DEBUG=y 时,会暴露调试设备
\_SB.ECT0:\_SB.ECT0._FST():读当前 FAND(与 TFN1._FST 一致);\_SB.ECT0._FSL(x):写 FAND(与 DPTF 写路径一致)。
- Windows:安装 ec-test-app(OpenDevicePartnership ectest)后,在管理员 PowerShell 中轮询:
while ($true) { ectest -acpi \_SB.ECT0._FST Start-Sleep -Seconds 1 } - Linux:若 ectool 支持读 FAND 或 EC RAM 中对应偏移,可写脚本轮询;或通过 cros_ec 接口读 thermal/风扇相关节点(视内核暴露而定)。
13.3 “监视过 FAND,但 DPTF 似乎没有改变 FAND”的排查
| 检查项 | 说明 |
|---|---|
| 1. 当前是否为 DPTF 控扇配置 | 若固件为 EC 全权控扇(CONFIG_MINIPC_EC_FULL_FAN_CONTROL=y 且 CONFIG_DRIVERS_INTEL_DPTF_DISABLE_ACTIVE_FAN=y),DPTF 风扇设备被隐藏(_STA=0),Windows 不会调用 _FSL,FAND 仅由 EC 在写 0xFF 后按自身 thermistor 变化。此时“DPTF 没改 FAND”是预期行为。若要验证 DPTF 控扇,需先改为 DPTF 控扇配置并重新刷写。 |
| 2. Windows 是否识别 DPTF 设备 | 设备管理器中查看是否有 Intel Dynamic Platform and Thermal Framework 或 ACPI 下的 INT3400、INT3404 等。若无 INT3404/DPTF 风扇设备,说明驱动未加载或 ACPI 未暴露,DPTF 不会写 FAND。 |
| 3. 是否安装 DPTF 驱动 | 从 Microsoft Update Catalog 搜索 “Intel DPTF” 或 “Dynamic Platform Thermal”,或从 OEM(如 Acer)下载该机型的 DPTF/热管理驱动并安装。未安装时 OS 不会根据 DPTF 表调用 _FSL。 |
| 4. 电源/热管理策略 | Windows 电源选项中是否有“Intel DPTF”或“平台热管理”相关项;部分 OEM 软件会关闭 DPTF 控扇或固定为静音策略,导致不写 FAND。可切换电源方案(平衡/高性能)后再观察 FAND。 |
| 5. 负载下观察 | 在确认固件为 DPTF 控扇、且设备管理器中有 INT3404 的前提下,对 CPU 施加负载(如压力测试),每隔数秒读一次 \_SB.ECT0._FST。若 FAND 始终不变,可判断为 DPTF 驱动未调用 _FSL 或策略未生效。 |
13.4 小结
- 监视 _FSL = 轮询读 FAND(通过 ECT0._FST 或等效接口)。
- FAND 不变时:先确认是 EC 控扇 还是 DPTF 控扇 的固件;若为 DPTF 控扇则检查 Windows 是否安装并加载 DPTF 驱动、INT3404 是否存在、电源/热管理策略是否启用 DPTF 控扇。
13.5 如何检查 _SB.ECT0._FST 是否暴露给系统
\_SB.ECT0 仅在固件开启 CONFIG_DRIVERS_INTEL_DPTF_ECTST_FAND_DEBUG=y 时存在;未开启则 ACPI 里没有该设备,无法读 _FST。
方法一:直接尝试执行(最直接)
-
Windows:安装 ec-test-app(ectest),在管理员 PowerShell 中执行:
ectest -acpi \_SB.ECT0._FST- 若返回数值(如 FAND 0–255)→ 已暴露。
- 若报错 “object not found” / “method does not exist” / 无法解析路径 → 未暴露或固件未开上述 CONFIG。
-
Linux:用 ACPI 调试接口(需 root):
# 若内核有 ACPI 调试,可尝试(部分内核未启用) echo '\_SB.ECT0._FST' > /sys/kernel/debug/acpi/custom_method cat /sys/kernel/debug/acpi/custom_method或使用 acpi_call 等模块/工具调用 ACPI 方法(若已安装)。
方法二: dump DSDT/SSDT 看是否有 ECT0 与 _FST
-
Windows:
- 用 SSDTTime 选 “Dump DSDT” 得到 DSDT.aml;或用 ACPICA acpidump:
acpidump -b -n DSDT得到 DSDT.dat(改名为 DSDT.aml)。 - 用 iasl 反编译:
iasl -d DSDT.aml,得到 DSDT.dsl。 - 在 DSDT.dsl 中搜索
ECT0、_FST:若在Scope (\_SB)下存在Device (ECT0)且其内有Method (_FST, ...),则固件已暴露该对象(是否可用还取决于 _STA 等)。
- 用 SSDTTime 选 “Dump DSDT” 得到 DSDT.aml;或用 ACPICA acpidump:
-
Linux:
sudo cat /sys/firmware/acpi/tables/DSDT > DSDT.aml iasl -d DSDT.aml grep -n "ECT0\|_FST" DSDT.dsl
方法三:设备/驱动侧(辅助)
- Windows:设备管理器中查看 ACPI 设备列表,是否有与 “ECT” 或 “DPTF ECT” 相关的设备(名称因 OEM 而异,不一定叫 ECT0)。
- 若固件未开
CONFIG_DRIVERS_INTEL_DPTF_ECTST_FAND_DEBUG,DSDT 中不会出现 ECT0,方法一、二都会得到“未暴露”的结论。
13.6 实测:ECT0._FSL 写入未生效的情况
-
环境:当前这台 Chromebox 刷入的 coreboot 配置为 EC 全权控扇(推测存在
CONFIG_MINIPC_EC_FULL_FAN_CONTROL=y之类),同时打开了 DPTF ECT 调试接口(CONFIG_DRIVERS_INTEL_DPTF_ECTST_FAND_DEBUG=y)。 -
实测命令:
# 1)读当前 FAND ectest -acpi \_SB.ECT0._FST # Argument[1] = 0xff (EC 自动控扇) # 2)写入 0x80 ectest -acpi \_SB.ECT0._FSL 0x80 # 3)再读 ectest -acpi \_SB.ECT0._FST # Argument[1] 仍为 0xff -
同样逻辑用
Test_ECT0_FST_FSL.ps1自动化验证时,效果一致:[1] Current FAND from _FST: FAND = 0xff[2] Writing FAND via _FSL(0x80) ...执行成功,ACPI 返回 Arg0=0x80;[3] Read-back FAND after _FSL: FAND = 0xff,后续轮询_FST仍为 0xff。
对 _FST/_FSL 返回值的具体解读:
- [1]
_FST= 0xff_FST返回包里的Argument[1] = 0xff,表示 EC 侧 当前 FAND=0xFF,即“自动控扇模式”。
- [2]
_FSL(0x80)返回 0x80- 这里的
Argument[0] = 0x80只是这次_FSL调用把参数“原样回显”,说明 ACPI 方法被成功执行,并不等于 EC 里的 FAND 已经变成 0x80。
- 这里的
- [3] 再次
_FST仍然 0xff- 再读
_FST时Argument[1]还是0xff,说明 EC 看到的 FAND 一直没有从 0xFF 变成 0x80,要么_FSL根本没写到 EC.FAND,要么写了但被 EC 逻辑立即覆盖回 0xFF。
- 再读
- [4] 持续监控
_FST也一直是 0xff- 说明在一段时间内 FAND 从未被真正设为 0x80,EC 自始至终都工作在自动控扇模式。
结论(就当前固件配置):
- 虽然 ACPI 中存在
\_SB.ECT0._FST / _FSL,但_FSL调用并没有真正改掉 EC 侧的 FAND 字段; - 更可能的实现是:coreboot 的 ACPI 里为调试方便加了
_FST/_FSL方法,但在 EC 全权控扇配置下,\_FSL要么只是“回显参数”(Return(Arg0)),要么即便临时写入 FAND,也被 EC 策略立即覆盖回 0xFF; - 因为 FAND 始终为 0xFF,EC 始终工作在“自动控扇”模式,DPTF/ECT0 这条路径实际处于旁路状态。
文档由分析过程整理,便于后续改 EC 或排查温控时查阅。
13.6 实测:ECT0._FSL 写入 0x80 后 FAND 始终为 0xFF 的根因
- 环境:当前 Chromebox 上已启用 DPTF + ECT0(
CONFIG_DRIVERS_INTEL_DPTF_ECTST_FAND_DEBUG=y),通过ECT0._FST/_FSL读写 FAND。 - 实测命令(节选自
logs.txt):