浅析linux设备驱动的clock(时钟)的注册

做嵌入式 linux 驱动的时候,难免会遇到clock,今天上网查阅关于clock的资料,发现网上大多数资料都是关于linux内核的时钟机制,而不是关于Linux设备驱动的时钟。

于是将自己今天学习的经验写出来,跟大家交流交流,有不对的地方希望高手们指出。

我会以三星的smdkc220开发板为例。

分析代码, 它将clock也看作一种设备,使用前也要register注册一下,但这个register又有别于一般的设备,不会在sysfs下device,driver或bus等目录下生成node。

先看注册函数exynos4_register_clocks(void),在arch\arm\mach-exynos\Clock-exynos4.c文件里,

void __init exynos4_register_clocks(void)
{

        //........ 省略 

        s3c_register_clksrc(exynos4_clksrcs, ARRAY_SIZE(exynos4_clksrcs));
s3c_register_clocks(exynos4_init_clocks, ARRAY_SIZE(exynos4_init_clocks));


s3c_register_clocks(exynos4_init_clocks_off, ARRAY_SIZE(exynos4_init_clocks_off));
s3c_disable_clocks(exynos4_init_clocks_off, ARRAY_SIZE(exynos4_init_clocks_off));


s3c_register_clocks(exynos4_init_audss_clocks, ARRAY_SIZE(exynos4_init_audss_clocks));
s3c_disable_clocks(exynos4_init_audss_clocks, ARRAY_SIZE(exynos4_init_audss_clocks));

        

        //...........省略

        s3c_pwmclk_init();
}


三星将clock按设备分类,比如jpeg, fimc, mipi-csi, mipi-dsi等设备,就归到exynos4_init_clocks_off这个数组中,比如exynos4_clksrcs数组里,就是各种各样的source clk。

不用管上面几个register函数长得不一样,其实他们的本质都是一样的,只是他们给不同类的clock注册,就用了不同的名字而已。


在执行register前,得先定义好clock,也就是各个数组的内容。

以exynos4_init_clocks_off为例,

static struct clk exynos4_init_clocks_off[] = {

                 //....省略

           {
.name = "csis",
.devname = "s3c-csis.0",
.enable = exynos4_clk_ip_cam_ctrl,
.ctrlbit = (1 << 4),

       }, {
.name = "fimc",
.devname = "s3c-fimc.0",
.enable = exynos4_clk_ip_cam_ctrl,
.ctrlbit = (1 << 0),

}, {
.name = "jpeg",
.enable = exynos4_clk_ip_cam_ctrl,
.ctrlbit = ((1 << 11) | (1 << 6)),

        }, {
.name = "dsim0",
.enable = exynos4_clk_ip_lcd0_ctrl,
.ctrlbit = (1 << 3),
}

        //..........省略

}


这里,就定义了jpeg, fimc, mipi-dsi, mipi-csi这四种设备的clk了,还给了他们clock名字(.name),enable的函数,还指出了在clock control寄存器里哪一位是控制他们各自的(ctrlbit)。但这里虽注册一下,最近三星还加多了一个devname这个东东,这个是指设备的名字,比如fimc这个设备名字叫s3c-fimc.0, 但它的时钟名字叫"fimc"。注意要将两者区别开。以后注册设备的时候会用到(注册clock是在注册设备之前先进行的)。


接下来,细看怎么注册吧。

void __init s3c_register_clocks(struct clk *clkp, int nr_clks)
{
int ret;


for (; nr_clks > 0; nr_clks--, clkp++) {
ret = s3c24xx_register_clock(clkp);


if (ret < 0) {
printk(KERN_ERR "Failed to register clock %s (%d)\n",
      clkp->name, ret);
}
}
}


回想起刚才exynos4_init_clocks_off[]数组吧,里面定义了某些设备的clock。通过参数 clkp 将数组传进来,然后用一个for循环, 对里面各个设备的clock进行分别注册。

这里调用了s3c24xx_register_clock(clkp)来对一个设备的clock进行注册。注册完后通过指针++,又指向下一下设备的clock。


int s3c24xx_register_clock(struct clk *clk)
{
if (clk->enable == NULL)
clk->enable = clk_null_enable;


/* fill up the clk_lookup structure and register it*/
clk->lookup.dev_id = clk->devname;
clk->lookup.con_id = clk->name;
clk->lookup.clk = clk;
clkdev_add(&clk->lookup);


return 0;
}

看这个注册函数,先判断之前定义好的设备clock里,有没有enable函数,没有的话,给它一个空的,有名无实的空函数。

然后,还记得刚才定义clock的时候,还给了它clock name和device name吧? 将这两个名字分别复制到 clk 结构体中 lookup结构体身上的成员变量里。

这里我们先看看clk结构体吧, 三星是这样定义它的,跟原来linux中的不一样。


struct clk {
struct list_head      list;
struct module        *owner;
struct clk           *parent;
const char           *name;
const char *devname;
int      id;
int      usage;
unsigned long         rate;
unsigned long         ctrlbit;


struct clk_ops *ops;
int    (*enable)(struct clk *, int enable);
struct clk_lookup lookup;
#if defined(CONFIG_PM_DEBUG) && defined(CONFIG_DEBUG_FS)
struct dentry *dent; /* For visible tree hierarchy */
#endif
};

刚才定义设备的clock的时候,name给了,devname也给了,enable函数也给了,ctrlbit也给了。但现在其他成员还是空的,以后慢慢会填 满的。

现在先看看clk_lookup结构体,这个成员有什么用呢?

顾名思义,lookup,肯定是查找的时候用到的。那什么时候查找呢,找什么呢?

在这里,我们将注册设备和注册设备的clock分开操作了,注册设备的clock是一次过完成的,一次将各种设备的clock注册好的,但是我们设备的注册,是个别进行的。

所以,在我们注册设备的时候,要在这一堆先前已经注册好的clock堆中,找到属于自己的那个clock。这时候就要靠clk_lookup结构体了。


struct clk_lookup {
struct list_head node;
const char *dev_id;
const char *con_id;
struct clk *clk;
};

看这结构,里面有dev_id, con_id两个成员,它们分别代表了clock name和device name。所以又回到之前,我们要先将clock name和device name复制到这两个成员里才行。于是就有了

clk->lookup.dev_id = clk->devname;
clk->lookup.con_id = clk->name;

然后clk->lookup.clk = clk;也是同理。

再回头看 clkdev_add(&clk->lookup);这句话是什么意思呢?

先说说在注册设备时我们怎么在clock堆 里找到自己的clock吧。 是通过一条clock链的(listhead),这是一条双向循环的链。这条链的每个结点是什么呢?

就是clk_lookup结构体!

现在我们知道每个clk_lookup都包含了各自设备的一些简单信息了吧(name,devname),这些信息足够丰富而又能将不同的clk_lookup结构体分开了。

clkdev_add(&clk->lookup); 这句话的意思就是, 将clk_lookup结构体,当成一个结点node,接到这条链的尾巴上。

你来看它的详细就知道,

void clkdev_add(struct clk_lookup *cl)
{
mutex_lock(&clocks_mutex);
list_add_tail(&cl->node, &clocks);
mutex_unlock(&clocks_mutex);
}

又顾名思义,list_add_tail,就是加到尾巴上嘛。

那将一件东西接到链子上,那这个东西上应该有个小挂钩或者其他什么东西才行啊~

这个就是clk_lookup里的node成员变量(在上面函数中就是 cl->node)

这个node是什么东西,看它在clk_lookup里的定义:

struct list_head node;

我们知道,list_head结构,就像一个前面有个小挂钩,后面又有个小挂钩的东西,很多的list_head通过彼此的小挂钩互相钩住,就形成了一条链了。

于是,各个不同的clk_lookup结构体,就通过node,互相接起来,形成了一条链了。

这条链形如下:

                       clk_lookup1        clk_lookup2           clk_lookup3

                 .... ------node1-----------node2-----------------node3----------....

                                  |                         |                                 |

                             *name1            *name2                   *name3

                                  |                         |                                 |

                        *devname1        *devname2              *devname3

                                  |                         |                                 |

                                ...                        ...                               ...


这条链是一开始就定义了的,我们通过static LIST_HEAD(clocks); 定义一条静态的空链,供全体使用。然后一个结点一个结点地往上接。

将clk_lookup结构体连接到链上之后,也就是将各个设备的clock连接到这条链上了,为什么?因为clk_lookup结构体里包含了一个叫clk的成员变量。它就是clock。

至此,我们就完成了注册了。

如果问注册的目的是什么?

我想,就是形成一条链,以后让在注册设备的时候查找呗。




小弟知识浅陋,有不正确的地方请大牛们指出。也欢迎同好者一起来交流。邮箱[email protected]。转载请注明出处。





你可能感兴趣的:(linux,struct,list,hierarchy,CAM,linux内核)