FakeESOut.cpp 12.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/*
 * FakeESOut.cpp
 *****************************************************************************
 * Copyright © 2014-2015 VideoLAN and VLC Authors
 *
 * 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
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Lesser General Public License for more details.
 *
 * 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.
 *****************************************************************************/
20 21 22 23 24

#ifdef HAVE_CONFIG_H
# include "config.h"
#endif

25 26
#include "FakeESOut.hpp"
#include "FakeESOutID.hpp"
27
#include "CommandsQueue.hpp"
28 29
#include <vlc_es_out.h>
#include <vlc_block.h>
30
#include <cassert>
31

32
using namespace adaptive;
33

34
FakeESOut::FakeESOut( es_out_t *es, CommandsQueue *queue )
35 36 37 38 39 40 41
    : real_es_out( es )
    , extrainfo( NULL )
    , commandsqueue( queue )
    , fakeesout( new es_out_t )
    , timestamps_offset( 0 )
    , timestamps_expected( 0 )
    , timestamps_check_done( false )
42 43 44 45 46 47 48 49
{
    fakeesout->pf_add = esOutAdd_Callback;
    fakeesout->pf_control = esOutControl_Callback;
    fakeesout->pf_del = esOutDel_Callback;
    fakeesout->pf_destroy = esOutDestroy_Callback;
    fakeesout->pf_send = esOutSend_Callback;
    fakeesout->p_sys = (es_out_sys_t*) this;

50
    vlc_mutex_init(&lock);
51 52 53 54 55 56 57 58 59 60 61 62
}

es_out_t * FakeESOut::getEsOut()
{
    return fakeesout;
}

FakeESOut::~FakeESOut()
{
    recycleAll();
    gc();

63
    delete fakeesout;
64
    vlc_mutex_destroy(&lock);
65 66
}

67 68 69 70 71 72 73 74 75
void FakeESOut::setExpectedTimestampOffset(mtime_t offset)
{
    vlc_mutex_lock(&lock);
    timestamps_offset = 0;
    timestamps_expected = offset;
    timestamps_check_done = false;
    vlc_mutex_unlock(&lock);
}

76 77
void FakeESOut::setTimestampOffset(mtime_t offset)
{
78
    vlc_mutex_lock(&lock);
79
    timestamps_offset = offset;
80
    timestamps_check_done = true;
81
    vlc_mutex_unlock(&lock);
82 83 84 85
}

void FakeESOut::setExtraInfoProvider( ExtraFMTInfoInterface *extra )
{
86
    vlc_mutex_lock(&lock);
87
    extrainfo = extra;
88
    vlc_mutex_unlock(&lock);
89 90
}

91
FakeESOutID * FakeESOut::createNewID( const es_format_t *p_fmt )
92 93 94 95
{
    es_format_t fmtcopy;
    es_format_Init( &fmtcopy, 0, 0 );
    es_format_Copy( &fmtcopy, p_fmt );
96
    fmtcopy.i_group = 0; /* Always ignore group for adaptive */
97
    fmtcopy.i_id = -1;
98

99 100
    vlc_mutex_lock(&lock);

101 102 103
    if( extrainfo )
        extrainfo->fillExtraFMTInfo( &fmtcopy );

104 105 106 107
    FakeESOutID *es_id = new (std::nothrow) FakeESOutID( this, &fmtcopy );
    if(likely(es_id))
        fakeesidlist.push_back( es_id );

108 109
    vlc_mutex_unlock(&lock);

110 111 112 113 114 115 116 117 118 119
    es_format_Clean( &fmtcopy );

    return es_id;
}

void FakeESOut::createOrRecycleRealEsID( FakeESOutID *es_id )
{
    std::list<FakeESOutID *>::iterator it;
    es_out_id_t *realid = NULL;

120 121
    vlc_mutex_lock(&lock);

122
    bool b_select = false;
123 124
    for( it=recycle_candidates.begin(); it!=recycle_candidates.end(); ++it )
    {
125 126
        FakeESOutID *cand = *it;
        if ( cand->isCompatible( es_id ) )
127
        {
128 129 130
            realid = cand->realESID();
            cand->setRealESID( NULL );
            delete *it;
131 132 133
            recycle_candidates.erase( it );
            break;
        }
134
        else if( cand->getFmt()->i_cat == es_id->getFmt()->i_cat && cand->realESID() )
135 136 137 138 139
        {
            /* We need to enforce same selection when not reused
               Otherwise the es will select any other compatible track
               and will end this in a activate/select loop when reactivating a track */
            es_out_Control( real_es_out, ES_OUT_GET_ES_STATE, cand->realESID(), &b_select );
140
            break;
141
        }
142 143
    }

144
    if( !realid )
145
    {
146
        realid = es_out_Add( real_es_out, es_id->getFmt() );
147 148 149
        if( b_select )
            es_out_Control( real_es_out, ES_OUT_SET_ES_STATE, realid, b_select );
    }
150

151
    es_id->setRealESID( realid );
152 153

    vlc_mutex_unlock(&lock);
154 155 156 157
}

mtime_t FakeESOut::getTimestampOffset() const
{
158 159 160 161
    vlc_mutex_lock(const_cast<vlc_mutex_t *>(&lock));
    mtime_t time = timestamps_offset;
    vlc_mutex_unlock(const_cast<vlc_mutex_t *>(&lock));
    return time;
162 163 164 165 166 167
}

size_t FakeESOut::esCount() const
{
    size_t i_count = 0;
    std::list<FakeESOutID *>::const_iterator it;
168
    vlc_mutex_lock(const_cast<vlc_mutex_t *>(&lock));
169 170 171
    for( it=fakeesidlist.begin(); it!=fakeesidlist.end(); ++it )
        if( (*it)->realESID() )
            i_count++;
172
    vlc_mutex_unlock(const_cast<vlc_mutex_t *>(&lock));
173 174
    return i_count;
}
175 176 177

void FakeESOut::schedulePCRReset()
{
178
    AbstractCommand *command = commandsqueue->factory()->creatEsOutControlResetPCRCommand();
179
    if( likely(command) )
180
        commandsqueue->Schedule( command );
181 182 183 184 185
}

void FakeESOut::scheduleAllForDeletion()
{
    std::list<FakeESOutID *>::const_iterator it;
186
    vlc_mutex_lock(&lock);
187 188 189 190 191
    for( it=fakeesidlist.begin(); it!=fakeesidlist.end(); ++it )
    {
        FakeESOutID *es_id = *it;
        if(!es_id->scheduledForDeletion())
        {
192
            AbstractCommand *command = commandsqueue->factory()->createEsOutDelCommand( es_id );
193 194
            if( likely(command) )
            {
195
                commandsqueue->Schedule( command );
196 197 198 199
                es_id->setScheduledForDeletion();
            }
        }
    }
200
    vlc_mutex_unlock(&lock);
201 202
}

203 204
void FakeESOut::recycleAll()
{
205
    /* Only used when demux is killed and commands queue is cancelled */
206 207
    commandsqueue->Abort( true );
    assert(commandsqueue->isEmpty());
208
    vlc_mutex_lock(&lock);
209
    recycle_candidates.splice( recycle_candidates.end(), fakeesidlist );
210
    vlc_mutex_unlock(&lock);
211 212 213 214
}

void FakeESOut::gc()
{
215
    vlc_mutex_lock(&lock);
216
    if( recycle_candidates.empty() )
217 218
    {
        vlc_mutex_unlock(&lock);
219
        return;
220
    }
221 222 223 224 225

    std::list<FakeESOutID *>::iterator it;
    for( it=recycle_candidates.begin(); it!=recycle_candidates.end(); ++it )
    {
        if( (*it)->realESID() )
226 227
        {
            es_out_Control( real_es_out, ES_OUT_SET_ES_STATE, (*it)->realESID(), false );
228
            es_out_Del( real_es_out, (*it)->realESID() );
229
        }
230 231 232
        delete *it;
    }
    recycle_candidates.clear();
233
    vlc_mutex_unlock(&lock);
234 235 236 237 238 239
}

bool FakeESOut::hasSelectedEs() const
{
    bool b_selected = false;
    std::list<FakeESOutID *>::const_iterator it;
240
    vlc_mutex_lock(const_cast<vlc_mutex_t *>(&lock));
241 242 243 244 245 246
    for( it=fakeesidlist.begin(); it!=fakeesidlist.end() && !b_selected; ++it )
    {
        FakeESOutID *esID = *it;
        if( esID->realESID() )
            es_out_Control( real_es_out, ES_OUT_GET_ES_STATE, esID->realESID(), &b_selected );
    }
247
    vlc_mutex_unlock(const_cast<vlc_mutex_t *>(&lock));
248 249 250
    return b_selected;
}

251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269
bool FakeESOut::drain()
{
    bool b_drained = true;
    std::list<FakeESOutID *>::const_iterator it;
    vlc_mutex_lock(&lock);
    for( it=fakeesidlist.begin(); it!=fakeesidlist.end(); ++it )
    {
        FakeESOutID *esID = *it;
        if( esID->realESID() )
        {
            bool b_empty;
            es_out_Control( real_es_out, ES_OUT_GET_EMPTY, &b_empty );
            b_drained &= b_empty;
        }
    }
    vlc_mutex_unlock(&lock);
    return b_drained;
}

270 271
bool FakeESOut::restarting() const
{
272 273 274 275
    vlc_mutex_lock(const_cast<vlc_mutex_t *>(&lock));
    bool b = !recycle_candidates.empty();
    vlc_mutex_unlock(const_cast<vlc_mutex_t *>(&lock));
    return b;
276 277 278 279
}

void FakeESOut::recycle( FakeESOutID *id )
{
280
    vlc_mutex_lock(&lock);
281 282
    fakeesidlist.remove( id );
    recycle_candidates.push_back( id );
283
    vlc_mutex_unlock(&lock);
284 285 286 287 288 289 290
}

/* Static callbacks */
/* Always pass Fake ES ID to slave demuxes, it is just an opaque struct to them */
es_out_id_t * FakeESOut::esOutAdd_Callback(es_out_t *fakees, const es_format_t *p_fmt)
{
    FakeESOut *me = (FakeESOut *) fakees->p_sys;
291 292 293 294

    if( p_fmt->i_cat != VIDEO_ES && p_fmt->i_cat != AUDIO_ES && p_fmt->i_cat != SPU_ES )
        return NULL;

295 296
    /* Feed the slave demux/stream_Demux with FakeESOutID struct,
     * we'll create real ES later on main demux on execution */
297
    FakeESOutID *es_id = me->createNewID( p_fmt );
298 299
    if( likely(es_id) )
    {
300
        assert(!es_id->scheduledForDeletion());
301
        AbstractCommand *command = me->commandsqueue->factory()->createEsOutAddCommand( es_id );
302 303
        if( likely(command) )
        {
304
            me->commandsqueue->Schedule( command );
305 306 307 308 309 310 311 312 313 314
            return reinterpret_cast<es_out_id_t *>(es_id);
        }
        else
        {
            delete es_id;
        }
    }
    return NULL;
}

315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
void FakeESOut::checkTimestampsStart(mtime_t i_start)
{
    if( i_start == VLC_TS_INVALID )
        return;

    vlc_mutex_lock(&lock);
    if( !timestamps_check_done )
    {
        if( i_start < CLOCK_FREQ ) /* Starts 0 */
            timestamps_offset = timestamps_expected;
        timestamps_check_done = true;
    }
    vlc_mutex_unlock(&lock);
}

330 331 332 333
int FakeESOut::esOutSend_Callback(es_out_t *fakees, es_out_id_t *p_es, block_t *p_block)
{
    FakeESOut *me = (FakeESOut *) fakees->p_sys;
    FakeESOutID *es_id = reinterpret_cast<FakeESOutID *>( p_es );
334
    assert(!es_id->scheduledForDeletion());
335 336 337

    me->checkTimestampsStart( p_block->i_dts );

338 339 340 341 342 343 344
    mtime_t offset = me->getTimestampOffset();
    if( p_block->i_dts > VLC_TS_INVALID )
    {
        p_block->i_dts += offset;
        if( p_block->i_pts > VLC_TS_INVALID )
                p_block->i_pts += offset;
    }
345
    AbstractCommand *command = me->commandsqueue->factory()->createEsOutSendCommand( es_id, p_block );
346 347
    if( likely(command) )
    {
348
        me->commandsqueue->Schedule( command );
349 350 351 352 353 354 355 356 357
        return VLC_SUCCESS;
    }
    return VLC_EGENERIC;
}

void FakeESOut::esOutDel_Callback(es_out_t *fakees, es_out_id_t *p_es)
{
    FakeESOut *me = (FakeESOut *) fakees->p_sys;
    FakeESOutID *es_id = reinterpret_cast<FakeESOutID *>( p_es );
358
    AbstractCommand *command = me->commandsqueue->factory()->createEsOutDelCommand( es_id );
359
    if( likely(command) )
360 361
    {
        es_id->setScheduledForDeletion();
362
        me->commandsqueue->Schedule( command );
363
    }
364 365 366 367 368 369 370 371 372 373 374 375 376
}

int FakeESOut::esOutControl_Callback(es_out_t *fakees, int i_query, va_list args)
{
    FakeESOut *me = (FakeESOut *) fakees->p_sys;

    switch( i_query )
    {
        case ES_OUT_SET_PCR:
        case ES_OUT_SET_GROUP_PCR:
        {
            int i_group;
            if( i_query == ES_OUT_SET_GROUP_PCR )
377
                i_group = va_arg( args, int );
378 379
            else
                i_group = 0;
380
            int64_t  pcr = va_arg( args, int64_t );
381
            pcr += me->getTimestampOffset();
382
            AbstractCommand *command = me->commandsqueue->factory()->createEsOutControlPCRCommand( i_group, pcr );
383 384
            if( likely(command) )
            {
385
                me->commandsqueue->Schedule( command );
386 387 388 389 390
                return VLC_SUCCESS;
            }
        }
        break;

391 392 393
        case ES_OUT_SET_GROUP_META:
        {
            static_cast<void>(va_arg( args, int )); /* ignore group */
394
            const vlc_meta_t *p_meta = va_arg( args, const vlc_meta_t * );
395 396 397 398 399 400 401 402 403
            AbstractCommand *command = me->commandsqueue->factory()->createEsOutMetaCommand( -1, p_meta );
            if( likely(command) )
            {
                me->commandsqueue->Schedule( command );
                return VLC_SUCCESS;
            }
        }
        break;

404 405 406
        /* For others, we don't have the delorean, so always lie */
        case ES_OUT_GET_ES_STATE:
        {
407
            static_cast<void>(va_arg( args, es_out_id_t * ));
408
            bool *pb = va_arg( args, bool * );
409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424
            *pb = true;
            // ft
        }
        case ES_OUT_SET_ES:
        case ES_OUT_SET_ES_DEFAULT:
        case ES_OUT_SET_ES_STATE:
            return VLC_SUCCESS;

    }

    return VLC_EGENERIC;
}

void FakeESOut::esOutDestroy_Callback(es_out_t *fakees)
{
    FakeESOut *me = (FakeESOut *) fakees->p_sys;
425
    AbstractCommand *command = me->commandsqueue->factory()->createEsOutDestroyCommand();
426
    if( likely(command) )
427
        me->commandsqueue->Schedule( command );
428 429
}
/* !Static callbacks */