Annotation of sys/dev/bluetooth/btms.c, Revision 1.1
1.1 ! nbrk 1: /* $OpenBSD: btms.c,v 1.1 2007/07/27 16:52:24 gwk Exp $ */
! 2: /* $NetBSD: btms.c,v 1.6 2007/03/04 06:01:45 christos Exp $ */
! 3:
! 4: /*-
! 5: * Copyright (c) 2006 Itronix Inc.
! 6: * All rights reserved.
! 7: *
! 8: * Written by Iain Hibbert for Itronix Inc.
! 9: *
! 10: * Redistribution and use in source and binary forms, with or without
! 11: * modification, are permitted provided that the following conditions
! 12: * are met:
! 13: * 1. Redistributions of source code must retain the above copyright
! 14: * notice, this list of conditions and the following disclaimer.
! 15: * 2. Redistributions in binary form must reproduce the above copyright
! 16: * notice, this list of conditions and the following disclaimer in the
! 17: * documentation and/or other materials provided with the distribution.
! 18: * 3. The name of Itronix Inc. may not be used to endorse
! 19: * or promote products derived from this software without specific
! 20: * prior written permission.
! 21: *
! 22: * THIS SOFTWARE IS PROVIDED BY ITRONIX INC. ``AS IS'' AND
! 23: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
! 24: * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
! 25: * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL ITRONIX INC. BE LIABLE FOR ANY
! 26: * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
! 27: * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
! 28: * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
! 29: * ON ANY THEORY OF LIABILITY, WHETHER IN
! 30: * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
! 31: * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
! 32: * POSSIBILITY OF SUCH DAMAGE.
! 33: */
! 34:
! 35: /*
! 36: * based on dev/usb/ums.c
! 37: */
! 38:
! 39: #include <sys/param.h>
! 40: #include <sys/conf.h>
! 41: #include <sys/device.h>
! 42: #include <sys/proc.h>
! 43: #include <sys/systm.h>
! 44:
! 45: #include <netbt/bluetooth.h>
! 46:
! 47: #include <dev/bluetooth/bthid.h>
! 48: #include <dev/bluetooth/bthidev.h>
! 49:
! 50: #include <dev/usb/hid.h>
! 51: #include <dev/usb/usb.h>
! 52: #include <dev/usb/usbhid.h>
! 53:
! 54: #include <dev/wscons/wsconsio.h>
! 55: #include <dev/wscons/wsmousevar.h>
! 56:
! 57: #define MAX_BUTTONS 31
! 58: #define BUTTON(n) (1 << (((n) == 1 || (n) == 2) ? 3 - (n) : (n)))
! 59: #define NOTMOUSE(f) (((f) & (HIO_CONST | HIO_RELATIVE)) != HIO_RELATIVE)
! 60:
! 61: struct btms_softc {
! 62: struct bthidev sc_hidev; /* device+ */
! 63:
! 64: struct device *sc_wsmouse; /* child */
! 65: int sc_enabled;
! 66: uint16_t sc_flags;
! 67:
! 68: /* locators */
! 69: struct hid_location sc_loc_x;
! 70: struct hid_location sc_loc_y;
! 71: struct hid_location sc_loc_z;
! 72: struct hid_location sc_loc_w;
! 73: struct hid_location sc_loc_button[MAX_BUTTONS];
! 74:
! 75: int sc_num_buttons;
! 76: uint32_t sc_buttons;
! 77: };
! 78:
! 79: /* sc_flags */
! 80: #define BTMS_REVZ (1 << 0) /* reverse Z direction */
! 81: #define BTMS_HASZ (1 << 1) /* has Z direction */
! 82: #define BTMS_HASW (1 << 2) /* has W direction */
! 83:
! 84: int btms_match(struct device *, struct cfdata *, void *);
! 85: void btms_attach(struct device *, struct device *, void *);
! 86: int btms_detach(struct device *, int);
! 87:
! 88: struct cfdriver btms_cd = {
! 89: NULL, "btms", DV_DULL
! 90: };
! 91:
! 92: const struct cfattach btms_ca = {
! 93: sizeof(struct btms_softc),
! 94: btms_match,
! 95: btms_attach,
! 96: btms_detach,
! 97: };
! 98:
! 99: /* wsmouse(4) accessops */
! 100: int btms_enable(void *);
! 101: int btms_ioctl(void *, unsigned long, void *, int, struct lwp *);
! 102: void btms_disable(void *);
! 103:
! 104: const struct wsmouse_accessops btms_accessops = {
! 105: btms_enable,
! 106: btms_ioctl,
! 107: btms_disable,
! 108: };
! 109:
! 110: /* bthid methods */
! 111: void btms_input(struct bthidev *, uint8_t *, int);
! 112:
! 113:
! 114: int
! 115: btms_match(struct device *parent, struct cfdata *match, void *aux)
! 116: {
! 117: struct bthidev_attach_args *ba = aux;
! 118:
! 119: if (hid_is_collection(ba->ba_desc, ba->ba_dlen, ba->ba_id,
! 120: HID_USAGE2(HUP_GENERIC_DESKTOP, HUG_MOUSE)))
! 121: return 1;
! 122:
! 123: return 0;
! 124: }
! 125:
! 126: void
! 127: btms_attach(struct device *parent, struct device *self, void *aux)
! 128: {
! 129: struct btms_softc *sc = (struct btms_softc *)self;
! 130: struct bthidev_attach_args *ba = aux;
! 131: struct wsmousedev_attach_args wsma;
! 132: struct hid_location *zloc;
! 133: uint32_t flags;
! 134: int i, hl;
! 135:
! 136: ba->ba_input = btms_input;
! 137:
! 138: /* control the horizontal */
! 139: hl = hid_locate(ba->ba_desc, ba->ba_dlen,
! 140: HID_USAGE2(HUP_GENERIC_DESKTOP, HUG_X), ba->ba_id, hid_input,
! 141: &sc->sc_loc_x, &flags);
! 142:
! 143: if (hl == 0 || NOTMOUSE(flags)) {
! 144: printf("\n%s: X report 0x%04x not supported\n",
! 145: sc->sc_hidev.sc_dev.dv_xname, flags);
! 146:
! 147: return;
! 148: }
! 149:
! 150: /* control the vertical */
! 151: hl = hid_locate(ba->ba_desc, ba->ba_dlen,
! 152: HID_USAGE2(HUP_GENERIC_DESKTOP, HUG_Y), ba->ba_id, hid_input,
! 153: &sc->sc_loc_y, &flags);
! 154:
! 155: if (hl == 0 || NOTMOUSE(flags)) {
! 156: printf("\n%s: Y report 0x%04x not supported\n",
! 157: sc->sc_hidev.sc_dev.dv_xname, flags);
! 158:
! 159: return;
! 160: }
! 161:
! 162: /* Try the wheel first as the Z activator since it's tradition. */
! 163: hl = hid_locate(ba->ba_desc, ba->ba_dlen,
! 164: HID_USAGE2(HUP_GENERIC_DESKTOP, HUG_WHEEL), ba->ba_id, hid_input,
! 165: &sc->sc_loc_z, &flags);
! 166:
! 167: zloc = &sc->sc_loc_z;
! 168: if (hl) {
! 169: if (NOTMOUSE(flags)) {
! 170: printf("\n%s: Wheel report 0x%04x not supported\n",
! 171: sc->sc_hidev.sc_dev.dv_xname, flags);
! 172:
! 173: /* ignore Bad Z coord */
! 174: sc->sc_loc_z.size = 0;
! 175: } else {
! 176: sc->sc_flags |= BTMS_HASZ;
! 177: /* Wheels need the Z axis reversed. */
! 178: sc->sc_flags ^= BTMS_REVZ;
! 179: /* Put Z on the W coordinate */
! 180: zloc = &sc->sc_loc_w;
! 181: }
! 182: }
! 183:
! 184: hl = hid_locate(ba->ba_desc, ba->ba_dlen,
! 185: HID_USAGE2(HUP_GENERIC_DESKTOP, HUG_Z), ba->ba_id, hid_input,
! 186: zloc, &flags);
! 187:
! 188: /*
! 189: * The horizontal component of the scrollball can also be given by
! 190: * Application Control Pan in the Consumer page, so if we didnt see
! 191: * any Z then check that.
! 192: */
! 193: if (!hl) {
! 194: hl = hid_locate(ba->ba_desc, ba->ba_dlen,
! 195: HID_USAGE2(HUP_CONSUMER, HUC_AC_PAN), ba->ba_id, hid_input,
! 196: zloc, &flags);
! 197: }
! 198:
! 199: if (hl) {
! 200: if (NOTMOUSE(flags))
! 201: zloc->size = 0; /* ignore Z */
! 202: else {
! 203: if (sc->sc_flags & BTMS_HASZ)
! 204: sc->sc_flags |= BTMS_HASW;
! 205: else
! 206: sc->sc_flags |= BTMS_HASZ;
! 207: }
! 208: }
! 209:
! 210: for (i = 1 ; i <= MAX_BUTTONS ; i++) {
! 211: hl = hid_locate(ba->ba_desc, ba->ba_dlen,
! 212: HID_USAGE2(HUP_BUTTON, i), ba->ba_id, hid_input,
! 213: &sc->sc_loc_button[i - 1], NULL);
! 214:
! 215: if (hl == 0)
! 216: break;
! 217: }
! 218: sc->sc_num_buttons = i - 1;
! 219:
! 220: printf(": %d button%s%s%s%s.\n", sc->sc_num_buttons,
! 221: sc->sc_num_buttons == 1 ? "" : "s",
! 222: sc->sc_flags & BTMS_HASW ? ", W" : "",
! 223: sc->sc_flags & BTMS_HASZ ? " and Z dir" : "",
! 224: sc->sc_flags & BTMS_HASW ? "s" : "");
! 225:
! 226: wsma.accessops = &btms_accessops;
! 227: wsma.accesscookie = sc;
! 228:
! 229: sc->sc_wsmouse = config_found((struct device *)sc,
! 230: &wsma, wsmousedevprint);
! 231: }
! 232:
! 233: int
! 234: btms_detach(struct device *self, int flags)
! 235: {
! 236: struct btms_softc *sc = (struct btms_softc *)self;
! 237: int err = 0;
! 238:
! 239: if (sc->sc_wsmouse != NULL) {
! 240: err = config_detach(sc->sc_wsmouse, flags);
! 241: sc->sc_wsmouse = NULL;
! 242: }
! 243:
! 244: return err;
! 245: }
! 246:
! 247: int
! 248: btms_enable(void *self)
! 249: {
! 250: struct btms_softc *sc = (struct btms_softc *)self;
! 251:
! 252: if (sc->sc_enabled)
! 253: return EBUSY;
! 254:
! 255: sc->sc_enabled = 1;
! 256: return 0;
! 257: }
! 258:
! 259: int
! 260: btms_ioctl(void *self, unsigned long cmd, void *data, int flag, struct lwp *l)
! 261: {
! 262: /* struct btms_softc *sc = (struct btms_softc *)self; */
! 263:
! 264: switch (cmd) {
! 265: case WSMOUSEIO_GTYPE:
! 266: *(uint *)data = WSMOUSE_TYPE_BLUETOOTH;
! 267: break;
! 268:
! 269: default:
! 270: return EPASSTHROUGH;
! 271: }
! 272:
! 273: return 0;
! 274: }
! 275:
! 276: void
! 277: btms_disable(void *self)
! 278: {
! 279: struct btms_softc *sc = (struct btms_softc *)self;
! 280:
! 281: sc->sc_enabled = 0;
! 282: }
! 283:
! 284: void
! 285: btms_input(struct bthidev *self, uint8_t *data, int len)
! 286: {
! 287: struct btms_softc *sc = (struct btms_softc *)self;
! 288: int dx, dy, dz, dw;
! 289: uint32_t buttons;
! 290: int i, s;
! 291:
! 292: if (sc->sc_wsmouse == NULL || sc->sc_enabled == 0)
! 293: return;
! 294:
! 295: dx = hid_get_data(data, &sc->sc_loc_x);
! 296: dy = -hid_get_data(data, &sc->sc_loc_y);
! 297: dz = hid_get_data(data, &sc->sc_loc_z);
! 298: dw = hid_get_data(data, &sc->sc_loc_w);
! 299:
! 300: if (sc->sc_flags & BTMS_REVZ)
! 301: dz = -dz;
! 302:
! 303: buttons = 0;
! 304: for (i = 0 ; i < sc->sc_num_buttons ; i++)
! 305: if (hid_get_data(data, &sc->sc_loc_button[i]))
! 306: buttons |= BUTTON(i);
! 307:
! 308: if (dx != 0 || dy != 0 || dz != 0 || dw != 0 ||
! 309: buttons != sc->sc_buttons) {
! 310: sc->sc_buttons = buttons;
! 311:
! 312: s = spltty();
! 313: wsmouse_input(sc->sc_wsmouse, buttons, dx, dy, dz, dw,
! 314: WSMOUSE_INPUT_DELTA);
! 315: splx(s);
! 316: }
! 317: }
CVSweb