Description
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
Type
Projects
Status
No status