The RFC patch below adds a new flag for dup3().
(For simplicity, I'm just reusing O_EXCL for now.)

I am assuming calling f_op->flush() twice instead of just once
before closing the file is not harmful.

The idea is to extend dup3() so that we can catch close() errors
for the descriptor to be replaced, if we want to. Existing code
will not see any changes. (Userspace might wish to replace a
descriptor exactly because there is a problem, so we don't want
to make this unconditional.

This approach relies on close() having a single error path in Linux:
the file has an f_op->flush() handler, and it returns an error.
Here, if the flag is included in dup3(), f_op->flush() is called early
for newfd, and if it fails, dup3() will return with -EIO.

It will still be called again in filp_close(), but since the descriptor
is to be closed, there should be no activity between the two
f_op->flush() calls, so the latter should not fail.

Is there a reason (besides having yet another O_ flag) why this
would not work, or why we would not want to do this?
Am I missing something?
I just want be able to know if a problem was detected, that's all.

Best,
    Nominal Animal

diff -u5 -bar linux-3.16-rc3/fs/file.c linux-3.16-rc3.new/fs/file.c
--- linux-3.16-rc3/fs/file.c    2014-06-30 00:11:36.000000000 +0300
+++ linux-3.16-rc3.new/fs/file.c        2014-07-02 02:04:23.708114821 +0300
@@ -816,21 +816,31 @@
 SYSCALL_DEFINE3(dup3, unsigned int, oldfd, unsigned int, newfd, int, flags)
 {
        int err = -EBADF;
-       struct file *file;
+       struct file *file, *tofree;
        struct files_struct *files = current->files;
 
-       if ((flags & ~O_CLOEXEC) != 0)
+       if ((flags & ~(O_CLOEXEC | O_EXCL)) != 0)
                return -EINVAL;
 
        if (unlikely(oldfd == newfd))
                return -EINVAL;
 
        if (newfd >= rlimit(RLIMIT_NOFILE))
                return -EBADF;
 
        spin_lock(&files->file_lock);
+       if (unlikely(flags & O_EXCL)) {
+               tofree = fcheck(newfd);
+               if (tofree && file_count(tofree) && tofree->f_op->flush) {
+                       spin_unlock(&files->file_lock);
+                       err = tofree->f_op->flush(tofree, files);
+                       if (unlikely(err < 0))
+                               return -EIO;
+                       spin_lock(&files->file_lock);
+               }
+       }
        err = expand_files(files, newfd);
        file = fcheck(oldfd);
        if (unlikely(!file))
                goto Ebadf;
        if (unlikely(err < 0)) {
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to