a575963da9
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
644 lines
27 KiB
C#
644 lines
27 KiB
C#
/* ****************************************************************************
|
|
*
|
|
* Copyright (c) Microsoft Corporation.
|
|
*
|
|
* This source code is subject to terms and conditions of the Microsoft Public License. A
|
|
* copy of the license can be found in the License.html file at the root of this distribution. If
|
|
* you cannot locate the Microsoft Public License, please send an email to
|
|
* dlr@microsoft.com. By using this source code in any fashion, you are agreeing to be bound
|
|
* by the terms of the Microsoft Public License.
|
|
*
|
|
* You must not remove this notice, or any other, from this software.
|
|
*
|
|
*
|
|
* ***************************************************************************/
|
|
using System; using Microsoft;
|
|
|
|
|
|
#if !SILVERLIGHT // ComObject
|
|
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.Globalization;
|
|
#if CODEPLEX_40
|
|
using System.Linq.Expressions;
|
|
#else
|
|
using Microsoft.Linq.Expressions;
|
|
#endif
|
|
using System.Reflection;
|
|
using System.Runtime.InteropServices;
|
|
using System.Security;
|
|
using System.Security.Permissions;
|
|
using ComTypes = System.Runtime.InteropServices.ComTypes;
|
|
|
|
#if CODEPLEX_40
|
|
namespace System.Dynamic {
|
|
#else
|
|
namespace Microsoft.Scripting {
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// An object that implements IDispatch
|
|
///
|
|
/// This currently has the following issues:
|
|
/// 1. If we prefer ComObjectWithTypeInfo over IDispatchComObject, then we will often not
|
|
/// IDispatchComObject since implementations of IDispatch often rely on a registered type library.
|
|
/// If we prefer IDispatchComObject over ComObjectWithTypeInfo, users get a non-ideal experience.
|
|
/// 2. IDispatch cannot distinguish between properties and methods with 0 arguments (and non-0
|
|
/// default arguments?). So obj.foo() is ambiguous as it could mean invoking method foo,
|
|
/// or it could mean invoking the function pointer returned by property foo.
|
|
/// We are attempting to find whether we need to call a method or a property by examining
|
|
/// the ITypeInfo associated with the IDispatch. ITypeInfo tell's use what parameters the method
|
|
/// expects, is it a method or a property, what is the default property of the object, how to
|
|
/// create an enumerator for collections etc.
|
|
/// 3. IronPython processes the signature and converts ref arguments into return values.
|
|
/// However, since the signature of a DispMethod is not available beforehand, this conversion
|
|
/// is not possible. There could be other signature conversions that may be affected. How does
|
|
/// VB6 deal with ref arguments and IDispatch?
|
|
///
|
|
/// We also support events for IDispatch objects:
|
|
/// Background:
|
|
/// COM objects support events through a mechanism known as Connect Points.
|
|
/// Connection Points are separate objects created off the actual COM
|
|
/// object (this is to prevent circular references between event sink
|
|
/// and event source). When clients want to sink events generated by
|
|
/// COM object they would implement callback interfaces (aka source
|
|
/// interfaces) and hand it over (advise) to the Connection Point.
|
|
///
|
|
/// Implementation details:
|
|
/// When IDispatchComObject.TryGetMember request is received we first check
|
|
/// whether the requested member is a property or a method. If this check
|
|
/// fails we will try to determine whether an event is requested. To do
|
|
/// so we will do the following set of steps:
|
|
/// 1. Verify the COM object implements IConnectionPointContainer
|
|
/// 2. Attempt to find COM object's coclass's description
|
|
/// a. Query the object for IProvideClassInfo interface. Go to 3, if found
|
|
/// b. From object's IDispatch retrieve primary interface description
|
|
/// c. Scan coclasses declared in object's type library.
|
|
/// d. Find coclass implementing this particular primary interface
|
|
/// 3. Scan coclass for all its source interfaces.
|
|
/// 4. Check whether to any of the methods on the source interfaces matches
|
|
/// the request name
|
|
///
|
|
/// Once we determine that TryGetMember requests an event we will return
|
|
/// an instance of BoundDispEvent class. This class has InPlaceAdd and
|
|
/// InPlaceSubtract operators defined. Calling InPlaceAdd operator will:
|
|
/// 1. An instance of ComEventSinksContainer class is created (unless
|
|
/// RCW already had one). This instance is hanged off the RCW in attempt
|
|
/// to bind the lifetime of event sinks to the lifetime of the RCW itself,
|
|
/// meaning event sink will be collected once the RCW is collected (this
|
|
/// is the same way event sinks lifetime is controlled by PIAs).
|
|
/// Notice: ComEventSinksContainer contains a Finalizer which will go and
|
|
/// unadvise all event sinks.
|
|
/// Notice: ComEventSinksContainer is a list of ComEventSink objects.
|
|
/// 2. Unless we have already created a ComEventSink for the required
|
|
/// source interface, we will create and advise a new ComEventSink. Each
|
|
/// ComEventSink implements a single source interface that COM object
|
|
/// supports.
|
|
/// 3. ComEventSink contains a map between method DISPIDs to the
|
|
/// multicast delegate that will be invoked when the event is raised.
|
|
/// 4. ComEventSink implements IReflect interface which is exposed as
|
|
/// custom IDispatch to COM consumers. This allows us to intercept calls
|
|
/// to IDispatch.Invoke and apply custom logic - in particular we will
|
|
/// just find and invoke the multicast delegate corresponding to the invoked
|
|
/// dispid.
|
|
/// </summary>
|
|
|
|
internal sealed class IDispatchComObject : ComObject, IDynamicMetaObjectProvider {
|
|
|
|
private readonly IDispatch _dispatchObject;
|
|
private ComTypeDesc _comTypeDesc;
|
|
private static readonly Dictionary<Guid, ComTypeDesc> _CacheComTypeDesc = new Dictionary<Guid, ComTypeDesc>();
|
|
|
|
internal IDispatchComObject(IDispatch rcw)
|
|
: base(rcw) {
|
|
_dispatchObject = rcw;
|
|
}
|
|
|
|
public override string ToString() {
|
|
ComTypeDesc ctd = _comTypeDesc;
|
|
string typeName = null;
|
|
|
|
if (ctd != null) {
|
|
typeName = ctd.TypeName;
|
|
}
|
|
|
|
if (String.IsNullOrEmpty(typeName)) {
|
|
typeName = "IDispatch";
|
|
}
|
|
|
|
return String.Format(CultureInfo.CurrentCulture, "{0} ({1})", RuntimeCallableWrapper.ToString(), typeName);
|
|
}
|
|
|
|
public ComTypeDesc ComTypeDesc {
|
|
get {
|
|
EnsureScanDefinedMethods();
|
|
return _comTypeDesc;
|
|
}
|
|
}
|
|
|
|
public IDispatch DispatchObject {
|
|
get {
|
|
return _dispatchObject;
|
|
}
|
|
}
|
|
|
|
private static int GetIDsOfNames(IDispatch dispatch, string name, out int dispId) {
|
|
int[] dispIds = new int[1];
|
|
Guid emtpyRiid = Guid.Empty;
|
|
int hresult = dispatch.TryGetIDsOfNames(
|
|
ref emtpyRiid,
|
|
new string[] { name },
|
|
1,
|
|
0,
|
|
dispIds);
|
|
|
|
dispId = dispIds[0];
|
|
return hresult;
|
|
}
|
|
|
|
static int Invoke(IDispatch dispatch, int memberDispId, out object result) {
|
|
Guid emtpyRiid = Guid.Empty;
|
|
ComTypes.DISPPARAMS dispParams = new ComTypes.DISPPARAMS();
|
|
ComTypes.EXCEPINFO excepInfo = new ComTypes.EXCEPINFO();
|
|
uint argErr;
|
|
int hresult = dispatch.TryInvoke(
|
|
memberDispId,
|
|
ref emtpyRiid,
|
|
0,
|
|
ComTypes.INVOKEKIND.INVOKE_PROPERTYGET,
|
|
ref dispParams,
|
|
out result,
|
|
out excepInfo,
|
|
out argErr);
|
|
|
|
return hresult;
|
|
}
|
|
|
|
internal bool TryGetGetItem(out ComMethodDesc value) {
|
|
ComMethodDesc methodDesc = _comTypeDesc.GetItem;
|
|
if (methodDesc != null) {
|
|
value = methodDesc;
|
|
return true;
|
|
}
|
|
|
|
return SlowTryGetGetItem(out value);
|
|
}
|
|
|
|
private bool SlowTryGetGetItem(out ComMethodDesc value) {
|
|
EnsureScanDefinedMethods();
|
|
|
|
ComMethodDesc methodDesc = _comTypeDesc.GetItem;
|
|
|
|
// Without type information, we really don't know whether or not we have a property getter.
|
|
if (methodDesc == null) {
|
|
string name = "[PROPERTYGET, DISPID(0)]";
|
|
|
|
_comTypeDesc.EnsureGetItem(new ComMethodDesc(name, ComDispIds.DISPID_VALUE, ComTypes.INVOKEKIND.INVOKE_PROPERTYGET));
|
|
methodDesc = _comTypeDesc.GetItem;
|
|
}
|
|
|
|
value = methodDesc;
|
|
return true;
|
|
}
|
|
|
|
internal bool TryGetSetItem(out ComMethodDesc value) {
|
|
ComMethodDesc methodDesc = _comTypeDesc.SetItem;
|
|
if (methodDesc != null) {
|
|
value = methodDesc;
|
|
return true;
|
|
}
|
|
|
|
return SlowTryGetSetItem(out value);
|
|
}
|
|
|
|
private bool SlowTryGetSetItem(out ComMethodDesc value) {
|
|
EnsureScanDefinedMethods();
|
|
|
|
ComMethodDesc methodDesc = _comTypeDesc.SetItem;
|
|
|
|
// Without type information, we really don't know whether or not we have a property setter.
|
|
if (methodDesc == null) {
|
|
string name = "[PROPERTYPUT, DISPID(0)]";
|
|
|
|
_comTypeDesc.EnsureSetItem(new ComMethodDesc(name, ComDispIds.DISPID_VALUE, ComTypes.INVOKEKIND.INVOKE_PROPERTYPUT));
|
|
methodDesc = _comTypeDesc.SetItem;
|
|
}
|
|
|
|
value = methodDesc;
|
|
return true;
|
|
}
|
|
|
|
internal bool TryGetMemberMethod(string name, out ComMethodDesc method) {
|
|
EnsureScanDefinedMethods();
|
|
return _comTypeDesc.TryGetFunc(name, out method);
|
|
}
|
|
|
|
internal bool TryGetMemberEvent(string name, out ComEventDesc @event) {
|
|
EnsureScanDefinedEvents();
|
|
return _comTypeDesc.TryGetEvent(name, out @event);
|
|
}
|
|
|
|
internal bool TryGetMemberMethodExplicit(string name, out ComMethodDesc method) {
|
|
EnsureScanDefinedMethods();
|
|
|
|
int dispId;
|
|
int hresult = GetIDsOfNames(_dispatchObject, name, out dispId);
|
|
|
|
if (hresult == ComHresults.S_OK) {
|
|
ComMethodDesc cmd = new ComMethodDesc(name, dispId, ComTypes.INVOKEKIND.INVOKE_FUNC);
|
|
_comTypeDesc.AddFunc(name, cmd);
|
|
method = cmd;
|
|
return true;
|
|
} else if (hresult == ComHresults.DISP_E_UNKNOWNNAME) {
|
|
method = null;
|
|
return false;
|
|
} else {
|
|
throw Error.CouldNotGetDispId(name, string.Format(CultureInfo.InvariantCulture, "0x{1:X})", hresult));
|
|
}
|
|
}
|
|
|
|
internal bool TryGetPropertySetterExplicit(string name, out ComMethodDesc method, Type limitType, bool holdsNull) {
|
|
EnsureScanDefinedMethods();
|
|
|
|
int dispId;
|
|
int hresult = GetIDsOfNames(_dispatchObject, name, out dispId);
|
|
|
|
if (hresult == ComHresults.S_OK) {
|
|
// we do not know whether we have put or putref here
|
|
// and we will not guess and pretend we found both.
|
|
ComMethodDesc put = new ComMethodDesc(name, dispId, ComTypes.INVOKEKIND.INVOKE_PROPERTYPUT);
|
|
_comTypeDesc.AddPut(name, put);
|
|
|
|
ComMethodDesc putref = new ComMethodDesc(name, dispId, ComTypes.INVOKEKIND.INVOKE_PROPERTYPUTREF);
|
|
_comTypeDesc.AddPutRef(name, putref);
|
|
|
|
if (ComBinderHelpers.PreferPut(limitType, holdsNull)) {
|
|
method = put;
|
|
} else {
|
|
method = putref;
|
|
}
|
|
return true;
|
|
} else if (hresult == ComHresults.DISP_E_UNKNOWNNAME) {
|
|
method = null;
|
|
return false;
|
|
} else {
|
|
throw Error.CouldNotGetDispId(name, string.Format(CultureInfo.InvariantCulture, "0x{1:X})", hresult));
|
|
}
|
|
}
|
|
|
|
internal override IList<string> GetMemberNames(bool dataOnly) {
|
|
EnsureScanDefinedMethods();
|
|
EnsureScanDefinedEvents();
|
|
|
|
return ComTypeDesc.GetMemberNames(dataOnly);
|
|
}
|
|
|
|
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Design", "CA1031:DoNotCatchGeneralExceptionTypes")]
|
|
internal override IList<KeyValuePair<string, object>> GetMembers(IEnumerable<string> names) {
|
|
if (names == null) {
|
|
names = GetMemberNames(true);
|
|
}
|
|
|
|
Type comType = RuntimeCallableWrapper.GetType();
|
|
|
|
var members = new List<KeyValuePair<string, object>>();
|
|
foreach (string name in names) {
|
|
if (name == null) {
|
|
continue;
|
|
}
|
|
|
|
ComMethodDesc method;
|
|
if (ComTypeDesc.TryGetFunc(name, out method) && method.IsDataMember) {
|
|
try {
|
|
object value = comType.InvokeMember(
|
|
method.Name,
|
|
BindingFlags.GetProperty,
|
|
null,
|
|
RuntimeCallableWrapper,
|
|
new object[0],
|
|
CultureInfo.InvariantCulture
|
|
);
|
|
members.Add(new KeyValuePair<string, object>(method.Name, value));
|
|
|
|
//evaluation failed for some reason. pass exception out
|
|
} catch (Exception ex) {
|
|
members.Add(new KeyValuePair<string, object>(method.Name, ex));
|
|
}
|
|
}
|
|
}
|
|
|
|
return members.ToArray();
|
|
}
|
|
|
|
DynamicMetaObject IDynamicMetaObjectProvider.GetMetaObject(Expression parameter) {
|
|
EnsureScanDefinedMethods();
|
|
return new IDispatchMetaObject(parameter, this);
|
|
}
|
|
|
|
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2201:DoNotRaiseReservedExceptionTypes")]
|
|
[SecurityCritical]
|
|
private static void GetFuncDescForDescIndex(ComTypes.ITypeInfo typeInfo, int funcIndex, out ComTypes.FUNCDESC funcDesc, out IntPtr funcDescHandle) {
|
|
IntPtr pFuncDesc = IntPtr.Zero;
|
|
typeInfo.GetFuncDesc(funcIndex, out pFuncDesc);
|
|
|
|
// GetFuncDesc should never return null, this is just to be safe
|
|
if (pFuncDesc == IntPtr.Zero) {
|
|
throw Error.CannotRetrieveTypeInformation();
|
|
}
|
|
|
|
funcDesc = (ComTypes.FUNCDESC)Marshal.PtrToStructure(pFuncDesc, typeof(ComTypes.FUNCDESC));
|
|
funcDescHandle = pFuncDesc;
|
|
}
|
|
|
|
#if CLR2
|
|
[SecurityCritical, SecurityTreatAsSafe]
|
|
#else
|
|
[SecuritySafeCritical]
|
|
#endif
|
|
private void EnsureScanDefinedEvents() {
|
|
// _comTypeDesc.Events is null if we have not yet attempted
|
|
// to scan the object for events.
|
|
if (_comTypeDesc != null && _comTypeDesc.Events != null) {
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Demand Full Trust to proceed with the operation.
|
|
//
|
|
|
|
new PermissionSet(PermissionState.Unrestricted).Demand();
|
|
|
|
// check type info in the type descriptions cache
|
|
ComTypes.ITypeInfo typeInfo = ComRuntimeHelpers.GetITypeInfoFromIDispatch(_dispatchObject, true);
|
|
if (typeInfo == null) {
|
|
_comTypeDesc = ComTypeDesc.CreateEmptyTypeDesc();
|
|
return;
|
|
}
|
|
|
|
ComTypes.TYPEATTR typeAttr = ComRuntimeHelpers.GetTypeAttrForTypeInfo(typeInfo);
|
|
|
|
if (_comTypeDesc == null) {
|
|
lock (_CacheComTypeDesc) {
|
|
if (_CacheComTypeDesc.TryGetValue(typeAttr.guid, out _comTypeDesc) == true &&
|
|
_comTypeDesc.Events != null) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
ComTypeDesc typeDesc = ComTypeDesc.FromITypeInfo(typeInfo, typeAttr);
|
|
|
|
ComTypes.ITypeInfo classTypeInfo = null;
|
|
Dictionary<string, ComEventDesc> events = null;
|
|
|
|
var cpc = RuntimeCallableWrapper as ComTypes.IConnectionPointContainer;
|
|
if (cpc == null) {
|
|
// No ICPC - this object does not support events
|
|
events = ComTypeDesc.EmptyEvents;
|
|
} else if ((classTypeInfo = GetCoClassTypeInfo(this.RuntimeCallableWrapper, typeInfo)) == null) {
|
|
// no class info found - this object may support events
|
|
// but we could not discover those
|
|
events = ComTypeDesc.EmptyEvents;
|
|
} else {
|
|
events = new Dictionary<string, ComEventDesc>();
|
|
|
|
ComTypes.TYPEATTR classTypeAttr = ComRuntimeHelpers.GetTypeAttrForTypeInfo(classTypeInfo);
|
|
for (int i = 0; i < classTypeAttr.cImplTypes; i++) {
|
|
int hRefType;
|
|
classTypeInfo.GetRefTypeOfImplType(i, out hRefType);
|
|
|
|
ComTypes.ITypeInfo interfaceTypeInfo;
|
|
classTypeInfo.GetRefTypeInfo(hRefType, out interfaceTypeInfo);
|
|
|
|
ComTypes.IMPLTYPEFLAGS flags;
|
|
classTypeInfo.GetImplTypeFlags(i, out flags);
|
|
if ((flags & ComTypes.IMPLTYPEFLAGS.IMPLTYPEFLAG_FSOURCE) != 0) {
|
|
ScanSourceInterface(interfaceTypeInfo, ref events);
|
|
}
|
|
}
|
|
|
|
if (events.Count == 0) {
|
|
events = ComTypeDesc.EmptyEvents;
|
|
}
|
|
}
|
|
|
|
lock (_CacheComTypeDesc) {
|
|
ComTypeDesc cachedTypeDesc;
|
|
if (_CacheComTypeDesc.TryGetValue(typeAttr.guid, out cachedTypeDesc)) {
|
|
_comTypeDesc = cachedTypeDesc;
|
|
} else {
|
|
_comTypeDesc = typeDesc;
|
|
_CacheComTypeDesc.Add(typeAttr.guid, _comTypeDesc);
|
|
}
|
|
_comTypeDesc.Events = events;
|
|
}
|
|
}
|
|
|
|
[SecurityCritical]
|
|
private static void ScanSourceInterface(ComTypes.ITypeInfo sourceTypeInfo, ref Dictionary<string, ComEventDesc> events) {
|
|
ComTypes.TYPEATTR sourceTypeAttribute = ComRuntimeHelpers.GetTypeAttrForTypeInfo(sourceTypeInfo);
|
|
|
|
for (int index = 0; index < sourceTypeAttribute.cFuncs; index++) {
|
|
IntPtr funcDescHandleToRelease = IntPtr.Zero;
|
|
|
|
try {
|
|
ComTypes.FUNCDESC funcDesc;
|
|
GetFuncDescForDescIndex(sourceTypeInfo, index, out funcDesc, out funcDescHandleToRelease);
|
|
|
|
// we are not interested in hidden or restricted functions for now.
|
|
if ((funcDesc.wFuncFlags & (int)ComTypes.FUNCFLAGS.FUNCFLAG_FHIDDEN) != 0) {
|
|
continue;
|
|
}
|
|
if ((funcDesc.wFuncFlags & (int)ComTypes.FUNCFLAGS.FUNCFLAG_FRESTRICTED) != 0) {
|
|
continue;
|
|
}
|
|
|
|
string name = ComRuntimeHelpers.GetNameOfMethod(sourceTypeInfo, funcDesc.memid);
|
|
name = name.ToUpper(System.Globalization.CultureInfo.InvariantCulture);
|
|
|
|
// Sometimes coclass has multiple source interfaces. Usually this is caused by
|
|
// adding new events and putting them on new interfaces while keeping the
|
|
// old interfaces around. This may cause name collisioning which we are
|
|
// resolving by keeping only the first event with the same name.
|
|
if (events.ContainsKey(name) == false) {
|
|
ComEventDesc eventDesc = new ComEventDesc();
|
|
eventDesc.dispid = funcDesc.memid;
|
|
eventDesc.sourceIID = sourceTypeAttribute.guid;
|
|
events.Add(name, eventDesc);
|
|
}
|
|
} finally {
|
|
if (funcDescHandleToRelease != IntPtr.Zero) {
|
|
sourceTypeInfo.ReleaseFuncDesc(funcDescHandleToRelease);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
[SecurityCritical]
|
|
private static ComTypes.ITypeInfo GetCoClassTypeInfo(object rcw, ComTypes.ITypeInfo typeInfo) {
|
|
Debug.Assert(typeInfo != null);
|
|
|
|
IProvideClassInfo provideClassInfo = rcw as IProvideClassInfo;
|
|
if (provideClassInfo != null) {
|
|
IntPtr typeInfoPtr = IntPtr.Zero;
|
|
try {
|
|
provideClassInfo.GetClassInfo(out typeInfoPtr);
|
|
if (typeInfoPtr != IntPtr.Zero) {
|
|
return Marshal.GetObjectForIUnknown(typeInfoPtr) as ComTypes.ITypeInfo;
|
|
}
|
|
} finally {
|
|
if (typeInfoPtr != IntPtr.Zero) {
|
|
Marshal.Release(typeInfoPtr);
|
|
}
|
|
}
|
|
}
|
|
|
|
// retrieving class information through IPCI has failed -
|
|
// we can try scanning the typelib to find the coclass
|
|
|
|
ComTypes.ITypeLib typeLib;
|
|
int typeInfoIndex;
|
|
typeInfo.GetContainingTypeLib(out typeLib, out typeInfoIndex);
|
|
string typeName = ComRuntimeHelpers.GetNameOfType(typeInfo);
|
|
|
|
ComTypeLibDesc typeLibDesc = ComTypeLibDesc.GetFromTypeLib(typeLib);
|
|
ComTypeClassDesc coclassDesc = typeLibDesc.GetCoClassForInterface(typeName);
|
|
if (coclassDesc == null) {
|
|
return null;
|
|
}
|
|
|
|
ComTypes.ITypeInfo typeInfoCoClass;
|
|
Guid coclassGuid = coclassDesc.Guid;
|
|
typeLib.GetTypeInfoOfGuid(ref coclassGuid, out typeInfoCoClass);
|
|
return typeInfoCoClass;
|
|
}
|
|
|
|
#if CLR2
|
|
[SecurityCritical, SecurityTreatAsSafe]
|
|
#else
|
|
[SecuritySafeCritical]
|
|
#endif
|
|
private void EnsureScanDefinedMethods() {
|
|
if (_comTypeDesc != null && _comTypeDesc.Funcs != null) {
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Demand Full Trust to proceed with the operation.
|
|
//
|
|
|
|
new PermissionSet(PermissionState.Unrestricted).Demand();
|
|
|
|
ComTypes.ITypeInfo typeInfo = ComRuntimeHelpers.GetITypeInfoFromIDispatch(_dispatchObject, true);
|
|
if (typeInfo == null) {
|
|
_comTypeDesc = ComTypeDesc.CreateEmptyTypeDesc();
|
|
return;
|
|
}
|
|
|
|
ComTypes.TYPEATTR typeAttr = ComRuntimeHelpers.GetTypeAttrForTypeInfo(typeInfo);
|
|
|
|
if (_comTypeDesc == null) {
|
|
lock (_CacheComTypeDesc) {
|
|
if (_CacheComTypeDesc.TryGetValue(typeAttr.guid, out _comTypeDesc) == true &&
|
|
_comTypeDesc.Funcs != null) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
ComTypeDesc typeDesc = ComTypeDesc.FromITypeInfo(typeInfo, typeAttr);
|
|
|
|
ComMethodDesc getItem = null;
|
|
ComMethodDesc setItem = null;
|
|
Hashtable funcs = new Hashtable(typeAttr.cFuncs);
|
|
Hashtable puts = new Hashtable();
|
|
Hashtable putrefs = new Hashtable();
|
|
|
|
for (int definedFuncIndex = 0; definedFuncIndex < typeAttr.cFuncs; definedFuncIndex++) {
|
|
IntPtr funcDescHandleToRelease = IntPtr.Zero;
|
|
|
|
try {
|
|
ComTypes.FUNCDESC funcDesc;
|
|
GetFuncDescForDescIndex(typeInfo, definedFuncIndex, out funcDesc, out funcDescHandleToRelease);
|
|
|
|
if ((funcDesc.wFuncFlags & (int)ComTypes.FUNCFLAGS.FUNCFLAG_FRESTRICTED) != 0) {
|
|
// This function is not meant for the script user to use.
|
|
continue;
|
|
}
|
|
|
|
ComMethodDesc method = new ComMethodDesc(typeInfo, funcDesc);
|
|
string name = method.Name.ToUpper(System.Globalization.CultureInfo.InvariantCulture);
|
|
|
|
if ((funcDesc.invkind & ComTypes.INVOKEKIND.INVOKE_PROPERTYPUT) != 0) {
|
|
puts.Add(name, method);
|
|
|
|
// for the special dispId == 0, we need to store
|
|
// the method descriptor for the Do(SetItem) binder.
|
|
if (method.DispId == ComDispIds.DISPID_VALUE && setItem == null) {
|
|
setItem = method;
|
|
}
|
|
continue;
|
|
}
|
|
if ((funcDesc.invkind & ComTypes.INVOKEKIND.INVOKE_PROPERTYPUTREF) != 0) {
|
|
putrefs.Add(name, method);
|
|
// for the special dispId == 0, we need to store
|
|
// the method descriptor for the Do(SetItem) binder.
|
|
if (method.DispId == ComDispIds.DISPID_VALUE && setItem == null) {
|
|
setItem = method;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (funcDesc.memid == ComDispIds.DISPID_NEWENUM) {
|
|
funcs.Add("GETENUMERATOR", method);
|
|
continue;
|
|
}
|
|
|
|
funcs.Add(name, method);
|
|
|
|
// for the special dispId == 0, we need to store the method descriptor
|
|
// for the Do(GetItem) binder.
|
|
if (funcDesc.memid == ComDispIds.DISPID_VALUE) {
|
|
getItem = method;
|
|
}
|
|
} finally {
|
|
if (funcDescHandleToRelease != IntPtr.Zero) {
|
|
typeInfo.ReleaseFuncDesc(funcDescHandleToRelease);
|
|
}
|
|
}
|
|
}
|
|
|
|
lock (_CacheComTypeDesc) {
|
|
ComTypeDesc cachedTypeDesc;
|
|
if (_CacheComTypeDesc.TryGetValue(typeAttr.guid, out cachedTypeDesc)) {
|
|
_comTypeDesc = cachedTypeDesc;
|
|
} else {
|
|
_comTypeDesc = typeDesc;
|
|
_CacheComTypeDesc.Add(typeAttr.guid, _comTypeDesc);
|
|
}
|
|
_comTypeDesc.Funcs = funcs;
|
|
_comTypeDesc.Puts = puts;
|
|
_comTypeDesc.PutRefs = putrefs;
|
|
_comTypeDesc.EnsureGetItem(getItem);
|
|
_comTypeDesc.EnsureSetItem(setItem);
|
|
}
|
|
}
|
|
|
|
internal bool TryGetPropertySetter(string name, out ComMethodDesc method, Type limitType, bool holdsNull) {
|
|
EnsureScanDefinedMethods();
|
|
|
|
if (ComBinderHelpers.PreferPut(limitType, holdsNull)) {
|
|
return _comTypeDesc.TryGetPut(name, out method) ||
|
|
_comTypeDesc.TryGetPutRef(name, out method);
|
|
} else {
|
|
return _comTypeDesc.TryGetPutRef(name, out method) ||
|
|
_comTypeDesc.TryGetPut(name, out method);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|