png: support more color types; fix idat sequencing

This commit is contained in:
K. Lange 2020-05-30 16:36:31 +09:00
parent 69324a4e3d
commit 431a6aaf2a

215
lib/png.c
View File

@ -98,6 +98,8 @@ static uint8_t _get(struct inflate_context * ctx) {
unsigned int size = read_32(c->f);
unsigned int type = read_32(c->f);
c->size = size;
if (type != PNG_IDAT) {
/* This isn't an IDAT? That's wrong! */
fprintf(stderr, "And this is the wrong type (0x%x), I'm just bailing.\n", type);
@ -131,27 +133,22 @@ static int paeth(int a, int b, int c) {
return c;
}
/**
* Handle decompressed output from the inflater
*
* Writes pixel data to the image, and applies relevant filters.
*/
static void _write(struct inflate_context * ctx, unsigned int sym) {
struct png_ctx * c = (ctx->input_priv);
static void write_pixel(struct png_ctx * c, uint32_t color) {
SPRITE((c->sprite), (c->x), (c->y)) = color;
/* Put this byte into the short buffer */
c->buffer[c->buf_off] = sym;
c->buf_off++;
/* If this is the beginning of a scanline... */
if (c->x == -1 && c->buf_off == 1) {
/* Then this is the scanline filter type */
c->sf = sym;
/* Reset the buffer, advance to the beginning of the actual scanline */
c->x = 0;
/* Reset the short buffer */
c->buf_off = 0;
} else if (c->buf_off == 4) {
/* Advance to next pixel */
c->x++;
if (c->x == (int)c->width) {
/* Advance to next line; next read is scanline filter type */
c->x = -1;
c->y++;
}
}
static void process_pixel_type_6(struct png_ctx * c) {
/*
* Obtain pixel data from short buffer;
* For color type 6, this is always in R G B A order in the
@ -204,18 +201,172 @@ static void _write(struct inflate_context * ctx, unsigned int sym) {
}
/* Write new pixel to the image */
SPRITE((c->sprite), (c->x), (c->y)) = rgba(r,g,b,a);
/* Reset the short buffer */
c->buf_off = 0;
/* Advance to next pixel */
c->x++;
if (c->x == (int)c->width) {
/* Advance to next line; next read is scanline filter type */
c->x = -1;
c->y++;
write_pixel(c, rgba(r,g,b,a));
}
static void process_pixel_type_2(struct png_ctx * c) {
/*
* Obtain pixel data from short buffer;
* For color type 6, this is always in R G B A order in the
* bytestream, so we don't have to worry about subpixel ordering
* or weird color masks.
*/
unsigned int r = c->buffer[0];
unsigned int g = c->buffer[1];
unsigned int b = c->buffer[2];
/* Apply filters */
if (c->sf == PNG_FILTER_SUB) {
/* Add raw value to the pixel on the left */
if (c->x > 0) {
uint32_t left = SPRITE((c->sprite), (c->x - 1), (c->y));
r += _RED(left);
g += _GRE(left);
b += _BLU(left);
}
} else if (c->sf == PNG_FILTER_UP) {
/* Add raw value to the pixel above */
if (c->y > 0) {
uint32_t up = SPRITE((c->sprite), (c->x), (c->y - 1));
r += _RED(up);
g += _GRE(up);
b += _BLU(up);
}
} else if (c->sf == PNG_FILTER_AVG) {
/* Add raw value to the average of the pixel above and left */
uint32_t left = (c->x > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y)) : 0;
uint32_t up = (c->y > 0) ? SPRITE((c->sprite), (c->x), (c->y - 1)) : 0;
r += ((int)_RED(left) + (int)_RED(up)) / 2;
g += ((int)_GRE(left) + (int)_GRE(up)) / 2;
b += ((int)_BLU(left) + (int)_BLU(up)) / 2;
} else if (c->sf == PNG_FILTER_PAETH) {
/* Use the Paeth predictor */
uint32_t left = (c->x > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y)) : 0;
uint32_t up = (c->y > 0) ? SPRITE((c->sprite), (c->x), (c->y - 1)) : 0;
uint32_t upleft = (c->x > 0 && c->y > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y - 1)) : 0;
r = ((int)r + paeth((int)_RED(left),(int)_RED(up),(int)_RED(upleft))) % 256;
g = ((int)g + paeth((int)_GRE(left),(int)_GRE(up),(int)_GRE(upleft))) % 256;
b = ((int)b + paeth((int)_BLU(left),(int)_BLU(up),(int)_BLU(upleft))) % 256;
}
/* Write new pixel to the image */
write_pixel(c, rgb(r,g,b));
}
static void process_pixel_type_4(struct png_ctx * c) {
/*
* Obtain pixel data from short buffer;
* For color type 6, this is always in R G B A order in the
* bytestream, so we don't have to worry about subpixel ordering
* or weird color masks.
*/
unsigned int b = c->buffer[0];
unsigned int a = c->buffer[1];
/* Apply filters */
if (c->sf == PNG_FILTER_SUB) {
/* Add raw value to the pixel on the left */
if (c->x > 0) {
uint32_t left = SPRITE((c->sprite), (c->x - 1), (c->y));
b += _BLU(left);
a += _ALP(left);
}
} else if (c->sf == PNG_FILTER_UP) {
/* Add raw value to the pixel above */
if (c->y > 0) {
uint32_t up = SPRITE((c->sprite), (c->x), (c->y - 1));
b += _BLU(up);
a += _ALP(up);
}
} else if (c->sf == PNG_FILTER_AVG) {
/* Add raw value to the average of the pixel above and left */
uint32_t left = (c->x > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y)) : 0;
uint32_t up = (c->y > 0) ? SPRITE((c->sprite), (c->x), (c->y - 1)) : 0;
b += ((int)_BLU(left) + (int)_BLU(up)) / 2;
a += ((int)_ALP(left) + (int)_ALP(up)) / 2;
} else if (c->sf == PNG_FILTER_PAETH) {
/* Use the Paeth predictor */
uint32_t left = (c->x > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y)) : 0;
uint32_t up = (c->y > 0) ? SPRITE((c->sprite), (c->x), (c->y - 1)) : 0;
uint32_t upleft = (c->x > 0 && c->y > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y - 1)) : 0;
b = ((int)b + paeth((int)_BLU(left),(int)_BLU(up),(int)_BLU(upleft))) % 256;
a = ((int)a + paeth((int)_ALP(left),(int)_ALP(up),(int)_ALP(upleft))) % 256;
}
/* Write new pixel to the image */
write_pixel(c, rgba(b,b,b,a));
}
static void process_pixel_type_0(struct png_ctx * c) {
unsigned int b = c->buffer[0];
if (c->sf == PNG_FILTER_SUB) {
if (c->x > 0) {
uint32_t left = SPRITE((c->sprite), (c->x - 1), (c->y));
b += _BLU(left);
}
} else if (c->sf == PNG_FILTER_UP) {
if (c->y > 0) {
uint32_t up = SPRITE((c->sprite), (c->x), (c->y - 1));
b += _BLU(up);
}
} else if (c->sf == PNG_FILTER_AVG) {
uint32_t left = (c->x > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y)) : 0;
uint32_t up = (c->y > 0) ? SPRITE((c->sprite), (c->x), (c->y - 1)) : 0;
b += ((int)_BLU(left) + (int)_BLU(up)) / 2;
} else if (c->sf == PNG_FILTER_PAETH) {
uint32_t left = (c->x > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y)) : 0;
uint32_t up = (c->y > 0) ? SPRITE((c->sprite), (c->x), (c->y - 1)) : 0;
uint32_t upleft = (c->x > 0 && c->y > 0) ? SPRITE((c->sprite), (c->x - 1), (c->y - 1)) : 0;
b = ((int)b + paeth((int)_BLU(left),(int)_BLU(up),(int)_BLU(upleft))) % 256;
}
/* Write new pixel to the image */
write_pixel(c, rgb(b,b,b));
}
/**
* Handle decompressed output from the inflater
*
* Writes pixel data to the image, and applies relevant filters.
*/
static void _write(struct inflate_context * ctx, unsigned int sym) {
struct png_ctx * c = (ctx->input_priv);
/* Put this byte into the short buffer */
c->buffer[c->buf_off] = sym;
c->buf_off++;
/* If this is the beginning of a scanline... */
if (c->x == -1 && c->buf_off == 1) {
/* Then this is the scanline filter type */
c->sf = sym;
/* Reset the buffer, advance to the beginning of the actual scanline */
c->x = 0;
c->buf_off = 0;
} else if (c->buf_off == 1 && c->color_type == 0) {
process_pixel_type_0(c);
} else if (c->buf_off == 2 && c->color_type == 4) {
process_pixel_type_4(c);
} else if (c->buf_off == 3 && c->color_type == 2) {
process_pixel_type_2(c);
} else if (c->buf_off == 4 && c->color_type == 6) {
process_pixel_type_6(c);
}
}
static int color_type_has_alpha(int c) {
switch (c) {
case 4:
case 6:
return ALPHA_EMBEDDED;
default:
return 0;
}
}
@ -275,14 +426,14 @@ int load_sprite_png(sprite_t * sprite, char * filename) {
if (c.interlace != 0 && c.interlace != 1) return 1;
if (c.bit_depth != 8) return 1; /* Sorry */
if (c.color_type != 6) return 1; /* Sorry */
if (c.color_type < 0 || c.color_type > 6 || (c.color_type & 1)) return 1; /* Sorry, no indexed support */
/* Allocate space */
sprite->width = c.width;
sprite->height = c.height;
sprite->bitmap = malloc(sizeof(uint32_t) * sprite->width * sprite->height);
sprite->masks = NULL;
sprite->alpha = (c.color_type == 4 || c.color_type == 6) ? ALPHA_EMBEDDED : 0;
sprite->alpha = color_type_has_alpha(c.color_type);
sprite->blank = 0;