从非托管代码更新性能计数器

Posted

技术标签:

【中文标题】从非托管代码更新性能计数器【英文标题】:Updating performance counters from unmanaged code 【发布时间】:2012-06-12 13:13:22 【问题描述】:

我在使用 PerflibV2 预览自定义性能计数器时遇到问题。

性能监视器按 GUID 显示我的自定义性能计数器组,当我想展开它时,会显示“无法加载计数器”。

我尝试将自己添加到“性能监视器用户”和“性能日志用户”组,但没有成功。

我google了一下,看了很多MSDN文章,都没有成功。

有人熟悉这个问题吗?

以下是我如何创建和添加自定义性能计数器的详细过程:

我需要创建一个性能计数器,该计数器将从我未管理的应用程序中更新。

我发现了两种方法:

封装托管性能计数器 API,这不是一个选项,因为它会影响性能;

使用提供所需功能的 PerflibV2;

作为一个测试应用程序,我创建了以下描述自定义性能计数器的 schema.xml 架构:

<!-- <?xml version="1.0" encoding="UTF-16"?> -->
<instrumentationManifest     
xmlns="http://schemas.microsoft.com/win/2004/08/events" 
xmlns:win="http://manifests.microsoft.com/win/2004/08/windows/events"
xmlns:xs="http://www.w3.org/2001/XMLSchema"    
>
<instrumentation>
    <counters xmlns="http://schemas.microsoft.com/win/2005/12/counters">
        <provider callback = "custom"
          applicationIdentity = "PerfCounters.exe"
          providerType = "userMode"
          providerGuid = "ab8e1320-965a-4cf9-9c07-fe25378c2a23">
            <counterSet
              guid = "ad36a036-c923-4794-b696-70577630b5cf"
              uri = "Microsoft.Windows.System.PerfCounters.MyCounterSet1"
              name = "My LogicalDisk" 
              description = "This is a sample counter set with multiple instances." 
              instances = "multiple">
                <counter id = "1"
                  uri = "Microsoft.Windows.System.PerfCounters.MyCounterSet1.MyCounter1"
                  name = "My Free Megabytes"
                  description = "First sample counter."
                  type = "perf_counter_rawcount"
                  detailLevel = "standard"
                  defaultScale = "1"/>
            </counterSet>
        </provider>
    </counters>
    </instrumentation>
</instrumentationManifest>

并执行:

ctrpp schema.xml

我将创建的文件添加到我的测试应用程序中,并且在我的测试应用程序中,大致如下:

PerfAutoInitialize();

ULONG instanceId = 0;
wchar_t instanceName[] = 't', 'e', 's', 't', 0;
PPERF_COUNTERSET_INSTANCE b = PerfCreateInstance(hDataSource_schema_1, &CtrSetGuid_schema_1_1, instanceName, instanceId);

我安装了性能计数器:

lodctr /m:schema.xml

我的 PerfCounters 应用程序在尝试从性能监视器读取计数器时启动并运行。

【问题讨论】:

【参考方案1】:

注册 PerfCounter 提供程序失败的原因有几个:

检查清单文件的架构是否有效。您可以根据 XSD definition file provided by Microsoft 验证文件。

如果您想检查,请使用lodctr 工具注册您的清单。确保以管理员身份运行 lodctr 工具。如果您的权利不足,它将默默地失败。注册清单后,您应该能够在 PerfCounter 对话框中看到 CounterSet 的 GUID。 (有关可以列出提供程序的工具,请参阅 Browsing Performance Counters。)

您必须同时生成头文件和资源文件(使用ctrpp-rc-o 选项)。资源文件必须添加到您的解决方案中。

构建您的应用程序,然后重新运行lodctr 工具,同时清单文件和您的 .exe 都在当前目录中。确保清单文件指向提供程序的 applicationIdentity 属性中二进制文件的文件名:

&lt;provider symbol="MyProvider" applicationIdentity="PerfCounterTest.exe" providerName="PerfCounterTest"

运行应用程序。当应用程序运行时,您应该能够在浏览器对话框中看到您的提供商的名称:

【讨论】:

我还必须提到,64 位 permon.exe 只有在 64 位自定义计数器进程正在运行时才会加载。我正在调试/运行 32 位,但它没有加载。我尝试了 SysWow64\perfmon,它成功了!【参考方案2】:

我不完全确定我的示例到底出了什么问题,但是有一个 Microsoft 示例,其中包含适用于 Windows 7 和 .NET Framework 3.5 SP1 的 Microsoft Windows SDK:

http://www.microsoft.com/en-us/download/confirmation.aspx?id=3138

安装 MS SDK 示例后位于:

C:\Program Files\Microsoft SDKs\Windows\v7.0\Samples\winbase\PerfCounters\Basic\CPP

它应该适用于 Windows 7(ctrcpp 参数较少,使用 PerfAutoInitialize() 和 PerfAutoCleanup() 代替 CounterInitialize() 和 CounterCleanup())。

应用程序在从 perfmon 添加计数器时崩溃,请参阅: Perflib 2 crashes when adding a counter (from Perfmon)

【讨论】:

【参考方案3】:

我知道这个问题已经得到解答,但我遇到了同样的问题,这是因为我忘记将生成的 .RC 文件包含到可执行文件中。当我用 stringtable、unlodctred 和lodctred 架构文件重新编译包含 .RC 文件的可执行文件时,它开始工作了。

【讨论】:

以上是关于从非托管代码更新性能计数器的主要内容,如果未能解决你的问题,请参考以下文章

RawFraction 性能计数器即使在删除性能类别后仍保持其状态

如何使用性能计数器监控 WCF 服务正常运行时间?

在 Azure 中使用 Visual Studio Online 加载测试性能计数器

RateOfCountsPerSecond64 类型的性能计数器的值始终为 0

Java高性能并发计数器之巅峰对决

无法让 nservicebus 性能计数器在 Windows Server 2008 R2 上的开发/自主机中工作