mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2026-03-22 07:27:12 +08:00
crypto: qat - add GEN6 firmware loader
Add support for the QAT GEN6 devices in the firmware loader. This includes handling firmware images signed with the RSA 3K and the XMSS algorithms. Co-developed-by: Suman Kumar Chakraborty <suman.kumar.chakraborty@intel.com> Signed-off-by: Suman Kumar Chakraborty <suman.kumar.chakraborty@intel.com> Signed-off-by: Jack Xu <jack.xu@intel.com> Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
This commit is contained in:
@@ -34,6 +34,8 @@
|
||||
#define PCI_DEVICE_ID_INTEL_QAT_402XXIOV 0x4945
|
||||
#define PCI_DEVICE_ID_INTEL_QAT_420XX 0x4946
|
||||
#define PCI_DEVICE_ID_INTEL_QAT_420XXIOV 0x4947
|
||||
#define PCI_DEVICE_ID_INTEL_QAT_6XXX 0x4948
|
||||
|
||||
#define ADF_DEVICE_FUSECTL_OFFSET 0x40
|
||||
#define ADF_DEVICE_LEGFUSE_OFFSET 0x4C
|
||||
#define ADF_DEVICE_FUSECTL_MASK 0x80000000
|
||||
|
||||
@@ -35,6 +35,7 @@ struct icp_qat_fw_loader_chip_info {
|
||||
u32 wakeup_event_val;
|
||||
bool fw_auth;
|
||||
bool css_3k;
|
||||
bool dual_sign;
|
||||
bool tgroup_share_ustore;
|
||||
u32 fcu_ctl_csr;
|
||||
u32 fcu_sts_csr;
|
||||
|
||||
@@ -7,6 +7,7 @@
|
||||
#define ICP_QAT_AC_C62X_DEV_TYPE 0x01000000
|
||||
#define ICP_QAT_AC_C3XXX_DEV_TYPE 0x02000000
|
||||
#define ICP_QAT_AC_4XXX_A_DEV_TYPE 0x08000000
|
||||
#define ICP_QAT_AC_6XXX_DEV_TYPE 0x80000000
|
||||
#define ICP_QAT_UCLO_MAX_AE 17
|
||||
#define ICP_QAT_UCLO_MAX_CTX 8
|
||||
#define ICP_QAT_UCLO_MAX_UIMAGE (ICP_QAT_UCLO_MAX_AE * ICP_QAT_UCLO_MAX_CTX)
|
||||
@@ -81,6 +82,21 @@
|
||||
#define ICP_QAT_CSS_RSA4K_MAX_IMAGE_LEN 0x40000
|
||||
#define ICP_QAT_CSS_RSA3K_MAX_IMAGE_LEN 0x30000
|
||||
|
||||
/* All lengths below are in bytes */
|
||||
#define ICP_QAT_DUALSIGN_OPAQUE_HDR_LEN 12
|
||||
#define ICP_QAT_DUALSIGN_OPAQUE_HDR_ALIGN_LEN 16
|
||||
#define ICP_QAT_DUALSIGN_OPAQUE_DATA_LEN 3540
|
||||
#define ICP_QAT_DUALSIGN_XMSS_PUBKEY_LEN 64
|
||||
#define ICP_QAT_DUALSIGN_XMSS_SIG_LEN 2692
|
||||
#define ICP_QAT_DUALSIGN_XMSS_SIG_ALIGN_LEN 2696
|
||||
#define ICP_QAT_DUALSIGN_MISC_INFO_LEN 16
|
||||
#define ICP_QAT_DUALSIGN_FW_TYPE_LEN 7
|
||||
#define ICP_QAT_DUALSIGN_MODULE_TYPE 0x14
|
||||
#define ICP_QAT_DUALSIGN_HDR_LEN 0x375
|
||||
#define ICP_QAT_DUALSIGN_HDR_VER 0x40001
|
||||
#define ICP_QAT_DUALSIGN_HDR_LEN_OFFSET 4
|
||||
#define ICP_QAT_DUALSIGN_HDR_VER_OFFSET 8
|
||||
|
||||
#define ICP_QAT_CTX_MODE(ae_mode) ((ae_mode) & 0xf)
|
||||
#define ICP_QAT_NN_MODE(ae_mode) (((ae_mode) >> 0x4) & 0xf)
|
||||
#define ICP_QAT_SHARED_USTORE_MODE(ae_mode) (((ae_mode) >> 0xb) & 0x1)
|
||||
@@ -440,6 +456,13 @@ struct icp_qat_fw_auth_desc {
|
||||
unsigned int img_ae_init_data_low;
|
||||
unsigned int img_ae_insts_high;
|
||||
unsigned int img_ae_insts_low;
|
||||
unsigned int cpp_mask;
|
||||
unsigned int reserved;
|
||||
unsigned int xmss_pubkey_high;
|
||||
unsigned int xmss_pubkey_low;
|
||||
unsigned int xmss_sig_high;
|
||||
unsigned int xmss_sig_low;
|
||||
unsigned int reserved2[2];
|
||||
};
|
||||
|
||||
struct icp_qat_auth_chunk {
|
||||
|
||||
@@ -698,6 +698,7 @@ static int qat_hal_chip_init(struct icp_qat_fw_loader_handle *handle,
|
||||
case PCI_DEVICE_ID_INTEL_QAT_401XX:
|
||||
case PCI_DEVICE_ID_INTEL_QAT_402XX:
|
||||
case PCI_DEVICE_ID_INTEL_QAT_420XX:
|
||||
case PCI_DEVICE_ID_INTEL_QAT_6XXX:
|
||||
handle->chip_info->mmp_sram_size = 0;
|
||||
handle->chip_info->nn = false;
|
||||
handle->chip_info->lm2lm3 = true;
|
||||
@@ -712,6 +713,8 @@ static int qat_hal_chip_init(struct icp_qat_fw_loader_handle *handle,
|
||||
handle->chip_info->wakeup_event_val = 0x80000000;
|
||||
handle->chip_info->fw_auth = true;
|
||||
handle->chip_info->css_3k = true;
|
||||
if (handle->pci_dev->device == PCI_DEVICE_ID_INTEL_QAT_6XXX)
|
||||
handle->chip_info->dual_sign = true;
|
||||
handle->chip_info->tgroup_share_ustore = true;
|
||||
handle->chip_info->fcu_ctl_csr = FCU_CONTROL_4XXX;
|
||||
handle->chip_info->fcu_sts_csr = FCU_STATUS_4XXX;
|
||||
|
||||
@@ -10,6 +10,7 @@
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/pci_ids.h>
|
||||
#include <linux/wordpart.h>
|
||||
#include "adf_accel_devices.h"
|
||||
#include "adf_common_drv.h"
|
||||
#include "icp_qat_uclo.h"
|
||||
@@ -737,6 +738,8 @@ qat_uclo_get_dev_type(struct icp_qat_fw_loader_handle *handle)
|
||||
case PCI_DEVICE_ID_INTEL_QAT_402XX:
|
||||
case PCI_DEVICE_ID_INTEL_QAT_420XX:
|
||||
return ICP_QAT_AC_4XXX_A_DEV_TYPE;
|
||||
case PCI_DEVICE_ID_INTEL_QAT_6XXX:
|
||||
return ICP_QAT_AC_6XXX_DEV_TYPE;
|
||||
default:
|
||||
pr_err("unsupported device 0x%x\n", handle->pci_dev->device);
|
||||
return 0;
|
||||
@@ -1035,17 +1038,30 @@ out_err:
|
||||
|
||||
static unsigned int qat_uclo_simg_hdr2sign_len(struct icp_qat_fw_loader_handle *handle)
|
||||
{
|
||||
if (handle->chip_info->dual_sign)
|
||||
return ICP_QAT_DUALSIGN_OPAQUE_DATA_LEN;
|
||||
|
||||
return ICP_QAT_AE_IMG_OFFSET(handle);
|
||||
}
|
||||
|
||||
static unsigned int qat_uclo_simg_hdr2cont_len(struct icp_qat_fw_loader_handle *handle)
|
||||
{
|
||||
if (handle->chip_info->dual_sign)
|
||||
return ICP_QAT_DUALSIGN_OPAQUE_DATA_LEN + ICP_QAT_DUALSIGN_MISC_INFO_LEN;
|
||||
|
||||
return ICP_QAT_AE_IMG_OFFSET(handle);
|
||||
}
|
||||
|
||||
static unsigned int qat_uclo_simg_fw_type(struct icp_qat_fw_loader_handle *handle, void *img_ptr)
|
||||
{
|
||||
struct icp_qat_css_hdr *hdr = img_ptr;
|
||||
char *fw_hdr = img_ptr;
|
||||
unsigned int offset;
|
||||
|
||||
if (handle->chip_info->dual_sign) {
|
||||
offset = qat_uclo_simg_hdr2sign_len(handle) + ICP_QAT_DUALSIGN_FW_TYPE_LEN;
|
||||
return *(fw_hdr + offset);
|
||||
}
|
||||
|
||||
return hdr->fw_type;
|
||||
}
|
||||
@@ -1390,16 +1406,27 @@ static int qat_uclo_check_image(struct icp_qat_fw_loader_handle *handle,
|
||||
if (handle->chip_info->fw_auth) {
|
||||
header_len = qat_uclo_simg_hdr2sign_len(handle);
|
||||
simg_type = qat_uclo_simg_fw_type(handle, image);
|
||||
|
||||
css_hdr = image;
|
||||
if ((css_hdr->header_len * css_dword_size) != header_len)
|
||||
goto err;
|
||||
if ((css_hdr->size * css_dword_size) != size)
|
||||
goto err;
|
||||
|
||||
if (handle->chip_info->dual_sign) {
|
||||
if (css_hdr->module_type != ICP_QAT_DUALSIGN_MODULE_TYPE)
|
||||
goto err;
|
||||
if (css_hdr->header_len != ICP_QAT_DUALSIGN_HDR_LEN)
|
||||
goto err;
|
||||
if (css_hdr->header_ver != ICP_QAT_DUALSIGN_HDR_VER)
|
||||
goto err;
|
||||
} else {
|
||||
if (css_hdr->header_len * css_dword_size != header_len)
|
||||
goto err;
|
||||
if (css_hdr->size * css_dword_size != size)
|
||||
goto err;
|
||||
if (size <= header_len)
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (fw_type != simg_type)
|
||||
goto err;
|
||||
if (size <= header_len)
|
||||
goto err;
|
||||
|
||||
size -= header_len;
|
||||
}
|
||||
|
||||
@@ -1515,6 +1542,115 @@ static int qat_uclo_build_auth_desc_RSA(struct icp_qat_fw_loader_handle *handle,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int qat_uclo_build_auth_desc_dualsign(struct icp_qat_fw_loader_handle *handle,
|
||||
char *image, unsigned int size,
|
||||
struct icp_firml_dram_desc *dram_desc,
|
||||
unsigned int fw_type,
|
||||
struct icp_qat_fw_auth_desc **desc)
|
||||
{
|
||||
struct icp_qat_simg_ae_mode *simg_ae_mode;
|
||||
struct icp_qat_fw_auth_desc *auth_desc;
|
||||
unsigned int chunk_offset, img_offset;
|
||||
u64 bus_addr, addr;
|
||||
char *virt_addr;
|
||||
|
||||
virt_addr = dram_desc->dram_base_addr_v;
|
||||
virt_addr += sizeof(struct icp_qat_auth_chunk);
|
||||
bus_addr = dram_desc->dram_bus_addr + sizeof(struct icp_qat_auth_chunk);
|
||||
|
||||
auth_desc = dram_desc->dram_base_addr_v;
|
||||
auth_desc->img_len = size - qat_uclo_simg_hdr2sign_len(handle);
|
||||
auth_desc->css_hdr_high = upper_32_bits(bus_addr);
|
||||
auth_desc->css_hdr_low = lower_32_bits(bus_addr);
|
||||
memcpy(virt_addr, image, ICP_QAT_DUALSIGN_OPAQUE_HDR_LEN);
|
||||
|
||||
img_offset = ICP_QAT_DUALSIGN_OPAQUE_HDR_LEN;
|
||||
chunk_offset = ICP_QAT_DUALSIGN_OPAQUE_HDR_ALIGN_LEN;
|
||||
|
||||
/* RSA pub key */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->fwsk_pub_high = upper_32_bits(addr);
|
||||
auth_desc->fwsk_pub_low = lower_32_bits(addr);
|
||||
memcpy(virt_addr + chunk_offset, image + img_offset, ICP_QAT_CSS_FWSK_MODULUS_LEN(handle));
|
||||
|
||||
img_offset += ICP_QAT_CSS_FWSK_MODULUS_LEN(handle);
|
||||
chunk_offset += ICP_QAT_CSS_FWSK_MODULUS_LEN(handle);
|
||||
/* RSA padding */
|
||||
memset(virt_addr + chunk_offset, 0, ICP_QAT_CSS_FWSK_PAD_LEN(handle));
|
||||
|
||||
chunk_offset += ICP_QAT_CSS_FWSK_PAD_LEN(handle);
|
||||
/* RSA exponent */
|
||||
memcpy(virt_addr + chunk_offset, image + img_offset, ICP_QAT_CSS_FWSK_EXPONENT_LEN(handle));
|
||||
|
||||
img_offset += ICP_QAT_CSS_FWSK_EXPONENT_LEN(handle);
|
||||
chunk_offset += ICP_QAT_CSS_FWSK_EXPONENT_LEN(handle);
|
||||
/* RSA signature */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->signature_high = upper_32_bits(addr);
|
||||
auth_desc->signature_low = lower_32_bits(addr);
|
||||
memcpy(virt_addr + chunk_offset, image + img_offset, ICP_QAT_CSS_SIGNATURE_LEN(handle));
|
||||
|
||||
img_offset += ICP_QAT_CSS_SIGNATURE_LEN(handle);
|
||||
chunk_offset += ICP_QAT_CSS_SIGNATURE_LEN(handle);
|
||||
/* XMSS pubkey */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->xmss_pubkey_high = upper_32_bits(addr);
|
||||
auth_desc->xmss_pubkey_low = lower_32_bits(addr);
|
||||
memcpy(virt_addr + chunk_offset, image + img_offset, ICP_QAT_DUALSIGN_XMSS_PUBKEY_LEN);
|
||||
|
||||
img_offset += ICP_QAT_DUALSIGN_XMSS_PUBKEY_LEN;
|
||||
chunk_offset += ICP_QAT_DUALSIGN_XMSS_PUBKEY_LEN;
|
||||
/* XMSS signature */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->xmss_sig_high = upper_32_bits(addr);
|
||||
auth_desc->xmss_sig_low = lower_32_bits(addr);
|
||||
memcpy(virt_addr + chunk_offset, image + img_offset, ICP_QAT_DUALSIGN_XMSS_SIG_LEN);
|
||||
|
||||
img_offset += ICP_QAT_DUALSIGN_XMSS_SIG_LEN;
|
||||
chunk_offset += ICP_QAT_DUALSIGN_XMSS_SIG_ALIGN_LEN;
|
||||
|
||||
if (dram_desc->dram_size < (chunk_offset + auth_desc->img_len)) {
|
||||
pr_err("auth chunk memory size is not enough to store data\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
/* Signed data */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->img_high = upper_32_bits(addr);
|
||||
auth_desc->img_low = lower_32_bits(addr);
|
||||
memcpy(virt_addr + chunk_offset, image + img_offset, auth_desc->img_len);
|
||||
|
||||
chunk_offset += ICP_QAT_DUALSIGN_MISC_INFO_LEN;
|
||||
/* AE firmware */
|
||||
if (fw_type == CSS_AE_FIRMWARE) {
|
||||
/* AE mode data */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->img_ae_mode_data_high = upper_32_bits(addr);
|
||||
auth_desc->img_ae_mode_data_low = lower_32_bits(addr);
|
||||
simg_ae_mode =
|
||||
(struct icp_qat_simg_ae_mode *)(virt_addr + chunk_offset);
|
||||
auth_desc->ae_mask = simg_ae_mode->ae_mask & handle->cfg_ae_mask;
|
||||
|
||||
chunk_offset += sizeof(struct icp_qat_simg_ae_mode);
|
||||
/* AE init seq */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->img_ae_init_data_high = upper_32_bits(addr);
|
||||
auth_desc->img_ae_init_data_low = lower_32_bits(addr);
|
||||
|
||||
chunk_offset += ICP_QAT_SIMG_AE_INIT_SEQ_LEN;
|
||||
/* AE instructions */
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->img_ae_insts_high = upper_32_bits(addr);
|
||||
auth_desc->img_ae_insts_low = lower_32_bits(addr);
|
||||
} else {
|
||||
addr = bus_addr + chunk_offset;
|
||||
auth_desc->img_ae_insts_high = upper_32_bits(addr);
|
||||
auth_desc->img_ae_insts_low = lower_32_bits(addr);
|
||||
}
|
||||
*desc = auth_desc;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int qat_uclo_map_auth_fw(struct icp_qat_fw_loader_handle *handle,
|
||||
char *image, unsigned int size,
|
||||
struct icp_qat_fw_auth_desc **desc)
|
||||
@@ -1533,6 +1669,10 @@ static int qat_uclo_map_auth_fw(struct icp_qat_fw_loader_handle *handle,
|
||||
auth_chunk->chunk_size = img_desc.dram_size;
|
||||
auth_chunk->chunk_bus_addr = img_desc.dram_bus_addr;
|
||||
|
||||
if (handle->chip_info->dual_sign)
|
||||
return qat_uclo_build_auth_desc_dualsign(handle, image, size, &img_desc,
|
||||
simg_fw_type, desc);
|
||||
|
||||
return qat_uclo_build_auth_desc_RSA(handle, image, size, &img_desc,
|
||||
simg_fw_type, desc);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user