mirror of https://github.com/adamdruppe/arsd.git
Refactor component/channel loop of image scaler
This commit is contained in:
parent
539480a2fa
commit
2804f426c4
371
pixmappaint.d
371
pixmappaint.d
|
@ -3092,7 +3092,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
};
|
}
|
||||||
|
|
||||||
auto dst = PixmapScannerRW(target);
|
auto dst = PixmapScannerRW(target);
|
||||||
|
|
||||||
|
@ -3170,57 +3170,105 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
multi,
|
multi,
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (immutable ib, ref c; pxInt.components) {
|
// ======== Interpolate X ========
|
||||||
|
auto sampleX(SamplingMode mode)(const size_t ib) {
|
||||||
|
pragma(inline, true);
|
||||||
|
|
||||||
// ======== Interpolate X ========
|
static if (mode == SamplingMode.multi) {
|
||||||
auto sampleX(SamplingMode mode)() {
|
const nLines = 1 + posSrcY[idxB] - posSrcY[idxT];
|
||||||
pragma(inline, true);
|
|
||||||
|
alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc;
|
||||||
|
ulong foreachLine(scope ForeachLineCallback apply) {
|
||||||
|
ulong linesSum = 0;
|
||||||
|
foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) {
|
||||||
|
const posLine = Point(posSrcX[idxL], lineY);
|
||||||
|
linesSum += apply(posLine);
|
||||||
|
}
|
||||||
|
return linesSum;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ========== None ==========
|
||||||
|
static if (directionX == none) {
|
||||||
|
static if (mode == SamplingMode.single) {
|
||||||
|
return (() @trusted => pxNeighs[idxTL].components.ptr[ib])();
|
||||||
|
}
|
||||||
|
|
||||||
|
static if (mode == SamplingMode.dual) {
|
||||||
|
return () @trusted {
|
||||||
|
ulong[2] result = [
|
||||||
|
pxNeighs[idxTL].components.ptr[ib],
|
||||||
|
pxNeighs[idxBL].components.ptr[ib],
|
||||||
|
];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
}
|
||||||
|
|
||||||
static if (mode == SamplingMode.multi) {
|
static if (mode == SamplingMode.multi) {
|
||||||
const nLines = 1 + posSrcY[idxB] - posSrcY[idxT];
|
const ySum = foreachLine(delegate(const Point posLine) {
|
||||||
|
const pxSrc = source.getPixel(posLine);
|
||||||
|
return ulong((() @trusted => pxSrc.components.ptr[ib])());
|
||||||
|
});
|
||||||
|
return (ySum / nLines);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc;
|
// ========== Down ==========
|
||||||
ulong foreachLine(scope ForeachLineCallback apply) {
|
static if (directionX == down) {
|
||||||
ulong linesSum = 0;
|
static if (mode == SamplingMode.single) {
|
||||||
foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) {
|
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
||||||
const posLine = Point(posSrcX[idxL], lineY);
|
const posSampling = Point(posSrcX[idxL], posSrcY[idxT]);
|
||||||
linesSum += apply(posLine);
|
const samplingOffset = source.scanTo(posSampling);
|
||||||
|
const srcSamples = () @trusted {
|
||||||
|
return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)];
|
||||||
|
}();
|
||||||
|
|
||||||
|
ulong xSum = 0;
|
||||||
|
|
||||||
|
foreach (srcSample; srcSamples) {
|
||||||
|
xSum += (() @trusted => srcSample.components.ptr[ib])();
|
||||||
|
}
|
||||||
|
|
||||||
|
return (xSum / nSamples);
|
||||||
|
}
|
||||||
|
|
||||||
|
static if (mode == SamplingMode.dual) {
|
||||||
|
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
||||||
|
const Point[2] posSampling = [
|
||||||
|
Point(posSrcX[idxL], posSrcY[idxT]),
|
||||||
|
Point(posSrcX[idxL], posSrcY[idxB]),
|
||||||
|
];
|
||||||
|
|
||||||
|
const int[2] samplingOffsets = [
|
||||||
|
source.scanTo(posSampling[0]),
|
||||||
|
source.scanTo(posSampling[1]),
|
||||||
|
];
|
||||||
|
|
||||||
|
const srcSamples2 = () @trusted {
|
||||||
|
const(const(Pixel)[])[2] result = [
|
||||||
|
source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)],
|
||||||
|
source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)],
|
||||||
|
];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
|
||||||
|
ulong[2] xSums = [0, 0];
|
||||||
|
|
||||||
|
foreach (idx, srcSamples; srcSamples2) {
|
||||||
|
foreach (srcSample; srcSamples) {
|
||||||
|
() @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }();
|
||||||
}
|
}
|
||||||
return linesSum;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
xSums[] /= nSamples;
|
||||||
|
return xSums;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ========== None ==========
|
static if (mode == SamplingMode.multi) {
|
||||||
static if (directionX == none) {
|
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
||||||
static if (mode == SamplingMode.single) {
|
|
||||||
return (() @trusted => pxNeighs[idxTL].components.ptr[ib])();
|
|
||||||
}
|
|
||||||
|
|
||||||
static if (mode == SamplingMode.dual) {
|
auto ySum = foreachLine(delegate(const Point posLine) {
|
||||||
return () @trusted {
|
const samplingOffset = source.scanTo(posLine);
|
||||||
ulong[2] result = [
|
|
||||||
pxNeighs[idxTL].components.ptr[ib],
|
|
||||||
pxNeighs[idxBL].components.ptr[ib],
|
|
||||||
];
|
|
||||||
return result;
|
|
||||||
}();
|
|
||||||
}
|
|
||||||
|
|
||||||
static if (mode == SamplingMode.multi) {
|
|
||||||
const ySum = foreachLine(delegate(const Point posLine) {
|
|
||||||
const pxSrc = source.getPixel(posLine);
|
|
||||||
return ulong((() @trusted => pxSrc.components.ptr[ib])());
|
|
||||||
});
|
|
||||||
return (ySum / nLines);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ========== Down ==========
|
|
||||||
static if (directionX == down) {
|
|
||||||
static if (mode == SamplingMode.single) {
|
|
||||||
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
|
||||||
const posSampling = Point(posSrcX[idxL], posSrcY[idxT]);
|
|
||||||
const samplingOffset = source.scanTo(posSampling);
|
|
||||||
const srcSamples = () @trusted {
|
const srcSamples = () @trusted {
|
||||||
return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)];
|
return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)];
|
||||||
}();
|
}();
|
||||||
|
@ -3231,162 +3279,117 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
xSum += (() @trusted => srcSample.components.ptr[ib])();
|
xSum += (() @trusted => srcSample.components.ptr[ib])();
|
||||||
}
|
}
|
||||||
|
|
||||||
return (xSum / nSamples);
|
return xSum;
|
||||||
|
});
|
||||||
|
|
||||||
|
ySum /= nSamples;
|
||||||
|
return (ySum / nLines);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ========== Up ==========
|
||||||
|
static if (directionX == up) {
|
||||||
|
|
||||||
|
if (posSrcX[0] == posSrcX[1]) {
|
||||||
|
static if (mode == SamplingMode.single) {
|
||||||
|
return (() @trusted => pxNeighs[idxTL].components.ptr[ib])();
|
||||||
|
}
|
||||||
|
static if (mode == SamplingMode.dual) {
|
||||||
|
return () @trusted {
|
||||||
|
ulong[2] result = [
|
||||||
|
pxNeighs[idxTL].components.ptr[ib],
|
||||||
|
pxNeighs[idxBL].components.ptr[ib],
|
||||||
|
];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
}
|
||||||
|
static if (mode == SamplingMode.multi) {
|
||||||
|
const ySum = foreachLine(delegate(const Point posLine) {
|
||||||
|
ulong xSum = 0;
|
||||||
|
const samplingOffset = source.scanTo(posLine);
|
||||||
|
return (() @trusted
|
||||||
|
=> source.data.ptr[samplingOffset].components.ptr[ib]
|
||||||
|
)();
|
||||||
|
});
|
||||||
|
return (ySum / nLines);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const ulong[2] weightsX = () {
|
||||||
|
ulong[2] result;
|
||||||
|
result[0] = (udecimalHalf + posSrcX[1] - posSrcCenterX).fractionalDigits;
|
||||||
|
result[1] = ulong(uint.max) + 1 - result[0];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
|
||||||
|
static if (mode == SamplingMode.single) {
|
||||||
|
ulong xSum = 0;
|
||||||
|
|
||||||
|
() @trusted {
|
||||||
|
xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]);
|
||||||
|
xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]);
|
||||||
|
}();
|
||||||
|
|
||||||
|
xSum >>= 32;
|
||||||
|
return xSum;
|
||||||
|
}
|
||||||
|
|
||||||
|
static if (mode == SamplingMode.dual) {
|
||||||
|
ulong[2] xSums = [0, 0];
|
||||||
|
|
||||||
|
() @trusted {
|
||||||
|
xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]);
|
||||||
|
xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]);
|
||||||
|
|
||||||
|
xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]);
|
||||||
|
xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]);
|
||||||
|
}();
|
||||||
|
|
||||||
|
foreach (ref sum; xSums) {
|
||||||
|
sum >>= 32;
|
||||||
}
|
}
|
||||||
|
|
||||||
static if (mode == SamplingMode.dual) {
|
return xSums;
|
||||||
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
}
|
||||||
const Point[2] posSampling = [
|
|
||||||
Point(posSrcX[idxL], posSrcY[idxT]),
|
|
||||||
Point(posSrcX[idxL], posSrcY[idxB]),
|
|
||||||
];
|
|
||||||
|
|
||||||
const int[2] samplingOffsets = [
|
static if (mode == SamplingMode.multi) {
|
||||||
source.scanTo(posSampling[0]),
|
const ySum = foreachLine(delegate(const Point posLine) {
|
||||||
source.scanTo(posSampling[1]),
|
ulong xSum = 0;
|
||||||
];
|
|
||||||
|
|
||||||
const srcSamples2 = () @trusted {
|
const samplingOffset = source.scanTo(posLine);
|
||||||
const(const(Pixel)[])[2] result = [
|
ubyte[2] pxcLR = () @trusted {
|
||||||
source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)],
|
ubyte[2] result = [
|
||||||
source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)],
|
source.data.ptr[samplingOffset].components.ptr[ib],
|
||||||
|
source.data.ptr[samplingOffset + 1].components.ptr[ib],
|
||||||
];
|
];
|
||||||
return result;
|
return result;
|
||||||
}();
|
}();
|
||||||
|
|
||||||
ulong[2] xSums = [0, 0];
|
xSum += (pxcLR[idxL] * weightsX[idxL]);
|
||||||
|
xSum += (pxcLR[idxR] * weightsX[idxR]);
|
||||||
|
|
||||||
foreach (idx, srcSamples; srcSamples2) {
|
return (xSum >> 32);
|
||||||
foreach (srcSample; srcSamples) {
|
});
|
||||||
() @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
xSums[] /= nSamples;
|
return (ySum / nLines);
|
||||||
return xSums;
|
|
||||||
}
|
|
||||||
|
|
||||||
static if (mode == SamplingMode.multi) {
|
|
||||||
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
|
||||||
|
|
||||||
auto ySum = foreachLine(delegate(const Point posLine) {
|
|
||||||
const samplingOffset = source.scanTo(posLine);
|
|
||||||
const srcSamples = () @trusted {
|
|
||||||
return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)];
|
|
||||||
}();
|
|
||||||
|
|
||||||
ulong xSum = 0;
|
|
||||||
|
|
||||||
foreach (srcSample; srcSamples) {
|
|
||||||
xSum += (() @trusted => srcSample.components.ptr[ib])();
|
|
||||||
}
|
|
||||||
|
|
||||||
return xSum;
|
|
||||||
});
|
|
||||||
|
|
||||||
ySum /= nSamples;
|
|
||||||
return (ySum / nLines);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ========== Up ==========
|
|
||||||
static if (directionX == up) {
|
|
||||||
|
|
||||||
if (posSrcX[0] == posSrcX[1]) {
|
|
||||||
static if (mode == SamplingMode.single) {
|
|
||||||
return (() @trusted => pxNeighs[idxTL].components.ptr[ib])();
|
|
||||||
}
|
|
||||||
static if (mode == SamplingMode.dual) {
|
|
||||||
return () @trusted {
|
|
||||||
ulong[2] result = [
|
|
||||||
pxNeighs[idxTL].components.ptr[ib],
|
|
||||||
pxNeighs[idxBL].components.ptr[ib],
|
|
||||||
];
|
|
||||||
return result;
|
|
||||||
}();
|
|
||||||
}
|
|
||||||
static if (mode == SamplingMode.multi) {
|
|
||||||
const ySum = foreachLine(delegate(const Point posLine) {
|
|
||||||
ulong xSum = 0;
|
|
||||||
const samplingOffset = source.scanTo(posLine);
|
|
||||||
return (() @trusted
|
|
||||||
=> source.data.ptr[samplingOffset].components.ptr[ib]
|
|
||||||
)();
|
|
||||||
});
|
|
||||||
return (ySum / nLines);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const ulong[2] weightsX = () {
|
|
||||||
ulong[2] result;
|
|
||||||
result[0] = (udecimalHalf + posSrcX[1] - posSrcCenterX).fractionalDigits;
|
|
||||||
result[1] = ulong(uint.max) + 1 - result[0];
|
|
||||||
return result;
|
|
||||||
}();
|
|
||||||
|
|
||||||
static if (mode == SamplingMode.single) {
|
|
||||||
ulong xSum = 0;
|
|
||||||
|
|
||||||
() @trusted {
|
|
||||||
xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]);
|
|
||||||
xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]);
|
|
||||||
}();
|
|
||||||
|
|
||||||
xSum >>= 32;
|
|
||||||
return xSum;
|
|
||||||
}
|
|
||||||
|
|
||||||
static if (mode == SamplingMode.dual) {
|
|
||||||
ulong[2] xSums = [0, 0];
|
|
||||||
|
|
||||||
() @trusted {
|
|
||||||
xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]);
|
|
||||||
xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]);
|
|
||||||
|
|
||||||
xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]);
|
|
||||||
xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]);
|
|
||||||
}();
|
|
||||||
|
|
||||||
foreach (ref sum; xSums) {
|
|
||||||
sum >>= 32;
|
|
||||||
}
|
|
||||||
|
|
||||||
return xSums;
|
|
||||||
}
|
|
||||||
|
|
||||||
static if (mode == SamplingMode.multi) {
|
|
||||||
const ySum = foreachLine(delegate(const Point posLine) {
|
|
||||||
ulong xSum = 0;
|
|
||||||
|
|
||||||
const samplingOffset = source.scanTo(posLine);
|
|
||||||
ubyte[2] pxcLR = () @trusted {
|
|
||||||
ubyte[2] result = [
|
|
||||||
source.data.ptr[samplingOffset].components.ptr[ib],
|
|
||||||
source.data.ptr[samplingOffset + 1].components.ptr[ib],
|
|
||||||
];
|
|
||||||
return result;
|
|
||||||
}();
|
|
||||||
|
|
||||||
xSum += (pxcLR[idxL] * weightsX[idxL]);
|
|
||||||
xSum += (pxcLR[idxR] * weightsX[idxR]);
|
|
||||||
|
|
||||||
return (xSum >> 32);
|
|
||||||
});
|
|
||||||
|
|
||||||
return (ySum / nLines);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// ======== Interpolate Y ========
|
// ======== Interpolate Y ========
|
||||||
static if (directionY == none) {
|
static if (directionY == none) {
|
||||||
c = clamp255(sampleX!(SamplingMode.single)());
|
foreach (immutable ib, ref c; pxInt.components) {
|
||||||
|
c = clamp255(sampleX!(SamplingMode.single)(ib));
|
||||||
}
|
}
|
||||||
static if (directionY == down) {
|
}
|
||||||
c = clamp255(sampleX!(SamplingMode.multi)());
|
static if (directionY == down) {
|
||||||
|
foreach (immutable ib, ref c; pxInt.components) {
|
||||||
|
c = clamp255(sampleX!(SamplingMode.multi)(ib));
|
||||||
}
|
}
|
||||||
static if (directionY == up) {
|
}
|
||||||
const xSums = sampleX!(SamplingMode.dual)();
|
static if (directionY == up) {
|
||||||
|
foreach (immutable ib, ref c; pxInt.components) {
|
||||||
|
const xSums = sampleX!(SamplingMode.dual)(ib);
|
||||||
|
|
||||||
ulong ySum = 0;
|
ulong ySum = 0;
|
||||||
ySum += (xSums[idxT] * weightsY[idxT]);
|
ySum += (xSums[idxT] * weightsY[idxT]);
|
||||||
|
|
Loading…
Reference in New Issue