下面开始分析linux/drivers/i
/* s
static int s
{
struct s
struct s
struct resource *res;
int ret;
/*这里pdev->dev.platform_data 在s
pdata = pdev->dev.platform_data;
if (!pdata) {
dev_err(&pdev->dev, "no platform data\n");
return -EINVAL;
}
/*申请一段sizeof(struct s
i
if (!i
dev_err(&pdev->dev, "no memory for state\n");
return -ENOMEM;
}
strlcpy(i
i
i
i
i
i
spin_lock_init(&i
init_waitqueue_head(&i
/* find the clock and enable it */
/*获得I
i
i
if (IS_ERR(i
dev_err(&pdev->dev, "cannot get clock\n");
ret = -ENOENT;
goto err_noclk;
}
dev_dbg(&pdev->dev, "clock source %p\n", i
clk_enable(i
/* map the registers */
/*获取系统的物理地址,中断等资源信息,并进行物理地址到虚拟地址的映射 */
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (res == NULL) {
dev_err(&pdev->dev, "cannot find IO resource\n");
ret = -ENOENT;
goto err_clk;
}
i
pdev->name);
if (i
dev_err(&pdev->dev, "cannot request IO\n");
ret = -ENXIO;
goto err_clk;
}
i
if (i
dev_err(&pdev->dev, "cannot map IO\n");
ret = -ENXIO;
goto err_ioarea;
}
dev_dbg(&pdev->dev, "registers %p (%p, %p)\n",
i
/* setup info block for the i
i
i
/* initialise the i
ret = s
if (ret != 0)
goto err_iomap;
/* find the IRQ for this unit (note, this relies on the init call to
* ensure no current IRQs pending
*/
i
if (ret <= 0) {
dev_err(&pdev->dev, "cannot find IRQ\n");
goto err_iomap;
}
ret = request_irq(i
dev_name(&pdev->dev), i
if (ret != 0) {
dev_err(&pdev->dev, "cannot claim IRQ %d\n", i
goto err_iomap;
}
ret = s
if (ret < 0) {
dev_err(&pdev->dev, "failed to register cpufreq notifier\n");
goto err_irq;
}
/* Note, previous versions of the driver used i
* to add the bus at any number. We now pass the bus number via
* the platform data, so if unset it will now default to always
* being bus 0.
*/
/* 向对应的I
i
ret = i
if (ret < 0) {
dev_err(&pdev->dev, "failed to add bus to i
goto err_cpufreq;
}
platform_set_drvdata(pdev, i
dev_info(&pdev->dev, "%s: S
return 0;
err_cpufreq:
s
err_irq:
free_irq(i
err_iomap:
iounmap(i
err_ioarea:
release_resource(i
kfree(i
err_clk:
clk_disable(i
clk_put(i
err_noclk:
kfree(i
return ret;
}
系统在初始化时会将系统硬件中的时钟注册进系统,用双向循环连接起来,在linux/arch/arm/plat-s
void __init s
{
/* initialise the clocks here, to allow other things like the
* console to use them, and to add new ones after the initialisation
*/
s
s
s
}
其中s
/* s
* Add all the clocks used by the s
* such as the S
* We cannot use a system device as we are needed before any
* of the init-calls that initialise the devices are actually
* done.*/
int __init s
{
unsigned long clkslow = __raw_readl(S
unsigned long clkcon = __raw_readl(S
struct clk *clkp;
struct clk *xtal;
int ret;
int ptr;
clk_upll.enable = s
if (s
printk(KERN_ERR "failed to register usb bus clock\n");
/* register clocks from clock array */
clkp = init_clocks;
for (ptr = 0; ptr < ARRAY_SIZE(init_clocks); ptr++, clkp++) {
/* ensure that we note the clock state */
clkp->usage = clkcon & clkp->ctrlbit ? 1 : 0;
ret = s
if (ret < 0) {
printk(KERN_ERR "Failed to register clock %s (%d)\n",
clkp->name, ret);
}
}
/* We must be careful disabling the clocks we are not intending to
* be using at boot time, as subsystems such as the LCD which do
* their own DMA requests to the bus can cause the system to lockup
* if they where in the middle of requesting bus access.
*
* Disabling the LCD clock if the LCD is active is very dangerous,
* and therefore the bootloader should be careful to not enable
* the LCD clock if it is not needed.
*/
/* install (and disable) the clocks we do not need immediately */
clkp = init_clocks_disable;
for (ptr = 0; ptr < ARRAY_SIZE(init_clocks_disable); ptr++, clkp++) {
ret = s
if (ret < 0) {
printk(KERN_ERR "Failed to register clock %s (%d)\n",
clkp->name, ret);
}
s
}
/* show the clock-slow value */
xtal = clk_get(NULL, "xtal");
printk("CLOCK: Slow mode (%ld.%ld MHz), %s, MPLL %s, UPLL %s\n",
print_mhz(clk_get_rate(xtal) /
( 2 * S
(clkslow & S
(clkslow & S
(clkslow & S
s
return 0;
}
根据注释中给的提示,时钟被分成了两部分,init_clocks和init_clocks_disable,其中init_clocks中的时钟是系统启动时会开启的,而init_clocks_disable中的时钟则在系统启动时会关闭。其中函数s
{
.name = "i
.id = -1,
.parent = &clk_p,
.enable = s
.ctrlbit = S
}
结构中保存了I
s
/* find the clock and enable it */
i
i
if (IS_ERR(i
dev_err(&pdev->dev, "cannot get clock\n");
ret = -ENOENT;
goto err_noclk;
}
dev_dbg(&pdev->dev, "clock source %p\n", i
clk_enable(i
clk_get(&pdev->dev, "i
struct clk *clk_get(struct device *dev, const char *id)
{
struct clk *p;
struct clk *clk = ERR_PTR(-ENOENT);
int idno;
if (dev == NULL || dev->bus != &platform_bus_type)
idno = -1;
else
idno = to_platform_device(dev)->id;
spin_lock(&clocks_lock);
list_for_each_entry(p, &clocks, list) {
if (p->id == idno &&
strcmp(id, p->name) == 0 &&
try_module_get(p->owner)) {
clk = p;
break;
}
}
s
/* s
static int s
{
unsigned long iicon = S
struct s
unsigned int freq;
/* get the plafrom data */
pdata = i
/* inititalise the gpio */
if (pdata->cfg_gpio)
pdata->cfg_gpio(to_platform_device(i
/* write slave address */
/* 写入从设备的地址 */
writeb(pdata->slave_addr, i
dev_info(i
/* 使能接收发送中断和I
writel(iicon, i
/* we need to work out the divisors for the clock... */
/*这里freq用来获取实际的I
if (s
writel(0, i
dev_err(i
return -EINVAL;
}
/* todo - check that the i
dev_info(i
dev_dbg(i
return 0;
}
/* s
*
* work out a divisor for the user requested frequency setting,
* either by the requested frequency, or scanning the acceptable
* range of frequencies until something is found
*/
static int s
{
struct s
/*从系统平台时钟队列中获取pclk的时钟频率,大小为50MHZ */
unsigned long clkin = clk_get_rate(i
unsigned int divs, div1;
unsigned long target_frequency;
u32 iiccon;
int freq;
i
clkin /= 1000; /* clkin now in KHz */
dev_dbg(i
target_frequency = pdata->frequency ? pdata->frequency : 100000;
target_frequency /= 1000; /* Target frequency now in KHz */
/* 目标频率在前面default_i
freq = s
if (freq > target_frequency) {
dev_err(i
"Unable to achieve desired frequency %luKHz." \
" Lowest achievable %dKHz\n", target_frequency, freq);
return -EINVAL;
}
*got = freq; /*通过传入的指针返回实际频率 */
/* 根据时钟选择和分频系数配置对应硬件寄存器 */
iiccon = readl(i
iiccon &= ~(S
iiccon |= (divs-1);
if (div1 == 512)
iiccon |= S
writel(iiccon, i
/* 判断是否为S
if (s
unsigned long sda_delay;
if (pdata->sda_delay) {
sda_delay = (freq / 1000) * pdata->sda_delay;
sda_delay /= 1000000;
sda_delay = DIV_ROUND_UP(sda_delay, 5);
if (sda_delay > 3)
sda_delay = 3;
sda_delay |= S
} else
sda_delay = 0;
dev_dbg(i
writel(sda_delay, i
}
return 0;
}
到这里,I