1 /* libbdev - block device interfacing library, by D.C. van Moolenbroek */
3 #include <minix/drivers.h>
4 #include <minix/bdev.h>
5 #include <minix/ioctl.h>
12 void bdev_driver(dev_t dev
, char *label
)
14 /* Associate a driver with the given (major) device, using its endpoint.
15 * File system usage note: typically called from mount and newdriver.
17 static int first
= TRUE
;
20 /* Initialize the driver endpoint array. */
26 bdev_update(dev
, label
);
29 static int bdev_retry(int *driver_tries
, int *transfer_tries
, int *result
)
31 /* Return TRUE iff the call result implies that we should retry the operation.
36 /* We get this error internally if the driver has restarted and the
37 * current operation may now go through. Check the retry count for
38 * driver restarts first, as we don't want to keep trying forever.
40 if (++*driver_tries
< DRIVER_TRIES
)
43 *result
= EDEADSRCDST
;
48 /* The 'transfer_tries' pointer is non-NULL if this was a transfer
49 * request. If we get back an I/O failure, keep retrying the request
50 * until we hit the transfer retry limit.
52 if (transfer_tries
!= NULL
&& ++*transfer_tries
< TRANSFER_TRIES
)
61 static int bdev_opcl(int req
, dev_t dev
, int access
)
63 /* Open or close the given minor device.
66 int r
, driver_tries
= 0;
69 memset(&m
, 0, sizeof(m
));
71 m
.m_lbdev_lblockdriver_msg
.minor
= minor(dev
);
72 m
.m_lbdev_lblockdriver_msg
.access
= access
;
74 r
= bdev_sendrec(dev
, &m
);
75 } while (bdev_retry(&driver_tries
, NULL
, &r
));
80 int bdev_open(dev_t dev
, int access
)
82 /* Open the given minor device.
83 * File system usage note: typically called from mount, after bdev_driver.
87 r
= bdev_opcl(BDEV_OPEN
, dev
, access
);
90 bdev_minor_add(dev
, access
);
95 int bdev_close(dev_t dev
)
97 /* Close the given minor device.
98 * File system usage note: typically called from unmount.
102 bdev_flush_asyn(dev
);
104 r
= bdev_opcl(BDEV_CLOSE
, dev
, 0);
112 static int bdev_rdwt_setup(int req
, dev_t dev
, u64_t pos
, char *buf
,
113 size_t count
, int flags
, message
*m
)
115 /* Set up a single-buffer read/write request.
121 assert((ssize_t
) count
>= 0);
123 if ((endpt
= bdev_driver_get(dev
)) == NONE
)
126 access
= (req
== BDEV_READ
) ? CPF_WRITE
: CPF_READ
;
128 grant
= cpf_grant_direct(endpt
, (vir_bytes
) buf
, count
, access
);
130 if (!GRANT_VALID(grant
)) {
131 printf("bdev: unable to allocate grant!\n");
135 memset(m
, 0, sizeof(*m
));
137 m
->m_lbdev_lblockdriver_msg
.minor
= minor(dev
);
138 m
->m_lbdev_lblockdriver_msg
.pos
= pos
;
139 m
->m_lbdev_lblockdriver_msg
.count
= count
;
140 m
->m_lbdev_lblockdriver_msg
.grant
= grant
;
141 m
->m_lbdev_lblockdriver_msg
.flags
= flags
;
146 static void bdev_rdwt_cleanup(const message
*m
)
148 /* Clean up a single-buffer read/write request.
151 cpf_revoke(m
->m_lbdev_lblockdriver_msg
.grant
);
154 static ssize_t
bdev_rdwt(int req
, dev_t dev
, u64_t pos
, char *buf
,
155 size_t count
, int flags
)
157 /* Perform a synchronous read or write call using a single buffer.
160 int r
, driver_tries
= 0, transfer_tries
= 0;
163 if ((r
= bdev_rdwt_setup(req
, dev
, pos
, buf
, count
, flags
, &m
)) != OK
)
166 r
= bdev_sendrec(dev
, &m
);
168 bdev_rdwt_cleanup(&m
);
169 } while (bdev_retry(&driver_tries
, &transfer_tries
, &r
));
174 static int bdev_vrdwt_setup(int req
, dev_t dev
, u64_t pos
, iovec_t
*vec
,
175 int count
, int flags
, message
*m
, iovec_s_t
*gvec
)
177 /* Set up a vectored read/write request.
184 assert(count
<= NR_IOREQS
);
186 if ((endpt
= bdev_driver_get(dev
)) == NONE
)
189 access
= (req
== BDEV_GATHER
) ? CPF_WRITE
: CPF_READ
;
192 for (i
= 0; i
< count
; i
++) {
193 grant
= cpf_grant_direct(endpt
, vec
[i
].iov_addr
, vec
[i
].iov_size
,
196 if (!GRANT_VALID(grant
)) {
197 printf("bdev: unable to allocate grant!\n");
199 for (i
--; i
>= 0; i
--)
200 cpf_revoke(gvec
[i
].iov_grant
);
205 gvec
[i
].iov_grant
= grant
;
206 gvec
[i
].iov_size
= vec
[i
].iov_size
;
208 assert(vec
[i
].iov_size
> 0);
209 assert((ssize_t
) (size
+ vec
[i
].iov_size
) > size
);
211 size
+= vec
[i
].iov_size
;
214 grant
= cpf_grant_direct(endpt
, (vir_bytes
) gvec
, sizeof(gvec
[0]) * count
,
217 if (!GRANT_VALID(grant
)) {
218 printf("bdev: unable to allocate grant!\n");
220 for (i
= count
- 1; i
>= 0; i
--)
221 cpf_revoke(gvec
[i
].iov_grant
);
226 memset(m
, 0, sizeof(*m
));
228 m
->m_lbdev_lblockdriver_msg
.minor
= minor(dev
);
229 m
->m_lbdev_lblockdriver_msg
.pos
= pos
;
230 m
->m_lbdev_lblockdriver_msg
.count
= count
;
231 m
->m_lbdev_lblockdriver_msg
.grant
= grant
;
232 m
->m_lbdev_lblockdriver_msg
.flags
= flags
;
237 static void bdev_vrdwt_cleanup(const message
*m
, iovec_s_t
*gvec
)
239 /* Clean up a vectored read/write request.
244 grant
= m
->m_lbdev_lblockdriver_msg
.grant
;
248 for (i
= m
->m_lbdev_lblockdriver_msg
.count
- 1; i
>= 0; i
--)
249 cpf_revoke(gvec
[i
].iov_grant
);
252 static ssize_t
bdev_vrdwt(int req
, dev_t dev
, u64_t pos
, iovec_t
*vec
,
253 int count
, int flags
)
255 /* Perform a synchronous read or write call using a vector of buffers.
257 iovec_s_t gvec
[NR_IOREQS
];
259 int r
, driver_tries
= 0, transfer_tries
= 0;
262 if ((r
= bdev_vrdwt_setup(req
, dev
, pos
, vec
, count
, flags
, &m
,
266 r
= bdev_sendrec(dev
, &m
);
268 bdev_vrdwt_cleanup(&m
, gvec
);
269 } while (bdev_retry(&driver_tries
, &transfer_tries
, &r
));
274 ssize_t
bdev_read(dev_t dev
, u64_t pos
, char *buf
, size_t count
, int flags
)
276 /* Perform a synchronous read call into a single buffer.
279 return bdev_rdwt(BDEV_READ
, dev
, pos
, buf
, count
, flags
);
282 ssize_t
bdev_write(dev_t dev
, u64_t pos
, char *buf
, size_t count
, int flags
)
284 /* Perform a synchronous write call from a single buffer.
287 return bdev_rdwt(BDEV_WRITE
, dev
, pos
, buf
, count
, flags
);
290 ssize_t
bdev_gather(dev_t dev
, u64_t pos
, iovec_t
*vec
, int count
, int flags
)
292 /* Perform a synchronous read call into a vector of buffers.
295 return bdev_vrdwt(BDEV_GATHER
, dev
, pos
, vec
, count
, flags
);
298 ssize_t
bdev_scatter(dev_t dev
, u64_t pos
, iovec_t
*vec
, int count
, int flags
)
300 /* Perform a synchronous write call from a vector of buffers.
303 return bdev_vrdwt(BDEV_SCATTER
, dev
, pos
, vec
, count
, flags
);
306 static int bdev_ioctl_setup(dev_t dev
, int request
, void *buf
,
307 endpoint_t user_endpt
, message
*m
)
309 /* Set up an I/O control request.
316 if ((endpt
= bdev_driver_get(dev
)) == NONE
)
319 if (_MINIX_IOCTL_BIG(request
))
320 size
= _MINIX_IOCTL_SIZE_BIG(request
);
322 size
= _MINIX_IOCTL_SIZE(request
);
325 if (_MINIX_IOCTL_IOR(request
)) access
|= CPF_WRITE
;
326 if (_MINIX_IOCTL_IOW(request
)) access
|= CPF_READ
;
328 /* The size may be 0, in which case 'buf' need not be a valid pointer. */
329 grant
= cpf_grant_direct(endpt
, (vir_bytes
) buf
, size
, access
);
331 if (!GRANT_VALID(grant
)) {
332 printf("bdev: unable to allocate grant!\n");
336 memset(m
, 0, sizeof(*m
));
337 m
->m_type
= BDEV_IOCTL
;
338 m
->m_lbdev_lblockdriver_msg
.minor
= minor(dev
);
339 m
->m_lbdev_lblockdriver_msg
.request
= request
;
340 m
->m_lbdev_lblockdriver_msg
.grant
= grant
;
341 m
->m_lbdev_lblockdriver_msg
.user
= user_endpt
;
346 static void bdev_ioctl_cleanup(const message
*m
)
348 /* Clean up an I/O control request.
351 cpf_revoke(m
->m_lbdev_lblockdriver_msg
.grant
);
354 int bdev_ioctl(dev_t dev
, int request
, void *buf
, endpoint_t user_endpt
)
356 /* Perform a synchronous I/O control request.
359 int r
, driver_tries
= 0;
362 if ((r
= bdev_ioctl_setup(dev
, request
, buf
, user_endpt
, &m
)) != OK
)
365 r
= bdev_sendrec(dev
, &m
);
367 bdev_ioctl_cleanup(&m
);
368 } while (bdev_retry(&driver_tries
, NULL
, &r
));
373 void bdev_flush_asyn(dev_t dev
)
375 /* Flush all ongoing asynchronous requests to the given minor device. This
376 * involves blocking until all I/O for it has completed.
377 * File system usage note: typically called from flush.
381 while ((call
= bdev_call_find(dev
)) != NULL
)
382 (void) bdev_wait_asyn(call
->id
);
385 static bdev_id_t
bdev_rdwt_asyn(int req
, dev_t dev
, u64_t pos
, char *buf
,
386 size_t count
, int flags
, bdev_callback_t callback
, bdev_param_t param
)
388 /* Perform an asynchronous read or write call using a single buffer.
393 if ((call
= bdev_call_alloc(1)) == NULL
)
396 if ((r
= bdev_rdwt_setup(req
, dev
, pos
, buf
, count
, flags
, &call
->msg
)) !=
398 bdev_call_free(call
);
403 if ((r
= bdev_senda(dev
, &call
->msg
, call
->id
)) != OK
) {
404 bdev_rdwt_cleanup(&call
->msg
);
406 bdev_call_free(call
);
412 call
->callback
= callback
;
414 call
->driver_tries
= 0;
415 call
->transfer_tries
= 0;
416 call
->vec
[0].iov_addr
= (vir_bytes
) buf
;
417 call
->vec
[0].iov_size
= count
;
422 static bdev_id_t
bdev_vrdwt_asyn(int req
, dev_t dev
, u64_t pos
, iovec_t
*vec
,
423 int count
, int flags
, bdev_callback_t callback
, bdev_param_t param
)
425 /* Perform an asynchronous read or write call using a vector of buffers.
430 if ((call
= bdev_call_alloc(count
)) == NULL
)
433 if ((r
= bdev_vrdwt_setup(req
, dev
, pos
, vec
, count
, flags
, &call
->msg
,
434 call
->gvec
)) != OK
) {
435 bdev_call_free(call
);
440 if ((r
= bdev_senda(dev
, &call
->msg
, call
->id
)) != OK
) {
441 bdev_vrdwt_cleanup(&call
->msg
, call
->gvec
);
443 bdev_call_free(call
);
449 call
->callback
= callback
;
451 call
->driver_tries
= 0;
452 call
->transfer_tries
= 0;
453 memcpy(call
->vec
, vec
, sizeof(vec
[0]) * count
);
458 bdev_id_t
bdev_read_asyn(dev_t dev
, u64_t pos
, char *buf
, size_t count
,
459 int flags
, bdev_callback_t callback
, bdev_param_t param
)
461 /* Perform an asynchronous read call into a single buffer.
464 return bdev_rdwt_asyn(BDEV_READ
, dev
, pos
, buf
, count
, flags
, callback
,
468 bdev_id_t
bdev_write_asyn(dev_t dev
, u64_t pos
, char *buf
, size_t count
,
469 int flags
, bdev_callback_t callback
, bdev_param_t param
)
471 /* Perform an asynchronous write call from a single buffer.
474 return bdev_rdwt_asyn(BDEV_WRITE
, dev
, pos
, buf
, count
, flags
, callback
,
478 bdev_id_t
bdev_gather_asyn(dev_t dev
, u64_t pos
, iovec_t
*vec
, int count
,
479 int flags
, bdev_callback_t callback
, bdev_param_t param
)
481 /* Perform an asynchronous read call into a vector of buffers.
484 return bdev_vrdwt_asyn(BDEV_GATHER
, dev
, pos
, vec
, count
, flags
, callback
,
488 bdev_id_t
bdev_scatter_asyn(dev_t dev
, u64_t pos
, iovec_t
*vec
, int count
,
489 int flags
, bdev_callback_t callback
, bdev_param_t param
)
491 /* Perform an asynchronous write call into a vector of buffers.
494 return bdev_vrdwt_asyn(BDEV_SCATTER
, dev
, pos
, vec
, count
, flags
, callback
,
498 bdev_id_t
bdev_ioctl_asyn(dev_t dev
, int request
, void *buf
,
499 endpoint_t user_endpt
, bdev_callback_t callback
, bdev_param_t param
)
501 /* Perform an asynchronous I/O control request.
506 if ((call
= bdev_call_alloc(1)) == NULL
)
509 if ((r
= bdev_ioctl_setup(dev
, request
, buf
, user_endpt
,
510 &call
->msg
)) != OK
) {
511 bdev_call_free(call
);
516 if ((r
= bdev_senda(dev
, &call
->msg
, call
->id
)) != OK
) {
517 bdev_ioctl_cleanup(&call
->msg
);
519 bdev_call_free(call
);
525 call
->callback
= callback
;
527 call
->driver_tries
= 0;
528 call
->vec
[0].iov_addr
= (vir_bytes
) buf
;
533 void bdev_callback_asyn(bdev_call_t
*call
, int result
)
535 /* Perform the callback for an asynchronous request, with the given result.
536 * Clean up the call structure afterwards.
539 /* If this was a transfer request and the result is EIO, we may want to retry
542 switch (call
->msg
.m_type
) {
547 if (result
== EIO
&& ++call
->transfer_tries
< TRANSFER_TRIES
) {
548 result
= bdev_senda(call
->dev
, &call
->msg
, call
->id
);
556 switch (call
->msg
.m_type
) {
559 bdev_rdwt_cleanup(&call
->msg
);
565 bdev_vrdwt_cleanup(&call
->msg
, call
->gvec
);
570 bdev_ioctl_cleanup(&call
->msg
);
578 /* Call the callback function. */
579 /* FIXME: we assume all reasonable ssize_t values can be stored in an int. */
580 call
->callback(call
->dev
, call
->id
, call
->param
, result
);
582 /* Free up the call structure. */
583 bdev_call_free(call
);
586 int bdev_restart_asyn(bdev_call_t
*call
)
588 /* The driver for the given call has restarted, and may now have a new
589 * endpoint. Recreate and resend the request for the given call.
593 /* Update and check the retry limit for driver restarts first. */
594 if (++call
->driver_tries
>= DRIVER_TRIES
)
597 /* Recreate all grants for the new endpoint. */
598 type
= call
->msg
.m_type
;
603 bdev_rdwt_cleanup(&call
->msg
);
605 r
= bdev_rdwt_setup(type
, call
->dev
,
606 call
->msg
.m_lbdev_lblockdriver_msg
.pos
,
607 (char *) call
->vec
[0].iov_addr
, call
->msg
.m_lbdev_lblockdriver_msg
.count
,
608 call
->msg
.m_lbdev_lblockdriver_msg
.flags
, &call
->msg
);
614 bdev_vrdwt_cleanup(&call
->msg
, call
->gvec
);
616 r
= bdev_vrdwt_setup(type
, call
->dev
,
617 call
->msg
.m_lbdev_lblockdriver_msg
.pos
,
618 call
->vec
, call
->msg
.m_lbdev_lblockdriver_msg
.count
, call
->msg
.m_lbdev_lblockdriver_msg
.flags
,
619 &call
->msg
, call
->gvec
);
624 bdev_ioctl_cleanup(&call
->msg
);
626 r
= bdev_ioctl_setup(call
->dev
, call
->msg
.m_lbdev_lblockdriver_msg
.request
,
627 (char *) call
->vec
[0].iov_addr
, call
->msg
.m_lbdev_lblockdriver_msg
.user
,
639 /* Try to resend the request. */
640 return bdev_senda(call
->dev
, &call
->msg
, call
->id
);