From 35f87ecf04416831c5675617b2bda4e2a031592f Mon Sep 17 00:00:00 2001 From: Stephen Toub Date: Sat, 11 Mar 2023 11:36:40 -0500 Subject: [PATCH] Avoid synchronizing access to TextWriter.Null in Console (#83296) If SetOut/Error are used to silence Console by using TextWriter.Null, we don't need to wrap the writer in a synchronized one like we normally do, since all operations are nops and there's no need to serialize access. --- .../System.Console/src/System/Console.cs | 29 ++++++++++++++----- 1 file changed, 21 insertions(+), 8 deletions(-) diff --git a/src/libraries/System.Console/src/System/Console.cs b/src/libraries/System.Console/src/System/Console.cs index cecef0a7e5b1fa..9a297e6c36b0b7 100644 --- a/src/libraries/System.Console/src/System/Console.cs +++ b/src/libraries/System.Console/src/System/Console.cs @@ -235,16 +235,16 @@ static TextWriter EnsureInitialized() private static TextWriter CreateOutputWriter(Stream outputStream) { - return TextWriter.Synchronized(outputStream == Stream.Null ? - StreamWriter.Null : - new StreamWriter( + return outputStream == Stream.Null ? + TextWriter.Null : + TextWriter.Synchronized(new StreamWriter( stream: outputStream, encoding: OutputEncoding.RemovePreamble(), // This ensures no prefix is written to the stream. bufferSize: WriteBufferSize, leaveOpen: true) - { - AutoFlush = true - }); + { + AutoFlush = true + }); } private static StrongBox? _isStdInRedirected; @@ -696,7 +696,15 @@ public static void SetOut(TextWriter newOut) { ArgumentNullException.ThrowIfNull(newOut); - newOut = TextWriter.Synchronized(newOut); + // Ensure all access to the writer is synchronized. If it's the known Null + // singleton writer, which may be used if someone wants to suppress all + // console output, we needn't add synchronization because all operations + // are nops. + if (newOut != TextWriter.Null) + { + newOut = TextWriter.Synchronized(newOut); + } + lock (s_syncObject) { s_isOutTextWriterRedirected = true; @@ -708,7 +716,12 @@ public static void SetError(TextWriter newError) { ArgumentNullException.ThrowIfNull(newError); - newError = TextWriter.Synchronized(newError); + // Ensure all access to the writer is synchronized. See comment in SetOut. + if (newError != TextWriter.Null) + { + newError = TextWriter.Synchronized(newError); + } + lock (s_syncObject) { s_isErrorTextWriterRedirected = true;