⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 acyclic_connector.hpp

📁 新版本TR1的stl
💻 HPP
字号:
/* /////////////////////////////////////////////////////////////////////////
 * File:        comstl/util/acyclic_connector.hpp
 *
 * Purpose:     A component for relating two COM objects without cycles.
 *
 * Created:     25th March 2006
 * Updated:     22nd March 2007
 *
 * Home:        http://stlsoft.org/
 *
 * Copyright (c) 2006-2007, Matthew Wilson and Synesis Software
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * - Redistributions of source code must retain the above copyright notice, this
 *   list of conditions and the following disclaimer.
 * - Redistributions in binary form must reproduce the above copyright notice,
 *   this list of conditions and the following disclaimer in the documentation
 *   and/or other materials provided with the distribution.
 * - Neither the name(s) of Matthew Wilson and Synesis Software nor the names of
 *   any contributors may be used to endorse or promote products derived from
 *   this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 *
 * ////////////////////////////////////////////////////////////////////// */


/** \file comstl/util/acyclic_connector.hpp
 *
 * \brief [C++ only; requires COM] Definition of the
 *   comstl::acyclic_connector class template
 *   (\ref group__library__utility__com "COM Utility" Library).
 */

#ifndef COMSTL_INCL_COMSTL_UTIL_HPP_ACYCLIC_CONNECTOR
#define COMSTL_INCL_COMSTL_UTIL_HPP_ACYCLIC_CONNECTOR

#ifndef STLSOFT_DOCUMENTATION_SKIP_SECTION
# define COMSTL_VER_COMSTL_UTIL_HPP_ACYCLIC_CONNECTOR_MAJOR     1
# define COMSTL_VER_COMSTL_UTIL_HPP_ACYCLIC_CONNECTOR_MINOR     2
# define COMSTL_VER_COMSTL_UTIL_HPP_ACYCLIC_CONNECTOR_REVISION  5
# define COMSTL_VER_COMSTL_UTIL_HPP_ACYCLIC_CONNECTOR_EDIT      17
#endif /* !STLSOFT_DOCUMENTATION_SKIP_SECTION */

/* /////////////////////////////////////////////////////////////////////////
 * Includes
 */

#ifndef COMSTL_INCL_COMSTL_H_COMSTL
# include <comstl/comstl.h>
#endif /* !COMSTL_INCL_COMSTL_H_COMSTL */
#ifndef COMSTL_INCL_COMSTL_UTIL_HPP_OBJECT_FUNCTIONS
# include <comstl/util/object_functions.h>
#endif /* !COMSTL_INCL_COMSTL_UTIL_HPP_OBJECT_FUNCTIONS */
#ifndef STLSOFT_INCL_STLSOFT_SYNCH_HPP_LOCK_SCOPE
# include <stlsoft/synch/lock_scope.hpp>
#endif /* !STLSOFT_INCL_STLSOFT_SYNCH_HPP_LOCK_SCOPE */

/* /////////////////////////////////////////////////////////////////////////
 * Namespace
 */

#ifndef _COMSTL_NO_NAMESPACE
# if defined(_STLSOFT_NO_NAMESPACE) || \
     defined(STLSOFT_DOCUMENTATION_SKIP_SECTION)
/* There is no stlsoft namespace, so must define ::comstl */
namespace comstl
{
# else
/* Define stlsoft::comstl_project */

namespace stlsoft
{

namespace comstl_project
{

# endif /* _STLSOFT_NO_NAMESPACE */
#endif /* !_COMSTL_NO_NAMESPACE */

/* /////////////////////////////////////////////////////////////////////////
 * Classes
 */

/** \brief Interface for acyclic communication.
 */
#ifdef STLSOFT_DOCUMENTATION_SKIP_SECTION
struct IAcyclicSide
    : public IUnknown
#else /* ? STLSOFT_DOCUMENTATION_SKIP_SECTION */
DECLARE_INTERFACE_(IAcyclicSide, IUnknown)
#endif /* STLSOFT_DOCUMENTATION_SKIP_SECTION */
{
    /** \brief Causes knowledge of the presence of the peer to be discarded
     */
    STDMETHOD_(void, Clear)() PURE;
    /** \brief Queries for the given interface on the peer.
     *
     * \param riid The interface identifier of the requested interface.
     * \param ppv Address of the interface pointer.
     *
     * \return A standard HRESULT status code indicating success/failure.
     */
    STDMETHOD(QueryPeer)(THIS_ REFIID riid, void **ppv) PURE;

    /** \brief returns the IID for the IAcyclicSide interface. */
    static REFIID iid()
    {
        static IID s_iid = { 0x8D5D0B0A, 0x4429, 0x4be1, { 0x8C, 0x00, 0xDE, 0xE0, 0xA8, 0xFF, 0xD0, 0xAF } };

        return s_iid;
    }
};

/** \brief A component that allows two objects to be connected without
 *   creating cyclic dependencies.
 *
 * Reference counting architectures, such as COM, rely on there being no
 * dependency cycles. A cyclic dependency is the situation where, say, 
 * object <b>a</b> holds a reference to object <b>b</b>, and object
 * <b>b</b> holds a reference to object <b>a</b>. Since neither will
 * release the references it holds (held on its efferent dependents)
 * until all references to itself (held by its afferent dependents) the
 * circle will never be broken.
 *
 * COM lore thus proscribes the use of mutual references, or prescribes
 * rigid protocols (e.g. Connection Points) for their management.
 *
 * acyclic_connector is designed to help in the case where two objects
 * need to have knowledge of each other, but not hold references on each
 * other. It works by acting as an intermediary that each of two objects
 * in the relationship keep informed as to their existance. 
 *
 * \param MX The type of the mutex that will be used to provide exclusive
 *  access to the instance state during the Clear() and QueryPeer() methods.
 */
template <ss_typename_param_k MX>
class acyclic_connector
{
/// \name Member Types
/// @{
public:
    typedef MX                      mutex_type;
private:
    typedef acyclic_connector<MX>   connector_type;
public:
    typedef acyclic_connector<MX>   class_type;
private:
    class side
        : public IAcyclicSide
    {
    /// \name Member Types
    /// @{
    public:
        typedef side            class_type;
    /// @}

    /// \name Construction
    /// @{
    public:
        ss_explicit_k side(connector_type &connector, LPUNKNOWN peer, IAcyclicSide **side)
            : m_connector(connector)
            , m_peer(peer)
            , m_refCount(1)
        {
            COMSTL_ASSERT(NULL != side);
            *side = this;
        }
    /// @}

    /// \name IUnknown
    /// @{
    public:
        STDMETHOD_(ULONG, AddRef)()
        {
            return ++m_refCount;
        }
        STDMETHOD_(ULONG, Release)()
        {
            class_type  &other  =   (this == &m_connector.m_left) ? m_connector.m_right : m_connector.m_left;

            if(0 == --m_refCount)
            {
                m_peer = NULL;

                if(0 == other.m_refCount)
                {
                    delete &m_connector;

                    return 0;
                }
            }

            return m_refCount;
        }
        STDMETHOD(QueryInterface)(REFIID riid, void **ppv)
        {
            if( IID_IUnknown == riid ||
                IAcyclicSide::iid() == riid)
            {
                *ppv = static_cast<LPUNKNOWN>(this);
                static_cast<LPUNKNOWN>(*ppv)->AddRef();
                return S_OK;
            }

            return E_NOINTERFACE;
        }
    /// @}

    /// \name IAcyclicSide
    /// @{
    public:
        STDMETHOD_(void, Clear)()
        {
            stlsoft::lock_scope<mutex_type> lock(m_connector.m_mx);

            m_peer = NULL;
        }
        STDMETHOD(QueryPeer)(THIS_ REFIID riid, void **ppv)
        {
            COMSTL_ASSERT(NULL != ppv);

            stlsoft::lock_scope<mutex_type> lock(m_connector.m_mx);
            class_type                      &other  =   (this == &m_connector.m_left) ? m_connector.m_right : m_connector.m_left;

            if(NULL == other.m_peer)
            {
                return E_POINTER;
            }
            else
            {
                return other.m_peer->QueryInterface(riid, ppv);
            }
        }
    /// @}

    /// \name Members
    /// @{
    private:
        connector_type  &m_connector;
        LPUNKNOWN       m_peer;
        LONG            m_refCount;
    /// @}
    };
    friend class side;
/// @}

/// \name Construction
/// @{
public:
    acyclic_connector(  LPUNKNOWN       leftPeer
                    ,   IAcyclicSide    **leftSide
                    ,   LPUNKNOWN       rightPeer
                    ,   IAcyclicSide    **rightSide);
private:
    ~acyclic_connector() stlsoft_throw_0();
/// @}

/// \name Implementation
/// @{
private:
    class_type &get_this_();
/// @}

/// \name Construction
/// @{
private:
    side        m_left;
    side        m_right;
    mutex_type  m_mx;
/// @}

/// \name Not to be implemented
/// @{
private:
    acyclic_connector(class_type const& rhs);
    class_type& operator =(class_type const& rhs);
/// @}
};

/* /////////////////////////////////////////////////////////////////////////
 * Implementation
 */

#ifndef STLSOFT_DOCUMENTATION_SKIP_SECTION

// acyclic_connector::side

#if 0
template <ss_typename_param_k MX>
inline acyclic_connector<MX>::side::side(connector_type &connector, LPUNKNOWN peer, IAcyclicSide **side)
    : m_connector(connector)
    , m_peer(peer)
    , m_refCount(1)
{
    COMSTL_ASSERT(NULL != side);
    *side = this;
}

template <ss_typename_param_k MX>
inline STDMETHODIMP_(ULONG) acyclic_connector<MX>::side::AddRef()
{
    return ++m_refCount;
}

template <ss_typename_param_k MX>
inline STDMETHODIMP_(ULONG) acyclic_connector<MX>::side::Release()
{
    class_type  &other  =   (this == &m_connector.m_left) ? m_connector.m_right : m_connector.m_left;

    if( 0 == --m_refCount &&
        0 == other.m_refCount)
    {
        delete &m_connector;

        return 0;
    }

    return m_refCount;
}

template <ss_typename_param_k MX>
inline STDMETHODIMP acyclic_connector<MX>::side::QueryInterface(REFIID riid, void **ppv)
{
    if( IID_IUnknown == riid ||
        IAcyclicSide::iid() == riid)
    {
        *ppv = static_cast<LPUNKNOWN>(this);
        static_cast<LPUNKNOWN>(*ppv)->AddRef();
        return S_OK;
    }

    return E_INTERFACE;
}
#endif /* 0 */

#if 0
template <ss_typename_param_k MX>
inline STDMETHODIMP_(void) acyclic_connector<MX>::side::Clear()
{
    stlsoft::lock_scope<mutex_type> lock(m_mx);

    m_peer = NULL;
}

template <ss_typename_param_k MX>
inline STDMETHODIMP acyclic_connector<MX>::side::QueryPeer(THIS_ REFIID riid, void **ppv)
{
    COMSTL_ASSERT(NULL != ppv);

    stlsoft::lock_scope<mutex_type> lock(m_mx);

    if(NULL == m_peer)
    {
        return E_POINTER;
    }
    else
    {
        return m_peer->QueryInterface(riid, ppv);
    }
}
#endif /* 0 */

// acyclic_connector

template <ss_typename_param_k MX>
inline acyclic_connector<MX>::acyclic_connector(LPUNKNOWN       leftPeer
                                            ,   IAcyclicSide    **leftSide
                                            ,   LPUNKNOWN       rightPeer
                                            ,   IAcyclicSide    **rightSide)
    : m_left(get_this_(), leftPeer, leftSide)
    , m_right(get_this_(), rightPeer, rightSide)
    , m_mx()
{
    COMSTL_MESSAGE_ASSERT("Acyclic connector cannot be initialised with null pointers", NULL != leftPeer);
    COMSTL_MESSAGE_ASSERT("Acyclic connector cannot be initialised with null pointers", NULL != rightPeer);
    COMSTL_MESSAGE_ASSERT("Acyclic connector cannot be initialised with null pointers", NULL != leftSide);
    COMSTL_MESSAGE_ASSERT("Acyclic connector cannot be initialised with null pointers", NULL != rightSide);
#if 0

    HRESULT hr;

    hr = get_object_identity(left, &m_left);
    if(FAILED(hr))
    {
#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
        STLSOFT_THROW_X(com_exception("Could not acquire left-side identity", hr));
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
    }
    else
    {
        hr = get_object_identity(right, &m_right);

        if(FAILED(hr))
        {
            m_left->Release();

#ifdef STLSOFT_CF_EXCEPTION_SUPPORT
            STLSOFT_THROW_X(com_exception("Could not acquire right-side identity", hr));
#else /* ? STLSOFT_CF_EXCEPTION_SUPPORT */
            m_left = NULL;
#endif /* STLSOFT_CF_EXCEPTION_SUPPORT */
        }
    }
#endif /* 0 */
}

template <ss_typename_param_k MX>
inline acyclic_connector<MX>::~acyclic_connector() stlsoft_throw_0()
{}

template <ss_typename_param_k MX>
inline ss_typename_type_k acyclic_connector<MX>::class_type &acyclic_connector<MX>::get_this_()
{
    return *this;
}

#endif /* !STLSOFT_DOCUMENTATION_SKIP_SECTION */

/* ////////////////////////////////////////////////////////////////////// */

#ifndef _COMSTL_NO_NAMESPACE
# if defined(_STLSOFT_NO_NAMESPACE) || \
     defined(STLSOFT_DOCUMENTATION_SKIP_SECTION)
} // namespace comstl
# else
} // namespace comstl_project
} // namespace stlsoft
# endif /* _STLSOFT_NO_NAMESPACE */
#endif /* !_COMSTL_NO_NAMESPACE */

/* ////////////////////////////////////////////////////////////////////// */

#endif /* !COMSTL_INCL_COMSTL_UTIL_HPP_ACYCLIC_CONNECTOR */

/* ////////////////////////////////////////////////////////////////////// */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -