aboutsummaryrefslogtreecommitdiffstats
path: root/fs/xfs/linux-2.6/xfs_ioctl32.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/xfs/linux-2.6/xfs_ioctl32.c')
-rw-r--r--fs/xfs/linux-2.6/xfs_ioctl32.c94
1 files changed, 90 insertions, 4 deletions
diff --git a/fs/xfs/linux-2.6/xfs_ioctl32.c b/fs/xfs/linux-2.6/xfs_ioctl32.c
index e7eb0d7e0d56..17993352aa64 100644
--- a/fs/xfs/linux-2.6/xfs_ioctl32.c
+++ b/fs/xfs/linux-2.6/xfs_ioctl32.c
@@ -476,6 +476,93 @@ xfs_compat_attrlist_by_handle(
476 return -error; 476 return -error;
477} 477}
478 478
479STATIC int
480xfs_compat_attrmulti_by_handle(
481 xfs_mount_t *mp,
482 void __user *arg,
483 struct inode *parinode)
484{
485 int error;
486 compat_xfs_attr_multiop_t *ops;
487 compat_xfs_fsop_attrmulti_handlereq_t am_hreq;
488 struct inode *inode;
489 unsigned int i, size;
490 char *attr_name;
491
492 if (!capable(CAP_SYS_ADMIN))
493 return -XFS_ERROR(EPERM);
494 if (copy_from_user(&am_hreq, arg,
495 sizeof(compat_xfs_fsop_attrmulti_handlereq_t)))
496 return -XFS_ERROR(EFAULT);
497
498 error = xfs_vget_fsop_handlereq_compat(mp, parinode, &am_hreq.hreq,
499 &inode);
500 if (error)
501 goto out;
502
503 error = E2BIG;
504 size = am_hreq.opcount * sizeof(compat_xfs_attr_multiop_t);
505 if (!size || size > 16 * PAGE_SIZE)
506 goto out_vn_rele;
507
508 error = ENOMEM;
509 ops = kmalloc(size, GFP_KERNEL);
510 if (!ops)
511 goto out_vn_rele;
512
513 error = EFAULT;
514 if (copy_from_user(ops, compat_ptr(am_hreq.ops), size))
515 goto out_kfree_ops;
516
517 attr_name = kmalloc(MAXNAMELEN, GFP_KERNEL);
518 if (!attr_name)
519 goto out_kfree_ops;
520
521
522 error = 0;
523 for (i = 0; i < am_hreq.opcount; i++) {
524 ops[i].am_error = strncpy_from_user(attr_name,
525 compat_ptr(ops[i].am_attrname),
526 MAXNAMELEN);
527 if (ops[i].am_error == 0 || ops[i].am_error == MAXNAMELEN)
528 error = -ERANGE;
529 if (ops[i].am_error < 0)
530 break;
531
532 switch (ops[i].am_opcode) {
533 case ATTR_OP_GET:
534 ops[i].am_error = xfs_attrmulti_attr_get(inode,
535 attr_name,
536 compat_ptr(ops[i].am_attrvalue),
537 &ops[i].am_length, ops[i].am_flags);
538 break;
539 case ATTR_OP_SET:
540 ops[i].am_error = xfs_attrmulti_attr_set(inode,
541 attr_name,
542 compat_ptr(ops[i].am_attrvalue),
543 ops[i].am_length, ops[i].am_flags);
544 break;
545 case ATTR_OP_REMOVE:
546 ops[i].am_error = xfs_attrmulti_attr_remove(inode,
547 attr_name, ops[i].am_flags);
548 break;
549 default:
550 ops[i].am_error = EINVAL;
551 }
552 }
553
554 if (copy_to_user(compat_ptr(am_hreq.ops), ops, size))
555 error = XFS_ERROR(EFAULT);
556
557 kfree(attr_name);
558 out_kfree_ops:
559 kfree(ops);
560 out_vn_rele:
561 iput(inode);
562 out:
563 return -error;
564}
565
479STATIC long 566STATIC long
480xfs_compat_ioctl( 567xfs_compat_ioctl(
481 xfs_inode_t *ip, 568 xfs_inode_t *ip,
@@ -499,10 +586,7 @@ xfs_compat_ioctl(
499 case XFS_IOC_GETBMAP: 586 case XFS_IOC_GETBMAP:
500 case XFS_IOC_GETBMAPA: 587 case XFS_IOC_GETBMAPA:
501 case XFS_IOC_GETBMAPX: 588 case XFS_IOC_GETBMAPX:
502/* not handled 589/* case XFS_IOC_FSSETDM_BY_HANDLE: not handled */
503 case XFS_IOC_FSSETDM_BY_HANDLE:
504 case XFS_IOC_ATTRMULTI_BY_HANDLE:
505*/
506 case XFS_IOC_FSCOUNTS: 590 case XFS_IOC_FSCOUNTS:
507 case XFS_IOC_SET_RESBLKS: 591 case XFS_IOC_SET_RESBLKS:
508 case XFS_IOC_GET_RESBLKS: 592 case XFS_IOC_GET_RESBLKS:
@@ -610,6 +694,8 @@ xfs_compat_ioctl(
610 } 694 }
611 case XFS_IOC_ATTRLIST_BY_HANDLE_32: 695 case XFS_IOC_ATTRLIST_BY_HANDLE_32:
612 return xfs_compat_attrlist_by_handle(mp, arg, inode); 696 return xfs_compat_attrlist_by_handle(mp, arg, inode);
697 case XFS_IOC_ATTRMULTI_BY_HANDLE_32:
698 return xfs_compat_attrmulti_by_handle(mp, arg, inode);
613 default: 699 default:
614 return -XFS_ERROR(ENOIOCTLCMD); 700 return -XFS_ERROR(ENOIOCTLCMD);
615 } 701 }