]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/base.cpp
Add explicit reference-counting base class
[user/henk/code/inspircd.git] / src / base.cpp
1 /*       +------------------------------------+
2  *       | Inspire Internet Relay Chat Daemon |
3  *       +------------------------------------+
4  *
5  *  InspIRCd: (C) 2002-2009 InspIRCd Development Team
6  * See: http://wiki.inspircd.org/Credits
7  *
8  * This program is free but copyrighted software; see
9  *            the file COPYING for details.
10  *
11  * ---------------------------------------------------
12  */
13
14 /* $Core */
15
16 #include "inspircd_config.h"
17 #include "base.h"
18 #include <time.h>
19 #include "inspircd.h"
20
21 const int bitfields[]           =       {1,2,4,8,16,32,64,128};
22 const int inverted_bitfields[]  =       {~1,~2,~4,~8,~16,~32,~64,~128};
23 std::map<std::string, ExtensionItem*> Extensible::extension_types;
24
25 classbase::classbase()
26 {
27 }
28
29 bool classbase::cull()
30 {
31         return true;
32 }
33
34 classbase::~classbase()
35 {
36 }
37
38 refcountbase::refcountbase() : refcount(0)
39 {
40 }
41
42 bool refcountbase::cull()
43 {
44         return (refcount == 0);
45 }
46
47 refcountbase::~refcountbase()
48 {
49 }
50
51 ExtensionItem::ExtensionItem(const std::string& Key, Module* mod) : key(Key), owner(mod)
52 {
53 }
54
55 ExtensionItem::~ExtensionItem()
56 {
57 }
58
59 void* ExtensionItem::get_raw(const Extensible* container)
60 {
61         ExtensibleStore::const_iterator i = container->extensions.find(key);
62         if (i == container->extensions.end())
63                 return NULL;
64         return i->second;
65 }
66
67 void* ExtensionItem::set_raw(Extensible* container, void* value)
68 {
69         std::pair<ExtensibleStore::iterator,bool> rv = 
70                 container->extensions.insert(std::make_pair(key, value));
71         if (rv.second)
72         {
73                 return NULL;
74         }
75         else
76         {
77                 void* old = rv.first->second;
78                 rv.first->second = value;
79                 return old;
80         }
81 }
82
83 void* ExtensionItem::unset_raw(Extensible* container)
84 {
85         ExtensibleStore::iterator i = container->extensions.find(key);
86         if (i == container->extensions.end())
87                 return NULL;
88         void* rv = i->second;
89         container->extensions.erase(i);
90         return rv;
91 }
92
93 bool Extensible::Register(ExtensionItem* item)
94 {
95         return Extensible::extension_types.insert(std::make_pair(item->key, item)).second;
96 }
97
98 std::vector<ExtensionItem*> Extensible::BeginUnregister(Module* module)
99 {
100         std::vector<ExtensionItem*> rv;
101         ExtensibleTypes::iterator i = extension_types.begin();
102         while (i != extension_types.end())
103         {
104                 ExtensibleTypes::iterator c = i++;
105                 if (c->second->owner == module)
106                 {
107                         rv.push_back(c->second);
108                         extension_types.erase(c);
109                 }
110         }
111         return rv;
112 }
113
114 void Extensible::doUnhookExtensions(const std::vector<ExtensionItem*>& toRemove)
115 {
116         for(std::vector<ExtensionItem*>::const_iterator i = toRemove.begin(); i != toRemove.end(); i++)
117         {
118                 ExtensibleStore::iterator e = extensions.find((**i).key);
119                 if (e != extensions.end())
120                 {
121                         (**i).free(e->second);
122                         extensions.erase(e);
123                 }
124         }
125 }
126
127 Extensible::~Extensible()
128 {
129         for(ExtensibleStore::iterator i = extensions.begin(); i != extensions.end(); ++i)
130         {
131                 ExtensionItem* type = GetItem(i->first);
132                 if (type)
133                         type->free(i->second);  
134                 else if (ServerInstance && ServerInstance->Logs)
135                         ServerInstance->Logs->Log("BASE", ERROR, "Extension type %s is not registered", i->first.c_str());
136         }
137 }
138
139 LocalExtItem::LocalExtItem(const std::string& Key, Module* mod) : ExtensionItem(Key, mod)
140 {
141 }
142
143 LocalExtItem::~LocalExtItem()
144 {
145 }
146
147 std::string LocalExtItem::serialize(SerializeFormat format, const Extensible* container, void* item)
148 {
149         return "";
150 }
151
152 void LocalExtItem::unserialize(SerializeFormat format, Extensible* container, const std::string& value)
153 {
154 }
155
156 LocalStringExt::LocalStringExt(const std::string& Key, Module* Owner)
157         : SimpleExtItem<std::string>(Key, Owner) { }
158
159 LocalStringExt::~LocalStringExt()
160 {
161 }
162
163 std::string LocalStringExt::serialize(SerializeFormat format, const Extensible* container, void* item)
164 {
165         if (item && format == FORMAT_USER)
166                 return *static_cast<std::string*>(item);
167         return "";
168 }
169
170 LocalIntExt::LocalIntExt(const std::string& Key, Module* mod) : LocalExtItem(Key, mod)
171 {
172 }
173
174 LocalIntExt::~LocalIntExt()
175 {
176 }
177
178 std::string LocalIntExt::serialize(SerializeFormat format, const Extensible* container, void* item)
179 {
180         if (format != FORMAT_USER)
181                 return "";
182         return ConvToStr(reinterpret_cast<intptr_t>(item));
183 }
184
185 intptr_t LocalIntExt::get(const Extensible* container)
186 {
187         return reinterpret_cast<intptr_t>(get_raw(container));
188 }
189
190 intptr_t LocalIntExt::set(Extensible* container, intptr_t value)
191 {
192         if (value)
193                 return reinterpret_cast<intptr_t>(set_raw(container, reinterpret_cast<void*>(value)));
194         else
195                 return reinterpret_cast<intptr_t>(unset_raw(container));
196 }
197
198 void LocalIntExt::free(void*)
199 {
200 }
201
202 StringExtItem::StringExtItem(const std::string& Key, Module* mod) : ExtensionItem(Key, mod)
203 {
204 }
205
206 StringExtItem::~StringExtItem()
207 {
208 }
209
210 std::string* StringExtItem::get(const Extensible* container)
211 {
212         return static_cast<std::string*>(get_raw(container));
213 }
214
215 std::string StringExtItem::serialize(SerializeFormat format, const Extensible* container, void* item)
216 {
217         return item ? *static_cast<std::string*>(item) : "";
218 }
219
220 void StringExtItem::unserialize(SerializeFormat format, Extensible* container, const std::string& value)
221 {
222         if (value.empty())
223                 unset(container);
224         else
225                 set(container, value);
226 }
227
228 void StringExtItem::set(Extensible* container, const std::string& value)
229 {
230         void* old = set_raw(container, new std::string(value));
231         delete static_cast<std::string*>(old);
232 }
233
234 void StringExtItem::unset(Extensible* container)
235 {
236         void* old = unset_raw(container);
237         delete static_cast<std::string*>(old);
238 }
239
240 void StringExtItem::free(void* item)
241 {
242         delete static_cast<std::string*>(item);
243 }