123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269 |
- /***************************************************************************
- * _ _ ____ _
- * Project ___| | | | _ \| |
- * / __| | | | |_) | |
- * | (__| |_| | _ <| |___
- * \___|\___/|_| \_\_____|
- *
- * Copyright (C) 1998 - 2004, Daniel Stenberg, <daniel@haxx.se>, et al.
- *
- * This software is licensed as described in the file COPYING, which
- * you should have received as part of this distribution. The terms
- * are also available at http://curl.haxx.se/docs/copyright.html.
- *
- * You may opt to use, copy, modify, merge, publish, distribute and/or sell
- * copies of the Software, and permit persons to whom the Software is
- * furnished to do so, under the terms of the COPYING file.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- * $Id: ssluse.c,v 1.93 2004/03/11 12:57:04 bagder Exp $
- ***************************************************************************/
- /*
- * The original SSL code for curl was written by
- * Linas Vepstas <linas@linas.org> and Sampo Kellomaki <sampo@iki.fi>
- */
- #include "setup.h"
- #include <string.h>
- #include <stdlib.h>
- #include <ctype.h>
- #ifdef HAVE_SYS_TYPES_H
- #include <sys/types.h>
- #endif
- #ifdef HAVE_SYS_SOCKET_H
- #include <sys/socket.h>
- #endif
- #include "urldata.h"
- #include "sendf.h"
- #include "formdata.h" /* for the boundary function */
- #include "url.h" /* for the ssl config check function */
- #include "inet_pton.h"
- #include "ssluse.h"
- #include "connect.h" /* Curl_ourerrno() proto */
- #ifdef USE_SSLEAY
- #include <openssl/rand.h>
- #include <openssl/x509v3.h>
- /* The last #include file should be: */
- #ifdef CURLDEBUG
- #include "memdebug.h"
- #endif
- #if OPENSSL_VERSION_NUMBER >= 0x0090581fL
- #define HAVE_SSL_GET1_SESSION 1
- #else
- #undef HAVE_SSL_GET1_SESSION
- #endif
- #if OPENSSL_VERSION_NUMBER >= 0x00904100L
- #define HAVE_USERDATA_IN_PWD_CALLBACK 1
- #else
- #undef HAVE_USERDATA_IN_PWD_CALLBACK
- #endif
- #if OPENSSL_VERSION_NUMBER >= 0x00907001L
- /* ENGINE_load_private_key() takes four arguments */
- #define HAVE_ENGINE_LOAD_FOUR_ARGS
- #else
- /* ENGINE_load_private_key() takes three arguments */
- #undef HAVE_ENGINE_LOAD_FOUR_ARGS
- #endif
- #ifndef HAVE_USERDATA_IN_PWD_CALLBACK
- static char global_passwd[64];
- #endif
- static int passwd_callback(char *buf, int num, int verify
- #if HAVE_USERDATA_IN_PWD_CALLBACK
- /* This was introduced in 0.9.4, we can set this
- using SSL_CTX_set_default_passwd_cb_userdata()
- */
- , void *global_passwd
- #endif
- )
- {
- if(verify)
- fprintf(stderr, "%s\n", buf);
- else {
- if(num > (int)strlen((char *)global_passwd)) {
- strcpy(buf, global_passwd);
- return strlen(buf);
- }
- }
- return 0;
- }
- static
- bool seed_enough(int nread)
- {
- #ifdef HAVE_RAND_STATUS
- nread = 0; /* to prevent compiler warnings */
- /* only available in OpenSSL 0.9.5a and later */
- if(RAND_status())
- return TRUE;
- #else
- if(nread > 500)
- /* this is a very silly decision to make */
- return TRUE;
- #endif
- return FALSE; /* not enough */
- }
- static
- int random_the_seed(struct SessionHandle *data)
- {
- char *buf = data->state.buffer; /* point to the big buffer */
- int nread=0;
- /* Q: should we add support for a random file name as a libcurl option?
- A: Yes, it is here */
- #ifndef RANDOM_FILE
- /* if RANDOM_FILE isn't defined, we only perform this if an option tells
- us to! */
- if(data->set.ssl.random_file)
- #define RANDOM_FILE "" /* doesn't matter won't be used */
- #endif
- {
- /* let the option override the define */
- nread += RAND_load_file((data->set.ssl.random_file?
- data->set.ssl.random_file:RANDOM_FILE),
- 16384);
- if(seed_enough(nread))
- return nread;
- }
- #if defined(HAVE_RAND_EGD)
- /* only available in OpenSSL 0.9.5 and later */
- /* EGD_SOCKET is set at configure time or not at all */
- #ifndef EGD_SOCKET
- /* If we don't have the define set, we only do this if the egd-option
- is set */
- if(data->set.ssl.egdsocket)
- #define EGD_SOCKET "" /* doesn't matter won't be used */
- #endif
- {
- /* If there's an option and a define, the option overrides the
- define */
- int ret = RAND_egd(data->set.ssl.egdsocket?
- data->set.ssl.egdsocket:EGD_SOCKET);
- if(-1 != ret) {
- nread += ret;
- if(seed_enough(nread))
- return nread;
- }
- }
- #endif
- /* If we get here, it means we need to seed the PRNG using a "silly"
- approach! */
- #ifdef HAVE_RAND_SCREEN
- /* This one gets a random value by reading the currently shown screen */
- RAND_screen();
- nread = 100; /* just a value */
- #else
- {
- int len;
- char *area;
- /* Changed call to RAND_seed to use the underlying RAND_add implementation
- * directly. Do this in a loop, with the amount of additional entropy
- * being dependent upon the algorithm used by Curl_FormBoundary(): N bytes
- * of a 7-bit ascii set. -- Richard Gorton, March 11 2003.
- */
-
- do {
- area = Curl_FormBoundary();
- if(!area)
- return 3; /* out of memory */
-
- len = strlen(area);
- RAND_add(area, len, (len >> 1));
- free(area); /* now remove the random junk */
- } while (!RAND_status());
- }
- #endif
- /* generates a default path for the random seed file */
- buf[0]=0; /* blank it first */
- RAND_file_name(buf, BUFSIZE);
- if(buf[0]) {
- /* we got a file name to try */
- nread += RAND_load_file(buf, 16384);
- if(seed_enough(nread))
- return nread;
- }
- infof(data, "libcurl is now using a weak random seed!\n");
- return nread;
- }
- #ifndef SSL_FILETYPE_ENGINE
- #define SSL_FILETYPE_ENGINE 42
- #endif
- static int do_file_type(const char *type)
- {
- if(!type || !type[0])
- return SSL_FILETYPE_PEM;
- if(curl_strequal(type, "PEM"))
- return SSL_FILETYPE_PEM;
- if(curl_strequal(type, "DER"))
- return SSL_FILETYPE_ASN1;
- if(curl_strequal(type, "ENG"))
- return SSL_FILETYPE_ENGINE;
- return -1;
- }
- static
- int cert_stuff(struct connectdata *conn,
- SSL_CTX* ctx,
- char *cert_file,
- const char *cert_type,
- char *key_file,
- const char *key_type)
- {
- struct SessionHandle *data = conn->data;
- int file_type;
- if(cert_file != NULL) {
- SSL *ssl;
- X509 *x509;
- if(data->set.key_passwd) {
- #ifndef HAVE_USERDATA_IN_PWD_CALLBACK
- /*
- * If password has been given, we store that in the global
- * area (*shudder*) for a while:
- */
- strcpy(global_passwd, data->set.key_passwd);
- #else
- /*
- * We set the password in the callback userdata
- */
- SSL_CTX_set_default_passwd_cb_userdata(ctx,
- data->set.key_passwd);
- #endif
- /* Set passwd callback: */
- SSL_CTX_set_default_passwd_cb(ctx, passwd_callback);
- }
- file_type = do_file_type(cert_type);
- switch(file_type) {
- case SSL_FILETYPE_PEM:
- /* SSL_CTX_use_certificate_chain_file() only works on PEM files */
- if(SSL_CTX_use_certificate_chain_file(ctx,
- cert_file) != 1) {
- failf(data, "unable to set certificate file (wrong password?)");
- return 0;
- }
- break;
- case SSL_FILETYPE_ASN1:
- /* SSL_CTX_use_certificate_file() works with either PEM or ASN1, but
- we use the case above for PEM so this can only be performed with
- ASN1 files. */
- if(SSL_CTX_use_certificate_file(ctx,
- cert_file,
- file_type) != 1) {
- failf(data, "unable to set certificate file (wrong password?)");
- return 0;
- }
- break;
- case SSL_FILETYPE_ENGINE:
- failf(data, "file type ENG for certificate not implemented");
- return 0;
- default:
- failf(data, "not supported file type '%s' for certificate", cert_type);
- return 0;
- }
- file_type = do_file_type(key_type);
- switch(file_type) {
- case SSL_FILETYPE_PEM:
- if(key_file == NULL)
- /* cert & key can only be in PEM case in the same file */
- key_file=cert_file;
- case SSL_FILETYPE_ASN1:
- if(SSL_CTX_use_PrivateKey_file(ctx, key_file, file_type) != 1) {
- failf(data, "unable to set private key file: '%s' type %s\n",
- key_file, key_type?key_type:"PEM");
- return 0;
- }
- break;
- case SSL_FILETYPE_ENGINE:
- #ifdef HAVE_OPENSSL_ENGINE_H
- { /* XXXX still needs some work */
- EVP_PKEY *priv_key = NULL;
- if(conn && conn->data && conn->data->engine) {
- #ifdef HAVE_ENGINE_LOAD_FOUR_ARGS
- UI_METHOD *ui_method = UI_OpenSSL();
- #endif
- if(!key_file || !key_file[0]) {
- failf(data, "no key set to load from crypto engine\n");
- return 0;
- }
- priv_key = ENGINE_load_private_key(conn->data->engine,key_file,
- #ifdef HAVE_ENGINE_LOAD_FOUR_ARGS
- ui_method,
- #endif
- data->set.key_passwd);
- if(!priv_key) {
- failf(data, "failed to load private key from crypto engine\n");
- return 0;
- }
- if(SSL_CTX_use_PrivateKey(ctx, priv_key) != 1) {
- failf(data, "unable to set private key\n");
- EVP_PKEY_free(priv_key);
- return 0;
- }
- EVP_PKEY_free(priv_key); /* we don't need the handle any more... */
- }
- else {
- failf(data, "crypto engine not set, can't load private key\n");
- return 0;
- }
- }
- #else
- failf(data, "file type ENG for private key not supported\n");
- return 0;
- #endif
- break;
- default:
- failf(data, "not supported file type for private key\n");
- return 0;
- }
- ssl=SSL_new(ctx);
- x509=SSL_get_certificate(ssl);
- /* This version was provided by Evan Jordan and is supposed to not
- leak memory as the previous version: */
- if(x509 != NULL) {
- EVP_PKEY *pktmp = X509_get_pubkey(x509);
- EVP_PKEY_copy_parameters(pktmp,SSL_get_privatekey(ssl));
- EVP_PKEY_free(pktmp);
- }
- SSL_free(ssl);
- /* If we are using DSA, we can copy the parameters from
- * the private key */
-
-
- /* Now we know that a key and cert have been set against
- * the SSL context */
- if(!SSL_CTX_check_private_key(ctx)) {
- failf(data, "Private key does not match the certificate public key");
- return(0);
- }
- #ifndef HAVE_USERDATA_IN_PWD_CALLBACK
- /* erase it now */
- memset(global_passwd, 0, sizeof(global_passwd));
- #endif
- }
- return(1);
- }
- static
- int cert_verify_callback(int ok, X509_STORE_CTX *ctx)
- {
- X509 *err_cert;
- char buf[256];
- err_cert=X509_STORE_CTX_get_current_cert(ctx);
- X509_NAME_oneline(X509_get_subject_name(err_cert),buf,256);
- return ok;
- }
- #endif
- #ifdef USE_SSLEAY
- /* "global" init done? */
- static int init_ssl=0;
- /* we have the "SSL is seeded" boolean global for the application to
- prevent multiple time-consuming seedings in vain */
- static bool ssl_seeded = FALSE;
- #endif
- /* Global init */
- void Curl_SSL_init(void)
- {
- #ifdef USE_SSLEAY
- /* make sure this is only done once */
- if(0 != init_ssl)
- return;
- init_ssl++; /* never again */
- #ifdef HAVE_ENGINE_LOAD_BUILTIN_ENGINES
- ENGINE_load_builtin_engines();
- #endif
- /* Lets get nice error messages */
- SSL_load_error_strings();
- /* Setup all the global SSL stuff */
- SSLeay_add_ssl_algorithms();
- #else
- /* SSL disabled, do nothing */
- #endif
- }
- /* Global cleanup */
- void Curl_SSL_cleanup(void)
- {
- #ifdef USE_SSLEAY
- if(init_ssl) {
- /* only cleanup if we did a previous init */
- /* Free the SSL error strings */
- ERR_free_strings();
-
- /* EVP_cleanup() removes all ciphers and digests from the
- table. */
- EVP_cleanup();
- #ifdef HAVE_ENGINE_cleanup
- ENGINE_cleanup();
- #endif
- #ifdef HAVE_CRYPTO_CLEANUP_ALL_EX_DATA
- /* this function was not present in 0.9.6b, but was added sometimes
- later */
- CRYPTO_cleanup_all_ex_data();
- #endif
- init_ssl=0; /* not inited any more */
- }
- #else
- /* SSL disabled, do nothing */
- #endif
- }
- #ifndef USE_SSLEAY
- void Curl_SSL_Close(struct connectdata *conn)
- {
- (void)conn;
- }
- #endif
- #ifdef USE_SSLEAY
- /*
- * This function is called when an SSL connection is closed.
- */
- void Curl_SSL_Close(struct connectdata *conn)
- {
- if(conn->ssl[FIRSTSOCKET].use) {
- int i;
- /*
- ERR_remove_state() frees the error queue associated with
- thread pid. If pid == 0, the current thread will have its
- error queue removed.
- Since error queue data structures are allocated
- automatically for new threads, they must be freed when
- threads are terminated in oder to avoid memory leaks.
- */
- ERR_remove_state(0);
- for(i=0; i<2; i++) {
- struct ssl_connect_data *connssl = &conn->ssl[i];
-
- if(connssl->handle) {
- (void)SSL_shutdown(connssl->handle);
- SSL_set_connect_state(connssl->handle);
-
- SSL_free (connssl->handle);
- connssl->handle = NULL;
- }
- if(connssl->ctx) {
- SSL_CTX_free (connssl->ctx);
- connssl->ctx = NULL;
- }
- connssl->use = FALSE; /* get back to ordinary socket usage */
- }
- }
- }
- /*
- * This sets up a session cache to the specified size.
- */
- CURLcode Curl_SSL_InitSessions(struct SessionHandle *data, long amount)
- {
- struct curl_ssl_session *session;
- if(data->state.session)
- /* this is just a precaution to prevent multiple inits */
- return CURLE_OK;
- session = (struct curl_ssl_session *)
- malloc(amount * sizeof(struct curl_ssl_session));
- if(!session)
- return CURLE_OUT_OF_MEMORY;
- /* "blank out" the newly allocated memory */
- memset(session, 0, amount * sizeof(struct curl_ssl_session));
- /* store the info in the SSL section */
- data->set.ssl.numsessions = amount;
- data->state.session = session;
- data->state.sessionage = 1; /* this is brand new */
- return CURLE_OK;
- }
- /*
- * Check if there's a session ID for the given connection in the cache,
- * and if there's one suitable, it is returned.
- */
- static int Get_SSL_Session(struct connectdata *conn,
- SSL_SESSION **ssl_sessionid)
- {
- struct curl_ssl_session *check;
- struct SessionHandle *data = conn->data;
- long i;
- for(i=0; i< data->set.ssl.numsessions; i++) {
- check = &data->state.session[i];
- if(!check->sessionid)
- /* not session ID means blank entry */
- continue;
- if(curl_strequal(conn->name, check->name) &&
- (conn->remote_port == check->remote_port) &&
- Curl_ssl_config_matches(&conn->ssl_config, &check->ssl_config)) {
- /* yes, we have a session ID! */
- data->state.sessionage++; /* increase general age */
- check->age = data->state.sessionage; /* set this as used in this age */
- *ssl_sessionid = check->sessionid;
- return FALSE;
- }
- }
- *ssl_sessionid = (SSL_SESSION *)NULL;
- return TRUE;
- }
- /*
- * Kill a single session ID entry in the cache.
- */
- static int Kill_Single_Session(struct curl_ssl_session *session)
- {
- if(session->sessionid) {
- /* defensive check */
- /* free the ID */
- SSL_SESSION_free(session->sessionid);
- session->sessionid=NULL;
- session->age = 0; /* fresh */
- Curl_free_ssl_config(&session->ssl_config);
- free(session->name);
- session->name = NULL; /* no name */
- return 0; /* ok */
- }
- else
- return 1;
- }
- /*
- * This function is called when the 'data' struct is going away. Close
- * down everything and free all resources!
- */
- int Curl_SSL_Close_All(struct SessionHandle *data)
- {
- int i;
- if(data->state.session) {
- for(i=0; i< data->set.ssl.numsessions; i++)
- /* the single-killer function handles empty table slots */
- Kill_Single_Session(&data->state.session[i]);
-
- /* free the cache data */
- free(data->state.session);
- }
- #ifdef HAVE_OPENSSL_ENGINE_H
- if(data->engine)
- {
- ENGINE_free(data->engine);
- data->engine = NULL;
- }
- #endif
- return 0;
- }
- /*
- * Extract the session id and store it in the session cache.
- */
- static int Store_SSL_Session(struct connectdata *conn,
- struct ssl_connect_data *ssl)
- {
- SSL_SESSION *ssl_sessionid;
- int i;
- struct SessionHandle *data=conn->data; /* the mother of all structs */
- struct curl_ssl_session *store = &data->state.session[0];
- int oldest_age=data->state.session[0].age; /* zero if unused */
- /* ask OpenSSL, say please */
- #ifdef HAVE_SSL_GET1_SESSION
- ssl_sessionid = SSL_get1_session(ssl->handle);
- /* SSL_get1_session() will increment the reference
- count and the session will stay in memory until explicitly freed with
- SSL_SESSION_free(3), regardless of its state.
- This function was introduced in openssl 0.9.5a. */
- #else
- ssl_sessionid = SSL_get_session(ssl->handle);
- /* if SSL_get1_session() is unavailable, use SSL_get_session().
- This is an inferior option because the session can be flushed
- at any time by openssl. It is included only so curl compiles
- under versions of openssl < 0.9.5a.
-
- WARNING: How curl behaves if it's session is flushed is
- untested.
- */
- #endif
- /* Now we should add the session ID and the host name to the cache, (remove
- the oldest if necessary) */
- /* find an empty slot for us, or find the oldest */
- for(i=1; (i<data->set.ssl.numsessions) &&
- data->state.session[i].sessionid; i++) {
- if(data->state.session[i].age < oldest_age) {
- oldest_age = data->state.session[i].age;
- store = &data->state.session[i];
- }
- }
- if(i == data->set.ssl.numsessions)
- /* cache is full, we must "kill" the oldest entry! */
- Kill_Single_Session(store);
- else
- store = &data->state.session[i]; /* use this slot */
-
- /* now init the session struct wisely */
- store->sessionid = ssl_sessionid;
- store->age = data->state.sessionage; /* set current age */
- store->name = strdup(conn->name); /* clone host name */
- store->remote_port = conn->remote_port; /* port number */
- Curl_clone_ssl_config(&conn->ssl_config, &store->ssl_config);
- return 0;
- }
- static int Curl_ASN1_UTCTIME_output(struct connectdata *conn,
- const char *prefix,
- ASN1_UTCTIME *tm)
- {
- char *asn1_string;
- int gmt=FALSE;
- int i;
- int year=0,month=0,day=0,hour=0,minute=0,second=0;
- struct SessionHandle *data = conn->data;
- if(!data->set.verbose)
- return 0;
- i=tm->length;
- asn1_string=(char *)tm->data;
- if(i < 10)
- return 1;
- if(asn1_string[i-1] == 'Z')
- gmt=TRUE;
- for (i=0; i<10; i++)
- if((asn1_string[i] > '9') || (asn1_string[i] < '0'))
- return 2;
- year= (asn1_string[0]-'0')*10+(asn1_string[1]-'0');
- if(year < 50)
- year+=100;
- month= (asn1_string[2]-'0')*10+(asn1_string[3]-'0');
- if((month > 12) || (month < 1))
- return 3;
- day= (asn1_string[4]-'0')*10+(asn1_string[5]-'0');
- hour= (asn1_string[6]-'0')*10+(asn1_string[7]-'0');
- minute= (asn1_string[8]-'0')*10+(asn1_string[9]-'0');
- if((asn1_string[10] >= '0') && (asn1_string[10] <= '9') &&
- (asn1_string[11] >= '0') && (asn1_string[11] <= '9'))
- second= (asn1_string[10]-'0')*10+(asn1_string[11]-'0');
-
- infof(data,
- "%s%04d-%02d-%02d %02d:%02d:%02d %s\n",
- prefix, year+1900, month, day, hour, minute, second, (gmt?"GMT":""));
- return 0;
- }
- #endif
- /* ====================================================== */
- #ifdef USE_SSLEAY
- static int
- cert_hostcheck(const char *certname, const char *hostname)
- {
- char *tmp;
- const char *certdomain;
-
- if(!certname ||
- strlen(certname)<3 ||
- !hostname ||
- !strlen(hostname)) /* sanity check */
- return 0;
- if(curl_strequal(certname, hostname)) /* trivial case */
- return 1;
- certdomain = certname + 1;
- if((certname[0] != '*') || (certdomain[0] != '.'))
- return 0; /* not a wildcard certificate, check failed */
-
- if(!strchr(certdomain+1, '.'))
- return 0; /* the certificate must have at least another dot in its name */
- /* find 'certdomain' within 'hostname' */
- tmp = strstr(hostname, certdomain);
- if(tmp) {
- /* ok the certname's domain matches the hostname, let's check that it's a
- tail-match */
- if(curl_strequal(tmp, certdomain))
- /* looks like a match. Just check we havent swallowed a '.' */
- return tmp == strchr(hostname, '.');
- else
- return 0;
- }
- return 0;
- }
- /* Quote from RFC2818 section 3.1 "Server Identity"
- If a subjectAltName extension of type dNSName is present, that MUST
- be used as the identity. Otherwise, the (most specific) Common Name
- field in the Subject field of the certificate MUST be used. Although
- the use of the Common Name is existing practice, it is deprecated and
- Certification Authorities are encouraged to use the dNSName instead.
- Matching is performed using the matching rules specified by
- [RFC2459]. If more than one identity of a given type is present in
- the certificate (e.g., more than one dNSName name, a match in any one
- of the set is considered acceptable.) Names may contain the wildcard
- character * which is considered to match any single domain name
- component or component fragment. E.g., *.a.com matches foo.a.com but
- not bar.foo.a.com. f*.com matches foo.com but not bar.com.
- In some cases, the URI is specified as an IP address rather than a
- hostname. In this case, the iPAddress subjectAltName must be present
- in the certificate and must exactly match the IP in the URI.
- */
- static CURLcode verifyhost(struct connectdata *conn,
- X509 *server_cert)
- {
- char peer_CN[257];
- bool matched = FALSE; /* no alternative match yet */
- int target = GEN_DNS; /* target type, GEN_DNS or GEN_IPADD */
- int addrlen = 0;
- struct SessionHandle *data = conn->data;
- STACK_OF(GENERAL_NAME) *altnames;
- #ifdef ENABLE_IPV6
- struct in6_addr addr;
- #else
- struct in_addr addr;
- #endif
-
- #ifdef ENABLE_IPV6
- if(conn->bits.ipv6_ip &&
- Curl_inet_pton(AF_INET6, conn->hostname, &addr)) {
- target = GEN_IPADD;
- addrlen = sizeof(struct in6_addr);
- }
- else
- #endif
- if(Curl_inet_pton(AF_INET, conn->hostname, &addr)) {
- target = GEN_IPADD;
- addrlen = sizeof(struct in_addr);
- }
-
- /* get a "list" of alternative names */
- altnames = X509_get_ext_d2i(server_cert, NID_subject_alt_name, NULL, NULL);
-
- if(altnames) {
- int hostlen = 0;
- int domainlen = 0;
- char *domain = NULL;
- int numalts;
- int i;
-
- if(GEN_DNS == target) {
- hostlen = strlen(conn->hostname);
- domain = strchr(conn->hostname, '.');
- if(domain)
- domainlen = strlen(domain);
- }
- /* get amount of alternatives, RFC2459 claims there MUST be at least
- one, but we don't depend on it... */
- numalts = sk_GENERAL_NAME_num(altnames);
- /* loop through all alternatives while none has matched */
- for (i=0; (i<numalts) && !matched; i++) {
- /* get a handle to alternative name number i */
- const GENERAL_NAME *check = sk_GENERAL_NAME_value(altnames, i);
- /* only check alternatives of the same type the target is */
- if(check->type == target) {
- /* get data and length */
- const char *altptr = (char *)ASN1_STRING_data(check->d.ia5);
- const int altlen = ASN1_STRING_length(check->d.ia5);
- switch(target) {
- case GEN_DNS: /* name comparison */
- /* Is this an exact match? */
- if((hostlen == altlen) &&
- curl_strnequal(conn->hostname, altptr, hostlen))
- matched = TRUE;
-
- /* Is this a wildcard match? */
- else if((altptr[0] == '*') &&
- (domainlen == altlen-1) &&
- curl_strnequal(domain, altptr+1, domainlen))
- matched = TRUE;
- break;
-
- case GEN_IPADD: /* IP address comparison */
- /* compare alternative IP address if the data chunk is the same size
- our server IP address is */
- if((altlen == addrlen) && !memcmp(altptr, &addr, altlen))
- matched = TRUE;
- break;
- }
- }
- }
- GENERAL_NAMES_free(altnames);
- }
-
- if(matched)
- /* an alternative name matched the server hostname */
- infof(data, "\t subjectAltName: %s matched\n", conn->hostname);
- else {
- bool obtain=FALSE;
- if(X509_NAME_get_text_by_NID(X509_get_subject_name(server_cert),
- NID_commonName,
- peer_CN,
- sizeof(peer_CN)) < 0) {
- if(data->set.ssl.verifyhost > 1) {
- failf(data,
- "SSL: unable to obtain common name from peer certificate");
- return CURLE_SSL_PEER_CERTIFICATE;
- }
- else {
- /* Consider verifyhost == 1 as an "OK" for a missing CN field, but we
- output a note about the situation */
- infof(data, "\t common name: WARNING couldn't obtain\n");
- }
- }
- else
- obtain = TRUE;
-
- if(obtain) {
- if(!cert_hostcheck(peer_CN, conn->hostname)) {
- if(data->set.ssl.verifyhost > 1) {
- failf(data, "SSL: certificate subject name '%s' does not match "
- "target host name '%s'", peer_CN, conn->hostname);
- return CURLE_SSL_PEER_CERTIFICATE;
- }
- else
- infof(data, "\t common name: %s (does not match '%s')\n",
- peer_CN, conn->hostname);
- }
- else
- infof(data, "\t common name: %s (matched)\n", peer_CN);
- }
- }
- return CURLE_OK;
- }
- #endif
- /* ====================================================== */
- CURLcode
- Curl_SSLConnect(struct connectdata *conn,
- int sockindex)
- {
- CURLcode retcode = CURLE_OK;
- #ifdef USE_SSLEAY
- struct SessionHandle *data = conn->data;
- int err;
- int what;
- char * str;
- SSL_METHOD *req_method;
- SSL_SESSION *ssl_sessionid=NULL;
- ASN1_TIME *certdate;
- curl_socket_t sockfd = conn->sock[sockindex];
- struct ssl_connect_data *connssl = &conn->ssl[sockindex];
- /* mark this is being ssl enabled from here on out. */
- connssl->use = TRUE;
- if(!ssl_seeded || data->set.ssl.random_file || data->set.ssl.egdsocket) {
- /* Make funny stuff to get random input */
- random_the_seed(data);
- ssl_seeded = TRUE;
- }
- /* check to see if we've been told to use an explicit SSL/TLS version */
- switch(data->set.ssl.version) {
- default:
- case CURL_SSLVERSION_DEFAULT:
- /* we try to figure out version */
- req_method = SSLv23_client_method();
- break;
- case CURL_SSLVERSION_TLSv1:
- req_method = TLSv1_client_method();
- break;
- case CURL_SSLVERSION_SSLv2:
- req_method = SSLv2_client_method();
- break;
- case CURL_SSLVERSION_SSLv3:
- req_method = SSLv3_client_method();
- break;
- }
-
- connssl->ctx = SSL_CTX_new(req_method);
- if(!connssl->ctx) {
- failf(data, "SSL: couldn't create a context!");
- return CURLE_OUT_OF_MEMORY;
- }
- /* OpenSSL contains code to work-around lots of bugs and flaws in various
- SSL-implementations. SSL_CTX_set_options() is used to enabled those
- work-arounds. The man page for this option states that SSL_OP_ALL enables
- ll the work-arounds and that "It is usually safe to use SSL_OP_ALL to
- enable the bug workaround options if compatibility with somewhat broken
- implementations is desired."
- */
- SSL_CTX_set_options(connssl->ctx, SSL_OP_ALL);
-
- if(data->set.cert) {
- if(!cert_stuff(conn,
- connssl->ctx,
- data->set.cert,
- data->set.cert_type,
- data->set.key,
- data->set.key_type)) {
- /* failf() is already done in cert_stuff() */
- return CURLE_SSL_CERTPROBLEM;
- }
- }
- if(data->set.ssl.cipher_list) {
- if(!SSL_CTX_set_cipher_list(connssl->ctx,
- data->set.ssl.cipher_list)) {
- failf(data, "failed setting cipher list");
- return CURLE_SSL_CIPHER;
- }
- }
- if (data->set.ssl.CAfile || data->set.ssl.CApath) {
- /* tell SSL where to find CA certificates that are used to verify
- the servers certificate. */
- if (!SSL_CTX_load_verify_locations(connssl->ctx, data->set.ssl.CAfile,
- data->set.ssl.CApath)) {
- if (data->set.ssl.verifypeer) {
- /* Fail if we insist on successfully verifying the server. */
- failf(data,"error setting certificate verify locations:\n"
- " CAfile: %s\n CApath: %s\n",
- data->set.ssl.CAfile ? data->set.ssl.CAfile : "none",
- data->set.ssl.CApath ? data->set.ssl.CApath : "none");
- return CURLE_SSL_CACERT;
- }
- else {
- /* Just continue with a warning if no strict certificate verification
- is required. */
- infof(data, "error setting certificate verify locations,"
- " continuing anyway:\n");
- }
- }
- else {
- /* Everything is fine. */
- infof(data, "successfully set certificate verify locations:\n");
- }
- infof(data,
- " CAfile: %s\n"
- " CApath: %s\n",
- data->set.ssl.CAfile ? data->set.ssl.CAfile : "none",
- data->set.ssl.CApath ? data->set.ssl.CApath : "none");
- }
- /* SSL always tries to verify the peer, this only says whether it should
- * fail to connect if the verification fails, or if it should continue
- * anyway. In the latter case the result of the verification is checked with
- * SSL_get_verify_result() below. */
- SSL_CTX_set_verify(connssl->ctx,
- data->set.ssl.verifypeer?SSL_VERIFY_PEER:SSL_VERIFY_NONE,
- cert_verify_callback);
- /* give application a chance to interfere with SSL set up. */
- if(data->set.ssl.fsslctx) {
- retcode = (*data->set.ssl.fsslctx)(data, connssl->ctx,
- data->set.ssl.fsslctxp);
- if(retcode) {
- failf(data,"error signaled by ssl ctx callback");
- return retcode;
- }
- }
- /* Lets make an SSL structure */
- connssl->handle = SSL_new(connssl->ctx);
- SSL_set_connect_state(connssl->handle);
- connssl->server_cert = 0x0;
- if(!conn->bits.reuse) {
- /* We're not re-using a connection, check if there's a cached ID we
- can/should use here! */
- if(!Get_SSL_Session(conn, &ssl_sessionid)) {
- /* we got a session id, use it! */
- SSL_set_session(connssl->handle, ssl_sessionid);
- /* Informational message */
- infof (data, "SSL re-using session ID\n");
- }
- }
- /* pass the raw socket into the SSL layers */
- SSL_set_fd(connssl->handle, sockfd);
- while(1) {
- fd_set writefd;
- fd_set readfd;
- struct timeval interval;
- long timeout_ms;
- /* Find out if any timeout is set. If not, use 300 seconds.
- Otherwise, figure out the most strict timeout of the two possible one
- and then how much time that has elapsed to know how much time we
- allow for the connect call */
- if(data->set.timeout || data->set.connecttimeout) {
- double has_passed;
- /* Evaluate in milliseconds how much time that has passed */
- has_passed = Curl_tvdiff(Curl_tvnow(), data->progress.start);
- #ifndef min
- #define min(a, b) ((a) < (b) ? (a) : (b))
- #endif
- /* get the most strict timeout of the ones converted to milliseconds */
- if(data->set.timeout &&
- (data->set.timeout>data->set.connecttimeout))
- timeout_ms = data->set.timeout*1000;
- else
- timeout_ms = data->set.connecttimeout*1000;
-
- /* subtract the passed time */
- timeout_ms -= (long)has_passed;
-
- if(timeout_ms < 0) {
- /* a precaution, no need to continue if time already is up */
- failf(data, "SSL connection timeout");
- return CURLE_OPERATION_TIMEOUTED;
- }
- }
- else
- /* no particular time-out has been set */
- timeout_ms=300000; /* milliseconds, default to five minutes */
- FD_ZERO(&writefd);
- FD_ZERO(&readfd);
- err = SSL_connect(connssl->handle);
- /* 1 is fine
- 0 is "not successful but was shut down controlled"
- <0 is "handshake was not successful, because a fatal error occurred" */
- if(1 != err) {
- int detail = SSL_get_error(connssl->handle, err);
- if(SSL_ERROR_WANT_READ == detail)
- FD_SET(sockfd, &readfd);
- else if(SSL_ERROR_WANT_WRITE == detail)
- FD_SET(sockfd, &writefd);
- else {
- /* untreated error */
- char error_buffer[120]; /* OpenSSL documents that this must be at least
- 120 bytes long. */
- detail = ERR_get_error(); /* Gets the earliest error code from the
- thread's error queue and removes the
- entry. */
- switch(detail) {
- case 0x1407E086:
- /* 1407E086:
- SSL routines:
- SSL2_SET_CERTIFICATE:
- certificate verify failed */
- case 0x14090086:
- /* 14090086:
- SSL routines:
- SSL3_GET_SERVER_CERTIFICATE:
- certificate verify failed */
- failf(data,
- "SSL certificate problem, verify that the CA cert is OK");
- return CURLE_SSL_CACERT;
- default:
- /* detail is already set to the SSL error above */
- failf(data, "SSL: %s", ERR_error_string(detail, error_buffer));
- /* OpenSSL 0.9.6 and later has a function named
- ERRO_error_string_n() that takes the size of the buffer as a third
- argument, and we should possibly switch to using that one in the
- future. */
- return CURLE_SSL_CONNECT_ERROR;
- }
- }
- }
- else
- /* we have been connected fine, get out of the connect loop */
- break;
- interval.tv_sec = timeout_ms/1000;
- timeout_ms -= interval.tv_sec*1000;
- interval.tv_usec = timeout_ms*1000;
- while(1) {
- what = select(sockfd+1, &readfd, &writefd, NULL, &interval);
- if(what > 0)
- /* reabable or writable, go loop in the outer loop */
- break;
- else if(0 == what) {
- /* timeout */
- failf(data, "SSL connection timeout");
- return CURLE_OPERATION_TIMEDOUT;
- }
- else {
- #if !defined(WIN32) && defined(EINTR)
- /* For platforms without EINTR all errnos are bad */
- if (errno == EINTR)
- continue; /* retry the select() */
- #endif
- /* anything other than the unimportant EINTR is fatally bad */
- failf(data, "select on SSL socket, errno: %d", Curl_ourerrno());
- return CURLE_SSL_CONNECT_ERROR;
- }
- } /* while()-loop for the select() */
- } /* while()-loop for the SSL_connect() */
- /* Informational message */
- infof (data, "SSL connection using %s\n",
- SSL_get_cipher(connssl->handle));
- if(!ssl_sessionid) {
- /* Since this is not a cached session ID, then we want to stach this one
- in the cache! */
- Store_SSL_Session(conn, connssl);
- }
-
- /* Get server's certificate (note: beware of dynamic allocation) - opt */
- /* major serious hack alert -- we should check certificates
- * to authenticate the server; otherwise we risk man-in-the-middle
- * attack
- */
- connssl->server_cert = SSL_get_peer_certificate(connssl->handle);
- if(!connssl->server_cert) {
- failf(data, "SSL: couldn't get peer certificate!");
- return CURLE_SSL_PEER_CERTIFICATE;
- }
- infof (data, "Server certificate:\n");
-
- str = X509_NAME_oneline(X509_get_subject_name(connssl->server_cert),
- NULL, 0);
- if(!str) {
- failf(data, "SSL: couldn't get X509-subject!");
- X509_free(connssl->server_cert);
- return CURLE_SSL_CONNECT_ERROR;
- }
- infof(data, "\t subject: %s\n", str);
- CRYPTO_free(str);
- certdate = X509_get_notBefore(connssl->server_cert);
- Curl_ASN1_UTCTIME_output(conn, "\t start date: ", certdate);
- certdate = X509_get_notAfter(connssl->server_cert);
- Curl_ASN1_UTCTIME_output(conn, "\t expire date: ", certdate);
- if(data->set.ssl.verifyhost) {
- retcode = verifyhost(conn, connssl->server_cert);
- if(retcode) {
- X509_free(connssl->server_cert);
- return retcode;
- }
- }
- str = X509_NAME_oneline(X509_get_issuer_name(connssl->server_cert),
- NULL, 0);
- if(!str) {
- failf(data, "SSL: couldn't get X509-issuer name!");
- retcode = CURLE_SSL_CONNECT_ERROR;
- }
- else {
- infof(data, "\t issuer: %s\n", str);
- CRYPTO_free(str);
- /* We could do all sorts of certificate verification stuff here before
- deallocating the certificate. */
-
- data->set.ssl.certverifyresult=SSL_get_verify_result(connssl->handle);
- if(data->set.ssl.certverifyresult != X509_V_OK) {
- if(data->set.ssl.verifypeer) {
- /* We probably never reach this, because SSL_connect() will fail
- and we return earlyer if verifypeer is set? */
- failf(data, "SSL certificate verify result: %d",
- data->set.ssl.certverifyresult);
- retcode = CURLE_SSL_PEER_CERTIFICATE;
- }
- else
- infof(data, "SSL certificate verify result: %d, continuing anyway.\n",
- data->set.ssl.certverifyresult);
- }
- else
- infof(data, "SSL certificate verify ok.\n");
- }
- X509_free(connssl->server_cert);
- #else /* USE_SSLEAY */
- (void)conn;
- (void)sockindex;
- #endif
- return retcode;
- }
|