summaryrefslogtreecommitdiffstats
path: root/usr.sbin/bluetooth/bthidd/session.c
diff options
context:
space:
mode:
authoremax <emax@FreeBSD.org>2006-09-07 21:47:49 +0000
committeremax <emax@FreeBSD.org>2006-09-07 21:47:49 +0000
commit0a0c0e69a8b3191ef395c191f4667746e8f79f90 (patch)
tree2d993bc06bdf62dbb75a647125df678ab96cd3be /usr.sbin/bluetooth/bthidd/session.c
parentd04edc5790dab1504a82e507350c255aac246f8b (diff)
downloadFreeBSD-src-0a0c0e69a8b3191ef395c191f4667746e8f79f90.zip
FreeBSD-src-0a0c0e69a8b3191ef395c191f4667746e8f79f90.tar.gz
Update bthidd(8) code and hook it up to the build.
bthidd(8) now was integrated with vkbd(4) and supports multiple keyboards via vkbd(4)/kbdmux(4). The code was tested with Apple Bluetooth keyboard and SE k700i cell phone (remote control feature). MFC after: 1 month
Diffstat (limited to 'usr.sbin/bluetooth/bthidd/session.c')
-rw-r--r--usr.sbin/bluetooth/bthidd/session.c86
1 files changed, 63 insertions, 23 deletions
diff --git a/usr.sbin/bluetooth/bthidd/session.c b/usr.sbin/bluetooth/bthidd/session.c
index eefd98a..b9f331b 100644
--- a/usr.sbin/bluetooth/bthidd/session.c
+++ b/usr.sbin/bluetooth/bthidd/session.c
@@ -1,7 +1,9 @@
/*
* session.c
- *
- * Copyright (c) 2004 Maksim Yevmenkin <m_evmenkin@yahoo.com>
+ */
+
+/*-
+ * Copyright (c) 2006 Maksim Yevmenkin <m_evmenkin@yahoo.com>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
@@ -25,17 +27,22 @@
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*
- * $Id: session.c,v 1.2 2004/11/17 21:59:42 max Exp $
+ * $Id: session.c,v 1.3 2006/09/07 21:06:53 max Exp $
* $FreeBSD$
*/
#include <sys/queue.h>
#include <assert.h>
#include <bluetooth.h>
+#include <errno.h>
+#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
+#include <syslog.h>
#include <unistd.h>
+#include <usbhid.h>
+#include "bthid_config.h"
#include "bthidd.h"
#include "kbd.h"
@@ -44,27 +51,51 @@
*/
bthid_session_p
-session_open(bthid_server_p srv, bdaddr_p bdaddr)
+session_open(bthid_server_p srv, hid_device_p const d)
{
- bthid_session_p s = NULL;
+ bthid_session_p s;
assert(srv != NULL);
- assert(bdaddr != NULL);
-
- if ((s = (bthid_session_p) malloc(sizeof(*s))) != NULL) {
- s->srv = srv;
- memcpy(&s->bdaddr, bdaddr, sizeof(s->bdaddr));
- s->ctrl = -1;
- s->intr = -1;
- s->state = CLOSED;
- s->keys = bit_alloc(kbd_maxkey());
- if (s->keys == NULL) {
+ assert(d != NULL);
+
+ if ((s = (bthid_session_p) malloc(sizeof(*s))) == NULL)
+ return (NULL);
+
+ s->srv = srv;
+ memcpy(&s->bdaddr, &d->bdaddr, sizeof(s->bdaddr));
+ s->ctrl = -1;
+ s->intr = -1;
+
+ if (d->keyboard) {
+ /* Open /dev/vkbdctl */
+ s->vkbd = open("/dev/vkbdctl", O_RDWR);
+ if (s->vkbd < 0) {
+ syslog(LOG_ERR, "Could not open /dev/vkbdctl " \
+ "for %s. %s (%d)", bt_ntoa(&s->bdaddr, NULL),
+ strerror(errno), errno);
free(s);
- s = NULL;
- } else
- LIST_INSERT_HEAD(&srv->sessions, s, next);
+ return (NULL);
+ }
+ } else
+ s->vkbd = -1;
+
+ s->state = CLOSED;
+
+ s->keys1 = bit_alloc(kbd_maxkey());
+ if (s->keys1 == NULL) {
+ free(s);
+ return (NULL);
}
+ s->keys2 = bit_alloc(kbd_maxkey());
+ if (s->keys2 == NULL) {
+ free(s->keys1);
+ free(s);
+ return (NULL);
+ }
+
+ LIST_INSERT_HEAD(&srv->sessions, s, next);
+
return (s);
}
@@ -75,7 +106,7 @@ session_open(bthid_server_p srv, bdaddr_p bdaddr)
bthid_session_p
session_by_bdaddr(bthid_server_p srv, bdaddr_p bdaddr)
{
- bthid_session_p s = NULL;
+ bthid_session_p s;
assert(srv != NULL);
assert(bdaddr != NULL);
@@ -92,15 +123,15 @@ session_by_bdaddr(bthid_server_p srv, bdaddr_p bdaddr)
*/
bthid_session_p
-session_by_fd(bthid_server_p srv, int fd)
+session_by_fd(bthid_server_p srv, int32_t fd)
{
- bthid_session_p s = NULL;
+ bthid_session_p s;
assert(srv != NULL);
assert(fd >= 0);
LIST_FOREACH(s, &srv->sessions, next)
- if (s->ctrl == fd || s->intr == fd)
+ if (s->ctrl == fd || s->intr == fd || s->vkbd == fd)
break;
return (s);
@@ -136,7 +167,16 @@ session_close(bthid_session_p s)
s->srv->maxfd --;
}
- free(s->keys);
+ if (s->vkbd != -1) {
+ FD_CLR(s->vkbd, &s->srv->rfdset);
+ close(s->vkbd);
+
+ if (s->srv->maxfd == s->vkbd)
+ s->srv->maxfd --;
+ }
+
+ free(s->keys1);
+ free(s->keys2);
memset(s, 0, sizeof(*s));
free(s);
OpenPOWER on IntegriCloud