You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

142 lines
6.0 KiB

  1. // Copyright 2021 Nick Brassel (@tzarc)
  2. // SPDX-License-Identifier: GPL-2.0-or-later
  3. #include "qp_internal.h"
  4. #include "qp_draw.h"
  5. #include "qp_comms.h"
  6. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  7. // Palette / Monochrome-format decoder
  8. static const qp_pixel_t qp_pixel_white = {.hsv888 = {.h = 0, .s = 0, .v = 255}};
  9. static const qp_pixel_t qp_pixel_black = {.hsv888 = {.h = 0, .s = 0, .v = 0}};
  10. bool qp_internal_bpp_capable(uint8_t bits_per_pixel) {
  11. #if !(QUANTUM_PAINTER_SUPPORTS_256_PALETTE)
  12. if (bits_per_pixel > 4) {
  13. qp_dprintf("qp_internal_decode_palette: image bpp greater than 4\n");
  14. return false;
  15. }
  16. #endif
  17. if (bits_per_pixel > 8) {
  18. qp_dprintf("qp_internal_decode_palette: image bpp greater than 8\n");
  19. return false;
  20. }
  21. return true;
  22. }
  23. bool qp_internal_decode_palette(painter_device_t device, uint32_t pixel_count, uint8_t bits_per_pixel, qp_internal_byte_input_callback input_callback, void* input_arg, qp_pixel_t* palette, qp_internal_pixel_output_callback output_callback, void* output_arg) {
  24. const uint8_t pixel_bitmask = (1 << bits_per_pixel) - 1;
  25. const uint8_t pixels_per_byte = 8 / bits_per_pixel;
  26. uint32_t remaining_pixels = pixel_count; // don't try to derive from byte_count, we may not use an entire byte
  27. while (remaining_pixels > 0) {
  28. uint8_t byteval = input_callback(input_arg);
  29. if (byteval < 0) {
  30. return false;
  31. }
  32. uint8_t loop_pixels = remaining_pixels < pixels_per_byte ? remaining_pixels : pixels_per_byte;
  33. for (uint8_t q = 0; q < loop_pixels; ++q) {
  34. if (!output_callback(palette, byteval & pixel_bitmask, output_arg)) {
  35. return false;
  36. }
  37. byteval >>= bits_per_pixel;
  38. }
  39. remaining_pixels -= loop_pixels;
  40. }
  41. return true;
  42. }
  43. bool qp_internal_decode_grayscale(painter_device_t device, uint32_t pixel_count, uint8_t bits_per_pixel, qp_internal_byte_input_callback input_callback, void* input_arg, qp_internal_pixel_output_callback output_callback, void* output_arg) {
  44. return qp_internal_decode_recolor(device, pixel_count, bits_per_pixel, input_callback, input_arg, qp_pixel_white, qp_pixel_black, output_callback, output_arg);
  45. }
  46. bool qp_internal_decode_recolor(painter_device_t device, uint32_t pixel_count, uint8_t bits_per_pixel, qp_internal_byte_input_callback input_callback, void* input_arg, qp_pixel_t fg_hsv888, qp_pixel_t bg_hsv888, qp_internal_pixel_output_callback output_callback, void* output_arg) {
  47. struct painter_driver_t* driver = (struct painter_driver_t*)device;
  48. int16_t steps = 1 << bits_per_pixel; // number of items we need to interpolate
  49. if (qp_internal_interpolate_palette(fg_hsv888, bg_hsv888, steps)) {
  50. if (!driver->driver_vtable->palette_convert(device, steps, qp_internal_global_pixel_lookup_table)) {
  51. return false;
  52. }
  53. }
  54. return qp_internal_decode_palette(device, pixel_count, bits_per_pixel, input_callback, input_arg, qp_internal_global_pixel_lookup_table, output_callback, output_arg);
  55. }
  56. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  57. // Progressive pull of bytes, push of pixels
  58. static inline int16_t qp_drawimage_byte_uncompressed_decoder(void* cb_arg) {
  59. struct qp_internal_byte_input_state* state = (struct qp_internal_byte_input_state*)cb_arg;
  60. state->curr = qp_stream_get(state->src_stream);
  61. return state->curr;
  62. }
  63. static inline int16_t qp_drawimage_byte_rle_decoder(void* cb_arg) {
  64. struct qp_internal_byte_input_state* state = (struct qp_internal_byte_input_state*)cb_arg;
  65. // Work out if we're parsing the initial marker byte
  66. if (state->rle.mode == MARKER_BYTE) {
  67. uint8_t c = qp_stream_get(state->src_stream);
  68. if (c >= 128) {
  69. state->rle.mode = NON_REPEATING_RUN; // non-repeated run
  70. state->rle.remain = c - 127;
  71. } else {
  72. state->rle.mode = REPEATING_RUN; // repeated run
  73. state->rle.remain = c;
  74. }
  75. state->curr = qp_stream_get(state->src_stream);
  76. }
  77. // Work out which byte we're returning
  78. uint8_t c = state->curr;
  79. // Decrement the counter of the bytes remaining
  80. state->rle.remain--;
  81. if (state->rle.remain > 0) {
  82. // If we're in a non-repeating run, queue up the next byte
  83. if (state->rle.mode == NON_REPEATING_RUN) {
  84. state->curr = qp_stream_get(state->src_stream);
  85. }
  86. } else {
  87. // Swap back to querying the marker byte mode
  88. state->rle.mode = MARKER_BYTE;
  89. }
  90. return c;
  91. }
  92. bool qp_internal_pixel_appender(qp_pixel_t* palette, uint8_t index, void* cb_arg) {
  93. struct qp_internal_pixel_output_state* state = (struct qp_internal_pixel_output_state*)cb_arg;
  94. struct painter_driver_t* driver = (struct painter_driver_t*)state->device;
  95. if (!driver->driver_vtable->append_pixels(state->device, qp_internal_global_pixdata_buffer, palette, state->pixel_write_pos++, 1, &index)) {
  96. return false;
  97. }
  98. // If we've hit the transmit limit, send out the entire buffer and reset the write position
  99. if (state->pixel_write_pos == state->max_pixels) {
  100. if (!driver->driver_vtable->pixdata(state->device, qp_internal_global_pixdata_buffer, state->pixel_write_pos)) {
  101. return false;
  102. }
  103. state->pixel_write_pos = 0;
  104. }
  105. return true;
  106. }
  107. qp_internal_byte_input_callback qp_internal_prepare_input_state(struct qp_internal_byte_input_state* input_state, painter_compression_t compression) {
  108. switch (compression) {
  109. case IMAGE_UNCOMPRESSED:
  110. return qp_drawimage_byte_uncompressed_decoder;
  111. case IMAGE_COMPRESSED_RLE:
  112. input_state->rle.mode = MARKER_BYTE;
  113. input_state->rle.remain = 0;
  114. return qp_drawimage_byte_rle_decoder;
  115. default:
  116. return NULL;
  117. }
  118. }