diff options
Diffstat (limited to 'drivers/media/video/saa7134')
-rw-r--r-- | drivers/media/video/saa7134/saa7134-input.c | 78 |
1 files changed, 78 insertions, 0 deletions
diff --git a/drivers/media/video/saa7134/saa7134-input.c b/drivers/media/video/saa7134/saa7134-input.c index e6217fe13339..ad08d13dffdd 100644 --- a/drivers/media/video/saa7134/saa7134-input.c +++ b/drivers/media/video/saa7134/saa7134-input.c | |||
@@ -198,6 +198,84 @@ static int get_key_beholdm6xx(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) | |||
198 | return 1; | 198 | return 1; |
199 | } | 199 | } |
200 | 200 | ||
201 | /* Common (grey or coloured) pinnacle PCTV remote handling | ||
202 | * | ||
203 | */ | ||
204 | static int get_key_pinnacle(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw, | ||
205 | int parity_offset, int marker, int code_modulo) | ||
206 | { | ||
207 | unsigned char b[4]; | ||
208 | unsigned int start = 0,parity = 0,code = 0; | ||
209 | |||
210 | /* poll IR chip */ | ||
211 | if (4 != i2c_master_recv(&ir->c, b, 4)) { | ||
212 | i2cdprintk("read error\n"); | ||
213 | return -EIO; | ||
214 | } | ||
215 | |||
216 | for (start = 0; start < ARRAY_SIZE(b); start++) { | ||
217 | if (b[start] == marker) { | ||
218 | code=b[(start+parity_offset + 1) % 4]; | ||
219 | parity=b[(start+parity_offset) % 4]; | ||
220 | } | ||
221 | } | ||
222 | |||
223 | /* Empty Request */ | ||
224 | if (parity == 0) | ||
225 | return 0; | ||
226 | |||
227 | /* Repeating... */ | ||
228 | if (ir->old == parity) | ||
229 | return 0; | ||
230 | |||
231 | ir->old = parity; | ||
232 | |||
233 | /* drop special codes when a key is held down a long time for the grey controller | ||
234 | In this case, the second bit of the code is asserted */ | ||
235 | if (marker == 0xfe && (code & 0x40)) | ||
236 | return 0; | ||
237 | |||
238 | code %= code_modulo; | ||
239 | |||
240 | *ir_raw = code; | ||
241 | *ir_key = code; | ||
242 | |||
243 | i2cdprintk("Pinnacle PCTV key %02x\n", code); | ||
244 | |||
245 | return 1; | ||
246 | } | ||
247 | |||
248 | /* The grey pinnacle PCTV remote | ||
249 | * | ||
250 | * There are one issue with this remote: | ||
251 | * - I2c packet does not change when the same key is pressed quickly. The workaround | ||
252 | * is to hold down each key for about half a second, so that another code is generated | ||
253 | * in the i2c packet, and the function can distinguish key presses. | ||
254 | * | ||
255 | * Sylvain Pasche <sylvain.pasche@gmail.com> | ||
256 | */ | ||
257 | static int get_key_pinnacle_grey(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) | ||
258 | { | ||
259 | |||
260 | return get_key_pinnacle(ir, ir_key, ir_raw, 1, 0xfe, 0xff); | ||
261 | } | ||
262 | |||
263 | |||
264 | /* The new pinnacle PCTV remote (with the colored buttons) | ||
265 | * | ||
266 | * Ricardo Cerqueira <v4l@cerqueira.org> | ||
267 | */ | ||
268 | static int get_key_pinnacle_color(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) | ||
269 | { | ||
270 | /* code_modulo parameter (0x88) is used to reduce code value to fit inside IR_KEYTAB_SIZE | ||
271 | * | ||
272 | * this is the only value that results in 42 unique | ||
273 | * codes < 128 | ||
274 | */ | ||
275 | |||
276 | return get_key_pinnacle(ir, ir_key, ir_raw, 2, 0x80, 0x88); | ||
277 | } | ||
278 | |||
201 | void saa7134_input_irq(struct saa7134_dev *dev) | 279 | void saa7134_input_irq(struct saa7134_dev *dev) |
202 | { | 280 | { |
203 | struct card_ir *ir = dev->remote; | 281 | struct card_ir *ir = dev->remote; |