00001 /* ///////////////////////////////////////////////////////////////////////// 00002 * File: stlsoft/util/pod_veneer.hpp 00003 * 00004 * Purpose: Contains the pod_veneer template class. 00005 * 00006 * Created: 19th January 2002 00007 * Updated: 10th August 2009 00008 * 00009 * Home: http://stlsoft.org/ 00010 * 00011 * Copyright (c) 2002-2009, Matthew Wilson and Synesis Software 00012 * All rights reserved. 00013 * 00014 * Redistribution and use in source and binary forms, with or without 00015 * modification, are permitted provided that the following conditions are met: 00016 * 00017 * - Redistributions of source code must retain the above copyright notice, this 00018 * list of conditions and the following disclaimer. 00019 * - Redistributions in binary form must reproduce the above copyright notice, 00020 * this list of conditions and the following disclaimer in the documentation 00021 * and/or other materials provided with the distribution. 00022 * - Neither the name(s) of Matthew Wilson and Synesis Software nor the names of 00023 * any contributors may be used to endorse or promote products derived from 00024 * this software without specific prior written permission. 00025 * 00026 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 00027 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 00028 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 00029 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE 00030 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 00031 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 00032 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 00033 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 00034 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 00035 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 00036 * POSSIBILITY OF SUCH DAMAGE. 00037 * 00038 * ////////////////////////////////////////////////////////////////////// */ 00039 00040 00047 #ifndef STLSOFT_INCL_STLSOFT_UTIL_HPP_POD_VENEER 00048 #define STLSOFT_INCL_STLSOFT_UTIL_HPP_POD_VENEER 00049 00050 #ifndef STLSOFT_DOCUMENTATION_SKIP_SECTION 00051 # define STLSOFT_VER_STLSOFT_UTIL_HPP_POD_VENEER_MAJOR 4 00052 # define STLSOFT_VER_STLSOFT_UTIL_HPP_POD_VENEER_MINOR 0 00053 # define STLSOFT_VER_STLSOFT_UTIL_HPP_POD_VENEER_REVISION 1 00054 # define STLSOFT_VER_STLSOFT_UTIL_HPP_POD_VENEER_EDIT 49 00055 #endif /* !STLSOFT_DOCUMENTATION_SKIP_SECTION */ 00056 00057 /* ///////////////////////////////////////////////////////////////////////// 00058 * Includes 00059 */ 00060 00061 #ifndef STLSOFT_INCL_STLSOFT_H_STLSOFT 00062 # include <stlsoft/stlsoft.h> 00063 #endif /* !STLSOFT_INCL_STLSOFT_H_STLSOFT */ 00064 #ifndef STLSOFT_INCL_STLSOFT_UTIL_HPP_CONSTRAINTS 00065 # include <stlsoft/util/constraints.hpp> 00066 #endif /* !STLSOFT_INCL_STLSOFT_UTIL_HPP_CONSTRAINTS */ 00067 00068 /* ///////////////////////////////////////////////////////////////////////// 00069 * Namespace 00070 */ 00071 00072 #ifndef _STLSOFT_NO_NAMESPACE 00073 namespace stlsoft 00074 { 00075 #endif /* _STLSOFT_NO_NAMESPACE */ 00076 00077 /* ///////////////////////////////////////////////////////////////////////// 00078 * Classes 00079 */ 00080 00081 // class pod_veneer 00082 00119 template< ss_typename_param_k T 00120 , ss_typename_param_k CF 00121 , ss_typename_param_k DF 00122 > 00123 class pod_veneer 00124 : public T 00125 { 00126 public: 00128 typedef T value_type; 00130 typedef T pod_type; 00132 typedef CF constructor_function_type; 00134 typedef DF destructor_function_type; 00136 typedef pod_veneer<T, CF, DF> class_type; 00137 00138 public: 00140 pod_veneer() 00141 { 00142 // It is entirely appropriate to constrain instantiations of this class 00143 // to default construction only, since the whole raison d'etre of this 00144 // template is to bring RAII semantics to POD types. 00145 00146 // Construct the pod 00147 constructor_function_type()(static_cast<pod_type *>(this)); 00148 } 00149 00151 ~pod_veneer() stlsoft_throw_0() 00152 { 00153 // There are four important points to note about the destructor 00154 // 00155 // 1. It does not throw an exception. Therefore, the destroying type 00156 // must not throw an exception in its function call operator 00157 // 2. It uses the must_be_pod constraint to ensure that the pod_type is 00158 // a POD type 00159 // 3. It uses the must_be_same_size constraint to ensure that the 00160 // generated class is the same size as the pod_type, so that arrays of 00161 // the generated class can be used where arrays of the pod_type could be 00162 // used. The runtime assert is included for those compilers that do not 00163 // implement compile-time asserts. 00164 // 4. It is non-virtual, since it should not introduce a vtable (which 00165 // would change the size of the generated class) 00166 00167 // 2. Must be POD 00168 stlsoft_constraint_must_be_pod(pod_type); 00169 00170 // 3. Must use EDO 00171 stlsoft_constraint_must_be_same_size(class_type, pod_type); 00172 #if defined(STLSOFT_COMPILER_IS_WATCOM) 00173 STLSOFT_ASSERT(sizeof(class_type) == sizeof(pod_type)); 00174 #else /* ? compiler */ 00175 STLSOFT_MESSAGE_ASSERT("pod_veneer used for inappropriate type", sizeof(class_type) == sizeof(pod_type)); 00176 #endif /* compiler */ 00177 00178 // Destroy the pod 00179 destructor_function_type()(static_cast<pod_type *>(this)); 00180 } 00181 }; 00182 00183 /* ////////////////////////////////////////////////////////////////////// */ 00184 00185 #ifndef _STLSOFT_NO_NAMESPACE 00186 } // namespace stlsoft 00187 #endif /* _STLSOFT_NO_NAMESPACE */ 00188 00189 /* ////////////////////////////////////////////////////////////////////// */ 00190 00191 #endif /* !STLSOFT_INCL_STLSOFT_UTIL_HPP_POD_VENEER */ 00192 00193 /* ///////////////////////////// end of file //////////////////////////// */