/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2016 Icinga Development Team (https://www.icinga.org/) * * * * This program is free software; you can redistribute it and/or * * modify it under the terms of the GNU General Public License * * as published by the Free Software Foundation; either version 2 * * of the License, or (at your option) any later version. * * * * This program 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 General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with this program; if not, write to the Free Software Foundation * * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. * ******************************************************************************/ #include "base/object.hpp" #include "base/value.hpp" #include "base/dictionary.hpp" #include "base/primitivetype.hpp" #include "base/utility.hpp" #include "base/timer.hpp" #include "base/logger.hpp" #include using namespace icinga; DEFINE_TYPE_INSTANCE(Object); #ifdef I2_LEAK_DEBUG static boost::mutex l_ObjectCountLock; static std::map l_ObjectCounts; static Timer::Ptr l_ObjectCountTimer; #endif /* I2_LEAK_DEBUG */ /** * Default constructor for the Object class. */ Object::Object(void) : m_References(0), m_Mutex(0) #ifdef I2_DEBUG , m_LockOwner(0) #endif /* I2_DEBUG */ { } /** * Destructor for the Object class. */ Object::~Object(void) { delete reinterpret_cast(m_Mutex); } /** * Returns a string representation for the object. */ String Object::ToString(void) const { return "Object of type '" + GetReflectionType()->GetName() + "'"; } #ifdef I2_DEBUG /** * Checks if the calling thread owns the lock on this object. * * @returns True if the calling thread owns the lock, false otherwise. */ bool Object::OwnsLock(void) const { #ifdef _WIN32 DWORD tid = InterlockedExchangeAdd(&m_LockOwner, 0); return (tid == GetCurrentThreadId()); #else /* _WIN32 */ pthread_t tid = __sync_fetch_and_add(&m_LockOwner, 0); return (tid == pthread_self()); #endif /* _WIN32 */ } #endif /* I2_DEBUG */ void Object::SetField(int id, const Value&, bool, const Value&) { if (id == 0) BOOST_THROW_EXCEPTION(std::runtime_error("Type field cannot be set.")); else BOOST_THROW_EXCEPTION(std::runtime_error("Invalid field ID.")); } Value Object::GetField(int id) const { if (id == 0) return GetReflectionType()->GetName(); else BOOST_THROW_EXCEPTION(std::runtime_error("Invalid field ID.")); } void Object::Validate(int types, const ValidationUtils& utils) { /* Nothing to do here. */ } void Object::ValidateField(int id, const Value& value, const ValidationUtils& utils) { /* Nothing to do here. */ } void Object::NotifyField(int id, const Value& cookie) { BOOST_THROW_EXCEPTION(std::runtime_error("Invalid field ID.")); } Object::Ptr Object::NavigateField(int id) const { BOOST_THROW_EXCEPTION(std::runtime_error("Invalid field ID.")); } Object::Ptr Object::Clone(void) const { BOOST_THROW_EXCEPTION(std::runtime_error("Object cannot be cloned.")); } Type::Ptr Object::GetReflectionType(void) const { return Object::TypeInstance; } #ifdef I2_LEAK_DEBUG void icinga::TypeAddObject(Object *object) { boost::mutex::scoped_lock lock(l_ObjectCountLock); String typeName = Utility::GetTypeName(typeid(*object)); l_ObjectCounts[typeName]++; } void icinga::TypeRemoveObject(Object *object) { boost::mutex::scoped_lock lock(l_ObjectCountLock); String typeName = Utility::GetTypeName(typeid(*object)); l_ObjectCounts[typeName]--; } static void TypeInfoTimerHandler(void) { boost::mutex::scoped_lock lock(l_ObjectCountLock); typedef std::map::value_type kv_pair; BOOST_FOREACH(kv_pair& kv, l_ObjectCounts) { if (kv.second == 0) continue; Log(LogInformation, "TypeInfo") << kv.second << " " << kv.first << " objects"; kv.second = 0; } } static void StartTypeInfoTimer(void) { l_ObjectCountTimer = new Timer(); l_ObjectCountTimer->SetInterval(10); l_ObjectCountTimer->OnTimerExpired.connect(boost::bind(TypeInfoTimerHandler)); l_ObjectCountTimer->Start(); } INITIALIZE_ONCE(StartTypeInfoTimer); #endif /* I2_LEAK_DEBUG */