android4.0 USB Camera实例(六)ffmpeg mpeg编码

前面本来说是做h264编码的 研究了两天发现ffmpeg里的h264编码似乎是要信赖第三方库x264 还是怎么简单怎么来吧所以就整了个mpeg编码 ffmpeg移植前面我有一篇ffmpeg解码里已经给了 详细链接在这http://blog.csdn.net/hclydao/article/details/18546757

怎么使用那里面也已经说了 这里主要是通过ffmpeg将yuv422格式转换成rgb 然后就是yuv422转成mpeg格式 接前面几篇 获取到yuv422数据后 为了能显示出来 所以先转换成rgb565的数据 接口函数例如以下

/*
* yuv to rgb
*/
JNIEXPORT jint JNICALL Java_com_hclydao_webcam_Ffmpeg_yuvtorgb(JNIEnv * env, jclass obj,const jbyteArray yuvdata, jbyteArray rgbdata,const jint dwidth,const jint dheight)
{
	jbyte *ydata = (jbyte*)(*env)->GetByteArrayElements(env, yuvdata, 0);
	jbyte *rdata = (jbyte*)(*env)->GetByteArrayElements(env, rgbdata, 0);
	AVFrame * rpicture=NULL;
	AVFrame * ypicture=NULL;
	struct SwsContext *swsctx = NULL;
	rpicture=avcodec_alloc_frame();
	ypicture=avcodec_alloc_frame();
	avpicture_fill((AVPicture *) rpicture, (uint8_t *)rdata, PIX_FMT_RGB565,dwidth,dheight);
	avpicture_fill((AVPicture *) ypicture, (uint8_t *)ydata, AV_PIX_FMT_YUYV422,mwidth,mheight);
	swsctx = sws_getContext(mwidth,mheight, AV_PIX_FMT_YUYV422,	dwidth, dheight,PIX_FMT_RGB565, SWS_BICUBIC, NULL, NULL, NULL);
	sws_scale(swsctx,(const uint8_t* const*)ypicture->data,ypicture->linesize,0,mheight,rpicture->data,rpicture->linesize);
	sws_freeContext(swsctx);
	av_free(rpicture);
	av_free(ypicture);
	(*env)->ReleaseByteArrayElements(env, yuvdata, ydata, 0);
	(*env)->ReleaseByteArrayElements(env, rgbdata, rdata, 0);
	return 0;
}

然后就是mpeg编码了 网上说ffmpeg仅仅能将yuv420p的编码 所以要先将yuv422转成yuv420p后在进行编码 相关接口函数例如以下

AVCodecContext *pCodecCtx= NULL;
AVPacket avpkt;
FILE * video_file;
unsigned char *outbuf=NULL;
unsigned char *yuv420buf=NULL;
static int outsize=0;
/*
* encording init
*/
JNIEXPORT jint JNICALL Java_com_hclydao_webcam_Ffmpeg_videoinit(JNIEnv * env, jclass obj,jbyteArray filename)
{
	LOGI("%s\n",__func__);
	AVCodec * pCodec=NULL;
	avcodec_register_all();
	pCodec=avcodec_find_encoder(AV_CODEC_ID_MPEG1VIDEO);
	if(pCodec == NULL) {
		LOGE("++++++++++++codec not found\n");
		return -1;
	} 
    pCodecCtx=avcodec_alloc_context3(pCodec);
    if (pCodecCtx == NULL) {
        LOGE("++++++Could not allocate video codec context\n");
        return -1;
    }
    /* put sample parameters */
    pCodecCtx->bit_rate = 400000;
    /* resolution must be a multiple of two */
    pCodecCtx->width = mwidth;
    pCodecCtx->height = mheight;
    /* frames per second */
    pCodecCtx->time_base= (AVRational){1,25};
    pCodecCtx->gop_size = 10; /* emit one intra frame every ten frames */
    pCodecCtx->max_b_frames=1;
    pCodecCtx->pix_fmt = AV_PIX_FMT_YUV420P;//AV_PIX_FMT_YUYV422;
    /* open it */
    if (avcodec_open2(pCodecCtx, pCodec, NULL) < 0) {
        LOGE("+++++++Could not open codec\n");
        return -1;
    }
	outsize = mwidth * mheight*2;
	outbuf = malloc(outsize*sizeof(char));
	yuv420buf = malloc(outsize*sizeof(char));
	jbyte *filedir = (jbyte*)(*env)->GetByteArrayElements(env, filename, 0);
  	if ((video_file = fopen(filedir, "wb")) == NULL) {
    	LOGE("++++++++++++open %s failed\n",filedir);
    	return -1;
  	}
	(*env)->ReleaseByteArrayElements(env, filename, filedir, 0);
	return 1;
}

JNIEXPORT jint JNICALL Java_com_hclydao_webcam_Ffmpeg_videostart(JNIEnv * env, jclass obj,jbyteArray yuvdata)
{
	int frameFinished=0,size=0;
	jbyte *ydata = (jbyte*)(*env)->GetByteArrayElements(env, yuvdata, 0);
	AVFrame * yuv420pframe=NULL;
	AVFrame * yuv422frame=NULL;
	struct SwsContext *swsctx = NULL;
	yuv420pframe=avcodec_alloc_frame();
	yuv422frame=avcodec_alloc_frame();
	avpicture_fill((AVPicture *) yuv420pframe, (uint8_t *)yuv420buf, AV_PIX_FMT_YUV420P,mwidth,mheight);
	avpicture_fill((AVPicture *) yuv422frame, (uint8_t *)ydata, AV_PIX_FMT_YUYV422,mwidth,mheight);
	swsctx = sws_getContext(mwidth,mheight, AV_PIX_FMT_YUYV422,	mwidth, mheight,AV_PIX_FMT_YUV420P, SWS_BICUBIC, NULL, NULL, NULL);
	sws_scale(swsctx,(const uint8_t* const*)yuv422frame->data,yuv422frame->linesize,0,mheight,yuv420pframe->data,yuv420pframe->linesize);
	av_init_packet(&avpkt);
	size = avcodec_encode_video2(pCodecCtx, &avpkt, yuv420pframe, &frameFinished);
    if (size < 0) {
    	LOGE("+++++Error encoding frame\n");
        return -1;
    }
	if(frameFinished)
		fwrite(avpkt.data,avpkt.size,1,video_file);
	av_free_packet(&avpkt);
	sws_freeContext(swsctx);
	av_free(yuv420pframe);
	av_free(yuv422frame);
	(*env)->ReleaseByteArrayElements(env, yuvdata, ydata, 0);
}

JNIEXPORT jint JNICALL Java_com_hclydao_webcam_Ffmpeg_videoclose(JNIEnv * env, jclass obj)
{
	fclose(video_file);
	avcodec_close(pCodecCtx);
	av_free(pCodecCtx);
	free(outbuf);
}
最后录下来的视频是能够用播放的 总感觉代码好像哪里写的不正确 bug总是有的 过程和原理搞清楚了其他就easy了

以下是我录的 至此摄像头这块临时就这样了

这測试我又又一次新建了一个project

整个project下载链接 请去我的资源里找吧 我上传了没显示出来 那个有20几M的就是的了

这个bug非常多 没怎么认真去写 假设原理和过程有问题的希望大家指出


转载于:https://www.cnblogs.com/ljbguanli/p/7299759.html

您好!对于在 Android 上使用 Camera2 API 和 FFmpeg 进行视频编码,您可以按照以下步骤操作: 1. 首先,您需要在您的 Android 项目中集成 FFmpeg。您可以通过在项目的 build.gradle 文件中添加 FFmpeg 库的引用来完成此操作。例如: ```groovy implementation 'com.writingminds:FFmpegAndroid:0.3.2' ``` 2. 接下来,您需要设置并打开 Camera2 API。您可以参考 Android 官方文档来了解如何使用 Camera2 API 进行摄像头操作。 3. 在获取到 Camera2 的帧数据后,您可以将其编码为视频文件。为此,您可以使用 FFmpeg 提供的命令行接口。以下是一个示例代码片段,演示如何将 Camera2 的帧数据编码为 H.264 格式的视频文件: ```java // 设置 FFmpeg 命令行参数 String[] ffmpegCmd = new String[] { "-y", // 覆盖输出文件 "-f", "rawvideo", "-vcodec", "rawvideo", "-pix_fmt", "nv21", // 请根据摄像头输出格式进行更改 "-s", previewSize.getWidth() + "x" + previewSize.getHeight(), // 请根据预览尺寸进行更改 "-i", "-", // 使用输入管道获取帧数据 "-c:v", "libx264", "-preset", "ultrafast", // 编码速度,可根据需求进行更改 "-tune", "zerolatency", // 实时性优化,可根据需求进行更改 "-crf", "23", // 视频质量,可根据需求进行更改 "-f", "mp4", outputFile.getAbsolutePath() }; // 创建 FFmpeg 进程并启动编码 Process ffmpegProcess = FFmpeg.getInstance(context).execute(ffmpegCmd, new ExecuteBinaryResponseHandler() { @Override public void onSuccess(String message) { // 编码成功 } @Override public void onFailure(String message) { // 编码失败 } }); // 将 Camera2 的帧数据写入 FFmpeg 的输入管道 OutputStream ffmpegInputStream = ffmpegProcess.getOutputStream(); ffmpegInputStream.write(frameData); ffmpegInputStream.flush(); ffmpegInputStream.close(); ``` 请注意,上述代码仅为示例,您需要根据您的项目需求进行适当的修改。 希望这能对您有所帮助!如有任何进一步的问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值