Merge remote-tracking branch 'remotes/kraxel/tags/seabios-1.12-20181120-pull-request...
[qemu/ar7.git] / include / io / channel.h
blobda2f1382002a59bcb429de5ee988388bb62f828e
1 /*
2 * QEMU I/O channels
4 * Copyright (c) 2015 Red Hat, Inc.
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
21 #ifndef QIO_CHANNEL_H
22 #define QIO_CHANNEL_H
24 #include "qemu-common.h"
25 #include "qom/object.h"
26 #include "qemu/coroutine.h"
27 #include "block/aio.h"
29 #define TYPE_QIO_CHANNEL "qio-channel"
30 #define QIO_CHANNEL(obj) \
31 OBJECT_CHECK(QIOChannel, (obj), TYPE_QIO_CHANNEL)
32 #define QIO_CHANNEL_CLASS(klass) \
33 OBJECT_CLASS_CHECK(QIOChannelClass, klass, TYPE_QIO_CHANNEL)
34 #define QIO_CHANNEL_GET_CLASS(obj) \
35 OBJECT_GET_CLASS(QIOChannelClass, obj, TYPE_QIO_CHANNEL)
37 typedef struct QIOChannel QIOChannel;
38 typedef struct QIOChannelClass QIOChannelClass;
40 #define QIO_CHANNEL_ERR_BLOCK -2
42 typedef enum QIOChannelFeature QIOChannelFeature;
44 enum QIOChannelFeature {
45 QIO_CHANNEL_FEATURE_FD_PASS,
46 QIO_CHANNEL_FEATURE_SHUTDOWN,
47 QIO_CHANNEL_FEATURE_LISTEN,
51 typedef enum QIOChannelShutdown QIOChannelShutdown;
53 enum QIOChannelShutdown {
54 QIO_CHANNEL_SHUTDOWN_READ = 1,
55 QIO_CHANNEL_SHUTDOWN_WRITE = 2,
56 QIO_CHANNEL_SHUTDOWN_BOTH = 3,
59 typedef gboolean (*QIOChannelFunc)(QIOChannel *ioc,
60 GIOCondition condition,
61 gpointer data);
63 /**
64 * QIOChannel:
66 * The QIOChannel defines the core API for a generic I/O channel
67 * class hierarchy. It is inspired by GIOChannel, but has the
68 * following differences
70 * - Use QOM to properly support arbitrary subclassing
71 * - Support use of iovecs for efficient I/O with multiple blocks
72 * - None of the character set translation, binary data exclusively
73 * - Direct support for QEMU Error object reporting
74 * - File descriptor passing
76 * This base class is abstract so cannot be instantiated. There
77 * will be subclasses for dealing with sockets, files, and higher
78 * level protocols such as TLS, WebSocket, etc.
81 struct QIOChannel {
82 Object parent;
83 unsigned int features; /* bitmask of QIOChannelFeatures */
84 char *name;
85 AioContext *ctx;
86 Coroutine *read_coroutine;
87 Coroutine *write_coroutine;
88 #ifdef _WIN32
89 HANDLE event; /* For use with GSource on Win32 */
90 #endif
93 /**
94 * QIOChannelClass:
96 * This class defines the contract that all subclasses
97 * must follow to provide specific channel implementations.
98 * The first five callbacks are mandatory to support, others
99 * provide additional optional features.
101 * Consult the corresponding public API docs for a description
102 * of the semantics of each callback
104 struct QIOChannelClass {
105 ObjectClass parent;
107 /* Mandatory callbacks */
108 ssize_t (*io_writev)(QIOChannel *ioc,
109 const struct iovec *iov,
110 size_t niov,
111 int *fds,
112 size_t nfds,
113 Error **errp);
114 ssize_t (*io_readv)(QIOChannel *ioc,
115 const struct iovec *iov,
116 size_t niov,
117 int **fds,
118 size_t *nfds,
119 Error **errp);
120 int (*io_close)(QIOChannel *ioc,
121 Error **errp);
122 GSource * (*io_create_watch)(QIOChannel *ioc,
123 GIOCondition condition);
124 int (*io_set_blocking)(QIOChannel *ioc,
125 bool enabled,
126 Error **errp);
128 /* Optional callbacks */
129 int (*io_shutdown)(QIOChannel *ioc,
130 QIOChannelShutdown how,
131 Error **errp);
132 void (*io_set_cork)(QIOChannel *ioc,
133 bool enabled);
134 void (*io_set_delay)(QIOChannel *ioc,
135 bool enabled);
136 off_t (*io_seek)(QIOChannel *ioc,
137 off_t offset,
138 int whence,
139 Error **errp);
140 void (*io_set_aio_fd_handler)(QIOChannel *ioc,
141 AioContext *ctx,
142 IOHandler *io_read,
143 IOHandler *io_write,
144 void *opaque);
147 /* General I/O handling functions */
150 * qio_channel_has_feature:
151 * @ioc: the channel object
152 * @feature: the feature to check support of
154 * Determine whether the channel implementation supports
155 * the optional feature named in @feature.
157 * Returns: true if supported, false otherwise.
159 bool qio_channel_has_feature(QIOChannel *ioc,
160 QIOChannelFeature feature);
163 * qio_channel_set_feature:
164 * @ioc: the channel object
165 * @feature: the feature to set support for
167 * Add channel support for the feature named in @feature.
169 void qio_channel_set_feature(QIOChannel *ioc,
170 QIOChannelFeature feature);
173 * qio_channel_set_name:
174 * @ioc: the channel object
175 * @name: the name of the channel
177 * Sets the name of the channel, which serves as an aid
178 * to debugging. The name is used when creating GSource
179 * watches for this channel.
181 void qio_channel_set_name(QIOChannel *ioc,
182 const char *name);
185 * qio_channel_readv_full:
186 * @ioc: the channel object
187 * @iov: the array of memory regions to read data into
188 * @niov: the length of the @iov array
189 * @fds: pointer to an array that will received file handles
190 * @nfds: pointer filled with number of elements in @fds on return
191 * @errp: pointer to a NULL-initialized error object
193 * Read data from the IO channel, storing it in the
194 * memory regions referenced by @iov. Each element
195 * in the @iov will be fully populated with data
196 * before the next one is used. The @niov parameter
197 * specifies the total number of elements in @iov.
199 * It is not required for all @iov to be filled with
200 * data. If the channel is in blocking mode, at least
201 * one byte of data will be read, but no more is
202 * guaranteed. If the channel is non-blocking and no
203 * data is available, it will return QIO_CHANNEL_ERR_BLOCK
205 * If the channel has passed any file descriptors,
206 * the @fds array pointer will be allocated and
207 * the elements filled with the received file
208 * descriptors. The @nfds pointer will be updated
209 * to indicate the size of the @fds array that
210 * was allocated. It is the callers responsibility
211 * to call close() on each file descriptor and to
212 * call g_free() on the array pointer in @fds.
214 * It is an error to pass a non-NULL @fds parameter
215 * unless qio_channel_has_feature() returns a true
216 * value for the QIO_CHANNEL_FEATURE_FD_PASS constant.
218 * Returns: the number of bytes read, or -1 on error,
219 * or QIO_CHANNEL_ERR_BLOCK if no data is available
220 * and the channel is non-blocking
222 ssize_t qio_channel_readv_full(QIOChannel *ioc,
223 const struct iovec *iov,
224 size_t niov,
225 int **fds,
226 size_t *nfds,
227 Error **errp);
231 * qio_channel_writev_full:
232 * @ioc: the channel object
233 * @iov: the array of memory regions to write data from
234 * @niov: the length of the @iov array
235 * @fds: an array of file handles to send
236 * @nfds: number of file handles in @fds
237 * @errp: pointer to a NULL-initialized error object
239 * Write data to the IO channel, reading it from the
240 * memory regions referenced by @iov. Each element
241 * in the @iov will be fully sent, before the next
242 * one is used. The @niov parameter specifies the
243 * total number of elements in @iov.
245 * It is not required for all @iov data to be fully
246 * sent. If the channel is in blocking mode, at least
247 * one byte of data will be sent, but no more is
248 * guaranteed. If the channel is non-blocking and no
249 * data can be sent, it will return QIO_CHANNEL_ERR_BLOCK
251 * If there are file descriptors to send, the @fds
252 * array should be non-NULL and provide the handles.
253 * All file descriptors will be sent if at least one
254 * byte of data was sent.
256 * It is an error to pass a non-NULL @fds parameter
257 * unless qio_channel_has_feature() returns a true
258 * value for the QIO_CHANNEL_FEATURE_FD_PASS constant.
260 * Returns: the number of bytes sent, or -1 on error,
261 * or QIO_CHANNEL_ERR_BLOCK if no data is can be sent
262 * and the channel is non-blocking
264 ssize_t qio_channel_writev_full(QIOChannel *ioc,
265 const struct iovec *iov,
266 size_t niov,
267 int *fds,
268 size_t nfds,
269 Error **errp);
272 * qio_channel_readv_all_eof:
273 * @ioc: the channel object
274 * @iov: the array of memory regions to read data into
275 * @niov: the length of the @iov array
276 * @errp: pointer to a NULL-initialized error object
278 * Read data from the IO channel, storing it in the
279 * memory regions referenced by @iov. Each element
280 * in the @iov will be fully populated with data
281 * before the next one is used. The @niov parameter
282 * specifies the total number of elements in @iov.
284 * The function will wait for all requested data
285 * to be read, yielding from the current coroutine
286 * if required.
288 * If end-of-file occurs before any data is read,
289 * no error is reported; otherwise, if it occurs
290 * before all requested data has been read, an error
291 * will be reported.
293 * Returns: 1 if all bytes were read, 0 if end-of-file
294 * occurs without data, or -1 on error
296 int qio_channel_readv_all_eof(QIOChannel *ioc,
297 const struct iovec *iov,
298 size_t niov,
299 Error **errp);
302 * qio_channel_readv_all:
303 * @ioc: the channel object
304 * @iov: the array of memory regions to read data into
305 * @niov: the length of the @iov array
306 * @errp: pointer to a NULL-initialized error object
308 * Read data from the IO channel, storing it in the
309 * memory regions referenced by @iov. Each element
310 * in the @iov will be fully populated with data
311 * before the next one is used. The @niov parameter
312 * specifies the total number of elements in @iov.
314 * The function will wait for all requested data
315 * to be read, yielding from the current coroutine
316 * if required.
318 * If end-of-file occurs before all requested data
319 * has been read, an error will be reported.
321 * Returns: 0 if all bytes were read, or -1 on error
323 int qio_channel_readv_all(QIOChannel *ioc,
324 const struct iovec *iov,
325 size_t niov,
326 Error **errp);
330 * qio_channel_writev_all:
331 * @ioc: the channel object
332 * @iov: the array of memory regions to write data from
333 * @niov: the length of the @iov array
334 * @errp: pointer to a NULL-initialized error object
336 * Write data to the IO channel, reading it from the
337 * memory regions referenced by @iov. Each element
338 * in the @iov will be fully sent, before the next
339 * one is used. The @niov parameter specifies the
340 * total number of elements in @iov.
342 * The function will wait for all requested data
343 * to be written, yielding from the current coroutine
344 * if required.
346 * Returns: 0 if all bytes were written, or -1 on error
348 int qio_channel_writev_all(QIOChannel *ioc,
349 const struct iovec *iov,
350 size_t niov,
351 Error **erp);
354 * qio_channel_readv:
355 * @ioc: the channel object
356 * @iov: the array of memory regions to read data into
357 * @niov: the length of the @iov array
358 * @errp: pointer to a NULL-initialized error object
360 * Behaves as qio_channel_readv_full() but does not support
361 * receiving of file handles.
363 ssize_t qio_channel_readv(QIOChannel *ioc,
364 const struct iovec *iov,
365 size_t niov,
366 Error **errp);
369 * qio_channel_writev:
370 * @ioc: the channel object
371 * @iov: the array of memory regions to write data from
372 * @niov: the length of the @iov array
373 * @errp: pointer to a NULL-initialized error object
375 * Behaves as qio_channel_writev_full() but does not support
376 * sending of file handles.
378 ssize_t qio_channel_writev(QIOChannel *ioc,
379 const struct iovec *iov,
380 size_t niov,
381 Error **errp);
384 * qio_channel_read:
385 * @ioc: the channel object
386 * @buf: the memory region to read data into
387 * @buflen: the length of @buf
388 * @errp: pointer to a NULL-initialized error object
390 * Behaves as qio_channel_readv_full() but does not support
391 * receiving of file handles, and only supports reading into
392 * a single memory region.
394 ssize_t qio_channel_read(QIOChannel *ioc,
395 char *buf,
396 size_t buflen,
397 Error **errp);
400 * qio_channel_write:
401 * @ioc: the channel object
402 * @buf: the memory regions to send data from
403 * @buflen: the length of @buf
404 * @errp: pointer to a NULL-initialized error object
406 * Behaves as qio_channel_writev_full() but does not support
407 * sending of file handles, and only supports writing from a
408 * single memory region.
410 ssize_t qio_channel_write(QIOChannel *ioc,
411 const char *buf,
412 size_t buflen,
413 Error **errp);
416 * qio_channel_read_all_eof:
417 * @ioc: the channel object
418 * @buf: the memory region to read data into
419 * @buflen: the number of bytes to @buf
420 * @errp: pointer to a NULL-initialized error object
422 * Reads @buflen bytes into @buf, possibly blocking or (if the
423 * channel is non-blocking) yielding from the current coroutine
424 * multiple times until the entire content is read. If end-of-file
425 * occurs immediately it is not an error, but if it occurs after
426 * data has been read it will return an error rather than a
427 * short-read. Otherwise behaves as qio_channel_read().
429 * Returns: 1 if all bytes were read, 0 if end-of-file occurs
430 * without data, or -1 on error
432 int qio_channel_read_all_eof(QIOChannel *ioc,
433 char *buf,
434 size_t buflen,
435 Error **errp);
438 * qio_channel_read_all:
439 * @ioc: the channel object
440 * @buf: the memory region to read data into
441 * @buflen: the number of bytes to @buf
442 * @errp: pointer to a NULL-initialized error object
444 * Reads @buflen bytes into @buf, possibly blocking or (if the
445 * channel is non-blocking) yielding from the current coroutine
446 * multiple times until the entire content is read. If end-of-file
447 * occurs it will return an error rather than a short-read. Otherwise
448 * behaves as qio_channel_read().
450 * Returns: 0 if all bytes were read, or -1 on error
452 int qio_channel_read_all(QIOChannel *ioc,
453 char *buf,
454 size_t buflen,
455 Error **errp);
458 * qio_channel_write_all:
459 * @ioc: the channel object
460 * @buf: the memory region to write data into
461 * @buflen: the number of bytes to @buf
462 * @errp: pointer to a NULL-initialized error object
464 * Writes @buflen bytes from @buf, possibly blocking or (if the
465 * channel is non-blocking) yielding from the current coroutine
466 * multiple times until the entire content is written. Otherwise
467 * behaves as qio_channel_write().
469 * Returns: 0 if all bytes were written, or -1 on error
471 int qio_channel_write_all(QIOChannel *ioc,
472 const char *buf,
473 size_t buflen,
474 Error **errp);
477 * qio_channel_set_blocking:
478 * @ioc: the channel object
479 * @enabled: the blocking flag state
480 * @errp: pointer to a NULL-initialized error object
482 * If @enabled is true, then the channel is put into
483 * blocking mode, otherwise it will be non-blocking.
485 * In non-blocking mode, read/write operations may
486 * return QIO_CHANNEL_ERR_BLOCK if they would otherwise
487 * block on I/O
489 int qio_channel_set_blocking(QIOChannel *ioc,
490 bool enabled,
491 Error **errp);
494 * qio_channel_close:
495 * @ioc: the channel object
496 * @errp: pointer to a NULL-initialized error object
498 * Close the channel, flushing any pending I/O
500 * Returns: 0 on success, -1 on error
502 int qio_channel_close(QIOChannel *ioc,
503 Error **errp);
506 * qio_channel_shutdown:
507 * @ioc: the channel object
508 * @how: the direction to shutdown
509 * @errp: pointer to a NULL-initialized error object
511 * Shutdowns transmission and/or receiving of data
512 * without closing the underlying transport.
514 * Not all implementations will support this facility,
515 * so may report an error. To avoid errors, the
516 * caller may check for the feature flag
517 * QIO_CHANNEL_FEATURE_SHUTDOWN prior to calling
518 * this method.
520 * Returns: 0 on success, -1 on error
522 int qio_channel_shutdown(QIOChannel *ioc,
523 QIOChannelShutdown how,
524 Error **errp);
527 * qio_channel_set_delay:
528 * @ioc: the channel object
529 * @enabled: the new flag state
531 * Controls whether the underlying transport is
532 * permitted to delay writes in order to merge
533 * small packets. If @enabled is true, then the
534 * writes may be delayed in order to opportunistically
535 * merge small packets into larger ones. If @enabled
536 * is false, writes are dispatched immediately with
537 * no delay.
539 * When @enabled is false, applications may wish to
540 * use the qio_channel_set_cork() method to explicitly
541 * control write merging.
543 * On channels which are backed by a socket, this
544 * API corresponds to the inverse of TCP_NODELAY flag,
545 * controlling whether the Nagle algorithm is active.
547 * This setting is merely a hint, so implementations are
548 * free to ignore this without it being considered an
549 * error.
551 void qio_channel_set_delay(QIOChannel *ioc,
552 bool enabled);
555 * qio_channel_set_cork:
556 * @ioc: the channel object
557 * @enabled: the new flag state
559 * Controls whether the underlying transport is
560 * permitted to dispatch data that is written.
561 * If @enabled is true, then any data written will
562 * be queued in local buffers until @enabled is
563 * set to false once again.
565 * This feature is typically used when the automatic
566 * write coalescing facility is disabled via the
567 * qio_channel_set_delay() method.
569 * On channels which are backed by a socket, this
570 * API corresponds to the TCP_CORK flag.
572 * This setting is merely a hint, so implementations are
573 * free to ignore this without it being considered an
574 * error.
576 void qio_channel_set_cork(QIOChannel *ioc,
577 bool enabled);
581 * qio_channel_seek:
582 * @ioc: the channel object
583 * @offset: the position to seek to, relative to @whence
584 * @whence: one of the (POSIX) SEEK_* constants listed below
585 * @errp: pointer to a NULL-initialized error object
587 * Moves the current I/O position within the channel
588 * @ioc, to be @offset. The value of @offset is
589 * interpreted relative to @whence:
591 * SEEK_SET - the position is set to @offset bytes
592 * SEEK_CUR - the position is moved by @offset bytes
593 * SEEK_END - the position is set to end of the file plus @offset bytes
595 * Not all implementations will support this facility,
596 * so may report an error.
598 * Returns: the new position on success, (off_t)-1 on failure
600 off_t qio_channel_io_seek(QIOChannel *ioc,
601 off_t offset,
602 int whence,
603 Error **errp);
607 * qio_channel_create_watch:
608 * @ioc: the channel object
609 * @condition: the I/O condition to monitor
611 * Create a new main loop source that is used to watch
612 * for the I/O condition @condition. Typically the
613 * qio_channel_add_watch() method would be used instead
614 * of this, since it directly attaches a callback to
615 * the source
617 * Returns: the new main loop source.
619 GSource *qio_channel_create_watch(QIOChannel *ioc,
620 GIOCondition condition);
623 * qio_channel_add_watch:
624 * @ioc: the channel object
625 * @condition: the I/O condition to monitor
626 * @func: callback to invoke when the source becomes ready
627 * @user_data: opaque data to pass to @func
628 * @notify: callback to free @user_data
630 * Create a new main loop source that is used to watch
631 * for the I/O condition @condition. The callback @func
632 * will be registered against the source, to be invoked
633 * when the source becomes ready. The optional @user_data
634 * will be passed to @func when it is invoked. The @notify
635 * callback will be used to free @user_data when the
636 * watch is deleted
638 * The returned source ID can be used with g_source_remove()
639 * to remove and free the source when no longer required.
640 * Alternatively the @func callback can return a FALSE
641 * value.
643 * Returns: the source ID
645 guint qio_channel_add_watch(QIOChannel *ioc,
646 GIOCondition condition,
647 QIOChannelFunc func,
648 gpointer user_data,
649 GDestroyNotify notify);
652 * qio_channel_add_watch_full:
653 * @ioc: the channel object
654 * @condition: the I/O condition to monitor
655 * @func: callback to invoke when the source becomes ready
656 * @user_data: opaque data to pass to @func
657 * @notify: callback to free @user_data
658 * @context: the context to run the watch source
660 * Similar as qio_channel_add_watch(), but allows to specify context
661 * to run the watch source.
663 * Returns: the source ID
665 guint qio_channel_add_watch_full(QIOChannel *ioc,
666 GIOCondition condition,
667 QIOChannelFunc func,
668 gpointer user_data,
669 GDestroyNotify notify,
670 GMainContext *context);
673 * qio_channel_add_watch_source:
674 * @ioc: the channel object
675 * @condition: the I/O condition to monitor
676 * @func: callback to invoke when the source becomes ready
677 * @user_data: opaque data to pass to @func
678 * @notify: callback to free @user_data
679 * @context: gcontext to bind the source to
681 * Similar as qio_channel_add_watch(), but allows to specify context
682 * to run the watch source, meanwhile return the GSource object
683 * instead of tag ID, with the GSource referenced already.
685 * Note: callers is responsible to unref the source when not needed.
687 * Returns: the source pointer
689 GSource *qio_channel_add_watch_source(QIOChannel *ioc,
690 GIOCondition condition,
691 QIOChannelFunc func,
692 gpointer user_data,
693 GDestroyNotify notify,
694 GMainContext *context);
697 * qio_channel_attach_aio_context:
698 * @ioc: the channel object
699 * @ctx: the #AioContext to set the handlers on
701 * Request that qio_channel_yield() sets I/O handlers on
702 * the given #AioContext. If @ctx is %NULL, qio_channel_yield()
703 * uses QEMU's main thread event loop.
705 * You can move a #QIOChannel from one #AioContext to another even if
706 * I/O handlers are set for a coroutine. However, #QIOChannel provides
707 * no synchronization between the calls to qio_channel_yield() and
708 * qio_channel_attach_aio_context().
710 * Therefore you should first call qio_channel_detach_aio_context()
711 * to ensure that the coroutine is not entered concurrently. Then,
712 * while the coroutine has yielded, call qio_channel_attach_aio_context(),
713 * and then aio_co_schedule() to place the coroutine on the new
714 * #AioContext. The calls to qio_channel_detach_aio_context()
715 * and qio_channel_attach_aio_context() should be protected with
716 * aio_context_acquire() and aio_context_release().
718 void qio_channel_attach_aio_context(QIOChannel *ioc,
719 AioContext *ctx);
722 * qio_channel_detach_aio_context:
723 * @ioc: the channel object
725 * Disable any I/O handlers set by qio_channel_yield(). With the
726 * help of aio_co_schedule(), this allows moving a coroutine that was
727 * paused by qio_channel_yield() to another context.
729 void qio_channel_detach_aio_context(QIOChannel *ioc);
732 * qio_channel_yield:
733 * @ioc: the channel object
734 * @condition: the I/O condition to wait for
736 * Yields execution from the current coroutine until the condition
737 * indicated by @condition becomes available. @condition must
738 * be either %G_IO_IN or %G_IO_OUT; it cannot contain both. In
739 * addition, no two coroutine can be waiting on the same condition
740 * and channel at the same time.
742 * This must only be called from coroutine context
744 void qio_channel_yield(QIOChannel *ioc,
745 GIOCondition condition);
748 * qio_channel_wait:
749 * @ioc: the channel object
750 * @condition: the I/O condition to wait for
752 * Block execution from the current thread until
753 * the condition indicated by @condition becomes
754 * available.
756 * This will enter a nested event loop to perform
757 * the wait.
759 void qio_channel_wait(QIOChannel *ioc,
760 GIOCondition condition);
763 * qio_channel_set_aio_fd_handler:
764 * @ioc: the channel object
765 * @ctx: the AioContext to set the handlers on
766 * @io_read: the read handler
767 * @io_write: the write handler
768 * @opaque: the opaque value passed to the handler
770 * This is used internally by qio_channel_yield(). It can
771 * be used by channel implementations to forward the handlers
772 * to another channel (e.g. from #QIOChannelTLS to the
773 * underlying socket).
775 void qio_channel_set_aio_fd_handler(QIOChannel *ioc,
776 AioContext *ctx,
777 IOHandler *io_read,
778 IOHandler *io_write,
779 void *opaque);
781 #endif /* QIO_CHANNEL_H */