Commit 6825cfd6 authored by Gilad Ben-Yossef's avatar Gilad Ben-Yossef Committed by Herbert Xu

crypto: ccree - simplify fragment ICV detection

The code detecting whether the ICV is fragmented was overly
complex and limited the number of fragments an ICV may be
comprised of with no reason in the current code, casuing the
new testmgr tests to fail.

This patch removes this legacy limitation and greatly simplifies
the code.
Signed-off-by: default avatarGilad Ben-Yossef <gilad@benyossef.com>
Signed-off-by: default avatarHerbert Xu <herbert@gondor.apana.org.au>
parent bd233baf
...@@ -602,55 +602,10 @@ void cc_unmap_aead_request(struct device *dev, struct aead_request *req) ...@@ -602,55 +602,10 @@ void cc_unmap_aead_request(struct device *dev, struct aead_request *req)
} }
} }
static int cc_get_aead_icv_nents(struct device *dev, struct scatterlist *sgl, static bool cc_is_icv_frag(unsigned int sgl_nents, unsigned int authsize,
unsigned int sgl_nents, unsigned int authsize, u32 last_entry_data_size)
u32 last_entry_data_size,
bool *is_icv_fragmented)
{ {
unsigned int icv_max_size = 0; return ((sgl_nents > 1) && (last_entry_data_size < authsize));
unsigned int icv_required_size = authsize > last_entry_data_size ?
(authsize - last_entry_data_size) :
authsize;
unsigned int nents;
unsigned int i;
if (sgl_nents < MAX_ICV_NENTS_SUPPORTED) {
*is_icv_fragmented = false;
return 0;
}
for (i = 0 ; i < (sgl_nents - MAX_ICV_NENTS_SUPPORTED) ; i++) {
if (!sgl)
break;
sgl = sg_next(sgl);
}
if (sgl)
icv_max_size = sgl->length;
if (last_entry_data_size > authsize) {
/* ICV attached to data in last entry (not fragmented!) */
nents = 0;
*is_icv_fragmented = false;
} else if (last_entry_data_size == authsize) {
/* ICV placed in whole last entry (not fragmented!) */
nents = 1;
*is_icv_fragmented = false;
} else if (icv_max_size > icv_required_size) {
nents = 1;
*is_icv_fragmented = true;
} else if (icv_max_size == icv_required_size) {
nents = 2;
*is_icv_fragmented = true;
} else {
dev_err(dev, "Unsupported num. of ICV fragments (> %d)\n",
MAX_ICV_NENTS_SUPPORTED);
nents = -1; /*unsupported*/
}
dev_dbg(dev, "is_frag=%s icv_nents=%u\n",
(*is_icv_fragmented ? "true" : "false"), nents);
return nents;
} }
static int cc_aead_chain_iv(struct cc_drvdata *drvdata, static int cc_aead_chain_iv(struct cc_drvdata *drvdata,
...@@ -817,16 +772,15 @@ static void cc_prepare_aead_data_dlli(struct aead_request *req, ...@@ -817,16 +772,15 @@ static void cc_prepare_aead_data_dlli(struct aead_request *req,
} }
} }
static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata, static void cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata,
struct aead_request *req, struct aead_request *req,
struct buffer_array *sg_data, struct buffer_array *sg_data,
u32 *src_last_bytes, u32 *dst_last_bytes, u32 *src_last_bytes, u32 *dst_last_bytes,
bool is_last_table) bool is_last_table)
{ {
struct aead_req_ctx *areq_ctx = aead_request_ctx(req); struct aead_req_ctx *areq_ctx = aead_request_ctx(req);
enum drv_crypto_direction direct = areq_ctx->gen_ctx.op_type; enum drv_crypto_direction direct = areq_ctx->gen_ctx.op_type;
unsigned int authsize = areq_ctx->req_authsize; unsigned int authsize = areq_ctx->req_authsize;
int rc = 0, icv_nents;
struct device *dev = drvdata_to_dev(drvdata); struct device *dev = drvdata_to_dev(drvdata);
struct scatterlist *sg; struct scatterlist *sg;
...@@ -837,14 +791,9 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata, ...@@ -837,14 +791,9 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata,
areq_ctx->src_offset, is_last_table, areq_ctx->src_offset, is_last_table,
&areq_ctx->src.mlli_nents); &areq_ctx->src.mlli_nents);
icv_nents = cc_get_aead_icv_nents(dev, areq_ctx->src_sgl, areq_ctx->is_icv_fragmented =
areq_ctx->src.nents, cc_is_icv_frag(areq_ctx->src.nents, authsize,
authsize, *src_last_bytes, *src_last_bytes);
&areq_ctx->is_icv_fragmented);
if (icv_nents < 0) {
rc = -ENOTSUPP;
goto prepare_data_mlli_exit;
}
if (areq_ctx->is_icv_fragmented) { if (areq_ctx->is_icv_fragmented) {
/* Backup happens only when ICV is fragmented, ICV /* Backup happens only when ICV is fragmented, ICV
...@@ -886,16 +835,11 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata, ...@@ -886,16 +835,11 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata,
areq_ctx->dst_offset, is_last_table, areq_ctx->dst_offset, is_last_table,
&areq_ctx->dst.mlli_nents); &areq_ctx->dst.mlli_nents);
icv_nents = cc_get_aead_icv_nents(dev, areq_ctx->src_sgl, areq_ctx->is_icv_fragmented =
areq_ctx->src.nents, cc_is_icv_frag(areq_ctx->src.nents, authsize,
authsize, *src_last_bytes, *src_last_bytes);
&areq_ctx->is_icv_fragmented);
if (icv_nents < 0) {
rc = -ENOTSUPP;
goto prepare_data_mlli_exit;
}
/* Backup happens only when ICV is fragmented, ICV /* Backup happens only when ICV is fragmented, ICV
* verification is made by CPU compare in order to simplify * verification is made by CPU compare in order to simplify
* MAC verification upon request completion * MAC verification upon request completion
*/ */
...@@ -923,14 +867,9 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata, ...@@ -923,14 +867,9 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata,
areq_ctx->src_offset, is_last_table, areq_ctx->src_offset, is_last_table,
&areq_ctx->src.mlli_nents); &areq_ctx->src.mlli_nents);
icv_nents = cc_get_aead_icv_nents(dev, areq_ctx->dst_sgl, areq_ctx->is_icv_fragmented =
areq_ctx->dst.nents, cc_is_icv_frag(areq_ctx->dst.nents, authsize,
authsize, *dst_last_bytes, *dst_last_bytes);
&areq_ctx->is_icv_fragmented);
if (icv_nents < 0) {
rc = -ENOTSUPP;
goto prepare_data_mlli_exit;
}
if (!areq_ctx->is_icv_fragmented) { if (!areq_ctx->is_icv_fragmented) {
sg = &areq_ctx->dst_sgl[areq_ctx->dst.nents - 1]; sg = &areq_ctx->dst_sgl[areq_ctx->dst.nents - 1];
...@@ -944,9 +883,6 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata, ...@@ -944,9 +883,6 @@ static int cc_prepare_aead_data_mlli(struct cc_drvdata *drvdata,
areq_ctx->icv_virt_addr = areq_ctx->mac_buf; areq_ctx->icv_virt_addr = areq_ctx->mac_buf;
} }
} }
prepare_data_mlli_exit:
return rc;
} }
static int cc_aead_chain_data(struct cc_drvdata *drvdata, static int cc_aead_chain_data(struct cc_drvdata *drvdata,
...@@ -1053,9 +989,9 @@ static int cc_aead_chain_data(struct cc_drvdata *drvdata, ...@@ -1053,9 +989,9 @@ static int cc_aead_chain_data(struct cc_drvdata *drvdata,
dst_mapped_nents > 1 || dst_mapped_nents > 1 ||
do_chain) { do_chain) {
areq_ctx->data_buff_type = CC_DMA_BUF_MLLI; areq_ctx->data_buff_type = CC_DMA_BUF_MLLI;
rc = cc_prepare_aead_data_mlli(drvdata, req, sg_data, cc_prepare_aead_data_mlli(drvdata, req, sg_data,
&src_last_bytes, &src_last_bytes, &dst_last_bytes,
&dst_last_bytes, is_last_table); is_last_table);
} else { } else {
areq_ctx->data_buff_type = CC_DMA_BUF_DLLI; areq_ctx->data_buff_type = CC_DMA_BUF_DLLI;
cc_prepare_aead_data_dlli(req, &src_last_bytes, cc_prepare_aead_data_dlli(req, &src_last_bytes,
......
...@@ -103,7 +103,6 @@ enum cc_std_body { ...@@ -103,7 +103,6 @@ enum cc_std_body {
#define MAX_REQUEST_QUEUE_SIZE 4096 #define MAX_REQUEST_QUEUE_SIZE 4096
#define MAX_MLLI_BUFF_SIZE 2080 #define MAX_MLLI_BUFF_SIZE 2080
#define MAX_ICV_NENTS_SUPPORTED 2
/* Definitions for HW descriptors DIN/DOUT fields */ /* Definitions for HW descriptors DIN/DOUT fields */
#define NS_BIT 1 #define NS_BIT 1
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment