amiga_imgv
changeset 8:a9edbbdb8213
forgot to add the ILBM image loader
author | John Tsiombikas <nuclear@member.fsf.org> |
---|---|
date | Sun, 29 Oct 2017 15:08:29 +0200 (2017-10-29) |
parents | 4c36d0f44aa6 |
children | 01bd8bbc46d4 |
files | src/image_ilbm.c |
diffstat | 1 files changed, 391 insertions(+), 0 deletions(-) [+] |
line diff
1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 1.2 +++ b/src/image_ilbm.c Sun Oct 29 15:08:29 2017 +0200 1.3 @@ -0,0 +1,391 @@ 1.4 +#include <stdio.h> 1.5 +#include <stdlib.h> 1.6 +#include <string.h> 1.7 +#include <errno.h> 1.8 +#include <assert.h> 1.9 +#include "image.h" 1.10 +#include "logger.h" 1.11 +#include "endian.h" 1.12 + 1.13 +#ifdef __GNUC__ 1.14 +#define PACKED __attribute__((packed)) 1.15 +#else 1.16 +#define PACKED 1.17 +#endif 1.18 + 1.19 +#define MKID(a, b, c, d) (((a) << 24) | ((b) << 16) | ((c) << 8) | (d)) 1.20 + 1.21 +#define IS_IFF_CONTAINER(id) ((id) == IFF_FORM || (id) == IFF_CAT || (id) == IFF_LIST) 1.22 + 1.23 +enum { 1.24 + IFF_FORM = MKID('F', 'O', 'R', 'M'), 1.25 + IFF_CAT = MKID('C', 'A', 'T', ' '), 1.26 + IFF_LIST = MKID('L', 'I', 'S', 'T'), 1.27 + IFF_ILBM = MKID('I', 'L', 'B', 'M'), 1.28 + IFF_PBM = MKID('P', 'B', 'M', ' '), 1.29 + IFF_BMHD = MKID('B', 'M', 'H', 'D'), 1.30 + IFF_CMAP = MKID('C', 'M', 'A', 'P'), 1.31 + IFF_BODY = MKID('B', 'O', 'D', 'Y'), 1.32 + IFF_CRNG = MKID('C', 'R', 'N', 'G') 1.33 +}; 1.34 + 1.35 + 1.36 +struct chdr { 1.37 + uint32_t id; 1.38 + uint32_t size; 1.39 +}; 1.40 + 1.41 +#ifdef __WATCOMC__ 1.42 +#pragma push(pack, 1) 1.43 +#endif 1.44 +struct bitmap_header { 1.45 + uint16_t width, height; 1.46 + int16_t xoffs, yoffs; 1.47 + uint8_t nplanes; 1.48 + uint8_t masking; 1.49 + uint8_t compression; 1.50 + uint8_t padding; 1.51 + uint16_t colorkey; 1.52 + uint8_t aspect_num, aspect_denom; 1.53 + int16_t pgwidth, pgheight; 1.54 +} PACKED; 1.55 +#ifdef __WATCOMC__ 1.56 +#pragma pop(pack) 1.57 +#endif 1.58 + 1.59 +enum { 1.60 + MASK_NONE, 1.61 + MASK_PLANE, 1.62 + MASK_COLORKEY, 1.63 + MASK_LASSO 1.64 +}; 1.65 + 1.66 +struct crng { 1.67 + uint16_t padding; 1.68 + uint16_t rate; 1.69 + uint16_t flags; 1.70 + uint8_t low, high; 1.71 +}; 1.72 + 1.73 +enum { 1.74 + CRNG_ENABLE = 1, 1.75 + CRNG_REVERSE = 2 1.76 +}; 1.77 + 1.78 +static int read_header(FILE *fp, struct chdr *hdr); 1.79 +static int read_ilbm_pbm(FILE *fp, uint32_t type, uint32_t size, struct ham_image *img); 1.80 +static int read_bmhd(FILE *fp, struct bitmap_header *bmhd); 1.81 +static int read_crng(FILE *fp, struct crng *crng); 1.82 +static int read_body_ilbm(FILE *fp, struct bitmap_header *bmhd, struct ham_image *img); 1.83 +static int read_body_pbm(FILE *fp, struct bitmap_header *bmhd, struct ham_image *img); 1.84 +static int read_compressed_scanline(FILE *fp, unsigned char *scanline, int width); 1.85 + 1.86 +int file_is_lbm(FILE *fp) 1.87 +{ 1.88 + uint32_t type; 1.89 + struct chdr hdr; 1.90 + 1.91 + while(read_header(fp, &hdr) != -1) { 1.92 + if(IS_IFF_CONTAINER(hdr.id)) { 1.93 + if(fread(&type, 4, 1, fp) < 1) { 1.94 + break; 1.95 + } 1.96 + type = ntohl(type); 1.97 + 1.98 + if(type == IFF_ILBM || type == IFF_PBM ) { 1.99 + rewind(fp); 1.100 + return 1; 1.101 + } 1.102 + hdr.size -= sizeof type; /* so we will seek fwd correctly */ 1.103 + } 1.104 + fseek(fp, hdr.size, SEEK_CUR); 1.105 + } 1.106 + fseek(fp, 0, SEEK_SET); 1.107 + return 0; 1.108 +} 1.109 + 1.110 +struct ham_image *load_ilbm(const char *fname) 1.111 +{ 1.112 + uint32_t type; 1.113 + struct chdr hdr; 1.114 + FILE *fp = 0; 1.115 + struct ham_image *img = 0; 1.116 + 1.117 + if(!(fp = fopen(fname, "rb"))) { 1.118 + logmsg("failed to open %s: %s\n", fname, strerror(errno)); 1.119 + return 0; 1.120 + } 1.121 + if(!(img = calloc(1, sizeof *img))) { 1.122 + logmsg("failed to allocate image structure\n"); 1.123 + goto err; 1.124 + } 1.125 + 1.126 + while(read_header(fp, &hdr) != -1) { 1.127 + if(IS_IFF_CONTAINER(hdr.id)) { 1.128 + if(fread(&type, 4, 1, fp) < 1) { 1.129 + break; 1.130 + } 1.131 + type = ntohl(type); 1.132 + hdr.size -= sizeof type; /* to compensate for having advanced 4 more bytes */ 1.133 + 1.134 + if(type == IFF_ILBM) { 1.135 + if(read_ilbm_pbm(fp, type, hdr.size, img) == -1) { 1.136 + goto err; 1.137 + } 1.138 + return img; 1.139 + } 1.140 + if(type == IFF_PBM) { 1.141 + if(read_ilbm_pbm(fp, type, hdr.size, img) == -1) { 1.142 + goto err; 1.143 + } 1.144 + return img; 1.145 + } 1.146 + } 1.147 + fseek(fp, hdr.size, SEEK_CUR); 1.148 + } 1.149 + 1.150 +err: 1.151 + fclose(fp); 1.152 + free(img); 1.153 + return 0; 1.154 +} 1.155 + 1.156 +static int read_header(FILE *fp, struct chdr *hdr) 1.157 +{ 1.158 + if(fread(hdr, 1, sizeof *hdr, fp) < sizeof *hdr) { 1.159 + return -1; 1.160 + } 1.161 + hdr->id = ntohl(hdr->id); 1.162 + hdr->size = ntohl(hdr->size); 1.163 + return 0; 1.164 +} 1.165 + 1.166 +static int read_ilbm_pbm(FILE *fp, uint32_t type, uint32_t size, struct ham_image *img) 1.167 +{ 1.168 + int i, palsz, res = -1; 1.169 + struct chdr hdr; 1.170 + struct bitmap_header bmhd; 1.171 + struct crng crng; 1.172 + struct colrange *crnode; 1.173 + unsigned char pal[3 * 256]; 1.174 + unsigned char *pptr; 1.175 + long start = ftell(fp); 1.176 + 1.177 + while(read_header(fp, &hdr) != -1 && ftell(fp) - start < size) { 1.178 + switch(hdr.id) { 1.179 + case IFF_BMHD: 1.180 + assert(hdr.size == 20); 1.181 + if(read_bmhd(fp, &bmhd) == -1) { 1.182 + return -1; 1.183 + } 1.184 + img->width = bmhd.width; 1.185 + img->height = bmhd.height; 1.186 + if(bmhd.nplanes > 8) { 1.187 + logmsg("%d planes found, only paletized LBM files supported\n", (int)bmhd.nplanes); 1.188 + return -1; 1.189 + } 1.190 + img->nbitplanes = bmhd.nplanes; 1.191 + 1.192 + if(!(img->pixels = malloc(img->width * img->height / 8 * img->nbitplanes))) { 1.193 + logmsg("failed to allocate %dx%d (%d bitplane) image\n", img->width, img->height, img->nbitplanes); 1.194 + return -1; 1.195 + } 1.196 + break; 1.197 + 1.198 + case IFF_CMAP: 1.199 + assert(hdr.size / 3 <= 256); 1.200 + 1.201 + palsz = hdr.size / 3; 1.202 + 1.203 + if(fread(pal, 1, hdr.size, fp) < hdr.size) { 1.204 + logmsg("failed to read colormap\n"); 1.205 + return -1; 1.206 + } 1.207 + pptr = pal; 1.208 + for(i=0; i<palsz; i++) { 1.209 + uint16_t r = ((uint16_t)*pptr++ << 4) & 0xf00; 1.210 + uint16_t g = (uint16_t)*pptr++ & 0xf0; 1.211 + uint16_t b = ((uint16_t)*pptr++ >> 4) & 0xf; 1.212 + img->palette[i] = r | g | b; 1.213 + } 1.214 + break; 1.215 + 1.216 + /* 1.217 + case IFF_CRNG: 1.218 + assert(hdr.size == sizeof crng); 1.219 + 1.220 + if(read_crng(fp, &crng) == -1) { 1.221 + logmsg("failed to read color cycling range chunk\n"); 1.222 + return -1; 1.223 + } 1.224 + if(crng.low != crng.high && crng.rate > 0) { 1.225 + if(!(crnode = malloc(sizeof *crnode))) { 1.226 + logmsg("failed to allocate color range node\n"); 1.227 + return -1; 1.228 + } 1.229 + crnode->low = crng.low; 1.230 + crnode->high = crng.high; 1.231 + crnode->cmode = (crng.flags & CRNG_REVERSE) ? CYCLE_REVERSE : CYCLE_NORMAL; 1.232 + crnode->rate = crng.rate; 1.233 + crnode->next = img->range; 1.234 + img->range = crnode; 1.235 + ++img->num_ranges; 1.236 + } 1.237 + break; 1.238 + */ 1.239 + 1.240 + case IFF_BODY: 1.241 + if(!img->pixels) { 1.242 + logmsg("malformed ILBM image: encountered BODY chunk before BMHD\n"); 1.243 + return -1; 1.244 + } 1.245 + if(type == IFF_ILBM) { 1.246 + if(read_body_ilbm(fp, &bmhd, img) == -1) { 1.247 + logmsg("failed to read interleaved pixel data\n"); 1.248 + return -1; 1.249 + } 1.250 + } else { 1.251 + assert(type == IFF_PBM); 1.252 + if(read_body_pbm(fp, &bmhd, img) == -1) { 1.253 + logmsg("failed to read linear pixel data\n"); 1.254 + return -1; 1.255 + } 1.256 + } 1.257 + res = 0; /* sucessfully read image */ 1.258 + break; 1.259 + 1.260 + default: 1.261 + /* skip unknown chunks */ 1.262 + fseek(fp, hdr.size, SEEK_CUR); 1.263 + if(ftell(fp) & 1) { 1.264 + /* chunks must start at even offsets */ 1.265 + fseek(fp, 1, SEEK_CUR); 1.266 + } 1.267 + } 1.268 + } 1.269 + 1.270 + return res; 1.271 +} 1.272 + 1.273 + 1.274 +static int read_bmhd(FILE *fp, struct bitmap_header *bmhd) 1.275 +{ 1.276 + if(fread(bmhd, sizeof *bmhd, 1, fp) < 1) { 1.277 + return -1; 1.278 + } 1.279 + bmhd->width = ntohs(bmhd->width); 1.280 + bmhd->height = ntohs(bmhd->height); 1.281 + bmhd->xoffs = ntohs(bmhd->xoffs); 1.282 + bmhd->yoffs = ntohs(bmhd->yoffs); 1.283 + bmhd->colorkey = ntohs(bmhd->colorkey); 1.284 + bmhd->pgwidth = ntohs(bmhd->pgwidth); 1.285 + bmhd->pgheight = ntohs(bmhd->pgheight); 1.286 + return 0; 1.287 +} 1.288 + 1.289 +static int read_crng(FILE *fp, struct crng *crng) 1.290 +{ 1.291 + if(fread(crng, sizeof *crng, 1, fp) < 1) { 1.292 + return -1; 1.293 + } 1.294 + crng->rate = ntohs(crng->rate); 1.295 + crng->flags = ntohs(crng->flags); 1.296 + return 0; 1.297 +} 1.298 + 1.299 +/* scanline: [bp0 row][bp1 row]...[bpN-1 row][opt mask row] 1.300 + * each uncompressed row is width / 8 bytes 1.301 + */ 1.302 +static int read_body_ilbm(FILE *fp, struct bitmap_header *bmhd, struct ham_image *img) 1.303 +{ 1.304 + int i, j; 1.305 + int rowsz = img->width / 8; 1.306 + unsigned char *dest = img->pixels; 1.307 + 1.308 + assert(bmhd->width == img->width); 1.309 + assert(bmhd->height == img->height); 1.310 + assert(img->pixels); 1.311 + 1.312 + for(i=0; i<img->height; i++) { 1.313 + 1.314 + for(j=0; j<bmhd->nplanes; j++) { 1.315 + /* read a row corresponding to bitplane j */ 1.316 + if(bmhd->compression) { 1.317 + if(read_compressed_scanline(fp, dest, rowsz) == -1) { 1.318 + return -1; 1.319 + } 1.320 + } else { 1.321 + if(fread(dest, 1, rowsz, fp) < rowsz) { 1.322 + return -1; 1.323 + } 1.324 + } 1.325 + dest += rowsz; 1.326 + } 1.327 + 1.328 + if(bmhd->masking & MASK_PLANE) { 1.329 + /* skip the mask (1bpp) */ 1.330 + fseek(fp, rowsz, SEEK_CUR); 1.331 + } 1.332 + } 1.333 + return 0; 1.334 +} 1.335 + 1.336 +static int read_body_pbm(FILE *fp, struct bitmap_header *bmhd, struct ham_image *img) 1.337 +{ 1.338 + /* 1.339 + int i; 1.340 + int npixels = img->width * img->height; 1.341 + unsigned char *dptr = img->pixels; 1.342 + 1.343 + assert(bmhd->width == img->width); 1.344 + assert(bmhd->height == img->height); 1.345 + assert(img->pixels); 1.346 + 1.347 + if(bmhd->compression) { 1.348 + for(i=0; i<img->height; i++) { 1.349 + if(read_compressed_scanline(fp, dptr, img->width) == -1) { 1.350 + return -1; 1.351 + } 1.352 + dptr += img->width; 1.353 + } 1.354 + 1.355 + } else { 1.356 + if(fread(img->pixels, 1, npixels, fp) < npixels) { 1.357 + return -1; 1.358 + } 1.359 + } 1.360 + return 0; 1.361 + */ 1.362 + return -1; /* TODO */ 1.363 +} 1.364 + 1.365 +static int read_compressed_scanline(FILE *fp, unsigned char *scanline, int width) 1.366 +{ 1.367 + int i, count, x = 0; 1.368 + signed char ctl; 1.369 + 1.370 + while(x < width) { 1.371 + if(fread(&ctl, 1, 1, fp) < 1) return -1; 1.372 + 1.373 + if(ctl == -128) continue; 1.374 + 1.375 + if(ctl >= 0) { 1.376 + count = ctl + 1; 1.377 + if(fread(scanline, 1, count, fp) < count) return -1; 1.378 + scanline += count; 1.379 + 1.380 + } else { 1.381 + unsigned char pixel; 1.382 + count = 1 - ctl; 1.383 + if(fread(&pixel, 1, 1, fp) < 1) return -1; 1.384 + 1.385 + for(i=0; i<count; i++) { 1.386 + *scanline++ = pixel; 1.387 + } 1.388 + } 1.389 + 1.390 + x += count; 1.391 + } 1.392 + 1.393 + return 0; 1.394 +}