chiark / gitweb /
doxygen + file naming
[disorder] / lib / speaker-protocol.c
CommitLineData
460b9539 1/*
2 * This file is part of DisOrder.
ea410ba1 3 * Copyright (C) 2005, 2007 Richard Kettlewell
460b9539 4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
18 * USA
19 */
ea410ba1 20/** @file lib/speaker-protocol.c
21 * @brief Speaker/server protocol support
22 */
460b9539 23
24#include <config.h>
25#include "types.h"
26
27#include <sys/socket.h>
28#include <string.h>
29#include <errno.h>
30#include <sys/uio.h>
31
ea410ba1 32#include "speaker-protocol.h"
460b9539 33#include "log.h"
34
ea410ba1 35/** @brief Send a speaker messages
36 * @param fd File descriptor to send to
37 * @param sm Pointer to message
38 * @param datafd File descriptor to pass with message or -1
39 */
460b9539 40void speaker_send(int fd, const struct speaker_message *sm, int datafd) {
41 struct msghdr m;
42 struct iovec iov;
43 union {
44 struct cmsghdr cmsg;
45 char size[CMSG_SPACE(sizeof (int))];
46 } u;
47 int ret;
48
49 memset(&m, 0, sizeof m);
50 m.msg_iov = &iov;
51 m.msg_iovlen = 1;
52 iov.iov_base = (void *)sm;
53 iov.iov_len = sizeof *sm;
54 if(datafd != -1) {
55 m.msg_control = (void *)&u.cmsg;
56 m.msg_controllen = sizeof u;
57 memset(&u, 0, sizeof u);
58 u.cmsg.cmsg_len = CMSG_LEN(sizeof (int));
59 u.cmsg.cmsg_level = SOL_SOCKET;
60 u.cmsg.cmsg_type = SCM_RIGHTS;
61 *(int *)CMSG_DATA(&u.cmsg) = datafd;
62 }
63 do {
64 ret = sendmsg(fd, &m, 0);
65 } while(ret < 0 && errno == EINTR);
66 if(ret < 0)
67 fatal(errno, "sendmsg");
68}
69
ea410ba1 70/** @brief Receive a speaker message
71 * @param fd File descriptor to read from
72 * @param sm Where to store received message
73 * @param datafd Where to store received file descriptor or NULL
74 * @return -ve on @c EAGAIN, 0 at EOF, +ve on success
75 *
76 * If @p datafd is NULL but a file descriptor is nonetheless received,
77 * the process is terminated with an error.
78 */
460b9539 79int speaker_recv(int fd, struct speaker_message *sm, int *datafd) {
80 struct msghdr m;
81 struct iovec iov;
82 union {
83 struct cmsghdr cmsg;
84 char size[CMSG_SPACE(sizeof (int))];
85 } u;
86 int ret;
87
88 memset(&m, 0, sizeof m);
89 m.msg_iov = &iov;
90 m.msg_iovlen = 1;
91 iov.iov_base = (void *)sm;
92 iov.iov_len = sizeof *sm;
93 if(datafd) {
94 m.msg_control = (void *)&u.cmsg;
95 m.msg_controllen = sizeof u;
96 memset(&u, 0, sizeof u);
97 u.cmsg.cmsg_len = CMSG_LEN(sizeof (int));
98 u.cmsg.cmsg_level = SOL_SOCKET;
99 u.cmsg.cmsg_type = SCM_RIGHTS;
100 *datafd = -1;
101 }
102 do {
103 ret = recvmsg(fd, &m, MSG_DONTWAIT);
104 } while(ret < 0 && errno == EINTR);
105 if(ret < 0) {
106 if(errno != EAGAIN) fatal(errno, "recvmsg");
107 return -1;
108 }
109 if((size_t)m.msg_controllen >= CMSG_LEN(sizeof (int))) {
110 if(!datafd)
111 fatal(0, "got an unexpected file descriptor from recvmsg");
112 else
113 *datafd = *(int *)CMSG_DATA(&u.cmsg);
114 }
115 return ret;
116}
117
118/*
119Local Variables:
120c-basic-offset:2
121comment-column:40
122fill-column:79
123indent-tabs-mode:nil
124End:
125*/