ir-raw-event.c 9.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
/* ir-raw-event.c - handle IR Pulse/Space event
 *
 * Copyright (C) 2010 by Mauro Carvalho Chehab <mchehab@redhat.com>
 *
 * This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation version 2 of the License.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 */

15
#include <linux/kthread.h>
16
#include <linux/mutex.h>
17
#include <linux/sched.h>
18
#include <linux/freezer.h>
19
#include "ir-core-priv.h"
20

21 22
/* Define the max number of pulse/space transitions to buffer */
#define MAX_IR_EVENT_SIZE      512
23

24 25 26
/* Used to keep track of IR raw clients, protected by ir_raw_handler_lock */
static LIST_HEAD(ir_raw_client_list);

27
/* Used to handle IR raw handler extensions */
28
static DEFINE_MUTEX(ir_raw_handler_lock);
29 30
static LIST_HEAD(ir_raw_handler_list);
static u64 available_protocols;
31

32
#ifdef MODULE
33 34
/* Used to load the decoders */
static struct work_struct wq_load;
35
#endif
36

37
static int ir_raw_event_thread(void *data)
38
{
39
	struct ir_raw_event ev;
40
	struct ir_raw_handler *handler;
41 42 43 44
	struct ir_raw_event_ctrl *raw = (struct ir_raw_event_ctrl *)data;

	while (!kthread_should_stop()) {
		try_to_freeze();
45

46
		mutex_lock(&ir_raw_handler_lock);
47 48 49 50 51 52 53

		while (kfifo_out(&raw->kfifo, &ev, sizeof(ev)) == sizeof(ev)) {
			list_for_each_entry(handler, &ir_raw_handler_list, list)
				handler->decode(raw->input_dev, ev);
			raw->prev_ev = ev;
		}

54
		mutex_unlock(&ir_raw_handler_lock);
55 56 57

		set_current_state(TASK_INTERRUPTIBLE);
		schedule();
58
	}
59 60

	return 0;
61 62 63 64 65
}

/**
 * ir_raw_event_store() - pass a pulse/space duration to the raw ir decoders
 * @input_dev:	the struct input_dev device descriptor
66
 * @ev:		the struct ir_raw_event descriptor of the pulse/space
67 68 69 70 71 72
 *
 * This routine (which may be called from an interrupt context) stores a
 * pulse/space duration for 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.
 */
73
int ir_raw_event_store(struct input_dev *input_dev, struct ir_raw_event *ev)
74
{
75
	struct ir_input_dev *ir = input_get_drvdata(input_dev);
76 77 78 79

	if (!ir->raw)
		return -EINVAL;

M
Maxim Levitsky 已提交
80 81 82
	IR_dprintk(2, "sample: (05%dus %s)\n",
		TO_US(ev->duration), TO_STR(ev->pulse));

83
	if (kfifo_in(&ir->raw->kfifo, ev, sizeof(*ev)) != sizeof(*ev))
84
		return -ENOMEM;
85

86 87 88
	return 0;
}
EXPORT_SYMBOL_GPL(ir_raw_event_store);
89

90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105
/**
 * 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 (which may be called from an interrupt context) is used to
 * store the beginning of an ir pulse or space (or the start/end of ir
 * reception) for the raw ir decoding state machines. 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; /* ns */
106
	struct ir_raw_event	ev;
107
	int			rc = 0;
108

109 110
	if (!ir->raw)
		return -EINVAL;
111

112 113
	now = ktime_get();
	delta = ktime_to_ns(ktime_sub(now, ir->raw->last_event));
114

115 116 117 118
	/* Check for a long duration since last event or if we're
	 * being called for the first time, note that delta can't
	 * possibly be negative.
	 */
119 120
	ev.duration = 0;
	if (delta > IR_MAX_DURATION || !ir->raw->last_type)
121
		type |= IR_START_EVENT;
122 123
	else
		ev.duration = delta;
124 125 126

	if (type & IR_START_EVENT)
		ir_raw_event_reset(input_dev);
127 128 129 130 131 132 133
	else if (ir->raw->last_type & IR_SPACE) {
		ev.pulse = false;
		rc = ir_raw_event_store(input_dev, &ev);
	} else if (ir->raw->last_type & IR_PULSE) {
		ev.pulse = true;
		rc = ir_raw_event_store(input_dev, &ev);
	} else
134
		return 0;
135

136 137
	ir->raw->last_event = now;
	ir->raw->last_type = type;
138 139
	return rc;
}
140
EXPORT_SYMBOL_GPL(ir_raw_event_store_edge);
141

142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
/**
 * ir_raw_event_store_with_filter() - pass next pulse/space to decoders with some processing
 * @input_dev:	the struct input_dev device descriptor
 * @type:	the type of the event that has occurred
 *
 * This routine (which may be called from an interrupt context) works
 * in similiar manner to ir_raw_event_store_edge.
 * This routine is intended for devices with limited internal buffer
 * It automerges samples of same type, and handles timeouts
 */
int ir_raw_event_store_with_filter(struct input_dev *input_dev,
						struct ir_raw_event *ev)
{
	struct ir_input_dev *ir = input_get_drvdata(input_dev);
	struct ir_raw_event_ctrl *raw = ir->raw;

	if (!raw || !ir->props)
		return -EINVAL;

	/* Ignore spaces in idle mode */
	if (ir->idle && !ev->pulse)
		return 0;
	else if (ir->idle)
		ir_raw_event_set_idle(input_dev, 0);

	if (!raw->this_ev.duration) {
		raw->this_ev = *ev;
	} else if (ev->pulse == raw->this_ev.pulse) {
		raw->this_ev.duration += ev->duration;
	} else {
		ir_raw_event_store(input_dev, &raw->this_ev);
		raw->this_ev = *ev;
	}

	/* Enter idle mode if nessesary */
	if (!ev->pulse && ir->props->timeout &&
		raw->this_ev.duration >= ir->props->timeout)
		ir_raw_event_set_idle(input_dev, 1);
	return 0;
}
EXPORT_SYMBOL_GPL(ir_raw_event_store_with_filter);

void ir_raw_event_set_idle(struct input_dev *input_dev, int idle)
{
	struct ir_input_dev *ir = input_get_drvdata(input_dev);
	struct ir_raw_event_ctrl *raw = ir->raw;
	ktime_t now;
	u64 delta;

	if (!ir->props)
		return;

	if (!ir->raw)
		goto out;

	if (idle) {
		IR_dprintk(2, "enter idle mode\n");
		raw->last_event = ktime_get();
	} else {
		IR_dprintk(2, "exit idle mode\n");

		now = ktime_get();
		delta = ktime_to_ns(ktime_sub(now, ir->raw->last_event));

		WARN_ON(raw->this_ev.pulse);

		raw->this_ev.duration =
			min(raw->this_ev.duration + delta,
						(u64)IR_MAX_DURATION);

		ir_raw_event_store(input_dev, &raw->this_ev);

		if (raw->this_ev.duration == IR_MAX_DURATION)
			ir_raw_event_reset(input_dev);

		raw->this_ev.duration = 0;
	}
out:
	if (ir->props->s_idle)
		ir->props->s_idle(ir->props->priv, idle);
	ir->idle = idle;
}
EXPORT_SYMBOL_GPL(ir_raw_event_set_idle);

226 227 228 229 230 231 232
/**
 * 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)
233
{
234
	struct ir_input_dev *ir = input_get_drvdata(input_dev);
235

236 237
	if (!ir->raw)
		return;
238

239
	wake_up_process(ir->raw->thread);
240 241
}
EXPORT_SYMBOL_GPL(ir_raw_event_handle);
242

243 244 245 246 247
/* used internally by the sysfs interface */
u64
ir_raw_get_allowed_protocols()
{
	u64 protocols;
248
	mutex_lock(&ir_raw_handler_lock);
249
	protocols = available_protocols;
250
	mutex_unlock(&ir_raw_handler_lock);
251 252 253 254 255 256 257 258 259 260
	return protocols;
}

/*
 * Used to (un)register raw event clients
 */
int ir_raw_event_register(struct input_dev *input_dev)
{
	struct ir_input_dev *ir = input_get_drvdata(input_dev);
	int rc;
261
	struct ir_raw_handler *handler;
262 263 264 265 266 267

	ir->raw = kzalloc(sizeof(*ir->raw), GFP_KERNEL);
	if (!ir->raw)
		return -ENOMEM;

	ir->raw->input_dev = input_dev;
268

269 270 271 272 273 274 275 276 277
	ir->raw->enabled_protocols = ~0;
	rc = kfifo_alloc(&ir->raw->kfifo, sizeof(s64) * MAX_IR_EVENT_SIZE,
			 GFP_KERNEL);
	if (rc < 0) {
		kfree(ir->raw);
		ir->raw = NULL;
		return rc;
	}

278 279 280 281
	ir->raw->thread = kthread_run(ir_raw_event_thread, ir->raw,
			"rc%u",  (unsigned int)ir->devno);

	if (IS_ERR(ir->raw->thread)) {
282 283
		int ret = PTR_ERR(ir->raw->thread);

284 285
		kfree(ir->raw);
		ir->raw = NULL;
286
		return ret;
287 288
	}

289
	mutex_lock(&ir_raw_handler_lock);
290 291 292 293
	list_add_tail(&ir->raw->list, &ir_raw_client_list);
	list_for_each_entry(handler, &ir_raw_handler_list, list)
		if (handler->raw_register)
			handler->raw_register(ir->raw->input_dev);
294
	mutex_unlock(&ir_raw_handler_lock);
295

296
	return 0;
297 298 299 300 301
}

void ir_raw_event_unregister(struct input_dev *input_dev)
{
	struct ir_input_dev *ir = input_get_drvdata(input_dev);
302
	struct ir_raw_handler *handler;
303 304 305 306

	if (!ir->raw)
		return;

307
	kthread_stop(ir->raw->thread);
308

309
	mutex_lock(&ir_raw_handler_lock);
310 311 312 313
	list_del(&ir->raw->list);
	list_for_each_entry(handler, &ir_raw_handler_list, list)
		if (handler->raw_unregister)
			handler->raw_unregister(ir->raw->input_dev);
314
	mutex_unlock(&ir_raw_handler_lock);
315 316 317 318 319 320

	kfifo_free(&ir->raw->kfifo);
	kfree(ir->raw);
	ir->raw = NULL;
}

321 322 323 324 325 326
/*
 * Extension interface - used to register the IR decoders
 */

int ir_raw_handler_register(struct ir_raw_handler *ir_raw_handler)
{
327 328
	struct ir_raw_event_ctrl *raw;

329
	mutex_lock(&ir_raw_handler_lock);
330
	list_add_tail(&ir_raw_handler->list, &ir_raw_handler_list);
331 332 333
	if (ir_raw_handler->raw_register)
		list_for_each_entry(raw, &ir_raw_client_list, list)
			ir_raw_handler->raw_register(raw->input_dev);
334
	available_protocols |= ir_raw_handler->protocols;
335
	mutex_unlock(&ir_raw_handler_lock);
336

337 338 339 340 341 342
	return 0;
}
EXPORT_SYMBOL(ir_raw_handler_register);

void ir_raw_handler_unregister(struct ir_raw_handler *ir_raw_handler)
{
343 344
	struct ir_raw_event_ctrl *raw;

345
	mutex_lock(&ir_raw_handler_lock);
346
	list_del(&ir_raw_handler->list);
347 348 349
	if (ir_raw_handler->raw_unregister)
		list_for_each_entry(raw, &ir_raw_client_list, list)
			ir_raw_handler->raw_unregister(raw->input_dev);
350
	available_protocols &= ~ir_raw_handler->protocols;
351
	mutex_unlock(&ir_raw_handler_lock);
352 353 354
}
EXPORT_SYMBOL(ir_raw_handler_unregister);

355
#ifdef MODULE
356 357 358 359 360
static void init_decoders(struct work_struct *work)
{
	/* Load the decoder modules */

	load_nec_decode();
361
	load_rc5_decode();
362
	load_rc6_decode();
363
	load_jvc_decode();
364
	load_sony_decode();
365
	load_lirc_codec();
366 367 368 369 370

	/* If needed, we may later add some init code. In this case,
	   it is needed to change the CONFIG_MODULE test at ir-core.h
	 */
}
371
#endif
372 373 374

void ir_raw_init(void)
{
375
#ifdef MODULE
376 377
	INIT_WORK(&wq_load, init_decoders);
	schedule_work(&wq_load);
378 379
#endif
}