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