243 lines
5.3 KiB
C
243 lines
5.3 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
/* Copyright (c) 2021 HiSilicon Limited. */
|
|
|
|
#ifndef ACC_MIG_H
|
|
#define ACC_MIG_H
|
|
|
|
#include <linux/mdev.h>
|
|
#include <linux/pci.h>
|
|
#include <linux/vfio.h>
|
|
|
|
#include "../qm.h"
|
|
|
|
#define VFIO_PCI_OFFSET_SHIFT 40
|
|
#define VFIO_PCI_OFFSET_TO_INDEX(off) ((off) >> VFIO_PCI_OFFSET_SHIFT)
|
|
#define VFIO_PCI_INDEX_TO_OFFSET(index) ((u64)(index) << VFIO_PCI_OFFSET_SHIFT)
|
|
#define VFIO_PCI_OFFSET_MASK (((u64)(1) << VFIO_PCI_OFFSET_SHIFT) - 1)
|
|
|
|
#define MIGRATION_REGION_SZ (sizeof(struct acc_vf_data) + \
|
|
sizeof(struct vfio_device_migration_info))
|
|
#define VFIO_DEV_DBG_LEN 256
|
|
#define VFIO_DBG_LOG_LEN 16
|
|
#define VFIO_DEVFN_MASK 0xFF
|
|
|
|
#define PCI_BAR_2 2
|
|
#define PCI_BAR_4 4
|
|
#define POLL_PERIOD 10
|
|
#define POLL_TIMEOUT 1000
|
|
#define QM_CACHE_WB_START 0x204
|
|
#define QM_CACHE_WB_DONE 0x208
|
|
#define QM_MB_CMD_PAUSE_QM 0xe
|
|
#define QM_ABNORMAL_INT_STATUS 0x100008
|
|
#define QM_IFC_INT_STATUS 0x0028
|
|
#define SEC_CORE_INT_STATUS 0x301008
|
|
#define HPRE_HAC_INT_STATUS 0x301800
|
|
#define HZIP_CORE_INT_STATUS 0x3010AC
|
|
|
|
#define QM_VFT_CFG_RDY 0x10006c
|
|
#define QM_VFT_CFG_OP_WR 0x100058
|
|
#define QM_VFT_CFG_TYPE 0x10005c
|
|
#define QM_VFT_CFG 0x100060
|
|
#define QM_VFT_CFG_OP_ENABLE 0x100054
|
|
#define QM_VFT_CFG_DATA_L 0x100064
|
|
#define QM_VFT_CFG_DATA_H 0x100068
|
|
|
|
#define ERROR_CHECK_TIMEOUT 100
|
|
#define CHECK_DELAY_TIME 100
|
|
|
|
#define QM_SQC_VFT_BASE_SHIFT_V2 28
|
|
#define QM_SQC_VFT_BASE_MASK_V2 GENMASK(15, 0)
|
|
#define QM_SQC_VFT_NUM_SHIFT_V2 45
|
|
#define QM_SQC_VFT_NUM_MASK_V2 GENMASK(9, 0)
|
|
|
|
/* mailbox */
|
|
#define QM_MB_CMD_SQC_BT 0x4
|
|
#define QM_MB_CMD_CQC_BT 0x5
|
|
#define QM_MB_CMD_SQC_VFT_V2 0x6
|
|
|
|
#define QM_MB_CMD_SEND_BASE 0x300
|
|
#define QM_MB_BUSY_SHIFT 13
|
|
#define QM_MB_OP_SHIFT 14
|
|
#define QM_MB_CMD_DATA_ADDR_L 0x304
|
|
#define QM_MB_CMD_DATA_ADDR_H 0x308
|
|
#define QM_MB_MAX_WAIT_CNT 6000
|
|
|
|
/* doorbell */
|
|
#define QM_DOORBELL_CMD_SQ 0
|
|
#define QM_DOORBELL_CMD_CQ 1
|
|
#define QM_DOORBELL_SQ_CQ_BASE_V2 0x1000
|
|
#define QM_DOORBELL_EQ_AEQ_BASE_V2 0x2000
|
|
#define QM_DB_CMD_SHIFT_V2 12
|
|
#define QM_DB_RAND_SHIFT_V2 16
|
|
#define QM_DB_INDEX_SHIFT_V2 32
|
|
#define QM_DB_PRIORITY_SHIFT_V2 48
|
|
|
|
/* RW regs */
|
|
#define QM_REGS_MAX_LEN 7
|
|
#define QM_REG_ADDR_OFFSET 0x0004
|
|
|
|
#define QM_XQC_ADDR_OFFSET 32U
|
|
#define QM_VF_AEQ_INT_MASK 0x0004
|
|
#define QM_VF_EQ_INT_MASK 0x000c
|
|
#define QM_IFC_INT_SOURCE_V 0x0020
|
|
#define QM_IFC_INT_MASK 0x0024
|
|
#define QM_IFC_INT_SET_V 0x002c
|
|
#define QM_QUE_ISO_CFG_V 0x0030
|
|
#define QM_PAGE_SIZE 0x0034
|
|
|
|
#define QM_EQC_DW0 0X8000
|
|
#define QM_AEQC_DW0 0X8020
|
|
|
|
struct qm_mailbox {
|
|
__le16 w0;
|
|
__le16 queue_num;
|
|
__le32 base_l;
|
|
__le32 base_h;
|
|
__le32 rsvd;
|
|
};
|
|
|
|
enum acc_type {
|
|
HISI_SEC = 0x1,
|
|
HISI_HPRE = 0x2,
|
|
HISI_ZIP = 0x3,
|
|
};
|
|
|
|
struct vf_acc_type {
|
|
const char *name;
|
|
u32 type;
|
|
};
|
|
|
|
static struct vf_acc_type vf_acc_types[] = {
|
|
{"hisi_sec2", HISI_SEC},
|
|
{"hisi_hpre", HISI_HPRE},
|
|
{"hisi_zip", HISI_ZIP},
|
|
};
|
|
|
|
enum mig_debug_cmd {
|
|
STATE_SAVE,
|
|
STATE_RESUME,
|
|
MB_TEST,
|
|
MIG_DATA_DUMP,
|
|
MIG_DEV_SHOW,
|
|
};
|
|
|
|
static const char * const vf_dev_state[] = {
|
|
"Stop",
|
|
"Running",
|
|
"Saving",
|
|
"Running & Saving",
|
|
"Resuming",
|
|
};
|
|
|
|
#define QM_MATCH_SIZE 32L
|
|
struct acc_vf_data {
|
|
/* QM match information */
|
|
u32 qp_num;
|
|
u32 acc_type;
|
|
u32 que_iso_cfg;
|
|
u32 qp_base;
|
|
/* QM reserved 4 match information */
|
|
u32 qm_rsv_state[4];
|
|
|
|
/* QM RW regs */
|
|
u32 aeq_int_mask;
|
|
u32 eq_int_mask;
|
|
u32 ifc_int_source;
|
|
u32 ifc_int_mask;
|
|
u32 ifc_int_set;
|
|
u32 page_size;
|
|
u32 vf_state;
|
|
|
|
/*
|
|
* QM_VF_MB has 4 regs don't need to migration
|
|
* mailbox regs writeback value will cause
|
|
* hardware to perform command operations
|
|
*/
|
|
|
|
/* QM_EQC_DW has 7 regs */
|
|
u32 qm_eqc_dw[7];
|
|
|
|
/* QM_AEQC_DW has 7 regs */
|
|
u32 qm_aeqc_dw[7];
|
|
|
|
/* QM reserved 5 regs */
|
|
u32 qm_rsv_regs[5];
|
|
|
|
/* qm memory init information */
|
|
dma_addr_t eqe_dma;
|
|
dma_addr_t aeqe_dma;
|
|
dma_addr_t sqc_dma;
|
|
dma_addr_t cqc_dma;
|
|
};
|
|
|
|
struct acc_vf_remap_irq_ctx {
|
|
struct eventfd_ctx *trigger;
|
|
struct virqfd *sync;
|
|
atomic_t cnt;
|
|
wait_queue_head_t waitq;
|
|
bool init;
|
|
};
|
|
|
|
struct acc_vf_migration {
|
|
__u32 vf_vendor;
|
|
__u32 vf_device;
|
|
__u32 handle;
|
|
struct pci_dev *pf_dev;
|
|
struct pci_dev *vf_dev;
|
|
struct hisi_qm *pf_qm;
|
|
struct hisi_qm *vf_qm;
|
|
int vf_id;
|
|
int refcnt;
|
|
u8 acc_type;
|
|
bool mig_ignore;
|
|
struct mutex reflock;
|
|
|
|
struct vfio_device_migration_info *mig_ctl;
|
|
struct acc_vf_data *vf_data;
|
|
bool in_dirty_track;
|
|
struct acc_vf_remap_irq_ctx remap_irq_ctx;
|
|
struct acc_vf_region *regions;
|
|
int num_regions;
|
|
struct dentry *debug_root;
|
|
};
|
|
|
|
struct acc_vf_region_ops {
|
|
int (*rw)(struct acc_vf_migration *acc_vf_dev,
|
|
char __user *buf, size_t count,
|
|
loff_t *ppos, bool iswrite);
|
|
void (*release)(struct acc_vf_migration *acc_vf_dev,
|
|
struct acc_vf_region *region);
|
|
int (*mmap)(struct acc_vf_migration *acc_vf_dev,
|
|
struct acc_vf_region *region,
|
|
struct vm_area_struct *vma);
|
|
int (*add_cap)(struct acc_vf_migration *acc_vf_dev,
|
|
struct acc_vf_region *region,
|
|
struct vfio_info_cap *caps);
|
|
};
|
|
|
|
struct acc_vf_region {
|
|
u32 type;
|
|
u32 subtype;
|
|
size_t size;
|
|
u32 flags;
|
|
const struct acc_vf_region_ops *ops;
|
|
void *data;
|
|
};
|
|
|
|
struct acc_vf_irqops {
|
|
int (*set_irqs)(struct acc_vf_migration *acc_vf_dev,
|
|
u32 flags, unsigned int index,
|
|
unsigned int start, unsigned int count,
|
|
void *data);
|
|
};
|
|
|
|
struct acc_vf_irq {
|
|
u32 type;
|
|
u32 subtype;
|
|
u32 flags;
|
|
u32 count;
|
|
const struct acc_vf_irqops *ops;
|
|
};
|
|
|
|
#endif /* ACC_MIG_H */
|