• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 
2 /* imageopmodule - Various operations on pictures */
3 
4 #ifdef sun
5 #define signed
6 #endif
7 
8 #include "Python.h"
9 
10 #if SIZEOF_INT == 4
11 typedef int Py_Int32;
12 typedef unsigned int Py_UInt32;
13 #else
14 #if SIZEOF_LONG == 4
15 typedef long Py_Int32;
16 typedef unsigned long Py_UInt32;
17 #else
18 #error "No 4-byte integral type"
19 #endif
20 #endif
21 
22 #define CHARP(cp, xmax, x, y) ((char *)(cp+y*xmax+x))
23 #define SHORTP(cp, xmax, x, y) ((short *)(cp+2*(y*xmax+x)))
24 #define LONGP(cp, xmax, x, y) ((Py_Int32 *)(cp+4*(y*xmax+x)))
25 
26 static PyObject *ImageopError;
27 static PyObject *ImageopDict;
28 
29 /**
30  * Check a coordonnate, make sure that (0 < value).
31  * Return 0 on error.
32  */
33 static int
check_coordonnate(int value,const char * name)34 check_coordonnate(int value, const char* name)
35 {
36     if ( 0 < value)
37         return 1;
38     PyErr_Format(PyExc_ValueError, "%s value is negative or nul", name);
39     return 0;
40 }
41 
42 /**
43  * Check integer overflow to make sure that product == x*y*size.
44  * Return 0 on error.
45  */
46 static int
check_multiply_size(int product,int x,const char * xname,int y,const char * yname,int size)47 check_multiply_size(int product, int x, const char* xname, int y, const char* yname, int size)
48 {
49     if ( !check_coordonnate(x, xname) )
50         return 0;
51     if ( !check_coordonnate(y, yname) )
52         return 0;
53     if ( size == (product / y) / x )
54         return 1;
55     PyErr_SetString(ImageopError, "String has incorrect length");
56     return 0;
57 }
58 
59 /**
60  * Check integer overflow to make sure that product == x*y.
61  * Return 0 on error.
62  */
63 static int
check_multiply(int product,int x,int y)64 check_multiply(int product, int x, int y)
65 {
66     return check_multiply_size(product, x, "x", y, "y", 1);
67 }
68 
69 /* If this function returns true (the default if anything goes wrong), we're
70    behaving in a backward-compatible way with respect to how multi-byte pixels
71    are stored in the strings.  The code in this module was originally written
72    for an SGI which is a big-endian system, and so the old code assumed that
73    4-byte integers hold the R, G, and B values in a particular order.
74    However, on little-endian systems the order is reversed, and so not
75    actually compatible with what gl.lrectwrite and imgfile expect.
76    (gl.lrectwrite and imgfile are also SGI-specific, however, it is
77    conceivable that the data handled here comes from or goes to an SGI or that
78    it is otherwise used in the expectation that the byte order in the strings
79    is as specified.)
80 
81    The function returns the value of the module variable
82    "backward_compatible", or 1 if the variable does not exist or is not an
83    int.
84  */
85 
86 static int
imageop_backward_compatible(void)87 imageop_backward_compatible(void)
88 {
89     static PyObject *bcos;
90     PyObject *bco;
91     long rc;
92 
93     if (ImageopDict == NULL) /* "cannot happen" */
94         return 1;
95     if (bcos == NULL) {
96         /* cache string object for future use */
97         bcos = PyString_FromString("backward_compatible");
98         if (bcos == NULL)
99             return 1;
100     }
101     bco = PyDict_GetItem(ImageopDict, bcos);
102     if (bco == NULL)
103         return 1;
104     if (!PyInt_Check(bco))
105         return 1;
106     rc = PyInt_AsLong(bco);
107     if (PyErr_Occurred()) {
108         /* not an integer, or too large, or something */
109         PyErr_Clear();
110         rc = 1;
111     }
112     return rc != 0;             /* convert to values 0, 1 */
113 }
114 
115 static PyObject *
imageop_crop(PyObject * self,PyObject * args)116 imageop_crop(PyObject *self, PyObject *args)
117 {
118     char *cp, *ncp;
119     short *nsp;
120     Py_Int32 *nlp;
121     int len, size, x, y, newx1, newx2, newy1, newy2, nlen;
122     int ix, iy, xstep, ystep;
123     PyObject *rv;
124 
125     if ( !PyArg_ParseTuple(args, "s#iiiiiii", &cp, &len, &size, &x, &y,
126                       &newx1, &newy1, &newx2, &newy2) )
127         return 0;
128 
129     if ( size != 1 && size != 2 && size != 4 ) {
130         PyErr_SetString(ImageopError, "Size should be 1, 2 or 4");
131         return 0;
132     }
133     if ( !check_multiply_size(len, x, "x", y, "y", size) )
134         return 0;
135 
136     xstep = (newx1 < newx2)? 1 : -1;
137     ystep = (newy1 < newy2)? 1 : -1;
138 
139     nlen = (abs(newx2-newx1)+1)*(abs(newy2-newy1)+1)*size;
140     if ( !check_multiply_size(nlen, abs(newx2-newx1)+1, "abs(newx2-newx1)+1", abs(newy2-newy1)+1, "abs(newy2-newy1)+1", size) )
141         return 0;
142     rv = PyString_FromStringAndSize(NULL, nlen);
143     if ( rv == 0 )
144         return 0;
145     ncp = (char *)PyString_AsString(rv);
146     nsp = (short *)ncp;
147     nlp = (Py_Int32 *)ncp;
148     newy2 += ystep;
149     newx2 += xstep;
150     for( iy = newy1; iy != newy2; iy+=ystep ) {
151         for ( ix = newx1; ix != newx2; ix+=xstep ) {
152             if ( iy < 0 || iy >= y || ix < 0 || ix >= x ) {
153                 if ( size == 1 )
154                     *ncp++ = 0;
155                 else
156                     *nlp++ = 0;
157             } else {
158                 if ( size == 1 )
159                     *ncp++ = *CHARP(cp, x, ix, iy);
160                 else if ( size == 2 )
161                     *nsp++ = *SHORTP(cp, x, ix, iy);
162                 else
163                     *nlp++ = *LONGP(cp, x, ix, iy);
164             }
165         }
166     }
167     return rv;
168 }
169 
170 static PyObject *
imageop_scale(PyObject * self,PyObject * args)171 imageop_scale(PyObject *self, PyObject *args)
172 {
173     char *cp, *ncp;
174     short *nsp;
175     Py_Int32 *nlp;
176     int len, size, x, y, newx, newy, nlen;
177     int ix, iy;
178     int oix, oiy;
179     PyObject *rv;
180 
181     if ( !PyArg_ParseTuple(args, "s#iiiii",
182                       &cp, &len, &size, &x, &y, &newx, &newy) )
183         return 0;
184 
185     if ( size != 1 && size != 2 && size != 4 ) {
186         PyErr_SetString(ImageopError, "Size should be 1, 2 or 4");
187         return 0;
188     }
189     if ( !check_multiply_size(len, x, "x", y, "y", size) )
190         return 0;
191     nlen = newx*newy*size;
192     if ( !check_multiply_size(nlen, newx, "newx", newy, "newy", size) )
193         return 0;
194 
195     rv = PyString_FromStringAndSize(NULL, nlen);
196     if ( rv == 0 )
197         return 0;
198     ncp = (char *)PyString_AsString(rv);
199     nsp = (short *)ncp;
200     nlp = (Py_Int32 *)ncp;
201     for( iy = 0; iy < newy; iy++ ) {
202         for ( ix = 0; ix < newx; ix++ ) {
203             oix = ix * x / newx;
204             oiy = iy * y / newy;
205             if ( size == 1 )
206                 *ncp++ = *CHARP(cp, x, oix, oiy);
207             else if ( size == 2 )
208                 *nsp++ = *SHORTP(cp, x, oix, oiy);
209             else
210                 *nlp++ = *LONGP(cp, x, oix, oiy);
211         }
212     }
213     return rv;
214 }
215 
216 /* Note: this routine can use a bit of optimizing */
217 
218 static PyObject *
imageop_tovideo(PyObject * self,PyObject * args)219 imageop_tovideo(PyObject *self, PyObject *args)
220 {
221     int maxx, maxy, x, y, len;
222     int i;
223     unsigned char *cp, *ncp;
224     int width;
225     PyObject *rv;
226 
227 
228     if ( !PyArg_ParseTuple(args, "s#iii", &cp, &len, &width, &maxx, &maxy) )
229         return 0;
230 
231     if ( width != 1 && width != 4 ) {
232         PyErr_SetString(ImageopError, "Size should be 1 or 4");
233         return 0;
234     }
235     if ( !check_multiply_size(len, maxx, "max", maxy, "maxy", width) )
236         return 0;
237 
238     rv = PyString_FromStringAndSize(NULL, len);
239     if ( rv == 0 )
240         return 0;
241     ncp = (unsigned char *)PyString_AsString(rv);
242 
243     if ( width == 1 ) {
244         memcpy(ncp, cp, maxx);                  /* Copy first line */
245         ncp += maxx;
246         for (y=1; y<maxy; y++) {                /* Interpolate other lines */
247             for(x=0; x<maxx; x++) {
248                 i = y*maxx + x;
249                 *ncp++ = ((int)cp[i] + (int)cp[i-maxx]) >> 1;
250             }
251         }
252     } else {
253         memcpy(ncp, cp, maxx*4);                        /* Copy first line */
254         ncp += maxx*4;
255         for (y=1; y<maxy; y++) {                /* Interpolate other lines */
256             for(x=0; x<maxx; x++) {
257                 i = (y*maxx + x)*4 + 1;
258                 *ncp++ = 0;                     /* Skip alfa comp */
259                 *ncp++ = ((int)cp[i] + (int)cp[i-4*maxx]) >> 1;
260                 i++;
261                 *ncp++ = ((int)cp[i] + (int)cp[i-4*maxx]) >> 1;
262                 i++;
263                 *ncp++ = ((int)cp[i] + (int)cp[i-4*maxx]) >> 1;
264             }
265         }
266     }
267     return rv;
268 }
269 
270 static PyObject *
imageop_grey2mono(PyObject * self,PyObject * args)271 imageop_grey2mono(PyObject *self, PyObject *args)
272 {
273     int tres, x, y, len;
274     unsigned char *cp, *ncp;
275     unsigned char ovalue;
276     PyObject *rv;
277     int i, bit;
278 
279 
280     if ( !PyArg_ParseTuple(args, "s#iii", &cp, &len, &x, &y, &tres) )
281         return 0;
282 
283     if ( !check_multiply(len, x, y) )
284         return 0;
285 
286     rv = PyString_FromStringAndSize(NULL, (len+7)/8);
287     if ( rv == 0 )
288         return 0;
289     ncp = (unsigned char *)PyString_AsString(rv);
290 
291     bit = 0x80;
292     ovalue = 0;
293     for ( i=0; i < len; i++ ) {
294         if ( (int)cp[i] > tres )
295             ovalue |= bit;
296         bit >>= 1;
297         if ( bit == 0 ) {
298             *ncp++ = ovalue;
299             bit = 0x80;
300             ovalue = 0;
301         }
302     }
303     if ( bit != 0x80 )
304         *ncp++ = ovalue;
305     return rv;
306 }
307 
308 static PyObject *
imageop_grey2grey4(PyObject * self,PyObject * args)309 imageop_grey2grey4(PyObject *self, PyObject *args)
310 {
311     int x, y, len;
312     unsigned char *cp, *ncp;
313     unsigned char ovalue;
314     PyObject *rv;
315     int i;
316     int pos;
317 
318 
319     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
320         return 0;
321 
322     if ( !check_multiply(len, x, y) )
323         return 0;
324 
325     rv = PyString_FromStringAndSize(NULL, (len+1)/2);
326     if ( rv == 0 )
327         return 0;
328     ncp = (unsigned char *)PyString_AsString(rv);
329     pos = 0;
330     ovalue = 0;
331     for ( i=0; i < len; i++ ) {
332         ovalue |= ((int)cp[i] & 0xf0) >> pos;
333         pos += 4;
334         if ( pos == 8 ) {
335             *ncp++ = ovalue;
336             ovalue = 0;
337             pos = 0;
338         }
339     }
340     if ( pos != 0 )
341         *ncp++ = ovalue;
342     return rv;
343 }
344 
345 static PyObject *
imageop_grey2grey2(PyObject * self,PyObject * args)346 imageop_grey2grey2(PyObject *self, PyObject *args)
347 {
348     int x, y, len;
349     unsigned char *cp, *ncp;
350     unsigned char ovalue;
351     PyObject *rv;
352     int i;
353     int pos;
354 
355 
356     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
357         return 0;
358 
359     if ( !check_multiply(len, x, y) )
360         return 0;
361 
362     rv = PyString_FromStringAndSize(NULL, (len+3)/4);
363     if ( rv == 0 )
364         return 0;
365     ncp = (unsigned char *)PyString_AsString(rv);
366     pos = 0;
367     ovalue = 0;
368     for ( i=0; i < len; i++ ) {
369         ovalue |= ((int)cp[i] & 0xc0) >> pos;
370         pos += 2;
371         if ( pos == 8 ) {
372             *ncp++ = ovalue;
373             ovalue = 0;
374             pos = 0;
375         }
376     }
377     if ( pos != 0 )
378         *ncp++ = ovalue;
379     return rv;
380 }
381 
382 static PyObject *
imageop_dither2mono(PyObject * self,PyObject * args)383 imageop_dither2mono(PyObject *self, PyObject *args)
384 {
385     int sum, x, y, len;
386     unsigned char *cp, *ncp;
387     unsigned char ovalue;
388     PyObject *rv;
389     int i, bit;
390 
391 
392     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
393         return 0;
394 
395     if ( !check_multiply(len, x, y) )
396         return 0;
397 
398     rv = PyString_FromStringAndSize(NULL, (len+7)/8);
399     if ( rv == 0 )
400         return 0;
401     ncp = (unsigned char *)PyString_AsString(rv);
402 
403     bit = 0x80;
404     ovalue = 0;
405     sum = 0;
406     for ( i=0; i < len; i++ ) {
407         sum += cp[i];
408         if ( sum >= 256 ) {
409             sum -= 256;
410             ovalue |= bit;
411         }
412         bit >>= 1;
413         if ( bit == 0 ) {
414             *ncp++ = ovalue;
415             bit = 0x80;
416             ovalue = 0;
417         }
418     }
419     if ( bit != 0x80 )
420         *ncp++ = ovalue;
421     return rv;
422 }
423 
424 static PyObject *
imageop_dither2grey2(PyObject * self,PyObject * args)425 imageop_dither2grey2(PyObject *self, PyObject *args)
426 {
427     int x, y, len;
428     unsigned char *cp, *ncp;
429     unsigned char ovalue;
430     PyObject *rv;
431     int i;
432     int pos;
433     int sum = 0, nvalue;
434 
435 
436     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
437         return 0;
438 
439     if ( !check_multiply(len, x, y) )
440         return 0;
441 
442     rv = PyString_FromStringAndSize(NULL, (len+3)/4);
443     if ( rv == 0 )
444         return 0;
445     ncp = (unsigned char *)PyString_AsString(rv);
446     pos = 1;
447     ovalue = 0;
448     for ( i=0; i < len; i++ ) {
449         sum += cp[i];
450         nvalue = sum & 0x180;
451         sum -= nvalue;
452         ovalue |= nvalue >> pos;
453         pos += 2;
454         if ( pos == 9 ) {
455             *ncp++ = ovalue;
456             ovalue = 0;
457             pos = 1;
458         }
459     }
460     if ( pos != 0 )
461         *ncp++ = ovalue;
462     return rv;
463 }
464 
465 static PyObject *
imageop_mono2grey(PyObject * self,PyObject * args)466 imageop_mono2grey(PyObject *self, PyObject *args)
467 {
468     int v0, v1, x, y, len, nlen;
469     unsigned char *cp, *ncp;
470     PyObject *rv;
471     int i, bit;
472 
473     if ( !PyArg_ParseTuple(args, "s#iiii", &cp, &len, &x, &y, &v0, &v1) )
474         return 0;
475 
476     nlen = x*y;
477     if ( !check_multiply(nlen, x, y) )
478         return 0;
479     if ( (nlen+7)/8 != len ) {
480         PyErr_SetString(ImageopError, "String has incorrect length");
481         return 0;
482     }
483 
484     rv = PyString_FromStringAndSize(NULL, nlen);
485     if ( rv == 0 )
486         return 0;
487     ncp = (unsigned char *)PyString_AsString(rv);
488 
489     bit = 0x80;
490     for ( i=0; i < nlen; i++ ) {
491         if ( *cp & bit )
492             *ncp++ = v1;
493         else
494             *ncp++ = v0;
495         bit >>= 1;
496         if ( bit == 0 ) {
497             bit = 0x80;
498             cp++;
499         }
500     }
501     return rv;
502 }
503 
504 static PyObject *
imageop_grey22grey(PyObject * self,PyObject * args)505 imageop_grey22grey(PyObject *self, PyObject *args)
506 {
507     int x, y, len, nlen;
508     unsigned char *cp, *ncp;
509     PyObject *rv;
510     int i, pos, value = 0, nvalue;
511 
512     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
513         return 0;
514 
515     nlen = x*y;
516     if ( !check_multiply(nlen, x, y) ) {
517         return 0;
518     }
519     if ( (nlen+3)/4 != len ) {
520         PyErr_SetString(ImageopError, "String has incorrect length");
521         return 0;
522     }
523 
524     rv = PyString_FromStringAndSize(NULL, nlen);
525     if ( rv == 0 )
526         return 0;
527     ncp = (unsigned char *)PyString_AsString(rv);
528 
529     pos = 0;
530     for ( i=0; i < nlen; i++ ) {
531         if ( pos == 0 ) {
532             value = *cp++;
533             pos = 8;
534         }
535         pos -= 2;
536         nvalue = (value >> pos) & 0x03;
537         *ncp++ = nvalue | (nvalue << 2) |
538                  (nvalue << 4) | (nvalue << 6);
539     }
540     return rv;
541 }
542 
543 static PyObject *
imageop_grey42grey(PyObject * self,PyObject * args)544 imageop_grey42grey(PyObject *self, PyObject *args)
545 {
546     int x, y, len, nlen;
547     unsigned char *cp, *ncp;
548     PyObject *rv;
549     int i, pos, value = 0, nvalue;
550 
551     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
552         return 0;
553 
554     nlen = x*y;
555     if ( !check_multiply(nlen, x, y) )
556         return 0;
557     if ( (nlen+1)/2 != len ) {
558         PyErr_SetString(ImageopError, "String has incorrect length");
559         return 0;
560     }
561 
562     rv = PyString_FromStringAndSize(NULL, nlen);
563     if ( rv == 0 )
564         return 0;
565     ncp = (unsigned char *)PyString_AsString(rv);
566 
567     pos = 0;
568     for ( i=0; i < nlen; i++ ) {
569         if ( pos == 0 ) {
570             value = *cp++;
571             pos = 8;
572         }
573         pos -= 4;
574         nvalue = (value >> pos) & 0x0f;
575         *ncp++ = nvalue | (nvalue << 4);
576     }
577     return rv;
578 }
579 
580 static PyObject *
imageop_rgb2rgb8(PyObject * self,PyObject * args)581 imageop_rgb2rgb8(PyObject *self, PyObject *args)
582 {
583     int x, y, len, nlen;
584     unsigned char *cp;
585     unsigned char *ncp;
586     PyObject *rv;
587     int i, r, g, b;
588     int backward_compatible = imageop_backward_compatible();
589 
590     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
591         return 0;
592 
593     if ( !check_multiply_size(len, x, "x", y, "y", 4) )
594         return 0;
595     nlen = x*y;
596     if ( !check_multiply(nlen, x, y) )
597         return 0;
598 
599     rv = PyString_FromStringAndSize(NULL, nlen);
600     if ( rv == 0 )
601         return 0;
602     ncp = (unsigned char *)PyString_AsString(rv);
603 
604     for ( i=0; i < nlen; i++ ) {
605         /* Bits in source: aaaaaaaa BBbbbbbb GGGggggg RRRrrrrr */
606         if (backward_compatible) {
607             Py_UInt32 value = * (Py_UInt32 *) cp;
608             cp += 4;
609             r = (int) ((value & 0xff) / 255. * 7. + .5);
610             g = (int) (((value >> 8) & 0xff) / 255. * 7. + .5);
611             b = (int) (((value >> 16) & 0xff) / 255. * 3. + .5);
612         } else {
613             cp++;                       /* skip alpha channel */
614             b = (int) (*cp++ / 255. * 3. + .5);
615             g = (int) (*cp++ / 255. * 7. + .5);
616             r = (int) (*cp++ / 255. * 7. + .5);
617         }
618         *ncp++ = (unsigned char)((r<<5) | (b<<3) | g);
619     }
620     return rv;
621 }
622 
623 static PyObject *
imageop_rgb82rgb(PyObject * self,PyObject * args)624 imageop_rgb82rgb(PyObject *self, PyObject *args)
625 {
626     int x, y, len, nlen;
627     unsigned char *cp;
628     unsigned char *ncp;
629     PyObject *rv;
630     int i, r, g, b;
631     unsigned char value;
632     int backward_compatible = imageop_backward_compatible();
633 
634     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
635         return 0;
636 
637     if ( !check_multiply(len, x, y) )
638         return 0;
639     nlen = x*y*4;
640     if ( !check_multiply_size(nlen, x, "x", y, "y", 4) )
641         return 0;
642 
643     rv = PyString_FromStringAndSize(NULL, nlen);
644     if ( rv == 0 )
645         return 0;
646     ncp = (unsigned char *)PyString_AsString(rv);
647 
648     for ( i=0; i < len; i++ ) {
649         /* Bits in source: RRRBBGGG
650         ** Red and Green are multiplied by 36.5, Blue by 85
651         */
652         value = *cp++;
653         r = (value >> 5) & 7;
654         g = (value     ) & 7;
655         b = (value >> 3) & 3;
656         r = (r<<5) | (r<<3) | (r>>1);
657         g = (g<<5) | (g<<3) | (g>>1);
658         b = (b<<6) | (b<<4) | (b<<2) | b;
659         if (backward_compatible) {
660             Py_UInt32 nvalue = r | (g<<8) | (b<<16);
661             * (Py_UInt32 *) ncp = nvalue;
662             ncp += 4;
663         } else {
664             *ncp++ = 0;
665             *ncp++ = b;
666             *ncp++ = g;
667             *ncp++ = r;
668         }
669     }
670     return rv;
671 }
672 
673 static PyObject *
imageop_rgb2grey(PyObject * self,PyObject * args)674 imageop_rgb2grey(PyObject *self, PyObject *args)
675 {
676     int x, y, len, nlen;
677     unsigned char *cp;
678     unsigned char *ncp;
679     PyObject *rv;
680     int i, r, g, b;
681     int nvalue;
682     int backward_compatible = imageop_backward_compatible();
683 
684     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
685         return 0;
686 
687     if ( !check_multiply_size(len, x, "x", y, "y", 4) )
688         return 0;
689     nlen = x*y;
690     if ( !check_multiply(nlen, x, y) )
691         return 0;
692 
693     rv = PyString_FromStringAndSize(NULL, nlen);
694     if ( rv == 0 )
695         return 0;
696     ncp = (unsigned char *)PyString_AsString(rv);
697 
698     for ( i=0; i < nlen; i++ ) {
699         if (backward_compatible) {
700             Py_UInt32 value = * (Py_UInt32 *) cp;
701             cp += 4;
702             r = (int) ((value & 0xff) / 255. * 7. + .5);
703             g = (int) (((value >> 8) & 0xff) / 255. * 7. + .5);
704             b = (int) (((value >> 16) & 0xff) / 255. * 3. + .5);
705         } else {
706             cp++;                       /* skip alpha channel */
707             b = *cp++;
708             g = *cp++;
709             r = *cp++;
710         }
711         nvalue = (int)(0.30*r + 0.59*g + 0.11*b);
712         if ( nvalue > 255 ) nvalue = 255;
713         *ncp++ = (unsigned char)nvalue;
714     }
715     return rv;
716 }
717 
718 static PyObject *
imageop_grey2rgb(PyObject * self,PyObject * args)719 imageop_grey2rgb(PyObject *self, PyObject *args)
720 {
721     int x, y, len, nlen;
722     unsigned char *cp;
723     unsigned char *ncp;
724     PyObject *rv;
725     int i;
726     unsigned char value;
727     int backward_compatible = imageop_backward_compatible();
728 
729     if ( !PyArg_ParseTuple(args, "s#ii", &cp, &len, &x, &y) )
730         return 0;
731 
732     if ( !check_multiply(len, x, y) )
733         return 0;
734     nlen = x*y*4;
735     if ( !check_multiply_size(nlen, x, "x", y, "y", 4) )
736         return 0;
737 
738     rv = PyString_FromStringAndSize(NULL, nlen);
739     if ( rv == 0 )
740         return 0;
741     ncp = (unsigned char *)PyString_AsString(rv);
742 
743     for ( i=0; i < len; i++ ) {
744         value = *cp++;
745         if (backward_compatible) {
746             * (Py_UInt32 *) ncp = (Py_UInt32) value | ((Py_UInt32) value << 8 ) | ((Py_UInt32) value << 16);
747             ncp += 4;
748         } else {
749             *ncp++ = 0;
750             *ncp++ = value;
751             *ncp++ = value;
752             *ncp++ = value;
753         }
754     }
755     return rv;
756 }
757 
758 static PyMethodDef imageop_methods[] = {
759     { "crop",                   imageop_crop, METH_VARARGS },
760     { "scale",                  imageop_scale, METH_VARARGS },
761     { "grey2mono",              imageop_grey2mono, METH_VARARGS },
762     { "grey2grey2",             imageop_grey2grey2, METH_VARARGS },
763     { "grey2grey4",             imageop_grey2grey4, METH_VARARGS },
764     { "dither2mono",            imageop_dither2mono, METH_VARARGS },
765     { "dither2grey2",           imageop_dither2grey2, METH_VARARGS },
766     { "mono2grey",              imageop_mono2grey, METH_VARARGS },
767     { "grey22grey",             imageop_grey22grey, METH_VARARGS },
768     { "grey42grey",             imageop_grey42grey, METH_VARARGS },
769     { "tovideo",                imageop_tovideo, METH_VARARGS },
770     { "rgb2rgb8",               imageop_rgb2rgb8, METH_VARARGS },
771     { "rgb82rgb",               imageop_rgb82rgb, METH_VARARGS },
772     { "rgb2grey",               imageop_rgb2grey, METH_VARARGS },
773     { "grey2rgb",               imageop_grey2rgb, METH_VARARGS },
774     { 0,                    0 }
775 };
776 
777 
778 PyMODINIT_FUNC
initimageop(void)779 initimageop(void)
780 {
781     PyObject *m;
782 
783     if (PyErr_WarnPy3k("the imageop module has been removed in "
784                        "Python 3.0", 2) < 0)
785         return;
786 
787     m = Py_InitModule("imageop", imageop_methods);
788     if (m == NULL)
789         return;
790     ImageopDict = PyModule_GetDict(m);
791     ImageopError = PyErr_NewException("imageop.error", NULL, NULL);
792     if (ImageopError != NULL)
793         PyDict_SetItemString(ImageopDict, "error", ImageopError);
794 }
795