File: src\Compilers\Core\Portable\InternalUtilities\SharedStopwatch.cs
Web Access
Project: src\src\Workspaces\Core\Portable\Microsoft.CodeAnalysis.Workspaces.csproj (Microsoft.CodeAnalysis.Workspaces)
// 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.Diagnostics;
 
namespace Roslyn.Utilities
{
    internal readonly struct SharedStopwatch
    {
        private static readonly Stopwatch s_stopwatch = Stopwatch.StartNew();
 
        private readonly TimeSpan _started;
 
        private SharedStopwatch(TimeSpan started)
        {
            _started = started;
        }
 
        public TimeSpan Elapsed => s_stopwatch.Elapsed - _started;
 
        public static SharedStopwatch StartNew()
        {
            // This call to StartNewCore isn't required, but is included to avoid measurement errors
            // which can occur during periods of high allocation activity. In some cases, calls to Stopwatch
            // operations can block at their return point on the completion of a background GC operation. When
            // this occurs, the GC wait time ends up included in the measured time span. In the event the first
            // call to StartNewCore blocked on a GC operation, the second call will most likely occur when the
            // GC is no longer active. In practice, a substantial improvement to the consistency of analyzer
            // timing data was observed.
            //
            // Note that the call to SharedStopwatch.Elapsed is not affected, because the GC wait will occur
            // after the timer has already recorded its stop time.
            _ = StartNewCore();
            return StartNewCore();
        }
 
        private static SharedStopwatch StartNewCore()
            => new SharedStopwatch(s_stopwatch.Elapsed);
    }
}