diff options
Diffstat (limited to 'drivers')
| -rw-r--r-- | drivers/s390/char/vmur.c | 75 |
1 files changed, 37 insertions, 38 deletions
diff --git a/drivers/s390/char/vmur.c b/drivers/s390/char/vmur.c index 161867cebd8c..1b758b51d7ed 100644 --- a/drivers/s390/char/vmur.c +++ b/drivers/s390/char/vmur.c | |||
| @@ -119,10 +119,12 @@ static void urdev_put(struct urdev *urd) | |||
| 119 | /* | 119 | /* |
| 120 | * Low-level functions to do I/O to a ur device. | 120 | * Low-level functions to do I/O to a ur device. |
| 121 | * alloc_chan_prog | 121 | * alloc_chan_prog |
| 122 | * free_chan_prog | ||
| 122 | * do_ur_io | 123 | * do_ur_io |
| 123 | * ur_int_handler | 124 | * ur_int_handler |
| 124 | * | 125 | * |
| 125 | * alloc_chan_prog allocates and builds the channel program | 126 | * alloc_chan_prog allocates and builds the channel program |
| 127 | * free_chan_prog frees memory of the channel program | ||
| 126 | * | 128 | * |
| 127 | * do_ur_io issues the channel program to the device and blocks waiting | 129 | * do_ur_io issues the channel program to the device and blocks waiting |
| 128 | * on a completion event it publishes at urd->io_done. The function | 130 | * on a completion event it publishes at urd->io_done. The function |
| @@ -137,6 +139,16 @@ static void urdev_put(struct urdev *urd) | |||
| 137 | * address pointer that alloc_chan_prog returned. | 139 | * address pointer that alloc_chan_prog returned. |
| 138 | */ | 140 | */ |
| 139 | 141 | ||
| 142 | static void free_chan_prog(struct ccw1 *cpa) | ||
| 143 | { | ||
| 144 | struct ccw1 *ptr = cpa; | ||
| 145 | |||
| 146 | while (ptr->cda) { | ||
| 147 | kfree((void *)(addr_t) ptr->cda); | ||
| 148 | ptr++; | ||
| 149 | } | ||
| 150 | kfree(cpa); | ||
| 151 | } | ||
| 140 | 152 | ||
| 141 | /* | 153 | /* |
| 142 | * alloc_chan_prog | 154 | * alloc_chan_prog |
| @@ -144,44 +156,45 @@ static void urdev_put(struct urdev *urd) | |||
| 144 | * with a final NOP CCW command-chained on (which ensures that CE and DE | 156 | * with a final NOP CCW command-chained on (which ensures that CE and DE |
| 145 | * are presented together in a single interrupt instead of as separate | 157 | * are presented together in a single interrupt instead of as separate |
| 146 | * interrupts unless an incorrect length indication kicks in first). The | 158 | * interrupts unless an incorrect length indication kicks in first). The |
| 147 | * data length in each CCW is reclen. The caller must ensure that count | 159 | * data length in each CCW is reclen. |
| 148 | * is an integral multiple of reclen. | ||
| 149 | * The channel program pointer returned by this function must be freed | ||
| 150 | * with kfree. The caller is responsible for checking that | ||
| 151 | * count/reclen is not ridiculously large. | ||
| 152 | */ | 160 | */ |
| 153 | static struct ccw1 *alloc_chan_prog(char *buf, size_t count, size_t reclen) | 161 | static struct ccw1 *alloc_chan_prog(const char __user *ubuf, int rec_count, |
| 162 | int reclen) | ||
| 154 | { | 163 | { |
| 155 | size_t num_ccws; | ||
| 156 | struct ccw1 *cpa; | 164 | struct ccw1 *cpa; |
| 165 | void *kbuf; | ||
| 157 | int i; | 166 | int i; |
| 158 | 167 | ||
| 159 | TRACE("alloc_chan_prog(%p, %zu, %zu)\n", buf, count, reclen); | 168 | TRACE("alloc_chan_prog(%p, %i, %i)\n", ubuf, rec_count, reclen); |
| 160 | 169 | ||
| 161 | /* | 170 | /* |
| 162 | * We chain a NOP onto the writes to force CE+DE together. | 171 | * We chain a NOP onto the writes to force CE+DE together. |
| 163 | * That means we allocate room for CCWs to cover count/reclen | 172 | * That means we allocate room for CCWs to cover count/reclen |
| 164 | * records plus a NOP. | 173 | * records plus a NOP. |
| 165 | */ | 174 | */ |
| 166 | num_ccws = count / reclen + 1; | 175 | cpa = kzalloc((rec_count + 1) * sizeof(struct ccw1), |
| 167 | cpa = kmalloc(num_ccws * sizeof(struct ccw1), GFP_KERNEL | GFP_DMA); | 176 | GFP_KERNEL | GFP_DMA); |
| 168 | if (!cpa) | 177 | if (!cpa) |
| 169 | return NULL; | 178 | return ERR_PTR(-ENOMEM); |
| 170 | 179 | ||
| 171 | for (i = 0; count; i++) { | 180 | for (i = 0; i < rec_count; i++) { |
| 172 | cpa[i].cmd_code = WRITE_CCW_CMD; | 181 | cpa[i].cmd_code = WRITE_CCW_CMD; |
| 173 | cpa[i].flags = CCW_FLAG_CC | CCW_FLAG_SLI; | 182 | cpa[i].flags = CCW_FLAG_CC | CCW_FLAG_SLI; |
| 174 | cpa[i].count = reclen; | 183 | cpa[i].count = reclen; |
| 175 | cpa[i].cda = __pa(buf); | 184 | kbuf = kmalloc(reclen, GFP_KERNEL | GFP_DMA); |
| 176 | buf += reclen; | 185 | if (!kbuf) { |
| 177 | count -= reclen; | 186 | free_chan_prog(cpa); |
| 187 | return ERR_PTR(-ENOMEM); | ||
| 188 | } | ||
| 189 | cpa[i].cda = (u32)(addr_t) kbuf; | ||
| 190 | if (copy_from_user(kbuf, ubuf, reclen)) { | ||
| 191 | free_chan_prog(cpa); | ||
| 192 | return ERR_PTR(-EFAULT); | ||
| 193 | } | ||
| 194 | ubuf += reclen; | ||
| 178 | } | 195 | } |
| 179 | /* The following NOP CCW forces CE+DE to be presented together */ | 196 | /* The following NOP CCW forces CE+DE to be presented together */ |
| 180 | cpa[i].cmd_code = CCW_CMD_NOOP; | 197 | cpa[i].cmd_code = CCW_CMD_NOOP; |
| 181 | cpa[i].flags = 0; | ||
| 182 | cpa[i].count = 0; | ||
| 183 | cpa[i].cda = 0; | ||
| 184 | |||
| 185 | return cpa; | 198 | return cpa; |
| 186 | } | 199 | } |
| 187 | 200 | ||
| @@ -325,24 +338,11 @@ static ssize_t do_write(struct urdev *urd, const char __user *udata, | |||
| 325 | size_t count, size_t reclen, loff_t *ppos) | 338 | size_t count, size_t reclen, loff_t *ppos) |
| 326 | { | 339 | { |
| 327 | struct ccw1 *cpa; | 340 | struct ccw1 *cpa; |
| 328 | char *buf; | ||
| 329 | int rc; | 341 | int rc; |
| 330 | 342 | ||
| 331 | /* Data buffer must be under 2GB line for fmt1 CCWs: hence GFP_DMA */ | 343 | cpa = alloc_chan_prog(udata, count / reclen, reclen); |
| 332 | buf = kmalloc(count, GFP_KERNEL | GFP_DMA); | 344 | if (IS_ERR(cpa)) |
| 333 | if (!buf) | 345 | return PTR_ERR(cpa); |
| 334 | return -ENOMEM; | ||
| 335 | |||
| 336 | if (copy_from_user(buf, udata, count)) { | ||
| 337 | rc = -EFAULT; | ||
| 338 | goto fail_kfree_buf; | ||
| 339 | } | ||
| 340 | |||
| 341 | cpa = alloc_chan_prog(buf, count, reclen); | ||
| 342 | if (!cpa) { | ||
| 343 | rc = -ENOMEM; | ||
| 344 | goto fail_kfree_buf; | ||
| 345 | } | ||
| 346 | 346 | ||
| 347 | rc = do_ur_io(urd, cpa); | 347 | rc = do_ur_io(urd, cpa); |
| 348 | if (rc) | 348 | if (rc) |
| @@ -354,10 +354,9 @@ static ssize_t do_write(struct urdev *urd, const char __user *udata, | |||
| 354 | } | 354 | } |
| 355 | *ppos += count; | 355 | *ppos += count; |
| 356 | rc = count; | 356 | rc = count; |
| 357 | |||
| 357 | fail_kfree_cpa: | 358 | fail_kfree_cpa: |
| 358 | kfree(cpa); | 359 | free_chan_prog(cpa); |
| 359 | fail_kfree_buf: | ||
| 360 | kfree(buf); | ||
| 361 | return rc; | 360 | return rc; |
| 362 | } | 361 | } |
| 363 | 362 | ||
