提交 36b8d291 编写于 作者: K Kai Ye 提交者: Zheng Zengkai

crypto: hisilicon/qm - set a qp error flag for userspace

driver inclusion
category: bugfix
bugzilla: https://gitee.com/openeuler/kernel/issues/I477YJ?from=project-issue

----------------------------------------------------------------------

Add one page memory for device or qp status. Set a qp error
flag in this page when device resetting. This error flag can
be seen in the userspace. It helps users to stop tasks when
device hardware occurs on the device.
Signed-off-by: NKai Ye <yekai13@huawei.com>
Reviewed-by: NHao Fang <fanghao11@huawei.com>
Reviewed-by: NMingqiang Ling <lingmingqiang@huawei.com>
Signed-off-by: NZheng Zengkai <zhengzengkai@huawei.com>
上级 a0464f0b
...@@ -233,6 +233,8 @@ ...@@ -233,6 +233,8 @@
#define QM_DBG_WRITE_LEN 1024 #define QM_DBG_WRITE_LEN 1024
#define QM_DBG_TMP_BUF_LEN 22 #define QM_DBG_TMP_BUF_LEN 22
#define QM_PCI_COMMAND_INVALID ~0 #define QM_PCI_COMMAND_INVALID ~0
#define QM_RESET_STOP_TX_OFFSET 1
#define QM_RESET_STOP_RX_OFFSET 2
#define WAIT_PERIOD 20 #define WAIT_PERIOD 20
#define REMOVE_WAIT_DELAY 10 #define REMOVE_WAIT_DELAY 10
...@@ -883,6 +885,20 @@ static irqreturn_t qm_mb_cmd_irq(int irq, void *data) ...@@ -883,6 +885,20 @@ static irqreturn_t qm_mb_cmd_irq(int irq, void *data)
return IRQ_HANDLED; return IRQ_HANDLED;
} }
static void qm_set_qp_disable(struct hisi_qp *qp, int offset)
{
u32 *addr;
if (qp->is_in_kernel)
return;
addr = (u32 *)(qp->qdma.va + qp->qdma.size) - offset;
*addr = 1;
/* make sure setup is completed */
mb();
}
static irqreturn_t qm_aeq_irq(int irq, void *data) static irqreturn_t qm_aeq_irq(int irq, void *data)
{ {
struct hisi_qm *qm = data; struct hisi_qm *qm = data;
...@@ -2467,6 +2483,15 @@ static void *qm_get_avail_sqe(struct hisi_qp *qp) ...@@ -2467,6 +2483,15 @@ static void *qm_get_avail_sqe(struct hisi_qp *qp)
return qp->sqe + sq_tail * qp->qm->sqe_size; return qp->sqe + sq_tail * qp->qm->sqe_size;
} }
static void hisi_qm_unset_hw_reset(struct hisi_qp *qp)
{
u64 *addr;
/* Use last 64 bits of DUS to reset status. */
addr = (u64 *)(qp->qdma.va + qp->qdma.size) - QM_RESET_STOP_TX_OFFSET;
*addr = 0;
}
static struct hisi_qp *qm_create_qp_nolock(struct hisi_qm *qm, u8 alg_type) static struct hisi_qp *qm_create_qp_nolock(struct hisi_qm *qm, u8 alg_type)
{ {
struct device *dev = &qm->pdev->dev; struct device *dev = &qm->pdev->dev;
...@@ -2492,7 +2517,7 @@ static struct hisi_qp *qm_create_qp_nolock(struct hisi_qm *qm, u8 alg_type) ...@@ -2492,7 +2517,7 @@ static struct hisi_qp *qm_create_qp_nolock(struct hisi_qm *qm, u8 alg_type)
} }
qp = &qm->qp_array[qp_id]; qp = &qm->qp_array[qp_id];
hisi_qm_unset_hw_reset(qp);
memset(qp->cqe, 0, sizeof(struct qm_cqe) * QM_Q_DEPTH); memset(qp->cqe, 0, sizeof(struct qm_cqe) * QM_Q_DEPTH);
qp->event_cb = NULL; qp->event_cb = NULL;
...@@ -2912,6 +2937,14 @@ static int hisi_qm_get_available_instances(struct uacce_device *uacce) ...@@ -2912,6 +2937,14 @@ static int hisi_qm_get_available_instances(struct uacce_device *uacce)
return hisi_qm_get_free_qp_num(uacce->priv); return hisi_qm_get_free_qp_num(uacce->priv);
} }
static void hisi_qm_set_hw_reset(struct hisi_qm *qm, int offset)
{
int i;
for (i = 0; i < qm->qp_num; i++)
qm_set_qp_disable(&qm->qp_array[i], offset);
}
static int hisi_qm_uacce_get_queue(struct uacce_device *uacce, static int hisi_qm_uacce_get_queue(struct uacce_device *uacce,
unsigned long arg, unsigned long arg,
struct uacce_queue *q) struct uacce_queue *q)
...@@ -3122,8 +3155,10 @@ static int qm_alloc_uacce(struct hisi_qm *qm) ...@@ -3122,8 +3155,10 @@ static int qm_alloc_uacce(struct hisi_qm *qm)
else else
mmio_page_nr = qm->db_interval / PAGE_SIZE; mmio_page_nr = qm->db_interval / PAGE_SIZE;
/* Add one more page for device or qp status */
dus_page_nr = (PAGE_SIZE - 1 + qm->sqe_size * QM_Q_DEPTH + dus_page_nr = (PAGE_SIZE - 1 + qm->sqe_size * QM_Q_DEPTH +
sizeof(struct qm_cqe) * QM_Q_DEPTH) >> PAGE_SHIFT; sizeof(struct qm_cqe) * QM_Q_DEPTH + PAGE_SIZE) >>
PAGE_SHIFT;
uacce->qf_pg_num[UACCE_QFRT_MMIO] = mmio_page_nr; uacce->qf_pg_num[UACCE_QFRT_MMIO] = mmio_page_nr;
uacce->qf_pg_num[UACCE_QFRT_DUS] = dus_page_nr; uacce->qf_pg_num[UACCE_QFRT_DUS] = dus_page_nr;
...@@ -3695,11 +3730,13 @@ int hisi_qm_stop(struct hisi_qm *qm, enum qm_stop_reason r) ...@@ -3695,11 +3730,13 @@ int hisi_qm_stop(struct hisi_qm *qm, enum qm_stop_reason r)
if (qm->status.stop_reason == QM_SOFT_RESET || if (qm->status.stop_reason == QM_SOFT_RESET ||
qm->status.stop_reason == QM_FLR) { qm->status.stop_reason == QM_FLR) {
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_TX_OFFSET);
ret = qm_stop_started_qp(qm); ret = qm_stop_started_qp(qm);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "Failed to stop started qp!\n"); dev_err(dev, "Failed to stop started qp!\n");
goto err_unlock; goto err_unlock;
} }
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_RX_OFFSET);
} }
/* Mask eq and aeq irq */ /* Mask eq and aeq irq */
...@@ -5058,6 +5095,8 @@ static int qm_controller_reset(struct hisi_qm *qm) ...@@ -5058,6 +5095,8 @@ static int qm_controller_reset(struct hisi_qm *qm)
ret = qm_controller_reset_prepare(qm); ret = qm_controller_reset_prepare(qm);
if (ret) { if (ret) {
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_TX_OFFSET);
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_RX_OFFSET);
clear_bit(QM_RST_SCHED, &qm->misc_ctl); clear_bit(QM_RST_SCHED, &qm->misc_ctl);
return ret; return ret;
} }
...@@ -5144,6 +5183,8 @@ void hisi_qm_reset_prepare(struct pci_dev *pdev) ...@@ -5144,6 +5183,8 @@ void hisi_qm_reset_prepare(struct pci_dev *pdev)
ret = hisi_qm_stop(qm, QM_FLR); ret = hisi_qm_stop(qm, QM_FLR);
if (ret) { if (ret) {
pci_err(pdev, "Failed to stop QM, ret = %d.\n", ret); pci_err(pdev, "Failed to stop QM, ret = %d.\n", ret);
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_TX_OFFSET);
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_RX_OFFSET);
return; return;
} }
...@@ -5330,6 +5371,8 @@ static void qm_pf_reset_vf_prepare(struct hisi_qm *qm, ...@@ -5330,6 +5371,8 @@ static void qm_pf_reset_vf_prepare(struct hisi_qm *qm,
} }
err_prepare: err_prepare:
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_TX_OFFSET);
hisi_qm_set_hw_reset(qm, QM_RESET_STOP_RX_OFFSET);
pci_save_state(pdev); pci_save_state(pdev);
ret = qm->ops->ping_pf(qm, cmd); ret = qm->ops->ping_pf(qm, cmd);
if (ret) if (ret)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册