diff options
Diffstat (limited to 'fs/sysfs')
| -rw-r--r-- | fs/sysfs/dir.c | 45 | ||||
| -rw-r--r-- | fs/sysfs/file.c | 3 |
2 files changed, 48 insertions, 0 deletions
diff --git a/fs/sysfs/dir.c b/fs/sysfs/dir.c index 3aa3434621ca..a5782e8c7f07 100644 --- a/fs/sysfs/dir.c +++ b/fs/sysfs/dir.c | |||
| @@ -372,6 +372,51 @@ int sysfs_rename_dir(struct kobject * kobj, const char *new_name) | |||
| 372 | return error; | 372 | return error; |
| 373 | } | 373 | } |
| 374 | 374 | ||
| 375 | int sysfs_move_dir(struct kobject *kobj, struct kobject *new_parent) | ||
| 376 | { | ||
| 377 | struct dentry *old_parent_dentry, *new_parent_dentry, *new_dentry; | ||
| 378 | struct sysfs_dirent *new_parent_sd, *sd; | ||
| 379 | int error; | ||
| 380 | |||
| 381 | if (!new_parent) | ||
| 382 | return -EINVAL; | ||
| 383 | |||
| 384 | old_parent_dentry = kobj->parent ? | ||
| 385 | kobj->parent->dentry : sysfs_mount->mnt_sb->s_root; | ||
| 386 | new_parent_dentry = new_parent->dentry; | ||
| 387 | |||
| 388 | again: | ||
| 389 | mutex_lock(&old_parent_dentry->d_inode->i_mutex); | ||
| 390 | if (!mutex_trylock(&new_parent_dentry->d_inode->i_mutex)) { | ||
| 391 | mutex_unlock(&old_parent_dentry->d_inode->i_mutex); | ||
| 392 | goto again; | ||
| 393 | } | ||
| 394 | |||
| 395 | new_parent_sd = new_parent_dentry->d_fsdata; | ||
| 396 | sd = kobj->dentry->d_fsdata; | ||
| 397 | |||
| 398 | new_dentry = lookup_one_len(kobj->name, new_parent_dentry, | ||
| 399 | strlen(kobj->name)); | ||
| 400 | if (IS_ERR(new_dentry)) { | ||
| 401 | error = PTR_ERR(new_dentry); | ||
| 402 | goto out; | ||
| 403 | } else | ||
| 404 | error = 0; | ||
| 405 | d_add(new_dentry, NULL); | ||
| 406 | d_move(kobj->dentry, new_dentry); | ||
| 407 | dput(new_dentry); | ||
| 408 | |||
| 409 | /* Remove from old parent's list and insert into new parent's list. */ | ||
| 410 | list_del_init(&sd->s_sibling); | ||
| 411 | list_add(&sd->s_sibling, &new_parent_sd->s_children); | ||
| 412 | |||
| 413 | out: | ||
| 414 | mutex_unlock(&new_parent_dentry->d_inode->i_mutex); | ||
| 415 | mutex_unlock(&old_parent_dentry->d_inode->i_mutex); | ||
| 416 | |||
| 417 | return error; | ||
| 418 | } | ||
| 419 | |||
| 375 | static int sysfs_dir_open(struct inode *inode, struct file *file) | 420 | static int sysfs_dir_open(struct inode *inode, struct file *file) |
| 376 | { | 421 | { |
| 377 | struct dentry * dentry = file->f_dentry; | 422 | struct dentry * dentry = file->f_dentry; |
diff --git a/fs/sysfs/file.c b/fs/sysfs/file.c index 298303b5a716..95c165101c98 100644 --- a/fs/sysfs/file.c +++ b/fs/sysfs/file.c | |||
| @@ -190,6 +190,9 @@ fill_write_buffer(struct sysfs_buffer * buffer, const char __user * buf, size_t | |||
| 190 | count = PAGE_SIZE - 1; | 190 | count = PAGE_SIZE - 1; |
| 191 | error = copy_from_user(buffer->page,buf,count); | 191 | error = copy_from_user(buffer->page,buf,count); |
| 192 | buffer->needs_read_fill = 1; | 192 | buffer->needs_read_fill = 1; |
| 193 | /* if buf is assumed to contain a string, terminate it by \0, | ||
| 194 | so e.g. sscanf() can scan the string easily */ | ||
| 195 | buffer->page[count] = 0; | ||
| 193 | return error ? -EFAULT : count; | 196 | return error ? -EFAULT : count; |
| 194 | } | 197 | } |
| 195 | 198 | ||
