/* * GRUB -- GRand Unified Bootloader * Copyright (C) 2008,2009 Free Software Foundation, Inc. * * GRUB is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * GRUB is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with GRUB. If not, see <http://www.gnu.org/licenses/>. */ #include <grub/bitmap.h> #include <grub/types.h> #include <grub/normal.h> #include <grub/dl.h> #include <grub/mm.h> #include <grub/misc.h> #include <grub/bufio.h> GRUB_MOD_LICENSE ("GPLv3+"); /* Uncomment following define to enable PNG debug. */ //#define PNG_DEBUG enum { PNG_COLOR_TYPE_GRAY = 0, PNG_COLOR_MASK_PALETTE = 1, PNG_COLOR_MASK_COLOR = 2, PNG_COLOR_MASK_ALPHA = 4, PNG_COLOR_TYPE_PALETTE = (PNG_COLOR_MASK_COLOR | PNG_COLOR_MASK_PALETTE), }; #define PNG_COMPRESSION_BASE 0 #define PNG_INTERLACE_NONE 0 #define PNG_INTERLACE_ADAM7 1 #define PNG_FILTER_TYPE_BASE 0 #define PNG_FILTER_VALUE_NONE 0 #define PNG_FILTER_VALUE_SUB 1 #define PNG_FILTER_VALUE_UP 2 #define PNG_FILTER_VALUE_AVG 3 #define PNG_FILTER_VALUE_PAETH 4 #define PNG_FILTER_VALUE_LAST 5 enum { PNG_CHUNK_IHDR = 0x49484452, PNG_CHUNK_IDAT = 0x49444154, PNG_CHUNK_IEND = 0x49454e44, PNG_CHUNK_PLTE = 0x504c5445 }; #define Z_DEFLATED 8 #define Z_FLAG_DICT 32 #define INFLATE_STORED 0 #define INFLATE_FIXED 1 #define INFLATE_DYNAMIC 2 #define WSIZE 0x8000 #define DEFLATE_HCLEN_BASE 4 #define DEFLATE_HCLEN_MAX 19 #define DEFLATE_HLIT_BASE 257 #define DEFLATE_HLIT_MAX 288 #define DEFLATE_HDIST_BASE 1 #define DEFLATE_HDIST_MAX 30 #define DEFLATE_HUFF_LEN 16 #ifdef PNG_DEBUG static grub_command_t cmd; #endif struct huff_table { int *values, *maxval, *offset; int num_values, max_length; }; struct grub_png_data { grub_file_t file; struct grub_video_bitmap **bitmap; int bit_count, bit_save; grub_uint32_t next_offset; unsigned image_width, image_height; int bpp, is_16bit; int raw_bytes, is_gray, is_alpha, is_palette; int row_bytes, color_bits; grub_uint8_t *image_data; int inside_idat, idat_remain; int code_values[DEFLATE_HLIT_MAX]; int code_maxval[DEFLATE_HUFF_LEN]; int code_offset[DEFLATE_HUFF_LEN]; int dist_values[DEFLATE_HDIST_MAX]; int dist_maxval[DEFLATE_HUFF_LEN]; int dist_offset[DEFLATE_HUFF_LEN]; grub_uint8_t palette[256][3]; struct huff_table code_table; struct huff_table dist_table; grub_uint8_t slide[WSIZE]; int wp; grub_uint8_t *cur_rgb; int cur_column, cur_filter, first_line; }; static grub_uint32_t grub_png_get_dword (struct grub_png_data *data) { grub_uint32_t r; r = 0; grub_file_read (data->file, &r, sizeof (grub_uint32_t)); return grub_be_to_cpu32 (r); } static grub_uint8_t grub_png_get_byte (struct grub_png_data *data) { grub_uint8_t r; if ((data->inside_idat) && (data->idat_remain == 0)) { grub_uint32_t len, type; do { /* Skip crc checksum. */ grub_png_get_dword (data); if (data->file->offset != data->next_offset) { grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: chunk size error"); return 0; } len = grub_png_get_dword (data); type = grub_png_get_dword (data); if (type != PNG_CHUNK_IDAT) { grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: unexpected end of data"); return 0; } data->next_offset = data->file->offset + len + 4; } while (len == 0); data->idat_remain = len; } r = 0; grub_file_read (data->file, &r, 1); if (data->inside_idat) data->idat_remain--; return r; } static int grub_png_get_bits (struct grub_png_data *data, int num) { int code, shift; if (data->bit_count == 0) { data->bit_save = grub_png_get_byte (data); data->bit_count = 8; } code = 0; shift = 0; while (grub_errno == 0) { int n; n = data->bit_count; if (n > num) n = num; code += (int) (data->bit_save & ((1 << n) - 1)) << shift; num -= n; if (!num) { data->bit_count -= n; data->bit_save >>= n; break; } shift += n; data->bit_save = grub_png_get_byte (data); data->bit_count = 8; } return code; } static grub_err_t grub_png_decode_image_palette (struct grub_png_data *data, unsigned len) { unsigned i = 0, j; if (len == 0) return GRUB_ERR_NONE; for (i = 0; 3 * i < len && i < 256; i++) for (j = 0; j < 3; j++) data->palette[i][j] = grub_png_get_byte (data); for (i *= 3; i < len; i++) grub_png_get_byte (data); grub_png_get_dword (data); return GRUB_ERR_NONE; } static grub_err_t grub_png_decode_image_header (struct grub_png_data *data) { int color_type; int color_bits; enum grub_video_blit_format blt; data->image_width = grub_png_get_dword (data); data->image_height = grub_png_get_dword (data); if ((!data->image_height) || (!data->image_width)) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: invalid image size"); color_bits = grub_png_get_byte (data); data->is_16bit = (color_bits == 16); color_type = grub_png_get_byte (data); /* According to PNG spec, no other types are valid. */ if ((color_type & ~(PNG_COLOR_MASK_ALPHA | PNG_COLOR_MASK_COLOR)) && (color_type != PNG_COLOR_TYPE_PALETTE)) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: color type not supported"); if (color_type == PNG_COLOR_TYPE_PALETTE) data->is_palette = 1; if (data->is_16bit && data->is_palette) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: color type not supported"); if (color_type & PNG_COLOR_MASK_ALPHA) blt = GRUB_VIDEO_BLIT_FORMAT_RGBA_8888; else blt = GRUB_VIDEO_BLIT_FORMAT_RGB_888; if (data->is_palette) data->bpp = 1; else if (color_type & PNG_COLOR_MASK_COLOR) data->bpp = 3; else { data->is_gray = 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) data->bpp++; if (grub_video_bitmap_create (data->bitmap, data->image_width, data->image_height, blt)) return grub_errno; if (data->is_16bit) 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 if (data->is_16bit || data->is_gray || data->is_palette) #endif { data->image_data = grub_malloc (data->image_height * data->row_bytes); if (grub_errno) return grub_errno; data->cur_rgb = data->image_data; } #ifndef GRUB_CPU_WORDS_BIGENDIAN else { data->image_data = 0; data->cur_rgb = (*data->bitmap)->data; } #endif data->raw_bytes = data->image_height * (data->row_bytes + 1); data->cur_column = 0; data->first_line = 1; if (grub_png_get_byte (data) != PNG_COMPRESSION_BASE) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: compression method not supported"); if (grub_png_get_byte (data) != PNG_FILTER_TYPE_BASE) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: filter method not supported"); if (grub_png_get_byte (data) != PNG_INTERLACE_NONE) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: interlace method not supported"); /* Skip crc checksum. */ grub_png_get_dword (data); return grub_errno; } /* Order of the bit length code lengths. */ static const grub_uint8_t bitorder[] = { 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15 }; /* Copy lengths for literal codes 257..285. */ static const int cplens[] = { 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31, 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0 }; /* Extra bits for literal codes 257..285. */ static const grub_uint8_t cplext[] = { 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 99, 99 }; /* 99==invalid */ /* Copy offsets for distance codes 0..29. */ static const int cpdist[] = { 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, 8193, 12289, 16385, 24577 }; /* Extra bits for distance codes. */ static const grub_uint8_t cpdext[] = { 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13 }; static void grub_png_init_huff_table (struct huff_table *ht, int cur_maxlen, int *cur_values, int *cur_maxval, int *cur_offset) { ht->values = cur_values; ht->maxval = cur_maxval; ht->offset = cur_offset; ht->num_values = 0; ht->max_length = cur_maxlen; grub_memset (cur_maxval, 0, sizeof (int) * cur_maxlen); } static void grub_png_insert_huff_item (struct huff_table *ht, int code, int len) { int i, n; if (len == 0) return; if (len > ht->max_length) { grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: invalid code length"); return; } n = 0; for (i = len; i < ht->max_length; i++) n += ht->maxval[i]; for (i = 0; i < n; i++) ht->values[ht->num_values - i] = ht->values[ht->num_values - i - 1]; ht->values[ht->num_values - n] = code; ht->num_values++; ht->maxval[len - 1]++; } static void grub_png_build_huff_table (struct huff_table *ht) { int base, ofs, i; base = 0; ofs = 0; for (i = 0; i < ht->max_length; i++) { base += ht->maxval[i]; ofs += ht->maxval[i]; ht->maxval[i] = base; ht->offset[i] = ofs - base; base <<= 1; } } static int grub_png_get_huff_code (struct grub_png_data *data, struct huff_table *ht) { int code, i; code = 0; for (i = 0; i < ht->max_length; i++) { code = (code << 1) + grub_png_get_bits (data, 1); if (code < ht->maxval[i]) return ht->values[code + ht->offset[i]]; } return 0; } static grub_err_t grub_png_init_fixed_block (struct grub_png_data *data) { int i; grub_png_init_huff_table (&data->code_table, DEFLATE_HUFF_LEN, data->code_values, data->code_maxval, data->code_offset); for (i = 0; i < 144; i++) grub_png_insert_huff_item (&data->code_table, i, 8); for (; i < 256; i++) grub_png_insert_huff_item (&data->code_table, i, 9); for (; i < 280; i++) grub_png_insert_huff_item (&data->code_table, i, 7); for (; i < DEFLATE_HLIT_MAX; i++) grub_png_insert_huff_item (&data->code_table, i, 8); grub_png_build_huff_table (&data->code_table); grub_png_init_huff_table (&data->dist_table, DEFLATE_HUFF_LEN, data->dist_values, data->dist_maxval, data->dist_offset); for (i = 0; i < DEFLATE_HDIST_MAX; i++) grub_png_insert_huff_item (&data->dist_table, i, 5); grub_png_build_huff_table (&data->dist_table); return grub_errno; } static grub_err_t grub_png_init_dynamic_block (struct grub_png_data *data) { int nl, nd, nb, i, prev; struct huff_table cl; int cl_values[sizeof (bitorder)]; int cl_maxval[8]; int cl_offset[8]; grub_uint8_t lens[DEFLATE_HCLEN_MAX]; nl = DEFLATE_HLIT_BASE + grub_png_get_bits (data, 5); nd = DEFLATE_HDIST_BASE + grub_png_get_bits (data, 5); nb = DEFLATE_HCLEN_BASE + grub_png_get_bits (data, 4); if ((nl > DEFLATE_HLIT_MAX) || (nd > DEFLATE_HDIST_MAX) || (nb > DEFLATE_HCLEN_MAX)) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: too much data"); grub_png_init_huff_table (&cl, 8, cl_values, cl_maxval, cl_offset); for (i = 0; i < nb; i++) lens[bitorder[i]] = grub_png_get_bits (data, 3); for (; i < DEFLATE_HCLEN_MAX; i++) lens[bitorder[i]] = 0; for (i = 0; i < DEFLATE_HCLEN_MAX; i++) grub_png_insert_huff_item (&cl, i, lens[i]); grub_png_build_huff_table (&cl); grub_png_init_huff_table (&data->code_table, DEFLATE_HUFF_LEN, data->code_values, data->code_maxval, data->code_offset); grub_png_init_huff_table (&data->dist_table, DEFLATE_HUFF_LEN, data->dist_values, data->dist_maxval, data->dist_offset); prev = 0; for (i = 0; i < nl + nd; i++) { int n, code; struct huff_table *ht; if (grub_errno) return grub_errno; if (i < nl) { ht = &data->code_table; code = i; } else { ht = &data->dist_table; code = i - nl; } n = grub_png_get_huff_code (data, &cl); if (n < 16) { grub_png_insert_huff_item (ht, code, n); prev = n; } else if (n == 16) { int c; c = 3 + grub_png_get_bits (data, 2); while (c > 0) { grub_png_insert_huff_item (ht, code++, prev); i++; c--; } i--; } else if (n == 17) i += 3 + grub_png_get_bits (data, 3) - 1; else i += 11 + grub_png_get_bits (data, 7) - 1; } grub_png_build_huff_table (&data->code_table); grub_png_build_huff_table (&data->dist_table); return grub_errno; } static grub_err_t grub_png_output_byte (struct grub_png_data *data, grub_uint8_t n) { if (--data->raw_bytes < 0) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "image size overflown"); if (data->cur_column == 0) { if (n >= PNG_FILTER_VALUE_LAST) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "invalid filter value"); data->cur_filter = n; } else *data->cur_rgb++ = n; data->cur_column++; if (data->cur_column == data->row_bytes + 1) { grub_uint8_t *blank_line = NULL; grub_uint8_t *cur = data->cur_rgb - data->row_bytes; grub_uint8_t *left = cur; grub_uint8_t *up; if (data->first_line) { blank_line = grub_zalloc (data->row_bytes); if (blank_line == NULL) return grub_errno; up = blank_line; } else up = cur - data->row_bytes; switch (data->cur_filter) { case PNG_FILTER_VALUE_SUB: { int i; cur += data->bpp; for (i = data->bpp; i < data->row_bytes; i++, cur++, left++) *cur += *left; break; } case PNG_FILTER_VALUE_UP: { int i; for (i = 0; i < data->row_bytes; i++, cur++, up++) *cur += *up; break; } case PNG_FILTER_VALUE_AVG: { int i; for (i = 0; i < data->bpp; i++, cur++, up++) *cur += *up >> 1; for (; i < data->row_bytes; i++, cur++, up++, left++) *cur += ((int) *up + (int) *left) >> 1; break; } case PNG_FILTER_VALUE_PAETH: { int i; grub_uint8_t *upper_left = up; for (i = 0; i < data->bpp; i++, cur++, up++) *cur += *up; for (; i < data->row_bytes; i++, cur++, up++, left++, upper_left++) { int a, b, c, pa, pb, pc; a = *left; b = *up; c = *upper_left; pa = b - c; pb = a - c; pc = pa + pb; if (pa < 0) pa = -pa; if (pb < 0) pb = -pb; if (pc < 0) pc = -pc; *cur += ((pa <= pb) && (pa <= pc)) ? a : (pb <= pc) ? b : c; } } } grub_free (blank_line); data->cur_column = 0; data->first_line = 0; } return grub_errno; } static grub_err_t grub_png_read_dynamic_block (struct grub_png_data *data) { while (grub_errno == 0) { int n; n = grub_png_get_huff_code (data, &data->code_table); if (n < 256) { data->slide[data->wp] = n; grub_png_output_byte (data, n); data->wp++; if (data->wp >= WSIZE) data->wp = 0; } else if (n == 256) break; else { int len, dist, pos; n -= 257; len = cplens[n]; if (cplext[n]) len += grub_png_get_bits (data, cplext[n]); n = grub_png_get_huff_code (data, &data->dist_table); dist = cpdist[n]; if (cpdext[n]) dist += grub_png_get_bits (data, cpdext[n]); pos = data->wp - dist; if (pos < 0) pos += WSIZE; while (len > 0) { data->slide[data->wp] = data->slide[pos]; grub_png_output_byte (data, data->slide[data->wp]); data->wp++; if (data->wp >= WSIZE) data->wp = 0; pos++; if (pos >= WSIZE) pos = 0; len--; } } } return grub_errno; } static grub_err_t grub_png_decode_image_data (struct grub_png_data *data) { grub_uint8_t cmf, flg; int final; cmf = grub_png_get_byte (data); flg = grub_png_get_byte (data); if ((cmf & 0xF) != Z_DEFLATED) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: only support deflate compression method"); if (flg & Z_FLAG_DICT) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: dictionary not supported"); do { int block_type; final = grub_png_get_bits (data, 1); block_type = grub_png_get_bits (data, 2); switch (block_type) { case INFLATE_STORED: { grub_uint16_t i, len; data->bit_count = 0; len = grub_png_get_byte (data); len += ((grub_uint16_t) grub_png_get_byte (data)) << 8; /* Skip NLEN field. */ grub_png_get_byte (data); grub_png_get_byte (data); for (i = 0; i < len; i++) grub_png_output_byte (data, grub_png_get_byte (data)); break; } case INFLATE_FIXED: grub_png_init_fixed_block (data); grub_png_read_dynamic_block (data); break; case INFLATE_DYNAMIC: grub_png_init_dynamic_block (data); grub_png_read_dynamic_block (data); break; default: return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: unknown block type"); } } while ((!final) && (grub_errno == 0)); /* Skip adler checksum. */ grub_png_get_dword (data); /* Skip crc checksum. */ grub_png_get_dword (data); return grub_errno; } static const grub_uint8_t png_magic[8] = { 0x89, 0x50, 0x4e, 0x47, 0xd, 0xa, 0x1a, 0x0a }; static void grub_png_convert_image (struct grub_png_data *data) { unsigned i; grub_uint8_t *d1, *d2; d1 = (*data->bitmap)->data; d2 = data->image_data + data->is_16bit; #ifndef GRUB_CPU_WORDS_BIGENDIAN #define R4 3 #define G4 2 #define B4 1 #define A4 0 #define R3 2 #define G3 1 #define B3 0 #else #define R4 0 #define G4 1 #define B4 2 #define A4 3 #define R3 0 #define G3 1 #define B3 2 #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; unsigned j; if (data->is_gray) { /* Generic formula is (0xff * i) / ((1U << data->color_bits) - 1) but for allowed bit depth of 1, 2 and for it's equivalent to (0xff / ((1U << data->color_bits) - 1)) * i Precompute the multipliers to avoid division. */ const grub_uint8_t multipliers[5] = { 0xff, 0xff, 0x55, 0x24, 0x11 }; for (i = 0; i < (1U << data->color_bits); i++) { grub_uint8_t col = multipliers[data->color_bits] * i; palette[i][0] = col; palette[i][1] = col; palette[i][2] = col; } } else grub_memcpy (palette, data->palette, 3 << data->color_bits); 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) { for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 3, d2++) { d1[R3] = data->palette[d2[0]][2]; d1[G3] = data->palette[d2[0]][1]; d1[B3] = data->palette[d2[0]][0]; } return; } if (data->is_gray) { switch (data->bpp) { case 4: /* 16-bit gray with alpha. */ for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 4, d2 += 4) { d1[R4] = d2[3]; d1[G4] = d2[3]; d1[B4] = d2[3]; d1[A4] = d2[1]; } break; case 2: if (data->is_16bit) /* 16-bit gray without alpha. */ { for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 4, d2 += 2) { d1[R3] = d2[1]; d1[G3] = d2[1]; d1[B3] = d2[1]; } } else /* 8-bit gray with alpha. */ { for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 4, d2 += 2) { d1[R4] = d2[1]; d1[G4] = d2[1]; d1[B4] = d2[1]; d1[A4] = d2[0]; } } break; /* 8-bit gray without alpha. */ case 1: for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 3, d2++) { d1[R3] = d2[0]; d1[G3] = d2[0]; d1[B3] = d2[0]; } break; } return; } { /* Only copy the upper 8 bit. */ #ifndef GRUB_CPU_WORDS_BIGENDIAN for (i = 0; i < (data->image_width * data->image_height * data->bpp >> 1); i++, d1++, d2 += 2) *d1 = *d2; #else switch (data->bpp) { /* 16-bit with alpha. */ case 8: for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 4, d2+=8) { d1[0] = d2[7]; d1[1] = d2[5]; d1[2] = d2[3]; d1[3] = d2[1]; } break; /* 16-bit without alpha. */ case 6: for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 3, d2+=6) { d1[0] = d2[5]; d1[1] = d2[3]; d1[2] = d2[1]; } break; case 4: /* 8-bit with alpha. */ for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 4, d2 += 4) { d1[0] = d2[3]; d1[1] = d2[2]; d1[2] = d2[1]; d1[3] = d2[0]; } break; /* 8-bit without alpha. */ case 3: for (i = 0; i < (data->image_width * data->image_height); i++, d1 += 3, d2 += 3) { d1[0] = d2[2]; d1[1] = d2[1]; d1[2] = d2[0]; } break; } #endif } } static grub_err_t grub_png_decode_png (struct grub_png_data *data) { grub_uint8_t magic[8]; if (grub_file_read (data->file, &magic[0], 8) != 8) return grub_errno; if (grub_memcmp (magic, png_magic, sizeof (png_magic))) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: not a png file"); while (1) { grub_uint32_t len, type; len = grub_png_get_dword (data); type = grub_png_get_dword (data); data->next_offset = data->file->offset + len + 4; switch (type) { case PNG_CHUNK_IHDR: grub_png_decode_image_header (data); break; case PNG_CHUNK_PLTE: grub_png_decode_image_palette (data, len); break; case PNG_CHUNK_IDAT: data->inside_idat = 1; data->idat_remain = len; data->bit_count = 0; grub_png_decode_image_data (data); data->inside_idat = 0; break; case PNG_CHUNK_IEND: if (data->image_data) grub_png_convert_image (data); return grub_errno; default: grub_file_seek (data->file, data->file->offset + len + 4); } if (grub_errno) break; if (data->file->offset != data->next_offset) return grub_error (GRUB_ERR_BAD_FILE_TYPE, "png: chunk size error"); } return grub_errno; } static grub_err_t grub_video_reader_png (struct grub_video_bitmap **bitmap, const char *filename) { grub_file_t file; struct grub_png_data *data; file = grub_buffile_open (filename, 0); if (!file) return grub_errno; data = grub_zalloc (sizeof (*data)); if (data != NULL) { data->file = file; data->bitmap = bitmap; grub_png_decode_png (data); grub_free (data->image_data); grub_free (data); } if (grub_errno != GRUB_ERR_NONE) { grub_video_bitmap_destroy (*bitmap); *bitmap = 0; } grub_file_close (file); return grub_errno; } #if defined(PNG_DEBUG) static grub_err_t grub_cmd_pngtest (grub_command_t cmd_d __attribute__ ((unused)), int argc, char **args) { struct grub_video_bitmap *bitmap = 0; if (argc != 1) return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected")); grub_video_reader_png (&bitmap, args[0]); if (grub_errno != GRUB_ERR_NONE) return grub_errno; grub_video_bitmap_destroy (bitmap); return GRUB_ERR_NONE; } #endif static struct grub_video_bitmap_reader png_reader = { .extension = ".png", .reader = grub_video_reader_png, .next = 0 }; GRUB_MOD_INIT (png) { grub_video_bitmap_reader_register (&png_reader); #if defined(PNG_DEBUG) cmd = grub_register_command ("pngtest", grub_cmd_pngtest, "FILE", "Tests loading of PNG bitmap."); #endif } GRUB_MOD_FINI (png) { #if defined(PNG_DEBUG) grub_unregister_command (cmd); #endif grub_video_bitmap_reader_unregister (&png_reader); }