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

原创 2020-03-05 17:00:00 时钟 i.MX6.驱动

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

相关产品 >

  • FETMX6Q-C核心板

    i.MX6Q核心板板层出不穷,要如何选择?飞凌解读i.mx6Q芯片超强性能为您推荐四核A9架构的i.MX6Q产品精选,包含iMX6Q 核心板、i.MX6Q 核心板、iMX6Q工业级核心板,欢迎采购。  i.MX6Q核心板基于NXP(原Freescale)Cortex-A9架构的i.MX6Q四核处理器设计,核心板小尺寸核心板搭配独特的超薄连接器,让设计随心所欲!

    了解详情
    FETMX6Q-C核心板
  • OKMX6Q-C开发板

    双千兆,可达限速 Ubuntu,OpenWrt|飞凌嵌入式iMX6Q开发板,板对板连接器,纤薄之际,次底板支持iMX6Q和iMX6DL核心板。i.MX6Q开发板与i.MX6DL开发板资源丰富,原理图、PCB、软件资源、硬件资源下载,技术支持等。欢迎选购

    了解详情
    OKMX6Q-C开发板
  • FETMX6Q-S核心板

    NXP iMX6Quad系列具有四个内核,运行频率达1.2 GHz,带有1 MB L2缓存和64位DDR3或2通道、32位LPDDR2支持。飞凌提供商业级iMX6Q核心板,工业级iMX6Q核心板,兼容一同底板。具有抗震,抗氧化,抗干扰,更快速升级产品等优势。保定飞凌嵌入式专注imx6,imx6开发板,飞思卡尔imx6等ARM嵌入式核心控制系统研发、设计和生产,是imx6,imx6开发板,飞思卡尔imx6专业提供者,imx6系列产品现已畅销全国,欢迎咨询!
    了解详情
    FETMX6Q-S核心板
  • OKMX6Q-S3开发板

    飞凌嵌入式提供iMX6Q开发板,iMX6解决方案,iMX6Q核心板,i.MX6Q开发板解决方案。iMX6Q稳定、快速、性价比高,欢迎选购 NXP iMX6系列芯片全支持,升级简配无忧替换。 了解详情
    OKMX6Q-S3开发板

推荐阅读 换一批 换一批