Skip to content

Commit 1831597

Browse files
chleroyksacilotto
authored andcommitted
crypto: talitos - Work around SEC6 ERRATA (AES-CTR mode data size error)
BugLink: https://bugs.launchpad.net/bugs/1918974 [ Upstream commit 416b846 ] Talitos Security Engine AESU considers any input data size that is not a multiple of 16 bytes to be an error. This is not a problem in general, except for Counter mode that is a stream cipher and can have an input of any size. Test Manager for ctr(aes) fails on 4th test vector which has a length of 499 while all previous vectors which have a 16 bytes multiple length succeed. As suggested by Freescale, round up the input data length to the nearest 16 bytes. Fixes: 5e75ae1 ("crypto: talitos - add new crypto modes") Signed-off-by: Christophe Leroy <[email protected]> Signed-off-by: Herbert Xu <[email protected]> Signed-off-by: Sasha Levin <[email protected]> Signed-off-by: Kamal Mostafa <[email protected]> Signed-off-by: Kelsey Skunberg <[email protected]>
1 parent 26d1bc6 commit 1831597

File tree

2 files changed

+17
-12
lines changed

2 files changed

+17
-12
lines changed

drivers/crypto/talitos.c

Lines changed: 16 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -1097,11 +1097,12 @@ static void ipsec_esp_decrypt_hwauth_done(struct device *dev,
10971097
*/
10981098
static int sg_to_link_tbl_offset(struct scatterlist *sg, int sg_count,
10991099
unsigned int offset, int datalen, int elen,
1100-
struct talitos_ptr *link_tbl_ptr)
1100+
struct talitos_ptr *link_tbl_ptr, int align)
11011101
{
11021102
int n_sg = elen ? sg_count + 1 : sg_count;
11031103
int count = 0;
11041104
int cryptlen = datalen + elen;
1105+
int padding = ALIGN(cryptlen, align) - cryptlen;
11051106

11061107
while (cryptlen && sg && n_sg--) {
11071108
unsigned int len = sg_dma_len(sg);
@@ -1125,7 +1126,7 @@ static int sg_to_link_tbl_offset(struct scatterlist *sg, int sg_count,
11251126
offset += datalen;
11261127
}
11271128
to_talitos_ptr(link_tbl_ptr + count,
1128-
sg_dma_address(sg) + offset, len, 0);
1129+
sg_dma_address(sg) + offset, sg_next(sg) ? len : len + padding, 0);
11291130
to_talitos_ptr_ext_set(link_tbl_ptr + count, 0, 0);
11301131
count++;
11311132
cryptlen -= len;
@@ -1148,33 +1149,34 @@ static int talitos_sg_map_ext(struct device *dev, struct scatterlist *src,
11481149
unsigned int len, struct talitos_edesc *edesc,
11491150
struct talitos_ptr *ptr, int sg_count,
11501151
unsigned int offset, int tbl_off, int elen,
1151-
bool force)
1152+
bool force, int align)
11521153
{
11531154
struct talitos_private *priv = dev_get_drvdata(dev);
11541155
bool is_sec1 = has_ftr_sec1(priv);
1156+
int aligned_len = ALIGN(len, align);
11551157

11561158
if (!src) {
11571159
to_talitos_ptr(ptr, 0, 0, is_sec1);
11581160
return 1;
11591161
}
11601162
to_talitos_ptr_ext_set(ptr, elen, is_sec1);
11611163
if (sg_count == 1 && !force) {
1162-
to_talitos_ptr(ptr, sg_dma_address(src) + offset, len, is_sec1);
1164+
to_talitos_ptr(ptr, sg_dma_address(src) + offset, aligned_len, is_sec1);
11631165
return sg_count;
11641166
}
11651167
if (is_sec1) {
1166-
to_talitos_ptr(ptr, edesc->dma_link_tbl + offset, len, is_sec1);
1168+
to_talitos_ptr(ptr, edesc->dma_link_tbl + offset, aligned_len, is_sec1);
11671169
return sg_count;
11681170
}
11691171
sg_count = sg_to_link_tbl_offset(src, sg_count, offset, len, elen,
1170-
&edesc->link_tbl[tbl_off]);
1172+
&edesc->link_tbl[tbl_off], align);
11711173
if (sg_count == 1 && !force) {
11721174
/* Only one segment now, so no link tbl needed*/
11731175
copy_talitos_ptr(ptr, &edesc->link_tbl[tbl_off], is_sec1);
11741176
return sg_count;
11751177
}
11761178
to_talitos_ptr(ptr, edesc->dma_link_tbl +
1177-
tbl_off * sizeof(struct talitos_ptr), len, is_sec1);
1179+
tbl_off * sizeof(struct talitos_ptr), aligned_len, is_sec1);
11781180
to_talitos_ptr_ext_or(ptr, DESC_PTR_LNKTBL_JUMP, is_sec1);
11791181

11801182
return sg_count;
@@ -1186,7 +1188,7 @@ static int talitos_sg_map(struct device *dev, struct scatterlist *src,
11861188
unsigned int offset, int tbl_off)
11871189
{
11881190
return talitos_sg_map_ext(dev, src, len, edesc, ptr, sg_count, offset,
1189-
tbl_off, 0, false);
1191+
tbl_off, 0, false, 1);
11901192
}
11911193

11921194
/*
@@ -1255,7 +1257,7 @@ static int ipsec_esp(struct talitos_edesc *edesc, struct aead_request *areq,
12551257

12561258
ret = talitos_sg_map_ext(dev, areq->src, cryptlen, edesc, &desc->ptr[4],
12571259
sg_count, areq->assoclen, tbl_off, elen,
1258-
false);
1260+
false, 1);
12591261

12601262
if (ret > 1) {
12611263
tbl_off += ret;
@@ -1275,7 +1277,7 @@ static int ipsec_esp(struct talitos_edesc *edesc, struct aead_request *areq,
12751277
elen = 0;
12761278
ret = talitos_sg_map_ext(dev, areq->dst, cryptlen, edesc, &desc->ptr[5],
12771279
sg_count, areq->assoclen, tbl_off, elen,
1278-
is_ipsec_esp && !encrypt);
1280+
is_ipsec_esp && !encrypt, 1);
12791281
tbl_off += ret;
12801282

12811283
if (!encrypt && is_ipsec_esp) {
@@ -1583,6 +1585,8 @@ static int common_nonsnoop(struct talitos_edesc *edesc,
15831585
bool sync_needed = false;
15841586
struct talitos_private *priv = dev_get_drvdata(dev);
15851587
bool is_sec1 = has_ftr_sec1(priv);
1588+
bool is_ctr = (desc->hdr & DESC_HDR_SEL0_MASK) == DESC_HDR_SEL0_AESU &&
1589+
(desc->hdr & DESC_HDR_MODE0_AESU_MASK) == DESC_HDR_MODE0_AESU_CTR;
15861590

15871591
/* first DWORD empty */
15881592

@@ -1603,8 +1607,8 @@ static int common_nonsnoop(struct talitos_edesc *edesc,
16031607
/*
16041608
* cipher in
16051609
*/
1606-
sg_count = talitos_sg_map(dev, areq->src, cryptlen, edesc,
1607-
&desc->ptr[3], sg_count, 0, 0);
1610+
sg_count = talitos_sg_map_ext(dev, areq->src, cryptlen, edesc, &desc->ptr[3],
1611+
sg_count, 0, 0, 0, false, is_ctr ? 16 : 1);
16081612
if (sg_count > 1)
16091613
sync_needed = true;
16101614

drivers/crypto/talitos.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -344,6 +344,7 @@ static inline bool has_ftr_sec1(struct talitos_private *priv)
344344

345345
/* primary execution unit mode (MODE0) and derivatives */
346346
#define DESC_HDR_MODE0_ENCRYPT cpu_to_be32(0x00100000)
347+
#define DESC_HDR_MODE0_AESU_MASK cpu_to_be32(0x00600000)
347348
#define DESC_HDR_MODE0_AESU_CBC cpu_to_be32(0x00200000)
348349
#define DESC_HDR_MODE0_AESU_CTR cpu_to_be32(0x00600000)
349350
#define DESC_HDR_MODE0_DEU_CBC cpu_to_be32(0x00400000)

0 commit comments

Comments
 (0)