驱动开发硬核特训 · Day 31:理解 I2C 子系统的驱动模型与实例剖析

news2025/5/24 14:44:37

📚 训练目标

  • 从驱动模型出发,掌握 I2C 子系统的核心结构;
  • 分析控制器与从设备的注册流程;
  • 结合 AT24 EEPROM 驱动源码与设备树实例,理解 i2c_client 与 i2c_driver 的交互;
  • 配套高质量练习题巩固理解。

一、I2C 子系统的本质:一个总线驱动子系统

在 Linux 内核中,I2C 是一个标准的总线型子系统(bus_type),不是 platform 设备模型。

I2C 子系统结构如下:

组件结构体描述
I2C 总线bus_type i2c_bus_type内核为 I2C 提供的驱动模型
控制器(主机)i2c_adapter所在 SoC 上的控制器,由 platform_driver 注册
从设备i2c_client掛载在总线上的外设设备(如 EEPROM、Codec)
驱动程序i2c_driver针对某类设备的驱动程序

控制器驱动:使用 platform_driver 绑定 SoC 资源,注册 i2c_adapter。

设备驱动:使用 i2c_driver 注册,等待与 i2c_client 匹配。
在这里插入图片描述


二、源码分析:控制器是如何注册进来的?

以 NXP i.MX8MP 的 I2C 控制器为例,驱动路径:drivers/i2c/busses/i2c-imx.c

static int i2c_imx_probe(struct platform_device *pdev)
{
    struct i2c_adapter *adapter;
    struct imx_i2c_struct *i2c_imx;

    // 1. 分配结构体
    i2c_imx = devm_kzalloc(&pdev->dev, sizeof(*i2c_imx), GFP_KERNEL);

    // 2. 获取寄存器资源、时钟、中断
    i2c_imx->base = devm_ioremap_resource(&pdev->dev, res);
    i2c_imx->clk = devm_clk_get(&pdev->dev, NULL);
    i2c_imx->irq = platform_get_irq(pdev, 0);

    // 3. 初始化 i2c_adapter 结构体
    adapter = &i2c_imx->adapter;
    adapter->owner = THIS_MODULE;
    adapter->algo  = &i2c_imx_algo;
    adapter->dev.parent = &pdev->dev;
    adapter->nr = pdev->id;

    strlcpy(adapter->name, "i.MX I2C Adapter", sizeof(adapter->name));

    // 4. 注册适配器到内核
    return i2c_add_numbered_adapter(adapter);
}

该驱动注册的是 platform_driver,但其核心工作是初始化并注册 i2c_adapter


三、设备是如何通过设备树挂载上来的?

设备树示例:

&i2c1 {
    status = "okay";

    eeprom@50 {
        compatible = "atmel,24c02";
        reg = <0x50>;
        pagesize = <16>;
    };
};

设备树加载过程:

  1. 控制器节点 i2c1 被 platform_driver 匹配;
  2. probe 中调用 i2c_add_adapter(),注册一条总线;
  3. 子节点 eeprom@50of_i2c_register_devices() 扫描;
  4. 创建 i2c_client 对象(地址为 0x50);
  5. 内核查找合适的 i2c_driver 进行绑定。

四、i2c_driver 是如何匹配和驱动设备的?

以 AT24 EEPROM 驱动为例,路径为 drivers/misc/eeprom/at24.c

1. 定义驱动匹配表和结构体

static const struct i2c_device_id at24_ids[] = {
    { "24c02", 0 }, { /* more */ }, {}
};

static const struct of_device_id at24_of_match[] = {
    { .compatible = "atmel,24c02" },
    { /* more */ }, {}
};

static struct i2c_driver at24_driver = {
    .driver = {
        .name = "at24",
        .of_match_table = at24_of_match,
    },
    .probe = at24_probe,
    .remove = at24_remove,
    .id_table = at24_ids,
};

module_i2c_driver(at24_driver);

2. 驱动入口:probe 函数

static int at24_probe(struct i2c_client *client,
                      const struct i2c_device_id *id)
{
    // 打印设备信息
    dev_info(&client->dev, "AT24 EEPROM detected at address 0x%x\n", client->addr);

    // 挂载到 MTD、NVMEM 或字符设备(不同内核版本机制不同)
    return devm_nvmem_register(...);
}

驱动完成初始化后,用户可通过 sysfsi2c-tools 访问 EEPROM 内容。


五、如何访问设备?—— 使用 i2c_transfer

int i2c_transfer(struct i2c_adapter *adap,
                 struct i2c_msg *msgs, int num);

示例:读 EEPROM

u8 addr = 0x00;
u8 buf[16];

struct i2c_msg msgs[2] = {
    {
        .addr = client->addr,
        .flags = 0,
        .buf = &addr,
        .len = 1,
    },
    {
        .addr = client->addr,
        .flags = I2C_M_RD,
        .buf = buf,
        .len = 16,
    }
};

i2c_transfer(client->adapter, msgs, 2);

六、练习题目(含答案)

💡 题目 1:以下哪个结构体代表 I2C 总线控制器?

A. struct i2c_driver
B. struct i2c_client
C. struct i2c_adapter
D. struct i2c_msg

✅ 答案:C

i2c_adapter 表示控制器,负责发起读写。


💡 题目 2:哪一个函数用于驱动注册设备驱动(如 AT24)?

A. platform_driver_register()
B. i2c_add_adapter()
C. i2c_register_driver()
D. of_register_platform_driver()

✅ 答案:C

i2c_register_driver() 会将 i2c_driver 注册到 i2c_bus_type 下,由内核匹配。


💡 题目 3:设备树中 reg = <0x50>; 表示的含义是什么?

A. 设备在系统内的 IRQ 编号
B. I2C 控制器的总线编号
C. 设备的物理地址
D. 设备在 I2C 总线上的从地址

✅ 答案:D

在 I2C 总线中,reg 对应从设备地址(如 0x50 即为 EEPROM 地址)。


七、实战建议与延伸学习

推荐进一步学习内容:

  • 使用 i2c-tools 进行测试:i2cdetect, i2cget, i2cset
  • 理解 regmap + i2c 的结合方式(如 codec 驱动)
  • 在 I2C 多路复用器(i2c-mux)场景下的子设备建模

八、结语

Linux 的 I2C 子系统是一个典型的“总线-设备-驱动”三段式模型。
控制器使用 platform_driver 驱动,而子设备由 i2c_driver 管理,匹配过程由 I2C 核心完成。

通过本文,我们不仅从结构上掌握了子系统的组织方式,也通过实际代码洞察了其内部运行机制,并借助练习巩固了要点。


视频教程请关注 B 站:“嵌入式 Jerry”

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2384672.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【python】局域网内通过python远程重启另一台windows电脑

&#x1f449;技__能&#x1f448;&#xff1a;C/C/C#/Python/Java/PHP/Vue/Node.js/HTML语言 &#x1f449;专__注&#x1f448;&#xff1a;专注主流机器人、人工智能等相关领域的开发、测试技术。 局域网内通过python远程重启另一台windows电脑 目录 局域网内通过python远程…

超越感官的实相:声、光、气味的科学与哲学探微

在人类的感官世界中&#xff0c;声、光、气味是日常生活中最直接的现象&#xff1a;我们聆听音乐、观赏光影、呼吸花香。然而&#xff0c;若深入探究它们的本质&#xff0c;科学与哲学竟以截然不同的视角&#xff0c;揭示了一个超越感官的实相世界。本文将从经典物理学、佛教哲…

什么是VR场景?VR与3D漫游到底有什么区别

在数字化时代&#xff0c;虚拟现实&#xff08;Virtual Reality, 简称VR&#xff09;场景与3D漫游作为两种前沿技术&#xff0c;改变着人们的生活方式和体验模式。通过计算机模拟真实或假想的场景&#xff0c;让用户仿佛身临其境&#xff0c;并能与虚拟环境进行互动。尽管VR场景…

python学习day2:进制+码制+逻辑运算符

进制 Python 中的进制表示与转换 进制的基本概念 二进制、八进制、十进制、十六进制的定义与特点不同进制在计算机科学中的应用场景 Python 中的进制表示 二进制表示&#xff1a;使用 0b 前缀八进制表示&#xff1a;使用 0o 前缀十六进制表示&#xff1a;使用 0x 前缀示例…

【分布式文件系统】FastDFS

1.简介 讲这个之前&#xff0c;相信很多人特别是学java的&#xff0c;肯定在做苍穹外卖的时候肯定接触过一个东西&#xff0c;叫做阿里云OSS&#xff0c;他们的功能都差不多&#xff0c;但是阿里云的这个是要付费的&#xff0c;而FastDFS是免费开源的&#xff0c;是由淘宝资深…

word为章节标题添加自动编号

问题&#xff1a; 如何为word文档中的多级标题添加自动编号&#xff1f; 方法&#xff1a; 1、首先为文档各级标题设置格式样式&#xff0c;一级标题使用样式中的“标题 1”&#xff0c;二级标题使用“标题 2”&#xff0c;三级使用“标题 3”&#xff0c;也就是直接在开始—…

无人机飞行间隔安全智能评估、安全风险评估

无人机空中安全飞行评估需结合改进碰撞模型、蒙特卡洛仿真、安全间隔反推及动态避障策略&#xff0c;通过多机型分类与实时数据融合&#xff0c;实现从理论建模到实际部署的全流程管控&#xff0c;为城市低空密集飞行提供安全保障。 需求 无人机飞行间隔安全智能评估 无人机…

【VLNs篇】03:VLMnav-端到端导航与视觉语言模型:将空间推理转化为问答

栏目内容论文标题End-to-End Navigation with Vision-Language Models: Transforming Spatial Reasoning into Question-Answering (端到端导航与视觉语言模型&#xff1a;将空间推理转化为问答)核心问题如何利用大型视觉语言模型&#xff08;VLM&#xff09;实现端到端的机器人…

PCB设计实践(二十五)贴片电阻与插件电阻的全面解析:差异、演进与应用场景

一、基础结构与技术原理差异 物理结构差异 贴片电阻&#xff08;SMD Resistor&#xff09;采用表面贴装技术&#xff08;SMT&#xff09;&#xff0c;其主体为扁平长方体或圆柱形结构&#xff0c;两端金属化电极直接与PCB焊盘接触。典型封装尺寸包括0402&#xff08;1.00.5mm&a…

Canvas设计图片编辑器全讲解(一)Canvas基础(万字图文讲解)

一、前序 近两年AI发展太过迅速&#xff0c;各类AI产品层出不穷&#xff0c;AI绘图/AI工作流/AI视频等平台的蓬勃发展&#xff0c;促使图片/视频等复杂内容的创作更加简单&#xff0c;让更多普通人有了图片和视频创作的机会。另一方面用户内容消费也逐渐向图片和视频倾斜。在“…

利用Qt绘图随机生成带多种干扰信息的数字图片

背景 在学习AutoML或ML的过程中&#xff0c;需要一些图片类型的数据做分类预测训练&#xff0c;于是想到尝试最简单的数字识别&#xff0c;且单个数字的识别&#xff0c;也就是y的取值只有10种可能&#xff0c;即0到9。 以下参考代码分别考虑了数字字体的大小、数字颜色的深浅…

STM32——从点灯到传感器控制

STM32基础外设开发&#xff1a;从点灯到传感器控制 一、前言 本篇文章总结STM32F10x系列基础外设开发实例&#xff0c;涵盖GPIO控制、按键检测、传感器应用等。所有代码基于标准库开发&#xff0c;适合STM32初学者参考。 二、硬件准备 STM32F10x系列开发板LED模块有源蜂鸣器…

java day14

接昨天&#xff0c;响应 响应 就是我们在处理请求的时候&#xff0c;里面的return 其实方法里面写的return的返回平常的什么字符串啊什么等等&#xff1b;这些东西都是直接返回&#xff1b;如果是一个对象的话&#xff0c;我们会按json的格式返回&#xff1b; 这些都依赖于一…

Tailwind css实战,基于Kooboo构建AI对话框页面(一)

在当今数字化时代&#xff0c;AI 助手已成为网站和应用不可或缺的一部分。本文将带你一步步使用 Tailwind CSS 和 Kooboo 构建一个现代化的 AI 对话界面框。 一、选择 Kooboo平台 的核心优势 智能提示&#xff1a;在输入 class 属性时&#xff0c;会自动触发 Tailwind CSS 规则…

重塑数学边界:人工智能如何引领数学研究的新纪元

目录 一、人工智能如何重新定义数学研究的边界 &#xff08;一&#xff09;数学与AI的关系&#xff1a;从基础理论到创新思维的回馈 &#xff08;二&#xff09;AI的创造力&#xff1a;突破传统推理的局限 &#xff08;三&#xff09;AI对数学研究的潜在贡献&#xff1a;创…

docker部署并测试翻译模型-CSANMT连续语义增强机器翻译

1.模型选择CSANMT-Translation模型&#xff1a; 2.修改docker-compose.yml文件&#xff0c;重新定义模型缓存路径和存储路径 其中MODELSCOPE_CACHE指定了模型的下载路径。 3.运行docker compose up -d --build&#xff0c;提示出现报错&#xff1a;Error response from daemon…

Spring Boot项目配置核心 - pom.xml的依赖管理与构建优化

基础架构 <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVe…

告别手抖困扰:全方位健康护理指南

手抖&#xff0c;医学上称为震颤&#xff0c;是常见的身体症状&#xff0c;可能由多种原因引发&#xff0c;了解其成因并采取科学护理措施&#xff0c;对改善症状、维护健康至关重要。 生理性手抖往往因情绪激动、过度劳累、大量饮用咖啡或酒精等引起&#xff0c;这种手抖通常较…

图解深度学习 - 特征工程(DL和ML的核心差异)

前言 深度学习通过自动化特征提取&#xff0c;简化了机器学习工作流程&#xff0c;它让解决问题变得更加简单。因为深度学习将特征工程完全自动化&#xff0c;而特征工程曾经是机器学习工作流程中最关键的一步。 一、机器学习特征工程 机器学习为什么需要特征工程&#xff08…

Datacom-hcia~Datacom-hcie学习笔记索引

hcia VLAN工作原理实验案例(超详细)https://blog.csdn.net/Fanmeang/article/details/145855768?spm1001.2014.3001.5502 交换机工作原理实验案例https://blog.csdn.net/Fanmeang/article/details/145802382?spm1001.2014.3001.5502 ARP理论实验案例&#xff08;超详细&am…