5.4 KiB
title, date, tags
| title | date | tags | ||||
|---|---|---|---|---|---|---|
| 如何在笔记本电脑插拔电源时自动切换 SCX 调度模式 | 2026-06-14 19:24:35 |
|
一直觉得scx_lavd和scx_flow支持自动判断负载来选择合适的调度模式很不错,但是我还是更信赖Nvidia工程师出品的scx_bpfland,尤其是CachyOS定义的scx_bpfland低延迟模式和省电模式在我笔记本上的表现都非常好。但有一个痛点——笔记本插电和用电池的场景对调度器的需求完全不同:
- 插电时:希望低延迟、高性能,用得爽,程序编译快游戏帧数高
- 电池时:希望尽可能省电,续航优先尽量用久一点,不在乎性能牺牲
每次手动敲命令切换实在太蠢了使用CachyOS的scx-manager也不太方便。研究了一下后发现可以借助 Linux 的 udev 规则来监听电源事件,实现全自动切换。记录一下折腾过程。
现状:scx_loader 配置文件
这是我的 /etc/scx_loader.toml(CachyOS已提前为各个调度器配好了 mode):
default_sched = "scx_bpfland"
default_mode = "LowLatency"
[scheds.scx_rusty]
auto_mode = []
gaming_mode = []
lowlatency_mode = []
powersave_mode = []
server_mode = []
[scheds.scx_p2dq]
auto_mode = []
gaming_mode = []
lowlatency_mode = ["-y"]
powersave_mode = []
server_mode = ["--keep-running"]
[scheds.scx_bpfland]
auto_mode = []
gaming_mode = ["-m", "performance"]
lowlatency_mode = ["-s", "5000", "-S", "500", "-l", "5000", "-m", "performance"]
powersave_mode = ["-m", "powersave"]
server_mode = ["-p"]
[scheds.scx_lavd]
auto_mode = ["--autopower"]
gaming_mode = ["--performance"]
lowlatency_mode = ["--performance"]
powersave_mode = ["--powersave"]
server_mode = []
[scheds.scx_tickless]
auto_mode = []
gaming_mode = ["-f", "5000", "-s", "5000"]
lowlatency_mode = ["-f", "5000", "-s", "1000"]
powersave_mode = ["-f", "50", "-p"]
server_mode = ["-f", "100"]
[scheds.scx_flash]
auto_mode = []
gaming_mode = []
lowlatency_mode = []
powersave_mode = []
server_mode = []
仔细观察 [scheds.scx_bpfland] 这一块:
lowlatency_mode正好对应插电状态想要的参数:-s 5000 -S 500 -l 5000 -m performancepowersave_mode正好对应电池状态想要的参数:-m powersave
不用自己写额外参数配置,直接复用已有的 mode 定义就好。
思路
scx_loader 提供了配套的命令行工具 scxctl,可以用来切换调度器和模式。剩下的就是找一个能在电源事件发生时自动执行命令的机制——Linux 内核的 udev 子系统天生就是干这个的。
整体流程很简单:插拔电源 → 内核发出 uevent → udev 匹配规则 → systemd-run 异步执行 → scxctl switch 切换模式。
步骤一:测试手动切换命令
先确认 scxctl 能正常工作(需要 root 权限):
# 模拟插电 → 低延迟/高性能模式
sudo scxctl switch -s bpfland -m lowlatency
# 模拟拔电 → 省电模式
sudo scxctl switch -s bpfland -m powersave
如果这两条命令都能成功切换,就可以继续下一步。
步骤二:编写 udev 规则
创建规则文件:
sudo nano /etc/udev/rules.d/99-scx-power.rules
写入以下内容:
# 插入电源 (AC) → 切换为 bpfland 的 lowlatency 模式
SUBSYSTEM=="power_supply", ATTR{type}=="Mains", ATTR{online}=="1", RUN+="/usr/bin/systemd-run --no-block /usr/bin/scxctl switch -s bpfland -m lowlatency"
# 拔出电源 (Battery) → 切换为 bpfland 的 powersave 模式
SUBSYSTEM=="power_supply", ATTR{type}=="Mains", ATTR{online}=="0", RUN+="/usr/bin/systemd-run --no-block /usr/bin/scxctl switch -s bpfland -m powersave"
为什么用
systemd-run --no-block包裹?
scxctl底层通过 D-Bus 与scx_loader守护进程通信。如果让udev直接执行,可能会因为 udev 的受限执行环境(缺少 D-Bus session、环境变量缺失)或者命令执行时间过长阻塞 udev 进程导致失败。交给 systemd 后台异步执行是最稳妥的做法。
步骤三:重载规则使其生效
sudo udevadm control --reload-rules
sudo udevadm trigger
udevadm trigger 会立即重放一次当前设备状态,所以如果此时已经插着电源,规则会立刻触发一次,可以顺便验证是否生效。
步骤四(可选):验证电源类型
绝大多数笔记本的外部电源在内核中的设备类型都是 Mains,上面规则直接就能用。如果插拔电源发现没反应,检查一下你的电源设备类型:
cat /sys/class/power_supply/*/type
如果输出的不是 Mains(比如是 AC 或其他名字),把 udev 规则里的 ATTR{type}=="Mains" 换成实际类型即可。
验证效果
插拔电源后,可以用以下命令确认模式是否已经切换:
scxctl get
总结
整个过程其实非常简单——核心思路就是「udev 监听电源事件 → systemd-run 异步调用 scxctl 切换 mode」。全程没有写一个新参数,完全复用 scx_loader.toml 里已有的 mode 定义。
这个方案同样适用于其他调度器:只要在 scx_loader.toml 里给对应调度器配好 lowlatency_mode 和 powersave_mode,然后把 udev 规则里的 -s bpfland 换成其他调度器名字就行。
这下笔记本插拔电源再也不用惦记着手动切调度器了:D