/* $Id: listeners.h 101967 2023-11-08 12:44:46Z vboxsync $ */ /** @file * MS COM / XPCOM Abstraction Layer - Listener helpers. */ /* * Copyright (C) 2010-2023 Oracle and/or its affiliates. * * This file is part of VirtualBox base platform packages, as * available from https://www.virtualbox.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, in version 3 of the * License. * * 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, see . * * The contents of this file may alternatively be used under the terms * of the Common Development and Distribution License Version 1.0 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included * in the VirtualBox distribution, in which case the provisions of the * CDDL are applicable instead of those of the GPL. * * You may elect to license modified versions of this file under the * terms and conditions of either the GPL or the CDDL or both. * * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0 */ #ifndef VBOX_INCLUDED_com_listeners_h #define VBOX_INCLUDED_com_listeners_h #ifndef RT_WITHOUT_PRAGMA_ONCE # pragma once #endif #include #include /** @defgroup grp_com_listeners Listener Helpers * @ingroup grp_com * @{ */ #ifdef VBOX_WITH_XPCOM # define NS_IMPL_QUERY_HEAD_INLINE() \ NS_IMETHODIMP QueryInterface(REFNSIID aIID, void **aInstancePtr) \ { \ NS_ASSERTION(aInstancePtr, "QueryInterface requires a non-NULL destination!"); \ nsISupports *foundInterface; # define NS_INTERFACE_MAP_BEGIN_INLINE() NS_IMPL_QUERY_HEAD_INLINE() # define NS_IMPL_QUERY_INTERFACE1_INLINE(a_i1) \ NS_INTERFACE_MAP_BEGIN_INLINE() \ NS_INTERFACE_MAP_ENTRY(a_i1) \ NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, a_i1) \ NS_INTERFACE_MAP_END #endif template class ListenerImpl : public ATL::CComObjectRootEx, VBOX_SCRIPTABLE_IMPL(IEventListener) { T* mListener; #ifdef RT_OS_WINDOWS /* FTM stuff */ ComPtr m_pUnkMarshaler; #else nsAutoRefCnt mRefCnt; NS_DECL_OWNINGTHREAD #endif public: ListenerImpl() { } virtual ~ListenerImpl() { } HRESULT init(T* aListener, TParam param) { mListener = aListener; return mListener->init(param); } HRESULT init(T* aListener) { mListener = aListener; return mListener->init(); } void uninit() { if (mListener) { mListener->uninit(); delete mListener; mListener = 0; } } HRESULT FinalConstruct() { #ifdef RT_OS_WINDOWS return CoCreateFreeThreadedMarshaler(this, &m_pUnkMarshaler.m_p); #else return S_OK; #endif } void FinalRelease() { uninit(); #ifdef RT_OS_WINDOWS m_pUnkMarshaler.setNull(); #endif } T* getWrapped() { return mListener; } DECLARE_NOT_AGGREGATABLE(ListenerImpl) DECLARE_PROTECT_FINAL_CONSTRUCT() #ifdef RT_OS_WINDOWS BEGIN_COM_MAP(ListenerImpl) COM_INTERFACE_ENTRY(IEventListener) COM_INTERFACE_ENTRY2(IDispatch, IEventListener) COM_INTERFACE_ENTRY_AGGREGATE(IID_IMarshal, m_pUnkMarshaler.m_p) END_COM_MAP() #else NS_IMETHOD_(nsrefcnt) AddRef(void) { NS_PRECONDITION(PRInt32(mRefCnt) >= 0, "illegal refcnt"); nsrefcnt count; count = ASMAtomicIncU32((volatile uint32_t *)&mRefCnt); NS_LOG_ADDREF(this, count, "ListenerImpl", sizeof(*this)); return count; } NS_IMETHOD_(nsrefcnt) Release(void) { nsrefcnt count; NS_PRECONDITION(0 != mRefCnt, "dup release"); count = ASMAtomicDecU32((volatile uint32_t *)&mRefCnt); NS_LOG_RELEASE(this, count, "ListenerImpl"); if (0 == count) { mRefCnt = 1; /* stabilize */ /* enable this to find non-threadsafe destructors: */ /* NS_ASSERT_OWNINGTHREAD(_class); */ NS_DELETEXPCOM(this); return 0; } return count; } NS_IMPL_QUERY_INTERFACE1_INLINE(IEventListener) #endif STDMETHOD(HandleEvent)(IEvent * aEvent) { VBoxEventType_T aType = VBoxEventType_Invalid; HRESULT hrc = aEvent->COMGETTER(Type)(&aType); AssertMsg(SUCCEEDED(hrc), ("hrc=%Rhrc\n", hrc)); RT_NOREF(hrc); return mListener->HandleEvent(aType, aEvent); } }; #ifdef VBOX_WITH_XPCOM # define VBOX_LISTENER_DECLARE(klazz) NS_DECL_CLASSINFO(klazz) #else # define VBOX_LISTENER_DECLARE(klazz) #endif /** @} */ #endif /* !VBOX_INCLUDED_com_listeners_h */