174 lines
4.8 KiB
C
174 lines
4.8 KiB
C
|
// File: crn_resampler.h
|
||
|
// RG: This is public domain code, originally derived from Graphics Gems 3, see: http://code.google.com/p/imageresampler/
|
||
|
#pragma once
|
||
|
|
||
|
namespace crnlib
|
||
|
{
|
||
|
#define CRNLIB_RESAMPLER_DEBUG_OPS 0
|
||
|
#define CRNLIB_RESAMPLER_DEFAULT_FILTER "lanczos4"
|
||
|
|
||
|
#define CRNLIB_RESAMPLER_MAX_DIMENSION 16384
|
||
|
|
||
|
// float or double
|
||
|
typedef float Resample_Real;
|
||
|
|
||
|
class Resampler
|
||
|
{
|
||
|
public:
|
||
|
typedef Resample_Real Sample;
|
||
|
|
||
|
struct Contrib
|
||
|
{
|
||
|
Resample_Real weight;
|
||
|
unsigned short pixel;
|
||
|
};
|
||
|
|
||
|
struct Contrib_List
|
||
|
{
|
||
|
unsigned short n;
|
||
|
Contrib* p;
|
||
|
};
|
||
|
|
||
|
enum Boundary_Op
|
||
|
{
|
||
|
BOUNDARY_WRAP = 0,
|
||
|
BOUNDARY_REFLECT = 1,
|
||
|
BOUNDARY_CLAMP = 2
|
||
|
};
|
||
|
|
||
|
enum Status
|
||
|
{
|
||
|
STATUS_OKAY = 0,
|
||
|
STATUS_OUT_OF_MEMORY = 1,
|
||
|
STATUS_BAD_FILTER_NAME = 2,
|
||
|
STATUS_SCAN_BUFFER_FULL = 3
|
||
|
};
|
||
|
|
||
|
// src_x/src_y - Input dimensions
|
||
|
// dst_x/dst_y - Output dimensions
|
||
|
// boundary_op - How to sample pixels near the image boundaries
|
||
|
// sample_low/sample_high - Clamp output samples to specified range, or disable clamping if sample_low >= sample_high
|
||
|
// Pclist_x/Pclist_y - Optional pointers to contributor lists from another instance of a Resampler
|
||
|
// src_x_ofs/src_y_ofs - Offset input image by specified amount (fractional values okay)
|
||
|
Resampler(
|
||
|
int src_x, int src_y,
|
||
|
int dst_x, int dst_y,
|
||
|
Boundary_Op boundary_op = BOUNDARY_CLAMP,
|
||
|
Resample_Real sample_low = 0.0f, Resample_Real sample_high = 0.0f,
|
||
|
const char* Pfilter_name = CRNLIB_RESAMPLER_DEFAULT_FILTER,
|
||
|
Contrib_List* Pclist_x = NULL,
|
||
|
Contrib_List* Pclist_y = NULL,
|
||
|
Resample_Real filter_x_scale = 1.0f,
|
||
|
Resample_Real filter_y_scale = 1.0f,
|
||
|
Resample_Real src_x_ofs = 0.0f,
|
||
|
Resample_Real src_y_ofs = 0.0f);
|
||
|
|
||
|
~Resampler();
|
||
|
|
||
|
// Reinits resampler so it can handle another frame.
|
||
|
void restart();
|
||
|
|
||
|
// false on out of memory.
|
||
|
bool put_line(const Sample* Psrc);
|
||
|
|
||
|
// NULL if no scanlines are currently available (give the resampler more scanlines!)
|
||
|
const Sample* get_line();
|
||
|
|
||
|
Status status() const { return m_status; }
|
||
|
|
||
|
// Returned contributor lists can be shared with another Resampler.
|
||
|
void get_clists(Contrib_List** ptr_clist_x, Contrib_List** ptr_clist_y);
|
||
|
Contrib_List* get_clist_x() const { return m_Pclist_x; }
|
||
|
Contrib_List* get_clist_y() const { return m_Pclist_y; }
|
||
|
|
||
|
// Filter accessors.
|
||
|
static int get_filter_num();
|
||
|
static const char* get_filter_name(int filter_num);
|
||
|
|
||
|
static Contrib_List* make_clist(
|
||
|
int src_x, int dst_x, Boundary_Op boundary_op,
|
||
|
Resample_Real (*Pfilter)(Resample_Real),
|
||
|
Resample_Real filter_support,
|
||
|
Resample_Real filter_scale,
|
||
|
Resample_Real src_ofs);
|
||
|
|
||
|
private:
|
||
|
Resampler();
|
||
|
Resampler(const Resampler& o);
|
||
|
Resampler& operator= (const Resampler& o);
|
||
|
|
||
|
#ifdef CRNLIB_RESAMPLER_DEBUG_OPS
|
||
|
int total_ops;
|
||
|
#endif
|
||
|
|
||
|
int m_intermediate_x;
|
||
|
|
||
|
int m_resample_src_x;
|
||
|
int m_resample_src_y;
|
||
|
int m_resample_dst_x;
|
||
|
int m_resample_dst_y;
|
||
|
|
||
|
Boundary_Op m_boundary_op;
|
||
|
|
||
|
Sample* m_Pdst_buf;
|
||
|
Sample* m_Ptmp_buf;
|
||
|
|
||
|
Contrib_List* m_Pclist_x;
|
||
|
Contrib_List* m_Pclist_y;
|
||
|
|
||
|
bool m_clist_x_forced;
|
||
|
bool m_clist_y_forced;
|
||
|
|
||
|
bool m_delay_x_resample;
|
||
|
|
||
|
int* m_Psrc_y_count;
|
||
|
unsigned char* m_Psrc_y_flag;
|
||
|
|
||
|
// The maximum number of scanlines that can be buffered at one time.
|
||
|
enum { MAX_SCAN_BUF_SIZE = CRNLIB_RESAMPLER_MAX_DIMENSION };
|
||
|
|
||
|
struct Scan_Buf
|
||
|
{
|
||
|
int scan_buf_y[MAX_SCAN_BUF_SIZE];
|
||
|
Sample* scan_buf_l[MAX_SCAN_BUF_SIZE];
|
||
|
};
|
||
|
|
||
|
Scan_Buf* m_Pscan_buf;
|
||
|
|
||
|
int m_cur_src_y;
|
||
|
int m_cur_dst_y;
|
||
|
|
||
|
Status m_status;
|
||
|
|
||
|
void resample_x(Sample* Pdst, const Sample* Psrc);
|
||
|
void scale_y_mov(Sample* Ptmp, const Sample* Psrc, Resample_Real weight, int dst_x);
|
||
|
void scale_y_add(Sample* Ptmp, const Sample* Psrc, Resample_Real weight, int dst_x);
|
||
|
void clamp(Sample* Pdst, int n);
|
||
|
void resample_y(Sample* Pdst);
|
||
|
|
||
|
static int reflect(const int j, const int src_x, const Boundary_Op boundary_op);
|
||
|
|
||
|
inline int count_ops(Contrib_List* Pclist, int k)
|
||
|
{
|
||
|
int i, t = 0;
|
||
|
for (i = 0; i < k; i++)
|
||
|
t += Pclist[i].n;
|
||
|
return (t);
|
||
|
}
|
||
|
|
||
|
Resample_Real m_lo;
|
||
|
Resample_Real m_hi;
|
||
|
|
||
|
inline Resample_Real clamp_sample(Resample_Real f) const
|
||
|
{
|
||
|
if (f < m_lo)
|
||
|
f = m_lo;
|
||
|
else if (f > m_hi)
|
||
|
f = m_hi;
|
||
|
return f;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
} // namespace crnlib
|
||
|
|