blob: 69fb3e8ccf2df382a50ddfd152b441b20d7ab873 [file] [log] [blame]
Denys Vlasenko2f59bf32017-04-07 20:45:08 +02001/* vi: set sw=4 ts=4: */
2/*
3 * Copyright (C) 2017 Denys Vlasenko <vda.linux@googlemail.com>
4 *
5 * Licensed under GPLv2, see file LICENSE in this source tree.
6 */
7//config:config SHRED
Denys Vlasenkob097a842018-12-28 03:20:17 +01008//config: bool "shred (4.9 kb)"
Denys Vlasenko2f59bf32017-04-07 20:45:08 +02009//config: default y
10//config: help
Denys Vlasenko72089cf2017-07-21 09:50:55 +020011//config: Overwrite a file to hide its contents, and optionally delete it
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020012
13//applet:IF_SHRED(APPLET(shred, BB_DIR_USR_BIN, BB_SUID_DROP))
14
15//kbuild:lib-$(CONFIG_SHRED) += shred.o
16
17//usage:#define shred_trivial_usage
Denys Vlasenkoe4700042021-06-15 10:39:33 +020018//usage: "[-fuz] [-n N] FILE..."
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020019//usage:#define shred_full_usage "\n\n"
20//usage: "Overwrite/delete FILEs\n"
21//usage: "\n -f Chmod to ensure writability"
22//usage: "\n -n N Overwrite N times (default 3)"
23//usage: "\n -z Final overwrite with zeros"
24//usage: "\n -u Remove file"
25//-x and -v are accepted but have no effect
26
27/* shred (GNU coreutils) 8.25:
28-f, --force change permissions to allow writing if necessary
29-u truncate and remove file after overwriting
30-z, --zero add a final overwrite with zeros to hide shredding
31-n, --iterations=N overwrite N times instead of the default (3)
32-v, --verbose show progress
33-x, --exact do not round file sizes up to the next full block; this is the default for non-regular files
34--random-source=FILE get random bytes from FILE
35-s, --size=N shred this many bytes (suffixes like K, M, G accepted)
36--remove[=HOW] like -u but give control on HOW to delete; See below
37*/
38
39#include "libbb.h"
40
41int shred_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
42int shred_main(int argc UNUSED_PARAM, char **argv)
43{
44 int rand_fd = rand_fd; /* for compiler */
45 int zero_fd;
46 unsigned num_iter = 3;
47 unsigned opt;
48 enum {
49 OPT_f = (1 << 0),
50 OPT_u = (1 << 1),
51 OPT_z = (1 << 2),
52 OPT_n = (1 << 3),
53 OPT_v = (1 << 4),
54 OPT_x = (1 << 5),
55 };
56
Denys Vlasenkoe4700042021-06-15 10:39:33 +020057 opt = getopt32(argv, "^" "fuzn:+vx" "\0" "-1"/*min 1 arg*/, &num_iter);
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020058 argv += optind;
59
60 zero_fd = xopen("/dev/zero", O_RDONLY);
61 if (num_iter != 0)
62 rand_fd = xopen("/dev/urandom", O_RDONLY);
63
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020064 for (;;) {
65 struct stat sb;
Denys Vlasenko0f7f1ae2017-04-07 21:10:00 +020066 const char *fname;
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020067 unsigned i;
Denys Vlasenko0f7f1ae2017-04-07 21:10:00 +020068 int fd;
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020069
Denys Vlasenko0f7f1ae2017-04-07 21:10:00 +020070 fname = *argv++;
71 if (!fname)
72 break;
73 fd = -1;
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020074 if (opt & OPT_f) {
Denys Vlasenko0f7f1ae2017-04-07 21:10:00 +020075 fd = open(fname, O_WRONLY);
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020076 if (fd < 0)
Denys Vlasenko0f7f1ae2017-04-07 21:10:00 +020077 chmod(fname, 0666);
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020078 }
79 if (fd < 0)
Denys Vlasenko0f7f1ae2017-04-07 21:10:00 +020080 fd = xopen(fname, O_WRONLY);
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020081
82 if (fstat(fd, &sb) == 0 && sb.st_size > 0) {
83 off_t size = sb.st_size;
84
85 for (i = 0; i < num_iter; i++) {
86 bb_copyfd_size(rand_fd, fd, size);
87 fdatasync(fd);
88 xlseek(fd, 0, SEEK_SET);
89 }
90 if (opt & OPT_z) {
91 bb_copyfd_size(zero_fd, fd, size);
92 fdatasync(fd);
93 }
94 if (opt & OPT_u) {
95 ftruncate(fd, 0);
Denys Vlasenko0f7f1ae2017-04-07 21:10:00 +020096 xunlink(fname);
Denys Vlasenko2f59bf32017-04-07 20:45:08 +020097 }
98 xclose(fd);
99 }
Denys Vlasenko2f59bf32017-04-07 20:45:08 +0200100 }
101
102 return EXIT_SUCCESS;
103}