File: EditAndContinue\EditAndContinueWorkspaceServiceTests.cs
Web Access
Project: ..\..\..\src\EditorFeatures\Test\Microsoft.CodeAnalysis.EditorFeatures.UnitTests.csproj (Microsoft.CodeAnalysis.EditorFeatures.UnitTests)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
#nullable disable
 
using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.IO;
using System.Linq;
using System.Reflection.Metadata;
using System.Reflection.Metadata.Ecma335;
using System.Reflection.PortableExecutable;
using System.Runtime.Remoting.Contexts;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.Test.Utilities;
using Microsoft.CodeAnalysis.Debugging;
using Microsoft.CodeAnalysis.Diagnostics;
using Microsoft.CodeAnalysis.EditAndContinue.Contracts;
using Microsoft.CodeAnalysis.Editor.UnitTests;
using Microsoft.CodeAnalysis.Editor.UnitTests.Diagnostics;
using Microsoft.CodeAnalysis.Editor.UnitTests.Workspaces;
using Microsoft.CodeAnalysis.Emit;
using Microsoft.CodeAnalysis.ExternalAccess.UnitTesting.Api;
using Microsoft.CodeAnalysis.ExternalAccess.Watch.Api;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.LanguageServer;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.Shared.TestHooks;
using Microsoft.CodeAnalysis.Test.Utilities;
using Microsoft.CodeAnalysis.Text;
using Microsoft.CodeAnalysis.UnitTests;
using Microsoft.VisualStudio.Text;
using Roslyn.Test.Utilities;
using Roslyn.Test.Utilities.TestGenerators;
using Roslyn.Utilities;
using Xunit;
 
namespace Microsoft.CodeAnalysis.EditAndContinue.UnitTests
{
    using static ActiveStatementTestHelpers;
 
    [UseExportProvider]
    public sealed partial class EditAndContinueWorkspaceServiceTests : TestBase
    {
        private static readonly Guid s_solutionTelemetryId = Guid.Parse("00000000-AAAA-AAAA-AAAA-000000000000");
        private static readonly Guid s_defaultProjectTelemetryId = Guid.Parse("00000000-AAAA-AAAA-AAAA-111111111111");
 
        private static readonly ActiveStatementSpanProvider s_noActiveSpans =
            (_, _, _) => new(ImmutableArray<ActiveStatementSpan>.Empty);
 
        private const TargetFramework DefaultTargetFramework = TargetFramework.NetStandard20;
 
        private Func<Project, CompilationOutputs> _mockCompilationOutputsProvider;
        private readonly List<string> _telemetryLog = new();
        private int _telemetryId;
 
        private readonly MockManagedEditAndContinueDebuggerService _debuggerService;
 
        public EditAndContinueWorkspaceServiceTests()
        {
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(Guid.NewGuid());
 
            _debuggerService = new MockManagedEditAndContinueDebuggerService()
            {
                LoadedModules = new Dictionary<Guid, ManagedHotReloadAvailability>()
            };
        }
 
        private TestWorkspace CreateWorkspace(out Solution solution, out EditAndContinueWorkspaceService service, Type[] additionalParts = null)
        {
            var workspace = new TestWorkspace(composition: FeaturesTestCompositions.Features.AddParts(additionalParts), solutionTelemetryId: s_solutionTelemetryId);
            solution = workspace.CurrentSolution;
            service = GetEditAndContinueService(workspace);
            return workspace;
        }
 
        private TestWorkspace CreateEditorWorkspace(out Solution solution, out EditAndContinueWorkspaceService service, out EditAndContinueLanguageService languageService, Type[] additionalParts = null)
        {
            var composition = EditorTestCompositions.EditorFeatures
                .RemoveParts(typeof(MockWorkspaceEventListenerProvider))
                .AddParts(
                    typeof(MockHostWorkspaceProvider),
                    typeof(MockManagedHotReloadService))
                .AddParts(additionalParts);
 
            var workspace = new TestWorkspace(composition: composition, solutionTelemetryId: s_solutionTelemetryId);
            ((MockHostWorkspaceProvider)workspace.GetService<IHostWorkspaceProvider>()).Workspace = workspace;
 
            solution = workspace.CurrentSolution;
            service = GetEditAndContinueService(workspace);
            languageService = workspace.GetService<EditAndContinueLanguageService>();
            return workspace;
        }
 
        private static SourceText GetAnalyzerConfigText((string key, string value)[] analyzerConfig)
            => CreateText("[*.*]" + Environment.NewLine + string.Join(Environment.NewLine, analyzerConfig.Select(c => $"{c.key} = {c.value}")));
 
        private static (Solution, Document) AddDefaultTestProject(
            Solution solution,
            string source,
            ISourceGenerator generator = null,
            string additionalFileText = null,
            (string key, string value)[] analyzerConfig = null)
        {
            solution = AddDefaultTestProject(solution, new[] { source }, generator, additionalFileText, analyzerConfig);
            return (solution, solution.Projects.Single().Documents.Single());
        }
 
        private static Solution AddDefaultTestProject(
            Solution solution,
            string[] sources,
            ISourceGenerator generator = null,
            string additionalFileText = null,
            (string key, string value)[] analyzerConfig = null)
        {
            var projectId = ProjectId.CreateNewId();
 
            var project = solution.
                AddProject(ProjectInfo.Create(projectId, VersionStamp.Create(), "proj", "proj", LanguageNames.CSharp, parseOptions: CSharpParseOptions.Default.WithNoRefSafetyRulesAttribute()).WithTelemetryId(s_defaultProjectTelemetryId)).GetProject(projectId).
                WithMetadataReferences(TargetFrameworkUtil.GetReferences(DefaultTargetFramework));
 
            solution = project.Solution;
 
            if (generator != null)
            {
                solution = solution.AddAnalyzerReference(project.Id, new TestGeneratorReference(generator));
            }
 
            if (additionalFileText != null)
            {
                solution = solution.AddAdditionalDocument(DocumentId.CreateNewId(project.Id), "additional", CreateText(additionalFileText));
            }
 
            if (analyzerConfig != null)
            {
                solution = solution.AddAnalyzerConfigDocument(
                    DocumentId.CreateNewId(project.Id),
                    name: "config",
                    GetAnalyzerConfigText(analyzerConfig),
                    filePath: Path.Combine(TempRoot.Root, "config"));
            }
 
            Document document = null;
            var i = 1;
            foreach (var source in sources)
            {
                var fileName = $"test{i++}.cs";
 
                document = solution.GetProject(project.Id).
                    AddDocument(fileName, CreateText(source), filePath: Path.Combine(TempRoot.Root, fileName));
 
                solution = document.Project.Solution;
            }
 
            return document.Project.Solution;
        }
 
        private EditAndContinueWorkspaceService GetEditAndContinueService(Workspace workspace)
        {
            var service = (EditAndContinueWorkspaceService)workspace.Services.GetRequiredService<IEditAndContinueWorkspaceService>();
            var accessor = service.GetTestAccessor();
            accessor.SetOutputProvider(project => _mockCompilationOutputsProvider(project));
            return service;
        }
 
        private async Task<DebuggingSession> StartDebuggingSessionAsync(
            EditAndContinueWorkspaceService service,
            Solution solution,
            CommittedSolution.DocumentState initialState = CommittedSolution.DocumentState.MatchesBuildOutput,
            IPdbMatchingSourceTextProvider sourceTextProvider = null)
        {
            var sessionId = await service.StartDebuggingSessionAsync(
                solution,
                _debuggerService,
                sourceTextProvider: sourceTextProvider ?? NullPdbMatchingSourceTextProvider.Instance,
                captureMatchingDocuments: ImmutableArray<DocumentId>.Empty,
                captureAllMatchingDocuments: false,
                reportDiagnostics: true,
                CancellationToken.None);
 
            var session = service.GetTestAccessor().GetDebuggingSession(sessionId);
 
            if (initialState != CommittedSolution.DocumentState.None)
            {
                SetDocumentsState(session, solution, initialState);
            }
 
            session.GetTestAccessor().SetTelemetryLogger((id, message) => _telemetryLog.Add($"{id}: {message.GetMessage()}"), () => ++_telemetryId);
 
            return session;
        }
 
        private void EnterBreakState(
            DebuggingSession session,
            ImmutableArray<ManagedActiveStatementDebugInfo> activeStatements = default,
            ImmutableArray<DocumentId> documentsWithRudeEdits = default)
        {
            _debuggerService.GetActiveStatementsImpl = () => activeStatements.NullToEmpty();
            session.BreakStateOrCapabilitiesChanged(inBreakState: true, out var documentsToReanalyze);
            AssertEx.Equal(documentsWithRudeEdits.NullToEmpty(), documentsToReanalyze);
        }
 
        private void ExitBreakState(
            DebuggingSession session,
            ImmutableArray<DocumentId> documentsWithRudeEdits = default)
        {
            _debuggerService.GetActiveStatementsImpl = () => ImmutableArray<ManagedActiveStatementDebugInfo>.Empty;
            session.BreakStateOrCapabilitiesChanged(inBreakState: false, out var documentsToReanalyze);
            AssertEx.Equal(documentsWithRudeEdits.NullToEmpty(), documentsToReanalyze);
        }
 
        private static void CapabilitiesChanged(
            DebuggingSession session,
            ImmutableArray<DocumentId> documentsWithRudeEdits = default)
        {
            session.BreakStateOrCapabilitiesChanged(inBreakState: null, out var documentsToReanalyze);
            AssertEx.Equal(documentsWithRudeEdits.NullToEmpty(), documentsToReanalyze);
        }
 
        private static void CommitSolutionUpdate(DebuggingSession session, ImmutableArray<DocumentId> documentsWithRudeEdits = default)
        {
            session.CommitSolutionUpdate(out var documentsToReanalyze);
            AssertEx.Equal(documentsWithRudeEdits.NullToEmpty(), documentsToReanalyze);
        }
 
        private static void EndDebuggingSession(DebuggingSession session, ImmutableArray<DocumentId> documentsWithRudeEdits = default)
        {
            session.EndSession(out var documentsToReanalyze, out _);
            AssertEx.Equal(documentsWithRudeEdits.NullToEmpty(), documentsToReanalyze);
        }
 
        private static async Task<(ModuleUpdates updates, ImmutableArray<DiagnosticData> diagnostics)> EmitSolutionUpdateAsync(
            DebuggingSession session,
            Solution solution,
            ActiveStatementSpanProvider activeStatementSpanProvider = null)
        {
            var result = await session.EmitSolutionUpdateAsync(solution, activeStatementSpanProvider ?? s_noActiveSpans, CancellationToken.None);
            return (result.ModuleUpdates, result.GetDiagnosticData(solution));
        }
 
        internal static void SetDocumentsState(DebuggingSession session, Solution solution, CommittedSolution.DocumentState state)
        {
            foreach (var project in solution.Projects)
            {
                foreach (var document in project.Documents)
                {
                    session.LastCommittedSolution.Test_SetDocumentState(document.Id, state);
                }
            }
        }
 
        private static IEnumerable<string> InspectDiagnostics(ImmutableArray<DiagnosticData> actual)
            => actual.Select(d => InspectDiagnostic(d));
 
        private static string InspectDiagnostic(DiagnosticData diagnostic)
            => $"{(string.IsNullOrWhiteSpace(diagnostic.DataLocation.MappedFileSpan.Path) ? diagnostic.ProjectId.ToString() : diagnostic.DataLocation.MappedFileSpan.ToString())}: {diagnostic.Severity} {diagnostic.Id}: {diagnostic.Message}";
 
        internal static Guid ReadModuleVersionId(Stream stream)
        {
            using var peReader = new PEReader(stream);
            var metadataReader = peReader.GetMetadataReader();
            var mvidHandle = metadataReader.GetModuleDefinition().Mvid;
            return metadataReader.GetGuid(mvidHandle);
        }
 
        private Guid EmitAndLoadLibraryToDebuggee(string source, string sourceFilePath = null, Encoding encoding = null, SourceHashAlgorithm checksumAlgorithm = SourceHashAlgorithms.Default, string assemblyName = "")
        {
            var moduleId = EmitLibrary(source, sourceFilePath, encoding, checksumAlgorithm, assemblyName);
            LoadLibraryToDebuggee(moduleId);
            return moduleId;
        }
 
        private void LoadLibraryToDebuggee(Guid moduleId, ManagedHotReloadAvailability availability = default)
        {
            _debuggerService.LoadedModules.Add(moduleId, availability);
        }
 
        private Guid EmitLibrary(
            string source,
            string sourceFilePath = null,
            Encoding encoding = null,
            SourceHashAlgorithm checksumAlgorithm = SourceHashAlgorithms.Default,
            string assemblyName = "",
            DebugInformationFormat pdbFormat = DebugInformationFormat.PortablePdb,
            ISourceGenerator generator = null,
            string additionalFileText = null,
            IEnumerable<(string, string)> analyzerOptions = null)
        {
            return EmitLibrary(new[] { (source, sourceFilePath ?? Path.Combine(TempRoot.Root, "test1.cs")) }, encoding, checksumAlgorithm, assemblyName, pdbFormat, generator, additionalFileText, analyzerOptions);
        }
 
        private Guid EmitLibrary(
            (string content, string filePath)[] sources,
            Encoding encoding = null,
            SourceHashAlgorithm checksumAlgorithm = SourceHashAlgorithms.Default,
            string assemblyName = "",
            DebugInformationFormat pdbFormat = DebugInformationFormat.PortablePdb,
            ISourceGenerator generator = null,
            string additionalFileText = null,
            IEnumerable<(string, string)> analyzerOptions = null)
        {
            encoding ??= Encoding.UTF8;
 
            var parseOptions = TestOptions.RegularPreview.WithNoRefSafetyRulesAttribute();
 
            var trees = sources.Select(source =>
            {
                var sourceText = SourceText.From(new MemoryStream(encoding.GetBytesWithPreamble(source.content)), encoding, checksumAlgorithm);
                return SyntaxFactory.ParseSyntaxTree(sourceText, parseOptions, source.filePath);
            });
 
            Compilation compilation = CSharpTestBase.CreateCompilation(trees.ToArray(), options: TestOptions.DebugDll, targetFramework: DefaultTargetFramework, assemblyName: assemblyName);
 
            if (generator != null)
            {
                var optionsProvider = (analyzerOptions != null) ? new EditAndContinueTestAnalyzerConfigOptionsProvider(analyzerOptions) : null;
                var additionalTexts = (additionalFileText != null) ? new[] { new InMemoryAdditionalText("additional_file", additionalFileText) } : null;
                var generatorDriver = CSharpGeneratorDriver.Create(new[] { generator }, additionalTexts, parseOptions, optionsProvider);
                generatorDriver.RunGeneratorsAndUpdateCompilation(compilation, out var outputCompilation, out var generatorDiagnostics);
                generatorDiagnostics.Verify();
                compilation = outputCompilation;
            }
 
            return EmitLibrary(compilation, pdbFormat);
        }
 
        private Guid EmitLibrary(Compilation compilation, DebugInformationFormat pdbFormat = DebugInformationFormat.PortablePdb)
        {
            var (peImage, pdbImage) = compilation.EmitToArrays(new EmitOptions(debugInformationFormat: pdbFormat));
            var symReader = SymReaderTestHelpers.OpenDummySymReader(pdbImage);
 
            var moduleMetadata = ModuleMetadata.CreateFromImage(peImage);
            var moduleId = moduleMetadata.GetModuleVersionId();
 
            // associate the binaries with the project (assumes a single project)
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(moduleId)
            {
                OpenAssemblyStreamImpl = () =>
                {
                    var stream = new MemoryStream();
                    peImage.WriteToStream(stream);
                    stream.Position = 0;
                    return stream;
                },
                OpenPdbStreamImpl = () =>
                {
                    var stream = new MemoryStream();
                    pdbImage.WriteToStream(stream);
                    stream.Position = 0;
                    return stream;
                }
            };
 
            return moduleId;
        }
 
        private static SourceText CreateText(string source)
            => SourceText.From(source, Encoding.UTF8, SourceHashAlgorithms.Default);
 
        private static SourceText CreateTextFromFile(string path)
        {
            using var stream = File.OpenRead(path);
            return SourceText.From(stream, Encoding.UTF8, SourceHashAlgorithms.Default);
        }
 
        private static TextSpan GetSpan(string str, string substr)
            => new TextSpan(str.IndexOf(substr), substr.Length);
 
        private static void VerifyReadersDisposed(IEnumerable<IDisposable> readers)
        {
            foreach (var reader in readers)
            {
                Assert.Throws<ObjectDisposedException>(() =>
                {
                    if (reader is MetadataReaderProvider md)
                    {
                        md.GetMetadataReader();
                    }
                    else
                    {
                        ((DebugInformationReaderProvider)reader).CreateEditAndContinueMethodDebugInfoReader();
                    }
                });
            }
        }
 
        private static DocumentInfo CreateDesignTimeOnlyDocument(ProjectId projectId, string name = "design-time-only.cs", string path = "design-time-only.cs")
        {
            var sourceText = CreateText("class DTO {}");
            return DocumentInfo.Create(
                DocumentId.CreateNewId(projectId, name),
                name: name,
                folders: Array.Empty<string>(),
                sourceCodeKind: SourceCodeKind.Regular,
                loader: TextLoader.From(TextAndVersion.Create(sourceText, VersionStamp.Create(), path)),
                filePath: path,
                isGenerated: false)
                .WithDesignTimeOnly(true);
        }
 
        internal sealed class FailingTextLoader : TextLoader
        {
            public override Task<TextAndVersion> LoadTextAndVersionAsync(LoadTextOptions options, CancellationToken cancellationToken)
            {
                Assert.True(false, $"Content of document should never be loaded");
                throw ExceptionUtilities.Unreachable();
            }
        }
 
        private static EditAndContinueLogEntry Row(int rowNumber, TableIndex table, EditAndContinueOperation operation)
            => new(MetadataTokens.Handle(table, rowNumber), operation);
 
        private static unsafe void VerifyEncLogMetadata(ImmutableArray<byte> delta, params EditAndContinueLogEntry[] expectedRows)
        {
            fixed (byte* ptr = delta.ToArray())
            {
                var reader = new MetadataReader(ptr, delta.Length);
                AssertEx.Equal(expectedRows, reader.GetEditAndContinueLogEntries(), itemInspector: EncLogRowToString);
            }
 
            static string EncLogRowToString(EditAndContinueLogEntry row)
            {
                TableIndex tableIndex;
                MetadataTokens.TryGetTableIndex(row.Handle.Kind, out tableIndex);
 
                return string.Format(
                    "Row({0}, TableIndex.{1}, EditAndContinueOperation.{2})",
                    MetadataTokens.GetRowNumber(row.Handle),
                    tableIndex,
                    row.Operation);
            }
        }
 
        private static void GenerateSource(GeneratorExecutionContext context)
        {
            foreach (var syntaxTree in context.Compilation.SyntaxTrees)
            {
                var fileName = PathUtilities.GetFileName(syntaxTree.FilePath);
 
                Generate(syntaxTree.GetText().ToString(), fileName);
 
                if (context.AnalyzerConfigOptions.GetOptions(syntaxTree).TryGetValue("enc_generator_output", out var optionValue))
                {
                    context.AddSource("GeneratedFromOptions_" + fileName, $"class G {{ int X => {optionValue}; }}");
                }
            }
 
            foreach (var additionalFile in context.AdditionalFiles)
            {
                Generate(additionalFile.GetText()!.ToString(), PathUtilities.GetFileName(additionalFile.Path));
            }
 
            void Generate(string source, string fileName)
            {
                var generatedSource = GetGeneratedCodeFromMarkedSource(source);
                if (generatedSource != null)
                {
                    context.AddSource($"Generated_{fileName}", generatedSource);
                }
            }
        }
 
        private static string GetGeneratedCodeFromMarkedSource(string markedSource)
        {
            const string OpeningMarker = "/* GENERATE:";
            const string ClosingMarker = "*/";
 
            var index = markedSource.IndexOf(OpeningMarker);
            if (index > 0)
            {
                index += OpeningMarker.Length;
                var closing = markedSource.IndexOf(ClosingMarker, index);
                return markedSource[index..closing].Trim();
            }
 
            return null;
        }
 
        [Theory]
        [CombinatorialData]
        public async Task StartDebuggingSession_CapturingDocuments(bool captureAllDocuments)
        {
            var encodingA = Encoding.BigEndianUnicode;
            var encodingB = Encoding.Unicode;
            var encodingC = Encoding.GetEncoding("SJIS");
            var encodingE = Encoding.UTF8;
 
            var sourceA1 = "class A {}";
            var sourceB1 = "class B { int F() => 1; }";
            var sourceB2 = "class B { int F() => 2; }";
            var sourceB3 = "class B { int F() => 3; }";
            var sourceC1 = "class C { const char L = 'ワ'; }";
            var sourceD1 = "dummy code";
            var sourceE1 = "class E { }";
            var sourceBytesA1 = encodingA.GetBytesWithPreamble(sourceA1);
            var sourceBytesB1 = encodingB.GetBytesWithPreamble(sourceB1);
            var sourceBytesC1 = encodingC.GetBytesWithPreamble(sourceC1);
            var sourceBytesD1 = Encoding.UTF8.GetBytesWithPreamble(sourceD1);
            var sourceBytesE1 = encodingE.GetBytesWithPreamble(sourceE1);
 
            var dir = Temp.CreateDirectory();
            var sourceFileA = dir.CreateFile("A.cs").WriteAllBytes(sourceBytesA1);
            var sourceFileB = dir.CreateFile("B.cs").WriteAllBytes(sourceBytesB1);
            var sourceFileC = dir.CreateFile("C.cs").WriteAllBytes(sourceBytesC1);
            var sourceFileD = dir.CreateFile("dummy").WriteAllBytes(sourceBytesD1);
            var sourceFileE = dir.CreateFile("E.cs").WriteAllBytes(sourceBytesE1);
            var sourceTreeA1 = SyntaxFactory.ParseSyntaxTree(SourceText.From(sourceBytesA1, sourceBytesA1.Length, encodingA, SourceHashAlgorithms.Default), TestOptions.Regular, sourceFileA.Path);
            var sourceTreeB1 = SyntaxFactory.ParseSyntaxTree(SourceText.From(sourceBytesB1, sourceBytesB1.Length, encodingB, SourceHashAlgorithms.Default), TestOptions.Regular, sourceFileB.Path);
            var sourceTreeC1 = SyntaxFactory.ParseSyntaxTree(SourceText.From(sourceBytesC1, sourceBytesC1.Length, encodingC, SourceHashAlgorithm.Sha1), TestOptions.Regular, sourceFileC.Path);
 
            // E is not included in the compilation:
            var compilation = CSharpTestBase.CreateCompilation(new[] { sourceTreeA1, sourceTreeB1, sourceTreeC1 }, options: TestOptions.DebugDll, targetFramework: DefaultTargetFramework, assemblyName: "P");
            EmitLibrary(compilation);
 
            // change content of B on disk:
            sourceFileB.WriteAllText(sourceB2, encodingB);
 
            // prepare workspace as if it was loaded from project files:
            using var _ = CreateWorkspace(out var solution, out var service, new[] { typeof(NoCompilationLanguageService) });
 
            var projectPId = ProjectId.CreateNewId();
            solution = solution
                .AddProject(projectPId, "P", "P", LanguageNames.CSharp)
                .WithProjectChecksumAlgorithm(projectPId, SourceHashAlgorithm.Sha1);
 
            var documentIdA = DocumentId.CreateNewId(projectPId, debugName: "A");
            solution = solution.AddDocument(DocumentInfo.Create(
                id: documentIdA,
                name: "A",
                loader: new WorkspaceFileTextLoader(solution.Services, sourceFileA.Path, encodingA),
                filePath: sourceFileA.Path));
 
            var documentIdB = DocumentId.CreateNewId(projectPId, debugName: "B");
            solution = solution.AddDocument(DocumentInfo.Create(
                id: documentIdB,
                name: "B",
                loader: new WorkspaceFileTextLoader(solution.Services, sourceFileB.Path, encodingB),
                filePath: sourceFileB.Path));
 
            var documentIdC = DocumentId.CreateNewId(projectPId, debugName: "C");
            solution = solution.AddDocument(DocumentInfo.Create(
                id: documentIdC,
                name: "C",
                loader: new WorkspaceFileTextLoader(solution.Services, sourceFileC.Path, encodingC),
                filePath: sourceFileC.Path));
 
            var documentIdE = DocumentId.CreateNewId(projectPId, debugName: "E");
            solution = solution.AddDocument(DocumentInfo.Create(
                id: documentIdE,
                name: "E",
                loader: new WorkspaceFileTextLoader(solution.Services, sourceFileE.Path, encodingE),
                filePath: sourceFileE.Path));
 
            // check that we are testing documents whose hash algorithm does not match the PDB (but the hash itself does):
            Assert.Equal(SourceHashAlgorithm.Sha1, solution.GetDocument(documentIdA).GetTextSynchronously(default).ChecksumAlgorithm);
            Assert.Equal(SourceHashAlgorithm.Sha1, solution.GetDocument(documentIdB).GetTextSynchronously(default).ChecksumAlgorithm);
            Assert.Equal(SourceHashAlgorithm.Sha1, solution.GetDocument(documentIdC).GetTextSynchronously(default).ChecksumAlgorithm);
            Assert.Equal(SourceHashAlgorithm.Sha1, solution.GetDocument(documentIdE).GetTextSynchronously(default).ChecksumAlgorithm);
 
            // design-time-only document with and without absolute path:
            solution = solution.
                AddDocument(CreateDesignTimeOnlyDocument(projectPId, name: "dt1.cs", path: Path.Combine(dir.Path, "dt1.cs"))).
                AddDocument(CreateDesignTimeOnlyDocument(projectPId, name: "dt2.cs", path: "dt2.cs"));
 
            // project that does not support EnC - the contents of documents in this project shouldn't be loaded:
            var projectQ = solution.AddProject("Q", "Q", NoCompilationConstants.LanguageName);
            solution = projectQ.Solution;
 
            solution = solution.AddDocument(DocumentInfo.Create(
                id: DocumentId.CreateNewId(projectQ.Id, debugName: "D"),
                name: "D",
                loader: new FailingTextLoader(),
                filePath: sourceFileD.Path));
 
            var captureMatchingDocuments = captureAllDocuments
                ? ImmutableArray<DocumentId>.Empty
                : (from project in solution.Projects from documentId in project.DocumentIds select documentId).ToImmutableArray();
 
            var sessionId = await service.StartDebuggingSessionAsync(solution, _debuggerService, NullPdbMatchingSourceTextProvider.Instance, captureMatchingDocuments, captureAllDocuments, reportDiagnostics: true, CancellationToken.None);
            var debuggingSession = service.GetTestAccessor().GetDebuggingSession(sessionId);
 
            var matchingDocuments = debuggingSession.LastCommittedSolution.Test_GetDocumentStates();
            AssertEx.Equal(new[]
            {
                "(A, MatchesBuildOutput)",
                "(C, MatchesBuildOutput)"
            }, matchingDocuments.Select(e => (solution.GetDocument(e.id).Name, e.state)).OrderBy(e => e.Name).Select(e => e.ToString()));
 
            // change content of B on disk again:
            sourceFileB.WriteAllText(sourceB3, encodingB);
            solution = solution.WithDocumentTextLoader(documentIdB, new WorkspaceFileTextLoader(solution.Services, sourceFileB.Path, encodingB), PreservationMode.PreserveValue);
 
            EnterBreakState(debuggingSession);
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            AssertEx.Equal(new[] { $"{projectPId}: Warning ENC1005: {string.Format(FeaturesResources.DocumentIsOutOfSyncWithDebuggee, sourceFileB.Path)}" }, InspectDiagnostics(emitDiagnostics));
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ProjectNotBuilt()
        {
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, "class C1 { void M() { System.Console.WriteLine(1); } }");
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(Guid.Empty);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // no changes:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document1, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // change the source:
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M() { System.Console.WriteLine(2); } }"));
            var document2 = solution.GetDocument(document1.Id);
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // changes in the project are ignored:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task DifferentDocumentWithSameContent()
        {
            var source = "class C1 { void M1() { System.Console.WriteLine(1); } }";
            var moduleFile = Temp.CreateFile().WriteAllBytes(TestResources.Basic.Members);
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source);
 
            solution = solution.WithProjectOutputFilePath(document.Project.Id, moduleFile.Path);
            _mockCompilationOutputsProvider = _ => new CompilationOutputFiles(moduleFile.Path);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // update the document
            var document1 = solution.GetDocument(document.Id);
            solution = solution.WithDocumentText(document.Id, CreateText(source));
            var document2 = solution.GetDocument(document.Id);
 
            Assert.Equal(document1.Id, document2.Id);
            Assert.NotSame(document1, document2);
 
            var diagnostics2 = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics2);
 
            // validate solution update status and emit - changes made during run mode are ignored:
            var (updates, _) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=0|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1"
            }, _telemetryLog);
        }
 
        [Theory]
        [CombinatorialData]
        public async Task ProjectThatDoesNotSupportEnC(bool breakMode)
        {
            using var _ = CreateWorkspace(out var solution, out var service, new[] { typeof(NoCompilationLanguageService) });
            var project = solution.AddProject("dummy_proj", "dummy_proj", NoCompilationConstants.LanguageName);
            var document = project.AddDocument("test", CreateText("dummy1"));
            solution = document.Project.Solution;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
            if (breakMode)
            {
                EnterBreakState(debuggingSession);
            }
 
            // no changes:
            var document1 = solution.Projects.Single().Documents.Single();
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document1, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // change the source:
            solution = solution.WithDocumentText(document1.Id, CreateText("dummy2"));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            var document2 = solution.GetDocument(document1.Id);
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
        }
 
        [Fact]
        public async Task DesignTimeOnlyDocument()
        {
            var moduleFile = Temp.CreateFile().WriteAllBytes(TestResources.Basic.Members);
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, "class C1 { void M() { System.Console.WriteLine(1); } }");
 
            var documentInfo = CreateDesignTimeOnlyDocument(document1.Project.Id);
            solution = solution.WithProjectOutputFilePath(document1.Project.Id, moduleFile.Path).AddDocument(documentInfo);
 
            _mockCompilationOutputsProvider = _ => new CompilationOutputFiles(moduleFile.Path);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // update a design-time-only source file:
            solution = solution.WithDocumentText(documentInfo.Id, CreateText("class UpdatedC2 {}"));
            var document2 = solution.GetDocument(documentInfo.Id);
 
            // no updates:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // validate solution update status and emit - changes made in design-time-only documents are ignored:
            var (updates, _) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=0|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1"
            }, _telemetryLog);
        }
 
        [Fact]
        public async Task DesignTimeOnlyDocument_Dynamic()
        {
            using var _ = CreateWorkspace(out var solution, out var service);
 
            (solution, var document) = AddDefaultTestProject(solution, "class C {}");
 
            var sourceText = CreateText("class D {}");
            var documentInfo = DocumentInfo.Create(
                DocumentId.CreateNewId(document.Project.Id),
                name: "design-time-only.cs",
                loader: TextLoader.From(TextAndVersion.Create(sourceText, VersionStamp.Create(), "design-time-only.cs")),
                filePath: "design-time-only.cs",
                isGenerated: false)
                .WithDesignTimeOnly(true);
 
            solution = solution.AddDocument(documentInfo);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
            EnterBreakState(debuggingSession);
 
            // change the source:
            var document1 = solution.GetDocument(documentInfo.Id);
            solution = solution.WithDocumentText(document1.Id, CreateText("class E {}"));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
        }
 
        [Theory]
        [CombinatorialData]
        public async Task DesignTimeOnlyDocument_Wpf([CombinatorialValues(LanguageNames.CSharp, LanguageNames.VisualBasic)] string language, bool delayLoad, bool open, bool designTimeOnlyAddedAfterSessionStarts)
        {
            var source = "class A { }";
            var sourceDesignTimeOnly = (language == LanguageNames.CSharp) ? "class B { }" : "Class C : End Class";
            var sourceDesignTimeOnly2 = (language == LanguageNames.CSharp) ? "class B2 { }" : "Class C2 : End Class";
 
            var dir = Temp.CreateDirectory();
 
            var extension = (language == LanguageNames.CSharp) ? ".cs" : ".vb";
 
            var sourceFileName = "a" + extension;
            var sourceFilePath = dir.CreateFile(sourceFileName).WriteAllText(source, Encoding.UTF8).Path;
 
            var designTimeOnlyFileName = "b.g.i" + extension;
            var designTimeOnlyFilePath = Path.Combine(dir.Path, designTimeOnlyFileName);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            // The workspace starts with 
            // [added == false] a version of the source that's not updated with the output of single file generator (or design-time build):
            // [added == true] without the output of single file generator (design-time build has not completed)
 
            var projectId = ProjectId.CreateNewId();
            var documentId = DocumentId.CreateNewId(projectId);
            var designTimeOnlyDocumentId = DocumentId.CreateNewId(projectId);
 
            solution = solution.
                AddProject(projectId, "test", "test", language).
                AddMetadataReferences(projectId, TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument(documentId, sourceFileName, CreateText(source), filePath: sourceFilePath);
 
            if (!designTimeOnlyAddedAfterSessionStarts)
            {
                solution = solution.AddDocument(designTimeOnlyDocumentId, designTimeOnlyFileName, SourceText.From(sourceDesignTimeOnly, Encoding.UTF8), filePath: designTimeOnlyFilePath);
            }
 
            // only compile actual source document, not design-time-only document:
            var moduleId = EmitLibrary(source, sourceFilePath: sourceFilePath);
 
            if (!delayLoad)
            {
                LoadLibraryToDebuggee(moduleId);
            }
 
            // make sure renames are not supported:
            _debuggerService.GetCapabilitiesImpl = () => ImmutableArray.Create("Baseline");
 
            var openDocumentIds = open ? ImmutableArray.Create(designTimeOnlyDocumentId) : ImmutableArray<DocumentId>.Empty;
            var sessionId = await service.StartDebuggingSessionAsync(solution, _debuggerService, NullPdbMatchingSourceTextProvider.Instance, captureMatchingDocuments: openDocumentIds, captureAllMatchingDocuments: false, reportDiagnostics: true, CancellationToken.None);
            var debuggingSession = service.GetTestAccessor().GetDebuggingSession(sessionId);
 
            if (designTimeOnlyAddedAfterSessionStarts)
            {
                solution = solution.AddDocument(designTimeOnlyDocumentId, designTimeOnlyFileName, SourceText.From(sourceDesignTimeOnly, Encoding.UTF8), filePath: designTimeOnlyFilePath);
            }
 
            var activeLineSpan = new LinePositionSpan(new(0, 0), new(0, 1));
            var activeStatements = ImmutableArray.Create(
                new ManagedActiveStatementDebugInfo(
                    new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000001, version: 1), ilOffset: 1),
                    designTimeOnlyFilePath,
                    activeLineSpan.ToSourceSpan(),
                    ActiveStatementFlags.NonLeafFrame | ActiveStatementFlags.MethodUpToDate));
 
            EnterBreakState(debuggingSession, activeStatements);
 
            // change the source (rude edit):
            solution = solution.WithDocumentText(designTimeOnlyDocumentId, CreateText(sourceDesignTimeOnly2));
 
            var designTimeOnlyDocument2 = solution.GetDocument(designTimeOnlyDocumentId);
 
            Assert.False(designTimeOnlyDocument2.State.SupportsEditAndContinue());
            Assert.True(designTimeOnlyDocument2.Project.SupportsEditAndContinue());
 
            var activeStatementMap = await debuggingSession.EditSession.BaseActiveStatements.GetValueAsync(CancellationToken.None);
            Assert.NotEmpty(activeStatementMap.DocumentPathMap);
 
            // Active statements in design-time documents should be left unchanged.
            var asSpans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(designTimeOnlyDocumentId), CancellationToken.None);
            Assert.Empty(asSpans.Single());
 
            // no Rude Edits reported:
            Assert.Empty(await service.GetDocumentDiagnosticsAsync(designTimeOnlyDocument2, s_noActiveSpans, CancellationToken.None));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(emitDiagnostics);
 
            if (delayLoad)
            {
                LoadLibraryToDebuggee(moduleId);
 
                // validate solution update status and emit:
                (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
                Assert.Equal(ModuleUpdateStatus.None, updates.Status);
                Assert.Empty(emitDiagnostics);
            }
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Theory]
        [CombinatorialData]
        public async Task ErrorReadingModuleFile(bool breakMode)
        {
            // module file is empty, which will cause a read error:
            var moduleFile = Temp.CreateFile();
 
            string expectedErrorMessage = null;
            try
            {
                using var stream = File.OpenRead(moduleFile.Path);
                using var peReader = new PEReader(stream);
                _ = peReader.GetMetadataReader();
            }
            catch (Exception e)
            {
                expectedErrorMessage = e.Message;
            }
 
            var source1 = "class C { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class C { void M() { System.Console.WriteLine(2); } }";
 
            using var _w = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, source1);
 
            _mockCompilationOutputsProvider = _ => new CompilationOutputFiles(moduleFile.Path);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            if (breakMode)
            {
                EnterBreakState(debuggingSession);
            }
 
            // change the source:
            solution = solution.WithDocumentText(document1.Id, CreateText(source2));
            var document2 = solution.GetDocument(document1.Id);
 
            // error not reported here since it might be intermittent and will be reported if the issue persist when applying the update:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            AssertEx.Equal(new[] { $"{document2.Project.Id}: Error ENC1001: {string.Format(FeaturesResources.ErrorReadingFile, moduleFile.Path, expectedErrorMessage)}" }, InspectDiagnostics(emitDiagnostics));
 
            // correct the error:
            EmitLibrary(source2);
 
            var (updates2, emitDiagnostics2) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Ready, updates2.Status);
            Assert.Empty(emitDiagnostics2);
 
            CommitSolutionUpdate(debuggingSession);
 
            if (breakMode)
            {
                ExitBreakState(debuggingSession);
            }
 
            EndDebuggingSession(debuggingSession);
 
            if (breakMode)
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=3",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=1|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges={00000000-AAAA-AAAA-AAAA-111111111111}",
                    "Debugging_EncSession_EditSession_EmitDeltaErrorId: SessionId=1|EditSessionId=2|ErrorId=ENC1001"
                }, _telemetryLog);
            }
            else
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=0|EmptySessionCount=0|HotReloadSessionCount=1|EmptyHotReloadSessionCount=1",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=1|InBreakState=False|Capabilities=31|ProjectIdsWithAppliedChanges={00000000-AAAA-AAAA-AAAA-111111111111}",
                    "Debugging_EncSession_EditSession_EmitDeltaErrorId: SessionId=1|EditSessionId=2|ErrorId=ENC1001"
                }, _telemetryLog);
            }
        }
 
        [Fact]
        public async Task ErrorReadingPdbFile()
        {
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("a.cs").WriteAllText(source1, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            var document1 = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument("a.cs", CreateText(source1), filePath: sourceFile.Path);
 
            var project = document1.Project;
            solution = project.Solution;
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path);
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(moduleId)
            {
                OpenPdbStreamImpl = () =>
                {
                    throw new IOException("Error");
                }
            };
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
            EnterBreakState(debuggingSession);
 
            // change the source:
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M() { System.Console.WriteLine(2); } }"));
            var document2 = solution.GetDocument(document1.Id);
 
            // error not reported here since it might be intermittent and will be reported if the issue persist when applying the update:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // an error occurred so we need to call update to determine whether we have changes to apply or not:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            AssertEx.Equal(new[] { $"{project.Id}: Warning ENC1006: {string.Format(FeaturesResources.UnableToReadSourceFileOrPdb, sourceFile.Path)}" }, InspectDiagnostics(emitDiagnostics));
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=0|EmptySessionCount=1|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1"
            }, _telemetryLog);
        }
 
        [Fact]
        public async Task ErrorReadingSourceFile()
        {
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("a.cs").WriteAllText(source1, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            var document1 = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(DefaultTargetFramework)).
                AddDocument("a.cs", SourceText.From(source1, Encoding.UTF8, SourceHashAlgorithm.Sha1), filePath: sourceFile.Path);
 
            var project = document1.Project;
            solution = project.Solution;
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path, checksumAlgorithm: SourceHashAlgorithms.Default);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
            EnterBreakState(debuggingSession);
 
            // change the source:
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M() { System.Console.WriteLine(2); } }"));
            var document2 = solution.GetDocument(document1.Id);
 
            using var fileLock = File.Open(sourceFile.Path, FileMode.Open, FileAccess.Read, FileShare.None);
 
            // error not reported here since it might be intermittent and will be reported if the issue persist when applying the update:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // an error occurred so we need to call update to determine whether we have changes to apply or not:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            AssertEx.Equal(new[] { $"{project.Id}: Warning ENC1006: {string.Format(FeaturesResources.UnableToReadSourceFileOrPdb, sourceFile.Path)}" }, InspectDiagnostics(emitDiagnostics));
 
            fileLock.Dispose();
 
            // try apply changes again:
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            Assert.NotEmpty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1",
                "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=0|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges="
            }, _telemetryLog);
        }
 
        [Theory]
        [CombinatorialData]
        public async Task FileAdded(bool breakMode)
        {
            var sourceA = "class C1 { void M() { System.Console.WriteLine(1); } }";
            var sourceB = "class C2 {}";
 
            var sourceFileA = Temp.CreateFile().WriteAllText(sourceA, Encoding.UTF8);
            var sourceFileB = Temp.CreateFile().WriteAllText(sourceB, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            var documentA = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument("test.cs", CreateText(sourceA), filePath: sourceFileA.Path);
 
            solution = documentA.Project.Solution;
 
            // Source B will be added while debugging.
            EmitAndLoadLibraryToDebuggee(sourceA, sourceFilePath: sourceFileA.Path);
 
            var project = documentA.Project;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            if (breakMode)
            {
                EnterBreakState(debuggingSession);
            }
 
            // add a source file:
            var documentB = project.AddDocument("file2.cs", CreateText(sourceB), filePath: sourceFileB.Path);
            solution = documentB.Project.Solution;
            documentB = solution.GetDocument(documentB.Id);
 
            var diagnostics2 = await service.GetDocumentDiagnosticsAsync(documentB, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics2);
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            debuggingSession.DiscardSolutionUpdate();
 
            if (breakMode)
            {
                ExitBreakState(debuggingSession);
            }
 
            EndDebuggingSession(debuggingSession);
 
            if (breakMode)
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=2",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=0|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges="
                }, _telemetryLog);
            }
            else
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=0|EmptySessionCount=0|HotReloadSessionCount=1|EmptyHotReloadSessionCount=0",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=0|InBreakState=False|Capabilities=31|ProjectIdsWithAppliedChanges="
                }, _telemetryLog);
            }
        }
 
        /// <summary>
        /// <code>
        ///                         F5   build
        ///                              complete
        ///                         │    │
        /// Workspace    ═════0═════╪════╪══════════1═══
        ///                   ▲     │               ▲ src file watcher
        ///                   │     │               │
        /// dll/pdb      ═0═══╪═════╪════1══════════╪═══
        ///                   │     │    ▲          │
        ///               ┌───┘     │    │          │
        ///               │      ┌──┼────┴──────────┘
        /// Source file  ═0══════1══╪═══════════════════
        ///
        /// Committed    ═══════════╪════0══════════1═══
        /// solution
        /// </code>
        /// </summary>
        [Theory]
        [CombinatorialData]
        public async Task ModuleDisallowsEditAndContinue_NoChanges(bool breakMode)
        {
            var source0 = "class C1 { void M() { System.Console.WriteLine(0); } }";
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("a.cs");
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            var project = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40));
 
            solution = project.Solution;
 
            // compile with source1:
            var moduleId = EmitLibrary(source1, sourceFilePath: sourceFile.Path);
            LoadLibraryToDebuggee(moduleId, new ManagedHotReloadAvailability(ManagedHotReloadAvailabilityStatus.NotAllowedForRuntime, "*message*"));
 
            // update the file with source1 before session starts:
            sourceFile.WriteAllText(source1, Encoding.UTF8);
 
            // source0 is loaded to workspace before session starts:
            var document0 = project.AddDocument("a.cs", CreateText(source0), filePath: sourceFile.Path);
            solution = document0.Project.Solution;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
 
            if (breakMode)
            {
                EnterBreakState(debuggingSession);
            }
 
            // workspace is updated to new version after build completed and the session started:
            solution = solution.WithDocumentText(document0.Id, CreateText(source1));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            if (breakMode)
            {
                ExitBreakState(debuggingSession);
            }
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ModuleDisallowsEditAndContinue_SourceGenerator_NoChanges()
        {
            var moduleId = Guid.NewGuid();
 
            var source1 = @"/* GENERATE class C1 { void M() { System.Console.WriteLine(1); } } */";
            var source2 = source1;
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source1, generator);
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(moduleId);
 
            LoadLibraryToDebuggee(moduleId, new ManagedHotReloadAvailability(ManagedHotReloadAvailabilityStatus.NotAllowedForRuntime, "*message*"));
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // update document with the same content:
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, CreateText(source2));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ModuleDisallowsEditAndContinue()
        {
            var moduleId = Guid.NewGuid();
 
            var source1 = @"
class C1
{
  void M()
  {
    System.Console.WriteLine(1);
    System.Console.WriteLine(2);
    System.Console.WriteLine(3);
  }
}";
            var source2 = @"
class C1
{
  void M()
  {
    System.Console.WriteLine(9);
    System.Console.WriteLine();
    System.Console.WriteLine(30);
  }
}";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source1, generator);
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(moduleId);
 
            LoadLibraryToDebuggee(moduleId, new ManagedHotReloadAvailability(ManagedHotReloadAvailabilityStatus.NotAllowedForRuntime, "*message*"));
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source:
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, CreateText(source2));
            var document2 = solution.GetDocument(document1.Id);
 
            // We do not report module diagnostics until emit.
            // This is to make the analysis deterministic (not dependent on the current state of the debuggee).
            var diagnostics1 = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics1);
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            AssertEx.Equal(new[] { $"{document2.FilePath}: (5,0)-(5,32): Error ENC2016: {string.Format(FeaturesResources.EditAndContinueDisallowedByProject, document2.Project.Name, "*message*")}" }, InspectDiagnostics(emitDiagnostics));
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.SetEqual(new[] { moduleId }, debuggingSession.GetTestAccessor().GetModulesPreparedForUpdate());
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1",
                "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=1|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges=",
                "Debugging_EncSession_EditSession_EmitDeltaErrorId: SessionId=1|EditSessionId=2|ErrorId=ENC2016"
            }, _telemetryLog);
        }
 
        private class TestSourceTextContainer : SourceTextContainer
        {
            public SourceText Text { get; set; }
 
            public override SourceText CurrentText => Text;
 
#pragma warning disable CS0067
            public override event EventHandler<TextChangeEventArgs> TextChanged;
#pragma warning restore
        }
 
        [Fact]
        public async Task Encodings()
        {
            var source1 = "class C1 { void M() { System.Console.WriteLine(\"ã\"); } }";
 
            var encoding = Encoding.GetEncoding(1252);
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("test.cs").WriteAllText(source1, encoding);
 
            using var workspace = CreateWorkspace(out var solution, out var service);
 
            var projectId = ProjectId.CreateNewId();
            var documentId = DocumentId.CreateNewId(projectId);
 
            solution = solution.
                AddProject(projectId, "test", "test", LanguageNames.CSharp).
                WithProjectChecksumAlgorithm(projectId, SourceHashAlgorithm.Sha1).
                AddMetadataReferences(projectId, TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument(documentId, "test.cs", SourceText.From(source1, encoding, SourceHashAlgorithm.Sha1), filePath: sourceFile.Path);
 
            // use different checksum alg to trigger PdbMatchingSourceTextProvider call:
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path, encoding: encoding, checksumAlgorithm: SourceHashAlgorithm.Sha256);
 
            var sourceTextProviderCalled = false;
            var sourceTextProvider = new MockPdbMatchingSourceTextProvider()
            {
                TryGetMatchingSourceTextImpl = (filePath, requiredChecksum, checksumAlgorithm) =>
                {
                    sourceTextProviderCalled = true;
 
                    // fall back to reading the file content:
                    return null;
                }
            };
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None, sourceTextProvider);
 
            EnterBreakState(debuggingSession);
 
            var (document, state) = await debuggingSession.LastCommittedSolution.GetDocumentAndStateAsync(documentId, currentDocument: null, CancellationToken.None);
            var text = await document.GetTextAsync();
            Assert.Same(encoding, text.Encoding);
            Assert.Equal(CommittedSolution.DocumentState.MatchesBuildOutput, state);
 
            Assert.True(sourceTextProviderCalled);
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Theory]
        [CombinatorialData]
        public async Task RudeEdits(bool breakMode)
        {
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class C1 { void M<T>() { System.Console.WriteLine(1); } }";
 
            var moduleId = Guid.NewGuid();
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source1);
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            if (breakMode)
            {
                EnterBreakState(debuggingSession);
            }
 
            // change the source (rude edit):
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, CreateText(source2));
            var document2 = solution.GetDocument(document1.Id);
 
            var diagnostics1 = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[] { "ENC0021: " + string.Format(FeaturesResources.Adding_0_requires_restarting_the_application, FeaturesResources.type_parameter) },
                diagnostics1.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            if (breakMode)
            {
                ExitBreakState(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
                EndDebuggingSession(debuggingSession);
            }
            else
            {
                EndDebuggingSession(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
            }
 
            AssertEx.SetEqual(new[] { moduleId }, debuggingSession.GetTestAccessor().GetModulesPreparedForUpdate());
 
            if (breakMode)
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=2",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=True|HadValidChanges=False|HadValidInsignificantChanges=False|RudeEditsCount=1|EmitDeltaErrorIdCount=0|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges=",
                    "Debugging_EncSession_EditSession_RudeEdit: SessionId=1|EditSessionId=2|RudeEditKind=21|RudeEditSyntaxKind=8910|RudeEditBlocking=True"
                }, _telemetryLog);
            }
            else
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=0|EmptySessionCount=0|HotReloadSessionCount=1|EmptyHotReloadSessionCount=0",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=True|HadValidChanges=False|HadValidInsignificantChanges=False|RudeEditsCount=1|EmitDeltaErrorIdCount=0|InBreakState=False|Capabilities=31|ProjectIdsWithAppliedChanges=",
                    "Debugging_EncSession_EditSession_RudeEdit: SessionId=1|EditSessionId=2|RudeEditKind=21|RudeEditSyntaxKind=8910|RudeEditBlocking=True"
                }, _telemetryLog);
            }
        }
 
        [Fact]
        public async Task DeferredApplyChangeWithActiveStatementRudeEdits()
        {
            var source1 = "class C { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class C { void M() { System.Console.WriteLine(2); } }";
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1);
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source1);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            var activeLineSpan1 = CreateText(source1).Lines.GetLinePositionSpan(GetSpan(source1, "System.Console.WriteLine(1);"));
            var activeLineSpan2 = CreateText(source2).Lines.GetLinePositionSpan(GetSpan(source2, "System.Console.WriteLine(2);"));
 
            var activeStatements = ImmutableArray.Create(
                new ManagedActiveStatementDebugInfo(
                    new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000001, version: 1), ilOffset: 1),
                    document.FilePath,
                    activeLineSpan1.ToSourceSpan(),
                    ActiveStatementFlags.NonLeafFrame | ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.PartiallyExecuted));
 
            EnterBreakState(debuggingSession, activeStatements);
 
            // change the source (rude edit):
            solution = solution.WithDocumentText(document.Id, CreateText(source2));
            var document2 = solution.GetDocument(document.Id);
 
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[] { "ENC0001: " + string.Format(FeaturesResources.Updating_an_active_statement_requires_restarting_the_application) },
                diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            // exit break state without applying the change:
            ExitBreakState(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            // enter break state again (with the same active statements)
 
            EnterBreakState(debuggingSession, activeStatements);
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[] { "ENC0001: " + string.Format(FeaturesResources.Updating_an_active_statement_requires_restarting_the_application) },
                diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            // exit break state without applying the change:
            ExitBreakState(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
 
            // apply the change:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            Assert.NotEmpty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            CommitSolutionUpdate(debuggingSession);
 
            EnterBreakState(debuggingSession, activeStatements);
 
            // no rude edits - changes have been applied
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            ExitBreakState(debuggingSession);
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task RudeEdits_SourceGenerators()
        {
            var sourceV1 = @"
/* GENERATE: class G { int X1() => 1; } */
 
class C { int Y => 1; }
";
            var sourceV2 = @"
/* GENERATE: class G { int X1<T>() => 1; } */
 
class C { int Y => 2; }
";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, sourceV1, generator: generator);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
            EnterBreakState(debuggingSession);
 
            // change the source:
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, CreateText(sourceV2));
 
            var generatedDocument = (await solution.Projects.Single().GetSourceGeneratedDocumentsAsync()).Single();
 
            var diagnostics1 = await service.GetDocumentDiagnosticsAsync(generatedDocument, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[] { "ENC0021: " + string.Format(FeaturesResources.Adding_0_requires_restarting_the_application, FeaturesResources.type_parameter) },
                diagnostics1.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            EndDebuggingSession(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(generatedDocument.Id));
        }
 
        [Theory]
        [CombinatorialData]
        public async Task RudeEdits_DocumentOutOfSync(bool breakMode)
        {
            var source0 = "class C1 { void M() { System.Console.WriteLine(0); } }";
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class C1 { void M<T>() { System.Console.WriteLine(1); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("a.cs");
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            var project = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40));
 
            solution = project.Solution;
 
            // compile with source0:
            var moduleId = EmitAndLoadLibraryToDebuggee(source0, sourceFilePath: sourceFile.Path);
 
            // update the file with source1 before session starts:
            sourceFile.WriteAllText(source1, Encoding.UTF8);
 
            // source1 is reflected in workspace before session starts:
            var document1 = project.AddDocument("a.cs", CreateText(source1), filePath: sourceFile.Path);
            solution = document1.Project.Solution;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
 
            if (breakMode)
            {
                EnterBreakState(debuggingSession);
            }
 
            // change the source (rude edit):
            solution = solution.WithDocumentText(document1.Id, CreateText(source2));
            var document2 = solution.GetDocument(document1.Id);
 
            // no Rude Edits, since the document is out-of-sync
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // since the document is out-of-sync we need to call update to determine whether we have changes to apply or not:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            AssertEx.Equal(new[] { $"{project.Id}: Warning ENC1005: {string.Format(FeaturesResources.DocumentIsOutOfSyncWithDebuggee, sourceFile.Path)}" }, InspectDiagnostics(emitDiagnostics));
 
            // update the file to match the build:
            sourceFile.WriteAllText(source0, Encoding.UTF8);
 
            // we do not reload the content of out-of-sync file for analyzer query:
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // debugger query will trigger reload of out-of-sync file content:
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            // now we see the rude edit:
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[]
                {
                    "ENC0038: " + FeaturesResources.Modifying_a_method_inside_the_context_of_a_generic_type_requires_restarting_the_application,
                    "ENC0021: " + string.Format(FeaturesResources.Adding_0_requires_restarting_the_application, FeaturesResources.type_parameter)
                },
                diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            if (breakMode)
            {
                ExitBreakState(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
                EndDebuggingSession(debuggingSession);
            }
            else
            {
                EndDebuggingSession(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
            }
 
            AssertEx.SetEqual(new[] { moduleId }, debuggingSession.GetTestAccessor().GetModulesPreparedForUpdate());
 
            if (breakMode)
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=2",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=True|HadValidChanges=False|HadValidInsignificantChanges=False|RudeEditsCount=2|EmitDeltaErrorIdCount=0|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges=",
                    "Debugging_EncSession_EditSession_RudeEdit: SessionId=1|EditSessionId=2|RudeEditKind=38|RudeEditSyntaxKind=8875|RudeEditBlocking=True",
                    "Debugging_EncSession_EditSession_RudeEdit: SessionId=1|EditSessionId=2|RudeEditKind=21|RudeEditSyntaxKind=8910|RudeEditBlocking=True"
                }, _telemetryLog);
            }
            else
            {
                AssertEx.Equal(new[]
                {
                    "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=0|EmptySessionCount=0|HotReloadSessionCount=1|EmptyHotReloadSessionCount=0",
                    "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=True|HadValidChanges=False|HadValidInsignificantChanges=False|RudeEditsCount=2|EmitDeltaErrorIdCount=0|InBreakState=False|Capabilities=31|ProjectIdsWithAppliedChanges=",
                    "Debugging_EncSession_EditSession_RudeEdit: SessionId=1|EditSessionId=2|RudeEditKind=38|RudeEditSyntaxKind=8875|RudeEditBlocking=True",
                    "Debugging_EncSession_EditSession_RudeEdit: SessionId=1|EditSessionId=2|RudeEditKind=21|RudeEditSyntaxKind=8910|RudeEditBlocking=True"
                }, _telemetryLog);
            }
        }
 
        [Fact]
        public async Task RudeEdits_DocumentWithoutSequencePoints()
        {
            var source1 = "abstract class C { public abstract void M(); }";
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("a.cs").WriteAllText(source1, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            // the workspace starts with a version of the source that's not updated with the output of single file generator (or design-time build):
            var document1 = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument("test.cs", CreateText(source1), filePath: sourceFile.Path);
 
            var project = document1.Project;
            solution = project.Solution;
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path);
 
            // do not initialize the document state - we will detect the state based on the PDB content.
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
 
            EnterBreakState(debuggingSession);
 
            // change the source (rude edit since the base document content matches the PDB checksum, so the document is not out-of-sync):
            solution = solution.WithDocumentText(document1.Id, CreateText("abstract class C { public abstract void M(); public abstract void N(); }"));
            var document2 = solution.Projects.Single().Documents.Single();
 
            // Rude Edits reported:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(
                new[] { "ENC0023: " + string.Format(FeaturesResources.Adding_an_abstract_0_or_overriding_an_inherited_0_requires_restarting_the_application, FeaturesResources.method) },
                diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            EndDebuggingSession(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
        }
 
        [Fact]
        public async Task RudeEdits_DelayLoadedModule()
        {
            var source1 = "class C { public void M() { } }";
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("a.cs").WriteAllText(source1, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            // the workspace starts with a version of the source that's not updated with the output of single file generator (or design-time build):
            var document1 = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument("test.cs", CreateText(source1), filePath: sourceFile.Path);
 
            var project = document1.Project;
            solution = project.Solution;
 
            var moduleId = EmitLibrary(source1, sourceFilePath: sourceFile.Path);
 
            // do not initialize the document state - we will detect the state based on the PDB content.
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
 
            EnterBreakState(debuggingSession);
 
            // change the source (rude edit) before the library is loaded:
            solution = solution.WithDocumentText(document1.Id, CreateText("class C { public void M<T>() { } }"));
            var document2 = solution.Projects.Single().Documents.Single();
 
            // Rude Edits reported:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(
                new[] { "ENC0021: " + string.Format(FeaturesResources.Adding_0_requires_restarting_the_application, FeaturesResources.type_parameter) },
                diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            // load library to the debuggee:
            LoadLibraryToDebuggee(moduleId);
 
            // Rude Edits still reported:
            diagnostics = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(
                new[] { "ENC0021: " + string.Format(FeaturesResources.Adding_0_requires_restarting_the_application, FeaturesResources.type_parameter) },
                diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            EndDebuggingSession(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(document2.Id));
        }
 
        [Fact]
        public async Task SyntaxError()
        {
            var moduleId = Guid.NewGuid();
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, "class C1 { void M() { System.Console.WriteLine(1); } }");
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source (compilation error):
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M() { "));
            var document2 = solution.Projects.Single().Documents.Single();
 
            // compilation errors are not reported via EnC diagnostic analyzer:
            var diagnostics1 = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics1);
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Blocked, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.SetEqual(new[] { moduleId }, debuggingSession.GetTestAccessor().GetModulesPreparedForUpdate());
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1",
                "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=True|HadRudeEdits=False|HadValidChanges=False|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=0|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges="
            }, _telemetryLog);
        }
 
        [Fact]
        public async Task SemanticError()
        {
            var sourceV1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, sourceV1);
 
            var moduleId = EmitAndLoadLibraryToDebuggee(sourceV1);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source (compilation error):
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M() { int i = 0L; System.Console.WriteLine(i); } }"));
            var document2 = solution.Projects.Single().Documents.Single();
 
            // compilation errors are not reported via EnC diagnostic analyzer:
            var diagnostics1 = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics1);
 
            // The EnC analyzer does not check for and block on all semantic errors as they are already reported by diagnostic analyzer.
            // Blocking update on semantic errors would be possible, but the status check is only an optimization to avoid emitting.
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Blocked, updates.Status);
            Assert.Empty(updates.Updates);
 
            // TODO: https://github.com/dotnet/roslyn/issues/36061
            // Semantic errors should not be reported in emit diagnostics.
 
            AssertEx.Equal(new[] { $"{document2.FilePath}: (0,30)-(0,32): Error CS0266: {string.Format(CSharpResources.ERR_NoImplicitConvCast, "long", "int")}" }, InspectDiagnostics(emitDiagnostics));
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.SetEqual(new[] { moduleId }, debuggingSession.GetTestAccessor().GetModulesPreparedForUpdate());
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1",
                "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=1|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges=",
                "Debugging_EncSession_EditSession_EmitDeltaErrorId: SessionId=1|EditSessionId=2|ErrorId=CS0266"
            }, _telemetryLog);
        }
 
        [Fact]
        public async Task HasChanges()
        {
            using var _ = CreateWorkspace(out var solution, out var service);
 
            var pathA = Path.Combine(TempRoot.Root, "A.cs");
            var pathB = Path.Combine(TempRoot.Root, "B.cs");
            var pathC = Path.Combine(TempRoot.Root, "C.cs");
            var pathD = Path.Combine(TempRoot.Root, "D.cs");
            var pathX = Path.Combine(TempRoot.Root, "X");
            var pathY = Path.Combine(TempRoot.Root, "Y");
            var pathCommon = Path.Combine(TempRoot.Root, "Common.cs");
 
            solution = solution.
                AddProject("A", "A", "C#").
                AddDocument("A.cs", "class Program { void Main() { System.Console.WriteLine(1); } }", filePath: pathA).Project.Solution.
                AddProject("B", "B", "C#").
                AddDocument("Common.cs", "class Common {}", filePath: pathCommon).Project.
                AddDocument("B.cs", "class B {}", filePath: pathB).Project.Solution.
                AddProject("C", "C", "C#").
                AddDocument("Common.cs", "class Common {}", filePath: pathCommon).Project.
                AddDocument("C.cs", "class C {}", filePath: pathC).Project.Solution;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
            EnterBreakState(debuggingSession);
 
            // change C.cs to have a compilation error:
            var oldSolution = solution;
            var projectC = solution.GetProjectsByName("C").Single();
            var documentC = projectC.Documents.Single(d => d.Name == "C.cs");
            solution = solution.WithDocumentText(documentC.Id, CreateText("class C { void M() { "));
 
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, CancellationToken.None));
 
            Assert.False(await EditSession.HasChangesAsync(oldSolution, solution, sourceFilePath: pathCommon, CancellationToken.None));
            Assert.False(await EditSession.HasChangesAsync(oldSolution, solution, sourceFilePath: pathB, CancellationToken.None));
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, sourceFilePath: pathC, CancellationToken.None));
            Assert.False(await EditSession.HasChangesAsync(oldSolution, solution, sourceFilePath: "NonexistentFile.cs", CancellationToken.None));
 
            // All projects must have no errors.
            var (updates, _) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Blocked, updates.Status);
 
            // add a project:
 
            oldSolution = solution;
            var projectD = solution.AddProject("D", "D", "C#");
            solution = projectD.Solution;
 
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, CancellationToken.None));
 
            // remove a project:
            Assert.True(await EditSession.HasChangesAsync(solution, solution.RemoveProject(projectD.Id), CancellationToken.None));
 
            EndDebuggingSession(debuggingSession);
        }
 
        public enum DocumentKind
        {
            Source,
            Additional,
            AnalyzerConfig,
        }
 
        [Theory]
        [CombinatorialData]
        public async Task HasChanges_Documents(DocumentKind documentKind)
        {
            using var _ = CreateWorkspace(out var solution, out var service);
 
            var pathX = Path.Combine(TempRoot.Root, "X.cs");
            var pathA = Path.Combine(TempRoot.Root, "A.cs");
 
            var generatorExecutionCount = 0;
            var generator = new TestSourceGenerator()
            {
                ExecuteImpl = context =>
                {
                    switch (documentKind)
                    {
                        case DocumentKind.Source:
                            context.AddSource("Generated.cs", context.Compilation.SyntaxTrees.SingleOrDefault(t => t.FilePath.EndsWith("X.cs"))?.ToString() ?? "none");
                            break;
 
                        case DocumentKind.Additional:
                            context.AddSource("Generated.cs", context.AdditionalFiles.FirstOrDefault()?.GetText().ToString() ?? "none");
                            break;
 
                        case DocumentKind.AnalyzerConfig:
                            var syntaxTree = context.Compilation.SyntaxTrees.Single(t => t.FilePath.EndsWith("A.cs"));
                            var content = context.AnalyzerConfigOptions.GetOptions(syntaxTree).TryGetValue("x", out var optionValue) ? optionValue.ToString() : "none";
 
                            context.AddSource("Generated.cs", content);
                            break;
                    }
 
                    generatorExecutionCount++;
                }
            };
 
            var project = solution.AddProject("A", "A", "C#").AddDocument("A.cs", "", filePath: pathA).Project;
            var projectId = project.Id;
            solution = project.Solution.AddAnalyzerReference(projectId, new TestGeneratorReference(generator));
            project = solution.GetRequiredProject(projectId);
            var generatedDocument = (await project.GetSourceGeneratedDocumentsAsync()).Single();
            var generatedDocumentId = generatedDocument.Id;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
            EnterBreakState(debuggingSession);
 
            Assert.Equal(1, generatorExecutionCount);
            var changedOrAddedDocuments = new PooledObjects.ArrayBuilder<Document>();
 
            //
            // Add document
            //
 
            generatorExecutionCount = 0;
            var oldSolution = solution;
            var documentId = DocumentId.CreateNewId(projectId);
            solution = documentKind switch
            {
                DocumentKind.Source => solution.AddDocument(documentId, "X", CreateText("xxx"), filePath: pathX),
                DocumentKind.Additional => solution.AddAdditionalDocument(documentId, "X", CreateText("xxx"), filePath: pathX),
                DocumentKind.AnalyzerConfig => solution.AddAnalyzerConfigDocument(documentId, "X", GetAnalyzerConfigText(new[] { ("x", "1") }), filePath: pathX),
                _ => throw ExceptionUtilities.Unreachable(),
            };
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, CancellationToken.None));
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, pathX, CancellationToken.None));
 
            // always returns false for source generated files:
            Assert.False(await EditSession.HasChangesAsync(oldSolution, solution, generatedDocument.FilePath, CancellationToken.None));
 
            // generator is not executed since we already know the solution changed without inspecting generated files:
            Assert.Equal(0, generatorExecutionCount);
 
            AssertEx.Equal(new[] { generatedDocumentId },
                await EditSession.GetChangedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), CancellationToken.None).ToImmutableArrayAsync(CancellationToken.None));
 
            await EditSession.PopulateChangedAndAddedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), changedOrAddedDocuments, CancellationToken.None);
            AssertEx.Equal(documentKind == DocumentKind.Source ? new[] { documentId, generatedDocumentId } : new[] { generatedDocumentId }, changedOrAddedDocuments.Select(d => d.Id));
 
            Assert.Equal(1, generatorExecutionCount);
 
            //
            // Update document to a different document snapshot but the same content
            //
 
            generatorExecutionCount = 0;
            oldSolution = solution;
 
            solution = documentKind switch
            {
                DocumentKind.Source => solution.WithDocumentText(documentId, CreateText("xxx")),
                DocumentKind.Additional => solution.WithAdditionalDocumentText(documentId, CreateText("xxx")),
                DocumentKind.AnalyzerConfig => solution.WithAnalyzerConfigDocumentText(documentId, GetAnalyzerConfigText(new[] { ("x", "1") })),
                _ => throw ExceptionUtilities.Unreachable(),
            };
            Assert.False(await EditSession.HasChangesAsync(oldSolution, solution, CancellationToken.None));
            Assert.False(await EditSession.HasChangesAsync(oldSolution, solution, pathX, CancellationToken.None));
 
            Assert.Equal(0, generatorExecutionCount);
 
            // source generator infrastructure compares content and reuses state if it matches (SourceGeneratedDocumentState.WithUpdatedGeneratedContent):
            AssertEx.Equal(documentKind == DocumentKind.Source ? new[] { documentId } : Array.Empty<DocumentId>(),
                await EditSession.GetChangedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), CancellationToken.None).ToImmutableArrayAsync(CancellationToken.None));
 
            await EditSession.PopulateChangedAndAddedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), changedOrAddedDocuments, CancellationToken.None);
            Assert.Empty(changedOrAddedDocuments);
 
            Assert.Equal(1, generatorExecutionCount);
 
            //
            // Update document content
            //
 
            generatorExecutionCount = 0;
            oldSolution = solution;
            solution = documentKind switch
            {
                DocumentKind.Source => solution.WithDocumentText(documentId, CreateText("xxx-changed")),
                DocumentKind.Additional => solution.WithAdditionalDocumentText(documentId, CreateText("xxx-changed")),
                DocumentKind.AnalyzerConfig => solution.WithAnalyzerConfigDocumentText(documentId, GetAnalyzerConfigText(new[] { ("x", "2") })),
                _ => throw ExceptionUtilities.Unreachable(),
            };
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, CancellationToken.None));
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, pathX, CancellationToken.None));
 
            AssertEx.Equal(documentKind == DocumentKind.Source ? new[] { documentId, generatedDocumentId } : new[] { generatedDocumentId },
                await EditSession.GetChangedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), CancellationToken.None).ToImmutableArrayAsync(CancellationToken.None));
 
            await EditSession.PopulateChangedAndAddedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), changedOrAddedDocuments, CancellationToken.None);
            AssertEx.Equal(documentKind == DocumentKind.Source ? new[] { documentId, generatedDocumentId } : new[] { generatedDocumentId }, changedOrAddedDocuments.Select(d => d.Id));
 
            Assert.Equal(1, generatorExecutionCount);
 
            //
            // Remove document
            //
 
            generatorExecutionCount = 0;
            oldSolution = solution;
            solution = documentKind switch
            {
                DocumentKind.Source => solution.RemoveDocument(documentId),
                DocumentKind.Additional => solution.RemoveAdditionalDocument(documentId),
                DocumentKind.AnalyzerConfig => solution.RemoveAnalyzerConfigDocument(documentId),
                _ => throw ExceptionUtilities.Unreachable(),
            };
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, CancellationToken.None));
            Assert.True(await EditSession.HasChangesAsync(oldSolution, solution, pathX, CancellationToken.None));
 
            Assert.Equal(0, generatorExecutionCount);
 
            AssertEx.Equal(new[] { generatedDocumentId },
                await EditSession.GetChangedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), CancellationToken.None).ToImmutableArrayAsync(CancellationToken.None));
 
            await EditSession.PopulateChangedAndAddedDocumentsAsync(oldSolution.GetProject(projectId), solution.GetProject(projectId), changedOrAddedDocuments, CancellationToken.None);
            AssertEx.Equal(new[] { generatedDocumentId }, changedOrAddedDocuments.Select(d => d.Id));
 
            Assert.Equal(1, generatorExecutionCount);
        }
 
        [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/1204")]
        [WorkItem("https://devdiv.visualstudio.com/DevDiv/_workitems/edit/1371694")]
        public async Task Project_Add()
        {
            var sourceA1 = "class A { void M() { System.Console.WriteLine(1); } }";
            var sourceB1 = "class B { int F() => 1; }";
            var sourceB2 = "class B { int G() => 1; }";
            var sourceB3 = "class B { int F() => 2; }";
 
            var dir = Temp.CreateDirectory();
            var sourceFileA = dir.CreateFile("a.cs").WriteAllText(sourceA1, Encoding.UTF8);
            var sourceFileB = dir.CreateFile("b.cs").WriteAllText(sourceB1, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
            solution = AddDefaultTestProject(solution, new[] { sourceA1 });
            var documentA1 = solution.Projects.Single().Documents.Single();
 
            var mvidA = EmitAndLoadLibraryToDebuggee(sourceA1, sourceFilePath: sourceFileA.Path, assemblyName: "A");
            var mvidB = EmitAndLoadLibraryToDebuggee(sourceB1, sourceFilePath: sourceFileB.Path, assemblyName: "B");
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // An active statement may be present in the added file since the file exists in the PDB:
            var activeLineSpanA1 = CreateText(sourceA1).Lines.GetLinePositionSpan(GetSpan(sourceA1, "System.Console.WriteLine(1);"));
            var activeLineSpanB1 = CreateText(sourceB1).Lines.GetLinePositionSpan(GetSpan(sourceB1, "1"));
 
            var activeStatements = ImmutableArray.Create(
                new ManagedActiveStatementDebugInfo(
                    new ManagedInstructionId(new ManagedMethodId(mvidA, token: 0x06000001, version: 1), ilOffset: 1),
                    sourceFileA.Path,
                    activeLineSpanA1.ToSourceSpan(),
                    ActiveStatementFlags.LeafFrame | ActiveStatementFlags.MethodUpToDate),
                new ManagedActiveStatementDebugInfo(
                    new ManagedInstructionId(new ManagedMethodId(mvidB, token: 0x06000001, version: 1), ilOffset: 1),
                    sourceFileB.Path,
                    activeLineSpanB1.ToSourceSpan(),
                    ActiveStatementFlags.LeafFrame | ActiveStatementFlags.MethodUpToDate));
 
            EnterBreakState(debuggingSession, activeStatements);
 
            // add project that matches assembly B and update the document:
 
            var documentB2 = solution.
                AddProject("B", "B", LanguageNames.CSharp).
                AddDocument("b.cs", CreateText(sourceB2), filePath: sourceFileB.Path);
 
            solution = documentB2.Project.Solution;
 
            // TODO: https://github.com/dotnet/roslyn/issues/1204
            // Should return span in document B since the document content matches the PDB.
            var baseSpans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(documentA1.Id, documentB2.Id), CancellationToken.None);
            AssertEx.Equal(new[]
            {
                "<empty>",
                "(0,21)-(0,22)"
            }, baseSpans.Select(spans => spans.IsEmpty ? "<empty>" : string.Join(",", spans.Select(s => s.LineSpan.ToString()))));
 
            var trackedActiveSpans = ImmutableArray.Create(
                new ActiveStatementSpan(1, activeLineSpanB1, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame, unmappedDocumentId: null));
 
            var currentSpans = await debuggingSession.GetAdjustedActiveStatementSpansAsync(documentB2, (_, _, _) => new(trackedActiveSpans), CancellationToken.None);
            // TODO: https://github.com/dotnet/roslyn/issues/1204
            // AssertEx.Equal(trackedActiveSpans, currentSpans);
            Assert.Empty(currentSpans);
 
            Assert.Equal(activeLineSpanB1,
                await debuggingSession.GetCurrentActiveStatementPositionAsync(documentB2.Project.Solution, (_, _, _) => new(trackedActiveSpans), activeStatements[1].ActiveInstruction, CancellationToken.None));
 
            var diagnostics = await service.GetDocumentDiagnosticsAsync(documentB2, s_noActiveSpans, CancellationToken.None);
 
            // TODO: https://github.com/dotnet/roslyn/issues/1204
            //AssertEx.Equal(
            //    new[] { "ENC0020: " + string.Format(FeaturesResources.Renaming_0_requires_restarting_the_application, FeaturesResources.method) },
            //    diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
            Assert.Empty(diagnostics);
 
            // update document with a valid change:
            solution = solution.WithDocumentText(documentB2.Id, CreateText(sourceB3));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
 
            // TODO: https://github.com/dotnet/roslyn/issues/1204
            // verify valid update
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
 
            ExitBreakState(debuggingSession);
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Theory]
        [CombinatorialData]
        public async Task Capabilities(bool breakState)
        {
            var source1 = "class C { void M() { } }";
            var source2 = "[System.Obsolete]class C { void M() { } }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            solution = AddDefaultTestProject(solution, new[] { source1 });
            var documentId = solution.Projects.Single().Documents.Single().Id;
 
            EmitAndLoadLibraryToDebuggee(source1);
 
            // attached to processes that allow updating custom attributes:
            _debuggerService.GetCapabilitiesImpl = () => ImmutableArray.Create("Baseline", "ChangeCustomAttributes");
 
            // F5
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // update document:
            solution = solution.WithDocumentText(documentId, CreateText(source2));
 
            var diagnostics = await service.GetDocumentDiagnosticsAsync(solution.GetDocument(documentId), s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            if (breakState)
            {
                EnterBreakState(debuggingSession);
            }
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(solution.GetDocument(documentId), s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            // attach to additional processes - at least one process that does not allow updating custom attributes:
            if (breakState)
            {
                ExitBreakState(debuggingSession);
            }
 
            _debuggerService.GetCapabilitiesImpl = () => ImmutableArray.Create("Baseline");
 
            if (breakState)
            {
                EnterBreakState(debuggingSession);
            }
            else
            {
                CapabilitiesChanged(debuggingSession);
            }
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(solution.GetDocument(documentId), s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[] { "ENC0101: " + string.Format(FeaturesResources.Updating_the_attributes_of_0_requires_restarting_the_application_because_it_is_not_supported_by_the_runtime, FeaturesResources.class_) },
               diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            if (breakState)
            {
                ExitBreakState(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(documentId));
            }
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(solution.GetDocument(documentId), s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[] { "ENC0101: " + string.Format(FeaturesResources.Updating_the_attributes_of_0_requires_restarting_the_application_because_it_is_not_supported_by_the_runtime, FeaturesResources.class_) },
               diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            // detach from processes that do not allow updating custom attributes:
            _debuggerService.GetCapabilitiesImpl = () => ImmutableArray.Create("Baseline", "ChangeCustomAttributes");
 
            if (breakState)
            {
                EnterBreakState(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(documentId));
            }
            else
            {
                CapabilitiesChanged(debuggingSession, documentsWithRudeEdits: ImmutableArray.Create(documentId));
            }
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(solution.GetDocument(documentId), s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            if (breakState)
            {
                ExitBreakState(debuggingSession);
            }
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                $"Debugging_EncSession: SolutionSessionId={{00000000-AAAA-AAAA-AAAA-000000000000}}|SessionId=1|SessionCount=0|EmptySessionCount={(breakState ? 3 : 0)}|HotReloadSessionCount=0|EmptyHotReloadSessionCount={(breakState ? 4 : 3)}"
            }, _telemetryLog);
        }
 
        [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/56431")]
        public async Task Capabilities_NoTypesEmitted()
        {
            var sourceV1 = @"
/* GENERATE:
class G
{
    int M()
    {
        return 1;
    }
}
*/
";
            var sourceV2 = @"
/* GENERATE:
class G
{
    // a change that won't cause a type to be emitted
    int M()
    {
        return 1;
    }
}
*/
";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1, generator);
 
            var moduleId = EmitLibrary(sourceV1, generator: generator);
            LoadLibraryToDebuggee(moduleId);
 
            // attached to processes that doesn't allow creating new types
            _debuggerService.GetCapabilitiesImpl = () => ImmutableArray.Create("Baseline");
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // change the source
            solution = solution.WithDocumentText(document1.Id, CreateText(sourceV2));
 
            // validate solution update status and emit
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check that no types have been updated. this used to throw
            var delta = updates.Updates.Single();
            Assert.Empty(delta.UpdatedTypes);
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task Capabilities_SynthesizedNewType()
        {
            var source1 = "class C { void M() { } }";
            var source2 = "class C { void M() { var x = new { Goo = 1 }; } }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            solution = AddDefaultTestProject(solution, new[] { source1 });
            var project = solution.Projects.Single();
            solution = solution.WithProjectParseOptions(project.Id, new CSharpParseOptions(LanguageVersion.CSharp10));
            var documentId = solution.Projects.Single().Documents.Single().Id;
 
            EmitAndLoadLibraryToDebuggee(source1);
 
            // attached to processes that doesn't allow creating new types
            _debuggerService.GetCapabilitiesImpl = () => ImmutableArray.Create("Baseline");
 
            // F5
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // update document:
            solution = solution.WithDocumentText(documentId, CreateText(source2));
            var document2 = solution.Projects.Single().Documents.Single();
 
            // These errors aren't reported as document diagnostics
            var diagnostics = await service.GetDocumentDiagnosticsAsync(solution.GetDocument(documentId), s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            // They are reported as emit diagnostics
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            AssertEx.Equal(new[] { $"{document2.Project.Id}: Error ENC1007: {FeaturesResources.ChangesRequiredSynthesizedType}" }, InspectDiagnostics(emitDiagnostics));
 
            // no emitted delta:
            Assert.Empty(updates.Updates);
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_EmitError()
        {
            var sourceV1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            (solution, var document) = AddDefaultTestProject(solution, sourceV1);
            EmitAndLoadLibraryToDebuggee(sourceV1);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source (valid edit but passing no encoding to emulate emit error):
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, SourceText.From("class C1 { void M() { System.Console.WriteLine(2); } }", encoding: null, SourceHashAlgorithms.Default));
            var document2 = solution.Projects.Single().Documents.Single();
 
            var diagnostics1 = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics1);
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            AssertEx.Equal(new[] { $"{document2.FilePath}: (0,0)-(0,54): Error CS8055: {string.Format(CSharpResources.ERR_EncodinglessSyntaxTree)}" }, InspectDiagnostics(emitDiagnostics));
 
            // no emitted delta:
            Assert.Empty(updates.Updates);
 
            // no pending update:
            Assert.Null(debuggingSession.GetTestAccessor().GetPendingSolutionUpdate());
 
            Assert.Throws<InvalidOperationException>(() => debuggingSession.CommitSolutionUpdate(out var _));
            Assert.Throws<InvalidOperationException>(() => debuggingSession.DiscardSolutionUpdate());
 
            // no change in non-remappable regions since we didn't have any active statements:
            Assert.Empty(debuggingSession.EditSession.NonRemappableRegions);
 
            // solution update status after discarding an update (still has update ready):
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Blocked, updates.Status);
            AssertEx.Equal(new[] { $"{document2.FilePath}: (0,0)-(0,54): Error CS8055: {string.Format(CSharpResources.ERR_EncodinglessSyntaxTree)}" }, InspectDiagnostics(emitDiagnostics));
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1",
                "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=1|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges=",
                "Debugging_EncSession_EditSession_EmitDeltaErrorId: SessionId=1|EditSessionId=2|ErrorId=CS8055"
            }, _telemetryLog);
        }
 
        [Theory]
        [InlineData(true)]
        [InlineData(false)]
        public async Task ValidSignificantChange_ApplyBeforeFileWatcherEvent(bool saveDocument)
        {
            // Scenarios tested:
            //
            // SaveDocument=true
            // workspace:     --V0-------------|--V2--------|------------|
            // file system:   --V0---------V1--|-----V2-----|------------|
            //                   \--build--/   F5    ^      F10  ^       F10
            //                                       save        file watcher: no-op
            // SaveDocument=false
            // workspace:     --V0-------------|--V2--------|----V1------|
            // file system:   --V0---------V1--|------------|------------|
            //                   \--build--/   F5           F10  ^       F10
            //                                                   file watcher: workspace update
 
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("test.cs").WriteAllText(source1, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            // the workspace starts with a version of the source that's not updated with the output of single file generator (or design-time build):
            var document1 = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(DefaultTargetFramework)).
                AddDocument("test.cs", CreateText("class C1 { void M() { System.Console.WriteLine(0); } }"), filePath: sourceFile.Path);
 
            var documentId = document1.Id;
            solution = document1.Project.Solution;
 
            var sourceTextProvider = new MockPdbMatchingSourceTextProvider()
            {
                TryGetMatchingSourceTextImpl = (filePath, requiredChecksum, checksumAlgorithm) =>
                {
                    Assert.Equal(sourceFile.Path, filePath);
                    AssertEx.Equal(requiredChecksum, CreateText(source1).GetChecksum());
                    Assert.Equal(SourceHashAlgorithms.Default, checksumAlgorithm);
 
                    return source1;
                }
            };
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None, sourceTextProvider);
 
            EnterBreakState(debuggingSession);
 
            // The user opens the source file and changes the source before Roslyn receives file watcher event.
            var source2 = "class C1 { void M() { System.Console.WriteLine(2); } }";
            solution = solution.WithDocumentText(documentId, CreateText(source2));
            var document2 = solution.GetDocument(documentId);
 
            // Save the document:
            if (saveDocument)
            {
                sourceFile.WriteAllText(source2, Encoding.UTF8);
            }
 
            // EnC service queries for a document, which triggers read of the source file from disk.
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
 
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            CommitSolutionUpdate(debuggingSession);
 
            ExitBreakState(debuggingSession);
 
            EnterBreakState(debuggingSession);
 
            // file watcher updates the workspace:
            solution = solution.WithDocumentText(documentId, CreateTextFromFile(sourceFile.Path));
            var document3 = solution.Projects.Single().Documents.Single();
 
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
 
            if (saveDocument)
            {
                Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            }
            else
            {
                Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
                debuggingSession.DiscardSolutionUpdate();
            }
 
            ExitBreakState(debuggingSession);
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_FileUpdateNotObservedBeforeDebuggingSessionStart()
        {
            // workspace:     --V0--------------V2-------|--------V3------------------V1--------------|
            // file system:   --V0---------V1-----V2-----|------------------------------V1------------|
            //                   \--build--/      ^save  F5   ^      ^F10 (no change)   ^save         F10 (ok)
            //                                                file watcher: no-op
            // build updates file from V0 -> V1
 
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class C1 { void M() { System.Console.WriteLine(2); } }";
            var source3 = "class C1 { void M() { System.Console.WriteLine(3); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("test.cs").WriteAllText(source2, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            // the workspace starts with a version of the source that's not updated with the output of single file generator (or design-time build):
            var document2 = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument("test.cs", CreateText(source2), filePath: sourceFile.Path);
 
            var documentId = document2.Id;
 
            var project = document2.Project;
            solution = project.Solution;
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
 
            EnterBreakState(debuggingSession);
 
            // user edits the file:
            solution = solution.WithDocumentText(documentId, CreateText(source3));
            var document3 = solution.Projects.Single().Documents.Single();
 
            // EnC service queries for a document, but the source file on disk doesn't match the PDB
 
            // We don't report rude edits for out-of-sync documents:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document3, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            // since the document is out-of-sync we need to call update to determine whether we have changes to apply or not:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            AssertEx.Equal(new[] { $"{project.Id}: Warning ENC1005: {string.Format(FeaturesResources.DocumentIsOutOfSyncWithDebuggee, sourceFile.Path)}" }, InspectDiagnostics(emitDiagnostics));
 
            // undo:
            solution = solution.WithDocumentText(documentId, CreateText(source1));
 
            var currentDocument = solution.GetDocument(documentId);
 
            // save (note that this call will fail to match the content with the PDB since it uses the content prior to the actual file write)
            // TODO: await debuggingSession.OnSourceFileUpdatedAsync(currentDocument);
            var (doc, state) = await debuggingSession.LastCommittedSolution.GetDocumentAndStateAsync(documentId, currentDocument, CancellationToken.None);
            Assert.Null(doc);
            Assert.Equal(CommittedSolution.DocumentState.OutOfSync, state);
            sourceFile.WriteAllText(source1, Encoding.UTF8);
 
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
 
            // the content actually hasn't changed:
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_AddedFileNotObservedBeforeDebuggingSessionStart()
        {
            // workspace:     ------|----V0---------------|----------
            // file system:   --V0--|---------------------|----------
            //                      F5   ^          ^F10 (no change)
            //                           file watcher observes the file
 
            var source1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("test.cs").WriteAllText(source1, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            // the workspace starts with no file
            var project = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40));
 
            solution = project.Solution;
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path);
 
            _debuggerService.IsEditAndContinueAvailable = _ => new ManagedHotReloadAvailability(ManagedHotReloadAvailabilityStatus.Attach, localizedMessage: "*attached*");
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
 
            // An active statement may be present in the added file since the file exists in the PDB:
            var activeInstruction1 = new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000001, version: 1), ilOffset: 1);
            var activeSpan1 = GetSpan(source1, "System.Console.WriteLine(1);");
            var sourceText1 = CreateText(source1);
            var activeLineSpan1 = sourceText1.Lines.GetLinePositionSpan(activeSpan1);
            var activeStatements = ImmutableArray.Create(
                new ManagedActiveStatementDebugInfo(
                    activeInstruction1,
                    "test.cs",
                    activeLineSpan1.ToSourceSpan(),
                    ActiveStatementFlags.LeafFrame));
 
            // disallow any edits (attach scenario)
            EnterBreakState(debuggingSession, activeStatements);
 
            // File watcher observes the document and adds it to the workspace:
 
            var document1 = project.AddDocument("test.cs", sourceText1, filePath: sourceFile.Path);
            solution = document1.Project.Solution;
 
            // We don't report rude edits for the added document:
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document1, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            // TODO: https://github.com/dotnet/roslyn/issues/49938
            // We currently create the AS map against the committed solution, which may not contain all documents.
            // var spans = await service.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(document1.Id), CancellationToken.None);
            // AssertEx.Equal(new[] { $"({activeLineSpan1}, LeafFrame)" }, spans.Single().Select(s => s.ToString()));
 
            // No changes.
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
 
            AssertEx.Empty(emitDiagnostics);
 
            EndDebuggingSession(debuggingSession);
        }
 
        [Theory]
        [CombinatorialData]
        public async Task ValidSignificantChange_DocumentOutOfSync(bool delayLoad)
        {
            var sourceOnDisk = "class C1 { void M() { System.Console.WriteLine(1); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("test.cs").WriteAllText(sourceOnDisk, Encoding.UTF8);
 
            using var _ = CreateWorkspace(out var solution, out var service);
 
            // the workspace starts with a version of the source that's not updated with the output of single file generator (or design-time build):
            var document1 = solution.
                AddProject("test", "test", LanguageNames.CSharp).
                AddMetadataReferences(TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument("test.cs", CreateText("class C1 { void M() { System.Console.WriteLine(0); } }"), filePath: sourceFile.Path);
 
            var project = document1.Project;
            solution = project.Solution;
 
            var moduleId = EmitLibrary(sourceOnDisk, sourceFilePath: sourceFile.Path);
 
            if (!delayLoad)
            {
                LoadLibraryToDebuggee(moduleId);
            }
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.None);
 
            EnterBreakState(debuggingSession);
 
            // no changes have been made to the project
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(updates.Updates);
            Assert.Empty(emitDiagnostics);
 
            // a file watcher observed a change and updated the document, so it now reflects the content on disk (the code that we compiled):
            solution = solution.WithDocumentText(document1.Id, CreateText(sourceOnDisk));
            var document3 = solution.Projects.Single().Documents.Single();
 
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document3, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // the content of the file is now exactly the same as the compiled document, so there is no change to be applied:
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            Assert.Empty(emitDiagnostics);
 
            EndDebuggingSession(debuggingSession);
 
            Assert.Empty(debuggingSession.GetTestAccessor().GetModulesPreparedForUpdate());
        }
 
        [Theory]
        [CombinatorialData]
        public async Task ValidSignificantChange_EmitSuccessful(bool breakMode, bool commitUpdate)
        {
            var sourceV1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
            var sourceV2 = "class C1 { void M() { System.Console.WriteLine(2); } }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1);
 
            var moduleId = EmitAndLoadLibraryToDebuggee(sourceV1);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            if (breakMode)
            {
                EnterBreakState(debuggingSession);
            }
 
            // change the source (valid edit):
            solution = solution.WithDocumentText(document1.Id, CreateText(sourceV2));
            var document2 = solution.GetDocument(document1.Id);
 
            var diagnostics1 = await service.GetDocumentDiagnosticsAsync(document2, s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics1);
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            ValidateDelta(updates.Updates.Single());
 
            void ValidateDelta(ModuleUpdate delta)
            {
                // check emitted delta:
                Assert.Empty(delta.ActiveStatements);
                Assert.NotEmpty(delta.ILDelta);
                Assert.NotEmpty(delta.MetadataDelta);
                Assert.NotEmpty(delta.PdbDelta);
                Assert.Equal(0x06000001, delta.UpdatedMethods.Single());
                Assert.Equal(0x02000002, delta.UpdatedTypes.Single());
                Assert.Equal(moduleId, delta.Module);
                Assert.Empty(delta.ExceptionRegions);
                Assert.Empty(delta.SequencePoints);
            }
 
            // the update should be stored on the service:
            var pendingUpdate = debuggingSession.GetTestAccessor().GetPendingSolutionUpdate();
            var (baselineProjectId, newBaseline) = pendingUpdate.EmitBaselines.Single();
            AssertEx.Equal(updates.Updates, pendingUpdate.Deltas);
            Assert.Equal(document2.Project.Id, baselineProjectId);
            Assert.Equal(moduleId, newBaseline.OriginalMetadata.GetModuleVersionId());
 
            var readers = debuggingSession.GetTestAccessor().GetBaselineModuleReaders();
            Assert.Equal(2, readers.Length);
            Assert.NotNull(readers[0]);
            Assert.NotNull(readers[1]);
 
            if (commitUpdate)
            {
                // all update providers either provided updates or had no change to apply:
                CommitSolutionUpdate(debuggingSession);
 
                Assert.Null(debuggingSession.GetTestAccessor().GetPendingSolutionUpdate());
 
                // no change in non-remappable regions since we didn't have any active statements:
                Assert.Empty(debuggingSession.EditSession.NonRemappableRegions);
 
                var baselineReaders = debuggingSession.GetTestAccessor().GetBaselineModuleReaders();
                Assert.Equal(2, baselineReaders.Length);
                Assert.Same(readers[0], baselineReaders[0]);
                Assert.Same(readers[1], baselineReaders[1]);
 
                // verify that baseline is added:
                Assert.Same(newBaseline, debuggingSession.GetTestAccessor().GetProjectEmitBaseline(document2.Project.Id));
 
                // solution update status after committing an update:
                (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
                Assert.Empty(emitDiagnostics);
                Assert.Equal(ModuleUpdateStatus.None, updates.Status);
            }
            else
            {
                // another update provider blocked the update:
                debuggingSession.DiscardSolutionUpdate();
 
                Assert.Null(debuggingSession.GetTestAccessor().GetPendingSolutionUpdate());
 
                // solution update status after committing an update:
                (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
                Assert.Empty(emitDiagnostics);
                Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
                ValidateDelta(updates.Updates.Single());
                debuggingSession.DiscardSolutionUpdate();
            }
 
            if (breakMode)
            {
                ExitBreakState(debuggingSession);
            }
 
            EndDebuggingSession(debuggingSession);
 
            // open module readers should be disposed when the debugging session ends:
            VerifyReadersDisposed(readers);
 
            AssertEx.SetEqual(new[] { moduleId }, debuggingSession.GetTestAccessor().GetModulesPreparedForUpdate());
 
            if (breakMode)
            {
                AssertEx.Equal(new[]
                {
                    $"Debugging_EncSession: SolutionSessionId={{00000000-AAAA-AAAA-AAAA-000000000000}}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount={(commitUpdate ? 3 : 2)}",
                    $"Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=0|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges={(commitUpdate ? "{00000000-AAAA-AAAA-AAAA-111111111111}" : "")}",
                }, _telemetryLog);
            }
            else
            {
                AssertEx.Equal(new[]
                {
                    $"Debugging_EncSession: SolutionSessionId={{00000000-AAAA-AAAA-AAAA-000000000000}}|SessionId=1|SessionCount=0|EmptySessionCount=0|HotReloadSessionCount=1|EmptyHotReloadSessionCount={(commitUpdate ? 1 : 0)}",
                    $"Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=0|InBreakState=False|Capabilities=31|ProjectIdsWithAppliedChanges={(commitUpdate ? "{00000000-AAAA-AAAA-AAAA-111111111111}" : "")}"
                }, _telemetryLog);
            }
        }
 
        [Theory]
        [InlineData(true)]
        [InlineData(false)]
        public async Task ValidSignificantChange_EmitSuccessful_UpdateDeferred(bool commitUpdate)
        {
            var dir = Temp.CreateDirectory();
 
            var sourceV1 = "class C1 { void M1() { int a = 1; System.Console.WriteLine(a); } void M2() { System.Console.WriteLine(1); } }";
            var compilationV1 = CSharpTestBase.CreateCompilation(sourceV1, parseOptions: TestOptions.Regular.WithNoRefSafetyRulesAttribute(), options: TestOptions.DebugDll, targetFramework: DefaultTargetFramework, assemblyName: "lib");
 
            var (peImage, pdbImage) = compilationV1.EmitToArrays(new EmitOptions(debugInformationFormat: DebugInformationFormat.PortablePdb));
            var moduleMetadata = ModuleMetadata.CreateFromImage(peImage);
            var moduleFile = dir.CreateFile("lib.dll").WriteAllBytes(peImage);
            var pdbFile = dir.CreateFile("lib.pdb").WriteAllBytes(pdbImage);
            var moduleId = moduleMetadata.GetModuleVersionId();
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1);
 
            _mockCompilationOutputsProvider = _ => new CompilationOutputFiles(moduleFile.Path, pdbFile.Path);
 
            // set up an active statement in the first method, so that we can test preservation of local signature.
            var activeStatements = ImmutableArray.Create(new ManagedActiveStatementDebugInfo(
                new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000001, version: 1), ilOffset: 0),
                documentName: document1.Name,
                sourceSpan: new SourceSpan(0, 15, 0, 16),
                ActiveStatementFlags.LeafFrame));
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // module is not loaded:
            EnterBreakState(debuggingSession, activeStatements);
 
            // change the source (valid edit):
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M1() { int a = 1; System.Console.WriteLine(a); } void M2() { System.Console.WriteLine(2); } }"));
            var document2 = solution.GetDocument(document1.Id);
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            Assert.Empty(emitDiagnostics);
 
            // delta to apply:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Equal(0x06000002, delta.UpdatedMethods.Single());
            Assert.Equal(0x02000002, delta.UpdatedTypes.Single());
            Assert.Equal(moduleId, delta.Module);
            Assert.Empty(delta.ExceptionRegions);
            Assert.Empty(delta.SequencePoints);
 
            // the update should be stored on the service:
            var pendingUpdate = debuggingSession.GetTestAccessor().GetPendingSolutionUpdate();
            var (baselineProjectId, newBaseline) = pendingUpdate.EmitBaselines.Single();
 
            var readers = debuggingSession.GetTestAccessor().GetBaselineModuleReaders();
            Assert.Equal(2, readers.Length);
            Assert.NotNull(readers[0]);
            Assert.NotNull(readers[1]);
 
            Assert.Equal(document2.Project.Id, baselineProjectId);
            Assert.Equal(moduleId, newBaseline.OriginalMetadata.GetModuleVersionId());
 
            if (commitUpdate)
            {
                CommitSolutionUpdate(debuggingSession);
                Assert.Null(debuggingSession.GetTestAccessor().GetPendingSolutionUpdate());
 
                // no change in non-remappable regions since we didn't have any active statements:
                Assert.Empty(debuggingSession.EditSession.NonRemappableRegions);
 
                // verify that baseline is added:
                Assert.Same(newBaseline, debuggingSession.GetTestAccessor().GetProjectEmitBaseline(document2.Project.Id));
 
                // solution update status after committing an update:
                ExitBreakState(debuggingSession);
 
                // make another update:
                EnterBreakState(debuggingSession);
 
                // Update M1 - this method has an active statement, so we will attempt to preserve the local signature.
                // Since the method hasn't been edited before we'll read the baseline PDB to get the signature token.
                // This validates that the Portable PDB reader can be used (and is not disposed) for a second generation edit.
                var document3 = solution.GetDocument(document1.Id);
                solution = solution.WithDocumentText(document3.Id, CreateText("class C1 { void M1() { int a = 3; System.Console.WriteLine(a); } void M2() { System.Console.WriteLine(2); } }"));
 
                (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
                Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
                Assert.Empty(emitDiagnostics);
                debuggingSession.DiscardSolutionUpdate();
            }
            else
            {
                debuggingSession.DiscardSolutionUpdate();
                Assert.Null(debuggingSession.GetTestAccessor().GetPendingSolutionUpdate());
            }
 
            ExitBreakState(debuggingSession);
            EndDebuggingSession(debuggingSession);
 
            // open module readers should be disposed when the debugging session ends:
            VerifyReadersDisposed(readers);
        }
 
        [Fact]
        public async Task ValidSignificantChange_PartialTypes()
        {
            var sourceA1 = @"
partial class C { int X = 1; void F() { X = 1; } }
 
partial class D { int U = 1; public D() { } }
partial class D { int W = 1; }
 
partial class E { int A; public E(int a) { A = a; } }
";
            var sourceB1 = @"
partial class C { int Y = 1; }
partial class E { int B; public E(int a, int b) { A = a; B = new System.Func<int>(() => b)(); } }
";
 
            var sourceA2 = @"
partial class C { int X = 2; void F() { X = 2; } }
 
partial class D { int U = 2; }
partial class D { int W = 2; public D() { } }
 
partial class E { int A = 1; public E(int a) { A = a; } }
";
            var sourceB2 = @"
partial class C { int Y = 2; }
partial class E { int B = 2; public E(int a, int b) { A = a; B = new System.Func<int>(() => b)(); } }
";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            solution = AddDefaultTestProject(solution, new[] { sourceA1, sourceB1 });
            var project = solution.Projects.Single();
 
            LoadLibraryToDebuggee(EmitLibrary(new[] { (sourceA1, "test1.cs"), (sourceB1, "test2.cs") }));
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source (valid edit):
            var documentA = project.Documents.First();
            var documentB = project.Documents.Skip(1).First();
            solution = solution.WithDocumentText(documentA.Id, CreateText(sourceA2));
            solution = solution.WithDocumentText(documentB.Id, CreateText(sourceB2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Equal(6, delta.UpdatedMethods.Length);  // F, C.C(), D.D(), E.E(int), E.E(int, int), lambda
            AssertEx.SetEqual(new[] { 0x02000002, 0x02000003, 0x02000004, 0x02000005 }, delta.UpdatedTypes, itemInspector: t => "0x" + t.ToString("X"));
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_SourceGenerators_DocumentUpdate_GeneratedDocumentUpdate()
        {
            var sourceV1 = @"
/* GENERATE: class G { int X => 1; } */
 
class C { int Y => 1; }
";
            var sourceV2 = @"
/* GENERATE: class G { int X => 2; } */
 
class C { int Y => 2; }
";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1, generator);
 
            var moduleId = EmitLibrary(sourceV1, generator: generator);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source (valid edit)
            solution = solution.WithDocumentText(document1.Id, CreateText(sourceV2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Equal(2, delta.UpdatedMethods.Length);
            AssertEx.Equal(new[] { 0x02000002, 0x02000003 }, delta.UpdatedTypes, itemInspector: t => "0x" + t.ToString("X"));
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_SourceGenerators_DocumentUpdate_GeneratedDocumentUpdate_LineChanges()
        {
            var sourceV1 = @"
/* GENERATE:
class G
{
    int M()
    {
        return 1;
    }
}
*/
";
            var sourceV2 = @"
/* GENERATE:
class G
{
 
    int M()
    {
        return 1;
    }
}
*/
";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1, generator);
 
            var moduleId = EmitLibrary(sourceV1, generator: generator);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source (valid edit):
            solution = solution.WithDocumentText(document1.Id, CreateText(sourceV2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
 
            var lineUpdate = delta.SequencePoints.Single();
            AssertEx.Equal(new[] { "3 -> 4" }, lineUpdate.LineUpdates.Select(edit => $"{edit.OldLine} -> {edit.NewLine}"));
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Empty(delta.UpdatedMethods);
            Assert.Empty(delta.UpdatedTypes);
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_SourceGenerators_DocumentUpdate_GeneratedDocumentInsert()
        {
            var sourceV1 = @"
partial class C { int X = 1; }
";
            var sourceV2 = @"
/* GENERATE: partial class C { int Y = 2; } */
 
partial class C { int X = 1; }
";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1, generator);
 
            var moduleId = EmitLibrary(sourceV1, generator: generator);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the source (valid edit):
            solution = solution.WithDocumentText(document1.Id, CreateText(sourceV2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Equal(1, delta.UpdatedMethods.Length); // constructor update
            Assert.Equal(0x02000002, delta.UpdatedTypes.Single());
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_SourceGenerators_AdditionalDocumentUpdate()
        {
            var source = @"
class C { int Y => 1; }
";
 
            var additionalSourceV1 = @"
/* GENERATE: class G { int X => 1; } */
";
            var additionalSourceV2 = @"
/* GENERATE: class G { int X => 2; } */
";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source, generator, additionalFileText: additionalSourceV1);
 
            var moduleId = EmitLibrary(source, generator: generator, additionalFileText: additionalSourceV1);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the additional source (valid edit):
            var additionalDocument1 = solution.Projects.Single().AdditionalDocuments.Single();
            solution = solution.WithAdditionalDocumentText(additionalDocument1.Id, CreateText(additionalSourceV2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Equal(1, delta.UpdatedMethods.Length);
            Assert.Equal(0x02000003, delta.UpdatedTypes.Single());
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_SourceGenerators_AnalyzerConfigUpdate()
        {
            var source = @"
class C { int Y => 1; }
";
 
            var configV1 = new[] { ("enc_generator_output", "1") };
            var configV2 = new[] { ("enc_generator_output", "2") };
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source, generator, analyzerConfig: configV1);
 
            var moduleId = EmitLibrary(source, generator: generator, analyzerOptions: configV1);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // change the additional source (valid edit):
            var configDocument1 = solution.Projects.Single().AnalyzerConfigDocuments.Single();
            solution = solution.WithAnalyzerConfigDocumentText(configDocument1.Id, GetAnalyzerConfigText(configV2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Equal(1, delta.UpdatedMethods.Length);
            Assert.Equal(0x02000003, delta.UpdatedTypes.Single());
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task ValidSignificantChange_SourceGenerators_DocumentRemove()
        {
            var source1 = "";
 
            var generator = new TestSourceGenerator()
            {
                ExecuteImpl = context => context.AddSource("generated", $"class G {{ int X => {context.Compilation.SyntaxTrees.Count()}; }}")
            };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, source1, generator);
 
            var moduleId = EmitLibrary(source1, generator: generator);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            // remove the source document (valid edit):
            solution = document1.Project.Solution.RemoveDocument(document1.Id);
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Equal(1, delta.UpdatedMethods.Length);
            Assert.Equal(0x02000002, delta.UpdatedTypes.Single());
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact]
        public async Task RudeEdit()
        {
            var source1 = "class C { void M() { } }";
            var source2 = "class C { void M() { var x = new { Goo = 1 }; } }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            solution = AddDefaultTestProject(solution, new[] { source1 });
            var project = solution.Projects.Single();
            solution = solution.WithProjectParseOptions(project.Id, new CSharpParseOptions(LanguageVersion.CSharp10));
            var documentId = solution.Projects.Single().Documents.Single().Id;
 
            EmitAndLoadLibraryToDebuggee(source1);
 
            // attached to processes that doesn't allow creating new types
            _debuggerService.GetCapabilitiesImpl = () => ImmutableArray.Create("Baseline");
 
            // F5
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // update document:
            solution = solution.WithDocumentText(documentId, CreateText(source2));
            var document2 = solution.Projects.Single().Documents.Single();
 
            // These errors aren't reported as document diagnostics
            var diagnostics = await service.GetDocumentDiagnosticsAsync(solution.GetDocument(documentId), s_noActiveSpans, CancellationToken.None);
            AssertEx.Empty(diagnostics);
 
            // They are reported as emit diagnostics
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            AssertEx.Equal(new[] { $"{document2.Project.Id}: Error ENC1007: {FeaturesResources.ChangesRequiredSynthesizedType}" }, InspectDiagnostics(emitDiagnostics));
 
            // no emitted delta:
            Assert.Empty(updates.Updates);
 
            EndDebuggingSession(debuggingSession);
        }
 
        /// <summary>
        /// Emulates two updates to Multi-TFM project.
        /// </summary>
        [Fact]
        public async Task TwoUpdatesWithLoadedAndUnloadedModule()
        {
            var dir = Temp.CreateDirectory();
 
            var source1 = "class A { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class A { void M() { System.Console.WriteLine(2); } }";
            var source3 = "class A { void M() { System.Console.WriteLine(3); } }";
 
            var compilationA = CSharpTestBase.CreateCompilation(source1, options: TestOptions.DebugDll, targetFramework: DefaultTargetFramework, assemblyName: "A");
            var compilationB = CSharpTestBase.CreateCompilation(source1, options: TestOptions.DebugDll, targetFramework: DefaultTargetFramework, assemblyName: "B");
 
            var (peImageA, pdbImageA) = compilationA.EmitToArrays(new EmitOptions(debugInformationFormat: DebugInformationFormat.PortablePdb));
            var moduleMetadataA = ModuleMetadata.CreateFromImage(peImageA);
            var moduleFileA = Temp.CreateFile("A.dll").WriteAllBytes(peImageA);
            var pdbFileA = dir.CreateFile("A.pdb").WriteAllBytes(pdbImageA);
            var moduleIdA = moduleMetadataA.GetModuleVersionId();
 
            var (peImageB, pdbImageB) = compilationB.EmitToArrays(new EmitOptions(debugInformationFormat: DebugInformationFormat.PortablePdb));
            var moduleMetadataB = ModuleMetadata.CreateFromImage(peImageB);
            var moduleFileB = dir.CreateFile("B.dll").WriteAllBytes(peImageB);
            var pdbFileB = dir.CreateFile("B.pdb").WriteAllBytes(pdbImageB);
            var moduleIdB = moduleMetadataB.GetModuleVersionId();
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var documentA) = AddDefaultTestProject(solution, source1);
            var projectA = documentA.Project;
 
            var projectB = solution.AddProject("B", "A", "C#").
                AddMetadataReferences(projectA.MetadataReferences).
                AddDocument("DocB", source1, filePath: Path.Combine(TempRoot.Root, "DocB.cs")).Project;
 
            solution = projectB.Solution;
 
            _mockCompilationOutputsProvider = project =>
                (project.Id == projectA.Id) ? new CompilationOutputFiles(moduleFileA.Path, pdbFileA.Path) :
                (project.Id == projectB.Id) ? new CompilationOutputFiles(moduleFileB.Path, pdbFileB.Path) :
                throw ExceptionUtilities.UnexpectedValue(project);
 
            // only module A is loaded
            LoadLibraryToDebuggee(moduleIdA);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession);
 
            //
            // First update.
            //
 
            solution = solution.WithDocumentText(projectA.Documents.Single().Id, CreateText(source2));
            solution = solution.WithDocumentText(projectB.Documents.Single().Id, CreateText(source2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            Assert.Empty(emitDiagnostics);
 
            var deltaA = updates.Updates.Single(d => d.Module == moduleIdA);
            var deltaB = updates.Updates.Single(d => d.Module == moduleIdB);
            Assert.Equal(2, updates.Updates.Length);
 
            // the update should be stored on the service:
            var pendingUpdate = debuggingSession.GetTestAccessor().GetPendingSolutionUpdate();
            var (_, newBaselineA1) = pendingUpdate.EmitBaselines.Single(b => b.ProjectId == projectA.Id);
            var (_, newBaselineB1) = pendingUpdate.EmitBaselines.Single(b => b.ProjectId == projectB.Id);
 
            var baselineA0 = newBaselineA1.GetInitialEmitBaseline();
            var baselineB0 = newBaselineB1.GetInitialEmitBaseline();
 
            var readers = debuggingSession.GetTestAccessor().GetBaselineModuleReaders();
            Assert.Equal(4, readers.Length);
            Assert.False(readers.Any(r => r is null));
 
            Assert.Equal(moduleIdA, newBaselineA1.OriginalMetadata.GetModuleVersionId());
            Assert.Equal(moduleIdB, newBaselineB1.OriginalMetadata.GetModuleVersionId());
 
            CommitSolutionUpdate(debuggingSession);
            Assert.Null(debuggingSession.GetTestAccessor().GetPendingSolutionUpdate());
 
            // no change in non-remappable regions since we didn't have any active statements:
            Assert.Empty(debuggingSession.EditSession.NonRemappableRegions);
 
            // verify that baseline is added for both modules:
            Assert.Same(newBaselineA1, debuggingSession.GetTestAccessor().GetProjectEmitBaseline(projectA.Id));
            Assert.Same(newBaselineB1, debuggingSession.GetTestAccessor().GetProjectEmitBaseline(projectB.Id));
 
            // solution update status after committing an update:(updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            ExitBreakState(debuggingSession);
            EnterBreakState(debuggingSession);
 
            //
            // Second update.
            //
 
            solution = solution.WithDocumentText(projectA.Documents.Single().Id, CreateText(source3));
            solution = solution.WithDocumentText(projectB.Documents.Single().Id, CreateText(source3));
 
            // validate solution update status and emit:
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
            Assert.Empty(emitDiagnostics);
 
            deltaA = updates.Updates.Single(d => d.Module == moduleIdA);
            deltaB = updates.Updates.Single(d => d.Module == moduleIdB);
            Assert.Equal(2, updates.Updates.Length);
 
            // the update should be stored on the service:
            pendingUpdate = debuggingSession.GetTestAccessor().GetPendingSolutionUpdate();
            var (_, newBaselineA2) = pendingUpdate.EmitBaselines.Single(b => b.ProjectId == projectA.Id);
            var (_, newBaselineB2) = pendingUpdate.EmitBaselines.Single(b => b.ProjectId == projectB.Id);
 
            Assert.NotSame(newBaselineA1, newBaselineA2);
            Assert.NotSame(newBaselineB1, newBaselineB2);
            Assert.Same(baselineA0, newBaselineA2.GetInitialEmitBaseline());
            Assert.Same(baselineB0, newBaselineB2.GetInitialEmitBaseline());
            Assert.Same(baselineA0.OriginalMetadata, newBaselineA2.OriginalMetadata);
            Assert.Same(baselineB0.OriginalMetadata, newBaselineB2.OriginalMetadata);
 
            // no new module readers:
            var baselineReaders = debuggingSession.GetTestAccessor().GetBaselineModuleReaders();
            AssertEx.Equal(readers, baselineReaders);
 
            CommitSolutionUpdate(debuggingSession);
            Assert.Null(debuggingSession.GetTestAccessor().GetPendingSolutionUpdate());
 
            // no change in non-remappable regions since we didn't have any active statements:
            Assert.Empty(debuggingSession.EditSession.NonRemappableRegions);
 
            // module readers tracked:
            baselineReaders = debuggingSession.GetTestAccessor().GetBaselineModuleReaders();
            AssertEx.Equal(readers, baselineReaders);
 
            // verify that baseline is updated for both modules:
            Assert.Same(newBaselineA2, debuggingSession.GetTestAccessor().GetProjectEmitBaseline(projectA.Id));
            Assert.Same(newBaselineB2, debuggingSession.GetTestAccessor().GetProjectEmitBaseline(projectB.Id));
 
            // solution update status after committing an update:
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.None, updates.Status);
 
            ExitBreakState(debuggingSession);
            EndDebuggingSession(debuggingSession);
 
            // open deferred module readers should be dispose when the debugging session ends:
            VerifyReadersDisposed(readers);
        }
 
        [Fact]
        public async Task ValidSignificantChange_BaselineCreationFailed_NoStream()
        {
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, "class C1 { void M() { System.Console.WriteLine(1); } }");
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(Guid.NewGuid())
            {
                OpenPdbStreamImpl = () => null,
                OpenAssemblyStreamImpl = () => null,
            };
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // module not loaded
            EnterBreakState(debuggingSession);
 
            // change the source (valid edit):
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M() { System.Console.WriteLine(2); } }"));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            AssertEx.Equal(new[] { $"{document1.Project.Id}: Error ENC1001: {string.Format(FeaturesResources.ErrorReadingFile, "test-pdb", new FileNotFoundException().Message)}" }, InspectDiagnostics(emitDiagnostics));
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
        }
 
        [Fact]
        public async Task ValidSignificantChange_BaselineCreationFailed_AssemblyReadError()
        {
            var sourceV1 = "class C1 { void M() { System.Console.WriteLine(1); } }";
            var compilationV1 = CSharpTestBase.CreateCompilationWithMscorlib40(sourceV1, options: TestOptions.DebugDll, assemblyName: "lib");
 
            var pdbStream = new MemoryStream();
            var peImage = compilationV1.EmitToArray(new EmitOptions(debugInformationFormat: DebugInformationFormat.PortablePdb), pdbStream: pdbStream);
            pdbStream.Position = 0;
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, sourceV1);
 
            _mockCompilationOutputsProvider = _ => new MockCompilationOutputs(Guid.NewGuid())
            {
                OpenPdbStreamImpl = () => pdbStream,
                OpenAssemblyStreamImpl = () => throw new IOException("*message*"),
            };
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // module not loaded
            EnterBreakState(debuggingSession);
 
            // change the source (valid edit):
            var document1 = solution.Projects.Single().Documents.Single();
            solution = solution.WithDocumentText(document1.Id, CreateText("class C1 { void M() { System.Console.WriteLine(2); } }"));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            AssertEx.Equal(new[] { $"{document.Project.Id}: Error ENC1001: {string.Format(FeaturesResources.ErrorReadingFile, "test-assembly", "*message*")}" }, InspectDiagnostics(emitDiagnostics));
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
 
            EndDebuggingSession(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                "Debugging_EncSession: SolutionSessionId={00000000-AAAA-AAAA-AAAA-000000000000}|SessionId=1|SessionCount=1|EmptySessionCount=0|HotReloadSessionCount=0|EmptyHotReloadSessionCount=1",
                "Debugging_EncSession_EditSession: SessionId=1|EditSessionId=2|HadCompilationErrors=False|HadRudeEdits=False|HadValidChanges=True|HadValidInsignificantChanges=False|RudeEditsCount=0|EmitDeltaErrorIdCount=1|InBreakState=True|Capabilities=31|ProjectIdsWithAppliedChanges=",
                "Debugging_EncSession_EditSession_EmitDeltaErrorId: SessionId=1|EditSessionId=2|ErrorId=ENC1001"
            }, _telemetryLog);
        }
 
        [Fact]
        public async Task ActiveStatements()
        {
            var sourceV1 = "class C { void F() { G(1); } void G(int a) => System.Console.WriteLine(1); }";
            var sourceV2 = "class C { int x; void F() { G(2); G(1); } void G(int a) => System.Console.WriteLine(2); }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1);
 
            var activeSpan11 = GetSpan(sourceV1, "G(1);");
            var activeSpan12 = GetSpan(sourceV1, "System.Console.WriteLine(1)");
            var activeSpan21 = GetSpan(sourceV2, "G(2); G(1);");
            var activeSpan22 = GetSpan(sourceV2, "System.Console.WriteLine(2)");
            var adjustedActiveSpan1 = GetSpan(sourceV2, "G(2);");
            var adjustedActiveSpan2 = GetSpan(sourceV2, "System.Console.WriteLine(2)");
 
            var documentId = document1.Id;
            var documentPath = document1.FilePath;
 
            var sourceTextV1 = document1.GetTextSynchronously(CancellationToken.None);
            var sourceTextV2 = CreateText(sourceV2);
 
            var activeLineSpan11 = sourceTextV1.Lines.GetLinePositionSpan(activeSpan11);
            var activeLineSpan12 = sourceTextV1.Lines.GetLinePositionSpan(activeSpan12);
            var activeLineSpan21 = sourceTextV2.Lines.GetLinePositionSpan(activeSpan21);
            var activeLineSpan22 = sourceTextV2.Lines.GetLinePositionSpan(activeSpan22);
            var adjustedActiveLineSpan1 = sourceTextV2.Lines.GetLinePositionSpan(adjustedActiveSpan1);
            var adjustedActiveLineSpan2 = sourceTextV2.Lines.GetLinePositionSpan(adjustedActiveSpan2);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // default if not called in a break state
            Assert.True((await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(document1.Id), CancellationToken.None)).IsDefault);
 
            var moduleId = Guid.NewGuid();
            var activeInstruction1 = new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000001, version: 1), ilOffset: 1);
            var activeInstruction2 = new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000002, version: 1), ilOffset: 1);
 
            var activeStatements = ImmutableArray.Create(
                new ManagedActiveStatementDebugInfo(
                    activeInstruction1,
                    documentPath,
                    activeLineSpan11.ToSourceSpan(),
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame),
                new ManagedActiveStatementDebugInfo(
                    activeInstruction2,
                    documentPath,
                    activeLineSpan12.ToSourceSpan(),
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame));
 
            EnterBreakState(debuggingSession, activeStatements);
 
            var activeStatementSpan11 = new ActiveStatementSpan(0, activeLineSpan11, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame, unmappedDocumentId: null);
            var activeStatementSpan12 = new ActiveStatementSpan(1, activeLineSpan12, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame, unmappedDocumentId: null);
 
            var baseSpans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(document1.Id), CancellationToken.None);
            AssertEx.Equal(new[]
            {
               activeStatementSpan11,
               activeStatementSpan12
            }, baseSpans.Single());
 
            var trackedActiveSpans1 = ImmutableArray.Create(activeStatementSpan11, activeStatementSpan12);
 
            var currentSpans = await debuggingSession.GetAdjustedActiveStatementSpansAsync(document1, (_, _, _) => new(trackedActiveSpans1), CancellationToken.None);
            AssertEx.Equal(trackedActiveSpans1, currentSpans);
 
            Assert.Equal(activeLineSpan11,
                await debuggingSession.GetCurrentActiveStatementPositionAsync(document1.Project.Solution, (_, _, _) => new(trackedActiveSpans1), activeInstruction1, CancellationToken.None));
 
            Assert.Equal(activeLineSpan12,
                await debuggingSession.GetCurrentActiveStatementPositionAsync(document1.Project.Solution, (_, _, _) => new(trackedActiveSpans1), activeInstruction2, CancellationToken.None));
 
            // change the source (valid edit):
            solution = solution.WithDocumentText(documentId, sourceTextV2);
            var document2 = solution.GetDocument(documentId);
 
            // tracking span update triggered by the edit:
            var activeStatementSpan21 = new ActiveStatementSpan(0, activeLineSpan21, ActiveStatementFlags.NonLeafFrame, unmappedDocumentId: null);
            var activeStatementSpan22 = new ActiveStatementSpan(1, activeLineSpan22, ActiveStatementFlags.LeafFrame, unmappedDocumentId: null);
            var trackedActiveSpans2 = ImmutableArray.Create(activeStatementSpan21, activeStatementSpan22);
 
            currentSpans = await debuggingSession.GetAdjustedActiveStatementSpansAsync(document2, (_, _, _) => new(trackedActiveSpans2), CancellationToken.None);
            AssertEx.Equal(new[] { adjustedActiveLineSpan1, adjustedActiveLineSpan2 }, currentSpans.Select(s => s.LineSpan));
 
            Assert.Equal(adjustedActiveLineSpan1,
                await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, (_, _, _) => new(trackedActiveSpans2), activeInstruction1, CancellationToken.None));
 
            Assert.Equal(adjustedActiveLineSpan2,
                await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, (_, _, _) => new(trackedActiveSpans2), activeInstruction2, CancellationToken.None));
        }
 
        [Theory]
        [CombinatorialData]
        public async Task ActiveStatements_SyntaxErrorOrOutOfSyncDocument(bool isOutOfSync)
        {
            var sourceV1 = "class C { void F() => G(1); void G(int a) => System.Console.WriteLine(1); }";
 
            // syntax error (missing ';') unless testing out-of-sync document
            var sourceV2 = isOutOfSync
                ? "class C { int x; void F() => G(1); void G(int a) => System.Console.WriteLine(2); }"
                : "class C { int x void F() => G(1); void G(int a) => System.Console.WriteLine(2); }";
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, sourceV1);
 
            var activeSpan11 = GetSpan(sourceV1, "G(1)");
            var activeSpan12 = GetSpan(sourceV1, "System.Console.WriteLine(1)");
 
            var documentId = document1.Id;
            var documentFilePath = document1.FilePath;
 
            var sourceTextV1 = await document1.GetTextAsync(CancellationToken.None);
            var sourceTextV2 = CreateText(sourceV2);
 
            var activeLineSpan11 = sourceTextV1.Lines.GetLinePositionSpan(activeSpan11);
            var activeLineSpan12 = sourceTextV1.Lines.GetLinePositionSpan(activeSpan12);
 
            var debuggingSession = await StartDebuggingSessionAsync(
                service,
                solution,
                isOutOfSync ? CommittedSolution.DocumentState.OutOfSync : CommittedSolution.DocumentState.MatchesBuildOutput);
 
            var moduleId = Guid.NewGuid();
            var activeInstruction1 = new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000001, version: 1), ilOffset: 1);
            var activeInstruction2 = new ManagedInstructionId(new ManagedMethodId(moduleId, token: 0x06000002, version: 1), ilOffset: 1);
 
            var activeStatements = ImmutableArray.Create(
                new ManagedActiveStatementDebugInfo(
                    activeInstruction1,
                    documentFilePath,
                    activeLineSpan11.ToSourceSpan(),
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame),
                new ManagedActiveStatementDebugInfo(
                    activeInstruction2,
                    documentFilePath,
                    activeLineSpan12.ToSourceSpan(),
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame));
 
            EnterBreakState(debuggingSession, activeStatements);
 
            var baseSpans = (await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(documentId), CancellationToken.None)).Single();
            AssertEx.Equal(new[]
            {
                new ActiveStatementSpan(0, activeLineSpan11, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame, unmappedDocumentId: null),
                new ActiveStatementSpan(1, activeLineSpan12, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame, unmappedDocumentId: null)
            }, baseSpans);
 
            // change the source (valid edit):
            solution = solution.WithDocumentText(documentId, sourceTextV2);
            var document2 = solution.GetDocument(documentId);
 
            // no adjustments made due to syntax error or out-of-sync document:
            var currentSpans = await debuggingSession.GetAdjustedActiveStatementSpansAsync(document2, (_, _, _) => ValueTaskFactory.FromResult(baseSpans), CancellationToken.None);
            AssertEx.Equal(new[] { activeLineSpan11, activeLineSpan12 }, currentSpans.Select(s => s.LineSpan));
 
            var currentSpan1 = await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, (_, _, _) => ValueTaskFactory.FromResult(baseSpans), activeInstruction1, CancellationToken.None);
            var currentSpan2 = await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, (_, _, _) => ValueTaskFactory.FromResult(baseSpans), activeInstruction2, CancellationToken.None);
            if (isOutOfSync)
            {
                Assert.Equal(baseSpans[0].LineSpan, currentSpan1.Value);
                Assert.Equal(baseSpans[1].LineSpan, currentSpan2.Value);
            }
            else
            {
                Assert.Null(currentSpan1);
                Assert.Null(currentSpan2);
            }
        }
 
        [Theory]
        [CombinatorialData]
        public async Task ActiveStatements_ForeignDocument(bool withPath, bool designTimeOnly)
        {
            var composition = FeaturesTestCompositions.Features.AddParts(typeof(NoCompilationLanguageService));
 
            using var _ = CreateWorkspace(out var solution, out var service, new[] { typeof(NoCompilationLanguageService) });
 
            var project = solution.AddProject("dummy_proj", "dummy_proj", designTimeOnly ? LanguageNames.CSharp : NoCompilationConstants.LanguageName);
            var filePath = withPath ? Path.Combine(TempRoot.Root, "test.cs") : null;
            var sourceText = CreateText("dummy1");
 
            var documentInfo = DocumentInfo.Create(
                DocumentId.CreateNewId(project.Id, "test"),
                name: "test",
                loader: TextLoader.From(TextAndVersion.Create(sourceText, VersionStamp.Create(), filePath)),
                filePath: filePath)
                .WithDesignTimeOnly(designTimeOnly);
 
            var document = project.Solution.AddDocument(documentInfo).GetDocument(documentInfo.Id);
 
            solution = document.Project.Solution;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            var activeStatements = ImmutableArray.Create(
                new ManagedActiveStatementDebugInfo(
                    new ManagedInstructionId(new ManagedMethodId(Guid.Empty, token: 0x06000001, version: 1), ilOffset: 0),
                    documentName: document.Name,
                    sourceSpan: new SourceSpan(0, 1, 0, 2),
                    ActiveStatementFlags.NonLeafFrame));
 
            EnterBreakState(debuggingSession, activeStatements);
 
            // active statements are not tracked in non-Roslyn projects:
            var currentSpans = await debuggingSession.GetAdjustedActiveStatementSpansAsync(document, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(currentSpans);
 
            var baseSpans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(document.Id), CancellationToken.None);
            Assert.Empty(baseSpans.Single());
 
            // update solution:
            solution = solution.WithDocumentText(document.Id, CreateText("dummy2"));
 
            baseSpans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(document.Id), CancellationToken.None);
            Assert.Empty(baseSpans.Single());
        }
 
        [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/24320")]
        public async Task ActiveStatements_LinkedDocuments()
        {
            var markedSources = new[]
            {
@"class Test1
{
    static void Main() => <AS:2>Project2::Test1.F();</AS:2>
    static void F() => <AS:1>Project4::Test2.M();</AS:1>
}",
@"class Test2 { static void M() => <AS:0>Console.WriteLine();</AS:0> }"
            };
 
            var module1 = Guid.NewGuid();
            var module2 = Guid.NewGuid();
            var module4 = Guid.NewGuid();
 
            var debugInfos = GetActiveStatementDebugInfosCSharp(
                markedSources,
                methodRowIds: new[] { 1, 2, 1 },
                modules: new[] { module4, module2, module1 });
 
            // Project1: Test1.cs, Test2.cs
            // Project2: Test1.cs (link from P1)
            // Project3: Test1.cs (link from P1)
            // Project4: Test2.cs (link from P1)
 
            using var _ = CreateWorkspace(out var solution, out var service);
            solution = AddDefaultTestProject(solution, ActiveStatementsDescription.ClearTags(markedSources));
 
            var documents = solution.Projects.Single().Documents;
            var doc1 = documents.First();
            var doc2 = documents.Skip(1).First();
            var text1 = await doc1.GetTextAsync();
            var text2 = await doc2.GetTextAsync();
 
            DocumentId AddProjectAndLinkDocument(string projectName, Document doc, SourceText text)
            {
                var p = solution.AddProject(projectName, projectName, "C#");
                var linkedDocId = DocumentId.CreateNewId(p.Id, projectName + "->" + doc.Name);
                solution = p.Solution.AddDocument(linkedDocId, doc.Name, text, filePath: doc.FilePath);
                return linkedDocId;
            }
 
            var docId3 = AddProjectAndLinkDocument("Project2", doc1, text1);
            var docId4 = AddProjectAndLinkDocument("Project3", doc1, text1);
            var docId5 = AddProjectAndLinkDocument("Project4", doc2, text2);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
            EnterBreakState(debuggingSession, debugInfos);
 
            // Base Active Statements
 
            var baseActiveStatementsMap = await debuggingSession.EditSession.BaseActiveStatements.GetValueAsync(CancellationToken.None).ConfigureAwait(false);
            var documentMap = baseActiveStatementsMap.DocumentPathMap;
 
            Assert.Equal(2, documentMap.Count);
 
            AssertEx.Equal(new[]
            {
                $"2: {doc1.FilePath}: (2,32)-(2,52) flags=[MethodUpToDate, NonLeafFrame]",
                $"1: {doc1.FilePath}: (3,29)-(3,49) flags=[MethodUpToDate, NonLeafFrame]"
            }, documentMap[doc1.FilePath].Select(InspectActiveStatement));
 
            AssertEx.Equal(new[]
            {
                $"0: {doc2.FilePath}: (0,39)-(0,59) flags=[LeafFrame, MethodUpToDate]",
            }, documentMap[doc2.FilePath].Select(InspectActiveStatement));
 
            Assert.Equal(3, baseActiveStatementsMap.InstructionMap.Count);
 
            var statements = baseActiveStatementsMap.InstructionMap.Values.OrderBy(v => v.Ordinal).ToArray();
            var s = statements[0];
            Assert.Equal(0x06000001, s.InstructionId.Method.Token);
            Assert.Equal(module4, s.InstructionId.Method.Module);
 
            s = statements[1];
            Assert.Equal(0x06000002, s.InstructionId.Method.Token);
            Assert.Equal(module2, s.InstructionId.Method.Module);
 
            s = statements[2];
            Assert.Equal(0x06000001, s.InstructionId.Method.Token);
            Assert.Equal(module1, s.InstructionId.Method.Module);
 
            var spans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(doc1.Id, doc2.Id, docId3, docId4, docId5), CancellationToken.None);
 
            AssertEx.Equal(new[]
            {
                "(2,32)-(2,52), (3,29)-(3,49)", // test1.cs
                "(0,39)-(0,59)",                // test2.cs
                "(2,32)-(2,52), (3,29)-(3,49)", // link test1.cs
                "(2,32)-(2,52), (3,29)-(3,49)", // link test1.cs
                "(0,39)-(0,59)"                 // link test2.cs
            }, spans.Select(docSpans => string.Join(", ", docSpans.Select(span => span.LineSpan))));
        }
 
        [Fact]
        public async Task ActiveStatements_OutOfSyncDocuments()
        {
            var markedSource1 =
@"class C
{
    static void M()
    {
        try
        {
        }
        catch (Exception e)
        {
            <AS:0>M();</AS:0>
        }
    }
}";
            var source2 =
 @"class C
{
    static void M()
    {
        try
        {
        }
        catch (Exception e)
        {
 
            M();
        }
    }
}";
 
            var markedSources = new[] { markedSource1 };
 
            var thread1 = Guid.NewGuid();
 
            // Thread1 stack trace: F (AS:0 leaf)
 
            var debugInfos = GetActiveStatementDebugInfosCSharp(
                markedSources,
                methodRowIds: new[] { 1 },
                ilOffsets: new[] { 1 },
                flags: new[]
                {
                    ActiveStatementFlags.LeafFrame | ActiveStatementFlags.MethodUpToDate
                });
 
            using var _ = CreateWorkspace(out var solution, out var service);
            solution = AddDefaultTestProject(solution, ActiveStatementsDescription.ClearTags(markedSources));
            var project = solution.Projects.Single();
            var document = project.Documents.Single();
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution, initialState: CommittedSolution.DocumentState.OutOfSync);
            EnterBreakState(debuggingSession, debugInfos);
 
            // update document to test a changed solution
            solution = solution.WithDocumentText(document.Id, CreateText(source2));
            document = solution.GetDocument(document.Id);
 
            var baseActiveStatementMap = await debuggingSession.EditSession.BaseActiveStatements.GetValueAsync(CancellationToken.None).ConfigureAwait(false);
 
            // Active Statements - available in out-of-sync documents, as they reflect the state of the debuggee and not the base document content
 
            Assert.Single(baseActiveStatementMap.DocumentPathMap);
 
            AssertEx.Equal(new[]
            {
                $"0: {document.FilePath}: (9,18)-(9,22) flags=[LeafFrame, MethodUpToDate]",
            }, baseActiveStatementMap.DocumentPathMap[document.FilePath].Select(InspectActiveStatement));
 
            Assert.Equal(1, baseActiveStatementMap.InstructionMap.Count);
 
            var activeStatement1 = baseActiveStatementMap.InstructionMap.Values.OrderBy(v => v.InstructionId.Method.Token).Single();
            Assert.Equal(0x06000001, activeStatement1.InstructionId.Method.Token);
            Assert.Equal(document.FilePath, activeStatement1.FilePath);
            Assert.True(activeStatement1.IsLeaf);
 
            // Active statement reported as unchanged as the containing document is out-of-sync:
            var baseSpans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(document.Id), CancellationToken.None);
            AssertEx.Equal(new[] { $"(9,18)-(9,22)" }, baseSpans.Single().Select(s => s.LineSpan.ToString()));
 
            // Whether or not an active statement is in an exception region is unknown if the document is out-of-sync:
            Assert.Null(await debuggingSession.IsActiveStatementInExceptionRegionAsync(solution, activeStatement1.InstructionId, CancellationToken.None));
 
            // Document got synchronized:
            debuggingSession.LastCommittedSolution.Test_SetDocumentState(document.Id, CommittedSolution.DocumentState.MatchesBuildOutput);
 
            // New location of the active statement reported:
            baseSpans = await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(document.Id), CancellationToken.None);
            AssertEx.Equal(new[] { $"(10,12)-(10,16)" }, baseSpans.Single().Select(s => s.LineSpan.ToString()));
 
            Assert.True(await debuggingSession.IsActiveStatementInExceptionRegionAsync(solution, activeStatement1.InstructionId, CancellationToken.None));
        }
 
        [Fact]
        public async Task ActiveStatements_SourceGeneratedDocuments_LineDirectives()
        {
            var markedSource1 = @"
/* GENERATE:
class C
{
    void F()
    {
#line 1 ""a.razor""
       <AS:0>F();</AS:0>
#line default
    }
}
*/
";
            var markedSource2 = @"
/* GENERATE:
class C
{
    void F()
    {
#line 2 ""a.razor""
       <AS:0>F();</AS:0>
#line default
    }
}
*/
";
            var source1 = ActiveStatementsDescription.ClearTags(markedSource1);
            var source2 = ActiveStatementsDescription.ClearTags(markedSource2);
 
            var additionalFileSourceV1 = @"
       xxxxxxxxxxxxxxxxx
";
 
            var generator = new TestSourceGenerator() { ExecuteImpl = GenerateSource };
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document1) = AddDefaultTestProject(solution, source1, generator, additionalFileText: additionalFileSourceV1);
 
            var generatedDocument1 = (await solution.Projects.Single().GetSourceGeneratedDocumentsAsync().ConfigureAwait(false)).Single();
 
            var moduleId = EmitLibrary(source1, generator: generator, additionalFileText: additionalFileSourceV1);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession, GetActiveStatementDebugInfosCSharp(
                new[] { GetGeneratedCodeFromMarkedSource(markedSource1) },
                filePaths: new[] { generatedDocument1.FilePath },
                modules: new[] { moduleId },
                methodRowIds: new[] { 1 },
                methodVersions: new[] { 1 },
                flags: new[]
                {
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame
                }));
 
            // change the source (valid edit)
            solution = solution.WithDocumentText(document1.Id, CreateText(source2));
 
            // validate solution update status and emit:
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            // check emitted delta:
            var delta = updates.Updates.Single();
            Assert.Empty(delta.ActiveStatements);
            Assert.NotEmpty(delta.ILDelta);
            Assert.NotEmpty(delta.MetadataDelta);
            Assert.NotEmpty(delta.PdbDelta);
            Assert.Empty(delta.UpdatedMethods);
            Assert.Empty(delta.UpdatedTypes);
 
            AssertEx.Equal(new[]
            {
                "a.razor: [0 -> 1]"
            }, delta.SequencePoints.Inspect());
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/54347")]
        public async Task ActiveStatements_EncSessionFollowedByHotReload()
        {
            var markedSource1 = @"
class C
{
    int F()
    {
        try
        {
            return 0;
        }
        catch
        {
            <AS:0>return 1;</AS:0>
        }
    }
}
";
            var markedSource2 = @"
class C
{
    int F()
    {
        try
        {
            return 0;
        }
        catch
        {
            <AS:0>return 2;</AS:0>
        }
    }
}
";
            var source1 = ActiveStatementsDescription.ClearTags(markedSource1);
            var source2 = ActiveStatementsDescription.ClearTags(markedSource2);
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, source1);
 
            var moduleId = EmitLibrary(source1);
            LoadLibraryToDebuggee(moduleId);
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EnterBreakState(debuggingSession, GetActiveStatementDebugInfosCSharp(
                new[] { markedSource1 },
                modules: new[] { moduleId },
                methodRowIds: new[] { 1 },
                methodVersions: new[] { 1 },
                flags: new[]
                {
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame
                }));
 
            // change the source (rude edit)
            solution = solution.WithDocumentText(document.Id, CreateText(source2));
            document = solution.GetDocument(document.Id);
 
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document, s_noActiveSpans, CancellationToken.None);
            AssertEx.Equal(new[] { "ENC0063: " + string.Format(FeaturesResources.Updating_a_0_around_an_active_statement_requires_restarting_the_application, CSharpFeaturesResources.catch_clause) },
                diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.RestartRequired, updates.Status);
 
            // undo the change
            solution = solution.WithDocumentText(document.Id, CreateText(source1));
            document = solution.GetDocument(document.Id);
 
            ExitBreakState(debuggingSession, ImmutableArray.Create(document.Id));
 
            // change the source (now a valid edit since there is no active statement)
            solution = solution.WithDocumentText(document.Id, CreateText(source2));
 
            diagnostics = await service.GetDocumentDiagnosticsAsync(document, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            // validate solution update status and emit (Hot Reload change):
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            debuggingSession.DiscardSolutionUpdate();
            EndDebuggingSession(debuggingSession);
        }
 
        /// <summary>
        /// Scenario:
        /// F5 a program that has function F that calls G. G has a long-running loop, which starts executing.
        /// The user makes following operations:
        /// 1) Break, edit F from version 1 to version 2, continue (change is applied), G is still running in its loop
        ///    Function remapping is produced for F v1 -> F v2.
        /// 2) Hot-reload edit F (without breaking) to version 3.
        ///    Function remapping is not produced for F v2 -> F v3. If G ever returned to F it will be remapped from F v1 -> F v2,
        ///    where F v2 is considered stale code. This is consistent with the semantic of Hot Reload: Hot Reloaded changes do not have
        ///    an effect until the method is called again. In this case the method is not called, it it returned into hence the stale
        ///    version executes.
        /// 3) Break and apply EnC edit. This edit is to F v3 (Hot Reload) of the method. We will produce remapping F v3 -> v4.
        /// </summary>
        [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/52100")]
        public async Task BreakStateRemappingFollowedUpByRunStateUpdate()
        {
            var markedSourceV1 =
@"class Test
{
    static bool B() => true;
 
    static void G() { while (B()); <AS:0>}</AS:0>
 
    static void F()
    {
        /*insert1[1]*/B();/*insert2[5]*/B();/*insert3[10]*/B();
        <AS:1>G();</AS:1>
    }
}";
            var markedSourceV2 = Update(markedSourceV1, marker: "1");
            var markedSourceV3 = Update(markedSourceV2, marker: "2");
            var markedSourceV4 = Update(markedSourceV3, marker: "3");
 
            var moduleId = EmitAndLoadLibraryToDebuggee(ActiveStatementsDescription.ClearTags(markedSourceV1));
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, ActiveStatementsDescription.ClearTags(markedSourceV1));
            var documentId = document.Id;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // EnC update F v1 -> v2
 
            EnterBreakState(debuggingSession, GetActiveStatementDebugInfosCSharp(
                new[] { markedSourceV1 },
                modules: new[] { moduleId, moduleId },
                methodRowIds: new[] { 2, 3 },
                methodVersions: new[] { 1, 1 },
                flags: new[]
                {
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame,    // G
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame, // F
                }));
 
            solution = solution.WithDocumentText(documentId, CreateText(ActiveStatementsDescription.ClearTags(markedSourceV2)));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(0x06000003, updates.Updates.Single().UpdatedMethods.Single());
            Assert.Equal(0x02000002, updates.Updates.Single().UpdatedTypes.Single());
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            CommitSolutionUpdate(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                $"0x06000002 v1 | AS {document.FilePath}: (4,41)-(4,42) => (4,41)-(4,42)",
                $"0x06000003 v1 | AS {document.FilePath}: (9,14)-(9,18) => (10,14)-(10,18)",
            }, InspectNonRemappableRegions(debuggingSession.EditSession.NonRemappableRegions));
 
            ExitBreakState(debuggingSession);
 
            // Hot Reload update F v2 -> v3
 
            solution = solution.WithDocumentText(documentId, CreateText(ActiveStatementsDescription.ClearTags(markedSourceV3)));
 
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(0x06000003, updates.Updates.Single().UpdatedMethods.Single());
            Assert.Equal(0x02000002, updates.Updates.Single().UpdatedTypes.Single());
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            CommitSolutionUpdate(debuggingSession);
 
            // the regions remain unchanged
            AssertEx.Equal(new[]
            {
                $"0x06000002 v1 | AS {document.FilePath}: (4,41)-(4,42) => (4,41)-(4,42)",
                $"0x06000003 v1 | AS {document.FilePath}: (9,14)-(9,18) => (10,14)-(10,18)",
            }, InspectNonRemappableRegions(debuggingSession.EditSession.NonRemappableRegions));
 
            // EnC update F v3 -> v4
 
            EnterBreakState(debuggingSession, GetActiveStatementDebugInfosCSharp(
                new[] { markedSourceV1 },       // matches F v1
                modules: new[] { moduleId, moduleId },
                methodRowIds: new[] { 2, 3 },
                methodVersions: new[] { 1, 1 }, // frame F v1 is still executing (G has not returned)
                flags: new[]
                {
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame,    // G
                    ActiveStatementFlags.Stale | ActiveStatementFlags.NonLeafFrame,        // F - not up-to-date anymore and since F v1 is followed by F v3 (hot-reload) it is now stale
                }));
 
            var spans = (await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(documentId), CancellationToken.None)).Single();
            AssertEx.Equal(new[]
            {
                new ActiveStatementSpan(0, new LinePositionSpan(new(4,41), new(4,42)), ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame, unmappedDocumentId: null),
            }, spans);
 
            solution = solution.WithDocumentText(documentId, CreateText(ActiveStatementsDescription.ClearTags(markedSourceV4)));
 
            (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(0x06000003, updates.Updates.Single().UpdatedMethods.Single());
            Assert.Equal(0x02000002, updates.Updates.Single().UpdatedTypes.Single());
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            CommitSolutionUpdate(debuggingSession);
 
            // Stale active statement region is gone.
            AssertEx.Equal(new[]
            {
                $"0x06000002 v1 | AS {document.FilePath}: (4,41)-(4,42) => (4,41)-(4,42)",
            }, InspectNonRemappableRegions(debuggingSession.EditSession.NonRemappableRegions));
 
            ExitBreakState(debuggingSession);
        }
 
        /// <summary>
        /// Scenario:
        /// - F5
        /// - edit, but not apply the edits
        /// - break
        /// </summary>
        [Fact]
        public async Task BreakInPresenceOfUnappliedChanges()
        {
            var markedSource1 =
@"class Test
{
    static bool B() => true;
    static void G() { while (B()); <AS:0>}</AS:0>
 
    static void F()
    {
        <AS:1>G();</AS:1>
    }
}";
 
            var markedSource2 =
@"class Test
{
    static bool B() => true;
    static void G() { while (B()); <AS:0>}</AS:0>
 
    static void F()
    {
        B();
        <AS:1>G();</AS:1>
    }
}";
 
            var markedSource3 =
@"class Test
{
    static bool B() => true;
    static void G() { while (B()); <AS:0>}</AS:0>
 
    static void F()
    {
        B();
        B();
        <AS:1>G();</AS:1>
    }
}";
 
            var moduleId = EmitAndLoadLibraryToDebuggee(ActiveStatementsDescription.ClearTags(markedSource1));
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, ActiveStatementsDescription.ClearTags(markedSource1));
            var documentId = document.Id;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // Update to snapshot 2, but don't apply
 
            solution = solution.WithDocumentText(documentId, CreateText(ActiveStatementsDescription.ClearTags(markedSource2)));
 
            // EnC update F v2 -> v3
 
            EnterBreakState(debuggingSession, GetActiveStatementDebugInfosCSharp(
                new[] { markedSource1 },
                modules: new[] { moduleId, moduleId },
                methodRowIds: new[] { 2, 3 },
                methodVersions: new[] { 1, 1 },
                flags: new[]
                {
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame,    // G
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame, // F
                }));
 
            // check that the active statement is mapped correctly to snapshot v2:
            var expectedSpanG1 = new LinePositionSpan(new LinePosition(3, 41), new LinePosition(3, 42));
            var expectedSpanF1 = new LinePositionSpan(new LinePosition(8, 14), new LinePosition(8, 18));
 
            var activeInstructionF1 = new ManagedInstructionId(new ManagedMethodId(moduleId, 0x06000003, version: 1), ilOffset: 0);
            var span = await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, s_noActiveSpans, activeInstructionF1, CancellationToken.None);
            Assert.Equal(expectedSpanF1, span.Value);
 
            var spans = (await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(documentId), CancellationToken.None)).Single();
            AssertEx.Equal(new[]
            {
                new ActiveStatementSpan(0, expectedSpanG1, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame, documentId),
                new ActiveStatementSpan(1, expectedSpanF1, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame, documentId)
            }, spans);
 
            solution = solution.WithDocumentText(documentId, CreateText(ActiveStatementsDescription.ClearTags(markedSource3)));
 
            // check that the active statement is mapped correctly to snapshot v3:
            var expectedSpanG2 = new LinePositionSpan(new LinePosition(3, 41), new LinePosition(3, 42));
            var expectedSpanF2 = new LinePositionSpan(new LinePosition(9, 14), new LinePosition(9, 18));
 
            span = await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, s_noActiveSpans, activeInstructionF1, CancellationToken.None);
            Assert.Equal(expectedSpanF2, span);
 
            spans = (await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(documentId), CancellationToken.None)).Single();
            AssertEx.Equal(new[]
            {
                new ActiveStatementSpan(0, expectedSpanG2, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame, documentId),
                new ActiveStatementSpan(1, expectedSpanF2, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.NonLeafFrame, documentId)
            }, spans);
 
            // no rude edits:
            var document1 = solution.GetDocument(documentId);
            var diagnostics = await service.GetDocumentDiagnosticsAsync(document1, s_noActiveSpans, CancellationToken.None);
            Assert.Empty(diagnostics);
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(0x06000003, updates.Updates.Single().UpdatedMethods.Single());
            Assert.Equal(0x02000002, updates.Updates.Single().UpdatedTypes.Single());
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            CommitSolutionUpdate(debuggingSession);
 
            AssertEx.Equal(new[]
            {
                $"0x06000002 v1 | AS {document.FilePath}: (3,41)-(3,42) => (3,41)-(3,42)",
                $"0x06000003 v1 | AS {document.FilePath}: (7,14)-(7,18) => (9,14)-(9,18)",
            }, InspectNonRemappableRegions(debuggingSession.EditSession.NonRemappableRegions));
 
            ExitBreakState(debuggingSession);
        }
 
        /// <summary>
        /// Scenario:
        /// - F5
        /// - edit and apply edit that deletes non-leaf active statement
        /// - break
        /// </summary>
        [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/52100")]
        public async Task BreakAfterRunModeChangeDeletesNonLeafActiveStatement()
        {
            var markedSource1 =
@"class Test
{
    static bool B() => true;
    static void G() { while (B()); <AS:0>}</AS:0>
 
    static void F()
    {
        <AS:1>G();</AS:1>
    }
}";
 
            var markedSource2 =
@"class Test
{
    static bool B() => true;
    static void G() { while (B()); <AS:0>}</AS:0>
 
    static void F()
    {
    }
}";
            var moduleId = EmitAndLoadLibraryToDebuggee(ActiveStatementsDescription.ClearTags(markedSource1));
 
            using var _ = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, ActiveStatementsDescription.ClearTags(markedSource1));
            var documentId = document.Id;
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            // Apply update: F v1 -> v2.
 
            solution = solution.WithDocumentText(documentId, CreateText(ActiveStatementsDescription.ClearTags(markedSource2)));
 
            var (updates, emitDiagnostics) = await EmitSolutionUpdateAsync(debuggingSession, solution);
            Assert.Empty(emitDiagnostics);
            Assert.Equal(0x06000003, updates.Updates.Single().UpdatedMethods.Single());
            Assert.Equal(0x02000002, updates.Updates.Single().UpdatedTypes.Single());
            Assert.Equal(ModuleUpdateStatus.Ready, updates.Status);
 
            CommitSolutionUpdate(debuggingSession);
 
            // Break
 
            EnterBreakState(debuggingSession, GetActiveStatementDebugInfosCSharp(
                new[] { markedSource1 },
                modules: new[] { moduleId, moduleId },
                methodRowIds: new[] { 2, 3 },
                methodVersions: new[] { 1, 1 },  // frame F v1 is still executing (G has not returned)
                flags: new[]
                {
                    ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame,    // G
                    ActiveStatementFlags.NonLeafFrame, // F
                }));
 
            // check that the active statement is mapped correctly to snapshot v2:
            var expectedSpanF1 = new LinePositionSpan(new LinePosition(7, 14), new LinePosition(7, 18));
            var expectedSpanG1 = new LinePositionSpan(new LinePosition(3, 41), new LinePosition(3, 42));
 
            var activeInstructionG1 = new ManagedInstructionId(new ManagedMethodId(moduleId, 0x06000002, version: 1), ilOffset: 0);
            var span = await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, s_noActiveSpans, activeInstructionG1, CancellationToken.None);
            Assert.Equal(expectedSpanG1, span);
 
            // Active statement in F has been deleted:
            var activeInstructionF1 = new ManagedInstructionId(new ManagedMethodId(moduleId, 0x06000003, version: 1), ilOffset: 0);
            span = await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, s_noActiveSpans, activeInstructionF1, CancellationToken.None);
            Assert.Null(span);
 
            var spans = (await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray.Create(documentId), CancellationToken.None)).Single();
            AssertEx.Equal(new[]
            {
                new ActiveStatementSpan(0, expectedSpanG1, ActiveStatementFlags.MethodUpToDate | ActiveStatementFlags.LeafFrame, unmappedDocumentId: null)
                // active statement in F has been deleted
            }, spans);
 
            ExitBreakState(debuggingSession);
        }
 
        [Fact]
        public async Task MultiSession()
        {
            var source1 = "class C { void M() { System.Console.WriteLine(); } }";
            var source3 = "class C { void M() { WriteLine(2); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFileA = dir.CreateFile("A.cs").WriteAllText(source1, Encoding.UTF8);
            var moduleId = EmitLibrary(source1, sourceFileA.Path, assemblyName: "Proj");
 
            using var _ = CreateWorkspace(out var solution, out var encService);
 
            var projectP = solution.
                AddProject("P", "P", LanguageNames.CSharp).
                WithMetadataReferences(TargetFrameworkUtil.GetReferences(DefaultTargetFramework));
 
            solution = projectP.Solution;
 
            var documentIdA = DocumentId.CreateNewId(projectP.Id, debugName: "A");
            solution = solution.AddDocument(DocumentInfo.Create(
                id: documentIdA,
                name: "A",
                loader: new WorkspaceFileTextLoader(solution.Services, sourceFileA.Path, Encoding.UTF8),
                filePath: sourceFileA.Path));
 
            var tasks = Enumerable.Range(0, 10).Select(async i =>
            {
                var sessionId = await encService.StartDebuggingSessionAsync(
                    solution,
                    _debuggerService,
                    NullPdbMatchingSourceTextProvider.Instance,
                    captureMatchingDocuments: ImmutableArray<DocumentId>.Empty,
                    captureAllMatchingDocuments: true,
                    reportDiagnostics: true,
                    CancellationToken.None);
 
                var solution1 = solution.WithDocumentText(documentIdA, CreateText("class C { void M() { System.Console.WriteLine(" + i + "); } }"));
 
                var result1 = await encService.EmitSolutionUpdateAsync(sessionId, solution1, s_noActiveSpans, CancellationToken.None);
                Assert.Empty(result1.Diagnostics);
                Assert.Equal(1, result1.ModuleUpdates.Updates.Length);
                encService.DiscardSolutionUpdate(sessionId);
 
                var solution2 = solution1.WithDocumentText(documentIdA, CreateText(source3));
 
                var result2 = await encService.EmitSolutionUpdateAsync(sessionId, solution2, s_noActiveSpans, CancellationToken.None);
                Assert.Equal("CS0103", result2.Diagnostics.Single().Diagnostics.Single().Id);
                Assert.Empty(result2.ModuleUpdates.Updates);
 
                encService.EndDebuggingSession(sessionId, out var _);
            });
 
            await Task.WhenAll(tasks);
 
            Assert.Empty(encService.GetTestAccessor().GetActiveDebuggingSessions());
        }
 
        [Fact]
        public async Task Disposal()
        {
            using var _1 = CreateWorkspace(out var solution, out var service);
            (solution, var document) = AddDefaultTestProject(solution, "class C { }");
 
            var debuggingSession = await StartDebuggingSessionAsync(service, solution);
 
            EndDebuggingSession(debuggingSession);
 
            // The folling methods shall not be called after the debugging session ended.
            await Assert.ThrowsAsync<ObjectDisposedException>(async () => await debuggingSession.EmitSolutionUpdateAsync(solution, s_noActiveSpans, CancellationToken.None));
            await Assert.ThrowsAsync<ObjectDisposedException>(async () => await debuggingSession.GetCurrentActiveStatementPositionAsync(solution, s_noActiveSpans, instructionId: default, CancellationToken.None));
            await Assert.ThrowsAsync<ObjectDisposedException>(async () => await debuggingSession.IsActiveStatementInExceptionRegionAsync(solution, instructionId: default, CancellationToken.None));
            Assert.Throws<ObjectDisposedException>(() => debuggingSession.BreakStateOrCapabilitiesChanged(inBreakState: true, out _));
            Assert.Throws<ObjectDisposedException>(() => debuggingSession.DiscardSolutionUpdate());
            Assert.Throws<ObjectDisposedException>(() => debuggingSession.CommitSolutionUpdate(out _));
            Assert.Throws<ObjectDisposedException>(() => debuggingSession.EndSession(out _, out _));
 
            // The following methods can be called at any point in time, so we must handle race with dispose gracefully.
            Assert.Empty(await debuggingSession.GetDocumentDiagnosticsAsync(document, s_noActiveSpans, CancellationToken.None));
            Assert.Empty(await debuggingSession.GetAdjustedActiveStatementSpansAsync(document, s_noActiveSpans, CancellationToken.None));
            Assert.True((await debuggingSession.GetBaseActiveStatementSpansAsync(solution, ImmutableArray<DocumentId>.Empty, CancellationToken.None)).IsDefault);
        }
 
        [Fact]
        public async Task WatchHotReloadServiceTest()
        {
            // See https://github.com/dotnet/sdk/blob/main/src/BuiltInTools/dotnet-watch/HotReload/CompilationHandler.cs#L125
 
            var source1 = "class C { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class C { void M() { System.Console.WriteLine(2); } }";
            var source3 = "class C { void M<T>() { System.Console.WriteLine(2); } }";
            var source4 = "class C { void M() { System.Console.WriteLine(2)/* missing semicolon */ }";
 
            var dir = Temp.CreateDirectory();
            var sourceFileA = dir.CreateFile("A.cs").WriteAllText(source1, Encoding.UTF8);
            var moduleId = EmitLibrary(source1, sourceFileA.Path, assemblyName: "Proj");
 
            using var workspace = CreateWorkspace(out var solution, out var encService);
 
            var projectId = ProjectId.CreateNewId();
            var projectP = solution.
                AddProject(ProjectInfo.Create(projectId, VersionStamp.Create(), "P", "P", LanguageNames.CSharp, parseOptions: CSharpParseOptions.Default.WithNoRefSafetyRulesAttribute())).GetProject(projectId).
                WithMetadataReferences(TargetFrameworkUtil.GetReferences(DefaultTargetFramework));
 
            solution = projectP.Solution;
 
            var documentIdA = DocumentId.CreateNewId(projectP.Id, debugName: "A");
            solution = solution.AddDocument(DocumentInfo.Create(
                id: documentIdA,
                name: "A",
                loader: new WorkspaceFileTextLoader(solution.Services, sourceFileA.Path, Encoding.UTF8),
                filePath: sourceFileA.Path));
 
            var hotReload = new WatchHotReloadService(workspace.Services, ImmutableArray.Create("Baseline", "AddDefinitionToExistingType", "NewTypeDefinition"));
 
            await hotReload.StartSessionAsync(solution, CancellationToken.None);
 
            var sessionId = hotReload.GetTestAccessor().SessionId;
            var session = encService.GetTestAccessor().GetDebuggingSession(sessionId);
            var matchingDocuments = session.LastCommittedSolution.Test_GetDocumentStates();
            AssertEx.Equal(new[]
            {
                "(A, MatchesBuildOutput)"
            }, matchingDocuments.Select(e => (solution.GetDocument(e.id).Name, e.state)).OrderBy(e => e.Name).Select(e => e.ToString()));
 
            // Valid update:
            solution = solution.WithDocumentText(documentIdA, CreateText(source2));
 
            var result = await hotReload.EmitSolutionUpdateAsync(solution, CancellationToken.None);
            Assert.Empty(result.diagnostics);
            Assert.Equal(1, result.updates.Length);
            AssertEx.Equal(new[] { 0x02000002 }, result.updates[0].UpdatedTypes);
 
            // Rude edit:
            solution = solution.WithDocumentText(documentIdA, CreateText(source3));
 
            result = await hotReload.EmitSolutionUpdateAsync(solution, CancellationToken.None);
            AssertEx.Equal(
                new[] { "ENC0021: " + string.Format(FeaturesResources.Adding_0_requires_restarting_the_application, FeaturesResources.type_parameter) },
                result.diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            Assert.Empty(result.updates);
 
            // Syntax error (not reported in diagnostics):
            solution = solution.WithDocumentText(documentIdA, CreateText(source4));
 
            result = await hotReload.EmitSolutionUpdateAsync(solution, CancellationToken.None);
            Assert.Empty(result.diagnostics);
            Assert.Empty(result.updates);
 
            hotReload.EndSession();
        }
 
        [Fact]
        public async Task UnitTestingHotReloadServiceTest()
        {
            var source1 = "class C { void M() { System.Console.WriteLine(1); } }";
            var source2 = "class C { void M() { System.Console.WriteLine(2); } }";
            var source3 = "class C { void M<T>() { System.Console.WriteLine(2); } }";
            var source4 = "class C { void M() { System.Console.WriteLine(2)/* missing semicolon */ }";
 
            var dir = Temp.CreateDirectory();
            var sourceFileA = dir.CreateFile("A.cs").WriteAllText(source1, Encoding.UTF8);
            var moduleId = EmitLibrary(source1, sourceFileA.Path, assemblyName: "Proj");
 
            using var workspace = CreateWorkspace(out var solution, out var encService);
 
            var projectP = solution.
                AddProject("P", "P", LanguageNames.CSharp).
                WithMetadataReferences(TargetFrameworkUtil.GetReferences(DefaultTargetFramework));
 
            solution = projectP.Solution;
 
            var documentIdA = DocumentId.CreateNewId(projectP.Id, debugName: "A");
            solution = solution.AddDocument(DocumentInfo.Create(
                id: documentIdA,
                name: "A",
                loader: new WorkspaceFileTextLoader(solution.Services, sourceFileA.Path, Encoding.UTF8),
                filePath: sourceFileA.Path));
 
            var hotReload = new UnitTestingHotReloadService(workspace.Services);
 
            await hotReload.StartSessionAsync(solution, ImmutableArray.Create("Baseline", "AddDefinitionToExistingType", "NewTypeDefinition"), CancellationToken.None);
 
            var sessionId = hotReload.GetTestAccessor().SessionId;
            var session = encService.GetTestAccessor().GetDebuggingSession(sessionId);
            var matchingDocuments = session.LastCommittedSolution.Test_GetDocumentStates();
            AssertEx.Equal(new[]
            {
                "(A, MatchesBuildOutput)"
            }, matchingDocuments.Select(e => (solution.GetDocument(e.id).Name, e.state)).OrderBy(e => e.Name).Select(e => e.ToString()));
 
            // Valid change
            solution = solution.WithDocumentText(documentIdA, CreateText(source2));
 
            var result = await hotReload.EmitSolutionUpdateAsync(solution, commitUpdates: true, CancellationToken.None);
            Assert.Empty(result.diagnostics);
            Assert.Equal(1, result.updates.Length);
 
            solution = solution.WithDocumentText(documentIdA, CreateText(source3));
 
            // Rude edit
            result = await hotReload.EmitSolutionUpdateAsync(solution, commitUpdates: true, CancellationToken.None);
            AssertEx.Equal(
                new[] { "ENC0021: " + string.Format(FeaturesResources.Adding_0_requires_restarting_the_application, FeaturesResources.type_parameter) },
                result.diagnostics.Select(d => $"{d.Id}: {d.GetMessage()}"));
 
            Assert.Empty(result.updates);
 
            // Syntax error is reported in the diagnostics:
            solution = solution.WithDocumentText(documentIdA, CreateText(source4));
 
            result = await hotReload.EmitSolutionUpdateAsync(solution, commitUpdates: true, CancellationToken.None);
            Assert.Equal(1, result.diagnostics.Length);
            Assert.Empty(result.updates);
 
            hotReload.EndSession();
        }
 
        [Fact]
        public async Task DefaultPdbMatchingSourceTextProvider()
        {
            var source1 = "class C1 { void M() { System.Console.WriteLine(\"a\"); } }";
            var source2 = "class C1 { void M() { System.Console.WriteLine(\"b\"); } }";
            var source3 = "class C1 { void M() { System.Console.WriteLine(\"c\"); } }";
 
            var dir = Temp.CreateDirectory();
            var sourceFile = dir.CreateFile("test.cs").WriteAllText(source1, Encoding.UTF8);
 
            using var workspace = CreateEditorWorkspace(out var solution, out var service, out var languageService);
            var sourceTextProvider = workspace.GetService<PdbMatchingSourceTextProvider>();
 
            var projectId = ProjectId.CreateNewId();
            var documentId = DocumentId.CreateNewId(projectId);
 
            solution = solution.
                AddProject(projectId, "test", "test", LanguageNames.CSharp).
                WithProjectChecksumAlgorithm(projectId, SourceHashAlgorithms.Default).
                AddMetadataReferences(projectId, TargetFrameworkUtil.GetReferences(TargetFramework.Mscorlib40)).
                AddDocument(DocumentInfo.Create(
                    documentId,
                    name: "test.cs",
                    loader: new WorkspaceFileTextLoader(workspace.Services.SolutionServices, sourceFile.Path, Encoding.UTF8),
                    filePath: sourceFile.Path));
 
            Assert.True(workspace.SetCurrentSolution(_ => solution, WorkspaceChangeKind.SolutionAdded));
            solution = workspace.CurrentSolution;
 
            var moduleId = EmitAndLoadLibraryToDebuggee(source1, sourceFilePath: sourceFile.Path);
 
            // hydrate document text and overwrite file content:
            var document1 = await solution.GetDocument(documentId).GetTextAsync();
            File.WriteAllText(sourceFile.Path, source2, Encoding.UTF8);
 
            await languageService.StartSessionAsync(CancellationToken.None);
            await languageService.EnterBreakStateAsync(CancellationToken.None);
 
            workspace.OnDocumentOpened(documentId, new TestSourceTextContainer()
            {
                Text = SourceText.From(source3, Encoding.UTF8, SourceHashAlgorithm.Sha1)
            });
 
            await workspace.GetService<AsynchronousOperationListenerProvider>().GetWaiter(FeatureAttribute.Workspace).ExpeditedWaitAsync();
 
            var (key, (documentState, version)) = sourceTextProvider.GetTestAccessor().GetDocumentsWithChangedLoaderByPath().Single();
            Assert.Equal(sourceFile.Path, key);
            Assert.Equal(solution.WorkspaceVersion, version);
            Assert.Equal(source1, (await documentState.GetTextAsync(CancellationToken.None)).ToString());
 
            // check committed document status:
            var debuggingSession = service.GetTestAccessor().GetActiveDebuggingSessions().Single();
            var (document, state) = await debuggingSession.LastCommittedSolution.GetDocumentAndStateAsync(documentId, currentDocument: null, CancellationToken.None);
            var text = await document.GetTextAsync();
            Assert.Equal(CommittedSolution.DocumentState.MatchesBuildOutput, state);
            Assert.Equal(source1, (await document.GetTextAsync(CancellationToken.None)).ToString());
 
            await languageService.EndSessionAsync(CancellationToken.None);
        }
    }
}