2012-05-31 18:53:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD access method
|
|
|
|
* MODULE: CryptoManager.h
|
|
|
|
* DESCRIPTION: Database encryption
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Initial
|
|
|
|
* Developer's Public License Version 1.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the
|
|
|
|
* License. You may obtain a copy of the License at
|
|
|
|
* http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed AS IS,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing rights
|
|
|
|
* and limitations under the License.
|
|
|
|
*
|
|
|
|
* The Original Code was created by Alex Peshkov
|
|
|
|
* for the Firebird Open Source RDBMS project.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2012 Alex Peshkov <peshkoff at mail.ru>
|
|
|
|
* and all contributors signed below.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef JRD_CRYPTO_MANAGER
|
|
|
|
#define JRD_CRYPTO_MANAGER
|
|
|
|
|
|
|
|
#include "../common/classes/alloc.h"
|
|
|
|
#include "../common/classes/fb_atomic.h"
|
|
|
|
#include "../common/classes/SyncObject.h"
|
|
|
|
#include "../common/classes/fb_string.h"
|
|
|
|
#include "../common/classes/objects_array.h"
|
2015-12-10 17:58:58 +01:00
|
|
|
#include "../common/classes/condition.h"
|
2016-02-03 14:20:41 +01:00
|
|
|
#include "../common/classes/MetaName.h"
|
2017-01-31 15:38:53 +01:00
|
|
|
#include "../common/classes/GetPlugins.h"
|
2012-06-01 09:09:42 +02:00
|
|
|
#include "../common/ThreadStart.h"
|
2012-06-05 12:06:31 +02:00
|
|
|
#include "../jrd/ods.h"
|
2015-03-20 19:02:30 +01:00
|
|
|
#include "../jrd/status.h"
|
2014-09-29 13:03:47 +02:00
|
|
|
#include "firebird/Interface.h"
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2016-09-15 16:37:50 +02:00
|
|
|
#define CRYPT_DEBUG(A)
|
|
|
|
|
2012-05-31 18:53:42 +02:00
|
|
|
// forward
|
|
|
|
|
|
|
|
class Config;
|
|
|
|
|
|
|
|
namespace Ods {
|
2016-06-03 14:52:46 +02:00
|
|
|
|
|
|
|
struct pag;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Firebird {
|
|
|
|
|
|
|
|
class ClumpletReader;
|
|
|
|
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace Jrd {
|
|
|
|
|
|
|
|
class Database;
|
|
|
|
class Attachment;
|
|
|
|
class jrd_file;
|
|
|
|
class BufferDesc;
|
|
|
|
class thread_db;
|
|
|
|
class Lock;
|
2015-12-10 17:58:58 +01:00
|
|
|
class PageSpace;
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2015-12-15 17:54:23 +01:00
|
|
|
|
|
|
|
//
|
|
|
|
// This very specific locking class makes it possible to perform traditional read/write locks,
|
|
|
|
// but in addition it can on special request perform some predefined action or (in a case when
|
|
|
|
// >=1 read lock is taken) set barrier for new locks (new locks will not be granted) and
|
|
|
|
// at the moment when last existing lock is released execute that predefined action in context
|
|
|
|
// of a thread, releasing last lock.
|
|
|
|
//
|
|
|
|
// In our case special request is done from AST handler - and therefore called ast.
|
|
|
|
// Read locks are done when performing IO+crypt activity - and called ioBegin/ioEnd.
|
|
|
|
// Write locks are done when some exclusive activity like changing crypt state is
|
|
|
|
// needed - they are full locks and called lockBegin/lockEnd.
|
|
|
|
//
|
|
|
|
|
2015-12-10 17:58:58 +01:00
|
|
|
class BarSync
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
class IBar
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void doOnTakenWriteSync(Jrd::thread_db* tdbb) = 0;
|
|
|
|
virtual void doOnAst(Jrd::thread_db* tdbb) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
BarSync(IBar* i)
|
|
|
|
: callback(i), counter(0), lockMode(0), flagWriteLock(false)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
class IoGuard
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
IoGuard(Jrd::thread_db* p_tdbb, BarSync& p_bs)
|
|
|
|
: tdbb(p_tdbb), bs(p_bs)
|
|
|
|
{
|
|
|
|
bs.ioBegin(tdbb);
|
|
|
|
}
|
|
|
|
|
|
|
|
~IoGuard()
|
|
|
|
{
|
|
|
|
bs.ioEnd(tdbb);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Jrd::thread_db* tdbb;
|
|
|
|
BarSync& bs;
|
|
|
|
};
|
|
|
|
|
|
|
|
class LockGuard
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
LockGuard(Jrd::thread_db* p_tdbb, BarSync& p_bs)
|
|
|
|
: tdbb(p_tdbb), bs(p_bs), flagLocked(false)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
void lock()
|
|
|
|
{
|
|
|
|
fb_assert(!flagLocked);
|
|
|
|
if (!flagLocked)
|
|
|
|
{
|
|
|
|
bs.lockBegin(tdbb);
|
|
|
|
flagLocked = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~LockGuard()
|
|
|
|
{
|
|
|
|
if (flagLocked)
|
|
|
|
{
|
|
|
|
bs.lockEnd(tdbb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Jrd::thread_db* tdbb;
|
|
|
|
BarSync& bs;
|
|
|
|
bool flagLocked;
|
|
|
|
};
|
|
|
|
|
|
|
|
void ioBegin(Jrd::thread_db* tdbb)
|
|
|
|
{
|
|
|
|
Firebird::MutexLockGuard g(mutex, FB_FUNCTION);
|
|
|
|
|
|
|
|
if (counter < 0)
|
|
|
|
{
|
2016-03-09 15:58:08 +01:00
|
|
|
if (!(flagWriteLock && (thread == getThreadId())))
|
2015-12-10 17:58:58 +01:00
|
|
|
{
|
2016-03-09 15:58:08 +01:00
|
|
|
if ((counter % BIG_VALUE == 0) && (!flagWriteLock))
|
2015-12-10 17:58:58 +01:00
|
|
|
{
|
2016-03-09 15:58:08 +01:00
|
|
|
if (lockMode)
|
|
|
|
{
|
|
|
|
// Someone is waiting for write lock
|
|
|
|
lockCond.notifyOne();
|
|
|
|
barCond.wait(mutex);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Ast done
|
|
|
|
callWriteLockHandler(tdbb);
|
|
|
|
counter = 0;
|
|
|
|
}
|
2015-12-10 17:58:58 +01:00
|
|
|
}
|
|
|
|
else
|
2016-03-09 15:58:08 +01:00
|
|
|
barCond.wait(mutex);
|
2015-12-10 17:58:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
++counter;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ioEnd(Jrd::thread_db* tdbb)
|
|
|
|
{
|
|
|
|
Firebird::MutexLockGuard g(mutex, FB_FUNCTION);
|
|
|
|
|
|
|
|
if (--counter < 0 && counter % BIG_VALUE == 0)
|
|
|
|
{
|
|
|
|
if (!(flagWriteLock && (thread == getThreadId())))
|
|
|
|
{
|
|
|
|
if (lockMode)
|
|
|
|
lockCond.notifyOne();
|
|
|
|
else
|
|
|
|
{
|
|
|
|
callWriteLockHandler(tdbb);
|
|
|
|
finishWriteLock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ast(Jrd::thread_db* tdbb)
|
|
|
|
{
|
|
|
|
Firebird::MutexLockGuard g(mutex, FB_FUNCTION);
|
|
|
|
if (counter >= 0)
|
|
|
|
{
|
|
|
|
counter -= BIG_VALUE;
|
|
|
|
}
|
|
|
|
callback->doOnAst(tdbb);
|
|
|
|
}
|
|
|
|
|
|
|
|
void lockBegin(Jrd::thread_db* tdbb)
|
|
|
|
{
|
|
|
|
Firebird::MutexLockGuard g(mutex, FB_FUNCTION);
|
|
|
|
|
|
|
|
if ((counter -= BIG_VALUE) != -BIG_VALUE)
|
|
|
|
{
|
|
|
|
++lockMode;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
lockCond.wait(mutex);
|
|
|
|
}
|
2015-12-13 02:02:02 +01:00
|
|
|
catch (const Firebird::Exception&)
|
2015-12-10 17:58:58 +01:00
|
|
|
{
|
|
|
|
--lockMode;
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
--lockMode;
|
|
|
|
}
|
|
|
|
|
|
|
|
thread = getThreadId();
|
|
|
|
flagWriteLock = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void lockEnd(Jrd::thread_db* tdbb)
|
|
|
|
{
|
|
|
|
Firebird::MutexLockGuard g(mutex, FB_FUNCTION);
|
|
|
|
|
|
|
|
flagWriteLock = false;
|
|
|
|
finishWriteLock();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void callWriteLockHandler(Jrd::thread_db* tdbb)
|
|
|
|
{
|
|
|
|
thread = getThreadId();
|
|
|
|
flagWriteLock = true;
|
|
|
|
callback->doOnTakenWriteSync(tdbb);
|
|
|
|
flagWriteLock = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void finishWriteLock()
|
|
|
|
{
|
|
|
|
if ((counter += BIG_VALUE) == 0)
|
|
|
|
barCond.notifyAll();
|
|
|
|
else
|
|
|
|
lockCond.notifyOne();
|
|
|
|
}
|
|
|
|
|
|
|
|
Firebird::Condition barCond, lockCond;
|
|
|
|
Firebird::Mutex mutex;
|
|
|
|
IBar* callback;
|
|
|
|
ThreadId thread;
|
|
|
|
int counter;
|
|
|
|
int lockMode;
|
|
|
|
bool flagWriteLock;
|
|
|
|
|
|
|
|
static const int BIG_VALUE = 1000000;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CryptoManager FB_FINAL : public Firebird::PermanentStorage, public BarSync::IBar
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
public:
|
2017-01-31 15:38:53 +01:00
|
|
|
typedef Firebird::GetPlugins<Firebird::IDbCryptPlugin> Factory;
|
2018-02-01 16:10:09 +01:00
|
|
|
typedef Firebird::HalfStaticArray<Attachment*, 16> AttVector;
|
2017-01-31 15:38:53 +01:00
|
|
|
|
2012-08-08 04:20:30 +02:00
|
|
|
explicit CryptoManager(thread_db* tdbb);
|
2012-05-31 18:53:42 +02:00
|
|
|
~CryptoManager();
|
|
|
|
|
2012-06-03 05:00:24 +02:00
|
|
|
void shutdown(thread_db* tdbb);
|
|
|
|
|
2016-02-03 14:20:41 +01:00
|
|
|
void prepareChangeCryptState(thread_db* tdbb, const Firebird::MetaName& plugName,
|
|
|
|
const Firebird::MetaName& key);
|
2012-05-31 18:53:42 +02:00
|
|
|
void changeCryptState(thread_db* tdbb, const Firebird::string& plugName);
|
|
|
|
void attach(thread_db* tdbb, Attachment* att);
|
|
|
|
void detach(thread_db* tdbb, Attachment* att);
|
|
|
|
|
|
|
|
void startCryptThread(thread_db* tdbb);
|
2016-09-15 16:37:50 +02:00
|
|
|
void terminateCryptThread(thread_db* tdbb, bool wait = false);
|
2016-02-26 13:55:34 +01:00
|
|
|
void stopThreadUsing(thread_db* tdbb, Attachment* att);
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2016-02-01 17:16:42 +01:00
|
|
|
class IOCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual bool callback(thread_db* tdbb, FbStatusVector* sv, Ods::pag* page) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool read(thread_db* tdbb, FbStatusVector* sv, Ods::pag* page, IOCallback* io);
|
|
|
|
bool write(thread_db* tdbb, FbStatusVector* sv, Ods::pag* page, IOCallback* io);
|
2012-05-31 18:53:42 +02:00
|
|
|
|
|
|
|
void cryptThread();
|
|
|
|
|
2017-01-09 18:28:11 +01:00
|
|
|
bool checkValidation(Firebird::IDbCryptPlugin* crypt);
|
2017-07-19 16:21:19 +02:00
|
|
|
void setDbInfo(Firebird::IDbCryptPlugin* cp);
|
2017-01-09 18:28:11 +01:00
|
|
|
|
2016-01-21 01:25:55 +01:00
|
|
|
ULONG getCurrentPage() const;
|
|
|
|
UCHAR getCurrentState() const;
|
2018-01-16 18:37:05 +01:00
|
|
|
const char* getKeyName() const;
|
2018-05-23 16:16:04 +02:00
|
|
|
const char* getPluginName() const;
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2016-02-01 17:16:42 +01:00
|
|
|
private:
|
2016-01-29 15:12:15 +01:00
|
|
|
enum IoResult {SUCCESS_ALL, FAILED_CRYPT, FAILED_IO};
|
2016-02-01 17:16:42 +01:00
|
|
|
IoResult internalRead(thread_db* tdbb, FbStatusVector* sv, Ods::pag* page, IOCallback* io);
|
|
|
|
IoResult internalWrite(thread_db* tdbb, FbStatusVector* sv, Ods::pag* page, IOCallback* io);
|
2016-01-29 15:12:15 +01:00
|
|
|
|
2012-06-05 12:06:31 +02:00
|
|
|
class Buffer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
operator Ods::pag*()
|
|
|
|
{
|
2015-09-01 09:24:10 +02:00
|
|
|
return reinterpret_cast<Ods::pag*>(FB_ALIGN(buf, PAGE_ALIGNMENT));
|
2012-06-05 12:06:31 +02:00
|
|
|
}
|
|
|
|
|
2015-12-10 17:58:58 +01:00
|
|
|
Ods::pag* operator->()
|
|
|
|
{
|
|
|
|
return reinterpret_cast<Ods::pag*>(FB_ALIGN(buf, PAGE_ALIGNMENT));
|
|
|
|
}
|
|
|
|
|
2012-06-05 12:06:31 +02:00
|
|
|
private:
|
2015-09-01 09:24:10 +02:00
|
|
|
char buf[MAX_PAGE_SIZE + PAGE_ALIGNMENT - 1];
|
2012-06-05 12:06:31 +02:00
|
|
|
};
|
|
|
|
|
2016-10-12 14:22:15 +02:00
|
|
|
class DbInfo;
|
|
|
|
friend class DbInfo;
|
|
|
|
|
2017-01-31 15:38:53 +01:00
|
|
|
class DbInfo FB_FINAL : public Firebird::RefCntIface<Firebird::IDbCryptInfoImpl<DbInfo, Firebird::CheckStatusWrapper> >
|
2016-10-12 14:22:15 +02:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
DbInfo(CryptoManager* cm)
|
|
|
|
: cryptoManager(cm)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
void destroy()
|
|
|
|
{
|
|
|
|
cryptoManager = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// IDbCryptInfo implementation
|
|
|
|
const char* getDatabaseFullPath(Firebird::CheckStatusWrapper* status);
|
|
|
|
|
|
|
|
int release()
|
|
|
|
{
|
|
|
|
if (--refCounter != 0)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
delete this;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
CryptoManager* cryptoManager;
|
|
|
|
};
|
|
|
|
|
2012-05-31 18:53:42 +02:00
|
|
|
static int blockingAstChangeCryptState(void*);
|
|
|
|
void blockingAstChangeCryptState();
|
|
|
|
|
2015-12-10 17:58:58 +01:00
|
|
|
// IBar's pure virtual functions are implemented here
|
|
|
|
void doOnTakenWriteSync(thread_db* tdbb);
|
|
|
|
void doOnAst(thread_db* tdbb);
|
|
|
|
|
2017-01-31 15:38:53 +01:00
|
|
|
void loadPlugin(thread_db* tdbb, const char* pluginName);
|
2018-02-01 16:10:09 +01:00
|
|
|
bool validateAttachment(thread_db* tdbb, Attachment* att, bool consume);
|
2012-05-31 18:53:42 +02:00
|
|
|
ULONG getLastPage(thread_db* tdbb);
|
2016-02-01 17:16:42 +01:00
|
|
|
void writeDbHeader(thread_db* tdbb, ULONG runpage);
|
2017-01-09 18:28:11 +01:00
|
|
|
void calcValidation(Firebird::string& valid, Firebird::IDbCryptPlugin* plugin);
|
|
|
|
void checkValidation();
|
2018-02-01 16:10:09 +01:00
|
|
|
void shutdownConsumers(thread_db* tdbb);
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2015-12-10 17:58:58 +01:00
|
|
|
void lockAndReadHeader(thread_db* tdbb, unsigned flags = 0);
|
|
|
|
static const unsigned CRYPT_HDR_INIT = 0x01;
|
|
|
|
static const unsigned CRYPT_HDR_NOWAIT = 0x02;
|
|
|
|
|
2016-06-03 14:52:46 +02:00
|
|
|
void addClumplet(Firebird::string& value, Firebird::ClumpletReader& block, UCHAR tag);
|
2017-01-31 15:38:53 +01:00
|
|
|
void calcDigitalSignature(thread_db* tdbb, Firebird::string& signature, const class Header& hdr);
|
|
|
|
void digitalySignDatabase(thread_db* tdbb, class CchHdr& hdr);
|
|
|
|
void checkDigitalSignature(thread_db* tdbb, const class Header& hdr);
|
2016-06-03 14:52:46 +02:00
|
|
|
|
2015-12-10 17:58:58 +01:00
|
|
|
BarSync sync;
|
2018-05-23 16:16:04 +02:00
|
|
|
Firebird::MetaName keyName, pluginName;
|
2016-01-29 16:46:06 +01:00
|
|
|
ULONG currentPage;
|
2018-02-01 16:10:09 +01:00
|
|
|
Firebird::Mutex pluginLoadMtx, cryptThreadMtx, holdersMutex;
|
|
|
|
AttVector keyProviders, keyConsumers;
|
2017-01-09 18:28:11 +01:00
|
|
|
Firebird::string hash;
|
2016-10-12 14:22:15 +02:00
|
|
|
Firebird::RefPtr<DbInfo> dbInfo;
|
2014-08-15 16:59:38 +02:00
|
|
|
Thread::Handle cryptThreadId;
|
2012-05-31 18:53:42 +02:00
|
|
|
Firebird::IDbCryptPlugin* cryptPlugin;
|
2017-01-31 15:38:53 +01:00
|
|
|
Factory* checkFactory;
|
2012-05-31 18:53:42 +02:00
|
|
|
Database& dbb;
|
|
|
|
Lock* stateLock;
|
|
|
|
Lock* threadLock;
|
2016-02-05 16:21:58 +01:00
|
|
|
Attachment* cryptAtt;
|
2015-12-15 17:54:23 +01:00
|
|
|
|
|
|
|
// This counter works only in a case when database encryption is changed.
|
|
|
|
// Traditional processing of AST can not be used for crypto manager.
|
|
|
|
// The problem is with taking state lock after AST.
|
|
|
|
// That should be done before next IO operation to guarantee database
|
|
|
|
// consistency, but IO operation may be requested from another AST
|
|
|
|
// (database cache blocking) when 'wait for a lock' operations are
|
|
|
|
// prohibited. I.e. we can't proceed with normal IO without a lock
|
|
|
|
// but at the same time can't take it.
|
|
|
|
|
|
|
|
// The solution is to use crypt versions counter in a lock (incremented
|
|
|
|
// by one each time one issues ALTER DATABASE ENCRYPT/DECRYPT), read
|
|
|
|
// it when lock can't be taken, store in slowIO variable, perform IO
|
|
|
|
// and compare stored value with current lock data. In case when values
|
|
|
|
// differ encryption status of database was changed during IO operation
|
|
|
|
// and such operation should be repeated. When data in lock does not
|
|
|
|
// change during IO that means that crypt rules remained the same even
|
|
|
|
// without state lock taken by us and therefore result of IO operation
|
|
|
|
// is correct. As soon as non-waiting attempt to take state lock succeeds
|
|
|
|
// slowIO mode is off (slowIO counter becomes zero) and we return to
|
|
|
|
// normal operation.
|
|
|
|
|
2015-12-10 17:58:58 +01:00
|
|
|
SINT64 slowIO;
|
2016-02-05 16:21:58 +01:00
|
|
|
bool crypt, process, down, run;
|
2016-09-15 16:37:50 +02:00
|
|
|
|
|
|
|
public:
|
|
|
|
Firebird::Mutex cryptAttMutex;
|
2012-05-31 18:53:42 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Jrd
|
|
|
|
|
|
|
|
|
|
|
|
#endif // JRD_CRYPTO_MANAGER
|