概述
设备树节点
i2c0: i2c@01c2ac00 {
compatible = "allwinner,sun6i-a31-i2c";
reg = <0x01c2ac00 0x400>;
interrupts = <GIC_SPI 6 IRQ_TYPE_LEVEL_HIGH>;
clocks = <&ccu CLK_BUS_I2C0>;
resets = <&ccu RST_BUS_I2C0>;
pinctrl-names = "default";
pinctrl-0 = <&i2c0_pins>;
status = "disabled";
#address-cells = <1>;
#size-cells = <0>;
};
控制器驱动源码路径
linux-5.1.0driversi2cbussesi2c-mv64xxx.c
控制器结构体与注册到核心层
linux-5.1.0includelinuxi2c.h
struct i2c_adapter adapter;
linux-5.1.0driversi2ci2c-core-base.c
int i2c_add_numbered_adapter(struct i2c_adapter *adap);
控制器设备树节点包括外设子节点,记录到adapter结构体
drv_data->adapter.dev.of_node = pd->dev.of_node;
iic控制器驱动platform的probe函数调用流程
mv64xxx_i2c_probe
i2c_add_numbered_adapter /* 添加I2C控制器 */
__i2c_add_numbered_adapter
i2c_register_adapter /* 注册I2C控制器 */
device_register /* I2C控制器设备注册 */
of_i2c_register_devices
/* 从设备树中把与该adapter匹配的iic控制器设备树节点找出来 */
bus = of_get_child_by_name(adap->dev.of_node, "i2c-bus");
/*解析设备树iic控制器节点,根据设备树的iic外设节点创建client */
client = of_i2c_register_device(adap, node);
i2c_new_device
// client是根据当前注册的adapter创建的,所以直接绑定adapter,以后不用再为client找adapter。
client->adapter = adap;
// i2c外设client都绑定该i2c总线,不管这个client属于哪个adapter,client与adapter的关系已在上一步表述清楚,这一步是外设与外设驱动的总线匹配关系。
// 不过该client属于哪个adapter,都是通过同一条i2c_bus_type的i2c_device_match函数匹配,换言之,外设驱动适合所有adapter的外设client,只要i2c_bus_type的i2c_device_match匹配成功。
client->dev.bus = &i2c_bus_type;
device_register /* 添加设备I2C从设备 */
i2c_scan_static_board_info /* 查找静态表,有些I2C设备是在代码中写死的,不是通过设备树的形式 */
i2c_new_device
client->dev.bus = &i2c_bus_type;
注册前为什么需要申请中断?
drv_data->irq = platform_get_irq(pd, 0);
rc = request_irq(drv_data->irq, mv64xxx_i2c_intr, 0, MV64XXX_I2C_CTLR_NAME, drv_data);
看设备树中断描述
interrupts = <GIC_SPI 6 IRQ_TYPE_LEVEL_HIGH>;
SPI:shared processor interrupts 的第6号中断,参考: 设备树中的interrupts属性解析
最后
以上就是大意发夹为你收集整理的全志V3s IIC控制器驱动分析的全部内容,希望文章能够帮你解决全志V3s IIC控制器驱动分析所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复