2012-05-31 18:53:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: Firebird samples.
|
|
|
|
* MODULE: CryptKeyHolder.cpp
|
|
|
|
* DESCRIPTION: Sample of how key holder may be written.
|
|
|
|
*
|
|
|
|
* 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): ______________________________________.
|
|
|
|
*/
|
|
|
|
|
2016-03-11 16:11:38 +01:00
|
|
|
#include "../interfaces/ifaceExamples.h"
|
2012-05-31 18:53:42 +02:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
2015-01-12 01:21:38 +01:00
|
|
|
class PluginModule : public IPluginModuleImpl<PluginModule, CheckStatusWrapper>
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
PluginModule()
|
2016-04-04 19:09:26 +02:00
|
|
|
: pluginManager(NULL)
|
2012-05-31 18:53:42 +02:00
|
|
|
{ }
|
|
|
|
|
|
|
|
~PluginModule()
|
|
|
|
{
|
2016-04-04 19:09:26 +02:00
|
|
|
if (pluginManager)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
pluginManager->unregisterModule(this);
|
|
|
|
doClean();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-04 19:09:26 +02:00
|
|
|
void registerMe(IPluginManager* m)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-04-04 19:09:26 +02:00
|
|
|
pluginManager = m;
|
|
|
|
pluginManager->registerModule(this);
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
|
2014-09-29 13:03:47 +02:00
|
|
|
void doClean()
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-04-04 19:09:26 +02:00
|
|
|
pluginManager = NULL;
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
|
2018-03-13 19:24:46 +01:00
|
|
|
void threadDetach() {};
|
|
|
|
|
2012-05-31 18:53:42 +02:00
|
|
|
private:
|
2016-04-04 19:09:26 +02:00
|
|
|
IPluginManager* pluginManager;
|
2012-05-31 18:53:42 +02:00
|
|
|
};
|
|
|
|
|
2015-01-12 01:21:38 +01:00
|
|
|
class CryptKeyHolder : public IKeyHolderPluginImpl<CryptKeyHolder, CheckStatusWrapper>
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
public:
|
2014-09-29 13:03:47 +02:00
|
|
|
explicit CryptKeyHolder(IPluginConfig* cnf) throw()
|
2016-02-03 14:20:41 +01:00
|
|
|
: callbackInterface(this), named(NULL), config(cnf), key(0), owner(NULL)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
config->addRef();
|
|
|
|
}
|
|
|
|
|
|
|
|
~CryptKeyHolder()
|
|
|
|
{
|
|
|
|
config->release();
|
|
|
|
}
|
|
|
|
|
|
|
|
// IKeyHolderPlugin implementation
|
2015-01-12 01:21:38 +01:00
|
|
|
int keyCallback(CheckStatusWrapper* status, ICryptKeyCallback* callback);
|
|
|
|
ICryptKeyCallback* keyHandle(CheckStatusWrapper* status, const char* keyName);
|
2017-03-16 10:16:30 +01:00
|
|
|
ICryptKeyCallback* chainHandle(CheckStatusWrapper* status);
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2014-09-29 13:03:47 +02:00
|
|
|
int release()
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
if (--refCounter == 0)
|
|
|
|
{
|
|
|
|
delete this;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-09-29 13:03:47 +02:00
|
|
|
void addRef()
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
++refCounter;
|
|
|
|
}
|
|
|
|
|
2014-09-29 13:03:47 +02:00
|
|
|
void setOwner(Firebird::IReferenceCounted* o)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
owner = o;
|
|
|
|
}
|
|
|
|
|
2014-09-29 13:03:47 +02:00
|
|
|
IReferenceCounted* getOwner()
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
return owner;
|
|
|
|
}
|
|
|
|
|
2016-03-11 16:11:38 +01:00
|
|
|
ISC_UCHAR getKey()
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
2017-01-09 18:28:48 +01:00
|
|
|
FB_BOOLEAN useOnlyOwnKeys(CheckStatusWrapper* status)
|
|
|
|
{
|
|
|
|
IConfigEntry* e = getEntry(status, "OnlyOwnKey");
|
|
|
|
if (!e)
|
|
|
|
return FB_TRUE; // safe default
|
|
|
|
|
|
|
|
FB_BOOLEAN rc = e->getBoolValue();
|
|
|
|
e->release();
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2012-05-31 18:53:42 +02:00
|
|
|
private:
|
2015-01-12 01:21:38 +01:00
|
|
|
class CallbackInterface : public ICryptKeyCallbackImpl<CallbackInterface, CheckStatusWrapper>
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
public:
|
2012-08-08 04:20:30 +02:00
|
|
|
explicit CallbackInterface(CryptKeyHolder* p)
|
2016-02-03 14:20:41 +01:00
|
|
|
: holder(p)
|
2012-05-31 18:53:42 +02:00
|
|
|
{ }
|
|
|
|
|
2014-09-29 13:03:47 +02:00
|
|
|
unsigned int callback(unsigned int, const void*, unsigned int length, void* buffer)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-03-11 16:11:38 +01:00
|
|
|
ISC_UCHAR k = holder->getKey();
|
2012-05-31 18:53:42 +02:00
|
|
|
if (!k)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (length > 0 && buffer)
|
|
|
|
{
|
|
|
|
memcpy(buffer, &k, 1);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-02-03 14:20:41 +01:00
|
|
|
private:
|
|
|
|
CryptKeyHolder* holder;
|
|
|
|
};
|
|
|
|
|
|
|
|
class NamedCallback : public ICryptKeyCallbackImpl<NamedCallback, CheckStatusWrapper>
|
|
|
|
{
|
|
|
|
public:
|
2016-03-11 16:11:38 +01:00
|
|
|
NamedCallback(NamedCallback* n, const char* nm, ISC_UCHAR k)
|
2016-02-03 14:20:41 +01:00
|
|
|
: next(n), key(k)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-02-03 14:20:41 +01:00
|
|
|
strncpy(name, nm, sizeof(name));
|
|
|
|
name[sizeof(name) - 1] = 0;
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
|
2016-02-03 14:20:41 +01:00
|
|
|
unsigned int callback(unsigned int, const void*, unsigned int length, void* buffer)
|
|
|
|
{
|
|
|
|
memcpy(buffer, &key, 1);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
~NamedCallback()
|
|
|
|
{
|
|
|
|
delete next;
|
|
|
|
}
|
|
|
|
|
|
|
|
char name[32];
|
|
|
|
NamedCallback* next;
|
2016-03-11 16:11:38 +01:00
|
|
|
ISC_UCHAR key;
|
2012-05-31 18:53:42 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
CallbackInterface callbackInterface;
|
2016-02-03 14:20:41 +01:00
|
|
|
NamedCallback *named;
|
2012-05-31 18:53:42 +02:00
|
|
|
|
|
|
|
IPluginConfig* config;
|
2016-03-11 16:11:38 +01:00
|
|
|
ISC_UCHAR key;
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2016-03-11 16:11:38 +01:00
|
|
|
FbSampleAtomic refCounter;
|
2014-09-29 13:03:47 +02:00
|
|
|
IReferenceCounted* owner;
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2016-02-03 14:20:41 +01:00
|
|
|
IConfigEntry* getEntry(CheckStatusWrapper* status, const char* entryName);
|
2012-05-31 18:53:42 +02:00
|
|
|
};
|
|
|
|
|
2016-02-03 14:20:41 +01:00
|
|
|
IConfigEntry* CryptKeyHolder::getEntry(CheckStatusWrapper* status, const char* entryName)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-02-03 14:20:41 +01:00
|
|
|
IConfig* def = config->getDefaultConfig(status);
|
|
|
|
if (status->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
IConfigEntry* confEntry = def->find(status, entryName);
|
|
|
|
def->release();
|
|
|
|
if (status->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return confEntry;
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
|
2015-01-12 01:21:38 +01:00
|
|
|
int CryptKeyHolder::keyCallback(CheckStatusWrapper* status, ICryptKeyCallback* callback)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
if (key != 0)
|
|
|
|
return 1;
|
|
|
|
|
2016-02-03 14:20:41 +01:00
|
|
|
IConfigEntry* confEntry = getEntry(status, "Auto");
|
2012-06-03 05:00:24 +02:00
|
|
|
|
2012-05-31 18:53:42 +02:00
|
|
|
if (confEntry)
|
|
|
|
{
|
2014-07-02 14:57:12 +02:00
|
|
|
FB_BOOLEAN b = confEntry->getBoolValue();
|
2012-05-31 18:53:42 +02:00
|
|
|
confEntry->release();
|
2014-07-02 14:57:12 +02:00
|
|
|
if (b)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
key = 0x5a;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (callback && callback->callback(0, NULL, 1, &key) != 1)
|
|
|
|
{
|
|
|
|
key = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-01-12 01:21:38 +01:00
|
|
|
ICryptKeyCallback* CryptKeyHolder::keyHandle(CheckStatusWrapper* status, const char* keyName)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-02-03 14:20:41 +01:00
|
|
|
if (keyName[0] == 0)
|
|
|
|
return &callbackInterface;
|
|
|
|
|
|
|
|
for (NamedCallback* n = named; n; n = n->next)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-02-03 14:20:41 +01:00
|
|
|
if (strcmp(keyName, n->name) == 0)
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
char kn[40];
|
|
|
|
strcpy(kn, "Key");
|
2016-03-03 23:53:31 +01:00
|
|
|
strncat(kn, keyName, sizeof(kn) - 3 - 1);
|
2016-02-03 14:20:41 +01:00
|
|
|
kn[sizeof(kn) - 1] = 0;
|
|
|
|
|
|
|
|
IConfigEntry* confEntry = getEntry(status, kn);
|
|
|
|
if (confEntry)
|
|
|
|
{
|
2016-02-19 11:39:11 +01:00
|
|
|
int k = confEntry->getIntValue();
|
2016-02-03 14:20:41 +01:00
|
|
|
confEntry->release();
|
|
|
|
if (k > 0 && k < 256)
|
|
|
|
{
|
2016-03-11 16:11:38 +01:00
|
|
|
named = new NamedCallback(named, keyName, static_cast<ISC_UCHAR>(k));
|
2016-02-03 14:20:41 +01:00
|
|
|
return named;
|
|
|
|
}
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
|
2016-02-03 14:20:41 +01:00
|
|
|
return NULL;
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
|
2017-03-16 10:16:30 +01:00
|
|
|
ICryptKeyCallback* CryptKeyHolder::chainHandle(CheckStatusWrapper* status)
|
|
|
|
{
|
|
|
|
return &callbackInterface;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-12 01:21:38 +01:00
|
|
|
class Factory : public IPluginFactoryImpl<Factory, CheckStatusWrapper>
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
|
|
|
public:
|
2015-01-12 01:21:38 +01:00
|
|
|
IPluginBase* createPlugin(CheckStatusWrapper* status, IPluginConfig* factoryParameter)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-04-04 19:09:26 +02:00
|
|
|
CryptKeyHolder* p = new CryptKeyHolder(factoryParameter);
|
|
|
|
p->addRef();
|
|
|
|
return p;
|
2012-05-31 18:53:42 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-04-04 19:09:26 +02:00
|
|
|
PluginModule module;
|
2012-05-31 18:53:42 +02:00
|
|
|
Factory factory;
|
|
|
|
|
|
|
|
} // anonymous namespace
|
|
|
|
|
2016-04-04 19:09:26 +02:00
|
|
|
extern "C" void FB_DLL_EXPORT FB_PLUGIN_ENTRY_POINT(IMaster* master)
|
2012-05-31 18:53:42 +02:00
|
|
|
{
|
2016-04-04 19:09:26 +02:00
|
|
|
IPluginManager* pluginManager = master->getPluginManager();
|
2012-05-31 18:53:42 +02:00
|
|
|
|
2016-04-04 19:09:26 +02:00
|
|
|
module.registerMe(pluginManager);
|
2015-02-18 16:01:17 +01:00
|
|
|
pluginManager->registerPluginFactory(IPluginManager::TYPE_KEY_HOLDER, "CryptKeyHolder_example",
|
2012-05-31 18:53:42 +02:00
|
|
|
&factory);
|
|
|
|
}
|
2016-04-04 19:09:26 +02:00
|
|
|
|