x11_timer.cpp
上传用户:riyaled888
上传日期:2009-03-27
资源大小:7338k
文件大小:4k
源码类别:

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * x11_timer.cpp
  3.  *****************************************************************************
  4.  * Copyright (C) 2003 VideoLAN
  5.  * $Id: x11_timer.cpp 6961 2004-03-05 17:34:23Z sam $
  6.  *
  7.  * Authors: Cyril Deguet     <asmax@via.ecp.fr>
  8.  *          Olivier Teuli鑢e <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., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  23.  *****************************************************************************/
  24. #ifdef X11_SKINS
  25. #include <unistd.h>
  26. #include <fcntl.h>
  27. #include "x11_timer.hpp"
  28. #include "x11_factory.hpp"
  29. X11Timer::X11Timer( intf_thread_t *pIntf, const Callback &rCallback ):
  30.     OSTimer( pIntf ), m_callback( rCallback )
  31. {
  32.     // Get the instance of timer loop
  33.     X11Factory *m_pOsFactory = (X11Factory*)(OSFactory::instance( pIntf ) );
  34.     m_pTimerLoop = m_pOsFactory->getTimerLoop();
  35. }
  36. X11Timer::~X11Timer()
  37. {
  38.     stop();
  39. }
  40. void X11Timer::start( int delay, bool oneShot )
  41. {
  42.     m_interval = 1000 * delay;
  43.     m_oneShot = oneShot;
  44.     m_nextDate = mdate() + m_interval;
  45.     m_pTimerLoop->addTimer( *this );
  46. }
  47. void X11Timer::stop()
  48. {
  49.     m_pTimerLoop->removeTimer( *this );
  50. }
  51. mtime_t X11Timer::getNextDate() const
  52. {
  53.     return m_nextDate;
  54. }
  55. bool X11Timer::execute()
  56. {
  57.     m_nextDate += m_interval;
  58.     // Execute the callback
  59.     (*(m_callback.getFunc()))( m_callback.getObj() );
  60.     return !m_oneShot;
  61. }
  62. X11TimerLoop::X11TimerLoop( intf_thread_t *pIntf, int connectionNumber ):
  63.     SkinObject( pIntf ), m_connectionNumber( connectionNumber )
  64. {
  65. }
  66. X11TimerLoop::~X11TimerLoop()
  67. {
  68. }
  69. void X11TimerLoop::addTimer( X11Timer &rTimer )
  70. {
  71.     m_timers.push_back( &rTimer );
  72. }
  73. void X11TimerLoop::removeTimer( X11Timer &rTimer )
  74. {
  75.     m_timers.remove( &rTimer );
  76. }
  77. void X11TimerLoop::waitNextTimer()
  78. {
  79.     mtime_t curDate = mdate();
  80.     mtime_t nextDate = LAST_MDATE;
  81.     X11Timer *nextTimer = NULL;
  82.     // Find the next timer to execute
  83.     list<X11Timer*>::const_iterator timer;
  84.     for( timer = m_timers.begin(); timer != m_timers.end(); timer++ )
  85.     {
  86.         mtime_t timerDate = (*timer)->getNextDate();
  87.         if( timerDate < nextDate )
  88.         {
  89.             nextTimer = *timer;
  90.             nextDate = timerDate;
  91.         }
  92.     }
  93.     if( nextTimer == NULL )
  94.     {
  95.         this->sleep( 1000 );
  96.     }
  97.     else
  98.     {
  99.         if( nextDate > curDate )
  100.         {
  101.             if( this->sleep( (nextDate - curDate ) / 1000 ) )
  102.             {
  103.                 // The sleep has been interrupted: stop here
  104.                 return;
  105.             }
  106.         }
  107.         // Execute the timer callback
  108.         if( ! nextTimer->execute() )
  109.         {
  110.             // Remove the timer if execute() returned false
  111.             m_timers.remove( nextTimer );
  112.         }
  113.     }
  114. }
  115. bool X11TimerLoop::sleep( int delay )
  116. {
  117.     // Timeout delay
  118.     struct timeval tv;
  119.     tv.tv_sec = delay / 1000;
  120.     tv.tv_usec = 1000 * (delay % 1000);
  121.     // FD set for select()
  122.     fd_set rfds;
  123.     FD_ZERO( &rfds );
  124.     FD_SET( m_connectionNumber, &rfds );
  125.     // Wait for an X11 event, or timeout
  126.     int num = select( m_connectionNumber + 1, &rfds, NULL, NULL, &tv );
  127.     return ( num > 0 );
  128. }
  129. #endif