00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031 #ifndef MORE_GEN_UTILITY_H
00032 #define MORE_GEN_UTILITY_H
00033
00034 #include <utility>
00035 #include <functional>
00036 #include <string>
00037 #include <typeinfo>
00038 #include <stdexcept>
00039 #include <cassert>
00040 #include <more/bits/config.h>
00041
00042
00043 namespace more {
00044 namespace gen {
00045
00046 struct instance_counter
00047 {
00048 instance_counter()
00049 : ic(new int)
00050 {
00051 *ic = 0;
00052 }
00053
00054 instance_counter(const instance_counter& x)
00055 : ic(x.ic)
00056 {
00057 ++*ic;
00058 }
00059
00060 ~instance_counter()
00061 {
00062 if (!*ic)
00063 delete ic;
00064 else
00065 --*ic;
00066 }
00067
00068 instance_counter& operator=(const instance_counter& x)
00069 {
00070 if (!*ic)
00071 delete ic;
00072 ic = x.ic;
00073 ++*ic;
00074 return *this;
00075 }
00076
00077 bool is_unique() const { return !*ic; }
00078
00079 private:
00080 mutable int* ic;
00081 };
00082
00083
00084
00085
00086 template<typename T> class handle;
00087
00088
00089
00090 struct handle_data
00091 {
00092 handle_data() : m_refcnt(1) {}
00093 void ref() const { ++m_refcnt; }
00094 unsigned int unref() const { return --m_refcnt; }
00095
00096 private:
00097 template<typename T> friend class handle;
00098 mutable int m_refcnt;
00099 };
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119 template<typename T>
00120 struct handle
00121 {
00122 handle() : m_ptr(0) {}
00123
00124 explicit handle(T* p) : m_ptr(p) {}
00125
00126 handle(handle const& x)
00127 : m_ptr(x.m_ptr)
00128 {
00129 if (m_ptr)
00130 ++m_ptr->m_refcnt;
00131 }
00132
00133 handle& operator=(handle const& x)
00134 {
00135 if (m_ptr && !--m_ptr->m_refcnt)
00136 delete m_ptr;
00137 m_ptr = x.m_ptr;
00138 if (m_ptr)
00139 ++m_ptr->m_refcnt;
00140 return *this;
00141 }
00142
00143 ~handle()
00144 {
00145 if (m_ptr && !--m_ptr->m_refcnt)
00146 delete m_ptr;
00147 }
00148
00149 operator bool() const { return m_ptr; }
00150
00151 bool is_owner() const { return m_ptr->m_refcnt == 1; }
00152
00153 void check_data() const
00154 {
00155 #ifndef MORE_DEBUG
00156 if (!m_ptr)
00157 throw std::logic_error(
00158 std::string("more::gen::handle: "
00159 "Accessing invalid object, typeid = ")
00160 + typeid(*this).name());
00161 #endif
00162 }
00163
00164 T* datachk() const
00165 {
00166 check_data();
00167 return m_ptr;
00168 }
00169
00170 T* data() const { return m_ptr; }
00171
00172 T* data_ref() const { m_ptr->ref(); return m_ptr; }
00173
00174
00175
00176
00177
00178 void set_data(T* ptr)
00179 {
00180 if (m_ptr && !--m_ptr->m_refcnt)
00181 delete m_ptr;
00182 m_ptr = ptr;
00183 }
00184
00185 private:
00186 mutable T *m_ptr;
00187 };
00188
00189
00190
00191
00192
00193
00194
00195
00196 struct noncopyable
00197 {
00198 noncopyable() {}
00199 private:
00200 noncopyable(noncopyable const&) {}
00201 noncopyable const& operator=(noncopyable const&) { return *this; }
00202 };
00203
00204
00205
00206
00207 template<typename T, typename U>
00208 struct refpair
00209 {
00210
00211 typedef T first_type;
00212
00213 typedef U second_type;
00214
00215 refpair(T& x, U& y) : first(x), second(y) {}
00216
00217 refpair(refpair const& rp) : first(rp.first), second(rp.second) {}
00218
00219 refpair& operator=(std::pair<T, U> const& p)
00220 {
00221 first = p.first;
00222 second = p.second;
00223 return *this;
00224 }
00225
00226 T& first;
00227
00228 U& second;
00229 };
00230
00231
00232 template<typename T, typename U>
00233 inline refpair<T, U> tie(T& x, U& y) { return refpair<T, U>(x, y); }
00234
00235
00236
00237
00238
00239
00240 template<typename Pair>
00241 struct select1st : std::unary_function<Pair, typename Pair::first_type>
00242 {
00243 typename Pair::first_type&
00244 operator()(Pair& x) const
00245 {
00246 return x.first;
00247 }
00248 };
00249
00250
00251
00252
00253
00254 template<typename Pair>
00255 struct select2nd : std::unary_function<Pair, typename Pair::second_type>
00256 {
00257 typename Pair::second_type&
00258 operator()(Pair& x) const
00259 {
00260 return x.second;
00261 }
00262 };
00263
00264 }}
00265
00266 #endif