Linux内核Thermal框架详解五、Thermal Core(4)
本文部分内容参考Linux Thermal 学习笔记 - 爱码网。特此致谢!
接前一篇文章Linux内核Thermal框架详解四、Thermal Core(3)
三、相关源码及分析
2. thermal_register_governors
上一回说到这一段代码:
for (__governor = __governor_thermal_table; \__governor < __governor_thermal_table_end; \__governor++) {ret = thermal_register_governor(*governor);if (ret) {pr_err("Failed to register governor: '%s'",(*governor)->name);break;}pr_info("Registered thermal governor '%s'",(*governor)->name);
}
__governor_thermal_table上一回仔细分析了其来龙去脉,现在该关注__governor这个变量了。
__governor这个变量在之前代码中出现过,如下:
struct thermal_governor **governor;
这是一个二重指针,即指向指针的指针。在上述代码中,它一开始指向了__governor_thermal_table,还记得__governor_thermal_table是怎样定义的吗?
在drivers/thermal/thermal_core.h中:
extern struct thermal_governor *__governor_thermal_table[];
extern struct thermal_governor *__governor_thermal_table_end[];
虽然这里是extern,但是这比include/asm-generic/vmlinux.lds.h中真正定义__governor_thermal_table的地方好理解。__governor_thermal_table实际上是一个指针数组。所以使用二重指针governor指向它也就合情合理了。
再来回顾一下各种governor策略:
- step_wise
drivers/thermal/gov_step_wise.c中:
static struct thermal_governor thermal_gov_step_wise = {.name = "step_wise",.throttle = step_wise_throttle,
};static struct thermal_governor *__thermal_table_entry_thermal_gov_step_wise \__used __section("__governor_thermal_table") = &thermal_gov_step_wise
- power_allocator
drivers/thermal/gov_power_allocator.c中:
static struct thermal_governor thermal_gov_power_allocator = {.name = "power_allocator",.bind_to_tz = power_allocator_bind,.unbind_from_tz = power_allocator_unbind,.throttle = power_allocator_throttle,
};static struct thermal_governor *__thermal_table_entry_thermal_gov_power_allocator \__used __section("__governor_thermal_table") = &thermal_gov_power_allocator
- fair_share
drivers/thermal/gov_fair_share.c中:
static struct thermal_governor thermal_gov_fair_share = {.name = "fair_share",.throttle = fair_share_throttle,
};static struct thermal_governor *__thermal_table_entry_thermal_gov_fair_share \__used __section("__governor_thermal_table") = &thermal_gov_fair_share
- user_space
drivers/thermal/gov_user_space.c中:
static struct thermal_governor thermal_gov_user_space = {.name = "user_space",.throttle = notify_user_space,.bind_to_tz = user_space_bind,
};static struct thermal_governor *__thermal_table_entry_thermal_gov_user_space \__used __section("__governor_thermal_table") = &thermal_gov_user_space
- bang_bang
drivers/thermal/gov_bang_bang.c中:
static struct thermal_governor thermal_gov_bang_bang = {.name = "bang_bang",.throttle = bang_bang_control,
};static struct thermal_governor *__thermal_table_entry_thermal_gov_bang_bang \__used __section("__governor_thermal_table") = &thermal_gov_bang_bang
弄清楚了以上细节后就能知道,本文开头的代码的意义是:遍历所有的governor策略并进行注册。注册具体都完成了哪些工作?下边接着来看。
(2)thermal_register_governor
thermal_register_governor函数同样在drivers/thermal/thermal_core.c中实现,代码如下:
int thermal_register_governor(struct thermal_governor *governor)
{int err;const char *name;struct thermal_zone_device *pos;if (!governor)return -EINVAL;mutex_lock(&thermal_governor_lock);err = -EBUSY;if (!__find_governor(governor->name)) {bool match_default;err = 0;list_add(&governor->governor_list, &thermal_governor_list);match_default = !strncmp(governor->name,DEFAULT_THERMAL_GOVERNOR,THERMAL_NAME_LENGTH);if (!def_governor && match_default)def_governor = governor;}mutex_lock(&thermal_list_lock);list_for_each_entry(pos, &thermal_tz_list, node) {/** only thermal zones with specified tz->tzp->governor_name* may run with tz->govenor unset*/if (pos->governor)continue;name = pos->tzp->governor_name;if (!strncasecmp(name, governor->name, THERMAL_NAME_LENGTH)) {int ret;ret = thermal_set_governor(pos, governor);if (ret)dev_err(&pos->device,"Failed to set governor %s for thermal zone %s: %d\n",governor->name, pos->type, ret);}}mutex_unlock(&thermal_list_lock);mutex_unlock(&thermal_governor_lock);return err;
}
函数虽然不算太长,但也不算太短,还是有一些内容的。逐段来看:
一开始是判断并确保入参governor不为空。
接下来加锁mutex_lock(&thermal_governor_lock)。thermal_governor_lock在同文件(drivers/thermal/thermal_core.c)中定义并初始化,代码如下:
static DEFINE_MUTEX(thermal_governor_lock);
接下来是一个判断if (!__find_governor(governor->name))。__find_governor函数同样在drivers/thermal/thermal_core.c中,代码如下:
/** Governor section: set of functions to handle thermal governors** Functions to help in the life cycle of thermal governors within* the thermal core and by the thermal governor code.*/static struct thermal_governor *__find_governor(const char *name)
{struct thermal_governor *pos;if (!name || !name[0])return def_governor;list_for_each_entry(pos, &thermal_governor_list, governor_list)if (!strncasecmp(name, pos->name, THERMAL_NAME_LENGTH))return pos;return NULL;
}
要弄清楚这个函数的功能,就必须弄清楚list_for_each_entry的含义。list_for_each_entry是一个宏,在include/linux/list.h中,代码如下:
/*** list_for_each_entry - iterate over list of given type* @pos: the type * to use as a loop cursor.* @head: the head for your list.* @member: the name of the list_head within the struct.*/
#define list_for_each_entry(pos, head, member) \for (pos = list_first_entry(head, typeof(*pos), member); \!list_entry_is_head(pos, head, member); \pos = list_next_entry(pos, member))
list_first_entry当然也在include/linux/list.h中,代码如下:
/*** list_first_entry - get the first element from a list* @ptr: the list head to take the element from.* @type: the type of the struct this is embedded in.* @member: the name of the list_head within the struct.** Note, that list is expected to be not empty.*/
#define list_first_entry(ptr, type, member) \list_entry((ptr)->next, type, member)
list_entry也在include/linux/linux.h中,就在list_first_entry宏定义的上边,代码如下:
/*** list_entry - get the struct for this entry* @ptr: the &struct list_head pointer.* @type: the type of the struct this is embedded in.* @member: the name of the list_head within the struct.*/
#define list_entry(ptr, type, member) \container_of(ptr, type, member)
由上,list_first_entry展开为:
#define list_first_entry container_of((ptr)->next, type, member)
list_entry_is_head同样在include/linux/linux.h中,代码如下:
/*** list_entry_is_head - test if the entry points to the head of the list* @pos: the type * to cursor* @head: the head for your list.* @member: the name of the list_head within the struct.*/
#define list_entry_is_head(pos, head, member) \(&pos->member == (head))
list_next_entry同样在include/linux/linux.h中,代码如下:
/*** list_next_entry - get the next element in list* @pos: the type * to cursor* @member: the name of the list_head within the struct.*/
#define list_next_entry(pos, member) \list_entry((pos)->member.next, typeof(*(pos)), member)
相关文章:
Linux内核Thermal框架详解五、Thermal Core(4)
本文部分内容参考Linux Thermal 学习笔记 - 爱码网。特此致谢! 接前一篇文章Linux内核Thermal框架详解四、Thermal Core(3) 三、相关源码及分析 2. thermal_register_governors 上一回说到这一段代码: for (__governor __gove…...
gcc 编译的过程
#include <stdio.h> #define PI 3.14 int main(int argc, char const *argv[]) { //打印IP的值printf("PI %lf\n", PI);return 0; }编译的过程:预处理、编译、汇编、链接1.预处理:宏替换、删除注释、头文件包含、条件编译 -E …...
Hadoop入个门
文章目录1️⃣、Hadoop概述1.1、Hadoop是什么1.2、三大发行版本1.3、优势1.4、组成HDFSYARNMapReduceHDFS、YARN、MapReduce三者关系1.6、大数据技术生态体系image-202303111027195802️⃣、Hadoop运行环境搭建2.1、虚拟机环境准备2.2、克隆虚拟机2.3、在hadoop2上安装JDK2.4、…...
python 从0到批量下载某站视频
简介:真实从0到1,童叟无欺~ 目标:用python批量下载某站搜索视频,以“CG 服装”为例 本章主要介绍如何用python把搜索到的视频直接下载到自己的本地文件夹中~ 介绍一下工作流1. 下载并安装python2. 测试pyt…...
【深度学习】神经网络和深度学习--卷积和池化的作用
深度学习通常指训练大型深度的神经网络的过程。 与传统的神经网络模型相比,深度学习模型在结构上与之非常相似;不同的是,深度学习模型的“深度”更大,“深度”的体现就是神经网络层数多,神经网络每一层的结点数多。 本…...
锦正茂风冷系列电源JCP-10-80的技术参数
JCP-10-80为高稳定度的双极性恒流电源,广泛应用于电磁铁、亥姆霍兹线圈等感性负载的励磁。电源采用线性电源结构,输出电流稳定度高,纹波和噪声低。电源输出电流可在正负额定电流*值之间连续变化,电流平滑连续过零,可使…...
Idea+maven+spring-cloud项目搭建系列--11-1 dubbo(zookeeper,nacos)注册中心
前言:dubbo rpc 服务需要将服务提供者和消费者信息进行注册,以便于消费端可以完成远程接口调用;注册中心是 Dubbo 服务治理的核心组件,Dubbo 依赖注册中心的协调实现服务(地址)发现,自动化的服务…...
Python3入门教程||Python3 迭代器与生成器||Python3 函数
Python3 迭代器与生成器 迭代器 迭代是 Python 最强大的功能之一,是访问集合元素的一种方式。。 迭代器是一个可以记住遍历的位置的对象。 迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束。迭代器只能往前不会后退。 迭代器有两…...
快速幂算法
快速幂算法 文章目录快速幂算法一、简单介绍二、计算7107^{10}710三、一般化1、计算ana^nan的快速方法:2、时间复杂度分析:四、代码五、参考资料一、简单介绍 快速幂(Exponentiation by squaring,平方求幂)是一种简…...
Hudi:问题总结(2)Flink-1.13.1消费kafka并插入hudi
问题一:java.lang.ClassNotFoundException: com.google.protobuf.MessageOrBuilder) 解决:字面意思,没找到类,将protobuf-java-3.2.0-jar包放到fink/lib/下 如果报commons-cli相关的错,就将commons-cli-1.4.jar放到f…...
Application工具方法
//注册这个接口registerActivityLifecycleCallbacks(activityLifecycleCallbacks);}Overridepublic void onTerminate() {//注销这个接口。unregisterActivityLifecycleCallbacks(activityLifecycleCallbacks);super.onTerminate();}public static List<Activity> activi…...
电脑游戏怎么录屏?其实很简单,只需要简单3步
电脑游戏一直是游戏爱好者最热衷的游戏之一。但是,有时候我们想分享我们在游戏中的精彩时刻,或者记录我们的游戏过程以便后续观看和学习。在这种情况下,录屏就成了必不可少的工具。但是,许多人可能不知道电脑游戏怎么录屏。在本文…...
【设计模式】go语言中的 [函数选项,单例,工厂,责任链] 常用的设计模式
文章目录前言一、函数选项模式二、单例模式三、工厂模式四、责任链模式前言 宿舍每人 温度38℃+ 大寄 设计模式很重要,设计模式其实就是为了解决某一类问题而形成的代码写法,设计模式很多,但是并不是每个都很常用,我们…...
2017系统分析师案例分析真题背记内容
前言 以下内容仅为个人根据当年系分案例真题问题整理的偏需要记背的考点答案,方便个人背诵和记忆使用。方便文字转语音,所以内容全为纯文字内容,以下内容仅供参考。 背记内容 微服务 微服务中应该包含的内容有:资源、对资源的…...
C++和C的区别
答:从宏观角度和微观角度分析微观角度:函数原型有区别,在c中,函数原型有参数和没有参数是不同的,并且允许申明多个同名的函数,只要他们的参数列表不同或者返回值不同即可,但是在c语言中不能。C引…...
【React教程】一、React简介
一、React简介 React是一个用于构建用户界面的JavaScript库,它是Facebook的内部项目,用来架设Instagram的网站,并于2013年5月开源。React主要用于构建Ul,很多人认为React 是 MVC 中的 V(视图)。由于拥有较高的性能&…...
运动蓝牙耳机什么牌子好,比较好的运动蓝牙耳机推荐
现在市面上的运动蓝牙耳机越来越多,在选择耳机的时候应该如何入手呢?最重要的是需要按照自己的需求来选择,但在耳机的配置上不能忽视的是耳机的防水等级,运动耳机对防水等级的要求更高,这样能够更好地防御汗水浸湿耳机…...
[深入理解SSD系列 闪存实战2.1] NAND FLASH特性串烧 | 不了解闪存特性,你能用好闪存产品吗?
前言 为了利用好闪存, 发挥闪存的优势, 以达到更好的性能和使用寿命, 那自然要求了解闪存特性。 闪存作为一种相对较新的存储介质, 有很多特别的特性。 一.闪存的特性 凡是采用Flash Memory的存储设备,可以统称为闪存存储。我们经常谈的固态硬盘(SSD),可以由volatile/…...
DJI ROS dji_sdk 源码分析|整体框架
DJI ROS dji_sdk 源码分析|整体框架launch文件CMakeLists.txtcpp文件main.cppOSDK 是一个用于开发无人机应用程序的开发工具包,基于OSDK 开发的应用程序能够运行在机载计算机上(如Manifold 2),开发者通过调用OSDK 中指定的接口能够…...
HT32合泰单片机开发环境搭建和配置教程
HT32合泰(Holtek)单片机开发环境搭建安装教程 前言 最近在准备合泰杯的比赛,在看合泰官方的PPT和数据手册学习,顺便做个合泰单片机的开发环境搭建教程。 合泰杯比赛发放的开发板是ESK32-30501,用的单片机是HT32F52352。 合泰杯官网地址&a…...
OpenClaw+ollama-QwQ-32B:自动化技术文档翻译与校对
OpenClawollama-QwQ-32B:自动化技术文档翻译与校对 1. 为什么需要自动化文档处理 作为技术文档工程师,我每天要处理大量多语言技术文档。传统工作流中,翻译、术语统一和格式校对这些重复性工作消耗了至少40%的有效工作时间。更痛苦的是&…...
GP2Y0A02YK0F红外测距传感器嵌入式驱动与非线性校准
1. 红外测距传感器技术解析与嵌入式实现1.1 GP2Y0A02YK0F传感器原理与特性GP2Y0A02YK0F是夏普(Sharp)公司推出的模拟输出型红外测距传感器模块,专为中距离非接触式距离检测设计。该模块采用三角测量法(Triangulation Method&#…...
10个AI概念让你从入门到精通:掌握AI产品核心技能,成为行业专家!
你调了三天 Prompt 效果还是稀烂,问算法同事他让你试试微调,你连微调和 RAG 的区别都说不清。不是你笨,是这些概念从来没人用 PM 能听懂的方式讲过。 所有人都觉得 AI PM 最重要的是会用 ChatGPT,但真正拉开差距的是你能不能在技术…...
GLM-4-9B-Chat-1M长文本推理:从百万字源码中识别安全漏洞模式与修复建议
GLM-4-9B-Chat-1M长文本推理:从百万字源码中识别安全漏洞模式与修复建议 1. 引言:当代码审计遇上百万字长文本 想象一下,你面对的是一个拥有数十万行、甚至上百万行代码的庞大项目。传统的代码审计工具,无论是静态分析还是人工审…...
GIS数据处理避坑指南:如何正确导入CGCS2000坐标系的CSV文件
GIS数据处理避坑指南:如何正确导入CGCS2000坐标系的CSV文件 在GIS数据处理工作中,坐标系的选择与数据导入是基础却极易出错的环节。许多初学者甚至有一定经验的数据工程师,都曾在CGCS2000坐标系下的CSV文件导入过程中踩过坑——明明数据检查无…...
VSAR总线回放功能详解:在线回放 vs 离线回放,你选对了吗?
在汽车总线开发与测试过程中,总线回放是一项非常实用的功能。无论是调试ECU、复现故障,还是分析历史数据,VSAR都能帮你轻松搞定!今天我们就来聊聊VSAR中的两大回放模式:在线回放 和 离线回放,看看它们有什么…...
阿里云代理商:阿里云无影云电脑部署 OpenClaw 接入钉钉机器人全攻略
在数智化办公时代,将 AI 助手与办公工具深度集成已成为企业提效的关键。阿里云无影云电脑支持 OpenClaw 部署后,接入钉钉机器人可实现智能问答、任务处理等自动化场景。以下为详细接入指南,助您一步到位!一、准备工作阿里云无影云…...
eDiary使用教程
eDiary使用教程CSDN文章 前言 在信息爆炸的今天,我们每天都有太多的思绪、工作笔记、生活点滴需要记录,却又担心隐私泄露,或是被臃肿的笔记软件拖慢效率。如果你也在寻找一款轻量、安全、无广告的本地记录工具,那么eDiary 电子日…...
Qwen3-32B-Chat百度技术生态适配:如何将私有服务接入百度智能云AI工作流
Qwen3-32B-Chat百度技术生态适配:如何将私有服务接入百度智能云AI工作流 1. 镜像概述与核心优势 Qwen3-32B-Chat 私有部署镜像专为 RTX 4090D 24GB 显存环境深度优化,基于 CUDA 12.4 和驱动 550.90.07 构建,提供开箱即用的大模型推理能力。…...
【源码解析】DolphinScheduler动态传参核心机制:从VarPool到下游Task的数据流转
1. 揭开DolphinScheduler动态传参的神秘面纱 第一次接触DolphinScheduler的任务传参功能时,我完全被各种参数传递方式搞晕了。官方文档里介绍的静态传参很好理解,就是在界面上提前配置好参数名和值。但实际开发中,我们经常遇到这样的场景&…...
