ffmpeg3 rgb编码为h264文件

举报
风吹稻花香 发表于 2021/06/04 23:52:51 2021/06/04
【摘要】 #include "stdafx.h"   #include <stdio.h>   #include <stdlib.h>   #include <string.h>     extern "C"...


#include "stdafx.h"  

#include <stdio.h>  

#include <stdlib.h>  

#include <string.h>  

 

extern "C"

{

#include <libavcodec\avcodec.h>    

#include <libavformat\avformat.h>    

#include <libswscale\swscale.h>    

#include <libavutil\pixfmt.h>    

#include <libavutil\imgutils.h>    

#include <libavutil/opt.h>  

#include "libavutil/time.h"

}

 

 

static AVCodecContext *c = NULL;

 

static AVFrame *frame;

 

static AVPacket pkt;

 

static FILE *file;

 

struct SwsContext *sws_context = NULL;

 

 

 

static void ffmpeg_encoder_set_frame_yuv_from_rgb(uint8_t *rgb) {

 

const int in_linesize[1] = { 3 * c->width };

 

sws_context = sws_getCachedContext(sws_context,

 

c->width, c->height, AV_PIX_FMT_RGB24,

 

c->width, c->height, AV_PIX_FMT_YUV420P,

 

0, 0, 0, 0);

 

sws_scale(sws_context, (const uint8_t * const *)&rgb, in_linesize, 0,

 

c->height, frame->data, frame->linesize);

 

}

 

 

 

uint8_t* generate_rgb(int width, int height, int pts, uint8_t *rgb) {

 

int x, y, cur;

 

rgb = (uint8_t *)realloc(rgb, 3 * sizeof(uint8_t) * height * width);

 

for (y = 0; y < height; y++) {

 

for (x = 0; x < width; x++) {

 

cur = 3 * (y * width + x);

 

rgb[cur + 0] = 0;

 

rgb[cur + 1] = 0;

 

rgb[cur + 2] = 0;

 

if ((frame->pts / 25) % 2 == 0) {

 

if (y < height / 2) {

 

if (x < width / 2) {

 

/* Black. */

 

}

 

else {

 

rgb[cur + 0] = 255;

 

}

 

}

 

else {

 

if (x < width / 2) {

 

rgb[cur + 1] = 255;

 

}

 

else {

 

rgb[cur + 2] = 255;

 

}

 

}

 

}

 

else {

 

if (y < height / 2) {

 

rgb[cur + 0] = 255;

 

if (x < width / 2) {

 

rgb[cur + 1] = 255;

 

}

 

else {

 

rgb[cur + 2] = 255;

 

}

 

}

 

else {

 

if (x < width / 2) {

 

rgb[cur + 1] = 255;

 

rgb[cur + 2] = 255;

 

}

 

else {

 

rgb[cur + 0] = 255;

 

rgb[cur + 1] = 255;

 

rgb[cur + 2] = 255;

 

}

 

}

 

}

 

}

 

}

 

return rgb;

 

}

 

 

 

/* Allocate resources and write header data to the output file. */

 

void ffmpeg_encoder_start(const char *filename, int codec_id, int fps, int width, int height) {

 

AVCodec *codec;

 

int ret;

 

 

 

codec = avcodec_find_encoder((AVCodecID)codec_id);

 

if (!codec) {

 

fprintf(stderr, "Codec not found\n");

 

exit(1);

 

}

 

c = avcodec_alloc_context3(codec);

 

if (!c) {

 

fprintf(stderr, "Could not allocate video codec context\n");

 

exit(1);

 

}

 

c->bit_rate = 400000;

 

c->width = width;

 

c->height = height;

 

c->time_base.num = 1;

 

c->time_base.den = fps;

 

c->keyint_min = 600;

 

c->pix_fmt = AV_PIX_FMT_YUV420P;

 

if (codec_id == AV_CODEC_ID_H264)

 

av_opt_set(c->priv_data, "preset", "slow", 0);

 

if (avcodec_open2(c, codec, NULL) < 0) {

 

fprintf(stderr, "Could not open codec\n");

 

exit(1);

 

}

 

file = fopen(filename, "wb");

 

if (!file) {

 

fprintf(stderr, "Could not open %s\n", filename);

 

exit(1);

 

}

 

frame = av_frame_alloc();

 

if (!frame) {

 

fprintf(stderr, "Could not allocate video frame\n");

 

exit(1);

 

}

 

frame->format = c->pix_fmt;

 

frame->width = c->width;

 

frame->height = c->height;

 

ret = av_image_alloc(frame->data, frame->linesize, c->width, c->height, c->pix_fmt, 32);

 

if (ret < 0) {

 

fprintf(stderr, "Could not allocate raw picture buffer\n");

 

exit(1);

 

}

 

}

 

 

 

/*

 

Write trailing data to the output file

 

and free resources allocated by ffmpeg_encoder_start.

 

*/

 

void ffmpeg_encoder_finish(void) {

 

uint8_t endcode[] = { 0, 0, 1, 0xb7 };

 

int got_output, ret;

 

do {

 

fflush(stdout);

 

ret = avcodec_encode_video2(c, &pkt, NULL, &got_output);

 

if (ret < 0) {

 

fprintf(stderr, "Error encoding frame\n");

 

exit(1);

 

}

 

if (got_output) {

 

fwrite(pkt.data, 1, pkt.size, file);

 

av_packet_unref(&pkt);

 

}

 

} while (got_output);

 

fwrite(endcode, 1, sizeof(endcode), file);

 

fclose(file);

 

avcodec_close(c);

 

av_free(c);

 

av_freep(&frame->data[0]);

 

av_frame_free(&frame);

 

}

 

 

 

/*

 

Encode one frame from an RGB24 input and save it to the output file.

 

Must be called after ffmpeg_encoder_start, and ffmpeg_encoder_finish

 

must be called after the last call to this function.

 

*/

 

void ffmpeg_encoder_encode_frame(uint8_t *rgb) {

 

int ret, got_output;

 

ffmpeg_encoder_set_frame_yuv_from_rgb(rgb);

 

av_init_packet(&pkt);

 

pkt.data = NULL;

 

pkt.size = 0;

 

if (frame->pts == 1) {

 

frame->key_frame = 1;

 

frame->pict_type = AV_PICTURE_TYPE_I;

 

}

 

else {

 

frame->key_frame = 0;

 

frame->pict_type = AV_PICTURE_TYPE_P;

 

}

 

ret = avcodec_encode_video2(c, &pkt, frame, &got_output);

 

if (ret < 0) {

 

fprintf(stderr, "Error encoding frame\n");

 

exit(1);

 

}

 

if (got_output) {

 

fwrite(pkt.data, 1, pkt.size, file);

 

av_packet_unref(&pkt);

 

}

 

}

 

 

int main(void) {

  av_register_all();

av_log_set_level(AV_LOG_INFO);
avformat_network_init();
avcodec_register_all();

 char *filename = "tmp.h264";

 int pts;

 int width = 320;

 int height = 240;

 uint8_t *rgb = NULL;

 ffmpeg_encoder_start(filename, AV_CODEC_ID_H264, 25, width, height);

 for (pts = 0; pts < 1000; pts++) {

 frame->pts = pts;

 rgb = generate_rgb(width, height, pts, rgb);

 ffmpeg_encoder_encode_frame(rgb);

 }

 ffmpeg_encoder_finish();

//encode_example("tmp.mpg", AV_CODEC_ID_MPEG1VIDEO);

 return 0;

 }

文章来源: blog.csdn.net,作者:网奇,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/jacke121/article/details/80374910

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。