async_queue.cpp
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:4k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * async_queue.cpp
  3.  *****************************************************************************
  4.  * Copyright (C) 2003 the VideoLAN team
  5.  * $Id: de19dc7b3710fbb4921a1a8bd0cefac36021e98b $
  6.  *
  7.  * Authors: Cyril Deguet     <asmax@via.ecp.fr>
  8.  *          Olivier Teulière <ipkiss@via.ecp.fr>
  9.  *
  10.  * This program is free software; you can redistribute it and/or modify
  11.  * it under the terms of the GNU General Public License as published by
  12.  * the Free Software Foundation; either version 2 of the License, or
  13.  * (at your option) any later version.
  14.  *
  15.  * This program is distributed in the hope that it will be useful,
  16.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  18.  * GNU General Public License for more details.
  19.  *
  20.  * You should have received a copy of the GNU General Public License
  21.  * along with this program; if not, write to the Free Software
  22.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  23.  *****************************************************************************/
  24. #include "async_queue.hpp"
  25. #include "../src/os_factory.hpp"
  26. #include "../src/os_timer.hpp"
  27. AsyncQueue::AsyncQueue( intf_thread_t *pIntf ): SkinObject( pIntf ),
  28.     m_cmdFlush( this )
  29. {
  30.     // Initialize the mutex
  31.     vlc_mutex_init( &m_lock );
  32.     // Create a timer
  33.     OSFactory *pOsFactory = OSFactory::instance( pIntf );
  34.     m_pTimer = pOsFactory->createOSTimer( m_cmdFlush );
  35.     // Flush the queue every 10 ms
  36.     m_pTimer->start( 10, false );
  37. }
  38. AsyncQueue::~AsyncQueue()
  39. {
  40.     delete( m_pTimer );
  41.     vlc_mutex_destroy( &m_lock );
  42. }
  43. AsyncQueue *AsyncQueue::instance( intf_thread_t *pIntf )
  44. {
  45.     if( ! pIntf->p_sys->p_queue )
  46.     {
  47.         AsyncQueue *pQueue;
  48.         pQueue = new AsyncQueue( pIntf );
  49.         if( pQueue )
  50.         {
  51.              // Initialization succeeded
  52.              pIntf->p_sys->p_queue = pQueue;
  53.         }
  54.      }
  55.      return pIntf->p_sys->p_queue;
  56. }
  57. void AsyncQueue::destroy( intf_thread_t *pIntf )
  58. {
  59.     if( pIntf->p_sys->p_queue )
  60.     {
  61.         delete pIntf->p_sys->p_queue;
  62.         pIntf->p_sys->p_queue = NULL;
  63.     }
  64. }
  65. void AsyncQueue::push( const CmdGenericPtr &rcCommand, bool removePrev )
  66. {
  67.     if( removePrev )
  68.     {
  69.         // Remove the commands of the same type
  70.         remove( rcCommand.get()->getType(), rcCommand );
  71.     }
  72.     m_cmdList.push_back( rcCommand );
  73. }
  74. void AsyncQueue::remove( const string &rType, const CmdGenericPtr &rcCommand )
  75. {
  76.     vlc_mutex_lock( &m_lock );
  77.     list<CmdGenericPtr>::iterator it;
  78.     for( it = m_cmdList.begin(); it != m_cmdList.end(); it++ )
  79.     {
  80.         // Remove the command if it is of the given type
  81.         if( (*it).get()->getType() == rType )
  82.         {
  83.             // Maybe the command wants to check if it must really be
  84.             // removed
  85.             if( rcCommand.get()->checkRemove( (*it).get() ) == true )
  86.             {
  87.                 list<CmdGenericPtr>::iterator itNew = it;
  88.                 itNew++;
  89.                 m_cmdList.erase( it );
  90.                 it = itNew;
  91.             }
  92.         }
  93.     }
  94.     vlc_mutex_unlock( &m_lock );
  95. }
  96. void AsyncQueue::flush()
  97. {
  98.     while (true)
  99.     {
  100.         vlc_mutex_lock( &m_lock );
  101.         if( m_cmdList.size() > 0 )
  102.         {
  103.             // Pop the first command from the queue
  104.             CmdGenericPtr cCommand = m_cmdList.front();
  105.             m_cmdList.pop_front();
  106.             // Unlock the mutex to avoid deadlocks if another thread wants to
  107.             // enqueue/remove a command while this one is processed
  108.             vlc_mutex_unlock( &m_lock );
  109.             // Execute the command
  110.             cCommand.get()->execute();
  111.         }
  112.         else
  113.         {
  114.             vlc_mutex_unlock( &m_lock );
  115.             break;
  116.         }
  117.     }
  118. }
  119. void AsyncQueue::CmdFlush::execute()
  120. {
  121.     // Flush the queue
  122.     m_pParent->flush();
  123. }