1 /* shred.c - Overwrite a file to securely delete
2  *
3  * Copyright 2014 Rob Landley <rob@landley.net>
4  *
5  * No standard
6 
7 USE_SHRED(NEWTOY(shred, "<1zxus#<1n#<1o#<0f", TOYFLAG_USR|TOYFLAG_BIN))
8 
9 config SHRED
10   bool "shred"
11   default y
12   help
13     usage: shred [-fuz] [-n COUNT] [-s SIZE] FILE...
14 
15     Securely delete a file by overwriting its contents with random data.
16 
17     -f        Force (chmod if necessary)
18     -n COUNT  Random overwrite iterations (default 1)
19     -o OFFSET Start at OFFSET
20     -s SIZE   Use SIZE instead of detecting file size
21     -u        unlink (actually delete file when done)
22     -x        Use exact size (default without -s rounds up to next 4k)
23     -z        zero at end
24 
25     Note: data journaling filesystems render this command useless, you must
26     overwrite all free space (fill up disk) to erase old data on those.
27 */
28 
29 #define FOR_shred
30 #include "toys.h"
31 
32 GLOBALS(
33   long offset;
34   long iterations;
35   long size;
36 
37   int ufd;
38 )
39 
40 void shred_main(void)
41 {
42   char **try;
43 
44   if (!(toys.optflags & FLAG_n)) TT.iterations++;
45   TT.ufd = xopenro("/dev/urandom");
46 
47   // We don't use loopfiles() here because "-" isn't stdin, and want to
48   // respond to files we can't open via chmod.
49 
50   for (try = toys.optargs; *try; try++) {
51     off_t pos = 0, len = TT.size;
52     int fd = open(*try, O_RDWR), iter = 0, throw;
53 
54     // do -f chmod if necessary
55     if (fd == -1 && (toys.optflags & FLAG_f)) {
56       chmod(*try, 0600);
57       fd = open(*try, O_RDWR);
58     }
59     if (fd == -1) {
60       perror_msg_raw(*try);
61       continue;
62     }
63 
64     // determine length
65     if (!len) len = fdlength(fd);
66     if (len<1) {
67       error_msg("%s: needs -s", *try);
68       close(fd);
69       continue;
70     }
71 
72     // Loop through, writing to this file
73     for (;;) {
74       // Advance to next -n or -z?
75 
76       if (pos >= len) {
77         pos = -1;
78         if (++iter == TT.iterations && (toys.optargs && FLAG_z)) {
79           memset(toybuf, 0, sizeof(toybuf));
80           continue;
81         }
82         if (iter >= TT.iterations) break;
83       }
84 
85       if (pos < TT.offset) {
86         if (TT.offset != lseek(fd, TT.offset, SEEK_SET)) {
87           perror_msg_raw(*try);
88           break;
89         }
90         pos = TT.offset;
91       }
92 
93       // Determine length, read random data if not zeroing, write.
94 
95       throw = sizeof(toybuf);
96       if (toys.optflags & FLAG_x)
97         if (len-pos < throw) throw = len-pos;
98 
99       if (iter != TT.iterations) xread(TT.ufd, toybuf, throw);
100       if (throw != writeall(fd, toybuf, throw)) perror_msg_raw(*try);
101       pos += throw;
102     }
103     if (toys.optflags & FLAG_u)
104       if (unlink(*try)) perror_msg("unlink '%s'", *try);
105   }
106 }
107