root / qemu-sockets.c @ 475dc65f
History | View | Annotate | Download (11.8 kB)
1 | 305b0eb2 | aliguori | /*
|
---|---|---|---|
2 | 305b0eb2 | aliguori | * inet and unix socket functions for qemu
|
3 | 305b0eb2 | aliguori | *
|
4 | 305b0eb2 | aliguori | * (c) 2008 Gerd Hoffmann <kraxel@redhat.com>
|
5 | 305b0eb2 | aliguori | *
|
6 | 305b0eb2 | aliguori | * This program is free software; you can redistribute it and/or modify
|
7 | 305b0eb2 | aliguori | * it under the terms of the GNU General Public License as published by
|
8 | 305b0eb2 | aliguori | * the Free Software Foundation; under version 2 of the License.
|
9 | 305b0eb2 | aliguori | *
|
10 | 305b0eb2 | aliguori | * This program is distributed in the hope that it will be useful,
|
11 | 305b0eb2 | aliguori | * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
12 | 305b0eb2 | aliguori | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
13 | 305b0eb2 | aliguori | * GNU General Public License for more details.
|
14 | 305b0eb2 | aliguori | */
|
15 | d247d25f | aliguori | #include <stdio.h> |
16 | d247d25f | aliguori | #include <stdlib.h> |
17 | d247d25f | aliguori | #include <string.h> |
18 | d247d25f | aliguori | #include <ctype.h> |
19 | d247d25f | aliguori | #include <errno.h> |
20 | d247d25f | aliguori | #include <unistd.h> |
21 | d247d25f | aliguori | |
22 | d247d25f | aliguori | #include "qemu_socket.h" |
23 | 47398b9c | blueswir1 | #include "qemu-common.h" /* for qemu_isdigit */ |
24 | d247d25f | aliguori | |
25 | d247d25f | aliguori | #ifndef AI_ADDRCONFIG
|
26 | d247d25f | aliguori | # define AI_ADDRCONFIG 0 |
27 | d247d25f | aliguori | #endif
|
28 | d247d25f | aliguori | |
29 | d247d25f | aliguori | static int sockets_debug = 0; |
30 | d247d25f | aliguori | static const int on=1, off=0; |
31 | d247d25f | aliguori | |
32 | d247d25f | aliguori | static int inet_getport(struct addrinfo *e) |
33 | d247d25f | aliguori | { |
34 | d247d25f | aliguori | struct sockaddr_in *i4;
|
35 | d247d25f | aliguori | struct sockaddr_in6 *i6;
|
36 | d247d25f | aliguori | |
37 | d247d25f | aliguori | switch (e->ai_family) {
|
38 | d247d25f | aliguori | case PF_INET6:
|
39 | d247d25f | aliguori | i6 = (void*)e->ai_addr;
|
40 | d247d25f | aliguori | return ntohs(i6->sin6_port);
|
41 | d247d25f | aliguori | case PF_INET:
|
42 | d247d25f | aliguori | i4 = (void*)e->ai_addr;
|
43 | d247d25f | aliguori | return ntohs(i4->sin_port);
|
44 | d247d25f | aliguori | default:
|
45 | d247d25f | aliguori | return 0; |
46 | d247d25f | aliguori | } |
47 | d247d25f | aliguori | } |
48 | d247d25f | aliguori | |
49 | d247d25f | aliguori | static void inet_setport(struct addrinfo *e, int port) |
50 | d247d25f | aliguori | { |
51 | d247d25f | aliguori | struct sockaddr_in *i4;
|
52 | d247d25f | aliguori | struct sockaddr_in6 *i6;
|
53 | d247d25f | aliguori | |
54 | d247d25f | aliguori | switch (e->ai_family) {
|
55 | d247d25f | aliguori | case PF_INET6:
|
56 | d247d25f | aliguori | i6 = (void*)e->ai_addr;
|
57 | d247d25f | aliguori | i6->sin6_port = htons(port); |
58 | d247d25f | aliguori | break;
|
59 | d247d25f | aliguori | case PF_INET:
|
60 | d247d25f | aliguori | i4 = (void*)e->ai_addr;
|
61 | d247d25f | aliguori | i4->sin_port = htons(port); |
62 | d247d25f | aliguori | break;
|
63 | d247d25f | aliguori | } |
64 | d247d25f | aliguori | } |
65 | d247d25f | aliguori | |
66 | d247d25f | aliguori | static const char *inet_strfamily(int family) |
67 | d247d25f | aliguori | { |
68 | d247d25f | aliguori | switch (family) {
|
69 | d247d25f | aliguori | case PF_INET6: return "ipv6"; |
70 | d247d25f | aliguori | case PF_INET: return "ipv4"; |
71 | d247d25f | aliguori | case PF_UNIX: return "unix"; |
72 | d247d25f | aliguori | } |
73 | d247d25f | aliguori | return "????"; |
74 | d247d25f | aliguori | } |
75 | d247d25f | aliguori | |
76 | d247d25f | aliguori | static void inet_print_addrinfo(const char *tag, struct addrinfo *res) |
77 | d247d25f | aliguori | { |
78 | d247d25f | aliguori | struct addrinfo *e;
|
79 | d247d25f | aliguori | char uaddr[INET6_ADDRSTRLEN+1]; |
80 | d247d25f | aliguori | char uport[33]; |
81 | d247d25f | aliguori | |
82 | d247d25f | aliguori | for (e = res; e != NULL; e = e->ai_next) { |
83 | d247d25f | aliguori | getnameinfo((struct sockaddr*)e->ai_addr,e->ai_addrlen,
|
84 | d247d25f | aliguori | uaddr,INET6_ADDRSTRLEN,uport,32,
|
85 | d247d25f | aliguori | NI_NUMERICHOST | NI_NUMERICSERV); |
86 | d247d25f | aliguori | fprintf(stderr,"%s: getaddrinfo: family %s, host %s, port %s\n",
|
87 | d247d25f | aliguori | tag, inet_strfamily(e->ai_family), uaddr, uport); |
88 | d247d25f | aliguori | } |
89 | d247d25f | aliguori | } |
90 | d247d25f | aliguori | |
91 | d247d25f | aliguori | int inet_listen(const char *str, char *ostr, int olen, |
92 | d247d25f | aliguori | int socktype, int port_offset) |
93 | d247d25f | aliguori | { |
94 | d247d25f | aliguori | struct addrinfo ai,*res,*e;
|
95 | d247d25f | aliguori | char addr[64]; |
96 | d247d25f | aliguori | char port[33]; |
97 | d247d25f | aliguori | char uaddr[INET6_ADDRSTRLEN+1]; |
98 | d247d25f | aliguori | char uport[33]; |
99 | d247d25f | aliguori | const char *opts, *h; |
100 | d247d25f | aliguori | int slisten,rc,pos,to,try_next;
|
101 | d247d25f | aliguori | |
102 | d247d25f | aliguori | memset(&ai,0, sizeof(ai)); |
103 | d247d25f | aliguori | ai.ai_flags = AI_PASSIVE | AI_ADDRCONFIG; |
104 | d247d25f | aliguori | ai.ai_family = PF_UNSPEC; |
105 | d247d25f | aliguori | ai.ai_socktype = socktype; |
106 | d247d25f | aliguori | |
107 | d247d25f | aliguori | /* parse address */
|
108 | d247d25f | aliguori | if (str[0] == ':') { |
109 | d247d25f | aliguori | /* no host given */
|
110 | d247d25f | aliguori | strcpy(addr,"");
|
111 | d247d25f | aliguori | if (1 != sscanf(str,":%32[^,]%n",port,&pos)) { |
112 | d247d25f | aliguori | fprintf(stderr, "%s: portonly parse error (%s)\n",
|
113 | d247d25f | aliguori | __FUNCTION__, str); |
114 | d247d25f | aliguori | return -1; |
115 | d247d25f | aliguori | } |
116 | d247d25f | aliguori | } else if (str[0] == '[') { |
117 | d247d25f | aliguori | /* IPv6 addr */
|
118 | d247d25f | aliguori | if (2 != sscanf(str,"[%64[^]]]:%32[^,]%n",addr,port,&pos)) { |
119 | d247d25f | aliguori | fprintf(stderr, "%s: ipv6 parse error (%s)\n",
|
120 | d247d25f | aliguori | __FUNCTION__, str); |
121 | d247d25f | aliguori | return -1; |
122 | d247d25f | aliguori | } |
123 | d247d25f | aliguori | ai.ai_family = PF_INET6; |
124 | 47398b9c | blueswir1 | } else if (qemu_isdigit(str[0])) { |
125 | d247d25f | aliguori | /* IPv4 addr */
|
126 | d247d25f | aliguori | if (2 != sscanf(str,"%64[0-9.]:%32[^,]%n",addr,port,&pos)) { |
127 | d247d25f | aliguori | fprintf(stderr, "%s: ipv4 parse error (%s)\n",
|
128 | d247d25f | aliguori | __FUNCTION__, str); |
129 | d247d25f | aliguori | return -1; |
130 | d247d25f | aliguori | } |
131 | d247d25f | aliguori | ai.ai_family = PF_INET; |
132 | d247d25f | aliguori | } else {
|
133 | d247d25f | aliguori | /* hostname */
|
134 | d247d25f | aliguori | if (2 != sscanf(str,"%64[^:]:%32[^,]%n",addr,port,&pos)) { |
135 | d247d25f | aliguori | fprintf(stderr, "%s: hostname parse error (%s)\n",
|
136 | d247d25f | aliguori | __FUNCTION__, str); |
137 | d247d25f | aliguori | return -1; |
138 | d247d25f | aliguori | } |
139 | d247d25f | aliguori | } |
140 | d247d25f | aliguori | |
141 | d247d25f | aliguori | /* parse options */
|
142 | d247d25f | aliguori | opts = str + pos; |
143 | d247d25f | aliguori | h = strstr(opts, ",to=");
|
144 | d247d25f | aliguori | to = h ? atoi(h+4) : 0; |
145 | d247d25f | aliguori | if (strstr(opts, ",ipv4")) |
146 | d247d25f | aliguori | ai.ai_family = PF_INET; |
147 | d247d25f | aliguori | if (strstr(opts, ",ipv6")) |
148 | d247d25f | aliguori | ai.ai_family = PF_INET6; |
149 | d247d25f | aliguori | |
150 | d247d25f | aliguori | /* lookup */
|
151 | d247d25f | aliguori | if (port_offset)
|
152 | d247d25f | aliguori | snprintf(port, sizeof(port), "%d", atoi(port) + port_offset); |
153 | d247d25f | aliguori | rc = getaddrinfo(strlen(addr) ? addr : NULL, port, &ai, &res);
|
154 | d247d25f | aliguori | if (rc != 0) { |
155 | d247d25f | aliguori | fprintf(stderr,"%s: getaddrinfo(%s,%s): %s\n", __FUNCTION__,
|
156 | d247d25f | aliguori | addr, port, gai_strerror(rc)); |
157 | d247d25f | aliguori | return -1; |
158 | d247d25f | aliguori | } |
159 | d247d25f | aliguori | if (sockets_debug)
|
160 | d247d25f | aliguori | inet_print_addrinfo(__FUNCTION__, res); |
161 | d247d25f | aliguori | |
162 | d247d25f | aliguori | /* create socket + bind */
|
163 | d247d25f | aliguori | for (e = res; e != NULL; e = e->ai_next) { |
164 | d247d25f | aliguori | getnameinfo((struct sockaddr*)e->ai_addr,e->ai_addrlen,
|
165 | d247d25f | aliguori | uaddr,INET6_ADDRSTRLEN,uport,32,
|
166 | d247d25f | aliguori | NI_NUMERICHOST | NI_NUMERICSERV); |
167 | d247d25f | aliguori | slisten = socket(e->ai_family, e->ai_socktype, e->ai_protocol); |
168 | d247d25f | aliguori | if (slisten < 0) { |
169 | d247d25f | aliguori | fprintf(stderr,"%s: socket(%s): %s\n", __FUNCTION__,
|
170 | d247d25f | aliguori | inet_strfamily(e->ai_family), strerror(errno)); |
171 | d247d25f | aliguori | continue;
|
172 | d247d25f | aliguori | } |
173 | d247d25f | aliguori | |
174 | d247d25f | aliguori | setsockopt(slisten,SOL_SOCKET,SO_REUSEADDR,(void*)&on,sizeof(on)); |
175 | d247d25f | aliguori | #ifdef IPV6_V6ONLY
|
176 | d247d25f | aliguori | if (e->ai_family == PF_INET6) {
|
177 | d247d25f | aliguori | /* listen on both ipv4 and ipv6 */
|
178 | d247d25f | aliguori | setsockopt(slisten,IPPROTO_IPV6,IPV6_V6ONLY,(void*)&off,sizeof(off)); |
179 | d247d25f | aliguori | } |
180 | d247d25f | aliguori | #endif
|
181 | d247d25f | aliguori | |
182 | d247d25f | aliguori | for (;;) {
|
183 | d247d25f | aliguori | if (bind(slisten, e->ai_addr, e->ai_addrlen) == 0) { |
184 | d247d25f | aliguori | if (sockets_debug)
|
185 | d247d25f | aliguori | fprintf(stderr,"%s: bind(%s,%s,%d): OK\n", __FUNCTION__,
|
186 | d247d25f | aliguori | inet_strfamily(e->ai_family), uaddr, inet_getport(e)); |
187 | d247d25f | aliguori | goto listen;
|
188 | d247d25f | aliguori | } |
189 | d247d25f | aliguori | try_next = to && (inet_getport(e) <= to + port_offset); |
190 | d247d25f | aliguori | if (!try_next || sockets_debug)
|
191 | d247d25f | aliguori | fprintf(stderr,"%s: bind(%s,%s,%d): %s\n", __FUNCTION__,
|
192 | d247d25f | aliguori | inet_strfamily(e->ai_family), uaddr, inet_getport(e), |
193 | d247d25f | aliguori | strerror(errno)); |
194 | d247d25f | aliguori | if (try_next) {
|
195 | d247d25f | aliguori | inet_setport(e, inet_getport(e) + 1);
|
196 | d247d25f | aliguori | continue;
|
197 | d247d25f | aliguori | } |
198 | d247d25f | aliguori | break;
|
199 | d247d25f | aliguori | } |
200 | d247d25f | aliguori | closesocket(slisten); |
201 | d247d25f | aliguori | } |
202 | d247d25f | aliguori | fprintf(stderr, "%s: FAILED\n", __FUNCTION__);
|
203 | d247d25f | aliguori | freeaddrinfo(res); |
204 | d247d25f | aliguori | return -1; |
205 | d247d25f | aliguori | |
206 | d247d25f | aliguori | listen:
|
207 | d247d25f | aliguori | if (listen(slisten,1) != 0) { |
208 | d247d25f | aliguori | perror("listen");
|
209 | d247d25f | aliguori | closesocket(slisten); |
210 | d247d25f | aliguori | return -1; |
211 | d247d25f | aliguori | } |
212 | d247d25f | aliguori | if (ostr) {
|
213 | d247d25f | aliguori | if (e->ai_family == PF_INET6) {
|
214 | d247d25f | aliguori | snprintf(ostr, olen, "[%s]:%d%s", uaddr,
|
215 | d247d25f | aliguori | inet_getport(e) - port_offset, opts); |
216 | d247d25f | aliguori | } else {
|
217 | d247d25f | aliguori | snprintf(ostr, olen, "%s:%d%s", uaddr,
|
218 | d247d25f | aliguori | inet_getport(e) - port_offset, opts); |
219 | d247d25f | aliguori | } |
220 | d247d25f | aliguori | } |
221 | d247d25f | aliguori | freeaddrinfo(res); |
222 | d247d25f | aliguori | return slisten;
|
223 | d247d25f | aliguori | } |
224 | d247d25f | aliguori | |
225 | d247d25f | aliguori | int inet_connect(const char *str, int socktype) |
226 | d247d25f | aliguori | { |
227 | d247d25f | aliguori | struct addrinfo ai,*res,*e;
|
228 | d247d25f | aliguori | char addr[64]; |
229 | d247d25f | aliguori | char port[33]; |
230 | d247d25f | aliguori | char uaddr[INET6_ADDRSTRLEN+1]; |
231 | d247d25f | aliguori | char uport[33]; |
232 | d247d25f | aliguori | int sock,rc;
|
233 | d247d25f | aliguori | |
234 | d247d25f | aliguori | memset(&ai,0, sizeof(ai)); |
235 | d247d25f | aliguori | ai.ai_flags = AI_CANONNAME | AI_ADDRCONFIG; |
236 | d247d25f | aliguori | ai.ai_family = PF_UNSPEC; |
237 | d247d25f | aliguori | ai.ai_socktype = socktype; |
238 | d247d25f | aliguori | |
239 | d247d25f | aliguori | /* parse address */
|
240 | d247d25f | aliguori | if (str[0] == '[') { |
241 | d247d25f | aliguori | /* IPv6 addr */
|
242 | d247d25f | aliguori | if (2 != sscanf(str,"[%64[^]]]:%32[^,]",addr,port)) { |
243 | d247d25f | aliguori | fprintf(stderr, "%s: ipv6 parse error (%s)\n",
|
244 | d247d25f | aliguori | __FUNCTION__, str); |
245 | d247d25f | aliguori | return -1; |
246 | d247d25f | aliguori | } |
247 | d247d25f | aliguori | ai.ai_family = PF_INET6; |
248 | 47398b9c | blueswir1 | } else if (qemu_isdigit(str[0])) { |
249 | d247d25f | aliguori | /* IPv4 addr */
|
250 | d247d25f | aliguori | if (2 != sscanf(str,"%64[0-9.]:%32[^,]",addr,port)) { |
251 | d247d25f | aliguori | fprintf(stderr, "%s: ipv4 parse error (%s)\n",
|
252 | d247d25f | aliguori | __FUNCTION__, str); |
253 | d247d25f | aliguori | return -1; |
254 | d247d25f | aliguori | } |
255 | d247d25f | aliguori | ai.ai_family = PF_INET; |
256 | d247d25f | aliguori | } else {
|
257 | d247d25f | aliguori | /* hostname */
|
258 | d247d25f | aliguori | if (2 != sscanf(str,"%64[^:]:%32[^,]",addr,port)) { |
259 | d247d25f | aliguori | fprintf(stderr, "%s: hostname parse error (%s)\n",
|
260 | d247d25f | aliguori | __FUNCTION__, str); |
261 | d247d25f | aliguori | return -1; |
262 | d247d25f | aliguori | } |
263 | d247d25f | aliguori | } |
264 | d247d25f | aliguori | |
265 | d247d25f | aliguori | /* parse options */
|
266 | d247d25f | aliguori | if (strstr(str, ",ipv4")) |
267 | d247d25f | aliguori | ai.ai_family = PF_INET; |
268 | d247d25f | aliguori | if (strstr(str, ",ipv6")) |
269 | d247d25f | aliguori | ai.ai_family = PF_INET6; |
270 | d247d25f | aliguori | |
271 | d247d25f | aliguori | /* lookup */
|
272 | d247d25f | aliguori | if (0 != (rc = getaddrinfo(addr, port, &ai, &res))) { |
273 | d247d25f | aliguori | fprintf(stderr,"getaddrinfo(%s,%s): %s\n", gai_strerror(rc),
|
274 | d247d25f | aliguori | addr, port); |
275 | d247d25f | aliguori | return -1; |
276 | d247d25f | aliguori | } |
277 | d247d25f | aliguori | if (sockets_debug)
|
278 | d247d25f | aliguori | inet_print_addrinfo(__FUNCTION__, res); |
279 | d247d25f | aliguori | |
280 | d247d25f | aliguori | for (e = res; e != NULL; e = e->ai_next) { |
281 | d247d25f | aliguori | if (getnameinfo((struct sockaddr*)e->ai_addr,e->ai_addrlen, |
282 | d247d25f | aliguori | uaddr,INET6_ADDRSTRLEN,uport,32,
|
283 | d247d25f | aliguori | NI_NUMERICHOST | NI_NUMERICSERV) != 0) {
|
284 | d247d25f | aliguori | fprintf(stderr,"%s: getnameinfo: oops\n", __FUNCTION__);
|
285 | d247d25f | aliguori | continue;
|
286 | d247d25f | aliguori | } |
287 | d247d25f | aliguori | sock = socket(e->ai_family, e->ai_socktype, e->ai_protocol); |
288 | d247d25f | aliguori | if (sock < 0) { |
289 | d247d25f | aliguori | fprintf(stderr,"%s: socket(%s): %s\n", __FUNCTION__,
|
290 | d247d25f | aliguori | inet_strfamily(e->ai_family), strerror(errno)); |
291 | d247d25f | aliguori | continue;
|
292 | d247d25f | aliguori | } |
293 | d247d25f | aliguori | setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,(void*)&on,sizeof(on)); |
294 | d247d25f | aliguori | |
295 | d247d25f | aliguori | /* connect to peer */
|
296 | d247d25f | aliguori | if (connect(sock,e->ai_addr,e->ai_addrlen) < 0) { |
297 | d247d25f | aliguori | if (sockets_debug || NULL == e->ai_next) |
298 | d247d25f | aliguori | fprintf(stderr, "%s: connect(%s,%s,%s,%s): %s\n", __FUNCTION__,
|
299 | d247d25f | aliguori | inet_strfamily(e->ai_family), |
300 | d247d25f | aliguori | e->ai_canonname, uaddr, uport, strerror(errno)); |
301 | d247d25f | aliguori | closesocket(sock); |
302 | d247d25f | aliguori | continue;
|
303 | d247d25f | aliguori | } |
304 | d247d25f | aliguori | if (sockets_debug)
|
305 | d247d25f | aliguori | fprintf(stderr, "%s: connect(%s,%s,%s,%s): OK\n", __FUNCTION__,
|
306 | d247d25f | aliguori | inet_strfamily(e->ai_family), |
307 | d247d25f | aliguori | e->ai_canonname, uaddr, uport); |
308 | d247d25f | aliguori | freeaddrinfo(res); |
309 | d247d25f | aliguori | return sock;
|
310 | d247d25f | aliguori | } |
311 | d247d25f | aliguori | freeaddrinfo(res); |
312 | d247d25f | aliguori | return -1; |
313 | d247d25f | aliguori | } |
314 | d247d25f | aliguori | |
315 | d247d25f | aliguori | #ifndef _WIN32
|
316 | d247d25f | aliguori | |
317 | d247d25f | aliguori | int unix_listen(const char *str, char *ostr, int olen) |
318 | d247d25f | aliguori | { |
319 | d247d25f | aliguori | struct sockaddr_un un;
|
320 | d247d25f | aliguori | char *path, *opts;
|
321 | d247d25f | aliguori | int sock, fd, len;
|
322 | d247d25f | aliguori | |
323 | d247d25f | aliguori | sock = socket(PF_UNIX, SOCK_STREAM, 0);
|
324 | d247d25f | aliguori | if (sock < 0) { |
325 | d247d25f | aliguori | perror("socket(unix)");
|
326 | d247d25f | aliguori | return -1; |
327 | d247d25f | aliguori | } |
328 | d247d25f | aliguori | |
329 | d247d25f | aliguori | opts = strchr(str, ',');
|
330 | d247d25f | aliguori | if (opts) {
|
331 | d247d25f | aliguori | len = opts - str; |
332 | d247d25f | aliguori | path = malloc(len+1);
|
333 | d247d25f | aliguori | snprintf(path, len+1, "%.*s", len, str); |
334 | d247d25f | aliguori | } else
|
335 | d247d25f | aliguori | path = strdup(str); |
336 | d247d25f | aliguori | |
337 | d247d25f | aliguori | memset(&un, 0, sizeof(un)); |
338 | d247d25f | aliguori | un.sun_family = AF_UNIX; |
339 | d247d25f | aliguori | if (path && strlen(path)) {
|
340 | d247d25f | aliguori | snprintf(un.sun_path, sizeof(un.sun_path), "%s", path); |
341 | d247d25f | aliguori | } else {
|
342 | d247d25f | aliguori | char *tmpdir = getenv("TMPDIR"); |
343 | d247d25f | aliguori | snprintf(un.sun_path, sizeof(un.sun_path), "%s/qemu-socket-XXXXXX", |
344 | d247d25f | aliguori | tmpdir ? tmpdir : "/tmp");
|
345 | d247d25f | aliguori | /*
|
346 | d247d25f | aliguori | * This dummy fd usage silences the mktemp() unsecure warning.
|
347 | d247d25f | aliguori | * Using mkstemp() doesn't make things more secure here
|
348 | d247d25f | aliguori | * though. bind() complains about existing files, so we have
|
349 | d247d25f | aliguori | * to unlink first and thus re-open the race window. The
|
350 | d247d25f | aliguori | * worst case possible is bind() failing, i.e. a DoS attack.
|
351 | d247d25f | aliguori | */
|
352 | d247d25f | aliguori | fd = mkstemp(un.sun_path); close(fd); |
353 | d247d25f | aliguori | } |
354 | d247d25f | aliguori | snprintf(ostr, olen, "%s%s", un.sun_path, opts ? opts : ""); |
355 | d247d25f | aliguori | |
356 | d247d25f | aliguori | unlink(un.sun_path); |
357 | d247d25f | aliguori | if (bind(sock, (struct sockaddr*) &un, sizeof(un)) < 0) { |
358 | d247d25f | aliguori | fprintf(stderr, "bind(unix:%s): %s\n", un.sun_path, strerror(errno));
|
359 | d247d25f | aliguori | goto err;
|
360 | d247d25f | aliguori | } |
361 | d247d25f | aliguori | if (listen(sock, 1) < 0) { |
362 | d247d25f | aliguori | fprintf(stderr, "listen(unix:%s): %s\n", un.sun_path, strerror(errno));
|
363 | d247d25f | aliguori | goto err;
|
364 | d247d25f | aliguori | } |
365 | d247d25f | aliguori | |
366 | d247d25f | aliguori | if (sockets_debug)
|
367 | d247d25f | aliguori | fprintf(stderr, "bind(unix:%s): OK\n", un.sun_path);
|
368 | d247d25f | aliguori | free(path); |
369 | d247d25f | aliguori | return sock;
|
370 | d247d25f | aliguori | |
371 | d247d25f | aliguori | err:
|
372 | d247d25f | aliguori | free(path); |
373 | d247d25f | aliguori | closesocket(sock); |
374 | d247d25f | aliguori | return -1; |
375 | d247d25f | aliguori | } |
376 | d247d25f | aliguori | |
377 | d247d25f | aliguori | int unix_connect(const char *path) |
378 | d247d25f | aliguori | { |
379 | d247d25f | aliguori | struct sockaddr_un un;
|
380 | d247d25f | aliguori | int sock;
|
381 | d247d25f | aliguori | |
382 | d247d25f | aliguori | sock = socket(PF_UNIX, SOCK_STREAM, 0);
|
383 | d247d25f | aliguori | if (sock < 0) { |
384 | d247d25f | aliguori | perror("socket(unix)");
|
385 | d247d25f | aliguori | return -1; |
386 | d247d25f | aliguori | } |
387 | d247d25f | aliguori | |
388 | d247d25f | aliguori | memset(&un, 0, sizeof(un)); |
389 | d247d25f | aliguori | un.sun_family = AF_UNIX; |
390 | d247d25f | aliguori | snprintf(un.sun_path, sizeof(un.sun_path), "%s", path); |
391 | d247d25f | aliguori | if (connect(sock, (struct sockaddr*) &un, sizeof(un)) < 0) { |
392 | d247d25f | aliguori | fprintf(stderr, "connect(unix:%s): %s\n", path, strerror(errno));
|
393 | d247d25f | aliguori | return -1; |
394 | d247d25f | aliguori | } |
395 | d247d25f | aliguori | |
396 | d247d25f | aliguori | if (sockets_debug)
|
397 | d247d25f | aliguori | fprintf(stderr, "connect(unix:%s): OK\n", path);
|
398 | d247d25f | aliguori | return sock;
|
399 | d247d25f | aliguori | } |
400 | d247d25f | aliguori | |
401 | d247d25f | aliguori | #else
|
402 | d247d25f | aliguori | |
403 | d247d25f | aliguori | int unix_listen(const char *path, char *ostr, int olen) |
404 | d247d25f | aliguori | { |
405 | d247d25f | aliguori | fprintf(stderr, "unix sockets are not available on windows\n");
|
406 | d247d25f | aliguori | return -1; |
407 | d247d25f | aliguori | } |
408 | d247d25f | aliguori | |
409 | d247d25f | aliguori | int unix_connect(const char *path) |
410 | d247d25f | aliguori | { |
411 | d247d25f | aliguori | fprintf(stderr, "unix sockets are not available on windows\n");
|
412 | d247d25f | aliguori | return -1; |
413 | d247d25f | aliguori | } |
414 | d247d25f | aliguori | |
415 | d247d25f | aliguori | #endif |