xref: /linux/drivers/crypto/hisilicon/debugfs.c (revision 001821b0e79716c4e17c71d8e053a23599a7a508)
1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright (c) 2022 HiSilicon Limited. */
3 #include <linux/hisi_acc_qm.h>
4 #include "qm_common.h"
5 
6 #define QM_DFX_BASE			0x0100000
7 #define QM_DFX_STATE1			0x0104000
8 #define QM_DFX_STATE2			0x01040C8
9 #define QM_DFX_COMMON			0x0000
10 #define QM_DFX_BASE_LEN			0x5A
11 #define QM_DFX_STATE1_LEN		0x2E
12 #define QM_DFX_STATE2_LEN		0x11
13 #define QM_DFX_COMMON_LEN		0xC3
14 #define QM_DFX_REGS_LEN			4UL
15 #define QM_DBG_TMP_BUF_LEN		22
16 #define QM_XQC_ADDR_MASK		GENMASK(31, 0)
17 #define CURRENT_FUN_MASK		GENMASK(5, 0)
18 #define CURRENT_Q_MASK			GENMASK(31, 16)
19 #define QM_SQE_ADDR_MASK		GENMASK(7, 0)
20 
21 #define QM_DFX_MB_CNT_VF		0x104010
22 #define QM_DFX_DB_CNT_VF		0x104020
23 #define QM_DFX_SQE_CNT_VF_SQN		0x104030
24 #define QM_DFX_CQE_CNT_VF_CQN		0x104040
25 #define QM_DFX_QN_SHIFT			16
26 #define QM_DFX_CNT_CLR_CE		0x100118
27 #define QM_DBG_WRITE_LEN		1024
28 #define QM_IN_IDLE_ST_REG		0x1040e4
29 #define QM_IN_IDLE_STATE		0x1
30 
31 static const char * const qm_debug_file_name[] = {
32 	[CURRENT_QM]   = "current_qm",
33 	[CURRENT_Q]    = "current_q",
34 	[CLEAR_ENABLE] = "clear_enable",
35 };
36 
37 static const char * const qm_s[] = {
38 	"work", "stop",
39 };
40 
41 struct qm_dfx_item {
42 	const char *name;
43 	u32 offset;
44 };
45 
46 struct qm_cmd_dump_item {
47 	const char *cmd;
48 	char *info_name;
49 	int (*dump_fn)(struct hisi_qm *qm, char *cmd, char *info_name);
50 };
51 
52 static struct qm_dfx_item qm_dfx_files[] = {
53 	{"err_irq", offsetof(struct qm_dfx, err_irq_cnt)},
54 	{"aeq_irq", offsetof(struct qm_dfx, aeq_irq_cnt)},
55 	{"abnormal_irq", offsetof(struct qm_dfx, abnormal_irq_cnt)},
56 	{"create_qp_err", offsetof(struct qm_dfx, create_qp_err_cnt)},
57 	{"mb_err", offsetof(struct qm_dfx, mb_err_cnt)},
58 };
59 
60 #define CNT_CYC_REGS_NUM		10
61 static const struct debugfs_reg32 qm_dfx_regs[] = {
62 	/* XXX_CNT are reading clear register */
63 	{"QM_ECC_1BIT_CNT               ",  0x104000},
64 	{"QM_ECC_MBIT_CNT               ",  0x104008},
65 	{"QM_DFX_MB_CNT                 ",  0x104018},
66 	{"QM_DFX_DB_CNT                 ",  0x104028},
67 	{"QM_DFX_SQE_CNT                ",  0x104038},
68 	{"QM_DFX_CQE_CNT                ",  0x104048},
69 	{"QM_DFX_SEND_SQE_TO_ACC_CNT    ",  0x104050},
70 	{"QM_DFX_WB_SQE_FROM_ACC_CNT    ",  0x104058},
71 	{"QM_DFX_ACC_FINISH_CNT         ",  0x104060},
72 	{"QM_DFX_CQE_ERR_CNT            ",  0x1040b4},
73 	{"QM_DFX_FUNS_ACTIVE_ST         ",  0x200},
74 	{"QM_ECC_1BIT_INF               ",  0x104004},
75 	{"QM_ECC_MBIT_INF               ",  0x10400c},
76 	{"QM_DFX_ACC_RDY_VLD0           ",  0x1040a0},
77 	{"QM_DFX_ACC_RDY_VLD1           ",  0x1040a4},
78 	{"QM_DFX_AXI_RDY_VLD            ",  0x1040a8},
79 	{"QM_DFX_FF_ST0                 ",  0x1040c8},
80 	{"QM_DFX_FF_ST1                 ",  0x1040cc},
81 	{"QM_DFX_FF_ST2                 ",  0x1040d0},
82 	{"QM_DFX_FF_ST3                 ",  0x1040d4},
83 	{"QM_DFX_FF_ST4                 ",  0x1040d8},
84 	{"QM_DFX_FF_ST5                 ",  0x1040dc},
85 	{"QM_DFX_FF_ST6                 ",  0x1040e0},
86 	{"QM_IN_IDLE_ST                 ",  0x1040e4},
87 	{"QM_CACHE_CTL                  ",  0x100050},
88 	{"QM_TIMEOUT_CFG                ",  0x100070},
89 	{"QM_DB_TIMEOUT_CFG             ",  0x100074},
90 	{"QM_FLR_PENDING_TIME_CFG       ",  0x100078},
91 	{"QM_ARUSR_MCFG1                ",  0x100088},
92 	{"QM_AWUSR_MCFG1                ",  0x100098},
93 	{"QM_AXI_M_CFG_ENABLE           ",  0x1000B0},
94 	{"QM_RAS_CE_THRESHOLD           ",  0x1000F8},
95 	{"QM_AXI_TIMEOUT_CTRL           ",  0x100120},
96 	{"QM_AXI_TIMEOUT_STATUS         ",  0x100124},
97 	{"QM_CQE_AGGR_TIMEOUT_CTRL      ",  0x100144},
98 	{"ACC_RAS_MSI_INT_SEL           ",  0x1040fc},
99 	{"QM_CQE_OUT                    ",  0x104100},
100 	{"QM_EQE_OUT                    ",  0x104104},
101 	{"QM_AEQE_OUT                   ",  0x104108},
102 	{"QM_DB_INFO0                   ",  0x104180},
103 	{"QM_DB_INFO1                   ",  0x104184},
104 	{"QM_AM_CTRL_GLOBAL             ",  0x300000},
105 	{"QM_AM_CURR_PORT_STS           ",  0x300100},
106 	{"QM_AM_CURR_TRANS_RETURN       ",  0x300150},
107 	{"QM_AM_CURR_RD_MAX_TXID        ",  0x300154},
108 	{"QM_AM_CURR_WR_MAX_TXID        ",  0x300158},
109 	{"QM_AM_ALARM_RRESP             ",  0x300180},
110 	{"QM_AM_ALARM_BRESP             ",  0x300184},
111 };
112 
113 static const struct debugfs_reg32 qm_vf_dfx_regs[] = {
114 	{"QM_DFX_FUNS_ACTIVE_ST         ",  0x200},
115 };
116 
117 /* define the QM's dfx regs region and region length */
118 static struct dfx_diff_registers qm_diff_regs[] = {
119 	{
120 		.reg_offset = QM_DFX_BASE,
121 		.reg_len = QM_DFX_BASE_LEN,
122 	}, {
123 		.reg_offset = QM_DFX_STATE1,
124 		.reg_len = QM_DFX_STATE1_LEN,
125 	}, {
126 		.reg_offset = QM_DFX_STATE2,
127 		.reg_len = QM_DFX_STATE2_LEN,
128 	}, {
129 		.reg_offset = QM_DFX_COMMON,
130 		.reg_len = QM_DFX_COMMON_LEN,
131 	},
132 };
133 
134 static struct hisi_qm *file_to_qm(struct debugfs_file *file)
135 {
136 	struct qm_debug *debug = file->debug;
137 
138 	return container_of(debug, struct hisi_qm, debug);
139 }
140 
141 static ssize_t qm_cmd_read(struct file *filp, char __user *buffer,
142 			   size_t count, loff_t *pos)
143 {
144 	char buf[QM_DBG_READ_LEN];
145 	int len;
146 
147 	len = scnprintf(buf, QM_DBG_READ_LEN, "%s\n",
148 			"Please echo help to cmd to get help information");
149 
150 	return simple_read_from_buffer(buffer, count, pos, buf, len);
151 }
152 
153 static void dump_show(struct hisi_qm *qm, void *info,
154 		     unsigned int info_size, char *info_name)
155 {
156 	struct device *dev = &qm->pdev->dev;
157 	u8 *info_curr = info;
158 	u32 i;
159 #define BYTE_PER_DW	4
160 
161 	dev_info(dev, "%s DUMP\n", info_name);
162 	for (i = 0; i < info_size; i += BYTE_PER_DW, info_curr += BYTE_PER_DW) {
163 		pr_info("DW%u: %02X%02X %02X%02X\n", i / BYTE_PER_DW,
164 			*(info_curr + 3), *(info_curr + 2), *(info_curr + 1), *(info_curr));
165 	}
166 }
167 
168 static int qm_sqc_dump(struct hisi_qm *qm, char *s, char *name)
169 {
170 	struct device *dev = &qm->pdev->dev;
171 	struct qm_sqc sqc;
172 	u32 qp_id;
173 	int ret;
174 
175 	if (!s)
176 		return -EINVAL;
177 
178 	ret = kstrtou32(s, 0, &qp_id);
179 	if (ret || qp_id >= qm->qp_num) {
180 		dev_err(dev, "Please input qp num (0-%u)", qm->qp_num - 1);
181 		return -EINVAL;
182 	}
183 
184 	ret = qm_set_and_get_xqc(qm, QM_MB_CMD_SQC, &sqc, qp_id, 1);
185 	if (!ret) {
186 		sqc.base_h = cpu_to_le32(QM_XQC_ADDR_MASK);
187 		sqc.base_l = cpu_to_le32(QM_XQC_ADDR_MASK);
188 		dump_show(qm, &sqc, sizeof(struct qm_sqc), name);
189 
190 		return 0;
191 	}
192 
193 	down_read(&qm->qps_lock);
194 	if (qm->sqc) {
195 		memcpy(&sqc, qm->sqc + qp_id * sizeof(struct qm_sqc), sizeof(struct qm_sqc));
196 		sqc.base_h = cpu_to_le32(QM_XQC_ADDR_MASK);
197 		sqc.base_l = cpu_to_le32(QM_XQC_ADDR_MASK);
198 		dump_show(qm, &sqc, sizeof(struct qm_sqc), "SOFT SQC");
199 	}
200 	up_read(&qm->qps_lock);
201 
202 	return 0;
203 }
204 
205 static int qm_cqc_dump(struct hisi_qm *qm, char *s, char *name)
206 {
207 	struct device *dev = &qm->pdev->dev;
208 	struct qm_cqc cqc;
209 	u32 qp_id;
210 	int ret;
211 
212 	if (!s)
213 		return -EINVAL;
214 
215 	ret = kstrtou32(s, 0, &qp_id);
216 	if (ret || qp_id >= qm->qp_num) {
217 		dev_err(dev, "Please input qp num (0-%u)", qm->qp_num - 1);
218 		return -EINVAL;
219 	}
220 
221 	ret = qm_set_and_get_xqc(qm, QM_MB_CMD_CQC, &cqc, qp_id, 1);
222 	if (!ret) {
223 		cqc.base_h = cpu_to_le32(QM_XQC_ADDR_MASK);
224 		cqc.base_l = cpu_to_le32(QM_XQC_ADDR_MASK);
225 		dump_show(qm, &cqc, sizeof(struct qm_cqc), name);
226 
227 		return 0;
228 	}
229 
230 	down_read(&qm->qps_lock);
231 	if (qm->cqc) {
232 		memcpy(&cqc, qm->cqc + qp_id * sizeof(struct qm_cqc), sizeof(struct qm_cqc));
233 		cqc.base_h = cpu_to_le32(QM_XQC_ADDR_MASK);
234 		cqc.base_l = cpu_to_le32(QM_XQC_ADDR_MASK);
235 		dump_show(qm, &cqc, sizeof(struct qm_cqc), "SOFT CQC");
236 	}
237 	up_read(&qm->qps_lock);
238 
239 	return 0;
240 }
241 
242 static int qm_eqc_aeqc_dump(struct hisi_qm *qm, char *s, char *name)
243 {
244 	struct device *dev = &qm->pdev->dev;
245 	struct qm_aeqc aeqc;
246 	struct qm_eqc eqc;
247 	size_t size;
248 	void *xeqc;
249 	int ret;
250 	u8 cmd;
251 
252 	if (strsep(&s, " ")) {
253 		dev_err(dev, "Please do not input extra characters!\n");
254 		return -EINVAL;
255 	}
256 
257 	if (!strcmp(name, "EQC")) {
258 		cmd = QM_MB_CMD_EQC;
259 		size = sizeof(struct qm_eqc);
260 		xeqc = &eqc;
261 	} else {
262 		cmd = QM_MB_CMD_AEQC;
263 		size = sizeof(struct qm_aeqc);
264 		xeqc = &aeqc;
265 	}
266 
267 	ret = qm_set_and_get_xqc(qm, cmd, xeqc, 0, 1);
268 	if (ret)
269 		return ret;
270 
271 	aeqc.base_h = cpu_to_le32(QM_XQC_ADDR_MASK);
272 	aeqc.base_l = cpu_to_le32(QM_XQC_ADDR_MASK);
273 	eqc.base_h = cpu_to_le32(QM_XQC_ADDR_MASK);
274 	eqc.base_l = cpu_to_le32(QM_XQC_ADDR_MASK);
275 	dump_show(qm, xeqc, size, name);
276 
277 	return ret;
278 }
279 
280 static int q_dump_param_parse(struct hisi_qm *qm, char *s,
281 			      u32 *e_id, u32 *q_id, u16 q_depth)
282 {
283 	struct device *dev = &qm->pdev->dev;
284 	unsigned int qp_num = qm->qp_num;
285 	char *presult;
286 	int ret;
287 
288 	presult = strsep(&s, " ");
289 	if (!presult) {
290 		dev_err(dev, "Please input qp number!\n");
291 		return -EINVAL;
292 	}
293 
294 	ret = kstrtou32(presult, 0, q_id);
295 	if (ret || *q_id >= qp_num) {
296 		dev_err(dev, "Please input qp num (0-%u)", qp_num - 1);
297 		return -EINVAL;
298 	}
299 
300 	presult = strsep(&s, " ");
301 	if (!presult) {
302 		dev_err(dev, "Please input sqe number!\n");
303 		return -EINVAL;
304 	}
305 
306 	ret = kstrtou32(presult, 0, e_id);
307 	if (ret || *e_id >= q_depth) {
308 		dev_err(dev, "Please input sqe num (0-%u)", q_depth - 1);
309 		return -EINVAL;
310 	}
311 
312 	if (strsep(&s, " ")) {
313 		dev_err(dev, "Please do not input extra characters!\n");
314 		return -EINVAL;
315 	}
316 
317 	return 0;
318 }
319 
320 static int qm_sq_dump(struct hisi_qm *qm, char *s, char *name)
321 {
322 	u16 sq_depth = qm->qp_array->sq_depth;
323 	struct hisi_qp *qp;
324 	u32 qp_id, sqe_id;
325 	void *sqe;
326 	int ret;
327 
328 	ret = q_dump_param_parse(qm, s, &sqe_id, &qp_id, sq_depth);
329 	if (ret)
330 		return ret;
331 
332 	sqe = kzalloc(qm->sqe_size, GFP_KERNEL);
333 	if (!sqe)
334 		return -ENOMEM;
335 
336 	qp = &qm->qp_array[qp_id];
337 	memcpy(sqe, qp->sqe + sqe_id * qm->sqe_size, qm->sqe_size);
338 	memset(sqe + qm->debug.sqe_mask_offset, QM_SQE_ADDR_MASK,
339 	       qm->debug.sqe_mask_len);
340 
341 	dump_show(qm, sqe, qm->sqe_size, name);
342 
343 	kfree(sqe);
344 
345 	return 0;
346 }
347 
348 static int qm_cq_dump(struct hisi_qm *qm, char *s, char *name)
349 {
350 	struct qm_cqe *cqe_curr;
351 	struct hisi_qp *qp;
352 	u32 qp_id, cqe_id;
353 	int ret;
354 
355 	ret = q_dump_param_parse(qm, s, &cqe_id, &qp_id, qm->qp_array->cq_depth);
356 	if (ret)
357 		return ret;
358 
359 	qp = &qm->qp_array[qp_id];
360 	cqe_curr = qp->cqe + cqe_id;
361 	dump_show(qm, cqe_curr, sizeof(struct qm_cqe), name);
362 
363 	return 0;
364 }
365 
366 static int qm_eq_aeq_dump(struct hisi_qm *qm, char *s, char *name)
367 {
368 	struct device *dev = &qm->pdev->dev;
369 	u16 xeq_depth;
370 	size_t size;
371 	void *xeqe;
372 	u32 xeqe_id;
373 	int ret;
374 
375 	if (!s)
376 		return -EINVAL;
377 
378 	ret = kstrtou32(s, 0, &xeqe_id);
379 	if (ret)
380 		return -EINVAL;
381 
382 	if (!strcmp(name, "EQE")) {
383 		xeq_depth = qm->eq_depth;
384 		size = sizeof(struct qm_eqe);
385 	} else {
386 		xeq_depth = qm->aeq_depth;
387 		size = sizeof(struct qm_aeqe);
388 	}
389 
390 	if (xeqe_id >= xeq_depth) {
391 		dev_err(dev, "Please input eqe or aeqe num (0-%u)", xeq_depth - 1);
392 		return -EINVAL;
393 	}
394 
395 	down_read(&qm->qps_lock);
396 
397 	if (qm->eqe && !strcmp(name, "EQE")) {
398 		xeqe = qm->eqe + xeqe_id;
399 	} else if (qm->aeqe && !strcmp(name, "AEQE")) {
400 		xeqe = qm->aeqe + xeqe_id;
401 	} else {
402 		ret = -EINVAL;
403 		goto err_unlock;
404 	}
405 
406 	dump_show(qm, xeqe, size, name);
407 
408 err_unlock:
409 	up_read(&qm->qps_lock);
410 	return ret;
411 }
412 
413 static int qm_dbg_help(struct hisi_qm *qm, char *s)
414 {
415 	struct device *dev = &qm->pdev->dev;
416 
417 	if (strsep(&s, " ")) {
418 		dev_err(dev, "Please do not input extra characters!\n");
419 		return -EINVAL;
420 	}
421 
422 	dev_info(dev, "available commands:\n");
423 	dev_info(dev, "sqc <num>\n");
424 	dev_info(dev, "cqc <num>\n");
425 	dev_info(dev, "eqc\n");
426 	dev_info(dev, "aeqc\n");
427 	dev_info(dev, "sq <num> <e>\n");
428 	dev_info(dev, "cq <num> <e>\n");
429 	dev_info(dev, "eq <e>\n");
430 	dev_info(dev, "aeq <e>\n");
431 
432 	return 0;
433 }
434 
435 static const struct qm_cmd_dump_item qm_cmd_dump_table[] = {
436 	{
437 		.cmd = "sqc",
438 		.info_name = "SQC",
439 		.dump_fn = qm_sqc_dump,
440 	}, {
441 		.cmd = "cqc",
442 		.info_name = "CQC",
443 		.dump_fn = qm_cqc_dump,
444 	}, {
445 		.cmd = "eqc",
446 		.info_name = "EQC",
447 		.dump_fn = qm_eqc_aeqc_dump,
448 	}, {
449 		.cmd = "aeqc",
450 		.info_name = "AEQC",
451 		.dump_fn = qm_eqc_aeqc_dump,
452 	}, {
453 		.cmd = "sq",
454 		.info_name = "SQE",
455 		.dump_fn = qm_sq_dump,
456 	}, {
457 		.cmd = "cq",
458 		.info_name = "CQE",
459 		.dump_fn = qm_cq_dump,
460 	}, {
461 		.cmd = "eq",
462 		.info_name = "EQE",
463 		.dump_fn = qm_eq_aeq_dump,
464 	}, {
465 		.cmd = "aeq",
466 		.info_name = "AEQE",
467 		.dump_fn = qm_eq_aeq_dump,
468 	},
469 };
470 
471 static int qm_cmd_write_dump(struct hisi_qm *qm, const char *cmd_buf)
472 {
473 	struct device *dev = &qm->pdev->dev;
474 	char *presult, *s, *s_tmp;
475 	int table_size, i, ret;
476 
477 	s = kstrdup(cmd_buf, GFP_KERNEL);
478 	if (!s)
479 		return -ENOMEM;
480 
481 	s_tmp = s;
482 	presult = strsep(&s, " ");
483 	if (!presult) {
484 		ret = -EINVAL;
485 		goto err_buffer_free;
486 	}
487 
488 	if (!strcmp(presult, "help")) {
489 		ret = qm_dbg_help(qm, s);
490 		goto err_buffer_free;
491 	}
492 
493 	table_size = ARRAY_SIZE(qm_cmd_dump_table);
494 	for (i = 0; i < table_size; i++) {
495 		if (!strcmp(presult, qm_cmd_dump_table[i].cmd)) {
496 			ret = qm_cmd_dump_table[i].dump_fn(qm, s,
497 				qm_cmd_dump_table[i].info_name);
498 			break;
499 		}
500 	}
501 
502 	if (i == table_size) {
503 		dev_info(dev, "Please echo help\n");
504 		ret = -EINVAL;
505 	}
506 
507 err_buffer_free:
508 	kfree(s_tmp);
509 
510 	return ret;
511 }
512 
513 static ssize_t qm_cmd_write(struct file *filp, const char __user *buffer,
514 			    size_t count, loff_t *pos)
515 {
516 	struct hisi_qm *qm = filp->private_data;
517 	char *cmd_buf, *cmd_buf_tmp;
518 	int ret;
519 
520 	if (*pos)
521 		return 0;
522 
523 	ret = hisi_qm_get_dfx_access(qm);
524 	if (ret)
525 		return ret;
526 
527 	/* Judge if the instance is being reset. */
528 	if (unlikely(atomic_read(&qm->status.flags) == QM_STOP)) {
529 		ret = 0;
530 		goto put_dfx_access;
531 	}
532 
533 	if (count > QM_DBG_WRITE_LEN) {
534 		ret = -ENOSPC;
535 		goto put_dfx_access;
536 	}
537 
538 	cmd_buf = memdup_user_nul(buffer, count);
539 	if (IS_ERR(cmd_buf)) {
540 		ret = PTR_ERR(cmd_buf);
541 		goto put_dfx_access;
542 	}
543 
544 	cmd_buf_tmp = strchr(cmd_buf, '\n');
545 	if (cmd_buf_tmp) {
546 		*cmd_buf_tmp = '\0';
547 		count = cmd_buf_tmp - cmd_buf + 1;
548 	}
549 
550 	ret = qm_cmd_write_dump(qm, cmd_buf);
551 	if (ret) {
552 		kfree(cmd_buf);
553 		goto put_dfx_access;
554 	}
555 
556 	kfree(cmd_buf);
557 
558 	ret = count;
559 
560 put_dfx_access:
561 	hisi_qm_put_dfx_access(qm);
562 	return ret;
563 }
564 
565 static const struct file_operations qm_cmd_fops = {
566 	.owner = THIS_MODULE,
567 	.open = simple_open,
568 	.read = qm_cmd_read,
569 	.write = qm_cmd_write,
570 };
571 
572 /**
573  * hisi_qm_regs_dump() - Dump registers's value.
574  * @s: debugfs file handle.
575  * @regset: accelerator registers information.
576  *
577  * Dump accelerator registers.
578  */
579 void hisi_qm_regs_dump(struct seq_file *s, struct debugfs_regset32 *regset)
580 {
581 	struct pci_dev *pdev = to_pci_dev(regset->dev);
582 	struct hisi_qm *qm = pci_get_drvdata(pdev);
583 	const struct debugfs_reg32 *regs = regset->regs;
584 	int regs_len = regset->nregs;
585 	int i, ret;
586 	u32 val;
587 
588 	ret = hisi_qm_get_dfx_access(qm);
589 	if (ret)
590 		return;
591 
592 	for (i = 0; i < regs_len; i++) {
593 		val = readl(regset->base + regs[i].offset);
594 		seq_printf(s, "%s= 0x%08x\n", regs[i].name, val);
595 	}
596 
597 	hisi_qm_put_dfx_access(qm);
598 }
599 EXPORT_SYMBOL_GPL(hisi_qm_regs_dump);
600 
601 static int qm_regs_show(struct seq_file *s, void *unused)
602 {
603 	struct hisi_qm *qm = s->private;
604 	struct debugfs_regset32 regset;
605 
606 	if (qm->fun_type == QM_HW_PF) {
607 		regset.regs = qm_dfx_regs;
608 		regset.nregs = ARRAY_SIZE(qm_dfx_regs);
609 	} else {
610 		regset.regs = qm_vf_dfx_regs;
611 		regset.nregs = ARRAY_SIZE(qm_vf_dfx_regs);
612 	}
613 
614 	regset.base = qm->io_base;
615 	regset.dev = &qm->pdev->dev;
616 
617 	hisi_qm_regs_dump(s, &regset);
618 
619 	return 0;
620 }
621 
622 DEFINE_SHOW_ATTRIBUTE(qm_regs);
623 
624 static u32 current_q_read(struct hisi_qm *qm)
625 {
626 	return readl(qm->io_base + QM_DFX_SQE_CNT_VF_SQN) >> QM_DFX_QN_SHIFT;
627 }
628 
629 static int current_q_write(struct hisi_qm *qm, u32 val)
630 {
631 	u32 tmp;
632 
633 	if (val >= qm->debug.curr_qm_qp_num)
634 		return -EINVAL;
635 
636 	tmp = val << QM_DFX_QN_SHIFT |
637 	      (readl(qm->io_base + QM_DFX_SQE_CNT_VF_SQN) & CURRENT_FUN_MASK);
638 	writel(tmp, qm->io_base + QM_DFX_SQE_CNT_VF_SQN);
639 
640 	tmp = val << QM_DFX_QN_SHIFT |
641 	      (readl(qm->io_base + QM_DFX_CQE_CNT_VF_CQN) & CURRENT_FUN_MASK);
642 	writel(tmp, qm->io_base + QM_DFX_CQE_CNT_VF_CQN);
643 
644 	return 0;
645 }
646 
647 static u32 clear_enable_read(struct hisi_qm *qm)
648 {
649 	return readl(qm->io_base + QM_DFX_CNT_CLR_CE);
650 }
651 
652 /* rd_clr_ctrl 1 enable read clear, otherwise 0 disable it */
653 static int clear_enable_write(struct hisi_qm *qm, u32 rd_clr_ctrl)
654 {
655 	if (rd_clr_ctrl > 1)
656 		return -EINVAL;
657 
658 	writel(rd_clr_ctrl, qm->io_base + QM_DFX_CNT_CLR_CE);
659 
660 	return 0;
661 }
662 
663 static u32 current_qm_read(struct hisi_qm *qm)
664 {
665 	return readl(qm->io_base + QM_DFX_MB_CNT_VF);
666 }
667 
668 static int qm_get_vf_qp_num(struct hisi_qm *qm, u32 fun_num)
669 {
670 	u32 remain_q_num, vfq_num;
671 	u32 num_vfs = qm->vfs_num;
672 
673 	vfq_num = (qm->ctrl_qp_num - qm->qp_num) / num_vfs;
674 	if (vfq_num >= qm->max_qp_num)
675 		return qm->max_qp_num;
676 
677 	remain_q_num = (qm->ctrl_qp_num - qm->qp_num) % num_vfs;
678 	if (vfq_num + remain_q_num <= qm->max_qp_num)
679 		return fun_num == num_vfs ? vfq_num + remain_q_num : vfq_num;
680 
681 	/*
682 	 * if vfq_num + remain_q_num > max_qp_num, the last VFs,
683 	 * each with one more queue.
684 	 */
685 	return fun_num + remain_q_num > num_vfs ? vfq_num + 1 : vfq_num;
686 }
687 
688 static int current_qm_write(struct hisi_qm *qm, u32 val)
689 {
690 	u32 tmp;
691 
692 	if (val > qm->vfs_num)
693 		return -EINVAL;
694 
695 	/* According PF or VF Dev ID to calculation curr_qm_qp_num and store */
696 	if (!val)
697 		qm->debug.curr_qm_qp_num = qm->qp_num;
698 	else
699 		qm->debug.curr_qm_qp_num = qm_get_vf_qp_num(qm, val);
700 
701 	writel(val, qm->io_base + QM_DFX_MB_CNT_VF);
702 	writel(val, qm->io_base + QM_DFX_DB_CNT_VF);
703 
704 	tmp = val |
705 	      (readl(qm->io_base + QM_DFX_SQE_CNT_VF_SQN) & CURRENT_Q_MASK);
706 	writel(tmp, qm->io_base + QM_DFX_SQE_CNT_VF_SQN);
707 
708 	tmp = val |
709 	      (readl(qm->io_base + QM_DFX_CQE_CNT_VF_CQN) & CURRENT_Q_MASK);
710 	writel(tmp, qm->io_base + QM_DFX_CQE_CNT_VF_CQN);
711 
712 	return 0;
713 }
714 
715 static ssize_t qm_debug_read(struct file *filp, char __user *buf,
716 			     size_t count, loff_t *pos)
717 {
718 	struct debugfs_file *file = filp->private_data;
719 	enum qm_debug_file index = file->index;
720 	struct hisi_qm *qm = file_to_qm(file);
721 	char tbuf[QM_DBG_TMP_BUF_LEN];
722 	u32 val;
723 	int ret;
724 
725 	ret = hisi_qm_get_dfx_access(qm);
726 	if (ret)
727 		return ret;
728 
729 	mutex_lock(&file->lock);
730 	switch (index) {
731 	case CURRENT_QM:
732 		val = current_qm_read(qm);
733 		break;
734 	case CURRENT_Q:
735 		val = current_q_read(qm);
736 		break;
737 	case CLEAR_ENABLE:
738 		val = clear_enable_read(qm);
739 		break;
740 	default:
741 		goto err_input;
742 	}
743 	mutex_unlock(&file->lock);
744 
745 	hisi_qm_put_dfx_access(qm);
746 	ret = scnprintf(tbuf, QM_DBG_TMP_BUF_LEN, "%u\n", val);
747 	return simple_read_from_buffer(buf, count, pos, tbuf, ret);
748 
749 err_input:
750 	mutex_unlock(&file->lock);
751 	hisi_qm_put_dfx_access(qm);
752 	return -EINVAL;
753 }
754 
755 static ssize_t qm_debug_write(struct file *filp, const char __user *buf,
756 			      size_t count, loff_t *pos)
757 {
758 	struct debugfs_file *file = filp->private_data;
759 	enum qm_debug_file index = file->index;
760 	struct hisi_qm *qm = file_to_qm(file);
761 	unsigned long val;
762 	char tbuf[QM_DBG_TMP_BUF_LEN];
763 	int len, ret;
764 
765 	if (*pos != 0)
766 		return 0;
767 
768 	if (count >= QM_DBG_TMP_BUF_LEN)
769 		return -ENOSPC;
770 
771 	len = simple_write_to_buffer(tbuf, QM_DBG_TMP_BUF_LEN - 1, pos, buf,
772 				     count);
773 	if (len < 0)
774 		return len;
775 
776 	tbuf[len] = '\0';
777 	if (kstrtoul(tbuf, 0, &val))
778 		return -EFAULT;
779 
780 	ret = hisi_qm_get_dfx_access(qm);
781 	if (ret)
782 		return ret;
783 
784 	mutex_lock(&file->lock);
785 	switch (index) {
786 	case CURRENT_QM:
787 		ret = current_qm_write(qm, val);
788 		break;
789 	case CURRENT_Q:
790 		ret = current_q_write(qm, val);
791 		break;
792 	case CLEAR_ENABLE:
793 		ret = clear_enable_write(qm, val);
794 		break;
795 	default:
796 		ret = -EINVAL;
797 	}
798 	mutex_unlock(&file->lock);
799 
800 	hisi_qm_put_dfx_access(qm);
801 
802 	if (ret)
803 		return ret;
804 
805 	return count;
806 }
807 
808 static const struct file_operations qm_debug_fops = {
809 	.owner = THIS_MODULE,
810 	.open = simple_open,
811 	.read = qm_debug_read,
812 	.write = qm_debug_write,
813 };
814 
815 static void dfx_regs_uninit(struct hisi_qm *qm,
816 		struct dfx_diff_registers *dregs, int reg_len)
817 {
818 	int i;
819 
820 	if (!dregs)
821 		return;
822 
823 	/* Setting the pointer is NULL to prevent double free */
824 	for (i = 0; i < reg_len; i++) {
825 		if (!dregs[i].regs)
826 			continue;
827 
828 		kfree(dregs[i].regs);
829 		dregs[i].regs = NULL;
830 	}
831 	kfree(dregs);
832 }
833 
834 static struct dfx_diff_registers *dfx_regs_init(struct hisi_qm *qm,
835 	const struct dfx_diff_registers *cregs, u32 reg_len)
836 {
837 	struct dfx_diff_registers *diff_regs;
838 	u32 j, base_offset;
839 	int i;
840 
841 	diff_regs = kcalloc(reg_len, sizeof(*diff_regs), GFP_KERNEL);
842 	if (!diff_regs)
843 		return ERR_PTR(-ENOMEM);
844 
845 	for (i = 0; i < reg_len; i++) {
846 		if (!cregs[i].reg_len)
847 			continue;
848 
849 		diff_regs[i].reg_offset = cregs[i].reg_offset;
850 		diff_regs[i].reg_len = cregs[i].reg_len;
851 		diff_regs[i].regs = kcalloc(QM_DFX_REGS_LEN, cregs[i].reg_len,
852 					 GFP_KERNEL);
853 		if (!diff_regs[i].regs)
854 			goto alloc_error;
855 
856 		for (j = 0; j < diff_regs[i].reg_len; j++) {
857 			base_offset = diff_regs[i].reg_offset +
858 					j * QM_DFX_REGS_LEN;
859 			diff_regs[i].regs[j] = readl(qm->io_base + base_offset);
860 		}
861 	}
862 
863 	return diff_regs;
864 
865 alloc_error:
866 	while (i > 0) {
867 		i--;
868 		kfree(diff_regs[i].regs);
869 	}
870 	kfree(diff_regs);
871 	return ERR_PTR(-ENOMEM);
872 }
873 
874 static int qm_diff_regs_init(struct hisi_qm *qm,
875 		struct dfx_diff_registers *dregs, u32 reg_len)
876 {
877 	int ret;
878 
879 	qm->debug.qm_diff_regs = dfx_regs_init(qm, qm_diff_regs, ARRAY_SIZE(qm_diff_regs));
880 	if (IS_ERR(qm->debug.qm_diff_regs)) {
881 		ret = PTR_ERR(qm->debug.qm_diff_regs);
882 		qm->debug.qm_diff_regs = NULL;
883 		return ret;
884 	}
885 
886 	qm->debug.acc_diff_regs = dfx_regs_init(qm, dregs, reg_len);
887 	if (IS_ERR(qm->debug.acc_diff_regs)) {
888 		dfx_regs_uninit(qm, qm->debug.qm_diff_regs, ARRAY_SIZE(qm_diff_regs));
889 		ret = PTR_ERR(qm->debug.acc_diff_regs);
890 		qm->debug.acc_diff_regs = NULL;
891 		return ret;
892 	}
893 
894 	return 0;
895 }
896 
897 static void qm_last_regs_uninit(struct hisi_qm *qm)
898 {
899 	struct qm_debug *debug = &qm->debug;
900 
901 	if (qm->fun_type == QM_HW_VF || !debug->qm_last_words)
902 		return;
903 
904 	kfree(debug->qm_last_words);
905 	debug->qm_last_words = NULL;
906 }
907 
908 static int qm_last_regs_init(struct hisi_qm *qm)
909 {
910 	int dfx_regs_num = ARRAY_SIZE(qm_dfx_regs);
911 	struct qm_debug *debug = &qm->debug;
912 	int i;
913 
914 	if (qm->fun_type == QM_HW_VF)
915 		return 0;
916 
917 	debug->qm_last_words = kcalloc(dfx_regs_num, sizeof(unsigned int), GFP_KERNEL);
918 	if (!debug->qm_last_words)
919 		return -ENOMEM;
920 
921 	for (i = 0; i < dfx_regs_num; i++) {
922 		debug->qm_last_words[i] = readl_relaxed(qm->io_base +
923 			qm_dfx_regs[i].offset);
924 	}
925 
926 	return 0;
927 }
928 
929 static void qm_diff_regs_uninit(struct hisi_qm *qm, u32 reg_len)
930 {
931 	dfx_regs_uninit(qm, qm->debug.acc_diff_regs, reg_len);
932 	qm->debug.acc_diff_regs = NULL;
933 	dfx_regs_uninit(qm, qm->debug.qm_diff_regs, ARRAY_SIZE(qm_diff_regs));
934 	qm->debug.qm_diff_regs = NULL;
935 }
936 
937 /**
938  * hisi_qm_regs_debugfs_init() - Allocate memory for registers.
939  * @qm: device qm handle.
940  * @dregs: diff registers handle.
941  * @reg_len: diff registers region length.
942  */
943 int hisi_qm_regs_debugfs_init(struct hisi_qm *qm,
944 		struct dfx_diff_registers *dregs, u32 reg_len)
945 {
946 	int ret;
947 
948 	if (!qm || !dregs)
949 		return -EINVAL;
950 
951 	if (qm->fun_type != QM_HW_PF)
952 		return 0;
953 
954 	ret = qm_last_regs_init(qm);
955 	if (ret) {
956 		dev_info(&qm->pdev->dev, "failed to init qm words memory!\n");
957 		return ret;
958 	}
959 
960 	ret = qm_diff_regs_init(qm, dregs, reg_len);
961 	if (ret) {
962 		qm_last_regs_uninit(qm);
963 		return ret;
964 	}
965 
966 	return 0;
967 }
968 EXPORT_SYMBOL_GPL(hisi_qm_regs_debugfs_init);
969 
970 /**
971  * hisi_qm_regs_debugfs_uninit() - Free memory for registers.
972  * @qm: device qm handle.
973  * @reg_len: diff registers region length.
974  */
975 void hisi_qm_regs_debugfs_uninit(struct hisi_qm *qm, u32 reg_len)
976 {
977 	if (!qm || qm->fun_type != QM_HW_PF)
978 		return;
979 
980 	qm_diff_regs_uninit(qm, reg_len);
981 	qm_last_regs_uninit(qm);
982 }
983 EXPORT_SYMBOL_GPL(hisi_qm_regs_debugfs_uninit);
984 
985 /**
986  * hisi_qm_acc_diff_regs_dump() - Dump registers's value.
987  * @qm: device qm handle.
988  * @s: Debugfs file handle.
989  * @dregs: diff registers handle.
990  * @regs_len: diff registers region length.
991  */
992 void hisi_qm_acc_diff_regs_dump(struct hisi_qm *qm, struct seq_file *s,
993 	struct dfx_diff_registers *dregs, u32 regs_len)
994 {
995 	u32 j, val, base_offset;
996 	int i, ret;
997 
998 	if (!qm || !s || !dregs)
999 		return;
1000 
1001 	ret = hisi_qm_get_dfx_access(qm);
1002 	if (ret)
1003 		return;
1004 
1005 	down_read(&qm->qps_lock);
1006 	for (i = 0; i < regs_len; i++) {
1007 		if (!dregs[i].reg_len)
1008 			continue;
1009 
1010 		for (j = 0; j < dregs[i].reg_len; j++) {
1011 			base_offset = dregs[i].reg_offset + j * QM_DFX_REGS_LEN;
1012 			val = readl(qm->io_base + base_offset);
1013 			if (val != dregs[i].regs[j])
1014 				seq_printf(s, "0x%08x = 0x%08x ---> 0x%08x\n",
1015 					   base_offset, dregs[i].regs[j], val);
1016 		}
1017 	}
1018 	up_read(&qm->qps_lock);
1019 
1020 	hisi_qm_put_dfx_access(qm);
1021 }
1022 EXPORT_SYMBOL_GPL(hisi_qm_acc_diff_regs_dump);
1023 
1024 void hisi_qm_show_last_dfx_regs(struct hisi_qm *qm)
1025 {
1026 	struct qm_debug *debug = &qm->debug;
1027 	struct pci_dev *pdev = qm->pdev;
1028 	u32 val;
1029 	int i;
1030 
1031 	if (qm->fun_type == QM_HW_VF || !debug->qm_last_words)
1032 		return;
1033 
1034 	for (i = 0; i < ARRAY_SIZE(qm_dfx_regs); i++) {
1035 		val = readl_relaxed(qm->io_base + qm_dfx_regs[i].offset);
1036 		if (debug->qm_last_words[i] != val)
1037 			pci_info(pdev, "%s \t= 0x%08x => 0x%08x\n",
1038 			qm_dfx_regs[i].name, debug->qm_last_words[i], val);
1039 	}
1040 }
1041 
1042 static int qm_diff_regs_show(struct seq_file *s, void *unused)
1043 {
1044 	struct hisi_qm *qm = s->private;
1045 
1046 	hisi_qm_acc_diff_regs_dump(qm, s, qm->debug.qm_diff_regs,
1047 					ARRAY_SIZE(qm_diff_regs));
1048 
1049 	return 0;
1050 }
1051 DEFINE_SHOW_ATTRIBUTE(qm_diff_regs);
1052 
1053 static int qm_state_show(struct seq_file *s, void *unused)
1054 {
1055 	struct hisi_qm *qm = s->private;
1056 	u32 val;
1057 	int ret;
1058 
1059 	/* If device is in suspended, directly return the idle state. */
1060 	ret = hisi_qm_get_dfx_access(qm);
1061 	if (!ret) {
1062 		val = readl(qm->io_base + QM_IN_IDLE_ST_REG);
1063 		hisi_qm_put_dfx_access(qm);
1064 	} else if (ret == -EAGAIN) {
1065 		val = QM_IN_IDLE_STATE;
1066 	} else {
1067 		return ret;
1068 	}
1069 
1070 	seq_printf(s, "%u\n", val);
1071 
1072 	return 0;
1073 }
1074 
1075 DEFINE_SHOW_ATTRIBUTE(qm_state);
1076 
1077 static ssize_t qm_status_read(struct file *filp, char __user *buffer,
1078 			      size_t count, loff_t *pos)
1079 {
1080 	struct hisi_qm *qm = filp->private_data;
1081 	char buf[QM_DBG_READ_LEN];
1082 	int val, len;
1083 
1084 	val = atomic_read(&qm->status.flags);
1085 	len = scnprintf(buf, QM_DBG_READ_LEN, "%s\n", qm_s[val]);
1086 
1087 	return simple_read_from_buffer(buffer, count, pos, buf, len);
1088 }
1089 
1090 static const struct file_operations qm_status_fops = {
1091 	.owner = THIS_MODULE,
1092 	.open = simple_open,
1093 	.read = qm_status_read,
1094 };
1095 
1096 static void qm_create_debugfs_file(struct hisi_qm *qm, struct dentry *dir,
1097 				   enum qm_debug_file index)
1098 {
1099 	struct debugfs_file *file = qm->debug.files + index;
1100 
1101 	file->index = index;
1102 	mutex_init(&file->lock);
1103 	file->debug = &qm->debug;
1104 
1105 	debugfs_create_file(qm_debug_file_name[index], 0600, dir, file,
1106 			    &qm_debug_fops);
1107 }
1108 
1109 static int qm_debugfs_atomic64_set(void *data, u64 val)
1110 {
1111 	if (val)
1112 		return -EINVAL;
1113 
1114 	atomic64_set((atomic64_t *)data, 0);
1115 
1116 	return 0;
1117 }
1118 
1119 static int qm_debugfs_atomic64_get(void *data, u64 *val)
1120 {
1121 	*val = atomic64_read((atomic64_t *)data);
1122 
1123 	return 0;
1124 }
1125 
1126 DEFINE_DEBUGFS_ATTRIBUTE(qm_atomic64_ops, qm_debugfs_atomic64_get,
1127 			 qm_debugfs_atomic64_set, "%llu\n");
1128 
1129 /**
1130  * hisi_qm_debug_init() - Initialize qm related debugfs files.
1131  * @qm: The qm for which we want to add debugfs files.
1132  *
1133  * Create qm related debugfs files.
1134  */
1135 void hisi_qm_debug_init(struct hisi_qm *qm)
1136 {
1137 	struct dfx_diff_registers *qm_regs = qm->debug.qm_diff_regs;
1138 	struct qm_dev_dfx *dev_dfx = &qm->debug.dev_dfx;
1139 	struct qm_dfx *dfx = &qm->debug.dfx;
1140 	struct dentry *qm_d;
1141 	void *data;
1142 	int i;
1143 
1144 	qm_d = debugfs_create_dir("qm", qm->debug.debug_root);
1145 	qm->debug.qm_d = qm_d;
1146 
1147 	/* only show this in PF */
1148 	if (qm->fun_type == QM_HW_PF) {
1149 		debugfs_create_file("qm_state", 0444, qm->debug.qm_d,
1150 					qm, &qm_state_fops);
1151 
1152 		qm_create_debugfs_file(qm, qm->debug.debug_root, CURRENT_QM);
1153 		for (i = CURRENT_Q; i < DEBUG_FILE_NUM; i++)
1154 			qm_create_debugfs_file(qm, qm->debug.qm_d, i);
1155 	}
1156 
1157 	if (qm_regs)
1158 		debugfs_create_file("diff_regs", 0444, qm->debug.qm_d,
1159 					qm, &qm_diff_regs_fops);
1160 
1161 	debugfs_create_file("regs", 0444, qm->debug.qm_d, qm, &qm_regs_fops);
1162 
1163 	debugfs_create_file("cmd", 0600, qm->debug.qm_d, qm, &qm_cmd_fops);
1164 
1165 	debugfs_create_file("status", 0444, qm->debug.qm_d, qm,
1166 			&qm_status_fops);
1167 
1168 	debugfs_create_u32("dev_state", 0444, qm->debug.qm_d, &dev_dfx->dev_state);
1169 	debugfs_create_u32("dev_timeout", 0644, qm->debug.qm_d, &dev_dfx->dev_timeout);
1170 
1171 	for (i = 0; i < ARRAY_SIZE(qm_dfx_files); i++) {
1172 		data = (atomic64_t *)((uintptr_t)dfx + qm_dfx_files[i].offset);
1173 		debugfs_create_file(qm_dfx_files[i].name,
1174 			0644,
1175 			qm_d,
1176 			data,
1177 			&qm_atomic64_ops);
1178 	}
1179 
1180 	if (test_bit(QM_SUPPORT_FUNC_QOS, &qm->caps))
1181 		hisi_qm_set_algqos_init(qm);
1182 }
1183 EXPORT_SYMBOL_GPL(hisi_qm_debug_init);
1184 
1185 /**
1186  * hisi_qm_debug_regs_clear() - clear qm debug related registers.
1187  * @qm: The qm for which we want to clear its debug registers.
1188  */
1189 void hisi_qm_debug_regs_clear(struct hisi_qm *qm)
1190 {
1191 	const struct debugfs_reg32 *regs;
1192 	int i;
1193 
1194 	/* clear current_qm */
1195 	writel(0x0, qm->io_base + QM_DFX_MB_CNT_VF);
1196 	writel(0x0, qm->io_base + QM_DFX_DB_CNT_VF);
1197 
1198 	/* clear current_q */
1199 	writel(0x0, qm->io_base + QM_DFX_SQE_CNT_VF_SQN);
1200 	writel(0x0, qm->io_base + QM_DFX_CQE_CNT_VF_CQN);
1201 
1202 	/*
1203 	 * these registers are reading and clearing, so clear them after
1204 	 * reading them.
1205 	 */
1206 	writel(0x1, qm->io_base + QM_DFX_CNT_CLR_CE);
1207 
1208 	regs = qm_dfx_regs;
1209 	for (i = 0; i < CNT_CYC_REGS_NUM; i++) {
1210 		readl(qm->io_base + regs->offset);
1211 		regs++;
1212 	}
1213 
1214 	/* clear clear_enable */
1215 	writel(0x0, qm->io_base + QM_DFX_CNT_CLR_CE);
1216 }
1217 EXPORT_SYMBOL_GPL(hisi_qm_debug_regs_clear);
1218