7

CLR メモリ診断ライブラリを使用して、実行中のプロセスのすべてのスレッドのスタック トレースを取得しています。

        var result = new Dictionary<int, string[]>();

        var pid = Process.GetCurrentProcess().Id;

        using (var dataTarget = DataTarget.AttachToProcess(pid, 5000, AttachFlag.Passive))
        {
            string dacLocation = dataTarget.ClrVersions[0].TryGetDacLocation();
            var runtime = dataTarget.CreateRuntime(dacLocation); //throws exception

            foreach (var t in runtime.Threads)
            {
                result.Add(
                    t.ManagedThreadId,
                    t.StackTrace.Select(f =>
                    {
                        if (f.Method != null)
                        {
                            return f.Method.Type.Name + "." + f.Method.Name;
                        }

                        return null;
                    }).ToArray()
                );
            }
        }

ここからこのコードを取得しましたが、他の人にとっては機能しているようですが、指定された行で例外がスローされ、次のメッセージが表示されますThis runtime is not initialized and contains no data.

dacLocationとして設定されていますC:\\Windows\\Microsoft.NET\\Framework\\v4.0.30319\\mscordacwks.dll

4

2 に答える 2

10

ClrMD は現在、.NET 4.6 をサポートしていません。この問題をたった 1 行で修正するオープン プル リクエストが GitHubにあります。もちろん、プロジェクトを複製して、この問題を示さない独自の ClrMD を構築することもできます。

または、過去数週間にわたって使用してきた一時的なハックを共有できます。

public static ClrRuntime CreateRuntimeHack(this DataTarget target, string dacLocation, int major, int minor)
{
    string dacFileNoExt = Path.GetFileNameWithoutExtension(dacLocation);
    if (dacFileNoExt.Contains("mscordacwks") && major == 4 && minor >= 5)
    {
        Type dacLibraryType = typeof(DataTarget).Assembly.GetType("Microsoft.Diagnostics.Runtime.DacLibrary");
        object dacLibrary = Activator.CreateInstance(dacLibraryType, target, dacLocation);
        Type v45RuntimeType = typeof(DataTarget).Assembly.GetType("Microsoft.Diagnostics.Runtime.Desktop.V45Runtime");
        object runtime = Activator.CreateInstance(v45RuntimeType, target, dacLibrary);
        return (ClrRuntime)runtime;
    }
    else
    {
        return target.CreateRuntime(dacLocation);
    }
}

私は知っています、それは恐ろしいことであり、リフレクションに依存しています。しかし、少なくとも今のところは機能しており、コードを変更する必要はありません。

于 2015-07-31T12:00:46.127 に答える