电源管理入门-12 clock驱动

news2026/4/2 1:42:14
电源管理的两个大方面就是电压和时钟。Clock 时钟就是 SoC 中的脉搏由它来控制各个部件按各自的节奏跳动。比如CPU主频设置串口的波特率设置I2S的采样率设置I2C的速率设置等等。这些不同的clock设置都需要从某个或某几个时钟源头而来最终开枝散叶形成一棵时钟树。1. clock驱动构架Linux的时钟子系统由CCFcommon clock framework框架管理CCF向上给用户提供了通用的时钟接口向下给驱动开发者提供硬件操作的接口。这个也是一个consumer、framework、provide****r的模式。其中其provider会比较复杂一些但是往往是由芯片厂商提供我们编写设备驱动要使用调频的时候只需要在consumer里面进行配置使用就可以了。1.1 Clock Provider介绍在SoC上器件很多会形成一个时钟树如下所示根节点一般是Oscillator有源振荡器或者Crystal无源振荡器。中间节点有很多种包括PLL锁相环用于提升频率的Divider分频器用于降频的Mux从多个clock path中选择一个Gate用来控制ON/OFF的。叶节点是使用 clock 作为输入的、有具体功能的 HW block。可通过cat /sys/kernel/debug/clk/clk_summary 查看这棵时钟树。image.png1.2 clock consumer介绍时钟的使用者clock子系统向consumer的提供通用的时钟API接口使其可以屏蔽底层硬件差异。提供给consumer操作的API如下struct clk *clk_get(struct device *dev, const char *id);struct clk *devm_clk_get(struct device *dev, const char *id);int clk_enable(struct clk *clk);//使能时钟不会睡眠 void clk_disable(struct clk *clk);//使能时钟不会睡眠 unsigned long clk_get_rate(struct clk *clk);void clk_put(struct clk *clk);long clk_round_rate(struct clk *clk, unsigned long rate);int clk_set_rate(struct clk *clk, unsigned long rate);int clk_set_parent(struct clk *clk, struct clk *parent);struct clk *clk_get_parent(struct clk *clk);int clk_prepare(struct clk *clk);void clk_unprepare(struct clk *clk);int clk_prepare_enable(struct clk *clk)//使能时钟可能会睡眠 void clk_disable_unprepare(struct clk *clk)//禁止时钟可能会睡眠 unsigned long clk_get_rate(struct clk *clk)//获取时钟频率2. Clock Provider根据 clock 的特点clock framework 将 clock 分为fixed rate、gate、devider、mux、fixed factor、composite六类。2.1 数据结构表示上面六类本质上都属于clock device内核把这些 clock HW block 的特性抽取出来用 struct clk_hw 来表示具体如下struct clk_hw{//指向CCF模块中对应 clock device 实例 struct clk_core *core;//clk是访问clk_core的实例。每当consumer通过clk_get对CCF中的clock device也就是clk_core发起访问的时候都需要获取一个句柄也就是clk struct clk *clk;//clock provider driver初始化时的数据数据被用来初始化clk_hw对应的clk_core数据结构。 const struct clk_init_data *init;};struct clk_init_data{//该clock设备的名字 const char *name;//clock provider driver进行具体的 HW 操作 const struct clk_ops *ops;//描述该clk_hw的拓扑结构 const char * const *parent_names;const struct clk_parent_data *parent_data;const struct clk_hw **parent_hws;u8 num_parents;unsigned long flags;};以固定频率的振动器 fixed rate 为例它的数据结构是struct clk_fixed_rate{//下面是fixed rate这种clock device特有的成员 struct clk_hw hw //基类 unsigned long fixed_rate;unsigned long fixed_accuracy;u8 flags;};其他clock硬件的表示也是如此。2.2 clock provider注册初始化clock驱动在时钟子系统中属于providerprovider是时钟的提供者即具体的clock驱动。clock驱动在Linux刚启动的时候就要完成比initcall都要早期因此clock驱动是在内核中进行实现。这里也叫clock device例如上面说的fixed rate属于硬件提供服务的。在其启动的时候根据DTS里面的配置进行注册。例如fixed rateCLK_OF_DECLARE(fixed_clk,fixed-clock, of_fixed_clk_setup);struct clk *clk_register_fixed_rate(struct device *dev, const char *name, const char *parent_name, unsigned long flags, unsigned long fixed_rate);其他的device如下clk_register_gate clk_register_divider clk_register_divider_table clk_register_mux clk_register_mux_table clk_register_fixed_factor clk_register_composite这些注册函数最终都会通过函数clk_register注册到 Common Clock Framework 中返回为 struct clk 指针。如下所示在内核的drivers/clk目录下可以看到各个芯片厂商对各自芯片clock驱动的实现2.3 DTS配置例如时钟源clocks{osc24M:osc24M{compatiblefixed-clock;#clock-cells 0;clock-output-nameosc24M;clock-frequency24000000;};};属性说明compatible驱动匹配名字#clock-cells提供输出时钟的路数。#clock-cells为0时代表输出一路时钟 #clock-cells为1时代表输出2路时钟。#clock-output-names输出时钟的名字#clock-frequency输出时钟的频率clock驱动编写的基本步骤实现struct clk_ops相关成员函数定义分配struct clk_onecell_data结构体初始化相关数据定义分配struct clk_init_data结构体初始化相关数据调用clk_register将时钟注册进框架调用clk_register_clkdev注册时钟设备调用of_clk_add_provider将clk provider存放到of_clk_provider链表中管理调用CLK_OF_DECLARE声明驱动2.4 clock驱动实现举例这里以fixed_clk为例fixed_clk针对像PLL这种具有固定频率的时钟对于PLL我们只需要实现.recalc_rate函数。设备树#define PLL0_CLK 0clocks{osc24M:osc24M{compatiblefixed-clock;#clock-cells 0;clock-output-namesosc24M;clock-frequency24000000;};pll0:pll0{compatiblexx, choogle-fixed-clk;#clock-cells 0;clock-idPLL0_CLK;clock-frequency1000000000;clock-output-namespll0;clocksosc24M;};};驱动#include linux/clk-provier.h#include linux/clkdev.h#include linux/clk.h#include linux/module.h#include linux/of.h#include linux/of_address.h#include linux/platform_device.h#include linux/slab.h#include linux/delay.h#define CLOCK_BASE 0X12340000#define CLOCK_SIZE 0X1000struct xx_fixed_clk{void __iomem *reg;//保存映射后寄存器基址 unsigned long fixed_rate;//频率 intid;//clockidstruct clk_hw*;} static unsigned long xx_pll0_fixed_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate){unsigned long recalc_rate;//硬件操作查询寄存器获得分频系数计算频率然后返回returnrecalc_rate;}static struct clk_ops xx_pll0_fixed_clk_ops{.recalc_ratexx_pll0_fixed_clk_recalc_rate,};struct clk_ops *xx_fixed_clk_ops[]{xx_pll0_fixed_clk_ops,};struct clk * __init xx_register_fixed_clk(const char *name, const char *parent_name, void __iomem *res_reg, u32 fixed_rate, int id, const struct clk_ops *ops){struct xx_fixed_clk *fixed_clk;struct clk *clk;struct clk_init_data init{};fixed_clkkzalloc(sizeof(*fixed_clk), GFP_KERNEL);if(!fixed_clk)returnERR_PTR(-ENOMEM);//初始化struct clk_init_data数据 init.namename;init.flagsCLK_IS_BASIC;init.parent_namesparent_name ?parent_name:NULL;init.num_parentsparent_name ?1:0;fixed_clk-regres_reg;//保存映射后的基址 fixed_clk-fixed_ratefixed_rate;//保存频率 fixed_clk-idid;//保存clockidfixed_clk-hw.initinit;//时钟注册 clkclk_register(NULL,fixed_clk-hw);if(IS_ERR(clk))kfree(fixed_clk);returnclk;}static void __init of_xx_fixed_clk_init(struct device_node *np){struct clk_onecell_data *clk_data;const char *clk_namenp-name;const char *parent_nameof_clk_get_parent_name(np,0);void __iomem *res_regioremap(CLOCK_BASE, CLOCK_SIZE);//寄存器基址映射 u32 rate-1;int clock_id, index, number;clk_datakmalloc(sizeof(struct clk_onecell_data), GFP_KERNEL);if(!clk_data)return;numberof_property_count_u32_elems(np,clock-id);clk_data-clkskcalloc(number, sizeof(struct clk*), GFP_KERNEL);if(!clk_data-clks)goto err_free_data;of_property_read_u32(np,clock-frequency,rate);/** * 操作寄存器初始化PLL时钟频率 *......*/for(index0;indexnumber;index){of_property_read_string_index(np,clock-output-names, index,clk_name);of_property_read_u32_index(np,clock-id, index,clock_id);clk_data-clks[index]xx_register_fixed_clk(clk_name, parent_name, res_reg, rate, clock_id, ak_fixed_clk_ops[pll_id]);if(IS_ERR(clk_data-clks[index])){pr_err(%s register fixed clk failed: clk_name:%s, index %d\n, __func__, clk_name, index);WARN_ON(true);continue;}clk_register_clkdev(clk_data-clks[index], clk_name, NULL);//注册时钟设备}clk_data-clk_numnumber;if(number1){of_clk_add_provider(np, of_clk_src_simple_get, clk_data-clks[0]);}else{of_clk_add_provider(np, of_clk_src_onecell_get, clk_data);}return;err_free_data: kfree(clk_data);}CLK_OF_DECLARE(xx_fixed_clk,xx,xx-fixed-clk, of_xx_fixed_clk_init);3. clock consumer主要就是获取clock和操作clock。3.1 获取clock即通过 clock 名称获取 struct clk 指针的过程由 clk_get、devm_clk_get、clk_get_sys、of_clk_get、of_clk_get_by_name、of_clk_get_from_provider 等接口负责实现这里以 clk_get 为例分析其实现过程struct clk *clk_get(struct device *dev, const char *con_id){const char *dev_iddev ? dev_name(dev):NULL;struct clk *clk;if(dev){//通过扫描所有“clock-names”中的值和传入的name比较如果相同获得它的index即“clock-names”中的第几个调用of_clk_get取得clock指针。 clk__of_clk_get_by_name(dev-of_node, dev_id, con_id);if(!IS_ERR(clk)||PTR_ERR(clk)-EPROBE_DEFER)returnclk;}returnclk_get_sys(dev_id, con_id);}3.2 操作clock//启动clock前的准备工作/停止clock后的善后工作。可能会睡眠。 int clk_prepare(struct clk *clk)void clk_unprepare(struct clk *clk)//启动/停止clock。不会睡眠。 static inline int clk_enable(struct clk *clk)static inline void clk_disable(struct clk *clk)//clock频率的获取和设置 static inline unsigned long clk_get_rate(struct clk *clk)static inline int clk_set_rate(struct clk *clk, unsigned long rate)static inline long clk_round_rate(struct clk *clk, unsigned long rate)//获取/选择clock的parent clock static inline int clk_set_parent(struct clk *clk, struct clk *parent)static inline struct clk *clk_get_parent(struct clk *clk)//将clk_prepare和clk_enable组合起来一起调用。将clk_disable和clk_unprepare组合起来一起调用 static inline int clk_prepare_enable(struct clk *clk)static inline void clk_disable_unprepare(struct clk *clk)3.3 实例操作我们在驱动consumer开发的时候需要使用clock这时需要在DTS里面配置例如mmc设备mmc0:mmc00x12345678{compatiblexx,xx-mmc0;......clocksperi PERI_MCI0;//指定mmc0的时钟来自PERI_MCI0PERI_MCI0的父时钟是peri clocks-namesmmc0;//时钟名调用devm_clk_get获取时钟时可以传入该名字......};以mmc的设备节点为例上述mmc0指定了时钟来自PERI_MCI0PERI_MCI0的父时钟是peri并将所指定的时钟给它命名为mmc0。使用方法如下/*1、获取时钟 */ host-clkdevm_clk_get(pdev-dev, NULL);//或者devm_clk_get(pdev-dev,mmc0)if(IS_ERR(host-clk)){dev_err(dev,failed to find clock source\n);retPTR_ERR(host-clk);goto probe_out_free_dev;}/*2、使能时钟 */ retclk_prepare_enable(host-clk);if(ret){dev_err(dev,failed to enable clock source.\n);goto probe_out_free_dev;}probe_out_free_dev: kfree(host);在驱动中操作时钟第一步需要获取struct clk指针句柄后续都通过该指针进行操作例如设置频率retclk_set_rate(host-clk,300000);获得频率retclk_get_rate(host-clk);注意devm_clk_get()的两个参数是二选一可以都传入也可以只传入一个参数。像i2c、mmc等这些外设驱动通常只需要使能门控即可因为这些外设并不是时钟源它们只有开关。如果直接调用clk_ser_rate函数设置频率clk_set_rate会向上传递即设置它的父时钟频率。例如在该例子中直接调用clk_set_rate函数最终设置的是时钟源peri的频率。4. SoC硬件中的使用在硬件中一般将clock的控制和reset搞到一起形成一个CRUclock reset unit。每个子系统例如NPU需要有自己独立的CRUCRU里面有PLL参考https://cloud.tencent.com/developer/article/1928685https://zhuanlan.zhihu.com/p/605593587https://bbs.16rd.com/thread-572748-1-1.html后记电源管理写了这么多篇慢慢套路就可以摸清楚了抓住主要构架思路剩下的就是招式问题了。DTS、consumerframework、provier、硬件树形组织等。先弄清楚这些东西不论调试什么问题就比较快了。“啥都懂一点啥都不精通干啥都能干干啥啥不是专业入门劝退堪称程序员杂家”。欢迎各位自己有博客公众号的留言申请转载多谢后续会继续更新纯干货分析欢迎分享给朋友欢迎点赞、收藏、在看、划线和评论交流公众号“那路谈OS与SoC嵌入式软件”欢迎关注个人文章汇总https://thatway1989.github.io

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2473864.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…