|
|
|
@ -64,15 +64,15 @@ namespace SixLabors.ImageSharp.Processing.Processors.Convolution |
|
|
|
var interest = Rectangle.Intersect(this.SourceRectangle, source.Bounds()); |
|
|
|
|
|
|
|
// Horizontal convolution
|
|
|
|
var horizontalOperation = new RowIntervalOperation(interest, firstPassPixels, source.PixelBuffer, this.KernelX, this.Configuration, this.PreserveAlpha); |
|
|
|
ParallelRowIterator.IterateRowIntervals<RowIntervalOperation, Vector4>( |
|
|
|
var horizontalOperation = new RowOperation(interest, firstPassPixels, source.PixelBuffer, this.KernelX, this.Configuration, this.PreserveAlpha); |
|
|
|
ParallelRowIterator.IterateRows<RowOperation, Vector4>( |
|
|
|
this.Configuration, |
|
|
|
interest, |
|
|
|
in horizontalOperation); |
|
|
|
|
|
|
|
// Vertical convolution
|
|
|
|
var verticalOperation = new RowIntervalOperation(interest, source.PixelBuffer, firstPassPixels, this.KernelY, this.Configuration, this.PreserveAlpha); |
|
|
|
ParallelRowIterator.IterateRowIntervals<RowIntervalOperation, Vector4>( |
|
|
|
var verticalOperation = new RowOperation(interest, source.PixelBuffer, firstPassPixels, this.KernelY, this.Configuration, this.PreserveAlpha); |
|
|
|
ParallelRowIterator.IterateRows<RowOperation, Vector4>( |
|
|
|
this.Configuration, |
|
|
|
interest, |
|
|
|
in verticalOperation); |
|
|
|
@ -81,7 +81,7 @@ namespace SixLabors.ImageSharp.Processing.Processors.Convolution |
|
|
|
/// <summary>
|
|
|
|
/// A <see langword="struct"/> implementing the convolution logic for <see cref="Convolution2PassProcessor{T}"/>.
|
|
|
|
/// </summary>
|
|
|
|
private readonly struct RowIntervalOperation : IRowIntervalOperation<Vector4> |
|
|
|
private readonly struct RowOperation : IRowOperation<Vector4> |
|
|
|
{ |
|
|
|
private readonly Rectangle bounds; |
|
|
|
private readonly Buffer2D<TPixel> targetPixels; |
|
|
|
@ -91,7 +91,7 @@ namespace SixLabors.ImageSharp.Processing.Processors.Convolution |
|
|
|
private readonly bool preserveAlpha; |
|
|
|
|
|
|
|
[MethodImpl(InliningOptions.ShortMethod)] |
|
|
|
public RowIntervalOperation( |
|
|
|
public RowOperation( |
|
|
|
Rectangle bounds, |
|
|
|
Buffer2D<TPixel> targetPixels, |
|
|
|
Buffer2D<TPixel> sourcePixels, |
|
|
|
@ -109,53 +109,50 @@ namespace SixLabors.ImageSharp.Processing.Processors.Convolution |
|
|
|
|
|
|
|
/// <inheritdoc/>
|
|
|
|
[MethodImpl(InliningOptions.ShortMethod)] |
|
|
|
public void Invoke(in RowInterval rows, Span<Vector4> span) |
|
|
|
public void Invoke(int y, Span<Vector4> span) |
|
|
|
{ |
|
|
|
ref Vector4 spanRef = ref MemoryMarshal.GetReference(span); |
|
|
|
|
|
|
|
int maxY = this.bounds.Bottom - 1; |
|
|
|
int maxX = this.bounds.Right - 1; |
|
|
|
|
|
|
|
for (int y = rows.Min; y < rows.Max; y++) |
|
|
|
{ |
|
|
|
Span<TPixel> targetRowSpan = this.targetPixels.GetRowSpan(y).Slice(this.bounds.X); |
|
|
|
PixelOperations<TPixel>.Instance.ToVector4(this.configuration, targetRowSpan.Slice(0, span.Length), span); |
|
|
|
Span<TPixel> targetRowSpan = this.targetPixels.GetRowSpan(y).Slice(this.bounds.X); |
|
|
|
PixelOperations<TPixel>.Instance.ToVector4(this.configuration, targetRowSpan.Slice(0, span.Length), span); |
|
|
|
|
|
|
|
if (this.preserveAlpha) |
|
|
|
if (this.preserveAlpha) |
|
|
|
{ |
|
|
|
for (int x = 0; x < this.bounds.Width; x++) |
|
|
|
{ |
|
|
|
for (int x = 0; x < this.bounds.Width; x++) |
|
|
|
{ |
|
|
|
DenseMatrixUtils.Convolve3( |
|
|
|
in this.kernel, |
|
|
|
this.sourcePixels, |
|
|
|
ref spanRef, |
|
|
|
y, |
|
|
|
x, |
|
|
|
this.bounds.Y, |
|
|
|
maxY, |
|
|
|
this.bounds.X, |
|
|
|
maxX); |
|
|
|
} |
|
|
|
DenseMatrixUtils.Convolve3( |
|
|
|
in this.kernel, |
|
|
|
this.sourcePixels, |
|
|
|
ref spanRef, |
|
|
|
y, |
|
|
|
x, |
|
|
|
this.bounds.Y, |
|
|
|
maxY, |
|
|
|
this.bounds.X, |
|
|
|
maxX); |
|
|
|
} |
|
|
|
else |
|
|
|
} |
|
|
|
else |
|
|
|
{ |
|
|
|
for (int x = 0; x < this.bounds.Width; x++) |
|
|
|
{ |
|
|
|
for (int x = 0; x < this.bounds.Width; x++) |
|
|
|
{ |
|
|
|
DenseMatrixUtils.Convolve4( |
|
|
|
in this.kernel, |
|
|
|
this.sourcePixels, |
|
|
|
ref spanRef, |
|
|
|
y, |
|
|
|
x, |
|
|
|
this.bounds.Y, |
|
|
|
maxY, |
|
|
|
this.bounds.X, |
|
|
|
maxX); |
|
|
|
} |
|
|
|
DenseMatrixUtils.Convolve4( |
|
|
|
in this.kernel, |
|
|
|
this.sourcePixels, |
|
|
|
ref spanRef, |
|
|
|
y, |
|
|
|
x, |
|
|
|
this.bounds.Y, |
|
|
|
maxY, |
|
|
|
this.bounds.X, |
|
|
|
maxX); |
|
|
|
} |
|
|
|
|
|
|
|
PixelOperations<TPixel>.Instance.FromVector4Destructive(this.configuration, span, targetRowSpan); |
|
|
|
} |
|
|
|
|
|
|
|
PixelOperations<TPixel>.Instance.FromVector4Destructive(this.configuration, span, targetRowSpan); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|