2 * ngIRCd -- The Next Generation IRC Daemon
3 * Copyright (c)2005-2008 Florian Westphal (fw@strlen.de).
4 * Copyright (c)2008-2014 Alexander Barton (alex@barton.de) and Contributors.
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 * Please read the file COPYING, README and AUTHORS for more information.
17 * SSL wrapper functions
34 #include "conn-func.h"
40 extern struct SSLOptions Conf_SSLOptions;
43 #include <openssl/err.h>
44 #include <openssl/rand.h>
46 static SSL_CTX * ssl_ctx;
49 static bool ConnSSL_LoadServerKey_openssl PARAMS(( SSL_CTX *c ));
50 static bool ConnSSL_SetVerifyProperties_openssl PARAMS(( SSL_CTX *c ));
53 #define MAX_CERT_CHAIN_LENGTH 10 /* XXX: do not hardcode */
56 #include <sys/types.h>
60 #include <gnutls/x509.h>
63 #define DH_BITS_MIN 1024
65 #define MAX_HASH_SIZE 64 /* from gnutls-int.h */
67 gnutls_x509_crl_t *crl_list;
68 gnutls_x509_crt_t *ca_list;
72 static gnutls_certificate_credentials_t x509_cred;
73 static gnutls_dh_params_t dh_params;
74 static gnutls_priority_t priorities_cache;
75 static bool ConnSSL_LoadServerKey_gnutls PARAMS(( void ));
76 static bool ConnSSL_SetVerifyProperties_gnutls PARAMS(( void ));
79 #define SHA256_STRING_LEN (32 * 2 + 1)
81 static bool ConnSSL_Init_SSL PARAMS(( CONNECTION *c ));
82 static int ConnectAccept PARAMS(( CONNECTION *c, bool connect ));
83 static int ConnSSL_HandleError PARAMS(( CONNECTION *c, const int code, const char *fname ));
86 static char * openreadclose(const char *name, size_t *len)
91 int fd = open(name, O_RDONLY);
93 Log(LOG_ERR, "Could not open %s: %s", name, strerror(errno));
97 Log(LOG_ERR, "Could not fstat %s: %s", name, strerror(errno));
100 if (!S_ISREG(s.st_mode)) {
101 Log(LOG_ERR, "%s: Not a regular file", name);
104 if (s.st_size <= 0) {
105 Log(LOG_ERR, "%s: invalid file length (size %ld <= 0)", name, (long) s.st_size);
108 buf = malloc(s.st_size);
110 Log(LOG_ERR, "Could not malloc %lu bytes for file %s: %s", s.st_size, name, strerror(errno));
113 br = read(fd, buf, s.st_size);
114 if (br != (ssize_t)s.st_size) {
115 Log(LOG_ERR, "Could not read file %s: read returned %ld, expected %ld: %s",
116 name, (long) br, (long) s.st_size, br == -1 ? strerror(errno):"short read?!");
117 memset(buf, 0, s.st_size);
132 * Log OpenSSL error message.
134 * @param msg The error message.
135 * @param info Additional information text or NULL.
138 LogOpenSSL_CertInfo(X509 *cert, const char *msg)
149 mem = BIO_new(BIO_s_mem());
152 X509_NAME_print_ex(mem, X509_get_subject_name (cert), 4, XN_FLAG_ONELINE);
153 X509_NAME_print_ex(mem, X509_get_issuer_name (cert), 4, XN_FLAG_ONELINE);
154 if (BIO_write(mem, "", 1) == 1) {
155 len = BIO_get_mem_data(mem, &memptr);
158 Log(LOG_INFO, "%s: \"%s\"", msg, memptr);
161 (void)BIO_set_close(mem, BIO_CLOSE);
166 LogOpenSSLError(const char *error, const char *info)
168 unsigned long err = ERR_get_error();
170 ? ERR_error_string(err, NULL)
171 : "Unable to determine error";
173 assert(error != NULL);
176 Log(LOG_ERR, "%s: %s (%s)", error, info, errmsg);
178 Log(LOG_ERR, "%s: %s", error, errmsg);
183 pem_passwd_cb(char *buf, int size, int rwflag, void *password)
185 array *pass = password;
188 (void)rwflag; /* rwflag is unused if DEBUG is not set. */
189 assert(rwflag == 0); /* 0 -> callback used for decryption.
190 * See SSL_CTX_set_default_passwd_cb(3) */
192 passlen = (int) array_bytes(pass);
194 LogDebug("pem_passwd_cb buf size %d, array size %d", size, passlen);
195 assert(passlen >= 0);
197 Log(LOG_ERR, "PEM password required but not set [in pem_passwd_cb()]!");
200 size = passlen > size ? size : passlen;
201 memcpy(buf, (char *)(array_start(pass)), size);
207 Verify_openssl(int preverify_ok, X509_STORE_CTX *ctx)
212 err_cert = X509_STORE_CTX_get_current_cert(ctx);
213 err = X509_STORE_CTX_get_error(ctx);
214 depth = X509_STORE_CTX_get_error_depth(ctx);
216 LogDebug("preverify_ok %d error:num=%d:%s:depth=%d",
217 preverify_ok, err, X509_verify_cert_error_string(err), depth);
219 if (preverify_ok != 1) {
221 * if certificates are not being enforced, ignore any errors.
222 * its possible to check if a connection has a valid certificate
223 * by testing the CONN_SSL_PEERCERT_OK flag.
224 * This can be used to enforce certificates for incoming servers,
225 * but not irc clients.
227 if (!Conf_SSLOptions.RequireClientCert)
230 Log(LOG_ERR, "verify error:num=%d:%s:depth=%d", err,
231 X509_verify_cert_error_string(err), depth);
245 if (!Conf_SSLOptions.DHFile) {
246 Log(LOG_NOTICE, "Configuration option \"DHFile\" not set!");
249 fp = fopen(Conf_SSLOptions.DHFile, "r");
251 Log(LOG_ERR, "%s: %s", Conf_SSLOptions.DHFile, strerror(errno));
254 dh_params = PEM_read_DHparams(fp, NULL, NULL, NULL);
256 Log(LOG_ERR, "%s: Failed to read SSL DH parameters!",
257 Conf_SSLOptions.DHFile);
263 #ifdef HAVE_LIBGNUTLS
264 bool need_dhgenerate = true;
266 gnutls_dh_params_t tmp_dh_params;
268 err = gnutls_dh_params_init(&tmp_dh_params);
270 Log(LOG_ERR, "Failed to initialize SSL DH parameters: %s",
271 gnutls_strerror(err));
274 if (Conf_SSLOptions.DHFile) {
275 gnutls_datum_t dhparms;
277 dhparms.data = (unsigned char *) openreadclose(Conf_SSLOptions.DHFile, &size);
280 err = gnutls_dh_params_import_pkcs3(tmp_dh_params, &dhparms, GNUTLS_X509_FMT_PEM);
282 need_dhgenerate = false;
285 "Failed to initialize SSL DH parameters: %s",
286 gnutls_strerror(err));
288 memset(dhparms.data, 0, size);
292 if (need_dhgenerate) {
294 "DHFile not set, generating %u bit DH parameters. This may take a while ...",
296 err = gnutls_dh_params_generate2(tmp_dh_params, DH_BITS);
298 Log(LOG_ERR, "Failed to generate SSL DH parameters: %s",
299 gnutls_strerror(err));
303 dh_params = tmp_dh_params;
309 void ConnSSL_Free(CONNECTION *c)
312 SSL *ssl = c->ssl_state.ssl;
316 c->ssl_state.ssl = NULL;
317 if (c->ssl_state.fingerprint) {
318 free(c->ssl_state.fingerprint);
319 c->ssl_state.fingerprint = NULL;
323 #ifdef HAVE_LIBGNUTLS
324 gnutls_session_t sess = c->ssl_state.gnutls_session;
325 if (Conn_OPTION_ISSET(c, CONN_SSL)) {
326 gnutls_bye(sess, GNUTLS_SHUT_RDWR);
330 assert(Conn_OPTION_ISSET(c, CONN_SSL));
331 /* can't just set bitmask to 0 -- there are other, non-ssl related flags, e.g. CONN_ZIP. */
332 Conn_OPTION_DEL(c, CONN_SSL_FLAGS_ALL);
337 ConnSSL_InitLibrary( void )
339 if (!Conf_SSLInUse()) {
340 LogDebug("SSL not in use, skipping initialization.");
349 SSL_load_error_strings();
352 if (!RAND_status()) {
353 Log(LOG_ERR, "OpenSSL PRNG not seeded: /dev/urandom missing?");
355 * it is probably best to fail and let the user install EGD or
356 * a similar program if no kernel random device is available.
357 * According to OpenSSL RAND_egd(3): "The automatic query of
358 * /var/run/egd-pool et al was added in OpenSSL 0.9.7";
359 * so it makes little sense to deal with PRNGD seeding ourselves.
361 array_free(&Conf_SSLOptions.ListenPorts);
365 newctx = SSL_CTX_new(SSLv23_method());
367 LogOpenSSLError("Failed to create SSL context", NULL);
368 array_free(&Conf_SSLOptions.ListenPorts);
372 if (!ConnSSL_LoadServerKey_openssl(newctx))
375 if (SSL_CTX_set_cipher_list(newctx, Conf_SSLOptions.CipherList) == 0) {
376 Log(LOG_ERR, "Failed to apply OpenSSL cipher list \"%s\"!",
377 Conf_SSLOptions.CipherList);
381 if (!ConnSSL_SetVerifyProperties_openssl(newctx))
384 SSL_CTX_set_options(newctx, SSL_OP_SINGLE_DH_USE|SSL_OP_NO_SSLv2);
385 SSL_CTX_set_mode(newctx, SSL_MODE_ENABLE_PARTIAL_WRITE);
386 SSL_CTX_set_verify(newctx, SSL_VERIFY_PEER|SSL_VERIFY_CLIENT_ONCE,
388 SSL_CTX_free(ssl_ctx);
390 Log(LOG_INFO, "%s initialized.", SSLeay_version(SSLEAY_VERSION));
393 SSL_CTX_free(newctx);
394 array_free(&Conf_SSLOptions.ListenPorts);
397 #ifdef HAVE_LIBGNUTLS
399 static bool initialized;
402 /* TODO: cannot reload gnutls keys: can't simply free x509
403 * context -- it may still be in use */
407 err = gnutls_global_init();
409 Log(LOG_ERR, "Failed to initialize GnuTLS: %s",
410 gnutls_strerror(err));
414 if (!ConnSSL_LoadServerKey_gnutls())
417 if (gnutls_priority_init(&priorities_cache, Conf_SSLOptions.CipherList,
418 NULL) != GNUTLS_E_SUCCESS) {
420 "Failed to apply GnuTLS cipher list \"%s\"!",
421 Conf_SSLOptions.CipherList);
425 if (!ConnSSL_SetVerifyProperties_gnutls())
428 Log(LOG_INFO, "GnuTLS %s initialized.", gnutls_check_version(NULL));
432 array_free(&Conf_SSLOptions.ListenPorts);
438 #ifdef HAVE_LIBGNUTLS
440 ConnSSL_SetVerifyProperties_gnutls(void)
444 if (Conf_SSLOptions.CAFile) {
445 err = gnutls_certificate_set_x509_trust_file(x509_cred, Conf_SSLOptions.CAFile, GNUTLS_X509_FMT_PEM);
447 Log(LOG_ERR, "Failed to load x509 trust file %s: %s", Conf_SSLOptions.CAFile, gnutls_strerror(err));
451 if (Conf_SSLOptions.CRLFile) {
452 err = gnutls_certificate_set_x509_crl_file(x509_cred, Conf_SSLOptions.CRLFile, GNUTLS_X509_FMT_PEM);
454 Log(LOG_ERR, "Failed to load x509 crl file %s: %s", Conf_SSLOptions.CRLFile, gnutls_strerror(err));
464 ConnSSL_LoadServerKey_gnutls(void)
467 const char *cert_file;
469 err = gnutls_certificate_allocate_credentials(&x509_cred);
471 Log(LOG_ERR, "Failed to allocate certificate credentials: %s",
472 gnutls_strerror(err));
476 cert_file = Conf_SSLOptions.CertFile ? Conf_SSLOptions.CertFile:Conf_SSLOptions.KeyFile;
478 Log(LOG_ERR, "No SSL server key configured!");
482 if (array_bytes(&Conf_SSLOptions.KeyFilePassword))
484 "Ignoring SSL \"KeyFilePassword\": Not supported by GnuTLS.");
486 if (!Load_DH_params())
489 gnutls_certificate_set_dh_params(x509_cred, dh_params);
490 err = gnutls_certificate_set_x509_key_file(x509_cred, cert_file, Conf_SSLOptions.KeyFile, GNUTLS_X509_FMT_PEM);
493 "Failed to set certificate key file (cert %s, key %s): %s",
495 Conf_SSLOptions.KeyFile ? Conf_SSLOptions.KeyFile : "(NULL)",
496 gnutls_strerror(err));
506 ConnSSL_LoadServerKey_openssl(SSL_CTX *ctx)
511 if (!Conf_SSLOptions.KeyFile) {
512 Log(LOG_ERR, "No SSL server key configured!");
516 SSL_CTX_set_default_passwd_cb(ctx, pem_passwd_cb);
517 SSL_CTX_set_default_passwd_cb_userdata(ctx, &Conf_SSLOptions.KeyFilePassword);
519 if (SSL_CTX_use_PrivateKey_file(ctx, Conf_SSLOptions.KeyFile, SSL_FILETYPE_PEM) != 1) {
520 array_free_wipe(&Conf_SSLOptions.KeyFilePassword);
521 LogOpenSSLError("Failed to add private key", Conf_SSLOptions.KeyFile);
525 cert_key = Conf_SSLOptions.CertFile ? Conf_SSLOptions.CertFile:Conf_SSLOptions.KeyFile;
526 if (SSL_CTX_use_certificate_chain_file(ctx, cert_key) != 1) {
527 array_free_wipe(&Conf_SSLOptions.KeyFilePassword);
528 LogOpenSSLError("Failed to load certificate chain", cert_key);
532 array_free_wipe(&Conf_SSLOptions.KeyFilePassword);
534 if (!SSL_CTX_check_private_key(ctx)) {
535 LogOpenSSLError("Server private key does not match certificate", NULL);
538 if (Load_DH_params()) {
539 if (SSL_CTX_set_tmp_dh(ctx, dh_params) != 1)
540 LogOpenSSLError("Error setting DH parameters", Conf_SSLOptions.DHFile);
541 /* don't return false here: the non-DH modes will still work */
550 ConnSSL_SetVerifyProperties_openssl(SSL_CTX * ctx)
552 X509_STORE *store = NULL;
554 int verify_flags = SSL_VERIFY_PEER;
557 if (!Conf_SSLOptions.CAFile)
560 if (SSL_CTX_load_verify_locations(ctx, Conf_SSLOptions.CAFile, NULL) != 1) {
561 LogOpenSSLError("SSL_CTX_load_verify_locations", NULL);
565 store = SSL_CTX_get_cert_store(ctx);
567 lookup = X509_STORE_add_lookup(store, X509_LOOKUP_file());
569 LogOpenSSLError("X509_STORE_add_lookup", Conf_SSLOptions.CRLFile);
573 if (1 != X509_load_crl_file(lookup, Conf_SSLOptions.CRLFile, X509_FILETYPE_PEM)) {
574 LogOpenSSLError("X509_load_crl_file", Conf_SSLOptions.CRLFile);
578 if (Conf_SSLOptions.RequireClientCert)
579 verify_flags |= SSL_VERIFY_FAIL_IF_NO_PEER_CERT;
581 SSL_CTX_set_verify(ctx, verify_flags, Verify_openssl);
582 SSL_CTX_set_verify_depth(ctx, MAX_CERT_CHAIN_LENGTH);
585 free(Conf_SSLOptions.CRLFile);
586 Conf_SSLOptions.CRLFile = NULL;
593 ConnSSL_Init_SSL(CONNECTION *c)
597 LogDebug("Initializing SSL ...");
603 "Can't initialize SSL context, OpenSSL initialization failed at startup!");
606 assert(c->ssl_state.ssl == NULL);
607 assert(c->ssl_state.fingerprint == NULL);
609 c->ssl_state.ssl = SSL_new(ssl_ctx);
610 if (!c->ssl_state.ssl) {
611 LogOpenSSLError("Failed to create SSL structure", NULL);
614 Conn_OPTION_ADD(c, CONN_SSL);
616 ret = SSL_set_fd(c->ssl_state.ssl, c->sock);
618 LogOpenSSLError("Failed to set SSL file descriptor", NULL);
623 #ifdef HAVE_LIBGNUTLS
624 Conn_OPTION_ADD(c, CONN_SSL);
625 ret = gnutls_priority_set(c->ssl_state.gnutls_session, priorities_cache);
626 if (ret != GNUTLS_E_SUCCESS) {
627 Log(LOG_ERR, "Failed to set GnuTLS session priorities: %s",
628 gnutls_strerror(ret));
633 * The intermediate (long) cast is here to avoid a warning like:
634 * "cast to pointer from integer of different size" on 64-bit platforms.
635 * There doesn't seem to be an alternate GNUTLS API we could use instead, see e.g.
636 * http://www.mail-archive.com/help-gnutls@gnu.org/msg00286.html
638 gnutls_transport_set_ptr(c->ssl_state.gnutls_session,
639 (gnutls_transport_ptr_t) (long) c->sock);
640 gnutls_certificate_server_set_request(c->ssl_state.gnutls_session,
641 GNUTLS_CERT_REQUEST);
642 ret = gnutls_credentials_set(c->ssl_state.gnutls_session,
643 GNUTLS_CRD_CERTIFICATE, x509_cred);
645 Log(LOG_ERR, "Failed to set SSL credentials: %s",
646 gnutls_strerror(ret));
650 gnutls_dh_set_prime_bits(c->ssl_state.gnutls_session, DH_BITS_MIN);
657 ConnSSL_PrepareConnect(CONNECTION *c, UNUSED CONF_SERVER *s)
660 #ifdef HAVE_LIBGNUTLS
663 err = gnutls_init(&c->ssl_state.gnutls_session, GNUTLS_CLIENT);
665 Log(LOG_ERR, "Failed to initialize new SSL session: %s",
666 gnutls_strerror(err));
670 ret = ConnSSL_Init_SSL(c);
673 Conn_OPTION_ADD(c, CONN_SSL_CONNECT);
675 assert(c->ssl_state.ssl);
678 SSL_set_verify(c->ssl_state.ssl, SSL_VERIFY_PEER, Verify_openssl);
680 SSL_set_verify(c->ssl_state.ssl, SSL_VERIFY_NONE, NULL);
687 * Check and handle error return codes after failed calls to SSL functions.
690 * SSL_connect(), SSL_accept(), SSL_do_handshake(), SSL_read(), SSL_peek(), or
691 * SSL_write() on ssl.
694 * gnutlsssl_read(), gnutls_write() or gnutls_handshake().
696 * @param c The connection handle.
697 * @prarm code The return code.
698 * @param fname The name of the function in which the error occurred.
699 * @return -1 on fatal errors, 0 if we can try again later.
702 ConnSSL_HandleError(CONNECTION * c, const int code, const char *fname)
705 int ret = SSL_ERROR_SYSCALL;
706 unsigned long sslerr;
707 int real_errno = errno;
709 ret = SSL_get_error(c->ssl_state.ssl, code);
712 case SSL_ERROR_WANT_READ:
713 io_event_del(c->sock, IO_WANTWRITE);
714 Conn_OPTION_ADD(c, CONN_SSL_WANT_READ);
715 return 0; /* try again later */
716 case SSL_ERROR_WANT_WRITE:
717 io_event_del(c->sock, IO_WANTREAD);
718 Conn_OPTION_ADD(c, CONN_SSL_WANT_WRITE); /* fall through */
720 return 0; /* try again later */
721 case SSL_ERROR_ZERO_RETURN:
722 LogDebug("SSL connection shut down normally.");
724 case SSL_ERROR_SYSCALL:
725 /* SSL_ERROR_WANT_CONNECT, SSL_ERROR_WANT_ACCEPT,
726 * and SSL_ERROR_WANT_X509_LOOKUP */
727 sslerr = ERR_get_error();
729 Log(LOG_ERR, "SSL error: %s [in %s()]!",
730 ERR_error_string(sslerr, NULL), fname);
732 switch (code) { /* EOF that violated protocol */
735 "SSL error, client disconnected [in %s()]!",
738 case -1: /* low level socket I/O error, check errno */
739 Log(LOG_ERR, "SSL error: %s [in %s()]!",
740 strerror(real_errno), fname);
745 LogOpenSSLError("SSL protocol error", fname);
748 Log(LOG_ERR, "Unknown SSL error %d [in %s()]!", ret, fname);
753 #ifdef HAVE_LIBGNUTLS
756 case GNUTLS_E_INTERRUPTED:
757 if (gnutls_record_get_direction(c->ssl_state.gnutls_session)) {
758 Conn_OPTION_ADD(c, CONN_SSL_WANT_WRITE);
759 io_event_del(c->sock, IO_WANTREAD);
761 Conn_OPTION_ADD(c, CONN_SSL_WANT_READ);
762 io_event_del(c->sock, IO_WANTWRITE);
767 if (gnutls_error_is_fatal(code)) {
768 Log(LOG_ERR, "SSL error: %s [%s].",
769 gnutls_strerror(code), fname);
779 #ifdef HAVE_LIBGNUTLS
780 static bool check_verification(unsigned output)
782 char errmsg[256] = "";
784 if (output & GNUTLS_CERT_SIGNER_NOT_FOUND)
785 strlcpy(errmsg, "No Issuer found", sizeof errmsg);
786 if (output & GNUTLS_CERT_SIGNER_NOT_CA) {
788 strlcat(errmsg, " ,", sizeof errmsg);
789 strlcat(errmsg, "Issuer is not a CA", sizeof errmsg);
791 if (output & GNUTLS_CERT_INSECURE_ALGORITHM) {
793 strlcat(errmsg, " ,", sizeof errmsg);
794 strlcat(errmsg, "Insecure Algorithm", sizeof errmsg);
796 if (output & GNUTLS_CERT_REVOKED) {
798 strlcat(errmsg, " ,", sizeof errmsg);
799 strlcat(errmsg, "Certificate Revoked", sizeof errmsg);
802 if (output & GNUTLS_CERT_INVALID)
803 assert(errmsg[0]); /* check for unhandled error */
805 if (!(output & GNUTLS_CERT_INVALID) && !errmsg[0]) {
806 LogDebug("Certificate verified.");
809 Log(LOG_ERR, "Certificate Validation failed: %s", errmsg);
813 static void *LogMalloc(size_t s)
815 void *mem = malloc(s);
817 Log(LOG_ERR, "Out of memory: Could not allocate %lu byte", (unsigned long) s);
822 LogGnuTLS_CertInfo(gnutls_x509_crt_t cert, const char *msg)
824 char *dn, *issuer_dn;
826 int err = gnutls_x509_crt_get_dn(cert, NULL, &size);
827 if (size == 0 || (err && err != GNUTLS_E_SHORT_MEMORY_BUFFER))
829 dn = LogMalloc(size);
832 err = gnutls_x509_crt_get_dn(cert, dn, &size);
835 Log(LOG_ERR, "gnutls_x509_crt_get_dn: %s", err ? gnutls_strerror(err) : "size == 0");
838 gnutls_x509_crt_get_issuer_dn(cert, NULL, &size);
840 issuer_dn = LogMalloc(size);
842 Log(LOG_INFO, "%s: Distinguished Name: %s", msg, dn);
846 gnutls_x509_crt_get_issuer_dn(cert, issuer_dn, &size);
847 Log(LOG_INFO, "%s: Distinguished Name: \"%s\", Issuer \"%s\"", msg, dn, issuer_dn);
854 ConnSSL_LogCertInfo( CONNECTION *c )
856 const char *comp_alg = "no compression";
857 bool cert_seen = false, cert_ok = false, cn_match = false;
861 X509 *client_cert = NULL;
862 SSL *ssl = c->ssl_state.ssl;
865 comp=SSL_get_current_compression(ssl);
867 Conn_OPTION_ADD(c, CONN_SSL_COMPRESSION);
868 comp_alg = SSL_COMP_get_name(comp);
870 Log(LOG_INFO, "Connection %d: initialized %s using cipher %s, %s.",
871 c->sock, SSL_get_version(ssl), SSL_get_cipher(ssl), comp_alg);
872 client_cert = SSL_get_peer_certificate(ssl);
874 int err = SSL_get_verify_result(ssl);
875 if (err == X509_V_OK)
878 Log(LOG_ERR, "Certificate Validation failed: %s", X509_verify_cert_error_string(err));
879 X509_NAME *subjectName;
881 subjectName = X509_get_subject_name(client_cert);
882 X509_NAME_get_text_by_NID(
883 subjectName, NID_commonName, cn, sizeof(cn));
884 if (strcmp (cn, c->host) == 0)
887 Log(LOG_ERR, "CN mismatch! Got \"%s\", expected \"%s\"", cn, c->host);
889 snprintf(msg, sizeof(msg), "%svalid peer certificate", cert_ok ? "":"in");
890 LogOpenSSL_CertInfo(client_cert, msg);
891 X509_free(client_cert);
895 #ifdef HAVE_LIBGNUTLS
898 gnutls_credentials_type_t cred;
899 gnutls_session_t sess = c->ssl_state.gnutls_session;
900 gnutls_cipher_algorithm_t cipher = gnutls_cipher_get(sess);
901 gnutls_compression_method_t comp = gnutls_compression_get(sess);
903 if (comp != GNUTLS_COMP_NULL)
904 comp_alg = gnutls_compression_get_name(comp);
905 Log(LOG_INFO, "Connection %d: initialized %s using cipher %s-%s, %s.",
907 gnutls_protocol_get_name(gnutls_protocol_get_version(sess)),
908 gnutls_cipher_get_name(cipher),
909 gnutls_mac_get_name(gnutls_mac_get(sess)),
911 ret = gnutls_certificate_verify_peers2(c->ssl_state.gnutls_session, &status);
913 Log(LOG_ERR, "gnutls_certificate_verify_peers2 failed: %s", gnutls_strerror(ret));
914 if (ret == 0 && check_verification(status))
917 cred = gnutls_auth_get_type (c->ssl_state.gnutls_session);
918 if (cred == GNUTLS_CRD_CERTIFICATE) {
919 gnutls_x509_crt_t cert;
920 unsigned cert_list_size;
921 const gnutls_datum_t *cert_list = gnutls_certificate_get_peers(sess, &cert_list_size);
923 Log(LOG_ERR, "gnutls_certificate_get_peers() failed");
926 assert(cert_list_size > 0);
927 gnutls_x509_crt_init(&cert);
928 gnutls_x509_crt_import(cert, cert_list, GNUTLS_X509_FMT_DER);
932 int err = gnutls_x509_crt_get_dn_by_oid(cert,
933 GNUTLS_OID_X520_COMMON_NAME,
935 if (size == 0 || (err && err != GNUTLS_E_SHORT_MEMORY_BUFFER))
936 goto done_cn_validation;
937 cn = LogMalloc(size);
939 goto done_cn_validation;
940 gnutls_x509_crt_get_dn_by_oid (cert,
941 GNUTLS_OID_X520_COMMON_NAME,
943 if (strcmp (cn, c->host) == 0)
946 Log(LOG_ERR, "CN mismatch! Got \"%s\", expected \"%s\"", cn, c->host);
950 snprintf(msg, sizeof(msg), "%svalid peer certificate", cert_ok ? "":"in");
951 LogGnuTLS_CertInfo(cert, msg);
952 gnutls_x509_crt_deinit(cert);
957 * can be used later to check if connection was authenticated, e.g. if inbound connection
958 * tries to register itself as server. could also restrict /OPER to authenticated connections, etc.
960 if (cert_ok && cn_match)
961 Conn_OPTION_ADD(c, CONN_SSL_PEERCERT_OK);
963 Log(LOG_INFO, "Peer did not present a certificate");
968 Accept incoming SSL connection.
970 1: SSL Connection established
972 -1: SSL Connection not established due to fatal error.
975 ConnSSL_Accept( CONNECTION *c )
978 if (!Conn_OPTION_ISSET(c, CONN_SSL)) {
979 #ifdef HAVE_LIBGNUTLS
980 gnutls_certificate_request_t req;
981 int err = gnutls_init(&c->ssl_state.gnutls_session, GNUTLS_SERVER);
983 Log(LOG_ERR, "Failed to initialize new SSL session: %s",
984 gnutls_strerror(err));
987 req = Conf_SSLOptions.RequireClientCert ? GNUTLS_CERT_REQUIRE : GNUTLS_CERT_REQUEST;
988 gnutls_certificate_server_set_request(c->ssl_state.gnutls_session, req);
990 if (!ConnSSL_Init_SSL(c))
993 return ConnectAccept(c, false );
998 ConnSSL_Connect( CONNECTION *c )
1002 assert(c->ssl_state.ssl);
1004 assert(Conn_OPTION_ISSET(c, CONN_SSL));
1005 return ConnectAccept(c, true);
1009 ConnSSL_InitCertFp( CONNECTION *c )
1011 const char hex[] = "0123456789abcdef";
1015 unsigned char digest[EVP_MAX_MD_SIZE];
1016 unsigned int digest_size;
1019 cert = SSL_get_peer_certificate(c->ssl_state.ssl);
1023 if (!X509_digest(cert, EVP_sha256(), digest, &digest_size)) {
1029 #endif /* HAVE_LIBSSL */
1030 #ifdef HAVE_LIBGNUTLS
1031 gnutls_x509_crt_t cert;
1032 unsigned int cert_list_size;
1033 const gnutls_datum_t *cert_list;
1034 unsigned char digest[MAX_HASH_SIZE];
1037 if (gnutls_certificate_type_get(c->ssl_state.gnutls_session) !=
1041 if (gnutls_x509_crt_init(&cert) != GNUTLS_E_SUCCESS)
1045 cert_list = gnutls_certificate_get_peers(c->ssl_state.gnutls_session,
1048 gnutls_x509_crt_deinit(cert);
1052 if (gnutls_x509_crt_import(cert, &cert_list[0],
1053 GNUTLS_X509_FMT_DER) != GNUTLS_E_SUCCESS) {
1054 gnutls_x509_crt_deinit(cert);
1058 digest_size = sizeof(digest);
1059 if (gnutls_x509_crt_get_fingerprint(cert, GNUTLS_DIG_SHA256, digest,
1061 gnutls_x509_crt_deinit(cert);
1065 gnutls_x509_crt_deinit(cert);
1066 #endif /* HAVE_LIBGNUTLS */
1068 assert(c->ssl_state.fingerprint == NULL);
1070 c->ssl_state.fingerprint = malloc(SHA256_STRING_LEN);
1071 if (!c->ssl_state.fingerprint)
1074 for (i = 0; i < (int)digest_size; i++) {
1075 c->ssl_state.fingerprint[i * 2] = hex[digest[i] / 16];
1076 c->ssl_state.fingerprint[i * 2 + 1] = hex[digest[i] % 16];
1078 c->ssl_state.fingerprint[i * 2] = '\0';
1083 /* accept/connect wrapper. if connect is true, connect to peer, otherwise wait for incoming connection */
1085 ConnectAccept( CONNECTION *c, bool connect)
1089 SSL *ssl = c->ssl_state.ssl;
1090 assert(ssl != NULL);
1092 ret = connect ? SSL_connect(ssl) : SSL_accept(ssl);
1094 return ConnSSL_HandleError(c, ret, connect ? "SSL_connect": "SSL_accept");
1096 #ifdef HAVE_LIBGNUTLS
1098 ret = gnutls_handshake(c->ssl_state.gnutls_session);
1100 return ConnSSL_HandleError(c, ret, "gnutls_handshake");
1101 #endif /* _GNUTLS */
1102 (void)ConnSSL_InitCertFp(c);
1104 Conn_OPTION_DEL(c, (CONN_SSL_WANT_WRITE|CONN_SSL_WANT_READ|CONN_SSL_CONNECT));
1105 ConnSSL_LogCertInfo(c);
1107 Conn_StartLogin(CONNECTION2ID(c));
1113 ConnSSL_Write(CONNECTION *c, const void *buf, size_t count)
1117 Conn_OPTION_DEL(c, CONN_SSL_WANT_WRITE|CONN_SSL_WANT_READ);
1121 bw = (ssize_t) SSL_write(c->ssl_state.ssl, buf, count);
1123 #ifdef HAVE_LIBGNUTLS
1124 bw = gnutls_write(c->ssl_state.gnutls_session, buf, count);
1128 if (ConnSSL_HandleError( c, bw, "ConnSSL_Write") == 0)
1129 errno = EAGAIN; /* try again */
1135 ConnSSL_Read(CONNECTION *c, void * buf, size_t count)
1139 Conn_OPTION_DEL(c, CONN_SSL_WANT_WRITE|CONN_SSL_WANT_READ);
1141 br = (ssize_t) SSL_read(c->ssl_state.ssl, buf, count);
1142 if (br > 0) /* on EOF we have to call ConnSSL_HandleError(), see SSL_read(3) */
1145 #ifdef HAVE_LIBGNUTLS
1146 br = gnutls_read(c->ssl_state.gnutls_session, buf, count);
1147 if (br >= 0) /* on EOF we must _not_ call ConnSSL_HandleError, see gnutls_record_recv(3) */
1150 /* error on read: switch ConnSSL_HandleError() return values -> 0 is "try again", so return -1 and set EAGAIN */
1151 if (ConnSSL_HandleError(c, br, "ConnSSL_Read") == 0) {
1160 ConnSSL_GetCipherInfo(CONNECTION *c, char *buf, size_t len)
1164 SSL *ssl = c->ssl_state.ssl;
1169 SSL_CIPHER_description(SSL_get_current_cipher(ssl), buf, len);
1170 nl = strchr(buf, '\n');
1175 #ifdef HAVE_LIBGNUTLS
1176 if (Conn_OPTION_ISSET(c, CONN_SSL)) {
1177 const char *name_cipher, *name_mac, *name_proto, *name_keyexchange;
1180 gnutls_session_t sess = c->ssl_state.gnutls_session;
1181 gnutls_cipher_algorithm_t cipher = gnutls_cipher_get(sess);
1182 name_cipher = gnutls_cipher_get_name(cipher);
1183 name_mac = gnutls_mac_get_name(gnutls_mac_get(sess));
1184 keysize = gnutls_cipher_get_key_size(cipher) * 8;
1185 name_proto = gnutls_protocol_get_name(gnutls_protocol_get_version(sess));
1186 name_keyexchange = gnutls_kx_get_name(gnutls_kx_get(sess));
1188 return snprintf(buf, len, "%s-%s%15s Kx=%s Enc=%s(%u) Mac=%s",
1189 name_cipher, name_mac, name_proto, name_keyexchange, name_cipher, keysize, name_mac) > 0;
1196 ConnSSL_GetCertFp(CONNECTION *c)
1198 return c->ssl_state.fingerprint;
1202 ConnSSL_SetCertFp(CONNECTION *c, const char *fingerprint)
1205 c->ssl_state.fingerprint = strndup(fingerprint, SHA256_STRING_LEN - 1);
1206 return c->ssl_state.fingerprint != NULL;
1211 ConnSSL_InitLibrary(void)
1216 #endif /* SSL_SUPPORT */