FakeESOut.cpp 11.5 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 42 43 44 45
{
    real_es_out = es;
    fakeesout = new es_out_t;

    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;

46 47
    commandsqueue = queue;

48 49
    timestamps_offset = 0;
    extrainfo = NULL;
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
}

void FakeESOut::setTimestampOffset(mtime_t offset)
{
69
    vlc_mutex_lock(&lock);
70
    timestamps_offset = offset;
71
    vlc_mutex_unlock(&lock);
72 73 74 75
}

void FakeESOut::setExtraInfoProvider( ExtraFMTInfoInterface *extra )
{
76
    vlc_mutex_lock(&lock);
77
    extrainfo = extra;
78
    vlc_mutex_unlock(&lock);
79 80
}

81
FakeESOutID * FakeESOut::createNewID( const es_format_t *p_fmt )
82 83 84 85
{
    es_format_t fmtcopy;
    es_format_Init( &fmtcopy, 0, 0 );
    es_format_Copy( &fmtcopy, p_fmt );
86
    fmtcopy.i_group = 0; /* Always ignore group for adaptive */
87
    fmtcopy.i_id = -1;
88

89 90
    vlc_mutex_lock(&lock);

91 92 93
    if( extrainfo )
        extrainfo->fillExtraFMTInfo( &fmtcopy );

94 95 96 97
    FakeESOutID *es_id = new (std::nothrow) FakeESOutID( this, &fmtcopy );
    if(likely(es_id))
        fakeesidlist.push_back( es_id );

98 99
    vlc_mutex_unlock(&lock);

100 101 102 103 104 105 106 107 108 109
    es_format_Clean( &fmtcopy );

    return es_id;
}

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

110 111
    vlc_mutex_lock(&lock);

112
    bool b_select = false;
113 114
    for( it=recycle_candidates.begin(); it!=recycle_candidates.end(); ++it )
    {
115 116
        FakeESOutID *cand = *it;
        if ( cand->isCompatible( es_id ) )
117
        {
118 119 120
            realid = cand->realESID();
            cand->setRealESID( NULL );
            delete *it;
121 122 123
            recycle_candidates.erase( it );
            break;
        }
124
        else if( cand->getFmt()->i_cat == es_id->getFmt()->i_cat && cand->realESID() )
125 126 127 128 129
        {
            /* 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 );
130
            break;
131
        }
132 133
    }

134
    if( !realid )
135
    {
136
        realid = es_out_Add( real_es_out, es_id->getFmt() );
137 138 139
        if( b_select )
            es_out_Control( real_es_out, ES_OUT_SET_ES_STATE, realid, b_select );
    }
140

141
    es_id->setRealESID( realid );
142 143

    vlc_mutex_unlock(&lock);
144 145 146 147
}

mtime_t FakeESOut::getTimestampOffset() const
{
148 149 150 151
    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;
152 153 154 155 156 157
}

size_t FakeESOut::esCount() const
{
    size_t i_count = 0;
    std::list<FakeESOutID *>::const_iterator it;
158
    vlc_mutex_lock(const_cast<vlc_mutex_t *>(&lock));
159 160 161
    for( it=fakeesidlist.begin(); it!=fakeesidlist.end(); ++it )
        if( (*it)->realESID() )
            i_count++;
162
    vlc_mutex_unlock(const_cast<vlc_mutex_t *>(&lock));
163 164
    return i_count;
}
165 166 167

void FakeESOut::schedulePCRReset()
{
168
    AbstractCommand *command = commandsqueue->factory()->creatEsOutControlResetPCRCommand();
169
    if( likely(command) )
170
        commandsqueue->Schedule( command );
171 172 173 174 175
}

void FakeESOut::scheduleAllForDeletion()
{
    std::list<FakeESOutID *>::const_iterator it;
176
    vlc_mutex_lock(&lock);
177 178 179 180 181
    for( it=fakeesidlist.begin(); it!=fakeesidlist.end(); ++it )
    {
        FakeESOutID *es_id = *it;
        if(!es_id->scheduledForDeletion())
        {
182
            AbstractCommand *command = commandsqueue->factory()->createEsOutDelCommand( es_id );
183 184
            if( likely(command) )
            {
185
                commandsqueue->Schedule( command );
186 187 188 189
                es_id->setScheduledForDeletion();
            }
        }
    }
190
    vlc_mutex_unlock(&lock);
191 192
}

193 194
void FakeESOut::recycleAll()
{
195
    /* Only used when demux is killed and commands queue is cancelled */
196 197
    commandsqueue->Abort( true );
    assert(commandsqueue->isEmpty());
198
    vlc_mutex_lock(&lock);
199
    recycle_candidates.splice( recycle_candidates.end(), fakeesidlist );
200
    vlc_mutex_unlock(&lock);
201 202 203 204
}

void FakeESOut::gc()
{
205
    vlc_mutex_lock(&lock);
206
    if( recycle_candidates.empty() )
207 208
    {
        vlc_mutex_unlock(&lock);
209
        return;
210
    }
211 212 213 214 215

    std::list<FakeESOutID *>::iterator it;
    for( it=recycle_candidates.begin(); it!=recycle_candidates.end(); ++it )
    {
        if( (*it)->realESID() )
216 217
        {
            es_out_Control( real_es_out, ES_OUT_SET_ES_STATE, (*it)->realESID(), false );
218
            es_out_Del( real_es_out, (*it)->realESID() );
219
        }
220 221 222
        delete *it;
    }
    recycle_candidates.clear();
223
    vlc_mutex_unlock(&lock);
224 225 226 227 228 229
}

bool FakeESOut::hasSelectedEs() const
{
    bool b_selected = false;
    std::list<FakeESOutID *>::const_iterator it;
230
    vlc_mutex_lock(const_cast<vlc_mutex_t *>(&lock));
231 232 233 234 235 236
    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 );
    }
237
    vlc_mutex_unlock(const_cast<vlc_mutex_t *>(&lock));
238 239 240
    return b_selected;
}

241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259
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;
}

260 261
bool FakeESOut::restarting() const
{
262 263 264 265
    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;
266 267 268 269
}

void FakeESOut::recycle( FakeESOutID *id )
{
270
    vlc_mutex_lock(&lock);
271 272
    fakeesidlist.remove( id );
    recycle_candidates.push_back( id );
273
    vlc_mutex_unlock(&lock);
274 275 276 277 278 279 280
}

/* 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;
281 282 283 284

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

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

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 );
309
    assert(!es_id->scheduledForDeletion());
310 311 312 313 314 315 316
    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;
    }
317
    AbstractCommand *command = me->commandsqueue->factory()->createEsOutSendCommand( es_id, p_block );
318 319
    if( likely(command) )
    {
320
        me->commandsqueue->Schedule( command );
321 322 323 324 325 326 327 328 329
        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 );
330
    AbstractCommand *command = me->commandsqueue->factory()->createEsOutDelCommand( es_id );
331
    if( likely(command) )
332 333
    {
        es_id->setScheduledForDeletion();
334
        me->commandsqueue->Schedule( command );
335
    }
336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353
}

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 )
                i_group = static_cast<int>(va_arg( args, int ));
            else
                i_group = 0;
            int64_t  pcr = static_cast<int64_t>(va_arg( args, int64_t ));
            pcr += me->getTimestampOffset();
354
            AbstractCommand *command = me->commandsqueue->factory()->createEsOutControlPCRCommand( i_group, pcr );
355 356
            if( likely(command) )
            {
357
                me->commandsqueue->Schedule( command );
358 359 360 361 362
                return VLC_SUCCESS;
            }
        }
        break;

363 364 365 366 367 368 369 370 371 372 373 374 375
        case ES_OUT_SET_GROUP_META:
        {
            static_cast<void>(va_arg( args, int )); /* ignore group */
            const vlc_meta_t *p_meta = static_cast<const vlc_meta_t *>(va_arg( args, const vlc_meta_t * ));
            AbstractCommand *command = me->commandsqueue->factory()->createEsOutMetaCommand( -1, p_meta );
            if( likely(command) )
            {
                me->commandsqueue->Schedule( command );
                return VLC_SUCCESS;
            }
        }
        break;

376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396
        /* For others, we don't have the delorean, so always lie */
        case ES_OUT_GET_ES_STATE:
        {
            static_cast<void*>(va_arg( args, es_out_id_t * ));
            bool *pb = static_cast<bool *>(va_arg( args, bool * ));
            *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;
397
    AbstractCommand *command = me->commandsqueue->factory()->createEsOutDestroyCommand();
398
    if( likely(command) )
399
        me->commandsqueue->Schedule( command );
400 401
}
/* !Static callbacks */