#include <m_pd.h>

#ifdef PD_MAJOR_VERSION
#include "s_stuff.h"
#else 
#include "m_imp.h"
#endif

#include "linuxhid.h"

#define LINUXMOUSE_DEVICE   "/dev/input/event0"
#define LINUXMOUSE_AXES     3

static char *version = "$Revision: 1.1 $";

/*------------------------------------------------------------------------------
 *  CLASS DEF
 */
static t_class *linuxmouse_class;

typedef struct _linuxmouse {
  t_object            x_obj;
  t_int               x_fd;
  t_symbol            *x_devname;
  int                 x_read_ok;
  int                 x_started;
#ifdef __gnu_linux__
  struct input_event  x_input_event; 
#endif
  t_outlet            *x_axis_out[LINUXMOUSE_AXES];
  t_outlet            *x_button_num_out;
  t_outlet            *x_button_val_out;
  unsigned char       x_buttons;
  unsigned char       x_axes;
} t_linuxmouse;

/*------------------------------------------------------------------------------
 * IMPLEMENTATION                    
 */

/* Actions */
void linuxmouse_stop(t_linuxmouse* x) {
	DEBUG(post("linuxmouse_stop"););

#ifdef __gnu_linux__
   if (x->x_fd >= 0 && x->x_started) { 
		sys_rmpollfn(x->x_fd);
		post("[linuxmouse] stopped");
		x->x_started = 0;
	} 
#endif
}

static int linuxmouse_close(t_linuxmouse *x) {
	DEBUG(post("linuxmouse_close"););

/* just to be safe, stop it first */
	linuxmouse_stop(x);
	
   if (x->x_fd < 0) {
		return 0;
	}
	else {		
		close (x->x_fd);
		post ("[linuxmouse] closed %s",x->x_devname->s_name);
		return 1;
	}
}

static int linuxmouse_open(t_linuxmouse *x, t_symbol *s) {
	int eventType, eventCode;
	char devicename[256] = "Unknown";
#ifdef __gnu_linux__
	unsigned long bitmask[EV_MAX][NBITS(KEY_MAX)];
#endif
	
	DEBUG(post("linuxmouse_open"););
	
	linuxmouse_close(x);
	
	/* set obj device name to parameter otherwise set to default */  
	if ( s != &s_ )  
		x->x_devname = s;
	
#ifdef __gnu_linux__	
	/* open device */
	if (x->x_devname) {
		/* open the device read-only, non-exclusive */
		x->x_fd = open (x->x_devname->s_name, O_RDONLY | O_NONBLOCK);
		/* test if device open */
		if (x->x_fd < 0 ) { 
			post("[linuxmouse] open %s failed",x->x_devname->s_name);
			x->x_fd = -1;
			return 0;
		}
	} else {
		post("[linuxmouse] no device set: %s",x->x_devname->s_name);
		return 1;
	}
	
/* read input_events from the LINUXMOUSE_DEVICE stream 
 * It seems that is just there to flush the event input buffer?
 */
	while (read (x->x_fd, &(x->x_input_event), sizeof(struct input_event)) > -1);
	
	/* get name of device */
	ioctl(x->x_fd, EVIOCGNAME(sizeof(devicename)), devicename);
	post ("Configuring %s on %s.",devicename,x->x_devname->s_name);
	post("\nSupported events:");

	/* get bitmask representing supported events (axes, buttons, etc.) */
	memset(bitmask, 0, sizeof(bitmask));
	ioctl(x->x_fd, EVIOCGBIT(0, EV_MAX), bitmask[0]);
	
	x->x_axes = 0;
	x->x_buttons = 0;
	
	/* cycle through all possible event types */
	for (eventType = 0; eventType < EV_MAX; eventType++) {
		if (test_bit(eventType, bitmask[0])) {
			post(" %s (type %d) ", events[eventType] ? events[eventType] : "?", eventType);
			//	post("Event type %d",eventType);
			
			/* get bitmask representing supported button types */
			ioctl(x->x_fd, EVIOCGBIT(eventType, KEY_MAX), bitmask[eventType]);
			
			/* cycle through all possible event codes (axes, keys, etc.) 
			 * testing to see which are supported  
			 */
			for (eventCode = 0; eventCode < KEY_MAX; eventCode++) {				
				if (test_bit(eventCode, bitmask[eventType])) {
					post("    Event code %d (%s)", eventCode, names[eventType] ? (names[eventType][eventCode] ? names[eventType][eventCode] : "?") : "?");
					
					if ( eventType == EV_KEY ) 
						x->x_buttons++;
					else if  ( eventType == EV_REL ) 
						x->x_axes++;
				}
			}
		}        
	}
	
	post ("\nUsing %d axes and %d buttons.", x->x_axes, x->x_buttons);
	post ("\nWARNING * WARNING * WARNING * WARNING * WARNING * WARNING * WARNING");
	post ("This object is under development!  The interface could change at anytime!");
	post ("As I write cross-platform versions, the interface might have to change.");
	post ("WARNING * WARNING * WARNING * WARNING * WARNING * WARNING * WARNING\n");
#endif
	
	return 1;
}

static int linuxmouse_read(t_linuxmouse *x,int fd) {
	int axis_num = 0;
	t_float button_num = 0;
	
	if (x->x_fd < 0) return 0;
	
#ifdef __gnu_linux__
	while (read (x->x_fd, &(x->x_input_event), sizeof(struct input_event)) > -1) {
		if  ( x->x_input_event.type == EV_REL ) {
			/* Relative Axes Event Type */
			switch ( x->x_input_event.code ) {
				case REL_X:
					axis_num = 0;
					break;
				case REL_Y:
					axis_num = 1;
					break;
				case REL_WHEEL:
					axis_num = 2;
					break;
			}
			outlet_float (x->x_axis_out[axis_num], (int)x->x_input_event.value);	
		}
		else if ( x->x_input_event.type == EV_KEY ) {
			/* key/button event type */
			switch ( x->x_input_event.code ) {
				case BTN_LEFT:
					button_num = 0;
					break;
				case BTN_RIGHT:
					button_num = 1;
					break;
				case BTN_MIDDLE:
					button_num = 2;
					break;
				case BTN_SIDE:
					button_num = 3;
					break;
				case BTN_EXTRA:
					button_num = 4;
					break;
				case BTN_FORWARD:
					button_num = 5;
					break;
				case BTN_BACK:
					button_num = 6;
					break;
			}
			outlet_float (x->x_button_val_out, x->x_input_event.value);
			outlet_float (x->x_button_num_out, button_num);
		}
 	}
#endif

	return 1;    
}

void linuxmouse_start(t_linuxmouse* x) {
	DEBUG(post("linuxmouse_start"););

/* if the device isn't open already, open it */
/* (I'll test this later -HCS) */
/*    if (x->x_fd < 0) linuxmouse_open(x,&s_); */
		
#ifdef __gnu_linux__
   if (x->x_fd >= 0 && !x->x_started) {
		sys_addpollfn(x->x_fd, (t_fdpollfn)linuxmouse_read, x);
		post("[linuxmouse] started");
		x->x_started = 1;
	} else {
		post("You need to set a input device (i.e /dev/input/event0)");
	}
#endif
}

/* setup functions */
static void linuxmouse_free(t_linuxmouse* x) {
	DEBUG(post("linuxmouse_free"););
    
	if (x->x_fd < 0) return;
	linuxmouse_stop(x);	
	close (x->x_fd);
}

static void *linuxmouse_new(t_symbol *s) {
	int i;
	t_linuxmouse *x = (t_linuxmouse *)pd_new(linuxmouse_class);
	
	DEBUG(post("linuxmouse_new"););
	
	post("[linuxmouse] %s, written by Hans-Christoph Steiner <hans@eds.org>",version);  
#ifndef __gnu_linux__
	post("    !! WARNING !! WARNING !! WARNING !! WARNING !! WARNING !! WARNING !!");
	post("     This is a dummy, since this object only works with a Linux kernel!");
	post("    !! WARNING !! WARNING !! WARNING !! WARNING !! WARNING !! WARNING !!");
#endif
	
	/* init vars */
	x->x_fd = -1;
	x->x_read_ok = 1;
	x->x_started = 0;
	x->x_devname = gensym("/dev/input/event0");

	/* create outlets for each axis */
	for (i = 0; i < LINUXMOUSE_AXES; i++) 
		x->x_axis_out[i] = outlet_new(&x->x_obj, &s_float);
	
	/* create outlets for buttons */
	x->x_button_num_out = outlet_new(&x->x_obj, &s_float);
	x->x_button_val_out = outlet_new(&x->x_obj, &s_float);

	if (!linuxmouse_open(x,s)) return x;
	
	return (x);
}

void linuxmouse_setup(void) {
	DEBUG(post("linuxmouse_setup"););
	linuxmouse_class = class_new(gensym("linuxmouse"), 
										  (t_newmethod)linuxmouse_new, 
										  (t_method)linuxmouse_free,
										  sizeof(t_linuxmouse), 0, A_DEFSYM, 0);
	
	/* add inlet message methods */
	class_addmethod(linuxmouse_class,(t_method) linuxmouse_open,gensym("open"),A_DEFSYM,0);
	class_addmethod(linuxmouse_class,(t_method) linuxmouse_close,gensym("close"),0);
	class_addmethod(linuxmouse_class,(t_method) linuxmouse_start,gensym("start"),0);
	class_addmethod(linuxmouse_class,(t_method) linuxmouse_stop,gensym("stop"),0);	
	class_addmethod(linuxmouse_class,(t_method) linuxmouse_start,gensym("poll"),0);
	class_addmethod(linuxmouse_class,(t_method) linuxmouse_stop,gensym("nopoll"),0);	
}