chacha20poly1305
[openssl.git] / ssl / ssl_cert.c
index eb41cfda93e70afd05ac98b0dea26a96256189fd..1180f15227b3e688f27ce90778c77edc81e8ac8f 100644 (file)
@@ -321,21 +321,16 @@ CERT *ssl_cert_dup(CERT *cert)
 
                if (cpk->chain)
                        {
-                       int j;
-                       rpk->chain = sk_X509_dup(cpk->chain);
+                       rpk->chain = X509_chain_up_ref(cpk->chain);
                        if (!rpk->chain)
                                {
                                SSLerr(SSL_F_SSL_CERT_DUP, ERR_R_MALLOC_FAILURE);
                                goto err;
                                }
-                       for (j = 0; j < sk_X509_num(rpk->chain); j++)
-                               {
-                               X509 *x = sk_X509_value(rpk->chain, j);
-                               CRYPTO_add(&x->references, 1, CRYPTO_LOCK_X509);
-                               }
                        }
                rpk->valid_flags = 0;
-                if (cert->pkeys[i].authz != NULL)
+#ifndef OPENSSL_NO_TLSEXT
+     if (cert->pkeys[i].authz != NULL)
                        {
                        /* Just copy everything. */
                        ret->pkeys[i].authz_length =
@@ -345,12 +340,30 @@ CERT *ssl_cert_dup(CERT *cert)
                        if (ret->pkeys[i].authz == NULL)
                                {
                                SSLerr(SSL_F_SSL_CERT_DUP, ERR_R_MALLOC_FAILURE);
-                               return(NULL);
+                               return NULL;
                                }
                        memcpy(ret->pkeys[i].authz,
                               cert->pkeys[i].authz,
                               cert->pkeys[i].authz_length);
                        }
+
+               if (cert->pkeys[i].serverinfo != NULL)
+                       {
+                       /* Just copy everything. */
+                       ret->pkeys[i].serverinfo =
+                               OPENSSL_malloc(cert->pkeys[i].serverinfo_length);
+                       if (ret->pkeys[i].serverinfo == NULL)
+                               {
+                               SSLerr(SSL_F_SSL_CERT_DUP, ERR_R_MALLOC_FAILURE);
+                               return NULL;
+                               }
+                       ret->pkeys[i].serverinfo_length =
+                               cert->pkeys[i].serverinfo_length;
+                       memcpy(ret->pkeys[i].serverinfo,
+                              cert->pkeys[i].serverinfo,
+                              cert->pkeys[i].serverinfo_length);
+                       }
+#endif
                }
        
        ret->references=1;
@@ -415,6 +428,8 @@ CERT *ssl_cert_dup(CERT *cert)
                ret->chain_store = cert->chain_store;
                }
 
+       ret->ciphers_raw = NULL;
+
        return(ret);
        
 #if !defined(OPENSSL_NO_DH) || !defined(OPENSSL_NO_ECDH)
@@ -464,8 +479,17 @@ void ssl_cert_clear_certs(CERT *c)
                        cpk->chain = NULL;
                        }
 #ifndef OPENSSL_NO_TLSEXT
-                if (cpk->authz != NULL)
+               if (cpk->authz)
+                       {
                        OPENSSL_free(cpk->authz);
+                       cpk->authz = NULL;
+                       }
+               if (cpk->serverinfo)
+                       {
+                       OPENSSL_free(cpk->serverinfo);
+                       cpk->serverinfo = NULL;
+                       cpk->serverinfo_length = 0;
+                       }
 #endif
                /* Clear all flags apart from explicit sign */
                cpk->valid_flags &= CERT_PKEY_EXPLICIT_SIGN;
@@ -517,6 +541,8 @@ void ssl_cert_free(CERT *c)
                X509_STORE_free(c->verify_store);
        if (c->chain_store)
                X509_STORE_free(c->chain_store);
+       if (c->ciphers_raw)
+               OPENSSL_free(c->ciphers_raw);
        OPENSSL_free(c);
        }
 
@@ -562,18 +588,11 @@ int ssl_cert_set0_chain(CERT *c, STACK_OF(X509) *chain)
 int ssl_cert_set1_chain(CERT *c, STACK_OF(X509) *chain)
        {
        STACK_OF(X509) *dchain;
-       X509 *x;
-       int i;
        if (!chain)
                return ssl_cert_set0_chain(c, NULL);
-       dchain = sk_X509_dup(chain);
+       dchain = X509_chain_up_ref(chain);
        if (!dchain)
                return 0;
-       for (i = 0; i < sk_X509_num(dchain); i++)
-               {
-               x = sk_X509_value(dchain, i);
-               CRYPTO_add(&x->references, 1, CRYPTO_LOCK_X509);
-               }
        if (!ssl_cert_set0_chain(c, dchain))
                {
                sk_X509_pop_free(dchain, X509_free);
@@ -684,6 +703,138 @@ int ssl_set_peer_cert_type(SESS_CERT *sc,int type)
        return(1);
        }
 
+#ifndef OPENSSL_NO_DANE
+/*
+ * return value:
+ * -1: format or digest error
+ *  0: match
+ *  1: no match
+ */
+int tlsa_cmp(const X509 *cert, const unsigned char *tlsa_record, unsigned int reclen)
+{
+       const EVP_MD *md;
+       unsigned char digest[EVP_MAX_MD_SIZE];
+       unsigned int len, selector, matching_type;
+       int ret;
+
+       if (reclen<3) return -1;
+
+       selector      = tlsa_record[1];
+       matching_type = tlsa_record[2];
+       tlsa_record   += 3;
+       reclen        -= 3;
+
+       switch (matching_type) {
+       case 0:                         /* exact match */
+               if (selector==0) {      /* full certificate */
+                       ret = EVP_Digest(tlsa_record,reclen,digest,&len,EVP_sha1(),NULL);
+                       return ret ? memcmp(cert->sha1_hash,digest,len)!=0 : -1;
+               }
+               else if (selector==1) { /* SubjectPublicKeyInfo */
+                       ASN1_BIT_STRING *key = X509_get0_pubkey_bitstr(cert);
+
+                       if (key == NULL) return -1;
+                       if (key->length != reclen) return 1;
+
+                       return memcmp(key->data,tlsa_record,reclen)!=0;
+               }
+               return -1;
+
+       case 1:                         /* SHA256 */
+       case 2:                         /* SHA512 */
+               md = matching_type==1 ? EVP_sha256() : EVP_sha512();
+
+               if (reclen!=EVP_MD_size(md)) return -1;
+
+               if (selector==0) {      /* full certificate */
+                       ret = X509_digest(cert,md,digest,&len);
+               }
+               else if (selector==1) { /* SubjectPublicKeyInfo */
+                       ret = X509_pubkey_digest(cert,md,digest,&len);
+               }
+               else
+                       return -1;
+
+               return ret ? memcmp(tlsa_record,digest,len)!=0 : -1;
+       default:
+               return -1;
+       }
+}
+
+int dane_verify_callback(int ok, X509_STORE_CTX *ctx)
+{
+       SSL *s = X509_STORE_CTX_get_ex_data(ctx,SSL_get_ex_data_X509_STORE_CTX_idx());
+       int depth=X509_STORE_CTX_get_error_depth(ctx);
+       X509 *cert = sk_X509_value(ctx->chain,depth);
+       unsigned int reclen, certificate_usage, witness_usage=0x100;
+       const unsigned char *tlsa_record = s->tlsa_record;
+       int tlsa_ret = -1;
+
+       if (s->verify_callback) ok = s->verify_callback(ok,ctx);
+
+       if (tlsa_record == NULL) return ok;
+
+       if (tlsa_record == (void*)-1) {
+               ctx->error = X509_V_ERR_INVALID_CA;     /* temporary code? */
+               return 0;
+       }
+
+       while ((reclen = *(unsigned int *)tlsa_record)) {
+               tlsa_record += sizeof(unsigned int);
+
+               /*
+                * tlsa_record[0]       Certificate Usage field
+                * tlsa_record[1]       Selector field
+                * tlsa_record[2]       Matching Type Field
+                * tlsa_record+3        Certificate Association data
+                */
+               certificate_usage = tlsa_record[0];
+
+               if (depth==0 || certificate_usage==0 || certificate_usage==2) {
+                       tlsa_ret = tlsa_cmp(cert,tlsa_record,reclen);
+                       if (tlsa_ret==0) {
+                               s->tlsa_witness = depth<<8|certificate_usage;
+                               break;
+                       }
+                       else if (tlsa_ret==-1)
+                               s->tlsa_witness = -1;   /* something phishy? */
+               }
+
+               tlsa_record += reclen;
+       }
+
+       if (depth==0) {
+               switch (s->tlsa_witness&0xff) {         /* witnessed usage */
+               case 0: /* CA constraint */
+                       if (s->tlsa_witness<0 && ctx->error==X509_V_OK)
+                               ctx->error = X509_V_ERR_INVALID_CA;
+                       return 0;
+               case 1: /* service certificate constraint */
+                       if (tlsa_ret!=0 && ctx->error==X509_V_OK)
+                               ctx->error = X509_V_ERR_CERT_UNTRUSTED;
+                       return 0;
+               case 2: /* trust anchor assertion */
+                       if ((s->tlsa_witness>>8)>0 && ctx->error==X509_V_ERR_UNABLE_TO_GET_ISSUER_CERT_LOCALLY)
+                               ctx->error = X509_V_OK;
+                       break;
+               case 3: /* domain-issued certificate */
+                       if (tlsa_ret==0)
+                               ctx->error = X509_V_OK; /* override all errors? */
+                       break;
+               default:/* there were TLSA records, but something phishy happened */
+                       ctx->error = X509_V_ERR_CERT_UNTRUSTED;
+                       return ok;
+               }
+       }
+
+       /*
+        * returning 1 makes verify procedure traverse the whole chain,
+        * not actually approve it...
+        */
+       return 1;
+}
+#endif
+
 int ssl_verify_cert_chain(SSL *s,STACK_OF(X509) *sk)
        {
        X509 *x;
@@ -705,6 +856,8 @@ int ssl_verify_cert_chain(SSL *s,STACK_OF(X509) *sk)
                SSLerr(SSL_F_SSL_VERIFY_CERT_CHAIN,ERR_R_X509_LIB);
                return(0);
                }
+       /* Set suite B flags if needed */
+       X509_STORE_CTX_set_flags(&ctx, tls1_suiteb(s));
 #if 0
        if (SSL_get_verify_depth(s) >= 0)
                X509_STORE_CTX_set_depth(&ctx, SSL_get_verify_depth(s));
@@ -723,8 +876,13 @@ int ssl_verify_cert_chain(SSL *s,STACK_OF(X509) *sk)
         */
        X509_VERIFY_PARAM_set1(X509_STORE_CTX_get0_param(&ctx), s->param);
 
+#ifndef OPENSSL_NO_DANE
+       X509_STORE_CTX_set_verify_cb(&ctx, dane_verify_callback);
+       s->tlsa_witness = -1;
+#else
        if (s->verify_callback)
                X509_STORE_CTX_set_verify_cb(&ctx, s->verify_callback);
+#endif
 
        if (s->ctx->app_verify_callback != NULL)
 #if 1 /* new with OpenSSL 0.9.7 */
@@ -1164,6 +1322,8 @@ int ssl_build_cert_chain(CERT *c, X509_STORE *chain_store, int flags)
                SSLerr(SSL_F_SSL_BUILD_CERT_CHAIN, ERR_R_X509_LIB);
                return 0;
                }
+       /* Set suite B flags if needed */
+       X509_STORE_CTX_set_flags(&xs_ctx, c->cert_flags & SSL_CERT_FLAG_SUITEB_128_LOS);
 
        i = X509_verify_cert(&xs_ctx);
        if (i > 0)