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

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了攻城狮解析丨时钟使用之注册和获取相关的知识,希望对你有一定的参考价值。

我们在编写或修改驱动时,经常会遇到时钟相关的问题,不知道从什么地方下手。在本文中,以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内定义的时钟。

以上是关于攻城狮解析丨时钟使用之注册和获取的主要内容,如果未能解决你的问题,请参考以下文章

大数据攻城狮之进阶技能-Github的使用

IC攻城狮求职宝典05钜泉光电笔试题

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

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

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

前端攻城狮该了解的 Vue.2x 响应式原理