xref: /dragonfly/sys/netbt/sco_socket.c (revision 7d84b73d)
1 /*-
2  * Copyright (c) 2006 Itronix Inc.
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  * 3. The name of Itronix Inc. may not be used to endorse
14  *    or promote products derived from this software without specific
15  *    prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY ITRONIX INC. ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
20  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL ITRONIX INC. BE LIABLE FOR ANY
21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24  * ON ANY THEORY OF LIABILITY, WHETHER IN
25  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27  * POSSIBILITY OF SUCH DAMAGE.
28  *
29  * $OpenBSD: sco_socket.c,v 1.1 2007/06/01 02:46:12 uwe Exp $
30  * $NetBSD: sco_socket.c,v 1.9 2007/04/21 06:15:23 plunky Exp $
31  */
32 
33 /* load symbolic names */
34 #ifdef BLUETOOTH_DEBUG
35 #define PRUREQUESTS
36 #define PRCOREQUESTS
37 #endif
38 
39 #include <sys/param.h>
40 #include <sys/domain.h>
41 #include <sys/kernel.h>
42 #include <sys/malloc.h>	/* for M_NOWAIT */
43 #include <sys/mbuf.h>
44 #include <sys/proc.h>
45 #include <sys/protosw.h>
46 #include <sys/socket.h>
47 #include <sys/socketvar.h>
48 #include <sys/systm.h>
49 #include <sys/bus.h>
50 
51 #include <sys/msgport2.h>
52 
53 #include <netbt/bluetooth.h>
54 #include <netbt/hci.h>
55 #include <netbt/sco.h>
56 
57 /*******************************************************************************
58  *
59  * SCO SOCK_SEQPACKET sockets - low latency audio data
60  */
61 
62 static void sco_connecting(void *);
63 static void sco_connected(void *);
64 static void sco_disconnected(void *, int);
65 static void *sco_newconn(void *, struct sockaddr_bt *, struct sockaddr_bt *);
66 static void sco_complete(void *, int);
67 static void sco_linkmode(void *, int);
68 static void sco_input(void *, struct mbuf *);
69 
70 static const struct btproto sco_proto = {
71 	sco_connecting,
72 	sco_connected,
73 	sco_disconnected,
74 	sco_newconn,
75 	sco_complete,
76 	sco_linkmode,
77 	sco_input,
78 };
79 
80 int sco_sendspace = 4096;
81 int sco_recvspace = 4096;
82 
83 /*
84  * get/set socket options
85  */
86 void
87 sco_ctloutput(netmsg_t msg)
88 {
89 	struct socket *so = msg->ctloutput.base.nm_so;
90 	struct sockopt *sopt = msg->ctloutput.nm_sopt;
91 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
92 	struct mbuf *m;
93 	int err = 0;
94 
95 #ifdef notyet			/* XXX */
96 	DPRINTFN(2, "req %s\n", prcorequests[req]);
97 #endif
98 
99 	if (pcb == NULL) {
100 		err = EINVAL;
101 		goto out;
102 	}
103 
104 	if (sopt->sopt_level != BTPROTO_SCO) {
105 		err = ENOPROTOOPT;
106 		goto out;
107 	}
108 
109 	switch(sopt->sopt_dir) {
110 	case PRCO_GETOPT:
111 		m = m_get(M_WAITOK, MT_DATA);
112 		m->m_len = sco_getopt(pcb, sopt->sopt_name, mtod(m, uint8_t *));
113 		if (m->m_len == 0) {
114 			m_freem(m);
115 			m = NULL;
116 			err = ENOPROTOOPT;
117 		}
118 		/* *opt = m; */
119 		/* XXX There are possible memory leaks (Griffin) */
120 		soopt_from_kbuf(sopt, mtod(m, void *), m->m_len);
121 		break;
122 
123 	case PRCO_SETOPT:
124 		m = m_get(M_WAITOK, MT_DATA);
125 		err = soopt_to_kbuf(sopt, mtod(m,void*), m->m_len, m->m_len);
126 
127 		if (m->m_len == 0) {
128 			m_freem(m);
129 			m = NULL;
130 			err = EIO;
131 		}
132 
133 		err = sco_setopt(pcb, sopt->sopt_name, mtod(m, uint8_t *));
134 		m_freem(m);
135 		break;
136 
137 	default:
138 		err = ENOPROTOOPT;
139 		break;
140 	}
141 out:
142 	lwkt_replymsg(&msg->ctloutput.base.lmsg, err);
143 }
144 
145 /*****************************************************************************
146  *
147  *	SCO Protocol socket callbacks
148  *
149  */
150 static void
151 sco_connecting(void *arg)
152 {
153 	struct socket *so = arg;
154 
155 	DPRINTF("Connecting\n");
156 	soisconnecting(so);
157 }
158 
159 static void
160 sco_connected(void *arg)
161 {
162 	struct socket *so = arg;
163 
164 	DPRINTF("Connected\n");
165 	soisconnected(so);
166 }
167 
168 static void
169 sco_disconnected(void *arg, int err)
170 {
171 	struct socket *so = arg;
172 
173 	DPRINTF("Disconnected (%d)\n", err);
174 
175 	so->so_error = err;
176 	soisdisconnected(so);
177 }
178 
179 static void *
180 sco_newconn(void *arg, struct sockaddr_bt *laddr,
181     struct sockaddr_bt *raddr)
182 {
183 	struct socket *so = arg;
184 
185 	DPRINTF("New Connection\n");
186 	so = sonewconn(so, 0);
187 	if (so == NULL)
188 		return NULL;
189 
190 	soisconnecting(so);
191 	return so->so_pcb;
192 }
193 
194 static void
195 sco_complete(void *arg, int num)
196 {
197 	struct socket *so = arg;
198 
199 	while (num-- > 0)
200 		sbdroprecord(&so->so_snd.sb);
201 
202 	sowwakeup(so);
203 }
204 
205 static void
206 sco_linkmode(void *arg, int mode)
207 {
208 }
209 
210 static void
211 sco_input(void *arg, struct mbuf *m)
212 {
213 	struct socket *so = arg;
214 
215 	/*
216 	 * since this data is time sensitive, if the buffer
217 	 * is full we just dump data until the latest one
218 	 * will fit.
219 	 */
220 
221 	while (m->m_pkthdr.len > sbspace(&so->so_rcv))
222 		sbdroprecord(&so->so_rcv.sb);
223 
224 	DPRINTFN(10, "received %d bytes\n", m->m_pkthdr.len);
225 
226 	sbappendrecord(&so->so_rcv.sb, m);
227 	sorwakeup(so);
228 }
229 
230 /*
231  * Implementation of usrreqs.
232  */
233 static void
234 sco_sdetach(netmsg_t msg)
235 {
236 	struct socket *so = msg->detach.base.nm_so;
237 	int error;
238 
239 	error = sco_detach((struct sco_pcb **)&so->so_pcb);
240 	lwkt_replymsg(&msg->detach.base.lmsg, error);
241 }
242 
243 /*
244  * NOTE: (so) is referenced from soabort*() and netmsg_pru_abort()
245  *	 will sofree() it when we return.
246  */
247 static void
248 sco_sabort (netmsg_t msg)
249 {
250 	struct socket *so = msg->abort.base.nm_so;
251 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
252 
253 	sco_disconnect(pcb, 0);
254 	soisdisconnected(so);
255 	sco_sdetach(msg);
256 	/* msg invalid now */
257 }
258 
259 static void
260 sco_sdisconnect (netmsg_t msg)
261 {
262 	struct socket *so = msg->abort.base.nm_so;
263  	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
264 	int error;
265 
266 	soisdisconnecting(so);
267 
268 	error = sco_disconnect(pcb, so->so_linger);
269 	lwkt_replymsg(&msg->disconnect.base.lmsg, error);
270 }
271 
272 static void
273 sco_sattach(netmsg_t msg)
274 {
275 	struct socket *so = msg->attach.base.nm_so;
276 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
277 	int error;
278 
279 	if (pcb) {
280 		error = EINVAL;
281 	} else {
282 		error = soreserve(so, sco_sendspace, sco_recvspace,NULL);
283 		if (error == 0) {
284 			error = sco_attach((struct sco_pcb **)&so->so_pcb,
285 					   &sco_proto, so);
286 		}
287 	}
288 	lwkt_replymsg(&msg->attach.base.lmsg, error);
289 }
290 
291 static void
292 sco_sbind(netmsg_t msg)
293 {
294 	struct socket *so = msg->bind.base.nm_so;
295 	struct sockaddr *nam = msg->bind.nm_nam;
296 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
297 	struct sockaddr_bt *sa;
298 	int error;
299 
300 	KKASSERT(nam != NULL);
301 	sa = (struct sockaddr_bt *)nam;
302 
303 	if (sa->bt_len != sizeof(struct sockaddr_bt)) {
304 		error = EINVAL;
305 	} else if (sa->bt_family != AF_BLUETOOTH) {
306 		error = EAFNOSUPPORT;
307 	} else {
308 		error = sco_bind(pcb, sa);
309 	}
310 	lwkt_replymsg(&msg->bind.base.lmsg, error);
311 }
312 
313 static void
314 sco_sconnect(netmsg_t msg)
315 {
316 	struct socket *so = msg->connect.base.nm_so;
317 	struct sockaddr *nam = msg->connect.nm_nam;
318 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
319 	struct sockaddr_bt *sa;
320 	int error;
321 
322 	KKASSERT(nam != NULL);
323 	sa = (struct sockaddr_bt *)nam;
324 
325 	if (sa->bt_len != sizeof(struct sockaddr_bt)) {
326 		error = EINVAL;
327 	} else if (sa->bt_family != AF_BLUETOOTH) {
328 		error = EAFNOSUPPORT;
329 	} else {
330 		soisconnecting(so);
331 		error = sco_connect(pcb, sa);
332 	}
333 	lwkt_replymsg(&msg->connect.base.lmsg, error);
334 }
335 
336 static void
337 sco_speeraddr(netmsg_t msg)
338 {
339 	struct socket *so = msg->peeraddr.base.nm_so;
340 	struct sockaddr **nam = msg->peeraddr.nm_nam;
341 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
342 	struct sockaddr_bt *sa, ssa;
343 	int error;
344 
345 	sa = &ssa;
346 	bzero(sa, sizeof *sa);
347 	sa->bt_len = sizeof(struct sockaddr_bt);
348 	sa->bt_family = AF_BLUETOOTH;
349 	error = sco_peeraddr(pcb, sa);
350 	*nam = dup_sockaddr((struct sockaddr *)sa);
351 	lwkt_replymsg(&msg->peeraddr.base.lmsg, error);
352 }
353 
354 static void
355 sco_ssockaddr(netmsg_t msg)
356 {
357 	struct socket *so = msg->sockaddr.base.nm_so;
358 	struct sockaddr **nam = msg->sockaddr.nm_nam;
359 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
360 	struct sockaddr_bt *sa, ssa;
361 	int error;
362 
363 	sa = &ssa;
364 	bzero(sa, sizeof *sa);
365 	sa->bt_len = sizeof(struct sockaddr_bt);
366 	sa->bt_family = AF_BLUETOOTH;
367 	error = sco_sockaddr(pcb, sa);
368 	*nam = dup_sockaddr((struct sockaddr *)sa);
369 
370 	lwkt_replymsg(&msg->sockaddr.base.lmsg, error);
371 }
372 
373 static void
374 sco_sshutdown(netmsg_t msg)
375 {
376 	socantsendmore(msg->shutdown.base.nm_so);
377 	lwkt_replymsg(&msg->shutdown.base.lmsg, 0);
378 }
379 
380 static void
381 sco_ssend(netmsg_t msg)
382 {
383 	struct socket *so = msg->send.base.nm_so;
384 	struct mbuf *m = msg->send.nm_m;
385 	struct mbuf *control = msg->send.nm_control;
386 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
387 	struct mbuf *m0;
388 	int error = 0;
389 
390 	KKASSERT(m != NULL);
391 	if (m->m_pkthdr.len == 0)
392 		goto out;
393 
394 	if (m->m_pkthdr.len > pcb->sp_mtu) {
395 		error = EMSGSIZE;
396 		goto out;
397 	}
398 
399 	m0 = m_copym(m, 0, M_COPYALL, M_NOWAIT);
400 	if (m0 == NULL) {
401 		error = ENOMEM;
402 		goto out;
403 	}
404 
405 	/* no use for that */
406 	if (control) {
407 		m_freem(control);
408 		control = NULL;
409 	}
410 
411 	sbappendrecord(&so->so_snd.sb, m);
412 	error = sco_send(pcb, m0);
413 	m = NULL;
414 out:
415 	if (m)
416 		m_freem(m);
417 	if (control)
418 		m_freem(control);
419 	lwkt_replymsg(&msg->send.base.lmsg, error);
420 }
421 
422 static void
423 sco_saccept(netmsg_t msg)
424 {
425 	struct socket *so = msg->accept.base.nm_so;
426 	struct sockaddr **nam = msg->accept.nm_nam;
427 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
428 	struct sockaddr_bt *sa, ssa;
429 	int error;
430 
431 	sa = &ssa;
432 	bzero(sa, sizeof *sa);
433 	sa->bt_len = sizeof(struct sockaddr_bt);
434 	sa->bt_family = AF_BLUETOOTH;
435 	error = sco_peeraddr(pcb, sa);
436 	*nam = dup_sockaddr((struct sockaddr *)sa);
437 
438 	lwkt_replymsg(&msg->accept.base.lmsg, error);
439 }
440 
441 static void
442 sco_slisten(netmsg_t msg)
443 {
444 	struct socket *so = msg->listen.base.nm_so;
445 	struct sco_pcb *pcb = (struct sco_pcb *)so->so_pcb;
446 	int error;
447 
448 	error = sco_listen(pcb);
449 	lwkt_replymsg(&msg->accept.base.lmsg, error);
450 }
451 
452 struct pr_usrreqs sco_usrreqs = {
453         .pru_abort = sco_sabort,
454         .pru_accept = sco_saccept,
455         .pru_attach = sco_sattach,
456         .pru_bind = sco_sbind,
457         .pru_connect = sco_sconnect,
458         .pru_connect2 = pr_generic_notsupp,
459         .pru_control = pr_generic_notsupp,
460         .pru_detach = sco_sdetach,
461         .pru_disconnect = sco_sdisconnect,
462         .pru_listen = sco_slisten,
463         .pru_peeraddr = sco_speeraddr,
464         .pru_rcvd = pr_generic_notsupp,
465         .pru_rcvoob = pr_generic_notsupp,
466         .pru_send = sco_ssend,
467         .pru_sense = pru_sense_null,
468         .pru_shutdown = sco_sshutdown,
469         .pru_sockaddr = sco_ssockaddr,
470         .pru_sosend = sosend,
471         .pru_soreceive = soreceive
472 };
473