1369 lines
41 KiB
Diff
1369 lines
41 KiB
Diff
diff -up openssl-1.0.2k/crypto/aes/asm/aes-s390x.pl.s390x-update openssl-1.0.2k/crypto/aes/asm/aes-s390x.pl
|
|
--- openssl-1.0.2k/crypto/aes/asm/aes-s390x.pl.s390x-update 2018-06-18 12:20:47.197994636 +0200
|
|
+++ openssl-1.0.2k/crypto/aes/asm/aes-s390x.pl 2018-06-18 13:27:37.109817203 +0200
|
|
@@ -122,6 +122,8 @@ sub _data_word()
|
|
}
|
|
|
|
$code=<<___;
|
|
+#include "s390x_arch.h"
|
|
+
|
|
.text
|
|
|
|
.type AES_Te,\@object
|
|
@@ -814,13 +816,10 @@ $code.=<<___ if (!$softonly);
|
|
ar %r5,%r0
|
|
|
|
larl %r1,OPENSSL_s390xcap_P
|
|
- lg %r0,0(%r1)
|
|
- tmhl %r0,0x4000 # check for message-security assist
|
|
- jz .Lekey_internal
|
|
-
|
|
llihh %r0,0x8000
|
|
srlg %r0,%r0,0(%r5)
|
|
- ng %r0,48(%r1) # check kmc capability vector
|
|
+ ng %r0,S390X_KM(%r1) # check availability of both km...
|
|
+ ng %r0,S390X_KMC(%r1) # ...and kmc support for given key length
|
|
jz .Lekey_internal
|
|
|
|
lmg %r0,%r1,0($inp) # just copy 128 bits...
|
|
@@ -1443,7 +1442,7 @@ $code.=<<___ if (0); ######### kmctr cod
|
|
larl %r1,OPENSSL_s390xcap_P
|
|
llihh %r0,0x8000 # check if kmctr supports the function code
|
|
srlg %r0,%r0,0($s0)
|
|
- ng %r0,64(%r1) # check kmctr capability vector
|
|
+ ng %r0,S390X_KMCTR(%r1) # check kmctr capability vector
|
|
lgr %r0,$s0
|
|
lgr %r1,$s1
|
|
jz .Lctr32_km_loop
|
|
@@ -1593,7 +1592,7 @@ $code.=<<___ if(1);
|
|
larl %r1,OPENSSL_s390xcap_P
|
|
llihh %r0,0x8000
|
|
srlg %r0,%r0,32($s1) # check for 32+function code
|
|
- ng %r0,32(%r1) # check km capability vector
|
|
+ ng %r0,S390X_KM(%r1) # check km capability vector
|
|
lgr %r0,$s0 # restore the function code
|
|
la %r1,0($key1) # restore $key1
|
|
jz .Lxts_km_vanilla
|
|
@@ -2220,7 +2219,6 @@ ___
|
|
}
|
|
$code.=<<___;
|
|
.string "AES for s390x, CRYPTOGAMS by <appro\@openssl.org>"
|
|
-.comm OPENSSL_s390xcap_P,80,8
|
|
___
|
|
|
|
$code =~ s/\`([^\`]*)\`/eval $1/gem;
|
|
diff -up openssl-1.0.2k/crypto/aes/Makefile.s390x-update openssl-1.0.2k/crypto/aes/Makefile
|
|
--- openssl-1.0.2k/crypto/aes/Makefile.s390x-update 2017-01-26 14:22:03.000000000 +0100
|
|
+++ openssl-1.0.2k/crypto/aes/Makefile 2018-06-18 13:27:37.108817179 +0200
|
|
@@ -92,6 +92,8 @@ aesv8-armx.S: asm/aesv8-armx.pl
|
|
$(PERL) asm/aesv8-armx.pl $(PERLASM_SCHEME) $@
|
|
aesv8-armx.o: aesv8-armx.S
|
|
|
|
+aes-s390x.o: aes-s390x.S
|
|
+
|
|
# GNU make "catch all"
|
|
aes-%.S: asm/aes-%.pl; $(PERL) $< $(PERLASM_SCHEME) > $@
|
|
aes-armv4.o: aes-armv4.S
|
|
diff -up openssl-1.0.2k/crypto/evp/e_aes.c.s390x-update openssl-1.0.2k/crypto/evp/e_aes.c
|
|
--- openssl-1.0.2k/crypto/evp/e_aes.c.s390x-update 2018-06-18 12:20:47.104992361 +0200
|
|
+++ openssl-1.0.2k/crypto/evp/e_aes.c 2018-06-18 13:28:07.033543735 +0200
|
|
@@ -854,6 +854,723 @@ static const EVP_CIPHER aes_##keylen##_#
|
|
const EVP_CIPHER *EVP_aes_##keylen##_##mode(void) \
|
|
{ return SPARC_AES_CAPABLE?&aes_t4_##keylen##_##mode:&aes_##keylen##_##mode; }
|
|
|
|
+#elif defined(OPENSSL_CPUID_OBJ) && defined(__s390__)
|
|
+/*
|
|
+ * IBM S390X support
|
|
+ */
|
|
+# include "s390x_arch.h"
|
|
+
|
|
+typedef struct {
|
|
+ union {
|
|
+ double align;
|
|
+ /*-
|
|
+ * KMA-GCM-AES parameter block
|
|
+ * (see z/Architecture Principles of Operation SA22-7832-11)
|
|
+ */
|
|
+ struct {
|
|
+ unsigned char reserved[12];
|
|
+ union {
|
|
+ unsigned int w;
|
|
+ unsigned char b[4];
|
|
+ } cv;
|
|
+ union {
|
|
+ unsigned long long g[2];
|
|
+ unsigned char b[16];
|
|
+ } t;
|
|
+ unsigned char h[16];
|
|
+ unsigned long long taadl;
|
|
+ unsigned long long tpcl;
|
|
+ union {
|
|
+ unsigned long long g[2];
|
|
+ unsigned int w[4];
|
|
+ } j0;
|
|
+ unsigned char k[32];
|
|
+ } param;
|
|
+ } kma;
|
|
+ unsigned int fc;
|
|
+ int key_set;
|
|
+
|
|
+ unsigned char *iv;
|
|
+ int ivlen;
|
|
+ int iv_set;
|
|
+ int iv_gen;
|
|
+
|
|
+ int taglen;
|
|
+
|
|
+ unsigned char ares[16];
|
|
+ unsigned char mres[16];
|
|
+ unsigned char kres[16];
|
|
+ int areslen;
|
|
+ int mreslen;
|
|
+ int kreslen;
|
|
+
|
|
+ int tls_aad_len;
|
|
+} S390X_AES_GCM_CTX;
|
|
+
|
|
+# define S390X_aes_128_CAPABLE ((OPENSSL_s390xcap_P.km[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_128)) &&\
|
|
+ (OPENSSL_s390xcap_P.kmc[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_128)))
|
|
+# define S390X_aes_192_CAPABLE ((OPENSSL_s390xcap_P.km[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_192)) &&\
|
|
+ (OPENSSL_s390xcap_P.kmc[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_192)))
|
|
+# define S390X_aes_256_CAPABLE ((OPENSSL_s390xcap_P.km[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_256)) &&\
|
|
+ (OPENSSL_s390xcap_P.kmc[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_256)))
|
|
+
|
|
+# define s390x_aes_init_key aes_init_key
|
|
+static int s390x_aes_init_key(EVP_CIPHER_CTX *ctx, const unsigned char *key,
|
|
+ const unsigned char *iv, int enc);
|
|
+
|
|
+# define S390X_aes_128_cbc_CAPABLE 1 /* checked by callee */
|
|
+# define S390X_aes_192_cbc_CAPABLE 1
|
|
+# define S390X_aes_256_cbc_CAPABLE 1
|
|
+
|
|
+# define s390x_aes_cbc_cipher aes_cbc_cipher
|
|
+static int s390x_aes_cbc_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+
|
|
+# define S390X_aes_128_ecb_CAPABLE 0
|
|
+# define S390X_aes_192_ecb_CAPABLE 0
|
|
+# define S390X_aes_256_ecb_CAPABLE 0
|
|
+
|
|
+# define s390x_aes_ecb_cipher aes_ecb_cipher
|
|
+static int s390x_aes_ecb_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+
|
|
+# define S390X_aes_128_ofb_CAPABLE 0
|
|
+# define S390X_aes_192_ofb_CAPABLE 0
|
|
+# define S390X_aes_256_ofb_CAPABLE 0
|
|
+
|
|
+# define s390x_aes_ofb_cipher aes_ofb_cipher
|
|
+static int s390x_aes_ofb_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+
|
|
+# define S390X_aes_128_cfb_CAPABLE 0
|
|
+# define S390X_aes_192_cfb_CAPABLE 0
|
|
+# define S390X_aes_256_cfb_CAPABLE 0
|
|
+
|
|
+# define s390x_aes_cfb_cipher aes_cfb_cipher
|
|
+static int s390x_aes_cfb_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+
|
|
+# define S390X_aes_128_cfb8_CAPABLE 0
|
|
+# define S390X_aes_192_cfb8_CAPABLE 0
|
|
+# define S390X_aes_256_cfb8_CAPABLE 0
|
|
+
|
|
+# define s390x_aes_cfb8_cipher aes_cfb8_cipher
|
|
+static int s390x_aes_cfb8_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+
|
|
+# define S390X_aes_128_cfb1_CAPABLE 0
|
|
+# define S390X_aes_192_cfb1_CAPABLE 0
|
|
+# define S390X_aes_256_cfb1_CAPABLE 0
|
|
+
|
|
+# define s390x_aes_cfb1_cipher aes_cfb1_cipher
|
|
+static int s390x_aes_cfb1_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+
|
|
+# define S390X_aes_128_ctr_CAPABLE 1 /* checked by callee */
|
|
+# define S390X_aes_192_ctr_CAPABLE 1
|
|
+# define S390X_aes_256_ctr_CAPABLE 1
|
|
+
|
|
+# define s390x_aes_ctr_cipher aes_ctr_cipher
|
|
+static int s390x_aes_ctr_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+
|
|
+# define S390X_aes_128_gcm_CAPABLE (S390X_aes_128_CAPABLE && \
|
|
+ (OPENSSL_s390xcap_P.kma[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_128)))
|
|
+# define S390X_aes_192_gcm_CAPABLE (S390X_aes_192_CAPABLE && \
|
|
+ (OPENSSL_s390xcap_P.kma[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_192)))
|
|
+# define S390X_aes_256_gcm_CAPABLE (S390X_aes_256_CAPABLE && \
|
|
+ (OPENSSL_s390xcap_P.kma[0] & \
|
|
+ S390X_CAPBIT(S390X_AES_256)))
|
|
+
|
|
+/* iv + padding length for iv lenghts != 12 */
|
|
+# define S390X_gcm_ivpadlen(i) ((((i) + 15) >> 4 << 4) + 16)
|
|
+
|
|
+static int s390x_aes_gcm_aad(S390X_AES_GCM_CTX *ctx, const unsigned char *aad,
|
|
+ size_t len)
|
|
+{
|
|
+ unsigned long long alen;
|
|
+ int n, rem;
|
|
+
|
|
+ if (ctx->kma.param.tpcl)
|
|
+ return -2;
|
|
+
|
|
+ alen = ctx->kma.param.taadl + len;
|
|
+ if (alen > (U64(1) << 61) || (sizeof(len) == 8 && alen < len))
|
|
+ return -1;
|
|
+ ctx->kma.param.taadl = alen;
|
|
+
|
|
+ n = ctx->areslen;
|
|
+ if (n) {
|
|
+ while (n && len) {
|
|
+ ctx->ares[n] = *aad;
|
|
+ n = (n + 1) & 0xf;
|
|
+ ++aad;
|
|
+ --len;
|
|
+ }
|
|
+ /* ctx->ares contains a complete block if offset has wrapped around */
|
|
+ if (!n) {
|
|
+ s390x_kma(ctx->ares, 16, NULL, 0, NULL, ctx->fc, &ctx->kma.param);
|
|
+ ctx->fc |= S390X_KMA_HS;
|
|
+ }
|
|
+ ctx->areslen = n;
|
|
+ }
|
|
+
|
|
+ rem = len & 0xf;
|
|
+
|
|
+ len &= ~0xf;
|
|
+ if (len) {
|
|
+ s390x_kma(aad, len, NULL, 0, NULL, ctx->fc, &ctx->kma.param);
|
|
+ aad += len;
|
|
+ ctx->fc |= S390X_KMA_HS;
|
|
+ }
|
|
+
|
|
+ if (rem) {
|
|
+ ctx->areslen = rem;
|
|
+
|
|
+ do {
|
|
+ --rem;
|
|
+ ctx->ares[rem] = aad[rem];
|
|
+ } while (rem);
|
|
+ }
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int s390x_aes_gcm(S390X_AES_GCM_CTX *ctx, const unsigned char *in,
|
|
+ unsigned char *out, size_t len)
|
|
+{
|
|
+ const unsigned char *inptr;
|
|
+ unsigned long long mlen;
|
|
+ union {
|
|
+ unsigned int w[4];
|
|
+ unsigned char b[16];
|
|
+ } buf;
|
|
+ size_t inlen;
|
|
+ int n, rem, i;
|
|
+
|
|
+ mlen = ctx->kma.param.tpcl + len;
|
|
+ if (mlen > ((U64(1) << 36) - 32) || (sizeof(len) == 8 && mlen < len))
|
|
+ return -1;
|
|
+ ctx->kma.param.tpcl = mlen;
|
|
+
|
|
+ n = ctx->mreslen;
|
|
+ if (n) {
|
|
+ inptr = in;
|
|
+ inlen = len;
|
|
+ while (n && inlen) {
|
|
+ ctx->mres[n] = *inptr;
|
|
+ n = (n + 1) & 0xf;
|
|
+ ++inptr;
|
|
+ --inlen;
|
|
+ }
|
|
+ /* ctx->mres contains a complete block if offset has wrapped around */
|
|
+ if (!n) {
|
|
+ s390x_kma(ctx->ares, ctx->areslen, ctx->mres, 16, buf.b,
|
|
+ ctx->fc | S390X_KMA_LAAD, &ctx->kma.param);
|
|
+ ctx->fc |= S390X_KMA_HS;
|
|
+ ctx->areslen = 0;
|
|
+
|
|
+ /* previous call already encrypted/decrypted its remainder,
|
|
+ * see comment below */
|
|
+ n = ctx->mreslen;
|
|
+ while (n) {
|
|
+ *out = buf.b[n];
|
|
+ n = (n + 1) & 0xf;
|
|
+ ++out;
|
|
+ ++in;
|
|
+ --len;
|
|
+ }
|
|
+ ctx->mreslen = 0;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ rem = len & 0xf;
|
|
+
|
|
+ len &= ~0xf;
|
|
+ if (len) {
|
|
+ s390x_kma(ctx->ares, ctx->areslen, in, len, out,
|
|
+ ctx->fc | S390X_KMA_LAAD, &ctx->kma.param);
|
|
+ in += len;
|
|
+ out += len;
|
|
+ ctx->fc |= S390X_KMA_HS;
|
|
+ ctx->areslen = 0;
|
|
+ }
|
|
+
|
|
+ /*-
|
|
+ * If there is a remainder, it has to be saved such that it can be
|
|
+ * processed by kma later. However, we also have to do the for-now
|
|
+ * unauthenticated encryption/decryption part here and now...
|
|
+ */
|
|
+ if (rem) {
|
|
+ if (!ctx->mreslen) {
|
|
+ buf.w[0] = ctx->kma.param.j0.w[0];
|
|
+ buf.w[1] = ctx->kma.param.j0.w[1];
|
|
+ buf.w[2] = ctx->kma.param.j0.w[2];
|
|
+ buf.w[3] = ctx->kma.param.cv.w + 1;
|
|
+ s390x_km(buf.b, 16, ctx->kres, ctx->fc & 0x1f, &ctx->kma.param.k);
|
|
+ }
|
|
+
|
|
+ n = ctx->mreslen;
|
|
+ for (i = 0; i < rem; i++) {
|
|
+ ctx->mres[n + i] = in[i];
|
|
+ out[i] = in[i] ^ ctx->kres[n + i];
|
|
+ }
|
|
+
|
|
+ ctx->mreslen += rem;
|
|
+ }
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static void s390x_aes_gcm_setiv(S390X_AES_GCM_CTX *ctx,
|
|
+ const unsigned char *iv)
|
|
+{
|
|
+ ctx->kma.param.t.g[0] = 0;
|
|
+ ctx->kma.param.t.g[1] = 0;
|
|
+ ctx->kma.param.tpcl = 0;
|
|
+ ctx->kma.param.taadl = 0;
|
|
+ ctx->mreslen = 0;
|
|
+ ctx->areslen = 0;
|
|
+ ctx->kreslen = 0;
|
|
+
|
|
+ if (ctx->ivlen == 12) {
|
|
+ memcpy(&ctx->kma.param.j0, iv, ctx->ivlen);
|
|
+ ctx->kma.param.j0.w[3] = 1;
|
|
+ ctx->kma.param.cv.w = 1;
|
|
+ } else {
|
|
+ /* ctx->iv has the right size and is already padded. */
|
|
+ memcpy(ctx->iv, iv, ctx->ivlen);
|
|
+ s390x_kma(ctx->iv, S390X_gcm_ivpadlen(ctx->ivlen), NULL, 0, NULL,
|
|
+ ctx->fc, &ctx->kma.param);
|
|
+ ctx->fc |= S390X_KMA_HS;
|
|
+
|
|
+ ctx->kma.param.j0.g[0] = ctx->kma.param.t.g[0];
|
|
+ ctx->kma.param.j0.g[1] = ctx->kma.param.t.g[1];
|
|
+ ctx->kma.param.cv.w = ctx->kma.param.j0.w[3];
|
|
+ ctx->kma.param.t.g[0] = 0;
|
|
+ ctx->kma.param.t.g[1] = 0;
|
|
+ }
|
|
+}
|
|
+
|
|
+static int s390x_aes_gcm_ctrl(EVP_CIPHER_CTX *c, int type, int arg, void *ptr)
|
|
+{
|
|
+ S390X_AES_GCM_CTX *gctx = EVP_C_DATA(S390X_AES_GCM_CTX, c);
|
|
+ S390X_AES_GCM_CTX *gctx_out;
|
|
+ EVP_CIPHER_CTX *out;
|
|
+ unsigned char *buf, *iv;
|
|
+ int ivlen, enc, len;
|
|
+
|
|
+ switch (type) {
|
|
+ case EVP_CTRL_INIT:
|
|
+ ivlen = c->cipher->iv_len;;
|
|
+ iv = c->iv;
|
|
+ gctx->key_set = 0;
|
|
+ gctx->iv_set = 0;
|
|
+ gctx->ivlen = ivlen;
|
|
+ gctx->iv = iv;
|
|
+ gctx->taglen = -1;
|
|
+ gctx->iv_gen = 0;
|
|
+ gctx->tls_aad_len = -1;
|
|
+ return 1;
|
|
+
|
|
+ case EVP_CTRL_GCM_SET_IVLEN:
|
|
+ if (arg <= 0)
|
|
+ return 0;
|
|
+
|
|
+ if (arg != 12) {
|
|
+ iv = c->iv;
|
|
+ len = S390X_gcm_ivpadlen(arg);
|
|
+
|
|
+ /* Allocate memory for iv if needed. */
|
|
+ if (gctx->ivlen == 12 || len > S390X_gcm_ivpadlen(gctx->ivlen)) {
|
|
+ if (gctx->iv != iv)
|
|
+ OPENSSL_free(gctx->iv);
|
|
+
|
|
+ gctx->iv = OPENSSL_malloc(len);
|
|
+ if (gctx->iv == NULL)
|
|
+ return 0;
|
|
+ }
|
|
+ /* Add padding. */
|
|
+ memset(gctx->iv + arg, 0, len - arg - 8);
|
|
+ *((unsigned long long *)(gctx->iv + len - 8)) = arg << 3;
|
|
+ }
|
|
+ gctx->ivlen = arg;
|
|
+ return 1;
|
|
+
|
|
+ case EVP_CTRL_GCM_SET_TAG:
|
|
+ buf = c->buf;
|
|
+ enc = c->encrypt;
|
|
+ if (arg <= 0 || arg > 16 || enc)
|
|
+ return 0;
|
|
+
|
|
+ memcpy(buf, ptr, arg);
|
|
+ gctx->taglen = arg;
|
|
+ return 1;
|
|
+
|
|
+ case EVP_CTRL_GCM_GET_TAG:
|
|
+ enc = c->encrypt;
|
|
+ if (arg <= 0 || arg > 16 || !enc || gctx->taglen < 0)
|
|
+ return 0;
|
|
+
|
|
+ memcpy(ptr, gctx->kma.param.t.b, arg);
|
|
+ return 1;
|
|
+
|
|
+ case EVP_CTRL_GCM_SET_IV_FIXED:
|
|
+ /* Special case: -1 length restores whole iv */
|
|
+ if (arg == -1) {
|
|
+ memcpy(gctx->iv, ptr, gctx->ivlen);
|
|
+ gctx->iv_gen = 1;
|
|
+ return 1;
|
|
+ }
|
|
+ /*
|
|
+ * Fixed field must be at least 4 bytes and invocation field at least
|
|
+ * 8.
|
|
+ */
|
|
+ if ((arg < 4) || (gctx->ivlen - arg) < 8)
|
|
+ return 0;
|
|
+
|
|
+ if (arg)
|
|
+ memcpy(gctx->iv, ptr, arg);
|
|
+
|
|
+ enc = c->encrypt;
|
|
+ if (enc && RAND_bytes(gctx->iv + arg, gctx->ivlen - arg) <= 0)
|
|
+ return 0;
|
|
+
|
|
+ gctx->iv_gen = 1;
|
|
+ return 1;
|
|
+
|
|
+ case EVP_CTRL_GCM_IV_GEN:
|
|
+ if (gctx->iv_gen == 0 || gctx->key_set == 0)
|
|
+ return 0;
|
|
+
|
|
+ s390x_aes_gcm_setiv(gctx, gctx->iv);
|
|
+
|
|
+ if (arg <= 0 || arg > gctx->ivlen)
|
|
+ arg = gctx->ivlen;
|
|
+
|
|
+ memcpy(ptr, gctx->iv + gctx->ivlen - arg, arg);
|
|
+ /*
|
|
+ * Invocation field will be at least 8 bytes in size and so no need
|
|
+ * to check wrap around or increment more than last 8 bytes.
|
|
+ */
|
|
+ (*(unsigned long long *)(gctx->iv + gctx->ivlen - 8))++;
|
|
+ gctx->iv_set = 1;
|
|
+ return 1;
|
|
+
|
|
+ case EVP_CTRL_GCM_SET_IV_INV:
|
|
+ enc = c->encrypt;
|
|
+ if (gctx->iv_gen == 0 || gctx->key_set == 0 || enc)
|
|
+ return 0;
|
|
+
|
|
+ memcpy(gctx->iv + gctx->ivlen - arg, ptr, arg);
|
|
+ s390x_aes_gcm_setiv(gctx, gctx->iv);
|
|
+ gctx->iv_set = 1;
|
|
+ return 1;
|
|
+
|
|
+ case EVP_CTRL_AEAD_TLS1_AAD:
|
|
+ /* Save the aad for later use. */
|
|
+ if (arg != EVP_AEAD_TLS1_AAD_LEN)
|
|
+ return 0;
|
|
+
|
|
+ buf = c->buf;
|
|
+ memcpy(buf, ptr, arg);
|
|
+ gctx->tls_aad_len = arg;
|
|
+
|
|
+ len = buf[arg - 2] << 8 | buf[arg - 1];
|
|
+ /* Correct length for explicit iv. */
|
|
+ if (len < EVP_GCM_TLS_EXPLICIT_IV_LEN)
|
|
+ return 0;
|
|
+ len -= EVP_GCM_TLS_EXPLICIT_IV_LEN;
|
|
+
|
|
+ /* If decrypting correct for tag too. */
|
|
+ enc = c->encrypt;
|
|
+ if (!enc) {
|
|
+ if (len < EVP_GCM_TLS_TAG_LEN)
|
|
+ return 0;
|
|
+ len -= EVP_GCM_TLS_TAG_LEN;
|
|
+ }
|
|
+ buf[arg - 2] = len >> 8;
|
|
+ buf[arg - 1] = len & 0xff;
|
|
+ /* Extra padding: tag appended to record. */
|
|
+ return EVP_GCM_TLS_TAG_LEN;
|
|
+
|
|
+ case EVP_CTRL_COPY:
|
|
+ out = ptr;
|
|
+ gctx_out = EVP_C_DATA(S390X_AES_GCM_CTX, out);
|
|
+ iv = c->iv;
|
|
+
|
|
+ if (gctx->iv == iv) {
|
|
+ gctx_out->iv = out->iv;
|
|
+ } else {
|
|
+ len = S390X_gcm_ivpadlen(gctx->ivlen);
|
|
+
|
|
+ gctx_out->iv = OPENSSL_malloc(len);
|
|
+ if (gctx_out->iv == NULL)
|
|
+ return 0;
|
|
+
|
|
+ memcpy(gctx_out->iv, gctx->iv, len);
|
|
+ }
|
|
+ return 1;
|
|
+
|
|
+ default:
|
|
+ return -1;
|
|
+ }
|
|
+}
|
|
+
|
|
+static int s390x_aes_gcm_init_key(EVP_CIPHER_CTX *ctx,
|
|
+ const unsigned char *key,
|
|
+ const unsigned char *iv, int enc)
|
|
+{
|
|
+ S390X_AES_GCM_CTX *gctx = EVP_C_DATA(S390X_AES_GCM_CTX, ctx);
|
|
+ int keylen;
|
|
+
|
|
+ if (iv == NULL && key == NULL)
|
|
+ return 1;
|
|
+
|
|
+ if (key != NULL) {
|
|
+ keylen = EVP_CIPHER_CTX_key_length(ctx);
|
|
+ memcpy(&gctx->kma.param.k, key, keylen);
|
|
+
|
|
+ /* Convert key size to function code. */
|
|
+ gctx->fc = S390X_AES_128 + (((keylen << 3) - 128) >> 6);
|
|
+ if (!enc)
|
|
+ gctx->fc |= S390X_DECRYPT;
|
|
+
|
|
+ if (iv == NULL && gctx->iv_set)
|
|
+ iv = gctx->iv;
|
|
+
|
|
+ if (iv != NULL) {
|
|
+ s390x_aes_gcm_setiv(gctx, iv);
|
|
+ gctx->iv_set = 1;
|
|
+ }
|
|
+ gctx->key_set = 1;
|
|
+ } else {
|
|
+ if (gctx->key_set)
|
|
+ s390x_aes_gcm_setiv(gctx, iv);
|
|
+ else
|
|
+ memcpy(gctx->iv, iv, gctx->ivlen);
|
|
+
|
|
+ gctx->iv_set = 1;
|
|
+ gctx->iv_gen = 0;
|
|
+ }
|
|
+ return 1;
|
|
+}
|
|
+
|
|
+static int s390x_aes_gcm_tls_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len)
|
|
+{
|
|
+ S390X_AES_GCM_CTX *gctx = EVP_C_DATA(S390X_AES_GCM_CTX, ctx);
|
|
+ const unsigned char *buf = ctx->buf;
|
|
+ const int enc = ctx->encrypt;
|
|
+ int rv = -1;
|
|
+
|
|
+ if (out != in || len < (EVP_GCM_TLS_EXPLICIT_IV_LEN + EVP_GCM_TLS_TAG_LEN))
|
|
+ return -1;
|
|
+
|
|
+ if (EVP_CIPHER_CTX_ctrl(ctx, enc ? EVP_CTRL_GCM_IV_GEN
|
|
+ : EVP_CTRL_GCM_SET_IV_INV,
|
|
+ EVP_GCM_TLS_EXPLICIT_IV_LEN, out) <= 0)
|
|
+ goto err;
|
|
+
|
|
+ in += EVP_GCM_TLS_EXPLICIT_IV_LEN;
|
|
+ out += EVP_GCM_TLS_EXPLICIT_IV_LEN;
|
|
+ len -= EVP_GCM_TLS_EXPLICIT_IV_LEN + EVP_GCM_TLS_TAG_LEN;
|
|
+
|
|
+ gctx->kma.param.taadl = gctx->tls_aad_len << 3;
|
|
+ gctx->kma.param.tpcl = len << 3;
|
|
+ s390x_kma(buf, gctx->tls_aad_len, in, len, out,
|
|
+ gctx->fc | S390X_KMA_LAAD | S390X_KMA_LPC, &gctx->kma.param);
|
|
+
|
|
+ if (enc) {
|
|
+ memcpy(out + len, gctx->kma.param.t.b, EVP_GCM_TLS_TAG_LEN);
|
|
+ rv = len + EVP_GCM_TLS_EXPLICIT_IV_LEN + EVP_GCM_TLS_TAG_LEN;
|
|
+ } else {
|
|
+ if (CRYPTO_memcmp(gctx->kma.param.t.b, in + len,
|
|
+ EVP_GCM_TLS_TAG_LEN)) {
|
|
+ OPENSSL_cleanse(out, len);
|
|
+ goto err;
|
|
+ }
|
|
+ rv = len;
|
|
+ }
|
|
+err:
|
|
+ gctx->iv_set = 0;
|
|
+ gctx->tls_aad_len = -1;
|
|
+ return rv;
|
|
+}
|
|
+
|
|
+static int s390x_aes_gcm_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len)
|
|
+{
|
|
+ S390X_AES_GCM_CTX *gctx = EVP_C_DATA(S390X_AES_GCM_CTX, ctx);
|
|
+ unsigned char *buf, tmp[16];
|
|
+ int enc;
|
|
+
|
|
+ if (!gctx->key_set)
|
|
+ return -1;
|
|
+
|
|
+ if (gctx->tls_aad_len >= 0)
|
|
+ return s390x_aes_gcm_tls_cipher(ctx, out, in, len);
|
|
+
|
|
+ if (!gctx->iv_set)
|
|
+ return -1;
|
|
+
|
|
+ if (in != NULL) {
|
|
+ if (out == NULL) {
|
|
+ if (s390x_aes_gcm_aad(gctx, in, len))
|
|
+ return -1;
|
|
+ } else {
|
|
+ if (s390x_aes_gcm(gctx, in, out, len))
|
|
+ return -1;
|
|
+ }
|
|
+ return len;
|
|
+ } else {
|
|
+ gctx->kma.param.taadl <<= 3;
|
|
+ gctx->kma.param.tpcl <<= 3;
|
|
+ s390x_kma(gctx->ares, gctx->areslen, gctx->mres, gctx->mreslen, tmp,
|
|
+ gctx->fc | S390X_KMA_LAAD | S390X_KMA_LPC, &gctx->kma.param);
|
|
+ /* recall that we already did en-/decrypt gctx->mres
|
|
+ * and returned it to caller... */
|
|
+ OPENSSL_cleanse(tmp, gctx->mreslen);
|
|
+ gctx->iv_set = 0;
|
|
+
|
|
+ enc = ctx->encrypt;
|
|
+ if (enc) {
|
|
+ gctx->taglen = 16;
|
|
+ } else {
|
|
+ if (gctx->taglen < 0)
|
|
+ return -1;
|
|
+
|
|
+ buf = ctx->buf;
|
|
+ if (CRYPTO_memcmp(buf, gctx->kma.param.t.b, gctx->taglen))
|
|
+ return -1;
|
|
+ }
|
|
+ return 0;
|
|
+ }
|
|
+}
|
|
+
|
|
+static int s390x_aes_gcm_cleanup(EVP_CIPHER_CTX *c)
|
|
+{
|
|
+ S390X_AES_GCM_CTX *gctx = EVP_C_DATA(S390X_AES_GCM_CTX, c);
|
|
+ const unsigned char *iv;
|
|
+
|
|
+ if (gctx == NULL)
|
|
+ return 0;
|
|
+
|
|
+ iv = c->iv;
|
|
+ if (iv != gctx->iv)
|
|
+ OPENSSL_free(gctx->iv);
|
|
+
|
|
+ OPENSSL_cleanse(gctx, sizeof(*gctx));
|
|
+ return 1;
|
|
+}
|
|
+
|
|
+# define S390X_AES_XTS_CTX EVP_AES_XTS_CTX
|
|
+# define S390X_aes_128_xts_CAPABLE 1 /* checked by callee */
|
|
+# define S390X_aes_256_xts_CAPABLE 1
|
|
+
|
|
+# define s390x_aes_xts_init_key aes_xts_init_key
|
|
+static int s390x_aes_xts_init_key(EVP_CIPHER_CTX *ctx,
|
|
+ const unsigned char *key,
|
|
+ const unsigned char *iv, int enc);
|
|
+# define s390x_aes_xts_cipher aes_xts_cipher
|
|
+static int s390x_aes_xts_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+# define s390x_aes_xts_ctrl aes_xts_ctrl
|
|
+static int s390x_aes_xts_ctrl(EVP_CIPHER_CTX *, int type, int arg, void *ptr);
|
|
+# define s390x_aes_xts_cleanup aes_xts_cleanup
|
|
+
|
|
+# define S390X_AES_CCM_CTX EVP_AES_CCM_CTX
|
|
+# define S390X_aes_128_ccm_CAPABLE 0
|
|
+# define S390X_aes_192_ccm_CAPABLE 0
|
|
+# define S390X_aes_256_ccm_CAPABLE 0
|
|
+
|
|
+# define s390x_aes_ccm_init_key aes_ccm_init_key
|
|
+static int s390x_aes_ccm_init_key(EVP_CIPHER_CTX *ctx,
|
|
+ const unsigned char *key,
|
|
+ const unsigned char *iv, int enc);
|
|
+# define s390x_aes_ccm_cipher aes_ccm_cipher
|
|
+static int s390x_aes_ccm_cipher(EVP_CIPHER_CTX *ctx, unsigned char *out,
|
|
+ const unsigned char *in, size_t len);
|
|
+# define s390x_aes_ccm_ctrl aes_ccm_ctrl
|
|
+static int s390x_aes_ccm_ctrl(EVP_CIPHER_CTX *, int type, int arg, void *ptr);
|
|
+# define s390x_aes_ccm_cleanup aes_ccm_cleanup
|
|
+
|
|
+# define BLOCK_CIPHER_generic(nid,keylen,blocksize,ivlen,nmode,mode, \
|
|
+ MODE,flags) \
|
|
+static const EVP_CIPHER s390x_aes_##keylen##_##mode = { \
|
|
+ nid##_##keylen##_##nmode,blocksize, \
|
|
+ keylen / 8, \
|
|
+ ivlen, \
|
|
+ flags | EVP_CIPH_##MODE##_MODE, \
|
|
+ s390x_aes_init_key, \
|
|
+ s390x_aes_##mode##_cipher, \
|
|
+ NULL, \
|
|
+ sizeof(EVP_AES_KEY), \
|
|
+ NULL, \
|
|
+ NULL, \
|
|
+ NULL, \
|
|
+ NULL \
|
|
+}; \
|
|
+static const EVP_CIPHER aes_##keylen##_##mode = { \
|
|
+ nid##_##keylen##_##nmode, \
|
|
+ blocksize, \
|
|
+ keylen / 8, \
|
|
+ ivlen, \
|
|
+ flags | EVP_CIPH_##MODE##_MODE, \
|
|
+ aes_init_key, \
|
|
+ aes_##mode##_cipher, \
|
|
+ NULL, \
|
|
+ sizeof(EVP_AES_KEY), \
|
|
+ NULL,NULL,NULL,NULL \
|
|
+}; \
|
|
+const EVP_CIPHER *EVP_aes_##keylen##_##mode(void) \
|
|
+{ \
|
|
+ return S390X_aes_##keylen##_##mode##_CAPABLE ? \
|
|
+ &s390x_aes_##keylen##_##mode : &aes_##keylen##_##mode; \
|
|
+}
|
|
+
|
|
+# define BLOCK_CIPHER_custom(nid,keylen,blocksize,ivlen,mode,MODE,flags)\
|
|
+static const EVP_CIPHER s390x_aes_##keylen##_##mode = { \
|
|
+ nid##_##keylen##_##mode, \
|
|
+ blocksize, \
|
|
+ (EVP_CIPH_##MODE##_MODE == EVP_CIPH_XTS_MODE ? 2 : 1) * keylen / 8, \
|
|
+ ivlen, \
|
|
+ flags | EVP_CIPH_##MODE##_MODE, \
|
|
+ s390x_aes_##mode##_init_key, \
|
|
+ s390x_aes_##mode##_cipher, \
|
|
+ s390x_aes_##mode##_cleanup, \
|
|
+ sizeof(S390X_AES_##MODE##_CTX), \
|
|
+ NULL, \
|
|
+ NULL, \
|
|
+ s390x_aes_##mode##_ctrl, \
|
|
+ NULL \
|
|
+}; \
|
|
+static const EVP_CIPHER aes_##keylen##_##mode = { \
|
|
+ nid##_##keylen##_##mode,blocksize, \
|
|
+ (EVP_CIPH_##MODE##_MODE == EVP_CIPH_XTS_MODE ? 2 : 1) * keylen / 8, \
|
|
+ ivlen, \
|
|
+ flags | EVP_CIPH_##MODE##_MODE, \
|
|
+ aes_##mode##_init_key, \
|
|
+ aes_##mode##_cipher, \
|
|
+ aes_##mode##_cleanup, \
|
|
+ sizeof(EVP_AES_##MODE##_CTX), \
|
|
+ NULL, \
|
|
+ NULL, \
|
|
+ aes_##mode##_ctrl, \
|
|
+ NULL \
|
|
+}; \
|
|
+const EVP_CIPHER *EVP_aes_##keylen##_##mode(void) \
|
|
+{ \
|
|
+ return S390X_aes_##keylen##_##mode##_CAPABLE ? \
|
|
+ &s390x_aes_##keylen##_##mode : &aes_##keylen##_##mode; \
|
|
+}
|
|
+
|
|
# else
|
|
|
|
# define BLOCK_CIPHER_generic(nid,keylen,blocksize,ivlen,nmode,mode,MODE,flags) \
|
|
diff -up openssl-1.0.2k/crypto/Makefile.s390x-update openssl-1.0.2k/crypto/Makefile
|
|
--- openssl-1.0.2k/crypto/Makefile.s390x-update 2018-06-18 12:20:49.079040659 +0200
|
|
+++ openssl-1.0.2k/crypto/Makefile 2018-06-18 13:27:53.065204592 +0200
|
|
@@ -77,6 +77,7 @@ alphacpuid.s: alphacpuid.pl
|
|
(preproc=$$$$.$@.S; trap "rm $$preproc" INT; \
|
|
$(PERL) alphacpuid.pl > $$preproc && \
|
|
$(CC) -E -P $$preproc > $@ && rm $$preproc)
|
|
+s390xcpuid.S: s390xcpuid.pl; $(PERL) s390xcpuid.pl $(PERLASM_SCHEME) $@
|
|
|
|
testapps:
|
|
[ -z "$(THIS)" ] || ( if echo $(SDIRS) | fgrep ' des '; \
|
|
diff -up openssl-1.0.2k/crypto/modes/asm/ghash-s390x.pl.s390x-update openssl-1.0.2k/crypto/modes/asm/ghash-s390x.pl
|
|
--- openssl-1.0.2k/crypto/modes/asm/ghash-s390x.pl.s390x-update 2018-06-18 12:20:47.170993976 +0200
|
|
+++ openssl-1.0.2k/crypto/modes/asm/ghash-s390x.pl 2018-06-18 13:27:37.110817228 +0200
|
|
@@ -73,6 +73,8 @@ $rem_4bit="%r14";
|
|
$sp="%r15";
|
|
|
|
$code.=<<___;
|
|
+#include "s390x_arch.h"
|
|
+
|
|
.text
|
|
|
|
.globl gcm_gmult_4bit
|
|
@@ -85,12 +87,13 @@ $code.=<<___ if(!$softonly && 0); # hard
|
|
tmhl %r0,0x4000 # check for message-security-assist
|
|
jz .Lsoft_gmult
|
|
lghi %r0,0
|
|
- lg %r1,24(%r1) # load second word of kimd capabilities vector
|
|
+ lg %r1,S390X_KIMD+8(%r1) # load second word of kimd capabilities
|
|
+ # vector
|
|
tmhh %r1,0x4000 # check for function 65
|
|
jz .Lsoft_gmult
|
|
stg %r0,16($sp) # arrange 16 bytes of zero input
|
|
stg %r0,24($sp)
|
|
- lghi %r0,65 # function 65
|
|
+ lghi %r0,S390X_GHASH # function 65
|
|
la %r1,0($Xi) # H lies right after Xi in gcm128_context
|
|
la $inp,16($sp)
|
|
lghi $len,16
|
|
@@ -119,16 +122,11 @@ gcm_ghash_4bit:
|
|
___
|
|
$code.=<<___ if(!$softonly);
|
|
larl %r1,OPENSSL_s390xcap_P
|
|
- lg %r0,0(%r1)
|
|
- tmhl %r0,0x4000 # check for message-security-assist
|
|
- jz .Lsoft_ghash
|
|
- lghi %r0,0
|
|
- la %r1,16($sp)
|
|
- .long 0xb93e0004 # kimd %r0,%r4
|
|
- lg %r1,24($sp)
|
|
- tmhh %r1,0x4000 # check for function 65
|
|
+ lg %r0,S390X_KIMD+8(%r1) # load second word of kimd capabilities
|
|
+ # vector
|
|
+ tmhh %r0,0x4000 # check for function 65
|
|
jz .Lsoft_ghash
|
|
- lghi %r0,65 # function 65
|
|
+ lghi %r0,S390X_GHASH # function 65
|
|
la %r1,0($Xi) # H lies right after Xi in gcm128_context
|
|
.long 0xb93e0004 # kimd %r0,$inp
|
|
brc 1,.-4 # pay attention to "partial completion"
|
|
diff -up openssl-1.0.2k/crypto/modes/Makefile.s390x-update openssl-1.0.2k/crypto/modes/Makefile
|
|
--- openssl-1.0.2k/crypto/modes/Makefile.s390x-update 2018-06-18 12:20:47.020990305 +0200
|
|
+++ openssl-1.0.2k/crypto/modes/Makefile 2018-06-18 13:27:37.110817228 +0200
|
|
@@ -71,6 +71,8 @@ ghash-%.S: asm/ghash-%.pl; $(PERL) $< $(
|
|
ghash-armv4.o: ghash-armv4.S
|
|
ghashv8-armx.o: ghashv8-armx.S
|
|
|
|
+ghash-s390x.o: ghash-s390x.S
|
|
+
|
|
files:
|
|
$(PERL) $(TOP)/util/files.pl Makefile >> $(TOP)/MINFO
|
|
|
|
diff -up openssl-1.0.2k/crypto/sha/asm/sha1-s390x.pl.s390x-update openssl-1.0.2k/crypto/sha/asm/sha1-s390x.pl
|
|
--- openssl-1.0.2k/crypto/sha/asm/sha1-s390x.pl.s390x-update 2018-06-18 12:20:47.174994073 +0200
|
|
+++ openssl-1.0.2k/crypto/sha/asm/sha1-s390x.pl 2018-06-18 13:27:37.112817276 +0200
|
|
@@ -152,6 +152,8 @@ ___
|
|
}
|
|
|
|
$code.=<<___;
|
|
+#include "s390x_arch.h"
|
|
+
|
|
.text
|
|
.align 64
|
|
.type Ktable,\@object
|
|
@@ -164,10 +166,7 @@ sha1_block_data_order:
|
|
___
|
|
$code.=<<___ if ($kimdfunc);
|
|
larl %r1,OPENSSL_s390xcap_P
|
|
- lg %r0,0(%r1)
|
|
- tmhl %r0,0x4000 # check for message-security assist
|
|
- jz .Lsoftware
|
|
- lg %r0,16(%r1) # check kimd capabilities
|
|
+ lg %r0,S390X_KIMD(%r1) # check kimd capabilities
|
|
tmhh %r0,`0x8000>>$kimdfunc`
|
|
jz .Lsoftware
|
|
lghi %r0,$kimdfunc
|
|
@@ -234,7 +233,6 @@ $code.=<<___;
|
|
br %r14
|
|
.size sha1_block_data_order,.-sha1_block_data_order
|
|
.string "SHA1 block transform for s390x, CRYPTOGAMS by <appro\@openssl.org>"
|
|
-.comm OPENSSL_s390xcap_P,80,8
|
|
___
|
|
|
|
$code =~ s/\`([^\`]*)\`/eval $1/gem;
|
|
diff -up openssl-1.0.2k/crypto/sha/asm/sha512-s390x.pl.s390x-update openssl-1.0.2k/crypto/sha/asm/sha512-s390x.pl
|
|
--- openssl-1.0.2k/crypto/sha/asm/sha512-s390x.pl.s390x-update 2018-06-18 12:20:47.179994196 +0200
|
|
+++ openssl-1.0.2k/crypto/sha/asm/sha512-s390x.pl 2018-06-18 13:27:37.112817276 +0200
|
|
@@ -163,6 +163,8 @@ ___
|
|
}
|
|
|
|
$code.=<<___;
|
|
+#include "s390x_arch.h"
|
|
+
|
|
.text
|
|
.align 64
|
|
.type $Table,\@object
|
|
@@ -237,10 +239,7 @@ $Func:
|
|
___
|
|
$code.=<<___ if ($kimdfunc);
|
|
larl %r1,OPENSSL_s390xcap_P
|
|
- lg %r0,0(%r1)
|
|
- tmhl %r0,0x4000 # check for message-security assist
|
|
- jz .Lsoftware
|
|
- lg %r0,16(%r1) # check kimd capabilities
|
|
+ lg %r0,S390X_KIMD(%r1) # check kimd capabilities
|
|
tmhh %r0,`0x8000>>$kimdfunc`
|
|
jz .Lsoftware
|
|
lghi %r0,$kimdfunc
|
|
@@ -308,7 +307,6 @@ $code.=<<___;
|
|
br %r14
|
|
.size $Func,.-$Func
|
|
.string "SHA${label} block transform for s390x, CRYPTOGAMS by <appro\@openssl.org>"
|
|
-.comm OPENSSL_s390xcap_P,80,8
|
|
___
|
|
|
|
$code =~ s/\`([^\`]*)\`/eval $1/gem;
|
|
diff -up openssl-1.0.2k/crypto/sha/Makefile.s390x-update openssl-1.0.2k/crypto/sha/Makefile
|
|
--- openssl-1.0.2k/crypto/sha/Makefile.s390x-update 2018-06-18 12:20:49.482050519 +0200
|
|
+++ openssl-1.0.2k/crypto/sha/Makefile 2018-06-18 13:27:37.112817276 +0200
|
|
@@ -100,6 +100,10 @@ sha1-armv8.o: sha1-armv8.S
|
|
sha256-armv8.o: sha256-armv8.S
|
|
sha512-armv8.o: sha512-armv8.S
|
|
|
|
+sha1-s390x.o: sha1-s390x.S
|
|
+sha256-s390x.o: sha256-s390x.S
|
|
+sha512-s390x.o: sha512-s390x.S
|
|
+
|
|
files:
|
|
$(PERL) $(TOP)/util/files.pl Makefile >> $(TOP)/MINFO
|
|
|
|
diff -up openssl-1.0.2k/crypto/s390x_arch.h.s390x-update openssl-1.0.2k/crypto/s390x_arch.h
|
|
--- openssl-1.0.2k/crypto/s390x_arch.h.s390x-update 2018-06-18 13:27:37.110817228 +0200
|
|
+++ openssl-1.0.2k/crypto/s390x_arch.h 2018-06-18 13:27:53.066204616 +0200
|
|
@@ -0,0 +1,93 @@
|
|
+/*
|
|
+ * Copyright 2017 The OpenSSL Project Authors. All Rights Reserved.
|
|
+ *
|
|
+ * Licensed under the OpenSSL license (the "License"). You may not use
|
|
+ * this file except in compliance with the License. You can obtain a copy
|
|
+ * in the file LICENSE in the source distribution or at
|
|
+ * https://www.openssl.org/source/license.html
|
|
+ */
|
|
+
|
|
+#ifndef S390X_ARCH_H
|
|
+# define S390X_ARCH_H
|
|
+
|
|
+# ifndef __ASSEMBLER__
|
|
+
|
|
+void s390x_km(const unsigned char *in, size_t len, unsigned char *out,
|
|
+ unsigned int fc, void *param);
|
|
+void s390x_kma(const unsigned char *aad, size_t alen, const unsigned char *in,
|
|
+ size_t len, unsigned char *out, unsigned int fc, void *param);
|
|
+
|
|
+/*
|
|
+ * The field elements of OPENSSL_s390xcap_P are the 64-bit words returned by
|
|
+ * the STFLE instruction followed by the 64-bit word pairs returned by
|
|
+ * instructions' QUERY functions. If STFLE returns fewer data or an instruction
|
|
+ * is not supported, the corresponding field elements are zero.
|
|
+ */
|
|
+struct OPENSSL_s390xcap_st {
|
|
+ unsigned long long stfle[4];
|
|
+ unsigned long long kimd[2];
|
|
+ unsigned long long klmd[2];
|
|
+ unsigned long long km[2];
|
|
+ unsigned long long kmc[2];
|
|
+ unsigned long long kmac[2];
|
|
+ unsigned long long kmctr[2];
|
|
+ unsigned long long kmo[2];
|
|
+ unsigned long long kmf[2];
|
|
+ unsigned long long prno[2];
|
|
+ unsigned long long kma[2];
|
|
+};
|
|
+
|
|
+extern struct OPENSSL_s390xcap_st OPENSSL_s390xcap_P;
|
|
+
|
|
+/* convert facility bit number or function code to bit mask */
|
|
+# define S390X_CAPBIT(i) (1ULL << (63 - (i) % 64))
|
|
+
|
|
+# endif
|
|
+
|
|
+/* OPENSSL_s390xcap_P offsets [bytes] */
|
|
+# define S390X_STFLE 0x00
|
|
+# define S390X_KIMD 0x20
|
|
+# define S390X_KLMD 0x30
|
|
+# define S390X_KM 0x40
|
|
+# define S390X_KMC 0x50
|
|
+# define S390X_KMAC 0x60
|
|
+# define S390X_KMCTR 0x70
|
|
+# define S390X_KMO 0x80
|
|
+# define S390X_KMF 0x90
|
|
+# define S390X_PRNO 0xa0
|
|
+# define S390X_KMA 0xb0
|
|
+
|
|
+/* Facility Bit Numbers */
|
|
+# define S390X_VX 129
|
|
+# define S390X_VXD 134
|
|
+# define S390X_VXE 135
|
|
+
|
|
+/* Function Codes */
|
|
+
|
|
+/* all instructions */
|
|
+# define S390X_QUERY 0
|
|
+
|
|
+/* kimd/klmd */
|
|
+# define S390X_SHA3_224 32
|
|
+# define S390X_SHA3_256 33
|
|
+# define S390X_SHA3_384 34
|
|
+# define S390X_SHA3_512 35
|
|
+# define S390X_SHAKE_128 36
|
|
+# define S390X_SHAKE_256 37
|
|
+# define S390X_GHASH 65
|
|
+
|
|
+/* km/kmc/kmac/kmctr/kmo/kmf/kma */
|
|
+# define S390X_AES_128 18
|
|
+# define S390X_AES_192 19
|
|
+# define S390X_AES_256 20
|
|
+
|
|
+/* prno */
|
|
+# define S390X_TRNG 114
|
|
+
|
|
+/* Register 0 Flags */
|
|
+# define S390X_DECRYPT 0x80
|
|
+# define S390X_KMA_LPC 0x100
|
|
+# define S390X_KMA_LAAD 0x200
|
|
+# define S390X_KMA_HS 0x400
|
|
+
|
|
+#endif
|
|
diff -up openssl-1.0.2k/crypto/s390xcap.c.s390x-update openssl-1.0.2k/crypto/s390xcap.c
|
|
--- openssl-1.0.2k/crypto/s390xcap.c.s390x-update 2017-01-26 14:22:03.000000000 +0100
|
|
+++ openssl-1.0.2k/crypto/s390xcap.c 2018-06-18 13:27:37.111817252 +0200
|
|
@@ -4,8 +4,7 @@
|
|
#include <setjmp.h>
|
|
#include <signal.h>
|
|
#include "cryptlib.h"
|
|
-
|
|
-extern unsigned long OPENSSL_s390xcap_P[];
|
|
+#include "s390x_arch.h"
|
|
|
|
static sigjmp_buf ill_jmp;
|
|
static void ill_handler(int sig)
|
|
@@ -13,30 +12,48 @@ static void ill_handler(int sig)
|
|
siglongjmp(ill_jmp, sig);
|
|
}
|
|
|
|
-unsigned long OPENSSL_s390x_facilities(void);
|
|
+void OPENSSL_s390x_facilities(void);
|
|
+void OPENSSL_vx_probe(void);
|
|
+
|
|
+struct OPENSSL_s390xcap_st OPENSSL_s390xcap_P;
|
|
|
|
void OPENSSL_cpuid_setup(void)
|
|
{
|
|
sigset_t oset;
|
|
struct sigaction ill_act, oact;
|
|
|
|
- if (OPENSSL_s390xcap_P[0])
|
|
+ if (OPENSSL_s390xcap_P.stfle[0])
|
|
return;
|
|
|
|
- OPENSSL_s390xcap_P[0] = 1UL << (8 * sizeof(unsigned long) - 1);
|
|
+ /* set a bit that will not be tested later */
|
|
+ OPENSSL_s390xcap_P.stfle[0] |= S390X_CAPBIT(0);
|
|
|
|
memset(&ill_act, 0, sizeof(ill_act));
|
|
ill_act.sa_handler = ill_handler;
|
|
sigfillset(&ill_act.sa_mask);
|
|
sigdelset(&ill_act.sa_mask, SIGILL);
|
|
+ sigdelset(&ill_act.sa_mask, SIGFPE);
|
|
sigdelset(&ill_act.sa_mask, SIGTRAP);
|
|
sigprocmask(SIG_SETMASK, &ill_act.sa_mask, &oset);
|
|
sigaction(SIGILL, &ill_act, &oact);
|
|
+ sigaction(SIGFPE, &ill_act, &oact);
|
|
|
|
/* protection against missing store-facility-list-extended */
|
|
if (sigsetjmp(ill_jmp, 1) == 0)
|
|
OPENSSL_s390x_facilities();
|
|
|
|
+ /* protection against disabled vector facility */
|
|
+ if ((OPENSSL_s390xcap_P.stfle[2] & S390X_CAPBIT(S390X_VX))
|
|
+ && (sigsetjmp(ill_jmp, 1) == 0)) {
|
|
+ OPENSSL_vx_probe();
|
|
+ } else {
|
|
+ OPENSSL_s390xcap_P.stfle[2] &= ~(S390X_CAPBIT(S390X_VX)
|
|
+ | S390X_CAPBIT(S390X_VXD)
|
|
+ | S390X_CAPBIT(S390X_VXE));
|
|
+ }
|
|
+
|
|
+ sigaction(SIGFPE, &oact, NULL);
|
|
+
|
|
sigaction(SIGILL, &oact, NULL);
|
|
sigprocmask(SIG_SETMASK, &oset, NULL);
|
|
}
|
|
diff -up openssl-1.0.2k/crypto/s390xcpuid.pl.s390x-update openssl-1.0.2k/crypto/s390xcpuid.pl
|
|
--- openssl-1.0.2k/crypto/s390xcpuid.pl.s390x-update 2018-06-18 13:27:53.067204641 +0200
|
|
+++ openssl-1.0.2k/crypto/s390xcpuid.pl 2018-06-18 13:27:53.067204641 +0200
|
|
@@ -0,0 +1,259 @@
|
|
+#! /usr/bin/env perl
|
|
+
|
|
+$flavour = shift;
|
|
+
|
|
+if ($flavour =~ /3[12]/) {
|
|
+ $SIZE_T=4;
|
|
+ $g="";
|
|
+} else {
|
|
+ $SIZE_T=8;
|
|
+ $g="g";
|
|
+}
|
|
+
|
|
+while (($output=shift) && ($output!~/\w[\w\-]*\.\w+$/)) {}
|
|
+open STDOUT,">$output";
|
|
+
|
|
+$ra="%r14";
|
|
+$sp="%r15";
|
|
+$stdframe=16*$SIZE_T+4*8;
|
|
+
|
|
+$code=<<___;
|
|
+#include "s390x_arch.h"
|
|
+
|
|
+.text
|
|
+
|
|
+.globl OPENSSL_s390x_facilities
|
|
+.type OPENSSL_s390x_facilities,\@function
|
|
+.align 16
|
|
+OPENSSL_s390x_facilities:
|
|
+ lghi %r0,0
|
|
+ larl %r4,OPENSSL_s390xcap_P
|
|
+
|
|
+ stg %r0,S390X_STFLE+8(%r4) # wipe capability vectors
|
|
+ stg %r0,S390X_STFLE+16(%r4)
|
|
+ stg %r0,S390X_STFLE+24(%r4)
|
|
+ stg %r0,S390X_KIMD(%r4)
|
|
+ stg %r0,S390X_KIMD+8(%r4)
|
|
+ stg %r0,S390X_KLMD(%r4)
|
|
+ stg %r0,S390X_KLMD+8(%r4)
|
|
+ stg %r0,S390X_KM(%r4)
|
|
+ stg %r0,S390X_KM+8(%r4)
|
|
+ stg %r0,S390X_KMC(%r4)
|
|
+ stg %r0,S390X_KMC+8(%r4)
|
|
+ stg %r0,S390X_KMAC(%r4)
|
|
+ stg %r0,S390X_KMAC+8(%r4)
|
|
+ stg %r0,S390X_KMCTR(%r4)
|
|
+ stg %r0,S390X_KMCTR+8(%r4)
|
|
+ stg %r0,S390X_KMO(%r4)
|
|
+ stg %r0,S390X_KMO+8(%r4)
|
|
+ stg %r0,S390X_KMF(%r4)
|
|
+ stg %r0,S390X_KMF+8(%r4)
|
|
+ stg %r0,S390X_PRNO(%r4)
|
|
+ stg %r0,S390X_PRNO+8(%r4)
|
|
+ stg %r0,S390X_KMA(%r4)
|
|
+ stg %r0,S390X_KMA+8(%r4)
|
|
+
|
|
+ .long 0xb2b04000 # stfle 0(%r4)
|
|
+ brc 8,.Ldone
|
|
+ lghi %r0,1
|
|
+ .long 0xb2b04000 # stfle 0(%r4)
|
|
+ brc 8,.Ldone
|
|
+ lghi %r0,2
|
|
+ .long 0xb2b04000 # stfle 0(%r4)
|
|
+.Ldone:
|
|
+ lmg %r2,%r3,S390X_STFLE(%r4)
|
|
+ tmhl %r2,0x4000 # check for message-security-assist
|
|
+ jz .Lret
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query kimd capabilities
|
|
+ la %r1,S390X_KIMD(%r4)
|
|
+ .long 0xb93e0002 # kimd %r0,%r2
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query klmd capabilities
|
|
+ la %r1,S390X_KLMD(%r4)
|
|
+ .long 0xb93f0002 # klmd %r0,%r2
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query km capability vector
|
|
+ la %r1,S390X_KM(%r4)
|
|
+ .long 0xb92e0042 # km %r4,%r2
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query kmc capability vector
|
|
+ la %r1,S390X_KMC(%r4)
|
|
+ .long 0xb92f0042 # kmc %r4,%r2
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query kmac capability vector
|
|
+ la %r1,S390X_KMAC(%r4)
|
|
+ .long 0xb91e0042 # kmac %r4,%r2
|
|
+
|
|
+ tmhh %r3,0x0004 # check for message-security-assist-4
|
|
+ jz .Lret
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query kmctr capability vector
|
|
+ la %r1,S390X_KMCTR(%r4)
|
|
+ .long 0xb92d2042 # kmctr %r4,%r2,%r2
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query kmo capability vector
|
|
+ la %r1,S390X_KMO(%r4)
|
|
+ .long 0xb92b0042 # kmo %r4,%r2
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query kmf capability vector
|
|
+ la %r1,S390X_KMF(%r4)
|
|
+ .long 0xb92a0042 # kmf %r4,%r2
|
|
+
|
|
+ tml %r2,0x40 # check for message-security-assist-5
|
|
+ jz .Lret
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query prno capability vector
|
|
+ la %r1,S390X_PRNO(%r4)
|
|
+ .long 0xb93c0042 # prno %r4,%r2
|
|
+
|
|
+ lg %r2,S390X_STFLE+16(%r4)
|
|
+ tmhl %r2,0x2000 # check for message-security-assist-8
|
|
+ jz .Lret
|
|
+
|
|
+ lghi %r0,S390X_QUERY # query kma capability vector
|
|
+ la %r1,S390X_KMA(%r4)
|
|
+ .long 0xb9294022 # kma %r2,%r4,%r2
|
|
+
|
|
+.Lret:
|
|
+ br $ra
|
|
+.size OPENSSL_s390x_facilities,.-OPENSSL_s390x_facilities
|
|
+
|
|
+.globl OPENSSL_rdtsc
|
|
+.type OPENSSL_rdtsc,\@function
|
|
+.align 16
|
|
+OPENSSL_rdtsc:
|
|
+ stck 16($sp)
|
|
+ lg %r2,16($sp)
|
|
+ br $ra
|
|
+.size OPENSSL_rdtsc,.-OPENSSL_rdtsc
|
|
+
|
|
+.globl OPENSSL_atomic_add
|
|
+.type OPENSSL_atomic_add,\@function
|
|
+.align 16
|
|
+OPENSSL_atomic_add:
|
|
+ l %r1,0(%r2)
|
|
+.Lspin: lr %r0,%r1
|
|
+ ar %r0,%r3
|
|
+ cs %r1,%r0,0(%r2)
|
|
+ brc 4,.Lspin
|
|
+ lgfr %r2,%r0 # OpenSSL expects the new value
|
|
+ br $ra
|
|
+.size OPENSSL_atomic_add,.-OPENSSL_atomic_add
|
|
+
|
|
+.globl OPENSSL_wipe_cpu
|
|
+.type OPENSSL_wipe_cpu,\@function
|
|
+.align 16
|
|
+OPENSSL_wipe_cpu:
|
|
+ xgr %r0,%r0
|
|
+ xgr %r1,%r1
|
|
+ lgr %r2,$sp
|
|
+ xgr %r3,%r3
|
|
+ xgr %r4,%r4
|
|
+ lzdr %f0
|
|
+ lzdr %f1
|
|
+ lzdr %f2
|
|
+ lzdr %f3
|
|
+ lzdr %f4
|
|
+ lzdr %f5
|
|
+ lzdr %f6
|
|
+ lzdr %f7
|
|
+ br $ra
|
|
+.size OPENSSL_wipe_cpu,.-OPENSSL_wipe_cpu
|
|
+
|
|
+.globl OPENSSL_cleanse
|
|
+.type OPENSSL_cleanse,\@function
|
|
+.align 16
|
|
+OPENSSL_cleanse:
|
|
+#if !defined(__s390x__) && !defined(__s390x)
|
|
+ llgfr %r3,%r3
|
|
+#endif
|
|
+ lghi %r4,15
|
|
+ lghi %r0,0
|
|
+ clgr %r3,%r4
|
|
+ jh .Lot
|
|
+ clgr %r3,%r0
|
|
+ bcr 8,%r14
|
|
+.Little:
|
|
+ stc %r0,0(%r2)
|
|
+ la %r2,1(%r2)
|
|
+ brctg %r3,.Little
|
|
+ br %r14
|
|
+.align 4
|
|
+.Lot: tmll %r2,7
|
|
+ jz .Laligned
|
|
+ stc %r0,0(%r2)
|
|
+ la %r2,1(%r2)
|
|
+ brctg %r3,.Lot
|
|
+.Laligned:
|
|
+ srlg %r4,%r3,3
|
|
+.Loop: stg %r0,0(%r2)
|
|
+ la %r2,8(%r2)
|
|
+ brctg %r4,.Loop
|
|
+ lghi %r4,7
|
|
+ ngr %r3,%r4
|
|
+ jnz .Little
|
|
+ br $ra
|
|
+.size OPENSSL_cleanse,.-OPENSSL_cleanse
|
|
+
|
|
+.globl OPENSSL_vx_probe
|
|
+.type OPENSSL_vx_probe,\@function
|
|
+.align 16
|
|
+OPENSSL_vx_probe:
|
|
+ .word 0xe700,0x0000,0x0044 # vzero %v0
|
|
+ br $ra
|
|
+.size OPENSSL_vx_probe,.-OPENSSL_vx_probe
|
|
+___
|
|
+
|
|
+################
|
|
+# void s390x_km(const unsigned char *in, size_t len, unsigned char *out,
|
|
+# unsigned int fc, void *param)
|
|
+{
|
|
+my ($in,$len,$out,$fc,$param) = map("%r$_",(2..6));
|
|
+$code.=<<___;
|
|
+.globl s390x_km
|
|
+.type s390x_km,\@function
|
|
+.align 16
|
|
+s390x_km:
|
|
+ lr %r0,$fc
|
|
+ l${g}r %r1,$param
|
|
+
|
|
+ .long 0xb92e0042 # km $out,$in
|
|
+ brc 1,.-4 # pay attention to "partial completion"
|
|
+
|
|
+ br $ra
|
|
+.size s390x_km,.-s390x_km
|
|
+___
|
|
+}
|
|
+
|
|
+################
|
|
+# void s390x_kma(const unsigned char *aad, size_t alen,
|
|
+# const unsigned char *in, size_t len,
|
|
+# unsigned char *out, unsigned int fc, void *param)
|
|
+{
|
|
+my ($aad,$alen,$in,$len,$out) = map("%r$_",(2..6));
|
|
+$code.=<<___;
|
|
+.globl s390x_kma
|
|
+.type s390x_kma,\@function
|
|
+.align 16
|
|
+s390x_kma:
|
|
+ st${g} $out,6*$SIZE_T($sp)
|
|
+ lm${g} %r0,%r1,$stdframe($sp)
|
|
+
|
|
+ .long 0xb9292064 # kma $out,$aad,$in
|
|
+ brc 1,.-4 # pay attention to "partial completion"
|
|
+
|
|
+ l${g} $out,6*$SIZE_T($sp)
|
|
+ br $ra
|
|
+.size s390x_kma,.-s390x_kma
|
|
+___
|
|
+}
|
|
+
|
|
+$code.=<<___;
|
|
+.section .init
|
|
+ brasl $ra,OPENSSL_cpuid_setup
|
|
+___
|
|
+
|
|
+$code =~ s/\`([^\`]*)\`/eval $1/gem;
|
|
+print $code;
|
|
+close STDOUT; # force flush
|