esp32s2: Implement parallel image capture
This commit is contained in:
parent
268717e427
commit
2cbdd18a20
@ -115,6 +115,10 @@ msgstr ""
|
|||||||
msgid "%q must be a tuple of length 2"
|
msgid "%q must be a tuple of length 2"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
|
#: ports/esp32s2/common-hal/imagecapture/ParallelImageCapture.c
|
||||||
|
msgid "%q must be between %d and %d"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
#: ports/raspberrypi/common-hal/rp2pio/StateMachine.c
|
#: ports/raspberrypi/common-hal/rp2pio/StateMachine.c
|
||||||
#: shared-bindings/canio/Match.c
|
#: shared-bindings/canio/Match.c
|
||||||
msgid "%q out of range"
|
msgid "%q out of range"
|
||||||
@ -3825,6 +3829,7 @@ msgstr ""
|
|||||||
#: ports/esp32s2/boards/artisense_rd00/mpconfigboard.h
|
#: ports/esp32s2/boards/artisense_rd00/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/atmegazero_esp32s2/mpconfigboard.h
|
#: ports/esp32s2/boards/atmegazero_esp32s2/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/electroniccats_bastwifi/mpconfigboard.h
|
#: ports/esp32s2/boards/electroniccats_bastwifi/mpconfigboard.h
|
||||||
|
#: ports/esp32s2/boards/espressif_kaluga_1.3/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/espressif_kaluga_1/mpconfigboard.h
|
#: ports/esp32s2/boards/espressif_kaluga_1/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/espressif_saola_1_wroom/mpconfigboard.h
|
#: ports/esp32s2/boards/espressif_saola_1_wroom/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/espressif_saola_1_wrover/mpconfigboard.h
|
#: ports/esp32s2/boards/espressif_saola_1_wrover/mpconfigboard.h
|
||||||
@ -3832,7 +3837,8 @@ msgstr ""
|
|||||||
#: ports/esp32s2/boards/franzininho_wifi_wrover/mpconfigboard.h
|
#: ports/esp32s2/boards/franzininho_wifi_wrover/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/lilygo_ttgo_t8_s2_st7789/mpconfigboard.h
|
#: ports/esp32s2/boards/lilygo_ttgo_t8_s2_st7789/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/microdev_micro_s2/mpconfigboard.h
|
#: ports/esp32s2/boards/microdev_micro_s2/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/muselab_nanoesp32_s2/mpconfigboard.h
|
#: ports/esp32s2/boards/muselab_nanoesp32_s2_wroom/mpconfigboard.h
|
||||||
|
#: ports/esp32s2/boards/muselab_nanoesp32_s2_wrover/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/targett_module_clip_wroom/mpconfigboard.h
|
#: ports/esp32s2/boards/targett_module_clip_wroom/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/targett_module_clip_wrover/mpconfigboard.h
|
#: ports/esp32s2/boards/targett_module_clip_wrover/mpconfigboard.h
|
||||||
#: ports/esp32s2/boards/unexpectedmaker_feathers2/mpconfigboard.h
|
#: ports/esp32s2/boards/unexpectedmaker_feathers2/mpconfigboard.h
|
||||||
|
@ -41,6 +41,15 @@ static i2s_t *i2s_instance[I2S_NUM_MAX];
|
|||||||
static QueueHandle_t i2s_queues[I2S_NUM_MAX];
|
static QueueHandle_t i2s_queues[I2S_NUM_MAX];
|
||||||
static TaskHandle_t i2s_tasks[I2S_NUM_MAX];
|
static TaskHandle_t i2s_tasks[I2S_NUM_MAX];
|
||||||
|
|
||||||
|
void port_i2s_allocate_i2s0(void) {
|
||||||
|
if (!i2s_instance[0]) {
|
||||||
|
i2s_instance[0] = (void *)~(intptr_t)0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
mp_raise_RuntimeError(translate("Peripheral in use"));
|
||||||
|
}
|
||||||
|
|
||||||
static int8_t port_i2s_allocate(void) {
|
static int8_t port_i2s_allocate(void) {
|
||||||
#if defined(I2S_NUM_1)
|
#if defined(I2S_NUM_1)
|
||||||
if (!i2s_instance[1]) {
|
if (!i2s_instance[1]) {
|
||||||
|
@ -59,3 +59,6 @@ bool port_i2s_playing(i2s_t *self);
|
|||||||
bool port_i2s_paused(i2s_t *self);
|
bool port_i2s_paused(i2s_t *self);
|
||||||
void port_i2s_pause(i2s_t *self);
|
void port_i2s_pause(i2s_t *self);
|
||||||
void port_i2s_resume(i2s_t *self);
|
void port_i2s_resume(i2s_t *self);
|
||||||
|
|
||||||
|
// some uses (imagecapture) can only operate on i2s0 and need their own init code
|
||||||
|
void port_i2s_allocate_i2s0(void);
|
||||||
|
@ -24,26 +24,108 @@
|
|||||||
* THE SOFTWARE.
|
* THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
#include "py/obj.h"
|
#include "py/obj.h"
|
||||||
#include "py/runtime.h"
|
#include "py/runtime.h"
|
||||||
|
|
||||||
|
#include "common-hal/audiobusio/__init__.h"
|
||||||
#include "common-hal/imagecapture/ParallelImageCapture.h"
|
#include "common-hal/imagecapture/ParallelImageCapture.h"
|
||||||
|
#include "cam.h"
|
||||||
|
|
||||||
void common_hal_imagecapture_parallelimagecapture_construct(imagecapture_parallelimagecapture_obj_t *self,
|
void common_hal_imagecapture_parallelimagecapture_construct(imagecapture_parallelimagecapture_obj_t *self,
|
||||||
const mcu_pin_obj_t *data0,
|
const uint8_t data_pins[],
|
||||||
|
uint8_t data_count,
|
||||||
const mcu_pin_obj_t *data_clock,
|
const mcu_pin_obj_t *data_clock,
|
||||||
const mcu_pin_obj_t *vertical_sync,
|
const mcu_pin_obj_t *vertical_sync,
|
||||||
const mcu_pin_obj_t *horizontal_reference,
|
const mcu_pin_obj_t *horizontal_reference) {
|
||||||
int data_count) {
|
|
||||||
mp_raise_NotImplementedError(NULL);
|
// only 8 bits is supported at present
|
||||||
|
if (data_count < 8 || data_count > 16) {
|
||||||
|
mp_raise_ValueError_varg(translate("%q must be between %d and %d"), MP_QSTR_data_count, 8, 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
// This will throw if unsuccessful. Everything following is guaranteed to succeed.
|
||||||
|
port_i2s_allocate_i2s0();
|
||||||
|
|
||||||
|
claim_pin(data_clock);
|
||||||
|
claim_pin(vertical_sync);
|
||||||
|
claim_pin(horizontal_reference);
|
||||||
|
|
||||||
|
self->data_count = data_count;
|
||||||
|
self->data_clock = data_clock->number;
|
||||||
|
self->vertical_sync = vertical_sync->number;
|
||||||
|
self->horizontal_reference = horizontal_reference->number;
|
||||||
|
|
||||||
|
self->config = (cam_config_t) {
|
||||||
|
.bit_width = data_count,
|
||||||
|
.pin = {
|
||||||
|
.pclk = data_clock->number,
|
||||||
|
.vsync = vertical_sync->number,
|
||||||
|
.hsync = horizontal_reference->number,
|
||||||
|
},
|
||||||
|
.vsync_invert = true,
|
||||||
|
.hsync_invert = false,
|
||||||
|
.size = 0,
|
||||||
|
.max_buffer_size = 8 * 1024,
|
||||||
|
.task_stack = 1024,
|
||||||
|
.task_pri = configMAX_PRIORITIES
|
||||||
|
};
|
||||||
|
|
||||||
|
for (int i = 0; i < data_count; i++) {
|
||||||
|
claim_pin_number(data_pins[i]);
|
||||||
|
self->config.pin_data[i] = data_pins[i];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void common_hal_imagecapture_parallelimagecapture_deinit(imagecapture_parallelimagecapture_obj_t *self) {
|
void common_hal_imagecapture_parallelimagecapture_deinit(imagecapture_parallelimagecapture_obj_t *self) {
|
||||||
|
reset_pin_number(self->data_clock);
|
||||||
|
self->data_clock = NO_PIN;
|
||||||
|
|
||||||
|
reset_pin_number(self->vertical_sync);
|
||||||
|
self->vertical_sync = NO_PIN;
|
||||||
|
|
||||||
|
reset_pin_number(self->horizontal_reference);
|
||||||
|
self->horizontal_reference = NO_PIN;
|
||||||
|
|
||||||
|
for (int i = 0; i < self->data_count; i++) {
|
||||||
|
if (self->config.pin_data[i] != NO_PIN) {
|
||||||
|
reset_pin_number(self->config.pin_data[i]);
|
||||||
|
self->config.pin_data[i] = NO_PIN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
port_i2s_reset_instance(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool common_hal_imagecapture_parallelimagecapture_deinited(imagecapture_parallelimagecapture_obj_t *self) {
|
bool common_hal_imagecapture_parallelimagecapture_deinited(imagecapture_parallelimagecapture_obj_t *self) {
|
||||||
return true;
|
return self->data_clock == NO_PIN;
|
||||||
}
|
}
|
||||||
|
|
||||||
void common_hal_imagecapture_parallelimagecapture_capture(imagecapture_parallelimagecapture_obj_t *self, void *buffer, size_t bufsize) {
|
void common_hal_imagecapture_parallelimagecapture_capture(imagecapture_parallelimagecapture_obj_t *self, void *buffer, size_t bufsize) {
|
||||||
|
size_t size = bufsize / 2; // count is in pixels
|
||||||
|
if (size != self->config.size || buffer != self->config.frame1_buffer) {
|
||||||
|
cam_deinit();
|
||||||
|
self->config.size = bufsize / 2; // count is in pixels(?)
|
||||||
|
self->config.frame1_buffer = buffer;
|
||||||
|
|
||||||
|
cam_init(&self->config);
|
||||||
|
} else {
|
||||||
|
cam_give(buffer);
|
||||||
|
}
|
||||||
|
cam_start();
|
||||||
|
|
||||||
|
while (!cam_ready()) {
|
||||||
|
RUN_BACKGROUND_TASKS;
|
||||||
|
if (mp_hal_is_interrupted()) {
|
||||||
|
self->config.size = 0; // force re-init next time
|
||||||
|
cam_stop();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *unused;
|
||||||
|
cam_take(&unused); // this just "returns" buffer
|
||||||
|
|
||||||
|
cam_stop();
|
||||||
}
|
}
|
||||||
|
@ -27,7 +27,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "shared-bindings/imagecapture/ParallelImageCapture.h"
|
#include "shared-bindings/imagecapture/ParallelImageCapture.h"
|
||||||
|
#include "cam.h"
|
||||||
|
|
||||||
struct imagecapture_parallelimagecapture_obj {
|
struct imagecapture_parallelimagecapture_obj {
|
||||||
mp_obj_base_t base;
|
mp_obj_base_t base;
|
||||||
|
cam_config_t config;
|
||||||
|
gpio_num_t data_clock;
|
||||||
|
gpio_num_t vertical_sync;
|
||||||
|
gpio_num_t horizontal_reference;
|
||||||
|
uint8_t data_count;
|
||||||
};
|
};
|
||||||
|
@ -66,6 +66,10 @@
|
|||||||
#include "common-hal/audiobusio/__init__.h"
|
#include "common-hal/audiobusio/__init__.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if CIRCUITPY_IMAGECAPTURE
|
||||||
|
#include "cam.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#define HEAP_SIZE (48 * 1024)
|
#define HEAP_SIZE (48 * 1024)
|
||||||
|
|
||||||
uint32_t *heap;
|
uint32_t *heap;
|
||||||
@ -155,6 +159,10 @@ safe_mode_t port_init(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void reset_port(void) {
|
void reset_port(void) {
|
||||||
|
#if CIRCUITPY_IMAGECAPTURE
|
||||||
|
cam_deinit();
|
||||||
|
#endif
|
||||||
|
|
||||||
reset_all_pins();
|
reset_all_pins();
|
||||||
|
|
||||||
// A larger delay so the idle task can run and do any IDF cleanup needed.
|
// A larger delay so the idle task can run and do any IDF cleanup needed.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user