/* * A swapon(8)/swapoff(8) for Linux 0.99. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "bitops.h" #include "blkdev.h" #include "xmalloc.h" #include "swap_constants.h" #include "nls.h" #include "fsprobe.h" #include "realpath.h" #include "pathnames.h" #include "sundries.h" #include "swapheader.h" #define PATH_MKSWAP "/sbin/mkswap" #ifdef HAVE_SYS_SWAP_H # include #endif #ifndef SWAPON_HAS_TWO_ARGS /* libc is insane, let's call the kernel */ # include # define swapon(path, flags) syscall(SYS_swapon, path, flags) # define swapoff(path) syscall(SYS_swapoff, path) #endif #define streq(s, t) (strcmp ((s), (t)) == 0) #define QUIET 1 #define CANONIC 1 #define MAX_PAGESIZE (64 * 1024) int all = 0; int priority = -1; /* non-prioritized swap by default */ /* If true, don't complain if the device/file doesn't exist */ int ifexists = 0; static struct option longswaponopts[] = { /* swapon only */ { "priority", required_argument, 0, 'p' }, { "ifexists", 0, 0, 'e' }, { "summary", 0, 0, 's' }, /* also for swapoff */ { "all", 0, 0, 'a' }, { "help", 0, 0, 'h' }, { "verbose", 0, 0, 'v' }, { "version", 0, 0, 'V' }, { NULL, 0, 0, 0 } }; static struct option *longswapoffopts = &longswaponopts[2]; static int cannot_find(const char *special); #define PRINT_USAGE_SPECIAL(_fp) \ fprintf(_fp, _( \ "The parameter:\n" \ " {-L label | LABEL=label} LABEL of device to be used\n" \ " {-U uuid | UUID=uuid} UUID of device to be used\n" \ " name of device to be used\n" \ " name of file to be used\n\n")) static void swapon_usage(FILE *fp, int n) { fprintf(fp, _("\nUsage:\n" " %1$s -a [-e] [-v] enable all swaps from /etc/fstab\n" " %1$s [-p priority] [-v] enable given swap\n" " %1$s -s display swap usage summary\n" " %1$s -h display help\n" " %1$s -V display version\n\n"), progname); PRINT_USAGE_SPECIAL(fp); exit(n); } static void swapoff_usage(FILE *fp, int n) { fprintf(fp, _("\nUsage:\n" " %1$s -a [-v] disable all swaps\n" " %1$s [-v] disable given swap\n" " %1$s -h display help\n" " %1$s -V display version\n\n"), progname); PRINT_USAGE_SPECIAL(fp); exit(n); } /* * contents of /proc/swaps */ static int numSwaps; static char **swapFiles; /* array of swap file and partition names */ static void read_proc_swaps(void) { FILE *swaps; char line[1024]; char *p, **q; numSwaps = 0; swapFiles = NULL; swaps = fopen(_PATH_PROC_SWAPS, "r"); if (swaps == NULL) return; /* nothing wrong */ /* skip the first line */ if (!fgets(line, sizeof(line), swaps)) { fprintf (stderr, _("%s: %s: unexpected file format\n"), progname, _PATH_PROC_SWAPS); fclose(swaps); return; } while (fgets(line, sizeof(line), swaps)) { /* * Cut the line "swap_device ... more info" after device. * This will fail with names with embedded spaces. */ for (p = line; *p && *p != ' '; p++); *p = 0; q = realloc(swapFiles, (numSwaps+1) * sizeof(*swapFiles)); if (q == NULL) break; swapFiles = q; swapFiles[numSwaps++] = strdup(line); } fclose(swaps); } static int is_in_proc_swaps(const char *fname) { int i; for (i = 0; i < numSwaps; i++) if (swapFiles[i] && !strcmp(fname, swapFiles[i])) return 1; return 0; } static int display_summary(void) { FILE *swaps; char line[1024] ; if ((swaps = fopen(_PATH_PROC_SWAPS, "r")) == NULL) { int errsv = errno; fprintf(stderr, "%s: %s: %s\n", progname, _PATH_PROC_SWAPS, strerror(errsv)); return -1; } while (fgets(line, sizeof(line), swaps)) printf("%s", line); fclose(swaps); return 0 ; } static int swap_is_suspend(const char *device) { const char *type = fsprobe_get_fstype_by_devname(device); /* S1SUSPEND/S2SUSPEND = * * "swsuspend" in libblkid * "suspend" in libvolume_id */ if (type && (strcmp(type, "suspend") == 0 || strcmp(type, "swsuspend") == 0)) return 1; return 0; } /* calls mkswap */ static int swap_reinitialize(const char *device) { const char *label = fsprobe_get_label_by_devname(device); const char *uuid = fsprobe_get_uuid_by_devname(device); pid_t pid; int status, ret; char *cmd[7]; int idx=0; switch((pid=fork())) { case -1: /* fork error */ fprintf(stderr, _("%s: cannot fork: %s\n"), progname, strerror(errno)); return -1; case 0: /* child */ cmd[idx++] = PATH_MKSWAP; if (label && *label) { cmd[idx++] = "-L"; cmd[idx++] = (char *) label; } if (uuid && *uuid) { cmd[idx++] = "-U"; cmd[idx++] = (char *) uuid; } cmd[idx++] = (char *) device; cmd[idx++] = NULL; execv(cmd[0], cmd); perror("execv"); exit(1); /* error */ default: /* parent */ do { if ((ret = waitpid(pid, &status, 0)) < 0 && errno == EINTR) continue; else if (ret < 0) { fprintf(stderr, _("%s: waitpid: %s\n"), progname, strerror(errno)); return -1; } } while (0); /* mkswap returns: 0=suss, 1=error */ if (WIFEXITED(status) && WEXITSTATUS(status)==0) return 0; /* ok */ } return -1; /* error */ } int swap_detect_signature(const char *buf) { if ((memcmp(buf, "SWAP-SPACE", 10) == 0) || (memcmp(buf, "SWAPSPACE2", 10) == 0)) return 1; return 0; } /* return the pagesize the swap format has been built with * as swap metadata depends on the pagesize, we have to * reinitialize if it does not match with the current pagesize * returns 0 if not a valid swap format */ unsigned int swap_get_pagesize(const char *dev) { int fd; char *buf; unsigned int page, last_page = 0; unsigned int pagesize = 0; unsigned long long size, swap_size; int swap_version = 0; int flip = 0; int datasz; struct swap_header_v1_2 *s; struct stat sb; fd = open(dev, O_RDONLY); if (fd == -1) { perror("open"); return 0; } /* get size */ if (fstat(fd, &sb)) { perror("fstat"); goto err; } if (S_ISBLK(sb.st_mode)) { if (blkdev_get_size(fd, &size)) { perror("blkdev_get_size"); goto err; } } else size = sb.st_size; buf = malloc(MAX_PAGESIZE); if (!buf) { perror("malloc"); goto err; } datasz = read(fd, buf, MAX_PAGESIZE); if (datasz == (ssize_t) -1) { perror("read"); goto err1; } for (page = 0x1000; page <= MAX_PAGESIZE; page <<= 1) { /* skip 32k pagesize since this does not seem to * be supported */ if (page == 0x8000) continue; /* the smallest swap area is PAGE_SIZE*10, it means * 40k, that's less than MAX_PAGESIZE */ if (datasz < (page - 10)) break; if (swap_detect_signature(buf + page - 10)) { pagesize = page; break; } } if (pagesize) { s = (struct swap_header_v1_2 *)buf; if (s->version == 1) { swap_version = 1; last_page = s->last_page; } else if (swab32(s->version) == 1) { flip = 1; swap_version = 1; last_page = swab32(s->last_page); } if (verbose) fprintf(stderr, _("found %sswap v%d signature string" " for %d KiB PAGE_SIZE\n"), flip ? "other-endian " : "", swap_version, pagesize / 1024); swap_size = (last_page + 1) * pagesize; if (swap_size > size) { if (verbose) fprintf(stderr, _("last_page 0x%08llx is larger" " than actual size of swapspace\n"), (unsigned long long)swap_size); pagesize = 0; } } err1: free(buf); err: close(fd); return pagesize; } static int do_swapon(const char *orig_special, int prio, int canonic) { int status; int reinitialize = 0; struct stat st; const char *special = orig_special; unsigned int swap_pagesize = 0; if (verbose) printf(_("%s on %s\n"), progname, orig_special); if (!canonic) { special = fsprobe_get_devname(orig_special); if (!special) return cannot_find(orig_special); } if (stat(special, &st) < 0) { int errsv = errno; fprintf(stderr, _("%s: cannot stat %s: %s\n"), progname, special, strerror(errsv)); return -1; } swap_pagesize = swap_get_pagesize(special); if (swap_pagesize && (getpagesize() != swap_pagesize)) { if (verbose) fprintf(stderr, _("%s: %s: swap format pagesize does not match." " Reinitializing the swap.\n"), progname, special); reinitialize = 1; } /* We have to reinitialize swap with old (=useless) software suspend * data. The problem is that if we don't do it, then we get data * corruption the next time an attempt at unsuspending is made. */ if (swap_is_suspend(special)) { fprintf(stdout, _("%s: %s: software suspend data detected. " "Reinitializing the swap.\n"), progname, special); reinitialize = 1; } if (reinitialize) { if (swap_reinitialize(special) < 0) return -1; } /* people generally dislike this warning - now it is printed only when `verbose' is set */ if (verbose) { int permMask = (S_ISBLK(st.st_mode) ? 07007 : 07077); if ((st.st_mode & permMask) != 0) { fprintf(stderr, _("%s: warning: %s has " "insecure permissions %04o, " "%04o suggested\n"), progname, special, st.st_mode & 07777, ~permMask & 0666); } } /* test for holes by LBT */ if (S_ISREG(st.st_mode)) { if (st.st_blocks * 512 < st.st_size) { fprintf(stderr, _("%s: Skipping file %s - it appears " "to have holes.\n"), progname, special); return -1; } } { int flags = 0; #ifdef SWAP_FLAG_PREFER if (prio >= 0) { if (prio > SWAP_FLAG_PRIO_MASK) prio = SWAP_FLAG_PRIO_MASK; flags = SWAP_FLAG_PREFER | ((prio & SWAP_FLAG_PRIO_MASK) << SWAP_FLAG_PRIO_SHIFT); } #endif status = swapon(special, flags); } if (status < 0) { int errsv = errno; fprintf(stderr, "%s: %s: %s\n", progname, orig_special, strerror(errsv)); } return status; } static int cannot_find(const char *special) { fprintf(stderr, _("%s: cannot find the device for %s\n"), progname, special); return -1; } static int swapon_by_label(const char *label, int prio) { const char *special = fsprobe_get_devname_by_label(label); return special ? do_swapon(special, prio, CANONIC) : cannot_find(label); } static int swapon_by_uuid(const char *uuid, int prio) { const char *special = fsprobe_get_devname_by_uuid(uuid); return special ? do_swapon(special, prio, CANONIC) : cannot_find(uuid); } static int do_swapoff(const char *orig_special, int quiet, int canonic) { const char *special = orig_special; if (verbose) printf(_("%s on %s\n"), progname, orig_special); if (!canonic) { special = fsprobe_get_devname(orig_special); if (!special) return cannot_find(orig_special); } if (swapoff(special) == 0) return 0; /* success */ if (errno == EPERM) { fprintf(stderr, _("Not superuser.\n")); exit(1); /* any further swapoffs will also fail */ } if (!quiet || errno == ENOMEM) { fprintf(stderr, "%s: %s: %s\n", progname, orig_special, strerror(errno)); } return -1; } static int swapoff_by_label(const char *label, int quiet) { const char *special = fsprobe_get_devname_by_label(label); return special ? do_swapoff(special, quiet, CANONIC) : cannot_find(label); } static int swapoff_by_uuid(const char *uuid, int quiet) { const char *special = fsprobe_get_devname_by_uuid(uuid); return special ? do_swapoff(special, quiet, CANONIC) : cannot_find(uuid); } static int swapon_all(void) { FILE *fp; struct mntent *fstab; int status = 0; read_proc_swaps(); fp = setmntent(_PATH_MNTTAB, "r"); if (fp == NULL) { int errsv = errno; fprintf(stderr, _("%s: cannot open %s: %s\n"), progname, _PATH_MNTTAB, strerror(errsv)); exit(2); } while ((fstab = getmntent(fp)) != NULL) { const char *special; int skip = 0; int pri = priority; char *opt, *opts; if (!streq(fstab->mnt_type, MNTTYPE_SWAP)) continue; opts = strdup(fstab->mnt_opts); for (opt = strtok(opts, ","); opt != NULL; opt = strtok(NULL, ",")) { if (strncmp(opt, "pri=", 4) == 0) pri = atoi(opt+4); if (strcmp(opt, "noauto") == 0) skip = 1; } free(opts); if (skip) continue; special = fsprobe_get_devname(fstab->mnt_fsname); if (!special) { if (!ifexists) status |= cannot_find(fstab->mnt_fsname); continue; } if (!is_in_proc_swaps(special) && (!ifexists || !access(special, R_OK))) status |= do_swapon(special, pri, CANONIC); free((void *) special); } fclose(fp); return status; } static const char **llist = NULL; static int llct = 0; static const char **ulist = NULL; static int ulct = 0; static void addl(const char *label) { llist = (const char **) xrealloc(llist, (++llct) * sizeof(char *)); llist[llct-1] = label; } static void addu(const char *uuid) { ulist = (const char **) xrealloc(ulist, (++ulct) * sizeof(char *)); ulist[ulct-1] = uuid; } static int main_swapon(int argc, char *argv[]) { int status = 0; int c, i; while ((c = getopt_long(argc, argv, "ahep:svVL:U:", longswaponopts, NULL)) != -1) { switch (c) { case 'a': /* all */ ++all; break; case 'h': /* help */ swapon_usage(stdout, 0); break; case 'p': /* priority */ priority = atoi(optarg); break; case 'L': addl(optarg); break; case 'U': addu(optarg); break; case 'e': /* ifexists */ ifexists = 1; break; case 's': /* status report */ status = display_summary(); exit(status); case 'v': /* be chatty */ ++verbose; break; case 'V': /* version */ printf("%s: (%s)\n", progname, PACKAGE_STRING); exit(0); case 0: break; case '?': default: swapon_usage(stderr, 1); } } argv += optind; if (!all && !llct && !ulct && *argv == NULL) swapon_usage(stderr, 2); if (ifexists && (!all || strcmp(progname, "swapon"))) swapon_usage(stderr, 1); if (all) status |= swapon_all(); for (i = 0; i < llct; i++) status |= swapon_by_label(llist[i], priority); for (i = 0; i < ulct; i++) status |= swapon_by_uuid(ulist[i], priority); while (*argv != NULL) status |= do_swapon(*argv++, priority, !CANONIC); return status; } static int main_swapoff(int argc, char *argv[]) { FILE *fp; struct mntent *fstab; int status = 0; int c, i; while ((c = getopt_long(argc, argv, "ahvVL:U:", longswapoffopts, NULL)) != -1) { switch (c) { case 'a': /* all */ ++all; break; case 'h': /* help */ swapoff_usage(stdout, 0); break; case 'v': /* be chatty */ ++verbose; break; case 'V': /* version */ printf("%s (%s)\n", progname, PACKAGE_STRING); exit(0); case 'L': addl(optarg); break; case 'U': addu(optarg); break; case 0: break; case '?': default: swapoff_usage(stderr, 1); } } argv += optind; if (!all && !llct && !ulct && *argv == NULL) swapoff_usage(stderr, 2); /* * swapoff any explicitly given arguments. * Complain in case the swapoff call fails. */ for (i = 0; i < llct; i++) status |= swapoff_by_label(llist[i], !QUIET); for (i = 0; i < ulct; i++) status |= swapoff_by_uuid(ulist[i], !QUIET); while (*argv != NULL) status |= do_swapoff(*argv++, !QUIET, !CANONIC); if (all) { /* * In case /proc/swaps exists, unswap stuff listed there. * We are quiet but report errors in status. * Errors might mean that /proc/swaps * exists as ordinary file, not in procfs. * do_swapoff() exits immediately on EPERM. */ read_proc_swaps(); for(i=0; imnt_type, MNTTYPE_SWAP)) continue; special = fsprobe_get_devname(fstab->mnt_fsname); if (!special) continue; if (!is_in_proc_swaps(special)) do_swapoff(special, QUIET, CANONIC); } fclose(fp); } return status; } int main(int argc, char *argv[]) { char *p; setlocale(LC_ALL, ""); bindtextdomain(PACKAGE, LOCALEDIR); textdomain(PACKAGE); progname = argv[0]; p = strrchr(progname, '/'); if (p) progname = p+1; if (streq(progname, "swapon")) return main_swapon(argc, argv); else return main_swapoff(argc, argv); }