/* * linux/fs/nfs/dir.c * * Copyright (C) 1992 Rick Sladkey * * nfs directory handling functions * * 10 Apr 1996 Added silly rename for unlink --okir * 28 Sep 1996 Improved directory cache --okir */ #include #include #include #include #include #include #include #include #include #include #include /* for fs functions */ /* * Head for a dircache entry. Currently still very simple; when * the cache grows larger, we will need a LRU list. */ struct nfs_dirent { dev_t dev; /* device number */ ino_t ino; /* inode number */ u32 cookie; /* cooke of first entry */ unsigned short valid : 1, /* data is valid */ locked : 1; /* entry locked */ unsigned int size; /* # of entries */ unsigned long age; /* last used */ unsigned long mtime; /* last attr stamp */ struct wait_queue * wait; struct nfs_entry * entry; }; static int nfs_dir_open(struct inode * inode, struct file * file); static long nfs_dir_read(struct inode *, struct file *, char *, unsigned long); static int nfs_readdir(struct inode *, struct file *, void *, filldir_t); static int nfs_lookup(struct inode *, const char *, int, struct inode **); static int nfs_create(struct inode *, const char *, int, int, struct inode **); static int nfs_mkdir(struct inode *, const char *, int, int); static int nfs_rmdir(struct inode *, const char *, int); static int nfs_unlink(struct inode *, const char *, int); static int nfs_symlink(struct inode *, const char *, int, const char *); static int nfs_link(struct inode *, struct inode *, const char *, int); static int nfs_mknod(struct inode *, const char *, int, int, int); static int nfs_rename(struct inode *, const char *, int, struct inode *, const char *, int, int); static struct file_operations nfs_dir_operations = { NULL, /* lseek - default */ nfs_dir_read, /* read - bad */ NULL, /* write - bad */ nfs_readdir, /* readdir */ NULL, /* select - default */ NULL, /* ioctl - default */ NULL, /* mmap */ nfs_dir_open, /* open - revalidate */ NULL, /* no special release code */ NULL /* fsync */ }; struct inode_operations nfs_dir_inode_operations = { &nfs_dir_operations, /* default directory file-ops */ nfs_create, /* create */ nfs_lookup, /* lookup */ nfs_link, /* link */ nfs_unlink, /* unlink */ nfs_symlink, /* symlink */ nfs_mkdir, /* mkdir */ nfs_rmdir, /* rmdir */ nfs_mknod, /* mknod */ nfs_rename, /* rename */ NULL, /* readlink */ NULL, /* follow_link */ NULL, /* readpage */ NULL, /* writepage */ NULL, /* bmap */ NULL, /* truncate */ NULL, /* permission */ NULL, /* smap */ NULL, /* updatepage */ nfs_revalidate, /* revalidate */ }; static int nfs_dir_open(struct inode *dir, struct file *file) { dfprintk(VFS, "NFS: nfs_dir_open(%x/%ld)\n", dir->i_dev, dir->i_ino); return nfs_revalidate_inode(NFS_SERVER(dir), dir); } static long nfs_dir_read(struct inode *inode, struct file *filp, char *buf, unsigned long count) { return -EISDIR; } static struct nfs_dirent dircache[NFS_MAX_DIRCACHE]; /* * We need to do caching of directory entries to prevent an * incredible amount of RPC traffic. Only the most recent open * directory is cached. This seems sufficient for most purposes. * Technically, we ought to flush the cache on close but this is * not a problem in practice. * * XXX: Do proper directory caching by stuffing data into the * page cache (may require some fiddling for rsize < PAGE_SIZE). */ static int nfs_readdir(struct inode *inode, struct file *filp, void *dirent, filldir_t filldir) { static struct wait_queue *readdir_wait = NULL; struct wait_queue **waitp = NULL; struct nfs_dirent *cache, *free; struct nfs_entry *entry; unsigned long age, dead; u32 cookie; int ismydir, result; int i, j, index = 0; dfprintk(VFS, "NFS: nfs_readdir(%x/%ld)\n", inode->i_dev, inode->i_ino); if (!inode || !S_ISDIR(inode->i_mode)) { printk("nfs_readdir: inode is NULL or not a directory\n"); return -EBADF; } if ((result = nfs_revalidate_inode(NFS_SERVER(inode), inode)) < 0) return result; /* * Try to find the entry in the cache */ again: if (waitp) { interruptible_sleep_on(waitp); if (current->signal & ~current->blocked) return -ERESTARTSYS; waitp = NULL; } cookie = filp->f_pos; entry = NULL; free = NULL; age = ~(unsigned long) 0; dead = jiffies - NFS_ATTRTIMEO(inode); for (i = 0, cache = dircache; i < NFS_MAX_DIRCACHE; i++, cache++) { /* dprintk("NFS: dircache[%d] valid %d locked %d\n", i, cache->valid, cache->locked); */ ismydir = (cache->dev == inode->i_dev && cache->ino == inode->i_ino); if (cache->locked) { if (!ismydir || cache->cookie != cookie) continue; dfprintk(DIRCACHE, "NFS: waiting on dircache entry\n"); waitp = &cache->wait; goto again; } if (ismydir && cache->mtime != NFS_OLDMTIME(inode)) cache->valid = 0; if (!cache->valid || cache->age < dead) { free = cache; age = 0; } else if (cache->age < age) { free = cache; age = cache->age; } if (!ismydir || !cache->valid) continue; if (cache->cookie == cookie && cache->size > 0) { entry = cache->entry + (index = 0); cache->locked = 1; break; } for (j = 0; j < cache->size; j++) { /* dprintk("NFS: examing entry %.*s @%d\n", (int) cache->entry[j].length, cache->entry[j].name, cache->entry[j].cookie); */ if (cache->entry[j].cookie != cookie) continue; if (j < cache->size - 1) { entry = cache->entry + (index = j + 1); } else if (cache->entry[j].eof) { return 0; } break; } if (entry) { dfprintk(DIRCACHE, "NFS: found dircache entry %d\n", (int)(cache - dircache)); cache->locked = 1; break; } } /* * Okay, entry not present in cache, or locked and inaccessible. * Set up the cache entry and attempt a READDIR call. */ if (entry == NULL) { if ((cache = free) == NULL) { dfprintk(DIRCACHE, "NFS: dircache contention\n"); waitp = &readdir_wait; goto again; } dfprintk(DIRCACHE, "NFS: using free dircache entry %d\n", (int)(free - dircache)); cache->cookie = cookie; cache->locked = 1; cache->valid = 0; cache->dev = inode->i_dev; cache->ino = inode->i_ino; if (!cache->entry) { cache->entry = (struct nfs_entry *) get_free_page(GFP_KERNEL); if (!cache->entry) { result = -ENOMEM; goto done; } } result = nfs_proc_readdir(NFS_SERVER(inode), NFS_FH(inode), cookie, PAGE_SIZE, cache->entry); if (result <= 0) goto done; cache->size = result; cache->valid = 1; entry = cache->entry + (index = 0); } cache->mtime = NFS_OLDMTIME(inode); cache->age = jiffies; /* * Yowza! We have a cache entry... */ while (index < cache->size) { int nextpos = entry->cookie; /* dprintk("NFS: filldir(%p, %.*s, %d, %d, %x, eof %x)\n", entry, (int) entry->length, entry->name, entry->length, (unsigned int) filp->f_pos, entry->fileid, entry->eof); */ if (filldir(dirent, entry->name, entry->length, cookie, entry->fileid) < 0) break; cookie = nextpos; if (nextpos != entry->cookie) { printk("nfs_readdir: shouldn't happen!\n"); break; } index++; entry++; } filp->f_pos = cookie; result = 0; /* XXX: May want to kick async readdir-ahead here. Not too hard * to do. */ done: dfprintk(DIRCACHE, "NFS: nfs_readdir complete\n"); cache->locked = 0; wake_up(&cache->wait); wake_up(&readdir_wait); return result; } /* * Invalidate dircache entries for inode */ void nfs_invalidate_dircache(struct inode *inode) { struct nfs_dirent *cache; dev_t dev = inode->i_dev; ino_t ino = inode->i_ino; int i; dfprintk(DIRCACHE, "NFS: invalidate dircache for %x/%ld\n", dev, (long)ino); for (i = 0, cache = dircache; i < NFS_MAX_DIRCACHE; i++, cache++) { if (!cache->locked && cache->dev == dev && cache->ino == ino) cache->valid = 0; /* brute force */ } } /* * Free directory cache memory * Called from cleanup_module */ void nfs_free_dircache(void) { struct nfs_dirent *cache; int i; dfprintk(DIRCACHE, "NFS: freeing dircache\n"); for (i = 0, cache = dircache; i < NFS_MAX_DIRCACHE; i++, cache++) { cache->valid = 0; if (cache->locked) { printk("nfs_kfree_cache: locked entry in dircache!\n"); continue; } if (cache->entry) free_page((unsigned long) cache->entry); cache->entry = NULL; } } /* * Lookup caching is a big win for performance but this is just * a trial to see how well it works on a small scale. * For example, bash does a lookup on ".." 13 times for each path * element when running pwd. Yes, hard to believe but true. * Try pwd in a filesystem mounted with noac. * * It trades a little cpu time and memory for a lot of network bandwidth. * Since the cache is not hashed yet, it is a good idea not to make it too * large because every lookup looks through the entire cache even * though most of them will fail. * * FIXME: The lookup cache should also cache failed lookups. This can * be a considerable win on diskless clients. */ static struct nfs_lookup_cache_entry { kdev_t dev; ino_t inode; char filename[NFS_MAXNAMLEN + 1]; struct nfs_fh fhandle; struct nfs_fattr fattr; int expiration_date; } nfs_lookup_cache[NFS_LOOKUP_CACHE_SIZE]; static struct nfs_lookup_cache_entry *nfs_lookup_cache_index(struct inode *dir, const char *filename) { struct nfs_lookup_cache_entry *entry; int i; for (i = 0; i < NFS_LOOKUP_CACHE_SIZE; i++) { entry = nfs_lookup_cache + i; if (entry->dev == dir->i_dev && entry->inode == dir->i_ino && !strncmp(filename, entry->filename, NFS_MAXNAMLEN)) return entry; } return NULL; } static int nfs_lookup_cache_lookup(struct inode *dir, const char *filename, struct nfs_fh *fhandle, struct nfs_fattr *fattr) { static int nfs_lookup_cache_in_use = 0; struct nfs_lookup_cache_entry *entry; dfprintk(LOOKUPCACHE, "NFS: lookup_cache_lookup(%x/%ld, %s)\n", dir->i_dev, dir->i_ino, filename); if (!nfs_lookup_cache_in_use) { memset(nfs_lookup_cache, 0, sizeof(nfs_lookup_cache)); nfs_lookup_cache_in_use = 1; } if ((entry = nfs_lookup_cache_index(dir, filename))) { if (jiffies > entry->expiration_date) { entry->dev = 0; return 0; } *fhandle = entry->fhandle; *fattr = entry->fattr; return 1; } return 0; } static void nfs_lookup_cache_add(struct inode *dir, const char *filename, struct nfs_fh *fhandle, struct nfs_fattr *fattr) { static int nfs_lookup_cache_pos = 0; struct nfs_lookup_cache_entry *entry; dfprintk(LOOKUPCACHE, "NFS: lookup_cache_add(%x/%ld, %s\n", dir->i_dev, dir->i_ino, filename); /* compensate for bug in SGI NFS server */ if (fattr->size == -1 || fattr->uid == -1 || fattr->gid == -1 || fattr->atime.seconds == -1 || fattr->mtime.seconds == -1) return; if (!(entry = nfs_lookup_cache_index(dir, filename))) { entry = nfs_lookup_cache + nfs_lookup_cache_pos++; if (nfs_lookup_cache_pos == NFS_LOOKUP_CACHE_SIZE) nfs_lookup_cache_pos = 0; } entry->dev = dir->i_dev; entry->inode = dir->i_ino; strcpy(entry->filename, filename); entry->fhandle = *fhandle; entry->fattr = *fattr; entry->expiration_date = jiffies + (S_ISDIR(fattr->mode) ? NFS_SERVER(dir)->acdirmin : NFS_SERVER(dir)->acregmin); } static void nfs_lookup_cache_remove(struct inode *dir, struct inode *inode, const char *filename) { struct nfs_lookup_cache_entry *entry; kdev_t dev; ino_t fileid; int i; if (inode) { dev = inode->i_dev; fileid = inode->i_ino; } else if ((entry = nfs_lookup_cache_index(dir, filename))) { dev = entry->dev; fileid = entry->fattr.fileid; } else return; dfprintk(LOOKUPCACHE, "NFS: lookup_cache_remove(%x/%ld)\n", dev, (long)fileid); for (i = 0; i < NFS_LOOKUP_CACHE_SIZE; i++) { entry = nfs_lookup_cache + i; if (entry->dev == dev && entry->fattr.fileid == fileid) entry->dev = 0; } } static void nfs_lookup_cache_refresh(struct inode *file, struct nfs_fattr *fattr) { struct nfs_lookup_cache_entry *entry; kdev_t dev = file->i_dev; int fileid = file->i_ino; int i; for (i = 0; i < NFS_LOOKUP_CACHE_SIZE; i++) { entry = nfs_lookup_cache + i; if (entry->dev == dev && entry->fattr.fileid == fileid) entry->fattr = *fattr; } } static int nfs_lookup(struct inode *dir, const char *__name, int len, struct inode **result) { struct nfs_fh fhandle; struct nfs_fattr fattr; char name[len > NFS_MAXNAMLEN? 1 : len+1]; int error; dfprintk(VFS, "NFS: lookup(%x/%ld, %.*s)\n", dir->i_dev, dir->i_ino, len, __name); *result = NULL; if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_lookup: inode is NULL or not a directory\n"); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(dir); return -ENAMETOOLONG; } memcpy(name,__name,len); name[len] = '\0'; if (len == 1 && name[0] == '.') { /* cheat for "." */ *result = dir; return 0; } if ((NFS_SERVER(dir)->flags & NFS_MOUNT_NOAC) || !nfs_lookup_cache_lookup(dir, name, &fhandle, &fattr)) { if ((error = nfs_proc_lookup(NFS_SERVER(dir), NFS_FH(dir), name, &fhandle, &fattr))) { iput(dir); return error; } nfs_lookup_cache_add(dir, name, &fhandle, &fattr); } if (!(*result = nfs_fhget(dir->i_sb, &fhandle, &fattr))) { iput(dir); return -EACCES; } iput(dir); return 0; } static int nfs_create(struct inode *dir, const char *name, int len, int mode, struct inode **result) { struct nfs_sattr sattr; struct nfs_fattr fattr; struct nfs_fh fhandle; int error; dfprintk(VFS, "NFS: create(%x/%ld, %s\n", dir->i_dev, dir->i_ino, name); *result = NULL; if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_create: inode is NULL or not a directory\n"); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(dir); return -ENAMETOOLONG; } sattr.mode = mode; sattr.uid = sattr.gid = sattr.size = (unsigned) -1; sattr.atime.seconds = sattr.mtime.seconds = (unsigned) -1; if ((error = nfs_proc_create(NFS_SERVER(dir), NFS_FH(dir), name, &sattr, &fhandle, &fattr))) { iput(dir); return error; } if (!(*result = nfs_fhget(dir->i_sb, &fhandle, &fattr))) { iput(dir); return -EACCES; } nfs_lookup_cache_add(dir, name, &fhandle, &fattr); nfs_invalidate_dircache(dir); iput(dir); return 0; } static int nfs_mknod(struct inode *dir, const char *name, int len, int mode, int rdev) { struct nfs_sattr sattr; struct nfs_fattr fattr; struct nfs_fh fhandle; int error; dfprintk(VFS, "NFS: mknod(%x/%ld, %s\n", dir->i_dev, dir->i_ino, name); if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_mknod: inode is NULL or not a directory\n"); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(dir); return -ENAMETOOLONG; } sattr.mode = mode; sattr.uid = sattr.gid = (unsigned) -1; if (S_ISCHR(mode) || S_ISBLK(mode)) sattr.size = rdev; /* get out your barf bag */ else sattr.size = (unsigned) -1; sattr.atime.seconds = sattr.mtime.seconds = (unsigned) -1; error = nfs_proc_create(NFS_SERVER(dir), NFS_FH(dir), name, &sattr, &fhandle, &fattr); if (!error) nfs_lookup_cache_add(dir, name, &fhandle, &fattr); nfs_invalidate_dircache(dir); iput(dir); return error; } static int nfs_mkdir(struct inode *dir, const char *name, int len, int mode) { struct nfs_sattr sattr; struct nfs_fattr fattr; struct nfs_fh fhandle; int error; dfprintk(VFS, "NFS: mkdir(%x/%ld, %s\n", dir->i_dev, dir->i_ino, name); if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_mkdir: inode is NULL or not a directory\n"); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(dir); return -ENAMETOOLONG; } sattr.mode = mode; sattr.uid = sattr.gid = sattr.size = (unsigned) -1; sattr.atime.seconds = sattr.mtime.seconds = (unsigned) -1; error = nfs_proc_mkdir(NFS_SERVER(dir), NFS_FH(dir), name, &sattr, &fhandle, &fattr); if (!error) nfs_lookup_cache_add(dir, name, &fhandle, &fattr); nfs_invalidate_dircache(dir); iput(dir); return error; } static int nfs_rmdir(struct inode *dir, const char *name, int len) { int error; dfprintk(VFS, "NFS: rmdir(%x/%ld, %s\n", dir->i_dev, dir->i_ino, name); if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_rmdir: inode is NULL or not a directory\n"); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(dir); return -ENAMETOOLONG; } error = nfs_proc_rmdir(NFS_SERVER(dir), NFS_FH(dir), name); if (!error) nfs_lookup_cache_remove(dir, NULL, name); nfs_invalidate_dircache(dir); iput(dir); return error; } static int nfs_sillyrename(struct inode *dir, const char *name, int len) { struct inode *inode; char silly[16]; int slen, ret; dir->i_count++; if (nfs_lookup(dir, name, len, &inode) < 0) return -EIO; /* arbitrary */ if (inode->i_count == 1) { iput(inode); return -EIO; } if (NFS_RENAMED_DIR(inode)) { iput(NFS_RENAMED_DIR(inode)); NFS_RENAMED_DIR(inode) = NULL; iput(inode); return -EIO; } slen = sprintf(silly, ".nfs%ld", inode->i_ino); if (len == slen && !strncmp(name, silly, len)) { iput(inode); return -EIO; /* DWIM */ } dfprintk(VFS, "NFS: sillyrename(%x/%ld, %s)\n", dir->i_dev, dir->i_ino, name); ret = nfs_proc_rename(NFS_SERVER(dir), NFS_FH(dir), name, NFS_FH(dir), silly); if (ret >= 0) { nfs_lookup_cache_remove(dir, NULL, name); nfs_lookup_cache_remove(dir, NULL, silly); NFS_RENAMED_DIR(inode) = dir; dir->i_count++; } nfs_invalidate_dircache(dir); iput(inode); return ret; } /* * When releasing the inode, finally remove any unlinked but open files. * Note that we have to clear the set of pending signals temporarily; * otherwise the RPC call will fail. */ void nfs_sillyrename_cleanup(struct inode *inode) { unsigned long oldsig; struct inode *dir = NFS_RENAMED_DIR(inode); char silly[14]; int error, slen; dfprintk(VFS, "NFS: sillyrename cleanup(%x/%ld)\n", inode->i_dev, inode->i_ino); oldsig = current->signal; current->signal = 0; slen = sprintf(silly, ".nfs%ld", inode->i_ino); error = nfs_proc_remove(NFS_SERVER(dir), NFS_FH(dir), silly); if (error < 0) printk("NFS: silly_rename cleanup failed (err %d)\n", -error); nfs_lookup_cache_remove(dir, NULL, silly); nfs_invalidate_dircache(dir); NFS_RENAMED_DIR(inode) = NULL; iput(dir); current->signal |= oldsig; } static int nfs_unlink(struct inode *dir, const char *name, int len) { int error; dfprintk(VFS, "NFS: unlink(%x/%ld, %s)\n", dir->i_dev, dir->i_ino, name); if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_unlink: inode is NULL or not a directory\n"); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(dir); return -ENAMETOOLONG; } if ((error = nfs_sillyrename(dir, name, len)) < 0) { error = nfs_proc_remove(NFS_SERVER(dir), NFS_FH(dir), name); if (!error) nfs_lookup_cache_remove(dir, NULL, name); } nfs_invalidate_dircache(dir); iput(dir); return error; } static int nfs_symlink(struct inode *dir, const char *name, int len, const char *symname) { struct nfs_sattr sattr; int error; dfprintk(VFS, "NFS: symlink(%x/%ld, %s, %s)\n", dir->i_dev, dir->i_ino, name, symname); if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_symlink: inode is NULL or not a directory\n"); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(dir); return -ENAMETOOLONG; } if (strlen(symname) > NFS_MAXPATHLEN) { iput(dir); return -ENAMETOOLONG; } sattr.mode = S_IFLNK | S_IRWXUGO; /* SunOS 4.1.2 crashes without this! */ sattr.uid = sattr.gid = sattr.size = (unsigned) -1; sattr.atime.seconds = sattr.mtime.seconds = (unsigned) -1; error = nfs_proc_symlink(NFS_SERVER(dir), NFS_FH(dir), name, symname, &sattr); nfs_invalidate_dircache(dir); iput(dir); return error; } static int nfs_link(struct inode *oldinode, struct inode *dir, const char *name, int len) { int error; dfprintk(VFS, "NFS: link(%x/%ld -> %x/%ld, %s)\n", oldinode->i_dev, oldinode->i_ino, dir->i_dev, dir->i_ino, name); if (!oldinode) { printk("nfs_link: old inode is NULL\n"); iput(oldinode); iput(dir); return -ENOENT; } if (!dir || !S_ISDIR(dir->i_mode)) { printk("nfs_link: dir is NULL or not a directory\n"); iput(oldinode); iput(dir); return -ENOENT; } if (len > NFS_MAXNAMLEN) { iput(oldinode); iput(dir); return -ENAMETOOLONG; } error = nfs_proc_link(NFS_SERVER(oldinode), NFS_FH(oldinode), NFS_FH(dir), name); if (!error) { nfs_lookup_cache_remove(dir, oldinode, NULL); NFS_READTIME(oldinode) = 0; /* force getattr */ } nfs_invalidate_dircache(dir); iput(oldinode); iput(dir); return error; } /* * RENAME * FIXME: Some nfsds, like the Linux user space nfsd, may generate a * different file handle for the same inode after a rename (e.g. when * moving to a different directory). A fail-safe method to do so would * be to look up old_dir/old_name, create a link to new_dir/new_name and * rename the old file using the silly_rename stuff. This way, the original * file in old_dir will go away when the last process iput()s the inode. */ static int nfs_rename(struct inode *old_dir, const char *old_name, int old_len, struct inode *new_dir, const char *new_name, int new_len, int must_be_dir) { int error; dfprintk(VFS, "NFS: rename(%x/%ld, %s -> %x/%ld, %s)\n", old_dir->i_dev, old_dir->i_ino, old_name, new_dir->i_dev, new_dir->i_ino, new_name); if (!old_dir || !S_ISDIR(old_dir->i_mode)) { printk("nfs_rename: old inode is NULL or not a directory\n"); iput(old_dir); iput(new_dir); return -ENOENT; } if (!new_dir || !S_ISDIR(new_dir->i_mode)) { printk("nfs_rename: new inode is NULL or not a directory\n"); iput(old_dir); iput(new_dir); return -ENOENT; } if (old_len > NFS_MAXNAMLEN || new_len > NFS_MAXNAMLEN) { iput(old_dir); iput(new_dir); return -ENAMETOOLONG; } /* We don't do rename() with trailing slashes over NFS now. Hmm. */ if (must_be_dir) return -EINVAL; error = nfs_proc_rename(NFS_SERVER(old_dir), NFS_FH(old_dir), old_name, NFS_FH(new_dir), new_name); if (!error) { nfs_lookup_cache_remove(old_dir, NULL, old_name); nfs_lookup_cache_remove(new_dir, NULL, new_name); } nfs_invalidate_dircache(old_dir); nfs_invalidate_dircache(new_dir); iput(old_dir); iput(new_dir); return error; } /* * Many nfs protocol calls return the new file attributes after * an operation. Here we update the inode to reflect the state * of the server's inode. */ void nfs_refresh_inode(struct inode *inode, struct nfs_fattr *fattr) { int was_empty; dfprintk(VFS, "NFS: refresh_inode(%x/%ld ct=%d)\n", inode->i_dev, inode->i_ino, inode->i_count); if (!inode || !fattr) { printk("nfs_refresh_inode: inode or fattr is NULL\n"); return; } if (inode->i_ino != fattr->fileid) { printk("nfs_refresh_inode: inode number mismatch\n"); return; } was_empty = (inode->i_mode == 0); inode->i_mode = fattr->mode; inode->i_nlink = fattr->nlink; inode->i_uid = fattr->uid; inode->i_gid = fattr->gid; /* Size changed from outside: invalidate caches on next read */ if (inode->i_size != fattr->size) { dfprintk(PAGECACHE, "NFS: cacheinv(%x/%ld)\n", inode->i_dev, inode->i_ino); NFS_CACHEINV(inode); } if (NFS_OLDMTIME(inode) != fattr->mtime.seconds) { dfprintk(PAGECACHE, "NFS: mtime change on %x/%ld\n", inode->i_dev, inode->i_ino); NFS_ATTRTIMEO(inode) = NFS_MINATTRTIMEO(inode); } inode->i_size = fattr->size; if (S_ISCHR(inode->i_mode) || S_ISBLK(inode->i_mode)) inode->i_rdev = to_kdev_t(fattr->rdev); else inode->i_rdev = 0; inode->i_blocks = fattr->blocks; inode->i_atime = fattr->atime.seconds; inode->i_mtime = fattr->mtime.seconds; inode->i_ctime = fattr->ctime.seconds; if (S_ISREG(inode->i_mode)) inode->i_op = &nfs_file_inode_operations; else if (S_ISDIR(inode->i_mode)) inode->i_op = &nfs_dir_inode_operations; else if (S_ISLNK(inode->i_mode)) inode->i_op = &nfs_symlink_inode_operations; else if (S_ISCHR(inode->i_mode)) inode->i_op = &chrdev_inode_operations; else if (S_ISBLK(inode->i_mode)) inode->i_op = &blkdev_inode_operations; else if (S_ISFIFO(inode->i_mode)) { if (was_empty) init_fifo(inode); } else inode->i_op = NULL; nfs_lookup_cache_refresh(inode, fattr); }