diff options
Diffstat (limited to 'fs/sysfs/bin.c')
-rw-r--r-- | fs/sysfs/bin.c | 195 |
1 files changed, 124 insertions, 71 deletions
diff --git a/fs/sysfs/bin.c b/fs/sysfs/bin.c index d3b9f5f07db1..135353f8a296 100644 --- a/fs/sysfs/bin.c +++ b/fs/sysfs/bin.c | |||
@@ -20,29 +20,41 @@ | |||
20 | 20 | ||
21 | #include "sysfs.h" | 21 | #include "sysfs.h" |
22 | 22 | ||
23 | struct bin_buffer { | ||
24 | struct mutex mutex; | ||
25 | void *buffer; | ||
26 | int mmapped; | ||
27 | }; | ||
28 | |||
23 | static int | 29 | static int |
24 | fill_read(struct dentry *dentry, char *buffer, loff_t off, size_t count) | 30 | fill_read(struct dentry *dentry, char *buffer, loff_t off, size_t count) |
25 | { | 31 | { |
26 | struct bin_attribute * attr = to_bin_attr(dentry); | 32 | struct sysfs_dirent *attr_sd = dentry->d_fsdata; |
27 | struct kobject * kobj = to_kobj(dentry->d_parent); | 33 | struct bin_attribute *attr = attr_sd->s_elem.bin_attr.bin_attr; |
34 | struct kobject *kobj = attr_sd->s_parent->s_elem.dir.kobj; | ||
35 | int rc; | ||
36 | |||
37 | /* need attr_sd for attr, its parent for kobj */ | ||
38 | if (!sysfs_get_active_two(attr_sd)) | ||
39 | return -ENODEV; | ||
28 | 40 | ||
29 | if (!attr->read) | 41 | rc = -EIO; |
30 | return -EIO; | 42 | if (attr->read) |
43 | rc = attr->read(kobj, attr, buffer, off, count); | ||
31 | 44 | ||
32 | return attr->read(kobj, buffer, off, count); | 45 | sysfs_put_active_two(attr_sd); |
46 | |||
47 | return rc; | ||
33 | } | 48 | } |
34 | 49 | ||
35 | static ssize_t | 50 | static ssize_t |
36 | read(struct file * file, char __user * userbuf, size_t count, loff_t * off) | 51 | read(struct file *file, char __user *userbuf, size_t bytes, loff_t *off) |
37 | { | 52 | { |
38 | char *buffer = file->private_data; | 53 | struct bin_buffer *bb = file->private_data; |
39 | struct dentry *dentry = file->f_path.dentry; | 54 | struct dentry *dentry = file->f_path.dentry; |
40 | int size = dentry->d_inode->i_size; | 55 | int size = dentry->d_inode->i_size; |
41 | loff_t offs = *off; | 56 | loff_t offs = *off; |
42 | int ret; | 57 | int count = min_t(size_t, bytes, PAGE_SIZE); |
43 | |||
44 | if (count > PAGE_SIZE) | ||
45 | count = PAGE_SIZE; | ||
46 | 58 | ||
47 | if (size) { | 59 | if (size) { |
48 | if (offs > size) | 60 | if (offs > size) |
@@ -51,43 +63,56 @@ read(struct file * file, char __user * userbuf, size_t count, loff_t * off) | |||
51 | count = size - offs; | 63 | count = size - offs; |
52 | } | 64 | } |
53 | 65 | ||
54 | ret = fill_read(dentry, buffer, offs, count); | 66 | mutex_lock(&bb->mutex); |
55 | if (ret < 0) | 67 | |
56 | return ret; | 68 | count = fill_read(dentry, bb->buffer, offs, count); |
57 | count = ret; | 69 | if (count < 0) |
70 | goto out_unlock; | ||
58 | 71 | ||
59 | if (copy_to_user(userbuf, buffer, count)) | 72 | if (copy_to_user(userbuf, bb->buffer, count)) { |
60 | return -EFAULT; | 73 | count = -EFAULT; |
74 | goto out_unlock; | ||
75 | } | ||
61 | 76 | ||
62 | pr_debug("offs = %lld, *off = %lld, count = %zd\n", offs, *off, count); | 77 | pr_debug("offs = %lld, *off = %lld, count = %d\n", offs, *off, count); |
63 | 78 | ||
64 | *off = offs + count; | 79 | *off = offs + count; |
65 | 80 | ||
81 | out_unlock: | ||
82 | mutex_unlock(&bb->mutex); | ||
66 | return count; | 83 | return count; |
67 | } | 84 | } |
68 | 85 | ||
69 | static int | 86 | static int |
70 | flush_write(struct dentry *dentry, char *buffer, loff_t offset, size_t count) | 87 | flush_write(struct dentry *dentry, char *buffer, loff_t offset, size_t count) |
71 | { | 88 | { |
72 | struct bin_attribute *attr = to_bin_attr(dentry); | 89 | struct sysfs_dirent *attr_sd = dentry->d_fsdata; |
73 | struct kobject *kobj = to_kobj(dentry->d_parent); | 90 | struct bin_attribute *attr = attr_sd->s_elem.bin_attr.bin_attr; |
91 | struct kobject *kobj = attr_sd->s_parent->s_elem.dir.kobj; | ||
92 | int rc; | ||
93 | |||
94 | /* need attr_sd for attr, its parent for kobj */ | ||
95 | if (!sysfs_get_active_two(attr_sd)) | ||
96 | return -ENODEV; | ||
97 | |||
98 | rc = -EIO; | ||
99 | if (attr->write) | ||
100 | rc = attr->write(kobj, attr, buffer, offset, count); | ||
74 | 101 | ||
75 | if (!attr->write) | 102 | sysfs_put_active_two(attr_sd); |
76 | return -EIO; | ||
77 | 103 | ||
78 | return attr->write(kobj, buffer, offset, count); | 104 | return rc; |
79 | } | 105 | } |
80 | 106 | ||
81 | static ssize_t write(struct file * file, const char __user * userbuf, | 107 | static ssize_t write(struct file *file, const char __user *userbuf, |
82 | size_t count, loff_t * off) | 108 | size_t bytes, loff_t *off) |
83 | { | 109 | { |
84 | char *buffer = file->private_data; | 110 | struct bin_buffer *bb = file->private_data; |
85 | struct dentry *dentry = file->f_path.dentry; | 111 | struct dentry *dentry = file->f_path.dentry; |
86 | int size = dentry->d_inode->i_size; | 112 | int size = dentry->d_inode->i_size; |
87 | loff_t offs = *off; | 113 | loff_t offs = *off; |
114 | int count = min_t(size_t, bytes, PAGE_SIZE); | ||
88 | 115 | ||
89 | if (count > PAGE_SIZE) | ||
90 | count = PAGE_SIZE; | ||
91 | if (size) { | 116 | if (size) { |
92 | if (offs > size) | 117 | if (offs > size) |
93 | return 0; | 118 | return 0; |
@@ -95,72 +120,100 @@ static ssize_t write(struct file * file, const char __user * userbuf, | |||
95 | count = size - offs; | 120 | count = size - offs; |
96 | } | 121 | } |
97 | 122 | ||
98 | if (copy_from_user(buffer, userbuf, count)) | 123 | mutex_lock(&bb->mutex); |
99 | return -EFAULT; | ||
100 | 124 | ||
101 | count = flush_write(dentry, buffer, offs, count); | 125 | if (copy_from_user(bb->buffer, userbuf, count)) { |
126 | count = -EFAULT; | ||
127 | goto out_unlock; | ||
128 | } | ||
129 | |||
130 | count = flush_write(dentry, bb->buffer, offs, count); | ||
102 | if (count > 0) | 131 | if (count > 0) |
103 | *off = offs + count; | 132 | *off = offs + count; |
133 | |||
134 | out_unlock: | ||
135 | mutex_unlock(&bb->mutex); | ||
104 | return count; | 136 | return count; |
105 | } | 137 | } |
106 | 138 | ||
107 | static int mmap(struct file *file, struct vm_area_struct *vma) | 139 | static int mmap(struct file *file, struct vm_area_struct *vma) |
108 | { | 140 | { |
109 | struct dentry *dentry = file->f_path.dentry; | 141 | struct bin_buffer *bb = file->private_data; |
110 | struct bin_attribute *attr = to_bin_attr(dentry); | 142 | struct sysfs_dirent *attr_sd = file->f_path.dentry->d_fsdata; |
111 | struct kobject *kobj = to_kobj(dentry->d_parent); | 143 | struct bin_attribute *attr = attr_sd->s_elem.bin_attr.bin_attr; |
144 | struct kobject *kobj = attr_sd->s_parent->s_elem.dir.kobj; | ||
145 | int rc; | ||
146 | |||
147 | mutex_lock(&bb->mutex); | ||
148 | |||
149 | /* need attr_sd for attr, its parent for kobj */ | ||
150 | if (!sysfs_get_active_two(attr_sd)) | ||
151 | return -ENODEV; | ||
112 | 152 | ||
113 | if (!attr->mmap) | 153 | rc = -EINVAL; |
114 | return -EINVAL; | 154 | if (attr->mmap) |
155 | rc = attr->mmap(kobj, attr, vma); | ||
115 | 156 | ||
116 | return attr->mmap(kobj, attr, vma); | 157 | if (rc == 0 && !bb->mmapped) |
158 | bb->mmapped = 1; | ||
159 | else | ||
160 | sysfs_put_active_two(attr_sd); | ||
161 | |||
162 | mutex_unlock(&bb->mutex); | ||
163 | |||
164 | return rc; | ||
117 | } | 165 | } |
118 | 166 | ||
119 | static int open(struct inode * inode, struct file * file) | 167 | static int open(struct inode * inode, struct file * file) |
120 | { | 168 | { |
121 | struct kobject *kobj = sysfs_get_kobject(file->f_path.dentry->d_parent); | 169 | struct sysfs_dirent *attr_sd = file->f_path.dentry->d_fsdata; |
122 | struct bin_attribute * attr = to_bin_attr(file->f_path.dentry); | 170 | struct bin_attribute *attr = attr_sd->s_elem.bin_attr.bin_attr; |
123 | int error = -EINVAL; | 171 | struct bin_buffer *bb = NULL; |
124 | 172 | int error; | |
125 | if (!kobj || !attr) | ||
126 | goto Done; | ||
127 | 173 | ||
128 | /* Grab the module reference for this attribute if we have one */ | 174 | /* need attr_sd for attr */ |
129 | error = -ENODEV; | 175 | if (!sysfs_get_active(attr_sd)) |
130 | if (!try_module_get(attr->attr.owner)) | 176 | return -ENODEV; |
131 | goto Done; | ||
132 | 177 | ||
133 | error = -EACCES; | 178 | error = -EACCES; |
134 | if ((file->f_mode & FMODE_WRITE) && !(attr->write || attr->mmap)) | 179 | if ((file->f_mode & FMODE_WRITE) && !(attr->write || attr->mmap)) |
135 | goto Error; | 180 | goto err_out; |
136 | if ((file->f_mode & FMODE_READ) && !(attr->read || attr->mmap)) | 181 | if ((file->f_mode & FMODE_READ) && !(attr->read || attr->mmap)) |
137 | goto Error; | 182 | goto err_out; |
138 | 183 | ||
139 | error = -ENOMEM; | 184 | error = -ENOMEM; |
140 | file->private_data = kmalloc(PAGE_SIZE, GFP_KERNEL); | 185 | bb = kzalloc(sizeof(*bb), GFP_KERNEL); |
141 | if (!file->private_data) | 186 | if (!bb) |
142 | goto Error; | 187 | goto err_out; |
143 | 188 | ||
144 | error = 0; | 189 | bb->buffer = kmalloc(PAGE_SIZE, GFP_KERNEL); |
145 | goto Done; | 190 | if (!bb->buffer) |
146 | 191 | goto err_out; | |
147 | Error: | 192 | |
148 | module_put(attr->attr.owner); | 193 | mutex_init(&bb->mutex); |
149 | Done: | 194 | file->private_data = bb; |
150 | if (error) | 195 | |
151 | kobject_put(kobj); | 196 | /* open succeeded, put active reference and pin attr_sd */ |
197 | sysfs_put_active(attr_sd); | ||
198 | sysfs_get(attr_sd); | ||
199 | return 0; | ||
200 | |||
201 | err_out: | ||
202 | sysfs_put_active(attr_sd); | ||
203 | kfree(bb); | ||
152 | return error; | 204 | return error; |
153 | } | 205 | } |
154 | 206 | ||
155 | static int release(struct inode * inode, struct file * file) | 207 | static int release(struct inode * inode, struct file * file) |
156 | { | 208 | { |
157 | struct kobject * kobj = to_kobj(file->f_path.dentry->d_parent); | 209 | struct sysfs_dirent *attr_sd = file->f_path.dentry->d_fsdata; |
158 | struct bin_attribute * attr = to_bin_attr(file->f_path.dentry); | 210 | struct bin_buffer *bb = file->private_data; |
159 | u8 * buffer = file->private_data; | 211 | |
160 | 212 | if (bb->mmapped) | |
161 | kobject_put(kobj); | 213 | sysfs_put_active_two(attr_sd); |
162 | module_put(attr->attr.owner); | 214 | sysfs_put(attr_sd); |
163 | kfree(buffer); | 215 | kfree(bb->buffer); |
216 | kfree(bb); | ||
164 | return 0; | 217 | return 0; |
165 | } | 218 | } |
166 | 219 | ||
@@ -181,9 +234,9 @@ const struct file_operations bin_fops = { | |||
181 | 234 | ||
182 | int sysfs_create_bin_file(struct kobject * kobj, struct bin_attribute * attr) | 235 | int sysfs_create_bin_file(struct kobject * kobj, struct bin_attribute * attr) |
183 | { | 236 | { |
184 | BUG_ON(!kobj || !kobj->dentry || !attr); | 237 | BUG_ON(!kobj || !kobj->sd || !attr); |
185 | 238 | ||
186 | return sysfs_add_file(kobj->dentry, &attr->attr, SYSFS_KOBJ_BIN_ATTR); | 239 | return sysfs_add_file(kobj->sd, &attr->attr, SYSFS_KOBJ_BIN_ATTR); |
187 | } | 240 | } |
188 | 241 | ||
189 | 242 | ||
@@ -195,7 +248,7 @@ int sysfs_create_bin_file(struct kobject * kobj, struct bin_attribute * attr) | |||
195 | 248 | ||
196 | void sysfs_remove_bin_file(struct kobject * kobj, struct bin_attribute * attr) | 249 | void sysfs_remove_bin_file(struct kobject * kobj, struct bin_attribute * attr) |
197 | { | 250 | { |
198 | if (sysfs_hash_and_remove(kobj->dentry, attr->attr.name) < 0) { | 251 | if (sysfs_hash_and_remove(kobj->sd, attr->attr.name) < 0) { |
199 | printk(KERN_ERR "%s: " | 252 | printk(KERN_ERR "%s: " |
200 | "bad dentry or inode or no such file: \"%s\"\n", | 253 | "bad dentry or inode or no such file: \"%s\"\n", |
201 | __FUNCTION__, attr->attr.name); | 254 | __FUNCTION__, attr->attr.name); |