攻城獅解析丨時(shí)鐘使用之注冊和獲?。ㄒ唬?/h1>
原創(chuàng)
2020-03-05 17:00:00
時(shí)鐘
i.MX6.驅(qū)動(dòng)
我們在編寫或修改驅(qū)動(dòng)時(shí),經(jīng)常會(huì)遇到時(shí)鐘相關(guān)的問題,不知道從什么地方下手。在本文中,以i.MX6的3.0.35版本的內(nèi)核舉例時(shí)鐘如何獲取和使用。
我們常見的獲取時(shí)鐘的方法
?
通過名稱獲取
例如:獲取時(shí)鐘clko
clko = clk_get(NULL, "clko_clk");
?
通過設(shè)備獲取
例如在音頻接口ssi的驅(qū)動(dòng)中獲時(shí)鐘,
ssi->clk = clk_get(&pdev->dev, NULL); //設(shè)備的名字是 "imx-ssi"
我們可以通過查看函數(shù)clk_get的原型來進(jìn)一步查看始終是怎么獲取的
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);
}
我們繼續(xù)查看函數(shù)clk_get_sys,此函數(shù)通過設(shè)備的名字或時(shí)鐘的名字來獲取時(shí)鐘
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);
}
此函數(shù)調(diào)用clk_find獲取時(shí)鐘,在時(shí)鐘列表內(nèi)按照設(shè)備名和時(shí)鐘名查找,并返回獲取到的時(shí)鐘。
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;
} // 先找設(shè)備名
if (p->con_id) {//再找時(shí)鐘名
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;}
時(shí)鐘的注冊
剛才提到了從時(shí)鐘列表內(nèi)按照設(shè)備或時(shí)鐘名稱獲取時(shí)鐘,時(shí)鐘列表是怎么生成的。
我們在i.Mx6板級文件初始化里調(diào)用了時(shí)鐘初始化函數(shù)。
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函數(shù)
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);
}
….
}
先看函數(shù)內(nèi)lookups數(shù)組內(nèi)定了許多時(shí)鐘
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, \
}
再看函數(shù)clkdev_add
void clkdev_add(struct clk_lookup *cl){
mutex_lock(&clocks_mutex);
list_add_tail(&cl->node, &clocks);
mutex_unlock(&clocks_mutex);
}
由此我們可以發(fā)現(xiàn)mx6_clocks_init函數(shù)將數(shù)組lookups內(nèi)的時(shí)鐘,全部添加到了鏈表clocks內(nèi)。我們再通過第一步的方法獲取lookups內(nèi)定義的時(shí)鐘。
相關(guān)產(chǎn)品
>
-
FETMX6Q-C核心板
i.MX6Q核心板板層出不窮,要如何選擇?飛凌解讀i.mx6Q芯片強(qiáng)性能為您推薦四核A9架構(gòu)的i.MX6Q產(chǎn)品精選,包含iMX6Q 核心板、i.MX6Q 核心板、iMX6Q工業(yè)級核心板,歡迎采購。 i.MX6Q核心板基于NXP(原Freescale)Cortex-A9架構(gòu)的i.MX6Q四核處理器設(shè)計(jì),核心板小尺寸核心板搭配獨(dú)特的薄款連接器,讓設(shè)計(jì)隨心所欲!
了解詳情
-
OKMX6Q-C開發(fā)板
雙千兆飛凌嵌入式iMX6Q開發(fā)板,板對板連接器,纖薄之際,次底板支持iMX6Q和iMX6DL核心板。i.MX6Q開發(fā)板與i.MX6DL開發(fā)板資源豐富,原理圖、PCB、軟件資源、硬件資源下載,技術(shù)支持等。歡迎選購
了解詳情
-
FETMX6Q-S核心板
NXP iMX6Quad系列具有四個(gè)內(nèi)核,運(yùn)行頻率達(dá)1.2 GHz,帶有1 MB L2緩存和64位DDR3或2通道、32位LPDDR2支持。飛凌提供商業(yè)級iMX6Q核心板,工業(yè)級iMX6Q核心板,兼容一同底板。具有抗震,抗氧化,抗干擾,更快速升級產(chǎn)品等優(yōu)勢。保定飛凌嵌入式專注imx6,imx6開發(fā)板,飛思卡爾imx6等ARM嵌入式核心控制系統(tǒng)研發(fā)、設(shè)計(jì)和生產(chǎn),是imx6,imx6開發(fā)板,飛思卡爾imx6提供者,imx6系列產(chǎn)品現(xiàn)已暢銷全國,歡迎咨詢!
了解詳情
-
OKMX6Q-S3開發(fā)板
飛凌嵌入式提供iMX6Q開發(fā)板,iMX6解決方案,iMX6Q核心板,i.MX6Q開發(fā)板解決方案。iMX6Q穩(wěn)定、快速、性價(jià)比高,歡迎選購 NXP iMX6系列芯片全支持,升級簡配無憂替換。
了解詳情
推薦閱讀
換一批
我們在編寫或修改驅(qū)動(dòng)時(shí),經(jīng)常會(huì)遇到時(shí)鐘相關(guān)的問題,不知道從什么地方下手。在本文中,以i.MX6的3.0.35版本的內(nèi)核舉例時(shí)鐘如何獲取和使用。
我們常見的獲取時(shí)鐘的方法
? 通過名稱獲取
例如:獲取時(shí)鐘clko
clko = clk_get(NULL, "clko_clk");
? 通過設(shè)備獲取
例如在音頻接口ssi的驅(qū)動(dòng)中獲時(shí)鐘,
ssi->clk = clk_get(&pdev->dev, NULL); //設(shè)備的名字是 "imx-ssi"
我們可以通過查看函數(shù)clk_get的原型來進(jìn)一步查看始終是怎么獲取的
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);
}
我們繼續(xù)查看函數(shù)clk_get_sys,此函數(shù)通過設(shè)備的名字或時(shí)鐘的名字來獲取時(shí)鐘
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);
}
此函數(shù)調(diào)用clk_find獲取時(shí)鐘,在時(shí)鐘列表內(nèi)按照設(shè)備名和時(shí)鐘名查找,并返回獲取到的時(shí)鐘。
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;
} // 先找設(shè)備名
if (p->con_id) {//再找時(shí)鐘名
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;}
時(shí)鐘的注冊
剛才提到了從時(shí)鐘列表內(nèi)按照設(shè)備或時(shí)鐘名稱獲取時(shí)鐘,時(shí)鐘列表是怎么生成的。
我們在i.Mx6板級文件初始化里調(diào)用了時(shí)鐘初始化函數(shù)。
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函數(shù)
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);
}
….
}
先看函數(shù)內(nèi)lookups數(shù)組內(nèi)定了許多時(shí)鐘
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, \
}
再看函數(shù)clkdev_add
void clkdev_add(struct clk_lookup *cl){
mutex_lock(&clocks_mutex);
list_add_tail(&cl->node, &clocks);
mutex_unlock(&clocks_mutex);
}
由此我們可以發(fā)現(xiàn)mx6_clocks_init函數(shù)將數(shù)組lookups內(nèi)的時(shí)鐘,全部添加到了鏈表clocks內(nèi)。我們再通過第一步的方法獲取lookups內(nèi)定義的時(shí)鐘。
相關(guān)產(chǎn)品 >
-
FETMX6Q-C核心板
i.MX6Q核心板板層出不窮,要如何選擇?飛凌解讀i.mx6Q芯片強(qiáng)性能為您推薦四核A9架構(gòu)的i.MX6Q產(chǎn)品精選,包含iMX6Q 核心板、i.MX6Q 核心板、iMX6Q工業(yè)級核心板,歡迎采購。 i.MX6Q核心板基于NXP(原Freescale)Cortex-A9架構(gòu)的i.MX6Q四核處理器設(shè)計(jì),核心板小尺寸核心板搭配獨(dú)特的薄款連接器,讓設(shè)計(jì)隨心所欲!
了解詳情 -
OKMX6Q-C開發(fā)板
雙千兆飛凌嵌入式iMX6Q開發(fā)板,板對板連接器,纖薄之際,次底板支持iMX6Q和iMX6DL核心板。i.MX6Q開發(fā)板與i.MX6DL開發(fā)板資源豐富,原理圖、PCB、軟件資源、硬件資源下載,技術(shù)支持等。歡迎選購
了解詳情 -
FETMX6Q-S核心板
NXP iMX6Quad系列具有四個(gè)內(nèi)核,運(yùn)行頻率達(dá)1.2 GHz,帶有1 MB L2緩存和64位DDR3或2通道、32位LPDDR2支持。飛凌提供商業(yè)級iMX6Q核心板,工業(yè)級iMX6Q核心板,兼容一同底板。具有抗震,抗氧化,抗干擾,更快速升級產(chǎn)品等優(yōu)勢。保定飛凌嵌入式專注imx6,imx6開發(fā)板,飛思卡爾imx6等ARM嵌入式核心控制系統(tǒng)研發(fā)、設(shè)計(jì)和生產(chǎn),是imx6,imx6開發(fā)板,飛思卡爾imx6提供者,imx6系列產(chǎn)品現(xiàn)已暢銷全國,歡迎咨詢!
了解詳情 -
OKMX6Q-S3開發(fā)板
飛凌嵌入式提供iMX6Q開發(fā)板,iMX6解決方案,iMX6Q核心板,i.MX6Q開發(fā)板解決方案。iMX6Q穩(wěn)定、快速、性價(jià)比高,歡迎選購 NXP iMX6系列芯片全支持,升級簡配無憂替換。 了解詳情