/* Convert a 40x96 image into two, 40x48 images suitable of loading */ /* with my kfest18 based code */ #include #include #include #include #include #include #include #include #define OUTPUT_C 0 #define OUTPUT_ASM 1 static int convert_color(int color) { int c=0; switch(color) { case 0x000000: c=0; break; /* black */ case 0xe31e60: c=1; break; /* magenta */ case 0x604ebd: c=2; break; /* dark blue */ case 0xff44fd: c=3; break; /* purple */ case 0x00a360: c=4; break; /* dark green */ case 0x9c9c9c: c=5; break; /* grey 1 */ case 0x14cffd: c=6; break; /* medium blue */ case 0xd0c3ff: c=7; break; /* light blue */ case 0x607203: c=8; break; /* brown */ case 0xff6a3c: c=9; break; /* orange */ case 0x9d9d9d: c=10; break; /* grey 2 */ case 0xffa0d0: c=11; break; /* pink */ case 0x14f53c: c=12; break; /* bright green */ case 0xd0dd8d: c=13; break; /* yellow */ case 0x72ffd0: c=14; break; /* aqua */ case 0xffffff: c=15; break; /* white */ default: printf("Unknown color %x\n",color); break; } return c; } /* expects a PNG where the xsize is *2 */ int loadpng(char *filename, unsigned char **image_ptr, int *xsize, int *ysize, int high) { int x,y; int color; FILE *infile; int debug=0; unsigned char *image,*out_ptr; int width, height; int a2_color; png_byte bit_depth; png_structp png_ptr; png_infop info_ptr; png_bytep *row_pointers; png_byte color_type; // int number_of_passes; unsigned char header[8]; /* open file and test for it being a png */ infile = fopen(filename, "rb"); if (infile==NULL) { fprintf(stderr,"Error! Could not open %s\n",filename); return -1; } /* Check the header */ fread(header, 1, 8, infile); if (png_sig_cmp(header, 0, 8)) { fprintf(stderr,"Error! %s is not a PNG file\n",filename); return -1; } /* initialize stuff */ png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); if (!png_ptr) { fprintf(stderr,"Error create_read_struct\n"); exit(-1); } info_ptr = png_create_info_struct(png_ptr); if (!info_ptr) { fprintf(stderr,"Error png_create_info_struct\n"); exit(-1); } png_init_io(png_ptr, infile); png_set_sig_bytes(png_ptr, 8); png_read_info(png_ptr, info_ptr); width = png_get_image_width(png_ptr, info_ptr); height = png_get_image_height(png_ptr, info_ptr); *xsize=width; *ysize=height/2; color_type = png_get_color_type(png_ptr, info_ptr); bit_depth = png_get_bit_depth(png_ptr, info_ptr); if (debug) { printf("PNG: width=%d height=%d depth=%d\n",width,height,bit_depth); if (color_type==PNG_COLOR_TYPE_RGB) printf("Type RGB\n"); else if (color_type==PNG_COLOR_TYPE_RGB_ALPHA) printf("Type RGBA\n"); else if (color_type==PNG_COLOR_TYPE_PALETTE) printf("Type palette\n"); } // number_of_passes = png_set_interlace_handling(png_ptr); png_read_update_info(png_ptr, info_ptr); row_pointers = (png_bytep*) malloc(sizeof(png_bytep) * height); for (y=0; y2) && (run<16)) { if (out_type==OUTPUT_C) { printf("0x%02X,0x%02X,",0xA0|run,last); } else { printf("$%02X,$%02X",0xA0|run,last); } size+=2; } if (run>=16) { if (out_type==OUTPUT_C) { printf("0x%02X,0x%02X,0x%02X,",0xA0,run,last); } else { printf("$%02X,$%02X,$%02X",0xA0,run,last); } size+=3; } return size; } int rle_smaller(int out_type, char *varname, int xsize,int ysize, unsigned char *image,int high) { int run=0; int x; int last=-1,next; int size=0; int count=0; x=0; /* Write out xsize and ysize */ if (out_type==OUTPUT_C) { fprintf(stdout,"unsigned char %s_%s[]={\n",varname, high?"high":"low"); fprintf(stdout,"\t0x%X, /* ysize=%d */",xsize,ysize); } else { fprintf(stdout,"%s_%s:",varname,high?"high":"low"); fprintf(stdout,"\t.byte $%X ; ysize=%d",xsize,ysize); } size+=2; /* Get first top/bottom color pair */ last=image[x]; run++; x++; while(1) { /* get next top/bottom color pair */ next=image[x]; if ((next&0xf0)==0xA0) { fprintf(stderr,"Warning! Using color A (grey2)!\n"); next&=~0xf0; next|=0x50; // substitute grey1 } /* If color change (or too big) then output our run */ /* Note 0xff for run length is special case meaning "finished" */ if ((next!=last) || (run>254)) { size+=print_run(count,out_type,run,last); count++; run=0; last=next; } x++; /* If we reach the end */ if (x>=xsize*(ysize/2)) { run++; size+=print_run(count,out_type,run,last); break; } run++; if (count>6) count=0; } /* Print closing marker */ if (out_type==OUTPUT_C) { fprintf(stdout,"0xA1,"); fprintf(stdout,"\t};\n"); } else { fprintf(stdout,"\n\t.byte $A1\n"); } size+=1; return size; } /* Converts a PNG to RLE compressed data */ int main(int argc, char **argv) { unsigned char *image; int xsize,ysize; int size=0; int out_type=OUTPUT_C; if (argc<4) { fprintf(stderr,"Usage:\t%s type INFILE varname\n\n",argv[0]); fprintf(stderr,"\ttype: c or asm\n"); fprintf(stderr,"\tvarname: label for graphic\n"); fprintf(stderr,"\n"); exit(-1); } if (!strcmp(argv[1],"c")) { out_type=OUTPUT_C; } else if (!strcmp(argv[1],"asm")) { out_type=OUTPUT_ASM; } if (loadpng(argv[2],&image,&xsize,&ysize,1)<0) { fprintf(stderr,"Error loading png!\n"); exit(-1); } fprintf(stderr,"Loaded image %d by %d\n",xsize,ysize); // size=rle_original(out_type,argv[3], // xsize,ysize,image); size=rle_smaller(out_type,argv[3], xsize,ysize,image,0); fprintf(stderr,"Size %d bytes\n",size); if (loadpng(argv[2],&image,&xsize,&ysize,0)<0) { fprintf(stderr,"Error loading png!\n"); exit(-1); } fprintf(stderr,"Loaded image %d by %d\n",xsize,ysize); // size=rle_original(out_type,argv[3], // xsize,ysize,image); size=rle_smaller(out_type,argv[3], xsize,ysize,image,1); fprintf(stderr,"Size %d bytes\n",size); return 0; }