dos33fsprogs/hgr-utils/png2hgr.c

318 lines
6.7 KiB
C
Raw Normal View History

2018-08-24 19:16:28 +00:00
/* Converts 280x192 8-bit PNG file with correct palette to Apple II HGR */
#define VERSION "0.0.1"
#include <stdio.h> /* For FILE I/O */
#include <string.h> /* For strncmp */
#include <fcntl.h> /* for open() */
#include <unistd.h> /* for lseek() */
#include <sys/stat.h> /* for file modes */
#include <stdlib.h> /* free() */
#include <png.h>
#define OUTPUT_C 0
#define OUTPUT_ASM 1
#define OUTPUT_RAW 2
/* If you want default black a different color try the below */
/* Useful if you are trying to xdraw to get red/blue */
#if 0
/* Default is BLACK0 */
#define DEFAULT_BLACK -1
#else
/* DEFAULT is BLACK1 */
#define DEFAULT_BLACK 1
#endif
2018-08-24 19:16:28 +00:00
static int debug=1;
static int convert_color(int color) {
int c=0;
switch(color) {
2018-08-24 19:16:28 +00:00
#if 0
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 */
2018-08-24 19:16:28 +00:00
#endif
/* These use the questionable palette my older code used */
/* Also handle the newer one */
case 0x000000: c=0; break; /* black */
case 0x1bcb01: c=1; break; /* bright green */
case 0x14f53c: c=1; break; /* bright green */
case 0xe434fe: c=2; break; /* magenta */
case 0xe31e60: c=2; break; /* magenta */
case 0xffffff: c=3; break; /* white */
case 0xcd5b01: c=5; break; /* orange */
case 0xff6a3c: c=5; break; /* orange */
case 0x1b9afe: c=6; break; /* medium blue */
case 0x14cffd: c=6; break; /* medium blue */
default:
2018-08-24 19:16:28 +00:00
fprintf(stderr,"Unknown color %x\n",color);
break;
}
return c;
}
/* expects a PNG where the xsize is *2 */
int loadpng(char *filename,
2018-08-24 19:16:28 +00:00
unsigned char **image_ptr, int *xsize, int *ysize) {
int x,y;
int color;
FILE *infile;
2018-08-24 19:16:28 +00:00
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;
2018-08-24 19:16:28 +00:00
int row_bytes;
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;
2018-08-24 19:16:28 +00:00
*ysize=height;
color_type = png_get_color_type(png_ptr, info_ptr);
bit_depth = png_get_bit_depth(png_ptr, info_ptr);
2018-08-24 19:16:28 +00:00
if (width!=280) {
fprintf(stderr,"Unknown width %d\n",width);
return -1;
}
if (height!=192) {
fprintf(stderr,"Unknown height %d\n",height);
return -1;
}
2018-08-24 19:16:28 +00:00
image=calloc(width*height,sizeof(unsigned char));
if (image==NULL) {
fprintf(stderr,"Error allocating image\n");
return -1;
}
if (debug) {
2018-08-24 19:16:28 +00:00
fprintf(stderr,"PNG: width=%d height=%d depth=%d\n",
width,height,bit_depth);
if (color_type==PNG_COLOR_TYPE_RGB) {
fprintf(stderr,"Type RGB\n");
}
else if (color_type==PNG_COLOR_TYPE_RGB_ALPHA) {
fprintf(stderr,"Type RGBA\n");
}
else if (color_type==PNG_COLOR_TYPE_PALETTE) {
fprintf(stderr,"Type palette\n");
}
}
/* If palette, expand to RGB automatically */
if (color_type == PNG_COLOR_TYPE_PALETTE) {
png_set_expand(png_ptr);
}
png_read_update_info(png_ptr, info_ptr);
2018-08-24 19:16:28 +00:00
row_bytes = png_get_rowbytes(png_ptr, info_ptr);
// *pChannels = (int)png_get_channels(png_ptr, info_ptr);
row_pointers = (png_bytep*) malloc(sizeof(png_bytep) * height);
for (y=0; y<height; y++) {
2018-08-24 19:16:28 +00:00
row_pointers[y] = (png_byte*)malloc(row_bytes);
}
png_read_image(png_ptr, row_pointers);
2018-08-24 19:16:28 +00:00
png_read_end(png_ptr, NULL);
fclose(infile);
out_ptr=image;
2018-08-24 19:16:28 +00:00
for(y=0;y<height;y++) {
for(x=0;x<width;x++) {
2018-08-24 19:16:28 +00:00
color= (row_pointers[y][x*3]<<16)+
(row_pointers[y][x*3+1]<<8)+
(row_pointers[y][x*3+2]);
// if (debug) {
// fprintf(stderr,"%x ",color);
// }
2018-08-24 19:16:28 +00:00
a2_color=convert_color(color);
2018-08-24 19:16:28 +00:00
if (debug) {
fprintf(stderr,"%x",a2_color);
}
2018-08-24 19:16:28 +00:00
*out_ptr=a2_color;
out_ptr++;
}
2018-08-24 19:16:28 +00:00
if (debug) fprintf(stderr,"\nNR: ");
}
2018-08-24 19:16:28 +00:00
*image_ptr=image;
return 0;
}
/* Converts a PNG to RAW 8K Hires Image */
static void print_help(char *name,int version) {
printf("\npcx2hgr version %s\n",VERSION);
if (version) exit(1);
printf("\nUsage: %s [-r] [-s] PCXFILE\n\n",name);
printf("\t[-r] raw, don't prepend with BLOAD addr/size\n");
printf("\t[-s] short, leave off bottom text area\n");
printf("\n");
exit(1);
}
2018-08-24 19:31:29 +00:00
static int hgr_offset_table[48]={
0x0000,0x0080,0x0100,0x0180,0x0200,0x0280,0x0300,0x0380,
0x0028,0x00A8,0x0128,0x01A8,0x0228,0x02A8,0x0328,0x03A8,
0x0050,0x00D0,0x0150,0x01D0,0x0250,0x02D0,0x0350,0x03D0,
};
2018-08-24 19:31:29 +00:00
static int hgr_offset(int y) {
2018-08-24 19:33:29 +00:00
int temp,temp2,address;
2018-08-24 19:31:29 +00:00
temp=y/8;
2018-08-24 19:33:29 +00:00
temp2=y%8;
2018-08-24 19:31:29 +00:00
2018-08-24 19:33:29 +00:00
temp2=temp2*0x400;
address=hgr_offset_table[temp]+temp2;
return address;
2018-08-24 19:31:29 +00:00
}
2018-08-24 19:16:28 +00:00
static unsigned char apple2_image[8192];
int main(int argc, char **argv) {
int xsize=0,ysize=0;
2018-08-24 19:31:29 +00:00
int c,x,y;
2018-08-24 19:16:28 +00:00
unsigned char *image;
2018-08-24 19:31:29 +00:00
unsigned char byte1,byte2;
char *filename;
/* Parse command line arguments */
while ( (c=getopt(argc, argv, "hv") ) != -1) {
switch(c) {
case 'h':
print_help(argv[0],0);
break;
case 'v':
print_help(argv[0],1);
break;
default:
print_help(argv[0],0);
break;
}
}
if (optind>=argc) {
printf("ERROR: Was expecting filename!\n");
exit(1);
}
filename=strdup(argv[optind]);
2018-08-24 19:16:28 +00:00
memset(apple2_image,0,8192);
2018-08-24 19:16:28 +00:00
if (loadpng(filename,&image,&xsize,&ysize)<0) {
fprintf(stderr,"Error loading png!\n");
exit(-1);
}
fprintf(stderr,"Loaded image %d by %d\n",xsize,ysize);
2018-08-24 19:31:29 +00:00
for(y=0;y<192;y++) {
for(x=0;x<20;x++) {
byte1=0xff;
apple2_image[hgr_offset(y)]=byte1;
}
}
2018-08-24 19:31:29 +00:00
fwrite(apple2_image,8192,sizeof(unsigned char),stdout);
return 0;
}