pcmcia: CompactFlash driver for PA Semi Electra boards
[pv_ops_mirror.git] / drivers / media / video / pvrusb2 / pvrusb2-encoder.c
blob20b614436d2cffc16b239025ec79736dda9ddab1
1 /*
3 * $Id$
5 * Copyright (C) 2005 Mike Isely <isely@pobox.com>
6 * Copyright (C) 2004 Aurelien Alleaume <slts@free.fr>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 #include <linux/device.h> // for linux/firmware.h
24 #include <linux/firmware.h>
25 #include "pvrusb2-util.h"
26 #include "pvrusb2-encoder.h"
27 #include "pvrusb2-hdw-internal.h"
28 #include "pvrusb2-debug.h"
29 #include "pvrusb2-fx2-cmd.h"
33 /* Firmware mailbox flags - definitions found from ivtv */
34 #define IVTV_MBOX_FIRMWARE_DONE 0x00000004
35 #define IVTV_MBOX_DRIVER_DONE 0x00000002
36 #define IVTV_MBOX_DRIVER_BUSY 0x00000001
38 #define MBOX_BASE 0x44
41 static int pvr2_encoder_write_words(struct pvr2_hdw *hdw,
42 unsigned int offs,
43 const u32 *data, unsigned int dlen)
45 unsigned int idx,addr;
46 unsigned int bAddr;
47 int ret;
48 unsigned int chunkCnt;
52 Format: First byte must be 0x01. Remaining 32 bit words are
53 spread out into chunks of 7 bytes each, with the first 4 bytes
54 being the data word (little endian), and the next 3 bytes
55 being the address where that data word is to be written (big
56 endian). Repeat request for additional words, with offset
57 adjusted accordingly.
60 while (dlen) {
61 chunkCnt = 8;
62 if (chunkCnt > dlen) chunkCnt = dlen;
63 memset(hdw->cmd_buffer,0,sizeof(hdw->cmd_buffer));
64 bAddr = 0;
65 hdw->cmd_buffer[bAddr++] = FX2CMD_MEM_WRITE_DWORD;
66 for (idx = 0; idx < chunkCnt; idx++) {
67 addr = idx + offs;
68 hdw->cmd_buffer[bAddr+6] = (addr & 0xffu);
69 hdw->cmd_buffer[bAddr+5] = ((addr>>8) & 0xffu);
70 hdw->cmd_buffer[bAddr+4] = ((addr>>16) & 0xffu);
71 PVR2_DECOMPOSE_LE(hdw->cmd_buffer, bAddr,data[idx]);
72 bAddr += 7;
74 ret = pvr2_send_request(hdw,
75 hdw->cmd_buffer,1+(chunkCnt*7),
76 NULL,0);
77 if (ret) return ret;
78 data += chunkCnt;
79 dlen -= chunkCnt;
80 offs += chunkCnt;
83 return 0;
87 static int pvr2_encoder_read_words(struct pvr2_hdw *hdw,
88 unsigned int offs,
89 u32 *data, unsigned int dlen)
91 unsigned int idx;
92 int ret;
93 unsigned int chunkCnt;
97 Format: First byte must be 0x02 (status check) or 0x28 (read
98 back block of 32 bit words). Next 6 bytes must be zero,
99 followed by a single byte of MBOX_BASE+offset for portion to
100 be read. Returned data is packed set of 32 bits words that
101 were read.
105 while (dlen) {
106 chunkCnt = 16;
107 if (chunkCnt > dlen) chunkCnt = dlen;
108 if (chunkCnt < 16) chunkCnt = 1;
109 hdw->cmd_buffer[0] =
110 ((chunkCnt == 1) ?
111 FX2CMD_MEM_READ_DWORD : FX2CMD_MEM_READ_64BYTES);
112 hdw->cmd_buffer[1] = 0;
113 hdw->cmd_buffer[2] = 0;
114 hdw->cmd_buffer[3] = 0;
115 hdw->cmd_buffer[4] = 0;
116 hdw->cmd_buffer[5] = ((offs>>16) & 0xffu);
117 hdw->cmd_buffer[6] = ((offs>>8) & 0xffu);
118 hdw->cmd_buffer[7] = (offs & 0xffu);
119 ret = pvr2_send_request(hdw,
120 hdw->cmd_buffer,8,
121 hdw->cmd_buffer,
122 (chunkCnt == 1 ? 4 : 16 * 4));
123 if (ret) return ret;
125 for (idx = 0; idx < chunkCnt; idx++) {
126 data[idx] = PVR2_COMPOSE_LE(hdw->cmd_buffer,idx*4);
128 data += chunkCnt;
129 dlen -= chunkCnt;
130 offs += chunkCnt;
133 return 0;
137 /* This prototype is set up to be compatible with the
138 cx2341x_mbox_func prototype in cx2341x.h, which should be in
139 kernels 2.6.18 or later. We do this so that we can enable
140 cx2341x.ko to write to our encoder (by handing it a pointer to this
141 function). For earlier kernels this doesn't really matter. */
142 static int pvr2_encoder_cmd(void *ctxt,
143 int cmd,
144 int arg_cnt_send,
145 int arg_cnt_recv,
146 u32 *argp)
148 unsigned int poll_count;
149 unsigned int try_count = 0;
150 int retry_flag;
151 int ret = 0;
152 unsigned int idx;
153 /* These sizes look to be limited by the FX2 firmware implementation */
154 u32 wrData[16];
155 u32 rdData[16];
156 struct pvr2_hdw *hdw = (struct pvr2_hdw *)ctxt;
161 The encoder seems to speak entirely using blocks 32 bit words.
162 In ivtv driver terms, this is a mailbox at MBOX_BASE which we
163 populate with data and watch what the hardware does with it.
164 The first word is a set of flags used to control the
165 transaction, the second word is the command to execute, the
166 third byte is zero (ivtv driver suggests that this is some
167 kind of return value), and the fourth byte is a specified
168 timeout (windows driver always uses 0x00060000 except for one
169 case when it is zero). All successive words are the argument
170 words for the command.
172 First, write out the entire set of words, with the first word
173 being zero.
175 Next, write out just the first word again, but set it to
176 IVTV_MBOX_DRIVER_DONE | IVTV_DRIVER_BUSY this time (which
177 probably means "go").
179 Next, read back the return count words. Check the first word,
180 which should have IVTV_MBOX_FIRMWARE_DONE set. If however
181 that bit is not set, then the command isn't done so repeat the
182 read until it is set.
184 Finally, write out just the first word again, but set it to
185 0x0 this time (which probably means "idle").
189 if (arg_cnt_send > (ARRAY_SIZE(wrData) - 4)) {
190 pvr2_trace(
191 PVR2_TRACE_ERROR_LEGS,
192 "Failed to write cx23416 command"
193 " - too many input arguments"
194 " (was given %u limit %lu)",
195 arg_cnt_send, (long unsigned) ARRAY_SIZE(wrData) - 4);
196 return -EINVAL;
199 if (arg_cnt_recv > (ARRAY_SIZE(rdData) - 4)) {
200 pvr2_trace(
201 PVR2_TRACE_ERROR_LEGS,
202 "Failed to write cx23416 command"
203 " - too many return arguments"
204 " (was given %u limit %lu)",
205 arg_cnt_recv, (long unsigned) ARRAY_SIZE(rdData) - 4);
206 return -EINVAL;
210 LOCK_TAKE(hdw->ctl_lock); do {
212 retry_flag = 0;
213 try_count++;
214 ret = 0;
215 wrData[0] = 0;
216 wrData[1] = cmd;
217 wrData[2] = 0;
218 wrData[3] = 0x00060000;
219 for (idx = 0; idx < arg_cnt_send; idx++) {
220 wrData[idx+4] = argp[idx];
222 for (; idx < ARRAY_SIZE(wrData) - 4; idx++) {
223 wrData[idx+4] = 0;
226 ret = pvr2_encoder_write_words(hdw,MBOX_BASE,wrData,idx);
227 if (ret) break;
228 wrData[0] = IVTV_MBOX_DRIVER_DONE|IVTV_MBOX_DRIVER_BUSY;
229 ret = pvr2_encoder_write_words(hdw,MBOX_BASE,wrData,1);
230 if (ret) break;
231 poll_count = 0;
232 while (1) {
233 poll_count++;
234 ret = pvr2_encoder_read_words(hdw,MBOX_BASE,rdData,
235 arg_cnt_recv+4);
236 if (ret) {
237 break;
239 if (rdData[0] & IVTV_MBOX_FIRMWARE_DONE) {
240 break;
242 if (rdData[0] && (poll_count < 1000)) continue;
243 if (!rdData[0]) {
244 retry_flag = !0;
245 pvr2_trace(
246 PVR2_TRACE_ERROR_LEGS,
247 "Encoder timed out waiting for us"
248 "; arranging to retry");
249 } else {
250 pvr2_trace(
251 PVR2_TRACE_ERROR_LEGS,
252 "***WARNING*** device's encoder"
253 " appears to be stuck"
254 " (status=0x%08x)",rdData[0]);
256 pvr2_trace(
257 PVR2_TRACE_ERROR_LEGS,
258 "Encoder command: 0x%02x",cmd);
259 for (idx = 4; idx < arg_cnt_send; idx++) {
260 pvr2_trace(
261 PVR2_TRACE_ERROR_LEGS,
262 "Encoder arg%d: 0x%08x",
263 idx-3,wrData[idx]);
265 ret = -EBUSY;
266 break;
268 if (retry_flag) {
269 if (try_count < 20) continue;
270 pvr2_trace(
271 PVR2_TRACE_ERROR_LEGS,
272 "Too many retries...");
273 ret = -EBUSY;
275 if (ret) {
276 pvr2_trace(
277 PVR2_TRACE_ERROR_LEGS,
278 "Giving up on command."
279 " It is likely that"
280 " this is a bad idea...");
281 break;
283 wrData[0] = 0x7;
284 for (idx = 0; idx < arg_cnt_recv; idx++) {
285 argp[idx] = rdData[idx+4];
288 wrData[0] = 0x0;
289 ret = pvr2_encoder_write_words(hdw,MBOX_BASE,wrData,1);
290 if (ret) break;
292 } while(0); LOCK_GIVE(hdw->ctl_lock);
294 return ret;
298 static int pvr2_encoder_vcmd(struct pvr2_hdw *hdw, int cmd,
299 int args, ...)
301 va_list vl;
302 unsigned int idx;
303 u32 data[12];
305 if (args > ARRAY_SIZE(data)) {
306 pvr2_trace(
307 PVR2_TRACE_ERROR_LEGS,
308 "Failed to write cx23416 command"
309 " - too many arguments"
310 " (was given %u limit %lu)",
311 args, (long unsigned) ARRAY_SIZE(data));
312 return -EINVAL;
315 va_start(vl, args);
316 for (idx = 0; idx < args; idx++) {
317 data[idx] = va_arg(vl, u32);
319 va_end(vl);
321 return pvr2_encoder_cmd(hdw,cmd,args,0,data);
325 /* This implements some extra setup for the encoder that seems to be
326 specific to the PVR USB2 hardware. */
327 static int pvr2_encoder_prep_config(struct pvr2_hdw *hdw)
329 int ret = 0;
330 int encMisc3Arg = 0;
332 #if 0
333 /* This inexplicable bit happens in the Hauppage windows
334 driver (for both 24xxx and 29xxx devices). However I
335 currently see no difference in behavior with or without
336 this stuff. Leave this here as a note of its existence,
337 but don't use it. */
338 LOCK_TAKE(hdw->ctl_lock); do {
339 u32 dat[1];
340 dat[0] = 0x80000640;
341 pvr2_encoder_write_words(hdw,0x01fe,dat,1);
342 pvr2_encoder_write_words(hdw,0x023e,dat,1);
343 } while(0); LOCK_GIVE(hdw->ctl_lock);
344 #endif
346 /* Mike Isely <isely@pobox.com> 26-Jan-2006 The windows driver
347 sends the following list of ENC_MISC commands (for both
348 24xxx and 29xxx devices). Meanings are not entirely clear,
349 however without the ENC_MISC(3,1) command then we risk
350 random perpetual video corruption whenever the video input
351 breaks up for a moment (like when switching channels). */
354 #if 0
355 /* This ENC_MISC(5,0) command seems to hurt 29xxx sync
356 performance on channel changes, but is not a problem on
357 24xxx devices. */
358 ret |= pvr2_encoder_vcmd(hdw, CX2341X_ENC_MISC,4, 5,0,0,0);
359 #endif
361 /* This ENC_MISC(3,encMisc3Arg) command is critical - without
362 it there will eventually be video corruption. Also, the
363 29xxx case is strange - the Windows driver is passing 1
364 regardless of device type but if we have 1 for 29xxx device
365 the video turns sluggish. */
366 switch (hdw->hdw_type) {
367 case PVR2_HDW_TYPE_24XXX: encMisc3Arg = 1; break;
368 case PVR2_HDW_TYPE_29XXX: encMisc3Arg = 0; break;
369 default: break;
371 ret |= pvr2_encoder_vcmd(hdw, CX2341X_ENC_MISC,4, 3,
372 encMisc3Arg,0,0);
374 ret |= pvr2_encoder_vcmd(hdw, CX2341X_ENC_MISC,4, 8,0,0,0);
376 #if 0
377 /* This ENC_MISC(4,1) command is poisonous, so it is commented
378 out. But I'm leaving it here anyway to document its
379 existence in the Windows driver. The effect of this
380 command is that apps displaying the stream become sluggish
381 with stuttering video. */
382 ret |= pvr2_encoder_vcmd(hdw, CX2341X_ENC_MISC,4, 4,1,0,0);
383 #endif
385 ret |= pvr2_encoder_vcmd(hdw, CX2341X_ENC_MISC,4, 0,3,0,0);
386 ret |= pvr2_encoder_vcmd(hdw, CX2341X_ENC_MISC,4,15,0,0,0);
388 return ret;
391 int pvr2_encoder_configure(struct pvr2_hdw *hdw)
393 int ret;
394 int val;
395 pvr2_trace(PVR2_TRACE_ENCODER,"pvr2_encoder_configure"
396 " (cx2341x module)");
397 hdw->enc_ctl_state.port = CX2341X_PORT_STREAMING;
398 hdw->enc_ctl_state.width = hdw->res_hor_val;
399 hdw->enc_ctl_state.height = hdw->res_ver_val;
400 hdw->enc_ctl_state.is_50hz = ((hdw->std_mask_cur & V4L2_STD_525_60) ?
401 0 : 1);
403 ret = 0;
405 ret |= pvr2_encoder_prep_config(hdw);
407 /* saa7115: 0xf0 */
408 val = 0xf0;
409 if (hdw->hdw_type == PVR2_HDW_TYPE_24XXX) {
410 /* ivtv cx25840: 0x140 */
411 val = 0x140;
414 if (!ret) ret = pvr2_encoder_vcmd(
415 hdw,CX2341X_ENC_SET_NUM_VSYNC_LINES, 2,
416 val, val);
418 /* setup firmware to notify us about some events (don't know why...) */
419 if (!ret) ret = pvr2_encoder_vcmd(
420 hdw,CX2341X_ENC_SET_EVENT_NOTIFICATION, 4,
421 0, 0, 0x10000000, 0xffffffff);
423 if (!ret) ret = pvr2_encoder_vcmd(
424 hdw,CX2341X_ENC_SET_VBI_LINE, 5,
425 0xffffffff,0,0,0,0);
427 if (ret) {
428 pvr2_trace(PVR2_TRACE_ERROR_LEGS,
429 "Failed to configure cx23416");
430 return ret;
433 ret = cx2341x_update(hdw,pvr2_encoder_cmd,
434 (hdw->enc_cur_valid ? &hdw->enc_cur_state : NULL),
435 &hdw->enc_ctl_state);
436 if (ret) {
437 pvr2_trace(PVR2_TRACE_ERROR_LEGS,
438 "Error from cx2341x module code=%d",ret);
439 return ret;
442 ret = 0;
444 if (!ret) ret = pvr2_encoder_vcmd(
445 hdw, CX2341X_ENC_INITIALIZE_INPUT, 0);
447 if (ret) {
448 pvr2_trace(PVR2_TRACE_ERROR_LEGS,
449 "Failed to initialize cx23416 video input");
450 return ret;
453 hdw->subsys_enabled_mask |= (1<<PVR2_SUBSYS_B_ENC_CFG);
454 memcpy(&hdw->enc_cur_state,&hdw->enc_ctl_state,
455 sizeof(struct cx2341x_mpeg_params));
456 hdw->enc_cur_valid = !0;
457 return 0;
461 int pvr2_encoder_start(struct pvr2_hdw *hdw)
463 int status;
465 /* unmask some interrupts */
466 pvr2_write_register(hdw, 0x0048, 0xbfffffff);
468 /* change some GPIO data */
469 pvr2_hdw_gpio_chg_dir(hdw,0xffffffff,0x00000481);
470 pvr2_hdw_gpio_chg_out(hdw,0xffffffff,0x00000000);
472 pvr2_encoder_vcmd(hdw,CX2341X_ENC_MUTE_VIDEO,1,
473 hdw->input_val == PVR2_CVAL_INPUT_RADIO ? 1 : 0);
475 switch (hdw->config) {
476 case pvr2_config_vbi:
477 status = pvr2_encoder_vcmd(hdw,CX2341X_ENC_START_CAPTURE,2,
478 0x01,0x14);
479 break;
480 case pvr2_config_mpeg:
481 status = pvr2_encoder_vcmd(hdw,CX2341X_ENC_START_CAPTURE,2,
482 0,0x13);
483 break;
484 default: /* Unhandled cases for now */
485 status = pvr2_encoder_vcmd(hdw,CX2341X_ENC_START_CAPTURE,2,
486 0,0x13);
487 break;
489 if (!status) {
490 hdw->subsys_enabled_mask |= (1<<PVR2_SUBSYS_B_ENC_RUN);
492 return status;
495 int pvr2_encoder_stop(struct pvr2_hdw *hdw)
497 int status;
499 /* mask all interrupts */
500 pvr2_write_register(hdw, 0x0048, 0xffffffff);
502 switch (hdw->config) {
503 case pvr2_config_vbi:
504 status = pvr2_encoder_vcmd(hdw,CX2341X_ENC_STOP_CAPTURE,3,
505 0x01,0x01,0x14);
506 break;
507 case pvr2_config_mpeg:
508 status = pvr2_encoder_vcmd(hdw,CX2341X_ENC_STOP_CAPTURE,3,
509 0x01,0,0x13);
510 break;
511 default: /* Unhandled cases for now */
512 status = pvr2_encoder_vcmd(hdw,CX2341X_ENC_STOP_CAPTURE,3,
513 0x01,0,0x13);
514 break;
517 /* change some GPIO data */
518 /* Note: Bit d7 of dir appears to control the LED. So we shut it
519 off here. */
520 pvr2_hdw_gpio_chg_dir(hdw,0xffffffff,0x00000401);
521 pvr2_hdw_gpio_chg_out(hdw,0xffffffff,0x00000000);
523 if (!status) {
524 hdw->subsys_enabled_mask &= ~(1<<PVR2_SUBSYS_B_ENC_RUN);
526 return status;
531 Stuff for Emacs to see, in order to encourage consistent editing style:
532 *** Local Variables: ***
533 *** mode: c ***
534 *** fill-column: 70 ***
535 *** tab-width: 8 ***
536 *** c-basic-offset: 8 ***
537 *** End: ***