From 24f175a609e58e082760c5e7b8907058573ac113 Mon Sep 17 00:00:00 2001
From: "Mirar (Pontus Hagland)" <pike@sort.mirar.org>
Date: Wed, 7 Jan 1998 04:08:39 +0100
Subject: [PATCH] adding Image.X - x encoding stuff

Rev: src/modules/Image/encodings/x.c:1.1
---
 .gitattributes                  |   1 +
 src/modules/Image/encodings/x.c | 692 ++++++++++++++++++++++++++++++++
 2 files changed, 693 insertions(+)
 create mode 100644 src/modules/Image/encodings/x.c

diff --git a/.gitattributes b/.gitattributes
index 14d8b51f7b..8205c58654 100644
--- a/.gitattributes
+++ b/.gitattributes
@@ -83,6 +83,7 @@ testfont binary
 /src/modules/Image/encodings/gif_lzw.h foreign_ident
 /src/modules/Image/encodings/png.c foreign_ident
 /src/modules/Image/encodings/pnm.c foreign_ident
+/src/modules/Image/encodings/x.c foreign_ident
 /src/modules/Image/font.c foreign_ident
 /src/modules/Image/illustration.pike foreign_ident
 /src/modules/Image/image.c foreign_ident
diff --git a/src/modules/Image/encodings/x.c b/src/modules/Image/encodings/x.c
new file mode 100644
index 0000000000..57269c69f4
--- /dev/null
+++ b/src/modules/Image/encodings/x.c
@@ -0,0 +1,692 @@
+/* $Id: x.c,v 1.1 1998/01/07 03:08:39 mirar Exp $ */
+
+/*
+**! module Image
+**! note
+**!	$Id: x.c,v 1.1 1998/01/07 03:08:39 mirar Exp $
+**! submodule X
+**!
+**!	This submodule handles encoding and decoding of
+**!	the binary formats of X11.
+**!
+**!	
+**!
+**! see also: Image, Image.image, Image.colortable
+*/
+
+#include <math.h>
+#include <ctype.h>
+
+#include "stralloc.h"
+#include "global.h"
+
+#include <sys/types.h>
+#include <sys/stat.h>
+#ifdef HAVE_NETINET_IN_H
+#include <netinet/in.h>
+#endif
+#ifdef HAVE_WINSOCK_H
+#include <winsock.h>
+#endif
+
+RCSID("$Id: x.c,v 1.1 1998/01/07 03:08:39 mirar Exp $");
+#include "pike_macros.h"
+#include "object.h"
+#include "constants.h"
+#include "interpret.h"
+#include "svalue.h"
+#include "threads.h"
+#include "array.h"
+#include "error.h"
+
+
+
+
+#include "image.h"
+#include "colortable.h"
+#include "builtin_functions.h"
+
+extern struct program *image_colortable_program;
+extern struct program *image_program;
+
+/*
+**! method string encode_truecolor(object image,int bpp,int alignbits,int rbits,int rshift,int gbits,int gshift,int bbits,int bshift)
+**! method string encode_truecolor_masks(object image,int bpp,int alignbits,int rmask,int gmask,int bmask)
+**! method string encode_truecolor(object image,int bpp,int alignbits,int rbits,int rshift,int gbits,int gshift,int bbits,int bshift,object ct)
+**! method string encode_truecolor_masks(object image,int bpp,int alignbits,int rmask,int gmask,int bmask,object ct)
+**!	Pack a image into a truecolor string. You will get a string
+**!	of packed red, green and blue bits; 
+**!	ie: 
+**!
+**!	<tt>encode_truecolor(img, 12,32, 3,5, 4,0, 3,8)</tt>
+**!	will give (aligned to even 32 bits for each row):<br>
+**!	<tt>0bbbrrr0 gggg0bbb rrr0gggg 0bbb</tt>...<br>
+**!	<tt><--pixel 1--><--pixel 2--> <--3--></tt><br>
+**!	<tt>10987654 32101098 76543210 1098</tt>... <- bit position
+**!	<tt> <-><->  <--></tt>
+**!	<tt>  |  |    +--- 4,0</tt>: 4 bits green shifted 0 bits
+**!	<tt>  |  +-------- 3,5</tt>: 3 bits red shifted 5 bits
+**!	<tt>  +----------- 3,8</tt>: 3 bits blue shifted 8 bits
+**!
+**!	The above call is equal to 
+**!	<br><tt>encode_truecolor_masks(img, 12,32, 224, 15, 768)</tt>
+**!	and
+**!	<br><tt>encode_truecolor(img, 12,32, 3,5,4,0,3,8, colortable(1<<3,1<<4,1<<3))</tt>. 
+**!	<br>The latter gives possibility to use dither algorithms, 
+**!	but is slightly slower.
+**!
+**! arg object image
+**!	the image object to encode
+**! arg int bpp
+**!	bits per pixel, how many bits each pixel should take
+**! arg int alignbits
+**!	the number of even bits each line should be padded to
+**! arg int rbits
+**! arg int gbits
+**! arg int bbits
+**!	bits for each basecolor
+**! arg int rshift
+**! arg int gshift
+**! arg int bshift
+**!	leftshifts for each basecolor
+**! arg int rmask
+**! arg int gmask
+**! arg int bmask
+**! 	masks for each basecolor (xbits and gbits are calculated from this),
+**!	needs to be massive (no zeroes among the ones in the mask).
+**! arg object ct
+**!	colortable object (for dithering, or whatever)
+**! 
+*/
+
+static void x_encode_truecolor(INT32 args)
+{
+   struct image *img;
+   struct neo_colortable *nct;
+   int rbits,rshift,gbits,gshift,bbits,bshift;
+   int bpp,alignbits;
+   unsigned long rfmask,gfmask,bfmask;
+   unsigned char *d;
+   struct pike_string *dest;
+   INT32 x,y;
+   rgb_group *s,*tmp=NULL;
+
+   if (args<9)
+      error("Image.X.encode_truecolor: too few arguments (expected 9 arguments)\n");
+   
+   if (sp[-args].type!=T_OBJECT ||
+       !(img=(struct image*)get_storage(sp[-args].u.object,image_program)))
+      error("Image.X.encode_truecolor: illegal argument 1 (expected image object)\n");
+   if (args>9)
+      if (sp[9-args].type!=T_OBJECT ||
+	  !(nct=(struct neo_colortable*)
+	    get_storage(sp[9-args].u.object,image_colortable_program)))
+	 error("Image.X.encode_truecolor: illegal argument 10 (expected colortable object)\n");
+      else;
+   else nct=NULL;
+	 
+   if (sp[1-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 2 (expected integer)\n");
+   else
+      bpp=sp[1-args].u.integer;
+
+   if (sp[2-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 3 (expected integer)\n");
+   else
+      alignbits=sp[2-args].u.integer;
+
+   if (!alignbits) alignbits=1;
+
+   if (sp[3-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 4 (expected integer)\n");
+   else
+      rbits=sp[3-args].u.integer;
+
+   if (sp[4-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 5 (expected integer)\n");
+   else
+      rshift=sp[4-args].u.integer;
+
+   if (sp[5-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 6 (expected integer)\n");
+   else
+      gbits=sp[5-args].u.integer;
+
+   if (sp[6-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 7 (expected integer)\n");
+   else
+      gshift=sp[6-args].u.integer;
+
+   if (sp[7-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 8 (expected integer)\n");
+   else
+      bbits=sp[7-args].u.integer;
+
+   if (sp[8-args].type!=T_INT)
+      error("Image.X.encode_truecolor: illegal argument 9 (expected integer)\n");
+   else
+      bshift=sp[8-args].u.integer;
+
+   if (nct) 
+   {
+      tmp=(rgb_group*)xalloc(sizeof(rgb_group)*img->xsize*img->ysize +1);
+      if (!image_colortable_map_image(nct,img->img,tmp,
+				      img->xsize*img->ysize,img->xsize))
+      {
+	 free(tmp);
+	 error("Image.X.encode_truecolor(): called colortable is not initiated\n");
+      }
+      s=tmp;
+   }
+   else
+      s=img->img;
+
+
+   dest=begin_shared_string(( ( (img->xsize*bpp+alignbits-1) / alignbits )
+			      * alignbits*img->ysize  +7 ) / 8);
+   d=(unsigned char*)(dest->str);
+   *d=0;
+
+   y=img->ysize;
+
+THREADS_ALLOW();
+   
+   if (!(rshift&7) && !(gshift&7) && !(bshift&7) 
+       && sizeof(COLORTYPE)==1
+       && rbits==8 && gbits==8 && bbits==8 && !(bpp&7) 
+       && (!(alignbits&7) || !(bpp%alignbits)) )
+   {
+      INT32 Bpp=bpp>>3;
+      INT32 rpos=-(rshift>>3)-1;
+      INT32 gpos=-(gshift>>3)-1;
+      INT32 bpos=-(bshift>>3)-1;
+      INT32 linemod=(alignbits-((img->xsize*bpp+alignbits-1)%alignbits)-1)>>3;
+
+      if (!linemod && Bpp==4 && rpos!=gpos && gpos!=bpos) 
+      {
+	 INT32 zpos=-4;
+	 while (rpos==zpos||gpos==zpos||bpos==zpos) zpos++;
+	 while (y--)
+	 {
+	    x=img->xsize;
+	    while (x--) d+=4,d[rpos]=s->r,d[gpos]=s->g,d[bpos]=s->b,d[zpos]=0; 
+	 }
+      }
+      else if (!linemod && Bpp==3 && rpos!=gpos && gpos!=bpos) 
+      {
+	 while (y--)
+	 {
+	    x=img->xsize;
+	    while (x--) d+=3,d[rpos]=s->r,d[gpos]=s->g,d[bpos]=s->b; 
+	 }
+      }
+      else
+      {
+	 MEMSET(d,0,( ( (img->xsize*bpp+alignbits-1) / alignbits )
+		      * alignbits*img->ysize  +7 ) / 8);
+	 while (y--)
+	 {
+	    x=img->xsize;
+	    while (x--) d+=Bpp,d[rpos]=s->r,d[gpos]=s->g,d[bpos]=s->b; 
+	    d+=linemod;
+	 }
+      }
+   }
+   else
+   {
+      INT32 rfshift,gfshift,bfshift,rzshift,gzshift,bzshift;
+      INT32 bpshift,blinemod,bit;
+
+      rfshift=rshift-(sizeof(COLORTYPE)*8-rbits);
+      gfshift=gshift-(sizeof(COLORTYPE)*8-gbits);
+      bfshift=bshift-(sizeof(COLORTYPE)*8-bbits);
+      if (rfshift<0) rzshift=-rfshift,rfshift=0; else rzshift=0;
+      if (gfshift<0) gzshift=-gfshift,gfshift=0; else gzshift=0;
+      if (bfshift<0) bzshift=-bfshift,bfshift=0; else bzshift=0;
+
+      rfmask=(((1<<rbits)-1)<<(sizeof(COLORTYPE)*8-rbits));
+      gfmask=(((1<<gbits)-1)<<(sizeof(COLORTYPE)*8-gbits));
+      bfmask=(((1<<bbits)-1)<<(sizeof(COLORTYPE)*8-bbits));
+      bpshift=sizeof(unsigned long)*8-bpp;
+      blinemod=alignbits-( (img->xsize*bpp+alignbits-1) % alignbits)-1;
+
+      bit=0;
+
+      while (y--)
+      {
+	 unsigned long bp;
+
+	 x=img->xsize;
+	 while (x--) /* write bits from this line */
+	 {
+	    register unsigned long b =
+	       ((((s->r&rfmask)>>rzshift)<<rfshift)|
+		(((s->g&gfmask)>>gzshift)<<gfshift)|
+		(((s->b&bfmask)>>bzshift)<<bfshift))<<bpshift;
+	    bp = bpp;
+	    while (bp>8-bit)
+	    {
+	       *d|=(unsigned char)((b>>24)&(((1<<(8-bit))-1)<<bit));
+	       b<<=8-bit;
+	       bp-=8-bit;
+	       *(++d)=0; bit=0;
+	    }
+	    *d|=b>>(24-(8-bp)+bit); 
+	    bit+=bp;
+	    if (bit==8) *(++d)=0,bit=0;
+	    s++;
+	 }
+	 bp=blinemod;
+	 while (bp>8-bit) *(++d)=0,bp-=8-bit,bit=0;
+	 bit+=bp;
+	 if (bit==8) *(++d)=0,bit=0;
+      }
+   }
+
+THREADS_DISALLOW();
+
+   if (nct) free(tmp);
+
+   pop_n_elems(args);
+   push_string(end_shared_string(dest));
+}
+
+static INLINE void x_examine_mask(struct svalue *mask,
+				  const char *what,
+				  int *bits,int *shift)
+{
+   unsigned long x;
+   if (mask->type!=T_INT)
+      error("Image.X.encode_truecolor_masks: illegal %s (expected integer)\n",what);
+
+   x=(unsigned long)mask->u.integer;
+   x&=(unsigned long)((INT32)-1); /* i hope this works... */
+   /* what i _really_ want to do is cast INT32 to unsigned INT32... */
+
+   *bits=0; 
+   *shift=0; 
+   if (!x) return; 
+   while (!(x&1)) x>>=1,(*shift)++;
+   while (x&1) x>>=1,(*bits)++;
+
+   if (x)
+      error("Image.X.encode_truecolor_masks: illegal %s (nonmassive bitfield)\n",what);
+}
+
+static void x_encode_truecolor_masks(INT32 args)
+{
+   struct object *ct;
+   int rbits,rshift,gbits,gshift,bbits,bshift;
+
+   if (args<6) 
+      error("Image.X.encode_truecolor_masks: too few arguments (expected 4 arguments)\n");
+   if (sp[-args].type!=T_OBJECT ||
+       !get_storage(sp[-args].u.object,image_program))
+      error("Image.X.encode_truecolor_masks: illegal argument 1 (expected image object)\n");
+
+   if (args>6)
+      if (sp[6-args].type!=T_OBJECT ||
+	  !get_storage(ct=sp[6-args].u.object,image_colortable_program))
+	 error("Image.X.encode_truecolor_masks: illegal argument 7 (expected colortable object)\n");
+      else;
+   else
+      ct=NULL;
+ 
+   if (sp[1-args].type!=T_INT)
+      error("Image.X.encode_truecolor_masks: illegal argument 2 (expected integer)\n");
+   if (sp[2-args].type!=T_INT)
+      error("Image.X.encode_truecolor_masks: illegal argument 3 (expected integer)\n");
+
+   x_examine_mask(sp+3-args,"argument 3 (red mask)",&rbits,&rshift);
+   x_examine_mask(sp+4-args,"argument 4 (blue mask)",&gbits,&gshift);
+   x_examine_mask(sp+5-args,"argument 5 (green mask)",&bbits,&bshift);
+
+   if (ct) ct->refs++;
+   pop_n_elems(args-3);
+   push_int(rbits);
+   push_int(rshift);
+   push_int(gbits);
+   push_int(gshift);
+   push_int(bbits);
+   push_int(bshift);
+   if (ct)
+   {
+      push_object(ct);
+      x_encode_truecolor(10);
+   }
+   else
+      x_encode_truecolor(9);
+}
+
+/*
+**! method string encode_pseudo(object image,int bpp,int alignbits,int vbpp,object colortable)
+**! method string encode_pseudo(object image,int bpp,int alignbits,int vbpp,object colortable,string translate)
+**!
+**! arg object image
+**!	the image object to encode
+**! arg int bpp
+**!	bits per pixel, how many bits each pixel should take
+**! arg int vbpp
+**!	value bits per pixel; how many bits per pixel that really 
+**!	contains information
+**! arg int alignbits
+**!	the number of even bits each line should be padded to
+**! arg object colortable
+**!	colortable to get indices for pseudocolor
+**! arg string translate
+**!	translate table for colors. Length of this string
+**!	should be 1<<vbpp (or 2<<vbpp if vbpp are greater than 8).
+**!
+**! note
+**!	currently, only upto 16 bits pseudocolor are supported.
+*/
+
+static void x_encode_pseudocolor_1byte_exact(INT32 args,
+					     struct image *img,
+					     struct neo_colortable *nct,
+					     int bpp,int vbpp,int alignbits,
+					     unsigned char* translate)
+{
+   struct pike_string *dest;
+   INT32 linemod=(alignbits-((img->xsize*bpp+alignbits-1)%alignbits)-1)>>3;
+   INT32 mask=((1<<vbpp)-1);
+
+   dest=begin_shared_string(img->xsize*img->ysize);
+
+   if (!image_colortable_index_8bit_image(nct,img->img,
+					  (unsigned char*)(dest->str),
+					  img->xsize*img->ysize,img->xsize))
+   {
+      free_string(end_shared_string(dest));
+      error("Image.x.encode_pseudocolor: colortable not initialised");
+   }
+
+   if (!translate && !linemod)
+   {
+      pop_n_elems(args);
+      push_string(end_shared_string(dest));
+      return;
+   }
+		
+   if (!linemod)
+   {
+      unsigned char *d;
+      INT32 n;
+      d=(unsigned char*)dest->str;
+      n=img->xsize*img->ysize;
+      while (n--) *d=translate[(*d)&mask],d++;
+      pop_n_elems(args);
+      push_string(end_shared_string(dest));
+      return;
+   }
+
+   do
+   {
+      unsigned char *d;
+      unsigned char *s;
+      INT32 y,x,m;
+
+      struct pike_string *dest2;
+      dest2=begin_shared_string((img->xsize+linemod)*img->ysize);
+
+      s=(unsigned char*)(dest->str);
+      d=(unsigned char*)(dest2->str);
+
+      y=img->ysize;
+      while (y--)
+      {
+	 if (translate) 
+	    { x=img->xsize; while (x--) *(d++)=translate[(*(s++))&mask]; }
+	 else MEMCPY(d,s,img->xsize),d+=img->xsize,s+=img->xsize;
+	 m=linemod;
+	 while (m--) *(d++)=0;
+      }
+
+      free_string(end_shared_string(dest));
+      pop_n_elems(args);
+      push_string(end_shared_string(dest2));
+   }
+   while (0);
+}
+					     
+static void x_encode_pseudocolor_1byte(INT32 args,
+				       struct image *img,
+				       struct neo_colortable *nct,
+				       int bpp,int vbpp,int alignbits,
+				       unsigned char* translate)
+{
+   struct pike_string *dest;
+   INT32 blinemod=(alignbits-((img->xsize*bpp+alignbits-1)%alignbits)-1);
+   unsigned char *d;
+   unsigned char *s;
+   INT32 y,x,bit,bp,b;
+   struct pike_string *dest2;
+
+   dest=begin_shared_string(img->xsize*img->ysize);
+
+   if (!image_colortable_index_8bit_image(nct,img->img,
+					  (unsigned char*)(dest->str),
+					  img->xsize*img->ysize,img->xsize))
+   {
+      free_string(end_shared_string(dest));
+      error("Image.x.encode_pseudocolor: colortable not initialised");
+   }
+
+   dest2=begin_shared_string(((img->xsize*bpp+blinemod)*img->ysize+7)/8);
+
+   s=(unsigned char*)(dest->str);
+   d=(unsigned char*)(dest2->str);
+   bit=0;
+   *d=0;
+
+   y=img->ysize;
+   while (y--)
+   {
+      if (translate) 
+      { 
+	 x=img->xsize; 
+	 while (x--) 
+	 {
+	    b=translate[*(s++)]<<(32-vbpp); 
+	    bp = bpp;
+	    while (bp>8-bit)
+	    {
+	       *d|=(unsigned char)((b>>24)&(((1<<(8-bit))-1)<<bit));
+	       b<<=8-bit;
+	       bp-=8-bit;
+	       *(++d)=0; bit=0;
+	    }
+	    *d|=b>>(24-(8-bp)+bit); 
+	    bit+=bp;
+	    if (bit==8) *(++d)=0,bit=0;
+	 }
+      }
+      else 
+      {
+	 x=img->xsize; 
+	 while (x--) 
+	 {
+	    b=(*(s++))<<(32-vbpp); 
+	    bp = bpp;
+	    while (bp>8-bit)
+	    {
+	       *d|=(unsigned char)((b>>24)&(((1<<(8-bit))-1)<<bit));
+	       b<<=8-bit;
+	       bp-=8-bit;
+	       *(++d)=0; bit=0;
+	    }
+	    *d|=b>>(24-(8-bp)+bit); 
+	    bit+=bp;
+	    if (bit==8) *(++d)=0,bit=0;
+	 }
+      }
+      bp=blinemod;
+      while (bp>8-bit) *(++d)=0,bp-=8-bit,bit=0;
+      bit+=bp;
+      if (bit==8) *(++d)=0,bit=0;
+   }
+
+   free_string(end_shared_string(dest));
+   pop_n_elems(args);
+   push_string(end_shared_string(dest2));
+}
+					     
+static void x_encode_pseudocolor_2byte(INT32 args,
+				       struct image *img,
+				       struct neo_colortable *nct,
+				       int bpp,int vbpp,int alignbits,
+				       unsigned short  *translate)
+{
+   struct pike_string *dest;
+   INT32 blinemod=(alignbits-((img->xsize*bpp+alignbits-1)%alignbits)-1);
+   unsigned char *d;
+   unsigned char *s;
+   INT32 y,x,bit,bp,b;
+   struct pike_string *dest2;
+
+   dest=begin_shared_string(img->xsize*img->ysize);
+
+   if (!image_colortable_index_8bit_image(nct,img->img,
+					  (unsigned char*)(dest->str),
+					  img->xsize*img->ysize,img->xsize))
+   {
+      free_string(end_shared_string(dest));
+      error("Image.x.encode_pseudocolor: colortable not initialised");
+   }
+
+   dest2=begin_shared_string(((img->xsize*bpp+blinemod)*img->ysize+7)/8);
+
+   s=(unsigned char*)(dest->str);
+   d=(unsigned char*)(dest2->str);
+   *d=0;
+   bit=0;
+
+   y=img->ysize;
+   while (y--)
+   {
+      if (translate) 
+      { 
+	 x=img->xsize; 
+	 while (x--) 
+	 {
+	    b=ntohs(translate[*(s++)])<<(32-vbpp); 
+	    bp = bpp;
+	    while (bp>8-bit)
+	    {
+	       *d|=(unsigned char)((b>>24)&(((1<<(8-bit))-1)<<bit));
+	       b<<=8-bit;
+	       bp-=8-bit;
+	       *(++d)=0; bit=0;
+	    }
+	    *d|=b>>(24-(8-bp)+bit); 
+	    bit+=bp;
+	    if (bit==8) *(++d)=0,bit=0;
+	 }
+      }
+      else 
+      {
+	 x=img->xsize; 
+	 while (x--) 
+	 {
+	    b=(*(s++))<<(32-vbpp); 
+	    bp = bpp;
+	    while (bp>8-bit)
+	    {
+	       *d|=(unsigned char)((b>>24)&(((1<<(8-bit))-1)<<bit));
+	       b<<=8-bit;
+	       bp-=8-bit;
+	       *(++d)=0; bit=0;
+	    }
+	    *d|=b>>(24-(8-bp)+bit); 
+	    bit+=bp;
+	    if (bit==8) *(++d)=0,bit=0;
+	 }
+      }
+      bp=blinemod;
+      while (bp>8-bit) *(++d)=0,bp-=8-bit,bit=0;
+      bit+=bp;
+      if (bit==8) *(++d)=0,bit=0;
+   }
+
+   free_string(end_shared_string(dest));
+   pop_n_elems(args);
+   push_string(end_shared_string(dest2));
+}
+					     
+static void x_encode_pseudocolor(INT32 args)
+{
+   INT32 bpp,alignbits,vbpp;
+   struct image *img;
+   struct neo_colortable *nct;
+   char *translate=NULL;
+   
+   if (args<4) 
+      error("Image.X.encode_pseudocolor: too few arguments");
+   if (sp[1-args].type!=T_INT)
+      error("Image.X.encode_pseudocolor: illegal argument 2 (expected integer)\n");
+   if (sp[2-args].type!=T_INT)
+      error("Image.X.encode_pseudocolor: illegal argument 3 (expected integer)\n");
+   if (sp[3-args].type!=T_INT)
+      error("Image.X.encode_pseudocolor: illegal argument 4 (expected integer)\n");
+   bpp=sp[1-args].u.integer;
+   alignbits=sp[2-args].u.integer;
+   vbpp=sp[3-args].u.integer;
+   if (!alignbits) alignbits=1;
+
+   if (sp[-args].type!=T_OBJECT ||
+       !(img=(struct image*)get_storage(sp[-args].u.object,image_program)))
+      error("Image.X.encode_pseudocolor: illegal argument 1 (expected image object)\n");
+   if (sp[4-args].type!=T_OBJECT ||
+       !(nct=(struct neo_colortable*)
+	 get_storage(sp[4-args].u.object,image_colortable_program)))
+      error("Image.X.encode_pseudocolor: illegal argument 4 (expected colortable object)\n");
+
+   if (args>5)
+      if (sp[5-args].type!=T_STRING)
+	 error("Image.X.encode_pseudocolor: illegal argument 6 (expected string)\n");
+      else if (sp[5-args].u.string->len!=((vbpp>8)?2:1)<<vbpp)
+	 error("Image.X.encode_pseudocolor: illegal argument 6 (expected translate string of length %d, not %d)\n",((vbpp>8)?2:1)<<vbpp,sp[5-args].u.string->len);
+      else 
+	 translate=sp[5-args].u.string->str;
+	 
+   if ( vbpp==8 && bpp==8 && !((bpp*img->xsize)%alignbits) )
+      x_encode_pseudocolor_1byte_exact(args,img,nct,vbpp,bpp,alignbits,
+				       (unsigned char*)translate);
+   else if (vbpp<=8) 
+      x_encode_pseudocolor_1byte(args,img,nct,bpp,vbpp,alignbits,
+				 (unsigned char*)translate);
+   else if (vbpp<=16) 
+      x_encode_pseudocolor_2byte(args,img,nct,bpp,vbpp,alignbits,
+				 (unsigned short*)translate);
+   else error("Image.X.encode_pseudocolor: sorry, too many bits\n");
+}
+
+/**** init module ********************************************/
+
+struct program *image_x_module_program=NULL;
+
+void init_image_x(void)
+{
+   start_new_program();
+   
+   add_function("encode_truecolor",x_encode_truecolor,
+		"function(object,int,int,int,int,int,int,int,int,void|object:string)",0);
+   add_function("encode_truecolor_masks",x_encode_truecolor_masks,
+		"function(object,int,int,int,int,int,void|object:string)",0);
+   add_function("encode_pseudocolor",x_encode_pseudocolor,
+		"function(object,int,int,int,object,void|string:string)",0);
+
+   image_x_module_program=end_program();
+   push_object(clone_object(image_x_module_program,0));
+   add_constant(make_shared_string("X"),sp-1,0);
+   pop_stack();
+}
+
+void exit_image_x(void)
+{
+  if(image_x_module_program)
+  {
+    free_program(image_x_module_program);
+    image_x_module_program=0;
+  }
+}
-- 
GitLab