xref: /openbmc/linux/drivers/mtd/ubi/cdev.c (revision 4a59c797a18917a5cf3ff7ade296b46134d91e6a)
1801c135cSArtem B. Bityutskiy /*
2801c135cSArtem B. Bityutskiy  * Copyright (c) International Business Machines Corp., 2006
3801c135cSArtem B. Bityutskiy  *
4801c135cSArtem B. Bityutskiy  * This program is free software; you can redistribute it and/or modify
5801c135cSArtem B. Bityutskiy  * it under the terms of the GNU General Public License as published by
6801c135cSArtem B. Bityutskiy  * the Free Software Foundation; either version 2 of the License, or
7801c135cSArtem B. Bityutskiy  * (at your option) any later version.
8801c135cSArtem B. Bityutskiy  *
9801c135cSArtem B. Bityutskiy  * This program is distributed in the hope that it will be useful,
10801c135cSArtem B. Bityutskiy  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11801c135cSArtem B. Bityutskiy  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
12801c135cSArtem B. Bityutskiy  * the GNU General Public License for more details.
13801c135cSArtem B. Bityutskiy  *
14801c135cSArtem B. Bityutskiy  * You should have received a copy of the GNU General Public License
15801c135cSArtem B. Bityutskiy  * along with this program; if not, write to the Free Software
16801c135cSArtem B. Bityutskiy  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17801c135cSArtem B. Bityutskiy  *
18801c135cSArtem B. Bityutskiy  * Author: Artem Bityutskiy (Битюцкий Артём)
19801c135cSArtem B. Bityutskiy  */
20801c135cSArtem B. Bityutskiy 
21801c135cSArtem B. Bityutskiy /*
22801c135cSArtem B. Bityutskiy  * This file includes implementation of UBI character device operations.
23801c135cSArtem B. Bityutskiy  *
24801c135cSArtem B. Bityutskiy  * There are two kinds of character devices in UBI: UBI character devices and
25801c135cSArtem B. Bityutskiy  * UBI volume character devices. UBI character devices allow users to
26801c135cSArtem B. Bityutskiy  * manipulate whole volumes: create, remove, and re-size them. Volume character
27801c135cSArtem B. Bityutskiy  * devices provide volume I/O capabilities.
28801c135cSArtem B. Bityutskiy  *
29801c135cSArtem B. Bityutskiy  * Major and minor numbers are assigned dynamically to both UBI and volume
30801c135cSArtem B. Bityutskiy  * character devices.
319f961b57SArtem Bityutskiy  *
329f961b57SArtem Bityutskiy  * Well, there is the third kind of character devices - the UBI control
339f961b57SArtem Bityutskiy  * character device, which allows to manipulate by UBI devices - create and
349f961b57SArtem Bityutskiy  * delete them. In other words, it is used for attaching and detaching MTD
359f961b57SArtem Bityutskiy  * devices.
36801c135cSArtem B. Bityutskiy  */
37801c135cSArtem B. Bityutskiy 
38801c135cSArtem B. Bityutskiy #include <linux/module.h>
39801c135cSArtem B. Bityutskiy #include <linux/stat.h>
405a0e3ad6STejun Heo #include <linux/slab.h>
41801c135cSArtem B. Bityutskiy #include <linux/ioctl.h>
42801c135cSArtem B. Bityutskiy #include <linux/capability.h>
439c9ec147SArtem Bityutskiy #include <linux/uaccess.h>
44f429b2eaSArtem Bityutskiy #include <linux/compat.h>
453013ee31SArtem Bityutskiy #include <linux/math64.h>
46801c135cSArtem B. Bityutskiy #include <mtd/ubi-user.h>
47801c135cSArtem B. Bityutskiy #include "ubi.h"
48801c135cSArtem B. Bityutskiy 
49801c135cSArtem B. Bityutskiy /**
50801c135cSArtem B. Bityutskiy  * get_exclusive - get exclusive access to an UBI volume.
51801c135cSArtem B. Bityutskiy  * @desc: volume descriptor
52801c135cSArtem B. Bityutskiy  *
53801c135cSArtem B. Bityutskiy  * This function changes UBI volume open mode to "exclusive". Returns previous
54801c135cSArtem B. Bityutskiy  * mode value (positive integer) in case of success and a negative error code
55801c135cSArtem B. Bityutskiy  * in case of failure.
56801c135cSArtem B. Bityutskiy  */
57801c135cSArtem B. Bityutskiy static int get_exclusive(struct ubi_volume_desc *desc)
58801c135cSArtem B. Bityutskiy {
59801c135cSArtem B. Bityutskiy 	int users, err;
60801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
61801c135cSArtem B. Bityutskiy 
62801c135cSArtem B. Bityutskiy 	spin_lock(&vol->ubi->volumes_lock);
63801c135cSArtem B. Bityutskiy 	users = vol->readers + vol->writers + vol->exclusive;
64801c135cSArtem B. Bityutskiy 	ubi_assert(users > 0);
65801c135cSArtem B. Bityutskiy 	if (users > 1) {
66801c135cSArtem B. Bityutskiy 		dbg_err("%d users for volume %d", users, vol->vol_id);
67801c135cSArtem B. Bityutskiy 		err = -EBUSY;
68801c135cSArtem B. Bityutskiy 	} else {
69801c135cSArtem B. Bityutskiy 		vol->readers = vol->writers = 0;
70801c135cSArtem B. Bityutskiy 		vol->exclusive = 1;
71801c135cSArtem B. Bityutskiy 		err = desc->mode;
72801c135cSArtem B. Bityutskiy 		desc->mode = UBI_EXCLUSIVE;
73801c135cSArtem B. Bityutskiy 	}
74801c135cSArtem B. Bityutskiy 	spin_unlock(&vol->ubi->volumes_lock);
75801c135cSArtem B. Bityutskiy 
76801c135cSArtem B. Bityutskiy 	return err;
77801c135cSArtem B. Bityutskiy }
78801c135cSArtem B. Bityutskiy 
79801c135cSArtem B. Bityutskiy /**
80801c135cSArtem B. Bityutskiy  * revoke_exclusive - revoke exclusive mode.
81801c135cSArtem B. Bityutskiy  * @desc: volume descriptor
82801c135cSArtem B. Bityutskiy  * @mode: new mode to switch to
83801c135cSArtem B. Bityutskiy  */
84801c135cSArtem B. Bityutskiy static void revoke_exclusive(struct ubi_volume_desc *desc, int mode)
85801c135cSArtem B. Bityutskiy {
86801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
87801c135cSArtem B. Bityutskiy 
88801c135cSArtem B. Bityutskiy 	spin_lock(&vol->ubi->volumes_lock);
89801c135cSArtem B. Bityutskiy 	ubi_assert(vol->readers == 0 && vol->writers == 0);
90801c135cSArtem B. Bityutskiy 	ubi_assert(vol->exclusive == 1 && desc->mode == UBI_EXCLUSIVE);
91801c135cSArtem B. Bityutskiy 	vol->exclusive = 0;
92801c135cSArtem B. Bityutskiy 	if (mode == UBI_READONLY)
93801c135cSArtem B. Bityutskiy 		vol->readers = 1;
94801c135cSArtem B. Bityutskiy 	else if (mode == UBI_READWRITE)
95801c135cSArtem B. Bityutskiy 		vol->writers = 1;
96801c135cSArtem B. Bityutskiy 	else
97801c135cSArtem B. Bityutskiy 		vol->exclusive = 1;
98801c135cSArtem B. Bityutskiy 	spin_unlock(&vol->ubi->volumes_lock);
99801c135cSArtem B. Bityutskiy 
100801c135cSArtem B. Bityutskiy 	desc->mode = mode;
101801c135cSArtem B. Bityutskiy }
102801c135cSArtem B. Bityutskiy 
103801c135cSArtem B. Bityutskiy static int vol_cdev_open(struct inode *inode, struct file *file)
104801c135cSArtem B. Bityutskiy {
105801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc;
106e73f4459SArtem Bityutskiy 	int vol_id = iminor(inode) - 1, mode, ubi_num;
107e73f4459SArtem Bityutskiy 
108e73f4459SArtem Bityutskiy 	ubi_num = ubi_major2num(imajor(inode));
1097d200e88SArtem Bityutskiy 	if (ubi_num < 0)
110e73f4459SArtem Bityutskiy 		return ubi_num;
111801c135cSArtem B. Bityutskiy 
112801c135cSArtem B. Bityutskiy 	if (file->f_mode & FMODE_WRITE)
113801c135cSArtem B. Bityutskiy 		mode = UBI_READWRITE;
114801c135cSArtem B. Bityutskiy 	else
115801c135cSArtem B. Bityutskiy 		mode = UBI_READONLY;
116801c135cSArtem B. Bityutskiy 
117e1cf7e6dSArtem Bityutskiy 	dbg_gen("open device %d, volume %d, mode %d",
118e1cf7e6dSArtem Bityutskiy 		ubi_num, vol_id, mode);
119801c135cSArtem B. Bityutskiy 
120e73f4459SArtem Bityutskiy 	desc = ubi_open_volume(ubi_num, vol_id, mode);
121801c135cSArtem B. Bityutskiy 	if (IS_ERR(desc))
122801c135cSArtem B. Bityutskiy 		return PTR_ERR(desc);
123801c135cSArtem B. Bityutskiy 
124801c135cSArtem B. Bityutskiy 	file->private_data = desc;
125801c135cSArtem B. Bityutskiy 	return 0;
126801c135cSArtem B. Bityutskiy }
127801c135cSArtem B. Bityutskiy 
128801c135cSArtem B. Bityutskiy static int vol_cdev_release(struct inode *inode, struct file *file)
129801c135cSArtem B. Bityutskiy {
130801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
131801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
132801c135cSArtem B. Bityutskiy 
133e1cf7e6dSArtem Bityutskiy 	dbg_gen("release device %d, volume %d, mode %d",
134e1cf7e6dSArtem Bityutskiy 		vol->ubi->ubi_num, vol->vol_id, desc->mode);
135801c135cSArtem B. Bityutskiy 
136801c135cSArtem B. Bityutskiy 	if (vol->updating) {
137801c135cSArtem B. Bityutskiy 		ubi_warn("update of volume %d not finished, volume is damaged",
138801c135cSArtem B. Bityutskiy 			 vol->vol_id);
139e653879cSArtem Bityutskiy 		ubi_assert(!vol->changing_leb);
140801c135cSArtem B. Bityutskiy 		vol->updating = 0;
14192ad8f37SArtem Bityutskiy 		vfree(vol->upd_buf);
142e653879cSArtem Bityutskiy 	} else if (vol->changing_leb) {
143c8566350SArtem Bityutskiy 		dbg_gen("only %lld of %lld bytes received for atomic LEB change"
144e653879cSArtem Bityutskiy 			" for volume %d:%d, cancel", vol->upd_received,
145e653879cSArtem Bityutskiy 			vol->upd_bytes, vol->ubi->ubi_num, vol->vol_id);
146e653879cSArtem Bityutskiy 		vol->changing_leb = 0;
147e653879cSArtem Bityutskiy 		vfree(vol->upd_buf);
148801c135cSArtem B. Bityutskiy 	}
149801c135cSArtem B. Bityutskiy 
150801c135cSArtem B. Bityutskiy 	ubi_close_volume(desc);
151801c135cSArtem B. Bityutskiy 	return 0;
152801c135cSArtem B. Bityutskiy }
153801c135cSArtem B. Bityutskiy 
154801c135cSArtem B. Bityutskiy static loff_t vol_cdev_llseek(struct file *file, loff_t offset, int origin)
155801c135cSArtem B. Bityutskiy {
156801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
157801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
158801c135cSArtem B. Bityutskiy 	loff_t new_offset;
159801c135cSArtem B. Bityutskiy 
160801c135cSArtem B. Bityutskiy 	if (vol->updating) {
161801c135cSArtem B. Bityutskiy 		/* Update is in progress, seeking is prohibited */
162801c135cSArtem B. Bityutskiy 		dbg_err("updating");
163801c135cSArtem B. Bityutskiy 		return -EBUSY;
164801c135cSArtem B. Bityutskiy 	}
165801c135cSArtem B. Bityutskiy 
166801c135cSArtem B. Bityutskiy 	switch (origin) {
167801c135cSArtem B. Bityutskiy 	case 0: /* SEEK_SET */
168801c135cSArtem B. Bityutskiy 		new_offset = offset;
169801c135cSArtem B. Bityutskiy 		break;
170801c135cSArtem B. Bityutskiy 	case 1: /* SEEK_CUR */
171801c135cSArtem B. Bityutskiy 		new_offset = file->f_pos + offset;
172801c135cSArtem B. Bityutskiy 		break;
173801c135cSArtem B. Bityutskiy 	case 2: /* SEEK_END */
174801c135cSArtem B. Bityutskiy 		new_offset = vol->used_bytes + offset;
175801c135cSArtem B. Bityutskiy 		break;
176801c135cSArtem B. Bityutskiy 	default:
177801c135cSArtem B. Bityutskiy 		return -EINVAL;
178801c135cSArtem B. Bityutskiy 	}
179801c135cSArtem B. Bityutskiy 
180801c135cSArtem B. Bityutskiy 	if (new_offset < 0 || new_offset > vol->used_bytes) {
181801c135cSArtem B. Bityutskiy 		dbg_err("bad seek %lld", new_offset);
182801c135cSArtem B. Bityutskiy 		return -EINVAL;
183801c135cSArtem B. Bityutskiy 	}
184801c135cSArtem B. Bityutskiy 
185c8566350SArtem Bityutskiy 	dbg_gen("seek volume %d, offset %lld, origin %d, new offset %lld",
186801c135cSArtem B. Bityutskiy 		vol->vol_id, offset, origin, new_offset);
187801c135cSArtem B. Bityutskiy 
188801c135cSArtem B. Bityutskiy 	file->f_pos = new_offset;
189801c135cSArtem B. Bityutskiy 	return new_offset;
190801c135cSArtem B. Bityutskiy }
191801c135cSArtem B. Bityutskiy 
19202c24a82SJosef Bacik static int vol_cdev_fsync(struct file *file, loff_t start, loff_t end, int datasync)
1931b24bc3aSCorentin Chary {
1941b24bc3aSCorentin Chary 	struct ubi_volume_desc *desc = file->private_data;
1951b24bc3aSCorentin Chary 	struct ubi_device *ubi = desc->vol->ubi;
19602c24a82SJosef Bacik 	struct inode *inode = file->f_path.dentry->d_inode;
19702c24a82SJosef Bacik 	int err;
19802c24a82SJosef Bacik 	mutex_lock(&inode->i_mutex);
19902c24a82SJosef Bacik 	err = ubi_sync(ubi->ubi_num);
20002c24a82SJosef Bacik 	mutex_unlock(&inode->i_mutex);
20102c24a82SJosef Bacik 	return err;
2021b24bc3aSCorentin Chary }
2031b24bc3aSCorentin Chary 
2041b24bc3aSCorentin Chary 
205801c135cSArtem B. Bityutskiy static ssize_t vol_cdev_read(struct file *file, __user char *buf, size_t count,
206801c135cSArtem B. Bityutskiy 			     loff_t *offp)
207801c135cSArtem B. Bityutskiy {
208801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
209801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
210801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
211ae616e1bSArtem Bityutskiy 	int err, lnum, off, len,  tbuf_size;
212801c135cSArtem B. Bityutskiy 	size_t count_save = count;
213801c135cSArtem B. Bityutskiy 	void *tbuf;
214801c135cSArtem B. Bityutskiy 
215c8566350SArtem Bityutskiy 	dbg_gen("read %zd bytes from offset %lld of volume %d",
216ae616e1bSArtem Bityutskiy 		count, *offp, vol->vol_id);
217801c135cSArtem B. Bityutskiy 
218801c135cSArtem B. Bityutskiy 	if (vol->updating) {
219801c135cSArtem B. Bityutskiy 		dbg_err("updating");
220801c135cSArtem B. Bityutskiy 		return -EBUSY;
221801c135cSArtem B. Bityutskiy 	}
222801c135cSArtem B. Bityutskiy 	if (vol->upd_marker) {
223801c135cSArtem B. Bityutskiy 		dbg_err("damaged volume, update marker is set");
224801c135cSArtem B. Bityutskiy 		return -EBADF;
225801c135cSArtem B. Bityutskiy 	}
226801c135cSArtem B. Bityutskiy 	if (*offp == vol->used_bytes || count == 0)
227801c135cSArtem B. Bityutskiy 		return 0;
228801c135cSArtem B. Bityutskiy 
229801c135cSArtem B. Bityutskiy 	if (vol->corrupted)
230c8566350SArtem Bityutskiy 		dbg_gen("read from corrupted volume %d", vol->vol_id);
231801c135cSArtem B. Bityutskiy 
232801c135cSArtem B. Bityutskiy 	if (*offp + count > vol->used_bytes)
233801c135cSArtem B. Bityutskiy 		count_save = count = vol->used_bytes - *offp;
234801c135cSArtem B. Bityutskiy 
235801c135cSArtem B. Bityutskiy 	tbuf_size = vol->usable_leb_size;
236801c135cSArtem B. Bityutskiy 	if (count < tbuf_size)
237801c135cSArtem B. Bityutskiy 		tbuf_size = ALIGN(count, ubi->min_io_size);
23892ad8f37SArtem Bityutskiy 	tbuf = vmalloc(tbuf_size);
239801c135cSArtem B. Bityutskiy 	if (!tbuf)
240801c135cSArtem B. Bityutskiy 		return -ENOMEM;
241801c135cSArtem B. Bityutskiy 
242801c135cSArtem B. Bityutskiy 	len = count > tbuf_size ? tbuf_size : count;
2433013ee31SArtem Bityutskiy 	lnum = div_u64_rem(*offp, vol->usable_leb_size, &off);
244801c135cSArtem B. Bityutskiy 
245801c135cSArtem B. Bityutskiy 	do {
246801c135cSArtem B. Bityutskiy 		cond_resched();
247801c135cSArtem B. Bityutskiy 
248801c135cSArtem B. Bityutskiy 		if (off + len >= vol->usable_leb_size)
249801c135cSArtem B. Bityutskiy 			len = vol->usable_leb_size - off;
250801c135cSArtem B. Bityutskiy 
25189b96b69SArtem Bityutskiy 		err = ubi_eba_read_leb(ubi, vol, lnum, tbuf, off, len, 0);
252801c135cSArtem B. Bityutskiy 		if (err)
253801c135cSArtem B. Bityutskiy 			break;
254801c135cSArtem B. Bityutskiy 
255801c135cSArtem B. Bityutskiy 		off += len;
256801c135cSArtem B. Bityutskiy 		if (off == vol->usable_leb_size) {
257801c135cSArtem B. Bityutskiy 			lnum += 1;
258801c135cSArtem B. Bityutskiy 			off -= vol->usable_leb_size;
259801c135cSArtem B. Bityutskiy 		}
260801c135cSArtem B. Bityutskiy 
261801c135cSArtem B. Bityutskiy 		count -= len;
262801c135cSArtem B. Bityutskiy 		*offp += len;
263801c135cSArtem B. Bityutskiy 
264801c135cSArtem B. Bityutskiy 		err = copy_to_user(buf, tbuf, len);
265801c135cSArtem B. Bityutskiy 		if (err) {
266801c135cSArtem B. Bityutskiy 			err = -EFAULT;
267801c135cSArtem B. Bityutskiy 			break;
268801c135cSArtem B. Bityutskiy 		}
269801c135cSArtem B. Bityutskiy 
270801c135cSArtem B. Bityutskiy 		buf += len;
271801c135cSArtem B. Bityutskiy 		len = count > tbuf_size ? tbuf_size : count;
272801c135cSArtem B. Bityutskiy 	} while (count);
273801c135cSArtem B. Bityutskiy 
27492ad8f37SArtem Bityutskiy 	vfree(tbuf);
275801c135cSArtem B. Bityutskiy 	return err ? err : count_save - count;
276801c135cSArtem B. Bityutskiy }
277801c135cSArtem B. Bityutskiy 
278801c135cSArtem B. Bityutskiy /*
279801c135cSArtem B. Bityutskiy  * This function allows to directly write to dynamic UBI volumes, without
280766fb95bSSidney Amani  * issuing the volume update operation.
281801c135cSArtem B. Bityutskiy  */
282801c135cSArtem B. Bityutskiy static ssize_t vol_cdev_direct_write(struct file *file, const char __user *buf,
283801c135cSArtem B. Bityutskiy 				     size_t count, loff_t *offp)
284801c135cSArtem B. Bityutskiy {
285801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
286801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
287801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
288ae616e1bSArtem Bityutskiy 	int lnum, off, len, tbuf_size, err = 0;
289801c135cSArtem B. Bityutskiy 	size_t count_save = count;
290801c135cSArtem B. Bityutskiy 	char *tbuf;
291801c135cSArtem B. Bityutskiy 
292766fb95bSSidney Amani 	if (!vol->direct_writes)
293766fb95bSSidney Amani 		return -EPERM;
294766fb95bSSidney Amani 
295c8566350SArtem Bityutskiy 	dbg_gen("requested: write %zd bytes to offset %lld of volume %u",
296ae616e1bSArtem Bityutskiy 		count, *offp, vol->vol_id);
297801c135cSArtem B. Bityutskiy 
298801c135cSArtem B. Bityutskiy 	if (vol->vol_type == UBI_STATIC_VOLUME)
299801c135cSArtem B. Bityutskiy 		return -EROFS;
300801c135cSArtem B. Bityutskiy 
3013013ee31SArtem Bityutskiy 	lnum = div_u64_rem(*offp, vol->usable_leb_size, &off);
302cadb40ccSKyungmin Park 	if (off & (ubi->min_io_size - 1)) {
303801c135cSArtem B. Bityutskiy 		dbg_err("unaligned position");
304801c135cSArtem B. Bityutskiy 		return -EINVAL;
305801c135cSArtem B. Bityutskiy 	}
306801c135cSArtem B. Bityutskiy 
307801c135cSArtem B. Bityutskiy 	if (*offp + count > vol->used_bytes)
308801c135cSArtem B. Bityutskiy 		count_save = count = vol->used_bytes - *offp;
309801c135cSArtem B. Bityutskiy 
310801c135cSArtem B. Bityutskiy 	/* We can write only in fractions of the minimum I/O unit */
311cadb40ccSKyungmin Park 	if (count & (ubi->min_io_size - 1)) {
312801c135cSArtem B. Bityutskiy 		dbg_err("unaligned write length");
313801c135cSArtem B. Bityutskiy 		return -EINVAL;
314801c135cSArtem B. Bityutskiy 	}
315801c135cSArtem B. Bityutskiy 
316801c135cSArtem B. Bityutskiy 	tbuf_size = vol->usable_leb_size;
317801c135cSArtem B. Bityutskiy 	if (count < tbuf_size)
318801c135cSArtem B. Bityutskiy 		tbuf_size = ALIGN(count, ubi->min_io_size);
31992ad8f37SArtem Bityutskiy 	tbuf = vmalloc(tbuf_size);
320801c135cSArtem B. Bityutskiy 	if (!tbuf)
321801c135cSArtem B. Bityutskiy 		return -ENOMEM;
322801c135cSArtem B. Bityutskiy 
323801c135cSArtem B. Bityutskiy 	len = count > tbuf_size ? tbuf_size : count;
324801c135cSArtem B. Bityutskiy 
325801c135cSArtem B. Bityutskiy 	while (count) {
326801c135cSArtem B. Bityutskiy 		cond_resched();
327801c135cSArtem B. Bityutskiy 
328801c135cSArtem B. Bityutskiy 		if (off + len >= vol->usable_leb_size)
329801c135cSArtem B. Bityutskiy 			len = vol->usable_leb_size - off;
330801c135cSArtem B. Bityutskiy 
331801c135cSArtem B. Bityutskiy 		err = copy_from_user(tbuf, buf, len);
332801c135cSArtem B. Bityutskiy 		if (err) {
333801c135cSArtem B. Bityutskiy 			err = -EFAULT;
334801c135cSArtem B. Bityutskiy 			break;
335801c135cSArtem B. Bityutskiy 		}
336801c135cSArtem B. Bityutskiy 
33789b96b69SArtem Bityutskiy 		err = ubi_eba_write_leb(ubi, vol, lnum, tbuf, off, len,
338801c135cSArtem B. Bityutskiy 					UBI_UNKNOWN);
339801c135cSArtem B. Bityutskiy 		if (err)
340801c135cSArtem B. Bityutskiy 			break;
341801c135cSArtem B. Bityutskiy 
342801c135cSArtem B. Bityutskiy 		off += len;
343801c135cSArtem B. Bityutskiy 		if (off == vol->usable_leb_size) {
344801c135cSArtem B. Bityutskiy 			lnum += 1;
345801c135cSArtem B. Bityutskiy 			off -= vol->usable_leb_size;
346801c135cSArtem B. Bityutskiy 		}
347801c135cSArtem B. Bityutskiy 
348801c135cSArtem B. Bityutskiy 		count -= len;
349801c135cSArtem B. Bityutskiy 		*offp += len;
350801c135cSArtem B. Bityutskiy 		buf += len;
351801c135cSArtem B. Bityutskiy 		len = count > tbuf_size ? tbuf_size : count;
352801c135cSArtem B. Bityutskiy 	}
353801c135cSArtem B. Bityutskiy 
35492ad8f37SArtem Bityutskiy 	vfree(tbuf);
355801c135cSArtem B. Bityutskiy 	return err ? err : count_save - count;
356801c135cSArtem B. Bityutskiy }
357801c135cSArtem B. Bityutskiy 
358801c135cSArtem B. Bityutskiy static ssize_t vol_cdev_write(struct file *file, const char __user *buf,
359801c135cSArtem B. Bityutskiy 			      size_t count, loff_t *offp)
360801c135cSArtem B. Bityutskiy {
361801c135cSArtem B. Bityutskiy 	int err = 0;
362801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
363801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
364801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
365801c135cSArtem B. Bityutskiy 
366e653879cSArtem Bityutskiy 	if (!vol->updating && !vol->changing_leb)
367801c135cSArtem B. Bityutskiy 		return vol_cdev_direct_write(file, buf, count, offp);
368801c135cSArtem B. Bityutskiy 
369e653879cSArtem Bityutskiy 	if (vol->updating)
3701b68d0eeSArtem Bityutskiy 		err = ubi_more_update_data(ubi, vol, buf, count);
371e653879cSArtem Bityutskiy 	else
372e653879cSArtem Bityutskiy 		err = ubi_more_leb_change_data(ubi, vol, buf, count);
373e653879cSArtem Bityutskiy 
374801c135cSArtem B. Bityutskiy 	if (err < 0) {
375e653879cSArtem Bityutskiy 		ubi_err("cannot accept more %zd bytes of data, error %d",
37601f7b309SArtem Bityutskiy 			count, err);
377801c135cSArtem B. Bityutskiy 		return err;
378801c135cSArtem B. Bityutskiy 	}
379801c135cSArtem B. Bityutskiy 
380801c135cSArtem B. Bityutskiy 	if (err) {
381801c135cSArtem B. Bityutskiy 		/*
382e653879cSArtem Bityutskiy 		 * The operation is finished, @err contains number of actually
383e653879cSArtem Bityutskiy 		 * written bytes.
384801c135cSArtem B. Bityutskiy 		 */
385801c135cSArtem B. Bityutskiy 		count = err;
386801c135cSArtem B. Bityutskiy 
387e653879cSArtem Bityutskiy 		if (vol->changing_leb) {
388e653879cSArtem Bityutskiy 			revoke_exclusive(desc, UBI_READWRITE);
389e653879cSArtem Bityutskiy 			return count;
390e653879cSArtem Bityutskiy 		}
391e653879cSArtem Bityutskiy 
392801c135cSArtem B. Bityutskiy 		err = ubi_check_volume(ubi, vol->vol_id);
393801c135cSArtem B. Bityutskiy 		if (err < 0)
394801c135cSArtem B. Bityutskiy 			return err;
395801c135cSArtem B. Bityutskiy 
396801c135cSArtem B. Bityutskiy 		if (err) {
397801c135cSArtem B. Bityutskiy 			ubi_warn("volume %d on UBI device %d is corrupted",
398801c135cSArtem B. Bityutskiy 				 vol->vol_id, ubi->ubi_num);
399801c135cSArtem B. Bityutskiy 			vol->corrupted = 1;
400801c135cSArtem B. Bityutskiy 		}
401801c135cSArtem B. Bityutskiy 		vol->checked = 1;
4020e0ee1ccSDmitry Pervushin 		ubi_volume_notify(ubi, vol, UBI_VOLUME_UPDATED);
403801c135cSArtem B. Bityutskiy 		revoke_exclusive(desc, UBI_READWRITE);
404801c135cSArtem B. Bityutskiy 	}
405801c135cSArtem B. Bityutskiy 
406801c135cSArtem B. Bityutskiy 	return count;
407801c135cSArtem B. Bityutskiy }
408801c135cSArtem B. Bityutskiy 
409f429b2eaSArtem Bityutskiy static long vol_cdev_ioctl(struct file *file, unsigned int cmd,
410f429b2eaSArtem Bityutskiy 			   unsigned long arg)
411801c135cSArtem B. Bityutskiy {
412801c135cSArtem B. Bityutskiy 	int err = 0;
413801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc = file->private_data;
414801c135cSArtem B. Bityutskiy 	struct ubi_volume *vol = desc->vol;
415801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi = vol->ubi;
416801c135cSArtem B. Bityutskiy 	void __user *argp = (void __user *)arg;
417801c135cSArtem B. Bityutskiy 
418801c135cSArtem B. Bityutskiy 	switch (cmd) {
419801c135cSArtem B. Bityutskiy 	/* Volume update command */
420801c135cSArtem B. Bityutskiy 	case UBI_IOCVOLUP:
421801c135cSArtem B. Bityutskiy 	{
422801c135cSArtem B. Bityutskiy 		int64_t bytes, rsvd_bytes;
423801c135cSArtem B. Bityutskiy 
424801c135cSArtem B. Bityutskiy 		if (!capable(CAP_SYS_RESOURCE)) {
425801c135cSArtem B. Bityutskiy 			err = -EPERM;
426801c135cSArtem B. Bityutskiy 			break;
427801c135cSArtem B. Bityutskiy 		}
428801c135cSArtem B. Bityutskiy 
429801c135cSArtem B. Bityutskiy 		err = copy_from_user(&bytes, argp, sizeof(int64_t));
430801c135cSArtem B. Bityutskiy 		if (err) {
431801c135cSArtem B. Bityutskiy 			err = -EFAULT;
432801c135cSArtem B. Bityutskiy 			break;
433801c135cSArtem B. Bityutskiy 		}
434801c135cSArtem B. Bityutskiy 
435801c135cSArtem B. Bityutskiy 		if (desc->mode == UBI_READONLY) {
436801c135cSArtem B. Bityutskiy 			err = -EROFS;
437801c135cSArtem B. Bityutskiy 			break;
438801c135cSArtem B. Bityutskiy 		}
439801c135cSArtem B. Bityutskiy 
44073789a3dSBruce Leonard 		rsvd_bytes = (long long)vol->reserved_pebs *
44173789a3dSBruce Leonard 					ubi->leb_size-vol->data_pad;
442801c135cSArtem B. Bityutskiy 		if (bytes < 0 || bytes > rsvd_bytes) {
443801c135cSArtem B. Bityutskiy 			err = -EINVAL;
444801c135cSArtem B. Bityutskiy 			break;
445801c135cSArtem B. Bityutskiy 		}
446801c135cSArtem B. Bityutskiy 
447801c135cSArtem B. Bityutskiy 		err = get_exclusive(desc);
448801c135cSArtem B. Bityutskiy 		if (err < 0)
449801c135cSArtem B. Bityutskiy 			break;
450801c135cSArtem B. Bityutskiy 
4511b68d0eeSArtem Bityutskiy 		err = ubi_start_update(ubi, vol, bytes);
452801c135cSArtem B. Bityutskiy 		if (bytes == 0)
453801c135cSArtem B. Bityutskiy 			revoke_exclusive(desc, UBI_READWRITE);
454801c135cSArtem B. Bityutskiy 		break;
455801c135cSArtem B. Bityutskiy 	}
456801c135cSArtem B. Bityutskiy 
457e653879cSArtem Bityutskiy 	/* Atomic logical eraseblock change command */
458e653879cSArtem Bityutskiy 	case UBI_IOCEBCH:
459e653879cSArtem Bityutskiy 	{
460e653879cSArtem Bityutskiy 		struct ubi_leb_change_req req;
461e653879cSArtem Bityutskiy 
462e653879cSArtem Bityutskiy 		err = copy_from_user(&req, argp,
463e653879cSArtem Bityutskiy 				     sizeof(struct ubi_leb_change_req));
464e653879cSArtem Bityutskiy 		if (err) {
465e653879cSArtem Bityutskiy 			err = -EFAULT;
466e653879cSArtem Bityutskiy 			break;
467e653879cSArtem Bityutskiy 		}
468e653879cSArtem Bityutskiy 
469e653879cSArtem Bityutskiy 		if (desc->mode == UBI_READONLY ||
470e653879cSArtem Bityutskiy 		    vol->vol_type == UBI_STATIC_VOLUME) {
471e653879cSArtem Bityutskiy 			err = -EROFS;
472e653879cSArtem Bityutskiy 			break;
473e653879cSArtem Bityutskiy 		}
474e653879cSArtem Bityutskiy 
475e653879cSArtem Bityutskiy 		/* Validate the request */
476e653879cSArtem Bityutskiy 		err = -EINVAL;
477e653879cSArtem Bityutskiy 		if (req.lnum < 0 || req.lnum >= vol->reserved_pebs ||
478e653879cSArtem Bityutskiy 		    req.bytes < 0 || req.lnum >= vol->usable_leb_size)
479e653879cSArtem Bityutskiy 			break;
480e653879cSArtem Bityutskiy 		if (req.dtype != UBI_LONGTERM && req.dtype != UBI_SHORTTERM &&
481e653879cSArtem Bityutskiy 		    req.dtype != UBI_UNKNOWN)
482e653879cSArtem Bityutskiy 			break;
483e653879cSArtem Bityutskiy 
484e653879cSArtem Bityutskiy 		err = get_exclusive(desc);
485e653879cSArtem Bityutskiy 		if (err < 0)
486e653879cSArtem Bityutskiy 			break;
487e653879cSArtem Bityutskiy 
488e653879cSArtem Bityutskiy 		err = ubi_start_leb_change(ubi, vol, &req);
489e653879cSArtem Bityutskiy 		if (req.bytes == 0)
490e653879cSArtem Bityutskiy 			revoke_exclusive(desc, UBI_READWRITE);
491e653879cSArtem Bityutskiy 		break;
492e653879cSArtem Bityutskiy 	}
493e653879cSArtem Bityutskiy 
494801c135cSArtem B. Bityutskiy 	/* Logical eraseblock erasure command */
495801c135cSArtem B. Bityutskiy 	case UBI_IOCEBER:
496801c135cSArtem B. Bityutskiy 	{
497801c135cSArtem B. Bityutskiy 		int32_t lnum;
498801c135cSArtem B. Bityutskiy 
499bf07803aSChristoph Hellwig 		err = get_user(lnum, (__user int32_t *)argp);
500801c135cSArtem B. Bityutskiy 		if (err) {
501801c135cSArtem B. Bityutskiy 			err = -EFAULT;
502801c135cSArtem B. Bityutskiy 			break;
503801c135cSArtem B. Bityutskiy 		}
504801c135cSArtem B. Bityutskiy 
505e653879cSArtem Bityutskiy 		if (desc->mode == UBI_READONLY ||
506e653879cSArtem Bityutskiy 		    vol->vol_type == UBI_STATIC_VOLUME) {
507801c135cSArtem B. Bityutskiy 			err = -EROFS;
508801c135cSArtem B. Bityutskiy 			break;
509801c135cSArtem B. Bityutskiy 		}
510801c135cSArtem B. Bityutskiy 
511801c135cSArtem B. Bityutskiy 		if (lnum < 0 || lnum >= vol->reserved_pebs) {
512801c135cSArtem B. Bityutskiy 			err = -EINVAL;
513801c135cSArtem B. Bityutskiy 			break;
514801c135cSArtem B. Bityutskiy 		}
515801c135cSArtem B. Bityutskiy 
516c8566350SArtem Bityutskiy 		dbg_gen("erase LEB %d:%d", vol->vol_id, lnum);
51789b96b69SArtem Bityutskiy 		err = ubi_eba_unmap_leb(ubi, vol, lnum);
518801c135cSArtem B. Bityutskiy 		if (err)
519801c135cSArtem B. Bityutskiy 			break;
520801c135cSArtem B. Bityutskiy 
521801c135cSArtem B. Bityutskiy 		err = ubi_wl_flush(ubi);
522801c135cSArtem B. Bityutskiy 		break;
523801c135cSArtem B. Bityutskiy 	}
524141e6ebdSCorentin Chary 
525141e6ebdSCorentin Chary 	/* Logical eraseblock map command */
526141e6ebdSCorentin Chary 	case UBI_IOCEBMAP:
527141e6ebdSCorentin Chary 	{
528141e6ebdSCorentin Chary 		struct ubi_map_req req;
529141e6ebdSCorentin Chary 
530141e6ebdSCorentin Chary 		err = copy_from_user(&req, argp, sizeof(struct ubi_map_req));
531141e6ebdSCorentin Chary 		if (err) {
532141e6ebdSCorentin Chary 			err = -EFAULT;
533141e6ebdSCorentin Chary 			break;
534141e6ebdSCorentin Chary 		}
535141e6ebdSCorentin Chary 		err = ubi_leb_map(desc, req.lnum, req.dtype);
536141e6ebdSCorentin Chary 		break;
537141e6ebdSCorentin Chary 	}
538c3da23beSCorentin Chary 
539c3da23beSCorentin Chary 	/* Logical eraseblock un-map command */
540c3da23beSCorentin Chary 	case UBI_IOCEBUNMAP:
541c3da23beSCorentin Chary 	{
542c3da23beSCorentin Chary 		int32_t lnum;
543c3da23beSCorentin Chary 
544c3da23beSCorentin Chary 		err = get_user(lnum, (__user int32_t *)argp);
545c3da23beSCorentin Chary 		if (err) {
546c3da23beSCorentin Chary 			err = -EFAULT;
547c3da23beSCorentin Chary 			break;
548c3da23beSCorentin Chary 		}
549c3da23beSCorentin Chary 		err = ubi_leb_unmap(desc, lnum);
550c3da23beSCorentin Chary 		break;
551c3da23beSCorentin Chary 	}
552a27ce8f5SCorentin Chary 
553a27ce8f5SCorentin Chary 	/* Check if logical eraseblock is mapped command */
554a27ce8f5SCorentin Chary 	case UBI_IOCEBISMAP:
555a27ce8f5SCorentin Chary 	{
556a27ce8f5SCorentin Chary 		int32_t lnum;
557a27ce8f5SCorentin Chary 
558a27ce8f5SCorentin Chary 		err = get_user(lnum, (__user int32_t *)argp);
559a27ce8f5SCorentin Chary 		if (err) {
560a27ce8f5SCorentin Chary 			err = -EFAULT;
561a27ce8f5SCorentin Chary 			break;
562a27ce8f5SCorentin Chary 		}
563a27ce8f5SCorentin Chary 		err = ubi_is_mapped(desc, lnum);
564a27ce8f5SCorentin Chary 		break;
565a27ce8f5SCorentin Chary 	}
566801c135cSArtem B. Bityutskiy 
567766fb95bSSidney Amani 	/* Set volume property command */
5686748482fSArtem Bityutskiy 	case UBI_IOCSETVOLPROP:
569766fb95bSSidney Amani 	{
5706748482fSArtem Bityutskiy 		struct ubi_set_vol_prop_req req;
571766fb95bSSidney Amani 
572766fb95bSSidney Amani 		err = copy_from_user(&req, argp,
5736748482fSArtem Bityutskiy 				     sizeof(struct ubi_set_vol_prop_req));
574766fb95bSSidney Amani 		if (err) {
575766fb95bSSidney Amani 			err = -EFAULT;
576766fb95bSSidney Amani 			break;
577766fb95bSSidney Amani 		}
578766fb95bSSidney Amani 		switch (req.property) {
5796748482fSArtem Bityutskiy 		case UBI_VOL_PROP_DIRECT_WRITE:
580f089c0b2SArtem Bityutskiy 			mutex_lock(&ubi->device_mutex);
581766fb95bSSidney Amani 			desc->vol->direct_writes = !!req.value;
582f089c0b2SArtem Bityutskiy 			mutex_unlock(&ubi->device_mutex);
583766fb95bSSidney Amani 			break;
584766fb95bSSidney Amani 		default:
585766fb95bSSidney Amani 			err = -EINVAL;
586766fb95bSSidney Amani 			break;
587766fb95bSSidney Amani 		}
588766fb95bSSidney Amani 		break;
589766fb95bSSidney Amani 	}
590766fb95bSSidney Amani 
591801c135cSArtem B. Bityutskiy 	default:
592801c135cSArtem B. Bityutskiy 		err = -ENOTTY;
593801c135cSArtem B. Bityutskiy 		break;
594801c135cSArtem B. Bityutskiy 	}
595801c135cSArtem B. Bityutskiy 	return err;
596801c135cSArtem B. Bityutskiy }
597801c135cSArtem B. Bityutskiy 
598801c135cSArtem B. Bityutskiy /**
599801c135cSArtem B. Bityutskiy  * verify_mkvol_req - verify volume creation request.
600801c135cSArtem B. Bityutskiy  * @ubi: UBI device description object
601801c135cSArtem B. Bityutskiy  * @req: the request to check
602801c135cSArtem B. Bityutskiy  *
603801c135cSArtem B. Bityutskiy  * This function zero if the request is correct, and %-EINVAL if not.
604801c135cSArtem B. Bityutskiy  */
605801c135cSArtem B. Bityutskiy static int verify_mkvol_req(const struct ubi_device *ubi,
606801c135cSArtem B. Bityutskiy 			    const struct ubi_mkvol_req *req)
607801c135cSArtem B. Bityutskiy {
608801c135cSArtem B. Bityutskiy 	int n, err = -EINVAL;
609801c135cSArtem B. Bityutskiy 
610801c135cSArtem B. Bityutskiy 	if (req->bytes < 0 || req->alignment < 0 || req->vol_type < 0 ||
611801c135cSArtem B. Bityutskiy 	    req->name_len < 0)
612801c135cSArtem B. Bityutskiy 		goto bad;
613801c135cSArtem B. Bityutskiy 
614801c135cSArtem B. Bityutskiy 	if ((req->vol_id < 0 || req->vol_id >= ubi->vtbl_slots) &&
615801c135cSArtem B. Bityutskiy 	    req->vol_id != UBI_VOL_NUM_AUTO)
616801c135cSArtem B. Bityutskiy 		goto bad;
617801c135cSArtem B. Bityutskiy 
618801c135cSArtem B. Bityutskiy 	if (req->alignment == 0)
619801c135cSArtem B. Bityutskiy 		goto bad;
620801c135cSArtem B. Bityutskiy 
621801c135cSArtem B. Bityutskiy 	if (req->bytes == 0)
622801c135cSArtem B. Bityutskiy 		goto bad;
623801c135cSArtem B. Bityutskiy 
624801c135cSArtem B. Bityutskiy 	if (req->vol_type != UBI_DYNAMIC_VOLUME &&
625801c135cSArtem B. Bityutskiy 	    req->vol_type != UBI_STATIC_VOLUME)
626801c135cSArtem B. Bityutskiy 		goto bad;
627801c135cSArtem B. Bityutskiy 
628801c135cSArtem B. Bityutskiy 	if (req->alignment > ubi->leb_size)
629801c135cSArtem B. Bityutskiy 		goto bad;
630801c135cSArtem B. Bityutskiy 
631cadb40ccSKyungmin Park 	n = req->alignment & (ubi->min_io_size - 1);
632801c135cSArtem B. Bityutskiy 	if (req->alignment != 1 && n)
633801c135cSArtem B. Bityutskiy 		goto bad;
634801c135cSArtem B. Bityutskiy 
635*4a59c797SRichard Weinberger 	if (!req->name[0] || !req->name_len)
636*4a59c797SRichard Weinberger 		goto bad;
637*4a59c797SRichard Weinberger 
638801c135cSArtem B. Bityutskiy 	if (req->name_len > UBI_VOL_NAME_MAX) {
639801c135cSArtem B. Bityutskiy 		err = -ENAMETOOLONG;
640801c135cSArtem B. Bityutskiy 		goto bad;
641801c135cSArtem B. Bityutskiy 	}
642801c135cSArtem B. Bityutskiy 
643a6ea4407SArtem Bityutskiy 	n = strnlen(req->name, req->name_len + 1);
644a6ea4407SArtem Bityutskiy 	if (n != req->name_len)
645a6ea4407SArtem Bityutskiy 		goto bad;
646a6ea4407SArtem Bityutskiy 
647801c135cSArtem B. Bityutskiy 	return 0;
648801c135cSArtem B. Bityutskiy 
649801c135cSArtem B. Bityutskiy bad:
650801c135cSArtem B. Bityutskiy 	dbg_err("bad volume creation request");
651801c135cSArtem B. Bityutskiy 	ubi_dbg_dump_mkvol_req(req);
652801c135cSArtem B. Bityutskiy 	return err;
653801c135cSArtem B. Bityutskiy }
654801c135cSArtem B. Bityutskiy 
655801c135cSArtem B. Bityutskiy /**
656801c135cSArtem B. Bityutskiy  * verify_rsvol_req - verify volume re-size request.
657801c135cSArtem B. Bityutskiy  * @ubi: UBI device description object
658801c135cSArtem B. Bityutskiy  * @req: the request to check
659801c135cSArtem B. Bityutskiy  *
660801c135cSArtem B. Bityutskiy  * This function returns zero if the request is correct, and %-EINVAL if not.
661801c135cSArtem B. Bityutskiy  */
662801c135cSArtem B. Bityutskiy static int verify_rsvol_req(const struct ubi_device *ubi,
663801c135cSArtem B. Bityutskiy 			    const struct ubi_rsvol_req *req)
664801c135cSArtem B. Bityutskiy {
665801c135cSArtem B. Bityutskiy 	if (req->bytes <= 0)
666801c135cSArtem B. Bityutskiy 		return -EINVAL;
667801c135cSArtem B. Bityutskiy 
668801c135cSArtem B. Bityutskiy 	if (req->vol_id < 0 || req->vol_id >= ubi->vtbl_slots)
669801c135cSArtem B. Bityutskiy 		return -EINVAL;
670801c135cSArtem B. Bityutskiy 
671801c135cSArtem B. Bityutskiy 	return 0;
672801c135cSArtem B. Bityutskiy }
673801c135cSArtem B. Bityutskiy 
674f40ac9cdSArtem Bityutskiy /**
675f40ac9cdSArtem Bityutskiy  * rename_volumes - rename UBI volumes.
676f40ac9cdSArtem Bityutskiy  * @ubi: UBI device description object
677f40ac9cdSArtem Bityutskiy  * @req: volumes re-name request
678f40ac9cdSArtem Bityutskiy  *
679f40ac9cdSArtem Bityutskiy  * This is a helper function for the volume re-name IOCTL which validates the
680f40ac9cdSArtem Bityutskiy  * the request, opens the volume and calls corresponding volumes management
681f40ac9cdSArtem Bityutskiy  * function. Returns zero in case of success and a negative error code in case
682f40ac9cdSArtem Bityutskiy  * of failure.
683f40ac9cdSArtem Bityutskiy  */
684f40ac9cdSArtem Bityutskiy static int rename_volumes(struct ubi_device *ubi,
685f40ac9cdSArtem Bityutskiy 			  struct ubi_rnvol_req *req)
686f40ac9cdSArtem Bityutskiy {
687f40ac9cdSArtem Bityutskiy 	int i, n, err;
688f40ac9cdSArtem Bityutskiy 	struct list_head rename_list;
689f40ac9cdSArtem Bityutskiy 	struct ubi_rename_entry *re, *re1;
690f40ac9cdSArtem Bityutskiy 
691f40ac9cdSArtem Bityutskiy 	if (req->count < 0 || req->count > UBI_MAX_RNVOL)
692f40ac9cdSArtem Bityutskiy 		return -EINVAL;
693f40ac9cdSArtem Bityutskiy 
694f40ac9cdSArtem Bityutskiy 	if (req->count == 0)
695f40ac9cdSArtem Bityutskiy 		return 0;
696f40ac9cdSArtem Bityutskiy 
697f40ac9cdSArtem Bityutskiy 	/* Validate volume IDs and names in the request */
698f40ac9cdSArtem Bityutskiy 	for (i = 0; i < req->count; i++) {
699f40ac9cdSArtem Bityutskiy 		if (req->ents[i].vol_id < 0 ||
700f40ac9cdSArtem Bityutskiy 		    req->ents[i].vol_id >= ubi->vtbl_slots)
701f40ac9cdSArtem Bityutskiy 			return -EINVAL;
702f40ac9cdSArtem Bityutskiy 		if (req->ents[i].name_len < 0)
703f40ac9cdSArtem Bityutskiy 			return -EINVAL;
704f40ac9cdSArtem Bityutskiy 		if (req->ents[i].name_len > UBI_VOL_NAME_MAX)
705f40ac9cdSArtem Bityutskiy 			return -ENAMETOOLONG;
706f40ac9cdSArtem Bityutskiy 		req->ents[i].name[req->ents[i].name_len] = '\0';
707f40ac9cdSArtem Bityutskiy 		n = strlen(req->ents[i].name);
708f40ac9cdSArtem Bityutskiy 		if (n != req->ents[i].name_len)
709f40ac9cdSArtem Bityutskiy 			err = -EINVAL;
710f40ac9cdSArtem Bityutskiy 	}
711f40ac9cdSArtem Bityutskiy 
712f40ac9cdSArtem Bityutskiy 	/* Make sure volume IDs and names are unique */
713f40ac9cdSArtem Bityutskiy 	for (i = 0; i < req->count - 1; i++) {
714f40ac9cdSArtem Bityutskiy 		for (n = i + 1; n < req->count; n++) {
715f40ac9cdSArtem Bityutskiy 			if (req->ents[i].vol_id == req->ents[n].vol_id) {
716f40ac9cdSArtem Bityutskiy 				dbg_err("duplicated volume id %d",
717f40ac9cdSArtem Bityutskiy 					req->ents[i].vol_id);
718f40ac9cdSArtem Bityutskiy 				return -EINVAL;
719f40ac9cdSArtem Bityutskiy 			}
720f40ac9cdSArtem Bityutskiy 			if (!strcmp(req->ents[i].name, req->ents[n].name)) {
721f40ac9cdSArtem Bityutskiy 				dbg_err("duplicated volume name \"%s\"",
722f40ac9cdSArtem Bityutskiy 					req->ents[i].name);
723f40ac9cdSArtem Bityutskiy 				return -EINVAL;
724f40ac9cdSArtem Bityutskiy 			}
725f40ac9cdSArtem Bityutskiy 		}
726f40ac9cdSArtem Bityutskiy 	}
727f40ac9cdSArtem Bityutskiy 
728f40ac9cdSArtem Bityutskiy 	/* Create the re-name list */
729f40ac9cdSArtem Bityutskiy 	INIT_LIST_HEAD(&rename_list);
730f40ac9cdSArtem Bityutskiy 	for (i = 0; i < req->count; i++) {
731f40ac9cdSArtem Bityutskiy 		int vol_id = req->ents[i].vol_id;
732f40ac9cdSArtem Bityutskiy 		int name_len = req->ents[i].name_len;
733f40ac9cdSArtem Bityutskiy 		const char *name = req->ents[i].name;
734f40ac9cdSArtem Bityutskiy 
735f40ac9cdSArtem Bityutskiy 		re = kzalloc(sizeof(struct ubi_rename_entry), GFP_KERNEL);
736f40ac9cdSArtem Bityutskiy 		if (!re) {
737f40ac9cdSArtem Bityutskiy 			err = -ENOMEM;
738f40ac9cdSArtem Bityutskiy 			goto out_free;
739f40ac9cdSArtem Bityutskiy 		}
740f40ac9cdSArtem Bityutskiy 
741f40ac9cdSArtem Bityutskiy 		re->desc = ubi_open_volume(ubi->ubi_num, vol_id, UBI_EXCLUSIVE);
742f40ac9cdSArtem Bityutskiy 		if (IS_ERR(re->desc)) {
743f40ac9cdSArtem Bityutskiy 			err = PTR_ERR(re->desc);
744f40ac9cdSArtem Bityutskiy 			dbg_err("cannot open volume %d, error %d", vol_id, err);
745f40ac9cdSArtem Bityutskiy 			kfree(re);
746f40ac9cdSArtem Bityutskiy 			goto out_free;
747f40ac9cdSArtem Bityutskiy 		}
748f40ac9cdSArtem Bityutskiy 
749f40ac9cdSArtem Bityutskiy 		/* Skip this re-naming if the name does not really change */
750f40ac9cdSArtem Bityutskiy 		if (re->desc->vol->name_len == name_len &&
751f40ac9cdSArtem Bityutskiy 		    !memcmp(re->desc->vol->name, name, name_len)) {
752f40ac9cdSArtem Bityutskiy 			ubi_close_volume(re->desc);
753f40ac9cdSArtem Bityutskiy 			kfree(re);
754f40ac9cdSArtem Bityutskiy 			continue;
755f40ac9cdSArtem Bityutskiy 		}
756f40ac9cdSArtem Bityutskiy 
757f40ac9cdSArtem Bityutskiy 		re->new_name_len = name_len;
758f40ac9cdSArtem Bityutskiy 		memcpy(re->new_name, name, name_len);
759f40ac9cdSArtem Bityutskiy 		list_add_tail(&re->list, &rename_list);
760f40ac9cdSArtem Bityutskiy 		dbg_msg("will rename volume %d from \"%s\" to \"%s\"",
761f40ac9cdSArtem Bityutskiy 			vol_id, re->desc->vol->name, name);
762f40ac9cdSArtem Bityutskiy 	}
763f40ac9cdSArtem Bityutskiy 
764f40ac9cdSArtem Bityutskiy 	if (list_empty(&rename_list))
765f40ac9cdSArtem Bityutskiy 		return 0;
766f40ac9cdSArtem Bityutskiy 
767f40ac9cdSArtem Bityutskiy 	/* Find out the volumes which have to be removed */
768f40ac9cdSArtem Bityutskiy 	list_for_each_entry(re, &rename_list, list) {
769f40ac9cdSArtem Bityutskiy 		struct ubi_volume_desc *desc;
770f40ac9cdSArtem Bityutskiy 		int no_remove_needed = 0;
771f40ac9cdSArtem Bityutskiy 
772f40ac9cdSArtem Bityutskiy 		/*
773f40ac9cdSArtem Bityutskiy 		 * Volume @re->vol_id is going to be re-named to
774f40ac9cdSArtem Bityutskiy 		 * @re->new_name, while its current name is @name. If a volume
775f40ac9cdSArtem Bityutskiy 		 * with name @re->new_name currently exists, it has to be
776f40ac9cdSArtem Bityutskiy 		 * removed, unless it is also re-named in the request (@req).
777f40ac9cdSArtem Bityutskiy 		 */
778f40ac9cdSArtem Bityutskiy 		list_for_each_entry(re1, &rename_list, list) {
779f40ac9cdSArtem Bityutskiy 			if (re->new_name_len == re1->desc->vol->name_len &&
780f40ac9cdSArtem Bityutskiy 			    !memcmp(re->new_name, re1->desc->vol->name,
781f40ac9cdSArtem Bityutskiy 				    re1->desc->vol->name_len)) {
782f40ac9cdSArtem Bityutskiy 				no_remove_needed = 1;
783f40ac9cdSArtem Bityutskiy 				break;
784f40ac9cdSArtem Bityutskiy 			}
785f40ac9cdSArtem Bityutskiy 		}
786f40ac9cdSArtem Bityutskiy 
787f40ac9cdSArtem Bityutskiy 		if (no_remove_needed)
788f40ac9cdSArtem Bityutskiy 			continue;
789f40ac9cdSArtem Bityutskiy 
790f40ac9cdSArtem Bityutskiy 		/*
791f40ac9cdSArtem Bityutskiy 		 * It seems we need to remove volume with name @re->new_name,
792f40ac9cdSArtem Bityutskiy 		 * if it exists.
793f40ac9cdSArtem Bityutskiy 		 */
794f2863c54SArtem Bityutskiy 		desc = ubi_open_volume_nm(ubi->ubi_num, re->new_name,
795f2863c54SArtem Bityutskiy 					  UBI_EXCLUSIVE);
796f40ac9cdSArtem Bityutskiy 		if (IS_ERR(desc)) {
797f40ac9cdSArtem Bityutskiy 			err = PTR_ERR(desc);
798f40ac9cdSArtem Bityutskiy 			if (err == -ENODEV)
799f40ac9cdSArtem Bityutskiy 				/* Re-naming into a non-existing volume name */
800f40ac9cdSArtem Bityutskiy 				continue;
801f40ac9cdSArtem Bityutskiy 
802f40ac9cdSArtem Bityutskiy 			/* The volume exists but busy, or an error occurred */
803f40ac9cdSArtem Bityutskiy 			dbg_err("cannot open volume \"%s\", error %d",
804f40ac9cdSArtem Bityutskiy 				re->new_name, err);
805f40ac9cdSArtem Bityutskiy 			goto out_free;
806f40ac9cdSArtem Bityutskiy 		}
807f40ac9cdSArtem Bityutskiy 
80801ebc12fSJulia Lawall 		re1 = kzalloc(sizeof(struct ubi_rename_entry), GFP_KERNEL);
80901ebc12fSJulia Lawall 		if (!re1) {
810f40ac9cdSArtem Bityutskiy 			err = -ENOMEM;
811f40ac9cdSArtem Bityutskiy 			ubi_close_volume(desc);
812f40ac9cdSArtem Bityutskiy 			goto out_free;
813f40ac9cdSArtem Bityutskiy 		}
814f40ac9cdSArtem Bityutskiy 
81501ebc12fSJulia Lawall 		re1->remove = 1;
81601ebc12fSJulia Lawall 		re1->desc = desc;
81701ebc12fSJulia Lawall 		list_add(&re1->list, &rename_list);
818f40ac9cdSArtem Bityutskiy 		dbg_msg("will remove volume %d, name \"%s\"",
81901ebc12fSJulia Lawall 			re1->desc->vol->vol_id, re1->desc->vol->name);
820f40ac9cdSArtem Bityutskiy 	}
821f40ac9cdSArtem Bityutskiy 
822f089c0b2SArtem Bityutskiy 	mutex_lock(&ubi->device_mutex);
823f40ac9cdSArtem Bityutskiy 	err = ubi_rename_volumes(ubi, &rename_list);
824f089c0b2SArtem Bityutskiy 	mutex_unlock(&ubi->device_mutex);
825f40ac9cdSArtem Bityutskiy 
826f40ac9cdSArtem Bityutskiy out_free:
827f40ac9cdSArtem Bityutskiy 	list_for_each_entry_safe(re, re1, &rename_list, list) {
828f40ac9cdSArtem Bityutskiy 		ubi_close_volume(re->desc);
829f40ac9cdSArtem Bityutskiy 		list_del(&re->list);
830f40ac9cdSArtem Bityutskiy 		kfree(re);
831f40ac9cdSArtem Bityutskiy 	}
832f40ac9cdSArtem Bityutskiy 	return err;
833f40ac9cdSArtem Bityutskiy }
834f40ac9cdSArtem Bityutskiy 
835f429b2eaSArtem Bityutskiy static long ubi_cdev_ioctl(struct file *file, unsigned int cmd,
836f429b2eaSArtem Bityutskiy 			   unsigned long arg)
837801c135cSArtem B. Bityutskiy {
838801c135cSArtem B. Bityutskiy 	int err = 0;
839801c135cSArtem B. Bityutskiy 	struct ubi_device *ubi;
840801c135cSArtem B. Bityutskiy 	struct ubi_volume_desc *desc;
841801c135cSArtem B. Bityutskiy 	void __user *argp = (void __user *)arg;
842801c135cSArtem B. Bityutskiy 
843801c135cSArtem B. Bityutskiy 	if (!capable(CAP_SYS_RESOURCE))
844801c135cSArtem B. Bityutskiy 		return -EPERM;
845801c135cSArtem B. Bityutskiy 
846f429b2eaSArtem Bityutskiy 	ubi = ubi_get_by_major(imajor(file->f_mapping->host));
847e73f4459SArtem Bityutskiy 	if (!ubi)
848e73f4459SArtem Bityutskiy 		return -ENODEV;
849801c135cSArtem B. Bityutskiy 
850801c135cSArtem B. Bityutskiy 	switch (cmd) {
851801c135cSArtem B. Bityutskiy 	/* Create volume command */
852801c135cSArtem B. Bityutskiy 	case UBI_IOCMKVOL:
853801c135cSArtem B. Bityutskiy 	{
854801c135cSArtem B. Bityutskiy 		struct ubi_mkvol_req req;
855801c135cSArtem B. Bityutskiy 
856c8566350SArtem Bityutskiy 		dbg_gen("create volume");
857897a316cSArtem Bityutskiy 		err = copy_from_user(&req, argp, sizeof(struct ubi_mkvol_req));
858801c135cSArtem B. Bityutskiy 		if (err) {
859801c135cSArtem B. Bityutskiy 			err = -EFAULT;
860801c135cSArtem B. Bityutskiy 			break;
861801c135cSArtem B. Bityutskiy 		}
862801c135cSArtem B. Bityutskiy 
863801c135cSArtem B. Bityutskiy 		err = verify_mkvol_req(ubi, &req);
864801c135cSArtem B. Bityutskiy 		if (err)
865801c135cSArtem B. Bityutskiy 			break;
866801c135cSArtem B. Bityutskiy 
867f089c0b2SArtem Bityutskiy 		mutex_lock(&ubi->device_mutex);
868801c135cSArtem B. Bityutskiy 		err = ubi_create_volume(ubi, &req);
869f089c0b2SArtem Bityutskiy 		mutex_unlock(&ubi->device_mutex);
870801c135cSArtem B. Bityutskiy 		if (err)
871801c135cSArtem B. Bityutskiy 			break;
872801c135cSArtem B. Bityutskiy 
873bf07803aSChristoph Hellwig 		err = put_user(req.vol_id, (__user int32_t *)argp);
874801c135cSArtem B. Bityutskiy 		if (err)
875801c135cSArtem B. Bityutskiy 			err = -EFAULT;
876801c135cSArtem B. Bityutskiy 
877801c135cSArtem B. Bityutskiy 		break;
878801c135cSArtem B. Bityutskiy 	}
879801c135cSArtem B. Bityutskiy 
880801c135cSArtem B. Bityutskiy 	/* Remove volume command */
881801c135cSArtem B. Bityutskiy 	case UBI_IOCRMVOL:
882801c135cSArtem B. Bityutskiy 	{
883801c135cSArtem B. Bityutskiy 		int vol_id;
884801c135cSArtem B. Bityutskiy 
885c8566350SArtem Bityutskiy 		dbg_gen("remove volume");
886bf07803aSChristoph Hellwig 		err = get_user(vol_id, (__user int32_t *)argp);
887801c135cSArtem B. Bityutskiy 		if (err) {
888801c135cSArtem B. Bityutskiy 			err = -EFAULT;
889801c135cSArtem B. Bityutskiy 			break;
890801c135cSArtem B. Bityutskiy 		}
891801c135cSArtem B. Bityutskiy 
892801c135cSArtem B. Bityutskiy 		desc = ubi_open_volume(ubi->ubi_num, vol_id, UBI_EXCLUSIVE);
893801c135cSArtem B. Bityutskiy 		if (IS_ERR(desc)) {
894801c135cSArtem B. Bityutskiy 			err = PTR_ERR(desc);
895801c135cSArtem B. Bityutskiy 			break;
896801c135cSArtem B. Bityutskiy 		}
897801c135cSArtem B. Bityutskiy 
898f089c0b2SArtem Bityutskiy 		mutex_lock(&ubi->device_mutex);
899f40ac9cdSArtem Bityutskiy 		err = ubi_remove_volume(desc, 0);
900f089c0b2SArtem Bityutskiy 		mutex_unlock(&ubi->device_mutex);
90140e4d0c1SArtem Bityutskiy 
902450f872aSArtem Bityutskiy 		/*
90340e4d0c1SArtem Bityutskiy 		 * The volume is deleted (unless an error occurred), and the
90440e4d0c1SArtem Bityutskiy 		 * 'struct ubi_volume' object will be freed when
90540e4d0c1SArtem Bityutskiy 		 * 'ubi_close_volume()' will call 'put_device()'.
906450f872aSArtem Bityutskiy 		 */
907801c135cSArtem B. Bityutskiy 		ubi_close_volume(desc);
908801c135cSArtem B. Bityutskiy 		break;
909801c135cSArtem B. Bityutskiy 	}
910801c135cSArtem B. Bityutskiy 
911801c135cSArtem B. Bityutskiy 	/* Re-size volume command */
912801c135cSArtem B. Bityutskiy 	case UBI_IOCRSVOL:
913801c135cSArtem B. Bityutskiy 	{
914801c135cSArtem B. Bityutskiy 		int pebs;
915801c135cSArtem B. Bityutskiy 		struct ubi_rsvol_req req;
916801c135cSArtem B. Bityutskiy 
917c8566350SArtem Bityutskiy 		dbg_gen("re-size volume");
918897a316cSArtem Bityutskiy 		err = copy_from_user(&req, argp, sizeof(struct ubi_rsvol_req));
919801c135cSArtem B. Bityutskiy 		if (err) {
920801c135cSArtem B. Bityutskiy 			err = -EFAULT;
921801c135cSArtem B. Bityutskiy 			break;
922801c135cSArtem B. Bityutskiy 		}
923801c135cSArtem B. Bityutskiy 
924801c135cSArtem B. Bityutskiy 		err = verify_rsvol_req(ubi, &req);
925801c135cSArtem B. Bityutskiy 		if (err)
926801c135cSArtem B. Bityutskiy 			break;
927801c135cSArtem B. Bityutskiy 
928801c135cSArtem B. Bityutskiy 		desc = ubi_open_volume(ubi->ubi_num, req.vol_id, UBI_EXCLUSIVE);
929801c135cSArtem B. Bityutskiy 		if (IS_ERR(desc)) {
930801c135cSArtem B. Bityutskiy 			err = PTR_ERR(desc);
931801c135cSArtem B. Bityutskiy 			break;
932801c135cSArtem B. Bityutskiy 		}
933801c135cSArtem B. Bityutskiy 
9343013ee31SArtem Bityutskiy 		pebs = div_u64(req.bytes + desc->vol->usable_leb_size - 1,
9353013ee31SArtem Bityutskiy 			       desc->vol->usable_leb_size);
936801c135cSArtem B. Bityutskiy 
937f089c0b2SArtem Bityutskiy 		mutex_lock(&ubi->device_mutex);
938801c135cSArtem B. Bityutskiy 		err = ubi_resize_volume(desc, pebs);
939f089c0b2SArtem Bityutskiy 		mutex_unlock(&ubi->device_mutex);
940801c135cSArtem B. Bityutskiy 		ubi_close_volume(desc);
941801c135cSArtem B. Bityutskiy 		break;
942801c135cSArtem B. Bityutskiy 	}
943801c135cSArtem B. Bityutskiy 
944f40ac9cdSArtem Bityutskiy 	/* Re-name volumes command */
945f40ac9cdSArtem Bityutskiy 	case UBI_IOCRNVOL:
946f40ac9cdSArtem Bityutskiy 	{
947f40ac9cdSArtem Bityutskiy 		struct ubi_rnvol_req *req;
948f40ac9cdSArtem Bityutskiy 
949f40ac9cdSArtem Bityutskiy 		dbg_msg("re-name volumes");
950f40ac9cdSArtem Bityutskiy 		req = kmalloc(sizeof(struct ubi_rnvol_req), GFP_KERNEL);
951f40ac9cdSArtem Bityutskiy 		if (!req) {
952f40ac9cdSArtem Bityutskiy 			err = -ENOMEM;
953f40ac9cdSArtem Bityutskiy 			break;
954f40ac9cdSArtem Bityutskiy 		};
955f40ac9cdSArtem Bityutskiy 
956f40ac9cdSArtem Bityutskiy 		err = copy_from_user(req, argp, sizeof(struct ubi_rnvol_req));
957f40ac9cdSArtem Bityutskiy 		if (err) {
958f40ac9cdSArtem Bityutskiy 			err = -EFAULT;
959f40ac9cdSArtem Bityutskiy 			kfree(req);
960f40ac9cdSArtem Bityutskiy 			break;
961f40ac9cdSArtem Bityutskiy 		}
962f40ac9cdSArtem Bityutskiy 
963f40ac9cdSArtem Bityutskiy 		err = rename_volumes(ubi, req);
964f40ac9cdSArtem Bityutskiy 		kfree(req);
965f40ac9cdSArtem Bityutskiy 		break;
966f40ac9cdSArtem Bityutskiy 	}
967f40ac9cdSArtem Bityutskiy 
968801c135cSArtem B. Bityutskiy 	default:
969801c135cSArtem B. Bityutskiy 		err = -ENOTTY;
970801c135cSArtem B. Bityutskiy 		break;
971801c135cSArtem B. Bityutskiy 	}
972801c135cSArtem B. Bityutskiy 
973e73f4459SArtem Bityutskiy 	ubi_put_device(ubi);
974801c135cSArtem B. Bityutskiy 	return err;
975801c135cSArtem B. Bityutskiy }
976801c135cSArtem B. Bityutskiy 
977f429b2eaSArtem Bityutskiy static long ctrl_cdev_ioctl(struct file *file, unsigned int cmd,
978f429b2eaSArtem Bityutskiy 			    unsigned long arg)
979897a316cSArtem Bityutskiy {
980897a316cSArtem Bityutskiy 	int err = 0;
981897a316cSArtem Bityutskiy 	void __user *argp = (void __user *)arg;
982897a316cSArtem Bityutskiy 
983897a316cSArtem Bityutskiy 	if (!capable(CAP_SYS_RESOURCE))
984897a316cSArtem Bityutskiy 		return -EPERM;
985897a316cSArtem Bityutskiy 
986897a316cSArtem Bityutskiy 	switch (cmd) {
987897a316cSArtem Bityutskiy 	/* Attach an MTD device command */
988897a316cSArtem Bityutskiy 	case UBI_IOCATT:
989897a316cSArtem Bityutskiy 	{
990897a316cSArtem Bityutskiy 		struct ubi_attach_req req;
991897a316cSArtem Bityutskiy 		struct mtd_info *mtd;
992897a316cSArtem Bityutskiy 
993c8566350SArtem Bityutskiy 		dbg_gen("attach MTD device");
994897a316cSArtem Bityutskiy 		err = copy_from_user(&req, argp, sizeof(struct ubi_attach_req));
995897a316cSArtem Bityutskiy 		if (err) {
996897a316cSArtem Bityutskiy 			err = -EFAULT;
997897a316cSArtem Bityutskiy 			break;
998897a316cSArtem Bityutskiy 		}
999897a316cSArtem Bityutskiy 
1000897a316cSArtem Bityutskiy 		if (req.mtd_num < 0 ||
1001897a316cSArtem Bityutskiy 		    (req.ubi_num < 0 && req.ubi_num != UBI_DEV_NUM_AUTO)) {
1002897a316cSArtem Bityutskiy 			err = -EINVAL;
1003897a316cSArtem Bityutskiy 			break;
1004897a316cSArtem Bityutskiy 		}
1005897a316cSArtem Bityutskiy 
1006897a316cSArtem Bityutskiy 		mtd = get_mtd_device(NULL, req.mtd_num);
1007897a316cSArtem Bityutskiy 		if (IS_ERR(mtd)) {
1008897a316cSArtem Bityutskiy 			err = PTR_ERR(mtd);
1009897a316cSArtem Bityutskiy 			break;
1010897a316cSArtem Bityutskiy 		}
1011897a316cSArtem Bityutskiy 
1012897a316cSArtem Bityutskiy 		/*
1013897a316cSArtem Bityutskiy 		 * Note, further request verification is done by
1014897a316cSArtem Bityutskiy 		 * 'ubi_attach_mtd_dev()'.
1015897a316cSArtem Bityutskiy 		 */
1016897a316cSArtem Bityutskiy 		mutex_lock(&ubi_devices_mutex);
1017897a316cSArtem Bityutskiy 		err = ubi_attach_mtd_dev(mtd, req.ubi_num, req.vid_hdr_offset);
1018897a316cSArtem Bityutskiy 		mutex_unlock(&ubi_devices_mutex);
1019897a316cSArtem Bityutskiy 		if (err < 0)
1020897a316cSArtem Bityutskiy 			put_mtd_device(mtd);
1021897a316cSArtem Bityutskiy 		else
1022897a316cSArtem Bityutskiy 			/* @err contains UBI device number */
1023897a316cSArtem Bityutskiy 			err = put_user(err, (__user int32_t *)argp);
1024897a316cSArtem Bityutskiy 
1025897a316cSArtem Bityutskiy 		break;
1026897a316cSArtem Bityutskiy 	}
1027897a316cSArtem Bityutskiy 
1028897a316cSArtem Bityutskiy 	/* Detach an MTD device command */
1029897a316cSArtem Bityutskiy 	case UBI_IOCDET:
1030897a316cSArtem Bityutskiy 	{
1031897a316cSArtem Bityutskiy 		int ubi_num;
1032897a316cSArtem Bityutskiy 
1033c8566350SArtem Bityutskiy 		dbg_gen("dettach MTD device");
1034897a316cSArtem Bityutskiy 		err = get_user(ubi_num, (__user int32_t *)argp);
1035897a316cSArtem Bityutskiy 		if (err) {
1036897a316cSArtem Bityutskiy 			err = -EFAULT;
1037897a316cSArtem Bityutskiy 			break;
1038897a316cSArtem Bityutskiy 		}
1039897a316cSArtem Bityutskiy 
1040897a316cSArtem Bityutskiy 		mutex_lock(&ubi_devices_mutex);
1041897a316cSArtem Bityutskiy 		err = ubi_detach_mtd_dev(ubi_num, 0);
1042897a316cSArtem Bityutskiy 		mutex_unlock(&ubi_devices_mutex);
1043897a316cSArtem Bityutskiy 		break;
1044897a316cSArtem Bityutskiy 	}
1045897a316cSArtem Bityutskiy 
1046897a316cSArtem Bityutskiy 	default:
1047897a316cSArtem Bityutskiy 		err = -ENOTTY;
1048897a316cSArtem Bityutskiy 		break;
1049897a316cSArtem Bityutskiy 	}
1050897a316cSArtem Bityutskiy 
1051897a316cSArtem Bityutskiy 	return err;
1052897a316cSArtem Bityutskiy }
1053897a316cSArtem Bityutskiy 
1054f429b2eaSArtem Bityutskiy #ifdef CONFIG_COMPAT
1055f429b2eaSArtem Bityutskiy static long vol_cdev_compat_ioctl(struct file *file, unsigned int cmd,
1056f429b2eaSArtem Bityutskiy 				  unsigned long arg)
1057f429b2eaSArtem Bityutskiy {
1058f429b2eaSArtem Bityutskiy 	unsigned long translated_arg = (unsigned long)compat_ptr(arg);
10599f961b57SArtem Bityutskiy 
1060f429b2eaSArtem Bityutskiy 	return vol_cdev_ioctl(file, cmd, translated_arg);
1061f429b2eaSArtem Bityutskiy }
1062f429b2eaSArtem Bityutskiy 
1063f429b2eaSArtem Bityutskiy static long ubi_cdev_compat_ioctl(struct file *file, unsigned int cmd,
1064f429b2eaSArtem Bityutskiy 				  unsigned long arg)
1065f429b2eaSArtem Bityutskiy {
1066f429b2eaSArtem Bityutskiy 	unsigned long translated_arg = (unsigned long)compat_ptr(arg);
1067f429b2eaSArtem Bityutskiy 
1068f429b2eaSArtem Bityutskiy 	return ubi_cdev_ioctl(file, cmd, translated_arg);
1069f429b2eaSArtem Bityutskiy }
1070f429b2eaSArtem Bityutskiy 
1071f429b2eaSArtem Bityutskiy static long ctrl_cdev_compat_ioctl(struct file *file, unsigned int cmd,
1072f429b2eaSArtem Bityutskiy 				   unsigned long arg)
1073f429b2eaSArtem Bityutskiy {
1074f429b2eaSArtem Bityutskiy 	unsigned long translated_arg = (unsigned long)compat_ptr(arg);
1075f429b2eaSArtem Bityutskiy 
1076f429b2eaSArtem Bityutskiy 	return ctrl_cdev_ioctl(file, cmd, translated_arg);
1077f429b2eaSArtem Bityutskiy }
1078f429b2eaSArtem Bityutskiy #else
1079f429b2eaSArtem Bityutskiy #define vol_cdev_compat_ioctl  NULL
1080f429b2eaSArtem Bityutskiy #define ubi_cdev_compat_ioctl  NULL
1081f429b2eaSArtem Bityutskiy #define ctrl_cdev_compat_ioctl NULL
1082f429b2eaSArtem Bityutskiy #endif
1083801c135cSArtem B. Bityutskiy 
1084801c135cSArtem B. Bityutskiy /* UBI volume character device operations */
10854d187a88SJan Engelhardt const struct file_operations ubi_vol_cdev_operations = {
1086801c135cSArtem B. Bityutskiy 	.owner          = THIS_MODULE,
1087801c135cSArtem B. Bityutskiy 	.open           = vol_cdev_open,
1088801c135cSArtem B. Bityutskiy 	.release        = vol_cdev_release,
1089801c135cSArtem B. Bityutskiy 	.llseek         = vol_cdev_llseek,
1090801c135cSArtem B. Bityutskiy 	.read           = vol_cdev_read,
1091801c135cSArtem B. Bityutskiy 	.write          = vol_cdev_write,
10921b24bc3aSCorentin Chary 	.fsync		= vol_cdev_fsync,
1093f429b2eaSArtem Bityutskiy 	.unlocked_ioctl = vol_cdev_ioctl,
1094f429b2eaSArtem Bityutskiy 	.compat_ioctl   = vol_cdev_compat_ioctl,
1095f429b2eaSArtem Bityutskiy };
1096f429b2eaSArtem Bityutskiy 
1097f429b2eaSArtem Bityutskiy /* UBI character device operations */
1098f429b2eaSArtem Bityutskiy const struct file_operations ubi_cdev_operations = {
1099f429b2eaSArtem Bityutskiy 	.owner          = THIS_MODULE,
1100f429b2eaSArtem Bityutskiy 	.llseek         = no_llseek,
1101f429b2eaSArtem Bityutskiy 	.unlocked_ioctl = ubi_cdev_ioctl,
1102f429b2eaSArtem Bityutskiy 	.compat_ioctl   = ubi_cdev_compat_ioctl,
1103f429b2eaSArtem Bityutskiy };
1104f429b2eaSArtem Bityutskiy 
1105f429b2eaSArtem Bityutskiy /* UBI control character device operations */
1106f429b2eaSArtem Bityutskiy const struct file_operations ubi_ctrl_cdev_operations = {
1107f429b2eaSArtem Bityutskiy 	.owner          = THIS_MODULE,
1108f429b2eaSArtem Bityutskiy 	.unlocked_ioctl = ctrl_cdev_ioctl,
1109f429b2eaSArtem Bityutskiy 	.compat_ioctl   = ctrl_cdev_compat_ioctl,
1110e10b376eSArtem Bityutskiy 	.llseek		= no_llseek,
1111801c135cSArtem B. Bityutskiy };
1112