1 /* SPDX-License-Identifier: GPL-2.0 */
3 * Cadence CDNSP DRD Driver.
5 * Copyright (C) 2020 Cadence.
10 #ifndef __LINUX_CDNSP_DEBUG
11 #define __LINUX_CDNSP_DEBUG
13 static inline const char *cdnsp_trb_comp_code_string(u8 status)
20 case COMP_DATA_BUFFER_ERROR:
21 return "Data Buffer Error";
22 case COMP_BABBLE_DETECTED_ERROR:
23 return "Babble Detected";
26 case COMP_RESOURCE_ERROR:
27 return "Resource Error";
28 case COMP_NO_SLOTS_AVAILABLE_ERROR:
29 return "No Slots Available Error";
30 case COMP_INVALID_STREAM_TYPE_ERROR:
31 return "Invalid Stream Type Error";
32 case COMP_SLOT_NOT_ENABLED_ERROR:
33 return "Slot Not Enabled Error";
34 case COMP_ENDPOINT_NOT_ENABLED_ERROR:
35 return "Endpoint Not Enabled Error";
36 case COMP_SHORT_PACKET:
37 return "Short Packet";
38 case COMP_RING_UNDERRUN:
39 return "Ring Underrun";
40 case COMP_RING_OVERRUN:
41 return "Ring Overrun";
42 case COMP_VF_EVENT_RING_FULL_ERROR:
43 return "VF Event Ring Full Error";
44 case COMP_PARAMETER_ERROR:
45 return "Parameter Error";
46 case COMP_CONTEXT_STATE_ERROR:
47 return "Context State Error";
48 case COMP_EVENT_RING_FULL_ERROR:
49 return "Event Ring Full Error";
50 case COMP_INCOMPATIBLE_DEVICE_ERROR:
51 return "Incompatible Device Error";
52 case COMP_MISSED_SERVICE_ERROR:
53 return "Missed Service Error";
54 case COMP_COMMAND_RING_STOPPED:
55 return "Command Ring Stopped";
56 case COMP_COMMAND_ABORTED:
57 return "Command Aborted";
60 case COMP_STOPPED_LENGTH_INVALID:
61 return "Stopped - Length Invalid";
62 case COMP_STOPPED_SHORT_PACKET:
63 return "Stopped - Short Packet";
64 case COMP_MAX_EXIT_LATENCY_TOO_LARGE_ERROR:
65 return "Max Exit Latency Too Large Error";
66 case COMP_ISOCH_BUFFER_OVERRUN:
67 return "Isoch Buffer Overrun";
68 case COMP_EVENT_LOST_ERROR:
69 return "Event Lost Error";
70 case COMP_UNDEFINED_ERROR:
71 return "Undefined Error";
72 case COMP_INVALID_STREAM_ID_ERROR:
73 return "Invalid Stream ID Error";
79 static inline const char *cdnsp_trb_type_string(u8 type)
89 return "Status Stage";
99 return "Enable Slot Command";
100 case TRB_DISABLE_SLOT:
101 return "Disable Slot Command";
103 return "Address Device Command";
105 return "Configure Endpoint Command";
106 case TRB_EVAL_CONTEXT:
107 return "Evaluate Context Command";
109 return "Reset Endpoint Command";
111 return "Stop Ring Command";
113 return "Set TR Dequeue Pointer Command";
115 return "Reset Device Command";
116 case TRB_FORCE_HEADER:
117 return "Force Header Command";
119 return "No-Op Command";
121 return "Transfer Event";
123 return "Command Completion Event";
124 case TRB_PORT_STATUS:
125 return "Port Status Change Event";
127 return "Device Controller Event";
128 case TRB_MFINDEX_WRAP:
129 return "MFINDEX Wrap Event";
130 case TRB_ENDPOINT_NRDY:
131 return "Endpoint Not ready";
132 case TRB_HALT_ENDPOINT:
133 return "Halt Endpoint";
134 case TRB_FLUSH_ENDPOINT:
135 return "FLush Endpoint";
141 static inline const char *cdnsp_ring_type_string(enum cdnsp_ring_type type)
163 static inline char *cdnsp_slot_state_string(u32 state)
166 case SLOT_STATE_ENABLED:
167 return "enabled/disabled";
168 case SLOT_STATE_DEFAULT:
170 case SLOT_STATE_ADDRESSED:
172 case SLOT_STATE_CONFIGURED:
179 static inline const char *cdnsp_decode_trb(char *str, size_t size, u32 field0,
180 u32 field1, u32 field2, u32 field3)
182 int ep_id = TRB_TO_EP_INDEX(field3) - 1;
183 int type = TRB_FIELD_TO_TYPE(field3);
188 ep_num = DIV_ROUND_UP(ep_id, 2);
192 ret = snprintf(str, size,
193 "LINK %08x%08x intr %ld type '%s' flags %c:%c:%c:%c",
194 field1, field0, GET_INTR_TARGET(field2),
195 cdnsp_trb_type_string(type),
196 field3 & TRB_IOC ? 'I' : 'i',
197 field3 & TRB_CHAIN ? 'C' : 'c',
198 field3 & TRB_TC ? 'T' : 't',
199 field3 & TRB_CYCLE ? 'C' : 'c');
203 case TRB_PORT_STATUS:
205 ret = snprintf(str, size,
206 "ep%d%s(%d) type '%s' TRB %08x%08x status '%s'"
207 " len %ld slot %ld flags %c:%c",
208 ep_num, ep_id % 2 ? "out" : "in",
209 TRB_TO_EP_INDEX(field3),
210 cdnsp_trb_type_string(type), field1, field0,
211 cdnsp_trb_comp_code_string(GET_COMP_CODE(field2)),
212 EVENT_TRB_LEN(field2), TRB_TO_SLOT_ID(field3),
213 field3 & EVENT_DATA ? 'E' : 'e',
214 field3 & TRB_CYCLE ? 'C' : 'c');
216 case TRB_MFINDEX_WRAP:
217 ret = snprintf(str, size, "%s: flags %c",
218 cdnsp_trb_type_string(type),
219 field3 & TRB_CYCLE ? 'C' : 'c');
222 ret = snprintf(str, size,
223 "type '%s' bRequestType %02x bRequest %02x "
224 "wValue %02x%02x wIndex %02x%02x wLength %d "
225 "length %ld TD size %ld intr %ld Setup ID %ld "
227 cdnsp_trb_type_string(type),
229 (field0 & 0xff00) >> 8,
230 (field0 & 0xff000000) >> 24,
231 (field0 & 0xff0000) >> 16,
232 (field1 & 0xff00) >> 8,
234 (field1 & 0xff000000) >> 16 |
235 (field1 & 0xff0000) >> 16,
236 TRB_LEN(field2), GET_TD_SIZE(field2),
237 GET_INTR_TARGET(field2),
238 TRB_SETUPID_TO_TYPE(field3),
239 field3 & TRB_IDT ? 'D' : 'd',
240 field3 & TRB_IOC ? 'I' : 'i',
241 field3 & TRB_CYCLE ? 'C' : 'c');
244 ret = snprintf(str, size,
245 "type '%s' Buffer %08x%08x length %ld TD size %ld "
246 "intr %ld flags %c:%c:%c:%c:%c:%c:%c",
247 cdnsp_trb_type_string(type),
248 field1, field0, TRB_LEN(field2),
250 GET_INTR_TARGET(field2),
251 field3 & TRB_IDT ? 'D' : 'i',
252 field3 & TRB_IOC ? 'I' : 'i',
253 field3 & TRB_CHAIN ? 'C' : 'c',
254 field3 & TRB_NO_SNOOP ? 'S' : 's',
255 field3 & TRB_ISP ? 'I' : 'i',
256 field3 & TRB_ENT ? 'E' : 'e',
257 field3 & TRB_CYCLE ? 'C' : 'c');
260 ret = snprintf(str, size,
261 "Buffer %08x%08x length %ld TD size %ld intr"
262 "%ld type '%s' flags %c:%c:%c:%c",
263 field1, field0, TRB_LEN(field2),
265 GET_INTR_TARGET(field2),
266 cdnsp_trb_type_string(type),
267 field3 & TRB_IOC ? 'I' : 'i',
268 field3 & TRB_CHAIN ? 'C' : 'c',
269 field3 & TRB_ENT ? 'E' : 'e',
270 field3 & TRB_CYCLE ? 'C' : 'c');
276 ret = snprintf(str, size,
277 "type '%s' Buffer %08x%08x length %ld "
278 "TD size %ld intr %ld "
279 "flags %c:%c:%c:%c:%c:%c:%c:%c:%c",
280 cdnsp_trb_type_string(type),
281 field1, field0, TRB_LEN(field2),
283 GET_INTR_TARGET(field2),
284 field3 & TRB_BEI ? 'B' : 'b',
285 field3 & TRB_IDT ? 'T' : 't',
286 field3 & TRB_IOC ? 'I' : 'i',
287 field3 & TRB_CHAIN ? 'C' : 'c',
288 field3 & TRB_NO_SNOOP ? 'S' : 's',
289 field3 & TRB_ISP ? 'I' : 'i',
290 field3 & TRB_ENT ? 'E' : 'e',
291 field3 & TRB_CYCLE ? 'C' : 'c',
292 !(field3 & TRB_EVENT_INVALIDATE) ? 'V' : 'v');
295 case TRB_ENABLE_SLOT:
296 ret = snprintf(str, size, "%s: flags %c",
297 cdnsp_trb_type_string(type),
298 field3 & TRB_CYCLE ? 'C' : 'c');
300 case TRB_DISABLE_SLOT:
301 ret = snprintf(str, size, "%s: slot %ld flags %c",
302 cdnsp_trb_type_string(type),
303 TRB_TO_SLOT_ID(field3),
304 field3 & TRB_CYCLE ? 'C' : 'c');
307 ret = snprintf(str, size,
308 "%s: ctx %08x%08x slot %ld flags %c:%c",
309 cdnsp_trb_type_string(type), field1, field0,
310 TRB_TO_SLOT_ID(field3),
311 field3 & TRB_BSR ? 'B' : 'b',
312 field3 & TRB_CYCLE ? 'C' : 'c');
315 ret = snprintf(str, size,
316 "%s: ctx %08x%08x slot %ld flags %c:%c",
317 cdnsp_trb_type_string(type), field1, field0,
318 TRB_TO_SLOT_ID(field3),
319 field3 & TRB_DC ? 'D' : 'd',
320 field3 & TRB_CYCLE ? 'C' : 'c');
322 case TRB_EVAL_CONTEXT:
323 ret = snprintf(str, size,
324 "%s: ctx %08x%08x slot %ld flags %c",
325 cdnsp_trb_type_string(type), field1, field0,
326 TRB_TO_SLOT_ID(field3),
327 field3 & TRB_CYCLE ? 'C' : 'c');
330 case TRB_HALT_ENDPOINT:
331 case TRB_FLUSH_ENDPOINT:
332 ret = snprintf(str, size,
333 "%s: ep%d%s(%d) ctx %08x%08x slot %ld flags %c",
334 cdnsp_trb_type_string(type),
335 ep_num, ep_id % 2 ? "out" : "in",
336 TRB_TO_EP_INDEX(field3), field1, field0,
337 TRB_TO_SLOT_ID(field3),
338 field3 & TRB_CYCLE ? 'C' : 'c');
341 ret = snprintf(str, size,
342 "%s: ep%d%s(%d) slot %ld sp %d flags %c",
343 cdnsp_trb_type_string(type),
344 ep_num, ep_id % 2 ? "out" : "in",
345 TRB_TO_EP_INDEX(field3),
346 TRB_TO_SLOT_ID(field3),
347 TRB_TO_SUSPEND_PORT(field3),
348 field3 & TRB_CYCLE ? 'C' : 'c');
351 ret = snprintf(str, size,
352 "%s: ep%d%s(%d) deq %08x%08x stream %ld slot %ld flags %c",
353 cdnsp_trb_type_string(type),
354 ep_num, ep_id % 2 ? "out" : "in",
355 TRB_TO_EP_INDEX(field3), field1, field0,
356 TRB_TO_STREAM_ID(field2),
357 TRB_TO_SLOT_ID(field3),
358 field3 & TRB_CYCLE ? 'C' : 'c');
361 ret = snprintf(str, size, "%s: slot %ld flags %c",
362 cdnsp_trb_type_string(type),
363 TRB_TO_SLOT_ID(field3),
364 field3 & TRB_CYCLE ? 'C' : 'c');
366 case TRB_ENDPOINT_NRDY:
367 temp = TRB_TO_HOST_STREAM(field2);
369 ret = snprintf(str, size,
370 "%s: ep%d%s(%d) H_SID %x%s%s D_SID %lx flags %c:%c",
371 cdnsp_trb_type_string(type),
372 ep_num, ep_id % 2 ? "out" : "in",
373 TRB_TO_EP_INDEX(field3), temp,
374 temp == STREAM_PRIME_ACK ? "(PRIME)" : "",
375 temp == STREAM_REJECTED ? "(REJECTED)" : "",
376 TRB_TO_DEV_STREAM(field0),
377 field3 & TRB_STAT ? 'S' : 's',
378 field3 & TRB_CYCLE ? 'C' : 'c');
381 ret = snprintf(str, size,
382 "type '%s' -> raw %08x %08x %08x %08x",
383 cdnsp_trb_type_string(type),
384 field0, field1, field2, field3);
388 pr_info("CDNSP: buffer overflowed.\n");
393 static inline const char *cdnsp_decode_slot_context(u32 info, u32 info2,
394 u32 int_target, u32 state)
396 static char str[1024];
401 speed = info & DEV_SPEED;
414 s = "super-speed plus";
420 ret = sprintf(str, "%s Ctx Entries %d",
421 s, (info & LAST_CTX_MASK) >> 27);
423 ret += sprintf(str + ret, " [Intr %ld] Addr %ld State %s",
424 GET_INTR_TARGET(int_target), state & DEV_ADDR_MASK,
425 cdnsp_slot_state_string(GET_SLOT_STATE(state)));
430 static inline const char *cdnsp_portsc_link_state_string(u32 portsc)
432 switch (portsc & PORT_PLS_MASK) {
454 return "Compliance mode";
466 static inline const char *cdnsp_decode_portsc(char *str, size_t size,
471 ret = snprintf(str, size, "%s %s %s Link:%s PortSpeed:%d ",
472 portsc & PORT_POWER ? "Powered" : "Powered-off",
473 portsc & PORT_CONNECT ? "Connected" : "Not-connected",
474 portsc & PORT_PED ? "Enabled" : "Disabled",
475 cdnsp_portsc_link_state_string(portsc),
476 DEV_PORT_SPEED(portsc));
478 if (portsc & PORT_RESET)
479 ret += snprintf(str + ret, size - ret, "In-Reset ");
481 ret += snprintf(str + ret, size - ret, "Change: ");
482 if (portsc & PORT_CSC)
483 ret += snprintf(str + ret, size - ret, "CSC ");
484 if (portsc & PORT_WRC)
485 ret += snprintf(str + ret, size - ret, "WRC ");
486 if (portsc & PORT_RC)
487 ret += snprintf(str + ret, size - ret, "PRC ");
488 if (portsc & PORT_PLC)
489 ret += snprintf(str + ret, size - ret, "PLC ");
490 if (portsc & PORT_CEC)
491 ret += snprintf(str + ret, size - ret, "CEC ");
492 ret += snprintf(str + ret, size - ret, "Wake: ");
493 if (portsc & PORT_WKCONN_E)
494 ret += snprintf(str + ret, size - ret, "WCE ");
495 if (portsc & PORT_WKDISC_E)
496 ret += snprintf(str + ret, size - ret, "WDE ");
501 static inline const char *cdnsp_ep_state_string(u8 state)
504 case EP_STATE_DISABLED:
506 case EP_STATE_RUNNING:
508 case EP_STATE_HALTED:
510 case EP_STATE_STOPPED:
519 static inline const char *cdnsp_ep_type_string(u8 type)
541 static inline const char *cdnsp_decode_ep_context(char *str, size_t size,
543 u64 deq, u32 tx_info)
545 u8 max_pstr, ep_state, interval, ep_type, burst, cerr, mult;
551 esit = CTX_TO_MAX_ESIT_PAYLOAD_HI(info) << 16 |
552 CTX_TO_MAX_ESIT_PAYLOAD_LO(tx_info);
554 ep_state = info & EP_STATE_MASK;
555 max_pstr = CTX_TO_EP_MAXPSTREAMS(info);
556 interval = CTX_TO_EP_INTERVAL(info);
557 mult = CTX_TO_EP_MULT(info) + 1;
558 lsa = !!(info & EP_HAS_LSA);
560 cerr = (info2 & (3 << 1)) >> 1;
561 ep_type = CTX_TO_EP_TYPE(info2);
562 hid = !!(info2 & (1 << 7));
563 burst = CTX_TO_MAX_BURST(info2);
564 maxp = MAX_PACKET_DECODED(info2);
566 avg = EP_AVG_TRB_LENGTH(tx_info);
568 ret = snprintf(str, size, "State %s mult %d max P. Streams %d %s",
569 cdnsp_ep_state_string(ep_state), mult,
570 max_pstr, lsa ? "LSA " : "");
572 ret += snprintf(str + ret, size - ret,
573 "interval %d us max ESIT payload %d CErr %d ",
574 (1 << interval) * 125, esit, cerr);
576 ret += snprintf(str + ret, size - ret,
577 "Type %s %sburst %d maxp %d deq %016llx ",
578 cdnsp_ep_type_string(ep_type), hid ? "HID" : "",
581 ret += snprintf(str + ret, size - ret, "avg trb len %d", avg);
586 #endif /*__LINUX_CDNSP_DEBUG*/