2023-12-14 19:09:00 +08:00
/* stb_image_resize2 - v2.04 - public domain image resizing
2023-10-10 08:23:04 +08:00
by Jeff Roberts ( v2 ) and Jorge L Rodriguez
http : //github.com/nothings/stb
Can be threaded with the extended API . SSE2 , AVX , Neon and WASM SIMD support . Only
scaling and translation is supported , no rotations or shears .
COMPILING & LINKING
In one C / C + + file that # includes this file , do this :
# define STB_IMAGE_RESIZE_IMPLEMENTATION
before the # include . That will create the implementation in that file .
PORTING FROM VERSION 1
The API has changed . You can continue to use the old version of stb_image_resize . h ,
which is available in the " deprecated/ " directory .
If you ' re using the old simple - to - use API , porting is straightforward .
( For more advanced APIs , read the documentation . )
stbir_resize_uint8 ( ) :
- call ` stbir_resize_uint8_linear ` , cast channel count to ` stbir_pixel_layout `
stbir_resize_float ( ) :
- call ` stbir_resize_float_linear ` , cast channel count to ` stbir_pixel_layout `
stbir_resize_uint8_srgb ( ) :
- function name is unchanged
- cast channel count to ` stbir_pixel_layout `
- above is sufficient unless your image has alpha and it ' s not RGBA / BGRA
- in that case , follow the below instructions for stbir_resize_uint8_srgb_edgemode
stbir_resize_uint8_srgb_edgemode ( )
- switch to the " medium complexity " API
- stbir_resize ( ) , very similar API but a few more parameters :
- pixel_layout : cast channel count to ` stbir_pixel_layout `
- data_type : STBIR_TYPE_UINT8_SRGB
- edge : unchanged ( STBIR_EDGE_WRAP , etc . )
- filter : STBIR_FILTER_DEFAULT
- which channel is alpha is specified in stbir_pixel_layout , see enum for details
EASY API CALLS :
Easy API downsamples w / Mitchell filter , upsamples w / cubic interpolation , clamps to edge .
stbir_resize_uint8_srgb ( input_pixels , input_w , input_h , input_stride_in_bytes ,
output_pixels , output_w , output_h , output_stride_in_bytes ,
pixel_layout_enum )
stbir_resize_uint8_linear ( input_pixels , input_w , input_h , input_stride_in_bytes ,
output_pixels , output_w , output_h , output_stride_in_bytes ,
pixel_layout_enum )
stbir_resize_float_linear ( input_pixels , input_w , input_h , input_stride_in_bytes ,
output_pixels , output_w , output_h , output_stride_in_bytes ,
pixel_layout_enum )
If you pass NULL or zero for the output_pixels , we will allocate the output buffer
for you and return it from the function ( free with free ( ) or STBIR_FREE ) .
As a special case , XX_stride_in_bytes of 0 means packed continuously in memory .
API LEVELS
There are three levels of API - easy - to - use , medium - complexity and extended - complexity .
See the " header file " section of the source for API documentation .
ADDITIONAL DOCUMENTATION
MEMORY ALLOCATION
By default , we use malloc and free for memory allocation . To override the
memory allocation , before the implementation # include , add a :
# define STBIR_MALLOC(size,user_data) ...
# define STBIR_FREE(ptr,user_data) ...
Each resize makes exactly one call to malloc / free ( unless you use the
extended API where you can do one allocation for many resizes ) . Under
address sanitizer , we do separate allocations to find overread / writes .
PERFORMANCE
This library was written with an emphasis on performance . When testing
stb_image_resize with RGBA , the fastest mode is STBIR_4CHANNEL with
STBIR_TYPE_UINT8 pixels and CLAMPed edges ( which is what many other resize
libs do by default ) . Also , make sure SIMD is turned on of course ( default
for 64 - bit targets ) . Avoid WRAP edge mode if you want the fastest speed .
This library also comes with profiling built - in . If you define STBIR_PROFILE ,
you can use the advanced API and get low - level profiling information by
calling stbir_resize_extended_profile_info ( ) or stbir_resize_split_profile_info ( )
after a resize .
SIMD
Most of the routines have optimized SSE2 , AVX , NEON and WASM versions .
On Microsoft compilers , we automatically turn on SIMD for 64 - bit x64 and
ARM ; for 32 - bit x86 and ARM , you select SIMD mode by defining STBIR_SSE2 or
STBIR_NEON . For AVX and AVX2 , we auto - select it by detecting the / arch : AVX
or / arch : AVX2 switches . You can also always manually turn SSE2 , AVX or AVX2
support on by defining STBIR_SSE2 , STBIR_AVX or STBIR_AVX2 .
On Linux , SSE2 and Neon is on by default for 64 - bit x64 or ARM64 . For 32 - bit ,
we select x86 SIMD mode by whether you have - msse2 , - mavx or - mavx2 enabled
on the command line . For 32 - bit ARM , you must pass - mfpu = neon - vfpv4 for both
clang and GCC , but GCC also requires an additional - mfp16 - format = ieee to
automatically enable NEON .
On x86 platforms , you can also define STBIR_FP16C to turn on FP16C instructions
for converting back and forth to half - floats . This is autoselected when we
are using AVX2 . Clang and GCC also require the - mf16c switch . ARM always uses
the built - in half float hardware NEON instructions .
You can also tell us to use multiply - add instructions with STBIR_USE_FMA .
Because x86 doesn ' t always have fma , we turn it off by default to maintain
determinism across all platforms . If you don ' t care about non - FMA determinism
and are willing to restrict yourself to more recent x86 CPUs ( around the AVX
timeframe ) , then fma will give you around a 15 % speedup .
You can force off SIMD in all cases by defining STBIR_NO_SIMD . You can turn
off AVX or AVX2 specifically with STBIR_NO_AVX or STBIR_NO_AVX2 . AVX is 10 %
to 40 % faster , and AVX2 is generally another 12 % .
ALPHA CHANNEL
Most of the resizing functions provide the ability to control how the alpha
channel of an image is processed .
When alpha represents transparency , it is important that when combining
colors with filtering , the pixels should not be treated equally ; they
should use a weighted average based on their alpha values . For example ,
if a pixel is 1 % opaque bright green and another pixel is 99 % opaque
black and you average them , the average will be 50 % opaque , but the
unweighted average and will be a middling green color , while the weighted
average will be nearly black . This means the unweighted version introduced
green energy that didn ' t exist in the source image .
( If you want to know why this makes sense , you can work out the math for
the following : consider what happens if you alpha composite a source image
over a fixed color and then average the output , vs . if you average the
source image pixels and then composite that over the same fixed color .
Only the weighted average produces the same result as the ground truth
composite - then - average result . )
Therefore , it is in general best to " alpha weight " the pixels when applying
filters to them . This essentially means multiplying the colors by the alpha
values before combining them , and then dividing by the alpha value at the
end .
The computer graphics industry introduced a technique called " premultiplied
alpha " or " associated alpha " in which image colors are stored in image files
already multiplied by their alpha . This saves some math when compositing ,
and also avoids the need to divide by the alpha at the end ( which is quite
inefficient ) . However , while premultiplied alpha is common in the movie CGI
industry , it is not commonplace in other industries like videogames , and most
consumer file formats are generally expected to contain not - premultiplied
colors . For example , Photoshop saves PNG files " unpremultiplied " , and web
browsers like Chrome and Firefox expect PNG images to be unpremultiplied .
Note that there are three possibilities that might describe your image
and resize expectation :
1. images are not premultiplied , alpha weighting is desired
2. images are not premultiplied , alpha weighting is not desired
3. images are premultiplied
Both case # 2 and case # 3 require the exact same math : no alpha weighting
should be applied or removed . Only case 1 requires extra math operations ;
the other two cases can be handled identically .
stb_image_resize expects case # 1 by default , applying alpha weighting to
images , expecting the input images to be unpremultiplied . This is what the
COLOR + ALPHA buffer types tell the resizer to do .
When you use the pixel layouts STBIR_RGBA , STBIR_BGRA , STBIR_ARGB ,
STBIR_ABGR , STBIR_RX , or STBIR_XR you are telling us that the pixels are
non - premultiplied . In these cases , the resizer will alpha weight the colors
( effectively creating the premultiplied image ) , do the filtering , and then
convert back to non - premult on exit .
When you use the pixel layouts STBIR_RGBA_PM , STBIR_RGBA_PM , STBIR_RGBA_PM ,
STBIR_RGBA_PM , STBIR_RX_PM or STBIR_XR_PM , you are telling that the pixels
ARE premultiplied . In this case , the resizer doesn ' t have to do the
premultipling - it can filter directly on the input . This about twice as
fast as the non - premultiplied case , so it ' s the right option if your data is
already setup correctly .
When you use the pixel layout STBIR_4CHANNEL or STBIR_2CHANNEL , you are
telling us that there is no channel that represents transparency ; it may be
RGB and some unrelated fourth channel that has been stored in the alpha
channel , but it is actually not alpha . No special processing will be
performed .
The difference between the generic 4 or 2 channel layouts , and the
specialized _PM versions is with the _PM versions you are telling us that
the data * is * alpha , just don ' t premultiply it . That ' s important when
using SRGB pixel formats , we need to know where the alpha is , because
it is converted linearly ( rather than with the SRGB converters ) .
Because alpha weighting produces the same effect as premultiplying , you
even have the option with non - premultiplied inputs to let the resizer
produce a premultiplied output . Because the intially computed alpha - weighted
output image is effectively premultiplied , this is actually more performant
than the normal path which un - premultiplies the output image as a final step .
Finally , when converting both in and out of non - premulitplied space ( for
example , when using STBIR_RGBA ) , we go to somewhat heroic measures to
ensure that areas with zero alpha value pixels get something reasonable
in the RGB values . If you don ' t care about the RGB values of zero alpha
pixels , you can call the stbir_set_non_pm_alpha_speed_over_quality ( )
function - this runs a premultiplied resize about 25 % faster . That said ,
when you really care about speed , using premultiplied pixels for both in
and out ( STBIR_RGBA_PM , etc ) much faster than both of these premultiplied
options .
PIXEL LAYOUT CONVERSION
The resizer can convert from some pixel layouts to others . When using the
stbir_set_pixel_layouts ( ) , you can , for example , specify STBIR_RGBA
on input , and STBIR_ARGB on output , and it will re - organize the channels
during the resize . Currently , you can only convert between two pixel
layouts with the same number of channels .
DETERMINISM
We commit to being deterministic ( from x64 to ARM to scalar to SIMD , etc ) .
This requires compiling with fast - math off ( using at least / fp : precise ) .
Also , you must turn off fp - contracting ( which turns mult + adds into fmas ) !
We attempt to do this with pragmas , but with Clang , you usually want to add
- ffp - contract = off to the command line as well .
For 32 - bit x86 , you must use SSE and SSE2 codegen for determinism . That is ,
if the scalar x87 unit gets used at all , we immediately lose determinism .
On Microsoft Visual Studio 2008 and earlier , from what we can tell there is
no way to be deterministic in 32 - bit x86 ( some x87 always leaks in , even
with fp : strict ) . On 32 - bit x86 GCC , determinism requires both - msse2 and
- fpmath = sse .
Note that we will not be deterministic with float data containing NaNs -
the NaNs will propagate differently on different SIMD and platforms .
If you turn on STBIR_USE_FMA , then we will be deterministic with other
fma targets , but we will differ from non - fma targets ( this is unavoidable ,
because a fma isn ' t simply an add with a mult - it also introduces a
rounding difference compared to non - fma instruction sequences .
FLOAT PIXEL FORMAT RANGE
Any range of values can be used for the non - alpha float data that you pass
in ( 0 to 1 , - 1 to 1 , whatever ) . However , if you are inputting float values
but * outputting * bytes or shorts , you must use a range of 0 to 1 so that we
scale back properly . The alpha channel must also be 0 to 1 for any format
that does premultiplication prior to resizing .
Note also that with float output , using filters with negative lobes , the
output filtered values might go slightly out of range . You can define
STBIR_FLOAT_LOW_CLAMP and / or STBIR_FLOAT_HIGH_CLAMP to specify the range
to clamp to on output , if that ' s important .
MAX / MIN SCALE FACTORS
The input pixel resolutions are in integers , and we do the internal pointer
resolution in size_t sized integers . However , the scale ratio from input
resolution to output resolution is calculated in float form . This means
the effective possible scale ratio is limited to 24 bits ( or 16 million
to 1 ) . As you get close to the size of the float resolution ( again , 16
million pixels wide or high ) , you might start seeing float inaccuracy
issues in general in the pipeline . If you have to do extreme resizes ,
you can usually do this is multiple stages ( using float intermediate
buffers ) .
FLIPPED IMAGES
Stride is just the delta from one scanline to the next . This means you can
use a negative stride to handle inverted images ( point to the final
scanline and use a negative stride ) . You can invert the input or output ,
using negative strides .
DEFAULT FILTERS
For functions which don ' t provide explicit control over what filters to
use , you can change the compile - time defaults with :
# define STBIR_DEFAULT_FILTER_UPSAMPLE STBIR_FILTER_something
# define STBIR_DEFAULT_FILTER_DOWNSAMPLE STBIR_FILTER_something
See stbir_filter in the header - file section for the list of filters .
NEW FILTERS
A number of 1 D filter kernels are supplied . For a list of supported
filters , see the stbir_filter enum . You can install your own filters by
using the stbir_set_filter_callbacks function .
PROGRESS
For interactive use with slow resize operations , you can use the the
scanline callbacks in the extended API . It would have to be a * very * large
image resample to need progress though - we ' re very fast .
CEIL and FLOOR
In scalar mode , the only functions we use from math . h are ceilf and floorf ,
but if you have your own versions , you can define the STBIR_CEILF ( v ) and
STBIR_FLOORF ( v ) macros and we ' ll use them instead . In SIMD , we just use
our own versions .
ASSERT
Define STBIR_ASSERT ( boolval ) to override assert ( ) and not use assert . h
FUTURE TODOS
* For polyphase integral filters , we just memcpy the coeffs to dupe
them , but we should indirect and use the same coeff memory .
* Add pixel layout conversions for sensible different channel counts
( maybe , 1 - > 3 / 4 , 3 - > 4 , 4 - > 1 , 3 - > 1 ) .
* For SIMD encode and decode scanline routines , do any pre - aligning
for bad input / output buffer alignments and pitch ?
* For very wide scanlines , we should we do vertical strips to stay within
L2 cache . Maybe do chunks of 1 K pixels at a time . There would be
some pixel reconversion , but probably dwarfed by things falling out
of cache . Probably also something possible with alternating between
scattering and gathering at high resize scales ?
* Rewrite the coefficient generator to do many at once .
* AVX - 512 vertical kernels - worried about downclocking here .
* Convert the reincludes to macros when we know they aren ' t changing .
* Experiment with pivoting the horizontal and always using the
vertical filters ( which are faster , but perhaps not enough to overcome
the pivot cost and the extra memory touches ) . Need to buffer the whole
image so have to balance memory use .
* Most of our code is internally function pointers , should we compile
all the SIMD stuff always and dynamically dispatch ?
CONTRIBUTORS
Jeff Roberts : 2.0 implementation , optimizations , SIMD
Martins Mozeiko : NEON simd , WASM simd , clang and GCC whisperer .
Fabian Giesen : half float and srgb converters
Sean Barrett : API design , optimizations
Jorge L Rodriguez : Original 1.0 implementation
Aras Pranckevicius : bugfixes for 1.0
Nathan Reed : warning fixes for 1.0
REVISIONS
2023-12-14 19:09:00 +08:00
2.04 ( 2023 - 11 - 17 ) Fix for rare AVX bug , shadowed symbol ( thanks Nikola Smiljanic ) .
2.03 ( 2023 - 11 - 01 ) ASAN and TSAN warnings fixed , minor tweaks .
2.00 ( 2023 - 10 - 10 ) mostly new source : new api , optimizations , simd , vertical - first , etc
( 2 x - 5 x faster without simd , 4 x - 12 x faster with simd )
( in some cases , 20 x to 40 x faster - resizing to very small for example )
2023-10-10 08:23:04 +08:00
0.96 ( 2019 - 03 - 04 ) fixed warnings
0.95 ( 2017 - 07 - 23 ) fixed warnings
0.94 ( 2017 - 03 - 18 ) fixed warnings
0.93 ( 2017 - 03 - 03 ) fixed bug with certain combinations of heights
0.92 ( 2017 - 01 - 02 ) fix integer overflow on large ( > 2 GB ) images
0.91 ( 2016 - 04 - 02 ) fix warnings ; fix handling of subpixel regions
0.90 ( 2014 - 09 - 17 ) first released version
LICENSE
See end of file for license information .
*/
# if !defined(STB_IMAGE_RESIZE_DO_HORIZONTALS) && !defined(STB_IMAGE_RESIZE_DO_VERTICALS) && !defined(STB_IMAGE_RESIZE_DO_CODERS) // for internal re-includes
# ifndef STBIR_INCLUDE_STB_IMAGE_RESIZE2_H
# define STBIR_INCLUDE_STB_IMAGE_RESIZE2_H
# include <stddef.h>
# ifdef _MSC_VER
typedef unsigned char stbir_uint8 ;
typedef unsigned short stbir_uint16 ;
typedef unsigned int stbir_uint32 ;
typedef unsigned __int64 stbir_uint64 ;
# else
# include <stdint.h>
typedef uint8_t stbir_uint8 ;
typedef uint16_t stbir_uint16 ;
typedef uint32_t stbir_uint32 ;
typedef uint64_t stbir_uint64 ;
# endif
# ifdef _M_IX86_FP
# if ( _M_IX86_FP >= 1 )
# ifndef STBIR_SSE
# define STBIR_SSE
# endif
# endif
# endif
# if defined(_x86_64) || defined( __x86_64__ ) || defined( _M_X64 ) || defined(__x86_64) || defined(_M_AMD64) || defined(__SSE2__) || defined(STBIR_SSE) || defined(STBIR_SSE2)
# ifndef STBIR_SSE2
# define STBIR_SSE2
# endif
# if defined(__AVX__) || defined(STBIR_AVX2)
# ifndef STBIR_AVX
# ifndef STBIR_NO_AVX
# define STBIR_AVX
# endif
# endif
# endif
# if defined(__AVX2__) || defined(STBIR_AVX2)
# ifndef STBIR_NO_AVX2
# ifndef STBIR_AVX2
# define STBIR_AVX2
# endif
# if defined( _MSC_VER ) && !defined(__clang__)
# ifndef STBIR_FP16C // FP16C instructions are on all AVX2 cpus, so we can autoselect it here on microsoft - clang needs -m16c
# define STBIR_FP16C
# endif
# endif
# endif
# endif
# ifdef __F16C__
# ifndef STBIR_FP16C // turn on FP16C instructions if the define is set (for clang and gcc)
# define STBIR_FP16C
# endif
# endif
# endif
# if defined( _M_ARM64 ) || defined( __aarch64__ ) || defined( __arm64__ ) || defined(_M_ARM) || (__ARM_NEON_FP & 4) != 0 && __ARM_FP16_FORMAT_IEEE != 0
# ifndef STBIR_NEON
# define STBIR_NEON
# endif
# endif
# if defined(_M_ARM)
# ifdef STBIR_USE_FMA
# undef STBIR_USE_FMA // no FMA for 32-bit arm on MSVC
# endif
# endif
# if defined(__wasm__) && defined(__wasm_simd128__)
# ifndef STBIR_WASM
# define STBIR_WASM
# endif
# endif
# ifndef STBIRDEF
# ifdef STB_IMAGE_RESIZE_STATIC
# define STBIRDEF static
# else
# ifdef __cplusplus
# define STBIRDEF extern "C"
# else
# define STBIRDEF extern
# endif
# endif
# endif
//////////////////////////////////////////////////////////////////////////////
//// start "header file" ///////////////////////////////////////////////////
//
// Easy-to-use API:
//
// * stride is the offset between successive rows of image data
// in memory, in bytes. specify 0 for packed continuously in memory
// * colorspace is linear or sRGB as specified by function name
// * Uses the default filters
// * Uses edge mode clamped
// * returned result is 1 for success or 0 in case of an error.
// stbir_pixel_layout specifies:
// number of channels
// order of channels
// whether color is premultiplied by alpha
// for back compatibility, you can cast the old channel count to an stbir_pixel_layout
typedef enum
{
STBIR_1CHANNEL = 1 ,
STBIR_2CHANNEL = 2 ,
STBIR_RGB = 3 , // 3-chan, with order specified (for channel flipping)
2023-12-14 19:09:00 +08:00
STBIR_BGR = 0 , // 3-chan, with order specified (for channel flipping)
2023-10-10 08:23:04 +08:00
STBIR_4CHANNEL = 5 ,
2023-12-14 19:09:00 +08:00
STBIR_RGBA = 4 , // alpha formats, where alpha is NOT premultiplied into color channels
2023-10-10 08:23:04 +08:00
STBIR_BGRA = 6 ,
STBIR_ARGB = 7 ,
STBIR_ABGR = 8 ,
STBIR_RA = 9 ,
STBIR_AR = 10 ,
2023-12-14 19:09:00 +08:00
STBIR_RGBA_PM = 11 , // alpha formats, where alpha is premultiplied into color channels
2023-10-10 08:23:04 +08:00
STBIR_BGRA_PM = 12 ,
STBIR_ARGB_PM = 13 ,
STBIR_ABGR_PM = 14 ,
STBIR_RA_PM = 15 ,
STBIR_AR_PM = 16 ,
2023-12-14 19:09:00 +08:00
STBIR_RGBA_NO_AW = 11 , // alpha formats, where NO alpha weighting is applied at all!
STBIR_BGRA_NO_AW = 12 , // these are just synonyms for the _PM flags (which also do
STBIR_ARGB_NO_AW = 13 , // no alpha weighting). These names just make it more clear
STBIR_ABGR_NO_AW = 14 , // for some folks).
STBIR_RA_NO_AW = 15 ,
STBIR_AR_NO_AW = 16 ,
2023-10-10 08:23:04 +08:00
} stbir_pixel_layout ;
//===============================================================
// Simple-complexity API
//
// If output_pixels is NULL (0), then we will allocate the buffer and return it to you.
//--------------------------------
STBIRDEF unsigned char * stbir_resize_uint8_srgb ( const unsigned char * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
unsigned char * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_type ) ;
STBIRDEF unsigned char * stbir_resize_uint8_linear ( const unsigned char * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
unsigned char * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_type ) ;
STBIRDEF float * stbir_resize_float_linear ( const float * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
float * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_type ) ;
//===============================================================
//===============================================================
// Medium-complexity API
//
// This extends the easy-to-use API as follows:
//
// * Can specify the datatype - U8, U8_SRGB, U16, FLOAT, HALF_FLOAT
// * Edge wrap can selected explicitly
// * Filter can be selected explicitly
//--------------------------------
typedef enum
{
STBIR_EDGE_CLAMP = 0 ,
STBIR_EDGE_REFLECT = 1 ,
STBIR_EDGE_WRAP = 2 , // this edge mode is slower and uses more memory
STBIR_EDGE_ZERO = 3 ,
} stbir_edge ;
typedef enum
{
STBIR_FILTER_DEFAULT = 0 , // use same filter type that easy-to-use API chooses
STBIR_FILTER_BOX = 1 , // A trapezoid w/1-pixel wide ramps, same result as box for integer scale ratios
STBIR_FILTER_TRIANGLE = 2 , // On upsampling, produces same results as bilinear texture filtering
STBIR_FILTER_CUBICBSPLINE = 3 , // The cubic b-spline (aka Mitchell-Netrevalli with B=1,C=0), gaussian-esque
STBIR_FILTER_CATMULLROM = 4 , // An interpolating cubic spline
STBIR_FILTER_MITCHELL = 5 , // Mitchell-Netrevalli filter with B=1/3, C=1/3
STBIR_FILTER_POINT_SAMPLE = 6 , // Simple point sampling
STBIR_FILTER_OTHER = 7 , // User callback specified
} stbir_filter ;
typedef enum
{
STBIR_TYPE_UINT8 = 0 ,
STBIR_TYPE_UINT8_SRGB = 1 ,
STBIR_TYPE_UINT8_SRGB_ALPHA = 2 , // alpha channel, when present, should also be SRGB (this is very unusual)
STBIR_TYPE_UINT16 = 3 ,
STBIR_TYPE_FLOAT = 4 ,
STBIR_TYPE_HALF_FLOAT = 5
} stbir_datatype ;
// medium api
STBIRDEF void * stbir_resize ( const void * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
void * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_layout , stbir_datatype data_type ,
stbir_edge edge , stbir_filter filter ) ;
//===============================================================
//===============================================================
// Extended-complexity API
//
// This API exposes all resize functionality.
//
// * Separate filter types for each axis
// * Separate edge modes for each axis
// * Separate input and output data types
// * Can specify regions with subpixel correctness
// * Can specify alpha flags
// * Can specify a memory callback
// * Can specify a callback data type for pixel input and output
// * Can be threaded for a single resize
// * Can be used to resize many frames without recalculating the sampler info
//
// Use this API as follows:
// 1) Call the stbir_resize_init function on a local STBIR_RESIZE structure
// 2) Call any of the stbir_set functions
// 3) Optionally call stbir_build_samplers() if you are going to resample multiple times
// with the same input and output dimensions (like resizing video frames)
// 4) Resample by calling stbir_resize_extended().
// 5) Call stbir_free_samplers() if you called stbir_build_samplers()
//--------------------------------
// Types:
// INPUT CALLBACK: this callback is used for input scanlines
typedef void const * stbir_input_callback ( void * optional_output , void const * input_ptr , int num_pixels , int x , int y , void * context ) ;
// OUTPUT CALLBACK: this callback is used for output scanlines
typedef void stbir_output_callback ( void const * output_ptr , int num_pixels , int y , void * context ) ;
// callbacks for user installed filters
typedef float stbir__kernel_callback ( float x , float scale , void * user_data ) ; // centered at zero
typedef float stbir__support_callback ( float scale , void * user_data ) ;
// internal structure with precomputed scaling
typedef struct stbir__info stbir__info ;
typedef struct STBIR_RESIZE // use the stbir_resize_init and stbir_override functions to set these values for future compatibility
{
void * user_data ;
void const * input_pixels ;
int input_w , input_h ;
double input_s0 , input_t0 , input_s1 , input_t1 ;
stbir_input_callback * input_cb ;
void * output_pixels ;
int output_w , output_h ;
int output_subx , output_suby , output_subw , output_subh ;
stbir_output_callback * output_cb ;
int input_stride_in_bytes ;
int output_stride_in_bytes ;
int splits ;
int fast_alpha ;
int needs_rebuild ;
int called_alloc ;
stbir_pixel_layout input_pixel_layout_public ;
stbir_pixel_layout output_pixel_layout_public ;
stbir_datatype input_data_type ;
stbir_datatype output_data_type ;
stbir_filter horizontal_filter , vertical_filter ;
stbir_edge horizontal_edge , vertical_edge ;
stbir__kernel_callback * horizontal_filter_kernel ; stbir__support_callback * horizontal_filter_support ;
stbir__kernel_callback * vertical_filter_kernel ; stbir__support_callback * vertical_filter_support ;
stbir__info * samplers ;
} STBIR_RESIZE ;
// extended complexity api
// First off, you must ALWAYS call stbir_resize_init on your resize structure before any of the other calls!
STBIRDEF void stbir_resize_init ( STBIR_RESIZE * resize ,
const void * input_pixels , int input_w , int input_h , int input_stride_in_bytes , // stride can be zero
void * output_pixels , int output_w , int output_h , int output_stride_in_bytes , // stride can be zero
stbir_pixel_layout pixel_layout , stbir_datatype data_type ) ;
//===============================================================
// You can update these parameters any time after resize_init and there is no cost
//--------------------------------
STBIRDEF void stbir_set_datatypes ( STBIR_RESIZE * resize , stbir_datatype input_type , stbir_datatype output_type ) ;
STBIRDEF void stbir_set_pixel_callbacks ( STBIR_RESIZE * resize , stbir_input_callback * input_cb , stbir_output_callback * output_cb ) ; // no callbacks by default
STBIRDEF void stbir_set_user_data ( STBIR_RESIZE * resize , void * user_data ) ; // pass back STBIR_RESIZE* by default
STBIRDEF void stbir_set_buffer_ptrs ( STBIR_RESIZE * resize , const void * input_pixels , int input_stride_in_bytes , void * output_pixels , int output_stride_in_bytes ) ;
//===============================================================
//===============================================================
// If you call any of these functions, you will trigger a sampler rebuild!
//--------------------------------
STBIRDEF int stbir_set_pixel_layouts ( STBIR_RESIZE * resize , stbir_pixel_layout input_pixel_layout , stbir_pixel_layout output_pixel_layout ) ; // sets new buffer layouts
STBIRDEF int stbir_set_edgemodes ( STBIR_RESIZE * resize , stbir_edge horizontal_edge , stbir_edge vertical_edge ) ; // CLAMP by default
STBIRDEF int stbir_set_filters ( STBIR_RESIZE * resize , stbir_filter horizontal_filter , stbir_filter vertical_filter ) ; // STBIR_DEFAULT_FILTER_UPSAMPLE/DOWNSAMPLE by default
STBIRDEF int stbir_set_filter_callbacks ( STBIR_RESIZE * resize , stbir__kernel_callback * horizontal_filter , stbir__support_callback * horizontal_support , stbir__kernel_callback * vertical_filter , stbir__support_callback * vertical_support ) ;
STBIRDEF int stbir_set_pixel_subrect ( STBIR_RESIZE * resize , int subx , int suby , int subw , int subh ) ; // sets both sub-regions (full regions by default)
STBIRDEF int stbir_set_input_subrect ( STBIR_RESIZE * resize , double s0 , double t0 , double s1 , double t1 ) ; // sets input sub-region (full region by default)
STBIRDEF int stbir_set_output_pixel_subrect ( STBIR_RESIZE * resize , int subx , int suby , int subw , int subh ) ; // sets output sub-region (full region by default)
// when inputting AND outputting non-premultiplied alpha pixels, we use a slower but higher quality technique
// that fills the zero alpha pixel's RGB values with something plausible. If you don't care about areas of
// zero alpha, you can call this function to get about a 25% speed improvement for STBIR_RGBA to STBIR_RGBA
// types of resizes.
STBIRDEF int stbir_set_non_pm_alpha_speed_over_quality ( STBIR_RESIZE * resize , int non_pma_alpha_speed_over_quality ) ;
//===============================================================
//===============================================================
// You can call build_samplers to prebuild all the internal data we need to resample.
// Then, if you call resize_extended many times with the same resize, you only pay the
// cost once.
// If you do call build_samplers, you MUST call free_samplers eventually.
//--------------------------------
// This builds the samplers and does one allocation
STBIRDEF int stbir_build_samplers ( STBIR_RESIZE * resize ) ;
// You MUST call this, if you call stbir_build_samplers or stbir_build_samplers_with_splits
STBIRDEF void stbir_free_samplers ( STBIR_RESIZE * resize ) ;
//===============================================================
// And this is the main function to perform the resize synchronously on one thread.
STBIRDEF int stbir_resize_extended ( STBIR_RESIZE * resize ) ;
//===============================================================
// Use these functions for multithreading.
// 1) You call stbir_build_samplers_with_splits first on the main thread
// 2) Then stbir_resize_with_split on each thread
// 3) stbir_free_samplers when done on the main thread
//--------------------------------
// This will build samplers for threading.
// You can pass in the number of threads you'd like to use (try_splits).
// It returns the number of splits (threads) that you can call it with.
/// It might be less if the image resize can't be split up that many ways.
STBIRDEF int stbir_build_samplers_with_splits ( STBIR_RESIZE * resize , int try_splits ) ;
// This function does a split of the resizing (you call this fuction for each
// split, on multiple threads). A split is a piece of the output resize pixel space.
// Note that you MUST call stbir_build_samplers_with_splits before stbir_resize_extended_split!
// Usually, you will always call stbir_resize_split with split_start as the thread_index
// and "1" for the split_count.
// But, if you have a weird situation where you MIGHT want 8 threads, but sometimes
// only 4 threads, you can use 0,2,4,6 for the split_start's and use "2" for the
// split_count each time to turn in into a 4 thread resize. (This is unusual).
STBIRDEF int stbir_resize_extended_split ( STBIR_RESIZE * resize , int split_start , int split_count ) ;
//===============================================================
//===============================================================
// Pixel Callbacks info:
//--------------------------------
// The input callback is super flexible - it calls you with the input address
// (based on the stride and base pointer), it gives you an optional_output
// pointer that you can fill, or you can just return your own pointer into
// your own data.
//
// You can also do conversion from non-supported data types if necessary - in
// this case, you ignore the input_ptr and just use the x and y parameters to
// calculate your own input_ptr based on the size of each non-supported pixel.
// (Something like the third example below.)
//
// You can also install just an input or just an output callback by setting the
// callback that you don't want to zero.
//
// First example, progress: (getting a callback that you can monitor the progress):
// void const * my_callback( void * optional_output, void const * input_ptr, int num_pixels, int x, int y, void * context )
// {
// percentage_done = y / input_height;
// return input_ptr; // use buffer from call
// }
//
// Next example, copying: (copy from some other buffer or stream):
// void const * my_callback( void * optional_output, void const * input_ptr, int num_pixels, int x, int y, void * context )
// {
// CopyOrStreamData( optional_output, other_data_src, num_pixels * pixel_width_in_bytes );
// return optional_output; // return the optional buffer that we filled
// }
//
// Third example, input another buffer without copying: (zero-copy from other buffer):
// void const * my_callback( void * optional_output, void const * input_ptr, int num_pixels, int x, int y, void * context )
// {
// void * pixels = ( (char*) other_image_base ) + ( y * other_image_stride ) + ( x * other_pixel_width_in_bytes );
// return pixels; // return pointer to your data without copying
// }
//
//
// The output callback is considerably simpler - it just calls you so that you can dump
// out each scanline. You could even directly copy out to disk if you have a simple format
// like TGA or BMP. You can also convert to other output types here if you want.
//
// Simple example:
// void const * my_output( void * output_ptr, int num_pixels, int y, void * context )
// {
// percentage_done = y / output_height;
// fwrite( output_ptr, pixel_width_in_bytes, num_pixels, output_file );
// }
//===============================================================
//===============================================================
// optional built-in profiling API
//--------------------------------
# ifdef STBIR_PROFILE
typedef struct STBIR_PROFILE_INFO
{
stbir_uint64 total_clocks ;
// how many clocks spent (of total_clocks) in the various resize routines, along with a string description
// there are "resize_count" number of zones
stbir_uint64 clocks [ 8 ] ;
char const * * descriptions ;
// count of clocks and descriptions
stbir_uint32 count ;
} STBIR_PROFILE_INFO ;
// use after calling stbir_resize_extended (or stbir_build_samplers or stbir_build_samplers_with_splits)
STBIRDEF void stbir_resize_build_profile_info ( STBIR_PROFILE_INFO * out_info , STBIR_RESIZE const * resize ) ;
// use after calling stbir_resize_extended
STBIRDEF void stbir_resize_extended_profile_info ( STBIR_PROFILE_INFO * out_info , STBIR_RESIZE const * resize ) ;
// use after calling stbir_resize_extended_split
STBIRDEF void stbir_resize_split_profile_info ( STBIR_PROFILE_INFO * out_info , STBIR_RESIZE const * resize , int split_start , int split_num ) ;
//===============================================================
# endif
//// end header file /////////////////////////////////////////////////////
# endif // STBIR_INCLUDE_STB_IMAGE_RESIZE2_H
# if defined(STB_IMAGE_RESIZE_IMPLEMENTATION) || defined(STB_IMAGE_RESIZE2_IMPLEMENTATION)
# ifndef STBIR_ASSERT
# include <assert.h>
# define STBIR_ASSERT(x) assert(x)
# endif
# ifndef STBIR_MALLOC
# include <stdlib.h>
# define STBIR_MALLOC(size,user_data) ((void)(user_data), malloc(size))
# define STBIR_FREE(ptr,user_data) ((void)(user_data), free(ptr))
// (we used the comma operator to evaluate user_data, to avoid "unused parameter" warnings)
# endif
# ifdef _MSC_VER
# define stbir__inline __forceinline
# else
# define stbir__inline __inline__
// Clang address sanitizer
# if defined(__has_feature)
# if __has_feature(address_sanitizer) || __has_feature(memory_sanitizer)
# ifndef STBIR__SEPARATE_ALLOCATIONS
# define STBIR__SEPARATE_ALLOCATIONS
# endif
# endif
# endif
# endif
// GCC and MSVC
# if defined(__SANITIZE_ADDRESS__)
# ifndef STBIR__SEPARATE_ALLOCATIONS
# define STBIR__SEPARATE_ALLOCATIONS
# endif
# endif
// Always turn off automatic FMA use - use STBIR_USE_FMA if you want.
// Otherwise, this is a determinism disaster.
# ifndef STBIR_DONT_CHANGE_FP_CONTRACT // override in case you don't want this behavior
# if defined(_MSC_VER) && !defined(__clang__)
# if _MSC_VER > 1200
# pragma fp_contract(off)
# endif
# elif defined(__GNUC__) && !defined(__clang__)
# pragma GCC optimize("fp-contract=off")
# else
# pragma STDC FP_CONTRACT OFF
# endif
# endif
# ifdef _MSC_VER
# define STBIR__UNUSED(v) (void)(v)
# else
# define STBIR__UNUSED(v) (void)sizeof(v)
# endif
# define STBIR__ARRAY_SIZE(a) (sizeof((a)) / sizeof((a)[0]))
# ifndef STBIR_DEFAULT_FILTER_UPSAMPLE
# define STBIR_DEFAULT_FILTER_UPSAMPLE STBIR_FILTER_CATMULLROM
# endif
# ifndef STBIR_DEFAULT_FILTER_DOWNSAMPLE
# define STBIR_DEFAULT_FILTER_DOWNSAMPLE STBIR_FILTER_MITCHELL
# endif
# ifndef STBIR__HEADER_FILENAME
# define STBIR__HEADER_FILENAME "stb_image_resize2.h"
# endif
// the internal pixel layout enums are in a different order, so we can easily do range comparisons of types
// the public pixel layout is ordered in a way that if you cast num_channels (1-4) to the enum, you get something sensible
typedef enum
{
STBIRI_1CHANNEL = 0 ,
STBIRI_2CHANNEL = 1 ,
STBIRI_RGB = 2 ,
STBIRI_BGR = 3 ,
STBIRI_4CHANNEL = 4 ,
STBIRI_RGBA = 5 ,
STBIRI_BGRA = 6 ,
STBIRI_ARGB = 7 ,
STBIRI_ABGR = 8 ,
STBIRI_RA = 9 ,
STBIRI_AR = 10 ,
STBIRI_RGBA_PM = 11 ,
STBIRI_BGRA_PM = 12 ,
STBIRI_ARGB_PM = 13 ,
STBIRI_ABGR_PM = 14 ,
STBIRI_RA_PM = 15 ,
STBIRI_AR_PM = 16 ,
} stbir_internal_pixel_layout ;
// define the public pixel layouts to not compile inside the implementation (to avoid accidental use)
# define STBIR_BGR bad_dont_use_in_implementation
# define STBIR_1CHANNEL STBIR_BGR
# define STBIR_2CHANNEL STBIR_BGR
# define STBIR_RGB STBIR_BGR
# define STBIR_RGBA STBIR_BGR
# define STBIR_4CHANNEL STBIR_BGR
# define STBIR_BGRA STBIR_BGR
# define STBIR_ARGB STBIR_BGR
# define STBIR_ABGR STBIR_BGR
# define STBIR_RA STBIR_BGR
# define STBIR_AR STBIR_BGR
# define STBIR_RGBA_PM STBIR_BGR
# define STBIR_BGRA_PM STBIR_BGR
# define STBIR_ARGB_PM STBIR_BGR
# define STBIR_ABGR_PM STBIR_BGR
# define STBIR_RA_PM STBIR_BGR
# define STBIR_AR_PM STBIR_BGR
// must match stbir_datatype
static unsigned char stbir__type_size [ ] = {
1 , 1 , 1 , 2 , 4 , 2 // STBIR_TYPE_UINT8,STBIR_TYPE_UINT8_SRGB,STBIR_TYPE_UINT8_SRGB_ALPHA,STBIR_TYPE_UINT16,STBIR_TYPE_FLOAT,STBIR_TYPE_HALF_FLOAT
} ;
// When gathering, the contributors are which source pixels contribute.
// When scattering, the contributors are which destination pixels are contributed to.
typedef struct
{
int n0 ; // First contributing pixel
int n1 ; // Last contributing pixel
} stbir__contributors ;
typedef struct
{
int lowest ; // First sample index for whole filter
int highest ; // Last sample index for whole filter
int widest ; // widest single set of samples for an output
} stbir__filter_extent_info ;
typedef struct
{
int n0 ; // First pixel of decode buffer to write to
int n1 ; // Last pixel of decode that will be written to
int pixel_offset_for_input ; // Pixel offset into input_scanline
} stbir__span ;
typedef struct stbir__scale_info
{
int input_full_size ;
int output_sub_size ;
float scale ;
float inv_scale ;
float pixel_shift ; // starting shift in output pixel space (in pixels)
int scale_is_rational ;
stbir_uint32 scale_numerator , scale_denominator ;
} stbir__scale_info ;
typedef struct
{
stbir__contributors * contributors ;
float * coefficients ;
stbir__contributors * gather_prescatter_contributors ;
float * gather_prescatter_coefficients ;
stbir__scale_info scale_info ;
float support ;
stbir_filter filter_enum ;
stbir__kernel_callback * filter_kernel ;
stbir__support_callback * filter_support ;
stbir_edge edge ;
int coefficient_width ;
int filter_pixel_width ;
int filter_pixel_margin ;
int num_contributors ;
int contributors_size ;
int coefficients_size ;
stbir__filter_extent_info extent_info ;
int is_gather ; // 0 = scatter, 1 = gather with scale >= 1, 2 = gather with scale < 1
int gather_prescatter_num_contributors ;
int gather_prescatter_coefficient_width ;
int gather_prescatter_contributors_size ;
int gather_prescatter_coefficients_size ;
} stbir__sampler ;
typedef struct
{
stbir__contributors conservative ;
int edge_sizes [ 2 ] ; // this can be less than filter_pixel_margin, if the filter and scaling falls off
stbir__span spans [ 2 ] ; // can be two spans, if doing input subrect with clamp mode WRAP
} stbir__extents ;
typedef struct
{
# ifdef STBIR_PROFILE
union
{
struct { stbir_uint64 total , looping , vertical , horizontal , decode , encode , alpha , unalpha ; } named ;
stbir_uint64 array [ 8 ] ;
} profile ;
stbir_uint64 * current_zone_excluded_ptr ;
# endif
float * decode_buffer ;
int ring_buffer_first_scanline ;
int ring_buffer_last_scanline ;
int ring_buffer_begin_index ; // first_scanline is at this index in the ring buffer
int start_output_y , end_output_y ;
int start_input_y , end_input_y ; // used in scatter only
# ifdef STBIR__SEPARATE_ALLOCATIONS
float * * ring_buffers ; // one pointer for each ring buffer
# else
float * ring_buffer ; // one big buffer that we index into
# endif
float * vertical_buffer ;
char no_cache_straddle [ 64 ] ;
} stbir__per_split_info ;
typedef void stbir__decode_pixels_func ( float * decode , int width_times_channels , void const * input ) ;
typedef void stbir__alpha_weight_func ( float * decode_buffer , int width_times_channels ) ;
typedef void stbir__horizontal_gather_channels_func ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer ,
stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width ) ;
typedef void stbir__alpha_unweight_func ( float * encode_buffer , int width_times_channels ) ;
typedef void stbir__encode_pixels_func ( void * output , int width_times_channels , float const * encode ) ;
struct stbir__info
{
# ifdef STBIR_PROFILE
union
{
struct { stbir_uint64 total , build , alloc , horizontal , vertical , cleanup , pivot ; } named ;
stbir_uint64 array [ 7 ] ;
} profile ;
stbir_uint64 * current_zone_excluded_ptr ;
# endif
stbir__sampler horizontal ;
stbir__sampler vertical ;
void const * input_data ;
void * output_data ;
int input_stride_bytes ;
int output_stride_bytes ;
int ring_buffer_length_bytes ; // The length of an individual entry in the ring buffer. The total number of ring buffers is stbir__get_filter_pixel_width(filter)
int ring_buffer_num_entries ; // Total number of entries in the ring buffer.
stbir_datatype input_type ;
stbir_datatype output_type ;
stbir_input_callback * in_pixels_cb ;
void * user_data ;
stbir_output_callback * out_pixels_cb ;
stbir__extents scanline_extents ;
void * alloced_mem ;
stbir__per_split_info * split_info ; // by default 1, but there will be N of these allocated based on the thread init you did
stbir__decode_pixels_func * decode_pixels ;
stbir__alpha_weight_func * alpha_weight ;
stbir__horizontal_gather_channels_func * horizontal_gather_channels ;
stbir__alpha_unweight_func * alpha_unweight ;
stbir__encode_pixels_func * encode_pixels ;
int alloced_total ;
int splits ; // count of splits
stbir_internal_pixel_layout input_pixel_layout_internal ;
stbir_internal_pixel_layout output_pixel_layout_internal ;
int input_color_and_type ;
int offset_x , offset_y ; // offset within output_data
int vertical_first ;
int channels ;
int effective_channels ; // same as channels, except on RGBA/ARGB (7), or XA/AX (3)
int alloc_ring_buffer_num_entries ; // Number of entries in the ring buffer that will be allocated
} ;
# define stbir__max_uint8_as_float 255.0f
# define stbir__max_uint16_as_float 65535.0f
# define stbir__max_uint8_as_float_inverted (1.0f / 255.0f)
# define stbir__max_uint16_as_float_inverted (1.0f / 65535.0f)
# define stbir__small_float ((float)1 / (1 << 20) / (1 << 20) / (1 << 20) / (1 << 20) / (1 << 20) / (1 << 20))
// min/max friendly
# define STBIR_CLAMP(x, xmin, xmax) do { \
if ( ( x ) < ( xmin ) ) ( x ) = ( xmin ) ; \
if ( ( x ) > ( xmax ) ) ( x ) = ( xmax ) ; \
} while ( 0 )
static stbir__inline int stbir__min ( int a , int b )
{
return a < b ? a : b ;
}
static stbir__inline int stbir__max ( int a , int b )
{
return a > b ? a : b ;
}
static float stbir__srgb_uchar_to_linear_float [ 256 ] = {
0.000000f , 0.000304f , 0.000607f , 0.000911f , 0.001214f , 0.001518f , 0.001821f , 0.002125f , 0.002428f , 0.002732f , 0.003035f ,
0.003347f , 0.003677f , 0.004025f , 0.004391f , 0.004777f , 0.005182f , 0.005605f , 0.006049f , 0.006512f , 0.006995f , 0.007499f ,
0.008023f , 0.008568f , 0.009134f , 0.009721f , 0.010330f , 0.010960f , 0.011612f , 0.012286f , 0.012983f , 0.013702f , 0.014444f ,
0.015209f , 0.015996f , 0.016807f , 0.017642f , 0.018500f , 0.019382f , 0.020289f , 0.021219f , 0.022174f , 0.023153f , 0.024158f ,
0.025187f , 0.026241f , 0.027321f , 0.028426f , 0.029557f , 0.030713f , 0.031896f , 0.033105f , 0.034340f , 0.035601f , 0.036889f ,
0.038204f , 0.039546f , 0.040915f , 0.042311f , 0.043735f , 0.045186f , 0.046665f , 0.048172f , 0.049707f , 0.051269f , 0.052861f ,
0.054480f , 0.056128f , 0.057805f , 0.059511f , 0.061246f , 0.063010f , 0.064803f , 0.066626f , 0.068478f , 0.070360f , 0.072272f ,
0.074214f , 0.076185f , 0.078187f , 0.080220f , 0.082283f , 0.084376f , 0.086500f , 0.088656f , 0.090842f , 0.093059f , 0.095307f ,
0.097587f , 0.099899f , 0.102242f , 0.104616f , 0.107023f , 0.109462f , 0.111932f , 0.114435f , 0.116971f , 0.119538f , 0.122139f ,
0.124772f , 0.127438f , 0.130136f , 0.132868f , 0.135633f , 0.138432f , 0.141263f , 0.144128f , 0.147027f , 0.149960f , 0.152926f ,
0.155926f , 0.158961f , 0.162029f , 0.165132f , 0.168269f , 0.171441f , 0.174647f , 0.177888f , 0.181164f , 0.184475f , 0.187821f ,
0.191202f , 0.194618f , 0.198069f , 0.201556f , 0.205079f , 0.208637f , 0.212231f , 0.215861f , 0.219526f , 0.223228f , 0.226966f ,
0.230740f , 0.234551f , 0.238398f , 0.242281f , 0.246201f , 0.250158f , 0.254152f , 0.258183f , 0.262251f , 0.266356f , 0.270498f ,
0.274677f , 0.278894f , 0.283149f , 0.287441f , 0.291771f , 0.296138f , 0.300544f , 0.304987f , 0.309469f , 0.313989f , 0.318547f ,
0.323143f , 0.327778f , 0.332452f , 0.337164f , 0.341914f , 0.346704f , 0.351533f , 0.356400f , 0.361307f , 0.366253f , 0.371238f ,
0.376262f , 0.381326f , 0.386430f , 0.391573f , 0.396755f , 0.401978f , 0.407240f , 0.412543f , 0.417885f , 0.423268f , 0.428691f ,
0.434154f , 0.439657f , 0.445201f , 0.450786f , 0.456411f , 0.462077f , 0.467784f , 0.473532f , 0.479320f , 0.485150f , 0.491021f ,
0.496933f , 0.502887f , 0.508881f , 0.514918f , 0.520996f , 0.527115f , 0.533276f , 0.539480f , 0.545725f , 0.552011f , 0.558340f ,
0.564712f , 0.571125f , 0.577581f , 0.584078f , 0.590619f , 0.597202f , 0.603827f , 0.610496f , 0.617207f , 0.623960f , 0.630757f ,
0.637597f , 0.644480f , 0.651406f , 0.658375f , 0.665387f , 0.672443f , 0.679543f , 0.686685f , 0.693872f , 0.701102f , 0.708376f ,
0.715694f , 0.723055f , 0.730461f , 0.737911f , 0.745404f , 0.752942f , 0.760525f , 0.768151f , 0.775822f , 0.783538f , 0.791298f ,
0.799103f , 0.806952f , 0.814847f , 0.822786f , 0.830770f , 0.838799f , 0.846873f , 0.854993f , 0.863157f , 0.871367f , 0.879622f ,
0.887923f , 0.896269f , 0.904661f , 0.913099f , 0.921582f , 0.930111f , 0.938686f , 0.947307f , 0.955974f , 0.964686f , 0.973445f ,
0.982251f , 0.991102f , 1.0f
} ;
typedef union
{
unsigned int u ;
float f ;
} stbir__FP32 ;
// From https://gist.github.com/rygorous/2203834
static const stbir_uint32 fp32_to_srgb8_tab4 [ 104 ] = {
0x0073000d , 0x007a000d , 0x0080000d , 0x0087000d , 0x008d000d , 0x0094000d , 0x009a000d , 0x00a1000d ,
0x00a7001a , 0x00b4001a , 0x00c1001a , 0x00ce001a , 0x00da001a , 0x00e7001a , 0x00f4001a , 0x0101001a ,
0x010e0033 , 0x01280033 , 0x01410033 , 0x015b0033 , 0x01750033 , 0x018f0033 , 0x01a80033 , 0x01c20033 ,
0x01dc0067 , 0x020f0067 , 0x02430067 , 0x02760067 , 0x02aa0067 , 0x02dd0067 , 0x03110067 , 0x03440067 ,
0x037800ce , 0x03df00ce , 0x044600ce , 0x04ad00ce , 0x051400ce , 0x057b00c5 , 0x05dd00bc , 0x063b00b5 ,
0x06970158 , 0x07420142 , 0x07e30130 , 0x087b0120 , 0x090b0112 , 0x09940106 , 0x0a1700fc , 0x0a9500f2 ,
0x0b0f01cb , 0x0bf401ae , 0x0ccb0195 , 0x0d950180 , 0x0e56016e , 0x0f0d015e , 0x0fbc0150 , 0x10630143 ,
0x11070264 , 0x1238023e , 0x1357021d , 0x14660201 , 0x156601e9 , 0x165a01d3 , 0x174401c0 , 0x182401af ,
0x18fe0331 , 0x1a9602fe , 0x1c1502d2 , 0x1d7e02ad , 0x1ed4028d , 0x201a0270 , 0x21520256 , 0x227d0240 ,
0x239f0443 , 0x25c003fe , 0x27bf03c4 , 0x29a10392 , 0x2b6a0367 , 0x2d1d0341 , 0x2ebe031f , 0x304d0300 ,
0x31d105b0 , 0x34a80555 , 0x37520507 , 0x39d504c5 , 0x3c37048b , 0x3e7c0458 , 0x40a8042a , 0x42bd0401 ,
0x44c20798 , 0x488e071e , 0x4c1c06b6 , 0x4f76065d , 0x52a50610 , 0x55ac05cc , 0x5892058f , 0x5b590559 ,
0x5e0c0a23 , 0x631c0980 , 0x67db08f6 , 0x6c55087f , 0x70940818 , 0x74a007bd , 0x787d076c , 0x7c330723 ,
} ;
static stbir__inline stbir_uint8 stbir__linear_to_srgb_uchar ( float in )
{
static const stbir__FP32 almostone = { 0x3f7fffff } ; // 1-eps
static const stbir__FP32 minval = { ( 127 - 13 ) < < 23 } ;
stbir_uint32 tab , bias , scale , t ;
stbir__FP32 f ;
// Clamp to [2^(-13), 1-eps]; these two values map to 0 and 1, respectively.
// The tests are carefully written so that NaNs map to 0, same as in the reference
// implementation.
if ( ! ( in > minval . f ) ) // written this way to catch NaNs
return 0 ;
if ( in > almostone . f )
return 255 ;
// Do the table lookup and unpack bias, scale
f . f = in ;
tab = fp32_to_srgb8_tab4 [ ( f . u - minval . u ) > > 20 ] ;
bias = ( tab > > 16 ) < < 9 ;
scale = tab & 0xffff ;
// Grab next-highest mantissa bits and perform linear interpolation
t = ( f . u > > 12 ) & 0xff ;
return ( unsigned char ) ( ( bias + scale * t ) > > 16 ) ;
}
# ifndef STBIR_FORCE_GATHER_FILTER_SCANLINES_AMOUNT
# define STBIR_FORCE_GATHER_FILTER_SCANLINES_AMOUNT 32 // when downsampling and <= 32 scanlines of buffering, use gather. gather used down to 1/8th scaling for 25% win.
# endif
2023-12-14 19:09:00 +08:00
# ifndef STBIR_FORCE_MINIMUM_SCANLINES_FOR_SPLITS
# define STBIR_FORCE_MINIMUM_SCANLINES_FOR_SPLITS 4 // when threading, what is the minimum number of scanlines for a split?
# endif
2023-10-10 08:23:04 +08:00
// restrict pointers for the output pointers
# if defined( _MSC_VER ) && !defined(__clang__)
# define STBIR_STREAMOUT_PTR( star ) star __restrict
# define STBIR_NO_UNROLL( ptr ) __assume(ptr) // this oddly keeps msvc from unrolling a loop
# elif defined( __clang__ )
# define STBIR_STREAMOUT_PTR( star ) star __restrict__
2023-10-12 08:28:56 +08:00
# define STBIR_NO_UNROLL( ptr ) __asm__ (""::"r"(ptr))
2023-10-10 08:23:04 +08:00
# elif defined( __GNUC__ )
# define STBIR_STREAMOUT_PTR( star ) star __restrict__
2023-10-12 08:28:56 +08:00
# define STBIR_NO_UNROLL( ptr ) __asm__ (""::"r"(ptr))
2023-10-10 08:23:04 +08:00
# else
# define STBIR_STREAMOUT_PTR( star ) star
# define STBIR_NO_UNROLL( ptr )
# endif
# ifdef STBIR_NO_SIMD // force simd off for whatever reason
// force simd off overrides everything else, so clear it all
# ifdef STBIR_SSE2
# undef STBIR_SSE2
# endif
# ifdef STBIR_AVX
# undef STBIR_AVX
# endif
# ifdef STBIR_NEON
# undef STBIR_NEON
# endif
# ifdef STBIR_AVX2
# undef STBIR_AVX2
# endif
# ifdef STBIR_FP16C
# undef STBIR_FP16C
# endif
# ifdef STBIR_WASM
# undef STBIR_WASM
# endif
# ifdef STBIR_SIMD
# undef STBIR_SIMD
# endif
# else // STBIR_SIMD
# ifdef STBIR_SSE2
# include <emmintrin.h>
# define stbir__simdf __m128
# define stbir__simdi __m128i
# define stbir_simdi_castf( reg ) _mm_castps_si128(reg)
# define stbir_simdf_casti( reg ) _mm_castsi128_ps(reg)
# define stbir__simdf_load( reg, ptr ) (reg) = _mm_loadu_ps( (float const*)(ptr) )
# define stbir__simdi_load( reg, ptr ) (reg) = _mm_loadu_si128 ( (stbir__simdi const*)(ptr) )
# define stbir__simdf_load1( out, ptr ) (out) = _mm_load_ss( (float const*)(ptr) ) // top values can be random (not denormal or nan for perf)
# define stbir__simdi_load1( out, ptr ) (out) = _mm_castps_si128( _mm_load_ss( (float const*)(ptr) ))
# define stbir__simdf_load1z( out, ptr ) (out) = _mm_load_ss( (float const*)(ptr) ) // top values must be zero
# define stbir__simdf_frep4( fvar ) _mm_set_ps1( fvar )
# define stbir__simdf_load1frep4( out, fvar ) (out) = _mm_set_ps1( fvar )
# define stbir__simdf_load2( out, ptr ) (out) = _mm_castsi128_ps( _mm_loadl_epi64( (__m128i*)(ptr)) ) // top values can be random (not denormal or nan for perf)
# define stbir__simdf_load2z( out, ptr ) (out) = _mm_castsi128_ps( _mm_loadl_epi64( (__m128i*)(ptr)) ) // top values must be zero
# define stbir__simdf_load2hmerge( out, reg, ptr ) (out) = _mm_castpd_ps(_mm_loadh_pd( _mm_castps_pd(reg), (double*)(ptr) ))
# define stbir__simdf_zeroP() _mm_setzero_ps()
# define stbir__simdf_zero( reg ) (reg) = _mm_setzero_ps()
# define stbir__simdf_store( ptr, reg ) _mm_storeu_ps( (float*)(ptr), reg )
# define stbir__simdf_store1( ptr, reg ) _mm_store_ss( (float*)(ptr), reg )
# define stbir__simdf_store2( ptr, reg ) _mm_storel_epi64( (__m128i*)(ptr), _mm_castps_si128(reg) )
# define stbir__simdf_store2h( ptr, reg ) _mm_storeh_pd( (double*)(ptr), _mm_castps_pd(reg) )
# define stbir__simdi_store( ptr, reg ) _mm_storeu_si128( (__m128i*)(ptr), reg )
# define stbir__simdi_store1( ptr, reg ) _mm_store_ss( (float*)(ptr), _mm_castsi128_ps(reg) )
# define stbir__simdi_store2( ptr, reg ) _mm_storel_epi64( (__m128i*)(ptr), (reg) )
# define stbir__prefetch( ptr ) _mm_prefetch((char*)(ptr), _MM_HINT_T0 )
# define stbir__simdi_expand_u8_to_u32(out0,out1,out2,out3,ireg) \
{ \
stbir__simdi zero = _mm_setzero_si128 ( ) ; \
out2 = _mm_unpacklo_epi8 ( ireg , zero ) ; \
out3 = _mm_unpackhi_epi8 ( ireg , zero ) ; \
out0 = _mm_unpacklo_epi16 ( out2 , zero ) ; \
out1 = _mm_unpackhi_epi16 ( out2 , zero ) ; \
out2 = _mm_unpacklo_epi16 ( out3 , zero ) ; \
out3 = _mm_unpackhi_epi16 ( out3 , zero ) ; \
}
# define stbir__simdi_expand_u8_to_1u32(out,ireg) \
{ \
stbir__simdi zero = _mm_setzero_si128 ( ) ; \
out = _mm_unpacklo_epi8 ( ireg , zero ) ; \
out = _mm_unpacklo_epi16 ( out , zero ) ; \
}
# define stbir__simdi_expand_u16_to_u32(out0,out1,ireg) \
{ \
stbir__simdi zero = _mm_setzero_si128 ( ) ; \
out0 = _mm_unpacklo_epi16 ( ireg , zero ) ; \
out1 = _mm_unpackhi_epi16 ( ireg , zero ) ; \
}
# define stbir__simdf_convert_float_to_i32( i, f ) (i) = _mm_cvttps_epi32(f)
# define stbir__simdf_convert_float_to_int( f ) _mm_cvtt_ss2si(f)
# define stbir__simdf_convert_float_to_uint8( f ) ((unsigned char)_mm_cvtsi128_si32(_mm_cvttps_epi32(_mm_max_ps(_mm_min_ps(f,STBIR__CONSTF(STBIR_max_uint8_as_float)),_mm_setzero_ps()))))
# define stbir__simdf_convert_float_to_short( f ) ((unsigned short)_mm_cvtsi128_si32(_mm_cvttps_epi32(_mm_max_ps(_mm_min_ps(f,STBIR__CONSTF(STBIR_max_uint16_as_float)),_mm_setzero_ps()))))
# define stbir__simdi_to_int( i ) _mm_cvtsi128_si32(i)
# define stbir__simdi_convert_i32_to_float(out, ireg) (out) = _mm_cvtepi32_ps( ireg )
# define stbir__simdf_add( out, reg0, reg1 ) (out) = _mm_add_ps( reg0, reg1 )
# define stbir__simdf_mult( out, reg0, reg1 ) (out) = _mm_mul_ps( reg0, reg1 )
# define stbir__simdf_mult_mem( out, reg, ptr ) (out) = _mm_mul_ps( reg, _mm_loadu_ps( (float const*)(ptr) ) )
# define stbir__simdf_mult1_mem( out, reg, ptr ) (out) = _mm_mul_ss( reg, _mm_load_ss( (float const*)(ptr) ) )
# define stbir__simdf_add_mem( out, reg, ptr ) (out) = _mm_add_ps( reg, _mm_loadu_ps( (float const*)(ptr) ) )
# define stbir__simdf_add1_mem( out, reg, ptr ) (out) = _mm_add_ss( reg, _mm_load_ss( (float const*)(ptr) ) )
# ifdef STBIR_USE_FMA // not on by default to maintain bit identical simd to non-simd
# include <immintrin.h>
# define stbir__simdf_madd( out, add, mul1, mul2 ) (out) = _mm_fmadd_ps( mul1, mul2, add )
# define stbir__simdf_madd1( out, add, mul1, mul2 ) (out) = _mm_fmadd_ss( mul1, mul2, add )
# define stbir__simdf_madd_mem( out, add, mul, ptr ) (out) = _mm_fmadd_ps( mul, _mm_loadu_ps( (float const*)(ptr) ), add )
# define stbir__simdf_madd1_mem( out, add, mul, ptr ) (out) = _mm_fmadd_ss( mul, _mm_load_ss( (float const*)(ptr) ), add )
# else
# define stbir__simdf_madd( out, add, mul1, mul2 ) (out) = _mm_add_ps( add, _mm_mul_ps( mul1, mul2 ) )
# define stbir__simdf_madd1( out, add, mul1, mul2 ) (out) = _mm_add_ss( add, _mm_mul_ss( mul1, mul2 ) )
# define stbir__simdf_madd_mem( out, add, mul, ptr ) (out) = _mm_add_ps( add, _mm_mul_ps( mul, _mm_loadu_ps( (float const*)(ptr) ) ) )
# define stbir__simdf_madd1_mem( out, add, mul, ptr ) (out) = _mm_add_ss( add, _mm_mul_ss( mul, _mm_load_ss( (float const*)(ptr) ) ) )
# endif
# define stbir__simdf_add1( out, reg0, reg1 ) (out) = _mm_add_ss( reg0, reg1 )
# define stbir__simdf_mult1( out, reg0, reg1 ) (out) = _mm_mul_ss( reg0, reg1 )
# define stbir__simdf_and( out, reg0, reg1 ) (out) = _mm_and_ps( reg0, reg1 )
# define stbir__simdf_or( out, reg0, reg1 ) (out) = _mm_or_ps( reg0, reg1 )
# define stbir__simdf_min( out, reg0, reg1 ) (out) = _mm_min_ps( reg0, reg1 )
# define stbir__simdf_max( out, reg0, reg1 ) (out) = _mm_max_ps( reg0, reg1 )
# define stbir__simdf_min1( out, reg0, reg1 ) (out) = _mm_min_ss( reg0, reg1 )
# define stbir__simdf_max1( out, reg0, reg1 ) (out) = _mm_max_ss( reg0, reg1 )
# define stbir__simdf_0123ABCDto3ABx( out, reg0, reg1 ) (out)=_mm_castsi128_ps( _mm_shuffle_epi32( _mm_castps_si128( _mm_shuffle_ps( reg1,reg0, (0<<0) + (1<<2) + (2<<4) + (3<<6) )), (3<<0) + (0<<2) + (1<<4) + (2<<6) ) )
# define stbir__simdf_0123ABCDto23Ax( out, reg0, reg1 ) (out)=_mm_castsi128_ps( _mm_shuffle_epi32( _mm_castps_si128( _mm_shuffle_ps( reg1,reg0, (0<<0) + (1<<2) + (2<<4) + (3<<6) )), (2<<0) + (3<<2) + (0<<4) + (1<<6) ) )
static const stbir__simdf STBIR_zeroones = { 0.0f , 1.0f , 0.0f , 1.0f } ;
static const stbir__simdf STBIR_onezeros = { 1.0f , 0.0f , 1.0f , 0.0f } ;
# define stbir__simdf_aaa1( out, alp, ones ) (out)=_mm_castsi128_ps( _mm_shuffle_epi32( _mm_castps_si128( _mm_movehl_ps( ones, alp ) ), (1<<0) + (1<<2) + (1<<4) + (2<<6) ) )
# define stbir__simdf_1aaa( out, alp, ones ) (out)=_mm_castsi128_ps( _mm_shuffle_epi32( _mm_castps_si128( _mm_movelh_ps( ones, alp ) ), (0<<0) + (2<<2) + (2<<4) + (2<<6) ) )
# define stbir__simdf_a1a1( out, alp, ones) (out) = _mm_or_ps( _mm_castsi128_ps( _mm_srli_epi64( _mm_castps_si128(alp), 32 ) ), STBIR_zeroones )
# define stbir__simdf_1a1a( out, alp, ones) (out) = _mm_or_ps( _mm_castsi128_ps( _mm_slli_epi64( _mm_castps_si128(alp), 32 ) ), STBIR_onezeros )
# define stbir__simdf_swiz( reg, one, two, three, four ) _mm_castsi128_ps( _mm_shuffle_epi32( _mm_castps_si128( reg ), (one<<0) + (two<<2) + (three<<4) + (four<<6) ) )
# define stbir__simdi_and( out, reg0, reg1 ) (out) = _mm_and_si128( reg0, reg1 )
# define stbir__simdi_or( out, reg0, reg1 ) (out) = _mm_or_si128( reg0, reg1 )
# define stbir__simdi_16madd( out, reg0, reg1 ) (out) = _mm_madd_epi16( reg0, reg1 )
# define stbir__simdf_pack_to_8bytes(out,aa,bb) \
{ \
stbir__simdf af , bf ; \
stbir__simdi a , b ; \
af = _mm_min_ps ( aa , STBIR_max_uint8_as_float ) ; \
bf = _mm_min_ps ( bb , STBIR_max_uint8_as_float ) ; \
af = _mm_max_ps ( af , _mm_setzero_ps ( ) ) ; \
bf = _mm_max_ps ( bf , _mm_setzero_ps ( ) ) ; \
a = _mm_cvttps_epi32 ( af ) ; \
b = _mm_cvttps_epi32 ( bf ) ; \
a = _mm_packs_epi32 ( a , b ) ; \
out = _mm_packus_epi16 ( a , a ) ; \
}
# define stbir__simdf_load4_transposed( o0, o1, o2, o3, ptr ) \
stbir__simdf_load ( o0 , ( ptr ) ) ; \
stbir__simdf_load ( o1 , ( ptr ) + 4 ) ; \
stbir__simdf_load ( o2 , ( ptr ) + 8 ) ; \
stbir__simdf_load ( o3 , ( ptr ) + 12 ) ; \
{ \
__m128 tmp0 , tmp1 , tmp2 , tmp3 ; \
tmp0 = _mm_unpacklo_ps ( o0 , o1 ) ; \
tmp2 = _mm_unpacklo_ps ( o2 , o3 ) ; \
tmp1 = _mm_unpackhi_ps ( o0 , o1 ) ; \
tmp3 = _mm_unpackhi_ps ( o2 , o3 ) ; \
o0 = _mm_movelh_ps ( tmp0 , tmp2 ) ; \
o1 = _mm_movehl_ps ( tmp2 , tmp0 ) ; \
o2 = _mm_movelh_ps ( tmp1 , tmp3 ) ; \
o3 = _mm_movehl_ps ( tmp3 , tmp1 ) ; \
}
# define stbir__interleave_pack_and_store_16_u8( ptr, r0, r1, r2, r3 ) \
r0 = _mm_packs_epi32 ( r0 , r1 ) ; \
r2 = _mm_packs_epi32 ( r2 , r3 ) ; \
r1 = _mm_unpacklo_epi16 ( r0 , r2 ) ; \
r3 = _mm_unpackhi_epi16 ( r0 , r2 ) ; \
r0 = _mm_unpacklo_epi16 ( r1 , r3 ) ; \
r2 = _mm_unpackhi_epi16 ( r1 , r3 ) ; \
r0 = _mm_packus_epi16 ( r0 , r2 ) ; \
stbir__simdi_store ( ptr , r0 ) ; \
# define stbir__simdi_32shr( out, reg, imm ) out = _mm_srli_epi32( reg, imm )
# if defined(_MSC_VER) && !defined(__clang__)
// msvc inits with 8 bytes
# define STBIR__CONST_32_TO_8( v ) (char)(unsigned char)((v)&255),(char)(unsigned char)(((v)>>8)&255),(char)(unsigned char)(((v)>>16)&255),(char)(unsigned char)(((v)>>24)&255)
# define STBIR__CONST_4_32i( v ) STBIR__CONST_32_TO_8( v ), STBIR__CONST_32_TO_8( v ), STBIR__CONST_32_TO_8( v ), STBIR__CONST_32_TO_8( v )
# define STBIR__CONST_4d_32i( v0, v1, v2, v3 ) STBIR__CONST_32_TO_8( v0 ), STBIR__CONST_32_TO_8( v1 ), STBIR__CONST_32_TO_8( v2 ), STBIR__CONST_32_TO_8( v3 )
# else
// everything else inits with long long's
# define STBIR__CONST_4_32i( v ) (long long)((((stbir_uint64)(stbir_uint32)(v))<<32)|((stbir_uint64)(stbir_uint32)(v))),(long long)((((stbir_uint64)(stbir_uint32)(v))<<32)|((stbir_uint64)(stbir_uint32)(v)))
# define STBIR__CONST_4d_32i( v0, v1, v2, v3 ) (long long)((((stbir_uint64)(stbir_uint32)(v1))<<32)|((stbir_uint64)(stbir_uint32)(v0))),(long long)((((stbir_uint64)(stbir_uint32)(v3))<<32)|((stbir_uint64)(stbir_uint32)(v2)))
# endif
# define STBIR__SIMDF_CONST(var, x) stbir__simdf var = { x, x, x, x }
# define STBIR__SIMDI_CONST(var, x) stbir__simdi var = { STBIR__CONST_4_32i(x) }
# define STBIR__CONSTF(var) (var)
# define STBIR__CONSTI(var) (var)
# if defined(STBIR_AVX) || defined(__SSE4_1__)
# include <smmintrin.h>
# define stbir__simdf_pack_to_8words(out,reg0,reg1) out = _mm_packus_epi32(_mm_cvttps_epi32(_mm_max_ps(_mm_min_ps(reg0,STBIR__CONSTF(STBIR_max_uint16_as_float)),_mm_setzero_ps())), _mm_cvttps_epi32(_mm_max_ps(_mm_min_ps(reg1,STBIR__CONSTF(STBIR_max_uint16_as_float)),_mm_setzero_ps())))
# else
STBIR__SIMDI_CONST ( stbir__s32_32768 , 32768 ) ;
STBIR__SIMDI_CONST ( stbir__s16_32768 , ( ( 32768 < < 16 ) | 32768 ) ) ;
# define stbir__simdf_pack_to_8words(out,reg0,reg1) \
{ \
stbir__simdi tmp0 , tmp1 ; \
tmp0 = _mm_cvttps_epi32 ( _mm_max_ps ( _mm_min_ps ( reg0 , STBIR__CONSTF ( STBIR_max_uint16_as_float ) ) , _mm_setzero_ps ( ) ) ) ; \
tmp1 = _mm_cvttps_epi32 ( _mm_max_ps ( _mm_min_ps ( reg1 , STBIR__CONSTF ( STBIR_max_uint16_as_float ) ) , _mm_setzero_ps ( ) ) ) ; \
tmp0 = _mm_sub_epi32 ( tmp0 , stbir__s32_32768 ) ; \
tmp1 = _mm_sub_epi32 ( tmp1 , stbir__s32_32768 ) ; \
out = _mm_packs_epi32 ( tmp0 , tmp1 ) ; \
out = _mm_sub_epi16 ( out , stbir__s16_32768 ) ; \
}
# endif
# define STBIR_SIMD
// if we detect AVX, set the simd8 defines
# ifdef STBIR_AVX
# include <immintrin.h>
# define STBIR_SIMD8
# define stbir__simdf8 __m256
# define stbir__simdi8 __m256i
# define stbir__simdf8_load( out, ptr ) (out) = _mm256_loadu_ps( (float const *)(ptr) )
# define stbir__simdi8_load( out, ptr ) (out) = _mm256_loadu_si256( (__m256i const *)(ptr) )
# define stbir__simdf8_mult( out, a, b ) (out) = _mm256_mul_ps( (a), (b) )
# define stbir__simdf8_store( ptr, out ) _mm256_storeu_ps( (float*)(ptr), out )
# define stbir__simdi8_store( ptr, reg ) _mm256_storeu_si256( (__m256i*)(ptr), reg )
# define stbir__simdf8_frep8( fval ) _mm256_set1_ps( fval )
# define stbir__simdf8_min( out, reg0, reg1 ) (out) = _mm256_min_ps( reg0, reg1 )
# define stbir__simdf8_max( out, reg0, reg1 ) (out) = _mm256_max_ps( reg0, reg1 )
# define stbir__simdf8_add4halves( out, bot4, top8 ) (out) = _mm_add_ps( bot4, _mm256_extractf128_ps( top8, 1 ) )
# define stbir__simdf8_mult_mem( out, reg, ptr ) (out) = _mm256_mul_ps( reg, _mm256_loadu_ps( (float const*)(ptr) ) )
# define stbir__simdf8_add_mem( out, reg, ptr ) (out) = _mm256_add_ps( reg, _mm256_loadu_ps( (float const*)(ptr) ) )
# define stbir__simdf8_add( out, a, b ) (out) = _mm256_add_ps( a, b )
# define stbir__simdf8_load1b( out, ptr ) (out) = _mm256_broadcast_ss( ptr )
# define stbir__simdf_load1rep4( out, ptr ) (out) = _mm_broadcast_ss( ptr ) // avx load instruction
# define stbir__simdi8_convert_i32_to_float(out, ireg) (out) = _mm256_cvtepi32_ps( ireg )
# define stbir__simdf8_convert_float_to_i32( i, f ) (i) = _mm256_cvttps_epi32(f)
# define stbir__simdf8_bot4s( out, a, b ) (out) = _mm256_permute2f128_ps(a,b, (0<<0)+(2<<4) )
# define stbir__simdf8_top4s( out, a, b ) (out) = _mm256_permute2f128_ps(a,b, (1<<0)+(3<<4) )
# define stbir__simdf8_gettop4( reg ) _mm256_extractf128_ps(reg,1)
# ifdef STBIR_AVX2
# define stbir__simdi8_expand_u8_to_u32(out0,out1,ireg) \
{ \
stbir__simdi8 a , zero = _mm256_setzero_si256 ( ) ; \
a = _mm256_permute4x64_epi64 ( _mm256_unpacklo_epi8 ( _mm256_permute4x64_epi64 ( _mm256_castsi128_si256 ( ireg ) , ( 0 < < 0 ) + ( 2 < < 2 ) + ( 1 < < 4 ) + ( 3 < < 6 ) ) , zero ) , ( 0 < < 0 ) + ( 2 < < 2 ) + ( 1 < < 4 ) + ( 3 < < 6 ) ) ; \
out0 = _mm256_unpacklo_epi16 ( a , zero ) ; \
out1 = _mm256_unpackhi_epi16 ( a , zero ) ; \
}
# define stbir__simdf8_pack_to_16bytes(out,aa,bb) \
{ \
stbir__simdi8 t ; \
stbir__simdf8 af , bf ; \
stbir__simdi8 a , b ; \
af = _mm256_min_ps ( aa , STBIR_max_uint8_as_floatX ) ; \
bf = _mm256_min_ps ( bb , STBIR_max_uint8_as_floatX ) ; \
af = _mm256_max_ps ( af , _mm256_setzero_ps ( ) ) ; \
bf = _mm256_max_ps ( bf , _mm256_setzero_ps ( ) ) ; \
a = _mm256_cvttps_epi32 ( af ) ; \
b = _mm256_cvttps_epi32 ( bf ) ; \
t = _mm256_permute4x64_epi64 ( _mm256_packs_epi32 ( a , b ) , ( 0 < < 0 ) + ( 2 < < 2 ) + ( 1 < < 4 ) + ( 3 < < 6 ) ) ; \
out = _mm256_castsi256_si128 ( _mm256_permute4x64_epi64 ( _mm256_packus_epi16 ( t , t ) , ( 0 < < 0 ) + ( 2 < < 2 ) + ( 1 < < 4 ) + ( 3 < < 6 ) ) ) ; \
}
# define stbir__simdi8_expand_u16_to_u32(out,ireg) out = _mm256_unpacklo_epi16( _mm256_permute4x64_epi64(_mm256_castsi128_si256(ireg),(0<<0)+(2<<2)+(1<<4)+(3<<6)), _mm256_setzero_si256() );
# define stbir__simdf8_pack_to_16words(out,aa,bb) \
{ \
stbir__simdf8 af , bf ; \
stbir__simdi8 a , b ; \
af = _mm256_min_ps ( aa , STBIR_max_uint16_as_floatX ) ; \
bf = _mm256_min_ps ( bb , STBIR_max_uint16_as_floatX ) ; \
af = _mm256_max_ps ( af , _mm256_setzero_ps ( ) ) ; \
bf = _mm256_max_ps ( bf , _mm256_setzero_ps ( ) ) ; \
a = _mm256_cvttps_epi32 ( af ) ; \
b = _mm256_cvttps_epi32 ( bf ) ; \
( out ) = _mm256_permute4x64_epi64 ( _mm256_packus_epi32 ( a , b ) , ( 0 < < 0 ) + ( 2 < < 2 ) + ( 1 < < 4 ) + ( 3 < < 6 ) ) ; \
}
# else
# define stbir__simdi8_expand_u8_to_u32(out0,out1,ireg) \
{ \
stbir__simdi a , zero = _mm_setzero_si128 ( ) ; \
a = _mm_unpacklo_epi8 ( ireg , zero ) ; \
out0 = _mm256_setr_m128i ( _mm_unpacklo_epi16 ( a , zero ) , _mm_unpackhi_epi16 ( a , zero ) ) ; \
a = _mm_unpackhi_epi8 ( ireg , zero ) ; \
out1 = _mm256_setr_m128i ( _mm_unpacklo_epi16 ( a , zero ) , _mm_unpackhi_epi16 ( a , zero ) ) ; \
}
# define stbir__simdf8_pack_to_16bytes(out,aa,bb) \
{ \
stbir__simdi t ; \
stbir__simdf8 af , bf ; \
stbir__simdi8 a , b ; \
af = _mm256_min_ps ( aa , STBIR_max_uint8_as_floatX ) ; \
bf = _mm256_min_ps ( bb , STBIR_max_uint8_as_floatX ) ; \
af = _mm256_max_ps ( af , _mm256_setzero_ps ( ) ) ; \
bf = _mm256_max_ps ( bf , _mm256_setzero_ps ( ) ) ; \
a = _mm256_cvttps_epi32 ( af ) ; \
b = _mm256_cvttps_epi32 ( bf ) ; \
out = _mm_packs_epi32 ( _mm256_castsi256_si128 ( a ) , _mm256_extractf128_si256 ( a , 1 ) ) ; \
out = _mm_packus_epi16 ( out , out ) ; \
t = _mm_packs_epi32 ( _mm256_castsi256_si128 ( b ) , _mm256_extractf128_si256 ( b , 1 ) ) ; \
t = _mm_packus_epi16 ( t , t ) ; \
out = _mm_castps_si128 ( _mm_shuffle_ps ( _mm_castsi128_ps ( out ) , _mm_castsi128_ps ( t ) , ( 0 < < 0 ) + ( 1 < < 2 ) + ( 0 < < 4 ) + ( 1 < < 6 ) ) ) ; \
}
# define stbir__simdi8_expand_u16_to_u32(out,ireg) \
{ \
stbir__simdi a , b , zero = _mm_setzero_si128 ( ) ; \
a = _mm_unpacklo_epi16 ( ireg , zero ) ; \
b = _mm_unpackhi_epi16 ( ireg , zero ) ; \
out = _mm256_insertf128_si256 ( _mm256_castsi128_si256 ( a ) , b , 1 ) ; \
}
# define stbir__simdf8_pack_to_16words(out,aa,bb) \
{ \
stbir__simdi t0 , t1 ; \
stbir__simdf8 af , bf ; \
stbir__simdi8 a , b ; \
af = _mm256_min_ps ( aa , STBIR_max_uint16_as_floatX ) ; \
bf = _mm256_min_ps ( bb , STBIR_max_uint16_as_floatX ) ; \
af = _mm256_max_ps ( af , _mm256_setzero_ps ( ) ) ; \
bf = _mm256_max_ps ( bf , _mm256_setzero_ps ( ) ) ; \
a = _mm256_cvttps_epi32 ( af ) ; \
b = _mm256_cvttps_epi32 ( bf ) ; \
t0 = _mm_packus_epi32 ( _mm256_castsi256_si128 ( a ) , _mm256_extractf128_si256 ( a , 1 ) ) ; \
t1 = _mm_packus_epi32 ( _mm256_castsi256_si128 ( b ) , _mm256_extractf128_si256 ( b , 1 ) ) ; \
out = _mm256_setr_m128i ( t0 , t1 ) ; \
}
# endif
static __m256i stbir_00001111 = { STBIR__CONST_4d_32i ( 0 , 0 , 0 , 0 ) , STBIR__CONST_4d_32i ( 1 , 1 , 1 , 1 ) } ;
# define stbir__simdf8_0123to00001111( out, in ) (out) = _mm256_permutevar_ps ( in, stbir_00001111 )
static __m256i stbir_22223333 = { STBIR__CONST_4d_32i ( 2 , 2 , 2 , 2 ) , STBIR__CONST_4d_32i ( 3 , 3 , 3 , 3 ) } ;
# define stbir__simdf8_0123to22223333( out, in ) (out) = _mm256_permutevar_ps ( in, stbir_22223333 )
# define stbir__simdf8_0123to2222( out, in ) (out) = stbir__simdf_swiz(_mm256_castps256_ps128(in), 2,2,2,2 )
# define stbir__simdf8_load4b( out, ptr ) (out) = _mm256_broadcast_ps( (__m128 const *)(ptr) )
static __m256i stbir_00112233 = { STBIR__CONST_4d_32i ( 0 , 0 , 1 , 1 ) , STBIR__CONST_4d_32i ( 2 , 2 , 3 , 3 ) } ;
# define stbir__simdf8_0123to00112233( out, in ) (out) = _mm256_permutevar_ps ( in, stbir_00112233 )
# define stbir__simdf8_add4( out, a8, b ) (out) = _mm256_add_ps( a8, _mm256_castps128_ps256( b ) )
static __m256i stbir_load6 = { STBIR__CONST_4_32i ( 0x80000000 ) , STBIR__CONST_4d_32i ( 0x80000000 , 0x80000000 , 0 , 0 ) } ;
# define stbir__simdf8_load6z( out, ptr ) (out) = _mm256_maskload_ps( ptr, stbir_load6 )
# define stbir__simdf8_0123to00000000( out, in ) (out) = _mm256_shuffle_ps ( in, in, (0<<0)+(0<<2)+(0<<4)+(0<<6) )
# define stbir__simdf8_0123to11111111( out, in ) (out) = _mm256_shuffle_ps ( in, in, (1<<0)+(1<<2)+(1<<4)+(1<<6) )
# define stbir__simdf8_0123to22222222( out, in ) (out) = _mm256_shuffle_ps ( in, in, (2<<0)+(2<<2)+(2<<4)+(2<<6) )
# define stbir__simdf8_0123to33333333( out, in ) (out) = _mm256_shuffle_ps ( in, in, (3<<0)+(3<<2)+(3<<4)+(3<<6) )
# define stbir__simdf8_0123to21032103( out, in ) (out) = _mm256_shuffle_ps ( in, in, (2<<0)+(1<<2)+(0<<4)+(3<<6) )
# define stbir__simdf8_0123to32103210( out, in ) (out) = _mm256_shuffle_ps ( in, in, (3<<0)+(2<<2)+(1<<4)+(0<<6) )
# define stbir__simdf8_0123to12301230( out, in ) (out) = _mm256_shuffle_ps ( in, in, (1<<0)+(2<<2)+(3<<4)+(0<<6) )
# define stbir__simdf8_0123to10321032( out, in ) (out) = _mm256_shuffle_ps ( in, in, (1<<0)+(0<<2)+(3<<4)+(2<<6) )
# define stbir__simdf8_0123to30123012( out, in ) (out) = _mm256_shuffle_ps ( in, in, (3<<0)+(0<<2)+(1<<4)+(2<<6) )
# define stbir__simdf8_0123to11331133( out, in ) (out) = _mm256_shuffle_ps ( in, in, (1<<0)+(1<<2)+(3<<4)+(3<<6) )
# define stbir__simdf8_0123to00220022( out, in ) (out) = _mm256_shuffle_ps ( in, in, (0<<0)+(0<<2)+(2<<4)+(2<<6) )
# define stbir__simdf8_aaa1( out, alp, ones ) (out) = _mm256_blend_ps( alp, ones, (1<<0)+(1<<1)+(1<<2)+(0<<3)+(1<<4)+(1<<5)+(1<<6)+(0<<7)); (out)=_mm256_shuffle_ps( out,out, (3<<0) + (3<<2) + (3<<4) + (0<<6) )
# define stbir__simdf8_1aaa( out, alp, ones ) (out) = _mm256_blend_ps( alp, ones, (0<<0)+(1<<1)+(1<<2)+(1<<3)+(0<<4)+(1<<5)+(1<<6)+(1<<7)); (out)=_mm256_shuffle_ps( out,out, (1<<0) + (0<<2) + (0<<4) + (0<<6) )
# define stbir__simdf8_a1a1( out, alp, ones) (out) = _mm256_blend_ps( alp, ones, (1<<0)+(0<<1)+(1<<2)+(0<<3)+(1<<4)+(0<<5)+(1<<6)+(0<<7)); (out)=_mm256_shuffle_ps( out,out, (1<<0) + (0<<2) + (3<<4) + (2<<6) )
# define stbir__simdf8_1a1a( out, alp, ones) (out) = _mm256_blend_ps( alp, ones, (0<<0)+(1<<1)+(0<<2)+(1<<3)+(0<<4)+(1<<5)+(0<<6)+(1<<7)); (out)=_mm256_shuffle_ps( out,out, (1<<0) + (0<<2) + (3<<4) + (2<<6) )
# define stbir__simdf8_zero( reg ) (reg) = _mm256_setzero_ps()
# ifdef STBIR_USE_FMA // not on by default to maintain bit identical simd to non-simd
# define stbir__simdf8_madd( out, add, mul1, mul2 ) (out) = _mm256_fmadd_ps( mul1, mul2, add )
# define stbir__simdf8_madd_mem( out, add, mul, ptr ) (out) = _mm256_fmadd_ps( mul, _mm256_loadu_ps( (float const*)(ptr) ), add )
2023-12-14 19:09:00 +08:00
# define stbir__simdf8_madd_mem4( out, add, mul, ptr )(out) = _mm256_fmadd_ps( _mm256_setr_m128( mul, _mm_setzero_ps() ), _mm256_setr_m128( _mm_loadu_ps( (float const*)(ptr) ), _mm_setzero_ps() ), add )
2023-10-10 08:23:04 +08:00
# else
# define stbir__simdf8_madd( out, add, mul1, mul2 ) (out) = _mm256_add_ps( add, _mm256_mul_ps( mul1, mul2 ) )
# define stbir__simdf8_madd_mem( out, add, mul, ptr ) (out) = _mm256_add_ps( add, _mm256_mul_ps( mul, _mm256_loadu_ps( (float const*)(ptr) ) ) )
2023-12-14 19:09:00 +08:00
# define stbir__simdf8_madd_mem4( out, add, mul, ptr ) (out) = _mm256_add_ps( add, _mm256_setr_m128( _mm_mul_ps( mul, _mm_loadu_ps( (float const*)(ptr) ) ), _mm_setzero_ps() ) )
2023-10-10 08:23:04 +08:00
# endif
# define stbir__if_simdf8_cast_to_simdf4( val ) _mm256_castps256_ps128( val )
# endif
# ifdef STBIR_FLOORF
# undef STBIR_FLOORF
# endif
# define STBIR_FLOORF stbir_simd_floorf
static stbir__inline float stbir_simd_floorf ( float x ) // martins floorf
{
# if defined(STBIR_AVX) || defined(__SSE4_1__) || defined(STBIR_SSE41)
__m128 t = _mm_set_ss ( x ) ;
return _mm_cvtss_f32 ( _mm_floor_ss ( t , t ) ) ;
# else
__m128 f = _mm_set_ss ( x ) ;
__m128 t = _mm_cvtepi32_ps ( _mm_cvttps_epi32 ( f ) ) ;
__m128 r = _mm_add_ss ( t , _mm_and_ps ( _mm_cmplt_ss ( f , t ) , _mm_set_ss ( - 1.0f ) ) ) ;
return _mm_cvtss_f32 ( r ) ;
# endif
}
# ifdef STBIR_CEILF
# undef STBIR_CEILF
# endif
# define STBIR_CEILF stbir_simd_ceilf
static stbir__inline float stbir_simd_ceilf ( float x ) // martins ceilf
{
# if defined(STBIR_AVX) || defined(__SSE4_1__) || defined(STBIR_SSE41)
__m128 t = _mm_set_ss ( x ) ;
return _mm_cvtss_f32 ( _mm_ceil_ss ( t , t ) ) ;
# else
__m128 f = _mm_set_ss ( x ) ;
__m128 t = _mm_cvtepi32_ps ( _mm_cvttps_epi32 ( f ) ) ;
__m128 r = _mm_add_ss ( t , _mm_and_ps ( _mm_cmplt_ss ( t , f ) , _mm_set_ss ( 1.0f ) ) ) ;
return _mm_cvtss_f32 ( r ) ;
# endif
}
# elif defined(STBIR_NEON)
# include <arm_neon.h>
# define stbir__simdf float32x4_t
# define stbir__simdi uint32x4_t
# define stbir_simdi_castf( reg ) vreinterpretq_u32_f32(reg)
# define stbir_simdf_casti( reg ) vreinterpretq_f32_u32(reg)
# define stbir__simdf_load( reg, ptr ) (reg) = vld1q_f32( (float const*)(ptr) )
# define stbir__simdi_load( reg, ptr ) (reg) = vld1q_u32( (uint32_t const*)(ptr) )
# define stbir__simdf_load1( out, ptr ) (out) = vld1q_dup_f32( (float const*)(ptr) ) // top values can be random (not denormal or nan for perf)
# define stbir__simdi_load1( out, ptr ) (out) = vld1q_dup_u32( (uint32_t const*)(ptr) )
# define stbir__simdf_load1z( out, ptr ) (out) = vld1q_lane_f32( (float const*)(ptr), vdupq_n_f32(0), 0 ) // top values must be zero
# define stbir__simdf_frep4( fvar ) vdupq_n_f32( fvar )
# define stbir__simdf_load1frep4( out, fvar ) (out) = vdupq_n_f32( fvar )
# define stbir__simdf_load2( out, ptr ) (out) = vcombine_f32( vld1_f32( (float const*)(ptr) ), vcreate_f32(0) ) // top values can be random (not denormal or nan for perf)
# define stbir__simdf_load2z( out, ptr ) (out) = vcombine_f32( vld1_f32( (float const*)(ptr) ), vcreate_f32(0) ) // top values must be zero
# define stbir__simdf_load2hmerge( out, reg, ptr ) (out) = vcombine_f32( vget_low_f32(reg), vld1_f32( (float const*)(ptr) ) )
# define stbir__simdf_zeroP() vdupq_n_f32(0)
# define stbir__simdf_zero( reg ) (reg) = vdupq_n_f32(0)
# define stbir__simdf_store( ptr, reg ) vst1q_f32( (float*)(ptr), reg )
# define stbir__simdf_store1( ptr, reg ) vst1q_lane_f32( (float*)(ptr), reg, 0)
# define stbir__simdf_store2( ptr, reg ) vst1_f32( (float*)(ptr), vget_low_f32(reg) )
# define stbir__simdf_store2h( ptr, reg ) vst1_f32( (float*)(ptr), vget_high_f32(reg) )
# define stbir__simdi_store( ptr, reg ) vst1q_u32( (uint32_t*)(ptr), reg )
# define stbir__simdi_store1( ptr, reg ) vst1q_lane_u32( (uint32_t*)(ptr), reg, 0 )
# define stbir__simdi_store2( ptr, reg ) vst1_u32( (uint32_t*)(ptr), vget_low_u32(reg) )
# define stbir__prefetch( ptr )
# define stbir__simdi_expand_u8_to_u32(out0,out1,out2,out3,ireg) \
{ \
uint16x8_t l = vmovl_u8 ( vget_low_u8 ( vreinterpretq_u8_u32 ( ireg ) ) ) ; \
uint16x8_t h = vmovl_u8 ( vget_high_u8 ( vreinterpretq_u8_u32 ( ireg ) ) ) ; \
out0 = vmovl_u16 ( vget_low_u16 ( l ) ) ; \
out1 = vmovl_u16 ( vget_high_u16 ( l ) ) ; \
out2 = vmovl_u16 ( vget_low_u16 ( h ) ) ; \
out3 = vmovl_u16 ( vget_high_u16 ( h ) ) ; \
}
# define stbir__simdi_expand_u8_to_1u32(out,ireg) \
{ \
uint16x8_t tmp = vmovl_u8 ( vget_low_u8 ( vreinterpretq_u8_u32 ( ireg ) ) ) ; \
out = vmovl_u16 ( vget_low_u16 ( tmp ) ) ; \
}
# define stbir__simdi_expand_u16_to_u32(out0,out1,ireg) \
{ \
uint16x8_t tmp = vreinterpretq_u16_u32 ( ireg ) ; \
out0 = vmovl_u16 ( vget_low_u16 ( tmp ) ) ; \
out1 = vmovl_u16 ( vget_high_u16 ( tmp ) ) ; \
}
# define stbir__simdf_convert_float_to_i32( i, f ) (i) = vreinterpretq_u32_s32( vcvtq_s32_f32(f) )
# define stbir__simdf_convert_float_to_int( f ) vgetq_lane_s32(vcvtq_s32_f32(f), 0)
# define stbir__simdi_to_int( i ) (int)vgetq_lane_u32(i, 0)
# define stbir__simdf_convert_float_to_uint8( f ) ((unsigned char)vgetq_lane_s32(vcvtq_s32_f32(vmaxq_f32(vminq_f32(f,STBIR__CONSTF(STBIR_max_uint8_as_float)),vdupq_n_f32(0))), 0))
# define stbir__simdf_convert_float_to_short( f ) ((unsigned short)vgetq_lane_s32(vcvtq_s32_f32(vmaxq_f32(vminq_f32(f,STBIR__CONSTF(STBIR_max_uint16_as_float)),vdupq_n_f32(0))), 0))
# define stbir__simdi_convert_i32_to_float(out, ireg) (out) = vcvtq_f32_s32( vreinterpretq_s32_u32(ireg) )
# define stbir__simdf_add( out, reg0, reg1 ) (out) = vaddq_f32( reg0, reg1 )
# define stbir__simdf_mult( out, reg0, reg1 ) (out) = vmulq_f32( reg0, reg1 )
# define stbir__simdf_mult_mem( out, reg, ptr ) (out) = vmulq_f32( reg, vld1q_f32( (float const*)(ptr) ) )
# define stbir__simdf_mult1_mem( out, reg, ptr ) (out) = vmulq_f32( reg, vld1q_dup_f32( (float const*)(ptr) ) )
# define stbir__simdf_add_mem( out, reg, ptr ) (out) = vaddq_f32( reg, vld1q_f32( (float const*)(ptr) ) )
# define stbir__simdf_add1_mem( out, reg, ptr ) (out) = vaddq_f32( reg, vld1q_dup_f32( (float const*)(ptr) ) )
# ifdef STBIR_USE_FMA // not on by default to maintain bit identical simd to non-simd (and also x64 no madd to arm madd)
# define stbir__simdf_madd( out, add, mul1, mul2 ) (out) = vfmaq_f32( add, mul1, mul2 )
# define stbir__simdf_madd1( out, add, mul1, mul2 ) (out) = vfmaq_f32( add, mul1, mul2 )
# define stbir__simdf_madd_mem( out, add, mul, ptr ) (out) = vfmaq_f32( add, mul, vld1q_f32( (float const*)(ptr) ) )
# define stbir__simdf_madd1_mem( out, add, mul, ptr ) (out) = vfmaq_f32( add, mul, vld1q_dup_f32( (float const*)(ptr) ) )
# else
# define stbir__simdf_madd( out, add, mul1, mul2 ) (out) = vaddq_f32( add, vmulq_f32( mul1, mul2 ) )
# define stbir__simdf_madd1( out, add, mul1, mul2 ) (out) = vaddq_f32( add, vmulq_f32( mul1, mul2 ) )
# define stbir__simdf_madd_mem( out, add, mul, ptr ) (out) = vaddq_f32( add, vmulq_f32( mul, vld1q_f32( (float const*)(ptr) ) ) )
# define stbir__simdf_madd1_mem( out, add, mul, ptr ) (out) = vaddq_f32( add, vmulq_f32( mul, vld1q_dup_f32( (float const*)(ptr) ) ) )
# endif
# define stbir__simdf_add1( out, reg0, reg1 ) (out) = vaddq_f32( reg0, reg1 )
# define stbir__simdf_mult1( out, reg0, reg1 ) (out) = vmulq_f32( reg0, reg1 )
# define stbir__simdf_and( out, reg0, reg1 ) (out) = vreinterpretq_f32_u32( vandq_u32( vreinterpretq_u32_f32(reg0), vreinterpretq_u32_f32(reg1) ) )
# define stbir__simdf_or( out, reg0, reg1 ) (out) = vreinterpretq_f32_u32( vorrq_u32( vreinterpretq_u32_f32(reg0), vreinterpretq_u32_f32(reg1) ) )
# define stbir__simdf_min( out, reg0, reg1 ) (out) = vminq_f32( reg0, reg1 )
# define stbir__simdf_max( out, reg0, reg1 ) (out) = vmaxq_f32( reg0, reg1 )
# define stbir__simdf_min1( out, reg0, reg1 ) (out) = vminq_f32( reg0, reg1 )
# define stbir__simdf_max1( out, reg0, reg1 ) (out) = vmaxq_f32( reg0, reg1 )
# define stbir__simdf_0123ABCDto3ABx( out, reg0, reg1 ) (out) = vextq_f32( reg0, reg1, 3 )
# define stbir__simdf_0123ABCDto23Ax( out, reg0, reg1 ) (out) = vextq_f32( reg0, reg1, 2 )
# define stbir__simdf_a1a1( out, alp, ones ) (out) = vzipq_f32(vuzpq_f32(alp, alp).val[1], ones).val[0]
# define stbir__simdf_1a1a( out, alp, ones ) (out) = vzipq_f32(ones, vuzpq_f32(alp, alp).val[0]).val[0]
# if defined( _M_ARM64 ) || defined( __aarch64__ ) || defined( __arm64__ )
# define stbir__simdf_aaa1( out, alp, ones ) (out) = vcopyq_laneq_f32(vdupq_n_f32(vgetq_lane_f32(alp, 3)), 3, ones, 3)
# define stbir__simdf_1aaa( out, alp, ones ) (out) = vcopyq_laneq_f32(vdupq_n_f32(vgetq_lane_f32(alp, 0)), 0, ones, 0)
# if defined( _MSC_VER ) && !defined(__clang__)
# define stbir_make16(a,b,c,d) vcombine_u8( \
vcreate_u8 ( ( 4 * a + 0 ) | ( ( 4 * a + 1 ) < < 8 ) | ( ( 4 * a + 2 ) < < 16 ) | ( ( 4 * a + 3 ) < < 24 ) | \
( ( stbir_uint64 ) ( 4 * b + 0 ) < < 32 ) | ( ( stbir_uint64 ) ( 4 * b + 1 ) < < 40 ) | ( ( stbir_uint64 ) ( 4 * b + 2 ) < < 48 ) | ( ( stbir_uint64 ) ( 4 * b + 3 ) < < 56 ) ) , \
vcreate_u8 ( ( 4 * c + 0 ) | ( ( 4 * c + 1 ) < < 8 ) | ( ( 4 * c + 2 ) < < 16 ) | ( ( 4 * c + 3 ) < < 24 ) | \
( ( stbir_uint64 ) ( 4 * d + 0 ) < < 32 ) | ( ( stbir_uint64 ) ( 4 * d + 1 ) < < 40 ) | ( ( stbir_uint64 ) ( 4 * d + 2 ) < < 48 ) | ( ( stbir_uint64 ) ( 4 * d + 3 ) < < 56 ) ) )
# else
# define stbir_make16(a,b,c,d) (uint8x16_t){4*a+0,4*a+1,4*a+2,4*a+3,4*b+0,4*b+1,4*b+2,4*b+3,4*c+0,4*c+1,4*c+2,4*c+3,4*d+0,4*d+1,4*d+2,4*d+3}
# endif
# define stbir__simdf_swiz( reg, one, two, three, four ) vreinterpretq_f32_u8( vqtbl1q_u8( vreinterpretq_u8_f32(reg), stbir_make16(one, two, three, four) ) )
# define stbir__simdi_16madd( out, reg0, reg1 ) \
{ \
int16x8_t r0 = vreinterpretq_s16_u32 ( reg0 ) ; \
int16x8_t r1 = vreinterpretq_s16_u32 ( reg1 ) ; \
int32x4_t tmp0 = vmull_s16 ( vget_low_s16 ( r0 ) , vget_low_s16 ( r1 ) ) ; \
int32x4_t tmp1 = vmull_s16 ( vget_high_s16 ( r0 ) , vget_high_s16 ( r1 ) ) ; \
( out ) = vreinterpretq_u32_s32 ( vpaddq_s32 ( tmp0 , tmp1 ) ) ; \
}
# else
# define stbir__simdf_aaa1( out, alp, ones ) (out) = vsetq_lane_f32(1.0f, vdupq_n_f32(vgetq_lane_f32(alp, 3)), 3)
# define stbir__simdf_1aaa( out, alp, ones ) (out) = vsetq_lane_f32(1.0f, vdupq_n_f32(vgetq_lane_f32(alp, 0)), 0)
# if defined( _MSC_VER ) && !defined(__clang__)
static stbir__inline uint8x8x2_t stbir_make8x2 ( float32x4_t reg )
{
uint8x8x2_t r = { { vget_low_u8 ( vreinterpretq_u8_f32 ( reg ) ) , vget_high_u8 ( vreinterpretq_u8_f32 ( reg ) ) } } ;
return r ;
}
# define stbir_make8(a,b) vcreate_u8( \
( 4 * a + 0 ) | ( ( 4 * a + 1 ) < < 8 ) | ( ( 4 * a + 2 ) < < 16 ) | ( ( 4 * a + 3 ) < < 24 ) | \
( ( stbir_uint64 ) ( 4 * b + 0 ) < < 32 ) | ( ( stbir_uint64 ) ( 4 * b + 1 ) < < 40 ) | ( ( stbir_uint64 ) ( 4 * b + 2 ) < < 48 ) | ( ( stbir_uint64 ) ( 4 * b + 3 ) < < 56 ) )
# else
# define stbir_make8x2(reg) (uint8x8x2_t){ { vget_low_u8(vreinterpretq_u8_f32(reg)), vget_high_u8(vreinterpretq_u8_f32(reg)) } }
# define stbir_make8(a,b) (uint8x8_t){4*a+0,4*a+1,4*a+2,4*a+3,4*b+0,4*b+1,4*b+2,4*b+3}
# endif
# define stbir__simdf_swiz( reg, one, two, three, four ) vreinterpretq_f32_u8( vcombine_u8( \
vtbl2_u8 ( stbir_make8x2 ( reg ) , stbir_make8 ( one , two ) ) , \
vtbl2_u8 ( stbir_make8x2 ( reg ) , stbir_make8 ( three , four ) ) ) )
# define stbir__simdi_16madd( out, reg0, reg1 ) \
{ \
int16x8_t r0 = vreinterpretq_s16_u32 ( reg0 ) ; \
int16x8_t r1 = vreinterpretq_s16_u32 ( reg1 ) ; \
int32x4_t tmp0 = vmull_s16 ( vget_low_s16 ( r0 ) , vget_low_s16 ( r1 ) ) ; \
int32x4_t tmp1 = vmull_s16 ( vget_high_s16 ( r0 ) , vget_high_s16 ( r1 ) ) ; \
int32x2_t out0 = vpadd_s32 ( vget_low_s32 ( tmp0 ) , vget_high_s32 ( tmp0 ) ) ; \
int32x2_t out1 = vpadd_s32 ( vget_low_s32 ( tmp1 ) , vget_high_s32 ( tmp1 ) ) ; \
( out ) = vreinterpretq_u32_s32 ( vcombine_s32 ( out0 , out1 ) ) ; \
}
# endif
# define stbir__simdi_and( out, reg0, reg1 ) (out) = vandq_u32( reg0, reg1 )
# define stbir__simdi_or( out, reg0, reg1 ) (out) = vorrq_u32( reg0, reg1 )
# define stbir__simdf_pack_to_8bytes(out,aa,bb) \
{ \
float32x4_t af = vmaxq_f32 ( vminq_f32 ( aa , STBIR__CONSTF ( STBIR_max_uint8_as_float ) ) , vdupq_n_f32 ( 0 ) ) ; \
float32x4_t bf = vmaxq_f32 ( vminq_f32 ( bb , STBIR__CONSTF ( STBIR_max_uint8_as_float ) ) , vdupq_n_f32 ( 0 ) ) ; \
int16x4_t ai = vqmovn_s32 ( vcvtq_s32_f32 ( af ) ) ; \
int16x4_t bi = vqmovn_s32 ( vcvtq_s32_f32 ( bf ) ) ; \
uint8x8_t out8 = vqmovun_s16 ( vcombine_s16 ( ai , bi ) ) ; \
out = vreinterpretq_u32_u8 ( vcombine_u8 ( out8 , out8 ) ) ; \
}
# define stbir__simdf_pack_to_8words(out,aa,bb) \
{ \
float32x4_t af = vmaxq_f32 ( vminq_f32 ( aa , STBIR__CONSTF ( STBIR_max_uint16_as_float ) ) , vdupq_n_f32 ( 0 ) ) ; \
float32x4_t bf = vmaxq_f32 ( vminq_f32 ( bb , STBIR__CONSTF ( STBIR_max_uint16_as_float ) ) , vdupq_n_f32 ( 0 ) ) ; \
int32x4_t ai = vcvtq_s32_f32 ( af ) ; \
int32x4_t bi = vcvtq_s32_f32 ( bf ) ; \
out = vreinterpretq_u32_u16 ( vcombine_u16 ( vqmovun_s32 ( ai ) , vqmovun_s32 ( bi ) ) ) ; \
}
# define stbir__interleave_pack_and_store_16_u8( ptr, r0, r1, r2, r3 ) \
{ \
int16x4x2_t tmp0 = vzip_s16 ( vqmovn_s32 ( vreinterpretq_s32_u32 ( r0 ) ) , vqmovn_s32 ( vreinterpretq_s32_u32 ( r2 ) ) ) ; \
int16x4x2_t tmp1 = vzip_s16 ( vqmovn_s32 ( vreinterpretq_s32_u32 ( r1 ) ) , vqmovn_s32 ( vreinterpretq_s32_u32 ( r3 ) ) ) ; \
uint8x8x2_t out = \
{ { \
vqmovun_s16 ( vcombine_s16 ( tmp0 . val [ 0 ] , tmp0 . val [ 1 ] ) ) , \
vqmovun_s16 ( vcombine_s16 ( tmp1 . val [ 0 ] , tmp1 . val [ 1 ] ) ) , \
} } ; \
vst2_u8 ( ptr , out ) ; \
}
# define stbir__simdf_load4_transposed( o0, o1, o2, o3, ptr ) \
{ \
float32x4x4_t tmp = vld4q_f32 ( ptr ) ; \
o0 = tmp . val [ 0 ] ; \
o1 = tmp . val [ 1 ] ; \
o2 = tmp . val [ 2 ] ; \
o3 = tmp . val [ 3 ] ; \
}
# define stbir__simdi_32shr( out, reg, imm ) out = vshrq_n_u32( reg, imm )
# if defined( _MSC_VER ) && !defined(__clang__)
# define STBIR__SIMDF_CONST(var, x) __declspec(align(8)) float var[] = { x, x, x, x }
# define STBIR__SIMDI_CONST(var, x) __declspec(align(8)) uint32_t var[] = { x, x, x, x }
# define STBIR__CONSTF(var) (*(const float32x4_t*)var)
# define STBIR__CONSTI(var) (*(const uint32x4_t*)var)
# else
# define STBIR__SIMDF_CONST(var, x) stbir__simdf var = { x, x, x, x }
# define STBIR__SIMDI_CONST(var, x) stbir__simdi var = { x, x, x, x }
# define STBIR__CONSTF(var) (var)
# define STBIR__CONSTI(var) (var)
# endif
# ifdef STBIR_FLOORF
# undef STBIR_FLOORF
# endif
# define STBIR_FLOORF stbir_simd_floorf
static stbir__inline float stbir_simd_floorf ( float x )
{
# if defined( _M_ARM64 ) || defined( __aarch64__ ) || defined( __arm64__ )
return vget_lane_f32 ( vrndm_f32 ( vdup_n_f32 ( x ) ) , 0 ) ;
# else
float32x2_t f = vdup_n_f32 ( x ) ;
float32x2_t t = vcvt_f32_s32 ( vcvt_s32_f32 ( f ) ) ;
uint32x2_t a = vclt_f32 ( f , t ) ;
uint32x2_t b = vreinterpret_u32_f32 ( vdup_n_f32 ( - 1.0f ) ) ;
float32x2_t r = vadd_f32 ( t , vreinterpret_f32_u32 ( vand_u32 ( a , b ) ) ) ;
return vget_lane_f32 ( r , 0 ) ;
# endif
}
# ifdef STBIR_CEILF
# undef STBIR_CEILF
# endif
# define STBIR_CEILF stbir_simd_ceilf
static stbir__inline float stbir_simd_ceilf ( float x )
{
# if defined( _M_ARM64 ) || defined( __aarch64__ ) || defined( __arm64__ )
return vget_lane_f32 ( vrndp_f32 ( vdup_n_f32 ( x ) ) , 0 ) ;
# else
float32x2_t f = vdup_n_f32 ( x ) ;
float32x2_t t = vcvt_f32_s32 ( vcvt_s32_f32 ( f ) ) ;
uint32x2_t a = vclt_f32 ( t , f ) ;
uint32x2_t b = vreinterpret_u32_f32 ( vdup_n_f32 ( 1.0f ) ) ;
float32x2_t r = vadd_f32 ( t , vreinterpret_f32_u32 ( vand_u32 ( a , b ) ) ) ;
return vget_lane_f32 ( r , 0 ) ;
# endif
}
# define STBIR_SIMD
# elif defined(STBIR_WASM)
# include <wasm_simd128.h>
# define stbir__simdf v128_t
# define stbir__simdi v128_t
# define stbir_simdi_castf( reg ) (reg)
# define stbir_simdf_casti( reg ) (reg)
# define stbir__simdf_load( reg, ptr ) (reg) = wasm_v128_load( (void const*)(ptr) )
# define stbir__simdi_load( reg, ptr ) (reg) = wasm_v128_load( (void const*)(ptr) )
# define stbir__simdf_load1( out, ptr ) (out) = wasm_v128_load32_splat( (void const*)(ptr) ) // top values can be random (not denormal or nan for perf)
# define stbir__simdi_load1( out, ptr ) (out) = wasm_v128_load32_splat( (void const*)(ptr) )
# define stbir__simdf_load1z( out, ptr ) (out) = wasm_v128_load32_zero( (void const*)(ptr) ) // top values must be zero
# define stbir__simdf_frep4( fvar ) wasm_f32x4_splat( fvar )
# define stbir__simdf_load1frep4( out, fvar ) (out) = wasm_f32x4_splat( fvar )
# define stbir__simdf_load2( out, ptr ) (out) = wasm_v128_load64_splat( (void const*)(ptr) ) // top values can be random (not denormal or nan for perf)
# define stbir__simdf_load2z( out, ptr ) (out) = wasm_v128_load64_zero( (void const*)(ptr) ) // top values must be zero
# define stbir__simdf_load2hmerge( out, reg, ptr ) (out) = wasm_v128_load64_lane( (void const*)(ptr), reg, 1 )
# define stbir__simdf_zeroP() wasm_f32x4_const_splat(0)
# define stbir__simdf_zero( reg ) (reg) = wasm_f32x4_const_splat(0)
# define stbir__simdf_store( ptr, reg ) wasm_v128_store( (void*)(ptr), reg )
# define stbir__simdf_store1( ptr, reg ) wasm_v128_store32_lane( (void*)(ptr), reg, 0 )
# define stbir__simdf_store2( ptr, reg ) wasm_v128_store64_lane( (void*)(ptr), reg, 0 )
# define stbir__simdf_store2h( ptr, reg ) wasm_v128_store64_lane( (void*)(ptr), reg, 1 )
# define stbir__simdi_store( ptr, reg ) wasm_v128_store( (void*)(ptr), reg )
# define stbir__simdi_store1( ptr, reg ) wasm_v128_store32_lane( (void*)(ptr), reg, 0 )
# define stbir__simdi_store2( ptr, reg ) wasm_v128_store64_lane( (void*)(ptr), reg, 0 )
# define stbir__prefetch( ptr )
# define stbir__simdi_expand_u8_to_u32(out0,out1,out2,out3,ireg) \
{ \
v128_t l = wasm_u16x8_extend_low_u8x16 ( ireg ) ; \
v128_t h = wasm_u16x8_extend_high_u8x16 ( ireg ) ; \
out0 = wasm_u32x4_extend_low_u16x8 ( l ) ; \
out1 = wasm_u32x4_extend_high_u16x8 ( l ) ; \
out2 = wasm_u32x4_extend_low_u16x8 ( h ) ; \
out3 = wasm_u32x4_extend_high_u16x8 ( h ) ; \
}
# define stbir__simdi_expand_u8_to_1u32(out,ireg) \
{ \
v128_t tmp = wasm_u16x8_extend_low_u8x16 ( ireg ) ; \
out = wasm_u32x4_extend_low_u16x8 ( tmp ) ; \
}
# define stbir__simdi_expand_u16_to_u32(out0,out1,ireg) \
{ \
out0 = wasm_u32x4_extend_low_u16x8 ( ireg ) ; \
out1 = wasm_u32x4_extend_high_u16x8 ( ireg ) ; \
}
# define stbir__simdf_convert_float_to_i32( i, f ) (i) = wasm_i32x4_trunc_sat_f32x4(f)
# define stbir__simdf_convert_float_to_int( f ) wasm_i32x4_extract_lane(wasm_i32x4_trunc_sat_f32x4(f), 0)
# define stbir__simdi_to_int( i ) wasm_i32x4_extract_lane(i, 0)
# define stbir__simdf_convert_float_to_uint8( f ) ((unsigned char)wasm_i32x4_extract_lane(wasm_i32x4_trunc_sat_f32x4(wasm_f32x4_max(wasm_f32x4_min(f,STBIR_max_uint8_as_float),wasm_f32x4_const_splat(0))), 0))
# define stbir__simdf_convert_float_to_short( f ) ((unsigned short)wasm_i32x4_extract_lane(wasm_i32x4_trunc_sat_f32x4(wasm_f32x4_max(wasm_f32x4_min(f,STBIR_max_uint16_as_float),wasm_f32x4_const_splat(0))), 0))
# define stbir__simdi_convert_i32_to_float(out, ireg) (out) = wasm_f32x4_convert_i32x4(ireg)
# define stbir__simdf_add( out, reg0, reg1 ) (out) = wasm_f32x4_add( reg0, reg1 )
# define stbir__simdf_mult( out, reg0, reg1 ) (out) = wasm_f32x4_mul( reg0, reg1 )
# define stbir__simdf_mult_mem( out, reg, ptr ) (out) = wasm_f32x4_mul( reg, wasm_v128_load( (void const*)(ptr) ) )
# define stbir__simdf_mult1_mem( out, reg, ptr ) (out) = wasm_f32x4_mul( reg, wasm_v128_load32_splat( (void const*)(ptr) ) )
# define stbir__simdf_add_mem( out, reg, ptr ) (out) = wasm_f32x4_add( reg, wasm_v128_load( (void const*)(ptr) ) )
# define stbir__simdf_add1_mem( out, reg, ptr ) (out) = wasm_f32x4_add( reg, wasm_v128_load32_splat( (void const*)(ptr) ) )
# define stbir__simdf_madd( out, add, mul1, mul2 ) (out) = wasm_f32x4_add( add, wasm_f32x4_mul( mul1, mul2 ) )
# define stbir__simdf_madd1( out, add, mul1, mul2 ) (out) = wasm_f32x4_add( add, wasm_f32x4_mul( mul1, mul2 ) )
# define stbir__simdf_madd_mem( out, add, mul, ptr ) (out) = wasm_f32x4_add( add, wasm_f32x4_mul( mul, wasm_v128_load( (void const*)(ptr) ) ) )
# define stbir__simdf_madd1_mem( out, add, mul, ptr ) (out) = wasm_f32x4_add( add, wasm_f32x4_mul( mul, wasm_v128_load32_splat( (void const*)(ptr) ) ) )
# define stbir__simdf_add1( out, reg0, reg1 ) (out) = wasm_f32x4_add( reg0, reg1 )
# define stbir__simdf_mult1( out, reg0, reg1 ) (out) = wasm_f32x4_mul( reg0, reg1 )
# define stbir__simdf_and( out, reg0, reg1 ) (out) = wasm_v128_and( reg0, reg1 )
# define stbir__simdf_or( out, reg0, reg1 ) (out) = wasm_v128_or( reg0, reg1 )
# define stbir__simdf_min( out, reg0, reg1 ) (out) = wasm_f32x4_min( reg0, reg1 )
# define stbir__simdf_max( out, reg0, reg1 ) (out) = wasm_f32x4_max( reg0, reg1 )
# define stbir__simdf_min1( out, reg0, reg1 ) (out) = wasm_f32x4_min( reg0, reg1 )
# define stbir__simdf_max1( out, reg0, reg1 ) (out) = wasm_f32x4_max( reg0, reg1 )
# define stbir__simdf_0123ABCDto3ABx( out, reg0, reg1 ) (out) = wasm_i32x4_shuffle( reg0, reg1, 3, 4, 5, -1 )
# define stbir__simdf_0123ABCDto23Ax( out, reg0, reg1 ) (out) = wasm_i32x4_shuffle( reg0, reg1, 2, 3, 4, -1 )
# define stbir__simdf_aaa1(out,alp,ones) (out) = wasm_i32x4_shuffle(alp, ones, 3, 3, 3, 4)
# define stbir__simdf_1aaa(out,alp,ones) (out) = wasm_i32x4_shuffle(alp, ones, 4, 0, 0, 0)
# define stbir__simdf_a1a1(out,alp,ones) (out) = wasm_i32x4_shuffle(alp, ones, 1, 4, 3, 4)
# define stbir__simdf_1a1a(out,alp,ones) (out) = wasm_i32x4_shuffle(alp, ones, 4, 0, 4, 2)
# define stbir__simdf_swiz( reg, one, two, three, four ) wasm_i32x4_shuffle(reg, reg, one, two, three, four)
# define stbir__simdi_and( out, reg0, reg1 ) (out) = wasm_v128_and( reg0, reg1 )
# define stbir__simdi_or( out, reg0, reg1 ) (out) = wasm_v128_or( reg0, reg1 )
# define stbir__simdi_16madd( out, reg0, reg1 ) (out) = wasm_i32x4_dot_i16x8( reg0, reg1 )
# define stbir__simdf_pack_to_8bytes(out,aa,bb) \
{ \
v128_t af = wasm_f32x4_max ( wasm_f32x4_min ( aa , STBIR_max_uint8_as_float ) , wasm_f32x4_const_splat ( 0 ) ) ; \
v128_t bf = wasm_f32x4_max ( wasm_f32x4_min ( bb , STBIR_max_uint8_as_float ) , wasm_f32x4_const_splat ( 0 ) ) ; \
v128_t ai = wasm_i32x4_trunc_sat_f32x4 ( af ) ; \
v128_t bi = wasm_i32x4_trunc_sat_f32x4 ( bf ) ; \
v128_t out16 = wasm_i16x8_narrow_i32x4 ( ai , bi ) ; \
out = wasm_u8x16_narrow_i16x8 ( out16 , out16 ) ; \
}
# define stbir__simdf_pack_to_8words(out,aa,bb) \
{ \
v128_t af = wasm_f32x4_max ( wasm_f32x4_min ( aa , STBIR_max_uint16_as_float ) , wasm_f32x4_const_splat ( 0 ) ) ; \
v128_t bf = wasm_f32x4_max ( wasm_f32x4_min ( bb , STBIR_max_uint16_as_float ) , wasm_f32x4_const_splat ( 0 ) ) ; \
v128_t ai = wasm_i32x4_trunc_sat_f32x4 ( af ) ; \
v128_t bi = wasm_i32x4_trunc_sat_f32x4 ( bf ) ; \
out = wasm_u16x8_narrow_i32x4 ( ai , bi ) ; \
}
# define stbir__interleave_pack_and_store_16_u8( ptr, r0, r1, r2, r3 ) \
{ \
v128_t tmp0 = wasm_i16x8_narrow_i32x4 ( r0 , r1 ) ; \
v128_t tmp1 = wasm_i16x8_narrow_i32x4 ( r2 , r3 ) ; \
v128_t tmp = wasm_u8x16_narrow_i16x8 ( tmp0 , tmp1 ) ; \
tmp = wasm_i8x16_shuffle ( tmp , tmp , 0 , 4 , 8 , 12 , 1 , 5 , 9 , 13 , 2 , 6 , 10 , 14 , 3 , 7 , 11 , 15 ) ; \
wasm_v128_store ( ( void * ) ( ptr ) , tmp ) ; \
}
# define stbir__simdf_load4_transposed( o0, o1, o2, o3, ptr ) \
{ \
v128_t t0 = wasm_v128_load ( ptr ) ; \
v128_t t1 = wasm_v128_load ( ptr + 4 ) ; \
v128_t t2 = wasm_v128_load ( ptr + 8 ) ; \
v128_t t3 = wasm_v128_load ( ptr + 12 ) ; \
v128_t s0 = wasm_i32x4_shuffle ( t0 , t1 , 0 , 4 , 2 , 6 ) ; \
v128_t s1 = wasm_i32x4_shuffle ( t0 , t1 , 1 , 5 , 3 , 7 ) ; \
v128_t s2 = wasm_i32x4_shuffle ( t2 , t3 , 0 , 4 , 2 , 6 ) ; \
v128_t s3 = wasm_i32x4_shuffle ( t2 , t3 , 1 , 5 , 3 , 7 ) ; \
o0 = wasm_i32x4_shuffle ( s0 , s2 , 0 , 1 , 4 , 5 ) ; \
o1 = wasm_i32x4_shuffle ( s1 , s3 , 0 , 1 , 4 , 5 ) ; \
o2 = wasm_i32x4_shuffle ( s0 , s2 , 2 , 3 , 6 , 7 ) ; \
o3 = wasm_i32x4_shuffle ( s1 , s3 , 2 , 3 , 6 , 7 ) ; \
}
# define stbir__simdi_32shr( out, reg, imm ) out = wasm_u32x4_shr( reg, imm )
typedef float stbir__f32x4 __attribute__ ( ( __vector_size__ ( 16 ) , __aligned__ ( 16 ) ) ) ;
# define STBIR__SIMDF_CONST(var, x) stbir__simdf var = (v128_t)(stbir__f32x4){ x, x, x, x }
# define STBIR__SIMDI_CONST(var, x) stbir__simdi var = { x, x, x, x }
# define STBIR__CONSTF(var) (var)
# define STBIR__CONSTI(var) (var)
# ifdef STBIR_FLOORF
# undef STBIR_FLOORF
# endif
# define STBIR_FLOORF stbir_simd_floorf
static stbir__inline float stbir_simd_floorf ( float x )
{
return wasm_f32x4_extract_lane ( wasm_f32x4_floor ( wasm_f32x4_splat ( x ) ) , 0 ) ;
}
# ifdef STBIR_CEILF
# undef STBIR_CEILF
# endif
# define STBIR_CEILF stbir_simd_ceilf
static stbir__inline float stbir_simd_ceilf ( float x )
{
return wasm_f32x4_extract_lane ( wasm_f32x4_ceil ( wasm_f32x4_splat ( x ) ) , 0 ) ;
}
# define STBIR_SIMD
# endif // SSE2/NEON/WASM
# endif // NO SIMD
# ifdef STBIR_SIMD8
# define stbir__simdfX stbir__simdf8
# define stbir__simdiX stbir__simdi8
# define stbir__simdfX_load stbir__simdf8_load
# define stbir__simdiX_load stbir__simdi8_load
# define stbir__simdfX_mult stbir__simdf8_mult
# define stbir__simdfX_add_mem stbir__simdf8_add_mem
# define stbir__simdfX_madd_mem stbir__simdf8_madd_mem
# define stbir__simdfX_store stbir__simdf8_store
# define stbir__simdiX_store stbir__simdi8_store
# define stbir__simdf_frepX stbir__simdf8_frep8
# define stbir__simdfX_madd stbir__simdf8_madd
# define stbir__simdfX_min stbir__simdf8_min
# define stbir__simdfX_max stbir__simdf8_max
# define stbir__simdfX_aaa1 stbir__simdf8_aaa1
# define stbir__simdfX_1aaa stbir__simdf8_1aaa
# define stbir__simdfX_a1a1 stbir__simdf8_a1a1
# define stbir__simdfX_1a1a stbir__simdf8_1a1a
# define stbir__simdfX_convert_float_to_i32 stbir__simdf8_convert_float_to_i32
# define stbir__simdfX_pack_to_words stbir__simdf8_pack_to_16words
# define stbir__simdfX_zero stbir__simdf8_zero
# define STBIR_onesX STBIR_ones8
# define STBIR_max_uint8_as_floatX STBIR_max_uint8_as_float8
# define STBIR_max_uint16_as_floatX STBIR_max_uint16_as_float8
# define STBIR_simd_point5X STBIR_simd_point58
# define stbir__simdfX_float_count 8
# define stbir__simdfX_0123to1230 stbir__simdf8_0123to12301230
# define stbir__simdfX_0123to2103 stbir__simdf8_0123to21032103
static const stbir__simdf8 STBIR_max_uint16_as_float_inverted8 = { stbir__max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted } ;
static const stbir__simdf8 STBIR_max_uint8_as_float_inverted8 = { stbir__max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted } ;
static const stbir__simdf8 STBIR_ones8 = { 1.0 , 1.0 , 1.0 , 1.0 , 1.0 , 1.0 , 1.0 , 1.0 } ;
static const stbir__simdf8 STBIR_simd_point58 = { 0.5 , 0.5 , 0.5 , 0.5 , 0.5 , 0.5 , 0.5 , 0.5 } ;
static const stbir__simdf8 STBIR_max_uint8_as_float8 = { stbir__max_uint8_as_float , stbir__max_uint8_as_float , stbir__max_uint8_as_float , stbir__max_uint8_as_float , stbir__max_uint8_as_float , stbir__max_uint8_as_float , stbir__max_uint8_as_float , stbir__max_uint8_as_float } ;
static const stbir__simdf8 STBIR_max_uint16_as_float8 = { stbir__max_uint16_as_float , stbir__max_uint16_as_float , stbir__max_uint16_as_float , stbir__max_uint16_as_float , stbir__max_uint16_as_float , stbir__max_uint16_as_float , stbir__max_uint16_as_float , stbir__max_uint16_as_float } ;
# else
# define stbir__simdfX stbir__simdf
# define stbir__simdiX stbir__simdi
# define stbir__simdfX_load stbir__simdf_load
# define stbir__simdiX_load stbir__simdi_load
# define stbir__simdfX_mult stbir__simdf_mult
# define stbir__simdfX_add_mem stbir__simdf_add_mem
# define stbir__simdfX_madd_mem stbir__simdf_madd_mem
# define stbir__simdfX_store stbir__simdf_store
# define stbir__simdiX_store stbir__simdi_store
# define stbir__simdf_frepX stbir__simdf_frep4
# define stbir__simdfX_madd stbir__simdf_madd
# define stbir__simdfX_min stbir__simdf_min
# define stbir__simdfX_max stbir__simdf_max
# define stbir__simdfX_aaa1 stbir__simdf_aaa1
# define stbir__simdfX_1aaa stbir__simdf_1aaa
# define stbir__simdfX_a1a1 stbir__simdf_a1a1
# define stbir__simdfX_1a1a stbir__simdf_1a1a
# define stbir__simdfX_convert_float_to_i32 stbir__simdf_convert_float_to_i32
# define stbir__simdfX_pack_to_words stbir__simdf_pack_to_8words
# define stbir__simdfX_zero stbir__simdf_zero
# define STBIR_onesX STBIR__CONSTF(STBIR_ones)
# define STBIR_simd_point5X STBIR__CONSTF(STBIR_simd_point5)
# define STBIR_max_uint8_as_floatX STBIR__CONSTF(STBIR_max_uint8_as_float)
# define STBIR_max_uint16_as_floatX STBIR__CONSTF(STBIR_max_uint16_as_float)
# define stbir__simdfX_float_count 4
# define stbir__if_simdf8_cast_to_simdf4( val ) ( val )
# define stbir__simdfX_0123to1230 stbir__simdf_0123to1230
# define stbir__simdfX_0123to2103 stbir__simdf_0123to2103
# endif
# if defined(STBIR_NEON) && !defined(_M_ARM)
# if defined( _MSC_VER ) && !defined(__clang__)
typedef __int16 stbir__FP16 ;
# else
typedef float16_t stbir__FP16 ;
# endif
# else // no NEON, or 32-bit ARM for MSVC
typedef union stbir__FP16
{
unsigned short u ;
} stbir__FP16 ;
# endif
# if !defined(STBIR_NEON) && !defined(STBIR_FP16C) || defined(STBIR_NEON) && defined(_M_ARM)
// Fabian's half float routines, see: https://gist.github.com/rygorous/2156668
static stbir__inline float stbir__half_to_float ( stbir__FP16 h )
{
static const stbir__FP32 magic = { ( 254 - 15 ) < < 23 } ;
static const stbir__FP32 was_infnan = { ( 127 + 16 ) < < 23 } ;
stbir__FP32 o ;
o . u = ( h . u & 0x7fff ) < < 13 ; // exponent/mantissa bits
o . f * = magic . f ; // exponent adjust
if ( o . f > = was_infnan . f ) // make sure Inf/NaN survive
o . u | = 255 < < 23 ;
o . u | = ( h . u & 0x8000 ) < < 16 ; // sign bit
return o . f ;
}
static stbir__inline stbir__FP16 stbir__float_to_half ( float val )
{
stbir__FP32 f32infty = { 255 < < 23 } ;
stbir__FP32 f16max = { ( 127 + 16 ) < < 23 } ;
stbir__FP32 denorm_magic = { ( ( 127 - 15 ) + ( 23 - 10 ) + 1 ) < < 23 } ;
unsigned int sign_mask = 0x80000000u ;
stbir__FP16 o = { 0 } ;
stbir__FP32 f ;
unsigned int sign ;
f . f = val ;
sign = f . u & sign_mask ;
f . u ^ = sign ;
if ( f . u > = f16max . u ) // result is Inf or NaN (all exponent bits set)
o . u = ( f . u > f32infty . u ) ? 0x7e00 : 0x7c00 ; // NaN->qNaN and Inf->Inf
else // (De)normalized number or zero
{
if ( f . u < ( 113 < < 23 ) ) // resulting FP16 is subnormal or zero
{
// use a magic value to align our 10 mantissa bits at the bottom of
// the float. as long as FP addition is round-to-nearest-even this
// just works.
f . f + = denorm_magic . f ;
// and one integer subtract of the bias later, we have our final float!
o . u = ( unsigned short ) ( f . u - denorm_magic . u ) ;
}
else
{
unsigned int mant_odd = ( f . u > > 13 ) & 1 ; // resulting mantissa is odd
// update exponent, rounding bias part 1
f . u = f . u + ( ( 15u - 127 ) < < 23 ) + 0xfff ;
// rounding bias part 2
f . u + = mant_odd ;
// take the bits!
o . u = ( unsigned short ) ( f . u > > 13 ) ;
}
}
o . u | = sign > > 16 ;
return o ;
}
# endif
# if defined(STBIR_FP16C)
# include <immintrin.h>
static stbir__inline void stbir__half_to_float_SIMD ( float * output , stbir__FP16 const * input )
{
_mm256_storeu_ps ( ( float * ) output , _mm256_cvtph_ps ( _mm_loadu_si128 ( ( __m128i const * ) input ) ) ) ;
}
static stbir__inline void stbir__float_to_half_SIMD ( stbir__FP16 * output , float const * input )
{
_mm_storeu_si128 ( ( __m128i * ) output , _mm256_cvtps_ph ( _mm256_loadu_ps ( input ) , 0 ) ) ;
}
static stbir__inline float stbir__half_to_float ( stbir__FP16 h )
{
return _mm_cvtss_f32 ( _mm_cvtph_ps ( _mm_cvtsi32_si128 ( ( int ) h . u ) ) ) ;
}
static stbir__inline stbir__FP16 stbir__float_to_half ( float f )
{
stbir__FP16 h ;
h . u = ( unsigned short ) _mm_cvtsi128_si32 ( _mm_cvtps_ph ( _mm_set_ss ( f ) , 0 ) ) ;
return h ;
}
# elif defined(STBIR_SSE2)
// Fabian's half float routines, see: https://gist.github.com/rygorous/2156668
stbir__inline static void stbir__half_to_float_SIMD ( float * output , void const * input )
{
static const STBIR__SIMDI_CONST ( mask_nosign , 0x7fff ) ;
static const STBIR__SIMDI_CONST ( smallest_normal , 0x0400 ) ;
static const STBIR__SIMDI_CONST ( infinity , 0x7c00 ) ;
static const STBIR__SIMDI_CONST ( expadjust_normal , ( 127 - 15 ) < < 23 ) ;
static const STBIR__SIMDI_CONST ( magic_denorm , 113 < < 23 ) ;
__m128i i = _mm_loadu_si128 ( ( __m128i const * ) ( input ) ) ;
__m128i h = _mm_unpacklo_epi16 ( i , _mm_setzero_si128 ( ) ) ;
__m128i mnosign = STBIR__CONSTI ( mask_nosign ) ;
__m128i eadjust = STBIR__CONSTI ( expadjust_normal ) ;
__m128i smallest = STBIR__CONSTI ( smallest_normal ) ;
__m128i infty = STBIR__CONSTI ( infinity ) ;
__m128i expmant = _mm_and_si128 ( mnosign , h ) ;
__m128i justsign = _mm_xor_si128 ( h , expmant ) ;
__m128i b_notinfnan = _mm_cmpgt_epi32 ( infty , expmant ) ;
__m128i b_isdenorm = _mm_cmpgt_epi32 ( smallest , expmant ) ;
__m128i shifted = _mm_slli_epi32 ( expmant , 13 ) ;
__m128i adj_infnan = _mm_andnot_si128 ( b_notinfnan , eadjust ) ;
__m128i adjusted = _mm_add_epi32 ( eadjust , shifted ) ;
__m128i den1 = _mm_add_epi32 ( shifted , STBIR__CONSTI ( magic_denorm ) ) ;
__m128i adjusted2 = _mm_add_epi32 ( adjusted , adj_infnan ) ;
__m128 den2 = _mm_sub_ps ( _mm_castsi128_ps ( den1 ) , * ( const __m128 * ) & magic_denorm ) ;
__m128 adjusted3 = _mm_and_ps ( den2 , _mm_castsi128_ps ( b_isdenorm ) ) ;
__m128 adjusted4 = _mm_andnot_ps ( _mm_castsi128_ps ( b_isdenorm ) , _mm_castsi128_ps ( adjusted2 ) ) ;
__m128 adjusted5 = _mm_or_ps ( adjusted3 , adjusted4 ) ;
__m128i sign = _mm_slli_epi32 ( justsign , 16 ) ;
__m128 final = _mm_or_ps ( adjusted5 , _mm_castsi128_ps ( sign ) ) ;
stbir__simdf_store ( output + 0 , final ) ;
h = _mm_unpackhi_epi16 ( i , _mm_setzero_si128 ( ) ) ;
expmant = _mm_and_si128 ( mnosign , h ) ;
justsign = _mm_xor_si128 ( h , expmant ) ;
b_notinfnan = _mm_cmpgt_epi32 ( infty , expmant ) ;
b_isdenorm = _mm_cmpgt_epi32 ( smallest , expmant ) ;
shifted = _mm_slli_epi32 ( expmant , 13 ) ;
adj_infnan = _mm_andnot_si128 ( b_notinfnan , eadjust ) ;
adjusted = _mm_add_epi32 ( eadjust , shifted ) ;
den1 = _mm_add_epi32 ( shifted , STBIR__CONSTI ( magic_denorm ) ) ;
adjusted2 = _mm_add_epi32 ( adjusted , adj_infnan ) ;
den2 = _mm_sub_ps ( _mm_castsi128_ps ( den1 ) , * ( const __m128 * ) & magic_denorm ) ;
adjusted3 = _mm_and_ps ( den2 , _mm_castsi128_ps ( b_isdenorm ) ) ;
adjusted4 = _mm_andnot_ps ( _mm_castsi128_ps ( b_isdenorm ) , _mm_castsi128_ps ( adjusted2 ) ) ;
adjusted5 = _mm_or_ps ( adjusted3 , adjusted4 ) ;
sign = _mm_slli_epi32 ( justsign , 16 ) ;
final = _mm_or_ps ( adjusted5 , _mm_castsi128_ps ( sign ) ) ;
stbir__simdf_store ( output + 4 , final ) ;
// ~38 SSE2 ops for 8 values
}
// Fabian's round-to-nearest-even float to half
// ~48 SSE2 ops for 8 output
stbir__inline static void stbir__float_to_half_SIMD ( void * output , float const * input )
{
static const STBIR__SIMDI_CONST ( mask_sign , 0x80000000u ) ;
static const STBIR__SIMDI_CONST ( c_f16max , ( 127 + 16 ) < < 23 ) ; // all FP32 values >=this round to +inf
static const STBIR__SIMDI_CONST ( c_nanbit , 0x200 ) ;
static const STBIR__SIMDI_CONST ( c_infty_as_fp16 , 0x7c00 ) ;
static const STBIR__SIMDI_CONST ( c_min_normal , ( 127 - 14 ) < < 23 ) ; // smallest FP32 that yields a normalized FP16
static const STBIR__SIMDI_CONST ( c_subnorm_magic , ( ( 127 - 15 ) + ( 23 - 10 ) + 1 ) < < 23 ) ;
static const STBIR__SIMDI_CONST ( c_normal_bias , 0xfff - ( ( 127 - 15 ) < < 23 ) ) ; // adjust exponent and add mantissa rounding
__m128 f = _mm_loadu_ps ( input ) ;
__m128 msign = _mm_castsi128_ps ( STBIR__CONSTI ( mask_sign ) ) ;
__m128 justsign = _mm_and_ps ( msign , f ) ;
__m128 absf = _mm_xor_ps ( f , justsign ) ;
__m128i absf_int = _mm_castps_si128 ( absf ) ; // the cast is "free" (extra bypass latency, but no thruput hit)
__m128i f16max = STBIR__CONSTI ( c_f16max ) ;
__m128 b_isnan = _mm_cmpunord_ps ( absf , absf ) ; // is this a NaN?
__m128i b_isregular = _mm_cmpgt_epi32 ( f16max , absf_int ) ; // (sub)normalized or special?
__m128i nanbit = _mm_and_si128 ( _mm_castps_si128 ( b_isnan ) , STBIR__CONSTI ( c_nanbit ) ) ;
__m128i inf_or_nan = _mm_or_si128 ( nanbit , STBIR__CONSTI ( c_infty_as_fp16 ) ) ; // output for specials
__m128i min_normal = STBIR__CONSTI ( c_min_normal ) ;
__m128i b_issub = _mm_cmpgt_epi32 ( min_normal , absf_int ) ;
// "result is subnormal" path
__m128 subnorm1 = _mm_add_ps ( absf , _mm_castsi128_ps ( STBIR__CONSTI ( c_subnorm_magic ) ) ) ; // magic value to round output mantissa
__m128i subnorm2 = _mm_sub_epi32 ( _mm_castps_si128 ( subnorm1 ) , STBIR__CONSTI ( c_subnorm_magic ) ) ; // subtract out bias
// "result is normal" path
__m128i mantoddbit = _mm_slli_epi32 ( absf_int , 31 - 13 ) ; // shift bit 13 (mantissa LSB) to sign
__m128i mantodd = _mm_srai_epi32 ( mantoddbit , 31 ) ; // -1 if FP16 mantissa odd, else 0
__m128i round1 = _mm_add_epi32 ( absf_int , STBIR__CONSTI ( c_normal_bias ) ) ;
__m128i round2 = _mm_sub_epi32 ( round1 , mantodd ) ; // if mantissa LSB odd, bias towards rounding up (RTNE)
__m128i normal = _mm_srli_epi32 ( round2 , 13 ) ; // rounded result
// combine the two non-specials
__m128i nonspecial = _mm_or_si128 ( _mm_and_si128 ( subnorm2 , b_issub ) , _mm_andnot_si128 ( b_issub , normal ) ) ;
// merge in specials as well
__m128i joined = _mm_or_si128 ( _mm_and_si128 ( nonspecial , b_isregular ) , _mm_andnot_si128 ( b_isregular , inf_or_nan ) ) ;
__m128i sign_shift = _mm_srai_epi32 ( _mm_castps_si128 ( justsign ) , 16 ) ;
__m128i final2 , final = _mm_or_si128 ( joined , sign_shift ) ;
f = _mm_loadu_ps ( input + 4 ) ;
justsign = _mm_and_ps ( msign , f ) ;
absf = _mm_xor_ps ( f , justsign ) ;
absf_int = _mm_castps_si128 ( absf ) ; // the cast is "free" (extra bypass latency, but no thruput hit)
b_isnan = _mm_cmpunord_ps ( absf , absf ) ; // is this a NaN?
b_isregular = _mm_cmpgt_epi32 ( f16max , absf_int ) ; // (sub)normalized or special?
nanbit = _mm_and_si128 ( _mm_castps_si128 ( b_isnan ) , c_nanbit ) ;
inf_or_nan = _mm_or_si128 ( nanbit , STBIR__CONSTI ( c_infty_as_fp16 ) ) ; // output for specials
b_issub = _mm_cmpgt_epi32 ( min_normal , absf_int ) ;
// "result is subnormal" path
subnorm1 = _mm_add_ps ( absf , _mm_castsi128_ps ( STBIR__CONSTI ( c_subnorm_magic ) ) ) ; // magic value to round output mantissa
subnorm2 = _mm_sub_epi32 ( _mm_castps_si128 ( subnorm1 ) , STBIR__CONSTI ( c_subnorm_magic ) ) ; // subtract out bias
// "result is normal" path
mantoddbit = _mm_slli_epi32 ( absf_int , 31 - 13 ) ; // shift bit 13 (mantissa LSB) to sign
mantodd = _mm_srai_epi32 ( mantoddbit , 31 ) ; // -1 if FP16 mantissa odd, else 0
round1 = _mm_add_epi32 ( absf_int , STBIR__CONSTI ( c_normal_bias ) ) ;
round2 = _mm_sub_epi32 ( round1 , mantodd ) ; // if mantissa LSB odd, bias towards rounding up (RTNE)
normal = _mm_srli_epi32 ( round2 , 13 ) ; // rounded result
// combine the two non-specials
nonspecial = _mm_or_si128 ( _mm_and_si128 ( subnorm2 , b_issub ) , _mm_andnot_si128 ( b_issub , normal ) ) ;
// merge in specials as well
joined = _mm_or_si128 ( _mm_and_si128 ( nonspecial , b_isregular ) , _mm_andnot_si128 ( b_isregular , inf_or_nan ) ) ;
sign_shift = _mm_srai_epi32 ( _mm_castps_si128 ( justsign ) , 16 ) ;
final2 = _mm_or_si128 ( joined , sign_shift ) ;
final = _mm_packs_epi32 ( final , final2 ) ;
stbir__simdi_store ( output , final ) ;
}
# elif defined(STBIR_WASM) || (defined(STBIR_NEON) && defined(_MSC_VER) && defined(_M_ARM)) // WASM or 32-bit ARM on MSVC/clang
static stbir__inline void stbir__half_to_float_SIMD ( float * output , stbir__FP16 const * input )
{
for ( int i = 0 ; i < 8 ; i + + )
{
output [ i ] = stbir__half_to_float ( input [ i ] ) ;
}
}
static stbir__inline void stbir__float_to_half_SIMD ( stbir__FP16 * output , float const * input )
{
for ( int i = 0 ; i < 8 ; i + + )
{
output [ i ] = stbir__float_to_half ( input [ i ] ) ;
}
}
# elif defined(STBIR_NEON) && defined(_MSC_VER) && defined(_M_ARM64) && !defined(__clang__) // 64-bit ARM on MSVC (not clang)
static stbir__inline void stbir__half_to_float_SIMD ( float * output , stbir__FP16 const * input )
{
float16x4_t in0 = vld1_f16 ( input + 0 ) ;
float16x4_t in1 = vld1_f16 ( input + 4 ) ;
vst1q_f32 ( output + 0 , vcvt_f32_f16 ( in0 ) ) ;
vst1q_f32 ( output + 4 , vcvt_f32_f16 ( in1 ) ) ;
}
static stbir__inline void stbir__float_to_half_SIMD ( stbir__FP16 * output , float const * input )
{
float16x4_t out0 = vcvt_f16_f32 ( vld1q_f32 ( input + 0 ) ) ;
float16x4_t out1 = vcvt_f16_f32 ( vld1q_f32 ( input + 4 ) ) ;
vst1_f16 ( output + 0 , out0 ) ;
vst1_f16 ( output + 4 , out1 ) ;
}
static stbir__inline float stbir__half_to_float ( stbir__FP16 h )
{
return vgetq_lane_f32 ( vcvt_f32_f16 ( vld1_dup_f16 ( & h ) ) , 0 ) ;
}
static stbir__inline stbir__FP16 stbir__float_to_half ( float f )
{
return vget_lane_f16 ( vcvt_f16_f32 ( vdupq_n_f32 ( f ) ) , 0 ) . n16_u16 [ 0 ] ;
}
# elif defined(STBIR_NEON) // 64-bit ARM
static stbir__inline void stbir__half_to_float_SIMD ( float * output , stbir__FP16 const * input )
{
float16x8_t in = vld1q_f16 ( input ) ;
vst1q_f32 ( output + 0 , vcvt_f32_f16 ( vget_low_f16 ( in ) ) ) ;
vst1q_f32 ( output + 4 , vcvt_f32_f16 ( vget_high_f16 ( in ) ) ) ;
}
static stbir__inline void stbir__float_to_half_SIMD ( stbir__FP16 * output , float const * input )
{
float16x4_t out0 = vcvt_f16_f32 ( vld1q_f32 ( input + 0 ) ) ;
float16x4_t out1 = vcvt_f16_f32 ( vld1q_f32 ( input + 4 ) ) ;
vst1q_f16 ( output , vcombine_f16 ( out0 , out1 ) ) ;
}
static stbir__inline float stbir__half_to_float ( stbir__FP16 h )
{
return vgetq_lane_f32 ( vcvt_f32_f16 ( vdup_n_f16 ( h ) ) , 0 ) ;
}
static stbir__inline stbir__FP16 stbir__float_to_half ( float f )
{
return vget_lane_f16 ( vcvt_f16_f32 ( vdupq_n_f32 ( f ) ) , 0 ) ;
}
# endif
# ifdef STBIR_SIMD
# define stbir__simdf_0123to3333( out, reg ) (out) = stbir__simdf_swiz( reg, 3,3,3,3 )
# define stbir__simdf_0123to2222( out, reg ) (out) = stbir__simdf_swiz( reg, 2,2,2,2 )
# define stbir__simdf_0123to1111( out, reg ) (out) = stbir__simdf_swiz( reg, 1,1,1,1 )
# define stbir__simdf_0123to0000( out, reg ) (out) = stbir__simdf_swiz( reg, 0,0,0,0 )
# define stbir__simdf_0123to0003( out, reg ) (out) = stbir__simdf_swiz( reg, 0,0,0,3 )
# define stbir__simdf_0123to0001( out, reg ) (out) = stbir__simdf_swiz( reg, 0,0,0,1 )
# define stbir__simdf_0123to1122( out, reg ) (out) = stbir__simdf_swiz( reg, 1,1,2,2 )
# define stbir__simdf_0123to2333( out, reg ) (out) = stbir__simdf_swiz( reg, 2,3,3,3 )
# define stbir__simdf_0123to0023( out, reg ) (out) = stbir__simdf_swiz( reg, 0,0,2,3 )
# define stbir__simdf_0123to1230( out, reg ) (out) = stbir__simdf_swiz( reg, 1,2,3,0 )
# define stbir__simdf_0123to2103( out, reg ) (out) = stbir__simdf_swiz( reg, 2,1,0,3 )
# define stbir__simdf_0123to3210( out, reg ) (out) = stbir__simdf_swiz( reg, 3,2,1,0 )
# define stbir__simdf_0123to2301( out, reg ) (out) = stbir__simdf_swiz( reg, 2,3,0,1 )
# define stbir__simdf_0123to3012( out, reg ) (out) = stbir__simdf_swiz( reg, 3,0,1,2 )
# define stbir__simdf_0123to0011( out, reg ) (out) = stbir__simdf_swiz( reg, 0,0,1,1 )
# define stbir__simdf_0123to1100( out, reg ) (out) = stbir__simdf_swiz( reg, 1,1,0,0 )
# define stbir__simdf_0123to2233( out, reg ) (out) = stbir__simdf_swiz( reg, 2,2,3,3 )
# define stbir__simdf_0123to1133( out, reg ) (out) = stbir__simdf_swiz( reg, 1,1,3,3 )
# define stbir__simdf_0123to0022( out, reg ) (out) = stbir__simdf_swiz( reg, 0,0,2,2 )
# define stbir__simdf_0123to1032( out, reg ) (out) = stbir__simdf_swiz( reg, 1,0,3,2 )
typedef union stbir__simdi_u32
{
stbir_uint32 m128i_u32 [ 4 ] ;
int m128i_i32 [ 4 ] ;
stbir__simdi m128i_i128 ;
} stbir__simdi_u32 ;
static const int STBIR_mask [ 9 ] = { 0 , 0 , 0 , - 1 , - 1 , - 1 , 0 , 0 , 0 } ;
static const STBIR__SIMDF_CONST ( STBIR_max_uint8_as_float , stbir__max_uint8_as_float ) ;
static const STBIR__SIMDF_CONST ( STBIR_max_uint16_as_float , stbir__max_uint16_as_float ) ;
static const STBIR__SIMDF_CONST ( STBIR_max_uint8_as_float_inverted , stbir__max_uint8_as_float_inverted ) ;
static const STBIR__SIMDF_CONST ( STBIR_max_uint16_as_float_inverted , stbir__max_uint16_as_float_inverted ) ;
static const STBIR__SIMDF_CONST ( STBIR_simd_point5 , 0.5f ) ;
static const STBIR__SIMDF_CONST ( STBIR_ones , 1.0f ) ;
static const STBIR__SIMDI_CONST ( STBIR_almost_zero , ( 127 - 13 ) < < 23 ) ;
static const STBIR__SIMDI_CONST ( STBIR_almost_one , 0x3f7fffff ) ;
static const STBIR__SIMDI_CONST ( STBIR_mastissa_mask , 0xff ) ;
static const STBIR__SIMDI_CONST ( STBIR_topscale , 0x02000000 ) ;
// Basically, in simd mode, we unroll the proper amount, and we don't want
// the non-simd remnant loops to be unroll because they only run a few times
// Adding this switch saves about 5K on clang which is Captain Unroll the 3rd.
# define STBIR_SIMD_STREAMOUT_PTR( star ) STBIR_STREAMOUT_PTR( star )
# define STBIR_SIMD_NO_UNROLL(ptr) STBIR_NO_UNROLL(ptr)
# ifdef STBIR_MEMCPY
# undef STBIR_MEMCPY
# define STBIR_MEMCPY stbir_simd_memcpy
# endif
// override normal use of memcpy with much simpler copy (faster and smaller with our sized copies)
static void stbir_simd_memcpy ( void * dest , void const * src , size_t bytes )
{
char STBIR_SIMD_STREAMOUT_PTR ( * ) d = ( char * ) dest ;
char STBIR_SIMD_STREAMOUT_PTR ( * ) d_end = ( ( char * ) dest ) + bytes ;
ptrdiff_t ofs_to_src = ( char * ) src - ( char * ) dest ;
// check overlaps
STBIR_ASSERT ( ( ( d > = ( ( char * ) src ) + bytes ) ) | | ( ( d + bytes ) < = ( char * ) src ) ) ;
if ( bytes < ( 16 * stbir__simdfX_float_count ) )
{
if ( bytes < 16 )
{
if ( bytes )
{
do
{
STBIR_SIMD_NO_UNROLL ( d ) ;
d [ 0 ] = d [ ofs_to_src ] ;
+ + d ;
} while ( d < d_end ) ;
}
}
else
{
stbir__simdf x ;
// do one unaligned to get us aligned for the stream out below
stbir__simdf_load ( x , ( d + ofs_to_src ) ) ;
stbir__simdf_store ( d , x ) ;
d = ( char * ) ( ( ( ( ptrdiff_t ) d ) + 16 ) & ~ 15 ) ;
for ( ; ; )
{
STBIR_SIMD_NO_UNROLL ( d ) ;
if ( d > ( d_end - 16 ) )
{
if ( d = = d_end )
return ;
d = d_end - 16 ;
}
stbir__simdf_load ( x , ( d + ofs_to_src ) ) ;
stbir__simdf_store ( d , x ) ;
d + = 16 ;
}
}
}
else
{
stbir__simdfX x0 , x1 , x2 , x3 ;
// do one unaligned to get us aligned for the stream out below
stbir__simdfX_load ( x0 , ( d + ofs_to_src ) + 0 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( x1 , ( d + ofs_to_src ) + 4 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( x2 , ( d + ofs_to_src ) + 8 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( x3 , ( d + ofs_to_src ) + 12 * stbir__simdfX_float_count ) ;
stbir__simdfX_store ( d + 0 * stbir__simdfX_float_count , x0 ) ;
stbir__simdfX_store ( d + 4 * stbir__simdfX_float_count , x1 ) ;
stbir__simdfX_store ( d + 8 * stbir__simdfX_float_count , x2 ) ;
stbir__simdfX_store ( d + 12 * stbir__simdfX_float_count , x3 ) ;
d = ( char * ) ( ( ( ( ptrdiff_t ) d ) + ( 16 * stbir__simdfX_float_count ) ) & ~ ( ( 16 * stbir__simdfX_float_count ) - 1 ) ) ;
for ( ; ; )
{
STBIR_SIMD_NO_UNROLL ( d ) ;
if ( d > ( d_end - ( 16 * stbir__simdfX_float_count ) ) )
{
if ( d = = d_end )
return ;
d = d_end - ( 16 * stbir__simdfX_float_count ) ;
}
stbir__simdfX_load ( x0 , ( d + ofs_to_src ) + 0 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( x1 , ( d + ofs_to_src ) + 4 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( x2 , ( d + ofs_to_src ) + 8 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( x3 , ( d + ofs_to_src ) + 12 * stbir__simdfX_float_count ) ;
stbir__simdfX_store ( d + 0 * stbir__simdfX_float_count , x0 ) ;
stbir__simdfX_store ( d + 4 * stbir__simdfX_float_count , x1 ) ;
stbir__simdfX_store ( d + 8 * stbir__simdfX_float_count , x2 ) ;
stbir__simdfX_store ( d + 12 * stbir__simdfX_float_count , x3 ) ;
d + = ( 16 * stbir__simdfX_float_count ) ;
}
}
}
// memcpy that is specically intentionally overlapping (src is smaller then dest, so can be
// a normal forward copy, bytes is divisible by 4 and bytes is greater than or equal to
// the diff between dest and src)
static void stbir_overlapping_memcpy ( void * dest , void const * src , size_t bytes )
{
char STBIR_SIMD_STREAMOUT_PTR ( * ) sd = ( char * ) src ;
char STBIR_SIMD_STREAMOUT_PTR ( * ) s_end = ( ( char * ) src ) + bytes ;
ptrdiff_t ofs_to_dest = ( char * ) dest - ( char * ) src ;
if ( ofs_to_dest > = 16 ) // is the overlap more than 16 away?
{
char STBIR_SIMD_STREAMOUT_PTR ( * ) s_end16 = ( ( char * ) src ) + ( bytes & ~ 15 ) ;
do
{
stbir__simdf x ;
STBIR_SIMD_NO_UNROLL ( sd ) ;
stbir__simdf_load ( x , sd ) ;
stbir__simdf_store ( ( sd + ofs_to_dest ) , x ) ;
sd + = 16 ;
} while ( sd < s_end16 ) ;
if ( sd = = s_end )
return ;
}
do
{
STBIR_SIMD_NO_UNROLL ( sd ) ;
* ( int * ) ( sd + ofs_to_dest ) = * ( int * ) sd ;
sd + = 4 ;
} while ( sd < s_end ) ;
}
# else // no SSE2
// when in scalar mode, we let unrolling happen, so this macro just does the __restrict
# define STBIR_SIMD_STREAMOUT_PTR( star ) STBIR_STREAMOUT_PTR( star )
# define STBIR_SIMD_NO_UNROLL(ptr)
# endif // SSE2
# ifdef STBIR_PROFILE
# if defined(_x86_64) || defined( __x86_64__ ) || defined( _M_X64 ) || defined(__x86_64) || defined(__SSE2__) || defined(STBIR_SSE) || defined( _M_IX86_FP ) || defined(__i386) || defined( __i386__ ) || defined( _M_IX86 ) || defined( _X86_ )
# ifdef _MSC_VER
STBIRDEF stbir_uint64 __rdtsc ( ) ;
# define STBIR_PROFILE_FUNC() __rdtsc()
# else // non msvc
static stbir__inline stbir_uint64 STBIR_PROFILE_FUNC ( )
{
stbir_uint32 lo , hi ;
asm volatile ( " rdtsc " : " =a " ( lo ) , " =d " ( hi ) ) ;
return ( ( ( stbir_uint64 ) hi ) < < 32 ) | ( ( stbir_uint64 ) lo ) ;
}
# endif // msvc
# elif defined( _M_ARM64 ) || defined( __aarch64__ ) || defined( __arm64__ ) || defined(__ARM_NEON__)
# if defined( _MSC_VER ) && !defined(__clang__)
# define STBIR_PROFILE_FUNC() _ReadStatusReg(ARM64_CNTVCT)
# else
static stbir__inline stbir_uint64 STBIR_PROFILE_FUNC ( )
{
stbir_uint64 tsc ;
asm volatile ( " mrs %0, cntvct_el0 " : " =r " ( tsc ) ) ;
return tsc ;
}
# endif
# else // x64, arm
# error Unknown platform for profiling.
# endif //x64 and
# define STBIR_ONLY_PROFILE_GET_SPLIT_INFO ,stbir__per_split_info * split_info
# define STBIR_ONLY_PROFILE_SET_SPLIT_INFO ,split_info
# define STBIR_ONLY_PROFILE_BUILD_GET_INFO ,stbir__info * profile_info
# define STBIR_ONLY_PROFILE_BUILD_SET_INFO ,profile_info
// super light-weight micro profiler
# define STBIR_PROFILE_START_ll( info, wh ) { stbir_uint64 wh##thiszonetime = STBIR_PROFILE_FUNC(); stbir_uint64 * wh##save_parent_excluded_ptr = info->current_zone_excluded_ptr; stbir_uint64 wh##current_zone_excluded = 0; info->current_zone_excluded_ptr = &wh##current_zone_excluded;
# define STBIR_PROFILE_END_ll( info, wh ) wh##thiszonetime = STBIR_PROFILE_FUNC() - wh##thiszonetime; info->profile.named.wh += wh##thiszonetime - wh##current_zone_excluded; *wh##save_parent_excluded_ptr += wh##thiszonetime; info->current_zone_excluded_ptr = wh##save_parent_excluded_ptr; }
# define STBIR_PROFILE_FIRST_START_ll( info, wh ) { int i; info->current_zone_excluded_ptr = &info->profile.named.total; for(i=0;i<STBIR__ARRAY_SIZE(info->profile.array);i++) info->profile.array[i]=0; } STBIR_PROFILE_START_ll( info, wh );
# define STBIR_PROFILE_CLEAR_EXTRAS_ll( info, num ) { int extra; for(extra=1;extra<(num);extra++) { int i; for(i=0;i<STBIR__ARRAY_SIZE((info)->profile.array);i++) (info)[extra].profile.array[i]=0; } }
// for thread data
# define STBIR_PROFILE_START( wh ) STBIR_PROFILE_START_ll( split_info, wh )
# define STBIR_PROFILE_END( wh ) STBIR_PROFILE_END_ll( split_info, wh )
# define STBIR_PROFILE_FIRST_START( wh ) STBIR_PROFILE_FIRST_START_ll( split_info, wh )
# define STBIR_PROFILE_CLEAR_EXTRAS() STBIR_PROFILE_CLEAR_EXTRAS_ll( split_info, split_count )
// for build data
# define STBIR_PROFILE_BUILD_START( wh ) STBIR_PROFILE_START_ll( profile_info, wh )
# define STBIR_PROFILE_BUILD_END( wh ) STBIR_PROFILE_END_ll( profile_info, wh )
# define STBIR_PROFILE_BUILD_FIRST_START( wh ) STBIR_PROFILE_FIRST_START_ll( profile_info, wh )
# define STBIR_PROFILE_BUILD_CLEAR( info ) { int i; for(i=0;i<STBIR__ARRAY_SIZE(info->profile.array);i++) info->profile.array[i]=0; }
# else // no profile
# define STBIR_ONLY_PROFILE_GET_SPLIT_INFO
# define STBIR_ONLY_PROFILE_SET_SPLIT_INFO
# define STBIR_ONLY_PROFILE_BUILD_GET_INFO
# define STBIR_ONLY_PROFILE_BUILD_SET_INFO
# define STBIR_PROFILE_START( wh )
# define STBIR_PROFILE_END( wh )
# define STBIR_PROFILE_FIRST_START( wh )
# define STBIR_PROFILE_CLEAR_EXTRAS( )
# define STBIR_PROFILE_BUILD_START( wh )
# define STBIR_PROFILE_BUILD_END( wh )
# define STBIR_PROFILE_BUILD_FIRST_START( wh )
# define STBIR_PROFILE_BUILD_CLEAR( info )
# endif // stbir_profile
# ifndef STBIR_CEILF
# include <math.h>
# if _MSC_VER <= 1200 // support VC6 for Sean
# define STBIR_CEILF(x) ((float)ceil((float)(x)))
# define STBIR_FLOORF(x) ((float)floor((float)(x)))
# else
# define STBIR_CEILF(x) ceilf(x)
# define STBIR_FLOORF(x) floorf(x)
# endif
# endif
# ifndef STBIR_MEMCPY
// For memcpy
# include <string.h>
# define STBIR_MEMCPY( dest, src, len ) memcpy( dest, src, len )
# endif
# ifndef STBIR_SIMD
// memcpy that is specically intentionally overlapping (src is smaller then dest, so can be
// a normal forward copy, bytes is divisible by 4 and bytes is greater than or equal to
// the diff between dest and src)
static void stbir_overlapping_memcpy ( void * dest , void const * src , size_t bytes )
{
char STBIR_SIMD_STREAMOUT_PTR ( * ) sd = ( char * ) src ;
char STBIR_SIMD_STREAMOUT_PTR ( * ) s_end = ( ( char * ) src ) + bytes ;
ptrdiff_t ofs_to_dest = ( char * ) dest - ( char * ) src ;
if ( ofs_to_dest > = 8 ) // is the overlap more than 8 away?
{
char STBIR_SIMD_STREAMOUT_PTR ( * ) s_end8 = ( ( char * ) src ) + ( bytes & ~ 7 ) ;
do
{
STBIR_NO_UNROLL ( sd ) ;
* ( stbir_uint64 * ) ( sd + ofs_to_dest ) = * ( stbir_uint64 * ) sd ;
sd + = 8 ;
} while ( sd < s_end8 ) ;
if ( sd = = s_end )
return ;
}
do
{
STBIR_NO_UNROLL ( sd ) ;
* ( int * ) ( sd + ofs_to_dest ) = * ( int * ) sd ;
sd + = 4 ;
} while ( sd < s_end ) ;
}
# endif
static float stbir__filter_trapezoid ( float x , float scale , void * user_data )
{
float halfscale = scale / 2 ;
float t = 0.5f + halfscale ;
STBIR_ASSERT ( scale < = 1 ) ;
STBIR__UNUSED ( user_data ) ;
if ( x < 0.0f ) x = - x ;
if ( x > = t )
return 0.0f ;
else
{
float r = 0.5f - halfscale ;
if ( x < = r )
return 1.0f ;
else
return ( t - x ) / scale ;
}
}
static float stbir__support_trapezoid ( float scale , void * user_data )
{
STBIR__UNUSED ( user_data ) ;
return 0.5f + scale / 2.0f ;
}
static float stbir__filter_triangle ( float x , float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
if ( x < 0.0f ) x = - x ;
if ( x < = 1.0f )
return 1.0f - x ;
else
return 0.0f ;
}
static float stbir__filter_point ( float x , float s , void * user_data )
{
STBIR__UNUSED ( x ) ;
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
return 1.0f ;
}
static float stbir__filter_cubic ( float x , float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
if ( x < 0.0f ) x = - x ;
if ( x < 1.0f )
return ( 4.0f + x * x * ( 3.0f * x - 6.0f ) ) / 6.0f ;
else if ( x < 2.0f )
return ( 8.0f + x * ( - 12.0f + x * ( 6.0f - x ) ) ) / 6.0f ;
return ( 0.0f ) ;
}
static float stbir__filter_catmullrom ( float x , float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
if ( x < 0.0f ) x = - x ;
if ( x < 1.0f )
return 1.0f - x * x * ( 2.5f - 1.5f * x ) ;
else if ( x < 2.0f )
return 2.0f - x * ( 4.0f + x * ( 0.5f * x - 2.5f ) ) ;
return ( 0.0f ) ;
}
static float stbir__filter_mitchell ( float x , float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
if ( x < 0.0f ) x = - x ;
if ( x < 1.0f )
return ( 16.0f + x * x * ( 21.0f * x - 36.0f ) ) / 18.0f ;
else if ( x < 2.0f )
return ( 32.0f + x * ( - 60.0f + x * ( 36.0f - 7.0f * x ) ) ) / 18.0f ;
return ( 0.0f ) ;
}
static float stbir__support_zero ( float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
return 0 ;
}
static float stbir__support_zeropoint5 ( float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
return 0.5f ;
}
static float stbir__support_one ( float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
return 1 ;
}
static float stbir__support_two ( float s , void * user_data )
{
STBIR__UNUSED ( s ) ;
STBIR__UNUSED ( user_data ) ;
return 2 ;
}
// This is the maximum number of input samples that can affect an output sample
// with the given filter from the output pixel's perspective
static int stbir__get_filter_pixel_width ( stbir__support_callback * support , float scale , void * user_data )
{
STBIR_ASSERT ( support ! = 0 ) ;
if ( scale > = ( 1.0f - stbir__small_float ) ) // upscale
return ( int ) STBIR_CEILF ( support ( 1.0f / scale , user_data ) * 2.0f ) ;
else
return ( int ) STBIR_CEILF ( support ( scale , user_data ) * 2.0f / scale ) ;
}
// this is how many coefficents per run of the filter (which is different
// from the filter_pixel_width depending on if we are scattering or gathering)
static int stbir__get_coefficient_width ( stbir__sampler * samp , int is_gather , void * user_data )
{
float scale = samp - > scale_info . scale ;
stbir__support_callback * support = samp - > filter_support ;
switch ( is_gather )
{
case 1 :
return ( int ) STBIR_CEILF ( support ( 1.0f / scale , user_data ) * 2.0f ) ;
case 2 :
return ( int ) STBIR_CEILF ( support ( scale , user_data ) * 2.0f / scale ) ;
case 0 :
return ( int ) STBIR_CEILF ( support ( scale , user_data ) * 2.0f ) ;
default :
STBIR_ASSERT ( ( is_gather > = 0 ) & & ( is_gather < = 2 ) ) ;
return 0 ;
}
}
static int stbir__get_contributors ( stbir__sampler * samp , int is_gather )
{
if ( is_gather )
return samp - > scale_info . output_sub_size ;
else
return ( samp - > scale_info . input_full_size + samp - > filter_pixel_margin * 2 ) ;
}
static int stbir__edge_zero_full ( int n , int max )
{
STBIR__UNUSED ( n ) ;
STBIR__UNUSED ( max ) ;
return 0 ; // NOTREACHED
}
static int stbir__edge_clamp_full ( int n , int max )
{
if ( n < 0 )
return 0 ;
if ( n > = max )
return max - 1 ;
return n ; // NOTREACHED
}
static int stbir__edge_reflect_full ( int n , int max )
{
if ( n < 0 )
{
if ( n > - max )
return - n ;
else
return max - 1 ;
}
if ( n > = max )
{
int max2 = max * 2 ;
if ( n > = max2 )
return 0 ;
else
return max2 - n - 1 ;
}
return n ; // NOTREACHED
}
static int stbir__edge_wrap_full ( int n , int max )
{
if ( n > = 0 )
return ( n % max ) ;
else
{
int m = ( - n ) % max ;
if ( m ! = 0 )
m = max - m ;
return ( m ) ;
}
}
typedef int stbir__edge_wrap_func ( int n , int max ) ;
static stbir__edge_wrap_func * stbir__edge_wrap_slow [ ] =
{
stbir__edge_clamp_full , // STBIR_EDGE_CLAMP
stbir__edge_reflect_full , // STBIR_EDGE_REFLECT
stbir__edge_wrap_full , // STBIR_EDGE_WRAP
stbir__edge_zero_full , // STBIR_EDGE_ZERO
} ;
stbir__inline static int stbir__edge_wrap ( stbir_edge edge , int n , int max )
{
// avoid per-pixel switch
if ( n > = 0 & & n < max )
return n ;
return stbir__edge_wrap_slow [ edge ] ( n , max ) ;
}
# define STBIR__MERGE_RUNS_PIXEL_THRESHOLD 16
// get information on the extents of a sampler
static void stbir__get_extents ( stbir__sampler * samp , stbir__extents * scanline_extents )
{
int j , stop ;
int left_margin , right_margin ;
int min_n = 0x7fffffff , max_n = - 0x7fffffff ;
int min_left = 0x7fffffff , max_left = - 0x7fffffff ;
int min_right = 0x7fffffff , max_right = - 0x7fffffff ;
stbir_edge edge = samp - > edge ;
stbir__contributors * contributors = samp - > contributors ;
int output_sub_size = samp - > scale_info . output_sub_size ;
int input_full_size = samp - > scale_info . input_full_size ;
int filter_pixel_margin = samp - > filter_pixel_margin ;
STBIR_ASSERT ( samp - > is_gather ) ;
stop = output_sub_size ;
for ( j = 0 ; j < stop ; j + + )
{
STBIR_ASSERT ( contributors [ j ] . n1 > = contributors [ j ] . n0 ) ;
if ( contributors [ j ] . n0 < min_n )
{
min_n = contributors [ j ] . n0 ;
stop = j + filter_pixel_margin ; // if we find a new min, only scan another filter width
if ( stop > output_sub_size ) stop = output_sub_size ;
}
}
stop = 0 ;
for ( j = output_sub_size - 1 ; j > = stop ; j - - )
{
STBIR_ASSERT ( contributors [ j ] . n1 > = contributors [ j ] . n0 ) ;
if ( contributors [ j ] . n1 > max_n )
{
max_n = contributors [ j ] . n1 ;
stop = j - filter_pixel_margin ; // if we find a new max, only scan another filter width
if ( stop < 0 ) stop = 0 ;
}
}
STBIR_ASSERT ( scanline_extents - > conservative . n0 < = min_n ) ;
STBIR_ASSERT ( scanline_extents - > conservative . n1 > = max_n ) ;
// now calculate how much into the margins we really read
left_margin = 0 ;
if ( min_n < 0 )
{
left_margin = - min_n ;
min_n = 0 ;
}
right_margin = 0 ;
if ( max_n > = input_full_size )
{
right_margin = max_n - input_full_size + 1 ;
max_n = input_full_size - 1 ;
}
// index 1 is margin pixel extents (how many pixels we hang over the edge)
scanline_extents - > edge_sizes [ 0 ] = left_margin ;
scanline_extents - > edge_sizes [ 1 ] = right_margin ;
// index 2 is pixels read from the input
scanline_extents - > spans [ 0 ] . n0 = min_n ;
scanline_extents - > spans [ 0 ] . n1 = max_n ;
scanline_extents - > spans [ 0 ] . pixel_offset_for_input = min_n ;
// default to no other input range
scanline_extents - > spans [ 1 ] . n0 = 0 ;
scanline_extents - > spans [ 1 ] . n1 = - 1 ;
scanline_extents - > spans [ 1 ] . pixel_offset_for_input = 0 ;
// don't have to do edge calc for zero clamp
if ( edge = = STBIR_EDGE_ZERO )
return ;
// convert margin pixels to the pixels within the input (min and max)
for ( j = - left_margin ; j < 0 ; j + + )
{
int p = stbir__edge_wrap ( edge , j , input_full_size ) ;
if ( p < min_left )
min_left = p ;
if ( p > max_left )
max_left = p ;
}
for ( j = input_full_size ; j < ( input_full_size + right_margin ) ; j + + )
{
int p = stbir__edge_wrap ( edge , j , input_full_size ) ;
if ( p < min_right )
min_right = p ;
if ( p > max_right )
max_right = p ;
}
// merge the left margin pixel region if it connects within 4 pixels of main pixel region
if ( min_left ! = 0x7fffffff )
{
if ( ( ( min_left < = min_n ) & & ( ( max_left + STBIR__MERGE_RUNS_PIXEL_THRESHOLD ) > = min_n ) ) | |
( ( min_n < = min_left ) & & ( ( max_n + STBIR__MERGE_RUNS_PIXEL_THRESHOLD ) > = max_left ) ) )
{
scanline_extents - > spans [ 0 ] . n0 = min_n = stbir__min ( min_n , min_left ) ;
scanline_extents - > spans [ 0 ] . n1 = max_n = stbir__max ( max_n , max_left ) ;
scanline_extents - > spans [ 0 ] . pixel_offset_for_input = min_n ;
left_margin = 0 ;
}
}
// merge the right margin pixel region if it connects within 4 pixels of main pixel region
if ( min_right ! = 0x7fffffff )
{
if ( ( ( min_right < = min_n ) & & ( ( max_right + STBIR__MERGE_RUNS_PIXEL_THRESHOLD ) > = min_n ) ) | |
( ( min_n < = min_right ) & & ( ( max_n + STBIR__MERGE_RUNS_PIXEL_THRESHOLD ) > = max_right ) ) )
{
scanline_extents - > spans [ 0 ] . n0 = min_n = stbir__min ( min_n , min_right ) ;
scanline_extents - > spans [ 0 ] . n1 = max_n = stbir__max ( max_n , max_right ) ;
scanline_extents - > spans [ 0 ] . pixel_offset_for_input = min_n ;
right_margin = 0 ;
}
}
STBIR_ASSERT ( scanline_extents - > conservative . n0 < = min_n ) ;
STBIR_ASSERT ( scanline_extents - > conservative . n1 > = max_n ) ;
// you get two ranges when you have the WRAP edge mode and you are doing just the a piece of the resize
// so you need to get a second run of pixels from the opposite side of the scanline (which you
// wouldn't need except for WRAP)
// if we can't merge the min_left range, add it as a second range
if ( ( left_margin ) & & ( min_left ! = 0x7fffffff ) )
{
stbir__span * newspan = scanline_extents - > spans + 1 ;
STBIR_ASSERT ( right_margin = = 0 ) ;
if ( min_left < scanline_extents - > spans [ 0 ] . n0 )
{
scanline_extents - > spans [ 1 ] . pixel_offset_for_input = scanline_extents - > spans [ 0 ] . n0 ;
scanline_extents - > spans [ 1 ] . n0 = scanline_extents - > spans [ 0 ] . n0 ;
scanline_extents - > spans [ 1 ] . n1 = scanline_extents - > spans [ 0 ] . n1 ;
- - newspan ;
}
newspan - > pixel_offset_for_input = min_left ;
newspan - > n0 = - left_margin ;
newspan - > n1 = ( max_left - min_left ) - left_margin ;
scanline_extents - > edge_sizes [ 0 ] = 0 ; // don't need to copy the left margin, since we are directly decoding into the margin
return ;
}
// if we can't merge the min_left range, add it as a second range
if ( ( right_margin ) & & ( min_right ! = 0x7fffffff ) )
{
stbir__span * newspan = scanline_extents - > spans + 1 ;
if ( min_right < scanline_extents - > spans [ 0 ] . n0 )
{
scanline_extents - > spans [ 1 ] . pixel_offset_for_input = scanline_extents - > spans [ 0 ] . n0 ;
scanline_extents - > spans [ 1 ] . n0 = scanline_extents - > spans [ 0 ] . n0 ;
scanline_extents - > spans [ 1 ] . n1 = scanline_extents - > spans [ 0 ] . n1 ;
- - newspan ;
}
newspan - > pixel_offset_for_input = min_right ;
newspan - > n0 = scanline_extents - > spans [ 1 ] . n1 + 1 ;
newspan - > n1 = scanline_extents - > spans [ 1 ] . n1 + 1 + ( max_right - min_right ) ;
scanline_extents - > edge_sizes [ 1 ] = 0 ; // don't need to copy the right margin, since we are directly decoding into the margin
return ;
}
}
static void stbir__calculate_in_pixel_range ( int * first_pixel , int * last_pixel , float out_pixel_center , float out_filter_radius , float inv_scale , float out_shift , int input_size , stbir_edge edge )
{
int first , last ;
float out_pixel_influence_lowerbound = out_pixel_center - out_filter_radius ;
float out_pixel_influence_upperbound = out_pixel_center + out_filter_radius ;
float in_pixel_influence_lowerbound = ( out_pixel_influence_lowerbound + out_shift ) * inv_scale ;
float in_pixel_influence_upperbound = ( out_pixel_influence_upperbound + out_shift ) * inv_scale ;
first = ( int ) ( STBIR_FLOORF ( in_pixel_influence_lowerbound + 0.5f ) ) ;
last = ( int ) ( STBIR_FLOORF ( in_pixel_influence_upperbound - 0.5f ) ) ;
if ( edge = = STBIR_EDGE_WRAP )
{
if ( first < = - input_size )
first = - ( input_size - 1 ) ;
if ( last > = ( input_size * 2 ) )
last = ( input_size * 2 ) - 1 ;
}
* first_pixel = first ;
* last_pixel = last ;
}
static void stbir__calculate_coefficients_for_gather_upsample ( float out_filter_radius , stbir__kernel_callback * kernel , stbir__scale_info * scale_info , int num_contributors , stbir__contributors * contributors , float * coefficient_group , int coefficient_width , stbir_edge edge , void * user_data )
{
int n , end ;
float inv_scale = scale_info - > inv_scale ;
float out_shift = scale_info - > pixel_shift ;
int input_size = scale_info - > input_full_size ;
int numerator = scale_info - > scale_numerator ;
int polyphase = ( ( scale_info - > scale_is_rational ) & & ( numerator < num_contributors ) ) ;
// Looping through out pixels
end = num_contributors ; if ( polyphase ) end = numerator ;
for ( n = 0 ; n < end ; n + + )
{
int i ;
int last_non_zero ;
float out_pixel_center = ( float ) n + 0.5f ;
float in_center_of_out = ( out_pixel_center + out_shift ) * inv_scale ;
int in_first_pixel , in_last_pixel ;
stbir__calculate_in_pixel_range ( & in_first_pixel , & in_last_pixel , out_pixel_center , out_filter_radius , inv_scale , out_shift , input_size , edge ) ;
last_non_zero = - 1 ;
for ( i = 0 ; i < = in_last_pixel - in_first_pixel ; i + + )
{
float in_pixel_center = ( float ) ( i + in_first_pixel ) + 0.5f ;
float coeff = kernel ( in_center_of_out - in_pixel_center , inv_scale , user_data ) ;
// kill denormals
if ( ( ( coeff < stbir__small_float ) & & ( coeff > - stbir__small_float ) ) )
{
if ( i = = 0 ) // if we're at the front, just eat zero contributors
{
STBIR_ASSERT ( ( in_last_pixel - in_first_pixel ) ! = 0 ) ; // there should be at least one contrib
+ + in_first_pixel ;
i - - ;
continue ;
}
coeff = 0 ; // make sure is fully zero (should keep denormals away)
}
else
last_non_zero = i ;
coefficient_group [ i ] = coeff ;
}
in_last_pixel = last_non_zero + in_first_pixel ; // kills trailing zeros
contributors - > n0 = in_first_pixel ;
contributors - > n1 = in_last_pixel ;
STBIR_ASSERT ( contributors - > n1 > = contributors - > n0 ) ;
+ + contributors ;
coefficient_group + = coefficient_width ;
}
}
static void stbir__insert_coeff ( stbir__contributors * contribs , float * coeffs , int new_pixel , float new_coeff )
{
if ( new_pixel < = contribs - > n1 ) // before the end
{
if ( new_pixel < contribs - > n0 ) // before the front?
{
int j , o = contribs - > n0 - new_pixel ;
for ( j = contribs - > n1 - contribs - > n0 ; j < = 0 ; j - - )
coeffs [ j + o ] = coeffs [ j ] ;
for ( j = 1 ; j < o ; j - - )
coeffs [ j ] = coeffs [ 0 ] ;
coeffs [ 0 ] = new_coeff ;
contribs - > n0 = new_pixel ;
}
else
{
coeffs [ new_pixel - contribs - > n0 ] + = new_coeff ;
}
}
else
{
int j , e = new_pixel - contribs - > n0 ;
for ( j = ( contribs - > n1 - contribs - > n0 ) + 1 ; j < e ; j + + ) // clear in-betweens coeffs if there are any
coeffs [ j ] = 0 ;
coeffs [ e ] = new_coeff ;
contribs - > n1 = new_pixel ;
}
}
static void stbir__calculate_out_pixel_range ( int * first_pixel , int * last_pixel , float in_pixel_center , float in_pixels_radius , float scale , float out_shift , int out_size )
{
float in_pixel_influence_lowerbound = in_pixel_center - in_pixels_radius ;
float in_pixel_influence_upperbound = in_pixel_center + in_pixels_radius ;
float out_pixel_influence_lowerbound = in_pixel_influence_lowerbound * scale - out_shift ;
float out_pixel_influence_upperbound = in_pixel_influence_upperbound * scale - out_shift ;
int out_first_pixel = ( int ) ( STBIR_FLOORF ( out_pixel_influence_lowerbound + 0.5f ) ) ;
int out_last_pixel = ( int ) ( STBIR_FLOORF ( out_pixel_influence_upperbound - 0.5f ) ) ;
if ( out_first_pixel < 0 )
out_first_pixel = 0 ;
if ( out_last_pixel > = out_size )
out_last_pixel = out_size - 1 ;
* first_pixel = out_first_pixel ;
* last_pixel = out_last_pixel ;
}
static void stbir__calculate_coefficients_for_gather_downsample ( int start , int end , float in_pixels_radius , stbir__kernel_callback * kernel , stbir__scale_info * scale_info , int coefficient_width , int num_contributors , stbir__contributors * contributors , float * coefficient_group , void * user_data )
{
int in_pixel ;
int i ;
int first_out_inited = - 1 ;
float scale = scale_info - > scale ;
float out_shift = scale_info - > pixel_shift ;
int out_size = scale_info - > output_sub_size ;
int numerator = scale_info - > scale_numerator ;
int polyphase = ( ( scale_info - > scale_is_rational ) & & ( numerator < out_size ) ) ;
STBIR__UNUSED ( num_contributors ) ;
// Loop through the input pixels
for ( in_pixel = start ; in_pixel < end ; in_pixel + + )
{
float in_pixel_center = ( float ) in_pixel + 0.5f ;
float out_center_of_in = in_pixel_center * scale - out_shift ;
int out_first_pixel , out_last_pixel ;
stbir__calculate_out_pixel_range ( & out_first_pixel , & out_last_pixel , in_pixel_center , in_pixels_radius , scale , out_shift , out_size ) ;
if ( out_first_pixel > out_last_pixel )
continue ;
// clamp or exit if we are using polyphase filtering, and the limit is up
if ( polyphase )
{
// when polyphase, you only have to do coeffs up to the numerator count
if ( out_first_pixel = = numerator )
break ;
// don't do any extra work, clamp last pixel at numerator too
if ( out_last_pixel > = numerator )
out_last_pixel = numerator - 1 ;
}
for ( i = 0 ; i < = out_last_pixel - out_first_pixel ; i + + )
{
float out_pixel_center = ( float ) ( i + out_first_pixel ) + 0.5f ;
float x = out_pixel_center - out_center_of_in ;
float coeff = kernel ( x , scale , user_data ) * scale ;
// kill the coeff if it's too small (avoid denormals)
if ( ( ( coeff < stbir__small_float ) & & ( coeff > - stbir__small_float ) ) )
coeff = 0.0f ;
{
int out = i + out_first_pixel ;
float * coeffs = coefficient_group + out * coefficient_width ;
stbir__contributors * contribs = contributors + out ;
// is this the first time this output pixel has been seen? Init it.
if ( out > first_out_inited )
{
STBIR_ASSERT ( out = = ( first_out_inited + 1 ) ) ; // ensure we have only advanced one at time
first_out_inited = out ;
contribs - > n0 = in_pixel ;
contribs - > n1 = in_pixel ;
coeffs [ 0 ] = coeff ;
}
else
{
// insert on end (always in order)
if ( coeffs [ 0 ] = = 0.0f ) // if the first coefficent is zero, then zap it for this coeffs
{
STBIR_ASSERT ( ( in_pixel - contribs - > n0 ) = = 1 ) ; // ensure that when we zap, we're at the 2nd pos
contribs - > n0 = in_pixel ;
}
contribs - > n1 = in_pixel ;
STBIR_ASSERT ( ( in_pixel - contribs - > n0 ) < coefficient_width ) ;
coeffs [ in_pixel - contribs - > n0 ] = coeff ;
}
}
}
}
}
static void stbir__cleanup_gathered_coefficients ( stbir_edge edge , stbir__filter_extent_info * filter_info , stbir__scale_info * scale_info , int num_contributors , stbir__contributors * contributors , float * coefficient_group , int coefficient_width )
{
int input_size = scale_info - > input_full_size ;
int input_last_n1 = input_size - 1 ;
int n , end ;
int lowest = 0x7fffffff ;
int highest = - 0x7fffffff ;
int widest = - 1 ;
int numerator = scale_info - > scale_numerator ;
int denominator = scale_info - > scale_denominator ;
int polyphase = ( ( scale_info - > scale_is_rational ) & & ( numerator < num_contributors ) ) ;
float * coeffs ;
stbir__contributors * contribs ;
// weight all the coeffs for each sample
coeffs = coefficient_group ;
contribs = contributors ;
end = num_contributors ; if ( polyphase ) end = numerator ;
for ( n = 0 ; n < end ; n + + )
{
int i ;
float filter_scale , total_filter = 0 ;
int e ;
// add all contribs
e = contribs - > n1 - contribs - > n0 ;
for ( i = 0 ; i < = e ; i + + )
{
total_filter + = coeffs [ i ] ;
STBIR_ASSERT ( ( coeffs [ i ] > = - 2.0f ) & & ( coeffs [ i ] < = 2.0f ) ) ; // check for wonky weights
}
// rescale
if ( ( total_filter < stbir__small_float ) & & ( total_filter > - stbir__small_float ) )
{
// all coeffs are extremely small, just zero it
contribs - > n1 = contribs - > n0 ;
coeffs [ 0 ] = 0.0f ;
}
else
{
// if the total isn't 1.0, rescale everything
if ( ( total_filter < ( 1.0f - stbir__small_float ) ) | | ( total_filter > ( 1.0f + stbir__small_float ) ) )
{
filter_scale = 1.0f / total_filter ;
// scale them all
for ( i = 0 ; i < = e ; i + + )
coeffs [ i ] * = filter_scale ;
}
}
+ + contribs ;
coeffs + = coefficient_width ;
}
// if we have a rational for the scale, we can exploit the polyphaseness to not calculate
// most of the coefficients, so we copy them here
if ( polyphase )
{
stbir__contributors * prev_contribs = contributors ;
stbir__contributors * cur_contribs = contributors + numerator ;
for ( n = numerator ; n < num_contributors ; n + + )
{
cur_contribs - > n0 = prev_contribs - > n0 + denominator ;
cur_contribs - > n1 = prev_contribs - > n1 + denominator ;
+ + cur_contribs ;
+ + prev_contribs ;
}
stbir_overlapping_memcpy ( coefficient_group + numerator * coefficient_width , coefficient_group , ( num_contributors - numerator ) * coefficient_width * sizeof ( coeffs [ 0 ] ) ) ;
}
coeffs = coefficient_group ;
contribs = contributors ;
for ( n = 0 ; n < num_contributors ; n + + )
{
int i ;
// in zero edge mode, just remove out of bounds contribs completely (since their weights are accounted for now)
if ( edge = = STBIR_EDGE_ZERO )
{
// shrink the right side if necessary
if ( contribs - > n1 > input_last_n1 )
contribs - > n1 = input_last_n1 ;
// shrink the left side
if ( contribs - > n0 < 0 )
{
int j , left , skips = 0 ;
skips = - contribs - > n0 ;
contribs - > n0 = 0 ;
// now move down the weights
left = contribs - > n1 - contribs - > n0 + 1 ;
if ( left > 0 )
{
for ( j = 0 ; j < left ; j + + )
coeffs [ j ] = coeffs [ j + skips ] ;
}
}
}
else if ( ( edge = = STBIR_EDGE_CLAMP ) | | ( edge = = STBIR_EDGE_REFLECT ) )
{
// for clamp and reflect, calculate the true inbounds position (based on edge type) and just add that to the existing weight
// right hand side first
if ( contribs - > n1 > input_last_n1 )
{
int start = contribs - > n0 ;
int endi = contribs - > n1 ;
contribs - > n1 = input_last_n1 ;
for ( i = input_size ; i < = endi ; i + + )
stbir__insert_coeff ( contribs , coeffs , stbir__edge_wrap_slow [ edge ] ( i , input_size ) , coeffs [ i - start ] ) ;
}
// now check left hand edge
if ( contribs - > n0 < 0 )
{
int save_n0 ;
float save_n0_coeff ;
float * c = coeffs - ( contribs - > n0 + 1 ) ;
// reinsert the coeffs with it reflected or clamped (insert accumulates, if the coeffs exist)
for ( i = - 1 ; i > contribs - > n0 ; i - - )
stbir__insert_coeff ( contribs , coeffs , stbir__edge_wrap_slow [ edge ] ( i , input_size ) , * c - - ) ;
save_n0 = contribs - > n0 ;
save_n0_coeff = c [ 0 ] ; // save it, since we didn't do the final one (i==n0), because there might be too many coeffs to hold (before we resize)!
// now slide all the coeffs down (since we have accumulated them in the positive contribs) and reset the first contrib
contribs - > n0 = 0 ;
for ( i = 0 ; i < = contribs - > n1 ; i + + )
coeffs [ i ] = coeffs [ i - save_n0 ] ;
// now that we have shrunk down the contribs, we insert the first one safely
stbir__insert_coeff ( contribs , coeffs , stbir__edge_wrap_slow [ edge ] ( save_n0 , input_size ) , save_n0_coeff ) ;
}
}
if ( contribs - > n0 < = contribs - > n1 )
{
int diff = contribs - > n1 - contribs - > n0 + 1 ;
while ( diff & & ( coeffs [ diff - 1 ] = = 0.0f ) )
- - diff ;
contribs - > n1 = contribs - > n0 + diff - 1 ;
if ( contribs - > n0 < = contribs - > n1 )
{
if ( contribs - > n0 < lowest )
lowest = contribs - > n0 ;
if ( contribs - > n1 > highest )
highest = contribs - > n1 ;
if ( diff > widest )
widest = diff ;
}
// re-zero out unused coefficients (if any)
for ( i = diff ; i < coefficient_width ; i + + )
coeffs [ i ] = 0.0f ;
}
+ + contribs ;
coeffs + = coefficient_width ;
}
filter_info - > lowest = lowest ;
filter_info - > highest = highest ;
filter_info - > widest = widest ;
}
static int stbir__pack_coefficients ( int num_contributors , stbir__contributors * contributors , float * coefficents , int coefficient_width , int widest , int row_width )
{
# define STBIR_MOVE_1( dest, src ) { STBIR_NO_UNROLL(dest); ((stbir_uint32*)(dest))[0] = ((stbir_uint32*)(src))[0]; }
# define STBIR_MOVE_2( dest, src ) { STBIR_NO_UNROLL(dest); ((stbir_uint64*)(dest))[0] = ((stbir_uint64*)(src))[0]; }
# ifdef STBIR_SIMD
# define STBIR_MOVE_4( dest, src ) { stbir__simdf t; STBIR_NO_UNROLL(dest); stbir__simdf_load( t, src ); stbir__simdf_store( dest, t ); }
# else
# define STBIR_MOVE_4( dest, src ) { STBIR_NO_UNROLL(dest); ((stbir_uint64*)(dest))[0] = ((stbir_uint64*)(src))[0]; ((stbir_uint64*)(dest))[1] = ((stbir_uint64*)(src))[1]; }
# endif
if ( coefficient_width ! = widest )
{
float * pc = coefficents ;
float * coeffs = coefficents ;
float * pc_end = coefficents + num_contributors * widest ;
switch ( widest )
{
case 1 :
do {
STBIR_MOVE_1 ( pc , coeffs ) ;
+ + pc ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 2 :
do {
STBIR_MOVE_2 ( pc , coeffs ) ;
pc + = 2 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 3 :
do {
STBIR_MOVE_2 ( pc , coeffs ) ;
STBIR_MOVE_1 ( pc + 2 , coeffs + 2 ) ;
pc + = 3 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 4 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
pc + = 4 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 5 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_1 ( pc + 4 , coeffs + 4 ) ;
pc + = 5 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 6 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_2 ( pc + 4 , coeffs + 4 ) ;
pc + = 6 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 7 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_2 ( pc + 4 , coeffs + 4 ) ;
STBIR_MOVE_1 ( pc + 6 , coeffs + 6 ) ;
pc + = 7 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 8 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_4 ( pc + 4 , coeffs + 4 ) ;
pc + = 8 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 9 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_4 ( pc + 4 , coeffs + 4 ) ;
STBIR_MOVE_1 ( pc + 8 , coeffs + 8 ) ;
pc + = 9 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 10 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_4 ( pc + 4 , coeffs + 4 ) ;
STBIR_MOVE_2 ( pc + 8 , coeffs + 8 ) ;
pc + = 10 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 11 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_4 ( pc + 4 , coeffs + 4 ) ;
STBIR_MOVE_2 ( pc + 8 , coeffs + 8 ) ;
STBIR_MOVE_1 ( pc + 10 , coeffs + 10 ) ;
pc + = 11 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
case 12 :
do {
STBIR_MOVE_4 ( pc , coeffs ) ;
STBIR_MOVE_4 ( pc + 4 , coeffs + 4 ) ;
STBIR_MOVE_4 ( pc + 8 , coeffs + 8 ) ;
pc + = 12 ;
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
default :
do {
float * copy_end = pc + widest - 4 ;
float * c = coeffs ;
do {
STBIR_NO_UNROLL ( pc ) ;
STBIR_MOVE_4 ( pc , c ) ;
pc + = 4 ;
c + = 4 ;
} while ( pc < = copy_end ) ;
copy_end + = 4 ;
while ( pc < copy_end )
{
STBIR_MOVE_1 ( pc , c ) ;
+ + pc ; + + c ;
}
coeffs + = coefficient_width ;
} while ( pc < pc_end ) ;
break ;
}
}
// some horizontal routines read one float off the end (which is then masked off), so put in a sentinal so we don't read an snan or denormal
coefficents [ widest * num_contributors ] = 8888.0f ;
// the minimum we might read for unrolled filters widths is 12. So, we need to
// make sure we never read outside the decode buffer, by possibly moving
// the sample area back into the scanline, and putting zeros weights first.
// we start on the right edge and check until we're well past the possible
// clip area (2*widest).
{
stbir__contributors * contribs = contributors + num_contributors - 1 ;
float * coeffs = coefficents + widest * ( num_contributors - 1 ) ;
// go until no chance of clipping (this is usually less than 8 lops)
2023-12-14 19:09:00 +08:00
while ( ( contribs > = contributors ) & & ( ( contribs - > n0 + widest * 2 ) > = row_width ) )
2023-10-10 08:23:04 +08:00
{
// might we clip??
if ( ( contribs - > n0 + widest ) > row_width )
{
int stop_range = widest ;
// if range is larger than 12, it will be handled by generic loops that can terminate on the exact length
// of this contrib n1, instead of a fixed widest amount - so calculate this
if ( widest > 12 )
{
int mod ;
// how far will be read in the n_coeff loop (which depends on the widest count mod4);
mod = widest & 3 ;
stop_range = ( ( ( contribs - > n1 - contribs - > n0 + 1 ) - mod + 3 ) & ~ 3 ) + mod ;
// the n_coeff loops do a minimum amount of coeffs, so factor that in!
if ( stop_range < ( 8 + mod ) ) stop_range = 8 + mod ;
}
// now see if we still clip with the refined range
if ( ( contribs - > n0 + stop_range ) > row_width )
{
int new_n0 = row_width - stop_range ;
int num = contribs - > n1 - contribs - > n0 + 1 ;
int backup = contribs - > n0 - new_n0 ;
float * from_co = coeffs + num - 1 ;
float * to_co = from_co + backup ;
STBIR_ASSERT ( ( new_n0 > = 0 ) & & ( new_n0 < contribs - > n0 ) ) ;
// move the coeffs over
while ( num )
{
* to_co - - = * from_co - - ;
- - num ;
}
// zero new positions
while ( to_co > = coeffs )
* to_co - - = 0 ;
// set new start point
contribs - > n0 = new_n0 ;
if ( widest > 12 )
{
int mod ;
// how far will be read in the n_coeff loop (which depends on the widest count mod4);
mod = widest & 3 ;
stop_range = ( ( ( contribs - > n1 - contribs - > n0 + 1 ) - mod + 3 ) & ~ 3 ) + mod ;
// the n_coeff loops do a minimum amount of coeffs, so factor that in!
if ( stop_range < ( 8 + mod ) ) stop_range = 8 + mod ;
}
}
}
- - contribs ;
coeffs - = widest ;
}
}
return widest ;
# undef STBIR_MOVE_1
# undef STBIR_MOVE_2
# undef STBIR_MOVE_4
}
static void stbir__calculate_filters ( stbir__sampler * samp , stbir__sampler * other_axis_for_pivot , void * user_data STBIR_ONLY_PROFILE_BUILD_GET_INFO )
{
int n ;
float scale = samp - > scale_info . scale ;
stbir__kernel_callback * kernel = samp - > filter_kernel ;
stbir__support_callback * support = samp - > filter_support ;
float inv_scale = samp - > scale_info . inv_scale ;
int input_full_size = samp - > scale_info . input_full_size ;
int gather_num_contributors = samp - > num_contributors ;
stbir__contributors * gather_contributors = samp - > contributors ;
float * gather_coeffs = samp - > coefficients ;
int gather_coefficient_width = samp - > coefficient_width ;
switch ( samp - > is_gather )
{
case 1 : // gather upsample
{
float out_pixels_radius = support ( inv_scale , user_data ) * scale ;
stbir__calculate_coefficients_for_gather_upsample ( out_pixels_radius , kernel , & samp - > scale_info , gather_num_contributors , gather_contributors , gather_coeffs , gather_coefficient_width , samp - > edge , user_data ) ;
STBIR_PROFILE_BUILD_START ( cleanup ) ;
stbir__cleanup_gathered_coefficients ( samp - > edge , & samp - > extent_info , & samp - > scale_info , gather_num_contributors , gather_contributors , gather_coeffs , gather_coefficient_width ) ;
STBIR_PROFILE_BUILD_END ( cleanup ) ;
}
break ;
case 0 : // scatter downsample (only on vertical)
case 2 : // gather downsample
{
float in_pixels_radius = support ( scale , user_data ) * inv_scale ;
int filter_pixel_margin = samp - > filter_pixel_margin ;
int input_end = input_full_size + filter_pixel_margin ;
// if this is a scatter, we do a downsample gather to get the coeffs, and then pivot after
if ( ! samp - > is_gather )
{
// check if we are using the same gather downsample on the horizontal as this vertical,
// if so, then we don't have to generate them, we can just pivot from the horizontal.
if ( other_axis_for_pivot )
{
gather_contributors = other_axis_for_pivot - > contributors ;
gather_coeffs = other_axis_for_pivot - > coefficients ;
gather_coefficient_width = other_axis_for_pivot - > coefficient_width ;
gather_num_contributors = other_axis_for_pivot - > num_contributors ;
samp - > extent_info . lowest = other_axis_for_pivot - > extent_info . lowest ;
samp - > extent_info . highest = other_axis_for_pivot - > extent_info . highest ;
samp - > extent_info . widest = other_axis_for_pivot - > extent_info . widest ;
goto jump_right_to_pivot ;
}
gather_contributors = samp - > gather_prescatter_contributors ;
gather_coeffs = samp - > gather_prescatter_coefficients ;
gather_coefficient_width = samp - > gather_prescatter_coefficient_width ;
gather_num_contributors = samp - > gather_prescatter_num_contributors ;
}
stbir__calculate_coefficients_for_gather_downsample ( - filter_pixel_margin , input_end , in_pixels_radius , kernel , & samp - > scale_info , gather_coefficient_width , gather_num_contributors , gather_contributors , gather_coeffs , user_data ) ;
STBIR_PROFILE_BUILD_START ( cleanup ) ;
stbir__cleanup_gathered_coefficients ( samp - > edge , & samp - > extent_info , & samp - > scale_info , gather_num_contributors , gather_contributors , gather_coeffs , gather_coefficient_width ) ;
STBIR_PROFILE_BUILD_END ( cleanup ) ;
if ( ! samp - > is_gather )
{
// if this is a scatter (vertical only), then we need to pivot the coeffs
stbir__contributors * scatter_contributors ;
int highest_set ;
jump_right_to_pivot :
STBIR_PROFILE_BUILD_START ( pivot ) ;
highest_set = ( - filter_pixel_margin ) - 1 ;
for ( n = 0 ; n < gather_num_contributors ; n + + )
{
int k ;
int gn0 = gather_contributors - > n0 , gn1 = gather_contributors - > n1 ;
int scatter_coefficient_width = samp - > coefficient_width ;
float * scatter_coeffs = samp - > coefficients + ( gn0 + filter_pixel_margin ) * scatter_coefficient_width ;
float * g_coeffs = gather_coeffs ;
scatter_contributors = samp - > contributors + ( gn0 + filter_pixel_margin ) ;
for ( k = gn0 ; k < = gn1 ; k + + )
{
float gc = * g_coeffs + + ;
if ( ( k > highest_set ) | | ( scatter_contributors - > n0 > scatter_contributors - > n1 ) )
{
{
// if we are skipping over several contributors, we need to clear the skipped ones
stbir__contributors * clear_contributors = samp - > contributors + ( highest_set + filter_pixel_margin + 1 ) ;
while ( clear_contributors < scatter_contributors )
{
clear_contributors - > n0 = 0 ;
clear_contributors - > n1 = - 1 ;
+ + clear_contributors ;
}
}
scatter_contributors - > n0 = n ;
scatter_contributors - > n1 = n ;
scatter_coeffs [ 0 ] = gc ;
highest_set = k ;
}
else
{
stbir__insert_coeff ( scatter_contributors , scatter_coeffs , n , gc ) ;
}
+ + scatter_contributors ;
scatter_coeffs + = scatter_coefficient_width ;
}
+ + gather_contributors ;
gather_coeffs + = gather_coefficient_width ;
}
// now clear any unset contribs
{
stbir__contributors * clear_contributors = samp - > contributors + ( highest_set + filter_pixel_margin + 1 ) ;
stbir__contributors * end_contributors = samp - > contributors + samp - > num_contributors ;
while ( clear_contributors < end_contributors )
{
clear_contributors - > n0 = 0 ;
clear_contributors - > n1 = - 1 ;
+ + clear_contributors ;
}
}
STBIR_PROFILE_BUILD_END ( pivot ) ;
}
}
break ;
}
}
//========================================================================================================
// scanline decoders and encoders
# define stbir__coder_min_num 1
# define STB_IMAGE_RESIZE_DO_CODERS
# include STBIR__HEADER_FILENAME
# define stbir__decode_suffix BGRA
# define stbir__decode_swizzle
# define stbir__decode_order0 2
# define stbir__decode_order1 1
# define stbir__decode_order2 0
# define stbir__decode_order3 3
# define stbir__encode_order0 2
# define stbir__encode_order1 1
# define stbir__encode_order2 0
# define stbir__encode_order3 3
# define stbir__coder_min_num 4
# define STB_IMAGE_RESIZE_DO_CODERS
# include STBIR__HEADER_FILENAME
# define stbir__decode_suffix ARGB
# define stbir__decode_swizzle
# define stbir__decode_order0 1
# define stbir__decode_order1 2
# define stbir__decode_order2 3
# define stbir__decode_order3 0
# define stbir__encode_order0 3
# define stbir__encode_order1 0
# define stbir__encode_order2 1
# define stbir__encode_order3 2
# define stbir__coder_min_num 4
# define STB_IMAGE_RESIZE_DO_CODERS
# include STBIR__HEADER_FILENAME
# define stbir__decode_suffix ABGR
# define stbir__decode_swizzle
# define stbir__decode_order0 3
# define stbir__decode_order1 2
# define stbir__decode_order2 1
# define stbir__decode_order3 0
# define stbir__encode_order0 3
# define stbir__encode_order1 2
# define stbir__encode_order2 1
# define stbir__encode_order3 0
# define stbir__coder_min_num 4
# define STB_IMAGE_RESIZE_DO_CODERS
# include STBIR__HEADER_FILENAME
# define stbir__decode_suffix AR
# define stbir__decode_swizzle
# define stbir__decode_order0 1
# define stbir__decode_order1 0
# define stbir__decode_order2 3
# define stbir__decode_order3 2
# define stbir__encode_order0 1
# define stbir__encode_order1 0
# define stbir__encode_order2 3
# define stbir__encode_order3 2
# define stbir__coder_min_num 2
# define STB_IMAGE_RESIZE_DO_CODERS
# include STBIR__HEADER_FILENAME
// fancy alpha means we expand to keep both premultipied and non-premultiplied color channels
static void stbir__fancy_alpha_weight_4ch ( float * out_buffer , int width_times_channels )
{
float STBIR_STREAMOUT_PTR ( * ) out = out_buffer ;
float const * end_decode = out_buffer + ( width_times_channels / 4 ) * 7 ; // decode buffer aligned to end of out_buffer
float STBIR_STREAMOUT_PTR ( * ) decode = ( float * ) end_decode - width_times_channels ;
// fancy alpha is stored internally as R G B A Rpm Gpm Bpm
# ifdef STBIR_SIMD
# ifdef STBIR_SIMD8
decode + = 16 ;
while ( decode < = end_decode )
{
stbir__simdf8 d0 , d1 , a0 , a1 , p0 , p1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdf8_load ( d0 , decode - 16 ) ;
stbir__simdf8_load ( d1 , decode - 16 + 8 ) ;
stbir__simdf8_0123to33333333 ( a0 , d0 ) ;
stbir__simdf8_0123to33333333 ( a1 , d1 ) ;
stbir__simdf8_mult ( p0 , a0 , d0 ) ;
stbir__simdf8_mult ( p1 , a1 , d1 ) ;
stbir__simdf8_bot4s ( a0 , d0 , p0 ) ;
stbir__simdf8_bot4s ( a1 , d1 , p1 ) ;
stbir__simdf8_top4s ( d0 , d0 , p0 ) ;
stbir__simdf8_top4s ( d1 , d1 , p1 ) ;
stbir__simdf8_store ( out , a0 ) ;
stbir__simdf8_store ( out + 7 , d0 ) ;
stbir__simdf8_store ( out + 14 , a1 ) ;
stbir__simdf8_store ( out + 21 , d1 ) ;
decode + = 16 ;
out + = 28 ;
}
decode - = 16 ;
# else
decode + = 8 ;
while ( decode < = end_decode )
{
stbir__simdf d0 , a0 , d1 , a1 , p0 , p1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdf_load ( d0 , decode - 8 ) ;
stbir__simdf_load ( d1 , decode - 8 + 4 ) ;
stbir__simdf_0123to3333 ( a0 , d0 ) ;
stbir__simdf_0123to3333 ( a1 , d1 ) ;
stbir__simdf_mult ( p0 , a0 , d0 ) ;
stbir__simdf_mult ( p1 , a1 , d1 ) ;
stbir__simdf_store ( out , d0 ) ;
stbir__simdf_store ( out + 4 , p0 ) ;
stbir__simdf_store ( out + 7 , d1 ) ;
stbir__simdf_store ( out + 7 + 4 , p1 ) ;
decode + = 8 ;
out + = 14 ;
}
decode - = 8 ;
# endif
// might be one last odd pixel
# ifdef STBIR_SIMD8
while ( decode < end_decode )
# else
if ( decode < end_decode )
# endif
{
stbir__simdf d , a , p ;
stbir__simdf_load ( d , decode ) ;
stbir__simdf_0123to3333 ( a , d ) ;
stbir__simdf_mult ( p , a , d ) ;
stbir__simdf_store ( out , d ) ;
stbir__simdf_store ( out + 4 , p ) ;
decode + = 4 ;
out + = 7 ;
}
# else
while ( decode < end_decode )
{
float r = decode [ 0 ] , g = decode [ 1 ] , b = decode [ 2 ] , alpha = decode [ 3 ] ;
out [ 0 ] = r ;
out [ 1 ] = g ;
out [ 2 ] = b ;
out [ 3 ] = alpha ;
out [ 4 ] = r * alpha ;
out [ 5 ] = g * alpha ;
out [ 6 ] = b * alpha ;
out + = 7 ;
decode + = 4 ;
}
# endif
}
static void stbir__fancy_alpha_weight_2ch ( float * out_buffer , int width_times_channels )
{
float STBIR_STREAMOUT_PTR ( * ) out = out_buffer ;
float const * end_decode = out_buffer + ( width_times_channels / 2 ) * 3 ;
float STBIR_STREAMOUT_PTR ( * ) decode = ( float * ) end_decode - width_times_channels ;
// for fancy alpha, turns into: [X A Xpm][X A Xpm],etc
# ifdef STBIR_SIMD
decode + = 8 ;
if ( decode < = end_decode )
{
do {
# ifdef STBIR_SIMD8
stbir__simdf8 d0 , a0 , p0 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdf8_load ( d0 , decode - 8 ) ;
stbir__simdf8_0123to11331133 ( p0 , d0 ) ;
stbir__simdf8_0123to00220022 ( a0 , d0 ) ;
stbir__simdf8_mult ( p0 , p0 , a0 ) ;
stbir__simdf_store2 ( out , stbir__if_simdf8_cast_to_simdf4 ( d0 ) ) ;
stbir__simdf_store ( out + 2 , stbir__if_simdf8_cast_to_simdf4 ( p0 ) ) ;
stbir__simdf_store2h ( out + 3 , stbir__if_simdf8_cast_to_simdf4 ( d0 ) ) ;
stbir__simdf_store2 ( out + 6 , stbir__simdf8_gettop4 ( d0 ) ) ;
stbir__simdf_store ( out + 8 , stbir__simdf8_gettop4 ( p0 ) ) ;
stbir__simdf_store2h ( out + 9 , stbir__simdf8_gettop4 ( d0 ) ) ;
# else
stbir__simdf d0 , a0 , d1 , a1 , p0 , p1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdf_load ( d0 , decode - 8 ) ;
stbir__simdf_load ( d1 , decode - 8 + 4 ) ;
stbir__simdf_0123to1133 ( p0 , d0 ) ;
stbir__simdf_0123to1133 ( p1 , d1 ) ;
stbir__simdf_0123to0022 ( a0 , d0 ) ;
stbir__simdf_0123to0022 ( a1 , d1 ) ;
stbir__simdf_mult ( p0 , p0 , a0 ) ;
stbir__simdf_mult ( p1 , p1 , a1 ) ;
stbir__simdf_store2 ( out , d0 ) ;
stbir__simdf_store ( out + 2 , p0 ) ;
stbir__simdf_store2h ( out + 3 , d0 ) ;
stbir__simdf_store2 ( out + 6 , d1 ) ;
stbir__simdf_store ( out + 8 , p1 ) ;
stbir__simdf_store2h ( out + 9 , d1 ) ;
# endif
decode + = 8 ;
out + = 12 ;
} while ( decode < = end_decode ) ;
}
decode - = 8 ;
# endif
while ( decode < end_decode )
{
float x = decode [ 0 ] , y = decode [ 1 ] ;
STBIR_SIMD_NO_UNROLL ( decode ) ;
out [ 0 ] = x ;
out [ 1 ] = y ;
out [ 2 ] = x * y ;
out + = 3 ;
decode + = 2 ;
}
}
static void stbir__fancy_alpha_unweight_4ch ( float * encode_buffer , int width_times_channels )
{
float STBIR_SIMD_STREAMOUT_PTR ( * ) encode = encode_buffer ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) input = encode_buffer ;
float const * end_output = encode_buffer + width_times_channels ;
// fancy RGBA is stored internally as R G B A Rpm Gpm Bpm
do {
float alpha = input [ 3 ] ;
# ifdef STBIR_SIMD
stbir__simdf i , ia ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
if ( alpha < stbir__small_float )
{
stbir__simdf_load ( i , input ) ;
stbir__simdf_store ( encode , i ) ;
}
else
{
stbir__simdf_load1frep4 ( ia , 1.0f / alpha ) ;
stbir__simdf_load ( i , input + 4 ) ;
stbir__simdf_mult ( i , i , ia ) ;
stbir__simdf_store ( encode , i ) ;
encode [ 3 ] = alpha ;
}
# else
if ( alpha < stbir__small_float )
{
encode [ 0 ] = input [ 0 ] ;
encode [ 1 ] = input [ 1 ] ;
encode [ 2 ] = input [ 2 ] ;
}
else
{
float ialpha = 1.0f / alpha ;
encode [ 0 ] = input [ 4 ] * ialpha ;
encode [ 1 ] = input [ 5 ] * ialpha ;
encode [ 2 ] = input [ 6 ] * ialpha ;
}
encode [ 3 ] = alpha ;
# endif
input + = 7 ;
encode + = 4 ;
} while ( encode < end_output ) ;
}
// format: [X A Xpm][X A Xpm] etc
static void stbir__fancy_alpha_unweight_2ch ( float * encode_buffer , int width_times_channels )
{
float STBIR_SIMD_STREAMOUT_PTR ( * ) encode = encode_buffer ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) input = encode_buffer ;
float const * end_output = encode_buffer + width_times_channels ;
do {
float alpha = input [ 1 ] ;
encode [ 0 ] = input [ 0 ] ;
if ( alpha > = stbir__small_float )
encode [ 0 ] = input [ 2 ] / alpha ;
encode [ 1 ] = alpha ;
input + = 3 ;
encode + = 2 ;
} while ( encode < end_output ) ;
}
static void stbir__simple_alpha_weight_4ch ( float * decode_buffer , int width_times_channels )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decode_buffer ;
float const * end_decode = decode_buffer + width_times_channels ;
# ifdef STBIR_SIMD
{
decode + = 2 * stbir__simdfX_float_count ;
while ( decode < = end_decode )
{
stbir__simdfX d0 , a0 , d1 , a1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdfX_load ( d0 , decode - 2 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( d1 , decode - 2 * stbir__simdfX_float_count + stbir__simdfX_float_count ) ;
stbir__simdfX_aaa1 ( a0 , d0 , STBIR_onesX ) ;
stbir__simdfX_aaa1 ( a1 , d1 , STBIR_onesX ) ;
stbir__simdfX_mult ( d0 , d0 , a0 ) ;
stbir__simdfX_mult ( d1 , d1 , a1 ) ;
stbir__simdfX_store ( decode - 2 * stbir__simdfX_float_count , d0 ) ;
stbir__simdfX_store ( decode - 2 * stbir__simdfX_float_count + stbir__simdfX_float_count , d1 ) ;
decode + = 2 * stbir__simdfX_float_count ;
}
decode - = 2 * stbir__simdfX_float_count ;
// few last pixels remnants
# ifdef STBIR_SIMD8
while ( decode < end_decode )
# else
if ( decode < end_decode )
# endif
{
stbir__simdf d , a ;
stbir__simdf_load ( d , decode ) ;
stbir__simdf_aaa1 ( a , d , STBIR__CONSTF ( STBIR_ones ) ) ;
stbir__simdf_mult ( d , d , a ) ;
stbir__simdf_store ( decode , d ) ;
decode + = 4 ;
}
}
# else
while ( decode < end_decode )
{
float alpha = decode [ 3 ] ;
decode [ 0 ] * = alpha ;
decode [ 1 ] * = alpha ;
decode [ 2 ] * = alpha ;
decode + = 4 ;
}
# endif
}
static void stbir__simple_alpha_weight_2ch ( float * decode_buffer , int width_times_channels )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decode_buffer ;
float const * end_decode = decode_buffer + width_times_channels ;
# ifdef STBIR_SIMD
decode + = 2 * stbir__simdfX_float_count ;
while ( decode < = end_decode )
{
stbir__simdfX d0 , a0 , d1 , a1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdfX_load ( d0 , decode - 2 * stbir__simdfX_float_count ) ;
stbir__simdfX_load ( d1 , decode - 2 * stbir__simdfX_float_count + stbir__simdfX_float_count ) ;
stbir__simdfX_a1a1 ( a0 , d0 , STBIR_onesX ) ;
stbir__simdfX_a1a1 ( a1 , d1 , STBIR_onesX ) ;
stbir__simdfX_mult ( d0 , d0 , a0 ) ;
stbir__simdfX_mult ( d1 , d1 , a1 ) ;
stbir__simdfX_store ( decode - 2 * stbir__simdfX_float_count , d0 ) ;
stbir__simdfX_store ( decode - 2 * stbir__simdfX_float_count + stbir__simdfX_float_count , d1 ) ;
decode + = 2 * stbir__simdfX_float_count ;
}
decode - = 2 * stbir__simdfX_float_count ;
# endif
while ( decode < end_decode )
{
float alpha = decode [ 1 ] ;
STBIR_SIMD_NO_UNROLL ( decode ) ;
decode [ 0 ] * = alpha ;
decode + = 2 ;
}
}
static void stbir__simple_alpha_unweight_4ch ( float * encode_buffer , int width_times_channels )
{
float STBIR_SIMD_STREAMOUT_PTR ( * ) encode = encode_buffer ;
float const * end_output = encode_buffer + width_times_channels ;
do {
float alpha = encode [ 3 ] ;
# ifdef STBIR_SIMD
stbir__simdf i , ia ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
if ( alpha > = stbir__small_float )
{
stbir__simdf_load1frep4 ( ia , 1.0f / alpha ) ;
stbir__simdf_load ( i , encode ) ;
stbir__simdf_mult ( i , i , ia ) ;
stbir__simdf_store ( encode , i ) ;
encode [ 3 ] = alpha ;
}
# else
if ( alpha > = stbir__small_float )
{
float ialpha = 1.0f / alpha ;
encode [ 0 ] * = ialpha ;
encode [ 1 ] * = ialpha ;
encode [ 2 ] * = ialpha ;
}
# endif
encode + = 4 ;
} while ( encode < end_output ) ;
}
static void stbir__simple_alpha_unweight_2ch ( float * encode_buffer , int width_times_channels )
{
float STBIR_SIMD_STREAMOUT_PTR ( * ) encode = encode_buffer ;
float const * end_output = encode_buffer + width_times_channels ;
do {
float alpha = encode [ 1 ] ;
if ( alpha > = stbir__small_float )
encode [ 0 ] / = alpha ;
encode + = 2 ;
} while ( encode < end_output ) ;
}
// only used in RGB->BGR or BGR->RGB
static void stbir__simple_flip_3ch ( float * decode_buffer , int width_times_channels )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decode_buffer ;
float const * end_decode = decode_buffer + width_times_channels ;
decode + = 12 ;
while ( decode < = end_decode )
{
float t0 , t1 , t2 , t3 ;
STBIR_NO_UNROLL ( decode ) ;
t0 = decode [ 0 ] ; t1 = decode [ 3 ] ; t2 = decode [ 6 ] ; t3 = decode [ 9 ] ;
decode [ 0 ] = decode [ 2 ] ; decode [ 3 ] = decode [ 5 ] ; decode [ 6 ] = decode [ 8 ] ; decode [ 9 ] = decode [ 11 ] ;
decode [ 2 ] = t0 ; decode [ 5 ] = t1 ; decode [ 8 ] = t2 ; decode [ 11 ] = t3 ;
decode + = 12 ;
}
decode - = 12 ;
while ( decode < end_decode )
{
float t = decode [ 0 ] ;
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = decode [ 2 ] ;
decode [ 2 ] = t ;
decode + = 3 ;
}
}
static void stbir__decode_scanline ( stbir__info const * stbir_info , int n , float * output_buffer STBIR_ONLY_PROFILE_GET_SPLIT_INFO )
{
int channels = stbir_info - > channels ;
int effective_channels = stbir_info - > effective_channels ;
int input_sample_in_bytes = stbir__type_size [ stbir_info - > input_type ] * channels ;
stbir_edge edge_horizontal = stbir_info - > horizontal . edge ;
stbir_edge edge_vertical = stbir_info - > vertical . edge ;
int row = stbir__edge_wrap ( edge_vertical , n , stbir_info - > vertical . scale_info . input_full_size ) ;
const void * input_plane_data = ( ( char * ) stbir_info - > input_data ) + ( ptrdiff_t ) row * ( ptrdiff_t ) stbir_info - > input_stride_bytes ;
stbir__span const * spans = stbir_info - > scanline_extents . spans ;
float * full_decode_buffer = output_buffer - stbir_info - > scanline_extents . conservative . n0 * effective_channels ;
// if we are on edge_zero, and we get in here with an out of bounds n, then the calculate filters has failed
STBIR_ASSERT ( ! ( edge_vertical = = STBIR_EDGE_ZERO & & ( n < 0 | | n > = stbir_info - > vertical . scale_info . input_full_size ) ) ) ;
do
{
float * decode_buffer ;
void const * input_data ;
float * end_decode ;
int width_times_channels ;
int width ;
if ( spans - > n1 < spans - > n0 )
break ;
width = spans - > n1 + 1 - spans - > n0 ;
decode_buffer = full_decode_buffer + spans - > n0 * effective_channels ;
end_decode = full_decode_buffer + ( spans - > n1 + 1 ) * effective_channels ;
width_times_channels = width * channels ;
// read directly out of input plane by default
input_data = ( ( char * ) input_plane_data ) + spans - > pixel_offset_for_input * input_sample_in_bytes ;
// if we have an input callback, call it to get the input data
if ( stbir_info - > in_pixels_cb )
{
// call the callback with a temp buffer (that they can choose to use or not). the temp is just right aligned memory in the decode_buffer itself
input_data = stbir_info - > in_pixels_cb ( ( ( char * ) end_decode ) - ( width * input_sample_in_bytes ) , input_plane_data , width , spans - > pixel_offset_for_input , row , stbir_info - > user_data ) ;
}
STBIR_PROFILE_START ( decode ) ;
// convert the pixels info the float decode_buffer, (we index from end_decode, so that when channels<effective_channels, we are right justified in the buffer)
stbir_info - > decode_pixels ( ( float * ) end_decode - width_times_channels , width_times_channels , input_data ) ;
STBIR_PROFILE_END ( decode ) ;
if ( stbir_info - > alpha_weight )
{
STBIR_PROFILE_START ( alpha ) ;
stbir_info - > alpha_weight ( decode_buffer , width_times_channels ) ;
STBIR_PROFILE_END ( alpha ) ;
}
+ + spans ;
} while ( spans < = ( & stbir_info - > scanline_extents . spans [ 1 ] ) ) ;
// handle the edge_wrap filter (all other types are handled back out at the calculate_filter stage)
// basically the idea here is that if we have the whole scanline in memory, we don't redecode the
// wrapped edge pixels, and instead just memcpy them from the scanline into the edge positions
if ( ( edge_horizontal = = STBIR_EDGE_WRAP ) & & ( stbir_info - > scanline_extents . edge_sizes [ 0 ] | stbir_info - > scanline_extents . edge_sizes [ 1 ] ) )
{
// this code only runs if we're in edge_wrap, and we're doing the entire scanline
int e , start_x [ 2 ] ;
int input_full_size = stbir_info - > horizontal . scale_info . input_full_size ;
start_x [ 0 ] = - stbir_info - > scanline_extents . edge_sizes [ 0 ] ; // left edge start x
start_x [ 1 ] = input_full_size ; // right edge
for ( e = 0 ; e < 2 ; e + + )
{
// do each margin
int margin = stbir_info - > scanline_extents . edge_sizes [ e ] ;
if ( margin )
{
int x = start_x [ e ] ;
float * marg = full_decode_buffer + x * effective_channels ;
float const * src = full_decode_buffer + stbir__edge_wrap ( edge_horizontal , x , input_full_size ) * effective_channels ;
STBIR_MEMCPY ( marg , src , margin * effective_channels * sizeof ( float ) ) ;
}
}
}
}
//=================
// Do 1 channel horizontal routines
# ifdef STBIR_SIMD
# define stbir__1_coeff_only() \
stbir__simdf tot , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1 ( c , hc ) ; \
stbir__simdf_mult1_mem ( tot , c , decode ) ;
# define stbir__2_coeff_only() \
stbir__simdf tot , c , d ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2z ( c , hc ) ; \
stbir__simdf_load2 ( d , decode ) ; \
stbir__simdf_mult ( tot , c , d ) ; \
stbir__simdf_0123to1230 ( c , tot ) ; \
stbir__simdf_add1 ( tot , tot , c ) ;
# define stbir__3_coeff_only() \
stbir__simdf tot , c , t ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( c , hc ) ; \
stbir__simdf_mult_mem ( tot , c , decode ) ; \
stbir__simdf_0123to1230 ( c , tot ) ; \
stbir__simdf_0123to2301 ( t , tot ) ; \
stbir__simdf_add1 ( tot , tot , c ) ; \
stbir__simdf_add1 ( tot , tot , t ) ;
# define stbir__store_output_tiny() \
stbir__simdf_store1 ( output , tot ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 1 ;
# define stbir__4_coeff_start() \
stbir__simdf tot , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( c , hc ) ; \
stbir__simdf_mult_mem ( tot , c , decode ) ; \
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( c , hc + ( ofs ) ) ; \
stbir__simdf_madd_mem ( tot , tot , c , decode + ( ofs ) ) ;
# define stbir__1_coeff_remnant( ofs ) \
{ stbir__simdf d ; \
stbir__simdf_load1z ( c , hc + ( ofs ) ) ; \
stbir__simdf_load1 ( d , decode + ( ofs ) ) ; \
stbir__simdf_madd ( tot , tot , d , c ) ; }
# define stbir__2_coeff_remnant( ofs ) \
{ stbir__simdf d ; \
stbir__simdf_load2z ( c , hc + ( ofs ) ) ; \
stbir__simdf_load2 ( d , decode + ( ofs ) ) ; \
stbir__simdf_madd ( tot , tot , d , c ) ; }
# define stbir__3_coeff_setup() \
stbir__simdf mask ; \
stbir__simdf_load ( mask , STBIR_mask + 3 ) ;
# define stbir__3_coeff_remnant( ofs ) \
stbir__simdf_load ( c , hc + ( ofs ) ) ; \
stbir__simdf_and ( c , c , mask ) ; \
stbir__simdf_madd_mem ( tot , tot , c , decode + ( ofs ) ) ;
# define stbir__store_output() \
stbir__simdf_0123to2301 ( c , tot ) ; \
stbir__simdf_add ( tot , tot , c ) ; \
stbir__simdf_0123to1230 ( c , tot ) ; \
stbir__simdf_add1 ( tot , tot , c ) ; \
stbir__simdf_store1 ( output , tot ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 1 ;
# else
# define stbir__1_coeff_only() \
float tot ; \
tot = decode [ 0 ] * hc [ 0 ] ;
# define stbir__2_coeff_only() \
float tot ; \
tot = decode [ 0 ] * hc [ 0 ] ; \
tot + = decode [ 1 ] * hc [ 1 ] ;
# define stbir__3_coeff_only() \
float tot ; \
tot = decode [ 0 ] * hc [ 0 ] ; \
tot + = decode [ 1 ] * hc [ 1 ] ; \
tot + = decode [ 2 ] * hc [ 2 ] ;
# define stbir__store_output_tiny() \
output [ 0 ] = tot ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 1 ;
# define stbir__4_coeff_start() \
float tot0 , tot1 , tot2 , tot3 ; \
tot0 = decode [ 0 ] * hc [ 0 ] ; \
tot1 = decode [ 1 ] * hc [ 1 ] ; \
tot2 = decode [ 2 ] * hc [ 2 ] ; \
tot3 = decode [ 3 ] * hc [ 3 ] ;
# define stbir__4_coeff_continue_from_4( ofs ) \
tot0 + = decode [ 0 + ( ofs ) ] * hc [ 0 + ( ofs ) ] ; \
tot1 + = decode [ 1 + ( ofs ) ] * hc [ 1 + ( ofs ) ] ; \
tot2 + = decode [ 2 + ( ofs ) ] * hc [ 2 + ( ofs ) ] ; \
tot3 + = decode [ 3 + ( ofs ) ] * hc [ 3 + ( ofs ) ] ;
# define stbir__1_coeff_remnant( ofs ) \
tot0 + = decode [ 0 + ( ofs ) ] * hc [ 0 + ( ofs ) ] ;
# define stbir__2_coeff_remnant( ofs ) \
tot0 + = decode [ 0 + ( ofs ) ] * hc [ 0 + ( ofs ) ] ; \
tot1 + = decode [ 1 + ( ofs ) ] * hc [ 1 + ( ofs ) ] ; \
# define stbir__3_coeff_remnant( ofs ) \
tot0 + = decode [ 0 + ( ofs ) ] * hc [ 0 + ( ofs ) ] ; \
tot1 + = decode [ 1 + ( ofs ) ] * hc [ 1 + ( ofs ) ] ; \
tot2 + = decode [ 2 + ( ofs ) ] * hc [ 2 + ( ofs ) ] ;
# define stbir__store_output() \
output [ 0 ] = ( tot0 + tot2 ) + ( tot1 + tot3 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 1 ;
# endif
# define STBIR__horizontal_channels 1
# define STB_IMAGE_RESIZE_DO_HORIZONTALS
# include STBIR__HEADER_FILENAME
//=================
// Do 2 channel horizontal routines
# ifdef STBIR_SIMD
# define stbir__1_coeff_only() \
stbir__simdf tot , c , d ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1z ( c , hc ) ; \
stbir__simdf_0123to0011 ( c , c ) ; \
stbir__simdf_load2 ( d , decode ) ; \
stbir__simdf_mult ( tot , d , c ) ;
# define stbir__2_coeff_only() \
stbir__simdf tot , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2 ( c , hc ) ; \
stbir__simdf_0123to0011 ( c , c ) ; \
stbir__simdf_mult_mem ( tot , c , decode ) ;
# define stbir__3_coeff_only() \
stbir__simdf tot , c , cs , d ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0011 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot , c , decode ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_load2z ( d , decode + 4 ) ; \
stbir__simdf_madd ( tot , tot , d , c ) ;
# define stbir__store_output_tiny() \
stbir__simdf_0123to2301 ( c , tot ) ; \
stbir__simdf_add ( tot , tot , c ) ; \
stbir__simdf_store2 ( output , tot ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 2 ;
# ifdef STBIR_SIMD8
# define stbir__4_coeff_start() \
stbir__simdf8 tot0 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc ) ; \
stbir__simdf8_0123to00112233 ( c , cs ) ; \
stbir__simdf8_mult_mem ( tot0 , c , decode ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00112233 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 2 ) ;
# define stbir__1_coeff_remnant( ofs ) \
{ stbir__simdf t ; \
stbir__simdf_load1z ( t , hc + ( ofs ) ) ; \
stbir__simdf_0123to0011 ( t , t ) ; \
stbir__simdf_mult_mem ( t , t , decode + ( ofs ) * 2 ) ; \
stbir__simdf8_add4 ( tot0 , tot0 , t ) ; }
# define stbir__2_coeff_remnant( ofs ) \
{ stbir__simdf t ; \
stbir__simdf_load2 ( t , hc + ( ofs ) ) ; \
stbir__simdf_0123to0011 ( t , t ) ; \
stbir__simdf_mult_mem ( t , t , decode + ( ofs ) * 2 ) ; \
stbir__simdf8_add4 ( tot0 , tot0 , t ) ; }
# define stbir__3_coeff_remnant( ofs ) \
{ stbir__simdf8 d ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00112233 ( c , cs ) ; \
stbir__simdf8_load6z ( d , decode + ( ofs ) * 2 ) ; \
stbir__simdf8_madd ( tot0 , tot0 , c , d ) ; }
# define stbir__store_output() \
2023-12-14 19:09:00 +08:00
{ stbir__simdf t , d ; \
2023-10-10 08:23:04 +08:00
stbir__simdf8_add4halves ( t , stbir__if_simdf8_cast_to_simdf4 ( tot0 ) , tot0 ) ; \
2023-12-14 19:09:00 +08:00
stbir__simdf_0123to2301 ( d , t ) ; \
stbir__simdf_add ( t , t , d ) ; \
2023-10-10 08:23:04 +08:00
stbir__simdf_store2 ( output , t ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 2 ; }
# else
# define stbir__4_coeff_start() \
stbir__simdf tot0 , tot1 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0011 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf_0123to2233 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot1 , c , decode + 4 ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0011 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 2 ) ; \
stbir__simdf_0123to2233 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 2 + 4 ) ;
# define stbir__1_coeff_remnant( ofs ) \
{ stbir__simdf d ; \
stbir__simdf_load1z ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0011 ( c , cs ) ; \
stbir__simdf_load2 ( d , decode + ( ofs ) * 2 ) ; \
stbir__simdf_madd ( tot0 , tot0 , d , c ) ; }
# define stbir__2_coeff_remnant( ofs ) \
stbir__simdf_load2 ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0011 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 2 ) ;
# define stbir__3_coeff_remnant( ofs ) \
{ stbir__simdf d ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0011 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 2 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_load2z ( d , decode + ( ofs ) * 2 + 4 ) ; \
stbir__simdf_madd ( tot1 , tot1 , d , c ) ; }
# define stbir__store_output() \
stbir__simdf_add ( tot0 , tot0 , tot1 ) ; \
stbir__simdf_0123to2301 ( c , tot0 ) ; \
stbir__simdf_add ( tot0 , tot0 , c ) ; \
stbir__simdf_store2 ( output , tot0 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 2 ;
# endif
# else
# define stbir__1_coeff_only() \
float tota , totb , c ; \
c = hc [ 0 ] ; \
tota = decode [ 0 ] * c ; \
totb = decode [ 1 ] * c ;
# define stbir__2_coeff_only() \
float tota , totb , c ; \
c = hc [ 0 ] ; \
tota = decode [ 0 ] * c ; \
totb = decode [ 1 ] * c ; \
c = hc [ 1 ] ; \
tota + = decode [ 2 ] * c ; \
totb + = decode [ 3 ] * c ;
// this weird order of add matches the simd
# define stbir__3_coeff_only() \
float tota , totb , c ; \
c = hc [ 0 ] ; \
tota = decode [ 0 ] * c ; \
totb = decode [ 1 ] * c ; \
c = hc [ 2 ] ; \
tota + = decode [ 4 ] * c ; \
totb + = decode [ 5 ] * c ; \
c = hc [ 1 ] ; \
tota + = decode [ 2 ] * c ; \
totb + = decode [ 3 ] * c ;
# define stbir__store_output_tiny() \
output [ 0 ] = tota ; \
output [ 1 ] = totb ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 2 ;
# define stbir__4_coeff_start() \
float tota0 , tota1 , tota2 , tota3 , totb0 , totb1 , totb2 , totb3 , c ; \
c = hc [ 0 ] ; \
tota0 = decode [ 0 ] * c ; \
totb0 = decode [ 1 ] * c ; \
c = hc [ 1 ] ; \
tota1 = decode [ 2 ] * c ; \
totb1 = decode [ 3 ] * c ; \
c = hc [ 2 ] ; \
tota2 = decode [ 4 ] * c ; \
totb2 = decode [ 5 ] * c ; \
c = hc [ 3 ] ; \
tota3 = decode [ 6 ] * c ; \
totb3 = decode [ 7 ] * c ;
# define stbir__4_coeff_continue_from_4( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 2 ] * c ; \
totb0 + = decode [ 1 + ( ofs ) * 2 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
tota1 + = decode [ 2 + ( ofs ) * 2 ] * c ; \
totb1 + = decode [ 3 + ( ofs ) * 2 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
tota2 + = decode [ 4 + ( ofs ) * 2 ] * c ; \
totb2 + = decode [ 5 + ( ofs ) * 2 ] * c ; \
c = hc [ 3 + ( ofs ) ] ; \
tota3 + = decode [ 6 + ( ofs ) * 2 ] * c ; \
totb3 + = decode [ 7 + ( ofs ) * 2 ] * c ;
# define stbir__1_coeff_remnant( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 2 ] * c ; \
totb0 + = decode [ 1 + ( ofs ) * 2 ] * c ;
# define stbir__2_coeff_remnant( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 2 ] * c ; \
totb0 + = decode [ 1 + ( ofs ) * 2 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
tota1 + = decode [ 2 + ( ofs ) * 2 ] * c ; \
totb1 + = decode [ 3 + ( ofs ) * 2 ] * c ;
# define stbir__3_coeff_remnant( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 2 ] * c ; \
totb0 + = decode [ 1 + ( ofs ) * 2 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
tota1 + = decode [ 2 + ( ofs ) * 2 ] * c ; \
totb1 + = decode [ 3 + ( ofs ) * 2 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
tota2 + = decode [ 4 + ( ofs ) * 2 ] * c ; \
totb2 + = decode [ 5 + ( ofs ) * 2 ] * c ;
# define stbir__store_output() \
output [ 0 ] = ( tota0 + tota2 ) + ( tota1 + tota3 ) ; \
output [ 1 ] = ( totb0 + totb2 ) + ( totb1 + totb3 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 2 ;
# endif
# define STBIR__horizontal_channels 2
# define STB_IMAGE_RESIZE_DO_HORIZONTALS
# include STBIR__HEADER_FILENAME
//=================
// Do 3 channel horizontal routines
# ifdef STBIR_SIMD
# define stbir__1_coeff_only() \
stbir__simdf tot , c , d ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1z ( c , hc ) ; \
stbir__simdf_0123to0001 ( c , c ) ; \
stbir__simdf_load ( d , decode ) ; \
stbir__simdf_mult ( tot , d , c ) ;
# define stbir__2_coeff_only() \
stbir__simdf tot , c , cs , d ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2 ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_load ( d , decode ) ; \
stbir__simdf_mult ( tot , d , c ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_load ( d , decode + 3 ) ; \
stbir__simdf_madd ( tot , tot , d , c ) ;
# define stbir__3_coeff_only() \
stbir__simdf tot , c , d , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_load ( d , decode ) ; \
stbir__simdf_mult ( tot , d , c ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_load ( d , decode + 3 ) ; \
stbir__simdf_madd ( tot , tot , d , c ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_load ( d , decode + 6 ) ; \
stbir__simdf_madd ( tot , tot , d , c ) ;
# define stbir__store_output_tiny() \
stbir__simdf_store2 ( output , tot ) ; \
stbir__simdf_0123to2301 ( tot , tot ) ; \
stbir__simdf_store1 ( output + 2 , tot ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 3 ;
# ifdef STBIR_SIMD8
// we're loading from the XXXYYY decode by -1 to get the XXXYYY into different halves of the AVX reg fyi
# define stbir__4_coeff_start() \
stbir__simdf8 tot0 , tot1 , c , cs ; stbir__simdf t ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc ) ; \
stbir__simdf8_0123to00001111 ( c , cs ) ; \
stbir__simdf8_mult_mem ( tot0 , c , decode - 1 ) ; \
stbir__simdf8_0123to22223333 ( c , cs ) ; \
stbir__simdf8_mult_mem ( tot1 , c , decode + 6 - 1 ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00001111 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 3 - 1 ) ; \
stbir__simdf8_0123to22223333 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 3 + 6 - 1 ) ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1rep4 ( t , hc + ( ofs ) ) ; \
stbir__simdf8_madd_mem4 ( tot0 , tot0 , t , decode + ( ofs ) * 3 - 1 ) ;
# define stbir__2_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) - 2 ) ; \
stbir__simdf8_0123to22223333 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 3 - 1 ) ;
# define stbir__3_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00001111 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 3 - 1 ) ; \
stbir__simdf8_0123to2222 ( t , cs ) ; \
stbir__simdf8_madd_mem4 ( tot1 , tot1 , t , decode + ( ofs ) * 3 + 6 - 1 ) ;
# define stbir__store_output() \
stbir__simdf8_add ( tot0 , tot0 , tot1 ) ; \
stbir__simdf_0123to1230 ( t , stbir__if_simdf8_cast_to_simdf4 ( tot0 ) ) ; \
stbir__simdf8_add4halves ( t , t , tot0 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 3 ; \
if ( output < output_end ) \
{ \
stbir__simdf_store ( output - 3 , t ) ; \
continue ; \
} \
{ stbir__simdf tt ; stbir__simdf_0123to2301 ( tt , t ) ; \
stbir__simdf_store2 ( output - 3 , t ) ; \
stbir__simdf_store1 ( output + 2 - 3 , tt ) ; } \
break ;
# else
# define stbir__4_coeff_start() \
stbir__simdf tot0 , tot1 , tot2 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0001 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf_0123to1122 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot1 , c , decode + 4 ) ; \
stbir__simdf_0123to2333 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot2 , c , decode + 8 ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0001 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 3 ) ; \
stbir__simdf_0123to1122 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 3 + 4 ) ; \
stbir__simdf_0123to2333 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot2 , tot2 , c , decode + ( ofs ) * 3 + 8 ) ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1z ( c , hc + ( ofs ) ) ; \
stbir__simdf_0123to0001 ( c , c ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 3 ) ;
# define stbir__2_coeff_remnant( ofs ) \
{ stbir__simdf d ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2z ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0001 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 3 ) ; \
stbir__simdf_0123to1122 ( c , cs ) ; \
stbir__simdf_load2z ( d , decode + ( ofs ) * 3 + 4 ) ; \
stbir__simdf_madd ( tot1 , tot1 , c , d ) ; }
# define stbir__3_coeff_remnant( ofs ) \
{ stbir__simdf d ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0001 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 3 ) ; \
stbir__simdf_0123to1122 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 3 + 4 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_load1z ( d , decode + ( ofs ) * 3 + 8 ) ; \
stbir__simdf_madd ( tot2 , tot2 , c , d ) ; }
# define stbir__store_output() \
stbir__simdf_0123ABCDto3ABx ( c , tot0 , tot1 ) ; \
stbir__simdf_0123ABCDto23Ax ( cs , tot1 , tot2 ) ; \
stbir__simdf_0123to1230 ( tot2 , tot2 ) ; \
stbir__simdf_add ( tot0 , tot0 , cs ) ; \
stbir__simdf_add ( c , c , tot2 ) ; \
stbir__simdf_add ( tot0 , tot0 , c ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 3 ; \
if ( output < output_end ) \
{ \
stbir__simdf_store ( output - 3 , tot0 ) ; \
continue ; \
} \
stbir__simdf_0123to2301 ( tot1 , tot0 ) ; \
stbir__simdf_store2 ( output - 3 , tot0 ) ; \
stbir__simdf_store1 ( output + 2 - 3 , tot1 ) ; \
break ;
# endif
# else
# define stbir__1_coeff_only() \
float tot0 , tot1 , tot2 , c ; \
c = hc [ 0 ] ; \
tot0 = decode [ 0 ] * c ; \
tot1 = decode [ 1 ] * c ; \
tot2 = decode [ 2 ] * c ;
# define stbir__2_coeff_only() \
float tot0 , tot1 , tot2 , c ; \
c = hc [ 0 ] ; \
tot0 = decode [ 0 ] * c ; \
tot1 = decode [ 1 ] * c ; \
tot2 = decode [ 2 ] * c ; \
c = hc [ 1 ] ; \
tot0 + = decode [ 3 ] * c ; \
tot1 + = decode [ 4 ] * c ; \
tot2 + = decode [ 5 ] * c ;
# define stbir__3_coeff_only() \
float tot0 , tot1 , tot2 , c ; \
c = hc [ 0 ] ; \
tot0 = decode [ 0 ] * c ; \
tot1 = decode [ 1 ] * c ; \
tot2 = decode [ 2 ] * c ; \
c = hc [ 1 ] ; \
tot0 + = decode [ 3 ] * c ; \
tot1 + = decode [ 4 ] * c ; \
tot2 + = decode [ 5 ] * c ; \
c = hc [ 2 ] ; \
tot0 + = decode [ 6 ] * c ; \
tot1 + = decode [ 7 ] * c ; \
tot2 + = decode [ 8 ] * c ;
# define stbir__store_output_tiny() \
output [ 0 ] = tot0 ; \
output [ 1 ] = tot1 ; \
output [ 2 ] = tot2 ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 3 ;
# define stbir__4_coeff_start() \
float tota0 , tota1 , tota2 , totb0 , totb1 , totb2 , totc0 , totc1 , totc2 , totd0 , totd1 , totd2 , c ; \
c = hc [ 0 ] ; \
tota0 = decode [ 0 ] * c ; \
tota1 = decode [ 1 ] * c ; \
tota2 = decode [ 2 ] * c ; \
c = hc [ 1 ] ; \
totb0 = decode [ 3 ] * c ; \
totb1 = decode [ 4 ] * c ; \
totb2 = decode [ 5 ] * c ; \
c = hc [ 2 ] ; \
totc0 = decode [ 6 ] * c ; \
totc1 = decode [ 7 ] * c ; \
totc2 = decode [ 8 ] * c ; \
c = hc [ 3 ] ; \
totd0 = decode [ 9 ] * c ; \
totd1 = decode [ 10 ] * c ; \
totd2 = decode [ 11 ] * c ;
# define stbir__4_coeff_continue_from_4( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 3 ] * c ; \
tota1 + = decode [ 1 + ( ofs ) * 3 ] * c ; \
tota2 + = decode [ 2 + ( ofs ) * 3 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
totb0 + = decode [ 3 + ( ofs ) * 3 ] * c ; \
totb1 + = decode [ 4 + ( ofs ) * 3 ] * c ; \
totb2 + = decode [ 5 + ( ofs ) * 3 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
totc0 + = decode [ 6 + ( ofs ) * 3 ] * c ; \
totc1 + = decode [ 7 + ( ofs ) * 3 ] * c ; \
totc2 + = decode [ 8 + ( ofs ) * 3 ] * c ; \
c = hc [ 3 + ( ofs ) ] ; \
totd0 + = decode [ 9 + ( ofs ) * 3 ] * c ; \
totd1 + = decode [ 10 + ( ofs ) * 3 ] * c ; \
totd2 + = decode [ 11 + ( ofs ) * 3 ] * c ;
# define stbir__1_coeff_remnant( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 3 ] * c ; \
tota1 + = decode [ 1 + ( ofs ) * 3 ] * c ; \
tota2 + = decode [ 2 + ( ofs ) * 3 ] * c ;
# define stbir__2_coeff_remnant( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 3 ] * c ; \
tota1 + = decode [ 1 + ( ofs ) * 3 ] * c ; \
tota2 + = decode [ 2 + ( ofs ) * 3 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
totb0 + = decode [ 3 + ( ofs ) * 3 ] * c ; \
totb1 + = decode [ 4 + ( ofs ) * 3 ] * c ; \
totb2 + = decode [ 5 + ( ofs ) * 3 ] * c ; \
# define stbir__3_coeff_remnant( ofs ) \
c = hc [ 0 + ( ofs ) ] ; \
tota0 + = decode [ 0 + ( ofs ) * 3 ] * c ; \
tota1 + = decode [ 1 + ( ofs ) * 3 ] * c ; \
tota2 + = decode [ 2 + ( ofs ) * 3 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
totb0 + = decode [ 3 + ( ofs ) * 3 ] * c ; \
totb1 + = decode [ 4 + ( ofs ) * 3 ] * c ; \
totb2 + = decode [ 5 + ( ofs ) * 3 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
totc0 + = decode [ 6 + ( ofs ) * 3 ] * c ; \
totc1 + = decode [ 7 + ( ofs ) * 3 ] * c ; \
totc2 + = decode [ 8 + ( ofs ) * 3 ] * c ;
# define stbir__store_output() \
output [ 0 ] = ( tota0 + totc0 ) + ( totb0 + totd0 ) ; \
output [ 1 ] = ( tota1 + totc1 ) + ( totb1 + totd1 ) ; \
output [ 2 ] = ( tota2 + totc2 ) + ( totb2 + totd2 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 3 ;
# endif
# define STBIR__horizontal_channels 3
# define STB_IMAGE_RESIZE_DO_HORIZONTALS
# include STBIR__HEADER_FILENAME
//=================
// Do 4 channel horizontal routines
# ifdef STBIR_SIMD
# define stbir__1_coeff_only() \
stbir__simdf tot , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1 ( c , hc ) ; \
stbir__simdf_0123to0000 ( c , c ) ; \
stbir__simdf_mult_mem ( tot , c , decode ) ;
# define stbir__2_coeff_only() \
stbir__simdf tot , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2 ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot , c , decode ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot , tot , c , decode + 4 ) ;
# define stbir__3_coeff_only() \
stbir__simdf tot , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot , c , decode ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot , tot , c , decode + 4 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot , tot , c , decode + 8 ) ;
# define stbir__store_output_tiny() \
stbir__simdf_store ( output , tot ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 4 ;
# ifdef STBIR_SIMD8
# define stbir__4_coeff_start() \
stbir__simdf8 tot0 , c , cs ; stbir__simdf t ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc ) ; \
stbir__simdf8_0123to00001111 ( c , cs ) ; \
stbir__simdf8_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf8_0123to22223333 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + 8 ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00001111 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 ) ; \
stbir__simdf8_0123to22223333 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 + 8 ) ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1rep4 ( t , hc + ( ofs ) ) ; \
stbir__simdf8_madd_mem4 ( tot0 , tot0 , t , decode + ( ofs ) * 4 ) ;
# define stbir__2_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) - 2 ) ; \
stbir__simdf8_0123to22223333 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 ) ;
# define stbir__3_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00001111 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 ) ; \
stbir__simdf8_0123to2222 ( t , cs ) ; \
stbir__simdf8_madd_mem4 ( tot0 , tot0 , t , decode + ( ofs ) * 4 + 8 ) ;
# define stbir__store_output() \
stbir__simdf8_add4halves ( t , stbir__if_simdf8_cast_to_simdf4 ( tot0 ) , tot0 ) ; \
stbir__simdf_store ( output , t ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 4 ;
# else
# define stbir__4_coeff_start() \
stbir__simdf tot0 , tot1 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot1 , c , decode + 4 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + 8 ) ; \
stbir__simdf_0123to3333 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + 12 ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 4 + 4 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 + 8 ) ; \
stbir__simdf_0123to3333 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 4 + 12 ) ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1 ( c , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , c ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 ) ;
# define stbir__2_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2 ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 4 + 4 ) ;
# define stbir__3_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 4 + 4 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 4 + 8 ) ;
# define stbir__store_output() \
stbir__simdf_add ( tot0 , tot0 , tot1 ) ; \
stbir__simdf_store ( output , tot0 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 4 ;
# endif
# else
# define stbir__1_coeff_only() \
float p0 , p1 , p2 , p3 , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 ] ; \
p0 = decode [ 0 ] * c ; \
p1 = decode [ 1 ] * c ; \
p2 = decode [ 2 ] * c ; \
p3 = decode [ 3 ] * c ;
# define stbir__2_coeff_only() \
float p0 , p1 , p2 , p3 , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 ] ; \
p0 = decode [ 0 ] * c ; \
p1 = decode [ 1 ] * c ; \
p2 = decode [ 2 ] * c ; \
p3 = decode [ 3 ] * c ; \
c = hc [ 1 ] ; \
p0 + = decode [ 4 ] * c ; \
p1 + = decode [ 5 ] * c ; \
p2 + = decode [ 6 ] * c ; \
p3 + = decode [ 7 ] * c ;
# define stbir__3_coeff_only() \
float p0 , p1 , p2 , p3 , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 ] ; \
p0 = decode [ 0 ] * c ; \
p1 = decode [ 1 ] * c ; \
p2 = decode [ 2 ] * c ; \
p3 = decode [ 3 ] * c ; \
c = hc [ 1 ] ; \
p0 + = decode [ 4 ] * c ; \
p1 + = decode [ 5 ] * c ; \
p2 + = decode [ 6 ] * c ; \
p3 + = decode [ 7 ] * c ; \
c = hc [ 2 ] ; \
p0 + = decode [ 8 ] * c ; \
p1 + = decode [ 9 ] * c ; \
p2 + = decode [ 10 ] * c ; \
p3 + = decode [ 11 ] * c ;
# define stbir__store_output_tiny() \
output [ 0 ] = p0 ; \
output [ 1 ] = p1 ; \
output [ 2 ] = p2 ; \
output [ 3 ] = p3 ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 4 ;
# define stbir__4_coeff_start() \
float x0 , x1 , x2 , x3 , y0 , y1 , y2 , y3 , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 ] ; \
x0 = decode [ 0 ] * c ; \
x1 = decode [ 1 ] * c ; \
x2 = decode [ 2 ] * c ; \
x3 = decode [ 3 ] * c ; \
c = hc [ 1 ] ; \
y0 = decode [ 4 ] * c ; \
y1 = decode [ 5 ] * c ; \
y2 = decode [ 6 ] * c ; \
y3 = decode [ 7 ] * c ; \
c = hc [ 2 ] ; \
x0 + = decode [ 8 ] * c ; \
x1 + = decode [ 9 ] * c ; \
x2 + = decode [ 10 ] * c ; \
x3 + = decode [ 11 ] * c ; \
c = hc [ 3 ] ; \
y0 + = decode [ 12 ] * c ; \
y1 + = decode [ 13 ] * c ; \
y2 + = decode [ 14 ] * c ; \
y3 + = decode [ 15 ] * c ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 4 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 4 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 4 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 4 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
y0 + = decode [ 4 + ( ofs ) * 4 ] * c ; \
y1 + = decode [ 5 + ( ofs ) * 4 ] * c ; \
y2 + = decode [ 6 + ( ofs ) * 4 ] * c ; \
y3 + = decode [ 7 + ( ofs ) * 4 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
x0 + = decode [ 8 + ( ofs ) * 4 ] * c ; \
x1 + = decode [ 9 + ( ofs ) * 4 ] * c ; \
x2 + = decode [ 10 + ( ofs ) * 4 ] * c ; \
x3 + = decode [ 11 + ( ofs ) * 4 ] * c ; \
c = hc [ 3 + ( ofs ) ] ; \
y0 + = decode [ 12 + ( ofs ) * 4 ] * c ; \
y1 + = decode [ 13 + ( ofs ) * 4 ] * c ; \
y2 + = decode [ 14 + ( ofs ) * 4 ] * c ; \
y3 + = decode [ 15 + ( ofs ) * 4 ] * c ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 4 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 4 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 4 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 4 ] * c ;
# define stbir__2_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 4 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 4 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 4 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 4 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
y0 + = decode [ 4 + ( ofs ) * 4 ] * c ; \
y1 + = decode [ 5 + ( ofs ) * 4 ] * c ; \
y2 + = decode [ 6 + ( ofs ) * 4 ] * c ; \
y3 + = decode [ 7 + ( ofs ) * 4 ] * c ;
# define stbir__3_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 4 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 4 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 4 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 4 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
y0 + = decode [ 4 + ( ofs ) * 4 ] * c ; \
y1 + = decode [ 5 + ( ofs ) * 4 ] * c ; \
y2 + = decode [ 6 + ( ofs ) * 4 ] * c ; \
y3 + = decode [ 7 + ( ofs ) * 4 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
x0 + = decode [ 8 + ( ofs ) * 4 ] * c ; \
x1 + = decode [ 9 + ( ofs ) * 4 ] * c ; \
x2 + = decode [ 10 + ( ofs ) * 4 ] * c ; \
x3 + = decode [ 11 + ( ofs ) * 4 ] * c ;
# define stbir__store_output() \
output [ 0 ] = x0 + y0 ; \
output [ 1 ] = x1 + y1 ; \
output [ 2 ] = x2 + y2 ; \
output [ 3 ] = x3 + y3 ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 4 ;
# endif
# define STBIR__horizontal_channels 4
# define STB_IMAGE_RESIZE_DO_HORIZONTALS
# include STBIR__HEADER_FILENAME
//=================
// Do 7 channel horizontal routines
# ifdef STBIR_SIMD
# define stbir__1_coeff_only() \
stbir__simdf tot0 , tot1 , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1 ( c , hc ) ; \
stbir__simdf_0123to0000 ( c , c ) ; \
stbir__simdf_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf_mult_mem ( tot1 , c , decode + 3 ) ;
# define stbir__2_coeff_only() \
stbir__simdf tot0 , tot1 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2 ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf_mult_mem ( tot1 , c , decode + 3 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + 7 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + 10 ) ;
# define stbir__3_coeff_only() \
stbir__simdf tot0 , tot1 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf_mult_mem ( tot1 , c , decode + 3 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + 7 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + 10 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + 14 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + 17 ) ;
# define stbir__store_output_tiny() \
stbir__simdf_store ( output + 3 , tot1 ) ; \
stbir__simdf_store ( output , tot0 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 7 ;
# ifdef STBIR_SIMD8
# define stbir__4_coeff_start() \
stbir__simdf8 tot0 , tot1 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc ) ; \
stbir__simdf8_0123to00000000 ( c , cs ) ; \
stbir__simdf8_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf8_0123to11111111 ( c , cs ) ; \
stbir__simdf8_mult_mem ( tot1 , c , decode + 7 ) ; \
stbir__simdf8_0123to22222222 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + 14 ) ; \
stbir__simdf8_0123to33333333 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot1 , tot1 , c , decode + 21 ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00000000 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ; \
stbir__simdf8_0123to11111111 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 7 ) ; \
stbir__simdf8_0123to22222222 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 + 14 ) ; \
stbir__simdf8_0123to33333333 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 21 ) ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load1b ( c , hc + ( ofs ) ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ;
# define stbir__2_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load1b ( c , hc + ( ofs ) ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ; \
stbir__simdf8_load1b ( c , hc + ( ofs ) + 1 ) ; \
stbir__simdf8_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 7 ) ;
# define stbir__3_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf8_load4b ( cs , hc + ( ofs ) ) ; \
stbir__simdf8_0123to00000000 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ; \
stbir__simdf8_0123to11111111 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 7 ) ; \
stbir__simdf8_0123to22222222 ( c , cs ) ; \
stbir__simdf8_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 + 14 ) ;
# define stbir__store_output() \
stbir__simdf8_add ( tot0 , tot0 , tot1 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 7 ; \
if ( output < output_end ) \
{ \
stbir__simdf8_store ( output - 7 , tot0 ) ; \
continue ; \
} \
stbir__simdf_store ( output - 7 + 3 , stbir__simdf_swiz ( stbir__simdf8_gettop4 ( tot0 ) , 0 , 0 , 1 , 2 ) ) ; \
stbir__simdf_store ( output - 7 , stbir__if_simdf8_cast_to_simdf4 ( tot0 ) ) ; \
break ;
# else
# define stbir__4_coeff_start() \
stbir__simdf tot0 , tot1 , tot2 , tot3 , c , cs ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot0 , c , decode ) ; \
stbir__simdf_mult_mem ( tot1 , c , decode + 3 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_mult_mem ( tot2 , c , decode + 7 ) ; \
stbir__simdf_mult_mem ( tot3 , c , decode + 10 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + 14 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + 17 ) ; \
stbir__simdf_0123to3333 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot2 , tot2 , c , decode + 21 ) ; \
stbir__simdf_madd_mem ( tot3 , tot3 , c , decode + 24 ) ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 3 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot2 , tot2 , c , decode + ( ofs ) * 7 + 7 ) ; \
stbir__simdf_madd_mem ( tot3 , tot3 , c , decode + ( ofs ) * 7 + 10 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 + 14 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 17 ) ; \
stbir__simdf_0123to3333 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot2 , tot2 , c , decode + ( ofs ) * 7 + 21 ) ; \
stbir__simdf_madd_mem ( tot3 , tot3 , c , decode + ( ofs ) * 7 + 24 ) ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load1 ( c , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , c ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 3 ) ; \
# define stbir__2_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load2 ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 3 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot2 , tot2 , c , decode + ( ofs ) * 7 + 7 ) ; \
stbir__simdf_madd_mem ( tot3 , tot3 , c , decode + ( ofs ) * 7 + 10 ) ;
# define stbir__3_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
stbir__simdf_load ( cs , hc + ( ofs ) ) ; \
stbir__simdf_0123to0000 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 3 ) ; \
stbir__simdf_0123to1111 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot2 , tot2 , c , decode + ( ofs ) * 7 + 7 ) ; \
stbir__simdf_madd_mem ( tot3 , tot3 , c , decode + ( ofs ) * 7 + 10 ) ; \
stbir__simdf_0123to2222 ( c , cs ) ; \
stbir__simdf_madd_mem ( tot0 , tot0 , c , decode + ( ofs ) * 7 + 14 ) ; \
stbir__simdf_madd_mem ( tot1 , tot1 , c , decode + ( ofs ) * 7 + 17 ) ;
# define stbir__store_output() \
stbir__simdf_add ( tot0 , tot0 , tot2 ) ; \
stbir__simdf_add ( tot1 , tot1 , tot3 ) ; \
stbir__simdf_store ( output + 3 , tot1 ) ; \
stbir__simdf_store ( output , tot0 ) ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 7 ;
# endif
# else
# define stbir__1_coeff_only() \
float tot0 , tot1 , tot2 , tot3 , tot4 , tot5 , tot6 , c ; \
c = hc [ 0 ] ; \
tot0 = decode [ 0 ] * c ; \
tot1 = decode [ 1 ] * c ; \
tot2 = decode [ 2 ] * c ; \
tot3 = decode [ 3 ] * c ; \
tot4 = decode [ 4 ] * c ; \
tot5 = decode [ 5 ] * c ; \
tot6 = decode [ 6 ] * c ;
# define stbir__2_coeff_only() \
float tot0 , tot1 , tot2 , tot3 , tot4 , tot5 , tot6 , c ; \
c = hc [ 0 ] ; \
tot0 = decode [ 0 ] * c ; \
tot1 = decode [ 1 ] * c ; \
tot2 = decode [ 2 ] * c ; \
tot3 = decode [ 3 ] * c ; \
tot4 = decode [ 4 ] * c ; \
tot5 = decode [ 5 ] * c ; \
tot6 = decode [ 6 ] * c ; \
c = hc [ 1 ] ; \
tot0 + = decode [ 7 ] * c ; \
tot1 + = decode [ 8 ] * c ; \
tot2 + = decode [ 9 ] * c ; \
tot3 + = decode [ 10 ] * c ; \
tot4 + = decode [ 11 ] * c ; \
tot5 + = decode [ 12 ] * c ; \
tot6 + = decode [ 13 ] * c ; \
# define stbir__3_coeff_only() \
float tot0 , tot1 , tot2 , tot3 , tot4 , tot5 , tot6 , c ; \
c = hc [ 0 ] ; \
tot0 = decode [ 0 ] * c ; \
tot1 = decode [ 1 ] * c ; \
tot2 = decode [ 2 ] * c ; \
tot3 = decode [ 3 ] * c ; \
tot4 = decode [ 4 ] * c ; \
tot5 = decode [ 5 ] * c ; \
tot6 = decode [ 6 ] * c ; \
c = hc [ 1 ] ; \
tot0 + = decode [ 7 ] * c ; \
tot1 + = decode [ 8 ] * c ; \
tot2 + = decode [ 9 ] * c ; \
tot3 + = decode [ 10 ] * c ; \
tot4 + = decode [ 11 ] * c ; \
tot5 + = decode [ 12 ] * c ; \
tot6 + = decode [ 13 ] * c ; \
c = hc [ 2 ] ; \
tot0 + = decode [ 14 ] * c ; \
tot1 + = decode [ 15 ] * c ; \
tot2 + = decode [ 16 ] * c ; \
tot3 + = decode [ 17 ] * c ; \
tot4 + = decode [ 18 ] * c ; \
tot5 + = decode [ 19 ] * c ; \
tot6 + = decode [ 20 ] * c ; \
# define stbir__store_output_tiny() \
output [ 0 ] = tot0 ; \
output [ 1 ] = tot1 ; \
output [ 2 ] = tot2 ; \
output [ 3 ] = tot3 ; \
output [ 4 ] = tot4 ; \
output [ 5 ] = tot5 ; \
output [ 6 ] = tot6 ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 7 ;
# define stbir__4_coeff_start() \
float x0 , x1 , x2 , x3 , x4 , x5 , x6 , y0 , y1 , y2 , y3 , y4 , y5 , y6 , c ; \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 ] ; \
x0 = decode [ 0 ] * c ; \
x1 = decode [ 1 ] * c ; \
x2 = decode [ 2 ] * c ; \
x3 = decode [ 3 ] * c ; \
x4 = decode [ 4 ] * c ; \
x5 = decode [ 5 ] * c ; \
x6 = decode [ 6 ] * c ; \
c = hc [ 1 ] ; \
y0 = decode [ 7 ] * c ; \
y1 = decode [ 8 ] * c ; \
y2 = decode [ 9 ] * c ; \
y3 = decode [ 10 ] * c ; \
y4 = decode [ 11 ] * c ; \
y5 = decode [ 12 ] * c ; \
y6 = decode [ 13 ] * c ; \
c = hc [ 2 ] ; \
x0 + = decode [ 14 ] * c ; \
x1 + = decode [ 15 ] * c ; \
x2 + = decode [ 16 ] * c ; \
x3 + = decode [ 17 ] * c ; \
x4 + = decode [ 18 ] * c ; \
x5 + = decode [ 19 ] * c ; \
x6 + = decode [ 20 ] * c ; \
c = hc [ 3 ] ; \
y0 + = decode [ 21 ] * c ; \
y1 + = decode [ 22 ] * c ; \
y2 + = decode [ 23 ] * c ; \
y3 + = decode [ 24 ] * c ; \
y4 + = decode [ 25 ] * c ; \
y5 + = decode [ 26 ] * c ; \
y6 + = decode [ 27 ] * c ;
# define stbir__4_coeff_continue_from_4( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 7 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 7 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 7 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 7 ] * c ; \
x4 + = decode [ 4 + ( ofs ) * 7 ] * c ; \
x5 + = decode [ 5 + ( ofs ) * 7 ] * c ; \
x6 + = decode [ 6 + ( ofs ) * 7 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
y0 + = decode [ 7 + ( ofs ) * 7 ] * c ; \
y1 + = decode [ 8 + ( ofs ) * 7 ] * c ; \
y2 + = decode [ 9 + ( ofs ) * 7 ] * c ; \
y3 + = decode [ 10 + ( ofs ) * 7 ] * c ; \
y4 + = decode [ 11 + ( ofs ) * 7 ] * c ; \
y5 + = decode [ 12 + ( ofs ) * 7 ] * c ; \
y6 + = decode [ 13 + ( ofs ) * 7 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
x0 + = decode [ 14 + ( ofs ) * 7 ] * c ; \
x1 + = decode [ 15 + ( ofs ) * 7 ] * c ; \
x2 + = decode [ 16 + ( ofs ) * 7 ] * c ; \
x3 + = decode [ 17 + ( ofs ) * 7 ] * c ; \
x4 + = decode [ 18 + ( ofs ) * 7 ] * c ; \
x5 + = decode [ 19 + ( ofs ) * 7 ] * c ; \
x6 + = decode [ 20 + ( ofs ) * 7 ] * c ; \
c = hc [ 3 + ( ofs ) ] ; \
y0 + = decode [ 21 + ( ofs ) * 7 ] * c ; \
y1 + = decode [ 22 + ( ofs ) * 7 ] * c ; \
y2 + = decode [ 23 + ( ofs ) * 7 ] * c ; \
y3 + = decode [ 24 + ( ofs ) * 7 ] * c ; \
y4 + = decode [ 25 + ( ofs ) * 7 ] * c ; \
y5 + = decode [ 26 + ( ofs ) * 7 ] * c ; \
y6 + = decode [ 27 + ( ofs ) * 7 ] * c ;
# define stbir__1_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 7 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 7 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 7 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 7 ] * c ; \
x4 + = decode [ 4 + ( ofs ) * 7 ] * c ; \
x5 + = decode [ 5 + ( ofs ) * 7 ] * c ; \
x6 + = decode [ 6 + ( ofs ) * 7 ] * c ; \
# define stbir__2_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 7 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 7 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 7 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 7 ] * c ; \
x4 + = decode [ 4 + ( ofs ) * 7 ] * c ; \
x5 + = decode [ 5 + ( ofs ) * 7 ] * c ; \
x6 + = decode [ 6 + ( ofs ) * 7 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
y0 + = decode [ 7 + ( ofs ) * 7 ] * c ; \
y1 + = decode [ 8 + ( ofs ) * 7 ] * c ; \
y2 + = decode [ 9 + ( ofs ) * 7 ] * c ; \
y3 + = decode [ 10 + ( ofs ) * 7 ] * c ; \
y4 + = decode [ 11 + ( ofs ) * 7 ] * c ; \
y5 + = decode [ 12 + ( ofs ) * 7 ] * c ; \
y6 + = decode [ 13 + ( ofs ) * 7 ] * c ; \
# define stbir__3_coeff_remnant( ofs ) \
STBIR_SIMD_NO_UNROLL ( decode ) ; \
c = hc [ 0 + ( ofs ) ] ; \
x0 + = decode [ 0 + ( ofs ) * 7 ] * c ; \
x1 + = decode [ 1 + ( ofs ) * 7 ] * c ; \
x2 + = decode [ 2 + ( ofs ) * 7 ] * c ; \
x3 + = decode [ 3 + ( ofs ) * 7 ] * c ; \
x4 + = decode [ 4 + ( ofs ) * 7 ] * c ; \
x5 + = decode [ 5 + ( ofs ) * 7 ] * c ; \
x6 + = decode [ 6 + ( ofs ) * 7 ] * c ; \
c = hc [ 1 + ( ofs ) ] ; \
y0 + = decode [ 7 + ( ofs ) * 7 ] * c ; \
y1 + = decode [ 8 + ( ofs ) * 7 ] * c ; \
y2 + = decode [ 9 + ( ofs ) * 7 ] * c ; \
y3 + = decode [ 10 + ( ofs ) * 7 ] * c ; \
y4 + = decode [ 11 + ( ofs ) * 7 ] * c ; \
y5 + = decode [ 12 + ( ofs ) * 7 ] * c ; \
y6 + = decode [ 13 + ( ofs ) * 7 ] * c ; \
c = hc [ 2 + ( ofs ) ] ; \
x0 + = decode [ 14 + ( ofs ) * 7 ] * c ; \
x1 + = decode [ 15 + ( ofs ) * 7 ] * c ; \
x2 + = decode [ 16 + ( ofs ) * 7 ] * c ; \
x3 + = decode [ 17 + ( ofs ) * 7 ] * c ; \
x4 + = decode [ 18 + ( ofs ) * 7 ] * c ; \
x5 + = decode [ 19 + ( ofs ) * 7 ] * c ; \
x6 + = decode [ 20 + ( ofs ) * 7 ] * c ; \
# define stbir__store_output() \
output [ 0 ] = x0 + y0 ; \
output [ 1 ] = x1 + y1 ; \
output [ 2 ] = x2 + y2 ; \
output [ 3 ] = x3 + y3 ; \
output [ 4 ] = x4 + y4 ; \
output [ 5 ] = x5 + y5 ; \
output [ 6 ] = x6 + y6 ; \
horizontal_coefficients + = coefficient_width ; \
+ + horizontal_contributors ; \
output + = 7 ;
# endif
# define STBIR__horizontal_channels 7
# define STB_IMAGE_RESIZE_DO_HORIZONTALS
# include STBIR__HEADER_FILENAME
// include all of the vertical resamplers (both scatter and gather versions)
# define STBIR__vertical_channels 1
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 1
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 2
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 2
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 3
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 3
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 4
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 4
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 5
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 5
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 6
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 6
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 7
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 7
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 8
# define STB_IMAGE_RESIZE_DO_VERTICALS
# include STBIR__HEADER_FILENAME
# define STBIR__vertical_channels 8
# define STB_IMAGE_RESIZE_DO_VERTICALS
# define STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# include STBIR__HEADER_FILENAME
typedef void STBIR_VERTICAL_GATHERFUNC ( float * output , float const * coeffs , float const * * inputs , float const * input0_end ) ;
static STBIR_VERTICAL_GATHERFUNC * stbir__vertical_gathers [ 8 ] =
{
stbir__vertical_gather_with_1_coeffs , stbir__vertical_gather_with_2_coeffs , stbir__vertical_gather_with_3_coeffs , stbir__vertical_gather_with_4_coeffs , stbir__vertical_gather_with_5_coeffs , stbir__vertical_gather_with_6_coeffs , stbir__vertical_gather_with_7_coeffs , stbir__vertical_gather_with_8_coeffs
} ;
static STBIR_VERTICAL_GATHERFUNC * stbir__vertical_gathers_continues [ 8 ] =
{
stbir__vertical_gather_with_1_coeffs_cont , stbir__vertical_gather_with_2_coeffs_cont , stbir__vertical_gather_with_3_coeffs_cont , stbir__vertical_gather_with_4_coeffs_cont , stbir__vertical_gather_with_5_coeffs_cont , stbir__vertical_gather_with_6_coeffs_cont , stbir__vertical_gather_with_7_coeffs_cont , stbir__vertical_gather_with_8_coeffs_cont
} ;
typedef void STBIR_VERTICAL_SCATTERFUNC ( float * * outputs , float const * coeffs , float const * input , float const * input_end ) ;
static STBIR_VERTICAL_SCATTERFUNC * stbir__vertical_scatter_sets [ 8 ] =
{
stbir__vertical_scatter_with_1_coeffs , stbir__vertical_scatter_with_2_coeffs , stbir__vertical_scatter_with_3_coeffs , stbir__vertical_scatter_with_4_coeffs , stbir__vertical_scatter_with_5_coeffs , stbir__vertical_scatter_with_6_coeffs , stbir__vertical_scatter_with_7_coeffs , stbir__vertical_scatter_with_8_coeffs
} ;
static STBIR_VERTICAL_SCATTERFUNC * stbir__vertical_scatter_blends [ 8 ] =
{
stbir__vertical_scatter_with_1_coeffs_cont , stbir__vertical_scatter_with_2_coeffs_cont , stbir__vertical_scatter_with_3_coeffs_cont , stbir__vertical_scatter_with_4_coeffs_cont , stbir__vertical_scatter_with_5_coeffs_cont , stbir__vertical_scatter_with_6_coeffs_cont , stbir__vertical_scatter_with_7_coeffs_cont , stbir__vertical_scatter_with_8_coeffs_cont
} ;
static void stbir__encode_scanline ( stbir__info const * stbir_info , void * output_buffer_data , float * encode_buffer , int row STBIR_ONLY_PROFILE_GET_SPLIT_INFO )
{
int num_pixels = stbir_info - > horizontal . scale_info . output_sub_size ;
int channels = stbir_info - > channels ;
int width_times_channels = num_pixels * channels ;
void * output_buffer ;
// un-alpha weight if we need to
if ( stbir_info - > alpha_unweight )
{
STBIR_PROFILE_START ( unalpha ) ;
stbir_info - > alpha_unweight ( encode_buffer , width_times_channels ) ;
STBIR_PROFILE_END ( unalpha ) ;
}
// write directly into output by default
output_buffer = output_buffer_data ;
// if we have an output callback, we first convert the decode buffer in place (and then hand that to the callback)
if ( stbir_info - > out_pixels_cb )
output_buffer = encode_buffer ;
STBIR_PROFILE_START ( encode ) ;
// convert into the output buffer
stbir_info - > encode_pixels ( output_buffer , width_times_channels , encode_buffer ) ;
STBIR_PROFILE_END ( encode ) ;
// if we have an output callback, call it to send the data
if ( stbir_info - > out_pixels_cb )
stbir_info - > out_pixels_cb ( output_buffer_data , num_pixels , row , stbir_info - > user_data ) ;
}
// Get the ring buffer pointer for an index
static float * stbir__get_ring_buffer_entry ( stbir__info const * stbir_info , stbir__per_split_info const * split_info , int index )
{
STBIR_ASSERT ( index < stbir_info - > ring_buffer_num_entries ) ;
# ifdef STBIR__SEPARATE_ALLOCATIONS
return split_info - > ring_buffers [ index ] ;
# else
return ( float * ) ( ( ( char * ) split_info - > ring_buffer ) + ( index * stbir_info - > ring_buffer_length_bytes ) ) ;
# endif
}
// Get the specified scan line from the ring buffer
static float * stbir__get_ring_buffer_scanline ( stbir__info const * stbir_info , stbir__per_split_info const * split_info , int get_scanline )
{
int ring_buffer_index = ( split_info - > ring_buffer_begin_index + ( get_scanline - split_info - > ring_buffer_first_scanline ) ) % stbir_info - > ring_buffer_num_entries ;
return stbir__get_ring_buffer_entry ( stbir_info , split_info , ring_buffer_index ) ;
}
static void stbir__resample_horizontal_gather ( stbir__info const * stbir_info , float * output_buffer , float const * input_buffer STBIR_ONLY_PROFILE_GET_SPLIT_INFO )
{
float const * decode_buffer = input_buffer - ( stbir_info - > scanline_extents . conservative . n0 * stbir_info - > effective_channels ) ;
STBIR_PROFILE_START ( horizontal ) ;
if ( ( stbir_info - > horizontal . filter_enum = = STBIR_FILTER_POINT_SAMPLE ) & & ( stbir_info - > horizontal . scale_info . scale = = 1.0f ) )
STBIR_MEMCPY ( output_buffer , input_buffer , stbir_info - > horizontal . scale_info . output_sub_size * sizeof ( float ) * stbir_info - > effective_channels ) ;
else
stbir_info - > horizontal_gather_channels ( output_buffer , stbir_info - > horizontal . scale_info . output_sub_size , decode_buffer , stbir_info - > horizontal . contributors , stbir_info - > horizontal . coefficients , stbir_info - > horizontal . coefficient_width ) ;
STBIR_PROFILE_END ( horizontal ) ;
}
static void stbir__resample_vertical_gather ( stbir__info const * stbir_info , stbir__per_split_info * split_info , int n , int contrib_n0 , int contrib_n1 , float const * vertical_coefficients )
{
float * encode_buffer = split_info - > vertical_buffer ;
float * decode_buffer = split_info - > decode_buffer ;
int vertical_first = stbir_info - > vertical_first ;
int width = ( vertical_first ) ? ( stbir_info - > scanline_extents . conservative . n1 - stbir_info - > scanline_extents . conservative . n0 + 1 ) : stbir_info - > horizontal . scale_info . output_sub_size ;
int width_times_channels = stbir_info - > effective_channels * width ;
STBIR_ASSERT ( stbir_info - > vertical . is_gather ) ;
// loop over the contributing scanlines and scale into the buffer
STBIR_PROFILE_START ( vertical ) ;
{
int k = 0 , total = contrib_n1 - contrib_n0 + 1 ;
STBIR_ASSERT ( total > 0 ) ;
do {
float const * inputs [ 8 ] ;
int i , cnt = total ; if ( cnt > 8 ) cnt = 8 ;
for ( i = 0 ; i < cnt ; i + + )
inputs [ i ] = stbir__get_ring_buffer_scanline ( stbir_info , split_info , k + i + contrib_n0 ) ;
// call the N scanlines at a time function (up to 8 scanlines of blending at once)
( ( k = = 0 ) ? stbir__vertical_gathers : stbir__vertical_gathers_continues ) [ cnt - 1 ] ( ( vertical_first ) ? decode_buffer : encode_buffer , vertical_coefficients + k , inputs , inputs [ 0 ] + width_times_channels ) ;
k + = cnt ;
total - = cnt ;
} while ( total ) ;
}
STBIR_PROFILE_END ( vertical ) ;
if ( vertical_first )
{
// Now resample the gathered vertical data in the horizontal axis into the encode buffer
stbir__resample_horizontal_gather ( stbir_info , encode_buffer , decode_buffer STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
}
stbir__encode_scanline ( stbir_info , ( ( char * ) stbir_info - > output_data ) + ( ( ptrdiff_t ) n * ( ptrdiff_t ) stbir_info - > output_stride_bytes ) ,
encode_buffer , n STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
}
static void stbir__decode_and_resample_for_vertical_gather_loop ( stbir__info const * stbir_info , stbir__per_split_info * split_info , int n )
{
int ring_buffer_index ;
float * ring_buffer ;
// Decode the nth scanline from the source image into the decode buffer.
stbir__decode_scanline ( stbir_info , n , split_info - > decode_buffer STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
// update new end scanline
split_info - > ring_buffer_last_scanline = n ;
// get ring buffer
ring_buffer_index = ( split_info - > ring_buffer_begin_index + ( split_info - > ring_buffer_last_scanline - split_info - > ring_buffer_first_scanline ) ) % stbir_info - > ring_buffer_num_entries ;
ring_buffer = stbir__get_ring_buffer_entry ( stbir_info , split_info , ring_buffer_index ) ;
// Now resample it into the ring buffer.
stbir__resample_horizontal_gather ( stbir_info , ring_buffer , split_info - > decode_buffer STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
// Now it's sitting in the ring buffer ready to be used as source for the vertical sampling.
}
static void stbir__vertical_gather_loop ( stbir__info const * stbir_info , stbir__per_split_info * split_info , int split_count )
{
int y , start_output_y , end_output_y ;
stbir__contributors * vertical_contributors = stbir_info - > vertical . contributors ;
float const * vertical_coefficients = stbir_info - > vertical . coefficients ;
STBIR_ASSERT ( stbir_info - > vertical . is_gather ) ;
start_output_y = split_info - > start_output_y ;
end_output_y = split_info [ split_count - 1 ] . end_output_y ;
vertical_contributors + = start_output_y ;
vertical_coefficients + = start_output_y * stbir_info - > vertical . coefficient_width ;
// initialize the ring buffer for gathering
split_info - > ring_buffer_begin_index = 0 ;
split_info - > ring_buffer_first_scanline = stbir_info - > vertical . extent_info . lowest ;
split_info - > ring_buffer_last_scanline = split_info - > ring_buffer_first_scanline - 1 ; // means "empty"
for ( y = start_output_y ; y < end_output_y ; y + + )
{
int in_first_scanline , in_last_scanline ;
in_first_scanline = vertical_contributors - > n0 ;
in_last_scanline = vertical_contributors - > n1 ;
// make sure the indexing hasn't broken
STBIR_ASSERT ( in_first_scanline > = split_info - > ring_buffer_first_scanline ) ;
// Load in new scanlines
while ( in_last_scanline > split_info - > ring_buffer_last_scanline )
{
STBIR_ASSERT ( ( split_info - > ring_buffer_last_scanline - split_info - > ring_buffer_first_scanline + 1 ) < = stbir_info - > ring_buffer_num_entries ) ;
// make sure there was room in the ring buffer when we add new scanlines
if ( ( split_info - > ring_buffer_last_scanline - split_info - > ring_buffer_first_scanline + 1 ) = = stbir_info - > ring_buffer_num_entries )
{
split_info - > ring_buffer_first_scanline + + ;
split_info - > ring_buffer_begin_index + + ;
}
if ( stbir_info - > vertical_first )
{
float * ring_buffer = stbir__get_ring_buffer_scanline ( stbir_info , split_info , + + split_info - > ring_buffer_last_scanline ) ;
// Decode the nth scanline from the source image into the decode buffer.
stbir__decode_scanline ( stbir_info , split_info - > ring_buffer_last_scanline , ring_buffer STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
}
else
{
stbir__decode_and_resample_for_vertical_gather_loop ( stbir_info , split_info , split_info - > ring_buffer_last_scanline + 1 ) ;
}
}
// Now all buffers should be ready to write a row of vertical sampling, so do it.
stbir__resample_vertical_gather ( stbir_info , split_info , y , in_first_scanline , in_last_scanline , vertical_coefficients ) ;
+ + vertical_contributors ;
vertical_coefficients + = stbir_info - > vertical . coefficient_width ;
}
}
# define STBIR__FLOAT_EMPTY_MARKER 3.0e+38F
# define STBIR__FLOAT_BUFFER_IS_EMPTY(ptr) ((ptr)[0]==STBIR__FLOAT_EMPTY_MARKER)
static void stbir__encode_first_scanline_from_scatter ( stbir__info const * stbir_info , stbir__per_split_info * split_info )
{
// evict a scanline out into the output buffer
float * ring_buffer_entry = stbir__get_ring_buffer_entry ( stbir_info , split_info , split_info - > ring_buffer_begin_index ) ;
// dump the scanline out
stbir__encode_scanline ( stbir_info , ( ( char * ) stbir_info - > output_data ) + ( ( ptrdiff_t ) split_info - > ring_buffer_first_scanline * ( ptrdiff_t ) stbir_info - > output_stride_bytes ) , ring_buffer_entry , split_info - > ring_buffer_first_scanline STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
// mark it as empty
ring_buffer_entry [ 0 ] = STBIR__FLOAT_EMPTY_MARKER ;
// advance the first scanline
split_info - > ring_buffer_first_scanline + + ;
if ( + + split_info - > ring_buffer_begin_index = = stbir_info - > ring_buffer_num_entries )
split_info - > ring_buffer_begin_index = 0 ;
}
static void stbir__horizontal_resample_and_encode_first_scanline_from_scatter ( stbir__info const * stbir_info , stbir__per_split_info * split_info )
{
// evict a scanline out into the output buffer
float * ring_buffer_entry = stbir__get_ring_buffer_entry ( stbir_info , split_info , split_info - > ring_buffer_begin_index ) ;
// Now resample it into the buffer.
stbir__resample_horizontal_gather ( stbir_info , split_info - > vertical_buffer , ring_buffer_entry STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
// dump the scanline out
stbir__encode_scanline ( stbir_info , ( ( char * ) stbir_info - > output_data ) + ( ( ptrdiff_t ) split_info - > ring_buffer_first_scanline * ( ptrdiff_t ) stbir_info - > output_stride_bytes ) , split_info - > vertical_buffer , split_info - > ring_buffer_first_scanline STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
// mark it as empty
ring_buffer_entry [ 0 ] = STBIR__FLOAT_EMPTY_MARKER ;
// advance the first scanline
split_info - > ring_buffer_first_scanline + + ;
if ( + + split_info - > ring_buffer_begin_index = = stbir_info - > ring_buffer_num_entries )
split_info - > ring_buffer_begin_index = 0 ;
}
static void stbir__resample_vertical_scatter ( stbir__info const * stbir_info , stbir__per_split_info * split_info , int n0 , int n1 , float const * vertical_coefficients , float const * vertical_buffer , float const * vertical_buffer_end )
{
STBIR_ASSERT ( ! stbir_info - > vertical . is_gather ) ;
STBIR_PROFILE_START ( vertical ) ;
{
int k = 0 , total = n1 - n0 + 1 ;
STBIR_ASSERT ( total > 0 ) ;
do {
float * outputs [ 8 ] ;
int i , n = total ; if ( n > 8 ) n = 8 ;
for ( i = 0 ; i < n ; i + + )
{
outputs [ i ] = stbir__get_ring_buffer_scanline ( stbir_info , split_info , k + i + n0 ) ;
if ( ( i ) & & ( STBIR__FLOAT_BUFFER_IS_EMPTY ( outputs [ i ] ) ! = STBIR__FLOAT_BUFFER_IS_EMPTY ( outputs [ 0 ] ) ) ) // make sure runs are of the same type
{
n = i ;
break ;
}
}
// call the scatter to N scanlines at a time function (up to 8 scanlines of scattering at once)
( ( STBIR__FLOAT_BUFFER_IS_EMPTY ( outputs [ 0 ] ) ) ? stbir__vertical_scatter_sets : stbir__vertical_scatter_blends ) [ n - 1 ] ( outputs , vertical_coefficients + k , vertical_buffer , vertical_buffer_end ) ;
k + = n ;
total - = n ;
} while ( total ) ;
}
STBIR_PROFILE_END ( vertical ) ;
}
typedef void stbir__handle_scanline_for_scatter_func ( stbir__info const * stbir_info , stbir__per_split_info * split_info ) ;
static void stbir__vertical_scatter_loop ( stbir__info const * stbir_info , stbir__per_split_info * split_info , int split_count )
{
int y , start_output_y , end_output_y , start_input_y , end_input_y ;
stbir__contributors * vertical_contributors = stbir_info - > vertical . contributors ;
float const * vertical_coefficients = stbir_info - > vertical . coefficients ;
stbir__handle_scanline_for_scatter_func * handle_scanline_for_scatter ;
void * scanline_scatter_buffer ;
void * scanline_scatter_buffer_end ;
int on_first_input_y , last_input_y ;
STBIR_ASSERT ( ! stbir_info - > vertical . is_gather ) ;
start_output_y = split_info - > start_output_y ;
end_output_y = split_info [ split_count - 1 ] . end_output_y ; // may do multiple split counts
start_input_y = split_info - > start_input_y ;
end_input_y = split_info [ split_count - 1 ] . end_input_y ;
// adjust for starting offset start_input_y
y = start_input_y + stbir_info - > vertical . filter_pixel_margin ;
vertical_contributors + = y ;
vertical_coefficients + = stbir_info - > vertical . coefficient_width * y ;
if ( stbir_info - > vertical_first )
{
handle_scanline_for_scatter = stbir__horizontal_resample_and_encode_first_scanline_from_scatter ;
scanline_scatter_buffer = split_info - > decode_buffer ;
scanline_scatter_buffer_end = ( ( char * ) scanline_scatter_buffer ) + sizeof ( float ) * stbir_info - > effective_channels * ( stbir_info - > scanline_extents . conservative . n1 - stbir_info - > scanline_extents . conservative . n0 + 1 ) ;
}
else
{
handle_scanline_for_scatter = stbir__encode_first_scanline_from_scatter ;
scanline_scatter_buffer = split_info - > vertical_buffer ;
scanline_scatter_buffer_end = ( ( char * ) scanline_scatter_buffer ) + sizeof ( float ) * stbir_info - > effective_channels * stbir_info - > horizontal . scale_info . output_sub_size ;
}
// initialize the ring buffer for scattering
split_info - > ring_buffer_first_scanline = start_output_y ;
split_info - > ring_buffer_last_scanline = - 1 ;
split_info - > ring_buffer_begin_index = - 1 ;
// mark all the buffers as empty to start
for ( y = 0 ; y < stbir_info - > ring_buffer_num_entries ; y + + )
stbir__get_ring_buffer_entry ( stbir_info , split_info , y ) [ 0 ] = STBIR__FLOAT_EMPTY_MARKER ; // only used on scatter
// do the loop in input space
on_first_input_y = 1 ; last_input_y = start_input_y ;
for ( y = start_input_y ; y < end_input_y ; y + + )
{
int out_first_scanline , out_last_scanline ;
out_first_scanline = vertical_contributors - > n0 ;
out_last_scanline = vertical_contributors - > n1 ;
STBIR_ASSERT ( out_last_scanline - out_first_scanline + 1 < = stbir_info - > ring_buffer_num_entries ) ;
if ( ( out_last_scanline > = out_first_scanline ) & & ( ( ( out_first_scanline > = start_output_y ) & & ( out_first_scanline < end_output_y ) ) | | ( ( out_last_scanline > = start_output_y ) & & ( out_last_scanline < end_output_y ) ) ) )
{
float const * vc = vertical_coefficients ;
// keep track of the range actually seen for the next resize
last_input_y = y ;
if ( ( on_first_input_y ) & & ( y > start_input_y ) )
split_info - > start_input_y = y ;
on_first_input_y = 0 ;
// clip the region
if ( out_first_scanline < start_output_y )
{
vc + = start_output_y - out_first_scanline ;
out_first_scanline = start_output_y ;
}
if ( out_last_scanline > = end_output_y )
out_last_scanline = end_output_y - 1 ;
// if very first scanline, init the index
if ( split_info - > ring_buffer_begin_index < 0 )
split_info - > ring_buffer_begin_index = out_first_scanline - start_output_y ;
STBIR_ASSERT ( split_info - > ring_buffer_begin_index < = out_first_scanline ) ;
// Decode the nth scanline from the source image into the decode buffer.
stbir__decode_scanline ( stbir_info , y , split_info - > decode_buffer STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
// When horizontal first, we resample horizontally into the vertical buffer before we scatter it out
if ( ! stbir_info - > vertical_first )
stbir__resample_horizontal_gather ( stbir_info , split_info - > vertical_buffer , split_info - > decode_buffer STBIR_ONLY_PROFILE_SET_SPLIT_INFO ) ;
// Now it's sitting in the buffer ready to be distributed into the ring buffers.
// evict from the ringbuffer, if we need are full
if ( ( ( split_info - > ring_buffer_last_scanline - split_info - > ring_buffer_first_scanline + 1 ) = = stbir_info - > ring_buffer_num_entries ) & &
( out_last_scanline > split_info - > ring_buffer_last_scanline ) )
handle_scanline_for_scatter ( stbir_info , split_info ) ;
// Now the horizontal buffer is ready to write to all ring buffer rows, so do it.
stbir__resample_vertical_scatter ( stbir_info , split_info , out_first_scanline , out_last_scanline , vc , ( float * ) scanline_scatter_buffer , ( float * ) scanline_scatter_buffer_end ) ;
// update the end of the buffer
if ( out_last_scanline > split_info - > ring_buffer_last_scanline )
split_info - > ring_buffer_last_scanline = out_last_scanline ;
}
+ + vertical_contributors ;
vertical_coefficients + = stbir_info - > vertical . coefficient_width ;
}
// now evict the scanlines that are left over in the ring buffer
while ( split_info - > ring_buffer_first_scanline < end_output_y )
handle_scanline_for_scatter ( stbir_info , split_info ) ;
// update the end_input_y if we do multiple resizes with the same data
+ + last_input_y ;
for ( y = 0 ; y < split_count ; y + + )
if ( split_info [ y ] . end_input_y > last_input_y )
split_info [ y ] . end_input_y = last_input_y ;
}
static stbir__kernel_callback * stbir__builtin_kernels [ ] = { 0 , stbir__filter_trapezoid , stbir__filter_triangle , stbir__filter_cubic , stbir__filter_catmullrom , stbir__filter_mitchell , stbir__filter_point } ;
static stbir__support_callback * stbir__builtin_supports [ ] = { 0 , stbir__support_trapezoid , stbir__support_one , stbir__support_two , stbir__support_two , stbir__support_two , stbir__support_zeropoint5 } ;
static void stbir__set_sampler ( stbir__sampler * samp , stbir_filter filter , stbir__kernel_callback * kernel , stbir__support_callback * support , stbir_edge edge , stbir__scale_info * scale_info , int always_gather , void * user_data )
{
// set filter
if ( filter = = 0 )
{
filter = STBIR_DEFAULT_FILTER_DOWNSAMPLE ; // default to downsample
if ( scale_info - > scale > = ( 1.0f - stbir__small_float ) )
{
if ( ( scale_info - > scale < = ( 1.0f + stbir__small_float ) ) & & ( STBIR_CEILF ( scale_info - > pixel_shift ) = = scale_info - > pixel_shift ) )
filter = STBIR_FILTER_POINT_SAMPLE ;
else
filter = STBIR_DEFAULT_FILTER_UPSAMPLE ;
}
}
samp - > filter_enum = filter ;
STBIR_ASSERT ( samp - > filter_enum ! = 0 ) ;
STBIR_ASSERT ( ( unsigned ) samp - > filter_enum < STBIR_FILTER_OTHER ) ;
samp - > filter_kernel = stbir__builtin_kernels [ filter ] ;
samp - > filter_support = stbir__builtin_supports [ filter ] ;
if ( kernel & & support )
{
samp - > filter_kernel = kernel ;
samp - > filter_support = support ;
samp - > filter_enum = STBIR_FILTER_OTHER ;
}
samp - > edge = edge ;
samp - > filter_pixel_width = stbir__get_filter_pixel_width ( samp - > filter_support , scale_info - > scale , user_data ) ;
// Gather is always better, but in extreme downsamples, you have to most or all of the data in memory
// For horizontal, we always have all the pixels, so we always use gather here (always_gather==1).
// For vertical, we use gather if scaling up (which means we will have samp->filter_pixel_width
// scanlines in memory at once).
samp - > is_gather = 0 ;
if ( scale_info - > scale > = ( 1.0f - stbir__small_float ) )
samp - > is_gather = 1 ;
else if ( ( always_gather ) | | ( samp - > filter_pixel_width < = STBIR_FORCE_GATHER_FILTER_SCANLINES_AMOUNT ) )
samp - > is_gather = 2 ;
// pre calculate stuff based on the above
samp - > coefficient_width = stbir__get_coefficient_width ( samp , samp - > is_gather , user_data ) ;
if ( edge = = STBIR_EDGE_WRAP )
if ( samp - > filter_pixel_width > ( scale_info - > input_full_size * 2 ) ) // this can only happen when shrinking to a single pixel
samp - > filter_pixel_width = scale_info - > input_full_size * 2 ;
// This is how much to expand buffers to account for filters seeking outside
// the image boundaries.
samp - > filter_pixel_margin = samp - > filter_pixel_width / 2 ;
samp - > num_contributors = stbir__get_contributors ( samp , samp - > is_gather ) ;
samp - > contributors_size = samp - > num_contributors * sizeof ( stbir__contributors ) ;
samp - > coefficients_size = samp - > num_contributors * samp - > coefficient_width * sizeof ( float ) + sizeof ( float ) ; // extra sizeof(float) is padding
samp - > gather_prescatter_contributors = 0 ;
samp - > gather_prescatter_coefficients = 0 ;
if ( samp - > is_gather = = 0 )
{
samp - > gather_prescatter_coefficient_width = samp - > filter_pixel_width ;
samp - > gather_prescatter_num_contributors = stbir__get_contributors ( samp , 2 ) ;
samp - > gather_prescatter_contributors_size = samp - > gather_prescatter_num_contributors * sizeof ( stbir__contributors ) ;
samp - > gather_prescatter_coefficients_size = samp - > gather_prescatter_num_contributors * samp - > gather_prescatter_coefficient_width * sizeof ( float ) ;
}
}
static void stbir__get_conservative_extents ( stbir__sampler * samp , stbir__contributors * range , void * user_data )
{
float scale = samp - > scale_info . scale ;
float out_shift = samp - > scale_info . pixel_shift ;
stbir__support_callback * support = samp - > filter_support ;
int input_full_size = samp - > scale_info . input_full_size ;
stbir_edge edge = samp - > edge ;
float inv_scale = samp - > scale_info . inv_scale ;
STBIR_ASSERT ( samp - > is_gather ! = 0 ) ;
if ( samp - > is_gather = = 1 )
{
int in_first_pixel , in_last_pixel ;
float out_filter_radius = support ( inv_scale , user_data ) * scale ;
stbir__calculate_in_pixel_range ( & in_first_pixel , & in_last_pixel , 0.5 , out_filter_radius , inv_scale , out_shift , input_full_size , edge ) ;
range - > n0 = in_first_pixel ;
stbir__calculate_in_pixel_range ( & in_first_pixel , & in_last_pixel , ( ( float ) ( samp - > scale_info . output_sub_size - 1 ) ) + 0.5f , out_filter_radius , inv_scale , out_shift , input_full_size , edge ) ;
range - > n1 = in_last_pixel ;
}
else if ( samp - > is_gather = = 2 ) // downsample gather, refine
{
float in_pixels_radius = support ( scale , user_data ) * inv_scale ;
int filter_pixel_margin = samp - > filter_pixel_margin ;
int output_sub_size = samp - > scale_info . output_sub_size ;
int input_end ;
int n ;
int in_first_pixel , in_last_pixel ;
// get a conservative area of the input range
stbir__calculate_in_pixel_range ( & in_first_pixel , & in_last_pixel , 0 , 0 , inv_scale , out_shift , input_full_size , edge ) ;
range - > n0 = in_first_pixel ;
stbir__calculate_in_pixel_range ( & in_first_pixel , & in_last_pixel , ( float ) output_sub_size , 0 , inv_scale , out_shift , input_full_size , edge ) ;
range - > n1 = in_last_pixel ;
// now go through the margin to the start of area to find bottom
n = range - > n0 + 1 ;
input_end = - filter_pixel_margin ;
while ( n > = input_end )
{
int out_first_pixel , out_last_pixel ;
stbir__calculate_out_pixel_range ( & out_first_pixel , & out_last_pixel , ( ( float ) n ) + 0.5f , in_pixels_radius , scale , out_shift , output_sub_size ) ;
if ( out_first_pixel > out_last_pixel )
break ;
if ( ( out_first_pixel < output_sub_size ) | | ( out_last_pixel > = 0 ) )
range - > n0 = n ;
- - n ;
}
// now go through the end of the area through the margin to find top
n = range - > n1 - 1 ;
input_end = n + 1 + filter_pixel_margin ;
while ( n < = input_end )
{
int out_first_pixel , out_last_pixel ;
stbir__calculate_out_pixel_range ( & out_first_pixel , & out_last_pixel , ( ( float ) n ) + 0.5f , in_pixels_radius , scale , out_shift , output_sub_size ) ;
if ( out_first_pixel > out_last_pixel )
break ;
if ( ( out_first_pixel < output_sub_size ) | | ( out_last_pixel > = 0 ) )
range - > n1 = n ;
+ + n ;
}
}
if ( samp - > edge = = STBIR_EDGE_WRAP )
{
// if we are wrapping, and we are very close to the image size (so the edges might merge), just use the scanline up to the edge
if ( ( range - > n0 > 0 ) & & ( range - > n1 > = input_full_size ) )
{
int marg = range - > n1 - input_full_size + 1 ;
if ( ( marg + STBIR__MERGE_RUNS_PIXEL_THRESHOLD ) > = range - > n0 )
range - > n0 = 0 ;
}
if ( ( range - > n0 < 0 ) & & ( range - > n1 < ( input_full_size - 1 ) ) )
{
int marg = - range - > n0 ;
if ( ( input_full_size - marg - STBIR__MERGE_RUNS_PIXEL_THRESHOLD - 1 ) < = range - > n1 )
range - > n1 = input_full_size - 1 ;
}
}
else
{
// for non-edge-wrap modes, we never read over the edge, so clamp
if ( range - > n0 < 0 )
range - > n0 = 0 ;
if ( range - > n1 > = input_full_size )
range - > n1 = input_full_size - 1 ;
}
}
static void stbir__get_split_info ( stbir__per_split_info * split_info , int splits , int output_height , int vertical_pixel_margin , int input_full_height )
{
int i , cur ;
int left = output_height ;
cur = 0 ;
for ( i = 0 ; i < splits ; i + + )
{
int each ;
split_info [ i ] . start_output_y = cur ;
each = left / ( splits - i ) ;
split_info [ i ] . end_output_y = cur + each ;
cur + = each ;
left - = each ;
// scatter range (updated to minimum as you run it)
split_info [ i ] . start_input_y = - vertical_pixel_margin ;
split_info [ i ] . end_input_y = input_full_height + vertical_pixel_margin ;
}
}
static void stbir__free_internal_mem ( stbir__info * info )
{
# define STBIR__FREE_AND_CLEAR( ptr ) { if ( ptr ) { void * p = (ptr); (ptr) = 0; STBIR_FREE( p, info->user_data); } }
if ( info )
{
# ifndef STBIR__SEPARATE_ALLOCATIONS
STBIR__FREE_AND_CLEAR ( info - > alloced_mem ) ;
# else
int i , j ;
if ( ( info - > vertical . gather_prescatter_contributors ) & & ( ( void * ) info - > vertical . gather_prescatter_contributors ! = ( void * ) info - > split_info [ 0 ] . decode_buffer ) )
{
STBIR__FREE_AND_CLEAR ( info - > vertical . gather_prescatter_coefficients ) ;
STBIR__FREE_AND_CLEAR ( info - > vertical . gather_prescatter_contributors ) ;
}
for ( i = 0 ; i < info - > splits ; i + + )
{
for ( j = 0 ; j < info - > alloc_ring_buffer_num_entries ; j + + )
{
# ifdef STBIR_SIMD8
if ( info - > effective_channels = = 3 )
- - info - > split_info [ i ] . ring_buffers [ j ] ; // avx in 3 channel mode needs one float at the start of the buffer
# endif
STBIR__FREE_AND_CLEAR ( info - > split_info [ i ] . ring_buffers [ j ] ) ;
}
# ifdef STBIR_SIMD8
if ( info - > effective_channels = = 3 )
- - info - > split_info [ i ] . decode_buffer ; // avx in 3 channel mode needs one float at the start of the buffer
# endif
STBIR__FREE_AND_CLEAR ( info - > split_info [ i ] . decode_buffer ) ;
STBIR__FREE_AND_CLEAR ( info - > split_info [ i ] . ring_buffers ) ;
STBIR__FREE_AND_CLEAR ( info - > split_info [ i ] . vertical_buffer ) ;
}
STBIR__FREE_AND_CLEAR ( info - > split_info ) ;
if ( info - > vertical . coefficients ! = info - > horizontal . coefficients )
{
STBIR__FREE_AND_CLEAR ( info - > vertical . coefficients ) ;
STBIR__FREE_AND_CLEAR ( info - > vertical . contributors ) ;
}
STBIR__FREE_AND_CLEAR ( info - > horizontal . coefficients ) ;
STBIR__FREE_AND_CLEAR ( info - > horizontal . contributors ) ;
STBIR__FREE_AND_CLEAR ( info - > alloced_mem ) ;
STBIR__FREE_AND_CLEAR ( info ) ;
# endif
}
# undef STBIR__FREE_AND_CLEAR
}
static int stbir__get_max_split ( int splits , int height )
{
int i ;
int max = 0 ;
for ( i = 0 ; i < splits ; i + + )
{
int each = height / ( splits - i ) ;
if ( each > max )
max = each ;
height - = each ;
}
return max ;
}
static stbir__horizontal_gather_channels_func * * stbir__horizontal_gather_n_coeffs_funcs [ 8 ] =
{
0 , stbir__horizontal_gather_1_channels_with_n_coeffs_funcs , stbir__horizontal_gather_2_channels_with_n_coeffs_funcs , stbir__horizontal_gather_3_channels_with_n_coeffs_funcs , stbir__horizontal_gather_4_channels_with_n_coeffs_funcs , 0 , 0 , stbir__horizontal_gather_7_channels_with_n_coeffs_funcs
} ;
static stbir__horizontal_gather_channels_func * * stbir__horizontal_gather_channels_funcs [ 8 ] =
{
0 , stbir__horizontal_gather_1_channels_funcs , stbir__horizontal_gather_2_channels_funcs , stbir__horizontal_gather_3_channels_funcs , stbir__horizontal_gather_4_channels_funcs , 0 , 0 , stbir__horizontal_gather_7_channels_funcs
} ;
// there are six resize classifications: 0 == vertical scatter, 1 == vertical gather < 1x scale, 2 == vertical gather 1x-2x scale, 4 == vertical gather < 3x scale, 4 == vertical gather > 3x scale, 5 == <=4 pixel height, 6 == <=4 pixel wide column
# define STBIR_RESIZE_CLASSIFICATIONS 8
static float stbir__compute_weights [ 5 ] [ STBIR_RESIZE_CLASSIFICATIONS ] [ 4 ] = // 5 = 0=1chan, 1=2chan, 2=3chan, 3=4chan, 4=7chan
{
{
{ 1.00000f , 1.00000f , 0.31250f , 1.00000f } ,
{ 0.56250f , 0.59375f , 0.00000f , 0.96875f } ,
{ 1.00000f , 0.06250f , 0.00000f , 1.00000f } ,
{ 0.00000f , 0.09375f , 1.00000f , 1.00000f } ,
{ 1.00000f , 1.00000f , 1.00000f , 1.00000f } ,
{ 0.03125f , 0.12500f , 1.00000f , 1.00000f } ,
{ 0.06250f , 0.12500f , 0.00000f , 1.00000f } ,
{ 0.00000f , 1.00000f , 0.00000f , 0.03125f } ,
} , {
{ 0.00000f , 0.84375f , 0.00000f , 0.03125f } ,
{ 0.09375f , 0.93750f , 0.00000f , 0.78125f } ,
{ 0.87500f , 0.21875f , 0.00000f , 0.96875f } ,
{ 0.09375f , 0.09375f , 1.00000f , 1.00000f } ,
{ 1.00000f , 1.00000f , 1.00000f , 1.00000f } ,
{ 0.03125f , 0.12500f , 1.00000f , 1.00000f } ,
{ 0.06250f , 0.12500f , 0.00000f , 1.00000f } ,
{ 0.00000f , 1.00000f , 0.00000f , 0.53125f } ,
} , {
{ 0.00000f , 0.53125f , 0.00000f , 0.03125f } ,
{ 0.06250f , 0.96875f , 0.00000f , 0.53125f } ,
{ 0.87500f , 0.18750f , 0.00000f , 0.93750f } ,
{ 0.00000f , 0.09375f , 1.00000f , 1.00000f } ,
{ 1.00000f , 1.00000f , 1.00000f , 1.00000f } ,
{ 0.03125f , 0.12500f , 1.00000f , 1.00000f } ,
{ 0.06250f , 0.12500f , 0.00000f , 1.00000f } ,
{ 0.00000f , 1.00000f , 0.00000f , 0.56250f } ,
} , {
{ 0.00000f , 0.50000f , 0.00000f , 0.71875f } ,
{ 0.06250f , 0.84375f , 0.00000f , 0.87500f } ,
{ 1.00000f , 0.50000f , 0.50000f , 0.96875f } ,
{ 1.00000f , 0.09375f , 0.31250f , 0.50000f } ,
{ 1.00000f , 1.00000f , 1.00000f , 1.00000f } ,
{ 1.00000f , 0.03125f , 0.03125f , 0.53125f } ,
{ 0.18750f , 0.12500f , 0.00000f , 1.00000f } ,
{ 0.00000f , 1.00000f , 0.03125f , 0.18750f } ,
} , {
{ 0.00000f , 0.59375f , 0.00000f , 0.96875f } ,
{ 0.06250f , 0.81250f , 0.06250f , 0.59375f } ,
{ 0.75000f , 0.43750f , 0.12500f , 0.96875f } ,
{ 0.87500f , 0.06250f , 0.18750f , 0.43750f } ,
{ 1.00000f , 1.00000f , 1.00000f , 1.00000f } ,
{ 0.15625f , 0.12500f , 1.00000f , 1.00000f } ,
{ 0.06250f , 0.12500f , 0.00000f , 1.00000f } ,
{ 0.00000f , 1.00000f , 0.03125f , 0.34375f } ,
}
} ;
// structure that allow us to query and override info for training the costs
typedef struct STBIR__V_FIRST_INFO
{
double v_cost , h_cost ;
int control_v_first ; // 0 = no control, 1 = force hori, 2 = force vert
int v_first ;
int v_resize_classification ;
int is_gather ;
} STBIR__V_FIRST_INFO ;
# ifdef STBIR__V_FIRST_INFO_BUFFER
static STBIR__V_FIRST_INFO STBIR__V_FIRST_INFO_BUFFER = { 0 } ;
# define STBIR__V_FIRST_INFO_POINTER &STBIR__V_FIRST_INFO_BUFFER
# else
# define STBIR__V_FIRST_INFO_POINTER 0
# endif
// Figure out whether to scale along the horizontal or vertical first.
// This only *super* important when you are scaling by a massively
// different amount in the vertical vs the horizontal (for example, if
// you are scaling by 2x in the width, and 0.5x in the height, then you
// want to do the vertical scale first, because it's around 3x faster
// in that order.
//
// In more normal circumstances, this makes a 20-40% differences, so
// it's good to get right, but not critical. The normal way that you
// decide which direction goes first is just figuring out which
// direction does more multiplies. But with modern CPUs with their
// fancy caches and SIMD and high IPC abilities, so there's just a lot
// more that goes into it.
//
// My handwavy sort of solution is to have an app that does a whole
// bunch of timing for both vertical and horizontal first modes,
// and then another app that can read lots of these timing files
// and try to search for the best weights to use. Dotimings.c
// is the app that does a bunch of timings, and vf_train.c is the
// app that solves for the best weights (and shows how well it
// does currently).
static int stbir__should_do_vertical_first ( float weights_table [ STBIR_RESIZE_CLASSIFICATIONS ] [ 4 ] , int horizontal_filter_pixel_width , float horizontal_scale , int horizontal_output_size , int vertical_filter_pixel_width , float vertical_scale , int vertical_output_size , int is_gather , STBIR__V_FIRST_INFO * info )
{
double v_cost , h_cost ;
float * weights ;
int vertical_first ;
int v_classification ;
// categorize the resize into buckets
if ( ( vertical_output_size < = 4 ) | | ( horizontal_output_size < = 4 ) )
v_classification = ( vertical_output_size < horizontal_output_size ) ? 6 : 7 ;
else if ( vertical_scale < = 1.0f )
v_classification = ( is_gather ) ? 1 : 0 ;
else if ( vertical_scale < = 2.0f )
v_classification = 2 ;
else if ( vertical_scale < = 3.0f )
v_classification = 3 ;
else if ( vertical_scale < = 4.0f )
v_classification = 5 ;
else
v_classification = 6 ;
// use the right weights
weights = weights_table [ v_classification ] ;
// this is the costs when you don't take into account modern CPUs with high ipc and simd and caches - wish we had a better estimate
h_cost = ( float ) horizontal_filter_pixel_width * weights [ 0 ] + horizontal_scale * ( float ) vertical_filter_pixel_width * weights [ 1 ] ;
v_cost = ( float ) vertical_filter_pixel_width * weights [ 2 ] + vertical_scale * ( float ) horizontal_filter_pixel_width * weights [ 3 ] ;
// use computation estimate to decide vertical first or not
vertical_first = ( v_cost < = h_cost ) ? 1 : 0 ;
// save these, if requested
if ( info )
{
info - > h_cost = h_cost ;
info - > v_cost = v_cost ;
info - > v_resize_classification = v_classification ;
info - > v_first = vertical_first ;
info - > is_gather = is_gather ;
}
// and this allows us to override everything for testing (see dotiming.c)
if ( ( info ) & & ( info - > control_v_first ) )
vertical_first = ( info - > control_v_first = = 2 ) ? 1 : 0 ;
return vertical_first ;
}
// layout lookups - must match stbir_internal_pixel_layout
static unsigned char stbir__pixel_channels [ ] = {
1 , 2 , 3 , 3 , 4 , // 1ch, 2ch, rgb, bgr, 4ch
4 , 4 , 4 , 4 , 2 , 2 , // RGBA,BGRA,ARGB,ABGR,RA,AR
4 , 4 , 4 , 4 , 2 , 2 , // RGBA_PM,BGRA_PM,ARGB_PM,ABGR_PM,RA_PM,AR_PM
} ;
// the internal pixel layout enums are in a different order, so we can easily do range comparisons of types
// the public pixel layout is ordered in a way that if you cast num_channels (1-4) to the enum, you get something sensible
static stbir_internal_pixel_layout stbir__pixel_layout_convert_public_to_internal [ ] = {
STBIRI_BGR , STBIRI_1CHANNEL , STBIRI_2CHANNEL , STBIRI_RGB , STBIRI_RGBA ,
STBIRI_4CHANNEL , STBIRI_BGRA , STBIRI_ARGB , STBIRI_ABGR , STBIRI_RA , STBIRI_AR ,
STBIRI_RGBA_PM , STBIRI_BGRA_PM , STBIRI_ARGB_PM , STBIRI_ABGR_PM , STBIRI_RA_PM , STBIRI_AR_PM ,
} ;
static stbir__info * stbir__alloc_internal_mem_and_build_samplers ( stbir__sampler * horizontal , stbir__sampler * vertical , stbir__contributors * conservative , stbir_pixel_layout input_pixel_layout_public , stbir_pixel_layout output_pixel_layout_public , int splits , int new_x , int new_y , int fast_alpha , void * user_data STBIR_ONLY_PROFILE_BUILD_GET_INFO )
{
static char stbir_channel_count_index [ 8 ] = { 9 , 0 , 1 , 2 , 3 , 9 , 9 , 4 } ;
stbir__info * info = 0 ;
void * alloced = 0 ;
int alloced_total = 0 ;
int vertical_first ;
int decode_buffer_size , ring_buffer_length_bytes , ring_buffer_size , vertical_buffer_size , alloc_ring_buffer_num_entries ;
int alpha_weighting_type = 0 ; // 0=none, 1=simple, 2=fancy
int conservative_split_output_size = stbir__get_max_split ( splits , vertical - > scale_info . output_sub_size ) ;
stbir_internal_pixel_layout input_pixel_layout = stbir__pixel_layout_convert_public_to_internal [ input_pixel_layout_public ] ;
stbir_internal_pixel_layout output_pixel_layout = stbir__pixel_layout_convert_public_to_internal [ output_pixel_layout_public ] ;
int channels = stbir__pixel_channels [ input_pixel_layout ] ;
int effective_channels = channels ;
// first figure out what type of alpha weighting to use (if any)
if ( ( horizontal - > filter_enum ! = STBIR_FILTER_POINT_SAMPLE ) | | ( vertical - > filter_enum ! = STBIR_FILTER_POINT_SAMPLE ) ) // no alpha weighting on point sampling
{
if ( ( input_pixel_layout > = STBIRI_RGBA ) & & ( input_pixel_layout < = STBIRI_AR ) & & ( output_pixel_layout > = STBIRI_RGBA ) & & ( output_pixel_layout < = STBIRI_AR ) )
{
if ( fast_alpha )
{
alpha_weighting_type = 4 ;
}
else
{
static int fancy_alpha_effective_cnts [ 6 ] = { 7 , 7 , 7 , 7 , 3 , 3 } ;
alpha_weighting_type = 2 ;
effective_channels = fancy_alpha_effective_cnts [ input_pixel_layout - STBIRI_RGBA ] ;
}
}
else if ( ( input_pixel_layout > = STBIRI_RGBA_PM ) & & ( input_pixel_layout < = STBIRI_AR_PM ) & & ( output_pixel_layout > = STBIRI_RGBA ) & & ( output_pixel_layout < = STBIRI_AR ) )
{
// input premult, output non-premult
alpha_weighting_type = 3 ;
}
else if ( ( input_pixel_layout > = STBIRI_RGBA ) & & ( input_pixel_layout < = STBIRI_AR ) & & ( output_pixel_layout > = STBIRI_RGBA_PM ) & & ( output_pixel_layout < = STBIRI_AR_PM ) )
{
// input non-premult, output premult
alpha_weighting_type = 1 ;
}
}
// channel in and out count must match currently
if ( channels ! = stbir__pixel_channels [ output_pixel_layout ] )
return 0 ;
// get vertical first
vertical_first = stbir__should_do_vertical_first ( stbir__compute_weights [ ( int ) stbir_channel_count_index [ effective_channels ] ] , horizontal - > filter_pixel_width , horizontal - > scale_info . scale , horizontal - > scale_info . output_sub_size , vertical - > filter_pixel_width , vertical - > scale_info . scale , vertical - > scale_info . output_sub_size , vertical - > is_gather , STBIR__V_FIRST_INFO_POINTER ) ;
// sometimes read one float off in some of the unrolled loops (with a weight of zero coeff, so it doesn't have an effect)
decode_buffer_size = ( conservative - > n1 - conservative - > n0 + 1 ) * effective_channels * sizeof ( float ) + sizeof ( float ) ; // extra float for padding
# if defined( STBIR__SEPARATE_ALLOCATIONS ) && defined(STBIR_SIMD8)
if ( effective_channels = = 3 )
decode_buffer_size + = sizeof ( float ) ; // avx in 3 channel mode needs one float at the start of the buffer (only with separate allocations)
# endif
ring_buffer_length_bytes = horizontal - > scale_info . output_sub_size * effective_channels * sizeof ( float ) + sizeof ( float ) ; // extra float for padding
// if we do vertical first, the ring buffer holds a whole decoded line
if ( vertical_first )
ring_buffer_length_bytes = ( decode_buffer_size + 15 ) & ~ 15 ;
if ( ( ring_buffer_length_bytes & 4095 ) = = 0 ) ring_buffer_length_bytes + = 64 * 3 ; // avoid 4k alias
// One extra entry because floating point precision problems sometimes cause an extra to be necessary.
alloc_ring_buffer_num_entries = vertical - > filter_pixel_width + 1 ;
// we never need more ring buffer entries than the scanlines we're outputting when in scatter mode
if ( ( ! vertical - > is_gather ) & & ( alloc_ring_buffer_num_entries > conservative_split_output_size ) )
alloc_ring_buffer_num_entries = conservative_split_output_size ;
ring_buffer_size = alloc_ring_buffer_num_entries * ring_buffer_length_bytes ;
// The vertical buffer is used differently, depending on whether we are scattering
// the vertical scanlines, or gathering them.
// If scattering, it's used at the temp buffer to accumulate each output.
// If gathering, it's just the output buffer.
vertical_buffer_size = horizontal - > scale_info . output_sub_size * effective_channels * sizeof ( float ) + sizeof ( float ) ; // extra float for padding
// we make two passes through this loop, 1st to add everything up, 2nd to allocate and init
for ( ; ; )
{
int i ;
void * advance_mem = alloced ;
int copy_horizontal = 0 ;
stbir__sampler * possibly_use_horizontal_for_pivot = 0 ;
# ifdef STBIR__SEPARATE_ALLOCATIONS
# define STBIR__NEXT_PTR( ptr, size, ntype ) if ( alloced ) { void * p = STBIR_MALLOC( size, user_data); if ( p == 0 ) { stbir__free_internal_mem( info ); return 0; } (ptr) = (ntype*)p; }
# else
# define STBIR__NEXT_PTR( ptr, size, ntype ) advance_mem = (void*) ( ( ((size_t)advance_mem) + 15 ) & ~15 ); if ( alloced ) ptr = (ntype*)advance_mem; advance_mem = ((char*)advance_mem) + (size);
# endif
STBIR__NEXT_PTR ( info , sizeof ( stbir__info ) , stbir__info ) ;
STBIR__NEXT_PTR ( info - > split_info , sizeof ( stbir__per_split_info ) * splits , stbir__per_split_info ) ;
if ( info )
{
static stbir__alpha_weight_func * fancy_alpha_weights [ 6 ] = { stbir__fancy_alpha_weight_4ch , stbir__fancy_alpha_weight_4ch , stbir__fancy_alpha_weight_4ch , stbir__fancy_alpha_weight_4ch , stbir__fancy_alpha_weight_2ch , stbir__fancy_alpha_weight_2ch } ;
static stbir__alpha_unweight_func * fancy_alpha_unweights [ 6 ] = { stbir__fancy_alpha_unweight_4ch , stbir__fancy_alpha_unweight_4ch , stbir__fancy_alpha_unweight_4ch , stbir__fancy_alpha_unweight_4ch , stbir__fancy_alpha_unweight_2ch , stbir__fancy_alpha_unweight_2ch } ;
static stbir__alpha_weight_func * simple_alpha_weights [ 6 ] = { stbir__simple_alpha_weight_4ch , stbir__simple_alpha_weight_4ch , stbir__simple_alpha_weight_4ch , stbir__simple_alpha_weight_4ch , stbir__simple_alpha_weight_2ch , stbir__simple_alpha_weight_2ch } ;
static stbir__alpha_unweight_func * simple_alpha_unweights [ 6 ] = { stbir__simple_alpha_unweight_4ch , stbir__simple_alpha_unweight_4ch , stbir__simple_alpha_unweight_4ch , stbir__simple_alpha_unweight_4ch , stbir__simple_alpha_unweight_2ch , stbir__simple_alpha_unweight_2ch } ;
// initialize info fields
info - > alloced_mem = alloced ;
info - > alloced_total = alloced_total ;
info - > channels = channels ;
info - > effective_channels = effective_channels ;
info - > offset_x = new_x ;
info - > offset_y = new_y ;
info - > alloc_ring_buffer_num_entries = alloc_ring_buffer_num_entries ;
info - > ring_buffer_num_entries = 0 ;
info - > ring_buffer_length_bytes = ring_buffer_length_bytes ;
info - > splits = splits ;
info - > vertical_first = vertical_first ;
info - > input_pixel_layout_internal = input_pixel_layout ;
info - > output_pixel_layout_internal = output_pixel_layout ;
// setup alpha weight functions
info - > alpha_weight = 0 ;
info - > alpha_unweight = 0 ;
// handle alpha weighting functions and overrides
if ( alpha_weighting_type = = 2 )
{
// high quality alpha multiplying on the way in, dividing on the way out
info - > alpha_weight = fancy_alpha_weights [ input_pixel_layout - STBIRI_RGBA ] ;
info - > alpha_unweight = fancy_alpha_unweights [ output_pixel_layout - STBIRI_RGBA ] ;
}
else if ( alpha_weighting_type = = 4 )
{
// fast alpha multiplying on the way in, dividing on the way out
info - > alpha_weight = simple_alpha_weights [ input_pixel_layout - STBIRI_RGBA ] ;
info - > alpha_unweight = simple_alpha_unweights [ output_pixel_layout - STBIRI_RGBA ] ;
}
else if ( alpha_weighting_type = = 1 )
{
// fast alpha on the way in, leave in premultiplied form on way out
info - > alpha_weight = simple_alpha_weights [ input_pixel_layout - STBIRI_RGBA ] ;
}
else if ( alpha_weighting_type = = 3 )
{
// incoming is premultiplied, fast alpha dividing on the way out - non-premultiplied output
info - > alpha_unweight = simple_alpha_unweights [ output_pixel_layout - STBIRI_RGBA ] ;
}
// handle 3-chan color flipping, using the alpha weight path
if ( ( ( input_pixel_layout = = STBIRI_RGB ) & & ( output_pixel_layout = = STBIRI_BGR ) ) | |
( ( input_pixel_layout = = STBIRI_BGR ) & & ( output_pixel_layout = = STBIRI_RGB ) ) )
{
// do the flipping on the smaller of the two ends
if ( horizontal - > scale_info . scale < 1.0f )
info - > alpha_unweight = stbir__simple_flip_3ch ;
else
info - > alpha_weight = stbir__simple_flip_3ch ;
}
}
// get all the per-split buffers
for ( i = 0 ; i < splits ; i + + )
{
STBIR__NEXT_PTR ( info - > split_info [ i ] . decode_buffer , decode_buffer_size , float ) ;
# ifdef STBIR__SEPARATE_ALLOCATIONS
# ifdef STBIR_SIMD8
if ( ( info ) & & ( effective_channels = = 3 ) )
+ + info - > split_info [ i ] . decode_buffer ; // avx in 3 channel mode needs one float at the start of the buffer
# endif
STBIR__NEXT_PTR ( info - > split_info [ i ] . ring_buffers , alloc_ring_buffer_num_entries * sizeof ( float * ) , float * ) ;
{
int j ;
for ( j = 0 ; j < alloc_ring_buffer_num_entries ; j + + )
{
STBIR__NEXT_PTR ( info - > split_info [ i ] . ring_buffers [ j ] , ring_buffer_length_bytes , float ) ;
# ifdef STBIR_SIMD8
if ( ( info ) & & ( effective_channels = = 3 ) )
+ + info - > split_info [ i ] . ring_buffers [ j ] ; // avx in 3 channel mode needs one float at the start of the buffer
# endif
}
}
# else
STBIR__NEXT_PTR ( info - > split_info [ i ] . ring_buffer , ring_buffer_size , float ) ;
# endif
STBIR__NEXT_PTR ( info - > split_info [ i ] . vertical_buffer , vertical_buffer_size , float ) ;
}
// alloc memory for to-be-pivoted coeffs (if necessary)
if ( vertical - > is_gather = = 0 )
{
int both ;
int temp_mem_amt ;
// when in vertical scatter mode, we first build the coefficients in gather mode, and then pivot after,
// that means we need two buffers, so we try to use the decode buffer and ring buffer for this. if that
// is too small, we just allocate extra memory to use as this temp.
both = vertical - > gather_prescatter_contributors_size + vertical - > gather_prescatter_coefficients_size ;
# ifdef STBIR__SEPARATE_ALLOCATIONS
temp_mem_amt = decode_buffer_size ;
# else
temp_mem_amt = ( decode_buffer_size + ring_buffer_size + vertical_buffer_size ) * splits ;
# endif
if ( temp_mem_amt > = both )
{
if ( info )
{
vertical - > gather_prescatter_contributors = ( stbir__contributors * ) info - > split_info [ 0 ] . decode_buffer ;
vertical - > gather_prescatter_coefficients = ( float * ) ( ( ( char * ) info - > split_info [ 0 ] . decode_buffer ) + vertical - > gather_prescatter_contributors_size ) ;
}
}
else
{
// ring+decode memory is too small, so allocate temp memory
STBIR__NEXT_PTR ( vertical - > gather_prescatter_contributors , vertical - > gather_prescatter_contributors_size , stbir__contributors ) ;
STBIR__NEXT_PTR ( vertical - > gather_prescatter_coefficients , vertical - > gather_prescatter_coefficients_size , float ) ;
}
}
STBIR__NEXT_PTR ( horizontal - > contributors , horizontal - > contributors_size , stbir__contributors ) ;
STBIR__NEXT_PTR ( horizontal - > coefficients , horizontal - > coefficients_size , float ) ;
// are the two filters identical?? (happens a lot with mipmap generation)
if ( ( horizontal - > filter_kernel = = vertical - > filter_kernel ) & & ( horizontal - > filter_support = = vertical - > filter_support ) & & ( horizontal - > edge = = vertical - > edge ) & & ( horizontal - > scale_info . output_sub_size = = vertical - > scale_info . output_sub_size ) )
{
float diff_scale = horizontal - > scale_info . scale - vertical - > scale_info . scale ;
float diff_shift = horizontal - > scale_info . pixel_shift - vertical - > scale_info . pixel_shift ;
if ( diff_scale < 0.0f ) diff_scale = - diff_scale ;
if ( diff_shift < 0.0f ) diff_shift = - diff_shift ;
if ( ( diff_scale < = stbir__small_float ) & & ( diff_shift < = stbir__small_float ) )
{
if ( horizontal - > is_gather = = vertical - > is_gather )
{
copy_horizontal = 1 ;
goto no_vert_alloc ;
}
// everything matches, but vertical is scatter, horizontal is gather, use horizontal coeffs for vertical pivot coeffs
possibly_use_horizontal_for_pivot = horizontal ;
}
}
STBIR__NEXT_PTR ( vertical - > contributors , vertical - > contributors_size , stbir__contributors ) ;
STBIR__NEXT_PTR ( vertical - > coefficients , vertical - > coefficients_size , float ) ;
no_vert_alloc :
if ( info )
{
STBIR_PROFILE_BUILD_START ( horizontal ) ;
stbir__calculate_filters ( horizontal , 0 , user_data STBIR_ONLY_PROFILE_BUILD_SET_INFO ) ;
// setup the horizontal gather functions
// start with defaulting to the n_coeffs functions (specialized on channels and remnant leftover)
info - > horizontal_gather_channels = stbir__horizontal_gather_n_coeffs_funcs [ effective_channels ] [ horizontal - > extent_info . widest & 3 ] ;
// but if the number of coeffs <= 12, use another set of special cases. <=12 coeffs is any enlarging resize, or shrinking resize down to about 1/3 size
if ( horizontal - > extent_info . widest < = 12 )
info - > horizontal_gather_channels = stbir__horizontal_gather_channels_funcs [ effective_channels ] [ horizontal - > extent_info . widest - 1 ] ;
info - > scanline_extents . conservative . n0 = conservative - > n0 ;
info - > scanline_extents . conservative . n1 = conservative - > n1 ;
// get exact extents
stbir__get_extents ( horizontal , & info - > scanline_extents ) ;
// pack the horizontal coeffs
horizontal - > coefficient_width = stbir__pack_coefficients ( horizontal - > num_contributors , horizontal - > contributors , horizontal - > coefficients , horizontal - > coefficient_width , horizontal - > extent_info . widest , info - > scanline_extents . conservative . n1 + 1 ) ;
STBIR_MEMCPY ( & info - > horizontal , horizontal , sizeof ( stbir__sampler ) ) ;
STBIR_PROFILE_BUILD_END ( horizontal ) ;
if ( copy_horizontal )
{
STBIR_MEMCPY ( & info - > vertical , horizontal , sizeof ( stbir__sampler ) ) ;
}
else
{
STBIR_PROFILE_BUILD_START ( vertical ) ;
stbir__calculate_filters ( vertical , possibly_use_horizontal_for_pivot , user_data STBIR_ONLY_PROFILE_BUILD_SET_INFO ) ;
STBIR_MEMCPY ( & info - > vertical , vertical , sizeof ( stbir__sampler ) ) ;
STBIR_PROFILE_BUILD_END ( vertical ) ;
}
// setup the vertical split ranges
stbir__get_split_info ( info - > split_info , info - > splits , info - > vertical . scale_info . output_sub_size , info - > vertical . filter_pixel_margin , info - > vertical . scale_info . input_full_size ) ;
// now we know precisely how many entries we need
info - > ring_buffer_num_entries = info - > vertical . extent_info . widest ;
// we never need more ring buffer entries than the scanlines we're outputting
if ( ( ! info - > vertical . is_gather ) & & ( info - > ring_buffer_num_entries > conservative_split_output_size ) )
info - > ring_buffer_num_entries = conservative_split_output_size ;
STBIR_ASSERT ( info - > ring_buffer_num_entries < = info - > alloc_ring_buffer_num_entries ) ;
// a few of the horizontal gather functions read one dword past the end (but mask it out), so put in a normal value so no snans or denormals accidentally sneak in
for ( i = 0 ; i < splits ; i + + )
{
int width , ofs ;
// find the right most span
if ( info - > scanline_extents . spans [ 0 ] . n1 > info - > scanline_extents . spans [ 1 ] . n1 )
width = info - > scanline_extents . spans [ 0 ] . n1 - info - > scanline_extents . spans [ 0 ] . n0 ;
else
width = info - > scanline_extents . spans [ 1 ] . n1 - info - > scanline_extents . spans [ 1 ] . n0 ;
// this calc finds the exact end of the decoded scanline for all filter modes.
// usually this is just the width * effective channels. But we have to account
// for the area to the left of the scanline for wrap filtering and alignment, this
// is stored as a negative value in info->scanline_extents.conservative.n0. Next,
// we need to skip the exact size of the right hand size filter area (again for
// wrap mode), this is in info->scanline_extents.edge_sizes[1]).
ofs = ( width + 1 - info - > scanline_extents . conservative . n0 + info - > scanline_extents . edge_sizes [ 1 ] ) * effective_channels ;
// place a known, but numerically valid value in the decode buffer
info - > split_info [ i ] . decode_buffer [ ofs ] = 9999.0f ;
// if vertical filtering first, place a known, but numerically valid value in the all
// of the ring buffer accumulators
if ( vertical_first )
{
int j ;
for ( j = 0 ; j < info - > ring_buffer_num_entries ; j + + )
{
stbir__get_ring_buffer_entry ( info , info - > split_info + i , j ) [ ofs ] = 9999.0f ;
}
}
}
}
# undef STBIR__NEXT_PTR
// is this the first time through loop?
if ( info = = 0 )
{
alloced_total = ( int ) ( 15 + ( size_t ) advance_mem ) ;
alloced = STBIR_MALLOC ( alloced_total , user_data ) ;
if ( alloced = = 0 )
return 0 ;
}
else
return info ; // success
}
}
static int stbir__perform_resize ( stbir__info const * info , int split_start , int split_count )
{
stbir__per_split_info * split_info = info - > split_info + split_start ;
STBIR_PROFILE_CLEAR_EXTRAS ( ) ;
STBIR_PROFILE_FIRST_START ( looping ) ;
if ( info - > vertical . is_gather )
stbir__vertical_gather_loop ( info , split_info , split_count ) ;
else
stbir__vertical_scatter_loop ( info , split_info , split_count ) ;
STBIR_PROFILE_END ( looping ) ;
return 1 ;
}
static void stbir__update_info_from_resize ( stbir__info * info , STBIR_RESIZE * resize )
{
static stbir__decode_pixels_func * decode_simple [ STBIR_TYPE_HALF_FLOAT - STBIR_TYPE_UINT8_SRGB + 1 ] =
{
/* 1ch-4ch */ stbir__decode_uint8_srgb , stbir__decode_uint8_srgb , 0 , stbir__decode_float_linear , stbir__decode_half_float_linear ,
} ;
static stbir__decode_pixels_func * decode_alphas [ STBIRI_AR - STBIRI_RGBA + 1 ] [ STBIR_TYPE_HALF_FLOAT - STBIR_TYPE_UINT8_SRGB + 1 ] =
{
{ /* RGBA */ stbir__decode_uint8_srgb4_linearalpha , stbir__decode_uint8_srgb , 0 , stbir__decode_float_linear , stbir__decode_half_float_linear } ,
{ /* BGRA */ stbir__decode_uint8_srgb4_linearalpha_BGRA , stbir__decode_uint8_srgb_BGRA , 0 , stbir__decode_float_linear_BGRA , stbir__decode_half_float_linear_BGRA } ,
{ /* ARGB */ stbir__decode_uint8_srgb4_linearalpha_ARGB , stbir__decode_uint8_srgb_ARGB , 0 , stbir__decode_float_linear_ARGB , stbir__decode_half_float_linear_ARGB } ,
{ /* ABGR */ stbir__decode_uint8_srgb4_linearalpha_ABGR , stbir__decode_uint8_srgb_ABGR , 0 , stbir__decode_float_linear_ABGR , stbir__decode_half_float_linear_ABGR } ,
{ /* RA */ stbir__decode_uint8_srgb2_linearalpha , stbir__decode_uint8_srgb , 0 , stbir__decode_float_linear , stbir__decode_half_float_linear } ,
{ /* AR */ stbir__decode_uint8_srgb2_linearalpha_AR , stbir__decode_uint8_srgb_AR , 0 , stbir__decode_float_linear_AR , stbir__decode_half_float_linear_AR } ,
} ;
static stbir__decode_pixels_func * decode_simple_scaled_or_not [ 2 ] [ 2 ] =
{
{ stbir__decode_uint8_linear_scaled , stbir__decode_uint8_linear } , { stbir__decode_uint16_linear_scaled , stbir__decode_uint16_linear } ,
} ;
static stbir__decode_pixels_func * decode_alphas_scaled_or_not [ STBIRI_AR - STBIRI_RGBA + 1 ] [ 2 ] [ 2 ] =
{
{ /* RGBA */ { stbir__decode_uint8_linear_scaled , stbir__decode_uint8_linear } , { stbir__decode_uint16_linear_scaled , stbir__decode_uint16_linear } } ,
{ /* BGRA */ { stbir__decode_uint8_linear_scaled_BGRA , stbir__decode_uint8_linear_BGRA } , { stbir__decode_uint16_linear_scaled_BGRA , stbir__decode_uint16_linear_BGRA } } ,
{ /* ARGB */ { stbir__decode_uint8_linear_scaled_ARGB , stbir__decode_uint8_linear_ARGB } , { stbir__decode_uint16_linear_scaled_ARGB , stbir__decode_uint16_linear_ARGB } } ,
{ /* ABGR */ { stbir__decode_uint8_linear_scaled_ABGR , stbir__decode_uint8_linear_ABGR } , { stbir__decode_uint16_linear_scaled_ABGR , stbir__decode_uint16_linear_ABGR } } ,
{ /* RA */ { stbir__decode_uint8_linear_scaled , stbir__decode_uint8_linear } , { stbir__decode_uint16_linear_scaled , stbir__decode_uint16_linear } } ,
{ /* AR */ { stbir__decode_uint8_linear_scaled_AR , stbir__decode_uint8_linear_AR } , { stbir__decode_uint16_linear_scaled_AR , stbir__decode_uint16_linear_AR } }
} ;
static stbir__encode_pixels_func * encode_simple [ STBIR_TYPE_HALF_FLOAT - STBIR_TYPE_UINT8_SRGB + 1 ] =
{
/* 1ch-4ch */ stbir__encode_uint8_srgb , stbir__encode_uint8_srgb , 0 , stbir__encode_float_linear , stbir__encode_half_float_linear ,
} ;
static stbir__encode_pixels_func * encode_alphas [ STBIRI_AR - STBIRI_RGBA + 1 ] [ STBIR_TYPE_HALF_FLOAT - STBIR_TYPE_UINT8_SRGB + 1 ] =
{
{ /* RGBA */ stbir__encode_uint8_srgb4_linearalpha , stbir__encode_uint8_srgb , 0 , stbir__encode_float_linear , stbir__encode_half_float_linear } ,
{ /* BGRA */ stbir__encode_uint8_srgb4_linearalpha_BGRA , stbir__encode_uint8_srgb_BGRA , 0 , stbir__encode_float_linear_BGRA , stbir__encode_half_float_linear_BGRA } ,
{ /* ARGB */ stbir__encode_uint8_srgb4_linearalpha_ARGB , stbir__encode_uint8_srgb_ARGB , 0 , stbir__encode_float_linear_ARGB , stbir__encode_half_float_linear_ARGB } ,
{ /* ABGR */ stbir__encode_uint8_srgb4_linearalpha_ABGR , stbir__encode_uint8_srgb_ABGR , 0 , stbir__encode_float_linear_ABGR , stbir__encode_half_float_linear_ABGR } ,
{ /* RA */ stbir__encode_uint8_srgb2_linearalpha , stbir__encode_uint8_srgb , 0 , stbir__encode_float_linear , stbir__encode_half_float_linear } ,
{ /* AR */ stbir__encode_uint8_srgb2_linearalpha_AR , stbir__encode_uint8_srgb_AR , 0 , stbir__encode_float_linear_AR , stbir__encode_half_float_linear_AR }
} ;
static stbir__encode_pixels_func * encode_simple_scaled_or_not [ 2 ] [ 2 ] =
{
{ stbir__encode_uint8_linear_scaled , stbir__encode_uint8_linear } , { stbir__encode_uint16_linear_scaled , stbir__encode_uint16_linear } ,
} ;
static stbir__encode_pixels_func * encode_alphas_scaled_or_not [ STBIRI_AR - STBIRI_RGBA + 1 ] [ 2 ] [ 2 ] =
{
{ /* RGBA */ { stbir__encode_uint8_linear_scaled , stbir__encode_uint8_linear } , { stbir__encode_uint16_linear_scaled , stbir__encode_uint16_linear } } ,
{ /* BGRA */ { stbir__encode_uint8_linear_scaled_BGRA , stbir__encode_uint8_linear_BGRA } , { stbir__encode_uint16_linear_scaled_BGRA , stbir__encode_uint16_linear_BGRA } } ,
{ /* ARGB */ { stbir__encode_uint8_linear_scaled_ARGB , stbir__encode_uint8_linear_ARGB } , { stbir__encode_uint16_linear_scaled_ARGB , stbir__encode_uint16_linear_ARGB } } ,
{ /* ABGR */ { stbir__encode_uint8_linear_scaled_ABGR , stbir__encode_uint8_linear_ABGR } , { stbir__encode_uint16_linear_scaled_ABGR , stbir__encode_uint16_linear_ABGR } } ,
{ /* RA */ { stbir__encode_uint8_linear_scaled , stbir__encode_uint8_linear } , { stbir__encode_uint16_linear_scaled , stbir__encode_uint16_linear } } ,
{ /* AR */ { stbir__encode_uint8_linear_scaled_AR , stbir__encode_uint8_linear_AR } , { stbir__encode_uint16_linear_scaled_AR , stbir__encode_uint16_linear_AR } }
} ;
stbir__decode_pixels_func * decode_pixels = 0 ;
stbir__encode_pixels_func * encode_pixels = 0 ;
stbir_datatype input_type , output_type ;
input_type = resize - > input_data_type ;
output_type = resize - > output_data_type ;
info - > input_data = resize - > input_pixels ;
info - > input_stride_bytes = resize - > input_stride_in_bytes ;
info - > output_stride_bytes = resize - > output_stride_in_bytes ;
// if we're completely point sampling, then we can turn off SRGB
if ( ( info - > horizontal . filter_enum = = STBIR_FILTER_POINT_SAMPLE ) & & ( info - > vertical . filter_enum = = STBIR_FILTER_POINT_SAMPLE ) )
{
if ( ( ( input_type = = STBIR_TYPE_UINT8_SRGB ) | | ( input_type = = STBIR_TYPE_UINT8_SRGB_ALPHA ) ) & &
( ( output_type = = STBIR_TYPE_UINT8_SRGB ) | | ( output_type = = STBIR_TYPE_UINT8_SRGB_ALPHA ) ) )
{
input_type = STBIR_TYPE_UINT8 ;
output_type = STBIR_TYPE_UINT8 ;
}
}
// recalc the output and input strides
if ( info - > input_stride_bytes = = 0 )
info - > input_stride_bytes = info - > channels * info - > horizontal . scale_info . input_full_size * stbir__type_size [ input_type ] ;
if ( info - > output_stride_bytes = = 0 )
info - > output_stride_bytes = info - > channels * info - > horizontal . scale_info . output_sub_size * stbir__type_size [ output_type ] ;
// calc offset
info - > output_data = ( ( char * ) resize - > output_pixels ) + ( ( ptrdiff_t ) info - > offset_y * ( ptrdiff_t ) resize - > output_stride_in_bytes ) + ( info - > offset_x * info - > channels * stbir__type_size [ output_type ] ) ;
info - > in_pixels_cb = resize - > input_cb ;
info - > user_data = resize - > user_data ;
info - > out_pixels_cb = resize - > output_cb ;
// setup the input format converters
if ( ( input_type = = STBIR_TYPE_UINT8 ) | | ( input_type = = STBIR_TYPE_UINT16 ) )
{
int non_scaled = 0 ;
// check if we can run unscaled - 0-255.0/0-65535.0 instead of 0-1.0 (which is a tiny bit faster when doing linear 8->8 or 16->16)
if ( ( ! info - > alpha_weight ) & & ( ! info - > alpha_unweight ) ) // don't short circuit when alpha weighting (get everything to 0-1.0 as usual)
if ( ( ( input_type = = STBIR_TYPE_UINT8 ) & & ( output_type = = STBIR_TYPE_UINT8 ) ) | | ( ( input_type = = STBIR_TYPE_UINT16 ) & & ( output_type = = STBIR_TYPE_UINT16 ) ) )
non_scaled = 1 ;
if ( info - > input_pixel_layout_internal < = STBIRI_4CHANNEL )
decode_pixels = decode_simple_scaled_or_not [ input_type = = STBIR_TYPE_UINT16 ] [ non_scaled ] ;
else
decode_pixels = decode_alphas_scaled_or_not [ ( info - > input_pixel_layout_internal - STBIRI_RGBA ) % ( STBIRI_AR - STBIRI_RGBA + 1 ) ] [ input_type = = STBIR_TYPE_UINT16 ] [ non_scaled ] ;
}
else
{
if ( info - > input_pixel_layout_internal < = STBIRI_4CHANNEL )
decode_pixels = decode_simple [ input_type - STBIR_TYPE_UINT8_SRGB ] ;
else
decode_pixels = decode_alphas [ ( info - > input_pixel_layout_internal - STBIRI_RGBA ) % ( STBIRI_AR - STBIRI_RGBA + 1 ) ] [ input_type - STBIR_TYPE_UINT8_SRGB ] ;
}
// setup the output format converters
if ( ( output_type = = STBIR_TYPE_UINT8 ) | | ( output_type = = STBIR_TYPE_UINT16 ) )
{
int non_scaled = 0 ;
// check if we can run unscaled - 0-255.0/0-65535.0 instead of 0-1.0 (which is a tiny bit faster when doing linear 8->8 or 16->16)
if ( ( ! info - > alpha_weight ) & & ( ! info - > alpha_unweight ) ) // don't short circuit when alpha weighting (get everything to 0-1.0 as usual)
if ( ( ( input_type = = STBIR_TYPE_UINT8 ) & & ( output_type = = STBIR_TYPE_UINT8 ) ) | | ( ( input_type = = STBIR_TYPE_UINT16 ) & & ( output_type = = STBIR_TYPE_UINT16 ) ) )
non_scaled = 1 ;
if ( info - > output_pixel_layout_internal < = STBIRI_4CHANNEL )
encode_pixels = encode_simple_scaled_or_not [ output_type = = STBIR_TYPE_UINT16 ] [ non_scaled ] ;
else
encode_pixels = encode_alphas_scaled_or_not [ ( info - > output_pixel_layout_internal - STBIRI_RGBA ) % ( STBIRI_AR - STBIRI_RGBA + 1 ) ] [ output_type = = STBIR_TYPE_UINT16 ] [ non_scaled ] ;
}
else
{
if ( info - > output_pixel_layout_internal < = STBIRI_4CHANNEL )
encode_pixels = encode_simple [ output_type - STBIR_TYPE_UINT8_SRGB ] ;
else
encode_pixels = encode_alphas [ ( info - > output_pixel_layout_internal - STBIRI_RGBA ) % ( STBIRI_AR - STBIRI_RGBA + 1 ) ] [ output_type - STBIR_TYPE_UINT8_SRGB ] ;
}
info - > input_type = input_type ;
info - > output_type = output_type ;
info - > decode_pixels = decode_pixels ;
info - > encode_pixels = encode_pixels ;
}
static void stbir__clip ( int * outx , int * outsubw , int outw , double * u0 , double * u1 )
{
double per , adj ;
int over ;
// do left/top edge
if ( * outx < 0 )
{
per = ( ( double ) * outx ) / ( ( double ) * outsubw ) ; // is negative
adj = per * ( * u1 - * u0 ) ;
* u0 - = adj ; // increases u0
* outx = 0 ;
}
// do right/bot edge
over = outw - ( * outx + * outsubw ) ;
if ( over < 0 )
{
per = ( ( double ) over ) / ( ( double ) * outsubw ) ; // is negative
adj = per * ( * u1 - * u0 ) ;
* u1 + = adj ; // decrease u1
* outsubw = outw - * outx ;
}
}
// converts a double to a rational that has less than one float bit of error (returns 0 if unable to do so)
static int stbir__double_to_rational ( double f , stbir_uint32 limit , stbir_uint32 * numer , stbir_uint32 * denom , int limit_denom ) // limit_denom (1) or limit numer (0)
{
double err ;
stbir_uint64 top , bot ;
stbir_uint64 numer_last = 0 ;
stbir_uint64 denom_last = 1 ;
stbir_uint64 numer_estimate = 1 ;
stbir_uint64 denom_estimate = 0 ;
// scale to past float error range
top = ( stbir_uint64 ) ( f * ( double ) ( 1 < < 25 ) ) ;
bot = 1 < < 25 ;
// keep refining, but usually stops in a few loops - usually 5 for bad cases
for ( ; ; )
{
stbir_uint64 est , temp ;
// hit limit, break out and do best full range estimate
if ( ( ( limit_denom ) ? denom_estimate : numer_estimate ) > = limit )
break ;
// is the current error less than 1 bit of a float? if so, we're done
if ( denom_estimate )
{
err = ( ( double ) numer_estimate / ( double ) denom_estimate ) - f ;
if ( err < 0.0 ) err = - err ;
if ( err < ( 1.0 / ( double ) ( 1 < < 24 ) ) )
{
// yup, found it
* numer = ( stbir_uint32 ) numer_estimate ;
* denom = ( stbir_uint32 ) denom_estimate ;
return 1 ;
}
}
// no more refinement bits left? break out and do full range estimate
if ( bot = = 0 )
break ;
// gcd the estimate bits
est = top / bot ;
temp = top % bot ;
top = bot ;
bot = temp ;
// move remainders
temp = est * denom_estimate + denom_last ;
denom_last = denom_estimate ;
denom_estimate = temp ;
// move remainders
temp = est * numer_estimate + numer_last ;
numer_last = numer_estimate ;
numer_estimate = temp ;
}
// we didn't fine anything good enough for float, use a full range estimate
if ( limit_denom )
{
numer_estimate = ( stbir_uint64 ) ( f * ( double ) limit + 0.5 ) ;
denom_estimate = limit ;
}
else
{
numer_estimate = limit ;
denom_estimate = ( stbir_uint64 ) ( ( ( double ) limit / f ) + 0.5 ) ;
}
* numer = ( stbir_uint32 ) numer_estimate ;
* denom = ( stbir_uint32 ) denom_estimate ;
err = ( denom_estimate ) ? ( ( ( double ) ( stbir_uint32 ) numer_estimate / ( double ) ( stbir_uint32 ) denom_estimate ) - f ) : 1.0 ;
if ( err < 0.0 ) err = - err ;
return ( err < ( 1.0 / ( double ) ( 1 < < 24 ) ) ) ? 1 : 0 ;
}
static int stbir__calculate_region_transform ( stbir__scale_info * scale_info , int output_full_range , int * output_offset , int output_sub_range , int input_full_range , double input_s0 , double input_s1 )
{
double output_range , input_range , output_s , input_s , ratio , scale ;
input_s = input_s1 - input_s0 ;
// null area
if ( ( output_full_range = = 0 ) | | ( input_full_range = = 0 ) | |
( output_sub_range = = 0 ) | | ( input_s < = stbir__small_float ) )
return 0 ;
// are either of the ranges completely out of bounds?
if ( ( * output_offset > = output_full_range ) | | ( ( * output_offset + output_sub_range ) < = 0 ) | | ( input_s0 > = ( 1.0f - stbir__small_float ) ) | | ( input_s1 < = stbir__small_float ) )
return 0 ;
output_range = ( double ) output_full_range ;
input_range = ( double ) input_full_range ;
output_s = ( ( double ) output_sub_range ) / output_range ;
// figure out the scaling to use
ratio = output_s / input_s ;
// save scale before clipping
scale = ( output_range / input_range ) * ratio ;
scale_info - > scale = ( float ) scale ;
scale_info - > inv_scale = ( float ) ( 1.0 / scale ) ;
// clip output area to left/right output edges (and adjust input area)
stbir__clip ( output_offset , & output_sub_range , output_full_range , & input_s0 , & input_s1 ) ;
// recalc input area
input_s = input_s1 - input_s0 ;
// after clipping do we have zero input area?
if ( input_s < = stbir__small_float )
return 0 ;
// calculate and store the starting source offsets in output pixel space
scale_info - > pixel_shift = ( float ) ( input_s0 * ratio * output_range ) ;
scale_info - > scale_is_rational = stbir__double_to_rational ( scale , ( scale < = 1.0 ) ? output_full_range : input_full_range , & scale_info - > scale_numerator , & scale_info - > scale_denominator , ( scale > = 1.0 ) ) ;
scale_info - > input_full_size = input_full_range ;
scale_info - > output_sub_size = output_sub_range ;
return 1 ;
}
static void stbir__init_and_set_layout ( STBIR_RESIZE * resize , stbir_pixel_layout pixel_layout , stbir_datatype data_type )
{
resize - > input_cb = 0 ;
resize - > output_cb = 0 ;
resize - > user_data = resize ;
resize - > samplers = 0 ;
resize - > called_alloc = 0 ;
resize - > horizontal_filter = STBIR_FILTER_DEFAULT ;
resize - > horizontal_filter_kernel = 0 ; resize - > horizontal_filter_support = 0 ;
resize - > vertical_filter = STBIR_FILTER_DEFAULT ;
resize - > vertical_filter_kernel = 0 ; resize - > vertical_filter_support = 0 ;
resize - > horizontal_edge = STBIR_EDGE_CLAMP ;
resize - > vertical_edge = STBIR_EDGE_CLAMP ;
resize - > input_s0 = 0 ; resize - > input_t0 = 0 ; resize - > input_s1 = 1 ; resize - > input_t1 = 1 ;
resize - > output_subx = 0 ; resize - > output_suby = 0 ; resize - > output_subw = resize - > output_w ; resize - > output_subh = resize - > output_h ;
resize - > input_data_type = data_type ;
resize - > output_data_type = data_type ;
resize - > input_pixel_layout_public = pixel_layout ;
resize - > output_pixel_layout_public = pixel_layout ;
2023-12-14 19:09:00 +08:00
resize - > needs_rebuild = 1 ;
2023-10-10 08:23:04 +08:00
}
STBIRDEF void stbir_resize_init ( STBIR_RESIZE * resize ,
const void * input_pixels , int input_w , int input_h , int input_stride_in_bytes , // stride can be zero
void * output_pixels , int output_w , int output_h , int output_stride_in_bytes , // stride can be zero
stbir_pixel_layout pixel_layout , stbir_datatype data_type )
{
resize - > input_pixels = input_pixels ;
resize - > input_w = input_w ;
resize - > input_h = input_h ;
resize - > input_stride_in_bytes = input_stride_in_bytes ;
resize - > output_pixels = output_pixels ;
resize - > output_w = output_w ;
resize - > output_h = output_h ;
resize - > output_stride_in_bytes = output_stride_in_bytes ;
resize - > fast_alpha = 0 ;
stbir__init_and_set_layout ( resize , pixel_layout , data_type ) ;
}
// You can update parameters any time after resize_init
STBIRDEF void stbir_set_datatypes ( STBIR_RESIZE * resize , stbir_datatype input_type , stbir_datatype output_type ) // by default, datatype from resize_init
{
resize - > input_data_type = input_type ;
resize - > output_data_type = output_type ;
2023-12-14 19:09:00 +08:00
if ( ( resize - > samplers ) & & ( ! resize - > needs_rebuild ) )
stbir__update_info_from_resize ( resize - > samplers , resize ) ;
2023-10-10 08:23:04 +08:00
}
STBIRDEF void stbir_set_pixel_callbacks ( STBIR_RESIZE * resize , stbir_input_callback * input_cb , stbir_output_callback * output_cb ) // no callbacks by default
{
resize - > input_cb = input_cb ;
resize - > output_cb = output_cb ;
2023-12-14 19:09:00 +08:00
if ( ( resize - > samplers ) & & ( ! resize - > needs_rebuild ) )
{
resize - > samplers - > in_pixels_cb = input_cb ;
resize - > samplers - > out_pixels_cb = output_cb ;
}
2023-10-10 08:23:04 +08:00
}
STBIRDEF void stbir_set_user_data ( STBIR_RESIZE * resize , void * user_data ) // pass back STBIR_RESIZE* by default
{
resize - > user_data = user_data ;
2023-12-14 19:09:00 +08:00
if ( ( resize - > samplers ) & & ( ! resize - > needs_rebuild ) )
resize - > samplers - > user_data = user_data ;
2023-10-10 08:23:04 +08:00
}
STBIRDEF void stbir_set_buffer_ptrs ( STBIR_RESIZE * resize , const void * input_pixels , int input_stride_in_bytes , void * output_pixels , int output_stride_in_bytes )
{
resize - > input_pixels = input_pixels ;
resize - > input_stride_in_bytes = input_stride_in_bytes ;
resize - > output_pixels = output_pixels ;
resize - > output_stride_in_bytes = output_stride_in_bytes ;
2023-12-14 19:09:00 +08:00
if ( ( resize - > samplers ) & & ( ! resize - > needs_rebuild ) )
stbir__update_info_from_resize ( resize - > samplers , resize ) ;
2023-10-10 08:23:04 +08:00
}
STBIRDEF int stbir_set_edgemodes ( STBIR_RESIZE * resize , stbir_edge horizontal_edge , stbir_edge vertical_edge ) // CLAMP by default
{
resize - > horizontal_edge = horizontal_edge ;
resize - > vertical_edge = vertical_edge ;
resize - > needs_rebuild = 1 ;
return 1 ;
}
STBIRDEF int stbir_set_filters ( STBIR_RESIZE * resize , stbir_filter horizontal_filter , stbir_filter vertical_filter ) // STBIR_DEFAULT_FILTER_UPSAMPLE/DOWNSAMPLE by default
{
resize - > horizontal_filter = horizontal_filter ;
resize - > vertical_filter = vertical_filter ;
resize - > needs_rebuild = 1 ;
return 1 ;
}
STBIRDEF int stbir_set_filter_callbacks ( STBIR_RESIZE * resize , stbir__kernel_callback * horizontal_filter , stbir__support_callback * horizontal_support , stbir__kernel_callback * vertical_filter , stbir__support_callback * vertical_support )
{
resize - > horizontal_filter_kernel = horizontal_filter ; resize - > horizontal_filter_support = horizontal_support ;
resize - > vertical_filter_kernel = vertical_filter ; resize - > vertical_filter_support = vertical_support ;
resize - > needs_rebuild = 1 ;
return 1 ;
}
STBIRDEF int stbir_set_pixel_layouts ( STBIR_RESIZE * resize , stbir_pixel_layout input_pixel_layout , stbir_pixel_layout output_pixel_layout ) // sets new pixel layouts
{
resize - > input_pixel_layout_public = input_pixel_layout ;
resize - > output_pixel_layout_public = output_pixel_layout ;
resize - > needs_rebuild = 1 ;
return 1 ;
}
STBIRDEF int stbir_set_non_pm_alpha_speed_over_quality ( STBIR_RESIZE * resize , int non_pma_alpha_speed_over_quality ) // sets alpha speed
{
resize - > fast_alpha = non_pma_alpha_speed_over_quality ;
resize - > needs_rebuild = 1 ;
return 1 ;
}
STBIRDEF int stbir_set_input_subrect ( STBIR_RESIZE * resize , double s0 , double t0 , double s1 , double t1 ) // sets input region (full region by default)
{
resize - > input_s0 = s0 ;
resize - > input_t0 = t0 ;
resize - > input_s1 = s1 ;
resize - > input_t1 = t1 ;
resize - > needs_rebuild = 1 ;
// are we inbounds?
if ( ( s1 < stbir__small_float ) | | ( ( s1 - s0 ) < stbir__small_float ) | |
( t1 < stbir__small_float ) | | ( ( t1 - t0 ) < stbir__small_float ) | |
( s0 > ( 1.0f - stbir__small_float ) ) | |
( t0 > ( 1.0f - stbir__small_float ) ) )
return 0 ;
return 1 ;
}
STBIRDEF int stbir_set_output_pixel_subrect ( STBIR_RESIZE * resize , int subx , int suby , int subw , int subh ) // sets input region (full region by default)
{
resize - > output_subx = subx ;
resize - > output_suby = suby ;
resize - > output_subw = subw ;
resize - > output_subh = subh ;
resize - > needs_rebuild = 1 ;
// are we inbounds?
if ( ( subx > = resize - > output_w ) | | ( ( subx + subw ) < = 0 ) | | ( suby > = resize - > output_h ) | | ( ( suby + subh ) < = 0 ) | | ( subw = = 0 ) | | ( subh = = 0 ) )
return 0 ;
return 1 ;
}
STBIRDEF int stbir_set_pixel_subrect ( STBIR_RESIZE * resize , int subx , int suby , int subw , int subh ) // sets both regions (full regions by default)
{
double s0 , t0 , s1 , t1 ;
s0 = ( ( double ) subx ) / ( ( double ) resize - > output_w ) ;
t0 = ( ( double ) suby ) / ( ( double ) resize - > output_h ) ;
s1 = ( ( double ) ( subx + subw ) ) / ( ( double ) resize - > output_w ) ;
t1 = ( ( double ) ( suby + subh ) ) / ( ( double ) resize - > output_h ) ;
resize - > input_s0 = s0 ;
resize - > input_t0 = t0 ;
resize - > input_s1 = s1 ;
resize - > input_t1 = t1 ;
resize - > output_subx = subx ;
resize - > output_suby = suby ;
resize - > output_subw = subw ;
resize - > output_subh = subh ;
resize - > needs_rebuild = 1 ;
// are we inbounds?
if ( ( subx > = resize - > output_w ) | | ( ( subx + subw ) < = 0 ) | | ( suby > = resize - > output_h ) | | ( ( suby + subh ) < = 0 ) | | ( subw = = 0 ) | | ( subh = = 0 ) )
return 0 ;
return 1 ;
}
static int stbir__perform_build ( STBIR_RESIZE * resize , int splits )
{
stbir__contributors conservative = { 0 , 0 } ;
stbir__sampler horizontal , vertical ;
int new_output_subx , new_output_suby ;
stbir__info * out_info ;
# ifdef STBIR_PROFILE
stbir__info profile_infod ; // used to contain building profile info before everything is allocated
stbir__info * profile_info = & profile_infod ;
# endif
// have we already built the samplers?
if ( resize - > samplers )
return 0 ;
# define STBIR_RETURN_ERROR_AND_ASSERT( exp ) STBIR_ASSERT( !(exp) ); if (exp) return 0;
STBIR_RETURN_ERROR_AND_ASSERT ( ( unsigned ) resize - > horizontal_filter > = STBIR_FILTER_OTHER )
STBIR_RETURN_ERROR_AND_ASSERT ( ( unsigned ) resize - > vertical_filter > = STBIR_FILTER_OTHER )
# undef STBIR_RETURN_ERROR_AND_ASSERT
if ( splits < = 0 )
return 0 ;
STBIR_PROFILE_BUILD_FIRST_START ( build ) ;
new_output_subx = resize - > output_subx ;
new_output_suby = resize - > output_suby ;
// do horizontal clip and scale calcs
if ( ! stbir__calculate_region_transform ( & horizontal . scale_info , resize - > output_w , & new_output_subx , resize - > output_subw , resize - > input_w , resize - > input_s0 , resize - > input_s1 ) )
return 0 ;
// do vertical clip and scale calcs
if ( ! stbir__calculate_region_transform ( & vertical . scale_info , resize - > output_h , & new_output_suby , resize - > output_subh , resize - > input_h , resize - > input_t0 , resize - > input_t1 ) )
return 0 ;
// if nothing to do, just return
if ( ( horizontal . scale_info . output_sub_size = = 0 ) | | ( vertical . scale_info . output_sub_size = = 0 ) )
return 0 ;
stbir__set_sampler ( & horizontal , resize - > horizontal_filter , resize - > horizontal_filter_kernel , resize - > horizontal_filter_support , resize - > horizontal_edge , & horizontal . scale_info , 1 , resize - > user_data ) ;
stbir__get_conservative_extents ( & horizontal , & conservative , resize - > user_data ) ;
stbir__set_sampler ( & vertical , resize - > vertical_filter , resize - > horizontal_filter_kernel , resize - > vertical_filter_support , resize - > vertical_edge , & vertical . scale_info , 0 , resize - > user_data ) ;
2023-12-14 19:09:00 +08:00
if ( ( vertical . scale_info . output_sub_size / splits ) < STBIR_FORCE_MINIMUM_SCANLINES_FOR_SPLITS ) // each split should be a minimum of 4 scanlines (handwavey choice)
2023-10-10 08:23:04 +08:00
{
2023-12-14 19:09:00 +08:00
splits = vertical . scale_info . output_sub_size / STBIR_FORCE_MINIMUM_SCANLINES_FOR_SPLITS ;
2023-10-10 08:23:04 +08:00
if ( splits = = 0 ) splits = 1 ;
}
STBIR_PROFILE_BUILD_START ( alloc ) ;
out_info = stbir__alloc_internal_mem_and_build_samplers ( & horizontal , & vertical , & conservative , resize - > input_pixel_layout_public , resize - > output_pixel_layout_public , splits , new_output_subx , new_output_suby , resize - > fast_alpha , resize - > user_data STBIR_ONLY_PROFILE_BUILD_SET_INFO ) ;
STBIR_PROFILE_BUILD_END ( alloc ) ;
STBIR_PROFILE_BUILD_END ( build ) ;
if ( out_info )
{
resize - > splits = splits ;
resize - > samplers = out_info ;
resize - > needs_rebuild = 0 ;
# ifdef STBIR_PROFILE
STBIR_MEMCPY ( & out_info - > profile , & profile_infod . profile , sizeof ( out_info - > profile ) ) ;
# endif
2023-12-14 19:09:00 +08:00
// update anything that can be changed without recalcing samplers
stbir__update_info_from_resize ( out_info , resize ) ;
2023-10-10 08:23:04 +08:00
return splits ;
}
return 0 ;
}
void stbir_free_samplers ( STBIR_RESIZE * resize )
{
if ( resize - > samplers )
{
stbir__free_internal_mem ( resize - > samplers ) ;
resize - > samplers = 0 ;
resize - > called_alloc = 0 ;
}
}
STBIRDEF int stbir_build_samplers_with_splits ( STBIR_RESIZE * resize , int splits )
{
if ( ( resize - > samplers = = 0 ) | | ( resize - > needs_rebuild ) )
{
if ( resize - > samplers )
stbir_free_samplers ( resize ) ;
resize - > called_alloc = 1 ;
return stbir__perform_build ( resize , splits ) ;
}
STBIR_PROFILE_BUILD_CLEAR ( resize - > samplers ) ;
return 1 ;
}
STBIRDEF int stbir_build_samplers ( STBIR_RESIZE * resize )
{
return stbir_build_samplers_with_splits ( resize , 1 ) ;
}
STBIRDEF int stbir_resize_extended ( STBIR_RESIZE * resize )
{
int result ;
if ( ( resize - > samplers = = 0 ) | | ( resize - > needs_rebuild ) )
{
int alloc_state = resize - > called_alloc ; // remember allocated state
if ( resize - > samplers )
{
stbir__free_internal_mem ( resize - > samplers ) ;
resize - > samplers = 0 ;
}
if ( ! stbir_build_samplers ( resize ) )
return 0 ;
resize - > called_alloc = alloc_state ;
// if build_samplers succeeded (above), but there are no samplers set, then
// the area to stretch into was zero pixels, so don't do anything and return
// success
if ( resize - > samplers = = 0 )
return 1 ;
}
else
{
// didn't build anything - clear it
STBIR_PROFILE_BUILD_CLEAR ( resize - > samplers ) ;
}
// do resize
result = stbir__perform_resize ( resize - > samplers , 0 , resize - > splits ) ;
// if we alloced, then free
if ( ! resize - > called_alloc )
{
stbir_free_samplers ( resize ) ;
resize - > samplers = 0 ;
}
return result ;
}
STBIRDEF int stbir_resize_extended_split ( STBIR_RESIZE * resize , int split_start , int split_count )
{
STBIR_ASSERT ( resize - > samplers ) ;
// if we're just doing the whole thing, call full
if ( ( split_start = = - 1 ) | | ( ( split_start = = 0 ) & & ( split_count = = resize - > splits ) ) )
return stbir_resize_extended ( resize ) ;
// you **must** build samplers first when using split resize
if ( ( resize - > samplers = = 0 ) | | ( resize - > needs_rebuild ) )
return 0 ;
if ( ( split_start > = resize - > splits ) | | ( split_start < 0 ) | | ( ( split_start + split_count ) > resize - > splits ) | | ( split_count < = 0 ) )
return 0 ;
// do resize
return stbir__perform_resize ( resize - > samplers , split_start , split_count ) ;
}
static int stbir__check_output_stuff ( void * * ret_ptr , int * ret_pitch , void * output_pixels , int type_size , int output_w , int output_h , int output_stride_in_bytes , stbir_internal_pixel_layout pixel_layout )
{
size_t size ;
int pitch ;
void * ptr ;
pitch = output_w * type_size * stbir__pixel_channels [ pixel_layout ] ;
if ( pitch = = 0 )
return 0 ;
if ( output_stride_in_bytes = = 0 )
output_stride_in_bytes = pitch ;
if ( output_stride_in_bytes < pitch )
return 0 ;
size = output_stride_in_bytes * output_h ;
if ( size = = 0 )
return 0 ;
* ret_ptr = 0 ;
* ret_pitch = output_stride_in_bytes ;
if ( output_pixels = = 0 )
{
ptr = STBIR_MALLOC ( size , 0 ) ;
if ( ptr = = 0 )
return 0 ;
* ret_ptr = ptr ;
* ret_pitch = pitch ;
}
return 1 ;
}
STBIRDEF unsigned char * stbir_resize_uint8_linear ( const unsigned char * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
unsigned char * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_layout )
{
STBIR_RESIZE resize ;
unsigned char * optr ;
int opitch ;
if ( ! stbir__check_output_stuff ( ( void * * ) & optr , & opitch , output_pixels , sizeof ( unsigned char ) , output_w , output_h , output_stride_in_bytes , stbir__pixel_layout_convert_public_to_internal [ pixel_layout ] ) )
return 0 ;
stbir_resize_init ( & resize ,
input_pixels , input_w , input_h , input_stride_in_bytes ,
( optr ) ? optr : output_pixels , output_w , output_h , opitch ,
pixel_layout , STBIR_TYPE_UINT8 ) ;
if ( ! stbir_resize_extended ( & resize ) )
{
if ( optr )
STBIR_FREE ( optr , 0 ) ;
return 0 ;
}
return ( optr ) ? optr : output_pixels ;
}
STBIRDEF unsigned char * stbir_resize_uint8_srgb ( const unsigned char * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
unsigned char * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_layout )
{
STBIR_RESIZE resize ;
unsigned char * optr ;
int opitch ;
if ( ! stbir__check_output_stuff ( ( void * * ) & optr , & opitch , output_pixels , sizeof ( unsigned char ) , output_w , output_h , output_stride_in_bytes , stbir__pixel_layout_convert_public_to_internal [ pixel_layout ] ) )
return 0 ;
stbir_resize_init ( & resize ,
input_pixels , input_w , input_h , input_stride_in_bytes ,
( optr ) ? optr : output_pixels , output_w , output_h , opitch ,
pixel_layout , STBIR_TYPE_UINT8_SRGB ) ;
if ( ! stbir_resize_extended ( & resize ) )
{
if ( optr )
STBIR_FREE ( optr , 0 ) ;
return 0 ;
}
return ( optr ) ? optr : output_pixels ;
}
STBIRDEF float * stbir_resize_float_linear ( const float * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
float * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_layout )
{
STBIR_RESIZE resize ;
float * optr ;
int opitch ;
if ( ! stbir__check_output_stuff ( ( void * * ) & optr , & opitch , output_pixels , sizeof ( float ) , output_w , output_h , output_stride_in_bytes , stbir__pixel_layout_convert_public_to_internal [ pixel_layout ] ) )
return 0 ;
stbir_resize_init ( & resize ,
input_pixels , input_w , input_h , input_stride_in_bytes ,
( optr ) ? optr : output_pixels , output_w , output_h , opitch ,
pixel_layout , STBIR_TYPE_FLOAT ) ;
if ( ! stbir_resize_extended ( & resize ) )
{
if ( optr )
STBIR_FREE ( optr , 0 ) ;
return 0 ;
}
return ( optr ) ? optr : output_pixels ;
}
STBIRDEF void * stbir_resize ( const void * input_pixels , int input_w , int input_h , int input_stride_in_bytes ,
void * output_pixels , int output_w , int output_h , int output_stride_in_bytes ,
stbir_pixel_layout pixel_layout , stbir_datatype data_type ,
stbir_edge edge , stbir_filter filter )
{
STBIR_RESIZE resize ;
float * optr ;
int opitch ;
if ( ! stbir__check_output_stuff ( ( void * * ) & optr , & opitch , output_pixels , stbir__type_size [ data_type ] , output_w , output_h , output_stride_in_bytes , stbir__pixel_layout_convert_public_to_internal [ pixel_layout ] ) )
return 0 ;
stbir_resize_init ( & resize ,
input_pixels , input_w , input_h , input_stride_in_bytes ,
( optr ) ? optr : output_pixels , output_w , output_h , output_stride_in_bytes ,
pixel_layout , data_type ) ;
resize . horizontal_edge = edge ;
resize . vertical_edge = edge ;
resize . horizontal_filter = filter ;
resize . vertical_filter = filter ;
if ( ! stbir_resize_extended ( & resize ) )
{
if ( optr )
STBIR_FREE ( optr , 0 ) ;
return 0 ;
}
return ( optr ) ? optr : output_pixels ;
}
# ifdef STBIR_PROFILE
STBIRDEF void stbir_resize_build_profile_info ( STBIR_PROFILE_INFO * info , STBIR_RESIZE const * resize )
{
static char const * bdescriptions [ 6 ] = { " Building " , " Allocating " , " Horizontal sampler " , " Vertical sampler " , " Coefficient cleanup " , " Coefficient piovot " } ;
stbir__info * samp = resize - > samplers ;
int i ;
typedef int testa [ ( STBIR__ARRAY_SIZE ( bdescriptions ) = = ( STBIR__ARRAY_SIZE ( samp - > profile . array ) - 1 ) ) ? 1 : - 1 ] ;
typedef int testb [ ( sizeof ( samp - > profile . array ) = = ( sizeof ( samp - > profile . named ) ) ) ? 1 : - 1 ] ;
typedef int testc [ ( sizeof ( info - > clocks ) > = ( sizeof ( samp - > profile . named ) ) ) ? 1 : - 1 ] ;
for ( i = 0 ; i < STBIR__ARRAY_SIZE ( bdescriptions ) ; i + + )
info - > clocks [ i ] = samp - > profile . array [ i + 1 ] ;
info - > total_clocks = samp - > profile . named . total ;
info - > descriptions = bdescriptions ;
info - > count = STBIR__ARRAY_SIZE ( bdescriptions ) ;
}
STBIRDEF void stbir_resize_split_profile_info ( STBIR_PROFILE_INFO * info , STBIR_RESIZE const * resize , int split_start , int split_count )
{
static char const * descriptions [ 7 ] = { " Looping " , " Vertical sampling " , " Horizontal sampling " , " Scanline input " , " Scanline output " , " Alpha weighting " , " Alpha unweighting " } ;
stbir__per_split_info * split_info ;
int s , i ;
typedef int testa [ ( STBIR__ARRAY_SIZE ( descriptions ) = = ( STBIR__ARRAY_SIZE ( split_info - > profile . array ) - 1 ) ) ? 1 : - 1 ] ;
typedef int testb [ ( sizeof ( split_info - > profile . array ) = = ( sizeof ( split_info - > profile . named ) ) ) ? 1 : - 1 ] ;
typedef int testc [ ( sizeof ( info - > clocks ) > = ( sizeof ( split_info - > profile . named ) ) ) ? 1 : - 1 ] ;
if ( split_start = = - 1 )
{
split_start = 0 ;
split_count = resize - > samplers - > splits ;
}
if ( ( split_start > = resize - > splits ) | | ( split_start < 0 ) | | ( ( split_start + split_count ) > resize - > splits ) | | ( split_count < = 0 ) )
{
info - > total_clocks = 0 ;
info - > descriptions = 0 ;
info - > count = 0 ;
return ;
}
split_info = resize - > samplers - > split_info + split_start ;
// sum up the profile from all the splits
for ( i = 0 ; i < STBIR__ARRAY_SIZE ( descriptions ) ; i + + )
{
stbir_uint64 sum = 0 ;
for ( s = 0 ; s < split_count ; s + + )
sum + = split_info [ s ] . profile . array [ i + 1 ] ;
info - > clocks [ i ] = sum ;
}
info - > total_clocks = split_info - > profile . named . total ;
info - > descriptions = descriptions ;
info - > count = STBIR__ARRAY_SIZE ( descriptions ) ;
}
STBIRDEF void stbir_resize_extended_profile_info ( STBIR_PROFILE_INFO * info , STBIR_RESIZE const * resize )
{
stbir_resize_split_profile_info ( info , resize , - 1 , 0 ) ;
}
# endif // STBIR_PROFILE
# undef STBIR_BGR
# undef STBIR_1CHANNEL
# undef STBIR_2CHANNEL
# undef STBIR_RGB
# undef STBIR_RGBA
# undef STBIR_4CHANNEL
# undef STBIR_BGRA
# undef STBIR_ARGB
# undef STBIR_ABGR
# undef STBIR_RA
# undef STBIR_AR
# undef STBIR_RGBA_PM
# undef STBIR_BGRA_PM
# undef STBIR_ARGB_PM
# undef STBIR_ABGR_PM
# undef STBIR_RA_PM
# undef STBIR_AR_PM
# endif // STB_IMAGE_RESIZE_IMPLEMENTATION
# else // STB_IMAGE_RESIZE_HORIZONTALS&STB_IMAGE_RESIZE_DO_VERTICALS
// we reinclude the header file to define all the horizontal functions
// specializing each function for the number of coeffs is 20-40% faster *OVERALL*
// by including the header file again this way, we can still debug the functions
# define STBIR_strs_join2( start, mid, end ) start##mid##end
# define STBIR_strs_join1( start, mid, end ) STBIR_strs_join2( start, mid, end )
# define STBIR_strs_join24( start, mid1, mid2, end ) start##mid1##mid2##end
# define STBIR_strs_join14( start, mid1, mid2, end ) STBIR_strs_join24( start, mid1, mid2, end )
# ifdef STB_IMAGE_RESIZE_DO_CODERS
# ifdef stbir__decode_suffix
# define STBIR__CODER_NAME( name ) STBIR_strs_join1( name, _, stbir__decode_suffix )
# else
# define STBIR__CODER_NAME( name ) name
# endif
# ifdef stbir__decode_swizzle
# define stbir__decode_simdf8_flip(reg) STBIR_strs_join1( STBIR_strs_join1( STBIR_strs_join1( STBIR_strs_join1( stbir__simdf8_0123to,stbir__decode_order0,stbir__decode_order1),stbir__decode_order2,stbir__decode_order3),stbir__decode_order0,stbir__decode_order1),stbir__decode_order2,stbir__decode_order3)(reg, reg)
# define stbir__decode_simdf4_flip(reg) STBIR_strs_join1( STBIR_strs_join1( stbir__simdf_0123to,stbir__decode_order0,stbir__decode_order1),stbir__decode_order2,stbir__decode_order3)(reg, reg)
# define stbir__encode_simdf8_unflip(reg) STBIR_strs_join1( STBIR_strs_join1( STBIR_strs_join1( STBIR_strs_join1( stbir__simdf8_0123to,stbir__encode_order0,stbir__encode_order1),stbir__encode_order2,stbir__encode_order3),stbir__encode_order0,stbir__encode_order1),stbir__encode_order2,stbir__encode_order3)(reg, reg)
# define stbir__encode_simdf4_unflip(reg) STBIR_strs_join1( STBIR_strs_join1( stbir__simdf_0123to,stbir__encode_order0,stbir__encode_order1),stbir__encode_order2,stbir__encode_order3)(reg, reg)
# else
# define stbir__decode_order0 0
# define stbir__decode_order1 1
# define stbir__decode_order2 2
# define stbir__decode_order3 3
# define stbir__encode_order0 0
# define stbir__encode_order1 1
# define stbir__encode_order2 2
# define stbir__encode_order3 3
# define stbir__decode_simdf8_flip(reg)
# define stbir__decode_simdf4_flip(reg)
# define stbir__encode_simdf8_unflip(reg)
# define stbir__encode_simdf4_unflip(reg)
# endif
# ifdef STBIR_SIMD8
# define stbir__encode_simdfX_unflip stbir__encode_simdf8_unflip
# else
# define stbir__encode_simdfX_unflip stbir__encode_simdf4_unflip
# endif
static void STBIR__CODER_NAME ( stbir__decode_uint8_linear_scaled ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float * decode_end = ( float * ) decode + width_times_channels ;
unsigned char const * input = ( unsigned char const * ) inputp ;
# ifdef STBIR_SIMD
unsigned char const * end_input_m16 = input + width_times_channels - 16 ;
if ( width_times_channels > = 16 )
{
decode_end - = 16 ;
for ( ; ; )
{
# ifdef STBIR_SIMD8
stbir__simdi i ; stbir__simdi8 o0 , o1 ;
stbir__simdf8 of0 , of1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi8_expand_u8_to_u32 ( o0 , o1 , i ) ;
stbir__simdi8_convert_i32_to_float ( of0 , o0 ) ;
stbir__simdi8_convert_i32_to_float ( of1 , o1 ) ;
stbir__simdf8_mult ( of0 , of0 , STBIR_max_uint8_as_float_inverted8 ) ;
stbir__simdf8_mult ( of1 , of1 , STBIR_max_uint8_as_float_inverted8 ) ;
stbir__decode_simdf8_flip ( of0 ) ;
stbir__decode_simdf8_flip ( of1 ) ;
stbir__simdf8_store ( decode + 0 , of0 ) ;
stbir__simdf8_store ( decode + 8 , of1 ) ;
# else
stbir__simdi i , o0 , o1 , o2 , o3 ;
stbir__simdf of0 , of1 , of2 , of3 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi_expand_u8_to_u32 ( o0 , o1 , o2 , o3 , i ) ;
stbir__simdi_convert_i32_to_float ( of0 , o0 ) ;
stbir__simdi_convert_i32_to_float ( of1 , o1 ) ;
stbir__simdi_convert_i32_to_float ( of2 , o2 ) ;
stbir__simdi_convert_i32_to_float ( of3 , o3 ) ;
stbir__simdf_mult ( of0 , of0 , STBIR__CONSTF ( STBIR_max_uint8_as_float_inverted ) ) ;
stbir__simdf_mult ( of1 , of1 , STBIR__CONSTF ( STBIR_max_uint8_as_float_inverted ) ) ;
stbir__simdf_mult ( of2 , of2 , STBIR__CONSTF ( STBIR_max_uint8_as_float_inverted ) ) ;
stbir__simdf_mult ( of3 , of3 , STBIR__CONSTF ( STBIR_max_uint8_as_float_inverted ) ) ;
stbir__decode_simdf4_flip ( of0 ) ;
stbir__decode_simdf4_flip ( of1 ) ;
stbir__decode_simdf4_flip ( of2 ) ;
stbir__decode_simdf4_flip ( of3 ) ;
stbir__simdf_store ( decode + 0 , of0 ) ;
stbir__simdf_store ( decode + 4 , of1 ) ;
stbir__simdf_store ( decode + 8 , of2 ) ;
stbir__simdf_store ( decode + 12 , of3 ) ;
# endif
decode + = 16 ;
input + = 16 ;
if ( decode < = decode_end )
continue ;
if ( decode = = ( decode_end + 16 ) )
break ;
decode = decode_end ; // backup and do last couple
input = end_input_m16 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
decode + = 4 ;
while ( decode < = decode_end )
{
STBIR_SIMD_NO_UNROLL ( decode ) ;
decode [ 0 - 4 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) * stbir__max_uint8_as_float_inverted ;
decode [ 1 - 4 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) * stbir__max_uint8_as_float_inverted ;
decode [ 2 - 4 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) * stbir__max_uint8_as_float_inverted ;
decode [ 3 - 4 ] = ( ( float ) ( input [ stbir__decode_order3 ] ) ) * stbir__max_uint8_as_float_inverted ;
decode + = 4 ;
input + = 4 ;
}
decode - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( decode < decode_end )
{
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) * stbir__max_uint8_as_float_inverted ;
# if stbir__coder_min_num >= 2
decode [ 1 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) * stbir__max_uint8_as_float_inverted ;
# endif
# if stbir__coder_min_num >= 3
decode [ 2 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) * stbir__max_uint8_as_float_inverted ;
# endif
decode + = stbir__coder_min_num ;
input + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__encode_uint8_linear_scaled ) ( void * outputp , int width_times_channels , float const * encode )
{
unsigned char STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( unsigned char * ) outputp ;
unsigned char * end_output = ( ( unsigned char * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
if ( width_times_channels > = stbir__simdfX_float_count * 2 )
{
float const * end_encode_m8 = encode + width_times_channels - stbir__simdfX_float_count * 2 ;
end_output - = stbir__simdfX_float_count * 2 ;
for ( ; ; )
{
stbir__simdfX e0 , e1 ;
stbir__simdi i ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdfX_madd_mem ( e0 , STBIR_simd_point5X , STBIR_max_uint8_as_floatX , encode ) ;
stbir__simdfX_madd_mem ( e1 , STBIR_simd_point5X , STBIR_max_uint8_as_floatX , encode + stbir__simdfX_float_count ) ;
stbir__encode_simdfX_unflip ( e0 ) ;
stbir__encode_simdfX_unflip ( e1 ) ;
# ifdef STBIR_SIMD8
stbir__simdf8_pack_to_16bytes ( i , e0 , e1 ) ;
stbir__simdi_store ( output , i ) ;
# else
stbir__simdf_pack_to_8bytes ( i , e0 , e1 ) ;
stbir__simdi_store2 ( output , i ) ;
# endif
encode + = stbir__simdfX_float_count * 2 ;
output + = stbir__simdfX_float_count * 2 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + stbir__simdfX_float_count * 2 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m8 ;
}
return ;
}
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
stbir__simdf e0 ;
stbir__simdi i0 ;
STBIR_NO_UNROLL ( encode ) ;
stbir__simdf_load ( e0 , encode ) ;
stbir__simdf_madd ( e0 , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint8_as_float ) , e0 ) ;
stbir__encode_simdf4_unflip ( e0 ) ;
stbir__simdf_pack_to_8bytes ( i0 , e0 , e0 ) ; // only use first 4
* ( int * ) ( output - 4 ) = stbir__simdi_to_int ( i0 ) ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
stbir__simdf e0 ;
STBIR_NO_UNROLL ( encode ) ;
stbir__simdf_madd1_mem ( e0 , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint8_as_float ) , encode + stbir__encode_order0 ) ; output [ 0 ] = stbir__simdf_convert_float_to_uint8 ( e0 ) ;
# if stbir__coder_min_num >= 2
stbir__simdf_madd1_mem ( e0 , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint8_as_float ) , encode + stbir__encode_order1 ) ; output [ 1 ] = stbir__simdf_convert_float_to_uint8 ( e0 ) ;
# endif
# if stbir__coder_min_num >= 3
stbir__simdf_madd1_mem ( e0 , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint8_as_float ) , encode + stbir__encode_order2 ) ; output [ 2 ] = stbir__simdf_convert_float_to_uint8 ( e0 ) ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
# else
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
float f ;
f = encode [ stbir__encode_order0 ] * stbir__max_uint8_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 0 - 4 ] = ( unsigned char ) f ;
f = encode [ stbir__encode_order1 ] * stbir__max_uint8_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 1 - 4 ] = ( unsigned char ) f ;
f = encode [ stbir__encode_order2 ] * stbir__max_uint8_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 2 - 4 ] = ( unsigned char ) f ;
f = encode [ stbir__encode_order3 ] * stbir__max_uint8_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 3 - 4 ] = ( unsigned char ) f ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
float f ;
STBIR_NO_UNROLL ( encode ) ;
f = encode [ stbir__encode_order0 ] * stbir__max_uint8_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 0 ] = ( unsigned char ) f ;
# if stbir__coder_min_num >= 2
f = encode [ stbir__encode_order1 ] * stbir__max_uint8_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 1 ] = ( unsigned char ) f ;
# endif
# if stbir__coder_min_num >= 3
f = encode [ stbir__encode_order2 ] * stbir__max_uint8_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 2 ] = ( unsigned char ) f ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
# endif
}
static void STBIR__CODER_NAME ( stbir__decode_uint8_linear ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float * decode_end = ( float * ) decode + width_times_channels ;
unsigned char const * input = ( unsigned char const * ) inputp ;
# ifdef STBIR_SIMD
unsigned char const * end_input_m16 = input + width_times_channels - 16 ;
if ( width_times_channels > = 16 )
{
decode_end - = 16 ;
for ( ; ; )
{
# ifdef STBIR_SIMD8
stbir__simdi i ; stbir__simdi8 o0 , o1 ;
stbir__simdf8 of0 , of1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi8_expand_u8_to_u32 ( o0 , o1 , i ) ;
stbir__simdi8_convert_i32_to_float ( of0 , o0 ) ;
stbir__simdi8_convert_i32_to_float ( of1 , o1 ) ;
stbir__decode_simdf8_flip ( of0 ) ;
stbir__decode_simdf8_flip ( of1 ) ;
stbir__simdf8_store ( decode + 0 , of0 ) ;
stbir__simdf8_store ( decode + 8 , of1 ) ;
# else
stbir__simdi i , o0 , o1 , o2 , o3 ;
stbir__simdf of0 , of1 , of2 , of3 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi_expand_u8_to_u32 ( o0 , o1 , o2 , o3 , i ) ;
stbir__simdi_convert_i32_to_float ( of0 , o0 ) ;
stbir__simdi_convert_i32_to_float ( of1 , o1 ) ;
stbir__simdi_convert_i32_to_float ( of2 , o2 ) ;
stbir__simdi_convert_i32_to_float ( of3 , o3 ) ;
stbir__decode_simdf4_flip ( of0 ) ;
stbir__decode_simdf4_flip ( of1 ) ;
stbir__decode_simdf4_flip ( of2 ) ;
stbir__decode_simdf4_flip ( of3 ) ;
stbir__simdf_store ( decode + 0 , of0 ) ;
stbir__simdf_store ( decode + 4 , of1 ) ;
stbir__simdf_store ( decode + 8 , of2 ) ;
stbir__simdf_store ( decode + 12 , of3 ) ;
# endif
decode + = 16 ;
input + = 16 ;
if ( decode < = decode_end )
continue ;
if ( decode = = ( decode_end + 16 ) )
break ;
decode = decode_end ; // backup and do last couple
input = end_input_m16 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
decode + = 4 ;
while ( decode < = decode_end )
{
STBIR_SIMD_NO_UNROLL ( decode ) ;
decode [ 0 - 4 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) ;
decode [ 1 - 4 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) ;
decode [ 2 - 4 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) ;
decode [ 3 - 4 ] = ( ( float ) ( input [ stbir__decode_order3 ] ) ) ;
decode + = 4 ;
input + = 4 ;
}
decode - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( decode < decode_end )
{
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) ;
# if stbir__coder_min_num >= 2
decode [ 1 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) ;
# endif
# if stbir__coder_min_num >= 3
decode [ 2 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) ;
# endif
decode + = stbir__coder_min_num ;
input + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__encode_uint8_linear ) ( void * outputp , int width_times_channels , float const * encode )
{
unsigned char STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( unsigned char * ) outputp ;
unsigned char * end_output = ( ( unsigned char * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
if ( width_times_channels > = stbir__simdfX_float_count * 2 )
{
float const * end_encode_m8 = encode + width_times_channels - stbir__simdfX_float_count * 2 ;
end_output - = stbir__simdfX_float_count * 2 ;
for ( ; ; )
{
stbir__simdfX e0 , e1 ;
stbir__simdi i ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdfX_add_mem ( e0 , STBIR_simd_point5X , encode ) ;
stbir__simdfX_add_mem ( e1 , STBIR_simd_point5X , encode + stbir__simdfX_float_count ) ;
stbir__encode_simdfX_unflip ( e0 ) ;
stbir__encode_simdfX_unflip ( e1 ) ;
# ifdef STBIR_SIMD8
stbir__simdf8_pack_to_16bytes ( i , e0 , e1 ) ;
stbir__simdi_store ( output , i ) ;
# else
stbir__simdf_pack_to_8bytes ( i , e0 , e1 ) ;
stbir__simdi_store2 ( output , i ) ;
# endif
encode + = stbir__simdfX_float_count * 2 ;
output + = stbir__simdfX_float_count * 2 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + stbir__simdfX_float_count * 2 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m8 ;
}
return ;
}
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
stbir__simdf e0 ;
stbir__simdi i0 ;
STBIR_NO_UNROLL ( encode ) ;
stbir__simdf_load ( e0 , encode ) ;
stbir__simdf_add ( e0 , STBIR__CONSTF ( STBIR_simd_point5 ) , e0 ) ;
stbir__encode_simdf4_unflip ( e0 ) ;
stbir__simdf_pack_to_8bytes ( i0 , e0 , e0 ) ; // only use first 4
* ( int * ) ( output - 4 ) = stbir__simdi_to_int ( i0 ) ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
# else
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
float f ;
f = encode [ stbir__encode_order0 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 0 - 4 ] = ( unsigned char ) f ;
f = encode [ stbir__encode_order1 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 1 - 4 ] = ( unsigned char ) f ;
f = encode [ stbir__encode_order2 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 2 - 4 ] = ( unsigned char ) f ;
f = encode [ stbir__encode_order3 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 3 - 4 ] = ( unsigned char ) f ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
float f ;
STBIR_NO_UNROLL ( encode ) ;
f = encode [ stbir__encode_order0 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 0 ] = ( unsigned char ) f ;
# if stbir__coder_min_num >= 2
f = encode [ stbir__encode_order1 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 1 ] = ( unsigned char ) f ;
# endif
# if stbir__coder_min_num >= 3
f = encode [ stbir__encode_order2 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 255 ) ; output [ 2 ] = ( unsigned char ) f ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__decode_uint8_srgb ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float const * decode_end = ( float * ) decode + width_times_channels ;
unsigned char const * input = ( unsigned char const * ) inputp ;
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
decode + = 4 ;
while ( decode < = decode_end )
{
decode [ 0 - 4 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order0 ] ] ;
decode [ 1 - 4 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order1 ] ] ;
decode [ 2 - 4 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order2 ] ] ;
decode [ 3 - 4 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order3 ] ] ;
decode + = 4 ;
input + = 4 ;
}
decode - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( decode < decode_end )
{
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order0 ] ] ;
# if stbir__coder_min_num >= 2
decode [ 1 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order1 ] ] ;
# endif
# if stbir__coder_min_num >= 3
decode [ 2 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order2 ] ] ;
# endif
decode + = stbir__coder_min_num ;
input + = stbir__coder_min_num ;
}
# endif
}
# define stbir__min_max_shift20( i, f ) \
stbir__simdf_max ( f , f , stbir_simdf_casti ( STBIR__CONSTI ( STBIR_almost_zero ) ) ) ; \
stbir__simdf_min ( f , f , stbir_simdf_casti ( STBIR__CONSTI ( STBIR_almost_one ) ) ) ; \
stbir__simdi_32shr ( i , stbir_simdi_castf ( f ) , 20 ) ;
# define stbir__scale_and_convert( i, f ) \
stbir__simdf_madd ( f , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint8_as_float ) , f ) ; \
stbir__simdf_max ( f , f , stbir__simdf_zeroP ( ) ) ; \
stbir__simdf_min ( f , f , STBIR__CONSTF ( STBIR_max_uint8_as_float ) ) ; \
stbir__simdf_convert_float_to_i32 ( i , f ) ;
# define stbir__linear_to_srgb_finish( i, f ) \
{ \
stbir__simdi temp ; \
stbir__simdi_32shr ( temp , stbir_simdi_castf ( f ) , 12 ) ; \
stbir__simdi_and ( temp , temp , STBIR__CONSTI ( STBIR_mastissa_mask ) ) ; \
stbir__simdi_or ( temp , temp , STBIR__CONSTI ( STBIR_topscale ) ) ; \
stbir__simdi_16madd ( i , i , temp ) ; \
stbir__simdi_32shr ( i , i , 16 ) ; \
}
# define stbir__simdi_table_lookup2( v0,v1, table ) \
{ \
stbir__simdi_u32 temp0 , temp1 ; \
temp0 . m128i_i128 = v0 ; \
temp1 . m128i_i128 = v1 ; \
temp0 . m128i_u32 [ 0 ] = table [ temp0 . m128i_i32 [ 0 ] ] ; temp0 . m128i_u32 [ 1 ] = table [ temp0 . m128i_i32 [ 1 ] ] ; temp0 . m128i_u32 [ 2 ] = table [ temp0 . m128i_i32 [ 2 ] ] ; temp0 . m128i_u32 [ 3 ] = table [ temp0 . m128i_i32 [ 3 ] ] ; \
temp1 . m128i_u32 [ 0 ] = table [ temp1 . m128i_i32 [ 0 ] ] ; temp1 . m128i_u32 [ 1 ] = table [ temp1 . m128i_i32 [ 1 ] ] ; temp1 . m128i_u32 [ 2 ] = table [ temp1 . m128i_i32 [ 2 ] ] ; temp1 . m128i_u32 [ 3 ] = table [ temp1 . m128i_i32 [ 3 ] ] ; \
v0 = temp0 . m128i_i128 ; \
v1 = temp1 . m128i_i128 ; \
}
# define stbir__simdi_table_lookup3( v0,v1,v2, table ) \
{ \
stbir__simdi_u32 temp0 , temp1 , temp2 ; \
temp0 . m128i_i128 = v0 ; \
temp1 . m128i_i128 = v1 ; \
temp2 . m128i_i128 = v2 ; \
temp0 . m128i_u32 [ 0 ] = table [ temp0 . m128i_i32 [ 0 ] ] ; temp0 . m128i_u32 [ 1 ] = table [ temp0 . m128i_i32 [ 1 ] ] ; temp0 . m128i_u32 [ 2 ] = table [ temp0 . m128i_i32 [ 2 ] ] ; temp0 . m128i_u32 [ 3 ] = table [ temp0 . m128i_i32 [ 3 ] ] ; \
temp1 . m128i_u32 [ 0 ] = table [ temp1 . m128i_i32 [ 0 ] ] ; temp1 . m128i_u32 [ 1 ] = table [ temp1 . m128i_i32 [ 1 ] ] ; temp1 . m128i_u32 [ 2 ] = table [ temp1 . m128i_i32 [ 2 ] ] ; temp1 . m128i_u32 [ 3 ] = table [ temp1 . m128i_i32 [ 3 ] ] ; \
temp2 . m128i_u32 [ 0 ] = table [ temp2 . m128i_i32 [ 0 ] ] ; temp2 . m128i_u32 [ 1 ] = table [ temp2 . m128i_i32 [ 1 ] ] ; temp2 . m128i_u32 [ 2 ] = table [ temp2 . m128i_i32 [ 2 ] ] ; temp2 . m128i_u32 [ 3 ] = table [ temp2 . m128i_i32 [ 3 ] ] ; \
v0 = temp0 . m128i_i128 ; \
v1 = temp1 . m128i_i128 ; \
v2 = temp2 . m128i_i128 ; \
}
# define stbir__simdi_table_lookup4( v0,v1,v2,v3, table ) \
{ \
stbir__simdi_u32 temp0 , temp1 , temp2 , temp3 ; \
temp0 . m128i_i128 = v0 ; \
temp1 . m128i_i128 = v1 ; \
temp2 . m128i_i128 = v2 ; \
temp3 . m128i_i128 = v3 ; \
temp0 . m128i_u32 [ 0 ] = table [ temp0 . m128i_i32 [ 0 ] ] ; temp0 . m128i_u32 [ 1 ] = table [ temp0 . m128i_i32 [ 1 ] ] ; temp0 . m128i_u32 [ 2 ] = table [ temp0 . m128i_i32 [ 2 ] ] ; temp0 . m128i_u32 [ 3 ] = table [ temp0 . m128i_i32 [ 3 ] ] ; \
temp1 . m128i_u32 [ 0 ] = table [ temp1 . m128i_i32 [ 0 ] ] ; temp1 . m128i_u32 [ 1 ] = table [ temp1 . m128i_i32 [ 1 ] ] ; temp1 . m128i_u32 [ 2 ] = table [ temp1 . m128i_i32 [ 2 ] ] ; temp1 . m128i_u32 [ 3 ] = table [ temp1 . m128i_i32 [ 3 ] ] ; \
temp2 . m128i_u32 [ 0 ] = table [ temp2 . m128i_i32 [ 0 ] ] ; temp2 . m128i_u32 [ 1 ] = table [ temp2 . m128i_i32 [ 1 ] ] ; temp2 . m128i_u32 [ 2 ] = table [ temp2 . m128i_i32 [ 2 ] ] ; temp2 . m128i_u32 [ 3 ] = table [ temp2 . m128i_i32 [ 3 ] ] ; \
temp3 . m128i_u32 [ 0 ] = table [ temp3 . m128i_i32 [ 0 ] ] ; temp3 . m128i_u32 [ 1 ] = table [ temp3 . m128i_i32 [ 1 ] ] ; temp3 . m128i_u32 [ 2 ] = table [ temp3 . m128i_i32 [ 2 ] ] ; temp3 . m128i_u32 [ 3 ] = table [ temp3 . m128i_i32 [ 3 ] ] ; \
v0 = temp0 . m128i_i128 ; \
v1 = temp1 . m128i_i128 ; \
v2 = temp2 . m128i_i128 ; \
v3 = temp3 . m128i_i128 ; \
}
static void STBIR__CODER_NAME ( stbir__encode_uint8_srgb ) ( void * outputp , int width_times_channels , float const * encode )
{
unsigned char STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( unsigned char * ) outputp ;
unsigned char * end_output = ( ( unsigned char * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
stbir_uint32 const * to_srgb = fp32_to_srgb8_tab4 - ( 127 - 13 ) * 8 ;
if ( width_times_channels > = 16 )
{
float const * end_encode_m16 = encode + width_times_channels - 16 ;
end_output - = 16 ;
for ( ; ; )
{
stbir__simdf f0 , f1 , f2 , f3 ;
stbir__simdi i0 , i1 , i2 , i3 ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdf_load4_transposed ( f0 , f1 , f2 , f3 , encode ) ;
stbir__min_max_shift20 ( i0 , f0 ) ;
stbir__min_max_shift20 ( i1 , f1 ) ;
stbir__min_max_shift20 ( i2 , f2 ) ;
stbir__min_max_shift20 ( i3 , f3 ) ;
stbir__simdi_table_lookup4 ( i0 , i1 , i2 , i3 , to_srgb ) ;
stbir__linear_to_srgb_finish ( i0 , f0 ) ;
stbir__linear_to_srgb_finish ( i1 , f1 ) ;
stbir__linear_to_srgb_finish ( i2 , f2 ) ;
stbir__linear_to_srgb_finish ( i3 , f3 ) ;
stbir__interleave_pack_and_store_16_u8 ( output , STBIR_strs_join1 ( i , , stbir__encode_order0 ) , STBIR_strs_join1 ( i , , stbir__encode_order1 ) , STBIR_strs_join1 ( i , , stbir__encode_order2 ) , STBIR_strs_join1 ( i , , stbir__encode_order3 ) ) ;
encode + = 16 ;
output + = 16 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + 16 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m16 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
STBIR_SIMD_NO_UNROLL ( encode ) ;
output [ 0 - 4 ] = stbir__linear_to_srgb_uchar ( encode [ stbir__encode_order0 ] ) ;
output [ 1 - 4 ] = stbir__linear_to_srgb_uchar ( encode [ stbir__encode_order1 ] ) ;
output [ 2 - 4 ] = stbir__linear_to_srgb_uchar ( encode [ stbir__encode_order2 ] ) ;
output [ 3 - 4 ] = stbir__linear_to_srgb_uchar ( encode [ stbir__encode_order3 ] ) ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
STBIR_NO_UNROLL ( encode ) ;
output [ 0 ] = stbir__linear_to_srgb_uchar ( encode [ stbir__encode_order0 ] ) ;
# if stbir__coder_min_num >= 2
output [ 1 ] = stbir__linear_to_srgb_uchar ( encode [ stbir__encode_order1 ] ) ;
# endif
# if stbir__coder_min_num >= 3
output [ 2 ] = stbir__linear_to_srgb_uchar ( encode [ stbir__encode_order2 ] ) ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
}
# if ( stbir__coder_min_num == 4 ) || ( ( stbir__coder_min_num == 1 ) && ( !defined(stbir__decode_swizzle) ) )
static void STBIR__CODER_NAME ( stbir__decode_uint8_srgb4_linearalpha ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float const * decode_end = ( float * ) decode + width_times_channels ;
unsigned char const * input = ( unsigned char const * ) inputp ;
do {
decode [ 0 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order0 ] ] ;
decode [ 1 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order1 ] ] ;
decode [ 2 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order2 ] ] ;
decode [ 3 ] = ( ( float ) input [ stbir__decode_order3 ] ) * stbir__max_uint8_as_float_inverted ;
input + = 4 ;
decode + = 4 ;
} while ( decode < decode_end ) ;
}
static void STBIR__CODER_NAME ( stbir__encode_uint8_srgb4_linearalpha ) ( void * outputp , int width_times_channels , float const * encode )
{
unsigned char STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( unsigned char * ) outputp ;
unsigned char * end_output = ( ( unsigned char * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
stbir_uint32 const * to_srgb = fp32_to_srgb8_tab4 - ( 127 - 13 ) * 8 ;
if ( width_times_channels > = 16 )
{
float const * end_encode_m16 = encode + width_times_channels - 16 ;
end_output - = 16 ;
for ( ; ; )
{
stbir__simdf f0 , f1 , f2 , f3 ;
stbir__simdi i0 , i1 , i2 , i3 ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdf_load4_transposed ( f0 , f1 , f2 , f3 , encode ) ;
stbir__min_max_shift20 ( i0 , f0 ) ;
stbir__min_max_shift20 ( i1 , f1 ) ;
stbir__min_max_shift20 ( i2 , f2 ) ;
stbir__scale_and_convert ( i3 , f3 ) ;
stbir__simdi_table_lookup3 ( i0 , i1 , i2 , to_srgb ) ;
stbir__linear_to_srgb_finish ( i0 , f0 ) ;
stbir__linear_to_srgb_finish ( i1 , f1 ) ;
stbir__linear_to_srgb_finish ( i2 , f2 ) ;
stbir__interleave_pack_and_store_16_u8 ( output , STBIR_strs_join1 ( i , , stbir__encode_order0 ) , STBIR_strs_join1 ( i , , stbir__encode_order1 ) , STBIR_strs_join1 ( i , , stbir__encode_order2 ) , STBIR_strs_join1 ( i , , stbir__encode_order3 ) ) ;
output + = 16 ;
encode + = 16 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + 16 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m16 ;
}
return ;
}
# endif
do {
float f ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
output [ stbir__decode_order0 ] = stbir__linear_to_srgb_uchar ( encode [ 0 ] ) ;
output [ stbir__decode_order1 ] = stbir__linear_to_srgb_uchar ( encode [ 1 ] ) ;
output [ stbir__decode_order2 ] = stbir__linear_to_srgb_uchar ( encode [ 2 ] ) ;
f = encode [ 3 ] * stbir__max_uint8_as_float + 0.5f ;
STBIR_CLAMP ( f , 0 , 255 ) ;
output [ stbir__decode_order3 ] = ( unsigned char ) f ;
output + = 4 ;
encode + = 4 ;
} while ( output < end_output ) ;
}
# endif
# if ( stbir__coder_min_num == 2 ) || ( ( stbir__coder_min_num == 1 ) && ( !defined(stbir__decode_swizzle) ) )
static void STBIR__CODER_NAME ( stbir__decode_uint8_srgb2_linearalpha ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float const * decode_end = ( float * ) decode + width_times_channels ;
unsigned char const * input = ( unsigned char const * ) inputp ;
decode + = 4 ;
while ( decode < = decode_end )
{
decode [ 0 - 4 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order0 ] ] ;
decode [ 1 - 4 ] = ( ( float ) input [ stbir__decode_order1 ] ) * stbir__max_uint8_as_float_inverted ;
decode [ 2 - 4 ] = stbir__srgb_uchar_to_linear_float [ input [ stbir__decode_order0 + 2 ] ] ;
decode [ 3 - 4 ] = ( ( float ) input [ stbir__decode_order1 + 2 ] ) * stbir__max_uint8_as_float_inverted ;
input + = 4 ;
decode + = 4 ;
}
decode - = 4 ;
if ( decode < decode_end )
{
decode [ 0 ] = stbir__srgb_uchar_to_linear_float [ stbir__decode_order0 ] ;
decode [ 1 ] = ( ( float ) input [ stbir__decode_order1 ] ) * stbir__max_uint8_as_float_inverted ;
}
}
static void STBIR__CODER_NAME ( stbir__encode_uint8_srgb2_linearalpha ) ( void * outputp , int width_times_channels , float const * encode )
{
unsigned char STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( unsigned char * ) outputp ;
unsigned char * end_output = ( ( unsigned char * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
stbir_uint32 const * to_srgb = fp32_to_srgb8_tab4 - ( 127 - 13 ) * 8 ;
if ( width_times_channels > = 16 )
{
float const * end_encode_m16 = encode + width_times_channels - 16 ;
end_output - = 16 ;
for ( ; ; )
{
stbir__simdf f0 , f1 , f2 , f3 ;
stbir__simdi i0 , i1 , i2 , i3 ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdf_load4_transposed ( f0 , f1 , f2 , f3 , encode ) ;
stbir__min_max_shift20 ( i0 , f0 ) ;
stbir__scale_and_convert ( i1 , f1 ) ;
stbir__min_max_shift20 ( i2 , f2 ) ;
stbir__scale_and_convert ( i3 , f3 ) ;
stbir__simdi_table_lookup2 ( i0 , i2 , to_srgb ) ;
stbir__linear_to_srgb_finish ( i0 , f0 ) ;
stbir__linear_to_srgb_finish ( i2 , f2 ) ;
stbir__interleave_pack_and_store_16_u8 ( output , STBIR_strs_join1 ( i , , stbir__encode_order0 ) , STBIR_strs_join1 ( i , , stbir__encode_order1 ) , STBIR_strs_join1 ( i , , stbir__encode_order2 ) , STBIR_strs_join1 ( i , , stbir__encode_order3 ) ) ;
output + = 16 ;
encode + = 16 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + 16 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m16 ;
}
return ;
}
# endif
do {
float f ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
output [ stbir__decode_order0 ] = stbir__linear_to_srgb_uchar ( encode [ 0 ] ) ;
f = encode [ 1 ] * stbir__max_uint8_as_float + 0.5f ;
STBIR_CLAMP ( f , 0 , 255 ) ;
output [ stbir__decode_order1 ] = ( unsigned char ) f ;
output + = 2 ;
encode + = 2 ;
} while ( output < end_output ) ;
}
# endif
static void STBIR__CODER_NAME ( stbir__decode_uint16_linear_scaled ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float * decode_end = ( float * ) decode + width_times_channels ;
unsigned short const * input = ( unsigned short const * ) inputp ;
# ifdef STBIR_SIMD
unsigned short const * end_input_m8 = input + width_times_channels - 8 ;
if ( width_times_channels > = 8 )
{
decode_end - = 8 ;
for ( ; ; )
{
# ifdef STBIR_SIMD8
stbir__simdi i ; stbir__simdi8 o ;
stbir__simdf8 of ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi8_expand_u16_to_u32 ( o , i ) ;
stbir__simdi8_convert_i32_to_float ( of , o ) ;
stbir__simdf8_mult ( of , of , STBIR_max_uint16_as_float_inverted8 ) ;
stbir__decode_simdf8_flip ( of ) ;
stbir__simdf8_store ( decode + 0 , of ) ;
# else
stbir__simdi i , o0 , o1 ;
stbir__simdf of0 , of1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi_expand_u16_to_u32 ( o0 , o1 , i ) ;
stbir__simdi_convert_i32_to_float ( of0 , o0 ) ;
stbir__simdi_convert_i32_to_float ( of1 , o1 ) ;
stbir__simdf_mult ( of0 , of0 , STBIR__CONSTF ( STBIR_max_uint16_as_float_inverted ) ) ;
stbir__simdf_mult ( of1 , of1 , STBIR__CONSTF ( STBIR_max_uint16_as_float_inverted ) ) ;
stbir__decode_simdf4_flip ( of0 ) ;
stbir__decode_simdf4_flip ( of1 ) ;
stbir__simdf_store ( decode + 0 , of0 ) ;
stbir__simdf_store ( decode + 4 , of1 ) ;
# endif
decode + = 8 ;
input + = 8 ;
if ( decode < = decode_end )
continue ;
if ( decode = = ( decode_end + 8 ) )
break ;
decode = decode_end ; // backup and do last couple
input = end_input_m8 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
decode + = 4 ;
while ( decode < = decode_end )
{
STBIR_SIMD_NO_UNROLL ( decode ) ;
decode [ 0 - 4 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) * stbir__max_uint16_as_float_inverted ;
decode [ 1 - 4 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) * stbir__max_uint16_as_float_inverted ;
decode [ 2 - 4 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) * stbir__max_uint16_as_float_inverted ;
decode [ 3 - 4 ] = ( ( float ) ( input [ stbir__decode_order3 ] ) ) * stbir__max_uint16_as_float_inverted ;
decode + = 4 ;
input + = 4 ;
}
decode - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( decode < decode_end )
{
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) * stbir__max_uint16_as_float_inverted ;
# if stbir__coder_min_num >= 2
decode [ 1 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) * stbir__max_uint16_as_float_inverted ;
# endif
# if stbir__coder_min_num >= 3
decode [ 2 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) * stbir__max_uint16_as_float_inverted ;
# endif
decode + = stbir__coder_min_num ;
input + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__encode_uint16_linear_scaled ) ( void * outputp , int width_times_channels , float const * encode )
{
unsigned short STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( unsigned short * ) outputp ;
unsigned short * end_output = ( ( unsigned short * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
{
if ( width_times_channels > = stbir__simdfX_float_count * 2 )
{
float const * end_encode_m8 = encode + width_times_channels - stbir__simdfX_float_count * 2 ;
end_output - = stbir__simdfX_float_count * 2 ;
for ( ; ; )
{
stbir__simdfX e0 , e1 ;
stbir__simdiX i ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdfX_madd_mem ( e0 , STBIR_simd_point5X , STBIR_max_uint16_as_floatX , encode ) ;
stbir__simdfX_madd_mem ( e1 , STBIR_simd_point5X , STBIR_max_uint16_as_floatX , encode + stbir__simdfX_float_count ) ;
stbir__encode_simdfX_unflip ( e0 ) ;
stbir__encode_simdfX_unflip ( e1 ) ;
stbir__simdfX_pack_to_words ( i , e0 , e1 ) ;
stbir__simdiX_store ( output , i ) ;
encode + = stbir__simdfX_float_count * 2 ;
output + = stbir__simdfX_float_count * 2 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + stbir__simdfX_float_count * 2 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m8 ;
}
return ;
}
}
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
stbir__simdf e ;
stbir__simdi i ;
STBIR_NO_UNROLL ( encode ) ;
stbir__simdf_load ( e , encode ) ;
stbir__simdf_madd ( e , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint16_as_float ) , e ) ;
stbir__encode_simdf4_unflip ( e ) ;
stbir__simdf_pack_to_8words ( i , e , e ) ; // only use first 4
stbir__simdi_store2 ( output - 4 , i ) ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
stbir__simdf e ;
STBIR_NO_UNROLL ( encode ) ;
stbir__simdf_madd1_mem ( e , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint16_as_float ) , encode + stbir__encode_order0 ) ; output [ 0 ] = stbir__simdf_convert_float_to_short ( e ) ;
# if stbir__coder_min_num >= 2
stbir__simdf_madd1_mem ( e , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint16_as_float ) , encode + stbir__encode_order1 ) ; output [ 1 ] = stbir__simdf_convert_float_to_short ( e ) ;
# endif
# if stbir__coder_min_num >= 3
stbir__simdf_madd1_mem ( e , STBIR__CONSTF ( STBIR_simd_point5 ) , STBIR__CONSTF ( STBIR_max_uint16_as_float ) , encode + stbir__encode_order2 ) ; output [ 2 ] = stbir__simdf_convert_float_to_short ( e ) ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
# else
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
float f ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
f = encode [ stbir__encode_order0 ] * stbir__max_uint16_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 0 - 4 ] = ( unsigned short ) f ;
f = encode [ stbir__encode_order1 ] * stbir__max_uint16_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 1 - 4 ] = ( unsigned short ) f ;
f = encode [ stbir__encode_order2 ] * stbir__max_uint16_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 2 - 4 ] = ( unsigned short ) f ;
f = encode [ stbir__encode_order3 ] * stbir__max_uint16_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 3 - 4 ] = ( unsigned short ) f ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
float f ;
STBIR_NO_UNROLL ( encode ) ;
f = encode [ stbir__encode_order0 ] * stbir__max_uint16_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 0 ] = ( unsigned short ) f ;
# if stbir__coder_min_num >= 2
f = encode [ stbir__encode_order1 ] * stbir__max_uint16_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 1 ] = ( unsigned short ) f ;
# endif
# if stbir__coder_min_num >= 3
f = encode [ stbir__encode_order2 ] * stbir__max_uint16_as_float + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 2 ] = ( unsigned short ) f ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
# endif
}
static void STBIR__CODER_NAME ( stbir__decode_uint16_linear ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float * decode_end = ( float * ) decode + width_times_channels ;
unsigned short const * input = ( unsigned short const * ) inputp ;
# ifdef STBIR_SIMD
unsigned short const * end_input_m8 = input + width_times_channels - 8 ;
if ( width_times_channels > = 8 )
{
decode_end - = 8 ;
for ( ; ; )
{
# ifdef STBIR_SIMD8
stbir__simdi i ; stbir__simdi8 o ;
stbir__simdf8 of ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi8_expand_u16_to_u32 ( o , i ) ;
stbir__simdi8_convert_i32_to_float ( of , o ) ;
stbir__decode_simdf8_flip ( of ) ;
stbir__simdf8_store ( decode + 0 , of ) ;
# else
stbir__simdi i , o0 , o1 ;
stbir__simdf of0 , of1 ;
STBIR_NO_UNROLL ( decode ) ;
stbir__simdi_load ( i , input ) ;
stbir__simdi_expand_u16_to_u32 ( o0 , o1 , i ) ;
stbir__simdi_convert_i32_to_float ( of0 , o0 ) ;
stbir__simdi_convert_i32_to_float ( of1 , o1 ) ;
stbir__decode_simdf4_flip ( of0 ) ;
stbir__decode_simdf4_flip ( of1 ) ;
stbir__simdf_store ( decode + 0 , of0 ) ;
stbir__simdf_store ( decode + 4 , of1 ) ;
# endif
decode + = 8 ;
input + = 8 ;
if ( decode < = decode_end )
continue ;
if ( decode = = ( decode_end + 8 ) )
break ;
decode = decode_end ; // backup and do last couple
input = end_input_m8 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
decode + = 4 ;
while ( decode < = decode_end )
{
STBIR_SIMD_NO_UNROLL ( decode ) ;
decode [ 0 - 4 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) ;
decode [ 1 - 4 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) ;
decode [ 2 - 4 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) ;
decode [ 3 - 4 ] = ( ( float ) ( input [ stbir__decode_order3 ] ) ) ;
decode + = 4 ;
input + = 4 ;
}
decode - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( decode < decode_end )
{
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = ( ( float ) ( input [ stbir__decode_order0 ] ) ) ;
# if stbir__coder_min_num >= 2
decode [ 1 ] = ( ( float ) ( input [ stbir__decode_order1 ] ) ) ;
# endif
# if stbir__coder_min_num >= 3
decode [ 2 ] = ( ( float ) ( input [ stbir__decode_order2 ] ) ) ;
# endif
decode + = stbir__coder_min_num ;
input + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__encode_uint16_linear ) ( void * outputp , int width_times_channels , float const * encode )
{
unsigned short STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( unsigned short * ) outputp ;
unsigned short * end_output = ( ( unsigned short * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
{
if ( width_times_channels > = stbir__simdfX_float_count * 2 )
{
float const * end_encode_m8 = encode + width_times_channels - stbir__simdfX_float_count * 2 ;
end_output - = stbir__simdfX_float_count * 2 ;
for ( ; ; )
{
stbir__simdfX e0 , e1 ;
stbir__simdiX i ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdfX_add_mem ( e0 , STBIR_simd_point5X , encode ) ;
stbir__simdfX_add_mem ( e1 , STBIR_simd_point5X , encode + stbir__simdfX_float_count ) ;
stbir__encode_simdfX_unflip ( e0 ) ;
stbir__encode_simdfX_unflip ( e1 ) ;
stbir__simdfX_pack_to_words ( i , e0 , e1 ) ;
stbir__simdiX_store ( output , i ) ;
encode + = stbir__simdfX_float_count * 2 ;
output + = stbir__simdfX_float_count * 2 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + stbir__simdfX_float_count * 2 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m8 ;
}
return ;
}
}
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
stbir__simdf e ;
stbir__simdi i ;
STBIR_NO_UNROLL ( encode ) ;
stbir__simdf_load ( e , encode ) ;
stbir__simdf_add ( e , STBIR__CONSTF ( STBIR_simd_point5 ) , e ) ;
stbir__encode_simdf4_unflip ( e ) ;
stbir__simdf_pack_to_8words ( i , e , e ) ; // only use first 4
stbir__simdi_store2 ( output - 4 , i ) ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
# else
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
float f ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
f = encode [ stbir__encode_order0 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 0 - 4 ] = ( unsigned short ) f ;
f = encode [ stbir__encode_order1 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 1 - 4 ] = ( unsigned short ) f ;
f = encode [ stbir__encode_order2 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 2 - 4 ] = ( unsigned short ) f ;
f = encode [ stbir__encode_order3 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 3 - 4 ] = ( unsigned short ) f ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
float f ;
STBIR_NO_UNROLL ( encode ) ;
f = encode [ stbir__encode_order0 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 0 ] = ( unsigned short ) f ;
# if stbir__coder_min_num >= 2
f = encode [ stbir__encode_order1 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 1 ] = ( unsigned short ) f ;
# endif
# if stbir__coder_min_num >= 3
f = encode [ stbir__encode_order2 ] + 0.5f ; STBIR_CLAMP ( f , 0 , 65535 ) ; output [ 2 ] = ( unsigned short ) f ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__decode_half_float_linear ) ( float * decodep , int width_times_channels , void const * inputp )
{
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float * decode_end = ( float * ) decode + width_times_channels ;
stbir__FP16 const * input = ( stbir__FP16 const * ) inputp ;
# ifdef STBIR_SIMD
if ( width_times_channels > = 8 )
{
stbir__FP16 const * end_input_m8 = input + width_times_channels - 8 ;
decode_end - = 8 ;
for ( ; ; )
{
STBIR_NO_UNROLL ( decode ) ;
stbir__half_to_float_SIMD ( decode , input ) ;
# ifdef stbir__decode_swizzle
# ifdef STBIR_SIMD8
{
stbir__simdf8 of ;
stbir__simdf8_load ( of , decode ) ;
stbir__decode_simdf8_flip ( of ) ;
stbir__simdf8_store ( decode , of ) ;
}
# else
{
stbir__simdf of0 , of1 ;
stbir__simdf_load ( of0 , decode ) ;
stbir__simdf_load ( of1 , decode + 4 ) ;
stbir__decode_simdf4_flip ( of0 ) ;
stbir__decode_simdf4_flip ( of1 ) ;
stbir__simdf_store ( decode , of0 ) ;
stbir__simdf_store ( decode + 4 , of1 ) ;
}
# endif
# endif
decode + = 8 ;
input + = 8 ;
if ( decode < = decode_end )
continue ;
if ( decode = = ( decode_end + 8 ) )
break ;
decode = decode_end ; // backup and do last couple
input = end_input_m8 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
decode + = 4 ;
while ( decode < = decode_end )
{
STBIR_SIMD_NO_UNROLL ( decode ) ;
decode [ 0 - 4 ] = stbir__half_to_float ( input [ stbir__decode_order0 ] ) ;
decode [ 1 - 4 ] = stbir__half_to_float ( input [ stbir__decode_order1 ] ) ;
decode [ 2 - 4 ] = stbir__half_to_float ( input [ stbir__decode_order2 ] ) ;
decode [ 3 - 4 ] = stbir__half_to_float ( input [ stbir__decode_order3 ] ) ;
decode + = 4 ;
input + = 4 ;
}
decode - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( decode < decode_end )
{
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = stbir__half_to_float ( input [ stbir__decode_order0 ] ) ;
# if stbir__coder_min_num >= 2
decode [ 1 ] = stbir__half_to_float ( input [ stbir__decode_order1 ] ) ;
# endif
# if stbir__coder_min_num >= 3
decode [ 2 ] = stbir__half_to_float ( input [ stbir__decode_order2 ] ) ;
# endif
decode + = stbir__coder_min_num ;
input + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__encode_half_float_linear ) ( void * outputp , int width_times_channels , float const * encode )
{
stbir__FP16 STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( stbir__FP16 * ) outputp ;
stbir__FP16 * end_output = ( ( stbir__FP16 * ) output ) + width_times_channels ;
# ifdef STBIR_SIMD
if ( width_times_channels > = 8 )
{
float const * end_encode_m8 = encode + width_times_channels - 8 ;
end_output - = 8 ;
for ( ; ; )
{
STBIR_SIMD_NO_UNROLL ( encode ) ;
# ifdef stbir__decode_swizzle
# ifdef STBIR_SIMD8
{
stbir__simdf8 of ;
stbir__simdf8_load ( of , encode ) ;
stbir__encode_simdf8_unflip ( of ) ;
stbir__float_to_half_SIMD ( output , ( float * ) & of ) ;
}
# else
{
stbir__simdf of [ 2 ] ;
stbir__simdf_load ( of [ 0 ] , encode ) ;
stbir__simdf_load ( of [ 1 ] , encode + 4 ) ;
stbir__encode_simdf4_unflip ( of [ 0 ] ) ;
stbir__encode_simdf4_unflip ( of [ 1 ] ) ;
stbir__float_to_half_SIMD ( output , ( float * ) of ) ;
}
# endif
# else
stbir__float_to_half_SIMD ( output , encode ) ;
# endif
encode + = 8 ;
output + = 8 ;
if ( output < = end_output )
continue ;
if ( output = = ( end_output + 8 ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m8 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
STBIR_SIMD_NO_UNROLL ( output ) ;
output [ 0 - 4 ] = stbir__float_to_half ( encode [ stbir__encode_order0 ] ) ;
output [ 1 - 4 ] = stbir__float_to_half ( encode [ stbir__encode_order1 ] ) ;
output [ 2 - 4 ] = stbir__float_to_half ( encode [ stbir__encode_order2 ] ) ;
output [ 3 - 4 ] = stbir__float_to_half ( encode [ stbir__encode_order3 ] ) ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
STBIR_NO_UNROLL ( output ) ;
output [ 0 ] = stbir__float_to_half ( encode [ stbir__encode_order0 ] ) ;
# if stbir__coder_min_num >= 2
output [ 1 ] = stbir__float_to_half ( encode [ stbir__encode_order1 ] ) ;
# endif
# if stbir__coder_min_num >= 3
output [ 2 ] = stbir__float_to_half ( encode [ stbir__encode_order2 ] ) ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
}
static void STBIR__CODER_NAME ( stbir__decode_float_linear ) ( float * decodep , int width_times_channels , void const * inputp )
{
# ifdef stbir__decode_swizzle
float STBIR_STREAMOUT_PTR ( * ) decode = decodep ;
float * decode_end = ( float * ) decode + width_times_channels ;
float const * input = ( float const * ) inputp ;
# ifdef STBIR_SIMD
if ( width_times_channels > = 16 )
{
float const * end_input_m16 = input + width_times_channels - 16 ;
decode_end - = 16 ;
for ( ; ; )
{
STBIR_NO_UNROLL ( decode ) ;
# ifdef stbir__decode_swizzle
# ifdef STBIR_SIMD8
{
stbir__simdf8 of0 , of1 ;
stbir__simdf8_load ( of0 , input ) ;
stbir__simdf8_load ( of1 , input + 8 ) ;
stbir__decode_simdf8_flip ( of0 ) ;
stbir__decode_simdf8_flip ( of1 ) ;
stbir__simdf8_store ( decode , of0 ) ;
stbir__simdf8_store ( decode + 8 , of1 ) ;
}
# else
{
stbir__simdf of0 , of1 , of2 , of3 ;
stbir__simdf_load ( of0 , input ) ;
stbir__simdf_load ( of1 , input + 4 ) ;
stbir__simdf_load ( of2 , input + 8 ) ;
stbir__simdf_load ( of3 , input + 12 ) ;
stbir__decode_simdf4_flip ( of0 ) ;
stbir__decode_simdf4_flip ( of1 ) ;
stbir__decode_simdf4_flip ( of2 ) ;
stbir__decode_simdf4_flip ( of3 ) ;
stbir__simdf_store ( decode , of0 ) ;
stbir__simdf_store ( decode + 4 , of1 ) ;
stbir__simdf_store ( decode + 8 , of2 ) ;
stbir__simdf_store ( decode + 12 , of3 ) ;
}
# endif
# endif
decode + = 16 ;
input + = 16 ;
if ( decode < = decode_end )
continue ;
if ( decode = = ( decode_end + 16 ) )
break ;
decode = decode_end ; // backup and do last couple
input = end_input_m16 ;
}
return ;
}
# endif
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
decode + = 4 ;
while ( decode < = decode_end )
{
STBIR_SIMD_NO_UNROLL ( decode ) ;
decode [ 0 - 4 ] = input [ stbir__decode_order0 ] ;
decode [ 1 - 4 ] = input [ stbir__decode_order1 ] ;
decode [ 2 - 4 ] = input [ stbir__decode_order2 ] ;
decode [ 3 - 4 ] = input [ stbir__decode_order3 ] ;
decode + = 4 ;
input + = 4 ;
}
decode - = 4 ;
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( decode < decode_end )
{
STBIR_NO_UNROLL ( decode ) ;
decode [ 0 ] = input [ stbir__decode_order0 ] ;
# if stbir__coder_min_num >= 2
decode [ 1 ] = input [ stbir__decode_order1 ] ;
# endif
# if stbir__coder_min_num >= 3
decode [ 2 ] = input [ stbir__decode_order2 ] ;
# endif
decode + = stbir__coder_min_num ;
input + = stbir__coder_min_num ;
}
# endif
# else
if ( ( void * ) decodep ! = inputp )
STBIR_MEMCPY ( decodep , inputp , width_times_channels * sizeof ( float ) ) ;
# endif
}
static void STBIR__CODER_NAME ( stbir__encode_float_linear ) ( void * outputp , int width_times_channels , float const * encode )
{
# if !defined( STBIR_FLOAT_HIGH_CLAMP ) && !defined(STBIR_FLOAT_LO_CLAMP) && !defined(stbir__decode_swizzle)
if ( ( void * ) outputp ! = ( void * ) encode )
STBIR_MEMCPY ( outputp , encode , width_times_channels * sizeof ( float ) ) ;
# else
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = ( float * ) outputp ;
float * end_output = ( ( float * ) output ) + width_times_channels ;
# ifdef STBIR_FLOAT_HIGH_CLAMP
# define stbir_scalar_hi_clamp( v ) if ( v > STBIR_FLOAT_HIGH_CLAMP ) v = STBIR_FLOAT_HIGH_CLAMP;
# else
# define stbir_scalar_hi_clamp( v )
# endif
# ifdef STBIR_FLOAT_LOW_CLAMP
# define stbir_scalar_lo_clamp( v ) if ( v < STBIR_FLOAT_LOW_CLAMP ) v = STBIR_FLOAT_LOW_CLAMP;
# else
# define stbir_scalar_lo_clamp( v )
# endif
# ifdef STBIR_SIMD
# ifdef STBIR_FLOAT_HIGH_CLAMP
const stbir__simdfX high_clamp = stbir__simdf_frepX ( STBIR_FLOAT_HIGH_CLAMP ) ;
# endif
# ifdef STBIR_FLOAT_LOW_CLAMP
const stbir__simdfX low_clamp = stbir__simdf_frepX ( STBIR_FLOAT_LOW_CLAMP ) ;
# endif
if ( width_times_channels > = ( stbir__simdfX_float_count * 2 ) )
{
float const * end_encode_m8 = encode + width_times_channels - ( stbir__simdfX_float_count * 2 ) ;
end_output - = ( stbir__simdfX_float_count * 2 ) ;
for ( ; ; )
{
stbir__simdfX e0 , e1 ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
stbir__simdfX_load ( e0 , encode ) ;
stbir__simdfX_load ( e1 , encode + stbir__simdfX_float_count ) ;
# ifdef STBIR_FLOAT_HIGH_CLAMP
stbir__simdfX_min ( e0 , e0 , high_clamp ) ;
stbir__simdfX_min ( e1 , e1 , high_clamp ) ;
# endif
# ifdef STBIR_FLOAT_LOW_CLAMP
stbir__simdfX_max ( e0 , e0 , low_clamp ) ;
stbir__simdfX_max ( e1 , e1 , low_clamp ) ;
# endif
stbir__encode_simdfX_unflip ( e0 ) ;
stbir__encode_simdfX_unflip ( e1 ) ;
stbir__simdfX_store ( output , e0 ) ;
stbir__simdfX_store ( output + stbir__simdfX_float_count , e1 ) ;
encode + = stbir__simdfX_float_count * 2 ;
output + = stbir__simdfX_float_count * 2 ;
if ( output < end_output )
continue ;
if ( output = = ( end_output + ( stbir__simdfX_float_count * 2 ) ) )
break ;
output = end_output ; // backup and do last couple
encode = end_encode_m8 ;
}
return ;
}
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
stbir__simdf e0 ;
STBIR_NO_UNROLL ( encode ) ;
stbir__simdf_load ( e0 , encode ) ;
# ifdef STBIR_FLOAT_HIGH_CLAMP
stbir__simdf_min ( e0 , e0 , high_clamp ) ;
# endif
# ifdef STBIR_FLOAT_LOW_CLAMP
stbir__simdf_max ( e0 , e0 , low_clamp ) ;
# endif
stbir__encode_simdf4_unflip ( e0 ) ;
stbir__simdf_store ( output - 4 , e0 ) ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
# else
// try to do blocks of 4 when you can
# if stbir__coder_min_num != 3 // doesn't divide cleanly by four
output + = 4 ;
while ( output < = end_output )
{
float e ;
STBIR_SIMD_NO_UNROLL ( encode ) ;
e = encode [ stbir__encode_order0 ] ; stbir_scalar_hi_clamp ( e ) ; stbir_scalar_lo_clamp ( e ) ; output [ 0 - 4 ] = e ;
e = encode [ stbir__encode_order1 ] ; stbir_scalar_hi_clamp ( e ) ; stbir_scalar_lo_clamp ( e ) ; output [ 1 - 4 ] = e ;
e = encode [ stbir__encode_order2 ] ; stbir_scalar_hi_clamp ( e ) ; stbir_scalar_lo_clamp ( e ) ; output [ 2 - 4 ] = e ;
e = encode [ stbir__encode_order3 ] ; stbir_scalar_hi_clamp ( e ) ; stbir_scalar_lo_clamp ( e ) ; output [ 3 - 4 ] = e ;
output + = 4 ;
encode + = 4 ;
}
output - = 4 ;
# endif
# endif
// do the remnants
# if stbir__coder_min_num < 4
while ( output < end_output )
{
float e ;
STBIR_NO_UNROLL ( encode ) ;
e = encode [ stbir__encode_order0 ] ; stbir_scalar_hi_clamp ( e ) ; stbir_scalar_lo_clamp ( e ) ; output [ 0 ] = e ;
# if stbir__coder_min_num >= 2
e = encode [ stbir__encode_order1 ] ; stbir_scalar_hi_clamp ( e ) ; stbir_scalar_lo_clamp ( e ) ; output [ 1 ] = e ;
# endif
# if stbir__coder_min_num >= 3
e = encode [ stbir__encode_order2 ] ; stbir_scalar_hi_clamp ( e ) ; stbir_scalar_lo_clamp ( e ) ; output [ 2 ] = e ;
# endif
output + = stbir__coder_min_num ;
encode + = stbir__coder_min_num ;
}
# endif
# endif
}
# undef stbir__decode_suffix
# undef stbir__decode_simdf8_flip
# undef stbir__decode_simdf4_flip
# undef stbir__decode_order0
# undef stbir__decode_order1
# undef stbir__decode_order2
# undef stbir__decode_order3
# undef stbir__encode_order0
# undef stbir__encode_order1
# undef stbir__encode_order2
# undef stbir__encode_order3
# undef stbir__encode_simdf8_unflip
# undef stbir__encode_simdf4_unflip
# undef stbir__encode_simdfX_unflip
# undef STBIR__CODER_NAME
# undef stbir__coder_min_num
# undef stbir__decode_swizzle
# undef stbir_scalar_hi_clamp
# undef stbir_scalar_lo_clamp
# undef STB_IMAGE_RESIZE_DO_CODERS
# elif defined( STB_IMAGE_RESIZE_DO_VERTICALS)
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# define STBIR_chans( start, end ) STBIR_strs_join14(start,STBIR__vertical_channels,end,_cont)
# else
# define STBIR_chans( start, end ) STBIR_strs_join1(start,STBIR__vertical_channels,end)
# endif
# if STBIR__vertical_channels >= 1
# define stbIF0( code ) code
# else
# define stbIF0( code )
# endif
# if STBIR__vertical_channels >= 2
# define stbIF1( code ) code
# else
# define stbIF1( code )
# endif
# if STBIR__vertical_channels >= 3
# define stbIF2( code ) code
# else
# define stbIF2( code )
# endif
# if STBIR__vertical_channels >= 4
# define stbIF3( code ) code
# else
# define stbIF3( code )
# endif
# if STBIR__vertical_channels >= 5
# define stbIF4( code ) code
# else
# define stbIF4( code )
# endif
# if STBIR__vertical_channels >= 6
# define stbIF5( code ) code
# else
# define stbIF5( code )
# endif
# if STBIR__vertical_channels >= 7
# define stbIF6( code ) code
# else
# define stbIF6( code )
# endif
# if STBIR__vertical_channels >= 8
# define stbIF7( code ) code
# else
# define stbIF7( code )
# endif
static void STBIR_chans ( stbir__vertical_scatter_with_ , _coeffs ) ( float * * outputs , float const * vertical_coefficients , float const * input , float const * input_end )
{
stbIF0 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output0 = outputs [ 0 ] ; float c0s = vertical_coefficients [ 0 ] ; )
stbIF1 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output1 = outputs [ 1 ] ; float c1s = vertical_coefficients [ 1 ] ; )
stbIF2 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output2 = outputs [ 2 ] ; float c2s = vertical_coefficients [ 2 ] ; )
stbIF3 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output3 = outputs [ 3 ] ; float c3s = vertical_coefficients [ 3 ] ; )
stbIF4 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output4 = outputs [ 4 ] ; float c4s = vertical_coefficients [ 4 ] ; )
stbIF5 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output5 = outputs [ 5 ] ; float c5s = vertical_coefficients [ 5 ] ; )
stbIF6 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output6 = outputs [ 6 ] ; float c6s = vertical_coefficients [ 6 ] ; )
stbIF7 ( float STBIR_SIMD_STREAMOUT_PTR ( * ) output7 = outputs [ 7 ] ; float c7s = vertical_coefficients [ 7 ] ; )
# ifdef STBIR_SIMD
{
stbIF0 ( stbir__simdfX c0 = stbir__simdf_frepX ( c0s ) ; )
stbIF1 ( stbir__simdfX c1 = stbir__simdf_frepX ( c1s ) ; )
stbIF2 ( stbir__simdfX c2 = stbir__simdf_frepX ( c2s ) ; )
stbIF3 ( stbir__simdfX c3 = stbir__simdf_frepX ( c3s ) ; )
stbIF4 ( stbir__simdfX c4 = stbir__simdf_frepX ( c4s ) ; )
stbIF5 ( stbir__simdfX c5 = stbir__simdf_frepX ( c5s ) ; )
stbIF6 ( stbir__simdfX c6 = stbir__simdf_frepX ( c6s ) ; )
stbIF7 ( stbir__simdfX c7 = stbir__simdf_frepX ( c7s ) ; )
while ( ( ( char * ) input_end - ( char * ) input ) > = ( 16 * stbir__simdfX_float_count ) )
{
stbir__simdfX o0 , o1 , o2 , o3 , r0 , r1 , r2 , r3 ;
STBIR_SIMD_NO_UNROLL ( output0 ) ;
stbir__simdfX_load ( r0 , input ) ; stbir__simdfX_load ( r1 , input + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input + ( 3 * stbir__simdfX_float_count ) ) ;
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( stbir__simdfX_load ( o0 , output0 ) ; stbir__simdfX_load ( o1 , output0 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output0 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output0 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c0 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c0 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c0 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c0 ) ;
stbir__simdfX_store ( output0 , o0 ) ; stbir__simdfX_store ( output0 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output0 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output0 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF1 ( stbir__simdfX_load ( o0 , output1 ) ; stbir__simdfX_load ( o1 , output1 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output1 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output1 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c1 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c1 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c1 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c1 ) ;
stbir__simdfX_store ( output1 , o0 ) ; stbir__simdfX_store ( output1 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output1 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output1 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF2 ( stbir__simdfX_load ( o0 , output2 ) ; stbir__simdfX_load ( o1 , output2 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output2 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output2 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c2 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c2 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c2 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c2 ) ;
stbir__simdfX_store ( output2 , o0 ) ; stbir__simdfX_store ( output2 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output2 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output2 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF3 ( stbir__simdfX_load ( o0 , output3 ) ; stbir__simdfX_load ( o1 , output3 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output3 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output3 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c3 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c3 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c3 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c3 ) ;
stbir__simdfX_store ( output3 , o0 ) ; stbir__simdfX_store ( output3 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output3 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output3 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF4 ( stbir__simdfX_load ( o0 , output4 ) ; stbir__simdfX_load ( o1 , output4 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output4 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output4 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c4 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c4 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c4 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c4 ) ;
stbir__simdfX_store ( output4 , o0 ) ; stbir__simdfX_store ( output4 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output4 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output4 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF5 ( stbir__simdfX_load ( o0 , output5 ) ; stbir__simdfX_load ( o1 , output5 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output5 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output5 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c5 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c5 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c5 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c5 ) ;
stbir__simdfX_store ( output5 , o0 ) ; stbir__simdfX_store ( output5 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output5 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output5 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF6 ( stbir__simdfX_load ( o0 , output6 ) ; stbir__simdfX_load ( o1 , output6 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output6 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output6 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c6 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c6 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c6 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c6 ) ;
stbir__simdfX_store ( output6 , o0 ) ; stbir__simdfX_store ( output6 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output6 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output6 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF7 ( stbir__simdfX_load ( o0 , output7 ) ; stbir__simdfX_load ( o1 , output7 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output7 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output7 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c7 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c7 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c7 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c7 ) ;
stbir__simdfX_store ( output7 , o0 ) ; stbir__simdfX_store ( output7 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output7 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output7 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
# else
stbIF0 ( stbir__simdfX_mult ( o0 , r0 , c0 ) ; stbir__simdfX_mult ( o1 , r1 , c0 ) ; stbir__simdfX_mult ( o2 , r2 , c0 ) ; stbir__simdfX_mult ( o3 , r3 , c0 ) ;
stbir__simdfX_store ( output0 , o0 ) ; stbir__simdfX_store ( output0 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output0 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output0 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF1 ( stbir__simdfX_mult ( o0 , r0 , c1 ) ; stbir__simdfX_mult ( o1 , r1 , c1 ) ; stbir__simdfX_mult ( o2 , r2 , c1 ) ; stbir__simdfX_mult ( o3 , r3 , c1 ) ;
stbir__simdfX_store ( output1 , o0 ) ; stbir__simdfX_store ( output1 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output1 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output1 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF2 ( stbir__simdfX_mult ( o0 , r0 , c2 ) ; stbir__simdfX_mult ( o1 , r1 , c2 ) ; stbir__simdfX_mult ( o2 , r2 , c2 ) ; stbir__simdfX_mult ( o3 , r3 , c2 ) ;
stbir__simdfX_store ( output2 , o0 ) ; stbir__simdfX_store ( output2 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output2 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output2 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF3 ( stbir__simdfX_mult ( o0 , r0 , c3 ) ; stbir__simdfX_mult ( o1 , r1 , c3 ) ; stbir__simdfX_mult ( o2 , r2 , c3 ) ; stbir__simdfX_mult ( o3 , r3 , c3 ) ;
stbir__simdfX_store ( output3 , o0 ) ; stbir__simdfX_store ( output3 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output3 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output3 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF4 ( stbir__simdfX_mult ( o0 , r0 , c4 ) ; stbir__simdfX_mult ( o1 , r1 , c4 ) ; stbir__simdfX_mult ( o2 , r2 , c4 ) ; stbir__simdfX_mult ( o3 , r3 , c4 ) ;
stbir__simdfX_store ( output4 , o0 ) ; stbir__simdfX_store ( output4 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output4 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output4 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF5 ( stbir__simdfX_mult ( o0 , r0 , c5 ) ; stbir__simdfX_mult ( o1 , r1 , c5 ) ; stbir__simdfX_mult ( o2 , r2 , c5 ) ; stbir__simdfX_mult ( o3 , r3 , c5 ) ;
stbir__simdfX_store ( output5 , o0 ) ; stbir__simdfX_store ( output5 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output5 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output5 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF6 ( stbir__simdfX_mult ( o0 , r0 , c6 ) ; stbir__simdfX_mult ( o1 , r1 , c6 ) ; stbir__simdfX_mult ( o2 , r2 , c6 ) ; stbir__simdfX_mult ( o3 , r3 , c6 ) ;
stbir__simdfX_store ( output6 , o0 ) ; stbir__simdfX_store ( output6 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output6 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output6 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
stbIF7 ( stbir__simdfX_mult ( o0 , r0 , c7 ) ; stbir__simdfX_mult ( o1 , r1 , c7 ) ; stbir__simdfX_mult ( o2 , r2 , c7 ) ; stbir__simdfX_mult ( o3 , r3 , c7 ) ;
stbir__simdfX_store ( output7 , o0 ) ; stbir__simdfX_store ( output7 + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output7 + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output7 + ( 3 * stbir__simdfX_float_count ) , o3 ) ; )
# endif
input + = ( 4 * stbir__simdfX_float_count ) ;
stbIF0 ( output0 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF1 ( output1 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF2 ( output2 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF3 ( output3 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF4 ( output4 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF5 ( output5 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF6 ( output6 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF7 ( output7 + = ( 4 * stbir__simdfX_float_count ) ; )
}
while ( ( ( char * ) input_end - ( char * ) input ) > = 16 )
{
stbir__simdf o0 , r0 ;
STBIR_SIMD_NO_UNROLL ( output0 ) ;
stbir__simdf_load ( r0 , input ) ;
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( stbir__simdf_load ( o0 , output0 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c0 ) ) ; stbir__simdf_store ( output0 , o0 ) ; )
stbIF1 ( stbir__simdf_load ( o0 , output1 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c1 ) ) ; stbir__simdf_store ( output1 , o0 ) ; )
stbIF2 ( stbir__simdf_load ( o0 , output2 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c2 ) ) ; stbir__simdf_store ( output2 , o0 ) ; )
stbIF3 ( stbir__simdf_load ( o0 , output3 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c3 ) ) ; stbir__simdf_store ( output3 , o0 ) ; )
stbIF4 ( stbir__simdf_load ( o0 , output4 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c4 ) ) ; stbir__simdf_store ( output4 , o0 ) ; )
stbIF5 ( stbir__simdf_load ( o0 , output5 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c5 ) ) ; stbir__simdf_store ( output5 , o0 ) ; )
stbIF6 ( stbir__simdf_load ( o0 , output6 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c6 ) ) ; stbir__simdf_store ( output6 , o0 ) ; )
stbIF7 ( stbir__simdf_load ( o0 , output7 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c7 ) ) ; stbir__simdf_store ( output7 , o0 ) ; )
# else
stbIF0 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c0 ) ) ; stbir__simdf_store ( output0 , o0 ) ; )
stbIF1 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c1 ) ) ; stbir__simdf_store ( output1 , o0 ) ; )
stbIF2 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c2 ) ) ; stbir__simdf_store ( output2 , o0 ) ; )
stbIF3 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c3 ) ) ; stbir__simdf_store ( output3 , o0 ) ; )
stbIF4 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c4 ) ) ; stbir__simdf_store ( output4 , o0 ) ; )
stbIF5 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c5 ) ) ; stbir__simdf_store ( output5 , o0 ) ; )
stbIF6 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c6 ) ) ; stbir__simdf_store ( output6 , o0 ) ; )
stbIF7 ( stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c7 ) ) ; stbir__simdf_store ( output7 , o0 ) ; )
# endif
input + = 4 ;
stbIF0 ( output0 + = 4 ; ) stbIF1 ( output1 + = 4 ; ) stbIF2 ( output2 + = 4 ; ) stbIF3 ( output3 + = 4 ; ) stbIF4 ( output4 + = 4 ; ) stbIF5 ( output5 + = 4 ; ) stbIF6 ( output6 + = 4 ; ) stbIF7 ( output7 + = 4 ; )
}
}
# else
while ( ( ( char * ) input_end - ( char * ) input ) > = 16 )
{
float r0 , r1 , r2 , r3 ;
STBIR_NO_UNROLL ( input ) ;
r0 = input [ 0 ] , r1 = input [ 1 ] , r2 = input [ 2 ] , r3 = input [ 3 ] ;
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( output0 [ 0 ] + = ( r0 * c0s ) ; output0 [ 1 ] + = ( r1 * c0s ) ; output0 [ 2 ] + = ( r2 * c0s ) ; output0 [ 3 ] + = ( r3 * c0s ) ; )
stbIF1 ( output1 [ 0 ] + = ( r0 * c1s ) ; output1 [ 1 ] + = ( r1 * c1s ) ; output1 [ 2 ] + = ( r2 * c1s ) ; output1 [ 3 ] + = ( r3 * c1s ) ; )
stbIF2 ( output2 [ 0 ] + = ( r0 * c2s ) ; output2 [ 1 ] + = ( r1 * c2s ) ; output2 [ 2 ] + = ( r2 * c2s ) ; output2 [ 3 ] + = ( r3 * c2s ) ; )
stbIF3 ( output3 [ 0 ] + = ( r0 * c3s ) ; output3 [ 1 ] + = ( r1 * c3s ) ; output3 [ 2 ] + = ( r2 * c3s ) ; output3 [ 3 ] + = ( r3 * c3s ) ; )
stbIF4 ( output4 [ 0 ] + = ( r0 * c4s ) ; output4 [ 1 ] + = ( r1 * c4s ) ; output4 [ 2 ] + = ( r2 * c4s ) ; output4 [ 3 ] + = ( r3 * c4s ) ; )
stbIF5 ( output5 [ 0 ] + = ( r0 * c5s ) ; output5 [ 1 ] + = ( r1 * c5s ) ; output5 [ 2 ] + = ( r2 * c5s ) ; output5 [ 3 ] + = ( r3 * c5s ) ; )
stbIF6 ( output6 [ 0 ] + = ( r0 * c6s ) ; output6 [ 1 ] + = ( r1 * c6s ) ; output6 [ 2 ] + = ( r2 * c6s ) ; output6 [ 3 ] + = ( r3 * c6s ) ; )
stbIF7 ( output7 [ 0 ] + = ( r0 * c7s ) ; output7 [ 1 ] + = ( r1 * c7s ) ; output7 [ 2 ] + = ( r2 * c7s ) ; output7 [ 3 ] + = ( r3 * c7s ) ; )
# else
stbIF0 ( output0 [ 0 ] = ( r0 * c0s ) ; output0 [ 1 ] = ( r1 * c0s ) ; output0 [ 2 ] = ( r2 * c0s ) ; output0 [ 3 ] = ( r3 * c0s ) ; )
stbIF1 ( output1 [ 0 ] = ( r0 * c1s ) ; output1 [ 1 ] = ( r1 * c1s ) ; output1 [ 2 ] = ( r2 * c1s ) ; output1 [ 3 ] = ( r3 * c1s ) ; )
stbIF2 ( output2 [ 0 ] = ( r0 * c2s ) ; output2 [ 1 ] = ( r1 * c2s ) ; output2 [ 2 ] = ( r2 * c2s ) ; output2 [ 3 ] = ( r3 * c2s ) ; )
stbIF3 ( output3 [ 0 ] = ( r0 * c3s ) ; output3 [ 1 ] = ( r1 * c3s ) ; output3 [ 2 ] = ( r2 * c3s ) ; output3 [ 3 ] = ( r3 * c3s ) ; )
stbIF4 ( output4 [ 0 ] = ( r0 * c4s ) ; output4 [ 1 ] = ( r1 * c4s ) ; output4 [ 2 ] = ( r2 * c4s ) ; output4 [ 3 ] = ( r3 * c4s ) ; )
stbIF5 ( output5 [ 0 ] = ( r0 * c5s ) ; output5 [ 1 ] = ( r1 * c5s ) ; output5 [ 2 ] = ( r2 * c5s ) ; output5 [ 3 ] = ( r3 * c5s ) ; )
stbIF6 ( output6 [ 0 ] = ( r0 * c6s ) ; output6 [ 1 ] = ( r1 * c6s ) ; output6 [ 2 ] = ( r2 * c6s ) ; output6 [ 3 ] = ( r3 * c6s ) ; )
stbIF7 ( output7 [ 0 ] = ( r0 * c7s ) ; output7 [ 1 ] = ( r1 * c7s ) ; output7 [ 2 ] = ( r2 * c7s ) ; output7 [ 3 ] = ( r3 * c7s ) ; )
# endif
input + = 4 ;
stbIF0 ( output0 + = 4 ; ) stbIF1 ( output1 + = 4 ; ) stbIF2 ( output2 + = 4 ; ) stbIF3 ( output3 + = 4 ; ) stbIF4 ( output4 + = 4 ; ) stbIF5 ( output5 + = 4 ; ) stbIF6 ( output6 + = 4 ; ) stbIF7 ( output7 + = 4 ; )
}
# endif
while ( input < input_end )
{
float r = input [ 0 ] ;
STBIR_NO_UNROLL ( output0 ) ;
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( output0 [ 0 ] + = ( r * c0s ) ; )
stbIF1 ( output1 [ 0 ] + = ( r * c1s ) ; )
stbIF2 ( output2 [ 0 ] + = ( r * c2s ) ; )
stbIF3 ( output3 [ 0 ] + = ( r * c3s ) ; )
stbIF4 ( output4 [ 0 ] + = ( r * c4s ) ; )
stbIF5 ( output5 [ 0 ] + = ( r * c5s ) ; )
stbIF6 ( output6 [ 0 ] + = ( r * c6s ) ; )
stbIF7 ( output7 [ 0 ] + = ( r * c7s ) ; )
# else
stbIF0 ( output0 [ 0 ] = ( r * c0s ) ; )
stbIF1 ( output1 [ 0 ] = ( r * c1s ) ; )
stbIF2 ( output2 [ 0 ] = ( r * c2s ) ; )
stbIF3 ( output3 [ 0 ] = ( r * c3s ) ; )
stbIF4 ( output4 [ 0 ] = ( r * c4s ) ; )
stbIF5 ( output5 [ 0 ] = ( r * c5s ) ; )
stbIF6 ( output6 [ 0 ] = ( r * c6s ) ; )
stbIF7 ( output7 [ 0 ] = ( r * c7s ) ; )
# endif
+ + input ;
stbIF0 ( + + output0 ; ) stbIF1 ( + + output1 ; ) stbIF2 ( + + output2 ; ) stbIF3 ( + + output3 ; ) stbIF4 ( + + output4 ; ) stbIF5 ( + + output5 ; ) stbIF6 ( + + output6 ; ) stbIF7 ( + + output7 ; )
}
}
static void STBIR_chans ( stbir__vertical_gather_with_ , _coeffs ) ( float * outputp , float const * vertical_coefficients , float const * * inputs , float const * input0_end )
{
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = outputp ;
stbIF0 ( float const * input0 = inputs [ 0 ] ; float c0s = vertical_coefficients [ 0 ] ; )
stbIF1 ( float const * input1 = inputs [ 1 ] ; float c1s = vertical_coefficients [ 1 ] ; )
stbIF2 ( float const * input2 = inputs [ 2 ] ; float c2s = vertical_coefficients [ 2 ] ; )
stbIF3 ( float const * input3 = inputs [ 3 ] ; float c3s = vertical_coefficients [ 3 ] ; )
stbIF4 ( float const * input4 = inputs [ 4 ] ; float c4s = vertical_coefficients [ 4 ] ; )
stbIF5 ( float const * input5 = inputs [ 5 ] ; float c5s = vertical_coefficients [ 5 ] ; )
stbIF6 ( float const * input6 = inputs [ 6 ] ; float c6s = vertical_coefficients [ 6 ] ; )
stbIF7 ( float const * input7 = inputs [ 7 ] ; float c7s = vertical_coefficients [ 7 ] ; )
# if ( STBIR__vertical_channels == 1 ) && !defined(STB_IMAGE_RESIZE_VERTICAL_CONTINUE)
// check single channel one weight
if ( ( c0s > = ( 1.0f - 0.000001f ) ) & & ( c0s < = ( 1.0f + 0.000001f ) ) )
{
STBIR_MEMCPY ( output , input0 , ( char * ) input0_end - ( char * ) input0 ) ;
return ;
}
# endif
# ifdef STBIR_SIMD
{
stbIF0 ( stbir__simdfX c0 = stbir__simdf_frepX ( c0s ) ; )
stbIF1 ( stbir__simdfX c1 = stbir__simdf_frepX ( c1s ) ; )
stbIF2 ( stbir__simdfX c2 = stbir__simdf_frepX ( c2s ) ; )
stbIF3 ( stbir__simdfX c3 = stbir__simdf_frepX ( c3s ) ; )
stbIF4 ( stbir__simdfX c4 = stbir__simdf_frepX ( c4s ) ; )
stbIF5 ( stbir__simdfX c5 = stbir__simdf_frepX ( c5s ) ; )
stbIF6 ( stbir__simdfX c6 = stbir__simdf_frepX ( c6s ) ; )
stbIF7 ( stbir__simdfX c7 = stbir__simdf_frepX ( c7s ) ; )
while ( ( ( char * ) input0_end - ( char * ) input0 ) > = ( 16 * stbir__simdfX_float_count ) )
{
stbir__simdfX o0 , o1 , o2 , o3 , r0 , r1 , r2 , r3 ;
STBIR_SIMD_NO_UNROLL ( output ) ;
// prefetch four loop iterations ahead (doesn't affect much for small resizes, but helps with big ones)
stbIF0 ( stbir__prefetch ( input0 + ( 16 * stbir__simdfX_float_count ) ) ; )
stbIF1 ( stbir__prefetch ( input1 + ( 16 * stbir__simdfX_float_count ) ) ; )
stbIF2 ( stbir__prefetch ( input2 + ( 16 * stbir__simdfX_float_count ) ) ; )
stbIF3 ( stbir__prefetch ( input3 + ( 16 * stbir__simdfX_float_count ) ) ; )
stbIF4 ( stbir__prefetch ( input4 + ( 16 * stbir__simdfX_float_count ) ) ; )
stbIF5 ( stbir__prefetch ( input5 + ( 16 * stbir__simdfX_float_count ) ) ; )
stbIF6 ( stbir__prefetch ( input6 + ( 16 * stbir__simdfX_float_count ) ) ; )
stbIF7 ( stbir__prefetch ( input7 + ( 16 * stbir__simdfX_float_count ) ) ; )
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( stbir__simdfX_load ( o0 , output ) ; stbir__simdfX_load ( o1 , output + stbir__simdfX_float_count ) ; stbir__simdfX_load ( o2 , output + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( o3 , output + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_load ( r0 , input0 ) ; stbir__simdfX_load ( r1 , input0 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input0 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input0 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c0 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c0 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c0 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c0 ) ; )
# else
stbIF0 ( stbir__simdfX_load ( r0 , input0 ) ; stbir__simdfX_load ( r1 , input0 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input0 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input0 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_mult ( o0 , r0 , c0 ) ; stbir__simdfX_mult ( o1 , r1 , c0 ) ; stbir__simdfX_mult ( o2 , r2 , c0 ) ; stbir__simdfX_mult ( o3 , r3 , c0 ) ; )
# endif
stbIF1 ( stbir__simdfX_load ( r0 , input1 ) ; stbir__simdfX_load ( r1 , input1 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input1 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input1 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c1 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c1 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c1 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c1 ) ; )
stbIF2 ( stbir__simdfX_load ( r0 , input2 ) ; stbir__simdfX_load ( r1 , input2 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input2 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input2 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c2 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c2 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c2 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c2 ) ; )
stbIF3 ( stbir__simdfX_load ( r0 , input3 ) ; stbir__simdfX_load ( r1 , input3 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input3 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input3 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c3 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c3 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c3 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c3 ) ; )
stbIF4 ( stbir__simdfX_load ( r0 , input4 ) ; stbir__simdfX_load ( r1 , input4 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input4 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input4 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c4 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c4 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c4 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c4 ) ; )
stbIF5 ( stbir__simdfX_load ( r0 , input5 ) ; stbir__simdfX_load ( r1 , input5 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input5 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input5 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c5 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c5 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c5 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c5 ) ; )
stbIF6 ( stbir__simdfX_load ( r0 , input6 ) ; stbir__simdfX_load ( r1 , input6 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input6 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input6 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c6 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c6 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c6 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c6 ) ; )
stbIF7 ( stbir__simdfX_load ( r0 , input7 ) ; stbir__simdfX_load ( r1 , input7 + stbir__simdfX_float_count ) ; stbir__simdfX_load ( r2 , input7 + ( 2 * stbir__simdfX_float_count ) ) ; stbir__simdfX_load ( r3 , input7 + ( 3 * stbir__simdfX_float_count ) ) ;
stbir__simdfX_madd ( o0 , o0 , r0 , c7 ) ; stbir__simdfX_madd ( o1 , o1 , r1 , c7 ) ; stbir__simdfX_madd ( o2 , o2 , r2 , c7 ) ; stbir__simdfX_madd ( o3 , o3 , r3 , c7 ) ; )
stbir__simdfX_store ( output , o0 ) ; stbir__simdfX_store ( output + stbir__simdfX_float_count , o1 ) ; stbir__simdfX_store ( output + ( 2 * stbir__simdfX_float_count ) , o2 ) ; stbir__simdfX_store ( output + ( 3 * stbir__simdfX_float_count ) , o3 ) ;
output + = ( 4 * stbir__simdfX_float_count ) ;
stbIF0 ( input0 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF1 ( input1 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF2 ( input2 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF3 ( input3 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF4 ( input4 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF5 ( input5 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF6 ( input6 + = ( 4 * stbir__simdfX_float_count ) ; ) stbIF7 ( input7 + = ( 4 * stbir__simdfX_float_count ) ; )
}
while ( ( ( char * ) input0_end - ( char * ) input0 ) > = 16 )
{
stbir__simdf o0 , r0 ;
STBIR_SIMD_NO_UNROLL ( output ) ;
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( stbir__simdf_load ( o0 , output ) ; stbir__simdf_load ( r0 , input0 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c0 ) ) ; )
# else
stbIF0 ( stbir__simdf_load ( r0 , input0 ) ; stbir__simdf_mult ( o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c0 ) ) ; )
# endif
stbIF1 ( stbir__simdf_load ( r0 , input1 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c1 ) ) ; )
stbIF2 ( stbir__simdf_load ( r0 , input2 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c2 ) ) ; )
stbIF3 ( stbir__simdf_load ( r0 , input3 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c3 ) ) ; )
stbIF4 ( stbir__simdf_load ( r0 , input4 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c4 ) ) ; )
stbIF5 ( stbir__simdf_load ( r0 , input5 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c5 ) ) ; )
stbIF6 ( stbir__simdf_load ( r0 , input6 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c6 ) ) ; )
stbIF7 ( stbir__simdf_load ( r0 , input7 ) ; stbir__simdf_madd ( o0 , o0 , r0 , stbir__if_simdf8_cast_to_simdf4 ( c7 ) ) ; )
stbir__simdf_store ( output , o0 ) ;
output + = 4 ;
stbIF0 ( input0 + = 4 ; ) stbIF1 ( input1 + = 4 ; ) stbIF2 ( input2 + = 4 ; ) stbIF3 ( input3 + = 4 ; ) stbIF4 ( input4 + = 4 ; ) stbIF5 ( input5 + = 4 ; ) stbIF6 ( input6 + = 4 ; ) stbIF7 ( input7 + = 4 ; )
}
}
# else
while ( ( ( char * ) input0_end - ( char * ) input0 ) > = 16 )
{
float o0 , o1 , o2 , o3 ;
STBIR_NO_UNROLL ( output ) ;
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( o0 = output [ 0 ] + input0 [ 0 ] * c0s ; o1 = output [ 1 ] + input0 [ 1 ] * c0s ; o2 = output [ 2 ] + input0 [ 2 ] * c0s ; o3 = output [ 3 ] + input0 [ 3 ] * c0s ; )
# else
stbIF0 ( o0 = input0 [ 0 ] * c0s ; o1 = input0 [ 1 ] * c0s ; o2 = input0 [ 2 ] * c0s ; o3 = input0 [ 3 ] * c0s ; )
# endif
stbIF1 ( o0 + = input1 [ 0 ] * c1s ; o1 + = input1 [ 1 ] * c1s ; o2 + = input1 [ 2 ] * c1s ; o3 + = input1 [ 3 ] * c1s ; )
stbIF2 ( o0 + = input2 [ 0 ] * c2s ; o1 + = input2 [ 1 ] * c2s ; o2 + = input2 [ 2 ] * c2s ; o3 + = input2 [ 3 ] * c2s ; )
stbIF3 ( o0 + = input3 [ 0 ] * c3s ; o1 + = input3 [ 1 ] * c3s ; o2 + = input3 [ 2 ] * c3s ; o3 + = input3 [ 3 ] * c3s ; )
stbIF4 ( o0 + = input4 [ 0 ] * c4s ; o1 + = input4 [ 1 ] * c4s ; o2 + = input4 [ 2 ] * c4s ; o3 + = input4 [ 3 ] * c4s ; )
stbIF5 ( o0 + = input5 [ 0 ] * c5s ; o1 + = input5 [ 1 ] * c5s ; o2 + = input5 [ 2 ] * c5s ; o3 + = input5 [ 3 ] * c5s ; )
stbIF6 ( o0 + = input6 [ 0 ] * c6s ; o1 + = input6 [ 1 ] * c6s ; o2 + = input6 [ 2 ] * c6s ; o3 + = input6 [ 3 ] * c6s ; )
stbIF7 ( o0 + = input7 [ 0 ] * c7s ; o1 + = input7 [ 1 ] * c7s ; o2 + = input7 [ 2 ] * c7s ; o3 + = input7 [ 3 ] * c7s ; )
output [ 0 ] = o0 ; output [ 1 ] = o1 ; output [ 2 ] = o2 ; output [ 3 ] = o3 ;
output + = 4 ;
stbIF0 ( input0 + = 4 ; ) stbIF1 ( input1 + = 4 ; ) stbIF2 ( input2 + = 4 ; ) stbIF3 ( input3 + = 4 ; ) stbIF4 ( input4 + = 4 ; ) stbIF5 ( input5 + = 4 ; ) stbIF6 ( input6 + = 4 ; ) stbIF7 ( input7 + = 4 ; )
}
# endif
while ( input0 < input0_end )
{
float o0 ;
STBIR_NO_UNROLL ( output ) ;
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
stbIF0 ( o0 = output [ 0 ] + input0 [ 0 ] * c0s ; )
# else
stbIF0 ( o0 = input0 [ 0 ] * c0s ; )
# endif
stbIF1 ( o0 + = input1 [ 0 ] * c1s ; )
stbIF2 ( o0 + = input2 [ 0 ] * c2s ; )
stbIF3 ( o0 + = input3 [ 0 ] * c3s ; )
stbIF4 ( o0 + = input4 [ 0 ] * c4s ; )
stbIF5 ( o0 + = input5 [ 0 ] * c5s ; )
stbIF6 ( o0 + = input6 [ 0 ] * c6s ; )
stbIF7 ( o0 + = input7 [ 0 ] * c7s ; )
output [ 0 ] = o0 ;
+ + output ;
stbIF0 ( + + input0 ; ) stbIF1 ( + + input1 ; ) stbIF2 ( + + input2 ; ) stbIF3 ( + + input3 ; ) stbIF4 ( + + input4 ; ) stbIF5 ( + + input5 ; ) stbIF6 ( + + input6 ; ) stbIF7 ( + + input7 ; )
}
}
# undef stbIF0
# undef stbIF1
# undef stbIF2
# undef stbIF3
# undef stbIF4
# undef stbIF5
# undef stbIF6
# undef stbIF7
# undef STB_IMAGE_RESIZE_DO_VERTICALS
# undef STBIR__vertical_channels
# undef STB_IMAGE_RESIZE_DO_HORIZONTALS
# undef STBIR_strs_join24
# undef STBIR_strs_join14
# undef STBIR_chans
# ifdef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# undef STB_IMAGE_RESIZE_VERTICAL_CONTINUE
# endif
# else // !STB_IMAGE_RESIZE_DO_VERTICALS
# define STBIR_chans( start, end ) STBIR_strs_join1(start,STBIR__horizontal_channels,end)
# ifndef stbir__2_coeff_only
# define stbir__2_coeff_only() \
stbir__1_coeff_only ( ) ; \
stbir__1_coeff_remnant ( 1 ) ;
# endif
# ifndef stbir__2_coeff_remnant
# define stbir__2_coeff_remnant( ofs ) \
stbir__1_coeff_remnant ( ofs ) ; \
stbir__1_coeff_remnant ( ( ofs ) + 1 ) ;
# endif
# ifndef stbir__3_coeff_only
# define stbir__3_coeff_only() \
stbir__2_coeff_only ( ) ; \
stbir__1_coeff_remnant ( 2 ) ;
# endif
# ifndef stbir__3_coeff_remnant
# define stbir__3_coeff_remnant( ofs ) \
stbir__2_coeff_remnant ( ofs ) ; \
stbir__1_coeff_remnant ( ( ofs ) + 2 ) ;
# endif
# ifndef stbir__3_coeff_setup
# define stbir__3_coeff_setup()
# endif
# ifndef stbir__4_coeff_start
# define stbir__4_coeff_start() \
stbir__2_coeff_only ( ) ; \
stbir__2_coeff_remnant ( 2 ) ;
# endif
# ifndef stbir__4_coeff_continue_from_4
# define stbir__4_coeff_continue_from_4( ofs ) \
stbir__2_coeff_remnant ( ofs ) ; \
stbir__2_coeff_remnant ( ( ofs ) + 2 ) ;
# endif
# ifndef stbir__store_output_tiny
# define stbir__store_output_tiny stbir__store_output
# endif
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_1_coeff ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__1_coeff_only ( ) ;
stbir__store_output_tiny ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_2_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__2_coeff_only ( ) ;
stbir__store_output_tiny ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_3_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__3_coeff_only ( ) ;
stbir__store_output_tiny ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_4_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_5_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__1_coeff_remnant ( 4 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_6_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__2_coeff_remnant ( 4 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_7_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
stbir__3_coeff_setup ( ) ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__3_coeff_remnant ( 4 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_8_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__4_coeff_continue_from_4 ( 4 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_9_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__4_coeff_continue_from_4 ( 4 ) ;
stbir__1_coeff_remnant ( 8 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_10_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__4_coeff_continue_from_4 ( 4 ) ;
stbir__2_coeff_remnant ( 8 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_11_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
stbir__3_coeff_setup ( ) ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__4_coeff_continue_from_4 ( 4 ) ;
stbir__3_coeff_remnant ( 8 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_12_coeffs ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
stbir__4_coeff_continue_from_4 ( 4 ) ;
stbir__4_coeff_continue_from_4 ( 8 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod0 ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
int n = ( ( horizontal_contributors - > n1 - horizontal_contributors - > n0 + 1 ) - 4 + 3 ) > > 2 ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
do {
hc + = 4 ;
decode + = STBIR__horizontal_channels * 4 ;
stbir__4_coeff_continue_from_4 ( 0 ) ;
- - n ;
} while ( n > 0 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod1 ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
int n = ( ( horizontal_contributors - > n1 - horizontal_contributors - > n0 + 1 ) - 5 + 3 ) > > 2 ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
do {
hc + = 4 ;
decode + = STBIR__horizontal_channels * 4 ;
stbir__4_coeff_continue_from_4 ( 0 ) ;
- - n ;
} while ( n > 0 ) ;
stbir__1_coeff_remnant ( 4 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod2 ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
int n = ( ( horizontal_contributors - > n1 - horizontal_contributors - > n0 + 1 ) - 6 + 3 ) > > 2 ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
do {
hc + = 4 ;
decode + = STBIR__horizontal_channels * 4 ;
stbir__4_coeff_continue_from_4 ( 0 ) ;
- - n ;
} while ( n > 0 ) ;
stbir__2_coeff_remnant ( 4 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static void STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod3 ) ( float * output_buffer , unsigned int output_sub_size , float const * decode_buffer , stbir__contributors const * horizontal_contributors , float const * horizontal_coefficients , int coefficient_width )
{
float const * output_end = output_buffer + output_sub_size * STBIR__horizontal_channels ;
float STBIR_SIMD_STREAMOUT_PTR ( * ) output = output_buffer ;
stbir__3_coeff_setup ( ) ;
do {
float const * decode = decode_buffer + horizontal_contributors - > n0 * STBIR__horizontal_channels ;
int n = ( ( horizontal_contributors - > n1 - horizontal_contributors - > n0 + 1 ) - 7 + 3 ) > > 2 ;
float const * hc = horizontal_coefficients ;
stbir__4_coeff_start ( ) ;
do {
hc + = 4 ;
decode + = STBIR__horizontal_channels * 4 ;
stbir__4_coeff_continue_from_4 ( 0 ) ;
- - n ;
} while ( n > 0 ) ;
stbir__3_coeff_remnant ( 4 ) ;
stbir__store_output ( ) ;
} while ( output < output_end ) ;
}
static stbir__horizontal_gather_channels_func * STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_funcs ) [ 4 ] =
{
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod0 ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod1 ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod2 ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_n_coeffs_mod3 ) ,
} ;
static stbir__horizontal_gather_channels_func * STBIR_chans ( stbir__horizontal_gather_ , _channels_funcs ) [ 12 ] =
{
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_1_coeff ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_2_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_3_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_4_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_5_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_6_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_7_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_8_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_9_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_10_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_11_coeffs ) ,
STBIR_chans ( stbir__horizontal_gather_ , _channels_with_12_coeffs ) ,
} ;
# undef STBIR__horizontal_channels
# undef STB_IMAGE_RESIZE_DO_HORIZONTALS
# undef stbir__1_coeff_only
# undef stbir__1_coeff_remnant
# undef stbir__2_coeff_only
# undef stbir__2_coeff_remnant
# undef stbir__3_coeff_only
# undef stbir__3_coeff_remnant
# undef stbir__3_coeff_setup
# undef stbir__4_coeff_start
# undef stbir__4_coeff_continue_from_4
# undef stbir__store_output
# undef stbir__store_output_tiny
# undef STBIR_chans
# endif // HORIZONALS
# undef STBIR_strs_join2
# undef STBIR_strs_join1
# endif // STB_IMAGE_RESIZE_DO_HORIZONTALS/VERTICALS/CODERS
/*
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
This software is available under 2 licenses - - choose whichever you prefer .
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
ALTERNATIVE A - MIT License
Copyright ( c ) 2017 Sean Barrett
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 .
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
ALTERNATIVE B - Public Domain ( www . unlicense . org )
This is free and unencumbered software released into the public domain .
Anyone is free to copy , modify , publish , use , compile , sell , or distribute this
software , either in source code form or as a compiled binary , for any purpose ,
commercial or non - commercial , and by any means .
In jurisdictions that recognize copyright laws , the author or authors of this
software dedicate any and all copyright interest in the software to the public
domain . We make this dedication for the benefit of the public at large and to
the detriment of our heirs and successors . We intend this dedication to be an
overt act of relinquishment in perpetuity of all present and future rights to
this software under copyright law .
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 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 .
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*/