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,避免混淆。

相关推荐
可可苏饼干31 分钟前
ELK(Elastic Stack)日志采集与分析
linux·运维·笔记·elk
大柏怎么被偷了42 分钟前
【Git】基本操作
linux·运维·git
小女孩真可爱42 分钟前
大模型学习记录(八)---------RAG评估
linux·人工智能·python
我在人间贩卖青春1 小时前
查看文件相关命令
linux·查看文件
番茄你个西红41 小时前
安装KingbaseES时服务器swap的设置
linux·数据库
python百炼成钢2 小时前
50.linux_USB驱动
linux·运维·服务器·驱动开发
jay2 小时前
ens2f0 IP 远程连线,balance-alb 模式配置双网卡(ens2f0 + ens6f0)Bond,避免断网
linux·运维·服务器·网络·tcp/ip
Evan芙2 小时前
用Shell脚本破解经典鸡兔同笼问题
linux·运维·网络
꧁坚持很酷꧂3 小时前
Ubuntu系统下Qt程序连接串口设备没有问题,但运行时出现Permission denied的解决方法
linux·qt·ubuntu