调用 IStorage 的 EnumElements 时发生访问冲突
Access violation when calling EnumElements of IStorage
我正在读取结构化存储文件。并尝试获取根结构的所有子元素。但是我在这样做时遇到了访问冲突异常。
这是原生方法,
[ComImport][Guid("0000000d-0000-0000-C000-000000000046")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IEnumSTATSTG
{
[PreserveSig] uint Next(uint celt, [MarshalAs(UnmanagedType.LPArray), Out] System.Runtime.InteropServices.ComTypes.STATSTG[] rgelt, out uint pceltFetched);
}
[ComImport][Guid("0000000b-0000-0000-C000-000000000046")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IStorage
{
[return: MarshalAs(UnmanagedType.Interface)]
IStream OpenStream([In, MarshalAs(UnmanagedType.BStr)] string pwcsName, IntPtr reserved1, [In, MarshalAs(UnmanagedType.U4)] int grfMode, [In, MarshalAs(UnmanagedType.U4)] int reserved2);
void EnumElements(
/* [in] */ uint reserved1,
/* [size_is][unique][in] */ IntPtr reserved2,
/* [in] */ uint reserved3,
/* [out] */ out IEnumSTATSTG ppenum);
}
[DllImport("ole32.dll", CharSet = CharSet.Unicode)]
internal static extern uint StgOpenStorageEx
(
[MarshalAs(UnmanagedType.LPWStr)] string name, uint accessMode,
uint storageFileFormat, uint fileBuffering, IntPtr options,
IntPtr reserved, ref Guid riid, [MarshalAs(UnmanagedType.Interface)] ref IStorage stg
);
这是我的调用代码。
IStorage _storageObject;
// Opening file,
NativeMethods.StgOpenStorageEx(path, (uint)STM.Read | STM.ShareDenyWrite, (uint)storageFileFormat, (uint)fileBuffering,
options, IntPtr.Zero, ref _iidIStorage, ref _storageObject);
// Here I am calling EnumElements, I get exception here.
IEnumSTATSTG pIEnumStatStg;
_storageObject.EnumElements(0, IntPtr.Zero, 0, out pIEnumStatStg);
请注意,如果我调用另一个方法 if IStorage,比如 OpenStream,它工作正常,
_storageObject.OpenStream(streamName, IntPtr.Zero, (int)accessMode, 0);
我在打开文件时尝试了不同的 STM 标志组合,但它不起作用。
正如您所怀疑的,您的接口声明是完全错误的。方法的名称无关紧要,就像 COM 中的所有名称一样,重要的是方法的 order。他们必须 co-incide 与界面的 v-table。这意味着您不能只省略方法 willy-nilly.
这是一个与 .NET 接口不同的细节,CLR 弄清楚如何将接口方法绑定到它们的实现,并且它可以这样做,因为它可以访问声明和实现。但它无法访问 COM 方法实现,它们隐藏在视线之外,通常在用 C++ 或 Delphi.
编写的 DLL 中
您的 OpenStream() 测试实际上不起作用。您将其声明为第一种方法,但实际上是第二种方法。您实际上是在调用 CreateStream()。它没有炸弹是一个意外,CreateStream 恰好也有 5 个参数,它们是相似的。 EnumElements()那点运气运行出来,其实是接口的第9个方法。您正在调用第二种方法 OpenStream。这次的论据完全错误,kaboom.
您可以在接口声明上使用快捷方式,但您必须为要跳过的方法使用占位符。喜欢:
[ComImport][Guid("0000000b-0000-0000-C000-000000000046")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IStorage
{
void Dummy1();
[return: MarshalAs(UnmanagedType.Interface)]
IStream OpenStream([In, MarshalAs(UnmanagedType.BStr)] string pwcsName, IntPtr reserved1, [In, MarshalAs(UnmanagedType.U4)] int grfMode, [In, MarshalAs(UnmanagedType.U4)] int reserved2);
void Dummy3();
void Dummy4();
void Dummy5();
void Dummy6();
void Dummy7();
void Dummy8();
void EnumElements(
/* [in] */ uint reserved1,
/* [size_is][unique][in] */ IntPtr reserved2,
/* [in] */ uint reserved3,
/* [out] */ out IEnumSTATSTG ppenum);
}
可以省略尾随方法。您必须以相同的方式修复您的 IEnumSTATSTG 声明。这里不是这种情况,但是如果接口继承自 IUnknown 或 IDispatch 以外的基接口,那么您还必须声明继承的方法。
我正在读取结构化存储文件。并尝试获取根结构的所有子元素。但是我在这样做时遇到了访问冲突异常。
这是原生方法,
[ComImport][Guid("0000000d-0000-0000-C000-000000000046")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IEnumSTATSTG
{
[PreserveSig] uint Next(uint celt, [MarshalAs(UnmanagedType.LPArray), Out] System.Runtime.InteropServices.ComTypes.STATSTG[] rgelt, out uint pceltFetched);
}
[ComImport][Guid("0000000b-0000-0000-C000-000000000046")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IStorage
{
[return: MarshalAs(UnmanagedType.Interface)]
IStream OpenStream([In, MarshalAs(UnmanagedType.BStr)] string pwcsName, IntPtr reserved1, [In, MarshalAs(UnmanagedType.U4)] int grfMode, [In, MarshalAs(UnmanagedType.U4)] int reserved2);
void EnumElements(
/* [in] */ uint reserved1,
/* [size_is][unique][in] */ IntPtr reserved2,
/* [in] */ uint reserved3,
/* [out] */ out IEnumSTATSTG ppenum);
}
[DllImport("ole32.dll", CharSet = CharSet.Unicode)]
internal static extern uint StgOpenStorageEx
(
[MarshalAs(UnmanagedType.LPWStr)] string name, uint accessMode,
uint storageFileFormat, uint fileBuffering, IntPtr options,
IntPtr reserved, ref Guid riid, [MarshalAs(UnmanagedType.Interface)] ref IStorage stg
);
这是我的调用代码。
IStorage _storageObject;
// Opening file,
NativeMethods.StgOpenStorageEx(path, (uint)STM.Read | STM.ShareDenyWrite, (uint)storageFileFormat, (uint)fileBuffering,
options, IntPtr.Zero, ref _iidIStorage, ref _storageObject);
// Here I am calling EnumElements, I get exception here.
IEnumSTATSTG pIEnumStatStg;
_storageObject.EnumElements(0, IntPtr.Zero, 0, out pIEnumStatStg);
请注意,如果我调用另一个方法 if IStorage,比如 OpenStream,它工作正常,
_storageObject.OpenStream(streamName, IntPtr.Zero, (int)accessMode, 0);
我在打开文件时尝试了不同的 STM 标志组合,但它不起作用。
正如您所怀疑的,您的接口声明是完全错误的。方法的名称无关紧要,就像 COM 中的所有名称一样,重要的是方法的 order。他们必须 co-incide 与界面的 v-table。这意味着您不能只省略方法 willy-nilly.
这是一个与 .NET 接口不同的细节,CLR 弄清楚如何将接口方法绑定到它们的实现,并且它可以这样做,因为它可以访问声明和实现。但它无法访问 COM 方法实现,它们隐藏在视线之外,通常在用 C++ 或 Delphi.
编写的 DLL 中您的 OpenStream() 测试实际上不起作用。您将其声明为第一种方法,但实际上是第二种方法。您实际上是在调用 CreateStream()。它没有炸弹是一个意外,CreateStream 恰好也有 5 个参数,它们是相似的。 EnumElements()那点运气运行出来,其实是接口的第9个方法。您正在调用第二种方法 OpenStream。这次的论据完全错误,kaboom.
您可以在接口声明上使用快捷方式,但您必须为要跳过的方法使用占位符。喜欢:
[ComImport][Guid("0000000b-0000-0000-C000-000000000046")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IStorage
{
void Dummy1();
[return: MarshalAs(UnmanagedType.Interface)]
IStream OpenStream([In, MarshalAs(UnmanagedType.BStr)] string pwcsName, IntPtr reserved1, [In, MarshalAs(UnmanagedType.U4)] int grfMode, [In, MarshalAs(UnmanagedType.U4)] int reserved2);
void Dummy3();
void Dummy4();
void Dummy5();
void Dummy6();
void Dummy7();
void Dummy8();
void EnumElements(
/* [in] */ uint reserved1,
/* [size_is][unique][in] */ IntPtr reserved2,
/* [in] */ uint reserved3,
/* [out] */ out IEnumSTATSTG ppenum);
}
可以省略尾随方法。您必须以相同的方式修复您的 IEnumSTATSTG 声明。这里不是这种情况,但是如果接口继承自 IUnknown 或 IDispatch 以外的基接口,那么您还必须声明继承的方法。