OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [mw/] [src/] [drivers/] [kbd_x11.c] - Rev 673

Go to most recent revision | Compare with Previous | Blame | View Log

/*
 * Copyright (c) 1999, 2000 Greg Haerr <greg@censoft.com>
 * Author: Tony Rogvall <tony@bluetail.com>
 *
 * Converted to scancode mode by Greg Haerr
 *
 * X11 Keyboard driver
 */
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/keysym.h>
#include <X11/XKBlib.h>
#include "device.h"
 
static int  X11_Open(KBDDEVICE *pkd);
static void X11_Close(void);
static void X11_GetModifierInfo(MWKEYMOD *modifiers, MWKEYMOD *curmodifiers);
static int  X11_Read(MWKEY *kbuf, MWKEYMOD *modifiers, MWSCANCODE *scancode);
 
static int init_modstate();
 
static MWKEYMOD key_modstate;
extern int escape_quits;
 
extern Display*     x11_dpy;
extern int          x11_scr;
extern Visual*      x11_vis;
extern Window       x11_win;
extern GC           x11_gc;
extern int          x11_setup_display();
 
#define X_SCR_MASK 0x80
#define X_CAP_MASK 0x2
#define X_NUM_MASK 0x10
 
KBDDEVICE kbddev = {
    X11_Open,
    X11_Close,
    X11_GetModifierInfo,
    X11_Read,
    NULL
};
 
/*
 * Open the keyboard.
 * This is real simple, we just use a special file handle
 * that allows non-blocking I/O, and put the terminal into
 * character mode.
 */
static int
X11_Open(KBDDEVICE *pkd)
{
    if (x11_setup_display() < 0)
	return -1;
 
    if(init_modstate() < 0)
    	return -1;
 
    /* return the x11 file descriptor for select */
    return ConnectionNumber(x11_dpy);  
}
 
/*
 * Close the keyboard.
 * This resets the terminal modes.
 */
static void
X11_Close(void)
{
    /* nop */
}
 
/*
 * Return the possible modifiers for the keyboard.
 */
static  void
X11_GetModifierInfo(MWKEYMOD *modifiers, MWKEYMOD *curmodifiers)
{
	if (modifiers)
		*modifiers = MWKMOD_SHIFT | MWKMOD_CTRL | MWKMOD_ALT;
	if (curmodifiers)
		*curmodifiers = key_modstate;
}
 
/*
 * This reads one keystroke from the keyboard, and the current state of
 * the modifier keys (ALT, SHIFT, etc).  Returns -1 on error, 0 if no data
 * is ready, 1 on a keypress, and 2 on keyrelease.
 * This is a non-blocking call.
 */
static int
X11_Read(MWKEY *kbuf, MWKEYMOD *modifiers, MWSCANCODE *scancode)
{
	XEvent ev;
	MWKEY mwkey;
 
	static int grabbed = 0;
	static int x11_accel_num;
	static int x11_accel_den;
	static int x11_thres;
 
    /* check if we have a KeyPressedEvent */
    if (XCheckMaskEvent(x11_dpy, KeyPressMask|KeyReleaseMask, &ev)) {
	KeySym sym = XKeycodeToKeysym(x11_dpy, ev.xkey.keycode, 0);
 
	if (sym == NoSymbol)
	    return -1;
 
	/* calculate kbd modifiers*/
	key_modstate &= (MWKMOD_NUM|MWKMOD_CAPS|MWKMOD_SCR); 
	if (ev.xkey.state & ControlMask)
		key_modstate |= MWKMOD_CTRL; 
	if (ev.xkey.state & ShiftMask)
		key_modstate |= MWKMOD_SHIFT;
	if (ev.xkey.state & Mod1Mask)
		key_modstate |= MWKMOD_ALT;
	if (ev.xkey.state & X_CAP_MASK)
		key_modstate |= MWKMOD_CAPS;
	if (ev.xkey.state & X_SCR_MASK)
		key_modstate |= MWKMOD_SCR;
	if (ev.xkey.state & X_NUM_MASK)
		key_modstate |= MWKMOD_NUM;
 
	if (sym == XK_Escape) {
	    mwkey = MWKEY_ESCAPE;
 
	    if (ev.xkey.state & ControlMask) {
		/* toggle grab control */
		if (grabbed) {
		    XUngrabPointer(x11_dpy, CurrentTime);
		    XUngrabKeyboard(x11_dpy, CurrentTime);
		    XChangePointerControl(x11_dpy, True, False, x11_accel_num, 
			  x11_accel_den, 0);
		    grabbed = 0;
		}
		else {
		    /* save pointer config */
		    XGetPointerControl(x11_dpy, &x11_accel_num, &x11_accel_den,
			    &x11_thres);
		    XChangePointerControl(x11_dpy, True, False, 1, 1, 0);
		    XGrabKeyboard(x11_dpy, x11_win,
			    True,  /* only to this window */
			    GrabModeAsync, GrabModeAsync, CurrentTime);
		    XGrabPointer(x11_dpy, x11_win, False,
				 PointerMotionMask | ButtonPressMask,
				 GrabModeAsync, GrabModeAsync, None, None,
				 CurrentTime);
		    grabbed = 1;
		}
		return 0;
	    } else if (grabbed)
		XChangePointerControl(x11_dpy, True, False, x11_accel_num, 
		      x11_accel_den, 0);
	    *kbuf = mwkey;
	    *modifiers = key_modstate;
	    *scancode = ev.xkey.keycode;
	    return (ev.xkey.type == KeyPress)? 1: 2;
	} else {
	    switch (sym) {
	    case XK_Delete:
		    mwkey = MWKEY_DELETE;
		    break;
	    case XK_Home:
		    mwkey = MWKEY_HOME;
		    break;
	    case XK_Left:
		    mwkey = MWKEY_LEFT;
		    break;
	    case XK_Up:
		    mwkey = MWKEY_UP;
		    break;
	    case XK_Right:
		    mwkey = MWKEY_RIGHT;
		    break;
	    case XK_Down:
		    mwkey = MWKEY_DOWN;
		    break;
	    case XK_Page_Up:
		    mwkey = MWKEY_PAGEUP;
		    break;
	    case XK_Page_Down:
		    mwkey = MWKEY_PAGEDOWN;
		    break;
	    case XK_End:
		    mwkey = MWKEY_END;
		    break;
	    case XK_Insert:
		    mwkey = MWKEY_INSERT;
		    break;
	    case XK_Pause:
	    case XK_Break:
		    mwkey = MWKEY_QUIT;
		    break;
	    case XK_Print:
	    case XK_Sys_Req:
		    mwkey = MWKEY_PRINT;
		    break;
	    case XK_Menu:
		    mwkey = MWKEY_MENU;
		    break;
	    case XK_Cancel:
		    mwkey = MWKEY_CANCEL;
		    break;
	    case XK_KP_Enter:
		    mwkey = MWKEY_KP_ENTER;
		    break;
	    case XK_KP_Home:
		    mwkey = MWKEY_KP7;
		    break;
	    case XK_KP_Left:
		    mwkey = MWKEY_KP4;
		    break;
	    case XK_KP_Up:
		    mwkey = MWKEY_KP8;
		    break;
	    case XK_KP_Right:
		    mwkey = MWKEY_KP6;
		    break;
	    case XK_KP_Down:
		    mwkey = MWKEY_KP2;
		    break;
	    case XK_KP_Page_Up:
		    mwkey = MWKEY_KP9;
		    break;
	    case XK_KP_Page_Down:
		    mwkey = MWKEY_KP3;
		    break;
	    case XK_KP_End:
		    mwkey = MWKEY_KP1;
		    break;
	    case XK_KP_Insert:
		    mwkey = MWKEY_KP0;
		    break;
	    case XK_KP_Delete:
		    mwkey = MWKEY_KP_PERIOD;
		    break;
	    case XK_KP_Equal:
		    mwkey = MWKEY_KP_EQUALS;
		    break;
	    case XK_KP_Multiply:
		    mwkey = MWKEY_KP_MULTIPLY;
		    break;
	    case XK_KP_Add:
		    mwkey = MWKEY_KP_PLUS;
		    break;
	    case XK_KP_Subtract:
		    mwkey = MWKEY_KP_MINUS;
		    break;
	    case XK_KP_Decimal:
		    mwkey = MWKEY_KP_PERIOD;
		    break;
	    case XK_KP_Divide:
		    mwkey = MWKEY_KP_DIVIDE;
		    break;
	    case XK_KP_5:
	    case XK_KP_Begin:
		    mwkey = MWKEY_KP5;
		    break;
	    case XK_F1:
		    mwkey = MWKEY_F1;
		    break;
	    case XK_F2:
		    mwkey = MWKEY_F2;
		    break;
	    case XK_F3:
		    mwkey = MWKEY_F3;
		    break;
	    case XK_F4:
		    mwkey = MWKEY_F4;
		    break;
	    case XK_F5:
		    mwkey = MWKEY_F5;
		    break;
	    case XK_F6:
		    mwkey = MWKEY_F6;
		    break;
	    case XK_F7:
		    mwkey = MWKEY_F7;
		    break;
	    case XK_F8:
		    mwkey = MWKEY_F8;
		    break;
	    case XK_F9:
		    mwkey = MWKEY_F9;
		    break;
	    case XK_F10:
		    mwkey = MWKEY_F10;
		    break;
	    case XK_F11:
		    mwkey = MWKEY_F11;
		    break;
	    case XK_F12:
		    mwkey = MWKEY_F12;
		    break;
 
	    /* state modifiers*/
	    case XK_Num_Lock:
		    /* not sent, used only for state*/
		    if (ev.xkey.type == KeyRelease)
		    	key_modstate ^= MWKMOD_NUM;
		    return 0;
	    case XK_Shift_Lock:
	    case XK_Caps_Lock:
		    /* not sent, used only for state*/
		    if (ev.xkey.type == KeyRelease)
		    	key_modstate ^= MWKMOD_CAPS;
		    return 0;
	    case XK_Scroll_Lock:
		    /* not sent, used only for state*/
		    if (ev.xkey.type == KeyRelease)
		    	key_modstate ^= MWKMOD_SCR;
		    return 0;
		    break;
	    case XK_Shift_L:
		    mwkey = MWKEY_LSHIFT;
		    break;
	    case XK_Shift_R:
		    mwkey = MWKEY_RSHIFT;
		    break;
	    case XK_Control_L:
		    mwkey = MWKEY_LCTRL;
		    break;
	    case XK_Control_R:
		    mwkey = MWKEY_RCTRL;
		    break;
	    case XK_Alt_L:
		    mwkey = MWKEY_LALT;
		    break;
	    case XK_Alt_R:
		    mwkey = MWKEY_RALT;
		    break;
	    case XK_Meta_L:
	    case XK_Super_L:
	    case XK_Hyper_L:
		    mwkey = MWKEY_LMETA;
		    break;
	    case XK_Meta_R:
	    case XK_Super_R:
	    case XK_Hyper_R:
		    mwkey = MWKEY_RMETA;
		    break;
	    default:
		    switch (sym) {
		    case XK_BackSpace:
		    case XK_Tab:
		    case XK_Return:
			break;
		    default:
		    	if (sym & 0xFF00)
			    fprintf(stderr, "Unhandled X11 keysym: %04x\n", (int)sym);
		    }
 
		    XLookupString(&ev.xkey, &mwkey, 1, &sym, NULL );
 
		    if (key_modstate & MWKMOD_CTRL)
				mwkey = sym & 0x1f;	/* Control code */ 
			else
				mwkey = sym & 0xff;	/* ASCII*/
 
		    break;
 
	    }
	    if (key_modstate & MWKMOD_NUM) {
		switch (mwkey) {
		case MWKEY_KP0:
		case MWKEY_KP1:
		case MWKEY_KP2:
		case MWKEY_KP3:
		case MWKEY_KP4:
		case MWKEY_KP5:
		case MWKEY_KP6:
		case MWKEY_KP7:
		case MWKEY_KP8:
		case MWKEY_KP9:
			mwkey = mwkey - MWKEY_KP0 + '0';
			break;
		case MWKEY_KP_PERIOD:
			mwkey = '.';
			break;
		case MWKEY_KP_DIVIDE:
			mwkey = '/';
			break;
		case MWKEY_KP_MULTIPLY:
			mwkey = '*';
			break;
		case MWKEY_KP_MINUS:
			mwkey = '-';
			break;
		case MWKEY_KP_PLUS:
			mwkey = '+';
			break;
		case MWKEY_KP_ENTER:
			mwkey = MWKEY_ENTER;
			break;
		case MWKEY_KP_EQUALS:
			mwkey = '-';
			break;
		}
	    }
	    *modifiers = key_modstate;
	    *scancode = ev.xkey.keycode;
	    *kbuf = mwkey;
 
	    //printf("mods: 0x%x  scan: 0x%x  key: 0x%x\n",*modifiers
	    //						,*scancode
	    //						,*kbuf);
	    return (ev.xkey.type == KeyPress)? 1 : 2;
	}
    }
    return 0;
}
 
#define NUM_LOCK_MASK    0x00000002
#define CAPS_LOCK_MASK   0x00000001
#define SCROLL_LOCK_MASK 0x00000004
 
/* initialise key_modstate */ 
static int init_modstate ()
{
	unsigned int state;
	int capsl, numl, scrolll;
 
	if(XkbGetIndicatorState (x11_dpy, XkbUseCoreKbd, &state) != Success) {
		fprintf(stderr, "Error reading Indicator status\n");
		return -1; 
	} 
	capsl = state & CAPS_LOCK_MASK;
	numl = state & NUM_LOCK_MASK;
	scrolll = state & SCROLL_LOCK_MASK;
 
	if(numl != 0)
		key_modstate |= MWKMOD_NUM;
 
	if(capsl != 0)
		key_modstate |= MWKMOD_CAPS;
 
	if(scrolll != 0)
		key_modstate |= MWKMOD_SCR;
 
	return 0;
}
 
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.