Pillow/libImaging/Antialias.c

304 lines
7.8 KiB
C
Raw Normal View History

2010-07-31 06:52:47 +04:00
/*
* The Python Imaging Library
* $Id$
*
* pilopen antialiasing support
2010-07-31 06:52:47 +04:00
*
* history:
* 2002-03-09 fl Created (for PIL 1.1.3)
* 2002-03-10 fl Added support for mode "F"
*
* Copyright (c) 1997-2002 by Secret Labs AB
*
* See the README file for information on usage and redistribution.
*/
#include "Imaging.h"
#include <math.h>
/* resampling filters (from antialias.py) */
struct filter {
float (*filter)(float x);
float support;
};
static inline float sinc_filter(float x)
{
if (x == 0.0)
return 1.0;
x = x * M_PI;
return sin(x) / x;
}
static inline float antialias_filter(float x)
{
/* lanczos (truncated sinc) */
if (-3.0 <= x && x < 3.0)
return sinc_filter(x) * sinc_filter(x/3);
return 0.0;
}
static struct filter ANTIALIAS = { antialias_filter, 3.0 };
static inline float nearest_filter(float x)
{
if (-0.5 <= x && x < 0.5)
return 1.0;
return 0.0;
}
static struct filter NEAREST = { nearest_filter, 0.5 };
static inline float bilinear_filter(float x)
{
if (x < 0.0)
x = -x;
if (x < 1.0)
return 1.0-x;
return 0.0;
}
static struct filter BILINEAR = { bilinear_filter, 1.0 };
static inline float bicubic_filter(float x)
{
2014-10-24 12:46:51 +04:00
/* http://en.wikipedia.org/wiki/Bicubic_interpolation#Bicubic_convolution_algorithm */
2014-10-25 12:39:03 +04:00
#define a -0.5
2010-07-31 06:52:47 +04:00
if (x < 0.0)
x = -x;
if (x < 1.0)
2014-10-24 12:46:51 +04:00
return ((a + 2.0) * x - (a + 3.0)) * x*x + 1;
2010-07-31 06:52:47 +04:00
if (x < 2.0)
2014-10-24 12:46:51 +04:00
return (((x - 5) * x + 8) * x - 4) * a;
2010-07-31 06:52:47 +04:00
return 0.0;
#undef a
}
static struct filter BICUBIC = { bicubic_filter, 2.0 };
2014-10-25 06:04:04 +04:00
static inline UINT8 clip8(float in)
{
int out = (int) in;
if (out >= 255)
return 255;
if (out <= 0)
return 0;
return (UINT8) out;
}
2010-07-31 06:52:47 +04:00
Imaging
2014-10-26 00:11:39 +04:00
ImagingStretchHorizaontal(Imaging imIn, int xsize, int filter)
2010-07-31 06:52:47 +04:00
{
/* FIXME: this is a quick and straightforward translation from a
python prototype. might need some further C-ification... */
ImagingSectionCookie cookie;
2014-10-26 00:11:39 +04:00
Imaging imOut;
2010-07-31 06:52:47 +04:00
struct filter *filterp;
float support, scale, filterscale;
2014-10-25 05:57:08 +04:00
float center, ww, ss;
int xx, yy, x, b, kmax, xmin, xmax;
int *xbounds;
float *k, *kk;
2010-07-31 06:52:47 +04:00
/* check filter */
switch (filter) {
case IMAGING_TRANSFORM_NEAREST:
filterp = &NEAREST;
break;
case IMAGING_TRANSFORM_ANTIALIAS:
filterp = &ANTIALIAS;
break;
case IMAGING_TRANSFORM_BILINEAR:
filterp = &BILINEAR;
break;
case IMAGING_TRANSFORM_BICUBIC:
filterp = &BICUBIC;
break;
default:
return (Imaging) ImagingError_ValueError(
"unsupported resampling filter"
);
}
/* prepare for horizontal stretch */
2014-10-26 00:11:39 +04:00
filterscale = scale = (float) imIn->xsize / xsize;
2010-07-31 06:52:47 +04:00
/* determine support size (length of resampling filter) */
support = filterp->support;
if (filterscale < 1.0) {
filterscale = 1.0;
}
2010-07-31 06:52:47 +04:00
support = support * filterscale;
2014-10-25 05:37:33 +04:00
/* maximum number of coofs */
kmax = (int) ceil(support) * 2 + 1;
2010-07-31 06:52:47 +04:00
/* coefficient buffer (with rounding safety margin) */
2014-10-26 00:11:39 +04:00
kk = malloc(xsize * kmax * sizeof(float));
2014-10-25 05:37:33 +04:00
if ( ! kk)
return (Imaging) ImagingError_MemoryError();
2014-10-26 00:11:39 +04:00
xbounds = malloc(xsize * 2 * sizeof(int));
2014-10-25 05:37:33 +04:00
if ( ! xbounds) {
free(kk);
2010-07-31 06:52:47 +04:00
return (Imaging) ImagingError_MemoryError();
2014-10-25 05:37:33 +04:00
}
2014-10-26 00:11:39 +04:00
for (xx = 0; xx < xsize; xx++) {
2014-10-25 05:37:33 +04:00
k = &kk[xx * kmax];
center = (xx + 0.5) * scale;
ww = 0.0;
ss = 1.0 / filterscale;
2014-10-25 05:57:08 +04:00
xmin = (int) floor(center - support);
if (xmin < 0)
xmin = 0;
xmax = (int) ceil(center + support);
if (xmax > imIn->xsize)
xmax = imIn->xsize;
for (x = xmin; x < xmax; x++) {
2014-10-25 05:37:33 +04:00
float w = filterp->filter((x - center + 0.5) * ss) * ss;
2014-10-25 05:57:08 +04:00
k[x - xmin] = w;
2014-10-25 05:50:54 +04:00
ww += w;
2014-10-25 05:37:33 +04:00
}
2014-10-25 05:57:08 +04:00
for (x = 0; x < xmax - xmin; x++) {
2014-10-25 05:50:54 +04:00
if (ww != 0.0)
k[x] /= ww;
}
xbounds[xx * 2 + 0] = xmin;
xbounds[xx * 2 + 1] = xmax;
2014-10-25 05:37:33 +04:00
}
2010-07-31 06:52:47 +04:00
2014-10-26 00:11:39 +04:00
imOut = ImagingNew(imIn->mode, xsize, imIn->ysize);
if ( ! imOut) {
free(kk);
free(xbounds);
return NULL;
}
2010-07-31 06:52:47 +04:00
ImagingSectionEnter(&cookie);
/* horizontal stretch */
2014-10-25 05:41:38 +04:00
for (yy = 0; yy < imOut->ysize; yy++) {
if (imIn->image8) {
/* 8-bit grayscale */
2014-10-26 00:11:39 +04:00
for (xx = 0; xx < xsize; xx++) {
2014-10-25 05:50:54 +04:00
xmin = xbounds[xx * 2 + 0];
xmax = xbounds[xx * 2 + 1];
2014-10-25 05:41:38 +04:00
k = &kk[xx * kmax];
2014-10-25 05:50:54 +04:00
ss = 0.5;
2014-10-25 05:57:08 +04:00
for (x = xmin; x < xmax; x++)
ss = ss + imIn->image8[yy][x] * k[x - xmin];
2014-10-25 06:04:04 +04:00
imOut->image8[yy][xx] = clip8(ss);
2010-07-31 06:52:47 +04:00
}
} else
switch(imIn->type) {
case IMAGING_TYPE_UINT8:
/* n-bit grayscale */
2014-10-26 00:11:39 +04:00
for (xx = 0; xx < xsize; xx++) {
2014-10-25 05:50:54 +04:00
xmin = xbounds[xx * 2 + 0];
xmax = xbounds[xx * 2 + 1];
2014-10-25 05:41:38 +04:00
k = &kk[xx * kmax];
for (b = 0; b < imIn->bands; b++) {
if (imIn->bands == 2 && b)
b = 3; /* hack to deal with LA images */
2014-10-25 05:50:54 +04:00
ss = 0.5;
2014-10-25 05:57:08 +04:00
for (x = xmin; x < xmax; x++)
ss = ss + (UINT8) imIn->image[yy][x*4+b] * k[x - xmin];
2014-10-25 06:04:04 +04:00
imOut->image[yy][xx*4+b] = clip8(ss);
2010-07-31 06:52:47 +04:00
}
}
break;
case IMAGING_TYPE_INT32:
/* 32-bit integer */
2014-10-26 00:11:39 +04:00
for (xx = 0; xx < xsize; xx++) {
2014-10-25 05:50:54 +04:00
xmin = xbounds[xx * 2 + 0];
xmax = xbounds[xx * 2 + 1];
2014-10-25 05:41:38 +04:00
k = &kk[xx * kmax];
2010-07-31 06:52:47 +04:00
ss = 0.0;
2014-10-25 05:57:08 +04:00
for (x = xmin; x < xmax; x++)
ss = ss + IMAGING_PIXEL_I(imIn, x, yy) * k[x - xmin];
2014-10-25 05:50:54 +04:00
IMAGING_PIXEL_I(imOut, xx, yy) = (int) ss;
2010-07-31 06:52:47 +04:00
}
break;
case IMAGING_TYPE_FLOAT32:
/* 32-bit float */
2014-10-26 00:11:39 +04:00
for (xx = 0; xx < xsize; xx++) {
2014-10-25 05:50:54 +04:00
xmin = xbounds[xx * 2 + 0];
xmax = xbounds[xx * 2 + 1];
2014-10-25 05:41:38 +04:00
k = &kk[xx * kmax];
ss = 0.0;
2014-10-25 05:57:08 +04:00
for (x = xmin; x < xmax; x++)
ss = ss + IMAGING_PIXEL_F(imIn, x, yy) * k[x - xmin];
2014-10-25 05:50:54 +04:00
IMAGING_PIXEL_F(imOut, xx, yy) = ss;
2010-07-31 06:52:47 +04:00
}
break;
default:
ImagingSectionLeave(&cookie);
return (Imaging) ImagingError_ModeError();
}
2010-07-31 06:52:47 +04:00
}
ImagingSectionLeave(&cookie);
2014-10-25 05:37:33 +04:00
free(kk);
free(xbounds);
2010-07-31 06:52:47 +04:00
return imOut;
}
Imaging
2014-10-26 00:11:39 +04:00
ImagingTransposeToNew(Imaging imIn)
{
Imaging imTemp = ImagingNew(imIn->mode, imIn->ysize, imIn->xsize);
if ( ! imTemp)
return NULL;
if ( ! ImagingTranspose(imTemp, imIn)) {
ImagingDelete(imTemp);
return NULL;
}
return imTemp;
}
Imaging
ImagingStretch(Imaging imIn, int xsize, int ysize, int filter)
{
Imaging imTemp1, imTemp2, imTemp3;
2014-10-26 00:11:39 +04:00
Imaging imOut;
if (strcmp(imIn->mode, "P") == 0 || strcmp(imIn->mode, "1") == 0)
return (Imaging) ImagingError_ModeError();
2014-10-26 00:11:39 +04:00
/* two-pass resize, first pass */
imTemp1 = ImagingStretchHorizaontal(imIn, xsize, filter);
if ( ! imTemp1)
return NULL;
/* transpose image once */
2014-10-26 00:11:39 +04:00
imTemp2 = ImagingTransposeToNew(imTemp1);
ImagingDelete(imTemp1);
2014-10-26 00:11:39 +04:00
if ( ! imTemp2)
return NULL;
/* second pass */
2014-10-26 00:11:39 +04:00
imTemp3 = ImagingStretchHorizaontal(imTemp2, ysize, filter);
ImagingDelete(imTemp2);
2014-10-26 00:11:39 +04:00
if ( ! imTemp3)
return NULL;
/* transpose result */
2014-10-26 00:11:39 +04:00
imOut = ImagingTransposeToNew(imTemp3);
ImagingDelete(imTemp3);
2014-10-26 00:11:39 +04:00
if ( ! imOut)
return NULL;
return imOut;
}