Re: 2.6.7-mm6

From: Christoph Hellwig
Date: Mon Jul 05 2004 - 05:46:06 EST


On Mon, Jul 05, 2004 at 11:18:04AM +0100, Christoph Hellwig wrote:
> > +use-llseek-instead-of-f_pos=-for-directory-seeking.patch
> >
> > Fix an nfsd problem when the client sends an insane directory offset.
>
> Please either use llseek() directly or renamed the thing to vfs_llseek()
> everywhere. Two names for exactly the same thing are a bad idea.
>
> (The latter sounds like the better idea to me)

Updated patch implementing my suggestion below:


--- 1.84/fs/nfsd/vfs.c 2004-06-01 11:27:57 +02:00
+++ edited/fs/nfsd/vfs.c 2004-07-05 14:37:24 +02:00
@@ -1477,10 +1477,12 @@
err = nfsd_open(rqstp, fhp, S_IFDIR, MAY_READ, &file);
if (err)
goto out;
- if (offset > ~(u32) 0)
- goto out_close;

- file.f_pos = offset;
+ offset = vfs_llseek(&file, offset, 0);
+ if (offset < 0) {
+ err = nfserrno((int)offset);
+ goto out_close;
+ }

/*
* Read the directory entries. This silly loop is necessary because
@@ -1496,7 +1498,7 @@
err = nfserrno(err);
else
err = cdp->err;
- *offsetp = file.f_pos;
+ *offsetp = vfs_llseek(&file, 0LL, 1);

if (err == nfserr_eof || err == nfserr_toosmall)
err = nfs_ok; /* can still be found in ->err */
--- 1.39/fs/read_write.c 2004-05-22 10:23:18 +02:00
+++ edited/fs/read_write.c 2004-07-05 14:38:14 +02:00
@@ -112,7 +112,7 @@

EXPORT_SYMBOL(default_llseek);

-static inline loff_t llseek(struct file *file, loff_t offset, int origin)
+inline loff_t vfs_llseek(struct file *file, loff_t offset, int origin)
{
loff_t (*fn)(struct file *, loff_t, int);

@@ -122,6 +122,8 @@
return fn(file, offset, origin);
}

+EXPORT_SYMBOL(vfs_llseek);
+
asmlinkage off_t sys_lseek(unsigned int fd, off_t offset, unsigned int origin)
{
off_t retval;
@@ -135,7 +137,7 @@

retval = -EINVAL;
if (origin <= 2) {
- loff_t res = llseek(file, offset, origin);
+ loff_t res = vfs_llseek(file, offset, origin);
retval = res;
if (res != (loff_t)retval)
retval = -EOVERFLOW; /* LFS: should only happen on 32 bit platforms */
@@ -165,7 +167,7 @@
if (origin > 2)
goto out_putf;

- offset = llseek(file, ((loff_t) offset_high << 32) | offset_low,
+ offset = vfs_llseek(file, ((loff_t) offset_high << 32) | offset_low,
origin);

retval = (int)offset;
--- 1.332/include/linux/fs.h 2004-07-02 07:23:47 +02:00
+++ edited/include/linux/fs.h 2004-07-05 14:41:03 +02:00
@@ -923,6 +923,7 @@
unsigned long, loff_t *);
extern ssize_t vfs_writev(struct file *, const struct iovec __user *,
unsigned long, loff_t *);
+extern loff_t vfs_llseek(struct file *file, loff_t offset, int origin);

/*
* NOTE: write_inode, delete_inode, clear_inode, put_inode can be called
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/