diff options
Diffstat (limited to 'fs/xfs/linux-2.6/xfs_ioctl.c')
-rw-r--r-- | fs/xfs/linux-2.6/xfs_ioctl.c | 63 |
1 files changed, 45 insertions, 18 deletions
diff --git a/fs/xfs/linux-2.6/xfs_ioctl.c b/fs/xfs/linux-2.6/xfs_ioctl.c index 5917808abbd6..47cfde6cfae2 100644 --- a/fs/xfs/linux-2.6/xfs_ioctl.c +++ b/fs/xfs/linux-2.6/xfs_ioctl.c | |||
@@ -349,19 +349,44 @@ xfs_open_by_handle( | |||
349 | return new_fd; | 349 | return new_fd; |
350 | } | 350 | } |
351 | 351 | ||
352 | /* | ||
353 | * This is a copy from fs/namei.c:vfs_readlink(), except for removing it's | ||
354 | * unused first argument. | ||
355 | */ | ||
356 | STATIC int | ||
357 | do_readlink( | ||
358 | char __user *buffer, | ||
359 | int buflen, | ||
360 | const char *link) | ||
361 | { | ||
362 | int len; | ||
363 | |||
364 | len = PTR_ERR(link); | ||
365 | if (IS_ERR(link)) | ||
366 | goto out; | ||
367 | |||
368 | len = strlen(link); | ||
369 | if (len > (unsigned) buflen) | ||
370 | len = buflen; | ||
371 | if (copy_to_user(buffer, link, len)) | ||
372 | len = -EFAULT; | ||
373 | out: | ||
374 | return len; | ||
375 | } | ||
376 | |||
377 | |||
352 | STATIC int | 378 | STATIC int |
353 | xfs_readlink_by_handle( | 379 | xfs_readlink_by_handle( |
354 | xfs_mount_t *mp, | 380 | xfs_mount_t *mp, |
355 | void __user *arg, | 381 | void __user *arg, |
356 | struct inode *parinode) | 382 | struct inode *parinode) |
357 | { | 383 | { |
358 | int error; | ||
359 | struct iovec aiov; | ||
360 | struct uio auio; | ||
361 | struct inode *inode; | 384 | struct inode *inode; |
362 | xfs_fsop_handlereq_t hreq; | 385 | xfs_fsop_handlereq_t hreq; |
363 | bhv_vnode_t *vp; | 386 | bhv_vnode_t *vp; |
364 | __u32 olen; | 387 | __u32 olen; |
388 | void *link; | ||
389 | int error; | ||
365 | 390 | ||
366 | if (!capable(CAP_SYS_ADMIN)) | 391 | if (!capable(CAP_SYS_ADMIN)) |
367 | return -XFS_ERROR(EPERM); | 392 | return -XFS_ERROR(EPERM); |
@@ -374,29 +399,31 @@ xfs_readlink_by_handle( | |||
374 | 399 | ||
375 | /* Restrict this handle operation to symlinks only. */ | 400 | /* Restrict this handle operation to symlinks only. */ |
376 | if (!S_ISLNK(inode->i_mode)) { | 401 | if (!S_ISLNK(inode->i_mode)) { |
377 | VN_RELE(vp); | 402 | error = -XFS_ERROR(EINVAL); |
378 | return -XFS_ERROR(EINVAL); | 403 | goto out_iput; |
379 | } | 404 | } |
380 | 405 | ||
381 | if (copy_from_user(&olen, hreq.ohandlen, sizeof(__u32))) { | 406 | if (copy_from_user(&olen, hreq.ohandlen, sizeof(__u32))) { |
382 | VN_RELE(vp); | 407 | error = -XFS_ERROR(EFAULT); |
383 | return -XFS_ERROR(EFAULT); | 408 | goto out_iput; |
384 | } | 409 | } |
385 | aiov.iov_len = olen; | ||
386 | aiov.iov_base = hreq.ohandle; | ||
387 | 410 | ||
388 | auio.uio_iov = (struct kvec *)&aiov; | 411 | link = kmalloc(MAXPATHLEN+1, GFP_KERNEL); |
389 | auio.uio_iovcnt = 1; | 412 | if (!link) |
390 | auio.uio_offset = 0; | 413 | goto out_iput; |
391 | auio.uio_segflg = UIO_USERSPACE; | ||
392 | auio.uio_resid = olen; | ||
393 | 414 | ||
394 | error = bhv_vop_readlink(vp, &auio, IO_INVIS, NULL); | 415 | error = -bhv_vop_readlink(vp, link); |
395 | VN_RELE(vp); | ||
396 | if (error) | 416 | if (error) |
397 | return -error; | 417 | goto out_kfree; |
418 | error = do_readlink(hreq.ohandle, olen, link); | ||
419 | if (error) | ||
420 | goto out_kfree; | ||
398 | 421 | ||
399 | return (olen - auio.uio_resid); | 422 | out_kfree: |
423 | kfree(link); | ||
424 | out_iput: | ||
425 | iput(inode); | ||
426 | return error; | ||
400 | } | 427 | } |
401 | 428 | ||
402 | STATIC int | 429 | STATIC int |