jpeg_hal.c 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. /*
  2. * Copyright (c) 2020, Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /****************************************************************************
  7. * Included Files
  8. ****************************************************************************/
  9. #include <string.h>
  10. #include <assert.h>
  11. #include <os_common_api.h>
  12. #include <jpeg_hal.h>
  13. #include <mem_manager.h>
  14. #include <logging/log.h>
  15. #ifdef CONFIG_DISPLAY
  16. #include <ui_region.h>
  17. #endif
  18. #include <memory/mem_cache.h>
  19. #include <spicache.h>
  20. #define JPEG_MIN_SIZE 16
  21. #define JPEG_HW_TIMEOUT (50) //ms
  22. /****************************************************************************
  23. * Private Data
  24. ****************************************************************************/
  25. static OS_MUTEX_DEFINE(g_parser_info_mutex);
  26. static OS_MUTEX_DEFINE(g_decode_mutex);
  27. static jpeg_parser_info_t g_parser_info;
  28. /****************************************************************************
  29. * Private Functions
  30. ****************************************************************************/
  31. static void _hal_jpeg_device_handler(int err_code, void *user_data)
  32. {
  33. LOG_INF("_hal_jpeg_device_handler err_code (%d)\n",err_code);
  34. }
  35. /****************************************************************************
  36. * Public Functions
  37. ****************************************************************************/
  38. int hal_jpeg_decode_open(hal_jpeg_handle_t *hjpeg)
  39. {
  40. return jpeg_open(hjpeg->device);
  41. }
  42. static int _hal_jpeg_decode_sub_image(hal_jpeg_handle_t *hjpeg,
  43. void *jpeg_src, int jpeg_size,
  44. void *bmp_buffer, int output_format, int output_stride,
  45. int win_x, int win_y, int win_w, int win_h)
  46. {
  47. int ret = 0;
  48. struct jpeg_info_t *jpeg_info = NULL;
  49. os_mutex_lock(&g_parser_info_mutex, OS_FOREVER);
  50. jpeg_parser_info_t *parser_info = &g_parser_info;
  51. memset(parser_info, 0 , sizeof(jpeg_parser_info_t));
  52. parser_info->jpeg_base = jpeg_src;
  53. parser_info->jpeg_size = jpeg_size;
  54. ret = jpeg_parser_process(parser_info, 1);
  55. if (ret) {
  56. LOG_ERR("jpeg_parser_process failed (%d)\n",ret);
  57. ret = -HAL_JPEG_PARSER_ERROR;
  58. goto err_exit;
  59. }
  60. jpeg_info = &parser_info->jpeg_info;
  61. jpeg_info->window_x = win_x;
  62. jpeg_info->window_y = win_y;
  63. jpeg_info->window_w = win_w;
  64. jpeg_info->window_h = win_h;
  65. jpeg_info->output_stride = output_stride;
  66. jpeg_info->output_bmp = bmp_buffer;
  67. jpeg_info->output_format = output_format;
  68. jpeg_info->stream_offset = 0;
  69. jpeg_info->stream_size = (jpeg_info->stream_size + 511) / 512 * 512;
  70. ret = jpeg_config(hjpeg->device, jpeg_info);
  71. if (ret) {
  72. LOG_ERR("jpeg_config failed (%d)\n",ret);
  73. ret = -HAL_JPEG_CONFIG_DECODER_ERROR;
  74. goto err_exit;
  75. }
  76. ret = jpeg_decode(hjpeg->device);
  77. if (ret) {
  78. LOG_ERR("jpeg_decode failed (%d)\n",ret);
  79. ret = -HAL_JPEG_DECODER_ERROR;
  80. goto err_exit;
  81. }
  82. ret = HAL_JPEG_ERROR_NONE;
  83. err_exit:
  84. os_mutex_unlock(&g_parser_info_mutex);
  85. return ret;
  86. }
  87. int hal_jpeg_decode(hal_jpeg_handle_t *hjpeg, void *jpeg_src, int jpeg_size,
  88. void *bmp_buffer, int output_format, int output_stride,
  89. int win_x, int win_y, int win_w, int win_h)
  90. {
  91. int ret = 0;
  92. struct jpeg_head_info *head_info = (struct jpeg_head_info *)jpeg_src;
  93. os_strace_u32x4(SYS_TRACE_ID_IMG_DECODE, 0, (uint32_t)jpeg_src, win_w, win_h);
  94. #ifndef CONFIG_SOC_NO_PSRAM
  95. if (buf_is_psram(jpeg_src)) {
  96. mem_dcache_flush(jpeg_src, jpeg_size);
  97. mem_dcache_sync();
  98. }
  99. if (buf_is_psram(bmp_buffer)) {
  100. if(output_format == HAL_JPEG_OUT_RGB888) {
  101. mem_dcache_flush(bmp_buffer, win_h * output_stride * 3);
  102. } else {
  103. mem_dcache_flush(bmp_buffer, win_h * output_stride * 2);
  104. }
  105. mem_dcache_sync();
  106. }
  107. #endif
  108. if (head_info->flag != JPEG_FLAG) {
  109. ret = _hal_jpeg_decode_sub_image(hjpeg, jpeg_src, jpeg_size,
  110. bmp_buffer, output_format, output_stride,
  111. win_x, win_y, win_w, win_h);
  112. return ret;
  113. }
  114. if (head_info->split_num <= 1) {
  115. ret = _hal_jpeg_decode_sub_image(hjpeg, (uint8_t *)jpeg_src + sizeof(*head_info),
  116. jpeg_size - sizeof(*head_info), bmp_buffer, output_format, output_stride,
  117. win_x, win_y, win_w, win_h);
  118. return ret;
  119. }
  120. #ifdef CONFIG_DISPLAY
  121. uint8_t h_block_num = (head_info->pic_height + head_info->split_height - 1)
  122. / head_info->split_height ;
  123. uint8_t w_block_num = (head_info->pic_width + head_info->split_width - 1)
  124. / head_info->split_width;
  125. uint8_t bytes_per_pixel = output_format ? 2 : 3;
  126. jpeg_sub_info_t sub_info[MAX_BLOCK_NUM];
  127. memcpy(sub_info, (void*)&head_info->offset, h_block_num * w_block_num * 4);
  128. for (int j = 0; j < h_block_num; j++) {
  129. for (int i = 0; i < w_block_num; i++) {
  130. int sub_jpeg_size = sub_info[j * w_block_num + i].size;
  131. int dec_w, dec_h;
  132. ui_region_t pic_region = {
  133. .x1 = i * head_info->split_width,
  134. .y1 = j * head_info->split_height,
  135. .x2 = (i + 1) * head_info->split_width - 1,
  136. .y2 = (j + 1) * head_info->split_height - 1,
  137. };
  138. if (pic_region.y2 > head_info->pic_height - 1) {
  139. pic_region.y2 = head_info->pic_height - 1;
  140. }
  141. if (pic_region.x2 > head_info->pic_width - 1) {
  142. pic_region.x2 = head_info->pic_width - 1;
  143. }
  144. ui_region_t output_region = {
  145. .x1 = win_x,
  146. .y1 = win_y,
  147. .x2 = win_x + win_w - 1,
  148. .y2 = win_y + win_h - 1,
  149. };
  150. ui_region_t dec_region;
  151. if (ui_region_intersect(&dec_region, &output_region, &pic_region) == false) {
  152. continue;
  153. }
  154. void* sub_jpeg_src = (void*)((uint32_t)jpeg_src + sub_info[j * w_block_num + i].offset);
  155. void* sub_out_dest = (void*)((uint32_t)bmp_buffer
  156. + (dec_region.x1 - win_x) * bytes_per_pixel
  157. + (dec_region.y1 - win_y) * output_stride * bytes_per_pixel);
  158. dec_w = dec_region.x2 - dec_region.x1 + 1;
  159. dec_h = dec_region.y2 - dec_region.y1 + 1;
  160. ret = _hal_jpeg_decode_sub_image(hjpeg, sub_jpeg_src, sub_jpeg_size, sub_out_dest,
  161. output_format, output_stride,
  162. dec_region.x1 - pic_region.x1 , dec_region.y1 - pic_region.y1,
  163. dec_w,
  164. dec_h);
  165. if (ret != 0) {
  166. LOG_ERR("jpeg_decode failed (%d)\n",ret);
  167. ret = -HAL_JPEG_DECODER_ERROR;
  168. goto err_exit;
  169. }
  170. if (dec_w * dec_h < JPEG_MIN_SIZE * JPEG_MIN_SIZE) {
  171. jpeg_decode_wait_finished(hjpeg->device, 1);
  172. } else {
  173. jpeg_decode_wait_finished(hjpeg->device, JPEG_HW_TIMEOUT);
  174. }
  175. }
  176. }
  177. err_exit:
  178. os_strace_end_call(SYS_TRACE_ID_IMG_DECODE);
  179. return ret;
  180. #else
  181. os_strace_end_call(SYS_TRACE_ID_IMG_DECODE);
  182. return 0;
  183. #endif
  184. }
  185. int hal_jpeg_get_picture_info(void *jpeg_src , uint32_t jpeg_size,
  186. uint32_t *picture_w, uint32_t *picture_h)
  187. {
  188. struct jpeg_head_info *head_info = (struct jpeg_head_info *)jpeg_src;
  189. jpeg_parser_info_t *parser_info = &g_parser_info;
  190. if (head_info->flag == JPEG_FLAG) {
  191. *picture_w = head_info->pic_width;
  192. *picture_h = head_info->pic_height;
  193. return 0;
  194. }
  195. os_mutex_lock(&g_parser_info_mutex, OS_FOREVER);
  196. memset(parser_info, 0, sizeof(jpeg_parser_info_t));
  197. parser_info->jpeg_base = jpeg_src;
  198. parser_info->jpeg_size = jpeg_size;
  199. if (jpeg_parser_process(parser_info, 1)) {
  200. os_mutex_unlock(&g_parser_info_mutex);
  201. return HAL_JPEG_PARSER_ERROR;
  202. }
  203. *picture_w = parser_info->jpeg_info.image_w;
  204. *picture_h = parser_info->jpeg_info.image_h;
  205. os_mutex_unlock(&g_parser_info_mutex);
  206. return 0;
  207. }
  208. int hal_jpeg_decode_wait_finised(hal_jpeg_handle_t *hjpeg, int timeout)
  209. {
  210. return jpeg_decode_wait_finished(hjpeg->device, timeout);
  211. }
  212. int hal_jpeg_decode_close(hal_jpeg_handle_t *hjpeg)
  213. {
  214. return jpeg_close(hjpeg->device);
  215. }
  216. int hal_jpeg_init(hal_jpeg_handle_t *hjpeg, uint32_t preferred_modes)
  217. {
  218. /* Check the jpeg peripheral existence */
  219. hjpeg->device = device_get_binding(CONFIG_JPEG_HW_DEV_NAME);
  220. if (hjpeg->device == NULL) {
  221. return -ENODEV;
  222. }
  223. /* Register the jpeg device instance callback */
  224. jpeg_register_callback(hjpeg->device, _hal_jpeg_device_handler, hjpeg);
  225. return 0;
  226. }
  227. int hal_jpeg_deinit(hal_jpeg_handle_t *hjpeg)
  228. {
  229. /* Assign invald value */
  230. hjpeg->device = NULL;
  231. return 0;
  232. }
  233. int jpg_decode(void* jpeg_src, int jpeg_size,
  234. void* bmp_buffer, int output_format, int output_stride,
  235. int win_x, int win_y, int win_w, int win_h)
  236. {
  237. static hal_jpeg_handle_t jpg_decoder;
  238. static bool jpg_inited = false;
  239. int res;
  240. int bytes_per_pixel = output_format ? 2 : 3;
  241. res = os_mutex_lock(&g_decode_mutex, OS_FOREVER);
  242. if (res) {
  243. return res;
  244. }
  245. if (jpg_inited == false) {
  246. res = hal_jpeg_init(&jpg_decoder, 0);
  247. if (res)
  248. return res;
  249. jpg_inited = true;
  250. }
  251. res = hal_jpeg_decode_open(&jpg_decoder);
  252. if (res) {
  253. LOG_ERR("hal_jpeg_decode_open fail\n");
  254. os_mutex_unlock(&g_decode_mutex);
  255. return res;
  256. }
  257. res = hal_jpeg_decode(&jpg_decoder, (void*)jpeg_src, (int)jpeg_size,
  258. bmp_buffer, output_format, output_stride,
  259. win_x, win_y, win_w, win_h);
  260. hal_jpeg_decode_wait_finised(&jpg_decoder, JPEG_HW_TIMEOUT);
  261. hal_jpeg_decode_close(&jpg_decoder);
  262. os_mutex_unlock(&g_decode_mutex);
  263. return res ? res : (win_w * win_h * bytes_per_pixel);
  264. }