涉及到的文件:
drivers/i2c/i2c-core.c
drivers/i2c/i2c-dev.c
drivers/i2c/busses/i2c-imx.c
等等
在下面分析的代码中,不想关或者不重要的,我会省略掉。
1. 适配器设备的注册
在Linux内核启动的过程中,会调用到mx6_sabresd_board_init函数
static void __init mx6_sabresd_board_init(void) { 。。。省略。。。 /*在这里修改了i2c0适配器上设备的类型,同时添加了平台数据,后面会分析到*/ strcpy(mxc_i2c0_board_info[0].type, "wm8962"); mxc_i2c0_board_info[0].platform_data = &wm8962_config_data; /*第一步:注册适配器设备:详见下面分析*/ imx6q_add_imx_i2c(0, &mx6q_sabresd_i2c_data); /*第二步:添加I2C从设备*/ i2c_register_board_info(0, mxc_i2c0_board_info, ARRAY_SIZE(mxc_i2c0_board_info)); 。。。省略。。。 }
下面我们来分析第一步:注册适配器设备
extern const struct imx_imx_i2c_data imx6q_imx_i2c_data[] __initconst; #define imx6q_add_imx_i2c(id, pdata) imx_add_imx_i2c(&imx6q_imx_i2c_data[id], pdata)
接着追到imx_add_imx_i2c,我们可以知道
1). 通过imx_add_platform_device ,我们知道I2C的适配器设备是放入了平台总线模型当中,并且名字为imx-i2c。
2). 第一个参数imx_imx_i2c_data是适配器的设备信息,包括I2C寄存器地址,中断号,资源res是通过data的重新赋值。它包括了I2C寄存器的基地址,中断号,总线编号。第二个参数
imxi2c_platform_data是存放在平台上的数据,下面我们来分析这两个参数。 struct platform_device *__init imx_add_imx_i2c( const struct imx_imx_i2c_data *data, const struct imxi2c_platform_data *pdata) { struct resource res[] = { { .start = data->iobase, .end = data->iobase + data->iosize - 1, .flags = IORESOURCE_MEM, }, { .start = data->irq, .end = data->irq, .flags = IORESOURCE_IRQ, }, }; return imx_add_platform_device("imx-i2c", data->id, res, ARRAY_SIZE(res), pdata, sizeof(*pdata)); }
我们先分析第一个参数imx_add_platform_device:
我们往回查看上述函数的关键数据
imx_imx_i2c_data* data= imx6q_imx_i2c_data[0] const struct imx_imx_i2c_data imx6q_imx_i2c_data[] __initconst = { #define imx6q_imx_i2c_data_entry(_id, _hwid) imx_imx_i2c_data_entry(MX6Q, _id, _hwid, SZ_4K) imx6q_imx_i2c_data_entry(0, 1), imx6q_imx_i2c_data_entry(1, 2), imx6q_imx_i2c_data_entry(2, 3), };
接着追imx_imx_i2c_data_entry
#define imx_imx_i2c_data_entry(soc, _id, _hwid, _size)
[_id] = imx_imx_i2c_data_entry_single(soc, _id, _hwid, _size)
再接着追imx_imx_i2c_data_entry
#define imx_imx_i2c_data_entry_single(soc, _id, _hwid, _size) { .id = _id, .iobase = soc ## _I2C ## _hwid ## _BASE_ADDR, .iosize = _size, .irq = soc ## _INT_I2C ## _hwid, }
通过一层层的代入展开,可以推导出:
imx6q_imx_i2c_data[0]= { .id = 0, .iobase = MX6Q_I2C1_BASE_ADDR, .iosize = _SZ_4K, .irq = MX6Q_INT_I2C1, }
第二个参数mx6q_sabresd_i2c_data是时钟系数:
static struct imxi2c_platform_data mx6q_sabresd_i2c_data = { .bitrate = 100000, };
2. 适配器驱动的流程
从之前的代码,我们得知适配器设备在平台驱动模型中的名字为imx-i2c。
其代码在drivers/i2c/busses/i2c-imx.c
static struct platform_driver i2c_imx_driver = { .remove = __exit_p(i2c_imx_remove), .driver = { .name = DRIVER_NAME, /*imx-i2c*/ .owner = THIS_MODULE, } }; /*通过平台模型注册*/ static int __init i2c_adap_imx_init(void) { return platform_driver_probe(&i2c_imx_driver, i2c_imx_probe); }
当名字imx-i2c匹配上的时候,会自动调用探测函数i2c_imx_probe,其原理属于平台驱动模型。在这里面会对I2C进行初始化:设置始终,初始化I2C寄存器,设置中断等。
static int __init i2c_imx_probe(struct platform_device *pdev) { struct imx_i2c_struct *i2c_imx; struct resource *res; struct imxi2c_platform_data *pdata; void __iomem *base; resource_size_t res_size; int irq; int ret; dev_dbg(&pdev->dev, "<%s> ", __func__); /*获取来自适配器设备上的内存资源:I2C寄存器地址*/ res = platform_get_resource(pdev, IORESOURCE_MEM, 0); if (!res) { dev_err(&pdev->dev, "can't get device resources "); return -ENOENT; } /*获取来自适配器设备上的中断号*/ irq = platform_get_irq(pdev, 0); if (irq < 0) { dev_err(&pdev->dev, "can't get irq number "); return -ENOENT; } /*获取平台数据:也就是之前分析的时钟系数*/ pdata = pdev->dev.platform_data; if (pdata && pdata->init) { ret = pdata->init(&pdev->dev); if (ret) return ret; } res_size = resource_size(res); /*将寄存器地址映射为虚拟内存*/ if (!request_mem_region(res->start, res_size, DRIVER_NAME)) { ret = -EBUSY; goto fail0; } base = ioremap(res->start, res_size); if (!base) { dev_err(&pdev->dev, "ioremap failed "); ret = -EIO; goto fail1; } i2c_imx = kzalloc(sizeof(struct imx_i2c_struct), GFP_KERNEL); if (!i2c_imx) { dev_err(&pdev->dev, "can't allocate interface "); ret = -ENOMEM; goto fail2; } /* Setup i2c_imx driver structure */ strcpy(i2c_imx->adapter.name, pdev->name); i2c_imx->adapter.owner = THIS_MODULE; i2c_imx->adapter.algo = &i2c_imx_algo; /*算法层:这个是跟具体硬件和协议相关的,下面分析*/ i2c_imx->adapter.dev.parent = &pdev->dev; i2c_imx->adapter.nr = pdev->id; /*名字*/ i2c_imx->irq = irq; i2c_imx->base = base; i2c_imx->res = res; /* 设置I2C时钟 */ i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); if (IS_ERR(i2c_imx->clk)) { ret = PTR_ERR(i2c_imx->clk); dev_err(&pdev->dev, "can't get I2C clock "); goto fail3; } /* 设置中断 */ ret = request_irq(i2c_imx->irq, i2c_imx_isr, 0, pdev->name, i2c_imx); if (ret) { dev_err(&pdev->dev, "can't claim irq %d ", i2c_imx->irq); goto fail4; } /* 等待队列 */ init_waitqueue_head(&i2c_imx->queue); /* 设置适配器数据 */ i2c_set_adapdata(&i2c_imx->adapter, i2c_imx); /* 设置时钟分频 */ if (pdata && pdata->bitrate) i2c_imx_set_clk(i2c_imx, pdata->bitrate); else i2c_imx_set_clk(i2c_imx, IMX_I2C_BIT_RATE); /* 设置I2C的寄存器 */ writeb(0, i2c_imx->base + IMX_I2C_I2CR); writeb(0, i2c_imx->base + IMX_I2C_I2SR); /* 注册适配器驱动:这个我们后面会分析 */ ret = i2c_add_numbered_adapter(&i2c_imx->adapter); if (ret < 0) { dev_err(&pdev->dev, "registration failed "); goto fail5; } /* 将i2c_imx数据放入平台总线中 */ platform_set_drvdata(pdev, i2c_imx); dev_dbg(&i2c_imx->adapter.dev, "claimed irq %d ", i2c_imx->irq); dev_dbg(&i2c_imx->adapter.dev, "device resources from 0x%x to 0x%x ", i2c_imx->res->start, i2c_imx->res->end); dev_dbg(&i2c_imx->adapter.dev, "allocated %d bytes at 0x%x ", res_size, i2c_imx->res->start); dev_dbg(&i2c_imx->adapter.dev, "adapter name: "%s" ", i2c_imx->adapter.name); dev_dbg(&i2c_imx->adapter.dev, "IMX I2C adapter registered "); return 0; /* Return OK */ fail5: free_irq(i2c_imx->irq, i2c_imx); fail4: clk_put(i2c_imx->clk); fail3: kfree(i2c_imx); fail2: iounmap(base); fail1: release_mem_region(res->start, resource_size(res)); fail0: if (pdata && pdata->exit) pdata->exit(&pdev->dev); return ret; /* Return error number */ }
/*这个函数的主要功能是提供该适配器能提供的I2C功能,提供判断*/
static u32 i2c_imx_func(struct i2c_adapter *adapter) { return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL; } static struct i2c_algorithm i2c_imx_algo = { .master_xfer = i2c_imx_xfer, /*具体的传输的实现*/ .functionality = i2c_imx_func, /*传输的方式*/ }; static int i2c_imx_xfer(struct i2c_adapter *adapter, struct i2c_msg *msgs, int num) { unsigned int i, temp; int result; /*将该适配器上的数据取出来*/ struct imx_i2c_struct *i2c_imx = i2c_get_adapdata(adapter); dev_dbg(&i2c_imx->adapter.dev, "<%s> ", __func__); /* I2C起始位 */ result = i2c_imx_start(i2c_imx); if (result) goto fail0; /* 对数据进行读写 */ for (i = 0; i < num; i++) { if (i) { dev_dbg(&i2c_imx->adapter.dev, "<%s> repeated start ", __func__); temp = readb(i2c_imx->base + IMX_I2C_I2CR); temp |= I2CR_RSTA; writeb(temp, i2c_imx->base + IMX_I2C_I2CR); result = i2c_imx_bus_busy(i2c_imx, 1); if (result) goto fail0; } dev_dbg(&i2c_imx->adapter.dev, "<%s> transfer message: %d ", __func__, i); /* write/read data */ #ifdef CONFIG_I2C_DEBUG_BUS temp = readb(i2c_imx->base + IMX_I2C_I2CR); dev_dbg(&i2c_imx->adapter.dev, "<%s> CONTROL: IEN=%d, IIEN=%d, " "MSTA=%d, MTX=%d, TXAK=%d, RSTA=%d ", __func__, (temp & I2CR_IEN ? 1 : 0), (temp & I2CR_IIEN ? 1 : 0), (temp & I2CR_MSTA ? 1 : 0), (temp & I2CR_MTX ? 1 : 0), (temp & I2CR_TXAK ? 1 : 0), (temp & I2CR_RSTA ? 1 : 0)); temp = readb(i2c_imx->base + IMX_I2C_I2SR); dev_dbg(&i2c_imx->adapter.dev, "<%s> STATUS: ICF=%d, IAAS=%d, IBB=%d, " "IAL=%d, SRW=%d, IIF=%d, RXAK=%d ", __func__, (temp & I2SR_ICF ? 1 : 0), (temp & I2SR_IAAS ? 1 : 0), (temp & I2SR_IBB ? 1 : 0), (temp & I2SR_IAL ? 1 : 0), (temp & I2SR_SRW ? 1 : 0), (temp & I2SR_IIF ? 1 : 0), (temp & I2SR_RXAK ? 1 : 0)); #endif if (msgs[i].flags & I2C_M_RD) result = i2c_imx_read(i2c_imx, &msgs[i]); else result = i2c_imx_write(i2c_imx, &msgs[i]); if (result) goto fail0; } fail0: /* 停止位 */ i2c_imx_stop(i2c_imx); dev_dbg(&i2c_imx->adapter.dev, "<%s> exit with: %s: %d ", __func__, (result < 0) ? "error" : "success msg", (result < 0) ? result : num); return (result < 0) ? result : num; }
这个函数里面得到所有函数就是最底层真正对硬件的操作了,比如:
static int i2c_imx_start(struct imx_i2c_struct *i2c_imx) { unsigned int temp = 0; struct imxi2c_platform_data *pdata; int result; dev_dbg(&i2c_imx->adapter.dev, "<%s> ", __func__); /* Currently on Arik/Rigel, the I2C clk is from IPG_PERCLK which is * sourced from IPG_CLK. In low bus freq mode, IPG_CLK is at 12MHz * and IPG_PERCLK is down to 4MHz. * Update I2C divider before set i2c clock. */ pdata = i2c_imx->adapter.dev.parent->platform_data; if (pdata && pdata->bitrate) i2c_imx_set_clk(i2c_imx, pdata->bitrate); else i2c_imx_set_clk(i2c_imx, IMX_I2C_BIT_RATE); clk_enable(i2c_imx->clk); writeb(i2c_imx->ifdr, i2c_imx->base + IMX_I2C_IFDR); /* Enable I2C controller */ writeb(0, i2c_imx->base + IMX_I2C_I2SR); writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); /* Wait controller to be stable */ udelay(50); /* Start I2C transaction */ temp = readb(i2c_imx->base + IMX_I2C_I2CR); temp |= I2CR_MSTA; writeb(temp, i2c_imx->base + IMX_I2C_I2CR); result = i2c_imx_bus_busy(i2c_imx, 1); if (result) return result; i2c_imx->stopped = 0; temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK; writeb(temp, i2c_imx->base + IMX_I2C_I2CR); return result; }
在i2c_add_numbered_adapter中会对最后的适配器驱动进行注册,在注册之前还会对I2C设备驱动进行注册。
int i2c_add_numbered_adapter(struct i2c_adapter *adap) { int id; int status; if (adap->nr & ~MAX_ID_MASK) return -EINVAL; retry: if (idr_pre_get(&i2c_adapter_idr, GFP_KERNEL) == 0) return -ENOMEM; mutex_lock(&core_lock); /* "above" here means "above or equal to", sigh; * we need the "equal to" result to force the result */ status = idr_get_new_above(&i2c_adapter_idr, adap, adap->nr, &id); if (status == 0 && id != adap->nr) { status = -EBUSY; idr_remove(&i2c_adapter_idr, id); } mutex_unlock(&core_lock); if (status == -EAGAIN) goto retry; if (status == 0) /*在这里真正注册I2C适配器*/ status = i2c_register_adapter(adap); return status; } static int i2c_register_adapter(struct i2c_adapter *adap) { int res = 0; /* Can't register until after driver model init */ if (unlikely(WARN_ON(!i2c_bus_type.p))) { res = -EAGAIN; goto out_list; } /* Sanity checks */ if (unlikely(adap->name[0] == '