/usr/include/diagnostics/macros/invariance_annotation.hpp is in libdiagnostics-dev 0.3.3-12build1.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 | /*
* Diagnostics - a unified framework for code annotation, logging,
* program monitoring, and unit-testing.
*
* Copyright (C) 2009 Christian Schallhart <christian@schallhart.net>,
* Michael Tautschnig <tautschnig@forsyte.de>
* 2008 model.in.tum.de group, FORSYTE group
* 2006-2007 model.in.tum.de group
* 2002-2005 Christian Schallhart
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
/**
* @file diagnostics/macros/invariance_annotation.hpp
*
* $Id: invariance_annotation.hpp,v 1.12 2005/06/23 09:54:22 esdentem Exp $
*
* @author Christian Schallhart
*
* @brief [LEVEL: beta] @ref DIAGNOSTICS_BASE_CLASS_INVARIANCE_ENTER
* and @ref DIAGNOSTICS_BASE_CLASS_INVARIANCE_EXIT, @ref
* DIAGNOSTICS_BASE_CLASS_INVARIANCE_GUARD, and @ref
* DIAGNOSTICS_BASE_CLASS_INVARIANCE_ASSERT.
*
* @test diagnostics/macros/invariance_annotation.t.cpp
*/
#ifndef DIAGNOSTICS__MACROS__INVARIANCE_ANNOTATION_HPP__INCLUDE_GUARD
#define DIAGNOSTICS__MACROS__INVARIANCE_ANNOTATION_HPP__INCLUDE_GUARD
// ::diagnostics::logging_facility::log
#include <diagnostics/frame/logging_facility.hpp>
// DIAGNOSTICS_BASE_CONCAT
#include <diagnostics/util/preprocessor.hpp>
////////////////////////////////////////////////////////////////////////////////
DIAGNOSTICS_NAMESPACE_BEGIN;
INTERNAL_NAMESPACE_BEGIN;
/**
* @brief Helper for Class-Invariance Checking. On construction and
* description, the invariance is checked. The check is performed by
* calling an argumentless void method. This helper is used when the
* invariance is allowed to throw.
*/
template<typename CLASS>
class Class_Invariance_Guard_Throw
{
public:
explicit inline Class_Invariance_Guard_Throw(CLASS const * const object);
inline ~Class_Invariance_Guard_Throw() noexcept(false);
inline void no_final_check() { m_final_check=false; }
private:
Class_Invariance_Guard_Throw(Class_Invariance_Guard_Throw const & other);
Class_Invariance_Guard_Throw & operator=(Class_Invariance_Guard_Throw const & other);
private:
CLASS const & m_object;
bool m_final_check;
};
template<typename CLASS>
Class_Invariance_Guard_Throw<CLASS>::Class_Invariance_Guard_Throw(CLASS const * const object)
: m_object(*object),
m_final_check(true)
{
(m_object.m_class_invariance)();
}
template<typename CLASS>
Class_Invariance_Guard_Throw<CLASS>::~Class_Invariance_Guard_Throw() noexcept(false)
{
if(m_final_check) (m_object.m_class_invariance)();
}
/**
* @brief Helper for Class-Invariance Checking. On construction and
* description, the invariance is checked. The check is performed by
* calling an argumentless void method.
*/
template<typename CLASS>
class Class_Invariance_Guard
{
public:
explicit inline Class_Invariance_Guard(CLASS const * const object);
inline ~Class_Invariance_Guard();
private:
Class_Invariance_Guard(Class_Invariance_Guard const & other);
Class_Invariance_Guard & operator=(Class_Invariance_Guard const & other);
private:
CLASS const & m_object;
};
template<typename CLASS>
Class_Invariance_Guard<CLASS>::Class_Invariance_Guard(CLASS const * const object)
: m_object(*object)
{
(m_object.m_class_invariance)();
}
template<typename CLASS>
Class_Invariance_Guard<CLASS>::~Class_Invariance_Guard()
{
(m_object.m_class_invariance)();
}
INTERNAL_NAMESPACE_END;
DIAGNOSTICS_NAMESPACE_END;
/**
* @brief an invariance block. The type Self must be defined in this
* context and has to refer to the current class. Self must have a
* public method void Self::m_class_invariance() const. This method
* must implement the invariance check. A new block is
* opened. Exceptions thrown by the invariance method are handled as
* well as possible.
*
* @attention If the block is left by an exception AND the invariance
* throws an exception, then the invariance's exception (if of type @a
* BASE) is logged with (@a LEVEL,@ref TYPE_TRACE,@a NAME,@a WHAT),
* and is discared afterwards. Execptions of type @ref
* diagnostics::unittest::Test_Exception can be handled explicitly --
* all other exceptions are unkown and coresponding message is
* generated. The exception of the method is rethrown. This might
* lead to counter-intuitive behaviors: Depending on whether an
* excpetion is thrown or not by the method, invariance violations
* lead to different behaviors. However, since invariance checking is
* used for mainly for debuging purposes, and any exception thrown by
* an invariance check is accompanied by a corresponding log-message,
* which indicates failure anyhow, this is acceptable. BUT IT IS NOT
* ACCEPTABLE TO EXPOSE SUCH A BEHAVIOR TO A CLIENT OF A MODULE.
*
* @attention THIS FEATURE MIGHT BE DELETED IN THE FUTURE
*/
#define DIAGNOSTICS_BASE_CLASS_THROWING_INVARIANCE_ENTER \
do { ::DIAGNOSTICS_NAMESPACE::INTERNAL_NAMESPACE::Class_Invariance_Guard_Throw<Self> \
internal__invariance_guard(this); try {
#define DIAGNOSTICS_INTERNAL_INVARIANCE_HANDLE_TE1(LEVEL,NR_WHAT) \
catch(::diagnostics::unittest::Test_Exception & e) { \
DIAGNOSTICS_BASE_LOG(LEVEL, \
::diagnostics::TYPE_TRACE, \
NR_WHAT, \
::std::string("dropped invariance exception EXCEPTION=\"") \
+ e.name() \
+ "\" WHAT=\"" \
+ e.what() \
+ "\""); \
}
#define DIAGNOSTICS_INTERNAL_INVARIANCE_HANDLE_TE0(LEVEL,NR_WHAT)
#define DIAGNOSTICS_BASE_CLASS_THROWING_INVARIANCE_EXIT(LEVEL,NR_WHAT,HANDLE_TE_EXLPLICITLY,BASE,STR_WHAT) \
} \
catch(...) { \
internal__invariance_guard.no_final_check(); \
try { \
this->m_class_invariance(); \
} \
DIAGNOSTICS_BASE_CONCAT(DIAGNOSTICS_INTERNAL_INVARIANCE_HANDLE_TE,HANDLE_TE_EXLPLICITLY)((LEVEL),(NR_WHAT)) \
catch(BASE & e) { \
DIAGNOSTICS_BASE_LOG((LEVEL), \
::diagnostics::TYPE_TRACE, \
(NR_WHAT), \
::std::string("dropped invariance exception ") \
+ STR_WHAT(e)); \
} \
catch(...){ \
DIAGNOSTICS_BASE_LOG((LEVEL), \
::diagnostics::TYPE_TRACE, \
(NR_WHAT), \
"dropped invariance exception"); \
} \
throw; \
} \
} while(false)
/**
* @brief an invariance guard. The type Self must be defined in this
* context and has to refer to the current class. Self must have a
* public method void Self::m_class_invariance() const. This method
* must implement the invariance check. A new block is opened.
*
* @attention if the invariance method throws, the behavior is undefined.
*/
#define DIAGNOSTICS_BASE_CLASS_INVARIANCE_GUARD \
::DIAGNOSTICS_NAMESPACE::INTERNAL_NAMESPACE::Class_Invariance_Guard<Self> \
internal__invariance_guard(this)
/**
* @brief an invariance block. Sets up a @ref
* DIAGNOSTICS_BASE_CLASS_INVARIANCE_GUARD within a new block.
*
* @attention if the invariance method throws, the behavior is undefined.
*/
#define DIAGNOSTICS_BASE_CLASS_INVARIANCE_ENTER \
do { DIAGNOSTICS_BASE_CLASS_INVARIANCE_GUARD; {
/**
* @brief end of an invariance block.
*/
#define DIAGNOSTICS_BASE_CLASS_INVARIANCE_EXIT \
} } while(false)
/**
* @brief Assert the current class invariance, i.e., calls the invariance.
*/
#define DIAGNOSTICS_BASE_CLASS_INVARIANCE_ASSERT (this->m_class_invariance())
#endif
// vim:ts=4:sw=4
|