/* vi: set sw=4 ts=4: */ /* * Mini rmmod implementation for busybox * * Copyright (C) 1999-2004 by Erik Andersen * Copyright (C) 2008 Timo Teras * * Licensed under GPLv2 or later, see file LICENSE in this source tree. */ //applet:IF_RMMOD(APPLET(rmmod, BB_DIR_SBIN, BB_SUID_DROP)) //usage:#if !ENABLE_MODPROBE_SMALL //usage:#define rmmod_trivial_usage //usage: "[-wfa] [MODULE]..." //usage:#define rmmod_full_usage "\n\n" //usage: "Unload kernel modules\n" //usage: "\n -w Wait until the module is no longer used" //usage: "\n -f Force unload" //usage: "\n -a Remove all unused modules (recursively)" //usage:#define rmmod_example_usage //usage: "$ rmmod tulip\n" //usage:#endif #include "libbb.h" #include "modutils.h" int rmmod_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; int rmmod_main(int argc UNUSED_PARAM, char **argv) { int n; unsigned flags = O_NONBLOCK | O_EXCL; /* Parse command line. */ n = getopt32(argv, "wfas"); // -s ignored argv += optind; if (n & 1) // --wait flags &= ~O_NONBLOCK; if (n & 2) // --force flags |= O_TRUNC; if (n & 4) { /* Unload _all_ unused modules via NULL delete_module() call */ if (bb_delete_module(NULL, flags) != 0 && errno != EFAULT) bb_perror_msg_and_die("rmmod"); return EXIT_SUCCESS; } if (!*argv) bb_show_usage(); n = ENABLE_FEATURE_2_4_MODULES && get_linux_version_code() < KERNEL_VERSION(2,6,0); while (*argv) { char modname[MODULE_NAME_LEN]; const char *bname; bname = bb_basename(*argv++); if (n) safe_strncpy(modname, bname, MODULE_NAME_LEN); else filename2modname(bname, modname); if (bb_delete_module(modname, flags)) bb_error_msg_and_die("can't unload '%s': %s", modname, moderror(errno)); } return EXIT_SUCCESS; }