@@ -76,7 +76,7 @@ Control_Sequence::Control_Sequence ( Track *track ) : Sequence( 0 ) | |||||
Control_Sequence::~Control_Sequence ( ) | Control_Sequence::~Control_Sequence ( ) | ||||
{ | { | ||||
Fl::remove_timeout( &Control_Sequence::process_osc, this ); | |||||
// Fl::remove_timeout( &Control_Sequence::process_osc, this ); | |||||
Loggable::block_start(); | Loggable::block_start(); | ||||
@@ -110,8 +110,6 @@ Control_Sequence::init ( void ) | |||||
interpolation( Linear ); | interpolation( Linear ); | ||||
frequency( 10 ); | frequency( 10 ); | ||||
Fl::add_timeout( _interval, &Control_Sequence::process_osc, this ); | |||||
} | } | ||||
@@ -140,7 +138,7 @@ Control_Sequence::get_unjournaled ( Log_Entry &e ) const | |||||
free( path ); | free( path ); | ||||
free( peer ); | free( peer ); | ||||
} | } | ||||
e.add( ":frequency", frequency() ); | |||||
/* e.add( ":frequency", frequency() ); */ | |||||
} | } | ||||
void | void | ||||
@@ -173,8 +171,8 @@ Control_Sequence::set ( Log_Entry &e ) | |||||
name( v ); | name( v ); | ||||
else if ( ! strcmp( ":interpolation", s ) ) | else if ( ! strcmp( ":interpolation", s ) ) | ||||
interpolation( (curve_type_e)atoi( v ) ); | interpolation( (curve_type_e)atoi( v ) ); | ||||
else if ( ! strcmp( ":frequency", s ) ) | |||||
frequency( atoi( v ) ); | |||||
/* else if ( ! strcmp( ":frequency", s ) ) */ | |||||
/* frequency( atoi( v ) ); */ | |||||
else if ( ! strcmp( ":osc_peer", s ) ) | else if ( ! strcmp( ":osc_peer", s ) ) | ||||
{ | { | ||||
_osc_connected_peer = strdup( v ); | _osc_connected_peer = strdup( v ); | ||||
@@ -380,18 +378,18 @@ Control_Sequence::menu_cb ( const Fl_Menu_ *m ) | |||||
interpolation( Linear ); | interpolation( Linear ); | ||||
else if ( ! strcmp( picked, "Interpolation/None" ) ) | else if ( ! strcmp( picked, "Interpolation/None" ) ) | ||||
interpolation( None ); | interpolation( None ); | ||||
else if ( ! strcmp( picked, "Frequency/1Hz" ) ) | |||||
frequency( 1 ); | |||||
else if ( ! strcmp( picked, "Frequency/5Hz" ) ) | |||||
frequency( 5 ); | |||||
else if ( ! strcmp( picked, "Frequency/10Hz" ) ) | |||||
frequency( 10 ); | |||||
else if ( ! strcmp( picked, "Frequency/20Hz" ) ) | |||||
frequency( 20 ); | |||||
else if ( ! strcmp( picked, "Frequency/30Hz" ) ) | |||||
frequency( 30 ); | |||||
else if ( ! strcmp( picked, "Frequency/60Hz" ) ) | |||||
frequency( 60 ); | |||||
/* else if ( ! strcmp( picked, "Frequency/1Hz" ) ) */ | |||||
/* frequency( 1 ); */ | |||||
/* else if ( ! strcmp( picked, "Frequency/5Hz" ) ) */ | |||||
/* frequency( 5 ); */ | |||||
/* else if ( ! strcmp( picked, "Frequency/10Hz" ) ) */ | |||||
/* frequency( 10 ); */ | |||||
/* else if ( ! strcmp( picked, "Frequency/20Hz" ) ) */ | |||||
/* frequency( 20 ); */ | |||||
/* else if ( ! strcmp( picked, "Frequency/30Hz" ) ) */ | |||||
/* frequency( 30 ); */ | |||||
/* else if ( ! strcmp( picked, "Frequency/60Hz" ) ) */ | |||||
/* frequency( 60 ); */ | |||||
else if ( ! strcmp( picked, "/Rename" ) ) | else if ( ! strcmp( picked, "/Rename" ) ) | ||||
{ | { | ||||
@@ -435,8 +433,6 @@ Control_Sequence::process_osc ( void *v ) | |||||
void | void | ||||
Control_Sequence::process_osc ( void ) | Control_Sequence::process_osc ( void ) | ||||
{ | { | ||||
Fl::repeat_timeout( _interval, &Control_Sequence::process_osc, this ); | |||||
if ( _osc_output && _osc_output->connected() ) | if ( _osc_output && _osc_output->connected() ) | ||||
{ | { | ||||
sample_t buf[1]; | sample_t buf[1]; | ||||
@@ -500,12 +496,12 @@ Control_Sequence::handle ( int m ) | |||||
menu.add( "Interpolation/None", 0, 0, 0, FL_MENU_RADIO | ( interpolation() == None ? FL_MENU_VALUE : 0 ) ); | menu.add( "Interpolation/None", 0, 0, 0, FL_MENU_RADIO | ( interpolation() == None ? FL_MENU_VALUE : 0 ) ); | ||||
menu.add( "Interpolation/Linear", 0, 0, 0, FL_MENU_RADIO | ( interpolation() == Linear ? FL_MENU_VALUE : 0 ) ); | menu.add( "Interpolation/Linear", 0, 0, 0, FL_MENU_RADIO | ( interpolation() == Linear ? FL_MENU_VALUE : 0 ) ); | ||||
menu.add( "Frequency/1Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 1 ? FL_MENU_VALUE : 0 ) ); | |||||
menu.add( "Frequency/5Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 5 ? FL_MENU_VALUE : 0 ) ); | |||||
menu.add( "Frequency/10Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 10 ? FL_MENU_VALUE : 0 ) ); | |||||
menu.add( "Frequency/20Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 20 ? FL_MENU_VALUE : 0 ) ); | |||||
menu.add( "Frequency/30Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 30 ? FL_MENU_VALUE : 0 ) ); | |||||
menu.add( "Frequency/60Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 60 ? FL_MENU_VALUE : 0 ) ); | |||||
/* menu.add( "Frequency/1Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 1 ? FL_MENU_VALUE : 0 ) ); */ | |||||
/* menu.add( "Frequency/5Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 5 ? FL_MENU_VALUE : 0 ) ); */ | |||||
/* menu.add( "Frequency/10Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 10 ? FL_MENU_VALUE : 0 ) ); */ | |||||
/* menu.add( "Frequency/20Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 20 ? FL_MENU_VALUE : 0 ) ); */ | |||||
/* menu.add( "Frequency/30Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 30 ? FL_MENU_VALUE : 0 ) ); */ | |||||
/* menu.add( "Frequency/60Hz", 0, 0, 0, FL_MENU_RADIO | ( frequency() == 60 ? FL_MENU_VALUE : 0 ) ); */ | |||||
menu.add( "Rename", 0, 0, 0 ); | menu.add( "Rename", 0, 0, 0 ); | ||||
@@ -62,8 +62,6 @@ private: | |||||
static void menu_cb ( Fl_Widget *w, void *v ); | static void menu_cb ( Fl_Widget *w, void *v ); | ||||
void menu_cb ( const Fl_Menu_ *m ); | void menu_cb ( const Fl_Menu_ *m ); | ||||
static void process_osc ( void *v ); | |||||
void process_osc ( void ); | |||||
float _rate; | float _rate; | ||||
@@ -85,6 +83,9 @@ protected: | |||||
public: | public: | ||||
static void process_osc ( void *v ); | |||||
void process_osc ( void ); | |||||
void connect_osc ( void ); | void connect_osc ( void ); | ||||
static bool draw_with_gradient; | static bool draw_with_gradient; | ||||
@@ -0,0 +1,71 @@ | |||||
/*******************************************************************************/ | |||||
/* Copyright (C) 2012 Jonathan Moore Liles */ | |||||
/* */ | |||||
/* 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; see the file COPYING. If not,write to the Free Software */ | |||||
/* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ | |||||
/*******************************************************************************/ | |||||
#include "OSC_Thread.H" | |||||
#include "Timeline.H" | |||||
#include <stdlib.h> | |||||
#include <unistd.h> | |||||
extern Timeline *timeline; | |||||
OSC_Thread::OSC_Thread ( ) | |||||
{ | |||||
// _thread.init(); | |||||
} | |||||
OSC_Thread::~OSC_Thread ( ) | |||||
{ | |||||
} | |||||
void | |||||
OSC_Thread::start ( ) | |||||
{ | |||||
_thread.clone( &OSC_Thread::process, this ); | |||||
} | |||||
void | |||||
OSC_Thread::process ( void ) | |||||
{ | |||||
_thread.name( "OSC" ); | |||||
for ( ;; ) | |||||
{ | |||||
usleep( 100 * 1000 ); | |||||
lock(); | |||||
timeline->process_osc(); | |||||
unlock(); | |||||
} | |||||
} | |||||
void * | |||||
OSC_Thread::process ( void *v ) | |||||
{ | |||||
OSC_Thread *t = (OSC_Thread*)v; | |||||
t->process(); | |||||
return NULL; | |||||
} | |||||
@@ -0,0 +1,38 @@ | |||||
/*******************************************************************************/ | |||||
/* Copyright (C) 2012 Jonathan Moore Liles */ | |||||
/* */ | |||||
/* 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; see the file COPYING. If not,write to the Free Software */ | |||||
/* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ | |||||
/*******************************************************************************/ | |||||
#pragma once | |||||
#include "Thread.H" | |||||
#include "Mutex.H" | |||||
class OSC_Thread : public Mutex | |||||
{ | |||||
Thread _thread; /* io thread */ | |||||
public: | |||||
OSC_Thread ( ); | |||||
virtual ~OSC_Thread ( ); | |||||
void start ( void ); | |||||
void process ( void ); | |||||
static void *process ( void * ); | |||||
}; |
@@ -51,6 +51,8 @@ | |||||
#include "TLE.H" | #include "TLE.H" | ||||
/* */ | /* */ | ||||
#include "OSC_Thread.H" | |||||
#include "NSM.H" | #include "NSM.H" | ||||
extern NSM_Client *nsm; | extern NSM_Client *nsm; | ||||
@@ -394,6 +396,7 @@ Timeline::Timeline ( int X, int Y, int W, int H, const char* L ) : BASE( X, Y, W | |||||
{ | { | ||||
Loggable::snapshot_callback( &Timeline::snapshot, this ); | Loggable::snapshot_callback( &Timeline::snapshot, this ); | ||||
osc_thread = 0; | |||||
_sample_rate = 0; | _sample_rate = 0; | ||||
box( FL_FLAT_BOX ); | box( FL_FLAT_BOX ); | ||||
@@ -1458,8 +1461,12 @@ Timeline::add_track ( Track *track ) | |||||
engine->lock(); | engine->lock(); | ||||
osc_thread->lock(); | |||||
tracks->add( track ); | tracks->add( track ); | ||||
osc_thread->unlock(); | |||||
engine->unlock(); | engine->unlock(); | ||||
/* FIXME: why is this necessary? doesn't the above add do DAMAGE_CHILD? */ | /* FIXME: why is this necessary? doesn't the above add do DAMAGE_CHILD? */ | ||||
@@ -1475,9 +1482,13 @@ Timeline::remove_track ( Track *track ) | |||||
engine->lock(); | engine->lock(); | ||||
osc_thread->lock(); | |||||
/* FIXME: what to do about track contents? */ | /* FIXME: what to do about track contents? */ | ||||
tracks->remove( track ); | tracks->remove( track ); | ||||
osc_thread->unlock(); | |||||
engine->unlock(); | engine->unlock(); | ||||
/* FIXME: why is this necessary? doesn't the above add do DAMAGE_CHILD? */ | /* FIXME: why is this necessary? doesn't the above add do DAMAGE_CHILD? */ | ||||
@@ -1585,6 +1596,13 @@ Timeline::init_osc ( const char *osc_port ) | |||||
/* poll so we can keep OSC handlers running in the GUI thread and avoid extra sync */ | /* poll so we can keep OSC handlers running in the GUI thread and avoid extra sync */ | ||||
Fl::add_timeout( OSC_INTERVAL, &Timeline::check_osc, this ); | Fl::add_timeout( OSC_INTERVAL, &Timeline::check_osc, this ); | ||||
if ( ! osc_thread ) | |||||
{ | |||||
osc_thread = new OSC_Thread(); | |||||
osc_thread->start(); | |||||
} | |||||
return 0; | return 0; | ||||
} | } | ||||
@@ -1664,6 +1682,7 @@ Timeline::discover_peers ( void ) | |||||
c->connect_osc(); | c->connect_osc(); | ||||
} | } | ||||
} | } | ||||
} | } | ||||
void | void | ||||
@@ -1697,5 +1716,22 @@ Timeline::add_osc_peers_to_menu ( Fl_Menu_Button *m, const char *prefix ) | |||||
osc->list_peers( &Timeline::peer_callback, this ); | osc->list_peers( &Timeline::peer_callback, this ); | ||||
} | } | ||||
/* runs in the OSC thread... */ | |||||
void | |||||
Timeline::process_osc ( void ) | |||||
{ | |||||
THREAD_ASSERT( OSC ); | |||||
/* reconnect OSC signals */ | |||||
for ( int i = tracks->children(); i-- ; ) | |||||
{ | |||||
Track *t = (Track*)tracks->child( i ); | |||||
for ( int j = t->control->children(); j--; ) | |||||
{ | |||||
Control_Sequence *c = (Control_Sequence*)t->control->child( j ); | |||||
c->process_osc(); | |||||
} | |||||
} | |||||
} | |||||
@@ -32,6 +32,7 @@ | |||||
#include <list> | #include <list> | ||||
#include "OSC/Endpoint.H" | #include "OSC/Endpoint.H" | ||||
#include "OSC_Thread.H" | |||||
class Fl_Scroll; | class Fl_Scroll; | ||||
class Fl_Pack; | class Fl_Pack; | ||||
@@ -120,7 +121,9 @@ class Timeline : public Fl_Single_Window, public RWLock | |||||
public: | public: | ||||
OSC::Endpoint *osc; | OSC::Endpoint *osc; | ||||
OSC_Thread *osc_thread; | |||||
void process_osc ( void ); | |||||
#undef Bars | #undef Bars | ||||
#undef Beats | #undef Beats | ||||
#undef None | #undef None | ||||
@@ -253,6 +256,7 @@ public: | |||||
private: | private: | ||||
static void snapshot ( void *v ) { ((Timeline*)v)->snapshot(); } | static void snapshot ( void *v ) { ((Timeline*)v)->snapshot(); } | ||||
void snapshot ( void ); | void snapshot ( void ); | ||||
@@ -581,8 +581,12 @@ Track::remove ( Control_Sequence *t ) | |||||
engine->lock(); | engine->lock(); | ||||
timeline->osc_thread->lock(); | |||||
control->remove( t ); | control->remove( t ); | ||||
timeline->osc_thread->unlock(); | |||||
engine->unlock(); | engine->unlock(); | ||||
resize(); | resize(); | ||||