cam_hal.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  1. // Copyright 2010-2020 Espressif Systems (Shanghai) PTE LTD
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. #include <stdio.h>
  15. #include <string.h>
  16. #include "esp_heap_caps.h"
  17. #include "ll_cam.h"
  18. #include "cam_hal.h"
  19. #if (ESP_IDF_VERSION_MAJOR == 3) && (ESP_IDF_VERSION_MINOR == 3)
  20. #include "rom/ets_sys.h"
  21. #else
  22. #include "esp_timer.h"
  23. #if CONFIG_IDF_TARGET_ESP32
  24. #include "esp32/rom/ets_sys.h" // will be removed in idf v5.0
  25. #elif CONFIG_IDF_TARGET_ESP32S2
  26. #include "esp32s2/rom/ets_sys.h"
  27. #elif CONFIG_IDF_TARGET_ESP32S3
  28. #include "esp32s3/rom/ets_sys.h"
  29. #endif
  30. #endif // ESP_IDF_VERSION_MAJOR
  31. #define ESP_CAMERA_ETS_PRINTF ets_printf
  32. #if CONFIG_CAM_TASK_STACK_SIZE
  33. #define CAM_TASK_STACK CONFIG_CAM_TASK_STACK_SIZE
  34. #else
  35. #define CAM_TASK_STACK (2*1024)
  36. #endif
  37. static const char *TAG = "cam_hal";
  38. static cam_obj_t *cam_obj = NULL;
  39. static const uint32_t JPEG_SOI_MARKER = 0xFFD8FF; // written in little-endian for esp32
  40. static const uint16_t JPEG_EOI_MARKER = 0xD9FF; // written in little-endian for esp32
  41. static int cam_verify_jpeg_soi(const uint8_t *inbuf, uint32_t length)
  42. {
  43. uint32_t sig = *((uint32_t *)inbuf) & 0xFFFFFF;
  44. if(sig != JPEG_SOI_MARKER) {
  45. for (uint32_t i = 0; i < length; i++) {
  46. sig = *((uint32_t *)(&inbuf[i])) & 0xFFFFFF;
  47. if (sig == JPEG_SOI_MARKER) {
  48. ESP_LOGW(TAG, "SOI: %d", (int) i);
  49. return i;
  50. }
  51. }
  52. ESP_LOGW(TAG, "NO-SOI");
  53. return -1;
  54. }
  55. return 0;
  56. }
  57. static int cam_verify_jpeg_eoi(const uint8_t *inbuf, uint32_t length)
  58. {
  59. int offset = -1;
  60. uint8_t *dptr = (uint8_t *)inbuf + length - 2;
  61. while (dptr > inbuf) {
  62. uint16_t sig = *((uint16_t *)dptr);
  63. if (JPEG_EOI_MARKER == sig) {
  64. offset = dptr - inbuf;
  65. //ESP_LOGW(TAG, "EOI: %d", length - (offset + 2));
  66. return offset;
  67. }
  68. dptr--;
  69. }
  70. return -1;
  71. }
  72. static bool cam_get_next_frame(int * frame_pos)
  73. {
  74. if(!cam_obj->frames[*frame_pos].en){
  75. for (int x = 0; x < cam_obj->frame_cnt; x++) {
  76. if (cam_obj->frames[x].en) {
  77. *frame_pos = x;
  78. return true;
  79. }
  80. }
  81. } else {
  82. return true;
  83. }
  84. return false;
  85. }
  86. static bool cam_start_frame(int * frame_pos)
  87. {
  88. if (cam_get_next_frame(frame_pos)) {
  89. if(ll_cam_start(cam_obj, *frame_pos)){
  90. // Vsync the frame manually
  91. ll_cam_do_vsync(cam_obj);
  92. uint64_t us = (uint64_t)esp_timer_get_time();
  93. cam_obj->frames[*frame_pos].fb.timestamp.tv_sec = us / 1000000UL;
  94. cam_obj->frames[*frame_pos].fb.timestamp.tv_usec = us % 1000000UL;
  95. return true;
  96. }
  97. }
  98. return false;
  99. }
  100. void IRAM_ATTR ll_cam_send_event(cam_obj_t *cam, cam_event_t cam_event, BaseType_t * HPTaskAwoken)
  101. {
  102. if (xQueueSendFromISR(cam->event_queue, (void *)&cam_event, HPTaskAwoken) != pdTRUE) {
  103. ll_cam_stop(cam);
  104. cam->state = CAM_STATE_IDLE;
  105. ESP_CAMERA_ETS_PRINTF(DRAM_STR("cam_hal: EV-%s-OVF\r\n"), cam_event==CAM_IN_SUC_EOF_EVENT ? DRAM_STR("EOF") : DRAM_STR("VSYNC"));
  106. }
  107. }
  108. //Copy fram from DMA dma_buffer to fram dma_buffer
  109. static void cam_task(void *arg)
  110. {
  111. int cnt = 0;
  112. int frame_pos = 0;
  113. cam_obj->state = CAM_STATE_IDLE;
  114. cam_event_t cam_event = 0;
  115. xQueueReset(cam_obj->event_queue);
  116. while (1) {
  117. xQueueReceive(cam_obj->event_queue, (void *)&cam_event, portMAX_DELAY);
  118. DBG_PIN_SET(1);
  119. switch (cam_obj->state) {
  120. case CAM_STATE_IDLE: {
  121. if (cam_event == CAM_VSYNC_EVENT) {
  122. //DBG_PIN_SET(1);
  123. if(cam_start_frame(&frame_pos)){
  124. cam_obj->frames[frame_pos].fb.len = 0;
  125. cam_obj->state = CAM_STATE_READ_BUF;
  126. }
  127. cnt = 0;
  128. }
  129. }
  130. break;
  131. case CAM_STATE_READ_BUF: {
  132. camera_fb_t * frame_buffer_event = &cam_obj->frames[frame_pos].fb;
  133. size_t pixels_per_dma = (cam_obj->dma_half_buffer_size * cam_obj->fb_bytes_per_pixel) / (cam_obj->dma_bytes_per_item * cam_obj->in_bytes_per_pixel);
  134. if (cam_event == CAM_IN_SUC_EOF_EVENT) {
  135. if(!cam_obj->psram_mode){
  136. if (cam_obj->fb_size < (frame_buffer_event->len + pixels_per_dma)) {
  137. ESP_LOGW(TAG, "FB-OVF");
  138. ll_cam_stop(cam_obj);
  139. DBG_PIN_SET(0);
  140. continue;
  141. }
  142. frame_buffer_event->len += ll_cam_memcpy(cam_obj,
  143. &frame_buffer_event->buf[frame_buffer_event->len],
  144. &cam_obj->dma_buffer[(cnt % cam_obj->dma_half_buffer_cnt) * cam_obj->dma_half_buffer_size],
  145. cam_obj->dma_half_buffer_size);
  146. }
  147. //Check for JPEG SOI in the first buffer. stop if not found
  148. if (cam_obj->jpeg_mode && cnt == 0 && cam_verify_jpeg_soi(frame_buffer_event->buf, frame_buffer_event->len) != 0) {
  149. ll_cam_stop(cam_obj);
  150. cam_obj->state = CAM_STATE_IDLE;
  151. }
  152. cnt++;
  153. } else if (cam_event == CAM_VSYNC_EVENT) {
  154. //DBG_PIN_SET(1);
  155. ll_cam_stop(cam_obj);
  156. if (cnt || !cam_obj->jpeg_mode || cam_obj->psram_mode) {
  157. if (cam_obj->jpeg_mode) {
  158. if (!cam_obj->psram_mode) {
  159. if (cam_obj->fb_size < (frame_buffer_event->len + pixels_per_dma)) {
  160. ESP_LOGW(TAG, "FB-OVF");
  161. cnt--;
  162. } else {
  163. frame_buffer_event->len += ll_cam_memcpy(cam_obj,
  164. &frame_buffer_event->buf[frame_buffer_event->len],
  165. &cam_obj->dma_buffer[(cnt % cam_obj->dma_half_buffer_cnt) * cam_obj->dma_half_buffer_size],
  166. cam_obj->dma_half_buffer_size);
  167. }
  168. }
  169. cnt++;
  170. }
  171. cam_obj->frames[frame_pos].en = 0;
  172. if (cam_obj->psram_mode) {
  173. if (cam_obj->jpeg_mode) {
  174. frame_buffer_event->len = cnt * cam_obj->dma_half_buffer_size;
  175. } else {
  176. frame_buffer_event->len = cam_obj->recv_size;
  177. }
  178. } else if (!cam_obj->jpeg_mode) {
  179. if (frame_buffer_event->len != cam_obj->fb_size) {
  180. cam_obj->frames[frame_pos].en = 1;
  181. ESP_LOGE(TAG, "FB-SIZE: %u != %u", frame_buffer_event->len, (unsigned) cam_obj->fb_size);
  182. }
  183. }
  184. //send frame
  185. if(!cam_obj->frames[frame_pos].en && xQueueSend(cam_obj->frame_buffer_queue, (void *)&frame_buffer_event, 0) != pdTRUE) {
  186. //pop frame buffer from the queue
  187. camera_fb_t * fb2 = NULL;
  188. if(xQueueReceive(cam_obj->frame_buffer_queue, &fb2, 0) == pdTRUE) {
  189. //push the new frame to the end of the queue
  190. if (xQueueSend(cam_obj->frame_buffer_queue, (void *)&frame_buffer_event, 0) != pdTRUE) {
  191. cam_obj->frames[frame_pos].en = 1;
  192. ESP_LOGE(TAG, "FBQ-SND");
  193. }
  194. //free the popped buffer
  195. cam_give(fb2);
  196. } else {
  197. //queue is full and we could not pop a frame from it
  198. cam_obj->frames[frame_pos].en = 1;
  199. ESP_LOGE(TAG, "FBQ-RCV");
  200. }
  201. }
  202. }
  203. if(!cam_start_frame(&frame_pos)){
  204. cam_obj->state = CAM_STATE_IDLE;
  205. } else {
  206. cam_obj->frames[frame_pos].fb.len = 0;
  207. }
  208. cnt = 0;
  209. }
  210. }
  211. break;
  212. }
  213. DBG_PIN_SET(0);
  214. }
  215. }
  216. static lldesc_t * allocate_dma_descriptors(uint32_t count, uint16_t size, uint8_t * buffer)
  217. {
  218. lldesc_t *dma = (lldesc_t *)heap_caps_malloc(count * sizeof(lldesc_t), MALLOC_CAP_DMA);
  219. if (dma == NULL) {
  220. return dma;
  221. }
  222. for (int x = 0; x < count; x++) {
  223. dma[x].size = size;
  224. dma[x].length = 0;
  225. dma[x].sosf = 0;
  226. dma[x].eof = 0;
  227. dma[x].owner = 1;
  228. dma[x].buf = (buffer + size * x);
  229. dma[x].empty = (uint32_t)&dma[(x + 1) % count];
  230. }
  231. return dma;
  232. }
  233. static esp_err_t cam_dma_config(const camera_config_t *config)
  234. {
  235. bool ret = ll_cam_dma_sizes(cam_obj);
  236. if (0 == ret) {
  237. return ESP_FAIL;
  238. }
  239. cam_obj->dma_node_cnt = (cam_obj->dma_buffer_size) / cam_obj->dma_node_buffer_size; // Number of DMA nodes
  240. cam_obj->frame_copy_cnt = cam_obj->recv_size / cam_obj->dma_half_buffer_size; // Number of interrupted copies, ping-pong copy
  241. ESP_LOGI(TAG, "buffer_size: %d, half_buffer_size: %d, node_buffer_size: %d, node_cnt: %d, total_cnt: %d",
  242. (int) cam_obj->dma_buffer_size, (int) cam_obj->dma_half_buffer_size, (int) cam_obj->dma_node_buffer_size,
  243. (int) cam_obj->dma_node_cnt, (int) cam_obj->frame_copy_cnt);
  244. cam_obj->dma_buffer = NULL;
  245. cam_obj->dma = NULL;
  246. cam_obj->frames = (cam_frame_t *)heap_caps_calloc(1, cam_obj->frame_cnt * sizeof(cam_frame_t), MALLOC_CAP_DEFAULT);
  247. CAM_CHECK(cam_obj->frames != NULL, "frames malloc failed", ESP_FAIL);
  248. uint8_t dma_align = 0;
  249. size_t fb_size = cam_obj->fb_size;
  250. if (cam_obj->psram_mode) {
  251. dma_align = ll_cam_get_dma_align(cam_obj);
  252. if (cam_obj->fb_size < cam_obj->recv_size) {
  253. fb_size = cam_obj->recv_size;
  254. }
  255. }
  256. /* Allocate memory for frame buffer */
  257. size_t alloc_size = fb_size * sizeof(uint8_t) + dma_align;
  258. uint32_t _caps = MALLOC_CAP_8BIT;
  259. if (CAMERA_FB_IN_DRAM == config->fb_location) {
  260. _caps |= MALLOC_CAP_INTERNAL;
  261. } else {
  262. _caps |= MALLOC_CAP_SPIRAM;
  263. }
  264. for (int x = 0; x < cam_obj->frame_cnt; x++) {
  265. cam_obj->frames[x].dma = NULL;
  266. cam_obj->frames[x].fb_offset = 0;
  267. cam_obj->frames[x].en = 0;
  268. ESP_LOGI(TAG, "Allocating %d Byte frame buffer in %s", alloc_size, _caps & MALLOC_CAP_SPIRAM ? "PSRAM" : "OnBoard RAM");
  269. #if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(4, 3, 0)
  270. // In IDF v4.2 and earlier, memory returned by heap_caps_aligned_alloc must be freed using heap_caps_aligned_free.
  271. // And heap_caps_aligned_free is deprecated on v4.3.
  272. cam_obj->frames[x].fb.buf = (uint8_t *)heap_caps_aligned_alloc(16, alloc_size, _caps);
  273. #else
  274. cam_obj->frames[x].fb.buf = (uint8_t *)heap_caps_malloc(alloc_size, _caps);
  275. #endif
  276. CAM_CHECK(cam_obj->frames[x].fb.buf != NULL, "frame buffer malloc failed", ESP_FAIL);
  277. if (cam_obj->psram_mode) {
  278. //align PSRAM buffer. TODO: save the offset so proper address can be freed later
  279. cam_obj->frames[x].fb_offset = dma_align - ((uint32_t)cam_obj->frames[x].fb.buf & (dma_align - 1));
  280. cam_obj->frames[x].fb.buf += cam_obj->frames[x].fb_offset;
  281. ESP_LOGI(TAG, "Frame[%d]: Offset: %u, Addr: 0x%08X", x, cam_obj->frames[x].fb_offset, (unsigned) cam_obj->frames[x].fb.buf);
  282. cam_obj->frames[x].dma = allocate_dma_descriptors(cam_obj->dma_node_cnt, cam_obj->dma_node_buffer_size, cam_obj->frames[x].fb.buf);
  283. CAM_CHECK(cam_obj->frames[x].dma != NULL, "frame dma malloc failed", ESP_FAIL);
  284. }
  285. cam_obj->frames[x].en = 1;
  286. }
  287. if (!cam_obj->psram_mode) {
  288. cam_obj->dma_buffer = (uint8_t *)heap_caps_malloc(cam_obj->dma_buffer_size * sizeof(uint8_t), MALLOC_CAP_DMA);
  289. if(NULL == cam_obj->dma_buffer) {
  290. ESP_LOGE(TAG,"%s(%d): DMA buffer %d Byte malloc failed, the current largest free block:%d Byte", __FUNCTION__, __LINE__,
  291. (int) cam_obj->dma_buffer_size, (int) heap_caps_get_largest_free_block(MALLOC_CAP_DMA));
  292. return ESP_FAIL;
  293. }
  294. cam_obj->dma = allocate_dma_descriptors(cam_obj->dma_node_cnt, cam_obj->dma_node_buffer_size, cam_obj->dma_buffer);
  295. CAM_CHECK(cam_obj->dma != NULL, "dma malloc failed", ESP_FAIL);
  296. }
  297. return ESP_OK;
  298. }
  299. esp_err_t cam_init(const camera_config_t *config)
  300. {
  301. CAM_CHECK(NULL != config, "config pointer is invalid", ESP_ERR_INVALID_ARG);
  302. esp_err_t ret = ESP_OK;
  303. cam_obj = (cam_obj_t *)heap_caps_calloc(1, sizeof(cam_obj_t), MALLOC_CAP_DMA);
  304. CAM_CHECK(NULL != cam_obj, "lcd_cam object malloc error", ESP_ERR_NO_MEM);
  305. cam_obj->swap_data = 0;
  306. cam_obj->vsync_pin = config->pin_vsync;
  307. cam_obj->vsync_invert = true;
  308. ll_cam_set_pin(cam_obj, config);
  309. ret = ll_cam_config(cam_obj, config);
  310. CAM_CHECK_GOTO(ret == ESP_OK, "ll_cam initialize failed", err);
  311. #if CAMERA_DBG_PIN_ENABLE
  312. PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[DBG_PIN_NUM], PIN_FUNC_GPIO);
  313. gpio_set_direction(DBG_PIN_NUM, GPIO_MODE_OUTPUT);
  314. gpio_set_pull_mode(DBG_PIN_NUM, GPIO_FLOATING);
  315. #endif
  316. ESP_LOGI(TAG, "cam init ok");
  317. return ESP_OK;
  318. err:
  319. free(cam_obj);
  320. cam_obj = NULL;
  321. return ESP_FAIL;
  322. }
  323. esp_err_t cam_config(const camera_config_t *config, framesize_t frame_size, uint16_t sensor_pid)
  324. {
  325. CAM_CHECK(NULL != config, "config pointer is invalid", ESP_ERR_INVALID_ARG);
  326. esp_err_t ret = ESP_OK;
  327. ret = ll_cam_set_sample_mode(cam_obj, (pixformat_t)config->pixel_format, config->xclk_freq_hz, sensor_pid);
  328. cam_obj->jpeg_mode = config->pixel_format == PIXFORMAT_JPEG;
  329. #if CONFIG_IDF_TARGET_ESP32
  330. cam_obj->psram_mode = false;
  331. #else
  332. cam_obj->psram_mode = (config->xclk_freq_hz == 16000000);
  333. #endif
  334. cam_obj->frame_cnt = config->fb_count;
  335. cam_obj->width = resolution[frame_size].width;
  336. cam_obj->height = resolution[frame_size].height;
  337. if(cam_obj->jpeg_mode){
  338. cam_obj->recv_size = cam_obj->width * cam_obj->height / 5;
  339. cam_obj->fb_size = cam_obj->recv_size;
  340. } else {
  341. cam_obj->recv_size = cam_obj->width * cam_obj->height * cam_obj->in_bytes_per_pixel;
  342. cam_obj->fb_size = cam_obj->width * cam_obj->height * cam_obj->fb_bytes_per_pixel;
  343. }
  344. ret = cam_dma_config(config);
  345. CAM_CHECK_GOTO(ret == ESP_OK, "cam_dma_config failed", err);
  346. cam_obj->event_queue = xQueueCreate(cam_obj->dma_half_buffer_cnt - 1, sizeof(cam_event_t));
  347. CAM_CHECK_GOTO(cam_obj->event_queue != NULL, "event_queue create failed", err);
  348. size_t frame_buffer_queue_len = cam_obj->frame_cnt;
  349. if (config->grab_mode == CAMERA_GRAB_LATEST && cam_obj->frame_cnt > 1) {
  350. frame_buffer_queue_len = cam_obj->frame_cnt - 1;
  351. }
  352. cam_obj->frame_buffer_queue = xQueueCreate(frame_buffer_queue_len, sizeof(camera_fb_t*));
  353. CAM_CHECK_GOTO(cam_obj->frame_buffer_queue != NULL, "frame_buffer_queue create failed", err);
  354. ret = ll_cam_init_isr(cam_obj);
  355. CAM_CHECK_GOTO(ret == ESP_OK, "cam intr alloc failed", err);
  356. #if CONFIG_CAMERA_CORE0
  357. xTaskCreatePinnedToCore(cam_task, "cam_task", CAM_TASK_STACK, NULL, configMAX_PRIORITIES - 2, &cam_obj->task_handle, 0);
  358. #elif CONFIG_CAMERA_CORE1
  359. xTaskCreatePinnedToCore(cam_task, "cam_task", CAM_TASK_STACK, NULL, configMAX_PRIORITIES - 2, &cam_obj->task_handle, 1);
  360. #else
  361. xTaskCreate(cam_task, "cam_task", CAM_TASK_STACK, NULL, configMAX_PRIORITIES - 2, &cam_obj->task_handle);
  362. #endif
  363. ESP_LOGI(TAG, "cam config ok");
  364. return ESP_OK;
  365. err:
  366. cam_deinit();
  367. return ESP_FAIL;
  368. }
  369. esp_err_t cam_deinit(void)
  370. {
  371. if (!cam_obj) {
  372. return ESP_FAIL;
  373. }
  374. cam_stop();
  375. if (cam_obj->task_handle) {
  376. vTaskDelete(cam_obj->task_handle);
  377. }
  378. if (cam_obj->event_queue) {
  379. vQueueDelete(cam_obj->event_queue);
  380. }
  381. if (cam_obj->frame_buffer_queue) {
  382. vQueueDelete(cam_obj->frame_buffer_queue);
  383. }
  384. if (cam_obj->dma) {
  385. free(cam_obj->dma);
  386. }
  387. if (cam_obj->dma_buffer) {
  388. free(cam_obj->dma_buffer);
  389. }
  390. if (cam_obj->frames) {
  391. for (int x = 0; x < cam_obj->frame_cnt; x++) {
  392. free(cam_obj->frames[x].fb.buf - cam_obj->frames[x].fb_offset);
  393. if (cam_obj->frames[x].dma) {
  394. free(cam_obj->frames[x].dma);
  395. }
  396. }
  397. free(cam_obj->frames);
  398. }
  399. ll_cam_deinit(cam_obj);
  400. free(cam_obj);
  401. cam_obj = NULL;
  402. return ESP_OK;
  403. }
  404. void cam_stop(void)
  405. {
  406. ll_cam_vsync_intr_enable(cam_obj, false);
  407. ll_cam_stop(cam_obj);
  408. }
  409. void cam_start(void)
  410. {
  411. ll_cam_vsync_intr_enable(cam_obj, true);
  412. }
  413. camera_fb_t *cam_take(TickType_t timeout)
  414. {
  415. camera_fb_t *dma_buffer = NULL;
  416. TickType_t start = xTaskGetTickCount();
  417. xQueueReceive(cam_obj->frame_buffer_queue, (void *)&dma_buffer, timeout);
  418. if (dma_buffer) {
  419. if(cam_obj->jpeg_mode){
  420. // find the end marker for JPEG. Data after that can be discarded
  421. int offset_e = cam_verify_jpeg_eoi(dma_buffer->buf, dma_buffer->len);
  422. if (offset_e >= 0) {
  423. // adjust buffer length
  424. dma_buffer->len = offset_e + sizeof(JPEG_EOI_MARKER);
  425. return dma_buffer;
  426. } else {
  427. ESP_LOGW(TAG, "NO-EOI");
  428. cam_give(dma_buffer);
  429. return cam_take(timeout - (xTaskGetTickCount() - start));//recurse!!!!
  430. }
  431. } else if(cam_obj->psram_mode && cam_obj->in_bytes_per_pixel != cam_obj->fb_bytes_per_pixel){
  432. //currently this is used only for YUV to GRAYSCALE
  433. dma_buffer->len = ll_cam_memcpy(cam_obj, dma_buffer->buf, dma_buffer->buf, dma_buffer->len);
  434. }
  435. return dma_buffer;
  436. } else {
  437. ESP_LOGW(TAG, "Failed to get the frame on time!");
  438. }
  439. return NULL;
  440. }
  441. void cam_give(camera_fb_t *dma_buffer)
  442. {
  443. for (int x = 0; x < cam_obj->frame_cnt; x++) {
  444. if (&cam_obj->frames[x].fb == dma_buffer) {
  445. cam_obj->frames[x].en = 1;
  446. break;
  447. }
  448. }
  449. }