68.range属性

简单来说,ranges 是一个地址转换表(Address Translation Table)

它的作用是:告诉内核,如何从"CPU 的视角"去访问"子总线(Bus)下面的设备"。

1. 为什么要转换?(核心痛点)

在嵌入式系统中,CPU 并不直接连接所有的外设。外设通常挂载在各种总线(Bus)上。

  • 子设备的世界: 设备自己认为自己在地址 0x0

  • CPU 的世界: CPU 看到的这个设备可能在物理地址 0x40000000

如果没有 ranges,CPU 就不知道该往哪里读写才能操作这个设备。ranges 就像一个桥梁,把子设备的局部地址映射到 CPU 的全局物理地址空间。

2. ranges 的语法格式

ranges 属性通常出现在父节点(通常是表示总线的节点,如 /soc/pcie)中。它的值是一个数字列表,格式如下:

<Child-Bus-Address Parent-Bus-Address Length>

这三个部分构成了"地址映射三元组":

  1. Child-Bus-Address (子总线地址):子设备自己在总线上的起始地址。

  2. Parent-Bus-Address (父总线地址):CPU(或父总线)看到的起始地址。

  3. Length (长度):这段映射区域的大小。

注意 :这三个字段的具体长度(由几个 32-bit 数组成)分别由子节点的 #address-cells、父节点的 #address-cells 和子节点的 #size-cells 决定。

3. 举例说明(由浅入深)

情况 A:带偏移的映射(最标准用法)

假设有一个自定义总线 mybus,挂在它下面的设备从 0x0 开始编址,但实际在 CPU 看来,这段空间在 0x10000000

cpp 复制代码
/ {
    #address-cells = <1>;
    #size-cells = <1>;

    mybus@10000000 {
        compatible = "simple-bus";
        #address-cells = <1>;
        #size-cells = <1>;

        /* 重点在这里 */
        /* 子地址 0x0  映射到  父地址 0x10000000  长度 0x1000 */
        ranges = <0x0 0x10000000 0x1000>;

        /* mybus 下面的设备 */
        gpio@0 {
            compatible = "ns16550";
            reg = <0x0 0x100>;  // 设备说自己在 0x0
        };
    };
};
  • 解析: 当内核驱动想要访问这个 gpio 时,它读到 reg = <0x0 0x100>

  • 转换: 内核查看父节点的 ranges,发现 0x0 对应父地址 0x10000000

  • 结果: CPU 最终访问的物理地址是 0x10000000

情况 B:空 ranges (Identity Mapping,1:1 映射)

这是我们在 ARM SoC 的设备树中最常见 的写法。如果父子地址空间是完全一致的(也就是说,硬件上没有进行特殊的地址重映射),我们可以留空 ranges

cpp 复制代码
ranges;
  • 含义: 子总线的地址空间 = 父总线的地址空间。

  • 例子: 大多数 ARM SoC 内部的 AHB/APB 总线都是直接映射到 CPU 寻址空间的,不需要转换,所以经常看到空的 ranges

情况 C:没有 ranges 属性

如果一个总线节点完全没有 ranges 属性。

  • 含义: 这是一个隔离的总线。CPU 无法直接通过内存读写(MMIO)来访问该总线下的设备。

  • 场景: 这种通常用于像 I2C 控制器这样的节点。I2C 设备挂在控制器下面,但 CPU 不能直接通过物理内存地址去读写 I2C 设备的寄存器,必须通过 I2C 驱动协议来通信。

相关推荐
阿拉斯攀登19 小时前
【RK3576 安卓 JNI/NDK 系列 09】RK3576 实战(三):JNI 调用 librga 实现 2D 硬件加速图像处理
android·驱动开发·rk3568·瑞芯微·rk安卓驱动·rk3576 rga加速
阿拉斯攀登1 天前
第 19 篇 驱动性能优化与功耗优化实战
android·驱动开发·瑞芯微·嵌入式驱动·安卓驱动
道一云黑板报1 天前
技术拆解:AI低代码架构设计与全链路落地实现
人工智能·驱动开发·低代码·ai·企业微信·ai编程·代码规范
LXY_BUAA1 天前
《嵌入式操作系统》_高级字符设备驱动_20260316
linux·运维·服务器·驱动开发
春日见2 天前
Matlab快速入门 基础语法教学
java·开发语言·驱动开发·matlab·docker·计算机外设
阿拉斯攀登2 天前
第 20 篇 RK 平台 NPU / 硬件编解码驱动适配与安卓调用
android·驱动开发·瑞芯微·rk安卓驱动
路溪非溪2 天前
Linux下iw工具的使用总结
linux·网络·arm开发·驱动开发
17(无规则自律)2 天前
【Linux驱动实战】:最简单的内核模块
linux·c语言·驱动开发·嵌入式硬件
阿拉斯攀登2 天前
第 12 篇 RK 平台安卓驱动实战 5:SPI 设备驱动开发,以 SPI 屏 / Flash 为例
android·驱动开发·rk3568·瑞芯微·嵌入式驱动·安卓驱动·spi 设备驱动
路溪非溪2 天前
关于Linux中的日志问题
linux·arm开发·驱动开发