123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- /* vi: set sw=4 ts=4: */
- /*
- * Copyright (C) 2017 Denys Vlasenko <vda.linux@googlemail.com>
- *
- * Licensed under GPLv2, see file LICENSE in this source tree.
- */
- //config:config SHRED
- //config: bool "shred (4.9 kb)"
- //config: default y
- //config: help
- //config: Overwrite a file to hide its contents, and optionally delete it
- //applet:IF_SHRED(APPLET(shred, BB_DIR_USR_BIN, BB_SUID_DROP))
- //kbuild:lib-$(CONFIG_SHRED) += shred.o
- //usage:#define shred_trivial_usage
- //usage: "[-fuz] [-n N] [-s SIZE] FILE..."
- //usage:#define shred_full_usage "\n\n"
- //usage: "Overwrite/delete FILEs\n"
- //usage: "\n -f Chmod to ensure writability"
- //usage: "\n -s SIZE Size to write"
- //usage: "\n -n N Overwrite N times (default 3)"
- //usage: "\n -z Final overwrite with zeros"
- //usage: "\n -u Remove file"
- //-x (exact: don't round up to 4k) and -v (verbose) are accepted but have no effect
- /* shred (GNU coreutils) 8.25:
- -f, --force change permissions to allow writing if necessary
- -u truncate and remove file after overwriting
- -z, --zero add a final overwrite with zeros to hide shredding
- -n, --iterations=N overwrite N times instead of the default (3)
- -v, --verbose show progress
- -x, --exact do not round file sizes up to the next full block; this is the default for non-regular files
- --random-source=FILE get random bytes from FILE
- -s, --size=N shred this many bytes (suffixes like K, M, G accepted)
- --remove[=HOW] like -u but give control on HOW to delete; See below
- */
- #include "libbb.h"
- int shred_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
- int shred_main(int argc UNUSED_PARAM, char **argv)
- {
- char *opt_s;
- int rand_fd = rand_fd; /* for compiler */
- int zero_fd;
- unsigned num_iter = 3;
- unsigned opt;
- enum {
- OPT_f = (1 << 0),
- OPT_u = (1 << 1),
- OPT_z = (1 << 2),
- OPT_n = (1 << 3),
- OPT_v = (1 << 4),
- OPT_x = (1 << 5),
- OPT_s = (1 << 6),
- };
- opt = getopt32(argv, "^" "fuzn:+vxs:" "\0" "-1"/*min 1 arg*/, &num_iter, &opt_s);
- argv += optind;
- zero_fd = xopen("/dev/zero", O_RDONLY);
- if (num_iter != 0)
- rand_fd = xopen("/dev/urandom", O_RDONLY);
- for (;;) {
- struct stat sb;
- const char *fname;
- unsigned i;
- int fd;
- fname = *argv++;
- if (!fname)
- break;
- fd = -1;
- if (opt & OPT_f) {
- fd = open(fname, O_WRONLY);
- if (fd < 0)
- chmod(fname, 0666);
- }
- if (fd < 0)
- fd = xopen(fname, O_WRONLY);
- if (fstat(fd, &sb) == 0 && sb.st_size > 0) {
- off_t size = sb.st_size;
- if (opt & OPT_s) {
- size = BB_STRTOOFF(opt_s, NULL, 0); /* accepts oct/hex */
- if (errno || size < 0) bb_show_usage();
- }
- for (i = 0; i < num_iter; i++) {
- bb_copyfd_size(rand_fd, fd, size);
- fdatasync(fd);
- xlseek(fd, 0, SEEK_SET);
- }
- if (opt & OPT_z) {
- bb_copyfd_size(zero_fd, fd, size);
- fdatasync(fd);
- }
- }
- if (opt & OPT_u) {
- ftruncate(fd, 0);
- xunlink(fname);
- }
- xclose(fd);
- }
- return EXIT_SUCCESS;
- }
|