/* * PROGRAM: Firebird interface. * MODULE: ImplementHelper.h * DESCRIPTION: Tools to help create interfaces. * * 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 "firebird/Plugin.h" #include "firebird/Timer.h" #include "firebird/Provider.h" #include "../common/classes/alloc.h" #include "gen/iberror.h" #include "../yvalve/gds_proto.h" #include "../common/classes/init.h" #include "../common/classes/auto.h" #include "../common/classes/RefCounted.h" #include "consts_pub.h" namespace Firebird { // If you need interface on stack, use template AutoPtr // as second parameter to store it. typedef SimpleDispose AutoDisposable; // Implement standard interface and plugin functions // Helps to implement generic versioned interfaces template class VersionedIface : public C { public: VersionedIface() { } int FB_CARG getVersion() { return V; } IPluginModule* FB_CARG getModule(); private: VersionedIface(const VersionedIface&); VersionedIface& operator=(const VersionedIface&); }; // Helps to implement versioned interfaces on stack or static template class AutoIface : public VersionedIface { public: AutoIface() { } void* operator new(size_t, void* memory) throw() { return memory; } }; // Helps to implement disposable interfaces template class DisposeIface : public VersionedIface, public GlobalStorage { public: DisposeIface() { } }; // Helps to implement standard interfaces template class RefCntIface : public VersionedIface, public GlobalStorage { public: RefCntIface() : refCounter(0) { } #ifdef DEV_BUILD ~RefCntIface() { fb_assert(refCounter.value() == 0); } #endif void FB_CARG addRef() { ++refCounter; } protected: AtomicCounter refCounter; }; // Helps to implement plugins template class StdPlugin : public RefCntIface { private: IRefCounted* owner; public: StdPlugin() : owner(NULL) { } IRefCounted* FB_CARG getOwner() { return owner; } void FB_CARG setOwner(IRefCounted* iface) { owner = iface; } }; // Trivial factory template class SimpleFactoryBase : public AutoIface { public: IPluginBase* FB_CARG createPlugin(IPluginConfig* factoryParameter) { P* p = new P(factoryParameter); p->addRef(); return p; } }; template class SimpleFactory : public Static > { }; // Ensure access to cached pointer to master interface class CachedMasterInterface { public: static void set(IMaster* master); protected: static IMaster* getMasterInterface(); }; // Base for interface type independent accessors template class AccessAutoInterface : public CachedMasterInterface { public: explicit AccessAutoInterface(C* aPtr) : ptr(aPtr) { } operator C*() { return ptr; } C* operator->() { return ptr; } private: C* ptr; }; // Master interface access class MasterInterfacePtr : public AccessAutoInterface { public: MasterInterfacePtr() : AccessAutoInterface(getMasterInterface()) { } }; // Generic plugins interface access class PluginManagerInterfacePtr : public AccessAutoInterface { public: PluginManagerInterfacePtr() : AccessAutoInterface(getMasterInterface()->getPluginManager()) { } /* explicit PluginManagerInterfacePtr(IMaster* master) : AccessAutoInterface(master->getPluginManager()) { } */ }; // Control timer interface access class TimerInterfacePtr : public AccessAutoInterface { public: TimerInterfacePtr() : AccessAutoInterface(getMasterInterface()->getTimerControl()) { } }; // Distributed transactions coordinator access class DtcInterfacePtr : public AccessAutoInterface { public: DtcInterfacePtr() : AccessAutoInterface(getMasterInterface()->getDtc()) { } }; // When process exits, dynamically loaded modules (for us plugin modules) // are unloaded first. As the result all global variables in plugin are already destroyed // when yvalve is starting fb_shutdown(). This causes almost unavoidable segfault. // To avoid it this class is added - it detects spontaneous (not by PluginManager) // module unload and notifies PluginManager about this said fact. class UnloadDetectorHelper : public VersionedIface { public: typedef void VoidNoParam(); explicit UnloadDetectorHelper(MemoryPool&) : cleanup(NULL), flagOsUnload(false) { } void registerMe() { PluginManagerInterfacePtr()->registerModule(this); flagOsUnload = true; } ~UnloadDetectorHelper() { if (flagOsUnload) { PluginManagerInterfacePtr()->unregisterModule(this); doClean(); } } bool unloadStarted() { return !flagOsUnload; } void setCleanup(VoidNoParam* function) { cleanup = function; } private: VoidNoParam* cleanup; bool flagOsUnload; void FB_CARG doClean() { flagOsUnload = false; if (cleanup) { cleanup(); cleanup = NULL; } } }; typedef GlobalPtr UnloadDetector; extern UnloadDetector myModule; template IPluginModule* VersionedIface::getModule() { return &myModule; } // Default replacement for missing virtual functions class DefaultMissingEntrypoint { public: virtual void FB_CARG noEntrypoint() { Arg::Gds(isc_wish_list).raise(); } }; // Helps to create update information template class MakeUpgradeInfo { public: MakeUpgradeInfo() { ui.missingFunctionClass = &missing; ui.clientModule = &myModule; } operator UpgradeInfo*() { return &ui; } private: M missing; struct UpgradeInfo ui; }; class InternalMessageBuffer : public FbMessage { public: InternalMessageBuffer(unsigned aBlrLength, const unsigned char* aBlr, unsigned aBufferLength, unsigned char* aBuffer) { blrLength = aBlrLength; blr = aBlr; bufferLength = aBufferLength; buffer = aBuffer; } }; } // namespace Firebird #endif // FB_COMMON_CLASSES_IMPLEMENT_HELPER