2000-02-09 01:28:47 +05:30
|
|
|
/* vi: set sw=4 ts=4: */
|
1999-10-19 02:52:59 +05:30
|
|
|
/*
|
|
|
|
* Mini swapon/swapoff implementation for busybox
|
|
|
|
*
|
2004-03-15 13:59:22 +05:30
|
|
|
* Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
|
1999-10-19 02:52:59 +05:30
|
|
|
*
|
2010-08-16 23:44:46 +05:30
|
|
|
* Licensed under GPLv2, see file LICENSE in this source tree.
|
1999-10-19 02:52:59 +05:30
|
|
|
*/
|
|
|
|
|
2011-04-11 06:59:49 +05:30
|
|
|
//usage:#define swapon_trivial_usage
|
|
|
|
//usage: "[-a]" IF_FEATURE_SWAPON_PRI(" [-p PRI]") " [DEVICE]"
|
|
|
|
//usage:#define swapon_full_usage "\n\n"
|
|
|
|
//usage: "Start swapping on DEVICE\n"
|
|
|
|
//usage: "\n -a Start swapping on all swap devices"
|
|
|
|
//usage: IF_FEATURE_SWAPON_PRI(
|
|
|
|
//usage: "\n -p PRI Set swap device priority"
|
|
|
|
//usage: )
|
|
|
|
//usage:
|
|
|
|
//usage:#define swapoff_trivial_usage
|
|
|
|
//usage: "[-a] [DEVICE]"
|
|
|
|
//usage:#define swapoff_full_usage "\n\n"
|
|
|
|
//usage: "Stop swapping on DEVICE\n"
|
|
|
|
//usage: "\n -a Stop swapping on all swap devices"
|
|
|
|
|
2007-05-27 00:30:18 +05:30
|
|
|
#include "libbb.h"
|
1999-10-19 02:52:59 +05:30
|
|
|
#include <mntent.h>
|
2011-07-08 12:19:40 +05:30
|
|
|
#ifndef __BIONIC__
|
|
|
|
# include <sys/swap.h>
|
|
|
|
#endif
|
1999-10-19 02:52:59 +05:30
|
|
|
|
2009-09-20 07:58:22 +05:30
|
|
|
#if ENABLE_FEATURE_MOUNT_LABEL
|
|
|
|
# include "volume_id.h"
|
|
|
|
#else
|
|
|
|
# define resolve_mount_spec(fsname) ((void)0)
|
|
|
|
#endif
|
|
|
|
|
2011-07-08 12:19:40 +05:30
|
|
|
#ifndef MNTTYPE_SWAP
|
|
|
|
# define MNTTYPE_SWAP "swap"
|
|
|
|
#endif
|
|
|
|
|
2008-05-19 04:35:34 +05:30
|
|
|
#if ENABLE_FEATURE_SWAPON_PRI
|
|
|
|
struct globals {
|
|
|
|
int flags;
|
2010-02-04 19:30:15 +05:30
|
|
|
} FIX_ALIASING;
|
2008-05-19 04:35:34 +05:30
|
|
|
#define G (*(struct globals*)&bb_common_bufsiz1)
|
|
|
|
#define g_flags (G.flags)
|
|
|
|
#else
|
|
|
|
#define g_flags 0
|
|
|
|
#endif
|
2011-09-21 05:29:15 +05:30
|
|
|
#define INIT_G() do { } while (0)
|
2008-05-19 04:35:34 +05:30
|
|
|
|
2006-09-13 03:12:17 +05:30
|
|
|
static int swap_enable_disable(char *device)
|
1999-10-19 02:52:59 +05:30
|
|
|
{
|
2000-02-09 01:28:47 +05:30
|
|
|
int status;
|
2002-11-03 05:55:23 +05:30
|
|
|
struct stat st;
|
|
|
|
|
2009-09-20 07:58:22 +05:30
|
|
|
resolve_mount_spec(&device);
|
2006-03-13 21:15:16 +05:30
|
|
|
xstat(device, &st);
|
2002-11-03 05:55:23 +05:30
|
|
|
|
2007-03-15 04:25:39 +05:30
|
|
|
#if ENABLE_DESKTOP
|
2002-11-03 05:55:23 +05:30
|
|
|
/* test for holes */
|
2005-09-13 07:59:39 +05:30
|
|
|
if (S_ISREG(st.st_mode))
|
2007-11-06 01:03:38 +05:30
|
|
|
if (st.st_blocks * (off_t)512 < st.st_size)
|
2007-03-15 04:25:39 +05:30
|
|
|
bb_error_msg("warning: swap file has holes");
|
2007-03-12 23:46:24 +05:30
|
|
|
#endif
|
2000-02-09 01:28:47 +05:30
|
|
|
|
2006-10-04 02:30:43 +05:30
|
|
|
if (applet_name[5] == 'n')
|
2008-05-19 04:35:34 +05:30
|
|
|
status = swapon(device, g_flags);
|
2000-02-09 01:28:47 +05:30
|
|
|
else
|
|
|
|
status = swapoff(device);
|
|
|
|
|
2002-10-26 15:57:42 +05:30
|
|
|
if (status != 0) {
|
2007-10-01 17:28:38 +05:30
|
|
|
bb_simple_perror_msg(device);
|
2005-09-16 10:11:20 +05:30
|
|
|
return 1;
|
2002-10-26 15:57:42 +05:30
|
|
|
}
|
2005-09-13 07:59:39 +05:30
|
|
|
|
2005-09-16 10:11:20 +05:30
|
|
|
return 0;
|
1999-10-19 02:52:59 +05:30
|
|
|
}
|
|
|
|
|
2002-10-26 15:57:42 +05:30
|
|
|
static int do_em_all(void)
|
1999-10-19 02:52:59 +05:30
|
|
|
{
|
|
|
|
struct mntent *m;
|
2005-09-13 07:59:39 +05:30
|
|
|
FILE *f;
|
|
|
|
int err;
|
1999-10-19 02:52:59 +05:30
|
|
|
|
2005-09-13 07:59:39 +05:30
|
|
|
f = setmntent("/etc/fstab", "r");
|
2000-12-22 07:18:07 +05:30
|
|
|
if (f == NULL)
|
2003-03-19 14:43:01 +05:30
|
|
|
bb_perror_msg_and_die("/etc/fstab");
|
2005-09-13 07:59:39 +05:30
|
|
|
|
|
|
|
err = 0;
|
2010-04-30 01:50:57 +05:30
|
|
|
while ((m = getmntent(f)) != NULL) {
|
|
|
|
if (strcmp(m->mnt_type, MNTTYPE_SWAP) == 0) {
|
|
|
|
/* swapon -a should ignore entries with noauto,
|
|
|
|
* but swapoff -a should process them */
|
|
|
|
if (applet_name[5] != 'n'
|
|
|
|
|| hasmntopt(m, MNTOPT_NOAUTO) == NULL
|
|
|
|
) {
|
2013-08-08 13:51:27 +05:30
|
|
|
#if ENABLE_FEATURE_SWAPON_PRI
|
2013-11-12 15:46:30 +05:30
|
|
|
char *p;
|
2013-08-08 13:51:27 +05:30
|
|
|
g_flags = 0; /* each swap space might have different flags */
|
2013-11-12 15:46:30 +05:30
|
|
|
p = hasmntopt(m, "pri");
|
2013-08-08 13:51:27 +05:30
|
|
|
if (p) {
|
|
|
|
/* Max allowed 32767 (==SWAP_FLAG_PRIO_MASK) */
|
2013-09-27 17:09:41 +05:30
|
|
|
unsigned int swap_prio = MIN(bb_strtou(p + 4 , NULL, 10), SWAP_FLAG_PRIO_MASK);
|
2013-08-08 13:51:27 +05:30
|
|
|
/* We want to allow "NNNN,foo", thus errno == EINVAL is allowed too */
|
|
|
|
if (errno != ERANGE) {
|
|
|
|
g_flags = SWAP_FLAG_PREFER |
|
|
|
|
(swap_prio << SWAP_FLAG_PRIO_SHIFT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2010-04-30 01:50:57 +05:30
|
|
|
err += swap_enable_disable(m->mnt_fsname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-09-13 07:59:39 +05:30
|
|
|
|
2010-04-30 01:50:57 +05:30
|
|
|
if (ENABLE_FEATURE_CLEAN_UP)
|
|
|
|
endmntent(f);
|
2005-09-13 07:59:39 +05:30
|
|
|
|
2002-10-26 15:57:42 +05:30
|
|
|
return err;
|
1999-10-19 02:52:59 +05:30
|
|
|
}
|
|
|
|
|
2007-10-11 15:35:36 +05:30
|
|
|
int swap_on_off_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
2008-07-05 14:48:54 +05:30
|
|
|
int swap_on_off_main(int argc UNUSED_PARAM, char **argv)
|
1999-10-19 02:52:59 +05:30
|
|
|
{
|
2005-09-16 10:11:20 +05:30
|
|
|
int ret;
|
2005-09-13 07:59:39 +05:30
|
|
|
|
2011-09-21 05:29:15 +05:30
|
|
|
INIT_G();
|
|
|
|
|
2008-05-19 04:35:34 +05:30
|
|
|
#if !ENABLE_FEATURE_SWAPON_PRI
|
2007-08-18 21:02:12 +05:30
|
|
|
ret = getopt32(argv, "a");
|
2008-05-19 04:35:34 +05:30
|
|
|
#else
|
2011-08-14 03:06:36 +05:30
|
|
|
if (applet_name[5] == 'n')
|
|
|
|
opt_complementary = "p+";
|
2008-05-19 04:35:34 +05:30
|
|
|
ret = getopt32(argv, (applet_name[5] == 'n') ? "ap:" : "a", &g_flags);
|
|
|
|
|
|
|
|
if (ret & 2) { // -p
|
|
|
|
g_flags = SWAP_FLAG_PREFER |
|
|
|
|
((g_flags & SWAP_FLAG_PRIO_MASK) << SWAP_FLAG_PRIO_SHIFT);
|
|
|
|
ret &= 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (ret /* & 1: not needed */) // -a
|
2005-09-13 07:00:19 +05:30
|
|
|
return do_em_all();
|
2005-09-13 07:59:39 +05:30
|
|
|
|
2008-05-19 04:35:34 +05:30
|
|
|
argv += optind;
|
|
|
|
if (!*argv)
|
|
|
|
bb_show_usage();
|
|
|
|
|
2007-03-12 23:46:24 +05:30
|
|
|
/* ret = 0; redundant */
|
2008-05-19 04:35:34 +05:30
|
|
|
do {
|
2005-09-16 10:11:20 +05:30
|
|
|
ret += swap_enable_disable(*argv);
|
2008-05-19 04:35:34 +05:30
|
|
|
} while (*++argv);
|
|
|
|
|
2005-09-16 10:11:20 +05:30
|
|
|
return ret;
|
1999-10-19 02:52:59 +05:30
|
|
|
}
|