8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-02-01 09:23:02 +01:00
firebird-mirror/src/common/classes/init.h

226 lines
4.5 KiB
C
Raw Normal View History

/*
* PROGRAM: Common Access Method
* MODULE: init.h
* DESCRIPTION: InitMutex, InitInstance - templates to help with initialization
*
* 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 Alexander Peshkoff
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2004 Alexander Peshkoff <peshkoff@mail.ru>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
#ifndef CLASSES_INIT_INSTANCE_H
#define CLASSES_INIT_INSTANCE_H
#include "fb_types.h"
#include "../common/classes/alloc.h"
namespace Firebird {
namespace StaticMutex
{
// Support for common mutex for various inits
extern Mutex* mutex;
void create();
void release();
}
// InstanceControl - interface for almost all global variables
class InstanceControl
{
public:
enum DtorPriority
{
PRIORITY_REGULAR,
PRIORITY_TLS_KEY
};
InstanceControl();
//
2009-03-05 01:52:25 +01:00
// GlobalPtr should not be directly derived from class with virtual functions -
// virtual table for its instances may become invalid in the moment,
// when cleanup is needed. Therefore indirect link via InstanceList and
2009-03-05 01:52:25 +01:00
// InstanceLink is established. This means more calls to memory allocator,
// but who cares for 100 global variables?
//
class InstanceList
{
public:
2009-03-05 07:05:50 +01:00
explicit InstanceList(DtorPriority p);
virtual ~InstanceList();
static void destructors();
private:
InstanceList* next;
DtorPriority priority;
virtual void dtor() = 0;
};
template <typename T, InstanceControl::DtorPriority P = InstanceControl::PRIORITY_REGULAR>
class InstanceLink : private InstanceList, public GlobalStorage
{
private:
T* link;
public:
2009-03-05 07:05:50 +01:00
explicit InstanceLink(T* l)
2009-04-04 18:39:31 +02:00
: InstanceControl::InstanceList(P), link(l)
{
fb_assert(link);
}
void dtor()
{
fb_assert(link);
if (link)
{
link->dtor();
link = NULL;
}
}
};
public:
static void destructors();
static void registerGdsCleanup(FPTR_VOID cleanup);
static void registerShutdown(FPTR_VOID shutdown);
};
// GlobalPtr - template to help declaring global varables
template <typename T>
class GlobalPtr : private InstanceControl
{
private:
T* instance;
public:
void dtor()
{
delete instance;
instance = 0;
}
2008-12-05 01:56:15 +01:00
GlobalPtr()
{
2008-01-23 20:03:16 +01:00
// This means - for objects with ctors/dtors that want to be global,
// provide ctor with MemoryPool& parameter. Even if it is ignored!
instance = FB_NEW(*getDefaultMemoryPool()) T(*getDefaultMemoryPool());
2010-02-08 08:57:33 +01:00
// Put ourselves into linked list for cleanup.
// Allocated pointer is saved by InstanceList::constructor.
new InstanceControl::InstanceLink<GlobalPtr>(this);
}
T* operator->() throw()
{
return instance;
}
operator T&() throw()
{
return *instance;
}
T* operator&() throw()
{
return instance;
}
};
// InitMutex - executes static void C::init() once and only once
template <typename C>
class InitMutex
{
private:
volatile bool flag;
public:
2008-12-05 01:56:15 +01:00
InitMutex()
: flag(false) { }
2008-12-05 01:56:15 +01:00
void init()
{
2010-02-08 08:57:33 +01:00
if (!flag)
{
MutexLockGuard guard(*StaticMutex::mutex);
2010-02-08 08:57:33 +01:00
if (!flag)
{
2008-02-04 12:59:42 +01:00
C::init();
flag = true;
2006-05-20 05:55:54 +02:00
}
}
2008-12-05 01:56:15 +01:00
}
void cleanup()
{
2010-02-08 08:57:33 +01:00
if (flag)
{
MutexLockGuard guard(*StaticMutex::mutex);
2010-02-08 08:57:33 +01:00
if (flag)
{
2008-02-04 12:59:42 +01:00
C::cleanup();
flag = false;
}
}
2008-12-05 01:56:15 +01:00
}
};
// InitInstance - allocate instance of class T on first request.
template <typename T>
class InitInstance : private InstanceControl
{
private:
T* instance;
volatile bool flag;
public:
2008-12-05 01:56:15 +01:00
InitInstance()
2010-02-11 02:08:20 +01:00
: instance(NULL), flag(false)
{ }
2008-12-05 01:56:15 +01:00
T& operator()()
{
2010-02-08 08:57:33 +01:00
if (!flag)
{
MutexLockGuard guard(*StaticMutex::mutex);
2010-02-08 08:57:33 +01:00
if (!flag)
{
instance = FB_NEW(*getDefaultMemoryPool()) T(*getDefaultMemoryPool());
2008-02-04 12:59:42 +01:00
flag = true;
2010-02-08 08:57:33 +01:00
// Put ourselves into linked list for cleanup.
// Allocated pointer is saved by InstanceList::constructor.
new InstanceControl::InstanceLink<InitInstance>(this);
}
}
return *instance;
}
void dtor()
{
MutexLockGuard guard(*StaticMutex::mutex);
flag = false;
delete instance;
2010-02-11 02:08:20 +01:00
instance = NULL;
}
};
} //namespace Firebird
#endif // CLASSES_INIT_INSTANCE_H