f-stack/dpdk/drivers/crypto/ccp/ccp_crypto.c

2960 lines
83 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(c) 2018 Advanced Micro Devices, Inc. All rights reserved.
*/
#include <dirent.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <sys/mman.h>
#include <sys/queue.h>
#include <sys/types.h>
#include <unistd.h>
#include <openssl/sha.h>
#include <openssl/cmac.h> /*sub key apis*/
#include <openssl/evp.h> /*sub key apis*/
#include <rte_hexdump.h>
#include <rte_memzone.h>
#include <rte_malloc.h>
#include <rte_memory.h>
#include <rte_spinlock.h>
#include <rte_string_fns.h>
#include <rte_cryptodev_pmd.h>
#include "ccp_dev.h"
#include "ccp_crypto.h"
#include "ccp_pci.h"
#include "ccp_pmd_private.h"
#include <openssl/conf.h>
#include <openssl/err.h>
#include <openssl/hmac.h>
/* SHA initial context values */
static uint32_t ccp_sha1_init[SHA_COMMON_DIGEST_SIZE / sizeof(uint32_t)] = {
SHA1_H4, SHA1_H3,
SHA1_H2, SHA1_H1,
SHA1_H0, 0x0U,
0x0U, 0x0U,
};
uint32_t ccp_sha224_init[SHA256_DIGEST_SIZE / sizeof(uint32_t)] = {
SHA224_H7, SHA224_H6,
SHA224_H5, SHA224_H4,
SHA224_H3, SHA224_H2,
SHA224_H1, SHA224_H0,
};
uint32_t ccp_sha256_init[SHA256_DIGEST_SIZE / sizeof(uint32_t)] = {
SHA256_H7, SHA256_H6,
SHA256_H5, SHA256_H4,
SHA256_H3, SHA256_H2,
SHA256_H1, SHA256_H0,
};
uint64_t ccp_sha384_init[SHA512_DIGEST_SIZE / sizeof(uint64_t)] = {
SHA384_H7, SHA384_H6,
SHA384_H5, SHA384_H4,
SHA384_H3, SHA384_H2,
SHA384_H1, SHA384_H0,
};
uint64_t ccp_sha512_init[SHA512_DIGEST_SIZE / sizeof(uint64_t)] = {
SHA512_H7, SHA512_H6,
SHA512_H5, SHA512_H4,
SHA512_H3, SHA512_H2,
SHA512_H1, SHA512_H0,
};
#if defined(_MSC_VER)
#define SHA3_CONST(x) x
#else
#define SHA3_CONST(x) x##L
#endif
/** 'Words' here refers to uint64_t */
#define SHA3_KECCAK_SPONGE_WORDS \
(((1600) / 8) / sizeof(uint64_t))
typedef struct sha3_context_ {
uint64_t saved;
/**
* The portion of the input message that we
* didn't consume yet
*/
union {
uint64_t s[SHA3_KECCAK_SPONGE_WORDS];
/* Keccak's state */
uint8_t sb[SHA3_KECCAK_SPONGE_WORDS * 8];
/**total 200 ctx size**/
};
unsigned int byteIndex;
/**
* 0..7--the next byte after the set one
* (starts from 0; 0--none are buffered)
*/
unsigned int wordIndex;
/**
* 0..24--the next word to integrate input
* (starts from 0)
*/
unsigned int capacityWords;
/**
* the double size of the hash output in
* words (e.g. 16 for Keccak 512)
*/
} sha3_context;
#ifndef SHA3_ROTL64
#define SHA3_ROTL64(x, y) \
(((x) << (y)) | ((x) >> ((sizeof(uint64_t)*8) - (y))))
#endif
static const uint64_t keccakf_rndc[24] = {
SHA3_CONST(0x0000000000000001UL), SHA3_CONST(0x0000000000008082UL),
SHA3_CONST(0x800000000000808aUL), SHA3_CONST(0x8000000080008000UL),
SHA3_CONST(0x000000000000808bUL), SHA3_CONST(0x0000000080000001UL),
SHA3_CONST(0x8000000080008081UL), SHA3_CONST(0x8000000000008009UL),
SHA3_CONST(0x000000000000008aUL), SHA3_CONST(0x0000000000000088UL),
SHA3_CONST(0x0000000080008009UL), SHA3_CONST(0x000000008000000aUL),
SHA3_CONST(0x000000008000808bUL), SHA3_CONST(0x800000000000008bUL),
SHA3_CONST(0x8000000000008089UL), SHA3_CONST(0x8000000000008003UL),
SHA3_CONST(0x8000000000008002UL), SHA3_CONST(0x8000000000000080UL),
SHA3_CONST(0x000000000000800aUL), SHA3_CONST(0x800000008000000aUL),
SHA3_CONST(0x8000000080008081UL), SHA3_CONST(0x8000000000008080UL),
SHA3_CONST(0x0000000080000001UL), SHA3_CONST(0x8000000080008008UL)
};
static const unsigned int keccakf_rotc[24] = {
1, 3, 6, 10, 15, 21, 28, 36, 45, 55, 2, 14, 27, 41, 56, 8, 25, 43, 62,
18, 39, 61, 20, 44
};
static const unsigned int keccakf_piln[24] = {
10, 7, 11, 17, 18, 3, 5, 16, 8, 21, 24, 4, 15, 23, 19, 13, 12, 2, 20,
14, 22, 9, 6, 1
};
static enum ccp_cmd_order
ccp_get_cmd_id(const struct rte_crypto_sym_xform *xform)
{
enum ccp_cmd_order res = CCP_CMD_NOT_SUPPORTED;
if (xform == NULL)
return res;
if (xform->type == RTE_CRYPTO_SYM_XFORM_AUTH) {
if (xform->next == NULL)
return CCP_CMD_AUTH;
else if (xform->next->type == RTE_CRYPTO_SYM_XFORM_CIPHER)
return CCP_CMD_HASH_CIPHER;
}
if (xform->type == RTE_CRYPTO_SYM_XFORM_CIPHER) {
if (xform->next == NULL)
return CCP_CMD_CIPHER;
else if (xform->next->type == RTE_CRYPTO_SYM_XFORM_AUTH)
return CCP_CMD_CIPHER_HASH;
}
if (xform->type == RTE_CRYPTO_SYM_XFORM_AEAD)
return CCP_CMD_COMBINED;
return res;
}
/* partial hash using openssl */
static int partial_hash_sha1(uint8_t *data_in, uint8_t *data_out)
{
SHA_CTX ctx;
if (!SHA1_Init(&ctx))
return -EFAULT;
SHA1_Transform(&ctx, data_in);
rte_memcpy(data_out, &ctx, SHA_DIGEST_LENGTH);
return 0;
}
static int partial_hash_sha224(uint8_t *data_in, uint8_t *data_out)
{
SHA256_CTX ctx;
if (!SHA224_Init(&ctx))
return -EFAULT;
SHA256_Transform(&ctx, data_in);
rte_memcpy(data_out, &ctx,
SHA256_DIGEST_LENGTH);
return 0;
}
static int partial_hash_sha256(uint8_t *data_in, uint8_t *data_out)
{
SHA256_CTX ctx;
if (!SHA256_Init(&ctx))
return -EFAULT;
SHA256_Transform(&ctx, data_in);
rte_memcpy(data_out, &ctx,
SHA256_DIGEST_LENGTH);
return 0;
}
static int partial_hash_sha384(uint8_t *data_in, uint8_t *data_out)
{
SHA512_CTX ctx;
if (!SHA384_Init(&ctx))
return -EFAULT;
SHA512_Transform(&ctx, data_in);
rte_memcpy(data_out, &ctx,
SHA512_DIGEST_LENGTH);
return 0;
}
static int partial_hash_sha512(uint8_t *data_in, uint8_t *data_out)
{
SHA512_CTX ctx;
if (!SHA512_Init(&ctx))
return -EFAULT;
SHA512_Transform(&ctx, data_in);
rte_memcpy(data_out, &ctx,
SHA512_DIGEST_LENGTH);
return 0;
}
static void
keccakf(uint64_t s[25])
{
int i, j, round;
uint64_t t, bc[5];
#define KECCAK_ROUNDS 24
for (round = 0; round < KECCAK_ROUNDS; round++) {
/* Theta */
for (i = 0; i < 5; i++)
bc[i] = s[i] ^ s[i + 5] ^ s[i + 10] ^ s[i + 15] ^
s[i + 20];
for (i = 0; i < 5; i++) {
t = bc[(i + 4) % 5] ^ SHA3_ROTL64(bc[(i + 1) % 5], 1);
for (j = 0; j < 25; j += 5)
s[j + i] ^= t;
}
/* Rho Pi */
t = s[1];
for (i = 0; i < 24; i++) {
j = keccakf_piln[i];
bc[0] = s[j];
s[j] = SHA3_ROTL64(t, keccakf_rotc[i]);
t = bc[0];
}
/* Chi */
for (j = 0; j < 25; j += 5) {
for (i = 0; i < 5; i++)
bc[i] = s[j + i];
for (i = 0; i < 5; i++)
s[j + i] ^= (~bc[(i + 1) % 5]) &
bc[(i + 2) % 5];
}
/* Iota */
s[0] ^= keccakf_rndc[round];
}
}
static void
sha3_Init224(void *priv)
{
sha3_context *ctx = (sha3_context *) priv;
memset(ctx, 0, sizeof(*ctx));
ctx->capacityWords = 2 * 224 / (8 * sizeof(uint64_t));
}
static void
sha3_Init256(void *priv)
{
sha3_context *ctx = (sha3_context *) priv;
memset(ctx, 0, sizeof(*ctx));
ctx->capacityWords = 2 * 256 / (8 * sizeof(uint64_t));
}
static void
sha3_Init384(void *priv)
{
sha3_context *ctx = (sha3_context *) priv;
memset(ctx, 0, sizeof(*ctx));
ctx->capacityWords = 2 * 384 / (8 * sizeof(uint64_t));
}
static void
sha3_Init512(void *priv)
{
sha3_context *ctx = (sha3_context *) priv;
memset(ctx, 0, sizeof(*ctx));
ctx->capacityWords = 2 * 512 / (8 * sizeof(uint64_t));
}
/* This is simply the 'update' with the padding block.
* The padding block is 0x01 || 0x00* || 0x80. First 0x01 and last 0x80
* bytes are always present, but they can be the same byte.
*/
static void
sha3_Update(void *priv, void const *bufIn, size_t len)
{
sha3_context *ctx = (sha3_context *) priv;
unsigned int old_tail = (8 - ctx->byteIndex) & 7;
size_t words;
unsigned int tail;
size_t i;
const uint8_t *buf = bufIn;
if (len < old_tail) {
while (len--)
ctx->saved |= (uint64_t) (*(buf++)) <<
((ctx->byteIndex++) * 8);
return;
}
if (old_tail) {
len -= old_tail;
while (old_tail--)
ctx->saved |= (uint64_t) (*(buf++)) <<
((ctx->byteIndex++) * 8);
ctx->s[ctx->wordIndex] ^= ctx->saved;
ctx->byteIndex = 0;
ctx->saved = 0;
if (++ctx->wordIndex ==
(SHA3_KECCAK_SPONGE_WORDS - ctx->capacityWords)) {
keccakf(ctx->s);
ctx->wordIndex = 0;
}
}
words = len / sizeof(uint64_t);
tail = len - words * sizeof(uint64_t);
for (i = 0; i < words; i++, buf += sizeof(uint64_t)) {
const uint64_t t = (uint64_t) (buf[0]) |
((uint64_t) (buf[1]) << 8 * 1) |
((uint64_t) (buf[2]) << 8 * 2) |
((uint64_t) (buf[3]) << 8 * 3) |
((uint64_t) (buf[4]) << 8 * 4) |
((uint64_t) (buf[5]) << 8 * 5) |
((uint64_t) (buf[6]) << 8 * 6) |
((uint64_t) (buf[7]) << 8 * 7);
ctx->s[ctx->wordIndex] ^= t;
if (++ctx->wordIndex ==
(SHA3_KECCAK_SPONGE_WORDS - ctx->capacityWords)) {
keccakf(ctx->s);
ctx->wordIndex = 0;
}
}
while (tail--)
ctx->saved |= (uint64_t) (*(buf++)) << ((ctx->byteIndex++) * 8);
}
int partial_hash_sha3_224(uint8_t *data_in, uint8_t *data_out)
{
sha3_context *ctx;
int i;
ctx = rte_zmalloc("sha3-ctx", sizeof(sha3_context), 0);
if (!ctx) {
CCP_LOG_ERR("sha3-ctx creation failed");
return -ENOMEM;
}
sha3_Init224(ctx);
sha3_Update(ctx, data_in, SHA3_224_BLOCK_SIZE);
for (i = 0; i < CCP_SHA3_CTX_SIZE; i++, data_out++)
*data_out = ctx->sb[CCP_SHA3_CTX_SIZE - i - 1];
rte_free(ctx);
return 0;
}
int partial_hash_sha3_256(uint8_t *data_in, uint8_t *data_out)
{
sha3_context *ctx;
int i;
ctx = rte_zmalloc("sha3-ctx", sizeof(sha3_context), 0);
if (!ctx) {
CCP_LOG_ERR("sha3-ctx creation failed");
return -ENOMEM;
}
sha3_Init256(ctx);
sha3_Update(ctx, data_in, SHA3_256_BLOCK_SIZE);
for (i = 0; i < CCP_SHA3_CTX_SIZE; i++, data_out++)
*data_out = ctx->sb[CCP_SHA3_CTX_SIZE - i - 1];
rte_free(ctx);
return 0;
}
int partial_hash_sha3_384(uint8_t *data_in, uint8_t *data_out)
{
sha3_context *ctx;
int i;
ctx = rte_zmalloc("sha3-ctx", sizeof(sha3_context), 0);
if (!ctx) {
CCP_LOG_ERR("sha3-ctx creation failed");
return -ENOMEM;
}
sha3_Init384(ctx);
sha3_Update(ctx, data_in, SHA3_384_BLOCK_SIZE);
for (i = 0; i < CCP_SHA3_CTX_SIZE; i++, data_out++)
*data_out = ctx->sb[CCP_SHA3_CTX_SIZE - i - 1];
rte_free(ctx);
return 0;
}
int partial_hash_sha3_512(uint8_t *data_in, uint8_t *data_out)
{
sha3_context *ctx;
int i;
ctx = rte_zmalloc("sha3-ctx", sizeof(sha3_context), 0);
if (!ctx) {
CCP_LOG_ERR("sha3-ctx creation failed");
return -ENOMEM;
}
sha3_Init512(ctx);
sha3_Update(ctx, data_in, SHA3_512_BLOCK_SIZE);
for (i = 0; i < CCP_SHA3_CTX_SIZE; i++, data_out++)
*data_out = ctx->sb[CCP_SHA3_CTX_SIZE - i - 1];
rte_free(ctx);
return 0;
}
static int generate_partial_hash(struct ccp_session *sess)
{
uint8_t ipad[sess->auth.block_size];
uint8_t opad[sess->auth.block_size];
uint8_t *ipad_t, *opad_t;
uint32_t *hash_value_be32, hash_temp32[8];
uint64_t *hash_value_be64, hash_temp64[8];
int i, count;
uint8_t *hash_value_sha3;
opad_t = ipad_t = (uint8_t *)sess->auth.key;
hash_value_be32 = (uint32_t *)((uint8_t *)sess->auth.pre_compute);
hash_value_be64 = (uint64_t *)((uint8_t *)sess->auth.pre_compute);
/* considering key size is always equal to block size of algorithm */
for (i = 0; i < sess->auth.block_size; i++) {
ipad[i] = (ipad_t[i] ^ HMAC_IPAD_VALUE);
opad[i] = (opad_t[i] ^ HMAC_OPAD_VALUE);
}
switch (sess->auth.algo) {
case CCP_AUTH_ALGO_SHA1_HMAC:
count = SHA1_DIGEST_SIZE >> 2;
if (partial_hash_sha1(ipad, (uint8_t *)hash_temp32))
return -1;
for (i = 0; i < count; i++, hash_value_be32++)
*hash_value_be32 = hash_temp32[count - 1 - i];
hash_value_be32 = (uint32_t *)((uint8_t *)sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha1(opad, (uint8_t *)hash_temp32))
return -1;
for (i = 0; i < count; i++, hash_value_be32++)
*hash_value_be32 = hash_temp32[count - 1 - i];
return 0;
case CCP_AUTH_ALGO_SHA224_HMAC:
count = SHA256_DIGEST_SIZE >> 2;
if (partial_hash_sha224(ipad, (uint8_t *)hash_temp32))
return -1;
for (i = 0; i < count; i++, hash_value_be32++)
*hash_value_be32 = hash_temp32[count - 1 - i];
hash_value_be32 = (uint32_t *)((uint8_t *)sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha224(opad, (uint8_t *)hash_temp32))
return -1;
for (i = 0; i < count; i++, hash_value_be32++)
*hash_value_be32 = hash_temp32[count - 1 - i];
return 0;
case CCP_AUTH_ALGO_SHA3_224_HMAC:
hash_value_sha3 = sess->auth.pre_compute;
if (partial_hash_sha3_224(ipad, hash_value_sha3))
return -1;
hash_value_sha3 = (uint8_t *)(sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha3_224(opad, hash_value_sha3))
return -1;
return 0;
case CCP_AUTH_ALGO_SHA256_HMAC:
count = SHA256_DIGEST_SIZE >> 2;
if (partial_hash_sha256(ipad, (uint8_t *)hash_temp32))
return -1;
for (i = 0; i < count; i++, hash_value_be32++)
*hash_value_be32 = hash_temp32[count - 1 - i];
hash_value_be32 = (uint32_t *)((uint8_t *)sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha256(opad, (uint8_t *)hash_temp32))
return -1;
for (i = 0; i < count; i++, hash_value_be32++)
*hash_value_be32 = hash_temp32[count - 1 - i];
return 0;
case CCP_AUTH_ALGO_SHA3_256_HMAC:
hash_value_sha3 = sess->auth.pre_compute;
if (partial_hash_sha3_256(ipad, hash_value_sha3))
return -1;
hash_value_sha3 = (uint8_t *)(sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha3_256(opad, hash_value_sha3))
return -1;
return 0;
case CCP_AUTH_ALGO_SHA384_HMAC:
count = SHA512_DIGEST_SIZE >> 3;
if (partial_hash_sha384(ipad, (uint8_t *)hash_temp64))
return -1;
for (i = 0; i < count; i++, hash_value_be64++)
*hash_value_be64 = hash_temp64[count - 1 - i];
hash_value_be64 = (uint64_t *)((uint8_t *)sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha384(opad, (uint8_t *)hash_temp64))
return -1;
for (i = 0; i < count; i++, hash_value_be64++)
*hash_value_be64 = hash_temp64[count - 1 - i];
return 0;
case CCP_AUTH_ALGO_SHA3_384_HMAC:
hash_value_sha3 = sess->auth.pre_compute;
if (partial_hash_sha3_384(ipad, hash_value_sha3))
return -1;
hash_value_sha3 = (uint8_t *)(sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha3_384(opad, hash_value_sha3))
return -1;
return 0;
case CCP_AUTH_ALGO_SHA512_HMAC:
count = SHA512_DIGEST_SIZE >> 3;
if (partial_hash_sha512(ipad, (uint8_t *)hash_temp64))
return -1;
for (i = 0; i < count; i++, hash_value_be64++)
*hash_value_be64 = hash_temp64[count - 1 - i];
hash_value_be64 = (uint64_t *)((uint8_t *)sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha512(opad, (uint8_t *)hash_temp64))
return -1;
for (i = 0; i < count; i++, hash_value_be64++)
*hash_value_be64 = hash_temp64[count - 1 - i];
return 0;
case CCP_AUTH_ALGO_SHA3_512_HMAC:
hash_value_sha3 = sess->auth.pre_compute;
if (partial_hash_sha3_512(ipad, hash_value_sha3))
return -1;
hash_value_sha3 = (uint8_t *)(sess->auth.pre_compute
+ sess->auth.ctx_len);
if (partial_hash_sha3_512(opad, hash_value_sha3))
return -1;
return 0;
default:
CCP_LOG_ERR("Invalid auth algo");
return -1;
}
}
/* prepare temporary keys K1 and K2 */
static void prepare_key(unsigned char *k, unsigned char *l, int bl)
{
int i;
/* Shift block to left, including carry */
for (i = 0; i < bl; i++) {
k[i] = l[i] << 1;
if (i < bl - 1 && l[i + 1] & 0x80)
k[i] |= 1;
}
/* If MSB set fixup with R */
if (l[0] & 0x80)
k[bl - 1] ^= bl == 16 ? 0x87 : 0x1b;
}
/* subkeys K1 and K2 generation for CMAC */
static int
generate_cmac_subkeys(struct ccp_session *sess)
{
const EVP_CIPHER *algo;
EVP_CIPHER_CTX *ctx;
unsigned char *ccp_ctx;
size_t i;
int dstlen, totlen;
unsigned char zero_iv[AES_BLOCK_SIZE] = {0};
unsigned char dst[2 * AES_BLOCK_SIZE] = {0};
unsigned char k1[AES_BLOCK_SIZE] = {0};
unsigned char k2[AES_BLOCK_SIZE] = {0};
if (sess->auth.ut.aes_type == CCP_AES_TYPE_128)
algo = EVP_aes_128_cbc();
else if (sess->auth.ut.aes_type == CCP_AES_TYPE_192)
algo = EVP_aes_192_cbc();
else if (sess->auth.ut.aes_type == CCP_AES_TYPE_256)
algo = EVP_aes_256_cbc();
else {
CCP_LOG_ERR("Invalid CMAC type length");
return -1;
}
ctx = EVP_CIPHER_CTX_new();
if (!ctx) {
CCP_LOG_ERR("ctx creation failed");
return -1;
}
if (EVP_EncryptInit(ctx, algo, (unsigned char *)sess->auth.key,
(unsigned char *)zero_iv) <= 0)
goto key_generate_err;
if (EVP_CIPHER_CTX_set_padding(ctx, 0) <= 0)
goto key_generate_err;
if (EVP_EncryptUpdate(ctx, dst, &dstlen, zero_iv,
AES_BLOCK_SIZE) <= 0)
goto key_generate_err;
if (EVP_EncryptFinal_ex(ctx, dst + dstlen, &totlen) <= 0)
goto key_generate_err;
memset(sess->auth.pre_compute, 0, CCP_SB_BYTES * 2);
ccp_ctx = (unsigned char *)(sess->auth.pre_compute + CCP_SB_BYTES - 1);
prepare_key(k1, dst, AES_BLOCK_SIZE);
for (i = 0; i < AES_BLOCK_SIZE; i++, ccp_ctx--)
*ccp_ctx = k1[i];
ccp_ctx = (unsigned char *)(sess->auth.pre_compute +
(2 * CCP_SB_BYTES) - 1);
prepare_key(k2, k1, AES_BLOCK_SIZE);
for (i = 0; i < AES_BLOCK_SIZE; i++, ccp_ctx--)
*ccp_ctx = k2[i];
EVP_CIPHER_CTX_free(ctx);
return 0;
key_generate_err:
CCP_LOG_ERR("CMAC Init failed");
return -1;
}
/* configure session */
static int
ccp_configure_session_cipher(struct ccp_session *sess,
const struct rte_crypto_sym_xform *xform)
{
const struct rte_crypto_cipher_xform *cipher_xform = NULL;
size_t i, j, x;
cipher_xform = &xform->cipher;
/* set cipher direction */
if (cipher_xform->op == RTE_CRYPTO_CIPHER_OP_ENCRYPT)
sess->cipher.dir = CCP_CIPHER_DIR_ENCRYPT;
else
sess->cipher.dir = CCP_CIPHER_DIR_DECRYPT;
/* set cipher key */
sess->cipher.key_length = cipher_xform->key.length;
rte_memcpy(sess->cipher.key, cipher_xform->key.data,
cipher_xform->key.length);
/* set iv parameters */
sess->iv.offset = cipher_xform->iv.offset;
sess->iv.length = cipher_xform->iv.length;
switch (cipher_xform->algo) {
case RTE_CRYPTO_CIPHER_AES_CTR:
sess->cipher.algo = CCP_CIPHER_ALGO_AES_CTR;
sess->cipher.um.aes_mode = CCP_AES_MODE_CTR;
sess->cipher.engine = CCP_ENGINE_AES;
break;
case RTE_CRYPTO_CIPHER_AES_ECB:
sess->cipher.algo = CCP_CIPHER_ALGO_AES_CBC;
sess->cipher.um.aes_mode = CCP_AES_MODE_ECB;
sess->cipher.engine = CCP_ENGINE_AES;
break;
case RTE_CRYPTO_CIPHER_AES_CBC:
sess->cipher.algo = CCP_CIPHER_ALGO_AES_CBC;
sess->cipher.um.aes_mode = CCP_AES_MODE_CBC;
sess->cipher.engine = CCP_ENGINE_AES;
break;
case RTE_CRYPTO_CIPHER_3DES_CBC:
sess->cipher.algo = CCP_CIPHER_ALGO_3DES_CBC;
sess->cipher.um.des_mode = CCP_DES_MODE_CBC;
sess->cipher.engine = CCP_ENGINE_3DES;
break;
default:
CCP_LOG_ERR("Unsupported cipher algo");
return -1;
}
switch (sess->cipher.engine) {
case CCP_ENGINE_AES:
if (sess->cipher.key_length == 16)
sess->cipher.ut.aes_type = CCP_AES_TYPE_128;
else if (sess->cipher.key_length == 24)
sess->cipher.ut.aes_type = CCP_AES_TYPE_192;
else if (sess->cipher.key_length == 32)
sess->cipher.ut.aes_type = CCP_AES_TYPE_256;
else {
CCP_LOG_ERR("Invalid cipher key length");
return -1;
}
for (i = 0; i < sess->cipher.key_length ; i++)
sess->cipher.key_ccp[sess->cipher.key_length - i - 1] =
sess->cipher.key[i];
break;
case CCP_ENGINE_3DES:
if (sess->cipher.key_length == 16)
sess->cipher.ut.des_type = CCP_DES_TYPE_128;
else if (sess->cipher.key_length == 24)
sess->cipher.ut.des_type = CCP_DES_TYPE_192;
else {
CCP_LOG_ERR("Invalid cipher key length");
return -1;
}
for (j = 0, x = 0; j < sess->cipher.key_length/8; j++, x += 8)
for (i = 0; i < 8; i++)
sess->cipher.key_ccp[(8 + x) - i - 1] =
sess->cipher.key[i + x];
break;
default:
CCP_LOG_ERR("Invalid CCP Engine");
return -ENOTSUP;
}
sess->cipher.nonce_phys = rte_mem_virt2phy(sess->cipher.nonce);
sess->cipher.key_phys = rte_mem_virt2phy(sess->cipher.key_ccp);
return 0;
}
static int
ccp_configure_session_auth(struct ccp_session *sess,
const struct rte_crypto_sym_xform *xform)
{
const struct rte_crypto_auth_xform *auth_xform = NULL;
size_t i;
auth_xform = &xform->auth;
sess->auth.digest_length = auth_xform->digest_length;
if (auth_xform->op == RTE_CRYPTO_AUTH_OP_GENERATE)
sess->auth.op = CCP_AUTH_OP_GENERATE;
else
sess->auth.op = CCP_AUTH_OP_VERIFY;
switch (auth_xform->algo) {
case RTE_CRYPTO_AUTH_MD5_HMAC:
if (sess->auth_opt) {
sess->auth.algo = CCP_AUTH_ALGO_MD5_HMAC;
sess->auth.offset = ((CCP_SB_BYTES << 1) -
MD5_DIGEST_SIZE);
sess->auth.key_length = auth_xform->key.length;
sess->auth.block_size = MD5_BLOCK_SIZE;
memset(sess->auth.key, 0, sess->auth.block_size);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
} else
return -1; /* HMAC MD5 not supported on CCP */
break;
case RTE_CRYPTO_AUTH_SHA1:
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.algo = CCP_AUTH_ALGO_SHA1;
sess->auth.ut.sha_type = CCP_SHA_TYPE_1;
sess->auth.ctx = (void *)ccp_sha1_init;
sess->auth.ctx_len = CCP_SB_BYTES;
sess->auth.offset = CCP_SB_BYTES - SHA1_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA1_HMAC:
if (sess->auth_opt) {
if (auth_xform->key.length > SHA1_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA1_HMAC;
sess->auth.offset = CCP_SB_BYTES - SHA1_DIGEST_SIZE;
sess->auth.block_size = SHA1_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
} else {
if (auth_xform->key.length > SHA1_BLOCK_SIZE)
return -1;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.algo = CCP_AUTH_ALGO_SHA1_HMAC;
sess->auth.ut.sha_type = CCP_SHA_TYPE_1;
sess->auth.ctx_len = CCP_SB_BYTES;
sess->auth.offset = CCP_SB_BYTES - SHA1_DIGEST_SIZE;
sess->auth.block_size = SHA1_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0,
sess->auth.ctx_len << 1);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
}
break;
case RTE_CRYPTO_AUTH_SHA224:
sess->auth.algo = CCP_AUTH_ALGO_SHA224;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_224;
sess->auth.ctx = (void *)ccp_sha224_init;
sess->auth.ctx_len = CCP_SB_BYTES;
sess->auth.offset = CCP_SB_BYTES - SHA224_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA224_HMAC:
if (sess->auth_opt) {
if (auth_xform->key.length > SHA224_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA224_HMAC;
sess->auth.offset = CCP_SB_BYTES - SHA224_DIGEST_SIZE;
sess->auth.block_size = SHA224_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
} else {
if (auth_xform->key.length > SHA224_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA224_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_224;
sess->auth.ctx_len = CCP_SB_BYTES;
sess->auth.offset = CCP_SB_BYTES - SHA224_DIGEST_SIZE;
sess->auth.block_size = SHA224_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0,
sess->auth.ctx_len << 1);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
}
break;
case RTE_CRYPTO_AUTH_SHA3_224:
sess->auth.algo = CCP_AUTH_ALGO_SHA3_224;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_224;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA224_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA3_224_HMAC:
if (auth_xform->key.length > SHA3_224_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA3_224_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_224;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA224_DIGEST_SIZE;
sess->auth.block_size = SHA3_224_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0, 2 * sess->auth.ctx_len);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
break;
case RTE_CRYPTO_AUTH_SHA256:
sess->auth.algo = CCP_AUTH_ALGO_SHA256;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_256;
sess->auth.ctx = (void *)ccp_sha256_init;
sess->auth.ctx_len = CCP_SB_BYTES;
sess->auth.offset = CCP_SB_BYTES - SHA256_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA256_HMAC:
if (sess->auth_opt) {
if (auth_xform->key.length > SHA256_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA256_HMAC;
sess->auth.offset = CCP_SB_BYTES - SHA256_DIGEST_SIZE;
sess->auth.block_size = SHA256_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
} else {
if (auth_xform->key.length > SHA256_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA256_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_256;
sess->auth.ctx_len = CCP_SB_BYTES;
sess->auth.offset = CCP_SB_BYTES - SHA256_DIGEST_SIZE;
sess->auth.block_size = SHA256_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0,
sess->auth.ctx_len << 1);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
}
break;
case RTE_CRYPTO_AUTH_SHA3_256:
sess->auth.algo = CCP_AUTH_ALGO_SHA3_256;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_256;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA256_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA3_256_HMAC:
if (auth_xform->key.length > SHA3_256_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA3_256_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_256;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA256_DIGEST_SIZE;
sess->auth.block_size = SHA3_256_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0, 2 * sess->auth.ctx_len);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
break;
case RTE_CRYPTO_AUTH_SHA384:
sess->auth.algo = CCP_AUTH_ALGO_SHA384;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_384;
sess->auth.ctx = (void *)ccp_sha384_init;
sess->auth.ctx_len = CCP_SB_BYTES << 1;
sess->auth.offset = (CCP_SB_BYTES << 1) - SHA384_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA384_HMAC:
if (sess->auth_opt) {
if (auth_xform->key.length > SHA384_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA384_HMAC;
sess->auth.offset = ((CCP_SB_BYTES << 1) -
SHA384_DIGEST_SIZE);
sess->auth.block_size = SHA384_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
} else {
if (auth_xform->key.length > SHA384_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA384_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_384;
sess->auth.ctx_len = CCP_SB_BYTES << 1;
sess->auth.offset = ((CCP_SB_BYTES << 1) -
SHA384_DIGEST_SIZE);
sess->auth.block_size = SHA384_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0,
sess->auth.ctx_len << 1);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
}
break;
case RTE_CRYPTO_AUTH_SHA3_384:
sess->auth.algo = CCP_AUTH_ALGO_SHA3_384;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_384;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA384_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA3_384_HMAC:
if (auth_xform->key.length > SHA3_384_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA3_384_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_384;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA384_DIGEST_SIZE;
sess->auth.block_size = SHA3_384_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0, 2 * sess->auth.ctx_len);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
break;
case RTE_CRYPTO_AUTH_SHA512:
sess->auth.algo = CCP_AUTH_ALGO_SHA512;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_512;
sess->auth.ctx = (void *)ccp_sha512_init;
sess->auth.ctx_len = CCP_SB_BYTES << 1;
sess->auth.offset = (CCP_SB_BYTES << 1) - SHA512_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA512_HMAC:
if (sess->auth_opt) {
if (auth_xform->key.length > SHA512_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA512_HMAC;
sess->auth.offset = ((CCP_SB_BYTES << 1) -
SHA512_DIGEST_SIZE);
sess->auth.block_size = SHA512_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
} else {
if (auth_xform->key.length > SHA512_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA512_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA_TYPE_512;
sess->auth.ctx_len = CCP_SB_BYTES << 1;
sess->auth.offset = ((CCP_SB_BYTES << 1) -
SHA512_DIGEST_SIZE);
sess->auth.block_size = SHA512_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0,
sess->auth.ctx_len << 1);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
}
break;
case RTE_CRYPTO_AUTH_SHA3_512:
sess->auth.algo = CCP_AUTH_ALGO_SHA3_512;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_512;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA512_DIGEST_SIZE;
break;
case RTE_CRYPTO_AUTH_SHA3_512_HMAC:
if (auth_xform->key.length > SHA3_512_BLOCK_SIZE)
return -1;
sess->auth.algo = CCP_AUTH_ALGO_SHA3_512_HMAC;
sess->auth.engine = CCP_ENGINE_SHA;
sess->auth.ut.sha_type = CCP_SHA3_TYPE_512;
sess->auth.ctx_len = CCP_SHA3_CTX_SIZE;
sess->auth.offset = CCP_SHA3_CTX_SIZE - SHA512_DIGEST_SIZE;
sess->auth.block_size = SHA3_512_BLOCK_SIZE;
sess->auth.key_length = auth_xform->key.length;
memset(sess->auth.key, 0, sess->auth.block_size);
memset(sess->auth.pre_compute, 0, 2 * sess->auth.ctx_len);
rte_memcpy(sess->auth.key, auth_xform->key.data,
auth_xform->key.length);
if (generate_partial_hash(sess))
return -1;
break;
case RTE_CRYPTO_AUTH_AES_CMAC:
sess->auth.algo = CCP_AUTH_ALGO_AES_CMAC;
sess->auth.engine = CCP_ENGINE_AES;
sess->auth.um.aes_mode = CCP_AES_MODE_CMAC;
sess->auth.key_length = auth_xform->key.length;
/* padding and hash result */
sess->auth.ctx_len = CCP_SB_BYTES << 1;
sess->auth.offset = AES_BLOCK_SIZE;
sess->auth.block_size = AES_BLOCK_SIZE;
if (sess->auth.key_length == 16)
sess->auth.ut.aes_type = CCP_AES_TYPE_128;
else if (sess->auth.key_length == 24)
sess->auth.ut.aes_type = CCP_AES_TYPE_192;
else if (sess->auth.key_length == 32)
sess->auth.ut.aes_type = CCP_AES_TYPE_256;
else {
CCP_LOG_ERR("Invalid CMAC key length");
return -1;
}
rte_memcpy(sess->auth.key, auth_xform->key.data,
sess->auth.key_length);
for (i = 0; i < sess->auth.key_length; i++)
sess->auth.key_ccp[sess->auth.key_length - i - 1] =
sess->auth.key[i];
if (generate_cmac_subkeys(sess))
return -1;
break;
default:
CCP_LOG_ERR("Unsupported hash algo");
return -ENOTSUP;
}
return 0;
}
static int
ccp_configure_session_aead(struct ccp_session *sess,
const struct rte_crypto_sym_xform *xform)
{
const struct rte_crypto_aead_xform *aead_xform = NULL;
size_t i;
aead_xform = &xform->aead;
sess->cipher.key_length = aead_xform->key.length;
rte_memcpy(sess->cipher.key, aead_xform->key.data,
aead_xform->key.length);
if (aead_xform->op == RTE_CRYPTO_AEAD_OP_ENCRYPT) {
sess->cipher.dir = CCP_CIPHER_DIR_ENCRYPT;
sess->auth.op = CCP_AUTH_OP_GENERATE;
} else {
sess->cipher.dir = CCP_CIPHER_DIR_DECRYPT;
sess->auth.op = CCP_AUTH_OP_VERIFY;
}
sess->aead_algo = aead_xform->algo;
sess->auth.aad_length = aead_xform->aad_length;
sess->auth.digest_length = aead_xform->digest_length;
/* set iv parameters */
sess->iv.offset = aead_xform->iv.offset;
sess->iv.length = aead_xform->iv.length;
switch (aead_xform->algo) {
case RTE_CRYPTO_AEAD_AES_GCM:
sess->cipher.algo = CCP_CIPHER_ALGO_AES_GCM;
sess->cipher.um.aes_mode = CCP_AES_MODE_GCTR;
sess->cipher.engine = CCP_ENGINE_AES;
if (sess->cipher.key_length == 16)
sess->cipher.ut.aes_type = CCP_AES_TYPE_128;
else if (sess->cipher.key_length == 24)
sess->cipher.ut.aes_type = CCP_AES_TYPE_192;
else if (sess->cipher.key_length == 32)
sess->cipher.ut.aes_type = CCP_AES_TYPE_256;
else {
CCP_LOG_ERR("Invalid aead key length");
return -1;
}
for (i = 0; i < sess->cipher.key_length; i++)
sess->cipher.key_ccp[sess->cipher.key_length - i - 1] =
sess->cipher.key[i];
sess->auth.algo = CCP_AUTH_ALGO_AES_GCM;
sess->auth.engine = CCP_ENGINE_AES;
sess->auth.um.aes_mode = CCP_AES_MODE_GHASH;
sess->auth.ctx_len = CCP_SB_BYTES;
sess->auth.offset = 0;
sess->auth.block_size = AES_BLOCK_SIZE;
sess->cmd_id = CCP_CMD_COMBINED;
break;
default:
CCP_LOG_ERR("Unsupported aead algo");
return -ENOTSUP;
}
sess->cipher.nonce_phys = rte_mem_virt2phy(sess->cipher.nonce);
sess->cipher.key_phys = rte_mem_virt2phy(sess->cipher.key_ccp);
return 0;
}
int
ccp_set_session_parameters(struct ccp_session *sess,
const struct rte_crypto_sym_xform *xform,
struct ccp_private *internals)
{
const struct rte_crypto_sym_xform *cipher_xform = NULL;
const struct rte_crypto_sym_xform *auth_xform = NULL;
const struct rte_crypto_sym_xform *aead_xform = NULL;
int ret = 0;
sess->auth_opt = internals->auth_opt;
sess->cmd_id = ccp_get_cmd_id(xform);
switch (sess->cmd_id) {
case CCP_CMD_CIPHER:
cipher_xform = xform;
break;
case CCP_CMD_AUTH:
auth_xform = xform;
break;
case CCP_CMD_CIPHER_HASH:
cipher_xform = xform;
auth_xform = xform->next;
break;
case CCP_CMD_HASH_CIPHER:
auth_xform = xform;
cipher_xform = xform->next;
break;
case CCP_CMD_COMBINED:
aead_xform = xform;
break;
default:
CCP_LOG_ERR("Unsupported cmd_id");
return -1;
}
/* Default IV length = 0 */
sess->iv.length = 0;
if (cipher_xform) {
ret = ccp_configure_session_cipher(sess, cipher_xform);
if (ret != 0) {
CCP_LOG_ERR("Invalid/unsupported cipher parameters");
return ret;
}
}
if (auth_xform) {
ret = ccp_configure_session_auth(sess, auth_xform);
if (ret != 0) {
CCP_LOG_ERR("Invalid/unsupported auth parameters");
return ret;
}
}
if (aead_xform) {
ret = ccp_configure_session_aead(sess, aead_xform);
if (ret != 0) {
CCP_LOG_ERR("Invalid/unsupported aead parameters");
return ret;
}
}
return ret;
}
/* calculate CCP descriptors requirement */
static inline int
ccp_cipher_slot(struct ccp_session *session)
{
int count = 0;
switch (session->cipher.algo) {
case CCP_CIPHER_ALGO_AES_CBC:
count = 2;
/**< op + passthrough for iv */
break;
case CCP_CIPHER_ALGO_AES_ECB:
count = 1;
/**<only op*/
break;
case CCP_CIPHER_ALGO_AES_CTR:
count = 2;
/**< op + passthrough for iv */
break;
case CCP_CIPHER_ALGO_3DES_CBC:
count = 2;
/**< op + passthrough for iv */
break;
default:
CCP_LOG_ERR("Unsupported cipher algo %d",
session->cipher.algo);
}
return count;
}
static inline int
ccp_auth_slot(struct ccp_session *session)
{
int count = 0;
switch (session->auth.algo) {
case CCP_AUTH_ALGO_SHA1:
case CCP_AUTH_ALGO_SHA224:
case CCP_AUTH_ALGO_SHA256:
case CCP_AUTH_ALGO_SHA384:
case CCP_AUTH_ALGO_SHA512:
count = 3;
/**< op + lsb passthrough cpy to/from*/
break;
case CCP_AUTH_ALGO_MD5_HMAC:
break;
case CCP_AUTH_ALGO_SHA1_HMAC:
case CCP_AUTH_ALGO_SHA224_HMAC:
case CCP_AUTH_ALGO_SHA256_HMAC:
if (session->auth_opt == 0)
count = 6;
break;
case CCP_AUTH_ALGO_SHA384_HMAC:
case CCP_AUTH_ALGO_SHA512_HMAC:
/**
* 1. Load PHash1 = H(k ^ ipad); to LSB
* 2. generate IHash = H(hash on meassage with PHash1
* as init values);
* 3. Retrieve IHash 2 slots for 384/512
* 4. Load Phash2 = H(k ^ opad); to LSB
* 5. generate FHash = H(hash on Ihash with Phash2
* as init value);
* 6. Retrieve HMAC output from LSB to host memory
*/
if (session->auth_opt == 0)
count = 7;
break;
case CCP_AUTH_ALGO_SHA3_224:
case CCP_AUTH_ALGO_SHA3_256:
case CCP_AUTH_ALGO_SHA3_384:
case CCP_AUTH_ALGO_SHA3_512:
count = 1;
/**< only op ctx and dst in host memory*/
break;
case CCP_AUTH_ALGO_SHA3_224_HMAC:
case CCP_AUTH_ALGO_SHA3_256_HMAC:
count = 3;
break;
case CCP_AUTH_ALGO_SHA3_384_HMAC:
case CCP_AUTH_ALGO_SHA3_512_HMAC:
count = 4;
/**
* 1. Op to Perform Ihash
* 2. Retrieve result from LSB to host memory
* 3. Perform final hash
*/
break;
case CCP_AUTH_ALGO_AES_CMAC:
count = 4;
/**
* op
* extra descriptor in padding case
* (k1/k2(255:128) with iv(127:0))
* Retrieve result
*/
break;
default:
CCP_LOG_ERR("Unsupported auth algo %d",
session->auth.algo);
}
return count;
}
static int
ccp_aead_slot(struct ccp_session *session)
{
int count = 0;
switch (session->aead_algo) {
case RTE_CRYPTO_AEAD_AES_GCM:
break;
default:
CCP_LOG_ERR("Unsupported aead algo %d",
session->aead_algo);
}
switch (session->auth.algo) {
case CCP_AUTH_ALGO_AES_GCM:
count = 5;
/**
* 1. Passthru iv
* 2. Hash AAD
* 3. GCTR
* 4. Reload passthru
* 5. Hash Final tag
*/
break;
default:
CCP_LOG_ERR("Unsupported combined auth ALGO %d",
session->auth.algo);
}
return count;
}
int
ccp_compute_slot_count(struct ccp_session *session)
{
int count = 0;
switch (session->cmd_id) {
case CCP_CMD_CIPHER:
count = ccp_cipher_slot(session);
break;
case CCP_CMD_AUTH:
count = ccp_auth_slot(session);
break;
case CCP_CMD_CIPHER_HASH:
case CCP_CMD_HASH_CIPHER:
count = ccp_cipher_slot(session);
count += ccp_auth_slot(session);
break;
case CCP_CMD_COMBINED:
count = ccp_aead_slot(session);
break;
default:
CCP_LOG_ERR("Unsupported cmd_id");
}
return count;
}
static uint8_t
algo_select(int sessalgo,
const EVP_MD **algo)
{
int res = 0;
switch (sessalgo) {
case CCP_AUTH_ALGO_MD5_HMAC:
*algo = EVP_md5();
break;
case CCP_AUTH_ALGO_SHA1_HMAC:
*algo = EVP_sha1();
break;
case CCP_AUTH_ALGO_SHA224_HMAC:
*algo = EVP_sha224();
break;
case CCP_AUTH_ALGO_SHA256_HMAC:
*algo = EVP_sha256();
break;
case CCP_AUTH_ALGO_SHA384_HMAC:
*algo = EVP_sha384();
break;
case CCP_AUTH_ALGO_SHA512_HMAC:
*algo = EVP_sha512();
break;
default:
res = -EINVAL;
break;
}
return res;
}
static int
process_cpu_auth_hmac(uint8_t *src, uint8_t *dst,
__rte_unused uint8_t *iv,
EVP_PKEY *pkey,
int srclen,
EVP_MD_CTX *ctx,
const EVP_MD *algo,
uint16_t d_len)
{
size_t dstlen;
unsigned char temp_dst[64];
if (EVP_DigestSignInit(ctx, NULL, algo, NULL, pkey) <= 0)
goto process_auth_err;
if (EVP_DigestSignUpdate(ctx, (char *)src, srclen) <= 0)
goto process_auth_err;
if (EVP_DigestSignFinal(ctx, temp_dst, &dstlen) <= 0)
goto process_auth_err;
memcpy(dst, temp_dst, d_len);
return 0;
process_auth_err:
CCP_LOG_ERR("Process cpu auth failed");
return -EINVAL;
}
static int cpu_crypto_auth(struct ccp_qp *qp,
struct rte_crypto_op *op,
struct ccp_session *sess,
EVP_MD_CTX *ctx)
{
uint8_t *src, *dst;
int srclen, status;
struct rte_mbuf *mbuf_src, *mbuf_dst;
const EVP_MD *algo = NULL;
EVP_PKEY *pkey;
algo_select(sess->auth.algo, &algo);
pkey = EVP_PKEY_new_mac_key(EVP_PKEY_HMAC, NULL, sess->auth.key,
sess->auth.key_length);
mbuf_src = op->sym->m_src;
mbuf_dst = op->sym->m_dst ? op->sym->m_dst : op->sym->m_src;
srclen = op->sym->auth.data.length;
src = rte_pktmbuf_mtod_offset(mbuf_src, uint8_t *,
op->sym->auth.data.offset);
if (sess->auth.op == CCP_AUTH_OP_VERIFY) {
dst = qp->temp_digest;
} else {
dst = op->sym->auth.digest.data;
if (dst == NULL) {
dst = rte_pktmbuf_mtod_offset(mbuf_dst, uint8_t *,
op->sym->auth.data.offset +
sess->auth.digest_length);
}
}
status = process_cpu_auth_hmac(src, dst, NULL,
pkey, srclen,
ctx,
algo,
sess->auth.digest_length);
if (status) {
op->status = RTE_CRYPTO_OP_STATUS_ERROR;
return status;
}
if (sess->auth.op == CCP_AUTH_OP_VERIFY) {
if (memcmp(dst, op->sym->auth.digest.data,
sess->auth.digest_length) != 0) {
op->status = RTE_CRYPTO_OP_STATUS_AUTH_FAILED;
} else {
op->status = RTE_CRYPTO_OP_STATUS_SUCCESS;
}
} else {
op->status = RTE_CRYPTO_OP_STATUS_SUCCESS;
}
EVP_PKEY_free(pkey);
return 0;
}
static void
ccp_perform_passthru(struct ccp_passthru *pst,
struct ccp_queue *cmd_q)
{
struct ccp_desc *desc;
union ccp_function function;
desc = &cmd_q->qbase_desc[cmd_q->qidx];
CCP_CMD_ENGINE(desc) = CCP_ENGINE_PASSTHRU;
CCP_CMD_SOC(desc) = 0;
CCP_CMD_IOC(desc) = 0;
CCP_CMD_INIT(desc) = 0;
CCP_CMD_EOM(desc) = 0;
CCP_CMD_PROT(desc) = 0;
function.raw = 0;
CCP_PT_BYTESWAP(&function) = pst->byte_swap;
CCP_PT_BITWISE(&function) = pst->bit_mod;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = pst->len;
if (pst->dir) {
CCP_CMD_SRC_LO(desc) = (uint32_t)(pst->src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(pst->src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = (uint32_t)(pst->dest_addr);
CCP_CMD_DST_HI(desc) = 0;
CCP_CMD_DST_MEM(desc) = CCP_MEMTYPE_SB;
if (pst->bit_mod != CCP_PASSTHRU_BITWISE_NOOP)
CCP_CMD_LSB_ID(desc) = cmd_q->sb_key;
} else {
CCP_CMD_SRC_LO(desc) = (uint32_t)(pst->src_addr);
CCP_CMD_SRC_HI(desc) = 0;
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SB;
CCP_CMD_DST_LO(desc) = (uint32_t)(pst->dest_addr);
CCP_CMD_DST_HI(desc) = high32_value(pst->dest_addr);
CCP_CMD_DST_MEM(desc) = CCP_MEMTYPE_SYSTEM;
}
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
}
static int
ccp_perform_hmac(struct rte_crypto_op *op,
struct ccp_queue *cmd_q)
{
struct ccp_session *session;
union ccp_function function;
struct ccp_desc *desc;
uint32_t tail;
phys_addr_t src_addr, dest_addr, dest_addr_t;
struct ccp_passthru pst;
uint64_t auth_msg_bits;
void *append_ptr;
uint8_t *addr;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
addr = session->auth.pre_compute;
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->auth.data.offset);
append_ptr = (void *)rte_pktmbuf_append(op->sym->m_src,
session->auth.ctx_len);
dest_addr = (phys_addr_t)rte_mem_virt2phy(append_ptr);
dest_addr_t = dest_addr;
/** Load PHash1 to LSB*/
pst.src_addr = (phys_addr_t)rte_mem_virt2phy((void *)addr);
pst.dest_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.len = session->auth.ctx_len;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
ccp_perform_passthru(&pst, cmd_q);
/**sha engine command descriptor for IntermediateHash*/
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_SHA;
CCP_CMD_SOC(desc) = 0;
CCP_CMD_IOC(desc) = 0;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
CCP_CMD_PROT(desc) = 0;
function.raw = 0;
CCP_SHA_TYPE(&function) = session->auth.ut.sha_type;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = op->sym->auth.data.length;
auth_msg_bits = (op->sym->auth.data.length +
session->auth.block_size) * 8;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_sha;
CCP_CMD_SHA_LO(desc) = ((uint32_t)auth_msg_bits);
CCP_CMD_SHA_HI(desc) = high32_value(auth_msg_bits);
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
/* Intermediate Hash value retrieve */
if ((session->auth.ut.sha_type == CCP_SHA_TYPE_384) ||
(session->auth.ut.sha_type == CCP_SHA_TYPE_512)) {
pst.src_addr =
(phys_addr_t)((cmd_q->sb_sha + 1) * CCP_SB_BYTES);
pst.dest_addr = dest_addr_t;
pst.len = CCP_SB_BYTES;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
pst.src_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.dest_addr = dest_addr_t + CCP_SB_BYTES;
pst.len = CCP_SB_BYTES;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
} else {
pst.src_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.dest_addr = dest_addr_t;
pst.len = session->auth.ctx_len;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
}
/** Load PHash2 to LSB*/
addr += session->auth.ctx_len;
pst.src_addr = (phys_addr_t)rte_mem_virt2phy((void *)addr);
pst.dest_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.len = session->auth.ctx_len;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
ccp_perform_passthru(&pst, cmd_q);
/**sha engine command descriptor for FinalHash*/
dest_addr_t += session->auth.offset;
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_SHA;
CCP_CMD_SOC(desc) = 0;
CCP_CMD_IOC(desc) = 0;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
CCP_CMD_PROT(desc) = 0;
function.raw = 0;
CCP_SHA_TYPE(&function) = session->auth.ut.sha_type;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = (session->auth.ctx_len -
session->auth.offset);
auth_msg_bits = (session->auth.block_size +
session->auth.ctx_len -
session->auth.offset) * 8;
CCP_CMD_SRC_LO(desc) = (uint32_t)(dest_addr_t);
CCP_CMD_SRC_HI(desc) = high32_value(dest_addr_t);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_sha;
CCP_CMD_SHA_LO(desc) = ((uint32_t)auth_msg_bits);
CCP_CMD_SHA_HI(desc) = high32_value(auth_msg_bits);
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
/* Retrieve hmac output */
pst.src_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.dest_addr = dest_addr;
pst.len = session->auth.ctx_len;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
if ((session->auth.ut.sha_type == CCP_SHA_TYPE_384) ||
(session->auth.ut.sha_type == CCP_SHA_TYPE_512))
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
else
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static int
ccp_perform_sha(struct rte_crypto_op *op,
struct ccp_queue *cmd_q)
{
struct ccp_session *session;
union ccp_function function;
struct ccp_desc *desc;
uint32_t tail;
phys_addr_t src_addr, dest_addr;
struct ccp_passthru pst;
void *append_ptr;
uint64_t auth_msg_bits;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->auth.data.offset);
append_ptr = (void *)rte_pktmbuf_append(op->sym->m_src,
session->auth.ctx_len);
dest_addr = (phys_addr_t)rte_mem_virt2phy(append_ptr);
/** Passthru sha context*/
pst.src_addr = (phys_addr_t)rte_mem_virt2phy((void *)
session->auth.ctx);
pst.dest_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.len = session->auth.ctx_len;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
ccp_perform_passthru(&pst, cmd_q);
/**prepare sha command descriptor*/
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_SHA;
CCP_CMD_SOC(desc) = 0;
CCP_CMD_IOC(desc) = 0;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
CCP_CMD_PROT(desc) = 0;
function.raw = 0;
CCP_SHA_TYPE(&function) = session->auth.ut.sha_type;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = op->sym->auth.data.length;
auth_msg_bits = op->sym->auth.data.length * 8;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_sha;
CCP_CMD_SHA_LO(desc) = ((uint32_t)auth_msg_bits);
CCP_CMD_SHA_HI(desc) = high32_value(auth_msg_bits);
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
/* Hash value retrieve */
pst.src_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.dest_addr = dest_addr;
pst.len = session->auth.ctx_len;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
if ((session->auth.ut.sha_type == CCP_SHA_TYPE_384) ||
(session->auth.ut.sha_type == CCP_SHA_TYPE_512))
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
else
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static int
ccp_perform_sha3_hmac(struct rte_crypto_op *op,
struct ccp_queue *cmd_q)
{
struct ccp_session *session;
struct ccp_passthru pst;
union ccp_function function;
struct ccp_desc *desc;
uint8_t *append_ptr;
uint32_t tail;
phys_addr_t src_addr, dest_addr, ctx_paddr, dest_addr_t;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->auth.data.offset);
append_ptr = (uint8_t *)rte_pktmbuf_append(op->sym->m_src,
session->auth.ctx_len);
if (!append_ptr) {
CCP_LOG_ERR("CCP MBUF append failed\n");
return -1;
}
dest_addr = (phys_addr_t)rte_mem_virt2phy((void *)append_ptr);
dest_addr_t = dest_addr + (session->auth.ctx_len / 2);
ctx_paddr = (phys_addr_t)rte_mem_virt2phy((void
*)session->auth.pre_compute);
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
/*desc1 for SHA3-Ihash operation */
CCP_CMD_ENGINE(desc) = CCP_ENGINE_SHA;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
function.raw = 0;
CCP_SHA_TYPE(&function) = session->auth.ut.sha_type;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = op->sym->auth.data.length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = (cmd_q->sb_sha * CCP_SB_BYTES);
CCP_CMD_DST_HI(desc) = 0;
CCP_CMD_DST_MEM(desc) = CCP_MEMTYPE_SB;
CCP_CMD_KEY_LO(desc) = ((uint32_t)ctx_paddr);
CCP_CMD_KEY_HI(desc) = high32_value(ctx_paddr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
/* Intermediate Hash value retrieve */
if ((session->auth.ut.sha_type == CCP_SHA3_TYPE_384) ||
(session->auth.ut.sha_type == CCP_SHA3_TYPE_512)) {
pst.src_addr =
(phys_addr_t)((cmd_q->sb_sha + 1) * CCP_SB_BYTES);
pst.dest_addr = dest_addr_t;
pst.len = CCP_SB_BYTES;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
pst.src_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.dest_addr = dest_addr_t + CCP_SB_BYTES;
pst.len = CCP_SB_BYTES;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
} else {
pst.src_addr = (phys_addr_t)(cmd_q->sb_sha * CCP_SB_BYTES);
pst.dest_addr = dest_addr_t;
pst.len = CCP_SB_BYTES;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
}
/**sha engine command descriptor for FinalHash*/
ctx_paddr += CCP_SHA3_CTX_SIZE;
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_SHA;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
function.raw = 0;
CCP_SHA_TYPE(&function) = session->auth.ut.sha_type;
CCP_CMD_FUNCTION(desc) = function.raw;
if (session->auth.ut.sha_type == CCP_SHA3_TYPE_224) {
dest_addr_t += (CCP_SB_BYTES - SHA224_DIGEST_SIZE);
CCP_CMD_LEN(desc) = SHA224_DIGEST_SIZE;
} else if (session->auth.ut.sha_type == CCP_SHA3_TYPE_256) {
CCP_CMD_LEN(desc) = SHA256_DIGEST_SIZE;
} else if (session->auth.ut.sha_type == CCP_SHA3_TYPE_384) {
dest_addr_t += (2 * CCP_SB_BYTES - SHA384_DIGEST_SIZE);
CCP_CMD_LEN(desc) = SHA384_DIGEST_SIZE;
} else {
CCP_CMD_LEN(desc) = SHA512_DIGEST_SIZE;
}
CCP_CMD_SRC_LO(desc) = ((uint32_t)dest_addr_t);
CCP_CMD_SRC_HI(desc) = high32_value(dest_addr_t);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = (uint32_t)dest_addr;
CCP_CMD_DST_HI(desc) = high32_value(dest_addr);
CCP_CMD_DST_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)ctx_paddr);
CCP_CMD_KEY_HI(desc) = high32_value(ctx_paddr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static int
ccp_perform_sha3(struct rte_crypto_op *op,
struct ccp_queue *cmd_q)
{
struct ccp_session *session;
union ccp_function function;
struct ccp_desc *desc;
uint8_t *ctx_addr, *append_ptr;
uint32_t tail;
phys_addr_t src_addr, dest_addr, ctx_paddr;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->auth.data.offset);
append_ptr = (uint8_t *)rte_pktmbuf_append(op->sym->m_src,
session->auth.ctx_len);
if (!append_ptr) {
CCP_LOG_ERR("CCP MBUF append failed\n");
return -1;
}
dest_addr = (phys_addr_t)rte_mem_virt2phy((void *)append_ptr);
ctx_addr = session->auth.sha3_ctx;
ctx_paddr = (phys_addr_t)rte_mem_virt2phy((void *)ctx_addr);
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
/* prepare desc for SHA3 operation */
CCP_CMD_ENGINE(desc) = CCP_ENGINE_SHA;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
function.raw = 0;
CCP_SHA_TYPE(&function) = session->auth.ut.sha_type;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = op->sym->auth.data.length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = ((uint32_t)dest_addr);
CCP_CMD_DST_HI(desc) = high32_value(dest_addr);
CCP_CMD_DST_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)ctx_paddr);
CCP_CMD_KEY_HI(desc) = high32_value(ctx_paddr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static int
ccp_perform_aes_cmac(struct rte_crypto_op *op,
struct ccp_queue *cmd_q)
{
struct ccp_session *session;
union ccp_function function;
struct ccp_passthru pst;
struct ccp_desc *desc;
uint32_t tail;
uint8_t *src_tb, *append_ptr, *ctx_addr;
phys_addr_t src_addr, dest_addr, key_addr;
int length, non_align_len;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
key_addr = rte_mem_virt2phy(session->auth.key_ccp);
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->auth.data.offset);
append_ptr = (uint8_t *)rte_pktmbuf_append(op->sym->m_src,
session->auth.ctx_len);
dest_addr = (phys_addr_t)rte_mem_virt2phy((void *)append_ptr);
function.raw = 0;
CCP_AES_ENCRYPT(&function) = CCP_CIPHER_DIR_ENCRYPT;
CCP_AES_MODE(&function) = session->auth.um.aes_mode;
CCP_AES_TYPE(&function) = session->auth.ut.aes_type;
if (op->sym->auth.data.length % session->auth.block_size == 0) {
ctx_addr = session->auth.pre_compute;
memset(ctx_addr, 0, AES_BLOCK_SIZE);
pst.src_addr = (phys_addr_t)rte_mem_virt2phy((void *)ctx_addr);
pst.dest_addr = (phys_addr_t)(cmd_q->sb_iv * CCP_SB_BYTES);
pst.len = CCP_SB_BYTES;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
ccp_perform_passthru(&pst, cmd_q);
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
/* prepare desc for aes-cmac command */
CCP_CMD_ENGINE(desc) = CCP_ENGINE_AES;
CCP_CMD_EOM(desc) = 1;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = op->sym->auth.data.length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail =
(uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
} else {
ctx_addr = session->auth.pre_compute + CCP_SB_BYTES;
memset(ctx_addr, 0, AES_BLOCK_SIZE);
pst.src_addr = (phys_addr_t)rte_mem_virt2phy((void *)ctx_addr);
pst.dest_addr = (phys_addr_t)(cmd_q->sb_iv * CCP_SB_BYTES);
pst.len = CCP_SB_BYTES;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
ccp_perform_passthru(&pst, cmd_q);
length = (op->sym->auth.data.length / AES_BLOCK_SIZE);
length *= AES_BLOCK_SIZE;
non_align_len = op->sym->auth.data.length - length;
/* prepare desc for aes-cmac command */
/*Command 1*/
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_AES;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
/*Command 2*/
append_ptr = append_ptr + CCP_SB_BYTES;
memset(append_ptr, 0, AES_BLOCK_SIZE);
src_tb = rte_pktmbuf_mtod_offset(op->sym->m_src,
uint8_t *,
op->sym->auth.data.offset +
length);
rte_memcpy(append_ptr, src_tb, non_align_len);
append_ptr[non_align_len] = CMAC_PAD_VALUE;
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_AES;
CCP_CMD_EOM(desc) = 1;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = AES_BLOCK_SIZE;
CCP_CMD_SRC_LO(desc) = ((uint32_t)(dest_addr + CCP_SB_BYTES));
CCP_CMD_SRC_HI(desc) = high32_value(dest_addr + CCP_SB_BYTES);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail =
(uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
}
/* Retrieve result */
pst.dest_addr = dest_addr;
pst.src_addr = (phys_addr_t)(cmd_q->sb_iv * CCP_SB_BYTES);
pst.len = CCP_SB_BYTES;
pst.dir = 0;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static int
ccp_perform_aes(struct rte_crypto_op *op,
struct ccp_queue *cmd_q,
struct ccp_batch_info *b_info)
{
struct ccp_session *session;
union ccp_function function;
uint8_t *lsb_buf;
struct ccp_passthru pst = {0};
struct ccp_desc *desc;
phys_addr_t src_addr, dest_addr, key_addr;
uint8_t *iv;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
function.raw = 0;
iv = rte_crypto_op_ctod_offset(op, uint8_t *, session->iv.offset);
if (session->cipher.um.aes_mode != CCP_AES_MODE_ECB) {
if (session->cipher.um.aes_mode == CCP_AES_MODE_CTR) {
rte_memcpy(session->cipher.nonce + AES_BLOCK_SIZE,
iv, session->iv.length);
pst.src_addr = (phys_addr_t)session->cipher.nonce_phys;
CCP_AES_SIZE(&function) = 0x1F;
} else {
lsb_buf =
&(b_info->lsb_buf[b_info->lsb_buf_idx*CCP_SB_BYTES]);
rte_memcpy(lsb_buf +
(CCP_SB_BYTES - session->iv.length),
iv, session->iv.length);
pst.src_addr = b_info->lsb_buf_phys +
(b_info->lsb_buf_idx * CCP_SB_BYTES);
b_info->lsb_buf_idx++;
}
pst.dest_addr = (phys_addr_t)(cmd_q->sb_iv * CCP_SB_BYTES);
pst.len = CCP_SB_BYTES;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
}
desc = &cmd_q->qbase_desc[cmd_q->qidx];
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->cipher.data.offset);
if (likely(op->sym->m_dst != NULL))
dest_addr = rte_pktmbuf_iova_offset(op->sym->m_dst,
op->sym->cipher.data.offset);
else
dest_addr = src_addr;
key_addr = session->cipher.key_phys;
/* prepare desc for aes command */
CCP_CMD_ENGINE(desc) = CCP_ENGINE_AES;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
CCP_AES_ENCRYPT(&function) = session->cipher.dir;
CCP_AES_MODE(&function) = session->cipher.um.aes_mode;
CCP_AES_TYPE(&function) = session->cipher.ut.aes_type;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = op->sym->cipher.data.length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = ((uint32_t)dest_addr);
CCP_CMD_DST_HI(desc) = high32_value(dest_addr);
CCP_CMD_DST_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
if (session->cipher.um.aes_mode != CCP_AES_MODE_ECB)
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static int
ccp_perform_3des(struct rte_crypto_op *op,
struct ccp_queue *cmd_q,
struct ccp_batch_info *b_info)
{
struct ccp_session *session;
union ccp_function function;
unsigned char *lsb_buf;
struct ccp_passthru pst;
struct ccp_desc *desc;
uint32_t tail;
uint8_t *iv;
phys_addr_t src_addr, dest_addr, key_addr;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
iv = rte_crypto_op_ctod_offset(op, uint8_t *, session->iv.offset);
switch (session->cipher.um.des_mode) {
case CCP_DES_MODE_CBC:
lsb_buf = &(b_info->lsb_buf[b_info->lsb_buf_idx*CCP_SB_BYTES]);
b_info->lsb_buf_idx++;
rte_memcpy(lsb_buf + (CCP_SB_BYTES - session->iv.length),
iv, session->iv.length);
pst.src_addr = (phys_addr_t)rte_mem_virt2phy((void *) lsb_buf);
pst.dest_addr = (phys_addr_t)(cmd_q->sb_iv * CCP_SB_BYTES);
pst.len = CCP_SB_BYTES;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_256BIT;
ccp_perform_passthru(&pst, cmd_q);
break;
case CCP_DES_MODE_CFB:
case CCP_DES_MODE_ECB:
CCP_LOG_ERR("Unsupported DES cipher mode");
return -ENOTSUP;
}
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->cipher.data.offset);
if (unlikely(op->sym->m_dst != NULL))
dest_addr =
rte_pktmbuf_iova_offset(op->sym->m_dst,
op->sym->cipher.data.offset);
else
dest_addr = src_addr;
key_addr = rte_mem_virt2phy(session->cipher.key_ccp);
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
/* prepare desc for des command */
CCP_CMD_ENGINE(desc) = CCP_ENGINE_3DES;
CCP_CMD_SOC(desc) = 0;
CCP_CMD_IOC(desc) = 0;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_EOM(desc) = 1;
CCP_CMD_PROT(desc) = 0;
function.raw = 0;
CCP_DES_ENCRYPT(&function) = session->cipher.dir;
CCP_DES_MODE(&function) = session->cipher.um.des_mode;
CCP_DES_TYPE(&function) = session->cipher.ut.des_type;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = op->sym->cipher.data.length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = ((uint32_t)dest_addr);
CCP_CMD_DST_HI(desc) = high32_value(dest_addr);
CCP_CMD_DST_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
if (session->cipher.um.des_mode)
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
/* Write the new tail address back to the queue register */
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
/* Turn the queue back on using our cached control register */
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static int
ccp_perform_aes_gcm(struct rte_crypto_op *op, struct ccp_queue *cmd_q)
{
struct ccp_session *session;
union ccp_function function;
uint8_t *iv;
struct ccp_passthru pst;
struct ccp_desc *desc;
uint32_t tail;
uint64_t *temp;
phys_addr_t src_addr, dest_addr, key_addr, aad_addr;
phys_addr_t digest_dest_addr;
int length, non_align_len;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
iv = rte_crypto_op_ctod_offset(op, uint8_t *, session->iv.offset);
key_addr = session->cipher.key_phys;
src_addr = rte_pktmbuf_iova_offset(op->sym->m_src,
op->sym->aead.data.offset);
if (unlikely(op->sym->m_dst != NULL))
dest_addr = rte_pktmbuf_iova_offset(op->sym->m_dst,
op->sym->aead.data.offset);
else
dest_addr = src_addr;
rte_pktmbuf_append(op->sym->m_src, session->auth.ctx_len);
digest_dest_addr = op->sym->aead.digest.phys_addr;
temp = (uint64_t *)(op->sym->aead.digest.data + AES_BLOCK_SIZE);
*temp++ = rte_bswap64(session->auth.aad_length << 3);
*temp = rte_bswap64(op->sym->aead.data.length << 3);
non_align_len = op->sym->aead.data.length % AES_BLOCK_SIZE;
length = CCP_ALIGN(op->sym->aead.data.length, AES_BLOCK_SIZE);
aad_addr = op->sym->aead.aad.phys_addr;
/* CMD1 IV Passthru */
rte_memcpy(session->cipher.nonce + AES_BLOCK_SIZE, iv,
session->iv.length);
pst.src_addr = session->cipher.nonce_phys;
pst.dest_addr = (phys_addr_t)(cmd_q->sb_iv * CCP_SB_BYTES);
pst.len = CCP_SB_BYTES;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
ccp_perform_passthru(&pst, cmd_q);
/* CMD2 GHASH-AAD */
function.raw = 0;
CCP_AES_ENCRYPT(&function) = CCP_AES_MODE_GHASH_AAD;
CCP_AES_MODE(&function) = CCP_AES_MODE_GHASH;
CCP_AES_TYPE(&function) = session->cipher.ut.aes_type;
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_AES;
CCP_CMD_INIT(desc) = 1;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = session->auth.aad_length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)aad_addr);
CCP_CMD_SRC_HI(desc) = high32_value(aad_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
/* CMD3 : GCTR Plain text */
function.raw = 0;
CCP_AES_ENCRYPT(&function) = session->cipher.dir;
CCP_AES_MODE(&function) = CCP_AES_MODE_GCTR;
CCP_AES_TYPE(&function) = session->cipher.ut.aes_type;
if (non_align_len == 0)
CCP_AES_SIZE(&function) = (AES_BLOCK_SIZE << 3) - 1;
else
CCP_AES_SIZE(&function) = (non_align_len << 3) - 1;
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_AES;
CCP_CMD_EOM(desc) = 1;
CCP_CMD_FUNCTION(desc) = function.raw;
CCP_CMD_LEN(desc) = length;
CCP_CMD_SRC_LO(desc) = ((uint32_t)src_addr);
CCP_CMD_SRC_HI(desc) = high32_value(src_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = ((uint32_t)dest_addr);
CCP_CMD_DST_HI(desc) = high32_value(dest_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
/* CMD4 : PT to copy IV */
pst.src_addr = session->cipher.nonce_phys;
pst.dest_addr = (phys_addr_t)(cmd_q->sb_iv * CCP_SB_BYTES);
pst.len = AES_BLOCK_SIZE;
pst.dir = 1;
pst.bit_mod = CCP_PASSTHRU_BITWISE_NOOP;
pst.byte_swap = CCP_PASSTHRU_BYTESWAP_NOOP;
ccp_perform_passthru(&pst, cmd_q);
/* CMD5 : GHASH-Final */
function.raw = 0;
CCP_AES_ENCRYPT(&function) = CCP_AES_MODE_GHASH_FINAL;
CCP_AES_MODE(&function) = CCP_AES_MODE_GHASH;
CCP_AES_TYPE(&function) = session->cipher.ut.aes_type;
desc = &cmd_q->qbase_desc[cmd_q->qidx];
memset(desc, 0, Q_DESC_SIZE);
CCP_CMD_ENGINE(desc) = CCP_ENGINE_AES;
CCP_CMD_FUNCTION(desc) = function.raw;
/* Last block (AAD_len || PT_len)*/
CCP_CMD_LEN(desc) = AES_BLOCK_SIZE;
CCP_CMD_SRC_LO(desc) = ((uint32_t)digest_dest_addr + AES_BLOCK_SIZE);
CCP_CMD_SRC_HI(desc) = high32_value(digest_dest_addr + AES_BLOCK_SIZE);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_DST_LO(desc) = ((uint32_t)digest_dest_addr);
CCP_CMD_DST_HI(desc) = high32_value(digest_dest_addr);
CCP_CMD_SRC_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_KEY_LO(desc) = ((uint32_t)key_addr);
CCP_CMD_KEY_HI(desc) = high32_value(key_addr);
CCP_CMD_KEY_MEM(desc) = CCP_MEMTYPE_SYSTEM;
CCP_CMD_LSB_ID(desc) = cmd_q->sb_iv;
cmd_q->qidx = (cmd_q->qidx + 1) % COMMANDS_PER_QUEUE;
rte_wmb();
tail = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx * Q_DESC_SIZE);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE, tail);
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
op->status = RTE_CRYPTO_OP_STATUS_NOT_PROCESSED;
return 0;
}
static inline int
ccp_crypto_cipher(struct rte_crypto_op *op,
struct ccp_queue *cmd_q,
struct ccp_batch_info *b_info)
{
int result = 0;
struct ccp_session *session;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
switch (session->cipher.algo) {
case CCP_CIPHER_ALGO_AES_CBC:
result = ccp_perform_aes(op, cmd_q, b_info);
b_info->desccnt += 2;
break;
case CCP_CIPHER_ALGO_AES_CTR:
result = ccp_perform_aes(op, cmd_q, b_info);
b_info->desccnt += 2;
break;
case CCP_CIPHER_ALGO_AES_ECB:
result = ccp_perform_aes(op, cmd_q, b_info);
b_info->desccnt += 1;
break;
case CCP_CIPHER_ALGO_3DES_CBC:
result = ccp_perform_3des(op, cmd_q, b_info);
b_info->desccnt += 2;
break;
default:
CCP_LOG_ERR("Unsupported cipher algo %d",
session->cipher.algo);
return -ENOTSUP;
}
return result;
}
static inline int
ccp_crypto_auth(struct rte_crypto_op *op,
struct ccp_queue *cmd_q,
struct ccp_batch_info *b_info)
{
int result = 0;
struct ccp_session *session;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
switch (session->auth.algo) {
case CCP_AUTH_ALGO_SHA1:
case CCP_AUTH_ALGO_SHA224:
case CCP_AUTH_ALGO_SHA256:
case CCP_AUTH_ALGO_SHA384:
case CCP_AUTH_ALGO_SHA512:
result = ccp_perform_sha(op, cmd_q);
b_info->desccnt += 3;
break;
case CCP_AUTH_ALGO_MD5_HMAC:
if (session->auth_opt == 0)
result = -1;
break;
case CCP_AUTH_ALGO_SHA1_HMAC:
case CCP_AUTH_ALGO_SHA224_HMAC:
case CCP_AUTH_ALGO_SHA256_HMAC:
if (session->auth_opt == 0) {
result = ccp_perform_hmac(op, cmd_q);
b_info->desccnt += 6;
}
break;
case CCP_AUTH_ALGO_SHA384_HMAC:
case CCP_AUTH_ALGO_SHA512_HMAC:
if (session->auth_opt == 0) {
result = ccp_perform_hmac(op, cmd_q);
b_info->desccnt += 7;
}
break;
case CCP_AUTH_ALGO_SHA3_224:
case CCP_AUTH_ALGO_SHA3_256:
case CCP_AUTH_ALGO_SHA3_384:
case CCP_AUTH_ALGO_SHA3_512:
result = ccp_perform_sha3(op, cmd_q);
b_info->desccnt += 1;
break;
case CCP_AUTH_ALGO_SHA3_224_HMAC:
case CCP_AUTH_ALGO_SHA3_256_HMAC:
result = ccp_perform_sha3_hmac(op, cmd_q);
b_info->desccnt += 3;
break;
case CCP_AUTH_ALGO_SHA3_384_HMAC:
case CCP_AUTH_ALGO_SHA3_512_HMAC:
result = ccp_perform_sha3_hmac(op, cmd_q);
b_info->desccnt += 4;
break;
case CCP_AUTH_ALGO_AES_CMAC:
result = ccp_perform_aes_cmac(op, cmd_q);
b_info->desccnt += 4;
break;
default:
CCP_LOG_ERR("Unsupported auth algo %d",
session->auth.algo);
return -ENOTSUP;
}
return result;
}
static inline int
ccp_crypto_aead(struct rte_crypto_op *op,
struct ccp_queue *cmd_q,
struct ccp_batch_info *b_info)
{
int result = 0;
struct ccp_session *session;
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
switch (session->auth.algo) {
case CCP_AUTH_ALGO_AES_GCM:
if (session->cipher.algo != CCP_CIPHER_ALGO_AES_GCM) {
CCP_LOG_ERR("Incorrect chain order");
return -1;
}
result = ccp_perform_aes_gcm(op, cmd_q);
b_info->desccnt += 5;
break;
default:
CCP_LOG_ERR("Unsupported aead algo %d",
session->aead_algo);
return -ENOTSUP;
}
return result;
}
int
process_ops_to_enqueue(struct ccp_qp *qp,
struct rte_crypto_op **op,
struct ccp_queue *cmd_q,
uint16_t nb_ops,
uint16_t total_nb_ops,
int slots_req,
uint16_t b_idx)
{
int i, result = 0;
struct ccp_batch_info *b_info;
struct ccp_session *session;
EVP_MD_CTX *auth_ctx = NULL;
if (rte_mempool_get(qp->batch_mp, (void **)&b_info)) {
CCP_LOG_ERR("batch info allocation failed");
return 0;
}
auth_ctx = EVP_MD_CTX_create();
if (unlikely(!auth_ctx)) {
CCP_LOG_ERR("Unable to create auth ctx");
return 0;
}
b_info->auth_ctr = 0;
/* populate batch info necessary for dequeue */
b_info->op_idx = 0;
b_info->b_idx = 0;
b_info->lsb_buf_idx = 0;
b_info->desccnt = 0;
b_info->cmd_q = cmd_q;
b_info->lsb_buf_phys =
(phys_addr_t)rte_mem_virt2phy((void *)b_info->lsb_buf);
rte_atomic64_sub(&b_info->cmd_q->free_slots, slots_req);
b_info->head_offset = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx *
Q_DESC_SIZE);
for (i = b_idx; i < (nb_ops+b_idx); i++) {
session = (struct ccp_session *)get_sym_session_private_data(
op[i]->sym->session,
ccp_cryptodev_driver_id);
switch (session->cmd_id) {
case CCP_CMD_CIPHER:
result = ccp_crypto_cipher(op[i], cmd_q, b_info);
break;
case CCP_CMD_AUTH:
if (session->auth_opt) {
b_info->auth_ctr++;
result = cpu_crypto_auth(qp, op[i],
session, auth_ctx);
} else
result = ccp_crypto_auth(op[i], cmd_q, b_info);
break;
case CCP_CMD_CIPHER_HASH:
result = ccp_crypto_cipher(op[i], cmd_q, b_info);
if (result)
break;
result = ccp_crypto_auth(op[i], cmd_q, b_info);
break;
case CCP_CMD_HASH_CIPHER:
if (session->auth_opt) {
result = cpu_crypto_auth(qp, op[i],
session, auth_ctx);
if (op[i]->status !=
RTE_CRYPTO_OP_STATUS_SUCCESS)
CCP_LOG_ERR("RTE_CRYPTO_OP_STATUS_AUTH_FAILED");
} else
result = ccp_crypto_auth(op[i], cmd_q, b_info);
if (result)
break;
result = ccp_crypto_cipher(op[i], cmd_q, b_info);
break;
case CCP_CMD_COMBINED:
result = ccp_crypto_aead(op[i], cmd_q, b_info);
break;
default:
CCP_LOG_ERR("Unsupported cmd_id");
result = -1;
}
if (unlikely(result < 0)) {
rte_atomic64_add(&b_info->cmd_q->free_slots,
(slots_req - b_info->desccnt));
break;
}
b_info->op[i] = op[i];
}
b_info->opcnt = i;
b_info->b_idx = b_idx;
b_info->total_nb_ops = total_nb_ops;
b_info->tail_offset = (uint32_t)(cmd_q->qbase_phys_addr + cmd_q->qidx *
Q_DESC_SIZE);
rte_wmb();
/* Write the new tail address back to the queue register */
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_TAIL_LO_BASE,
b_info->tail_offset);
/* Turn the queue back on using our cached control register */
CCP_WRITE_REG(cmd_q->reg_base, CMD_Q_CONTROL_BASE,
cmd_q->qcontrol | CMD_Q_RUN);
rte_ring_enqueue(qp->processed_pkts, (void *)b_info);
EVP_MD_CTX_destroy(auth_ctx);
return i-b_idx;
}
static inline void ccp_auth_dq_prepare(struct rte_crypto_op *op)
{
struct ccp_session *session;
uint8_t *digest_data, *addr;
struct rte_mbuf *m_last;
int offset, digest_offset;
uint8_t digest_le[64];
session = (struct ccp_session *)get_sym_session_private_data(
op->sym->session,
ccp_cryptodev_driver_id);
if (session->cmd_id == CCP_CMD_COMBINED) {
digest_data = op->sym->aead.digest.data;
digest_offset = op->sym->aead.data.offset +
op->sym->aead.data.length;
} else {
digest_data = op->sym->auth.digest.data;
digest_offset = op->sym->auth.data.offset +
op->sym->auth.data.length;
}
m_last = rte_pktmbuf_lastseg(op->sym->m_src);
addr = (uint8_t *)((char *)m_last->buf_addr + m_last->data_off +
m_last->data_len - session->auth.ctx_len);
rte_mb();
offset = session->auth.offset;
if (session->auth.engine == CCP_ENGINE_SHA)
if ((session->auth.ut.sha_type != CCP_SHA_TYPE_1) &&
(session->auth.ut.sha_type != CCP_SHA_TYPE_224) &&
(session->auth.ut.sha_type != CCP_SHA_TYPE_256)) {
/* All other algorithms require byte
* swap done by host
*/
unsigned int i;
offset = session->auth.ctx_len -
session->auth.offset - 1;
for (i = 0; i < session->auth.digest_length; i++)
digest_le[i] = addr[offset - i];
offset = 0;
addr = digest_le;
}
op->status = RTE_CRYPTO_OP_STATUS_SUCCESS;
if (session->auth.op == CCP_AUTH_OP_VERIFY) {
if (memcmp(addr + offset, digest_data,
session->auth.digest_length) != 0)
op->status = RTE_CRYPTO_OP_STATUS_AUTH_FAILED;
} else {
if (unlikely(digest_data == 0))
digest_data = rte_pktmbuf_mtod_offset(
op->sym->m_dst, uint8_t *,
digest_offset);
rte_memcpy(digest_data, addr + offset,
session->auth.digest_length);
}
/* Trim area used for digest from mbuf. */
rte_pktmbuf_trim(op->sym->m_src,
session->auth.ctx_len);
}
static int
ccp_prepare_ops(struct ccp_qp *qp,
struct rte_crypto_op **op_d,
struct ccp_batch_info *b_info,
uint16_t nb_ops)
{
int i, min_ops;
struct ccp_session *session;
EVP_MD_CTX *auth_ctx = NULL;
auth_ctx = EVP_MD_CTX_create();
if (unlikely(!auth_ctx)) {
CCP_LOG_ERR("Unable to create auth ctx");
return 0;
}
min_ops = RTE_MIN(nb_ops, b_info->opcnt);
for (i = b_info->b_idx; i < min_ops; i++) {
op_d[i] = b_info->op[b_info->b_idx + b_info->op_idx++];
session = (struct ccp_session *)get_sym_session_private_data(
op_d[i]->sym->session,
ccp_cryptodev_driver_id);
switch (session->cmd_id) {
case CCP_CMD_CIPHER:
op_d[i]->status = RTE_CRYPTO_OP_STATUS_SUCCESS;
break;
case CCP_CMD_AUTH:
if (session->auth_opt == 0)
ccp_auth_dq_prepare(op_d[i]);
break;
case CCP_CMD_CIPHER_HASH:
if (session->auth_opt)
cpu_crypto_auth(qp, op_d[i],
session, auth_ctx);
else
ccp_auth_dq_prepare(op_d[i]);
break;
case CCP_CMD_HASH_CIPHER:
if (session->auth_opt)
op_d[i]->status = RTE_CRYPTO_OP_STATUS_SUCCESS;
else
ccp_auth_dq_prepare(op_d[i]);
break;
case CCP_CMD_COMBINED:
ccp_auth_dq_prepare(op_d[i]);
break;
default:
CCP_LOG_ERR("Unsupported cmd_id");
}
}
EVP_MD_CTX_destroy(auth_ctx);
b_info->opcnt -= min_ops;
return min_ops;
}
int
process_ops_to_dequeue(struct ccp_qp *qp,
struct rte_crypto_op **op,
uint16_t nb_ops,
uint16_t *total_nb_ops)
{
struct ccp_batch_info *b_info;
uint32_t cur_head_offset;
if (qp->b_info != NULL) {
b_info = qp->b_info;
if (unlikely(b_info->op_idx > 0))
goto success;
} else if (rte_ring_dequeue(qp->processed_pkts,
(void **)&b_info))
return 0;
if (b_info->auth_ctr == b_info->opcnt)
goto success;
*total_nb_ops = b_info->total_nb_ops;
cur_head_offset = CCP_READ_REG(b_info->cmd_q->reg_base,
CMD_Q_HEAD_LO_BASE);
if (b_info->head_offset < b_info->tail_offset) {
if ((cur_head_offset >= b_info->head_offset) &&
(cur_head_offset < b_info->tail_offset)) {
qp->b_info = b_info;
return 0;
}
} else if (b_info->tail_offset != b_info->head_offset) {
if ((cur_head_offset >= b_info->head_offset) ||
(cur_head_offset < b_info->tail_offset)) {
qp->b_info = b_info;
return 0;
}
}
success:
*total_nb_ops = b_info->total_nb_ops;
nb_ops = ccp_prepare_ops(qp, op, b_info, nb_ops);
rte_atomic64_add(&b_info->cmd_q->free_slots, b_info->desccnt);
b_info->desccnt = 0;
if (b_info->opcnt > 0) {
qp->b_info = b_info;
} else {
rte_mempool_put(qp->batch_mp, (void *)b_info);
qp->b_info = NULL;
}
return nb_ops;
}