hush/util-linux/swaponoff.c
Matt Whitlock 0a53b203f9 add discard option -d to swapon
Signed-off-by: Matt Whitlock <busybox@mattwhitlock.name>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2014-03-23 18:36:23 +01:00

231 lines
5.4 KiB
C

/* vi: set sw=4 ts=4: */
/*
* Mini swapon/swapoff implementation for busybox
*
* Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
*
* Licensed under GPLv2, see file LICENSE in this source tree.
*/
//usage:#define swapon_trivial_usage
//usage: "[-a]" IF_FEATURE_SWAPON_DISCARD(" [-d[POL]]") 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_DISCARD(
//usage: "\n -d[POL] Discard blocks at swapon (POL=once),"
//usage: "\n as freed (POL=pages), or both (POL omitted)"
//usage: )
//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"
#include "libbb.h"
#include <mntent.h>
#ifndef __BIONIC__
# include <sys/swap.h>
#endif
#if ENABLE_FEATURE_MOUNT_LABEL
# include "volume_id.h"
#else
# define resolve_mount_spec(fsname) ((void)0)
#endif
#ifndef MNTTYPE_SWAP
# define MNTTYPE_SWAP "swap"
#endif
#if ENABLE_FEATURE_SWAPON_DISCARD
#ifndef SWAP_FLAG_DISCARD
#define SWAP_FLAG_DISCARD 0x10000
#endif
#ifndef SWAP_FLAG_DISCARD_ONCE
#define SWAP_FLAG_DISCARD_ONCE 0x20000
#endif
#ifndef SWAP_FLAG_DISCARD_PAGES
#define SWAP_FLAG_DISCARD_PAGES 0x40000
#endif
#define SWAP_FLAG_DISCARD_MASK \
(SWAP_FLAG_DISCARD | SWAP_FLAG_DISCARD_ONCE | SWAP_FLAG_DISCARD_PAGES)
#endif
#if ENABLE_FEATURE_SWAPON_DISCARD || ENABLE_FEATURE_SWAPON_PRI
struct globals {
int flags;
} FIX_ALIASING;
#define G (*(struct globals*)&bb_common_bufsiz1)
#define g_flags (G.flags)
#else
#define g_flags 0
#endif
#define INIT_G() do { } while (0)
static int swap_enable_disable(char *device)
{
int status;
struct stat st;
resolve_mount_spec(&device);
xstat(device, &st);
#if ENABLE_DESKTOP
/* test for holes */
if (S_ISREG(st.st_mode))
if (st.st_blocks * (off_t)512 < st.st_size)
bb_error_msg("warning: swap file has holes");
#endif
if (applet_name[5] == 'n')
status = swapon(device, g_flags);
else
status = swapoff(device);
if (status != 0) {
bb_simple_perror_msg(device);
return 1;
}
return 0;
}
static int do_em_all(void)
{
struct mntent *m;
FILE *f;
int err;
#ifdef G
int cl_flags = g_flags;
#endif
f = setmntent("/etc/fstab", "r");
if (f == NULL)
bb_perror_msg_and_die("/etc/fstab");
err = 0;
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
) {
#if ENABLE_FEATURE_SWAPON_DISCARD || ENABLE_FEATURE_SWAPON_PRI
char *p;
g_flags = cl_flags; /* each swap space might have different flags */
#if ENABLE_FEATURE_SWAPON_DISCARD
p = hasmntopt(m, "discard");
if (p) {
if (p[7] == '=') {
if (strncmp(p + 8, "once", 4) == 0 && (p[12] == ',' || p[12] == '\0'))
g_flags = (g_flags & ~SWAP_FLAG_DISCARD_MASK) | SWAP_FLAG_DISCARD | SWAP_FLAG_DISCARD_ONCE;
else if (strncmp(p + 8, "pages", 5) == 0 && (p[13] == ',' || p[13] == '\0'))
g_flags = (g_flags & ~SWAP_FLAG_DISCARD_MASK) | SWAP_FLAG_DISCARD | SWAP_FLAG_DISCARD_PAGES;
}
else if (p[7] == ',' || p[7] == '\0')
g_flags = (g_flags & ~SWAP_FLAG_DISCARD_MASK) | SWAP_FLAG_DISCARD;
}
#endif
#if ENABLE_FEATURE_SWAPON_PRI
p = hasmntopt(m, "pri");
if (p) {
/* Max allowed 32767 (== SWAP_FLAG_PRIO_MASK) */
unsigned prio = bb_strtou(p + 4, NULL, 10);
/* We want to allow "NNNN,foo", thus errno == EINVAL is allowed too */
if (errno != ERANGE) {
g_flags = (g_flags & ~SWAP_FLAG_PRIO_MASK) | SWAP_FLAG_PREFER |
MIN(prio, SWAP_FLAG_PRIO_MASK);
}
}
#endif
#endif
err += swap_enable_disable(m->mnt_fsname);
}
}
}
if (ENABLE_FEATURE_CLEAN_UP)
endmntent(f);
return err;
}
int swap_on_off_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
int swap_on_off_main(int argc UNUSED_PARAM, char **argv)
{
int ret;
#if ENABLE_FEATURE_SWAPON_DISCARD
char *discard = NULL;
#endif
#if ENABLE_FEATURE_SWAPON_PRI
unsigned prio;
#endif
INIT_G();
#if !ENABLE_FEATURE_SWAPON_DISCARD && !ENABLE_FEATURE_SWAPON_PRI
ret = getopt32(argv, "a");
#else
#if ENABLE_FEATURE_SWAPON_PRI
if (applet_name[5] == 'n')
opt_complementary = "p+";
#endif
ret = getopt32(argv, (applet_name[5] == 'n') ?
#if ENABLE_FEATURE_SWAPON_DISCARD
"d::"
#endif
#if ENABLE_FEATURE_SWAPON_PRI
"p:"
#endif
"a" : "a"
#if ENABLE_FEATURE_SWAPON_DISCARD
, &discard
#endif
#if ENABLE_FEATURE_SWAPON_PRI
, &prio
#endif
);
#endif
#if ENABLE_FEATURE_SWAPON_DISCARD
if (ret & 1) { // -d
if (!discard)
g_flags |= SWAP_FLAG_DISCARD;
else if (strcmp(discard, "once") == 0)
g_flags |= SWAP_FLAG_DISCARD | SWAP_FLAG_DISCARD_ONCE;
else if (strcmp(discard, "pages") == 0)
g_flags |= SWAP_FLAG_DISCARD | SWAP_FLAG_DISCARD_PAGES;
else
bb_show_usage();
}
ret >>= 1;
#endif
#if ENABLE_FEATURE_SWAPON_PRI
if (ret & 1) // -p
g_flags |= SWAP_FLAG_PREFER |
MIN(prio, SWAP_FLAG_PRIO_MASK);
ret >>= 1;
#endif
if (ret /* & 1: not needed */) // -a
return do_em_all();
argv += optind;
if (!*argv)
bb_show_usage();
/* ret = 0; redundant */
do {
ret += swap_enable_disable(*argv);
} while (*++argv);
return ret;
}