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