【问题标题】:Converting RGB frames to YUV420P using FFmpeg/C++ [duplicate]使用 FFmpeg/C++ 将 RGB 帧转换为 YUV420P [重复]
【发布时间】:2014-04-27 08:22:16
【问题描述】:

我尝试了代码based on this tutorial。它的作用是将视频的前 5 帧以 RGB 格式写入内存。我需要将这些 RGB 帧转换为 YUV420P 格式,以便我可以将它们传递给 YUV420P 编码器。我尝试使用sws_scale,但生成的帧会丢失颜色信息并且还会出现缩放问题。此外,查看了各种问题,但没有一个给出可行的解决方案。我对 FFmpeg 库完全陌生,并且正在研究各种教程以掌握它。我希望有人可以提出一个很好的解决方案。非常感谢您的投入!

以下是代码:

#include <ffmpeg/avcodec.h>
#include <ffmpeg/avformat.h>

#include <stdio.h>

void SaveFrame(AVFrame *pFrame, int width, int height, int iFrame) {
  FILE *pFile;
  char szFilename[32];
  int  y;

  // Open file
  sprintf(szFilename, "frame%d.ppm", iFrame);
  pFile=fopen(szFilename, "wb");
  if(pFile==NULL)
    return;

  // Write header
  fprintf(pFile, "P6\n%d %d\n255\n", width, height);

  // Write pixel data
  for(y=0; y<height; y++)
    fwrite(pFrame->data[0]+y*pFrame->linesize[0], 1, width*3, pFile);

  // Close file
  fclose(pFile);
}

int main(int argc, char *argv[]) {
  AVFormatContext *pFormatCtx;
  int             i, videoStream;
  AVCodecContext  *pCodecCtx;
  AVCodec         *pCodec;
  AVFrame         *pFrame; 
  AVFrame         *pFrameRGB;
  AVPacket        packet;
  int             frameFinished;
  int             numBytes;
  uint8_t         *buffer;

  if(argc < 2) {
    printf("Please provide a movie file\n");
    return -1;
  }
  // Register all formats and codecs
  av_register_all();

  // Open video file
  if(av_open_input_file(&pFormatCtx, argv[1], NULL, 0, NULL)!=0)
    return -1; // Couldn't open file

  // Retrieve stream information
  if(av_find_stream_info(pFormatCtx)<0)
    return -1; // Couldn't find stream information

  // Dump information about file onto standard error
  dump_format(pFormatCtx, 0, argv[1], 0);

  // Find the first video stream
  videoStream=-1;
  for(i=0; i<pFormatCtx->nb_streams; i++)
    if(pFormatCtx->streams[i]->codec->codec_type==CODEC_TYPE_VIDEO) {
      videoStream=i;
      break;
    }
  if(videoStream==-1)
    return -1; // Didn't find a video stream

  // Get a pointer to the codec context for the video stream
  pCodecCtx=pFormatCtx->streams[videoStream]->codec;

  // Find the decoder for the video stream
  pCodec=avcodec_find_decoder(pCodecCtx->codec_id);
  if(pCodec==NULL) {
    fprintf(stderr, "Unsupported codec!\n");
    return -1; // Codec not found
  }
  // Open codec
  if(avcodec_open(pCodecCtx, pCodec)<0)
    return -1; // Could not open codec

  // Allocate video frame
  pFrame=avcodec_alloc_frame();

  // Allocate an AVFrame structure
  pFrameRGB=avcodec_alloc_frame();
  if(pFrameRGB==NULL)
    return -1;

  // Determine required buffer size and allocate buffer
  numBytes=avpicture_get_size(PIX_FMT_RGB24, pCodecCtx->width,
                  pCodecCtx->height);
  buffer=(uint8_t *)av_malloc(numBytes*sizeof(uint8_t));

  // Assign appropriate parts of buffer to image planes in pFrameRGB
  // Note that pFrameRGB is an AVFrame, but AVFrame is a superset
  // of AVPicture
  avpicture_fill((AVPicture *)pFrameRGB, buffer, PIX_FMT_RGB24,
         pCodecCtx->width, pCodecCtx->height);

  // Read frames and save first five frames to disk
  i=0;
  while(av_read_frame(pFormatCtx, &packet)>=0) {
    // Is this a packet from the video stream?
    if(packet.stream_index==videoStream) {
      // Decode video frame
      avcodec_decode_video(pCodecCtx, pFrame, &frameFinished, 
               packet.data, packet.size);

      // Did we get a video frame?
      if(frameFinished) {
    // Convert the image from its native format to RGB
    img_convert((AVPicture *)pFrameRGB, PIX_FMT_RGB24, 
                    (AVPicture*)pFrame, pCodecCtx->pix_fmt, pCodecCtx->width, 
                    pCodecCtx->height);

    // Save the frame to disk
    if(++i<=5)
      SaveFrame(pFrameRGB, pCodecCtx->width, pCodecCtx->height, 
            i);
      }
    }

    // Free the packet that was allocated by av_read_frame
    av_free_packet(&packet);
  }

  // Free the RGB image
  av_free(buffer);
  av_free(pFrameRGB);

  // Free the YUV frame
  av_free(pFrame);

  // Close the codec
  avcodec_close(pCodecCtx);

  // Close the video file
  av_close_input_file(pFormatCtx);

  return 0;
}

【问题讨论】:

  • 我试图完全理解你在这里做什么。您有 RGB24 的帧,您希望它们进行 YUV 编码,然后您想解码并转换回 RGB24 以显示它们......对吗?
  • @Jack 是的,这就是整个场景。我有 RGB 帧。我想将它们转换为 YUV420P 格式,使用 YUV420P 编解码器进行编码和解码,最后将解码后的图像转换回 RGB 以显示它们。
  • 从解码器生成 RGB 是没有意义的,只是将它们转换回 YUV 进行编码。当您要求 RGB 时,解码器会做什么,它会解码 YUV,然后在内部将其转换为 RGB,因此 - 双重工作。

标签: c++ ffmpeg video-encoding libavcodec libav


【解决方案1】:

您从其他人的网站复制并粘贴了旧代码,这些代码甚至无法使用最新的 libav* 进行编译。您在询问 sws_scale,而这段代码甚至没有使用它。所以代码与问题100%无关。最重要的是,你以前问过这个问题,我已经回答过了。 ffmpeg: RGB to YUV conversion loses color and scale

【讨论】:

    猜你喜欢
    • 2018-11-19
    • 2018-02-06
    • 2018-03-14
    • 1970-01-01
    • 2021-11-25
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多