aboutsummaryrefslogblamecommitdiffstats
path: root/src/event/DeferredMonitor.cxx
blob: 62edb7817560446334f46916d6346ed17024b7ec (plain) (tree)

























                                                                          


                        




                                           
      




                           



                                    


                                                         
      

 














                                        
    
                      
 









                                               








                                                            

      
/*
 * Copyright (C) 2003-2013 The Music Player Daemon Project
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 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 General Public License for more details.
 *
 * You should have received a copy of the GNU 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.
 */

#include "config.h"
#include "DeferredMonitor.hxx"
#include "Loop.hxx"

void
DeferredMonitor::Cancel()
{
#ifdef USE_EPOLL
	pending = false;
#else
	const ScopeLock protect(mutex);
	if (source_id != 0) {
		g_source_remove(source_id);
		source_id = 0;
	}
#endif
}

void
DeferredMonitor::Schedule()
{
#ifdef USE_EPOLL
	if (!pending.exchange(true))
		fd.Write();
#else
	const ScopeLock protect(mutex);
	if (source_id == 0)
		source_id = loop.AddIdle(Callback, this);
#endif
}

#ifdef USE_EPOLL

bool
DeferredMonitor::OnSocketReady(unsigned)
{
	fd.Read();

	if (pending.exchange(false))
		RunDeferred();

	return true;
}

#else

void
DeferredMonitor::Run()
{
	{
		const ScopeLock protect(mutex);
		if (source_id == 0)
			/* cancelled */
			return;

		source_id = 0;
	}

	RunDeferred();
}

gboolean
DeferredMonitor::Callback(gpointer data)
{
	DeferredMonitor &monitor = *(DeferredMonitor *)data;
	monitor.Run();
	return false;
}

#endif