xref: /OK3568_Linux_fs/kernel/drivers/crypto/hisilicon/sec2/sec_main.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /* Copyright (c) 2019 HiSilicon Limited. */
3*4882a593Smuzhiyun 
4*4882a593Smuzhiyun #include <linux/acpi.h>
5*4882a593Smuzhiyun #include <linux/aer.h>
6*4882a593Smuzhiyun #include <linux/bitops.h>
7*4882a593Smuzhiyun #include <linux/debugfs.h>
8*4882a593Smuzhiyun #include <linux/init.h>
9*4882a593Smuzhiyun #include <linux/io.h>
10*4882a593Smuzhiyun #include <linux/iommu.h>
11*4882a593Smuzhiyun #include <linux/kernel.h>
12*4882a593Smuzhiyun #include <linux/module.h>
13*4882a593Smuzhiyun #include <linux/pci.h>
14*4882a593Smuzhiyun #include <linux/seq_file.h>
15*4882a593Smuzhiyun #include <linux/topology.h>
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun #include "sec.h"
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun #define SEC_VF_NUM			63
20*4882a593Smuzhiyun #define SEC_QUEUE_NUM_V1		4096
21*4882a593Smuzhiyun #define SEC_QUEUE_NUM_V2		1024
22*4882a593Smuzhiyun #define SEC_PF_PCI_DEVICE_ID		0xa255
23*4882a593Smuzhiyun #define SEC_VF_PCI_DEVICE_ID		0xa256
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun #define SEC_BD_ERR_CHK_EN0		0xEFFFFFFF
26*4882a593Smuzhiyun #define SEC_BD_ERR_CHK_EN1		0x7ffff7fd
27*4882a593Smuzhiyun #define SEC_BD_ERR_CHK_EN3		0xffffbfff
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun #define SEC_SQE_SIZE			128
30*4882a593Smuzhiyun #define SEC_SQ_SIZE			(SEC_SQE_SIZE * QM_Q_DEPTH)
31*4882a593Smuzhiyun #define SEC_PF_DEF_Q_NUM		256
32*4882a593Smuzhiyun #define SEC_PF_DEF_Q_BASE		0
33*4882a593Smuzhiyun #define SEC_CTX_Q_NUM_DEF		2
34*4882a593Smuzhiyun #define SEC_CTX_Q_NUM_MAX		32
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun #define SEC_CTRL_CNT_CLR_CE		0x301120
37*4882a593Smuzhiyun #define SEC_CTRL_CNT_CLR_CE_BIT		BIT(0)
38*4882a593Smuzhiyun #define SEC_ENGINE_PF_CFG_OFF		0x300000
39*4882a593Smuzhiyun #define SEC_ACC_COMMON_REG_OFF		0x1000
40*4882a593Smuzhiyun #define SEC_CORE_INT_SOURCE		0x301010
41*4882a593Smuzhiyun #define SEC_CORE_INT_MASK		0x301000
42*4882a593Smuzhiyun #define SEC_CORE_INT_STATUS		0x301008
43*4882a593Smuzhiyun #define SEC_CORE_SRAM_ECC_ERR_INFO	0x301C14
44*4882a593Smuzhiyun #define SEC_ECC_NUM(err)			(((err) >> 16) & 0xFF)
45*4882a593Smuzhiyun #define SEC_ECC_ADDR(err)			((err) >> 0)
46*4882a593Smuzhiyun #define SEC_CORE_INT_DISABLE		0x0
47*4882a593Smuzhiyun #define SEC_CORE_INT_ENABLE		0x1ff
48*4882a593Smuzhiyun #define SEC_CORE_INT_CLEAR		0x1ff
49*4882a593Smuzhiyun #define SEC_SAA_ENABLE			0x17f
50*4882a593Smuzhiyun 
51*4882a593Smuzhiyun #define SEC_RAS_CE_REG			0x301050
52*4882a593Smuzhiyun #define SEC_RAS_FE_REG			0x301054
53*4882a593Smuzhiyun #define SEC_RAS_NFE_REG			0x301058
54*4882a593Smuzhiyun #define SEC_RAS_CE_ENB_MSK		0x88
55*4882a593Smuzhiyun #define SEC_RAS_FE_ENB_MSK		0x0
56*4882a593Smuzhiyun #define SEC_RAS_NFE_ENB_MSK		0x177
57*4882a593Smuzhiyun #define SEC_RAS_DISABLE			0x0
58*4882a593Smuzhiyun #define SEC_MEM_START_INIT_REG		0x0100
59*4882a593Smuzhiyun #define SEC_MEM_INIT_DONE_REG		0x0104
60*4882a593Smuzhiyun 
61*4882a593Smuzhiyun #define SEC_CONTROL_REG			0x0200
62*4882a593Smuzhiyun #define SEC_TRNG_EN_SHIFT		8
63*4882a593Smuzhiyun #define SEC_CLK_GATE_ENABLE		BIT(3)
64*4882a593Smuzhiyun #define SEC_CLK_GATE_DISABLE		(~BIT(3))
65*4882a593Smuzhiyun #define SEC_AXI_SHUTDOWN_ENABLE	BIT(12)
66*4882a593Smuzhiyun #define SEC_AXI_SHUTDOWN_DISABLE	0xFFFFEFFF
67*4882a593Smuzhiyun 
68*4882a593Smuzhiyun #define SEC_INTERFACE_USER_CTRL0_REG	0x0220
69*4882a593Smuzhiyun #define SEC_INTERFACE_USER_CTRL1_REG	0x0224
70*4882a593Smuzhiyun #define SEC_SAA_EN_REG					0x0270
71*4882a593Smuzhiyun #define SEC_BD_ERR_CHK_EN_REG0		0x0380
72*4882a593Smuzhiyun #define SEC_BD_ERR_CHK_EN_REG1		0x0384
73*4882a593Smuzhiyun #define SEC_BD_ERR_CHK_EN_REG3		0x038c
74*4882a593Smuzhiyun 
75*4882a593Smuzhiyun #define SEC_USER0_SMMU_NORMAL		(BIT(23) | BIT(15))
76*4882a593Smuzhiyun #define SEC_USER1_SMMU_NORMAL		(BIT(31) | BIT(23) | BIT(15) | BIT(7))
77*4882a593Smuzhiyun #define SEC_CORE_INT_STATUS_M_ECC	BIT(2)
78*4882a593Smuzhiyun 
79*4882a593Smuzhiyun #define SEC_DELAY_10_US			10
80*4882a593Smuzhiyun #define SEC_POLL_TIMEOUT_US		1000
81*4882a593Smuzhiyun #define SEC_DBGFS_VAL_MAX_LEN		20
82*4882a593Smuzhiyun #define SEC_SINGLE_PORT_MAX_TRANS	0x2060
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun #define SEC_SQE_MASK_OFFSET		64
85*4882a593Smuzhiyun #define SEC_SQE_MASK_LEN		48
86*4882a593Smuzhiyun 
87*4882a593Smuzhiyun #define SEC_ADDR(qm, offset) ((qm)->io_base + (offset) + \
88*4882a593Smuzhiyun 			     SEC_ENGINE_PF_CFG_OFF + SEC_ACC_COMMON_REG_OFF)
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun struct sec_hw_error {
91*4882a593Smuzhiyun 	u32 int_msk;
92*4882a593Smuzhiyun 	const char *msg;
93*4882a593Smuzhiyun };
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun struct sec_dfx_item {
96*4882a593Smuzhiyun 	const char *name;
97*4882a593Smuzhiyun 	u32 offset;
98*4882a593Smuzhiyun };
99*4882a593Smuzhiyun 
100*4882a593Smuzhiyun static const char sec_name[] = "hisi_sec2";
101*4882a593Smuzhiyun static struct dentry *sec_debugfs_root;
102*4882a593Smuzhiyun 
103*4882a593Smuzhiyun static struct hisi_qm_list sec_devices = {
104*4882a593Smuzhiyun 	.register_to_crypto	= sec_register_to_crypto,
105*4882a593Smuzhiyun 	.unregister_from_crypto	= sec_unregister_from_crypto,
106*4882a593Smuzhiyun };
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun static const struct sec_hw_error sec_hw_errors[] = {
109*4882a593Smuzhiyun 	{.int_msk = BIT(0), .msg = "sec_axi_rresp_err_rint"},
110*4882a593Smuzhiyun 	{.int_msk = BIT(1), .msg = "sec_axi_bresp_err_rint"},
111*4882a593Smuzhiyun 	{.int_msk = BIT(2), .msg = "sec_ecc_2bit_err_rint"},
112*4882a593Smuzhiyun 	{.int_msk = BIT(3), .msg = "sec_ecc_1bit_err_rint"},
113*4882a593Smuzhiyun 	{.int_msk = BIT(4), .msg = "sec_req_trng_timeout_rint"},
114*4882a593Smuzhiyun 	{.int_msk = BIT(5), .msg = "sec_fsm_hbeat_rint"},
115*4882a593Smuzhiyun 	{.int_msk = BIT(6), .msg = "sec_channel_req_rng_timeout_rint"},
116*4882a593Smuzhiyun 	{.int_msk = BIT(7), .msg = "sec_bd_err_rint"},
117*4882a593Smuzhiyun 	{.int_msk = BIT(8), .msg = "sec_chain_buff_err_rint"},
118*4882a593Smuzhiyun 	{ /* sentinel */ }
119*4882a593Smuzhiyun };
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun static const char * const sec_dbg_file_name[] = {
122*4882a593Smuzhiyun 	[SEC_CURRENT_QM] = "current_qm",
123*4882a593Smuzhiyun 	[SEC_CLEAR_ENABLE] = "clear_enable",
124*4882a593Smuzhiyun };
125*4882a593Smuzhiyun 
126*4882a593Smuzhiyun static struct sec_dfx_item sec_dfx_labels[] = {
127*4882a593Smuzhiyun 	{"send_cnt", offsetof(struct sec_dfx, send_cnt)},
128*4882a593Smuzhiyun 	{"recv_cnt", offsetof(struct sec_dfx, recv_cnt)},
129*4882a593Smuzhiyun 	{"send_busy_cnt", offsetof(struct sec_dfx, send_busy_cnt)},
130*4882a593Smuzhiyun 	{"recv_busy_cnt", offsetof(struct sec_dfx, recv_busy_cnt)},
131*4882a593Smuzhiyun 	{"err_bd_cnt", offsetof(struct sec_dfx, err_bd_cnt)},
132*4882a593Smuzhiyun 	{"invalid_req_cnt", offsetof(struct sec_dfx, invalid_req_cnt)},
133*4882a593Smuzhiyun 	{"done_flag_cnt", offsetof(struct sec_dfx, done_flag_cnt)},
134*4882a593Smuzhiyun };
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun static const struct debugfs_reg32 sec_dfx_regs[] = {
137*4882a593Smuzhiyun 	{"SEC_PF_ABNORMAL_INT_SOURCE    ",  0x301010},
138*4882a593Smuzhiyun 	{"SEC_SAA_EN                    ",  0x301270},
139*4882a593Smuzhiyun 	{"SEC_BD_LATENCY_MIN            ",  0x301600},
140*4882a593Smuzhiyun 	{"SEC_BD_LATENCY_MAX            ",  0x301608},
141*4882a593Smuzhiyun 	{"SEC_BD_LATENCY_AVG            ",  0x30160C},
142*4882a593Smuzhiyun 	{"SEC_BD_NUM_IN_SAA0            ",  0x301670},
143*4882a593Smuzhiyun 	{"SEC_BD_NUM_IN_SAA1            ",  0x301674},
144*4882a593Smuzhiyun 	{"SEC_BD_NUM_IN_SEC             ",  0x301680},
145*4882a593Smuzhiyun 	{"SEC_ECC_1BIT_CNT              ",  0x301C00},
146*4882a593Smuzhiyun 	{"SEC_ECC_1BIT_INFO             ",  0x301C04},
147*4882a593Smuzhiyun 	{"SEC_ECC_2BIT_CNT              ",  0x301C10},
148*4882a593Smuzhiyun 	{"SEC_ECC_2BIT_INFO             ",  0x301C14},
149*4882a593Smuzhiyun 	{"SEC_BD_SAA0                   ",  0x301C20},
150*4882a593Smuzhiyun 	{"SEC_BD_SAA1                   ",  0x301C24},
151*4882a593Smuzhiyun 	{"SEC_BD_SAA2                   ",  0x301C28},
152*4882a593Smuzhiyun 	{"SEC_BD_SAA3                   ",  0x301C2C},
153*4882a593Smuzhiyun 	{"SEC_BD_SAA4                   ",  0x301C30},
154*4882a593Smuzhiyun 	{"SEC_BD_SAA5                   ",  0x301C34},
155*4882a593Smuzhiyun 	{"SEC_BD_SAA6                   ",  0x301C38},
156*4882a593Smuzhiyun 	{"SEC_BD_SAA7                   ",  0x301C3C},
157*4882a593Smuzhiyun 	{"SEC_BD_SAA8                   ",  0x301C40},
158*4882a593Smuzhiyun };
159*4882a593Smuzhiyun 
sec_pf_q_num_set(const char * val,const struct kernel_param * kp)160*4882a593Smuzhiyun static int sec_pf_q_num_set(const char *val, const struct kernel_param *kp)
161*4882a593Smuzhiyun {
162*4882a593Smuzhiyun 	return q_num_set(val, kp, SEC_PF_PCI_DEVICE_ID);
163*4882a593Smuzhiyun }
164*4882a593Smuzhiyun 
165*4882a593Smuzhiyun static const struct kernel_param_ops sec_pf_q_num_ops = {
166*4882a593Smuzhiyun 	.set = sec_pf_q_num_set,
167*4882a593Smuzhiyun 	.get = param_get_int,
168*4882a593Smuzhiyun };
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun static u32 pf_q_num = SEC_PF_DEF_Q_NUM;
171*4882a593Smuzhiyun module_param_cb(pf_q_num, &sec_pf_q_num_ops, &pf_q_num, 0444);
172*4882a593Smuzhiyun MODULE_PARM_DESC(pf_q_num, "Number of queues in PF(v1 2-4096, v2 2-1024)");
173*4882a593Smuzhiyun 
sec_ctx_q_num_set(const char * val,const struct kernel_param * kp)174*4882a593Smuzhiyun static int sec_ctx_q_num_set(const char *val, const struct kernel_param *kp)
175*4882a593Smuzhiyun {
176*4882a593Smuzhiyun 	u32 ctx_q_num;
177*4882a593Smuzhiyun 	int ret;
178*4882a593Smuzhiyun 
179*4882a593Smuzhiyun 	if (!val)
180*4882a593Smuzhiyun 		return -EINVAL;
181*4882a593Smuzhiyun 
182*4882a593Smuzhiyun 	ret = kstrtou32(val, 10, &ctx_q_num);
183*4882a593Smuzhiyun 	if (ret)
184*4882a593Smuzhiyun 		return -EINVAL;
185*4882a593Smuzhiyun 
186*4882a593Smuzhiyun 	if (!ctx_q_num || ctx_q_num > SEC_CTX_Q_NUM_MAX || ctx_q_num & 0x1) {
187*4882a593Smuzhiyun 		pr_err("ctx queue num[%u] is invalid!\n", ctx_q_num);
188*4882a593Smuzhiyun 		return -EINVAL;
189*4882a593Smuzhiyun 	}
190*4882a593Smuzhiyun 
191*4882a593Smuzhiyun 	return param_set_int(val, kp);
192*4882a593Smuzhiyun }
193*4882a593Smuzhiyun 
194*4882a593Smuzhiyun static const struct kernel_param_ops sec_ctx_q_num_ops = {
195*4882a593Smuzhiyun 	.set = sec_ctx_q_num_set,
196*4882a593Smuzhiyun 	.get = param_get_int,
197*4882a593Smuzhiyun };
198*4882a593Smuzhiyun static u32 ctx_q_num = SEC_CTX_Q_NUM_DEF;
199*4882a593Smuzhiyun module_param_cb(ctx_q_num, &sec_ctx_q_num_ops, &ctx_q_num, 0444);
200*4882a593Smuzhiyun MODULE_PARM_DESC(ctx_q_num, "Queue num in ctx (2 default, 2, 4, ..., 32)");
201*4882a593Smuzhiyun 
202*4882a593Smuzhiyun static const struct kernel_param_ops vfs_num_ops = {
203*4882a593Smuzhiyun 	.set = vfs_num_set,
204*4882a593Smuzhiyun 	.get = param_get_int,
205*4882a593Smuzhiyun };
206*4882a593Smuzhiyun 
207*4882a593Smuzhiyun static u32 vfs_num;
208*4882a593Smuzhiyun module_param_cb(vfs_num, &vfs_num_ops, &vfs_num, 0444);
209*4882a593Smuzhiyun MODULE_PARM_DESC(vfs_num, "Number of VFs to enable(1-63), 0(default)");
210*4882a593Smuzhiyun 
sec_destroy_qps(struct hisi_qp ** qps,int qp_num)211*4882a593Smuzhiyun void sec_destroy_qps(struct hisi_qp **qps, int qp_num)
212*4882a593Smuzhiyun {
213*4882a593Smuzhiyun 	hisi_qm_free_qps(qps, qp_num);
214*4882a593Smuzhiyun 	kfree(qps);
215*4882a593Smuzhiyun }
216*4882a593Smuzhiyun 
sec_create_qps(void)217*4882a593Smuzhiyun struct hisi_qp **sec_create_qps(void)
218*4882a593Smuzhiyun {
219*4882a593Smuzhiyun 	int node = cpu_to_node(smp_processor_id());
220*4882a593Smuzhiyun 	u32 ctx_num = ctx_q_num;
221*4882a593Smuzhiyun 	struct hisi_qp **qps;
222*4882a593Smuzhiyun 	int ret;
223*4882a593Smuzhiyun 
224*4882a593Smuzhiyun 	qps = kcalloc(ctx_num, sizeof(struct hisi_qp *), GFP_KERNEL);
225*4882a593Smuzhiyun 	if (!qps)
226*4882a593Smuzhiyun 		return NULL;
227*4882a593Smuzhiyun 
228*4882a593Smuzhiyun 	ret = hisi_qm_alloc_qps_node(&sec_devices, ctx_num, 0, node, qps);
229*4882a593Smuzhiyun 	if (!ret)
230*4882a593Smuzhiyun 		return qps;
231*4882a593Smuzhiyun 
232*4882a593Smuzhiyun 	kfree(qps);
233*4882a593Smuzhiyun 	return NULL;
234*4882a593Smuzhiyun }
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun 
237*4882a593Smuzhiyun static const struct pci_device_id sec_dev_ids[] = {
238*4882a593Smuzhiyun 	{ PCI_DEVICE(PCI_VENDOR_ID_HUAWEI, SEC_PF_PCI_DEVICE_ID) },
239*4882a593Smuzhiyun 	{ PCI_DEVICE(PCI_VENDOR_ID_HUAWEI, SEC_VF_PCI_DEVICE_ID) },
240*4882a593Smuzhiyun 	{ 0, }
241*4882a593Smuzhiyun };
242*4882a593Smuzhiyun MODULE_DEVICE_TABLE(pci, sec_dev_ids);
243*4882a593Smuzhiyun 
sec_get_endian(struct hisi_qm * qm)244*4882a593Smuzhiyun static u8 sec_get_endian(struct hisi_qm *qm)
245*4882a593Smuzhiyun {
246*4882a593Smuzhiyun 	u32 reg;
247*4882a593Smuzhiyun 
248*4882a593Smuzhiyun 	/*
249*4882a593Smuzhiyun 	 * As for VF, it is a wrong way to get endian setting by
250*4882a593Smuzhiyun 	 * reading a register of the engine
251*4882a593Smuzhiyun 	 */
252*4882a593Smuzhiyun 	if (qm->pdev->is_virtfn) {
253*4882a593Smuzhiyun 		dev_err_ratelimited(&qm->pdev->dev,
254*4882a593Smuzhiyun 				    "cannot access a register in VF!\n");
255*4882a593Smuzhiyun 		return SEC_LE;
256*4882a593Smuzhiyun 	}
257*4882a593Smuzhiyun 	reg = readl_relaxed(qm->io_base + SEC_ENGINE_PF_CFG_OFF +
258*4882a593Smuzhiyun 			    SEC_ACC_COMMON_REG_OFF + SEC_CONTROL_REG);
259*4882a593Smuzhiyun 
260*4882a593Smuzhiyun 	/* BD little endian mode */
261*4882a593Smuzhiyun 	if (!(reg & BIT(0)))
262*4882a593Smuzhiyun 		return SEC_LE;
263*4882a593Smuzhiyun 
264*4882a593Smuzhiyun 	/* BD 32-bits big endian mode */
265*4882a593Smuzhiyun 	else if (!(reg & BIT(1)))
266*4882a593Smuzhiyun 		return SEC_32BE;
267*4882a593Smuzhiyun 
268*4882a593Smuzhiyun 	/* BD 64-bits big endian mode */
269*4882a593Smuzhiyun 	else
270*4882a593Smuzhiyun 		return SEC_64BE;
271*4882a593Smuzhiyun }
272*4882a593Smuzhiyun 
sec_engine_init(struct hisi_qm * qm)273*4882a593Smuzhiyun static int sec_engine_init(struct hisi_qm *qm)
274*4882a593Smuzhiyun {
275*4882a593Smuzhiyun 	int ret;
276*4882a593Smuzhiyun 	u32 reg;
277*4882a593Smuzhiyun 
278*4882a593Smuzhiyun 	/* disable clock gate control */
279*4882a593Smuzhiyun 	reg = readl_relaxed(SEC_ADDR(qm, SEC_CONTROL_REG));
280*4882a593Smuzhiyun 	reg &= SEC_CLK_GATE_DISABLE;
281*4882a593Smuzhiyun 	writel_relaxed(reg, SEC_ADDR(qm, SEC_CONTROL_REG));
282*4882a593Smuzhiyun 
283*4882a593Smuzhiyun 	writel_relaxed(0x1, SEC_ADDR(qm, SEC_MEM_START_INIT_REG));
284*4882a593Smuzhiyun 
285*4882a593Smuzhiyun 	ret = readl_relaxed_poll_timeout(SEC_ADDR(qm, SEC_MEM_INIT_DONE_REG),
286*4882a593Smuzhiyun 					 reg, reg & 0x1, SEC_DELAY_10_US,
287*4882a593Smuzhiyun 					 SEC_POLL_TIMEOUT_US);
288*4882a593Smuzhiyun 	if (ret) {
289*4882a593Smuzhiyun 		pci_err(qm->pdev, "fail to init sec mem\n");
290*4882a593Smuzhiyun 		return ret;
291*4882a593Smuzhiyun 	}
292*4882a593Smuzhiyun 
293*4882a593Smuzhiyun 	reg = readl_relaxed(SEC_ADDR(qm, SEC_CONTROL_REG));
294*4882a593Smuzhiyun 	reg |= (0x1 << SEC_TRNG_EN_SHIFT);
295*4882a593Smuzhiyun 	writel_relaxed(reg, SEC_ADDR(qm, SEC_CONTROL_REG));
296*4882a593Smuzhiyun 
297*4882a593Smuzhiyun 	reg = readl_relaxed(SEC_ADDR(qm, SEC_INTERFACE_USER_CTRL0_REG));
298*4882a593Smuzhiyun 	reg |= SEC_USER0_SMMU_NORMAL;
299*4882a593Smuzhiyun 	writel_relaxed(reg, SEC_ADDR(qm, SEC_INTERFACE_USER_CTRL0_REG));
300*4882a593Smuzhiyun 
301*4882a593Smuzhiyun 	reg = readl_relaxed(SEC_ADDR(qm, SEC_INTERFACE_USER_CTRL1_REG));
302*4882a593Smuzhiyun 	reg |= SEC_USER1_SMMU_NORMAL;
303*4882a593Smuzhiyun 	writel_relaxed(reg, SEC_ADDR(qm, SEC_INTERFACE_USER_CTRL1_REG));
304*4882a593Smuzhiyun 
305*4882a593Smuzhiyun 	writel(SEC_SINGLE_PORT_MAX_TRANS,
306*4882a593Smuzhiyun 	       qm->io_base + AM_CFG_SINGLE_PORT_MAX_TRANS);
307*4882a593Smuzhiyun 
308*4882a593Smuzhiyun 	writel(SEC_SAA_ENABLE, SEC_ADDR(qm, SEC_SAA_EN_REG));
309*4882a593Smuzhiyun 
310*4882a593Smuzhiyun 	/* Enable sm4 extra mode, as ctr/ecb */
311*4882a593Smuzhiyun 	writel_relaxed(SEC_BD_ERR_CHK_EN0,
312*4882a593Smuzhiyun 		       SEC_ADDR(qm, SEC_BD_ERR_CHK_EN_REG0));
313*4882a593Smuzhiyun 	/* Enable sm4 xts mode multiple iv */
314*4882a593Smuzhiyun 	writel_relaxed(SEC_BD_ERR_CHK_EN1,
315*4882a593Smuzhiyun 		       SEC_ADDR(qm, SEC_BD_ERR_CHK_EN_REG1));
316*4882a593Smuzhiyun 	writel_relaxed(SEC_BD_ERR_CHK_EN3,
317*4882a593Smuzhiyun 		       SEC_ADDR(qm, SEC_BD_ERR_CHK_EN_REG3));
318*4882a593Smuzhiyun 
319*4882a593Smuzhiyun 	/* config endian */
320*4882a593Smuzhiyun 	reg = readl_relaxed(SEC_ADDR(qm, SEC_CONTROL_REG));
321*4882a593Smuzhiyun 	reg |= sec_get_endian(qm);
322*4882a593Smuzhiyun 	writel_relaxed(reg, SEC_ADDR(qm, SEC_CONTROL_REG));
323*4882a593Smuzhiyun 
324*4882a593Smuzhiyun 	return 0;
325*4882a593Smuzhiyun }
326*4882a593Smuzhiyun 
sec_set_user_domain_and_cache(struct hisi_qm * qm)327*4882a593Smuzhiyun static int sec_set_user_domain_and_cache(struct hisi_qm *qm)
328*4882a593Smuzhiyun {
329*4882a593Smuzhiyun 	/* qm user domain */
330*4882a593Smuzhiyun 	writel(AXUSER_BASE, qm->io_base + QM_ARUSER_M_CFG_1);
331*4882a593Smuzhiyun 	writel(ARUSER_M_CFG_ENABLE, qm->io_base + QM_ARUSER_M_CFG_ENABLE);
332*4882a593Smuzhiyun 	writel(AXUSER_BASE, qm->io_base + QM_AWUSER_M_CFG_1);
333*4882a593Smuzhiyun 	writel(AWUSER_M_CFG_ENABLE, qm->io_base + QM_AWUSER_M_CFG_ENABLE);
334*4882a593Smuzhiyun 	writel(WUSER_M_CFG_ENABLE, qm->io_base + QM_WUSER_M_CFG_ENABLE);
335*4882a593Smuzhiyun 
336*4882a593Smuzhiyun 	/* qm cache */
337*4882a593Smuzhiyun 	writel(AXI_M_CFG, qm->io_base + QM_AXI_M_CFG);
338*4882a593Smuzhiyun 	writel(AXI_M_CFG_ENABLE, qm->io_base + QM_AXI_M_CFG_ENABLE);
339*4882a593Smuzhiyun 
340*4882a593Smuzhiyun 	/* disable FLR triggered by BME(bus master enable) */
341*4882a593Smuzhiyun 	writel(PEH_AXUSER_CFG, qm->io_base + QM_PEH_AXUSER_CFG);
342*4882a593Smuzhiyun 	writel(PEH_AXUSER_CFG_ENABLE, qm->io_base + QM_PEH_AXUSER_CFG_ENABLE);
343*4882a593Smuzhiyun 
344*4882a593Smuzhiyun 	/* enable sqc,cqc writeback */
345*4882a593Smuzhiyun 	writel(SQC_CACHE_ENABLE | CQC_CACHE_ENABLE | SQC_CACHE_WB_ENABLE |
346*4882a593Smuzhiyun 	       CQC_CACHE_WB_ENABLE | FIELD_PREP(SQC_CACHE_WB_THRD, 1) |
347*4882a593Smuzhiyun 	       FIELD_PREP(CQC_CACHE_WB_THRD, 1), qm->io_base + QM_CACHE_CTL);
348*4882a593Smuzhiyun 
349*4882a593Smuzhiyun 	return sec_engine_init(qm);
350*4882a593Smuzhiyun }
351*4882a593Smuzhiyun 
352*4882a593Smuzhiyun /* sec_debug_regs_clear() - clear the sec debug regs */
sec_debug_regs_clear(struct hisi_qm * qm)353*4882a593Smuzhiyun static void sec_debug_regs_clear(struct hisi_qm *qm)
354*4882a593Smuzhiyun {
355*4882a593Smuzhiyun 	int i;
356*4882a593Smuzhiyun 
357*4882a593Smuzhiyun 	/* clear current_qm */
358*4882a593Smuzhiyun 	writel(0x0, qm->io_base + QM_DFX_MB_CNT_VF);
359*4882a593Smuzhiyun 	writel(0x0, qm->io_base + QM_DFX_DB_CNT_VF);
360*4882a593Smuzhiyun 
361*4882a593Smuzhiyun 	/* clear sec dfx regs */
362*4882a593Smuzhiyun 	writel(0x1, qm->io_base + SEC_CTRL_CNT_CLR_CE);
363*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(sec_dfx_regs); i++)
364*4882a593Smuzhiyun 		readl(qm->io_base + sec_dfx_regs[i].offset);
365*4882a593Smuzhiyun 
366*4882a593Smuzhiyun 	/* clear rdclr_en */
367*4882a593Smuzhiyun 	writel(0x0, qm->io_base + SEC_CTRL_CNT_CLR_CE);
368*4882a593Smuzhiyun 
369*4882a593Smuzhiyun 	hisi_qm_debug_regs_clear(qm);
370*4882a593Smuzhiyun }
371*4882a593Smuzhiyun 
sec_hw_error_enable(struct hisi_qm * qm)372*4882a593Smuzhiyun static void sec_hw_error_enable(struct hisi_qm *qm)
373*4882a593Smuzhiyun {
374*4882a593Smuzhiyun 	u32 val;
375*4882a593Smuzhiyun 
376*4882a593Smuzhiyun 	if (qm->ver == QM_HW_V1) {
377*4882a593Smuzhiyun 		writel(SEC_CORE_INT_DISABLE, qm->io_base + SEC_CORE_INT_MASK);
378*4882a593Smuzhiyun 		pci_info(qm->pdev, "V1 not support hw error handle\n");
379*4882a593Smuzhiyun 		return;
380*4882a593Smuzhiyun 	}
381*4882a593Smuzhiyun 
382*4882a593Smuzhiyun 	val = readl(SEC_ADDR(qm, SEC_CONTROL_REG));
383*4882a593Smuzhiyun 
384*4882a593Smuzhiyun 	/* clear SEC hw error source if having */
385*4882a593Smuzhiyun 	writel(SEC_CORE_INT_CLEAR, qm->io_base + SEC_CORE_INT_SOURCE);
386*4882a593Smuzhiyun 
387*4882a593Smuzhiyun 	/* enable SEC hw error interrupts */
388*4882a593Smuzhiyun 	writel(SEC_CORE_INT_ENABLE, qm->io_base + SEC_CORE_INT_MASK);
389*4882a593Smuzhiyun 
390*4882a593Smuzhiyun 	/* enable RAS int */
391*4882a593Smuzhiyun 	writel(SEC_RAS_CE_ENB_MSK, qm->io_base + SEC_RAS_CE_REG);
392*4882a593Smuzhiyun 	writel(SEC_RAS_FE_ENB_MSK, qm->io_base + SEC_RAS_FE_REG);
393*4882a593Smuzhiyun 	writel(SEC_RAS_NFE_ENB_MSK, qm->io_base + SEC_RAS_NFE_REG);
394*4882a593Smuzhiyun 
395*4882a593Smuzhiyun 	/* enable SEC block master OOO when m-bit error occur */
396*4882a593Smuzhiyun 	val = val | SEC_AXI_SHUTDOWN_ENABLE;
397*4882a593Smuzhiyun 
398*4882a593Smuzhiyun 	writel(val, SEC_ADDR(qm, SEC_CONTROL_REG));
399*4882a593Smuzhiyun }
400*4882a593Smuzhiyun 
sec_hw_error_disable(struct hisi_qm * qm)401*4882a593Smuzhiyun static void sec_hw_error_disable(struct hisi_qm *qm)
402*4882a593Smuzhiyun {
403*4882a593Smuzhiyun 	u32 val;
404*4882a593Smuzhiyun 
405*4882a593Smuzhiyun 	val = readl(SEC_ADDR(qm, SEC_CONTROL_REG));
406*4882a593Smuzhiyun 
407*4882a593Smuzhiyun 	/* disable RAS int */
408*4882a593Smuzhiyun 	writel(SEC_RAS_DISABLE, qm->io_base + SEC_RAS_CE_REG);
409*4882a593Smuzhiyun 	writel(SEC_RAS_DISABLE, qm->io_base + SEC_RAS_FE_REG);
410*4882a593Smuzhiyun 	writel(SEC_RAS_DISABLE, qm->io_base + SEC_RAS_NFE_REG);
411*4882a593Smuzhiyun 
412*4882a593Smuzhiyun 	/* disable SEC hw error interrupts */
413*4882a593Smuzhiyun 	writel(SEC_CORE_INT_DISABLE, qm->io_base + SEC_CORE_INT_MASK);
414*4882a593Smuzhiyun 
415*4882a593Smuzhiyun 	/* disable SEC block master OOO when m-bit error occur */
416*4882a593Smuzhiyun 	val = val & SEC_AXI_SHUTDOWN_DISABLE;
417*4882a593Smuzhiyun 
418*4882a593Smuzhiyun 	writel(val, SEC_ADDR(qm, SEC_CONTROL_REG));
419*4882a593Smuzhiyun }
420*4882a593Smuzhiyun 
sec_current_qm_read(struct sec_debug_file * file)421*4882a593Smuzhiyun static u32 sec_current_qm_read(struct sec_debug_file *file)
422*4882a593Smuzhiyun {
423*4882a593Smuzhiyun 	struct hisi_qm *qm = file->qm;
424*4882a593Smuzhiyun 
425*4882a593Smuzhiyun 	return readl(qm->io_base + QM_DFX_MB_CNT_VF);
426*4882a593Smuzhiyun }
427*4882a593Smuzhiyun 
sec_current_qm_write(struct sec_debug_file * file,u32 val)428*4882a593Smuzhiyun static int sec_current_qm_write(struct sec_debug_file *file, u32 val)
429*4882a593Smuzhiyun {
430*4882a593Smuzhiyun 	struct hisi_qm *qm = file->qm;
431*4882a593Smuzhiyun 	u32 vfq_num;
432*4882a593Smuzhiyun 	u32 tmp;
433*4882a593Smuzhiyun 
434*4882a593Smuzhiyun 	if (val > qm->vfs_num)
435*4882a593Smuzhiyun 		return -EINVAL;
436*4882a593Smuzhiyun 
437*4882a593Smuzhiyun 	/* According PF or VF Dev ID to calculation curr_qm_qp_num and store */
438*4882a593Smuzhiyun 	if (!val) {
439*4882a593Smuzhiyun 		qm->debug.curr_qm_qp_num = qm->qp_num;
440*4882a593Smuzhiyun 	} else {
441*4882a593Smuzhiyun 		vfq_num = (qm->ctrl_qp_num - qm->qp_num) / qm->vfs_num;
442*4882a593Smuzhiyun 
443*4882a593Smuzhiyun 		if (val == qm->vfs_num)
444*4882a593Smuzhiyun 			qm->debug.curr_qm_qp_num =
445*4882a593Smuzhiyun 				qm->ctrl_qp_num - qm->qp_num -
446*4882a593Smuzhiyun 				(qm->vfs_num - 1) * vfq_num;
447*4882a593Smuzhiyun 		else
448*4882a593Smuzhiyun 			qm->debug.curr_qm_qp_num = vfq_num;
449*4882a593Smuzhiyun 	}
450*4882a593Smuzhiyun 
451*4882a593Smuzhiyun 	writel(val, qm->io_base + QM_DFX_MB_CNT_VF);
452*4882a593Smuzhiyun 	writel(val, qm->io_base + QM_DFX_DB_CNT_VF);
453*4882a593Smuzhiyun 
454*4882a593Smuzhiyun 	tmp = val |
455*4882a593Smuzhiyun 	      (readl(qm->io_base + QM_DFX_SQE_CNT_VF_SQN) & CURRENT_Q_MASK);
456*4882a593Smuzhiyun 	writel(tmp, qm->io_base + QM_DFX_SQE_CNT_VF_SQN);
457*4882a593Smuzhiyun 
458*4882a593Smuzhiyun 	tmp = val |
459*4882a593Smuzhiyun 	      (readl(qm->io_base + QM_DFX_CQE_CNT_VF_CQN) & CURRENT_Q_MASK);
460*4882a593Smuzhiyun 	writel(tmp, qm->io_base + QM_DFX_CQE_CNT_VF_CQN);
461*4882a593Smuzhiyun 
462*4882a593Smuzhiyun 	return 0;
463*4882a593Smuzhiyun }
464*4882a593Smuzhiyun 
sec_clear_enable_read(struct sec_debug_file * file)465*4882a593Smuzhiyun static u32 sec_clear_enable_read(struct sec_debug_file *file)
466*4882a593Smuzhiyun {
467*4882a593Smuzhiyun 	struct hisi_qm *qm = file->qm;
468*4882a593Smuzhiyun 
469*4882a593Smuzhiyun 	return readl(qm->io_base + SEC_CTRL_CNT_CLR_CE) &
470*4882a593Smuzhiyun 			SEC_CTRL_CNT_CLR_CE_BIT;
471*4882a593Smuzhiyun }
472*4882a593Smuzhiyun 
sec_clear_enable_write(struct sec_debug_file * file,u32 val)473*4882a593Smuzhiyun static int sec_clear_enable_write(struct sec_debug_file *file, u32 val)
474*4882a593Smuzhiyun {
475*4882a593Smuzhiyun 	struct hisi_qm *qm = file->qm;
476*4882a593Smuzhiyun 	u32 tmp;
477*4882a593Smuzhiyun 
478*4882a593Smuzhiyun 	if (val != 1 && val)
479*4882a593Smuzhiyun 		return -EINVAL;
480*4882a593Smuzhiyun 
481*4882a593Smuzhiyun 	tmp = (readl(qm->io_base + SEC_CTRL_CNT_CLR_CE) &
482*4882a593Smuzhiyun 	       ~SEC_CTRL_CNT_CLR_CE_BIT) | val;
483*4882a593Smuzhiyun 	writel(tmp, qm->io_base + SEC_CTRL_CNT_CLR_CE);
484*4882a593Smuzhiyun 
485*4882a593Smuzhiyun 	return 0;
486*4882a593Smuzhiyun }
487*4882a593Smuzhiyun 
sec_debug_read(struct file * filp,char __user * buf,size_t count,loff_t * pos)488*4882a593Smuzhiyun static ssize_t sec_debug_read(struct file *filp, char __user *buf,
489*4882a593Smuzhiyun 			       size_t count, loff_t *pos)
490*4882a593Smuzhiyun {
491*4882a593Smuzhiyun 	struct sec_debug_file *file = filp->private_data;
492*4882a593Smuzhiyun 	char tbuf[SEC_DBGFS_VAL_MAX_LEN];
493*4882a593Smuzhiyun 	u32 val;
494*4882a593Smuzhiyun 	int ret;
495*4882a593Smuzhiyun 
496*4882a593Smuzhiyun 	spin_lock_irq(&file->lock);
497*4882a593Smuzhiyun 
498*4882a593Smuzhiyun 	switch (file->index) {
499*4882a593Smuzhiyun 	case SEC_CURRENT_QM:
500*4882a593Smuzhiyun 		val = sec_current_qm_read(file);
501*4882a593Smuzhiyun 		break;
502*4882a593Smuzhiyun 	case SEC_CLEAR_ENABLE:
503*4882a593Smuzhiyun 		val = sec_clear_enable_read(file);
504*4882a593Smuzhiyun 		break;
505*4882a593Smuzhiyun 	default:
506*4882a593Smuzhiyun 		spin_unlock_irq(&file->lock);
507*4882a593Smuzhiyun 		return -EINVAL;
508*4882a593Smuzhiyun 	}
509*4882a593Smuzhiyun 
510*4882a593Smuzhiyun 	spin_unlock_irq(&file->lock);
511*4882a593Smuzhiyun 	ret = snprintf(tbuf, SEC_DBGFS_VAL_MAX_LEN, "%u\n", val);
512*4882a593Smuzhiyun 
513*4882a593Smuzhiyun 	return simple_read_from_buffer(buf, count, pos, tbuf, ret);
514*4882a593Smuzhiyun }
515*4882a593Smuzhiyun 
sec_debug_write(struct file * filp,const char __user * buf,size_t count,loff_t * pos)516*4882a593Smuzhiyun static ssize_t sec_debug_write(struct file *filp, const char __user *buf,
517*4882a593Smuzhiyun 			       size_t count, loff_t *pos)
518*4882a593Smuzhiyun {
519*4882a593Smuzhiyun 	struct sec_debug_file *file = filp->private_data;
520*4882a593Smuzhiyun 	char tbuf[SEC_DBGFS_VAL_MAX_LEN];
521*4882a593Smuzhiyun 	unsigned long val;
522*4882a593Smuzhiyun 	int len, ret;
523*4882a593Smuzhiyun 
524*4882a593Smuzhiyun 	if (*pos != 0)
525*4882a593Smuzhiyun 		return 0;
526*4882a593Smuzhiyun 
527*4882a593Smuzhiyun 	if (count >= SEC_DBGFS_VAL_MAX_LEN)
528*4882a593Smuzhiyun 		return -ENOSPC;
529*4882a593Smuzhiyun 
530*4882a593Smuzhiyun 	len = simple_write_to_buffer(tbuf, SEC_DBGFS_VAL_MAX_LEN - 1,
531*4882a593Smuzhiyun 				     pos, buf, count);
532*4882a593Smuzhiyun 	if (len < 0)
533*4882a593Smuzhiyun 		return len;
534*4882a593Smuzhiyun 
535*4882a593Smuzhiyun 	tbuf[len] = '\0';
536*4882a593Smuzhiyun 	if (kstrtoul(tbuf, 0, &val))
537*4882a593Smuzhiyun 		return -EFAULT;
538*4882a593Smuzhiyun 
539*4882a593Smuzhiyun 	spin_lock_irq(&file->lock);
540*4882a593Smuzhiyun 
541*4882a593Smuzhiyun 	switch (file->index) {
542*4882a593Smuzhiyun 	case SEC_CURRENT_QM:
543*4882a593Smuzhiyun 		ret = sec_current_qm_write(file, val);
544*4882a593Smuzhiyun 		if (ret)
545*4882a593Smuzhiyun 			goto err_input;
546*4882a593Smuzhiyun 		break;
547*4882a593Smuzhiyun 	case SEC_CLEAR_ENABLE:
548*4882a593Smuzhiyun 		ret = sec_clear_enable_write(file, val);
549*4882a593Smuzhiyun 		if (ret)
550*4882a593Smuzhiyun 			goto err_input;
551*4882a593Smuzhiyun 		break;
552*4882a593Smuzhiyun 	default:
553*4882a593Smuzhiyun 		ret = -EINVAL;
554*4882a593Smuzhiyun 		goto err_input;
555*4882a593Smuzhiyun 	}
556*4882a593Smuzhiyun 
557*4882a593Smuzhiyun 	spin_unlock_irq(&file->lock);
558*4882a593Smuzhiyun 
559*4882a593Smuzhiyun 	return count;
560*4882a593Smuzhiyun 
561*4882a593Smuzhiyun  err_input:
562*4882a593Smuzhiyun 	spin_unlock_irq(&file->lock);
563*4882a593Smuzhiyun 	return ret;
564*4882a593Smuzhiyun }
565*4882a593Smuzhiyun 
566*4882a593Smuzhiyun static const struct file_operations sec_dbg_fops = {
567*4882a593Smuzhiyun 	.owner = THIS_MODULE,
568*4882a593Smuzhiyun 	.open = simple_open,
569*4882a593Smuzhiyun 	.read = sec_debug_read,
570*4882a593Smuzhiyun 	.write = sec_debug_write,
571*4882a593Smuzhiyun };
572*4882a593Smuzhiyun 
sec_debugfs_atomic64_get(void * data,u64 * val)573*4882a593Smuzhiyun static int sec_debugfs_atomic64_get(void *data, u64 *val)
574*4882a593Smuzhiyun {
575*4882a593Smuzhiyun 	*val = atomic64_read((atomic64_t *)data);
576*4882a593Smuzhiyun 
577*4882a593Smuzhiyun 	return 0;
578*4882a593Smuzhiyun }
579*4882a593Smuzhiyun 
sec_debugfs_atomic64_set(void * data,u64 val)580*4882a593Smuzhiyun static int sec_debugfs_atomic64_set(void *data, u64 val)
581*4882a593Smuzhiyun {
582*4882a593Smuzhiyun 	if (val)
583*4882a593Smuzhiyun 		return -EINVAL;
584*4882a593Smuzhiyun 
585*4882a593Smuzhiyun 	atomic64_set((atomic64_t *)data, 0);
586*4882a593Smuzhiyun 
587*4882a593Smuzhiyun 	return 0;
588*4882a593Smuzhiyun }
589*4882a593Smuzhiyun 
590*4882a593Smuzhiyun DEFINE_DEBUGFS_ATTRIBUTE(sec_atomic64_ops, sec_debugfs_atomic64_get,
591*4882a593Smuzhiyun 			 sec_debugfs_atomic64_set, "%lld\n");
592*4882a593Smuzhiyun 
sec_core_debug_init(struct hisi_qm * qm)593*4882a593Smuzhiyun static int sec_core_debug_init(struct hisi_qm *qm)
594*4882a593Smuzhiyun {
595*4882a593Smuzhiyun 	struct sec_dev *sec = container_of(qm, struct sec_dev, qm);
596*4882a593Smuzhiyun 	struct device *dev = &qm->pdev->dev;
597*4882a593Smuzhiyun 	struct sec_dfx *dfx = &sec->debug.dfx;
598*4882a593Smuzhiyun 	struct debugfs_regset32 *regset;
599*4882a593Smuzhiyun 	struct dentry *tmp_d;
600*4882a593Smuzhiyun 	int i;
601*4882a593Smuzhiyun 
602*4882a593Smuzhiyun 	tmp_d = debugfs_create_dir("sec_dfx", qm->debug.debug_root);
603*4882a593Smuzhiyun 
604*4882a593Smuzhiyun 	regset = devm_kzalloc(dev, sizeof(*regset), GFP_KERNEL);
605*4882a593Smuzhiyun 	if (!regset)
606*4882a593Smuzhiyun 		return -ENOMEM;
607*4882a593Smuzhiyun 
608*4882a593Smuzhiyun 	regset->regs = sec_dfx_regs;
609*4882a593Smuzhiyun 	regset->nregs = ARRAY_SIZE(sec_dfx_regs);
610*4882a593Smuzhiyun 	regset->base = qm->io_base;
611*4882a593Smuzhiyun 
612*4882a593Smuzhiyun 	if (qm->pdev->device == SEC_PF_PCI_DEVICE_ID)
613*4882a593Smuzhiyun 		debugfs_create_regset32("regs", 0444, tmp_d, regset);
614*4882a593Smuzhiyun 
615*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(sec_dfx_labels); i++) {
616*4882a593Smuzhiyun 		atomic64_t *data = (atomic64_t *)((uintptr_t)dfx +
617*4882a593Smuzhiyun 					sec_dfx_labels[i].offset);
618*4882a593Smuzhiyun 		debugfs_create_file(sec_dfx_labels[i].name, 0644,
619*4882a593Smuzhiyun 				   tmp_d, data, &sec_atomic64_ops);
620*4882a593Smuzhiyun 	}
621*4882a593Smuzhiyun 
622*4882a593Smuzhiyun 	return 0;
623*4882a593Smuzhiyun }
624*4882a593Smuzhiyun 
sec_debug_init(struct hisi_qm * qm)625*4882a593Smuzhiyun static int sec_debug_init(struct hisi_qm *qm)
626*4882a593Smuzhiyun {
627*4882a593Smuzhiyun 	struct sec_dev *sec = container_of(qm, struct sec_dev, qm);
628*4882a593Smuzhiyun 	int i;
629*4882a593Smuzhiyun 
630*4882a593Smuzhiyun 	if (qm->pdev->device == SEC_PF_PCI_DEVICE_ID) {
631*4882a593Smuzhiyun 		for (i = SEC_CURRENT_QM; i < SEC_DEBUG_FILE_NUM; i++) {
632*4882a593Smuzhiyun 			spin_lock_init(&sec->debug.files[i].lock);
633*4882a593Smuzhiyun 			sec->debug.files[i].index = i;
634*4882a593Smuzhiyun 			sec->debug.files[i].qm = qm;
635*4882a593Smuzhiyun 
636*4882a593Smuzhiyun 			debugfs_create_file(sec_dbg_file_name[i], 0600,
637*4882a593Smuzhiyun 						  qm->debug.debug_root,
638*4882a593Smuzhiyun 						  sec->debug.files + i,
639*4882a593Smuzhiyun 						  &sec_dbg_fops);
640*4882a593Smuzhiyun 		}
641*4882a593Smuzhiyun 	}
642*4882a593Smuzhiyun 
643*4882a593Smuzhiyun 	return sec_core_debug_init(qm);
644*4882a593Smuzhiyun }
645*4882a593Smuzhiyun 
sec_debugfs_init(struct hisi_qm * qm)646*4882a593Smuzhiyun static int sec_debugfs_init(struct hisi_qm *qm)
647*4882a593Smuzhiyun {
648*4882a593Smuzhiyun 	struct device *dev = &qm->pdev->dev;
649*4882a593Smuzhiyun 	int ret;
650*4882a593Smuzhiyun 
651*4882a593Smuzhiyun 	qm->debug.debug_root = debugfs_create_dir(dev_name(dev),
652*4882a593Smuzhiyun 						  sec_debugfs_root);
653*4882a593Smuzhiyun 	qm->debug.sqe_mask_offset = SEC_SQE_MASK_OFFSET;
654*4882a593Smuzhiyun 	qm->debug.sqe_mask_len = SEC_SQE_MASK_LEN;
655*4882a593Smuzhiyun 	ret = hisi_qm_debug_init(qm);
656*4882a593Smuzhiyun 	if (ret)
657*4882a593Smuzhiyun 		goto failed_to_create;
658*4882a593Smuzhiyun 
659*4882a593Smuzhiyun 	ret = sec_debug_init(qm);
660*4882a593Smuzhiyun 	if (ret)
661*4882a593Smuzhiyun 		goto failed_to_create;
662*4882a593Smuzhiyun 
663*4882a593Smuzhiyun 
664*4882a593Smuzhiyun 	return 0;
665*4882a593Smuzhiyun 
666*4882a593Smuzhiyun failed_to_create:
667*4882a593Smuzhiyun 	debugfs_remove_recursive(sec_debugfs_root);
668*4882a593Smuzhiyun 
669*4882a593Smuzhiyun 	return ret;
670*4882a593Smuzhiyun }
671*4882a593Smuzhiyun 
sec_debugfs_exit(struct hisi_qm * qm)672*4882a593Smuzhiyun static void sec_debugfs_exit(struct hisi_qm *qm)
673*4882a593Smuzhiyun {
674*4882a593Smuzhiyun 	debugfs_remove_recursive(qm->debug.debug_root);
675*4882a593Smuzhiyun }
676*4882a593Smuzhiyun 
sec_log_hw_error(struct hisi_qm * qm,u32 err_sts)677*4882a593Smuzhiyun static void sec_log_hw_error(struct hisi_qm *qm, u32 err_sts)
678*4882a593Smuzhiyun {
679*4882a593Smuzhiyun 	const struct sec_hw_error *errs = sec_hw_errors;
680*4882a593Smuzhiyun 	struct device *dev = &qm->pdev->dev;
681*4882a593Smuzhiyun 	u32 err_val;
682*4882a593Smuzhiyun 
683*4882a593Smuzhiyun 	while (errs->msg) {
684*4882a593Smuzhiyun 		if (errs->int_msk & err_sts) {
685*4882a593Smuzhiyun 			dev_err(dev, "%s [error status=0x%x] found\n",
686*4882a593Smuzhiyun 				errs->msg, errs->int_msk);
687*4882a593Smuzhiyun 
688*4882a593Smuzhiyun 			if (SEC_CORE_INT_STATUS_M_ECC & errs->int_msk) {
689*4882a593Smuzhiyun 				err_val = readl(qm->io_base +
690*4882a593Smuzhiyun 						SEC_CORE_SRAM_ECC_ERR_INFO);
691*4882a593Smuzhiyun 				dev_err(dev, "multi ecc sram num=0x%x\n",
692*4882a593Smuzhiyun 					SEC_ECC_NUM(err_val));
693*4882a593Smuzhiyun 			}
694*4882a593Smuzhiyun 		}
695*4882a593Smuzhiyun 		errs++;
696*4882a593Smuzhiyun 	}
697*4882a593Smuzhiyun }
698*4882a593Smuzhiyun 
sec_get_hw_err_status(struct hisi_qm * qm)699*4882a593Smuzhiyun static u32 sec_get_hw_err_status(struct hisi_qm *qm)
700*4882a593Smuzhiyun {
701*4882a593Smuzhiyun 	return readl(qm->io_base + SEC_CORE_INT_STATUS);
702*4882a593Smuzhiyun }
703*4882a593Smuzhiyun 
sec_clear_hw_err_status(struct hisi_qm * qm,u32 err_sts)704*4882a593Smuzhiyun static void sec_clear_hw_err_status(struct hisi_qm *qm, u32 err_sts)
705*4882a593Smuzhiyun {
706*4882a593Smuzhiyun 	writel(err_sts, qm->io_base + SEC_CORE_INT_SOURCE);
707*4882a593Smuzhiyun }
708*4882a593Smuzhiyun 
sec_open_axi_master_ooo(struct hisi_qm * qm)709*4882a593Smuzhiyun static void sec_open_axi_master_ooo(struct hisi_qm *qm)
710*4882a593Smuzhiyun {
711*4882a593Smuzhiyun 	u32 val;
712*4882a593Smuzhiyun 
713*4882a593Smuzhiyun 	val = readl(SEC_ADDR(qm, SEC_CONTROL_REG));
714*4882a593Smuzhiyun 	writel(val & SEC_AXI_SHUTDOWN_DISABLE, SEC_ADDR(qm, SEC_CONTROL_REG));
715*4882a593Smuzhiyun 	writel(val | SEC_AXI_SHUTDOWN_ENABLE, SEC_ADDR(qm, SEC_CONTROL_REG));
716*4882a593Smuzhiyun }
717*4882a593Smuzhiyun 
718*4882a593Smuzhiyun static const struct hisi_qm_err_ini sec_err_ini = {
719*4882a593Smuzhiyun 	.hw_init		= sec_set_user_domain_and_cache,
720*4882a593Smuzhiyun 	.hw_err_enable		= sec_hw_error_enable,
721*4882a593Smuzhiyun 	.hw_err_disable		= sec_hw_error_disable,
722*4882a593Smuzhiyun 	.get_dev_hw_err_status	= sec_get_hw_err_status,
723*4882a593Smuzhiyun 	.clear_dev_hw_err_status = sec_clear_hw_err_status,
724*4882a593Smuzhiyun 	.log_dev_hw_err		= sec_log_hw_error,
725*4882a593Smuzhiyun 	.open_axi_master_ooo	= sec_open_axi_master_ooo,
726*4882a593Smuzhiyun 	.err_info		= {
727*4882a593Smuzhiyun 		.ce			= QM_BASE_CE,
728*4882a593Smuzhiyun 		.nfe			= QM_BASE_NFE | QM_ACC_DO_TASK_TIMEOUT |
729*4882a593Smuzhiyun 					  QM_ACC_WB_NOT_READY_TIMEOUT,
730*4882a593Smuzhiyun 		.fe			= 0,
731*4882a593Smuzhiyun 		.ecc_2bits_mask		= SEC_CORE_INT_STATUS_M_ECC,
732*4882a593Smuzhiyun 		.msi_wr_port		= BIT(0),
733*4882a593Smuzhiyun 		.acpi_rst		= "SRST",
734*4882a593Smuzhiyun 	}
735*4882a593Smuzhiyun };
736*4882a593Smuzhiyun 
sec_pf_probe_init(struct sec_dev * sec)737*4882a593Smuzhiyun static int sec_pf_probe_init(struct sec_dev *sec)
738*4882a593Smuzhiyun {
739*4882a593Smuzhiyun 	struct hisi_qm *qm = &sec->qm;
740*4882a593Smuzhiyun 	int ret;
741*4882a593Smuzhiyun 
742*4882a593Smuzhiyun 	if (qm->ver == QM_HW_V1)
743*4882a593Smuzhiyun 		qm->ctrl_qp_num = SEC_QUEUE_NUM_V1;
744*4882a593Smuzhiyun 	else
745*4882a593Smuzhiyun 		qm->ctrl_qp_num = SEC_QUEUE_NUM_V2;
746*4882a593Smuzhiyun 
747*4882a593Smuzhiyun 	qm->err_ini = &sec_err_ini;
748*4882a593Smuzhiyun 
749*4882a593Smuzhiyun 	ret = sec_set_user_domain_and_cache(qm);
750*4882a593Smuzhiyun 	if (ret)
751*4882a593Smuzhiyun 		return ret;
752*4882a593Smuzhiyun 
753*4882a593Smuzhiyun 	hisi_qm_dev_err_init(qm);
754*4882a593Smuzhiyun 	sec_debug_regs_clear(qm);
755*4882a593Smuzhiyun 
756*4882a593Smuzhiyun 	return 0;
757*4882a593Smuzhiyun }
758*4882a593Smuzhiyun 
sec_qm_init(struct hisi_qm * qm,struct pci_dev * pdev)759*4882a593Smuzhiyun static int sec_qm_init(struct hisi_qm *qm, struct pci_dev *pdev)
760*4882a593Smuzhiyun {
761*4882a593Smuzhiyun 	int ret;
762*4882a593Smuzhiyun 
763*4882a593Smuzhiyun 	qm->pdev = pdev;
764*4882a593Smuzhiyun 	qm->ver = pdev->revision;
765*4882a593Smuzhiyun 	qm->sqe_size = SEC_SQE_SIZE;
766*4882a593Smuzhiyun 	qm->dev_name = sec_name;
767*4882a593Smuzhiyun 
768*4882a593Smuzhiyun 	qm->fun_type = (pdev->device == SEC_PF_PCI_DEVICE_ID) ?
769*4882a593Smuzhiyun 			QM_HW_PF : QM_HW_VF;
770*4882a593Smuzhiyun 	if (qm->fun_type == QM_HW_PF) {
771*4882a593Smuzhiyun 		qm->qp_base = SEC_PF_DEF_Q_BASE;
772*4882a593Smuzhiyun 		qm->qp_num = pf_q_num;
773*4882a593Smuzhiyun 		qm->debug.curr_qm_qp_num = pf_q_num;
774*4882a593Smuzhiyun 		qm->qm_list = &sec_devices;
775*4882a593Smuzhiyun 	} else if (qm->fun_type == QM_HW_VF && qm->ver == QM_HW_V1) {
776*4882a593Smuzhiyun 		/*
777*4882a593Smuzhiyun 		 * have no way to get qm configure in VM in v1 hardware,
778*4882a593Smuzhiyun 		 * so currently force PF to uses SEC_PF_DEF_Q_NUM, and force
779*4882a593Smuzhiyun 		 * to trigger only one VF in v1 hardware.
780*4882a593Smuzhiyun 		 * v2 hardware has no such problem.
781*4882a593Smuzhiyun 		 */
782*4882a593Smuzhiyun 		qm->qp_base = SEC_PF_DEF_Q_NUM;
783*4882a593Smuzhiyun 		qm->qp_num = SEC_QUEUE_NUM_V1 - SEC_PF_DEF_Q_NUM;
784*4882a593Smuzhiyun 	}
785*4882a593Smuzhiyun 
786*4882a593Smuzhiyun 	/*
787*4882a593Smuzhiyun 	 * WQ_HIGHPRI: SEC request must be low delayed,
788*4882a593Smuzhiyun 	 * so need a high priority workqueue.
789*4882a593Smuzhiyun 	 * WQ_UNBOUND: SEC task is likely with long
790*4882a593Smuzhiyun 	 * running CPU intensive workloads.
791*4882a593Smuzhiyun 	 */
792*4882a593Smuzhiyun 	qm->wq = alloc_workqueue("%s", WQ_HIGHPRI | WQ_MEM_RECLAIM |
793*4882a593Smuzhiyun 				 WQ_UNBOUND, num_online_cpus(),
794*4882a593Smuzhiyun 				 pci_name(qm->pdev));
795*4882a593Smuzhiyun 	if (!qm->wq) {
796*4882a593Smuzhiyun 		pci_err(qm->pdev, "fail to alloc workqueue\n");
797*4882a593Smuzhiyun 		return -ENOMEM;
798*4882a593Smuzhiyun 	}
799*4882a593Smuzhiyun 
800*4882a593Smuzhiyun 	ret = hisi_qm_init(qm);
801*4882a593Smuzhiyun 	if (ret)
802*4882a593Smuzhiyun 		destroy_workqueue(qm->wq);
803*4882a593Smuzhiyun 
804*4882a593Smuzhiyun 	return ret;
805*4882a593Smuzhiyun }
806*4882a593Smuzhiyun 
sec_qm_uninit(struct hisi_qm * qm)807*4882a593Smuzhiyun static void sec_qm_uninit(struct hisi_qm *qm)
808*4882a593Smuzhiyun {
809*4882a593Smuzhiyun 	hisi_qm_uninit(qm);
810*4882a593Smuzhiyun }
811*4882a593Smuzhiyun 
sec_probe_init(struct sec_dev * sec)812*4882a593Smuzhiyun static int sec_probe_init(struct sec_dev *sec)
813*4882a593Smuzhiyun {
814*4882a593Smuzhiyun 	struct hisi_qm *qm = &sec->qm;
815*4882a593Smuzhiyun 	int ret;
816*4882a593Smuzhiyun 
817*4882a593Smuzhiyun 	if (qm->fun_type == QM_HW_PF) {
818*4882a593Smuzhiyun 		ret = sec_pf_probe_init(sec);
819*4882a593Smuzhiyun 		if (ret)
820*4882a593Smuzhiyun 			return ret;
821*4882a593Smuzhiyun 	}
822*4882a593Smuzhiyun 
823*4882a593Smuzhiyun 	return 0;
824*4882a593Smuzhiyun }
825*4882a593Smuzhiyun 
sec_probe_uninit(struct hisi_qm * qm)826*4882a593Smuzhiyun static void sec_probe_uninit(struct hisi_qm *qm)
827*4882a593Smuzhiyun {
828*4882a593Smuzhiyun 	hisi_qm_dev_err_uninit(qm);
829*4882a593Smuzhiyun 
830*4882a593Smuzhiyun 	destroy_workqueue(qm->wq);
831*4882a593Smuzhiyun }
832*4882a593Smuzhiyun 
sec_iommu_used_check(struct sec_dev * sec)833*4882a593Smuzhiyun static void sec_iommu_used_check(struct sec_dev *sec)
834*4882a593Smuzhiyun {
835*4882a593Smuzhiyun 	struct iommu_domain *domain;
836*4882a593Smuzhiyun 	struct device *dev = &sec->qm.pdev->dev;
837*4882a593Smuzhiyun 
838*4882a593Smuzhiyun 	domain = iommu_get_domain_for_dev(dev);
839*4882a593Smuzhiyun 
840*4882a593Smuzhiyun 	/* Check if iommu is used */
841*4882a593Smuzhiyun 	sec->iommu_used = false;
842*4882a593Smuzhiyun 	if (domain) {
843*4882a593Smuzhiyun 		if (domain->type & __IOMMU_DOMAIN_PAGING)
844*4882a593Smuzhiyun 			sec->iommu_used = true;
845*4882a593Smuzhiyun 		dev_info(dev, "SMMU Opened, the iommu type = %u\n",
846*4882a593Smuzhiyun 			domain->type);
847*4882a593Smuzhiyun 	}
848*4882a593Smuzhiyun }
849*4882a593Smuzhiyun 
sec_probe(struct pci_dev * pdev,const struct pci_device_id * id)850*4882a593Smuzhiyun static int sec_probe(struct pci_dev *pdev, const struct pci_device_id *id)
851*4882a593Smuzhiyun {
852*4882a593Smuzhiyun 	struct sec_dev *sec;
853*4882a593Smuzhiyun 	struct hisi_qm *qm;
854*4882a593Smuzhiyun 	int ret;
855*4882a593Smuzhiyun 
856*4882a593Smuzhiyun 	sec = devm_kzalloc(&pdev->dev, sizeof(*sec), GFP_KERNEL);
857*4882a593Smuzhiyun 	if (!sec)
858*4882a593Smuzhiyun 		return -ENOMEM;
859*4882a593Smuzhiyun 
860*4882a593Smuzhiyun 	qm = &sec->qm;
861*4882a593Smuzhiyun 	ret = sec_qm_init(qm, pdev);
862*4882a593Smuzhiyun 	if (ret) {
863*4882a593Smuzhiyun 		pci_err(pdev, "Failed to init SEC QM (%d)!\n", ret);
864*4882a593Smuzhiyun 		return ret;
865*4882a593Smuzhiyun 	}
866*4882a593Smuzhiyun 
867*4882a593Smuzhiyun 	sec->ctx_q_num = ctx_q_num;
868*4882a593Smuzhiyun 	sec_iommu_used_check(sec);
869*4882a593Smuzhiyun 
870*4882a593Smuzhiyun 	ret = sec_probe_init(sec);
871*4882a593Smuzhiyun 	if (ret) {
872*4882a593Smuzhiyun 		pci_err(pdev, "Failed to probe!\n");
873*4882a593Smuzhiyun 		goto err_qm_uninit;
874*4882a593Smuzhiyun 	}
875*4882a593Smuzhiyun 
876*4882a593Smuzhiyun 	ret = hisi_qm_start(qm);
877*4882a593Smuzhiyun 	if (ret) {
878*4882a593Smuzhiyun 		pci_err(pdev, "Failed to start sec qm!\n");
879*4882a593Smuzhiyun 		goto err_probe_uninit;
880*4882a593Smuzhiyun 	}
881*4882a593Smuzhiyun 
882*4882a593Smuzhiyun 	ret = sec_debugfs_init(qm);
883*4882a593Smuzhiyun 	if (ret)
884*4882a593Smuzhiyun 		pci_warn(pdev, "Failed to init debugfs!\n");
885*4882a593Smuzhiyun 
886*4882a593Smuzhiyun 	ret = hisi_qm_alg_register(qm, &sec_devices);
887*4882a593Smuzhiyun 	if (ret < 0) {
888*4882a593Smuzhiyun 		pr_err("Failed to register driver to crypto.\n");
889*4882a593Smuzhiyun 		goto err_qm_stop;
890*4882a593Smuzhiyun 	}
891*4882a593Smuzhiyun 
892*4882a593Smuzhiyun 	if (qm->fun_type == QM_HW_PF && vfs_num) {
893*4882a593Smuzhiyun 		ret = hisi_qm_sriov_enable(pdev, vfs_num);
894*4882a593Smuzhiyun 		if (ret < 0)
895*4882a593Smuzhiyun 			goto err_alg_unregister;
896*4882a593Smuzhiyun 	}
897*4882a593Smuzhiyun 
898*4882a593Smuzhiyun 	return 0;
899*4882a593Smuzhiyun 
900*4882a593Smuzhiyun err_alg_unregister:
901*4882a593Smuzhiyun 	hisi_qm_alg_unregister(qm, &sec_devices);
902*4882a593Smuzhiyun 
903*4882a593Smuzhiyun err_qm_stop:
904*4882a593Smuzhiyun 	sec_debugfs_exit(qm);
905*4882a593Smuzhiyun 	hisi_qm_stop(qm, QM_NORMAL);
906*4882a593Smuzhiyun 
907*4882a593Smuzhiyun err_probe_uninit:
908*4882a593Smuzhiyun 	sec_probe_uninit(qm);
909*4882a593Smuzhiyun 
910*4882a593Smuzhiyun err_qm_uninit:
911*4882a593Smuzhiyun 	sec_qm_uninit(qm);
912*4882a593Smuzhiyun 
913*4882a593Smuzhiyun 	return ret;
914*4882a593Smuzhiyun }
915*4882a593Smuzhiyun 
sec_remove(struct pci_dev * pdev)916*4882a593Smuzhiyun static void sec_remove(struct pci_dev *pdev)
917*4882a593Smuzhiyun {
918*4882a593Smuzhiyun 	struct hisi_qm *qm = pci_get_drvdata(pdev);
919*4882a593Smuzhiyun 
920*4882a593Smuzhiyun 	hisi_qm_wait_task_finish(qm, &sec_devices);
921*4882a593Smuzhiyun 	hisi_qm_alg_unregister(qm, &sec_devices);
922*4882a593Smuzhiyun 	if (qm->fun_type == QM_HW_PF && qm->vfs_num)
923*4882a593Smuzhiyun 		hisi_qm_sriov_disable(pdev, qm->is_frozen);
924*4882a593Smuzhiyun 
925*4882a593Smuzhiyun 	sec_debugfs_exit(qm);
926*4882a593Smuzhiyun 
927*4882a593Smuzhiyun 	(void)hisi_qm_stop(qm, QM_NORMAL);
928*4882a593Smuzhiyun 
929*4882a593Smuzhiyun 	if (qm->fun_type == QM_HW_PF)
930*4882a593Smuzhiyun 		sec_debug_regs_clear(qm);
931*4882a593Smuzhiyun 
932*4882a593Smuzhiyun 	sec_probe_uninit(qm);
933*4882a593Smuzhiyun 
934*4882a593Smuzhiyun 	sec_qm_uninit(qm);
935*4882a593Smuzhiyun }
936*4882a593Smuzhiyun 
937*4882a593Smuzhiyun static const struct pci_error_handlers sec_err_handler = {
938*4882a593Smuzhiyun 	.error_detected = hisi_qm_dev_err_detected,
939*4882a593Smuzhiyun 	.slot_reset =  hisi_qm_dev_slot_reset,
940*4882a593Smuzhiyun 	.reset_prepare		= hisi_qm_reset_prepare,
941*4882a593Smuzhiyun 	.reset_done		= hisi_qm_reset_done,
942*4882a593Smuzhiyun };
943*4882a593Smuzhiyun 
944*4882a593Smuzhiyun static struct pci_driver sec_pci_driver = {
945*4882a593Smuzhiyun 	.name = "hisi_sec2",
946*4882a593Smuzhiyun 	.id_table = sec_dev_ids,
947*4882a593Smuzhiyun 	.probe = sec_probe,
948*4882a593Smuzhiyun 	.remove = sec_remove,
949*4882a593Smuzhiyun 	.err_handler = &sec_err_handler,
950*4882a593Smuzhiyun 	.sriov_configure = hisi_qm_sriov_configure,
951*4882a593Smuzhiyun 	.shutdown = hisi_qm_dev_shutdown,
952*4882a593Smuzhiyun };
953*4882a593Smuzhiyun 
sec_register_debugfs(void)954*4882a593Smuzhiyun static void sec_register_debugfs(void)
955*4882a593Smuzhiyun {
956*4882a593Smuzhiyun 	if (!debugfs_initialized())
957*4882a593Smuzhiyun 		return;
958*4882a593Smuzhiyun 
959*4882a593Smuzhiyun 	sec_debugfs_root = debugfs_create_dir("hisi_sec2", NULL);
960*4882a593Smuzhiyun }
961*4882a593Smuzhiyun 
sec_unregister_debugfs(void)962*4882a593Smuzhiyun static void sec_unregister_debugfs(void)
963*4882a593Smuzhiyun {
964*4882a593Smuzhiyun 	debugfs_remove_recursive(sec_debugfs_root);
965*4882a593Smuzhiyun }
966*4882a593Smuzhiyun 
sec_init(void)967*4882a593Smuzhiyun static int __init sec_init(void)
968*4882a593Smuzhiyun {
969*4882a593Smuzhiyun 	int ret;
970*4882a593Smuzhiyun 
971*4882a593Smuzhiyun 	hisi_qm_init_list(&sec_devices);
972*4882a593Smuzhiyun 	sec_register_debugfs();
973*4882a593Smuzhiyun 
974*4882a593Smuzhiyun 	ret = pci_register_driver(&sec_pci_driver);
975*4882a593Smuzhiyun 	if (ret < 0) {
976*4882a593Smuzhiyun 		sec_unregister_debugfs();
977*4882a593Smuzhiyun 		pr_err("Failed to register pci driver.\n");
978*4882a593Smuzhiyun 		return ret;
979*4882a593Smuzhiyun 	}
980*4882a593Smuzhiyun 
981*4882a593Smuzhiyun 	return 0;
982*4882a593Smuzhiyun }
983*4882a593Smuzhiyun 
sec_exit(void)984*4882a593Smuzhiyun static void __exit sec_exit(void)
985*4882a593Smuzhiyun {
986*4882a593Smuzhiyun 	pci_unregister_driver(&sec_pci_driver);
987*4882a593Smuzhiyun 	sec_unregister_debugfs();
988*4882a593Smuzhiyun }
989*4882a593Smuzhiyun 
990*4882a593Smuzhiyun module_init(sec_init);
991*4882a593Smuzhiyun module_exit(sec_exit);
992*4882a593Smuzhiyun 
993*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
994*4882a593Smuzhiyun MODULE_AUTHOR("Zaibo Xu <xuzaibo@huawei.com>");
995*4882a593Smuzhiyun MODULE_AUTHOR("Longfang Liu <liulongfang@huawei.com>");
996*4882a593Smuzhiyun MODULE_AUTHOR("Kai Ye <yekai13@huawei.com>");
997*4882a593Smuzhiyun MODULE_AUTHOR("Wei Zhang <zhangwei375@huawei.com>");
998*4882a593Smuzhiyun MODULE_DESCRIPTION("Driver for HiSilicon SEC accelerator");
999