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>/ 下,可能会有 bind 和 unbind 文件:
- 生成条件 :
- 驱动通过标准 API 注册(如
platform_driver_register()、i2c_add_driver()、pci_register_driver())。 - 驱动实现了
probe()和remove()。 - 内核启用了
CONFIG_SYSFS和CONFIG_HOTPLUG。
- 驱动通过标准 API 注册(如
- 功能 :
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
其他总线
- media 、mipi-dsi 、iio 、gpio 、soc 、rpmsg 、tee 、scmi_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,避免混淆。