Path: blob/master/drivers/crypto/hisilicon/zip/zip_main.c
26292 views
// SPDX-License-Identifier: GPL-2.01/* Copyright (c) 2019 HiSilicon Limited. */2#include <linux/acpi.h>3#include <linux/bitops.h>4#include <linux/debugfs.h>5#include <linux/init.h>6#include <linux/io.h>7#include <linux/kernel.h>8#include <linux/module.h>9#include <linux/pci.h>10#include <linux/pm_runtime.h>11#include <linux/seq_file.h>12#include <linux/topology.h>13#include <linux/uacce.h>14#include "zip.h"1516#define CAP_FILE_PERMISSION 044417#define PCI_DEVICE_ID_HUAWEI_ZIP_PF 0xa2501819#define HZIP_QUEUE_NUM_V1 40962021#define HZIP_CLOCK_GATE_CTRL 0x30100422#define HZIP_DECOMP_CHECK_ENABLE BIT(16)23#define HZIP_FSM_MAX_CNT 0x3010082425#define HZIP_PORT_ARCA_CHE_0 0x30104026#define HZIP_PORT_ARCA_CHE_1 0x30104427#define HZIP_PORT_AWCA_CHE_0 0x30106028#define HZIP_PORT_AWCA_CHE_1 0x30106429#define HZIP_CACHE_ALL_EN 0xffffffff3031#define HZIP_BD_RUSER_32_63 0x30111032#define HZIP_SGL_RUSER_32_63 0x30111c33#define HZIP_DATA_RUSER_32_63 0x30112834#define HZIP_DATA_WUSER_32_63 0x30113435#define HZIP_BD_WUSER_32_63 0x3011403637#define HZIP_QM_IDEL_STATUS 0x3040e43839#define HZIP_CORE_DFX_BASE 0x30100040#define HZIP_CORE_DFX_DECOMP_BASE 0x30400041#define HZIP_CORE_DFX_COMP_0 0x30200042#define HZIP_CORE_DFX_COMP_1 0x30300043#define HZIP_CORE_DFX_DECOMP_0 0x30400044#define HZIP_CORE_DFX_DECOMP_1 0x30500045#define HZIP_CORE_DFX_DECOMP_2 0x30600046#define HZIP_CORE_DFX_DECOMP_3 0x30700047#define HZIP_CORE_DFX_DECOMP_4 0x30800048#define HZIP_CORE_DFX_DECOMP_5 0x30900049#define HZIP_CORE_REGS_BASE_LEN 0xB050#define HZIP_CORE_REGS_DFX_LEN 0x2851#define HZIP_CORE_ADDR_INTRVL 0x10005253#define HZIP_CORE_INT_SOURCE 0x3010A054#define HZIP_CORE_INT_MASK_REG 0x3010A455#define HZIP_CORE_INT_SET 0x3010A856#define HZIP_CORE_INT_STATUS 0x3010AC57#define HZIP_CORE_INT_STATUS_M_ECC BIT(1)58#define HZIP_CORE_SRAM_ECC_ERR_INFO 0x30114859#define HZIP_CORE_INT_RAS_CE_ENB 0x30116060#define HZIP_CORE_INT_RAS_NFE_ENB 0x30116461#define HZIP_CORE_INT_RAS_FE_ENB 0x30116862#define HZIP_CORE_INT_RAS_FE_ENB_MASK 0x063#define HZIP_OOO_SHUTDOWN_SEL 0x30120C64#define HZIP_SRAM_ECC_ERR_NUM_SHIFT 1665#define HZIP_SRAM_ECC_ERR_ADDR_SHIFT 2466#define HZIP_CORE_INT_MASK_ALL GENMASK(12, 0)67#define HZIP_SQE_SIZE 12868#define HZIP_PF_DEF_Q_NUM 6469#define HZIP_PF_DEF_Q_BASE 070#define HZIP_CTX_Q_NUM_DEF 27172#define HZIP_SOFT_CTRL_CNT_CLR_CE 0x30100073#define HZIP_SOFT_CTRL_CNT_CLR_CE_BIT BIT(0)74#define HZIP_SOFT_CTRL_ZIP_CONTROL 0x30100C75#define HZIP_AXI_SHUTDOWN_ENABLE BIT(14)76#define HZIP_WR_PORT BIT(11)7778#define HZIP_ALG_ZLIB_BIT GENMASK(1, 0)79#define HZIP_ALG_GZIP_BIT GENMASK(3, 2)80#define HZIP_ALG_DEFLATE_BIT GENMASK(5, 4)81#define HZIP_ALG_LZ77_BIT GENMASK(7, 6)8283#define HZIP_BUF_SIZE 2284#define HZIP_SQE_MASK_OFFSET 6485#define HZIP_SQE_MASK_LEN 488687#define HZIP_CNT_CLR_CE_EN BIT(0)88#define HZIP_RO_CNT_CLR_CE_EN BIT(2)89#define HZIP_RD_CNT_CLR_CE_EN (HZIP_CNT_CLR_CE_EN | \90HZIP_RO_CNT_CLR_CE_EN)9192#define HZIP_PREFETCH_CFG 0x3011B093#define HZIP_SVA_TRANS 0x3011C494#define HZIP_PREFETCH_ENABLE (~(BIT(26) | BIT(17) | BIT(0)))95#define HZIP_SVA_PREFETCH_DISABLE BIT(26)96#define HZIP_SVA_DISABLE_READY (BIT(26) | BIT(30))97#define HZIP_SHAPER_RATE_COMPRESS 75098#define HZIP_SHAPER_RATE_DECOMPRESS 14099#define HZIP_DELAY_1_US 1100#define HZIP_POLL_TIMEOUT_US 1000101102/* clock gating */103#define HZIP_PEH_CFG_AUTO_GATE 0x3011A8104#define HZIP_PEH_CFG_AUTO_GATE_EN BIT(0)105#define HZIP_CORE_GATED_EN GENMASK(15, 8)106#define HZIP_CORE_GATED_OOO_EN BIT(29)107#define HZIP_CLOCK_GATED_EN (HZIP_CORE_GATED_EN | \108HZIP_CORE_GATED_OOO_EN)109110/* zip comp high performance */111#define HZIP_HIGH_PERF_OFFSET 0x301208112113enum {114HZIP_HIGH_COMP_RATE,115HZIP_HIGH_COMP_PERF,116};117118static const char hisi_zip_name[] = "hisi_zip";119static struct dentry *hzip_debugfs_root;120121struct hisi_zip_hw_error {122u32 int_msk;123const char *msg;124};125126struct zip_dfx_item {127const char *name;128u32 offset;129};130131static const struct qm_dev_alg zip_dev_algs[] = { {132.alg_msk = HZIP_ALG_ZLIB_BIT,133.alg = "zlib\n",134}, {135.alg_msk = HZIP_ALG_GZIP_BIT,136.alg = "gzip\n",137}, {138.alg_msk = HZIP_ALG_DEFLATE_BIT,139.alg = "deflate\n",140}, {141.alg_msk = HZIP_ALG_LZ77_BIT,142.alg = "lz77_zstd\n",143},144};145146static struct hisi_qm_list zip_devices = {147.register_to_crypto = hisi_zip_register_to_crypto,148.unregister_from_crypto = hisi_zip_unregister_from_crypto,149};150151static struct zip_dfx_item zip_dfx_files[] = {152{"send_cnt", offsetof(struct hisi_zip_dfx, send_cnt)},153{"recv_cnt", offsetof(struct hisi_zip_dfx, recv_cnt)},154{"send_busy_cnt", offsetof(struct hisi_zip_dfx, send_busy_cnt)},155{"err_bd_cnt", offsetof(struct hisi_zip_dfx, err_bd_cnt)},156};157158static const struct hisi_zip_hw_error zip_hw_error[] = {159{ .int_msk = BIT(0), .msg = "zip_ecc_1bitt_err" },160{ .int_msk = BIT(1), .msg = "zip_ecc_2bit_err" },161{ .int_msk = BIT(2), .msg = "zip_axi_rresp_err" },162{ .int_msk = BIT(3), .msg = "zip_axi_bresp_err" },163{ .int_msk = BIT(4), .msg = "zip_src_addr_parse_err" },164{ .int_msk = BIT(5), .msg = "zip_dst_addr_parse_err" },165{ .int_msk = BIT(6), .msg = "zip_pre_in_addr_err" },166{ .int_msk = BIT(7), .msg = "zip_pre_in_data_err" },167{ .int_msk = BIT(8), .msg = "zip_com_inf_err" },168{ .int_msk = BIT(9), .msg = "zip_enc_inf_err" },169{ .int_msk = BIT(10), .msg = "zip_pre_out_err" },170{ .int_msk = BIT(11), .msg = "zip_axi_poison_err" },171{ .int_msk = BIT(12), .msg = "zip_sva_err" },172{ /* sentinel */ }173};174175enum ctrl_debug_file_index {176HZIP_CLEAR_ENABLE,177HZIP_DEBUG_FILE_NUM,178};179180static const char * const ctrl_debug_file_name[] = {181[HZIP_CLEAR_ENABLE] = "clear_enable",182};183184struct ctrl_debug_file {185enum ctrl_debug_file_index index;186spinlock_t lock;187struct hisi_zip_ctrl *ctrl;188};189190/*191* One ZIP controller has one PF and multiple VFs, some global configurations192* which PF has need this structure.193*194* Just relevant for PF.195*/196struct hisi_zip_ctrl {197struct hisi_zip *hisi_zip;198struct ctrl_debug_file files[HZIP_DEBUG_FILE_NUM];199};200201enum zip_cap_type {202ZIP_QM_NFE_MASK_CAP = 0x0,203ZIP_QM_RESET_MASK_CAP,204ZIP_QM_OOO_SHUTDOWN_MASK_CAP,205ZIP_QM_CE_MASK_CAP,206ZIP_NFE_MASK_CAP,207ZIP_RESET_MASK_CAP,208ZIP_OOO_SHUTDOWN_MASK_CAP,209ZIP_CE_MASK_CAP,210ZIP_CLUSTER_NUM_CAP,211ZIP_CORE_TYPE_NUM_CAP,212ZIP_CORE_NUM_CAP,213ZIP_CLUSTER_COMP_NUM_CAP,214ZIP_CLUSTER_DECOMP_NUM_CAP,215ZIP_DECOMP_ENABLE_BITMAP,216ZIP_COMP_ENABLE_BITMAP,217ZIP_DRV_ALG_BITMAP,218ZIP_DEV_ALG_BITMAP,219ZIP_CORE1_ALG_BITMAP,220ZIP_CORE2_ALG_BITMAP,221ZIP_CORE3_ALG_BITMAP,222ZIP_CORE4_ALG_BITMAP,223ZIP_CORE5_ALG_BITMAP,224ZIP_CAP_MAX225};226227static struct hisi_qm_cap_info zip_basic_cap_info[] = {228{ZIP_QM_NFE_MASK_CAP, 0x3124, 0, GENMASK(31, 0), 0x0, 0x1C57, 0x7C77},229{ZIP_QM_RESET_MASK_CAP, 0x3128, 0, GENMASK(31, 0), 0x0, 0xC57, 0x6C77},230{ZIP_QM_OOO_SHUTDOWN_MASK_CAP, 0x3128, 0, GENMASK(31, 0), 0x0, 0x4, 0x6C77},231{ZIP_QM_CE_MASK_CAP, 0x312C, 0, GENMASK(31, 0), 0x0, 0x8, 0x8},232{ZIP_NFE_MASK_CAP, 0x3130, 0, GENMASK(31, 0), 0x0, 0x7FE, 0x1FFE},233{ZIP_RESET_MASK_CAP, 0x3134, 0, GENMASK(31, 0), 0x0, 0x7FE, 0x7FE},234{ZIP_OOO_SHUTDOWN_MASK_CAP, 0x3134, 0, GENMASK(31, 0), 0x0, 0x2, 0x7FE},235{ZIP_CE_MASK_CAP, 0x3138, 0, GENMASK(31, 0), 0x0, 0x1, 0x1},236{ZIP_CLUSTER_NUM_CAP, 0x313C, 28, GENMASK(3, 0), 0x1, 0x1, 0x1},237{ZIP_CORE_TYPE_NUM_CAP, 0x313C, 24, GENMASK(3, 0), 0x2, 0x2, 0x2},238{ZIP_CORE_NUM_CAP, 0x313C, 16, GENMASK(7, 0), 0x8, 0x8, 0x5},239{ZIP_CLUSTER_COMP_NUM_CAP, 0x313C, 8, GENMASK(7, 0), 0x2, 0x2, 0x2},240{ZIP_CLUSTER_DECOMP_NUM_CAP, 0x313C, 0, GENMASK(7, 0), 0x6, 0x6, 0x3},241{ZIP_DECOMP_ENABLE_BITMAP, 0x3140, 16, GENMASK(15, 0), 0xFC, 0xFC, 0x1C},242{ZIP_COMP_ENABLE_BITMAP, 0x3140, 0, GENMASK(15, 0), 0x3, 0x3, 0x3},243{ZIP_DRV_ALG_BITMAP, 0x3144, 0, GENMASK(31, 0), 0x0, 0x0, 0x30},244{ZIP_DEV_ALG_BITMAP, 0x3148, 0, GENMASK(31, 0), 0xF, 0xF, 0x3F},245{ZIP_CORE1_ALG_BITMAP, 0x314C, 0, GENMASK(31, 0), 0x5, 0x5, 0xD5},246{ZIP_CORE2_ALG_BITMAP, 0x3150, 0, GENMASK(31, 0), 0x5, 0x5, 0xD5},247{ZIP_CORE3_ALG_BITMAP, 0x3154, 0, GENMASK(31, 0), 0xA, 0xA, 0x2A},248{ZIP_CORE4_ALG_BITMAP, 0x3158, 0, GENMASK(31, 0), 0xA, 0xA, 0x2A},249{ZIP_CORE5_ALG_BITMAP, 0x315C, 0, GENMASK(31, 0), 0xA, 0xA, 0x2A},250{ZIP_CAP_MAX, 0x317c, 0, GENMASK(0, 0), 0x0, 0x0, 0x0}251};252253static const struct hisi_qm_cap_query_info zip_cap_query_info[] = {254{QM_RAS_NFE_TYPE, "QM_RAS_NFE_TYPE ", 0x3124, 0x0, 0x1C57, 0x7C77},255{QM_RAS_NFE_RESET, "QM_RAS_NFE_RESET ", 0x3128, 0x0, 0xC57, 0x6C77},256{QM_RAS_CE_TYPE, "QM_RAS_CE_TYPE ", 0x312C, 0x0, 0x8, 0x8},257{ZIP_RAS_NFE_TYPE, "ZIP_RAS_NFE_TYPE ", 0x3130, 0x0, 0x7FE, 0x1FFE},258{ZIP_RAS_NFE_RESET, "ZIP_RAS_NFE_RESET ", 0x3134, 0x0, 0x7FE, 0x7FE},259{ZIP_RAS_CE_TYPE, "ZIP_RAS_CE_TYPE ", 0x3138, 0x0, 0x1, 0x1},260{ZIP_CORE_INFO, "ZIP_CORE_INFO ", 0x313C, 0x12080206, 0x12080206, 0x12050203},261{ZIP_CORE_EN, "ZIP_CORE_EN ", 0x3140, 0xFC0003, 0xFC0003, 0x1C0003},262{ZIP_DRV_ALG_BITMAP_TB, "ZIP_DRV_ALG_BITMAP ", 0x3144, 0x0, 0x0, 0x30},263{ZIP_ALG_BITMAP, "ZIP_ALG_BITMAP ", 0x3148, 0xF, 0xF, 0x3F},264{ZIP_CORE1_BITMAP, "ZIP_CORE1_BITMAP ", 0x314C, 0x5, 0x5, 0xD5},265{ZIP_CORE2_BITMAP, "ZIP_CORE2_BITMAP ", 0x3150, 0x5, 0x5, 0xD5},266{ZIP_CORE3_BITMAP, "ZIP_CORE3_BITMAP ", 0x3154, 0xA, 0xA, 0x2A},267{ZIP_CORE4_BITMAP, "ZIP_CORE4_BITMAP ", 0x3158, 0xA, 0xA, 0x2A},268{ZIP_CORE5_BITMAP, "ZIP_CORE5_BITMAP ", 0x315C, 0xA, 0xA, 0x2A},269};270271static const struct debugfs_reg32 hzip_dfx_regs[] = {272{"HZIP_GET_BD_NUM ", 0x00},273{"HZIP_GET_RIGHT_BD ", 0x04},274{"HZIP_GET_ERROR_BD ", 0x08},275{"HZIP_DONE_BD_NUM ", 0x0c},276{"HZIP_WORK_CYCLE ", 0x10},277{"HZIP_IDLE_CYCLE ", 0x18},278{"HZIP_MAX_DELAY ", 0x20},279{"HZIP_MIN_DELAY ", 0x24},280{"HZIP_AVG_DELAY ", 0x28},281{"HZIP_MEM_VISIBLE_DATA ", 0x30},282{"HZIP_MEM_VISIBLE_ADDR ", 0x34},283{"HZIP_CONSUMED_BYTE ", 0x38},284{"HZIP_PRODUCED_BYTE ", 0x40},285{"HZIP_COMP_INF ", 0x70},286{"HZIP_PRE_OUT ", 0x78},287{"HZIP_BD_RD ", 0x7c},288{"HZIP_BD_WR ", 0x80},289{"HZIP_GET_BD_AXI_ERR_NUM ", 0x84},290{"HZIP_GET_BD_PARSE_ERR_NUM ", 0x88},291{"HZIP_ADD_BD_AXI_ERR_NUM ", 0x8c},292{"HZIP_DECOMP_STF_RELOAD_CURR_ST ", 0x94},293{"HZIP_DECOMP_LZ77_CURR_ST ", 0x9c},294};295296static const struct debugfs_reg32 hzip_com_dfx_regs[] = {297{"HZIP_CLOCK_GATE_CTRL ", 0x301004},298{"HZIP_CORE_INT_RAS_CE_ENB ", 0x301160},299{"HZIP_CORE_INT_RAS_NFE_ENB ", 0x301164},300{"HZIP_CORE_INT_RAS_FE_ENB ", 0x301168},301{"HZIP_UNCOM_ERR_RAS_CTRL ", 0x30116C},302};303304static const struct debugfs_reg32 hzip_dump_dfx_regs[] = {305{"HZIP_GET_BD_NUM ", 0x00},306{"HZIP_GET_RIGHT_BD ", 0x04},307{"HZIP_GET_ERROR_BD ", 0x08},308{"HZIP_DONE_BD_NUM ", 0x0c},309{"HZIP_MAX_DELAY ", 0x20},310};311312/* define the ZIP's dfx regs region and region length */313static struct dfx_diff_registers hzip_diff_regs[] = {314{315.reg_offset = HZIP_CORE_DFX_BASE,316.reg_len = HZIP_CORE_REGS_BASE_LEN,317}, {318.reg_offset = HZIP_CORE_DFX_COMP_0,319.reg_len = HZIP_CORE_REGS_DFX_LEN,320}, {321.reg_offset = HZIP_CORE_DFX_COMP_1,322.reg_len = HZIP_CORE_REGS_DFX_LEN,323}, {324.reg_offset = HZIP_CORE_DFX_DECOMP_0,325.reg_len = HZIP_CORE_REGS_DFX_LEN,326}, {327.reg_offset = HZIP_CORE_DFX_DECOMP_1,328.reg_len = HZIP_CORE_REGS_DFX_LEN,329}, {330.reg_offset = HZIP_CORE_DFX_DECOMP_2,331.reg_len = HZIP_CORE_REGS_DFX_LEN,332}, {333.reg_offset = HZIP_CORE_DFX_DECOMP_3,334.reg_len = HZIP_CORE_REGS_DFX_LEN,335}, {336.reg_offset = HZIP_CORE_DFX_DECOMP_4,337.reg_len = HZIP_CORE_REGS_DFX_LEN,338}, {339.reg_offset = HZIP_CORE_DFX_DECOMP_5,340.reg_len = HZIP_CORE_REGS_DFX_LEN,341},342};343344static int hzip_diff_regs_show(struct seq_file *s, void *unused)345{346struct hisi_qm *qm = s->private;347348hisi_qm_acc_diff_regs_dump(qm, s, qm->debug.acc_diff_regs,349ARRAY_SIZE(hzip_diff_regs));350351return 0;352}353DEFINE_SHOW_ATTRIBUTE(hzip_diff_regs);354355static int perf_mode_set(const char *val, const struct kernel_param *kp)356{357int ret;358u32 n;359360if (!val)361return -EINVAL;362363ret = kstrtou32(val, 10, &n);364if (ret != 0 || (n != HZIP_HIGH_COMP_PERF &&365n != HZIP_HIGH_COMP_RATE))366return -EINVAL;367368return param_set_int(val, kp);369}370371static const struct kernel_param_ops zip_com_perf_ops = {372.set = perf_mode_set,373.get = param_get_int,374};375376/*377* perf_mode = 0 means enable high compression rate mode,378* perf_mode = 1 means enable high compression performance mode.379* These two modes only apply to the compression direction.380*/381static u32 perf_mode = HZIP_HIGH_COMP_RATE;382module_param_cb(perf_mode, &zip_com_perf_ops, &perf_mode, 0444);383MODULE_PARM_DESC(perf_mode, "ZIP high perf mode 0(default), 1(enable)");384385static const struct kernel_param_ops zip_uacce_mode_ops = {386.set = uacce_mode_set,387.get = param_get_int,388};389390/*391* uacce_mode = 0 means zip only register to crypto,392* uacce_mode = 1 means zip both register to crypto and uacce.393*/394static u32 uacce_mode = UACCE_MODE_NOUACCE;395module_param_cb(uacce_mode, &zip_uacce_mode_ops, &uacce_mode, 0444);396MODULE_PARM_DESC(uacce_mode, UACCE_MODE_DESC);397398static bool pf_q_num_flag;399static int pf_q_num_set(const char *val, const struct kernel_param *kp)400{401pf_q_num_flag = true;402403return hisi_qm_q_num_set(val, kp, PCI_DEVICE_ID_HUAWEI_ZIP_PF);404}405406static const struct kernel_param_ops pf_q_num_ops = {407.set = pf_q_num_set,408.get = param_get_int,409};410411static u32 pf_q_num = HZIP_PF_DEF_Q_NUM;412module_param_cb(pf_q_num, &pf_q_num_ops, &pf_q_num, 0444);413MODULE_PARM_DESC(pf_q_num, "Number of queues in PF(v1 2-4096, v2 2-1024)");414415static const struct kernel_param_ops vfs_num_ops = {416.set = vfs_num_set,417.get = param_get_int,418};419420static u32 vfs_num;421module_param_cb(vfs_num, &vfs_num_ops, &vfs_num, 0444);422MODULE_PARM_DESC(vfs_num, "Number of VFs to enable(1-63), 0(default)");423424static const struct pci_device_id hisi_zip_dev_ids[] = {425{ PCI_DEVICE(PCI_VENDOR_ID_HUAWEI, PCI_DEVICE_ID_HUAWEI_ZIP_PF) },426{ PCI_DEVICE(PCI_VENDOR_ID_HUAWEI, PCI_DEVICE_ID_HUAWEI_ZIP_VF) },427{ 0, }428};429MODULE_DEVICE_TABLE(pci, hisi_zip_dev_ids);430431int zip_create_qps(struct hisi_qp **qps, int qp_num, int node)432{433if (node == NUMA_NO_NODE)434node = cpu_to_node(raw_smp_processor_id());435436return hisi_qm_alloc_qps_node(&zip_devices, qp_num, 0, node, qps);437}438439bool hisi_zip_alg_support(struct hisi_qm *qm, u32 alg)440{441u32 cap_val;442443cap_val = qm->cap_tables.dev_cap_table[ZIP_DRV_ALG_BITMAP_TB].cap_val;444if ((alg & cap_val) == alg)445return true;446447return false;448}449450static int hisi_zip_set_high_perf(struct hisi_qm *qm)451{452u32 val;453int ret;454455val = readl_relaxed(qm->io_base + HZIP_HIGH_PERF_OFFSET);456if (perf_mode == HZIP_HIGH_COMP_PERF)457val |= HZIP_HIGH_COMP_PERF;458else459val &= ~HZIP_HIGH_COMP_PERF;460461/* Set perf mode */462writel(val, qm->io_base + HZIP_HIGH_PERF_OFFSET);463ret = readl_relaxed_poll_timeout(qm->io_base + HZIP_HIGH_PERF_OFFSET,464val, val == perf_mode, HZIP_DELAY_1_US,465HZIP_POLL_TIMEOUT_US);466if (ret)467pci_err(qm->pdev, "failed to set perf mode\n");468469return ret;470}471472static void hisi_zip_open_sva_prefetch(struct hisi_qm *qm)473{474u32 val;475int ret;476477if (!test_bit(QM_SUPPORT_SVA_PREFETCH, &qm->caps))478return;479480/* Enable prefetch */481val = readl_relaxed(qm->io_base + HZIP_PREFETCH_CFG);482val &= HZIP_PREFETCH_ENABLE;483writel(val, qm->io_base + HZIP_PREFETCH_CFG);484485ret = readl_relaxed_poll_timeout(qm->io_base + HZIP_PREFETCH_CFG,486val, !(val & HZIP_SVA_PREFETCH_DISABLE),487HZIP_DELAY_1_US, HZIP_POLL_TIMEOUT_US);488if (ret)489pci_err(qm->pdev, "failed to open sva prefetch\n");490}491492static void hisi_zip_close_sva_prefetch(struct hisi_qm *qm)493{494u32 val;495int ret;496497if (!test_bit(QM_SUPPORT_SVA_PREFETCH, &qm->caps))498return;499500val = readl_relaxed(qm->io_base + HZIP_PREFETCH_CFG);501val |= HZIP_SVA_PREFETCH_DISABLE;502writel(val, qm->io_base + HZIP_PREFETCH_CFG);503504ret = readl_relaxed_poll_timeout(qm->io_base + HZIP_SVA_TRANS,505val, !(val & HZIP_SVA_DISABLE_READY),506HZIP_DELAY_1_US, HZIP_POLL_TIMEOUT_US);507if (ret)508pci_err(qm->pdev, "failed to close sva prefetch\n");509}510511static void hisi_zip_enable_clock_gate(struct hisi_qm *qm)512{513u32 val;514515if (qm->ver < QM_HW_V3)516return;517518val = readl(qm->io_base + HZIP_CLOCK_GATE_CTRL);519val |= HZIP_CLOCK_GATED_EN;520writel(val, qm->io_base + HZIP_CLOCK_GATE_CTRL);521522val = readl(qm->io_base + HZIP_PEH_CFG_AUTO_GATE);523val |= HZIP_PEH_CFG_AUTO_GATE_EN;524writel(val, qm->io_base + HZIP_PEH_CFG_AUTO_GATE);525}526527static int hisi_zip_set_user_domain_and_cache(struct hisi_qm *qm)528{529void __iomem *base = qm->io_base;530u32 dcomp_bm, comp_bm;531u32 zip_core_en;532533/* qm user domain */534writel(AXUSER_BASE, base + QM_ARUSER_M_CFG_1);535writel(ARUSER_M_CFG_ENABLE, base + QM_ARUSER_M_CFG_ENABLE);536writel(AXUSER_BASE, base + QM_AWUSER_M_CFG_1);537writel(AWUSER_M_CFG_ENABLE, base + QM_AWUSER_M_CFG_ENABLE);538writel(WUSER_M_CFG_ENABLE, base + QM_WUSER_M_CFG_ENABLE);539540/* qm cache */541writel(AXI_M_CFG, base + QM_AXI_M_CFG);542writel(AXI_M_CFG_ENABLE, base + QM_AXI_M_CFG_ENABLE);543544/* disable FLR triggered by BME(bus master enable) */545writel(PEH_AXUSER_CFG, base + QM_PEH_AXUSER_CFG);546writel(PEH_AXUSER_CFG_ENABLE, base + QM_PEH_AXUSER_CFG_ENABLE);547548/* cache */549writel(HZIP_CACHE_ALL_EN, base + HZIP_PORT_ARCA_CHE_0);550writel(HZIP_CACHE_ALL_EN, base + HZIP_PORT_ARCA_CHE_1);551writel(HZIP_CACHE_ALL_EN, base + HZIP_PORT_AWCA_CHE_0);552writel(HZIP_CACHE_ALL_EN, base + HZIP_PORT_AWCA_CHE_1);553554/* user domain configurations */555writel(AXUSER_BASE, base + HZIP_BD_RUSER_32_63);556writel(AXUSER_BASE, base + HZIP_BD_WUSER_32_63);557558if (qm->use_sva && qm->ver == QM_HW_V2) {559writel(AXUSER_BASE | AXUSER_SSV, base + HZIP_DATA_RUSER_32_63);560writel(AXUSER_BASE | AXUSER_SSV, base + HZIP_DATA_WUSER_32_63);561writel(AXUSER_BASE | AXUSER_SSV, base + HZIP_SGL_RUSER_32_63);562} else {563writel(AXUSER_BASE, base + HZIP_DATA_RUSER_32_63);564writel(AXUSER_BASE, base + HZIP_DATA_WUSER_32_63);565writel(AXUSER_BASE, base + HZIP_SGL_RUSER_32_63);566}567568/* let's open all compression/decompression cores */569570zip_core_en = qm->cap_tables.dev_cap_table[ZIP_CORE_EN].cap_val;571dcomp_bm = (zip_core_en >> zip_basic_cap_info[ZIP_DECOMP_ENABLE_BITMAP].shift) &572zip_basic_cap_info[ZIP_DECOMP_ENABLE_BITMAP].mask;573comp_bm = (zip_core_en >> zip_basic_cap_info[ZIP_COMP_ENABLE_BITMAP].shift) &574zip_basic_cap_info[ZIP_COMP_ENABLE_BITMAP].mask;575writel(HZIP_DECOMP_CHECK_ENABLE | dcomp_bm | comp_bm, base + HZIP_CLOCK_GATE_CTRL);576577/* enable sqc,cqc writeback */578writel(SQC_CACHE_ENABLE | CQC_CACHE_ENABLE | SQC_CACHE_WB_ENABLE |579CQC_CACHE_WB_ENABLE | FIELD_PREP(SQC_CACHE_WB_THRD, 1) |580FIELD_PREP(CQC_CACHE_WB_THRD, 1), base + QM_CACHE_CTL);581582hisi_zip_enable_clock_gate(qm);583584return hisi_dae_set_user_domain(qm);585}586587static void hisi_zip_master_ooo_ctrl(struct hisi_qm *qm, bool enable)588{589u32 val1, val2;590591val1 = readl(qm->io_base + HZIP_SOFT_CTRL_ZIP_CONTROL);592if (enable) {593val1 |= HZIP_AXI_SHUTDOWN_ENABLE;594val2 = hisi_qm_get_hw_info(qm, zip_basic_cap_info,595ZIP_OOO_SHUTDOWN_MASK_CAP, qm->cap_ver);596} else {597val1 &= ~HZIP_AXI_SHUTDOWN_ENABLE;598val2 = 0x0;599}600601if (qm->ver > QM_HW_V2)602writel(val2, qm->io_base + HZIP_OOO_SHUTDOWN_SEL);603604writel(val1, qm->io_base + HZIP_SOFT_CTRL_ZIP_CONTROL);605}606607static void hisi_zip_hw_error_enable(struct hisi_qm *qm)608{609u32 nfe, ce;610611if (qm->ver == QM_HW_V1) {612writel(HZIP_CORE_INT_MASK_ALL,613qm->io_base + HZIP_CORE_INT_MASK_REG);614dev_info(&qm->pdev->dev, "Does not support hw error handle\n");615return;616}617618nfe = hisi_qm_get_hw_info(qm, zip_basic_cap_info, ZIP_NFE_MASK_CAP, qm->cap_ver);619ce = hisi_qm_get_hw_info(qm, zip_basic_cap_info, ZIP_CE_MASK_CAP, qm->cap_ver);620621/* clear ZIP hw error source if having */622writel(ce | nfe | HZIP_CORE_INT_RAS_FE_ENB_MASK, qm->io_base + HZIP_CORE_INT_SOURCE);623624/* configure error type */625writel(ce, qm->io_base + HZIP_CORE_INT_RAS_CE_ENB);626writel(HZIP_CORE_INT_RAS_FE_ENB_MASK, qm->io_base + HZIP_CORE_INT_RAS_FE_ENB);627writel(nfe, qm->io_base + HZIP_CORE_INT_RAS_NFE_ENB);628629hisi_zip_master_ooo_ctrl(qm, true);630631/* enable ZIP hw error interrupts */632writel(0, qm->io_base + HZIP_CORE_INT_MASK_REG);633634hisi_dae_hw_error_enable(qm);635}636637static void hisi_zip_hw_error_disable(struct hisi_qm *qm)638{639u32 nfe, ce;640641/* disable ZIP hw error interrupts */642nfe = hisi_qm_get_hw_info(qm, zip_basic_cap_info, ZIP_NFE_MASK_CAP, qm->cap_ver);643ce = hisi_qm_get_hw_info(qm, zip_basic_cap_info, ZIP_CE_MASK_CAP, qm->cap_ver);644writel(ce | nfe | HZIP_CORE_INT_RAS_FE_ENB_MASK, qm->io_base + HZIP_CORE_INT_MASK_REG);645646hisi_zip_master_ooo_ctrl(qm, false);647648hisi_dae_hw_error_disable(qm);649}650651static inline struct hisi_qm *file_to_qm(struct ctrl_debug_file *file)652{653struct hisi_zip *hisi_zip = file->ctrl->hisi_zip;654655return &hisi_zip->qm;656}657658static u32 clear_enable_read(struct hisi_qm *qm)659{660return readl(qm->io_base + HZIP_SOFT_CTRL_CNT_CLR_CE) &661HZIP_SOFT_CTRL_CNT_CLR_CE_BIT;662}663664static int clear_enable_write(struct hisi_qm *qm, u32 val)665{666u32 tmp;667668if (val != 1 && val != 0)669return -EINVAL;670671tmp = (readl(qm->io_base + HZIP_SOFT_CTRL_CNT_CLR_CE) &672~HZIP_SOFT_CTRL_CNT_CLR_CE_BIT) | val;673writel(tmp, qm->io_base + HZIP_SOFT_CTRL_CNT_CLR_CE);674675return 0;676}677678static ssize_t hisi_zip_ctrl_debug_read(struct file *filp, char __user *buf,679size_t count, loff_t *pos)680{681struct ctrl_debug_file *file = filp->private_data;682struct hisi_qm *qm = file_to_qm(file);683char tbuf[HZIP_BUF_SIZE];684u32 val;685int ret;686687ret = hisi_qm_get_dfx_access(qm);688if (ret)689return ret;690691spin_lock_irq(&file->lock);692switch (file->index) {693case HZIP_CLEAR_ENABLE:694val = clear_enable_read(qm);695break;696default:697goto err_input;698}699spin_unlock_irq(&file->lock);700701hisi_qm_put_dfx_access(qm);702ret = scnprintf(tbuf, sizeof(tbuf), "%u\n", val);703return simple_read_from_buffer(buf, count, pos, tbuf, ret);704705err_input:706spin_unlock_irq(&file->lock);707hisi_qm_put_dfx_access(qm);708return -EINVAL;709}710711static ssize_t hisi_zip_ctrl_debug_write(struct file *filp,712const char __user *buf,713size_t count, loff_t *pos)714{715struct ctrl_debug_file *file = filp->private_data;716struct hisi_qm *qm = file_to_qm(file);717char tbuf[HZIP_BUF_SIZE];718unsigned long val;719int len, ret;720721if (*pos != 0)722return 0;723724if (count >= HZIP_BUF_SIZE)725return -ENOSPC;726727len = simple_write_to_buffer(tbuf, HZIP_BUF_SIZE - 1, pos, buf, count);728if (len < 0)729return len;730731tbuf[len] = '\0';732ret = kstrtoul(tbuf, 0, &val);733if (ret)734return ret;735736ret = hisi_qm_get_dfx_access(qm);737if (ret)738return ret;739740spin_lock_irq(&file->lock);741switch (file->index) {742case HZIP_CLEAR_ENABLE:743ret = clear_enable_write(qm, val);744if (ret)745goto err_input;746break;747default:748ret = -EINVAL;749goto err_input;750}751752ret = count;753754err_input:755spin_unlock_irq(&file->lock);756hisi_qm_put_dfx_access(qm);757return ret;758}759760static const struct file_operations ctrl_debug_fops = {761.owner = THIS_MODULE,762.open = simple_open,763.read = hisi_zip_ctrl_debug_read,764.write = hisi_zip_ctrl_debug_write,765};766767static int zip_debugfs_atomic64_set(void *data, u64 val)768{769if (val)770return -EINVAL;771772atomic64_set((atomic64_t *)data, 0);773774return 0;775}776777static int zip_debugfs_atomic64_get(void *data, u64 *val)778{779*val = atomic64_read((atomic64_t *)data);780781return 0;782}783784DEFINE_DEBUGFS_ATTRIBUTE(zip_atomic64_ops, zip_debugfs_atomic64_get,785zip_debugfs_atomic64_set, "%llu\n");786787static int hisi_zip_regs_show(struct seq_file *s, void *unused)788{789hisi_qm_regs_dump(s, s->private);790791return 0;792}793794DEFINE_SHOW_ATTRIBUTE(hisi_zip_regs);795796static void __iomem *get_zip_core_addr(struct hisi_qm *qm, int core_num)797{798u8 zip_comp_core_num;799u32 zip_core_info;800801zip_core_info = qm->cap_tables.dev_cap_table[ZIP_CORE_INFO].cap_val;802zip_comp_core_num = (zip_core_info >> zip_basic_cap_info[ZIP_CLUSTER_COMP_NUM_CAP].shift) &803zip_basic_cap_info[ZIP_CLUSTER_COMP_NUM_CAP].mask;804805if (core_num < zip_comp_core_num)806return qm->io_base + HZIP_CORE_DFX_BASE +807(core_num + 1) * HZIP_CORE_ADDR_INTRVL;808809return qm->io_base + HZIP_CORE_DFX_DECOMP_BASE +810(core_num - zip_comp_core_num) * HZIP_CORE_ADDR_INTRVL;811}812813static int hisi_zip_core_debug_init(struct hisi_qm *qm)814{815u32 zip_core_num, zip_comp_core_num;816struct device *dev = &qm->pdev->dev;817struct debugfs_regset32 *regset;818u32 zip_core_info;819struct dentry *tmp_d;820char buf[HZIP_BUF_SIZE];821int i;822823zip_core_info = qm->cap_tables.dev_cap_table[ZIP_CORE_INFO].cap_val;824zip_core_num = (zip_core_info >> zip_basic_cap_info[ZIP_CORE_NUM_CAP].shift) &825zip_basic_cap_info[ZIP_CORE_NUM_CAP].mask;826zip_comp_core_num = (zip_core_info >> zip_basic_cap_info[ZIP_CLUSTER_COMP_NUM_CAP].shift) &827zip_basic_cap_info[ZIP_CLUSTER_COMP_NUM_CAP].mask;828829for (i = 0; i < zip_core_num; i++) {830if (i < zip_comp_core_num)831scnprintf(buf, sizeof(buf), "comp_core%d", i);832else833scnprintf(buf, sizeof(buf), "decomp_core%d",834i - zip_comp_core_num);835836regset = devm_kzalloc(dev, sizeof(*regset), GFP_KERNEL);837if (!regset)838return -ENOENT;839840regset->regs = hzip_dfx_regs;841regset->nregs = ARRAY_SIZE(hzip_dfx_regs);842regset->base = get_zip_core_addr(qm, i);843regset->dev = dev;844845tmp_d = debugfs_create_dir(buf, qm->debug.debug_root);846debugfs_create_file("regs", 0444, tmp_d, regset,847&hisi_zip_regs_fops);848}849850return 0;851}852853static int zip_cap_regs_show(struct seq_file *s, void *unused)854{855struct hisi_qm *qm = s->private;856u32 i, size;857858size = qm->cap_tables.qm_cap_size;859for (i = 0; i < size; i++)860seq_printf(s, "%s= 0x%08x\n", qm->cap_tables.qm_cap_table[i].name,861qm->cap_tables.qm_cap_table[i].cap_val);862863size = qm->cap_tables.dev_cap_size;864for (i = 0; i < size; i++)865seq_printf(s, "%s= 0x%08x\n", qm->cap_tables.dev_cap_table[i].name,866qm->cap_tables.dev_cap_table[i].cap_val);867868return 0;869}870871DEFINE_SHOW_ATTRIBUTE(zip_cap_regs);872873static void hisi_zip_dfx_debug_init(struct hisi_qm *qm)874{875struct dfx_diff_registers *hzip_regs = qm->debug.acc_diff_regs;876struct hisi_zip *zip = container_of(qm, struct hisi_zip, qm);877struct hisi_zip_dfx *dfx = &zip->dfx;878struct dentry *tmp_dir;879void *data;880int i;881882tmp_dir = debugfs_create_dir("zip_dfx", qm->debug.debug_root);883for (i = 0; i < ARRAY_SIZE(zip_dfx_files); i++) {884data = (atomic64_t *)((uintptr_t)dfx + zip_dfx_files[i].offset);885debugfs_create_file(zip_dfx_files[i].name,8860644, tmp_dir, data,887&zip_atomic64_ops);888}889890if (qm->fun_type == QM_HW_PF && hzip_regs)891debugfs_create_file("diff_regs", 0444, tmp_dir,892qm, &hzip_diff_regs_fops);893894debugfs_create_file("cap_regs", CAP_FILE_PERMISSION,895qm->debug.debug_root, qm, &zip_cap_regs_fops);896}897898static int hisi_zip_ctrl_debug_init(struct hisi_qm *qm)899{900struct hisi_zip *zip = container_of(qm, struct hisi_zip, qm);901int i;902903for (i = HZIP_CLEAR_ENABLE; i < HZIP_DEBUG_FILE_NUM; i++) {904spin_lock_init(&zip->ctrl->files[i].lock);905zip->ctrl->files[i].ctrl = zip->ctrl;906zip->ctrl->files[i].index = i;907908debugfs_create_file(ctrl_debug_file_name[i], 0600,909qm->debug.debug_root,910zip->ctrl->files + i,911&ctrl_debug_fops);912}913914return hisi_zip_core_debug_init(qm);915}916917static int hisi_zip_debugfs_init(struct hisi_qm *qm)918{919struct device *dev = &qm->pdev->dev;920int ret;921922ret = hisi_qm_regs_debugfs_init(qm, hzip_diff_regs, ARRAY_SIZE(hzip_diff_regs));923if (ret) {924dev_warn(dev, "Failed to init ZIP diff regs!\n");925return ret;926}927928qm->debug.sqe_mask_offset = HZIP_SQE_MASK_OFFSET;929qm->debug.sqe_mask_len = HZIP_SQE_MASK_LEN;930qm->debug.debug_root = debugfs_create_dir(dev_name(dev),931hzip_debugfs_root);932933hisi_qm_debug_init(qm);934935if (qm->fun_type == QM_HW_PF) {936ret = hisi_zip_ctrl_debug_init(qm);937if (ret)938goto debugfs_remove;939}940941hisi_zip_dfx_debug_init(qm);942943return 0;944945debugfs_remove:946debugfs_remove_recursive(qm->debug.debug_root);947hisi_qm_regs_debugfs_uninit(qm, ARRAY_SIZE(hzip_diff_regs));948return ret;949}950951/* hisi_zip_debug_regs_clear() - clear the zip debug regs */952static void hisi_zip_debug_regs_clear(struct hisi_qm *qm)953{954u32 zip_core_info;955u8 zip_core_num;956int i, j;957958zip_core_info = qm->cap_tables.dev_cap_table[ZIP_CORE_INFO].cap_val;959zip_core_num = (zip_core_info >> zip_basic_cap_info[ZIP_CORE_NUM_CAP].shift) &960zip_basic_cap_info[ZIP_CORE_NUM_CAP].mask;961962/* enable register read_clear bit */963writel(HZIP_RD_CNT_CLR_CE_EN, qm->io_base + HZIP_SOFT_CTRL_CNT_CLR_CE);964for (i = 0; i < zip_core_num; i++)965for (j = 0; j < ARRAY_SIZE(hzip_dfx_regs); j++)966readl(get_zip_core_addr(qm, i) +967hzip_dfx_regs[j].offset);968969/* disable register read_clear bit */970writel(0x0, qm->io_base + HZIP_SOFT_CTRL_CNT_CLR_CE);971972hisi_qm_debug_regs_clear(qm);973}974975static void hisi_zip_debugfs_exit(struct hisi_qm *qm)976{977debugfs_remove_recursive(qm->debug.debug_root);978979hisi_qm_regs_debugfs_uninit(qm, ARRAY_SIZE(hzip_diff_regs));980981if (qm->fun_type == QM_HW_PF) {982hisi_zip_debug_regs_clear(qm);983qm->debug.curr_qm_qp_num = 0;984}985}986987static int hisi_zip_show_last_regs_init(struct hisi_qm *qm)988{989int core_dfx_regs_num = ARRAY_SIZE(hzip_dump_dfx_regs);990int com_dfx_regs_num = ARRAY_SIZE(hzip_com_dfx_regs);991struct qm_debug *debug = &qm->debug;992void __iomem *io_base;993u32 zip_core_info;994u32 zip_core_num;995int i, j, idx;996997zip_core_info = qm->cap_tables.dev_cap_table[ZIP_CORE_INFO].cap_val;998zip_core_num = (zip_core_info >> zip_basic_cap_info[ZIP_CORE_NUM_CAP].shift) &999zip_basic_cap_info[ZIP_CORE_NUM_CAP].mask;10001001debug->last_words = kcalloc(core_dfx_regs_num * zip_core_num + com_dfx_regs_num,1002sizeof(unsigned int), GFP_KERNEL);1003if (!debug->last_words)1004return -ENOMEM;10051006for (i = 0; i < com_dfx_regs_num; i++) {1007io_base = qm->io_base + hzip_com_dfx_regs[i].offset;1008debug->last_words[i] = readl_relaxed(io_base);1009}10101011for (i = 0; i < zip_core_num; i++) {1012io_base = get_zip_core_addr(qm, i);1013for (j = 0; j < core_dfx_regs_num; j++) {1014idx = com_dfx_regs_num + i * core_dfx_regs_num + j;1015debug->last_words[idx] = readl_relaxed(1016io_base + hzip_dump_dfx_regs[j].offset);1017}1018}10191020return 0;1021}10221023static void hisi_zip_show_last_regs_uninit(struct hisi_qm *qm)1024{1025struct qm_debug *debug = &qm->debug;10261027if (qm->fun_type == QM_HW_VF || !debug->last_words)1028return;10291030kfree(debug->last_words);1031debug->last_words = NULL;1032}10331034static void hisi_zip_show_last_dfx_regs(struct hisi_qm *qm)1035{1036int core_dfx_regs_num = ARRAY_SIZE(hzip_dump_dfx_regs);1037int com_dfx_regs_num = ARRAY_SIZE(hzip_com_dfx_regs);1038u32 zip_core_num, zip_comp_core_num;1039struct qm_debug *debug = &qm->debug;1040char buf[HZIP_BUF_SIZE];1041u32 zip_core_info;1042void __iomem *base;1043int i, j, idx;1044u32 val;10451046if (qm->fun_type == QM_HW_VF || !debug->last_words)1047return;10481049for (i = 0; i < com_dfx_regs_num; i++) {1050val = readl_relaxed(qm->io_base + hzip_com_dfx_regs[i].offset);1051if (debug->last_words[i] != val)1052pci_info(qm->pdev, "com_dfx: %s \t= 0x%08x => 0x%08x\n",1053hzip_com_dfx_regs[i].name, debug->last_words[i], val);1054}10551056zip_core_info = qm->cap_tables.dev_cap_table[ZIP_CORE_INFO].cap_val;1057zip_core_num = (zip_core_info >> zip_basic_cap_info[ZIP_CORE_NUM_CAP].shift) &1058zip_basic_cap_info[ZIP_CORE_NUM_CAP].mask;1059zip_comp_core_num = (zip_core_info >> zip_basic_cap_info[ZIP_CLUSTER_COMP_NUM_CAP].shift) &1060zip_basic_cap_info[ZIP_CLUSTER_COMP_NUM_CAP].mask;10611062for (i = 0; i < zip_core_num; i++) {1063if (i < zip_comp_core_num)1064scnprintf(buf, sizeof(buf), "Comp_core-%d", i);1065else1066scnprintf(buf, sizeof(buf), "Decomp_core-%d",1067i - zip_comp_core_num);1068base = get_zip_core_addr(qm, i);10691070pci_info(qm->pdev, "==>%s:\n", buf);1071/* dump last word for dfx regs during control resetting */1072for (j = 0; j < core_dfx_regs_num; j++) {1073idx = com_dfx_regs_num + i * core_dfx_regs_num + j;1074val = readl_relaxed(base + hzip_dump_dfx_regs[j].offset);1075if (debug->last_words[idx] != val)1076pci_info(qm->pdev, "%s \t= 0x%08x => 0x%08x\n",1077hzip_dump_dfx_regs[j].name,1078debug->last_words[idx], val);1079}1080}1081}10821083static void hisi_zip_log_hw_error(struct hisi_qm *qm, u32 err_sts)1084{1085const struct hisi_zip_hw_error *err = zip_hw_error;1086struct device *dev = &qm->pdev->dev;1087u32 err_val;10881089while (err->msg) {1090if (err->int_msk & err_sts) {1091dev_err(dev, "%s [error status=0x%x] found\n",1092err->msg, err->int_msk);10931094if (err->int_msk & HZIP_CORE_INT_STATUS_M_ECC) {1095err_val = readl(qm->io_base +1096HZIP_CORE_SRAM_ECC_ERR_INFO);1097dev_err(dev, "hisi-zip multi ecc sram num=0x%x\n",1098((err_val >>1099HZIP_SRAM_ECC_ERR_NUM_SHIFT) & 0xFF));1100}1101}1102err++;1103}1104}11051106static u32 hisi_zip_get_hw_err_status(struct hisi_qm *qm)1107{1108return readl(qm->io_base + HZIP_CORE_INT_STATUS);1109}11101111static void hisi_zip_clear_hw_err_status(struct hisi_qm *qm, u32 err_sts)1112{1113writel(err_sts, qm->io_base + HZIP_CORE_INT_SOURCE);1114}11151116static void hisi_zip_disable_error_report(struct hisi_qm *qm, u32 err_type)1117{1118u32 nfe_mask;11191120nfe_mask = hisi_qm_get_hw_info(qm, zip_basic_cap_info, ZIP_NFE_MASK_CAP, qm->cap_ver);1121writel(nfe_mask & (~err_type), qm->io_base + HZIP_CORE_INT_RAS_NFE_ENB);1122}11231124static void hisi_zip_open_axi_master_ooo(struct hisi_qm *qm)1125{1126u32 val;11271128val = readl(qm->io_base + HZIP_SOFT_CTRL_ZIP_CONTROL);11291130writel(val & ~HZIP_AXI_SHUTDOWN_ENABLE,1131qm->io_base + HZIP_SOFT_CTRL_ZIP_CONTROL);11321133writel(val | HZIP_AXI_SHUTDOWN_ENABLE,1134qm->io_base + HZIP_SOFT_CTRL_ZIP_CONTROL);11351136hisi_dae_open_axi_master_ooo(qm);1137}11381139static void hisi_zip_close_axi_master_ooo(struct hisi_qm *qm)1140{1141u32 nfe_enb;11421143/* Disable ECC Mbit error report. */1144nfe_enb = readl(qm->io_base + HZIP_CORE_INT_RAS_NFE_ENB);1145writel(nfe_enb & ~HZIP_CORE_INT_STATUS_M_ECC,1146qm->io_base + HZIP_CORE_INT_RAS_NFE_ENB);11471148/* Inject zip ECC Mbit error to block master ooo. */1149writel(HZIP_CORE_INT_STATUS_M_ECC,1150qm->io_base + HZIP_CORE_INT_SET);1151}11521153static enum acc_err_result hisi_zip_get_err_result(struct hisi_qm *qm)1154{1155enum acc_err_result zip_result = ACC_ERR_NONE;1156enum acc_err_result dae_result;1157u32 err_status;11581159/* Get device hardware new error status */1160err_status = hisi_zip_get_hw_err_status(qm);1161if (err_status) {1162if (err_status & qm->err_info.ecc_2bits_mask)1163qm->err_status.is_dev_ecc_mbit = true;1164hisi_zip_log_hw_error(qm, err_status);11651166if (err_status & qm->err_info.dev_reset_mask) {1167/* Disable the same error reporting until device is recovered. */1168hisi_zip_disable_error_report(qm, err_status);1169return ACC_ERR_NEED_RESET;1170} else {1171hisi_zip_clear_hw_err_status(qm, err_status);1172}1173}11741175dae_result = hisi_dae_get_err_result(qm);11761177return (zip_result == ACC_ERR_NEED_RESET ||1178dae_result == ACC_ERR_NEED_RESET) ?1179ACC_ERR_NEED_RESET : ACC_ERR_RECOVERED;1180}11811182static bool hisi_zip_dev_is_abnormal(struct hisi_qm *qm)1183{1184u32 err_status;11851186err_status = hisi_zip_get_hw_err_status(qm);1187if (err_status & qm->err_info.dev_shutdown_mask)1188return true;11891190return hisi_dae_dev_is_abnormal(qm);1191}11921193static int hisi_zip_set_priv_status(struct hisi_qm *qm)1194{1195return hisi_dae_close_axi_master_ooo(qm);1196}11971198static void hisi_zip_err_info_init(struct hisi_qm *qm)1199{1200struct hisi_qm_err_info *err_info = &qm->err_info;12011202err_info->fe = HZIP_CORE_INT_RAS_FE_ENB_MASK;1203err_info->ce = hisi_qm_get_hw_info(qm, zip_basic_cap_info, ZIP_QM_CE_MASK_CAP, qm->cap_ver);1204err_info->nfe = hisi_qm_get_hw_info(qm, zip_basic_cap_info,1205ZIP_QM_NFE_MASK_CAP, qm->cap_ver);1206err_info->ecc_2bits_mask = HZIP_CORE_INT_STATUS_M_ECC;1207err_info->qm_shutdown_mask = hisi_qm_get_hw_info(qm, zip_basic_cap_info,1208ZIP_QM_OOO_SHUTDOWN_MASK_CAP, qm->cap_ver);1209err_info->dev_shutdown_mask = hisi_qm_get_hw_info(qm, zip_basic_cap_info,1210ZIP_OOO_SHUTDOWN_MASK_CAP, qm->cap_ver);1211err_info->qm_reset_mask = hisi_qm_get_hw_info(qm, zip_basic_cap_info,1212ZIP_QM_RESET_MASK_CAP, qm->cap_ver);1213err_info->dev_reset_mask = hisi_qm_get_hw_info(qm, zip_basic_cap_info,1214ZIP_RESET_MASK_CAP, qm->cap_ver);1215err_info->msi_wr_port = HZIP_WR_PORT;1216err_info->acpi_rst = "ZRST";1217}12181219static const struct hisi_qm_err_ini hisi_zip_err_ini = {1220.hw_init = hisi_zip_set_user_domain_and_cache,1221.hw_err_enable = hisi_zip_hw_error_enable,1222.hw_err_disable = hisi_zip_hw_error_disable,1223.get_dev_hw_err_status = hisi_zip_get_hw_err_status,1224.clear_dev_hw_err_status = hisi_zip_clear_hw_err_status,1225.open_axi_master_ooo = hisi_zip_open_axi_master_ooo,1226.close_axi_master_ooo = hisi_zip_close_axi_master_ooo,1227.open_sva_prefetch = hisi_zip_open_sva_prefetch,1228.close_sva_prefetch = hisi_zip_close_sva_prefetch,1229.show_last_dfx_regs = hisi_zip_show_last_dfx_regs,1230.err_info_init = hisi_zip_err_info_init,1231.get_err_result = hisi_zip_get_err_result,1232.set_priv_status = hisi_zip_set_priv_status,1233.dev_is_abnormal = hisi_zip_dev_is_abnormal,1234};12351236static int hisi_zip_pf_probe_init(struct hisi_zip *hisi_zip)1237{1238struct hisi_qm *qm = &hisi_zip->qm;1239struct hisi_zip_ctrl *ctrl;1240int ret;12411242ctrl = devm_kzalloc(&qm->pdev->dev, sizeof(*ctrl), GFP_KERNEL);1243if (!ctrl)1244return -ENOMEM;12451246hisi_zip->ctrl = ctrl;1247ctrl->hisi_zip = hisi_zip;12481249ret = hisi_zip_set_user_domain_and_cache(qm);1250if (ret)1251return ret;12521253ret = hisi_zip_set_high_perf(qm);1254if (ret)1255return ret;12561257hisi_zip_open_sva_prefetch(qm);1258hisi_qm_dev_err_init(qm);1259hisi_zip_debug_regs_clear(qm);12601261ret = hisi_zip_show_last_regs_init(qm);1262if (ret)1263pci_err(qm->pdev, "Failed to init last word regs!\n");12641265return ret;1266}12671268static int zip_pre_store_cap_reg(struct hisi_qm *qm)1269{1270struct hisi_qm_cap_record *zip_cap;1271struct pci_dev *pdev = qm->pdev;1272size_t i, size;12731274size = ARRAY_SIZE(zip_cap_query_info);1275zip_cap = devm_kzalloc(&pdev->dev, sizeof(*zip_cap) * size, GFP_KERNEL);1276if (!zip_cap)1277return -ENOMEM;12781279for (i = 0; i < size; i++) {1280zip_cap[i].type = zip_cap_query_info[i].type;1281zip_cap[i].name = zip_cap_query_info[i].name;1282zip_cap[i].cap_val = hisi_qm_get_cap_value(qm, zip_cap_query_info,1283i, qm->cap_ver);1284}12851286qm->cap_tables.dev_cap_table = zip_cap;1287qm->cap_tables.dev_cap_size = size;12881289return 0;1290}12911292static int hisi_zip_qm_init(struct hisi_qm *qm, struct pci_dev *pdev)1293{1294u64 alg_msk;1295int ret;12961297qm->pdev = pdev;1298qm->mode = uacce_mode;1299qm->sqe_size = HZIP_SQE_SIZE;1300qm->dev_name = hisi_zip_name;13011302qm->fun_type = (pdev->device == PCI_DEVICE_ID_HUAWEI_ZIP_PF) ?1303QM_HW_PF : QM_HW_VF;1304if (qm->fun_type == QM_HW_PF) {1305qm->qp_base = HZIP_PF_DEF_Q_BASE;1306qm->qp_num = pf_q_num;1307qm->debug.curr_qm_qp_num = pf_q_num;1308qm->qm_list = &zip_devices;1309qm->err_ini = &hisi_zip_err_ini;1310if (pf_q_num_flag)1311set_bit(QM_MODULE_PARAM, &qm->misc_ctl);1312} else if (qm->fun_type == QM_HW_VF && qm->ver == QM_HW_V1) {1313/*1314* have no way to get qm configure in VM in v1 hardware,1315* so currently force PF to uses HZIP_PF_DEF_Q_NUM, and force1316* to trigger only one VF in v1 hardware.1317*1318* v2 hardware has no such problem.1319*/1320qm->qp_base = HZIP_PF_DEF_Q_NUM;1321qm->qp_num = HZIP_QUEUE_NUM_V1 - HZIP_PF_DEF_Q_NUM;1322}13231324ret = hisi_qm_init(qm);1325if (ret) {1326pci_err(qm->pdev, "Failed to init zip qm configures!\n");1327return ret;1328}13291330/* Fetch and save the value of capability registers */1331ret = zip_pre_store_cap_reg(qm);1332if (ret) {1333pci_err(qm->pdev, "Failed to pre-store capability registers!\n");1334goto err_qm_uninit;1335}13361337alg_msk = qm->cap_tables.dev_cap_table[ZIP_ALG_BITMAP].cap_val;1338ret = hisi_qm_set_algs(qm, alg_msk, zip_dev_algs, ARRAY_SIZE(zip_dev_algs));1339if (ret) {1340pci_err(qm->pdev, "Failed to set zip algs!\n");1341goto err_qm_uninit;1342}13431344ret = hisi_dae_set_alg(qm);1345if (ret)1346goto err_qm_uninit;13471348return 0;13491350err_qm_uninit:1351hisi_qm_uninit(qm);1352return ret;1353}13541355static void hisi_zip_qm_uninit(struct hisi_qm *qm)1356{1357hisi_qm_uninit(qm);1358}13591360static int hisi_zip_probe_init(struct hisi_zip *hisi_zip)1361{1362u32 type_rate = HZIP_SHAPER_RATE_COMPRESS;1363struct hisi_qm *qm = &hisi_zip->qm;1364int ret;13651366if (qm->fun_type == QM_HW_PF) {1367ret = hisi_zip_pf_probe_init(hisi_zip);1368if (ret)1369return ret;1370/* enable shaper type 0 */1371if (qm->ver >= QM_HW_V3) {1372type_rate |= QM_SHAPER_ENABLE;13731374/* ZIP need to enable shaper type 1 */1375type_rate |= HZIP_SHAPER_RATE_DECOMPRESS << QM_SHAPER_TYPE1_OFFSET;1376qm->type_rate = type_rate;1377}1378}13791380return 0;1381}13821383static void hisi_zip_probe_uninit(struct hisi_qm *qm)1384{1385if (qm->fun_type == QM_HW_VF)1386return;13871388hisi_zip_show_last_regs_uninit(qm);1389hisi_zip_close_sva_prefetch(qm);1390hisi_qm_dev_err_uninit(qm);1391}13921393static int hisi_zip_probe(struct pci_dev *pdev, const struct pci_device_id *id)1394{1395struct hisi_zip *hisi_zip;1396struct hisi_qm *qm;1397int ret;13981399hisi_zip = devm_kzalloc(&pdev->dev, sizeof(*hisi_zip), GFP_KERNEL);1400if (!hisi_zip)1401return -ENOMEM;14021403qm = &hisi_zip->qm;14041405ret = hisi_zip_qm_init(qm, pdev);1406if (ret) {1407pci_err(pdev, "Failed to init ZIP QM (%d)!\n", ret);1408return ret;1409}14101411ret = hisi_zip_probe_init(hisi_zip);1412if (ret) {1413pci_err(pdev, "Failed to probe (%d)!\n", ret);1414goto err_qm_uninit;1415}14161417ret = hisi_qm_start(qm);1418if (ret)1419goto err_probe_uninit;14201421ret = hisi_zip_debugfs_init(qm);1422if (ret)1423pci_err(pdev, "failed to init debugfs (%d)!\n", ret);14241425hisi_qm_add_list(qm, &zip_devices);1426ret = hisi_qm_alg_register(qm, &zip_devices, HZIP_CTX_Q_NUM_DEF);1427if (ret < 0) {1428pci_err(pdev, "failed to register driver to crypto!\n");1429goto err_qm_del_list;1430}14311432if (qm->uacce) {1433ret = uacce_register(qm->uacce);1434if (ret) {1435pci_err(pdev, "failed to register uacce (%d)!\n", ret);1436goto err_qm_alg_unregister;1437}1438}14391440if (qm->fun_type == QM_HW_PF && vfs_num > 0) {1441ret = hisi_qm_sriov_enable(pdev, vfs_num);1442if (ret < 0)1443goto err_qm_alg_unregister;1444}14451446hisi_qm_pm_init(qm);14471448return 0;14491450err_qm_alg_unregister:1451hisi_qm_alg_unregister(qm, &zip_devices, HZIP_CTX_Q_NUM_DEF);14521453err_qm_del_list:1454hisi_qm_del_list(qm, &zip_devices);1455hisi_zip_debugfs_exit(qm);1456hisi_qm_stop(qm, QM_NORMAL);14571458err_probe_uninit:1459hisi_zip_probe_uninit(qm);14601461err_qm_uninit:1462hisi_zip_qm_uninit(qm);14631464return ret;1465}14661467static void hisi_zip_remove(struct pci_dev *pdev)1468{1469struct hisi_qm *qm = pci_get_drvdata(pdev);14701471hisi_qm_pm_uninit(qm);1472hisi_qm_wait_task_finish(qm, &zip_devices);1473hisi_qm_alg_unregister(qm, &zip_devices, HZIP_CTX_Q_NUM_DEF);1474hisi_qm_del_list(qm, &zip_devices);14751476if (qm->fun_type == QM_HW_PF && qm->vfs_num)1477hisi_qm_sriov_disable(pdev, true);14781479hisi_zip_debugfs_exit(qm);1480hisi_qm_stop(qm, QM_NORMAL);1481hisi_zip_probe_uninit(qm);1482hisi_zip_qm_uninit(qm);1483}14841485static const struct dev_pm_ops hisi_zip_pm_ops = {1486SET_RUNTIME_PM_OPS(hisi_qm_suspend, hisi_qm_resume, NULL)1487};14881489static const struct pci_error_handlers hisi_zip_err_handler = {1490.error_detected = hisi_qm_dev_err_detected,1491.slot_reset = hisi_qm_dev_slot_reset,1492.reset_prepare = hisi_qm_reset_prepare,1493.reset_done = hisi_qm_reset_done,1494};14951496static struct pci_driver hisi_zip_pci_driver = {1497.name = "hisi_zip",1498.id_table = hisi_zip_dev_ids,1499.probe = hisi_zip_probe,1500.remove = hisi_zip_remove,1501.sriov_configure = IS_ENABLED(CONFIG_PCI_IOV) ?1502hisi_qm_sriov_configure : NULL,1503.err_handler = &hisi_zip_err_handler,1504.shutdown = hisi_qm_dev_shutdown,1505.driver.pm = &hisi_zip_pm_ops,1506};15071508struct pci_driver *hisi_zip_get_pf_driver(void)1509{1510return &hisi_zip_pci_driver;1511}1512EXPORT_SYMBOL_GPL(hisi_zip_get_pf_driver);15131514static void hisi_zip_register_debugfs(void)1515{1516if (!debugfs_initialized())1517return;15181519hzip_debugfs_root = debugfs_create_dir("hisi_zip", NULL);1520}15211522static void hisi_zip_unregister_debugfs(void)1523{1524debugfs_remove_recursive(hzip_debugfs_root);1525}15261527static int __init hisi_zip_init(void)1528{1529int ret;15301531hisi_qm_init_list(&zip_devices);1532hisi_zip_register_debugfs();15331534ret = pci_register_driver(&hisi_zip_pci_driver);1535if (ret < 0) {1536hisi_zip_unregister_debugfs();1537pr_err("Failed to register pci driver.\n");1538}15391540return ret;1541}15421543static void __exit hisi_zip_exit(void)1544{1545pci_unregister_driver(&hisi_zip_pci_driver);1546hisi_zip_unregister_debugfs();1547}15481549module_init(hisi_zip_init);1550module_exit(hisi_zip_exit);15511552MODULE_LICENSE("GPL v2");1553MODULE_AUTHOR("Zhou Wang <[email protected]>");1554MODULE_DESCRIPTION("Driver for HiSilicon ZIP accelerator");155515561557