最可靠方式是使用systemd-modules-load.service,通过/etc/modules-load.d/下.conf文件声明模块名;每行一个模块名(不含.ko后缀),名称须与lsmod或/lib/modules/$(uname -r)/中实际文件一致,空行和#开头行被忽略。
Linux 系统启动时自动加载内核模块,最可靠、最主流的方式是使用 systemd-modules-load.service,通过 /etc/modules-load.d/ 下的配置文件声明模块名。其他方法(如 /etc/rc.local 或自定义 init 脚本)在现代 systemd 系统中不是首选,且容易被忽略或失效。
只需在 /etc/modules-load.d/ 目录下新建一个以 .conf 结尾的文件,每行写一个要加载的模块名(不含 .ko 后缀),空行和 # 开头的注释会被忽略。
lsmod 输出或 /lib/modules/$(uname -r)/ 下实际存在的文件名一致(例如 ip_vs_rr 对应 ip_vs_rr.ko)modprobe 会自动从 modules.dep 解析依赖并加载8139too 是 rtl8139 的别名),写别名也可以,但建议优先用标准模块名echo "ip_vs_rr" | sudo tee /etc/modules-load.d/ipvs.conf
执行 systemctl restart systemd-modules-load.service 报错,常见原因是模块尚未安装到 /lib/modules/$(uname -r)/,或模块名拼写错误,或模块依赖未满足(比如 ip_vs_rr 依赖 ip_vs,但后者没列在配置里)。
systemd-modules-load.service 在启动时调用 modprobe <module>,不检查模块是否存在,只看返回码modprobe: FATAL: Module xxx not found in directory /lib/modules/5.15.0-xx-generic
sudo modprobe xxx,看是否成功;失败则用 modinfo xxx 检查模块路径和依赖sudo make modules_install(自己编译的模块)或安装了对应内核的 linux-modules-extra 包(Ubuntu/Debian)某些模块无法通过静态配置提前加载,因为它们依赖硬件探测结果或用户空间事件,modules-load.d 不适用。
e1000e、usbserial):由 udev 根据 modules.alias 动态触发,无需手动配置nf_conntrack_ftp ports=21):modules-load.d 不支持传参,得改用 /etc/modprobe.d/*.conf 配置 options 行blacklist 屏蔽的模块:即使写进 modules-load.d,也会被 modprobe 忽略(检查 /etc/modprobe.d/*.conf 中是否有 blacklist xxx)dracut 或 update-initramfs 重新生成镜像真正关键的是区分“系统级静态加载”和“设备级动态加载”。modules-load.d 只管前者;后者靠内核总线子系统 + udev + modules.alias 协同完成,不需要你干预——除非你发现某个热插拔设备没被识别,那才该去查 modinfo 和 dmesg | grep -i pci。