我最近花了一些时间分析OutputDebugString方法。在我的另一个实验中,我需要一个仅依赖于本机API的OutputDebugString版本。在实现它的过程中,我发现了一些关于OutputDebugString的有趣的事实,也许您也会感兴趣。
简而言之,OutputDebugString尝试将消息发送到附加到给定进程的调试器,如果没有调试器侦听,则尝试将全局节映射到进程内存中并将调试消息保存在其中。我使用本机API实现OutputDebugStringA(ANSI版本)的示例如下:
void NTAPI RtlOutputDebugStringA(_In_opt_ LPCSTR OutputString) {
if (OutputString) {
EXCEPTION_RECORD exceptionRecord{ };
exceptionRecord.ExceptionCode = DBG\_PRINTEXCEPTION\_C;
exceptionRecord.NumberParameters = ;
exceptionRecord.ExceptionInformation\[\] = strlen(OutputString) + ;
exceptionRecord.ExceptionInformation\[\] = reinterpret\_cast<ULONG\_PTR>(OutputString);
\_\_try {
RtlRaiseException(&exceptionRecord);
} \_\_except (EXCEPTION\_EXECUTE\_HANDLER) {
NotifyGlobalDebugOutputMonitor(OutputString);
}
}
}
RtlOutputDebugStringA(以及OutputDebugString)检查调试器存在的方式非常有趣:它引发了一个异常。如果有一个调试器正在侦听,它将吞了异常(异常代码:0x40010006L表示ANSI消息,0x4001000A表示UNICODE消息),并且处理程序将永远不会执行。我们都知道异常是昂贵的,我们应该只在特殊情况下使用它们。因此,对跟踪的每个写操作都抛出异常似乎不正确。稍后我将向您展示一些基准测试结果和解决此问题的简单方法(我想您已经知道了)。但是关于NotifyGlobalDebugOutputMonitor方法的前几句话。它使用一个全局映射部分、两个事件对象和一个互斥对象来编写调试消息。事件对象和互斥锁保护该节防止并发使用。我不会过多地讨论这个问题;如果你感兴趣,可以看看张玉武关于代码项目的优秀文章。您还可以在github上查看我的实现的源代码(它只是一个POC,所以请不要在接近生产的地方使用它)。可以说,在系统中运行调试输出监视器(如DebugView)也会对OutputDebugString性能产生负面影响,特别是当多个进程同时向调试输出写入数据时。
手机扫一扫
移动阅读更方便
你可能感兴趣的文章