summaryrefslogtreecommitdiff
path: root/libavformat
diff options
context:
space:
mode:
authorMichael Niedermayer <michaelni@gmx.at>2008-02-20 03:20:06 +0000
committerMichael Niedermayer <michaelni@gmx.at>2008-02-20 03:20:06 +0000
commitff2635b273269dff524bc89242902011b68794ff (patch)
treef643dca3a71abc8308fde12d4bd65c34997427d6 /libavformat
parent8815bf011e309aea8774a0cd584cbfadc16558c0 (diff)
Make timestamp interpolation work with mpeg2 field pictures.
Cleaner/simpler solutions are welcome. Originally committed as revision 12156 to svn://svn.ffmpeg.org/ffmpeg/trunk
Diffstat (limited to 'libavformat')
-rw-r--r--libavformat/avformat.h4
-rw-r--r--libavformat/utils.c30
2 files changed, 23 insertions, 11 deletions
diff --git a/libavformat/avformat.h b/libavformat/avformat.h
index 06396eebff..cc09e02776 100644
--- a/libavformat/avformat.h
+++ b/libavformat/avformat.h
@@ -337,8 +337,8 @@ typedef struct AVStream {
struct AVCodecParserContext *parser;
int64_t cur_dts;
- int last_IP_duration;
- int64_t last_IP_pts;
+ int last_IP_duration[2];
+ int64_t last_IP_pts[2];
/* av_seek_frame() support */
AVIndexEntry *index_entries; /**< only used if the format does not
support seeking natively */
diff --git a/libavformat/utils.c b/libavformat/utils.c
index 79d502cd9e..4936349d7d 100644
--- a/libavformat/utils.c
+++ b/libavformat/utils.c
@@ -702,21 +702,30 @@ static void compute_pkt_fields(AVFormatContext *s, AVStream *st,
/* interpolate PTS and DTS if they are not present */
if(delay <=1){
if (presentation_delayed) {
+ int fields= 2 + (pc ? pc->repeat_pict : 0);
+ int field_duration= pkt->duration / fields;
+ int parity= pc ? pc->parity : 0;
/* DTS = decompression timestamp */
/* PTS = presentation timestamp */
if (pkt->dts == AV_NOPTS_VALUE)
- pkt->dts = st->last_IP_pts;
+ pkt->dts = st->last_IP_pts[parity];
update_initial_timestamps(s, pkt->stream_index, pkt->dts, pkt->pts);
if (pkt->dts == AV_NOPTS_VALUE)
pkt->dts = st->cur_dts;
/* this is tricky: the dts must be incremented by the duration
of the frame we are displaying, i.e. the last I- or P-frame */
- if (st->last_IP_duration == 0)
- st->last_IP_duration = pkt->duration;
- st->cur_dts = pkt->dts + st->last_IP_duration;
- st->last_IP_duration = pkt->duration;
- st->last_IP_pts= pkt->pts;
+ st->cur_dts = pkt->dts;
+ for(i=0; i<fields; i++){
+ int p= (parity + i)&1;
+ if(!st->last_IP_duration[p])
+ st->last_IP_duration[p]= field_duration;
+ st->cur_dts += st->last_IP_duration[p];
+ st->last_IP_pts[p]= pkt->pts;
+ if(pkt->pts != AV_NOPTS_VALUE)
+ st->last_IP_pts[p] += i*field_duration;
+ st->last_IP_duration[p]= field_duration;
+ }
/* cannot compute PTS if not present (we can compute it only
by knowing the future */
} else if(pkt->pts != AV_NOPTS_VALUE || pkt->dts != AV_NOPTS_VALUE || pkt->duration){
@@ -1014,7 +1023,8 @@ static void av_read_frame_flush(AVFormatContext *s)
av_parser_close(st->parser);
st->parser = NULL;
}
- st->last_IP_pts = AV_NOPTS_VALUE;
+ st->last_IP_pts[0] =
+ st->last_IP_pts[1] = AV_NOPTS_VALUE;
st->cur_dts = AV_NOPTS_VALUE; /* we set the current DTS to an unspecified origin */
}
}
@@ -1622,7 +1632,8 @@ static void av_estimate_timings_from_pts(AVFormatContext *ic, offset_t old_offse
for(i=0; i<ic->nb_streams; i++){
st= ic->streams[i];
st->cur_dts= st->first_dts;
- st->last_IP_pts = AV_NOPTS_VALUE;
+ st->last_IP_pts[0] =
+ st->last_IP_pts[1] = AV_NOPTS_VALUE;
}
}
@@ -2181,7 +2192,8 @@ AVStream *av_new_stream(AVFormatContext *s, int id)
/* default pts setting is MPEG-like */
av_set_pts_info(st, 33, 1, 90000);
- st->last_IP_pts = AV_NOPTS_VALUE;
+ st->last_IP_pts[0] =
+ st->last_IP_pts[1] = AV_NOPTS_VALUE;
for(i=0; i<MAX_REORDER_DELAY+1; i++)
st->pts_buffer[i]= AV_NOPTS_VALUE;