/* * Copyright © 2004-2008 Red Hat, Inc. * * 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 of Red Hat * not be used in advertising or publicity pertaining to distribution * of the software without specific, written prior permission. Red * Hat makes no representations about the suitability of this software * for any purpose. It is provided "as is" without express or implied * warranty. * * THE AUTHORS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN * NO EVENT SHALL THE AUTHORS 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. * * Authors: * Kristian Høgsberg (krh@redhat.com) * Adam Jackson (ajax@redhat.com) */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include #include #include #include #include #include #include "evdev.h" #include #include #include /* 2.4 compatibility */ #ifndef EVIOCGRAB #define EVIOCGRAB _IOW('E', 0x90, int) #endif #ifndef BTN_TASK #define BTN_TASK 0x117 #endif #ifndef EV_SYN #define EV_SYN EV_RST #endif /* end compat */ #define ArrayLength(a) (sizeof(a) / (sizeof((a)[0]))) /* evdev flags */ #define EVDEV_KEYBOARD_EVENTS (1 << 0) #define EVDEV_BUTTON_EVENTS (1 << 1) #define EVDEV_RELATIVE_EVENTS (1 << 2) #define EVDEV_ABSOLUTE_EVENTS (1 << 3) #define EVDEV_TOUCHPAD (1 << 4) #define MIN_KEYCODE 8 #define GLYPHS_PER_KEY 2 #define AltMask Mod1Mask #define NumLockMask Mod2Mask #define AltLangMask Mod3Mask #define KanaMask Mod4Mask #define ScrollLockMask Mod5Mask #define CAPSFLAG 1 #define NUMFLAG 2 #define SCROLLFLAG 4 #define MODEFLAG 8 #define COMPOSEFLAG 16 static const char *evdevDefaults[] = { "XkbRules", "base", "XkbModel", "evdev", "XkbLayout", "us", NULL }; static void SetXkbOption(InputInfoPtr pInfo, char *name, char **option) { char *s; if ((s = xf86SetStrOption(pInfo->options, name, NULL))) { if (!s[0]) { xfree(s); *option = NULL; } else { *option = s; xf86Msg(X_CONFIG, "%s: %s: \"%s\"\n", pInfo->name, name, s); } } } static int wheel_up_button = 4; static int wheel_down_button = 5; static int wheel_left_button = 6; static int wheel_right_button = 7; static void PostButtonClicks(InputInfoPtr pInfo, int button, int count) { int i; for (i = 0; i < count; i++) { xf86PostButtonEvent(pInfo->dev, 0, button, 1, 0, 0); xf86PostButtonEvent(pInfo->dev, 0, button, 0, 0, 0); } } static void PostKbdEvent(InputInfoPtr pInfo, struct input_event *ev, int value) { /* filter repeat events for chording keys */ if (value == 2 && (ev->code == KEY_LEFTCTRL || ev->code == KEY_RIGHTCTRL || ev->code == KEY_LEFTSHIFT || ev->code == KEY_RIGHTSHIFT || ev->code == KEY_LEFTALT || ev->code == KEY_RIGHTALT || ev->code == KEY_LEFTMETA || ev->code == KEY_RIGHTMETA || ev->code == KEY_CAPSLOCK || ev->code == KEY_NUMLOCK || ev->code == KEY_SCROLLLOCK)) /* XXX windows keys? */ return; xf86PostKeyboardEvent(pInfo->dev, ev->code + MIN_KEYCODE, value); } static void EvdevReadInput(InputInfoPtr pInfo) { struct input_event ev; int len, value; int dx, dy; unsigned int abs; EvdevPtr pEvdev = pInfo->private; dx = 0; dy = 0; abs = 0; while (xf86WaitForInput (pInfo->fd, 0) > 0) { len = read(pInfo->fd, &ev, sizeof ev); if (len != sizeof ev) { /* The kernel promises that we always only read a complete * event, so len != sizeof ev is an error. */ xf86Msg(X_ERROR, "Read error: %s\n", strerror(errno)); break; } /* Get the signed value, earlier kernels had this as unsigned */ value = ev.value; switch (ev.type) { case EV_REL: switch (ev.code) { case REL_X: dx += value; break; case REL_Y: dy += value; break; case REL_WHEEL: if (value > 0) PostButtonClicks(pInfo, wheel_up_button, value); if (value < 0) PostButtonClicks(pInfo, wheel_down_button, -value); break; case REL_DIAL: case REL_HWHEEL: if (value > 0) PostButtonClicks(pInfo, wheel_right_button, value); if (value < 0) PostButtonClicks(pInfo, wheel_left_button, -value); break; } break; case EV_ABS: switch (ev.code) { case ABS_X: pEvdev->abs_x = value; abs = 1; break; case ABS_Y: pEvdev->abs_y = value; abs = 1; break; } break; case EV_KEY: /* don't repeat mouse buttons */ if (ev.code >= BTN_MOUSE && ev.code < KEY_OK) if (value == 2) break; switch (ev.code) { /* swap here, pretend we're an X-conformant device. */ case BTN_LEFT: if (!EvdevMBEmuFilterEvent(pInfo, ev.code, value)) xf86PostButtonEvent(pInfo->dev, 0, 1, value, 0, 0); break; case BTN_RIGHT: if (!EvdevMBEmuFilterEvent(pInfo, ev.code, value)) xf86PostButtonEvent(pInfo->dev, 0, 3, value, 0, 0); break; case BTN_MIDDLE: EvdevMBEmuEnable(pInfo, FALSE); xf86PostButtonEvent(pInfo->dev, 0, 2, value, 0, 0); break; case BTN_SIDE: case BTN_EXTRA: case BTN_FORWARD: case BTN_BACK: case BTN_TASK: xf86PostButtonEvent(pInfo->dev, 0, ev.code - BTN_LEFT + 5, value, 0, 0); break; case BTN_TOUCH: case BTN_TOOL_PEN: case BTN_TOOL_RUBBER: case BTN_TOOL_BRUSH: case BTN_TOOL_PENCIL: case BTN_TOOL_AIRBRUSH: case BTN_TOOL_FINGER: case BTN_TOOL_MOUSE: case BTN_TOOL_LENS: pEvdev->tool = value ? ev.code : 0; break; default: if (ev.code > BTN_TASK && ev.code < KEY_OK) break; PostKbdEvent(pInfo, &ev, value); break; } break; case EV_SYN: break; } } /* convert to relative motion for touchpads */ if (pEvdev->flags & EVDEV_TOUCHPAD) { abs = 0; if (pEvdev->tool) { /* meaning, touch is active */ if (pEvdev->old_x != -1) dx = pEvdev->abs_x - pEvdev->old_x; if (pEvdev->old_y != -1) dy = pEvdev->abs_y - pEvdev->old_y; pEvdev->old_x = pEvdev->abs_x; pEvdev->old_y = pEvdev->abs_y; } else { pEvdev->old_x = pEvdev->old_y = -1; } } if (dx != 0 || dy != 0) xf86PostMotionEvent(pInfo->dev, FALSE, 0, 2, dx, dy); /* * Some devices only generate valid abs coords when BTN_DIGI is * pressed. On wacom tablets, this means that the pen is in * proximity of the tablet. After the pen is removed, BTN_DIGI is * released, and a (0, 0) absolute event is generated. Checking * pEvdev->digi here, lets us ignore that event. pEvdev is * initialized to 1 so devices that doesn't use this scheme still * just works. */ if (abs && pEvdev->tool) { xf86PostMotionEvent(pInfo->dev, TRUE, 0, 2, pEvdev->abs_x, pEvdev->abs_y); } } #define TestBit(bit, array) (array[(bit) / 8] & (1 << ((bit) % 8))) static void EvdevPtrCtrlProc(DeviceIntPtr device, PtrCtrl *ctrl) { /* Nothing to do, dix handles all settings */ } /* FIXME: this map works with evdev keyboards, but all the xkb maps * probably don't. The easiest is to remap the event keycodes. */ static KeySym map[] = { /* 0x00 */ NoSymbol, NoSymbol, /* 0x01 */ XK_Escape, NoSymbol, /* 0x02 */ XK_1, XK_exclam, /* 0x03 */ XK_2, XK_at, /* 0x04 */ XK_3, XK_numbersign, /* 0x05 */ XK_4, XK_dollar, /* 0x06 */ XK_5, XK_percent, /* 0x07 */ XK_6, XK_asciicircum, /* 0x08 */ XK_7, XK_ampersand, /* 0x09 */ XK_8, XK_asterisk, /* 0x0a */ XK_9, XK_parenleft, /* 0x0b */ XK_0, XK_parenright, /* 0x0c */ XK_minus, XK_underscore, /* 0x0d */ XK_equal, XK_plus, /* 0x0e */ XK_BackSpace, NoSymbol, /* 0x0f */ XK_Tab, XK_ISO_Left_Tab, /* 0x10 */ XK_Q, NoSymbol, /* 0x11 */ XK_W, NoSymbol, /* 0x12 */ XK_E, NoSymbol, /* 0x13 */ XK_R, NoSymbol, /* 0x14 */ XK_T, NoSymbol, /* 0x15 */ XK_Y, NoSymbol, /* 0x16 */ XK_U, NoSymbol, /* 0x17 */ XK_I, NoSymbol, /* 0x18 */ XK_O, NoSymbol, /* 0x19 */ XK_P, NoSymbol, /* 0x1a */ XK_bracketleft, XK_braceleft, /* 0x1b */ XK_bracketright,XK_braceright, /* 0x1c */ XK_Return, NoSymbol, /* 0x1d */ XK_Control_L, NoSymbol, /* 0x1e */ XK_A, NoSymbol, /* 0x1f */ XK_S, NoSymbol, /* 0x20 */ XK_D, NoSymbol, /* 0x21 */ XK_F, NoSymbol, /* 0x22 */ XK_G, NoSymbol, /* 0x23 */ XK_H, NoSymbol, /* 0x24 */ XK_J, NoSymbol, /* 0x25 */ XK_K, NoSymbol, /* 0x26 */ XK_L, NoSymbol, /* 0x27 */ XK_semicolon, XK_colon, /* 0x28 */ XK_quoteright, XK_quotedbl, /* 0x29 */ XK_quoteleft, XK_asciitilde, /* 0x2a */ XK_Shift_L, NoSymbol, /* 0x2b */ XK_backslash, XK_bar, /* 0x2c */ XK_Z, NoSymbol, /* 0x2d */ XK_X, NoSymbol, /* 0x2e */ XK_C, NoSymbol, /* 0x2f */ XK_V, NoSymbol, /* 0x30 */ XK_B, NoSymbol, /* 0x31 */ XK_N, NoSymbol, /* 0x32 */ XK_M, NoSymbol, /* 0x33 */ XK_comma, XK_less, /* 0x34 */ XK_period, XK_greater, /* 0x35 */ XK_slash, XK_question, /* 0x36 */ XK_Shift_R, NoSymbol, /* 0x37 */ XK_KP_Multiply, NoSymbol, /* 0x38 */ XK_Alt_L, XK_Meta_L, /* 0x39 */ XK_space, NoSymbol, /* 0x3a */ XK_Caps_Lock, NoSymbol, /* 0x3b */ XK_F1, NoSymbol, /* 0x3c */ XK_F2, NoSymbol, /* 0x3d */ XK_F3, NoSymbol, /* 0x3e */ XK_F4, NoSymbol, /* 0x3f */ XK_F5, NoSymbol, /* 0x40 */ XK_F6, NoSymbol, /* 0x41 */ XK_F7, NoSymbol, /* 0x42 */ XK_F8, NoSymbol, /* 0x43 */ XK_F9, NoSymbol, /* 0x44 */ XK_F10, NoSymbol, /* 0x45 */ XK_Num_Lock, NoSymbol, /* 0x46 */ XK_Scroll_Lock, NoSymbol, /* These KP keys should have the KP_7 keysyms in the numlock * modifer... ? */ /* 0x47 */ XK_KP_Home, XK_KP_7, /* 0x48 */ XK_KP_Up, XK_KP_8, /* 0x49 */ XK_KP_Prior, XK_KP_9, /* 0x4a */ XK_KP_Subtract, NoSymbol, /* 0x4b */ XK_KP_Left, XK_KP_4, /* 0x4c */ XK_KP_Begin, XK_KP_5, /* 0x4d */ XK_KP_Right, XK_KP_6, /* 0x4e */ XK_KP_Add, NoSymbol, /* 0x4f */ XK_KP_End, XK_KP_1, /* 0x50 */ XK_KP_Down, XK_KP_2, /* 0x51 */ XK_KP_Next, XK_KP_3, /* 0x52 */ XK_KP_Insert, XK_KP_0, /* 0x53 */ XK_KP_Delete, XK_KP_Decimal, /* 0x54 */ NoSymbol, NoSymbol, /* 0x55 */ XK_F13, NoSymbol, /* 0x56 */ XK_less, XK_greater, /* 0x57 */ XK_F11, NoSymbol, /* 0x58 */ XK_F12, NoSymbol, /* 0x59 */ XK_F14, NoSymbol, /* 0x5a */ XK_F15, NoSymbol, /* 0x5b */ XK_F16, NoSymbol, /* 0x5c */ XK_F17, NoSymbol, /* 0x5d */ XK_F18, NoSymbol, /* 0x5e */ XK_F19, NoSymbol, /* 0x5f */ XK_F20, NoSymbol, /* 0x60 */ XK_KP_Enter, NoSymbol, /* 0x61 */ XK_Control_R, NoSymbol, /* 0x62 */ XK_KP_Divide, NoSymbol, /* 0x63 */ XK_Print, XK_Sys_Req, /* 0x64 */ XK_Alt_R, XK_Meta_R, /* 0x65 */ NoSymbol, NoSymbol, /* KEY_LINEFEED */ /* 0x66 */ XK_Home, NoSymbol, /* 0x67 */ XK_Up, NoSymbol, /* 0x68 */ XK_Prior, NoSymbol, /* 0x69 */ XK_Left, NoSymbol, /* 0x6a */ XK_Right, NoSymbol, /* 0x6b */ XK_End, NoSymbol, /* 0x6c */ XK_Down, NoSymbol, /* 0x6d */ XK_Next, NoSymbol, /* 0x6e */ XK_Insert, NoSymbol, /* 0x6f */ XK_Delete, NoSymbol, /* 0x70 */ NoSymbol, NoSymbol, /* KEY_MACRO */ /* 0x71 */ NoSymbol, NoSymbol, /* 0x72 */ NoSymbol, NoSymbol, /* 0x73 */ NoSymbol, NoSymbol, /* 0x74 */ NoSymbol, NoSymbol, /* 0x75 */ XK_KP_Equal, NoSymbol, /* 0x76 */ NoSymbol, NoSymbol, /* 0x77 */ NoSymbol, NoSymbol, /* 0x78 */ XK_F21, NoSymbol, /* 0x79 */ XK_F22, NoSymbol, /* 0x7a */ XK_F23, NoSymbol, /* 0x7b */ XK_F24, NoSymbol, /* 0x7c */ XK_KP_Separator, NoSymbol, /* 0x7d */ XK_Meta_L, NoSymbol, /* 0x7e */ XK_Meta_R, NoSymbol, /* 0x7f */ XK_Multi_key, NoSymbol, /* 0x80 */ NoSymbol, NoSymbol, /* 0x81 */ NoSymbol, NoSymbol, /* 0x82 */ NoSymbol, NoSymbol, /* 0x83 */ NoSymbol, NoSymbol, /* 0x84 */ NoSymbol, NoSymbol, /* 0x85 */ NoSymbol, NoSymbol, /* 0x86 */ NoSymbol, NoSymbol, /* 0x87 */ NoSymbol, NoSymbol, /* 0x88 */ NoSymbol, NoSymbol, /* 0x89 */ NoSymbol, NoSymbol, /* 0x8a */ NoSymbol, NoSymbol, /* 0x8b */ NoSymbol, NoSymbol, /* 0x8c */ NoSymbol, NoSymbol, /* 0x8d */ NoSymbol, NoSymbol, /* 0x8e */ NoSymbol, NoSymbol, /* 0x8f */ NoSymbol, NoSymbol, /* 0x90 */ NoSymbol, NoSymbol, /* 0x91 */ NoSymbol, NoSymbol, /* 0x92 */ NoSymbol, NoSymbol, /* 0x93 */ NoSymbol, NoSymbol, /* 0x94 */ NoSymbol, NoSymbol, /* 0x95 */ NoSymbol, NoSymbol, /* 0x96 */ NoSymbol, NoSymbol, /* 0x97 */ NoSymbol, NoSymbol, /* 0x98 */ NoSymbol, NoSymbol, /* 0x99 */ NoSymbol, NoSymbol, /* 0x9a */ NoSymbol, NoSymbol, /* 0x9b */ NoSymbol, NoSymbol, /* 0x9c */ NoSymbol, NoSymbol, /* 0x9d */ NoSymbol, NoSymbol, /* 0x9e */ NoSymbol, NoSymbol, /* 0x9f */ NoSymbol, NoSymbol, /* 0xa0 */ NoSymbol, NoSymbol, /* 0xa1 */ NoSymbol, NoSymbol, /* 0xa2 */ NoSymbol, NoSymbol, /* 0xa3 */ NoSymbol, NoSymbol, /* 0xa4 */ NoSymbol, NoSymbol, /* 0xa5 */ NoSymbol, NoSymbol, /* 0xa6 */ NoSymbol, NoSymbol, /* 0xa7 */ NoSymbol, NoSymbol, /* 0xa8 */ NoSymbol, NoSymbol, /* 0xa9 */ NoSymbol, NoSymbol, /* 0xaa */ NoSymbol, NoSymbol, /* 0xab */ NoSymbol, NoSymbol, /* 0xac */ NoSymbol, NoSymbol, /* 0xad */ NoSymbol, NoSymbol, /* 0xae */ NoSymbol, NoSymbol, /* 0xaf */ NoSymbol, NoSymbol, /* 0xb0 */ NoSymbol, NoSymbol, /* 0xb1 */ NoSymbol, NoSymbol, /* 0xb2 */ NoSymbol, NoSymbol, /* 0xb3 */ NoSymbol, NoSymbol, /* 0xb4 */ NoSymbol, NoSymbol, /* 0xb5 */ NoSymbol, NoSymbol, /* 0xb6 */ NoSymbol, NoSymbol, /* 0xb7 */ NoSymbol, NoSymbol, /* 0xb8 */ NoSymbol, NoSymbol, /* 0xb9 */ NoSymbol, NoSymbol, /* 0xba */ NoSymbol, NoSymbol, /* 0xbb */ NoSymbol, NoSymbol, /* 0xbc */ NoSymbol, NoSymbol, /* 0xbd */ NoSymbol, NoSymbol, /* 0xbe */ NoSymbol, NoSymbol, /* 0xbf */ NoSymbol, NoSymbol, /* 0xc0 */ NoSymbol, NoSymbol, /* 0xc1 */ NoSymbol, NoSymbol, /* 0xc2 */ NoSymbol, NoSymbol, /* 0xc3 */ NoSymbol, NoSymbol, /* 0xc4 */ NoSymbol, NoSymbol, /* 0xc5 */ NoSymbol, NoSymbol, /* 0xc6 */ NoSymbol, NoSymbol, /* 0xc7 */ NoSymbol, NoSymbol, /* 0xc8 */ NoSymbol, NoSymbol, /* 0xc9 */ NoSymbol, NoSymbol, /* 0xca */ NoSymbol, NoSymbol, /* 0xcb */ NoSymbol, NoSymbol, /* 0xcc */ NoSymbol, NoSymbol, /* 0xcd */ NoSymbol, NoSymbol, /* 0xce */ NoSymbol, NoSymbol, /* 0xcf */ NoSymbol, NoSymbol, /* 0xd0 */ NoSymbol, NoSymbol, /* 0xd1 */ NoSymbol, NoSymbol, /* 0xd2 */ NoSymbol, NoSymbol, /* 0xd3 */ NoSymbol, NoSymbol, /* 0xd4 */ NoSymbol, NoSymbol, /* 0xd5 */ NoSymbol, NoSymbol, /* 0xd6 */ NoSymbol, NoSymbol, /* 0xd7 */ NoSymbol, NoSymbol, /* 0xd8 */ NoSymbol, NoSymbol, /* 0xd9 */ NoSymbol, NoSymbol, /* 0xda */ NoSymbol, NoSymbol, /* 0xdb */ NoSymbol, NoSymbol, /* 0xdc */ NoSymbol, NoSymbol, /* 0xdd */ NoSymbol, NoSymbol, /* 0xde */ NoSymbol, NoSymbol, /* 0xdf */ NoSymbol, NoSymbol, /* 0xe0 */ NoSymbol, NoSymbol, /* 0xe1 */ NoSymbol, NoSymbol, /* 0xe2 */ NoSymbol, NoSymbol, /* 0xe3 */ NoSymbol, NoSymbol, /* 0xe4 */ NoSymbol, NoSymbol, /* 0xe5 */ NoSymbol, NoSymbol, /* 0xe6 */ NoSymbol, NoSymbol, /* 0xe7 */ NoSymbol, NoSymbol, /* 0xe8 */ NoSymbol, NoSymbol, /* 0xe9 */ NoSymbol, NoSymbol, /* 0xea */ NoSymbol, NoSymbol, /* 0xeb */ NoSymbol, NoSymbol, /* 0xec */ NoSymbol, NoSymbol, /* 0xed */ NoSymbol, NoSymbol, /* 0xee */ NoSymbol, NoSymbol, /* 0xef */ NoSymbol, NoSymbol, /* 0xf0 */ NoSymbol, NoSymbol, /* 0xf1 */ NoSymbol, NoSymbol, /* 0xf2 */ NoSymbol, NoSymbol, /* 0xf3 */ NoSymbol, NoSymbol, /* 0xf4 */ NoSymbol, NoSymbol, /* 0xf5 */ NoSymbol, NoSymbol, /* 0xf6 */ NoSymbol, NoSymbol, /* 0xf7 */ NoSymbol, NoSymbol, }; static void EvdevKbdCtrl(DeviceIntPtr device, KeybdCtrl *ctrl) { static struct { int xbit, code; } bits[] = { { CAPSFLAG, LED_CAPSL }, { NUMFLAG, LED_NUML }, { SCROLLFLAG, LED_SCROLLL }, { MODEFLAG, LED_KANA }, { COMPOSEFLAG, LED_COMPOSE } }; InputInfoPtr pInfo; struct input_event ev[ArrayLength(bits)]; int i; pInfo = device->public.devicePrivate; for (i = 0; i < ArrayLength(bits); i++) { ev[i].type = EV_LED; ev[i].code = bits[i].code; ev[i].value = (ctrl->leds & bits[i].xbit) > 0; } write(pInfo->fd, ev, sizeof ev); } static int EvdevAddKeyClass(DeviceIntPtr device) { InputInfoPtr pInfo; KeySymsRec keySyms; CARD8 modMap[MAP_LENGTH]; KeySym sym; int i, j; EvdevPtr pEvdev; static struct { KeySym keysym; CARD8 mask; } modifiers[] = { { XK_Shift_L, ShiftMask }, { XK_Shift_R, ShiftMask }, { XK_Control_L, ControlMask }, { XK_Control_R, ControlMask }, { XK_Caps_Lock, LockMask }, { XK_Alt_L, AltMask }, { XK_Alt_R, AltMask }, { XK_Meta_L, Mod4Mask }, { XK_Meta_R, Mod4Mask }, { XK_Num_Lock, NumLockMask }, { XK_Scroll_Lock, ScrollLockMask }, { XK_Mode_switch, AltLangMask } }; /* TODO: * Ctrl-Alt-Backspace and other Ctrl-Alt-stuff should work * Get keyboard repeat under control (right now caps lock repeats!) */ pInfo = device->public.devicePrivate; pEvdev = pInfo->private; /* Compute the modifier map */ memset(modMap, 0, sizeof modMap); for (i = 0; i < ArrayLength(map) / GLYPHS_PER_KEY; i++) { sym = map[i * GLYPHS_PER_KEY]; for (j = 0; j < ArrayLength(modifiers); j++) { if (modifiers[j].keysym == sym) modMap[i + MIN_KEYCODE] = modifiers[j].mask; } } keySyms.map = map; keySyms.mapWidth = GLYPHS_PER_KEY; keySyms.minKeyCode = MIN_KEYCODE; keySyms.maxKeyCode = MIN_KEYCODE + ArrayLength(map) / GLYPHS_PER_KEY - 1; #ifdef XKB if (pEvdev->noXkb) #endif { xf86Msg(X_CONFIG, "XKB: disabled\n"); if (!InitKeyboardDeviceStruct((DevicePtr)device, &keySyms, modMap, NULL, EvdevKbdCtrl)) return !Success; } #ifdef XKB else { SetXkbOption(pInfo, "xkb_rules", &pEvdev->xkb_rules); if (!pEvdev->xkb_rules) SetXkbOption(pInfo, "XkbRules", &pEvdev->xkb_rules); /* sorry, no model change allowed for you */ xf86ReplaceStrOption(pInfo->options, "xkb_model", "evdev"); SetXkbOption(pInfo, "xkb_model", &pEvdev->xkb_model); SetXkbOption(pInfo, "xkb_layout", &pEvdev->xkb_layout); if (!pEvdev->xkb_layout) SetXkbOption(pInfo, "XkbLayout", &pEvdev->xkb_layout); SetXkbOption(pInfo, "xkb_variant", &pEvdev->xkb_variant); if (!pEvdev->xkb_variant) SetXkbOption(pInfo, "XkbVariant", &pEvdev->xkb_variant); SetXkbOption(pInfo, "xkb_options", &pEvdev->xkb_options); if (!pEvdev->xkb_options) SetXkbOption(pInfo, "XkbOptions", &pEvdev->xkb_options); XkbSetRulesDflts(pEvdev->xkb_rules, pEvdev->xkb_model, pEvdev->xkb_layout, pEvdev->xkb_variant, pEvdev->xkb_options); if (!XkbInitKeyboardDeviceStruct(device, &pEvdev->xkbnames, &keySyms, modMap, NULL, EvdevKbdCtrl)) return !Success; } #endif pInfo->flags |= XI86_KEYBOARD_CAPABLE; return Success; } static int EvdevAddAbsClass(DeviceIntPtr device) { InputInfoPtr pInfo; EvdevPtr pEvdev; struct input_absinfo absinfo_x, absinfo_y; pInfo = device->public.devicePrivate; pEvdev = pInfo->private; if (ioctl(pInfo->fd, EVIOCGABS(ABS_X), &absinfo_x) < 0) { xf86Msg(X_ERROR, "ioctl EVIOCGABS failed: %s\n", strerror(errno)); return !Success; } if (ioctl(pInfo->fd, EVIOCGABS(ABS_Y), &absinfo_y) < 0) { xf86Msg(X_ERROR, "ioctl EVIOCGABS failed: %s\n", strerror(errno)); return !Success; } pEvdev->min_x = absinfo_x.minimum; pEvdev->max_x = absinfo_x.maximum; pEvdev->min_y = absinfo_y.minimum; pEvdev->max_y = absinfo_y.maximum; if (!InitValuatorClassDeviceStruct(device, 2, #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 3 GetMotionHistory, #endif GetMotionHistorySize(), Absolute)) return !Success; /* X valuator */ xf86InitValuatorAxisStruct(device, 0, pEvdev->min_x, pEvdev->max_x, 10000, 0, 10000); xf86InitValuatorDefaults(device, 0); /* Y valuator */ xf86InitValuatorAxisStruct(device, 1, pEvdev->min_y, pEvdev->max_y, 10000, 0, 10000); xf86InitValuatorDefaults(device, 1); xf86MotionHistoryAllocate(pInfo); if (!InitPtrFeedbackClassDeviceStruct(device, EvdevPtrCtrlProc)) return !Success; pInfo->flags |= XI86_POINTER_CAPABLE; return Success; } static int EvdevAddRelClass(DeviceIntPtr device) { InputInfoPtr pInfo; pInfo = device->public.devicePrivate; if (!InitValuatorClassDeviceStruct(device, 2, #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 3 GetMotionHistory, #endif GetMotionHistorySize(), Relative)) return !Success; /* X valuator */ xf86InitValuatorAxisStruct(device, 0, -1, -1, 1, 0, 1); xf86InitValuatorDefaults(device, 0); /* Y valuator */ xf86InitValuatorAxisStruct(device, 1, -1, -1, 1, 0, 1); xf86InitValuatorDefaults(device, 1); xf86MotionHistoryAllocate(pInfo); if (!InitPtrFeedbackClassDeviceStruct(device, EvdevPtrCtrlProc)) return !Success; pInfo->flags |= XI86_POINTER_CAPABLE; xf86MotionHistoryAllocate(pInfo); return Success; } static int EvdevAddButtonClass(DeviceIntPtr device) { CARD8 map[32]; InputInfoPtr pInfo; int i; pInfo = device->public.devicePrivate; /* FIXME: count number of actual buttons */ for (i = 0; i < ArrayLength(map); i++) map[i] = i; if (!InitButtonClassDeviceStruct(device, ArrayLength(map), map)) return !Success; return Success; } static int EvdevInit(DeviceIntPtr device) { InputInfoPtr pInfo; EvdevPtr pEvdev; pInfo = device->public.devicePrivate; pEvdev = pInfo->private; /* FIXME: This doesn't add buttons for keyboards with scrollwheels. */ if (pEvdev->flags & EVDEV_KEYBOARD_EVENTS) EvdevAddKeyClass(device); if (pEvdev->flags & EVDEV_BUTTON_EVENTS) EvdevAddButtonClass(device); /* We don't allow relative and absolute axes on the same device. Reason Reason being that some devices (MS Optical Desktop 2000) register both rel and abs axes for x/y. The abs axes register min/max, this min/max then also applies to the relative device (the mouse) and caps it at 0..255 for both axis. So unless you have a small screen, you won't be enjoying it much. FIXME: somebody volunteer to fix this. */ if (pEvdev->flags & EVDEV_RELATIVE_EVENTS) EvdevAddRelClass(device); else if (pEvdev->flags & EVDEV_ABSOLUTE_EVENTS) EvdevAddAbsClass(device); return Success; } static int EvdevProc(DeviceIntPtr device, int what) { InputInfoPtr pInfo; EvdevPtr pEvdev; pInfo = device->public.devicePrivate; pEvdev = pInfo->private; switch (what) { case DEVICE_INIT: return EvdevInit(device); case DEVICE_ON: if (!pEvdev->kernel24 && ioctl(pInfo->fd, EVIOCGRAB, (void *)1)) xf86Msg(X_WARNING, "%s: Grab failed (%s)\n", pInfo->name, strerror(errno)); xf86AddEnabledDevice(pInfo); device->public.on = TRUE; break; case DEVICE_OFF: if (!pEvdev->kernel24 && ioctl(pInfo->fd, EVIOCGRAB, (void *)0)) xf86Msg(X_WARNING, "%s: Release failed (%s)\n", pInfo->name, strerror(errno)); xf86RemoveEnabledDevice(pInfo); device->public.on = FALSE; break; case DEVICE_CLOSE: xf86Msg(X_INFO, "%s: Close\n", pInfo->name); break; } return Success; } static Bool EvdevConvert(InputInfoPtr pInfo, int first, int num, int v0, int v1, int v2, int v3, int v4, int v5, int *x, int *y) { EvdevPtr pEvdev = pInfo->private; int screenWidth = screenInfo.screens[pEvdev->screen]->width; int screenHeight = screenInfo.screens[pEvdev->screen]->height; if (first != 0 || num != 2) return FALSE; /* on absolute touchpads, don't warp on initial touch */ if (pEvdev->flags & EVDEV_TOUCHPAD) { *x = v0; *y = v0; return TRUE; } *x = (v0 - pEvdev->min_x) * screenWidth / (pEvdev->max_x - pEvdev->min_x); *y = (v1 - pEvdev->min_y) * screenHeight / (pEvdev->max_y - pEvdev->min_y); return TRUE; } static int EvdevProbe(InputInfoPtr pInfo) { char key_bitmask[(KEY_MAX + 7) / 8]; char rel_bitmask[(REL_MAX + 7) / 8]; char abs_bitmask[(ABS_MAX + 7) / 8]; int i, has_axes, has_buttons, has_keys; EvdevPtr pEvdev = pInfo->private; if (ioctl(pInfo->fd, EVIOCGRAB, (void *)1)) { if (errno == EINVAL) { /* keyboards are unsafe in 2.4 */ pEvdev->kernel24 = 1; } else { xf86Msg(X_ERROR, "Grab failed. Device already configured?\n"); return 1; } } else { ioctl(pInfo->fd, EVIOCGRAB, (void *)0); } if (ioctl(pInfo->fd, EVIOCGBIT(EV_REL, sizeof(rel_bitmask)), rel_bitmask) < 0) { xf86Msg(X_ERROR, "ioctl EVIOCGBIT failed: %s\n", strerror(errno)); return 1; } if (ioctl(pInfo->fd, EVIOCGBIT(EV_ABS, sizeof(abs_bitmask)), abs_bitmask) < 0) { xf86Msg(X_ERROR, "ioctl EVIOCGBIT failed: %s\n", strerror(errno)); return 1; } if (ioctl(pInfo->fd, EVIOCGBIT(EV_KEY, sizeof(key_bitmask)), key_bitmask) < 0) { xf86Msg(X_ERROR, "ioctl EVIOCGBIT failed: %s\n", strerror(errno)); return 1; } has_axes = FALSE; has_buttons = FALSE; has_keys = FALSE; if (TestBit(REL_X, rel_bitmask) && TestBit(REL_Y, rel_bitmask)) { xf86Msg(X_INFO, "%s: Found x and y relative axes\n", pInfo->name); pEvdev->flags |= EVDEV_RELATIVE_EVENTS; has_axes = TRUE; } if (TestBit(ABS_X, abs_bitmask) && TestBit(ABS_Y, abs_bitmask)) { xf86Msg(X_INFO, "%s: Found x and y absolute axes\n", pInfo->name); pEvdev->flags |= EVDEV_ABSOLUTE_EVENTS; if (TestBit(BTN_TOUCH, key_bitmask)) { xf86Msg(X_INFO, "%s: Found absolute touchpad\n", pInfo->name); pEvdev->flags |= EVDEV_TOUCHPAD; pEvdev->old_x = pEvdev->old_y = -1; } has_axes = TRUE; } if (TestBit(BTN_LEFT, key_bitmask)) { xf86Msg(X_INFO, "%s: Found mouse buttons\n", pInfo->name); pEvdev->flags |= EVDEV_BUTTON_EVENTS; has_buttons = TRUE; } for (i = 0; i < BTN_MISC; i++) if (TestBit(i, key_bitmask)) break; if (i < BTN_MISC) { xf86Msg(X_INFO, "%s: Found keys\n", pInfo->name); pEvdev->flags |= EVDEV_KEYBOARD_EVENTS; has_keys = TRUE; } if (has_axes && has_buttons) { xf86Msg(X_INFO, "%s: Configuring as mouse\n", pInfo->name); pInfo->flags |= XI86_POINTER_CAPABLE | XI86_SEND_DRAG_EVENTS | XI86_CONFIGURED; pInfo->type_name = XI_MOUSE; } if (has_keys) { if (pEvdev->kernel24) { xf86Msg(X_INFO, "%s: Kernel < 2.6 is too old, ignoring keyboard\n", pInfo->name); } else { xf86Msg(X_INFO, "%s: Configuring as keyboard\n", pInfo->name); pInfo->flags |= XI86_KEYBOARD_CAPABLE | XI86_CONFIGURED; pInfo->type_name = XI_KEYBOARD; } } if ((pInfo->flags & XI86_CONFIGURED) == 0) { xf86Msg(X_WARNING, "%s: Don't know how to use device\n", pInfo->name); return 1; } return 0; } static InputInfoPtr EvdevPreInit(InputDriverPtr drv, IDevPtr dev, int flags) { InputInfoPtr pInfo; MessageType deviceFrom = X_CONFIG; const char *device; EvdevPtr pEvdev; if (!(pInfo = xf86AllocateInput(drv, 0))) return NULL; /* Initialise the InputInfoRec. */ pInfo->name = dev->identifier; pInfo->flags = 0; pInfo->type_name = "UNKNOWN"; pInfo->device_control = EvdevProc; pInfo->read_input = EvdevReadInput; pInfo->history_size = 0; pInfo->control_proc = NULL; pInfo->close_proc = NULL; pInfo->switch_mode = NULL; pInfo->conversion_proc = EvdevConvert; pInfo->reverse_conversion_proc = NULL; pInfo->dev = NULL; pInfo->private_flags = 0; pInfo->always_core_feedback = 0; pInfo->conf_idev = dev; if (!(pEvdev = xcalloc(sizeof(EvdevRec), 1))) return pInfo; pInfo->private = pEvdev; xf86CollectInputOptions(pInfo, evdevDefaults, NULL); xf86ProcessCommonOptions(pInfo, pInfo->options); pEvdev->screen = xf86SetIntOption(pInfo->options, "ScreenNumber", 0); /* * We initialize pEvdev->tool to 1 so that device that doesn't use * proximity will still report events. */ pEvdev->tool = 1; EvdevMBEmuPreInit(pInfo); device = xf86CheckStrOption(dev->commonOptions, "Path", NULL); if (!device) device = xf86CheckStrOption(dev->commonOptions, "Device", NULL); if (!device) { xf86Msg(X_ERROR, "%s: No device specified.\n", pInfo->name); xf86DeleteInput(pInfo, 0); return NULL; } xf86Msg(deviceFrom, "%s: Device: \"%s\"\n", pInfo->name, device); do { pInfo->fd = open(device, O_RDWR, 0); } while (pInfo->fd < 0 && errno == EINTR); if (pInfo->fd < 0) { xf86Msg(X_ERROR, "Unable to open evdev device \"%s\".\n", device); xf86DeleteInput(pInfo, 0); return NULL; } pEvdev->noXkb = noXkbExtension; /* parse the XKB options during kbd setup */ if (EvdevProbe(pInfo)) { xf86DeleteInput(pInfo, 0); return NULL; } return pInfo; } _X_EXPORT InputDriverRec EVDEV = { 1, "evdev", NULL, EvdevPreInit, NULL, NULL, 0 }; static void EvdevUnplug(pointer p) { } static pointer EvdevPlug(pointer module, pointer options, int *errmaj, int *errmin) { xf86AddInputDriver(&EVDEV, module, 0); return module; } static XF86ModuleVersionInfo EvdevVersionRec = { "evdev", MODULEVENDORSTRING, MODINFOSTRING1, MODINFOSTRING2, 0, /* Missing from SDK: XORG_VERSION_CURRENT, */ 1, 0, 0, ABI_CLASS_XINPUT, ABI_XINPUT_VERSION, MOD_CLASS_XINPUT, {0, 0, 0, 0} }; _X_EXPORT XF86ModuleData evdevModuleData = { &EvdevVersionRec, EvdevPlug, EvdevUnplug };