jpeg.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958
  1. /*
  2. * GRUB -- GRand Unified Bootloader
  3. * Copyright (C) 2008 Free Software Foundation, Inc.
  4. *
  5. * GRUB is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * GRUB is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
  17. */
  18. #include <grub/bitmap.h>
  19. #include <grub/types.h>
  20. #include <grub/normal.h>
  21. #include <grub/dl.h>
  22. #include <grub/mm.h>
  23. #include <grub/misc.h>
  24. #include <grub/bufio.h>
  25. #include <grub/safemath.h>
  26. GRUB_MOD_LICENSE ("GPLv3+");
  27. /* Uncomment following define to enable JPEG debug. */
  28. //#define JPEG_DEBUG
  29. #define JPEG_ESC_CHAR 0xFF
  30. #define JPEG_SAMPLING_1x1 0x11
  31. enum
  32. {
  33. JPEG_MARKER_SOF0 = 0xc0,
  34. JPEG_MARKER_DHT = 0xc4,
  35. JPEG_MARKER_SOI = 0xd8,
  36. JPEG_MARKER_EOI = 0xd9,
  37. JPEG_MARKER_RST0 = 0xd0,
  38. JPEG_MARKER_RST1 = 0xd1,
  39. JPEG_MARKER_RST2 = 0xd2,
  40. JPEG_MARKER_RST3 = 0xd3,
  41. JPEG_MARKER_RST4 = 0xd4,
  42. JPEG_MARKER_RST5 = 0xd5,
  43. JPEG_MARKER_RST6 = 0xd6,
  44. JPEG_MARKER_RST7 = 0xd7,
  45. JPEG_MARKER_SOS = 0xda,
  46. JPEG_MARKER_DQT = 0xdb,
  47. JPEG_MARKER_DRI = 0xdd,
  48. };
  49. #define SHIFT_BITS 8
  50. #define CONST(x) ((int) ((x) * (1L << SHIFT_BITS) + 0.5))
  51. #define JPEG_UNIT_SIZE 8
  52. static const grub_uint8_t jpeg_zigzag_order[64] = {
  53. 0, 1, 8, 16, 9, 2, 3, 10,
  54. 17, 24, 32, 25, 18, 11, 4, 5,
  55. 12, 19, 26, 33, 40, 48, 41, 34,
  56. 27, 20, 13, 6, 7, 14, 21, 28,
  57. 35, 42, 49, 56, 57, 50, 43, 36,
  58. 29, 22, 15, 23, 30, 37, 44, 51,
  59. 58, 59, 52, 45, 38, 31, 39, 46,
  60. 53, 60, 61, 54, 47, 55, 62, 63
  61. };
  62. #ifdef JPEG_DEBUG
  63. static grub_command_t cmd;
  64. #endif
  65. typedef int jpeg_data_unit_t[64];
  66. struct grub_jpeg_data
  67. {
  68. grub_file_t file;
  69. struct grub_video_bitmap **bitmap;
  70. grub_uint8_t *bitmap_ptr;
  71. unsigned image_width;
  72. unsigned image_height;
  73. grub_uint8_t *huff_value[4];
  74. int huff_offset[4][16];
  75. int huff_maxval[4][16];
  76. grub_uint8_t quan_table[2][64];
  77. int comp_index[3][3];
  78. jpeg_data_unit_t ydu[4];
  79. jpeg_data_unit_t crdu;
  80. jpeg_data_unit_t cbdu;
  81. unsigned log_vs, log_hs;
  82. int dri;
  83. unsigned r1;
  84. int dc_value[3];
  85. int color_components;
  86. int bit_mask, bit_save;
  87. };
  88. static grub_uint8_t
  89. grub_jpeg_get_byte (struct grub_jpeg_data *data)
  90. {
  91. grub_uint8_t r;
  92. grub_ssize_t bytes_read;
  93. r = 0;
  94. bytes_read = grub_file_read (data->file, &r, 1);
  95. if (bytes_read != 1)
  96. {
  97. grub_error (GRUB_ERR_BAD_FILE_TYPE,
  98. "jpeg: unexpected end of data");
  99. return 0;
  100. }
  101. return r;
  102. }
  103. static grub_uint16_t
  104. grub_jpeg_get_word (struct grub_jpeg_data *data)
  105. {
  106. grub_uint16_t r;
  107. grub_ssize_t bytes_read;
  108. r = 0;
  109. bytes_read = grub_file_read (data->file, &r, sizeof (grub_uint16_t));
  110. if (bytes_read != sizeof (grub_uint16_t))
  111. {
  112. grub_error (GRUB_ERR_BAD_FILE_TYPE,
  113. "jpeg: unexpected end of data");
  114. return 0;
  115. }
  116. return grub_be_to_cpu16 (r);
  117. }
  118. static int
  119. grub_jpeg_get_bit (struct grub_jpeg_data *data)
  120. {
  121. int ret;
  122. if (data->bit_mask == 0)
  123. {
  124. data->bit_save = grub_jpeg_get_byte (data);
  125. if (grub_errno != GRUB_ERR_NONE) {
  126. grub_error (GRUB_ERR_BAD_FILE_TYPE,
  127. "jpeg: file read error");
  128. return 0;
  129. }
  130. if (data->bit_save == JPEG_ESC_CHAR)
  131. {
  132. if (grub_jpeg_get_byte (data) != 0)
  133. {
  134. grub_error (GRUB_ERR_BAD_FILE_TYPE,
  135. "jpeg: invalid 0xFF in data stream");
  136. return 0;
  137. }
  138. if (grub_errno != GRUB_ERR_NONE)
  139. {
  140. grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: file read error");
  141. return 0;
  142. }
  143. }
  144. data->bit_mask = 0x80;
  145. }
  146. ret = ((data->bit_save & data->bit_mask) != 0);
  147. data->bit_mask >>= 1;
  148. return ret;
  149. }
  150. static int
  151. grub_jpeg_get_number (struct grub_jpeg_data *data, int num)
  152. {
  153. int value, i, msb;
  154. if (num == 0)
  155. return 0;
  156. msb = value = grub_jpeg_get_bit (data);
  157. for (i = 1; i < num && grub_errno == GRUB_ERR_NONE; i++)
  158. value = (value << 1) + (grub_jpeg_get_bit (data) != 0);
  159. if (!msb)
  160. value += 1 - (1 << num);
  161. return value;
  162. }
  163. static int
  164. grub_jpeg_get_huff_code (struct grub_jpeg_data *data, int id)
  165. {
  166. int code;
  167. unsigned i;
  168. code = 0;
  169. for (i = 0; i < ARRAY_SIZE (data->huff_maxval[id]); i++)
  170. {
  171. code <<= 1;
  172. if (grub_jpeg_get_bit (data))
  173. code++;
  174. if (code < data->huff_maxval[id][i])
  175. return data->huff_value[id][code + data->huff_offset[id][i]];
  176. }
  177. grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: huffman decode fails");
  178. return 0;
  179. }
  180. static grub_err_t
  181. grub_jpeg_decode_huff_table (struct grub_jpeg_data *data)
  182. {
  183. int id, ac, n, base, ofs;
  184. grub_uint32_t next_marker;
  185. grub_uint8_t count[16];
  186. unsigned i;
  187. next_marker = data->file->offset;
  188. next_marker += grub_jpeg_get_word (data);
  189. if (next_marker > data->file->size)
  190. {
  191. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  192. "jpeg: invalid huffman table");
  193. }
  194. while (data->file->offset + sizeof (count) + 1 <= next_marker)
  195. {
  196. id = grub_jpeg_get_byte (data);
  197. if (grub_errno != GRUB_ERR_NONE)
  198. return grub_errno;
  199. ac = (id >> 4) & 1;
  200. id &= 0xF;
  201. if (id > 1)
  202. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  203. "jpeg: too many huffman tables");
  204. if (grub_file_read (data->file, &count, sizeof (count)) !=
  205. sizeof (count))
  206. return grub_errno;
  207. n = 0;
  208. for (i = 0; i < ARRAY_SIZE (count); i++)
  209. n += count[i];
  210. id += ac * 2;
  211. if (data->huff_value[id] != NULL)
  212. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  213. "jpeg: attempt to reallocate huffman table");
  214. data->huff_value[id] = grub_malloc (n);
  215. if (grub_errno)
  216. return grub_errno;
  217. if (grub_file_read (data->file, data->huff_value[id], n) != n)
  218. return grub_errno;
  219. base = 0;
  220. ofs = 0;
  221. for (i = 0; i < ARRAY_SIZE (count); i++)
  222. {
  223. base += count[i];
  224. ofs += count[i];
  225. data->huff_maxval[id][i] = base;
  226. data->huff_offset[id][i] = ofs - base;
  227. base <<= 1;
  228. }
  229. }
  230. if (data->file->offset != next_marker)
  231. grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: extra byte in huffman table");
  232. return grub_errno;
  233. }
  234. static grub_err_t
  235. grub_jpeg_decode_quan_table (struct grub_jpeg_data *data)
  236. {
  237. int id;
  238. grub_uint32_t next_marker;
  239. next_marker = data->file->offset;
  240. next_marker += grub_jpeg_get_word (data);
  241. if (grub_errno != GRUB_ERR_NONE)
  242. return grub_errno;
  243. if (next_marker > data->file->size)
  244. {
  245. /* Should never be set beyond the size of the file. */
  246. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid next reference");
  247. }
  248. while (data->file->offset + sizeof (data->quan_table[id]) + 1
  249. <= next_marker)
  250. {
  251. id = grub_jpeg_get_byte (data);
  252. if (grub_errno != GRUB_ERR_NONE)
  253. return grub_errno;
  254. if (id >= 0x10) /* Upper 4-bit is precision. */
  255. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  256. "jpeg: only 8-bit precision is supported");
  257. if (id > 1)
  258. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  259. "jpeg: too many quantization tables");
  260. if (grub_file_read (data->file, &data->quan_table[id],
  261. sizeof (data->quan_table[id]))
  262. != sizeof (data->quan_table[id]))
  263. return grub_errno;
  264. }
  265. if (data->file->offset != next_marker)
  266. grub_error (GRUB_ERR_BAD_FILE_TYPE,
  267. "jpeg: extra byte in quantization table");
  268. return grub_errno;
  269. }
  270. static grub_err_t
  271. grub_jpeg_decode_sof (struct grub_jpeg_data *data)
  272. {
  273. int i, cc;
  274. grub_uint32_t next_marker;
  275. next_marker = data->file->offset;
  276. next_marker += grub_jpeg_get_word (data);
  277. if (grub_errno != GRUB_ERR_NONE)
  278. return grub_errno;
  279. if (grub_jpeg_get_byte (data) != 8)
  280. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  281. "jpeg: only 8-bit precision is supported");
  282. data->image_height = grub_jpeg_get_word (data);
  283. data->image_width = grub_jpeg_get_word (data);
  284. grub_dprintf ("jpeg", "image height: %d\n", data->image_height);
  285. grub_dprintf ("jpeg", "image width: %d\n", data->image_width);
  286. if ((!data->image_height) || (!data->image_width) ||
  287. (data->image_height > IMAGE_HW_MAX_PX) || (data->image_width > IMAGE_HW_MAX_PX))
  288. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid image size");
  289. cc = grub_jpeg_get_byte (data);
  290. if (cc != 1 && cc != 3)
  291. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  292. "jpeg: component count must be 1 or 3");
  293. data->color_components = cc;
  294. for (i = 0; i < cc; i++)
  295. {
  296. int id, ss;
  297. id = grub_jpeg_get_byte (data) - 1;
  298. if ((id < 0) || (id >= 3))
  299. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid index");
  300. ss = grub_jpeg_get_byte (data); /* Sampling factor. */
  301. if (grub_errno != GRUB_ERR_NONE)
  302. return grub_errno;
  303. if (!id)
  304. {
  305. grub_uint8_t vs, hs;
  306. vs = ss & 0xF; /* Vertical sampling. */
  307. hs = ss >> 4; /* Horizontal sampling. */
  308. if ((vs > 2) || (hs > 2) || (vs == 0) || (hs == 0))
  309. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  310. "jpeg: sampling method not supported");
  311. data->log_vs = (vs == 2);
  312. data->log_hs = (hs == 2);
  313. }
  314. else if (ss != JPEG_SAMPLING_1x1)
  315. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  316. "jpeg: sampling method not supported");
  317. data->comp_index[id][0] = grub_jpeg_get_byte (data);
  318. if (data->comp_index[id][0] > 1)
  319. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  320. "jpeg: too many quantization tables");
  321. }
  322. if (data->file->offset != next_marker)
  323. grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: extra byte in sof");
  324. return grub_errno;
  325. }
  326. static grub_err_t
  327. grub_jpeg_decode_dri (struct grub_jpeg_data *data)
  328. {
  329. if (grub_jpeg_get_word (data) != 4)
  330. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  331. "jpeg: DRI marker length must be 4");
  332. data->dri = grub_jpeg_get_word (data);
  333. return grub_errno;
  334. }
  335. static void
  336. grub_jpeg_idct_transform (jpeg_data_unit_t du)
  337. {
  338. int *pd;
  339. int i;
  340. int t0, t1, t2, t3, t4, t5, t6, t7;
  341. int v0, v1, v2, v3, v4;
  342. pd = du;
  343. for (i = 0; i < JPEG_UNIT_SIZE; i++, pd++)
  344. {
  345. if ((pd[JPEG_UNIT_SIZE * 1] | pd[JPEG_UNIT_SIZE * 2] |
  346. pd[JPEG_UNIT_SIZE * 3] | pd[JPEG_UNIT_SIZE * 4] |
  347. pd[JPEG_UNIT_SIZE * 5] | pd[JPEG_UNIT_SIZE * 6] |
  348. pd[JPEG_UNIT_SIZE * 7]) == 0)
  349. {
  350. pd[JPEG_UNIT_SIZE * 0] <<= SHIFT_BITS;
  351. pd[JPEG_UNIT_SIZE * 1] = pd[JPEG_UNIT_SIZE * 2]
  352. = pd[JPEG_UNIT_SIZE * 3] = pd[JPEG_UNIT_SIZE * 4]
  353. = pd[JPEG_UNIT_SIZE * 5] = pd[JPEG_UNIT_SIZE * 6]
  354. = pd[JPEG_UNIT_SIZE * 7] = pd[JPEG_UNIT_SIZE * 0];
  355. continue;
  356. }
  357. t0 = pd[JPEG_UNIT_SIZE * 0];
  358. t1 = pd[JPEG_UNIT_SIZE * 2];
  359. t2 = pd[JPEG_UNIT_SIZE * 4];
  360. t3 = pd[JPEG_UNIT_SIZE * 6];
  361. v4 = (t1 + t3) * CONST (0.541196100);
  362. v0 = ((t0 + t2) << SHIFT_BITS);
  363. v1 = ((t0 - t2) << SHIFT_BITS);
  364. v2 = v4 - t3 * CONST (1.847759065);
  365. v3 = v4 + t1 * CONST (0.765366865);
  366. t0 = v0 + v3;
  367. t3 = v0 - v3;
  368. t1 = v1 + v2;
  369. t2 = v1 - v2;
  370. t4 = pd[JPEG_UNIT_SIZE * 7];
  371. t5 = pd[JPEG_UNIT_SIZE * 5];
  372. t6 = pd[JPEG_UNIT_SIZE * 3];
  373. t7 = pd[JPEG_UNIT_SIZE * 1];
  374. v0 = t4 + t7;
  375. v1 = t5 + t6;
  376. v2 = t4 + t6;
  377. v3 = t5 + t7;
  378. v4 = (v2 + v3) * CONST (1.175875602);
  379. v0 *= CONST (0.899976223);
  380. v1 *= CONST (2.562915447);
  381. v2 = v2 * CONST (1.961570560) - v4;
  382. v3 = v3 * CONST (0.390180644) - v4;
  383. t4 = t4 * CONST (0.298631336) - v0 - v2;
  384. t5 = t5 * CONST (2.053119869) - v1 - v3;
  385. t6 = t6 * CONST (3.072711026) - v1 - v2;
  386. t7 = t7 * CONST (1.501321110) - v0 - v3;
  387. pd[JPEG_UNIT_SIZE * 0] = t0 + t7;
  388. pd[JPEG_UNIT_SIZE * 7] = t0 - t7;
  389. pd[JPEG_UNIT_SIZE * 1] = t1 + t6;
  390. pd[JPEG_UNIT_SIZE * 6] = t1 - t6;
  391. pd[JPEG_UNIT_SIZE * 2] = t2 + t5;
  392. pd[JPEG_UNIT_SIZE * 5] = t2 - t5;
  393. pd[JPEG_UNIT_SIZE * 3] = t3 + t4;
  394. pd[JPEG_UNIT_SIZE * 4] = t3 - t4;
  395. }
  396. pd = du;
  397. for (i = 0; i < JPEG_UNIT_SIZE; i++, pd += JPEG_UNIT_SIZE)
  398. {
  399. if ((pd[1] | pd[2] | pd[3] | pd[4] | pd[5] | pd[6] | pd[7]) == 0)
  400. {
  401. pd[0] >>= (SHIFT_BITS + 3);
  402. pd[1] = pd[2] = pd[3] = pd[4] = pd[5] = pd[6] = pd[7] = pd[0];
  403. continue;
  404. }
  405. v4 = (pd[2] + pd[6]) * CONST (0.541196100);
  406. v0 = (pd[0] + pd[4]) << SHIFT_BITS;
  407. v1 = (pd[0] - pd[4]) << SHIFT_BITS;
  408. v2 = v4 - pd[6] * CONST (1.847759065);
  409. v3 = v4 + pd[2] * CONST (0.765366865);
  410. t0 = v0 + v3;
  411. t3 = v0 - v3;
  412. t1 = v1 + v2;
  413. t2 = v1 - v2;
  414. t4 = pd[7];
  415. t5 = pd[5];
  416. t6 = pd[3];
  417. t7 = pd[1];
  418. v0 = t4 + t7;
  419. v1 = t5 + t6;
  420. v2 = t4 + t6;
  421. v3 = t5 + t7;
  422. v4 = (v2 + v3) * CONST (1.175875602);
  423. v0 *= CONST (0.899976223);
  424. v1 *= CONST (2.562915447);
  425. v2 = v2 * CONST (1.961570560) - v4;
  426. v3 = v3 * CONST (0.390180644) - v4;
  427. t4 = t4 * CONST (0.298631336) - v0 - v2;
  428. t5 = t5 * CONST (2.053119869) - v1 - v3;
  429. t6 = t6 * CONST (3.072711026) - v1 - v2;
  430. t7 = t7 * CONST (1.501321110) - v0 - v3;
  431. pd[0] = (t0 + t7) >> (SHIFT_BITS * 2 + 3);
  432. pd[7] = (t0 - t7) >> (SHIFT_BITS * 2 + 3);
  433. pd[1] = (t1 + t6) >> (SHIFT_BITS * 2 + 3);
  434. pd[6] = (t1 - t6) >> (SHIFT_BITS * 2 + 3);
  435. pd[2] = (t2 + t5) >> (SHIFT_BITS * 2 + 3);
  436. pd[5] = (t2 - t5) >> (SHIFT_BITS * 2 + 3);
  437. pd[3] = (t3 + t4) >> (SHIFT_BITS * 2 + 3);
  438. pd[4] = (t3 - t4) >> (SHIFT_BITS * 2 + 3);
  439. }
  440. for (i = 0; i < JPEG_UNIT_SIZE * JPEG_UNIT_SIZE; i++)
  441. {
  442. du[i] += 128;
  443. if (du[i] < 0)
  444. du[i] = 0;
  445. if (du[i] > 255)
  446. du[i] = 255;
  447. }
  448. }
  449. static grub_err_t
  450. grub_jpeg_decode_du (struct grub_jpeg_data *data, int id, jpeg_data_unit_t du)
  451. {
  452. int h1, h2, qt;
  453. unsigned pos;
  454. grub_memset (du, 0, sizeof (jpeg_data_unit_t));
  455. qt = data->comp_index[id][0];
  456. h1 = data->comp_index[id][1];
  457. h2 = data->comp_index[id][2];
  458. data->dc_value[id] +=
  459. grub_jpeg_get_number (data, grub_jpeg_get_huff_code (data, h1));
  460. if (grub_errno != GRUB_ERR_NONE)
  461. return grub_errno;
  462. du[0] = data->dc_value[id] * (int) data->quan_table[qt][0];
  463. pos = 1;
  464. while (pos < ARRAY_SIZE (data->quan_table[qt]))
  465. {
  466. int num, val;
  467. num = grub_jpeg_get_huff_code (data, h2);
  468. if (!num)
  469. break;
  470. val = grub_jpeg_get_number (data, num & 0xF);
  471. num >>= 4;
  472. pos += num;
  473. if (grub_errno != GRUB_ERR_NONE)
  474. return grub_errno;
  475. if (pos >= ARRAY_SIZE (jpeg_zigzag_order))
  476. {
  477. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  478. "jpeg: invalid position in zigzag order!?");
  479. }
  480. du[jpeg_zigzag_order[pos]] = val * (int) data->quan_table[qt][pos];
  481. pos++;
  482. }
  483. grub_jpeg_idct_transform (du);
  484. return GRUB_ERR_NONE;
  485. }
  486. static void
  487. grub_jpeg_ycrcb_to_rgb (int yy, int cr, int cb, grub_uint8_t * rgb)
  488. {
  489. int dd;
  490. cr -= 128;
  491. cb -= 128;
  492. /* Red */
  493. dd = yy + ((cr * CONST (1.402)) >> SHIFT_BITS);
  494. if (dd < 0)
  495. dd = 0;
  496. if (dd > 255)
  497. dd = 255;
  498. #ifdef GRUB_CPU_WORDS_BIGENDIAN
  499. rgb[2] = dd;
  500. #else
  501. *(rgb++) = dd;
  502. #endif
  503. /* Green */
  504. dd = yy - ((cb * CONST (0.34414) + cr * CONST (0.71414)) >> SHIFT_BITS);
  505. if (dd < 0)
  506. dd = 0;
  507. if (dd > 255)
  508. dd = 255;
  509. #ifdef GRUB_CPU_WORDS_BIGENDIAN
  510. rgb[1] = dd;
  511. #else
  512. *(rgb++) = dd;
  513. #endif
  514. /* Blue */
  515. dd = yy + ((cb * CONST (1.772)) >> SHIFT_BITS);
  516. if (dd < 0)
  517. dd = 0;
  518. if (dd > 255)
  519. dd = 255;
  520. #ifdef GRUB_CPU_WORDS_BIGENDIAN
  521. rgb[0] = dd;
  522. rgb += 3;
  523. #else
  524. *(rgb++) = dd;
  525. #endif
  526. }
  527. static grub_err_t
  528. grub_jpeg_decode_sos (struct grub_jpeg_data *data)
  529. {
  530. int i, cc;
  531. grub_uint32_t data_offset;
  532. data_offset = data->file->offset;
  533. data_offset += grub_jpeg_get_word (data);
  534. cc = grub_jpeg_get_byte (data);
  535. if (grub_errno != GRUB_ERR_NONE)
  536. return grub_errno;
  537. if (cc != 3 && cc != 1)
  538. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  539. "jpeg: component count must be 1 or 3");
  540. data->color_components = cc;
  541. for (i = 0; i < cc; i++)
  542. {
  543. int id, ht;
  544. id = grub_jpeg_get_byte (data) - 1;
  545. if ((id < 0) || (id >= 3))
  546. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid index");
  547. if (grub_errno != GRUB_ERR_NONE)
  548. return grub_errno;
  549. ht = grub_jpeg_get_byte (data);
  550. data->comp_index[id][1] = (ht >> 4);
  551. data->comp_index[id][2] = (ht & 0xF) + 2;
  552. if ((data->comp_index[id][1] < 0) || (data->comp_index[id][1] > 3) ||
  553. (data->comp_index[id][2] < 0) || (data->comp_index[id][2] > 3))
  554. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid hufftable index");
  555. if (grub_errno != GRUB_ERR_NONE)
  556. return grub_errno;
  557. }
  558. grub_jpeg_get_byte (data); /* Skip 3 unused bytes. */
  559. grub_jpeg_get_word (data);
  560. if (grub_errno != GRUB_ERR_NONE)
  561. return grub_errno;
  562. if (data->file->offset != data_offset)
  563. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: extra byte in sos");
  564. if (*data->bitmap)
  565. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: too many start of scan blocks");
  566. if (grub_video_bitmap_create (data->bitmap, data->image_width,
  567. data->image_height,
  568. GRUB_VIDEO_BLIT_FORMAT_RGB_888))
  569. return grub_errno;
  570. data->bitmap_ptr = (*data->bitmap)->data;
  571. return GRUB_ERR_NONE;
  572. }
  573. static grub_err_t
  574. grub_jpeg_decode_data (struct grub_jpeg_data *data)
  575. {
  576. unsigned c1, vb, hb, nr1, nc1;
  577. unsigned stride_a, stride_b, stride;
  578. int rst = data->dri;
  579. grub_err_t err = GRUB_ERR_NONE;
  580. vb = 8 << data->log_vs;
  581. hb = 8 << data->log_hs;
  582. nr1 = (data->image_height + vb - 1) >> (3 + data->log_vs);
  583. nc1 = (data->image_width + hb - 1) >> (3 + data->log_hs);
  584. if (data->bitmap_ptr == NULL)
  585. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  586. "jpeg: attempted to decode data before start of stream");
  587. if (grub_mul(vb, data->image_width, &stride_a) ||
  588. grub_mul(hb, nc1, &stride_b) ||
  589. grub_sub(stride_a, stride_b, &stride))
  590. return grub_error (GRUB_ERR_BAD_FILE_TYPE,
  591. "jpeg: cannot decode image with these dimensions");
  592. for (; data->r1 < nr1 && (!data->dri || rst);
  593. data->r1++, data->bitmap_ptr += stride * 3)
  594. for (c1 = 0; c1 < nc1 && (!data->dri || rst);
  595. c1++, rst--, data->bitmap_ptr += hb * 3)
  596. {
  597. unsigned r2, c2, nr2, nc2;
  598. grub_uint8_t *ptr2;
  599. for (r2 = 0; r2 < (1U << data->log_vs); r2++)
  600. for (c2 = 0; c2 < (1U << data->log_hs); c2++)
  601. {
  602. err = grub_jpeg_decode_du (data, 0, data->ydu[r2 * 2 + c2]);
  603. if (err != GRUB_ERR_NONE)
  604. return err;
  605. }
  606. if (data->color_components >= 3)
  607. {
  608. err = grub_jpeg_decode_du (data, 1, data->cbdu);
  609. if (err != GRUB_ERR_NONE)
  610. return err;
  611. err = grub_jpeg_decode_du (data, 2, data->crdu);
  612. if (err != GRUB_ERR_NONE)
  613. return err;
  614. }
  615. nr2 = (data->r1 == nr1 - 1) ? (data->image_height - data->r1 * vb) : vb;
  616. nc2 = (c1 == nc1 - 1) ? (data->image_width - c1 * hb) : hb;
  617. ptr2 = data->bitmap_ptr;
  618. for (r2 = 0; r2 < nr2; r2++, ptr2 += (data->image_width - nc2) * 3)
  619. for (c2 = 0; c2 < nc2; c2++, ptr2 += 3)
  620. {
  621. unsigned i0;
  622. int yy;
  623. i0 = (r2 >> data->log_vs) * 8 + (c2 >> data->log_hs);
  624. yy = data->ydu[(r2 / 8) * 2 + (c2 / 8)][(r2 % 8) * 8 + (c2 % 8)];
  625. if (data->color_components >= 3)
  626. {
  627. int cr, cb;
  628. cr = data->crdu[i0];
  629. cb = data->cbdu[i0];
  630. grub_jpeg_ycrcb_to_rgb (yy, cr, cb, ptr2);
  631. }
  632. else
  633. {
  634. ptr2[0] = yy;
  635. ptr2[1] = yy;
  636. ptr2[2] = yy;
  637. }
  638. }
  639. }
  640. return grub_errno;
  641. }
  642. static void
  643. grub_jpeg_reset (struct grub_jpeg_data *data)
  644. {
  645. data->bit_mask = 0x0;
  646. data->dc_value[0] = 0;
  647. data->dc_value[1] = 0;
  648. data->dc_value[2] = 0;
  649. }
  650. static grub_uint8_t
  651. grub_jpeg_get_marker (struct grub_jpeg_data *data)
  652. {
  653. grub_uint8_t r;
  654. r = grub_jpeg_get_byte (data);
  655. if (r != JPEG_ESC_CHAR)
  656. {
  657. grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid maker");
  658. return 0;
  659. }
  660. return grub_jpeg_get_byte (data);
  661. }
  662. static grub_err_t
  663. grub_jpeg_decode_jpeg (struct grub_jpeg_data *data)
  664. {
  665. if (grub_jpeg_get_marker (data) != JPEG_MARKER_SOI) /* Start Of Image. */
  666. return grub_error (GRUB_ERR_BAD_FILE_TYPE, "jpeg: invalid jpeg file");
  667. while (grub_errno == 0)
  668. {
  669. grub_uint8_t marker;
  670. marker = grub_jpeg_get_marker (data);
  671. if (grub_errno)
  672. break;
  673. grub_dprintf ("jpeg", "jpeg marker: %x\n", marker);
  674. switch (marker)
  675. {
  676. case JPEG_MARKER_DHT: /* Define Huffman Table. */
  677. grub_jpeg_decode_huff_table (data);
  678. break;
  679. case JPEG_MARKER_DQT: /* Define Quantization Table. */
  680. grub_jpeg_decode_quan_table (data);
  681. break;
  682. case JPEG_MARKER_SOF0: /* Start Of Frame 0. */
  683. grub_jpeg_decode_sof (data);
  684. break;
  685. case JPEG_MARKER_DRI: /* Define Restart Interval. */
  686. grub_jpeg_decode_dri (data);
  687. break;
  688. case JPEG_MARKER_SOS: /* Start Of Scan. */
  689. if (grub_jpeg_decode_sos (data))
  690. break;
  691. /* FALLTHROUGH */
  692. case JPEG_MARKER_RST0: /* Restart. */
  693. case JPEG_MARKER_RST1:
  694. case JPEG_MARKER_RST2:
  695. case JPEG_MARKER_RST3:
  696. case JPEG_MARKER_RST4:
  697. case JPEG_MARKER_RST5:
  698. case JPEG_MARKER_RST6:
  699. case JPEG_MARKER_RST7:
  700. grub_jpeg_decode_data (data);
  701. grub_jpeg_reset (data);
  702. break;
  703. case JPEG_MARKER_EOI: /* End Of Image. */
  704. return grub_errno;
  705. default: /* Skip unrecognized marker. */
  706. {
  707. grub_uint16_t sz;
  708. sz = grub_jpeg_get_word (data);
  709. if (grub_errno)
  710. return (grub_errno);
  711. grub_file_seek (data->file, data->file->offset + sz - 2);
  712. }
  713. }
  714. }
  715. return grub_errno;
  716. }
  717. static grub_err_t
  718. grub_video_reader_jpeg (struct grub_video_bitmap **bitmap,
  719. const char *filename)
  720. {
  721. grub_file_t file;
  722. struct grub_jpeg_data *data;
  723. file = grub_buffile_open (filename, GRUB_FILE_TYPE_PIXMAP, 0);
  724. if (!file)
  725. return grub_errno;
  726. data = grub_zalloc (sizeof (*data));
  727. if (data != NULL)
  728. {
  729. int i;
  730. data->file = file;
  731. data->bitmap = bitmap;
  732. grub_jpeg_decode_jpeg (data);
  733. for (i = 0; i < 4; i++)
  734. grub_free (data->huff_value[i]);
  735. grub_free (data);
  736. }
  737. if (grub_errno != GRUB_ERR_NONE)
  738. {
  739. grub_video_bitmap_destroy (*bitmap);
  740. *bitmap = 0;
  741. }
  742. grub_file_close (file);
  743. return grub_errno;
  744. }
  745. #if defined(JPEG_DEBUG)
  746. static grub_err_t
  747. grub_cmd_jpegtest (grub_command_t cmdd __attribute__ ((unused)),
  748. int argc, char **args)
  749. {
  750. struct grub_video_bitmap *bitmap = 0;
  751. if (argc != 1)
  752. return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected"));
  753. grub_video_reader_jpeg (&bitmap, args[0]);
  754. if (grub_errno != GRUB_ERR_NONE)
  755. return grub_errno;
  756. grub_video_bitmap_destroy (bitmap);
  757. return GRUB_ERR_NONE;
  758. }
  759. #endif
  760. static struct grub_video_bitmap_reader jpg_reader = {
  761. .extension = ".jpg",
  762. .reader = grub_video_reader_jpeg,
  763. .next = 0
  764. };
  765. static struct grub_video_bitmap_reader jpeg_reader = {
  766. .extension = ".jpeg",
  767. .reader = grub_video_reader_jpeg,
  768. .next = 0
  769. };
  770. GRUB_MOD_INIT (jpeg)
  771. {
  772. grub_video_bitmap_reader_register (&jpg_reader);
  773. grub_video_bitmap_reader_register (&jpeg_reader);
  774. #if defined(JPEG_DEBUG)
  775. cmd = grub_register_command ("jpegtest", grub_cmd_jpegtest,
  776. "FILE", "Tests loading of JPEG bitmap.");
  777. #endif
  778. }
  779. GRUB_MOD_FINI (jpeg)
  780. {
  781. #if defined(JPEG_DEBUG)
  782. grub_unregister_command (cmd);
  783. #endif
  784. grub_video_bitmap_reader_unregister (&jpeg_reader);
  785. grub_video_bitmap_reader_unregister (&jpg_reader);
  786. }