/*
* Copyright (c) 2007 Wayne Meissner
*
* This file is part of gstreamer-java.
*
* This code is free software: you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License version 3 only, as
* published by the Free Software Foundation.
*
* This code 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
* version 3 for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* version 3 along with this work. If not, see <http://www.gnu.org/licenses/>.
*/
package org.gstreamer.lowlevel;
import static org.gstreamer.lowlevel.GlibAPI.GLIB_API;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
import org.gstreamer.Gst;
import com.sun.jna.Pointer;
/**
* The GLib main loop.
*/
public class MainLoop extends RefCountedObject {
private static final List<Runnable> bgTasks = new LinkedList<Runnable>();
/**
* Creates a new instance of {@code MainLoop}
*
* <p> This will create a new main loop on the default gstreamer main context.
*
*/
public MainLoop() {
super(initializer(GLIB_API.g_main_loop_new(Gst.getMainContext(), false)));
}
/**
* Creates a new instance of {@code MainLoop}
*
* <p> This variant is used internally.
*
* @param init internal initialization data.
*/
public MainLoop(Initializer init) {
super(init);
}
/**
* Instructs a main loop to stop processing and return from {@link #run}.
*/
public void quit() {
invokeLater(new Runnable() {
public void run() {
GLIB_API.g_main_loop_quit(MainLoop.this);
}
});
}
/**
* Enter a loop, processing all events.
* <p> The loop will continue processing events until {@link #quit} is
* called.
*/
public void run() {
GLIB_API.g_main_loop_run(this);
}
/**
* Returns whether this main loop is currently processing or not.
*
* @return <tt>true</tt> if the main loop is currently being run.
*/
public boolean isRunning() {
return GLIB_API.g_main_loop_is_running(this);
}
/**
* Gets the main context for this main loop.
*
* @return a main context.
*/
public GMainContext getMainContext() {
return GLIB_API.g_main_loop_get_context(this);
}
/**
* Runs the main loop in a background thread.
*/
public void startInBackground() {
bgThread = new java.lang.Thread(new Runnable() {
public void run() {
MainLoop.this.run();
}
}, "MainLoop");
bgThread.setDaemon(true);
bgThread.setName("gmainloop");
bgThread.start();
}
/**
* Invokes a task on the main loop thread.
* <p> This method will wait until the task has completed before returning.
*
* @param r the task to invoke.
*/
public void invokeAndWait(Runnable r) {
FutureTask<Object> task = new FutureTask<Object>(r, null);
invokeLater(task);
try {
task.get();
} catch (InterruptedException ex) {
throw new RuntimeException(ex.getCause());
} catch (ExecutionException ex) {
throw new RuntimeException(ex.getCause());
}
}
private static final GlibAPI.GSourceFunc bgCallback = new GlibAPI.GSourceFunc() {
public boolean callback(Pointer source) {
// System.out.println("Running g_idle callbacks");
List<Runnable> tasks = new ArrayList<Runnable>();
synchronized (bgTasks) {
tasks.addAll(bgTasks);
bgTasks.clear();
}
for (Runnable r : tasks) {
r.run();
}
GLIB_API.g_source_unref(source);
return false;
}
};
/**
* Invokes a task on the main loop thread.
* <p> This method returns immediately, without waiting for the task to
* complete.
*
* @param r the task to invoke.
*/
public void invokeLater(final Runnable r) {
// System.out.println("Scheduling idle callbacks");
synchronized (bgTasks) {
boolean empty = bgTasks.isEmpty();
bgTasks.add(r);
// Only trigger the callback if there were no existing elements in the list
// otherwise it is already triggered
if (empty) {
GSource source = GLIB_API.g_idle_source_new();
GLIB_API.g_source_set_callback(source, bgCallback, source, null);
source.attach(Gst.getMainContext());
source.disown(); // gets destroyed in the callback
}
}
}
//--------------------------------------------------------------------------
// protected methods
//
/**
* Increases the reference count on the native {@code GMainLoop}
*/
@Override
protected void ref() {
GLIB_API.g_main_loop_ref(this);
}
/**
* Decreases the reference count on the native {@code GMainLoop}
*/
@Override
protected void unref() {
GLIB_API.g_main_loop_unref(this);
}
/**
* Frees the native {@code GMainLoop}
*/
@Override
protected void disposeNativeHandle(Pointer ptr) {
GLIB_API.g_main_loop_unref(ptr);
}
//--------------------------------------------------------------------------
// Instance variables
//
private Thread bgThread;
}