aboutsummaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
Diffstat (limited to 'fs')
-rw-r--r--fs/anon_inodes.c25
1 files changed, 12 insertions, 13 deletions
diff --git a/fs/anon_inodes.c b/fs/anon_inodes.c
index b4a75880f6fd..23321889d9b0 100644
--- a/fs/anon_inodes.c
+++ b/fs/anon_inodes.c
@@ -76,7 +76,6 @@ int anon_inode_getfd(int *pfd, struct inode **pinode, struct file **pfile,
76{ 76{
77 struct qstr this; 77 struct qstr this;
78 struct dentry *dentry; 78 struct dentry *dentry;
79 struct inode *inode;
80 struct file *file; 79 struct file *file;
81 int error, fd; 80 int error, fd;
82 81
@@ -86,15 +85,9 @@ int anon_inode_getfd(int *pfd, struct inode **pinode, struct file **pfile,
86 if (!file) 85 if (!file)
87 return -ENFILE; 86 return -ENFILE;
88 87
89 inode = igrab(anon_inode_inode);
90 if (IS_ERR(inode)) {
91 error = PTR_ERR(inode);
92 goto err_put_filp;
93 }
94
95 error = get_unused_fd(); 88 error = get_unused_fd();
96 if (error < 0) 89 if (error < 0)
97 goto err_iput; 90 goto err_put_filp;
98 fd = error; 91 fd = error;
99 92
100 /* 93 /*
@@ -108,14 +101,22 @@ int anon_inode_getfd(int *pfd, struct inode **pinode, struct file **pfile,
108 dentry = d_alloc(anon_inode_mnt->mnt_sb->s_root, &this); 101 dentry = d_alloc(anon_inode_mnt->mnt_sb->s_root, &this);
109 if (!dentry) 102 if (!dentry)
110 goto err_put_unused_fd; 103 goto err_put_unused_fd;
104
105 /*
106 * We know the anon_inode inode count is always greater than zero,
107 * so we can avoid doing an igrab() and we can use an open-coded
108 * atomic_inc().
109 */
110 atomic_inc(&anon_inode_inode->i_count);
111
111 dentry->d_op = &anon_inodefs_dentry_operations; 112 dentry->d_op = &anon_inodefs_dentry_operations;
112 /* Do not publish this dentry inside the global dentry hash table */ 113 /* Do not publish this dentry inside the global dentry hash table */
113 dentry->d_flags &= ~DCACHE_UNHASHED; 114 dentry->d_flags &= ~DCACHE_UNHASHED;
114 d_instantiate(dentry, inode); 115 d_instantiate(dentry, anon_inode_inode);
115 116
116 file->f_path.mnt = mntget(anon_inode_mnt); 117 file->f_path.mnt = mntget(anon_inode_mnt);
117 file->f_path.dentry = dentry; 118 file->f_path.dentry = dentry;
118 file->f_mapping = inode->i_mapping; 119 file->f_mapping = anon_inode_inode->i_mapping;
119 120
120 file->f_pos = 0; 121 file->f_pos = 0;
121 file->f_flags = O_RDWR; 122 file->f_flags = O_RDWR;
@@ -127,14 +128,12 @@ int anon_inode_getfd(int *pfd, struct inode **pinode, struct file **pfile,
127 fd_install(fd, file); 128 fd_install(fd, file);
128 129
129 *pfd = fd; 130 *pfd = fd;
130 *pinode = inode; 131 *pinode = anon_inode_inode;
131 *pfile = file; 132 *pfile = file;
132 return 0; 133 return 0;
133 134
134err_put_unused_fd: 135err_put_unused_fd:
135 put_unused_fd(fd); 136 put_unused_fd(fd);
136err_iput:
137 iput(inode);
138err_put_filp: 137err_put_filp:
139 put_filp(file); 138 put_filp(file);
140 return error; 139 return error;