攻城狮解析丨时钟使用之注册和获取(一)

我们在编写或修改驱动时,经常会遇到时钟相关的问题,不知道从什么地方下手。在本文中,以i.MX6的3.0.35版本的内核举例时钟如何获取和使用。

我们常见的获取时钟的方法
? 通过名称获取
例如:获取时钟clko

clko = clk_get(NULL, "clko_clk");

? 通过设备获取
例如在音频接口ssi的驱动中获时钟,

ssi->clk = clk_get(&pdev->dev, NULL); //设备的名字是 "imx-ssi"

我们可以通过查看函数clk_get的原型来进一步查看始终是怎么获取的

struct clk clk_get(struct device dev, const char *con_id)

{

    const char *dev_id = dev ? dev_name(dev) : NULL;

     return clk_get_sys(dev_id, con_id);

}

我们继续查看函数clk_get_sys,此函数通过设备的名字或时钟的名字来获取时钟

struct clk clk_get_sys(const char dev_id, const char *con_id)

{

     struct clk_lookup *cl;

     mutex_lock(&clocks_mutex); 

     cl = clk_find(dev_id, con_id);

     if (cl && !__clk_get(cl->clk))

             cl = NULL;

     mutex_unlock(&clocks_mutex);

     return cl ? cl->clk : ERR_PTR(-ENOENT);

}

此函数调用clk_find获取时钟,在时钟列表内按照设备名和时钟名查找,并返回获取到的时钟。

static struct clk_lookup clk_find(const char dev_id, const char *con_id)

{

     struct clk_lookup *p, *cl = NULL;

     int match, best = 0;

     list_for_each_entry(p, &clocks, node) {

             match = 0;

             if (p->dev_id) {

                     if (!dev_id || strcmp(p->dev_id,dev_id))

                             continue;

                     match += 2;

             } // 先找设备名

             if (p->con_id) {//再找时钟名

                     if (!con_id || strcmp(p->con_id,con_id))

                             continue;

                     match += 1;

             }

             if (match > best) {

                     cl = p;

                     if (match != 3)

                             best=match;

                     else

                             break;

             }

     }

     return cl;}

时钟的注册
刚才提到了从时钟列表内按照设备或时钟名称获取时钟,时钟列表是怎么生成的。

我们在i.Mx6板级文件初始化里调用了时钟初始化函数。

static void __init mx6_sabresd_timer_init(void)

{

     struct clk *uart_clk;

#ifdef CONFIG_LOCAL_TIMERS

     twd_base = ioremap(LOCAL_TWD_ADDR,SZ_256);

     BUG_ON(!twd_base);

#endif

     mx6_clocks_init(32768, 24000000, 0, 0);

     uart_clk = clk_get_sys("imx-uart.0", NULL);

     early_console_setup(UART1_BASE_ADDR, uart_clk);

}

我们查看 mx6_clocks_init函数

int __init mx6_clocks_init(unsigned long ckil, unsigned long osc,

     unsigned long ckih1, unsigned long ckih2)

{

    …

     for (i = 0; i < ARRAY_SIZE(lookups); i++){

             clkdev_add(&lookups[i]);

             clk_debug_register(lookups[i].clk);

     }

….

}

先看函数内 lookups数组内定了许多时钟

static struct clk_lookup lookups[] = {

 ……

_REGISTER_CLOCK("imx-ssi.0", NULL, ssi1_clk),

     _REGISTER_CLOCK("imx-ssi.1", NULL, ssi2_clk),

     _REGISTER_CLOCK("imx-ssi.2", NULL, ssi3_clk),

_REGISTER_CLOCK(NULL, "clko_clk", clko_clk),

……

}

clk_lookup的定义

struct clk_lookup {

     struct list_head          node;

     const char                *dev_id;

     const char                *con_id;

     struct clk                *clk;

};

clk_lookup的赋值

#define _REGISTER_CLOCK(d, n, c) \

     {
             .dev_id = d,
             .con_id = n,
             .clk = &c,
     }

再看函数 clkdev_add

void clkdev_add(struct clk_lookup *cl){

     mutex_lock(&clocks_mutex);

     list_add_tail(&cl->node, &clocks);

     mutex_unlock(&clocks_mutex);

}

由此我们可以发现mx6_clocks_init函数将数组lookups内的时钟,全部添加到了链表clocks内。我们再通过第一步的方法获取lookups内定义的时钟。

原文地址:https://blog.51cto.com/14771178/2485915

时间: 2024-11-10 08:28:44

攻城狮解析丨时钟使用之注册和获取(一)的相关文章

攻城狮解析丨开发板电源架构解析之OK4418-C(下)

上一期对 <OKMX6UL-C开发板底 板电源架构> 进行了分析,相信对用户是有一定的帮助的.这一期准备对同样有点复杂的OKxx18_C的电源架构做一个详细的梳理. 我们要知道飞凌S5P4418/688系列 FETxx18核 心板 支持以下三种供电方式: ? ADP供电,通过J1(电源插座)供入底板,经过U11(FDS4435)直接供给核心板PMIC. ? USB供电,由OTG接口直接供给核心板PMIC. ? VBAT供电,由锂电池供给核心板PMIC. ▲核心板PMIC(NXE2000)锂电池

攻城狮自述丨OK1043A-C DPDK环境体验

飞凌嵌入式今年6月推出的--FET1043A-C核心板 ,采用NXP公司的QorIQ? LS1043A处理器设计,拥有四颗ARMv8-A架构的Cortex-A53核,主频1.6G,低功耗,高能效.四通道高达10GB的SerDes包含多种灵活配置,在飞凌设计的配套底板中最大程度上发挥了QorIQ? LS1043A 处理器网络性能,采用一个万兆,六个千兆的设计,配合处理器内部的DPAA1加速引擎,再加上2GB大容量DDR4内存简直就是网络性能怪兽. 硬件如此的优秀,那么传统的Linux内核还是否与其

攻城狮分享丨i.MX6X的uboot自动适配logo的配置方案

有很多客户对于调试屏幕有很多困扰,经常在自己调试屏幕的时候,出现各种各样的问题.为了帮助大家更好更快的调试屏幕,飞凌研发工程师又 在 i.MX6X的uboot 阶段添加了一项新的功能,让屏幕logo适配更方便更快捷 ,我们一起来看看这一项新的功能吧. 该功能目前适配内核系统为Linux3.0.35版本的i.MX6X平台,操作起来是很简单的,我们只需要把屏幕的分辨率以及刷新频率设置一下,大部分的屏幕就都可以自己适配了,方便又省心. 下面我们就来看看如何操作吧! IMX6X开发板 上电后 3秒钟之内

转:攻城狮修炼秘籍

漫校园生活,孜孜不倦的你,今天书多读一点! 海康威视为你的假期精心准备了精神的饕餮大餐! 把自己投入知识的海洋吧!为即将成为一名高冷俊俏的程序猿和媛而吹响最后的号角: 各类秘籍自助区 一.测试攻城狮 <软件测试> <测试之道> 二.硬件攻城狮 基础理论:工程数学.积分变换.信号与系统.数字信号处理.自动控制原理.英语 专业课程:数电.模电.嵌入式系统.热设计.DDR3.flash.开关电源 (若有实践经验,可只看基础理论) 三.应用软件开发攻城狮 必读: <UNIX环境高级编

【云栖大会】程序猿、攻城狮的大聚会 他们眼中的云栖大会是啥样

2016年云栖大会进行得如火如荼,"烧脑"."前沿"."高端"."技术"."一票难求"--这些关键词把这场"干货"分享大会衬托得神秘满满,有一群人却沉浸其中,自得其乐. 他们的名字叫--程序猿.攻城狮. 从最初的站长大会,到阿里云开发者大会,再到云栖大会:从200名草根站长到4万名业内外参与者:从单一峰会发展至超过100场论坛的"巨无霸"大会:从纯谈技术到融合音乐节.

攻城狮在路上(叁)Linux(三十)--- 光盘写入工具

一.基本步骤: 1.用mkisofs命令将所需备份的数据构建成镜像文件. 2.用cdrecord命令将镜像文件刻录至光盘或者DVD中. 二.mkisofs:新建镜像文件 mkisofs [-0 镜像文件] [-rv] [-m file] 待备份文件... [-V vol] graft -point isodir=Systemdir... 参数说明: -o:后面指定镜像文件 -r:通过RockRidge产生支持UNIX/Linux的文件数据,可以记录较多信息. -v:显示构建过程 -m file:

攻城狮在路上(叁)Linux(二十九)--- 完整备份工具:dump以及restore

一.dump命令: 该命令既可以针对整个文件系统进行备份,也可以仅针对目录来备份.还可以指定不同的备份等级(-0~-9共10个等级). dump -W:列出在/etc/fstab中具有dump设置的分区是否备份过. 命令格式: dump [-Suvj] [-level] [-f 备份文件] 待备份数据 参数说明: -S:仅列出后面的待备份数据所需要的磁盘空间大小. -u:将这次dump的时间记录到/etc/dumpdates文件中. -v:将dump的文件过程显示出来. -j:加入bzip2的支

攻城狮在路上(叁)Linux(二十五)--- linux内存交换空间(swap)的构建

swap的功能是应付物理内存不足的状况,用硬盘来暂时放置内存中的信息. 对于一般主机,物理内存都差不多够用,所以也就不会用到swap,但是对于服务器而言,当遇到大量网络请求时或许就会用到. 当swap被使用的时候,主机的硬盘灯就会闪烁不停. 本篇介绍两种方式:1.设置一个swap分区   2.创建一个虚拟内存的文件. 一.使用物理分区构建swap: 1.首先是分区: A.fdisk /dev/sda; <== 根据后续提示创建一个分区. B.修改分区的ID,因为fdisk默认将分区的ID作为文件

攻城狮在路上(叁)Linux(二十六)--- linux文件系统的特殊查看与操作

一.boot sector 与 super block的关系: 1.boot sector用于存放引导装载程序,占用1024个字节. 2.super block的大小也为1024字节. 3.若block大小为1k,则boot sector和super block各占一个block. 4.若block大于1K(2K/4K)时,则两者都位于第一个block中. 二.磁盘空间的浪费问题:暂不考虑. 三.利用GUN的parted命令进行分区行为: 因为fdisk不支持高于2TB的分区. 命令格式: pa