blob: e6cc45ebb33da0a22f4b333f88a72a53b6696862 [file] [log] [blame]
Alexander Shishkine443b332012-05-11 17:25:46 +03001#include <linux/kernel.h>
Alexander Shishkin3c37bb62013-03-30 02:46:17 +02002#include <linux/device.h>
3#include <linux/types.h>
4#include <linux/spinlock.h>
Alexander Shishkine443b332012-05-11 17:25:46 +03005#include <linux/usb/ch9.h>
6#include <linux/usb/gadget.h>
Alexander Shishkine443b332012-05-11 17:25:46 +03007
8#include "ci.h"
9#include "udc.h"
10#include "bits.h"
11#include "debug.h"
12
13/* Interrupt statistics */
14#define ISR_MASK 0x1F
15static struct isr_statistics {
16 u32 test;
17 u32 ui;
18 u32 uei;
19 u32 pci;
20 u32 uri;
21 u32 sli;
22 u32 none;
23 struct {
24 u32 cnt;
25 u32 buf[ISR_MASK+1];
26 u32 idx;
27 } hndl;
28} isr_statistics;
29
30void dbg_interrupt(u32 intmask)
31{
32 if (!intmask) {
33 isr_statistics.none++;
34 return;
35 }
36
37 isr_statistics.hndl.buf[isr_statistics.hndl.idx++] = intmask;
38 isr_statistics.hndl.idx &= ISR_MASK;
39 isr_statistics.hndl.cnt++;
40
41 if (USBi_URI & intmask)
42 isr_statistics.uri++;
43 if (USBi_PCI & intmask)
44 isr_statistics.pci++;
45 if (USBi_UEI & intmask)
46 isr_statistics.uei++;
47 if (USBi_UI & intmask)
48 isr_statistics.ui++;
49 if (USBi_SLI & intmask)
50 isr_statistics.sli++;
51}
52
53/**
54 * hw_register_read: reads all device registers (execute without interruption)
55 * @buf: destination buffer
56 * @size: buffer size
57 *
58 * This function returns number of registers read
59 */
Richard Zhao26c696c2012-07-07 22:56:40 +080060static size_t hw_register_read(struct ci13xxx *ci, u32 *buf, size_t size)
Alexander Shishkine443b332012-05-11 17:25:46 +030061{
62 unsigned i;
63
Richard Zhao26c696c2012-07-07 22:56:40 +080064 if (size > ci->hw_bank.size)
65 size = ci->hw_bank.size;
Alexander Shishkine443b332012-05-11 17:25:46 +030066
67 for (i = 0; i < size; i++)
Richard Zhao26c696c2012-07-07 22:56:40 +080068 buf[i] = hw_read(ci, i * sizeof(u32), ~0);
Alexander Shishkine443b332012-05-11 17:25:46 +030069
70 return size;
71}
72
73/**
74 * hw_register_write: writes to register
75 * @addr: register address
76 * @data: register value
77 *
78 * This function returns an error code
79 */
Richard Zhao26c696c2012-07-07 22:56:40 +080080static int hw_register_write(struct ci13xxx *ci, u16 addr, u32 data)
Alexander Shishkine443b332012-05-11 17:25:46 +030081{
82 /* align */
83 addr /= sizeof(u32);
84
Richard Zhao26c696c2012-07-07 22:56:40 +080085 if (addr >= ci->hw_bank.size)
Alexander Shishkine443b332012-05-11 17:25:46 +030086 return -EINVAL;
87
88 /* align */
89 addr *= sizeof(u32);
90
Richard Zhao26c696c2012-07-07 22:56:40 +080091 hw_write(ci, addr, ~0, data);
Alexander Shishkine443b332012-05-11 17:25:46 +030092 return 0;
93}
94
95/**
96 * hw_intr_clear: disables interrupt & clears interrupt status (execute without
97 * interruption)
98 * @n: interrupt bit
99 *
100 * This function returns an error code
101 */
Richard Zhao26c696c2012-07-07 22:56:40 +0800102static int hw_intr_clear(struct ci13xxx *ci, int n)
Alexander Shishkine443b332012-05-11 17:25:46 +0300103{
104 if (n >= REG_BITS)
105 return -EINVAL;
106
Richard Zhao26c696c2012-07-07 22:56:40 +0800107 hw_write(ci, OP_USBINTR, BIT(n), 0);
108 hw_write(ci, OP_USBSTS, BIT(n), BIT(n));
Alexander Shishkine443b332012-05-11 17:25:46 +0300109 return 0;
110}
111
112/**
113 * hw_intr_force: enables interrupt & forces interrupt status (execute without
114 * interruption)
115 * @n: interrupt bit
116 *
117 * This function returns an error code
118 */
Richard Zhao26c696c2012-07-07 22:56:40 +0800119static int hw_intr_force(struct ci13xxx *ci, int n)
Alexander Shishkine443b332012-05-11 17:25:46 +0300120{
121 if (n >= REG_BITS)
122 return -EINVAL;
123
Richard Zhao26c696c2012-07-07 22:56:40 +0800124 hw_write(ci, CAP_TESTMODE, TESTMODE_FORCE, TESTMODE_FORCE);
125 hw_write(ci, OP_USBINTR, BIT(n), BIT(n));
126 hw_write(ci, OP_USBSTS, BIT(n), BIT(n));
127 hw_write(ci, CAP_TESTMODE, TESTMODE_FORCE, 0);
Alexander Shishkine443b332012-05-11 17:25:46 +0300128 return 0;
129}
130
131/**
132 * show_device: prints information about device capabilities and status
133 *
134 * Check "device.h" for details
135 */
136static ssize_t show_device(struct device *dev, struct device_attribute *attr,
137 char *buf)
138{
Richard Zhao26c696c2012-07-07 22:56:40 +0800139 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
140 struct usb_gadget *gadget = &ci->gadget;
Alexander Shishkine443b332012-05-11 17:25:46 +0300141 int n = 0;
142
143 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800144 dev_err(ci->dev, "[%s] EINVAL\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300145 return 0;
146 }
147
148 n += scnprintf(buf + n, PAGE_SIZE - n, "speed = %d\n",
149 gadget->speed);
150 n += scnprintf(buf + n, PAGE_SIZE - n, "max_speed = %d\n",
151 gadget->max_speed);
Alexander Shishkine443b332012-05-11 17:25:46 +0300152 n += scnprintf(buf + n, PAGE_SIZE - n, "is_otg = %d\n",
153 gadget->is_otg);
154 n += scnprintf(buf + n, PAGE_SIZE - n, "is_a_peripheral = %d\n",
155 gadget->is_a_peripheral);
156 n += scnprintf(buf + n, PAGE_SIZE - n, "b_hnp_enable = %d\n",
157 gadget->b_hnp_enable);
158 n += scnprintf(buf + n, PAGE_SIZE - n, "a_hnp_support = %d\n",
159 gadget->a_hnp_support);
160 n += scnprintf(buf + n, PAGE_SIZE - n, "a_alt_hnp_support = %d\n",
161 gadget->a_alt_hnp_support);
162 n += scnprintf(buf + n, PAGE_SIZE - n, "name = %s\n",
163 (gadget->name ? gadget->name : ""));
164
165 return n;
166}
167static DEVICE_ATTR(device, S_IRUSR, show_device, NULL);
168
169/**
170 * show_driver: prints information about attached gadget (if any)
171 *
172 * Check "device.h" for details
173 */
174static ssize_t show_driver(struct device *dev, struct device_attribute *attr,
175 char *buf)
176{
Richard Zhao26c696c2012-07-07 22:56:40 +0800177 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
178 struct usb_gadget_driver *driver = ci->driver;
Alexander Shishkine443b332012-05-11 17:25:46 +0300179 int n = 0;
180
181 if (attr == NULL || buf == NULL) {
182 dev_err(dev, "[%s] EINVAL\n", __func__);
183 return 0;
184 }
185
186 if (driver == NULL)
187 return scnprintf(buf, PAGE_SIZE,
188 "There is no gadget attached!\n");
189
190 n += scnprintf(buf + n, PAGE_SIZE - n, "function = %s\n",
191 (driver->function ? driver->function : ""));
192 n += scnprintf(buf + n, PAGE_SIZE - n, "max speed = %d\n",
193 driver->max_speed);
194
195 return n;
196}
197static DEVICE_ATTR(driver, S_IRUSR, show_driver, NULL);
198
199/* Maximum event message length */
200#define DBG_DATA_MSG 64UL
201
202/* Maximum event messages */
203#define DBG_DATA_MAX 128UL
204
205/* Event buffer descriptor */
206static struct {
207 char (buf[DBG_DATA_MAX])[DBG_DATA_MSG]; /* buffer */
208 unsigned idx; /* index */
209 unsigned tty; /* print to console? */
210 rwlock_t lck; /* lock */
211} dbg_data = {
212 .idx = 0,
213 .tty = 0,
Thomas Gleixnereece09e2011-07-17 21:25:03 +0200214 .lck = __RW_LOCK_UNLOCKED(dbg_data.lck)
Alexander Shishkine443b332012-05-11 17:25:46 +0300215};
216
217/**
218 * dbg_dec: decrements debug event index
219 * @idx: buffer index
220 */
221static void dbg_dec(unsigned *idx)
222{
223 *idx = (*idx - 1) & (DBG_DATA_MAX-1);
224}
225
226/**
227 * dbg_inc: increments debug event index
228 * @idx: buffer index
229 */
230static void dbg_inc(unsigned *idx)
231{
232 *idx = (*idx + 1) & (DBG_DATA_MAX-1);
233}
234
235/**
236 * dbg_print: prints the common part of the event
237 * @addr: endpoint address
238 * @name: event name
239 * @status: status
240 * @extra: extra information
241 */
242static void dbg_print(u8 addr, const char *name, int status, const char *extra)
243{
244 struct timeval tval;
245 unsigned int stamp;
246 unsigned long flags;
247
248 write_lock_irqsave(&dbg_data.lck, flags);
249
250 do_gettimeofday(&tval);
251 stamp = tval.tv_sec & 0xFFFF; /* 2^32 = 4294967296. Limit to 4096s */
252 stamp = stamp * 1000000 + tval.tv_usec;
253
254 scnprintf(dbg_data.buf[dbg_data.idx], DBG_DATA_MSG,
255 "%04X\t? %02X %-7.7s %4i ?\t%s\n",
256 stamp, addr, name, status, extra);
257
258 dbg_inc(&dbg_data.idx);
259
260 write_unlock_irqrestore(&dbg_data.lck, flags);
261
262 if (dbg_data.tty != 0)
263 pr_notice("%04X\t? %02X %-7.7s %4i ?\t%s\n",
264 stamp, addr, name, status, extra);
265}
266
267/**
268 * dbg_done: prints a DONE event
269 * @addr: endpoint address
270 * @td: transfer descriptor
271 * @status: status
272 */
273void dbg_done(u8 addr, const u32 token, int status)
274{
275 char msg[DBG_DATA_MSG];
276
277 scnprintf(msg, sizeof(msg), "%d %02X",
278 (int)(token & TD_TOTAL_BYTES) >> ffs_nr(TD_TOTAL_BYTES),
279 (int)(token & TD_STATUS) >> ffs_nr(TD_STATUS));
280 dbg_print(addr, "DONE", status, msg);
281}
282
283/**
284 * dbg_event: prints a generic event
285 * @addr: endpoint address
286 * @name: event name
287 * @status: status
288 */
289void dbg_event(u8 addr, const char *name, int status)
290{
291 if (name != NULL)
292 dbg_print(addr, name, status, "");
293}
294
295/*
296 * dbg_queue: prints a QUEUE event
297 * @addr: endpoint address
298 * @req: USB request
299 * @status: status
300 */
301void dbg_queue(u8 addr, const struct usb_request *req, int status)
302{
303 char msg[DBG_DATA_MSG];
304
305 if (req != NULL) {
306 scnprintf(msg, sizeof(msg),
307 "%d %d", !req->no_interrupt, req->length);
308 dbg_print(addr, "QUEUE", status, msg);
309 }
310}
311
312/**
313 * dbg_setup: prints a SETUP event
314 * @addr: endpoint address
315 * @req: setup request
316 */
317void dbg_setup(u8 addr, const struct usb_ctrlrequest *req)
318{
319 char msg[DBG_DATA_MSG];
320
321 if (req != NULL) {
322 scnprintf(msg, sizeof(msg),
323 "%02X %02X %04X %04X %d", req->bRequestType,
324 req->bRequest, le16_to_cpu(req->wValue),
325 le16_to_cpu(req->wIndex), le16_to_cpu(req->wLength));
326 dbg_print(addr, "SETUP", 0, msg);
327 }
328}
329
330/**
331 * show_events: displays the event buffer
332 *
333 * Check "device.h" for details
334 */
335static ssize_t show_events(struct device *dev, struct device_attribute *attr,
336 char *buf)
337{
338 unsigned long flags;
339 unsigned i, j, n = 0;
340
341 if (attr == NULL || buf == NULL) {
342 dev_err(dev->parent, "[%s] EINVAL\n", __func__);
343 return 0;
344 }
345
346 read_lock_irqsave(&dbg_data.lck, flags);
347
348 i = dbg_data.idx;
349 for (dbg_dec(&i); i != dbg_data.idx; dbg_dec(&i)) {
350 n += strlen(dbg_data.buf[i]);
351 if (n >= PAGE_SIZE) {
352 n -= strlen(dbg_data.buf[i]);
353 break;
354 }
355 }
356 for (j = 0, dbg_inc(&i); j < n; dbg_inc(&i))
357 j += scnprintf(buf + j, PAGE_SIZE - j,
358 "%s", dbg_data.buf[i]);
359
360 read_unlock_irqrestore(&dbg_data.lck, flags);
361
362 return n;
363}
364
365/**
366 * store_events: configure if events are going to be also printed to console
367 *
368 * Check "device.h" for details
369 */
370static ssize_t store_events(struct device *dev, struct device_attribute *attr,
371 const char *buf, size_t count)
372{
373 unsigned tty;
374
375 if (attr == NULL || buf == NULL) {
376 dev_err(dev, "[%s] EINVAL\n", __func__);
377 goto done;
378 }
379
380 if (sscanf(buf, "%u", &tty) != 1 || tty > 1) {
381 dev_err(dev, "<1|0>: enable|disable console log\n");
382 goto done;
383 }
384
385 dbg_data.tty = tty;
386 dev_info(dev, "tty = %u", dbg_data.tty);
387
388 done:
389 return count;
390}
391static DEVICE_ATTR(events, S_IRUSR | S_IWUSR, show_events, store_events);
392
393/**
394 * show_inters: interrupt status, enable status and historic
395 *
396 * Check "device.h" for details
397 */
398static ssize_t show_inters(struct device *dev, struct device_attribute *attr,
399 char *buf)
400{
Richard Zhao26c696c2012-07-07 22:56:40 +0800401 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300402 unsigned long flags;
403 u32 intr;
404 unsigned i, j, n = 0;
405
406 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800407 dev_err(ci->dev, "[%s] EINVAL\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300408 return 0;
409 }
410
Richard Zhao26c696c2012-07-07 22:56:40 +0800411 spin_lock_irqsave(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300412
413 /*n += scnprintf(buf + n, PAGE_SIZE - n,
Richard Zhao26c696c2012-07-07 22:56:40 +0800414 "status = %08x\n", hw_read_intr_status(ci));
Alexander Shishkine443b332012-05-11 17:25:46 +0300415 n += scnprintf(buf + n, PAGE_SIZE - n,
Richard Zhao26c696c2012-07-07 22:56:40 +0800416 "enable = %08x\n", hw_read_intr_enable(ci));*/
Alexander Shishkine443b332012-05-11 17:25:46 +0300417
418 n += scnprintf(buf + n, PAGE_SIZE - n, "*test = %d\n",
419 isr_statistics.test);
420 n += scnprintf(buf + n, PAGE_SIZE - n, "? ui = %d\n",
421 isr_statistics.ui);
422 n += scnprintf(buf + n, PAGE_SIZE - n, "? uei = %d\n",
423 isr_statistics.uei);
424 n += scnprintf(buf + n, PAGE_SIZE - n, "? pci = %d\n",
425 isr_statistics.pci);
426 n += scnprintf(buf + n, PAGE_SIZE - n, "? uri = %d\n",
427 isr_statistics.uri);
428 n += scnprintf(buf + n, PAGE_SIZE - n, "? sli = %d\n",
429 isr_statistics.sli);
430 n += scnprintf(buf + n, PAGE_SIZE - n, "*none = %d\n",
431 isr_statistics.none);
432 n += scnprintf(buf + n, PAGE_SIZE - n, "*hndl = %d\n",
433 isr_statistics.hndl.cnt);
434
435 for (i = isr_statistics.hndl.idx, j = 0; j <= ISR_MASK; j++, i++) {
436 i &= ISR_MASK;
437 intr = isr_statistics.hndl.buf[i];
438
439 if (USBi_UI & intr)
440 n += scnprintf(buf + n, PAGE_SIZE - n, "ui ");
441 intr &= ~USBi_UI;
442 if (USBi_UEI & intr)
443 n += scnprintf(buf + n, PAGE_SIZE - n, "uei ");
444 intr &= ~USBi_UEI;
445 if (USBi_PCI & intr)
446 n += scnprintf(buf + n, PAGE_SIZE - n, "pci ");
447 intr &= ~USBi_PCI;
448 if (USBi_URI & intr)
449 n += scnprintf(buf + n, PAGE_SIZE - n, "uri ");
450 intr &= ~USBi_URI;
451 if (USBi_SLI & intr)
452 n += scnprintf(buf + n, PAGE_SIZE - n, "sli ");
453 intr &= ~USBi_SLI;
454 if (intr)
455 n += scnprintf(buf + n, PAGE_SIZE - n, "??? ");
456 if (isr_statistics.hndl.buf[i])
457 n += scnprintf(buf + n, PAGE_SIZE - n, "\n");
458 }
459
Richard Zhao26c696c2012-07-07 22:56:40 +0800460 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300461
462 return n;
463}
464
465/**
466 * store_inters: enable & force or disable an individual interrutps
467 * (to be used for test purposes only)
468 *
469 * Check "device.h" for details
470 */
471static ssize_t store_inters(struct device *dev, struct device_attribute *attr,
472 const char *buf, size_t count)
473{
Richard Zhao26c696c2012-07-07 22:56:40 +0800474 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300475 unsigned long flags;
476 unsigned en, bit;
477
478 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800479 dev_err(ci->dev, "EINVAL\n");
Alexander Shishkine443b332012-05-11 17:25:46 +0300480 goto done;
481 }
482
483 if (sscanf(buf, "%u %u", &en, &bit) != 2 || en > 1) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800484 dev_err(ci->dev, "<1|0> <bit>: enable|disable interrupt\n");
Alexander Shishkine443b332012-05-11 17:25:46 +0300485 goto done;
486 }
487
Richard Zhao26c696c2012-07-07 22:56:40 +0800488 spin_lock_irqsave(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300489 if (en) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800490 if (hw_intr_force(ci, bit))
Alexander Shishkine443b332012-05-11 17:25:46 +0300491 dev_err(dev, "invalid bit number\n");
492 else
493 isr_statistics.test++;
494 } else {
Richard Zhao26c696c2012-07-07 22:56:40 +0800495 if (hw_intr_clear(ci, bit))
Alexander Shishkine443b332012-05-11 17:25:46 +0300496 dev_err(dev, "invalid bit number\n");
497 }
Richard Zhao26c696c2012-07-07 22:56:40 +0800498 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300499
500 done:
501 return count;
502}
503static DEVICE_ATTR(inters, S_IRUSR | S_IWUSR, show_inters, store_inters);
504
505/**
506 * show_port_test: reads port test mode
507 *
508 * Check "device.h" for details
509 */
510static ssize_t show_port_test(struct device *dev,
511 struct device_attribute *attr, char *buf)
512{
Richard Zhao26c696c2012-07-07 22:56:40 +0800513 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300514 unsigned long flags;
515 unsigned mode;
516
517 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800518 dev_err(ci->dev, "EINVAL\n");
Alexander Shishkine443b332012-05-11 17:25:46 +0300519 return 0;
520 }
521
Richard Zhao26c696c2012-07-07 22:56:40 +0800522 spin_lock_irqsave(&ci->lock, flags);
523 mode = hw_port_test_get(ci);
524 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300525
526 return scnprintf(buf, PAGE_SIZE, "mode = %u\n", mode);
527}
528
529/**
530 * store_port_test: writes port test mode
531 *
532 * Check "device.h" for details
533 */
534static ssize_t store_port_test(struct device *dev,
535 struct device_attribute *attr,
536 const char *buf, size_t count)
537{
Richard Zhao26c696c2012-07-07 22:56:40 +0800538 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300539 unsigned long flags;
540 unsigned mode;
541
542 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800543 dev_err(ci->dev, "[%s] EINVAL\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300544 goto done;
545 }
546
547 if (sscanf(buf, "%u", &mode) != 1) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800548 dev_err(ci->dev, "<mode>: set port test mode");
Alexander Shishkine443b332012-05-11 17:25:46 +0300549 goto done;
550 }
551
Richard Zhao26c696c2012-07-07 22:56:40 +0800552 spin_lock_irqsave(&ci->lock, flags);
553 if (hw_port_test_set(ci, mode))
554 dev_err(ci->dev, "invalid mode\n");
555 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300556
557 done:
558 return count;
559}
560static DEVICE_ATTR(port_test, S_IRUSR | S_IWUSR,
561 show_port_test, store_port_test);
562
563/**
564 * show_qheads: DMA contents of all queue heads
565 *
566 * Check "device.h" for details
567 */
568static ssize_t show_qheads(struct device *dev, struct device_attribute *attr,
569 char *buf)
570{
Richard Zhao26c696c2012-07-07 22:56:40 +0800571 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300572 unsigned long flags;
573 unsigned i, j, n = 0;
574
575 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800576 dev_err(ci->dev, "[%s] EINVAL\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300577 return 0;
578 }
579
Richard Zhao26c696c2012-07-07 22:56:40 +0800580 spin_lock_irqsave(&ci->lock, flags);
581 for (i = 0; i < ci->hw_ep_max/2; i++) {
582 struct ci13xxx_ep *mEpRx = &ci->ci13xxx_ep[i];
Alexander Shishkine443b332012-05-11 17:25:46 +0300583 struct ci13xxx_ep *mEpTx =
Richard Zhao26c696c2012-07-07 22:56:40 +0800584 &ci->ci13xxx_ep[i + ci->hw_ep_max/2];
Alexander Shishkine443b332012-05-11 17:25:46 +0300585 n += scnprintf(buf + n, PAGE_SIZE - n,
586 "EP=%02i: RX=%08X TX=%08X\n",
587 i, (u32)mEpRx->qh.dma, (u32)mEpTx->qh.dma);
588 for (j = 0; j < (sizeof(struct ci13xxx_qh)/sizeof(u32)); j++) {
589 n += scnprintf(buf + n, PAGE_SIZE - n,
590 " %04X: %08X %08X\n", j,
591 *((u32 *)mEpRx->qh.ptr + j),
592 *((u32 *)mEpTx->qh.ptr + j));
593 }
594 }
Richard Zhao26c696c2012-07-07 22:56:40 +0800595 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300596
597 return n;
598}
599static DEVICE_ATTR(qheads, S_IRUSR, show_qheads, NULL);
600
601/**
602 * show_registers: dumps all registers
603 *
604 * Check "device.h" for details
605 */
606#define DUMP_ENTRIES 512
607static ssize_t show_registers(struct device *dev,
608 struct device_attribute *attr, char *buf)
609{
Richard Zhao26c696c2012-07-07 22:56:40 +0800610 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300611 unsigned long flags;
612 u32 *dump;
613 unsigned i, k, n = 0;
614
615 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800616 dev_err(ci->dev, "[%s] EINVAL\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300617 return 0;
618 }
619
620 dump = kmalloc(sizeof(u32) * DUMP_ENTRIES, GFP_KERNEL);
621 if (!dump) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800622 dev_err(ci->dev, "%s: out of memory\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300623 return 0;
624 }
625
Richard Zhao26c696c2012-07-07 22:56:40 +0800626 spin_lock_irqsave(&ci->lock, flags);
627 k = hw_register_read(ci, dump, DUMP_ENTRIES);
628 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300629
630 for (i = 0; i < k; i++) {
631 n += scnprintf(buf + n, PAGE_SIZE - n,
632 "reg[0x%04X] = 0x%08X\n",
633 i * (unsigned)sizeof(u32), dump[i]);
634 }
635 kfree(dump);
636
637 return n;
638}
639
640/**
641 * store_registers: writes value to register address
642 *
643 * Check "device.h" for details
644 */
645static ssize_t store_registers(struct device *dev,
646 struct device_attribute *attr,
647 const char *buf, size_t count)
648{
Richard Zhao26c696c2012-07-07 22:56:40 +0800649 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300650 unsigned long addr, data, flags;
651
652 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800653 dev_err(ci->dev, "[%s] EINVAL\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300654 goto done;
655 }
656
657 if (sscanf(buf, "%li %li", &addr, &data) != 2) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800658 dev_err(ci->dev,
Alexander Shishkine443b332012-05-11 17:25:46 +0300659 "<addr> <data>: write data to register address\n");
660 goto done;
661 }
662
Richard Zhao26c696c2012-07-07 22:56:40 +0800663 spin_lock_irqsave(&ci->lock, flags);
664 if (hw_register_write(ci, addr, data))
665 dev_err(ci->dev, "invalid address range\n");
666 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300667
668 done:
669 return count;
670}
671static DEVICE_ATTR(registers, S_IRUSR | S_IWUSR,
672 show_registers, store_registers);
673
674/**
675 * show_requests: DMA contents of all requests currently queued (all endpts)
676 *
677 * Check "device.h" for details
678 */
679static ssize_t show_requests(struct device *dev, struct device_attribute *attr,
680 char *buf)
681{
Richard Zhao26c696c2012-07-07 22:56:40 +0800682 struct ci13xxx *ci = container_of(dev, struct ci13xxx, gadget.dev);
Alexander Shishkine443b332012-05-11 17:25:46 +0300683 unsigned long flags;
684 struct list_head *ptr = NULL;
685 struct ci13xxx_req *req = NULL;
686 unsigned i, j, n = 0, qSize = sizeof(struct ci13xxx_td)/sizeof(u32);
687
688 if (attr == NULL || buf == NULL) {
Richard Zhao26c696c2012-07-07 22:56:40 +0800689 dev_err(ci->dev, "[%s] EINVAL\n", __func__);
Alexander Shishkine443b332012-05-11 17:25:46 +0300690 return 0;
691 }
692
Richard Zhao26c696c2012-07-07 22:56:40 +0800693 spin_lock_irqsave(&ci->lock, flags);
694 for (i = 0; i < ci->hw_ep_max; i++)
695 list_for_each(ptr, &ci->ci13xxx_ep[i].qh.queue)
Alexander Shishkine443b332012-05-11 17:25:46 +0300696 {
697 req = list_entry(ptr, struct ci13xxx_req, queue);
698
699 n += scnprintf(buf + n, PAGE_SIZE - n,
700 "EP=%02i: TD=%08X %s\n",
Richard Zhao26c696c2012-07-07 22:56:40 +0800701 i % ci->hw_ep_max/2, (u32)req->dma,
702 ((i < ci->hw_ep_max/2) ? "RX" : "TX"));
Alexander Shishkine443b332012-05-11 17:25:46 +0300703
704 for (j = 0; j < qSize; j++)
705 n += scnprintf(buf + n, PAGE_SIZE - n,
706 " %04X: %08X\n", j,
707 *((u32 *)req->ptr + j));
708 }
Richard Zhao26c696c2012-07-07 22:56:40 +0800709 spin_unlock_irqrestore(&ci->lock, flags);
Alexander Shishkine443b332012-05-11 17:25:46 +0300710
711 return n;
712}
713static DEVICE_ATTR(requests, S_IRUSR, show_requests, NULL);
714
715/**
716 * dbg_create_files: initializes the attribute interface
717 * @dev: device
718 *
719 * This function returns an error code
720 */
721int dbg_create_files(struct device *dev)
722{
723 int retval = 0;
724
725 if (dev == NULL)
726 return -EINVAL;
727 retval = device_create_file(dev, &dev_attr_device);
728 if (retval)
729 goto done;
730 retval = device_create_file(dev, &dev_attr_driver);
731 if (retval)
732 goto rm_device;
733 retval = device_create_file(dev, &dev_attr_events);
734 if (retval)
735 goto rm_driver;
736 retval = device_create_file(dev, &dev_attr_inters);
737 if (retval)
738 goto rm_events;
739 retval = device_create_file(dev, &dev_attr_port_test);
740 if (retval)
741 goto rm_inters;
742 retval = device_create_file(dev, &dev_attr_qheads);
743 if (retval)
744 goto rm_port_test;
745 retval = device_create_file(dev, &dev_attr_registers);
746 if (retval)
747 goto rm_qheads;
748 retval = device_create_file(dev, &dev_attr_requests);
749 if (retval)
750 goto rm_registers;
751 return 0;
752
753 rm_registers:
754 device_remove_file(dev, &dev_attr_registers);
755 rm_qheads:
756 device_remove_file(dev, &dev_attr_qheads);
757 rm_port_test:
758 device_remove_file(dev, &dev_attr_port_test);
759 rm_inters:
760 device_remove_file(dev, &dev_attr_inters);
761 rm_events:
762 device_remove_file(dev, &dev_attr_events);
763 rm_driver:
764 device_remove_file(dev, &dev_attr_driver);
765 rm_device:
766 device_remove_file(dev, &dev_attr_device);
767 done:
768 return retval;
769}
770
771/**
772 * dbg_remove_files: destroys the attribute interface
773 * @dev: device
774 *
775 * This function returns an error code
776 */
777int dbg_remove_files(struct device *dev)
778{
779 if (dev == NULL)
780 return -EINVAL;
781 device_remove_file(dev, &dev_attr_requests);
782 device_remove_file(dev, &dev_attr_registers);
783 device_remove_file(dev, &dev_attr_qheads);
784 device_remove_file(dev, &dev_attr_port_test);
785 device_remove_file(dev, &dev_attr_inters);
786 device_remove_file(dev, &dev_attr_events);
787 device_remove_file(dev, &dev_attr_driver);
788 device_remove_file(dev, &dev_attr_device);
789 return 0;
790}