Skip to content

Commit 677c4a0

Browse files
committed
s390x assembly pack: process x25519 and x448 non-canonical values
...in constant time. Signed-off-by: Patrick Steuer <[email protected]> Reviewed-by: Matt Caswell <[email protected]> (Merged from #10339)
1 parent 6376c22 commit 677c4a0

File tree

2 files changed

+37
-8
lines changed

2 files changed

+37
-8
lines changed

crypto/ec/ecx_meth.c

+9-8
Original file line numberDiff line numberDiff line change
@@ -854,6 +854,7 @@ static const EVP_PKEY_METHOD ed448_pkey_meth = {
854854

855855
#ifdef S390X_EC_ASM
856856
# include "s390x_arch.h"
857+
# include "internal/constant_time.h"
857858

858859
static void s390x_x25519_mod_p(unsigned char u[32])
859860
{
@@ -867,16 +868,16 @@ static void s390x_x25519_mod_p(unsigned char u[32])
867868
u_red[31] = (unsigned char)c;
868869
c >>= 8;
869870

870-
for (i = 30; c > 0 && i >= 0; i--) {
871+
for (i = 30; i >= 0; i--) {
871872
c += (unsigned int)u_red[i];
872873
u_red[i] = (unsigned char)c;
873874
c >>= 8;
874875
}
875876

876-
if (u_red[0] & 0x80) {
877-
u_red[0] &= 0x7f;
878-
memcpy(u, u_red, sizeof(u_red));
879-
}
877+
c = (u_red[0] & 0x80) >> 7;
878+
u_red[0] &= 0x7f;
879+
constant_time_cond_swap_buff(0 - (unsigned char)c,
880+
u, u_red, sizeof(u_red));
880881
}
881882

882883
static void s390x_x448_mod_p(unsigned char u[56])
@@ -901,14 +902,14 @@ static void s390x_x448_mod_p(unsigned char u[56])
901902
u_red[27] = (unsigned char)c;
902903
c >>= 8;
903904

904-
for (i = 26; c > 0 && i >= 0; i--) {
905+
for (i = 26; i >= 0; i--) {
905906
c += (unsigned int)u_red[i];
906907
u_red[i] = (unsigned char)c;
907908
c >>= 8;
908909
}
909910

910-
if (c)
911-
memcpy(u, u_red, sizeof(u_red));
911+
constant_time_cond_swap_buff(0 - (unsigned char)c,
912+
u, u_red, sizeof(u_red));
912913
}
913914

914915
static int s390x_x25519_mul(unsigned char u_dst[32],

include/internal/constant_time.h

+28
Original file line numberDiff line numberDiff line change
@@ -352,6 +352,34 @@ static ossl_inline void constant_time_cond_swap_64(uint64_t mask, uint64_t *a,
352352
*b ^= xor;
353353
}
354354

355+
/*
356+
* mask must be 0xFF or 0x00.
357+
* "constant time" is per len.
358+
*
359+
* if (mask) {
360+
* unsigned char tmp[len];
361+
*
362+
* memcpy(tmp, a, len);
363+
* memcpy(a, b);
364+
* memcpy(b, tmp);
365+
* }
366+
*/
367+
static ossl_inline void constant_time_cond_swap_buff(unsigned char mask,
368+
unsigned char *a,
369+
unsigned char *b,
370+
size_t len)
371+
{
372+
size_t i;
373+
unsigned char tmp;
374+
375+
for (i = 0; i < len; i++) {
376+
tmp = a[i] ^ b[i];
377+
tmp &= mask;
378+
a[i] ^= tmp;
379+
b[i] ^= tmp;
380+
}
381+
}
382+
355383
/*
356384
* table is a two dimensional array of bytes. Each row has rowsize elements.
357385
* Copies row number idx into out. rowsize and numrows are not considered

0 commit comments

Comments
 (0)