2 * Wireless Host Controller (WHC) debug.
4 * Copyright (C) 2008 Cambridge Silicon Radio Ltd.
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License version
8 * 2 as published by the Free Software Foundation.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 #include <linux/slab.h>
19 #include <linux/kernel.h>
20 #include <linux/debugfs.h>
21 #include <linux/seq_file.h>
22 #include <linux/export.h>
24 #include "../../wusbcore/wusbhc.h"
34 static void qset_print(struct seq_file
*s
, struct whc_qset
*qset
)
36 static const char *qh_type
[] = {
37 "ctrl", "isoc", "bulk", "intr", "rsvd", "rsvd", "rsvd", "lpintr", };
39 struct urb
*urb
= NULL
;
42 seq_printf(s
, "qset %08x", (u32
)qset
->qset_dma
);
43 if (&qset
->list_node
== qset
->whc
->async_list
.prev
) {
44 seq_printf(s
, " (dummy)\n");
46 seq_printf(s
, " ep%d%s-%s maxpkt: %d\n",
47 qset
->qh
.info1
& 0x0f,
48 (qset
->qh
.info1
>> 4) & 0x1 ? "in" : "out",
49 qh_type
[(qset
->qh
.info1
>> 5) & 0x7],
50 (qset
->qh
.info1
>> 16) & 0xffff);
52 seq_printf(s
, " -> %08x\n", (u32
)qset
->qh
.link
);
53 seq_printf(s
, " info: %08x %08x %08x\n",
54 qset
->qh
.info1
, qset
->qh
.info2
, qset
->qh
.info3
);
55 seq_printf(s
, " sts: %04x errs: %d curwin: %08x\n",
56 qset
->qh
.status
, qset
->qh
.err_count
, qset
->qh
.cur_window
);
57 seq_printf(s
, " TD: sts: %08x opts: %08x\n",
58 qset
->qh
.overlay
.qtd
.status
, qset
->qh
.overlay
.qtd
.options
);
60 for (i
= 0; i
< WHCI_QSET_TD_MAX
; i
++) {
61 seq_printf(s
, " %c%c TD[%d]: sts: %08x opts: %08x ptr: %08x\n",
62 i
== qset
->td_start
? 'S' : ' ',
63 i
== qset
->td_end
? 'E' : ' ',
64 i
, qset
->qtd
[i
].status
, qset
->qtd
[i
].options
,
65 (u32
)qset
->qtd
[i
].page_list_ptr
);
67 seq_printf(s
, " ntds: %d\n", qset
->ntds
);
68 list_for_each_entry(std
, &qset
->stds
, list_node
) {
69 if (urb
!= std
->urb
) {
71 seq_printf(s
, " urb %p transferred: %d bytes\n", urb
,
75 seq_printf(s
, " sTD[%td]: %zu bytes @ %08x\n",
76 std
->qtd
- &qset
->qtd
[0],
77 std
->len
, std
->num_pointers
?
78 (u32
)(std
->pl_virt
[0].buf_ptr
) : (u32
)std
->dma_addr
);
80 seq_printf(s
, " sTD[-]: %zd bytes @ %08x\n",
81 std
->len
, std
->num_pointers
?
82 (u32
)(std
->pl_virt
[0].buf_ptr
) : (u32
)std
->dma_addr
);
86 static int di_print(struct seq_file
*s
, void *p
)
88 struct whc
*whc
= s
->private;
92 for (d
= 0; d
< whc
->n_devices
; d
++) {
93 struct di_buf_entry
*di
= &whc
->di_buf
[d
];
95 bitmap_scnprintf(buf
, sizeof(buf
),
96 (unsigned long *)di
->availability_info
, UWB_NUM_MAS
);
98 seq_printf(s
, "DI[%d]\n", d
);
99 seq_printf(s
, " availability: %s\n", buf
);
100 seq_printf(s
, " %c%c key idx: %d dev addr: %d\n",
101 (di
->addr_sec_info
& WHC_DI_SECURE
) ? 'S' : ' ',
102 (di
->addr_sec_info
& WHC_DI_DISABLE
) ? 'D' : ' ',
103 (di
->addr_sec_info
& WHC_DI_KEY_IDX_MASK
) >> 8,
104 (di
->addr_sec_info
& WHC_DI_DEV_ADDR_MASK
));
109 static int asl_print(struct seq_file
*s
, void *p
)
111 struct whc
*whc
= s
->private;
112 struct whc_qset
*qset
;
114 list_for_each_entry(qset
, &whc
->async_list
, list_node
) {
121 static int pzl_print(struct seq_file
*s
, void *p
)
123 struct whc
*whc
= s
->private;
124 struct whc_qset
*qset
;
127 for (period
= 0; period
< 5; period
++) {
128 seq_printf(s
, "Period %d\n", period
);
129 list_for_each_entry(qset
, &whc
->periodic_list
[period
], list_node
) {
136 static int di_open(struct inode
*inode
, struct file
*file
)
138 return single_open(file
, di_print
, inode
->i_private
);
141 static int asl_open(struct inode
*inode
, struct file
*file
)
143 return single_open(file
, asl_print
, inode
->i_private
);
146 static int pzl_open(struct inode
*inode
, struct file
*file
)
148 return single_open(file
, pzl_print
, inode
->i_private
);
151 static const struct file_operations di_fops
= {
155 .release
= single_release
,
156 .owner
= THIS_MODULE
,
159 static const struct file_operations asl_fops
= {
163 .release
= single_release
,
164 .owner
= THIS_MODULE
,
167 static const struct file_operations pzl_fops
= {
171 .release
= single_release
,
172 .owner
= THIS_MODULE
,
175 void whc_dbg_init(struct whc
*whc
)
177 if (whc
->wusbhc
.pal
.debugfs_dir
== NULL
)
180 whc
->dbg
= kzalloc(sizeof(struct whc_dbg
), GFP_KERNEL
);
181 if (whc
->dbg
== NULL
)
184 whc
->dbg
->di_f
= debugfs_create_file("di", 0444,
185 whc
->wusbhc
.pal
.debugfs_dir
, whc
,
187 whc
->dbg
->asl_f
= debugfs_create_file("asl", 0444,
188 whc
->wusbhc
.pal
.debugfs_dir
, whc
,
190 whc
->dbg
->pzl_f
= debugfs_create_file("pzl", 0444,
191 whc
->wusbhc
.pal
.debugfs_dir
, whc
,
195 void whc_dbg_clean_up(struct whc
*whc
)
198 debugfs_remove(whc
->dbg
->pzl_f
);
199 debugfs_remove(whc
->dbg
->asl_f
);
200 debugfs_remove(whc
->dbg
->di_f
);