/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef mozilla_mscom_AgileReference_h #define mozilla_mscom_AgileReference_h #include "mozilla/RefPtr.h" #include namespace mozilla { namespace mscom { /** * This class encapsulates an "agile reference." These are references that * allow you to pass COM interfaces between apartments. When you have an * interface that you would like to pass between apartments, you wrap that * interface in an AgileReference and pass the agile reference instead. Then * you unwrap the interface by calling AgileReference::Resolve. * * Sample usage: * * // In the multithreaded apartment, foo is an IFoo* * auto myAgileRef = MakeUnique(IID_IFoo, foo); * * // myAgileRef is passed to our main thread, which runs in a single-threaded * // apartment: * * RefPtr foo; * HRESULT hr = myAgileRef->Resolve(IID_IFoo, getter_AddRefs(foo)); * // Now foo may be called from the main thread */ class AgileReference { public: AgileReference(REFIID aIid, IUnknown* aObject); AgileReference(AgileReference&& aOther); ~AgileReference(); explicit operator bool() const { return mAgileRef || mGitCookie; } HRESULT Resolve(REFIID aIid, void** aOutInterface); AgileReference(const AgileReference& aOther) = delete; AgileReference& operator=(const AgileReference& aOther) = delete; AgileReference& operator=(AgileReference&& aOther) = delete; private: IGlobalInterfaceTable* ObtainGit(); private: IID mIid; RefPtr mAgileRef; DWORD mGitCookie; }; } // namespace mscom } // namespace mozilla #endif // mozilla_mscom_AgileReference_h