123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763 |
- /*
- * GRUB -- GRand Unified Bootloader
- * Copyright (C) 2008 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>
- /* Uncomment following define to enable JPEG debug. */
- //#define JPEG_DEBUG
- #define JPEG_ESC_CHAR 0xFF
- #define JPEG_SAMPLING_1x1 0x11
- #define JPEG_MARKER_SOI 0xd8
- #define JPEG_MARKER_EOI 0xd9
- #define JPEG_MARKER_DHT 0xc4
- #define JPEG_MARKER_DQT 0xdb
- #define JPEG_MARKER_SOF0 0xc0
- #define JPEG_MARKER_SOS 0xda
- #define SHIFT_BITS 8
- #define CONST(x) ((int) ((x) * (1L << SHIFT_BITS) + 0.5))
- #define JPEG_UNIT_SIZE 8
- static const grub_uint8_t jpeg_zigzag_order[64] = {
- 0, 1, 8, 16, 9, 2, 3, 10,
- 17, 24, 32, 25, 18, 11, 4, 5,
- 12, 19, 26, 33, 40, 48, 41, 34,
- 27, 20, 13, 6, 7, 14, 21, 28,
- 35, 42, 49, 56, 57, 50, 43, 36,
- 29, 22, 15, 23, 30, 37, 44, 51,
- 58, 59, 52, 45, 38, 31, 39, 46,
- 53, 60, 61, 54, 47, 55, 62, 63
- };
- #ifdef JPEG_DEBUG
- static grub_command_t cmd;
- #endif
- typedef int jpeg_data_unit_t[64];
- struct grub_jpeg_data
- {
- grub_file_t file;
- struct grub_video_bitmap **bitmap;
- int image_width;
- int image_height;
- grub_uint8_t *huff_value[4];
- int huff_offset[4][16];
- int huff_maxval[4][16];
- grub_uint8_t quan_table[2][64];
- int comp_index[3][3];
- jpeg_data_unit_t ydu[4];
- jpeg_data_unit_t crdu;
- jpeg_data_unit_t cbdu;
- int vs, hs;
- int dc_value[3];
- int bit_mask, bit_save;
- };
- static grub_uint8_t
- grub_jpeg_get_byte (struct grub_jpeg_data *data)
- {
- grub_uint8_t r;
- r = 0;
- grub_file_read (data->file, &r, 1);
- return r;
- }
- static grub_uint16_t
- grub_jpeg_get_word (struct grub_jpeg_data *data)
- {
- grub_uint16_t r;
- r = 0;
- grub_file_read (data->file, &r, sizeof (grub_uint16_t));
- return grub_be_to_cpu16 (r);
- }
- static int
- grub_jpeg_get_bit (struct grub_jpeg_data *data)
- {
- int ret;
- if (data->bit_mask == 0)
- {
- data->bit_save = grub_jpeg_get_byte (data);
- if (data->bit_save == JPEG_ESC_CHAR)
- {
- if (grub_jpeg_get_byte (data) != 0)
- {
- grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: invalid 0xFF in data stream");
- return 0;
- }
- }
- data->bit_mask = 0x80;
- }
- ret = ((data->bit_save & data->bit_mask) != 0);
- data->bit_mask >>= 1;
- return ret;
- }
- static int
- grub_jpeg_get_number (struct grub_jpeg_data *data, int num)
- {
- int value, i, msb;
- if (num == 0)
- return 0;
- msb = value = grub_jpeg_get_bit (data);
- for (i = 1; i < num; i++)
- value = (value << 1) + (grub_jpeg_get_bit (data) != 0);
- if (!msb)
- value += 1 - (1 << num);
- return value;
- }
- static int
- grub_jpeg_get_huff_code (struct grub_jpeg_data *data, int id)
- {
- int code;
- unsigned i;
- code = 0;
- for (i = 0; i < ARRAY_SIZE (data->huff_maxval[id]); i++)
- {
- code <<= 1;
- if (grub_jpeg_get_bit (data))
- code++;
- if (code < data->huff_maxval[id][i])
- return data->huff_value[id][code + data->huff_offset[id][i]];
- }
- grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: huffman decode fails");
- return 0;
- }
- static grub_err_t
- grub_jpeg_decode_huff_table (struct grub_jpeg_data *data)
- {
- int id, ac, n, base, ofs;
- grub_uint32_t next_marker;
- grub_uint8_t count[16];
- unsigned i;
- next_marker = data->file->offset;
- next_marker += grub_jpeg_get_word (data);
- while (data->file->offset + sizeof (count) + 1 <= next_marker)
- {
- id = grub_jpeg_get_byte (data);
- ac = (id >> 4) & 1;
- id &= 0xF;
- if (id > 1)
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: too many huffman tables");
- if (grub_file_read (data->file, &count, sizeof (count)) !=
- sizeof (count))
- return grub_errno;
- n = 0;
- for (i = 0; i < ARRAY_SIZE (count); i++)
- n += count[i];
- id += ac * 2;
- data->huff_value[id] = grub_malloc (n);
- if (grub_errno)
- return grub_errno;
- if (grub_file_read (data->file, data->huff_value[id], n) != n)
- return grub_errno;
- base = 0;
- ofs = 0;
- for (i = 0; i < ARRAY_SIZE (count); i++)
- {
- base += count[i];
- ofs += count[i];
- data->huff_maxval[id][i] = base;
- data->huff_offset[id][i] = ofs - base;
- base <<= 1;
- }
- }
- if (data->file->offset != next_marker)
- grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: extra byte in huffman table");
- return grub_errno;
- }
- static grub_err_t
- grub_jpeg_decode_quan_table (struct grub_jpeg_data *data)
- {
- int id;
- grub_uint32_t next_marker;
- next_marker = data->file->offset;
- next_marker += grub_jpeg_get_word (data);
- while (data->file->offset + sizeof (data->quan_table[id]) + 1
- <= next_marker)
- {
- id = grub_jpeg_get_byte (data);
- if (id >= 0x10) /* Upper 4-bit is precision. */
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: only 8-bit precision is supported");
- if (id > 1)
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: too many quantization tables");
- if (grub_file_read (data->file, &data->quan_table[id],
- sizeof (data->quan_table[id]))
- != sizeof (data->quan_table[id]))
- return grub_errno;
- }
- if (data->file->offset != next_marker)
- grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: extra byte in quantization table");
- return grub_errno;
- }
- static grub_err_t
- grub_jpeg_decode_sof (struct grub_jpeg_data *data)
- {
- int i, cc;
- grub_uint32_t next_marker;
- next_marker = data->file->offset;
- next_marker += grub_jpeg_get_word (data);
- if (grub_jpeg_get_byte (data) != 8)
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: only 8-bit precision is supported");
- data->image_height = grub_jpeg_get_word (data);
- data->image_width = grub_jpeg_get_word (data);
- if ((!data->image_height) || (!data->image_width))
- return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid image size");
- cc = grub_jpeg_get_byte (data);
- if (cc != 3)
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: component count must be 3");
- for (i = 0; i < cc; i++)
- {
- int id, ss;
- id = grub_jpeg_get_byte (data) - 1;
- if ((id < 0) || (id >= 3))
- return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid index");
- ss = grub_jpeg_get_byte (data); /* Sampling factor. */
- if (!id)
- {
- data->vs = ss & 0xF; /* Vertical sampling. */
- data->hs = ss >> 4; /* Horizontal sampling. */
- if ((data->vs > 2) || (data->hs > 2))
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: sampling method not supported");
- }
- else if (ss != JPEG_SAMPLING_1x1)
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: sampling method not supported");
- data->comp_index[id][0] = grub_jpeg_get_byte (data);
- }
- if (data->file->offset != next_marker)
- grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: extra byte in sof");
- return grub_errno;
- }
- static void
- grub_jpeg_idct_transform (jpeg_data_unit_t du)
- {
- int *pd;
- int i;
- int t0, t1, t2, t3, t4, t5, t6, t7;
- int v0, v1, v2, v3, v4;
- pd = du;
- for (i = 0; i < JPEG_UNIT_SIZE; i++, pd++)
- {
- if ((pd[JPEG_UNIT_SIZE * 1] | pd[JPEG_UNIT_SIZE * 2] |
- pd[JPEG_UNIT_SIZE * 3] | pd[JPEG_UNIT_SIZE * 4] |
- pd[JPEG_UNIT_SIZE * 5] | pd[JPEG_UNIT_SIZE * 6] |
- pd[JPEG_UNIT_SIZE * 7]) == 0)
- {
- pd[JPEG_UNIT_SIZE * 0] <<= SHIFT_BITS;
- pd[JPEG_UNIT_SIZE * 1] = pd[JPEG_UNIT_SIZE * 2]
- = pd[JPEG_UNIT_SIZE * 3] = pd[JPEG_UNIT_SIZE * 4]
- = pd[JPEG_UNIT_SIZE * 5] = pd[JPEG_UNIT_SIZE * 6]
- = pd[JPEG_UNIT_SIZE * 7] = pd[JPEG_UNIT_SIZE * 0];
- continue;
- }
- t0 = pd[JPEG_UNIT_SIZE * 0];
- t1 = pd[JPEG_UNIT_SIZE * 2];
- t2 = pd[JPEG_UNIT_SIZE * 4];
- t3 = pd[JPEG_UNIT_SIZE * 6];
- v4 = (t1 + t3) * CONST (0.541196100);
- v0 = ((t0 + t2) << SHIFT_BITS);
- v1 = ((t0 - t2) << SHIFT_BITS);
- v2 = v4 - t3 * CONST (1.847759065);
- v3 = v4 + t1 * CONST (0.765366865);
- t0 = v0 + v3;
- t3 = v0 - v3;
- t1 = v1 + v2;
- t2 = v1 - v2;
- t4 = pd[JPEG_UNIT_SIZE * 7];
- t5 = pd[JPEG_UNIT_SIZE * 5];
- t6 = pd[JPEG_UNIT_SIZE * 3];
- t7 = pd[JPEG_UNIT_SIZE * 1];
- v0 = t4 + t7;
- v1 = t5 + t6;
- v2 = t4 + t6;
- v3 = t5 + t7;
- v4 = (v2 + v3) * CONST (1.175875602);
- v0 *= CONST (0.899976223);
- v1 *= CONST (2.562915447);
- v2 = v2 * CONST (1.961570560) - v4;
- v3 = v3 * CONST (0.390180644) - v4;
- t4 = t4 * CONST (0.298631336) - v0 - v2;
- t5 = t5 * CONST (2.053119869) - v1 - v3;
- t6 = t6 * CONST (3.072711026) - v1 - v2;
- t7 = t7 * CONST (1.501321110) - v0 - v3;
- pd[JPEG_UNIT_SIZE * 0] = t0 + t7;
- pd[JPEG_UNIT_SIZE * 7] = t0 - t7;
- pd[JPEG_UNIT_SIZE * 1] = t1 + t6;
- pd[JPEG_UNIT_SIZE * 6] = t1 - t6;
- pd[JPEG_UNIT_SIZE * 2] = t2 + t5;
- pd[JPEG_UNIT_SIZE * 5] = t2 - t5;
- pd[JPEG_UNIT_SIZE * 3] = t3 + t4;
- pd[JPEG_UNIT_SIZE * 4] = t3 - t4;
- }
- pd = du;
- for (i = 0; i < JPEG_UNIT_SIZE; i++, pd += JPEG_UNIT_SIZE)
- {
- if ((pd[1] | pd[2] | pd[3] | pd[4] | pd[5] | pd[6] | pd[7]) == 0)
- {
- pd[0] >>= (SHIFT_BITS + 3);
- pd[1] = pd[2] = pd[3] = pd[4] = pd[5] = pd[6] = pd[7] = pd[0];
- continue;
- }
- v4 = (pd[2] + pd[6]) * CONST (0.541196100);
- v0 = (pd[0] + pd[4]) << SHIFT_BITS;
- v1 = (pd[0] - pd[4]) << SHIFT_BITS;
- v2 = v4 - pd[6] * CONST (1.847759065);
- v3 = v4 + pd[2] * CONST (0.765366865);
- t0 = v0 + v3;
- t3 = v0 - v3;
- t1 = v1 + v2;
- t2 = v1 - v2;
- t4 = pd[7];
- t5 = pd[5];
- t6 = pd[3];
- t7 = pd[1];
- v0 = t4 + t7;
- v1 = t5 + t6;
- v2 = t4 + t6;
- v3 = t5 + t7;
- v4 = (v2 + v3) * CONST (1.175875602);
- v0 *= CONST (0.899976223);
- v1 *= CONST (2.562915447);
- v2 = v2 * CONST (1.961570560) - v4;
- v3 = v3 * CONST (0.390180644) - v4;
- t4 = t4 * CONST (0.298631336) - v0 - v2;
- t5 = t5 * CONST (2.053119869) - v1 - v3;
- t6 = t6 * CONST (3.072711026) - v1 - v2;
- t7 = t7 * CONST (1.501321110) - v0 - v3;
- pd[0] = (t0 + t7) >> (SHIFT_BITS * 2 + 3);
- pd[7] = (t0 - t7) >> (SHIFT_BITS * 2 + 3);
- pd[1] = (t1 + t6) >> (SHIFT_BITS * 2 + 3);
- pd[6] = (t1 - t6) >> (SHIFT_BITS * 2 + 3);
- pd[2] = (t2 + t5) >> (SHIFT_BITS * 2 + 3);
- pd[5] = (t2 - t5) >> (SHIFT_BITS * 2 + 3);
- pd[3] = (t3 + t4) >> (SHIFT_BITS * 2 + 3);
- pd[4] = (t3 - t4) >> (SHIFT_BITS * 2 + 3);
- }
- for (i = 0; i < JPEG_UNIT_SIZE * JPEG_UNIT_SIZE; i++)
- {
- du[i] += 128;
- if (du[i] < 0)
- du[i] = 0;
- if (du[i] > 255)
- du[i] = 255;
- }
- }
- static void
- grub_jpeg_decode_du (struct grub_jpeg_data *data, int id, jpeg_data_unit_t du)
- {
- int h1, h2, qt;
- unsigned pos;
- grub_memset (du, 0, sizeof (jpeg_data_unit_t));
- qt = data->comp_index[id][0];
- h1 = data->comp_index[id][1];
- h2 = data->comp_index[id][2];
- data->dc_value[id] +=
- grub_jpeg_get_number (data, grub_jpeg_get_huff_code (data, h1));
- du[0] = data->dc_value[id] * (int) data->quan_table[qt][0];
- pos = 1;
- while (pos < ARRAY_SIZE (data->quan_table[qt]))
- {
- int num, val;
- num = grub_jpeg_get_huff_code (data, h2);
- if (!num)
- break;
- val = grub_jpeg_get_number (data, num & 0xF);
- num >>= 4;
- pos += num;
- du[jpeg_zigzag_order[pos]] = val * (int) data->quan_table[qt][pos];
- pos++;
- }
- grub_jpeg_idct_transform (du);
- }
- static void
- grub_jpeg_ycrcb_to_rgb (int yy, int cr, int cb, grub_uint8_t * rgb)
- {
- int dd;
- cr -= 128;
- cb -= 128;
- /* Red */
- dd = yy + ((cr * CONST (1.402)) >> SHIFT_BITS);
- if (dd < 0)
- dd = 0;
- if (dd > 255)
- dd = 255;
- *(rgb++) = dd;
- /* Green */
- dd = yy - ((cb * CONST (0.34414) + cr * CONST (0.71414)) >> SHIFT_BITS);
- if (dd < 0)
- dd = 0;
- if (dd > 255)
- dd = 255;
- *(rgb++) = dd;
- /* Blue */
- dd = yy + ((cb * CONST (1.772)) >> SHIFT_BITS);
- if (dd < 0)
- dd = 0;
- if (dd > 255)
- dd = 255;
- *(rgb++) = dd;
- }
- static grub_err_t
- grub_jpeg_decode_sos (struct grub_jpeg_data *data)
- {
- int i, cc, r1, c1, nr1, nc1, vb, hb;
- grub_uint8_t *ptr1;
- grub_uint32_t data_offset;
- data_offset = data->file->offset;
- data_offset += grub_jpeg_get_word (data);
- cc = grub_jpeg_get_byte (data);
- if (cc != 3)
- return grub_error (GRUB_ERR_BAD_FILE_TYPE,
- "jpeg: component count must be 3");
- for (i = 0; i < cc; i++)
- {
- int id, ht;
- id = grub_jpeg_get_byte (data) - 1;
- if ((id < 0) || (id >= 3))
- return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid index");
- ht = grub_jpeg_get_byte (data);
- data->comp_index[id][1] = (ht >> 4);
- data->comp_index[id][2] = (ht & 0xF) + 2;
- }
- grub_jpeg_get_byte (data); /* Skip 3 unused bytes. */
- grub_jpeg_get_word (data);
- if (data->file->offset != data_offset)
- return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: extra byte in sos");
- if (grub_video_bitmap_create (data->bitmap, data->image_width,
- data->image_height,
- GRUB_VIDEO_BLIT_FORMAT_RGB_888))
- return grub_errno;
- data->bit_mask = 0x0;
- vb = data->vs * 8;
- hb = data->hs * 8;
- nr1 = (data->image_height + vb - 1) / vb;
- nc1 = (data->image_width + hb - 1) / hb;
- ptr1 = (*data->bitmap)->data;
- for (r1 = 0; r1 < nr1;
- r1++, ptr1 += (vb * data->image_width - hb * nc1) * 3)
- for (c1 = 0; c1 < nc1; c1++, ptr1 += hb * 3)
- {
- int r2, c2, nr2, nc2;
- grub_uint8_t *ptr2;
- for (r2 = 0; r2 < data->vs; r2++)
- for (c2 = 0; c2 < data->hs; c2++)
- grub_jpeg_decode_du (data, 0, data->ydu[r2 * 2 + c2]);
- grub_jpeg_decode_du (data, 1, data->cbdu);
- grub_jpeg_decode_du (data, 2, data->crdu);
- if (grub_errno)
- return grub_errno;
- nr2 = (r1 == nr1 - 1) ? (data->image_height - r1 * vb) : vb;
- nc2 = (c1 == nc1 - 1) ? (data->image_width - c1 * hb) : hb;
- ptr2 = ptr1;
- for (r2 = 0; r2 < nr2; r2++, ptr2 += (data->image_width - nc2) * 3)
- for (c2 = 0; c2 < nc2; c2++, ptr2 += 3)
- {
- int i0, yy, cr, cb;
- i0 = (r2 / data->vs) * 8 + (c2 / data->hs);
- cr = data->crdu[i0];
- cb = data->cbdu[i0];
- yy =
- data->ydu[(r2 / 8) * 2 + (c2 / 8)][(r2 % 8) * 8 + (c2 % 8)];
- grub_jpeg_ycrcb_to_rgb (yy, cr, cb, ptr2);
- }
- }
- return grub_errno;
- }
- static grub_uint8_t
- grub_jpeg_get_marker (struct grub_jpeg_data *data)
- {
- grub_uint8_t r;
- r = grub_jpeg_get_byte (data);
- if (r != JPEG_ESC_CHAR)
- {
- grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid maker");
- return 0;
- }
- return grub_jpeg_get_byte (data);
- }
- static grub_err_t
- grub_jpeg_decode_jpeg (struct grub_jpeg_data *data)
- {
- if (grub_jpeg_get_marker (data) != JPEG_MARKER_SOI) /* Start Of Image. */
- return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid jpeg file");
- while (grub_errno == 0)
- {
- grub_uint8_t marker;
- marker = grub_jpeg_get_marker (data);
- if (grub_errno)
- break;
- #ifdef JPEG_DEBUG
- grub_printf ("jpeg marker: %x\n", marker);
- #endif
- switch (marker)
- {
- case JPEG_MARKER_DHT: /* Define Huffman Table. */
- grub_jpeg_decode_huff_table (data);
- break;
- case JPEG_MARKER_DQT: /* Define Quantization Table. */
- grub_jpeg_decode_quan_table (data);
- break;
- case JPEG_MARKER_SOF0: /* Start Of Frame 0. */
- grub_jpeg_decode_sof (data);
- break;
- case JPEG_MARKER_SOS: /* Start Of Scan. */
- grub_jpeg_decode_sos (data);
- break;
- case JPEG_MARKER_EOI: /* End Of Image. */
- return grub_errno;
- default: /* Skip unrecognized marker. */
- {
- grub_uint16_t sz;
- sz = grub_jpeg_get_word (data);
- if (grub_errno)
- return (grub_errno);
- grub_file_seek (data->file, data->file->offset + sz - 2);
- }
- }
- }
- return grub_errno;
- }
- static grub_err_t
- grub_video_reader_jpeg (struct grub_video_bitmap **bitmap,
- const char *filename)
- {
- grub_file_t file;
- struct grub_jpeg_data *data;
- file = grub_buffile_open (filename, 0);
- if (!file)
- return grub_errno;
- data = grub_zalloc (sizeof (*data));
- if (data != NULL)
- {
- int i;
- data->file = file;
- data->bitmap = bitmap;
- grub_jpeg_decode_jpeg (data);
- for (i = 0; i < 4; i++)
- if (data->huff_value[i])
- grub_free (data->huff_value[i]);
- 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(JPEG_DEBUG)
- static grub_err_t
- grub_cmd_jpegtest (grub_command_t cmd __attribute__ ((unused)),
- int argc, char **args)
- {
- struct grub_video_bitmap *bitmap = 0;
- if (argc != 1)
- return grub_error (GRUB_ERR_BAD_ARGUMENT, "file name required");
- grub_video_reader_jpeg (&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 jpg_reader = {
- .extension = ".jpg",
- .reader = grub_video_reader_jpeg,
- .next = 0
- };
- static struct grub_video_bitmap_reader jpeg_reader = {
- .extension = ".jpeg",
- .reader = grub_video_reader_jpeg,
- .next = 0
- };
- GRUB_MOD_INIT (jpeg)
- {
- grub_video_bitmap_reader_register (&jpg_reader);
- grub_video_bitmap_reader_register (&jpeg_reader);
- #if defined(JPEG_DEBUG)
- cmd = grub_register_command ("jpegtest", grub_cmd_jpegtest,
- "FILE", "Tests loading of JPEG bitmap.");
- #endif
- }
- GRUB_MOD_FINI (jpeg)
- {
- #if defined(JPEG_DEBUG)
- grub_unregister_command (cmd);
- #endif
- grub_video_bitmap_reader_unregister (&jpeg_reader);
- grub_video_bitmap_reader_unregister (&jpg_reader);
- }
|