From 1812747542355ecd356d0c6b967c418375a590b2 Mon Sep 17 00:00:00 2001 From: Juan Sebastian Hoyos Ayala Date: Fri, 17 Mar 2023 01:46:30 -0700 Subject: [PATCH 1/6] General cleanup for dotnet-dump command binding --- src/Tools/dotnet-dump/Dumper.cs | 115 ++++++++++++++----------------- src/Tools/dotnet-dump/Program.cs | 23 +++++-- 2 files changed, 69 insertions(+), 69 deletions(-) diff --git a/src/Tools/dotnet-dump/Dumper.cs b/src/Tools/dotnet-dump/Dumper.cs index badcdf036e..058b28d201 100644 --- a/src/Tools/dotnet-dump/Dumper.cs +++ b/src/Tools/dotnet-dump/Dumper.cs @@ -28,112 +28,97 @@ public enum DumpTypeOption Triage // A small dump containing module lists, thread lists, exception information, all stacks and PII removed. } - public Dumper() + internal static int Collect(DumpCollectionConfig config, IConsole console) { - } - - public int Collect(IConsole console, int processId, string output, bool diag, bool crashreport, DumpTypeOption type, string name) - { - Console.WriteLine(name); - if (name != null) + if (config.ProcessName is not null) { - if (processId != 0) + if (config.ProcessId != 0) { - Console.WriteLine("Can only specify either --name or --process-id option."); + console.Error.WriteLine("Can only specify either --name or --process-id option."); return -1; } - processId = CommandUtils.FindProcessIdWithName(name); - if (processId < 0) + + config.ProcessId = CommandUtils.FindProcessIdWithName(config.ProcessName); + + if (config.ProcessId < 0) { return -1; } } - if (processId == 0) + if (config.ProcessId == 0) { - Console.Error.WriteLine("ProcessId is required."); + console.Error.WriteLine("ProcessId is required."); return -1; } - if (processId < 0) + if (config.ProcessId < 0) { - Console.Error.WriteLine($"The PID cannot be negative: {processId}"); + console.Error.WriteLine($"The PID cannot be negative: {config.ProcessId}"); return -1; } try { - if (output == null) + if (config.DumpOutputPath is null) { // Build timestamp based file path string timestamp = $"{DateTime.Now:yyyyMMdd_HHmmss}"; - output = Path.Combine(Directory.GetCurrentDirectory(), RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? $"dump_{timestamp}.dmp" : $"core_{timestamp}"); + config.DumpOutputPath = Path.Combine(Directory.GetCurrentDirectory(), RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? $"dump_{timestamp}.dmp" : $"core_{timestamp}"); } - // Make sure the dump path is NOT relative. This path could be sent to the runtime - // process on Linux which may have a different current directory. - output = Path.GetFullPath(output); - - // Display the type of dump and dump path - string dumpTypeMessage = null; - switch (type) + else { - case DumpTypeOption.Full: - dumpTypeMessage = "full"; - break; - case DumpTypeOption.Heap: - dumpTypeMessage = "dump with heap"; - break; - case DumpTypeOption.Mini: - dumpTypeMessage = "dump"; - break; - case DumpTypeOption.Triage: - dumpTypeMessage = "triage dump"; - break; + // Make sure the dump path is NOT relative. This path could be sent to the runtime + // process on Linux which may have a different current directory. + config.DumpOutputPath = Path.GetFullPath(config.DumpOutputPath); } - console.Out.WriteLine($"Writing {dumpTypeMessage} to {output}"); + + string dumpTypeMessage = config.DumpType switch + { + DumpTypeOption.Full => "full", + DumpTypeOption.Heap => "dump with heap", + DumpTypeOption.Mini => "dump", + DumpTypeOption.Triage => "triage dump", + _ => throw new ArgumentException("Invalid dump type.") + }; + + console.Out.WriteLine($"Writing {dumpTypeMessage} to {config.DumpOutputPath}"); if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) { - if (crashreport) + if (config.GenerateCrashReport) { - Console.WriteLine("Crash reports not supported on Windows."); + console.Error.WriteLine("Crash reports not supported on Windows."); return -1; } - Windows.CollectDump(processId, output, type); + if (config.EnableDiagnosticOutput) + { + console.Error.WriteLine("Verbose diagnostics not supported on Windows."); + return -1; + } + + Windows.CollectDump(config.ProcessId, config.DumpOutputPath, config.DumpType); } else { - DiagnosticsClient client = new(processId); + DiagnosticsClient client = new(config.ProcessId); - DumpType dumpType = DumpType.Normal; - switch (type) + DumpType dumpType = config.DumpType switch { - case DumpTypeOption.Full: - dumpType = DumpType.Full; - break; - case DumpTypeOption.Heap: - dumpType = DumpType.WithHeap; - break; - case DumpTypeOption.Mini: - dumpType = DumpType.Normal; - break; - case DumpTypeOption.Triage: - dumpType = DumpType.Triage; - break; - } + DumpTypeOption.Full => DumpType.Full, + DumpTypeOption.Heap => DumpType.WithHeap, + DumpTypeOption.Mini => dumpType = DumpType.Normal, + DumpTypeOption.Triage => DumpType.Triage, + _ => throw new ArgumentException("Invalid dump type.") + }; WriteDumpFlags flags = WriteDumpFlags.None; - if (diag) - { - flags |= WriteDumpFlags.LoggingEnabled; - } - if (crashreport) - { - flags |= WriteDumpFlags.CrashReportEnabled; - } + if (config.EnableDiagnosticOutput) { flags |= WriteDumpFlags.LoggingEnabled; } + if (config.GenerateCrashReport) { flags |= WriteDumpFlags.CrashReportEnabled; } + // Send the command to the runtime to initiate the core dump - client.WriteDump(dumpType, output, flags); + client.WriteDump(dumpType, config.DumpOutputPath, flags); } } catch (Exception ex) when diff --git a/src/Tools/dotnet-dump/Program.cs b/src/Tools/dotnet-dump/Program.cs index ec1facfb43..640605d936 100644 --- a/src/Tools/dotnet-dump/Program.cs +++ b/src/Tools/dotnet-dump/Program.cs @@ -13,6 +13,15 @@ namespace Microsoft.Diagnostics.Tools.Dump { + // Make sure the name of the fields match the option names. + internal record struct DumpCollectionConfig( + int ProcessId, + string ProcessName, + string DumpOutputPath, + bool EnableDiagnosticOutput, + bool GenerateCrashReport, + Dumper.DumpTypeOption DumpType); + internal static class Program { public static Task Main(string[] args) @@ -31,9 +40,9 @@ private static Command CollectCommand() => new(name: "collect", description: "Capture dumps from a process") { // Handler - CommandHandler.Create(new Dumper().Collect), + CommandHandler.Create(Dumper.Collect), // Options - ProcessIdOption(), OutputOption(), DiagnosticLoggingOption(), CrashReportOption(), TypeOption(), ProcessNameOption() + ProcessIdOption(), ProcessNameOption(), OutputOption(), DiagnosticLoggingOption(), CrashReportOption(), TypeOption() }; private static Option ProcessIdOption() => @@ -41,6 +50,7 @@ private static Option ProcessIdOption() => aliases: new[] { "-p", "--process-id" }, description: "The process id to collect a memory dump.") { + Name = nameof(DumpCollectionConfig.ProcessId), Argument = new Argument(name: "pid") }; @@ -49,15 +59,17 @@ private static Option ProcessNameOption() => aliases: new[] { "-n", "--name" }, description: "The name of the process to collect a memory dump.") { - Argument = new Argument(name: "name") + Name = nameof(DumpCollectionConfig.ProcessName), + Argument = new Argument(name: "ProcessName") }; private static Option OutputOption() => new( aliases: new[] { "-o", "--output" }, - description: @"The path where collected dumps should be written. Defaults to '.\dump_YYYYMMDD_HHMMSS.dmp' on Windows and './core_YYYYMMDD_HHMMSS' + description: @"The path where collected dumps should be written. Defaults to '.\dump_YYYYMMDD_HHMMSS.dmp' on Windows and './core_YYYYMMDD_HHMMSS' on Linux where YYYYMMDD is Year/Month/Day and HHMMSS is Hour/Minute/Second. Otherwise, it is the full path and file name of the dump.") { + Name = nameof(DumpCollectionConfig.DumpOutputPath), Argument = new Argument(name: "output_dump_path") }; @@ -66,6 +78,7 @@ private static Option DiagnosticLoggingOption() => alias: "--diag", description: "Enable dump collection diagnostic logging.") { + Name = nameof(DumpCollectionConfig.EnableDiagnosticOutput), Argument = new Argument(name: "diag") }; @@ -74,6 +87,7 @@ private static Option CrashReportOption() => alias: "--crashreport", description: "Enable crash report generation.") { + Name = nameof(DumpCollectionConfig.GenerateCrashReport), Argument = new Argument(name: "crashreport") }; @@ -82,6 +96,7 @@ private static Option TypeOption() => alias: "--type", description: @"The dump type determines the kinds of information that are collected from the process. There are several types: Full - The largest dump containing all memory including the module images. Heap - A large and relatively comprehensive dump containing module lists, thread lists, all stacks, exception information, handle information, and all memory except for mapped images. Mini - A small dump containing module lists, thread lists, exception information and all stacks. Triage - A small dump containing module lists, thread lists, exception information, all stacks and PII removed.") { + Name = nameof(DumpCollectionConfig.DumpType), Argument = new Argument(name: "dump_type", getDefaultValue: () => Dumper.DumpTypeOption.Full) }; From 365bc24fb52d5d823df091aa0dac6ccde3f85d3c Mon Sep 17 00:00:00 2001 From: Juan Sebastian Hoyos Ayala Date: Fri, 17 Mar 2023 02:41:48 -0700 Subject: [PATCH 2/6] Enable dotnet-dump target custom diagnostic port --- ...icrosoft.Diagnostics.NETCore.Client.csproj | 1 + src/Tools/dotnet-dump/Dumper.cs | 63 ++++++++++--------- src/Tools/dotnet-dump/Program.cs | 13 +++- src/Tools/dotnet-dump/dotnet-dump.csproj | 3 +- 4 files changed, 49 insertions(+), 31 deletions(-) diff --git a/src/Microsoft.Diagnostics.NETCore.Client/Microsoft.Diagnostics.NETCore.Client.csproj b/src/Microsoft.Diagnostics.NETCore.Client/Microsoft.Diagnostics.NETCore.Client.csproj index de97cad6fd..2339c28393 100644 --- a/src/Microsoft.Diagnostics.NETCore.Client/Microsoft.Diagnostics.NETCore.Client.csproj +++ b/src/Microsoft.Diagnostics.NETCore.Client/Microsoft.Diagnostics.NETCore.Client.csproj @@ -29,6 +29,7 @@ + diff --git a/src/Tools/dotnet-dump/Dumper.cs b/src/Tools/dotnet-dump/Dumper.cs index 058b28d201..d7faefa0f2 100644 --- a/src/Tools/dotnet-dump/Dumper.cs +++ b/src/Tools/dotnet-dump/Dumper.cs @@ -6,6 +6,8 @@ using System.CommandLine.IO; using System.IO; using System.Runtime.InteropServices; +using System.Threading; +using System.Threading.Tasks; using Microsoft.Diagnostics.NETCore.Client; using Microsoft.Internal.Common.Utils; @@ -30,31 +32,8 @@ public enum DumpTypeOption internal static int Collect(DumpCollectionConfig config, IConsole console) { - if (config.ProcessName is not null) + if (!CommandUtils.ValidateArgumentsForAttach(config.ProcessId, config.ProcessName, config.DiagnosticPort, out int targetPprocessId)) { - if (config.ProcessId != 0) - { - console.Error.WriteLine("Can only specify either --name or --process-id option."); - return -1; - } - - config.ProcessId = CommandUtils.FindProcessIdWithName(config.ProcessName); - - if (config.ProcessId < 0) - { - return -1; - } - } - - if (config.ProcessId == 0) - { - console.Error.WriteLine("ProcessId is required."); - return -1; - } - - if (config.ProcessId < 0) - { - console.Error.WriteLine($"The PID cannot be negative: {config.ProcessId}"); return -1; } @@ -98,12 +77,10 @@ internal static int Collect(DumpCollectionConfig config, IConsole console) return -1; } - Windows.CollectDump(config.ProcessId, config.DumpOutputPath, config.DumpType); + Windows.CollectDump(targetPprocessId, config.DumpOutputPath, config.DumpType); } else { - DiagnosticsClient client = new(config.ProcessId); - DumpType dumpType = config.DumpType switch { DumpTypeOption.Full => DumpType.Full, @@ -113,9 +90,33 @@ internal static int Collect(DumpCollectionConfig config, IConsole console) _ => throw new ArgumentException("Invalid dump type.") }; + DiagnosticsClient client; + if (!string.IsNullOrEmpty(config.DiagnosticPort)) + { + IpcEndpointConfig portConfig = IpcEndpointConfig.Parse(config.DiagnosticPort); + if (portConfig.IsListenConfig) + { + console.Error.WriteLine("dotnet-dump only supports connect mode to a runtime."); + return -1; + } + + client = new DiagnosticsClient(portConfig); + } + else + { + client = new DiagnosticsClient(targetPprocessId); + } + WriteDumpFlags flags = WriteDumpFlags.None; - if (config.EnableDiagnosticOutput) { flags |= WriteDumpFlags.LoggingEnabled; } - if (config.GenerateCrashReport) { flags |= WriteDumpFlags.CrashReportEnabled; } + if (config.EnableDiagnosticOutput) + { + console.Out.WriteLine("Diagnostic output requested. Logging will appear in the console of the target process."); + flags |= WriteDumpFlags.LoggingEnabled; + } + if (config.GenerateCrashReport) + { + flags |= WriteDumpFlags.CrashReportEnabled; + } // Send the command to the runtime to initiate the core dump client.WriteDump(dumpType, config.DumpOutputPath, flags); @@ -134,6 +135,10 @@ NotSupportedException or DiagnosticsClientException) { console.Error.WriteLine($"{ex.Message}"); + if (!config.EnableDiagnosticOutput) + { + console.Error.WriteLine($"Consider rerunning the command with diagnostic output enabled."); + } return -1; } diff --git a/src/Tools/dotnet-dump/Program.cs b/src/Tools/dotnet-dump/Program.cs index 640605d936..a13292f0ea 100644 --- a/src/Tools/dotnet-dump/Program.cs +++ b/src/Tools/dotnet-dump/Program.cs @@ -7,6 +7,7 @@ using System.CommandLine.Invocation; using System.CommandLine.Parsing; using System.IO; +using System.Threading; using System.Threading.Tasks; using Microsoft.Internal.Common.Commands; using Microsoft.Tools.Common; @@ -17,6 +18,7 @@ namespace Microsoft.Diagnostics.Tools.Dump internal record struct DumpCollectionConfig( int ProcessId, string ProcessName, + string DiagnosticPort, string DumpOutputPath, bool EnableDiagnosticOutput, bool GenerateCrashReport, @@ -42,7 +44,7 @@ private static Command CollectCommand() => // Handler CommandHandler.Create(Dumper.Collect), // Options - ProcessIdOption(), ProcessNameOption(), OutputOption(), DiagnosticLoggingOption(), CrashReportOption(), TypeOption() + ProcessIdOption(), ProcessNameOption(), DiagnosticPortOption(), OutputOption(), DiagnosticLoggingOption(), CrashReportOption(), TypeOption() }; private static Option ProcessIdOption() => @@ -63,6 +65,15 @@ private static Option ProcessNameOption() => Argument = new Argument(name: "ProcessName") }; + private static Option DiagnosticPortOption() => + new( + alias: "--diagnostic-port", + description: @"The path to a diagnostic port to be used.") + { + Name = nameof(DumpCollectionConfig.DiagnosticPort), + Argument = new Argument(name: "diagnosticPort") + }; + private static Option OutputOption() => new( aliases: new[] { "-o", "--output" }, diff --git a/src/Tools/dotnet-dump/dotnet-dump.csproj b/src/Tools/dotnet-dump/dotnet-dump.csproj index 1699965407..3603b0d636 100644 --- a/src/Tools/dotnet-dump/dotnet-dump.csproj +++ b/src/Tools/dotnet-dump/dotnet-dump.csproj @@ -27,8 +27,9 @@ + - + From 500a99c97dec5973cd631655b845a2c4e155ce1a Mon Sep 17 00:00:00 2001 From: Juan Sebastian Hoyos Ayala Date: Fri, 12 May 2023 13:45:53 -0700 Subject: [PATCH 3/6] Lint and update the IPC spec to include CreateCoreDump2 and CreateCoreDump3 --- documentation/design-docs/ipc-protocol.md | 263 ++++++++++++++++------ 1 file changed, 200 insertions(+), 63 deletions(-) diff --git a/documentation/design-docs/ipc-protocol.md b/documentation/design-docs/ipc-protocol.md index 24fb0e5ec3..5c2619b6f2 100644 --- a/documentation/design-docs/ipc-protocol.md +++ b/documentation/design-docs/ipc-protocol.md @@ -4,10 +4,10 @@ This spec describes the IPC Protocol to be used for communicating with the dotnet core runtime's Diagnostics Server from an external client over a platform-specific transport, e.g., Unix Domain Sockets. - ### Terminology The protocol will use the following names for various constructs and behaviors defined in this spec: + * *Diagnostic IPC Protocol*: The protocol defined in this spec * *Diagnostic Server*: The server in the runtime that receives/sends Diagnostic IPC Procotol communication. * *Commands*: The functionality being invoked in the runtime that communicates over the Diagnostic IPC Protocol, e.g., "Start an EventPipe stream". These are encoded as a `command_set` and a `command_id`. @@ -20,7 +20,7 @@ The protocol will use the following names for various constructs and behaviors d ## General Flow -All communication with the Diagnostic Server will begin with a Diagnostic IPC Message sent from the client to the server. The server will respond with a Diagnostic IPC Message. After this, the client and runtime _may_ reuse the Pipe for any Command specific communication which is referred to as an Optional Continuation. +All communication with the Diagnostic Server will begin with a Diagnostic IPC Message sent from the client to the server. The server will respond with a Diagnostic IPC Message. After this, the client and runtime *may* reuse the Pipe for any Command specific communication which is referred to as an Optional Continuation. ![Generic Flow](ipc-protocol-genericflow.svg) @@ -33,6 +33,7 @@ connection closed ``` Example flow for EventPipe: + ``` runtime <- client : [ magic; size; EventPipe CollectTracing ][ stream config struct ] <- Diagnostic IPC Message runtime -> client : [ magic; size; Server OK ][ sessionId ] <- Diagnostic IPC Message @@ -47,11 +48,12 @@ connection closed The protocol will be communicated over a platform-specific transport. On Unix/Linux based platforms, a Unix Domain Socket will be used, and on Windows, a Named Pipe will be used. -#### Naming and Location Conventions +### Naming and Location Conventions Unix Domain Sockets (MacOS and *nix): The socket is placed in one of two places: + 1. The directory specified in `$TMPDIR` 2. `/tmp` if `$TMPDIR` is undefined/empty @@ -60,11 +62,13 @@ In order to ensure filename uniqueness, a `disambiguation key` is generated. On > NOTE: If the target application is running inside an application sandbox on MacOS, the transport will be placed in the Application Group container directory. This is a convention for all sandboxed applications on MacOS. socket name: + ```c dotnet-diagnostic-{%d:PID}-{%llu:disambiguation key}-socket ``` Named Pipes (Windows): + ``` \\.\pipe\dotnet-diagnostic-{%d:PID} ``` @@ -162,6 +166,7 @@ For example, this IPC Message is the generic OK message which has an empty Paylo ### Headers Every Diagnostic IPC Message will start with a header and every header will: + * start with a magic version number and a size * `sizeof(IpcHeader) == 20` * encode numbers little-endian @@ -181,12 +186,11 @@ struct IpcHeader The `reserved` field is reserved for future use. It is unused in `DOTNET_IPC_V1` and must be 0x0000. - ### Payloads -Payloads are Command specific data encoded into a Diagnostic IPC Message. The size of the payload is implicitly encoded in the Header's `size` field as `PayloadSize = header.size - sizeof(struct IpcHeader)`. A Payload _may_ be 0 bytes long if it empty. The encoding of data in the Payload is Command specific. +Payloads are Command specific data encoded into a Diagnostic IPC Message. The size of the payload is implicitly encoded in the Header's `size` field as `PayloadSize = header.size - sizeof(struct IpcHeader)`. A Payload *may* be 0 bytes long if it empty. The encoding of data in the Payload is Command specific. -Payloads are either encoded as fixed size structures that can be `memcpy`'ed , _or_: +Payloads are either encoded as fixed size structures that can be `memcpy`'ed , *or*: * `X, Y, Z` means encode bytes for `X` followed by bytes for `Y` followed by bytes for `Z` * `uint` = 4 little endian bytes @@ -351,17 +355,22 @@ enum class EventPipeCommandId : uint8_t CollectTracing2 = 0x03, // create/start a given session with/without rundown } ``` -See: [EventPipe Commands](#EventPipe-Commands) + +See: [EventPipe Commands](#eventpipe-commands) ```c++ enum class DumpCommandId : uint8_t { // reserved = 0x00, CreateCoreDump = 0x01, + CreateCoreDump2 = 0x02, + CreateCoreDump3 = 0x03, + CreateCoreDump4 = 0x04, // future } ``` -See: [Dump Commands](#Dump-Commands) + +See: [Dump Commands](#dump-commands) ```c++ enum class ProfilerCommandId : uint8_t @@ -370,8 +379,9 @@ enum class ProfilerCommandId : uint8_t AttachProfiler = 0x01, // future } -``` -See: [Profiler Commands](#Profiler-Commands) +``` + +See: [Profiler Commands](#profiler-commands) ```c++ enum class ProcessCommandId : uint8_t @@ -383,7 +393,8 @@ enum class ProcessCommandId : uint8_t // future } ``` -See: [Process Commands](#Process-Commands) + +See: [Process Commands](#process-commands) Commands may use the generic `{ magic="DOTNET_IPC_V1"; size=20; command_set=0xFF (Server); command_id=0x00 (OK); reserved = 0x0000; }` to indicate success rather than having a command specific success `command_id`. @@ -400,6 +411,7 @@ enum class EventPipeCommandId : uint8_t CollectTracing2 = 0x03, // create/start a given session with/without rundown } ``` + EventPipe Payloads are encoded with the following rules: * `X, Y, Z` means encode bytes for `X` followed by bytes for `Y` followed by bytes for `Z` @@ -416,13 +428,13 @@ Command Code: `0x0202` The `CollectTracing` Command is used to start a streaming session of event data. The runtime will attempt to start a session and respond with a success message with a payload of the `sessionId`. The event data is streamed in the `nettrace` format. The stream begins after the response Message from the runtime to the client. The client is expected to continue to listen on the transport until the connection is closed. -In the event there is an [error](#Errors), the runtime will attempt to send an error message and subsequently close the connection. +In the event there is an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. -The client is expected to send a [`StopTracing`](#StopTracing) command to the runtime in order to stop the stream, as there is a "run down" at the end of a stream session that transmits additional metadata. +The client is expected to send a [`StopTracing`](#stoptracing) command to the runtime in order to stop the stream, as there is a "run down" at the end of a stream session that transmits additional metadata. If the stream is stopped prematurely due to a client or server error, the `nettrace` file generated will be incomplete and should be considered corrupted. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0202; 0x0000 }` @@ -431,6 +443,7 @@ Header: `{ Magic; Size; 0x0202; 0x0000 }` * `array providers`: The providers to turn on for the streaming session A `provider_config` is composed of the following data: + * `ulong keywords`: The keywords to turn on with this providers * `uint logLevel`: The level of information to turn on * `string provider_name`: The name of the provider @@ -438,16 +451,18 @@ A `provider_config` is composed of the following data: > see ETW documentation for a more detailed explanation of Keywords, Filters, and Log Level. -#### Returns (as an IPC Message Payload): +#### Returns (as an IPC Message Payload) Header: `{ Magic; 28; 0xFF00; 0x0000; }` `CollectTracing` returns: + * `ulong sessionId`: the ID for the stream session starting on the current connection -##### Details: +##### Details Input: + ``` Payload { @@ -456,7 +471,7 @@ Payload array providers } -provider_config +provider_config { ulong keywords, uint logLevel, @@ -466,12 +481,14 @@ provider_config ``` Returns: + ```c Payload { ulong sessionId } ``` + Followed by an Optional Continuation of a `nettrace` format stream of events. ### `CollectTracing2` @@ -480,7 +497,7 @@ Command Code: `0x0203` The `CollectTracing2` Command is an extension of the `CollectTracing` command - its behavior is the same as `CollectTracing` command, except that it has another field that lets you specify whether rundown events should be fired by the runtime. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0203; 0x0000 }` @@ -490,23 +507,26 @@ Header: `{ Magic; Size; 0x0203; 0x0000 }` * `array providers`: The providers to turn on for the streaming session A `provider_config` is composed of the following data: + * `ulong keywords`: The keywords to turn on with this providers * `uint logLevel`: The level of information to turn on * `string provider_name`: The name of the provider * `string filter_data` (optional): Filter information > see ETW documentation for a more detailed explanation of Keywords, Filters, and Log Level. -> -#### Returns (as an IPC Message Payload): +> +#### Returns (as an IPC Message Payload) Header: `{ Magic; 28; 0xFF00; 0x0000; }` `CollectTracing2` returns: + * `ulong sessionId`: the ID for the stream session starting on the current connection -##### Details: +##### Details Input: + ``` Payload { @@ -516,7 +536,7 @@ Payload array providers } -provider_config +provider_config { ulong keywords, uint logLevel, @@ -526,36 +546,38 @@ provider_config ``` Returns: + ```c Payload { ulong sessionId } ``` + Followed by an Optional Continuation of a `nettrace` format stream of events. -### `StopTracing` +### `StopTracing` Command Code: `0x0201` -The `StopTracing` command is used to stop a specific streaming session. Clients are expected to use this command to stop streaming sessions started with [`CollectStreaming`](#CollectStreaming). +The `StopTracing` command is used to stop a specific streaming session. Clients are expected to use this command to stop streaming sessions started with [`CollectTracing`](#collecttracing) or [`CollectTracing2`](#collecttracing2). -#### Inputs: +#### Inputs Header: `{ Magic; 28; 0x0201; 0x0000 }` * `ulong sessionId`: The ID for the streaming session to stop -#### Returns: +#### Returns Header: `{ Magic; 28; 0xFF00; 0x0000 }` * `ulong sessionId`: the ID for the streaming session that was stopped - -##### Details: +##### Details Inputs: + ```c Payload { @@ -564,6 +586,7 @@ Payload ``` Returns: + ```c Payload { @@ -579,9 +602,9 @@ Command Code: `0x0101` The `CreateCoreDump` command is used to instruct the runtime to generate a core dump of the process. The command will keep the connection open while the dump is generated and then respond with a message containing an `HRESULT` indicating success or failure. -In the event of an [error](#Errors), the runtime will attempt to send an error message and subsequently close the connection. +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0101; 0x0000 }` @@ -594,16 +617,18 @@ Header: `{ Magic; Size; 0x0101; 0x0000 }` * `uint diagnostics`: If set to 1, log to console the dump generation diagnostics * `0` or `1` for on or off -#### Returns (as an IPC Message Payload): +#### Returns (as an IPC Message Payload) Header: `{ Magic; 28; 0xFF00; 0x0000; }` `CreateCoreDump` returns: + * `int32 hresult`: The result of creating the core dump (`0` indicates success) -##### Details: +##### Details Input: + ``` Payload { @@ -614,6 +639,56 @@ Payload ``` Returns: + +```c +Payload +{ + int32 hresult +} +``` + +### `CreateCoreDump2` + +Command Code: `0x0102` + +The `CreateCoreDump2` command is used to instruct the runtime to generate a core dump of the process, much like `CreateCoreDump`. The main difference is the the v2 command accepts a set of flags instead of the boolean diagnostics field used in *v1* to describe additional diagnostic artifacts. + +#### Inputs + +Header: `{ Magic; Size; 0x0102; 0x0000 }` + +* `string dumpName`: As described in [`CreateCoreDump`](#createcoredump). +* `uint dumpType`: As described in [`CreateCoreDump`](#createcoredump). +* `uint flags`: Flags as defined by the following: + + * `GenerateDumpFlagsNone = 0x00`: No diagnostics assets are created. + * `GenerateDumpFlagsLoggingEnabled = 0x01`: General logging enabled - streamed to the console of the target process. + * `GenerateDumpFlagsVerboseLoggingEnabled = 0x02`: Verbose logging enabled - streamed to the console of the target process. + * `GenerateDumpFlagsCrashReportEnabled = 0x04`: Generate crashdump report next to the generated dump. + +#### Returns (as an IPC Message Payload) + +Header: `{ Magic; 28; 0xFF00; 0x0000; }` + +`CreateCoreDump2` returns: + +* `int32 hresult`: The result of creating the core dump (`0` indicates success) + +##### Details + +Input: + +``` +Payload +{ + string dumpName, + uint dumpType, + uint flags +} +``` + +Returns: + ```c Payload { @@ -621,6 +696,53 @@ Payload } ``` +### `CreateCoreDump3` + +Command Code: `0x0103` + +The `CreateCoreDump3` command is used to instruct the runtime to generate a core dump of the process, with inputs identical to `CreateCoreDump2`. The main difference is the response stream may contain an error string that can be propagated to clients. + +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. + +#### Inputs + +Header: `{ Magic; Size; 0x0103; 0x0000 }` + +All payload fields are identical to `CreateCoreDump2` + +#### Returns (as an IPC Message Payload) + +Header: `{ Magic; Size; 0xFF00; 0x0000; }` + +`CreateCoreDump3` returns: + +* `int32 hresult`: The result of creating the core dump (`0` indicates success). +* `string error`: Optionally the payload may have an error describing the collection issues. + +##### Details + + +Input: + +``` +Payload +{ + string dumpName, + uint dumpType, + uint flags +} +``` + +Returns: + +```c +Payload +{ + int32 hresult + string errorString +} +``` + ## Profiler Commands ### `AttachProfiler` @@ -629,9 +751,9 @@ Command Code: `0x0301` The `AttachProfiler` command is used to attach a profiler to the runtime. The command will keep the connection open while the profiler is being attached and then respond with a message containing an `HRESULT` indicating success or failure. -In the event of an [error](#Errors), the runtime will attempt to send an error message and subsequently close the connection. +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0301; 0x0000 }` @@ -641,21 +763,24 @@ Header: `{ Magic; Size; 0x0301; 0x0000 }` * `array clientData`: The data being provided to the profiler Where a `CLSID` is a fixed size struct consisting of: + * `uint x` * `byte s1` * `byte s2` * `byte[8] c` -#### Returns (as an IPC Message Payload): +#### Returns (as an IPC Message Payload) Header: `{ Magic; 28; 0xFF00; 0x0000; }` `AttachProfiler` returns: + * `int32 hresult`: The result of attaching the profiler (`0` indicates success) -##### Details: +##### Details Input: + ``` Payload { @@ -668,6 +793,7 @@ Payload ``` Returns: + ```c Payload { @@ -685,19 +811,20 @@ Command Code: `0x0400` The `ProcessInfo` command queries the runtime for some basic information about the process. -In the event of an [error](#Errors), the runtime will attempt to send an error message and subsequently close the connection. +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0400; 0x0000 }` There is no payload. -#### Returns (as an IPC Message Payload): +#### Returns (as an IPC Message Payload) Header: `{ Magic; size; 0xFF00; 0x0000; }` Payload: + * `int64 processId`: the process id in the process's PID-space * `GUID runtimeCookie`: a 128-bit GUID that should be unique across PID-spaces * `string commandLine`: the command line that invoked the process @@ -715,9 +842,10 @@ Payload: * ARM64 => `"arm64"` * Other => `"Unknown"` -##### Details: +##### Details Returns: + ```c++ struct Payload { @@ -735,17 +863,17 @@ Command Code: `0x0401` If the target .NET application has been configured Diagnostic Ports configured to suspend with `DOTNET_DiagnosticPorts` or `DOTNET_DefaultDiagnosticPortSuspend` has been set to `1` (`0` is the default value), then the runtime will pause during `EEStartupHelper` in `ceemain.cpp` and wait for an event to be set. (See [Diagnostic Ports](#diagnostic-ports) for more details) -The `ResumeRuntime` command sets the necessary event to resume runtime startup. If the .NET application _has not_ been configured to with Diagnostics Monitor Address or the runtime has _already_ been resumed, this command is a no-op. +The `ResumeRuntime` command sets the necessary event to resume runtime startup. If the .NET application *has not* been configured to with Diagnostics Monitor Address or the runtime has *already* been resumed, this command is a no-op. -In the event of an [error](#Errors), the runtime will attempt to send an error message and subsequently close the connection. +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0401; 0x0000 }` There is no payload. -#### Returns (as an IPC Message Payload): +#### Returns (as an IPC Message Payload) Header: `{ Magic; size; 0xFF00; 0x0000; }` @@ -757,30 +885,33 @@ Command Code: `0x0402` The `ProcessEnvironment` command queries the runtime for its environment block. -In the event of an [error](#Errors), the runtime will attempt to send an error message and subsequently close the connection. +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0402; 0x0000 }` There is no payload. -#### Returns (as an IPC Message Payload + continuation): +#### Returns (as an IPC Message Payload + continuation) Header: `{ Magic; size; 0xFF00; 0x0000; }` Payload: + * `uint32_t nIncomingBytes`: the number of bytes to expect in the continuation stream * `uint16_t future`: unused Continuation: + * `Array> environmentBlock`: The environment block written as a length prefixed array of length prefixed arrays of `WCHAR`. Note: it is valid for `nIncomingBytes` to be `4` and the continuation to simply contain the value `0`. -##### Details: +##### Details Returns: + ```c++ struct Payload { @@ -797,19 +928,20 @@ Command Code: `0x0404` The `ProcessInfo2` command queries the runtime for some basic information about the process. The returned payload has the same information as that of the `ProcessInfo` command in addition to the managed entrypoint assembly name and CLR product version. -In the event of an [error](#Errors), the runtime will attempt to send an error message and subsequently close the connection. +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. -#### Inputs: +#### Inputs Header: `{ Magic; Size; 0x0402; 0x0000 }` There is no payload. -#### Returns (as an IPC Message Payload): +#### Returns (as an IPC Message Payload) Header: `{ Magic; size; 0xFF00; 0x0000; }` Payload: + * `int64 processId`: the process id in the process's PID-space * `GUID runtimeCookie`: a 128-bit GUID that should be unique across PID-spaces * `string commandLine`: the command line that invoked the process @@ -829,9 +961,10 @@ Payload: * `string managedEntrypointAssemblyName`: the assembly name from the assembly identity of the entrypoint assembly of the process. This is the same value that is returned from executing `System.Reflection.Assembly.GetEntryAssembly().GetName().Name` in the target process. * `string clrProductVersion`: the product version of the CLR of the process; may contain prerelease label information e.g. `6.0.0-preview.6.#####` -##### Details: +##### Details Returns: + ```c++ struct Payload { @@ -850,6 +983,7 @@ struct Payload In the event an error occurs in the handling of an Ipc Message, the Diagnostic Server will attempt to send an Ipc Message encoding the error and subsequently close the connection. The connection will be closed **regardless** of the success of sending the error message. The Client is expected to be resilient in the event of a connection being abruptly closed. Errors are `HRESULTS` encoded as `int32_t` when sent back to the client. There are a few Diagnostics IPC specific `HRESULT`s: + ```c #define CORDIAGIPC_E_BAD_ENCODING = 0x80131384 #define CORDIAGIPC_E_UNKNOWN_COMMAND = 0x80131385 @@ -858,6 +992,7 @@ Errors are `HRESULTS` encoded as `int32_t` when sent back to the client. There ``` Diagnostic Server errors are sent as a Diagnostic IPC Message with: + * a `command_set` of `0xFF` * a `command_id` of `0xFF` * a Payload consisting of a `int32_t` representing the error encountered (described above) @@ -865,6 +1000,7 @@ Diagnostic Server errors are sent as a Diagnostic IPC Message with: All errors will result in the Server closing the connection. Error response Messages will be sent when: + * the client sends an improperly encoded Diagnostic IPC Message * the client uses an unknown `command` * the client uses an unknown `magic` version string @@ -927,7 +1063,7 @@ For example, if the Diagnostic Server finds incorrectly encoded data while parsi -# Diagnostic Ports +## Diagnostic Ports > Available since .NET 5.0 @@ -935,7 +1071,7 @@ A Diagnostic Port is a mechanism for communicating the Diagnostics IPC Protocol .NET applications can configure Diagnostic Ports with the following environment variables: - * `DOTNET_DiagnosticPorts=[,tag[...]][;[,tag[...]][...]]` +> `DOTNET_DiagnosticPorts=[,tag[...]][;[,tag[...]][...]]` where: @@ -947,7 +1083,7 @@ where: Example usage: ```shell -$ export DOTNET_DiagnosticPorts=$DOTNET_DiagnosticPorts;~/mydiagport.sock,nosuspend; +export DOTNET_DiagnosticPorts=$DOTNET_DiagnosticPorts;~/mydiagport.sock,nosuspend; ``` Any diagnostic ports specified in this configuration will be created in addition to the default port (`dotnet-diagnostic--`). The suspend mode of the default port is set via the new environment variable `DOTNET_DefaultDiagnosticPortSuspend` which defaults to `0` for `nosuspend`. @@ -962,14 +1098,14 @@ The runtime will make a best effort attempt to generate a port from a port confi When a Diagnostic Port is configured, the runtime will attempt to connect to the provided address in a retry loop while also listening on the traditional server. The retry loop has an initial timeout of 10ms with a falloff factor of 1.25x and a max timeout of 500 ms. A successful connection will result in an infinite timeout. The runtime is resilient to the remote end of the Diagnostic Port failing, e.g., closing, not `Accepting`, etc. -## Advertise Protocol +### Advertise Protocol Upon successful connection, the runtime will send a fixed-size, 34 byte buffer containing the following information: - * `char[8] magic`: (8 bytes) `"ADVR_V1\0"` (ASCII chars + null byte) - * `GUID runtimeCookie`: (16 bytes) CLR Instance Cookie (little-endian) - * `uint64_t processId`: (8 bytes) PID (little-endian) - * `uint16_t future`: (2 bytes) unused for future-proofing +* `char[8] magic`: (8 bytes) `"ADVR_V1\0"` (ASCII chars + null byte) +* `GUID runtimeCookie`: (16 bytes) CLR Instance Cookie (little-endian) +* `uint64_t processId`: (8 bytes) PID (little-endian) +* `uint16_t future`: (2 bytes) unused for future-proofing With the following layout: @@ -1026,11 +1162,12 @@ With the following layout: This is a one-way transmission with no expectation of an ACK. The tool owning the Diagnostic Port is expected to consume this message and then hold on to the now active connection until it chooses to send a Diagnostics IPC command. -## Dataflow +### Dataflow -Due to the potential for an *optional continuation* in the Diagnostics IPC Protocol, each successful connection between the runtime and a Diagnostic Port is only usable **once**. As a result, a .NET process will attempt to _reconnect_ to the diagnostic port immediately after every command that is sent across an active connection. +Due to the potential for an *optional continuation* in the Diagnostics IPC Protocol, each successful connection between the runtime and a Diagnostic Port is only usable **once**. As a result, a .NET process will attempt to *reconnect* to the diagnostic port immediately after every command that is sent across an active connection. A typical dataflow has 2 actors, the Target application, `T` and the Diagnostics Monitor Application, `M`, and communicates like so: + ``` T -> : Target attempts to connect to M, which may not exist yet // M comes into existence @@ -1040,4 +1177,4 @@ T <- M : [ Diagnostics IPC Protocol ] - Monitor sends a Diagnostics IPC Protocol T -> M : [ Advertise ] - Target reconnects to Monitor with a _new_ connection and re-sends the advertise message ``` -It is important to emphasize that a connection **_should not_** be reused for multiple Diagnostic IPC Protocol commands. +It is important to emphasize that a connection ***should not*** be reused for multiple Diagnostic IPC Protocol commands. From 79ec21ed788b354d68f03acb54d2df9530115139 Mon Sep 17 00:00:00 2001 From: Juan Sebastian Hoyos Ayala Date: Fri, 12 May 2023 16:39:52 -0700 Subject: [PATCH 4/6] Add GenerateCoreDump4 --- .../DiagnosticsClient/DiagnosticsClient.cs | 133 ++++++++++++++---- .../DiagnosticsClient/WriteDumpFlags.cs | 6 +- .../DiagnosticsIpc/IpcCommands.cs | 1 + src/Tools/dotnet-dump/dotnet-dump.csproj | 3 +- 4 files changed, 110 insertions(+), 33 deletions(-) diff --git a/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/DiagnosticsClient.cs b/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/DiagnosticsClient.cs index 3168df5e15..d407b0fb1b 100644 --- a/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/DiagnosticsClient.cs +++ b/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/DiagnosticsClient.cs @@ -18,6 +18,7 @@ namespace Microsoft.Diagnostics.NETCore.Client /// public sealed class DiagnosticsClient { + private const string DumpOperationName = "Write dump"; private readonly IpcEndpoint _endpoint; public DiagnosticsClient(int processId) : @@ -129,27 +130,59 @@ public void WriteDump(DumpType dumpType, string dumpPath, bool logDumpGeneration /// Trigger a core dump generation. /// /// Type of the dump to be generated - /// Full path to the dump to be generated. By default it is /tmp/coredump.{pid} + /// Full path to the dump to be generated. /// logging and crash report flags. On runtimes less than 6.0, only LoggingEnabled is supported. public void WriteDump(DumpType dumpType, string dumpPath, WriteDumpFlags flags) { - IpcMessage request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump3, dumpType, dumpPath, flags); + WriteDump(dumpType, dumpPath, flags, logPath: null); + } + + /// + /// Trigger a core dump generation. + /// + /// Type of the dump to be generated + /// Full path to the dump to be generated. + /// logging and crash report flags. On runtimes less than 6.0, only LoggingEnabled is supported. + /// Full path to the log to be generated. If null or empty and flags requests logging - the default behavior is... + public void WriteDump(DumpType dumpType, string dumpPath, WriteDumpFlags flags, string logPath) + { + IpcMessage request = CreateWriteDumpMessageV4(dumpType, dumpPath, flags, logPath); IpcMessage response = IpcClient.SendMessage(_endpoint, request); - if (!ValidateResponseMessage(response, "Write dump", ValidateResponseOptions.UnknownCommandReturnsFalse | ValidateResponseOptions.ErrorMessageReturned)) + + if (ValidateResponseMessage(response, DumpOperationName, ValidateResponseOptions.UnknownCommandReturnsFalse | ValidateResponseOptions.ErrorMessageReturned)) { - request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump2, dumpType, dumpPath, flags); - response = IpcClient.SendMessage(_endpoint, request); - if (!ValidateResponseMessage(response, "Write dump", ValidateResponseOptions.UnknownCommandReturnsFalse)) - { - if ((flags & ~WriteDumpFlags.LoggingEnabled) != 0) - { - throw new ArgumentException($"Only {nameof(WriteDumpFlags.LoggingEnabled)} flag is supported by this runtime version", nameof(flags)); - } - request = CreateWriteDumpMessage(dumpType, dumpPath, logDumpGeneration: (flags & WriteDumpFlags.LoggingEnabled) != 0); - response = IpcClient.SendMessage(_endpoint, request); - ValidateResponseMessage(response, "Write dump"); - } + return; + } + + if ((flags & (WriteDumpFlags.LogToFile | WriteDumpFlags.VerboseLoggingEnabled)) != 0) + { + throw new ArgumentException($"{nameof(WriteDumpFlags.LogToFile)} and {nameof(WriteDumpFlags.VerboseLoggingEnabled)} flags are not supported by the current runtime.", nameof(flags)); + } + + request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump3, dumpType, dumpPath, flags); + response = IpcClient.SendMessage(_endpoint, request); + + if (ValidateResponseMessage(response, DumpOperationName, ValidateResponseOptions.UnknownCommandReturnsFalse | ValidateResponseOptions.ErrorMessageReturned)) + { + return; + } + + request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump2, dumpType, dumpPath, flags); + response = IpcClient.SendMessage(_endpoint, request); + + if (ValidateResponseMessage(response, DumpOperationName, ValidateResponseOptions.UnknownCommandReturnsFalse)) + { + return; + } + + if ((flags & ~WriteDumpFlags.LoggingEnabled) != 0) + { + throw new ArgumentException($"Only {nameof(WriteDumpFlags.LoggingEnabled)} flag is supported by this runtime version", nameof(flags)); } + + request = CreateWriteDumpMessage(dumpType, dumpPath, logDumpGeneration: (flags & WriteDumpFlags.LoggingEnabled) != 0); + response = IpcClient.SendMessage(_endpoint, request); + ValidateResponseMessage(response, DumpOperationName); } /// @@ -171,25 +204,54 @@ public Task WriteDumpAsync(DumpType dumpType, string dumpPath, bool logDumpGener /// Full path to the dump to be generated. By default it is /tmp/coredump.{pid} /// logging and crash report flags. On runtimes less than 6.0, only LoggingEnabled is supported. /// The token to monitor for cancellation requests. - public async Task WriteDumpAsync(DumpType dumpType, string dumpPath, WriteDumpFlags flags, CancellationToken token) + public Task WriteDumpAsync(DumpType dumpType, string dumpPath, WriteDumpFlags flags, CancellationToken token) { - IpcMessage request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump3, dumpType, dumpPath, flags); + return WriteDumpAsync(dumpType, dumpPath, flags, logPath: null, token); + } + + /// + /// Trigger a core dump generation. + /// + /// Type of the dump to be generated + /// Full path to the dump to be generated. By default it is /tmp/coredump.{pid} + /// logging and crash report flags. On runtimes less than 6.0, only LoggingEnabled is supported. + /// Full path to the log to be generated. If null or empty and flags requests logging - the default behavior is... + /// The token to monitor for cancellation requests. + public async Task WriteDumpAsync(DumpType dumpType, string dumpPath, WriteDumpFlags flags, string logPath, CancellationToken token) + { + IpcMessage request = CreateWriteDumpMessageV4(dumpType, dumpPath, flags, logPath); IpcMessage response = await IpcClient.SendMessageAsync(_endpoint, request, token).ConfigureAwait(false); - if (!ValidateResponseMessage(response, "Write dump", ValidateResponseOptions.UnknownCommandReturnsFalse | ValidateResponseOptions.ErrorMessageReturned)) + if (ValidateResponseMessage(response, DumpOperationName, ValidateResponseOptions.UnknownCommandReturnsFalse | ValidateResponseOptions.ErrorMessageReturned)) { - request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump2, dumpType, dumpPath, flags); - response = await IpcClient.SendMessageAsync(_endpoint, request, token).ConfigureAwait(false); - if (!ValidateResponseMessage(response, "Write dump", ValidateResponseOptions.UnknownCommandReturnsFalse)) - { - if ((flags & ~WriteDumpFlags.LoggingEnabled) != 0) - { - throw new ArgumentException($"Only {nameof(WriteDumpFlags.LoggingEnabled)} flag is supported by this runtime version", nameof(flags)); - } - request = CreateWriteDumpMessage(dumpType, dumpPath, logDumpGeneration: (flags & WriteDumpFlags.LoggingEnabled) != 0); - response = await IpcClient.SendMessageAsync(_endpoint, request, token).ConfigureAwait(false); - ValidateResponseMessage(response, "Write dump"); - } + return; + } + + if ((flags & (WriteDumpFlags.LogToFile | WriteDumpFlags.VerboseLoggingEnabled)) != 0) + { + throw new ArgumentException($"{nameof(WriteDumpFlags.LogToFile)} and {nameof(WriteDumpFlags.VerboseLoggingEnabled)} flags are not supported by the current runtime.", nameof(flags)); } + + request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump3, dumpType, dumpPath, flags); + response = await IpcClient.SendMessageAsync(_endpoint, request, token).ConfigureAwait(false); + if (ValidateResponseMessage(response, DumpOperationName, ValidateResponseOptions.UnknownCommandReturnsFalse | ValidateResponseOptions.ErrorMessageReturned)) + { + return; + } + + request = CreateWriteDumpMessage(DumpCommandId.GenerateCoreDump2, dumpType, dumpPath, flags); + response = await IpcClient.SendMessageAsync(_endpoint, request, token).ConfigureAwait(false); + if (ValidateResponseMessage(response, DumpOperationName, ValidateResponseOptions.UnknownCommandReturnsFalse)) + { + return; + } + + if ((flags & ~WriteDumpFlags.LoggingEnabled) != 0) + { + throw new ArgumentException($"Only {nameof(WriteDumpFlags.LoggingEnabled)} flag is supported by this runtime version", nameof(flags)); + } + request = CreateWriteDumpMessage(dumpType, dumpPath, logDumpGeneration: (flags & WriteDumpFlags.LoggingEnabled) != 0); + response = await IpcClient.SendMessageAsync(_endpoint, request, token).ConfigureAwait(false); + ValidateResponseMessage(response, DumpOperationName); } /// @@ -566,6 +628,17 @@ private static IpcMessage CreateWriteDumpMessage(DumpCommandId command, DumpType return new IpcMessage(DiagnosticsServerCommandSet.Dump, (byte)command, payload); } + private static IpcMessage CreateWriteDumpMessageV4(DumpType dumpType, string dumpPath, WriteDumpFlags flags, string logPath) + { + if (string.IsNullOrEmpty(dumpPath)) + { + throw new ArgumentNullException($"{nameof(dumpPath)} required"); + } + + byte[] payload = SerializePayload(dumpPath, (uint)dumpType, (uint)flags, logPath); + return new IpcMessage(DiagnosticsServerCommandSet.Dump, (byte)DumpCommandId.GenerateCoreDump4, payload); + } + private static ProcessInfo GetProcessInfoFromResponse(IpcResponse response, string operationName) { ValidateResponseMessage(response.Message, operationName); diff --git a/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/WriteDumpFlags.cs b/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/WriteDumpFlags.cs index 25bf8dab3e..9da3fb7e4e 100644 --- a/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/WriteDumpFlags.cs +++ b/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsClient/WriteDumpFlags.cs @@ -3,11 +3,13 @@ namespace Microsoft.Diagnostics.NETCore.Client { - public enum WriteDumpFlags + [System.Flags] + public enum WriteDumpFlags : uint { None = 0x00, LoggingEnabled = 0x01, VerboseLoggingEnabled = 0x02, - CrashReportEnabled = 0x04 + CrashReportEnabled = 0x04, + LogToFile = 0x8 } } diff --git a/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsIpc/IpcCommands.cs b/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsIpc/IpcCommands.cs index da251b7750..362e92320e 100644 --- a/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsIpc/IpcCommands.cs +++ b/src/Microsoft.Diagnostics.NETCore.Client/DiagnosticsIpc/IpcCommands.cs @@ -32,6 +32,7 @@ internal enum DumpCommandId : byte GenerateCoreDump = 0x01, GenerateCoreDump2 = 0x02, GenerateCoreDump3 = 0x03, + GenerateCoreDump4 = 0x04, } internal enum ProfilerCommandId : byte diff --git a/src/Tools/dotnet-dump/dotnet-dump.csproj b/src/Tools/dotnet-dump/dotnet-dump.csproj index 3603b0d636..517f8929c0 100644 --- a/src/Tools/dotnet-dump/dotnet-dump.csproj +++ b/src/Tools/dotnet-dump/dotnet-dump.csproj @@ -25,9 +25,10 @@ + - + From 368856a564b18351ccb6e49334f0dd621f6e2aae Mon Sep 17 00:00:00 2001 From: Juan Sebastian Hoyos Ayala Date: Mon, 15 May 2023 03:41:35 -0700 Subject: [PATCH 5/6] Add option to log to file in dotnet-dump --- src/Tools/dotnet-dump/Dumper.cs | 40 ++++++++++++++++++++++--------- src/Tools/dotnet-dump/Program.cs | 41 ++++++++++++++++++++++++++------ 2 files changed, 63 insertions(+), 18 deletions(-) diff --git a/src/Tools/dotnet-dump/Dumper.cs b/src/Tools/dotnet-dump/Dumper.cs index d7faefa0f2..56c569f06c 100644 --- a/src/Tools/dotnet-dump/Dumper.cs +++ b/src/Tools/dotnet-dump/Dumper.cs @@ -6,8 +6,6 @@ using System.CommandLine.IO; using System.IO; using System.Runtime.InteropServices; -using System.Threading; -using System.Threading.Tasks; using Microsoft.Diagnostics.NETCore.Client; using Microsoft.Internal.Common.Utils; @@ -30,9 +28,16 @@ public enum DumpTypeOption Triage // A small dump containing module lists, thread lists, exception information, all stacks and PII removed. } + public enum LogLevelOption + { + None, // No logging. + Diag, // Diagnostic logging, useful for first level. + Verbose, // Verbose loggging, useful to debug unwinding but also slows down target process. + } + internal static int Collect(DumpCollectionConfig config, IConsole console) { - if (!CommandUtils.ValidateArgumentsForAttach(config.ProcessId, config.ProcessName, config.DiagnosticPort, out int targetPprocessId)) + if (!CommandUtils.ValidateArgumentsForAttach(config.ProcessId, config.ProcessName, config.DiagnosticPort, out int targetProcessId)) { return -1; } @@ -71,13 +76,13 @@ internal static int Collect(DumpCollectionConfig config, IConsole console) return -1; } - if (config.EnableDiagnosticOutput) + if (config.LogLevel != LogLevelOption.None) { - console.Error.WriteLine("Verbose diagnostics not supported on Windows."); + console.Error.WriteLine("Diagnostic logging not supported on Windows."); return -1; } - Windows.CollectDump(targetPprocessId, config.DumpOutputPath, config.DumpType); + Windows.CollectDump(targetProcessId, config.DumpOutputPath, config.DumpType); } else { @@ -104,22 +109,35 @@ internal static int Collect(DumpCollectionConfig config, IConsole console) } else { - client = new DiagnosticsClient(targetPprocessId); + client = new DiagnosticsClient(targetProcessId); } WriteDumpFlags flags = WriteDumpFlags.None; - if (config.EnableDiagnosticOutput) + + flags |= config.LogLevel switch + { + LogLevelOption.None => WriteDumpFlags.None, + LogLevelOption.Diag => WriteDumpFlags.LoggingEnabled, + LogLevelOption.Verbose => WriteDumpFlags.VerboseLoggingEnabled, + _ => throw new ArgumentException($"Invalid log level supplied: {config.LogLevel}") + }; + + if (config.LogToFile || config.DiagnosticLogPath is not null) + { + flags |= WriteDumpFlags.LogToFile; + } + else if (flags != WriteDumpFlags.None) { console.Out.WriteLine("Diagnostic output requested. Logging will appear in the console of the target process."); - flags |= WriteDumpFlags.LoggingEnabled; } + if (config.GenerateCrashReport) { flags |= WriteDumpFlags.CrashReportEnabled; } // Send the command to the runtime to initiate the core dump - client.WriteDump(dumpType, config.DumpOutputPath, flags); + client.WriteDump(dumpType, config.DumpOutputPath, flags, config.DiagnosticLogPath); } } catch (Exception ex) when @@ -135,7 +153,7 @@ NotSupportedException or DiagnosticsClientException) { console.Error.WriteLine($"{ex.Message}"); - if (!config.EnableDiagnosticOutput) + if (config.LogLevel == LogLevelOption.None) { console.Error.WriteLine($"Consider rerunning the command with diagnostic output enabled."); } diff --git a/src/Tools/dotnet-dump/Program.cs b/src/Tools/dotnet-dump/Program.cs index a13292f0ea..5785ce0fe3 100644 --- a/src/Tools/dotnet-dump/Program.cs +++ b/src/Tools/dotnet-dump/Program.cs @@ -7,20 +7,21 @@ using System.CommandLine.Invocation; using System.CommandLine.Parsing; using System.IO; -using System.Threading; using System.Threading.Tasks; using Microsoft.Internal.Common.Commands; using Microsoft.Tools.Common; namespace Microsoft.Diagnostics.Tools.Dump { - // Make sure the name of the fields match the option names. + // Make sure the name of the fields match the option's argument names. internal record struct DumpCollectionConfig( int ProcessId, string ProcessName, string DiagnosticPort, string DumpOutputPath, - bool EnableDiagnosticOutput, + Dumper.LogLevelOption LogLevel, + bool LogToFile, + string DiagnosticLogPath, bool GenerateCrashReport, Dumper.DumpTypeOption DumpType); @@ -44,7 +45,8 @@ private static Command CollectCommand() => // Handler CommandHandler.Create(Dumper.Collect), // Options - ProcessIdOption(), ProcessNameOption(), DiagnosticPortOption(), OutputOption(), DiagnosticLoggingOption(), CrashReportOption(), TypeOption() + ProcessIdOption(), ProcessNameOption(), DiagnosticPortOption(), + OutputOption(), DiagnosticLoggingOption(), FileLoggingOption(), LoggingPathOption(), CrashReportOption(), TypeOption() }; private static Option ProcessIdOption() => @@ -86,11 +88,36 @@ private static Option OutputOption() => private static Option DiagnosticLoggingOption() => new( - alias: "--diag", + alias: "--log", description: "Enable dump collection diagnostic logging.") { - Name = nameof(DumpCollectionConfig.EnableDiagnosticOutput), - Argument = new Argument(name: "diag") + Name = nameof(DumpCollectionConfig.LogLevel), + Argument = new Argument(name: "log", getDefaultValue: static () => Dumper.LogLevelOption.None) + }; + + // Until we update system.commandline to a newer version, the semmantics of this are... not ideal: + // if the user wants the log to go to file they either have to use logToFile for the default value, + // or they have to use logFilePath. + // See https://github.com/dotnet/command-line-api/pull/1462 + private static Option FileLoggingOption() => + new( + alias: "--log-to-file", + description: "Route diagnostic logging to a file.", + getDefaultValue: static () => false) + { + Name = nameof(DumpCollectionConfig.LogToFile), + Argument = new Argument(name: nameof(DumpCollectionConfig.LogToFile)), + IsRequired = false + }; + + private static Option LoggingPathOption() => + new( + alias: "--log-file-path", + description: "Route diagnostic logging to a specific file.") + { + Name = nameof(DumpCollectionConfig.DiagnosticLogPath), + Argument = new Argument(name: nameof(DumpCollectionConfig.DiagnosticLogPath)), + IsRequired = false }; private static Option CrashReportOption() => From fde4566318ca2bd070b865f6e6cf55d2a5b6985f Mon Sep 17 00:00:00 2001 From: Juan Sebastian Hoyos Ayala Date: Wed, 17 May 2023 19:26:37 -0700 Subject: [PATCH 6/6] Add CreateCoreDump4 spec --- documentation/design-docs/ipc-protocol.md | 52 +++++++++++++++++++++++ 1 file changed, 52 insertions(+) diff --git a/documentation/design-docs/ipc-protocol.md b/documentation/design-docs/ipc-protocol.md index 5c2619b6f2..313fc29a64 100644 --- a/documentation/design-docs/ipc-protocol.md +++ b/documentation/design-docs/ipc-protocol.md @@ -743,6 +743,58 @@ Payload } ``` +### `CreateCoreDump4` + +Command Code: `0x0104` + +The `CreateCoreDump4` command is used to instruct the runtime to generate a core dump of the process. It augments `CreateCoreDump3` with the ability to collect verbose logs and to redirect them to any given file. + +In the event of an [error](#errors), the runtime will attempt to send an error message and subsequently close the connection. + +#### Inputs + +Header: `{ Magic; Size; 0x0104; 0x0000 }` + +* `string dumpName`: As described in [`CreateCoreDump`](#createcoredump). +* `uint dumpType`: As described in [`CreateCoreDump`](#createcoredump). +* `uint flags`: Flags as defined by [`CreateCoreDump3`](#createcoredump3), with the following addition: + * `GenerateDumpFlagsLogToFile = 0x8`: Generate crashdump report next to the generated dump. +* `string logPath`: The path to use for logging. In case this is null or empty, the runtime will log to a file adjacent to the dump. + +#### Returns (as an IPC Message Payload) + +Header: `{ Magic; Size; 0xFF00; 0x0000; }` + +`CreateCoreDump4` returns: + +* `int32 hresult`: The result of creating the core dump (`0` indicates success). +* `string error`: Optionally the payload may have an error describing the collection issues. + +##### Details + + +Input: + +``` +Payload +{ + string dumpName, + uint dumpType, + uint flags + string logPath +} +``` + +Returns: + +```c +Payload +{ + int32 hresult + string errorString +} +``` + ## Profiler Commands ### `AttachProfiler`