/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/ /*** This file is part of systemd. Copyright 2010 Lennart Poettering systemd is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. systemd is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with systemd; If not, see . ***/ #include #include #include #include #include "macro.h" #include "execute.h" #include "kmod-setup.h" static const char * const kmod_table[] = { "autofs4", "/sys/class/misc/autofs", "ipv6", "/sys/module/ipv6", "unix", "/proc/net/unix" }; int kmod_setup(void) { unsigned i, n = 0; const char * cmdline[3 + ELEMENTSOF(kmod_table) + 1]; ExecCommand command; ExecContext context; pid_t pid; int r; siginfo_t status; for (i = 0; i < ELEMENTSOF(kmod_table); i += 2) { if (access(kmod_table[i+1], F_OK) >= 0) continue; log_debug("Your kernel apparently lacks built-in %s support. Might be a good idea to compile it in. " "We'll now try to work around this by calling '/sbin/modprobe %s'...", kmod_table[i], kmod_table[i]); cmdline[3 + n++] = kmod_table[i]; } if (n <= 0) return 0; cmdline[0] = "/sbin/modprobe"; cmdline[1] = "-qab"; cmdline[2] = "--"; cmdline[3 + n] = NULL; zero(command); zero(context); command.path = (char*) cmdline[0]; command.argv = (char**) cmdline; exec_context_init(&context); r = exec_spawn(&command, NULL, &context, NULL, 0, NULL, false, false, false, false, NULL, &pid); exec_context_done(&context); if (r < 0) return r; if ((r = wait_for_terminate(pid, &status)) < 0) return -errno; if (status.si_code == CLD_EXITED) { if (status.si_status != 0) { log_warning("/sbin/modprobe failed with error code %i.", status.si_status); return -EPROTO; } log_debug("/sbin/modprobe succeeded."); return 0; } else if (status.si_code == CLD_KILLED || status.si_code == CLD_DUMPED) { log_warning("/sbin/modprobe terminated by signal %s.", signal_to_string(status.si_status)); return -EPROTO; } log_warning("/sbin/modprobe failed due to unknown reason."); return -EPROTO; }