xref: /openbmc/linux/drivers/mtd/ubi/cdev.c (revision 669d2044)
11a59d1b8SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
2801c135cSArtem B. Bityutskiy /*
3801c135cSArtem B. Bityutskiy  * Copyright (c) International Business Machines Corp., 2006
4801c135cSArtem B. Bityutskiy  *
5801c135cSArtem B. Bityutskiy  * Author: Artem Bityutskiy (Битюцкий Артём)
6801c135cSArtem B. Bityutskiy  */
7801c135cSArtem B. Bityutskiy 
8801c135cSArtem B. Bityutskiy /*
9801c135cSArtem B. Bityutskiy  * This file includes implementation of UBI character device operations.
10801c135cSArtem B. Bityutskiy  *
11801c135cSArtem B. Bityutskiy  * There are two kinds of character devices in UBI: UBI character devices and
12801c135cSArtem B. Bityutskiy  * UBI volume character devices. UBI character devices allow users to
13801c135cSArtem B. Bityutskiy  * manipulate whole volumes: create, remove, and re-size them. Volume character
14801c135cSArtem B. Bityutskiy  * devices provide volume I/O capabilities.
15801c135cSArtem B. Bityutskiy  *
16801c135cSArtem B. Bityutskiy  * Major and minor numbers are assigned dynamically to both UBI and volume
17801c135cSArtem B. Bityutskiy  * character devices.
189f961b57SArtem Bityutskiy  *
199f961b57SArtem Bityutskiy  * Well, there is the third kind of character devices - the UBI control
209f961b57SArtem Bityutskiy  * character device, which allows to manipulate by UBI devices - create and
219f961b57SArtem Bityutskiy  * delete them. In other words, it is used for attaching and detaching MTD
229f961b57SArtem Bityutskiy  * devices.
23801c135cSArtem B. Bityutskiy  */
24801c135cSArtem B. Bityutskiy 
25801c135cSArtem B. Bityutskiy #include <linux/module.h>
26801c135cSArtem B. Bityutskiy #include <linux/stat.h>
275a0e3ad6STejun Heo #include <linux/slab.h>
28801c135cSArtem B. Bityutskiy #include <linux/ioctl.h>
29801c135cSArtem B. Bityutskiy #include <linux/capability.h>
309c9ec147SArtem Bityutskiy #include <linux/uaccess.h>
31f429b2eaSArtem Bityutskiy #include <linux/compat.h>
323013ee31SArtem Bityutskiy #include <linux/math64.h>
33801c135cSArtem B. Bityutskiy #include <mtd/ubi-user.h>
34801c135cSArtem B. Bityutskiy #include "ubi.h"
35801c135cSArtem B. Bityutskiy 
36801c135cSArtem B. Bityutskiy /**
37801c135cSArtem B. Bityutskiy  * get_exclusive - get exclusive access to an UBI volume.
38801c135cSArtem B. Bityutskiy  * @desc: volume descriptor
39801c135cSArtem B. Bityutskiy  *
40801c135cSArtem B. Bityutskiy  * This function changes UBI volume open mode to "exclusive". Returns previous
41801c135cSArtem B. Bityutskiy  * mode value (positive integer) in case of success and a negative error code
42801c135cSArtem B. Bityutskiy  * in case of failure.
43801c135cSArtem B. Bityutskiy  */
get_exclusive(struct ubi_volume_desc * desc)4445fc5c81STanya Brokhman static int get_exclusive(struct ubi_volume_desc *desc)
45801c135cSArtem B. Bityutskiy {
46801c135cSArtem B. Bityutskiy 	int users, err;
47801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
48801c135cSArtem B. Bityutskiy 
49801c135cSArtem B. Bityutskiy 	spin_lock(&vol->ubi->volumes_lock);
50fafdd2bfSRichard Weinberger 	users = vol->readers + vol->writers + vol->exclusive + vol->metaonly;
51801c135cSArtem B. Bityutskiy 	ubi_assert(users > 0);
52801c135cSArtem B. Bityutskiy 	if (users > 1) {
5345fc5c81STanya Brokhman 		ubi_err(vol->ubi, "%d users for volume %d", users, vol->vol_id);
54801c135cSArtem B. Bityutskiy 		err = -EBUSY;
55801c135cSArtem B. Bityutskiy 	} else {
56fafdd2bfSRichard Weinberger 		vol->readers = vol->writers = vol->metaonly = 0;
57801c135cSArtem B. Bityutskiy 		vol->exclusive = 1;
58801c135cSArtem B. Bityutskiy 		err = desc->mode;
59801c135cSArtem B. Bityutskiy 		desc->mode = UBI_EXCLUSIVE;
60801c135cSArtem B. Bityutskiy 	}
61801c135cSArtem B. Bityutskiy 	spin_unlock(&vol->ubi->volumes_lock);
62801c135cSArtem B. Bityutskiy 
63801c135cSArtem B. Bityutskiy 	return err;
64801c135cSArtem B. Bityutskiy }
65801c135cSArtem B. Bityutskiy 
66801c135cSArtem B. Bityutskiy /**
67801c135cSArtem B. Bityutskiy  * revoke_exclusive - revoke exclusive mode.
68801c135cSArtem B. Bityutskiy  * @desc: volume descriptor
69801c135cSArtem B. Bityutskiy  * @mode: new mode to switch to
70801c135cSArtem B. Bityutskiy  */
revoke_exclusive(struct ubi_volume_desc * desc,int mode)71801c135cSArtem B. Bityutskiy static void revoke_exclusive(struct ubi_volume_desc *desc, int mode)
72801c135cSArtem B. Bityutskiy {
73801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
74801c135cSArtem B. Bityutskiy 
75801c135cSArtem B. Bityutskiy 	spin_lock(&vol->ubi->volumes_lock);
76fafdd2bfSRichard Weinberger 	ubi_assert(vol->readers == 0 && vol->writers == 0 && vol->metaonly == 0);
77801c135cSArtem B. Bityutskiy 	ubi_assert(vol->exclusive == 1 && desc->mode == UBI_EXCLUSIVE);
78801c135cSArtem B. Bityutskiy 	vol->exclusive = 0;
79801c135cSArtem B. Bityutskiy 	if (mode == UBI_READONLY)
80801c135cSArtem B. Bityutskiy 		vol->readers = 1;
81801c135cSArtem B. Bityutskiy 	else if (mode == UBI_READWRITE)
82801c135cSArtem B. Bityutskiy 		vol->writers = 1;
83fafdd2bfSRichard Weinberger 	else if (mode == UBI_METAONLY)
84fafdd2bfSRichard Weinberger 		vol->metaonly = 1;
85801c135cSArtem B. Bityutskiy 	else
86801c135cSArtem B. Bityutskiy 		vol->exclusive = 1;
87801c135cSArtem B. Bityutskiy 	spin_unlock(&vol->ubi->volumes_lock);
88801c135cSArtem B. Bityutskiy 
89801c135cSArtem B. Bityutskiy 	desc->mode = mode;
90801c135cSArtem B. Bityutskiy }
91801c135cSArtem B. Bityutskiy 
vol_cdev_open(struct inode * inode,struct file * file)92801c135cSArtem B. Bityutskiy static int vol_cdev_open(struct inode *inode, struct file *file)
93801c135cSArtem B. Bityutskiy {
94801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc;
95e73f4459SArtem Bityutskiy 	int vol_id = iminor(inode) - 1, mode, ubi_num;
96e73f4459SArtem Bityutskiy 
97e73f4459SArtem Bityutskiy 	ubi_num = ubi_major2num(imajor(inode));
987d200e88SArtem Bityutskiy 	if (ubi_num < 0)
99e73f4459SArtem Bityutskiy 		return ubi_num;
100801c135cSArtem B. Bityutskiy 
101801c135cSArtem B. Bityutskiy 	if (file->f_mode & FMODE_WRITE)
102801c135cSArtem B. Bityutskiy 		mode = UBI_READWRITE;
103801c135cSArtem B. Bityutskiy 	else
104801c135cSArtem B. Bityutskiy 		mode = UBI_READONLY;
105801c135cSArtem B. Bityutskiy 
106e1cf7e6dSArtem Bityutskiy 	dbg_gen("open device %d, volume %d, mode %d",
107e1cf7e6dSArtem Bityutskiy 		ubi_num, vol_id, mode);
108801c135cSArtem B. Bityutskiy 
109e73f4459SArtem Bityutskiy 	desc = ubi_open_volume(ubi_num, vol_id, mode);
110801c135cSArtem B. Bityutskiy 	if (IS_ERR(desc))
111801c135cSArtem B. Bityutskiy 		return PTR_ERR(desc);
112801c135cSArtem B. Bityutskiy 
113801c135cSArtem B. Bityutskiy 	file->private_data = desc;
114801c135cSArtem B. Bityutskiy 	return 0;
115801c135cSArtem B. Bityutskiy }
116801c135cSArtem B. Bityutskiy 
vol_cdev_release(struct inode * inode,struct file * file)117801c135cSArtem B. Bityutskiy static int vol_cdev_release(struct inode *inode, struct file *file)
118801c135cSArtem B. Bityutskiy {
119801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
120801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
121801c135cSArtem B. Bityutskiy 
122e1cf7e6dSArtem Bityutskiy 	dbg_gen("release device %d, volume %d, mode %d",
123e1cf7e6dSArtem Bityutskiy 		vol->ubi->ubi_num, vol->vol_id, desc->mode);
124801c135cSArtem B. Bityutskiy 
125801c135cSArtem B. Bityutskiy 	if (vol->updating) {
12632608703STanya Brokhman 		ubi_warn(vol->ubi, "update of volume %d not finished, volume is damaged",
127801c135cSArtem B. Bityutskiy 			 vol->vol_id);
128e653879cSArtem Bityutskiy 		ubi_assert(!vol->changing_leb);
129801c135cSArtem B. Bityutskiy 		vol->updating = 0;
13092ad8f37SArtem Bityutskiy 		vfree(vol->upd_buf);
131e653879cSArtem Bityutskiy 	} else if (vol->changing_leb) {
132049333ceSArtem Bityutskiy 		dbg_gen("only %lld of %lld bytes received for atomic LEB change for volume %d:%d, cancel",
133049333ceSArtem Bityutskiy 			vol->upd_received, vol->upd_bytes, vol->ubi->ubi_num,
134049333ceSArtem Bityutskiy 			vol->vol_id);
135e653879cSArtem Bityutskiy 		vol->changing_leb = 0;
136e653879cSArtem Bityutskiy 		vfree(vol->upd_buf);
137801c135cSArtem B. Bityutskiy 	}
138801c135cSArtem B. Bityutskiy 
139801c135cSArtem B. Bityutskiy 	ubi_close_volume(desc);
140801c135cSArtem B. Bityutskiy 	return 0;
141801c135cSArtem B. Bityutskiy }
142801c135cSArtem B. Bityutskiy 
vol_cdev_llseek(struct file * file,loff_t offset,int origin)143801c135cSArtem B. Bityutskiy static loff_t vol_cdev_llseek(struct file *file, loff_t offset, int origin)
144801c135cSArtem B. Bityutskiy {
145801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
146801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
147801c135cSArtem B. Bityutskiy 
148801c135cSArtem B. Bityutskiy 	if (vol->updating) {
149801c135cSArtem B. Bityutskiy 		/* Update is in progress, seeking is prohibited */
15032608703STanya Brokhman 		ubi_err(vol->ubi, "updating");
151801c135cSArtem B. Bityutskiy 		return -EBUSY;
152801c135cSArtem B. Bityutskiy 	}
153801c135cSArtem B. Bityutskiy 
1544a1f2f38SAl Viro 	return fixed_size_llseek(file, offset, origin, vol->used_bytes);
155801c135cSArtem B. Bityutskiy }
156801c135cSArtem B. Bityutskiy 
vol_cdev_fsync(struct file * file,loff_t start,loff_t end,int datasync)157049333ceSArtem Bityutskiy static int vol_cdev_fsync(struct file *file, loff_t start, loff_t end,
158049333ceSArtem Bityutskiy 			  int datasync)
1591b24bc3aSCorentin Chary {
1601b24bc3aSCorentin Chary 	struct ubi_volume_desc *desc = file->private_data;
1611b24bc3aSCorentin Chary 	struct ubi_device *ubi = desc->vol->ubi;
162496ad9aaSAl Viro 	struct inode *inode = file_inode(file);
16302c24a82SJosef Bacik 	int err;
1645955102cSAl Viro 	inode_lock(inode);
16502c24a82SJosef Bacik 	err = ubi_sync(ubi->ubi_num);
1665955102cSAl Viro 	inode_unlock(inode);
16702c24a82SJosef Bacik 	return err;
1681b24bc3aSCorentin Chary }
1691b24bc3aSCorentin Chary 
1701b24bc3aSCorentin Chary 
vol_cdev_read(struct file * file,__user char * buf,size_t count,loff_t * offp)171801c135cSArtem B. Bityutskiy static ssize_t vol_cdev_read(struct file *file, __user char *buf, size_t count,
172801c135cSArtem B. Bityutskiy 			     loff_t *offp)
173801c135cSArtem B. Bityutskiy {
174801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
175801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
176801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
177ae616e1bSArtem Bityutskiy 	int err, lnum, off, len,  tbuf_size;
178801c135cSArtem B. Bityutskiy 	size_t count_save = count;
179801c135cSArtem B. Bityutskiy 	void *tbuf;
180801c135cSArtem B. Bityutskiy 
181c8566350SArtem Bityutskiy 	dbg_gen("read %zd bytes from offset %lld of volume %d",
182ae616e1bSArtem Bityutskiy 		count, *offp, vol->vol_id);
183801c135cSArtem B. Bityutskiy 
184801c135cSArtem B. Bityutskiy 	if (vol->updating) {
18532608703STanya Brokhman 		ubi_err(vol->ubi, "updating");
186801c135cSArtem B. Bityutskiy 		return -EBUSY;
187801c135cSArtem B. Bityutskiy 	}
188801c135cSArtem B. Bityutskiy 	if (vol->upd_marker) {
18932608703STanya Brokhman 		ubi_err(vol->ubi, "damaged volume, update marker is set");
190801c135cSArtem B. Bityutskiy 		return -EBADF;
191801c135cSArtem B. Bityutskiy 	}
192801c135cSArtem B. Bityutskiy 	if (*offp == vol->used_bytes || count == 0)
193801c135cSArtem B. Bityutskiy 		return 0;
194801c135cSArtem B. Bityutskiy 
195801c135cSArtem B. Bityutskiy 	if (vol->corrupted)
196c8566350SArtem Bityutskiy 		dbg_gen("read from corrupted volume %d", vol->vol_id);
197801c135cSArtem B. Bityutskiy 
198801c135cSArtem B. Bityutskiy 	if (*offp + count > vol->used_bytes)
199801c135cSArtem B. Bityutskiy 		count_save = count = vol->used_bytes - *offp;
200801c135cSArtem B. Bityutskiy 
201801c135cSArtem B. Bityutskiy 	tbuf_size = vol->usable_leb_size;
202801c135cSArtem B. Bityutskiy 	if (count < tbuf_size)
203801c135cSArtem B. Bityutskiy 		tbuf_size = ALIGN(count, ubi->min_io_size);
20492ad8f37SArtem Bityutskiy 	tbuf = vmalloc(tbuf_size);
205801c135cSArtem B. Bityutskiy 	if (!tbuf)
206801c135cSArtem B. Bityutskiy 		return -ENOMEM;
207801c135cSArtem B. Bityutskiy 
208801c135cSArtem B. Bityutskiy 	len = count > tbuf_size ? tbuf_size : count;
2093013ee31SArtem Bityutskiy 	lnum = div_u64_rem(*offp, vol->usable_leb_size, &off);
210801c135cSArtem B. Bityutskiy 
211801c135cSArtem B. Bityutskiy 	do {
212801c135cSArtem B. Bityutskiy 		cond_resched();
213801c135cSArtem B. Bityutskiy 
214801c135cSArtem B. Bityutskiy 		if (off + len >= vol->usable_leb_size)
215801c135cSArtem B. Bityutskiy 			len = vol->usable_leb_size - off;
216801c135cSArtem B. Bityutskiy 
21789b96b69SArtem Bityutskiy 		err = ubi_eba_read_leb(ubi, vol, lnum, tbuf, off, len, 0);
218801c135cSArtem B. Bityutskiy 		if (err)
219801c135cSArtem B. Bityutskiy 			break;
220801c135cSArtem B. Bityutskiy 
221801c135cSArtem B. Bityutskiy 		off += len;
222801c135cSArtem B. Bityutskiy 		if (off == vol->usable_leb_size) {
223801c135cSArtem B. Bityutskiy 			lnum += 1;
224801c135cSArtem B. Bityutskiy 			off -= vol->usable_leb_size;
225801c135cSArtem B. Bityutskiy 		}
226801c135cSArtem B. Bityutskiy 
227801c135cSArtem B. Bityutskiy 		count -= len;
228801c135cSArtem B. Bityutskiy 		*offp += len;
229801c135cSArtem B. Bityutskiy 
230801c135cSArtem B. Bityutskiy 		err = copy_to_user(buf, tbuf, len);
231801c135cSArtem B. Bityutskiy 		if (err) {
232801c135cSArtem B. Bityutskiy 			err = -EFAULT;
233801c135cSArtem B. Bityutskiy 			break;
234801c135cSArtem B. Bityutskiy 		}
235801c135cSArtem B. Bityutskiy 
236801c135cSArtem B. Bityutskiy 		buf += len;
237801c135cSArtem B. Bityutskiy 		len = count > tbuf_size ? tbuf_size : count;
238801c135cSArtem B. Bityutskiy 	} while (count);
239801c135cSArtem B. Bityutskiy 
24092ad8f37SArtem Bityutskiy 	vfree(tbuf);
241801c135cSArtem B. Bityutskiy 	return err ? err : count_save - count;
242801c135cSArtem B. Bityutskiy }
243801c135cSArtem B. Bityutskiy 
244801c135cSArtem B. Bityutskiy /*
245801c135cSArtem B. Bityutskiy  * This function allows to directly write to dynamic UBI volumes, without
246766fb95bSSidney Amani  * issuing the volume update operation.
247801c135cSArtem B. Bityutskiy  */
vol_cdev_direct_write(struct file * file,const char __user * buf,size_t count,loff_t * offp)248801c135cSArtem B. Bityutskiy static ssize_t vol_cdev_direct_write(struct file *file, const char __user *buf,
249801c135cSArtem B. Bityutskiy 				     size_t count, loff_t *offp)
250801c135cSArtem B. Bityutskiy {
251801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
252801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
253801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
254ae616e1bSArtem Bityutskiy 	int lnum, off, len, tbuf_size, err = 0;
255801c135cSArtem B. Bityutskiy 	size_t count_save = count;
256801c135cSArtem B. Bityutskiy 	char *tbuf;
257801c135cSArtem B. Bityutskiy 
258766fb95bSSidney Amani 	if (!vol->direct_writes)
259766fb95bSSidney Amani 		return -EPERM;
260766fb95bSSidney Amani 
261c8566350SArtem Bityutskiy 	dbg_gen("requested: write %zd bytes to offset %lld of volume %u",
262ae616e1bSArtem Bityutskiy 		count, *offp, vol->vol_id);
263801c135cSArtem B. Bityutskiy 
264801c135cSArtem B. Bityutskiy 	if (vol->vol_type == UBI_STATIC_VOLUME)
265801c135cSArtem B. Bityutskiy 		return -EROFS;
266801c135cSArtem B. Bityutskiy 
2673013ee31SArtem Bityutskiy 	lnum = div_u64_rem(*offp, vol->usable_leb_size, &off);
268cadb40ccSKyungmin Park 	if (off & (ubi->min_io_size - 1)) {
26932608703STanya Brokhman 		ubi_err(ubi, "unaligned position");
270801c135cSArtem B. Bityutskiy 		return -EINVAL;
271801c135cSArtem B. Bityutskiy 	}
272801c135cSArtem B. Bityutskiy 
273801c135cSArtem B. Bityutskiy 	if (*offp + count > vol->used_bytes)
274801c135cSArtem B. Bityutskiy 		count_save = count = vol->used_bytes - *offp;
275801c135cSArtem B. Bityutskiy 
276801c135cSArtem B. Bityutskiy 	/* We can write only in fractions of the minimum I/O unit */
277cadb40ccSKyungmin Park 	if (count & (ubi->min_io_size - 1)) {
27832608703STanya Brokhman 		ubi_err(ubi, "unaligned write length");
279801c135cSArtem B. Bityutskiy 		return -EINVAL;
280801c135cSArtem B. Bityutskiy 	}
281801c135cSArtem B. Bityutskiy 
282801c135cSArtem B. Bityutskiy 	tbuf_size = vol->usable_leb_size;
283801c135cSArtem B. Bityutskiy 	if (count < tbuf_size)
284801c135cSArtem B. Bityutskiy 		tbuf_size = ALIGN(count, ubi->min_io_size);
28592ad8f37SArtem Bityutskiy 	tbuf = vmalloc(tbuf_size);
286801c135cSArtem B. Bityutskiy 	if (!tbuf)
287801c135cSArtem B. Bityutskiy 		return -ENOMEM;
288801c135cSArtem B. Bityutskiy 
289801c135cSArtem B. Bityutskiy 	len = count > tbuf_size ? tbuf_size : count;
290801c135cSArtem B. Bityutskiy 
291801c135cSArtem B. Bityutskiy 	while (count) {
292801c135cSArtem B. Bityutskiy 		cond_resched();
293801c135cSArtem B. Bityutskiy 
294801c135cSArtem B. Bityutskiy 		if (off + len >= vol->usable_leb_size)
295801c135cSArtem B. Bityutskiy 			len = vol->usable_leb_size - off;
296801c135cSArtem B. Bityutskiy 
297801c135cSArtem B. Bityutskiy 		err = copy_from_user(tbuf, buf, len);
298801c135cSArtem B. Bityutskiy 		if (err) {
299801c135cSArtem B. Bityutskiy 			err = -EFAULT;
300801c135cSArtem B. Bityutskiy 			break;
301801c135cSArtem B. Bityutskiy 		}
302801c135cSArtem B. Bityutskiy 
303b36a261eSRichard Weinberger 		err = ubi_eba_write_leb(ubi, vol, lnum, tbuf, off, len);
304801c135cSArtem B. Bityutskiy 		if (err)
305801c135cSArtem B. Bityutskiy 			break;
306801c135cSArtem B. Bityutskiy 
307801c135cSArtem B. Bityutskiy 		off += len;
308801c135cSArtem B. Bityutskiy 		if (off == vol->usable_leb_size) {
309801c135cSArtem B. Bityutskiy 			lnum += 1;
310801c135cSArtem B. Bityutskiy 			off -= vol->usable_leb_size;
311801c135cSArtem B. Bityutskiy 		}
312801c135cSArtem B. Bityutskiy 
313801c135cSArtem B. Bityutskiy 		count -= len;
314801c135cSArtem B. Bityutskiy 		*offp += len;
315801c135cSArtem B. Bityutskiy 		buf += len;
316801c135cSArtem B. Bityutskiy 		len = count > tbuf_size ? tbuf_size : count;
317801c135cSArtem B. Bityutskiy 	}
318801c135cSArtem B. Bityutskiy 
31992ad8f37SArtem Bityutskiy 	vfree(tbuf);
320801c135cSArtem B. Bityutskiy 	return err ? err : count_save - count;
321801c135cSArtem B. Bityutskiy }
322801c135cSArtem B. Bityutskiy 
vol_cdev_write(struct file * file,const char __user * buf,size_t count,loff_t * offp)323801c135cSArtem B. Bityutskiy static ssize_t vol_cdev_write(struct file *file, const char __user *buf,
324801c135cSArtem B. Bityutskiy 			      size_t count, loff_t *offp)
325801c135cSArtem B. Bityutskiy {
326801c135cSArtem B. Bityutskiy 	int err = 0;
327801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
328801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
329801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
330801c135cSArtem B. Bityutskiy 
331e653879cSArtem Bityutskiy 	if (!vol->updating && !vol->changing_leb)
332801c135cSArtem B. Bityutskiy 		return vol_cdev_direct_write(file, buf, count, offp);
333801c135cSArtem B. Bityutskiy 
334e653879cSArtem Bityutskiy 	if (vol->updating)
3351b68d0eeSArtem Bityutskiy 		err = ubi_more_update_data(ubi, vol, buf, count);
336e653879cSArtem Bityutskiy 	else
337e653879cSArtem Bityutskiy 		err = ubi_more_leb_change_data(ubi, vol, buf, count);
338e653879cSArtem Bityutskiy 
339801c135cSArtem B. Bityutskiy 	if (err < 0) {
34032608703STanya Brokhman 		ubi_err(ubi, "cannot accept more %zd bytes of data, error %d",
34101f7b309SArtem Bityutskiy 			count, err);
342801c135cSArtem B. Bityutskiy 		return err;
343801c135cSArtem B. Bityutskiy 	}
344801c135cSArtem B. Bityutskiy 
345801c135cSArtem B. Bityutskiy 	if (err) {
346801c135cSArtem B. Bityutskiy 		/*
347e653879cSArtem Bityutskiy 		 * The operation is finished, @err contains number of actually
348e653879cSArtem Bityutskiy 		 * written bytes.
349801c135cSArtem B. Bityutskiy 		 */
350801c135cSArtem B. Bityutskiy 		count = err;
351801c135cSArtem B. Bityutskiy 
352e653879cSArtem Bityutskiy 		if (vol->changing_leb) {
353e653879cSArtem Bityutskiy 			revoke_exclusive(desc, UBI_READWRITE);
354e653879cSArtem Bityutskiy 			return count;
355e653879cSArtem Bityutskiy 		}
356e653879cSArtem Bityutskiy 
357c355aa46SQuentin Schulz 		/*
358c355aa46SQuentin Schulz 		 * We voluntarily do not take into account the skip_check flag
359c355aa46SQuentin Schulz 		 * as we want to make sure what we wrote was correctly written.
360c355aa46SQuentin Schulz 		 */
361801c135cSArtem B. Bityutskiy 		err = ubi_check_volume(ubi, vol->vol_id);
362801c135cSArtem B. Bityutskiy 		if (err < 0)
363801c135cSArtem B. Bityutskiy 			return err;
364801c135cSArtem B. Bityutskiy 
365801c135cSArtem B. Bityutskiy 		if (err) {
36632608703STanya Brokhman 			ubi_warn(ubi, "volume %d on UBI device %d is corrupted",
367801c135cSArtem B. Bityutskiy 				 vol->vol_id, ubi->ubi_num);
368801c135cSArtem B. Bityutskiy 			vol->corrupted = 1;
369801c135cSArtem B. Bityutskiy 		}
370801c135cSArtem B. Bityutskiy 		vol->checked = 1;
3710e0ee1ccSDmitry Pervushin 		ubi_volume_notify(ubi, vol, UBI_VOLUME_UPDATED);
372801c135cSArtem B. Bityutskiy 		revoke_exclusive(desc, UBI_READWRITE);
373801c135cSArtem B. Bityutskiy 	}
374801c135cSArtem B. Bityutskiy 
375801c135cSArtem B. Bityutskiy 	return count;
376801c135cSArtem B. Bityutskiy }
377801c135cSArtem B. Bityutskiy 
vol_cdev_ioctl(struct file * file,unsigned int cmd,unsigned long arg)378f429b2eaSArtem Bityutskiy static long vol_cdev_ioctl(struct file *file, unsigned int cmd,
379f429b2eaSArtem Bityutskiy 			   unsigned long arg)
380801c135cSArtem B. Bityutskiy {
381801c135cSArtem B. Bityutskiy 	int err = 0;
382801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
383801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
384801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
385801c135cSArtem B. Bityutskiy 	void __user *argp = (void __user *)arg;
386801c135cSArtem B. Bityutskiy 
387801c135cSArtem B. Bityutskiy 	switch (cmd) {
388801c135cSArtem B. Bityutskiy 	/* Volume update command */
389801c135cSArtem B. Bityutskiy 	case UBI_IOCVOLUP:
390801c135cSArtem B. Bityutskiy 	{
391801c135cSArtem B. Bityutskiy 		int64_t bytes, rsvd_bytes;
392801c135cSArtem B. Bityutskiy 
393801c135cSArtem B. Bityutskiy 		if (!capable(CAP_SYS_RESOURCE)) {
394801c135cSArtem B. Bityutskiy 			err = -EPERM;
395801c135cSArtem B. Bityutskiy 			break;
396801c135cSArtem B. Bityutskiy 		}
397801c135cSArtem B. Bityutskiy 
398801c135cSArtem B. Bityutskiy 		err = copy_from_user(&bytes, argp, sizeof(int64_t));
399801c135cSArtem B. Bityutskiy 		if (err) {
400801c135cSArtem B. Bityutskiy 			err = -EFAULT;
401801c135cSArtem B. Bityutskiy 			break;
402801c135cSArtem B. Bityutskiy 		}
403801c135cSArtem B. Bityutskiy 
404801c135cSArtem B. Bityutskiy 		if (desc->mode == UBI_READONLY) {
405801c135cSArtem B. Bityutskiy 			err = -EROFS;
406801c135cSArtem B. Bityutskiy 			break;
407801c135cSArtem B. Bityutskiy 		}
408801c135cSArtem B. Bityutskiy 
40973789a3dSBruce Leonard 		rsvd_bytes = (long long)vol->reserved_pebs *
4105f09aaa9SBoris Brezillon 					vol->usable_leb_size;
411801c135cSArtem B. Bityutskiy 		if (bytes < 0 || bytes > rsvd_bytes) {
412801c135cSArtem B. Bityutskiy 			err = -EINVAL;
413801c135cSArtem B. Bityutskiy 			break;
414801c135cSArtem B. Bityutskiy 		}
415801c135cSArtem B. Bityutskiy 
41645fc5c81STanya Brokhman 		err = get_exclusive(desc);
417801c135cSArtem B. Bityutskiy 		if (err < 0)
418801c135cSArtem B. Bityutskiy 			break;
419801c135cSArtem B. Bityutskiy 
4201b68d0eeSArtem Bityutskiy 		err = ubi_start_update(ubi, vol, bytes);
421fda322a1SEzequiel Garcia 		if (bytes == 0) {
422fda322a1SEzequiel Garcia 			ubi_volume_notify(ubi, vol, UBI_VOLUME_UPDATED);
423801c135cSArtem B. Bityutskiy 			revoke_exclusive(desc, UBI_READWRITE);
424fda322a1SEzequiel Garcia 		}
425801c135cSArtem B. Bityutskiy 		break;
426801c135cSArtem B. Bityutskiy 	}
427801c135cSArtem B. Bityutskiy 
428e653879cSArtem Bityutskiy 	/* Atomic logical eraseblock change command */
429e653879cSArtem Bityutskiy 	case UBI_IOCEBCH:
430e653879cSArtem Bityutskiy 	{
431e653879cSArtem Bityutskiy 		struct ubi_leb_change_req req;
432e653879cSArtem Bityutskiy 
433e653879cSArtem Bityutskiy 		err = copy_from_user(&req, argp,
434e653879cSArtem Bityutskiy 				     sizeof(struct ubi_leb_change_req));
435e653879cSArtem Bityutskiy 		if (err) {
436e653879cSArtem Bityutskiy 			err = -EFAULT;
437e653879cSArtem Bityutskiy 			break;
438e653879cSArtem Bityutskiy 		}
439e653879cSArtem Bityutskiy 
440e653879cSArtem Bityutskiy 		if (desc->mode == UBI_READONLY ||
441e653879cSArtem Bityutskiy 		    vol->vol_type == UBI_STATIC_VOLUME) {
442e653879cSArtem Bityutskiy 			err = -EROFS;
443e653879cSArtem Bityutskiy 			break;
444e653879cSArtem Bityutskiy 		}
445e653879cSArtem Bityutskiy 
446e653879cSArtem Bityutskiy 		/* Validate the request */
447e653879cSArtem Bityutskiy 		err = -EINVAL;
4489a5f09acSBoris Brezillon 		if (!ubi_leb_valid(vol, req.lnum) ||
449299d0c5bSBrian Norris 		    req.bytes < 0 || req.bytes > vol->usable_leb_size)
450e653879cSArtem Bityutskiy 			break;
451e653879cSArtem Bityutskiy 
45245fc5c81STanya Brokhman 		err = get_exclusive(desc);
453e653879cSArtem Bityutskiy 		if (err < 0)
454e653879cSArtem Bityutskiy 			break;
455e653879cSArtem Bityutskiy 
456e653879cSArtem Bityutskiy 		err = ubi_start_leb_change(ubi, vol, &req);
457e653879cSArtem Bityutskiy 		if (req.bytes == 0)
458e653879cSArtem Bityutskiy 			revoke_exclusive(desc, UBI_READWRITE);
459e653879cSArtem Bityutskiy 		break;
460e653879cSArtem Bityutskiy 	}
461e653879cSArtem Bityutskiy 
462801c135cSArtem B. Bityutskiy 	/* Logical eraseblock erasure command */
463801c135cSArtem B. Bityutskiy 	case UBI_IOCEBER:
464801c135cSArtem B. Bityutskiy 	{
465801c135cSArtem B. Bityutskiy 		int32_t lnum;
466801c135cSArtem B. Bityutskiy 
467bf07803aSChristoph Hellwig 		err = get_user(lnum, (__user int32_t *)argp);
468801c135cSArtem B. Bityutskiy 		if (err) {
469801c135cSArtem B. Bityutskiy 			err = -EFAULT;
470801c135cSArtem B. Bityutskiy 			break;
471801c135cSArtem B. Bityutskiy 		}
472801c135cSArtem B. Bityutskiy 
473e653879cSArtem Bityutskiy 		if (desc->mode == UBI_READONLY ||
474e653879cSArtem Bityutskiy 		    vol->vol_type == UBI_STATIC_VOLUME) {
475801c135cSArtem B. Bityutskiy 			err = -EROFS;
476801c135cSArtem B. Bityutskiy 			break;
477801c135cSArtem B. Bityutskiy 		}
478801c135cSArtem B. Bityutskiy 
4799a5f09acSBoris Brezillon 		if (!ubi_leb_valid(vol, lnum)) {
480801c135cSArtem B. Bityutskiy 			err = -EINVAL;
481801c135cSArtem B. Bityutskiy 			break;
482801c135cSArtem B. Bityutskiy 		}
483801c135cSArtem B. Bityutskiy 
484c8566350SArtem Bityutskiy 		dbg_gen("erase LEB %d:%d", vol->vol_id, lnum);
48589b96b69SArtem Bityutskiy 		err = ubi_eba_unmap_leb(ubi, vol, lnum);
486801c135cSArtem B. Bityutskiy 		if (err)
487801c135cSArtem B. Bityutskiy 			break;
488801c135cSArtem B. Bityutskiy 
48962f38455SJoel Reardon 		err = ubi_wl_flush(ubi, UBI_ALL, UBI_ALL);
490801c135cSArtem B. Bityutskiy 		break;
491801c135cSArtem B. Bityutskiy 	}
492141e6ebdSCorentin Chary 
493141e6ebdSCorentin Chary 	/* Logical eraseblock map command */
494141e6ebdSCorentin Chary 	case UBI_IOCEBMAP:
495141e6ebdSCorentin Chary 	{
496141e6ebdSCorentin Chary 		struct ubi_map_req req;
497141e6ebdSCorentin Chary 
498141e6ebdSCorentin Chary 		err = copy_from_user(&req, argp, sizeof(struct ubi_map_req));
499141e6ebdSCorentin Chary 		if (err) {
500141e6ebdSCorentin Chary 			err = -EFAULT;
501141e6ebdSCorentin Chary 			break;
502141e6ebdSCorentin Chary 		}
503b36a261eSRichard Weinberger 		err = ubi_leb_map(desc, req.lnum);
504141e6ebdSCorentin Chary 		break;
505141e6ebdSCorentin Chary 	}
506c3da23beSCorentin Chary 
507c3da23beSCorentin Chary 	/* Logical eraseblock un-map command */
508c3da23beSCorentin Chary 	case UBI_IOCEBUNMAP:
509c3da23beSCorentin Chary 	{
510c3da23beSCorentin Chary 		int32_t lnum;
511c3da23beSCorentin Chary 
512c3da23beSCorentin Chary 		err = get_user(lnum, (__user int32_t *)argp);
513c3da23beSCorentin Chary 		if (err) {
514c3da23beSCorentin Chary 			err = -EFAULT;
515c3da23beSCorentin Chary 			break;
516c3da23beSCorentin Chary 		}
517c3da23beSCorentin Chary 		err = ubi_leb_unmap(desc, lnum);
518c3da23beSCorentin Chary 		break;
519c3da23beSCorentin Chary 	}
520a27ce8f5SCorentin Chary 
521a27ce8f5SCorentin Chary 	/* Check if logical eraseblock is mapped command */
522a27ce8f5SCorentin Chary 	case UBI_IOCEBISMAP:
523a27ce8f5SCorentin Chary 	{
524a27ce8f5SCorentin Chary 		int32_t lnum;
525a27ce8f5SCorentin Chary 
526a27ce8f5SCorentin Chary 		err = get_user(lnum, (__user int32_t *)argp);
527a27ce8f5SCorentin Chary 		if (err) {
528a27ce8f5SCorentin Chary 			err = -EFAULT;
529a27ce8f5SCorentin Chary 			break;
530a27ce8f5SCorentin Chary 		}
531a27ce8f5SCorentin Chary 		err = ubi_is_mapped(desc, lnum);
532a27ce8f5SCorentin Chary 		break;
533a27ce8f5SCorentin Chary 	}
534801c135cSArtem B. Bityutskiy 
535766fb95bSSidney Amani 	/* Set volume property command */
5366748482fSArtem Bityutskiy 	case UBI_IOCSETVOLPROP:
537766fb95bSSidney Amani 	{
5386748482fSArtem Bityutskiy 		struct ubi_set_vol_prop_req req;
539766fb95bSSidney Amani 
540766fb95bSSidney Amani 		err = copy_from_user(&req, argp,
5416748482fSArtem Bityutskiy 				     sizeof(struct ubi_set_vol_prop_req));
542766fb95bSSidney Amani 		if (err) {
543766fb95bSSidney Amani 			err = -EFAULT;
544766fb95bSSidney Amani 			break;
545766fb95bSSidney Amani 		}
546766fb95bSSidney Amani 		switch (req.property) {
5476748482fSArtem Bityutskiy 		case UBI_VOL_PROP_DIRECT_WRITE:
548f089c0b2SArtem Bityutskiy 			mutex_lock(&ubi->device_mutex);
549766fb95bSSidney Amani 			desc->vol->direct_writes = !!req.value;
550f089c0b2SArtem Bityutskiy 			mutex_unlock(&ubi->device_mutex);
551766fb95bSSidney Amani 			break;
552766fb95bSSidney Amani 		default:
553766fb95bSSidney Amani 			err = -EINVAL;
554766fb95bSSidney Amani 			break;
555766fb95bSSidney Amani 		}
556766fb95bSSidney Amani 		break;
557766fb95bSSidney Amani 	}
558766fb95bSSidney Amani 
5598af87188SArtem Bityutskiy 	/* Create a R/O block device on top of the UBI volume */
5608af87188SArtem Bityutskiy 	case UBI_IOCVOLCRBLK:
5619d54c8a3SEzequiel Garcia 	{
5629d54c8a3SEzequiel Garcia 		struct ubi_volume_info vi;
5639d54c8a3SEzequiel Garcia 
5649d54c8a3SEzequiel Garcia 		ubi_get_volume_info(desc, &vi);
5654d283ee2SArtem Bityutskiy 		err = ubiblock_create(&vi);
5669d54c8a3SEzequiel Garcia 		break;
5679d54c8a3SEzequiel Garcia 	}
5689d54c8a3SEzequiel Garcia 
5698af87188SArtem Bityutskiy 	/* Remove the R/O block device */
5708af87188SArtem Bityutskiy 	case UBI_IOCVOLRMBLK:
5719d54c8a3SEzequiel Garcia 	{
5729d54c8a3SEzequiel Garcia 		struct ubi_volume_info vi;
5739d54c8a3SEzequiel Garcia 
5749d54c8a3SEzequiel Garcia 		ubi_get_volume_info(desc, &vi);
5754d283ee2SArtem Bityutskiy 		err = ubiblock_remove(&vi);
5769d54c8a3SEzequiel Garcia 		break;
5779d54c8a3SEzequiel Garcia 	}
5789d54c8a3SEzequiel Garcia 
579801c135cSArtem B. Bityutskiy 	default:
580801c135cSArtem B. Bityutskiy 		err = -ENOTTY;
581801c135cSArtem B. Bityutskiy 		break;
582801c135cSArtem B. Bityutskiy 	}
583801c135cSArtem B. Bityutskiy 	return err;
584801c135cSArtem B. Bityutskiy }
585801c135cSArtem B. Bityutskiy 
586801c135cSArtem B. Bityutskiy /**
587801c135cSArtem B. Bityutskiy  * verify_mkvol_req - verify volume creation request.
588801c135cSArtem B. Bityutskiy  * @ubi: UBI device description object
589801c135cSArtem B. Bityutskiy  * @req: the request to check
590801c135cSArtem B. Bityutskiy  *
591801c135cSArtem B. Bityutskiy  * This function zero if the request is correct, and %-EINVAL if not.
592801c135cSArtem B. Bityutskiy  */
verify_mkvol_req(const struct ubi_device * ubi,const struct ubi_mkvol_req * req)593801c135cSArtem B. Bityutskiy static int verify_mkvol_req(const struct ubi_device *ubi,
594801c135cSArtem B. Bityutskiy 			    const struct ubi_mkvol_req *req)
595801c135cSArtem B. Bityutskiy {
596801c135cSArtem B. Bityutskiy 	int n, err = -EINVAL;
597801c135cSArtem B. Bityutskiy 
598801c135cSArtem B. Bityutskiy 	if (req->bytes < 0 || req->alignment < 0 || req->vol_type < 0 ||
599801c135cSArtem B. Bityutskiy 	    req->name_len < 0)
600801c135cSArtem B. Bityutskiy 		goto bad;
601801c135cSArtem B. Bityutskiy 
602801c135cSArtem B. Bityutskiy 	if ((req->vol_id < 0 || req->vol_id >= ubi->vtbl_slots) &&
603801c135cSArtem B. Bityutskiy 	    req->vol_id != UBI_VOL_NUM_AUTO)
604801c135cSArtem B. Bityutskiy 		goto bad;
605801c135cSArtem B. Bityutskiy 
606801c135cSArtem B. Bityutskiy 	if (req->alignment == 0)
607801c135cSArtem B. Bityutskiy 		goto bad;
608801c135cSArtem B. Bityutskiy 
609801c135cSArtem B. Bityutskiy 	if (req->bytes == 0)
610801c135cSArtem B. Bityutskiy 		goto bad;
611801c135cSArtem B. Bityutskiy 
612801c135cSArtem B. Bityutskiy 	if (req->vol_type != UBI_DYNAMIC_VOLUME &&
613801c135cSArtem B. Bityutskiy 	    req->vol_type != UBI_STATIC_VOLUME)
614801c135cSArtem B. Bityutskiy 		goto bad;
615801c135cSArtem B. Bityutskiy 
616c355aa46SQuentin Schulz 	if (req->flags & ~UBI_VOL_VALID_FLGS)
617c355aa46SQuentin Schulz 		goto bad;
618c355aa46SQuentin Schulz 
619c355aa46SQuentin Schulz 	if (req->flags & UBI_VOL_SKIP_CRC_CHECK_FLG &&
620c355aa46SQuentin Schulz 	    req->vol_type != UBI_STATIC_VOLUME)
621c355aa46SQuentin Schulz 		goto bad;
622c355aa46SQuentin Schulz 
623801c135cSArtem B. Bityutskiy 	if (req->alignment > ubi->leb_size)
624801c135cSArtem B. Bityutskiy 		goto bad;
625801c135cSArtem B. Bityutskiy 
626cadb40ccSKyungmin Park 	n = req->alignment & (ubi->min_io_size - 1);
627801c135cSArtem B. Bityutskiy 	if (req->alignment != 1 && n)
628801c135cSArtem B. Bityutskiy 		goto bad;
629801c135cSArtem B. Bityutskiy 
6304a59c797SRichard Weinberger 	if (!req->name[0] || !req->name_len)
6314a59c797SRichard Weinberger 		goto bad;
6324a59c797SRichard Weinberger 
633801c135cSArtem B. Bityutskiy 	if (req->name_len > UBI_VOL_NAME_MAX) {
634801c135cSArtem B. Bityutskiy 		err = -ENAMETOOLONG;
635801c135cSArtem B. Bityutskiy 		goto bad;
636801c135cSArtem B. Bityutskiy 	}
637801c135cSArtem B. Bityutskiy 
638a6ea4407SArtem Bityutskiy 	n = strnlen(req->name, req->name_len + 1);
639a6ea4407SArtem Bityutskiy 	if (n != req->name_len)
640a6ea4407SArtem Bityutskiy 		goto bad;
641a6ea4407SArtem Bityutskiy 
642801c135cSArtem B. Bityutskiy 	return 0;
643801c135cSArtem B. Bityutskiy 
644801c135cSArtem B. Bityutskiy bad:
64532608703STanya Brokhman 	ubi_err(ubi, "bad volume creation request");
646718c00bbSArtem Bityutskiy 	ubi_dump_mkvol_req(req);
647801c135cSArtem B. Bityutskiy 	return err;
648801c135cSArtem B. Bityutskiy }
649801c135cSArtem B. Bityutskiy 
650801c135cSArtem B. Bityutskiy /**
651801c135cSArtem B. Bityutskiy  * verify_rsvol_req - verify volume re-size request.
652801c135cSArtem B. Bityutskiy  * @ubi: UBI device description object
653801c135cSArtem B. Bityutskiy  * @req: the request to check
654801c135cSArtem B. Bityutskiy  *
655801c135cSArtem B. Bityutskiy  * This function returns zero if the request is correct, and %-EINVAL if not.
656801c135cSArtem B. Bityutskiy  */
verify_rsvol_req(const struct ubi_device * ubi,const struct ubi_rsvol_req * req)657801c135cSArtem B. Bityutskiy static int verify_rsvol_req(const struct ubi_device *ubi,
658801c135cSArtem B. Bityutskiy 			    const struct ubi_rsvol_req *req)
659801c135cSArtem B. Bityutskiy {
660801c135cSArtem B. Bityutskiy 	if (req->bytes <= 0)
661801c135cSArtem B. Bityutskiy 		return -EINVAL;
662801c135cSArtem B. Bityutskiy 
663801c135cSArtem B. Bityutskiy 	if (req->vol_id < 0 || req->vol_id >= ubi->vtbl_slots)
664801c135cSArtem B. Bityutskiy 		return -EINVAL;
665801c135cSArtem B. Bityutskiy 
666801c135cSArtem B. Bityutskiy 	return 0;
667801c135cSArtem B. Bityutskiy }
668801c135cSArtem B. Bityutskiy 
669f40ac9cdSArtem Bityutskiy /**
670f40ac9cdSArtem Bityutskiy  * rename_volumes - rename UBI volumes.
671f40ac9cdSArtem Bityutskiy  * @ubi: UBI device description object
672f40ac9cdSArtem Bityutskiy  * @req: volumes re-name request
673f40ac9cdSArtem Bityutskiy  *
674f40ac9cdSArtem Bityutskiy  * This is a helper function for the volume re-name IOCTL which validates the
675ec1f97f5SJilin Yuan  * request, opens the volume and calls corresponding volumes management
676f40ac9cdSArtem Bityutskiy  * function. Returns zero in case of success and a negative error code in case
677f40ac9cdSArtem Bityutskiy  * of failure.
678f40ac9cdSArtem Bityutskiy  */
rename_volumes(struct ubi_device * ubi,struct ubi_rnvol_req * req)679f40ac9cdSArtem Bityutskiy static int rename_volumes(struct ubi_device *ubi,
680f40ac9cdSArtem Bityutskiy 			  struct ubi_rnvol_req *req)
681f40ac9cdSArtem Bityutskiy {
682f40ac9cdSArtem Bityutskiy 	int i, n, err;
683f40ac9cdSArtem Bityutskiy 	struct list_head rename_list;
684f40ac9cdSArtem Bityutskiy 	struct ubi_rename_entry *re, *re1;
685f40ac9cdSArtem Bityutskiy 
686f40ac9cdSArtem Bityutskiy 	if (req->count < 0 || req->count > UBI_MAX_RNVOL)
687f40ac9cdSArtem Bityutskiy 		return -EINVAL;
688f40ac9cdSArtem Bityutskiy 
689f40ac9cdSArtem Bityutskiy 	if (req->count == 0)
690f40ac9cdSArtem Bityutskiy 		return 0;
691f40ac9cdSArtem Bityutskiy 
692f40ac9cdSArtem Bityutskiy 	/* Validate volume IDs and names in the request */
693f40ac9cdSArtem Bityutskiy 	for (i = 0; i < req->count; i++) {
694f40ac9cdSArtem Bityutskiy 		if (req->ents[i].vol_id < 0 ||
695f40ac9cdSArtem Bityutskiy 		    req->ents[i].vol_id >= ubi->vtbl_slots)
696f40ac9cdSArtem Bityutskiy 			return -EINVAL;
697f40ac9cdSArtem Bityutskiy 		if (req->ents[i].name_len < 0)
698f40ac9cdSArtem Bityutskiy 			return -EINVAL;
699f40ac9cdSArtem Bityutskiy 		if (req->ents[i].name_len > UBI_VOL_NAME_MAX)
700f40ac9cdSArtem Bityutskiy 			return -ENAMETOOLONG;
701f40ac9cdSArtem Bityutskiy 		req->ents[i].name[req->ents[i].name_len] = '\0';
702f40ac9cdSArtem Bityutskiy 		n = strlen(req->ents[i].name);
703f40ac9cdSArtem Bityutskiy 		if (n != req->ents[i].name_len)
7047fbbd057SDan Carpenter 			return -EINVAL;
705f40ac9cdSArtem Bityutskiy 	}
706f40ac9cdSArtem Bityutskiy 
707f40ac9cdSArtem Bityutskiy 	/* Make sure volume IDs and names are unique */
708f40ac9cdSArtem Bityutskiy 	for (i = 0; i < req->count - 1; i++) {
709f40ac9cdSArtem Bityutskiy 		for (n = i + 1; n < req->count; n++) {
710f40ac9cdSArtem Bityutskiy 			if (req->ents[i].vol_id == req->ents[n].vol_id) {
71132608703STanya Brokhman 				ubi_err(ubi, "duplicated volume id %d",
712f40ac9cdSArtem Bityutskiy 					req->ents[i].vol_id);
713f40ac9cdSArtem Bityutskiy 				return -EINVAL;
714f40ac9cdSArtem Bityutskiy 			}
715f40ac9cdSArtem Bityutskiy 			if (!strcmp(req->ents[i].name, req->ents[n].name)) {
71632608703STanya Brokhman 				ubi_err(ubi, "duplicated volume name \"%s\"",
717f40ac9cdSArtem Bityutskiy 					req->ents[i].name);
718f40ac9cdSArtem Bityutskiy 				return -EINVAL;
719f40ac9cdSArtem Bityutskiy 			}
720f40ac9cdSArtem Bityutskiy 		}
721f40ac9cdSArtem Bityutskiy 	}
722f40ac9cdSArtem Bityutskiy 
723f40ac9cdSArtem Bityutskiy 	/* Create the re-name list */
724f40ac9cdSArtem Bityutskiy 	INIT_LIST_HEAD(&rename_list);
725f40ac9cdSArtem Bityutskiy 	for (i = 0; i < req->count; i++) {
726f40ac9cdSArtem Bityutskiy 		int vol_id = req->ents[i].vol_id;
727f40ac9cdSArtem Bityutskiy 		int name_len = req->ents[i].name_len;
728f40ac9cdSArtem Bityutskiy 		const char *name = req->ents[i].name;
729f40ac9cdSArtem Bityutskiy 
730f40ac9cdSArtem Bityutskiy 		re = kzalloc(sizeof(struct ubi_rename_entry), GFP_KERNEL);
731f40ac9cdSArtem Bityutskiy 		if (!re) {
732f40ac9cdSArtem Bityutskiy 			err = -ENOMEM;
733f40ac9cdSArtem Bityutskiy 			goto out_free;
734f40ac9cdSArtem Bityutskiy 		}
735f40ac9cdSArtem Bityutskiy 
736892abde5SRichard Weinberger 		re->desc = ubi_open_volume(ubi->ubi_num, vol_id, UBI_METAONLY);
737f40ac9cdSArtem Bityutskiy 		if (IS_ERR(re->desc)) {
738f40ac9cdSArtem Bityutskiy 			err = PTR_ERR(re->desc);
73932608703STanya Brokhman 			ubi_err(ubi, "cannot open volume %d, error %d",
74032608703STanya Brokhman 				vol_id, err);
741f40ac9cdSArtem Bityutskiy 			kfree(re);
742f40ac9cdSArtem Bityutskiy 			goto out_free;
743f40ac9cdSArtem Bityutskiy 		}
744f40ac9cdSArtem Bityutskiy 
745f40ac9cdSArtem Bityutskiy 		/* Skip this re-naming if the name does not really change */
746f40ac9cdSArtem Bityutskiy 		if (re->desc->vol->name_len == name_len &&
747f40ac9cdSArtem Bityutskiy 		    !memcmp(re->desc->vol->name, name, name_len)) {
748f40ac9cdSArtem Bityutskiy 			ubi_close_volume(re->desc);
749f40ac9cdSArtem Bityutskiy 			kfree(re);
750f40ac9cdSArtem Bityutskiy 			continue;
751f40ac9cdSArtem Bityutskiy 		}
752f40ac9cdSArtem Bityutskiy 
753f40ac9cdSArtem Bityutskiy 		re->new_name_len = name_len;
754f40ac9cdSArtem Bityutskiy 		memcpy(re->new_name, name, name_len);
755f40ac9cdSArtem Bityutskiy 		list_add_tail(&re->list, &rename_list);
756719bb840SArtem Bityutskiy 		dbg_gen("will rename volume %d from \"%s\" to \"%s\"",
757f40ac9cdSArtem Bityutskiy 			vol_id, re->desc->vol->name, name);
758f40ac9cdSArtem Bityutskiy 	}
759f40ac9cdSArtem Bityutskiy 
760f40ac9cdSArtem Bityutskiy 	if (list_empty(&rename_list))
761f40ac9cdSArtem Bityutskiy 		return 0;
762f40ac9cdSArtem Bityutskiy 
763f40ac9cdSArtem Bityutskiy 	/* Find out the volumes which have to be removed */
764f40ac9cdSArtem Bityutskiy 	list_for_each_entry(re, &rename_list, list) {
765f40ac9cdSArtem Bityutskiy 		struct ubi_volume_desc *desc;
766f40ac9cdSArtem Bityutskiy 		int no_remove_needed = 0;
767f40ac9cdSArtem Bityutskiy 
768f40ac9cdSArtem Bityutskiy 		/*
769f40ac9cdSArtem Bityutskiy 		 * Volume @re->vol_id is going to be re-named to
770f40ac9cdSArtem Bityutskiy 		 * @re->new_name, while its current name is @name. If a volume
771f40ac9cdSArtem Bityutskiy 		 * with name @re->new_name currently exists, it has to be
772f40ac9cdSArtem Bityutskiy 		 * removed, unless it is also re-named in the request (@req).
773f40ac9cdSArtem Bityutskiy 		 */
774f40ac9cdSArtem Bityutskiy 		list_for_each_entry(re1, &rename_list, list) {
775f40ac9cdSArtem Bityutskiy 			if (re->new_name_len == re1->desc->vol->name_len &&
776f40ac9cdSArtem Bityutskiy 			    !memcmp(re->new_name, re1->desc->vol->name,
777f40ac9cdSArtem Bityutskiy 				    re1->desc->vol->name_len)) {
778f40ac9cdSArtem Bityutskiy 				no_remove_needed = 1;
779f40ac9cdSArtem Bityutskiy 				break;
780f40ac9cdSArtem Bityutskiy 			}
781f40ac9cdSArtem Bityutskiy 		}
782f40ac9cdSArtem Bityutskiy 
783f40ac9cdSArtem Bityutskiy 		if (no_remove_needed)
784f40ac9cdSArtem Bityutskiy 			continue;
785f40ac9cdSArtem Bityutskiy 
786f40ac9cdSArtem Bityutskiy 		/*
787f40ac9cdSArtem Bityutskiy 		 * It seems we need to remove volume with name @re->new_name,
788f40ac9cdSArtem Bityutskiy 		 * if it exists.
789f40ac9cdSArtem Bityutskiy 		 */
790f2863c54SArtem Bityutskiy 		desc = ubi_open_volume_nm(ubi->ubi_num, re->new_name,
791f2863c54SArtem Bityutskiy 					  UBI_EXCLUSIVE);
792f40ac9cdSArtem Bityutskiy 		if (IS_ERR(desc)) {
793f40ac9cdSArtem Bityutskiy 			err = PTR_ERR(desc);
794f40ac9cdSArtem Bityutskiy 			if (err == -ENODEV)
795f40ac9cdSArtem Bityutskiy 				/* Re-naming into a non-existing volume name */
796f40ac9cdSArtem Bityutskiy 				continue;
797f40ac9cdSArtem Bityutskiy 
798f40ac9cdSArtem Bityutskiy 			/* The volume exists but busy, or an error occurred */
79932608703STanya Brokhman 			ubi_err(ubi, "cannot open volume \"%s\", error %d",
800f40ac9cdSArtem Bityutskiy 				re->new_name, err);
801f40ac9cdSArtem Bityutskiy 			goto out_free;
802f40ac9cdSArtem Bityutskiy 		}
803f40ac9cdSArtem Bityutskiy 
80401ebc12fSJulia Lawall 		re1 = kzalloc(sizeof(struct ubi_rename_entry), GFP_KERNEL);
80501ebc12fSJulia Lawall 		if (!re1) {
806f40ac9cdSArtem Bityutskiy 			err = -ENOMEM;
807f40ac9cdSArtem Bityutskiy 			ubi_close_volume(desc);
808f40ac9cdSArtem Bityutskiy 			goto out_free;
809f40ac9cdSArtem Bityutskiy 		}
810f40ac9cdSArtem Bityutskiy 
81101ebc12fSJulia Lawall 		re1->remove = 1;
81201ebc12fSJulia Lawall 		re1->desc = desc;
81301ebc12fSJulia Lawall 		list_add(&re1->list, &rename_list);
814719bb840SArtem Bityutskiy 		dbg_gen("will remove volume %d, name \"%s\"",
81501ebc12fSJulia Lawall 			re1->desc->vol->vol_id, re1->desc->vol->name);
816f40ac9cdSArtem Bityutskiy 	}
817f40ac9cdSArtem Bityutskiy 
818f089c0b2SArtem Bityutskiy 	mutex_lock(&ubi->device_mutex);
819f40ac9cdSArtem Bityutskiy 	err = ubi_rename_volumes(ubi, &rename_list);
820f089c0b2SArtem Bityutskiy 	mutex_unlock(&ubi->device_mutex);
821f40ac9cdSArtem Bityutskiy 
822f40ac9cdSArtem Bityutskiy out_free:
823f40ac9cdSArtem Bityutskiy 	list_for_each_entry_safe(re, re1, &rename_list, list) {
824f40ac9cdSArtem Bityutskiy 		ubi_close_volume(re->desc);
825f40ac9cdSArtem Bityutskiy 		list_del(&re->list);
826f40ac9cdSArtem Bityutskiy 		kfree(re);
827f40ac9cdSArtem Bityutskiy 	}
828f40ac9cdSArtem Bityutskiy 	return err;
829f40ac9cdSArtem Bityutskiy }
830f40ac9cdSArtem Bityutskiy 
ubi_cdev_ioctl(struct file * file,unsigned int cmd,unsigned long arg)831f429b2eaSArtem Bityutskiy static long ubi_cdev_ioctl(struct file *file, unsigned int cmd,
832f429b2eaSArtem Bityutskiy 			   unsigned long arg)
833801c135cSArtem B. Bityutskiy {
834801c135cSArtem B. Bityutskiy 	int err = 0;
835801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi;
836801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc;
837801c135cSArtem B. Bityutskiy 	void __user *argp = (void __user *)arg;
838801c135cSArtem B. Bityutskiy 
839801c135cSArtem B. Bityutskiy 	if (!capable(CAP_SYS_RESOURCE))
840801c135cSArtem B. Bityutskiy 		return -EPERM;
841801c135cSArtem B. Bityutskiy 
842f429b2eaSArtem Bityutskiy 	ubi = ubi_get_by_major(imajor(file->f_mapping->host));
843e73f4459SArtem Bityutskiy 	if (!ubi)
844e73f4459SArtem Bityutskiy 		return -ENODEV;
845801c135cSArtem B. Bityutskiy 
846801c135cSArtem B. Bityutskiy 	switch (cmd) {
847801c135cSArtem B. Bityutskiy 	/* Create volume command */
848801c135cSArtem B. Bityutskiy 	case UBI_IOCMKVOL:
849801c135cSArtem B. Bityutskiy 	{
850801c135cSArtem B. Bityutskiy 		struct ubi_mkvol_req req;
851801c135cSArtem B. Bityutskiy 
852c8566350SArtem Bityutskiy 		dbg_gen("create volume");
853897a316cSArtem Bityutskiy 		err = copy_from_user(&req, argp, sizeof(struct ubi_mkvol_req));
854801c135cSArtem B. Bityutskiy 		if (err) {
855801c135cSArtem B. Bityutskiy 			err = -EFAULT;
856801c135cSArtem B. Bityutskiy 			break;
857801c135cSArtem B. Bityutskiy 		}
858801c135cSArtem B. Bityutskiy 
859801c135cSArtem B. Bityutskiy 		err = verify_mkvol_req(ubi, &req);
860801c135cSArtem B. Bityutskiy 		if (err)
861801c135cSArtem B. Bityutskiy 			break;
862801c135cSArtem B. Bityutskiy 
863f089c0b2SArtem Bityutskiy 		mutex_lock(&ubi->device_mutex);
864801c135cSArtem B. Bityutskiy 		err = ubi_create_volume(ubi, &req);
865f089c0b2SArtem Bityutskiy 		mutex_unlock(&ubi->device_mutex);
866801c135cSArtem B. Bityutskiy 		if (err)
867801c135cSArtem B. Bityutskiy 			break;
868801c135cSArtem B. Bityutskiy 
869bf07803aSChristoph Hellwig 		err = put_user(req.vol_id, (__user int32_t *)argp);
870801c135cSArtem B. Bityutskiy 		if (err)
871801c135cSArtem B. Bityutskiy 			err = -EFAULT;
872801c135cSArtem B. Bityutskiy 
873801c135cSArtem B. Bityutskiy 		break;
874801c135cSArtem B. Bityutskiy 	}
875801c135cSArtem B. Bityutskiy 
876801c135cSArtem B. Bityutskiy 	/* Remove volume command */
877801c135cSArtem B. Bityutskiy 	case UBI_IOCRMVOL:
878801c135cSArtem B. Bityutskiy 	{
879801c135cSArtem B. Bityutskiy 		int vol_id;
880801c135cSArtem B. Bityutskiy 
881c8566350SArtem Bityutskiy 		dbg_gen("remove volume");
882bf07803aSChristoph Hellwig 		err = get_user(vol_id, (__user int32_t *)argp);
883801c135cSArtem B. Bityutskiy 		if (err) {
884801c135cSArtem B. Bityutskiy 			err = -EFAULT;
885801c135cSArtem B. Bityutskiy 			break;
886801c135cSArtem B. Bityutskiy 		}
887801c135cSArtem B. Bityutskiy 
888801c135cSArtem B. Bityutskiy 		desc = ubi_open_volume(ubi->ubi_num, vol_id, UBI_EXCLUSIVE);
889801c135cSArtem B. Bityutskiy 		if (IS_ERR(desc)) {
890801c135cSArtem B. Bityutskiy 			err = PTR_ERR(desc);
891801c135cSArtem B. Bityutskiy 			break;
892801c135cSArtem B. Bityutskiy 		}
893801c135cSArtem B. Bityutskiy 
894f089c0b2SArtem Bityutskiy 		mutex_lock(&ubi->device_mutex);
895f40ac9cdSArtem Bityutskiy 		err = ubi_remove_volume(desc, 0);
896f089c0b2SArtem Bityutskiy 		mutex_unlock(&ubi->device_mutex);
89740e4d0c1SArtem Bityutskiy 
898450f872aSArtem Bityutskiy 		/*
89940e4d0c1SArtem Bityutskiy 		 * The volume is deleted (unless an error occurred), and the
90040e4d0c1SArtem Bityutskiy 		 * 'struct ubi_volume' object will be freed when
90140e4d0c1SArtem Bityutskiy 		 * 'ubi_close_volume()' will call 'put_device()'.
902450f872aSArtem Bityutskiy 		 */
903801c135cSArtem B. Bityutskiy 		ubi_close_volume(desc);
904801c135cSArtem B. Bityutskiy 		break;
905801c135cSArtem B. Bityutskiy 	}
906801c135cSArtem B. Bityutskiy 
907801c135cSArtem B. Bityutskiy 	/* Re-size volume command */
908801c135cSArtem B. Bityutskiy 	case UBI_IOCRSVOL:
909801c135cSArtem B. Bityutskiy 	{
910801c135cSArtem B. Bityutskiy 		int pebs;
911801c135cSArtem B. Bityutskiy 		struct ubi_rsvol_req req;
912801c135cSArtem B. Bityutskiy 
913c8566350SArtem Bityutskiy 		dbg_gen("re-size volume");
914897a316cSArtem Bityutskiy 		err = copy_from_user(&req, argp, sizeof(struct ubi_rsvol_req));
915801c135cSArtem B. Bityutskiy 		if (err) {
916801c135cSArtem B. Bityutskiy 			err = -EFAULT;
917801c135cSArtem B. Bityutskiy 			break;
918801c135cSArtem B. Bityutskiy 		}
919801c135cSArtem B. Bityutskiy 
920801c135cSArtem B. Bityutskiy 		err = verify_rsvol_req(ubi, &req);
921801c135cSArtem B. Bityutskiy 		if (err)
922801c135cSArtem B. Bityutskiy 			break;
923801c135cSArtem B. Bityutskiy 
924801c135cSArtem B. Bityutskiy 		desc = ubi_open_volume(ubi->ubi_num, req.vol_id, UBI_EXCLUSIVE);
925801c135cSArtem B. Bityutskiy 		if (IS_ERR(desc)) {
926801c135cSArtem B. Bityutskiy 			err = PTR_ERR(desc);
927801c135cSArtem B. Bityutskiy 			break;
928801c135cSArtem B. Bityutskiy 		}
929801c135cSArtem B. Bityutskiy 
9303013ee31SArtem Bityutskiy 		pebs = div_u64(req.bytes + desc->vol->usable_leb_size - 1,
9313013ee31SArtem Bityutskiy 			       desc->vol->usable_leb_size);
932801c135cSArtem B. Bityutskiy 
933f089c0b2SArtem Bityutskiy 		mutex_lock(&ubi->device_mutex);
934801c135cSArtem B. Bityutskiy 		err = ubi_resize_volume(desc, pebs);
935f089c0b2SArtem Bityutskiy 		mutex_unlock(&ubi->device_mutex);
936801c135cSArtem B. Bityutskiy 		ubi_close_volume(desc);
937801c135cSArtem B. Bityutskiy 		break;
938801c135cSArtem B. Bityutskiy 	}
939801c135cSArtem B. Bityutskiy 
940f40ac9cdSArtem Bityutskiy 	/* Re-name volumes command */
941f40ac9cdSArtem Bityutskiy 	case UBI_IOCRNVOL:
942f40ac9cdSArtem Bityutskiy 	{
943f40ac9cdSArtem Bityutskiy 		struct ubi_rnvol_req *req;
944f40ac9cdSArtem Bityutskiy 
945719bb840SArtem Bityutskiy 		dbg_gen("re-name volumes");
946f40ac9cdSArtem Bityutskiy 		req = kmalloc(sizeof(struct ubi_rnvol_req), GFP_KERNEL);
947f40ac9cdSArtem Bityutskiy 		if (!req) {
948f40ac9cdSArtem Bityutskiy 			err = -ENOMEM;
949f40ac9cdSArtem Bityutskiy 			break;
950a396ce4bSRichard Weinberger 		}
951f40ac9cdSArtem Bityutskiy 
952f40ac9cdSArtem Bityutskiy 		err = copy_from_user(req, argp, sizeof(struct ubi_rnvol_req));
953f40ac9cdSArtem Bityutskiy 		if (err) {
954f40ac9cdSArtem Bityutskiy 			err = -EFAULT;
955f40ac9cdSArtem Bityutskiy 			kfree(req);
956f40ac9cdSArtem Bityutskiy 			break;
957f40ac9cdSArtem Bityutskiy 		}
958f40ac9cdSArtem Bityutskiy 
959f40ac9cdSArtem Bityutskiy 		err = rename_volumes(ubi, req);
960f40ac9cdSArtem Bityutskiy 		kfree(req);
961f40ac9cdSArtem Bityutskiy 		break;
962f40ac9cdSArtem Bityutskiy 	}
963f40ac9cdSArtem Bityutskiy 
964663586c0SRichard Weinberger 	/* Check a specific PEB for bitflips and scrub it if needed */
965663586c0SRichard Weinberger 	case UBI_IOCRPEB:
966663586c0SRichard Weinberger 	{
967663586c0SRichard Weinberger 		int pnum;
968663586c0SRichard Weinberger 
969663586c0SRichard Weinberger 		err = get_user(pnum, (__user int32_t *)argp);
970663586c0SRichard Weinberger 		if (err) {
971663586c0SRichard Weinberger 			err = -EFAULT;
972663586c0SRichard Weinberger 			break;
973663586c0SRichard Weinberger 		}
974663586c0SRichard Weinberger 
975663586c0SRichard Weinberger 		err = ubi_bitflip_check(ubi, pnum, 0);
976663586c0SRichard Weinberger 		break;
977663586c0SRichard Weinberger 	}
978663586c0SRichard Weinberger 
979663586c0SRichard Weinberger 	/* Force scrubbing for a specific PEB */
980663586c0SRichard Weinberger 	case UBI_IOCSPEB:
981663586c0SRichard Weinberger 	{
982663586c0SRichard Weinberger 		int pnum;
983663586c0SRichard Weinberger 
984663586c0SRichard Weinberger 		err = get_user(pnum, (__user int32_t *)argp);
985663586c0SRichard Weinberger 		if (err) {
986663586c0SRichard Weinberger 			err = -EFAULT;
987663586c0SRichard Weinberger 			break;
988663586c0SRichard Weinberger 		}
989663586c0SRichard Weinberger 
990663586c0SRichard Weinberger 		err = ubi_bitflip_check(ubi, pnum, 1);
991663586c0SRichard Weinberger 		break;
992663586c0SRichard Weinberger 	}
993663586c0SRichard Weinberger 
994801c135cSArtem B. Bityutskiy 	default:
995801c135cSArtem B. Bityutskiy 		err = -ENOTTY;
996801c135cSArtem B. Bityutskiy 		break;
997801c135cSArtem B. Bityutskiy 	}
998801c135cSArtem B. Bityutskiy 
999e73f4459SArtem Bityutskiy 	ubi_put_device(ubi);
1000801c135cSArtem B. Bityutskiy 	return err;
1001801c135cSArtem B. Bityutskiy }
1002801c135cSArtem B. Bityutskiy 
ctrl_cdev_ioctl(struct file * file,unsigned int cmd,unsigned long arg)1003f429b2eaSArtem Bityutskiy static long ctrl_cdev_ioctl(struct file *file, unsigned int cmd,
1004f429b2eaSArtem Bityutskiy 			    unsigned long arg)
1005897a316cSArtem Bityutskiy {
1006897a316cSArtem Bityutskiy 	int err = 0;
1007897a316cSArtem Bityutskiy 	void __user *argp = (void __user *)arg;
1008897a316cSArtem Bityutskiy 
1009897a316cSArtem Bityutskiy 	if (!capable(CAP_SYS_RESOURCE))
1010897a316cSArtem Bityutskiy 		return -EPERM;
1011897a316cSArtem Bityutskiy 
1012897a316cSArtem Bityutskiy 	switch (cmd) {
1013897a316cSArtem Bityutskiy 	/* Attach an MTD device command */
1014897a316cSArtem Bityutskiy 	case UBI_IOCATT:
1015897a316cSArtem Bityutskiy 	{
1016897a316cSArtem Bityutskiy 		struct ubi_attach_req req;
1017897a316cSArtem Bityutskiy 		struct mtd_info *mtd;
1018897a316cSArtem Bityutskiy 
1019c8566350SArtem Bityutskiy 		dbg_gen("attach MTD device");
1020897a316cSArtem Bityutskiy 		err = copy_from_user(&req, argp, sizeof(struct ubi_attach_req));
1021897a316cSArtem Bityutskiy 		if (err) {
1022897a316cSArtem Bityutskiy 			err = -EFAULT;
1023897a316cSArtem Bityutskiy 			break;
1024897a316cSArtem Bityutskiy 		}
1025897a316cSArtem Bityutskiy 
1026897a316cSArtem Bityutskiy 		if (req.mtd_num < 0 ||
1027897a316cSArtem Bityutskiy 		    (req.ubi_num < 0 && req.ubi_num != UBI_DEV_NUM_AUTO)) {
1028897a316cSArtem Bityutskiy 			err = -EINVAL;
1029897a316cSArtem Bityutskiy 			break;
1030897a316cSArtem Bityutskiy 		}
1031897a316cSArtem Bityutskiy 
1032897a316cSArtem Bityutskiy 		mtd = get_mtd_device(NULL, req.mtd_num);
1033897a316cSArtem Bityutskiy 		if (IS_ERR(mtd)) {
1034897a316cSArtem Bityutskiy 			err = PTR_ERR(mtd);
1035897a316cSArtem Bityutskiy 			break;
1036897a316cSArtem Bityutskiy 		}
1037897a316cSArtem Bityutskiy 
1038897a316cSArtem Bityutskiy 		/*
1039897a316cSArtem Bityutskiy 		 * Note, further request verification is done by
1040897a316cSArtem Bityutskiy 		 * 'ubi_attach_mtd_dev()'.
1041897a316cSArtem Bityutskiy 		 */
1042897a316cSArtem Bityutskiy 		mutex_lock(&ubi_devices_mutex);
1043256334c3SRichard Genoud 		err = ubi_attach_mtd_dev(mtd, req.ubi_num, req.vid_hdr_offset,
1044*669d2044SZhihao Cheng 					 req.max_beb_per1024, !!req.disable_fm);
1045897a316cSArtem Bityutskiy 		mutex_unlock(&ubi_devices_mutex);
1046897a316cSArtem Bityutskiy 		if (err < 0)
1047897a316cSArtem Bityutskiy 			put_mtd_device(mtd);
1048897a316cSArtem Bityutskiy 		else
1049897a316cSArtem Bityutskiy 			/* @err contains UBI device number */
1050897a316cSArtem Bityutskiy 			err = put_user(err, (__user int32_t *)argp);
1051897a316cSArtem Bityutskiy 
1052897a316cSArtem Bityutskiy 		break;
1053897a316cSArtem Bityutskiy 	}
1054897a316cSArtem Bityutskiy 
1055897a316cSArtem Bityutskiy 	/* Detach an MTD device command */
1056897a316cSArtem Bityutskiy 	case UBI_IOCDET:
1057897a316cSArtem Bityutskiy 	{
1058897a316cSArtem Bityutskiy 		int ubi_num;
1059897a316cSArtem Bityutskiy 
10602ce7be1bSPeter Meerwald 		dbg_gen("detach MTD device");
1061897a316cSArtem Bityutskiy 		err = get_user(ubi_num, (__user int32_t *)argp);
1062897a316cSArtem Bityutskiy 		if (err) {
1063897a316cSArtem Bityutskiy 			err = -EFAULT;
1064897a316cSArtem Bityutskiy 			break;
1065897a316cSArtem Bityutskiy 		}
1066897a316cSArtem Bityutskiy 
1067897a316cSArtem Bityutskiy 		mutex_lock(&ubi_devices_mutex);
1068897a316cSArtem Bityutskiy 		err = ubi_detach_mtd_dev(ubi_num, 0);
1069897a316cSArtem Bityutskiy 		mutex_unlock(&ubi_devices_mutex);
1070897a316cSArtem Bityutskiy 		break;
1071897a316cSArtem Bityutskiy 	}
1072897a316cSArtem Bityutskiy 
1073897a316cSArtem Bityutskiy 	default:
1074897a316cSArtem Bityutskiy 		err = -ENOTTY;
1075897a316cSArtem Bityutskiy 		break;
1076897a316cSArtem Bityutskiy 	}
1077897a316cSArtem Bityutskiy 
1078897a316cSArtem Bityutskiy 	return err;
1079897a316cSArtem Bityutskiy }
1080897a316cSArtem Bityutskiy 
1081801c135cSArtem B. Bityutskiy /* UBI volume character device operations */
10824d187a88SJan Engelhardt const struct file_operations ubi_vol_cdev_operations = {
1083801c135cSArtem B. Bityutskiy 	.owner          = THIS_MODULE,
1084801c135cSArtem B. Bityutskiy 	.open           = vol_cdev_open,
1085801c135cSArtem B. Bityutskiy 	.release        = vol_cdev_release,
1086801c135cSArtem B. Bityutskiy 	.llseek         = vol_cdev_llseek,
1087801c135cSArtem B. Bityutskiy 	.read           = vol_cdev_read,
1088801c135cSArtem B. Bityutskiy 	.write          = vol_cdev_write,
10891b24bc3aSCorentin Chary 	.fsync		= vol_cdev_fsync,
1090f429b2eaSArtem Bityutskiy 	.unlocked_ioctl = vol_cdev_ioctl,
1091407e9ef7SArnd Bergmann 	.compat_ioctl   = compat_ptr_ioctl,
1092f429b2eaSArtem Bityutskiy };
1093f429b2eaSArtem Bityutskiy 
1094f429b2eaSArtem Bityutskiy /* UBI character device operations */
1095f429b2eaSArtem Bityutskiy const struct file_operations ubi_cdev_operations = {
1096f429b2eaSArtem Bityutskiy 	.owner          = THIS_MODULE,
1097f429b2eaSArtem Bityutskiy 	.llseek         = no_llseek,
1098f429b2eaSArtem Bityutskiy 	.unlocked_ioctl = ubi_cdev_ioctl,
1099407e9ef7SArnd Bergmann 	.compat_ioctl   = compat_ptr_ioctl,
1100f429b2eaSArtem Bityutskiy };
1101f429b2eaSArtem Bityutskiy 
1102f429b2eaSArtem Bityutskiy /* UBI control character device operations */
1103f429b2eaSArtem Bityutskiy const struct file_operations ubi_ctrl_cdev_operations = {
1104f429b2eaSArtem Bityutskiy 	.owner          = THIS_MODULE,
1105f429b2eaSArtem Bityutskiy 	.unlocked_ioctl = ctrl_cdev_ioctl,
1106407e9ef7SArnd Bergmann 	.compat_ioctl   = compat_ptr_ioctl,
1107e10b376eSArtem Bityutskiy 	.llseek		= no_llseek,
1108801c135cSArtem B. Bityutskiy };
1109