2 * Copyright (c) 2011 Venkatesh Srinivas <vsrinivas@dragonflybsd.org>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
17 * This file contains support for the POSIX 1003.1B AIO/LIO facility.
19 * This version of AIO is aimed at standards compliance; it is not aimed
20 * at either reasonability or performance. It merely wraps synchronous I/O
23 * This version cannot perform asynchronous notification of I/O completion
24 * on DragonFly via SIGEV_SIGNAL.
26 * 2) SIGEV_SIGNAL code is present, but if-ed out as 'not yet'; Dfly
27 * does not support sigqueue(), so we cannot control the payload to
28 * a SIGINFO-signal from userspace. Programs using AIO signals use
29 * the payload field to carry pointers to the completed AIO structure,
30 * which is not yet possible here.
32 * It would be possible for this version to support SIGEV_KEVENT.
36 #include <sys/types.h>
38 #include <sys/signal.h>
39 #include <sys/queue.h>
46 static void *_aio_thr_start(void *);
49 _aio_notify(struct sigevent
*sigevp
)
58 switch (sigevp
->sigev_notify
) {
64 sigevp
->sigev_notify_attributes
,
72 sig
= sigevp
->sigev_signo
;
73 sv
= sigevp
->sigev_value
;
74 sigqueue(pid
, sig
, sv
);
85 _aio_thr_start(void *vsigevp
)
87 struct sigevent
*sigevp
= vsigevp
;
88 sigevp
->sigev_notify_function(sigevp
->sigev_value
);
95 * Asynchronously read from a file
98 aio_read(struct aiocb
*ap
)
101 if ((ap
->aio_sigevent
.sigev_notify
!= SIGEV_NONE
) &&
102 (ap
->aio_sigevent
.sigev_notify
!= SIGEV_THREAD
))
106 ap
->_aio_val
= pread(ap
->aio_fildes
,
107 (void *) ap
->aio_buf
,
110 ap
->_aio_err
= errno
;
112 _aio_notify(&ap
->aio_sigevent
);
118 aio_write(struct aiocb
*ap
)
121 if ((ap
->aio_sigevent
.sigev_notify
!= SIGEV_NONE
) &&
122 (ap
->aio_sigevent
.sigev_notify
!= SIGEV_THREAD
))
126 ap
->_aio_val
= pwrite(ap
->aio_fildes
,
127 (void *) ap
->aio_buf
,
130 ap
->_aio_err
= errno
;
132 _aio_notify(&ap
->aio_sigevent
);
138 aio_fsync(int op
, struct aiocb
*ap
)
141 if ((ap
->aio_sigevent
.sigev_notify
!= SIGEV_NONE
) &&
142 (ap
->aio_sigevent
.sigev_notify
!= SIGEV_THREAD
))
146 ap
->_aio_val
= fsync(ap
->aio_fildes
);
147 ap
->_aio_err
= errno
;
149 _aio_notify(&ap
->aio_sigevent
);
155 lio_listio(int mode
, struct aiocb
*const apv
[], int nent
,
156 struct sigevent
*sigevp
)
162 (sigevp
->sigev_notify
!= SIGEV_NONE
) &&
163 (sigevp
->sigev_notify
!= SIGEV_THREAD
))
167 for (i
= 0; i
< nent
; i
++)
168 switch (apv
[i
]->aio_lio_opcode
) {
191 * Get I/O completion status
193 * Returns EINPROGRESS until I/O is complete. Returns ECANCELED if
194 * I/O is canceled. Returns I/O status if operation completed.
196 * This routine does not block.
199 aio_error(const struct aiocb
*ap
)
201 return (ap
->_aio_err
);
207 * Finish up I/O, releasing I/O resources and returns the value
208 * that would have been associated with a synchronous request.
211 aio_return(struct aiocb
*ap
)
213 return (ap
->_aio_val
);
217 aio_cancel(int fildes
, struct aiocb
*aiocbp
)
221 /* must be a valid file descriptor */
222 if (fstat(fildes
, &sb
)) {
226 return (AIO_ALLDONE
);
230 aio_suspend(const struct aiocb
*const list
[], int nent
, const struct timespec
*timo
)