* grub-core/video/readers/png.c: Support narrow (4-/2-/1-bpp) PNG.

This commit is contained in:
Vladimir 'phcoder' Serbinenko 2013-10-02 13:22:56 +02:00
parent be74aeb39e
commit c573914389
2 changed files with 74 additions and 19 deletions

View file

@ -1,3 +1,7 @@
2013-10-02 Vladimir Serbinenko <phcoder@gmail.com>
* grub-core/video/readers/png.c: Support narrow (4-/2-/1-bpp) PNG.
2013-10-01 Vladimir Testov <vladimir.testov@rosalab.ru> 2013-10-01 Vladimir Testov <vladimir.testov@rosalab.ru>
* grub-core/tests/checksums.h: Corrected due to changes in * grub-core/tests/checksums.h: Corrected due to changes in

View file

@ -99,6 +99,7 @@ struct grub_png_data
int image_width, image_height, bpp, is_16bit; int image_width, image_height, bpp, is_16bit;
int raw_bytes, is_gray, is_alpha, is_palette; int raw_bytes, is_gray, is_alpha, is_palette;
int row_bytes, color_bits;
grub_uint8_t *image_data; grub_uint8_t *image_data;
int inside_idat, idat_remain; int inside_idat, idat_remain;
@ -253,12 +254,10 @@ grub_png_decode_image_header (struct grub_png_data *data)
return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: invalid image size"); return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: invalid image size");
color_bits = grub_png_get_byte (data); color_bits = grub_png_get_byte (data);
if ((color_bits != 8) && (color_bits != 16))
return grub_error (GRUB_ERR_BAD_FILE_TYPE,
"png: bit depth must be 8 or 16");
data->is_16bit = (color_bits == 16); data->is_16bit = (color_bits == 16);
color_type = grub_png_get_byte (data); color_type = grub_png_get_byte (data);
/* According to PNG spec, no other types are valid. */ /* According to PNG spec, no other types are valid. */
if ((color_type & ~(PNG_COLOR_MASK_ALPHA | PNG_COLOR_MASK_COLOR)) if ((color_type & ~(PNG_COLOR_MASK_ALPHA | PNG_COLOR_MASK_COLOR))
&& (color_type != PNG_COLOR_TYPE_PALETTE)) && (color_type != PNG_COLOR_TYPE_PALETTE))
@ -283,6 +282,12 @@ grub_png_decode_image_header (struct grub_png_data *data)
data->bpp = 1; data->bpp = 1;
} }
if ((color_bits != 8) && (color_bits != 16)
&& (color_bits != 4
|| !(data->is_gray || data->is_palette)))
return grub_error (GRUB_ERR_BAD_FILE_TYPE,
"png: bit depth must be 8 or 16");
if (color_type & PNG_COLOR_MASK_ALPHA) if (color_type & PNG_COLOR_MASK_ALPHA)
data->bpp++; data->bpp++;
@ -294,12 +299,16 @@ grub_png_decode_image_header (struct grub_png_data *data)
if (data->is_16bit) if (data->is_16bit)
data->bpp <<= 1; data->bpp <<= 1;
data->color_bits = color_bits;
data->row_bytes = data->image_width * data->bpp;
if (data->color_bits <= 4)
data->row_bytes = (data->image_width * data->color_bits + 7) / 8;
#ifndef GRUB_CPU_WORDS_BIGENDIAN #ifndef GRUB_CPU_WORDS_BIGENDIAN
if (data->is_16bit || data->is_gray || data->is_palette) if (data->is_16bit || data->is_gray || data->is_palette)
#endif #endif
{ {
data->image_data = grub_malloc (data->image_height * data->image_data = grub_malloc (data->image_height * data->row_bytes);
data->image_width * data->bpp);
if (grub_errno) if (grub_errno)
return grub_errno; return grub_errno;
@ -313,7 +322,7 @@ grub_png_decode_image_header (struct grub_png_data *data)
} }
#endif #endif
data->raw_bytes = data->image_height * (data->image_width * data->bpp + 1); data->raw_bytes = data->image_height * (data->row_bytes + 1);
data->cur_column = 0; data->cur_column = 0;
data->first_line = 1; data->first_line = 1;
@ -567,8 +576,6 @@ grub_png_init_dynamic_block (struct grub_png_data *data)
static grub_err_t static grub_err_t
grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n) grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n)
{ {
int row_bytes;
if (--data->raw_bytes < 0) if (--data->raw_bytes < 0)
return grub_error (GRUB_ERR_BAD_FILE_TYPE, "image size overflown"); return grub_error (GRUB_ERR_BAD_FILE_TYPE, "image size overflown");
@ -583,24 +590,23 @@ grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n)
*data->cur_rgb++ = n; *data->cur_rgb++ = n;
data->cur_column++; data->cur_column++;
row_bytes = data->image_width * data->bpp; if (data->cur_column == data->row_bytes + 1)
if (data->cur_column == row_bytes + 1)
{ {
grub_uint8_t *blank_line = NULL; grub_uint8_t *blank_line = NULL;
grub_uint8_t *cur = data->cur_rgb - row_bytes; grub_uint8_t *cur = data->cur_rgb - data->row_bytes;
grub_uint8_t *left = cur; grub_uint8_t *left = cur;
grub_uint8_t *up; grub_uint8_t *up;
if (data->first_line) if (data->first_line)
{ {
blank_line = grub_zalloc (row_bytes); blank_line = grub_zalloc (data->row_bytes);
if (blank_line == NULL) if (blank_line == NULL)
return grub_errno; return grub_errno;
up = blank_line; up = blank_line;
} }
else else
up = cur - row_bytes; up = cur - data->row_bytes;
switch (data->cur_filter) switch (data->cur_filter)
{ {
@ -609,7 +615,7 @@ grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n)
int i; int i;
cur += data->bpp; cur += data->bpp;
for (i = data->bpp; i < row_bytes; i++, cur++, left++) for (i = data->bpp; i < data->row_bytes; i++, cur++, left++)
*cur += *left; *cur += *left;
break; break;
@ -618,7 +624,7 @@ grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n)
{ {
int i; int i;
for (i = 0; i < row_bytes; i++, cur++, up++) for (i = 0; i < data->row_bytes; i++, cur++, up++)
*cur += *up; *cur += *up;
break; break;
@ -630,7 +636,7 @@ grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n)
for (i = 0; i < data->bpp; i++, cur++, up++) for (i = 0; i < data->bpp; i++, cur++, up++)
*cur += *up >> 1; *cur += *up >> 1;
for (; i < row_bytes; i++, cur++, up++, left++) for (; i < data->row_bytes; i++, cur++, up++, left++)
*cur += ((int) *up + (int) *left) >> 1; *cur += ((int) *up + (int) *left) >> 1;
break; break;
@ -643,7 +649,7 @@ grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n)
for (i = 0; i < data->bpp; i++, cur++, up++) for (i = 0; i < data->bpp; i++, cur++, up++)
*cur += *up; *cur += *up;
for (; i < row_bytes; i++, cur++, up++, left++, upper_left++) for (; i < data->row_bytes; i++, cur++, up++, left++, upper_left++)
{ {
int a, b, c, pa, pb, pc; int a, b, c, pa, pb, pc;
@ -836,6 +842,48 @@ grub_png_convert_image (struct grub_png_data *data)
#define B3 2 #define B3 2
#endif #endif
if (data->color_bits <= 4)
{
grub_uint8_t palette[16][3];
grub_uint8_t *d1c, *d2c;
int shift;
int mask = (1 << data->color_bits) - 1;
int j;
if (data->is_gray)
for (i = 0; i < (1 << data->color_bits); i++)
{
grub_uint8_t col = (0xff * i) / ((1 << data->color_bits) - 1);
palette[i][0] = col;
palette[i][1] = col;
palette[i][2] = col;
}
else
grub_memcpy (palette, data->palette, 16 * 3);
d1c = d1;
d2c = d2;
for (j = 0; j < data->image_height; j++, d1c += data->image_width * 3,
d2c += data->row_bytes)
{
d1 = d1c;
d2 = d2c;
shift = 8 - data->color_bits;
for (i = 0; i < data->image_width; i++, d1 += 3)
{
grub_uint8_t col = (d2[0] >> shift) & mask;
d1[R3] = data->palette[col][2];
d1[G3] = data->palette[col][1];
d1[B3] = data->palette[col][0];
shift -= data->color_bits;
if (shift < 0)
{
d2++;
shift += 8;
}
}
}
return;
}
if (data->is_palette) if (data->is_palette)
{ {
for (i = 0; i < (data->image_width * data->image_height); for (i = 0; i < (data->image_width * data->image_height);
@ -845,8 +893,10 @@ grub_png_convert_image (struct grub_png_data *data)
d1[G3] = data->palette[d2[0]][1]; d1[G3] = data->palette[d2[0]][1];
d1[B3] = data->palette[d2[0]][0]; d1[B3] = data->palette[d2[0]][0];
} }
return;
} }
else if (data->is_gray)
if (data->is_gray)
{ {
switch (data->bpp) switch (data->bpp)
{ {
@ -896,8 +946,9 @@ grub_png_convert_image (struct grub_png_data *data)
} }
break; break;
} }
return;
} }
else
{ {
/* Only copy the upper 8 bit. */ /* Only copy the upper 8 bit. */
#ifndef GRUB_CPU_WORDS_BIGENDIAN #ifndef GRUB_CPU_WORDS_BIGENDIAN