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