Skip to content

[API Proposal]: Allow the user to provide state for a ComWrappers.GetOrCreateObjectForComInstance operation #113622

Open
@jkoritzinsky

Description

@jkoritzinsky

Background and motivation

As part of the Interop Type Mapping Proposal, one important aspect we've discussed for foreign language projections to improve their performance (and avoid relying on the type map in the majority of cases), is to use call-site information about types to avoid needing to go through the map for strongly-typed APIs.
 
Currently, there is no mechanism to pass down additional state to ComWrappers.CreateObject, so any foreign language projection based on COM (ie. CsWinRT) must use something else such as [ThreadStatic] statics on their ComWrappers implementation to pass down call-site state as a side channel.

Additionally, CsWinRT has had to add additional tracking and lifetime extensions to support some of their primitives that should be immediately unwrapped, ie. IReference<T> implementations, as mentioned in #113591.

In #113591, an API change is proposed to allow CsWinRT to use another side-channel to return an object without going through ComWrappers. In the comments, a protected property is proposed to provide operation-scoped state.

This API proposal aims to solve both of the above problems. It allows the user to provide their own "state" object that will be available in CreateObjectWithState and allows the user to specify flags based on the created wrapper object with the wrapperFlags out parameter.

API Proposal

namespace System.Runtime.InteropServices;

+[Flags]
+public enum CreatedWrapperFlags
+{
+    None = 0,
+    // Same as CreateObjectFlags.TrackerObject, but decided based on inspecting the COM object directly while creating the wrapper.
+    TrackerObject = 0x1,
+    /// <summary>
+    /// The managed object doesn't keep the native object alive. It represents an equivalent value.
+    /// </summary>
+    /// <remarks>
+    /// Using this flag results in the following changes:
+    /// <see cref="ComWrappers.TryGetComInstance" /> will return <c>false</c> for the returned object.
+    /// The features provided by the <see cref="CreateObjectFlags.TrackerObject" /> flag will be disabled.
+    /// Integration between <see cref="System.WeakReference" /> and the returned object via the native <c>IWeakReferenceSource</c> interface will not work.
+    /// <see cref="CreateObjectFlags.UniqueInstance" /> behavior is implied.
+    /// Diagnostics tooling support to unwrap objects returned by `CreateObject` will not see this object as a wrapper.
+    /// The same object can be returned from `CreateObject` wrapping different COM objects.
+    /// </remarks>
+    NonWrapping = 0x2
+}

public abstract class ComWrappers
{
   protected abstract object? CreateObject(IntPtr externalComObject, CreateObjectFlags flags);
+    protected virtual object? CreateObject(IntPtr externalComObject, CreateObjectFlags flags, object? userState, out CreatedWrapperFlags wrapperFlags)
+    {
+        wrapperFlags = CreatedWrapperFlags.None;
+        return CreateObject(externalComObject, flags);
+    }

   public object GetOrCreateObjectForComInstance(IntPtr externalComObject, CreateObjectFlags flags);
+    public object GetOrCreateObjectForComInstance(IntPtr externalComObject, CreateObjectFlags flags, object? userState);
}

API Usage

class WinRTComWrappers  : ComWrappers
{
    protected override object? CreateObject(IntPtr externalComObject, ref CreateObjectFlags flags, object? userState, out CreatedWrapperFlags wrapperFlags)
    {
        if (CheckIfShouldCreateAnRcw(...))
        {
            wrapperFlags = CreatedWrapperFlags.None;
            return rcw;
        }

        // Otherwise, unbox directly
        wrapperFlags = CreatedWrapperFlags.NonWrapping;
        return UnboxTheValue(...);
    }
}

Alternative Designs

The designs in #113581 and #113591 are alternative options we've considered.

Risks

These API changes would make it less obvious which members on ComWrappers to implement in subclasses, as there are now two overloads of CreateObject.

Metadata

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    • Status

      No status

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions