/* $XFree86$ */ /* * Copyright 2005 by J. Kean Johnston * * Permission to use, copy, modify, distribute, and sell this software and its * documentation for any purpose is hereby granted without fee, provided that * the above copyright notice appear in all copies and that both that * copyright notice and this permission notice appear in supporting * documentation, and that the name J. Kean Johnston not be used in * advertising or publicity pertaining to distribution of the software without * specific, written prior permission. J. Kean Johnston makes no * representations about the suitability of this software for any purpose. * It is provided "as is" without express or implied warranty. * * J. KEAN JOHNSTON DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO * EVENT SHALL J. KEAN JOHNSTON BE LIABLE FOR ANY SPECIAL, INDIRECT OR * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF * USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR * PERFORMANCE OF THIS SOFTWARE. */ /* $XConsortium$ */ /* * Based on sco_io.c which is * (C) Copyright 2003 J. Kean Johnston * * Based on lnx_kbd.c which is * Copyright (c) 2002 by The XFree86 Project, Inc. * * Based on the code from lnx_io.c which is * Copyright 1992 by Orest Zborowski * Copyright 1993 by David Dawes */ #define NEED_EVENTS #include "X.h" #include "compiler.h" #define _NEED_SYSI86 #include "xf86.h" #include "xf86Priv.h" #include "xf86OSpriv.h" #include "xf86_OSlib.h" #include "xf86Xinput.h" #include "xf86OSKbd.h" #include "atKeynames.h" #include "sco_kbd.h" #include #include #include static KbdProtocolRec protocols[] = { { "standard", PROT_STD }, { NULL, PROT_UNKNOWN_KBD } }; extern Bool VTSwitchEnabled; #ifdef USE_VT_SYSREQ extern Bool VTSysreqToggle; #endif static void SoundBell(InputInfoPtr pInfo, int loudness, int pitch, int duration) { if (loudness && pitch) { ioctl(pInfo->fd, KIOCSOUND, 1193180 / pitch); usleep(duration * loudness * 20); ioctl(pInfo->fd, KIOCSOUND, 0); } } static void SetKbdLeds(InputInfoPtr pInfo, int leds) { int real_leds = 0; static int once = 1; /* * sleep the first time through under SCO. There appears to be a * timing problem in the driver which causes the keyboard to be lost. * This usleep stops it from occurring. NOTE: this was in the old code. * I am not convinced it is true any longer, but it doesn't hurt to * leave this in here. */ if (once) { usleep(100); once = 0; } #ifdef LED_CAP if (leds & XLED1) real_leds |= LED_CAP; if (leds & XLED2) real_leds |= LED_NUM; if (leds & XLED3) real_leds |= LED_SCR; #ifdef LED_COMP if (leds & XLED4) real_leds |= LED_COMP; #else if (leds & XLED4) real_leds |= LED_SCR; #endif #endif ioctl(pInfo->fd, KDSETLED, real_leds); } static int GetKbdLeds(InputInfoPtr pInfo) { int real_leds, leds = 0; ioctl(pInfo->fd, KDGETLED, &real_leds); if (real_leds & LED_CAP) leds |= XLED1; if (real_leds & LED_NUM) leds |= XLED2; if (real_leds & LED_SCR) leds |= XLED3; return(leds); } /* * NOTE: Only OpenServer Release 5.0.6 with Release Supplement 5.0.6A * and later have the required ioctl. 5.0.6A or higher is HIGHLY * recommended. The console driver is quite a different beast on that OS. */ #undef rate static void SetKbdRepeat(InputInfoPtr pInfo, char rad) { #if defined(KBIO_SETRATE) KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; int i; int value = 0x7f; /* Maximum delay with slowest rate */ int delay = 250; /* Default delay */ int rate = 300; /* Default repeat rate */ static int valid_rates[] = { 300, 267, 240, 218, 200, 185, 171, 160, 150, 133, 120, 109, 100, 92, 86, 80, 75, 67, 60, 55, 50, 46, 43, 40, 37, 33, 30, 27, 25, 23, 21, 20 }; #define RATE_COUNT (sizeof( valid_rates ) / sizeof( int )) static int valid_delays[] = { 250, 500, 750, 1000 }; #define DELAY_COUNT (sizeof( valid_delays ) / sizeof( int )) if (pKbd->rate >= 0) rate = pKbd->rate * 10; if (pKbd->delay >= 0) delay = pKbd->delay; for (i = 0; i < RATE_COUNT; i++) if (rate >= valid_rates[i]) { value &= 0x60; value |= i; break; } for (i = 0; i < DELAY_COUNT; i++) if (delay <= valid_delays[i]) { value &= 0x1f; value |= i << 5; break; } ioctl (pInfo->fd, KBIO_SETRATE, value); #endif /* defined(KBIO_SETRATE) */ } static int KbdInit(InputInfoPtr pInfo, int what) { KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; ScoKbdPrivPtr priv = (ScoKbdPrivPtr) pKbd->private; if (pKbd->isConsole) { priv->use_tcs = 1; priv->use_kd = 1; priv->no_nmap = 1; priv->no_emap = 1; priv->orig_getsc = 0; if (ioctl (pInfo->fd, TCGETSC, &priv->orig_getsc) < 0) priv->use_tcs = 0; if (ioctl (pInfo->fd, KDGKBMODE, &priv->orig_kbm) < 0) priv->use_kd = 0; if (!priv->use_tcs && !priv->use_kd) { xf86Msg (X_ERROR, "KbdInit: Could not determine keyboard mode\n"); return !Success; } /* * One day this should be fixed to translate normal ASCII characters * back into scancodes or into events that XFree86 wants, but not * now. For the time being, we only support scancode mode screens. */ if (priv->use_tcs && !(priv->orig_getsc & KB_ISSCANCODE)) { xf86Msg (X_ERROR, "KbdInit: Keyboard can not send scancodes\n"); return !Success; } /* * We need to get the original keyboard map and NUL out the lock * modifiers. This prevents the scancode API from messing with * the keyboard LED's. We restore the original map when we exit. */ if (ioctl (pInfo->fd, GIO_KEYMAP, &priv->keymap) < 0) { xf86Msg (X_ERROR, "KbdInit: Failed to get keyboard map (%s)\n", strerror(errno)); return !Success; } if (ioctl (pInfo->fd, GIO_KEYMAP, &priv->noledmap) < 0) { xf86Msg (X_ERROR, "KbdInit: Failed to get keyboard map (%s)\n", strerror(errno)); return !Success; } else { int i, j; for (i = 0; i < priv->noledmap.n_keys; i++) { for (j = 0; j < NUM_STATES; j++) { if (IS_SPECIAL(priv->noledmap, i, j) && ((priv->noledmap.key[i].map[j] == K_CLK) || (priv->noledmap.key[i].map[j] == K_NLK) || (priv->noledmap.key[i].map[j] == K_SLK))) { priv->noledmap.key[i].map[j] = K_NOP; } } } } if (ioctl (pInfo->fd, XCGETA, &priv->kbdtty) < 0) { xf86Msg (X_ERROR, "KbdInit: Failed to get terminal modes (%s)\n", strerror(errno)); return !Success; } priv->sc_mapbuf = xalloc (10*BSIZE); priv->sc_mapbuf2 = xalloc(10*BSIZE); /* Get the emap */ if (ioctl (pInfo->fd, LDGMAP, priv->sc_mapbuf) < 0) { if (errno != ENAVAIL) { xf86Msg (X_ERROR, "KbdInit: Failed to retrieve e-map (%s)\n", strerror (errno)); return !Success; } priv->no_emap = 0; } /* Get the nmap */ if (ioctl (pInfo->fd, NMGMAP, priv->sc_mapbuf2) < 0) { if (errno != ENAVAIL) { xf86Msg (X_ERROR, "KbdInit: Failed to retrieve n-map (%s)\n", strerror (errno)); return !Success; } priv->no_nmap = 0; } } /* End of if we are on a console */ return Success; } static int KbdOn(InputInfoPtr pInfo, int what) { KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; ScoKbdPrivPtr priv = (ScoKbdPrivPtr) pKbd->private; struct termios newtio; if (pKbd->isConsole) { ioctl (pInfo->fd, LDNMAP); /* Turn e-mapping off */ ioctl (pInfo->fd, NMNMAP); /* Turn n-mapping off */ newtio = priv->kbdtty; /* structure copy */ newtio.c_iflag = (IGNPAR | IGNBRK) & (~PARMRK) & (~ISTRIP); newtio.c_oflag = 0; newtio.c_cflag = CREAD | CS8 | B9600; newtio.c_lflag = 0; newtio.c_cc[VTIME]=0; newtio.c_cc[VMIN]=1; cfsetispeed(&newtio, 9600); cfsetospeed(&newtio, 9600); ioctl(pInfo->fd, XCSETA, &newtio); /* Now tell the keyboard driver to send us raw scancodes */ if (priv->use_tcs) { int nm = priv->orig_getsc; nm &= ~KB_XSCANCODE; ioctl (pInfo->fd, TCSETSC, &nm); } if (priv->use_kd) ioctl (pInfo->fd, KDSKBMODE, K_RAW); ioctl (pInfo->fd, PIO_KEYMAP, &priv->noledmap); } return Success; } static int KbdOff(InputInfoPtr pInfo, int what) { KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; ScoKbdPrivPtr priv = (ScoKbdPrivPtr) pKbd->private; if (pKbd->isConsole) { /* Revert back to original translate scancode mode */ if (priv->use_tcs) ioctl (pInfo->fd, TCSETSC, &priv->orig_getsc); if (priv->use_kd) ioctl (pInfo->fd, KDSKBMODE, priv->orig_kbm); ioctl (pInfo->fd, PIO_KEYMAP, &priv->keymap); if (priv->no_emap) ioctl (pInfo->fd, LDSMAP, priv->sc_mapbuf); if (priv->no_nmap) ioctl (pInfo->fd, NMSMAP, priv->sc_mapbuf2); ioctl(pInfo->fd, XCSETA, &priv->kbdtty); } return Success; } static int GetSpecialKey(InputInfoPtr pInfo, int scanCode) { KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; int specialkey = scanCode; if (pKbd->CustomKeycodes) { specialkey = pKbd->specialMap->map[scanCode]; } return specialkey; } #define ModifierSet(k) ((modifiers & (k)) == (k)) static Bool SpecialKey(InputInfoPtr pInfo, int key, Bool down, int modifiers) { KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; if(!pKbd->vtSwitchSupported) return FALSE; if ((!ModifierSet(ShiftMask)) && ((ModifierSet(ControlMask | AltMask)) || (ModifierSet(ControlMask | AltLangMask)))) { if (VTSwitchEnabled && !xf86Info.vtSysreq) { switch (key) { case KEY_F1: case KEY_F2: case KEY_F3: case KEY_F4: case KEY_F5: case KEY_F6: case KEY_F7: case KEY_F8: case KEY_F9: case KEY_F10: if (down) { int sts = key - KEY_F1; if (sts != xf86Info.vtno) { ioctl(pInfo->fd, VT_ACTIVATE, sts); } return TRUE; } case KEY_F11: case KEY_F12: if (down) { int sts = key - KEY_F11 + 10; if (sts != xf86Info.vtno) { ioctl(pInfo->fd, VT_ACTIVATE, sts); } return TRUE; } } } } #ifdef USE_VT_SYSREQ if (VTSwitchEnabled && xf86Info.vtSysreq) { switch (key) { case KEY_F1: case KEY_F2: case KEY_F3: case KEY_F4: case KEY_F5: case KEY_F6: case KEY_F7: case KEY_F8: case KEY_F9: case KEY_F10: if (VTSysreqToggle && down) { ioctl(pInfo->fd, VT_ACTIVATE, key - KEY_F1); VTSysreqToggle = FALSE; return TRUE; } break; case KEY_F11: case KEY_F12: if (VTSysreqToggle && down) { ioctl(pInfo->fd, VT_ACTIVATE, key - KEY_F11 + 10); VTSysreqToggle = FALSE; return TRUE; } break; /* Ignore these keys -- ie don't let them cancel an alt-sysreq */ case KEY_Alt: case KEY_AltLang: break; case KEY_SysReqest: if (!(ModifierSet(ShiftMask) || ModifierSet(ControlMask))) { if ((ModifierSet(AltMask) || ModifierSet(AltLangMask)) && down) VTSysreqToggle = TRUE; } break; default: /* * We only land here when Alt-SysReq is followed by a * non-switching key. */ if (VTSysreqToggle) VTSysreqToggle = FALSE; } } #endif /* USE_VT_SYSREQ */ return FALSE; } static void stdReadInput(InputInfoPtr pInfo) { KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; unsigned char rBuf[64]; int nBytes, i; if ((nBytes = read( pInfo->fd, (char *)rBuf, sizeof(rBuf))) > 0) { for (i = 0; i < nBytes; i++) { pKbd->PostEvent(pInfo, rBuf[i] & 0x7f, rBuf[i] & 0x80 ? FALSE : TRUE); } } } static Bool OpenKeyboard(InputInfoPtr pInfo) { KbdDevPtr pKbd = (KbdDevPtr) pInfo->private; int i; KbdProtocolId prot = PROT_UNKNOWN_KBD; char *s; s = xf86SetStrOption(pInfo->options, "Protocol", NULL); for (i = 0; protocols[i].name; i++) { if (xf86NameCmp(s, protocols[i].name) == 0) { prot = protocols[i].id; break; } } switch (prot) { case PROT_STD: pInfo->read_input = stdReadInput; break; default: xf86Msg(X_ERROR,"\"%s\" is not a valid keyboard protocol name\n", s); xfree(s); return FALSE; } xf86Msg(X_CONFIG, "%s: Protocol: %s\n", pInfo->name, s); xfree(s); s = xf86SetStrOption(pInfo->options, "Device", NULL); if (s == NULL) { pInfo->fd = xf86Info.consoleFd; pKbd->isConsole = TRUE; } else { pInfo->fd = open(s, O_RDONLY | O_NONBLOCK | O_EXCL); if (pInfo->fd == -1) { xf86Msg(X_ERROR, "%s: cannot open \"%s\"\n", pInfo->name, s); xfree(s); return FALSE; } pKbd->isConsole = FALSE; xfree(s); } if (pKbd->isConsole) pKbd->vtSwitchSupported = TRUE; return TRUE; } _X_EXPORT Bool xf86OSKbdPreInit(InputInfoPtr pInfo) { KbdDevPtr pKbd = pInfo->private; pKbd->KbdInit = KbdInit; pKbd->KbdOn = KbdOn; pKbd->KbdOff = KbdOff; pKbd->Bell = SoundBell; pKbd->SetLeds = SetKbdLeds; pKbd->GetLeds = GetKbdLeds; pKbd->SetKbdRepeat = SetKbdRepeat; pKbd->KbdGetMapping = KbdGetMapping; pKbd->SpecialKey = SpecialKey; pKbd->GetSpecialKey = GetSpecialKey; pKbd->OpenKeyboard = OpenKeyboard; pKbd->RemapScanCode = ATScancode; pKbd->vtSwitchSupported = FALSE; pKbd->private = xcalloc(sizeof(ScoKbdPrivRec), 1); if (pKbd->private == NULL) { xf86Msg(X_ERROR,"can't allocate keyboard OS private data\n"); return FALSE; } return TRUE; }