5 * Copyright (C) 2005 Mike Isely <isely@pobox.com>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 #include <linux/string.h>
23 #include <linux/slab.h>
24 #include "pvrusb2-debugifc.h"
25 #include "pvrusb2-hdw.h"
26 #include "pvrusb2-debug.h"
27 #include "pvrusb2-i2c-core.h"
29 struct debugifc_mask_item
{
34 static struct debugifc_mask_item mask_items
[] = {
35 {"ENC_FIRMWARE",(1<<PVR2_SUBSYS_B_ENC_FIRMWARE
)},
36 {"ENC_CFG",(1<<PVR2_SUBSYS_B_ENC_CFG
)},
37 {"DIG_RUN",(1<<PVR2_SUBSYS_B_DIGITIZER_RUN
)},
38 {"USB_RUN",(1<<PVR2_SUBSYS_B_USBSTREAM_RUN
)},
39 {"ENC_RUN",(1<<PVR2_SUBSYS_B_ENC_RUN
)},
43 static unsigned int debugifc_count_whitespace(const char *buf
,
49 for (scnt
= 0; scnt
< count
; scnt
++) {
51 if (ch
== ' ') continue;
52 if (ch
== '\t') continue;
53 if (ch
== '\n') continue;
60 static unsigned int debugifc_count_nonwhitespace(const char *buf
,
66 for (scnt
= 0; scnt
< count
; scnt
++) {
69 if (ch
== '\t') break;
70 if (ch
== '\n') break;
76 static unsigned int debugifc_isolate_word(const char *buf
,unsigned int count
,
78 unsigned int *wlenPtr
)
81 unsigned int consume_cnt
= 0;
87 scnt
= debugifc_count_whitespace(buf
,count
);
88 consume_cnt
+= scnt
; count
-= scnt
; buf
+= scnt
;
89 if (!count
) goto done
;
91 scnt
= debugifc_count_nonwhitespace(buf
,count
);
95 consume_cnt
+= scnt
; count
-= scnt
; buf
+= scnt
;
104 static int debugifc_parse_unsigned_number(const char *buf
,unsigned int count
,
111 if ((count
>= 2) && (buf
[0] == '0') &&
112 ((buf
[1] == 'x') || (buf
[1] == 'X'))) {
116 } else if ((count
>= 1) && (buf
[0] == '0')) {
122 if ((ch
>= '0') && (ch
<= '9')) {
124 } else if ((ch
>= 'a') && (ch
<= 'f')) {
126 } else if ((ch
>= 'A') && (ch
<= 'F')) {
131 if (val
>= radix
) return -EINVAL
;
140 static int debugifc_match_keyword(const char *buf
,unsigned int count
,
144 if (!keyword
) return 0;
145 kl
= strlen(keyword
);
146 if (kl
!= count
) return 0;
147 return !memcmp(buf
,keyword
,kl
);
151 static unsigned long debugifc_find_mask(const char *buf
,unsigned int count
)
153 struct debugifc_mask_item
*mip
;
155 for (idx
= 0; idx
< ARRAY_SIZE(mask_items
); idx
++) {
156 mip
= mask_items
+ idx
;
157 if (debugifc_match_keyword(buf
,count
,mip
->name
)) {
165 static int debugifc_print_mask(char *buf
,unsigned int sz
,
166 unsigned long msk
,unsigned long val
)
168 struct debugifc_mask_item
*mip
;
172 for (idx
= 0; idx
< ARRAY_SIZE(mask_items
); idx
++) {
173 mip
= mask_items
+ idx
;
174 if (!(mip
->msk
& msk
)) continue;
175 ccnt
= scnprintf(buf
,sz
,"%s%c%s",
177 ((mip
->msk
& val
) ? '+' : '-'),
186 static unsigned int debugifc_parse_subsys_mask(const char *buf
,
188 unsigned long *mskPtr
,
189 unsigned long *valPtr
)
192 unsigned int consume_cnt
= 0;
196 unsigned long m1
,msk
,val
;
202 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
204 consume_cnt
+= scnt
; count
-= scnt
; buf
+= scnt
;
208 if (wlen
) switch (wptr
[0]) {
220 m1
= debugifc_find_mask(wptr
,wlen
);
223 if (!mode
) val
|= m1
;
231 int pvr2_debugifc_print_info(struct pvr2_hdw
*hdw
,char *buf
,unsigned int acnt
)
235 struct pvr2_hdw_debug_info dbg
;
237 pvr2_hdw_get_debug_info(hdw
,&dbg
);
239 ccnt
= scnprintf(buf
,acnt
,"big lock %s; ctl lock %s",
240 (dbg
.big_lock_held
? "held" : "free"),
241 (dbg
.ctl_lock_held
? "held" : "free"));
242 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
243 if (dbg
.ctl_lock_held
) {
244 ccnt
= scnprintf(buf
,acnt
,"; cmd_state=%d cmd_code=%d"
245 " cmd_wlen=%d cmd_rlen=%d"
246 " wpend=%d rpend=%d tmout=%d rstatus=%d"
248 dbg
.cmd_debug_state
,dbg
.cmd_code
,
249 dbg
.cmd_debug_write_len
,
250 dbg
.cmd_debug_read_len
,
251 dbg
.cmd_debug_write_pend
,
252 dbg
.cmd_debug_read_pend
,
253 dbg
.cmd_debug_timeout
,
254 dbg
.cmd_debug_rstatus
,
255 dbg
.cmd_debug_wstatus
);
256 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
258 ccnt
= scnprintf(buf
,acnt
,"\n");
259 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
261 buf
,acnt
,"driver flags: %s %s %s\n",
262 (dbg
.flag_init_ok
? "initialized" : "uninitialized"),
263 (dbg
.flag_ok
? "ok" : "fail"),
264 (dbg
.flag_disconnected
? "disconnected" : "connected"));
265 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
266 ccnt
= scnprintf(buf
,acnt
,"Subsystems enabled / configured: ");
267 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
268 ccnt
= debugifc_print_mask(buf
,acnt
,dbg
.subsys_flags
,dbg
.subsys_flags
);
269 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
270 ccnt
= scnprintf(buf
,acnt
,"\n");
271 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
272 ccnt
= scnprintf(buf
,acnt
,"Subsystems disabled / unconfigured: ");
273 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
274 ccnt
= debugifc_print_mask(buf
,acnt
,~dbg
.subsys_flags
,dbg
.subsys_flags
);
275 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
276 ccnt
= scnprintf(buf
,acnt
,"\n");
277 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
279 ccnt
= scnprintf(buf
,acnt
,"Attached I2C modules:\n");
280 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
281 ccnt
= pvr2_i2c_report(hdw
,buf
,acnt
);
282 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
288 int pvr2_debugifc_print_status(struct pvr2_hdw
*hdw
,
289 char *buf
,unsigned int acnt
)
295 u32 gpio_dir
,gpio_in
,gpio_out
;
297 ret
= pvr2_hdw_is_hsm(hdw
);
298 ccnt
= scnprintf(buf
,acnt
,"USB link speed: %s\n",
299 (ret
< 0 ? "FAIL" : (ret
? "high" : "full")));
300 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
302 gpio_dir
= 0; gpio_in
= 0; gpio_out
= 0;
303 pvr2_hdw_gpio_get_dir(hdw
,&gpio_dir
);
304 pvr2_hdw_gpio_get_out(hdw
,&gpio_out
);
305 pvr2_hdw_gpio_get_in(hdw
,&gpio_in
);
306 ccnt
= scnprintf(buf
,acnt
,"GPIO state: dir=0x%x in=0x%x out=0x%x\n",
307 gpio_dir
,gpio_in
,gpio_out
);
308 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
310 ccnt
= scnprintf(buf
,acnt
,"Streaming is %s\n",
311 pvr2_hdw_get_streaming(hdw
) ? "on" : "off");
312 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
314 msk
= pvr2_hdw_subsys_get(hdw
);
315 ccnt
= scnprintf(buf
,acnt
,"Subsystems enabled / configured: ");
316 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
317 ccnt
= debugifc_print_mask(buf
,acnt
,msk
,msk
);
318 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
319 ccnt
= scnprintf(buf
,acnt
,"\n");
320 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
321 ccnt
= scnprintf(buf
,acnt
,"Subsystems disabled / unconfigured: ");
322 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
323 ccnt
= debugifc_print_mask(buf
,acnt
,~msk
,msk
);
324 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
325 ccnt
= scnprintf(buf
,acnt
,"\n");
326 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
328 msk
= pvr2_hdw_subsys_stream_get(hdw
);
329 ccnt
= scnprintf(buf
,acnt
,"Subsystems stopped on stream shutdown: ");
330 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
331 ccnt
= debugifc_print_mask(buf
,acnt
,msk
,msk
);
332 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
333 ccnt
= scnprintf(buf
,acnt
,"\n");
334 bcnt
+= ccnt
; acnt
-= ccnt
; buf
+= ccnt
;
340 static int pvr2_debugifc_do1cmd(struct pvr2_hdw
*hdw
,const char *buf
,
347 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
349 count
-= scnt
; buf
+= scnt
;
352 pvr2_trace(PVR2_TRACE_DEBUGIFC
,"debugifc cmd: \"%.*s\"",wlen
,wptr
);
353 if (debugifc_match_keyword(wptr
,wlen
,"reset")) {
354 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
355 if (!scnt
) return -EINVAL
;
356 count
-= scnt
; buf
+= scnt
;
357 if (!wptr
) return -EINVAL
;
358 if (debugifc_match_keyword(wptr
,wlen
,"cpu")) {
359 pvr2_hdw_cpureset_assert(hdw
,!0);
360 pvr2_hdw_cpureset_assert(hdw
,0);
362 } else if (debugifc_match_keyword(wptr
,wlen
,"bus")) {
363 pvr2_hdw_device_reset(hdw
);
364 } else if (debugifc_match_keyword(wptr
,wlen
,"soft")) {
365 return pvr2_hdw_cmd_powerup(hdw
);
366 } else if (debugifc_match_keyword(wptr
,wlen
,"deep")) {
367 return pvr2_hdw_cmd_deep_reset(hdw
);
368 } else if (debugifc_match_keyword(wptr
,wlen
,"firmware")) {
369 return pvr2_upload_firmware2(hdw
);
370 } else if (debugifc_match_keyword(wptr
,wlen
,"decoder")) {
371 return pvr2_hdw_cmd_decoder_reset(hdw
);
374 } else if (debugifc_match_keyword(wptr
,wlen
,"subsys_flags")) {
375 unsigned long msk
= 0;
376 unsigned long val
= 0;
377 if (debugifc_parse_subsys_mask(buf
,count
,&msk
,&val
) != count
) {
378 pvr2_trace(PVR2_TRACE_DEBUGIFC
,
379 "debugifc parse error on subsys mask");
382 pvr2_hdw_subsys_bit_chg(hdw
,msk
,val
);
384 } else if (debugifc_match_keyword(wptr
,wlen
,"stream_flags")) {
385 unsigned long msk
= 0;
386 unsigned long val
= 0;
387 if (debugifc_parse_subsys_mask(buf
,count
,&msk
,&val
) != count
) {
388 pvr2_trace(PVR2_TRACE_DEBUGIFC
,
389 "debugifc parse error on stream mask");
392 pvr2_hdw_subsys_stream_bit_chg(hdw
,msk
,val
);
394 } else if (debugifc_match_keyword(wptr
,wlen
,"cpufw")) {
395 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
396 if (!scnt
) return -EINVAL
;
397 count
-= scnt
; buf
+= scnt
;
398 if (!wptr
) return -EINVAL
;
399 if (debugifc_match_keyword(wptr
,wlen
,"fetch")) {
400 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
402 count
-= scnt
; buf
+= scnt
;
403 if (debugifc_match_keyword(wptr
,wlen
,"prom")) {
404 pvr2_hdw_cpufw_set_enabled(hdw
,!0,!0);
405 } else if (debugifc_match_keyword(wptr
,wlen
,
407 pvr2_hdw_cpufw_set_enabled(hdw
,0,!0);
412 pvr2_hdw_cpufw_set_enabled(hdw
,0,!0);
414 } else if (debugifc_match_keyword(wptr
,wlen
,"done")) {
415 pvr2_hdw_cpufw_set_enabled(hdw
,0,0);
420 } else if (debugifc_match_keyword(wptr
,wlen
,"gpio")) {
424 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
425 if (!scnt
) return -EINVAL
;
426 count
-= scnt
; buf
+= scnt
;
427 if (!wptr
) return -EINVAL
;
428 if (debugifc_match_keyword(wptr
,wlen
,"dir")) {
430 } else if (!debugifc_match_keyword(wptr
,wlen
,"out")) {
433 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
434 if (!scnt
) return -EINVAL
;
435 count
-= scnt
; buf
+= scnt
;
436 if (!wptr
) return -EINVAL
;
437 ret
= debugifc_parse_unsigned_number(wptr
,wlen
,&msk
);
439 scnt
= debugifc_isolate_word(buf
,count
,&wptr
,&wlen
);
441 ret
= debugifc_parse_unsigned_number(wptr
,wlen
,&val
);
448 ret
= pvr2_hdw_gpio_chg_dir(hdw
,msk
,val
);
450 ret
= pvr2_hdw_gpio_chg_out(hdw
,msk
,val
);
454 pvr2_trace(PVR2_TRACE_DEBUGIFC
,
455 "debugifc failed to recognize cmd: \"%.*s\"",wlen
,wptr
);
460 int pvr2_debugifc_docmd(struct pvr2_hdw
*hdw
,const char *buf
,
463 unsigned int bcnt
= 0;
467 for (bcnt
= 0; bcnt
< count
; bcnt
++) {
468 if (buf
[bcnt
] == '\n') break;
471 ret
= pvr2_debugifc_do1cmd(hdw
,buf
,bcnt
);
472 if (ret
< 0) return ret
;
473 if (bcnt
< count
) bcnt
++;
483 Stuff for Emacs to see, in order to encourage consistent editing style:
484 *** Local Variables: ***
486 *** fill-column: 75 ***
488 *** c-basic-offset: 8 ***