当然,也是从网上下载的。经过一番修改编译,运行结果正确。
由于dts/pts,有的播放器播放速度有所差异。
这里就直接就共享出来(也可以去下载区下载)。
#ifndef __GH_RTSP_2_MP4_H__
#define __GH_RTSP_2_MP4_H__
#include
#include
#include
#ifdef __cplusplus
extern "C" {
#endif
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#ifdef __cplusplus
}
#endif
#endif
#include
#include
#include
#include "gh_rtsp2mp4.h"
#define RTSP "rtsp://admin:[email protected]"
//经过实验,这个值最好
#define PTS_VALUE 4500
#define MAX_FRAMES 300
static bool g_RunningFlag = true;
static int rtsp2mp4(const char* pInputFileName, const char* pOutputFileName)
{
AVOutputFormat *ofmt = NULL;
//Input AVFormatContext and Output AVFormatContext
AVFormatContext *ifmt_ctx = NULL, *ofmt_ctx = NULL;
AVPacket pkt;
int ret=0, i=0;
int video_index = -1;
int frame_index = 0;
int dts = 0;
int pts = 0;
av_register_all();
//Network
avformat_network_init();
//Input
if ((ret = avformat_open_input(&ifmt_ctx, pInputFileName, 0, 0)) < 0)
{
printf("Could not open input file.");
goto end;
}
if ((ret = avformat_find_stream_info(ifmt_ctx, 0)) < 0)
{
printf("Failed to retrieve input stream information");
goto end;
}
for (i = 0; inb_streams; i++)
{
if (ifmt_ctx->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO)
{
video_index = i;
break;
}
}
av_dump_format(ifmt_ctx, 0, pInputFileName, 0);
//Output
avformat_alloc_output_context2(&ofmt_ctx, NULL, NULL, pOutputFileName); //RTMP
if (!ofmt_ctx)
{
printf("Could not create output context\n");
ret = AVERROR_UNKNOWN;
goto end;
}
ofmt = ofmt_ctx->oformat;
for (i = 0; i < ifmt_ctx->nb_streams; i++)
{
//Create output AVStream according to input AVStream
AVStream *in_stream = ifmt_ctx->streams[i];
AVStream *out_stream = avformat_new_stream(ofmt_ctx, in_stream->codec->codec);
if (!out_stream)
{
printf("Failed allocating output stream\n");
ret = AVERROR_UNKNOWN;
goto end;
}
//Copy the settings of AVCodecContext
ret = avcodec_copy_context(out_stream->codec, in_stream->codec);
if (ret < 0)
{
printf("Failed to copy context from input to output stream codec context\n");
goto end;
}
out_stream->codec->codec_tag = 0;
if (ofmt_ctx->oformat->flags & AVFMT_GLOBALHEADER)
{
out_stream->codec->flags |= AV_CODEC_FLAG_GLOBAL_HEADER;
}
}
//Dump Format------------------
av_dump_format(ofmt_ctx, 0, pOutputFileName, 1);
//Open output URL
if (!(ofmt->flags & AVFMT_NOFILE))
{
ret = avio_open(&ofmt_ctx->pb, pOutputFileName, AVIO_FLAG_WRITE);
if (ret < 0)
{
printf("Could not open output URL '%s'", pOutputFileName);
goto end;
}
}
//Write file header
ret = avformat_write_header(ofmt_ctx, NULL);
if (ret < 0)
{
printf("Error occurred when opening output URL\n");
goto end;
}
#if USE_H264BSF
AVBitStreamFilterContext* h264bsfc = av_bitstream_filter_init("h264_mp4toannexb");
#endif
while (g_RunningFlag)
{
AVStream *in_stream, *out_stream;
//Get an AVPacket
ret = av_read_frame(ifmt_ctx, &pkt);
if (ret < 0)
{
break;
}
in_stream = ifmt_ctx->streams[pkt.stream_index];
out_stream = ofmt_ctx->streams[pkt.stream_index];
//Convert PTS/DTS
#if 1
if (pts == 0)
{
pts = pkt.pts;
dts = pkt.dts;
}
else
{
pkt.pts += pts;
pkt.dts += dts;
}
if (pkt.dts < pkt.pts)
{
pkt.dts = pkt.pts;
}
#else
pkt.pts += 4500*frame_index;
pkt.dts += 4500*frame_index;
#endif
pkt.duration = av_rescale_q(pkt.duration, in_stream->time_base, out_stream->time_base);
pkt.pos = -1;
if (pkt.stream_index == video_index)
{
//printf("Receive %8d video frames from RTSP\n", frame_index);
frame_index++;
#if USE_H264BSF
av_bitstream_filter_filter(h264bsfc, in_stream->codec, NULL, &pkt.data, &pkt.size, pkt.data, pkt.size, 0);
#endif
}
ret = av_interleaved_write_frame(ofmt_ctx, &pkt);
if (ret < 0 && frame_index > 3)
{
printf("Error muxing packet\n");
break;
}
av_free_packet(&pkt);
if (frame_index > MAX_FRAMES)
{
break;
}
}
#if USE_H264BSF
av_bitstream_filter_close(h264bsfc);
#endif
//Write file trailer
av_write_trailer(ofmt_ctx);
end:
avformat_close_input(&ifmt_ctx);
/* close output */
if (ofmt_ctx && !(ofmt->flags & AVFMT_NOFILE))
{
avio_close(ofmt_ctx->pb);
}
avformat_free_context(ofmt_ctx);
if (ret < 0 && ret != AVERROR_EOF)
{
printf("Error occurred.\n");
return -1;
}
return 0;
}
int main(int argc, char **argv)
{
rtsp2mp4(RTSP, "rtsp.mp4");
return 0;
}
APP:= gh_rtsp2mp4
SRCS:= gh_rtsp2mp4.cpp
INCS:= $(wildcard *.h)
OBJS:= $(SRCS:.cpp=.o)
#GST
CPPFLAGS+= -DGST_FLAG \
-I/usr/local/include
LIBS := -L /usr/lib/x86_64-linux-gnu -lavcodec -lavutil -lswresample -lavformat \
-L /usr/linclude -lz -llzma -lbz2
# -L /usr/local/lib -lopencv_core -lopencv_imgcodecs -lopencv_imgproc -lopencv_highgui \
LIBS += $(GLLINK)
# LIBS += -lGL -lGLU -lglut
#LIBS+= `pkg-config --libs $(PKGS)`
all: $(APP)
.o: .cpp $(INCS) Makefile
g++ -g -c $@ $(CPPFLAGS) $<
$(APP): $(OBJS) Makefile
g++ -g -o $(APP) $(OBJS) $(LIBS)
clean:
rm -rf $(OBJS) $(APP)