Linux 磁盘热插拔与自动挂载

3次阅读

linux 不自动挂载 usb/nvme 设备,需用 udev 规则触发;应精准匹配 id_bus==”usb” 和 id_fs_type,挂载到 /media/$user/ 下并设 uid/gid,ntfs 需禁用 windows 快速启动或加 remove_hiberfile 选项。

Linux 磁盘热插拔与自动挂载

udev 规则写不对,设备插上根本不会自动挂载

Linux 不会默认为新插入的 USB 硬盘或 NVMe 热插拔盘自动挂载,得靠 udev 规则触发挂载动作。但规则里如果用 SYMLINKKERNEL=="sd*" 这类宽泛匹配,容易误触系统盘或已有设备,导致挂载错乱甚至系统卡死。

实操建议:

  • 先用 udevadm monitor --subsystem-match=block 插拔设备,确认实际触发的 DEVNAME(如 /dev/sdb)和关键属性(如 ID_MODEL="USB_3.0_Flash_Disk"
  • 规则文件放在 /etc/udev/rules.d/99-usb-mount.rules,内容必须包含 ENV{ID_BUS}=="usb" + ENV{ID_FS_TYPE}!="",过滤掉无文件系统的块设备
  • 避免在规则里直接调用 mount,改用 RUN+="/bin/sh -c 'mkdir -p /mnt/usb-%k && mount -o uid=1000,gid=1000 /dev/%k /mnt/usb-%k 2>/dev/NULL'",否则 udev 同步阻塞会导致插拔响应延迟

systemd-mount 挂载后权限不对,普通用户无法读写

systemd-mount --no-block --automount=yes 虽能自动建挂点并监听,但默认以 root 权限挂载,且不设 uid/gid,导致桌面环境里的用户看不到文件或提示“Permission denied”。

实操建议:

  • 必须加 --Property=UID=1000--property=GID=1000(替换为你实际用户的 ID,可用 id -u 查)
  • 对 FAT32/exFAT 设备,额外加 --property=FSType=vfat 并在 mount 选项里补 umask=000,否则 Windows 写入的文件默认只有 owner 可读
  • 不要依赖 systemd-mount --discover 扫描已插设备——它只对当前未挂载的生效,热插拔必须靠 udev 触发

NTFS 分区挂载失败,报错 “The disk contains an unclean file system”

Windows 快速启动(Fast Startup)启用时,NTFS 分区会被“假卸载”,Linux 下 ntfs-3g 默认拒绝挂载,报错信息就是 The disk contains an unclean file system,不是磁盘坏了。

实操建议:

  • 临时解决:加 -o remove_hiberfile 强制清除休眠签名(⚠️会丢 Windows 未保存的休眠状态)
  • 长期方案:Windows 里关掉快速启动(电源选项 → 选择电源按钮的功能 → 更改当前不可用设置 → 取消勾选“启用快速启动”)
  • 如果仍报错,检查是否用了 ntfs-3g 而非内核原生 ntfs 驱动(后者只读),确认 lsmod | grep ntfs3 有输出才支持可写 NTFS3

自动挂载后桌面图标不显示,Nautilus/Dolphin 找不到挂载点

GNOME 和 KDE 默认只监控 /media/用户名//run/media/用户名/ 下的挂载点,自己建在 /mnt/usb-* 的路径不会被桌面环境识别,图标、弹窗、侧边栏都不出现。

实操建议:

  • 挂载目标路径必须是 /media/$USER/xxx/run/media/$USER/xxx,其中 $USER 要用真实用户名(不能写 %u,udev 不解析这个)
  • 确保目录属主是对应用户:RUN+="/bin/sh -c 'mkdir -p /media/$USER/usb-%k && chown $USER:$USER /media/$USER/usb-%k'"
  • 如果用 systemd-mount,必须加 --no-pager 并确保 systemd-logind 正常运行,否则 /run/media/ 下的子目录不会被桌面服务 watch 到

udev 规则和挂载路径的组合非常敏感,少一个条件或路径写错一级,整个流程就静默失败——没有错误提示,也没有图标,只能靠 journalctl -u systemd-udevdfindmnt 交叉验证。

text=ZqhQzanResources