diff --git a/src/SixLabors.Fonts/Buffer{T}.cs b/src/SixLabors.Fonts/Buffer{T}.cs index 5b8ec3f1..d3bd0437 100644 --- a/src/SixLabors.Fonts/Buffer{T}.cs +++ b/src/SixLabors.Fonts/Buffer{T}.cs @@ -26,7 +26,7 @@ public Buffer(int length) this.buffer = ArrayPool.Shared.Rent(bufferSizeInBytes); this.length = length; - ByteMemoryManager manager = new(this.buffer); + using ByteMemoryManager manager = new(this.buffer); this.Memory = manager.Memory.Slice(0, this.length); this.span = this.Memory.Span; diff --git a/src/SixLabors.Fonts/FileFontMetrics.cs b/src/SixLabors.Fonts/FileFontMetrics.cs index d316c5e8..eadd93ca 100644 --- a/src/SixLabors.Fonts/FileFontMetrics.cs +++ b/src/SixLabors.Fonts/FileFontMetrics.cs @@ -164,7 +164,7 @@ public static FileFontMetrics[] LoadFontCollection(string path) { using FileStream fs = File.OpenRead(path); long startPos = fs.Position; - BigEndianBinaryReader reader = new(fs, true); + using BigEndianBinaryReader reader = new(fs, true); TtcHeader ttcHeader = TtcHeader.Read(reader); FileFontMetrics[] fonts = new FileFontMetrics[(int)ttcHeader.NumFonts]; diff --git a/src/SixLabors.Fonts/FontCollection.cs b/src/SixLabors.Fonts/FontCollection.cs index e3815e43..437d430d 100644 --- a/src/SixLabors.Fonts/FontCollection.cs +++ b/src/SixLabors.Fonts/FontCollection.cs @@ -234,7 +234,7 @@ private IEnumerable AddCollectionImpl( out IEnumerable descriptions) { long startPos = stream.Position; - BigEndianBinaryReader reader = new(stream, true); + using BigEndianBinaryReader reader = new(stream, true); TtcHeader ttcHeader = TtcHeader.Read(reader); List result = new((int)ttcHeader.NumFonts); HashSet installedFamilies = new(); diff --git a/src/SixLabors.Fonts/FontDescription.cs b/src/SixLabors.Fonts/FontDescription.cs index 27d38777..ce20c981 100644 --- a/src/SixLabors.Fonts/FontDescription.cs +++ b/src/SixLabors.Fonts/FontDescription.cs @@ -92,7 +92,7 @@ public static FontDescription LoadDescription(string path) Guard.NotNullOrWhiteSpace(path, nameof(path)); using FileStream fs = File.OpenRead(path); - var reader = new FontReader(fs); + using var reader = new FontReader(fs); return LoadDescription(reader); } @@ -106,7 +106,7 @@ public static FontDescription LoadDescription(Stream stream) Guard.NotNull(stream, nameof(stream)); // Only read the name tables. - var reader = new FontReader(stream); + using var reader = new FontReader(stream); return LoadDescription(reader); } @@ -152,14 +152,14 @@ public static FontDescription[] LoadFontCollectionDescriptions(string path) public static FontDescription[] LoadFontCollectionDescriptions(Stream stream) { long startPos = stream.Position; - var reader = new BigEndianBinaryReader(stream, true); + using var reader = new BigEndianBinaryReader(stream, true); var ttcHeader = TtcHeader.Read(reader); var result = new FontDescription[(int)ttcHeader.NumFonts]; for (int i = 0; i < ttcHeader.NumFonts; ++i) { stream.Position = startPos + ttcHeader.OffsetTable[i]; - var fontReader = new FontReader(stream); + using var fontReader = new FontReader(stream); result[i] = LoadDescription(fontReader); } diff --git a/src/SixLabors.Fonts/FontReader.cs b/src/SixLabors.Fonts/FontReader.cs index 37efd0e0..2c27ef40 100644 --- a/src/SixLabors.Fonts/FontReader.cs +++ b/src/SixLabors.Fonts/FontReader.cs @@ -9,21 +9,23 @@ namespace SixLabors.Fonts; -internal sealed class FontReader +internal sealed class FontReader : IDisposable { private readonly Stream stream; private readonly Dictionary loadedTables = new(); private readonly TableLoader loader; + private readonly bool isOwnedStream; + private bool isDisposed; + internal FontReader(Stream stream, TableLoader loader) { this.loader = loader; Func loadHeader = TableHeader.Read; - long startOfFilePosition = stream.Position; this.stream = stream; - var reader = new BigEndianBinaryReader(stream, true); + using var reader = new BigEndianBinaryReader(stream, true); // we should immediately read the table header to learn which tables we have and what order they are in uint version = reader.ReadUInt32(); @@ -85,13 +87,14 @@ internal FontReader(Stream stream, TableLoader loader) this.CompressedTableData = true; this.Headers = Woff2Utils.ReadWoff2Headers(reader, tableCount); + this.isOwnedStream = true; + byte[] compressedBuffer = reader.ReadBytes((int)totalCompressedSize); var decompressedStream = new MemoryStream(); using var input = new MemoryStream(compressedBuffer); using var decompressor = new BrotliStream(input, CompressionMode.Decompress); decompressor.CopyTo(decompressedStream); decompressedStream.Position = 0; - this.stream.Dispose(); this.stream = decompressedStream; return; } @@ -197,4 +200,18 @@ public bool TryGetReaderAtTablePosition(string tableName, [NotNullWhen(returnVal reader = header?.CreateReader(this.stream); return reader != null; } + + public void Dispose() + { + if (this.isDisposed) + { + return; + } + + if (this.isOwnedStream) + { + this.stream.Dispose(); + this.isDisposed = true; + } + } } diff --git a/src/SixLabors.Fonts/Native/MacSystemFontsEnumerator.cs b/src/SixLabors.Fonts/Native/MacSystemFontsEnumerator.cs index c59f9568..9f28aff6 100644 --- a/src/SixLabors.Fonts/Native/MacSystemFontsEnumerator.cs +++ b/src/SixLabors.Fonts/Native/MacSystemFontsEnumerator.cs @@ -18,7 +18,7 @@ namespace SixLabors.Fonts.Native; /// Internally, it calls the native CoreText's method to retrieve /// the list of fonts so using this class must be guarded by RuntimeInformation.IsOSPlatform(OSPlatform.OSX). /// -internal class MacSystemFontsEnumerator : IEnumerable, IEnumerator +internal sealed class MacSystemFontsEnumerator : IEnumerable, IEnumerator { private static readonly ArrayPool BytePool = ArrayPool.Shared; diff --git a/src/SixLabors.Fonts/StreamFontMetrics.cs b/src/SixLabors.Fonts/StreamFontMetrics.cs index 4b8ca433..afb1b9f1 100644 --- a/src/SixLabors.Fonts/StreamFontMetrics.cs +++ b/src/SixLabors.Fonts/StreamFontMetrics.cs @@ -333,7 +333,7 @@ internal override void UpdatePositions(GlyphPositioningCollection collection) public static StreamFontMetrics LoadFont(string path) { using FileStream fs = File.OpenRead(path); - var reader = new FontReader(fs); + using var reader = new FontReader(fs); return LoadFont(reader); } @@ -357,7 +357,7 @@ public static StreamFontMetrics LoadFont(string path, long offset) /// a . public static StreamFontMetrics LoadFont(Stream stream) { - var reader = new FontReader(stream); + using var reader = new FontReader(stream); return LoadFont(reader); } diff --git a/tests/SixLabors.Fonts.Tests/FontReaderTests.cs b/tests/SixLabors.Fonts.Tests/FontReaderTests.cs index 5f256e70..2c098327 100644 --- a/tests/SixLabors.Fonts.Tests/FontReaderTests.cs +++ b/tests/SixLabors.Fonts.Tests/FontReaderTests.cs @@ -15,7 +15,7 @@ public void ReadTrueTypeOutlineType() var writer = new BigEndianBinaryWriter(); writer.WriteTrueTypeFileHeader(0, 0, 0, 0); - var reader = new FontReader(writer.GetStream()); + using var reader = new FontReader(writer.GetStream()); Assert.Equal(OutlineType.TrueType, reader.OutlineType); } @@ -25,7 +25,7 @@ public void ReadCffOutlineType() var writer = new BigEndianBinaryWriter(); writer.WriteCffFileHeader(0, 0, 0, 0); - var reader = new FontReader(writer.GetStream()); + using var reader = new FontReader(writer.GetStream()); Assert.Equal(OutlineType.CFF, reader.OutlineType); } @@ -37,7 +37,7 @@ public void ReadTableHeaders() writer.WriteTableHeader("name", 0, 10, 0); writer.WriteTableHeader("cmap", 0, 1, 0); - var reader = new FontReader(writer.GetStream()); + using var reader = new FontReader(writer.GetStream()); Assert.Equal(2, reader.Headers.Count); } @@ -59,7 +59,7 @@ public void ReadCMapTable() new byte[] { 2, 9 }) }); - var reader = new FontReader(writer.GetStream()); + using var reader = new FontReader(writer.GetStream()); CMapTable cmap = reader.GetTable(); Assert.NotNull(cmap); } @@ -67,8 +67,8 @@ public void ReadCMapTable() [Fact] public void ReadFont_WithWoffFormat_EqualsTtf() { - var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); - var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff1Data()); + using var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); + using var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff1Data()); Assert.Equal(fontReaderTtf.Headers.Count, fontReaderWoff.Headers.Count); foreach (string key in fontReaderTtf.Headers.Keys) @@ -80,9 +80,9 @@ public void ReadFont_WithWoffFormat_EqualsTtf() [Fact] public void GlyphsCount_WithWoffFormat_EqualsTtf() { - var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff1Data()); + using var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff1Data()); GlyphTable glyphsWoff = fontReaderWoff.GetTable(); - var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); + using var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); GlyphTable glyphsTtf = fontReaderTtf.GetTable(); Assert.Equal(glyphsTtf.GlyphCount, glyphsWoff.GlyphCount); @@ -91,8 +91,8 @@ public void GlyphsCount_WithWoffFormat_EqualsTtf() [Fact] public void ReadFont_WithWoff2Format_EqualsTtf() { - var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); - var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff2Data()); + using var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); + using var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff2Data()); Assert.Equal(fontReaderTtf.Headers.Count, fontReaderWoff.Headers.Count); foreach (string key in fontReaderTtf.Headers.Keys) @@ -104,9 +104,9 @@ public void ReadFont_WithWoff2Format_EqualsTtf() [Fact] public void GlyphsCount_WithWoff2Format_EqualsTtf() { - var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff2Data()); + using var fontReaderWoff = new FontReader(TestFonts.OpensSansWoff2Data()); GlyphTable glyphsWoff = fontReaderWoff.GetTable(); - var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); + using var fontReaderTtf = new FontReader(TestFonts.OpenSansTtfData()); GlyphTable glyphsTtf = fontReaderTtf.GetTable(); Assert.Equal(glyphsTtf.GlyphCount, glyphsWoff.GlyphCount); diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/CMapTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/CMapTableTests.cs index 1d6acc79..b7e351d6 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/CMapTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/CMapTableTests.cs @@ -35,7 +35,11 @@ public void ShouldThrowExceptionWhenTableCouldNotBeFound() using (System.IO.MemoryStream stream = writer.GetStream()) { - InvalidFontTableException exception = Assert.Throws(() => CMapTable.Load(new FontReader(stream))); + InvalidFontTableException exception = Assert.Throws(() => + { + using var reader = new FontReader(stream); + CMapTable.Load(reader); + }); Assert.Equal("cmap", exception.Table); } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/ColrTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/ColrTableTests.cs index b72299f9..7f60b602 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/ColrTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/ColrTableTests.cs @@ -14,9 +14,10 @@ public void ShouldReturnNullWhenTableCouldNotBeFound() var writer = new BigEndianBinaryWriter(); writer.WriteTrueTypeFileHeader(); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - Assert.Null(ColrTable.Load(new FontReader(stream))); + using var reader = new FontReader(stream); + Assert.Null(ColrTable.Load(reader)); } } @@ -47,12 +48,12 @@ public void ShouldReturnTableValues() } }); - using (System.IO.Stream stream = TestFonts.TwemojiMozillaData()) + using (Stream stream = TestFonts.TwemojiMozillaData()) { - var reader = new FontReader(stream); + using var reader = new FontReader(stream); ColrTable tbl = reader.GetTable(); - System.Span layers = tbl.GetLayers(15); + Span layers = tbl.GetLayers(15); Assert.Equal(2, layers.Length); } } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/HorizontalHeadTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/HorizontalHeadTableTests.cs index 31940af0..95d55403 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/HorizontalHeadTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/HorizontalHeadTableTests.cs @@ -35,9 +35,10 @@ public void ShouldReturnNullWhenTableCouldNotBeFound() var writer = new BigEndianBinaryWriter(); writer.WriteTrueTypeFileHeader(); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - Assert.Null(HorizontalHeadTable.Load(new FontReader(stream))); + using var reader = new FontReader(stream); + Assert.Null(HorizontalHeadTable.Load(reader)); } } } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/IndexLocationTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/IndexLocationTableTests.cs index ce11426c..1052cc05 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/IndexLocationTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/IndexLocationTableTests.cs @@ -15,10 +15,13 @@ public void ShouldThrowExceptionWhenHeadTableCouldNotBeFound() var writer = new BigEndianBinaryWriter(); writer.WriteTrueTypeFileHeader(); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - MissingFontTableException exception = Assert.Throws( - () => IndexLocationTable.Load(new FontReader(stream))); + MissingFontTableException exception = Assert.Throws(() => + { + using var reader = new FontReader(stream); + IndexLocationTable.Load(reader); + }); Assert.Equal("head", exception.Table); } @@ -40,10 +43,13 @@ public void ShouldThrowExceptionWhenMaximumProfileTableCouldNotBeFound() 0, HeadTable.IndexLocationFormats.Offset16)); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - InvalidFontTableException exception = Assert.Throws( - () => IndexLocationTable.Load(new FontReader(stream))); + InvalidFontTableException exception = Assert.Throws(() => + { + using var reader = new FontReader(stream); + IndexLocationTable.Load(reader); + }); Assert.Equal("maxp", exception.Table); } @@ -65,9 +71,10 @@ public void ShouldReturnNullWhenTableCouldNotBeFound() 0, HeadTable.IndexLocationFormats.Offset16)); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - Assert.Null(IndexLocationTable.Load(new FontReader(stream))); + using var reader = new FontReader(stream); + Assert.Null(IndexLocationTable.Load(reader)); } } } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/KerningTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/KerningTableTests.cs index 8f0eae73..b0183591 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/KerningTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/KerningTableTests.cs @@ -1,4 +1,4 @@ -// Copyright (c) Six Labors. +// Copyright (c) Six Labors. // Licensed under the Six Labors Split License. using SixLabors.Fonts.Tables.General.Kern; @@ -13,9 +13,10 @@ public void ShouldReturnDefaultValueWhenTableCouldNotBeFound() var writer = new BigEndianBinaryWriter(); writer.WriteTrueTypeFileHeader(); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - var table = KerningTable.Load(new FontReader(stream)); + using var reader = new FontReader(stream); + var table = KerningTable.Load(reader); Assert.NotNull(table); } } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/MaximumProfileTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/MaximumProfileTableTests.cs index 6e260136..54f123d6 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/MaximumProfileTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/MaximumProfileTableTests.cs @@ -15,8 +15,11 @@ public void ShouldThrowExceptionWhenTableCouldNotBeFound() using (MemoryStream stream = writer.GetStream()) { - InvalidFontTableException exception = Assert.Throws( - () => MaximumProfileTable.Load(new FontReader(stream))); + InvalidFontTableException exception = Assert.Throws(() => + { + using var reader = new FontReader(stream); + MaximumProfileTable.Load(reader); + }); Assert.Equal("maxp", exception.Table); } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/NameTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/NameTableTests.cs index 12ae2dc1..25cec2c7 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/NameTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/NameTableTests.cs @@ -76,10 +76,13 @@ public void ShouldThrowExceptionWhenTableCouldNotBeFound() var writer = new BigEndianBinaryWriter(); writer.WriteTrueTypeFileHeader(); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - InvalidFontTableException exception = Assert.Throws( - () => NameTable.Load(new FontReader(stream))); + InvalidFontTableException exception = Assert.Throws(() => + { + using var reader = new FontReader(stream); + NameTable.Load(reader); + }); Assert.Equal("name", exception.Table); } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/Os2TableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/Os2TableTests.cs index ed6dde0b..cfa4a2f0 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/Os2TableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/Os2TableTests.cs @@ -1,4 +1,4 @@ -// Copyright (c) Six Labors. +// Copyright (c) Six Labors. // Licensed under the Six Labors Split License. using SixLabors.Fonts.Tables.General; @@ -13,9 +13,10 @@ public void ShouldReturnNullWhenTableCouldNotBeFound() var writer = new BigEndianBinaryWriter(); writer.WriteTrueTypeFileHeader(); - using (System.IO.MemoryStream stream = writer.GetStream()) + using (MemoryStream stream = writer.GetStream()) { - Assert.Null(OS2Table.Load(new FontReader(stream))); + using var reader = new FontReader(stream); + Assert.Null(OS2Table.Load(reader)); } } } diff --git a/tests/SixLabors.Fonts.Tests/Tables/General/VerticalHeadTableTests.cs b/tests/SixLabors.Fonts.Tests/Tables/General/VerticalHeadTableTests.cs index b73683c3..5ea44e76 100644 --- a/tests/SixLabors.Fonts.Tests/Tables/General/VerticalHeadTableTests.cs +++ b/tests/SixLabors.Fonts.Tests/Tables/General/VerticalHeadTableTests.cs @@ -36,6 +36,7 @@ public void ShouldReturnNullWhenTableCouldNotBeFound() writer.WriteTrueTypeFileHeader(); using MemoryStream stream = writer.GetStream(); - Assert.Null(VerticalHeadTable.Load(new FontReader(stream))); + using var reader = new FontReader(stream); + Assert.Null(VerticalHeadTable.Load(reader)); } }