/* * PROGRAM: Firebird interface. * MODULE: ImplementHelper.h * DESCRIPTION: Tools to help write plugins. * * 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) 2010 Alex Peshkov * and all contributors signed below. * * All Rights Reserved. * Contributor(s): ______________________________________. * * */ #ifndef FB_COMMON_CLASSES_IMPLEMENT_HELPER #define FB_COMMON_CLASSES_IMPLEMENT_HELPER #include "FirebirdPluginApi.h" #include "../common/classes/alloc.h" #include "gen/iberror.h" #include "../common/config/config.h" #include "../yvalve/gds_proto.h" #include "../common/classes/init.h" namespace Firebird { // If you need interface on stack, use template AutoPtr // as second parameter to store it. class AutoInterface { public: static void clear(Interface* ptr) { if (ptr) { ptr->release(); } } }; // Implement standard interface and plugin functions template class Versioned : public C, public S { public: int FB_CARG version() { return V; } }; template class StackIface : public Versioned { #ifdef DEV_BUILD private: int counter; public: // StackIface is destroyed in same frame where created, // therefore final release() call is not expected, // therefore initialize check counter with 0, not 1. StackIface() : counter(0) { } ~StackIface() { fb_assert(counter == 0); } #endif public: void FB_CARG addRef() { #ifdef DEV_BUILD ++counter; #endif } int FB_CARG release() { #ifdef DEV_BUILD --counter; #endif return 1; } void* operator new(size_t, void* memory) throw() { return memory; } }; template class StdIface : public Versioned { public: StdIface() : refCounter(1) { } #ifdef DEV_BUILD ~StdIface() { // fb_assert(refCounter.value() == 0); } #endif void FB_CARG addRef() { ++refCounter; } protected: AtomicCounter refCounter; }; template class StdPlugin : public StdIface { private: Interface* own; public: StdPlugin() : own(NULL) { } Interface* FB_CARG owner(Interface* iface) { Interface* rc = own; own = iface; return rc; } }; // Trivial factory template class SimpleFactoryBase : public StackIface { public: Plugin* FB_CARG createPlugin(IFactoryParameter* factoryParameter) { return new P(factoryParameter); } }; template class SimpleFactory : public Static > { }; class DefaultMissingEntrypoint { public: virtual void FB_CARG noEntrypoint() { Arg::Gds(isc_wish_list).raise(); } }; // Template to help with loop in the set of plugins template class PluginsSet { public: PluginsSet(unsigned int interfaceType, unsigned int desiredVersion, const char* namesList = NULL) : masterInterface(fb_get_master_interface()), pluginInterface(masterInterface->getPluginInterface()), missing(), pluginSet(pluginInterface->getPlugins(interfaceType, namesList ? namesList : Config::getPlugins(interfaceType), desiredVersion, &missing)), currentPlugin(NULL) { getPlugin(); } bool hasData() { return currentPlugin; } const char* name() { return hasData() ? pluginSet->name() : NULL; } const char* module() { return hasData() ? pluginSet->module() : NULL; } P* plugin() { return currentPlugin; } void next() { if (hasData()) { pluginInterface->releasePlugin(currentPlugin); currentPlugin = NULL; pluginSet->next(); getPlugin(); } } void set(const char* newName) { if (hasData()) { pluginInterface->releasePlugin(currentPlugin); currentPlugin = NULL; } pluginSet->set(newName); getPlugin(); } ~PluginsSet() { if (hasData()) { pluginInterface->releasePlugin(currentPlugin); currentPlugin = NULL; } if (pluginSet) pluginSet->release(); } private: AutoPtr masterInterface; AutoPtr pluginInterface; M missing; // AutoPtr pluginSet; IPluginSet *pluginSet; P* currentPlugin; void getPlugin() { currentPlugin = (P*) pluginSet->plugin(); } }; class PluginInterface : public AutoPtr { public: PluginInterface() : AutoPtr(NULL) { IMaster* mi = fb_get_master_interface(); reset(mi->getPluginInterface()); mi->release(); } }; } // namespace Firebird #endif // FB_COMMON_CLASSES_IMPLEMENT_HELPER