On Mon, Feb 07, 2005 at 05:48:44PM -0500, Trond Myklebust wrote: > Have you applied SGI's IRIX patches to your server (the one that makes > the cookies take 32-bit values)?
Can't. They're 6.4. And in any case it's 31-bits values glibc is limited to, which is the problem with the sgis (they're 32-bits unsigned). > Alternatively, you can forward-port the old 2.4.x cookie hack to 2.6.x > (that should be fairly trivial to do). You can find the patch on > > http://client.linux-nfs.org/Linux-2.4.x/2.4.26/linux-2.4.26-02-seekdir.dif Made a new version that's less hackish, with an irix32 mount flag to handle the case. With that patch, a mount -o irix32 and -32bitclient on the sgi side everything works as it should. Specifically, what the irix32 flag does is: - convert the getdents64 d_off entries from unsigned 32 bits as the nfs server sends them to signed 32 bits as glibc wants. Side effect, the directory filp->f_pos is signed 32bits too. It does not touch the cached nfs server replies which are pristine. - accept _llseek with negative positions and convert them back to 32-bits unsigned when needed Using a mount flag ensures no damage is done in mixed environments and also allows someday glibc to handle 64-bits cookies correctly. If the patch looks like it's going to be accepted, I'll send the (trivial) mount changes to whoever the util-linux maintainer is. And does someone happens to know who maintains the mount/nfs options man page? It may fuzz a little, fc3 has a minor patch in another part of dir.c. OG. Signed-off-by: Olivier Galibert <[EMAIL PROTECTED]> fs/nfs/dir.c | 53 +++++++++++++++++++++++++++++++++++++++++----- fs/nfs/inode.c | 1 include/linux/nfs_mount.h | 3 +- 3 files changed, 51 insertions(+), 6 deletions(-) --- ../save/kernel-2.6.10/linux-2.6.10/fs/nfs/dir.c 2005-02-08 13:49:13.000000000 +0100 +++ linux-2.6.10/fs/nfs/dir.c 2005-02-08 22:27:35.000000000 +0100 @@ -31,6 +31,7 @@ #include <linux/pagemap.h> #include <linux/smp_lock.h> #include <linux/namei.h> +#include <linux/mount.h> #include "delegation.h" @@ -38,6 +39,7 @@ /* #define NFS_DEBUG_VERBOSE 1 */ static int nfs_opendir(struct inode *, struct file *); +static loff_t nfs_dir_llseek(struct file *, loff_t, int); static int nfs_readdir(struct file *, void *, filldir_t); static struct dentry *nfs_lookup(struct inode *, struct dentry *, struct nameidata *); static int nfs_cached_lookup(struct inode *, struct dentry *, @@ -54,6 +56,7 @@ static int nfs_fsync_dir(struct file *, struct dentry *, int); struct file_operations nfs_dir_operations = { + .llseek = nfs_dir_llseek, .read = generic_read_dir, .readdir = nfs_readdir, .open = nfs_opendir, @@ -96,6 +99,7 @@ #endif /* CONFIG_NFS_V4 */ + /* * Open file */ @@ -112,6 +116,33 @@ return res; } + +/* + * Allow seeking to negative positions in the irix32 case + */ +static loff_t nfs_dir_llseek(struct file *file, loff_t offset, int origin) +{ + switch (origin) { + case 1: + if (offset == 0) { + offset = file->f_pos; + break; + } + case 2: + return -EINVAL; + } + + if (offset < 0 && !(NFS_SB(file->f_vfsmnt->mnt_sb)->flags & NFS_MOUNT_IRIX32)) + return -EINVAL; + + if (offset != file->f_pos) { + file->f_pos = offset; + file->f_version = 0; + } + return (offset <= 0) ? 0 : offset; +} + + typedef u32 * (*decode_dirent_t)(u32 *, struct nfs_entry *, int); typedef struct { struct file *file; @@ -324,7 +355,7 @@ unsigned long fileid; int loop_count = 0, res; - + int irix32 = NFS_SB(file->f_vfsmnt->mnt_sb)->flags & NFS_MOUNT_IRIX32; dfprintk(VFS, "NFS: nfs_do_filldir() filling starting @ cookie %Lu\n", (long long)desc->target); for(;;) { @@ -337,11 +368,19 @@ if (desc->plus && (entry->fattr->valid & NFS_ATTR_FATTR)) d_type = nfs_type_to_d_type(entry->fattr->type); - res = filldir(dirent, entry->name, entry->len, - entry->prev_cookie, fileid, d_type); + if (!irix32) + res = filldir(dirent, entry->name, entry->len, + entry->prev_cookie, fileid, d_type); + else + res = filldir(dirent, entry->name, entry->len, + (s32)entry->prev_cookie, fileid, d_type); if (res < 0) break; - file->f_pos = desc->target = entry->cookie; + desc->target = entry->cookie; + if (!irix32) + file->f_pos = entry->cookie; + else + file->f_pos = (s32)entry->cookie; if (dir_decode(desc) != 0) { desc->page_index ++; break; @@ -429,6 +468,7 @@ struct nfs_fh fh; struct nfs_fattr fattr; long res; + int irix32 = NFS_SB(filp->f_vfsmnt->mnt_sb)->flags & NFS_MOUNT_IRIX32; lock_kernel(); @@ -447,7 +487,10 @@ memset(desc, 0, sizeof(*desc)); desc->file = filp; - desc->target = filp->f_pos; + if (!irix32) + desc->target = filp->f_pos; + else + desc->target = (u32)filp->f_pos; desc->decode = NFS_PROTO(inode)->decode_dirent; desc->plus = NFS_USE_READDIRPLUS(inode); diff -ru ../save/kernel-2.6.10/linux-2.6.10/fs/nfs/inode.c linux-2.6.10/fs/nfs/inode.c --- ../save/kernel-2.6.10/linux-2.6.10/fs/nfs/inode.c 2004-12-24 22:35:24.000000000 +0100 +++ linux-2.6.10/fs/nfs/inode.c 2005-02-08 17:55:30.000000000 +0100 @@ -524,6 +524,7 @@ { NFS_MOUNT_NOAC, ",noac", "" }, { NFS_MOUNT_NONLM, ",nolock", ",lock" }, { NFS_MOUNT_BROKEN_SUID, ",broken_suid", "" }, + { NFS_MOUNT_IRIX32, ",irix32", "" }, { 0, NULL, NULL } }; struct proc_nfs_info *nfs_infop; diff -ru ../save/kernel-2.6.10/linux-2.6.10/include/linux/nfs_mount.h linux-2.6.10/include/linux/nfs_mount.h --- ../save/kernel-2.6.10/linux-2.6.10/include/linux/nfs_mount.h 2004-12-24 22:33:49.000000000 +0100 +++ linux-2.6.10/include/linux/nfs_mount.h 2005-02-08 17:54:22.000000000 +0100 @@ -20,7 +20,7 @@ * mount-to-kernel version compatibility. Some of these aren't used yet * but here they are anyway. */ -#define NFS_MOUNT_VERSION 6 +#define NFS_MOUNT_VERSION 7 #define NFS_MAX_CONTEXT_LEN 256 struct nfs_mount_data { @@ -60,6 +60,7 @@ #define NFS_MOUNT_BROKEN_SUID 0x0400 /* 4 */ #define NFS_MOUNT_STRICTLOCK 0x1000 /* reserved for NFSv4 */ #define NFS_MOUNT_SECFLAVOUR 0x2000 /* 5 */ +#define NFS_MOUNT_IRIX32 0x4000 /* 7 */ #define NFS_MOUNT_FLAGMASK 0xFFFF #endif - To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to [EMAIL PROTECTED] More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/