162306a36Sopenharmony_ci.. SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci.. include:: ../disclaimer-zh_CN.rst 362306a36Sopenharmony_ci 462306a36Sopenharmony_ci:Original: Documentation/power/energy-model.rst 562306a36Sopenharmony_ci 662306a36Sopenharmony_ci:翻译: 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci 唐艺舟 Tang Yizhou <tangyeechou@gmail.com> 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci============ 1162306a36Sopenharmony_ci设备能量模型 1262306a36Sopenharmony_ci============ 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci1. 概述 1562306a36Sopenharmony_ci------- 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci能量模型(EM)框架是一种驱动程序与内核子系统之间的接口。其中驱动程序了解不同 1862306a36Sopenharmony_ci性能层级的设备所消耗的功率,而内核子系统愿意使用该信息做出能量感知决策。 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci设备所消耗的功率的信息来源在不同的平台上可能有很大的不同。这些功率成本在某些 2162306a36Sopenharmony_ci情况下可以使用设备树数据来估算。在其它情况下,固件会更清楚。或者,用户空间可能 2262306a36Sopenharmony_ci是最清楚的。以此类推。为了避免每一个客户端子系统对每一种可能的信息源自己重新 2362306a36Sopenharmony_ci实现支持,EM框架作为一个抽象层介入,它在内核中对功率成本表的格式进行标准化, 2462306a36Sopenharmony_ci因此能够避免多余的工作。 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_ci功率值可以用微瓦或“抽象刻度”表示。多个子系统可能使用EM,由系统集成商来检查 2762306a36Sopenharmony_ci功率值刻度类型的要求是否满足。可以在能量感知调度器的文档中找到一个例子 2862306a36Sopenharmony_ciDocumentation/scheduler/sched-energy.rst。对于一些子系统,比如热能或 2962306a36Sopenharmony_cipowercap,用“抽象刻度”描述功率值可能会导致问题。这些子系统对过去使用的功率的 3062306a36Sopenharmony_ci估算值更感兴趣,因此可能需要真实的微瓦。这些要求的一个例子可以在智能功率分配 3162306a36Sopenharmony_ciDocumentation/driver-api/thermal/power_allocator.rst文档中找到。 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_ci内核子系统可能(基于EM内部标志位)实现了对EM注册设备是否具有不一致刻度的自动 3462306a36Sopenharmony_ci检查。要记住的重要事情是,当功率值以“抽象刻度”表示时,从中推导以微焦耳为单位 3562306a36Sopenharmony_ci的真实能量消耗是不可能的。 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci下图描述了一个驱动的例子(这里是针对Arm的,但该方法适用于任何体系结构),它 3862306a36Sopenharmony_ci向EM框架提供了功率成本,感兴趣的客户端可从中读取数据:: 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci +---------------+ +-----------------+ +---------------+ 4162306a36Sopenharmony_ci | Thermal (IPA) | | Scheduler (EAS) | | Other | 4262306a36Sopenharmony_ci +---------------+ +-----------------+ +---------------+ 4362306a36Sopenharmony_ci | | em_cpu_energy() | 4462306a36Sopenharmony_ci | | em_cpu_get() | 4562306a36Sopenharmony_ci +---------+ | +---------+ 4662306a36Sopenharmony_ci | | | 4762306a36Sopenharmony_ci v v v 4862306a36Sopenharmony_ci +---------------------+ 4962306a36Sopenharmony_ci | Energy Model | 5062306a36Sopenharmony_ci | Framework | 5162306a36Sopenharmony_ci +---------------------+ 5262306a36Sopenharmony_ci ^ ^ ^ 5362306a36Sopenharmony_ci | | | em_dev_register_perf_domain() 5462306a36Sopenharmony_ci +----------+ | +---------+ 5562306a36Sopenharmony_ci | | | 5662306a36Sopenharmony_ci +---------------+ +---------------+ +--------------+ 5762306a36Sopenharmony_ci | cpufreq-dt | | arm_scmi | | Other | 5862306a36Sopenharmony_ci +---------------+ +---------------+ +--------------+ 5962306a36Sopenharmony_ci ^ ^ ^ 6062306a36Sopenharmony_ci | | | 6162306a36Sopenharmony_ci +--------------+ +---------------+ +--------------+ 6262306a36Sopenharmony_ci | Device Tree | | Firmware | | ? | 6362306a36Sopenharmony_ci +--------------+ +---------------+ +--------------+ 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci对于CPU设备,EM框架管理着系统中每个“性能域”的功率成本表。一个性能域是一组 6662306a36Sopenharmony_ci性能一起伸缩的CPU。性能域通常与CPUFreq策略具有1对1映射。一个性能域中的 6762306a36Sopenharmony_ci所有CPU要求具有相同的微架构。不同性能域中的CPU可以有不同的微架构。 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci2. 核心API 7162306a36Sopenharmony_ci---------- 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci2.1 配置选项 7462306a36Sopenharmony_ci^^^^^^^^^^^^ 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ci必须使能CONFIG_ENERGY_MODEL才能使用EM框架。 7762306a36Sopenharmony_ci 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci2.2 性能域的注册 8062306a36Sopenharmony_ci^^^^^^^^^^^^^^^^ 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_ci“高级”EM的注册 8362306a36Sopenharmony_ci~~~~~~~~~~~~~~~~ 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ci“高级”EM因它允许驱动提供更精确的功率模型而得名。它并不受限于框架中的一些已 8662306a36Sopenharmony_ci实现的数学公式(就像“简单”EM那样)。它可以更好地反映每个性能状态的实际功率 8762306a36Sopenharmony_ci测量。因此,在EM静态功率(漏电流功率)是重要的情况下,应该首选这种注册方式。 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci驱动程序应通过以下API将性能域注册到EM框架中:: 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_ci int em_dev_register_perf_domain(struct device *dev, unsigned int nr_states, 9262306a36Sopenharmony_ci struct em_data_callback *cb, cpumask_t *cpus, bool microwatts); 9362306a36Sopenharmony_ci 9462306a36Sopenharmony_ci驱动程序必须提供一个回调函数,为每个性能状态返回<频率,功率>元组。驱动程序 9562306a36Sopenharmony_ci提供的回调函数可以自由地从任何相关位置(DT、固件......)以及以任何被认为是 9662306a36Sopenharmony_ci必要的方式获取数据。只有对于CPU设备,驱动程序必须使用cpumask指定性能域的CPU。 9762306a36Sopenharmony_ci对于CPU以外的其他设备,最后一个参数必须被设置为NULL。 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci最后一个参数“microwatts”(微瓦)设置成正确的值是很重要的,使用EM的内核 10062306a36Sopenharmony_ci子系统可能会依赖这个标志来检查所有的EM设备是否使用相同的刻度。如果有不同的 10162306a36Sopenharmony_ci刻度,这些子系统可能决定返回警告/错误,停止工作或崩溃(panic)。 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_ci关于实现这个回调函数的驱动程序的例子,参见第3节。或者在第2.4节阅读这个API 10462306a36Sopenharmony_ci的更多文档。 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ci使用DT的EM注册 10762306a36Sopenharmony_ci============== 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ciEM也可以使用OPP框架和DT "操作点-v2 "中的信息注册。DT中的每个OPP条目都可 11062306a36Sopenharmony_ci以用一个包含微瓦特功率值的属性 "op-microwatt "来扩展。这个OPP DT属性允 11162306a36Sopenharmony_ci许平台注册反映总功率(静态+动态)的EM功率值。这些功率值可能直接来自实验和 11262306a36Sopenharmony_ci测量。 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_ci“人工”EM的注册 11562306a36Sopenharmony_ci============== 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_ci有一个选项可以为缺少关于每个性能状态的功率值的详细知识的驱动程序提供一个自 11862306a36Sopenharmony_ci定义回调。回调.get_cost()是可选的,它提供EAS使用的“成本”值。这对那些只提 11962306a36Sopenharmony_ci供CPU类型之间相对效率信息的平台很有用,人们可以利用这些信息来创建一个抽象的 12062306a36Sopenharmony_ci功率模型。但是,考虑到输入功率值的大小限制,即使是抽象的功率模型有时也很难装 12162306a36Sopenharmony_ci进去。.get_cost()允许提供反映CPU效率的“成本”值。这将允许提供EAS信息,它 12262306a36Sopenharmony_ci与EM内部计算'成本'值的公式有不同的关系。要为这样的平台注册EM,驱动程序必须 12362306a36Sopenharmony_ci将标志“microwatts”设置为0,提供.get_power()回调和.get_cost()回调。EM 12462306a36Sopenharmony_ci框架会在注册过程中正确处理这样的平台。这种平台会被设置EM_PERF_DOMAIN_ARTIFICIAL 12562306a36Sopenharmony_ci标志。其他使用EM的框架应该特别注意测试和正确对待这个标志。 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_ci“简单”EM的注册 12862306a36Sopenharmony_ci~~~~~~~~~~~~~~~~ 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_ci“简单”EM是用框架的辅助函数cpufreq_register_em_with_opp()注册的。它实现了 13162306a36Sopenharmony_ci一个和以下数学公式紧密相关的功率模型:: 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_ci Power = C * V^2 * f 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_ci使用这种方法注册的EM可能无法正确反映真实设备的物理特性,例如当静态功率 13662306a36Sopenharmony_ci(漏电流功率)很重要时。 13762306a36Sopenharmony_ci 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_ci2.3 访问性能域 14062306a36Sopenharmony_ci^^^^^^^^^^^^^^ 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_ci有两个API函数提供对能量模型的访问。em_cpu_get()以CPU id为参数,em_pd_get() 14362306a36Sopenharmony_ci以设备指针为参数。使用哪个接口取决于子系统,但对于CPU设备来说,这两个函数都返 14462306a36Sopenharmony_ci回相同的性能域。 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci对CPU的能量模型感兴趣的子系统可以通过em_cpu_get() API检索它。在创建性能域时 14762306a36Sopenharmony_ci分配一次能量模型表,它保存在内存中不被修改。 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci一个性能域所消耗的能量可以使用em_cpu_energy() API来估算。该估算假定CPU设备 15062306a36Sopenharmony_ci使用的CPUfreq监管器是schedutil。当前该计算不能提供给其它类型的设备。 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci关于上述API的更多细节可以在 ``<linux/energy_model.h>`` 或第2.4节中找到。 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci2.4 API的细节描述 15662306a36Sopenharmony_ci^^^^^^^^^^^^^^^^^ 15762306a36Sopenharmony_ci参见 include/linux/energy_model.h 和 kernel/power/energy_model.c 的kernel doc。 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci3. 驱动示例 16062306a36Sopenharmony_ci----------- 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ciCPUFreq框架支持专用的回调函数,用于为指定的CPU(们)注册EM: 16362306a36Sopenharmony_cicpufreq_driver::register_em()。这个回调必须为每个特定的驱动程序正确实现, 16462306a36Sopenharmony_ci因为框架会在设置过程中适时地调用它。本节提供了一个简单的例子,展示CPUFreq驱动 16562306a36Sopenharmony_ci在能量模型框架中使用(假的)“foo”协议注册性能域。该驱动实现了一个est_power() 16662306a36Sopenharmony_ci函数提供给EM框架:: 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci -> drivers/cpufreq/foo_cpufreq.c 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci 01 static int est_power(struct device *dev, unsigned long *mW, 17162306a36Sopenharmony_ci 02 unsigned long *KHz) 17262306a36Sopenharmony_ci 03 { 17362306a36Sopenharmony_ci 04 long freq, power; 17462306a36Sopenharmony_ci 05 17562306a36Sopenharmony_ci 06 /* 使用“foo”协议设置频率上限 */ 17662306a36Sopenharmony_ci 07 freq = foo_get_freq_ceil(dev, *KHz); 17762306a36Sopenharmony_ci 08 if (freq < 0); 17862306a36Sopenharmony_ci 09 return freq; 17962306a36Sopenharmony_ci 10 18062306a36Sopenharmony_ci 11 /* 估算相关频率下设备的功率成本 */ 18162306a36Sopenharmony_ci 12 power = foo_estimate_power(dev, freq); 18262306a36Sopenharmony_ci 13 if (power < 0); 18362306a36Sopenharmony_ci 14 return power; 18462306a36Sopenharmony_ci 15 18562306a36Sopenharmony_ci 16 /* 将这些值返回给EM框架 */ 18662306a36Sopenharmony_ci 17 *mW = power; 18762306a36Sopenharmony_ci 18 *KHz = freq; 18862306a36Sopenharmony_ci 19 18962306a36Sopenharmony_ci 20 return 0; 19062306a36Sopenharmony_ci 21 } 19162306a36Sopenharmony_ci 22 19262306a36Sopenharmony_ci 23 static void foo_cpufreq_register_em(struct cpufreq_policy *policy) 19362306a36Sopenharmony_ci 24 { 19462306a36Sopenharmony_ci 25 struct em_data_callback em_cb = EM_DATA_CB(est_power); 19562306a36Sopenharmony_ci 26 struct device *cpu_dev; 19662306a36Sopenharmony_ci 27 int nr_opp; 19762306a36Sopenharmony_ci 28 19862306a36Sopenharmony_ci 29 cpu_dev = get_cpu_device(cpumask_first(policy->cpus)); 19962306a36Sopenharmony_ci 30 20062306a36Sopenharmony_ci 31 /* 查找该策略支持的OPP数量 */ 20162306a36Sopenharmony_ci 32 nr_opp = foo_get_nr_opp(policy); 20262306a36Sopenharmony_ci 33 20362306a36Sopenharmony_ci 34 /* 并注册新的性能域 */ 20462306a36Sopenharmony_ci 35 em_dev_register_perf_domain(cpu_dev, nr_opp, &em_cb, policy->cpus, 20562306a36Sopenharmony_ci 36 true); 20662306a36Sopenharmony_ci 37 } 20762306a36Sopenharmony_ci 38 20862306a36Sopenharmony_ci 39 static struct cpufreq_driver foo_cpufreq_driver = { 20962306a36Sopenharmony_ci 40 .register_em = foo_cpufreq_register_em, 21062306a36Sopenharmony_ci 41 }; 211