/* * Copyright (C) 2004 Red Hat Inc. * Copyright (C) 2005 Martin Koegler * Copyright (C) 2010 TigerVNC Team * Copyright (C) 2010 m-privacy GmbH * * This is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This software is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this software; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, * USA. */ #ifdef HAVE_CONFIG_H #include <config.h> #endif #ifndef HAVE_GNUTLS #error "This header should not be compiled without HAVE_GNUTLS defined" #endif #include <stdlib.h> #ifndef WIN32 #include <unistd.h> #endif #include <rfb/CSecurityTLS.h> #include <rfb/SSecurityVeNCrypt.h> #include <rfb/CConnection.h> #include <rfb/LogWriter.h> #include <rfb/Exception.h> #include <rfb/UserMsgBox.h> #include <rdr/TLSInStream.h> #include <rdr/TLSOutStream.h> #include <os/os.h> #include <os/print.h> #include <os/tls.h> #include <gnutls/x509.h> /* * GNUTLS 2.6.5 and older didn't have some variables defined so don't use them. * GNUTLS 1.X.X defined LIBGNUTLS_VERSION_NUMBER so treat it as "old" gnutls as * well */ #if (defined(GNUTLS_VERSION_NUMBER) && GNUTLS_VERSION_NUMBER < 0x020606) || \ defined(LIBGNUTLS_VERSION_NUMBER) #define WITHOUT_X509_TIMES #endif /* Ancient GNUTLS... */ #if !defined(GNUTLS_VERSION_NUMBER) && !defined(LIBGNUTLS_VERSION_NUMBER) #define WITHOUT_X509_TIMES #endif using namespace rfb; StringParameter CSecurityTLS::x509ca("x509ca", "X509 CA certificate", "", ConfViewer); StringParameter CSecurityTLS::x509crl("x509crl", "X509 CRL file", "", ConfViewer); static LogWriter vlog("TLS"); static LogWriter vlog_raw("RawTLS"); static void debug_log(int level, const char* str) { vlog_raw.debug("[%d]: %s", level, str); } void CSecurityTLS::initGlobal() { static bool globalInitDone = false; if (!globalInitDone) { gnutls_global_init(); /* 100 means debug log */ if (vlog_raw.getLevel() >= 100) { gnutls_global_set_log_level(10); gnutls_global_set_log_function(debug_log); } globalInitDone = true; } } CSecurityTLS::CSecurityTLS(bool _anon) : session(0), anon_cred(0), anon(_anon), fis(0), fos(0) { cafile = x509ca.getData(); crlfile = x509crl.getData(); } void CSecurityTLS::setDefaults() { char* homeDir = NULL; if (getvnchomedir(&homeDir) == -1) { vlog.error("Could not obtain VNC home directory path"); return; } int len = strlen(homeDir) + 1; CharArray caDefault(len + 11); CharArray crlDefault(len + 12); sprintf(caDefault.buf, "%sx509_ca.pem", homeDir); sprintf(crlDefault.buf, "%s509_crl.pem", homeDir); delete [] homeDir; if (!fileexists(caDefault.buf)) x509ca.setDefaultStr(strdup(caDefault.buf)); if (!fileexists(crlDefault.buf)) x509crl.setDefaultStr(strdup(crlDefault.buf)); } void CSecurityTLS::shutdown(bool needbye) { if (session && needbye) if (gnutls_bye(session, GNUTLS_SHUT_RDWR) != GNUTLS_E_SUCCESS) vlog.error("gnutls_bye failed"); if (anon_cred) { gnutls_anon_free_client_credentials(anon_cred); anon_cred = 0; } if (cert_cred) { gnutls_certificate_free_credentials(cert_cred); cert_cred = 0; } if (session) { gnutls_deinit(session); session = 0; gnutls_global_deinit(); } } CSecurityTLS::~CSecurityTLS() { shutdown(true); if (fis) delete fis; if (fos) delete fos; delete[] cafile; delete[] crlfile; } bool CSecurityTLS::processMsg(CConnection* cc) { rdr::InStream* is = cc->getInStream(); rdr::OutStream* os = cc->getOutStream(); client = cc; initGlobal(); if (!session) { if (!is->checkNoWait(1)) return false; if (is->readU8() == 0) { rdr::U32 result = is->readU32(); CharArray reason; if (result == secResultFailed || result == secResultTooMany) reason.buf = is->readString(); else reason.buf = strDup("Authentication failure (protocol error)"); throw AuthFailureException(reason.buf); } if (gnutls_init(&session, GNUTLS_CLIENT) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_init failed"); if (gnutls_set_default_priority(session) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_set_default_priority failed"); setParam(); } rdr::TLSInStream *tlsis = new rdr::TLSInStream(is, session); rdr::TLSOutStream *tlsos = new rdr::TLSOutStream(os, session); int err; err = gnutls_handshake(session); if (err != GNUTLS_E_SUCCESS) { delete tlsis; delete tlsos; if (!gnutls_error_is_fatal(err)) return false; vlog.error("TLS Handshake failed: %s\n", gnutls_strerror (err)); shutdown(false); throw AuthFailureException("TLS Handshake failed"); } checkSession(); cc->setStreams(fis = tlsis, fos = tlsos); return true; } void CSecurityTLS::setParam() { static const int kx_anon_priority[] = { GNUTLS_KX_ANON_DH, 0 }; static const int kx_priority[] = { GNUTLS_KX_DHE_DSS, GNUTLS_KX_RSA, GNUTLS_KX_DHE_RSA, GNUTLS_KX_SRP, 0 }; if (anon) { if (gnutls_kx_set_priority(session, kx_anon_priority) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_kx_set_priority failed"); if (gnutls_anon_allocate_client_credentials(&anon_cred) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_anon_allocate_client_credentials failed"); if (gnutls_credentials_set(session, GNUTLS_CRD_ANON, anon_cred) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_credentials_set failed"); vlog.debug("Anonymous session has been set"); } else { if (gnutls_kx_set_priority(session, kx_priority) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_kx_set_priority failed"); if (gnutls_certificate_allocate_credentials(&cert_cred) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_certificate_allocate_credentials failed"); if (*cafile && gnutls_certificate_set_x509_trust_file(cert_cred,cafile,GNUTLS_X509_FMT_PEM) < 0) throw AuthFailureException("load of CA cert failed"); /* Load previously saved certs */ char *homeDir = NULL; int err; if (getvnchomedir(&homeDir) == -1) vlog.error("Could not obtain VNC home directory path"); else { CharArray caSave(strlen(homeDir) + 19 + 1); sprintf(caSave.buf, "%sx509_savedcerts.pem", homeDir); delete [] homeDir; err = gnutls_certificate_set_x509_trust_file(cert_cred, caSave.buf, GNUTLS_X509_FMT_PEM); if (err < 0) vlog.debug("Failed to load saved server certificates from %s", caSave.buf); } if (*crlfile && gnutls_certificate_set_x509_crl_file(cert_cred,crlfile,GNUTLS_X509_FMT_PEM) < 0) throw AuthFailureException("load of CRL failed"); if (gnutls_credentials_set(session, GNUTLS_CRD_CERTIFICATE, cert_cred) != GNUTLS_E_SUCCESS) throw AuthFailureException("gnutls_credentials_set failed"); vlog.debug("X509 session has been set"); } } void CSecurityTLS::checkSession() { const unsigned allowed_errors = GNUTLS_CERT_INVALID | GNUTLS_CERT_SIGNER_NOT_FOUND | GNUTLS_CERT_SIGNER_NOT_CA; unsigned int status; const gnutls_datum *cert_list; unsigned int cert_list_size = 0; int err; gnutls_datum info; if (anon) return; if (gnutls_certificate_type_get(session) != GNUTLS_CRT_X509) throw AuthFailureException("unsupported certificate type"); err = gnutls_certificate_verify_peers2(session, &status); if (err != 0) { vlog.error("server certificate verification failed: %s", gnutls_strerror(err)); throw AuthFailureException("server certificate verification failed"); } if (status & GNUTLS_CERT_REVOKED) throw AuthFailureException("server certificate has been revoked"); #ifndef WITHOUT_X509_TIMES if (status & GNUTLS_CERT_NOT_ACTIVATED) throw AuthFailureException("server certificate has not been activated"); if (status & GNUTLS_CERT_EXPIRED) { vlog.debug("server certificate has expired"); if (!msg->showMsgBox(UserMsgBox::M_YESNO, "certificate has expired", "The certificate of the server has expired, " "do you want to continue?")) throw AuthFailureException("server certificate has expired"); } #endif /* Process other errors later */ cert_list = gnutls_certificate_get_peers(session, &cert_list_size); if (!cert_list_size) throw AuthFailureException("empty certificate chain"); /* Process only server's certificate, not issuer's certificate */ gnutls_x509_crt crt; gnutls_x509_crt_init(&crt); if (gnutls_x509_crt_import(crt, &cert_list[0], GNUTLS_X509_FMT_DER) < 0) throw AuthFailureException("decoding of certificate failed"); if (gnutls_x509_crt_check_hostname(crt, client->getServerName()) == 0) { char buf[255]; vlog.debug("hostname mismatch"); snprintf(buf, sizeof(buf), "Hostname (%s) does not match any certificate, " "do you want to continue?", client->getServerName()); buf[sizeof(buf) - 1] = '\0'; if (!msg->showMsgBox(UserMsgBox::M_YESNO, "hostname mismatch", buf)) throw AuthFailureException("hostname mismatch"); } if (status == 0) { /* Everything is fine (hostname + verification) */ gnutls_x509_crt_deinit(crt); return; } if (status & GNUTLS_CERT_INVALID) vlog.debug("server certificate invalid"); if (status & GNUTLS_CERT_SIGNER_NOT_FOUND) vlog.debug("server cert signer not found"); if (status & GNUTLS_CERT_SIGNER_NOT_CA) vlog.debug("server cert signer not CA"); if ((status & (~allowed_errors)) != 0) { /* No other errors are allowed */ vlog.debug("GNUTLS status of certificate verification: %u", status); throw AuthFailureException("Invalid status of server certificate verification"); } vlog.debug("Saved server certificates don't match"); if (gnutls_x509_crt_print(crt, GNUTLS_CRT_PRINT_ONELINE, &info)) { /* * GNUTLS doesn't correctly export gnutls_free symbol which is * a function pointer. Linking with Visual Studio 2008 Express will * fail when you call gnutls_free(). */ #if WIN32 free(info.data); #else gnutls_free(info.data); #endif throw AuthFailureException("Could not find certificate to display"); } size_t out_size = 0; char *out_buf = NULL; char *certinfo = NULL; int len = 0; vlog.debug("certificate issuer unknown"); len = snprintf(NULL, 0, "This certificate has been signed by an unknown " "authority:\n\n%s\n\nDo you want to save it and " "continue?\n ", info.data); if (len < 0) AuthFailureException("certificate decoding error"); vlog.debug("%s", info.data); certinfo = new char[len]; if (certinfo == NULL) throw AuthFailureException("Out of memory"); snprintf(certinfo, len, "This certificate has been signed by an unknown " "authority:\n\n%s\n\nDo you want to save it and " "continue? ", info.data); for (int i = 0; i < len - 1; i++) if (certinfo[i] == ',' && certinfo[i + 1] == ' ') certinfo[i] = '\n'; if (!msg->showMsgBox(UserMsgBox::M_YESNO, "certificate issuer unknown", certinfo)) { delete [] certinfo; throw AuthFailureException("certificate issuer unknown"); } delete [] certinfo; if (gnutls_x509_crt_export(crt, GNUTLS_X509_FMT_PEM, NULL, &out_size) == GNUTLS_E_SHORT_MEMORY_BUFFER) AuthFailureException("Out of memory"); // Save cert out_buf = new char[out_size]; if (out_buf == NULL) AuthFailureException("Out of memory"); if (gnutls_x509_crt_export(crt, GNUTLS_X509_FMT_PEM, out_buf, &out_size) < 0) AuthFailureException("certificate issuer unknown, and certificate " "export failed"); char *homeDir = NULL; if (getvnchomedir(&homeDir) == -1) vlog.error("Could not obtain VNC home directory path"); else { FILE *f; CharArray caSave(strlen(homeDir) + 1 + 19); sprintf(caSave.buf, "%sx509_savedcerts.pem", homeDir); delete [] homeDir; f = fopen(caSave.buf, "a+"); if (!f) msg->showMsgBox(UserMsgBox::M_OK, "certificate save failed", "Could not save the certificate"); else { fprintf(f, "%s\n", out_buf); fclose(f); } } delete [] out_buf; gnutls_x509_crt_deinit(crt); /* * GNUTLS doesn't correctly export gnutls_free symbol which is * a function pointer. Linking with Visual Studio 2008 Express will * fail when you call gnutls_free(). */ #if WIN32 free(info.data); #else gnutls_free(info.data); #endif }