linux-input.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
@ 2010-04-07 20:18 David Härdeman
  2010-04-07 21:01 ` Jon Smirl
                   ` (3 more replies)
  0 siblings, 4 replies; 13+ messages in thread
From: David Härdeman @ 2010-04-07 20:18 UTC (permalink / raw)
  To: mchehab; +Cc: linux-input, linux-media

drivers/media/IR/ir-raw-event.c is currently written with the assumption 
that all "raw" hardware will generate events only on state change (i.e.  
when a pulse or space starts).

However, some hardware (like mceusb, probably the most popular IR receiver
out there) only generates duration data (and that data is buffered so using
any kind of timing on the data is futile).

Furthermore, using signed int's to represent pulse/space durations in ms
is a well-known approach to anyone with experience in writing ir decoders.

This patch (which has been tested this time) is still a RFC on my proposed
interface changes.

Changes since last version:

o RC5x and NECx support no longer added in patch (separate patches to follow)

o The use of a kfifo has been left given feedback from Jon, Andy, Mauro

o The RX decoding is now handled via a workqueue (I can break that up into a
  separate patch later, but I think it helps the discussion to have it in for
  now), with inspiration from Andy's code.

o Separate reset operations are no longer added to decoders, a duration of
  zero is instead used to signal a reset (which allows the reset request to
  be inserted into the kfifo).

o Not sent using quilt...Mauro, does it still trip up your MUA?

Not changed:

o int's are still used to represent pulse/space durations in ms. Mauro and I
  seem to disagree on this one but I'm right :)

Index: ir/drivers/media/IR/ir-raw-event.c
===================================================================
--- ir.orig/drivers/media/IR/ir-raw-event.c	2010-04-06 12:16:27.000000000 +0200
+++ ir/drivers/media/IR/ir-raw-event.c	2010-04-07 21:32:13.961850481 +0200
@@ -15,13 +15,14 @@
 #include <media/ir-core.h>
 #include <linux/workqueue.h>
 #include <linux/spinlock.h>
+#include <linux/sched.h>
 
 /* Define the max number of bit transitions per IR keycode */
 #define MAX_IR_EVENT_SIZE	256
 
 /* Used to handle IR raw handler extensions */
 static LIST_HEAD(ir_raw_handler_list);
-static spinlock_t ir_raw_handler_lock;
+static DEFINE_SPINLOCK(ir_raw_handler_lock);
 
 /**
  * RUN_DECODER()	- runs an operation on all IR decoders
@@ -53,19 +54,30 @@
 /* Used to load the decoders */
 static struct work_struct wq_load;
 
+static void ir_raw_event_work(struct work_struct *work)
+{
+	int d;
+	struct ir_raw_event_ctrl *raw =
+		container_of(work, struct ir_raw_event_ctrl, rx_work);
+
+	while (kfifo_out(&raw->kfifo, &d, sizeof(d)) == sizeof(d))
+		RUN_DECODER(decode, raw->input_dev, d);
+}
+
 int ir_raw_event_register(struct input_dev *input_dev)
 {
 	struct ir_input_dev *ir = input_get_drvdata(input_dev);
-	int rc, size;
+	int rc;
 
 	ir->raw = kzalloc(sizeof(*ir->raw), GFP_KERNEL);
 	if (!ir->raw)
 		return -ENOMEM;
 
-	size = sizeof(struct ir_raw_event) * MAX_IR_EVENT_SIZE * 2;
-	size = roundup_pow_of_two(size);
+	ir->raw->input_dev = input_dev;
+	INIT_WORK(&ir->raw->rx_work, ir_raw_event_work);
 
-	rc = kfifo_alloc(&ir->raw->kfifo, size, GFP_KERNEL);
+	rc = kfifo_alloc(&ir->raw->kfifo, sizeof(int) * MAX_IR_EVENT_SIZE,
+			 GFP_KERNEL);
 	if (rc < 0) {
 		kfree(ir->raw);
 		ir->raw = NULL;
@@ -91,6 +103,7 @@
 	if (!ir->raw)
 		return;
 
+	cancel_work_sync(&ir->raw->rx_work);
 	RUN_DECODER(raw_unregister, input_dev);
 
 	kfifo_free(&ir->raw->kfifo);
@@ -99,74 +112,85 @@
 }
 EXPORT_SYMBOL_GPL(ir_raw_event_unregister);
 
-int ir_raw_event_store(struct input_dev *input_dev, enum raw_event_type type)
+/**
+ * ir_raw_event_store() - pass a pulse/space duration to the raw ir decoders
+ * @input_dev:	the struct input_dev device descriptor
+ * @duration:	duration of the pulse or space
+ *
+ * This routine passes a pulse/space duration to the raw ir decoding state
+ * machines. Pulses are signalled as positive values and spaces as negative
+ * values. A zero value will reset the decoding state machines.
+ */
+int ir_raw_event_store(struct input_dev *input_dev, int duration)
 {
-	struct ir_input_dev	*ir = input_get_drvdata(input_dev);
-	struct timespec		ts;
-	struct ir_raw_event	event;
-	int			rc;
+	struct ir_input_dev *ir = input_get_drvdata(input_dev);
 
 	if (!ir->raw)
 		return -EINVAL;
 
-	event.type = type;
-	event.delta.tv_sec = 0;
-	event.delta.tv_nsec = 0;
+	if (kfifo_in(&ir->raw->kfifo, &duration, sizeof(duration)) != sizeof(duration))
+		return -ENOMEM;
 
-	ktime_get_ts(&ts);
+	return 0;
+}
+EXPORT_SYMBOL_GPL(ir_raw_event_store);
 
-	if (timespec_equal(&ir->raw->last_event, &event.delta))
-		event.type |= IR_START_EVENT;
-	else
-		event.delta = timespec_sub(ts, ir->raw->last_event);
+/**
+ * ir_raw_event_store_edge() - notify raw ir decoders of the start of a pulse/space
+ * @input_dev:	the struct input_dev device descriptor
+ * @type:	the type of the event that has occurred
+ *
+ * This routine is used to notify the raw ir decoders on the beginning of an
+ * ir pulse or space (or the start/end of ir reception). This is used by
+ * hardware which does not provide durations directly but only interrupts
+ * (or similar events) on state change.
+ */
+int ir_raw_event_store_edge(struct input_dev *input_dev, enum raw_event_type type)
+{
+	struct ir_input_dev	*ir = input_get_drvdata(input_dev);
+	ktime_t			now;
+	s64			delta; /* us */
+	int			rc = 0;
 
-	memcpy(&ir->raw->last_event, &ts, sizeof(ts));
+	if (!ir->raw)
+		return -EINVAL;
 
-	if (event.delta.tv_sec) {
-		event.type |= IR_START_EVENT;
-		event.delta.tv_sec = 0;
-		event.delta.tv_nsec = 0;
-	}
+	now = ktime_get();
+	delta = ktime_us_delta(now, ir->raw->last_event);
 
-	kfifo_in(&ir->raw->kfifo, &event, sizeof(event));
+	/* Are we called for the first time? */
+	if (!ir->raw->last_type)
+		type |= IR_START_EVENT;
+
+	if (type & IR_START_EVENT)
+		ir_raw_event_reset(input_dev);
+	else if (ir->raw->last_type & IR_SPACE)
+		rc = ir_raw_event_store(input_dev, (int)-delta);
+	else if (ir->raw->last_type & IR_PULSE)
+		rc = ir_raw_event_store(input_dev, (int)delta);
+	else
+		return 0;
 
+	ir->raw->last_event = now;
+	ir->raw->last_type = type;
 	return rc;
 }
-EXPORT_SYMBOL_GPL(ir_raw_event_store);
+EXPORT_SYMBOL_GPL(ir_raw_event_store_edge);
 
-int ir_raw_event_handle(struct input_dev *input_dev)
+/**
+ * ir_raw_event_handle() - schedules the decoding of stored ir data
+ * @input_dev:	the struct input_dev device descriptor
+ *
+ * This routine will signal the workqueue to start decoding stored ir data.
+ */
+void ir_raw_event_handle(struct input_dev *input_dev)
 {
-	struct ir_input_dev		*ir = input_get_drvdata(input_dev);
-	int				rc;
-	struct ir_raw_event		ev;
-	int 				len, i;
-
-	/*
-	 * Store the events into a temporary buffer. This allows calling more than
-	 * one decoder to deal with the received data
-	 */
-	len = kfifo_len(&ir->raw->kfifo) / sizeof(ev);
-	if (!len)
-		return 0;
-
-	for (i = 0; i < len; i++) {
-		rc = kfifo_out(&ir->raw->kfifo, &ev, sizeof(ev));
-		if (rc != sizeof(ev)) {
-			IR_dprintk(1, "overflow error: received %d instead of %zd\n",
-				   rc, sizeof(ev));
-			return -EINVAL;
-		}
-		IR_dprintk(2, "event type %d, time before event: %07luus\n",
-			ev.type, (ev.delta.tv_nsec + 500) / 1000);
-		rc = RUN_DECODER(decode, input_dev, &ev);
-	}
+	struct ir_input_dev *ir = input_get_drvdata(input_dev);
 
-	/*
-	 * Call all ir decoders. This allows decoding the same event with
-	 * more than one protocol handler.
-	 */
+	if (!ir->raw)
+		return;
 
-	return rc;
+	schedule_work(&ir->raw->rx_work);
 }
 EXPORT_SYMBOL_GPL(ir_raw_event_handle);
 
@@ -205,8 +229,6 @@
 
 void ir_raw_init(void)
 {
-	spin_lock_init(&ir_raw_handler_lock);
-
 #ifdef MODULE
 	INIT_WORK(&wq_load, init_decoders);
 	schedule_work(&wq_load);
Index: ir/include/media/ir-core.h
===================================================================
--- ir.orig/include/media/ir-core.h	2010-04-06 12:16:27.000000000 +0200
+++ ir/include/media/ir-core.h	2010-04-07 18:17:53.524850074 +0200
@@ -57,14 +57,12 @@
 	void		(*close)(void *priv);
 };
 
-struct ir_raw_event {
-	struct timespec		delta;	/* Time spent before event */
-	enum raw_event_type	type;	/* event type */
-};
-
 struct ir_raw_event_ctrl {
-	struct kfifo			kfifo;		/* fifo for the pulse/space events */
-	struct timespec			last_event;	/* when last event occurred */
+	struct work_struct		rx_work;	/* for the rx decoding workqueue */
+	struct kfifo			kfifo;		/* fifo for the pulse/space durations */
+	ktime_t				last_event;	/* when last event occurred */
+	enum raw_event_type		last_type;	/* last event type */
+	struct input_dev		*input_dev;	/* pointer to the parent input_dev */
 };
 
 struct ir_input_dev {
@@ -89,8 +87,7 @@
 struct ir_raw_handler {
 	struct list_head list;
 
-	int (*decode)(struct input_dev *input_dev,
-		      struct ir_raw_event *ev);
+	int (*decode)(struct input_dev *input_dev, int duration);
 	int (*raw_register)(struct input_dev *input_dev);
 	int (*raw_unregister)(struct input_dev *input_dev);
 };
@@ -146,8 +143,13 @@
 /* Routines from ir-raw-event.c */
 int ir_raw_event_register(struct input_dev *input_dev);
 void ir_raw_event_unregister(struct input_dev *input_dev);
-int ir_raw_event_store(struct input_dev *input_dev, enum raw_event_type type);
-int ir_raw_event_handle(struct input_dev *input_dev);
+void ir_raw_event_handle(struct input_dev *input_dev);
+int ir_raw_event_store(struct input_dev *input_dev, int duration);
+int ir_raw_event_store_edge(struct input_dev *input_dev, enum raw_event_type type);
+static inline void ir_raw_event_reset(struct input_dev *dev) {
+	ir_raw_event_store(input_dev, 0);
+	ir_raw_event_handle(input_dev);
+}
 int ir_raw_handler_register(struct ir_raw_handler *ir_raw_handler);
 void ir_raw_handler_unregister(struct ir_raw_handler *ir_raw_handler);
 void ir_raw_init(void);
Index: ir/drivers/media/IR/ir-nec-decoder.c
===================================================================
--- ir.orig/drivers/media/IR/ir-nec-decoder.c	2010-04-06 12:16:27.000000000 +0200
+++ ir/drivers/media/IR/ir-nec-decoder.c	2010-04-07 18:32:10.892853025 +0200
@@ -13,23 +13,23 @@
  */
 
 #include <media/ir-core.h>
+#include <linux/bitrev.h>
 
 #define NEC_NBITS		32
-#define NEC_UNIT		559979 /* ns */
-#define NEC_HEADER_MARK		(16 * NEC_UNIT)
-#define NEC_HEADER_SPACE	(8 * NEC_UNIT)
-#define NEC_REPEAT_SPACE	(4 * NEC_UNIT)
-#define NEC_MARK		(NEC_UNIT)
-#define NEC_0_SPACE		(NEC_UNIT)
-#define NEC_1_SPACE		(3 * NEC_UNIT)
+#define NEC_UNIT		562	/* us */
+#define NEC_HEADER_MARK		16
+#define NEC_HEADER_SPACE	-8
+#define NEC_REPEAT_SPACE	-4
+#define NEC_MARK		1
+#define NEC_0_SPACE		-1
+#define NEC_1_SPACE		-3
 
 /* Used to register nec_decoder clients */
 static LIST_HEAD(decoder_list);
-static spinlock_t decoder_lock;
+static DEFINE_SPINLOCK(decoder_lock);
 
 enum nec_state {
 	STATE_INACTIVE,
-	STATE_HEADER_MARK,
 	STATE_HEADER_SPACE,
 	STATE_MARK,
 	STATE_SPACE,
@@ -37,13 +37,6 @@
 	STATE_TRAILER_SPACE,
 };
 
-struct nec_code {
-	u8	address;
-	u8	not_address;
-	u8	command;
-	u8	not_command;
-};
-
 struct decoder_data {
 	struct list_head	list;
 	struct ir_input_dev	*ir_dev;
@@ -51,7 +44,7 @@
 
 	/* State machine control */
 	enum nec_state		state;
-	struct nec_code		nec_code;
+	u32			nec_bits;
 	unsigned		count;
 };
 
@@ -62,7 +55,6 @@
  *
  * Returns the struct decoder_data that corresponds to a device
  */
-
 static struct decoder_data *get_decoder_data(struct  ir_input_dev *ir_dev)
 {
 	struct decoder_data *data = NULL;
@@ -123,20 +115,20 @@
 	.attrs	= decoder_attributes,
 };
 
-
 /**
  * ir_nec_decode() - Decode one NEC pulse or space
  * @input_dev:	the struct input_dev descriptor of the device
- * @ev:		event array with type/duration of pulse/space
+ * @duration:	duration in us of pulse/space
  *
  * This function returns -EINVAL if the pulse violates the state machine
  */
-static int ir_nec_decode(struct input_dev *input_dev,
-			 struct ir_raw_event *ev)
+static int ir_nec_decode(struct input_dev *input_dev, int duration)
 {
 	struct decoder_data *data;
 	struct ir_input_dev *ir_dev = input_get_drvdata(input_dev);
-	int bit, last_bit;
+	int d;
+	u32 scancode;
+	u8 address, not_address, command, not_command;
 
 	data = get_decoder_data(ir_dev);
 	if (!data)
@@ -145,150 +137,106 @@
 	if (!data->enabled)
 		return 0;
 
-	/* Except for the initial event, what matters is the previous bit */
-	bit = (ev->type & IR_PULSE) ? 1 : 0;
-
-	last_bit = !bit;
-
-	/* Discards spurious space last_bits when inactive */
-
-	/* Very long delays are considered as start events */
-	if (ev->delta.tv_nsec > NEC_HEADER_MARK + NEC_HEADER_SPACE - NEC_UNIT / 2)
+	if (duration == 0) {
 		data->state = STATE_INACTIVE;
+		return 0;
+	}
 
-	if (ev->type & IR_START_EVENT)
-		data->state = STATE_INACTIVE;
+	d = DIV_ROUND_CLOSEST(abs(duration), NEC_UNIT);
+	if (duration < 0)
+		d = -d;
+
+	IR_dprintk(2, "NEC decode started at state %d (%i units, %ius)\n",
+		   data->state, d, duration);
 
 	switch (data->state) {
-	case STATE_INACTIVE:
-		if (!bit)		/* PULSE marks the start event */
-			return 0;
 
-		data->count = 0;
-		data->state = STATE_HEADER_MARK;
-		memset (&data->nec_code, 0, sizeof(data->nec_code));
-		return 0;
-	case STATE_HEADER_MARK:
-		if (!last_bit)
-			goto err;
-		if (ev->delta.tv_nsec < NEC_HEADER_MARK - 6 * NEC_UNIT)
-			goto err;
-		data->state = STATE_HEADER_SPACE;
+	case STATE_INACTIVE:
+		if (d == NEC_HEADER_MARK) {
+			data->count = 0;
+			data->state = STATE_HEADER_SPACE;
+		}
 		return 0;
+
 	case STATE_HEADER_SPACE:
-		if (last_bit)
-			goto err;
-		if (ev->delta.tv_nsec >= NEC_HEADER_SPACE - NEC_UNIT / 2) {
+		if (d == NEC_HEADER_SPACE) {
 			data->state = STATE_MARK;
 			return 0;
-		}
-
-		if (ev->delta.tv_nsec >= NEC_REPEAT_SPACE - NEC_UNIT / 2) {
+		} else if (d == NEC_REPEAT_SPACE) {
 			ir_repeat(input_dev);
 			IR_dprintk(1, "Repeat last key\n");
 			data->state = STATE_TRAILER_MARK;
 			return 0;
 		}
-		goto err;
+		break;
+
 	case STATE_MARK:
-		if (!last_bit)
-			goto err;
-		if ((ev->delta.tv_nsec > NEC_MARK + NEC_UNIT / 2) ||
-		    (ev->delta.tv_nsec < NEC_MARK - NEC_UNIT / 2))
-			goto err;
-		data->state = STATE_SPACE;
-		return 0;
+		if (d == NEC_MARK) {
+			data->state = STATE_SPACE;
+			return 0;
+		}
+		break;
+
 	case STATE_SPACE:
-		if (last_bit)
-			goto err;
+		if (d != NEC_0_SPACE && d != NEC_1_SPACE)
+			break;
 
-		if ((ev->delta.tv_nsec >= NEC_0_SPACE - NEC_UNIT / 2) &&
-		    (ev->delta.tv_nsec < NEC_0_SPACE + NEC_UNIT / 2))
-			bit = 0;
-		else if ((ev->delta.tv_nsec >= NEC_1_SPACE - NEC_UNIT / 2) &&
-		         (ev->delta.tv_nsec < NEC_1_SPACE + NEC_UNIT / 2))
-			bit = 1;
-		else {
-			IR_dprintk(1, "Decode failed at %d-th bit (%s) @%luus\n",
-				data->count,
-				last_bit ? "pulse" : "space",
-				(ev->delta.tv_nsec + 500) / 1000);
+		data->nec_bits <<= 1;
+		if (d == NEC_1_SPACE)
+			data->nec_bits |= 1;
+		data->count++;
 
-			goto err2;
+		if (data->count != NEC_NBITS) {
+			data->state = STATE_MARK;
+			return 0;
 		}
 
-		/* Ok, we've got a valid bit. proccess it */
-		if (bit) {
-			int shift = data->count;
-
-			/*
-			 * NEC transmit bytes on this temporal order:
-			 * address | not address | command | not command
-			 */
-			if (shift < 8) {
-				data->nec_code.address |= 1 << shift;
-			} else if (shift < 16) {
-				data->nec_code.not_address |= 1 << (shift - 8);
-			} else if (shift < 24) {
-				data->nec_code.command |= 1 << (shift - 16);
-			} else {
-				data->nec_code.not_command |= 1 << (shift - 24);
-			}
+		address     = bitrev8((data->nec_bits >> 24) & 0xff);
+		not_address = bitrev8((data->nec_bits >> 16) & 0xff);
+		command	    = bitrev8((data->nec_bits >>  8) & 0xff);
+		not_command = bitrev8((data->nec_bits >>  0) & 0xff);
+
+		if ((command ^ not_command) != 0xff) {
+			IR_dprintk(1, "NEC checksum error: received 0x%08x\n",
+				   data->nec_bits);
+			break;
 		}
-		if (++data->count == NEC_NBITS) {
-			u32 scancode;
-			/*
-			 * Fixme: may need to accept Extended NEC protocol?
-			 */
-			if ((data->nec_code.command ^ data->nec_code.not_command) != 0xff)
-				goto checksum_err;
-
-			if ((data->nec_code.address ^ data->nec_code.not_address) != 0xff) {
-				/* Extended NEC */
-				scancode = data->nec_code.address << 16 |
-					   data->nec_code.not_address << 8 |
-					   data->nec_code.command;
-				IR_dprintk(1, "NEC scancode 0x%06x\n", scancode);
-			} else {
-				/* normal NEC */
-				scancode = data->nec_code.address << 8 |
-					   data->nec_code.command;
-				IR_dprintk(1, "NEC scancode 0x%04x\n", scancode);
-			}
-			ir_keydown(input_dev, scancode, 0);
 
-			data->state = STATE_TRAILER_MARK;
-		} else
-			data->state = STATE_MARK;
+		if ((address ^ not_address) != 0xff) {
+			/* Extended NEC */
+			scancode = address     << 16 |
+				   not_address <<  8 |
+				   command;
+			IR_dprintk(1, "NEC (Ext) scancode 0x%06x\n", scancode);
+		} else {
+			/* normal NEC */
+			scancode = address << 8 | command;
+			IR_dprintk(1, "NEC scancode 0x%04x\n", scancode);
+		}
+
+		ir_keydown(input_dev, scancode, 0);
+		data->state = STATE_TRAILER_MARK;
 		return 0;
+
 	case STATE_TRAILER_MARK:
-		if (!last_bit)
-			goto err;
-		data->state = STATE_TRAILER_SPACE;
-		return 0;
+		if (d > 0) {
+			data->state = STATE_TRAILER_SPACE;
+			return 0;
+		}
+		break;
+
 	case STATE_TRAILER_SPACE:
-		if (last_bit)
-			goto err;
-		data->state = STATE_INACTIVE;
-		return 0;
-	}
+		if (d < 0) {
+			data->state = STATE_INACTIVE;
+			return 0;
+		}
 
-err:
-	IR_dprintk(1, "NEC decoded failed at state %d (%s) @ %luus\n",
-		   data->state,
-		   bit ? "pulse" : "space",
-		   (ev->delta.tv_nsec + 500) / 1000);
-err2:
-	data->state = STATE_INACTIVE;
-	return -EINVAL;
+		break;
+	}
 
-checksum_err:
+	IR_dprintk(1, "NEC decode failed at state %d (%i units, %ius)\n",
+		   data->state, d, duration);
 	data->state = STATE_INACTIVE;
-	IR_dprintk(1, "NEC checksum error: received 0x%02x%02x%02x%02x\n",
-		   data->nec_code.address,
-		   data->nec_code.not_address,
-		   data->nec_code.command,
-		   data->nec_code.not_command);
 	return -EINVAL;
 }
 
Index: ir/drivers/media/IR/ir-rc5-decoder.c
===================================================================
--- ir.orig/drivers/media/IR/ir-rc5-decoder.c	2010-04-06 12:16:51.784849187 +0200
+++ ir/drivers/media/IR/ir-rc5-decoder.c	2010-04-07 21:34:14.816870395 +0200
@@ -15,31 +15,22 @@
 /*
  * This code only handles 14 bits RC-5 protocols. There are other variants
  * that use a different number of bits. This is currently unsupported
- * It considers a carrier of 36 kHz, with a total of 14 bits, where
- * the first two bits are start bits, and a third one is a filing bit
  */
 
 #include <media/ir-core.h>
 
-static unsigned int ir_rc5_remote_gap = 888888;
-
-#define RC5_NBITS		14
-#define RC5_BIT			(ir_rc5_remote_gap * 2)
-#define RC5_DURATION		(ir_rc5_remote_gap * RC5_NBITS)
+#define RC5_UNIT		889 /* us */
+#define	RC5_BITS		14
 
 /* Used to register rc5_decoder clients */
 static LIST_HEAD(decoder_list);
-static spinlock_t decoder_lock;
+static DEFINE_SPINLOCK(decoder_lock);
 
 enum rc5_state {
 	STATE_INACTIVE,
-	STATE_MARKSPACE,
-	STATE_TRAILER,
-};
-
-struct rc5_code {
-	u8	address;
-	u8	command;
+	STATE_BIT_START,
+	STATE_BIT_END,
+	STATE_FINISHED,
 };
 
 struct decoder_data {
@@ -49,8 +40,9 @@
 
 	/* State machine control */
 	enum rc5_state		state;
-	struct rc5_code		rc5_code;
-	unsigned		code, elapsed, last_bit, last_code;
+	u32			rc5_bits;
+	int			last_delta;
+	unsigned		count;
 };
 
 
@@ -122,18 +114,19 @@
 };
 
 /**
- * handle_event() - Decode one RC-5 pulse or space
+ * ir_rc5_decode() - Decode one RC-5 pulse or space
  * @input_dev:	the struct input_dev descriptor of the device
- * @ev:		event array with type/duration of pulse/space
+ * @duration:	duration of pulse/space
  *
  * This function returns -EINVAL if the pulse violates the state machine
  */
-static int ir_rc5_decode(struct input_dev *input_dev,
-			struct ir_raw_event *ev)
+static int ir_rc5_decode(struct input_dev *input_dev, int duration)
 {
 	struct decoder_data *data;
 	struct ir_input_dev *ir_dev = input_get_drvdata(input_dev);
-	int is_pulse, scancode, delta, toggle;
+	u8 command, system, toggle;
+	u32 scancode;
+	int d;
 
 	data = get_decoder_data(ir_dev);
 	if (!data)
@@ -142,79 +135,86 @@
 	if (!data->enabled)
 		return 0;
 
-	delta = DIV_ROUND_CLOSEST(ev->delta.tv_nsec, ir_rc5_remote_gap);
+	if (duration == 0) {
+		data->state = STATE_INACTIVE;
+		return 0;
+	}
 
-	/* The duration time refers to the last bit time */
-	is_pulse = (ev->type & IR_PULSE) ? 1 : 0;
+	d = DIV_ROUND_CLOSEST(abs(duration), RC5_UNIT);
+	if (duration < 0)
+		d = -d;
+
+again:
+	IR_dprintk(2, "RC5 decode started at state %i (%i units, %ius)\n",
+		   data->state, d, duration);
 
-	/* Very long delays are considered as start events */
-	if (delta > RC5_DURATION || (ev->type & IR_START_EVENT))
-		data->state = STATE_INACTIVE;
+	if (d == 0 && data->state != STATE_FINISHED)
+		return 0;
 
 	switch (data->state) {
+
 	case STATE_INACTIVE:
-	IR_dprintk(2, "currently inative. Start bit (%s) @%uus\n",
-		   is_pulse ? "pulse" : "space",
-		   (unsigned)(ev->delta.tv_nsec + 500) / 1000);
-
-		/* Discards the initial start space */
-		if (!is_pulse)
-			goto err;
-		data->code = 1;
-		data->last_bit = 1;
-		data->elapsed = 0;
-		memset(&data->rc5_code, 0, sizeof(data->rc5_code));
-		data->state = STATE_MARKSPACE;
-		return 0;
-	case STATE_MARKSPACE:
-		if (delta != 1)
-			data->last_bit = data->last_bit ? 0 : 1;
+		if ((d == 1) || (d == 2)) {
+			data->state = STATE_BIT_START;
+			data->count = 1;
+			d--;
+			goto again;
+		}
+		break;
 
-		data->elapsed += delta;
+	case STATE_BIT_START:
+		if (abs(d) == 1) {
+			data->rc5_bits <<= 1;
+			if (d == -1)
+				data->rc5_bits |= 1;
+			data->count++;
+			data->last_delta = d;
+
+			/*
+			 * If the last bit is zero, a space will "merge"
+			 * with the silence after the command.
+			 */
+			if ((data->count == data->wanted_bits) && (d == 1))
+				data->state = STATE_FINISHED;
+			else
+				data->state = STATE_BIT_END;
 
-		if ((data->elapsed % 2) == 1)
 			return 0;
+		}
+		break;
 
-		data->code <<= 1;
-		data->code |= data->last_bit;
-
-		/* Fill the 2 unused bits at the command with 0 */
-		if (data->elapsed / 2 == 6)
-			data->code <<= 2;
-
-		if (data->elapsed >= (RC5_NBITS - 1) * 2) {
-			scancode = data->code;
-
-			/* Check for the start bits */
-			if ((scancode & 0xc000) != 0xc000) {
-				IR_dprintk(1, "Code 0x%04x doesn't have two start bits. It is not RC-5\n", scancode);
-				goto err;
-			}
-
-			toggle = (scancode & 0x2000) ? 1 : 0;
-
-			if (scancode == data->last_code) {
-				IR_dprintk(1, "RC-5 repeat\n");
-				ir_repeat(input_dev);
-			} else {
-				data->last_code = scancode;
-				scancode &= 0x1fff;
-				IR_dprintk(1, "RC-5 scancode 0x%04x\n", scancode);
-
-				ir_keydown(input_dev, scancode, 0);
-			}
-			data->state = STATE_TRAILER;
+	case STATE_BIT_END:
+		/* delta and last_delta signedness must differ */
+		if (d * data->last_delta < 0) {
+			if (data->count == RC5_BITS)
+				data->state = STATE_FINISHED;
+			else
+				data->state = STATE_BIT_START;
+
+			if (d > 0)
+				d--;
+			else if (d < 0)
+				d++;
+			goto again;
 		}
-		return 0;
-	case STATE_TRAILER:
+		break;
+
+	case STATE_FINISHED:
+		command  = (data->rc5_bits & 0x0003F) >> 0;
+		system   = (data->rc5_bits & 0x007C0) >> 6;
+		toggle   = (data->rc5_bits & 0x00800) ? 1 : 0;
+		command += (data->rc5_bits & 0x01000) ? 0 : 0x40;
+		scancode = system << 8 | command;
+
+		IR_dprintk(1, "RC5 scancode 0x%04x (toggle: %u)\n",
+			   scancode, toggle);
+		ir_keydown(input_dev, scancode, toggle);
 		data->state = STATE_INACTIVE;
 		return 0;
 	}
 
-err:
-	IR_dprintk(1, "RC-5 decoded failed at %s @ %luus\n",
-		   is_pulse ? "pulse" : "space",
-		   (ev->delta.tv_nsec + 500) / 1000);
+	IR_dprintk(1, "RC5 decode failed at state %i (%i units, %ius)\n",
+		   data->state, d, duration);
 	data->state = STATE_INACTIVE;
 	return -EINVAL;
 }
Index: ir/drivers/media/video/saa7134/saa7134-input.c
===================================================================
--- ir.orig/drivers/media/video/saa7134/saa7134-input.c	2010-04-06 12:30:16.428854774 +0200
+++ ir/drivers/media/video/saa7134/saa7134-input.c	2010-04-07 18:05:27.756852639 +0200
@@ -1021,7 +1021,7 @@
 	saa_clearb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
 	saa_setb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
 	space = saa_readl(SAA7134_GPIO_GPSTATUS0 >> 2) & ir->mask_keydown;
-	ir_raw_event_store(dev->remote->dev, space ? IR_SPACE : IR_PULSE);
+	ir_raw_event_store_edge(dev->remote->dev, space ? IR_SPACE : IR_PULSE);
 
 
 	/*

-- 
David Härdeman
--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-07 20:18 [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations David Härdeman
@ 2010-04-07 21:01 ` Jon Smirl
  2010-04-08  0:18 ` Jon Smirl
                   ` (2 subsequent siblings)
  3 siblings, 0 replies; 13+ messages in thread
From: Jon Smirl @ 2010-04-07 21:01 UTC (permalink / raw)
  To: David Härdeman; +Cc: mchehab, linux-input, linux-media

On Wed, Apr 7, 2010 at 4:18 PM, David Härdeman <david@hardeman.nu> wrote:
> o Not sent using quilt...Mauro, does it still trip up your MUA?

Try Stacked git (stg). Stacked git is quilt rewritten specifically for git.
http://www.procode.org/stgit/

I have been using it for kernel patches for over four years. So far no
one has complained about it (they have complained about my pilot
errors using it).

-- 
Jon Smirl
jonsmirl@gmail.com

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-07 20:18 [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations David Härdeman
  2010-04-07 21:01 ` Jon Smirl
@ 2010-04-08  0:18 ` Jon Smirl
  2010-04-08  0:44 ` Andy Walls
  2010-04-08  5:10 ` Mauro Carvalho Chehab
  3 siblings, 0 replies; 13+ messages in thread
From: Jon Smirl @ 2010-04-08  0:18 UTC (permalink / raw)
  To: David Härdeman; +Cc: mchehab, linux-input, linux-media

On Wed, Apr 7, 2010 at 4:18 PM, David Härdeman <david@hardeman.nu> wrote:
> drivers/media/IR/ir-raw-event.c is currently written with the assumption
> that all "raw" hardware will generate events only on state change (i.e.
> when a pulse or space starts).
>
> However, some hardware (like mceusb, probably the most popular IR receiver
> out there) only generates duration data (and that data is buffered so using
> any kind of timing on the data is futile).
>
> Furthermore, using signed int's to represent pulse/space durations in ms
> is a well-known approach to anyone with experience in writing ir decoders.
>
> This patch (which has been tested this time) is still a RFC on my proposed
> interface changes.
>
> Changes since last version:
>
> o RC5x and NECx support no longer added in patch (separate patches to follow)
>
> o The use of a kfifo has been left given feedback from Jon, Andy, Mauro
>
> o The RX decoding is now handled via a workqueue (I can break that up into a
>  separate patch later, but I think it helps the discussion to have it in for
>  now), with inspiration from Andy's code.
>
> o Separate reset operations are no longer added to decoders, a duration of
>  zero is instead used to signal a reset (which allows the reset request to
>  be inserted into the kfifo).
>
> o Not sent using quilt...Mauro, does it still trip up your MUA?
>
> Not changed:
>
> o int's are still used to represent pulse/space durations in ms. Mauro and I
>  seem to disagree on this one but I'm right :)
>
> Index: ir/drivers/media/IR/ir-raw-event.c
> ===================================================================
> --- ir.orig/drivers/media/IR/ir-raw-event.c     2010-04-06 12:16:27.000000000 +0200
> +++ ir/drivers/media/IR/ir-raw-event.c  2010-04-07 21:32:13.961850481 +0200
> @@ -15,13 +15,14 @@
>  #include <media/ir-core.h>
>  #include <linux/workqueue.h>
>  #include <linux/spinlock.h>
> +#include <linux/sched.h>
>
>  /* Define the max number of bit transitions per IR keycode */
>  #define MAX_IR_EVENT_SIZE      256
>
>  /* Used to handle IR raw handler extensions */
>  static LIST_HEAD(ir_raw_handler_list);
> -static spinlock_t ir_raw_handler_lock;
> +static DEFINE_SPINLOCK(ir_raw_handler_lock);
>
>  /**
>  * RUN_DECODER()       - runs an operation on all IR decoders
> @@ -53,19 +54,30 @@
>  /* Used to load the decoders */
>  static struct work_struct wq_load;
>
> +static void ir_raw_event_work(struct work_struct *work)
> +{
> +       int d;
> +       struct ir_raw_event_ctrl *raw =
> +               container_of(work, struct ir_raw_event_ctrl, rx_work);
> +
> +       while (kfifo_out(&raw->kfifo, &d, sizeof(d)) == sizeof(d))
> +               RUN_DECODER(decode, raw->input_dev, d);
> +}
> +
>  int ir_raw_event_register(struct input_dev *input_dev)
>  {
>        struct ir_input_dev *ir = input_get_drvdata(input_dev);
> -       int rc, size;
> +       int rc;
>
>        ir->raw = kzalloc(sizeof(*ir->raw), GFP_KERNEL);
>        if (!ir->raw)
>                return -ENOMEM;
>
> -       size = sizeof(struct ir_raw_event) * MAX_IR_EVENT_SIZE * 2;
> -       size = roundup_pow_of_two(size);
> +       ir->raw->input_dev = input_dev;
> +       INIT_WORK(&ir->raw->rx_work, ir_raw_event_work);
>
> -       rc = kfifo_alloc(&ir->raw->kfifo, size, GFP_KERNEL);
> +       rc = kfifo_alloc(&ir->raw->kfifo, sizeof(int) * MAX_IR_EVENT_SIZE,
> +                        GFP_KERNEL);
>        if (rc < 0) {
>                kfree(ir->raw);
>                ir->raw = NULL;
> @@ -91,6 +103,7 @@
>        if (!ir->raw)
>                return;
>
> +       cancel_work_sync(&ir->raw->rx_work);
>        RUN_DECODER(raw_unregister, input_dev);
>
>        kfifo_free(&ir->raw->kfifo);
> @@ -99,74 +112,85 @@
>  }
>  EXPORT_SYMBOL_GPL(ir_raw_event_unregister);
>
> -int ir_raw_event_store(struct input_dev *input_dev, enum raw_event_type type)
> +/**
> + * ir_raw_event_store() - pass a pulse/space duration to the raw ir decoders
> + * @input_dev: the struct input_dev device descriptor
> + * @duration:  duration of the pulse or space
> + *
> + * This routine passes a pulse/space duration to the raw ir decoding state
> + * machines. Pulses are signalled as positive values and spaces as negative
> + * values. A zero value will reset the decoding state machines.

Mention that this routine is safe to call from interrupt context

> + */
> +int ir_raw_event_store(struct input_dev *input_dev, int duration)
>  {
> -       struct ir_input_dev     *ir = input_get_drvdata(input_dev);
> -       struct timespec         ts;
> -       struct ir_raw_event     event;
> -       int                     rc;
> +       struct ir_input_dev *ir = input_get_drvdata(input_dev);
>
>        if (!ir->raw)
>                return -EINVAL;
>
> -       event.type = type;
> -       event.delta.tv_sec = 0;
> -       event.delta.tv_nsec = 0;
> +       if (kfifo_in(&ir->raw->kfifo, &duration, sizeof(duration)) != sizeof(duration))
> +               return -ENOMEM;
>
> -       ktime_get_ts(&ts);
> +       return 0;
> +}
> +EXPORT_SYMBOL_GPL(ir_raw_event_store);
>
> -       if (timespec_equal(&ir->raw->last_event, &event.delta))
> -               event.type |= IR_START_EVENT;
> -       else
> -               event.delta = timespec_sub(ts, ir->raw->last_event);
> +/**
> + * ir_raw_event_store_edge() - notify raw ir decoders of the start of a pulse/space
> + * @input_dev: the struct input_dev device descriptor
> + * @type:      the type of the event that has occurred
> + *
> + * This routine is used to notify the raw ir decoders on the beginning of an
> + * ir pulse or space (or the start/end of ir reception). This is used by
> + * hardware which does not provide durations directly but only interrupts
> + * (or similar events) on state change.

Mention that this routine is safe to call from interrupt context

> + */
> +int ir_raw_event_store_edge(struct input_dev *input_dev, enum raw_event_type type)
> +{
> +       struct ir_input_dev     *ir = input_get_drvdata(input_dev);
> +       ktime_t                 now;
> +       s64                     delta; /* us */
> +       int                     rc = 0;
>
> -       memcpy(&ir->raw->last_event, &ts, sizeof(ts));
> +       if (!ir->raw)
> +               return -EINVAL;
>
> -       if (event.delta.tv_sec) {
> -               event.type |= IR_START_EVENT;
> -               event.delta.tv_sec = 0;
> -               event.delta.tv_nsec = 0;
> -       }
> +       now = ktime_get();
> +       delta = ktime_us_delta(now, ir->raw->last_event);
>
> -       kfifo_in(&ir->raw->kfifo, &event, sizeof(event));
> +       /* Are we called for the first time? */
> +       if (!ir->raw->last_type)
> +               type |= IR_START_EVENT;
> +
> +       if (type & IR_START_EVENT)
> +               ir_raw_event_reset(input_dev);

--- make sure this is ok to call from interrupt context

> +       else if (ir->raw->last_type & IR_SPACE)
> +               rc = ir_raw_event_store(input_dev, (int)-delta);
> +       else if (ir->raw->last_type & IR_PULSE)
> +               rc = ir_raw_event_store(input_dev, (int)delta);
> +       else
> +               return 0;
>
> +       ir->raw->last_event = now;
> +       ir->raw->last_type = type;
>        return rc;
>  }
> -EXPORT_SYMBOL_GPL(ir_raw_event_store);
> +EXPORT_SYMBOL_GPL(ir_raw_event_store_edge);
>
> -int ir_raw_event_handle(struct input_dev *input_dev)
> +/**
> + * ir_raw_event_handle() - schedules the decoding of stored ir data
> + * @input_dev: the struct input_dev device descriptor
> + *
> + * This routine will signal the workqueue to start decoding stored ir data.
> + */
> +void ir_raw_event_handle(struct input_dev *input_dev)
>  {
> -       struct ir_input_dev             *ir = input_get_drvdata(input_dev);
> -       int                             rc;
> -       struct ir_raw_event             ev;
> -       int                             len, i;
> -
> -       /*
> -        * Store the events into a temporary buffer. This allows calling more than
> -        * one decoder to deal with the received data
> -        */
> -       len = kfifo_len(&ir->raw->kfifo) / sizeof(ev);
> -       if (!len)
> -               return 0;
> -
> -       for (i = 0; i < len; i++) {
> -               rc = kfifo_out(&ir->raw->kfifo, &ev, sizeof(ev));
> -               if (rc != sizeof(ev)) {
> -                       IR_dprintk(1, "overflow error: received %d instead of %zd\n",
> -                                  rc, sizeof(ev));
> -                       return -EINVAL;
> -               }
> -               IR_dprintk(2, "event type %d, time before event: %07luus\n",
> -                       ev.type, (ev.delta.tv_nsec + 500) / 1000);
> -               rc = RUN_DECODER(decode, input_dev, &ev);
> -       }
> +       struct ir_input_dev *ir = input_get_drvdata(input_dev);
>
> -       /*
> -        * Call all ir decoders. This allows decoding the same event with
> -        * more than one protocol handler.
> -        */
> +       if (!ir->raw)
> +               return;
>
> -       return rc;
> +       schedule_work(&ir->raw->rx_work);
>  }
>  EXPORT_SYMBOL_GPL(ir_raw_event_handle);
>
> @@ -205,8 +229,6 @@
>
>  void ir_raw_init(void)
>  {
> -       spin_lock_init(&ir_raw_handler_lock);
> -
>  #ifdef MODULE
>        INIT_WORK(&wq_load, init_decoders);
>        schedule_work(&wq_load);
> Index: ir/include/media/ir-core.h
> ===================================================================
> --- ir.orig/include/media/ir-core.h     2010-04-06 12:16:27.000000000 +0200
> +++ ir/include/media/ir-core.h  2010-04-07 18:17:53.524850074 +0200
> @@ -57,14 +57,12 @@
>        void            (*close)(void *priv);
>  };
>
> -struct ir_raw_event {
> -       struct timespec         delta;  /* Time spent before event */
> -       enum raw_event_type     type;   /* event type */
> -};
> -
>  struct ir_raw_event_ctrl {
> -       struct kfifo                    kfifo;          /* fifo for the pulse/space events */
> -       struct timespec                 last_event;     /* when last event occurred */
> +       struct work_struct              rx_work;        /* for the rx decoding workqueue */
> +       struct kfifo                    kfifo;          /* fifo for the pulse/space durations */
> +       ktime_t                         last_event;     /* when last event occurred */
> +       enum raw_event_type             last_type;      /* last event type */
> +       struct input_dev                *input_dev;     /* pointer to the parent input_dev */
>  };
>
>  struct ir_input_dev {
> @@ -89,8 +87,7 @@
>  struct ir_raw_handler {
>        struct list_head list;
>
> -       int (*decode)(struct input_dev *input_dev,
> -                     struct ir_raw_event *ev);
> +       int (*decode)(struct input_dev *input_dev, int duration);
>        int (*raw_register)(struct input_dev *input_dev);
>        int (*raw_unregister)(struct input_dev *input_dev);
>  };
> @@ -146,8 +143,13 @@
>  /* Routines from ir-raw-event.c */
>  int ir_raw_event_register(struct input_dev *input_dev);
>  void ir_raw_event_unregister(struct input_dev *input_dev);
> -int ir_raw_event_store(struct input_dev *input_dev, enum raw_event_type type);
> -int ir_raw_event_handle(struct input_dev *input_dev);
> +void ir_raw_event_handle(struct input_dev *input_dev);
> +int ir_raw_event_store(struct input_dev *input_dev, int duration);
> +int ir_raw_event_store_edge(struct input_dev *input_dev, enum raw_event_type type);
> +static inline void ir_raw_event_reset(struct input_dev *dev) {
> +       ir_raw_event_store(input_dev, 0);
> +       ir_raw_event_handle(input_dev);
> +}
>  int ir_raw_handler_register(struct ir_raw_handler *ir_raw_handler);
>  void ir_raw_handler_unregister(struct ir_raw_handler *ir_raw_handler);
>  void ir_raw_init(void);
> Index: ir/drivers/media/IR/ir-nec-decoder.c
> ===================================================================
> --- ir.orig/drivers/media/IR/ir-nec-decoder.c   2010-04-06 12:16:27.000000000 +0200
> +++ ir/drivers/media/IR/ir-nec-decoder.c        2010-04-07 18:32:10.892853025 +0200
> @@ -13,23 +13,23 @@
>  */
>
>  #include <media/ir-core.h>
> +#include <linux/bitrev.h>
>
>  #define NEC_NBITS              32
> -#define NEC_UNIT               559979 /* ns */
> -#define NEC_HEADER_MARK                (16 * NEC_UNIT)
> -#define NEC_HEADER_SPACE       (8 * NEC_UNIT)
> -#define NEC_REPEAT_SPACE       (4 * NEC_UNIT)
> -#define NEC_MARK               (NEC_UNIT)
> -#define NEC_0_SPACE            (NEC_UNIT)
> -#define NEC_1_SPACE            (3 * NEC_UNIT)
> +#define NEC_UNIT               562     /* us */
> +#define NEC_HEADER_MARK                16
> +#define NEC_HEADER_SPACE       -8
> +#define NEC_REPEAT_SPACE       -4
> +#define NEC_MARK               1
> +#define NEC_0_SPACE            -1
> +#define NEC_1_SPACE            -3
>
>  /* Used to register nec_decoder clients */
>  static LIST_HEAD(decoder_list);
> -static spinlock_t decoder_lock;
> +static DEFINE_SPINLOCK(decoder_lock);
>
>  enum nec_state {
>        STATE_INACTIVE,
> -       STATE_HEADER_MARK,
>        STATE_HEADER_SPACE,
>        STATE_MARK,
>        STATE_SPACE,
> @@ -37,13 +37,6 @@
>        STATE_TRAILER_SPACE,
>  };
>
> -struct nec_code {
> -       u8      address;
> -       u8      not_address;
> -       u8      command;
> -       u8      not_command;
> -};
> -
>  struct decoder_data {
>        struct list_head        list;
>        struct ir_input_dev     *ir_dev;
> @@ -51,7 +44,7 @@
>
>        /* State machine control */
>        enum nec_state          state;
> -       struct nec_code         nec_code;
> +       u32                     nec_bits;
>        unsigned                count;
>  };
>
> @@ -62,7 +55,6 @@
>  *
>  * Returns the struct decoder_data that corresponds to a device
>  */
> -
>  static struct decoder_data *get_decoder_data(struct  ir_input_dev *ir_dev)
>  {
>        struct decoder_data *data = NULL;
> @@ -123,20 +115,20 @@
>        .attrs  = decoder_attributes,
>  };
>
> -
>  /**
>  * ir_nec_decode() - Decode one NEC pulse or space
>  * @input_dev: the struct input_dev descriptor of the device
> - * @ev:                event array with type/duration of pulse/space
> + * @duration:  duration in us of pulse/space
>  *
>  * This function returns -EINVAL if the pulse violates the state machine
>  */
> -static int ir_nec_decode(struct input_dev *input_dev,
> -                        struct ir_raw_event *ev)
> +static int ir_nec_decode(struct input_dev *input_dev, int duration)
>  {
>        struct decoder_data *data;
>        struct ir_input_dev *ir_dev = input_get_drvdata(input_dev);
> -       int bit, last_bit;
> +       int d;
> +       u32 scancode;
> +       u8 address, not_address, command, not_command;
>
>        data = get_decoder_data(ir_dev);
>        if (!data)
> @@ -145,150 +137,106 @@
>        if (!data->enabled)
>                return 0;
>
> -       /* Except for the initial event, what matters is the previous bit */
> -       bit = (ev->type & IR_PULSE) ? 1 : 0;
> -
> -       last_bit = !bit;
> -
> -       /* Discards spurious space last_bits when inactive */
> -
> -       /* Very long delays are considered as start events */
> -       if (ev->delta.tv_nsec > NEC_HEADER_MARK + NEC_HEADER_SPACE - NEC_UNIT / 2)
> +       if (duration == 0) {
>                data->state = STATE_INACTIVE;
> +               return 0;
> +       }
>
> -       if (ev->type & IR_START_EVENT)
> -               data->state = STATE_INACTIVE;
> +       d = DIV_ROUND_CLOSEST(abs(duration), NEC_UNIT);
> +       if (duration < 0)
> +               d = -d;

d = abs(duration)
d = DIV_ROUND_CLOSEST(d, NEC_UNIT);


> +
> +       IR_dprintk(2, "NEC decode started at state %d (%i units, %ius)\n",
> +                  data->state, d, duration);
>
>        switch (data->state) {
> -       case STATE_INACTIVE:
> -               if (!bit)               /* PULSE marks the start event */
> -                       return 0;
>
> -               data->count = 0;
> -               data->state = STATE_HEADER_MARK;
> -               memset (&data->nec_code, 0, sizeof(data->nec_code));
> -               return 0;
> -       case STATE_HEADER_MARK:
> -               if (!last_bit)
> -                       goto err;
> -               if (ev->delta.tv_nsec < NEC_HEADER_MARK - 6 * NEC_UNIT)
> -                       goto err;
> -               data->state = STATE_HEADER_SPACE;
> +       case STATE_INACTIVE:
> +               if (d == NEC_HEADER_MARK) {
> +                       data->count = 0;
> +                       data->state = STATE_HEADER_SPACE;
> +               }
>                return 0;
> +
>        case STATE_HEADER_SPACE:
> -               if (last_bit)
> -                       goto err;
> -               if (ev->delta.tv_nsec >= NEC_HEADER_SPACE - NEC_UNIT / 2) {
> +               if (d == NEC_HEADER_SPACE) {
>                        data->state = STATE_MARK;
>                        return 0;
> -               }
> -
> -               if (ev->delta.tv_nsec >= NEC_REPEAT_SPACE - NEC_UNIT / 2) {
> +               } else if (d == NEC_REPEAT_SPACE) {
>                        ir_repeat(input_dev);
>                        IR_dprintk(1, "Repeat last key\n");
>                        data->state = STATE_TRAILER_MARK;
>                        return 0;
>                }
> -               goto err;
> +               break;
> +
>        case STATE_MARK:
> -               if (!last_bit)
> -                       goto err;
> -               if ((ev->delta.tv_nsec > NEC_MARK + NEC_UNIT / 2) ||
> -                   (ev->delta.tv_nsec < NEC_MARK - NEC_UNIT / 2))
> -                       goto err;
> -               data->state = STATE_SPACE;
> -               return 0;
> +               if (d == NEC_MARK) {
> +                       data->state = STATE_SPACE;
> +                       return 0;
> +               }
> +               break;
> +
>        case STATE_SPACE:
> -               if (last_bit)
> -                       goto err;
> +               if (d != NEC_0_SPACE && d != NEC_1_SPACE)
> +                       break;
>
> -               if ((ev->delta.tv_nsec >= NEC_0_SPACE - NEC_UNIT / 2) &&
> -                   (ev->delta.tv_nsec < NEC_0_SPACE + NEC_UNIT / 2))
> -                       bit = 0;
> -               else if ((ev->delta.tv_nsec >= NEC_1_SPACE - NEC_UNIT / 2) &&
> -                        (ev->delta.tv_nsec < NEC_1_SPACE + NEC_UNIT / 2))
> -                       bit = 1;
> -               else {
> -                       IR_dprintk(1, "Decode failed at %d-th bit (%s) @%luus\n",
> -                               data->count,
> -                               last_bit ? "pulse" : "space",
> -                               (ev->delta.tv_nsec + 500) / 1000);
> +               data->nec_bits <<= 1;
> +               if (d == NEC_1_SPACE)
> +                       data->nec_bits |= 1;
> +               data->count++;
>
> -                       goto err2;
> +               if (data->count != NEC_NBITS) {
> +                       data->state = STATE_MARK;
> +                       return 0;
>                }
>
> -               /* Ok, we've got a valid bit. proccess it */
> -               if (bit) {
> -                       int shift = data->count;
> -
> -                       /*
> -                        * NEC transmit bytes on this temporal order:
> -                        * address | not address | command | not command
> -                        */
> -                       if (shift < 8) {
> -                               data->nec_code.address |= 1 << shift;
> -                       } else if (shift < 16) {
> -                               data->nec_code.not_address |= 1 << (shift - 8);
> -                       } else if (shift < 24) {
> -                               data->nec_code.command |= 1 << (shift - 16);
> -                       } else {
> -                               data->nec_code.not_command |= 1 << (shift - 24);
> -                       }
> +               address     = bitrev8((data->nec_bits >> 24) & 0xff);
> +               not_address = bitrev8((data->nec_bits >> 16) & 0xff);
> +               command     = bitrev8((data->nec_bits >>  8) & 0xff);
> +               not_command = bitrev8((data->nec_bits >>  0) & 0xff);
> +
> +               if ((command ^ not_command) != 0xff) {
> +                       IR_dprintk(1, "NEC checksum error: received 0x%08x\n",
> +                                  data->nec_bits);
> +                       break;
>                }
> -               if (++data->count == NEC_NBITS) {
> -                       u32 scancode;
> -                       /*
> -                        * Fixme: may need to accept Extended NEC protocol?
> -                        */
> -                       if ((data->nec_code.command ^ data->nec_code.not_command) != 0xff)
> -                               goto checksum_err;
> -
> -                       if ((data->nec_code.address ^ data->nec_code.not_address) != 0xff) {
> -                               /* Extended NEC */
> -                               scancode = data->nec_code.address << 16 |
> -                                          data->nec_code.not_address << 8 |
> -                                          data->nec_code.command;
> -                               IR_dprintk(1, "NEC scancode 0x%06x\n", scancode);
> -                       } else {
> -                               /* normal NEC */
> -                               scancode = data->nec_code.address << 8 |
> -                                          data->nec_code.command;
> -                               IR_dprintk(1, "NEC scancode 0x%04x\n", scancode);
> -                       }
> -                       ir_keydown(input_dev, scancode, 0);
>
> -                       data->state = STATE_TRAILER_MARK;
> -               } else
> -                       data->state = STATE_MARK;
> +               if ((address ^ not_address) != 0xff) {
> +                       /* Extended NEC */
> +                       scancode = address     << 16 |
> +                                  not_address <<  8 |
> +                                  command;
> +                       IR_dprintk(1, "NEC (Ext) scancode 0x%06x\n", scancode);
> +               } else {
> +                       /* normal NEC */
> +                       scancode = address << 8 | command;
> +                       IR_dprintk(1, "NEC scancode 0x%04x\n", scancode);
> +               }
> +
> +               ir_keydown(input_dev, scancode, 0);
> +               data->state = STATE_TRAILER_MARK;
>                return 0;
> +
>        case STATE_TRAILER_MARK:
> -               if (!last_bit)
> -                       goto err;
> -               data->state = STATE_TRAILER_SPACE;
> -               return 0;
> +               if (d > 0) {
> +                       data->state = STATE_TRAILER_SPACE;
> +                       return 0;
> +               }
> +               break;
> +
>        case STATE_TRAILER_SPACE:
> -               if (last_bit)
> -                       goto err;
> -               data->state = STATE_INACTIVE;
> -               return 0;
> -       }
> +               if (d < 0) {
> +                       data->state = STATE_INACTIVE;
> +                       return 0;
> +               }
>
> -err:
> -       IR_dprintk(1, "NEC decoded failed at state %d (%s) @ %luus\n",
> -                  data->state,
> -                  bit ? "pulse" : "space",
> -                  (ev->delta.tv_nsec + 500) / 1000);
> -err2:
> -       data->state = STATE_INACTIVE;
> -       return -EINVAL;
> +               break;
> +       }
>
> -checksum_err:
> +       IR_dprintk(1, "NEC decode failed at state %d (%i units, %ius)\n",
> +                  data->state, d, duration);
>        data->state = STATE_INACTIVE;
> -       IR_dprintk(1, "NEC checksum error: received 0x%02x%02x%02x%02x\n",
> -                  data->nec_code.address,
> -                  data->nec_code.not_address,
> -                  data->nec_code.command,
> -                  data->nec_code.not_command);
>        return -EINVAL;
>  }
>
> Index: ir/drivers/media/IR/ir-rc5-decoder.c
> ===================================================================
> --- ir.orig/drivers/media/IR/ir-rc5-decoder.c   2010-04-06 12:16:51.784849187 +0200
> +++ ir/drivers/media/IR/ir-rc5-decoder.c        2010-04-07 21:34:14.816870395 +0200
> @@ -15,31 +15,22 @@
>  /*
>  * This code only handles 14 bits RC-5 protocols. There are other variants
>  * that use a different number of bits. This is currently unsupported
> - * It considers a carrier of 36 kHz, with a total of 14 bits, where
> - * the first two bits are start bits, and a third one is a filing bit
>  */
>
>  #include <media/ir-core.h>
>
> -static unsigned int ir_rc5_remote_gap = 888888;
> -
> -#define RC5_NBITS              14
> -#define RC5_BIT                        (ir_rc5_remote_gap * 2)
> -#define RC5_DURATION           (ir_rc5_remote_gap * RC5_NBITS)
> +#define RC5_UNIT               889 /* us */
> +#define        RC5_BITS                14
>
>  /* Used to register rc5_decoder clients */
>  static LIST_HEAD(decoder_list);
> -static spinlock_t decoder_lock;
> +static DEFINE_SPINLOCK(decoder_lock);
>
>  enum rc5_state {
>        STATE_INACTIVE,
> -       STATE_MARKSPACE,
> -       STATE_TRAILER,
> -};
> -
> -struct rc5_code {
> -       u8      address;
> -       u8      command;
> +       STATE_BIT_START,
> +       STATE_BIT_END,
> +       STATE_FINISHED,
>  };
>
>  struct decoder_data {
> @@ -49,8 +40,9 @@
>
>        /* State machine control */
>        enum rc5_state          state;
> -       struct rc5_code         rc5_code;
> -       unsigned                code, elapsed, last_bit, last_code;
> +       u32                     rc5_bits;
> +       int                     last_delta;
> +       unsigned                count;
>  };
>
>
> @@ -122,18 +114,19 @@
>  };
>
>  /**
> - * handle_event() - Decode one RC-5 pulse or space
> + * ir_rc5_decode() - Decode one RC-5 pulse or space
>  * @input_dev: the struct input_dev descriptor of the device
> - * @ev:                event array with type/duration of pulse/space
> + * @duration:  duration of pulse/space
>  *
>  * This function returns -EINVAL if the pulse violates the state machine
>  */
> -static int ir_rc5_decode(struct input_dev *input_dev,
> -                       struct ir_raw_event *ev)
> +static int ir_rc5_decode(struct input_dev *input_dev, int duration)
>  {
>        struct decoder_data *data;
>        struct ir_input_dev *ir_dev = input_get_drvdata(input_dev);
> -       int is_pulse, scancode, delta, toggle;
> +       u8 command, system, toggle;
> +       u32 scancode;
> +       int d;
>
>        data = get_decoder_data(ir_dev);
>        if (!data)
> @@ -142,79 +135,86 @@
>        if (!data->enabled)
>                return 0;
>
> -       delta = DIV_ROUND_CLOSEST(ev->delta.tv_nsec, ir_rc5_remote_gap);
> +       if (duration == 0) {
> +               data->state = STATE_INACTIVE;
> +               return 0;
> +       }
>
> -       /* The duration time refers to the last bit time */
> -       is_pulse = (ev->type & IR_PULSE) ? 1 : 0;
> +       d = DIV_ROUND_CLOSEST(abs(duration), RC5_UNIT);
> +       if (duration < 0)
> +               d = -d;

d = abs(duration)
d = DIV_ROUND_CLOSEST(d, RC5_UNIT);

> +
> +again:
> +       IR_dprintk(2, "RC5 decode started at state %i (%i units, %ius)\n",
> +                  data->state, d, duration);
>
> -       /* Very long delays are considered as start events */
> -       if (delta > RC5_DURATION || (ev->type & IR_START_EVENT))
> -               data->state = STATE_INACTIVE;
> +       if (d == 0 && data->state != STATE_FINISHED)
> +               return 0;
>
>        switch (data->state) {
> +
>        case STATE_INACTIVE:
> -       IR_dprintk(2, "currently inative. Start bit (%s) @%uus\n",
> -                  is_pulse ? "pulse" : "space",
> -                  (unsigned)(ev->delta.tv_nsec + 500) / 1000);
> -
> -               /* Discards the initial start space */
> -               if (!is_pulse)
> -                       goto err;
> -               data->code = 1;
> -               data->last_bit = 1;
> -               data->elapsed = 0;
> -               memset(&data->rc5_code, 0, sizeof(data->rc5_code));
> -               data->state = STATE_MARKSPACE;
> -               return 0;
> -       case STATE_MARKSPACE:
> -               if (delta != 1)
> -                       data->last_bit = data->last_bit ? 0 : 1;
> +               if ((d == 1) || (d == 2)) {
> +                       data->state = STATE_BIT_START;
> +                       data->count = 1;
> +                       d--;
> +                       goto again;
> +               }
> +               break;
>
> -               data->elapsed += delta;
> +       case STATE_BIT_START:
> +               if (abs(d) == 1) {
> +                       data->rc5_bits <<= 1;
> +                       if (d == -1)
> +                               data->rc5_bits |= 1;
> +                       data->count++;
> +                       data->last_delta = d;
> +
> +                       /*
> +                        * If the last bit is zero, a space will "merge"
> +                        * with the silence after the command.
> +                        */
> +                       if ((data->count == data->wanted_bits) && (d == 1))
> +                               data->state = STATE_FINISHED;
> +                       else
> +                               data->state = STATE_BIT_END;
>
> -               if ((data->elapsed % 2) == 1)
>                        return 0;
> +               }
> +               break;
>
> -               data->code <<= 1;
> -               data->code |= data->last_bit;
> -
> -               /* Fill the 2 unused bits at the command with 0 */
> -               if (data->elapsed / 2 == 6)
> -                       data->code <<= 2;
> -
> -               if (data->elapsed >= (RC5_NBITS - 1) * 2) {
> -                       scancode = data->code;
> -
> -                       /* Check for the start bits */
> -                       if ((scancode & 0xc000) != 0xc000) {
> -                               IR_dprintk(1, "Code 0x%04x doesn't have two start bits. It is not RC-5\n", scancode);
> -                               goto err;
> -                       }
> -
> -                       toggle = (scancode & 0x2000) ? 1 : 0;
> -
> -                       if (scancode == data->last_code) {
> -                               IR_dprintk(1, "RC-5 repeat\n");
> -                               ir_repeat(input_dev);
> -                       } else {
> -                               data->last_code = scancode;
> -                               scancode &= 0x1fff;
> -                               IR_dprintk(1, "RC-5 scancode 0x%04x\n", scancode);
> -
> -                               ir_keydown(input_dev, scancode, 0);
> -                       }
> -                       data->state = STATE_TRAILER;
> +       case STATE_BIT_END:
> +               /* delta and last_delta signedness must differ */
> +               if (d * data->last_delta < 0) {
> +                       if (data->count == RC5_BITS)
> +                               data->state = STATE_FINISHED;
> +                       else
> +                               data->state = STATE_BIT_START;
> +
> +                       if (d > 0)
> +                               d--;
> +                       else if (d < 0)
> +                               d++;
> +                       goto again;
>                }
> -               return 0;
> -       case STATE_TRAILER:
> +               break;
> +
> +       case STATE_FINISHED:
> +               command  = (data->rc5_bits & 0x0003F) >> 0;
> +               system   = (data->rc5_bits & 0x007C0) >> 6;
> +               toggle   = (data->rc5_bits & 0x00800) ? 1 : 0;
> +               command += (data->rc5_bits & 0x01000) ? 0 : 0x40;
> +               scancode = system << 8 | command;
> +
> +               IR_dprintk(1, "RC5 scancode 0x%04x (toggle: %u)\n",
> +                          scancode, toggle);
> +               ir_keydown(input_dev, scancode, toggle);
>                data->state = STATE_INACTIVE;
>                return 0;
>        }
>
> -err:
> -       IR_dprintk(1, "RC-5 decoded failed at %s @ %luus\n",
> -                  is_pulse ? "pulse" : "space",
> -                  (ev->delta.tv_nsec + 500) / 1000);
> +       IR_dprintk(1, "RC5 decode failed at state %i (%i units, %ius)\n",
> +                  data->state, d, duration);
>        data->state = STATE_INACTIVE;
>        return -EINVAL;
>  }
> Index: ir/drivers/media/video/saa7134/saa7134-input.c
> ===================================================================
> --- ir.orig/drivers/media/video/saa7134/saa7134-input.c 2010-04-06 12:30:16.428854774 +0200
> +++ ir/drivers/media/video/saa7134/saa7134-input.c      2010-04-07 18:05:27.756852639 +0200
> @@ -1021,7 +1021,7 @@
>        saa_clearb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
>        saa_setb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
>        space = saa_readl(SAA7134_GPIO_GPSTATUS0 >> 2) & ir->mask_keydown;
> -       ir_raw_event_store(dev->remote->dev, space ? IR_SPACE : IR_PULSE);
> +       ir_raw_event_store_edge(dev->remote->dev, space ? IR_SPACE : IR_PULSE);
>
>
>        /*
>
> --
> David Härdeman
> --
> To unsubscribe from this list: send the line "unsubscribe linux-input" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html
>



-- 
Jon Smirl
jonsmirl@gmail.com

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-07 20:18 [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations David Härdeman
  2010-04-07 21:01 ` Jon Smirl
  2010-04-08  0:18 ` Jon Smirl
@ 2010-04-08  0:44 ` Andy Walls
  2010-04-08  5:10 ` Mauro Carvalho Chehab
  3 siblings, 0 replies; 13+ messages in thread
From: Andy Walls @ 2010-04-08  0:44 UTC (permalink / raw)
  To: David Härdeman; +Cc: mchehab, linux-input, linux-media

On Wed, 2010-04-07 at 22:18 +0200, David Härdeman wrote:
> drivers/media/IR/ir-raw-event.c is currently written with the assumption 
> that all "raw" hardware will generate events only on state change (i.e.  
> when a pulse or space starts).
> 
> However, some hardware (like mceusb, probably the most popular IR receiver
> out there) only generates duration data (and that data is buffered so using
> any kind of timing on the data is futile).
> 
> Furthermore, using signed int's to represent pulse/space durations in ms
> is a well-known approach to anyone with experience in writing ir decoders.
> 
> This patch (which has been tested this time) is still a RFC on my proposed
> interface changes.
> 
> Changes since last version:
> 
> o RC5x and NECx support no longer added in patch (separate patches to follow)
> 
> o The use of a kfifo has been left given feedback from Jon, Andy, Mauro
> 
> o The RX decoding is now handled via a workqueue (I can break that up into a
>   separate patch later, but I think it helps the discussion to have it in for
>   now), with inspiration from Andy's code.

I haven't looked at your patches yet (sorry, busy month for me), but I'd
like to explain my motivation for a workqueue and you can see if the
rationale fits here:


I did the heavy lifting for IR Rx from integrated IR controllers in a
workqueue to keep hard IRQ handlers for video bridge chips as short as
possible.  Getting the hard IRQ handler done in the minimum amount of
time and PCI MMIO cycles was the objective.

Video buffer DMA done IRQ's a much more improtant than dawdling with IR.
The work queue can take as long as it needs to decode IR pulses (convert
timebase, parallel decode, or FFT for all I cared) and generate an input
keypress. :)

Regards,
Andy



> o Separate reset operations are no longer added to decoders, a duration of
>   zero is instead used to signal a reset (which allows the reset request to
>   be inserted into the kfifo).
> 
> o Not sent using quilt...Mauro, does it still trip up your MUA?
> 
> Not changed:
> 
> o int's are still used to represent pulse/space durations in ms. Mauro and I
>   seem to disagree on this one but I'm right :)
> 
> Index: ir/drivers/media/IR/ir-raw-event.c
> ===================================================================
> --- ir.orig/drivers/media/IR/ir-raw-event.c	2010-04-06 12:16:27.000000000 +0200
> +++ ir/drivers/media/IR/ir-raw-event.c	2010-04-07 21:32:13.961850481 +0200
> @@ -15,13 +15,14 @@
>  #include <media/ir-core.h>
>  #include <linux/workqueue.h>
>  #include <linux/spinlock.h>
> +#include <linux/sched.h>
>  
>  /* Define the max number of bit transitions per IR keycode */
>  #define MAX_IR_EVENT_SIZE	256
>  
>  /* Used to handle IR raw handler extensions */
>  static LIST_HEAD(ir_raw_handler_list);
> -static spinlock_t ir_raw_handler_lock;
> +static DEFINE_SPINLOCK(ir_raw_handler_lock);
>  
>  /**
>   * RUN_DECODER()	- runs an operation on all IR decoders
> @@ -53,19 +54,30 @@
>  /* Used to load the decoders */
>  static struct work_struct wq_load;
>  
> +static void ir_raw_event_work(struct work_struct *work)
> +{
> +	int d;
> +	struct ir_raw_event_ctrl *raw =
> +		container_of(work, struct ir_raw_event_ctrl, rx_work);
> +
> +	while (kfifo_out(&raw->kfifo, &d, sizeof(d)) == sizeof(d))
> +		RUN_DECODER(decode, raw->input_dev, d);
> +}
> +
>  int ir_raw_event_register(struct input_dev *input_dev)
>  {
>  	struct ir_input_dev *ir = input_get_drvdata(input_dev);
> -	int rc, size;
> +	int rc;
>  
>  	ir->raw = kzalloc(sizeof(*ir->raw), GFP_KERNEL);
>  	if (!ir->raw)
>  		return -ENOMEM;
>  
> -	size = sizeof(struct ir_raw_event) * MAX_IR_EVENT_SIZE * 2;
> -	size = roundup_pow_of_two(size);
> +	ir->raw->input_dev = input_dev;
> +	INIT_WORK(&ir->raw->rx_work, ir_raw_event_work);
>  
> -	rc = kfifo_alloc(&ir->raw->kfifo, size, GFP_KERNEL);
> +	rc = kfifo_alloc(&ir->raw->kfifo, sizeof(int) * MAX_IR_EVENT_SIZE,
> +			 GFP_KERNEL);
>  	if (rc < 0) {
>  		kfree(ir->raw);
>  		ir->raw = NULL;
> @@ -91,6 +103,7 @@
>  	if (!ir->raw)
>  		return;
>  
> +	cancel_work_sync(&ir->raw->rx_work);
>  	RUN_DECODER(raw_unregister, input_dev);
>  
>  	kfifo_free(&ir->raw->kfifo);
> @@ -99,74 +112,85 @@
>  }
>  EXPORT_SYMBOL_GPL(ir_raw_event_unregister);
>  
> -int ir_raw_event_store(struct input_dev *input_dev, enum raw_event_type type)
> +/**
> + * ir_raw_event_store() - pass a pulse/space duration to the raw ir decoders
> + * @input_dev:	the struct input_dev device descriptor
> + * @duration:	duration of the pulse or space
> + *
> + * This routine passes a pulse/space duration to the raw ir decoding state
> + * machines. Pulses are signalled as positive values and spaces as negative
> + * values. A zero value will reset the decoding state machines.
> + */
> +int ir_raw_event_store(struct input_dev *input_dev, int duration)
>  {
> -	struct ir_input_dev	*ir = input_get_drvdata(input_dev);
> -	struct timespec		ts;
> -	struct ir_raw_event	event;
> -	int			rc;
> +	struct ir_input_dev *ir = input_get_drvdata(input_dev);
>  
>  	if (!ir->raw)
>  		return -EINVAL;
>  
> -	event.type = type;
> -	event.delta.tv_sec = 0;
> -	event.delta.tv_nsec = 0;
> +	if (kfifo_in(&ir->raw->kfifo, &duration, sizeof(duration)) != sizeof(duration))
> +		return -ENOMEM;
>  
> -	ktime_get_ts(&ts);
> +	return 0;
> +}
> +EXPORT_SYMBOL_GPL(ir_raw_event_store);
>  
> -	if (timespec_equal(&ir->raw->last_event, &event.delta))
> -		event.type |= IR_START_EVENT;
> -	else
> -		event.delta = timespec_sub(ts, ir->raw->last_event);
> +/**
> + * ir_raw_event_store_edge() - notify raw ir decoders of the start of a pulse/space
> + * @input_dev:	the struct input_dev device descriptor
> + * @type:	the type of the event that has occurred
> + *
> + * This routine is used to notify the raw ir decoders on the beginning of an
> + * ir pulse or space (or the start/end of ir reception). This is used by
> + * hardware which does not provide durations directly but only interrupts
> + * (or similar events) on state change.
> + */
> +int ir_raw_event_store_edge(struct input_dev *input_dev, enum raw_event_type type)
> +{
> +	struct ir_input_dev	*ir = input_get_drvdata(input_dev);
> +	ktime_t			now;
> +	s64			delta; /* us */
> +	int			rc = 0;
>  
> -	memcpy(&ir->raw->last_event, &ts, sizeof(ts));
> +	if (!ir->raw)
> +		return -EINVAL;
>  
> -	if (event.delta.tv_sec) {
> -		event.type |= IR_START_EVENT;
> -		event.delta.tv_sec = 0;
> -		event.delta.tv_nsec = 0;
> -	}
> +	now = ktime_get();
> +	delta = ktime_us_delta(now, ir->raw->last_event);
>  
> -	kfifo_in(&ir->raw->kfifo, &event, sizeof(event));
> +	/* Are we called for the first time? */
> +	if (!ir->raw->last_type)
> +		type |= IR_START_EVENT;
> +
> +	if (type & IR_START_EVENT)
> +		ir_raw_event_reset(input_dev);
> +	else if (ir->raw->last_type & IR_SPACE)
> +		rc = ir_raw_event_store(input_dev, (int)-delta);
> +	else if (ir->raw->last_type & IR_PULSE)
> +		rc = ir_raw_event_store(input_dev, (int)delta);
> +	else
> +		return 0;
>  
> +	ir->raw->last_event = now;
> +	ir->raw->last_type = type;
>  	return rc;
>  }
> -EXPORT_SYMBOL_GPL(ir_raw_event_store);
> +EXPORT_SYMBOL_GPL(ir_raw_event_store_edge);
>  
> -int ir_raw_event_handle(struct input_dev *input_dev)
> +/**
> + * ir_raw_event_handle() - schedules the decoding of stored ir data
> + * @input_dev:	the struct input_dev device descriptor
> + *
> + * This routine will signal the workqueue to start decoding stored ir data.
> + */
> +void ir_raw_event_handle(struct input_dev *input_dev)
>  {
> -	struct ir_input_dev		*ir = input_get_drvdata(input_dev);
> -	int				rc;
> -	struct ir_raw_event		ev;
> -	int 				len, i;
> -
> -	/*
> -	 * Store the events into a temporary buffer. This allows calling more than
> -	 * one decoder to deal with the received data
> -	 */
> -	len = kfifo_len(&ir->raw->kfifo) / sizeof(ev);
> -	if (!len)
> -		return 0;
> -
> -	for (i = 0; i < len; i++) {
> -		rc = kfifo_out(&ir->raw->kfifo, &ev, sizeof(ev));
> -		if (rc != sizeof(ev)) {
> -			IR_dprintk(1, "overflow error: received %d instead of %zd\n",
> -				   rc, sizeof(ev));
> -			return -EINVAL;
> -		}
> -		IR_dprintk(2, "event type %d, time before event: %07luus\n",
> -			ev.type, (ev.delta.tv_nsec + 500) / 1000);
> -		rc = RUN_DECODER(decode, input_dev, &ev);
> -	}
> +	struct ir_input_dev *ir = input_get_drvdata(input_dev);
>  
> -	/*
> -	 * Call all ir decoders. This allows decoding the same event with
> -	 * more than one protocol handler.
> -	 */
> +	if (!ir->raw)
> +		return;
>  
> -	return rc;
> +	schedule_work(&ir->raw->rx_work);
>  }
>  EXPORT_SYMBOL_GPL(ir_raw_event_handle);
>  
> @@ -205,8 +229,6 @@
>  
>  void ir_raw_init(void)
>  {
> -	spin_lock_init(&ir_raw_handler_lock);
> -
>  #ifdef MODULE
>  	INIT_WORK(&wq_load, init_decoders);
>  	schedule_work(&wq_load);
> Index: ir/include/media/ir-core.h
> ===================================================================
> --- ir.orig/include/media/ir-core.h	2010-04-06 12:16:27.000000000 +0200
> +++ ir/include/media/ir-core.h	2010-04-07 18:17:53.524850074 +0200
> @@ -57,14 +57,12 @@
>  	void		(*close)(void *priv);
>  };
>  
> -struct ir_raw_event {
> -	struct timespec		delta;	/* Time spent before event */
> -	enum raw_event_type	type;	/* event type */
> -};
> -
>  struct ir_raw_event_ctrl {
> -	struct kfifo			kfifo;		/* fifo for the pulse/space events */
> -	struct timespec			last_event;	/* when last event occurred */
> +	struct work_struct		rx_work;	/* for the rx decoding workqueue */
> +	struct kfifo			kfifo;		/* fifo for the pulse/space durations */
> +	ktime_t				last_event;	/* when last event occurred */
> +	enum raw_event_type		last_type;	/* last event type */
> +	struct input_dev		*input_dev;	/* pointer to the parent input_dev */
>  };
>  
>  struct ir_input_dev {
> @@ -89,8 +87,7 @@
>  struct ir_raw_handler {
>  	struct list_head list;
>  
> -	int (*decode)(struct input_dev *input_dev,
> -		      struct ir_raw_event *ev);
> +	int (*decode)(struct input_dev *input_dev, int duration);
>  	int (*raw_register)(struct input_dev *input_dev);
>  	int (*raw_unregister)(struct input_dev *input_dev);
>  };
> @@ -146,8 +143,13 @@
>  /* Routines from ir-raw-event.c */
>  int ir_raw_event_register(struct input_dev *input_dev);
>  void ir_raw_event_unregister(struct input_dev *input_dev);
> -int ir_raw_event_store(struct input_dev *input_dev, enum raw_event_type type);
> -int ir_raw_event_handle(struct input_dev *input_dev);
> +void ir_raw_event_handle(struct input_dev *input_dev);
> +int ir_raw_event_store(struct input_dev *input_dev, int duration);
> +int ir_raw_event_store_edge(struct input_dev *input_dev, enum raw_event_type type);
> +static inline void ir_raw_event_reset(struct input_dev *dev) {
> +	ir_raw_event_store(input_dev, 0);
> +	ir_raw_event_handle(input_dev);
> +}
>  int ir_raw_handler_register(struct ir_raw_handler *ir_raw_handler);
>  void ir_raw_handler_unregister(struct ir_raw_handler *ir_raw_handler);
>  void ir_raw_init(void);
> Index: ir/drivers/media/IR/ir-nec-decoder.c
> ===================================================================
> --- ir.orig/drivers/media/IR/ir-nec-decoder.c	2010-04-06 12:16:27.000000000 +0200
> +++ ir/drivers/media/IR/ir-nec-decoder.c	2010-04-07 18:32:10.892853025 +0200
> @@ -13,23 +13,23 @@
>   */
>  
>  #include <media/ir-core.h>
> +#include <linux/bitrev.h>
>  
>  #define NEC_NBITS		32
> -#define NEC_UNIT		559979 /* ns */
> -#define NEC_HEADER_MARK		(16 * NEC_UNIT)
> -#define NEC_HEADER_SPACE	(8 * NEC_UNIT)
> -#define NEC_REPEAT_SPACE	(4 * NEC_UNIT)
> -#define NEC_MARK		(NEC_UNIT)
> -#define NEC_0_SPACE		(NEC_UNIT)
> -#define NEC_1_SPACE		(3 * NEC_UNIT)
> +#define NEC_UNIT		562	/* us */
> +#define NEC_HEADER_MARK		16
> +#define NEC_HEADER_SPACE	-8
> +#define NEC_REPEAT_SPACE	-4
> +#define NEC_MARK		1
> +#define NEC_0_SPACE		-1
> +#define NEC_1_SPACE		-3
>  
>  /* Used to register nec_decoder clients */
>  static LIST_HEAD(decoder_list);
> -static spinlock_t decoder_lock;
> +static DEFINE_SPINLOCK(decoder_lock);
>  
>  enum nec_state {
>  	STATE_INACTIVE,
> -	STATE_HEADER_MARK,
>  	STATE_HEADER_SPACE,
>  	STATE_MARK,
>  	STATE_SPACE,
> @@ -37,13 +37,6 @@
>  	STATE_TRAILER_SPACE,
>  };
>  
> -struct nec_code {
> -	u8	address;
> -	u8	not_address;
> -	u8	command;
> -	u8	not_command;
> -};
> -
>  struct decoder_data {
>  	struct list_head	list;
>  	struct ir_input_dev	*ir_dev;
> @@ -51,7 +44,7 @@
>  
>  	/* State machine control */
>  	enum nec_state		state;
> -	struct nec_code		nec_code;
> +	u32			nec_bits;
>  	unsigned		count;
>  };
>  
> @@ -62,7 +55,6 @@
>   *
>   * Returns the struct decoder_data that corresponds to a device
>   */
> -
>  static struct decoder_data *get_decoder_data(struct  ir_input_dev *ir_dev)
>  {
>  	struct decoder_data *data = NULL;
> @@ -123,20 +115,20 @@
>  	.attrs	= decoder_attributes,
>  };
>  
> -
>  /**
>   * ir_nec_decode() - Decode one NEC pulse or space
>   * @input_dev:	the struct input_dev descriptor of the device
> - * @ev:		event array with type/duration of pulse/space
> + * @duration:	duration in us of pulse/space
>   *
>   * This function returns -EINVAL if the pulse violates the state machine
>   */
> -static int ir_nec_decode(struct input_dev *input_dev,
> -			 struct ir_raw_event *ev)
> +static int ir_nec_decode(struct input_dev *input_dev, int duration)
>  {
>  	struct decoder_data *data;
>  	struct ir_input_dev *ir_dev = input_get_drvdata(input_dev);
> -	int bit, last_bit;
> +	int d;
> +	u32 scancode;
> +	u8 address, not_address, command, not_command;
>  
>  	data = get_decoder_data(ir_dev);
>  	if (!data)
> @@ -145,150 +137,106 @@
>  	if (!data->enabled)
>  		return 0;
>  
> -	/* Except for the initial event, what matters is the previous bit */
> -	bit = (ev->type & IR_PULSE) ? 1 : 0;
> -
> -	last_bit = !bit;
> -
> -	/* Discards spurious space last_bits when inactive */
> -
> -	/* Very long delays are considered as start events */
> -	if (ev->delta.tv_nsec > NEC_HEADER_MARK + NEC_HEADER_SPACE - NEC_UNIT / 2)
> +	if (duration == 0) {
>  		data->state = STATE_INACTIVE;
> +		return 0;
> +	}
>  
> -	if (ev->type & IR_START_EVENT)
> -		data->state = STATE_INACTIVE;
> +	d = DIV_ROUND_CLOSEST(abs(duration), NEC_UNIT);
> +	if (duration < 0)
> +		d = -d;
> +
> +	IR_dprintk(2, "NEC decode started at state %d (%i units, %ius)\n",
> +		   data->state, d, duration);
>  
>  	switch (data->state) {
> -	case STATE_INACTIVE:
> -		if (!bit)		/* PULSE marks the start event */
> -			return 0;
>  
> -		data->count = 0;
> -		data->state = STATE_HEADER_MARK;
> -		memset (&data->nec_code, 0, sizeof(data->nec_code));
> -		return 0;
> -	case STATE_HEADER_MARK:
> -		if (!last_bit)
> -			goto err;
> -		if (ev->delta.tv_nsec < NEC_HEADER_MARK - 6 * NEC_UNIT)
> -			goto err;
> -		data->state = STATE_HEADER_SPACE;
> +	case STATE_INACTIVE:
> +		if (d == NEC_HEADER_MARK) {
> +			data->count = 0;
> +			data->state = STATE_HEADER_SPACE;
> +		}
>  		return 0;
> +
>  	case STATE_HEADER_SPACE:
> -		if (last_bit)
> -			goto err;
> -		if (ev->delta.tv_nsec >= NEC_HEADER_SPACE - NEC_UNIT / 2) {
> +		if (d == NEC_HEADER_SPACE) {
>  			data->state = STATE_MARK;
>  			return 0;
> -		}
> -
> -		if (ev->delta.tv_nsec >= NEC_REPEAT_SPACE - NEC_UNIT / 2) {
> +		} else if (d == NEC_REPEAT_SPACE) {
>  			ir_repeat(input_dev);
>  			IR_dprintk(1, "Repeat last key\n");
>  			data->state = STATE_TRAILER_MARK;
>  			return 0;
>  		}
> -		goto err;
> +		break;
> +
>  	case STATE_MARK:
> -		if (!last_bit)
> -			goto err;
> -		if ((ev->delta.tv_nsec > NEC_MARK + NEC_UNIT / 2) ||
> -		    (ev->delta.tv_nsec < NEC_MARK - NEC_UNIT / 2))
> -			goto err;
> -		data->state = STATE_SPACE;
> -		return 0;
> +		if (d == NEC_MARK) {
> +			data->state = STATE_SPACE;
> +			return 0;
> +		}
> +		break;
> +
>  	case STATE_SPACE:
> -		if (last_bit)
> -			goto err;
> +		if (d != NEC_0_SPACE && d != NEC_1_SPACE)
> +			break;
>  
> -		if ((ev->delta.tv_nsec >= NEC_0_SPACE - NEC_UNIT / 2) &&
> -		    (ev->delta.tv_nsec < NEC_0_SPACE + NEC_UNIT / 2))
> -			bit = 0;
> -		else if ((ev->delta.tv_nsec >= NEC_1_SPACE - NEC_UNIT / 2) &&
> -		         (ev->delta.tv_nsec < NEC_1_SPACE + NEC_UNIT / 2))
> -			bit = 1;
> -		else {
> -			IR_dprintk(1, "Decode failed at %d-th bit (%s) @%luus\n",
> -				data->count,
> -				last_bit ? "pulse" : "space",
> -				(ev->delta.tv_nsec + 500) / 1000);
> +		data->nec_bits <<= 1;
> +		if (d == NEC_1_SPACE)
> +			data->nec_bits |= 1;
> +		data->count++;
>  
> -			goto err2;
> +		if (data->count != NEC_NBITS) {
> +			data->state = STATE_MARK;
> +			return 0;
>  		}
>  
> -		/* Ok, we've got a valid bit. proccess it */
> -		if (bit) {
> -			int shift = data->count;
> -
> -			/*
> -			 * NEC transmit bytes on this temporal order:
> -			 * address | not address | command | not command
> -			 */
> -			if (shift < 8) {
> -				data->nec_code.address |= 1 << shift;
> -			} else if (shift < 16) {
> -				data->nec_code.not_address |= 1 << (shift - 8);
> -			} else if (shift < 24) {
> -				data->nec_code.command |= 1 << (shift - 16);
> -			} else {
> -				data->nec_code.not_command |= 1 << (shift - 24);
> -			}
> +		address     = bitrev8((data->nec_bits >> 24) & 0xff);
> +		not_address = bitrev8((data->nec_bits >> 16) & 0xff);
> +		command	    = bitrev8((data->nec_bits >>  8) & 0xff);
> +		not_command = bitrev8((data->nec_bits >>  0) & 0xff);
> +
> +		if ((command ^ not_command) != 0xff) {
> +			IR_dprintk(1, "NEC checksum error: received 0x%08x\n",
> +				   data->nec_bits);
> +			break;
>  		}
> -		if (++data->count == NEC_NBITS) {
> -			u32 scancode;
> -			/*
> -			 * Fixme: may need to accept Extended NEC protocol?
> -			 */
> -			if ((data->nec_code.command ^ data->nec_code.not_command) != 0xff)
> -				goto checksum_err;
> -
> -			if ((data->nec_code.address ^ data->nec_code.not_address) != 0xff) {
> -				/* Extended NEC */
> -				scancode = data->nec_code.address << 16 |
> -					   data->nec_code.not_address << 8 |
> -					   data->nec_code.command;
> -				IR_dprintk(1, "NEC scancode 0x%06x\n", scancode);
> -			} else {
> -				/* normal NEC */
> -				scancode = data->nec_code.address << 8 |
> -					   data->nec_code.command;
> -				IR_dprintk(1, "NEC scancode 0x%04x\n", scancode);
> -			}
> -			ir_keydown(input_dev, scancode, 0);
>  
> -			data->state = STATE_TRAILER_MARK;
> -		} else
> -			data->state = STATE_MARK;
> +		if ((address ^ not_address) != 0xff) {
> +			/* Extended NEC */
> +			scancode = address     << 16 |
> +				   not_address <<  8 |
> +				   command;
> +			IR_dprintk(1, "NEC (Ext) scancode 0x%06x\n", scancode);
> +		} else {
> +			/* normal NEC */
> +			scancode = address << 8 | command;
> +			IR_dprintk(1, "NEC scancode 0x%04x\n", scancode);
> +		}
> +
> +		ir_keydown(input_dev, scancode, 0);
> +		data->state = STATE_TRAILER_MARK;
>  		return 0;
> +
>  	case STATE_TRAILER_MARK:
> -		if (!last_bit)
> -			goto err;
> -		data->state = STATE_TRAILER_SPACE;
> -		return 0;
> +		if (d > 0) {
> +			data->state = STATE_TRAILER_SPACE;
> +			return 0;
> +		}
> +		break;
> +
>  	case STATE_TRAILER_SPACE:
> -		if (last_bit)
> -			goto err;
> -		data->state = STATE_INACTIVE;
> -		return 0;
> -	}
> +		if (d < 0) {
> +			data->state = STATE_INACTIVE;
> +			return 0;
> +		}
>  
> -err:
> -	IR_dprintk(1, "NEC decoded failed at state %d (%s) @ %luus\n",
> -		   data->state,
> -		   bit ? "pulse" : "space",
> -		   (ev->delta.tv_nsec + 500) / 1000);
> -err2:
> -	data->state = STATE_INACTIVE;
> -	return -EINVAL;
> +		break;
> +	}
>  
> -checksum_err:
> +	IR_dprintk(1, "NEC decode failed at state %d (%i units, %ius)\n",
> +		   data->state, d, duration);
>  	data->state = STATE_INACTIVE;
> -	IR_dprintk(1, "NEC checksum error: received 0x%02x%02x%02x%02x\n",
> -		   data->nec_code.address,
> -		   data->nec_code.not_address,
> -		   data->nec_code.command,
> -		   data->nec_code.not_command);
>  	return -EINVAL;
>  }
>  
> Index: ir/drivers/media/IR/ir-rc5-decoder.c
> ===================================================================
> --- ir.orig/drivers/media/IR/ir-rc5-decoder.c	2010-04-06 12:16:51.784849187 +0200
> +++ ir/drivers/media/IR/ir-rc5-decoder.c	2010-04-07 21:34:14.816870395 +0200
> @@ -15,31 +15,22 @@
>  /*
>   * This code only handles 14 bits RC-5 protocols. There are other variants
>   * that use a different number of bits. This is currently unsupported
> - * It considers a carrier of 36 kHz, with a total of 14 bits, where
> - * the first two bits are start bits, and a third one is a filing bit
>   */
>  
>  #include <media/ir-core.h>
>  
> -static unsigned int ir_rc5_remote_gap = 888888;
> -
> -#define RC5_NBITS		14
> -#define RC5_BIT			(ir_rc5_remote_gap * 2)
> -#define RC5_DURATION		(ir_rc5_remote_gap * RC5_NBITS)
> +#define RC5_UNIT		889 /* us */
> +#define	RC5_BITS		14
>  
>  /* Used to register rc5_decoder clients */
>  static LIST_HEAD(decoder_list);
> -static spinlock_t decoder_lock;
> +static DEFINE_SPINLOCK(decoder_lock);
>  
>  enum rc5_state {
>  	STATE_INACTIVE,
> -	STATE_MARKSPACE,
> -	STATE_TRAILER,
> -};
> -
> -struct rc5_code {
> -	u8	address;
> -	u8	command;
> +	STATE_BIT_START,
> +	STATE_BIT_END,
> +	STATE_FINISHED,
>  };
>  
>  struct decoder_data {
> @@ -49,8 +40,9 @@
>  
>  	/* State machine control */
>  	enum rc5_state		state;
> -	struct rc5_code		rc5_code;
> -	unsigned		code, elapsed, last_bit, last_code;
> +	u32			rc5_bits;
> +	int			last_delta;
> +	unsigned		count;
>  };
>  
> 
> @@ -122,18 +114,19 @@
>  };
>  
>  /**
> - * handle_event() - Decode one RC-5 pulse or space
> + * ir_rc5_decode() - Decode one RC-5 pulse or space
>   * @input_dev:	the struct input_dev descriptor of the device
> - * @ev:		event array with type/duration of pulse/space
> + * @duration:	duration of pulse/space
>   *
>   * This function returns -EINVAL if the pulse violates the state machine
>   */
> -static int ir_rc5_decode(struct input_dev *input_dev,
> -			struct ir_raw_event *ev)
> +static int ir_rc5_decode(struct input_dev *input_dev, int duration)
>  {
>  	struct decoder_data *data;
>  	struct ir_input_dev *ir_dev = input_get_drvdata(input_dev);
> -	int is_pulse, scancode, delta, toggle;
> +	u8 command, system, toggle;
> +	u32 scancode;
> +	int d;
>  
>  	data = get_decoder_data(ir_dev);
>  	if (!data)
> @@ -142,79 +135,86 @@
>  	if (!data->enabled)
>  		return 0;
>  
> -	delta = DIV_ROUND_CLOSEST(ev->delta.tv_nsec, ir_rc5_remote_gap);
> +	if (duration == 0) {
> +		data->state = STATE_INACTIVE;
> +		return 0;
> +	}
>  
> -	/* The duration time refers to the last bit time */
> -	is_pulse = (ev->type & IR_PULSE) ? 1 : 0;
> +	d = DIV_ROUND_CLOSEST(abs(duration), RC5_UNIT);
> +	if (duration < 0)
> +		d = -d;
> +
> +again:
> +	IR_dprintk(2, "RC5 decode started at state %i (%i units, %ius)\n",
> +		   data->state, d, duration);
>  
> -	/* Very long delays are considered as start events */
> -	if (delta > RC5_DURATION || (ev->type & IR_START_EVENT))
> -		data->state = STATE_INACTIVE;
> +	if (d == 0 && data->state != STATE_FINISHED)
> +		return 0;
>  
>  	switch (data->state) {
> +
>  	case STATE_INACTIVE:
> -	IR_dprintk(2, "currently inative. Start bit (%s) @%uus\n",
> -		   is_pulse ? "pulse" : "space",
> -		   (unsigned)(ev->delta.tv_nsec + 500) / 1000);
> -
> -		/* Discards the initial start space */
> -		if (!is_pulse)
> -			goto err;
> -		data->code = 1;
> -		data->last_bit = 1;
> -		data->elapsed = 0;
> -		memset(&data->rc5_code, 0, sizeof(data->rc5_code));
> -		data->state = STATE_MARKSPACE;
> -		return 0;
> -	case STATE_MARKSPACE:
> -		if (delta != 1)
> -			data->last_bit = data->last_bit ? 0 : 1;
> +		if ((d == 1) || (d == 2)) {
> +			data->state = STATE_BIT_START;
> +			data->count = 1;
> +			d--;
> +			goto again;
> +		}
> +		break;
>  
> -		data->elapsed += delta;
> +	case STATE_BIT_START:
> +		if (abs(d) == 1) {
> +			data->rc5_bits <<= 1;
> +			if (d == -1)
> +				data->rc5_bits |= 1;
> +			data->count++;
> +			data->last_delta = d;
> +
> +			/*
> +			 * If the last bit is zero, a space will "merge"
> +			 * with the silence after the command.
> +			 */
> +			if ((data->count == data->wanted_bits) && (d == 1))
> +				data->state = STATE_FINISHED;
> +			else
> +				data->state = STATE_BIT_END;
>  
> -		if ((data->elapsed % 2) == 1)
>  			return 0;
> +		}
> +		break;
>  
> -		data->code <<= 1;
> -		data->code |= data->last_bit;
> -
> -		/* Fill the 2 unused bits at the command with 0 */
> -		if (data->elapsed / 2 == 6)
> -			data->code <<= 2;
> -
> -		if (data->elapsed >= (RC5_NBITS - 1) * 2) {
> -			scancode = data->code;
> -
> -			/* Check for the start bits */
> -			if ((scancode & 0xc000) != 0xc000) {
> -				IR_dprintk(1, "Code 0x%04x doesn't have two start bits. It is not RC-5\n", scancode);
> -				goto err;
> -			}
> -
> -			toggle = (scancode & 0x2000) ? 1 : 0;
> -
> -			if (scancode == data->last_code) {
> -				IR_dprintk(1, "RC-5 repeat\n");
> -				ir_repeat(input_dev);
> -			} else {
> -				data->last_code = scancode;
> -				scancode &= 0x1fff;
> -				IR_dprintk(1, "RC-5 scancode 0x%04x\n", scancode);
> -
> -				ir_keydown(input_dev, scancode, 0);
> -			}
> -			data->state = STATE_TRAILER;
> +	case STATE_BIT_END:
> +		/* delta and last_delta signedness must differ */
> +		if (d * data->last_delta < 0) {
> +			if (data->count == RC5_BITS)
> +				data->state = STATE_FINISHED;
> +			else
> +				data->state = STATE_BIT_START;
> +
> +			if (d > 0)
> +				d--;
> +			else if (d < 0)
> +				d++;
> +			goto again;
>  		}
> -		return 0;
> -	case STATE_TRAILER:
> +		break;
> +
> +	case STATE_FINISHED:
> +		command  = (data->rc5_bits & 0x0003F) >> 0;
> +		system   = (data->rc5_bits & 0x007C0) >> 6;
> +		toggle   = (data->rc5_bits & 0x00800) ? 1 : 0;
> +		command += (data->rc5_bits & 0x01000) ? 0 : 0x40;
> +		scancode = system << 8 | command;
> +
> +		IR_dprintk(1, "RC5 scancode 0x%04x (toggle: %u)\n",
> +			   scancode, toggle);
> +		ir_keydown(input_dev, scancode, toggle);
>  		data->state = STATE_INACTIVE;
>  		return 0;
>  	}
>  
> -err:
> -	IR_dprintk(1, "RC-5 decoded failed at %s @ %luus\n",
> -		   is_pulse ? "pulse" : "space",
> -		   (ev->delta.tv_nsec + 500) / 1000);
> +	IR_dprintk(1, "RC5 decode failed at state %i (%i units, %ius)\n",
> +		   data->state, d, duration);
>  	data->state = STATE_INACTIVE;
>  	return -EINVAL;
>  }
> Index: ir/drivers/media/video/saa7134/saa7134-input.c
> ===================================================================
> --- ir.orig/drivers/media/video/saa7134/saa7134-input.c	2010-04-06 12:30:16.428854774 +0200
> +++ ir/drivers/media/video/saa7134/saa7134-input.c	2010-04-07 18:05:27.756852639 +0200
> @@ -1021,7 +1021,7 @@
>  	saa_clearb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
>  	saa_setb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
>  	space = saa_readl(SAA7134_GPIO_GPSTATUS0 >> 2) & ir->mask_keydown;
> -	ir_raw_event_store(dev->remote->dev, space ? IR_SPACE : IR_PULSE);
> +	ir_raw_event_store_edge(dev->remote->dev, space ? IR_SPACE : IR_PULSE);
>  
> 
>  	/*
> 
> -- 
> David Härdeman
> --
> To unsubscribe from this list: send the line "unsubscribe linux-media" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-07 20:18 [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations David Härdeman
                   ` (2 preceding siblings ...)
  2010-04-08  0:44 ` Andy Walls
@ 2010-04-08  5:10 ` Mauro Carvalho Chehab
  2010-04-08 11:23   ` David Härdeman
  2010-04-08 12:41   ` Jon Smirl
  3 siblings, 2 replies; 13+ messages in thread
From: Mauro Carvalho Chehab @ 2010-04-08  5:10 UTC (permalink / raw)
  To: David Härdeman; +Cc: linux-input, linux-media

David Härdeman wrote:
> drivers/media/IR/ir-raw-event.c is currently written with the assumption 
> that all "raw" hardware will generate events only on state change (i.e.  
> when a pulse or space starts).
> 
> However, some hardware (like mceusb, probably the most popular IR receiver
> out there) only generates duration data (and that data is buffered so using
> any kind of timing on the data is futile).
> 
> Furthermore, using signed int's to represent pulse/space durations in ms
> is a well-known approach to anyone with experience in writing ir decoders.
> 
> This patch (which has been tested this time) is still a RFC on my proposed
> interface changes.
> 
> Changes since last version:
> 
> o RC5x and NECx support no longer added in patch (separate patches to follow)
> 
> o The use of a kfifo has been left given feedback from Jon, Andy, Mauro

Ok.

> o The RX decoding is now handled via a workqueue (I can break that up into a
>   separate patch later, but I think it helps the discussion to have it in for
>   now), with inspiration from Andy's code.

I'm in doubt about that. the workqueue is called just after an event. this means
that, just after every IRQ trigger (assuming the worse case), the workqueue will 
be called.

On the previous code, it is drivers responsibility to call the function that 
de-queue. On saa7134, I've scheduled it to wake after 15 ms. So, instead of
32 wakeups, just one is done, and the additional delay introduced by it is not
enough to disturb the user.

 > o Separate reset operations are no longer added to decoders, a duration of
>   zero is instead used to signal a reset (which allows the reset request to
>   be inserted into the kfifo).
> 
> o Not sent using quilt...Mauro, does it still trip up your MUA?

Thank you! Btw, git mailsend doesn't have any troubles.

> Not changed:
> 
> o int's are still used to represent pulse/space durations in ms. Mauro and I
>   seem to disagree on this one but I'm right :)

:)

We both have different opinions on that. I didn't hear a good argument from you
why your're right and I am wrong ;)

Maybe we can try to make a deal on it.

What you're really doing is:

struct rc_event {
	u32	mark : 1;
	u32	duration :31;	/* in microsseconds */
};

Please, correct me if I'm wrong, but I suspect that the technical reasons behind your
proposal are:
	1) to save space at kfifo and argument exchange with the functions;
	2) nanoseconds is too short for a carrier at the order of 10^4 Hz;

My proposal is to do something like:

struct rc_event {
	enum rc_event_type type;
	u64 duration		/* in nanoseconds */

My rationale are:
	1) To make the decoder code less obfuscated;
	2) To use the same time measurement as used on kernel timers, avoiding an uneeded division
for IRQ and polling-based devices.

It might have some other non-technical issues, like foo/bar uses this/that, this means less changes
on some code, etc, but we shouldn't consider those non-technical issues when discussing
the architecture.

So, here's the deal:

Let's do something in-between. While I still think that using a different measure for
duration will add an unnecessary runtime conversion from kernel ktime into
microsseconds, for me, the most important point is to avoid obfuscating the code.

So, we can define a opaque type:

typedef u32 mark_duration_t;

To represent the rc_event struct (this isn't a number anymore - it is a struct with one
bit for mark/space and 31 bits for unsigned duration). The use of an opaque type may
avoid people to do common mistakes.

And use some macros to convert from this type, like:
		
#define DURATION(mark_duration)	abs(mark_duration)
#define MARK(duration)	(abs(duration))
#define SPACE(duration)	(-abs(duration))
#define IS_MARK(mark_duration)	((duration > 0) ? 1 : 0)
#define IS_SPACE(mark_duration)	((duration < 0) ? 1 : 0)
#define DURATION(mark_duration)	abs(mark_duration)
#define TO_UNITS(mark_duration, unit)	\
	do { \
		a = DIV_ROUND_CLOSEST(DURATION(mark_duration), unit); \
		a = (mark_duration < 0) ? -a: a; \
	} while (0)

And use it along the decoders:

<from your nec decoder>

instead of:

> +#define NEC_UNIT		562	/* us */
> +#define NEC_HEADER_MARK		16
> +#define NEC_HEADER_SPACE	-8
>
> d = DIV_ROUND_CLOSEST(abs(duration), NEC_UNIT);
> +	if (duration < 0)
> +		d = -d;

With macros, we'll have:

#define NEC_UNIT		DURATION(562)	/* us */
#define NEC_HEADER_MARK		MARK(16)	/* units */
#define NEC_HEADER_SPACE	SPACE(8)	/* units */

d = TO_UNITS(duration, NEC_UNIT);


<from your rc5 decoder>

Instead of this obfuscated code:

> d = DIV_ROUND_CLOSEST(abs(duration), RC5_UNIT);
> +	if (duration < 0)
> +		d = -d;
>
> +	case STATE_BIT_START:
> +		if (abs(d) == 1) {
> +			data->rc5_bits <<= 1;
> +			if (d == -1)
> +				data->rc5_bits |= 1;
> +			data->count++;
> +			data->last_delta = d;
> +


A less obfuscated code will be:

	d = TO_UNITS(duration, RC5_UNIT);

	case STATE_BIT_START:
		if (DURATION(d) == 1) {
			data->rc5_bits <<= 1;
			if (IS_SPACE(d))
				data->rc5_bits |= 1;
			data->count++;
			data->last_delta = d;


The compiled code will be identical, but it the code is now clearer, 
as, on all places that the opaque type is being used, a
macro is properly indicating what part of the "struct mark/duration" were used.


PS.: Macros untested - it is 2am here and I'm a little tired, so probably they
are not 100% ;) I hope you got the idea.

> 
> Index: ir/drivers/media/IR/ir-raw-event.c
> ===================================================================
> --- ir.orig/drivers/media/IR/ir-raw-event.c	2010-04-06 12:16:27.000000000 +0200
> +++ ir/drivers/media/IR/ir-raw-event.c	2010-04-07 21:32:13.961850481 +0200
> @@ -15,13 +15,14 @@
>  #include <media/ir-core.h>
>  #include <linux/workqueue.h>
>  #include <linux/spinlock.h>
> +#include <linux/sched.h>
>  
>  /* Define the max number of bit transitions per IR keycode */
>  #define MAX_IR_EVENT_SIZE	256
>  
>  /* Used to handle IR raw handler extensions */
>  static LIST_HEAD(ir_raw_handler_list);
> -static spinlock_t ir_raw_handler_lock;
> +static DEFINE_SPINLOCK(ir_raw_handler_lock);

(just a side note: I had to do the above change already, due to some lock troubles I'm
having - Patch were already send to the -git tree).


Cheers,
Mauro
--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08  5:10 ` Mauro Carvalho Chehab
@ 2010-04-08 11:23   ` David Härdeman
  2010-04-08 11:50     ` Mauro Carvalho Chehab
  2010-04-08 12:41   ` Jon Smirl
  1 sibling, 1 reply; 13+ messages in thread
From: David Härdeman @ 2010-04-08 11:23 UTC (permalink / raw)
  To: Mauro Carvalho Chehab; +Cc: linux-input, linux-media

On Thu, Apr 08, 2010 at 02:10:40AM -0300, Mauro Carvalho Chehab wrote:
> David Härdeman wrote:
> > 
> > o The RX decoding is now handled via a workqueue (I can break that up into a
> >   separate patch later, but I think it helps the discussion to have it in for
> >   now), with inspiration from Andy's code.
> 
> I'm in doubt about that. the workqueue is called just after an event. this means
> that, just after every IRQ trigger (assuming the worse case), the workqueue will 
> be called.

No

> On the previous code, it is drivers responsibility to call the function that 
> de-queue. On saa7134, I've scheduled it to wake after 15 ms. So, instead of
> 32 wakeups, just one is done, and the additional delay introduced by it is not
> enough to disturb the user.

It's still the case with my patch, the ir_raw_event_handle() function is 
still there and it will call schedule_work().

>> o int's are still used to represent pulse/space durations in ms.
>>   Mauro and I seem to disagree on this one but I'm right :)
> 
> :)
> 
> We both have different opinions on that. I didn't hear a good argument from you
> why your're right and I am wrong ;)
> 
> Maybe we can try to make a deal on it.
> 
> What you're really doing is:
> 
> struct rc_event {
> 	u32	mark : 1;
> 	u32	duration :31;	/* in microsseconds */
> };
> 
> Please, correct me if I'm wrong, but I suspect that the technical reasons behind your
> proposal are:
> 	1) to save space at kfifo and argument exchange with the functions;
> 	2) nanoseconds is too short for a carrier at the order of 10^4 
> 	Hz;
> 
> My proposal is to do something like:
> 
> struct rc_event {
> 	enum rc_event_type type;
> 	u64 duration		/* in nanoseconds */
> 
> My rationale are:
> 	1) To make the decoder code less obfuscated;

Subjective

> 	2) To use the same time measurement as used on kernel timers, avoiding an uneeded division
> for IRQ and polling-based devices.

Are you sure you don't want to rewrite ir_raw_event_store_edge() and 
ir_raw_event_store() in assembly?
> 
> It might have some other non-technical issues, like foo/bar uses this/that, this means less changes
> on some code, etc, but we shouldn't consider those non-technical issues when discussing
> the architecture.
> 
> So, here's the deal:
>
> 
> Let's do something in-between. While I still think that using a different measure for
> duration will add an unnecessary runtime conversion from kernel ktime into
> microsseconds, for me, the most important point is to avoid obfuscating the code.
> 
> So, we can define a opaque type:
> 
> typedef u32 mark_duration_t;
> 
> To represent the rc_event struct (this isn't a number anymore - it is a struct with one
> bit for mark/space and 31 bits for unsigned duration). The use of an opaque type may
> avoid people to do common mistakes.

I've seldom seen a case where a "typedef gobbledygook" is considered 
clearer than a native data type.

> And use some macros to convert from this type, like:
> 		
> #define DURATION(mark_duration)	abs(mark_duration)
> #define MARK(duration)	(abs(duration))
> #define SPACE(duration)	(-abs(duration))
> #define IS_MARK(mark_duration)	((duration > 0) ? 1 : 0)
> #define IS_SPACE(mark_duration)	((duration < 0) ? 1 : 0)
> #define DURATION(mark_duration)	abs(mark_duration)
> #define TO_UNITS(mark_duration, unit)	\
> 	do { \
> 		a = DIV_ROUND_CLOSEST(DURATION(mark_duration), unit); \
> 		a = (mark_duration < 0) ? -a: a; \
> 	} while (0)
> 
> And use it along the decoders:

If you think a couple of defines would make it that much clearer, I can 
add some defines. If the division in ktime_us_delta() worries you that 
much, I can avoid it as well.

So how about:

s64 duration; /* signed to represent pulse/space, in ns */

This is the return value from ktime subtraction, so no conversion 
necessary. Then I'll also add defines along your lines.

New patch coming up...

-- 
David Härdeman

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08 11:23   ` David Härdeman
@ 2010-04-08 11:50     ` Mauro Carvalho Chehab
  2010-04-08 12:43       ` David Härdeman
  0 siblings, 1 reply; 13+ messages in thread
From: Mauro Carvalho Chehab @ 2010-04-08 11:50 UTC (permalink / raw)
  To: David Härdeman; +Cc: linux-input, linux-media

David Härdeman wrote:
> On Thu, Apr 08, 2010 at 02:10:40AM -0300, Mauro Carvalho Chehab wrote:
>> David Härdeman wrote:
>>> o The RX decoding is now handled via a workqueue (I can break that up into a
>>>   separate patch later, but I think it helps the discussion to have it in for
>>>   now), with inspiration from Andy's code.
>> I'm in doubt about that. the workqueue is called just after an event. this means
>> that, just after every IRQ trigger (assuming the worse case), the workqueue will 
>> be called.
> 
> No
> 
>> On the previous code, it is drivers responsibility to call the function that 
>> de-queue. On saa7134, I've scheduled it to wake after 15 ms. So, instead of
>> 32 wakeups, just one is done, and the additional delay introduced by it is not
>> enough to disturb the user.
> 
> It's still the case with my patch, the ir_raw_event_handle() function is 
> still there and it will call schedule_work().

OK!

>>> o int's are still used to represent pulse/space durations in ms.
>>>   Mauro and I seem to disagree on this one but I'm right :)
>> :)
>>
>> We both have different opinions on that. I didn't hear a good argument from you
>> why your're right and I am wrong ;)
>>
>> Maybe we can try to make a deal on it.
>>
>> What you're really doing is:
>>
>> struct rc_event {
>> 	u32	mark : 1;
>> 	u32	duration :31;	/* in microsseconds */
>> };
>>
>> Please, correct me if I'm wrong, but I suspect that the technical reasons behind your
>> proposal are:
>> 	1) to save space at kfifo and argument exchange with the functions;
>> 	2) nanoseconds is too short for a carrier at the order of 10^4 
>> 	Hz;
>>
>> My proposal is to do something like:
>>
>> struct rc_event {
>> 	enum rc_event_type type;
>> 	u64 duration		/* in nanoseconds */
>>
>> My rationale are:
>> 	1) To make the decoder code less obfuscated;
> 
> Subjective
> 
>> 	2) To use the same time measurement as used on kernel timers, avoiding an uneeded division
>> for IRQ and polling-based devices.
> 
> Are you sure you don't want to rewrite ir_raw_event_store_edge() and 
> ir_raw_event_store() in assembly?

I want to take just the opposite direction ;) By reading the decoders on the first patch
you submitted me in priv, I remembered the time I used to program in assembler, back on
eigties. On that time, there programming magazines used to put a challenge requesting for
the smallest/fastest (and trickiest) code to solve a problem ;) Of course, nobody, author
included, were capable of understanding those codes after a few weeks, without spending
a few hours to get the tricks.

What I'm trying to say is that those protocol demods are tricky enough 
to require a lot of time to understand what the code is really doing. 
As understanding the code is a requirement for reviewers and for bug fixes, 
we should make life easier, by find the better way to implement the logic that
will help the decoder understanding.

>> It might have some other non-technical issues, like foo/bar uses this/that, this means less changes
>> on some code, etc, but we shouldn't consider those non-technical issues when discussing
>> the architecture.
>>
>> So, here's the deal:
>>
>>
>> Let's do something in-between. While I still think that using a different measure for
>> duration will add an unnecessary runtime conversion from kernel ktime into
>> microsseconds, for me, the most important point is to avoid obfuscating the code.
>>
>> So, we can define a opaque type:
>>
>> typedef u32 mark_duration_t;
>>
>> To represent the rc_event struct (this isn't a number anymore - it is a struct with one
>> bit for mark/space and 31 bits for unsigned duration). The use of an opaque type may
>> avoid people to do common mistakes.
> 
> I've seldom seen a case where a "typedef gobbledygook" is considered 
> clearer than a native data type.
> 
>> And use some macros to convert from this type, like:
>> 		
>> #define DURATION(mark_duration)	abs(mark_duration)
>> #define MARK(duration)	(abs(duration))
>> #define SPACE(duration)	(-abs(duration))
>> #define IS_MARK(mark_duration)	((duration > 0) ? 1 : 0)
>> #define IS_SPACE(mark_duration)	((duration < 0) ? 1 : 0)
>> #define DURATION(mark_duration)	abs(mark_duration)
>> #define TO_UNITS(mark_duration, unit)	\
>> 	do { \
>> 		a = DIV_ROUND_CLOSEST(DURATION(mark_duration), unit); \
>> 		a = (mark_duration < 0) ? -a: a; \
>> 	} while (0)
>>
>> And use it along the decoders:
> 
> If you think a couple of defines would make it that much clearer, I can 
> add some defines. If the division in ktime_us_delta() worries you that 
> much, I can avoid it as well.
> 
> So how about:
> 
> s64 duration; /* signed to represent pulse/space, in ns */
> 
> This is the return value from ktime subtraction, so no conversion 
> necessary. Then I'll also add defines along your lines.

Ok.
> 
> New patch coming up...

Ah, I just forgot to comment those lines (sorry, too tired yesterday):

>-	size = sizeof(struct ir_raw_event) * MAX_IR_EVENT_SIZE * 2;
>-	size = roundup_pow_of_two(size);
>+	ir->raw->input_dev = input_dev;
>+	INIT_WORK(&ir->raw->rx_work, ir_raw_event_work);
> 
>-	rc = kfifo_alloc(&ir->raw->kfifo, size, GFP_KERNEL);
>+	rc = kfifo_alloc(&ir->raw->kfifo, sizeof(int) * MAX_IR_EVENT_SIZE,
>+			 GFP_KERNEL);

kfifo logic requires a power of two buffer to work, so, please keep the
original roundup_pow_of_two() logic, or add a comment before MAX_IR_EVENT_SIZE.

-- 

Cheers,
Mauro

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08  5:10 ` Mauro Carvalho Chehab
  2010-04-08 11:23   ` David Härdeman
@ 2010-04-08 12:41   ` Jon Smirl
  2010-04-08 13:06     ` Mauro Carvalho Chehab
  1 sibling, 1 reply; 13+ messages in thread
From: Jon Smirl @ 2010-04-08 12:41 UTC (permalink / raw)
  To: Mauro Carvalho Chehab; +Cc: David Härdeman, linux-input, linux-media

On Thu, Apr 8, 2010 at 1:10 AM, Mauro Carvalho Chehab
<mchehab@infradead.org> wrote:
> David Härdeman wrote:
>> drivers/media/IR/ir-raw-event.c is currently written with the assumption
>> that all "raw" hardware will generate events only on state change (i.e.
>> when a pulse or space starts).
>>
>> However, some hardware (like mceusb, probably the most popular IR receiver
>> out there) only generates duration data (and that data is buffered so using
>> any kind of timing on the data is futile).
>>
>> Furthermore, using signed int's to represent pulse/space durations in ms
>> is a well-known approach to anyone with experience in writing ir decoders.
>>
>> This patch (which has been tested this time) is still a RFC on my proposed
>> interface changes.
>>
>> Changes since last version:
>>
>> o RC5x and NECx support no longer added in patch (separate patches to follow)
>>
>> o The use of a kfifo has been left given feedback from Jon, Andy, Mauro
>
> Ok.
>
>> o The RX decoding is now handled via a workqueue (I can break that up into a
>>   separate patch later, but I think it helps the discussion to have it in for
>>   now), with inspiration from Andy's code.
>
> I'm in doubt about that. the workqueue is called just after an event. this means
> that, just after every IRQ trigger (assuming the worse case), the workqueue will
> be called.
>
> On the previous code, it is drivers responsibility to call the function that
> de-queue. On saa7134, I've scheduled it to wake after 15 ms. So, instead of
> 32 wakeups, just one is done, and the additional delay introduced by it is not
> enough to disturb the user.

The wakeup is variable when the default thread is used. My quad core
desktop wakes up on every pulse. My embedded system wakes up about
every 15 pulses. The embedded system called schedule_work() fifteen
times from the IRQ, but the kernel collapsed them into a single
wakeup. I'd stick with the default thread and let the kernel get
around to processing IR whenever it has some time.

A workqueue has to be used at some point in the system. The input
subsystem calls that send messages to user space can't be called from
interrupt context.  I believe in handing off to the workqueue as soon
as possible for IR signals.

Keep this code in the core to simplify writing the drivers. My GPIO
timer driver example is very simple.

If you're worried about performance, none of this code matters. What
is more important is localizing memory accesses to avoid processor
cache misses. A cache miss can equal 1000 divides.

>
>  > o Separate reset operations are no longer added to decoders, a duration of
>>   zero is instead used to signal a reset (which allows the reset request to
>>   be inserted into the kfifo).
>>
>> o Not sent using quilt...Mauro, does it still trip up your MUA?
>
> Thank you! Btw, git mailsend doesn't have any troubles.
>
>> Not changed:
>>
>> o int's are still used to represent pulse/space durations in ms. Mauro and I
>>   seem to disagree on this one but I'm right :)
>
> :)
>
> We both have different opinions on that. I didn't hear a good argument from you
> why your're right and I am wrong ;)
>
> Maybe we can try to make a deal on it.
>
> What you're really doing is:
>
> struct rc_event {
>        u32     mark : 1;
>        u32     duration :31;   /* in microsseconds */
> };
>
> Please, correct me if I'm wrong, but I suspect that the technical reasons behind your
> proposal are:
>        1) to save space at kfifo and argument exchange with the functions;
>        2) nanoseconds is too short for a carrier at the order of 10^4 Hz;
>
> My proposal is to do something like:
>
> struct rc_event {
>        enum rc_event_type type;
>        u64 duration            /* in nanoseconds */
>
> My rationale are:
>        1) To make the decoder code less obfuscated;
>        2) To use the same time measurement as used on kernel timers, avoiding an uneeded division
> for IRQ and polling-based devices.
>
> It might have some other non-technical issues, like foo/bar uses this/that, this means less changes
> on some code, etc, but we shouldn't consider those non-technical issues when discussing
> the architecture.
>
> So, here's the deal:
>
> Let's do something in-between. While I still think that using a different measure for
> duration will add an unnecessary runtime conversion from kernel ktime into
> microsseconds, for me, the most important point is to avoid obfuscating the code.
>
> So, we can define a opaque type:
>
> typedef u32 mark_duration_t;
>
> To represent the rc_event struct (this isn't a number anymore - it is a struct with one
> bit for mark/space and 31 bits for unsigned duration). The use of an opaque type may
> avoid people to do common mistakes.
>
> And use some macros to convert from this type, like:
>
> #define DURATION(mark_duration) abs(mark_duration)
> #define MARK(duration)  (abs(duration))
> #define SPACE(duration) (-abs(duration))
> #define IS_MARK(mark_duration)  ((duration > 0) ? 1 : 0)
> #define IS_SPACE(mark_duration) ((duration < 0) ? 1 : 0)
> #define DURATION(mark_duration) abs(mark_duration)
> #define TO_UNITS(mark_duration, unit)   \
>        do { \
>                a = DIV_ROUND_CLOSEST(DURATION(mark_duration), unit); \
>                a = (mark_duration < 0) ? -a: a; \
>        } while (0)
>
> And use it along the decoders:
>
> <from your nec decoder>
>
> instead of:
>
>> +#define NEC_UNIT             562     /* us */
>> +#define NEC_HEADER_MARK              16
>> +#define NEC_HEADER_SPACE     -8
>>
>> d = DIV_ROUND_CLOSEST(abs(duration), NEC_UNIT);
>> +     if (duration < 0)
>> +             d = -d;
>
> With macros, we'll have:
>
> #define NEC_UNIT                DURATION(562)   /* us */
> #define NEC_HEADER_MARK         MARK(16)        /* units */
> #define NEC_HEADER_SPACE        SPACE(8)        /* units */
>
> d = TO_UNITS(duration, NEC_UNIT);
>
>
> <from your rc5 decoder>
>
> Instead of this obfuscated code:
>
>> d = DIV_ROUND_CLOSEST(abs(duration), RC5_UNIT);
>> +     if (duration < 0)
>> +             d = -d;
>>
>> +     case STATE_BIT_START:
>> +             if (abs(d) == 1) {
>> +                     data->rc5_bits <<= 1;
>> +                     if (d == -1)
>> +                             data->rc5_bits |= 1;
>> +                     data->count++;
>> +                     data->last_delta = d;
>> +
>
>
> A less obfuscated code will be:
>
>        d = TO_UNITS(duration, RC5_UNIT);
>
>        case STATE_BIT_START:
>                if (DURATION(d) == 1) {
>                        data->rc5_bits <<= 1;
>                        if (IS_SPACE(d))
>                                data->rc5_bits |= 1;
>                        data->count++;
>                        data->last_delta = d;
>
>
> The compiled code will be identical, but it the code is now clearer,
> as, on all places that the opaque type is being used, a
> macro is properly indicating what part of the "struct mark/duration" were used.
>
>
> PS.: Macros untested - it is 2am here and I'm a little tired, so probably they
> are not 100% ;) I hope you got the idea.
>
>>
>> Index: ir/drivers/media/IR/ir-raw-event.c
>> ===================================================================
>> --- ir.orig/drivers/media/IR/ir-raw-event.c   2010-04-06 12:16:27.000000000 +0200
>> +++ ir/drivers/media/IR/ir-raw-event.c        2010-04-07 21:32:13.961850481 +0200
>> @@ -15,13 +15,14 @@
>>  #include <media/ir-core.h>
>>  #include <linux/workqueue.h>
>>  #include <linux/spinlock.h>
>> +#include <linux/sched.h>
>>
>>  /* Define the max number of bit transitions per IR keycode */
>>  #define MAX_IR_EVENT_SIZE    256
>>
>>  /* Used to handle IR raw handler extensions */
>>  static LIST_HEAD(ir_raw_handler_list);
>> -static spinlock_t ir_raw_handler_lock;
>> +static DEFINE_SPINLOCK(ir_raw_handler_lock);
>
> (just a side note: I had to do the above change already, due to some lock troubles I'm
> having - Patch were already send to the -git tree).
>
>
> Cheers,
> Mauro
> --
> To unsubscribe from this list: send the line "unsubscribe linux-input" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html
>



-- 
Jon Smirl
jonsmirl@gmail.com
--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08 11:50     ` Mauro Carvalho Chehab
@ 2010-04-08 12:43       ` David Härdeman
  2010-04-08 13:07         ` Mauro Carvalho Chehab
  0 siblings, 1 reply; 13+ messages in thread
From: David Härdeman @ 2010-04-08 12:43 UTC (permalink / raw)
  To: Mauro Carvalho Chehab; +Cc: linux-input, linux-media

On Thu, Apr 08, 2010 at 08:50:48AM -0300, Mauro Carvalho Chehab wrote:
> >-	size = sizeof(struct ir_raw_event) * MAX_IR_EVENT_SIZE * 2;
> >-	size = roundup_pow_of_two(size);
> >+	ir->raw->input_dev = input_dev;
> >+	INIT_WORK(&ir->raw->rx_work, ir_raw_event_work);
> > 
> >-	rc = kfifo_alloc(&ir->raw->kfifo, size, GFP_KERNEL);
> >+	rc = kfifo_alloc(&ir->raw->kfifo, sizeof(int) * MAX_IR_EVENT_SIZE,
> >+			 GFP_KERNEL);
> 
> kfifo logic requires a power of two buffer to work, so, please keep the
> original roundup_pow_of_two() logic, or add a comment before MAX_IR_EVENT_SIZE.

No, kfifo_alloc() takes care of the rounding up. See the code for 
kfifo_alloc() in kernel/kfifo.c.

-- 
David Härdeman
--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08 12:41   ` Jon Smirl
@ 2010-04-08 13:06     ` Mauro Carvalho Chehab
  2010-04-08 15:53       ` David Härdeman
  0 siblings, 1 reply; 13+ messages in thread
From: Mauro Carvalho Chehab @ 2010-04-08 13:06 UTC (permalink / raw)
  To: Jon Smirl; +Cc: David Härdeman, linux-input, linux-media

Jon Smirl wrote:
> On Thu, Apr 8, 2010 at 1:10 AM, Mauro Carvalho Chehab
> <mchehab@infradead.org> wrote:
>> David Härdeman wrote:
>>> drivers/media/IR/ir-raw-event.c is currently written with the assumption
>>> that all "raw" hardware will generate events only on state change (i.e.
>>> when a pulse or space starts).
>>>
>>> However, some hardware (like mceusb, probably the most popular IR receiver
>>> out there) only generates duration data (and that data is buffered so using
>>> any kind of timing on the data is futile).
>>>
>>> Furthermore, using signed int's to represent pulse/space durations in ms
>>> is a well-known approach to anyone with experience in writing ir decoders.
>>>
>>> This patch (which has been tested this time) is still a RFC on my proposed
>>> interface changes.
>>>
>>> Changes since last version:
>>>
>>> o RC5x and NECx support no longer added in patch (separate patches to follow)
>>>
>>> o The use of a kfifo has been left given feedback from Jon, Andy, Mauro
>> Ok.
>>
>>> o The RX decoding is now handled via a workqueue (I can break that up into a
>>>   separate patch later, but I think it helps the discussion to have it in for
>>>   now), with inspiration from Andy's code.
>> I'm in doubt about that. the workqueue is called just after an event. this means
>> that, just after every IRQ trigger (assuming the worse case), the workqueue will
>> be called.
>>
>> On the previous code, it is drivers responsibility to call the function that
>> de-queue. On saa7134, I've scheduled it to wake after 15 ms. So, instead of
>> 32 wakeups, just one is done, and the additional delay introduced by it is not
>> enough to disturb the user.
> 
> The wakeup is variable when the default thread is used. My quad core
> desktop wakes up on every pulse. My embedded system wakes up about
> every 15 pulses. The embedded system called schedule_work() fifteen
> times from the IRQ, but the kernel collapsed them into a single
> wakeup. I'd stick with the default thread and let the kernel get
> around to processing IR whenever it has some time.

Makes sense.

> A workqueue has to be used at some point in the system. The input
> subsystem calls that send messages to user space can't be called from
> interrupt context.  I believe in handing off to the workqueue as soon
> as possible for IR signals.

I'm ok on using a workqueue for it.
> 
> Keep this code in the core to simplify writing the drivers. My GPIO
> timer driver example is very simple.
> 
> If you're worried about performance, none of this code matters. What
> is more important is localizing memory accesses to avoid processor
> cache misses. A cache miss can equal 1000 divides.

Agreed.


-- 

Cheers,
Mauro
--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08 12:43       ` David Härdeman
@ 2010-04-08 13:07         ` Mauro Carvalho Chehab
  0 siblings, 0 replies; 13+ messages in thread
From: Mauro Carvalho Chehab @ 2010-04-08 13:07 UTC (permalink / raw)
  To: David Härdeman; +Cc: linux-input, linux-media

David Härdeman wrote:
> On Thu, Apr 08, 2010 at 08:50:48AM -0300, Mauro Carvalho Chehab wrote:
>>> -	size = sizeof(struct ir_raw_event) * MAX_IR_EVENT_SIZE * 2;
>>> -	size = roundup_pow_of_two(size);
>>> +	ir->raw->input_dev = input_dev;
>>> +	INIT_WORK(&ir->raw->rx_work, ir_raw_event_work);
>>>
>>> -	rc = kfifo_alloc(&ir->raw->kfifo, size, GFP_KERNEL);
>>> +	rc = kfifo_alloc(&ir->raw->kfifo, sizeof(int) * MAX_IR_EVENT_SIZE,
>>> +			 GFP_KERNEL);
>> kfifo logic requires a power of two buffer to work, so, please keep the
>> original roundup_pow_of_two() logic, or add a comment before MAX_IR_EVENT_SIZE.
> 
> No, kfifo_alloc() takes care of the rounding up. See the code for 
> kfifo_alloc() in kernel/kfifo.c.
> 
Ok.

-- 

Cheers,
Mauro

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08 13:06     ` Mauro Carvalho Chehab
@ 2010-04-08 15:53       ` David Härdeman
  2010-04-08 17:04         ` Mauro Carvalho Chehab
  0 siblings, 1 reply; 13+ messages in thread
From: David Härdeman @ 2010-04-08 15:53 UTC (permalink / raw)
  To: Mauro Carvalho Chehab; +Cc: Jon Smirl, linux-input, linux-media

On Thu, Apr 08, 2010 at 10:06:53AM -0300, Mauro Carvalho Chehab wrote:
> Jon Smirl wrote:
> > On Thu, Apr 8, 2010 at 1:10 AM, Mauro Carvalho Chehab
> > <mchehab@infradead.org> wrote:
> >> On the previous code, it is drivers responsibility to call the 
> >> function that
> >> de-queue. On saa7134, I've scheduled it to wake after 15 ms. So, instead of
> >> 32 wakeups, just one is done, and the additional delay introduced by it is not
> >> enough to disturb the user.
> > 
> > The wakeup is variable when the default thread is used. My quad core
> > desktop wakes up on every pulse. My embedded system wakes up about
> > every 15 pulses. The embedded system called schedule_work() fifteen
> > times from the IRQ, but the kernel collapsed them into a single
> > wakeup. I'd stick with the default thread and let the kernel get
> > around to processing IR whenever it has some time.
> 
> Makes sense.

Given Jon's experience, it would perhaps make sense to remove 
ir_raw_event_handle() and call schedule_work() from every call to 
ir_raw_event_store()?

One thing less for IR drivers to care about...

-- 
David Härdeman
--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

^ permalink raw reply	[flat|nested] 13+ messages in thread

* Re: [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations
  2010-04-08 15:53       ` David Härdeman
@ 2010-04-08 17:04         ` Mauro Carvalho Chehab
  0 siblings, 0 replies; 13+ messages in thread
From: Mauro Carvalho Chehab @ 2010-04-08 17:04 UTC (permalink / raw)
  To: David Härdeman; +Cc: Jon Smirl, linux-input, linux-media

David Härdeman wrote:
> On Thu, Apr 08, 2010 at 10:06:53AM -0300, Mauro Carvalho Chehab wrote:
>> Jon Smirl wrote:
>>> On Thu, Apr 8, 2010 at 1:10 AM, Mauro Carvalho Chehab
>>> <mchehab@infradead.org> wrote:
>>>> On the previous code, it is drivers responsibility to call the 
>>>> function that
>>>> de-queue. On saa7134, I've scheduled it to wake after 15 ms. So, instead of
>>>> 32 wakeups, just one is done, and the additional delay introduced by it is not
>>>> enough to disturb the user.
>>> The wakeup is variable when the default thread is used. My quad core
>>> desktop wakes up on every pulse. My embedded system wakes up about
>>> every 15 pulses. The embedded system called schedule_work() fifteen
>>> times from the IRQ, but the kernel collapsed them into a single
>>> wakeup. I'd stick with the default thread and let the kernel get
>>> around to processing IR whenever it has some time.
>> Makes sense.
> 
> Given Jon's experience, it would perhaps make sense to remove 
> ir_raw_event_handle() and call schedule_work() from every call to 
> ir_raw_event_store()?
> 
> One thing less for IR drivers to care about...

Maybe, on a separate patch, but let's do it by the end of the changes,
to let people to give us some feedback about the practical effects
on the users side, and the corresponding perf impacts. 

I won't mind to move the mod_timer stuff from saa7134 to the core, 
as a way to easy this change.

-- 

Cheers,
Mauro

^ permalink raw reply	[flat|nested] 13+ messages in thread

end of thread, other threads:[~2010-04-08 17:04 UTC | newest]

Thread overview: 13+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2010-04-07 20:18 [RFC2] Teach drivers/media/IR/ir-raw-event.c to use durations David Härdeman
2010-04-07 21:01 ` Jon Smirl
2010-04-08  0:18 ` Jon Smirl
2010-04-08  0:44 ` Andy Walls
2010-04-08  5:10 ` Mauro Carvalho Chehab
2010-04-08 11:23   ` David Härdeman
2010-04-08 11:50     ` Mauro Carvalho Chehab
2010-04-08 12:43       ` David Härdeman
2010-04-08 13:07         ` Mauro Carvalho Chehab
2010-04-08 12:41   ` Jon Smirl
2010-04-08 13:06     ` Mauro Carvalho Chehab
2010-04-08 15:53       ` David Härdeman
2010-04-08 17:04         ` Mauro Carvalho Chehab

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).