1

使用 h264 编解码器的 avi 文件。我需要打开它并提取每一帧的时间戳。

我试图在 opencv 中这样做,但它返回错误的结果。

我已经修改了 fffmpeg/doc/examples 文件夹中存在的示例 filtering_video.c 的代码,以获取时间戳。

我有以下代码,但编译它给了我一个关于 UNIT64_C 的奇怪结果。环顾四周,我发现这是一个 ffmpeg 问题。

所以 2 个问题:1. 这段代码看起来是否能够返回和打印帧时间戳 2. UINT64_C 问题的任何解决方案?

使用 ubuntu 11.04 和最新的 ffmpeg。

代码

#include <unistd.h>

#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>

static AVFormatContext *fmt_ctx;
static AVCodecContext *dec_ctx;
static int video_stream_index = -1;

static int open_input_file(const char *filename)
{
int ret;
AVCodec *dec;

if ((ret = avformat_open_input(&fmt_ctx, filename, NULL, NULL)) < 0)
{
    printf("Cannot open input file\n");
    return ret;
}

if ((ret = avformat_find_stream_info(fmt_ctx, NULL)) < 0)
{
    printf("Cannot find stream information\n");
    return ret;
}

/* select the video stream */
ret = av_find_best_stream(fmt_ctx, AVMEDIA_TYPE_VIDEO, -1, -1, &dec, 0);
if (ret < 0)
{
    printf("Cannot find a video stream in the input file\n");
    return ret;
}
video_stream_index = ret;
dec_ctx = fmt_ctx->streams[video_stream_index]->codec;

/* init the video decoder */
if ((ret = avcodec_open2(dec_ctx, dec, NULL)) < 0)
{
    printf("Cannot open video decoder\n");
    return ret;
}

return 0;
}

int main(int argc, char **argv)
{
int ret;
AVPacket packet;
AVFrame frame;
int got_frame;

if (argc != 2)
{
    fprintf(stderr, "Usage: %s file\n", argv[0]);
    exit(1);
}

avcodec_register_all();
av_register_all();

if ((ret = open_input_file(argv[1])) < 0)
    goto end;

/* read all packets */
while (1)
{
    if ((ret = av_read_frame(fmt_ctx, &packet)) < 0)
        break;

    if (packet.stream_index == video_stream_index)
    {
        avcodec_get_frame_defaults(&frame);
        got_frame = 0;
        ret = avcodec_decode_video2(dec_ctx, &frame, &got_frame, &packet);
        if (ret < 0)
        {
           printf("Error decoding video\n");
            break;
        }

        if (got_frame)
        {
            frame.pts = av_frame_get_best_effort_timestamp(&frame);
            printf("frame timestamp %ld\n", frame.pts);
        }

    }
}
    av_free_packet(&packet);
end:
if (dec_ctx)
    avcodec_close(dec_ctx);
avformat_close_input(&fmt_ctx);

exit(0);
}

编译器错误

/usr/local/include/libavutil/common.h||In function ‘int32_t    av_clipl_int32_c(int64_t)’:|
/usr/local/include/libavutil/common.h|173|error: ‘UINT64_C’ was not declared in this scope|
4

3 回答 3

2

UINT64_C在任何其他标头之前声明了<stdint.h>它,它不应该再抱怨了。

于 2012-07-29T20:29:32.897 回答
0

的定义UINT64_C在一个条件内:

/* The ISO C99 standard specifies that in C++ implementations these
   should only be defined if explicitly requested.  */
#if !defined __cplusplus || defined __STDC_CONSTANT_MACROS

看起来如果你像我一样使用来自 C++98 的 ffmpeg,你需要

#define __STDC_CONSTANT_MACROS

在你面前

#include <stdint.h>
于 2013-01-26T02:13:33.010 回答
0

我发现 stdint.h 只为 C 定义了宏 INT64_C,而不是为 C++ 定义。由于我不想对 ffmpeg 的标头进行太多麻烦,因此我将此定义放在了我的代码中,尤其是来自 ffmpeg 的所有包含

#ifndef INT64_C
#define INT64_C(c) (c ## LL)
#define UINT64_C(c) (c ## ULL)
#endif
于 2013-11-28T08:07:17.497 回答