rc-core-priv.h 7.0 KB
Newer Older
1 2 3
/*
 * Remote Controller core raw events header
 *
4
 * Copyright (C) 2010 by Mauro Carvalho Chehab
5 6 7 8 9 10 11 12 13 14 15
 *
 * 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.
 */

16 17
#ifndef _RC_CORE_PRIV
#define _RC_CORE_PRIV
18 19

#include <linux/slab.h>
20
#include <linux/spinlock.h>
21
#include <media/rc-core.h>
22 23 24 25

struct ir_raw_handler {
	struct list_head list;

26
	u64 protocols; /* which are handled by this handler */
27
	int (*decode)(struct rc_dev *dev, struct ir_raw_event event);
28 29
	int (*encode)(u64 protocols, const struct rc_scancode_filter *scancode,
		      struct ir_raw_event *events, unsigned int max);
30 31

	/* These two should only be used by the lirc decoder */
32 33
	int (*raw_register)(struct rc_dev *dev);
	int (*raw_unregister)(struct rc_dev *dev);
34 35 36
};

struct ir_raw_event_ctrl {
37
	struct list_head		list;		/* to keep track of raw clients */
38
	struct task_struct		*thread;
39
	spinlock_t			lock;
40
	struct kfifo_rec_ptr_1		kfifo;		/* fifo for the pulse/space durations */
41 42
	ktime_t				last_event;	/* when last event occurred */
	enum raw_event_type		last_type;	/* last event type */
43
	struct rc_dev			*dev;		/* pointer to the parent rc_dev */
44 45 46

	/* raw decoder state follows */
	struct ir_raw_event prev_ev;
47
	struct ir_raw_event this_ev;
48 49 50 51
	struct nec_dec {
		int state;
		unsigned count;
		u32 bits;
52 53
		bool is_nec_x;
		bool necx_repeat;
54 55 56 57 58
	} nec;
	struct rc5_dec {
		int state;
		u32 bits;
		unsigned count;
59
		bool is_rc5x;
60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
	} rc5;
	struct rc6_dec {
		int state;
		u8 header;
		u32 body;
		bool toggle;
		unsigned count;
		unsigned wanted_bits;
	} rc6;
	struct sony_dec {
		int state;
		u32 bits;
		unsigned count;
	} sony;
	struct jvc_dec {
		int state;
		u16 bits;
		u16 old_bits;
		unsigned count;
		bool first;
		bool toggle;
	} jvc;
82 83 84 85 86
	struct sanyo_dec {
		int state;
		unsigned count;
		u64 bits;
	} sanyo;
87 88 89 90 91 92
	struct sharp_dec {
		int state;
		unsigned count;
		u32 bits;
		unsigned int pulse_len;
	} sharp;
93 94 95 96 97 98 99 100 101 102 103
	struct mce_kbd_dec {
		struct input_dev *idev;
		struct timer_list rx_timeout;
		char name[64];
		char phys[64];
		int state;
		u8 header;
		u32 body;
		unsigned count;
		unsigned wanted_bits;
	} mce_kbd;
104
	struct lirc_codec {
105
		struct rc_dev *dev;
106
		struct lirc_driver *drv;
107
		int carrier_low;
108 109 110 111 112 113

		ktime_t gap_start;
		u64 gap_duration;
		bool gap;
		bool send_timeout_reports;

114
	} lirc;
115 116 117 118 119
	struct xmp_dec {
		int state;
		unsigned count;
		u32 durations[16];
	} xmp;
120 121 122
};

/* macros for IR decoders */
123 124
static inline bool geq_margin(unsigned d1, unsigned d2, unsigned margin)
{
125 126 127
	return d1 > (d2 - margin);
}

128 129
static inline bool eq_margin(unsigned d1, unsigned d2, unsigned margin)
{
130 131 132
	return ((d1 > (d2 - margin)) && (d1 < (d2 + margin)));
}

133 134
static inline bool is_transition(struct ir_raw_event *x, struct ir_raw_event *y)
{
135 136 137
	return x->pulse != y->pulse;
}

138 139
static inline void decrease_duration(struct ir_raw_event *ev, unsigned duration)
{
140 141 142 143 144 145
	if (duration > ev->duration)
		ev->duration = 0;
	else
		ev->duration -= duration;
}

146 147 148 149 150 151
/* Returns true if event is normal pulse/space event */
static inline bool is_timing_event(struct ir_raw_event ev)
{
	return !ev.carrier_report && !ev.reset;
}

M
Maxim Levitsky 已提交
152
#define TO_US(duration)			DIV_ROUND_CLOSEST((duration), 1000)
153
#define TO_STR(is_pulse)		((is_pulse) ? "pulse" : "space")
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
/* functions for IR encoders */

static inline void init_ir_raw_event_duration(struct ir_raw_event *ev,
					      unsigned int pulse,
					      u32 duration)
{
	init_ir_raw_event(ev);
	ev->duration = duration;
	ev->pulse = pulse;
}

/**
 * struct ir_raw_timings_manchester - Manchester coding timings
 * @leader:		duration of leader pulse (if any) 0 if continuing
 *			existing signal (see @pulse_space_start)
 * @pulse_space_start:	1 for starting with pulse (0 for starting with space)
 * @clock:		duration of each pulse/space in ns
 * @invert:		if set clock logic is inverted
 *			(0 = space + pulse, 1 = pulse + space)
 * @trailer_space:	duration of trailer space in ns
 */
struct ir_raw_timings_manchester {
	unsigned int leader;
	unsigned int pulse_space_start:1;
	unsigned int clock;
	unsigned int invert:1;
	unsigned int trailer_space;
};

int ir_raw_gen_manchester(struct ir_raw_event **ev, unsigned int max,
			  const struct ir_raw_timings_manchester *timings,
			  unsigned int n, unsigned int data);

188
/*
189
 * Routines from rc-raw.c to be used internally and by decoders
190
 */
191
u64 ir_raw_get_allowed_protocols(void);
192
u64 ir_raw_get_encode_protocols(void);
193 194
int ir_raw_event_register(struct rc_dev *dev);
void ir_raw_event_unregister(struct rc_dev *dev);
195 196 197 198 199 200 201 202 203 204 205 206 207
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);

/*
 * Decoder initialization code
 *
 * Those load logic are called during ir-core init, and automatically
 * loads the compiled decoders for their usage with IR raw events
 */

/* from ir-nec-decoder.c */
#ifdef CONFIG_IR_NEC_DECODER_MODULE
208
#define load_nec_decode()	request_module_nowait("ir-nec-decoder")
209
#else
210
static inline void load_nec_decode(void) { }
211 212 213 214
#endif

/* from ir-rc5-decoder.c */
#ifdef CONFIG_IR_RC5_DECODER_MODULE
215
#define load_rc5_decode()	request_module_nowait("ir-rc5-decoder")
216
#else
217
static inline void load_rc5_decode(void) { }
218 219
#endif

220
/* from ir-rc6-decoder.c */
221
#ifdef CONFIG_IR_RC6_DECODER_MODULE
222
#define load_rc6_decode()	request_module_nowait("ir-rc6-decoder")
223
#else
224
static inline void load_rc6_decode(void) { }
225 226
#endif

227 228
/* from ir-jvc-decoder.c */
#ifdef CONFIG_IR_JVC_DECODER_MODULE
229
#define load_jvc_decode()	request_module_nowait("ir-jvc-decoder")
230
#else
231
static inline void load_jvc_decode(void) { }
232 233
#endif

234 235
/* from ir-sony-decoder.c */
#ifdef CONFIG_IR_SONY_DECODER_MODULE
236
#define load_sony_decode()	request_module_nowait("ir-sony-decoder")
237
#else
238
static inline void load_sony_decode(void) { }
239 240
#endif

241 242
/* from ir-sanyo-decoder.c */
#ifdef CONFIG_IR_SANYO_DECODER_MODULE
243
#define load_sanyo_decode()	request_module_nowait("ir-sanyo-decoder")
244 245 246 247
#else
static inline void load_sanyo_decode(void) { }
#endif

248 249 250 251 252 253 254
/* from ir-sharp-decoder.c */
#ifdef CONFIG_IR_SHARP_DECODER_MODULE
#define load_sharp_decode()	request_module_nowait("ir-sharp-decoder")
#else
static inline void load_sharp_decode(void) { }
#endif

255 256
/* from ir-mce_kbd-decoder.c */
#ifdef CONFIG_IR_MCE_KBD_DECODER_MODULE
257
#define load_mce_kbd_decode()	request_module_nowait("ir-mce_kbd-decoder")
258
#else
259
static inline void load_mce_kbd_decode(void) { }
260 261
#endif

262 263
/* from ir-lirc-codec.c */
#ifdef CONFIG_IR_LIRC_CODEC_MODULE
264
#define load_lirc_codec()	request_module_nowait("ir-lirc-codec")
265
#else
266
static inline void load_lirc_codec(void) { }
267 268
#endif

269 270 271 272 273 274 275
/* from ir-xmp-decoder.c */
#ifdef CONFIG_IR_XMP_DECODER_MODULE
#define load_xmp_decode()      request_module_nowait("ir-xmp-decoder")
#else
static inline void load_xmp_decode(void) { }
#endif

276

277
#endif /* _RC_CORE_PRIV */