Skip to content

Commit

Permalink
Merge pull request #392 from drewnoakes/pool-buffers
Browse files Browse the repository at this point in the history
Use ArrayPool for poolable byte buffers
  • Loading branch information
drewnoakes authored Feb 1, 2024
2 parents 877ee15 + f7622ba commit af0aa35
Show file tree
Hide file tree
Showing 9 changed files with 179 additions and 92 deletions.
6 changes: 5 additions & 1 deletion MetadataExtractor/Formats/Apple/BplistReader.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
// Copyright (c) Drew Noakes and contributors. All Rights Reserved. Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.

using System.Buffers;

namespace MetadataExtractor.Formats.Apple;

/// <summary>
Expand Down Expand Up @@ -120,7 +122,7 @@ object HandleInt(byte marker)

Dictionary<byte, byte> HandleDict(byte count)
{
var keyRefs = new byte[count];
var keyRefs = ArrayPool<byte>.Shared.Rent(count);

for (int j = 0; j < count; j++)
{
Expand All @@ -134,6 +136,8 @@ Dictionary<byte, byte> HandleDict(byte count)
map.Add(keyRefs[j], reader.GetByte());
}

ArrayPool<byte>.Shared.Return(keyRefs);

return map;
}

Expand Down
30 changes: 20 additions & 10 deletions MetadataExtractor/Formats/Flir/FlirReader.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// Copyright (c) Drew Noakes and contributors. All Rights Reserved. Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.

using System.Buffers;
using MetadataExtractor.Formats.Jpeg;
using static MetadataExtractor.Formats.Flir.FlirCameraInfoDirectory;

Expand Down Expand Up @@ -35,26 +36,35 @@ public IEnumerable<Directory> ReadJpegSegments(IEnumerable<JpegSegment> segments
if (length == 0)
return [];

var buffer = new byte[length];
using var merged = new MemoryStream(buffer);
byte[] buffer = ArrayPool<byte>.Shared.Rent(length);

foreach (var segment in segments)
try
{
// Skip segments not starting with the required preamble
if (segment.Span.StartsWith(preamble))
using var merged = new MemoryStream(buffer);

foreach (var segment in segments)
{
merged.Write(segment.Bytes, preambleLength, segment.Bytes.Length - preambleLength);
// Skip segments not starting with the required preamble
if (segment.Span.StartsWith(preamble))
{
merged.Write(segment.Bytes, preambleLength, segment.Bytes.Length - preambleLength);
}
}
}

return Extract(new ByteArrayReader(buffer));
return Extract(new ByteArrayReader(buffer));
}
finally
{
ArrayPool<byte>.Shared.Return(buffer);
}
}

public IEnumerable<Directory> Extract(IndexedReader reader)
{
var header = reader.GetUInt32(0);
Span<byte> header = stackalloc byte[4];
reader.GetBytes(0, header);

if (header != 0x46464600)
if (!header.SequenceEqual("FFF\0"u8))
{
var flirHeaderDirectory = new FlirHeaderDirectory();
flirHeaderDirectory.AddError("Unexpected FFF header bytes.");
Expand Down
120 changes: 70 additions & 50 deletions MetadataExtractor/Formats/Gif/GifReader.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// Copyright (c) Drew Noakes and contributors. All Rights Reserved. Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.

using System.Buffers;
using MetadataExtractor.Formats.Icc;
using MetadataExtractor.Formats.Xmp;

Expand Down Expand Up @@ -234,45 +235,42 @@ private static GifCommentDirectory ReadCommentBlock(SequentialReader reader, byt
if (blockSizeBytes != 11)
return new ErrorDirectory($"Invalid GIF application extension block size. Expected 11, got {blockSizeBytes}.");

var extensionType = reader.GetString(blockSizeBytes, Encoding.UTF8);
Span<byte> extensionType = stackalloc byte[11];

switch (extensionType)
reader.GetBytes(extensionType);

if (extensionType.SequenceEqual("XMP DataXMP"u8))
{
case "XMP DataXMP":
{
// XMP data extension
var xmpBytes = GatherBytes(reader);
int xmpLength = xmpBytes.Length - 257; // Exclude the "magic trailer", see XMP Specification Part 3, 1.1.2 GIF
// Only extract valid blocks
return xmpLength > 0
? new XmpReader().Extract(xmpBytes, 0, xmpBytes.Length - 257)
: null;
}
case "ICCRGBG1012":
{
// ICC profile extension
var iccBytes = GatherBytes(reader, reader.GetByte());
return iccBytes.Length != 0
? new IccReader().Extract(new ByteArrayReader(iccBytes))
: null;
}
case "NETSCAPE2.0":
{
reader.Skip(2);
// Netscape's animated GIF extension
// Iteration count (0 means infinite)
var iterationCount = reader.GetUInt16();
// Skip terminator
reader.Skip(1);
var animationDirectory = new GifAnimationDirectory();
animationDirectory.Set(GifAnimationDirectory.TagIterationCount, iterationCount);
return animationDirectory;
}
default:
{
SkipBlocks(reader);
return null;
}
// XMP data extension
var xmpBytes = GatherXmpBytes(reader);
return xmpBytes is not null
? new XmpReader().Extract(xmpBytes)
: null;
}
else if (extensionType.SequenceEqual("ICCRGBG1012"u8))
{
// ICC profile extension
var iccBytes = GatherBytes(reader, reader.GetByte());
return iccBytes.Length != 0
? new IccReader().Extract(new ByteArrayReader(iccBytes))
: null;
}
else if (extensionType.SequenceEqual("NETSCAPE2.0"u8))
{
reader.Skip(2);
// Netscape's animated GIF extension
// Iteration count (0 means infinite)
var iterationCount = reader.GetUInt16();
// Skip terminator
reader.Skip(1);
var animationDirectory = new GifAnimationDirectory();
animationDirectory.Set(GifAnimationDirectory.TagIterationCount, iterationCount);
return animationDirectory;
}
else
{
SkipBlocks(reader);
return null;
}
}

Expand Down Expand Up @@ -329,36 +327,58 @@ private static GifImageDirectory ReadImageBlock(SequentialReader reader)

#region Utility methods

private static byte[] GatherBytes(SequentialReader reader)
private static byte[]? GatherXmpBytes(SequentialReader reader)
{
var bytes = new MemoryStream();
var buffer = new byte[257];
// GatherXmpBytes differs from GatherBytes in that this method includes the "length"
// bytes in its output.

var stream = new MemoryStream();
var buffer = ArrayPool<byte>.Shared.Rent(byte.MaxValue + 1);

while (true)
{
var b = reader.GetByte();
if (b == 0)
return bytes.ToArray();
buffer[0] = b;
reader.GetBytes(buffer, 1, b);
bytes.Write(buffer, 0, b + 1);
var len = reader.GetByte();
if (len == 0)
break;
buffer[0] = len;
reader.GetBytes(buffer, offset: 1, count: len);
stream.Write(buffer, 0, len + 1);
}

ArrayPool<byte>.Shared.Return(buffer);

// Exclude the "magic trailer", see XMP Specification Part 3, 1.1.2 GIF
int xmpLength = checked((int)stream.Length) - 257;

if (xmpLength <= 0)
{
return null;
}

stream.SetLength(xmpLength);

return stream.ToArray();
}

private static byte[] GatherBytes(SequentialReader reader, int firstLength)
private static byte[] GatherBytes(SequentialReader reader, byte firstLength)
{
var buffer = new MemoryStream();
var stream = new MemoryStream();
var buffer = ArrayPool<byte>.Shared.Rent(byte.MaxValue);

var length = firstLength;

while (length > 0)
{
buffer.Write(reader.GetBytes(length), 0, length);
reader.GetBytes(buffer.AsSpan().Slice(0, length));

stream.Write(buffer, 0, length);

length = reader.GetByte();
}

return buffer.ToArray();
ArrayPool<byte>.Shared.Return(buffer);

return stream.ToArray();
}

private static void SkipBlocks(SequentialReader reader)
Expand Down
11 changes: 8 additions & 3 deletions MetadataExtractor/Formats/Icc/IccReader.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// Copyright (c) Drew Noakes and contributors. All Rights Reserved. Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.

using System.Buffers;
using MetadataExtractor.Formats.Jpeg;

namespace MetadataExtractor.Formats.Icc
Expand Down Expand Up @@ -37,14 +38,14 @@ public IEnumerable<Directory> ReadJpegSegments(IEnumerable<JpegSegment> segments
byte[] buffer;
if (iccSegments.Count == 1)
{
buffer = new byte[iccSegments[0].Bytes.Length - JpegSegmentPreambleLength];
buffer = ArrayPool<byte>.Shared.Rent(iccSegments[0].Bytes.Length - JpegSegmentPreambleLength);
Array.Copy(iccSegments[0].Bytes, JpegSegmentPreambleLength, buffer, 0, iccSegments[0].Bytes.Length - JpegSegmentPreambleLength);
}
else
{
// Concatenate all buffers
var totalLength = iccSegments.Sum(s => s.Bytes.Length - JpegSegmentPreambleLength);
buffer = new byte[totalLength];
buffer = ArrayPool<byte>.Shared.Rent(totalLength);
for (int i = 0, pos = 0; i < iccSegments.Count; i++)
{
var segment = iccSegments[i];
Expand All @@ -53,7 +54,11 @@ public IEnumerable<Directory> ReadJpegSegments(IEnumerable<JpegSegment> segments
}
}

return new Directory[] { Extract(new ByteArrayReader(buffer)) };
Directory directory = Extract(new ByteArrayReader(buffer));

ArrayPool<byte>.Shared.Return(buffer);

return [directory];
}

public IccDirectory Extract(IndexedReader reader)
Expand Down
6 changes: 5 additions & 1 deletion MetadataExtractor/Formats/Raf/RafMetadataReader.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// Copyright (c) Drew Noakes and contributors. All Rights Reserved. Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.

using System.Buffers;
using MetadataExtractor.Formats.Jpeg;

namespace MetadataExtractor.Formats.Raf
Expand All @@ -14,7 +15,8 @@ public static IReadOnlyList<Directory> ReadMetadata(Stream stream)
if (!stream.CanSeek)
throw new ArgumentException("Must support seek", nameof(stream));

var data = new byte[512];
var data = ArrayPool<byte>.Shared.Rent(512);

var bytesRead = stream.Read(data, 0, 512);

if (bytesRead == 0)
Expand All @@ -32,6 +34,8 @@ public static IReadOnlyList<Directory> ReadMetadata(Stream stream)
}
}

ArrayPool<byte>.Shared.Return(data);

return JpegMetadataReader.ReadMetadata(stream);
}
}
Expand Down
9 changes: 3 additions & 6 deletions MetadataExtractor/Formats/Tga/TgaExtensionReader.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
// Copyright (c) Drew Noakes and contributors. All Rights Reserved. Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.

using System.Buffers;

namespace MetadataExtractor.Formats.Tga
{
/// <summary>Reads TGA image file extension area.</summary>
Expand Down Expand Up @@ -55,12 +57,7 @@ protected override void Populate(Stream stream, int offset, TgaExtensionDirector

string GetString(int length)
{
var buffer = new byte[length];
reader.GetBytes(buffer, 0, length);
int i = 0;
while (i < buffer.Length && buffer[i] != '\0')
++i;
return Encoding.ASCII.GetString(buffer, 0, i).TrimEnd();
return reader.GetNullTerminatedString(length, Encoding.ASCII, moveToMaxLength: true).TrimEnd();
}

bool TryGetDateTime(out DateTime dateTime)
Expand Down
25 changes: 22 additions & 3 deletions MetadataExtractor/IO/IndexedReader.cs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
// Copyright (c) Drew Noakes and contributors. All Rights Reserved. Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.


using System.Buffers;
using System.Buffers.Binary;

namespace MetadataExtractor.IO
Expand Down Expand Up @@ -321,13 +322,31 @@ public string GetString(int index, int bytesRequested, Encoding encoding)
{
// This check is important on .NET Framework
if (bytesRequested is 0)
{
return "";
}
else if (bytesRequested < 256)
{
Span<byte> bytes = stackalloc byte[bytesRequested];

Span<byte> bytes = bytesRequested < 256 ? stackalloc byte[bytesRequested] : new byte[bytesRequested];
GetBytes(index, bytes);

GetBytes(index, bytes);
return encoding.GetString(bytes);
}
else
{
byte[] bytes = ArrayPool<byte>.Shared.Rent(bytesRequested);

Span<byte> span = bytes.AsSpan().Slice(0, bytesRequested);

GetBytes(index, span);

return encoding.GetString(bytes);
var s = encoding.GetString(span);

ArrayPool<byte>.Shared.Return(bytes);

return s;
}
}

/// <summary>
Expand Down
Loading

0 comments on commit af0aa35

Please sign in to comment.