diff --git a/src/Nethermind/Nethermind.Consensus/Processing/BlockchainProcessor.cs b/src/Nethermind/Nethermind.Consensus/Processing/BlockchainProcessor.cs index 3b94c7a7487..b302da9177e 100644 --- a/src/Nethermind/Nethermind.Consensus/Processing/BlockchainProcessor.cs +++ b/src/Nethermind/Nethermind.Consensus/Processing/BlockchainProcessor.cs @@ -164,7 +164,7 @@ public async Task StopAsync(bool processRemainingBlocks = false) _blockQueue.CompleteAdding(); } - await Task.WhenAll((_recoveryTask ?? Task.CompletedTask), (_processorTask ?? Task.CompletedTask)); + await Task.WhenAll(_recoveryTask ?? Task.CompletedTask, _processorTask ?? Task.CompletedTask); if (_logger.IsInfo) _logger.Info("Blockchain Processor shutdown complete.. please wait for all components to close"); } diff --git a/src/Nethermind/Nethermind.Consensus/Producers/MultipleBlockProducer.cs b/src/Nethermind/Nethermind.Consensus/Producers/MultipleBlockProducer.cs index 57836b2e474..13f9f26e28a 100644 --- a/src/Nethermind/Nethermind.Consensus/Producers/MultipleBlockProducer.cs +++ b/src/Nethermind/Nethermind.Consensus/Producers/MultipleBlockProducer.cs @@ -7,6 +7,7 @@ using System.Threading; using System.Threading.Tasks; using Nethermind.Core; +using Nethermind.Core.Collections; using Nethermind.Evm.Tracing; using Nethermind.Logging; @@ -32,23 +33,20 @@ protected MultipleBlockProducer( public async Task BuildBlock(BlockHeader? parentHeader, IBlockTracer? blockTracer = null, PayloadAttributes? payloadAttributes = null, CancellationToken? token = null) { - Task[] produceTasks = new Task[_blockProducers.Length]; + using ArrayPoolList> produceTasks = new(_blockProducers.Length); for (int i = 0; i < _blockProducers.Length; i++) { T blockProducerInfo = _blockProducers[i]; - if (!blockProducerInfo.Condition.CanProduce(parentHeader)) - { - produceTasks[i] = Task.FromResult(null); - continue; - } - produceTasks[i] = blockProducerInfo.BlockProducer.BuildBlock(parentHeader, blockProducerInfo.BlockTracer, cancellationToken: token); + produceTasks.Add(!blockProducerInfo.Condition.CanProduce(parentHeader!) + ? Task.FromResult(null) + : blockProducerInfo.BlockProducer.BuildBlock(parentHeader, blockProducerInfo.BlockTracer, cancellationToken: token)); } IEnumerable<(Block? Block, T BlockProducer)> blocksWithProducers; try { - Block?[] blocks = await Task.WhenAll(produceTasks); + Block?[] blocks = await Task.WhenAll(produceTasks.AsSpan()); blocksWithProducers = blocks.Zip(_blockProducers); } catch (OperationCanceledException) diff --git a/src/Nethermind/Nethermind.Core.Test/Encoding/TxDecoderTests.cs b/src/Nethermind/Nethermind.Core.Test/Encoding/TxDecoderTests.cs index ad7f29b2668..b2cb2ecdfcd 100644 --- a/src/Nethermind/Nethermind.Core.Test/Encoding/TxDecoderTests.cs +++ b/src/Nethermind/Nethermind.Core.Test/Encoding/TxDecoderTests.cs @@ -7,6 +7,7 @@ using System.Threading.Tasks; using FluentAssertions; using FluentAssertions.Numeric; +using Nethermind.Core.Collections; using Nethermind.Core.Crypto; using Nethermind.Core.Eip2930; using Nethermind.Core.Extensions; @@ -66,7 +67,7 @@ public class TxDecoderTests [TestCaseSource(nameof(TestCaseSource))] [Repeat(10)] // Might wanna increase this to double check when changing logic as on lower value, it does not reproduce. - public void CanCorrectlyCalculateTxHash_when_called_concurrently((Transaction Tx, string Description) testCase) + public async Task CanCorrectlyCalculateTxHash_when_called_concurrently((Transaction Tx, string Description) testCase) { Transaction tx = testCase.Tx; @@ -78,14 +79,12 @@ public void CanCorrectlyCalculateTxHash_when_called_concurrently((Transaction Tx decodedTx.SetPreHash(rlp.Bytes); - IEnumerable>>> tasks = Enumerable + using ArrayPoolList>>> tasks = Enumerable .Range(0, 32) - .Select((_) => - Task.Factory - .StartNew(() => decodedTx.Hash.Should().Be(expectedHash), - TaskCreationOptions.RunContinuationsAsynchronously)); + .Select(_ => Task.Factory.StartNew(() => decodedTx.Hash.Should().Be(expectedHash), TaskCreationOptions.RunContinuationsAsynchronously)) + .ToPooledList(32); - Task.WaitAll(tasks.ToArray()); + await Task.WhenAll>>(tasks.AsSpan()); } [TestCaseSource(nameof(TestCaseSource))] diff --git a/src/Nethermind/Nethermind.Core/PubSub/CompositePublisher.cs b/src/Nethermind/Nethermind.Core/PubSub/CompositePublisher.cs index cfb3541fa4b..02e1a8addc5 100644 --- a/src/Nethermind/Nethermind.Core/PubSub/CompositePublisher.cs +++ b/src/Nethermind/Nethermind.Core/PubSub/CompositePublisher.cs @@ -1,34 +1,28 @@ // SPDX-FileCopyrightText: 2022 Demerzel Solutions Limited // SPDX-License-Identifier: LGPL-3.0-only +using System; using System.Threading.Tasks; +using Nethermind.Core.Collections; namespace Nethermind.Core.PubSub { - public class CompositePublisher : IPublisher + public class CompositePublisher(params IPublisher[] publishers) : IPublisher { - private readonly IPublisher[] _publishers; - - public CompositePublisher(params IPublisher[] publishers) - { - _publishers = publishers; - } - public async Task PublishAsync(T data) where T : class { - // TODO: .Net 9 stackalloc - Task[] tasks = new Task[_publishers.Length]; - for (int i = 0; i < _publishers.Length; i++) + using ArrayPoolList tasks = new(publishers.Length); + for (int i = 0; i < publishers.Length; i++) { - tasks[i] = _publishers[i].PublishAsync(data); + tasks.Add(publishers[i].PublishAsync(data)); } - await Task.WhenAll(tasks); + await Task.WhenAll(tasks.AsSpan()); } public void Dispose() { - foreach (IPublisher publisher in _publishers) + foreach (IPublisher publisher in publishers) { publisher.Dispose(); } diff --git a/src/Nethermind/Nethermind.Db/RocksDbInitializer.cs b/src/Nethermind/Nethermind.Db/RocksDbInitializer.cs index 9e0e61cf742..26be4d0ff27 100644 --- a/src/Nethermind/Nethermind.Db/RocksDbInitializer.cs +++ b/src/Nethermind/Nethermind.Db/RocksDbInitializer.cs @@ -4,6 +4,7 @@ using System; using System.Collections.Generic; using System.Threading.Tasks; +using Nethermind.Core.Collections; namespace Nethermind.Db { @@ -68,13 +69,13 @@ protected void InitAll() protected async Task InitAllAsync() { - HashSet allInitializers = new(); - foreach (var registration in _registrations) + using ArrayPoolList allInitializers = new(_registrations.Count); + foreach (Action registration in _registrations) { allInitializers.Add(Task.Run(() => registration.Invoke())); } - await Task.WhenAll(allInitializers); + await Task.WhenAll(allInitializers.AsSpan()); } protected static string GetTitleDbName(string dbName) => char.ToUpper(dbName[0]) + dbName[1..]; diff --git a/src/Nethermind/Nethermind.Evm.Test/Tracing/GethLikeCallTracerTests.cs b/src/Nethermind/Nethermind.Evm.Test/Tracing/GethLikeCallTracerTests.cs index af49a60c100..10eea16fa7a 100644 --- a/src/Nethermind/Nethermind.Evm.Test/Tracing/GethLikeCallTracerTests.cs +++ b/src/Nethermind/Nethermind.Evm.Test/Tracing/GethLikeCallTracerTests.cs @@ -17,7 +17,7 @@ namespace Nethermind.Evm.Test.Tracing; [TestFixture] public class GethLikeCallTracerTests : VirtualMachineTestsBase { - private static readonly JsonSerializerOptions SerializerOptions = EthereumJsonSerializer.JsonOptionsIndented; + private static readonly JsonSerializerOptions SerializerOptions = new(EthereumJsonSerializer.JsonOptionsIndented) { NewLine = "\n" }; private const string? WithLog = """{"withLog":true}"""; private const string? OnlyTopCall = """{"onlyTopCall":true}"""; private const string? WithLogAndOnlyTopCall = """{"withLog":true,"onlyTopCall":true}"""; @@ -27,14 +27,8 @@ private string ExecuteCallTrace(byte[] code, string? tracerConfig = null) (_, Transaction tx) = PrepareTx(MainnetSpecProvider.CancunActivation, 100000, code); NativeCallTracer tracer = new(tx, GetGethTraceOptions(tracerConfig)); - GethLikeTxTrace callTrace = Execute( - tracer, - code, - MainnetSpecProvider.CancunActivation) - .BuildResult(); - return JsonSerializer.Serialize(callTrace.CustomTracerResult?.Value, SerializerOptions) - // fix for windows, can be done better in .NET 9: https://github.com/dotnet/runtime/issues/84117 - .ReplaceLineEndings("\n"); + GethLikeTxTrace callTrace = Execute(tracer, code, MainnetSpecProvider.CancunActivation).BuildResult(); + return JsonSerializer.Serialize(callTrace.CustomTracerResult?.Value, SerializerOptions); } private static GethTraceOptions GetGethTraceOptions(string? config) => GethTraceOptions.Default with diff --git a/src/Nethermind/Nethermind.Network.Discovery/NodesLocator.cs b/src/Nethermind/Nethermind.Network.Discovery/NodesLocator.cs index ef0a4b3d7f2..254bedef24b 100644 --- a/src/Nethermind/Nethermind.Network.Discovery/NodesLocator.cs +++ b/src/Nethermind/Nethermind.Network.Discovery/NodesLocator.cs @@ -3,7 +3,9 @@ using System.Text; using Nethermind.Core; +using Nethermind.Core.Collections; using Nethermind.Core.Crypto; +using Nethermind.Core.Extensions; using Nethermind.Logging; using Nethermind.Network.Discovery.Lifecycle; using Nethermind.Network.Discovery.Messages; @@ -128,8 +130,8 @@ public async Task LocateNodesAsync(byte[]? searchedNodeId, CancellationToken can int count = failRequestCount > 0 ? failRequestCount : _discoveryConfig.Concurrency; IEnumerable nodesToSend = tryCandidates.Skip(nodesTriedCount).Take(count); - IEnumerable> sendFindNodeTasks = SendFindNodes(searchedNodeId, nodesToSend, alreadyTriedNodes); - Result?[] results = await Task.WhenAll(sendFindNodeTasks); + using ArrayPoolList> sendFindNodeTasks = SendFindNodes(searchedNodeId, nodesToSend, alreadyTriedNodes).ToPooledList(count); + Result[] results = await Task.WhenAll(sendFindNodeTasks.AsSpan()); if (results.Length == 0) { diff --git a/src/Nethermind/Nethermind.Network/CompositeNodeSource.cs b/src/Nethermind/Nethermind.Network/CompositeNodeSource.cs index 1402ebcf725..811eb1c3a81 100644 --- a/src/Nethermind/Nethermind.Network/CompositeNodeSource.cs +++ b/src/Nethermind/Nethermind.Network/CompositeNodeSource.cs @@ -8,6 +8,8 @@ using System.Threading; using System.Threading.Channels; using System.Threading.Tasks; +using Nethermind.Core.Collections; +using Nethermind.Core.Extensions; using Nethermind.Stats.Model; namespace Nethermind.Network; @@ -20,14 +22,13 @@ public async IAsyncEnumerable DiscoverNodes([EnumeratorCancellation] Cance { Channel ch = Channel.CreateBounded(1); - // TODO: .Net 9 stackalloc - Task[] feedTasks = _nodeSources.Select(async innerSource => + using ArrayPoolList feedTasks = _nodeSources.Select(async innerSource => { await foreach (Node node in innerSource.DiscoverNodes(cancellationToken)) { await ch.Writer.WriteAsync(node, cancellationToken); } - }).ToArray(); + }).ToPooledList(_nodeSources.Length * 16); try { @@ -38,7 +39,7 @@ public async IAsyncEnumerable DiscoverNodes([EnumeratorCancellation] Cance } finally { - await Task.WhenAll(feedTasks); + await Task.WhenAll(feedTasks.AsSpan()); } } diff --git a/src/Nethermind/Nethermind.Network/PeerManager.cs b/src/Nethermind/Nethermind.Network/PeerManager.cs index acad3a3d396..9984806be4a 100644 --- a/src/Nethermind/Nethermind.Network/PeerManager.cs +++ b/src/Nethermind/Nethermind.Network/PeerManager.cs @@ -12,7 +12,9 @@ using FastEnumUtility; using Nethermind.Core; using Nethermind.Core.Attributes; +using Nethermind.Core.Collections; using Nethermind.Core.Crypto; +using Nethermind.Core.Extensions; using Nethermind.Logging; using Nethermind.Network.Config; using Nethermind.Network.P2P; @@ -206,7 +208,7 @@ private class CandidateSelection private async Task RunPeerUpdateLoop() { Channel taskChannel = Channel.CreateBounded(1); - List? tasks = Enumerable.Range(0, _outgoingConnectParallelism).Select(async (idx) => + using ArrayPoolList tasks = Enumerable.Range(0, _outgoingConnectParallelism).Select(async idx => { await foreach (Peer peer in taskChannel.Reader.ReadAllAsync(_cancellationTokenSource.Token)) { @@ -226,7 +228,7 @@ private async Task RunPeerUpdateLoop() } } if (_logger.IsDebug) _logger.Debug($"Connect worker {idx} completed"); - }).ToList(); + }).ToPooledList(_outgoingConnectParallelism); int loopCount = 0; long previousActivePeersCount = 0; @@ -359,7 +361,7 @@ private async Task RunPeerUpdateLoop() } taskChannel.Writer.Complete(); - await Task.WhenAll(tasks); + await Task.WhenAll(tasks.AsSpan()); } private bool EnsureAvailableActivePeerSlot() diff --git a/src/Nethermind/Nethermind.Network/SessionMonitor.cs b/src/Nethermind/Nethermind.Network/SessionMonitor.cs index 0c2f73f2fbc..11ae1942a5e 100644 --- a/src/Nethermind/Nethermind.Network/SessionMonitor.cs +++ b/src/Nethermind/Nethermind.Network/SessionMonitor.cs @@ -5,6 +5,7 @@ using System.Collections.Concurrent; using System.Collections.Generic; using System.Linq; +using System.Runtime.InteropServices; using System.Threading; using System.Threading.Tasks; @@ -84,7 +85,7 @@ private async Task SendPingMessagesAsync() if (_pingTasks.Count > 0) { - bool[] tasks = await Task.WhenAll(_pingTasks); + bool[] tasks = await Task.WhenAll(CollectionsMarshal.AsSpan(_pingTasks)); int tasksLength = tasks.Length; if (tasksLength != 0) { diff --git a/src/Nethermind/Nethermind.State/PersistentStorageProvider.cs b/src/Nethermind/Nethermind.State/PersistentStorageProvider.cs index 95b9b6f2fc8..9bbbf99282d 100644 --- a/src/Nethermind/Nethermind.State/PersistentStorageProvider.cs +++ b/src/Nethermind/Nethermind.State/PersistentStorageProvider.cs @@ -347,7 +347,7 @@ public void CommitTrees(IBlockCommitter blockCommitter) } } - Task.WaitAll(commitTask); + Task.WaitAll(commitTask.AsSpan()); _toUpdateRoots.Clear(); // only needed here as there is no control over cached storage size otherwise diff --git a/src/Nethermind/Nethermind.Synchronization/Trie/TrieNodeRecovery.cs b/src/Nethermind/Nethermind.Synchronization/Trie/TrieNodeRecovery.cs index fa1c10d99ec..85c20a2da21 100644 --- a/src/Nethermind/Nethermind.Synchronization/Trie/TrieNodeRecovery.cs +++ b/src/Nethermind/Nethermind.Synchronization/Trie/TrieNodeRecovery.cs @@ -2,6 +2,7 @@ // SPDX-License-Identifier: LGPL-3.0-only using System; +using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; @@ -47,7 +48,8 @@ protected TrieNodeRecovery(ISyncPeerPool syncPeerPool, ILogManager? logManager) { while (keyRecoveries.Count > 0) { - Task<(Recovery, byte[]?)> task = await Task.WhenAny(keyRecoveries.Select(kr => kr.Task!)); + using ArrayPoolList>? tasks = keyRecoveries.Select(kr => kr.Task!).ToPooledList(keyRecoveries.Count); + Task<(Recovery, byte[]?)> task = await Task.WhenAny<(Recovery, byte[]?)>(tasks.AsSpan()); (Recovery Recovery, byte[]? Data) result = await task; if (result.Data is null) { @@ -57,7 +59,7 @@ protected TrieNodeRecovery(ISyncPeerPool syncPeerPool, ILogManager? logManager) else { if (_logger.IsWarn) _logger.Warn($"Successfully recovered from peer {result.Recovery.Peer} with {result.Data.Length} bytes!"); - cts.Cancel(); + await cts.CancelAsync(); return result.Data; } } diff --git a/src/Nethermind/Nethermind.Trie/BatchedTrieVisitor.cs b/src/Nethermind/Nethermind.Trie/BatchedTrieVisitor.cs index 99bff648354..2cfc57b4556 100644 --- a/src/Nethermind/Nethermind.Trie/BatchedTrieVisitor.cs +++ b/src/Nethermind/Nethermind.Trie/BatchedTrieVisitor.cs @@ -142,12 +142,11 @@ public void Start( try { - // TODO: .Net 9 stackalloc - Task[]? tasks = Enumerable.Range(0, trieVisitContext.MaxDegreeOfParallelism) + using ArrayPoolList tasks = Enumerable.Range(0, trieVisitContext.MaxDegreeOfParallelism) .Select(_ => Task.Run(BatchedThread)) - .ToArray(); + .ToPooledList(trieVisitContext.MaxDegreeOfParallelism); - Task.WaitAll(tasks); + Task.WaitAll(tasks.AsSpan()); } catch (Exception) { diff --git a/src/Nethermind/Nethermind.Trie/PatriciaTree.cs b/src/Nethermind/Nethermind.Trie/PatriciaTree.cs index 6bdc280969c..822afe17cd9 100644 --- a/src/Nethermind/Nethermind.Trie/PatriciaTree.cs +++ b/src/Nethermind/Nethermind.Trie/PatriciaTree.cs @@ -197,7 +197,6 @@ Task CreateTaskForPath(TreePath childPath, TrieNode childNode, int idx) => Task. committer.ReturnConcurrencyQuota(); }); - // TODO: .Net 9 stackalloc ArrayPoolList? childTasks = null; for (int i = 0; i < 16; i++) @@ -230,7 +229,7 @@ Task CreateTaskForPath(TreePath childPath, TrieNode childNode, int idx) => Task. if (childTasks is not null) { - Task.WaitAll(childTasks.ToArray()); + Task.WaitAll(childTasks.AsSpan()); childTasks.Dispose(); } } diff --git a/src/Nethermind/Nethermind.Trie/Pruning/TrieStore.cs b/src/Nethermind/Nethermind.Trie/Pruning/TrieStore.cs index 33c5da9e863..cdccc13d808 100644 --- a/src/Nethermind/Nethermind.Trie/Pruning/TrieStore.cs +++ b/src/Nethermind/Nethermind.Trie/Pruning/TrieStore.cs @@ -3,6 +3,7 @@ using System; using System.Collections.Concurrent; +using System.Collections.Generic; using System.Diagnostics; using System.Diagnostics.CodeAnalysis; using System.Linq; @@ -802,11 +803,11 @@ void TopLevelPersist(TrieNode tn, Hash256? address2, TreePath path) }); } - Task.WaitAll(parallelStartNodes.Select(entry => Task.Run(() => - { - (TrieNode trieNode, Hash256? address2, TreePath path2) = entry; - PersistNodeStartingFrom(trieNode, address2, path2, persistedNodeRecorder, writeFlags, disposeQueue); - }))); + using ArrayPoolList persistNodeStartingFromTasks = parallelStartNodes.Select( + entry => Task.Run(() => PersistNodeStartingFrom(entry.trieNode, entry.address2, entry.path, persistedNodeRecorder, writeFlags, disposeQueue))) + .ToPooledList(parallelStartNodes.Count); + + Task.WaitAll(persistNodeStartingFromTasks.AsSpan()); disposeQueue.CompleteAdding(); Task.WaitAll(_disposeTasks);