For O_DIRECT opens we're currently checking that the fs supports
O_DIRECT at write(2)-time.
This is a forward-port of Andrea's patch which moves the check to
open() time. Seems more sensible.
}
if (arg & O_DIRECT) {
+ if (inode->i_mapping && inode->i_mapping->a_ops) {
+ if (!inode->i_mapping->a_ops->direct_IO)
+ return -EINVAL;
+ }
+
/*
* alloc_kiovec() can sleep and we are only serialized by
* the big kernel lock here, so abuse the i_sem to serialize
}
f->f_flags &= ~(O_CREAT | O_EXCL | O_NOCTTY | O_TRUNC);
+ /* NB: we're sure to have correct a_ops only after f_op->open */
+ if (f->f_flags & O_DIRECT) {
+ error = -EINVAL;
+ if (inode->i_mapping && inode->i_mapping->a_ops)
+ if (!inode->i_mapping->a_ops->direct_IO)
+ goto cleanup_all;
+ }
+
return f;
cleanup_all:
retval = -EINVAL;
if ((offset & blocksize_mask) || (count & blocksize_mask))
goto out_free;
- if (!mapping->a_ops->direct_IO)
- goto out_free;
/*
* Flush to disk exclusively the _data_, metadata must remain