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

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * x11_timer.cpp
  3.  *****************************************************************************
  4.  * Copyright (C) 2003 the VideoLAN team
  5.  * $Id: 1719748c03c7d5b8eb44a7ca974d07e2389a76f9 $
  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. #ifdef X11_SKINS
  25. #include <unistd.h>
  26. #include <fcntl.h>
  27. #include <poll.h>
  28. #include "x11_timer.hpp"
  29. #include "x11_factory.hpp"
  30. #include "../commands/cmd_generic.hpp"
  31. X11Timer::X11Timer( intf_thread_t *pIntf, CmdGeneric &rCmd ):
  32.     OSTimer( pIntf ), m_rCommand( rCmd )
  33. {
  34.     // Get the instance of timer loop
  35.     X11Factory *m_pOsFactory = (X11Factory*)(OSFactory::instance( pIntf ) );
  36.     m_pTimerLoop = m_pOsFactory->getTimerLoop();
  37. }
  38. X11Timer::~X11Timer()
  39. {
  40.     stop();
  41. }
  42. void X11Timer::start( int delay, bool oneShot )
  43. {
  44.     m_interval = 1000 * delay;
  45.     m_oneShot = oneShot;
  46.     m_nextDate = mdate() + m_interval;
  47.     m_pTimerLoop->addTimer( *this );
  48. }
  49. void X11Timer::stop()
  50. {
  51.     m_pTimerLoop->removeTimer( *this );
  52. }
  53. mtime_t X11Timer::getNextDate() const
  54. {
  55.     return m_nextDate;
  56. }
  57. bool X11Timer::execute()
  58. {
  59.     m_nextDate += m_interval;
  60.     // Execute the callback
  61.     m_rCommand.execute();
  62.     return !m_oneShot;
  63. }
  64. X11TimerLoop::X11TimerLoop( intf_thread_t *pIntf, int connectionNumber ):
  65.     SkinObject( pIntf ), m_connectionNumber( connectionNumber )
  66. {
  67. }
  68. X11TimerLoop::~X11TimerLoop()
  69. {
  70. }
  71. void X11TimerLoop::addTimer( X11Timer &rTimer )
  72. {
  73.     m_timers.push_back( &rTimer );
  74. }
  75. void X11TimerLoop::removeTimer( X11Timer &rTimer )
  76. {
  77.     m_timers.remove( &rTimer );
  78. }
  79. void X11TimerLoop::waitNextTimer()
  80. {
  81.     mtime_t curDate = mdate();
  82.     mtime_t nextDate = LAST_MDATE;
  83.     X11Timer *nextTimer = NULL;
  84.     // Find the next timer to execute
  85.     list<X11Timer*>::const_iterator timer;
  86.     for( timer = m_timers.begin(); timer != m_timers.end(); timer++ )
  87.     {
  88.         mtime_t timerDate = (*timer)->getNextDate();
  89.         if( timerDate < nextDate )
  90.         {
  91.             nextTimer = *timer;
  92.             nextDate = timerDate;
  93.         }
  94.     }
  95.     if( nextTimer == NULL )
  96.     {
  97.         this->sleep( 1000 );
  98.     }
  99.     else
  100.     {
  101.         if( nextDate > curDate )
  102.         {
  103.             if( this->sleep( (nextDate - curDate ) / 1000 ) )
  104.             {
  105.                 // The sleep has been interrupted: stop here
  106.                 return;
  107.             }
  108.         }
  109.         // Execute the timer callback
  110.         if( ! nextTimer->execute() )
  111.         {
  112.             // Remove the timer if execute() returned false
  113.             m_timers.remove( nextTimer );
  114.         }
  115.     }
  116. }
  117. bool X11TimerLoop::sleep( int delay )
  118. {
  119.     struct pollfd ufd;
  120.     memset( &ufd, 0, sizeof (ufd) );
  121.     ufd.fd = m_connectionNumber;
  122.     ufd.events = POLLIN;
  123.     // Wait for an X11 event, or timeout
  124.     return poll( &ufd, 1, delay ) > 0;
  125. }
  126. #endif