/* clang-format off */ //$ nobt //$ nocpp /** * @file avir.h * * @brief The "main" inclusion file with all required classes and functions. * * This is the "main" inclusion file for the "AVIR" image resizer. This * inclusion file contains implementation of the AVIR image resizing algorithm * in its entirety. Also includes several classes and functions that can be * useful elsewhere. * * AVIR Copyright (c) 2015-2019 Aleksey Vaneev * * @mainpage * * @section intro_sec Introduction * * Description is available at https://github.com/avaneev/avir * * AVIR is devoted to women. Your digital photos can look good at any size! * * @section license License * * AVIR License Agreement * * The MIT License (MIT) * * Copyright (c) 2015-2019 Aleksey Vaneev * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. * * Please credit the author of this library in your documentation in the * following way: "AVIR image resizing algorithm designed by Aleksey Vaneev" * * @version 2.4 */ #ifndef AVIR_CIMAGERESIZER_INCLUDED #define AVIR_CIMAGERESIZER_INCLUDED #include "third_party/avir/notice.h" #include "libc/bits/xmmintrin.h" #include "libc/str/str.h" #include "libc/mem/mem.h" #include "libc/bits/bits.h" #include "libc/math.h" namespace avir { /** * The macro defines AVIR version string. */ #define AVIR_VERSION "2.4" /** * The macro equals to "pi" constant, fills 53-bit floating point mantissa. * Undefined at the end of file. */ #define AVIR_PI 3.1415926535897932 /** * The macro equals to "pi divided by 2" constant, fills 53-bit floating * point mantissa. Undefined at the end of file. */ #define AVIR_PId2 1.5707963267948966 /** * Rounding function, based on the (int) typecast. Biased result. Not suitable * for numbers >= 2^31. * * @param d Value to round. * @return Rounded value. Some bias may be introduced. */ template inline T round(const T d) { return (d < 0.0 ? -(T)(int)((T)0.5 - d) : (T)(int)(d + (T)0.5)); } /** * Template function "clamps" (clips) the specified value so that it is not * lesser than "minv", and not greater than "maxv". * * @param Value Value to clamp. * @param minv Minimal allowed value. * @param maxv Maximal allowed value. * @return The clamped value. */ template inline T clamp(const T& Value, const T minv, const T maxv) { if (Value < minv) { return (minv); } else if (Value > maxv) { return (maxv); } else { return (Value); } } /** * Power 2.4 approximation function, designed for sRGB gamma correction. * * @param x Argument, in the range 0.09 to 1. * @return Value raised into power 2.4, approximate. */ template inline T pow24_sRGB(const T x) { const double x2 = x * x; const double x3 = x2 * x; const double x4 = x2 * x2; return ((T)(0.0985766365536824 + 0.839474952656502 * x2 + 0.363287814061725 * x3 - 0.0125559718896615 / (0.12758338921578 + 0.290283465468235 * x) - 0.231757513261358 * x - 0.0395365717969074 * x4)); } /** * Power 1/2.4 approximation function, designed for sRGB gamma correction. * * @param x Argument, in the range 0.003 to 1. * @return Value raised into power 1/2.4, approximate. */ template inline T pow24i_sRGB(const T x) { const double sx = sqrt(x); const double ssx = sqrt(sx); const double sssx = sqrt(ssx); return ((T)(0.000213364515060263 + 0.0149409239419218 * x + 0.433973412731747 * sx + ssx * (0.659628181609715 * sssx - 0.0380957908841466 - 0.0706476137208521 * sx))); } /** * Function approximately linearizes the sRGB gamma value. * * @param s sRGB gamma value, in the range 0 to 1. * @return Linearized sRGB gamma value, approximated. */ template inline T convertSRGB2Lin(const T s) { const T a = (T)0.055; if (s <= (T)0.04045) { return (s / (T)12.92); } return (pow24_sRGB((s + a) / ((T)1 + a))); } /** * Function approximately de-linearizes the linear gamma value. * * @param s Linear gamma value, in the range 0 to 1. * @return sRGB gamma value, approximated. */ template inline T convertLin2SRGB(const T s) { const T a = (T)0.055; if (s <= (T)0.0031308) { return ((T)12.92 * s); } return (((T)1 + a) * pow24i_sRGB(s) - a); } /** * Function converts (via typecast) specified array of type T1 values of * length l into array of type T2 values. If T1 is the same as T2, copy * operation is performed. When copying data at overlapping address spaces, * "op" should be lower than "ip". * * @param ip Input buffer. * @param[out] op Output buffer. * @param l The number of elements to copy. * @param ip Input buffer pointer increment. * @param op Output buffer pointer increment. */ template inline void copyArray(const T1* ip, T2* op, int l, const int ipinc = 1, const int opinc = 1) { while (l > 0) { *op = (T2)*ip; op += opinc; ip += ipinc; l--; } } /** * Function adds values located in array "ip" to array "op". * * @param ip Input buffer. * @param[out] op Output buffer. * @param l The number of elements to add. * @param ip Input buffer pointer increment. * @param op Output buffer pointer increment. */ template inline void addArray(const T1* ip, T2* op, int l, const int ipinc = 1, const int opinc = 1) { while (l > 0) { *op += *ip; op += opinc; ip += ipinc; l--; } } /** * Function that replicates a set of adjacent elements several times in a row. * This operation is usually used to replicate pixels at the start or end of * image's scanline. * * @param ip Source array. * @param ipl Source array length (usually 1..4, but can be any number). * @param[out] op Destination buffer. * @param l Number of times the source array should be replicated (the * destination buffer should be able to hold ipl * l number of elements). * @param opinc Destination buffer position increment after replicating the * source array. This value should be equal to at least ipl. */ template inline void replicateArray(const T1* const ip, const int ipl, T2* op, int l, const int opinc) { if (ipl == 1) { while (l > 0) { op[0] = ip[0]; op += opinc; l--; } } else if (ipl == 4) { while (l > 0) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; op[3] = ip[3]; op += opinc; l--; } } else if (ipl == 3) { while (l > 0) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; op += opinc; l--; } } else if (ipl == 2) { while (l > 0) { op[0] = ip[0]; op[1] = ip[1]; op += opinc; l--; } } else { while (l > 0) { int i; for (i = 0; i < ipl; i++) { op[i] = ip[i]; } op += opinc; l--; } } } /** * Function calculates frequency response of the specified FIR filter at the * specified circular frequency. Phase can be calculated as atan2( im, re ). * Function uses computationally-efficient oscillators instead of "cos" and * "sin" functions. * * @param flt FIR filter's coefficients. * @param fltlen Number of coefficients (taps) in the filter. * @param th Circular frequency [0; pi]. * @param[out] re0 Resulting real part of the complex frequency response. * @param[out] im0 Resulting imaginary part of the complex frequency response. * @param fltlat Filter's latency in samples (taps). */ template inline void calcFIRFilterResponse(const T* flt, int fltlen, const double th, double& re0, double& im0, const int fltlat = 0) { const double sincr = 2.0 * cos(th); double cvalue1; double svalue1; if (fltlat == 0) { cvalue1 = 1.0; svalue1 = 0.0; } else { cvalue1 = cos(-fltlat * th); svalue1 = sin(-fltlat * th); } double cvalue2 = cos(-(fltlat + 1) * th); double svalue2 = sin(-(fltlat + 1) * th); double re = 0.0; double im = 0.0; while (fltlen > 0) { re += cvalue1 * flt[0]; im += svalue1 * flt[0]; flt++; fltlen--; double tmp = cvalue1; cvalue1 = sincr * cvalue1 - cvalue2; cvalue2 = tmp; tmp = svalue1; svalue1 = sincr * svalue1 - svalue2; svalue2 = tmp; } re0 = re; im0 = im; } /** * Function normalizes FIR filter so that its frequency response at DC is * equal to DCGain. * * @param[in,out] p Filter coefficients. * @param l Filter length. * @param DCGain Filter's gain at DC. * @param pstep "p" array step. */ template inline void normalizeFIRFilter(T* const p, const int l, const double DCGain, const int pstep = 1) { double s = 0.0; T* pp = p; int i = l; while (i > 0) { s += *pp; pp += pstep; i--; } s = DCGain / s; pp = p; i = l; while (i > 0) { *pp = (T)(*pp * s); pp += pstep; i--; } } /** * @brief Memory buffer class for element array storage, with capacity * tracking. * * Allows easier handling of memory blocks allocation and automatic * deallocation for arrays (buffers) consisting of elements of specified * class. Tracks buffer's capacity in "int" variable; unsuitable for * allocation of very large memory blocks (with more than 2 billion elements). * * This class manages memory space only - it does not perform element class * construction (initialization) operations. Buffer's required memory address * alignment specification is supported. * * Uses standard library to allocate and deallocate memory. * * @tparam T Buffer element's type. * @tparam capint Buffer capacity's type to use. Use size_t for large buffers. */ template class CBuffer { public: CBuffer() : Data(NULL), DataAligned(NULL), Capacity(0), Alignment(0) {} /** * Constructor creates the buffer with the specified capacity. * * @param aCapacity Buffer's capacity. * @param aAlignment Buffer's required memory address alignment. 0 - use * stdlib's default alignment. */ CBuffer(const capint aCapacity, const int aAlignment = 0) { allocinit(aCapacity, aAlignment); } CBuffer(const CBuffer& Source) { allocinit(Source.Capacity, Source.Alignment); memcpy(DataAligned, Source.DataAligned, Capacity * sizeof(T)); } ~CBuffer() { freeData(); } CBuffer& operator=(const CBuffer& Source) { alloc(Source.Capacity, Source.Alignment); memcpy(DataAligned, Source.DataAligned, Capacity * sizeof(T)); return (*this); } /** * Function allocates memory so that the specified number of elements * can be stored in *this buffer object. * * @param aCapacity Storage for this number of elements to allocate. * @param aAlignment Buffer's required memory address alignment, * power-of-2 values only. 0 - use stdlib's default alignment. */ void alloc(const capint aCapacity, const int aAlignment = 0) { freeData(); allocinit(aCapacity, aAlignment); } /** * Function deallocates any previously allocated buffer. */ void free() { freeData(); Data = NULL; DataAligned = NULL; Capacity = 0; Alignment = 0; } /** * @return The capacity of the element buffer. */ capint getCapacity() const { return (Capacity); } /** * Function "forces" *this buffer to have an arbitary capacity. Calling * this function invalidates all further operations except deleting *this * object. This function should not be usually used at all. Function can * be used to "model" certain buffer capacity without calling a costly * memory allocation function. * * @param NewCapacity A new "forced" capacity. */ void forceCapacity(const capint NewCapacity) { Capacity = NewCapacity; } /** * Function reallocates *this buffer to a larger size so that it will be * able to hold the specified number of elements. Downsizing is not * performed. Alignment is not changed. * * @param NewCapacity New (increased) capacity. * @param DoDataCopy "True" if data in the buffer should be retained. */ void increaseCapacity(const capint NewCapacity, const bool DoDataCopy = true) { if (NewCapacity < Capacity) { return; } if (DoDataCopy) { const capint PrevCapacity = Capacity; T* const PrevData = Data; T* const PrevDataAligned = DataAligned; allocinit(NewCapacity, Alignment); memcpy(DataAligned, PrevDataAligned, PrevCapacity * sizeof(T)); ::free(PrevData); } else { ::free(Data); allocinit(NewCapacity, Alignment); } } /** * Function "truncates" (reduces) capacity of the buffer without * reallocating it. Alignment is not changed. * * @param NewCapacity New required capacity. */ void truncateCapacity(const capint NewCapacity) { if (NewCapacity >= Capacity) { return; } Capacity = NewCapacity; } /** * Function increases capacity so that the specified number of * elements can be stored. This function increases the previous capacity * value by third the current capacity value until space for the required * number of elements is available. Alignment is not changed. * * @param ReqCapacity Required capacity. */ void updateCapacity(const capint ReqCapacity) { if (ReqCapacity <= Capacity) { return; } capint NewCapacity = Capacity; while (NewCapacity < ReqCapacity) { NewCapacity += NewCapacity / 3 + 1; } increaseCapacity(NewCapacity); } operator T*() const { return (DataAligned); } private: T* Data; ///< Element buffer pointer. ///< T* DataAligned; ///< Memory address-aligned element buffer pointer. ///< capint Capacity; ///< Element buffer capacity. ///< int Alignment; ///< Memory address alignment in use. 0 - use stdlib's ///< default alignment. ///< /** * Internal element buffer allocation function used during object * construction. * * @param aCapacity Storage for this number of elements to allocate. * @param aAlignment Buffer's required memory address alignment. 0 - use * stdlib's default alignment. */ void allocinit(const capint aCapacity, const int aAlignment) { if (aAlignment == 0) { Data = (T*)::malloc(aCapacity * sizeof(T)); DataAligned = Data; Alignment = 0; } else { Data = (T*)::malloc(aCapacity * sizeof(T) + aAlignment); DataAligned = alignptr(Data, aAlignment); Alignment = aAlignment; } Capacity = aCapacity; } /** * Function frees a previously allocated Data buffer. */ void freeData() { ::free(Data); } /** * Function modifies the specified pointer so that it becomes memory * address-aligned. * * @param ptr Pointer to align. * @param align Alignment in bytes to apply. * @return Pointer aligned to align bytes. Works with power-of-2 * alignments only. If no alignment is necessary, "align" bytes will be * added to the pointer value. */ template inline Tp alignptr(const Tp ptr, const uintptr_t align) { return ((Tp)((uintptr_t)ptr + align - ((uintptr_t)ptr & (align - 1)))); } }; /** * Function optimizes the length of the symmetric-odd FIR filter by removing * left- and rightmost elements that are below specific threshold. * * Synthetic test shows that filter gets optimized in 2..3% of cases and in * each such case optimization reduces filter length by 6..8%. Optimization, * however, may skew the results of algorithm modeling and complexity * calculation leading to a choice of a less optimal algorithm. * * @param[in,out] Flt Buffer that contains filter being optimized. * @param[in,out] FltLatency Variable that holds the current latency of the * filter. May be adjusted on function return. * @param Threshold Threshold level. */ template inline void optimizeFIRFilter(CBuffer& Flt, int& FltLatency, T const Threshold = (T)0.00001) { int i; // Optimize length. for (i = 0; i <= FltLatency; i++) { if (fabs(Flt[i]) >= Threshold || i == FltLatency) { if (i > 0) { const int NewCapacity = Flt.getCapacity() - i * 2; copyArray(&Flt[i], &Flt[0], NewCapacity); Flt.truncateCapacity(NewCapacity); FltLatency -= i; } break; } } } /** * @brief Array of structured objects. * * Implements allocation of a linear array of objects of class T (which are * initialized), addressable via operator[]. Each object is created via the * "operator new". New object insertions are quick since implementation uses * prior space allocation (capacity), thus not requiring frequent memory block * reallocations. * * @tparam T Array element's type. */ template class CStructArray { public: CStructArray() : ItemCount(0) {} CStructArray(const CStructArray& Source) : ItemCount(0), Items(Source.getItemCount()) { while (ItemCount < Source.getItemCount()) { Items[ItemCount] = new T(Source[ItemCount]); ItemCount++; } } ~CStructArray() { clear(); } CStructArray& operator=(const CStructArray& Source) { clear(); const int NewCount = Source.ItemCount; Items.updateCapacity(NewCount); while (ItemCount < NewCount) { Items[ItemCount] = new T(Source[ItemCount]); ItemCount++; } return (*this); } T& operator[](const int Index) { return (*Items[Index]); } const T& operator[](const int Index) const { return (*Items[Index]); } /** * Function creates a new object of type T with the default constructor * and adds this object to the array. * * @return Reference to a newly added object. */ T& add() { if (ItemCount == Items.getCapacity()) { Items.increaseCapacity(ItemCount * 3 / 2 + 1); } Items[ItemCount] = new T(); ItemCount++; return ((*this)[ItemCount - 1]); } /** * Function changes number of allocated items. New items are created with * the default constructor. If NewCount is below the current item count, * items that are above NewCount range will be destructed. * * @param NewCount New requested item count. */ void setItemCount(const int NewCount) { if (NewCount > ItemCount) { Items.increaseCapacity(NewCount); while (ItemCount < NewCount) { Items[ItemCount] = new T(); ItemCount++; } } else { while (ItemCount > NewCount) { ItemCount--; delete Items[ItemCount]; } } } /** * Function erases all items of *this array. */ void clear() { while (ItemCount > 0) { ItemCount--; delete Items[ItemCount]; } } /** * @return The number of allocated items. */ int getItemCount() const { return (ItemCount); } private: int ItemCount; ///< The number of items available in the array. ///< CBuffer Items; ///< Element buffer. ///< }; /** * @brief Sine signal generator class. * * Class implements sine signal generator without biasing, with * constructor-based initalization only. This generator uses oscillator * instead of "sin" function. */ class CSineGen { public: /** * Constructor initializes *this sine signal generator. * * @param si Sine function increment, in radians. * @param ph Starting phase, in radians. Add 0.5 * AVIR_PI for cosine * function. */ CSineGen(const double si, const double ph) : svalue1(sin(ph)), svalue2(sin(ph - si)), sincr(2.0 * cos(si)) {} /** * @return The next value of the sine function, without biasing. */ double generate() { const double res = svalue1; svalue1 = sincr * res - svalue2; svalue2 = res; return (res); } private: double svalue1; ///< Current sine value. ///< double svalue2; ///< Previous sine value. ///< double sincr; ///< Sine value increment. ///< }; /** * @brief Peaked Cosine window function generator class. * * Class implements Peaked Cosine window function generator. Generates the * right-handed half of the window function. The Alpha parameter of this * window function offers the control of the balance between the early and * later taps of the filter. E.g. at Alpha=1 both early and later taps are * attenuated, but at Alpha=4 mostly later taps are attenuated. This offers a * great control over ringing artifacts produced by a low-pass filter in image * processing, without compromising achieved image sharpness. */ class CDSPWindowGenPeakedCosine { public: /** * Constructor initializes *this window function generator. * * @param aAlpha Alpha parameter, affects the peak shape (peak * augmentation) of the window function. Should be >= 1.0. * @param aLen2 Half filter's length (non-truncated). */ CDSPWindowGenPeakedCosine(const double aAlpha, const double aLen2) : Alpha(aAlpha), Len2(aLen2), wn(0), w1(AVIR_PId2 / Len2, AVIR_PI * 0.5) {} /** * @return The next Peaked Cosine window function coefficient. */ double generate() { const double h = pow(wn / Len2, Alpha); wn++; return (w1.generate() * (1.0 - h)); } private: double Alpha; ///< Alpha parameter, affects the peak shape of window. ///< double Len2; ///< Half length of the window function. ///< int wn; ///< Window function integer position. 0 - center of the ///< window function. ///< CSineGen w1; ///< Sine-wave generator. ///< }; /** * @brief FIR filter-based equalizer generator. * * Class implements an object used to generate symmetric-odd FIR filters with * the specified frequency response (aka paragraphic equalizer). The * calculated filter is windowed by the Peaked Cosine window function. * * In image processing, due to short length of filters being used (6-8 taps) * the resulting frequency response of the filter is approximate and may be * mathematically imperfect, but still adequate to the visual requirements. * * On a side note, this equalizer generator can be successfully used for audio * signal equalization as well: for example, it is used in almost the same * form in Voxengo Marvel GEQ equalizer plug-in. * * Filter generation is based on decomposition of frequency range into * spectral bands, with each band represented by linear and ramp "kernels". * When the filter is built, these kernels are combined together with * different weights that approximate the required frequency response. */ class CDSPFIREQ { public: /** * Function initializes *this object with the required parameters. The * gain of frequencies beyond the MinFreq..MaxFreq range are controlled by * the first and the last band's gain. * * @param SampleRate Processing sample rate (use 2 for image processing). * @param aFilterLength Required filter length in samples (taps). The * actual filter length is truncated to an integer value. * @param aBandCount Number of band crossover points required to control, * including bands at MinFreq and MaxFreq. * @param MinFreq Minimal frequency that should be controlled. * @param MaxFreq Maximal frequency that should be controlled. * @param IsLogBands "True" if the bands should be spaced logarithmically. * @param WFAlpha Peaked Cosine window function's Alpha parameter. */ void init(const double SampleRate, const double aFilterLength, const int aBandCount, const double MinFreq, const double MaxFreq, const bool IsLogBands, const double WFAlpha) { FilterLength = aFilterLength; BandCount = aBandCount; CenterFreqs.alloc(BandCount); z = (int)ceil(FilterLength * 0.5); zi = z + (z & 1); z2 = z * 2; CBuffer oscbuf(z2); initOscBuf(oscbuf); CBuffer winbuf(z); initWinBuf(winbuf, WFAlpha); UseFirstVirtBand = (MinFreq > 0.0); const int k = zi * (BandCount + (UseFirstVirtBand ? 1 : 0)); Kernels1.alloc(k); Kernels2.alloc(k); double m; // Frequency step multiplier. double mo; // Frequency step offset (addition). if (IsLogBands) { m = exp(log(MaxFreq / MinFreq) / (BandCount - 1)); mo = 0.0; } else { m = 1.0; mo = (MaxFreq - MinFreq) / (BandCount - 1); } double f = MinFreq; double x1 = 0.0; double x2; int si; if (UseFirstVirtBand) { si = 0; } else { si = 1; CenterFreqs[0] = 0.0; f = f * m + mo; } double* kernbuf1 = &Kernels1[0]; double* kernbuf2 = &Kernels2[0]; int i; for (i = si; i < BandCount; i++) { x2 = f * 2.0 / SampleRate; CenterFreqs[i] = x2; fillBandKernel(x1, x2, kernbuf1, kernbuf2, oscbuf, winbuf); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; f = f * m + mo; } if (x1 < 1.0) { UseLastVirtBand = true; fillBandKernel(x1, 1.0, kernbuf1, kernbuf2, oscbuf, winbuf); } else { UseLastVirtBand = false; } } /** * @return Filter's length, in samples (taps). */ int getFilterLength() const { return (z2 - 1); } /** * @return Filter's latency (group delay), in samples (taps). */ int getFilterLatency() const { return (z - 1); } /** * Function creates symmetric-odd FIR filter with the specified gain * levels at band crossover points. * * @param BandGains Array of linear gain levels, count=BandCount specified * in the init() function. * @param[out] Filter Output filter buffer, length = getFilterLength(). */ void buildFilter(const double* const BandGains, double* const Filter) { const double* kernbuf1 = &Kernels1[0]; const double* kernbuf2 = &Kernels2[0]; double x1 = 0.0; double y1 = BandGains[0]; double x2; double y2; int i; int si; if (UseFirstVirtBand) { si = 1; x2 = CenterFreqs[0]; y2 = y1; } else { si = 2; x2 = CenterFreqs[1]; y2 = BandGains[1]; } copyBandKernel(Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - x2 * y1); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; y1 = y2; for (i = si; i < BandCount; i++) { x2 = CenterFreqs[i]; y2 = BandGains[i]; addBandKernel(Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - x2 * y1); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; y1 = y2; } if (UseLastVirtBand) { addBandKernel(Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - y1); } for (i = 0; i < z - 1; i++) { Filter[z + i] = Filter[z - 2 - i]; } } /** * Function calculates filter's length (in samples) and latency depending * on the required non-truncated filter length. * * @param aFilterLength Required filter length in samples (non-truncated). * @param[out] Latency Resulting latency (group delay) of the filter, * in samples (taps). * @return Filter length in samples (taps). */ static int calcFilterLength(const double aFilterLength, int& Latency) { const int l = (int)ceil(aFilterLength * 0.5); Latency = l - 1; return (l * 2 - 1); } private: double FilterLength; ///< Length of filter. ///< int z; ///< Equals (int) ceil( FilterLength * 0.5 ). ///< int zi; ///< Equals "z" if z is even, or z + 1 if z is odd. Used as a ///< Kernels1 and Kernels2 size multiplier and kernel buffer ///< increment to make sure each kernel buffer is 16-byte aligned. ///< int z2; ///< Equals z * 2. ///< int BandCount; ///< Number of controllable bands. ///< CBuffer CenterFreqs; ///< Center frequencies for all bands, ///< normalized to 0.0-1.0 range. ///< CBuffer Kernels1; ///< Half-length kernel buffers for each ///< spectral band (linear part). ///< CBuffer Kernels2; ///< Half-length kernel buffers for each ///< spectral band (ramp part). ///< bool UseFirstVirtBand; ///< "True" if the first virtual band ///< (between 0.0 and MinFreq) should be used. The ///< first virtual band won't be used if MinFreq ///< equals 0.0. ///< bool UseLastVirtBand; ///< "True" if the last virtual band (between ///< MaxFreq and SampleRate * 0.5) should be used. The ///< last virtual band won't be used if MaxFreq * 2.0 ///< equals SampleRate. ///< /** * Function initializes the "oscbuf" used in the fillBandKernel() * function. * * @param oscbuf Oscillator buffer, length = z * 2. */ void initOscBuf(double* oscbuf) const { int i = z; while (i > 0) { oscbuf[0] = 0.0; oscbuf[1] = 1.0; oscbuf += 2; i--; } } /** * Function initializes window function buffer. This function generates * Peaked Cosine window function. * * @param winbuf Windowing buffer. * @param Alpha Peaked Cosine alpha parameter. */ void initWinBuf(double* winbuf, const double Alpha) const { CDSPWindowGenPeakedCosine wf(Alpha, FilterLength * 0.5); int i; for (i = 1; i <= z; i++) { winbuf[z - i] = wf.generate(); } } /** * Function fills first half of symmetric-odd FIR kernel for the band. * This function should be called successively for adjacent bands. * Previous band's x2 should be equal to current band's x1. A band kernel * consists of 2 elements: linear kernel and ramp kernel. * * @param x1 Band's left corner frequency (0..1). * @param x2 Band's right corner frequency (0..1). * @param kernbuf1 Band kernel buffer 1 (linear part), length = z. * @param kernbuf2 Band kernel buffer 2 (ramp part), length = z. * @param oscbuf Oscillation buffer. Before the first call of the * fillBandKernel() should be initialized with the call of the * initOscBuf() function. * @param winbuf Buffer that contains windowing function. */ void fillBandKernel(const double x1, const double x2, double* kernbuf1, double* kernbuf2, double* oscbuf, const double* const winbuf) { const double s2_incr = AVIR_PI * x2; const double s2_coeff = 2.0 * cos(s2_incr); double s2_value1 = sin(s2_incr * (-z + 1)); double c2_value1 = sin(s2_incr * (-z + 1) + AVIR_PI * 0.5); oscbuf[0] = sin(s2_incr * -z); oscbuf[1] = sin(s2_incr * -z + AVIR_PI * 0.5); int ks; for (ks = 1; ks < z; ks++) { const int ks2 = ks * 2; const double s1_value1 = oscbuf[ks2]; const double c1_value1 = oscbuf[ks2 + 1]; oscbuf[ks2] = s2_value1; oscbuf[ks2 + 1] = c2_value1; const double x = AVIR_PI * (ks - z); const double v0 = winbuf[ks - 1] / ((x1 - x2) * x); kernbuf1[ks - 1] = (x2 * s2_value1 - x1 * s1_value1 + (c2_value1 - c1_value1) / x) * v0; kernbuf2[ks - 1] = (s2_value1 - s1_value1) * v0; s2_value1 = s2_coeff * s2_value1 - oscbuf[ks2 - 2]; c2_value1 = s2_coeff * c2_value1 - oscbuf[ks2 - 1]; } kernbuf1[z - 1] = (x2 * x2 - x1 * x1) / (x1 - x2) * 0.5; kernbuf2[z - 1] = -1.0; } /** * Function copies band kernel's elements to the output buffer. * * @param outbuf Output buffer. * @param kernbuf1 Kernel buffer 1 (linear part). * @param kernbuf2 Kernel buffer 2 (ramp part). * @param c Multiplier for linear kernel element. * @param d Multiplier for ramp kernel element. */ void copyBandKernel(double* outbuf, const double* const kernbuf1, const double* const kernbuf2, const double c, const double d) const { int ks; for (ks = 0; ks < z; ks++) { outbuf[ks] = c * kernbuf1[ks] + d * kernbuf2[ks]; } } /** * Function adds band kernel's elements to the output buffer. * * @param outbuf Output buffer. * @param kernbuf1 Kernel buffer 1 (linear part). * @param kernbuf2 Kernel buffer 2 (ramp part). * @param c Multiplier for linear kernel element. * @param d Multiplier for ramp kernel element. */ void addBandKernel(double* outbuf, const double* const kernbuf1, const double* const kernbuf2, const double c, const double d) const { int ks; for (ks = 0; ks < z; ks++) { outbuf[ks] += c * kernbuf1[ks] + d * kernbuf2[ks]; } } }; /** * @brief Low-pass filter windowed by Peaked Cosine window function. * * This class implements calculation of linear-phase symmetric-odd FIR * low-pass filter windowed by the Peaked Cosine window function, for image * processing applications. */ class CDSPPeakedCosineLPF { public: int fl2; ///< Half filter's length, excluding the peak value. This value ///< can be also used as filter's latency in samples (taps). ///< int FilterLen; ///< Filter's length in samples (taps). ///< /** * Constructor initalizes *this object. * * @param aLen2 Half-length (non-truncated) of low-pass filter, in samples * (taps). * @param aFreq2 Low-pass filter's corner frequency [0; pi]. * @param aAlpha Peaked Cosine window function Alpha parameter. */ CDSPPeakedCosineLPF(const double aLen2, const double aFreq2, const double aAlpha) : fl2((int)ceil(aLen2) - 1), FilterLen(fl2 + fl2 + 1), Len2(aLen2), Freq2(aFreq2), Alpha(aAlpha) {} /** * Function generates a linear-phase low-pass filter windowed by Peaked * Cosine window function. * * @param[out] op Output buffer, length = FilterLen (fl2 * 2 + 1). * @param DCGain Required gain at DC. The resulting filter will be * normalized to achieve this DC gain. */ template void generateLPF(T* op, const double DCGain) { CDSPWindowGenPeakedCosine wf(Alpha, Len2); CSineGen f2(Freq2, 0.0); op += fl2; T* op2 = op; f2.generate(); int t = 1; *op = (T)(Freq2 * wf.generate() / AVIR_PI); double s = *op; while (t <= fl2) { const double v = f2.generate() * wf.generate() / t / AVIR_PI; op++; op2--; *op = (T)v; *op2 = (T)v; s += *op + *op2; t++; } t = FilterLen; s = DCGain / s; while (t > 0) { *op2 = (T)(*op2 * s); op2++; t--; } } private: double Len2; ///< Half-length (non-truncated) of low-pass filter, in ///< samples (taps). ///< double Freq2; ///< Low-pass filter's corner frequency. ///< double Alpha; ///< Peaked Cosine window function Alpha parameter. ///< }; /** * @brief Buffer class for parametrized low-pass filter. * * This class extends the CBuffer< double > class by adding several variables * that define a symmetric-odd FIR low-pass filter windowed by Peaked Cosine * window function. This class can be used to compare filters without * comparing their buffer contents. */ class CFltBuffer : public CBuffer { public: double Len2; ///< Half-length (non-truncated) of low-pass filters, in ///< samples (taps). ///< double Freq; ///< Low-pass filter's corner frequency. ///< double Alpha; ///< Peaked Cosine window function Alpha parameter. ///< double DCGain; ///< DC gain applied to the filter. ///< CFltBuffer() : CBuffer(), Len2(0.0), Freq(0.0), Alpha(0.0), DCGain(0.0) {} /** * @param b2 Filter buffer to compare *this object to. * @return Operator returns "true" if both filters have same parameters. */ bool operator==(const CFltBuffer& b2) const { return (Len2 == b2.Len2 && Freq == b2.Freq && Alpha == b2.Alpha && DCGain == b2.DCGain); } }; /** * @brief Sinc function-based fractional delay filter bank. * * Class implements storage and initialization of a bank of sinc * function-based fractional delay filters, expressed as 1st order polynomial * interpolation coefficients. The filters are produced from a single "long" * windowed low-pass filter. Also supports 0th-order ("nearest neighbor") * interpolation. * * This class also supports multiplication of each fractional delay filter by * an external filter (usually a low-pass filter). * * @tparam fptype Specifies storage type of the filter coefficients bank. The * filters are initially calculated using the "double" precision. */ template class CDSPFracFilterBankLin { public: CDSPFracFilterBankLin() : Order(-1) {} /** * Copy constructor copies a limited set of parameters of the source * filter bank. The actual filters are not copied. Such copying is used * during filtering steps "modeling" stage. A further init() function * call is required. * * @param s Source filter bank. */ void copyInitParams(const CDSPFracFilterBankLin& s) { WFLen2 = s.WFLen2; WFFreq = s.WFFreq; WFAlpha = s.WFAlpha; FracCount = s.FracCount; Order = s.Order; Alignment = s.Alignment; SrcFilterLen = s.SrcFilterLen; FilterLen = s.FilterLen; FilterSize = s.FilterSize; IsSrcTableBuilt = false; ExtFilter = s.ExtFilter; TableFillFlags.alloc(s.TableFillFlags.getCapacity()); int i; // Copy table fill flags, but shifted so that further initialization // is still possible (such feature should not be used, though). for (i = 0; i < TableFillFlags.getCapacity(); i++) { TableFillFlags[i] = (uint8_t)(s.TableFillFlags[i] << 2); } } /** * Operator compares *this filter bank and another filter bank and returns * "true" if their parameters are equal. Alignment is not taken into * account. * * @param s Filter bank to compare to. * @return "True" if compared banks have equal parameters. */ bool operator==(const CDSPFracFilterBankLin& s) const { return (Order == s.Order && WFLen2 == s.WFLen2 && WFFreq == s.WFFreq && WFAlpha == s.WFAlpha && FracCount == s.FracCount && ExtFilter == s.ExtFilter); } /** * Function initializes (builds) the filter bank based on the supplied * parameters. If the supplied parameters are equal to previously defined * parameters, function does nothing (alignment is assumed to be never * changing between the init() function calls). * * @param ReqFracCount Required number of fractional delays in the filter * bank. The minimal value is 2. * @param ReqOrder Required order of the interpolation polynomial * (0 or 1). * @param BaseLen Low-pass filter's base length, in samples (taps). * Affects the actual length of the filter and its overall steepness. * @param Cutoff Low-pass filter's normalized cutoff frequency [0; 1]. * @param aWFAlpha Peaked Cosine window function's Alpha parameter. * @param aExtFilter External filter to apply to each fractional delay * filter. * @param aAlignment Memory alignment of the filter bank, power-of-2 * value. 0 - use default stdlib alignment. * @param FltLenAlign Filter's length alignment, power-of-2 value. */ void init(const int ReqFracCount, const int ReqOrder, const double BaseLen, const double Cutoff, const double aWFAlpha, const CFltBuffer& aExtFilter, const int aAlignment = 0, const int FltLenAlign = 1) { double NewWFLen2 = 0.5 * BaseLen * ReqFracCount; double NewWFFreq = AVIR_PI * Cutoff / ReqFracCount; double NewWFAlpha = aWFAlpha; if (ReqOrder == Order && NewWFLen2 == WFLen2 && NewWFFreq == WFFreq && NewWFAlpha == WFAlpha && ReqFracCount == FracCount && aExtFilter == ExtFilter) { IsInitRequired = false; return; } WFLen2 = NewWFLen2; WFFreq = NewWFFreq; WFAlpha = NewWFAlpha; FracCount = ReqFracCount; Order = ReqOrder; Alignment = aAlignment; ExtFilter = aExtFilter; CDSPPeakedCosineLPF p(WFLen2, WFFreq, WFAlpha); SrcFilterLen = (p.fl2 / ReqFracCount + 1) * 2; const int ElementSize = ReqOrder + 1; FilterLen = SrcFilterLen; if (ExtFilter.getCapacity() > 0) { FilterLen += ExtFilter.getCapacity() - 1; } FilterLen = (FilterLen + FltLenAlign - 1) & ~(FltLenAlign - 1); FilterSize = FilterLen * ElementSize; IsSrcTableBuilt = false; IsInitRequired = true; } /** * @return The length of each fractional delay filter, in samples (taps). * Always an even value. */ int getFilterLen() const { return (FilterLen); } /** * @return The number of fractional filters in use by *this bank. */ int getFracCount() const { return (FracCount); } /** * @return The order of the interpolation polynomial. */ int getOrder() const { return (Order); } /** * Function returns the pointer to the specified interpolation table * filter. * * @param i Filter (fractional delay) index, in the range 0 to * ReqFracCount - 1, inclusive. * @return Pointer to filter. Higher order polynomial coefficients are * stored after after previous order coefficients, separated by FilterLen * elements. */ const fptype* getFilter(const int i) { if (!IsSrcTableBuilt) { buildSrcTable(); } fptype* const Res = &Table[i * FilterSize]; if ((TableFillFlags[i] & 2) == 0) { createFilter(i); TableFillFlags[i] |= 2; if (Order > 0) { createFilter(i + 1); const fptype* const Res2 = Res + FilterSize; fptype* const op = Res + FilterLen; int j; // Create higher-order interpolation coefficients (linear // interpolation). for (j = 0; j < FilterLen; j++) { op[j] = Res2[j] - Res[j]; } } } return (Res); } /** * Function makes sure all fractional delay filters were created. */ void createAllFilters() { int i; for (i = 0; i < FracCount; i++) { getFilter(i); } } /** * Function returns an approximate initialization complexity, expressed in * the number of multiply-add operations. This includes fractional delay * filters calculation and multiplication by an external filter. This * function can only be called after the init() function. * * @param FracUseMap Fractional delays use map, each element corresponds * to a single fractional delay, will be compared to the internal table * fill flags. This map should include 0 and 1 values only. * @return The complexity of the initialization, expressed in the number * of multiply-add operations. */ int calcInitComplexity(const CBuffer& FracUseMap) const { const int FltInitCost = 65; // Cost to initialize a single sample // of the fractional delay filter. const int FltUseCost = FilterLen * Order + SrcFilterLen * ExtFilter.getCapacity(); // Cost to use a single // fractional delay filter. const int ucb[2] = {0, FltUseCost}; int ic; int i; if (IsInitRequired) { ic = FracCount * SrcFilterLen * FltInitCost; for (i = 0; i < FracCount; i++) { ic += ucb[FracUseMap[i]]; } } else { ic = 0; for (i = 0; i < FracCount; i++) { if (FracUseMap[i] != 0) { ic += ucb[TableFillFlags[i] == 0 ? 1 : 0]; } } } return (ic); } private: static const int InterpPoints = 2; ///< The maximal number of points the ///< interpolation is based on. ///< double WFLen2; ///< Window function's Len2 parameter. ///< double WFFreq; ///< Window function's Freq parameter. ///< double WFAlpha; ///< Window function's Alpha parameter. ///< int FracCount; ///< The required number of fractional delay filters. ///< int Order; ///< The order of the interpolation polynomial. ///< int Alignment; ///< The required filter table alignment. ///< int SrcFilterLen; ///< Length of the "source" filters. This is always an ///< even value. ///< int FilterLen; ///< Specifies the number of samples (taps) each fractional ///< delay filter has. This is always an even value, adjusted ///< by the FltLenAlign. ///< int FilterSize; ///< The size of a single filter element, equals ///< FilterLen * ElementSize. ///< bool IsInitRequired; ///< "True" if SrcTable filter table initialization ///< is required. This value is available only after the ///< call to the init() function. ///< CBuffer Table; ///< Interpolation table, size equals to ///< ReqFracCount * FilterLen * ElementSize. ///< CBuffer TableFillFlags; ///< Contains ReqFracCount + 1 ///< elements. Bit 0 of every element is 1 if Table ///< already contains the filter from SrcTable filtered ///< by ExtFilter. Bit 1 of every element means higher ///< order coefficients were filled for the filter. ///< CFltBuffer ExtFilter; ///< External filter that should be applied to every ///< fractional delay filter. Can be empty. Half of ///< this filter's capacity is used as latency (group ///< delay) value of the filter. ///< CBuffer SrcTable; ///< Source table of delay filters, contains ///< ReqFracCount + 1 elements. This table is used ///< to fill the Table with the actual filters, ///< filtered by an external filter. ///< bool IsSrcTableBuilt; ///< "True" if the SrcTable was built already. This ///< variable is set to "false" in the init() function. ///< /** * Function builds source table used in the createFilter() function. */ void buildSrcTable() { IsSrcTableBuilt = true; IsInitRequired = false; CDSPPeakedCosineLPF p(WFLen2, WFFreq, WFAlpha); const int BufLen = SrcFilterLen * FracCount + InterpPoints - 1; const int BufOffs = InterpPoints / 2 - 1; const int BufCenter = SrcFilterLen * FracCount / 2 + BufOffs; CBuffer Buf(BufLen); memset(Buf, 0, (BufCenter - p.fl2) * sizeof(double)); int i = BufLen - BufCenter - p.fl2 - 1; memset(&Buf[BufLen - i], 0, i * sizeof(double)); p.generateLPF(&Buf[BufCenter - p.fl2], FracCount); SrcTable.alloc((FracCount + 1) * SrcFilterLen); TableFillFlags.alloc(FracCount + 1); int j; double* op0 = SrcTable; for (i = FracCount; i >= 0; i--) { TableFillFlags[i] = 0; double* p = Buf + BufOffs + i; for (j = 0; j < SrcFilterLen; j++) { op0[0] = p[0]; op0++; p += FracCount; } } Table.alloc((FracCount + 1) * FilterSize, Alignment); } /** * Function creates the specified filter in the Table by copying it from * the SrcTable and filtering by ExtFilter. Function does nothing if * filter was already created. * * @param k Filter index to create, in the range 0 to FracCount, * inclusive. */ void createFilter(const int k) { if (TableFillFlags[k] != 0) { return; } TableFillFlags[k] |= 1; const int ExtFilterLatency = ExtFilter.getCapacity() / 2; const int ResLatency = ExtFilterLatency + SrcFilterLen / 2; int ResLen = SrcFilterLen; if (ExtFilter.getCapacity() > 0) { ResLen += ExtFilter.getCapacity() - 1; } const int ResOffs = FilterLen / 2 - ResLatency; fptype* op = &Table[k * FilterSize]; int i; for (i = 0; i < ResOffs; i++) { op[i] = 0.0; } for (i = ResOffs + ResLen; i < FilterLen; i++) { op[i] = 0.0; } op += ResOffs; const double* const srcflt = &SrcTable[k * SrcFilterLen]; if (ExtFilter.getCapacity() == 0) { for (i = 0; i < ResLen; i++) { op[i] = (fptype)srcflt[i]; } return; } // Perform convolution of extflt and srcflt. const double* const extflt = &ExtFilter[0]; int j; for (j = 0; j < ResLen; j++) { int k = 0; int l = j - ExtFilter.getCapacity() + 1; int r = l + ExtFilter.getCapacity(); if (l < 0) { k -= l; l = 0; } if (r > SrcFilterLen) { r = SrcFilterLen; } const double* const extfltb = extflt + k; const double* const srcfltb = srcflt + l; double s = 0.0; l = r - l; for (i = 0; i < l; i++) { s += extfltb[i] * srcfltb[i]; } op[j] = (fptype)s; } } }; /** * @brief Thread pool for multi-threaded image resizing operation. * * This base class is used to organize a multi-threaded image resizing * operation. The thread pool should consist of threads that initially wait * for a signal. Upon receiving a signal (via the startAllWorkloads() * function) each previously added thread should execute its workload's * process() function once, and return to the wait signal state again. The * thread pool should be also able to efficiently wait for all workloads to * finish via the waitAllWorkloadsToFinish() function. * * The image resizing algorithm makes calls to functions of this class. */ class CImageResizerThreadPool { public: CImageResizerThreadPool() {} virtual ~CImageResizerThreadPool() {} /** * @brief Thread pool's workload object class. * * This class should be used as a base class for objects that perform the * actual work spread over several threads. */ class CWorkload { public: virtual ~CWorkload() {} /** * Function that gets called from the thread when thread pool's * startAllWorkloads() function is called. */ virtual void process() = 0; }; /** * @return The suggested number of workloads (and their associated * threads) to add. The minimal value this function can return is 1. The * usual value may depend on the number of physical and virtual cores * present in the system, and on other considerations. */ virtual int getSuggestedWorkloadCount() const { return (1); } /** * Function adds a new workload (and possibly thread) to the thread pool. * The caller decides how many parallel workloads (and threads) it * requires, but this number will not exceed the value returned by the * getSuggestedWorkloadCount() function. It is implementation-specific how * many workloads to associate with a single thread. But for efficiency * reasons each workload should be associated with its own thread. * * Note that the same set of workload objects will be processed each time * the startAllWorkloads() function is called. This means that workload * objects are added only once. The caller changes the state of the * workload objects and then calls the startAllWorkloads() function to * process them. * * @param Workload Workload object whose process() function will be called * from within the thread when the startAllWorkloads() function is called. */ virtual void addWorkload(CWorkload* const Workload) {} /** * Function starts all workloads associated with threads previously added * via the addWorkload() function. It is assumed that this function * performs the necessary "memory barrier" (or "cache sync") kind of * operation so that all threads catch up the prior changes made to the * workload objects during their wait state. */ virtual void startAllWorkloads() {} /** * Function waits for all workloads to finish. */ virtual void waitAllWorkloadsToFinish() {} /** * Function removes all workloads previously added via the addWorkload() * function. This function gets called only after the * waitAllWorkloadsToFinish() function call. */ virtual void removeAllWorkloads() {} }; /** * @brief Resizing algorithm parameters structure. * * This structure holds all selectable parameters used by the resizing * algorithm at various stages, for both downsizing and upsizing. There are no * other parameters exist that can optimize the performance of the resizing * algorithm. Filter length parameters can take fractional values. * * Beside quality, these parameters (except Alpha parameters) directly affect * the computative cost of the resizing algorithm. It is possible to trade * the visual quality for computative cost. * * Anti-alias filtering during downsizing can be defined as a considerable * reduction of contrast of smallest features of an image. Unfortunately, such * de-contrasting partially affects features of all sizes thus producing a * non-linearity of frequency response. All pre-defined parameter sets are * described by 3 values separated by slashes. The first value is the * de-contrasting factor of small features (which are being removed) while * the second value is the de-contrasting factor of large features (which * should remain intact), with value of 1 equating to "no contrast change". * The third value is the optimization score (see below), with value of 0 * equating to the "perfect" linearity of frequency response. * * The pre-defined parameter sets offered by this library were auto-optimized * for the given LPFltBaseLen, IntFltLen and CorrFltAlpha values. The * optimization goal was to minimize the score: the sum of squares of the * difference between original and processed images (which was not actually * resized, k=1). The original image was a 0.5 megapixel uniformly-distributed * white-noise image with pixel intensities in the 0-1 range. Such goal * converges very well and produces filtering system with the flattest * frequency response possible for the given constraints. With this goal, * increasing the LPFltBaseLen value reduces the general amount of aliasing * artifacts. */ struct CImageResizerParams { double CorrFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the correction filter. The ///< "usable" values are in the narrow range 1.0 to 1.5. ///< double CorrFltLen; ///< Correction filter's length in samples (taps). The ///< "usable" range is narrow, 5.5 to 8, as to minimize ///< the "overcorrection" which is mathematically precise, ///< but visually unacceptable. ///< double IntFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the interpolation low-pass filter. ///< The "usable" values are in the range 1.5 to 2.5. ///< double IntFltCutoff; ///< Interpolation low-pass filter's cutoff frequency ///< (normalized, [0; 1]). The "usable" range is 0.6 to ///< 0.8. ///< double IntFltLen; ///< Interpolation low-pass filter's length in samples ///< (taps). The length value should be at least 18 or ///< otherwise a "dark grid" artifact will be introduced if ///< a further sharpening is applied. IntFltLen together ///< with other IntFlt parameters should be tuned in a way ///< that produces the flattest frequency response in 0-0.5 ///< normalized frequency range (this range is due to 2X ///< upsampling). ///< double LPFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the low-pass filter. The "usable" ///< values are in the range 1.5 to 6.5. ///< double LPFltBaseLen; ///< Base length of the low-pass (aka anti-aliasing ///< or reconstruction) filter, in samples (taps), ///< further adjusted by the actual cutoff frequency, ///< upsampling and downsampling factors. The "usable" ///< range is between 6 and 9. ///< double LPFltCutoffMult; ///< Low-pass filter's cutoff frequency ///< multiplier. This value can be both below and ///< above 1.0 as low-pass filters are inserted on ///< downsampling and upsampling steps and always ///< have corner frequency equal to or below 0.5pi. ///< This multiplier shifts low-pass filter's corner ///< frequency towards lower (if below 1.0) or higher ///< (if above 1.0) frequencies. This multiplier can ///< be way below 1.0 since any additional ///< high-frequency damping will be partially ///< corrected by the correction filter. The "usable" ///< range is 0.3 to 1.0. ///< CImageResizerParams() : HBFltAlpha(1.75395), HBFltCutoff(0.40356), HBFltLen(22.00000) {} double HBFltAlpha; ///< Half-band filter's Alpha. Assigned internally. ///< double HBFltCutoff; ///< Half-band filter's cutoff point [0; 1]. Assigned ///< internally. ///< double HBFltLen; ///< Length of the half-band low-pass filter. Assigned ///< internally. Internally used to perform 2X or higher ///< downsampling. These filter parameters should be treated ///< as "technical" and do not require adjustment as they ///< were tuned to suit all combinations of other ///< parameters. This half-band filter provides a wide ///< transition band (for minimal ringing artifacts) and a ///< high stop-band attenuation (for minimal aliasing). ///< }; /** * @brief The default set of resizing algorithm parameters * (10.01/1.029/0.019169). * * This is the default set of resizing parameters that was designed to deliver * a sharp image while still providing a low amount of ringing artifacts, and * having a reasonable computational cost. */ struct CImageResizerParamsDef : public CImageResizerParams { CImageResizerParamsDef() { CorrFltAlpha = 1.0; // 10.01/1.88/1.029(522.43)/0.019169:258648,446808 CorrFltLen = 6.30770; IntFltAlpha = 2.27825; IntFltCutoff = 0.75493; IntFltLen = 18.0; LPFltAlpha = 3.40127; LPFltBaseLen = 7.78; LPFltCutoffMult = 0.78797; } }; /** * @brief Set of resizing algorithm parameters for ultra-low-ringing * performance (7.69/1.069/0.000245). * * This set of resizing algorithm parameters offers the lowest amount of * ringing this library is capable of providing while still offering a decent * quality. Low ringing is attained at the expense of higher aliasing * artifacts and a slightly reduced contrast. */ struct CImageResizerParamsULR : public CImageResizerParams { CImageResizerParamsULR() { CorrFltAlpha = 1.0; // 7.69/1.97/1.069(31445.45)/0.000245:258627,436845 CorrFltLen = 5.83280; IntFltAlpha = 2.11453; IntFltCutoff = 0.73986; IntFltLen = 18.0; LPFltAlpha = 1.73455; LPFltBaseLen = 6.40; LPFltCutoffMult = 0.61314; } }; /** * @brief Set of resizing algorithm parameters for low-ringing performance * (7.86/1.065/0.000106). * * This set of resizing algorithm parameters offers a very low-ringing * performance at the expense of higher aliasing artifacts and a slightly * reduced contrast. */ struct CImageResizerParamsLR : public CImageResizerParams { CImageResizerParamsLR() { CorrFltAlpha = 1.0; // 7.86/1.96/1.065(73865.02)/0.000106:258636,437381 CorrFltLen = 5.87671; IntFltAlpha = 2.25322; IntFltCutoff = 0.74090; IntFltLen = 18.0; LPFltAlpha = 1.79306; LPFltBaseLen = 7.00; LPFltCutoffMult = 0.68881; } }; /** * @brief Set of resizing algorithm parameters for lower-ringing performance * (8.86/1.046/0.010168). * * This set of resizing algorithm parameters offers a lower-ringing * performance in comparison to the default setting, at the expense of higher * aliasing artifacts and a slightly reduced contrast. */ struct CImageResizerParamsLow : public CImageResizerParams { CImageResizerParamsLow() { CorrFltAlpha = 1.0; // 8.86/1.92/1.046(871.54)/0.010168:258647,442252 CorrFltLen = 6.09757; IntFltAlpha = 2.36704; IntFltCutoff = 0.74674; IntFltLen = 18.0; LPFltAlpha = 2.19427; LPFltBaseLen = 7.66; LPFltCutoffMult = 0.75380; } }; /** * @brief Set of resizing algorithm parameters for low-aliasing * resizing (11.81/1.012/0.038379). * * This set of resizing algorithm parameters offers a considerable * anti-aliasing performance with a good frequency response linearity (and * contrast). This is an intermediate setting between the default and Ultra * parameters. */ struct CImageResizerParamsHigh : public CImageResizerParams { CImageResizerParamsHigh() { CorrFltAlpha = 1.0; // 11.81/1.83/1.012(307.84)/0.038379:258660,452719 CorrFltLen = 6.80909; IntFltAlpha = 2.44917; IntFltCutoff = 0.75856; IntFltLen = 18.0; LPFltAlpha = 4.39527; LPFltBaseLen = 8.18; LPFltCutoffMult = 0.79172; } }; /** * @brief Set of resizing algorithm parameters for ultra low-aliasing * resizing (13.65/1.001/0.000483). * * This set of resizing algorithm parameters offers a very considerable * anti-aliasing performance with a good frequency response linearity (and * contrast). This set of parameters is computationally expensive and may * produce ringing artifacts on sharp features. */ struct CImageResizerParamsUltra : public CImageResizerParams { CImageResizerParamsUltra() { CorrFltAlpha = 1.0; // 13.65/1.79/1.001(28288.41)/0.000483:258658,457974 CorrFltLen = 7.48060; IntFltAlpha = 1.93750; IntFltCutoff = 0.75462; IntFltLen = 18.0; LPFltAlpha = 5.55209; LPFltBaseLen = 8.34; LPFltCutoffMult = 0.78002; } }; /** * @brief Image resizing variables class. * * This is an utility "catch all" class that defines various variables used * during image resizing. Several variables that are explicitly initialized in * this class' constructor are also used as additional "input" variables to * the image resizing function. These variables will not be changed by the * avir::CImageResizer<>::resizeImage() function. */ class CImageResizerVars { public: int ElCount; ///< The number of "fptype" elements used to store 1 pixel. ///< int ElCountIO; ///< The number of source and destination image's elements ///< used to store 1 pixel. ///< int fppack; ///< The number of atomic types stored in a single "fptype" ///< element. ///< int fpalign; ///< Suggested alignment size in bytes. This is not a ///< required alignment, because image resizing algorithm cannot ///< be made to have a strictly aligned data access in all cases ///< (e.g. de-interleaved interpolation cannot perform aligned ///< accesses). ///< int elalign; ///< Length alignment of arrays of elements. This applies to ///< filters and intermediate buffers: this constant forces ///< filters and scanlines to have a length which is a multiple ///< of this value, for more efficient SIMD implementation. ///< int packmode; ///< 0 if interleaved packing, 1 if de-interleaved. ///< int BufLen[2]; ///< Intermediate buffers' lengths in "fptype" elements. int BufOffs[2]; ///< Offsets into the intermediate buffers, used to ///< provide prefix elements required during processing so ///< that no "out of range" access happens. This offset is a ///< multiple of ElCount if pixels are stored in interleaved ///< form. ///< double k; ///< Resizing step coefficient, updated to reflect the actually ///< used coefficient during resizing. ///< double o; ///< Starting pixel offset inside the source image, updated to ///< reflect the actually used offset during resizing. ///< int ResizeStep; ///< Index of the resizing step in the latest filtering ///< steps array. ///< double InGammaMult; ///< Input gamma multiplier, used to convert input ///< data to 0 to 1 range. 0.0 if no gamma is in use. ///< double OutGammaMult; ///< Output gamma multiplier, used to convert data to ///< 0 to 255/65535 range. 0.0 if no gamma is in use. ///< double ox; ///< Start X pixel offset within source image (can be ///< negative). Positive offset moves image to the left. ///< double oy; ///< Start Y pixel offset within source image (can be ///< negative). Positive offset moves image to the top. ///< CImageResizerThreadPool* ThreadPool; ///< Thread pool to be used by the ///< image resizing function. Set to NULL to use ///< single-threaded processing. ///< bool UseSRGBGamma; ///< Perform sRGB gamma linearization (correction). ///< int BuildMode; ///< The build mode to use, for debugging purposes. Set to ///< -1 to select a minimal-complexity mode automatically. All ///< build modes deliver similar results with minor ///< deviations. ///< int RndSeed; ///< Random seed parameter. This parameter may be incremented ///< after each random generator initialization. The use of this ///< variable depends on the ditherer implementation. ///< CImageResizerVars() : ox(0.0), oy(0.0), ThreadPool(NULL), UseSRGBGamma(false), BuildMode(-1), RndSeed(0) {} }; /** * @brief Image resizer's filtering step class. * * Class defines data to perform a single filtering step over a whole * horizontal or vertical scanline. Resizing consists of 1 or more steps that * may be performed before the actual resizing takes place. Filtering may also * follow a resizing step. Each step must ensure that scanline data contains * enough pixels to perform the next step (which may be resizing) without * exceeding scanline's bounds. * * A derived class must implement several "const" and "static" functions that * are used to perform the actual filtering in interleaved or de-interleaved * mode. * * @tparam fptype Floating point type to use for storing pixel elements. SIMD * types can be used: in this case each element may hold a whole pixel. * @tparam fptypeatom The atomic type the "fptype" consists of. */ template class CImageResizerFilterStep { public: bool IsUpsample; ///< "True" if this step is an upsampling step, "false" ///< if downsampling step. Should be set to "false" if ///< ResampleFactor equals 0. ///< int ResampleFactor; ///< Resample factor (>=1). If 0, this is a resizing ///< step. This value should be >1 if IsUpsample equals ///< "true". ///< CBuffer Flt; ///< Filter to use at this step. ///< CFltBuffer FltOrig; ///< Originally-designed filter. This buffer may not ///< be assigned. Assigned by filters that precede the ///< resizing step if such filter is planned to be ///< embedded into the interpolation filter as "external" ///< filter. If IsUpsample=true and this filter buffer is ///< not empty, the upsampling step will not itself apply ///< any filtering over upsampled input scanline. ///< double DCGain; ///< DC gain which was applied to the filter. Not defined ///< if ResampleFactor = 0. ///< int FltLatency; ///< Filter's latency (group delay, shift) in pixels. ///< const CImageResizerVars* Vars; ///< Image resizing-related variables. ///< int InLen; ///< Input scanline's length in pixels. ///< int InBuf; ///< Input buffer index, 0 or 1. ///< int InPrefix; ///< Required input prefix pixels. These prefix pixels will ///< be filled with source scanline's first pixel value. If ///< IsUpsample is "true", this is the additional number of ///< times the first pixel will be filtered before processing ///< scanline, this number is also reflected in the OutPrefix. ///< int InSuffix; ///< Required input suffix pixels. These suffix pixels will ///< be filled with source scanline's last pixel value. If ///< IsUpsample is "true", this is the additional number of ///< times the last pixel will be filtered before processing ///< scanline, this number is also reflected in the OutSuffix. ///< int InElIncr; ///< Pixel element increment within the input buffer, used ///< during de-interleaved processing: in this case each ///< image's channel is stored independently, InElIncr elements ///< apart. ///< int OutLen; ///< Length of the resulting scanline. ///< int OutBuf; ///< Output buffer index. 0 or 1; 2 for the last step. ///< int OutPrefix; ///< Required output prefix pixels. These prefix pixels ///< will not be pre-filled with any values. Value is valid ///< only if IsUpsample equals "true". ///< int OutSuffix; ///< Required input suffix pixels. These suffix pixels will ///< not be pre-filled with any values. Value is valid only if ///< IsUpsample equals "true". ///< int OutElIncr; ///< Pixel element increment within the output buffer, used ///< during de-interleaved processing. Equals to the ///< InBufElIncr of the next step. ///< CBuffer PrefixDC; ///< DC component fluctuations added at the ///< start of the resulting scanline, used when ///< IsUpsample equals "true". ///< CBuffer SuffixDC; ///< DC component fluctuations added at the ///< end of the resulting scanline, used when ///< IsUpsample equals "true". ///< int EdgePixelCount; ///< The number of edge pixels added. Affects the ///< initial position within the input scanline, used to ///< produce edge pixels. This variable is used and ///< should be defined when IsUpsample=false and ///< ResampleFactor>0. When assigning this variable it is ///< also necessary to update InPrefix, OutLen and Vars.o ///< variables. ///< static const int EdgePixelCountDef = 3; ///< The default number of pixels ///< additionally produced at scanline edges during filtering. This is ///< required to reduce edge artifacts. ///< /** * @brief Resizing position structure. * * Structure holds resizing position and pointer to fractional delay * filter. */ struct CResizePos { int SrcPosInt; ///< Source scanline position. ///< int fti; ///< Fractional delay filter index. ///< const fptype* ftp; ///< Fractional delay filter pointer. ///< fptypeatom x; ///< Interpolation coefficient between delay filters. ///< int SrcOffs; ///< Source scanline offset. ///< }; /** * @brief Resizing positions buffer class. * * This class combines buffer together with variables that define resizing * stepping. */ class CRPosBuf : public CBuffer { public: double k; ///< Resizing step. ///< double o; ///< Resizing offset. ///< int FracCount; ///< The number of fractional delay filters in a filter ///< bank used together with this buffer. ///< }; /** * @brief Resizing positions buffer array class. * * This class combines structure array of the CRPosBuf class objects with * the function that locates or creates buffer with the required resizing * stepping. */ class CRPosBufArray : public CStructArray { public: using CStructArray::add; using CStructArray::getItemCount; /** * Function returns the resizing positions buffer with the required * stepping. If no such buffer exists, it is created. * * @param k Resizing step. * @param o Resizing offset. * @param FracCount The number of fractional delay filters in a filter * bank used together with this buffer. * @return Reference to the CRPosBuf object. */ CRPosBuf& getRPosBuf(const double k, const double o, const int FracCount) { int i; for (i = 0; i < getItemCount(); i++) { CRPosBuf& Buf = (*this)[i]; if (Buf.k == k && Buf.o == o && Buf.FracCount == FracCount) { return (Buf); } } CRPosBuf& NewBuf = add(); NewBuf.k = k; NewBuf.o = o; NewBuf.FracCount = FracCount; return (NewBuf); } }; CRPosBuf* RPosBuf; ///< Resizing positions buffer. Used when ///< ResampleFactor equals 0 (resizing step). ///< CDSPFracFilterBankLin* FltBank; ///< Filter bank in use by *this ///< resizing step. ///< }; /** * @brief Interleaved filtering steps implementation class. * * This class implements scanline filtering functions in interleaved mode. * This means that each pixel is processed independently, not in groups. * * @tparam fptype Floating point type to use for storing pixel elements. SIMD * types can be used: in this case each element may hold a whole pixel. * @tparam fptypeatom The atomic type the "fptype" consists of. */ template class CImageResizerFilterStepINL : public CImageResizerFilterStep { public: using CImageResizerFilterStep::IsUpsample; using CImageResizerFilterStep::ResampleFactor; using CImageResizerFilterStep::Flt; using CImageResizerFilterStep::FltOrig; using CImageResizerFilterStep::FltLatency; using CImageResizerFilterStep::Vars; using CImageResizerFilterStep::InLen; using CImageResizerFilterStep::InPrefix; using CImageResizerFilterStep::InSuffix; using CImageResizerFilterStep::OutLen; using CImageResizerFilterStep::OutPrefix; using CImageResizerFilterStep::OutSuffix; using CImageResizerFilterStep::PrefixDC; using CImageResizerFilterStep::SuffixDC; using CImageResizerFilterStep::RPosBuf; using CImageResizerFilterStep::FltBank; using CImageResizerFilterStep::EdgePixelCount; /** * Function performs "packing" of a scanline and type conversion. * Scanline, depending on the "fptype" can be potentially stored as a * packed SIMD values having a certain atomic type. If required, the sRGB * gamma correction is applied. * * @param ip Input scanline. * @param op0 Output scanline. * @param l0 The number of pixels to "pack". */ template void packScanline(const Tin* ip, fptype* const op0, const int l0) const { const int ElCount = Vars->ElCount; const int ElCountIO = Vars->ElCountIO; fptype* op = op0; int l = l0; if (!Vars->UseSRGBGamma) { if (ElCountIO == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; op += ElCount; ip++; l--; } } else if (ElCountIO == 4) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; v[1] = (fptypeatom)ip[1]; v[2] = (fptypeatom)ip[2]; v[3] = (fptypeatom)ip[3]; op += ElCount; ip += 4; l--; } } else if (ElCountIO == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; v[1] = (fptypeatom)ip[1]; v[2] = (fptypeatom)ip[2]; op += ElCount; ip += 3; l--; } } else if (ElCountIO == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; v[1] = (fptypeatom)ip[1]; op += ElCount; ip += 2; l--; } } } else { const fptypeatom gm = (fptypeatom)Vars->InGammaMult; if (ElCountIO == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); op += ElCount; ip++; l--; } } else if (ElCountIO == 4) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); v[1] = convertSRGB2Lin((fptypeatom)ip[1] * gm); v[2] = convertSRGB2Lin((fptypeatom)ip[2] * gm); v[3] = convertSRGB2Lin((fptypeatom)ip[3] * gm); op += ElCount; ip += 4; l--; } } else if (ElCountIO == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); v[1] = convertSRGB2Lin((fptypeatom)ip[1] * gm); v[2] = convertSRGB2Lin((fptypeatom)ip[2] * gm); op += ElCount; ip += 3; l--; } } else if (ElCountIO == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); v[1] = convertSRGB2Lin((fptypeatom)ip[1] * gm); op += ElCount; ip += 2; l--; } } } const int ZeroCount = ElCount * Vars->fppack - ElCountIO; op = op0; l = l0; if (ZeroCount == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)op + ElCountIO; v[0] = (fptypeatom)0; op += ElCount; l--; } } else if (ZeroCount == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)op + ElCountIO; v[0] = (fptypeatom)0; v[1] = (fptypeatom)0; op += ElCount; l--; } } else if (ZeroCount == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)op + ElCountIO; v[0] = (fptypeatom)0; v[1] = (fptypeatom)0; v[2] = (fptypeatom)0; op += ElCount; l--; } } } /** * Function applies Linear to sRGB gamma correction to the specified * scanline. * * @param p Scanline. * @param l The number of pixels to de-linearize. * @param Vars0 Image resizing-related variables. */ static void applySRGBGamma(fptype* p, int l, const CImageResizerVars& Vars0) { const int ElCount = Vars0.ElCount; const int ElCountIO = Vars0.ElCountIO; const fptypeatom gm = (fptypeatom)Vars0.OutGammaMult; if (ElCountIO == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; p += ElCount; l--; } } else if (ElCountIO == 4) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; v[1] = convertLin2SRGB(v[1]) * gm; v[2] = convertLin2SRGB(v[2]) * gm; v[3] = convertLin2SRGB(v[3]) * gm; p += ElCount; l--; } } else if (ElCountIO == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; v[1] = convertLin2SRGB(v[1]) * gm; v[2] = convertLin2SRGB(v[2]) * gm; p += ElCount; l--; } } else if (ElCountIO == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; v[1] = convertLin2SRGB(v[1]) * gm; p += ElCount; l--; } } } /** * Function converts vertical scanline to horizontal scanline. This * function is called by the image resizer when image is resized * vertically. This means that the vertical scanline is stored in the * same format produced by the packScanline() and maintained by other * filtering functions. * * @param ip Input vertical scanline. * @param op Output buffer (temporary buffer used during resizing). * @param SrcLen The number of pixels in the input scanline, also used to * calculate input buffer increment. * @param SrcIncr Input buffer increment to the next vertical pixel. */ void convertVtoH(const fptype* ip, fptype* op, const int SrcLen, const int SrcIncr) const { const int ElCount = Vars->ElCount; int j; if (ElCount == 1) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; ip += SrcIncr; op++; } } else if (ElCount == 4) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; op[3] = ip[3]; ip += SrcIncr; op += 4; } } else if (ElCount == 3) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; ip += SrcIncr; op += 3; } } else if (ElCount == 2) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; op[1] = ip[1]; ip += SrcIncr; op += 2; } } } /** * Function performs "unpacking" of a scanline and type conversion * (truncation is used when floating point is converted to integer). * Scanline, depending on the "fptype" can be potentially stored as a * packed SIMD values having a certain atomic type. The unpacking function * assumes that scanline is stored in the style produced by the * packScanline() function. * * @param ip Input scanline. * @param op Output scanline. * @param l The number of pixels to "unpack". * @param Vars0 Image resizing-related variables. */ template static void unpackScanline(const fptype* ip, Tout* op, int l, const CImageResizerVars& Vars0) { const int ElCount = Vars0.ElCount; const int ElCountIO = Vars0.ElCountIO; if (ElCountIO == 1) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; ip += ElCount; op++; l--; } } else if (ElCountIO == 4) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; op[1] = (Tout)v[1]; op[2] = (Tout)v[2]; op[3] = (Tout)v[3]; ip += ElCount; op += 4; l--; } } else if (ElCountIO == 3) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; op[1] = (Tout)v[1]; op[2] = (Tout)v[2]; ip += ElCount; op += 3; l--; } } else if (ElCountIO == 2) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; op[1] = (Tout)v[1]; ip += ElCount; op += 2; l--; } } } /** * Function prepares input scanline buffer for *this filtering step. * Left- and right-most pixels are replicated to make sure no buffer * overrun happens. Such approach also allows to bypass any pointer * range checks. * * @param Src Source buffer. */ void prepareInBuf(fptype* Src) const { if (IsUpsample || InPrefix + InSuffix == 0) { return; } const int ElCount = Vars->ElCount; replicateArray(Src, ElCount, Src - ElCount, InPrefix, -ElCount); Src += (InLen - 1) * ElCount; replicateArray(Src, ElCount, Src + ElCount, InSuffix, ElCount); } /** * Function peforms scanline upsampling with filtering. * * @param Src Source scanline buffer (length = this -> InLen). Source * scanline increment will be equal to ElCount. * @param Dst Destination scanline buffer. */ void doUpsample(const fptype* const Src, fptype* const Dst) const { const int ElCount = Vars->ElCount; fptype* op0 = &Dst[-OutPrefix * ElCount]; memset(&op0->value, 0, (OutPrefix + OutLen + OutSuffix) * ElCount * sizeof(op0->value)); const fptype* ip = Src; const int opstep = ElCount * ResampleFactor; int l; if (FltOrig.getCapacity() > 0) { // Do not perform filtering, only upsample. op0 += (OutPrefix % ResampleFactor) * ElCount; l = OutPrefix / ResampleFactor; if (ElCount == 1) { while (l > 0) { op0[0] = ip[0]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0 += opstep; l--; } } else if (ElCount == 4) { while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0[3] = ip[3]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0[3] = ip[3]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0[3] = ip[3]; op0 += opstep; l--; } } else if (ElCount == 3) { while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0 += opstep; l--; } } else if (ElCount == 2) { while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0 += opstep; l--; } } return; } const fptype* const f = Flt; const int flen = Flt.getCapacity(); fptype* op; int i; if (ElCount == 1) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[i] += f[i] * ip[0]; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[i] += f[i] * ip[0]; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[i] += f[i] * ip[0]; } op0 += opstep; l--; } } else if (ElCount == 4) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op[3] += f[i] * ip[3]; op += 4; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op[3] += f[i] * ip[3]; op += 4; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op[3] += f[i] * ip[3]; op += 4; } op0 += opstep; l--; } } else if (ElCount == 3) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op += 3; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op += 3; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op += 3; } op0 += opstep; l--; } } else if (ElCount == 2) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op += 2; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op += 2; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op += 2; } op0 += opstep; l--; } } op = op0; const fptype* dc = SuffixDC; l = SuffixDC.getCapacity(); if (ElCount == 1) { for (i = 0; i < l; i++) { op[i] += ip[0] * dc[i]; } } else if (ElCount == 4) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; op[3] += ip[3] * dc[0]; dc++; op += 4; l--; } } else if (ElCount == 3) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; dc++; op += 3; l--; } } else if (ElCount == 2) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; dc++; op += 2; l--; } } ip = Src; op = Dst - InPrefix * opstep; dc = PrefixDC; l = PrefixDC.getCapacity(); if (ElCount == 1) { for (i = 0; i < l; i++) { op[i] += ip[0] * dc[i]; } } else if (ElCount == 4) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; op[3] += ip[3] * dc[0]; dc++; op += 4; l--; } } else if (ElCount == 3) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; dc++; op += 3; l--; } } else if (ElCount == 2) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; dc++; op += 2; l--; } } } /** * Function peforms scanline filtering with optional downsampling. * Function makes use of the symmetry of the filter. * * @param Src Source scanline buffer (length = this -> InLen). Source * scanline increment will be equal to ElCount. * @param Dst Destination scanline buffer. * @param DstIncr Destination scanline buffer increment, used for * horizontal or vertical scanline stepping. */ void doFilter(const fptype* const Src, fptype* Dst, const int DstIncr) const { const int ElCount = Vars->ElCount; const fptype* const f = &Flt[FltLatency]; const int flen = FltLatency + 1; const int ipstep = ElCount * ResampleFactor; const fptype* ip = Src - EdgePixelCount * ipstep; const fptype* ip1; const fptype* ip2; int l = OutLen; int i; if (ElCount == 1) { while (l > 0) { fptype s = f[0] * ip[0]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1++; ip2--; s += f[i] * (ip1[0] + ip2[0]); } Dst[0] = s; Dst += DstIncr; ip += ipstep; l--; } } else if (ElCount == 4) { while (l > 0) { fptype s1 = f[0] * ip[0]; fptype s2 = f[0] * ip[1]; fptype s3 = f[0] * ip[2]; fptype s4 = f[0] * ip[3]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1 += 4; ip2 -= 4; s1 += f[i] * (ip1[0] + ip2[0]); s2 += f[i] * (ip1[1] + ip2[1]); s3 += f[i] * (ip1[2] + ip2[2]); s4 += f[i] * (ip1[3] + ip2[3]); } Dst[0] = s1; Dst[1] = s2; Dst[2] = s3; Dst[3] = s4; Dst += DstIncr; ip += ipstep; l--; } } else if (ElCount == 3) { while (l > 0) { fptype s1 = f[0] * ip[0]; fptype s2 = f[0] * ip[1]; fptype s3 = f[0] * ip[2]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1 += 3; ip2 -= 3; s1 += f[i] * (ip1[0] + ip2[0]); s2 += f[i] * (ip1[1] + ip2[1]); s3 += f[i] * (ip1[2] + ip2[2]); } Dst[0] = s1; Dst[1] = s2; Dst[2] = s3; Dst += DstIncr; ip += ipstep; l--; } } else if (ElCount == 2) { while (l > 0) { fptype s1 = f[0] * ip[0]; fptype s2 = f[0] * ip[1]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1 += 2; ip2 -= 2; s1 += f[i] * (ip1[0] + ip2[0]); s2 += f[i] * (ip1[1] + ip2[1]); } Dst[0] = s1; Dst[1] = s2; Dst += DstIncr; ip += ipstep; l--; } } } /** * Function performs resizing of a single scanline. This function does * not "know" about the length of the source scanline buffer. This buffer * should be padded with enough pixels so that ( SrcPos - FilterLenD2 ) is * always >= 0 and ( SrcPos + ( DstLineLen - 1 ) * k + FilterLenD2 + 1 ) * does not exceed source scanline's buffer length. SrcLine's increment is * assumed to be equal to ElCount. * * @param SrcLine Source scanline buffer. * @param DstLine Destination (resized) scanline buffer. * @param DstLineIncr Destination scanline position increment, used for * horizontal or vertical scanline stepping. * @param xx Temporary buffer, of size FltBank -> getFilterLen(), must be * aligned by fpclass :: fpalign. */ void doResize(const fptype* SrcLine, fptype* DstLine, const int DstLineIncr, fptype* const) const { const int IntFltLen = FltBank->getFilterLen(); const int ElCount = Vars->ElCount; const typename CImageResizerFilterStep::CResizePos* rpos = &(*RPosBuf)[0]; const typename CImageResizerFilterStep< fptype, fptypeatom>::CResizePos* const rpose = rpos + OutLen; #define AVIR_RESIZE_PART1 \ while (rpos < rpose) { \ const fptype x = (fptype)rpos->x; \ const fptype* const ftp = rpos->ftp; \ const fptype* const ftp2 = ftp + IntFltLen; \ const fptype* Src = SrcLine + rpos->SrcOffs; \ int i; #define AVIR_RESIZE_PART1nx \ while (rpos < rpose) { \ const fptype* const ftp = rpos->ftp; \ const fptype* Src = SrcLine + rpos->SrcOffs; \ int i; #define AVIR_RESIZE_PART2 \ DstLine += DstLineIncr; \ rpos++; \ } if (FltBank->getOrder() == 1) { if (ElCount == 1) { AVIR_RESIZE_PART1 fptype sum = 0.0; for (i = 0; i < IntFltLen; i++) { sum += (ftp[i] + ftp2[i] * x) * Src[i]; } DstLine[0] = sum; AVIR_RESIZE_PART2 } else if (ElCount == 4) { AVIR_RESIZE_PART1 fptype sum[4]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; sum[3] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i] + ftp2[i] * x; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; sum[3] += xx * Src[3]; Src += 4; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; DstLine[3] = sum[3]; AVIR_RESIZE_PART2 } else if (ElCount == 3) { AVIR_RESIZE_PART1 fptype sum[3]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i] + ftp2[i] * x; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; Src += 3; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; AVIR_RESIZE_PART2 } else if (ElCount == 2) { AVIR_RESIZE_PART1 fptype sum[2]; sum[0] = 0.0; sum[1] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i] + ftp2[i] * x; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; Src += 2; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; AVIR_RESIZE_PART2 } } else { if (ElCount == 1) { AVIR_RESIZE_PART1nx fptype sum = 0.0; for (i = 0; i < IntFltLen; i++) { sum += ftp[i] * Src[i]; } DstLine[0] = sum; AVIR_RESIZE_PART2 } else if (ElCount == 4) { AVIR_RESIZE_PART1nx fptype sum[4]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; sum[3] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i]; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; sum[3] += xx * Src[3]; Src += 4; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; DstLine[3] = sum[3]; AVIR_RESIZE_PART2 } else if (ElCount == 3) { AVIR_RESIZE_PART1nx fptype sum[3]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i]; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; Src += 3; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; AVIR_RESIZE_PART2 } else if (ElCount == 2) { AVIR_RESIZE_PART1nx fptype sum[2]; sum[0] = 0.0; sum[1] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i]; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; Src += 2; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; AVIR_RESIZE_PART2 } } } #undef AVIR_RESIZE_PART2 #undef AVIR_RESIZE_PART1nx #undef AVIR_RESIZE_PART1 }; /** * @brief Image resizer's default dithering class. * * This class defines an object that performs rounding, clipping and dithering * operations over horizontal scanline pixels before scanline is stored in the * output buffer. * * The ditherer should expect the same storage order of the pixels in a * scanline as used in the "filtering step" class. So, a separate ditherer * class should be defined for each scanline pixel storage style. The default * ditherer implements a simple rounding without dithering: it can be used for * an efficient dithering method which can be multi-threaded. * * @tparam fptype Floating point type to use for storing pixel data. SIMD * types can be used. */ template class CImageResizerDithererDefINL { public: /** * Function initializes the ditherer object. * * @param aLen Scanline length in pixels to process. * @param aVars Image resizing-related variables. * @param aTrMul Bit-depth truncation multiplier. 1 - no additional * truncation. * @param aPkOut Peak output value allowed. */ void init(const int aLen, const CImageResizerVars& aVars, const double aTrMul, const double aPkOut) { Len = aLen; Vars = &aVars; LenE = aLen * Vars->ElCount; TrMul0 = aTrMul; PkOut0 = aPkOut; } /** * @return "True" if dithering is recursive relative to scanlines meaning * multi-threaded execution is not supported by this dithering method. */ static bool isRecursive() { return (false); } /** * Function performs rounding and clipping operations. * * @param ResScanline The buffer containing the final scanline. */ void dither(fptype* const ResScanline) const { const fptype c0 = 0.0; const fptype PkOut = (fptype)PkOut0; int j; if (TrMul0 == 1.0) { // Optimization - do not perform bit depth truncation. for (j = 0; j < LenE; j++) { ResScanline[j] = clamp(round(ResScanline[j]), c0, PkOut); } } else { const fptype TrMul = (fptype)TrMul0; for (j = 0; j < LenE; j++) { const fptype z0 = round(ResScanline[j] / TrMul) * TrMul; ResScanline[j] = clamp(z0, c0, PkOut); } } } protected: int Len; ///< Scanline's length in pixels. ///< const CImageResizerVars* Vars; ///< Image resizing-related variables. ///< int LenE; ///< = LenE * ElCount. ///< double TrMul0; ///< Bit-depth truncation multiplier. ///< double PkOut0; ///< Peak output value allowed. ///< }; /** * @brief Image resizer's error-diffusion dithering class, interleaved mode. * * This ditherer implements error-diffusion dithering which looks good, and * whose results are compressed by PNG well. This implementation uses * weighting coefficients obtained via machine optimization and visual * evaluation. * * @tparam fptype Floating point type to use for storing pixel data. SIMD * types can be used. */ template class CImageResizerDithererErrdINL : public CImageResizerDithererDefINL { public: /** * Function initializes the ditherer object. * * @param aLen Scanline length in pixels to process. * @param aVars Image resizing-related variables. * @param aTrMul Bit-depth truncation multiplier. 1 - no additional * truncation. * @param aPkOut Peak output value allowed. */ void init(const int aLen, const CImageResizerVars& aVars, const double aTrMul, const double aPkOut) { CImageResizerDithererDefINL::init(aLen, aVars, aTrMul, aPkOut); ResScanlineDith0.alloc(LenE + Vars->ElCount, sizeof(fptype)); ResScanlineDith = ResScanlineDith0 + Vars->ElCount; int i; for (i = 0; i < LenE + Vars->ElCount; i++) { ResScanlineDith0[i] = 0.0; } } static bool isRecursive() { return (true); } void dither(fptype* const ResScanline) { const int ElCount = Vars->ElCount; const fptype c0 = 0.0; const fptype TrMul = (fptype)TrMul0; const fptype PkOut = (fptype)PkOut0; int j; for (j = 0; j < LenE; j++) { ResScanline[j] += ResScanlineDith[j]; ResScanlineDith[j] = 0.0; } for (j = 0; j < LenE - ElCount; j++) { // Perform rounding, noise estimation and saturation. const fptype z0 = round(ResScanline[j] / TrMul) * TrMul; const fptype Noise = ResScanline[j] - z0; ResScanline[j] = clamp(z0, c0, PkOut); ResScanline[j + ElCount] += Noise * (fptype)0.364842; ResScanlineDith[j - ElCount] += Noise * (fptype)0.207305; ResScanlineDith[j] += Noise * (fptype)0.364842; ResScanlineDith[j + ElCount] += Noise * (fptype)0.063011; } while (j < LenE) { const fptype z0 = round(ResScanline[j] / TrMul) * TrMul; const fptype Noise = ResScanline[j] - z0; ResScanline[j] = clamp(z0, c0, PkOut); ResScanlineDith[j - ElCount] += Noise * (fptype)0.207305; ResScanlineDith[j] += Noise * (fptype)0.364842; j++; } } protected: using CImageResizerDithererDefINL::Len; using CImageResizerDithererDefINL::Vars; using CImageResizerDithererDefINL::LenE; using CImageResizerDithererDefINL::TrMul0; using CImageResizerDithererDefINL::PkOut0; CBuffer ResScanlineDith0; ///< Error diffusion buffer. ///< fptype* ResScanlineDith; ///< Error diffusion buffer pointer which skips ///< the first ElCount elements. ///< }; /** * @brief Floating-point processing definition and abstraction class. * * This class defines several constants and typedefs that point to classes * that should be used by the image resizing algorithm. Such "definition * class" can be used to define alternative scanline processing algorithms * (e.g. SIMD) and image scanline packing styles used during processing. This * class also offers an abstraction layer for dithering, rounding and * clamping (saturation) operation. * * The fpclass_def class can be used to define processing using both SIMD and * non-SIMD types, but using algorithms that are operate on interleaved pixels * and non-SIMD optimized themselves. * * @tparam afptype Floating point type to use for storing intermediate data * and variables. For variables that are not used in intensive calculations * the "double" type is always used. On the latest Intel processors (like * i7-4770K) there is almost no performance difference between "double" and * "float". Image quality differences between "double" and "float" are not * apparent on 8-bit images. At the same time the "float" uses half amount of * working memory the "double" type uses. SIMD types can be used. The * functions round() and clamp() in the "avir" or other visible namespace * should be available for the specified type. SIMD types allow to perform * resizing of images with more than 4 channels, to be exact 4 * SIMD element * number (e.g. 16 for float4), without modification of the image resizing * algorithm required. * @tparam afptypeatom The atomic type the "afptype" consists of. * @tparam adith Ditherer class to use during processing. */ template > class fpclass_def { public: typedef afptype fptype; ///< Floating-point type to use during processing. ///< typedef afptypeatom fptypeatom; ///< Atomic type "fptype" consists of. ///< static const int fppack = sizeof(fptype) / sizeof(fptypeatom); ///< ///< The number of atomic types stored in a single ///< "fptype" element. ///< static const int fpalign = sizeof(fptype); ///< Suggested alignment size ///< in bytes. This is not a required alignment, because ///< image resizing algorithm cannot be made to have a ///< strictly aligned data access at all steps (e.g. ///< interpolation cannot perform aligned accesses). ///< static const int elalign = 1; ///< Length alignment of arrays of elements. ///< This applies to filters and intermediate buffers: this constant ///< forces filters and scanlines to have a length which is a multiple ///< of this value, for more efficient SIMD implementation. ///< static const int packmode = 0; ///< 0 if interleaved packing, 1 if ///< de-interleaved. ///< typedef CImageResizerFilterStepINL CFilterStep; ///< ///< Filtering step class to use during processing. ///< typedef adith CDitherer; ///< Ditherer class to use during processing. ///< }; /** * @brief Image resizer class. * * The object of this class can be used to resize 1-4 channel images to any * required size. Resizing is performed by utilizing interpolated sinc * fractional delay filters plus (if necessary) a cascade of built-in * sinc function-based 2X upsampling or 2X downsampling stages, followed by a * correction filtering. * * Object of this class can be allocated on stack. * * @tparam fpclass Floating-point processing definition class to use. See * avir::fpclass_def for more details. */ template > class CImageResizer { public: /** * Constructor initializes the resizer. * * @param aResBitDepth Required bit depth of resulting image (1-16). If * integer value output is used (e.g. uint8_t), the bit depth also affects * rounding: for example, if aResBitDepth=6 and "Tout" is uint8_t, the * result will be rounded to 6 most significant bits (2 least significant * bits truncated, with dithering applied). * @param aSrcBitDepth Source image's real bit-depth. Set to 0 to use * aResBitDepth. * @param aParams Resizing algorithm's parameters to use. Leave out for * default values. Can be useful when performing automatic optimization of * parameters. */ CImageResizer(const int aResBitDepth = 8, const int aSrcBitDepth = 0, const CImageResizerParams& aParams = CImageResizerParamsDef()) : Params(aParams), ResBitDepth(aResBitDepth) { SrcBitDepth = (aSrcBitDepth == 0 ? ResBitDepth : aSrcBitDepth); initFilterBank(FixedFilterBank, 1.0, false, CFltBuffer()); FixedFilterBank.createAllFilters(); } /** * Function resizes image. * * @param SrcBuf Source image buffer. * @param SrcWidth Source image width. * @param SrcHeight Source image height. * @param SrcScanlineSize Physical size of source scanline in elements * (not bytes). If this value is below 1, SrcWidth * ElCountIO will be * used as the physical source scanline size. * @param[out] NewBuf Buffer to accept the resized image. Can be equal to * SrcBuf if the size of the resized image is smaller or equal to source * image in size. * @param NewWidth New image width. * @param NewHeight New image height. * @param ElCountIO The number of elements (channels) used to store each * source and destination pixel (1-4). * @param k Resizing step (one output pixel corresponds to "k" input * pixels). A downsizing factor if > 1.0; upsizing factor if <= 1.0. * Multiply by -1 if you would like to bypass "ox" and "oy" adjustment * which is done by default to produce a centered image. If step value * equals 0, the step value will be chosen automatically and independently * for horizontal and vertical resizing. * @param[in,out] aVars Pointer to variables structure to be passed to the * image resizing function. Can be NULL. Only variables that are * initialized in default constructor of this structure are accepted by * this function. These variables will not be changed by this function. * All other variables can be modified by this function. The access to * this object is not thread-safe, each concurrent instance of this * function should use a separate aVars object. * @tparam Tin Input buffer element's type. Can be uint8_t (0-255 value * range), uint16_t (0-65535 value range), float (0.0-1.0 value range), * double (0.0-1.0 value range). Larger integer types are treated as * uint16_t. Signed integer types are unsupported. * @tparam Tout Output buffer element's type. Can be uint8_t (0-255 value * range), uint16_t (0-65535 value range), float (0.0-1.0 value range), * double (0.0-1.0 value range). Larger integer types are treated as * uint16_t. Signed integer types are unsupported. */ template void resizeImage(const Tin* const SrcBuf, const int SrcWidth, const int SrcHeight, int SrcScanlineSize, Tout* const NewBuf, const int NewWidth, const int NewHeight, const int ElCountIO, const double k, CImageResizerVars* const aVars = NULL) const { if (SrcWidth == 0 || SrcHeight == 0) { memset(NewBuf, 0, (size_t)NewWidth * NewHeight * sizeof(Tout)); return; } else if (NewWidth == 0 || NewHeight == 0) { return; } CImageResizerVars DefVars; CImageResizerVars& Vars = (aVars == NULL ? DefVars : *aVars); CImageResizerThreadPool DefThreadPool; CImageResizerThreadPool& ThreadPool = (Vars.ThreadPool == NULL ? DefThreadPool : *Vars.ThreadPool); // Define resizing steps, also optionally modify offsets so that // resizing produces a "centered" image. double kx; double ky; double ox = Vars.ox; double oy = Vars.oy; if (k == 0.0) { if (NewWidth > SrcWidth) { kx = (double)(SrcWidth - 1) / (NewWidth - 1); } else { kx = (double)SrcWidth / NewWidth; ox += (kx - 1.0) * 0.5; } if (NewHeight > SrcHeight) { ky = (double)(SrcHeight - 1) / (NewHeight - 1); } else { ky = (double)SrcHeight / NewHeight; oy += (ky - 1.0) * 0.5; } } else if (k > 0.0) { kx = k; ky = k; if (k > 1.0) { const double ko = (k - 1.0) * 0.5; ox += ko; oy += ko; } } else { kx = -k; ky = -k; } // Evaluate pre-multipliers used on the output stage. const bool IsInFloat = ((Tin)0.4 != 0); const bool IsOutFloat = ((Tout)0.4 != 0); double OutMul; // Output multiplier. if (Vars.UseSRGBGamma) { if (IsInFloat) { Vars.InGammaMult = 1.0; } else { Vars.InGammaMult = 1.0 / (sizeof(Tin) == 1 ? 255.0 : 65535.0); } if (IsOutFloat) { Vars.OutGammaMult = 1.0; } else { Vars.OutGammaMult = (sizeof(Tout) == 1 ? 255.0 : 65535.0); } OutMul = 1.0; } else { if (IsOutFloat) { OutMul = 1.0; } else { OutMul = (sizeof(Tout) == 1 ? 255.0 : 65535.0); } if (!IsInFloat) { OutMul /= (sizeof(Tin) == 1 ? 255.0 : 65535.0); } } // Fill widely-used variables. const int ElCount = (ElCountIO + fpclass ::fppack - 1) / fpclass ::fppack; const int NewWidthE = NewWidth * ElCount; if (SrcScanlineSize < 1) { SrcScanlineSize = SrcWidth * ElCountIO; } Vars.ElCount = ElCount; Vars.ElCountIO = ElCountIO; Vars.fppack = fpclass ::fppack; Vars.fpalign = fpclass ::fpalign; Vars.elalign = fpclass ::elalign; Vars.packmode = fpclass ::packmode; // Horizontal scanline filtering and resizing. CDSPFracFilterBankLin FltBank; CFilterSteps FltSteps; typename CFilterStep ::CRPosBufArray RPosBufArray; CBuffer UsedFracMap; // Perform the filtering steps modeling at various modes, find the // most efficient mode for both horizontal and vertical resizing. int UseBuildMode = 1; const int BuildModeCount = (FixedFilterBank.getOrder() == 0 ? 4 : 2); int m; if (Vars.BuildMode >= 0) { UseBuildMode = Vars.BuildMode; } else { int BestScore = 0x7FFFFFFF; for (m = 0; m < BuildModeCount; m++) { CDSPFracFilterBankLin TmpBank; CFilterSteps TmpSteps; Vars.k = kx; Vars.o = ox; buildFilterSteps(TmpSteps, Vars, TmpBank, OutMul, m, true); updateFilterStepBuffers(TmpSteps, Vars, RPosBufArray, SrcWidth, NewWidth); fillUsedFracMap(TmpSteps[Vars.ResizeStep], UsedFracMap); const int c = calcComplexity(TmpSteps, Vars, UsedFracMap, SrcHeight); if (c < BestScore) { UseBuildMode = m; BestScore = c; } } } // Perform the actual filtering steps building. Vars.k = kx; Vars.o = ox; buildFilterSteps(FltSteps, Vars, FltBank, OutMul, UseBuildMode, false); updateFilterStepBuffers(FltSteps, Vars, RPosBufArray, SrcWidth, NewWidth); updateBufLenAndRPosPtrs(FltSteps, Vars, NewWidth); const int ThreadCount = ThreadPool.getSuggestedWorkloadCount(); // Includes the current thread. CStructArray > td; td.setItemCount(ThreadCount); int i; for (i = 0; i < ThreadCount; i++) { if (i > 0) { ThreadPool.addWorkload(&td[i]); } td[i].init(i, ThreadCount, FltSteps, Vars); td[i].initScanlineQueue(td[i].sopResizeH, SrcHeight, SrcWidth); } CBuffer FltBuf( (size_t)NewWidthE * SrcHeight, fpclass ::fpalign); // Temporary buffer that receives // horizontally-filtered and resized image. for (i = 0; i < SrcHeight; i++) { td[i % ThreadCount].addScanlineToQueue( (void*)&SrcBuf[(size_t)i * SrcScanlineSize], &FltBuf[(size_t)i * NewWidthE]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); // Vertical scanline filtering and resizing, reuse previously defined // filtering steps if possible. const int PrevUseBuildMode = UseBuildMode; if (Vars.BuildMode >= 0) { UseBuildMode = Vars.BuildMode; } else { CImageResizerVars TmpVars(Vars); int BestScore = 0x7FFFFFFF; for (m = 0; m < BuildModeCount; m++) { CDSPFracFilterBankLin TmpBank; TmpBank.copyInitParams(FltBank); CFilterSteps TmpSteps; TmpVars.k = ky; TmpVars.o = oy; buildFilterSteps(TmpSteps, TmpVars, TmpBank, 1.0, m, true); updateFilterStepBuffers(TmpSteps, TmpVars, RPosBufArray, SrcHeight, NewHeight); fillUsedFracMap(TmpSteps[TmpVars.ResizeStep], UsedFracMap); const int c = calcComplexity(TmpSteps, TmpVars, UsedFracMap, NewWidth); if (c < BestScore) { UseBuildMode = m; BestScore = c; } } } Vars.k = ky; Vars.o = oy; if (UseBuildMode == PrevUseBuildMode && ky == kx) { if (OutMul != 1.0) { modifyCorrFilterDCGain(FltSteps, 1.0 / OutMul); } } else { buildFilterSteps(FltSteps, Vars, FltBank, 1.0, UseBuildMode, false); } updateFilterStepBuffers(FltSteps, Vars, RPosBufArray, SrcHeight, NewHeight); updateBufLenAndRPosPtrs(FltSteps, Vars, NewWidth); if (IsOutFloat && sizeof(FltBuf[0]) == sizeof(Tout) && fpclass ::packmode == 0) { // In-place output. for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopResizeV, NewWidth, SrcHeight, NewWidthE, NewWidthE); } for (i = 0; i < NewWidth; i++) { td[i % ThreadCount].addScanlineToQueue( &FltBuf[(size_t)i * ElCount], (fptype*)&NewBuf[(size_t)i * ElCount]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); ThreadPool.removeAllWorkloads(); return; } CBuffer ResBuf((size_t)NewWidthE * NewHeight, fpclass ::fpalign); for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopResizeV, NewWidth, SrcHeight, NewWidthE, NewWidthE); } const int im = (fpclass ::packmode == 0 ? ElCount : 1); for (i = 0; i < NewWidth; i++) { td[i % ThreadCount].addScanlineToQueue(&FltBuf[(size_t)i * im], &ResBuf[(size_t)i * im]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); if (IsOutFloat) { // Perform output, but skip dithering. for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopUnpackH, NewHeight, NewWidth); } for (i = 0; i < NewHeight; i++) { td[i % ThreadCount].addScanlineToQueue( &ResBuf[(size_t)i * NewWidthE], &NewBuf[(size_t)i * NewWidth * ElCountIO]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); ThreadPool.removeAllWorkloads(); return; } // Perform output with dithering (for integer output only). int TruncBits; // The number of lower bits to truncate and dither. int OutRange; // Output range. if (sizeof(Tout) == 1) { TruncBits = 8 - ResBitDepth; OutRange = 255; } else { TruncBits = 16 - ResBitDepth; OutRange = 65535; } const double PkOut = OutRange; const double TrMul = (TruncBits > 0 ? PkOut / (OutRange >> TruncBits) : 1.0); if (CDitherer ::isRecursive()) { td[0].getDitherer().init(NewWidth, Vars, TrMul, PkOut); if (Vars.UseSRGBGamma) { for (i = 0; i < NewHeight; i++) { fptype* const ResScanline = &ResBuf[(size_t)i * NewWidthE]; CFilterStep ::applySRGBGamma(ResScanline, NewWidth, Vars); td[0].getDitherer().dither(ResScanline); CFilterStep ::unpackScanline( ResScanline, &NewBuf[(size_t)i * NewWidth * ElCountIO], NewWidth, Vars); } } else { for (i = 0; i < NewHeight; i++) { fptype* const ResScanline = &ResBuf[(size_t)i * NewWidthE]; td[0].getDitherer().dither(ResScanline); CFilterStep ::unpackScanline( ResScanline, &NewBuf[(size_t)i * NewWidth * ElCountIO], NewWidth, Vars); } } } else { for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopDitherAndUnpackH, NewHeight, NewWidth); td[i].getDitherer().init(NewWidth, Vars, TrMul, PkOut); } for (i = 0; i < NewHeight; i++) { td[i % ThreadCount].addScanlineToQueue( &ResBuf[(size_t)i * NewWidthE], &NewBuf[(size_t)i * NewWidth * ElCountIO]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); } ThreadPool.removeAllWorkloads(); } private: typedef typename fpclass ::fptype fptype; ///< Floating-point type to use ///< during processing. ///< typedef typename fpclass ::CFilterStep CFilterStep; ///< Filtering step ///< class to use during processing. ///< typedef typename fpclass ::CDitherer CDitherer; ///< Ditherer class to ///< use during processing. ///< CImageResizerParams Params; ///< Algorithm's parameters currently in use. ///< int SrcBitDepth; ///< Bit resolution of the source image. ///< int ResBitDepth; ///< Bit resolution of the resulting image. ///< CDSPFracFilterBankLin FixedFilterBank; ///< Fractional delay ///< filter bank with fixed characteristics, mainly for ///< upsizing cases. ///< /** * @brief Filtering steps array. * * The object of this class stores filtering steps together. */ typedef CStructArray CFilterSteps; /** * Function initializes the filter bank in the specified resizing step * according to the source and resulting image bit depths. * * @param FltBank Filter bank to initialize. * @param CutoffMult Cutoff multiplier, 0 to 1. 1 corresponds to 0.5pi * cutoff point. * @param ForceHiOrder "True" if a high-order interpolation should be * forced which requires considerably less resources for initialization. * @param ExtFilter External filter to apply to interpolation filter. */ void initFilterBank(CDSPFracFilterBankLin& FltBank, const double CutoffMult, const bool ForceHiOrder, const CFltBuffer& ExtFilter) const { const int IntBitDepth = (ResBitDepth > SrcBitDepth ? ResBitDepth : SrcBitDepth); const double SNR = -6.02 * (IntBitDepth + 3); int UseOrder; int FracCount; // The number of fractional delay filters sampled by // the filter bank. This variable affects the // signal-to-noise ratio at interpolation stage. // Theoretically, at UseOrder==1, 8-bit image resizing // requires 66.2 dB SNR or 11. 16-bit resizing requires // 114.4 dB SNR or 150. At UseOrder=0 the required number of // filters is exponentially higher. if (ForceHiOrder || IntBitDepth > 8) { UseOrder = 1; // -146 dB max FracCount = (int)ceil(0.23134052 * exp(-0.058062929 * SNR)); } else { UseOrder = 0; // -72 dB max FracCount = (int)ceil(0.33287686 * exp(-0.11334583 * SNR)); } if (FracCount < 2) { FracCount = 2; } FltBank.init(FracCount, UseOrder, Params.IntFltLen / CutoffMult, Params.IntFltCutoff * CutoffMult, Params.IntFltAlpha, ExtFilter, fpclass ::fpalign, fpclass ::elalign); } /** * Function allocates filter buffer taking "fpclass" alignments into * account. The allocated buffer may be larger than the requested size: in * this case the additional elements will be zeroed by this function. * * @param Flt Filter buffer. * @param ReqCapacity The required filter buffer's capacity. * @param IsModel "True" if filtering steps modeling is performed without * actual filter allocation. * @param FltExt If non-NULL this variable will receive the number of * elements the filter was extended by. */ static void allocFilter(CBuffer& Flt, const int ReqCapacity, const bool IsModel = false, int* const FltExt = NULL) { int UseCapacity = (ReqCapacity + fpclass ::elalign - 1) & ~(fpclass ::elalign - 1); int Ext = UseCapacity - ReqCapacity; if (FltExt != NULL) { *FltExt = Ext; } if (IsModel) { Flt.forceCapacity(UseCapacity); return; } Flt.alloc(UseCapacity, fpclass ::fpalign); while (Ext > 0) { Ext--; Flt[ReqCapacity + Ext] = 0.0; } } /** * Function assigns filter parameters to the specified filtering step * object. * * @param fs Filtering step to assign parameter to. This step cannot be * the last step if ResampleFactor greater than 1 was specified. * @param IsUpsample "True" if upsampling step. Should be set to "false" * if FltCutoff is negative. * @param ResampleFactor Resampling factor of this filter (>=1). * @param FltCutoff Filter cutoff point. This value will be divided by the * ResampleFactor if IsUpsample equals "true". If zero value was * specified, the "half-band" predefined filter will be created. In this * case the ResampleFactor will modify the filter cutoff point. * @param DCGain DC gain to apply to the filter. Assigned to filtering * step's DCGain variable. * @param UseFltOrig "True" if the originally-designed filter should be * left in filtering step's FltOrig buffer. Otherwise it will be freed. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ void assignFilterParams(CFilterStep& fs, const bool IsUpsample, const int ResampleFactor, const double FltCutoff, const double DCGain, const bool UseFltOrig, const bool IsModel) const { double FltAlpha; double Len2; double Freq; if (FltCutoff == 0.0) { const double m = 2.0 / ResampleFactor; FltAlpha = Params.HBFltAlpha; Len2 = 0.5 * Params.HBFltLen / m; Freq = AVIR_PI * Params.HBFltCutoff * m; } else { FltAlpha = Params.LPFltAlpha; Len2 = 0.25 * Params.LPFltBaseLen / FltCutoff; Freq = AVIR_PI * Params.LPFltCutoffMult * FltCutoff; } if (IsUpsample) { Len2 *= ResampleFactor; Freq /= ResampleFactor; fs.DCGain = DCGain * ResampleFactor; } else { fs.DCGain = DCGain; } fs.FltOrig.Len2 = Len2; fs.FltOrig.Freq = Freq; fs.FltOrig.Alpha = FltAlpha; fs.FltOrig.DCGain = fs.DCGain; CDSPPeakedCosineLPF w(Len2, Freq, FltAlpha); fs.IsUpsample = IsUpsample; fs.ResampleFactor = ResampleFactor; fs.FltLatency = w.fl2; int FltExt; // Filter's extension due to fpclass :: elalign. if (IsModel) { allocFilter(fs.Flt, w.FilterLen, true, &FltExt); if (UseFltOrig) { // Allocate a real buffer even in modeling mode since this // filter may be copied by the filter bank. fs.FltOrig.alloc(w.FilterLen); memset(&fs.FltOrig[0], 0, w.FilterLen * sizeof(fs.FltOrig[0])); } } else { fs.FltOrig.alloc(w.FilterLen); w.generateLPF(&fs.FltOrig[0], 1.0); optimizeFIRFilter(fs.FltOrig, fs.FltLatency); normalizeFIRFilter(&fs.FltOrig[0], fs.FltOrig.getCapacity(), fs.DCGain); allocFilter(fs.Flt, fs.FltOrig.getCapacity(), false, &FltExt); copyArray(&fs.FltOrig[0], &fs.Flt[0], fs.FltOrig.getCapacity()); if (!UseFltOrig) { fs.FltOrig.free(); } } if (IsUpsample) { int l = fs.Flt.getCapacity() - fs.FltLatency - ResampleFactor - FltExt; allocFilter(fs.PrefixDC, l, IsModel); allocFilter(fs.SuffixDC, fs.FltLatency, IsModel); if (IsModel) { return; } // Create prefix and suffix "tails" used during upsampling. const fptype* ip = &fs.Flt[fs.FltLatency + ResampleFactor]; copyArray(ip, &fs.PrefixDC[0], l); while (true) { ip += ResampleFactor; l -= ResampleFactor; if (l <= 0) { break; } addArray(ip, &fs.PrefixDC[0], l); } l = fs.FltLatency; fptype* op = &fs.SuffixDC[0]; copyArray(&fs.Flt[0], op, l); while (true) { op += ResampleFactor; l -= ResampleFactor; if (l <= 0) { break; } addArray(&fs.Flt[0], op, l); } } else if (!UseFltOrig) { fs.EdgePixelCount = fs.EdgePixelCountDef; } } /** * Function adds a correction filter that tries to achieve a linear * frequency response at all frequencies. The actual resulting response * may feature a slight damping of the highest frequencies since a * suitably short correction filter cannot fix steep high-frequency * damping. * * This function assumes that the resizing step is currently the last * step, even if it was not inserted yet: this allows placement of the * correction filter both before and after the resizing step. * * @param Steps Filtering steps. * @param bw Resulting bandwidth relative to the original bandwidth (which * is 1.0), usually 1/k. Should be <= 1.0. * @param IsPreCorrection "True" if the filtering step was already created * and it is first in the Steps array. "True" also adds edge pixels to * reduce edge artifacts. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ void addCorrectionFilter(CFilterSteps& Steps, const double bw, const bool IsPreCorrection, const bool IsModel) const { CFilterStep& fs = (IsPreCorrection ? Steps[0] : Steps.add()); fs.IsUpsample = false; fs.ResampleFactor = 1; fs.DCGain = 1.0; fs.EdgePixelCount = (IsPreCorrection ? fs.EdgePixelCountDef : 0); if (IsModel) { allocFilter( fs.Flt, CDSPFIREQ ::calcFilterLength(Params.CorrFltLen, fs.FltLatency), true); return; } const int BinCount = 65; // Frequency response bins to control. const int BinCount1 = BinCount - 1; double curbw = 1.0; // Bandwidth of the filter at the current step. int i; int j; double re; double im; CBuffer Bins(BinCount); // Adjustment introduced by all // steps at all frequencies of interest. for (j = 0; j < BinCount; j++) { Bins[j] = 1.0; } const int si = (IsPreCorrection ? 1 : 0); for (i = si; i < Steps.getItemCount() - (si ^ 1); i++) { const CFilterStep& fs = Steps[i]; if (fs.IsUpsample) { curbw *= fs.ResampleFactor; if (fs.FltOrig.getCapacity() > 0) { continue; } } const double dcg = 1.0 / fs.DCGain; // DC gain correction. const fptype* Flt; int FltLen; if (fs.ResampleFactor == 0) { Flt = fs.FltBank->getFilter(0); FltLen = fs.FltBank->getFilterLen(); } else { Flt = &fs.Flt[0]; FltLen = fs.Flt.getCapacity(); } // Calculate frequency response adjustment introduced by the // filter at this step, within the bounds of bandwidth of // interest. for (j = 0; j < BinCount; j++) { const double th = AVIR_PI * bw / curbw * j / BinCount1; calcFIRFilterResponse(Flt, FltLen, th, re, im); Bins[j] /= sqrt(re * re + im * im) * dcg; } if (!fs.IsUpsample && fs.ResampleFactor > 1) { curbw /= fs.ResampleFactor; } } // Calculate filter. CDSPFIREQ EQ; EQ.init(bw * 2.0, Params.CorrFltLen, BinCount, 0.0, bw, false, Params.CorrFltAlpha); fs.FltLatency = EQ.getFilterLatency(); CBuffer Filter(EQ.getFilterLength()); EQ.buildFilter(Bins, &Filter[0]); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); optimizeFIRFilter(Filter, fs.FltLatency); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); allocFilter(fs.Flt, Filter.getCapacity()); copyArray(&Filter[0], &fs.Flt[0], Filter.getCapacity()); // Print a theoretically achieved final frequency response at various // feature sizes (from DC to 1 pixel). Values above 255 means features // become brighter, values below 255 means features become dimmer. /* const double sbw = ( bw > 1.0 ? 1.0 / bw : 1.0 ); for( j = 0; j < BinCount; j++ ) { const double th = AVIR_PI * sbw * j / BinCount1; calcFIRFilterResponse( &fs.Flt[ 0 ], fs.Flt.getCapacity(), th, re, im ); printf( "%f\n", sqrt( re * re + im * im ) / Bins[ j ] * 255 ); } printf( "***\n" );*/ } /** * Function adds a sharpening filter if image is being upsized. Such * sharpening allows to spot interpolation filter's stop-band attenuation: * if attenuation is too weak, a "dark grid" and other artifacts may * become visible. * * It is assumed that 40 decibel stop-band attenuation should be * considered a required minimum: this allows application of (deliberately * strong) 64X sharpening without spotting any artifacts. * * @param Steps Filtering steps. * @param bw Resulting bandwidth relative to the original bandwidth (which * is 1.0), usually 1/k. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ static void addSharpenTest(CFilterSteps& Steps, const double bw, const bool IsModel) { if (bw <= 1.0) { return; } const double FltLen = 10.0 * bw; CFilterStep& fs = Steps.add(); fs.IsUpsample = false; fs.ResampleFactor = 1; fs.DCGain = 1.0; fs.EdgePixelCount = 0; if (IsModel) { allocFilter(fs.Flt, CDSPFIREQ ::calcFilterLength(FltLen, fs.FltLatency), true); return; } const int BinCount = 200; CBuffer Bins(BinCount); int Thresh = (int)round(BinCount / bw * 1.75); if (Thresh > BinCount) { Thresh = BinCount; } int j; for (j = 0; j < Thresh; j++) { Bins[j] = 1.0; } for (j = Thresh; j < BinCount; j++) { Bins[j] = 256.0; } CDSPFIREQ EQ; EQ.init(bw * 2.0, FltLen, BinCount, 0.0, bw, false, 1.7); fs.FltLatency = EQ.getFilterLatency(); CBuffer Filter(EQ.getFilterLength()); EQ.buildFilter(Bins, &Filter[0]); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); optimizeFIRFilter(Filter, fs.FltLatency); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); allocFilter(fs.Flt, Filter.getCapacity()); copyArray(&Filter[0], &fs.Flt[0], Filter.getCapacity()); /* for( j = 0; j < BinCount; j++ ) { const double th = AVIR_PI * j / ( BinCount - 1 ); double re; double im; calcFIRFilterResponse( &fs.Flt[ 0 ], fs.Flt.getCapacity(), th, re, im ); printf( "%f\n", sqrt( re * re + im * im )); } printf( "***\n" );*/ } /** * Function builds sequence of filtering steps depending on the specified * resizing coefficient. The last steps included are always the resizing * step then (possibly) the correction step. * * @param Steps Array that receives filtering steps. * @param[out] Vars Variables object. * @param FltBank Filter bank to initialize and use. * @param DCGain The overall DC gain to apply. This DC gain is applied to * the first filtering step only (upsampling or filtering step). * @param ModeFlags Build mode flags to use. This is a bitmap of switches * that enable or disable certain algorithm features. * @param IsModel "True" if filtering steps modeling is performed without * the actual filter allocation and building. */ void buildFilterSteps(CFilterSteps& Steps, CImageResizerVars& Vars, CDSPFracFilterBankLin& FltBank, const double DCGain, const int ModeFlags, const bool IsModel) const { Steps.clear(); const bool DoFltAndIntCombo = ((ModeFlags & 1) != 0); // Do filter // and interpolator combining. const bool ForceHiOrderInt = ((ModeFlags & 2) != 0); // Force use // of a higher-order interpolation. const bool UseHalfband = ((ModeFlags & 4) != 0); // Use half-band // filter. const double bw = 1.0 / Vars.k; // Resulting bandwidth. const int UpsampleFactor = ((int)floor(Vars.k) < 2 ? 2 : 1); double IntCutoffMult; // Interpolation filter cutoff multiplier. CFilterStep* ReuseStep; // If not NULL, resizing step should use // this step object instead of creating a new one. CFilterStep* ExtFltStep; // Use FltOrig of this step as the external // filter to applied to the interpolator. bool IsPreCorrection; // "True" if the correction filter is applied // first. double FltCutoff; // Cutoff frequency of the first filtering step. double corrbw; ///< Bandwidth at the correction step. if (Vars.k <= 1.0) { IsPreCorrection = true; FltCutoff = 1.0; corrbw = 1.0; Steps.add(); } else { IsPreCorrection = false; FltCutoff = bw; corrbw = bw; } // Add 1 upsampling or several downsampling filters. if (UpsampleFactor > 1) { CFilterStep& fs = Steps.add(); assignFilterParams(fs, true, UpsampleFactor, FltCutoff, DCGain, DoFltAndIntCombo, IsModel); IntCutoffMult = FltCutoff * 2.0 / UpsampleFactor; ReuseStep = NULL; ExtFltStep = (DoFltAndIntCombo ? &fs : NULL); } else { int DownsampleFactor; while (true) { DownsampleFactor = (int)floor(0.5 / FltCutoff); bool DoHBFltAdd; if (DownsampleFactor > 16) { // Add half-band filter unconditionally in order to keep // filter lengths lower for more precise frequency // response and less edge artifacts. DoHBFltAdd = true; DownsampleFactor = 16; } else { DoHBFltAdd = (UseHalfband && DownsampleFactor > 1); } if (DoHBFltAdd) { assignFilterParams(Steps.add(), false, DownsampleFactor, 0.0, 1.0, false, IsModel); FltCutoff *= DownsampleFactor; } else { if (DownsampleFactor < 1) { DownsampleFactor = 1; } break; } } CFilterStep& fs = Steps.add(); assignFilterParams(fs, false, DownsampleFactor, FltCutoff, DCGain, DoFltAndIntCombo, IsModel); IntCutoffMult = FltCutoff / 0.5; if (DoFltAndIntCombo) { ReuseStep = &fs; ExtFltStep = &fs; } else { IntCutoffMult *= DownsampleFactor; ReuseStep = NULL; ExtFltStep = NULL; } } // Insert resizing and correction steps. CFilterStep& fs = (ReuseStep == NULL ? Steps.add() : *ReuseStep); Vars.ResizeStep = Steps.getItemCount() - 1; fs.IsUpsample = false; fs.ResampleFactor = 0; fs.DCGain = (ExtFltStep == NULL ? 1.0 : ExtFltStep->DCGain); initFilterBank(FltBank, IntCutoffMult, ForceHiOrderInt, (ExtFltStep == NULL ? fs.FltOrig : ExtFltStep->FltOrig)); if (FltBank == FixedFilterBank) { fs.FltBank = (CDSPFracFilterBankLin*)&FixedFilterBank; } else { fs.FltBank = &FltBank; } addCorrectionFilter(Steps, corrbw, IsPreCorrection, IsModel); // addSharpenTest( Steps, bw, IsModel ); } /** * Function extends *this upsampling step so that it produces more * upsampled pixels that cover the prefix and suffix needs of the next * step. After the call to this function the InPrefix and InSuffix * variables of the next step will be set to zero. * * @param fs Upsampling filtering step. * @param NextStep The next step structure. */ static void extendUpsample(CFilterStep& fs, CFilterStep& NextStep) { fs.InPrefix = (NextStep.InPrefix + fs.ResampleFactor - 1) / fs.ResampleFactor; fs.OutPrefix += fs.InPrefix * fs.ResampleFactor; NextStep.InPrefix = 0; fs.InSuffix = (NextStep.InSuffix + fs.ResampleFactor - 1) / fs.ResampleFactor; fs.OutSuffix += fs.InSuffix * fs.ResampleFactor; NextStep.InSuffix = 0; } /** * Function fills resizing step's RPosBuf array, excluding the actual * "ftp" pointers and "SrcOffs" offsets. * * This array should be cleared if the resizing step or offset were * changed. Otherwise this function only fills the elements required to * cover resizing step's OutLen. * * This function is called by the updateFilterStepBuffers() function. * * @param fs Resizing step. * @param Vars Variables object. */ static void fillRPosBuf(CFilterStep& fs, const CImageResizerVars& Vars) { const int PrevLen = fs.RPosBuf->getCapacity(); if (fs.OutLen > PrevLen) { fs.RPosBuf->increaseCapacity(fs.OutLen); } typename CFilterStep ::CResizePos* rpos = &(*fs.RPosBuf)[PrevLen]; const int FracCount = fs.FltBank->getFracCount(); const double o = Vars.o; const double k = Vars.k; int i; for (i = PrevLen; i < fs.OutLen; i++) { const double SrcPos = o + k * i; const int SrcPosInt = (int)floor(SrcPos); const double x = (SrcPos - SrcPosInt) * FracCount; const int fti = (int)x; rpos->x = (typename fpclass ::fptypeatom)(x - fti); rpos->fti = fti; rpos->SrcPosInt = SrcPosInt; rpos++; } } /** * Function updates filtering step buffer lengths depending on the * specified source and new scanline lengths. This function should be * called after the buildFilterSteps() function. * * @param Steps Array that receives filtering steps. * @param[out] Vars Variables object, will receive buffer size and length. * This function expects "k" and "o" variable values that will be * adjusted by this function. * @param RPosBufArray Resizing position buffers array, used to obtain * buffer to initialize and use (will be reused if it is already fully or * partially filled). * @param SrcLen Source scanline's length in pixels. * @param NewLen New scanline's length in pixels. */ static void updateFilterStepBuffers( CFilterSteps& Steps, CImageResizerVars& Vars, typename CFilterStep ::CRPosBufArray& RPosBufArray, int SrcLen, const int NewLen) { int upstep = -1; int InBuf = 0; int i; for (i = 0; i < Steps.getItemCount(); i++) { CFilterStep& fs = Steps[i]; fs.Vars = &Vars; fs.InLen = SrcLen; fs.InBuf = InBuf; fs.OutBuf = (InBuf + 1) & 1; if (fs.IsUpsample) { upstep = i; Vars.k *= fs.ResampleFactor; Vars.o *= fs.ResampleFactor; fs.InPrefix = 0; fs.InSuffix = 0; fs.OutLen = fs.InLen * fs.ResampleFactor; fs.OutPrefix = fs.FltLatency; fs.OutSuffix = fs.Flt.getCapacity() - fs.FltLatency - fs.ResampleFactor; int l0 = fs.OutPrefix + fs.OutLen + fs.OutSuffix; int l = fs.InLen * fs.ResampleFactor + fs.SuffixDC.getCapacity(); if (l > l0) { fs.OutSuffix += l - l0; } l0 = fs.OutLen + fs.OutSuffix; if (fs.PrefixDC.getCapacity() > l0) { fs.OutSuffix += fs.PrefixDC.getCapacity() - l0; } } else if (fs.ResampleFactor == 0) { const int FilterLenD2 = fs.FltBank->getFilterLen() / 2; const int FilterLenD21 = FilterLenD2 - 1; const int ResizeLPix = (int)floor(Vars.o) - FilterLenD21; fs.InPrefix = (ResizeLPix < 0 ? -ResizeLPix : 0); const int ResizeRPix = (int)floor(Vars.o + (NewLen - 1) * Vars.k) + FilterLenD2 + 1; fs.InSuffix = (ResizeRPix > fs.InLen ? ResizeRPix - fs.InLen : 0); fs.OutLen = NewLen; fs.RPosBuf = &RPosBufArray.getRPosBuf(Vars.k, Vars.o, fs.FltBank->getFracCount()); fillRPosBuf(fs, Vars); } else { Vars.k /= fs.ResampleFactor; Vars.o /= fs.ResampleFactor; Vars.o += fs.EdgePixelCount; fs.InPrefix = fs.FltLatency; fs.InSuffix = fs.Flt.getCapacity() - fs.FltLatency - 1; // Additionally extend OutLen to produce more precise edge // pixels. fs.OutLen = (fs.InLen + fs.ResampleFactor - 1) / fs.ResampleFactor + fs.EdgePixelCount; fs.InSuffix += (fs.OutLen - 1) * fs.ResampleFactor + 1 - fs.InLen; fs.InPrefix += fs.EdgePixelCount * fs.ResampleFactor; fs.OutLen += fs.EdgePixelCount; } InBuf = fs.OutBuf; SrcLen = fs.OutLen; } Steps[Steps.getItemCount() - 1].OutBuf = 2; if (upstep != -1) { extendUpsample(Steps[upstep], Steps[upstep + 1]); } } /** * Function calculates an optimal intermediate buffer length that will * cover all needs of the specified filtering steps. This function should * be called after the updateFilterStepBuffers() function. * * Function also updates resizing step's RPosBuf pointers to the filter * bank and SrcOffs values. * * @param Steps Filtering steps. * @param[out] Vars Variables object, will receive buffer size and length. * @param ResElIncr Resulting (final) element increment, used to produce * de-interleaved result. For horizontal processing this value is equal * to last step's OutLen, for vertical processing this value is equal to * resulting image's width. */ static void updateBufLenAndRPosPtrs(CFilterSteps& Steps, CImageResizerVars& Vars, const int ResElIncr) { int MaxPrefix[2] = {0, 0}; int MaxLen[2] = {0, 0}; int i; for (i = 0; i < Steps.getItemCount(); i++) { CFilterStep& fs = Steps[i]; const int ib = fs.InBuf; if (fs.InPrefix > MaxPrefix[ib]) { MaxPrefix[ib] = fs.InPrefix; } int l = fs.InLen + fs.InSuffix; if (l > MaxLen[ib]) { MaxLen[ib] = l; } fs.InElIncr = fs.InPrefix + l; if (fs.OutBuf == 2) { break; } const int ob = fs.OutBuf; if (fs.IsUpsample) { if (fs.OutPrefix > MaxPrefix[ob]) { MaxPrefix[ob] = fs.OutPrefix; } l = fs.OutLen + fs.OutSuffix; if (l > MaxLen[ob]) { MaxLen[ob] = l; } } else { if (fs.OutLen > MaxLen[ob]) { MaxLen[ob] = fs.OutLen; } } } // Update OutElIncr values of all steps. for (i = 0; i < Steps.getItemCount(); i++) { CFilterStep& fs = Steps[i]; if (fs.OutBuf == 2) { fs.OutElIncr = ResElIncr; break; } CFilterStep& fs2 = Steps[i + 1]; if (fs.IsUpsample) { fs.OutElIncr = fs.OutPrefix + fs.OutLen + fs.OutSuffix; if (fs.OutElIncr > fs2.InElIncr) { fs2.InElIncr = fs.OutElIncr; } else { fs.OutElIncr = fs2.InElIncr; } } else { fs.OutElIncr = fs2.InElIncr; } } // Update temporary buffer's length. for (i = 0; i < 2; i++) { Vars.BufLen[i] = MaxPrefix[i] + MaxLen[i]; Vars.BufOffs[i] = MaxPrefix[i]; if (Vars.packmode == 0) { Vars.BufOffs[i] *= Vars.ElCount; } Vars.BufLen[i] *= Vars.ElCount; } // Update RPosBuf pointers and SrcOffs. CFilterStep& fs = Steps[Vars.ResizeStep]; typename CFilterStep ::CResizePos* rpos = &(*fs.RPosBuf)[0]; const int em = (fpclass ::packmode == 0 ? Vars.ElCount : 1); const int FilterLenD21 = fs.FltBank->getFilterLen() / 2 - 1; for (i = 0; i < fs.OutLen; i++) { rpos->ftp = fs.FltBank->getFilter(rpos->fti); rpos->SrcOffs = (rpos->SrcPosInt - FilterLenD21) * em; rpos++; } } /** * Function modifies the overall (DC) gain of the correction filter in the * pre-built filtering steps array. * * @param Steps Filtering steps. * @param m Multiplier to apply to the correction filter. */ void modifyCorrFilterDCGain(CFilterSteps& Steps, const double m) const { CBuffer* Flt; const int z = Steps.getItemCount() - 1; if (!Steps[z].IsUpsample && Steps[z].ResampleFactor == 1) { Flt = &Steps[z].Flt; } else { Flt = &Steps[0].Flt; } int i; for (i = 0; i < Flt->getCapacity(); i++) { (*Flt)[i] = (fptype)((double)(*Flt)[i] * m); } } /** * Function builds a map of used fractional delay filters based on the * resizing positions buffer. * * @param fs Resizing step. * @param[out] UsedFracMap Map of used fractional delay filters. */ static void fillUsedFracMap(const CFilterStep& fs, CBuffer& UsedFracMap) { const int FracCount = fs.FltBank->getFracCount(); UsedFracMap.increaseCapacity(FracCount, false); memset(&UsedFracMap[0], 0, FracCount * sizeof(UsedFracMap[0])); typename CFilterStep ::CResizePos* rpos = &(*fs.RPosBuf)[0]; int i; for (i = 0; i < fs.OutLen; i++) { UsedFracMap[rpos->fti] |= 1; rpos++; } } /** * Function calculates the overall filtering steps complexity per * scanline. Each complexity unit corresponds to a single multiply-add * operation. Data copy and pointer math operations are not included in * this calculation, it is assumed that they correlate to the multiply-add * operations. Calculation also does not include final rounding, dithering * and clamping operations since they cannot be optimized out anyway. * * Calculation of the CRPosBuf buffer is not included since it cannot be * avoided. * * This function should be called after the updateFilterStepBuffers() * function. * * @param Steps Filtering steps array. * @param Vars Variables object. * @param UsedFracMap The map of used fractional delay filters. * @param ScanlineCount Scanline count. */ static int calcComplexity(const CFilterSteps& Steps, const CImageResizerVars& Vars, const CBuffer& UsedFracMap, const int ScanlineCount) { int fcnum; // Filter complexity multiplier numerator. int fcdenom; // Filter complexity multiplier denominator. if (Vars.packmode != 0) { fcnum = 1; fcdenom = 1; } else { // In interleaved processing mode, filters require 1 less // multiplication per 2 multiply-add instructions. fcnum = 3; fcdenom = 4; } int s = 0; // Complexity per one scanline. int s2 = 0; // Complexity per all scanlines. int i; for (i = 0; i < Steps.getItemCount(); i++) { const CFilterStep& fs = Steps[i]; s2 += 65 * fs.Flt.getCapacity(); // Filter creation complexity. if (fs.IsUpsample) { if (fs.FltOrig.getCapacity() > 0) { continue; } s += (fs.Flt.getCapacity() * (fs.InPrefix + fs.InLen + fs.InSuffix) + fs.SuffixDC.getCapacity() + fs.PrefixDC.getCapacity()) * Vars.ElCount; } else if (fs.ResampleFactor == 0) { s += fs.FltBank->getFilterLen() * (fs.FltBank->getOrder() + Vars.ElCount) * fs.OutLen; s2 += fs.FltBank->calcInitComplexity(UsedFracMap); } else { s += fs.Flt.getCapacity() * Vars.ElCount * fs.OutLen * fcnum / fcdenom; } } return (s + s2 / ScanlineCount); } /** * @brief Thread-isolated data used for scanline processing. * * This structure holds data necessary for image's horizontal or vertical * scanline processing, including scanline processing queue. * * @tparam Tin Source element data type. Intermediate buffers store data * in floating point format. * @tparam Tout Destination element data type. Intermediate buffers store * data in floating point format. */ template class CThreadData : public CImageResizerThreadPool ::CWorkload { public: virtual void process() { processScanlineQueue(); } /** * This enumeration lists possible scanline operations. */ enum EScanlineOperation { sopResizeH, ///< Resize horizontal scanline. ///< sopResizeV, ///< Resize vertical scanline. ///< sopDitherAndUnpackH, ///< Dither and unpack horizontal scanline. ///< sopUnpackH ///< Unpack horizontal scanline. ///< }; /** * Function initializes *this thread data object and assigns certain * variables provided by the higher level code. * * @param aThreadIndex Index of this thread data (0-based). * @param aThreadCount Total number of threads used during processing. * @param aSteps Filtering steps. * @param aVars Image resizer variables. */ void init(const int aThreadIndex, const int aThreadCount, const CFilterSteps& aSteps, const CImageResizerVars& aVars) { ThreadIndex = aThreadIndex; ThreadCount = aThreadCount; Steps = &aSteps; Vars = &aVars; } /** * Function initializes scanline processing queue, and updates * capacities of intermediate buffers. * * @param aOp Operation to perform over scanline. * @param TotalLines The total number of scanlines that will be * processed by all threads. * @param aSrcLen Source scanline length in pixels. * @param aSrcIncr Source scanline buffer increment. Ignored in * horizontal scanline processing. * @param aResIncr Resulting scanline buffer increment. Ignored in * horizontal scanline processing. */ void initScanlineQueue(const EScanlineOperation aOp, const int TotalLines, const int aSrcLen, const int aSrcIncr = 0, const int aResIncr = 0) { const int l = Vars->BufLen[0] + Vars->BufLen[1]; if (Bufs.getCapacity() < l) { Bufs.alloc(l, fpclass ::fpalign); } BufPtrs[0] = Bufs + Vars->BufOffs[0]; BufPtrs[1] = Bufs + Vars->BufLen[0] + Vars->BufOffs[1]; int j; int ml = 0; for (j = 0; j < Steps->getItemCount(); j++) { const CFilterStep& fs = (*Steps)[j]; if (fs.ResampleFactor == 0 && ml < fs.FltBank->getFilterLen()) { ml = fs.FltBank->getFilterLen(); } } TmpFltBuf.alloc(ml, fpclass ::fpalign); ScanlineOp = aOp; SrcLen = aSrcLen; SrcIncr = aSrcIncr; ResIncr = aResIncr; QueueLen = 0; Queue.increaseCapacity((TotalLines + ThreadCount - 1) / ThreadCount, false); } /** * Function adds a scanline to the queue buffer. The * initScanlineQueue() function should be called before calling this * function. The number of calls to this add function should not * exceed the TotalLines spread over all threads. * * @param SrcBuf Source scanline buffer. * @param ResBuf Resulting scanline buffer. */ void addScanlineToQueue(void* const SrcBuf, void* const ResBuf) { Queue[QueueLen].SrcBuf = SrcBuf; Queue[QueueLen].ResBuf = ResBuf; QueueLen++; } /** * Function processes all queued scanlines. */ void processScanlineQueue() { int i; switch (ScanlineOp) { case sopResizeH: { for (i = 0; i < QueueLen; i++) { resizeScanlineH((Tin*)Queue[i].SrcBuf, (fptype*)Queue[i].ResBuf); } break; } case sopResizeV: { for (i = 0; i < QueueLen; i++) { resizeScanlineV((fptype*)Queue[i].SrcBuf, (fptype*)Queue[i].ResBuf); } break; } case sopDitherAndUnpackH: { if (Vars->UseSRGBGamma) { for (i = 0; i < QueueLen; i++) { CFilterStep ::applySRGBGamma((fptype*)Queue[i].SrcBuf, SrcLen, *Vars); Ditherer.dither((fptype*)Queue[i].SrcBuf); CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } else { for (i = 0; i < QueueLen; i++) { Ditherer.dither((fptype*)Queue[i].SrcBuf); CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } break; } case sopUnpackH: { if (Vars->UseSRGBGamma) { for (i = 0; i < QueueLen; i++) { CFilterStep ::applySRGBGamma((fptype*)Queue[i].SrcBuf, SrcLen, *Vars); CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } else { for (i = 0; i < QueueLen; i++) { CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } break; } } } /** * Function returns ditherer object associated with *this thread data * object. */ CDitherer& getDitherer() { return (Ditherer); } private: int ThreadIndex; ///< Thread index. ///< int ThreadCount; ///< Thread count. ///< const CFilterSteps* Steps; ///< Filtering steps. ///< const CImageResizerVars* Vars; ///< Image resizer variables. ///< CBuffer Bufs; ///< Flip-flop intermediate buffers. ///< fptype* BufPtrs[3]; ///< Flip-flop buffer pointers (referenced by ///< filtering step's InBuf and OutBuf indices). ///< CBuffer TmpFltBuf; ///< Temporary buffer used in the ///< doResize() function, aligned by fpclass :: fpalign. ///< EScanlineOperation ScanlineOp; ///< Operation to perform over ///< scanline. ///< int SrcLen; ///< Source scanline length in the last queue. ///< int SrcIncr; ///< Source scanline buffer increment in the last queue. ///< int ResIncr; ///< Resulting scanline buffer increment in the last ///< queue. ///< CDitherer Ditherer; ///< Ditherer object to use. ///< /** * @brief Scanline processing queue item. * * Scanline processing queue item. */ struct CQueueItem { void* SrcBuf; ///< Source scanline buffer, will by typecasted to ///< Tin or fptype*. ///< void* ResBuf; ///< Resulting scanline buffer, will by typecasted ///< to Tout or fptype*. ///< }; CBuffer Queue; ///< Scanline processing queue. ///< int QueueLen; ///< Queue length. ///< /** * Function resizes a single horizontal scanline. * * @param SrcBuf Source scanline buffer. Can be either horizontal or * vertical. * @param ResBuf Resulting scanline buffer. */ void resizeScanlineH(const Tin* const SrcBuf, fptype* const ResBuf) { (*Steps)[0].packScanline(SrcBuf, BufPtrs[0], SrcLen); BufPtrs[2] = ResBuf; int j; for (j = 0; j < Steps->getItemCount(); j++) { const CFilterStep& fs = (*Steps)[j]; fs.prepareInBuf(BufPtrs[fs.InBuf]); const int DstIncr = (Vars->packmode == 0 ? Vars->ElCount : 1); if (fs.ResampleFactor != 0) { if (fs.IsUpsample) { fs.doUpsample(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf]); } else { fs.doFilter(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr); } } else { fs.doResize(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr, TmpFltBuf); } } } /** * Function resizes a single vertical scanline. * * @param SrcBuf Source scanline buffer. Can be either horizontal or * vertical. * @param ResBuf Resulting scanline buffer. */ void resizeScanlineV(const fptype* const SrcBuf, fptype* const ResBuf) { (*Steps)[0].convertVtoH(SrcBuf, BufPtrs[0], SrcLen, SrcIncr); BufPtrs[2] = ResBuf; int j; for (j = 0; j < Steps->getItemCount(); j++) { const CFilterStep& fs = (*Steps)[j]; fs.prepareInBuf(BufPtrs[fs.InBuf]); const int DstIncr = (fs.OutBuf == 2 ? ResIncr : (Vars->packmode == 0 ? Vars->ElCount : 1)); if (fs.ResampleFactor != 0) { if (fs.IsUpsample) { fs.doUpsample(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf]); } else { fs.doFilter(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr); } } else { fs.doResize(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr, TmpFltBuf); } } } }; }; #undef AVIR_PI #undef AVIR_PId2 } // namespace avir #endif // AVIR_CIMAGERESIZER_INCLUDED //$ nobt //$ nocpp /** * @file avir.h * * @brief The "main" inclusion file with all required classes and functions. * * This is the "main" inclusion file for the "AVIR" image resizer. This * inclusion file contains implementation of the AVIR image resizing algorithm * in its entirety. Also includes several classes and functions that can be * useful elsewhere. * * AVIR Copyright (c) 2015-2019 Aleksey Vaneev * * @mainpage * * @section intro_sec Introduction * * Description is available at https://github.com/avaneev/avir * * AVIR is devoted to women. Your digital photos can look good at any size! * * @section license License * * AVIR License Agreement * * The MIT License (MIT) * * Copyright (c) 2015-2019 Aleksey Vaneev * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. * * Please credit the author of this library in your documentation in the * following way: "AVIR image resizing algorithm designed by Aleksey Vaneev" * * @version 2.4 */ #ifndef AVIR_CIMAGERESIZER_INCLUDED #define AVIR_CIMAGERESIZER_INCLUDED #include #include #include #include namespace avir { /** * The macro defines AVIR version string. */ #define AVIR_VERSION "2.4" /** * The macro equals to "pi" constant, fills 53-bit floating point mantissa. * Undefined at the end of file. */ #define AVIR_PI 3.1415926535897932 /** * The macro equals to "pi divided by 2" constant, fills 53-bit floating * point mantissa. Undefined at the end of file. */ #define AVIR_PId2 1.5707963267948966 /** * Rounding function, based on the (int) typecast. Biased result. Not suitable * for numbers >= 2^31. * * @param d Value to round. * @return Rounded value. Some bias may be introduced. */ template inline T round(const T d) { return (d < 0.0 ? -(T)(int)((T)0.5 - d) : (T)(int)(d + (T)0.5)); } /** * Template function "clamps" (clips) the specified value so that it is not * lesser than "minv", and not greater than "maxv". * * @param Value Value to clamp. * @param minv Minimal allowed value. * @param maxv Maximal allowed value. * @return The clamped value. */ template inline T clamp(const T& Value, const T minv, const T maxv) { if (Value < minv) { return (minv); } else if (Value > maxv) { return (maxv); } else { return (Value); } } /** * Power 2.4 approximation function, designed for sRGB gamma correction. * * @param x Argument, in the range 0.09 to 1. * @return Value raised into power 2.4, approximate. */ template inline T pow24_sRGB(const T x) { const double x2 = x * x; const double x3 = x2 * x; const double x4 = x2 * x2; return ((T)(0.0985766365536824 + 0.839474952656502 * x2 + 0.363287814061725 * x3 - 0.0125559718896615 / (0.12758338921578 + 0.290283465468235 * x) - 0.231757513261358 * x - 0.0395365717969074 * x4)); } /** * Power 1/2.4 approximation function, designed for sRGB gamma correction. * * @param x Argument, in the range 0.003 to 1. * @return Value raised into power 1/2.4, approximate. */ template inline T pow24i_sRGB(const T x) { const double sx = sqrt(x); const double ssx = sqrt(sx); const double sssx = sqrt(ssx); return ((T)(0.000213364515060263 + 0.0149409239419218 * x + 0.433973412731747 * sx + ssx * (0.659628181609715 * sssx - 0.0380957908841466 - 0.0706476137208521 * sx))); } /** * Function approximately linearizes the sRGB gamma value. * * @param s sRGB gamma value, in the range 0 to 1. * @return Linearized sRGB gamma value, approximated. */ template inline T convertSRGB2Lin(const T s) { const T a = (T)0.055; if (s <= (T)0.04045) { return (s / (T)12.92); } return (pow24_sRGB((s + a) / ((T)1 + a))); } /** * Function approximately de-linearizes the linear gamma value. * * @param s Linear gamma value, in the range 0 to 1. * @return sRGB gamma value, approximated. */ template inline T convertLin2SRGB(const T s) { const T a = (T)0.055; if (s <= (T)0.0031308) { return ((T)12.92 * s); } return (((T)1 + a) * pow24i_sRGB(s) - a); } /** * Function converts (via typecast) specified array of type T1 values of * length l into array of type T2 values. If T1 is the same as T2, copy * operation is performed. When copying data at overlapping address spaces, * "op" should be lower than "ip". * * @param ip Input buffer. * @param[out] op Output buffer. * @param l The number of elements to copy. * @param ip Input buffer pointer increment. * @param op Output buffer pointer increment. */ template inline void copyArray(const T1* ip, T2* op, int l, const int ipinc = 1, const int opinc = 1) { while (l > 0) { *op = (T2)*ip; op += opinc; ip += ipinc; l--; } } /** * Function adds values located in array "ip" to array "op". * * @param ip Input buffer. * @param[out] op Output buffer. * @param l The number of elements to add. * @param ip Input buffer pointer increment. * @param op Output buffer pointer increment. */ template inline void addArray(const T1* ip, T2* op, int l, const int ipinc = 1, const int opinc = 1) { while (l > 0) { *op += *ip; op += opinc; ip += ipinc; l--; } } /** * Function that replicates a set of adjacent elements several times in a row. * This operation is usually used to replicate pixels at the start or end of * image's scanline. * * @param ip Source array. * @param ipl Source array length (usually 1..4, but can be any number). * @param[out] op Destination buffer. * @param l Number of times the source array should be replicated (the * destination buffer should be able to hold ipl * l number of elements). * @param opinc Destination buffer position increment after replicating the * source array. This value should be equal to at least ipl. */ template inline void replicateArray(const T1* const ip, const int ipl, T2* op, int l, const int opinc) { if (ipl == 1) { while (l > 0) { op[0] = ip[0]; op += opinc; l--; } } else if (ipl == 4) { while (l > 0) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; op[3] = ip[3]; op += opinc; l--; } } else if (ipl == 3) { while (l > 0) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; op += opinc; l--; } } else if (ipl == 2) { while (l > 0) { op[0] = ip[0]; op[1] = ip[1]; op += opinc; l--; } } else { while (l > 0) { int i; for (i = 0; i < ipl; i++) { op[i] = ip[i]; } op += opinc; l--; } } } /** * Function calculates frequency response of the specified FIR filter at the * specified circular frequency. Phase can be calculated as atan2( im, re ). * Function uses computationally-efficient oscillators instead of "cos" and * "sin" functions. * * @param flt FIR filter's coefficients. * @param fltlen Number of coefficients (taps) in the filter. * @param th Circular frequency [0; pi]. * @param[out] re0 Resulting real part of the complex frequency response. * @param[out] im0 Resulting imaginary part of the complex frequency response. * @param fltlat Filter's latency in samples (taps). */ template inline void calcFIRFilterResponse(const T* flt, int fltlen, const double th, double& re0, double& im0, const int fltlat = 0) { const double sincr = 2.0 * cos(th); double cvalue1; double svalue1; if (fltlat == 0) { cvalue1 = 1.0; svalue1 = 0.0; } else { cvalue1 = cos(-fltlat * th); svalue1 = sin(-fltlat * th); } double cvalue2 = cos(-(fltlat + 1) * th); double svalue2 = sin(-(fltlat + 1) * th); double re = 0.0; double im = 0.0; while (fltlen > 0) { re += cvalue1 * flt[0]; im += svalue1 * flt[0]; flt++; fltlen--; double tmp = cvalue1; cvalue1 = sincr * cvalue1 - cvalue2; cvalue2 = tmp; tmp = svalue1; svalue1 = sincr * svalue1 - svalue2; svalue2 = tmp; } re0 = re; im0 = im; } /** * Function normalizes FIR filter so that its frequency response at DC is * equal to DCGain. * * @param[in,out] p Filter coefficients. * @param l Filter length. * @param DCGain Filter's gain at DC. * @param pstep "p" array step. */ template inline void normalizeFIRFilter(T* const p, const int l, const double DCGain, const int pstep = 1) { double s = 0.0; T* pp = p; int i = l; while (i > 0) { s += *pp; pp += pstep; i--; } s = DCGain / s; pp = p; i = l; while (i > 0) { *pp = (T)(*pp * s); pp += pstep; i--; } } /** * @brief Memory buffer class for element array storage, with capacity * tracking. * * Allows easier handling of memory blocks allocation and automatic * deallocation for arrays (buffers) consisting of elements of specified * class. Tracks buffer's capacity in "int" variable; unsuitable for * allocation of very large memory blocks (with more than 2 billion elements). * * This class manages memory space only - it does not perform element class * construction (initialization) operations. Buffer's required memory address * alignment specification is supported. * * Uses standard library to allocate and deallocate memory. * * @tparam T Buffer element's type. * @tparam capint Buffer capacity's type to use. Use size_t for large buffers. */ template class CBuffer { public: CBuffer() : Data(NULL), DataAligned(NULL), Capacity(0), Alignment(0) {} /** * Constructor creates the buffer with the specified capacity. * * @param aCapacity Buffer's capacity. * @param aAlignment Buffer's required memory address alignment. 0 - use * stdlib's default alignment. */ CBuffer(const capint aCapacity, const int aAlignment = 0) { allocinit(aCapacity, aAlignment); } CBuffer(const CBuffer& Source) { allocinit(Source.Capacity, Source.Alignment); memcpy(DataAligned, Source.DataAligned, Capacity * sizeof(T)); } ~CBuffer() { freeData(); } CBuffer& operator=(const CBuffer& Source) { alloc(Source.Capacity, Source.Alignment); memcpy(DataAligned, Source.DataAligned, Capacity * sizeof(T)); return (*this); } /** * Function allocates memory so that the specified number of elements * can be stored in *this buffer object. * * @param aCapacity Storage for this number of elements to allocate. * @param aAlignment Buffer's required memory address alignment, * power-of-2 values only. 0 - use stdlib's default alignment. */ void alloc(const capint aCapacity, const int aAlignment = 0) { freeData(); allocinit(aCapacity, aAlignment); } /** * Function deallocates any previously allocated buffer. */ void free() { freeData(); Data = NULL; DataAligned = NULL; Capacity = 0; Alignment = 0; } /** * @return The capacity of the element buffer. */ capint getCapacity() const { return (Capacity); } /** * Function "forces" *this buffer to have an arbitary capacity. Calling * this function invalidates all further operations except deleting *this * object. This function should not be usually used at all. Function can * be used to "model" certain buffer capacity without calling a costly * memory allocation function. * * @param NewCapacity A new "forced" capacity. */ void forceCapacity(const capint NewCapacity) { Capacity = NewCapacity; } /** * Function reallocates *this buffer to a larger size so that it will be * able to hold the specified number of elements. Downsizing is not * performed. Alignment is not changed. * * @param NewCapacity New (increased) capacity. * @param DoDataCopy "True" if data in the buffer should be retained. */ void increaseCapacity(const capint NewCapacity, const bool DoDataCopy = true) { if (NewCapacity < Capacity) { return; } if (DoDataCopy) { const capint PrevCapacity = Capacity; T* const PrevData = Data; T* const PrevDataAligned = DataAligned; allocinit(NewCapacity, Alignment); memcpy(DataAligned, PrevDataAligned, PrevCapacity * sizeof(T)); ::free(PrevData); } else { ::free(Data); allocinit(NewCapacity, Alignment); } } /** * Function "truncates" (reduces) capacity of the buffer without * reallocating it. Alignment is not changed. * * @param NewCapacity New required capacity. */ void truncateCapacity(const capint NewCapacity) { if (NewCapacity >= Capacity) { return; } Capacity = NewCapacity; } /** * Function increases capacity so that the specified number of * elements can be stored. This function increases the previous capacity * value by third the current capacity value until space for the required * number of elements is available. Alignment is not changed. * * @param ReqCapacity Required capacity. */ void updateCapacity(const capint ReqCapacity) { if (ReqCapacity <= Capacity) { return; } capint NewCapacity = Capacity; while (NewCapacity < ReqCapacity) { NewCapacity += NewCapacity / 3 + 1; } increaseCapacity(NewCapacity); } operator T*() const { return (DataAligned); } private: T* Data; ///< Element buffer pointer. ///< T* DataAligned; ///< Memory address-aligned element buffer pointer. ///< capint Capacity; ///< Element buffer capacity. ///< int Alignment; ///< Memory address alignment in use. 0 - use stdlib's ///< default alignment. ///< /** * Internal element buffer allocation function used during object * construction. * * @param aCapacity Storage for this number of elements to allocate. * @param aAlignment Buffer's required memory address alignment. 0 - use * stdlib's default alignment. */ void allocinit(const capint aCapacity, const int aAlignment) { if (aAlignment == 0) { Data = (T*)::malloc(aCapacity * sizeof(T)); DataAligned = Data; Alignment = 0; } else { Data = (T*)::malloc(aCapacity * sizeof(T) + aAlignment); DataAligned = alignptr(Data, aAlignment); Alignment = aAlignment; } Capacity = aCapacity; } /** * Function frees a previously allocated Data buffer. */ void freeData() { ::free(Data); } /** * Function modifies the specified pointer so that it becomes memory * address-aligned. * * @param ptr Pointer to align. * @param align Alignment in bytes to apply. * @return Pointer aligned to align bytes. Works with power-of-2 * alignments only. If no alignment is necessary, "align" bytes will be * added to the pointer value. */ template inline Tp alignptr(const Tp ptr, const uintptr_t align) { return ((Tp)((uintptr_t)ptr + align - ((uintptr_t)ptr & (align - 1)))); } }; /** * Function optimizes the length of the symmetric-odd FIR filter by removing * left- and rightmost elements that are below specific threshold. * * Synthetic test shows that filter gets optimized in 2..3% of cases and in * each such case optimization reduces filter length by 6..8%. Optimization, * however, may skew the results of algorithm modeling and complexity * calculation leading to a choice of a less optimal algorithm. * * @param[in,out] Flt Buffer that contains filter being optimized. * @param[in,out] FltLatency Variable that holds the current latency of the * filter. May be adjusted on function return. * @param Threshold Threshold level. */ template inline void optimizeFIRFilter(CBuffer& Flt, int& FltLatency, T const Threshold = (T)0.00001) { int i; // Optimize length. for (i = 0; i <= FltLatency; i++) { if (fabs(Flt[i]) >= Threshold || i == FltLatency) { if (i > 0) { const int NewCapacity = Flt.getCapacity() - i * 2; copyArray(&Flt[i], &Flt[0], NewCapacity); Flt.truncateCapacity(NewCapacity); FltLatency -= i; } break; } } } /** * @brief Array of structured objects. * * Implements allocation of a linear array of objects of class T (which are * initialized), addressable via operator[]. Each object is created via the * "operator new". New object insertions are quick since implementation uses * prior space allocation (capacity), thus not requiring frequent memory block * reallocations. * * @tparam T Array element's type. */ template class CStructArray { public: CStructArray() : ItemCount(0) {} CStructArray(const CStructArray& Source) : ItemCount(0), Items(Source.getItemCount()) { while (ItemCount < Source.getItemCount()) { Items[ItemCount] = new T(Source[ItemCount]); ItemCount++; } } ~CStructArray() { clear(); } CStructArray& operator=(const CStructArray& Source) { clear(); const int NewCount = Source.ItemCount; Items.updateCapacity(NewCount); while (ItemCount < NewCount) { Items[ItemCount] = new T(Source[ItemCount]); ItemCount++; } return (*this); } T& operator[](const int Index) { return (*Items[Index]); } const T& operator[](const int Index) const { return (*Items[Index]); } /** * Function creates a new object of type T with the default constructor * and adds this object to the array. * * @return Reference to a newly added object. */ T& add() { if (ItemCount == Items.getCapacity()) { Items.increaseCapacity(ItemCount * 3 / 2 + 1); } Items[ItemCount] = new T(); ItemCount++; return ((*this)[ItemCount - 1]); } /** * Function changes number of allocated items. New items are created with * the default constructor. If NewCount is below the current item count, * items that are above NewCount range will be destructed. * * @param NewCount New requested item count. */ void setItemCount(const int NewCount) { if (NewCount > ItemCount) { Items.increaseCapacity(NewCount); while (ItemCount < NewCount) { Items[ItemCount] = new T(); ItemCount++; } } else { while (ItemCount > NewCount) { ItemCount--; delete Items[ItemCount]; } } } /** * Function erases all items of *this array. */ void clear() { while (ItemCount > 0) { ItemCount--; delete Items[ItemCount]; } } /** * @return The number of allocated items. */ int getItemCount() const { return (ItemCount); } private: int ItemCount; ///< The number of items available in the array. ///< CBuffer Items; ///< Element buffer. ///< }; /** * @brief Sine signal generator class. * * Class implements sine signal generator without biasing, with * constructor-based initalization only. This generator uses oscillator * instead of "sin" function. */ class CSineGen { public: /** * Constructor initializes *this sine signal generator. * * @param si Sine function increment, in radians. * @param ph Starting phase, in radians. Add 0.5 * AVIR_PI for cosine * function. */ CSineGen(const double si, const double ph) : svalue1(sin(ph)), svalue2(sin(ph - si)), sincr(2.0 * cos(si)) {} /** * @return The next value of the sine function, without biasing. */ double generate() { const double res = svalue1; svalue1 = sincr * res - svalue2; svalue2 = res; return (res); } private: double svalue1; ///< Current sine value. ///< double svalue2; ///< Previous sine value. ///< double sincr; ///< Sine value increment. ///< }; /** * @brief Peaked Cosine window function generator class. * * Class implements Peaked Cosine window function generator. Generates the * right-handed half of the window function. The Alpha parameter of this * window function offers the control of the balance between the early and * later taps of the filter. E.g. at Alpha=1 both early and later taps are * attenuated, but at Alpha=4 mostly later taps are attenuated. This offers a * great control over ringing artifacts produced by a low-pass filter in image * processing, without compromising achieved image sharpness. */ class CDSPWindowGenPeakedCosine { public: /** * Constructor initializes *this window function generator. * * @param aAlpha Alpha parameter, affects the peak shape (peak * augmentation) of the window function. Should be >= 1.0. * @param aLen2 Half filter's length (non-truncated). */ CDSPWindowGenPeakedCosine(const double aAlpha, const double aLen2) : Alpha(aAlpha), Len2(aLen2), wn(0), w1(AVIR_PId2 / Len2, AVIR_PI * 0.5) {} /** * @return The next Peaked Cosine window function coefficient. */ double generate() { const double h = pow(wn / Len2, Alpha); wn++; return (w1.generate() * (1.0 - h)); } private: double Alpha; ///< Alpha parameter, affects the peak shape of window. ///< double Len2; ///< Half length of the window function. ///< int wn; ///< Window function integer position. 0 - center of the ///< window function. ///< CSineGen w1; ///< Sine-wave generator. ///< }; /** * @brief FIR filter-based equalizer generator. * * Class implements an object used to generate symmetric-odd FIR filters with * the specified frequency response (aka paragraphic equalizer). The * calculated filter is windowed by the Peaked Cosine window function. * * In image processing, due to short length of filters being used (6-8 taps) * the resulting frequency response of the filter is approximate and may be * mathematically imperfect, but still adequate to the visual requirements. * * On a side note, this equalizer generator can be successfully used for audio * signal equalization as well: for example, it is used in almost the same * form in Voxengo Marvel GEQ equalizer plug-in. * * Filter generation is based on decomposition of frequency range into * spectral bands, with each band represented by linear and ramp "kernels". * When the filter is built, these kernels are combined together with * different weights that approximate the required frequency response. */ class CDSPFIREQ { public: /** * Function initializes *this object with the required parameters. The * gain of frequencies beyond the MinFreq..MaxFreq range are controlled by * the first and the last band's gain. * * @param SampleRate Processing sample rate (use 2 for image processing). * @param aFilterLength Required filter length in samples (taps). The * actual filter length is truncated to an integer value. * @param aBandCount Number of band crossover points required to control, * including bands at MinFreq and MaxFreq. * @param MinFreq Minimal frequency that should be controlled. * @param MaxFreq Maximal frequency that should be controlled. * @param IsLogBands "True" if the bands should be spaced logarithmically. * @param WFAlpha Peaked Cosine window function's Alpha parameter. */ void init(const double SampleRate, const double aFilterLength, const int aBandCount, const double MinFreq, const double MaxFreq, const bool IsLogBands, const double WFAlpha) { FilterLength = aFilterLength; BandCount = aBandCount; CenterFreqs.alloc(BandCount); z = (int)ceil(FilterLength * 0.5); zi = z + (z & 1); z2 = z * 2; CBuffer oscbuf(z2); initOscBuf(oscbuf); CBuffer winbuf(z); initWinBuf(winbuf, WFAlpha); UseFirstVirtBand = (MinFreq > 0.0); const int k = zi * (BandCount + (UseFirstVirtBand ? 1 : 0)); Kernels1.alloc(k); Kernels2.alloc(k); double m; // Frequency step multiplier. double mo; // Frequency step offset (addition). if (IsLogBands) { m = exp(log(MaxFreq / MinFreq) / (BandCount - 1)); mo = 0.0; } else { m = 1.0; mo = (MaxFreq - MinFreq) / (BandCount - 1); } double f = MinFreq; double x1 = 0.0; double x2; int si; if (UseFirstVirtBand) { si = 0; } else { si = 1; CenterFreqs[0] = 0.0; f = f * m + mo; } double* kernbuf1 = &Kernels1[0]; double* kernbuf2 = &Kernels2[0]; int i; for (i = si; i < BandCount; i++) { x2 = f * 2.0 / SampleRate; CenterFreqs[i] = x2; fillBandKernel(x1, x2, kernbuf1, kernbuf2, oscbuf, winbuf); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; f = f * m + mo; } if (x1 < 1.0) { UseLastVirtBand = true; fillBandKernel(x1, 1.0, kernbuf1, kernbuf2, oscbuf, winbuf); } else { UseLastVirtBand = false; } } /** * @return Filter's length, in samples (taps). */ int getFilterLength() const { return (z2 - 1); } /** * @return Filter's latency (group delay), in samples (taps). */ int getFilterLatency() const { return (z - 1); } /** * Function creates symmetric-odd FIR filter with the specified gain * levels at band crossover points. * * @param BandGains Array of linear gain levels, count=BandCount specified * in the init() function. * @param[out] Filter Output filter buffer, length = getFilterLength(). */ void buildFilter(const double* const BandGains, double* const Filter) { const double* kernbuf1 = &Kernels1[0]; const double* kernbuf2 = &Kernels2[0]; double x1 = 0.0; double y1 = BandGains[0]; double x2; double y2; int i; int si; if (UseFirstVirtBand) { si = 1; x2 = CenterFreqs[0]; y2 = y1; } else { si = 2; x2 = CenterFreqs[1]; y2 = BandGains[1]; } copyBandKernel(Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - x2 * y1); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; y1 = y2; for (i = si; i < BandCount; i++) { x2 = CenterFreqs[i]; y2 = BandGains[i]; addBandKernel(Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - x2 * y1); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; y1 = y2; } if (UseLastVirtBand) { addBandKernel(Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - y1); } for (i = 0; i < z - 1; i++) { Filter[z + i] = Filter[z - 2 - i]; } } /** * Function calculates filter's length (in samples) and latency depending * on the required non-truncated filter length. * * @param aFilterLength Required filter length in samples (non-truncated). * @param[out] Latency Resulting latency (group delay) of the filter, * in samples (taps). * @return Filter length in samples (taps). */ static int calcFilterLength(const double aFilterLength, int& Latency) { const int l = (int)ceil(aFilterLength * 0.5); Latency = l - 1; return (l * 2 - 1); } private: double FilterLength; ///< Length of filter. ///< int z; ///< Equals (int) ceil( FilterLength * 0.5 ). ///< int zi; ///< Equals "z" if z is even, or z + 1 if z is odd. Used as a ///< Kernels1 and Kernels2 size multiplier and kernel buffer ///< increment to make sure each kernel buffer is 16-byte aligned. ///< int z2; ///< Equals z * 2. ///< int BandCount; ///< Number of controllable bands. ///< CBuffer CenterFreqs; ///< Center frequencies for all bands, ///< normalized to 0.0-1.0 range. ///< CBuffer Kernels1; ///< Half-length kernel buffers for each ///< spectral band (linear part). ///< CBuffer Kernels2; ///< Half-length kernel buffers for each ///< spectral band (ramp part). ///< bool UseFirstVirtBand; ///< "True" if the first virtual band ///< (between 0.0 and MinFreq) should be used. The ///< first virtual band won't be used if MinFreq ///< equals 0.0. ///< bool UseLastVirtBand; ///< "True" if the last virtual band (between ///< MaxFreq and SampleRate * 0.5) should be used. The ///< last virtual band won't be used if MaxFreq * 2.0 ///< equals SampleRate. ///< /** * Function initializes the "oscbuf" used in the fillBandKernel() * function. * * @param oscbuf Oscillator buffer, length = z * 2. */ void initOscBuf(double* oscbuf) const { int i = z; while (i > 0) { oscbuf[0] = 0.0; oscbuf[1] = 1.0; oscbuf += 2; i--; } } /** * Function initializes window function buffer. This function generates * Peaked Cosine window function. * * @param winbuf Windowing buffer. * @param Alpha Peaked Cosine alpha parameter. */ void initWinBuf(double* winbuf, const double Alpha) const { CDSPWindowGenPeakedCosine wf(Alpha, FilterLength * 0.5); int i; for (i = 1; i <= z; i++) { winbuf[z - i] = wf.generate(); } } /** * Function fills first half of symmetric-odd FIR kernel for the band. * This function should be called successively for adjacent bands. * Previous band's x2 should be equal to current band's x1. A band kernel * consists of 2 elements: linear kernel and ramp kernel. * * @param x1 Band's left corner frequency (0..1). * @param x2 Band's right corner frequency (0..1). * @param kernbuf1 Band kernel buffer 1 (linear part), length = z. * @param kernbuf2 Band kernel buffer 2 (ramp part), length = z. * @param oscbuf Oscillation buffer. Before the first call of the * fillBandKernel() should be initialized with the call of the * initOscBuf() function. * @param winbuf Buffer that contains windowing function. */ void fillBandKernel(const double x1, const double x2, double* kernbuf1, double* kernbuf2, double* oscbuf, const double* const winbuf) { const double s2_incr = AVIR_PI * x2; const double s2_coeff = 2.0 * cos(s2_incr); double s2_value1 = sin(s2_incr * (-z + 1)); double c2_value1 = sin(s2_incr * (-z + 1) + AVIR_PI * 0.5); oscbuf[0] = sin(s2_incr * -z); oscbuf[1] = sin(s2_incr * -z + AVIR_PI * 0.5); int ks; for (ks = 1; ks < z; ks++) { const int ks2 = ks * 2; const double s1_value1 = oscbuf[ks2]; const double c1_value1 = oscbuf[ks2 + 1]; oscbuf[ks2] = s2_value1; oscbuf[ks2 + 1] = c2_value1; const double x = AVIR_PI * (ks - z); const double v0 = winbuf[ks - 1] / ((x1 - x2) * x); kernbuf1[ks - 1] = (x2 * s2_value1 - x1 * s1_value1 + (c2_value1 - c1_value1) / x) * v0; kernbuf2[ks - 1] = (s2_value1 - s1_value1) * v0; s2_value1 = s2_coeff * s2_value1 - oscbuf[ks2 - 2]; c2_value1 = s2_coeff * c2_value1 - oscbuf[ks2 - 1]; } kernbuf1[z - 1] = (x2 * x2 - x1 * x1) / (x1 - x2) * 0.5; kernbuf2[z - 1] = -1.0; } /** * Function copies band kernel's elements to the output buffer. * * @param outbuf Output buffer. * @param kernbuf1 Kernel buffer 1 (linear part). * @param kernbuf2 Kernel buffer 2 (ramp part). * @param c Multiplier for linear kernel element. * @param d Multiplier for ramp kernel element. */ void copyBandKernel(double* outbuf, const double* const kernbuf1, const double* const kernbuf2, const double c, const double d) const { int ks; for (ks = 0; ks < z; ks++) { outbuf[ks] = c * kernbuf1[ks] + d * kernbuf2[ks]; } } /** * Function adds band kernel's elements to the output buffer. * * @param outbuf Output buffer. * @param kernbuf1 Kernel buffer 1 (linear part). * @param kernbuf2 Kernel buffer 2 (ramp part). * @param c Multiplier for linear kernel element. * @param d Multiplier for ramp kernel element. */ void addBandKernel(double* outbuf, const double* const kernbuf1, const double* const kernbuf2, const double c, const double d) const { int ks; for (ks = 0; ks < z; ks++) { outbuf[ks] += c * kernbuf1[ks] + d * kernbuf2[ks]; } } }; /** * @brief Low-pass filter windowed by Peaked Cosine window function. * * This class implements calculation of linear-phase symmetric-odd FIR * low-pass filter windowed by the Peaked Cosine window function, for image * processing applications. */ class CDSPPeakedCosineLPF { public: int fl2; ///< Half filter's length, excluding the peak value. This value ///< can be also used as filter's latency in samples (taps). ///< int FilterLen; ///< Filter's length in samples (taps). ///< /** * Constructor initalizes *this object. * * @param aLen2 Half-length (non-truncated) of low-pass filter, in samples * (taps). * @param aFreq2 Low-pass filter's corner frequency [0; pi]. * @param aAlpha Peaked Cosine window function Alpha parameter. */ CDSPPeakedCosineLPF(const double aLen2, const double aFreq2, const double aAlpha) : fl2((int)ceil(aLen2) - 1), FilterLen(fl2 + fl2 + 1), Len2(aLen2), Freq2(aFreq2), Alpha(aAlpha) {} /** * Function generates a linear-phase low-pass filter windowed by Peaked * Cosine window function. * * @param[out] op Output buffer, length = FilterLen (fl2 * 2 + 1). * @param DCGain Required gain at DC. The resulting filter will be * normalized to achieve this DC gain. */ template void generateLPF(T* op, const double DCGain) { CDSPWindowGenPeakedCosine wf(Alpha, Len2); CSineGen f2(Freq2, 0.0); op += fl2; T* op2 = op; f2.generate(); int t = 1; *op = (T)(Freq2 * wf.generate() / AVIR_PI); double s = *op; while (t <= fl2) { const double v = f2.generate() * wf.generate() / t / AVIR_PI; op++; op2--; *op = (T)v; *op2 = (T)v; s += *op + *op2; t++; } t = FilterLen; s = DCGain / s; while (t > 0) { *op2 = (T)(*op2 * s); op2++; t--; } } private: double Len2; ///< Half-length (non-truncated) of low-pass filter, in ///< samples (taps). ///< double Freq2; ///< Low-pass filter's corner frequency. ///< double Alpha; ///< Peaked Cosine window function Alpha parameter. ///< }; /** * @brief Buffer class for parametrized low-pass filter. * * This class extends the CBuffer< double > class by adding several variables * that define a symmetric-odd FIR low-pass filter windowed by Peaked Cosine * window function. This class can be used to compare filters without * comparing their buffer contents. */ class CFltBuffer : public CBuffer { public: double Len2; ///< Half-length (non-truncated) of low-pass filters, in ///< samples (taps). ///< double Freq; ///< Low-pass filter's corner frequency. ///< double Alpha; ///< Peaked Cosine window function Alpha parameter. ///< double DCGain; ///< DC gain applied to the filter. ///< CFltBuffer() : CBuffer(), Len2(0.0), Freq(0.0), Alpha(0.0), DCGain(0.0) {} /** * @param b2 Filter buffer to compare *this object to. * @return Operator returns "true" if both filters have same parameters. */ bool operator==(const CFltBuffer& b2) const { return (Len2 == b2.Len2 && Freq == b2.Freq && Alpha == b2.Alpha && DCGain == b2.DCGain); } }; /** * @brief Sinc function-based fractional delay filter bank. * * Class implements storage and initialization of a bank of sinc * function-based fractional delay filters, expressed as 1st order polynomial * interpolation coefficients. The filters are produced from a single "long" * windowed low-pass filter. Also supports 0th-order ("nearest neighbor") * interpolation. * * This class also supports multiplication of each fractional delay filter by * an external filter (usually a low-pass filter). * * @tparam fptype Specifies storage type of the filter coefficients bank. The * filters are initially calculated using the "double" precision. */ template class CDSPFracFilterBankLin { public: CDSPFracFilterBankLin() : Order(-1) {} /** * Copy constructor copies a limited set of parameters of the source * filter bank. The actual filters are not copied. Such copying is used * during filtering steps "modeling" stage. A further init() function * call is required. * * @param s Source filter bank. */ void copyInitParams(const CDSPFracFilterBankLin& s) { WFLen2 = s.WFLen2; WFFreq = s.WFFreq; WFAlpha = s.WFAlpha; FracCount = s.FracCount; Order = s.Order; Alignment = s.Alignment; SrcFilterLen = s.SrcFilterLen; FilterLen = s.FilterLen; FilterSize = s.FilterSize; IsSrcTableBuilt = false; ExtFilter = s.ExtFilter; TableFillFlags.alloc(s.TableFillFlags.getCapacity()); int i; // Copy table fill flags, but shifted so that further initialization // is still possible (such feature should not be used, though). for (i = 0; i < TableFillFlags.getCapacity(); i++) { TableFillFlags[i] = (uint8_t)(s.TableFillFlags[i] << 2); } } /** * Operator compares *this filter bank and another filter bank and returns * "true" if their parameters are equal. Alignment is not taken into * account. * * @param s Filter bank to compare to. * @return "True" if compared banks have equal parameters. */ bool operator==(const CDSPFracFilterBankLin& s) const { return (Order == s.Order && WFLen2 == s.WFLen2 && WFFreq == s.WFFreq && WFAlpha == s.WFAlpha && FracCount == s.FracCount && ExtFilter == s.ExtFilter); } /** * Function initializes (builds) the filter bank based on the supplied * parameters. If the supplied parameters are equal to previously defined * parameters, function does nothing (alignment is assumed to be never * changing between the init() function calls). * * @param ReqFracCount Required number of fractional delays in the filter * bank. The minimal value is 2. * @param ReqOrder Required order of the interpolation polynomial * (0 or 1). * @param BaseLen Low-pass filter's base length, in samples (taps). * Affects the actual length of the filter and its overall steepness. * @param Cutoff Low-pass filter's normalized cutoff frequency [0; 1]. * @param aWFAlpha Peaked Cosine window function's Alpha parameter. * @param aExtFilter External filter to apply to each fractional delay * filter. * @param aAlignment Memory alignment of the filter bank, power-of-2 * value. 0 - use default stdlib alignment. * @param FltLenAlign Filter's length alignment, power-of-2 value. */ void init(const int ReqFracCount, const int ReqOrder, const double BaseLen, const double Cutoff, const double aWFAlpha, const CFltBuffer& aExtFilter, const int aAlignment = 0, const int FltLenAlign = 1) { double NewWFLen2 = 0.5 * BaseLen * ReqFracCount; double NewWFFreq = AVIR_PI * Cutoff / ReqFracCount; double NewWFAlpha = aWFAlpha; if (ReqOrder == Order && NewWFLen2 == WFLen2 && NewWFFreq == WFFreq && NewWFAlpha == WFAlpha && ReqFracCount == FracCount && aExtFilter == ExtFilter) { IsInitRequired = false; return; } WFLen2 = NewWFLen2; WFFreq = NewWFFreq; WFAlpha = NewWFAlpha; FracCount = ReqFracCount; Order = ReqOrder; Alignment = aAlignment; ExtFilter = aExtFilter; CDSPPeakedCosineLPF p(WFLen2, WFFreq, WFAlpha); SrcFilterLen = (p.fl2 / ReqFracCount + 1) * 2; const int ElementSize = ReqOrder + 1; FilterLen = SrcFilterLen; if (ExtFilter.getCapacity() > 0) { FilterLen += ExtFilter.getCapacity() - 1; } FilterLen = (FilterLen + FltLenAlign - 1) & ~(FltLenAlign - 1); FilterSize = FilterLen * ElementSize; IsSrcTableBuilt = false; IsInitRequired = true; } /** * @return The length of each fractional delay filter, in samples (taps). * Always an even value. */ int getFilterLen() const { return (FilterLen); } /** * @return The number of fractional filters in use by *this bank. */ int getFracCount() const { return (FracCount); } /** * @return The order of the interpolation polynomial. */ int getOrder() const { return (Order); } /** * Function returns the pointer to the specified interpolation table * filter. * * @param i Filter (fractional delay) index, in the range 0 to * ReqFracCount - 1, inclusive. * @return Pointer to filter. Higher order polynomial coefficients are * stored after after previous order coefficients, separated by FilterLen * elements. */ const fptype* getFilter(const int i) { if (!IsSrcTableBuilt) { buildSrcTable(); } fptype* const Res = &Table[i * FilterSize]; if ((TableFillFlags[i] & 2) == 0) { createFilter(i); TableFillFlags[i] |= 2; if (Order > 0) { createFilter(i + 1); const fptype* const Res2 = Res + FilterSize; fptype* const op = Res + FilterLen; int j; // Create higher-order interpolation coefficients (linear // interpolation). for (j = 0; j < FilterLen; j++) { op[j] = Res2[j] - Res[j]; } } } return (Res); } /** * Function makes sure all fractional delay filters were created. */ void createAllFilters() { int i; for (i = 0; i < FracCount; i++) { getFilter(i); } } /** * Function returns an approximate initialization complexity, expressed in * the number of multiply-add operations. This includes fractional delay * filters calculation and multiplication by an external filter. This * function can only be called after the init() function. * * @param FracUseMap Fractional delays use map, each element corresponds * to a single fractional delay, will be compared to the internal table * fill flags. This map should include 0 and 1 values only. * @return The complexity of the initialization, expressed in the number * of multiply-add operations. */ int calcInitComplexity(const CBuffer& FracUseMap) const { const int FltInitCost = 65; // Cost to initialize a single sample // of the fractional delay filter. const int FltUseCost = FilterLen * Order + SrcFilterLen * ExtFilter.getCapacity(); // Cost to use a single // fractional delay filter. const int ucb[2] = {0, FltUseCost}; int ic; int i; if (IsInitRequired) { ic = FracCount * SrcFilterLen * FltInitCost; for (i = 0; i < FracCount; i++) { ic += ucb[FracUseMap[i]]; } } else { ic = 0; for (i = 0; i < FracCount; i++) { if (FracUseMap[i] != 0) { ic += ucb[TableFillFlags[i] == 0 ? 1 : 0]; } } } return (ic); } private: static const int InterpPoints = 2; ///< The maximal number of points the ///< interpolation is based on. ///< double WFLen2; ///< Window function's Len2 parameter. ///< double WFFreq; ///< Window function's Freq parameter. ///< double WFAlpha; ///< Window function's Alpha parameter. ///< int FracCount; ///< The required number of fractional delay filters. ///< int Order; ///< The order of the interpolation polynomial. ///< int Alignment; ///< The required filter table alignment. ///< int SrcFilterLen; ///< Length of the "source" filters. This is always an ///< even value. ///< int FilterLen; ///< Specifies the number of samples (taps) each fractional ///< delay filter has. This is always an even value, adjusted ///< by the FltLenAlign. ///< int FilterSize; ///< The size of a single filter element, equals ///< FilterLen * ElementSize. ///< bool IsInitRequired; ///< "True" if SrcTable filter table initialization ///< is required. This value is available only after the ///< call to the init() function. ///< CBuffer Table; ///< Interpolation table, size equals to ///< ReqFracCount * FilterLen * ElementSize. ///< CBuffer TableFillFlags; ///< Contains ReqFracCount + 1 ///< elements. Bit 0 of every element is 1 if Table ///< already contains the filter from SrcTable filtered ///< by ExtFilter. Bit 1 of every element means higher ///< order coefficients were filled for the filter. ///< CFltBuffer ExtFilter; ///< External filter that should be applied to every ///< fractional delay filter. Can be empty. Half of ///< this filter's capacity is used as latency (group ///< delay) value of the filter. ///< CBuffer SrcTable; ///< Source table of delay filters, contains ///< ReqFracCount + 1 elements. This table is used ///< to fill the Table with the actual filters, ///< filtered by an external filter. ///< bool IsSrcTableBuilt; ///< "True" if the SrcTable was built already. This ///< variable is set to "false" in the init() function. ///< /** * Function builds source table used in the createFilter() function. */ void buildSrcTable() { IsSrcTableBuilt = true; IsInitRequired = false; CDSPPeakedCosineLPF p(WFLen2, WFFreq, WFAlpha); const int BufLen = SrcFilterLen * FracCount + InterpPoints - 1; const int BufOffs = InterpPoints / 2 - 1; const int BufCenter = SrcFilterLen * FracCount / 2 + BufOffs; CBuffer Buf(BufLen); memset(Buf, 0, (BufCenter - p.fl2) * sizeof(double)); int i = BufLen - BufCenter - p.fl2 - 1; memset(&Buf[BufLen - i], 0, i * sizeof(double)); p.generateLPF(&Buf[BufCenter - p.fl2], FracCount); abort(); SrcTable.alloc((FracCount + 1) * SrcFilterLen); TableFillFlags.alloc(FracCount + 1); int j; double* op0 = SrcTable; for (i = FracCount; i >= 0; i--) { TableFillFlags[i] = 0; double* p = Buf + BufOffs + i; for (j = 0; j < SrcFilterLen; j++) { op0[0] = p[0]; op0++; p += FracCount; } } Table.alloc((FracCount + 1) * FilterSize, Alignment); } /** * Function creates the specified filter in the Table by copying it from * the SrcTable and filtering by ExtFilter. Function does nothing if * filter was already created. * * @param k Filter index to create, in the range 0 to FracCount, * inclusive. */ void createFilter(const int k) { if (TableFillFlags[k] != 0) { return; } TableFillFlags[k] |= 1; const int ExtFilterLatency = ExtFilter.getCapacity() / 2; const int ResLatency = ExtFilterLatency + SrcFilterLen / 2; int ResLen = SrcFilterLen; if (ExtFilter.getCapacity() > 0) { ResLen += ExtFilter.getCapacity() - 1; } const int ResOffs = FilterLen / 2 - ResLatency; fptype* op = &Table[k * FilterSize]; int i; for (i = 0; i < ResOffs; i++) { op[i] = 0.0; } for (i = ResOffs + ResLen; i < FilterLen; i++) { op[i] = 0.0; } op += ResOffs; const double* const srcflt = &SrcTable[k * SrcFilterLen]; if (ExtFilter.getCapacity() == 0) { for (i = 0; i < ResLen; i++) { op[i] = (fptype)srcflt[i]; } return; } // Perform convolution of extflt and srcflt. const double* const extflt = &ExtFilter[0]; int j; for (j = 0; j < ResLen; j++) { int k = 0; int l = j - ExtFilter.getCapacity() + 1; int r = l + ExtFilter.getCapacity(); if (l < 0) { k -= l; l = 0; } if (r > SrcFilterLen) { r = SrcFilterLen; } const double* const extfltb = extflt + k; const double* const srcfltb = srcflt + l; double s = 0.0; l = r - l; for (i = 0; i < l; i++) { s += extfltb[i] * srcfltb[i]; } op[j] = (fptype)s; } } }; /** * @brief Thread pool for multi-threaded image resizing operation. * * This base class is used to organize a multi-threaded image resizing * operation. The thread pool should consist of threads that initially wait * for a signal. Upon receiving a signal (via the startAllWorkloads() * function) each previously added thread should execute its workload's * process() function once, and return to the wait signal state again. The * thread pool should be also able to efficiently wait for all workloads to * finish via the waitAllWorkloadsToFinish() function. * * The image resizing algorithm makes calls to functions of this class. */ class CImageResizerThreadPool { public: CImageResizerThreadPool() {} virtual ~CImageResizerThreadPool() {} /** * @brief Thread pool's workload object class. * * This class should be used as a base class for objects that perform the * actual work spread over several threads. */ class CWorkload { public: virtual ~CWorkload() {} /** * Function that gets called from the thread when thread pool's * startAllWorkloads() function is called. */ virtual void process() = 0; }; /** * @return The suggested number of workloads (and their associated * threads) to add. The minimal value this function can return is 1. The * usual value may depend on the number of physical and virtual cores * present in the system, and on other considerations. */ virtual int getSuggestedWorkloadCount() const { return (1); } /** * Function adds a new workload (and possibly thread) to the thread pool. * The caller decides how many parallel workloads (and threads) it * requires, but this number will not exceed the value returned by the * getSuggestedWorkloadCount() function. It is implementation-specific how * many workloads to associate with a single thread. But for efficiency * reasons each workload should be associated with its own thread. * * Note that the same set of workload objects will be processed each time * the startAllWorkloads() function is called. This means that workload * objects are added only once. The caller changes the state of the * workload objects and then calls the startAllWorkloads() function to * process them. * * @param Workload Workload object whose process() function will be called * from within the thread when the startAllWorkloads() function is called. */ virtual void addWorkload(CWorkload* const Workload) {} /** * Function starts all workloads associated with threads previously added * via the addWorkload() function. It is assumed that this function * performs the necessary "memory barrier" (or "cache sync") kind of * operation so that all threads catch up the prior changes made to the * workload objects during their wait state. */ virtual void startAllWorkloads() {} /** * Function waits for all workloads to finish. */ virtual void waitAllWorkloadsToFinish() {} /** * Function removes all workloads previously added via the addWorkload() * function. This function gets called only after the * waitAllWorkloadsToFinish() function call. */ virtual void removeAllWorkloads() {} }; /** * @brief Resizing algorithm parameters structure. * * This structure holds all selectable parameters used by the resizing * algorithm at various stages, for both downsizing and upsizing. There are no * other parameters exist that can optimize the performance of the resizing * algorithm. Filter length parameters can take fractional values. * * Beside quality, these parameters (except Alpha parameters) directly affect * the computative cost of the resizing algorithm. It is possible to trade * the visual quality for computative cost. * * Anti-alias filtering during downsizing can be defined as a considerable * reduction of contrast of smallest features of an image. Unfortunately, such * de-contrasting partially affects features of all sizes thus producing a * non-linearity of frequency response. All pre-defined parameter sets are * described by 3 values separated by slashes. The first value is the * de-contrasting factor of small features (which are being removed) while * the second value is the de-contrasting factor of large features (which * should remain intact), with value of 1 equating to "no contrast change". * The third value is the optimization score (see below), with value of 0 * equating to the "perfect" linearity of frequency response. * * The pre-defined parameter sets offered by this library were auto-optimized * for the given LPFltBaseLen, IntFltLen and CorrFltAlpha values. The * optimization goal was to minimize the score: the sum of squares of the * difference between original and processed images (which was not actually * resized, k=1). The original image was a 0.5 megapixel uniformly-distributed * white-noise image with pixel intensities in the 0-1 range. Such goal * converges very well and produces filtering system with the flattest * frequency response possible for the given constraints. With this goal, * increasing the LPFltBaseLen value reduces the general amount of aliasing * artifacts. */ struct CImageResizerParams { double CorrFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the correction filter. The ///< "usable" values are in the narrow range 1.0 to 1.5. ///< double CorrFltLen; ///< Correction filter's length in samples (taps). The ///< "usable" range is narrow, 5.5 to 8, as to minimize ///< the "overcorrection" which is mathematically precise, ///< but visually unacceptable. ///< double IntFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the interpolation low-pass filter. ///< The "usable" values are in the range 1.5 to 2.5. ///< double IntFltCutoff; ///< Interpolation low-pass filter's cutoff frequency ///< (normalized, [0; 1]). The "usable" range is 0.6 to ///< 0.8. ///< double IntFltLen; ///< Interpolation low-pass filter's length in samples ///< (taps). The length value should be at least 18 or ///< otherwise a "dark grid" artifact will be introduced if ///< a further sharpening is applied. IntFltLen together ///< with other IntFlt parameters should be tuned in a way ///< that produces the flattest frequency response in 0-0.5 ///< normalized frequency range (this range is due to 2X ///< upsampling). ///< double LPFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the low-pass filter. The "usable" ///< values are in the range 1.5 to 6.5. ///< double LPFltBaseLen; ///< Base length of the low-pass (aka anti-aliasing ///< or reconstruction) filter, in samples (taps), ///< further adjusted by the actual cutoff frequency, ///< upsampling and downsampling factors. The "usable" ///< range is between 6 and 9. ///< double LPFltCutoffMult; ///< Low-pass filter's cutoff frequency ///< multiplier. This value can be both below and ///< above 1.0 as low-pass filters are inserted on ///< downsampling and upsampling steps and always ///< have corner frequency equal to or below 0.5pi. ///< This multiplier shifts low-pass filter's corner ///< frequency towards lower (if below 1.0) or higher ///< (if above 1.0) frequencies. This multiplier can ///< be way below 1.0 since any additional ///< high-frequency damping will be partially ///< corrected by the correction filter. The "usable" ///< range is 0.3 to 1.0. ///< CImageResizerParams() : HBFltAlpha(1.75395), HBFltCutoff(0.40356), HBFltLen(22.00000) {} double HBFltAlpha; ///< Half-band filter's Alpha. Assigned internally. ///< double HBFltCutoff; ///< Half-band filter's cutoff point [0; 1]. Assigned ///< internally. ///< double HBFltLen; ///< Length of the half-band low-pass filter. Assigned ///< internally. Internally used to perform 2X or higher ///< downsampling. These filter parameters should be treated ///< as "technical" and do not require adjustment as they ///< were tuned to suit all combinations of other ///< parameters. This half-band filter provides a wide ///< transition band (for minimal ringing artifacts) and a ///< high stop-band attenuation (for minimal aliasing). ///< }; /** * @brief The default set of resizing algorithm parameters * (10.01/1.029/0.019169). * * This is the default set of resizing parameters that was designed to deliver * a sharp image while still providing a low amount of ringing artifacts, and * having a reasonable computational cost. */ struct CImageResizerParamsDef : public CImageResizerParams { CImageResizerParamsDef() { CorrFltAlpha = 1.0; // 10.01/1.88/1.029(522.43)/0.019169:258648,446808 CorrFltLen = 6.30770; IntFltAlpha = 2.27825; IntFltCutoff = 0.75493; IntFltLen = 18.0; LPFltAlpha = 3.40127; LPFltBaseLen = 7.78; LPFltCutoffMult = 0.78797; } }; /** * @brief Set of resizing algorithm parameters for ultra-low-ringing * performance (7.69/1.069/0.000245). * * This set of resizing algorithm parameters offers the lowest amount of * ringing this library is capable of providing while still offering a decent * quality. Low ringing is attained at the expense of higher aliasing * artifacts and a slightly reduced contrast. */ struct CImageResizerParamsULR : public CImageResizerParams { CImageResizerParamsULR() { CorrFltAlpha = 1.0; // 7.69/1.97/1.069(31445.45)/0.000245:258627,436845 CorrFltLen = 5.83280; IntFltAlpha = 2.11453; IntFltCutoff = 0.73986; IntFltLen = 18.0; LPFltAlpha = 1.73455; LPFltBaseLen = 6.40; LPFltCutoffMult = 0.61314; } }; /** * @brief Set of resizing algorithm parameters for low-ringing performance * (7.86/1.065/0.000106). * * This set of resizing algorithm parameters offers a very low-ringing * performance at the expense of higher aliasing artifacts and a slightly * reduced contrast. */ struct CImageResizerParamsLR : public CImageResizerParams { CImageResizerParamsLR() { CorrFltAlpha = 1.0; // 7.86/1.96/1.065(73865.02)/0.000106:258636,437381 CorrFltLen = 5.87671; IntFltAlpha = 2.25322; IntFltCutoff = 0.74090; IntFltLen = 18.0; LPFltAlpha = 1.79306; LPFltBaseLen = 7.00; LPFltCutoffMult = 0.68881; } }; /** * @brief Set of resizing algorithm parameters for lower-ringing performance * (8.86/1.046/0.010168). * * This set of resizing algorithm parameters offers a lower-ringing * performance in comparison to the default setting, at the expense of higher * aliasing artifacts and a slightly reduced contrast. */ struct CImageResizerParamsLow : public CImageResizerParams { CImageResizerParamsLow() { CorrFltAlpha = 1.0; // 8.86/1.92/1.046(871.54)/0.010168:258647,442252 CorrFltLen = 6.09757; IntFltAlpha = 2.36704; IntFltCutoff = 0.74674; IntFltLen = 18.0; LPFltAlpha = 2.19427; LPFltBaseLen = 7.66; LPFltCutoffMult = 0.75380; } }; /** * @brief Set of resizing algorithm parameters for low-aliasing * resizing (11.81/1.012/0.038379). * * This set of resizing algorithm parameters offers a considerable * anti-aliasing performance with a good frequency response linearity (and * contrast). This is an intermediate setting between the default and Ultra * parameters. */ struct CImageResizerParamsHigh : public CImageResizerParams { CImageResizerParamsHigh() { CorrFltAlpha = 1.0; // 11.81/1.83/1.012(307.84)/0.038379:258660,452719 CorrFltLen = 6.80909; IntFltAlpha = 2.44917; IntFltCutoff = 0.75856; IntFltLen = 18.0; LPFltAlpha = 4.39527; LPFltBaseLen = 8.18; LPFltCutoffMult = 0.79172; } }; /** * @brief Set of resizing algorithm parameters for ultra low-aliasing * resizing (13.65/1.001/0.000483). * * This set of resizing algorithm parameters offers a very considerable * anti-aliasing performance with a good frequency response linearity (and * contrast). This set of parameters is computationally expensive and may * produce ringing artifacts on sharp features. */ struct CImageResizerParamsUltra : public CImageResizerParams { CImageResizerParamsUltra() { CorrFltAlpha = 1.0; // 13.65/1.79/1.001(28288.41)/0.000483:258658,457974 CorrFltLen = 7.48060; IntFltAlpha = 1.93750; IntFltCutoff = 0.75462; IntFltLen = 18.0; LPFltAlpha = 5.55209; LPFltBaseLen = 8.34; LPFltCutoffMult = 0.78002; } }; /** * @brief Image resizing variables class. * * This is an utility "catch all" class that defines various variables used * during image resizing. Several variables that are explicitly initialized in * this class' constructor are also used as additional "input" variables to * the image resizing function. These variables will not be changed by the * avir::CImageResizer<>::resizeImage() function. */ class CImageResizerVars { public: int ElCount; ///< The number of "fptype" elements used to store 1 pixel. ///< int ElCountIO; ///< The number of source and destination image's elements ///< used to store 1 pixel. ///< int fppack; ///< The number of atomic types stored in a single "fptype" ///< element. ///< int fpalign; ///< Suggested alignment size in bytes. This is not a ///< required alignment, because image resizing algorithm cannot ///< be made to have a strictly aligned data access in all cases ///< (e.g. de-interleaved interpolation cannot perform aligned ///< accesses). ///< int elalign; ///< Length alignment of arrays of elements. This applies to ///< filters and intermediate buffers: this constant forces ///< filters and scanlines to have a length which is a multiple ///< of this value, for more efficient SIMD implementation. ///< int packmode; ///< 0 if interleaved packing, 1 if de-interleaved. ///< int BufLen[2]; ///< Intermediate buffers' lengths in "fptype" elements. int BufOffs[2]; ///< Offsets into the intermediate buffers, used to ///< provide prefix elements required during processing so ///< that no "out of range" access happens. This offset is a ///< multiple of ElCount if pixels are stored in interleaved ///< form. ///< double k; ///< Resizing step coefficient, updated to reflect the actually ///< used coefficient during resizing. ///< double o; ///< Starting pixel offset inside the source image, updated to ///< reflect the actually used offset during resizing. ///< int ResizeStep; ///< Index of the resizing step in the latest filtering ///< steps array. ///< double InGammaMult; ///< Input gamma multiplier, used to convert input ///< data to 0 to 1 range. 0.0 if no gamma is in use. ///< double OutGammaMult; ///< Output gamma multiplier, used to convert data to ///< 0 to 255/65535 range. 0.0 if no gamma is in use. ///< double ox; ///< Start X pixel offset within source image (can be ///< negative). Positive offset moves image to the left. ///< double oy; ///< Start Y pixel offset within source image (can be ///< negative). Positive offset moves image to the top. ///< CImageResizerThreadPool* ThreadPool; ///< Thread pool to be used by the ///< image resizing function. Set to NULL to use ///< single-threaded processing. ///< bool UseSRGBGamma; ///< Perform sRGB gamma linearization (correction). ///< int BuildMode; ///< The build mode to use, for debugging purposes. Set to ///< -1 to select a minimal-complexity mode automatically. All ///< build modes deliver similar results with minor ///< deviations. ///< int RndSeed; ///< Random seed parameter. This parameter may be incremented ///< after each random generator initialization. The use of this ///< variable depends on the ditherer implementation. ///< CImageResizerVars() : ox(0.0), oy(0.0), ThreadPool(NULL), UseSRGBGamma(false), BuildMode(-1), RndSeed(0) {} }; /** * @brief Image resizer's filtering step class. * * Class defines data to perform a single filtering step over a whole * horizontal or vertical scanline. Resizing consists of 1 or more steps that * may be performed before the actual resizing takes place. Filtering may also * follow a resizing step. Each step must ensure that scanline data contains * enough pixels to perform the next step (which may be resizing) without * exceeding scanline's bounds. * * A derived class must implement several "const" and "static" functions that * are used to perform the actual filtering in interleaved or de-interleaved * mode. * * @tparam fptype Floating point type to use for storing pixel elements. SIMD * types can be used: in this case each element may hold a whole pixel. * @tparam fptypeatom The atomic type the "fptype" consists of. */ template class CImageResizerFilterStep { public: bool IsUpsample; ///< "True" if this step is an upsampling step, "false" ///< if downsampling step. Should be set to "false" if ///< ResampleFactor equals 0. ///< int ResampleFactor; ///< Resample factor (>=1). If 0, this is a resizing ///< step. This value should be >1 if IsUpsample equals ///< "true". ///< CBuffer Flt; ///< Filter to use at this step. ///< CFltBuffer FltOrig; ///< Originally-designed filter. This buffer may not ///< be assigned. Assigned by filters that precede the ///< resizing step if such filter is planned to be ///< embedded into the interpolation filter as "external" ///< filter. If IsUpsample=true and this filter buffer is ///< not empty, the upsampling step will not itself apply ///< any filtering over upsampled input scanline. ///< double DCGain; ///< DC gain which was applied to the filter. Not defined ///< if ResampleFactor = 0. ///< int FltLatency; ///< Filter's latency (group delay, shift) in pixels. ///< const CImageResizerVars* Vars; ///< Image resizing-related variables. ///< int InLen; ///< Input scanline's length in pixels. ///< int InBuf; ///< Input buffer index, 0 or 1. ///< int InPrefix; ///< Required input prefix pixels. These prefix pixels will ///< be filled with source scanline's first pixel value. If ///< IsUpsample is "true", this is the additional number of ///< times the first pixel will be filtered before processing ///< scanline, this number is also reflected in the OutPrefix. ///< int InSuffix; ///< Required input suffix pixels. These suffix pixels will ///< be filled with source scanline's last pixel value. If ///< IsUpsample is "true", this is the additional number of ///< times the last pixel will be filtered before processing ///< scanline, this number is also reflected in the OutSuffix. ///< int InElIncr; ///< Pixel element increment within the input buffer, used ///< during de-interleaved processing: in this case each ///< image's channel is stored independently, InElIncr elements ///< apart. ///< int OutLen; ///< Length of the resulting scanline. ///< int OutBuf; ///< Output buffer index. 0 or 1; 2 for the last step. ///< int OutPrefix; ///< Required output prefix pixels. These prefix pixels ///< will not be pre-filled with any values. Value is valid ///< only if IsUpsample equals "true". ///< int OutSuffix; ///< Required input suffix pixels. These suffix pixels will ///< not be pre-filled with any values. Value is valid only if ///< IsUpsample equals "true". ///< int OutElIncr; ///< Pixel element increment within the output buffer, used ///< during de-interleaved processing. Equals to the ///< InBufElIncr of the next step. ///< CBuffer PrefixDC; ///< DC component fluctuations added at the ///< start of the resulting scanline, used when ///< IsUpsample equals "true". ///< CBuffer SuffixDC; ///< DC component fluctuations added at the ///< end of the resulting scanline, used when ///< IsUpsample equals "true". ///< int EdgePixelCount; ///< The number of edge pixels added. Affects the ///< initial position within the input scanline, used to ///< produce edge pixels. This variable is used and ///< should be defined when IsUpsample=false and ///< ResampleFactor>0. When assigning this variable it is ///< also necessary to update InPrefix, OutLen and Vars.o ///< variables. ///< static const int EdgePixelCountDef = 3; ///< The default number of pixels ///< additionally produced at scanline edges during filtering. This is ///< required to reduce edge artifacts. ///< /** * @brief Resizing position structure. * * Structure holds resizing position and pointer to fractional delay * filter. */ struct CResizePos { int SrcPosInt; ///< Source scanline position. ///< int fti; ///< Fractional delay filter index. ///< const fptype* ftp; ///< Fractional delay filter pointer. ///< fptypeatom x; ///< Interpolation coefficient between delay filters. ///< int SrcOffs; ///< Source scanline offset. ///< }; /** * @brief Resizing positions buffer class. * * This class combines buffer together with variables that define resizing * stepping. */ class CRPosBuf : public CBuffer { public: double k; ///< Resizing step. ///< double o; ///< Resizing offset. ///< int FracCount; ///< The number of fractional delay filters in a filter ///< bank used together with this buffer. ///< }; /** * @brief Resizing positions buffer array class. * * This class combines structure array of the CRPosBuf class objects with * the function that locates or creates buffer with the required resizing * stepping. */ class CRPosBufArray : public CStructArray { public: using CStructArray::add; using CStructArray::getItemCount; /** * Function returns the resizing positions buffer with the required * stepping. If no such buffer exists, it is created. * * @param k Resizing step. * @param o Resizing offset. * @param FracCount The number of fractional delay filters in a filter * bank used together with this buffer. * @return Reference to the CRPosBuf object. */ CRPosBuf& getRPosBuf(const double k, const double o, const int FracCount) { int i; for (i = 0; i < getItemCount(); i++) { CRPosBuf& Buf = (*this)[i]; if (Buf.k == k && Buf.o == o && Buf.FracCount == FracCount) { return (Buf); } } CRPosBuf& NewBuf = add(); NewBuf.k = k; NewBuf.o = o; NewBuf.FracCount = FracCount; return (NewBuf); } }; CRPosBuf* RPosBuf; ///< Resizing positions buffer. Used when ///< ResampleFactor equals 0 (resizing step). ///< CDSPFracFilterBankLin* FltBank; ///< Filter bank in use by *this ///< resizing step. ///< }; /** * @brief Interleaved filtering steps implementation class. * * This class implements scanline filtering functions in interleaved mode. * This means that each pixel is processed independently, not in groups. * * @tparam fptype Floating point type to use for storing pixel elements. SIMD * types can be used: in this case each element may hold a whole pixel. * @tparam fptypeatom The atomic type the "fptype" consists of. */ template class CImageResizerFilterStepINL : public CImageResizerFilterStep { public: using CImageResizerFilterStep::IsUpsample; using CImageResizerFilterStep::ResampleFactor; using CImageResizerFilterStep::Flt; using CImageResizerFilterStep::FltOrig; using CImageResizerFilterStep::FltLatency; using CImageResizerFilterStep::Vars; using CImageResizerFilterStep::InLen; using CImageResizerFilterStep::InPrefix; using CImageResizerFilterStep::InSuffix; using CImageResizerFilterStep::OutLen; using CImageResizerFilterStep::OutPrefix; using CImageResizerFilterStep::OutSuffix; using CImageResizerFilterStep::PrefixDC; using CImageResizerFilterStep::SuffixDC; using CImageResizerFilterStep::RPosBuf; using CImageResizerFilterStep::FltBank; using CImageResizerFilterStep::EdgePixelCount; /** * Function performs "packing" of a scanline and type conversion. * Scanline, depending on the "fptype" can be potentially stored as a * packed SIMD values having a certain atomic type. If required, the sRGB * gamma correction is applied. * * @param ip Input scanline. * @param op0 Output scanline. * @param l0 The number of pixels to "pack". */ template void packScanline(const Tin* ip, fptype* const op0, const int l0) const { const int ElCount = Vars->ElCount; const int ElCountIO = Vars->ElCountIO; fptype* op = op0; int l = l0; if (!Vars->UseSRGBGamma) { if (ElCountIO == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; op += ElCount; ip++; l--; } } else if (ElCountIO == 4) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; v[1] = (fptypeatom)ip[1]; v[2] = (fptypeatom)ip[2]; v[3] = (fptypeatom)ip[3]; op += ElCount; ip += 4; l--; } } else if (ElCountIO == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; v[1] = (fptypeatom)ip[1]; v[2] = (fptypeatom)ip[2]; op += ElCount; ip += 3; l--; } } else if (ElCountIO == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = (fptypeatom)ip[0]; v[1] = (fptypeatom)ip[1]; op += ElCount; ip += 2; l--; } } } else { const fptypeatom gm = (fptypeatom)Vars->InGammaMult; if (ElCountIO == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); op += ElCount; ip++; l--; } } else if (ElCountIO == 4) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); v[1] = convertSRGB2Lin((fptypeatom)ip[1] * gm); v[2] = convertSRGB2Lin((fptypeatom)ip[2] * gm); v[3] = convertSRGB2Lin((fptypeatom)ip[3] * gm); op += ElCount; ip += 4; l--; } } else if (ElCountIO == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); v[1] = convertSRGB2Lin((fptypeatom)ip[1] * gm); v[2] = convertSRGB2Lin((fptypeatom)ip[2] * gm); op += ElCount; ip += 3; l--; } } else if (ElCountIO == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)op; v[0] = convertSRGB2Lin((fptypeatom)ip[0] * gm); v[1] = convertSRGB2Lin((fptypeatom)ip[1] * gm); op += ElCount; ip += 2; l--; } } } const int ZeroCount = ElCount * Vars->fppack - ElCountIO; op = op0; l = l0; if (ZeroCount == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)op + ElCountIO; v[0] = (fptypeatom)0; op += ElCount; l--; } } else if (ZeroCount == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)op + ElCountIO; v[0] = (fptypeatom)0; v[1] = (fptypeatom)0; op += ElCount; l--; } } else if (ZeroCount == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)op + ElCountIO; v[0] = (fptypeatom)0; v[1] = (fptypeatom)0; v[2] = (fptypeatom)0; op += ElCount; l--; } } } /** * Function applies Linear to sRGB gamma correction to the specified * scanline. * * @param p Scanline. * @param l The number of pixels to de-linearize. * @param Vars0 Image resizing-related variables. */ static void applySRGBGamma(fptype* p, int l, const CImageResizerVars& Vars0) { const int ElCount = Vars0.ElCount; const int ElCountIO = Vars0.ElCountIO; const fptypeatom gm = (fptypeatom)Vars0.OutGammaMult; if (ElCountIO == 1) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; p += ElCount; l--; } } else if (ElCountIO == 4) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; v[1] = convertLin2SRGB(v[1]) * gm; v[2] = convertLin2SRGB(v[2]) * gm; v[3] = convertLin2SRGB(v[3]) * gm; p += ElCount; l--; } } else if (ElCountIO == 3) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; v[1] = convertLin2SRGB(v[1]) * gm; v[2] = convertLin2SRGB(v[2]) * gm; p += ElCount; l--; } } else if (ElCountIO == 2) { while (l > 0) { fptypeatom* v = (fptypeatom*)p; v[0] = convertLin2SRGB(v[0]) * gm; v[1] = convertLin2SRGB(v[1]) * gm; p += ElCount; l--; } } } /** * Function converts vertical scanline to horizontal scanline. This * function is called by the image resizer when image is resized * vertically. This means that the vertical scanline is stored in the * same format produced by the packScanline() and maintained by other * filtering functions. * * @param ip Input vertical scanline. * @param op Output buffer (temporary buffer used during resizing). * @param SrcLen The number of pixels in the input scanline, also used to * calculate input buffer increment. * @param SrcIncr Input buffer increment to the next vertical pixel. */ void convertVtoH(const fptype* ip, fptype* op, const int SrcLen, const int SrcIncr) const { const int ElCount = Vars->ElCount; int j; if (ElCount == 1) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; ip += SrcIncr; op++; } } else if (ElCount == 4) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; op[3] = ip[3]; ip += SrcIncr; op += 4; } } else if (ElCount == 3) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; op[1] = ip[1]; op[2] = ip[2]; ip += SrcIncr; op += 3; } } else if (ElCount == 2) { for (j = 0; j < SrcLen; j++) { op[0] = ip[0]; op[1] = ip[1]; ip += SrcIncr; op += 2; } } } /** * Function performs "unpacking" of a scanline and type conversion * (truncation is used when floating point is converted to integer). * Scanline, depending on the "fptype" can be potentially stored as a * packed SIMD values having a certain atomic type. The unpacking function * assumes that scanline is stored in the style produced by the * packScanline() function. * * @param ip Input scanline. * @param op Output scanline. * @param l The number of pixels to "unpack". * @param Vars0 Image resizing-related variables. */ template static void unpackScanline(const fptype* ip, Tout* op, int l, const CImageResizerVars& Vars0) { const int ElCount = Vars0.ElCount; const int ElCountIO = Vars0.ElCountIO; if (ElCountIO == 1) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; ip += ElCount; op++; l--; } } else if (ElCountIO == 4) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; op[1] = (Tout)v[1]; op[2] = (Tout)v[2]; op[3] = (Tout)v[3]; ip += ElCount; op += 4; l--; } } else if (ElCountIO == 3) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; op[1] = (Tout)v[1]; op[2] = (Tout)v[2]; ip += ElCount; op += 3; l--; } } else if (ElCountIO == 2) { while (l > 0) { const fptypeatom* v = (const fptypeatom*)ip; op[0] = (Tout)v[0]; op[1] = (Tout)v[1]; ip += ElCount; op += 2; l--; } } } /** * Function prepares input scanline buffer for *this filtering step. * Left- and right-most pixels are replicated to make sure no buffer * overrun happens. Such approach also allows to bypass any pointer * range checks. * * @param Src Source buffer. */ void prepareInBuf(fptype* Src) const { if (IsUpsample || InPrefix + InSuffix == 0) { return; } const int ElCount = Vars->ElCount; replicateArray(Src, ElCount, Src - ElCount, InPrefix, -ElCount); Src += (InLen - 1) * ElCount; replicateArray(Src, ElCount, Src + ElCount, InSuffix, ElCount); } /** * Function peforms scanline upsampling with filtering. * * @param Src Source scanline buffer (length = this -> InLen). Source * scanline increment will be equal to ElCount. * @param Dst Destination scanline buffer. */ void doUpsample(const fptype* const Src, fptype* const Dst) const { const int ElCount = Vars->ElCount; fptype* op0 = &Dst[-OutPrefix * ElCount]; memset(op0, 0, (OutPrefix + OutLen + OutSuffix) * ElCount * sizeof(fptype)); const fptype* ip = Src; const int opstep = ElCount * ResampleFactor; int l; if (FltOrig.getCapacity() > 0) { // Do not perform filtering, only upsample. op0 += (OutPrefix % ResampleFactor) * ElCount; l = OutPrefix / ResampleFactor; if (ElCount == 1) { while (l > 0) { op0[0] = ip[0]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0 += opstep; l--; } } else if (ElCount == 4) { while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0[3] = ip[3]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0[3] = ip[3]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0[3] = ip[3]; op0 += opstep; l--; } } else if (ElCount == 3) { while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0[2] = ip[2]; op0 += opstep; l--; } } else if (ElCount == 2) { while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while (l >= 0) { op0[0] = ip[0]; op0[1] = ip[1]; op0 += opstep; l--; } } return; } const fptype* const f = Flt; const int flen = Flt.getCapacity(); fptype* op; int i; if (ElCount == 1) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[i] += f[i] * ip[0]; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[i] += f[i] * ip[0]; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[i] += f[i] * ip[0]; } op0 += opstep; l--; } } else if (ElCount == 4) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op[3] += f[i] * ip[3]; op += 4; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op[3] += f[i] * ip[3]; op += 4; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op[3] += f[i] * ip[3]; op += 4; } op0 += opstep; l--; } } else if (ElCount == 3) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op += 3; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op += 3; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op[2] += f[i] * ip[2]; op += 3; } op0 += opstep; l--; } } else if (ElCount == 2) { l = InPrefix; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op += 2; } op0 += opstep; l--; } l = InLen - 1; while (l > 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op += 2; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while (l >= 0) { op = op0; for (i = 0; i < flen; i++) { op[0] += f[i] * ip[0]; op[1] += f[i] * ip[1]; op += 2; } op0 += opstep; l--; } } op = op0; const fptype* dc = SuffixDC; l = SuffixDC.getCapacity(); if (ElCount == 1) { for (i = 0; i < l; i++) { op[i] += ip[0] * dc[i]; } } else if (ElCount == 4) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; op[3] += ip[3] * dc[0]; dc++; op += 4; l--; } } else if (ElCount == 3) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; dc++; op += 3; l--; } } else if (ElCount == 2) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; dc++; op += 2; l--; } } ip = Src; op = Dst - InPrefix * opstep; dc = PrefixDC; l = PrefixDC.getCapacity(); if (ElCount == 1) { for (i = 0; i < l; i++) { op[i] += ip[0] * dc[i]; } } else if (ElCount == 4) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; op[3] += ip[3] * dc[0]; dc++; op += 4; l--; } } else if (ElCount == 3) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; op[2] += ip[2] * dc[0]; dc++; op += 3; l--; } } else if (ElCount == 2) { while (l > 0) { op[0] += ip[0] * dc[0]; op[1] += ip[1] * dc[0]; dc++; op += 2; l--; } } } /** * Function peforms scanline filtering with optional downsampling. * Function makes use of the symmetry of the filter. * * @param Src Source scanline buffer (length = this -> InLen). Source * scanline increment will be equal to ElCount. * @param Dst Destination scanline buffer. * @param DstIncr Destination scanline buffer increment, used for * horizontal or vertical scanline stepping. */ void doFilter(const fptype* const Src, fptype* Dst, const int DstIncr) const { const int ElCount = Vars->ElCount; const fptype* const f = &Flt[FltLatency]; const int flen = FltLatency + 1; const int ipstep = ElCount * ResampleFactor; const fptype* ip = Src - EdgePixelCount * ipstep; const fptype* ip1; const fptype* ip2; int l = OutLen; int i; if (ElCount == 1) { while (l > 0) { fptype s = f[0] * ip[0]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1++; ip2--; s += f[i] * (ip1[0] + ip2[0]); } Dst[0] = s; Dst += DstIncr; ip += ipstep; l--; } } else if (ElCount == 4) { while (l > 0) { fptype s1 = f[0] * ip[0]; fptype s2 = f[0] * ip[1]; fptype s3 = f[0] * ip[2]; fptype s4 = f[0] * ip[3]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1 += 4; ip2 -= 4; s1 += f[i] * (ip1[0] + ip2[0]); s2 += f[i] * (ip1[1] + ip2[1]); s3 += f[i] * (ip1[2] + ip2[2]); s4 += f[i] * (ip1[3] + ip2[3]); } Dst[0] = s1; Dst[1] = s2; Dst[2] = s3; Dst[3] = s4; Dst += DstIncr; ip += ipstep; l--; } } else if (ElCount == 3) { while (l > 0) { fptype s1 = f[0] * ip[0]; fptype s2 = f[0] * ip[1]; fptype s3 = f[0] * ip[2]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1 += 3; ip2 -= 3; s1 += f[i] * (ip1[0] + ip2[0]); s2 += f[i] * (ip1[1] + ip2[1]); s3 += f[i] * (ip1[2] + ip2[2]); } Dst[0] = s1; Dst[1] = s2; Dst[2] = s3; Dst += DstIncr; ip += ipstep; l--; } } else if (ElCount == 2) { while (l > 0) { fptype s1 = f[0] * ip[0]; fptype s2 = f[0] * ip[1]; ip1 = ip; ip2 = ip; for (i = 1; i < flen; i++) { ip1 += 2; ip2 -= 2; s1 += f[i] * (ip1[0] + ip2[0]); s2 += f[i] * (ip1[1] + ip2[1]); } Dst[0] = s1; Dst[1] = s2; Dst += DstIncr; ip += ipstep; l--; } } } /** * Function performs resizing of a single scanline. This function does * not "know" about the length of the source scanline buffer. This buffer * should be padded with enough pixels so that ( SrcPos - FilterLenD2 ) is * always >= 0 and ( SrcPos + ( DstLineLen - 1 ) * k + FilterLenD2 + 1 ) * does not exceed source scanline's buffer length. SrcLine's increment is * assumed to be equal to ElCount. * * @param SrcLine Source scanline buffer. * @param DstLine Destination (resized) scanline buffer. * @param DstLineIncr Destination scanline position increment, used for * horizontal or vertical scanline stepping. * @param xx Temporary buffer, of size FltBank -> getFilterLen(), must be * aligned by fpclass :: fpalign. */ void doResize(const fptype* SrcLine, fptype* DstLine, const int DstLineIncr, fptype* const) const { const int IntFltLen = FltBank->getFilterLen(); const int ElCount = Vars->ElCount; const typename CImageResizerFilterStep::CResizePos* rpos = &(*RPosBuf)[0]; const typename CImageResizerFilterStep< fptype, fptypeatom>::CResizePos* const rpose = rpos + OutLen; #define AVIR_RESIZE_PART1 \ while (rpos < rpose) { \ const fptype x = (fptype)rpos->x; \ const fptype* const ftp = rpos->ftp; \ const fptype* const ftp2 = ftp + IntFltLen; \ const fptype* Src = SrcLine + rpos->SrcOffs; \ int i; #define AVIR_RESIZE_PART1nx \ while (rpos < rpose) { \ const fptype* const ftp = rpos->ftp; \ const fptype* Src = SrcLine + rpos->SrcOffs; \ int i; #define AVIR_RESIZE_PART2 \ DstLine += DstLineIncr; \ rpos++; \ } if (FltBank->getOrder() == 1) { if (ElCount == 1) { AVIR_RESIZE_PART1 fptype sum = 0.0; for (i = 0; i < IntFltLen; i++) { sum += (ftp[i] + ftp2[i] * x) * Src[i]; } DstLine[0] = sum; AVIR_RESIZE_PART2 } else if (ElCount == 4) { AVIR_RESIZE_PART1 fptype sum[4]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; sum[3] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i] + ftp2[i] * x; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; sum[3] += xx * Src[3]; Src += 4; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; DstLine[3] = sum[3]; AVIR_RESIZE_PART2 } else if (ElCount == 3) { AVIR_RESIZE_PART1 fptype sum[3]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i] + ftp2[i] * x; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; Src += 3; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; AVIR_RESIZE_PART2 } else if (ElCount == 2) { AVIR_RESIZE_PART1 fptype sum[2]; sum[0] = 0.0; sum[1] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i] + ftp2[i] * x; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; Src += 2; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; AVIR_RESIZE_PART2 } } else { if (ElCount == 1) { AVIR_RESIZE_PART1nx fptype sum = 0.0; for (i = 0; i < IntFltLen; i++) { sum += ftp[i] * Src[i]; } DstLine[0] = sum; AVIR_RESIZE_PART2 } else if (ElCount == 4) { AVIR_RESIZE_PART1nx fptype sum[4]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; sum[3] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i]; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; sum[3] += xx * Src[3]; Src += 4; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; DstLine[3] = sum[3]; AVIR_RESIZE_PART2 } else if (ElCount == 3) { AVIR_RESIZE_PART1nx fptype sum[3]; sum[0] = 0.0; sum[1] = 0.0; sum[2] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i]; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; sum[2] += xx * Src[2]; Src += 3; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; DstLine[2] = sum[2]; AVIR_RESIZE_PART2 } else if (ElCount == 2) { AVIR_RESIZE_PART1nx fptype sum[2]; sum[0] = 0.0; sum[1] = 0.0; for (i = 0; i < IntFltLen; i++) { const fptype xx = ftp[i]; sum[0] += xx * Src[0]; sum[1] += xx * Src[1]; Src += 2; } DstLine[0] = sum[0]; DstLine[1] = sum[1]; AVIR_RESIZE_PART2 } } } #undef AVIR_RESIZE_PART2 #undef AVIR_RESIZE_PART1nx #undef AVIR_RESIZE_PART1 }; /** * @brief Image resizer's default dithering class. * * This class defines an object that performs rounding, clipping and dithering * operations over horizontal scanline pixels before scanline is stored in the * output buffer. * * The ditherer should expect the same storage order of the pixels in a * scanline as used in the "filtering step" class. So, a separate ditherer * class should be defined for each scanline pixel storage style. The default * ditherer implements a simple rounding without dithering: it can be used for * an efficient dithering method which can be multi-threaded. * * @tparam fptype Floating point type to use for storing pixel data. SIMD * types can be used. */ template class CImageResizerDithererDefINL { public: /** * Function initializes the ditherer object. * * @param aLen Scanline length in pixels to process. * @param aVars Image resizing-related variables. * @param aTrMul Bit-depth truncation multiplier. 1 - no additional * truncation. * @param aPkOut Peak output value allowed. */ void init(const int aLen, const CImageResizerVars& aVars, const double aTrMul, const double aPkOut) { Len = aLen; Vars = &aVars; LenE = aLen * Vars->ElCount; TrMul0 = aTrMul; PkOut0 = aPkOut; } /** * @return "True" if dithering is recursive relative to scanlines meaning * multi-threaded execution is not supported by this dithering method. */ static bool isRecursive() { return (false); } /** * Function performs rounding and clipping operations. * * @param ResScanline The buffer containing the final scanline. */ void dither(fptype* const ResScanline) const { const fptype c0 = 0.0; const fptype PkOut = (fptype)PkOut0; int j; if (TrMul0 == 1.0) { // Optimization - do not perform bit depth truncation. for (j = 0; j < LenE; j++) { ResScanline[j] = clamp(round(ResScanline[j]), c0, PkOut); } } else { const fptype TrMul = (fptype)TrMul0; for (j = 0; j < LenE; j++) { const fptype z0 = round(ResScanline[j] / TrMul) * TrMul; ResScanline[j] = clamp(z0, c0, PkOut); } } } protected: int Len; ///< Scanline's length in pixels. ///< const CImageResizerVars* Vars; ///< Image resizing-related variables. ///< int LenE; ///< = LenE * ElCount. ///< double TrMul0; ///< Bit-depth truncation multiplier. ///< double PkOut0; ///< Peak output value allowed. ///< }; /** * @brief Image resizer's error-diffusion dithering class, interleaved mode. * * This ditherer implements error-diffusion dithering which looks good, and * whose results are compressed by PNG well. This implementation uses * weighting coefficients obtained via machine optimization and visual * evaluation. * * @tparam fptype Floating point type to use for storing pixel data. SIMD * types can be used. */ template class CImageResizerDithererErrdINL : public CImageResizerDithererDefINL { public: /** * Function initializes the ditherer object. * * @param aLen Scanline length in pixels to process. * @param aVars Image resizing-related variables. * @param aTrMul Bit-depth truncation multiplier. 1 - no additional * truncation. * @param aPkOut Peak output value allowed. */ void init(const int aLen, const CImageResizerVars& aVars, const double aTrMul, const double aPkOut) { CImageResizerDithererDefINL::init(aLen, aVars, aTrMul, aPkOut); ResScanlineDith0.alloc(LenE + Vars->ElCount, sizeof(fptype)); ResScanlineDith = ResScanlineDith0 + Vars->ElCount; int i; for (i = 0; i < LenE + Vars->ElCount; i++) { ResScanlineDith0[i] = 0.0; } } static bool isRecursive() { return (true); } void dither(fptype* const ResScanline) { const int ElCount = Vars->ElCount; const fptype c0 = 0.0; const fptype TrMul = (fptype)TrMul0; const fptype PkOut = (fptype)PkOut0; int j; for (j = 0; j < LenE; j++) { ResScanline[j] += ResScanlineDith[j]; ResScanlineDith[j] = 0.0; } for (j = 0; j < LenE - ElCount; j++) { // Perform rounding, noise estimation and saturation. const fptype z0 = round(ResScanline[j] / TrMul) * TrMul; const fptype Noise = ResScanline[j] - z0; ResScanline[j] = clamp(z0, c0, PkOut); ResScanline[j + ElCount] += Noise * (fptype)0.364842; ResScanlineDith[j - ElCount] += Noise * (fptype)0.207305; ResScanlineDith[j] += Noise * (fptype)0.364842; ResScanlineDith[j + ElCount] += Noise * (fptype)0.063011; } while (j < LenE) { const fptype z0 = round(ResScanline[j] / TrMul) * TrMul; const fptype Noise = ResScanline[j] - z0; ResScanline[j] = clamp(z0, c0, PkOut); ResScanlineDith[j - ElCount] += Noise * (fptype)0.207305; ResScanlineDith[j] += Noise * (fptype)0.364842; j++; } } protected: using CImageResizerDithererDefINL::Len; using CImageResizerDithererDefINL::Vars; using CImageResizerDithererDefINL::LenE; using CImageResizerDithererDefINL::TrMul0; using CImageResizerDithererDefINL::PkOut0; CBuffer ResScanlineDith0; ///< Error diffusion buffer. ///< fptype* ResScanlineDith; ///< Error diffusion buffer pointer which skips ///< the first ElCount elements. ///< }; /** * @brief Floating-point processing definition and abstraction class. * * This class defines several constants and typedefs that point to classes * that should be used by the image resizing algorithm. Such "definition * class" can be used to define alternative scanline processing algorithms * (e.g. SIMD) and image scanline packing styles used during processing. This * class also offers an abstraction layer for dithering, rounding and * clamping (saturation) operation. * * The fpclass_def class can be used to define processing using both SIMD and * non-SIMD types, but using algorithms that are operate on interleaved pixels * and non-SIMD optimized themselves. * * @tparam afptype Floating point type to use for storing intermediate data * and variables. For variables that are not used in intensive calculations * the "double" type is always used. On the latest Intel processors (like * i7-4770K) there is almost no performance difference between "double" and * "float". Image quality differences between "double" and "float" are not * apparent on 8-bit images. At the same time the "float" uses half amount of * working memory the "double" type uses. SIMD types can be used. The * functions round() and clamp() in the "avir" or other visible namespace * should be available for the specified type. SIMD types allow to perform * resizing of images with more than 4 channels, to be exact 4 * SIMD element * number (e.g. 16 for float4), without modification of the image resizing * algorithm required. * @tparam afptypeatom The atomic type the "afptype" consists of. * @tparam adith Ditherer class to use during processing. */ template > class fpclass_def { public: typedef afptype fptype; ///< Floating-point type to use during processing. ///< typedef afptypeatom fptypeatom; ///< Atomic type "fptype" consists of. ///< static const int fppack = sizeof(fptype) / sizeof(fptypeatom); ///< ///< The number of atomic types stored in a single ///< "fptype" element. ///< static const int fpalign = sizeof(fptype); ///< Suggested alignment size ///< in bytes. This is not a required alignment, because ///< image resizing algorithm cannot be made to have a ///< strictly aligned data access at all steps (e.g. ///< interpolation cannot perform aligned accesses). ///< static const int elalign = 1; ///< Length alignment of arrays of elements. ///< This applies to filters and intermediate buffers: this constant ///< forces filters and scanlines to have a length which is a multiple ///< of this value, for more efficient SIMD implementation. ///< static const int packmode = 0; ///< 0 if interleaved packing, 1 if ///< de-interleaved. ///< typedef CImageResizerFilterStepINL CFilterStep; ///< ///< Filtering step class to use during processing. ///< typedef adith CDitherer; ///< Ditherer class to use during processing. ///< }; /** * @brief Image resizer class. * * The object of this class can be used to resize 1-4 channel images to any * required size. Resizing is performed by utilizing interpolated sinc * fractional delay filters plus (if necessary) a cascade of built-in * sinc function-based 2X upsampling or 2X downsampling stages, followed by a * correction filtering. * * Object of this class can be allocated on stack. * * @tparam fpclass Floating-point processing definition class to use. See * avir::fpclass_def for more details. */ template > class CImageResizer { public: /** * Constructor initializes the resizer. * * @param aResBitDepth Required bit depth of resulting image (1-16). If * integer value output is used (e.g. uint8_t), the bit depth also affects * rounding: for example, if aResBitDepth=6 and "Tout" is uint8_t, the * result will be rounded to 6 most significant bits (2 least significant * bits truncated, with dithering applied). * @param aSrcBitDepth Source image's real bit-depth. Set to 0 to use * aResBitDepth. * @param aParams Resizing algorithm's parameters to use. Leave out for * default values. Can be useful when performing automatic optimization of * parameters. */ CImageResizer(const int aResBitDepth = 8, const int aSrcBitDepth = 0, const CImageResizerParams& aParams = CImageResizerParamsDef()) : Params(aParams), ResBitDepth(aResBitDepth) { SrcBitDepth = (aSrcBitDepth == 0 ? ResBitDepth : aSrcBitDepth); initFilterBank(FixedFilterBank, 1.0, false, CFltBuffer()); FixedFilterBank.createAllFilters(); } /** * Function resizes image. * * @param SrcBuf Source image buffer. * @param SrcWidth Source image width. * @param SrcHeight Source image height. * @param SrcScanlineSize Physical size of source scanline in elements * (not bytes). If this value is below 1, SrcWidth * ElCountIO will be * used as the physical source scanline size. * @param[out] NewBuf Buffer to accept the resized image. Can be equal to * SrcBuf if the size of the resized image is smaller or equal to source * image in size. * @param NewWidth New image width. * @param NewHeight New image height. * @param ElCountIO The number of elements (channels) used to store each * source and destination pixel (1-4). * @param k Resizing step (one output pixel corresponds to "k" input * pixels). A downsizing factor if > 1.0; upsizing factor if <= 1.0. * Multiply by -1 if you would like to bypass "ox" and "oy" adjustment * which is done by default to produce a centered image. If step value * equals 0, the step value will be chosen automatically and independently * for horizontal and vertical resizing. * @param[in,out] aVars Pointer to variables structure to be passed to the * image resizing function. Can be NULL. Only variables that are * initialized in default constructor of this structure are accepted by * this function. These variables will not be changed by this function. * All other variables can be modified by this function. The access to * this object is not thread-safe, each concurrent instance of this * function should use a separate aVars object. * @tparam Tin Input buffer element's type. Can be uint8_t (0-255 value * range), uint16_t (0-65535 value range), float (0.0-1.0 value range), * double (0.0-1.0 value range). Larger integer types are treated as * uint16_t. Signed integer types are unsupported. * @tparam Tout Output buffer element's type. Can be uint8_t (0-255 value * range), uint16_t (0-65535 value range), float (0.0-1.0 value range), * double (0.0-1.0 value range). Larger integer types are treated as * uint16_t. Signed integer types are unsupported. */ template void resizeImage(const Tin* const SrcBuf, const int SrcWidth, const int SrcHeight, int SrcScanlineSize, Tout* const NewBuf, const int NewWidth, const int NewHeight, const int ElCountIO, const double k, CImageResizerVars* const aVars = NULL) const { if (SrcWidth == 0 || SrcHeight == 0) { memset(NewBuf, 0, (size_t)NewWidth * NewHeight * sizeof(Tout)); return; } else if (NewWidth == 0 || NewHeight == 0) { return; } CImageResizerVars DefVars; CImageResizerVars& Vars = (aVars == NULL ? DefVars : *aVars); CImageResizerThreadPool DefThreadPool; CImageResizerThreadPool& ThreadPool = (Vars.ThreadPool == NULL ? DefThreadPool : *Vars.ThreadPool); // Define resizing steps, also optionally modify offsets so that // resizing produces a "centered" image. double kx; double ky; double ox = Vars.ox; double oy = Vars.oy; if (k == 0.0) { if (NewWidth > SrcWidth) { kx = (double)(SrcWidth - 1) / (NewWidth - 1); } else { kx = (double)SrcWidth / NewWidth; ox += (kx - 1.0) * 0.5; } if (NewHeight > SrcHeight) { ky = (double)(SrcHeight - 1) / (NewHeight - 1); } else { ky = (double)SrcHeight / NewHeight; oy += (ky - 1.0) * 0.5; } } else if (k > 0.0) { kx = k; ky = k; if (k > 1.0) { const double ko = (k - 1.0) * 0.5; ox += ko; oy += ko; } } else { kx = -k; ky = -k; } // Evaluate pre-multipliers used on the output stage. const bool IsInFloat = ((Tin)0.4 != 0); const bool IsOutFloat = ((Tout)0.4 != 0); double OutMul; // Output multiplier. if (Vars.UseSRGBGamma) { if (IsInFloat) { Vars.InGammaMult = 1.0; } else { Vars.InGammaMult = 1.0 / (sizeof(Tin) == 1 ? 255.0 : 65535.0); } if (IsOutFloat) { Vars.OutGammaMult = 1.0; } else { Vars.OutGammaMult = (sizeof(Tout) == 1 ? 255.0 : 65535.0); } OutMul = 1.0; } else { if (IsOutFloat) { OutMul = 1.0; } else { OutMul = (sizeof(Tout) == 1 ? 255.0 : 65535.0); } if (!IsInFloat) { OutMul /= (sizeof(Tin) == 1 ? 255.0 : 65535.0); } } // Fill widely-used variables. const int ElCount = (ElCountIO + fpclass ::fppack - 1) / fpclass ::fppack; const int NewWidthE = NewWidth * ElCount; if (SrcScanlineSize < 1) { SrcScanlineSize = SrcWidth * ElCountIO; } Vars.ElCount = ElCount; Vars.ElCountIO = ElCountIO; Vars.fppack = fpclass ::fppack; Vars.fpalign = fpclass ::fpalign; Vars.elalign = fpclass ::elalign; Vars.packmode = fpclass ::packmode; // Horizontal scanline filtering and resizing. CDSPFracFilterBankLin FltBank; CFilterSteps FltSteps; typename CFilterStep ::CRPosBufArray RPosBufArray; CBuffer UsedFracMap; // Perform the filtering steps modeling at various modes, find the // most efficient mode for both horizontal and vertical resizing. int UseBuildMode = 1; const int BuildModeCount = (FixedFilterBank.getOrder() == 0 ? 4 : 2); int m; if (Vars.BuildMode >= 0) { UseBuildMode = Vars.BuildMode; } else { int BestScore = 0x7FFFFFFF; for (m = 0; m < BuildModeCount; m++) { CDSPFracFilterBankLin TmpBank; CFilterSteps TmpSteps; Vars.k = kx; Vars.o = ox; buildFilterSteps(TmpSteps, Vars, TmpBank, OutMul, m, true); updateFilterStepBuffers(TmpSteps, Vars, RPosBufArray, SrcWidth, NewWidth); fillUsedFracMap(TmpSteps[Vars.ResizeStep], UsedFracMap); const int c = calcComplexity(TmpSteps, Vars, UsedFracMap, SrcHeight); if (c < BestScore) { UseBuildMode = m; BestScore = c; } } } // Perform the actual filtering steps building. Vars.k = kx; Vars.o = ox; buildFilterSteps(FltSteps, Vars, FltBank, OutMul, UseBuildMode, false); updateFilterStepBuffers(FltSteps, Vars, RPosBufArray, SrcWidth, NewWidth); updateBufLenAndRPosPtrs(FltSteps, Vars, NewWidth); const int ThreadCount = ThreadPool.getSuggestedWorkloadCount(); // Includes the current thread. CStructArray > td; td.setItemCount(ThreadCount); int i; for (i = 0; i < ThreadCount; i++) { if (i > 0) { ThreadPool.addWorkload(&td[i]); } td[i].init(i, ThreadCount, FltSteps, Vars); td[i].initScanlineQueue(td[i].sopResizeH, SrcHeight, SrcWidth); } CBuffer FltBuf( (size_t)NewWidthE * SrcHeight, fpclass ::fpalign); // Temporary buffer that receives // horizontally-filtered and resized image. for (i = 0; i < SrcHeight; i++) { td[i % ThreadCount].addScanlineToQueue( (void*)&SrcBuf[(size_t)i * SrcScanlineSize], &FltBuf[(size_t)i * NewWidthE]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); // Vertical scanline filtering and resizing, reuse previously defined // filtering steps if possible. const int PrevUseBuildMode = UseBuildMode; if (Vars.BuildMode >= 0) { UseBuildMode = Vars.BuildMode; } else { CImageResizerVars TmpVars(Vars); int BestScore = 0x7FFFFFFF; for (m = 0; m < BuildModeCount; m++) { CDSPFracFilterBankLin TmpBank; TmpBank.copyInitParams(FltBank); CFilterSteps TmpSteps; TmpVars.k = ky; TmpVars.o = oy; buildFilterSteps(TmpSteps, TmpVars, TmpBank, 1.0, m, true); updateFilterStepBuffers(TmpSteps, TmpVars, RPosBufArray, SrcHeight, NewHeight); fillUsedFracMap(TmpSteps[TmpVars.ResizeStep], UsedFracMap); const int c = calcComplexity(TmpSteps, TmpVars, UsedFracMap, NewWidth); if (c < BestScore) { UseBuildMode = m; BestScore = c; } } } Vars.k = ky; Vars.o = oy; if (UseBuildMode == PrevUseBuildMode && ky == kx) { if (OutMul != 1.0) { modifyCorrFilterDCGain(FltSteps, 1.0 / OutMul); } } else { buildFilterSteps(FltSteps, Vars, FltBank, 1.0, UseBuildMode, false); } updateFilterStepBuffers(FltSteps, Vars, RPosBufArray, SrcHeight, NewHeight); updateBufLenAndRPosPtrs(FltSteps, Vars, NewWidth); if (IsOutFloat && sizeof(FltBuf[0]) == sizeof(Tout) && fpclass ::packmode == 0) { // In-place output. for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopResizeV, NewWidth, SrcHeight, NewWidthE, NewWidthE); } for (i = 0; i < NewWidth; i++) { td[i % ThreadCount].addScanlineToQueue( &FltBuf[(size_t)i * ElCount], (fptype*)&NewBuf[(size_t)i * ElCount]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); ThreadPool.removeAllWorkloads(); return; } CBuffer ResBuf((size_t)NewWidthE * NewHeight, fpclass ::fpalign); for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopResizeV, NewWidth, SrcHeight, NewWidthE, NewWidthE); } const int im = (fpclass ::packmode == 0 ? ElCount : 1); for (i = 0; i < NewWidth; i++) { td[i % ThreadCount].addScanlineToQueue(&FltBuf[(size_t)i * im], &ResBuf[(size_t)i * im]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); if (IsOutFloat) { // Perform output, but skip dithering. for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopUnpackH, NewHeight, NewWidth); } for (i = 0; i < NewHeight; i++) { td[i % ThreadCount].addScanlineToQueue( &ResBuf[(size_t)i * NewWidthE], &NewBuf[(size_t)i * NewWidth * ElCountIO]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); ThreadPool.removeAllWorkloads(); return; } // Perform output with dithering (for integer output only). int TruncBits; // The number of lower bits to truncate and dither. int OutRange; // Output range. if (sizeof(Tout) == 1) { TruncBits = 8 - ResBitDepth; OutRange = 255; } else { TruncBits = 16 - ResBitDepth; OutRange = 65535; } const double PkOut = OutRange; const double TrMul = (TruncBits > 0 ? PkOut / (OutRange >> TruncBits) : 1.0); if (CDitherer ::isRecursive()) { td[0].getDitherer().init(NewWidth, Vars, TrMul, PkOut); if (Vars.UseSRGBGamma) { for (i = 0; i < NewHeight; i++) { fptype* const ResScanline = &ResBuf[(size_t)i * NewWidthE]; CFilterStep ::applySRGBGamma(ResScanline, NewWidth, Vars); td[0].getDitherer().dither(ResScanline); CFilterStep ::unpackScanline( ResScanline, &NewBuf[(size_t)i * NewWidth * ElCountIO], NewWidth, Vars); } } else { for (i = 0; i < NewHeight; i++) { fptype* const ResScanline = &ResBuf[(size_t)i * NewWidthE]; td[0].getDitherer().dither(ResScanline); CFilterStep ::unpackScanline( ResScanline, &NewBuf[(size_t)i * NewWidth * ElCountIO], NewWidth, Vars); } } } else { for (i = 0; i < ThreadCount; i++) { td[i].initScanlineQueue(td[i].sopDitherAndUnpackH, NewHeight, NewWidth); td[i].getDitherer().init(NewWidth, Vars, TrMul, PkOut); } for (i = 0; i < NewHeight; i++) { td[i % ThreadCount].addScanlineToQueue( &ResBuf[(size_t)i * NewWidthE], &NewBuf[(size_t)i * NewWidth * ElCountIO]); } ThreadPool.startAllWorkloads(); td[0].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); } ThreadPool.removeAllWorkloads(); } private: typedef typename fpclass ::fptype fptype; ///< Floating-point type to use ///< during processing. ///< typedef typename fpclass ::CFilterStep CFilterStep; ///< Filtering step ///< class to use during processing. ///< typedef typename fpclass ::CDitherer CDitherer; ///< Ditherer class to ///< use during processing. ///< CImageResizerParams Params; ///< Algorithm's parameters currently in use. ///< int SrcBitDepth; ///< Bit resolution of the source image. ///< int ResBitDepth; ///< Bit resolution of the resulting image. ///< CDSPFracFilterBankLin FixedFilterBank; ///< Fractional delay ///< filter bank with fixed characteristics, mainly for ///< upsizing cases. ///< /** * @brief Filtering steps array. * * The object of this class stores filtering steps together. */ typedef CStructArray CFilterSteps; /** * Function initializes the filter bank in the specified resizing step * according to the source and resulting image bit depths. * * @param FltBank Filter bank to initialize. * @param CutoffMult Cutoff multiplier, 0 to 1. 1 corresponds to 0.5pi * cutoff point. * @param ForceHiOrder "True" if a high-order interpolation should be * forced which requires considerably less resources for initialization. * @param ExtFilter External filter to apply to interpolation filter. */ void initFilterBank(CDSPFracFilterBankLin& FltBank, const double CutoffMult, const bool ForceHiOrder, const CFltBuffer& ExtFilter) const { const int IntBitDepth = (ResBitDepth > SrcBitDepth ? ResBitDepth : SrcBitDepth); const double SNR = -6.02 * (IntBitDepth + 3); int UseOrder; int FracCount; // The number of fractional delay filters sampled by // the filter bank. This variable affects the // signal-to-noise ratio at interpolation stage. // Theoretically, at UseOrder==1, 8-bit image resizing // requires 66.2 dB SNR or 11. 16-bit resizing requires // 114.4 dB SNR or 150. At UseOrder=0 the required number of // filters is exponentially higher. if (ForceHiOrder || IntBitDepth > 8) { UseOrder = 1; // -146 dB max FracCount = (int)ceil(0.23134052 * exp(-0.058062929 * SNR)); } else { UseOrder = 0; // -72 dB max FracCount = (int)ceil(0.33287686 * exp(-0.11334583 * SNR)); } if (FracCount < 2) { FracCount = 2; } FltBank.init(FracCount, UseOrder, Params.IntFltLen / CutoffMult, Params.IntFltCutoff * CutoffMult, Params.IntFltAlpha, ExtFilter, fpclass ::fpalign, fpclass ::elalign); } /** * Function allocates filter buffer taking "fpclass" alignments into * account. The allocated buffer may be larger than the requested size: in * this case the additional elements will be zeroed by this function. * * @param Flt Filter buffer. * @param ReqCapacity The required filter buffer's capacity. * @param IsModel "True" if filtering steps modeling is performed without * actual filter allocation. * @param FltExt If non-NULL this variable will receive the number of * elements the filter was extended by. */ static void allocFilter(CBuffer& Flt, const int ReqCapacity, const bool IsModel = false, int* const FltExt = NULL) { int UseCapacity = (ReqCapacity + fpclass ::elalign - 1) & ~(fpclass ::elalign - 1); int Ext = UseCapacity - ReqCapacity; if (FltExt != NULL) { *FltExt = Ext; } if (IsModel) { Flt.forceCapacity(UseCapacity); return; } Flt.alloc(UseCapacity, fpclass ::fpalign); while (Ext > 0) { Ext--; Flt[ReqCapacity + Ext] = 0.0; } } /** * Function assigns filter parameters to the specified filtering step * object. * * @param fs Filtering step to assign parameter to. This step cannot be * the last step if ResampleFactor greater than 1 was specified. * @param IsUpsample "True" if upsampling step. Should be set to "false" * if FltCutoff is negative. * @param ResampleFactor Resampling factor of this filter (>=1). * @param FltCutoff Filter cutoff point. This value will be divided by the * ResampleFactor if IsUpsample equals "true". If zero value was * specified, the "half-band" predefined filter will be created. In this * case the ResampleFactor will modify the filter cutoff point. * @param DCGain DC gain to apply to the filter. Assigned to filtering * step's DCGain variable. * @param UseFltOrig "True" if the originally-designed filter should be * left in filtering step's FltOrig buffer. Otherwise it will be freed. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ void assignFilterParams(CFilterStep& fs, const bool IsUpsample, const int ResampleFactor, const double FltCutoff, const double DCGain, const bool UseFltOrig, const bool IsModel) const { double FltAlpha; double Len2; double Freq; if (FltCutoff == 0.0) { const double m = 2.0 / ResampleFactor; FltAlpha = Params.HBFltAlpha; Len2 = 0.5 * Params.HBFltLen / m; Freq = AVIR_PI * Params.HBFltCutoff * m; } else { FltAlpha = Params.LPFltAlpha; Len2 = 0.25 * Params.LPFltBaseLen / FltCutoff; Freq = AVIR_PI * Params.LPFltCutoffMult * FltCutoff; } if (IsUpsample) { Len2 *= ResampleFactor; Freq /= ResampleFactor; fs.DCGain = DCGain * ResampleFactor; } else { fs.DCGain = DCGain; } fs.FltOrig.Len2 = Len2; fs.FltOrig.Freq = Freq; fs.FltOrig.Alpha = FltAlpha; fs.FltOrig.DCGain = fs.DCGain; CDSPPeakedCosineLPF w(Len2, Freq, FltAlpha); fs.IsUpsample = IsUpsample; fs.ResampleFactor = ResampleFactor; fs.FltLatency = w.fl2; int FltExt; // Filter's extension due to fpclass :: elalign. if (IsModel) { allocFilter(fs.Flt, w.FilterLen, true, &FltExt); if (UseFltOrig) { // Allocate a real buffer even in modeling mode since this // filter may be copied by the filter bank. fs.FltOrig.alloc(w.FilterLen); memset(&fs.FltOrig[0], 0, w.FilterLen * sizeof(fs.FltOrig[0])); } } else { fs.FltOrig.alloc(w.FilterLen); w.generateLPF(&fs.FltOrig[0], 1.0); optimizeFIRFilter(fs.FltOrig, fs.FltLatency); normalizeFIRFilter(&fs.FltOrig[0], fs.FltOrig.getCapacity(), fs.DCGain); allocFilter(fs.Flt, fs.FltOrig.getCapacity(), false, &FltExt); copyArray(&fs.FltOrig[0], &fs.Flt[0], fs.FltOrig.getCapacity()); if (!UseFltOrig) { fs.FltOrig.free(); } } if (IsUpsample) { int l = fs.Flt.getCapacity() - fs.FltLatency - ResampleFactor - FltExt; allocFilter(fs.PrefixDC, l, IsModel); allocFilter(fs.SuffixDC, fs.FltLatency, IsModel); if (IsModel) { return; } // Create prefix and suffix "tails" used during upsampling. const fptype* ip = &fs.Flt[fs.FltLatency + ResampleFactor]; copyArray(ip, &fs.PrefixDC[0], l); while (true) { ip += ResampleFactor; l -= ResampleFactor; if (l <= 0) { break; } addArray(ip, &fs.PrefixDC[0], l); } l = fs.FltLatency; fptype* op = &fs.SuffixDC[0]; copyArray(&fs.Flt[0], op, l); while (true) { op += ResampleFactor; l -= ResampleFactor; if (l <= 0) { break; } addArray(&fs.Flt[0], op, l); } } else if (!UseFltOrig) { fs.EdgePixelCount = fs.EdgePixelCountDef; } } /** * Function adds a correction filter that tries to achieve a linear * frequency response at all frequencies. The actual resulting response * may feature a slight damping of the highest frequencies since a * suitably short correction filter cannot fix steep high-frequency * damping. * * This function assumes that the resizing step is currently the last * step, even if it was not inserted yet: this allows placement of the * correction filter both before and after the resizing step. * * @param Steps Filtering steps. * @param bw Resulting bandwidth relative to the original bandwidth (which * is 1.0), usually 1/k. Should be <= 1.0. * @param IsPreCorrection "True" if the filtering step was already created * and it is first in the Steps array. "True" also adds edge pixels to * reduce edge artifacts. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ void addCorrectionFilter(CFilterSteps& Steps, const double bw, const bool IsPreCorrection, const bool IsModel) const { CFilterStep& fs = (IsPreCorrection ? Steps[0] : Steps.add()); fs.IsUpsample = false; fs.ResampleFactor = 1; fs.DCGain = 1.0; fs.EdgePixelCount = (IsPreCorrection ? fs.EdgePixelCountDef : 0); if (IsModel) { allocFilter( fs.Flt, CDSPFIREQ ::calcFilterLength(Params.CorrFltLen, fs.FltLatency), true); return; } const int BinCount = 65; // Frequency response bins to control. const int BinCount1 = BinCount - 1; double curbw = 1.0; // Bandwidth of the filter at the current step. int i; int j; double re; double im; CBuffer Bins(BinCount); // Adjustment introduced by all // steps at all frequencies of interest. for (j = 0; j < BinCount; j++) { Bins[j] = 1.0; } const int si = (IsPreCorrection ? 1 : 0); for (i = si; i < Steps.getItemCount() - (si ^ 1); i++) { const CFilterStep& fs = Steps[i]; if (fs.IsUpsample) { curbw *= fs.ResampleFactor; if (fs.FltOrig.getCapacity() > 0) { continue; } } const double dcg = 1.0 / fs.DCGain; // DC gain correction. const fptype* Flt; int FltLen; if (fs.ResampleFactor == 0) { Flt = fs.FltBank->getFilter(0); FltLen = fs.FltBank->getFilterLen(); } else { Flt = &fs.Flt[0]; FltLen = fs.Flt.getCapacity(); } // Calculate frequency response adjustment introduced by the // filter at this step, within the bounds of bandwidth of // interest. for (j = 0; j < BinCount; j++) { const double th = AVIR_PI * bw / curbw * j / BinCount1; calcFIRFilterResponse(Flt, FltLen, th, re, im); Bins[j] /= sqrt(re * re + im * im) * dcg; } if (!fs.IsUpsample && fs.ResampleFactor > 1) { curbw /= fs.ResampleFactor; } } // Calculate filter. CDSPFIREQ EQ; EQ.init(bw * 2.0, Params.CorrFltLen, BinCount, 0.0, bw, false, Params.CorrFltAlpha); fs.FltLatency = EQ.getFilterLatency(); CBuffer Filter(EQ.getFilterLength()); EQ.buildFilter(Bins, &Filter[0]); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); optimizeFIRFilter(Filter, fs.FltLatency); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); allocFilter(fs.Flt, Filter.getCapacity()); copyArray(&Filter[0], &fs.Flt[0], Filter.getCapacity()); // Print a theoretically achieved final frequency response at various // feature sizes (from DC to 1 pixel). Values above 255 means features // become brighter, values below 255 means features become dimmer. /* const double sbw = ( bw > 1.0 ? 1.0 / bw : 1.0 ); for( j = 0; j < BinCount; j++ ) { const double th = AVIR_PI * sbw * j / BinCount1; calcFIRFilterResponse( &fs.Flt[ 0 ], fs.Flt.getCapacity(), th, re, im ); printf( "%f\n", sqrt( re * re + im * im ) / Bins[ j ] * 255 ); } printf( "***\n" );*/ } /** * Function adds a sharpening filter if image is being upsized. Such * sharpening allows to spot interpolation filter's stop-band attenuation: * if attenuation is too weak, a "dark grid" and other artifacts may * become visible. * * It is assumed that 40 decibel stop-band attenuation should be * considered a required minimum: this allows application of (deliberately * strong) 64X sharpening without spotting any artifacts. * * @param Steps Filtering steps. * @param bw Resulting bandwidth relative to the original bandwidth (which * is 1.0), usually 1/k. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ static void addSharpenTest(CFilterSteps& Steps, const double bw, const bool IsModel) { if (bw <= 1.0) { return; } const double FltLen = 10.0 * bw; CFilterStep& fs = Steps.add(); fs.IsUpsample = false; fs.ResampleFactor = 1; fs.DCGain = 1.0; fs.EdgePixelCount = 0; if (IsModel) { allocFilter(fs.Flt, CDSPFIREQ ::calcFilterLength(FltLen, fs.FltLatency), true); return; } const int BinCount = 200; CBuffer Bins(BinCount); int Thresh = (int)round(BinCount / bw * 1.75); if (Thresh > BinCount) { Thresh = BinCount; } int j; for (j = 0; j < Thresh; j++) { Bins[j] = 1.0; } for (j = Thresh; j < BinCount; j++) { Bins[j] = 256.0; } CDSPFIREQ EQ; EQ.init(bw * 2.0, FltLen, BinCount, 0.0, bw, false, 1.7); fs.FltLatency = EQ.getFilterLatency(); CBuffer Filter(EQ.getFilterLength()); EQ.buildFilter(Bins, &Filter[0]); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); optimizeFIRFilter(Filter, fs.FltLatency); normalizeFIRFilter(&Filter[0], Filter.getCapacity(), 1.0); allocFilter(fs.Flt, Filter.getCapacity()); copyArray(&Filter[0], &fs.Flt[0], Filter.getCapacity()); /* for( j = 0; j < BinCount; j++ ) { const double th = AVIR_PI * j / ( BinCount - 1 ); double re; double im; calcFIRFilterResponse( &fs.Flt[ 0 ], fs.Flt.getCapacity(), th, re, im ); printf( "%f\n", sqrt( re * re + im * im )); } printf( "***\n" );*/ } /** * Function builds sequence of filtering steps depending on the specified * resizing coefficient. The last steps included are always the resizing * step then (possibly) the correction step. * * @param Steps Array that receives filtering steps. * @param[out] Vars Variables object. * @param FltBank Filter bank to initialize and use. * @param DCGain The overall DC gain to apply. This DC gain is applied to * the first filtering step only (upsampling or filtering step). * @param ModeFlags Build mode flags to use. This is a bitmap of switches * that enable or disable certain algorithm features. * @param IsModel "True" if filtering steps modeling is performed without * the actual filter allocation and building. */ void buildFilterSteps(CFilterSteps& Steps, CImageResizerVars& Vars, CDSPFracFilterBankLin& FltBank, const double DCGain, const int ModeFlags, const bool IsModel) const { Steps.clear(); const bool DoFltAndIntCombo = ((ModeFlags & 1) != 0); // Do filter // and interpolator combining. const bool ForceHiOrderInt = ((ModeFlags & 2) != 0); // Force use // of a higher-order interpolation. const bool UseHalfband = ((ModeFlags & 4) != 0); // Use half-band // filter. const double bw = 1.0 / Vars.k; // Resulting bandwidth. const int UpsampleFactor = ((int)floor(Vars.k) < 2 ? 2 : 1); double IntCutoffMult; // Interpolation filter cutoff multiplier. CFilterStep* ReuseStep; // If not NULL, resizing step should use // this step object instead of creating a new one. CFilterStep* ExtFltStep; // Use FltOrig of this step as the external // filter to applied to the interpolator. bool IsPreCorrection; // "True" if the correction filter is applied // first. double FltCutoff; // Cutoff frequency of the first filtering step. double corrbw; ///< Bandwidth at the correction step. if (Vars.k <= 1.0) { IsPreCorrection = true; FltCutoff = 1.0; corrbw = 1.0; Steps.add(); } else { IsPreCorrection = false; FltCutoff = bw; corrbw = bw; } // Add 1 upsampling or several downsampling filters. if (UpsampleFactor > 1) { CFilterStep& fs = Steps.add(); assignFilterParams(fs, true, UpsampleFactor, FltCutoff, DCGain, DoFltAndIntCombo, IsModel); IntCutoffMult = FltCutoff * 2.0 / UpsampleFactor; ReuseStep = NULL; ExtFltStep = (DoFltAndIntCombo ? &fs : NULL); } else { int DownsampleFactor; while (true) { DownsampleFactor = (int)floor(0.5 / FltCutoff); bool DoHBFltAdd; if (DownsampleFactor > 16) { // Add half-band filter unconditionally in order to keep // filter lengths lower for more precise frequency // response and less edge artifacts. DoHBFltAdd = true; DownsampleFactor = 16; } else { DoHBFltAdd = (UseHalfband && DownsampleFactor > 1); } if (DoHBFltAdd) { assignFilterParams(Steps.add(), false, DownsampleFactor, 0.0, 1.0, false, IsModel); FltCutoff *= DownsampleFactor; } else { if (DownsampleFactor < 1) { DownsampleFactor = 1; } break; } } CFilterStep& fs = Steps.add(); assignFilterParams(fs, false, DownsampleFactor, FltCutoff, DCGain, DoFltAndIntCombo, IsModel); IntCutoffMult = FltCutoff / 0.5; if (DoFltAndIntCombo) { ReuseStep = &fs; ExtFltStep = &fs; } else { IntCutoffMult *= DownsampleFactor; ReuseStep = NULL; ExtFltStep = NULL; } } // Insert resizing and correction steps. CFilterStep& fs = (ReuseStep == NULL ? Steps.add() : *ReuseStep); Vars.ResizeStep = Steps.getItemCount() - 1; fs.IsUpsample = false; fs.ResampleFactor = 0; fs.DCGain = (ExtFltStep == NULL ? 1.0 : ExtFltStep->DCGain); initFilterBank(FltBank, IntCutoffMult, ForceHiOrderInt, (ExtFltStep == NULL ? fs.FltOrig : ExtFltStep->FltOrig)); if (FltBank == FixedFilterBank) { fs.FltBank = (CDSPFracFilterBankLin*)&FixedFilterBank; } else { fs.FltBank = &FltBank; } addCorrectionFilter(Steps, corrbw, IsPreCorrection, IsModel); // addSharpenTest( Steps, bw, IsModel ); } /** * Function extends *this upsampling step so that it produces more * upsampled pixels that cover the prefix and suffix needs of the next * step. After the call to this function the InPrefix and InSuffix * variables of the next step will be set to zero. * * @param fs Upsampling filtering step. * @param NextStep The next step structure. */ static void extendUpsample(CFilterStep& fs, CFilterStep& NextStep) { fs.InPrefix = (NextStep.InPrefix + fs.ResampleFactor - 1) / fs.ResampleFactor; fs.OutPrefix += fs.InPrefix * fs.ResampleFactor; NextStep.InPrefix = 0; fs.InSuffix = (NextStep.InSuffix + fs.ResampleFactor - 1) / fs.ResampleFactor; fs.OutSuffix += fs.InSuffix * fs.ResampleFactor; NextStep.InSuffix = 0; } /** * Function fills resizing step's RPosBuf array, excluding the actual * "ftp" pointers and "SrcOffs" offsets. * * This array should be cleared if the resizing step or offset were * changed. Otherwise this function only fills the elements required to * cover resizing step's OutLen. * * This function is called by the updateFilterStepBuffers() function. * * @param fs Resizing step. * @param Vars Variables object. */ static void fillRPosBuf(CFilterStep& fs, const CImageResizerVars& Vars) { const int PrevLen = fs.RPosBuf->getCapacity(); if (fs.OutLen > PrevLen) { fs.RPosBuf->increaseCapacity(fs.OutLen); } typename CFilterStep ::CResizePos* rpos = &(*fs.RPosBuf)[PrevLen]; const int FracCount = fs.FltBank->getFracCount(); const double o = Vars.o; const double k = Vars.k; int i; for (i = PrevLen; i < fs.OutLen; i++) { const double SrcPos = o + k * i; const int SrcPosInt = (int)floor(SrcPos); const double x = (SrcPos - SrcPosInt) * FracCount; const int fti = (int)x; rpos->x = (typename fpclass ::fptypeatom)(x - fti); rpos->fti = fti; rpos->SrcPosInt = SrcPosInt; rpos++; } } /** * Function updates filtering step buffer lengths depending on the * specified source and new scanline lengths. This function should be * called after the buildFilterSteps() function. * * @param Steps Array that receives filtering steps. * @param[out] Vars Variables object, will receive buffer size and length. * This function expects "k" and "o" variable values that will be * adjusted by this function. * @param RPosBufArray Resizing position buffers array, used to obtain * buffer to initialize and use (will be reused if it is already fully or * partially filled). * @param SrcLen Source scanline's length in pixels. * @param NewLen New scanline's length in pixels. */ static void updateFilterStepBuffers( CFilterSteps& Steps, CImageResizerVars& Vars, typename CFilterStep ::CRPosBufArray& RPosBufArray, int SrcLen, const int NewLen) { int upstep = -1; int InBuf = 0; int i; for (i = 0; i < Steps.getItemCount(); i++) { CFilterStep& fs = Steps[i]; fs.Vars = &Vars; fs.InLen = SrcLen; fs.InBuf = InBuf; fs.OutBuf = (InBuf + 1) & 1; if (fs.IsUpsample) { upstep = i; Vars.k *= fs.ResampleFactor; Vars.o *= fs.ResampleFactor; fs.InPrefix = 0; fs.InSuffix = 0; fs.OutLen = fs.InLen * fs.ResampleFactor; fs.OutPrefix = fs.FltLatency; fs.OutSuffix = fs.Flt.getCapacity() - fs.FltLatency - fs.ResampleFactor; int l0 = fs.OutPrefix + fs.OutLen + fs.OutSuffix; int l = fs.InLen * fs.ResampleFactor + fs.SuffixDC.getCapacity(); if (l > l0) { fs.OutSuffix += l - l0; } l0 = fs.OutLen + fs.OutSuffix; if (fs.PrefixDC.getCapacity() > l0) { fs.OutSuffix += fs.PrefixDC.getCapacity() - l0; } } else if (fs.ResampleFactor == 0) { const int FilterLenD2 = fs.FltBank->getFilterLen() / 2; const int FilterLenD21 = FilterLenD2 - 1; const int ResizeLPix = (int)floor(Vars.o) - FilterLenD21; fs.InPrefix = (ResizeLPix < 0 ? -ResizeLPix : 0); const int ResizeRPix = (int)floor(Vars.o + (NewLen - 1) * Vars.k) + FilterLenD2 + 1; fs.InSuffix = (ResizeRPix > fs.InLen ? ResizeRPix - fs.InLen : 0); fs.OutLen = NewLen; fs.RPosBuf = &RPosBufArray.getRPosBuf(Vars.k, Vars.o, fs.FltBank->getFracCount()); fillRPosBuf(fs, Vars); } else { Vars.k /= fs.ResampleFactor; Vars.o /= fs.ResampleFactor; Vars.o += fs.EdgePixelCount; fs.InPrefix = fs.FltLatency; fs.InSuffix = fs.Flt.getCapacity() - fs.FltLatency - 1; // Additionally extend OutLen to produce more precise edge // pixels. fs.OutLen = (fs.InLen + fs.ResampleFactor - 1) / fs.ResampleFactor + fs.EdgePixelCount; fs.InSuffix += (fs.OutLen - 1) * fs.ResampleFactor + 1 - fs.InLen; fs.InPrefix += fs.EdgePixelCount * fs.ResampleFactor; fs.OutLen += fs.EdgePixelCount; } InBuf = fs.OutBuf; SrcLen = fs.OutLen; } Steps[Steps.getItemCount() - 1].OutBuf = 2; if (upstep != -1) { extendUpsample(Steps[upstep], Steps[upstep + 1]); } } /** * Function calculates an optimal intermediate buffer length that will * cover all needs of the specified filtering steps. This function should * be called after the updateFilterStepBuffers() function. * * Function also updates resizing step's RPosBuf pointers to the filter * bank and SrcOffs values. * * @param Steps Filtering steps. * @param[out] Vars Variables object, will receive buffer size and length. * @param ResElIncr Resulting (final) element increment, used to produce * de-interleaved result. For horizontal processing this value is equal * to last step's OutLen, for vertical processing this value is equal to * resulting image's width. */ static void updateBufLenAndRPosPtrs(CFilterSteps& Steps, CImageResizerVars& Vars, const int ResElIncr) { int MaxPrefix[2] = {0, 0}; int MaxLen[2] = {0, 0}; int i; for (i = 0; i < Steps.getItemCount(); i++) { CFilterStep& fs = Steps[i]; const int ib = fs.InBuf; if (fs.InPrefix > MaxPrefix[ib]) { MaxPrefix[ib] = fs.InPrefix; } int l = fs.InLen + fs.InSuffix; if (l > MaxLen[ib]) { MaxLen[ib] = l; } fs.InElIncr = fs.InPrefix + l; if (fs.OutBuf == 2) { break; } const int ob = fs.OutBuf; if (fs.IsUpsample) { if (fs.OutPrefix > MaxPrefix[ob]) { MaxPrefix[ob] = fs.OutPrefix; } l = fs.OutLen + fs.OutSuffix; if (l > MaxLen[ob]) { MaxLen[ob] = l; } } else { if (fs.OutLen > MaxLen[ob]) { MaxLen[ob] = fs.OutLen; } } } // Update OutElIncr values of all steps. for (i = 0; i < Steps.getItemCount(); i++) { CFilterStep& fs = Steps[i]; if (fs.OutBuf == 2) { fs.OutElIncr = ResElIncr; break; } CFilterStep& fs2 = Steps[i + 1]; if (fs.IsUpsample) { fs.OutElIncr = fs.OutPrefix + fs.OutLen + fs.OutSuffix; if (fs.OutElIncr > fs2.InElIncr) { fs2.InElIncr = fs.OutElIncr; } else { fs.OutElIncr = fs2.InElIncr; } } else { fs.OutElIncr = fs2.InElIncr; } } // Update temporary buffer's length. for (i = 0; i < 2; i++) { Vars.BufLen[i] = MaxPrefix[i] + MaxLen[i]; Vars.BufOffs[i] = MaxPrefix[i]; if (Vars.packmode == 0) { Vars.BufOffs[i] *= Vars.ElCount; } Vars.BufLen[i] *= Vars.ElCount; } // Update RPosBuf pointers and SrcOffs. CFilterStep& fs = Steps[Vars.ResizeStep]; typename CFilterStep ::CResizePos* rpos = &(*fs.RPosBuf)[0]; const int em = (fpclass ::packmode == 0 ? Vars.ElCount : 1); const int FilterLenD21 = fs.FltBank->getFilterLen() / 2 - 1; for (i = 0; i < fs.OutLen; i++) { rpos->ftp = fs.FltBank->getFilter(rpos->fti); rpos->SrcOffs = (rpos->SrcPosInt - FilterLenD21) * em; rpos++; } } /** * Function modifies the overall (DC) gain of the correction filter in the * pre-built filtering steps array. * * @param Steps Filtering steps. * @param m Multiplier to apply to the correction filter. */ void modifyCorrFilterDCGain(CFilterSteps& Steps, const double m) const { CBuffer* Flt; const int z = Steps.getItemCount() - 1; if (!Steps[z].IsUpsample && Steps[z].ResampleFactor == 1) { Flt = &Steps[z].Flt; } else { Flt = &Steps[0].Flt; } int i; for (i = 0; i < Flt->getCapacity(); i++) { (*Flt)[i] = (fptype)((double)(*Flt)[i] * m); } } /** * Function builds a map of used fractional delay filters based on the * resizing positions buffer. * * @param fs Resizing step. * @param[out] UsedFracMap Map of used fractional delay filters. */ static void fillUsedFracMap(const CFilterStep& fs, CBuffer& UsedFracMap) { const int FracCount = fs.FltBank->getFracCount(); UsedFracMap.increaseCapacity(FracCount, false); memset(&UsedFracMap[0], 0, FracCount * sizeof(UsedFracMap[0])); typename CFilterStep ::CResizePos* rpos = &(*fs.RPosBuf)[0]; int i; for (i = 0; i < fs.OutLen; i++) { UsedFracMap[rpos->fti] |= 1; rpos++; } } /** * Function calculates the overall filtering steps complexity per * scanline. Each complexity unit corresponds to a single multiply-add * operation. Data copy and pointer math operations are not included in * this calculation, it is assumed that they correlate to the multiply-add * operations. Calculation also does not include final rounding, dithering * and clamping operations since they cannot be optimized out anyway. * * Calculation of the CRPosBuf buffer is not included since it cannot be * avoided. * * This function should be called after the updateFilterStepBuffers() * function. * * @param Steps Filtering steps array. * @param Vars Variables object. * @param UsedFracMap The map of used fractional delay filters. * @param ScanlineCount Scanline count. */ static int calcComplexity(const CFilterSteps& Steps, const CImageResizerVars& Vars, const CBuffer& UsedFracMap, const int ScanlineCount) { int fcnum; // Filter complexity multiplier numerator. int fcdenom; // Filter complexity multiplier denominator. if (Vars.packmode != 0) { fcnum = 1; fcdenom = 1; } else { // In interleaved processing mode, filters require 1 less // multiplication per 2 multiply-add instructions. fcnum = 3; fcdenom = 4; } int s = 0; // Complexity per one scanline. int s2 = 0; // Complexity per all scanlines. int i; for (i = 0; i < Steps.getItemCount(); i++) { const CFilterStep& fs = Steps[i]; s2 += 65 * fs.Flt.getCapacity(); // Filter creation complexity. if (fs.IsUpsample) { if (fs.FltOrig.getCapacity() > 0) { continue; } s += (fs.Flt.getCapacity() * (fs.InPrefix + fs.InLen + fs.InSuffix) + fs.SuffixDC.getCapacity() + fs.PrefixDC.getCapacity()) * Vars.ElCount; } else if (fs.ResampleFactor == 0) { s += fs.FltBank->getFilterLen() * (fs.FltBank->getOrder() + Vars.ElCount) * fs.OutLen; s2 += fs.FltBank->calcInitComplexity(UsedFracMap); } else { s += fs.Flt.getCapacity() * Vars.ElCount * fs.OutLen * fcnum / fcdenom; } } return (s + s2 / ScanlineCount); } /** * @brief Thread-isolated data used for scanline processing. * * This structure holds data necessary for image's horizontal or vertical * scanline processing, including scanline processing queue. * * @tparam Tin Source element data type. Intermediate buffers store data * in floating point format. * @tparam Tout Destination element data type. Intermediate buffers store * data in floating point format. */ template class CThreadData : public CImageResizerThreadPool ::CWorkload { public: virtual void process() { processScanlineQueue(); } /** * This enumeration lists possible scanline operations. */ enum EScanlineOperation { sopResizeH, ///< Resize horizontal scanline. ///< sopResizeV, ///< Resize vertical scanline. ///< sopDitherAndUnpackH, ///< Dither and unpack horizontal scanline. ///< sopUnpackH ///< Unpack horizontal scanline. ///< }; /** * Function initializes *this thread data object and assigns certain * variables provided by the higher level code. * * @param aThreadIndex Index of this thread data (0-based). * @param aThreadCount Total number of threads used during processing. * @param aSteps Filtering steps. * @param aVars Image resizer variables. */ void init(const int aThreadIndex, const int aThreadCount, const CFilterSteps& aSteps, const CImageResizerVars& aVars) { ThreadIndex = aThreadIndex; ThreadCount = aThreadCount; Steps = &aSteps; Vars = &aVars; } /** * Function initializes scanline processing queue, and updates * capacities of intermediate buffers. * * @param aOp Operation to perform over scanline. * @param TotalLines The total number of scanlines that will be * processed by all threads. * @param aSrcLen Source scanline length in pixels. * @param aSrcIncr Source scanline buffer increment. Ignored in * horizontal scanline processing. * @param aResIncr Resulting scanline buffer increment. Ignored in * horizontal scanline processing. */ void initScanlineQueue(const EScanlineOperation aOp, const int TotalLines, const int aSrcLen, const int aSrcIncr = 0, const int aResIncr = 0) { const int l = Vars->BufLen[0] + Vars->BufLen[1]; if (Bufs.getCapacity() < l) { Bufs.alloc(l, fpclass ::fpalign); } BufPtrs[0] = Bufs + Vars->BufOffs[0]; BufPtrs[1] = Bufs + Vars->BufLen[0] + Vars->BufOffs[1]; int j; int ml = 0; for (j = 0; j < Steps->getItemCount(); j++) { const CFilterStep& fs = (*Steps)[j]; if (fs.ResampleFactor == 0 && ml < fs.FltBank->getFilterLen()) { ml = fs.FltBank->getFilterLen(); } } TmpFltBuf.alloc(ml, fpclass ::fpalign); ScanlineOp = aOp; SrcLen = aSrcLen; SrcIncr = aSrcIncr; ResIncr = aResIncr; QueueLen = 0; Queue.increaseCapacity((TotalLines + ThreadCount - 1) / ThreadCount, false); } /** * Function adds a scanline to the queue buffer. The * initScanlineQueue() function should be called before calling this * function. The number of calls to this add function should not * exceed the TotalLines spread over all threads. * * @param SrcBuf Source scanline buffer. * @param ResBuf Resulting scanline buffer. */ void addScanlineToQueue(void* const SrcBuf, void* const ResBuf) { Queue[QueueLen].SrcBuf = SrcBuf; Queue[QueueLen].ResBuf = ResBuf; QueueLen++; } /** * Function processes all queued scanlines. */ void processScanlineQueue() { int i; switch (ScanlineOp) { case sopResizeH: { for (i = 0; i < QueueLen; i++) { resizeScanlineH((Tin*)Queue[i].SrcBuf, (fptype*)Queue[i].ResBuf); } break; } case sopResizeV: { for (i = 0; i < QueueLen; i++) { resizeScanlineV((fptype*)Queue[i].SrcBuf, (fptype*)Queue[i].ResBuf); } break; } case sopDitherAndUnpackH: { if (Vars->UseSRGBGamma) { for (i = 0; i < QueueLen; i++) { CFilterStep ::applySRGBGamma((fptype*)Queue[i].SrcBuf, SrcLen, *Vars); Ditherer.dither((fptype*)Queue[i].SrcBuf); CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } else { for (i = 0; i < QueueLen; i++) { Ditherer.dither((fptype*)Queue[i].SrcBuf); CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } break; } case sopUnpackH: { if (Vars->UseSRGBGamma) { for (i = 0; i < QueueLen; i++) { CFilterStep ::applySRGBGamma((fptype*)Queue[i].SrcBuf, SrcLen, *Vars); CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } else { for (i = 0; i < QueueLen; i++) { CFilterStep ::unpackScanline((fptype*)Queue[i].SrcBuf, (Tout*)Queue[i].ResBuf, SrcLen, *Vars); } } break; } } } /** * Function returns ditherer object associated with *this thread data * object. */ CDitherer& getDitherer() { return (Ditherer); } private: int ThreadIndex; ///< Thread index. ///< int ThreadCount; ///< Thread count. ///< const CFilterSteps* Steps; ///< Filtering steps. ///< const CImageResizerVars* Vars; ///< Image resizer variables. ///< CBuffer Bufs; ///< Flip-flop intermediate buffers. ///< fptype* BufPtrs[3]; ///< Flip-flop buffer pointers (referenced by ///< filtering step's InBuf and OutBuf indices). ///< CBuffer TmpFltBuf; ///< Temporary buffer used in the ///< doResize() function, aligned by fpclass :: fpalign. ///< EScanlineOperation ScanlineOp; ///< Operation to perform over ///< scanline. ///< int SrcLen; ///< Source scanline length in the last queue. ///< int SrcIncr; ///< Source scanline buffer increment in the last queue. ///< int ResIncr; ///< Resulting scanline buffer increment in the last ///< queue. ///< CDitherer Ditherer; ///< Ditherer object to use. ///< /** * @brief Scanline processing queue item. * * Scanline processing queue item. */ struct CQueueItem { void* SrcBuf; ///< Source scanline buffer, will by typecasted to ///< Tin or fptype*. ///< void* ResBuf; ///< Resulting scanline buffer, will by typecasted ///< to Tout or fptype*. ///< }; CBuffer Queue; ///< Scanline processing queue. ///< int QueueLen; ///< Queue length. ///< /** * Function resizes a single horizontal scanline. * * @param SrcBuf Source scanline buffer. Can be either horizontal or * vertical. * @param ResBuf Resulting scanline buffer. */ void resizeScanlineH(const Tin* const SrcBuf, fptype* const ResBuf) { (*Steps)[0].packScanline(SrcBuf, BufPtrs[0], SrcLen); BufPtrs[2] = ResBuf; int j; for (j = 0; j < Steps->getItemCount(); j++) { const CFilterStep& fs = (*Steps)[j]; fs.prepareInBuf(BufPtrs[fs.InBuf]); const int DstIncr = (Vars->packmode == 0 ? Vars->ElCount : 1); if (fs.ResampleFactor != 0) { if (fs.IsUpsample) { fs.doUpsample(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf]); } else { fs.doFilter(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr); } } else { fs.doResize(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr, TmpFltBuf); } } } /** * Function resizes a single vertical scanline. * * @param SrcBuf Source scanline buffer. Can be either horizontal or * vertical. * @param ResBuf Resulting scanline buffer. */ void resizeScanlineV(const fptype* const SrcBuf, fptype* const ResBuf) { (*Steps)[0].convertVtoH(SrcBuf, BufPtrs[0], SrcLen, SrcIncr); BufPtrs[2] = ResBuf; int j; for (j = 0; j < Steps->getItemCount(); j++) { const CFilterStep& fs = (*Steps)[j]; fs.prepareInBuf(BufPtrs[fs.InBuf]); const int DstIncr = (fs.OutBuf == 2 ? ResIncr : (Vars->packmode == 0 ? Vars->ElCount : 1)); if (fs.ResampleFactor != 0) { if (fs.IsUpsample) { fs.doUpsample(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf]); } else { fs.doFilter(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr); } } else { fs.doResize(BufPtrs[fs.InBuf], BufPtrs[fs.OutBuf], DstIncr, TmpFltBuf); } } } }; }; #undef AVIR_PI #undef AVIR_PId2 } // namespace avir #endif // AVIR_CIMAGERESIZER_INCLUDED /* clang-format off */ //$ nobt //$ nocpp #include "libc/calls/calls.h" /** * @file avir.h * * @brief The "main" inclusion file with all required classes and functions. * * This is the "main" inclusion file for the "AVIR" image resizer. This * inclusion file contains implementation of the AVIR image resizing algorithm * in its entirety. Also includes several classes and functions that can be * useful elsewhere. * * AVIR Copyright (c) 2015-2019 Aleksey Vaneev * * @mainpage * * @section intro_sec Introduction * * Description is available at https://github.com/avaneev/avir * * AVIR is devoted to women. Your digital photos can look good at any size! * * @section license License * * AVIR License Agreement * * The MIT License (MIT) * * Copyright (c) 2015-2019 Aleksey Vaneev * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. * * Please credit the author of this library in your documentation in the * following way: "AVIR image resizing algorithm designed by Aleksey Vaneev" * * @version 2.4 */ #ifndef AVIR_CIMAGERESIZER_INCLUDED #define AVIR_CIMAGERESIZER_INCLUDED #include #include #include #include namespace avir { /** * The macro defines AVIR version string. */ #define AVIR_VERSION "2.4" /** * The macro equals to "pi" constant, fills 53-bit floating point mantissa. * Undefined at the end of file. */ #define AVIR_PI 3.1415926535897932 /** * The macro equals to "pi divided by 2" constant, fills 53-bit floating * point mantissa. Undefined at the end of file. */ #define AVIR_PId2 1.5707963267948966 /** * Rounding function, based on the (int) typecast. Biased result. Not suitable * for numbers >= 2^31. * * @param d Value to round. * @return Rounded value. Some bias may be introduced. */ template< class T > inline T round( const T d ) { return( d < 0.0 ? -(T) (int) ( (T) 0.5 - d ) : (T) (int) ( d + (T) 0.5 )); } /** * Template function "clamps" (clips) the specified value so that it is not * lesser than "minv", and not greater than "maxv". * * @param Value Value to clamp. * @param minv Minimal allowed value. * @param maxv Maximal allowed value. * @return The clamped value. */ template< class T > inline T clamp( const T& Value, const T minv, const T maxv ) { if( Value < minv ) { return( minv ); } else if( Value > maxv ) { return( maxv ); } else { return( Value ); } } /** * Power 2.4 approximation function, designed for sRGB gamma correction. * * @param x Argument, in the range 0.09 to 1. * @return Value raised into power 2.4, approximate. */ template< class T > inline T pow24_sRGB( const T x ) { const double x2 = x * x; const double x3 = x2 * x; const double x4 = x2 * x2; return( (T) ( 0.0985766365536824 + 0.839474952656502 * x2 + 0.363287814061725 * x3 - 0.0125559718896615 / ( 0.12758338921578 + 0.290283465468235 * x ) - 0.231757513261358 * x - 0.0395365717969074 * x4 )); } /** * Power 1/2.4 approximation function, designed for sRGB gamma correction. * * @param x Argument, in the range 0.003 to 1. * @return Value raised into power 1/2.4, approximate. */ template< class T > inline T pow24i_sRGB( const T x ) { const double sx = sqrt( x ); const double ssx = sqrt( sx ); const double sssx = sqrt( ssx ); return( (T) ( 0.000213364515060263 + 0.0149409239419218 * x + 0.433973412731747 * sx + ssx * ( 0.659628181609715 * sssx - 0.0380957908841466 - 0.0706476137208521 * sx ))); } /** * Function approximately linearizes the sRGB gamma value. * * @param s sRGB gamma value, in the range 0 to 1. * @return Linearized sRGB gamma value, approximated. */ template< class T > inline T convertSRGB2Lin( const T s ) { const T a = (T) 0.055; if( s <= (T) 0.04045 ) { return( s / (T) 12.92 ); } return( pow24_sRGB(( s + a ) / ( (T) 1 + a ))); } /** * Function approximately de-linearizes the linear gamma value. * * @param s Linear gamma value, in the range 0 to 1. * @return sRGB gamma value, approximated. */ template< class T > inline T convertLin2SRGB( const T s ) { const T a = (T) 0.055; if( s <= (T) 0.0031308 ) { return( (T) 12.92 * s ); } return(( (T) 1 + a ) * pow24i_sRGB( s ) - a ); } /** * Function converts (via typecast) specified array of type T1 values of * length l into array of type T2 values. If T1 is the same as T2, copy * operation is performed. When copying data at overlapping address spaces, * "op" should be lower than "ip". * * @param ip Input buffer. * @param[out] op Output buffer. * @param l The number of elements to copy. * @param ip Input buffer pointer increment. * @param op Output buffer pointer increment. */ template< class T1, class T2 > inline void copyArray( const T1* ip, T2* op, int l, const int ipinc = 1, const int opinc = 1 ) { while( l > 0 ) { *op = (T2) *ip; op += opinc; ip += ipinc; l--; } } /** * Function adds values located in array "ip" to array "op". * * @param ip Input buffer. * @param[out] op Output buffer. * @param l The number of elements to add. * @param ip Input buffer pointer increment. * @param op Output buffer pointer increment. */ template< class T1, class T2 > inline void addArray( const T1* ip, T2* op, int l, const int ipinc = 1, const int opinc = 1 ) { while( l > 0 ) { *op += *ip; op += opinc; ip += ipinc; l--; } } /** * Function that replicates a set of adjacent elements several times in a row. * This operation is usually used to replicate pixels at the start or end of * image's scanline. * * @param ip Source array. * @param ipl Source array length (usually 1..4, but can be any number). * @param[out] op Destination buffer. * @param l Number of times the source array should be replicated (the * destination buffer should be able to hold ipl * l number of elements). * @param opinc Destination buffer position increment after replicating the * source array. This value should be equal to at least ipl. */ template< class T1, class T2 > inline void replicateArray( const T1* const ip, const int ipl, T2* op, int l, const int opinc ) { if( ipl == 1 ) { while( l > 0 ) { op[ 0 ] = ip[ 0 ]; op += opinc; l--; } } else if( ipl == 4 ) { while( l > 0 ) { op[ 0 ] = ip[ 0 ]; op[ 1 ] = ip[ 1 ]; op[ 2 ] = ip[ 2 ]; op[ 3 ] = ip[ 3 ]; op += opinc; l--; } } else if( ipl == 3 ) { while( l > 0 ) { op[ 0 ] = ip[ 0 ]; op[ 1 ] = ip[ 1 ]; op[ 2 ] = ip[ 2 ]; op += opinc; l--; } } else if( ipl == 2 ) { while( l > 0 ) { op[ 0 ] = ip[ 0 ]; op[ 1 ] = ip[ 1 ]; op += opinc; l--; } } else { while( l > 0 ) { int i; for( i = 0; i < ipl; i++ ) { op[ i ] = ip[ i ]; } op += opinc; l--; } } } /** * Function calculates frequency response of the specified FIR filter at the * specified circular frequency. Phase can be calculated as atan2( im, re ). * Function uses computationally-efficient oscillators instead of "cos" and * "sin" functions. * * @param flt FIR filter's coefficients. * @param fltlen Number of coefficients (taps) in the filter. * @param th Circular frequency [0; pi]. * @param[out] re0 Resulting real part of the complex frequency response. * @param[out] im0 Resulting imaginary part of the complex frequency response. * @param fltlat Filter's latency in samples (taps). */ template< class T > inline void calcFIRFilterResponse( const T* flt, int fltlen, const double th, double& re0, double& im0, const int fltlat = 0 ) { const double sincr = 2.0 * cos( th ); double cvalue1; double svalue1; if( fltlat == 0 ) { cvalue1 = 1.0; svalue1 = 0.0; } else { cvalue1 = cos( -fltlat * th ); svalue1 = sin( -fltlat * th ); } double cvalue2 = cos( -( fltlat + 1 ) * th ); double svalue2 = sin( -( fltlat + 1 ) * th ); double re = 0.0; double im = 0.0; while( fltlen > 0 ) { re += cvalue1 * flt[ 0 ]; im += svalue1 * flt[ 0 ]; flt++; fltlen--; double tmp = cvalue1; cvalue1 = sincr * cvalue1 - cvalue2; cvalue2 = tmp; tmp = svalue1; svalue1 = sincr * svalue1 - svalue2; svalue2 = tmp; } re0 = re; im0 = im; } /** * Function normalizes FIR filter so that its frequency response at DC is * equal to DCGain. * * @param[in,out] p Filter coefficients. * @param l Filter length. * @param DCGain Filter's gain at DC. * @param pstep "p" array step. */ template< class T > inline void normalizeFIRFilter( T* const p, const int l, const double DCGain, const int pstep = 1 ) { double s = 0.0; T* pp = p; int i = l; while( i > 0 ) { s += *pp; pp += pstep; i--; } s = DCGain / s; pp = p; i = l; while( i > 0 ) { *pp = (T) ( *pp * s ); pp += pstep; i--; } } /** * @brief Memory buffer class for element array storage, with capacity * tracking. * * Allows easier handling of memory blocks allocation and automatic * deallocation for arrays (buffers) consisting of elements of specified * class. Tracks buffer's capacity in "int" variable; unsuitable for * allocation of very large memory blocks (with more than 2 billion elements). * * This class manages memory space only - it does not perform element class * construction (initialization) operations. Buffer's required memory address * alignment specification is supported. * * Uses standard library to allocate and deallocate memory. * * @tparam T Buffer element's type. * @tparam capint Buffer capacity's type to use. Use size_t for large buffers. */ template< class T, typename capint = int > class CBuffer { public: CBuffer() : Data( NULL ) , DataAligned( NULL ) , Capacity( 0 ) , Alignment( 0 ) { } /** * Constructor creates the buffer with the specified capacity. * * @param aCapacity Buffer's capacity. * @param aAlignment Buffer's required memory address alignment. 0 - use * stdlib's default alignment. */ CBuffer( const capint aCapacity, const int aAlignment = 0 ) { allocinit( aCapacity, aAlignment ); } CBuffer( const CBuffer& Source ) { allocinit( Source.Capacity, Source.Alignment ); memcpy( DataAligned, Source.DataAligned, Capacity * sizeof( T )); } ~CBuffer() { freeData(); } CBuffer& operator = ( const CBuffer& Source ) { alloc( Source.Capacity, Source.Alignment ); memcpy( DataAligned, Source.DataAligned, Capacity * sizeof( T )); return( *this ); } /** * Function allocates memory so that the specified number of elements * can be stored in *this buffer object. * * @param aCapacity Storage for this number of elements to allocate. * @param aAlignment Buffer's required memory address alignment, * power-of-2 values only. 0 - use stdlib's default alignment. */ void alloc( const capint aCapacity, const int aAlignment = 0 ) { freeData(); allocinit( aCapacity, aAlignment ); } /** * Function deallocates any previously allocated buffer. */ void free() { freeData(); Data = NULL; DataAligned = NULL; Capacity = 0; Alignment = 0; } /** * @return The capacity of the element buffer. */ capint getCapacity() const { return( Capacity ); } /** * Function "forces" *this buffer to have an arbitary capacity. Calling * this function invalidates all further operations except deleting *this * object. This function should not be usually used at all. Function can * be used to "model" certain buffer capacity without calling a costly * memory allocation function. * * @param NewCapacity A new "forced" capacity. */ void forceCapacity( const capint NewCapacity ) { Capacity = NewCapacity; } /** * Function reallocates *this buffer to a larger size so that it will be * able to hold the specified number of elements. Downsizing is not * performed. Alignment is not changed. * * @param NewCapacity New (increased) capacity. * @param DoDataCopy "True" if data in the buffer should be retained. */ void increaseCapacity( const capint NewCapacity, const bool DoDataCopy = true ) { if( NewCapacity < Capacity ) { return; } if( DoDataCopy ) { const capint PrevCapacity = Capacity; T* const PrevData = Data; T* const PrevDataAligned = DataAligned; allocinit( NewCapacity, Alignment ); memcpy( DataAligned, PrevDataAligned, PrevCapacity * sizeof( T )); :: free( PrevData ); } else { :: free( Data ); allocinit( NewCapacity, Alignment ); } } /** * Function "truncates" (reduces) capacity of the buffer without * reallocating it. Alignment is not changed. * * @param NewCapacity New required capacity. */ void truncateCapacity( const capint NewCapacity ) { if( NewCapacity >= Capacity ) { return; } Capacity = NewCapacity; } /** * Function increases capacity so that the specified number of * elements can be stored. This function increases the previous capacity * value by third the current capacity value until space for the required * number of elements is available. Alignment is not changed. * * @param ReqCapacity Required capacity. */ void updateCapacity( const capint ReqCapacity ) { if( ReqCapacity <= Capacity ) { return; } capint NewCapacity = Capacity; while( NewCapacity < ReqCapacity ) { NewCapacity += NewCapacity / 3 + 1; } increaseCapacity( NewCapacity ); } operator T* () const { return( DataAligned ); } private: T* Data; ///< Element buffer pointer. ///< T* DataAligned; ///< Memory address-aligned element buffer pointer. ///< capint Capacity; ///< Element buffer capacity. ///< int Alignment; ///< Memory address alignment in use. 0 - use stdlib's ///< default alignment. ///< /** * Internal element buffer allocation function used during object * construction. * * @param aCapacity Storage for this number of elements to allocate. * @param aAlignment Buffer's required memory address alignment. 0 - use * stdlib's default alignment. */ void allocinit( const capint aCapacity, const int aAlignment ) { if( aAlignment == 0 ) { Data = (T*) :: malloc( aCapacity * sizeof( T )); DataAligned = Data; Alignment = 0; } else { Data = (T*) :: malloc( aCapacity * sizeof( T ) + aAlignment ); DataAligned = alignptr( Data, aAlignment ); Alignment = aAlignment; } Capacity = aCapacity; } /** * Function frees a previously allocated Data buffer. */ void freeData() { :: free( Data ); } /** * Function modifies the specified pointer so that it becomes memory * address-aligned. * * @param ptr Pointer to align. * @param align Alignment in bytes to apply. * @return Pointer aligned to align bytes. Works with power-of-2 * alignments only. If no alignment is necessary, "align" bytes will be * added to the pointer value. */ template< class Tp > inline Tp alignptr( const Tp ptr, const uintptr_t align ) { return( (Tp) ( (uintptr_t) ptr + align - ( (uintptr_t) ptr & ( align - 1 ))) ); } }; /** * Function optimizes the length of the symmetric-odd FIR filter by removing * left- and rightmost elements that are below specific threshold. * * Synthetic test shows that filter gets optimized in 2..3% of cases and in * each such case optimization reduces filter length by 6..8%. Optimization, * however, may skew the results of algorithm modeling and complexity * calculation leading to a choice of a less optimal algorithm. * * @param[in,out] Flt Buffer that contains filter being optimized. * @param[in,out] FltLatency Variable that holds the current latency of the * filter. May be adjusted on function return. * @param Threshold Threshold level. */ template< class T > inline void optimizeFIRFilter( CBuffer< T >& Flt, int& FltLatency, T const Threshold = (T) 0.00001 ) { int i; // Optimize length. for( i = 0; i <= FltLatency; i++ ) { if( fabs( Flt[ i ]) >= Threshold || i == FltLatency ) { if( i > 0 ) { const int NewCapacity = Flt.getCapacity() - i * 2; copyArray( &Flt[ i ], &Flt[ 0 ], NewCapacity ); Flt.truncateCapacity( NewCapacity ); FltLatency -= i; } break; } } } /** * @brief Array of structured objects. * * Implements allocation of a linear array of objects of class T (which are * initialized), addressable via operator[]. Each object is created via the * "operator new". New object insertions are quick since implementation uses * prior space allocation (capacity), thus not requiring frequent memory block * reallocations. * * @tparam T Array element's type. */ template< class T > class CStructArray { public: CStructArray() : ItemCount( 0 ) { } CStructArray( const CStructArray& Source ) : ItemCount( 0 ) , Items( Source.getItemCount() ) { while( ItemCount < Source.getItemCount() ) { Items[ ItemCount ] = new T( Source[ ItemCount ]); ItemCount++; } } ~CStructArray() { clear(); } CStructArray& operator = ( const CStructArray& Source ) { clear(); const int NewCount = Source.ItemCount; Items.updateCapacity( NewCount ); while( ItemCount < NewCount ) { Items[ ItemCount ] = new T( Source[ ItemCount ]); ItemCount++; } return( *this ); } T& operator []( const int Index ) { return( *Items[ Index ]); } const T& operator []( const int Index ) const { return( *Items[ Index ]); } /** * Function creates a new object of type T with the default constructor * and adds this object to the array. * * @return Reference to a newly added object. */ T& add() { if( ItemCount == Items.getCapacity() ) { Items.increaseCapacity( ItemCount * 3 / 2 + 1 ); } Items[ ItemCount ] = new T(); ItemCount++; return( (*this)[ ItemCount - 1 ]); } /** * Function changes number of allocated items. New items are created with * the default constructor. If NewCount is below the current item count, * items that are above NewCount range will be destructed. * * @param NewCount New requested item count. */ void setItemCount( const int NewCount ) { if( NewCount > ItemCount ) { Items.increaseCapacity( NewCount ); while( ItemCount < NewCount ) { Items[ ItemCount ] = new T(); ItemCount++; } } else { while( ItemCount > NewCount ) { ItemCount--; delete Items[ ItemCount ]; } } } /** * Function erases all items of *this array. */ void clear() { while( ItemCount > 0 ) { ItemCount--; delete Items[ ItemCount ]; } } /** * @return The number of allocated items. */ int getItemCount() const { return( ItemCount ); } private: int ItemCount; ///< The number of items available in the array. ///< CBuffer< T* > Items; ///< Element buffer. ///< }; /** * @brief Sine signal generator class. * * Class implements sine signal generator without biasing, with * constructor-based initalization only. This generator uses oscillator * instead of "sin" function. */ class CSineGen { public: /** * Constructor initializes *this sine signal generator. * * @param si Sine function increment, in radians. * @param ph Starting phase, in radians. Add 0.5 * AVIR_PI for cosine * function. */ CSineGen( const double si, const double ph ) : svalue1( sin( ph )) , svalue2( sin( ph - si )) , sincr( 2.0 * cos( si )) { } /** * @return The next value of the sine function, without biasing. */ double generate() { const double res = svalue1; svalue1 = sincr * res - svalue2; svalue2 = res; return( res ); } private: double svalue1; ///< Current sine value. ///< double svalue2; ///< Previous sine value. ///< double sincr; ///< Sine value increment. ///< }; /** * @brief Peaked Cosine window function generator class. * * Class implements Peaked Cosine window function generator. Generates the * right-handed half of the window function. The Alpha parameter of this * window function offers the control of the balance between the early and * later taps of the filter. E.g. at Alpha=1 both early and later taps are * attenuated, but at Alpha=4 mostly later taps are attenuated. This offers a * great control over ringing artifacts produced by a low-pass filter in image * processing, without compromising achieved image sharpness. */ class CDSPWindowGenPeakedCosine { public: /** * Constructor initializes *this window function generator. * * @param aAlpha Alpha parameter, affects the peak shape (peak * augmentation) of the window function. Should be >= 1.0. * @param aLen2 Half filter's length (non-truncated). */ CDSPWindowGenPeakedCosine( const double aAlpha, const double aLen2 ) : Alpha( aAlpha ) , Len2( aLen2 ) , wn( 0 ) , w1( AVIR_PId2 / Len2, AVIR_PI * 0.5 ) { } /** * @return The next Peaked Cosine window function coefficient. */ double generate() { const double h = pow( wn / Len2, Alpha ); wn++; return( w1.generate() * ( 1.0 - h )); } private: double Alpha; ///< Alpha parameter, affects the peak shape of window. ///< double Len2; ///< Half length of the window function. ///< int wn; ///< Window function integer position. 0 - center of the ///< window function. ///< CSineGen w1; ///< Sine-wave generator. ///< }; /** * @brief FIR filter-based equalizer generator. * * Class implements an object used to generate symmetric-odd FIR filters with * the specified frequency response (aka paragraphic equalizer). The * calculated filter is windowed by the Peaked Cosine window function. * * In image processing, due to short length of filters being used (6-8 taps) * the resulting frequency response of the filter is approximate and may be * mathematically imperfect, but still adequate to the visual requirements. * * On a side note, this equalizer generator can be successfully used for audio * signal equalization as well: for example, it is used in almost the same * form in Voxengo Marvel GEQ equalizer plug-in. * * Filter generation is based on decomposition of frequency range into * spectral bands, with each band represented by linear and ramp "kernels". * When the filter is built, these kernels are combined together with * different weights that approximate the required frequency response. */ class CDSPFIREQ { public: /** * Function initializes *this object with the required parameters. The * gain of frequencies beyond the MinFreq..MaxFreq range are controlled by * the first and the last band's gain. * * @param SampleRate Processing sample rate (use 2 for image processing). * @param aFilterLength Required filter length in samples (taps). The * actual filter length is truncated to an integer value. * @param aBandCount Number of band crossover points required to control, * including bands at MinFreq and MaxFreq. * @param MinFreq Minimal frequency that should be controlled. * @param MaxFreq Maximal frequency that should be controlled. * @param IsLogBands "True" if the bands should be spaced logarithmically. * @param WFAlpha Peaked Cosine window function's Alpha parameter. */ void init( const double SampleRate, const double aFilterLength, const int aBandCount, const double MinFreq, const double MaxFreq, const bool IsLogBands, const double WFAlpha ) { FilterLength = aFilterLength; BandCount = aBandCount; CenterFreqs.alloc( BandCount ); z = (int) ceil( FilterLength * 0.5 ); zi = z + ( z & 1 ); z2 = z * 2; CBuffer< double > oscbuf( z2 ); initOscBuf( oscbuf ); CBuffer< double > winbuf( z ); initWinBuf( winbuf, WFAlpha ); UseFirstVirtBand = ( MinFreq > 0.0 ); const int k = zi * ( BandCount + ( UseFirstVirtBand ? 1 : 0 )); Kernels1.alloc( k ); Kernels2.alloc( k ); double m; // Frequency step multiplier. double mo; // Frequency step offset (addition). if( IsLogBands ) { m = exp( log( MaxFreq / MinFreq ) / ( BandCount - 1 )); mo = 0.0; } else { m = 1.0; mo = ( MaxFreq - MinFreq ) / ( BandCount - 1 ); } double f = MinFreq; double x1 = 0.0; double x2; int si; if( UseFirstVirtBand ) { si = 0; } else { si = 1; CenterFreqs[ 0 ] = 0.0; f = f * m + mo; } double* kernbuf1 = &Kernels1[ 0 ]; double* kernbuf2 = &Kernels2[ 0 ]; int i; for( i = si; i < BandCount; i++ ) { x2 = f * 2.0 / SampleRate; CenterFreqs[ i ] = x2; fillBandKernel( x1, x2, kernbuf1, kernbuf2, oscbuf, winbuf ); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; f = f * m + mo; } if( x1 < 1.0 ) { UseLastVirtBand = true; fillBandKernel( x1, 1.0, kernbuf1, kernbuf2, oscbuf, winbuf ); } else { UseLastVirtBand = false; } } /** * @return Filter's length, in samples (taps). */ int getFilterLength() const { return( z2 - 1 ); } /** * @return Filter's latency (group delay), in samples (taps). */ int getFilterLatency() const { return( z - 1 ); } /** * Function creates symmetric-odd FIR filter with the specified gain * levels at band crossover points. * * @param BandGains Array of linear gain levels, count=BandCount specified * in the init() function. * @param[out] Filter Output filter buffer, length = getFilterLength(). */ void buildFilter( const double* const BandGains, double* const Filter ) { const double* kernbuf1 = &Kernels1[ 0 ]; const double* kernbuf2 = &Kernels2[ 0 ]; double x1 = 0.0; double y1 = BandGains[ 0 ]; double x2; double y2; int i; int si; if( UseFirstVirtBand ) { si = 1; x2 = CenterFreqs[ 0 ]; y2 = y1; } else { si = 2; x2 = CenterFreqs[ 1 ]; y2 = BandGains[ 1 ]; } copyBandKernel( Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - x2 * y1 ); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; y1 = y2; for( i = si; i < BandCount; i++ ) { x2 = CenterFreqs[ i ]; y2 = BandGains[ i ]; addBandKernel( Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - x2 * y1 ); kernbuf1 += zi; kernbuf2 += zi; x1 = x2; y1 = y2; } if( UseLastVirtBand ) { addBandKernel( Filter, kernbuf1, kernbuf2, y1 - y2, x1 * y2 - y1 ); } for( i = 0; i < z - 1; i++ ) { Filter[ z + i ] = Filter[ z - 2 - i ]; } } /** * Function calculates filter's length (in samples) and latency depending * on the required non-truncated filter length. * * @param aFilterLength Required filter length in samples (non-truncated). * @param[out] Latency Resulting latency (group delay) of the filter, * in samples (taps). * @return Filter length in samples (taps). */ static int calcFilterLength( const double aFilterLength, int& Latency ) { const int l = (int) ceil( aFilterLength * 0.5 ); Latency = l - 1; return( l * 2 - 1 ); } private: double FilterLength; ///< Length of filter. ///< int z; ///< Equals (int) ceil( FilterLength * 0.5 ). ///< int zi; ///< Equals "z" if z is even, or z + 1 if z is odd. Used as a ///< Kernels1 and Kernels2 size multiplier and kernel buffer increment ///< to make sure each kernel buffer is 16-byte aligned. ///< int z2; ///< Equals z * 2. ///< int BandCount; ///< Number of controllable bands. ///< CBuffer< double > CenterFreqs; ///< Center frequencies for all bands, ///< normalized to 0.0-1.0 range. ///< CBuffer< double > Kernels1; ///< Half-length kernel buffers for each ///< spectral band (linear part). ///< CBuffer< double > Kernels2; ///< Half-length kernel buffers for each ///< spectral band (ramp part). ///< bool UseFirstVirtBand; ///< "True" if the first virtual band ///< (between 0.0 and MinFreq) should be used. The first virtual band ///< won't be used if MinFreq equals 0.0. ///< bool UseLastVirtBand; ///< "True" if the last virtual band (between ///< MaxFreq and SampleRate * 0.5) should be used. The last virtual ///< band won't be used if MaxFreq * 2.0 equals SampleRate. ///< /** * Function initializes the "oscbuf" used in the fillBandKernel() * function. * * @param oscbuf Oscillator buffer, length = z * 2. */ void initOscBuf( double* oscbuf ) const { int i = z; while( i > 0 ) { oscbuf[ 0 ] = 0.0; oscbuf[ 1 ] = 1.0; oscbuf += 2; i--; } } /** * Function initializes window function buffer. This function generates * Peaked Cosine window function. * * @param winbuf Windowing buffer. * @param Alpha Peaked Cosine alpha parameter. */ void initWinBuf( double* winbuf, const double Alpha ) const { CDSPWindowGenPeakedCosine wf( Alpha, FilterLength * 0.5 ); int i; for( i = 1; i <= z; i++ ) { winbuf[ z - i ] = wf.generate(); } } /** * Function fills first half of symmetric-odd FIR kernel for the band. * This function should be called successively for adjacent bands. * Previous band's x2 should be equal to current band's x1. A band kernel * consists of 2 elements: linear kernel and ramp kernel. * * @param x1 Band's left corner frequency (0..1). * @param x2 Band's right corner frequency (0..1). * @param kernbuf1 Band kernel buffer 1 (linear part), length = z. * @param kernbuf2 Band kernel buffer 2 (ramp part), length = z. * @param oscbuf Oscillation buffer. Before the first call of the * fillBandKernel() should be initialized with the call of the * initOscBuf() function. * @param winbuf Buffer that contains windowing function. */ void fillBandKernel( const double x1, const double x2, double* kernbuf1, double* kernbuf2, double* oscbuf, const double* const winbuf ) { const double s2_incr = AVIR_PI * x2; const double s2_coeff = 2.0 * cos( s2_incr ); double s2_value1 = sin( s2_incr * ( -z + 1 )); double c2_value1 = sin( s2_incr * ( -z + 1 ) + AVIR_PI * 0.5 ); oscbuf[ 0 ] = sin( s2_incr * -z ); oscbuf[ 1 ] = sin( s2_incr * -z + AVIR_PI * 0.5 ); int ks; for( ks = 1; ks < z; ks++ ) { const int ks2 = ks * 2; const double s1_value1 = oscbuf[ ks2 ]; const double c1_value1 = oscbuf[ ks2 + 1 ]; oscbuf[ ks2 ] = s2_value1; oscbuf[ ks2 + 1 ] = c2_value1; const double x = AVIR_PI * ( ks - z ); const double v0 = winbuf[ ks - 1 ] / (( x1 - x2 ) * x ); kernbuf1[ ks - 1 ] = ( x2 * s2_value1 - x1 * s1_value1 + ( c2_value1 - c1_value1 ) / x ) * v0; kernbuf2[ ks - 1 ] = ( s2_value1 - s1_value1 ) * v0; s2_value1 = s2_coeff * s2_value1 - oscbuf[ ks2 - 2 ]; c2_value1 = s2_coeff * c2_value1 - oscbuf[ ks2 - 1 ]; } kernbuf1[ z - 1 ] = ( x2 * x2 - x1 * x1 ) / ( x1 - x2 ) * 0.5; kernbuf2[ z - 1 ] = -1.0; } /** * Function copies band kernel's elements to the output buffer. * * @param outbuf Output buffer. * @param kernbuf1 Kernel buffer 1 (linear part). * @param kernbuf2 Kernel buffer 2 (ramp part). * @param c Multiplier for linear kernel element. * @param d Multiplier for ramp kernel element. */ void copyBandKernel( double* outbuf, const double* const kernbuf1, const double* const kernbuf2, const double c, const double d ) const { int ks; for( ks = 0; ks < z; ks++ ) { outbuf[ ks ] = c * kernbuf1[ ks ] + d * kernbuf2[ ks ]; } } /** * Function adds band kernel's elements to the output buffer. * * @param outbuf Output buffer. * @param kernbuf1 Kernel buffer 1 (linear part). * @param kernbuf2 Kernel buffer 2 (ramp part). * @param c Multiplier for linear kernel element. * @param d Multiplier for ramp kernel element. */ void addBandKernel( double* outbuf, const double* const kernbuf1, const double* const kernbuf2, const double c, const double d ) const { int ks; for( ks = 0; ks < z; ks++ ) { outbuf[ ks ] += c * kernbuf1[ ks ] + d * kernbuf2[ ks ]; } } }; /** * @brief Low-pass filter windowed by Peaked Cosine window function. * * This class implements calculation of linear-phase symmetric-odd FIR * low-pass filter windowed by the Peaked Cosine window function, for image * processing applications. */ class CDSPPeakedCosineLPF { public: int fl2; ///< Half filter's length, excluding the peak value. This value ///< can be also used as filter's latency in samples (taps). ///< int FilterLen; ///< Filter's length in samples (taps). ///< /** * Constructor initalizes *this object. * * @param aLen2 Half-length (non-truncated) of low-pass filter, in samples * (taps). * @param aFreq2 Low-pass filter's corner frequency [0; pi]. * @param aAlpha Peaked Cosine window function Alpha parameter. */ CDSPPeakedCosineLPF( const double aLen2, const double aFreq2, const double aAlpha ) : fl2( (int) ceil( aLen2 ) - 1 ) , FilterLen( fl2 + fl2 + 1 ) , Len2( aLen2 ) , Freq2( aFreq2 ) , Alpha( aAlpha ) { } /** * Function generates a linear-phase low-pass filter windowed by Peaked * Cosine window function. * * @param[out] op Output buffer, length = FilterLen (fl2 * 2 + 1). * @param DCGain Required gain at DC. The resulting filter will be * normalized to achieve this DC gain. */ template< class T > void generateLPF( T* op, const double DCGain ) { CDSPWindowGenPeakedCosine wf( Alpha, Len2 ); CSineGen f2( Freq2, 0.0 ); op += fl2; T* op2 = op; f2.generate(); int t = 1; *op = (T) ( Freq2 * wf.generate() / AVIR_PI ); double s = *op; while( t <= fl2 ) { const double v = f2.generate() * wf.generate() / t / AVIR_PI; op++; op2--; *op = (T) v; *op2 = (T) v; s += *op + *op2; t++; } t = FilterLen; s = DCGain / s; while( t > 0 ) { *op2 = (T) ( *op2 * s ); op2++; t--; } } private: double Len2; ///< Half-length (non-truncated) of low-pass filter, in ///< samples (taps). ///< double Freq2; ///< Low-pass filter's corner frequency. ///< double Alpha; ///< Peaked Cosine window function Alpha parameter. ///< }; /** * @brief Buffer class for parametrized low-pass filter. * * This class extends the CBuffer< double > class by adding several variables * that define a symmetric-odd FIR low-pass filter windowed by Peaked Cosine * window function. This class can be used to compare filters without * comparing their buffer contents. */ class CFltBuffer : public CBuffer< double > { public: double Len2; ///< Half-length (non-truncated) of low-pass filters, in ///< samples (taps). ///< double Freq; ///< Low-pass filter's corner frequency. ///< double Alpha; ///< Peaked Cosine window function Alpha parameter. ///< double DCGain; ///< DC gain applied to the filter. ///< CFltBuffer() : CBuffer< double >() , Len2( 0.0 ) , Freq( 0.0 ) , Alpha( 0.0 ) , DCGain( 0.0 ) { } /** * @param b2 Filter buffer to compare *this object to. * @return Operator returns "true" if both filters have same parameters. */ bool operator == ( const CFltBuffer& b2 ) const { return( Len2 == b2.Len2 && Freq == b2.Freq && Alpha == b2.Alpha && DCGain == b2.DCGain ); } }; /** * @brief Sinc function-based fractional delay filter bank. * * Class implements storage and initialization of a bank of sinc * function-based fractional delay filters, expressed as 1st order polynomial * interpolation coefficients. The filters are produced from a single "long" * windowed low-pass filter. Also supports 0th-order ("nearest neighbor") * interpolation. * * This class also supports multiplication of each fractional delay filter by * an external filter (usually a low-pass filter). * * @tparam fptype Specifies storage type of the filter coefficients bank. The * filters are initially calculated using the "double" precision. */ template< class fptype > class CDSPFracFilterBankLin { public: CDSPFracFilterBankLin() : Order( -1 ) { } /** * Copy constructor copies a limited set of parameters of the source * filter bank. The actual filters are not copied. Such copying is used * during filtering steps "modeling" stage. A further init() function * call is required. * * @param s Source filter bank. */ void copyInitParams( const CDSPFracFilterBankLin& s ) { WFLen2 = s.WFLen2; WFFreq = s.WFFreq; WFAlpha = s.WFAlpha; FracCount = s.FracCount; Order = s.Order; Alignment = s.Alignment; SrcFilterLen = s.SrcFilterLen; FilterLen = s.FilterLen; FilterSize = s.FilterSize; IsSrcTableBuilt = false; ExtFilter = s.ExtFilter; TableFillFlags.alloc( s.TableFillFlags.getCapacity() ); int i; // Copy table fill flags, but shifted so that further initialization // is still possible (such feature should not be used, though). for( i = 0; i < TableFillFlags.getCapacity(); i++ ) { TableFillFlags[ i ] = (uint8_t) ( s.TableFillFlags[ i ] << 2 ); } } /** * Operator compares *this filter bank and another filter bank and returns * "true" if their parameters are equal. Alignment is not taken into * account. * * @param s Filter bank to compare to. * @return "True" if compared banks have equal parameters. */ bool operator == ( const CDSPFracFilterBankLin& s ) const { return( Order == s.Order && WFLen2 == s.WFLen2 && WFFreq == s.WFFreq && WFAlpha == s.WFAlpha && FracCount == s.FracCount && ExtFilter == s.ExtFilter ); } /** * Function initializes (builds) the filter bank based on the supplied * parameters. If the supplied parameters are equal to previously defined * parameters, function does nothing (alignment is assumed to be never * changing between the init() function calls). * * @param ReqFracCount Required number of fractional delays in the filter * bank. The minimal value is 2. * @param ReqOrder Required order of the interpolation polynomial * (0 or 1). * @param BaseLen Low-pass filter's base length, in samples (taps). * Affects the actual length of the filter and its overall steepness. * @param Cutoff Low-pass filter's normalized cutoff frequency [0; 1]. * @param aWFAlpha Peaked Cosine window function's Alpha parameter. * @param aExtFilter External filter to apply to each fractional delay * filter. * @param aAlignment Memory alignment of the filter bank, power-of-2 * value. 0 - use default stdlib alignment. * @param FltLenAlign Filter's length alignment, power-of-2 value. */ void init( const int ReqFracCount, const int ReqOrder, const double BaseLen, const double Cutoff, const double aWFAlpha, const CFltBuffer& aExtFilter, const int aAlignment = 0, const int FltLenAlign = 1 ) { double NewWFLen2 = 0.5 * BaseLen * ReqFracCount; double NewWFFreq = AVIR_PI * Cutoff / ReqFracCount; double NewWFAlpha = aWFAlpha; if( ReqOrder == Order && NewWFLen2 == WFLen2 && NewWFFreq == WFFreq && NewWFAlpha == WFAlpha && ReqFracCount == FracCount && aExtFilter == ExtFilter ) { IsInitRequired = false; return; } WFLen2 = NewWFLen2; WFFreq = NewWFFreq; WFAlpha = NewWFAlpha; FracCount = ReqFracCount; Order = ReqOrder; Alignment = aAlignment; ExtFilter = aExtFilter; CDSPPeakedCosineLPF p( WFLen2, WFFreq, WFAlpha ); SrcFilterLen = ( p.fl2 / ReqFracCount + 1 ) * 2; const int ElementSize = ReqOrder + 1; FilterLen = SrcFilterLen; if( ExtFilter.getCapacity() > 0 ) { FilterLen += ExtFilter.getCapacity() - 1; } FilterLen = ( FilterLen + FltLenAlign - 1 ) & ~( FltLenAlign - 1 ); FilterSize = FilterLen * ElementSize; IsSrcTableBuilt = false; IsInitRequired = true; } /** * @return The length of each fractional delay filter, in samples (taps). * Always an even value. */ int getFilterLen() const { return( FilterLen ); } /** * @return The number of fractional filters in use by *this bank. */ int getFracCount() const { return( FracCount ); } /** * @return The order of the interpolation polynomial. */ int getOrder() const { return( Order ); } /** * Function returns the pointer to the specified interpolation table * filter. * * @param i Filter (fractional delay) index, in the range 0 to * ReqFracCount - 1, inclusive. * @return Pointer to filter. Higher order polynomial coefficients are * stored after after previous order coefficients, separated by FilterLen * elements. */ const fptype* getFilter( const int i ) { if( !IsSrcTableBuilt ) { buildSrcTable(); } fptype* const Res = &Table[ i * FilterSize ]; if(( TableFillFlags[ i ] & 2 ) == 0 ) { createFilter( i ); TableFillFlags[ i ] |= 2; if( Order > 0 ) { createFilter( i + 1 ); const fptype* const Res2 = Res + FilterSize; fptype* const op = Res + FilterLen; int j; // Create higher-order interpolation coefficients (linear // interpolation). for( j = 0; j < FilterLen; j++ ) { op[ j ] = Res2[ j ] - Res[ j ]; } } } return( Res ); } /** * Function makes sure all fractional delay filters were created. */ void createAllFilters() { int i; for( i = 0; i < FracCount; i++ ) { getFilter( i ); } } /** * Function returns an approximate initialization complexity, expressed in * the number of multiply-add operations. This includes fractional delay * filters calculation and multiplication by an external filter. This * function can only be called after the init() function. * * @param FracUseMap Fractional delays use map, each element corresponds * to a single fractional delay, will be compared to the internal table * fill flags. This map should include 0 and 1 values only. * @return The complexity of the initialization, expressed in the number * of multiply-add operations. */ int calcInitComplexity( const CBuffer< uint8_t >& FracUseMap ) const { const int FltInitCost = 65; // Cost to initialize a single sample // of the fractional delay filter. const int FltUseCost = FilterLen * Order + SrcFilterLen * ExtFilter.getCapacity(); // Cost to use a single // fractional delay filter. const int ucb[ 2 ] = { 0, FltUseCost }; int ic; int i; if( IsInitRequired ) { ic = FracCount * SrcFilterLen * FltInitCost; for( i = 0; i < FracCount; i++ ) { ic += ucb[ FracUseMap[ i ]]; } } else { ic = 0; for( i = 0; i < FracCount; i++ ) { if( FracUseMap[ i ] != 0 ) { ic += ucb[ TableFillFlags[ i ] == 0 ? 1 : 0 ]; } } } return( ic ); } private: static const int InterpPoints = 2; ///< The maximal number of points the ///< interpolation is based on. ///< double WFLen2; ///< Window function's Len2 parameter. ///< double WFFreq; ///< Window function's Freq parameter. ///< double WFAlpha; ///< Window function's Alpha parameter. ///< int FracCount; ///< The required number of fractional delay filters. ///< int Order; ///< The order of the interpolation polynomial. ///< int Alignment; ///< The required filter table alignment. ///< int SrcFilterLen; ///< Length of the "source" filters. This is always an ///< even value. ///< int FilterLen; ///< Specifies the number of samples (taps) each fractional ///< delay filter has. This is always an even value, adjusted by the ///< FltLenAlign. ///< int FilterSize; ///< The size of a single filter element, equals ///< FilterLen * ElementSize. ///< bool IsInitRequired; ///< "True" if SrcTable filter table initialization ///< is required. This value is available only after the call to the ///< init() function. ///< CBuffer< fptype > Table; ///< Interpolation table, size equals to ///< ReqFracCount * FilterLen * ElementSize. ///< CBuffer< uint8_t > TableFillFlags; ///< Contains ReqFracCount + 1 ///< elements. Bit 0 of every element is 1 if Table already contains ///< the filter from SrcTable filtered by ExtFilter. Bit 1 of every ///< element means higher order coefficients were filled for the ///< filter. ///< CFltBuffer ExtFilter; ///< External filter that should be applied to every ///< fractional delay filter. Can be empty. Half of this filter's ///< capacity is used as latency (group delay) value of the filter. ///< CBuffer< double > SrcTable; ///< Source table of delay filters, contains ///< ReqFracCount + 1 elements. This table is used to fill the Table ///< with the actual filters, filtered by an external filter. ///< bool IsSrcTableBuilt; ///< "True" if the SrcTable was built already. This ///< variable is set to "false" in the init() function. ///< /** * Function builds source table used in the createFilter() function. */ void buildSrcTable() { IsSrcTableBuilt = true; IsInitRequired = false; CDSPPeakedCosineLPF p( WFLen2, WFFreq, WFAlpha ); const int BufLen = SrcFilterLen * FracCount + InterpPoints - 1; const int BufOffs = InterpPoints / 2 - 1; const int BufCenter = SrcFilterLen * FracCount / 2 + BufOffs; CBuffer< double > Buf( BufLen ); memset( Buf, 0, ( BufCenter - p.fl2 ) * sizeof( double )); int i = BufLen - BufCenter - p.fl2 - 1; memset( &Buf[ BufLen - i ], 0, i * sizeof( double )); p.generateLPF( &Buf[ BufCenter - p.fl2 ], FracCount ); asm("int3"); SrcTable.alloc(( FracCount + 1 ) * SrcFilterLen ); TableFillFlags.alloc( FracCount + 1 ); int j; double* op0 = SrcTable; for( i = FracCount; i >= 0; i-- ) { TableFillFlags[ i ] = 0; double* p = Buf + BufOffs + i; for( j = 0; j < SrcFilterLen; j++ ) { op0[ 0 ] = p[ 0 ]; op0++; p += FracCount; } } Table.alloc(( FracCount + 1 ) * FilterSize, Alignment ); } /** * Function creates the specified filter in the Table by copying it from * the SrcTable and filtering by ExtFilter. Function does nothing if * filter was already created. * * @param k Filter index to create, in the range 0 to FracCount, * inclusive. */ void createFilter( const int k ) { if( TableFillFlags[ k ] != 0 ) { return; } TableFillFlags[ k ] |= 1; const int ExtFilterLatency = ExtFilter.getCapacity() / 2; const int ResLatency = ExtFilterLatency + SrcFilterLen / 2; int ResLen = SrcFilterLen; if( ExtFilter.getCapacity() > 0 ) { ResLen += ExtFilter.getCapacity() - 1; } const int ResOffs = FilterLen / 2 - ResLatency; fptype* op = &Table[ k * FilterSize ]; int i; for( i = 0; i < ResOffs; i++ ) { op[ i ] = 0.0; } for( i = ResOffs + ResLen; i < FilterLen; i++ ) { op[ i ] = 0.0; } op += ResOffs; const double* const srcflt = &SrcTable[ k * SrcFilterLen ]; if( ExtFilter.getCapacity() == 0 ) { for( i = 0; i < ResLen; i++ ) { op[ i ] = (fptype) srcflt[ i ]; } return; } // Perform convolution of extflt and srcflt. const double* const extflt = &ExtFilter[ 0 ]; int j; for( j = 0; j < ResLen; j++ ) { int k = 0; int l = j - ExtFilter.getCapacity() + 1; int r = l + ExtFilter.getCapacity(); if( l < 0 ) { k -= l; l = 0; } if( r > SrcFilterLen ) { r = SrcFilterLen; } const double* const extfltb = extflt + k; const double* const srcfltb = srcflt + l; double s = 0.0; l = r - l; for( i = 0; i < l; i++ ) { s += extfltb[ i ] * srcfltb[ i ]; } op[ j ] = (fptype) s; } } }; /** * @brief Thread pool for multi-threaded image resizing operation. * * This base class is used to organize a multi-threaded image resizing * operation. The thread pool should consist of threads that initially wait * for a signal. Upon receiving a signal (via the startAllWorkloads() * function) each previously added thread should execute its workload's * process() function once, and return to the wait signal state again. The * thread pool should be also able to efficiently wait for all workloads to * finish via the waitAllWorkloadsToFinish() function. * * The image resizing algorithm makes calls to functions of this class. */ class CImageResizerThreadPool { public: CImageResizerThreadPool() { } virtual ~CImageResizerThreadPool() { } /** * @brief Thread pool's workload object class. * * This class should be used as a base class for objects that perform the * actual work spread over several threads. */ class CWorkload { public: virtual ~CWorkload() { } /** * Function that gets called from the thread when thread pool's * startAllWorkloads() function is called. */ virtual void process() = 0; }; /** * @return The suggested number of workloads (and their associated * threads) to add. The minimal value this function can return is 1. The * usual value may depend on the number of physical and virtual cores * present in the system, and on other considerations. */ virtual int getSuggestedWorkloadCount() const { return( 1 ); } /** * Function adds a new workload (and possibly thread) to the thread pool. * The caller decides how many parallel workloads (and threads) it * requires, but this number will not exceed the value returned by the * getSuggestedWorkloadCount() function. It is implementation-specific how * many workloads to associate with a single thread. But for efficiency * reasons each workload should be associated with its own thread. * * Note that the same set of workload objects will be processed each time * the startAllWorkloads() function is called. This means that workload * objects are added only once. The caller changes the state of the * workload objects and then calls the startAllWorkloads() function to * process them. * * @param Workload Workload object whose process() function will be called * from within the thread when the startAllWorkloads() function is called. */ virtual void addWorkload( CWorkload* const Workload ) { } /** * Function starts all workloads associated with threads previously added * via the addWorkload() function. It is assumed that this function * performs the necessary "memory barrier" (or "cache sync") kind of * operation so that all threads catch up the prior changes made to the * workload objects during their wait state. */ virtual void startAllWorkloads() { } /** * Function waits for all workloads to finish. */ virtual void waitAllWorkloadsToFinish() { } /** * Function removes all workloads previously added via the addWorkload() * function. This function gets called only after the * waitAllWorkloadsToFinish() function call. */ virtual void removeAllWorkloads() { } }; /** * @brief Resizing algorithm parameters structure. * * This structure holds all selectable parameters used by the resizing * algorithm at various stages, for both downsizing and upsizing. There are no * other parameters exist that can optimize the performance of the resizing * algorithm. Filter length parameters can take fractional values. * * Beside quality, these parameters (except Alpha parameters) directly affect * the computative cost of the resizing algorithm. It is possible to trade * the visual quality for computative cost. * * Anti-alias filtering during downsizing can be defined as a considerable * reduction of contrast of smallest features of an image. Unfortunately, such * de-contrasting partially affects features of all sizes thus producing a * non-linearity of frequency response. All pre-defined parameter sets are * described by 3 values separated by slashes. The first value is the * de-contrasting factor of small features (which are being removed) while * the second value is the de-contrasting factor of large features (which * should remain intact), with value of 1 equating to "no contrast change". * The third value is the optimization score (see below), with value of 0 * equating to the "perfect" linearity of frequency response. * * The pre-defined parameter sets offered by this library were auto-optimized * for the given LPFltBaseLen, IntFltLen and CorrFltAlpha values. The * optimization goal was to minimize the score: the sum of squares of the * difference between original and processed images (which was not actually * resized, k=1). The original image was a 0.5 megapixel uniformly-distributed * white-noise image with pixel intensities in the 0-1 range. Such goal * converges very well and produces filtering system with the flattest * frequency response possible for the given constraints. With this goal, * increasing the LPFltBaseLen value reduces the general amount of aliasing * artifacts. */ struct CImageResizerParams { double CorrFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the correction filter. The "usable" values are ///< in the narrow range 1.0 to 1.5. ///< double CorrFltLen; ///< Correction filter's length in samples (taps). The ///< "usable" range is narrow, 5.5 to 8, as to minimize the ///< "overcorrection" which is mathematically precise, but visually ///< unacceptable. ///< double IntFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the interpolation low-pass filter. The "usable" ///< values are in the range 1.5 to 2.5. ///< double IntFltCutoff; ///< Interpolation low-pass filter's cutoff frequency ///< (normalized, [0; 1]). The "usable" range is 0.6 to 0.8. ///< double IntFltLen; ///< Interpolation low-pass filter's length in samples ///< (taps). The length value should be at least 18 or otherwise a ///< "dark grid" artifact will be introduced if a further sharpening ///< is applied. IntFltLen together with other IntFlt parameters ///< should be tuned in a way that produces the flattest frequency ///< response in 0-0.5 normalized frequency range (this range is due ///< to 2X upsampling). ///< double LPFltAlpha; ///< Alpha parameter of the Peaked Cosine window ///< function used on the low-pass filter. The "usable" values are ///< in the range 1.5 to 6.5. ///< double LPFltBaseLen; ///< Base length of the low-pass (aka anti-aliasing ///< or reconstruction) filter, in samples (taps), further adjusted by ///< the actual cutoff frequency, upsampling and downsampling factors. ///< The "usable" range is between 6 and 9. ///< double LPFltCutoffMult; ///< Low-pass filter's cutoff frequency ///< multiplier. This value can be both below and above 1.0 as ///< low-pass filters are inserted on downsampling and upsampling ///< steps and always have corner frequency equal to or below 0.5pi. ///< This multiplier shifts low-pass filter's corner frequency towards ///< lower (if below 1.0) or higher (if above 1.0) frequencies. This ///< multiplier can be way below 1.0 since any additional ///< high-frequency damping will be partially corrected by the ///< correction filter. The "usable" range is 0.3 to 1.0. ///< CImageResizerParams() : HBFltAlpha( 1.75395 ) , HBFltCutoff( 0.40356 ) , HBFltLen( 22.00000 ) { } double HBFltAlpha; ///< Half-band filter's Alpha. Assigned internally. ///< double HBFltCutoff; ///< Half-band filter's cutoff point [0; 1]. Assigned ///< internally. ///< double HBFltLen; ///< Length of the half-band low-pass filter. Assigned ///< internally. Internally used to perform 2X or higher downsampling. ///< These filter parameters should be treated as "technical" and do ///< not require adjustment as they were tuned to suit all ///< combinations of other parameters. This half-band filter provides ///< a wide transition band (for minimal ringing artifacts) and a high ///< stop-band attenuation (for minimal aliasing). ///< }; /** * @brief The default set of resizing algorithm parameters * (10.01/1.029/0.019169). * * This is the default set of resizing parameters that was designed to deliver * a sharp image while still providing a low amount of ringing artifacts, and * having a reasonable computational cost. */ struct CImageResizerParamsDef : public CImageResizerParams { CImageResizerParamsDef() { CorrFltAlpha = 1.0;//10.01/1.88/1.029(522.43)/0.019169:258648,446808 CorrFltLen = 6.30770; IntFltAlpha = 2.27825; IntFltCutoff = 0.75493; IntFltLen = 18.0; LPFltAlpha = 3.40127; LPFltBaseLen = 7.78; LPFltCutoffMult = 0.78797; } }; /** * @brief Set of resizing algorithm parameters for ultra-low-ringing * performance (7.69/1.069/0.000245). * * This set of resizing algorithm parameters offers the lowest amount of * ringing this library is capable of providing while still offering a decent * quality. Low ringing is attained at the expense of higher aliasing * artifacts and a slightly reduced contrast. */ struct CImageResizerParamsULR : public CImageResizerParams { CImageResizerParamsULR() { CorrFltAlpha = 1.0;//7.69/1.97/1.069(31445.45)/0.000245:258627,436845 CorrFltLen = 5.83280; IntFltAlpha = 2.11453; IntFltCutoff = 0.73986; IntFltLen = 18.0; LPFltAlpha = 1.73455; LPFltBaseLen = 6.40; LPFltCutoffMult = 0.61314; } }; /** * @brief Set of resizing algorithm parameters for low-ringing performance * (7.86/1.065/0.000106). * * This set of resizing algorithm parameters offers a very low-ringing * performance at the expense of higher aliasing artifacts and a slightly * reduced contrast. */ struct CImageResizerParamsLR : public CImageResizerParams { CImageResizerParamsLR() { CorrFltAlpha = 1.0;//7.86/1.96/1.065(73865.02)/0.000106:258636,437381 CorrFltLen = 5.87671; IntFltAlpha = 2.25322; IntFltCutoff = 0.74090; IntFltLen = 18.0; LPFltAlpha = 1.79306; LPFltBaseLen = 7.00; LPFltCutoffMult = 0.68881; } }; /** * @brief Set of resizing algorithm parameters for lower-ringing performance * (8.86/1.046/0.010168). * * This set of resizing algorithm parameters offers a lower-ringing * performance in comparison to the default setting, at the expense of higher * aliasing artifacts and a slightly reduced contrast. */ struct CImageResizerParamsLow : public CImageResizerParams { CImageResizerParamsLow() { CorrFltAlpha = 1.0;//8.86/1.92/1.046(871.54)/0.010168:258647,442252 CorrFltLen = 6.09757; IntFltAlpha = 2.36704; IntFltCutoff = 0.74674; IntFltLen = 18.0; LPFltAlpha = 2.19427; LPFltBaseLen = 7.66; LPFltCutoffMult = 0.75380; } }; /** * @brief Set of resizing algorithm parameters for low-aliasing * resizing (11.81/1.012/0.038379). * * This set of resizing algorithm parameters offers a considerable * anti-aliasing performance with a good frequency response linearity (and * contrast). This is an intermediate setting between the default and Ultra * parameters. */ struct CImageResizerParamsHigh : public CImageResizerParams { CImageResizerParamsHigh() { CorrFltAlpha = 1.0;//11.81/1.83/1.012(307.84)/0.038379:258660,452719 CorrFltLen = 6.80909; IntFltAlpha = 2.44917; IntFltCutoff = 0.75856; IntFltLen = 18.0; LPFltAlpha = 4.39527; LPFltBaseLen = 8.18; LPFltCutoffMult = 0.79172; } }; /** * @brief Set of resizing algorithm parameters for ultra low-aliasing * resizing (13.65/1.001/0.000483). * * This set of resizing algorithm parameters offers a very considerable * anti-aliasing performance with a good frequency response linearity (and * contrast). This set of parameters is computationally expensive and may * produce ringing artifacts on sharp features. */ struct CImageResizerParamsUltra : public CImageResizerParams { CImageResizerParamsUltra() { CorrFltAlpha = 1.0;//13.65/1.79/1.001(28288.41)/0.000483:258658,457974 CorrFltLen = 7.48060; IntFltAlpha = 1.93750; IntFltCutoff = 0.75462; IntFltLen = 18.0; LPFltAlpha = 5.55209; LPFltBaseLen = 8.34; LPFltCutoffMult = 0.78002; } }; /** * @brief Image resizing variables class. * * This is an utility "catch all" class that defines various variables used * during image resizing. Several variables that are explicitly initialized in * this class' constructor are also used as additional "input" variables to * the image resizing function. These variables will not be changed by the * avir::CImageResizer<>::resizeImage() function. */ class CImageResizerVars { public: int ElCount; ///< The number of "fptype" elements used to store 1 pixel. ///< int ElCountIO; ///< The number of source and destination image's elements ///< used to store 1 pixel. ///< int fppack; ///< The number of atomic types stored in a single "fptype" ///< element. ///< int fpalign; ///< Suggested alignment size in bytes. This is not a ///< required alignment, because image resizing algorithm cannot be ///< made to have a strictly aligned data access in all cases (e.g. ///< de-interleaved interpolation cannot perform aligned accesses). ///< int elalign; ///< Length alignment of arrays of elements. This applies to ///< filters and intermediate buffers: this constant forces filters ///< and scanlines to have a length which is a multiple of this value, ///< for more efficient SIMD implementation. ///< int packmode; ///< 0 if interleaved packing, 1 if de-interleaved. ///< int BufLen[ 2 ]; ///< Intermediate buffers' lengths in "fptype" elements. int BufOffs[ 2 ]; ///< Offsets into the intermediate buffers, used to ///< provide prefix elements required during processing so that no ///< "out of range" access happens. This offset is a multiple of ///< ElCount if pixels are stored in interleaved form. ///< double k; ///< Resizing step coefficient, updated to reflect the actually ///< used coefficient during resizing. ///< double o; ///< Starting pixel offset inside the source image, updated to ///< reflect the actually used offset during resizing. ///< int ResizeStep; ///< Index of the resizing step in the latest filtering ///< steps array. ///< double InGammaMult; ///< Input gamma multiplier, used to convert input ///< data to 0 to 1 range. 0.0 if no gamma is in use. ///< double OutGammaMult; ///< Output gamma multiplier, used to convert data to ///< 0 to 255/65535 range. 0.0 if no gamma is in use. ///< double ox; ///< Start X pixel offset within source image (can be ///< negative). Positive offset moves image to the left. ///< double oy; ///< Start Y pixel offset within source image (can be ///< negative). Positive offset moves image to the top. ///< CImageResizerThreadPool* ThreadPool; ///< Thread pool to be used by the ///< image resizing function. Set to NULL to use single-threaded ///< processing. ///< bool UseSRGBGamma; ///< Perform sRGB gamma linearization (correction). ///< int BuildMode; ///< The build mode to use, for debugging purposes. Set to ///< -1 to select a minimal-complexity mode automatically. All build ///< modes deliver similar results with minor deviations. ///< int RndSeed; ///< Random seed parameter. This parameter may be incremented ///< after each random generator initialization. The use of this ///< variable depends on the ditherer implementation. ///< CImageResizerVars() : ox( 0.0 ) , oy( 0.0 ) , ThreadPool( NULL ) , UseSRGBGamma( false ) , BuildMode( -1 ) , RndSeed( 0 ) { } }; /** * @brief Image resizer's filtering step class. * * Class defines data to perform a single filtering step over a whole * horizontal or vertical scanline. Resizing consists of 1 or more steps that * may be performed before the actual resizing takes place. Filtering may also * follow a resizing step. Each step must ensure that scanline data contains * enough pixels to perform the next step (which may be resizing) without * exceeding scanline's bounds. * * A derived class must implement several "const" and "static" functions that * are used to perform the actual filtering in interleaved or de-interleaved * mode. * * @tparam fptype Floating point type to use for storing pixel elements. SIMD * types can be used: in this case each element may hold a whole pixel. * @tparam fptypeatom The atomic type the "fptype" consists of. */ template< class fptype, class fptypeatom > class CImageResizerFilterStep { public: bool IsUpsample; ///< "True" if this step is an upsampling step, "false" ///< if downsampling step. Should be set to "false" if ResampleFactor ///< equals 0. ///< int ResampleFactor; ///< Resample factor (>=1). If 0, this is a resizing ///< step. This value should be >1 if IsUpsample equals "true". ///< CBuffer< fptype > Flt; ///< Filter to use at this step. ///< CFltBuffer FltOrig; ///< Originally-designed filter. This buffer may not ///< be assigned. Assigned by filters that precede the resizing step ///< if such filter is planned to be embedded into the interpolation ///< filter as "external" filter. If IsUpsample=true and this filter ///< buffer is not empty, the upsampling step will not itself apply ///< any filtering over upsampled input scanline. ///< double DCGain; ///< DC gain which was applied to the filter. Not defined ///< if ResampleFactor = 0. ///< int FltLatency; ///< Filter's latency (group delay, shift) in pixels. ///< const CImageResizerVars* Vars; ///< Image resizing-related variables. ///< int InLen; ///< Input scanline's length in pixels. ///< int InBuf; ///< Input buffer index, 0 or 1. ///< int InPrefix; ///< Required input prefix pixels. These prefix pixels will ///< be filled with source scanline's first pixel value. If IsUpsample ///< is "true", this is the additional number of times the first pixel ///< will be filtered before processing scanline, this number is also ///< reflected in the OutPrefix. ///< int InSuffix; ///< Required input suffix pixels. These suffix pixels will ///< be filled with source scanline's last pixel value. If IsUpsample ///< is "true", this is the additional number of times the last pixel ///< will be filtered before processing scanline, this number is also ///< reflected in the OutSuffix. ///< int InElIncr; ///< Pixel element increment within the input buffer, used ///< during de-interleaved processing: in this case each image's ///< channel is stored independently, InElIncr elements apart. ///< int OutLen; ///< Length of the resulting scanline. ///< int OutBuf; ///< Output buffer index. 0 or 1; 2 for the last step. ///< int OutPrefix; ///< Required output prefix pixels. These prefix pixels ///< will not be pre-filled with any values. Value is valid only if ///< IsUpsample equals "true". ///< int OutSuffix; ///< Required input suffix pixels. These suffix pixels will ///< not be pre-filled with any values. Value is valid only if ///< IsUpsample equals "true". ///< int OutElIncr; ///< Pixel element increment within the output buffer, used ///< during de-interleaved processing. Equals to the InBufElIncr of ///< the next step. ///< CBuffer< fptype > PrefixDC; ///< DC component fluctuations added at the ///< start of the resulting scanline, used when IsUpsample equals ///< "true". ///< CBuffer< fptype > SuffixDC; ///< DC component fluctuations added at the ///< end of the resulting scanline, used when IsUpsample equals ///< "true". ///< int EdgePixelCount; ///< The number of edge pixels added. Affects the ///< initial position within the input scanline, used to produce edge ///< pixels. This variable is used and should be defined when ///< IsUpsample=false and ResampleFactor>0. When assigning this ///< variable it is also necessary to update InPrefix, OutLen and ///< Vars.o variables. ///< static const int EdgePixelCountDef = 3; ///< The default number of pixels ///< additionally produced at scanline edges during filtering. This is ///< required to reduce edge artifacts. ///< /** * @brief Resizing position structure. * * Structure holds resizing position and pointer to fractional delay * filter. */ struct CResizePos { int SrcPosInt; ///< Source scanline position. ///< int fti; ///< Fractional delay filter index. ///< const fptype* ftp; ///< Fractional delay filter pointer. ///< fptypeatom x; ///< Interpolation coefficient between delay filters. ///< int SrcOffs; ///< Source scanline offset. ///< }; /** * @brief Resizing positions buffer class. * * This class combines buffer together with variables that define resizing * stepping. */ class CRPosBuf : public CBuffer< CResizePos > { public: double k; ///< Resizing step. ///< double o; ///< Resizing offset. ///< int FracCount; ///< The number of fractional delay filters in a filter ///< bank used together with this buffer. ///< }; /** * @brief Resizing positions buffer array class. * * This class combines structure array of the CRPosBuf class objects with * the function that locates or creates buffer with the required resizing * stepping. */ class CRPosBufArray : public CStructArray< CRPosBuf > { public: using CStructArray< CRPosBuf > :: add; using CStructArray< CRPosBuf > :: getItemCount; /** * Function returns the resizing positions buffer with the required * stepping. If no such buffer exists, it is created. * * @param k Resizing step. * @param o Resizing offset. * @param FracCount The number of fractional delay filters in a filter * bank used together with this buffer. * @return Reference to the CRPosBuf object. */ CRPosBuf& getRPosBuf( const double k, const double o, const int FracCount ) { int i; for( i = 0; i < getItemCount(); i++ ) { CRPosBuf& Buf = (*this)[ i ]; if( Buf.k == k && Buf.o == o && Buf.FracCount == FracCount ) { return( Buf ); } } CRPosBuf& NewBuf = add(); NewBuf.k = k; NewBuf.o = o; NewBuf.FracCount = FracCount; return( NewBuf ); } }; CRPosBuf* RPosBuf; ///< Resizing positions buffer. Used when ///< ResampleFactor equals 0 (resizing step). ///< CDSPFracFilterBankLin< fptype >* FltBank; ///< Filter bank in use by *this ///< resizing step. ///< }; /** * @brief Interleaved filtering steps implementation class. * * This class implements scanline filtering functions in interleaved mode. * This means that each pixel is processed independently, not in groups. * * @tparam fptype Floating point type to use for storing pixel elements. SIMD * types can be used: in this case each element may hold a whole pixel. * @tparam fptypeatom The atomic type the "fptype" consists of. */ template< class fptype, class fptypeatom > class CImageResizerFilterStepINL : public CImageResizerFilterStep< fptype, fptypeatom > { public: using CImageResizerFilterStep< fptype, fptypeatom > :: IsUpsample; using CImageResizerFilterStep< fptype, fptypeatom > :: ResampleFactor; using CImageResizerFilterStep< fptype, fptypeatom > :: Flt; using CImageResizerFilterStep< fptype, fptypeatom > :: FltOrig; using CImageResizerFilterStep< fptype, fptypeatom > :: FltLatency; using CImageResizerFilterStep< fptype, fptypeatom > :: Vars; using CImageResizerFilterStep< fptype, fptypeatom > :: InLen; using CImageResizerFilterStep< fptype, fptypeatom > :: InPrefix; using CImageResizerFilterStep< fptype, fptypeatom > :: InSuffix; using CImageResizerFilterStep< fptype, fptypeatom > :: OutLen; using CImageResizerFilterStep< fptype, fptypeatom > :: OutPrefix; using CImageResizerFilterStep< fptype, fptypeatom > :: OutSuffix; using CImageResizerFilterStep< fptype, fptypeatom > :: PrefixDC; using CImageResizerFilterStep< fptype, fptypeatom > :: SuffixDC; using CImageResizerFilterStep< fptype, fptypeatom > :: RPosBuf; using CImageResizerFilterStep< fptype, fptypeatom > :: FltBank; using CImageResizerFilterStep< fptype, fptypeatom > :: EdgePixelCount; /** * Function performs "packing" of a scanline and type conversion. * Scanline, depending on the "fptype" can be potentially stored as a * packed SIMD values having a certain atomic type. If required, the sRGB * gamma correction is applied. * * @param ip Input scanline. * @param op0 Output scanline. * @param l0 The number of pixels to "pack". */ template< class Tin > void packScanline( const Tin* ip, fptype* const op0, const int l0 ) const { const int ElCount = Vars -> ElCount; const int ElCountIO = Vars -> ElCountIO; fptype* op = op0; int l = l0; if( !Vars -> UseSRGBGamma ) { if( ElCountIO == 1 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = (fptypeatom) ip[ 0 ]; op += ElCount; ip++; l--; } } else if( ElCountIO == 4 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = (fptypeatom) ip[ 0 ]; v[ 1 ] = (fptypeatom) ip[ 1 ]; v[ 2 ] = (fptypeatom) ip[ 2 ]; v[ 3 ] = (fptypeatom) ip[ 3 ]; op += ElCount; ip += 4; l--; } } else if( ElCountIO == 3 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = (fptypeatom) ip[ 0 ]; v[ 1 ] = (fptypeatom) ip[ 1 ]; v[ 2 ] = (fptypeatom) ip[ 2 ]; op += ElCount; ip += 3; l--; } } else if( ElCountIO == 2 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = (fptypeatom) ip[ 0 ]; v[ 1 ] = (fptypeatom) ip[ 1 ]; op += ElCount; ip += 2; l--; } } } else { const fptypeatom gm = (fptypeatom) Vars -> InGammaMult; if( ElCountIO == 1 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = convertSRGB2Lin( (fptypeatom) ip[ 0 ] * gm ); op += ElCount; ip++; l--; } } else if( ElCountIO == 4 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = convertSRGB2Lin( (fptypeatom) ip[ 0 ] * gm ); v[ 1 ] = convertSRGB2Lin( (fptypeatom) ip[ 1 ] * gm ); v[ 2 ] = convertSRGB2Lin( (fptypeatom) ip[ 2 ] * gm ); v[ 3 ] = convertSRGB2Lin( (fptypeatom) ip[ 3 ] * gm ); op += ElCount; ip += 4; l--; } } else if( ElCountIO == 3 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = convertSRGB2Lin( (fptypeatom) ip[ 0 ] * gm ); v[ 1 ] = convertSRGB2Lin( (fptypeatom) ip[ 1 ] * gm ); v[ 2 ] = convertSRGB2Lin( (fptypeatom) ip[ 2 ] * gm ); op += ElCount; ip += 3; l--; } } else if( ElCountIO == 2 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op; v[ 0 ] = convertSRGB2Lin( (fptypeatom) ip[ 0 ] * gm ); v[ 1 ] = convertSRGB2Lin( (fptypeatom) ip[ 1 ] * gm ); op += ElCount; ip += 2; l--; } } } const int ZeroCount = ElCount * Vars -> fppack - ElCountIO; op = op0; l = l0; if( ZeroCount == 1 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op + ElCountIO; v[ 0 ] = (fptypeatom) 0; op += ElCount; l--; } } else if( ZeroCount == 2 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op + ElCountIO; v[ 0 ] = (fptypeatom) 0; v[ 1 ] = (fptypeatom) 0; op += ElCount; l--; } } else if( ZeroCount == 3 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) op + ElCountIO; v[ 0 ] = (fptypeatom) 0; v[ 1 ] = (fptypeatom) 0; v[ 2 ] = (fptypeatom) 0; op += ElCount; l--; } } } /** * Function applies Linear to sRGB gamma correction to the specified * scanline. * * @param p Scanline. * @param l The number of pixels to de-linearize. * @param Vars0 Image resizing-related variables. */ static void applySRGBGamma( fptype* p, int l, const CImageResizerVars& Vars0 ) { const int ElCount = Vars0.ElCount; const int ElCountIO = Vars0.ElCountIO; const fptypeatom gm = (fptypeatom) Vars0.OutGammaMult; if( ElCountIO == 1 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) p; v[ 0 ] = convertLin2SRGB( v[ 0 ]) * gm; p += ElCount; l--; } } else if( ElCountIO == 4 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) p; v[ 0 ] = convertLin2SRGB( v[ 0 ]) * gm; v[ 1 ] = convertLin2SRGB( v[ 1 ]) * gm; v[ 2 ] = convertLin2SRGB( v[ 2 ]) * gm; v[ 3 ] = convertLin2SRGB( v[ 3 ]) * gm; p += ElCount; l--; } } else if( ElCountIO == 3 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) p; v[ 0 ] = convertLin2SRGB( v[ 0 ]) * gm; v[ 1 ] = convertLin2SRGB( v[ 1 ]) * gm; v[ 2 ] = convertLin2SRGB( v[ 2 ]) * gm; p += ElCount; l--; } } else if( ElCountIO == 2 ) { while( l > 0 ) { fptypeatom* v = (fptypeatom*) p; v[ 0 ] = convertLin2SRGB( v[ 0 ]) * gm; v[ 1 ] = convertLin2SRGB( v[ 1 ]) * gm; p += ElCount; l--; } } } /** * Function converts vertical scanline to horizontal scanline. This * function is called by the image resizer when image is resized * vertically. This means that the vertical scanline is stored in the * same format produced by the packScanline() and maintained by other * filtering functions. * * @param ip Input vertical scanline. * @param op Output buffer (temporary buffer used during resizing). * @param SrcLen The number of pixels in the input scanline, also used to * calculate input buffer increment. * @param SrcIncr Input buffer increment to the next vertical pixel. */ void convertVtoH( const fptype* ip, fptype* op, const int SrcLen, const int SrcIncr ) const { const int ElCount = Vars -> ElCount; int j; if( ElCount == 1 ) { for( j = 0; j < SrcLen; j++ ) { op[ 0 ] = ip[ 0 ]; ip += SrcIncr; op++; } } else if( ElCount == 4 ) { for( j = 0; j < SrcLen; j++ ) { op[ 0 ] = ip[ 0 ]; op[ 1 ] = ip[ 1 ]; op[ 2 ] = ip[ 2 ]; op[ 3 ] = ip[ 3 ]; ip += SrcIncr; op += 4; } } else if( ElCount == 3 ) { for( j = 0; j < SrcLen; j++ ) { op[ 0 ] = ip[ 0 ]; op[ 1 ] = ip[ 1 ]; op[ 2 ] = ip[ 2 ]; ip += SrcIncr; op += 3; } } else if( ElCount == 2 ) { for( j = 0; j < SrcLen; j++ ) { op[ 0 ] = ip[ 0 ]; op[ 1 ] = ip[ 1 ]; ip += SrcIncr; op += 2; } } } /** * Function performs "unpacking" of a scanline and type conversion * (truncation is used when floating point is converted to integer). * Scanline, depending on the "fptype" can be potentially stored as a * packed SIMD values having a certain atomic type. The unpacking function * assumes that scanline is stored in the style produced by the * packScanline() function. * * @param ip Input scanline. * @param op Output scanline. * @param l The number of pixels to "unpack". * @param Vars0 Image resizing-related variables. */ template< class Tout > static void unpackScanline( const fptype* ip, Tout* op, int l, const CImageResizerVars& Vars0 ) { const int ElCount = Vars0.ElCount; const int ElCountIO = Vars0.ElCountIO; /* dprintf(2, "BOOP ElCount=%d ElCountIO=%d l=%d op=%p\n", ElCount, ElCountIO, l, op); */ if( ElCountIO == 1 ) { while( l > 0 ) { const fptypeatom* v = (const fptypeatom*) ip; op[ 0 ] = (Tout) v[ 0 ]; ip += ElCount; op++; l--; } } else if( ElCountIO == 4 ) { while( l > 0 ) { const fptypeatom* v = (const fptypeatom*) ip; op[ 0 ] = (Tout) v[ 0 ]; op[ 1 ] = (Tout) v[ 1 ]; op[ 2 ] = (Tout) v[ 2 ]; op[ 3 ] = (Tout) v[ 3 ]; ip += ElCount; op += 4; l--; } } else if( ElCountIO == 3 ) { while( l > 0 ) { const fptypeatom* v = (const fptypeatom*) ip; /* DebugBreak(); */ /* dprintf(2, "BOOP ElCount=%d ElCountIO=%d l=%d op=%p v=%p v[0]=%d\n", ElCount, ElCountIO, l, op, v, (Tout)v[0]); */ op[ 0 ] = (Tout) v[ 0 ]; op[ 1 ] = (Tout) v[ 1 ]; op[ 2 ] = (Tout) v[ 2 ]; ip += ElCount; op += 3; l--; } } else if( ElCountIO == 2 ) { while( l > 0 ) { const fptypeatom* v = (const fptypeatom*) ip; op[ 0 ] = (Tout) v[ 0 ]; op[ 1 ] = (Tout) v[ 1 ]; ip += ElCount; op += 2; l--; } } } /** * Function prepares input scanline buffer for *this filtering step. * Left- and right-most pixels are replicated to make sure no buffer * overrun happens. Such approach also allows to bypass any pointer * range checks. * * @param Src Source buffer. */ void prepareInBuf( fptype* Src ) const { if( IsUpsample || InPrefix + InSuffix == 0 ) { return; } const int ElCount = Vars -> ElCount; replicateArray( Src, ElCount, Src - ElCount, InPrefix, -ElCount ); Src += ( InLen - 1 ) * ElCount; replicateArray( Src, ElCount, Src + ElCount, InSuffix, ElCount ); } /** * Function peforms scanline upsampling with filtering. * * @param Src Source scanline buffer (length = this -> InLen). Source * scanline increment will be equal to ElCount. * @param Dst Destination scanline buffer. */ void doUpsample( const fptype* const Src, fptype* const Dst ) const { const int ElCount = Vars -> ElCount; fptype* op0 = &Dst[ -OutPrefix * ElCount ]; memset( op0, 0, ( OutPrefix + OutLen + OutSuffix ) * ElCount * sizeof( fptype )); const fptype* ip = Src; const int opstep = ElCount * ResampleFactor; int l; if( FltOrig.getCapacity() > 0 ) { // Do not perform filtering, only upsample. op0 += ( OutPrefix % ResampleFactor ) * ElCount; l = OutPrefix / ResampleFactor; if( ElCount == 1 ) { while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while( l >= 0 ) { op0[ 0 ] = ip[ 0 ]; op0 += opstep; l--; } } else if( ElCount == 4 ) { while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0[ 2 ] = ip[ 2 ]; op0[ 3 ] = ip[ 3 ]; op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0[ 2 ] = ip[ 2 ]; op0[ 3 ] = ip[ 3 ]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while( l >= 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0[ 2 ] = ip[ 2 ]; op0[ 3 ] = ip[ 3 ]; op0 += opstep; l--; } } else if( ElCount == 3 ) { while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0[ 2 ] = ip[ 2 ]; op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0[ 2 ] = ip[ 2 ]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while( l >= 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0[ 2 ] = ip[ 2 ]; op0 += opstep; l--; } } else if( ElCount == 2 ) { while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0 += opstep; ip += ElCount; l--; } l = OutSuffix / ResampleFactor; while( l >= 0 ) { op0[ 0 ] = ip[ 0 ]; op0[ 1 ] = ip[ 1 ]; op0 += opstep; l--; } } return; } const fptype* const f = Flt; const int flen = Flt.getCapacity(); fptype* op; int i; if( ElCount == 1 ) { l = InPrefix; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ i ] += f[ i ] * ip[ 0 ]; } op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ i ] += f[ i ] * ip[ 0 ]; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while( l >= 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ i ] += f[ i ] * ip[ 0 ]; } op0 += opstep; l--; } } else if( ElCount == 4 ) { l = InPrefix; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op[ 2 ] += f[ i ] * ip[ 2 ]; op[ 3 ] += f[ i ] * ip[ 3 ]; op += 4; } op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op[ 2 ] += f[ i ] * ip[ 2 ]; op[ 3 ] += f[ i ] * ip[ 3 ]; op += 4; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while( l >= 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op[ 2 ] += f[ i ] * ip[ 2 ]; op[ 3 ] += f[ i ] * ip[ 3 ]; op += 4; } op0 += opstep; l--; } } else if( ElCount == 3 ) { l = InPrefix; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op[ 2 ] += f[ i ] * ip[ 2 ]; op += 3; } op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op[ 2 ] += f[ i ] * ip[ 2 ]; op += 3; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while( l >= 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op[ 2 ] += f[ i ] * ip[ 2 ]; op += 3; } op0 += opstep; l--; } } else if( ElCount == 2 ) { l = InPrefix; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op += 2; } op0 += opstep; l--; } l = InLen - 1; while( l > 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op += 2; } ip += ElCount; op0 += opstep; l--; } l = InSuffix; while( l >= 0 ) { op = op0; for( i = 0; i < flen; i++ ) { op[ 0 ] += f[ i ] * ip[ 0 ]; op[ 1 ] += f[ i ] * ip[ 1 ]; op += 2; } op0 += opstep; l--; } } op = op0; const fptype* dc = SuffixDC; l = SuffixDC.getCapacity(); if( ElCount == 1 ) { for( i = 0; i < l; i++ ) { op[ i ] += ip[ 0 ] * dc[ i ]; } } else if( ElCount == 4 ) { while( l > 0 ) { op[ 0 ] += ip[ 0 ] * dc[ 0 ]; op[ 1 ] += ip[ 1 ] * dc[ 0 ]; op[ 2 ] += ip[ 2 ] * dc[ 0 ]; op[ 3 ] += ip[ 3 ] * dc[ 0 ]; dc++; op += 4; l--; } } else if( ElCount == 3 ) { while( l > 0 ) { op[ 0 ] += ip[ 0 ] * dc[ 0 ]; op[ 1 ] += ip[ 1 ] * dc[ 0 ]; op[ 2 ] += ip[ 2 ] * dc[ 0 ]; dc++; op += 3; l--; } } else if( ElCount == 2 ) { while( l > 0 ) { op[ 0 ] += ip[ 0 ] * dc[ 0 ]; op[ 1 ] += ip[ 1 ] * dc[ 0 ]; dc++; op += 2; l--; } } ip = Src; op = Dst - InPrefix * opstep; dc = PrefixDC; l = PrefixDC.getCapacity(); if( ElCount == 1 ) { for( i = 0; i < l; i++ ) { op[ i ] += ip[ 0 ] * dc[ i ]; } } else if( ElCount == 4 ) { while( l > 0 ) { op[ 0 ] += ip[ 0 ] * dc[ 0 ]; op[ 1 ] += ip[ 1 ] * dc[ 0 ]; op[ 2 ] += ip[ 2 ] * dc[ 0 ]; op[ 3 ] += ip[ 3 ] * dc[ 0 ]; dc++; op += 4; l--; } } else if( ElCount == 3 ) { while( l > 0 ) { op[ 0 ] += ip[ 0 ] * dc[ 0 ]; op[ 1 ] += ip[ 1 ] * dc[ 0 ]; op[ 2 ] += ip[ 2 ] * dc[ 0 ]; dc++; op += 3; l--; } } else if( ElCount == 2 ) { while( l > 0 ) { op[ 0 ] += ip[ 0 ] * dc[ 0 ]; op[ 1 ] += ip[ 1 ] * dc[ 0 ]; dc++; op += 2; l--; } } } /** * Function peforms scanline filtering with optional downsampling. * Function makes use of the symmetry of the filter. * * @param Src Source scanline buffer (length = this -> InLen). Source * scanline increment will be equal to ElCount. * @param Dst Destination scanline buffer. * @param DstIncr Destination scanline buffer increment, used for * horizontal or vertical scanline stepping. */ void doFilter( const fptype* const Src, fptype* Dst, const int DstIncr ) const { const int ElCount = Vars -> ElCount; const fptype* const f = &Flt[ FltLatency ]; const int flen = FltLatency + 1; const int ipstep = ElCount * ResampleFactor; const fptype* ip = Src - EdgePixelCount * ipstep; const fptype* ip1; const fptype* ip2; int l = OutLen; int i; if( ElCount == 1 ) { while( l > 0 ) { fptype s = f[ 0 ] * ip[ 0 ]; ip1 = ip; ip2 = ip; for( i = 1; i < flen; i++ ) { ip1++; ip2--; s += f[ i ] * ( ip1[ 0 ] + ip2[ 0 ]); } Dst[ 0 ] = s; Dst += DstIncr; ip += ipstep; l--; } } else if( ElCount == 4 ) { while( l > 0 ) { fptype s1 = f[ 0 ] * ip[ 0 ]; fptype s2 = f[ 0 ] * ip[ 1 ]; fptype s3 = f[ 0 ] * ip[ 2 ]; fptype s4 = f[ 0 ] * ip[ 3 ]; ip1 = ip; ip2 = ip; for( i = 1; i < flen; i++ ) { ip1 += 4; ip2 -= 4; s1 += f[ i ] * ( ip1[ 0 ] + ip2[ 0 ]); s2 += f[ i ] * ( ip1[ 1 ] + ip2[ 1 ]); s3 += f[ i ] * ( ip1[ 2 ] + ip2[ 2 ]); s4 += f[ i ] * ( ip1[ 3 ] + ip2[ 3 ]); } Dst[ 0 ] = s1; Dst[ 1 ] = s2; Dst[ 2 ] = s3; Dst[ 3 ] = s4; Dst += DstIncr; ip += ipstep; l--; } } else if( ElCount == 3 ) { while( l > 0 ) { fptype s1 = f[ 0 ] * ip[ 0 ]; fptype s2 = f[ 0 ] * ip[ 1 ]; fptype s3 = f[ 0 ] * ip[ 2 ]; ip1 = ip; ip2 = ip; for( i = 1; i < flen; i++ ) { ip1 += 3; ip2 -= 3; s1 += f[ i ] * ( ip1[ 0 ] + ip2[ 0 ]); s2 += f[ i ] * ( ip1[ 1 ] + ip2[ 1 ]); s3 += f[ i ] * ( ip1[ 2 ] + ip2[ 2 ]); } Dst[ 0 ] = s1; Dst[ 1 ] = s2; Dst[ 2 ] = s3; Dst += DstIncr; ip += ipstep; l--; } } else if( ElCount == 2 ) { while( l > 0 ) { fptype s1 = f[ 0 ] * ip[ 0 ]; fptype s2 = f[ 0 ] * ip[ 1 ]; ip1 = ip; ip2 = ip; for( i = 1; i < flen; i++ ) { ip1 += 2; ip2 -= 2; s1 += f[ i ] * ( ip1[ 0 ] + ip2[ 0 ]); s2 += f[ i ] * ( ip1[ 1 ] + ip2[ 1 ]); } Dst[ 0 ] = s1; Dst[ 1 ] = s2; Dst += DstIncr; ip += ipstep; l--; } } } /** * Function performs resizing of a single scanline. This function does * not "know" about the length of the source scanline buffer. This buffer * should be padded with enough pixels so that ( SrcPos - FilterLenD2 ) is * always >= 0 and ( SrcPos + ( DstLineLen - 1 ) * k + FilterLenD2 + 1 ) * does not exceed source scanline's buffer length. SrcLine's increment is * assumed to be equal to ElCount. * * @param SrcLine Source scanline buffer. * @param DstLine Destination (resized) scanline buffer. * @param DstLineIncr Destination scanline position increment, used for * horizontal or vertical scanline stepping. * @param xx Temporary buffer, of size FltBank -> getFilterLen(), must be * aligned by fpclass :: fpalign. */ void doResize( const fptype* SrcLine, fptype* DstLine, const int DstLineIncr, fptype* const ) const { const int IntFltLen = FltBank -> getFilterLen(); const int ElCount = Vars -> ElCount; const typename CImageResizerFilterStep< fptype, fptypeatom > :: CResizePos* rpos = &(*RPosBuf)[ 0 ]; const typename CImageResizerFilterStep< fptype, fptypeatom > :: CResizePos* const rpose = rpos + OutLen; #define AVIR_RESIZE_PART1 \ while( rpos < rpose ) \ { \ const fptype x = (fptype) rpos -> x; \ const fptype* const ftp = rpos -> ftp; \ const fptype* const ftp2 = ftp + IntFltLen; \ const fptype* Src = SrcLine + rpos -> SrcOffs; \ int i; #define AVIR_RESIZE_PART1nx \ while( rpos < rpose ) \ { \ const fptype* const ftp = rpos -> ftp; \ const fptype* Src = SrcLine + rpos -> SrcOffs; \ int i; #define AVIR_RESIZE_PART2 \ DstLine += DstLineIncr; \ rpos++; \ } if( FltBank -> getOrder() == 1 ) { if( ElCount == 1 ) { AVIR_RESIZE_PART1 fptype sum = 0.0; for( i = 0; i < IntFltLen; i++ ) { sum += ( ftp[ i ] + ftp2[ i ] * x ) * Src[ i ]; } DstLine[ 0 ] = sum; AVIR_RESIZE_PART2 } else if( ElCount == 4 ) { AVIR_RESIZE_PART1 fptype sum[ 4 ]; sum[ 0 ] = 0.0; sum[ 1 ] = 0.0; sum[ 2 ] = 0.0; sum[ 3 ] = 0.0; for( i = 0; i < IntFltLen; i++ ) { const fptype xx = ftp[ i ] + ftp2[ i ] * x; sum[ 0 ] += xx * Src[ 0 ]; sum[ 1 ] += xx * Src[ 1 ]; sum[ 2 ] += xx * Src[ 2 ]; sum[ 3 ] += xx * Src[ 3 ]; Src += 4; } DstLine[ 0 ] = sum[ 0 ]; DstLine[ 1 ] = sum[ 1 ]; DstLine[ 2 ] = sum[ 2 ]; DstLine[ 3 ] = sum[ 3 ]; AVIR_RESIZE_PART2 } else if( ElCount == 3 ) { AVIR_RESIZE_PART1 fptype sum[ 3 ]; sum[ 0 ] = 0.0; sum[ 1 ] = 0.0; sum[ 2 ] = 0.0; for( i = 0; i < IntFltLen; i++ ) { const fptype xx = ftp[ i ] + ftp2[ i ] * x; sum[ 0 ] += xx * Src[ 0 ]; sum[ 1 ] += xx * Src[ 1 ]; sum[ 2 ] += xx * Src[ 2 ]; Src += 3; } DstLine[ 0 ] = sum[ 0 ]; DstLine[ 1 ] = sum[ 1 ]; DstLine[ 2 ] = sum[ 2 ]; AVIR_RESIZE_PART2 } else if( ElCount == 2 ) { AVIR_RESIZE_PART1 fptype sum[ 2 ]; sum[ 0 ] = 0.0; sum[ 1 ] = 0.0; for( i = 0; i < IntFltLen; i++ ) { const fptype xx = ftp[ i ] + ftp2[ i ] * x; sum[ 0 ] += xx * Src[ 0 ]; sum[ 1 ] += xx * Src[ 1 ]; Src += 2; } DstLine[ 0 ] = sum[ 0 ]; DstLine[ 1 ] = sum[ 1 ]; AVIR_RESIZE_PART2 } } else { if( ElCount == 1 ) { AVIR_RESIZE_PART1nx fptype sum = 0.0; for( i = 0; i < IntFltLen; i++ ) { sum += ftp[ i ] * Src[ i ]; } DstLine[ 0 ] = sum; AVIR_RESIZE_PART2 } else if( ElCount == 4 ) { AVIR_RESIZE_PART1nx fptype sum[ 4 ]; sum[ 0 ] = 0.0; sum[ 1 ] = 0.0; sum[ 2 ] = 0.0; sum[ 3 ] = 0.0; for( i = 0; i < IntFltLen; i++ ) { const fptype xx = ftp[ i ]; sum[ 0 ] += xx * Src[ 0 ]; sum[ 1 ] += xx * Src[ 1 ]; sum[ 2 ] += xx * Src[ 2 ]; sum[ 3 ] += xx * Src[ 3 ]; Src += 4; } DstLine[ 0 ] = sum[ 0 ]; DstLine[ 1 ] = sum[ 1 ]; DstLine[ 2 ] = sum[ 2 ]; DstLine[ 3 ] = sum[ 3 ]; AVIR_RESIZE_PART2 } else if( ElCount == 3 ) { AVIR_RESIZE_PART1nx fptype sum[ 3 ]; sum[ 0 ] = 0.0; sum[ 1 ] = 0.0; sum[ 2 ] = 0.0; for( i = 0; i < IntFltLen; i++ ) { const fptype xx = ftp[ i ]; sum[ 0 ] += xx * Src[ 0 ]; sum[ 1 ] += xx * Src[ 1 ]; sum[ 2 ] += xx * Src[ 2 ]; Src += 3; } DstLine[ 0 ] = sum[ 0 ]; DstLine[ 1 ] = sum[ 1 ]; DstLine[ 2 ] = sum[ 2 ]; AVIR_RESIZE_PART2 } else if( ElCount == 2 ) { AVIR_RESIZE_PART1nx fptype sum[ 2 ]; sum[ 0 ] = 0.0; sum[ 1 ] = 0.0; for( i = 0; i < IntFltLen; i++ ) { const fptype xx = ftp[ i ]; sum[ 0 ] += xx * Src[ 0 ]; sum[ 1 ] += xx * Src[ 1 ]; Src += 2; } DstLine[ 0 ] = sum[ 0 ]; DstLine[ 1 ] = sum[ 1 ]; AVIR_RESIZE_PART2 } } } #undef AVIR_RESIZE_PART2 #undef AVIR_RESIZE_PART1nx #undef AVIR_RESIZE_PART1 }; /** * @brief Image resizer's default dithering class. * * This class defines an object that performs rounding, clipping and dithering * operations over horizontal scanline pixels before scanline is stored in the * output buffer. * * The ditherer should expect the same storage order of the pixels in a * scanline as used in the "filtering step" class. So, a separate ditherer * class should be defined for each scanline pixel storage style. The default * ditherer implements a simple rounding without dithering: it can be used for * an efficient dithering method which can be multi-threaded. * * @tparam fptype Floating point type to use for storing pixel data. SIMD * types can be used. */ template< class fptype > class CImageResizerDithererDefINL { public: /** * Function initializes the ditherer object. * * @param aLen Scanline length in pixels to process. * @param aVars Image resizing-related variables. * @param aTrMul Bit-depth truncation multiplier. 1 - no additional * truncation. * @param aPkOut Peak output value allowed. */ void init( const int aLen, const CImageResizerVars& aVars, const double aTrMul, const double aPkOut ) { Len = aLen; Vars = &aVars; LenE = aLen * Vars -> ElCount; TrMul0 = aTrMul; PkOut0 = aPkOut; } /** * @return "True" if dithering is recursive relative to scanlines meaning * multi-threaded execution is not supported by this dithering method. */ static bool isRecursive() { return( false ); } /** * Function performs rounding and clipping operations. * * @param ResScanline The buffer containing the final scanline. */ void dither( fptype* const ResScanline ) const { const fptype c0 = 0.0; const fptype PkOut = (fptype) PkOut0; int j; if( TrMul0 == 1.0 ) { // Optimization - do not perform bit depth truncation. for( j = 0; j < LenE; j++ ) { ResScanline[ j ] = clamp( round( ResScanline[ j ]), c0, PkOut ); } } else { const fptype TrMul = (fptype) TrMul0; for( j = 0; j < LenE; j++ ) { const fptype z0 = round( ResScanline[ j ] / TrMul ) * TrMul; ResScanline[ j ] = clamp( z0, c0, PkOut ); } } } protected: int Len; ///< Scanline's length in pixels. ///< const CImageResizerVars* Vars; ///< Image resizing-related variables. ///< int LenE; ///< = LenE * ElCount. ///< double TrMul0; ///< Bit-depth truncation multiplier. ///< double PkOut0; ///< Peak output value allowed. ///< }; /** * @brief Image resizer's error-diffusion dithering class, interleaved mode. * * This ditherer implements error-diffusion dithering which looks good, and * whose results are compressed by PNG well. This implementation uses * weighting coefficients obtained via machine optimization and visual * evaluation. * * @tparam fptype Floating point type to use for storing pixel data. SIMD * types can be used. */ template< class fptype > class CImageResizerDithererErrdINL : public CImageResizerDithererDefINL< fptype > { public: /** * Function initializes the ditherer object. * * @param aLen Scanline length in pixels to process. * @param aVars Image resizing-related variables. * @param aTrMul Bit-depth truncation multiplier. 1 - no additional * truncation. * @param aPkOut Peak output value allowed. */ void init( const int aLen, const CImageResizerVars& aVars, const double aTrMul, const double aPkOut ) { CImageResizerDithererDefINL< fptype > :: init( aLen, aVars, aTrMul, aPkOut ); ResScanlineDith0.alloc( LenE + Vars -> ElCount, sizeof( fptype )); ResScanlineDith = ResScanlineDith0 + Vars -> ElCount; int i; for( i = 0; i < LenE + Vars -> ElCount; i++ ) { ResScanlineDith0[ i ] = 0.0; } } static bool isRecursive() { return( true ); } void dither( fptype* const ResScanline ) { const int ElCount = Vars -> ElCount; const fptype c0 = 0.0; const fptype TrMul = (fptype) TrMul0; const fptype PkOut = (fptype) PkOut0; int j; for( j = 0; j < LenE; j++ ) { ResScanline[ j ] += ResScanlineDith[ j ]; ResScanlineDith[ j ] = 0.0; } for( j = 0; j < LenE - ElCount; j++ ) { // Perform rounding, noise estimation and saturation. const fptype z0 = round( ResScanline[ j ] / TrMul ) * TrMul; const fptype Noise = ResScanline[ j ] - z0; ResScanline[ j ] = clamp( z0, c0, PkOut ); ResScanline[ j + ElCount ] += Noise * (fptype) 0.364842; ResScanlineDith[ j - ElCount ] += Noise * (fptype) 0.207305; ResScanlineDith[ j ] += Noise * (fptype) 0.364842; ResScanlineDith[ j + ElCount ] += Noise * (fptype) 0.063011; } while( j < LenE ) { const fptype z0 = round( ResScanline[ j ] / TrMul ) * TrMul; const fptype Noise = ResScanline[ j ] - z0; ResScanline[ j ] = clamp( z0, c0, PkOut ); ResScanlineDith[ j - ElCount ] += Noise * (fptype) 0.207305; ResScanlineDith[ j ] += Noise * (fptype) 0.364842; j++; } } protected: using CImageResizerDithererDefINL< fptype > :: Len; using CImageResizerDithererDefINL< fptype > :: Vars; using CImageResizerDithererDefINL< fptype > :: LenE; using CImageResizerDithererDefINL< fptype > :: TrMul0; using CImageResizerDithererDefINL< fptype > :: PkOut0; CBuffer< fptype > ResScanlineDith0; ///< Error diffusion buffer. ///< fptype* ResScanlineDith; ///< Error diffusion buffer pointer which skips ///< the first ElCount elements. ///< }; /** * @brief Floating-point processing definition and abstraction class. * * This class defines several constants and typedefs that point to classes * that should be used by the image resizing algorithm. Such "definition * class" can be used to define alternative scanline processing algorithms * (e.g. SIMD) and image scanline packing styles used during processing. This * class also offers an abstraction layer for dithering, rounding and * clamping (saturation) operation. * * The fpclass_def class can be used to define processing using both SIMD and * non-SIMD types, but using algorithms that are operate on interleaved pixels * and non-SIMD optimized themselves. * * @tparam afptype Floating point type to use for storing intermediate data * and variables. For variables that are not used in intensive calculations * the "double" type is always used. On the latest Intel processors (like * i7-4770K) there is almost no performance difference between "double" and * "float". Image quality differences between "double" and "float" are not * apparent on 8-bit images. At the same time the "float" uses half amount of * working memory the "double" type uses. SIMD types can be used. The * functions round() and clamp() in the "avir" or other visible namespace * should be available for the specified type. SIMD types allow to perform * resizing of images with more than 4 channels, to be exact 4 * SIMD element * number (e.g. 16 for float4), without modification of the image resizing * algorithm required. * @tparam afptypeatom The atomic type the "afptype" consists of. * @tparam adith Ditherer class to use during processing. */ template< class afptype, class afptypeatom = afptype, class adith = CImageResizerDithererDefINL< afptype > > class fpclass_def { public: typedef afptype fptype; ///< Floating-point type to use during processing. ///< typedef afptypeatom fptypeatom; ///< Atomic type "fptype" consists of. ///< static const int fppack = sizeof( fptype ) / sizeof( fptypeatom ); ///< ///< The number of atomic types stored in a single "fptype" element. ///< static const int fpalign = sizeof( fptype ); ///< Suggested alignment size ///< in bytes. This is not a required alignment, because image ///< resizing algorithm cannot be made to have a strictly aligned data ///< access at all steps (e.g. interpolation cannot perform aligned ///< accesses). ///< static const int elalign = 1; ///< Length alignment of arrays of elements. ///< This applies to filters and intermediate buffers: this constant ///< forces filters and scanlines to have a length which is a multiple ///< of this value, for more efficient SIMD implementation. ///< static const int packmode = 0; ///< 0 if interleaved packing, 1 if ///< de-interleaved. ///< typedef CImageResizerFilterStepINL< fptype, fptypeatom > CFilterStep; ///< ///< Filtering step class to use during processing. ///< typedef adith CDitherer; ///< Ditherer class to use during processing. ///< }; /** * @brief Image resizer class. * * The object of this class can be used to resize 1-4 channel images to any * required size. Resizing is performed by utilizing interpolated sinc * fractional delay filters plus (if necessary) a cascade of built-in * sinc function-based 2X upsampling or 2X downsampling stages, followed by a * correction filtering. * * Object of this class can be allocated on stack. * * @tparam fpclass Floating-point processing definition class to use. See * avir::fpclass_def for more details. */ template< class fpclass = fpclass_def< float > > class CImageResizer { public: /** * Constructor initializes the resizer. * * @param aResBitDepth Required bit depth of resulting image (1-16). If * integer value output is used (e.g. uint8_t), the bit depth also affects * rounding: for example, if aResBitDepth=6 and "Tout" is uint8_t, the * result will be rounded to 6 most significant bits (2 least significant * bits truncated, with dithering applied). * @param aSrcBitDepth Source image's real bit-depth. Set to 0 to use * aResBitDepth. * @param aParams Resizing algorithm's parameters to use. Leave out for * default values. Can be useful when performing automatic optimization of * parameters. */ CImageResizer( const int aResBitDepth = 8, const int aSrcBitDepth = 0, const CImageResizerParams& aParams = CImageResizerParamsDef() ) : Params( aParams ) , ResBitDepth( aResBitDepth ) { SrcBitDepth = ( aSrcBitDepth == 0 ? ResBitDepth : aSrcBitDepth ); initFilterBank( FixedFilterBank, 1.0, false, CFltBuffer() ); FixedFilterBank.createAllFilters(); } /** * Function resizes image. * * @param SrcBuf Source image buffer. * @param SrcWidth Source image width. * @param SrcHeight Source image height. * @param SrcScanlineSize Physical size of source scanline in elements * (not bytes). If this value is below 1, SrcWidth * ElCountIO will be * used as the physical source scanline size. * @param[out] NewBuf Buffer to accept the resized image. Can be equal to * SrcBuf if the size of the resized image is smaller or equal to source * image in size. * @param NewWidth New image width. * @param NewHeight New image height. * @param ElCountIO The number of elements (channels) used to store each * source and destination pixel (1-4). * @param k Resizing step (one output pixel corresponds to "k" input * pixels). A downsizing factor if > 1.0; upsizing factor if <= 1.0. * Multiply by -1 if you would like to bypass "ox" and "oy" adjustment * which is done by default to produce a centered image. If step value * equals 0, the step value will be chosen automatically and independently * for horizontal and vertical resizing. * @param[in,out] aVars Pointer to variables structure to be passed to the * image resizing function. Can be NULL. Only variables that are * initialized in default constructor of this structure are accepted by * this function. These variables will not be changed by this function. * All other variables can be modified by this function. The access to * this object is not thread-safe, each concurrent instance of this * function should use a separate aVars object. * @tparam Tin Input buffer element's type. Can be uint8_t (0-255 value * range), uint16_t (0-65535 value range), float (0.0-1.0 value range), * double (0.0-1.0 value range). Larger integer types are treated as * uint16_t. Signed integer types are unsupported. * @tparam Tout Output buffer element's type. Can be uint8_t (0-255 value * range), uint16_t (0-65535 value range), float (0.0-1.0 value range), * double (0.0-1.0 value range). Larger integer types are treated as * uint16_t. Signed integer types are unsupported. */ template< class Tin, class Tout > void resizeImage( const Tin* const SrcBuf, const int SrcWidth, const int SrcHeight, int SrcScanlineSize, Tout* const NewBuf, const int NewWidth, const int NewHeight, const int ElCountIO, const double k, CImageResizerVars* const aVars = NULL ) const { if( SrcWidth == 0 || SrcHeight == 0 ) { memset( NewBuf, 0, (size_t) NewWidth * NewHeight * sizeof( Tout )); return; } else if( NewWidth == 0 || NewHeight == 0 ) { return; } CImageResizerVars DefVars; CImageResizerVars& Vars = ( aVars == NULL ? DefVars : *aVars ); CImageResizerThreadPool DefThreadPool; CImageResizerThreadPool& ThreadPool = ( Vars.ThreadPool == NULL ? DefThreadPool : *Vars.ThreadPool ); // Define resizing steps, also optionally modify offsets so that // resizing produces a "centered" image. double kx; double ky; double ox = Vars.ox; double oy = Vars.oy; if( k == 0.0 ) { if( NewWidth > SrcWidth ) { kx = (double) ( SrcWidth - 1 ) / ( NewWidth - 1 ); } else { kx = (double) SrcWidth / NewWidth; ox += ( kx - 1.0 ) * 0.5; } if( NewHeight > SrcHeight ) { ky = (double) ( SrcHeight - 1 ) / ( NewHeight - 1 ); } else { ky = (double) SrcHeight / NewHeight; oy += ( ky - 1.0 ) * 0.5; } } else if( k > 0.0 ) { kx = k; ky = k; if( k > 1.0 ) { const double ko = ( k - 1.0 ) * 0.5; ox += ko; oy += ko; } } else { kx = -k; ky = -k; } // Evaluate pre-multipliers used on the output stage. const bool IsInFloat = ( (Tin) 0.4 != 0 ); const bool IsOutFloat = ( (Tout) 0.4 != 0 ); double OutMul; // Output multiplier. if( Vars.UseSRGBGamma ) { if( IsInFloat ) { Vars.InGammaMult = 1.0; } else { Vars.InGammaMult = 1.0 / ( sizeof( Tin ) == 1 ? 255.0 : 65535.0 ); } if( IsOutFloat ) { Vars.OutGammaMult = 1.0; } else { Vars.OutGammaMult = ( sizeof( Tout ) == 1 ? 255.0 : 65535.0 ); } OutMul = 1.0; } else { if( IsOutFloat ) { OutMul = 1.0; } else { OutMul = ( sizeof( Tout ) == 1 ? 255.0 : 65535.0 ); } if( !IsInFloat ) { OutMul /= ( sizeof( Tin ) == 1 ? 255.0 : 65535.0 ); } } // Fill widely-used variables. const int ElCount = ( ElCountIO + fpclass :: fppack - 1 ) / fpclass :: fppack; const int NewWidthE = NewWidth * ElCount; if( SrcScanlineSize < 1 ) { SrcScanlineSize = SrcWidth * ElCountIO; } Vars.ElCount = ElCount; Vars.ElCountIO = ElCountIO; Vars.fppack = fpclass :: fppack; Vars.fpalign = fpclass :: fpalign; Vars.elalign = fpclass :: elalign; Vars.packmode = fpclass :: packmode; // Horizontal scanline filtering and resizing. CDSPFracFilterBankLin< fptype > FltBank; CFilterSteps FltSteps; typename CFilterStep :: CRPosBufArray RPosBufArray; CBuffer< uint8_t > UsedFracMap; // Perform the filtering steps modeling at various modes, find the // most efficient mode for both horizontal and vertical resizing. int UseBuildMode = 1; const int BuildModeCount = ( FixedFilterBank.getOrder() == 0 ? 4 : 2 ); int m; if( Vars.BuildMode >= 0 ) { UseBuildMode = Vars.BuildMode; } else { int BestScore = 0x7FFFFFFF; for( m = 0; m < BuildModeCount; m++ ) { CDSPFracFilterBankLin< fptype > TmpBank; CFilterSteps TmpSteps; Vars.k = kx; Vars.o = ox; buildFilterSteps( TmpSteps, Vars, TmpBank, OutMul, m, true ); updateFilterStepBuffers( TmpSteps, Vars, RPosBufArray, SrcWidth, NewWidth ); fillUsedFracMap( TmpSteps[ Vars.ResizeStep ], UsedFracMap ); const int c = calcComplexity( TmpSteps, Vars, UsedFracMap, SrcHeight ); if( c < BestScore ) { UseBuildMode = m; BestScore = c; } } } // Perform the actual filtering steps building. Vars.k = kx; Vars.o = ox; buildFilterSteps( FltSteps, Vars, FltBank, OutMul, UseBuildMode, false ); updateFilterStepBuffers( FltSteps, Vars, RPosBufArray, SrcWidth, NewWidth ); updateBufLenAndRPosPtrs( FltSteps, Vars, NewWidth ); const int ThreadCount = ThreadPool.getSuggestedWorkloadCount(); // Includes the current thread. CStructArray< CThreadData< Tin, Tout > > td; td.setItemCount( ThreadCount ); int i; for( i = 0; i < ThreadCount; i++ ) { if( i > 0 ) { ThreadPool.addWorkload( &td[ i ]); } td[ i ].init( i, ThreadCount, FltSteps, Vars ); td[ i ].initScanlineQueue( td[ i ].sopResizeH, SrcHeight, SrcWidth ); } CBuffer< fptype, size_t > FltBuf( (size_t) NewWidthE * SrcHeight, fpclass :: fpalign ); // Temporary buffer that receives // horizontally-filtered and resized image. for( i = 0; i < SrcHeight; i++ ) { td[ i % ThreadCount ].addScanlineToQueue( (void*) &SrcBuf[ (size_t) i * SrcScanlineSize ], &FltBuf[ (size_t) i * NewWidthE ]); } ThreadPool.startAllWorkloads(); td[ 0 ].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); // Vertical scanline filtering and resizing, reuse previously defined // filtering steps if possible. const int PrevUseBuildMode = UseBuildMode; if( Vars.BuildMode >= 0 ) { UseBuildMode = Vars.BuildMode; } else { CImageResizerVars TmpVars( Vars ); int BestScore = 0x7FFFFFFF; for( m = 0; m < BuildModeCount; m++ ) { CDSPFracFilterBankLin< fptype > TmpBank; TmpBank.copyInitParams( FltBank ); CFilterSteps TmpSteps; TmpVars.k = ky; TmpVars.o = oy; buildFilterSteps( TmpSteps, TmpVars, TmpBank, 1.0, m, true ); updateFilterStepBuffers( TmpSteps, TmpVars, RPosBufArray, SrcHeight, NewHeight ); fillUsedFracMap( TmpSteps[ TmpVars.ResizeStep ], UsedFracMap ); const int c = calcComplexity( TmpSteps, TmpVars, UsedFracMap, NewWidth ); if( c < BestScore ) { UseBuildMode = m; BestScore = c; } } } Vars.k = ky; Vars.o = oy; if( UseBuildMode == PrevUseBuildMode && ky == kx ) { if( OutMul != 1.0 ) { modifyCorrFilterDCGain( FltSteps, 1.0 / OutMul ); } } else { buildFilterSteps( FltSteps, Vars, FltBank, 1.0, UseBuildMode, false ); } updateFilterStepBuffers( FltSteps, Vars, RPosBufArray, SrcHeight, NewHeight ); updateBufLenAndRPosPtrs( FltSteps, Vars, NewWidth ); if( IsOutFloat && sizeof( FltBuf[ 0 ]) == sizeof( Tout ) && fpclass :: packmode == 0 ) { // In-place output. for( i = 0; i < ThreadCount; i++ ) { td[ i ].initScanlineQueue( td[ i ].sopResizeV, NewWidth, SrcHeight, NewWidthE, NewWidthE ); } for( i = 0; i < NewWidth; i++ ) { td[ i % ThreadCount ].addScanlineToQueue( &FltBuf[ (size_t) i * ElCount ], (fptype*) &NewBuf[ (size_t) i * ElCount ]); } ThreadPool.startAllWorkloads(); td[ 0 ].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); ThreadPool.removeAllWorkloads(); return; } CBuffer< fptype, size_t > ResBuf( (size_t) NewWidthE * NewHeight, fpclass :: fpalign ); for( i = 0; i < ThreadCount; i++ ) { td[ i ].initScanlineQueue( td[ i ].sopResizeV, NewWidth, SrcHeight, NewWidthE, NewWidthE ); } const int im = ( fpclass :: packmode == 0 ? ElCount : 1 ); for( i = 0; i < NewWidth; i++ ) { td[ i % ThreadCount ].addScanlineToQueue( &FltBuf[ (size_t) i * im ], &ResBuf[ (size_t) i * im ]); } ThreadPool.startAllWorkloads(); td[ 0 ].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); if( IsOutFloat ) { // Perform output, but skip dithering. for( i = 0; i < ThreadCount; i++ ) { td[ i ].initScanlineQueue( td[ i ].sopUnpackH, NewHeight, NewWidth ); } for( i = 0; i < NewHeight; i++ ) { td[ i % ThreadCount ].addScanlineToQueue( &ResBuf[ (size_t) i * NewWidthE ], &NewBuf[ (size_t) i * NewWidth * ElCountIO ]); } ThreadPool.startAllWorkloads(); td[ 0 ].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); ThreadPool.removeAllWorkloads(); return; } // Perform output with dithering (for integer output only). int TruncBits; // The number of lower bits to truncate and dither. int OutRange; // Output range. if( sizeof( Tout ) == 1 ) { TruncBits = 8 - ResBitDepth; OutRange = 255; } else { TruncBits = 16 - ResBitDepth; OutRange = 65535; } const double PkOut = OutRange; const double TrMul = ( TruncBits > 0 ? PkOut / ( OutRange >> TruncBits ) : 1.0 ); if( CDitherer :: isRecursive() ) { td[ 0 ].getDitherer().init( NewWidth, Vars, TrMul, PkOut ); if( Vars.UseSRGBGamma ) { for( i = 0; i < NewHeight; i++ ) { fptype* const ResScanline = &ResBuf[ (size_t) i * NewWidthE ]; CFilterStep :: applySRGBGamma( ResScanline, NewWidth, Vars ); td[ 0 ].getDitherer().dither( ResScanline ); CFilterStep :: unpackScanline( ResScanline, &NewBuf[ (size_t) i * NewWidth * ElCountIO ], NewWidth, Vars ); } } else { for( i = 0; i < NewHeight; i++ ) { fptype* const ResScanline = &ResBuf[ (size_t) i * NewWidthE ]; td[ 0 ].getDitherer().dither( ResScanline ); CFilterStep :: unpackScanline( ResScanline, &NewBuf[ (size_t) i * NewWidth * ElCountIO ], NewWidth, Vars ); } } } else { for( i = 0; i < ThreadCount; i++ ) { td[ i ].initScanlineQueue( td[ i ].sopDitherAndUnpackH, NewHeight, NewWidth ); td[ i ].getDitherer().init( NewWidth, Vars, TrMul, PkOut ); } for( i = 0; i < NewHeight; i++ ) { td[ i % ThreadCount ].addScanlineToQueue( &ResBuf[ (size_t) i * NewWidthE ], &NewBuf[ (size_t) i * NewWidth * ElCountIO ]); } ThreadPool.startAllWorkloads(); td[ 0 ].processScanlineQueue(); ThreadPool.waitAllWorkloadsToFinish(); } ThreadPool.removeAllWorkloads(); } private: typedef typename fpclass :: fptype fptype; ///< Floating-point type to use ///< during processing. ///< typedef typename fpclass :: CFilterStep CFilterStep; ///< Filtering step ///< class to use during processing. ///< typedef typename fpclass :: CDitherer CDitherer; ///< Ditherer class to ///< use during processing. ///< CImageResizerParams Params; ///< Algorithm's parameters currently in use. ///< int SrcBitDepth; ///< Bit resolution of the source image. ///< int ResBitDepth; ///< Bit resolution of the resulting image. ///< CDSPFracFilterBankLin< fptype > FixedFilterBank; ///< Fractional delay ///< filter bank with fixed characteristics, mainly for upsizing ///< cases. ///< /** * @brief Filtering steps array. * * The object of this class stores filtering steps together. */ typedef CStructArray< CFilterStep > CFilterSteps; /** * Function initializes the filter bank in the specified resizing step * according to the source and resulting image bit depths. * * @param FltBank Filter bank to initialize. * @param CutoffMult Cutoff multiplier, 0 to 1. 1 corresponds to 0.5pi * cutoff point. * @param ForceHiOrder "True" if a high-order interpolation should be * forced which requires considerably less resources for initialization. * @param ExtFilter External filter to apply to interpolation filter. */ void initFilterBank( CDSPFracFilterBankLin< fptype >& FltBank, const double CutoffMult, const bool ForceHiOrder, const CFltBuffer& ExtFilter ) const { const int IntBitDepth = ( ResBitDepth > SrcBitDepth ? ResBitDepth : SrcBitDepth ); const double SNR = -6.02 * ( IntBitDepth + 3 ); int UseOrder; int FracCount; // The number of fractional delay filters sampled by // the filter bank. This variable affects the signal-to-noise // ratio at interpolation stage. Theoretically, at UseOrder==1, // 8-bit image resizing requires 66.2 dB SNR or 11. 16-bit // resizing requires 114.4 dB SNR or 150. At UseOrder=0 the // required number of filters is exponentially higher. if( ForceHiOrder || IntBitDepth > 8 ) { UseOrder = 1; // -146 dB max FracCount = (int) ceil( 0.23134052 * exp( -0.058062929 * SNR )); } else { UseOrder = 0; // -72 dB max FracCount = (int) ceil( 0.33287686 * exp( -0.11334583 * SNR )); } if( FracCount < 2 ) { FracCount = 2; } FltBank.init( FracCount, UseOrder, Params.IntFltLen / CutoffMult, Params.IntFltCutoff * CutoffMult, Params.IntFltAlpha, ExtFilter, fpclass :: fpalign, fpclass :: elalign ); } /** * Function allocates filter buffer taking "fpclass" alignments into * account. The allocated buffer may be larger than the requested size: in * this case the additional elements will be zeroed by this function. * * @param Flt Filter buffer. * @param ReqCapacity The required filter buffer's capacity. * @param IsModel "True" if filtering steps modeling is performed without * actual filter allocation. * @param FltExt If non-NULL this variable will receive the number of * elements the filter was extended by. */ static void allocFilter( CBuffer< fptype >& Flt, const int ReqCapacity, const bool IsModel = false, int* const FltExt = NULL ) { int UseCapacity = ( ReqCapacity + fpclass :: elalign - 1 ) & ~( fpclass :: elalign - 1 ); int Ext = UseCapacity - ReqCapacity; if( FltExt != NULL ) { *FltExt = Ext; } if( IsModel ) { Flt.forceCapacity( UseCapacity ); return; } Flt.alloc( UseCapacity, fpclass :: fpalign ); while( Ext > 0 ) { Ext--; Flt[ ReqCapacity + Ext ] = 0.0; } } /** * Function assigns filter parameters to the specified filtering step * object. * * @param fs Filtering step to assign parameter to. This step cannot be * the last step if ResampleFactor greater than 1 was specified. * @param IsUpsample "True" if upsampling step. Should be set to "false" * if FltCutoff is negative. * @param ResampleFactor Resampling factor of this filter (>=1). * @param FltCutoff Filter cutoff point. This value will be divided by the * ResampleFactor if IsUpsample equals "true". If zero value was * specified, the "half-band" predefined filter will be created. In this * case the ResampleFactor will modify the filter cutoff point. * @param DCGain DC gain to apply to the filter. Assigned to filtering * step's DCGain variable. * @param UseFltOrig "True" if the originally-designed filter should be * left in filtering step's FltOrig buffer. Otherwise it will be freed. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ void assignFilterParams( CFilterStep& fs, const bool IsUpsample, const int ResampleFactor, const double FltCutoff, const double DCGain, const bool UseFltOrig, const bool IsModel ) const { double FltAlpha; double Len2; double Freq; if( FltCutoff == 0.0 ) { const double m = 2.0 / ResampleFactor; FltAlpha = Params.HBFltAlpha; Len2 = 0.5 * Params.HBFltLen / m; Freq = AVIR_PI * Params.HBFltCutoff * m; } else { FltAlpha = Params.LPFltAlpha; Len2 = 0.25 * Params.LPFltBaseLen / FltCutoff; Freq = AVIR_PI * Params.LPFltCutoffMult * FltCutoff; } if( IsUpsample ) { Len2 *= ResampleFactor; Freq /= ResampleFactor; fs.DCGain = DCGain * ResampleFactor; } else { fs.DCGain = DCGain; } fs.FltOrig.Len2 = Len2; fs.FltOrig.Freq = Freq; fs.FltOrig.Alpha = FltAlpha; fs.FltOrig.DCGain = fs.DCGain; CDSPPeakedCosineLPF w( Len2, Freq, FltAlpha ); fs.IsUpsample = IsUpsample; fs.ResampleFactor = ResampleFactor; fs.FltLatency = w.fl2; int FltExt; // Filter's extension due to fpclass :: elalign. if( IsModel ) { allocFilter( fs.Flt, w.FilterLen, true, &FltExt ); if( UseFltOrig ) { // Allocate a real buffer even in modeling mode since this // filter may be copied by the filter bank. fs.FltOrig.alloc( w.FilterLen ); memset( &fs.FltOrig[ 0 ], 0, w.FilterLen * sizeof( fs.FltOrig[ 0 ])); } } else { fs.FltOrig.alloc( w.FilterLen ); w.generateLPF( &fs.FltOrig[ 0 ], 1.0 ); optimizeFIRFilter( fs.FltOrig, fs.FltLatency ); normalizeFIRFilter( &fs.FltOrig[ 0 ], fs.FltOrig.getCapacity(), fs.DCGain ); allocFilter( fs.Flt, fs.FltOrig.getCapacity(), false, &FltExt ); copyArray( &fs.FltOrig[ 0 ], &fs.Flt[ 0 ], fs.FltOrig.getCapacity() ); if( !UseFltOrig ) { fs.FltOrig.free(); } } if( IsUpsample ) { int l = fs.Flt.getCapacity() - fs.FltLatency - ResampleFactor - FltExt; allocFilter( fs.PrefixDC, l, IsModel ); allocFilter( fs.SuffixDC, fs.FltLatency, IsModel ); if( IsModel ) { return; } // Create prefix and suffix "tails" used during upsampling. const fptype* ip = &fs.Flt[ fs.FltLatency + ResampleFactor ]; copyArray( ip, &fs.PrefixDC[ 0 ], l ); while( true ) { ip += ResampleFactor; l -= ResampleFactor; if( l <= 0 ) { break; } addArray( ip, &fs.PrefixDC[ 0 ], l ); } l = fs.FltLatency; fptype* op = &fs.SuffixDC[ 0 ]; copyArray( &fs.Flt[ 0 ], op, l ); while( true ) { op += ResampleFactor; l -= ResampleFactor; if( l <= 0 ) { break; } addArray( &fs.Flt[ 0 ], op, l ); } } else if( !UseFltOrig ) { fs.EdgePixelCount = fs.EdgePixelCountDef; } } /** * Function adds a correction filter that tries to achieve a linear * frequency response at all frequencies. The actual resulting response * may feature a slight damping of the highest frequencies since a * suitably short correction filter cannot fix steep high-frequency * damping. * * This function assumes that the resizing step is currently the last * step, even if it was not inserted yet: this allows placement of the * correction filter both before and after the resizing step. * * @param Steps Filtering steps. * @param bw Resulting bandwidth relative to the original bandwidth (which * is 1.0), usually 1/k. Should be <= 1.0. * @param IsPreCorrection "True" if the filtering step was already created * and it is first in the Steps array. "True" also adds edge pixels to * reduce edge artifacts. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ void addCorrectionFilter( CFilterSteps& Steps, const double bw, const bool IsPreCorrection, const bool IsModel ) const { CFilterStep& fs = ( IsPreCorrection ? Steps[ 0 ] : Steps.add() ); fs.IsUpsample = false; fs.ResampleFactor = 1; fs.DCGain = 1.0; fs.EdgePixelCount = ( IsPreCorrection ? fs.EdgePixelCountDef : 0 ); if( IsModel ) { allocFilter( fs.Flt, CDSPFIREQ :: calcFilterLength( Params.CorrFltLen, fs.FltLatency ), true ); return; } const int BinCount = 65; // Frequency response bins to control. const int BinCount1 = BinCount - 1; double curbw = 1.0; // Bandwidth of the filter at the current step. int i; int j; double re; double im; CBuffer< double > Bins( BinCount ); // Adjustment introduced by all // steps at all frequencies of interest. for( j = 0; j < BinCount; j++ ) { Bins[ j ] = 1.0; } const int si = ( IsPreCorrection ? 1 : 0 ); for( i = si; i < Steps.getItemCount() - ( si ^ 1 ); i++ ) { const CFilterStep& fs = Steps[ i ]; if( fs.IsUpsample ) { curbw *= fs.ResampleFactor; if( fs.FltOrig.getCapacity() > 0 ) { continue; } } const double dcg = 1.0 / fs.DCGain; // DC gain correction. const fptype* Flt; int FltLen; if( fs.ResampleFactor == 0 ) { Flt = fs.FltBank -> getFilter( 0 ); FltLen = fs.FltBank -> getFilterLen(); } else { Flt = &fs.Flt[ 0 ]; FltLen = fs.Flt.getCapacity(); } // Calculate frequency response adjustment introduced by the // filter at this step, within the bounds of bandwidth of // interest. for( j = 0; j < BinCount; j++ ) { const double th = AVIR_PI * bw / curbw * j / BinCount1; calcFIRFilterResponse( Flt, FltLen, th, re, im ); Bins[ j ] /= sqrt( re * re + im * im ) * dcg; } if( !fs.IsUpsample && fs.ResampleFactor > 1 ) { curbw /= fs.ResampleFactor; } } // Calculate filter. CDSPFIREQ EQ; EQ.init( bw * 2.0, Params.CorrFltLen, BinCount, 0.0, bw, false, Params.CorrFltAlpha ); fs.FltLatency = EQ.getFilterLatency(); CBuffer< double > Filter( EQ.getFilterLength() ); EQ.buildFilter( Bins, &Filter[ 0 ]); normalizeFIRFilter( &Filter[ 0 ], Filter.getCapacity(), 1.0 ); optimizeFIRFilter( Filter, fs.FltLatency ); normalizeFIRFilter( &Filter[ 0 ], Filter.getCapacity(), 1.0 ); allocFilter( fs.Flt, Filter.getCapacity() ); copyArray( &Filter[ 0 ], &fs.Flt[ 0 ], Filter.getCapacity() ); // Print a theoretically achieved final frequency response at various // feature sizes (from DC to 1 pixel). Values above 255 means features // become brighter, values below 255 means features become dimmer. /* const double sbw = ( bw > 1.0 ? 1.0 / bw : 1.0 ); for( j = 0; j < BinCount; j++ ) { const double th = AVIR_PI * sbw * j / BinCount1; calcFIRFilterResponse( &fs.Flt[ 0 ], fs.Flt.getCapacity(), th, re, im ); printf( "%f\n", sqrt( re * re + im * im ) / Bins[ j ] * 255 ); } printf( "***\n" );*/ } /** * Function adds a sharpening filter if image is being upsized. Such * sharpening allows to spot interpolation filter's stop-band attenuation: * if attenuation is too weak, a "dark grid" and other artifacts may * become visible. * * It is assumed that 40 decibel stop-band attenuation should be * considered a required minimum: this allows application of (deliberately * strong) 64X sharpening without spotting any artifacts. * * @param Steps Filtering steps. * @param bw Resulting bandwidth relative to the original bandwidth (which * is 1.0), usually 1/k. * @param IsModel "True" if filtering steps modeling is performed without * actual filter building. */ static void addSharpenTest( CFilterSteps& Steps, const double bw, const bool IsModel ) { if( bw <= 1.0 ) { return; } const double FltLen = 10.0 * bw; CFilterStep& fs = Steps.add(); fs.IsUpsample = false; fs.ResampleFactor = 1; fs.DCGain = 1.0; fs.EdgePixelCount = 0; if( IsModel ) { allocFilter( fs.Flt, CDSPFIREQ :: calcFilterLength( FltLen, fs.FltLatency ), true ); return; } const int BinCount = 200; CBuffer< double > Bins( BinCount ); int Thresh = (int) round( BinCount / bw * 1.75 ); if( Thresh > BinCount ) { Thresh = BinCount; } int j; for( j = 0; j < Thresh; j++ ) { Bins[ j ] = 1.0; } for( j = Thresh; j < BinCount; j++ ) { Bins[ j ] = 256.0; } CDSPFIREQ EQ; EQ.init( bw * 2.0, FltLen, BinCount, 0.0, bw, false, 1.7 ); fs.FltLatency = EQ.getFilterLatency(); CBuffer< double > Filter( EQ.getFilterLength() ); EQ.buildFilter( Bins, &Filter[ 0 ]); normalizeFIRFilter( &Filter[ 0 ], Filter.getCapacity(), 1.0 ); optimizeFIRFilter( Filter, fs.FltLatency ); normalizeFIRFilter( &Filter[ 0 ], Filter.getCapacity(), 1.0 ); allocFilter( fs.Flt, Filter.getCapacity() ); copyArray( &Filter[ 0 ], &fs.Flt[ 0 ], Filter.getCapacity() ); /* for( j = 0; j < BinCount; j++ ) { const double th = AVIR_PI * j / ( BinCount - 1 ); double re; double im; calcFIRFilterResponse( &fs.Flt[ 0 ], fs.Flt.getCapacity(), th, re, im ); printf( "%f\n", sqrt( re * re + im * im )); } printf( "***\n" );*/ } /** * Function builds sequence of filtering steps depending on the specified * resizing coefficient. The last steps included are always the resizing * step then (possibly) the correction step. * * @param Steps Array that receives filtering steps. * @param[out] Vars Variables object. * @param FltBank Filter bank to initialize and use. * @param DCGain The overall DC gain to apply. This DC gain is applied to * the first filtering step only (upsampling or filtering step). * @param ModeFlags Build mode flags to use. This is a bitmap of switches * that enable or disable certain algorithm features. * @param IsModel "True" if filtering steps modeling is performed without * the actual filter allocation and building. */ void buildFilterSteps( CFilterSteps& Steps, CImageResizerVars& Vars, CDSPFracFilterBankLin< fptype >& FltBank, const double DCGain, const int ModeFlags, const bool IsModel ) const { Steps.clear(); const bool DoFltAndIntCombo = (( ModeFlags & 1 ) != 0 ); // Do filter // and interpolator combining. const bool ForceHiOrderInt = (( ModeFlags & 2 ) != 0 ); // Force use // of a higher-order interpolation. const bool UseHalfband = (( ModeFlags & 4 ) != 0 ); // Use half-band // filter. const double bw = 1.0 / Vars.k; // Resulting bandwidth. const int UpsampleFactor = ( (int) floor( Vars.k ) < 2 ? 2 : 1 ); double IntCutoffMult; // Interpolation filter cutoff multiplier. CFilterStep* ReuseStep; // If not NULL, resizing step should use // this step object instead of creating a new one. CFilterStep* ExtFltStep; // Use FltOrig of this step as the external // filter to applied to the interpolator. bool IsPreCorrection; // "True" if the correction filter is applied // first. double FltCutoff; // Cutoff frequency of the first filtering step. double corrbw; ///< Bandwidth at the correction step. if( Vars.k <= 1.0 ) { IsPreCorrection = true; FltCutoff = 1.0; corrbw = 1.0; Steps.add(); } else { IsPreCorrection = false; FltCutoff = bw; corrbw = bw; } // Add 1 upsampling or several downsampling filters. if( UpsampleFactor > 1 ) { CFilterStep& fs = Steps.add(); assignFilterParams( fs, true, UpsampleFactor, FltCutoff, DCGain, DoFltAndIntCombo, IsModel ); IntCutoffMult = FltCutoff * 2.0 / UpsampleFactor; ReuseStep = NULL; ExtFltStep = ( DoFltAndIntCombo ? &fs : NULL ); } else { int DownsampleFactor; while( true ) { DownsampleFactor = (int) floor( 0.5 / FltCutoff ); bool DoHBFltAdd; if( DownsampleFactor > 16 ) { // Add half-band filter unconditionally in order to keep // filter lengths lower for more precise frequency // response and less edge artifacts. DoHBFltAdd = true; DownsampleFactor = 16; } else { DoHBFltAdd = ( UseHalfband && DownsampleFactor > 1 ); } if( DoHBFltAdd ) { assignFilterParams( Steps.add(), false, DownsampleFactor, 0.0, 1.0, false, IsModel ); FltCutoff *= DownsampleFactor; } else { if( DownsampleFactor < 1 ) { DownsampleFactor = 1; } break; } } CFilterStep& fs = Steps.add(); assignFilterParams( fs, false, DownsampleFactor, FltCutoff, DCGain, DoFltAndIntCombo, IsModel ); IntCutoffMult = FltCutoff / 0.5; if( DoFltAndIntCombo ) { ReuseStep = &fs; ExtFltStep = &fs; } else { IntCutoffMult *= DownsampleFactor; ReuseStep = NULL; ExtFltStep = NULL; } } // Insert resizing and correction steps. CFilterStep& fs = ( ReuseStep == NULL ? Steps.add() : *ReuseStep ); Vars.ResizeStep = Steps.getItemCount() - 1; fs.IsUpsample = false; fs.ResampleFactor = 0; fs.DCGain = ( ExtFltStep == NULL ? 1.0 : ExtFltStep -> DCGain ); initFilterBank( FltBank, IntCutoffMult, ForceHiOrderInt, ( ExtFltStep == NULL ? fs.FltOrig : ExtFltStep -> FltOrig )); if( FltBank == FixedFilterBank ) { fs.FltBank = (CDSPFracFilterBankLin< fptype >*) &FixedFilterBank; } else { fs.FltBank = &FltBank; } addCorrectionFilter( Steps, corrbw, IsPreCorrection, IsModel ); //addSharpenTest( Steps, bw, IsModel ); } /** * Function extends *this upsampling step so that it produces more * upsampled pixels that cover the prefix and suffix needs of the next * step. After the call to this function the InPrefix and InSuffix * variables of the next step will be set to zero. * * @param fs Upsampling filtering step. * @param NextStep The next step structure. */ static void extendUpsample( CFilterStep& fs, CFilterStep& NextStep ) { fs.InPrefix = ( NextStep.InPrefix + fs.ResampleFactor - 1 ) / fs.ResampleFactor; fs.OutPrefix += fs.InPrefix * fs.ResampleFactor; NextStep.InPrefix = 0; fs.InSuffix = ( NextStep.InSuffix + fs.ResampleFactor - 1 ) / fs.ResampleFactor; fs.OutSuffix += fs.InSuffix * fs.ResampleFactor; NextStep.InSuffix = 0; } /** * Function fills resizing step's RPosBuf array, excluding the actual * "ftp" pointers and "SrcOffs" offsets. * * This array should be cleared if the resizing step or offset were * changed. Otherwise this function only fills the elements required to * cover resizing step's OutLen. * * This function is called by the updateFilterStepBuffers() function. * * @param fs Resizing step. * @param Vars Variables object. */ static void fillRPosBuf( CFilterStep& fs, const CImageResizerVars& Vars ) { const int PrevLen = fs.RPosBuf -> getCapacity(); if( fs.OutLen > PrevLen ) { fs.RPosBuf -> increaseCapacity( fs.OutLen ); } typename CFilterStep :: CResizePos* rpos = &(*fs.RPosBuf)[ PrevLen ]; const int FracCount = fs.FltBank -> getFracCount(); const double o = Vars.o; const double k = Vars.k; int i; for( i = PrevLen; i < fs.OutLen; i++ ) { const double SrcPos = o + k * i; const int SrcPosInt = (int) floor( SrcPos ); const double x = ( SrcPos - SrcPosInt ) * FracCount; const int fti = (int) x; rpos -> x = (typename fpclass :: fptypeatom) ( x - fti ); rpos -> fti = fti; rpos -> SrcPosInt = SrcPosInt; rpos++; } } /** * Function updates filtering step buffer lengths depending on the * specified source and new scanline lengths. This function should be * called after the buildFilterSteps() function. * * @param Steps Array that receives filtering steps. * @param[out] Vars Variables object, will receive buffer size and length. * This function expects "k" and "o" variable values that will be * adjusted by this function. * @param RPosBufArray Resizing position buffers array, used to obtain * buffer to initialize and use (will be reused if it is already fully or * partially filled). * @param SrcLen Source scanline's length in pixels. * @param NewLen New scanline's length in pixels. */ static void updateFilterStepBuffers( CFilterSteps& Steps, CImageResizerVars& Vars, typename CFilterStep :: CRPosBufArray& RPosBufArray, int SrcLen, const int NewLen ) { int upstep = -1; int InBuf = 0; int i; for( i = 0; i < Steps.getItemCount(); i++ ) { CFilterStep& fs = Steps[ i ]; fs.Vars = &Vars; fs.InLen = SrcLen; fs.InBuf = InBuf; fs.OutBuf = ( InBuf + 1 ) & 1; if( fs.IsUpsample ) { upstep = i; Vars.k *= fs.ResampleFactor; Vars.o *= fs.ResampleFactor; fs.InPrefix = 0; fs.InSuffix = 0; fs.OutLen = fs.InLen * fs.ResampleFactor; fs.OutPrefix = fs.FltLatency; fs.OutSuffix = fs.Flt.getCapacity() - fs.FltLatency - fs.ResampleFactor; int l0 = fs.OutPrefix + fs.OutLen + fs.OutSuffix; int l = fs.InLen * fs.ResampleFactor + fs.SuffixDC.getCapacity(); if( l > l0 ) { fs.OutSuffix += l - l0; } l0 = fs.OutLen + fs.OutSuffix; if( fs.PrefixDC.getCapacity() > l0 ) { fs.OutSuffix += fs.PrefixDC.getCapacity() - l0; } } else if( fs.ResampleFactor == 0 ) { const int FilterLenD2 = fs.FltBank -> getFilterLen() / 2; const int FilterLenD21 = FilterLenD2 - 1; const int ResizeLPix = (int) floor( Vars.o ) - FilterLenD21; fs.InPrefix = ( ResizeLPix < 0 ? -ResizeLPix : 0 ); const int ResizeRPix = (int) floor( Vars.o + ( NewLen - 1 ) * Vars.k ) + FilterLenD2 + 1; fs.InSuffix = ( ResizeRPix > fs.InLen ? ResizeRPix - fs.InLen : 0 ); fs.OutLen = NewLen; fs.RPosBuf = &RPosBufArray.getRPosBuf( Vars.k, Vars.o, fs.FltBank -> getFracCount() ); fillRPosBuf( fs, Vars ); } else { Vars.k /= fs.ResampleFactor; Vars.o /= fs.ResampleFactor; Vars.o += fs.EdgePixelCount; fs.InPrefix = fs.FltLatency; fs.InSuffix = fs.Flt.getCapacity() - fs.FltLatency - 1; // Additionally extend OutLen to produce more precise edge // pixels. fs.OutLen = ( fs.InLen + fs.ResampleFactor - 1 ) / fs.ResampleFactor + fs.EdgePixelCount; fs.InSuffix += ( fs.OutLen - 1 ) * fs.ResampleFactor + 1 - fs.InLen; fs.InPrefix += fs.EdgePixelCount * fs.ResampleFactor; fs.OutLen += fs.EdgePixelCount; } InBuf = fs.OutBuf; SrcLen = fs.OutLen; } Steps[ Steps.getItemCount() - 1 ].OutBuf = 2; if( upstep != -1 ) { extendUpsample( Steps[ upstep ], Steps[ upstep + 1 ]); } } /** * Function calculates an optimal intermediate buffer length that will * cover all needs of the specified filtering steps. This function should * be called after the updateFilterStepBuffers() function. * * Function also updates resizing step's RPosBuf pointers to the filter * bank and SrcOffs values. * * @param Steps Filtering steps. * @param[out] Vars Variables object, will receive buffer size and length. * @param ResElIncr Resulting (final) element increment, used to produce * de-interleaved result. For horizontal processing this value is equal * to last step's OutLen, for vertical processing this value is equal to * resulting image's width. */ static void updateBufLenAndRPosPtrs( CFilterSteps& Steps, CImageResizerVars& Vars, const int ResElIncr ) { int MaxPrefix[ 2 ] = { 0, 0 }; int MaxLen[ 2 ] = { 0, 0 }; int i; for( i = 0; i < Steps.getItemCount(); i++ ) { CFilterStep& fs = Steps[ i ]; const int ib = fs.InBuf; if( fs.InPrefix > MaxPrefix[ ib ]) { MaxPrefix[ ib ] = fs.InPrefix; } int l = fs.InLen + fs.InSuffix; if( l > MaxLen[ ib ]) { MaxLen[ ib ] = l; } fs.InElIncr = fs.InPrefix + l; if( fs.OutBuf == 2 ) { break; } const int ob = fs.OutBuf; if( fs.IsUpsample ) { if( fs.OutPrefix > MaxPrefix[ ob ]) { MaxPrefix[ ob ] = fs.OutPrefix; } l = fs.OutLen + fs.OutSuffix; if( l > MaxLen[ ob ]) { MaxLen[ ob ] = l; } } else { if( fs.OutLen > MaxLen[ ob ]) { MaxLen[ ob ] = fs.OutLen; } } } // Update OutElIncr values of all steps. for( i = 0; i < Steps.getItemCount(); i++ ) { CFilterStep& fs = Steps[ i ]; if( fs.OutBuf == 2 ) { fs.OutElIncr = ResElIncr; break; } CFilterStep& fs2 = Steps[ i + 1 ]; if( fs.IsUpsample ) { fs.OutElIncr = fs.OutPrefix + fs.OutLen + fs.OutSuffix; if( fs.OutElIncr > fs2.InElIncr ) { fs2.InElIncr = fs.OutElIncr; } else { fs.OutElIncr = fs2.InElIncr; } } else { fs.OutElIncr = fs2.InElIncr; } } // Update temporary buffer's length. for( i = 0; i < 2; i++ ) { Vars.BufLen[ i ] = MaxPrefix[ i ] + MaxLen[ i ]; Vars.BufOffs[ i ] = MaxPrefix[ i ]; if( Vars.packmode == 0 ) { Vars.BufOffs[ i ] *= Vars.ElCount; } Vars.BufLen[ i ] *= Vars.ElCount; } // Update RPosBuf pointers and SrcOffs. CFilterStep& fs = Steps[ Vars.ResizeStep ]; typename CFilterStep :: CResizePos* rpos = &(*fs.RPosBuf)[ 0 ]; const int em = ( fpclass :: packmode == 0 ? Vars.ElCount : 1 ); const int FilterLenD21 = fs.FltBank -> getFilterLen() / 2 - 1; for( i = 0; i < fs.OutLen; i++ ) { rpos -> ftp = fs.FltBank -> getFilter( rpos -> fti ); rpos -> SrcOffs = ( rpos -> SrcPosInt - FilterLenD21 ) * em; rpos++; } } /** * Function modifies the overall (DC) gain of the correction filter in the * pre-built filtering steps array. * * @param Steps Filtering steps. * @param m Multiplier to apply to the correction filter. */ void modifyCorrFilterDCGain( CFilterSteps& Steps, const double m ) const { CBuffer< fptype >* Flt; const int z = Steps.getItemCount() - 1; if( !Steps[ z ].IsUpsample && Steps[ z ].ResampleFactor == 1 ) { Flt = &Steps[ z ].Flt; } else { Flt = &Steps[ 0 ].Flt; } int i; for( i = 0; i < Flt -> getCapacity(); i++ ) { (*Flt)[ i ] = (fptype) ( (double) (*Flt)[ i ] * m ); } } /** * Function builds a map of used fractional delay filters based on the * resizing positions buffer. * * @param fs Resizing step. * @param[out] UsedFracMap Map of used fractional delay filters. */ static void fillUsedFracMap( const CFilterStep& fs, CBuffer< uint8_t >& UsedFracMap ) { const int FracCount = fs.FltBank -> getFracCount(); UsedFracMap.increaseCapacity( FracCount, false ); memset( &UsedFracMap[ 0 ], 0, FracCount * sizeof( UsedFracMap[ 0 ])); typename CFilterStep :: CResizePos* rpos = &(*fs.RPosBuf)[ 0 ]; int i; for( i = 0; i < fs.OutLen; i++ ) { UsedFracMap[ rpos -> fti ] |= 1; rpos++; } } /** * Function calculates the overall filtering steps complexity per * scanline. Each complexity unit corresponds to a single multiply-add * operation. Data copy and pointer math operations are not included in * this calculation, it is assumed that they correlate to the multiply-add * operations. Calculation also does not include final rounding, dithering * and clamping operations since they cannot be optimized out anyway. * * Calculation of the CRPosBuf buffer is not included since it cannot be * avoided. * * This function should be called after the updateFilterStepBuffers() * function. * * @param Steps Filtering steps array. * @param Vars Variables object. * @param UsedFracMap The map of used fractional delay filters. * @param ScanlineCount Scanline count. */ static int calcComplexity( const CFilterSteps& Steps, const CImageResizerVars& Vars, const CBuffer< uint8_t >& UsedFracMap, const int ScanlineCount ) { int fcnum; // Filter complexity multiplier numerator. int fcdenom; // Filter complexity multiplier denominator. if( Vars.packmode != 0 ) { fcnum = 1; fcdenom = 1; } else { // In interleaved processing mode, filters require 1 less // multiplication per 2 multiply-add instructions. fcnum = 3; fcdenom = 4; } int s = 0; // Complexity per one scanline. int s2 = 0; // Complexity per all scanlines. int i; for( i = 0; i < Steps.getItemCount(); i++ ) { const CFilterStep& fs = Steps[ i ]; s2 += 65 * fs.Flt.getCapacity(); // Filter creation complexity. if( fs.IsUpsample ) { if( fs.FltOrig.getCapacity() > 0 ) { continue; } s += ( fs.Flt.getCapacity() * ( fs.InPrefix + fs.InLen + fs.InSuffix ) + fs.SuffixDC.getCapacity() + fs.PrefixDC.getCapacity() ) * Vars.ElCount; } else if( fs.ResampleFactor == 0 ) { s += fs.FltBank -> getFilterLen() * ( fs.FltBank -> getOrder() + Vars.ElCount ) * fs.OutLen; s2 += fs.FltBank -> calcInitComplexity( UsedFracMap ); } else { s += fs.Flt.getCapacity() * Vars.ElCount * fs.OutLen * fcnum / fcdenom; } } return( s + s2 / ScanlineCount ); } /** * @brief Thread-isolated data used for scanline processing. * * This structure holds data necessary for image's horizontal or vertical * scanline processing, including scanline processing queue. * * @tparam Tin Source element data type. Intermediate buffers store data * in floating point format. * @tparam Tout Destination element data type. Intermediate buffers store * data in floating point format. */ template< class Tin, class Tout > class CThreadData : public CImageResizerThreadPool :: CWorkload { public: virtual void process() { processScanlineQueue(); } /** * This enumeration lists possible scanline operations. */ enum EScanlineOperation { sopResizeH, ///< Resize horizontal scanline. ///< sopResizeV, ///< Resize vertical scanline. ///< sopDitherAndUnpackH, ///< Dither and unpack horizontal scanline. ///< sopUnpackH ///< Unpack horizontal scanline. ///< }; /** * Function initializes *this thread data object and assigns certain * variables provided by the higher level code. * * @param aThreadIndex Index of this thread data (0-based). * @param aThreadCount Total number of threads used during processing. * @param aSteps Filtering steps. * @param aVars Image resizer variables. */ void init( const int aThreadIndex, const int aThreadCount, const CFilterSteps& aSteps, const CImageResizerVars& aVars ) { ThreadIndex = aThreadIndex; ThreadCount = aThreadCount; Steps = &aSteps; Vars = &aVars; } /** * Function initializes scanline processing queue, and updates * capacities of intermediate buffers. * * @param aOp Operation to perform over scanline. * @param TotalLines The total number of scanlines that will be * processed by all threads. * @param aSrcLen Source scanline length in pixels. * @param aSrcIncr Source scanline buffer increment. Ignored in * horizontal scanline processing. * @param aResIncr Resulting scanline buffer increment. Ignored in * horizontal scanline processing. */ void initScanlineQueue( const EScanlineOperation aOp, const int TotalLines, const int aSrcLen, const int aSrcIncr = 0, const int aResIncr = 0 ) { const int l = Vars -> BufLen[ 0 ] + Vars -> BufLen[ 1 ]; if( Bufs.getCapacity() < l ) { Bufs.alloc( l, fpclass :: fpalign ); } BufPtrs[ 0 ] = Bufs + Vars -> BufOffs[ 0 ]; BufPtrs[ 1 ] = Bufs + Vars -> BufLen[ 0 ] + Vars -> BufOffs[ 1 ]; int j; int ml = 0; for( j = 0; j < Steps -> getItemCount(); j++ ) { const CFilterStep& fs = (*Steps)[ j ]; if( fs.ResampleFactor == 0 && ml < fs.FltBank -> getFilterLen() ) { ml = fs.FltBank -> getFilterLen(); } } TmpFltBuf.alloc( ml, fpclass :: fpalign ); ScanlineOp = aOp; SrcLen = aSrcLen; SrcIncr = aSrcIncr; ResIncr = aResIncr; QueueLen = 0; Queue.increaseCapacity(( TotalLines + ThreadCount - 1 ) / ThreadCount, false ); } /** * Function adds a scanline to the queue buffer. The * initScanlineQueue() function should be called before calling this * function. The number of calls to this add function should not * exceed the TotalLines spread over all threads. * * @param SrcBuf Source scanline buffer. * @param ResBuf Resulting scanline buffer. */ void addScanlineToQueue( void* const SrcBuf, void* const ResBuf ) { Queue[ QueueLen ].SrcBuf = SrcBuf; Queue[ QueueLen ].ResBuf = ResBuf; QueueLen++; } /** * Function processes all queued scanlines. */ void processScanlineQueue() { int i; switch( ScanlineOp ) { case sopResizeH: { for( i = 0; i < QueueLen; i++ ) { resizeScanlineH( (Tin*) Queue[ i ].SrcBuf, (fptype*) Queue[ i ].ResBuf ); } break; } case sopResizeV: { for( i = 0; i < QueueLen; i++ ) { resizeScanlineV( (fptype*) Queue[ i ].SrcBuf, (fptype*) Queue[ i ].ResBuf ); } break; } case sopDitherAndUnpackH: { if( Vars -> UseSRGBGamma ) { for( i = 0; i < QueueLen; i++ ) { CFilterStep :: applySRGBGamma( (fptype*) Queue[ i ].SrcBuf, SrcLen, *Vars ); Ditherer.dither( (fptype*) Queue[ i ].SrcBuf ); CFilterStep :: unpackScanline( (fptype*) Queue[ i ].SrcBuf, (Tout*) Queue[ i ].ResBuf, SrcLen, *Vars ); } } else { for( i = 0; i < QueueLen; i++ ) { Ditherer.dither( (fptype*) Queue[ i ].SrcBuf ); CFilterStep :: unpackScanline( (fptype*) Queue[ i ].SrcBuf, (Tout*) Queue[ i ].ResBuf, SrcLen, *Vars ); } } break; } case sopUnpackH: { if( Vars -> UseSRGBGamma ) { for( i = 0; i < QueueLen; i++ ) { CFilterStep :: applySRGBGamma( (fptype*) Queue[ i ].SrcBuf, SrcLen, *Vars ); CFilterStep :: unpackScanline( (fptype*) Queue[ i ].SrcBuf, (Tout*) Queue[ i ].ResBuf, SrcLen, *Vars ); } } else { for( i = 0; i < QueueLen; i++ ) { CFilterStep :: unpackScanline( (fptype*) Queue[ i ].SrcBuf, (Tout*) Queue[ i ].ResBuf, SrcLen, *Vars ); } } break; } } } /** * Function returns ditherer object associated with *this thread data * object. */ CDitherer& getDitherer() { return( Ditherer ); } private: int ThreadIndex; ///< Thread index. ///< int ThreadCount; ///< Thread count. ///< const CFilterSteps* Steps; ///< Filtering steps. ///< const CImageResizerVars* Vars; ///< Image resizer variables. ///< CBuffer< fptype > Bufs; ///< Flip-flop intermediate buffers. ///< fptype* BufPtrs[ 3 ]; ///< Flip-flop buffer pointers (referenced by ///< filtering step's InBuf and OutBuf indices). ///< CBuffer< fptype > TmpFltBuf; ///< Temporary buffer used in the ///< doResize() function, aligned by fpclass :: fpalign. ///< EScanlineOperation ScanlineOp; ///< Operation to perform over ///< scanline. ///< int SrcLen; ///< Source scanline length in the last queue. ///< int SrcIncr; ///< Source scanline buffer increment in the last queue. ///< int ResIncr; ///< Resulting scanline buffer increment in the last ///< queue. ///< CDitherer Ditherer; ///< Ditherer object to use. ///< /** * @brief Scanline processing queue item. * * Scanline processing queue item. */ struct CQueueItem { void* SrcBuf; ///< Source scanline buffer, will by typecasted to ///< Tin or fptype*. ///< void* ResBuf; ///< Resulting scanline buffer, will by typecasted ///< to Tout or fptype*. ///< }; CBuffer< CQueueItem > Queue; ///< Scanline processing queue. ///< int QueueLen; ///< Queue length. ///< /** * Function resizes a single horizontal scanline. * * @param SrcBuf Source scanline buffer. Can be either horizontal or * vertical. * @param ResBuf Resulting scanline buffer. */ void resizeScanlineH( const Tin* const SrcBuf, fptype* const ResBuf ) { (*Steps)[ 0 ].packScanline( SrcBuf, BufPtrs[ 0 ], SrcLen ); BufPtrs[ 2 ] = ResBuf; int j; for( j = 0; j < Steps -> getItemCount(); j++ ) { const CFilterStep& fs = (*Steps)[ j ]; fs.prepareInBuf( BufPtrs[ fs.InBuf ]); const int DstIncr = ( Vars -> packmode == 0 ? Vars -> ElCount : 1 ); if( fs.ResampleFactor != 0 ) { if( fs.IsUpsample ) { fs.doUpsample( BufPtrs[ fs.InBuf ], BufPtrs[ fs.OutBuf ]); } else { fs.doFilter( BufPtrs[ fs.InBuf ], BufPtrs[ fs.OutBuf ], DstIncr ); } } else { fs.doResize( BufPtrs[ fs.InBuf ], BufPtrs[ fs.OutBuf ], DstIncr, TmpFltBuf ); } } } /** * Function resizes a single vertical scanline. * * @param SrcBuf Source scanline buffer. Can be either horizontal or * vertical. * @param ResBuf Resulting scanline buffer. */ void resizeScanlineV( const fptype* const SrcBuf, fptype* const ResBuf ) { (*Steps)[ 0 ].convertVtoH( SrcBuf, BufPtrs[ 0 ], SrcLen, SrcIncr ); BufPtrs[ 2 ] = ResBuf; int j; for( j = 0; j < Steps -> getItemCount(); j++ ) { const CFilterStep& fs = (*Steps)[ j ]; fs.prepareInBuf( BufPtrs[ fs.InBuf ]); const int DstIncr = ( fs.OutBuf == 2 ? ResIncr : ( Vars -> packmode == 0 ? Vars -> ElCount : 1 )); if( fs.ResampleFactor != 0 ) { if( fs.IsUpsample ) { fs.doUpsample( BufPtrs[ fs.InBuf ], BufPtrs[ fs.OutBuf ]); } else { fs.doFilter( BufPtrs[ fs.InBuf ], BufPtrs[ fs.OutBuf ], DstIncr ); } } else { fs.doResize( BufPtrs[ fs.InBuf ], BufPtrs[ fs.OutBuf ], DstIncr, TmpFltBuf ); } } } }; }; #undef AVIR_PI #undef AVIR_PId2 } // namespace avir #endif // AVIR_CIMAGERESIZER_INCLUDED