From dc78bd21af1c11556448c4905c23a86d9e8650b1 Mon Sep 17 00:00:00 2001 From: Curtis Wensley Date: Tue, 26 May 2020 23:44:19 -0700 Subject: [PATCH] Wpf: Fix huge performance penalty using Graphics on many Bitmaps --- src/Eto.Wpf/WpfExtensions.cs | 6 ++++- test/Eto.Test.Wpf/UnitTests/BitmapTests.cs | 31 ++++++++++++++++++++++ 2 files changed, 36 insertions(+), 1 deletion(-) create mode 100755 test/Eto.Test.Wpf/UnitTests/BitmapTests.cs diff --git a/src/Eto.Wpf/WpfExtensions.cs b/src/Eto.Wpf/WpfExtensions.cs index 419c51d078..1d2193314e 100755 --- a/src/Eto.Wpf/WpfExtensions.cs +++ b/src/Eto.Wpf/WpfExtensions.cs @@ -330,7 +330,11 @@ public static void RenderWithCollect(this swm.Imaging.RenderTargetBitmap bitmap, // fix memory leak with RenderTargetBitmap. See http://stackoverflow.com/questions/14786490/wpf-memory-leak-using-rendertargetbitmap // Reproducible with the // GC.Collect alone seems to fix the issue. Adding GC.WaitForPendingFinalizers may impact performance. - GC.Collect(); + + // Note: this may no longer be an issue with the latest version of .NET after some testing, however + // we keep it here to keep memory in check, but instead make it non-blocking so it doesn't cause a huge + // performance penalty creating many Bitmaps with a Graphics object. + GC.Collect(GC.MaxGeneration, GCCollectionMode.Forced, false); } diff --git a/test/Eto.Test.Wpf/UnitTests/BitmapTests.cs b/test/Eto.Test.Wpf/UnitTests/BitmapTests.cs new file mode 100755 index 0000000000..1f40801e61 --- /dev/null +++ b/test/Eto.Test.Wpf/UnitTests/BitmapTests.cs @@ -0,0 +1,31 @@ +using System; +using System.Diagnostics; +using Eto.Drawing; +using Eto.Test.UnitTests; +using NUnit.Framework; + +namespace Eto.Test.Wpf.UnitTests +{ + [TestFixture] + public class BitmapTests : TestBase + { + [Test, Timeout(1000)] + public void CreatingManySmallBitmapsShouldBeFast() + { + var sw = new Stopwatch(); + sw.Start(); + for (int i = 0; i < 100; i++) + { + var bmp = new Bitmap(20, 20, PixelFormat.Format32bppRgba); + + using (var g = new Graphics(bmp)) + { + g.Clear(Colors.Blue); + } + } + sw.Stop(); + Console.WriteLine($"Total time: {sw.Elapsed}"); + } + + } +} \ No newline at end of file