summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJim Meyering <meyering@redhat.com>2009-08-04 19:54:58 +0200
committerJim Meyering <meyering@redhat.com>2009-08-06 09:41:50 +0200
commit5929322ccb1f9d27c1b07b746d37419d17a7cbf6 (patch)
tree685ca3efe05ce619b33392ea499ad87c1c7a620c
parent365fb90eaae52c92d6af8e87769e72b872492f4a (diff)
downloadcoreutils-5929322ccb1f9d27c1b07b746d37419d17a7cbf6.tar.xz
dd: work around buffer length restrictions with oflag=direct (O_DIRECT)
dd oflag=direct would fail to copy a file with size that is not a multiple of 512 (destination file system specific) * NEWS (Bug fixes): Mention it. * src/dd.c (iwrite): Turn off O_DIRECT for any smaller-than-obs-sized write. Don't bother to restore it. * tests/dd/direct: New test for the above. * tests/Makefile.am (TESTS): Add dd/direct. * doc/coreutils.texi (dd invocation): Mention oflag=direct buffer size restriction. Details in http://thread.gmane.org/gmane.comp.gnu.coreutils.bugs/17586 Reported by Eric Sandeen.
-rw-r--r--NEWS3
-rw-r--r--doc/coreutils.texi4
-rw-r--r--src/dd.c10
-rw-r--r--tests/Makefile.am1
-rwxr-xr-xtests/dd/direct40
5 files changed, 57 insertions, 1 deletions
diff --git a/NEWS b/NEWS
index 0a0d4a719..c61f6667b 100644
--- a/NEWS
+++ b/NEWS
@@ -4,6 +4,9 @@ GNU coreutils NEWS -*- outline -*-
** Bug fixes
+ dd's oflag=direct option now works even when the size of the input
+ is not a multiple of e.g., 512 bytes.
+
install runs faster again with SELinux enabled
[introduced in coreutils-7.0]
diff --git a/doc/coreutils.texi b/doc/coreutils.texi
index acec76e4c..90a54b268 100644
--- a/doc/coreutils.texi
+++ b/doc/coreutils.texi
@@ -7861,6 +7861,10 @@ same time.
@opindex direct
@cindex direct I/O
Use direct I/O for data, avoiding the buffer cache.
+Note that the kernel may impose restrictions on read or write buffer sizes.
+For example, with an ext4 destination file system and a linux-based kernel,
+using @samp{oflag=direct} will cause writes to fail with @code{EINVAL} if the
+output buffer size is not a multiple of 512.
@item directory
@opindex directory
diff --git a/src/dd.c b/src/dd.c
index 9a9d22ae7..43ad7189f 100644
--- a/src/dd.c
+++ b/src/dd.c
@@ -837,6 +837,14 @@ iwrite (int fd, char const *buf, size_t size)
{
size_t total_written = 0;
+ if ((output_flags & O_DIRECT) && size < output_blocksize)
+ {
+ int old_flags = fcntl (STDOUT_FILENO, F_GETFL);
+ if (fcntl (STDOUT_FILENO, F_SETFL, old_flags & ~O_DIRECT) != 0)
+ error (0, errno, _("failed to turn off O_DIRECT: %s"),
+ quote (output_file));
+ }
+
while (total_written < size)
{
ssize_t nwritten;
@@ -1897,7 +1905,7 @@ main (int argc, char **argv)
|| S_ISDIR (stdout_stat.st_mode)
|| S_TYPEISSHM (&stdout_stat))
error (EXIT_FAILURE, ftruncate_errno,
- _("truncating at %"PRIuMAX" bytes in output file %s"),
+ _("failed to truncate to %"PRIuMAX" bytes in output file %s"),
size, quote (output_file));
}
}
diff --git a/tests/Makefile.am b/tests/Makefile.am
index ad19d0264..6ad61331c 100644
--- a/tests/Makefile.am
+++ b/tests/Makefile.am
@@ -298,6 +298,7 @@ TESTS = \
cp/src-base-dot \
cp/symlink-slash \
cp/thru-dangling \
+ dd/direct \
dd/misc \
dd/not-rewound \
dd/reblock \
diff --git a/tests/dd/direct b/tests/dd/direct
new file mode 100755
index 000000000..7e80bee15
--- /dev/null
+++ b/tests/dd/direct
@@ -0,0 +1,40 @@
+#!/bin/sh
+# ensure that dd's oflag=direct works
+
+# Copyright (C) 2009 Free Software Foundation, Inc.
+
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+
+# You should have received a copy of the GNU General Public License
+# along with this program. If not, see <http://www.gnu.org/licenses/>.
+
+if test "$VERBOSE" = yes; then
+ set -x
+ dd --version
+fi
+
+. $srcdir/test-lib.sh
+
+truncate -s 8192 in || framework_failure
+dd if=in oflag=direct of=out 2> /dev/null \
+ || skip_test_ 'this file system lacks support for O_DIRECT'
+
+truncate -s 511 short || framework_failure
+truncate -s 8191 m1 || framework_failure
+truncate -s 8193 p1 || framework_failure
+
+fail=0
+for i in short m1 p1; do
+ rm -f out
+ dd if=$i oflag=direct of=out || fail=1
+done
+
+Exit $fail