summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPádraig Brady <P@draigBrady.com>2013-11-04 23:14:11 +0000
committerPádraig Brady <P@draigBrady.com>2013-11-06 17:43:13 +0000
commit2c6736f92fc7d2f310714473ea84ceff57e01da2 (patch)
tree0eb4d044558effa82483537b5b58c8531d04691d
parentebaf961f3ad9ae7e8f9258e46eadb0e0d5e30ade (diff)
downloadcoreutils-2c6736f92fc7d2f310714473ea84ceff57e01da2.tar.xz
shred: write larger chunks when possible
* src/shred.c (dopass): When not needing to write periodic patterns, use a 64KiB block size to reduce the number of write system calls.
-rw-r--r--NEWS5
-rw-r--r--src/shred.c34
2 files changed, 32 insertions, 7 deletions
diff --git a/NEWS b/NEWS
index b4804a286..bb35583b6 100644
--- a/NEWS
+++ b/NEWS
@@ -87,8 +87,9 @@ GNU coreutils NEWS -*- outline -*-
Reservoir sampling is used to limit memory usage based on the number of
outputs, rather than the number of inputs.
- shred once again uses direct I/O where available.
- [Regression introduced in coreutils-6.0]
+ shred once again uses direct I/O where available, and increases write block
+ size from 12KiB to 64KiB when possible.
+ [Direct I/O regression introduced in coreutils-6.0]
split --line-bytes=SIZE, now only allocates memory as needed rather
than allocating SIZE bytes at program start.
diff --git a/src/shred.c b/src/shred.c
index 5c9765cd2..9ff723865 100644
--- a/src/shred.c
+++ b/src/shred.c
@@ -222,6 +222,25 @@ to be recovered later.\n\
exit (status);
}
+/*
+ * Determine if pattern type is periodic or not.
+ */
+static bool
+periodic_pattern (int type)
+{
+ if (type <= 0)
+ return false;
+
+ unsigned char r[3];
+ unsigned int bits = type & 0xfff;
+
+ bits |= bits << 12;
+ r[0] = (bits >> 4) & 255;
+ r[1] = (bits >> 8) & 255;
+ r[2] = bits & 255;
+
+ return (r[0] != r[1]) || (r[0] != r[2]);
+}
/*
* Fill a buffer with a fixed pattern.
@@ -361,9 +380,13 @@ dopass (int fd, char const *qname, off_t *sizep, int type,
/* Fill pattern buffer. Aligning it to a page so we can do direct I/O. */
size_t page_size = getpagesize ();
-#define OUTPUT_SIZE (12 * 1024)
+#define PERIODIC_OUTPUT_SIZE (12 * 1024)
+#define NONPERIODIC_OUTPUT_SIZE (64 * 1024)
+ verify (PERIODIC_OUTPUT_SIZE % 3 == 0);
+ size_t output_size = periodic_pattern (type)
+ ? PERIODIC_OUTPUT_SIZE : NONPERIODIC_OUTPUT_SIZE;
#define PAGE_ALIGN_SLOP (page_size - 1) /* So directio works */
-#define FILLPATTERN_SIZE (((OUTPUT_SIZE + 2) / 3) * 3) /* Multiple of 3 */
+#define FILLPATTERN_SIZE (((output_size + 2) / 3) * 3) /* Multiple of 3 */
#define PATTERNBUF_SIZE (PAGE_ALIGN_SLOP + FILLPATTERN_SIZE)
void *fill_pattern_mem = xmalloc (PATTERNBUF_SIZE);
unsigned char *pbuf = ptr_align (fill_pattern_mem, page_size);
@@ -408,8 +431,8 @@ dopass (int fd, char const *qname, off_t *sizep, int type,
while (true)
{
/* How much to write this time? */
- lim = OUTPUT_SIZE;
- if (0 <= size && size - offset < OUTPUT_SIZE)
+ lim = output_size;
+ if (0 <= size && size - offset < output_size)
{
if (size < offset)
break;
@@ -456,7 +479,8 @@ dopass (int fd, char const *qname, off_t *sizep, int type,
out. Thus, it shouldn't give up on bad blocks. This
code works because lim is always a multiple of
SECTOR_SIZE, except at the end. */
- verify (OUTPUT_SIZE % SECTOR_SIZE == 0);
+ verify (PERIODIC_OUTPUT_SIZE % SECTOR_SIZE == 0);
+ verify (NONPERIODIC_OUTPUT_SIZE % SECTOR_SIZE == 0);
if (errnum == EIO && 0 <= size && (soff | SECTOR_MASK) < lim)
{
size_t soff1 = (soff | SECTOR_MASK) + 1;