162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0 */
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Cadence CDNSP DRD Driver.
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2020 Cadence.
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Author: Pawel Laszczak <pawell@cadence.com>
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci */
1062306a36Sopenharmony_ci#ifndef __LINUX_CDNSP_DEBUG
1162306a36Sopenharmony_ci#define __LINUX_CDNSP_DEBUG
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_cistatic inline const char *cdnsp_trb_comp_code_string(u8 status)
1462306a36Sopenharmony_ci{
1562306a36Sopenharmony_ci	switch (status) {
1662306a36Sopenharmony_ci	case COMP_INVALID:
1762306a36Sopenharmony_ci		return "Invalid";
1862306a36Sopenharmony_ci	case COMP_SUCCESS:
1962306a36Sopenharmony_ci		return "Success";
2062306a36Sopenharmony_ci	case COMP_DATA_BUFFER_ERROR:
2162306a36Sopenharmony_ci		return "Data Buffer Error";
2262306a36Sopenharmony_ci	case COMP_BABBLE_DETECTED_ERROR:
2362306a36Sopenharmony_ci		return "Babble Detected";
2462306a36Sopenharmony_ci	case COMP_TRB_ERROR:
2562306a36Sopenharmony_ci		return "TRB Error";
2662306a36Sopenharmony_ci	case COMP_RESOURCE_ERROR:
2762306a36Sopenharmony_ci		return "Resource Error";
2862306a36Sopenharmony_ci	case COMP_NO_SLOTS_AVAILABLE_ERROR:
2962306a36Sopenharmony_ci		return "No Slots Available Error";
3062306a36Sopenharmony_ci	case COMP_INVALID_STREAM_TYPE_ERROR:
3162306a36Sopenharmony_ci		return "Invalid Stream Type Error";
3262306a36Sopenharmony_ci	case COMP_SLOT_NOT_ENABLED_ERROR:
3362306a36Sopenharmony_ci		return "Slot Not Enabled Error";
3462306a36Sopenharmony_ci	case COMP_ENDPOINT_NOT_ENABLED_ERROR:
3562306a36Sopenharmony_ci		return "Endpoint Not Enabled Error";
3662306a36Sopenharmony_ci	case COMP_SHORT_PACKET:
3762306a36Sopenharmony_ci		return "Short Packet";
3862306a36Sopenharmony_ci	case COMP_RING_UNDERRUN:
3962306a36Sopenharmony_ci		return "Ring Underrun";
4062306a36Sopenharmony_ci	case COMP_RING_OVERRUN:
4162306a36Sopenharmony_ci		return "Ring Overrun";
4262306a36Sopenharmony_ci	case COMP_VF_EVENT_RING_FULL_ERROR:
4362306a36Sopenharmony_ci		return "VF Event Ring Full Error";
4462306a36Sopenharmony_ci	case COMP_PARAMETER_ERROR:
4562306a36Sopenharmony_ci		return "Parameter Error";
4662306a36Sopenharmony_ci	case COMP_CONTEXT_STATE_ERROR:
4762306a36Sopenharmony_ci		return "Context State Error";
4862306a36Sopenharmony_ci	case COMP_EVENT_RING_FULL_ERROR:
4962306a36Sopenharmony_ci		return "Event Ring Full Error";
5062306a36Sopenharmony_ci	case COMP_INCOMPATIBLE_DEVICE_ERROR:
5162306a36Sopenharmony_ci		return "Incompatible Device Error";
5262306a36Sopenharmony_ci	case COMP_MISSED_SERVICE_ERROR:
5362306a36Sopenharmony_ci		return "Missed Service Error";
5462306a36Sopenharmony_ci	case COMP_COMMAND_RING_STOPPED:
5562306a36Sopenharmony_ci		return "Command Ring Stopped";
5662306a36Sopenharmony_ci	case COMP_COMMAND_ABORTED:
5762306a36Sopenharmony_ci		return "Command Aborted";
5862306a36Sopenharmony_ci	case COMP_STOPPED:
5962306a36Sopenharmony_ci		return "Stopped";
6062306a36Sopenharmony_ci	case COMP_STOPPED_LENGTH_INVALID:
6162306a36Sopenharmony_ci		return "Stopped - Length Invalid";
6262306a36Sopenharmony_ci	case COMP_STOPPED_SHORT_PACKET:
6362306a36Sopenharmony_ci		return "Stopped - Short Packet";
6462306a36Sopenharmony_ci	case COMP_MAX_EXIT_LATENCY_TOO_LARGE_ERROR:
6562306a36Sopenharmony_ci		return "Max Exit Latency Too Large Error";
6662306a36Sopenharmony_ci	case COMP_ISOCH_BUFFER_OVERRUN:
6762306a36Sopenharmony_ci		return "Isoch Buffer Overrun";
6862306a36Sopenharmony_ci	case COMP_EVENT_LOST_ERROR:
6962306a36Sopenharmony_ci		return "Event Lost Error";
7062306a36Sopenharmony_ci	case COMP_UNDEFINED_ERROR:
7162306a36Sopenharmony_ci		return "Undefined Error";
7262306a36Sopenharmony_ci	case COMP_INVALID_STREAM_ID_ERROR:
7362306a36Sopenharmony_ci		return "Invalid Stream ID Error";
7462306a36Sopenharmony_ci	default:
7562306a36Sopenharmony_ci		return "Unknown!!";
7662306a36Sopenharmony_ci	}
7762306a36Sopenharmony_ci}
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_cistatic inline const char *cdnsp_trb_type_string(u8 type)
8062306a36Sopenharmony_ci{
8162306a36Sopenharmony_ci	switch (type) {
8262306a36Sopenharmony_ci	case TRB_NORMAL:
8362306a36Sopenharmony_ci		return "Normal";
8462306a36Sopenharmony_ci	case TRB_SETUP:
8562306a36Sopenharmony_ci		return "Setup Stage";
8662306a36Sopenharmony_ci	case TRB_DATA:
8762306a36Sopenharmony_ci		return "Data Stage";
8862306a36Sopenharmony_ci	case TRB_STATUS:
8962306a36Sopenharmony_ci		return "Status Stage";
9062306a36Sopenharmony_ci	case TRB_ISOC:
9162306a36Sopenharmony_ci		return "Isoch";
9262306a36Sopenharmony_ci	case TRB_LINK:
9362306a36Sopenharmony_ci		return "Link";
9462306a36Sopenharmony_ci	case TRB_EVENT_DATA:
9562306a36Sopenharmony_ci		return "Event Data";
9662306a36Sopenharmony_ci	case TRB_TR_NOOP:
9762306a36Sopenharmony_ci		return "No-Op";
9862306a36Sopenharmony_ci	case TRB_ENABLE_SLOT:
9962306a36Sopenharmony_ci		return "Enable Slot Command";
10062306a36Sopenharmony_ci	case TRB_DISABLE_SLOT:
10162306a36Sopenharmony_ci		return "Disable Slot Command";
10262306a36Sopenharmony_ci	case TRB_ADDR_DEV:
10362306a36Sopenharmony_ci		return "Address Device Command";
10462306a36Sopenharmony_ci	case TRB_CONFIG_EP:
10562306a36Sopenharmony_ci		return "Configure Endpoint Command";
10662306a36Sopenharmony_ci	case TRB_EVAL_CONTEXT:
10762306a36Sopenharmony_ci		return "Evaluate Context Command";
10862306a36Sopenharmony_ci	case TRB_RESET_EP:
10962306a36Sopenharmony_ci		return "Reset Endpoint Command";
11062306a36Sopenharmony_ci	case TRB_STOP_RING:
11162306a36Sopenharmony_ci		return "Stop Ring Command";
11262306a36Sopenharmony_ci	case TRB_SET_DEQ:
11362306a36Sopenharmony_ci		return "Set TR Dequeue Pointer Command";
11462306a36Sopenharmony_ci	case TRB_RESET_DEV:
11562306a36Sopenharmony_ci		return "Reset Device Command";
11662306a36Sopenharmony_ci	case TRB_FORCE_HEADER:
11762306a36Sopenharmony_ci		return "Force Header Command";
11862306a36Sopenharmony_ci	case TRB_CMD_NOOP:
11962306a36Sopenharmony_ci		return "No-Op Command";
12062306a36Sopenharmony_ci	case TRB_TRANSFER:
12162306a36Sopenharmony_ci		return "Transfer Event";
12262306a36Sopenharmony_ci	case TRB_COMPLETION:
12362306a36Sopenharmony_ci		return "Command Completion Event";
12462306a36Sopenharmony_ci	case TRB_PORT_STATUS:
12562306a36Sopenharmony_ci		return "Port Status Change Event";
12662306a36Sopenharmony_ci	case TRB_HC_EVENT:
12762306a36Sopenharmony_ci		return "Device Controller Event";
12862306a36Sopenharmony_ci	case TRB_MFINDEX_WRAP:
12962306a36Sopenharmony_ci		return "MFINDEX Wrap Event";
13062306a36Sopenharmony_ci	case TRB_ENDPOINT_NRDY:
13162306a36Sopenharmony_ci		return "Endpoint Not ready";
13262306a36Sopenharmony_ci	case TRB_HALT_ENDPOINT:
13362306a36Sopenharmony_ci		return "Halt Endpoint";
13462306a36Sopenharmony_ci	case TRB_FLUSH_ENDPOINT:
13562306a36Sopenharmony_ci		return "FLush Endpoint";
13662306a36Sopenharmony_ci	default:
13762306a36Sopenharmony_ci		return "UNKNOWN";
13862306a36Sopenharmony_ci	}
13962306a36Sopenharmony_ci}
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_cistatic inline const char *cdnsp_ring_type_string(enum cdnsp_ring_type type)
14262306a36Sopenharmony_ci{
14362306a36Sopenharmony_ci	switch (type) {
14462306a36Sopenharmony_ci	case TYPE_CTRL:
14562306a36Sopenharmony_ci		return "CTRL";
14662306a36Sopenharmony_ci	case TYPE_ISOC:
14762306a36Sopenharmony_ci		return "ISOC";
14862306a36Sopenharmony_ci	case TYPE_BULK:
14962306a36Sopenharmony_ci		return "BULK";
15062306a36Sopenharmony_ci	case TYPE_INTR:
15162306a36Sopenharmony_ci		return "INTR";
15262306a36Sopenharmony_ci	case TYPE_STREAM:
15362306a36Sopenharmony_ci		return "STREAM";
15462306a36Sopenharmony_ci	case TYPE_COMMAND:
15562306a36Sopenharmony_ci		return "CMD";
15662306a36Sopenharmony_ci	case TYPE_EVENT:
15762306a36Sopenharmony_ci		return "EVENT";
15862306a36Sopenharmony_ci	}
15962306a36Sopenharmony_ci
16062306a36Sopenharmony_ci	return "UNKNOWN";
16162306a36Sopenharmony_ci}
16262306a36Sopenharmony_ci
16362306a36Sopenharmony_cistatic inline char *cdnsp_slot_state_string(u32 state)
16462306a36Sopenharmony_ci{
16562306a36Sopenharmony_ci	switch (state) {
16662306a36Sopenharmony_ci	case SLOT_STATE_ENABLED:
16762306a36Sopenharmony_ci		return "enabled/disabled";
16862306a36Sopenharmony_ci	case SLOT_STATE_DEFAULT:
16962306a36Sopenharmony_ci		return "default";
17062306a36Sopenharmony_ci	case SLOT_STATE_ADDRESSED:
17162306a36Sopenharmony_ci		return "addressed";
17262306a36Sopenharmony_ci	case SLOT_STATE_CONFIGURED:
17362306a36Sopenharmony_ci		return "configured";
17462306a36Sopenharmony_ci	default:
17562306a36Sopenharmony_ci		return "reserved";
17662306a36Sopenharmony_ci	}
17762306a36Sopenharmony_ci}
17862306a36Sopenharmony_ci
17962306a36Sopenharmony_cistatic inline const char *cdnsp_decode_trb(char *str, size_t size, u32 field0,
18062306a36Sopenharmony_ci					   u32 field1, u32 field2, u32 field3)
18162306a36Sopenharmony_ci{
18262306a36Sopenharmony_ci	int ep_id = TRB_TO_EP_INDEX(field3) - 1;
18362306a36Sopenharmony_ci	int type = TRB_FIELD_TO_TYPE(field3);
18462306a36Sopenharmony_ci	unsigned int ep_num;
18562306a36Sopenharmony_ci	int ret;
18662306a36Sopenharmony_ci	u32 temp;
18762306a36Sopenharmony_ci
18862306a36Sopenharmony_ci	ep_num = DIV_ROUND_UP(ep_id, 2);
18962306a36Sopenharmony_ci
19062306a36Sopenharmony_ci	switch (type) {
19162306a36Sopenharmony_ci	case TRB_LINK:
19262306a36Sopenharmony_ci		ret = snprintf(str, size,
19362306a36Sopenharmony_ci			       "LINK %08x%08x intr %ld type '%s' flags %c:%c:%c:%c",
19462306a36Sopenharmony_ci			       field1, field0, GET_INTR_TARGET(field2),
19562306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
19662306a36Sopenharmony_ci			       field3 & TRB_IOC ? 'I' : 'i',
19762306a36Sopenharmony_ci			       field3 & TRB_CHAIN ? 'C' : 'c',
19862306a36Sopenharmony_ci			       field3 & TRB_TC ? 'T' : 't',
19962306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
20062306a36Sopenharmony_ci		break;
20162306a36Sopenharmony_ci	case TRB_TRANSFER:
20262306a36Sopenharmony_ci	case TRB_COMPLETION:
20362306a36Sopenharmony_ci	case TRB_PORT_STATUS:
20462306a36Sopenharmony_ci	case TRB_HC_EVENT:
20562306a36Sopenharmony_ci		ret = snprintf(str, size,
20662306a36Sopenharmony_ci			       "ep%d%s(%d) type '%s' TRB %08x%08x status '%s'"
20762306a36Sopenharmony_ci			       " len %ld slot %ld flags %c:%c",
20862306a36Sopenharmony_ci			       ep_num, ep_id % 2 ? "out" : "in",
20962306a36Sopenharmony_ci			       TRB_TO_EP_INDEX(field3),
21062306a36Sopenharmony_ci			       cdnsp_trb_type_string(type), field1, field0,
21162306a36Sopenharmony_ci			       cdnsp_trb_comp_code_string(GET_COMP_CODE(field2)),
21262306a36Sopenharmony_ci			       EVENT_TRB_LEN(field2), TRB_TO_SLOT_ID(field3),
21362306a36Sopenharmony_ci			       field3 & EVENT_DATA ? 'E' : 'e',
21462306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
21562306a36Sopenharmony_ci		break;
21662306a36Sopenharmony_ci	case TRB_MFINDEX_WRAP:
21762306a36Sopenharmony_ci		ret = snprintf(str, size, "%s: flags %c",
21862306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
21962306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
22062306a36Sopenharmony_ci		break;
22162306a36Sopenharmony_ci	case TRB_SETUP:
22262306a36Sopenharmony_ci		ret = snprintf(str, size,
22362306a36Sopenharmony_ci			       "type '%s' bRequestType %02x bRequest %02x "
22462306a36Sopenharmony_ci			       "wValue %02x%02x wIndex %02x%02x wLength %d "
22562306a36Sopenharmony_ci			       "length %ld TD size %ld intr %ld Setup ID %ld "
22662306a36Sopenharmony_ci			       "flags %c:%c:%c",
22762306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
22862306a36Sopenharmony_ci			       field0 & 0xff,
22962306a36Sopenharmony_ci			       (field0 & 0xff00) >> 8,
23062306a36Sopenharmony_ci			       (field0 & 0xff000000) >> 24,
23162306a36Sopenharmony_ci			       (field0 & 0xff0000) >> 16,
23262306a36Sopenharmony_ci			       (field1 & 0xff00) >> 8,
23362306a36Sopenharmony_ci			       field1 & 0xff,
23462306a36Sopenharmony_ci			       (field1 & 0xff000000) >> 16 |
23562306a36Sopenharmony_ci			       (field1 & 0xff0000) >> 16,
23662306a36Sopenharmony_ci			       TRB_LEN(field2), GET_TD_SIZE(field2),
23762306a36Sopenharmony_ci			       GET_INTR_TARGET(field2),
23862306a36Sopenharmony_ci			       TRB_SETUPID_TO_TYPE(field3),
23962306a36Sopenharmony_ci			       field3 & TRB_IDT ? 'D' : 'd',
24062306a36Sopenharmony_ci			       field3 & TRB_IOC ? 'I' : 'i',
24162306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
24262306a36Sopenharmony_ci		break;
24362306a36Sopenharmony_ci	case TRB_DATA:
24462306a36Sopenharmony_ci		ret = snprintf(str, size,
24562306a36Sopenharmony_ci			       "type '%s' Buffer %08x%08x length %ld TD size %ld "
24662306a36Sopenharmony_ci			       "intr %ld flags %c:%c:%c:%c:%c:%c:%c",
24762306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
24862306a36Sopenharmony_ci			       field1, field0, TRB_LEN(field2),
24962306a36Sopenharmony_ci			       GET_TD_SIZE(field2),
25062306a36Sopenharmony_ci			       GET_INTR_TARGET(field2),
25162306a36Sopenharmony_ci			       field3 & TRB_IDT ? 'D' : 'i',
25262306a36Sopenharmony_ci			       field3 & TRB_IOC ? 'I' : 'i',
25362306a36Sopenharmony_ci			       field3 & TRB_CHAIN ? 'C' : 'c',
25462306a36Sopenharmony_ci			       field3 & TRB_NO_SNOOP ? 'S' : 's',
25562306a36Sopenharmony_ci			       field3 & TRB_ISP ? 'I' : 'i',
25662306a36Sopenharmony_ci			       field3 & TRB_ENT ? 'E' : 'e',
25762306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
25862306a36Sopenharmony_ci		break;
25962306a36Sopenharmony_ci	case TRB_STATUS:
26062306a36Sopenharmony_ci		ret = snprintf(str, size,
26162306a36Sopenharmony_ci			       "Buffer %08x%08x length %ld TD size %ld intr"
26262306a36Sopenharmony_ci			       "%ld type '%s' flags %c:%c:%c:%c",
26362306a36Sopenharmony_ci			       field1, field0, TRB_LEN(field2),
26462306a36Sopenharmony_ci			       GET_TD_SIZE(field2),
26562306a36Sopenharmony_ci			       GET_INTR_TARGET(field2),
26662306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
26762306a36Sopenharmony_ci			       field3 & TRB_IOC ? 'I' : 'i',
26862306a36Sopenharmony_ci			       field3 & TRB_CHAIN ? 'C' : 'c',
26962306a36Sopenharmony_ci			       field3 & TRB_ENT ? 'E' : 'e',
27062306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
27162306a36Sopenharmony_ci		break;
27262306a36Sopenharmony_ci	case TRB_NORMAL:
27362306a36Sopenharmony_ci	case TRB_ISOC:
27462306a36Sopenharmony_ci	case TRB_EVENT_DATA:
27562306a36Sopenharmony_ci	case TRB_TR_NOOP:
27662306a36Sopenharmony_ci		ret = snprintf(str, size,
27762306a36Sopenharmony_ci			       "type '%s' Buffer %08x%08x length %ld "
27862306a36Sopenharmony_ci			       "TD size %ld intr %ld "
27962306a36Sopenharmony_ci			       "flags %c:%c:%c:%c:%c:%c:%c:%c:%c",
28062306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
28162306a36Sopenharmony_ci			       field1, field0, TRB_LEN(field2),
28262306a36Sopenharmony_ci			       GET_TD_SIZE(field2),
28362306a36Sopenharmony_ci			       GET_INTR_TARGET(field2),
28462306a36Sopenharmony_ci			       field3 & TRB_BEI ? 'B' : 'b',
28562306a36Sopenharmony_ci			       field3 & TRB_IDT ? 'T' : 't',
28662306a36Sopenharmony_ci			       field3 & TRB_IOC ? 'I' : 'i',
28762306a36Sopenharmony_ci			       field3 & TRB_CHAIN ? 'C' : 'c',
28862306a36Sopenharmony_ci			       field3 & TRB_NO_SNOOP ? 'S' : 's',
28962306a36Sopenharmony_ci			       field3 & TRB_ISP ? 'I' : 'i',
29062306a36Sopenharmony_ci			       field3 & TRB_ENT ? 'E' : 'e',
29162306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c',
29262306a36Sopenharmony_ci			       !(field3 & TRB_EVENT_INVALIDATE) ? 'V' : 'v');
29362306a36Sopenharmony_ci		break;
29462306a36Sopenharmony_ci	case TRB_CMD_NOOP:
29562306a36Sopenharmony_ci	case TRB_ENABLE_SLOT:
29662306a36Sopenharmony_ci		ret = snprintf(str, size, "%s: flags %c",
29762306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
29862306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
29962306a36Sopenharmony_ci		break;
30062306a36Sopenharmony_ci	case TRB_DISABLE_SLOT:
30162306a36Sopenharmony_ci		ret = snprintf(str, size, "%s: slot %ld flags %c",
30262306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
30362306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
30462306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
30562306a36Sopenharmony_ci		break;
30662306a36Sopenharmony_ci	case TRB_ADDR_DEV:
30762306a36Sopenharmony_ci		ret = snprintf(str, size,
30862306a36Sopenharmony_ci			       "%s: ctx %08x%08x slot %ld flags %c:%c",
30962306a36Sopenharmony_ci			       cdnsp_trb_type_string(type), field1, field0,
31062306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
31162306a36Sopenharmony_ci			       field3 & TRB_BSR ? 'B' : 'b',
31262306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
31362306a36Sopenharmony_ci		break;
31462306a36Sopenharmony_ci	case TRB_CONFIG_EP:
31562306a36Sopenharmony_ci		ret = snprintf(str, size,
31662306a36Sopenharmony_ci			       "%s: ctx %08x%08x slot %ld flags %c:%c",
31762306a36Sopenharmony_ci			       cdnsp_trb_type_string(type), field1, field0,
31862306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
31962306a36Sopenharmony_ci			       field3 & TRB_DC ? 'D' : 'd',
32062306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
32162306a36Sopenharmony_ci		break;
32262306a36Sopenharmony_ci	case TRB_EVAL_CONTEXT:
32362306a36Sopenharmony_ci		ret = snprintf(str, size,
32462306a36Sopenharmony_ci			       "%s: ctx %08x%08x slot %ld flags %c",
32562306a36Sopenharmony_ci			       cdnsp_trb_type_string(type), field1, field0,
32662306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
32762306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
32862306a36Sopenharmony_ci		break;
32962306a36Sopenharmony_ci	case TRB_RESET_EP:
33062306a36Sopenharmony_ci	case TRB_HALT_ENDPOINT:
33162306a36Sopenharmony_ci	case TRB_FLUSH_ENDPOINT:
33262306a36Sopenharmony_ci		ret = snprintf(str, size,
33362306a36Sopenharmony_ci			       "%s: ep%d%s(%d) ctx %08x%08x slot %ld flags %c",
33462306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
33562306a36Sopenharmony_ci			       ep_num, ep_id % 2 ? "out" : "in",
33662306a36Sopenharmony_ci			       TRB_TO_EP_INDEX(field3), field1, field0,
33762306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
33862306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
33962306a36Sopenharmony_ci		break;
34062306a36Sopenharmony_ci	case TRB_STOP_RING:
34162306a36Sopenharmony_ci		ret = snprintf(str, size,
34262306a36Sopenharmony_ci			       "%s: ep%d%s(%d) slot %ld sp %d flags %c",
34362306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
34462306a36Sopenharmony_ci			       ep_num, ep_id % 2 ? "out" : "in",
34562306a36Sopenharmony_ci			       TRB_TO_EP_INDEX(field3),
34662306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
34762306a36Sopenharmony_ci			       TRB_TO_SUSPEND_PORT(field3),
34862306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
34962306a36Sopenharmony_ci		break;
35062306a36Sopenharmony_ci	case TRB_SET_DEQ:
35162306a36Sopenharmony_ci		ret = snprintf(str, size,
35262306a36Sopenharmony_ci			       "%s: ep%d%s(%d) deq %08x%08x stream %ld slot %ld  flags %c",
35362306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
35462306a36Sopenharmony_ci			       ep_num, ep_id % 2 ? "out" : "in",
35562306a36Sopenharmony_ci			       TRB_TO_EP_INDEX(field3), field1, field0,
35662306a36Sopenharmony_ci			       TRB_TO_STREAM_ID(field2),
35762306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
35862306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
35962306a36Sopenharmony_ci		break;
36062306a36Sopenharmony_ci	case TRB_RESET_DEV:
36162306a36Sopenharmony_ci		ret = snprintf(str, size, "%s: slot %ld flags %c",
36262306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
36362306a36Sopenharmony_ci			       TRB_TO_SLOT_ID(field3),
36462306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
36562306a36Sopenharmony_ci		break;
36662306a36Sopenharmony_ci	case TRB_ENDPOINT_NRDY:
36762306a36Sopenharmony_ci		temp = TRB_TO_HOST_STREAM(field2);
36862306a36Sopenharmony_ci
36962306a36Sopenharmony_ci		ret = snprintf(str, size,
37062306a36Sopenharmony_ci			       "%s: ep%d%s(%d) H_SID %x%s%s D_SID %lx flags %c:%c",
37162306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
37262306a36Sopenharmony_ci			       ep_num, ep_id % 2 ? "out" : "in",
37362306a36Sopenharmony_ci			       TRB_TO_EP_INDEX(field3), temp,
37462306a36Sopenharmony_ci			       temp == STREAM_PRIME_ACK ? "(PRIME)" : "",
37562306a36Sopenharmony_ci			       temp == STREAM_REJECTED ? "(REJECTED)" : "",
37662306a36Sopenharmony_ci			       TRB_TO_DEV_STREAM(field0),
37762306a36Sopenharmony_ci			       field3 & TRB_STAT ? 'S' : 's',
37862306a36Sopenharmony_ci			       field3 & TRB_CYCLE ? 'C' : 'c');
37962306a36Sopenharmony_ci		break;
38062306a36Sopenharmony_ci	default:
38162306a36Sopenharmony_ci		ret = snprintf(str, size,
38262306a36Sopenharmony_ci			       "type '%s' -> raw %08x %08x %08x %08x",
38362306a36Sopenharmony_ci			       cdnsp_trb_type_string(type),
38462306a36Sopenharmony_ci			       field0, field1, field2, field3);
38562306a36Sopenharmony_ci	}
38662306a36Sopenharmony_ci
38762306a36Sopenharmony_ci	if (ret >= size)
38862306a36Sopenharmony_ci		pr_info("CDNSP: buffer overflowed.\n");
38962306a36Sopenharmony_ci
39062306a36Sopenharmony_ci	return str;
39162306a36Sopenharmony_ci}
39262306a36Sopenharmony_ci
39362306a36Sopenharmony_cistatic inline const char *cdnsp_decode_slot_context(u32 info, u32 info2,
39462306a36Sopenharmony_ci						    u32 int_target, u32 state)
39562306a36Sopenharmony_ci{
39662306a36Sopenharmony_ci	static char str[1024];
39762306a36Sopenharmony_ci	int ret = 0;
39862306a36Sopenharmony_ci	u32 speed;
39962306a36Sopenharmony_ci	char *s;
40062306a36Sopenharmony_ci
40162306a36Sopenharmony_ci	speed = info & DEV_SPEED;
40262306a36Sopenharmony_ci
40362306a36Sopenharmony_ci	switch (speed) {
40462306a36Sopenharmony_ci	case SLOT_SPEED_FS:
40562306a36Sopenharmony_ci		s = "full-speed";
40662306a36Sopenharmony_ci		break;
40762306a36Sopenharmony_ci	case SLOT_SPEED_HS:
40862306a36Sopenharmony_ci		s = "high-speed";
40962306a36Sopenharmony_ci		break;
41062306a36Sopenharmony_ci	case SLOT_SPEED_SS:
41162306a36Sopenharmony_ci		s = "super-speed";
41262306a36Sopenharmony_ci		break;
41362306a36Sopenharmony_ci	case SLOT_SPEED_SSP:
41462306a36Sopenharmony_ci		s = "super-speed plus";
41562306a36Sopenharmony_ci		break;
41662306a36Sopenharmony_ci	default:
41762306a36Sopenharmony_ci		s = "UNKNOWN speed";
41862306a36Sopenharmony_ci	}
41962306a36Sopenharmony_ci
42062306a36Sopenharmony_ci	ret = sprintf(str, "%s Ctx Entries %d",
42162306a36Sopenharmony_ci		      s, (info & LAST_CTX_MASK) >> 27);
42262306a36Sopenharmony_ci
42362306a36Sopenharmony_ci	ret += sprintf(str + ret, " [Intr %ld] Addr %ld State %s",
42462306a36Sopenharmony_ci		       GET_INTR_TARGET(int_target), state & DEV_ADDR_MASK,
42562306a36Sopenharmony_ci		       cdnsp_slot_state_string(GET_SLOT_STATE(state)));
42662306a36Sopenharmony_ci
42762306a36Sopenharmony_ci	return str;
42862306a36Sopenharmony_ci}
42962306a36Sopenharmony_ci
43062306a36Sopenharmony_cistatic inline const char *cdnsp_portsc_link_state_string(u32 portsc)
43162306a36Sopenharmony_ci{
43262306a36Sopenharmony_ci	switch (portsc & PORT_PLS_MASK) {
43362306a36Sopenharmony_ci	case XDEV_U0:
43462306a36Sopenharmony_ci		return "U0";
43562306a36Sopenharmony_ci	case XDEV_U1:
43662306a36Sopenharmony_ci		return "U1";
43762306a36Sopenharmony_ci	case XDEV_U2:
43862306a36Sopenharmony_ci		return "U2";
43962306a36Sopenharmony_ci	case XDEV_U3:
44062306a36Sopenharmony_ci		return "U3";
44162306a36Sopenharmony_ci	case XDEV_DISABLED:
44262306a36Sopenharmony_ci		return "Disabled";
44362306a36Sopenharmony_ci	case XDEV_RXDETECT:
44462306a36Sopenharmony_ci		return "RxDetect";
44562306a36Sopenharmony_ci	case XDEV_INACTIVE:
44662306a36Sopenharmony_ci		return "Inactive";
44762306a36Sopenharmony_ci	case XDEV_POLLING:
44862306a36Sopenharmony_ci		return "Polling";
44962306a36Sopenharmony_ci	case XDEV_RECOVERY:
45062306a36Sopenharmony_ci		return "Recovery";
45162306a36Sopenharmony_ci	case XDEV_HOT_RESET:
45262306a36Sopenharmony_ci		return "Hot Reset";
45362306a36Sopenharmony_ci	case XDEV_COMP_MODE:
45462306a36Sopenharmony_ci		return "Compliance mode";
45562306a36Sopenharmony_ci	case XDEV_TEST_MODE:
45662306a36Sopenharmony_ci		return "Test mode";
45762306a36Sopenharmony_ci	case XDEV_RESUME:
45862306a36Sopenharmony_ci		return "Resume";
45962306a36Sopenharmony_ci	default:
46062306a36Sopenharmony_ci		break;
46162306a36Sopenharmony_ci	}
46262306a36Sopenharmony_ci
46362306a36Sopenharmony_ci	return "Unknown";
46462306a36Sopenharmony_ci}
46562306a36Sopenharmony_ci
46662306a36Sopenharmony_cistatic inline const char *cdnsp_decode_portsc(char *str, size_t size,
46762306a36Sopenharmony_ci					      u32 portsc)
46862306a36Sopenharmony_ci{
46962306a36Sopenharmony_ci	int ret;
47062306a36Sopenharmony_ci
47162306a36Sopenharmony_ci	ret = snprintf(str, size, "%s %s %s Link:%s PortSpeed:%d ",
47262306a36Sopenharmony_ci		       portsc & PORT_POWER ? "Powered" : "Powered-off",
47362306a36Sopenharmony_ci		       portsc & PORT_CONNECT ? "Connected" : "Not-connected",
47462306a36Sopenharmony_ci		       portsc & PORT_PED ? "Enabled" : "Disabled",
47562306a36Sopenharmony_ci		       cdnsp_portsc_link_state_string(portsc),
47662306a36Sopenharmony_ci		       DEV_PORT_SPEED(portsc));
47762306a36Sopenharmony_ci
47862306a36Sopenharmony_ci	if (portsc & PORT_RESET)
47962306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "In-Reset ");
48062306a36Sopenharmony_ci
48162306a36Sopenharmony_ci	ret += snprintf(str + ret, size - ret, "Change: ");
48262306a36Sopenharmony_ci	if (portsc & PORT_CSC)
48362306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "CSC ");
48462306a36Sopenharmony_ci	if (portsc & PORT_WRC)
48562306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "WRC ");
48662306a36Sopenharmony_ci	if (portsc & PORT_RC)
48762306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "PRC ");
48862306a36Sopenharmony_ci	if (portsc & PORT_PLC)
48962306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "PLC ");
49062306a36Sopenharmony_ci	if (portsc & PORT_CEC)
49162306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "CEC ");
49262306a36Sopenharmony_ci	ret += snprintf(str + ret, size - ret, "Wake: ");
49362306a36Sopenharmony_ci	if (portsc & PORT_WKCONN_E)
49462306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "WCE ");
49562306a36Sopenharmony_ci	if (portsc & PORT_WKDISC_E)
49662306a36Sopenharmony_ci		ret += snprintf(str + ret, size - ret, "WDE ");
49762306a36Sopenharmony_ci
49862306a36Sopenharmony_ci	return str;
49962306a36Sopenharmony_ci}
50062306a36Sopenharmony_ci
50162306a36Sopenharmony_cistatic inline const char *cdnsp_ep_state_string(u8 state)
50262306a36Sopenharmony_ci{
50362306a36Sopenharmony_ci	switch (state) {
50462306a36Sopenharmony_ci	case EP_STATE_DISABLED:
50562306a36Sopenharmony_ci		return "disabled";
50662306a36Sopenharmony_ci	case EP_STATE_RUNNING:
50762306a36Sopenharmony_ci		return "running";
50862306a36Sopenharmony_ci	case EP_STATE_HALTED:
50962306a36Sopenharmony_ci		return "halted";
51062306a36Sopenharmony_ci	case EP_STATE_STOPPED:
51162306a36Sopenharmony_ci		return "stopped";
51262306a36Sopenharmony_ci	case EP_STATE_ERROR:
51362306a36Sopenharmony_ci		return "error";
51462306a36Sopenharmony_ci	default:
51562306a36Sopenharmony_ci		return "INVALID";
51662306a36Sopenharmony_ci	}
51762306a36Sopenharmony_ci}
51862306a36Sopenharmony_ci
51962306a36Sopenharmony_cistatic inline const char *cdnsp_ep_type_string(u8 type)
52062306a36Sopenharmony_ci{
52162306a36Sopenharmony_ci	switch (type) {
52262306a36Sopenharmony_ci	case ISOC_OUT_EP:
52362306a36Sopenharmony_ci		return "Isoc OUT";
52462306a36Sopenharmony_ci	case BULK_OUT_EP:
52562306a36Sopenharmony_ci		return "Bulk OUT";
52662306a36Sopenharmony_ci	case INT_OUT_EP:
52762306a36Sopenharmony_ci		return "Int OUT";
52862306a36Sopenharmony_ci	case CTRL_EP:
52962306a36Sopenharmony_ci		return "Ctrl";
53062306a36Sopenharmony_ci	case ISOC_IN_EP:
53162306a36Sopenharmony_ci		return "Isoc IN";
53262306a36Sopenharmony_ci	case BULK_IN_EP:
53362306a36Sopenharmony_ci		return "Bulk IN";
53462306a36Sopenharmony_ci	case INT_IN_EP:
53562306a36Sopenharmony_ci		return "Int IN";
53662306a36Sopenharmony_ci	default:
53762306a36Sopenharmony_ci		return "INVALID";
53862306a36Sopenharmony_ci	}
53962306a36Sopenharmony_ci}
54062306a36Sopenharmony_ci
54162306a36Sopenharmony_cistatic inline const char *cdnsp_decode_ep_context(char *str, size_t size,
54262306a36Sopenharmony_ci						  u32 info, u32 info2,
54362306a36Sopenharmony_ci						  u64 deq, u32 tx_info)
54462306a36Sopenharmony_ci{
54562306a36Sopenharmony_ci	u8 max_pstr, ep_state, interval, ep_type, burst, cerr, mult;
54662306a36Sopenharmony_ci	bool lsa, hid;
54762306a36Sopenharmony_ci	u16 maxp, avg;
54862306a36Sopenharmony_ci	u32 esit;
54962306a36Sopenharmony_ci	int ret;
55062306a36Sopenharmony_ci
55162306a36Sopenharmony_ci	esit = CTX_TO_MAX_ESIT_PAYLOAD_HI(info) << 16 |
55262306a36Sopenharmony_ci	       CTX_TO_MAX_ESIT_PAYLOAD_LO(tx_info);
55362306a36Sopenharmony_ci
55462306a36Sopenharmony_ci	ep_state = info & EP_STATE_MASK;
55562306a36Sopenharmony_ci	max_pstr = CTX_TO_EP_MAXPSTREAMS(info);
55662306a36Sopenharmony_ci	interval = CTX_TO_EP_INTERVAL(info);
55762306a36Sopenharmony_ci	mult = CTX_TO_EP_MULT(info) + 1;
55862306a36Sopenharmony_ci	lsa = !!(info & EP_HAS_LSA);
55962306a36Sopenharmony_ci
56062306a36Sopenharmony_ci	cerr = (info2 & (3 << 1)) >> 1;
56162306a36Sopenharmony_ci	ep_type = CTX_TO_EP_TYPE(info2);
56262306a36Sopenharmony_ci	hid = !!(info2 & (1 << 7));
56362306a36Sopenharmony_ci	burst = CTX_TO_MAX_BURST(info2);
56462306a36Sopenharmony_ci	maxp = MAX_PACKET_DECODED(info2);
56562306a36Sopenharmony_ci
56662306a36Sopenharmony_ci	avg = EP_AVG_TRB_LENGTH(tx_info);
56762306a36Sopenharmony_ci
56862306a36Sopenharmony_ci	ret = snprintf(str, size, "State %s mult %d max P. Streams %d %s",
56962306a36Sopenharmony_ci		       cdnsp_ep_state_string(ep_state), mult,
57062306a36Sopenharmony_ci		       max_pstr, lsa ? "LSA " : "");
57162306a36Sopenharmony_ci
57262306a36Sopenharmony_ci	ret += snprintf(str + ret, size - ret,
57362306a36Sopenharmony_ci			"interval %d us max ESIT payload %d CErr %d ",
57462306a36Sopenharmony_ci			(1 << interval) * 125, esit, cerr);
57562306a36Sopenharmony_ci
57662306a36Sopenharmony_ci	ret += snprintf(str + ret, size - ret,
57762306a36Sopenharmony_ci			"Type %s %sburst %d maxp %d deq %016llx ",
57862306a36Sopenharmony_ci			cdnsp_ep_type_string(ep_type), hid ? "HID" : "",
57962306a36Sopenharmony_ci			burst, maxp, deq);
58062306a36Sopenharmony_ci
58162306a36Sopenharmony_ci	ret += snprintf(str + ret, size - ret, "avg trb len %d", avg);
58262306a36Sopenharmony_ci
58362306a36Sopenharmony_ci	return str;
58462306a36Sopenharmony_ci}
58562306a36Sopenharmony_ci
58662306a36Sopenharmony_ci#endif /*__LINUX_CDNSP_DEBUG*/
587