File: Windows\Win32\System\Com\StandardDispatch.cs
Web Access
Project: src\src\System.Windows.Forms.Primitives\src\System.Windows.Forms.Primitives.csproj (System.Windows.Forms.Primitives)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using Windows.Win32.System.Ole;
using Windows.Win32.System.Variant;
 
namespace Windows.Win32.System.Com;
 
/// <summary>
///  Base class for providing <see cref="IDispatch"/> services through a standard dispatch implementation
///  generated from a type library.
/// </summary>
/// <remarks>
///  <para>
///   This is roughly analogous to Win32's CreateStdDispatch which creates a simple wrapper that passes
///   through to <see cref="ITypeInfo"/> for basic <see cref="IDispatch"/> support.
///  </para>
/// </remarks>
internal abstract unsafe class StandardDispatch<T> : IDispatch.Interface, IDispatchEx.Interface, IDisposable
    where T : unmanaged, IComIID
{
    private ITypeInfo* _typeInfo;
    private readonly object _instance;
 
    /// <summary>
    ///  Construct a new instance with the specified backing <see cref="ITypeInfo"/>.
    /// </summary>
    /// <param name="instance">
    ///  Specifies the target that implements the <typeparamref name="T"/> interface. The normal behavior is to use
    ///  <see langword="this"/> as the instance to dispatch on (when <see langword="null"/> is passed).
    /// </param>
    public StandardDispatch(ITypeInfo* typeInfo, object? instance = default)
    {
        if (typeInfo is null)
        {
            throw new ArgumentNullException(nameof(typeInfo));
        }
 
        _instance = instance ?? this;
 
#if DEBUG
        typeInfo->GetTypeAttr(out TYPEATTR* typeAttributes).ThrowOnFailure();
        try
        {
            if (typeAttributes->guid != T.Guid)
            {
                throw new ArgumentException("Interface guid doesn't match type info", nameof(typeInfo));
            }
        }
        finally
        {
            typeInfo->ReleaseTypeAttr(typeAttributes);
        }
#endif
 
        _typeInfo = typeInfo;
        _typeInfo->AddRef();
    }
 
    HRESULT IDispatch.Interface.GetTypeInfoCount(uint* pctinfo)
    {
        if (pctinfo is null)
        {
            return HRESULT.E_POINTER;
        }
 
        *pctinfo = 1;
        return HRESULT.S_OK;
    }
 
    HRESULT IDispatch.Interface.GetTypeInfo(uint iTInfo, uint lcid, ITypeInfo** ppTInfo)
    {
        if (ppTInfo is null)
        {
            return HRESULT.E_POINTER;
        }
 
        if (iTInfo != 0)
        {
            *ppTInfo = null;
            return HRESULT.DISP_E_BADINDEX;
        }
 
        _typeInfo->AddRef();
        *ppTInfo = _typeInfo;
        return HRESULT.S_OK;
    }
 
    HRESULT IDispatch.Interface.GetIDsOfNames(Guid* riid, PWSTR* rgszNames, uint cNames, uint lcid, int* rgDispId)
    {
        // This must be IID_NULL
        if (riid != IID.NULL())
        {
            return HRESULT.DISP_E_UNKNOWNINTERFACE;
        }
 
        return _typeInfo->GetIDsOfNames(rgszNames, cNames, rgDispId);
    }
 
    HRESULT IDispatch.Interface.Invoke(
        int dispIdMember,
        Guid* riid,
        uint lcid,
        DISPATCH_FLAGS wFlags,
        DISPPARAMS* pDispParams,
        VARIANT* pVarResult,
        EXCEPINFO* pExcepInfo,
        uint* pArgErr)
    {
        if (pDispParams is null)
        {
            return HRESULT.E_INVALIDARG;
        }
 
        HRESULT hr = MapDotNetHRESULTs(Invoke(
            dispIdMember,
            lcid,
            wFlags,
            pDispParams,
            pVarResult,
            pExcepInfo,
            pArgErr));
 
        if (hr != HRESULT.DISP_E_MEMBERNOTFOUND)
        {
            return hr;
        }
 
        // The override couldn't find it, pass it along via the ITypeInfo.
        using ComScope<T> @interface = new(ComHelpers.GetComPointer<T>(_instance));
        return _typeInfo->Invoke(@interface, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, pArgErr);
    }
 
    HRESULT IDispatchEx.Interface.GetDispID(BSTR bstrName, uint grfdex, int* pid)
        => bstrName.IsNull || pid is null ? HRESULT.E_POINTER : GetDispID(bstrName, grfdex, pid);
 
    protected virtual HRESULT GetDispID(BSTR bstrName, uint grfdex, int* pid) => HRESULT.E_NOTIMPL;
 
    HRESULT IDispatchEx.Interface.GetMemberName(int id, BSTR* pbstrName)
        => pbstrName is null ? HRESULT.E_POINTER : GetMemberName(id, pbstrName);
 
    protected virtual HRESULT GetMemberName(int id, BSTR* pbstrName) => HRESULT.E_NOTIMPL;
 
    HRESULT IDispatchEx.Interface.GetNextDispID(uint grfdex, int id, int* pid)
    {
        if (pid is null)
        {
            return HRESULT.E_POINTER;
        }
 
        *pid = PInvokeCore.DISPID_UNKNOWN;
 
        return GetNextDispID(grfdex, id, pid);
    }
 
    protected virtual HRESULT GetNextDispID(uint grfdex, int id, int* pid) => HRESULT.E_NOTIMPL;
 
    HRESULT IDispatchEx.Interface.InvokeEx(
        int id,
        uint lcid,
        ushort wFlags,
        DISPPARAMS* pdp,
        VARIANT* pvarRes,
        EXCEPINFO* pei,
        IServiceProvider* pspCaller)
    {
        if (pdp is null)
        {
            return HRESULT.E_INVALIDARG;
        }
 
        HRESULT hr = MapDotNetHRESULTs(Invoke(
            id,
            lcid,
            (DISPATCH_FLAGS)wFlags,
            pdp,
            pvarRes,
            pei,
            argumentError: null));
 
        if (hr != HRESULT.DISP_E_MEMBERNOTFOUND)
        {
            return hr;
        }
 
        // The override couldn't find it, pass it along via the ITypeInfo.
        using ComScope<T> @interface = new(ComHelpers.GetComPointer<T>(_instance));
        return _typeInfo->Invoke(@interface, id, (DISPATCH_FLAGS)wFlags, pdp, pvarRes, pei, puArgErr: null);
    }
 
    protected virtual HRESULT Invoke(
        int dispId,
        uint lcid,
        DISPATCH_FLAGS flags,
        DISPPARAMS* parameters,
        VARIANT* result,
        EXCEPINFO* exceptionInfo,
        uint* argumentError)
        => HRESULT.DISP_E_MEMBERNOTFOUND;
 
    HRESULT IDispatchEx.Interface.GetMemberProperties(int id, uint grfdexFetch, FDEX_PROP_FLAGS* pgrfdex)
    {
        if (pgrfdex is null)
        {
            return HRESULT.E_POINTER;
        }
 
        if (id == PInvokeCore.DISPID_UNKNOWN)
        {
            return HRESULT.E_INVALIDARG;
        }
 
        *pgrfdex = default;
 
        HRESULT hr = GetMemberProperties(id, out FDEX_PROP_FLAGS properties);
        if (hr.Succeeded)
        {
            // Filter to the requested properties
            *pgrfdex = properties & (FDEX_PROP_FLAGS)grfdexFetch;
        }
        else
        {
            *pgrfdex = default;
        }
 
        return hr;
    }
 
    protected virtual HRESULT GetMemberProperties(int dispId, out FDEX_PROP_FLAGS properties)
    {
        properties = default;
        return HRESULT.E_NOTIMPL;
    }
 
    // .NET COM Interop returns E_NOTIMPL for these three.
 
    HRESULT IDispatchEx.Interface.DeleteMemberByName(BSTR bstrName, uint grfdex) => HRESULT.E_NOTIMPL;
    HRESULT IDispatchEx.Interface.DeleteMemberByDispID(int id) => HRESULT.E_NOTIMPL;
 
    HRESULT IDispatchEx.Interface.GetNameSpaceParent(IUnknown** ppunk)
    {
        if (ppunk is null)
        {
            return HRESULT.E_POINTER;
        }
 
        *ppunk = null;
        return HRESULT.E_NOTIMPL;
    }
 
    protected virtual void Dispose(bool disposing)
    {
        if (_typeInfo is not null)
        {
            _typeInfo->Release();
            _typeInfo = null;
        }
    }
 
    private static HRESULT MapDotNetHRESULTs(HRESULT hr)
    {
        // Following along with .NET COM interop
 
        if (hr == HRESULT.COR_E_OVERFLOW)
        {
            return HRESULT.DISP_E_OVERFLOW;
        }
        else if (hr == HRESULT.COR_E_INVALIDOLEVARIANTTYPE)
        {
            return HRESULT.DISP_E_BADVARTYPE;
        }
        else if (hr == HRESULT.COR_E_ARGUMENT)
        {
            return HRESULT.E_INVALIDARG;
        }
        else if (hr == HRESULT.COR_E_SAFEARRAYTYPEMISMATCH)
        {
            return HRESULT.DISP_E_TYPEMISMATCH;
        }
        else if (hr == HRESULT.COR_E_MISSINGMEMBER || hr == HRESULT.COR_E_MISSINGMETHOD)
        {
            return HRESULT.DISP_E_MEMBERNOTFOUND;
        }
 
        // .NET maps this, we would need to populate EXCEPINFO to do the same
        //
        // else if (hr == HRESULT.COR_E_TARGETINVOCATION)
        // {
        //     return HRESULT.DISP_E_EXCEPTION;
        // }
 
        return hr;
    }
 
    public void Dispose()
    {
        Dispose(disposing: true);
        GC.SuppressFinalize(this);
    }
}