【发布时间】:2013-07-16 10:52:29
【问题描述】:
我正在尝试在我的 linux 机器上的 C++ 线程中定期调用采样函数。我想在很短的时间(理想情况下为 1 毫秒)后重新启动我的功能,但我发现 1 毫秒期间消耗的功率(以瓦特为单位)高得令人望而却步:系统运行时的功率水平是我的两倍周期为 5 毫秒。保持低功耗是我想要的功能的主要关注点。
具体来说,
void* loop_and_sample(void* arg) {
while(1) {
sample();
nanosleep((struct timespec[]){{0,1000000}}, NULL);
}
}
需要 2 倍的力量:
void* loop_and_sample(void* arg) {
while(1) {
sample();
nanosleep((struct timespec[]){{0,5000000}}, NULL);
}
}
我已经确定采样器在 2 个频率下的功耗差异可以忽略不计,并且额外的功耗来自睡眠调用。也就是说,即使我在上面的两个 sn-ps 中注释掉了 sample() 行,第二个仍然需要一半的功率。关于如何减少睡眠呼叫消耗的功率的任何想法?
仅供参考,我在 24 核 Intel Xeon 上运行 Ubuntu 3.2.0,搜索 /boot/config 的频率显示如下:
cat /boot/config-3.2.0-48-generic | egrep 'HZ'
CONFIG_RCU_FAST_NO_HZ=y
CONFIG_NO_HZ=y
# CONFIG_HZ_100 is not set
CONFIG_HZ_250=y
# CONFIG_HZ_300 is not set
# CONFIG_HZ_1000 is not set
CONFIG_HZ=250
CONFIG_MACHZ_WDT=m
但是,运行此脚本:http://www.advenage.com/topics/linux-timer-interrupt-frequency.php,我发现我的内核定时器中断至少为 4016 Hz(这是我想要采样的频率的 4 倍)。感谢您的帮助!
【问题讨论】:
-
请指定您的硬件以及您正在运行的内核版本。电源状态管理是深奥的巫术,并且高度依赖于硬件。
-
Mark B(下图)可能是正确的,所以这是一个很长的镜头,但是...您可以尝试使用 timerfd(请参阅linux.die.net/man/2/timerfd_create)以防这是内部实现的问题小间隔的 nanosleep。
-
Ubuntu 3.2.0(如前文所述)和Intel Xeon 24核机(现已加入主体)
-
sample是做什么的?可以改为基于事件吗? -
示例实际上是在测量功率(通过读取英特尔的新 RAPL 计数器),我希望功率测量值为 1/ms,即计数器刷新率,所以我认为它不能基于事件
标签: c++ linux sleep power-saving