2 This file is part of PulseAudio.
4 Copyright 2004-2006 Lennart Poettering
5 Copyright 2006-2007 Pierre Ossman <ossman@cendio.se> for Cendio AB
7 PulseAudio is free software; you can redistribute it and/or modify
8 it under the terms of the GNU Lesser General Public License as
9 published by the Free Software Foundation; either version 2.1 of the
10 License, or (at your option) any later version.
12 PulseAudio is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Lesser General Public License for more details.
17 You should have received a copy of the GNU Lesser General Public
18 License along with PulseAudio; if not, write to the Free Software
19 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
35 #include <pulse/xmalloc.h>
37 #include <pulsecore/core-error.h>
38 #include <pulsecore/core-util.h>
39 #include <pulsecore/socket.h>
40 #include <pulsecore/socket-util.h>
41 #include <pulsecore/log.h>
42 #include <pulsecore/macro.h>
44 #include "iochannel.h"
48 int ifd_type
, ofd_type
;
49 pa_mainloop_api
* mainloop
;
51 pa_iochannel_cb_t callback
;
59 pa_io_event
* input_event
, *output_event
;
62 static void callback(pa_mainloop_api
* m
, pa_io_event
*e
, int fd
, pa_io_event_flags_t f
, void *userdata
);
64 static void delete_events(pa_iochannel
*io
) {
68 io
->mainloop
->io_free(io
->input_event
);
70 if (io
->output_event
&& io
->output_event
!= io
->input_event
)
71 io
->mainloop
->io_free(io
->output_event
);
73 io
->input_event
= io
->output_event
= NULL
;
76 static void enable_events(pa_iochannel
*io
) {
84 if (io
->ifd
== io
->ofd
&& io
->ifd
>= 0) {
85 pa_io_event_flags_t f
= PA_IO_EVENT_NULL
;
88 f
|= PA_IO_EVENT_INPUT
;
90 f
|= PA_IO_EVENT_OUTPUT
;
92 pa_assert(io
->input_event
== io
->output_event
);
94 if (f
!= PA_IO_EVENT_NULL
) {
96 io
->mainloop
->io_enable(io
->input_event
, f
);
98 io
->input_event
= io
->output_event
= io
->mainloop
->io_new(io
->mainloop
, io
->ifd
, f
, callback
, io
);
107 io
->mainloop
->io_enable(io
->input_event
, PA_IO_EVENT_INPUT
);
109 io
->input_event
= io
->mainloop
->io_new(io
->mainloop
, io
->ifd
, PA_IO_EVENT_INPUT
, callback
, io
);
110 } else if (io
->input_event
) {
111 io
->mainloop
->io_free(io
->input_event
);
112 io
->input_event
= NULL
;
118 if (io
->output_event
)
119 io
->mainloop
->io_enable(io
->output_event
, PA_IO_EVENT_OUTPUT
);
121 io
->output_event
= io
->mainloop
->io_new(io
->mainloop
, io
->ofd
, PA_IO_EVENT_OUTPUT
, callback
, io
);
122 } else if (io
->input_event
) {
123 io
->mainloop
->io_free(io
->output_event
);
124 io
->output_event
= NULL
;
130 static void callback(pa_mainloop_api
* m
, pa_io_event
*e
, int fd
, pa_io_event_flags_t f
, void *userdata
) {
131 pa_iochannel
*io
= userdata
;
132 pa_bool_t changed
= FALSE
;
139 if ((f
& (PA_IO_EVENT_HANGUP
|PA_IO_EVENT_ERROR
)) && !io
->hungup
) {
144 if ((f
& PA_IO_EVENT_INPUT
) && !io
->readable
) {
147 pa_assert(e
== io
->input_event
);
150 if ((f
& PA_IO_EVENT_OUTPUT
) && !io
->writable
) {
153 pa_assert(e
== io
->output_event
);
160 io
->callback(io
, io
->userdata
);
164 pa_iochannel
* pa_iochannel_new(pa_mainloop_api
*m
, int ifd
, int ofd
) {
168 pa_assert(ifd
>= 0 || ofd
>= 0);
170 io
= pa_xnew0(pa_iochannel
, 1);
176 pa_make_fd_nonblock(io
->ifd
);
178 if (io
->ofd
>= 0 && io
->ofd
!= io
->ifd
)
179 pa_make_fd_nonblock(io
->ofd
);
185 void pa_iochannel_free(pa_iochannel
*io
) {
193 if (io
->ofd
>= 0 && io
->ofd
!= io
->ifd
)
200 pa_bool_t
pa_iochannel_is_readable(pa_iochannel
*io
) {
203 return io
->readable
|| io
->hungup
;
206 pa_bool_t
pa_iochannel_is_writable(pa_iochannel
*io
) {
209 return io
->writable
&& !io
->hungup
;
212 pa_bool_t
pa_iochannel_is_hungup(pa_iochannel
*io
) {
218 ssize_t
pa_iochannel_write(pa_iochannel
*io
, const void*data
, size_t l
) {
224 pa_assert(io
->ofd
>= 0);
226 if ((r
= pa_write(io
->ofd
, data
, l
, &io
->ofd_type
)) >= 0) {
227 io
->writable
= io
->hungup
= FALSE
;
234 ssize_t
pa_iochannel_read(pa_iochannel
*io
, void*data
, size_t l
) {
239 pa_assert(io
->ifd
>= 0);
241 if ((r
= pa_read(io
->ifd
, data
, l
, &io
->ifd_type
)) >= 0) {
243 /* We also reset the hangup flag here to ensure that another
244 * IO callback is triggered so that we will again call into
246 io
->readable
= io
->hungup
= FALSE
;
255 pa_bool_t
pa_iochannel_creds_supported(pa_iochannel
*io
) {
258 struct sockaddr_un un
;
259 struct sockaddr_storage storage
;
265 pa_assert(io
->ifd
>= 0);
266 pa_assert(io
->ofd
== io
->ifd
);
269 if (getsockname(io
->ifd
, &sa
.sa
, &l
) < 0)
272 return sa
.sa
.sa_family
== AF_UNIX
;
275 int pa_iochannel_creds_enable(pa_iochannel
*io
) {
279 pa_assert(io
->ifd
>= 0);
281 if (setsockopt(io
->ifd
, SOL_SOCKET
, SO_PASSCRED
, &t
, sizeof(t
)) < 0) {
282 pa_log_error("setsockopt(SOL_SOCKET, SO_PASSCRED): %s", pa_cstrerror(errno
));
289 ssize_t
pa_iochannel_write_with_creds(pa_iochannel
*io
, const void*data
, size_t l
, const pa_creds
*ucred
) {
295 uint8_t data
[CMSG_SPACE(sizeof(struct ucred
))];
302 pa_assert(io
->ofd
>= 0);
305 iov
.iov_base
= (void*) data
;
309 cmsg
.hdr
.cmsg_len
= CMSG_LEN(sizeof(struct ucred
));
310 cmsg
.hdr
.cmsg_level
= SOL_SOCKET
;
311 cmsg
.hdr
.cmsg_type
= SCM_CREDENTIALS
;
313 u
= (struct ucred
*) CMSG_DATA(&cmsg
.hdr
);
327 mh
.msg_control
= &cmsg
;
328 mh
.msg_controllen
= sizeof(cmsg
);
330 if ((r
= sendmsg(io
->ofd
, &mh
, MSG_NOSIGNAL
)) >= 0) {
331 io
->writable
= io
->hungup
= FALSE
;
338 ssize_t
pa_iochannel_read_with_creds(pa_iochannel
*io
, void*data
, size_t l
, pa_creds
*creds
, pa_bool_t
*creds_valid
) {
344 uint8_t data
[CMSG_SPACE(sizeof(struct ucred
))];
350 pa_assert(io
->ifd
>= 0);
352 pa_assert(creds_valid
);
362 mh
.msg_control
= &cmsg
;
363 mh
.msg_controllen
= sizeof(cmsg
);
365 if ((r
= recvmsg(io
->ifd
, &mh
, 0)) >= 0) {
368 *creds_valid
= FALSE
;
370 for (cmh
= CMSG_FIRSTHDR(&mh
); cmh
; cmh
= CMSG_NXTHDR(&mh
, cmh
)) {
372 if (cmh
->cmsg_level
== SOL_SOCKET
&& cmh
->cmsg_type
== SCM_CREDENTIALS
) {
374 pa_assert(cmh
->cmsg_len
== CMSG_LEN(sizeof(struct ucred
)));
375 memcpy(&u
, CMSG_DATA(cmh
), sizeof(struct ucred
));
384 io
->readable
= io
->hungup
= FALSE
;
391 #endif /* HAVE_CREDS */
393 void pa_iochannel_set_callback(pa_iochannel
*io
, pa_iochannel_cb_t _callback
, void *userdata
) {
396 io
->callback
= _callback
;
397 io
->userdata
= userdata
;
400 void pa_iochannel_set_noclose(pa_iochannel
*io
, pa_bool_t b
) {
406 void pa_iochannel_socket_peer_to_string(pa_iochannel
*io
, char*s
, size_t l
) {
411 pa_socket_peer_to_string(io
->ifd
, s
, l
);
414 int pa_iochannel_socket_set_rcvbuf(pa_iochannel
*io
, size_t l
) {
417 return pa_socket_set_rcvbuf(io
->ifd
, l
);
420 int pa_iochannel_socket_set_sndbuf(pa_iochannel
*io
, size_t l
) {
423 return pa_socket_set_sndbuf(io
->ofd
, l
);
426 pa_mainloop_api
* pa_iochannel_get_mainloop_api(pa_iochannel
*io
) {
432 int pa_iochannel_get_recv_fd(pa_iochannel
*io
) {
438 int pa_iochannel_get_send_fd(pa_iochannel
*io
) {
444 pa_bool_t
pa_iochannel_socket_is_local(pa_iochannel
*io
) {
447 if (pa_socket_is_local(io
->ifd
))
450 if (io
->ifd
!= io
->ofd
)
451 if (pa_socket_is_local(io
->ofd
))