1 /*- 2 * Copyright (c) 1992, 1993 Erik Forsberg. 3 * Copyright (c) 1996, 1997 Kazutaka YOKOTA. 4 * All rights reserved. 5 * 6 * Redistribution and use in source and binary forms, with or without 7 * modification, are permitted provided that the following conditions 8 * are met: 9 * 1. Redistributions of source code must retain the above copyright 10 * notice, this list of conditions and the following disclaimer. 11 * 12 * THIS SOFTWARE IS PROVIDED BY ``AS IS'' AND ANY EXPRESS OR IMPLIED 13 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 14 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN 15 * NO EVENT SHALL I BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 16 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 17 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 18 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF 19 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING 20 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 21 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 22 * 23 * $FreeBSD$ 24 */ 25 26 /* 27 * Ported to 386bsd Oct 17, 1992 28 * Sandi Donno, Computer Science, University of Cape Town, South Africa 29 * Please send bug reports to sandi@cs.uct.ac.za 30 * 31 * Thanks are also due to Rick Macklem, rick@snowhite.cis.uoguelph.ca - 32 * although I was only partially successful in getting the alpha release 33 * of his "driver for the Logitech and ATI Inport Bus mice for use with 34 * 386bsd and the X386 port" to work with my Microsoft mouse, I nevertheless 35 * found his code to be an invaluable reference when porting this driver 36 * to 386bsd. 37 * 38 * Further modifications for latest 386BSD+patchkit and port to NetBSD, 39 * Andrew Herbert <andrew@werple.apana.org.au> - 8 June 1993 40 * 41 * Cloned from the Microsoft Bus Mouse driver, also by Erik Forsberg, by 42 * Andrew Herbert - 12 June 1993 43 * 44 * Modified for PS/2 mouse by Charles Hannum <mycroft@ai.mit.edu> 45 * - 13 June 1993 46 * 47 * Modified for PS/2 AUX mouse by Shoji Yuen <yuen@nuie.nagoya-u.ac.jp> 48 * - 24 October 1993 49 * 50 * Hardware access routines and probe logic rewritten by 51 * Kazutaka Yokota <yokota@zodiac.mech.utsunomiya-u.ac.jp> 52 * - 3, 14, 22 October 1996. 53 * - 12 November 1996. IOCTLs and rearranging `psmread', `psmioctl'... 54 * - 14, 30 November 1996. Uses `kbdio.c'. 55 * - 13 December 1996. Uses queuing version of `kbdio.c'. 56 * - January/February 1997. Tweaked probe logic for 57 * HiNote UltraII/Latitude/Armada laptops. 58 * - 30 July 1997. Added APM support. 59 * - 5 March 1997. Defined driver configuration flags (PSM_CONFIG_XXX). 60 * Improved sync check logic. 61 * Vendor specific support routines. 62 */ 63 64 #include "opt_psm.h" 65 66 #include <sys/param.h> 67 #include <sys/systm.h> 68 #include <sys/kernel.h> 69 #include <sys/module.h> 70 #include <sys/bus.h> 71 #include <sys/conf.h> 72 #include <sys/poll.h> 73 #include <sys/syslog.h> 74 #include <sys/malloc.h> 75 #include <machine/bus.h> 76 #include <sys/rman.h> 77 #include <sys/select.h> 78 #include <sys/uio.h> 79 80 #include <machine/clock.h> 81 #include <machine/limits.h> 82 #include <machine/mouse.h> 83 #include <machine/resource.h> 84 85 #include <isa/isavar.h> 86 #include <dev/kbd/atkbdcreg.h> 87 88 /* 89 * Driver specific options: the following options may be set by 90 * `options' statements in the kernel configuration file. 91 */ 92 93 /* debugging */ 94 #ifndef PSM_DEBUG 95 #define PSM_DEBUG 0 /* logging: 0: none, 1: brief, 2: verbose */ 96 #endif 97 98 /* features */ 99 100 /* #define PSM_HOOKRESUME hook the system resume event */ 101 /* #define PSM_RESETAFTERSUSPEND reset the device at the resume event */ 102 103 #ifdef PSM_HOOKAPM 104 #undef PSM_HOOKRESUME 105 #define PSM_HOOKRESUME 1 106 #endif /* PSM_HOOKAPM */ 107 108 #ifndef PSM_HOOKRESUME 109 #undef PSM_RESETAFTERSUSPEND 110 #endif /* PSM_HOOKRESUME */ 111 112 /* end of driver specific options */ 113 114 /* input queue */ 115 #define PSM_BUFSIZE 960 116 #define PSM_SMALLBUFSIZE 240 117 118 /* operation levels */ 119 #define PSM_LEVEL_BASE 0 120 #define PSM_LEVEL_STANDARD 1 121 #define PSM_LEVEL_NATIVE 2 122 #define PSM_LEVEL_MIN PSM_LEVEL_BASE 123 #define PSM_LEVEL_MAX PSM_LEVEL_NATIVE 124 125 /* Logitech PS2++ protocol */ 126 #define MOUSE_PS2PLUS_CHECKBITS(b) \ 127 ((((b[2] & 0x03) << 2) | 0x02) == (b[1] & 0x0f)) 128 #define MOUSE_PS2PLUS_PACKET_TYPE(b) \ 129 (((b[0] & 0x30) >> 2) | ((b[1] & 0x30) >> 4)) 130 131 /* some macros */ 132 #define PSM_UNIT(dev) (minor(dev) >> 1) 133 #define PSM_NBLOCKIO(dev) (minor(dev) & 1) 134 #define PSM_MKMINOR(unit,block) (((unit) << 1) | ((block) ? 0:1)) 135 136 #ifndef max 137 #define max(x,y) ((x) > (y) ? (x) : (y)) 138 #endif 139 #ifndef min 140 #define min(x,y) ((x) < (y) ? (x) : (y)) 141 #endif 142 143 #define abs(x) (((x) < 0) ? -(x) : (x)) 144 145 /* ring buffer */ 146 typedef struct ringbuf { 147 int count; /* # of valid elements in the buffer */ 148 int head; /* head pointer */ 149 int tail; /* tail poiner */ 150 unsigned char buf[PSM_BUFSIZE]; 151 } ringbuf_t; 152 153 /* driver control block */ 154 struct psm_softc { /* Driver status information */ 155 struct selinfo rsel; /* Process selecting for Input */ 156 unsigned char state; /* Mouse driver state */ 157 int config; /* driver configuration flags */ 158 int flags; /* other flags */ 159 KBDC kbdc; /* handle to access the keyboard controller */ 160 int addr; /* I/O port address */ 161 mousehw_t hw; /* hardware information */ 162 mousemode_t mode; /* operation mode */ 163 mousemode_t dflt_mode; /* default operation mode */ 164 mousestatus_t status; /* accumulated mouse movement */ 165 ringbuf_t queue; /* mouse status queue */ 166 unsigned char ipacket[16]; /* interim input buffer */ 167 int inputbytes; /* # of bytes in the input buffer */ 168 int button; /* the latest button state */ 169 int xold; /* previous absolute X position */ 170 int yold; /* previous absolute Y position */ 171 }; 172 devclass_t psm_devclass; 173 #define PSM_SOFTC(unit) ((struct psm_softc*)devclass_get_softc(psm_devclass, unit)) 174 175 /* driver state flags (state) */ 176 #define PSM_VALID 0x80 177 #define PSM_OPEN 1 /* Device is open */ 178 #define PSM_ASLP 2 /* Waiting for mouse data */ 179 180 /* driver configuration flags (config) */ 181 #define PSM_CONFIG_RESOLUTION 0x000f /* resolution */ 182 #define PSM_CONFIG_ACCEL 0x00f0 /* acceleration factor */ 183 #define PSM_CONFIG_NOCHECKSYNC 0x0100 /* disable sync. test */ 184 #define PSM_CONFIG_NOIDPROBE 0x0200 /* disable mouse model probe */ 185 #define PSM_CONFIG_NORESET 0x0400 /* don't reset the mouse */ 186 #define PSM_CONFIG_FORCETAP 0x0800 /* assume `tap' action exists */ 187 #define PSM_CONFIG_IGNPORTERROR 0x1000 /* ignore error in aux port test */ 188 189 #define PSM_CONFIG_FLAGS (PSM_CONFIG_RESOLUTION \ 190 | PSM_CONFIG_ACCEL \ 191 | PSM_CONFIG_NOCHECKSYNC \ 192 | PSM_CONFIG_NOIDPROBE \ 193 | PSM_CONFIG_NORESET \ 194 | PSM_CONFIG_FORCETAP \ 195 | PSM_CONFIG_IGNPORTERROR) 196 197 /* other flags (flags) */ 198 #define PSM_FLAGS_FINGERDOWN 0x0001 /* VersaPad finger down */ 199 200 /* for backward compatibility */ 201 #define OLD_MOUSE_GETHWINFO _IOR('M', 1, old_mousehw_t) 202 #define OLD_MOUSE_GETMODE _IOR('M', 2, old_mousemode_t) 203 #define OLD_MOUSE_SETMODE _IOW('M', 3, old_mousemode_t) 204 205 typedef struct old_mousehw { 206 int buttons; 207 int iftype; 208 int type; 209 int hwid; 210 } old_mousehw_t; 211 212 typedef struct old_mousemode { 213 int protocol; 214 int rate; 215 int resolution; 216 int accelfactor; 217 } old_mousemode_t; 218 219 /* packet formatting function */ 220 typedef int packetfunc_t __P((struct psm_softc *, unsigned char *, 221 int *, int, mousestatus_t *)); 222 223 /* function prototypes */ 224 static int psmprobe __P((device_t)); 225 static int psmattach __P((device_t)); 226 static int psmresume __P((device_t)); 227 228 static d_open_t psmopen; 229 static d_close_t psmclose; 230 static d_read_t psmread; 231 static d_ioctl_t psmioctl; 232 static d_poll_t psmpoll; 233 234 static int enable_aux_dev __P((KBDC)); 235 static int disable_aux_dev __P((KBDC)); 236 static int get_mouse_status __P((KBDC, int *, int, int)); 237 static int get_aux_id __P((KBDC)); 238 static int set_mouse_sampling_rate __P((KBDC, int)); 239 static int set_mouse_scaling __P((KBDC, int)); 240 static int set_mouse_resolution __P((KBDC, int)); 241 #ifdef PSM_RESETAFTERSUSPEND 242 static int set_mouse_mode __P((KBDC)); 243 #endif /* PSM_RESETAFTERSUSPEND */ 244 static int get_mouse_buttons __P((KBDC)); 245 static int is_a_mouse __P((int)); 246 static void recover_from_error __P((KBDC)); 247 static int restore_controller __P((KBDC, int)); 248 #ifdef PSM_RESETAFTERSUSPEND 249 static int reinitialize __P((int, mousemode_t *)); 250 #endif 251 static int doopen __P((int, int)); 252 static char *model_name(int); 253 static void psmintr(void*); 254 255 /* vendor specific features */ 256 typedef int probefunc_t __P((struct psm_softc *)); 257 258 static int mouse_id_proc1 __P((KBDC, int, int, int *)); 259 static probefunc_t enable_groller; 260 static probefunc_t enable_gmouse; 261 static probefunc_t enable_aglide; 262 static probefunc_t enable_kmouse; 263 static probefunc_t enable_msintelli; 264 static probefunc_t enable_mmanplus; 265 static probefunc_t enable_versapad; 266 static int tame_mouse __P((struct psm_softc *, mousestatus_t *, unsigned char *)); 267 268 static struct { 269 int model; 270 unsigned char syncmask; 271 int packetsize; 272 probefunc_t *probefunc; 273 } vendortype[] = { 274 { MOUSE_MODEL_NET, /* Genius NetMouse */ 275 0xc8, MOUSE_INTELLI_PACKETSIZE, enable_gmouse, }, 276 { MOUSE_MODEL_NETSCROLL, /* Genius NetScroll */ 277 0xc8, 6, enable_groller, }, 278 { MOUSE_MODEL_GLIDEPOINT, /* ALPS GlidePoint */ 279 0xc0, MOUSE_PS2_PACKETSIZE, enable_aglide, }, 280 { MOUSE_MODEL_MOUSEMANPLUS, /* Logitech MouseMan+ */ 281 0x08, MOUSE_PS2_PACKETSIZE, enable_mmanplus, }, 282 { MOUSE_MODEL_THINK, /* Kensignton ThinkingMouse */ 283 0x80, MOUSE_PS2_PACKETSIZE, enable_kmouse, }, 284 { MOUSE_MODEL_INTELLI, /* Microsoft IntelliMouse */ 285 0xc8, MOUSE_INTELLI_PACKETSIZE, enable_msintelli, }, 286 { MOUSE_MODEL_VERSAPAD, /* Interlink electronics VersaPad */ 287 0xe8, MOUSE_PS2VERSA_PACKETSIZE, enable_versapad, }, 288 { MOUSE_MODEL_GENERIC, 289 0xc0, MOUSE_PS2_PACKETSIZE, NULL, }, 290 }; 291 #define GENERIC_MOUSE_ENTRY 7 292 293 /* device driver declarateion */ 294 static device_method_t psm_methods[] = { 295 /* Device interface */ 296 DEVMETHOD(device_probe, psmprobe), 297 DEVMETHOD(device_attach, psmattach), 298 DEVMETHOD(device_resume, psmresume), 299 300 { 0, 0 } 301 }; 302 303 static driver_t psm_driver = { 304 "psm", 305 psm_methods, 306 sizeof(struct psm_softc), 307 }; 308 309 #define CDEV_MAJOR 21 310 311 static struct cdevsw psm_cdevsw = { 312 /* open */ psmopen, 313 /* close */ psmclose, 314 /* read */ psmread, 315 /* write */ nowrite, 316 /* ioctl */ psmioctl, 317 /* poll */ psmpoll, 318 /* mmap */ nommap, 319 /* strategy */ nostrategy, 320 /* name */ "psm", 321 /* maj */ CDEV_MAJOR, 322 /* dump */ nodump, 323 /* psize */ nopsize, 324 /* flags */ 0, 325 /* bmaj */ -1 326 }; 327 328 /* debug message level */ 329 static int verbose = PSM_DEBUG; 330 331 /* device I/O routines */ 332 static int 333 enable_aux_dev(KBDC kbdc) 334 { 335 int res; 336 337 res = send_aux_command(kbdc, PSMC_ENABLE_DEV); 338 if (verbose >= 2) 339 log(LOG_DEBUG, "psm: ENABLE_DEV return code:%04x\n", res); 340 341 return (res == PSM_ACK); 342 } 343 344 static int 345 disable_aux_dev(KBDC kbdc) 346 { 347 int res; 348 349 res = send_aux_command(kbdc, PSMC_DISABLE_DEV); 350 if (verbose >= 2) 351 log(LOG_DEBUG, "psm: DISABLE_DEV return code:%04x\n", res); 352 353 return (res == PSM_ACK); 354 } 355 356 static int 357 get_mouse_status(KBDC kbdc, int *status, int flag, int len) 358 { 359 int cmd; 360 int res; 361 int i; 362 363 switch (flag) { 364 case 0: 365 default: 366 cmd = PSMC_SEND_DEV_STATUS; 367 break; 368 case 1: 369 cmd = PSMC_SEND_DEV_DATA; 370 break; 371 } 372 empty_aux_buffer(kbdc, 5); 373 res = send_aux_command(kbdc, cmd); 374 if (verbose >= 2) 375 log(LOG_DEBUG, "psm: SEND_AUX_DEV_%s return code:%04x\n", 376 (flag == 1) ? "DATA" : "STATUS", res); 377 if (res != PSM_ACK) 378 return 0; 379 380 for (i = 0; i < len; ++i) { 381 status[i] = read_aux_data(kbdc); 382 if (status[i] < 0) 383 break; 384 } 385 386 if (verbose) { 387 log(LOG_DEBUG, "psm: %s %02x %02x %02x\n", 388 (flag == 1) ? "data" : "status", status[0], status[1], status[2]); 389 } 390 391 return i; 392 } 393 394 static int 395 get_aux_id(KBDC kbdc) 396 { 397 int res; 398 int id; 399 400 empty_aux_buffer(kbdc, 5); 401 res = send_aux_command(kbdc, PSMC_SEND_DEV_ID); 402 if (verbose >= 2) 403 log(LOG_DEBUG, "psm: SEND_DEV_ID return code:%04x\n", res); 404 if (res != PSM_ACK) 405 return (-1); 406 407 /* 10ms delay */ 408 DELAY(10000); 409 410 id = read_aux_data(kbdc); 411 if (verbose >= 2) 412 log(LOG_DEBUG, "psm: device ID: %04x\n", id); 413 414 return id; 415 } 416 417 static int 418 set_mouse_sampling_rate(KBDC kbdc, int rate) 419 { 420 int res; 421 422 res = send_aux_command_and_data(kbdc, PSMC_SET_SAMPLING_RATE, rate); 423 if (verbose >= 2) 424 log(LOG_DEBUG, "psm: SET_SAMPLING_RATE (%d) %04x\n", rate, res); 425 426 return ((res == PSM_ACK) ? rate : -1); 427 } 428 429 static int 430 set_mouse_scaling(KBDC kbdc, int scale) 431 { 432 int res; 433 434 switch (scale) { 435 case 1: 436 default: 437 scale = PSMC_SET_SCALING11; 438 break; 439 case 2: 440 scale = PSMC_SET_SCALING21; 441 break; 442 } 443 res = send_aux_command(kbdc, scale); 444 if (verbose >= 2) 445 log(LOG_DEBUG, "psm: SET_SCALING%s return code:%04x\n", 446 (scale == PSMC_SET_SCALING21) ? "21" : "11", res); 447 448 return (res == PSM_ACK); 449 } 450 451 /* `val' must be 0 through PSMD_MAX_RESOLUTION */ 452 static int 453 set_mouse_resolution(KBDC kbdc, int val) 454 { 455 int res; 456 457 res = send_aux_command_and_data(kbdc, PSMC_SET_RESOLUTION, val); 458 if (verbose >= 2) 459 log(LOG_DEBUG, "psm: SET_RESOLUTION (%d) %04x\n", val, res); 460 461 return ((res == PSM_ACK) ? val : -1); 462 } 463 464 #ifdef PSM_RESETAFTERSUSPEND 465 /* 466 * NOTE: once `set_mouse_mode()' is called, the mouse device must be 467 * re-enabled by calling `enable_aux_dev()' 468 */ 469 static int 470 set_mouse_mode(KBDC kbdc) 471 { 472 int res; 473 474 res = send_aux_command(kbdc, PSMC_SET_STREAM_MODE); 475 if (verbose >= 2) 476 log(LOG_DEBUG, "psm: SET_STREAM_MODE return code:%04x\n", res); 477 478 return (res == PSM_ACK); 479 } 480 #endif /* PSM_RESETAFTERSUSPEND */ 481 482 483 static int 484 get_mouse_buttons(KBDC kbdc) 485 { 486 int c = 2; /* assume two buttons by default */ 487 int status[3]; 488 489 /* 490 * NOTE: a special sequence to obtain Logitech Mouse specific 491 * information: set resolution to 25 ppi, set scaling to 1:1, set 492 * scaling to 1:1, set scaling to 1:1. Then the second byte of the 493 * mouse status bytes is the number of available buttons. 494 * Some manufactures also support this sequence. 495 */ 496 if (set_mouse_resolution(kbdc, PSMD_RES_LOW) != PSMD_RES_LOW) 497 return c; 498 if (set_mouse_scaling(kbdc, 1) && set_mouse_scaling(kbdc, 1) 499 && set_mouse_scaling(kbdc, 1) 500 && (get_mouse_status(kbdc, status, 0, 3) >= 3)) { 501 if (status[1] != 0) 502 return status[1]; 503 } 504 return c; 505 } 506 507 /* misc subroutines */ 508 /* 509 * Someday, I will get the complete list of valid pointing devices and 510 * their IDs... XXX 511 */ 512 static int 513 is_a_mouse(int id) 514 { 515 #if 0 516 static int valid_ids[] = { 517 PSM_MOUSE_ID, /* mouse */ 518 PSM_BALLPOINT_ID, /* ballpoint device */ 519 PSM_INTELLI_ID, /* Intellimouse */ 520 -1 /* end of table */ 521 }; 522 int i; 523 524 for (i = 0; valid_ids[i] >= 0; ++i) 525 if (valid_ids[i] == id) 526 return TRUE; 527 return FALSE; 528 #else 529 return TRUE; 530 #endif 531 } 532 533 static char * 534 model_name(int model) 535 { 536 static struct { 537 int model_code; 538 char *model_name; 539 } models[] = { 540 { MOUSE_MODEL_NETSCROLL, "NetScroll Mouse" }, 541 { MOUSE_MODEL_NET, "NetMouse" }, 542 { MOUSE_MODEL_GLIDEPOINT, "GlidePoint" }, 543 { MOUSE_MODEL_THINK, "ThinkingMouse" }, 544 { MOUSE_MODEL_INTELLI, "IntelliMouse" }, 545 { MOUSE_MODEL_MOUSEMANPLUS, "MouseMan+" }, 546 { MOUSE_MODEL_VERSAPAD, "VersaPad" }, 547 { MOUSE_MODEL_GENERIC, "Generic PS/2 mouse" }, 548 { MOUSE_MODEL_UNKNOWN, NULL }, 549 }; 550 int i; 551 552 for (i = 0; models[i].model_code != MOUSE_MODEL_UNKNOWN; ++i) { 553 if (models[i].model_code == model) 554 return models[i].model_name; 555 } 556 return "Unknown"; 557 } 558 559 static void 560 recover_from_error(KBDC kbdc) 561 { 562 /* discard anything left in the output buffer */ 563 empty_both_buffers(kbdc, 10); 564 565 #if 0 566 /* 567 * NOTE: KBDC_RESET_KBD may not restore the communication between the 568 * keyboard and the controller. 569 */ 570 reset_kbd(kbdc); 571 #else 572 /* 573 * NOTE: somehow diagnostic and keyboard port test commands bring the 574 * keyboard back. 575 */ 576 if (!test_controller(kbdc)) 577 log(LOG_ERR, "psm: keyboard controller failed.\n"); 578 /* if there isn't a keyboard in the system, the following error is OK */ 579 if (test_kbd_port(kbdc) != 0) { 580 if (verbose) 581 log(LOG_ERR, "psm: keyboard port failed.\n"); 582 } 583 #endif 584 } 585 586 static int 587 restore_controller(KBDC kbdc, int command_byte) 588 { 589 empty_both_buffers(kbdc, 10); 590 591 if (!set_controller_command_byte(kbdc, 0xff, command_byte)) { 592 log(LOG_ERR, "psm: failed to restore the keyboard controller " 593 "command byte.\n"); 594 return FALSE; 595 } else { 596 return TRUE; 597 } 598 } 599 600 #ifdef PSM_RESETAFTERSUSPEND 601 /* 602 * Re-initialize the aux port and device. The aux port must be enabled 603 * and its interrupt must be disabled before calling this routine. 604 * The aux device will be disabled before returning. 605 * The keyboard controller must be locked via `kbdc_lock()' before 606 * calling this routine. 607 */ 608 static int 609 reinitialize(int unit, mousemode_t *mode) 610 { 611 struct psm_softc *sc = PSM_SOFTC(unit); 612 KBDC kbdc = sc->kbdc; 613 int stat[3]; 614 int i; 615 616 switch((i = test_aux_port(kbdc))) { 617 case 1: /* ignore this error */ 618 case PSM_ACK: 619 if (verbose) 620 log(LOG_DEBUG, "psm%d: strange result for test aux port (%d).\n", 621 unit, i); 622 /* fall though */ 623 case 0: /* no error */ 624 break; 625 case -1: /* time out */ 626 default: /* error */ 627 recover_from_error(kbdc); 628 if (sc->config & PSM_CONFIG_IGNPORTERROR) 629 break; 630 log(LOG_ERR, "psm%d: the aux port is not functioning (%d).\n", 631 unit, i); 632 return FALSE; 633 } 634 635 if (sc->config & PSM_CONFIG_NORESET) { 636 /* 637 * Don't try to reset the pointing device. It may possibly be 638 * left in the unknown state, though... 639 */ 640 } else { 641 /* 642 * NOTE: some controllers appears to hang the `keyboard' when 643 * the aux port doesn't exist and `PSMC_RESET_DEV' is issued. 644 */ 645 if (!reset_aux_dev(kbdc)) { 646 recover_from_error(kbdc); 647 log(LOG_ERR, "psm%d: failed to reset the aux device.\n", unit); 648 return FALSE; 649 } 650 } 651 652 /* 653 * both the aux port and the aux device is functioning, see 654 * if the device can be enabled. 655 */ 656 if (!enable_aux_dev(kbdc) || !disable_aux_dev(kbdc)) { 657 log(LOG_ERR, "psm%d: failed to enable the aux device.\n", unit); 658 return FALSE; 659 } 660 empty_both_buffers(kbdc, 10); /* remove stray data if any */ 661 662 if (sc->config & PSM_CONFIG_NOIDPROBE) { 663 i = GENERIC_MOUSE_ENTRY; 664 } else { 665 /* FIXME: hardware ID, mouse buttons? */ 666 667 /* other parameters */ 668 for (i = 0; vendortype[i].probefunc != NULL; ++i) { 669 if ((*vendortype[i].probefunc)(sc)) { 670 if (verbose >= 2) 671 log(LOG_ERR, "psm%d: found %s\n", 672 unit, model_name(vendortype[i].model)); 673 break; 674 } 675 } 676 } 677 678 sc->hw.model = vendortype[i].model; 679 sc->mode.packetsize = vendortype[i].packetsize; 680 681 /* set mouse parameters */ 682 if (mode != (mousemode_t *)NULL) { 683 if (mode->rate > 0) 684 mode->rate = set_mouse_sampling_rate(kbdc, mode->rate); 685 if (mode->resolution >= 0) 686 mode->resolution = set_mouse_resolution(kbdc, mode->resolution); 687 set_mouse_scaling(kbdc, 1); 688 set_mouse_mode(kbdc); 689 } 690 691 /* request a data packet and extract sync. bits */ 692 if (get_mouse_status(kbdc, stat, 1, 3) < 3) { 693 log(LOG_DEBUG, "psm%d: failed to get data (reinitialize).\n", unit); 694 sc->mode.syncmask[0] = 0; 695 } else { 696 sc->mode.syncmask[1] = stat[0] & sc->mode.syncmask[0]; /* syncbits */ 697 /* the NetScroll Mouse will send three more bytes... Ignore them */ 698 empty_aux_buffer(kbdc, 5); 699 } 700 701 /* just check the status of the mouse */ 702 if (get_mouse_status(kbdc, stat, 0, 3) < 3) 703 log(LOG_DEBUG, "psm%d: failed to get status (reinitialize).\n", unit); 704 705 return TRUE; 706 } 707 #endif /* PSM_RESETAFTERSUSPEND */ 708 709 static int 710 doopen(int unit, int command_byte) 711 { 712 struct psm_softc *sc = PSM_SOFTC(unit); 713 int stat[3]; 714 715 /* enable the mouse device */ 716 if (!enable_aux_dev(sc->kbdc)) { 717 /* MOUSE ERROR: failed to enable the mouse because: 718 * 1) the mouse is faulty, 719 * 2) the mouse has been removed(!?) 720 * In the latter case, the keyboard may have hung, and need 721 * recovery procedure... 722 */ 723 recover_from_error(sc->kbdc); 724 #if 0 725 /* FIXME: we could reset the mouse here and try to enable 726 * it again. But it will take long time and it's not a good 727 * idea to disable the keyboard that long... 728 */ 729 if (!reinitialize(unit, &sc->mode) || !enable_aux_dev(sc->kbdc)) { 730 recover_from_error(sc->kbdc); 731 #else 732 { 733 #endif 734 restore_controller(sc->kbdc, command_byte); 735 /* mark this device is no longer available */ 736 sc->state &= ~PSM_VALID; 737 log(LOG_ERR, "psm%d: failed to enable the device (doopen).\n", 738 unit); 739 return (EIO); 740 } 741 } 742 743 if (get_mouse_status(sc->kbdc, stat, 0, 3) < 3) 744 log(LOG_DEBUG, "psm%d: failed to get status (doopen).\n", unit); 745 746 /* enable the aux port and interrupt */ 747 if (!set_controller_command_byte(sc->kbdc, 748 kbdc_get_device_mask(sc->kbdc), 749 (command_byte & KBD_KBD_CONTROL_BITS) 750 | KBD_ENABLE_AUX_PORT | KBD_ENABLE_AUX_INT)) { 751 /* CONTROLLER ERROR */ 752 disable_aux_dev(sc->kbdc); 753 restore_controller(sc->kbdc, command_byte); 754 log(LOG_ERR, "psm%d: failed to enable the aux interrupt (doopen).\n", 755 unit); 756 return (EIO); 757 } 758 759 return (0); 760 } 761 762 /* psm driver entry points */ 763 764 #define endprobe(v) { if (bootverbose) \ 765 --verbose; \ 766 kbdc_set_device_mask(sc->kbdc, mask); \ 767 kbdc_lock(sc->kbdc, FALSE); \ 768 free(sc, M_DEVBUF); \ 769 return (v); \ 770 } 771 772 static int 773 psmprobe(device_t dev) 774 { 775 int unit = device_get_unit(dev); 776 struct psm_softc *sc = device_get_softc(dev); 777 uintptr_t port; 778 uintptr_t flags; 779 int stat[3]; 780 int command_byte; 781 int mask; 782 int i; 783 784 #if 0 785 kbdc_debug(TRUE); 786 #endif 787 BUS_READ_IVAR(device_get_parent(dev), dev, KBDC_IVAR_PORT, &port); 788 BUS_READ_IVAR(device_get_parent(dev), dev, KBDC_IVAR_FLAGS, &flags); 789 790 sc->addr = port; 791 sc->kbdc = kbdc_open(sc->addr); 792 sc->config = flags & PSM_CONFIG_FLAGS; 793 sc->flags = 0; 794 if (bootverbose) 795 ++verbose; 796 797 device_set_desc(dev, "PS/2 Mouse"); 798 799 if (!kbdc_lock(sc->kbdc, TRUE)) { 800 printf("psm%d: unable to lock the controller.\n", unit); 801 if (bootverbose) 802 --verbose; 803 return (ENXIO); 804 } 805 806 /* 807 * NOTE: two bits in the command byte controls the operation of the 808 * aux port (mouse port): the aux port disable bit (bit 5) and the aux 809 * port interrupt (IRQ 12) enable bit (bit 2). 810 */ 811 812 /* discard anything left after the keyboard initialization */ 813 empty_both_buffers(sc->kbdc, 10); 814 815 /* save the current command byte; it will be used later */ 816 mask = kbdc_get_device_mask(sc->kbdc) & ~KBD_AUX_CONTROL_BITS; 817 command_byte = get_controller_command_byte(sc->kbdc); 818 if (verbose) 819 printf("psm%d: current command byte:%04x\n", unit, command_byte); 820 if (command_byte == -1) { 821 /* CONTROLLER ERROR */ 822 printf("psm%d: unable to get the current command byte value.\n", 823 unit); 824 endprobe(ENXIO); 825 } 826 827 /* 828 * disable the keyboard port while probing the aux port, which must be 829 * enabled during this routine 830 */ 831 if (!set_controller_command_byte(sc->kbdc, 832 KBD_KBD_CONTROL_BITS | KBD_AUX_CONTROL_BITS, 833 KBD_DISABLE_KBD_PORT | KBD_DISABLE_KBD_INT 834 | KBD_ENABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 835 /* 836 * this is CONTROLLER ERROR; I don't know how to recover 837 * from this error... 838 */ 839 restore_controller(sc->kbdc, command_byte); 840 printf("psm%d: unable to set the command byte.\n", unit); 841 endprobe(ENXIO); 842 } 843 write_controller_command(sc->kbdc, KBDC_ENABLE_AUX_PORT); 844 845 /* 846 * NOTE: `test_aux_port()' is designed to return with zero if the aux 847 * port exists and is functioning. However, some controllers appears 848 * to respond with zero even when the aux port doesn't exist. (It may 849 * be that this is only the case when the controller DOES have the aux 850 * port but the port is not wired on the motherboard.) The keyboard 851 * controllers without the port, such as the original AT, are 852 * supporsed to return with an error code or simply time out. In any 853 * case, we have to continue probing the port even when the controller 854 * passes this test. 855 * 856 * XXX: some controllers erroneously return the error code 1 when 857 * it has the perfectly functional aux port. We have to ignore this 858 * error code. Even if the controller HAS error with the aux port, 859 * it will be detected later... 860 * XXX: another incompatible controller returns PSM_ACK (0xfa)... 861 */ 862 switch ((i = test_aux_port(sc->kbdc))) { 863 case 1: /* ignore this error */ 864 case PSM_ACK: 865 if (verbose) 866 printf("psm%d: strange result for test aux port (%d).\n", 867 unit, i); 868 /* fall though */ 869 case 0: /* no error */ 870 break; 871 case -1: /* time out */ 872 default: /* error */ 873 recover_from_error(sc->kbdc); 874 if (sc->config & PSM_CONFIG_IGNPORTERROR) 875 break; 876 restore_controller(sc->kbdc, command_byte); 877 if (verbose) 878 printf("psm%d: the aux port is not functioning (%d).\n", 879 unit, i); 880 endprobe(ENXIO); 881 } 882 883 if (sc->config & PSM_CONFIG_NORESET) { 884 /* 885 * Don't try to reset the pointing device. It may possibly be 886 * left in the unknown state, though... 887 */ 888 } else { 889 /* 890 * NOTE: some controllers appears to hang the `keyboard' when the aux 891 * port doesn't exist and `PSMC_RESET_DEV' is issued. 892 */ 893 if (!reset_aux_dev(sc->kbdc)) { 894 recover_from_error(sc->kbdc); 895 restore_controller(sc->kbdc, command_byte); 896 if (verbose) 897 printf("psm%d: failed to reset the aux device.\n", unit); 898 endprobe(ENXIO); 899 } 900 } 901 902 /* 903 * both the aux port and the aux device is functioning, see if the 904 * device can be enabled. NOTE: when enabled, the device will start 905 * sending data; we shall immediately disable the device once we know 906 * the device can be enabled. 907 */ 908 if (!enable_aux_dev(sc->kbdc) || !disable_aux_dev(sc->kbdc)) { 909 /* MOUSE ERROR */ 910 recover_from_error(sc->kbdc); 911 restore_controller(sc->kbdc, command_byte); 912 if (verbose) 913 printf("psm%d: failed to enable the aux device.\n", unit); 914 endprobe(ENXIO); 915 } 916 917 /* save the default values after reset */ 918 if (get_mouse_status(sc->kbdc, stat, 0, 3) >= 3) { 919 sc->dflt_mode.rate = sc->mode.rate = stat[2]; 920 sc->dflt_mode.resolution = sc->mode.resolution = stat[1]; 921 } else { 922 sc->dflt_mode.rate = sc->mode.rate = -1; 923 sc->dflt_mode.resolution = sc->mode.resolution = -1; 924 } 925 926 /* hardware information */ 927 sc->hw.iftype = MOUSE_IF_PS2; 928 929 /* verify the device is a mouse */ 930 sc->hw.hwid = get_aux_id(sc->kbdc); 931 if (!is_a_mouse(sc->hw.hwid)) { 932 restore_controller(sc->kbdc, command_byte); 933 if (verbose) 934 printf("psm%d: unknown device type (%d).\n", unit, sc->hw.hwid); 935 endprobe(ENXIO); 936 } 937 switch (sc->hw.hwid) { 938 case PSM_BALLPOINT_ID: 939 sc->hw.type = MOUSE_TRACKBALL; 940 break; 941 case PSM_MOUSE_ID: 942 case PSM_INTELLI_ID: 943 sc->hw.type = MOUSE_MOUSE; 944 break; 945 default: 946 sc->hw.type = MOUSE_UNKNOWN; 947 break; 948 } 949 950 if (sc->config & PSM_CONFIG_NOIDPROBE) { 951 sc->hw.buttons = 2; 952 i = GENERIC_MOUSE_ENTRY; 953 } else { 954 /* # of buttons */ 955 sc->hw.buttons = get_mouse_buttons(sc->kbdc); 956 957 /* other parameters */ 958 for (i = 0; vendortype[i].probefunc != NULL; ++i) { 959 if ((*vendortype[i].probefunc)(sc)) { 960 if (verbose >= 2) 961 printf("psm%d: found %s\n", 962 unit, model_name(vendortype[i].model)); 963 break; 964 } 965 } 966 } 967 968 sc->hw.model = vendortype[i].model; 969 970 sc->dflt_mode.level = PSM_LEVEL_BASE; 971 sc->dflt_mode.packetsize = MOUSE_PS2_PACKETSIZE; 972 sc->dflt_mode.accelfactor = (sc->config & PSM_CONFIG_ACCEL) >> 4; 973 if (sc->config & PSM_CONFIG_NOCHECKSYNC) 974 sc->dflt_mode.syncmask[0] = 0; 975 else 976 sc->dflt_mode.syncmask[0] = vendortype[i].syncmask; 977 if (sc->config & PSM_CONFIG_FORCETAP) 978 sc->mode.syncmask[0] &= ~MOUSE_PS2_TAP; 979 sc->dflt_mode.syncmask[1] = 0; /* syncbits */ 980 sc->mode = sc->dflt_mode; 981 sc->mode.packetsize = vendortype[i].packetsize; 982 983 /* set mouse parameters */ 984 #if 0 985 /* 986 * A version of Logitech FirstMouse+ won't report wheel movement, 987 * if SET_DEFAULTS is sent... Don't use this command. 988 * This fix was found by Takashi Nishida. 989 */ 990 i = send_aux_command(sc->kbdc, PSMC_SET_DEFAULTS); 991 if (verbose >= 2) 992 printf("psm%d: SET_DEFAULTS return code:%04x\n", unit, i); 993 #endif 994 if (sc->config & PSM_CONFIG_RESOLUTION) { 995 sc->mode.resolution 996 = set_mouse_resolution(sc->kbdc, 997 (sc->config & PSM_CONFIG_RESOLUTION) - 1); 998 } else if (sc->mode.resolution >= 0) { 999 sc->mode.resolution 1000 = set_mouse_resolution(sc->kbdc, sc->dflt_mode.resolution); 1001 } 1002 if (sc->mode.rate > 0) { 1003 sc->mode.rate = set_mouse_sampling_rate(sc->kbdc, sc->dflt_mode.rate); 1004 } 1005 set_mouse_scaling(sc->kbdc, 1); 1006 1007 /* request a data packet and extract sync. bits */ 1008 if (get_mouse_status(sc->kbdc, stat, 1, 3) < 3) { 1009 printf("psm%d: failed to get data.\n", unit); 1010 sc->mode.syncmask[0] = 0; 1011 } else { 1012 sc->mode.syncmask[1] = stat[0] & sc->mode.syncmask[0]; /* syncbits */ 1013 /* the NetScroll Mouse will send three more bytes... Ignore them */ 1014 empty_aux_buffer(sc->kbdc, 5); 1015 } 1016 1017 /* just check the status of the mouse */ 1018 /* 1019 * NOTE: XXX there are some arcane controller/mouse combinations out 1020 * there, which hung the controller unless there is data transmission 1021 * after ACK from the mouse. 1022 */ 1023 if (get_mouse_status(sc->kbdc, stat, 0, 3) < 3) { 1024 printf("psm%d: failed to get status.\n", unit); 1025 } else { 1026 /* 1027 * When in its native mode, some mice operate with different 1028 * default parameters than in the PS/2 compatible mode. 1029 */ 1030 sc->dflt_mode.rate = sc->mode.rate = stat[2]; 1031 sc->dflt_mode.resolution = sc->mode.resolution = stat[1]; 1032 } 1033 1034 /* disable the aux port for now... */ 1035 if (!set_controller_command_byte(sc->kbdc, 1036 KBD_KBD_CONTROL_BITS | KBD_AUX_CONTROL_BITS, 1037 (command_byte & KBD_KBD_CONTROL_BITS) 1038 | KBD_DISABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 1039 /* 1040 * this is CONTROLLER ERROR; I don't know the proper way to 1041 * recover from this error... 1042 */ 1043 restore_controller(sc->kbdc, command_byte); 1044 printf("psm%d: unable to set the command byte.\n", unit); 1045 endprobe(ENXIO); 1046 } 1047 1048 /* done */ 1049 kbdc_set_device_mask(sc->kbdc, mask | KBD_AUX_CONTROL_BITS); 1050 kbdc_lock(sc->kbdc, FALSE); 1051 return (0); 1052 } 1053 1054 static int 1055 psmattach(device_t dev) 1056 { 1057 int unit = device_get_unit(dev); 1058 struct psm_softc *sc = device_get_softc(dev); 1059 void *ih; 1060 struct resource *res; 1061 uintptr_t irq; 1062 int zero = 0; 1063 1064 if (sc == NULL) /* shouldn't happen */ 1065 return (ENXIO); 1066 1067 /* Setup initial state */ 1068 sc->state = PSM_VALID; 1069 1070 /* Done */ 1071 make_dev(&psm_cdevsw, PSM_MKMINOR(unit, FALSE), 0, 0, 0666, "psm%d", unit); 1072 make_dev(&psm_cdevsw, PSM_MKMINOR(unit, TRUE), 0, 0, 0666, "bpsm%d", unit); 1073 1074 if (!verbose) { 1075 printf("psm%d: model %s, device ID %d\n", 1076 unit, model_name(sc->hw.model), sc->hw.hwid & 0x00ff); 1077 } else { 1078 printf("psm%d: model %s, device ID %d-%02x, %d buttons\n", 1079 unit, model_name(sc->hw.model), 1080 sc->hw.hwid & 0x00ff, sc->hw.hwid >> 8, sc->hw.buttons); 1081 printf("psm%d: config:%08x, flags:%08x, packet size:%d\n", 1082 unit, sc->config, sc->flags, sc->mode.packetsize); 1083 printf("psm%d: syncmask:%02x, syncbits:%02x\n", 1084 unit, sc->mode.syncmask[0], sc->mode.syncmask[1]); 1085 } 1086 1087 if (bootverbose) 1088 --verbose; 1089 1090 BUS_READ_IVAR(device_get_parent(dev), dev, KBDC_IVAR_IRQ, &irq); 1091 res = bus_alloc_resource(dev, SYS_RES_IRQ, &zero, irq, irq, 1, 1092 RF_SHAREABLE | RF_ACTIVE); 1093 BUS_SETUP_INTR(device_get_parent(dev), dev, res, INTR_TYPE_TTY, 1094 psmintr, sc, &ih); 1095 1096 return (0); 1097 } 1098 1099 static int 1100 psmopen(dev_t dev, int flag, int fmt, struct proc *p) 1101 { 1102 int unit = PSM_UNIT(dev); 1103 struct psm_softc *sc; 1104 int command_byte; 1105 int err; 1106 int s; 1107 1108 /* Get device data */ 1109 sc = PSM_SOFTC(unit); 1110 if ((sc == NULL) || (sc->state & PSM_VALID) == 0) 1111 /* the device is no longer valid/functioning */ 1112 return (ENXIO); 1113 1114 /* Disallow multiple opens */ 1115 if (sc->state & PSM_OPEN) 1116 return (EBUSY); 1117 1118 device_busy(devclass_get_device(psm_devclass, unit)); 1119 1120 /* Initialize state */ 1121 sc->rsel.si_flags = 0; 1122 sc->rsel.si_pid = 0; 1123 sc->mode.level = sc->dflt_mode.level; 1124 sc->mode.protocol = sc->dflt_mode.protocol; 1125 1126 /* flush the event queue */ 1127 sc->queue.count = 0; 1128 sc->queue.head = 0; 1129 sc->queue.tail = 0; 1130 sc->status.flags = 0; 1131 sc->status.button = 0; 1132 sc->status.obutton = 0; 1133 sc->status.dx = 0; 1134 sc->status.dy = 0; 1135 sc->status.dz = 0; 1136 sc->button = 0; 1137 1138 /* empty input buffer */ 1139 bzero(sc->ipacket, sizeof(sc->ipacket)); 1140 sc->inputbytes = 0; 1141 1142 /* don't let timeout routines in the keyboard driver to poll the kbdc */ 1143 if (!kbdc_lock(sc->kbdc, TRUE)) 1144 return (EIO); 1145 1146 /* save the current controller command byte */ 1147 s = spltty(); 1148 command_byte = get_controller_command_byte(sc->kbdc); 1149 1150 /* enable the aux port and temporalily disable the keyboard */ 1151 if ((command_byte == -1) 1152 || !set_controller_command_byte(sc->kbdc, 1153 kbdc_get_device_mask(sc->kbdc), 1154 KBD_DISABLE_KBD_PORT | KBD_DISABLE_KBD_INT 1155 | KBD_ENABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 1156 /* CONTROLLER ERROR; do you know how to get out of this? */ 1157 kbdc_lock(sc->kbdc, FALSE); 1158 splx(s); 1159 log(LOG_ERR, "psm%d: unable to set the command byte (psmopen).\n", 1160 unit); 1161 return (EIO); 1162 } 1163 /* 1164 * Now that the keyboard controller is told not to generate 1165 * the keyboard and mouse interrupts, call `splx()' to allow 1166 * the other tty interrupts. The clock interrupt may also occur, 1167 * but timeout routines will be blocked by the poll flag set 1168 * via `kbdc_lock()' 1169 */ 1170 splx(s); 1171 1172 /* enable the mouse device */ 1173 err = doopen(unit, command_byte); 1174 1175 /* done */ 1176 if (err == 0) 1177 sc->state |= PSM_OPEN; 1178 kbdc_lock(sc->kbdc, FALSE); 1179 return (err); 1180 } 1181 1182 static int 1183 psmclose(dev_t dev, int flag, int fmt, struct proc *p) 1184 { 1185 int unit = PSM_UNIT(dev); 1186 struct psm_softc *sc = PSM_SOFTC(unit); 1187 int stat[3]; 1188 int command_byte; 1189 int s; 1190 1191 /* don't let timeout routines in the keyboard driver to poll the kbdc */ 1192 if (!kbdc_lock(sc->kbdc, TRUE)) 1193 return (EIO); 1194 1195 /* save the current controller command byte */ 1196 s = spltty(); 1197 command_byte = get_controller_command_byte(sc->kbdc); 1198 if (command_byte == -1) { 1199 kbdc_lock(sc->kbdc, FALSE); 1200 splx(s); 1201 return (EIO); 1202 } 1203 1204 /* disable the aux interrupt and temporalily disable the keyboard */ 1205 if (!set_controller_command_byte(sc->kbdc, 1206 kbdc_get_device_mask(sc->kbdc), 1207 KBD_DISABLE_KBD_PORT | KBD_DISABLE_KBD_INT 1208 | KBD_ENABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 1209 log(LOG_ERR, "psm%d: failed to disable the aux int (psmclose).\n", 1210 PSM_UNIT(dev)); 1211 /* CONTROLLER ERROR; 1212 * NOTE: we shall force our way through. Because the only 1213 * ill effect we shall see is that we may not be able 1214 * to read ACK from the mouse, and it doesn't matter much 1215 * so long as the mouse will accept the DISABLE command. 1216 */ 1217 } 1218 splx(s); 1219 1220 /* remove anything left in the output buffer */ 1221 empty_aux_buffer(sc->kbdc, 10); 1222 1223 /* disable the aux device, port and interrupt */ 1224 if (sc->state & PSM_VALID) { 1225 if (!disable_aux_dev(sc->kbdc)) { 1226 /* MOUSE ERROR; 1227 * NOTE: we don't return error and continue, pretending 1228 * we have successfully disabled the device. It's OK because 1229 * the interrupt routine will discard any data from the mouse 1230 * hereafter. 1231 */ 1232 log(LOG_ERR, "psm%d: failed to disable the device (psmclose).\n", 1233 PSM_UNIT(dev)); 1234 } 1235 1236 if (get_mouse_status(sc->kbdc, stat, 0, 3) < 3) 1237 log(LOG_DEBUG, "psm%d: failed to get status (psmclose).\n", 1238 PSM_UNIT(dev)); 1239 } 1240 1241 if (!set_controller_command_byte(sc->kbdc, 1242 kbdc_get_device_mask(sc->kbdc), 1243 (command_byte & KBD_KBD_CONTROL_BITS) 1244 | KBD_DISABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 1245 /* CONTROLLER ERROR; 1246 * we shall ignore this error; see the above comment. 1247 */ 1248 log(LOG_ERR, "psm%d: failed to disable the aux port (psmclose).\n", 1249 PSM_UNIT(dev)); 1250 } 1251 1252 /* remove anything left in the output buffer */ 1253 empty_aux_buffer(sc->kbdc, 10); 1254 1255 /* close is almost always successful */ 1256 sc->state &= ~PSM_OPEN; 1257 kbdc_lock(sc->kbdc, FALSE); 1258 device_unbusy(devclass_get_device(psm_devclass, unit)); 1259 return (0); 1260 } 1261 1262 static int 1263 tame_mouse(struct psm_softc *sc, mousestatus_t *status, unsigned char *buf) 1264 { 1265 static unsigned char butmapps2[8] = { 1266 0, 1267 MOUSE_PS2_BUTTON1DOWN, 1268 MOUSE_PS2_BUTTON2DOWN, 1269 MOUSE_PS2_BUTTON1DOWN | MOUSE_PS2_BUTTON2DOWN, 1270 MOUSE_PS2_BUTTON3DOWN, 1271 MOUSE_PS2_BUTTON1DOWN | MOUSE_PS2_BUTTON3DOWN, 1272 MOUSE_PS2_BUTTON2DOWN | MOUSE_PS2_BUTTON3DOWN, 1273 MOUSE_PS2_BUTTON1DOWN | MOUSE_PS2_BUTTON2DOWN | MOUSE_PS2_BUTTON3DOWN, 1274 }; 1275 static unsigned char butmapmsc[8] = { 1276 MOUSE_MSC_BUTTON1UP | MOUSE_MSC_BUTTON2UP | MOUSE_MSC_BUTTON3UP, 1277 MOUSE_MSC_BUTTON2UP | MOUSE_MSC_BUTTON3UP, 1278 MOUSE_MSC_BUTTON1UP | MOUSE_MSC_BUTTON3UP, 1279 MOUSE_MSC_BUTTON3UP, 1280 MOUSE_MSC_BUTTON1UP | MOUSE_MSC_BUTTON2UP, 1281 MOUSE_MSC_BUTTON2UP, 1282 MOUSE_MSC_BUTTON1UP, 1283 0, 1284 }; 1285 int mapped; 1286 int i; 1287 1288 if (sc->mode.level == PSM_LEVEL_BASE) { 1289 mapped = status->button & ~MOUSE_BUTTON4DOWN; 1290 if (status->button & MOUSE_BUTTON4DOWN) 1291 mapped |= MOUSE_BUTTON1DOWN; 1292 status->button = mapped; 1293 buf[0] = MOUSE_PS2_SYNC | butmapps2[mapped & MOUSE_STDBUTTONS]; 1294 i = max(min(status->dx, 255), -256); 1295 if (i < 0) 1296 buf[0] |= MOUSE_PS2_XNEG; 1297 buf[1] = i; 1298 i = max(min(status->dy, 255), -256); 1299 if (i < 0) 1300 buf[0] |= MOUSE_PS2_YNEG; 1301 buf[2] = i; 1302 return MOUSE_PS2_PACKETSIZE; 1303 } else if (sc->mode.level == PSM_LEVEL_STANDARD) { 1304 buf[0] = MOUSE_MSC_SYNC | butmapmsc[status->button & MOUSE_STDBUTTONS]; 1305 i = max(min(status->dx, 255), -256); 1306 buf[1] = i >> 1; 1307 buf[3] = i - buf[1]; 1308 i = max(min(status->dy, 255), -256); 1309 buf[2] = i >> 1; 1310 buf[4] = i - buf[2]; 1311 i = max(min(status->dz, 127), -128); 1312 buf[5] = (i >> 1) & 0x7f; 1313 buf[6] = (i - (i >> 1)) & 0x7f; 1314 buf[7] = (~status->button >> 3) & 0x7f; 1315 return MOUSE_SYS_PACKETSIZE; 1316 } 1317 return sc->inputbytes;; 1318 } 1319 1320 static int 1321 psmread(dev_t dev, struct uio *uio, int flag) 1322 { 1323 register struct psm_softc *sc = PSM_SOFTC(PSM_UNIT(dev)); 1324 unsigned char buf[PSM_SMALLBUFSIZE]; 1325 int error = 0; 1326 int s; 1327 int l; 1328 1329 if ((sc->state & PSM_VALID) == 0) 1330 return EIO; 1331 1332 /* block until mouse activity occured */ 1333 s = spltty(); 1334 while (sc->queue.count <= 0) { 1335 if (PSM_NBLOCKIO(dev)) { 1336 splx(s); 1337 return EWOULDBLOCK; 1338 } 1339 sc->state |= PSM_ASLP; 1340 error = tsleep((caddr_t) sc, PZERO | PCATCH, "psmrea", 0); 1341 sc->state &= ~PSM_ASLP; 1342 if (error) { 1343 splx(s); 1344 return error; 1345 } else if ((sc->state & PSM_VALID) == 0) { 1346 /* the device disappeared! */ 1347 splx(s); 1348 return EIO; 1349 } 1350 } 1351 splx(s); 1352 1353 /* copy data to the user land */ 1354 while ((sc->queue.count > 0) && (uio->uio_resid > 0)) { 1355 s = spltty(); 1356 l = min(sc->queue.count, uio->uio_resid); 1357 if (l > sizeof(buf)) 1358 l = sizeof(buf); 1359 if (l > sizeof(sc->queue.buf) - sc->queue.head) { 1360 bcopy(&sc->queue.buf[sc->queue.head], &buf[0], 1361 sizeof(sc->queue.buf) - sc->queue.head); 1362 bcopy(&sc->queue.buf[0], 1363 &buf[sizeof(sc->queue.buf) - sc->queue.head], 1364 l - (sizeof(sc->queue.buf) - sc->queue.head)); 1365 } else { 1366 bcopy(&sc->queue.buf[sc->queue.head], &buf[0], l); 1367 } 1368 sc->queue.count -= l; 1369 sc->queue.head = (sc->queue.head + l) % sizeof(sc->queue.buf); 1370 splx(s); 1371 error = uiomove(buf, l, uio); 1372 if (error) 1373 break; 1374 } 1375 1376 return error; 1377 } 1378 1379 static int 1380 block_mouse_data(struct psm_softc *sc, int *c) 1381 { 1382 int s; 1383 1384 if (!kbdc_lock(sc->kbdc, TRUE)) 1385 return EIO; 1386 1387 s = spltty(); 1388 *c = get_controller_command_byte(sc->kbdc); 1389 if ((*c == -1) 1390 || !set_controller_command_byte(sc->kbdc, 1391 kbdc_get_device_mask(sc->kbdc), 1392 KBD_DISABLE_KBD_PORT | KBD_DISABLE_KBD_INT 1393 | KBD_ENABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 1394 /* this is CONTROLLER ERROR */ 1395 splx(s); 1396 kbdc_lock(sc->kbdc, FALSE); 1397 return EIO; 1398 } 1399 1400 /* 1401 * The device may be in the middle of status data transmission. 1402 * The transmission will be interrupted, thus, incomplete status 1403 * data must be discarded. Although the aux interrupt is disabled 1404 * at the keyboard controller level, at most one aux interrupt 1405 * may have already been pending and a data byte is in the 1406 * output buffer; throw it away. Note that the second argument 1407 * to `empty_aux_buffer()' is zero, so that the call will just 1408 * flush the internal queue. 1409 * `psmintr()' will be invoked after `splx()' if an interrupt is 1410 * pending; it will see no data and returns immediately. 1411 */ 1412 empty_aux_buffer(sc->kbdc, 0); /* flush the queue */ 1413 read_aux_data_no_wait(sc->kbdc); /* throw away data if any */ 1414 sc->inputbytes = 0; 1415 splx(s); 1416 1417 return 0; 1418 } 1419 1420 static int 1421 unblock_mouse_data(struct psm_softc *sc, int c) 1422 { 1423 int error = 0; 1424 1425 /* 1426 * We may have seen a part of status data during `set_mouse_XXX()'. 1427 * they have been queued; flush it. 1428 */ 1429 empty_aux_buffer(sc->kbdc, 0); 1430 1431 /* restore ports and interrupt */ 1432 if (!set_controller_command_byte(sc->kbdc, 1433 kbdc_get_device_mask(sc->kbdc), 1434 c & (KBD_KBD_CONTROL_BITS | KBD_AUX_CONTROL_BITS))) { 1435 /* CONTROLLER ERROR; this is serious, we may have 1436 * been left with the inaccessible keyboard and 1437 * the disabled mouse interrupt. 1438 */ 1439 error = EIO; 1440 } 1441 1442 kbdc_lock(sc->kbdc, FALSE); 1443 return error; 1444 } 1445 1446 static int 1447 psmioctl(dev_t dev, u_long cmd, caddr_t addr, int flag, struct proc *p) 1448 { 1449 struct psm_softc *sc = PSM_SOFTC(PSM_UNIT(dev)); 1450 mousemode_t mode; 1451 mousestatus_t status; 1452 #if (defined(MOUSE_GETVARS)) 1453 mousevar_t *var; 1454 #endif 1455 mousedata_t *data; 1456 int stat[3]; 1457 int command_byte; 1458 int error = 0; 1459 int s; 1460 1461 /* Perform IOCTL command */ 1462 switch (cmd) { 1463 1464 case OLD_MOUSE_GETHWINFO: 1465 s = spltty(); 1466 ((old_mousehw_t *)addr)->buttons = sc->hw.buttons; 1467 ((old_mousehw_t *)addr)->iftype = sc->hw.iftype; 1468 ((old_mousehw_t *)addr)->type = sc->hw.type; 1469 ((old_mousehw_t *)addr)->hwid = sc->hw.hwid & 0x00ff; 1470 splx(s); 1471 break; 1472 1473 case MOUSE_GETHWINFO: 1474 s = spltty(); 1475 *(mousehw_t *)addr = sc->hw; 1476 if (sc->mode.level == PSM_LEVEL_BASE) 1477 ((mousehw_t *)addr)->model = MOUSE_MODEL_GENERIC; 1478 splx(s); 1479 break; 1480 1481 case OLD_MOUSE_GETMODE: 1482 s = spltty(); 1483 switch (sc->mode.level) { 1484 case PSM_LEVEL_BASE: 1485 ((old_mousemode_t *)addr)->protocol = MOUSE_PROTO_PS2; 1486 break; 1487 case PSM_LEVEL_STANDARD: 1488 ((old_mousemode_t *)addr)->protocol = MOUSE_PROTO_SYSMOUSE; 1489 break; 1490 case PSM_LEVEL_NATIVE: 1491 ((old_mousemode_t *)addr)->protocol = MOUSE_PROTO_PS2; 1492 break; 1493 } 1494 ((old_mousemode_t *)addr)->rate = sc->mode.rate; 1495 ((old_mousemode_t *)addr)->resolution = sc->mode.resolution; 1496 ((old_mousemode_t *)addr)->accelfactor = sc->mode.accelfactor; 1497 splx(s); 1498 break; 1499 1500 case MOUSE_GETMODE: 1501 s = spltty(); 1502 *(mousemode_t *)addr = sc->mode; 1503 ((mousemode_t *)addr)->resolution = 1504 MOUSE_RES_LOW - sc->mode.resolution; 1505 switch (sc->mode.level) { 1506 case PSM_LEVEL_BASE: 1507 ((mousemode_t *)addr)->protocol = MOUSE_PROTO_PS2; 1508 ((mousemode_t *)addr)->packetsize = MOUSE_PS2_PACKETSIZE; 1509 break; 1510 case PSM_LEVEL_STANDARD: 1511 ((mousemode_t *)addr)->protocol = MOUSE_PROTO_SYSMOUSE; 1512 ((mousemode_t *)addr)->packetsize = MOUSE_SYS_PACKETSIZE; 1513 ((mousemode_t *)addr)->syncmask[0] = MOUSE_SYS_SYNCMASK; 1514 ((mousemode_t *)addr)->syncmask[1] = MOUSE_SYS_SYNC; 1515 break; 1516 case PSM_LEVEL_NATIVE: 1517 /* FIXME: this isn't quite correct... XXX */ 1518 ((mousemode_t *)addr)->protocol = MOUSE_PROTO_PS2; 1519 break; 1520 } 1521 splx(s); 1522 break; 1523 1524 case OLD_MOUSE_SETMODE: 1525 case MOUSE_SETMODE: 1526 if (cmd == OLD_MOUSE_SETMODE) { 1527 mode.rate = ((old_mousemode_t *)addr)->rate; 1528 /* 1529 * resolution old I/F new I/F 1530 * default 0 0 1531 * low 1 -2 1532 * medium low 2 -3 1533 * medium high 3 -4 1534 * high 4 -5 1535 */ 1536 if (((old_mousemode_t *)addr)->resolution > 0) 1537 mode.resolution = -((old_mousemode_t *)addr)->resolution - 1; 1538 mode.accelfactor = ((old_mousemode_t *)addr)->accelfactor; 1539 mode.level = -1; 1540 } else { 1541 mode = *(mousemode_t *)addr; 1542 } 1543 1544 /* adjust and validate parameters. */ 1545 if (mode.rate > UCHAR_MAX) 1546 return EINVAL; 1547 if (mode.rate == 0) 1548 mode.rate = sc->dflt_mode.rate; 1549 else if (mode.rate == -1) 1550 /* don't change the current setting */ 1551 ; 1552 else if (mode.rate < 0) 1553 return EINVAL; 1554 if (mode.resolution >= UCHAR_MAX) 1555 return EINVAL; 1556 if (mode.resolution >= 200) 1557 mode.resolution = MOUSE_RES_HIGH; 1558 else if (mode.resolution >= 100) 1559 mode.resolution = MOUSE_RES_MEDIUMHIGH; 1560 else if (mode.resolution >= 50) 1561 mode.resolution = MOUSE_RES_MEDIUMLOW; 1562 else if (mode.resolution > 0) 1563 mode.resolution = MOUSE_RES_LOW; 1564 if (mode.resolution == MOUSE_RES_DEFAULT) 1565 mode.resolution = sc->dflt_mode.resolution; 1566 else if (mode.resolution == -1) 1567 /* don't change the current setting */ 1568 ; 1569 else if (mode.resolution < 0) /* MOUSE_RES_LOW/MEDIUM/HIGH */ 1570 mode.resolution = MOUSE_RES_LOW - mode.resolution; 1571 if (mode.level == -1) 1572 /* don't change the current setting */ 1573 mode.level = sc->mode.level; 1574 else if ((mode.level < PSM_LEVEL_MIN) || (mode.level > PSM_LEVEL_MAX)) 1575 return EINVAL; 1576 if (mode.accelfactor == -1) 1577 /* don't change the current setting */ 1578 mode.accelfactor = sc->mode.accelfactor; 1579 else if (mode.accelfactor < 0) 1580 return EINVAL; 1581 1582 /* don't allow anybody to poll the keyboard controller */ 1583 error = block_mouse_data(sc, &command_byte); 1584 if (error) 1585 return error; 1586 1587 /* set mouse parameters */ 1588 if (mode.rate > 0) 1589 mode.rate = set_mouse_sampling_rate(sc->kbdc, mode.rate); 1590 if (mode.resolution >= 0) 1591 mode.resolution = set_mouse_resolution(sc->kbdc, mode.resolution); 1592 set_mouse_scaling(sc->kbdc, 1); 1593 get_mouse_status(sc->kbdc, stat, 0, 3); 1594 1595 s = spltty(); 1596 sc->mode.rate = mode.rate; 1597 sc->mode.resolution = mode.resolution; 1598 sc->mode.accelfactor = mode.accelfactor; 1599 sc->mode.level = mode.level; 1600 splx(s); 1601 1602 unblock_mouse_data(sc, command_byte); 1603 break; 1604 1605 case MOUSE_GETLEVEL: 1606 *(int *)addr = sc->mode.level; 1607 break; 1608 1609 case MOUSE_SETLEVEL: 1610 if ((*(int *)addr < PSM_LEVEL_MIN) || (*(int *)addr > PSM_LEVEL_MAX)) 1611 return EINVAL; 1612 sc->mode.level = *(int *)addr; 1613 break; 1614 1615 case MOUSE_GETSTATUS: 1616 s = spltty(); 1617 status = sc->status; 1618 sc->status.flags = 0; 1619 sc->status.obutton = sc->status.button; 1620 sc->status.button = 0; 1621 sc->status.dx = 0; 1622 sc->status.dy = 0; 1623 sc->status.dz = 0; 1624 splx(s); 1625 *(mousestatus_t *)addr = status; 1626 break; 1627 1628 #if (defined(MOUSE_GETVARS)) 1629 case MOUSE_GETVARS: 1630 var = (mousevar_t *)addr; 1631 bzero(var, sizeof(*var)); 1632 s = spltty(); 1633 var->var[0] = MOUSE_VARS_PS2_SIG; 1634 var->var[1] = sc->config; 1635 var->var[2] = sc->flags; 1636 splx(s); 1637 break; 1638 1639 case MOUSE_SETVARS: 1640 return ENODEV; 1641 #endif /* MOUSE_GETVARS */ 1642 1643 case MOUSE_READSTATE: 1644 case MOUSE_READDATA: 1645 data = (mousedata_t *)addr; 1646 if (data->len > sizeof(data->buf)/sizeof(data->buf[0])) 1647 return EINVAL; 1648 1649 error = block_mouse_data(sc, &command_byte); 1650 if (error) 1651 return error; 1652 if ((data->len = get_mouse_status(sc->kbdc, data->buf, 1653 (cmd == MOUSE_READDATA) ? 1 : 0, data->len)) <= 0) 1654 error = EIO; 1655 unblock_mouse_data(sc, command_byte); 1656 break; 1657 1658 #if (defined(MOUSE_SETRESOLUTION)) 1659 case MOUSE_SETRESOLUTION: 1660 mode.resolution = *(int *)addr; 1661 if (mode.resolution >= UCHAR_MAX) 1662 return EINVAL; 1663 else if (mode.resolution >= 200) 1664 mode.resolution = MOUSE_RES_HIGH; 1665 else if (mode.resolution >= 100) 1666 mode.resolution = MOUSE_RES_MEDIUMHIGH; 1667 else if (mode.resolution >= 50) 1668 mode.resolution = MOUSE_RES_MEDIUMLOW; 1669 else if (mode.resolution > 0) 1670 mode.resolution = MOUSE_RES_LOW; 1671 if (mode.resolution == MOUSE_RES_DEFAULT) 1672 mode.resolution = sc->dflt_mode.resolution; 1673 else if (mode.resolution == -1) 1674 mode.resolution = sc->mode.resolution; 1675 else if (mode.resolution < 0) /* MOUSE_RES_LOW/MEDIUM/HIGH */ 1676 mode.resolution = MOUSE_RES_LOW - mode.resolution; 1677 1678 error = block_mouse_data(sc, &command_byte); 1679 if (error) 1680 return error; 1681 sc->mode.resolution = set_mouse_resolution(sc->kbdc, mode.resolution); 1682 if (sc->mode.resolution != mode.resolution) 1683 error = EIO; 1684 unblock_mouse_data(sc, command_byte); 1685 break; 1686 #endif /* MOUSE_SETRESOLUTION */ 1687 1688 #if (defined(MOUSE_SETRATE)) 1689 case MOUSE_SETRATE: 1690 mode.rate = *(int *)addr; 1691 if (mode.rate > UCHAR_MAX) 1692 return EINVAL; 1693 if (mode.rate == 0) 1694 mode.rate = sc->dflt_mode.rate; 1695 else if (mode.rate < 0) 1696 mode.rate = sc->mode.rate; 1697 1698 error = block_mouse_data(sc, &command_byte); 1699 if (error) 1700 return error; 1701 sc->mode.rate = set_mouse_sampling_rate(sc->kbdc, mode.rate); 1702 if (sc->mode.rate != mode.rate) 1703 error = EIO; 1704 unblock_mouse_data(sc, command_byte); 1705 break; 1706 #endif /* MOUSE_SETRATE */ 1707 1708 #if (defined(MOUSE_SETSCALING)) 1709 case MOUSE_SETSCALING: 1710 if ((*(int *)addr <= 0) || (*(int *)addr > 2)) 1711 return EINVAL; 1712 1713 error = block_mouse_data(sc, &command_byte); 1714 if (error) 1715 return error; 1716 if (!set_mouse_scaling(sc->kbdc, *(int *)addr)) 1717 error = EIO; 1718 unblock_mouse_data(sc, command_byte); 1719 break; 1720 #endif /* MOUSE_SETSCALING */ 1721 1722 #if (defined(MOUSE_GETHWID)) 1723 case MOUSE_GETHWID: 1724 error = block_mouse_data(sc, &command_byte); 1725 if (error) 1726 return error; 1727 sc->hw.hwid &= ~0x00ff; 1728 sc->hw.hwid |= get_aux_id(sc->kbdc); 1729 *(int *)addr = sc->hw.hwid & 0x00ff; 1730 unblock_mouse_data(sc, command_byte); 1731 break; 1732 #endif /* MOUSE_GETHWID */ 1733 1734 default: 1735 return ENOTTY; 1736 } 1737 1738 return error; 1739 } 1740 1741 static void 1742 psmintr(void *arg) 1743 { 1744 /* 1745 * the table to turn PS/2 mouse button bits (MOUSE_PS2_BUTTON?DOWN) 1746 * into `mousestatus' button bits (MOUSE_BUTTON?DOWN). 1747 */ 1748 static int butmap[8] = { 1749 0, 1750 MOUSE_BUTTON1DOWN, 1751 MOUSE_BUTTON3DOWN, 1752 MOUSE_BUTTON1DOWN | MOUSE_BUTTON3DOWN, 1753 MOUSE_BUTTON2DOWN, 1754 MOUSE_BUTTON1DOWN | MOUSE_BUTTON2DOWN, 1755 MOUSE_BUTTON2DOWN | MOUSE_BUTTON3DOWN, 1756 MOUSE_BUTTON1DOWN | MOUSE_BUTTON2DOWN | MOUSE_BUTTON3DOWN 1757 }; 1758 static int butmap_versapad[8] = { 1759 0, 1760 MOUSE_BUTTON3DOWN, 1761 0, 1762 MOUSE_BUTTON3DOWN, 1763 MOUSE_BUTTON1DOWN, 1764 MOUSE_BUTTON1DOWN | MOUSE_BUTTON3DOWN, 1765 MOUSE_BUTTON1DOWN, 1766 MOUSE_BUTTON1DOWN | MOUSE_BUTTON3DOWN 1767 }; 1768 register struct psm_softc *sc = arg; 1769 mousestatus_t ms; 1770 int x, y, z; 1771 int c; 1772 int l; 1773 int x0, y0; 1774 1775 /* read until there is nothing to read */ 1776 while((c = read_aux_data_no_wait(sc->kbdc)) != -1) { 1777 1778 /* discard the byte if the device is not open */ 1779 if ((sc->state & PSM_OPEN) == 0) 1780 continue; 1781 1782 /* 1783 * Check sync bits. We check for overflow bits and the bit 3 1784 * for most mice. True, the code doesn't work if overflow 1785 * condition occurs. But we expect it rarely happens... 1786 */ 1787 if ((sc->inputbytes == 0) 1788 && ((c & sc->mode.syncmask[0]) != sc->mode.syncmask[1])) { 1789 log(LOG_DEBUG, "psmintr: out of sync (%04x != %04x).\n", 1790 c & sc->mode.syncmask[0], sc->mode.syncmask[1]); 1791 continue; 1792 } 1793 1794 sc->ipacket[sc->inputbytes++] = c; 1795 if (sc->inputbytes < sc->mode.packetsize) 1796 continue; 1797 1798 #if 0 1799 log(LOG_DEBUG, "psmintr: %02x %02x %02x %02x %02x %02x\n", 1800 sc->ipacket[0], sc->ipacket[1], sc->ipacket[2], 1801 sc->ipacket[3], sc->ipacket[4], sc->ipacket[5]); 1802 #endif 1803 1804 c = sc->ipacket[0]; 1805 1806 /* 1807 * A kludge for Kensington device! 1808 * The MSB of the horizontal count appears to be stored in 1809 * a strange place. This kludge doesn't affect other mice 1810 * because the bit is the overflow bit which is, in most cases, 1811 * expected to be zero when we reach here. XXX 1812 */ 1813 if (sc->hw.model != MOUSE_MODEL_VERSAPAD) 1814 sc->ipacket[1] |= (c & MOUSE_PS2_XOVERFLOW) ? 0x80 : 0; 1815 1816 /* ignore the overflow bits... */ 1817 x = (c & MOUSE_PS2_XNEG) ? sc->ipacket[1] - 256 : sc->ipacket[1]; 1818 y = (c & MOUSE_PS2_YNEG) ? sc->ipacket[2] - 256 : sc->ipacket[2]; 1819 z = 0; 1820 ms.obutton = sc->button; /* previous button state */ 1821 ms.button = butmap[c & MOUSE_PS2_BUTTONS]; 1822 /* `tapping' action */ 1823 if (sc->config & PSM_CONFIG_FORCETAP) 1824 ms.button |= ((c & MOUSE_PS2_TAP)) ? 0 : MOUSE_BUTTON4DOWN; 1825 1826 switch (sc->hw.model) { 1827 1828 case MOUSE_MODEL_INTELLI: 1829 case MOUSE_MODEL_NET: 1830 /* wheel data is in the fourth byte */ 1831 z = (char)sc->ipacket[3]; 1832 break; 1833 1834 case MOUSE_MODEL_MOUSEMANPLUS: 1835 /* 1836 * PS2++ protocl packet 1837 * 1838 * b7 b6 b5 b4 b3 b2 b1 b0 1839 * byte 1: * 1 p3 p2 1 * * * 1840 * byte 2: c1 c2 p1 p0 d1 d0 1 0 1841 * 1842 * p3-p0: packet type 1843 * c1, c2: c1 & c2 == 1, if p2 == 0 1844 * c1 & c2 == 0, if p2 == 1 1845 * 1846 * packet type: 0 (device type) 1847 * See comments in enable_mmanplus() below. 1848 * 1849 * packet type: 1 (wheel data) 1850 * 1851 * b7 b6 b5 b4 b3 b2 b1 b0 1852 * byte 3: h * B5 B4 s d2 d1 d0 1853 * 1854 * h: 1, if horizontal roller data 1855 * 0, if vertical roller data 1856 * B4, B5: button 4 and 5 1857 * s: sign bit 1858 * d2-d0: roller data 1859 * 1860 * packet type: 2 (reserved) 1861 */ 1862 if (((c & MOUSE_PS2PLUS_SYNCMASK) == MOUSE_PS2PLUS_SYNC) 1863 && (abs(x) > 191) 1864 && MOUSE_PS2PLUS_CHECKBITS(sc->ipacket)) { 1865 /* the extended data packet encodes button and wheel events */ 1866 switch (MOUSE_PS2PLUS_PACKET_TYPE(sc->ipacket)) { 1867 case 1: 1868 /* wheel data packet */ 1869 x = y = 0; 1870 if (sc->ipacket[2] & 0x80) { 1871 /* horizontal roller count - ignore it XXX*/ 1872 } else { 1873 /* vertical roller count */ 1874 z = (sc->ipacket[2] & MOUSE_PS2PLUS_ZNEG) 1875 ? (sc->ipacket[2] & 0x0f) - 16 1876 : (sc->ipacket[2] & 0x0f); 1877 } 1878 ms.button |= (sc->ipacket[2] & MOUSE_PS2PLUS_BUTTON4DOWN) 1879 ? MOUSE_BUTTON4DOWN : 0; 1880 ms.button |= (sc->ipacket[2] & MOUSE_PS2PLUS_BUTTON5DOWN) 1881 ? MOUSE_BUTTON5DOWN : 0; 1882 break; 1883 case 2: 1884 /* this packet type is reserved, and currently ignored */ 1885 /* FALL THROUGH */ 1886 case 0: 1887 /* device type packet - shouldn't happen */ 1888 /* FALL THROUGH */ 1889 default: 1890 x = y = 0; 1891 ms.button = ms.obutton; 1892 log(LOG_DEBUG, "psmintr: unknown PS2++ packet type %d: " 1893 "0x%02x 0x%02x 0x%02x\n", 1894 MOUSE_PS2PLUS_PACKET_TYPE(sc->ipacket), 1895 sc->ipacket[0], sc->ipacket[1], sc->ipacket[2]); 1896 break; 1897 } 1898 } else { 1899 /* preserve button states */ 1900 ms.button |= ms.obutton & MOUSE_EXTBUTTONS; 1901 } 1902 break; 1903 1904 case MOUSE_MODEL_GLIDEPOINT: 1905 /* `tapping' action */ 1906 ms.button |= ((c & MOUSE_PS2_TAP)) ? 0 : MOUSE_BUTTON4DOWN; 1907 break; 1908 1909 case MOUSE_MODEL_NETSCROLL: 1910 /* three addtional bytes encode button and wheel events */ 1911 ms.button |= (sc->ipacket[3] & MOUSE_PS2_BUTTON3DOWN) 1912 ? MOUSE_BUTTON4DOWN : 0; 1913 z = (sc->ipacket[3] & MOUSE_PS2_XNEG) 1914 ? sc->ipacket[4] - 256 : sc->ipacket[4]; 1915 break; 1916 1917 case MOUSE_MODEL_THINK: 1918 /* the fourth button state in the first byte */ 1919 ms.button |= (c & MOUSE_PS2_TAP) ? MOUSE_BUTTON4DOWN : 0; 1920 break; 1921 1922 case MOUSE_MODEL_VERSAPAD: 1923 /* VersaPad PS/2 absolute mode message format 1924 * 1925 * [packet1] 7 6 5 4 3 2 1 0(LSB) 1926 * ipacket[0]: 1 1 0 A 1 L T R 1927 * ipacket[1]: H7 H6 H5 H4 H3 H2 H1 H0 1928 * ipacket[2]: V7 V6 V5 V4 V3 V2 V1 V0 1929 * ipacket[3]: 1 1 1 A 1 L T R 1930 * ipacket[4]:V11 V10 V9 V8 H11 H10 H9 H8 1931 * ipacket[5]: 0 P6 P5 P4 P3 P2 P1 P0 1932 * 1933 * [note] 1934 * R: right physical mouse button (1=on) 1935 * T: touch pad virtual button (1=tapping) 1936 * L: left physical mouse button (1=on) 1937 * A: position data is valid (1=valid) 1938 * H: horizontal data (12bit signed integer. H11 is sign bit.) 1939 * V: vertical data (12bit signed integer. V11 is sign bit.) 1940 * P: pressure data 1941 * 1942 * Tapping is mapped to MOUSE_BUTTON4. 1943 */ 1944 ms.button = butmap_versapad[c & MOUSE_PS2VERSA_BUTTONS]; 1945 ms.button |= (c & MOUSE_PS2VERSA_TAP) ? MOUSE_BUTTON4DOWN : 0; 1946 x = y = 0; 1947 if (c & MOUSE_PS2VERSA_IN_USE) { 1948 x0 = sc->ipacket[1] | (((sc->ipacket[4]) & 0x0f) << 8); 1949 y0 = sc->ipacket[2] | (((sc->ipacket[4]) & 0xf0) << 4); 1950 if (x0 & 0x800) 1951 x0 -= 0x1000; 1952 if (y0 & 0x800) 1953 y0 -= 0x1000; 1954 if (sc->flags & PSM_FLAGS_FINGERDOWN) { 1955 x = sc->xold - x0; 1956 y = y0 - sc->yold; 1957 if (x < 0) /* XXX */ 1958 x++; 1959 else if (x) 1960 x--; 1961 if (y < 0) 1962 y++; 1963 else if (y) 1964 y--; 1965 } else { 1966 sc->flags |= PSM_FLAGS_FINGERDOWN; 1967 } 1968 sc->xold = x0; 1969 sc->yold = y0; 1970 } else { 1971 sc->flags &= ~PSM_FLAGS_FINGERDOWN; 1972 } 1973 c = ((x < 0) ? MOUSE_PS2_XNEG : 0) 1974 | ((y < 0) ? MOUSE_PS2_YNEG : 0); 1975 break; 1976 1977 case MOUSE_MODEL_GENERIC: 1978 default: 1979 break; 1980 } 1981 1982 /* scale values */ 1983 if (sc->mode.accelfactor >= 1) { 1984 if (x != 0) { 1985 x = x * x / sc->mode.accelfactor; 1986 if (x == 0) 1987 x = 1; 1988 if (c & MOUSE_PS2_XNEG) 1989 x = -x; 1990 } 1991 if (y != 0) { 1992 y = y * y / sc->mode.accelfactor; 1993 if (y == 0) 1994 y = 1; 1995 if (c & MOUSE_PS2_YNEG) 1996 y = -y; 1997 } 1998 } 1999 2000 ms.dx = x; 2001 ms.dy = y; 2002 ms.dz = z; 2003 ms.flags = ((x || y || z) ? MOUSE_POSCHANGED : 0) 2004 | (ms.obutton ^ ms.button); 2005 2006 if (sc->mode.level < PSM_LEVEL_NATIVE) 2007 sc->inputbytes = tame_mouse(sc, &ms, sc->ipacket); 2008 2009 sc->status.flags |= ms.flags; 2010 sc->status.dx += ms.dx; 2011 sc->status.dy += ms.dy; 2012 sc->status.dz += ms.dz; 2013 sc->status.button = ms.button; 2014 sc->button = ms.button; 2015 2016 /* queue data */ 2017 if (sc->queue.count + sc->inputbytes < sizeof(sc->queue.buf)) { 2018 l = min(sc->inputbytes, sizeof(sc->queue.buf) - sc->queue.tail); 2019 bcopy(&sc->ipacket[0], &sc->queue.buf[sc->queue.tail], l); 2020 if (sc->inputbytes > l) 2021 bcopy(&sc->ipacket[l], &sc->queue.buf[0], sc->inputbytes - l); 2022 sc->queue.tail = 2023 (sc->queue.tail + sc->inputbytes) % sizeof(sc->queue.buf); 2024 sc->queue.count += sc->inputbytes; 2025 } 2026 sc->inputbytes = 0; 2027 2028 if (sc->state & PSM_ASLP) { 2029 sc->state &= ~PSM_ASLP; 2030 wakeup((caddr_t) sc); 2031 } 2032 selwakeup(&sc->rsel); 2033 } 2034 } 2035 2036 static int 2037 psmpoll(dev_t dev, int events, struct proc *p) 2038 { 2039 struct psm_softc *sc = PSM_SOFTC(PSM_UNIT(dev)); 2040 int s; 2041 int revents = 0; 2042 2043 /* Return true if a mouse event available */ 2044 s = spltty(); 2045 if (events & (POLLIN | POLLRDNORM)) { 2046 if (sc->queue.count > 0) 2047 revents |= events & (POLLIN | POLLRDNORM); 2048 else 2049 selrecord(p, &sc->rsel); 2050 } 2051 splx(s); 2052 2053 return (revents); 2054 } 2055 2056 /* vendor/model specific routines */ 2057 2058 static int mouse_id_proc1(KBDC kbdc, int res, int scale, int *status) 2059 { 2060 if (set_mouse_resolution(kbdc, res) != res) 2061 return FALSE; 2062 if (set_mouse_scaling(kbdc, scale) 2063 && set_mouse_scaling(kbdc, scale) 2064 && set_mouse_scaling(kbdc, scale) 2065 && (get_mouse_status(kbdc, status, 0, 3) >= 3)) 2066 return TRUE; 2067 return FALSE; 2068 } 2069 2070 #if notyet 2071 /* Logitech MouseMan Cordless II */ 2072 static int 2073 enable_lcordless(struct psm_softc *sc) 2074 { 2075 int status[3]; 2076 int ch; 2077 2078 if (!mouse_id_proc1(sc->kbdc, PSMD_RES_HIGH, 2, status)) 2079 return FALSE; 2080 if (status[1] == PSMD_RES_HIGH) 2081 return FALSE; 2082 ch = (status[0] & 0x07) - 1; /* channel # */ 2083 if ((ch <= 0) || (ch > 4)) 2084 return FALSE; 2085 /* 2086 * status[1]: always one? 2087 * status[2]: battery status? (0-100) 2088 */ 2089 return TRUE; 2090 } 2091 #endif /* notyet */ 2092 2093 /* Genius NetScroll Mouse */ 2094 static int 2095 enable_groller(struct psm_softc *sc) 2096 { 2097 int status[3]; 2098 2099 /* 2100 * The special sequence to enable the fourth button and the 2101 * roller. Immediately after this sequence check status bytes. 2102 * if the mouse is NetScroll, the second and the third bytes are 2103 * '3' and 'D'. 2104 */ 2105 2106 /* 2107 * If the mouse is an ordinary PS/2 mouse, the status bytes should 2108 * look like the following. 2109 * 2110 * byte 1 bit 7 always 0 2111 * bit 6 stream mode (0) 2112 * bit 5 disabled (0) 2113 * bit 4 1:1 scaling (0) 2114 * bit 3 always 0 2115 * bit 0-2 button status 2116 * byte 2 resolution (PSMD_RES_HIGH) 2117 * byte 3 report rate (?) 2118 */ 2119 2120 if (!mouse_id_proc1(sc->kbdc, PSMD_RES_HIGH, 1, status)) 2121 return FALSE; 2122 if ((status[1] != '3') || (status[2] != 'D')) 2123 return FALSE; 2124 /* FIXME!! */ 2125 sc->hw.buttons = get_mouse_buttons(sc->kbdc); 2126 sc->hw.buttons = 4; 2127 return TRUE; 2128 } 2129 2130 /* Genius NetMouse/NetMouse Pro */ 2131 static int 2132 enable_gmouse(struct psm_softc *sc) 2133 { 2134 int status[3]; 2135 2136 /* 2137 * The special sequence to enable the middle, "rubber" button. 2138 * Immediately after this sequence check status bytes. 2139 * if the mouse is NetMouse, NetMouse Pro, or ASCII MIE Mouse, 2140 * the second and the third bytes are '3' and 'U'. 2141 * NOTE: NetMouse reports that it has three buttons although it has 2142 * two buttons and a rubber button. NetMouse Pro and MIE Mouse 2143 * say they have three buttons too and they do have a button on the 2144 * side... 2145 */ 2146 if (!mouse_id_proc1(sc->kbdc, PSMD_RES_HIGH, 1, status)) 2147 return FALSE; 2148 if ((status[1] != '3') || (status[2] != 'U')) 2149 return FALSE; 2150 return TRUE; 2151 } 2152 2153 /* ALPS GlidePoint */ 2154 static int 2155 enable_aglide(struct psm_softc *sc) 2156 { 2157 int status[3]; 2158 2159 /* 2160 * The special sequence to obtain ALPS GlidePoint specific 2161 * information. Immediately after this sequence, status bytes will 2162 * contain something interesting. 2163 * NOTE: ALPS produces several models of GlidePoint. Some of those 2164 * do not respond to this sequence, thus, cannot be detected this way. 2165 */ 2166 if (set_mouse_sampling_rate(sc->kbdc, 100) != 100) 2167 return FALSE; 2168 if (!mouse_id_proc1(sc->kbdc, PSMD_RES_LOW, 2, status)) 2169 return FALSE; 2170 if ((status[1] == PSMD_RES_LOW) || (status[2] == 100)) 2171 return FALSE; 2172 return TRUE; 2173 } 2174 2175 /* Kensington ThinkingMouse/Trackball */ 2176 static int 2177 enable_kmouse(struct psm_softc *sc) 2178 { 2179 static unsigned char rate[] = { 20, 60, 40, 20, 20, 60, 40, 20, 20 }; 2180 KBDC kbdc = sc->kbdc; 2181 int status[3]; 2182 int id1; 2183 int id2; 2184 int i; 2185 2186 id1 = get_aux_id(kbdc); 2187 if (set_mouse_sampling_rate(kbdc, 10) != 10) 2188 return FALSE; 2189 /* 2190 * The device is now in the native mode? It returns a different 2191 * ID value... 2192 */ 2193 id2 = get_aux_id(kbdc); 2194 if ((id1 == id2) || (id2 != 2)) 2195 return FALSE; 2196 2197 if (set_mouse_resolution(kbdc, PSMD_RES_LOW) != PSMD_RES_LOW) 2198 return FALSE; 2199 #if PSM_DEBUG >= 2 2200 /* at this point, resolution is LOW, sampling rate is 10/sec */ 2201 if (get_mouse_status(kbdc, status, 0, 3) < 3) 2202 return FALSE; 2203 #endif 2204 2205 /* 2206 * The special sequence to enable the third and fourth buttons. 2207 * Otherwise they behave like the first and second buttons. 2208 */ 2209 for (i = 0; i < sizeof(rate)/sizeof(rate[0]); ++i) { 2210 if (set_mouse_sampling_rate(kbdc, rate[i]) != rate[i]) 2211 return FALSE; 2212 } 2213 2214 /* 2215 * At this point, the device is using default resolution and 2216 * sampling rate for the native mode. 2217 */ 2218 if (get_mouse_status(kbdc, status, 0, 3) < 3) 2219 return FALSE; 2220 if ((status[1] == PSMD_RES_LOW) || (status[2] == rate[i - 1])) 2221 return FALSE; 2222 2223 /* the device appears be enabled by this sequence, diable it for now */ 2224 disable_aux_dev(kbdc); 2225 empty_aux_buffer(kbdc, 5); 2226 2227 return TRUE; 2228 } 2229 2230 /* Logitech MouseMan+/FirstMouse+ */ 2231 static int 2232 enable_mmanplus(struct psm_softc *sc) 2233 { 2234 static char res[] = { 2235 -1, PSMD_RES_LOW, PSMD_RES_HIGH, PSMD_RES_MEDIUM_HIGH, 2236 PSMD_RES_MEDIUM_LOW, -1, PSMD_RES_HIGH, PSMD_RES_MEDIUM_LOW, 2237 PSMD_RES_MEDIUM_HIGH, PSMD_RES_HIGH, 2238 }; 2239 KBDC kbdc = sc->kbdc; 2240 int data[3]; 2241 int i; 2242 2243 /* the special sequence to enable the fourth button and the roller. */ 2244 for (i = 0; i < sizeof(res)/sizeof(res[0]); ++i) { 2245 if (res[i] < 0) { 2246 if (!set_mouse_scaling(kbdc, 1)) 2247 return FALSE; 2248 } else { 2249 if (set_mouse_resolution(kbdc, res[i]) != res[i]) 2250 return FALSE; 2251 } 2252 } 2253 2254 if (get_mouse_status(kbdc, data, 1, 3) < 3) 2255 return FALSE; 2256 2257 /* 2258 * PS2++ protocl, packet type 0 2259 * 2260 * b7 b6 b5 b4 b3 b2 b1 b0 2261 * byte 1: * 1 p3 p2 1 * * * 2262 * byte 2: 1 1 p1 p0 m1 m0 1 0 2263 * byte 3: m7 m6 m5 m4 m3 m2 m1 m0 2264 * 2265 * p3-p0: packet type: 0 2266 * m7-m0: model ID: MouseMan+:0x50, FirstMouse+:0x51,... 2267 */ 2268 /* check constant bits */ 2269 if ((data[0] & MOUSE_PS2PLUS_SYNCMASK) != MOUSE_PS2PLUS_SYNC) 2270 return FALSE; 2271 if ((data[1] & 0xc3) != 0xc2) 2272 return FALSE; 2273 /* check d3-d0 in byte 2 */ 2274 if (!MOUSE_PS2PLUS_CHECKBITS(data)) 2275 return FALSE; 2276 /* check p3-p0 */ 2277 if (MOUSE_PS2PLUS_PACKET_TYPE(data) != 0) 2278 return FALSE; 2279 2280 sc->hw.hwid &= 0x00ff; 2281 sc->hw.hwid |= data[2] << 8; /* save model ID */ 2282 2283 /* 2284 * MouseMan+ (or FirstMouse+) is now in its native mode, in which 2285 * the wheel and the fourth button events are encoded in the 2286 * special data packet. The mouse may be put in the IntelliMouse mode 2287 * if it is initialized by the IntelliMouse's method. 2288 */ 2289 return TRUE; 2290 } 2291 2292 /* MS IntelliMouse */ 2293 static int 2294 enable_msintelli(struct psm_softc *sc) 2295 { 2296 /* 2297 * Logitech MouseMan+ and FirstMouse+ will also respond to this 2298 * probe routine and act like IntelliMouse. 2299 */ 2300 2301 static unsigned char rate[] = { 200, 100, 80, }; 2302 KBDC kbdc = sc->kbdc; 2303 int id; 2304 int i; 2305 2306 /* the special sequence to enable the third button and the roller. */ 2307 for (i = 0; i < sizeof(rate)/sizeof(rate[0]); ++i) { 2308 if (set_mouse_sampling_rate(kbdc, rate[i]) != rate[i]) 2309 return FALSE; 2310 } 2311 /* the device will give the genuine ID only after the above sequence */ 2312 id = get_aux_id(kbdc); 2313 if (id != PSM_INTELLI_ID) 2314 return FALSE; 2315 2316 sc->hw.hwid = id; 2317 sc->hw.buttons = 3; 2318 2319 return TRUE; 2320 } 2321 2322 /* Interlink electronics VersaPad */ 2323 static int 2324 enable_versapad(struct psm_softc *sc) 2325 { 2326 KBDC kbdc = sc->kbdc; 2327 int data[3]; 2328 2329 set_mouse_resolution(kbdc, PSMD_RES_MEDIUM_HIGH); /* set res. 2 */ 2330 set_mouse_sampling_rate(kbdc, 100); /* set rate 100 */ 2331 set_mouse_scaling(kbdc, 1); /* set scale 1:1 */ 2332 set_mouse_scaling(kbdc, 1); /* set scale 1:1 */ 2333 set_mouse_scaling(kbdc, 1); /* set scale 1:1 */ 2334 set_mouse_scaling(kbdc, 1); /* set scale 1:1 */ 2335 if (get_mouse_status(kbdc, data, 0, 3) < 3) /* get status */ 2336 return FALSE; 2337 if (data[2] != 0xa || data[1] != 0 ) /* rate == 0xa && res. == 0 */ 2338 return FALSE; 2339 set_mouse_scaling(kbdc, 1); /* set scale 1:1 */ 2340 2341 return TRUE; /* PS/2 absolute mode */ 2342 } 2343 2344 static int 2345 psmresume(device_t dev) 2346 { 2347 #ifdef PSM_HOOKRESUME 2348 struct psm_softc *sc = device_get_softc(dev); 2349 int unit = device_get_unit(dev); 2350 int err = 0; 2351 int s; 2352 int c; 2353 2354 if (verbose >= 2) 2355 log(LOG_NOTICE, "psm%d: system resume hook called.\n", unit); 2356 2357 /* don't let anybody mess with the aux device */ 2358 if (!kbdc_lock(sc->kbdc, TRUE)) 2359 return (EIO); 2360 s = spltty(); 2361 2362 /* save the current controller command byte */ 2363 empty_both_buffers(sc->kbdc, 10); 2364 c = get_controller_command_byte(sc->kbdc); 2365 if (verbose >= 2) 2366 log(LOG_DEBUG, "psm%d: current command byte: %04x (psmresume).\n", 2367 unit, c); 2368 2369 /* enable the aux port but disable the aux interrupt and the keyboard */ 2370 if ((c == -1) || !set_controller_command_byte(sc->kbdc, 2371 kbdc_get_device_mask(sc->kbdc), 2372 KBD_DISABLE_KBD_PORT | KBD_DISABLE_KBD_INT 2373 | KBD_ENABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 2374 /* CONTROLLER ERROR */ 2375 splx(s); 2376 kbdc_lock(sc->kbdc, FALSE); 2377 log(LOG_ERR, "psm%d: unable to set the command byte (psmresume).\n", 2378 unit); 2379 return (EIO); 2380 } 2381 2382 /* flush any data */ 2383 if (sc->state & PSM_VALID) { 2384 disable_aux_dev(sc->kbdc); /* this may fail; but never mind... */ 2385 empty_aux_buffer(sc->kbdc, 10); 2386 } 2387 sc->inputbytes = 0; 2388 2389 #ifdef PSM_RESETAFTERSUSPEND 2390 /* try to detect the aux device; are you still there? */ 2391 if (reinitialize(unit, &sc->mode)) { 2392 /* yes */ 2393 sc->state |= PSM_VALID; 2394 } else { 2395 /* the device has gone! */ 2396 restore_controller(sc->kbdc, c); 2397 sc->state &= ~PSM_VALID; 2398 log(LOG_ERR, "psm%d: the aux device has gone! (psmresume).\n", 2399 unit); 2400 err = ENXIO; 2401 } 2402 #endif /* PSM_RESETAFTERSUSPEND */ 2403 splx(s); 2404 2405 /* restore the driver state */ 2406 if ((sc->state & PSM_OPEN) && (err == 0)) { 2407 /* enable the aux device and the port again */ 2408 err = doopen(unit, c); 2409 if (err != 0) 2410 log(LOG_ERR, "psm%d: failed to enable the device (psmresume).\n", 2411 unit); 2412 } else { 2413 /* restore the keyboard port and disable the aux port */ 2414 if (!set_controller_command_byte(sc->kbdc, 2415 kbdc_get_device_mask(sc->kbdc), 2416 (c & KBD_KBD_CONTROL_BITS) 2417 | KBD_DISABLE_AUX_PORT | KBD_DISABLE_AUX_INT)) { 2418 /* CONTROLLER ERROR */ 2419 log(LOG_ERR, "psm%d: failed to disable the aux port (psmresume).\n", 2420 unit); 2421 err = EIO; 2422 } 2423 } 2424 2425 /* done */ 2426 kbdc_lock(sc->kbdc, FALSE); 2427 if ((sc->state & PSM_ASLP) && !(sc->state & PSM_VALID)) { 2428 /* 2429 * Release the blocked process; it must be notified that the device 2430 * cannot be accessed anymore. 2431 */ 2432 sc->state &= ~PSM_ASLP; 2433 wakeup((caddr_t)sc); 2434 } 2435 2436 if (verbose >= 2) 2437 log(LOG_DEBUG, "psm%d: system resume hook exiting.\n", unit); 2438 2439 return (err); 2440 #else /* !PSM_HOOKRESUME */ 2441 return (0); 2442 #endif /* PSM_HOOKRESUME */ 2443 } 2444 2445 DRIVER_MODULE(psm, atkbdc, psm_driver, psm_devclass, 0, 0); 2446