2020-06-30 12:34:31 +02:00
|
|
|
/****************************************************************************
|
2021-09-16 04:03:33 +02:00
|
|
|
* apps/examples/camera/camera_main.c
|
2020-06-30 12:34:31 +02:00
|
|
|
*
|
2021-09-16 04:03:33 +02:00
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
|
|
* this work for additional information regarding copyright ownership. The
|
|
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance with the
|
|
|
|
* License. You may obtain a copy of the License at
|
2020-06-30 12:34:31 +02:00
|
|
|
*
|
2021-09-16 04:03:33 +02:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2020-06-30 12:34:31 +02:00
|
|
|
*
|
2021-09-16 04:03:33 +02:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
2020-06-30 12:34:31 +02:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
#include <nuttx/video/video.h>
|
|
|
|
|
|
|
|
#include "camera_fileutil.h"
|
2021-09-16 04:03:33 +02:00
|
|
|
#include "camera_bkgd.h"
|
|
|
|
|
2020-06-30 12:34:31 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Pre-processor Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#define IMAGE_JPG_SIZE (512*1024) /* 512kB for FullHD Jpeg file. */
|
2021-09-16 04:03:33 +02:00
|
|
|
#define IMAGE_RGB_SIZE (320*240*2) /* QVGA RGB565 */
|
2020-06-30 12:34:31 +02:00
|
|
|
|
|
|
|
#define VIDEO_BUFNUM (3)
|
|
|
|
#define STILL_BUFNUM (1)
|
|
|
|
|
|
|
|
#define MAX_CAPTURE_NUM (100)
|
|
|
|
#define DEFAULT_CAPTURE_NUM (10)
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
#define START_CAPTURE_TIME (5) /* seconds */
|
|
|
|
#define KEEP_VIDEO_TIME (10) /* seconds */
|
2020-06-30 12:34:31 +02:00
|
|
|
|
|
|
|
#define APP_STATE_BEFORE_CAPTURE (0)
|
|
|
|
#define APP_STATE_UNDER_CAPTURE (1)
|
|
|
|
#define APP_STATE_AFTER_CAPTURE (2)
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Types
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
struct v_buffer
|
|
|
|
{
|
2022-12-07 03:36:41 +01:00
|
|
|
FAR uint32_t *start;
|
2020-06-30 12:34:31 +02:00
|
|
|
uint32_t length;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct v_buffer v_buffer_t;
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Function Prototypes
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int camera_prepare(int fd, enum v4l2_buf_type type,
|
|
|
|
uint32_t buf_mode, uint32_t pixformat,
|
|
|
|
uint16_t hsize, uint16_t vsize,
|
2022-12-07 03:36:41 +01:00
|
|
|
FAR struct v_buffer **vbuf,
|
2020-06-30 12:34:31 +02:00
|
|
|
uint8_t buffernum, int buffersize);
|
2022-12-07 03:36:41 +01:00
|
|
|
static void free_buffer(FAR struct v_buffer *buffers, uint8_t bufnum);
|
|
|
|
static int parse_arguments(int argc, FAR char *argv[],
|
|
|
|
FAR int *capture_num,
|
|
|
|
FAR enum v4l2_buf_type *type);
|
|
|
|
static int get_camimage(int fd, FAR struct v4l2_buffer *v4l2_buf,
|
2022-12-06 12:20:55 +01:00
|
|
|
enum v4l2_buf_type buf_type);
|
2022-12-07 03:36:41 +01:00
|
|
|
static int release_camimage(int fd, FAR struct v4l2_buffer *v4l2_buf);
|
2020-06-30 12:34:31 +02:00
|
|
|
static int start_stillcapture(int v_fd, enum v4l2_buf_type capture_type);
|
|
|
|
static int stop_stillcapture(int v_fd, enum v4l2_buf_type capture_type);
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: camera_prepare()
|
|
|
|
*
|
|
|
|
* Description:
|
2022-12-06 12:20:55 +01:00
|
|
|
* Allocate frame buffer for camera and queue the allocated buffer
|
2020-06-30 12:34:31 +02:00
|
|
|
* into video driver.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int camera_prepare(int fd, enum v4l2_buf_type type,
|
|
|
|
uint32_t buf_mode, uint32_t pixformat,
|
|
|
|
uint16_t hsize, uint16_t vsize,
|
2022-12-07 03:36:41 +01:00
|
|
|
FAR struct v_buffer **vbuf,
|
2020-06-30 12:34:31 +02:00
|
|
|
uint8_t buffernum, int buffersize)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
int cnt;
|
|
|
|
struct v4l2_format fmt =
|
|
|
|
{
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
struct v4l2_requestbuffers req =
|
|
|
|
{
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
struct v4l2_buffer buf =
|
|
|
|
{
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
/* VIDIOC_S_FMT set format */
|
|
|
|
|
|
|
|
fmt.type = type;
|
|
|
|
fmt.fmt.pix.width = hsize;
|
|
|
|
fmt.fmt.pix.height = vsize;
|
|
|
|
fmt.fmt.pix.field = V4L2_FIELD_ANY;
|
|
|
|
fmt.fmt.pix.pixelformat = pixformat;
|
|
|
|
|
2022-12-07 03:44:39 +01:00
|
|
|
ret = ioctl(fd, VIDIOC_S_FMT, (uintptr_t)&fmt);
|
2020-06-30 12:34:31 +02:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to VIDIOC_S_FMT: errno = %d\n", errno);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-12-06 12:21:37 +01:00
|
|
|
/* VIDIOC_REQBUFS initiate user pointer I/O */
|
|
|
|
|
|
|
|
req.type = type;
|
|
|
|
req.memory = V4L2_MEMORY_USERPTR;
|
|
|
|
req.count = buffernum;
|
|
|
|
req.mode = buf_mode;
|
|
|
|
|
2022-12-07 03:44:39 +01:00
|
|
|
ret = ioctl(fd, VIDIOC_REQBUFS, (uintptr_t)&req);
|
2022-12-06 12:21:37 +01:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to VIDIOC_REQBUFS: errno = %d\n", errno);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-06-30 12:34:31 +02:00
|
|
|
/* Prepare video memory to store images */
|
|
|
|
|
|
|
|
*vbuf = malloc(sizeof(v_buffer_t) * buffernum);
|
|
|
|
if (!(*vbuf))
|
|
|
|
{
|
|
|
|
printf("Out of memory for array of v_buffer_t[%d]\n", buffernum);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (cnt = 0; cnt < buffernum; cnt++)
|
|
|
|
{
|
|
|
|
(*vbuf)[cnt].length = buffersize;
|
|
|
|
|
|
|
|
/* Note:
|
|
|
|
* VIDIOC_QBUF set buffer pointer.
|
|
|
|
* Buffer pointer must be 32bytes aligned.
|
|
|
|
*/
|
|
|
|
|
2022-12-06 12:20:55 +01:00
|
|
|
(*vbuf)[cnt].start = memalign(32, buffersize);
|
2020-06-30 12:34:31 +02:00
|
|
|
if (!(*vbuf)[cnt].start)
|
|
|
|
{
|
|
|
|
printf("Out of memory for image buffer of %d/%d\n",
|
|
|
|
cnt, buffernum);
|
|
|
|
|
|
|
|
/* Release allocated memory. */
|
|
|
|
|
2022-12-06 12:20:55 +01:00
|
|
|
while (cnt--)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
|
|
|
free((*vbuf)[cnt].start);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(*vbuf);
|
|
|
|
*vbuf = NULL;
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VIDIOC_QBUF enqueue buffer */
|
|
|
|
|
|
|
|
for (cnt = 0; cnt < buffernum; cnt++)
|
|
|
|
{
|
|
|
|
memset(&buf, 0, sizeof(v4l2_buffer_t));
|
|
|
|
buf.type = type;
|
|
|
|
buf.memory = V4L2_MEMORY_USERPTR;
|
|
|
|
buf.index = cnt;
|
2022-12-07 03:44:39 +01:00
|
|
|
buf.m.userptr = (uintptr_t)(*vbuf)[cnt].start;
|
2020-06-30 12:34:31 +02:00
|
|
|
buf.length = (*vbuf)[cnt].length;
|
|
|
|
|
2022-12-07 03:44:39 +01:00
|
|
|
ret = ioctl(fd, VIDIOC_QBUF, (uintptr_t)&buf);
|
2020-06-30 12:34:31 +02:00
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
printf("Fail QBUF %d\n", errno);
|
|
|
|
free_buffer(*vbuf, buffernum);
|
|
|
|
*vbuf = NULL;
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VIDIOC_STREAMON start stream */
|
|
|
|
|
2022-12-07 03:44:39 +01:00
|
|
|
ret = ioctl(fd, VIDIOC_STREAMON, (uintptr_t)&type);
|
2020-06-30 12:34:31 +02:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to VIDIOC_STREAMON: errno = %d\n", errno);
|
|
|
|
free_buffer(*vbuf, buffernum);
|
|
|
|
*vbuf = NULL;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: free_buffer()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* All free allocated memory of v_buffer.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-12-07 03:36:41 +01:00
|
|
|
static void free_buffer(FAR struct v_buffer *buffers, uint8_t bufnum)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
|
|
|
uint8_t cnt;
|
|
|
|
if (buffers)
|
|
|
|
{
|
|
|
|
for (cnt = 0; cnt < bufnum; cnt++)
|
|
|
|
{
|
|
|
|
if (buffers[cnt].start)
|
|
|
|
{
|
|
|
|
free(buffers[cnt].start);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
free(buffers);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: parse_argument()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Parse and decode commandline arguments.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-12-07 03:36:41 +01:00
|
|
|
static int parse_arguments(int argc, FAR char *argv[],
|
|
|
|
FAR int *capture_num,
|
|
|
|
FAR enum v4l2_buf_type *type)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
|
|
|
if (argc == 1)
|
|
|
|
{
|
|
|
|
*capture_num = DEFAULT_CAPTURE_NUM;
|
|
|
|
*type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
}
|
|
|
|
else if (argc == 2)
|
|
|
|
{
|
|
|
|
if (strncmp(argv[1], "-jpg", 5) == 0)
|
|
|
|
{
|
|
|
|
*capture_num = DEFAULT_CAPTURE_NUM;
|
|
|
|
*type = V4L2_BUF_TYPE_STILL_CAPTURE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*capture_num = atoi(argv[1]);
|
2022-12-06 12:20:55 +01:00
|
|
|
if (*capture_num < 0 || *capture_num > MAX_CAPTURE_NUM)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
|
|
|
printf("Invalid capture num(%d). must be >=0 and <=%d\n",
|
|
|
|
*capture_num, MAX_CAPTURE_NUM);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
*type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (argc == 3)
|
|
|
|
{
|
|
|
|
if (strncmp(argv[1], "-jpg", 5) == 0)
|
|
|
|
{
|
|
|
|
*capture_num = atoi(argv[2]);
|
2022-12-06 12:20:55 +01:00
|
|
|
if (*capture_num < 0 || *capture_num > MAX_CAPTURE_NUM)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
|
|
|
printf("Invalid capture num(%d). must be >=0 and <=%d\n",
|
|
|
|
*capture_num, MAX_CAPTURE_NUM);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
*type = V4L2_BUF_TYPE_STILL_CAPTURE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Invalid argument 1 : %s\n", argv[1]);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Too many arguments\n");
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: get_camimage()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* DQBUF camera frame buffer from video driver with taken picture data.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-12-07 03:36:41 +01:00
|
|
|
static int get_camimage(int fd, FAR struct v4l2_buffer *v4l2_buf,
|
2022-12-06 12:20:55 +01:00
|
|
|
enum v4l2_buf_type buf_type)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* VIDIOC_DQBUF acquires captured data. */
|
|
|
|
|
|
|
|
memset(v4l2_buf, 0, sizeof(v4l2_buffer_t));
|
|
|
|
v4l2_buf->type = buf_type;
|
|
|
|
v4l2_buf->memory = V4L2_MEMORY_USERPTR;
|
|
|
|
|
2022-12-07 03:44:39 +01:00
|
|
|
ret = ioctl(fd, VIDIOC_DQBUF, (uintptr_t)v4l2_buf);
|
2020-06-30 12:34:31 +02:00
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
printf("Fail DQBUF %d\n", errno);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: release_camimage()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Re-QBUF to set used frame buffer into video driver.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-12-07 03:36:41 +01:00
|
|
|
static int release_camimage(int fd, FAR struct v4l2_buffer *v4l2_buf)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* VIDIOC_QBUF sets buffer pointer into video driver again. */
|
|
|
|
|
2022-12-07 03:44:39 +01:00
|
|
|
ret = ioctl(fd, VIDIOC_QBUF, (uintptr_t)v4l2_buf);
|
2020-06-30 12:34:31 +02:00
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
printf("Fail QBUF %d\n", errno);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: start_stillcapture()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Start STILL capture stream by TAKEPICT_START if buf_type is
|
|
|
|
* STILL_CAPTURE.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int start_stillcapture(int v_fd, enum v4l2_buf_type capture_type)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (capture_type == V4L2_BUF_TYPE_STILL_CAPTURE)
|
|
|
|
{
|
|
|
|
ret = ioctl(v_fd, VIDIOC_TAKEPICT_START, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to start taking picture\n");
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: stop_stillcapture()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Stop STILL capture stream by TAKEPICT_STOP if buf_type is STILL_CAPTURE.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static int stop_stillcapture(int v_fd, enum v4l2_buf_type capture_type)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (capture_type == V4L2_BUF_TYPE_STILL_CAPTURE)
|
|
|
|
{
|
|
|
|
ret = ioctl(v_fd, VIDIOC_TAKEPICT_STOP, false);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("Failed to stop taking picture\n");
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2022-10-24 15:52:02 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: get_imgsensor_name()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Get image sensor driver name by querying device capabilities.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static FAR const char *get_imgsensor_name(int fd)
|
|
|
|
{
|
|
|
|
static struct v4l2_capability cap;
|
|
|
|
|
2022-12-07 03:44:39 +01:00
|
|
|
ioctl(fd, VIDIOC_QUERYCAP, (uintptr_t)&cap);
|
2022-10-24 15:52:02 +02:00
|
|
|
|
|
|
|
return (FAR const char *)cap.driver;
|
|
|
|
}
|
|
|
|
|
2020-06-30 12:34:31 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: main()
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* main routine of this example.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int main(int argc, FAR char *argv[])
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
int v_fd;
|
|
|
|
int capture_num = DEFAULT_CAPTURE_NUM;
|
|
|
|
enum v4l2_buf_type capture_type = V4L2_BUF_TYPE_STILL_CAPTURE;
|
|
|
|
struct v4l2_buffer v4l2_buf;
|
2022-10-24 15:52:02 +02:00
|
|
|
FAR const char *save_dir;
|
|
|
|
FAR const char *sensor;
|
|
|
|
uint16_t w;
|
|
|
|
uint16_t h;
|
2021-09-16 04:03:33 +02:00
|
|
|
int is_eternal;
|
|
|
|
int app_state;
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
struct timeval start;
|
|
|
|
struct timeval now;
|
|
|
|
struct timeval delta;
|
|
|
|
struct timeval wait;
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2022-12-07 03:36:41 +01:00
|
|
|
FAR struct v_buffer *buffers_video = NULL;
|
|
|
|
FAR struct v_buffer *buffers_still = NULL;
|
2020-06-30 12:34:31 +02:00
|
|
|
|
|
|
|
/* ===== Parse and Check arguments ===== */
|
|
|
|
|
|
|
|
ret = parse_arguments(argc, argv, &capture_num, &capture_type);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
printf("usage: %s ([-jpg]) ([capture num])\n", argv[0]);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ===== Initialization Code ===== */
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
/* Initialize NX graphics subsystem to use LCD */
|
|
|
|
|
|
|
|
#ifdef CONFIG_EXAMPLES_CAMERA_OUTPUT_LCD
|
|
|
|
ret = nximage_initialize();
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("camera_main: Failed to get NX handle: %d\n", errno);
|
|
|
|
return ERROR;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-06-30 12:34:31 +02:00
|
|
|
/* Select storage to save image files */
|
|
|
|
|
|
|
|
save_dir = futil_initialize();
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
/* Initialize video driver to create a device file */
|
|
|
|
|
|
|
|
ret = video_initialize("/dev/video");
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
printf("ERROR: Failed to initialize video: errno = %d\n", errno);
|
|
|
|
goto exit_without_cleaning_videodriver;
|
|
|
|
}
|
|
|
|
|
2020-06-30 12:34:31 +02:00
|
|
|
/* Open the device file. */
|
|
|
|
|
|
|
|
v_fd = open("/dev/video", 0);
|
|
|
|
if (v_fd < 0)
|
|
|
|
{
|
|
|
|
printf("ERROR: Failed to open video.errno = %d\n", errno);
|
|
|
|
ret = ERROR;
|
|
|
|
goto exit_without_cleaning_buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Prepare for STILL_CAPTURE stream.
|
|
|
|
*
|
|
|
|
* The video buffer mode is V4L2_BUF_MODE_FIFO mode.
|
|
|
|
* In this FIFO mode, if all VIDIOC_QBUFed frame buffers are captured image
|
|
|
|
* and no additional frame buffers are VIDIOC_QBUFed, the capture stops and
|
|
|
|
* waits for new VIDIOC_QBUFed frame buffer.
|
|
|
|
* And when new VIDIOC_QBUF is executed, the capturing is resumed.
|
|
|
|
*
|
2022-10-24 15:52:02 +02:00
|
|
|
* Allocate frame buffers for JPEG size (512KB).
|
|
|
|
* Set FULLHD size in ISX012 case, QUADVGA size in ISX019 case or other
|
|
|
|
* image sensors,
|
2020-06-30 12:34:31 +02:00
|
|
|
* Number of frame buffers is defined as STILL_BUFNUM(1).
|
|
|
|
* And all allocated memorys are VIDIOC_QBUFed.
|
|
|
|
*/
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
if (capture_num != 0)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
2022-10-24 15:52:02 +02:00
|
|
|
/* Determine image size from connected image sensor name,
|
|
|
|
* because video driver does not support VIDIOC_ENUM_FRAMESIZES
|
|
|
|
* for now.
|
|
|
|
*/
|
|
|
|
|
|
|
|
sensor = get_imgsensor_name(v_fd);
|
|
|
|
if (strncmp(sensor, "ISX012", strlen("ISX012")) == 0)
|
|
|
|
{
|
|
|
|
w = VIDEO_HSIZE_FULLHD;
|
|
|
|
h = VIDEO_VSIZE_FULLHD;
|
|
|
|
}
|
|
|
|
else if (strncmp(sensor, "ISX019", strlen("ISX019")) == 0)
|
|
|
|
{
|
|
|
|
w = VIDEO_HSIZE_QUADVGA;
|
|
|
|
h = VIDEO_VSIZE_QUADVGA;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
w = VIDEO_HSIZE_QUADVGA;
|
|
|
|
h = VIDEO_VSIZE_QUADVGA;
|
|
|
|
}
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
ret = camera_prepare(v_fd, V4L2_BUF_TYPE_STILL_CAPTURE,
|
|
|
|
V4L2_BUF_MODE_FIFO, V4L2_PIX_FMT_JPEG,
|
2022-10-24 15:52:02 +02:00
|
|
|
w, h,
|
2021-09-16 04:03:33 +02:00
|
|
|
&buffers_still, STILL_BUFNUM, IMAGE_JPG_SIZE);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
2020-06-30 12:34:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Prepare for VIDEO_CAPTURE stream.
|
|
|
|
*
|
|
|
|
* The video buffer mode is V4L2_BUF_MODE_RING mode.
|
|
|
|
* In this RING mode, if all VIDIOC_QBUFed frame buffers are captured image
|
|
|
|
* and no additional frame buffers are VIDIOC_QBUFed, the capture continues
|
|
|
|
* as the oldest image in the V4L2_BUF_QBUFed frame buffer is reused in
|
|
|
|
* order from the captured frame buffer and a new camera image is
|
|
|
|
* recaptured.
|
|
|
|
*
|
2021-09-16 04:03:33 +02:00
|
|
|
* Allocate freame buffers for QVGA RGB565 size (320x240x2=150KB).
|
2020-06-30 12:34:31 +02:00
|
|
|
* Number of frame buffers is defined as VIDEO_BUFNUM(3).
|
|
|
|
* And all allocated memorys are VIDIOC_QBUFed.
|
|
|
|
*/
|
|
|
|
|
|
|
|
ret = camera_prepare(v_fd, V4L2_BUF_TYPE_VIDEO_CAPTURE,
|
2021-09-16 04:03:33 +02:00
|
|
|
V4L2_BUF_MODE_RING, V4L2_PIX_FMT_RGB565,
|
2020-06-30 12:34:31 +02:00
|
|
|
VIDEO_HSIZE_QVGA, VIDEO_VSIZE_QVGA,
|
2021-09-16 04:03:33 +02:00
|
|
|
&buffers_video, VIDEO_BUFNUM, IMAGE_RGB_SIZE);
|
2020-06-30 12:34:31 +02:00
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
/* This application has 3 states.
|
|
|
|
*
|
|
|
|
* APP_STATE_BEFORE_CAPTURE:
|
2022-10-24 15:52:02 +02:00
|
|
|
* This state waits 5 seconds (defined as START_CAPTURE_TIME)
|
2021-09-16 04:03:33 +02:00
|
|
|
* with displaying preview (VIDEO_CAPTURE stream image) on LCD.
|
|
|
|
* After 5 seconds, state will be changed to APP_STATE_UNDER_CAPTURE.
|
|
|
|
*
|
|
|
|
* APP_STATE_UNDER_CAPTURE:
|
|
|
|
* This state will start taking picture and store the image into files.
|
|
|
|
* Number of taking pictures is set capture_num valiable.
|
|
|
|
* It can be changed by command line argument.
|
|
|
|
* After finishing taking pictures, the state will be changed to
|
|
|
|
* APP_STATE_AFTER_CAPTURE.
|
|
|
|
*
|
|
|
|
* APP_STATE_AFTER_CAPTURE:
|
2022-10-24 15:52:02 +02:00
|
|
|
* This state waits 10 seconds (defined as KEEP_VIDEO_TIME)
|
2021-09-16 04:03:33 +02:00
|
|
|
* with displaying preview (VIDEO_CAPTURE stream image) on LCD.
|
|
|
|
* After 10 seconds, this application will be finished.
|
|
|
|
*
|
|
|
|
* Notice:
|
|
|
|
* If capture_num is set '0', state will stay APP_STATE_BEFORE_CAPTURE.
|
|
|
|
*/
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
app_state = APP_STATE_BEFORE_CAPTURE;
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
/* Show this application behavior. */
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
if (capture_num == 0)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
2021-09-16 04:03:33 +02:00
|
|
|
is_eternal = 1;
|
|
|
|
printf("Start video this mode is eternal."
|
|
|
|
" (Non stop, non save files.)\n");
|
|
|
|
#ifndef CONFIG_EXAMPLES_CAMERA_OUTPUT_LCD
|
|
|
|
printf("This mode should be run with LCD display\n");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
is_eternal = 0;
|
|
|
|
wait.tv_sec = START_CAPTURE_TIME;
|
|
|
|
wait.tv_usec = 0;
|
|
|
|
printf("Take %d pictures as %s file in %s after %d seconds.\n",
|
|
|
|
capture_num,
|
2022-12-06 12:20:55 +01:00
|
|
|
capture_type == V4L2_BUF_TYPE_STILL_CAPTURE ? "JPEG" : "RGB",
|
2021-09-16 04:03:33 +02:00
|
|
|
save_dir, START_CAPTURE_TIME);
|
|
|
|
printf(" After finishing taking pictures,"
|
|
|
|
" this app will be finished after %d seconds.\n",
|
|
|
|
KEEP_VIDEO_TIME);
|
|
|
|
}
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
gettimeofday(&start, NULL);
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
/* ===== Main Loop ===== */
|
2020-06-30 12:34:31 +02:00
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
switch (app_state)
|
2020-06-30 12:34:31 +02:00
|
|
|
{
|
2021-09-16 04:03:33 +02:00
|
|
|
/* BEFORE_CAPTURE and AFTER_CAPTURE is waiting for expiring the
|
|
|
|
* time.
|
2022-10-24 15:52:02 +02:00
|
|
|
* In the meantime, Capturing VIDEO image to show pre-view on LCD.
|
2021-09-16 04:03:33 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
case APP_STATE_BEFORE_CAPTURE:
|
|
|
|
case APP_STATE_AFTER_CAPTURE:
|
|
|
|
ret = get_camimage(v_fd, &v4l2_buf, V4L2_BUF_TYPE_VIDEO_CAPTURE);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_EXAMPLES_CAMERA_OUTPUT_LCD
|
2022-12-07 03:36:41 +01:00
|
|
|
nximage_draw((FAR void *)v4l2_buf.m.userptr,
|
2021-09-16 04:03:33 +02:00
|
|
|
VIDEO_HSIZE_QVGA, VIDEO_VSIZE_QVGA);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
ret = release_camimage(v_fd, &v4l2_buf);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_eternal)
|
|
|
|
{
|
|
|
|
gettimeofday(&now, NULL);
|
|
|
|
timersub(&now, &start, &delta);
|
|
|
|
if (timercmp(&delta, &wait, >))
|
|
|
|
{
|
|
|
|
printf("Expire time is pasted. GoTo next state.\n");
|
|
|
|
if (app_state == APP_STATE_BEFORE_CAPTURE)
|
|
|
|
{
|
|
|
|
app_state = APP_STATE_UNDER_CAPTURE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ret = OK;
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break; /* Finish APP_STATE_BEFORE_CAPTURE or APP_STATE_AFTER_CAPTURE */
|
|
|
|
|
|
|
|
/* UNDER_CAPTURE is taking pictures until number of capture_num
|
|
|
|
* value.
|
|
|
|
* This state stays until finishing all pictures.
|
|
|
|
*/
|
|
|
|
|
|
|
|
case APP_STATE_UNDER_CAPTURE:
|
2022-10-24 15:52:02 +02:00
|
|
|
printf("Start capturing...\n");
|
2021-09-16 04:03:33 +02:00
|
|
|
ret = start_stillcapture(v_fd, capture_type);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (capture_num)
|
|
|
|
{
|
|
|
|
ret = get_camimage(v_fd, &v4l2_buf, capture_type);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
|
|
|
|
futil_writeimage(
|
2022-12-07 03:36:41 +01:00
|
|
|
(FAR uint8_t *)v4l2_buf.m.userptr,
|
2021-09-16 04:03:33 +02:00
|
|
|
(size_t)v4l2_buf.bytesused,
|
2022-12-06 12:20:55 +01:00
|
|
|
capture_type == V4L2_BUF_TYPE_VIDEO_CAPTURE ?
|
2021-09-16 04:03:33 +02:00
|
|
|
"RGB" : "JPG");
|
|
|
|
|
|
|
|
ret = release_camimage(v_fd, &v4l2_buf);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
|
|
|
|
capture_num--;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = stop_stillcapture(v_fd, capture_type);
|
|
|
|
if (ret != OK)
|
|
|
|
{
|
|
|
|
goto exit_this_app;
|
|
|
|
}
|
|
|
|
|
|
|
|
app_state = APP_STATE_AFTER_CAPTURE;
|
|
|
|
wait.tv_sec = KEEP_VIDEO_TIME;
|
|
|
|
wait.tv_usec = 0;
|
|
|
|
gettimeofday(&start, NULL);
|
2022-10-24 15:52:02 +02:00
|
|
|
printf("Finished capturing...\n");
|
2021-09-16 04:03:33 +02:00
|
|
|
break; /* Finish APP_STATE_UNDER_CAPTURE */
|
|
|
|
|
|
|
|
default:
|
2022-10-24 15:52:02 +02:00
|
|
|
printf("Unknown error is occurred.. state=%d\n", app_state);
|
2021-09-16 04:03:33 +02:00
|
|
|
goto exit_this_app;
|
|
|
|
break;
|
2020-06-30 12:34:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
exit_this_app:
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
/* Close video device file makes dequeue all buffers */
|
2020-06-30 12:34:31 +02:00
|
|
|
|
|
|
|
close(v_fd);
|
|
|
|
|
|
|
|
free_buffer(buffers_video, VIDEO_BUFNUM);
|
|
|
|
free_buffer(buffers_still, STILL_BUFNUM);
|
|
|
|
|
|
|
|
exit_without_cleaning_buffer:
|
|
|
|
video_uninitialize();
|
|
|
|
|
2021-09-16 04:03:33 +02:00
|
|
|
exit_without_cleaning_videodriver:
|
|
|
|
#ifdef CONFIG_EXAMPLES_CAMERA_OUTPUT_LCD
|
|
|
|
nximage_finalize();
|
|
|
|
#endif
|
2020-06-30 12:34:31 +02:00
|
|
|
return ret;
|
|
|
|
}
|