fedora core 6 1.2949 + vserver 2.2.0
[linux-2.6.git] / fs / hfsplus / dir.c
index 656b5a2..e886ac8 100644 (file)
 #include <linux/sched.h>
 #include <linux/slab.h>
 #include <linux/random.h>
-#include <linux/version.h>
 
 #include "hfsplus_fs.h"
 #include "hfsplus_raw.h"
 
+static inline void hfsplus_instantiate(struct dentry *dentry,
+                                      struct inode *inode, u32 cnid)
+{
+       dentry->d_fsdata = (void *)(unsigned long)cnid;
+       d_instantiate(dentry, inode);
+}
+
 /* Find the entry inside dir named dentry->d_name */
 static struct dentry *hfsplus_lookup(struct inode *dir, struct dentry *dentry,
                                     struct nameidata *nd)
@@ -33,7 +39,7 @@ static struct dentry *hfsplus_lookup(struct inode *dir, struct dentry *dentry,
        sb = dir->i_sb;
        dentry->d_fsdata = NULL;
        hfs_find_init(HFSPLUS_SB(sb).cat_tree, &fd);
-       hfsplus_cat_build_key(fd.search_key, dir->i_ino, &dentry->d_name);
+       hfsplus_cat_build_key(sb, fd.search_key, dir->i_ino, &dentry->d_name);
 again:
        err = hfs_brec_read(&fd, &entry, sizeof(entry));
        if (err) {
@@ -52,6 +58,7 @@ again:
                        goto fail;
                }
                cnid = be32_to_cpu(entry.folder.id);
+               dentry->d_fsdata = (void *)(unsigned long)cnid;
        } else if (type == HFSPLUS_FILE) {
                if (fd.entrylength < sizeof(struct hfsplus_cat_file)) {
                        err = -EIO;
@@ -59,25 +66,32 @@ again:
                }
                cnid = be32_to_cpu(entry.file.id);
                if (entry.file.user_info.fdType == cpu_to_be32(HFSP_HARDLINK_TYPE) &&
-                   entry.file.user_info.fdCreator == cpu_to_be32(HFSP_HFSPLUS_CREATOR)) {
+                   entry.file.user_info.fdCreator == cpu_to_be32(HFSP_HFSPLUS_CREATOR) &&
+                   (entry.file.create_date == HFSPLUS_I(HFSPLUS_SB(sb).hidden_dir).create_date ||
+                    entry.file.create_date == HFSPLUS_I(sb->s_root->d_inode).create_date) &&
+                   HFSPLUS_SB(sb).hidden_dir) {
                        struct qstr str;
                        char name[32];
 
                        if (dentry->d_fsdata) {
-                               err = -ENOENT;
-                               inode = NULL;
-                               goto out;
+                               /*
+                                * We found a link pointing to another link,
+                                * so ignore it and treat it as regular file.
+                                */
+                               cnid = (unsigned long)dentry->d_fsdata;
+                               linkid = 0;
+                       } else {
+                               dentry->d_fsdata = (void *)(unsigned long)cnid;
+                               linkid = be32_to_cpu(entry.file.permissions.dev);
+                               str.len = sprintf(name, "iNode%d", linkid);
+                               str.name = name;
+                               hfsplus_cat_build_key(sb, fd.search_key, HFSPLUS_SB(sb).hidden_dir->i_ino, &str);
+                               goto again;
                        }
-                       dentry->d_fsdata = (void *)(unsigned long)cnid;
-                       linkid = be32_to_cpu(entry.file.permissions.dev);
-                       str.len = sprintf(name, "iNode%d", linkid);
-                       str.name = name;
-                       hfsplus_cat_build_key(fd.search_key, HFSPLUS_SB(sb).hidden_dir->i_ino, &str);
-                       goto again;
                } else if (!dentry->d_fsdata)
                        dentry->d_fsdata = (void *)(unsigned long)cnid;
        } else {
-               printk("HFS+-fs: Illegal catalog entry type in lookup\n");
+               printk(KERN_ERR "hfs: invalid catalog entry type in lookup\n");
                err = -EIO;
                goto fail;
        }
@@ -97,7 +111,7 @@ fail:
 
 static int hfsplus_readdir(struct file *filp, void *dirent, filldir_t filldir)
 {
-       struct inode *inode = filp->f_dentry->d_inode;
+       struct inode *inode = filp->f_path.dentry->d_inode;
        struct super_block *sb = inode->i_sb;
        int len, err;
        char strbuf[HFSPLUS_MAX_STRLEN + 1];
@@ -110,7 +124,7 @@ static int hfsplus_readdir(struct file *filp, void *dirent, filldir_t filldir)
                return 0;
 
        hfs_find_init(HFSPLUS_SB(sb).cat_tree, &fd);
-       hfsplus_cat_build_key(fd.search_key, inode->i_ino, NULL);
+       hfsplus_cat_build_key(sb, fd.search_key, inode->i_ino, NULL);
        err = hfs_brec_find(&fd);
        if (err)
                goto out;
@@ -125,12 +139,12 @@ static int hfsplus_readdir(struct file *filp, void *dirent, filldir_t filldir)
        case 1:
                hfs_bnode_read(fd.bnode, &entry, fd.entryoffset, fd.entrylength);
                if (be16_to_cpu(entry.type) != HFSPLUS_FOLDER_THREAD) {
-                       printk("HFS+-fs: bad catalog folder thread\n");
+                       printk(KERN_ERR "hfs: bad catalog folder thread\n");
                        err = -EIO;
                        goto out;
                }
                if (fd.entrylength < HFSPLUS_MIN_THREAD_SZ) {
-                       printk("HFS+-fs: truncated catalog thread\n");
+                       printk(KERN_ERR "hfs: truncated catalog thread\n");
                        err = -EIO;
                        goto out;
                }
@@ -149,19 +163,19 @@ static int hfsplus_readdir(struct file *filp, void *dirent, filldir_t filldir)
 
        for (;;) {
                if (be32_to_cpu(fd.key->cat.parent) != inode->i_ino) {
-                       printk("HFS+-fs: walked past end of dir\n");
+                       printk(KERN_ERR "hfs: walked past end of dir\n");
                        err = -EIO;
                        goto out;
                }
                hfs_bnode_read(fd.bnode, &entry, fd.entryoffset, fd.entrylength);
                type = be16_to_cpu(entry.type);
                len = HFSPLUS_MAX_STRLEN;
-               err = hfsplus_uni2asc(&fd.key->cat.name, strbuf, &len);
+               err = hfsplus_uni2asc(sb, &fd.key->cat.name, strbuf, &len);
                if (err)
                        goto out;
                if (type == HFSPLUS_FOLDER) {
                        if (fd.entrylength < sizeof(struct hfsplus_cat_folder)) {
-                               printk("HFS+-fs: small dir entry\n");
+                               printk(KERN_ERR "hfs: small dir entry\n");
                                err = -EIO;
                                goto out;
                        }
@@ -173,7 +187,7 @@ static int hfsplus_readdir(struct file *filp, void *dirent, filldir_t filldir)
                                break;
                } else if (type == HFSPLUS_FILE) {
                        if (fd.entrylength < sizeof(struct hfsplus_cat_file)) {
-                               printk("HFS+-fs: small file entry\n");
+                               printk(KERN_ERR "hfs: small file entry\n");
                                err = -EIO;
                                goto out;
                        }
@@ -181,7 +195,7 @@ static int hfsplus_readdir(struct file *filp, void *dirent, filldir_t filldir)
                                    be32_to_cpu(entry.file.id), DT_REG))
                                break;
                } else {
-                       printk("HFS+-fs: bad catalog entry type\n");
+                       printk(KERN_ERR "hfs: bad catalog entry type\n");
                        err = -EIO;
                        goto out;
                }
@@ -220,8 +234,8 @@ static int hfsplus_dir_release(struct inode *inode, struct file *file)
        return 0;
 }
 
-int hfsplus_create(struct inode *dir, struct dentry *dentry, int mode,
-                  struct nameidata *nd)
+static int hfsplus_create(struct inode *dir, struct dentry *dentry, int mode,
+                         struct nameidata *nd)
 {
        struct inode *inode;
        int res;
@@ -233,16 +247,17 @@ int hfsplus_create(struct inode *dir, struct dentry *dentry, int mode,
        res = hfsplus_create_cat(inode->i_ino, dir, &dentry->d_name, inode);
        if (res) {
                inode->i_nlink = 0;
+               hfsplus_delete_inode(inode);
                iput(inode);
                return res;
        }
-       dentry->d_fsdata = (void *)inode->i_ino;
-       d_instantiate(dentry, inode);
+       hfsplus_instantiate(dentry, inode, inode->i_ino);
        mark_inode_dirty(inode);
        return 0;
 }
 
-int hfsplus_link(struct dentry *src_dentry, struct inode *dst_dir, struct dentry *dst_dentry)
+static int hfsplus_link(struct dentry *src_dentry, struct inode *dst_dir,
+                       struct dentry *dst_dentry)
 {
        struct super_block *sb = dst_dir->i_sb;
        struct inode *inode = src_dentry->d_inode;
@@ -283,11 +298,10 @@ int hfsplus_link(struct dentry *src_dentry, struct inode *dst_dir, struct dentry
        if (res)
                return res;
 
-       inode->i_nlink++;
-       dst_dentry->d_fsdata = (void *)(unsigned long)cnid;
-       d_instantiate(dst_dentry, inode);
+       inc_nlink(inode);
+       hfsplus_instantiate(dst_dentry, inode, cnid);
        atomic_inc(&inode->i_count);
-       inode->i_ctime = CURRENT_TIME;
+       inode->i_ctime = CURRENT_TIME_SEC;
        mark_inode_dirty(inode);
        HFSPLUS_SB(sb).file_count++;
        sb->s_dirt = 1;
@@ -295,7 +309,7 @@ int hfsplus_link(struct dentry *src_dentry, struct inode *dst_dir, struct dentry
        return 0;
 }
 
-int hfsplus_unlink(struct inode *dir, struct dentry *dentry)
+static int hfsplus_unlink(struct inode *dir, struct dentry *dentry)
 {
        struct super_block *sb = dir->i_sb;
        struct inode *inode = dentry->d_inode;
@@ -323,7 +337,8 @@ int hfsplus_unlink(struct inode *dir, struct dentry *dentry)
        if (res)
                return res;
 
-       inode->i_nlink--;
+       if (inode->i_nlink > 0)
+               drop_nlink(inode);
        hfsplus_delete_inode(inode);
        if (inode->i_ino != cnid && !inode->i_nlink) {
                if (!atomic_read(&HFSPLUS_I(inode).opencnt)) {
@@ -332,14 +347,15 @@ int hfsplus_unlink(struct inode *dir, struct dentry *dentry)
                                hfsplus_delete_inode(inode);
                } else
                        inode->i_flags |= S_DEAD;
-       }
-       inode->i_ctime = CURRENT_TIME;
+       } else
+               clear_nlink(inode);
+       inode->i_ctime = CURRENT_TIME_SEC;
        mark_inode_dirty(inode);
 
        return res;
 }
 
-int hfsplus_mkdir(struct inode *dir, struct dentry *dentry, int mode)
+static int hfsplus_mkdir(struct inode *dir, struct dentry *dentry, int mode)
 {
        struct inode *inode;
        int res;
@@ -351,15 +367,16 @@ int hfsplus_mkdir(struct inode *dir, struct dentry *dentry, int mode)
        res = hfsplus_create_cat(inode->i_ino, dir, &dentry->d_name, inode);
        if (res) {
                inode->i_nlink = 0;
+               hfsplus_delete_inode(inode);
                iput(inode);
                return res;
        }
-       d_instantiate(dentry, inode);
+       hfsplus_instantiate(dentry, inode, inode->i_ino);
        mark_inode_dirty(inode);
        return 0;
 }
 
-int hfsplus_rmdir(struct inode *dir, struct dentry *dentry)
+static int hfsplus_rmdir(struct inode *dir, struct dentry *dentry)
 {
        struct inode *inode;
        int res;
@@ -370,14 +387,15 @@ int hfsplus_rmdir(struct inode *dir, struct dentry *dentry)
        res = hfsplus_delete_cat(inode->i_ino, dir, &dentry->d_name);
        if (res)
                return res;
-       inode->i_nlink = 0;
-       inode->i_ctime = CURRENT_TIME;
+       clear_nlink(inode);
+       inode->i_ctime = CURRENT_TIME_SEC;
        hfsplus_delete_inode(inode);
        mark_inode_dirty(inode);
        return 0;
 }
 
-int hfsplus_symlink(struct inode *dir, struct dentry *dentry, const char *symname)
+static int hfsplus_symlink(struct inode *dir, struct dentry *dentry,
+                          const char *symname)
 {
        struct super_block *sb;
        struct inode *inode;
@@ -391,7 +409,8 @@ int hfsplus_symlink(struct inode *dir, struct dentry *dentry, const char *symnam
        res = page_symlink(inode, symname, strlen(symname) + 1);
        if (res) {
                inode->i_nlink = 0;
-               iput (inode);
+               hfsplus_delete_inode(inode);
+               iput(inode);
                return res;
        }
 
@@ -399,15 +418,15 @@ int hfsplus_symlink(struct inode *dir, struct dentry *dentry, const char *symnam
        res = hfsplus_create_cat(inode->i_ino, dir, &dentry->d_name, inode);
 
        if (!res) {
-               dentry->d_fsdata = (void *)inode->i_ino;
-               d_instantiate(dentry, inode);
+               hfsplus_instantiate(dentry, inode, inode->i_ino);
                mark_inode_dirty(inode);
        }
 
        return res;
 }
 
-int hfsplus_mknod(struct inode *dir, struct dentry *dentry, int mode, dev_t rdev)
+static int hfsplus_mknod(struct inode *dir, struct dentry *dentry,
+                        int mode, dev_t rdev)
 {
        struct super_block *sb;
        struct inode *inode;
@@ -421,19 +440,19 @@ int hfsplus_mknod(struct inode *dir, struct dentry *dentry, int mode, dev_t rdev
        res = hfsplus_create_cat(inode->i_ino, dir, &dentry->d_name, inode);
        if (res) {
                inode->i_nlink = 0;
+               hfsplus_delete_inode(inode);
                iput(inode);
                return res;
        }
        init_special_inode(inode, mode, rdev);
-       dentry->d_fsdata = (void *)inode->i_ino;
-       d_instantiate(dentry, inode);
+       hfsplus_instantiate(dentry, inode, inode->i_ino);
        mark_inode_dirty(inode);
 
        return 0;
 }
 
-int hfsplus_rename(struct inode *old_dir, struct dentry *old_dentry,
-                  struct inode *new_dir, struct dentry *new_dentry)
+static int hfsplus_rename(struct inode *old_dir, struct dentry *old_dentry,
+                         struct inode *new_dir, struct dentry *new_dentry)
 {
        int res;
 
@@ -464,7 +483,7 @@ struct inode_operations hfsplus_dir_inode_operations = {
        .rename         = hfsplus_rename,
 };
 
-struct file_operations hfsplus_dir_operations = {
+const struct file_operations hfsplus_dir_operations = {
        .read           = generic_read_dir,
        .readdir        = hfsplus_readdir,
        .ioctl          = hfsplus_ioctl,