mirror of
https://github.com/MariaDB/server.git
synced 2025-01-20 05:52:27 +01:00
682 lines
19 KiB
C++
682 lines
19 KiB
C++
/*
|
|
Copyright (C) 2000-2007 MySQL AB
|
|
|
|
This program 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; version 2 of the License.
|
|
|
|
This program 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 program; see the file COPYING. If not, write to the
|
|
Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston,
|
|
MA 02110-1301 USA.
|
|
*/
|
|
|
|
|
|
/* yaSSL internal header defines SSL supporting types not specified in the
|
|
* draft along with type conversion functions and openssl compatibility
|
|
*/
|
|
|
|
|
|
#ifndef yaSSL_INT_HPP
|
|
#define yaSSL_INT_HPP
|
|
|
|
#include "yassl_imp.hpp"
|
|
#include "yassl_error.hpp"
|
|
#include "crypto_wrapper.hpp"
|
|
#include "cert_wrapper.hpp"
|
|
#include "log.hpp"
|
|
#include "lock.hpp"
|
|
#include "openssl/ssl.h" // ASN1_STRING and DH
|
|
|
|
// Check if _POSIX_THREADS should be forced
|
|
#if !defined(_POSIX_THREADS) && (defined(__NETWARE__) || defined(__hpux))
|
|
// HPUX does not define _POSIX_THREADS as it's not _fully_ implemented
|
|
// Netware supports pthreads but does not announce it
|
|
#define _POSIX_THREADS
|
|
#endif
|
|
|
|
#ifdef _POSIX_THREADS
|
|
#include <pthread.h>
|
|
#endif
|
|
|
|
|
|
namespace STL = STL_NAMESPACE;
|
|
|
|
|
|
namespace yaSSL {
|
|
|
|
|
|
// State Machine for Record Layer Protocol
|
|
enum RecordLayerState {
|
|
recordNotReady = 0, // fatal error, no more processing
|
|
recordReady
|
|
};
|
|
|
|
|
|
// State Machine for HandShake Protocol
|
|
enum HandShakeState {
|
|
handShakeNotReady = 0, // fatal error, no more processing
|
|
preHandshake, // initial state
|
|
inHandshake, // handshake started
|
|
handShakeReady // handshake done
|
|
};
|
|
|
|
|
|
// client input HandShake state, use if HandShakeState == inHandShake
|
|
enum ClientState {
|
|
serverNull = 0,
|
|
serverHelloComplete,
|
|
serverCertComplete,
|
|
serverKeyExchangeComplete,
|
|
serverHelloDoneComplete,
|
|
serverFinishedComplete
|
|
};
|
|
|
|
|
|
// server input HandShake state, use if HandShakeState == inHandShake
|
|
enum ServerState {
|
|
clientNull = 0,
|
|
clientHelloComplete,
|
|
clientKeyExchangeComplete,
|
|
clientFinishedComplete
|
|
};
|
|
|
|
|
|
// client connect state for nonblocking restart
|
|
enum ConnectState {
|
|
CONNECT_BEGIN = 0,
|
|
CLIENT_HELLO_SENT,
|
|
FIRST_REPLY_DONE,
|
|
FINISHED_DONE,
|
|
SECOND_REPLY_DONE
|
|
};
|
|
|
|
|
|
// server accpet state for nonblocking restart
|
|
enum AcceptState {
|
|
ACCEPT_BEGIN = 0,
|
|
ACCEPT_FIRST_REPLY_DONE,
|
|
SERVER_HELLO_DONE,
|
|
ACCEPT_SECOND_REPLY_DONE,
|
|
ACCEPT_FINISHED_DONE,
|
|
ACCEPT_THIRD_REPLY_DONE
|
|
};
|
|
|
|
|
|
// combines all states
|
|
class States {
|
|
RecordLayerState recordLayer_;
|
|
HandShakeState handshakeLayer_;
|
|
ClientState clientState_;
|
|
ServerState serverState_;
|
|
ConnectState connectState_;
|
|
AcceptState acceptState_;
|
|
char errorString_[MAX_ERROR_SZ];
|
|
YasslError what_;
|
|
public:
|
|
States();
|
|
|
|
const RecordLayerState& getRecord() const;
|
|
const HandShakeState& getHandShake() const;
|
|
const ClientState& getClient() const;
|
|
const ServerState& getServer() const;
|
|
const ConnectState& GetConnect() const;
|
|
const AcceptState& GetAccept() const;
|
|
const char* getString() const;
|
|
YasslError What() const;
|
|
|
|
RecordLayerState& useRecord();
|
|
HandShakeState& useHandShake();
|
|
ClientState& useClient();
|
|
ServerState& useServer();
|
|
ConnectState& UseConnect();
|
|
AcceptState& UseAccept();
|
|
char* useString();
|
|
void SetError(YasslError);
|
|
private:
|
|
States(const States&); // hide copy
|
|
States& operator=(const States&); // and assign
|
|
};
|
|
|
|
|
|
// holds all factories
|
|
class sslFactory {
|
|
MessageFactory messageFactory_; // creates new messages by type
|
|
HandShakeFactory handShakeFactory_; // creates new handshake types
|
|
ServerKeyFactory serverKeyFactory_; // creates new server key types
|
|
ClientKeyFactory clientKeyFactory_; // creates new client key types
|
|
|
|
sslFactory(); // only GetSSL_Factory creates
|
|
public:
|
|
const MessageFactory& getMessage() const;
|
|
const HandShakeFactory& getHandShake() const;
|
|
const ServerKeyFactory& getServerKey() const;
|
|
const ClientKeyFactory& getClientKey() const;
|
|
|
|
friend sslFactory& GetSSL_Factory(); // singleton creator
|
|
private:
|
|
sslFactory(const sslFactory&); // hide copy
|
|
sslFactory& operator=(const sslFactory&); // and assign
|
|
};
|
|
|
|
|
|
#undef X509_NAME // wincrypt.h clash
|
|
|
|
// openSSL X509 names
|
|
class X509_NAME {
|
|
char* name_;
|
|
size_t sz_;
|
|
ASN1_STRING entry_;
|
|
public:
|
|
X509_NAME(const char*, size_t sz);
|
|
~X509_NAME();
|
|
|
|
const char* GetName() const;
|
|
ASN1_STRING* GetEntry(int i);
|
|
size_t GetLength() const;
|
|
private:
|
|
X509_NAME(const X509_NAME&); // hide copy
|
|
X509_NAME& operator=(const X509_NAME&); // and assign
|
|
};
|
|
|
|
|
|
class StringHolder {
|
|
ASN1_STRING asnString_;
|
|
public:
|
|
StringHolder(const char* str, int sz);
|
|
~StringHolder();
|
|
|
|
ASN1_STRING* GetString();
|
|
private:
|
|
StringHolder(const StringHolder&); // hide copy
|
|
StringHolder& operator=(const StringHolder&); // and assign
|
|
};
|
|
|
|
|
|
// openSSL X509
|
|
class X509 {
|
|
X509_NAME issuer_;
|
|
X509_NAME subject_;
|
|
StringHolder beforeDate_; // not valid before
|
|
StringHolder afterDate_; // not valid after
|
|
public:
|
|
X509(const char* i, size_t, const char* s, size_t,
|
|
const char* b, int, const char* a, int);
|
|
~X509() {}
|
|
|
|
X509_NAME* GetIssuer();
|
|
X509_NAME* GetSubject();
|
|
|
|
ASN1_STRING* GetBefore();
|
|
ASN1_STRING* GetAfter();
|
|
|
|
private:
|
|
X509(const X509&); // hide copy
|
|
X509& operator=(const X509&); // and assign
|
|
};
|
|
|
|
|
|
// openSSL bignum
|
|
struct BIGNUM {
|
|
/*
|
|
gcc 2.96 fix: because of two Integer classes (yaSSL::Integer and
|
|
TaoCrypt::Integer), we need to explicitly state the namespace
|
|
here to let gcc 2.96 deduce the correct type.
|
|
*/
|
|
yaSSL::Integer int_;
|
|
void assign(const byte* b, uint s) { int_.assign(b,s); }
|
|
};
|
|
|
|
|
|
// openSSL session
|
|
class SSL_SESSION {
|
|
opaque sessionID_[ID_LEN];
|
|
opaque master_secret_[SECRET_LEN];
|
|
Cipher suite_[SUITE_LEN];
|
|
uint bornOn_; // create time in seconds
|
|
uint timeout_; // timeout in seconds
|
|
RandomPool& random_; // will clean master secret
|
|
X509* peerX509_;
|
|
public:
|
|
explicit SSL_SESSION(RandomPool&);
|
|
SSL_SESSION(const SSL&, RandomPool&);
|
|
~SSL_SESSION();
|
|
|
|
const opaque* GetID() const;
|
|
const opaque* GetSecret() const;
|
|
const Cipher* GetSuite() const;
|
|
uint GetBornOn() const;
|
|
uint GetTimeOut() const;
|
|
X509* GetPeerX509() const;
|
|
void SetTimeOut(uint);
|
|
|
|
SSL_SESSION& operator=(const SSL_SESSION&); // allow assign for resumption
|
|
private:
|
|
SSL_SESSION(const SSL_SESSION&); // hide copy
|
|
|
|
void CopyX509(X509*);
|
|
};
|
|
|
|
|
|
// holds all sessions
|
|
class Sessions {
|
|
STL::list<SSL_SESSION*> list_;
|
|
RandomPool random_; // for session cleaning
|
|
Mutex mutex_; // no-op for single threaded
|
|
|
|
Sessions() {} // only GetSessions can create
|
|
public:
|
|
SSL_SESSION* lookup(const opaque*, SSL_SESSION* copy = 0);
|
|
void add(const SSL&);
|
|
void remove(const opaque*);
|
|
|
|
~Sessions();
|
|
|
|
friend Sessions& GetSessions(); // singleton creator
|
|
private:
|
|
Sessions(const Sessions&); // hide copy
|
|
Sessions& operator=(const Sessions&); // and assign
|
|
};
|
|
|
|
|
|
#ifdef _POSIX_THREADS
|
|
typedef pthread_t THREAD_ID_T;
|
|
#else
|
|
typedef DWORD THREAD_ID_T;
|
|
#endif
|
|
|
|
// thread error data
|
|
struct ThreadError {
|
|
THREAD_ID_T threadID_;
|
|
int errorID_;
|
|
};
|
|
|
|
|
|
// holds all errors
|
|
class Errors {
|
|
STL::list<ThreadError> list_;
|
|
Mutex mutex_;
|
|
|
|
Errors() {} // only GetErrors can create
|
|
public:
|
|
int Lookup(bool peek); // self lookup
|
|
void Add(int);
|
|
void Remove(); // remove self
|
|
|
|
~Errors() {}
|
|
|
|
friend Errors& GetErrors(); // singleton creator
|
|
private:
|
|
Errors(const Errors&); // hide copy
|
|
Errors& operator=(const Errors); // and assign
|
|
};
|
|
|
|
|
|
Sessions& GetSessions(); // forward singletons
|
|
sslFactory& GetSSL_Factory();
|
|
Errors& GetErrors();
|
|
|
|
|
|
// openSSL method and context types
|
|
class SSL_METHOD {
|
|
ProtocolVersion version_;
|
|
ConnectionEnd side_;
|
|
bool verifyPeer_; // request or send certificate
|
|
bool verifyNone_; // whether to verify certificate
|
|
bool failNoCert_;
|
|
bool multipleProtocol_; // for SSLv23 compatibility
|
|
public:
|
|
SSL_METHOD(ConnectionEnd ce, ProtocolVersion pv,
|
|
bool multipleProtocol = false);
|
|
|
|
ProtocolVersion getVersion() const;
|
|
ConnectionEnd getSide() const;
|
|
|
|
void setVerifyPeer();
|
|
void setVerifyNone();
|
|
void setFailNoCert();
|
|
|
|
bool verifyPeer() const;
|
|
bool verifyNone() const;
|
|
bool failNoCert() const;
|
|
bool multipleProtocol() const;
|
|
private:
|
|
SSL_METHOD(const SSL_METHOD&); // hide copy
|
|
SSL_METHOD& operator=(const SSL_METHOD&); // and assign
|
|
};
|
|
|
|
|
|
struct Ciphers {
|
|
bool setSuites_; // user set suites from default
|
|
byte suites_[MAX_SUITE_SZ]; // new suites
|
|
int suiteSz_; // suite length in bytes
|
|
|
|
Ciphers() : setSuites_(false), suiteSz_(0) {}
|
|
};
|
|
|
|
|
|
struct DH; // forward
|
|
|
|
|
|
// save for SSL construction
|
|
struct DH_Parms {
|
|
Integer p_;
|
|
Integer g_;
|
|
bool set_; // if set by user
|
|
|
|
DH_Parms() : set_(false) {}
|
|
};
|
|
|
|
|
|
enum StatsField {
|
|
Accept, Connect, AcceptGood, ConnectGood, AcceptRenegotiate,
|
|
ConnectRenegotiate, Hits, CbHits, CacheFull, Misses, Timeouts, Number,
|
|
GetCacheSize, VerifyMode, VerifyDepth
|
|
};
|
|
|
|
|
|
// SSL stats
|
|
struct Stats {
|
|
long accept_;
|
|
long connect_;
|
|
long acceptGood_;
|
|
long connectGood_;
|
|
long acceptRenegotiate_;
|
|
long connectRenegotiate_;
|
|
|
|
long hits_;
|
|
long cbHits_;
|
|
long cacheFull_;
|
|
long misses_;
|
|
long timeouts_;
|
|
long number_;
|
|
long getCacheSize_;
|
|
|
|
int verifyMode_;
|
|
int verifyDepth_;
|
|
public:
|
|
Stats() : accept_(0), connect_(0), acceptGood_(0), connectGood_(0),
|
|
acceptRenegotiate_(0), connectRenegotiate_(0), hits_(0), cbHits_(0),
|
|
cacheFull_(0), misses_(0), timeouts_(0), number_(0), getCacheSize_(0),
|
|
verifyMode_(0), verifyDepth_(0)
|
|
{}
|
|
private:
|
|
Stats(const Stats&); // hide copy
|
|
Stats& operator=(const Stats&); // and assign
|
|
};
|
|
|
|
|
|
// the SSL context
|
|
class SSL_CTX {
|
|
public:
|
|
typedef STL::list<x509*> CertList;
|
|
private:
|
|
SSL_METHOD* method_;
|
|
x509* certificate_;
|
|
x509* privateKey_;
|
|
CertList caList_;
|
|
Ciphers ciphers_;
|
|
DH_Parms dhParms_;
|
|
pem_password_cb passwordCb_;
|
|
void* userData_;
|
|
bool sessionCacheOff_;
|
|
Stats stats_;
|
|
Mutex mutex_; // for Stats
|
|
public:
|
|
explicit SSL_CTX(SSL_METHOD* meth);
|
|
~SSL_CTX();
|
|
|
|
const x509* getCert() const;
|
|
const x509* getKey() const;
|
|
const SSL_METHOD* getMethod() const;
|
|
const Ciphers& GetCiphers() const;
|
|
const DH_Parms& GetDH_Parms() const;
|
|
const Stats& GetStats() const;
|
|
pem_password_cb GetPasswordCb() const;
|
|
void* GetUserData() const;
|
|
bool GetSessionCacheOff() const;
|
|
|
|
void setVerifyPeer();
|
|
void setVerifyNone();
|
|
void setFailNoCert();
|
|
bool SetCipherList(const char*);
|
|
bool SetDH(const DH&);
|
|
void SetPasswordCb(pem_password_cb cb);
|
|
void SetUserData(void*);
|
|
void SetSessionCacheOff();
|
|
|
|
void IncrementStats(StatsField);
|
|
void AddCA(x509* ca);
|
|
const CertList& GetCA_List() const;
|
|
|
|
friend int read_file(SSL_CTX*, const char*, int, CertType);
|
|
private:
|
|
SSL_CTX(const SSL_CTX&); // hide copy
|
|
SSL_CTX& operator=(const SSL_CTX&); // and assign
|
|
};
|
|
|
|
|
|
// holds all cryptographic types
|
|
class Crypto {
|
|
Digest* digest_; // agreed upon digest
|
|
BulkCipher* cipher_; // agreed upon cipher
|
|
DiffieHellman* dh_; // dh parms
|
|
RandomPool random_; // random number generator
|
|
CertManager cert_; // manages certificates
|
|
public:
|
|
explicit Crypto();
|
|
~Crypto();
|
|
|
|
const Digest& get_digest() const;
|
|
const BulkCipher& get_cipher() const;
|
|
const DiffieHellman& get_dh() const;
|
|
const RandomPool& get_random() const;
|
|
const CertManager& get_certManager() const;
|
|
|
|
Digest& use_digest();
|
|
BulkCipher& use_cipher();
|
|
DiffieHellman& use_dh();
|
|
RandomPool& use_random();
|
|
CertManager& use_certManager();
|
|
|
|
void SetDH(DiffieHellman*);
|
|
void SetDH(const DH_Parms&);
|
|
void setDigest(Digest*);
|
|
void setCipher(BulkCipher*);
|
|
|
|
bool DhSet();
|
|
private:
|
|
Crypto(const Crypto&); // hide copy
|
|
Crypto& operator=(const Crypto&); // and assign
|
|
};
|
|
|
|
|
|
// holds all handshake and verify hashes
|
|
class sslHashes {
|
|
MD5 md5HandShake_; // md5 handshake hash
|
|
SHA shaHandShake_; // sha handshake hash
|
|
Finished verify_; // peer's verify hash
|
|
Hashes certVerify_; // peer's cert verify hash
|
|
public:
|
|
sslHashes() {}
|
|
|
|
const MD5& get_MD5() const;
|
|
const SHA& get_SHA() const;
|
|
const Finished& get_verify() const;
|
|
const Hashes& get_certVerify() const;
|
|
|
|
MD5& use_MD5();
|
|
SHA& use_SHA();
|
|
Finished& use_verify();
|
|
Hashes& use_certVerify();
|
|
private:
|
|
sslHashes(const sslHashes&); // hide copy
|
|
sslHashes& operator=(const sslHashes&); // and assign
|
|
};
|
|
|
|
|
|
// holds input and output buffers
|
|
class Buffers {
|
|
public:
|
|
typedef STL::list<input_buffer*> inputList;
|
|
typedef STL::list<output_buffer*> outputList;
|
|
private:
|
|
inputList dataList_; // list of users app data / handshake
|
|
outputList handShakeList_; // buffered handshake msgs
|
|
input_buffer* rawInput_; // buffered raw input yet to process
|
|
public:
|
|
Buffers();
|
|
~Buffers();
|
|
|
|
const inputList& getData() const;
|
|
const outputList& getHandShake() const;
|
|
|
|
inputList& useData();
|
|
outputList& useHandShake();
|
|
|
|
void SetRawInput(input_buffer*); // takes ownership
|
|
input_buffer* TakeRawInput(); // takes ownership
|
|
private:
|
|
Buffers(const Buffers&); // hide copy
|
|
Buffers& operator=(const Buffers&); // and assign
|
|
};
|
|
|
|
|
|
// wraps security parameters
|
|
class Security {
|
|
Connection conn_; // connection information
|
|
Parameters parms_; // may be pending
|
|
SSL_SESSION resumeSession_; // if resuming
|
|
SSL_CTX* ctx_; // context used to init
|
|
bool resuming_; // trying to resume
|
|
public:
|
|
Security(ProtocolVersion, RandomPool&, ConnectionEnd, const Ciphers&,
|
|
SSL_CTX*, bool);
|
|
|
|
const SSL_CTX* GetContext() const;
|
|
const Connection& get_connection() const;
|
|
const Parameters& get_parms() const;
|
|
const SSL_SESSION& get_resume() const;
|
|
bool get_resuming() const;
|
|
|
|
Connection& use_connection();
|
|
Parameters& use_parms();
|
|
SSL_SESSION& use_resume();
|
|
|
|
void set_resuming(bool b);
|
|
private:
|
|
Security(const Security&); // hide copy
|
|
Security& operator=(const Security&); // and assign
|
|
};
|
|
|
|
|
|
// THE SSL type
|
|
class SSL {
|
|
Crypto crypto_; // agreed crypto agents
|
|
Security secure_; // Connection and Session parms
|
|
States states_; // Record and HandShake states
|
|
sslHashes hashes_; // handshake, finished hashes
|
|
Socket socket_; // socket wrapper
|
|
Buffers buffers_; // buffered handshakes and data
|
|
Log log_; // logger
|
|
|
|
// optimization variables
|
|
bool has_data_; // buffered data ready?
|
|
public:
|
|
SSL(SSL_CTX* ctx);
|
|
|
|
// gets and uses
|
|
const Crypto& getCrypto() const;
|
|
const Security& getSecurity() const;
|
|
const States& getStates() const;
|
|
const sslHashes& getHashes() const;
|
|
const sslFactory& getFactory() const;
|
|
const Socket& getSocket() const;
|
|
YasslError GetError() const;
|
|
bool GetMultiProtocol() const;
|
|
bool CompressionOn() const;
|
|
|
|
Crypto& useCrypto();
|
|
Security& useSecurity();
|
|
States& useStates();
|
|
sslHashes& useHashes();
|
|
Socket& useSocket();
|
|
Log& useLog();
|
|
Buffers& useBuffers();
|
|
|
|
bool HasData() const;
|
|
|
|
// sets
|
|
void set_pending(Cipher suite);
|
|
void set_random(const opaque*, ConnectionEnd);
|
|
void set_sessionID(const opaque*);
|
|
void set_session(SSL_SESSION*);
|
|
void set_preMaster(const opaque*, uint);
|
|
void set_masterSecret(const opaque*);
|
|
void SetError(YasslError);
|
|
int SetCompression();
|
|
void UnSetCompression();
|
|
|
|
// helpers
|
|
bool isTLS() const;
|
|
bool isTLSv1_1() const;
|
|
void order_error();
|
|
void makeMasterSecret();
|
|
void makeTLSMasterSecret();
|
|
void addData(input_buffer* data);
|
|
void fillData(Data&);
|
|
void PeekData(Data&);
|
|
void addBuffer(output_buffer* b);
|
|
void flushBuffer();
|
|
void verifyState(const RecordLayerHeader&);
|
|
void verifyState(const HandShakeHeader&);
|
|
void verifyState(ClientState);
|
|
void verifyState(ServerState);
|
|
void verfiyHandShakeComplete();
|
|
void matchSuite(const opaque*, uint length);
|
|
void deriveKeys();
|
|
void deriveTLSKeys();
|
|
void Send(const byte*, uint);
|
|
|
|
uint bufferedData();
|
|
uint get_SEQIncrement(bool);
|
|
|
|
const byte* get_macSecret(bool);
|
|
private:
|
|
void storeKeys(const opaque*);
|
|
void setKeys();
|
|
void verifyClientState(HandShakeType);
|
|
void verifyServerState(HandShakeType);
|
|
|
|
SSL(const SSL&); // hide copy
|
|
const SSL& operator=(const SSL&); // and assign
|
|
};
|
|
|
|
|
|
// compression
|
|
int Compress(const byte*, int, input_buffer&);
|
|
int DeCompress(input_buffer&, int, input_buffer&);
|
|
|
|
|
|
// conversion functions
|
|
void c32to24(uint32, uint24&);
|
|
void c24to32(const uint24, uint32&);
|
|
|
|
uint32 c24to32(const uint24);
|
|
|
|
void ato16(const opaque*, uint16&);
|
|
void ato24(const opaque*, uint24&);
|
|
|
|
void c16toa(uint16, opaque*);
|
|
void c24toa(const uint24, opaque*);
|
|
void c32toa(uint32 u32, opaque*);
|
|
|
|
|
|
} // naemspace
|
|
|
|
#endif // yaSSL_INT_HPP
|