cpython/Modules/imgfile.c

549 lines
12 KiB
C
Raw Normal View History

1992-03-04 12:40:03 -04:00
/***********************************************************
2000-06-30 20:50:40 -03:00
Copyright (c) 2000, BeOpen.com.
Copyright (c) 1995-2000, Corporation for National Research Initiatives.
Copyright (c) 1990-1995, Stichting Mathematisch Centrum.
All rights reserved.
1992-03-04 12:40:03 -04:00
2000-06-30 20:50:40 -03:00
See the file "Misc/COPYRIGHT" for information on usage and
redistribution of this file, and for a DISCLAIMER OF ALL WARRANTIES.
1992-03-04 12:40:03 -04:00
******************************************************************/
/* IMGFILE module - Interface to sgi libimage */
1992-03-04 12:40:03 -04:00
1992-06-03 14:06:36 -03:00
/* XXX This modele should be done better at some point. It should return
1992-03-04 12:40:03 -04:00
** an object of image file class, and have routines to manipulate these
** image files in a neater way (so you can get rgb images off a greyscale
** file, for instance, or do a straight display without having to get the
** image bits into python, etc).
**
** Warning: this module is very non-reentrant (esp. the readscaled stuff)
1992-03-04 12:40:03 -04:00
*/
1996-12-20 17:56:08 -04:00
#include "Python.h"
1992-03-04 12:40:03 -04:00
#include <gl/image.h>
1993-01-19 11:33:13 -04:00
#include "/usr/people/4Dgifts/iristools/include/izoom.h"
/* Bunch of missing extern decls; keep gcc -Wall happy... */
extern void i_seterror();
extern void iclose();
extern void filterzoom();
extern void putrow();
extern void getrow();
1996-12-20 17:56:08 -04:00
static PyObject * ImgfileError; /* Exception we raise for various trouble */
1992-03-04 12:40:03 -04:00
static int top_to_bottom; /* True if we want top-to-bottom images */
1992-03-04 12:40:03 -04:00
/* The image library does not always call the error hander :-(,
therefore we have a global variable indicating that it was called.
It is cleared by imgfile_open(). */
1992-03-04 12:40:03 -04:00
1992-03-23 14:21:32 -04:00
static int error_called;
/* The error handler */
1996-12-05 19:43:35 -04:00
static void
imgfile_error(str)
1996-12-20 17:56:08 -04:00
char *str;
1992-03-23 14:21:32 -04:00
{
1996-12-20 17:56:08 -04:00
PyErr_SetString(ImgfileError, str);
error_called = 1;
return; /* To imglib, which will return a failure indicator */
1992-03-23 14:21:32 -04:00
}
1992-03-04 12:40:03 -04:00
/* Open an image file and return a pointer to it.
Make sure we raise an exception if we fail. */
static IMAGE *
1992-06-03 14:06:36 -03:00
imgfile_open(fname)
1996-12-20 17:56:08 -04:00
char *fname;
1992-06-03 14:06:36 -03:00
{
1996-12-20 17:56:08 -04:00
IMAGE *image;
i_seterror(imgfile_error);
error_called = 0;
errno = 0;
if ( (image = iopen(fname, "r")) == NULL ) {
/* Error may already be set by imgfile_error */
if ( !error_called ) {
if (errno)
PyErr_SetFromErrno(ImgfileError);
else
PyErr_SetString(ImgfileError,
"Can't open image file");
}
return NULL;
}
return image;
1992-03-04 12:40:03 -04:00
}
1996-12-20 17:56:08 -04:00
static PyObject *
imgfile_ttob(self, args)
1996-12-20 17:56:08 -04:00
PyObject *self;
PyObject *args;
{
1996-12-20 17:56:08 -04:00
int newval;
PyObject *rv;
1996-12-20 17:56:08 -04:00
if (!PyArg_Parse(args, "i", &newval))
return NULL;
rv = PyInt_FromLong(top_to_bottom);
top_to_bottom = newval;
return rv;
}
1996-12-20 17:56:08 -04:00
static PyObject *
1992-03-04 12:40:03 -04:00
imgfile_read(self, args)
1996-12-20 17:56:08 -04:00
PyObject *self;
PyObject *args;
1992-03-04 12:40:03 -04:00
{
1996-12-20 17:56:08 -04:00
char *fname;
PyObject *rv;
int xsize, ysize, zsize;
char *cdatap;
long *idatap;
static short rs[8192], gs[8192], bs[8192];
int x, y;
IMAGE *image;
int yfirst, ylast, ystep;
if ( !PyArg_Parse(args, "s", &fname) )
return NULL;
1992-06-03 14:06:36 -03:00
1996-12-20 17:56:08 -04:00
if ( (image = imgfile_open(fname)) == NULL )
return NULL;
1992-03-04 12:40:03 -04:00
1996-12-20 17:56:08 -04:00
if ( image->colormap != CM_NORMAL ) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can only handle CM_NORMAL images");
return NULL;
}
if ( BPP(image->type) != 1 ) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can't handle imgfiles with bpp!=1");
return NULL;
}
xsize = image->xsize;
ysize = image->ysize;
zsize = image->zsize;
if ( zsize != 1 && zsize != 3) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can only handle 1 or 3 byte pixels");
return NULL;
}
if ( xsize > 8192 ) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can't handle image with > 8192 columns");
return NULL;
}
if ( zsize == 3 ) zsize = 4;
rv = PyString_FromStringAndSize((char *)NULL, xsize*ysize*zsize);
if ( rv == NULL ) {
iclose(image);
return NULL;
}
cdatap = PyString_AsString(rv);
idatap = (long *)cdatap;
if (top_to_bottom) {
yfirst = ysize-1;
ylast = -1;
ystep = -1;
1992-03-04 12:40:03 -04:00
} else {
1996-12-20 17:56:08 -04:00
yfirst = 0;
ylast = ysize;
ystep = 1;
}
for ( y=yfirst; y != ylast && !error_called; y += ystep ) {
if ( zsize == 1 ) {
getrow(image, rs, y, 0);
for(x=0; x<xsize; x++ )
*cdatap++ = rs[x];
} else {
getrow(image, rs, y, 0);
getrow(image, gs, y, 1);
getrow(image, bs, y, 2);
for(x=0; x<xsize; x++ )
*idatap++ = (rs[x] & 0xff) |
((gs[x] & 0xff)<<8) |
((bs[x] & 0xff)<<16);
}
1992-03-04 12:40:03 -04:00
}
1996-12-20 17:56:08 -04:00
iclose(image);
if ( error_called ) {
Py_DECREF(rv);
return NULL;
}
return rv;
1992-03-04 12:40:03 -04:00
}
static IMAGE *glob_image;
static long *glob_datap;
static int glob_width, glob_z, glob_ysize;
1996-12-05 19:43:35 -04:00
static void
xs_get(buf, y)
1996-12-20 17:56:08 -04:00
short *buf;
int y;
{
1996-12-20 17:56:08 -04:00
if (top_to_bottom)
getrow(glob_image, buf, (glob_ysize-1-y), glob_z);
else
getrow(glob_image, buf, y, glob_z);
}
1996-12-05 19:43:35 -04:00
static void
xs_put_c(buf, y)
1996-12-20 17:56:08 -04:00
short *buf;
int y;
{
1996-12-20 17:56:08 -04:00
char *datap = (char *)glob_datap + y*glob_width;
int width = glob_width;
1996-12-20 17:56:08 -04:00
while ( width-- )
*datap++ = (*buf++) & 0xff;
}
1996-12-05 19:43:35 -04:00
static void
xs_put_0(buf, y)
1996-12-20 17:56:08 -04:00
short *buf;
int y;
{
1996-12-20 17:56:08 -04:00
long *datap = glob_datap + y*glob_width;
int width = glob_width;
1996-12-20 17:56:08 -04:00
while ( width-- )
*datap++ = (*buf++) & 0xff;
}
1996-12-05 19:43:35 -04:00
static void
xs_put_12(buf, y)
1996-12-20 17:56:08 -04:00
short *buf;
int y;
{
1996-12-20 17:56:08 -04:00
long *datap = glob_datap + y*glob_width;
int width = glob_width;
1996-12-20 17:56:08 -04:00
while ( width-- )
*datap++ |= ((*buf++) & 0xff) << (glob_z*8);
}
static void
xscale(image, xsize, ysize, zsize, datap, xnew, ynew, fmode, blur)
1996-12-20 17:56:08 -04:00
IMAGE *image;
int xsize, ysize, zsize;
long *datap;
int xnew, ynew;
int fmode;
double blur;
{
1996-12-20 17:56:08 -04:00
glob_image = image;
glob_datap = datap;
glob_width = xnew;
glob_ysize = ysize;
if ( zsize == 1 ) {
glob_z = 0;
filterzoom(xs_get, xs_put_c, xsize, ysize,
xnew, ynew, fmode, blur);
} else {
glob_z = 0;
filterzoom(xs_get, xs_put_0, xsize, ysize,
xnew, ynew, fmode, blur);
glob_z = 1;
filterzoom(xs_get, xs_put_12, xsize, ysize,
xnew, ynew, fmode, blur);
glob_z = 2;
filterzoom(xs_get, xs_put_12, xsize, ysize,
xnew, ynew, fmode, blur);
}
}
1996-12-20 17:56:08 -04:00
static PyObject *
1992-06-03 14:06:36 -03:00
imgfile_readscaled(self, args)
1996-12-20 17:56:08 -04:00
PyObject *self;
PyObject *args;
1992-06-03 14:06:36 -03:00
{
1996-12-20 17:56:08 -04:00
char *fname;
PyObject *rv;
int xsize, ysize, zsize;
char *cdatap;
long *idatap;
static short rs[8192], gs[8192], bs[8192];
int x, y;
int xwtd, ywtd, xorig, yorig;
float xfac, yfac;
int cnt;
IMAGE *image;
char *filter;
double blur;
int extended;
int fmode = 0;
int yfirst, ylast, ystep;
/*
** Parse args. Funny, since arg 4 and 5 are optional
** (filter name and blur factor). Also, 4 or 5 arguments indicates
** extended scale algorithm in stead of simple-minded pixel drop/dup.
*/
extended = 0;
cnt = PyTuple_Size(args);
if ( cnt == 5 ) {
extended = 1;
if ( !PyArg_Parse(args, "(siisd)",
&fname, &xwtd, &ywtd, &filter, &blur) )
return NULL;
} else if ( cnt == 4 ) {
extended = 1;
if ( !PyArg_Parse(args, "(siis)",
&fname, &xwtd, &ywtd, &filter) )
return NULL;
blur = 1.0;
} else if ( !PyArg_Parse(args, "(sii)", &fname, &xwtd, &ywtd) )
return NULL;
/*
** Check parameters, open file and check type, rows, etc.
*/
if ( extended ) {
if ( strcmp(filter, "impulse") == 0 )
fmode = IMPULSE;
else if ( strcmp( filter, "box") == 0 )
fmode = BOX;
else if ( strcmp( filter, "triangle") == 0 )
fmode = TRIANGLE;
else if ( strcmp( filter, "quadratic") == 0 )
fmode = QUADRATIC;
else if ( strcmp( filter, "gaussian") == 0 )
fmode = GAUSSIAN;
else {
PyErr_SetString(ImgfileError, "Unknown filter type");
return NULL;
}
}
1996-12-20 17:56:08 -04:00
if ( (image = imgfile_open(fname)) == NULL )
return NULL;
1992-06-03 14:06:36 -03:00
1996-12-20 17:56:08 -04:00
if ( image->colormap != CM_NORMAL ) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can only handle CM_NORMAL images");
return NULL;
}
if ( BPP(image->type) != 1 ) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can't handle imgfiles with bpp!=1");
return NULL;
}
xsize = image->xsize;
ysize = image->ysize;
zsize = image->zsize;
if ( zsize != 1 && zsize != 3) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can only handle 1 or 3 byte pixels");
return NULL;
}
if ( xsize > 8192 ) {
iclose(image);
PyErr_SetString(ImgfileError,
"Can't handle image with > 8192 columns");
return NULL;
}
1992-06-03 14:06:36 -03:00
1996-12-20 17:56:08 -04:00
if ( zsize == 3 ) zsize = 4;
rv = PyString_FromStringAndSize(NULL, xwtd*ywtd*zsize);
if ( rv == NULL ) {
iclose(image);
return NULL;
}
PyFPE_START_PROTECT("readscaled", return 0)
1996-12-20 17:56:08 -04:00
xfac = (float)xsize/(float)xwtd;
yfac = (float)ysize/(float)ywtd;
1997-03-14 00:32:50 -04:00
PyFPE_END_PROTECT(yfac)
1996-12-20 17:56:08 -04:00
cdatap = PyString_AsString(rv);
idatap = (long *)cdatap;
if ( extended ) {
xscale(image, xsize, ysize, zsize,
idatap, xwtd, ywtd, fmode, blur);
} else {
if (top_to_bottom) {
yfirst = ywtd-1;
ylast = -1;
ystep = -1;
} else {
yfirst = 0;
ylast = ywtd;
ystep = 1;
}
1996-12-20 17:56:08 -04:00
for ( y=yfirst; y != ylast && !error_called; y += ystep ) {
yorig = (int)(y*yfac);
if ( zsize == 1 ) {
getrow(image, rs, yorig, 0);
for(x=0; x<xwtd; x++ ) {
*cdatap++ = rs[(int)(x*xfac)];
}
} else {
getrow(image, rs, yorig, 0);
getrow(image, gs, yorig, 1);
getrow(image, bs, yorig, 2);
for(x=0; x<xwtd; x++ ) {
xorig = (int)(x*xfac);
*idatap++ = (rs[xorig] & 0xff) |
((gs[xorig] & 0xff)<<8) |
((bs[xorig] & 0xff)<<16);
}
}
}
1992-06-03 14:06:36 -03:00
}
1996-12-20 17:56:08 -04:00
iclose(image);
if ( error_called ) {
Py_DECREF(rv);
return NULL;
}
return rv;
1992-06-03 14:06:36 -03:00
}
1996-12-20 17:56:08 -04:00
static PyObject *
1992-03-04 12:40:03 -04:00
imgfile_getsizes(self, args)
1996-12-20 17:56:08 -04:00
PyObject *self;
PyObject *args;
1992-03-04 12:40:03 -04:00
{
1996-12-20 17:56:08 -04:00
char *fname;
PyObject *rv;
IMAGE *image;
1992-03-04 12:40:03 -04:00
1996-12-20 17:56:08 -04:00
if ( !PyArg_Parse(args, "s", &fname) )
return NULL;
1996-12-20 17:56:08 -04:00
if ( (image = imgfile_open(fname)) == NULL )
return NULL;
rv = Py_BuildValue("(iii)", image->xsize, image->ysize, image->zsize);
iclose(image);
return rv;
1992-03-04 12:40:03 -04:00
}
1996-12-20 17:56:08 -04:00
static PyObject *
1992-08-20 08:54:27 -03:00
imgfile_write(self, args)
1996-12-20 17:56:08 -04:00
PyObject *self;
PyObject *args;
1992-08-20 08:54:27 -03:00
{
1996-12-20 17:56:08 -04:00
IMAGE *image;
char *fname;
int xsize, ysize, zsize, len;
char *cdatap;
long *idatap;
short rs[8192], gs[8192], bs[8192];
short r, g, b;
long rgb;
int x, y;
int yfirst, ylast, ystep;
if ( !PyArg_Parse(args, "(ss#iii)",
&fname, &cdatap, &len, &xsize, &ysize, &zsize) )
return NULL;
1992-08-20 08:54:27 -03:00
1996-12-20 17:56:08 -04:00
if ( zsize != 1 && zsize != 3 ) {
PyErr_SetString(ImgfileError,
"Can only handle 1 or 3 byte pixels");
return NULL;
}
if ( len != xsize * ysize * (zsize == 1 ? 1 : 4) ) {
PyErr_SetString(ImgfileError, "Data does not match sizes");
return NULL;
}
if ( xsize > 8192 ) {
PyErr_SetString(ImgfileError,
"Can't handle image with > 8192 columns");
return NULL;
}
error_called = 0;
errno = 0;
image =iopen(fname, "w", RLE(1), 3, xsize, ysize, zsize);
if ( image == 0 ) {
if ( ! error_called ) {
if (errno)
PyErr_SetFromErrno(ImgfileError);
else
PyErr_SetString(ImgfileError,
"Can't create image file");
}
return NULL;
}
1992-08-20 08:54:27 -03:00
1996-12-20 17:56:08 -04:00
idatap = (long *)cdatap;
1992-08-20 08:54:27 -03:00
1996-12-20 17:56:08 -04:00
if (top_to_bottom) {
yfirst = ysize-1;
ylast = -1;
ystep = -1;
1992-08-20 08:54:27 -03:00
} else {
1996-12-20 17:56:08 -04:00
yfirst = 0;
ylast = ysize;
ystep = 1;
1992-08-20 08:54:27 -03:00
}
1996-12-20 17:56:08 -04:00
for ( y=yfirst; y != ylast && !error_called; y += ystep ) {
if ( zsize == 1 ) {
for( x=0; x<xsize; x++ )
rs[x] = *cdatap++;
putrow(image, rs, y, 0);
} else {
for( x=0; x<xsize; x++ ) {
rgb = *idatap++;
r = rgb & 0xff;
g = (rgb >> 8 ) & 0xff;
b = (rgb >> 16 ) & 0xff;
rs[x] = r;
gs[x] = g;
bs[x] = b;
}
putrow(image, rs, y, 0);
putrow(image, gs, y, 1);
putrow(image, bs, y, 2);
}
}
iclose(image);
if ( error_called )
return NULL;
Py_INCREF(Py_None);
return Py_None;
1992-08-20 08:54:27 -03:00
}
1992-03-04 12:40:03 -04:00
1996-12-20 17:56:08 -04:00
static PyMethodDef imgfile_methods[] = {
{ "getsizes", imgfile_getsizes },
{ "read", imgfile_read },
{ "readscaled", imgfile_readscaled, 1},
{ "write", imgfile_write },
{ "ttob", imgfile_ttob },
{ NULL, NULL } /* Sentinel */
1992-03-04 12:40:03 -04:00
};
void
initimgfile()
{
1996-12-20 17:56:08 -04:00
PyObject *m, *d;
m = Py_InitModule("imgfile", imgfile_methods);
d = PyModule_GetDict(m);
ImgfileError = PyErr_NewException("imgfile.error", NULL, NULL);
if (ImgfileError != NULL)
PyDict_SetItemString(d, "error", ImgfileError);
1992-03-04 12:40:03 -04:00
}
1996-12-20 17:56:08 -04:00