aboutsummaryrefslogtreecommitdiff
path: root/libdiskfs/io-prenotify.c
diff options
context:
space:
mode:
Diffstat (limited to 'libdiskfs/io-prenotify.c')
-rw-r--r--libdiskfs/io-prenotify.c72
1 files changed, 72 insertions, 0 deletions
diff --git a/libdiskfs/io-prenotify.c b/libdiskfs/io-prenotify.c
new file mode 100644
index 00000000..f88cb286
--- /dev/null
+++ b/libdiskfs/io-prenotify.c
@@ -0,0 +1,72 @@
+/*
+ Copyright (C) 1994, 1995, 1996 Free Software Foundation
+
+ 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 2, 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, write to the Free Software
+ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
+
+#include "priv.h"
+#include "io_S.h"
+
+/* Implement io_prenotify as described in <hurd/io.defs>.
+
+ We set the prenotify size to be the allocated size of the file;
+ then users are forced to call this routine before writing past
+ that, and we can do allocation (orreturn ENOSPC if necessary. */
+kern_return_t
+diskfs_S_io_prenotify (struct protid *cred,
+ vm_offset_t start __attribute__ ((unused)),
+ vm_offset_t end)
+{
+ struct node *np;
+ int err = 0;
+
+ if (!cred)
+ return EOPNOTSUPP;
+
+ np = cred->po->np;
+
+ /* Clamp it down */
+ mutex_lock (&np->lock);
+
+ if (!cred->mapped)
+ {
+ err = EINVAL;
+ goto out;
+ }
+
+ err = iohelp_verify_user_conch (&np->conch, cred);
+ if (err)
+ goto out;
+
+ iohelp_fetch_shared_data (cred);
+
+ if ((off_t) end < np->allocsize)
+ {
+ /* The user didn't need to do this, so we'll make sure they
+ have the right shared page info. */
+ spin_lock (&cred->mapped->lock);
+ iohelp_put_shared_data (cred);
+ spin_unlock (&cred->mapped->lock);
+ goto out;
+ }
+
+ err = diskfs_grow (np, end, cred);
+ if (diskfs_synchronous)
+ diskfs_node_update (np, 1);
+ if (!err && np->filemod_reqs)
+ diskfs_notice_filechange (np, FILE_CHANGED_EXTEND, 0, end);
+ out:
+ mutex_unlock (&np->lock);
+ return err;
+}