/* * kmod - one tool to rule them all * * Copyright (C) 2011-2013 ProFUSION embedded systems * * This program 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. * * This program 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 this program. If not, see <http://www.gnu.org/licenses/>. */ #include <errno.h> #include <getopt.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <shared/util.h> #include <libkmod/libkmod.h> #include "kmod.h" static const char options_s[] = "+hV"; static const struct option options[] = { { "help", no_argument, NULL, 'h' }, { "version", no_argument, NULL, 'V' }, {} }; static const struct kmod_cmd kmod_cmd_help; static const struct kmod_cmd *kmod_cmds[] = { &kmod_cmd_help, &kmod_cmd_list, &kmod_cmd_static_nodes, #ifdef ENABLE_EXPERIMENTAL &kmod_cmd_insert, &kmod_cmd_remove, #endif }; static const struct kmod_cmd *kmod_compat_cmds[] = { &kmod_cmd_compat_lsmod, &kmod_cmd_compat_rmmod, &kmod_cmd_compat_insmod, &kmod_cmd_compat_modinfo, &kmod_cmd_compat_modprobe, &kmod_cmd_compat_depmod, }; static int kmod_help(int argc, char *argv[]) { size_t i; printf("kmod - Manage kernel modules: list, load, unload, etc\n" "Usage:\n" "\t%s [options] command [command_options]\n\n" "Options:\n" "\t-V, --version show version\n" "\t-h, --help show this help\n\n" "Commands:\n", basename(argv[0])); for (i = 0; i < ARRAY_SIZE(kmod_cmds); i++) { if (kmod_cmds[i]->help != NULL) { printf(" %-12s %s\n", kmod_cmds[i]->name, kmod_cmds[i]->help); } } puts("\nkmod also handles gracefully if called from following symlinks:"); for (i = 0; i < ARRAY_SIZE(kmod_compat_cmds); i++) { if (kmod_compat_cmds[i]->help != NULL) { printf(" %-12s %s\n", kmod_compat_cmds[i]->name, kmod_compat_cmds[i]->help); } } return EXIT_SUCCESS; } static const struct kmod_cmd kmod_cmd_help = { .name = "help", .cmd = kmod_help, .help = "Show help message", }; static int handle_kmod_commands(int argc, char *argv[]) { const char *cmd; int err = 0; size_t i; for (;;) { int c; c = getopt_long(argc, argv, options_s, options, NULL); if (c == -1) break; switch (c) { case 'h': kmod_help(argc, argv); return EXIT_SUCCESS; case 'V': puts(PACKAGE " version " VERSION); puts(KMOD_FEATURES); return EXIT_SUCCESS; case '?': return EXIT_FAILURE; default: fprintf(stderr, "Error: unexpected getopt_long() value '%c'.\n", c); return EXIT_FAILURE; } } if (optind >= argc) { fputs("missing command\n", stderr); goto fail; } cmd = argv[optind]; for (i = 0, err = -EINVAL; i < ARRAY_SIZE(kmod_cmds); i++) { if (streq(kmod_cmds[i]->name, cmd)) { err = kmod_cmds[i]->cmd(--argc, ++argv); break; } } if (err < 0) { fprintf(stderr, "invalid command '%s'\n", cmd); goto fail; } return err; fail: kmod_help(argc, argv); return EXIT_FAILURE; } static int handle_kmod_compat_commands(int argc, char *argv[]) { const char *cmd; size_t i; cmd = basename(argv[0]); for (i = 0; i < ARRAY_SIZE(kmod_compat_cmds); i++) { if (streq(kmod_compat_cmds[i]->name, cmd)) return kmod_compat_cmds[i]->cmd(argc, argv); } return -ENOENT; } int main(int argc, char *argv[]) { int err; if (streq(program_invocation_short_name, "kmod")) err = handle_kmod_commands(argc, argv); else err = handle_kmod_compat_commands(argc, argv); return err; }