//
// Copyright (c) James Jackson-South and contributors.
// Licensed under the Apache License, Version 2.0.
//
using System.Buffers;
namespace SixLabors.ImageSharp.Benchmarks
{
using System;
using BenchmarkDotNet.Attributes;
using SixLabors.ImageSharp.PixelFormats;
using CoreSize = SixLabors.Primitives.Size;
using System.Numerics;
using SixLabors.Memory;
using SixLabors.ImageSharp.PixelFormats.PixelBlenders;
public class PorterDuffBulkVsPixel : BenchmarkBase
{
private void BulkVectorConvert(Span destination, Span background, Span source, Span amount)
where TPixel : struct, IPixel
{
Guard.MustBeGreaterThanOrEqualTo(background.Length, destination.Length, nameof(background.Length));
Guard.MustBeGreaterThanOrEqualTo(source.Length, destination.Length, nameof(source.Length));
Guard.MustBeGreaterThanOrEqualTo(amount.Length, destination.Length, nameof(amount.Length));
using (IMemoryOwner buffer = Configuration.Default.MemoryAllocator.Allocate(destination.Length * 3))
{
Span destinationSpan = buffer.Slice(0, destination.Length);
Span backgroundSpan = buffer.Slice(destination.Length, destination.Length);
Span sourceSpan = buffer.Slice(destination.Length * 2, destination.Length);
PixelOperations.Instance.ToVector4(background, backgroundSpan, destination.Length);
PixelOperations.Instance.ToVector4(source, sourceSpan, destination.Length);
for (int i = 0; i < destination.Length; i++)
{
destinationSpan[i] = PorterDuffFunctions.Normal(backgroundSpan[i], sourceSpan[i], amount[i]);
}
PixelOperations.Instance.PackFromVector4(destinationSpan, destination, destination.Length);
}
}
private void BulkPixelConvert(Span destination, Span background, Span source, Span amount)
where TPixel : struct, IPixel
{
Guard.MustBeGreaterThanOrEqualTo(destination.Length, background.Length, nameof(destination));
Guard.MustBeGreaterThanOrEqualTo(source.Length, background.Length, nameof(destination));
Guard.MustBeGreaterThanOrEqualTo(amount.Length, background.Length, nameof(destination));
for (int i = 0; i < destination.Length; i++)
{
destination[i] = PorterDuffFunctions.Normal(destination[i], source[i], amount[i]);
}
}
[Benchmark(Description = "ImageSharp BulkVectorConvert")]
public CoreSize BulkVectorConvert()
{
using (var image = new Image(800, 800))
{
using (IMemoryOwner amounts = Configuration.Default.MemoryAllocator.Allocate(image.Width))
{
amounts.GetSpan().Fill(1);
Buffer2D pixels = image.GetRootFramePixelBuffer();
for (int y = 0; y < image.Height; y++)
{
Span span = pixels.GetRowSpan(y);
this.BulkVectorConvert(span, span, span, amounts.GetSpan());
}
return new CoreSize(image.Width, image.Height);
}
}
}
[Benchmark(Description = "ImageSharp BulkPixelConvert")]
public CoreSize BulkPixelConvert()
{
using (var image = new Image(800, 800))
{
using (IMemoryOwner amounts = Configuration.Default.MemoryAllocator.Allocate(image.Width))
{
amounts.GetSpan().Fill(1);
Buffer2D pixels = image.GetRootFramePixelBuffer();
for (int y = 0; y < image.Height; y++)
{
Span span = pixels.GetRowSpan(y);
this.BulkPixelConvert(span, span, span, amounts.GetSpan());
}
return new CoreSize(image.Width, image.Height);
}
}
}
}
}