clock.c 21.2 KB
Newer Older
1
/*****************************************************************************
2
 * clock.c: Clock/System date convertions, stream management
3
 *****************************************************************************
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
4
 * Copyright (C) 1999-2008 VLC authors and VideoLAN
5
 * Copyright (C) 2008 Laurent Aimar
6
 * $Id$
7 8
 *
 * Authors: Christophe Massiot <massiot@via.ecp.fr>
Laurent Aimar's avatar
Laurent Aimar committed
9
 *          Laurent Aimar < fenrir _AT_ videolan _DOT_ org >
10
 *
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
11 12 13
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation; either version 2.1 of the License, or
14
 * (at your option) any later version.
15
 *
16 17
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
18 19
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Lesser General Public License for more details.
20
 *
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
21 22 23
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24 25 26 27 28
 *****************************************************************************/

/*****************************************************************************
 * Preamble
 *****************************************************************************/
29 30 31 32
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif

33
#include <vlc_common.h>
34
#include <vlc_input.h>
35
#include "clock.h"
36
#include <assert.h>
37

38 39 40 41 42
/* TODO:
 * - clean up locking once clock code is stable
 *
 */

43
/*
44
 * DISCUSSION : SYNCHRONIZATION METHOD
45
 *
46 47 48 49 50 51 52
 * In some cases we can impose the pace of reading (when reading from a
 * file or a pipe), and for the synchronization we simply sleep() until
 * it is time to deliver the packet to the decoders. When reading from
 * the network, we must be read at the same pace as the server writes,
 * otherwise the kernel's buffer will trash packets. The risk is now to
 * overflow the input buffers in case the server goes too fast, that is
 * why we do these calculations :
53
 *
54 55 56 57
 * We compute a mean for the pcr because we want to eliminate the
 * network jitter and keep the low frequency variations. The mean is
 * in fact a low pass filter and the jitter is a high frequency signal
 * that is why it is eliminated by the filter/average.
58
 *
59 60 61 62 63 64 65
 * The low frequency variations enable us to synchronize the client clock
 * with the server clock because they represent the time variation between
 * the 2 clocks. Those variations (ie the filtered pcr) are used to compute
 * the presentation dates for the audio and video frames. With those dates
 * we can decode (or trash) the MPEG2 stream at "exactly" the same rate
 * as it is sent by the server and so we keep the synchronization between
 * the server and the client.
66
 *
67 68
 * It is a very important matter if you want to avoid underflow or overflow
 * in all the FIFOs, but it may be not enough.
69 70
 */

Laurent Aimar's avatar
Laurent Aimar committed
71
/* i_cr_average : Maximum number of samples used to compute the
72 73 74 75 76
 * dynamic average value.
 * We use the following formula :
 * new_average = (old_average * c_average + new_sample_value) / (c_average +1)
 */

77

78
/*****************************************************************************
79
 * Constants
80
 *****************************************************************************/
81 82

/* Maximum gap allowed between two CRs. */
83
#define CR_MAX_GAP (60 * CLOCK_FREQ)
84

85 86
/* Latency introduced on DVDs with CR == 0 on chapter change - this is from
 * my dice --Meuuh */
Laurent Aimar's avatar
Laurent Aimar committed
87
#define CR_MEAN_PTS_GAP (300000)
88

89 90 91 92 93 94 95 96 97
/* Rate (in 1/256) at which we will read faster to try to increase our
 * internal buffer (if we control the pace of the source).
 */
#define CR_BUFFERING_RATE (48)

/* Extra internal buffer value (in CLOCK_FREQ)
 * It is 60s max, remember as it is limited by the size it takes by es_out.c
 * it can be really large.
 */
98 99 100
//#define CR_BUFFERING_TARGET (60000000)
/* Due to some problems in es_out, we cannot use a large value yet */
#define CR_BUFFERING_TARGET (100000)
101

102 103 104
/*****************************************************************************
 * Structures
 *****************************************************************************/
105 106 107 108 109 110 111 112 113 114 115 116

/**
 * This structure holds long term average
 */
typedef struct
{
    mtime_t i_value;
    int     i_residue;

    int     i_count;
    int     i_divider;
} average_t;
117 118
static void    AvgInit( average_t *, int i_divider );
static void    AvgClean( average_t * );
119

120 121 122
static void    AvgReset( average_t * );
static void    AvgUpdate( average_t *, mtime_t i_value );
static mtime_t AvgGet( average_t * );
123
static void    AvgRescale( average_t *, int i_divider );
124

Laurent Aimar's avatar
Laurent Aimar committed
125 126 127 128 129 130 131 132 133 134 135 136 137
/* */
typedef struct
{
    mtime_t i_stream;
    mtime_t i_system;
} clock_point_t;

static inline clock_point_t clock_point_Create( mtime_t i_stream, mtime_t i_system )
{
    clock_point_t p = { .i_stream = i_stream, .i_system = i_system };
    return p;
}

138 139 140
/* */
#define INPUT_CLOCK_LATE_COUNT (3)

141
/* */
142 143
struct input_clock_t
{
144 145 146
    /* */
    vlc_mutex_t lock;

Laurent Aimar's avatar
Laurent Aimar committed
147 148
    /* Last point
     * It is used to detect unexpected stream discontinuities */
Laurent Aimar's avatar
Laurent Aimar committed
149
    clock_point_t last;
150

151
    /* Maximal timestamp returned by input_clock_ConvertTS (in system unit) */
152
    mtime_t i_ts_max;
153

154 155 156
    /* Amount of extra buffering expressed in stream clock */
    mtime_t i_buffering_duration;

Laurent Aimar's avatar
Laurent Aimar committed
157
    /* Clock drift */
158 159
    mtime_t i_next_drift_update;
    average_t drift;
160

161 162 163 164 165 166 167
    /* Late statistics */
    struct
    {
        mtime_t  pi_value[INPUT_CLOCK_LATE_COUNT];
        unsigned i_index;
    } late;

168 169 170 171
    /* Reference point */
    clock_point_t ref;
    bool          b_has_reference;

172 173 174 175
    /* External clock drift */
    mtime_t       i_external_clock;
    bool          b_has_external_clock;

Laurent Aimar's avatar
Laurent Aimar committed
176
    /* Current modifiers */
177
    bool    b_paused;
Laurent Aimar's avatar
Laurent Aimar committed
178
    int     i_rate;
179
    mtime_t i_pts_delay;
180
    mtime_t i_pause_date;
181 182
};

Laurent Aimar's avatar
Laurent Aimar committed
183
static mtime_t ClockStreamToSystem( input_clock_t *, mtime_t i_stream );
Laurent Aimar's avatar
Laurent Aimar committed
184
static mtime_t ClockSystemToStream( input_clock_t *, mtime_t i_system );
185

186 187
static mtime_t ClockGetTsOffset( input_clock_t * );

188
/*****************************************************************************
Laurent Aimar's avatar
Laurent Aimar committed
189
 * input_clock_New: create a new clock
190
 *****************************************************************************/
191
input_clock_t *input_clock_New( int i_rate )
192
{
193 194 195 196
    input_clock_t *cl = malloc( sizeof(*cl) );
    if( !cl )
        return NULL;

197
    vlc_mutex_init( &cl->lock );
198
    cl->b_has_reference = false;
199
    cl->ref = clock_point_Create( VLC_TS_INVALID, VLC_TS_INVALID );
200
    cl->b_has_external_clock = false;
201

202
    cl->last = clock_point_Create( VLC_TS_INVALID, VLC_TS_INVALID );
203

204
    cl->i_ts_max = VLC_TS_INVALID;
205

206 207
    cl->i_buffering_duration = 0;

208
    cl->i_next_drift_update = VLC_TS_INVALID;
209
    AvgInit( &cl->drift, 10 );
210

211 212 213 214
    cl->late.i_index = 0;
    for( int i = 0; i < INPUT_CLOCK_LATE_COUNT; i++ )
        cl->late.pi_value[i] = 0;

Laurent Aimar's avatar
Laurent Aimar committed
215
    cl->i_rate = i_rate;
216
    cl->i_pts_delay = 0;
217
    cl->b_paused = false;
218
    cl->i_pause_date = VLC_TS_INVALID;
Laurent Aimar's avatar
Laurent Aimar committed
219

220 221 222 223
    return cl;
}

/*****************************************************************************
Laurent Aimar's avatar
Laurent Aimar committed
224
 * input_clock_Delete: destroy a new clock
225
 *****************************************************************************/
Laurent Aimar's avatar
Laurent Aimar committed
226
void input_clock_Delete( input_clock_t *cl )
227
{
228
    AvgClean( &cl->drift );
229
    vlc_mutex_destroy( &cl->lock );
230
    free( cl );
231 232 233
}

/*****************************************************************************
234
 * input_clock_Update: manages a clock reference
235 236 237
 *
 *  i_ck_stream: date in stream clock
 *  i_ck_system: date in system clock
238
 *****************************************************************************/
239 240
void input_clock_Update( input_clock_t *cl, vlc_object_t *p_log,
                         bool *pb_late,
241
                         bool b_can_pace_control, bool b_buffering_allowed,
Laurent Aimar's avatar
Laurent Aimar committed
242
                         mtime_t i_ck_stream, mtime_t i_ck_system )
243
{
244
    bool b_reset_reference = false;
245

246 247
    assert( i_ck_stream > VLC_TS_INVALID && i_ck_system > VLC_TS_INVALID );

248
    vlc_mutex_lock( &cl->lock );
249

250
    if( !cl->b_has_reference )
251
    {
252 253
        /* */
        b_reset_reference= true;
254
    }
255
    else if( cl->last.i_stream > VLC_TS_INVALID &&
Laurent Aimar's avatar
Laurent Aimar committed
256 257
             ( (cl->last.i_stream - i_ck_stream) > CR_MAX_GAP ||
               (cl->last.i_stream - i_ck_stream) < -CR_MAX_GAP ) )
258
    {
259 260 261
        /* Stream discontinuity, for which we haven't received a
         * warning from the stream control facilities (dd-edited
         * stream ?). */
262
        msg_Warn( p_log, "clock gap, unexpected stream discontinuity" );
263
        cl->i_ts_max = VLC_TS_INVALID;
264 265

        /* */
266
        msg_Warn( p_log, "feeding synchro with a new reference point trying to recover from clock gap" );
267 268
        b_reset_reference= true;
    }
269 270

    /* */
271 272
    if( b_reset_reference )
    {
273
        cl->i_next_drift_update = VLC_TS_INVALID;
274
        AvgReset( &cl->drift );
275 276

        /* Feed synchro with a new reference point. */
Laurent Aimar's avatar
Laurent Aimar committed
277 278 279
        cl->b_has_reference = true;
        cl->ref = clock_point_Create( i_ck_stream,
                                      __MAX( cl->i_ts_max + CR_MEAN_PTS_GAP, i_ck_system ) );
280
        cl->b_has_external_clock = false;
281
    }
282

283 284
    /* Compute the drift between the stream clock and the system clock
     * when we don't control the source pace */
285
    if( !b_can_pace_control && cl->i_next_drift_update < i_ck_system )
286
    {
287
        const mtime_t i_converted = ClockSystemToStream( cl, i_ck_system );
288

289
        AvgUpdate( &cl->drift, i_converted - i_ck_stream );
290

291
        cl->i_next_drift_update = i_ck_system + CLOCK_FREQ/5; /* FIXME why that */
292
    }
293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312

    /* Update the extra buffering value */
    if( !b_can_pace_control || b_reset_reference )
    {
        cl->i_buffering_duration = 0;
    }
    else if( b_buffering_allowed )
    {
        /* Try to bufferize more than necessary by reading
         * CR_BUFFERING_RATE/256 faster until we have CR_BUFFERING_TARGET.
         */
        const mtime_t i_duration = __MAX( i_ck_stream - cl->last.i_stream, 0 );

        cl->i_buffering_duration += ( i_duration * CR_BUFFERING_RATE + 255 ) / 256;
        if( cl->i_buffering_duration > CR_BUFFERING_TARGET )
            cl->i_buffering_duration = CR_BUFFERING_TARGET;
    }
    //fprintf( stderr, "input_clock_Update: %d :: %lld\n", b_buffering_allowed, cl->i_buffering_duration/1000 );

    /* */
Laurent Aimar's avatar
Laurent Aimar committed
313
    cl->last = clock_point_Create( i_ck_stream, i_ck_system );
314

315 316 317
    /* It does not take the decoder latency into account but it is not really
     * the goal of the clock here */
    const mtime_t i_system_expected = ClockStreamToSystem( cl, i_ck_stream + AvgGet( &cl->drift ) );
318 319 320 321 322 323 324
    const mtime_t i_late = ( i_ck_system - cl->i_pts_delay ) - i_system_expected;
    *pb_late = i_late > 0;
    if( i_late > 0 )
    {
        cl->late.pi_value[cl->late.i_index] = i_late;
        cl->late.i_index = ( cl->late.i_index + 1 ) % INPUT_CLOCK_LATE_COUNT;
    }
325

326
    vlc_mutex_unlock( &cl->lock );
327 328
}

Laurent Aimar's avatar
Laurent Aimar committed
329
/*****************************************************************************
Laurent Aimar's avatar
Laurent Aimar committed
330
 * input_clock_Reset:
Laurent Aimar's avatar
Laurent Aimar committed
331
 *****************************************************************************/
Laurent Aimar's avatar
Laurent Aimar committed
332
void input_clock_Reset( input_clock_t *cl )
Laurent Aimar's avatar
Laurent Aimar committed
333
{
334 335
    vlc_mutex_lock( &cl->lock );

336
    cl->b_has_reference = false;
337
    cl->ref = clock_point_Create( VLC_TS_INVALID, VLC_TS_INVALID );
338
    cl->b_has_external_clock = false;
339
    cl->i_ts_max = VLC_TS_INVALID;
340 341

    vlc_mutex_unlock( &cl->lock );
Laurent Aimar's avatar
Laurent Aimar committed
342 343
}

344
/*****************************************************************************
345
 * input_clock_ChangeRate:
346
 *****************************************************************************/
347
void input_clock_ChangeRate( input_clock_t *cl, int i_rate )
348
{
349 350
    vlc_mutex_lock( &cl->lock );

351
    if( cl->b_has_reference )
352
    {
353 354 355
        /* Move the reference point (as if we were playing at the new rate
         * from the start */
        cl->ref.i_system = cl->last.i_system - (cl->last.i_system - cl->ref.i_system) * i_rate / cl->i_rate;
356
    }
357
    cl->i_rate = i_rate;
358 359

    vlc_mutex_unlock( &cl->lock );
360 361
}

362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385
/*****************************************************************************
 * input_clock_ChangePause:
 *****************************************************************************/
void input_clock_ChangePause( input_clock_t *cl, bool b_paused, mtime_t i_date )
{
    vlc_mutex_lock( &cl->lock );
    assert( (!cl->b_paused) != (!b_paused) );

    if( cl->b_paused )
    {
        const mtime_t i_duration = i_date - cl->i_pause_date;

        if( cl->b_has_reference && i_duration > 0 )
        {
            cl->ref.i_system += i_duration;
            cl->last.i_system += i_duration;
        }
    }
    cl->i_pause_date = i_date;
    cl->b_paused = b_paused;

    vlc_mutex_unlock( &cl->lock );
}

386
/*****************************************************************************
Laurent Aimar's avatar
Laurent Aimar committed
387
 * input_clock_GetWakeup
388
 *****************************************************************************/
389
mtime_t input_clock_GetWakeup( input_clock_t *cl )
390
{
391
    mtime_t i_wakeup = 0;
392

393 394 395 396
    vlc_mutex_lock( &cl->lock );

    /* Synchronized, we can wait */
    if( cl->b_has_reference )
397
        i_wakeup = ClockStreamToSystem( cl, cl->last.i_stream + AvgGet( &cl->drift ) - cl->i_buffering_duration );
398 399 400 401

    vlc_mutex_unlock( &cl->lock );

    return i_wakeup;
402 403 404
}

/*****************************************************************************
405
 * input_clock_ConvertTS
406
 *****************************************************************************/
407
int input_clock_ConvertTS( vlc_object_t *p_object, input_clock_t *cl,
408 409
                           int *pi_rate, mtime_t *pi_ts0, mtime_t *pi_ts1,
                           mtime_t i_ts_bound )
410
{
411
    assert( pi_ts0 );
412 413
    vlc_mutex_lock( &cl->lock );

414 415 416
    if( pi_rate )
        *pi_rate = cl->i_rate;

417
    if( !cl->b_has_reference )
418 419
    {
        vlc_mutex_unlock( &cl->lock );
420 421
        msg_Err(p_object, "Timestamp conversion failed for %"PRId64": "
                "no reference clock", *pi_ts0);
422
        *pi_ts0 = VLC_TS_INVALID;
423
        if( pi_ts1 )
424
            *pi_ts1 = VLC_TS_INVALID;
425
        return VLC_EGENERIC;
426
    }
427

428
    /* */
429
    const mtime_t i_ts_buffering = cl->i_buffering_duration * cl->i_rate / INPUT_RATE_DEFAULT;
430 431
    const mtime_t i_ts_delay = cl->i_pts_delay + ClockGetTsOffset( cl );

432
    /* */
433
    if( *pi_ts0 > VLC_TS_INVALID )
434 435 436 437
    {
        *pi_ts0 = ClockStreamToSystem( cl, *pi_ts0 + AvgGet( &cl->drift ) );
        if( *pi_ts0 > cl->i_ts_max )
            cl->i_ts_max = *pi_ts0;
438
        *pi_ts0 += i_ts_delay;
439
    }
440

Rafaël Carré's avatar
Rafaël Carré committed
441
    /* XXX we do not update i_ts_max on purpose */
442
    if( pi_ts1 && *pi_ts1 > VLC_TS_INVALID )
443 444
    {
        *pi_ts1 = ClockStreamToSystem( cl, *pi_ts1 + AvgGet( &cl->drift ) ) +
445
                  i_ts_delay;
446
    }
447

448
    vlc_mutex_unlock( &cl->lock );
449 450

    /* Check ts validity */
451 452
    if (i_ts_bound != INT64_MAX && *pi_ts0 > VLC_TS_INVALID) {
        if (*pi_ts0 >= mdate() + i_ts_delay + i_ts_buffering + i_ts_bound) {
453
            msg_Err(p_object,
454 455 456 457 458 459
                "Timestamp conversion failed (delay %"PRId64", buffering "
                "%"PRId64", bound %"PRId64")",
                i_ts_delay, i_ts_buffering, i_ts_bound);
            return VLC_EGENERIC;
        }
    }
460

461
    return VLC_SUCCESS;
462
}
463 464 465 466 467 468 469 470 471 472 473 474 475
/*****************************************************************************
 * input_clock_GetRate: Return current rate
 *****************************************************************************/
int input_clock_GetRate( input_clock_t *cl )
{
    int i_rate;

    vlc_mutex_lock( &cl->lock );
    i_rate = cl->i_rate;
    vlc_mutex_unlock( &cl->lock );

    return i_rate;
}
476

477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499
int input_clock_GetState( input_clock_t *cl,
                          mtime_t *pi_stream_start, mtime_t *pi_system_start,
                          mtime_t *pi_stream_duration, mtime_t *pi_system_duration )
{
    vlc_mutex_lock( &cl->lock );

    if( !cl->b_has_reference )
    {
        vlc_mutex_unlock( &cl->lock );
        return VLC_EGENERIC;
    }

    *pi_stream_start = cl->ref.i_stream;
    *pi_system_start = cl->ref.i_system;

    *pi_stream_duration = cl->last.i_stream - cl->ref.i_stream;
    *pi_system_duration = cl->last.i_system - cl->ref.i_system;

    vlc_mutex_unlock( &cl->lock );

    return VLC_SUCCESS;
}

500
void input_clock_ChangeSystemOrigin( input_clock_t *cl, bool b_absolute, mtime_t i_system )
501 502 503 504
{
    vlc_mutex_lock( &cl->lock );

    assert( cl->b_has_reference );
505 506 507 508 509 510 511 512 513 514 515 516 517 518
    mtime_t i_offset;
    if( b_absolute )
    {
        i_offset = i_system - cl->ref.i_system - ClockGetTsOffset( cl );
    }
    else
    {
        if( !cl->b_has_external_clock )
        {
            cl->b_has_external_clock = true;
            cl->i_external_clock     = i_system;
        }
        i_offset = i_system - cl->i_external_clock;
    }
519 520 521 522 523 524 525

    cl->ref.i_system += i_offset;
    cl->last.i_system += i_offset;

    vlc_mutex_unlock( &cl->lock );
}

526
void input_clock_GetSystemOrigin( input_clock_t *cl, mtime_t *pi_system, mtime_t *pi_delay )
527 528 529 530 531
{
    vlc_mutex_lock( &cl->lock );

    assert( cl->b_has_reference );

532 533 534
    *pi_system = cl->ref.i_system;
    if( pi_delay )
        *pi_delay  = cl->i_pts_delay;
535 536 537 538

    vlc_mutex_unlock( &cl->lock );
}

539 540 541 542 543 544
#warning "input_clock_SetJitter needs more work"
void input_clock_SetJitter( input_clock_t *cl,
                            mtime_t i_pts_delay, int i_cr_average )
{
    vlc_mutex_lock( &cl->lock );

545 546
    /* Update late observations */
    const mtime_t i_delay_delta = i_pts_delay - cl->i_pts_delay;
547 548 549 550 551 552 553 554
    mtime_t pi_late[INPUT_CLOCK_LATE_COUNT];
    for( int i = 0; i < INPUT_CLOCK_LATE_COUNT; i++ )
        pi_late[i] = __MAX( cl->late.pi_value[(cl->late.i_index + 1 + i)%INPUT_CLOCK_LATE_COUNT] - i_delay_delta, 0 );

    for( int i = 0; i < INPUT_CLOCK_LATE_COUNT; i++ )
        cl->late.pi_value[i] = 0;
    cl->late.i_index = 0;

555 556
    for( int i = 0; i < INPUT_CLOCK_LATE_COUNT; i++ )
    {
557 558 559 560
        if( pi_late[i] <= 0 )
            continue;
        cl->late.pi_value[cl->late.i_index] = pi_late[i];
        cl->late.i_index = ( cl->late.i_index + 1 ) % INPUT_CLOCK_LATE_COUNT;
561 562
    }

563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578
    /* TODO always save the value, and when rebuffering use the new one if smaller
     * TODO when increasing -> force rebuffering
     */
    if( cl->i_pts_delay < i_pts_delay )
        cl->i_pts_delay = i_pts_delay;

    /* */
    if( i_cr_average < 10 )
        i_cr_average = 10;

    if( cl->drift.i_divider != i_cr_average )
        AvgRescale( &cl->drift, i_cr_average );

    vlc_mutex_unlock( &cl->lock );
}

579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600
mtime_t input_clock_GetJitter( input_clock_t *cl )
{
    vlc_mutex_lock( &cl->lock );

#if INPUT_CLOCK_LATE_COUNT != 3
#   error "unsupported INPUT_CLOCK_LATE_COUNT"
#endif
    /* Find the median of the last late values
     * It works pretty well at rejecting bad values
     *
     * XXX we only increase pts_delay over time, decreasing it is
     * not that easy if we want to be robust.
     */
    const mtime_t *p = cl->late.pi_value;
    mtime_t i_late_median = p[0] + p[1] + p[2] - __MIN(__MIN(p[0],p[1]),p[2]) - __MAX(__MAX(p[0],p[1]),p[2]);
    mtime_t i_pts_delay = cl->i_pts_delay ;

    vlc_mutex_unlock( &cl->lock );

    return i_pts_delay + i_late_median;
}

Laurent Aimar's avatar
Laurent Aimar committed
601 602 603
/*****************************************************************************
 * ClockStreamToSystem: converts a movie clock to system date
 *****************************************************************************/
Laurent Aimar's avatar
Laurent Aimar committed
604
static mtime_t ClockStreamToSystem( input_clock_t *cl, mtime_t i_stream )
Laurent Aimar's avatar
Laurent Aimar committed
605 606
{
    if( !cl->b_has_reference )
607
        return VLC_TS_INVALID;
Laurent Aimar's avatar
Laurent Aimar committed
608

Laurent Aimar's avatar
Laurent Aimar committed
609
    return ( i_stream - cl->ref.i_stream ) * cl->i_rate / INPUT_RATE_DEFAULT +
Laurent Aimar's avatar
Laurent Aimar committed
610 611 612 613 614 615 616 617 618 619 620 621
           cl->ref.i_system;
}

/*****************************************************************************
 * ClockSystemToStream: converts a system date to movie clock
 *****************************************************************************
 * Caution : a valid reference point is needed for this to operate.
 *****************************************************************************/
static mtime_t ClockSystemToStream( input_clock_t *cl, mtime_t i_system )
{
    assert( cl->b_has_reference );
    return ( i_system - cl->ref.i_system ) * INPUT_RATE_DEFAULT / cl->i_rate +
Laurent Aimar's avatar
Laurent Aimar committed
622
            cl->ref.i_stream;
Laurent Aimar's avatar
Laurent Aimar committed
623 624
}

625 626 627 628 629 630 631 632 633
/**
 * It returns timestamp display offset due to ref/last modfied on rate changes
 * It ensures that currently converted dates are not changed.
 */
static mtime_t ClockGetTsOffset( input_clock_t *cl )
{
    return cl->i_pts_delay * ( cl->i_rate - INPUT_RATE_DEFAULT ) / INPUT_RATE_DEFAULT;
}

634 635 636
/*****************************************************************************
 * Long term average helpers
 *****************************************************************************/
637
static void AvgInit( average_t *p_avg, int i_divider )
638 639
{
    p_avg->i_divider = i_divider;
640
    AvgReset( p_avg );
641
}
642
static void AvgClean( average_t *p_avg )
643 644 645
{
    VLC_UNUSED(p_avg);
}
646
static void AvgReset( average_t *p_avg )
647 648 649 650 651
{
    p_avg->i_value = 0;
    p_avg->i_residue = 0;
    p_avg->i_count = 0;
}
652
static void AvgUpdate( average_t *p_avg, mtime_t i_value )
653 654 655 656 657 658 659 660 661 662 663
{
    const int i_f0 = __MIN( p_avg->i_divider - 1, p_avg->i_count );
    const int i_f1 = p_avg->i_divider - i_f0;

    const mtime_t i_tmp = i_f0 * p_avg->i_value + i_f1 * i_value + p_avg->i_residue;

    p_avg->i_value   = i_tmp / p_avg->i_divider;
    p_avg->i_residue = i_tmp % p_avg->i_divider;

    p_avg->i_count++;
}
664
static mtime_t AvgGet( average_t *p_avg )
665 666 667
{
    return p_avg->i_value;
}
668 669 670
static void AvgRescale( average_t *p_avg, int i_divider )
{
    const mtime_t i_tmp = p_avg->i_value * p_avg->i_divider + p_avg->i_residue;
Laurent Aimar's avatar
Laurent Aimar committed
671

672 673 674 675
    p_avg->i_divider = i_divider;
    p_avg->i_value   = i_tmp / p_avg->i_divider;
    p_avg->i_residue = i_tmp % p_avg->i_divider;
}