Better Nohalo comments
This commit is contained in:
parent
4f9e33aaf4
commit
4112bcc70a
@ -1,7 +1,7 @@
|
|||||||
/* Nohalo subdivision followed by LBB (Locally Bounded Bicubic)
|
/* Nohalo subdivision followed by LBB (Locally Bounded Bicubic)
|
||||||
* interpolation
|
* interpolation
|
||||||
*
|
*
|
||||||
* N. Robidoux and C. Racette 11/5--17/5/2009
|
* N. Robidoux, C. Racette and J. Cupitt 11/5--18/5/2009
|
||||||
*
|
*
|
||||||
* N. Robidoux 1/4-29/5/2009
|
* N. Robidoux 1/4-29/5/2009
|
||||||
*
|
*
|
||||||
@ -68,7 +68,7 @@
|
|||||||
*
|
*
|
||||||
* "Nohalo" is a resampler with a mission: smoothly straightening
|
* "Nohalo" is a resampler with a mission: smoothly straightening
|
||||||
* oblique lines without undesirable side-effects. In particular,
|
* oblique lines without undesirable side-effects. In particular,
|
||||||
* without much blurring and with absolutely no added haloing.
|
* without much blurring and with no added haloing.
|
||||||
*
|
*
|
||||||
* In this code, one Nohalo subdivision is performed. The
|
* In this code, one Nohalo subdivision is performed. The
|
||||||
* interpolation is finished with LBB (Locally Bounded Bicubic).
|
* interpolation is finished with LBB (Locally Bounded Bicubic).
|
||||||
@ -79,28 +79,35 @@
|
|||||||
* Nohalo is interpolatory
|
* Nohalo is interpolatory
|
||||||
* =======================
|
* =======================
|
||||||
*
|
*
|
||||||
* That is, nohalo preserves point values: If asked for the value at
|
* That is, Nohalo preserves point values: If asked for the value at
|
||||||
* the center of an input pixel, the sampler returns the corresponding
|
* the center of an input pixel, the sampler returns the corresponding
|
||||||
* value, unchanged. In addition, because nohalo is continuous, if
|
* value, unchanged. In addition, because Nohalo is continuous, if
|
||||||
* asked for a value at a location "very close" to the center of an
|
* asked for a value at a location "very close" to the center of an
|
||||||
* input pixel, then the sampler returns a value "very close" to
|
* input pixel, then the sampler returns a value "very close" to
|
||||||
* it. (Nohalo is not smoothing like, say, B-Spline
|
* it. (Nohalo is not smoothing like, say, B-Spline
|
||||||
* pseudo-interpolation.)
|
* pseudo-interpolation.)
|
||||||
*
|
*
|
||||||
* ========================================================
|
* ====================================================================
|
||||||
* Nohalo is co-monotone (this is why it's called "nohalo")
|
* Nohalo subdivision is co-monotone (this is why it's called "no-halo")
|
||||||
* ========================================================
|
* ====================================================================
|
||||||
*
|
*
|
||||||
* What monotonicity more or less means here is that the resampled
|
* One consequence of monotonicity is that additional subdivided
|
||||||
* value is in the range of the four closest input values. This
|
* values are in the range of the four closest input values, which is
|
||||||
* property is why there is no added haloing. It also implies that
|
* a form of local boundedness. (Note: plain vanilla bilinear and
|
||||||
* clamping is unnecessary (provided abyss values are within the range
|
* nearest neighbour are also co-monotone.) LBB is also locally
|
||||||
* of acceptable values, which is always the case). (Note: plain
|
* bounded. Consequently, Nohalo subdivision followed by LBB is
|
||||||
* vanilla bilinear is also co-monotone.)
|
* locally bounded. When used as a finishing scheme for Nohalo, the
|
||||||
|
* standard LBB bounds imply that the final interpolated value is in
|
||||||
|
* the range of the nine closest input values. This property is why
|
||||||
|
* there is very little added haloing, even when a finishing scheme
|
||||||
|
* which is not strictly monotone. Another consequence of local
|
||||||
|
* boundedness is that clamping is unnecessary (provided abyss values
|
||||||
|
* are within the range of acceptable values, which is "always" the
|
||||||
|
* case).
|
||||||
*
|
*
|
||||||
* Note: If the abyss policy is an extrapolating one---for example,
|
* Note: If the abyss policy is an extrapolating one---for example,
|
||||||
* linear or bilinear extrapolation---clamping is still unnecessary
|
* linear or bilinear extrapolation---clamping is still unnecessary
|
||||||
* unless one attempts to resample outside of the convex hull of the
|
* UNLESS one attempts to resample outside of the convex hull of the
|
||||||
* input pixel positions. Consequence: the "corner" image size
|
* input pixel positions. Consequence: the "corner" image size
|
||||||
* convention does not require clamping when using linear
|
* convention does not require clamping when using linear
|
||||||
* extrapolation abyss policy when performing image resizing, but the
|
* extrapolation abyss policy when performing image resizing, but the
|
||||||
@ -115,6 +122,10 @@
|
|||||||
* Nohalo is a local method
|
* Nohalo is a local method
|
||||||
* ========================
|
* ========================
|
||||||
*
|
*
|
||||||
|
* The interpolated pixel value when using Nohalo subdivision followed
|
||||||
|
* by LBB only depends on the 21 (5x5 minus the four corners) closest
|
||||||
|
* input values.
|
||||||
|
*
|
||||||
* ===============================
|
* ===============================
|
||||||
* Nohalo is second order accurate
|
* Nohalo is second order accurate
|
||||||
* ===============================
|
* ===============================
|
||||||
@ -122,15 +133,15 @@
|
|||||||
* (Except possibly near the boundary: it is easy to make this
|
* (Except possibly near the boundary: it is easy to make this
|
||||||
* property carry over everywhere but this requires a tuned abyss
|
* property carry over everywhere but this requires a tuned abyss
|
||||||
* policy---linear extrapolation, say---or building the boundary
|
* policy---linear extrapolation, say---or building the boundary
|
||||||
* conditions inside the sampler.) Nohalo is exact on linear
|
* conditions inside the sampler.) Nohalo+LBB is exact on linear
|
||||||
* intensity profiles, meaning that if the input pixel values (in the
|
* intensity profiles, meaning that if the input pixel values (in the
|
||||||
* stencil) are obtained from a function of the form f(x,y) = a + b*x
|
* stencil) are obtained from a function of the form f(x,y) = a + b*x
|
||||||
* + c*y (a, b, c constants), then the computed pixel value is exactly
|
* + c*y (a, b, c constants), then the computed pixel value is exactly
|
||||||
* the value of f(x,y) at the asked-for sampling location. The
|
* the value of f(x,y) at the asked-for sampling location. The
|
||||||
* boundary condition which is emulated by VIPS throught the "extend"
|
* boundary condition which is emulated by VIPS through the "extend"
|
||||||
* extension of the input image---this corresponds to the nearest
|
* extension of the input image---this corresponds to the nearest
|
||||||
* neighbour abyss policy---does NOT make this resampler exact on
|
* neighbour abyss policy---does NOT make this resampler exact on
|
||||||
* linears at the boundary. It does, however, guarantee that no
|
* linears near the boundary. It does, however, guarantee that no
|
||||||
* clamping is required even when resampled values are computed at
|
* clamping is required even when resampled values are computed at
|
||||||
* positions outside of the extent of the input image (when
|
* positions outside of the extent of the input image (when
|
||||||
* extrapolation is required).
|
* extrapolation is required).
|
||||||
@ -139,23 +150,23 @@
|
|||||||
* Nohalo is nonlinear
|
* Nohalo is nonlinear
|
||||||
* ===================
|
* ===================
|
||||||
*
|
*
|
||||||
* In particular, resampling a sum of images may not be the same as
|
* Both Nohalo and LBB are nonlinear, consequently their composition
|
||||||
* summing the resamples. (This occurs even without taking into account
|
* is nonlinear. In particular, resampling a sum of images may not be
|
||||||
* over and underflow issues: images can only take values within a
|
* the same as summing the resamples. (This occurs even without taking
|
||||||
* banded range, and consequently no sampler is truly linear.)
|
* into account over and underflow issues: images can only take values
|
||||||
|
* within a banded range, and consequently no sampler is truly
|
||||||
|
* linear.)
|
||||||
*
|
*
|
||||||
* ====================
|
* ====================
|
||||||
* Weaknesses of nohalo
|
* Weaknesses of Nohalo
|
||||||
* ====================
|
* ====================
|
||||||
*
|
*
|
||||||
* In some cases, the initial subdivision computation is wasted:
|
* In some cases, the initial subdivision computation is wasted:
|
||||||
*
|
*
|
||||||
* If a region is bichromatic, the nonlinear component of the level 1
|
* If a region is bi-chromatic, the nonlinear component of Nohalo
|
||||||
* nohalo is zero in the interior of the region, and consequently
|
* subdivision is zero in the interior of the region, and consequently
|
||||||
* nohalo boils down to bilinear. For such images, either stick to
|
* Nohalo subdivision boils down to bilinear. For such images, LBB is
|
||||||
* bilinear, or use a higher level (quality) setting. (There is no
|
* probably a better choice.
|
||||||
* real harm in using nohalo when it boils down to bilinear if one
|
|
||||||
* does not mind wasting cycles.)
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
@ -741,9 +752,25 @@ lbbicubic( const double c00,
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Computation of the four min and four max over 3x3 input data
|
* Computation of the four min and four max over 3x3 input data
|
||||||
* sub-blocks of the 4x4 input stencil. (Because there is
|
* sub-blocks of the 4x4 input stencil.
|
||||||
* redundancy, only 17 minima and 17 maxima are needed; if done with
|
*
|
||||||
* conditional moves, only 28 different flags are involved.)
|
* Surprisingly, we have not succeeded in using the fact that the
|
||||||
|
* data comes from the (co-monotone) method Nohalo so that it is
|
||||||
|
* known ahead of time that
|
||||||
|
*
|
||||||
|
* dos_thr is between dos_two and dos_fou
|
||||||
|
*
|
||||||
|
* tre_two is between dos_two and qua_two
|
||||||
|
*
|
||||||
|
* tre_fou is between dos_fou and qua_fou
|
||||||
|
*
|
||||||
|
* qua_thr is between qua_two and qua_fou
|
||||||
|
*
|
||||||
|
* tre_thr is in the convex hull of dos_two, dos_fou, qua_two and qua_fou
|
||||||
|
*
|
||||||
|
* to minimize the number of flags and conditional moves.
|
||||||
|
*
|
||||||
|
* Suggestions welcome!
|
||||||
*/
|
*/
|
||||||
const double m1 = (dos_two <= dos_thr) ? dos_two : dos_thr ;
|
const double m1 = (dos_two <= dos_thr) ? dos_two : dos_thr ;
|
||||||
const double M1 = (dos_two <= dos_thr) ? dos_thr : dos_two ;
|
const double M1 = (dos_two <= dos_thr) ? dos_thr : dos_two ;
|
||||||
@ -1033,20 +1060,20 @@ lbbicubic( const double c00,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Call Nohalo with an interpolator as a parameter.
|
* Call Nohalo+LBB with a careful type conversion as a parameter.
|
||||||
*
|
*
|
||||||
* It would be nice to do this with templates somehow---for one thing
|
* It would be nice to do this with templates somehow---for one thing
|
||||||
* this would allow code comments!---but we can't figure a clean way
|
* this would allow code comments!---but we can't figure a clean way
|
||||||
* to do it.
|
* to do it.
|
||||||
*/
|
*/
|
||||||
#define NOHALO_INTER( inter ) \
|
#define NOHALO_CONVERSION( conversion ) \
|
||||||
template <typename T> static void inline \
|
template <typename T> static void inline \
|
||||||
nohalo_ ## inter( PEL* restrict pout, \
|
nohalo_ ## conversion( PEL* restrict pout, \
|
||||||
const PEL* restrict pin, \
|
const PEL* restrict pin, \
|
||||||
const int bands, \
|
const int bands, \
|
||||||
const int lskip, \
|
const int lskip, \
|
||||||
const double x_0, \
|
const double x_0, \
|
||||||
const double y_0 ) \
|
const double y_0 ) \
|
||||||
{ \
|
{ \
|
||||||
T* restrict out = (T *) pout; \
|
T* restrict out = (T *) pout; \
|
||||||
\
|
\
|
||||||
@ -1261,28 +1288,28 @@ lbbicubic( const double c00,
|
|||||||
qua_thr, \
|
qua_thr, \
|
||||||
qua_fou ); \
|
qua_fou ); \
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
const T result = to_ ## inter<T>( double_result ); \
|
const T result = to_ ## conversion<T>( double_result ); \
|
||||||
in++; \
|
in++; \
|
||||||
*out++ = result; \
|
*out++ = result; \
|
||||||
} \
|
} \
|
||||||
\
|
\
|
||||||
} while (--band); \
|
} while (--band); \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
NOHALO_INTER( fptypes )
|
NOHALO_CONVERSION( fptypes )
|
||||||
NOHALO_INTER( withsign )
|
NOHALO_CONVERSION( withsign )
|
||||||
NOHALO_INTER( nosign )
|
NOHALO_CONVERSION( nosign )
|
||||||
|
|
||||||
|
|
||||||
#define CALL( T, inter ) \
|
#define CALL( T, conversion ) \
|
||||||
nohalo_ ## inter<T>( out, \
|
nohalo_ ## conversion<T>( out, \
|
||||||
p, \
|
p, \
|
||||||
bands, \
|
bands, \
|
||||||
lskip, \
|
lskip, \
|
||||||
relative_x, \
|
relative_x, \
|
||||||
relative_y );
|
relative_y );
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
Loading…
Reference in New Issue
Block a user