gpt4 book ai didi

c# - 检测是否从 P/Invoke 调用

转载 作者:太空狗 更新时间:2023-10-29 23:51:07 25 4
gpt4 key购买 nike

我正在使用用 C# 编写的托管服务。
该服务使用 P/Invoke 与非托管 COM 对象交互。
服务为对象提供对回调接口(interface)的引用。
该对象会在适当的时候调用接口(interface)上的方法。

我的问题:在运行时从服务代码中的任何一点,是否可以确定当前线程是否由于 P/Invoke 回调而正在执行?

换句话说,我基本上想做一个堆栈跟踪并检查基础调用是否是一个 P/Invoke 回调——没有任何对回调接口(interface)或其方法名称的引用。

希望这是有道理的!


编辑:我笼统地问这个问题的原因是因为我想知道这个问题是否有答案。我知道设计可能……有问题……但此时我看不到任何替代解决方案。以下是一些可能想要为我的更具体问题提供解决方案的人的更多背景信息。

我要处理的问题归结为公寓(STA 与 MTA)。

COM 对象是与硬件设备交互的接口(interface)的一部分。该服务管理多个硬件设备(即 COM 对象的多个实例在运行时存在,使用相同的回调接口(interface))。

该服务在 MTA 中运行。 COM 对象实例通常也运行在 MTA 中(因为对象是由服务实例化的)。但是,为了与某些硬件驱动程序交互,COM 对象有时必须在 STA 中运行。在那些情况下,服务使用带有内部 STA 线程的包装类来代理与 STA 中的 COM 对象实例的交互。包装器使用与 COM 对象相同的接口(interface),以保持同一个代码路径。

大多数情况下,包装器解决方案都可以正常工作。问题在于处理来自 COM 对象的回调,因为服务需要在回调执行期间调用对象上的方法。如果您尝试通过 STA 线程代理这些方法调用,它会导致死锁,大概是因为调用在 STA 消息队列上被阻塞(在回调完成之前不会被处理)。

当 STA 实例在服务上调用回调时,P/Invoke 会按照我们的预期在 MTA 线程中执行这些回调(因为服务在 MTA 中运行)。让我没想到的是,在回调过程中,只需绕过代理线程,就可以直接调用COM对象上的方法。 P/Invoke 必须以某种方式意识到线程起源于 STA 中的对象。

所以问题的症结在于,在处理STA时,必须根据上下文以不同的方式调用相同的方法。如果服务想从它自己管理的 MTA 线程中调用一个方法,那么它必须通过 STA 代理线程;但是,如果服务想要在回调期间调用相同的方法,则它必须直接调用该方法。如果我可以区分这两种情况,那么我就有了解决问题的方法。

在你建议之前...不,将所有必需的信息作为参数传递给回调以避免在回调期间调用方法是不切实际/不可能的。不,回调完成后无法检索所需信息。 COM 对象的上下文在回调完成后丢失。试图保护它是不切实际的。

最佳答案

这有帮助吗

using System.Diagnostics; 

StackTrace stackTrace = new StackTrace(); // Get call stack
StackFrame[] stackFrames = stackTrace.GetFrames(); // Get method calls (frames)

// write call stack method names
foreach (StackFrame stackFrame in stackFrames)
{
Console.WriteLine(stackFrame.GetMethod().Name); // Write method name
}

关于c# - 检测是否从 P/Invoke 调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23829261/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com