Linux 总线模型与 bind/unbind 完整解析

Linux 总线模型与 bind/unbind 完整解析

1. 为什么 /sys/bus/ 不是"全部总线"

在 Linux 设备模型中,每种总线类型(platform、PCI、I²C、SPI、USB 等)都会在 /sys/bus/ 下注册一个目录。你看到的目录取决于:

  • 内核配置 :如果内核没启用某个总线(比如 CAN 总线),就不会有 /sys/bus/can/
  • 驱动是否启用 :如果驱动没注册成功,对应的 drivers/ 子目录也不会出现。

因此 /sys/bus/ 是动态的,不是固定全集。


2. bind/unbind 的原理

/sys/bus/<bus>/drivers/<driver>/ 下,可能会有 bindunbind 文件:

  • 生成条件
    • 驱动通过标准 API 注册(如 platform_driver_register()i2c_add_driver()pci_register_driver())。
    • 驱动实现了 probe()remove()
    • 内核启用了 CONFIG_SYSFSCONFIG_HOTPLUG
  • 功能
    • bind:手动绑定设备到驱动,触发 probe()
    • unbind:手动解绑设备,触发 remove()

3. /sys/bus/platform/ 特殊文件

除了 devices/drivers/,platform 总线下还有:

  • drivers_autoprobe :控制是否自动绑定驱动(写 N 禁止自动匹配)。
  • drivers_probe:手动触发某个设备的驱动探测。
  • uevent :向用户空间发送设备事件(如 add/remove),常用于测试 udev 规则。

4. 总线分层与路径映射

  • platform vs pci
    • platform:SoC 内部外设控制器(如 Rockchip 的 PCIe RC)。
    • pci:由 RC 初始化出来的标准 PCI 总线,挂载具体的 PCIe 设备。
    • 二者是父子关系:先有 platform 层的 RC,后有 pci 层的设备枚举。
  • I²C/SPI/MMC/USB :都是总线抽象,设备 ID 命名规则不同(I²C 的 0-0014,SPI 的 spi0.0,PCI 的 0000:01:00.0,USB 的 1-1.3:1.0)。

5. 常见总线的操作示例

Platform

bash 复制代码
echo fe150000.pcie > /sys/bus/platform/drivers/rk-pcie/unbind
echo fe150000.pcie > /sys/bus/platform/drivers/rk-pcie/bind

PCI

bash 复制代码
echo 0000:01:00.0 > /sys/bus/pci/drivers/nvme/unbind
echo 0000:01:00.0 > /sys/bus/pci/drivers/nvme/bind

I²C

bash 复制代码
echo 0-0014 > /sys/bus/i2c/drivers/goodix-ts/unbind
echo 0-0014 > /sys/bus/i2c/drivers/goodix-ts/bind

SPI

bash 复制代码
echo spi0.0 > /sys/bus/spi/drivers/m25p80/unbind
echo spi0.0 > /sys/bus/spi/drivers/m25p80/bind

USB

bash 复制代码
echo 1-1.2 > /sys/bus/usb/drivers/usb/unbind
echo 1-1.2 > /sys/bus/usb/drivers/usb/bind

USB-Serial

bash 复制代码
echo 1-1.3:1.0 > /sys/bus/usb-serial/drivers/pl2303/unbind
echo 1-1.3:1.0 > /sys/bus/usb-serial/drivers/pl2303/bind

MMC/SDIO

bash 复制代码
echo mmc1:0001 > /sys/bus/sdio/drivers/bcmdhd/unbind
echo mmc1:0001 > /sys/bus/sdio/drivers/bcmdhd/bind

HID

bash 复制代码
echo 0003:046D:C534.0001 > /sys/bus/hid/drivers/hid-generic/unbind
echo 0003:046D:C534.0001 > /sys/bus/hid/drivers/hid-generic/bind

Virtio

bash 复制代码
echo virtio0 > /sys/bus/virtio/drivers/virtio_net/unbind
echo virtio0 > /sys/bus/virtio/drivers/virtio_net/bind

SCSI

bash 复制代码
echo 0:0:0:0 > /sys/bus/scsi/drivers/sd/unbind
echo 0:0:0:0 > /sys/bus/scsi/drivers/sd/bind

Type-C

bash 复制代码
echo typec0 > /sys/bus/typec/drivers/typec/unbind
echo typec0 > /sys/bus/typec/drivers/typec/bind

MDIO

bash 复制代码
echo f1c00000.mdio:01 > /sys/bus/mdio_bus/drivers/<phy-driver>/unbind
echo f1c00000.mdio:01 > /sys/bus/mdio_bus/drivers/<phy-driver>/bind

其他总线

  • mediamipi-dsiiiogpiosocrpmsgteescmi_protocol 等总线的用法与上述类似:在 /sys/bus/<bus>/drivers/<driver>/ 下执行 echo 设备名到 unbind/bind

6. 调试策略

  • 控制器问题 → 操作 platform/amba/soc 下的驱动。
  • 设备问题 → 操作 pci/i2c/spi/usb/mmc/scsi 等总线下的驱动。
  • 没有 bind/unbind 文件 → 驱动未实现 remove() 或为 built-in。
  • 重绑定影响范围 → platform 层重绑定可能重置控制器,会影响其下所有设备。
  • 日志观察 → 用 dmesg 监控 probe/remove 的打印,结合设备状态变化判断是否成功。

7. 总结

Linux 设备模型通过 sysfs 提供了统一的驱动绑定机制。不同总线类型(platform、PCI、I²C、SPI、USB、MMC、SCSI、HID、Virtio、Type-C 等)都可以支持 bind/unbind,前提是驱动按规范注册并实现了 probe/remove。在调试时要分清层次:

  • platform → 控制器本身
  • pci/i2c/spi/usb 等 → 总线设备

这样才能正确选择在哪个目录下执行 bind/unbind,避免混淆。

相关推荐
JiMoKuangXiangQu6 小时前
Linux USB 设备驱动框架简析
linux·usb 设备驱动
skywalk81636 小时前
快速启动wiki维基百科服务器 kiwix-serve --port=8080 wikipedia_zh_physics_mini_2025-12.zim
linux·运维·服务器·wiki
zl_dfq6 小时前
Linux 之 【文件】(文件共识原理、open、close、访问文件的本质、文件描述符)
linux
那些年的笔记6 小时前
Ubuntu22.04 英文界面转成中文界面
linux·运维·服务器
新兴AI民工6 小时前
【Linux内核七】进程管理模块:进程调度管理器sched_class
linux·服务器·linux内核
快乐的划水a6 小时前
上下文简析
linux·运维·服务器
HABuo6 小时前
【linux进程控制(一)】进程创建&退出-->fork&退出码详谈
linux·运维·服务器·c语言·c++·ubuntu·centos
EndingCoder6 小时前
高级类型:联合类型和类型别名
linux·服务器·前端·ubuntu·typescript
2301_765715146 小时前
Linux中组合使用多个命令的技巧与实现
linux·运维·chrome
想唱rap6 小时前
MySQL内置函数
linux·运维·服务器·数据库·c++·mysql