]> www.infradead.org Git - users/hch/misc.git/commitdiff
crypto: qat - enable dc chaining service
authorAdam Guerin <adam.guerin@intel.com>
Thu, 14 Sep 2023 14:14:13 +0000 (15:14 +0100)
committerHerbert Xu <herbert@gondor.apana.org.au>
Wed, 20 Sep 2023 05:15:54 +0000 (13:15 +0800)
QAT GEN4 devices support chained compression operations. These
allow, with a single request to firmware, to hash then compress
data.

Extend the configuration to enable such mode. The cfg_services
operations in sysfs are extended to allow the string "dcc". When
selected, the driver downloads to the device both the symmetric
crypto and the compression firmware images and sends an admin message
to firmware which enables `chained` operations.
In addition, it sets the device's capabilities as the combination
of compression and symmetric crypto capabilities, while excluding
the ICP_ACCEL_CAPABILITIES_CRYPTO_SYMMETRIC bit to indicate
that in this mode, symmetric crypto instances are not supported.

When "dcc" is enabled, the device will handle compression requests
as if the "dc" configuration is loaded ("dcc" is a variation of "dc")
and the driver will register the acomp algorithms.

As for the other extended configurations, "dcc" is only available for
qat_4xxx devices and the chaining service will be only accessible from
user space.

Signed-off-by: Adam Guerin <adam.guerin@intel.com>
Reviewed-by: Giovanni Cabiddu <giovanni.cabiddu@intel.com>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
Documentation/ABI/testing/sysfs-driver-qat
drivers/crypto/intel/qat/qat_4xxx/adf_4xxx_hw_data.c
drivers/crypto/intel/qat/qat_4xxx/adf_drv.c
drivers/crypto/intel/qat/qat_common/adf_admin.c
drivers/crypto/intel/qat/qat_common/adf_cfg_services.h
drivers/crypto/intel/qat/qat_common/adf_cfg_strings.h
drivers/crypto/intel/qat/qat_common/icp_qat_fw_init_admin.h

index ef6d6c57105efbf199495f3c9718a2ae9e0eb68f..96834d103a09e2ad0f156d9b2756f274de53a24b 100644 (file)
@@ -29,6 +29,8 @@ Description:  (RW) Reports the current configuration of the QAT device.
                  services
                * asym;sym: identical to sym;asym
                * dc: the device is configured for running compression services
+               * dcc: identical to dc but enables the dc chaining feature,
+                 hash then compression. If this is not required chose dc
                * sym: the device is configured for running symmetric crypto
                  services
                * asym: the device is configured for running asymmetric crypto
index cc2285b9f17b4ae313511d743a728c9b00f3c363..12b5d18191119d14f9122bc07a21ba820453628c 100644 (file)
@@ -76,11 +76,18 @@ static const struct adf_fw_config adf_fw_sym_dc_config[] = {
        {0x100, ADF_FW_ADMIN_OBJ},
 };
 
+static const struct adf_fw_config adf_fw_dcc_config[] = {
+       {0xF0, ADF_FW_DC_OBJ},
+       {0xF, ADF_FW_SYM_OBJ},
+       {0x100, ADF_FW_ADMIN_OBJ},
+};
+
 static_assert(ARRAY_SIZE(adf_fw_cy_config) == ARRAY_SIZE(adf_fw_dc_config));
 static_assert(ARRAY_SIZE(adf_fw_cy_config) == ARRAY_SIZE(adf_fw_sym_config));
 static_assert(ARRAY_SIZE(adf_fw_cy_config) == ARRAY_SIZE(adf_fw_asym_config));
 static_assert(ARRAY_SIZE(adf_fw_cy_config) == ARRAY_SIZE(adf_fw_asym_dc_config));
 static_assert(ARRAY_SIZE(adf_fw_cy_config) == ARRAY_SIZE(adf_fw_sym_dc_config));
+static_assert(ARRAY_SIZE(adf_fw_cy_config) == ARRAY_SIZE(adf_fw_dcc_config));
 
 /* Worker thread to service arbiter mappings */
 static const u32 default_thrd_to_arb_map[ADF_4XXX_MAX_ACCELENGINES] = {
@@ -95,6 +102,12 @@ static const u32 thrd_to_arb_map_dc[ADF_4XXX_MAX_ACCELENGINES] = {
        0x0
 };
 
+static const u32 thrd_to_arb_map_dcc[ADF_4XXX_MAX_ACCELENGINES] = {
+       0x00000000, 0x00000000, 0x00000000, 0x00000000,
+       0x0000FFFF, 0x0000FFFF, 0x0000FFFF, 0x0000FFFF,
+       0x0
+};
+
 static struct adf_hw_device_class adf_4xxx_class = {
        .name = ADF_4XXX_DEVICE_NAME,
        .type = DEV_4XXX,
@@ -189,6 +202,7 @@ static u32 get_accel_cap(struct adf_accel_dev *accel_dev)
 {
        struct pci_dev *pdev = accel_dev->accel_pci_dev.pci_dev;
        u32 capabilities_sym, capabilities_asym, capabilities_dc;
+       u32 capabilities_dcc;
        u32 fusectl1;
 
        /* Read accelerator capabilities mask */
@@ -261,6 +275,14 @@ static u32 get_accel_cap(struct adf_accel_dev *accel_dev)
                return capabilities_sym | capabilities_asym;
        case SVC_DC:
                return capabilities_dc;
+       case SVC_DCC:
+               /*
+                * Sym capabilities are available for chaining operations,
+                * but sym crypto instances cannot be supported
+                */
+               capabilities_dcc = capabilities_dc | capabilities_sym;
+               capabilities_dcc &= ~ICP_ACCEL_CAPABILITIES_CRYPTO_SYMMETRIC;
+               return capabilities_dcc;
        case SVC_SYM:
                return capabilities_sym;
        case SVC_ASYM:
@@ -286,6 +308,8 @@ static const u32 *adf_get_arbiter_mapping(struct adf_accel_dev *accel_dev)
        switch (get_service_enabled(accel_dev)) {
        case SVC_DC:
                return thrd_to_arb_map_dc;
+       case SVC_DCC:
+               return thrd_to_arb_map_dcc;
        default:
                return default_thrd_to_arb_map;
        }
@@ -383,6 +407,9 @@ static const char *uof_get_name(struct adf_accel_dev *accel_dev, u32 obj_num,
        case SVC_DC:
                id = adf_fw_dc_config[obj_num].obj;
                break;
+       case SVC_DCC:
+               id = adf_fw_dcc_config[obj_num].obj;
+               break;
        case SVC_SYM:
                id = adf_fw_sym_config[obj_num].obj;
                break;
@@ -429,6 +456,8 @@ static u32 uof_get_ae_mask(struct adf_accel_dev *accel_dev, u32 obj_num)
                return adf_fw_cy_config[obj_num].ae_mask;
        case SVC_DC:
                return adf_fw_dc_config[obj_num].ae_mask;
+       case SVC_DCC:
+               return adf_fw_dcc_config[obj_num].ae_mask;
        case SVC_CY2:
                return adf_fw_cy_config[obj_num].ae_mask;
        case SVC_SYM:
index 204a00a204f2dd480020cd9494819e2301fdc6b1..90f5c1ca7b8d855ac8af38acea702da87b1e0be8 100644 (file)
@@ -279,6 +279,7 @@ int adf_gen4_dev_config(struct adf_accel_dev *accel_dev)
                ret = adf_crypto_dev_config(accel_dev);
                break;
        case SVC_DC:
+       case SVC_DCC:
                ret = adf_comp_dev_config(accel_dev);
                break;
        default:
index ff790823b86861c5102ac7b011d32686d04ccd36..194d64d4b99a1b4ac350df8fee3c19064e835cee 100644 (file)
@@ -8,6 +8,7 @@
 #include <linux/dma-mapping.h>
 #include "adf_accel_devices.h"
 #include "adf_common_drv.h"
+#include "adf_cfg.h"
 #include "adf_heartbeat.h"
 #include "icp_qat_fw_init_admin.h"
 
@@ -212,6 +213,17 @@ int adf_get_fw_timestamp(struct adf_accel_dev *accel_dev, u64 *timestamp)
        return 0;
 }
 
+static int adf_set_chaining(struct adf_accel_dev *accel_dev)
+{
+       u32 ae_mask = GET_HW_DATA(accel_dev)->ae_mask;
+       struct icp_qat_fw_init_admin_resp resp = { };
+       struct icp_qat_fw_init_admin_req req = { };
+
+       req.cmd_id = ICP_QAT_FW_DC_CHAIN_INIT;
+
+       return adf_send_admin(accel_dev, &req, &resp, ae_mask);
+}
+
 static int adf_get_dc_capabilities(struct adf_accel_dev *accel_dev,
                                   u32 *capabilities)
 {
@@ -284,6 +296,19 @@ int adf_send_admin_hb_timer(struct adf_accel_dev *accel_dev, uint32_t ticks)
        return adf_send_admin(accel_dev, &req, &resp, ae_mask);
 }
 
+static bool is_dcc_enabled(struct adf_accel_dev *accel_dev)
+{
+       char services[ADF_CFG_MAX_VAL_LEN_IN_BYTES] = {0};
+       int ret;
+
+       ret = adf_cfg_get_param_value(accel_dev, ADF_GENERAL_SEC,
+                                     ADF_SERVICES_ENABLED, services);
+       if (ret)
+               return false;
+
+       return !strcmp(services, "dcc");
+}
+
 /**
  * adf_send_admin_init() - Function sends init message to FW
  * @accel_dev: Pointer to acceleration device.
@@ -297,6 +322,16 @@ int adf_send_admin_init(struct adf_accel_dev *accel_dev)
        u32 dc_capabilities = 0;
        int ret;
 
+       ret = adf_set_fw_constants(accel_dev);
+       if (ret)
+               return ret;
+
+       if (is_dcc_enabled(accel_dev)) {
+               ret = adf_set_chaining(accel_dev);
+               if (ret)
+                       return ret;
+       }
+
        ret = adf_get_dc_capabilities(accel_dev, &dc_capabilities);
        if (ret) {
                dev_err(&GET_DEV(accel_dev), "Cannot get dc capabilities\n");
@@ -304,10 +339,6 @@ int adf_send_admin_init(struct adf_accel_dev *accel_dev)
        }
        accel_dev->hw_device->extended_dc_capabilities = dc_capabilities;
 
-       ret = adf_set_fw_constants(accel_dev);
-       if (ret)
-               return ret;
-
        return adf_init_ae(accel_dev);
 }
 EXPORT_SYMBOL_GPL(adf_send_admin_init);
index 7fcb3b8f148a6ef0a849492f066f8a32e5ed8438..b353d40c5c6d0ac60ab5a647d0759fbb85512296 100644 (file)
@@ -9,6 +9,7 @@ enum adf_services {
        SVC_CY = 0,
        SVC_CY2,
        SVC_DC,
+       SVC_DCC,
        SVC_SYM,
        SVC_ASYM,
        SVC_DC_ASYM,
@@ -21,6 +22,7 @@ static const char *const adf_cfg_services[] = {
        [SVC_CY] = ADF_CFG_CY,
        [SVC_CY2] = ADF_CFG_ASYM_SYM,
        [SVC_DC] = ADF_CFG_DC,
+       [SVC_DCC] = ADF_CFG_DCC,
        [SVC_SYM] = ADF_CFG_SYM,
        [SVC_ASYM] = ADF_CFG_ASYM,
        [SVC_DC_ASYM] = ADF_CFG_DC_ASYM,
index 6066dc637352cae50e943fe0583e4340d0d83f45..322b76903a737d4e0fce0371a355f6a1f17fb0b0 100644 (file)
@@ -32,6 +32,7 @@
 #define ADF_CFG_DC_ASYM "dc;asym"
 #define ADF_CFG_SYM_DC "sym;dc"
 #define ADF_CFG_DC_SYM "dc;sym"
+#define ADF_CFG_DCC "dcc"
 #define ADF_SERVICES_ENABLED "ServicesEnabled"
 #define ADF_PM_IDLE_SUPPORT "PmIdleSupport"
 #define ADF_ETRMGR_COALESCING_ENABLED "InterruptCoalescingEnabled"
index 3e968a4bcc9cd51e8c52c77220694e517c565fa6..019a6443834e0b11ca395e565f7ae410303ff5fc 100644 (file)
@@ -16,6 +16,7 @@ enum icp_qat_fw_init_admin_cmd_id {
        ICP_QAT_FW_HEARTBEAT_SYNC = 7,
        ICP_QAT_FW_HEARTBEAT_GET = 8,
        ICP_QAT_FW_COMP_CAPABILITY_GET = 9,
+       ICP_QAT_FW_DC_CHAIN_INIT = 11,
        ICP_QAT_FW_HEARTBEAT_TIMER_SET = 13,
        ICP_QAT_FW_TIMER_GET = 19,
        ICP_QAT_FW_PM_STATE_CONFIG = 128,