f-stack/dpdk/drivers/crypto/ipsec_mb/ipsec_mb_private.h

492 lines
13 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(c) 2021 Intel Corporation
*/
#ifndef _IPSEC_MB_PRIVATE_H_
#define _IPSEC_MB_PRIVATE_H_
#if defined(RTE_ARCH_ARM)
#include <ipsec-mb.h>
#else
#include <intel-ipsec-mb.h>
#endif
#include <cryptodev_pmd.h>
#include <bus_vdev_driver.h>
#if defined(RTE_LIB_SECURITY)
#define IPSEC_MB_DOCSIS_SEC_ENABLED 1
#include <rte_security.h>
#include <rte_security_driver.h>
#endif
/* Maximum length for digest */
#define DIGEST_LENGTH_MAX 64
/* Maximum length for memzone name */
#define IPSEC_MB_MAX_MZ_NAME 32
/* ipsec mb multi-process queue pair config */
#define IPSEC_MB_MP_MSG "ipsec_mb_mp_msg"
enum ipsec_mb_vector_mode {
IPSEC_MB_NOT_SUPPORTED = 0,
IPSEC_MB_SSE,
IPSEC_MB_AVX,
IPSEC_MB_AVX2,
IPSEC_MB_AVX512,
IPSEC_MB_ARM64,
};
extern enum ipsec_mb_vector_mode vector_mode;
/** IMB_MGR instances, one per thread */
extern RTE_DEFINE_PER_LCORE(IMB_MGR *, mb_mgr);
#define CRYPTODEV_NAME_AESNI_MB_PMD crypto_aesni_mb
/**< IPSEC Multi buffer aesni_mb PMD device name */
#define CRYPTODEV_NAME_AESNI_GCM_PMD crypto_aesni_gcm
/**< IPSEC Multi buffer PMD aesni_gcm device name */
#define CRYPTODEV_NAME_KASUMI_PMD crypto_kasumi
/**< IPSEC Multi buffer PMD kasumi device name */
#define CRYPTODEV_NAME_SNOW3G_PMD crypto_snow3g
/**< IPSEC Multi buffer PMD snow3g device name */
#define CRYPTODEV_NAME_ZUC_PMD crypto_zuc
/**< IPSEC Multi buffer PMD zuc device name */
#define CRYPTODEV_NAME_CHACHA20_POLY1305_PMD crypto_chacha20_poly1305
/**< IPSEC Multi buffer PMD chacha20_poly1305 device name */
/** PMD LOGTYPE DRIVER, common to all PMDs */
extern int ipsec_mb_logtype_driver;
#define IPSEC_MB_LOG(level, fmt, ...) \
rte_log(RTE_LOG_##level, ipsec_mb_logtype_driver, \
"%s() line %u: " fmt "\n", __func__, __LINE__, ##__VA_ARGS__)
/** All supported device types */
enum ipsec_mb_pmd_types {
IPSEC_MB_PMD_TYPE_AESNI_MB = 0,
IPSEC_MB_PMD_TYPE_AESNI_GCM,
IPSEC_MB_PMD_TYPE_KASUMI,
IPSEC_MB_PMD_TYPE_SNOW3G,
IPSEC_MB_PMD_TYPE_ZUC,
IPSEC_MB_PMD_TYPE_CHACHA20_POLY1305,
IPSEC_MB_N_PMD_TYPES
};
/** Crypto operations */
enum ipsec_mb_operation {
IPSEC_MB_OP_ENCRYPT_THEN_HASH_GEN = 0,
IPSEC_MB_OP_HASH_VERIFY_THEN_DECRYPT,
IPSEC_MB_OP_HASH_GEN_THEN_ENCRYPT,
IPSEC_MB_OP_DECRYPT_THEN_HASH_VERIFY,
IPSEC_MB_OP_ENCRYPT_ONLY,
IPSEC_MB_OP_DECRYPT_ONLY,
IPSEC_MB_OP_HASH_GEN_ONLY,
IPSEC_MB_OP_HASH_VERIFY_ONLY,
IPSEC_MB_OP_AEAD_AUTHENTICATED_DECRYPT,
IPSEC_MB_OP_AEAD_AUTHENTICATED_ENCRYPT,
IPSEC_MB_OP_NOT_SUPPORTED
};
extern uint8_t pmd_driver_id_aesni_mb;
extern uint8_t pmd_driver_id_aesni_gcm;
extern uint8_t pmd_driver_id_kasumi;
extern uint8_t pmd_driver_id_snow3g;
extern uint8_t pmd_driver_id_zuc;
extern uint8_t pmd_driver_id_chacha20_poly1305;
/** Helper function. Gets driver ID based on PMD type */
static __rte_always_inline uint8_t
ipsec_mb_get_driver_id(enum ipsec_mb_pmd_types pmd_type)
{
switch (pmd_type) {
case IPSEC_MB_PMD_TYPE_AESNI_MB:
return pmd_driver_id_aesni_mb;
case IPSEC_MB_PMD_TYPE_AESNI_GCM:
return pmd_driver_id_aesni_gcm;
case IPSEC_MB_PMD_TYPE_KASUMI:
return pmd_driver_id_kasumi;
case IPSEC_MB_PMD_TYPE_SNOW3G:
return pmd_driver_id_snow3g;
case IPSEC_MB_PMD_TYPE_ZUC:
return pmd_driver_id_zuc;
case IPSEC_MB_PMD_TYPE_CHACHA20_POLY1305:
return pmd_driver_id_chacha20_poly1305;
default:
break;
}
return UINT8_MAX;
}
/** Common private data structure for each PMD */
struct ipsec_mb_dev_private {
enum ipsec_mb_pmd_types pmd_type;
/**< PMD type */
uint32_t max_nb_queue_pairs;
/**< Max number of queue pairs supported by device */
__extension__ uint8_t priv[];
};
/** IPSEC Multi buffer queue pair common queue pair data for all PMDs */
struct ipsec_mb_qp {
uint16_t id;
/**< Queue Pair Identifier */
char name[RTE_CRYPTODEV_NAME_MAX_LEN];
struct rte_ring *ingress_queue;
/**< Ring for placing operations ready for processing */
struct rte_mempool *sess_mp;
/**< Session Private Data Mempool */
struct rte_cryptodev_stats stats;
/**< Queue pair statistics */
enum ipsec_mb_pmd_types pmd_type;
/**< pmd type */
uint8_t digest_idx;
/**< Index of the next
* slot to be used in temp_digests,
* to store the digest for a given operation
*/
uint16_t qp_used_by_pid;
/**< The process id used for queue pairs **/
IMB_MGR *mb_mgr;
/**< Multi buffer manager */
const struct rte_memzone *mb_mgr_mz;
/**< Shared memzone for storing mb_mgr */
__extension__ uint8_t additional_data[];
/**< Storing PMD specific additional data */
};
/** Request types for IPC. */
enum ipsec_mb_mp_req_type {
RTE_IPSEC_MB_MP_REQ_NONE, /**< unknown event type */
RTE_IPSEC_MB_MP_REQ_QP_SET, /**< Queue pair setup request */
RTE_IPSEC_MB_MP_REQ_QP_FREE /**< Queue pair free request */
};
/** Parameters for IPC. */
struct ipsec_mb_mp_param {
enum ipsec_mb_mp_req_type type; /**< IPC request type */
int dev_id;
/**< The identifier of the device */
int qp_id;
/**< The index of the queue pair to be configured */
int socket_id;
/**< Socket to allocate resources on */
uint16_t process_id;
/**< The pid who send out the requested */
uint32_t nb_descriptors;
/**< Number of descriptors per queue pair */
void *mp_session;
/**< The mempool for creating session in sessionless mode */
int result;
/**< The request result for response message */
};
int
ipsec_mb_ipc_request(const struct rte_mp_msg *mp_msg, const void *peer);
static __rte_always_inline void *
ipsec_mb_get_qp_private_data(struct ipsec_mb_qp *qp)
{
return (void *)qp->additional_data;
}
/** Helper function. Allocates job manager */
static __rte_always_inline IMB_MGR *
alloc_init_mb_mgr(void)
{
IMB_MGR *mb_mgr = alloc_mb_mgr(0);
if (unlikely(mb_mgr == NULL)) {
IPSEC_MB_LOG(ERR, "Failed to allocate IMB_MGR data\n");
return NULL;
}
init_mb_mgr_auto(mb_mgr, NULL);
return mb_mgr;
}
/** Helper function. Gets per thread job manager */
static __rte_always_inline IMB_MGR *
get_per_thread_mb_mgr(void)
{
if (unlikely(RTE_PER_LCORE(mb_mgr) == NULL))
RTE_PER_LCORE(mb_mgr) = alloc_init_mb_mgr();
return RTE_PER_LCORE(mb_mgr);
}
/** Helper function. Gets mode and chained xforms from the xform */
static __rte_always_inline int
ipsec_mb_parse_xform(const struct rte_crypto_sym_xform *xform,
enum ipsec_mb_operation *mode,
const struct rte_crypto_sym_xform **auth_xform,
const struct rte_crypto_sym_xform **cipher_xform,
const struct rte_crypto_sym_xform **aead_xform)
{
if (xform == NULL) {
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
const struct rte_crypto_sym_xform *next = xform->next;
if (xform->type == RTE_CRYPTO_SYM_XFORM_CIPHER) {
if (next == NULL) {
if (xform->cipher.op == RTE_CRYPTO_CIPHER_OP_ENCRYPT) {
*mode = IPSEC_MB_OP_ENCRYPT_ONLY;
*cipher_xform = xform;
*auth_xform = NULL;
return 0;
}
*mode = IPSEC_MB_OP_DECRYPT_ONLY;
*cipher_xform = xform;
*auth_xform = NULL;
return 0;
}
if (next->type != RTE_CRYPTO_SYM_XFORM_AUTH) {
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
if (xform->cipher.op == RTE_CRYPTO_CIPHER_OP_ENCRYPT) {
if (next->auth.op != RTE_CRYPTO_AUTH_OP_GENERATE) {
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
*mode = IPSEC_MB_OP_ENCRYPT_THEN_HASH_GEN;
*cipher_xform = xform;
*auth_xform = xform->next;
return 0;
}
if (next->auth.op != RTE_CRYPTO_AUTH_OP_VERIFY) {
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
*mode = IPSEC_MB_OP_DECRYPT_THEN_HASH_VERIFY;
*cipher_xform = xform;
*auth_xform = xform->next;
return 0;
}
if (xform->type == RTE_CRYPTO_SYM_XFORM_AUTH) {
if (next == NULL) {
if (xform->auth.op == RTE_CRYPTO_AUTH_OP_GENERATE) {
*mode = IPSEC_MB_OP_HASH_GEN_ONLY;
*auth_xform = xform;
*cipher_xform = NULL;
return 0;
}
*mode = IPSEC_MB_OP_HASH_VERIFY_ONLY;
*auth_xform = xform;
*cipher_xform = NULL;
return 0;
}
if (next->type != RTE_CRYPTO_SYM_XFORM_CIPHER) {
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
if (xform->auth.op == RTE_CRYPTO_AUTH_OP_GENERATE) {
if (next->cipher.op != RTE_CRYPTO_CIPHER_OP_ENCRYPT) {
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
*mode = IPSEC_MB_OP_HASH_GEN_THEN_ENCRYPT;
*auth_xform = xform;
*cipher_xform = xform->next;
return 0;
}
if (next->cipher.op != RTE_CRYPTO_CIPHER_OP_DECRYPT) {
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
*mode = IPSEC_MB_OP_HASH_VERIFY_THEN_DECRYPT;
*auth_xform = xform;
*cipher_xform = xform->next;
return 0;
}
if (xform->type == RTE_CRYPTO_SYM_XFORM_AEAD) {
if (xform->aead.op == RTE_CRYPTO_AEAD_OP_ENCRYPT) {
/*
* CCM requires to hash first and cipher later
* when encrypting
*/
if (xform->aead.algo == RTE_CRYPTO_AEAD_AES_CCM) {
*mode = IPSEC_MB_OP_AEAD_AUTHENTICATED_DECRYPT;
*aead_xform = xform;
return 0;
} else {
*mode =
IPSEC_MB_OP_AEAD_AUTHENTICATED_ENCRYPT;
*aead_xform = xform;
return 0;
}
} else {
if (xform->aead.algo == RTE_CRYPTO_AEAD_AES_CCM) {
*mode = IPSEC_MB_OP_AEAD_AUTHENTICATED_ENCRYPT;
*aead_xform = xform;
return 0;
}
*mode = IPSEC_MB_OP_AEAD_AUTHENTICATED_DECRYPT;
*aead_xform = xform;
return 0;
}
}
*mode = IPSEC_MB_OP_NOT_SUPPORTED;
return -ENOTSUP;
}
/** Device creation function */
int
ipsec_mb_create(struct rte_vdev_device *vdev,
enum ipsec_mb_pmd_types pmd_type);
/** Device remove function */
int
ipsec_mb_remove(struct rte_vdev_device *vdev);
/** Configure queue pair PMD type specific data */
typedef int (*ipsec_mb_queue_pair_configure_t)(struct ipsec_mb_qp *qp);
/** Configure session PMD type specific data */
typedef int (*ipsec_mb_session_configure_t)(IMB_MGR *mbr_mgr,
void *session_private,
const struct rte_crypto_sym_xform *xform);
/** Configure internals PMD type specific data */
typedef int (*ipsec_mb_dev_configure_t)(struct rte_cryptodev *dev);
/** Per PMD type operation and data */
struct ipsec_mb_internals {
uint8_t is_configured;
dequeue_pkt_burst_t dequeue_burst;
ipsec_mb_dev_configure_t dev_config;
ipsec_mb_queue_pair_configure_t queue_pair_configure;
ipsec_mb_session_configure_t session_configure;
const struct rte_cryptodev_capabilities *caps;
struct rte_cryptodev_ops *ops;
struct rte_security_ops *security_ops;
uint64_t feature_flags;
uint32_t session_priv_size;
uint32_t qp_priv_size;
uint32_t internals_priv_size;
};
/** Global PMD type specific data */
extern struct ipsec_mb_internals ipsec_mb_pmds[IPSEC_MB_N_PMD_TYPES];
int
ipsec_mb_config(struct rte_cryptodev *dev,
struct rte_cryptodev_config *config);
int
ipsec_mb_start(struct rte_cryptodev *dev);
void
ipsec_mb_stop(struct rte_cryptodev *dev);
int
ipsec_mb_close(struct rte_cryptodev *dev);
void
ipsec_mb_stats_get(struct rte_cryptodev *dev,
struct rte_cryptodev_stats *stats);
void
ipsec_mb_stats_reset(struct rte_cryptodev *dev);
void
ipsec_mb_info_get(struct rte_cryptodev *dev,
struct rte_cryptodev_info *dev_info);
int
ipsec_mb_qp_release(struct rte_cryptodev *dev, uint16_t qp_id);
int
ipsec_mb_qp_set_unique_name(struct rte_cryptodev *dev, struct ipsec_mb_qp *qp);
int
ipsec_mb_qp_setup(struct rte_cryptodev *dev, uint16_t qp_id,
const struct rte_cryptodev_qp_conf *qp_conf,
int socket_id);
/** Returns the size of the aesni multi-buffer session structure */
unsigned
ipsec_mb_sym_session_get_size(struct rte_cryptodev *dev);
/** Configure an aesni multi-buffer session from a crypto xform chain */
int ipsec_mb_sym_session_configure(
struct rte_cryptodev *dev,
struct rte_crypto_sym_xform *xform,
struct rte_cryptodev_sym_session *sess);
/** Clear the memory of session so it does not leave key material behind */
void
ipsec_mb_sym_session_clear(struct rte_cryptodev *dev,
struct rte_cryptodev_sym_session *sess);
/** Get session from op. If sessionless create a session */
static __rte_always_inline void *
ipsec_mb_get_session_private(struct ipsec_mb_qp *qp, struct rte_crypto_op *op)
{
struct rte_cryptodev_sym_session *sess = NULL;
uint32_t driver_id = ipsec_mb_get_driver_id(qp->pmd_type);
struct rte_crypto_sym_op *sym_op = op->sym;
uint8_t sess_type = op->sess_type;
void *_sess;
struct ipsec_mb_internals *pmd_data = &ipsec_mb_pmds[qp->pmd_type];
switch (sess_type) {
case RTE_CRYPTO_OP_WITH_SESSION:
if (likely(sym_op->session != NULL))
sess = sym_op->session;
else
goto error_exit;
break;
case RTE_CRYPTO_OP_SESSIONLESS:
if (!qp->sess_mp ||
rte_mempool_get(qp->sess_mp, (void **)&_sess))
return NULL;
sess = _sess;
if (sess->sess_data_sz < pmd_data->session_priv_size) {
rte_mempool_put(qp->sess_mp, _sess);
goto error_exit;
}
if (unlikely(pmd_data->session_configure(qp->mb_mgr,
CRYPTODEV_GET_SYM_SESS_PRIV(sess), sym_op->xform) != 0)) {
rte_mempool_put(qp->sess_mp, _sess);
goto error_exit;
}
sess->driver_id = driver_id;
sym_op->session = sess;
break;
default:
IPSEC_MB_LOG(ERR, "Unrecognized session type %u", sess_type);
}
return CRYPTODEV_GET_SYM_SESS_PRIV(sess);
error_exit:
op->status = RTE_CRYPTO_OP_STATUS_INVALID_SESSION;
return NULL;
}
#endif /* _IPSEC_MB_PRIVATE_H_ */