mirror of
https://github.com/mpv-player/mpv
synced 2025-01-12 18:02:36 +00:00
dvd, bd: fix A/V sync
Slightly less robust, but simpler, and usually guarantees that audio and video are properly in sync.
This commit is contained in:
parent
1d7a68d75c
commit
34fdf082d8
@ -16,6 +16,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <math.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
#include "common/common.h"
|
#include "common/common.h"
|
||||||
@ -37,15 +38,18 @@ struct priv {
|
|||||||
// might add the streams only later.
|
// might add the streams only later.
|
||||||
struct sh_stream *dvd_subs[32];
|
struct sh_stream *dvd_subs[32];
|
||||||
// Used to rewrite the raw MPEG timestamps to playback time.
|
// Used to rewrite the raw MPEG timestamps to playback time.
|
||||||
struct {
|
|
||||||
double base_time; // playback display start time of current segment
|
double base_time; // playback display start time of current segment
|
||||||
double base_dts; // packet DTS that maps to base_time
|
double base_dts; // packet DTS that maps to base_time
|
||||||
double last_dts; // DTS of previously demuxed packet
|
double last_dts; // DTS of previously demuxed packet
|
||||||
} pts[STREAM_TYPE_COUNT];
|
|
||||||
double seek_pts;
|
double seek_pts;
|
||||||
bool seek_reinit; // needs reinit after seek
|
bool seek_reinit; // needs reinit after seek
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// If the timestamp difference between subsequent packets is this big, assume
|
||||||
|
// a reset. It should be big enough to account for 1. low video framerates and
|
||||||
|
// large audio frames, and 2. bad interleaving.
|
||||||
|
#define DTS_RESET_THRESHOLD 5.0
|
||||||
|
|
||||||
static void reselect_streams(demuxer_t *demuxer)
|
static void reselect_streams(demuxer_t *demuxer)
|
||||||
{
|
{
|
||||||
struct priv *p = demuxer->priv;
|
struct priv *p = demuxer->priv;
|
||||||
@ -189,11 +193,8 @@ static void reset_pts(demuxer_t *demuxer)
|
|||||||
|
|
||||||
MP_VERBOSE(demuxer, "reset to time: %f\n", base);
|
MP_VERBOSE(demuxer, "reset to time: %f\n", base);
|
||||||
|
|
||||||
for (int n = 0; n < STREAM_TYPE_COUNT; n++) {
|
p->base_dts = p->last_dts = MP_NOPTS_VALUE;
|
||||||
p->pts[n].base_dts = p->pts[n].last_dts = MP_NOPTS_VALUE;
|
p->base_time = base;
|
||||||
p->pts[n].base_time = base;
|
|
||||||
}
|
|
||||||
|
|
||||||
p->seek_reinit = false;
|
p->seek_reinit = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -222,32 +223,30 @@ static int d_fill_buffer(demuxer_t *demuxer)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int t = sh->type;
|
|
||||||
|
|
||||||
// Subtitle timestamps are not continuous, so the heuristic below can't be
|
|
||||||
// applied. Instead, use the video stream as reference.
|
|
||||||
if (t == STREAM_SUB)
|
|
||||||
t = STREAM_VIDEO;
|
|
||||||
|
|
||||||
MP_TRACE(demuxer, "ipts: %d %f %f\n", sh->type, pkt->pts, pkt->dts);
|
MP_TRACE(demuxer, "ipts: %d %f %f\n", sh->type, pkt->pts, pkt->dts);
|
||||||
|
|
||||||
if (sh->type == STREAM_SUB) {
|
if (sh->type == STREAM_SUB) {
|
||||||
if (p->pts[t].base_dts == MP_NOPTS_VALUE)
|
if (p->base_dts == MP_NOPTS_VALUE)
|
||||||
MP_WARN(demuxer, "subtitle packet along PTS reset, report a bug\n");
|
MP_WARN(demuxer, "subtitle packet along PTS reset\n");
|
||||||
} else if (pkt->dts != MP_NOPTS_VALUE) {
|
} else if (pkt->dts != MP_NOPTS_VALUE) {
|
||||||
if (p->pts[t].base_dts == MP_NOPTS_VALUE)
|
// Use the very first DTS to rebase the start time of the MPEG stream
|
||||||
p->pts[t].base_dts = p->pts[t].last_dts = pkt->dts;
|
// to the playback time.
|
||||||
|
if (p->base_dts == MP_NOPTS_VALUE)
|
||||||
|
p->base_dts = pkt->dts;
|
||||||
|
|
||||||
if (pkt->dts < p->pts[t].last_dts || pkt->dts > p->pts[t].last_dts + 0.5)
|
if (p->last_dts == MP_NOPTS_VALUE)
|
||||||
{
|
p->last_dts = pkt->dts;
|
||||||
MP_VERBOSE(demuxer, "PTS discontinuity on stream %d\n", sh->type);
|
|
||||||
p->pts[t].base_time += p->pts[t].last_dts - p->pts[t].base_dts;
|
if (fabs(p->last_dts - pkt->dts) >= DTS_RESET_THRESHOLD) {
|
||||||
p->pts[t].base_dts = p->pts[t].last_dts = pkt->dts - pkt->duration;
|
MP_WARN(demuxer, "PTS discontinuity: %f->%f\n", p->last_dts, pkt->dts);
|
||||||
|
p->base_time += p->last_dts - p->base_dts;
|
||||||
|
p->base_dts = pkt->dts - pkt->duration;
|
||||||
}
|
}
|
||||||
p->pts[t].last_dts = pkt->dts;
|
p->last_dts = pkt->dts;
|
||||||
}
|
}
|
||||||
if (p->pts[t].base_dts != MP_NOPTS_VALUE) {
|
|
||||||
double delta = -p->pts[t].base_dts + p->pts[t].base_time;
|
if (p->base_dts != MP_NOPTS_VALUE) {
|
||||||
|
double delta = -p->base_dts + p->base_time;
|
||||||
if (pkt->pts != MP_NOPTS_VALUE)
|
if (pkt->pts != MP_NOPTS_VALUE)
|
||||||
pkt->pts += delta;
|
pkt->pts += delta;
|
||||||
if (pkt->dts != MP_NOPTS_VALUE)
|
if (pkt->dts != MP_NOPTS_VALUE)
|
||||||
|
Loading…
Reference in New Issue
Block a user