aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/mtd/ubi/cdev.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/mtd/ubi/cdev.c')
-rw-r--r--drivers/mtd/ubi/cdev.c95
1 files changed, 80 insertions, 15 deletions
diff --git a/drivers/mtd/ubi/cdev.c b/drivers/mtd/ubi/cdev.c
index 01978b57e9cb..a60a3a24c2a1 100644
--- a/drivers/mtd/ubi/cdev.c
+++ b/drivers/mtd/ubi/cdev.c
@@ -44,17 +44,6 @@
44#include <asm/div64.h> 44#include <asm/div64.h>
45#include "ubi.h" 45#include "ubi.h"
46 46
47/*
48 * Maximum sequence numbers of UBI and volume character device IOCTLs (direct
49 * logical eraseblock erase is a debug-only feature).
50 */
51#define UBI_CDEV_IOC_MAX_SEQ 2
52#ifndef CONFIG_MTD_UBI_DEBUG_USERSPACE_IO
53#define VOL_CDEV_IOC_MAX_SEQ 1
54#else
55#define VOL_CDEV_IOC_MAX_SEQ 2
56#endif
57
58/** 47/**
59 * get_exclusive - get exclusive access to an UBI volume. 48 * get_exclusive - get exclusive access to an UBI volume.
60 * @desc: volume descriptor 49 * @desc: volume descriptor
@@ -582,8 +571,7 @@ static int ubi_cdev_ioctl(struct inode *inode, struct file *file,
582 struct ubi_mkvol_req req; 571 struct ubi_mkvol_req req;
583 572
584 dbg_msg("create volume"); 573 dbg_msg("create volume");
585 err = copy_from_user(&req, argp, 574 err = copy_from_user(&req, argp, sizeof(struct ubi_mkvol_req));
586 sizeof(struct ubi_mkvol_req));
587 if (err) { 575 if (err) {
588 err = -EFAULT; 576 err = -EFAULT;
589 break; 577 break;
@@ -647,8 +635,7 @@ static int ubi_cdev_ioctl(struct inode *inode, struct file *file,
647 struct ubi_rsvol_req req; 635 struct ubi_rsvol_req req;
648 636
649 dbg_msg("re-size volume"); 637 dbg_msg("re-size volume");
650 err = copy_from_user(&req, argp, 638 err = copy_from_user(&req, argp, sizeof(struct ubi_rsvol_req));
651 sizeof(struct ubi_rsvol_req));
652 if (err) { 639 if (err) {
653 err = -EFAULT; 640 err = -EFAULT;
654 break; 641 break;
@@ -684,8 +671,86 @@ static int ubi_cdev_ioctl(struct inode *inode, struct file *file,
684 return err; 671 return err;
685} 672}
686 673
674static int ctrl_cdev_ioctl(struct inode *inode, struct file *file,
675 unsigned int cmd, unsigned long arg)
676{
677 int err = 0;
678 void __user *argp = (void __user *)arg;
679
680 if (!capable(CAP_SYS_RESOURCE))
681 return -EPERM;
682
683 switch (cmd) {
684 /* Attach an MTD device command */
685 case UBI_IOCATT:
686 {
687 struct ubi_attach_req req;
688 struct mtd_info *mtd;
689
690 dbg_msg("attach MTD device");
691 err = copy_from_user(&req, argp, sizeof(struct ubi_attach_req));
692 if (err) {
693 err = -EFAULT;
694 break;
695 }
696
697 if (req.mtd_num < 0 ||
698 (req.ubi_num < 0 && req.ubi_num != UBI_DEV_NUM_AUTO)) {
699 err = -EINVAL;
700 break;
701 }
702
703 mtd = get_mtd_device(NULL, req.mtd_num);
704 if (IS_ERR(mtd)) {
705 err = PTR_ERR(mtd);
706 break;
707 }
708
709 /*
710 * Note, further request verification is done by
711 * 'ubi_attach_mtd_dev()'.
712 */
713 mutex_lock(&ubi_devices_mutex);
714 err = ubi_attach_mtd_dev(mtd, req.ubi_num, req.vid_hdr_offset);
715 mutex_unlock(&ubi_devices_mutex);
716 if (err < 0)
717 put_mtd_device(mtd);
718 else
719 /* @err contains UBI device number */
720 err = put_user(err, (__user int32_t *)argp);
721
722 break;
723 }
724
725 /* Detach an MTD device command */
726 case UBI_IOCDET:
727 {
728 int ubi_num;
729
730 dbg_msg("dettach MTD device");
731 err = get_user(ubi_num, (__user int32_t *)argp);
732 if (err) {
733 err = -EFAULT;
734 break;
735 }
736
737 mutex_lock(&ubi_devices_mutex);
738 err = ubi_detach_mtd_dev(ubi_num, 0);
739 mutex_unlock(&ubi_devices_mutex);
740 break;
741 }
742
743 default:
744 err = -ENOTTY;
745 break;
746 }
747
748 return err;
749}
750
687/* UBI control character device operations */ 751/* UBI control character device operations */
688struct file_operations ubi_ctrl_cdev_operations = { 752struct file_operations ubi_ctrl_cdev_operations = {
753 .ioctl = ctrl_cdev_ioctl,
689 .owner = THIS_MODULE, 754 .owner = THIS_MODULE,
690}; 755};
691 756