調用完函數i2c_init後(見 I2C子系統之I2C bus初始化——I2C_init() http://www.linuxidc.com/Linux/2012-08/68261.htm) 系統將成功創建i2c總線。初始化完畢總線後還需要接著初始化i2c設備和i2c驅動(一般是先初始化device),對於2440來說,linux內核中的device初始化一般是通過platform device來初始化的,platform device的初始化在/arch/arm/mach-s3c2440/mach-smdk2440.c文件中的smdk2440_machine_init()函數中完成。
函數。如下:
- static void __init smdk2440_machine_init(void)
- {
- ... ...
- s3c_i2c0_set_platdata(NULL);
-
- i2c_register_board_info(0, tq2440_i2c_devs,
- ARRAY_SIZE(tq2440_i2c_devs));
- platform_add_devices(smdk2440_devices, ARRAY_SIZE(smdk2440_devices));
- ... ...
- }
smdk2440_machine_init中和i2c有關的函數就三個。需要注意的是,此處的i2c_register_board_info()函數
未後來添加的,原生內核此處並未調用此函數。博主參考mini2440,確定需要調用此函數,後面分析。
談後首先描述下上訴三個函數的基本功能,再進行源碼分析。
1.s3c_i2c_set_platdata。構建platform_device型設備s3c_device_i2c0。
2.i2c_register_board_info。將板上相關的i2c設備信息統一注冊到i2c設
備鏈__i2c_board_list上。此處就為at24c02的相關參數。
3.platform_add_device。將之前創建的platform_device型設備s3c_device_i2c0
添加到platform_bus總線。
下面依次對上述三個函數進行源碼分析。
1.s3c_i2c0_set_platdata
3c_i2c0_set_platdata函數如下:
- void __init s3c_i2c0_set_platdata(struct s3c2410_platform_i2c *pd)
- {
- struct s3c2410_platform_i2c *npd;
-
- if (!pd)
- pd = &default_i2c_data0;
-
- npd = kmemdup(pd, sizeof(struct s3c2410_platform_i2c), GFP_KERNEL);
- if (!npd)
- printk(KERN_ERR "%s: no memory for platform data\n", __func__);
- else if (!npd->cfg_gpio)
- npd->cfg_gpio = s3c_i2c0_cfg_gpio;
-
- s3c_device_i2c0.dev.platform_data = npd;
- }
該函數首先判斷pd為真
- if (!pd)
- pd = &default_i2c_data0;
然後將default_i2c_data0賦值給pd
- static struct s3c2410_platform_i2c default_i2c_data0 __initdata = {
- .flags = 0,
- .slave_addr = 0x50,
- .frequency = 100*1000,
- .sda_delay = 100,
- };
然後通過
- npd = kmemdup(pd, sizeof(struct s3c2410_platform_i2c), GFP_KERNEL);
-
- 。。。 。。。
-
- s3c_device_i2c0.dev.platform_data = npd;
實現將default_i2c_data0的值保存到s3c_device_i2c0的dev.platform_data下
- struct platform_device s3c_device_i2c0 = {
- .name = "s3c2410-i2c",
- #ifdef CONFIG_S3C_DEV_I2C1
- .id = 0,
- #else
- .id = -1,
- #endif
- .num_resources = ARRAY_SIZE(s3c_i2c_resource),
- .resource = s3c_i2c_resource,
- };
此時s3c_device_i2c0.dev.platform_data中就保存slave_addr、frequency等信息。
在s3c24xx_i2c_probe函數中,對3c2440的i2c相關控制寄存器的初始化操作會將
這些參數填入到相關寄存器。此處值得一提的是name變量,name變量的初始賦值是s3c2410-i2c
但是會在後面的/arch/arm/mach-s3c2440/s3c24xx.c文件中的函數s3c244x_map_io()中修改成s3c2440-i2c。
這個函數的作用就是用來修改和s3c2410不同的外設用的。