当前位置: 首页 > news >正文

rkmedia_vi_get_frame_test.c 代码解析

使用示例:

录像:
rkmedia_vi_get_frame_test -a /etc/iqfiles/ -I 1 -o 1080.nv12
然后用yuvplayer.exe可以播放。

录像10帧:
rkmedia_vi_get_frame_test -a /etc/iqfiles/ -I 1 -o 1080.nv12 -c 10

解析代码:


#include <assert.h>
#include <fcntl.h>
#include <getopt.h>
#include <pthread.h>
#include <signal.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <unistd.h>#include "common/sample_common.h"
#include "rkmedia_api.h"typedef struct {  //用于传递输出参数char *file_path;int frame_cnt;
} OutputArgs;static bool quit = false;
static void sigterm_handler(int sig) {  //用于控制程序退出fprintf(stderr, "signal %d\n", sig);quit = true;
}static void *GetMediaBuffer(void *arg) {  //获取视频帧数据并保存到文件OutputArgs *outArgs = (OutputArgs *)arg;  //使用`OutputArgs`结构体指针`outArgs`来获取输出参数,其中包括保存路径`save_path`和帧计数`save_cnt`。char *save_path = outArgs->file_path;   //保存路径int save_cnt = outArgs->frame_cnt;    //帧计数int frame_id = 0;FILE *save_file = NULL;if (save_path) {save_file = fopen(save_path, "w");  //打开保存路径指定的文件if (!save_file)printf("ERROR: Open %s failed!\n", save_path);}MEDIA_BUFFER mb = NULL;while (!quit) { //进入循环,当全局变量`quit`为假时持续循环。mb = RK_MPI_SYS_GetMediaBuffer(RK_ID_VI, 0, -1);  //获取视频帧的媒体缓冲区`mb`,并检查是否获取成功if (!mb) {printf("RK_MPI_SYS_GetMediaBuffer get null buffer!\n");break;}//获取媒体缓冲区的图像信息`stImageInfo`,并打印相关信息,如帧指针、文件描述符、大小、时间戳等。MB_IMAGE_INFO_S stImageInfo = {0};int ret = RK_MPI_MB_GetImageInfo(mb, &stImageInfo);if (ret)printf("Warn: Get image info failed! ret = %d\n", ret);printf("Get Frame:ptr:%p, fd:%d, size:%zu, mode:%d, channel:%d, ""timestamp:%lld, ImgInfo:<wxh %dx%d, fmt 0x%x>\n",RK_MPI_MB_GetPtr(mb), RK_MPI_MB_GetFD(mb), RK_MPI_MB_GetSize(mb),RK_MPI_MB_GetModeID(mb), RK_MPI_MB_GetChannelID(mb),RK_MPI_MB_GetTimestamp(mb), stImageInfo.u32Width,stImageInfo.u32Height, stImageInfo.enImgType);//如果已经打开了保存文件,则将媒体缓冲区中的帧数据写入到文件中,并打印保存信息。if (save_file) { fwrite(RK_MPI_MB_GetPtr(mb), 1, RK_MPI_MB_GetSize(mb), save_file);printf("#Save frame-%d to %s\n", frame_id++, save_path);}RK_MPI_MB_ReleaseBuffer(mb);  //释放媒体缓冲区。if (save_cnt > 0) //如果设置了帧计数`save_cnt`且大于0,则递减计数。save_cnt--;// exit when completeif (!save_cnt) {  //当完成保存指定数量的帧后,设置全局变量`quit`为真,退出循环。quit = true;break;}}if (save_file)fclose(save_file);return NULL;
}static RK_CHAR optstr[] = "?::a::w:h:c:o:d:I:M:";
static const struct option long_options[] = {{"aiq", optional_argument, NULL, 'a'},{"device_name", required_argument, NULL, 'd'},{"width", required_argument, NULL, 'w'},{"height", required_argument, NULL, 'h'},{"frame_cnt", required_argument, NULL, 'c'},{"output", required_argument, NULL, 'o'},{"camid", required_argument, NULL, 'I'},{"multictx", required_argument, NULL, 'M'},{"help", optional_argument, NULL, '?'},{NULL, 0, NULL, 0},
};static void print_usage(const RK_CHAR *name) {printf("usage example:\n");
#ifdef RKAIQprintf("\t%s [-a [iqfiles_dir]] ""[-I 0] ""[-M 0] ""[-w 1920] ""[-h 1080]""[-c 10] ""[-d rkispp_scale0] ""[-o out.nv12] \n",name);printf("\t-a | --aiq: enable aiq with dirpath provided, eg:-a ""/oem/etc/iqfiles/, ""set dirpath emtpty to using path by default, without this option aiq ""should run in other application\n");printf("\t-M | --multictx: switch of multictx in isp, set 0 to disable, set ""1 to enable. Default: 0\n");
#elseprintf("\t%s [-w 1920] ""[-h 1080] ""[-c 10] ""[-d rkispp_scale0] ""[-I 0] ""[-o out.nv12] \n",name);
#endifprintf("\t-w | --width: VI width, Default:1920\n");printf("\t-h | --heght: VI height, Default:1080\n");printf("\t-d | --device_name: set device node(v4l2), Default:rkispp_scale0\n");printf("\t-I | --camid: camera ctx id, Default 0\n");printf("\t-c | --frame_cnt: record frame, Default:-1(unlimit)\n");printf("\t-o | --output: output path, Default:NULL\n");printf("Notice: fmt always NV12\n");
}int main(int argc, char *argv[]) {RK_U32 u32Width = 1920;RK_U32 u32Height = 1080;int frameCnt = -1;RK_CHAR *pDeviceName = "rkispp_scale0";RK_CHAR *pOutPath = NULL;RK_CHAR *pIqfilesPath = NULL;RK_S32 s32CamId = 0;
#ifdef RKAIQRK_BOOL bMultictx = RK_FALSE;
#endifint c;int ret = 0;//使用了 getopt_long 函数对命令行参数进行解析。然后根据解析得到的参数配置相应的采集参数,比如视频宽高、帧率等。while ((c = getopt_long(argc, argv, optstr, long_options, NULL)) != -1) {const char *tmp_optarg = optarg;switch (c) {case 'a':if (!optarg && NULL != argv[optind] && '-' != argv[optind][0]) {tmp_optarg = argv[optind++];}if (tmp_optarg) {pIqfilesPath = (char *)tmp_optarg;} else {pIqfilesPath = "/oem/etc/iqfiles/";}break;case 'w':u32Width = atoi(optarg);break;case 'h':u32Height = atoi(optarg);break;case 'c':frameCnt = atoi(optarg);break;case 'o':pOutPath = optarg;break;case 'd':pDeviceName = optarg;break;case 'I':s32CamId = atoi(optarg);break;
#ifdef RKAIQcase 'M':if (atoi(optarg)) {bMultictx = RK_TRUE;}break;
#endifcase '?':default:print_usage(argv[0]);return 0;}}printf("#####Device: %s\n", pDeviceName);printf("#####Resolution: %dx%d\n", u32Width, u32Height);printf("#####Frame Count to save: %d\n", frameCnt);printf("#####Output Path: %s\n", pOutPath);printf("#CameraIdx: %d\n\n", s32CamId);signal(SIGINT, sigterm_handler);if (pIqfilesPath) {
#ifdef RKAIQprintf("#####Aiq xml dirpath: %s\n\n", pIqfilesPath);printf("#bMultictx: %d\n\n", bMultictx);// rk_aiq_working_mode_t hdr_mode = RK_AIQ_WORKING_MODE_NORMAL;rk_aiq_working_mode_t hdr_mode = RK_AIQ_WORKING_MODE_ISP_HDR2; //替换为HDR模式int fps = 30;SAMPLE_COMM_ISP_Init(s32CamId, hdr_mode, bMultictx, pIqfilesPath);SAMPLE_COMM_ISP_Run(s32CamId);SAMPLE_COMM_ISP_SetFrameRate(s32CamId, fps);
#endif}//在相机初始化部分,通过 RK_MPI_VI_SetChnAttr 和 RK_MPI_VI_EnableChn 函数配置并启用了视频输入通道。RK_MPI_SYS_Init();VI_CHN_ATTR_S vi_chn_attr;vi_chn_attr.pcVideoNode = pDeviceName;vi_chn_attr.u32BufCnt = 3;vi_chn_attr.u32Width = u32Width;vi_chn_attr.u32Height = u32Height;vi_chn_attr.enPixFmt = IMAGE_TYPE_NV12;vi_chn_attr.enWorkMode = VI_WORK_MODE_NORMAL;vi_chn_attr.enBufType = VI_CHN_BUF_TYPE_MMAP;ret = RK_MPI_VI_SetChnAttr(s32CamId, 0, &vi_chn_attr);ret |= RK_MPI_VI_EnableChn(s32CamId, 0);if (ret) {printf("Create VI[0] failed! ret=%d\n", ret);return -1;}printf("%s initial finish\n", __func__);//在视频采集和处理部分,创建了一个线程用于获取媒体缓冲区,并通过 RK_MPI_VI_StartStream 开始采集。然后通过一个循环来等待退出信号,最后在退出时释放资源。pthread_t read_thread;OutputArgs outArgs = {pOutPath, frameCnt};pthread_create(&read_thread, NULL, GetMediaBuffer, &outArgs); //ret = RK_MPI_VI_StartStream(s32CamId, 0);if (ret) {printf("Start VI[0] failed! ret=%d\n", ret);return -1;}while (!quit) {usleep(500000);}RK_MPI_VI_DisableChn(s32CamId, 0);#ifdef RKAIQif (pIqfilesPath)SAMPLE_COMM_ISP_Stop(s32CamId);
#endifprintf("%s exit!\n", __func__);return 0;
}
http://www.lryc.cn/news/250217.html

相关文章:

  • 探究Kafka原理-3.生产者消费者API原理解析
  • Linux系统iptables扩展
  • Openwrt 系统安装 插件名称与中文释义
  • [原创]Delphi的SizeOf(), Length(), 动态数组, 静态数组的关系.
  • C++(20):bind_front
  • 【spring】bean的后处理器
  • Centos7安装docker、java、python环境
  • 简单小结类与对象
  • ABAP 如何获取内表行的索引值(index) ?
  • ESP32-Web-Server编程- 使用表格(Table)实时显示设备信息
  • vue3 Hooks函数使用及常用utils封装
  • matlab 无迹卡尔曼滤波
  • 大脑--学习方法
  • 4.C转python
  • YOLOv5项目实战(5)— 算法模型优化和服务器部署
  • JavaScript类型判断:解密变量真实身份的神奇技巧
  • MT6893_天玑 1200芯片规格参数介绍_datasheet规格书
  • 【Android踩过的坑】13.Android Studio 运行成功,但APP没有安装上的问题
  • redis安装配置
  • 企业数字化转型应对传统网络挑战的关键策略
  • Java 多线程基础 线程4种创建方式
  • C++概念相关练习题
  • 区间合并笔记
  • 青少年CTF之PHP特性练习(1-5)
  • 《opencv实用探索·七》一文看懂图像卷积运算
  • RPA机器人如何确保敏感数据的安全性
  • 微信号被封了怎么办
  • 关于 ls -s 输出文件大小的单位问题的讨论
  • JSON.stringify方法详解 后端接受JSON数据格式
  • vue请求如何按顺序执行