| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144 | #pragma once#include "ReferenceCounter.h"#include "os/Win32/WindowsHeaders.h"#include <windows.foundation.h>#include <windows.foundation.collections.h>#include <wrl.h>#include <vector>namespace il2cpp{namespace winrt{    template<typename T>    struct ResultTypeTraits    {        typedef typename ABI::Windows::Foundation::Internal::GetLogicalType<typename T::TResult_complex>::type LogicalType;        typedef typename ABI::Windows::Foundation::Internal::GetAbiType<typename T::TResult_complex>::type ResultType;    };    template<>    struct ResultTypeTraits<ABI::Windows::Foundation::IAsyncAction>    {        typedef void LogicalType;        typedef void ResultType;    };    template<typename OperationType, typename HandlerType>    class SynchronousOperationBase : public Microsoft::WRL::RuntimeClass<Microsoft::WRL::RuntimeClassFlags<Microsoft::WRL::WinRtClassicComMix>, Microsoft::WRL::FtmBase, HandlerType>    {    protected:        HANDLE m_Event;        HRESULT m_HR;        inline SynchronousOperationBase(OperationType* op)        {            m_Event = CreateEventExW(nullptr, nullptr, CREATE_EVENT_MANUAL_RESET, EVENT_ALL_ACCESS);            Assert(m_Event && "CreateEventExW failed.");        }        inline ~SynchronousOperationBase()        {            CloseHandle(m_Event);            // This will be not 0 if it's allocated on the stack            // This class must be allocated on the heap for correct COM ref counting!            IL2CPP_ASSERT(GetRefCount() == 0);        }    public:        inline HRESULT Wait()        {            auto waitResult = WaitForSingleObjectEx(m_Event, INFINITE, FALSE);            if (waitResult != WAIT_OBJECT_0)                return E_FAIL;            return m_HR;        }    };    template<typename T>    class SynchronousOperation : public SynchronousOperationBase<ABI::Windows::Foundation::IAsyncOperation<T>, ABI::Windows::Foundation::IAsyncOperationCompletedHandler<T> >    {    private:        using OperationType = ABI::Windows::Foundation::IAsyncOperation<T>;        typedef typename ResultTypeTraits<OperationType>::ResultType ResultType;        ResultType m_Result;    public:        inline SynchronousOperation(OperationType* op) :            SynchronousOperationBase(op),            m_Result(ResultType())        {            // NOTE: this is in the derived class because it might immediately complete.            // If this was called from the base class, you'd get a callback on a partially            // constructed vtable and crash.            auto hr = op->put_Completed(this);            Assert(SUCCEEDED(hr) && "IAsyncOperation<T>::put_Completed failed.");        }        ~SynchronousOperation()        {            ReferenceCounter<ResultType>::Release(m_Result);        }        HRESULT GetResults(ResultType* result)        {            auto hr = Wait();            if (FAILED(hr))                return hr;            *result = m_Result;            ReferenceCounter<ResultType>::AddRef(*result);            return S_OK;        }        virtual HRESULT STDMETHODCALLTYPE Invoke(ABI::Windows::Foundation::IAsyncOperation<T>* asyncInfo, ABI::Windows::Foundation::AsyncStatus status) override        {            m_HR = asyncInfo->GetResults(&m_Result);            SetEvent(m_Event);            return S_OK;        }        static Microsoft::WRL::ComPtr<SynchronousOperation<T> > Make(OperationType* op)        {            return Microsoft::WRL::Make<SynchronousOperation<T> >(op);        }    };    template<>    class SynchronousOperation<void> : public SynchronousOperationBase<ABI::Windows::Foundation::IAsyncAction, ABI::Windows::Foundation::IAsyncActionCompletedHandler>    {    private:        using OperationType = ABI::Windows::Foundation::IAsyncAction;    public:        inline SynchronousOperation(OperationType* op) :            SynchronousOperationBase(op)        {            // NOTE: this is in the derived class because it might immediately complete.            // If this was called from the base class, you'd get a callback on a partially            // constructed vtable and crash.            auto hr = op->put_Completed(this);            Assert(SUCCEEDED(hr) && "IAsyncAction::put_Completed failed.");        }        virtual HRESULT STDMETHODCALLTYPE Invoke(ABI::Windows::Foundation::IAsyncAction* asyncInfo, ABI::Windows::Foundation::AsyncStatus status) override        {            m_HR = asyncInfo->GetResults();            SetEvent(m_Event);            return S_OK;        }    };    template<typename T>    Microsoft::WRL::ComPtr<SynchronousOperation<typename ResultTypeTraits<T>::LogicalType> > MakeSynchronousOperation(T* op)    {        return Microsoft::WRL::Make<SynchronousOperation<typename ResultTypeTraits<T>::LogicalType> >(op);    }}}
 |