changeset 468:85cf9c0da052 prerelease

merge
author Giulio Moro <giuliomoro@yahoo.it>
date Mon, 20 Jun 2016 17:08:02 +0100 (2016-06-20)
parents 03a2cd5f151b (current diff) 67c45feec3c3 (diff)
children 898d57ac58a5 5a936f8e9447
files examples/08-PureData/basic_libpd/_main.pd examples/08-PureData/basic_libpd/m_pd.h examples/08-PureData/basic_libpd/main.cpp examples/08-PureData/basic_libpd/render.cpp examples/08-PureData/basic_libpd/s_stuff.h examples/08-PureData/basic_libpd/z_libpd.h examples/7segment/render.cpp examples/airharp/Junction.cpp examples/airharp/Junction.h examples/airharp/MassSpringDamper.cpp examples/airharp/MassSpringDamper.h examples/airharp/Plectrum.cpp examples/airharp/Plectrum.h examples/airharp/String.cpp examples/airharp/String.h examples/airharp/Waveguide.cpp examples/airharp/Waveguide.h examples/airharp/render.cpp examples/analogDigitalDemo/main.cpp examples/analogDigitalDemo/render.cpp examples/audio_in_FFT/main.cpp examples/audio_in_FFT/render.cpp examples/basic/main.cpp examples/basic/render.cpp examples/basic_FFT_phase_vocoder/SampleData.h examples/basic_FFT_phase_vocoder/main.cpp examples/basic_FFT_phase_vocoder/render.cpp examples/basic_FFT_phase_vocoder/sample.wav examples/basic_analog_input/main.cpp examples/basic_analog_input/render.cpp examples/basic_analog_output/main.cpp examples/basic_analog_output/render.cpp examples/basic_blink/render.cpp examples/basic_button/render.cpp examples/basic_midi/render.cpp examples/basic_network/main.cpp examples/basic_passthru/main.cpp examples/basic_passthru/render.cpp examples/basic_pulseIn/render.cpp examples/basic_writeFile/main.cpp examples/basic_writeFile/render.cpp examples/bucket_brigade_chorus/render.cpp examples/cape_test/main.cpp examples/cape_test/render.cpp examples/core/link_core.sh examples/core/render.cpp examples/d-box/ADSR.cpp examples/d-box/ADSR.h examples/d-box/AnalogInput.cpp examples/d-box/AnalogInput.h examples/d-box/Biquad.cpp examples/d-box/Biquad.h examples/d-box/DBoxSynth.h examples/d-box/DboxSensors.cpp examples/d-box/DboxSensors.h examples/d-box/FIRfilter.h examples/d-box/FeedbackOscillator.cpp examples/d-box/FeedbackOscillator.h examples/d-box/OscillatorBank.cpp examples/d-box/OscillatorBank.h examples/d-box/PinkNoise.cpp examples/d-box/PinkNoise.h examples/d-box/StatusLED.cpp examples/d-box/StatusLED.h examples/d-box/audio_routines.S examples/d-box/config.h examples/d-box/how_to_build_dbox.txt examples/d-box/logger.cpp examples/d-box/logger.h examples/d-box/main.cpp examples/d-box/prio.cpp examples/d-box/prio.h examples/d-box/render.cpp examples/d-box/sensors.cpp examples/d-box/sensors.h examples/d-box/spear_parser.cpp examples/d-box/spear_parser.h examples/filter_FIR/FIRfilter.h examples/filter_FIR/SampleData.h examples/filter_FIR/longsample.wav examples/filter_FIR/main.cpp examples/filter_FIR/render.cpp examples/filter_IIR/SampleData.h examples/filter_IIR/longsample.wav examples/filter_IIR/main.cpp examples/filter_IIR/render.cpp examples/gpioAnalogLoopbackTest/render.cpp examples/level_meter/render.cpp examples/loggingSensors/main.cpp examples/loggingSensors/render.cpp examples/measure_noisefloor/main.cpp examples/measure_noisefloor/render.cpp examples/minimal/render.cpp examples/mpr121/I2C_MPR121.cpp examples/mpr121/I2C_MPR121.h examples/mpr121/render.cpp examples/osc/render.cpp examples/oscillator_bank/audio_routines.S examples/oscillator_bank/main.cpp examples/oscillator_bank/render.cpp examples/pd/basicAnalogIn/_main.pd examples/pd/basicAnalogOut/_main.pd examples/pd/circularBuffer/_main.pd examples/pd/digital/bela_digital-help.pd examples/pd/digital/digital_example.pd examples/pd/envelopeTrigger/_main.pd examples/pd/hello-midi/_main.pd examples/pd/hello-midi/countTo.pd examples/pd/hello-midi/maxArray.pd examples/pd/hello-midi/mono.pd examples/pd/hello-world/_main.pd examples/pd/karplusStrong/_main.pd examples/pd/karplusStrong/count.pd examples/pd/rubberDuckie/_main.pd examples/pd/samphold/_main.pd examples/pd/simple-fm/_main.pd examples/pd/tableScrubbing/_main.pd examples/pd/tableScrubbing/rt.wav examples/samples/SampleData.h examples/samples/main.cpp examples/samples/render.cpp examples/samples/sample.wav examples/scope_analogue/render.cpp examples/scope_basic/render.cpp examples/stepper/render.cpp examples/tank_wars/game.cpp examples/tank_wars/game.h examples/tank_wars/main.cpp examples/tank_wars/render.cpp examples/tank_wars/vector_graphics.cpp examples/tank_wars/vector_graphics.h
diffstat 260 files changed, 17468 insertions(+), 15827 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/01-Basics/minimal/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,120 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <Bela.h>
+
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+bool setup(BelaContext *context, void *userData)
+{
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+void render(BelaContext *context, void *userData)
+{
+
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+
+\example 01-minimal
+
+The bare bones
+----------------------
+
+The structure of a render.cpp file
+----------------------------------
+A render.cpp file has three functions: `setup()`, `render()` and `cleanup()`.
+
+`setup()` is an initialisation function which runs before audio rendering begins. 
+It is called once when the project starts. Use it to prepare any memory or 
+resources that will be needed in `render()`.
+
+`render()` is a function that is regularly called, over and over continuously, at 
+the highest priority by the audio engine. It is used to process audio and 
+sensor data. This function is called regularly by the system every time there 
+is a new block of audio and/or sensor data to process.
+
+`cleanup()` is a function that is called when the program stops, to finish up any 
+processes that might still be running.
+
+Here we will briefly explain each function and the structure of the render.cpp 
+
+Before any of the functions
+---------------------------
+At the top of the file, include any libraries you might need.
+
+Additionally, declare any global variables. In these tutorial sketches, all 
+global variables are preceded by a `g` so we always know which variables are 
+global - `gSampleData`, for example. It's not mandatory but is a really good way 
+of keeping track of what's global and what's not.
+
+Sometimes it's necessary to access a variable from another file, such as 
+main.cpp. In this case, precede this variable with the keyword `extern`.
+
+Function arguments
+------------------
+`setup()`, `render()` and `cleanup()` each take the same arguments. These are:
+
+`0ext *context`
+`void *userData`
+
+These arguments are pointers to data structures. The main one that's used is 
+`context`, which is a pointer to a data structure containing lots of information 
+you need.
+
+Take a look at what's in the data structure [here]
+(https://code.soundsoftware.ac.uk/projects/beaglert/embedded/structBeagleRTContext.html).
+
+You can access any of these bits of information about current audio and sensor 
+settings and pointers to data buffers that are contained in the data structure 
+like this: `context->name_of_item`.
+
+For example, `context->audioChannels` returns the number of audio channels. 
+`context->audioSampleRate` returns the audio sample rate. 
+`context->audioIn[n]` would give you the current input sample (assuming that 
+your input is mono - if it's not you will have to account for multiple channels).
+
+Note that `audioIn`, `audioOut`, `analogIn`, `analogOut` are all arrays (buffers).
+
+*/
\ No newline at end of file
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/01-Basics/passthrough/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,107 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --help [-h]:             Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "h", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, 0) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/01-Basics/passthrough/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,119 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <Bela.h>
+#include <rtdk.h>
+
+bool setup(BelaContext *context, void *userData)
+{
+	// Nothing to do here...
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	// Simplest possible case: pass inputs through to outputs
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		for(unsigned int ch = 0; ch < context->audioChannels; ch++){
+			// Two equivalent ways to write this code
+
+			// The long way, using the buffers directly:
+			// context->audioOut[n * context->audioChannels + ch] =
+			// 		context->audioIn[n * context->audioChannels + ch];
+
+			// Or using the macros:
+			audioWrite(context, n, ch, audioRead(context, n, ch));
+		}
+	}
+
+	// Same with analog channelss
+	for(unsigned int n = 0; n < context->analogFrames; n++) {
+		for(unsigned int ch = 0; ch < context->analogChannels; ch++) {
+			// Two equivalent ways to write this code
+
+			// The long way, using the buffers directly:
+			// context->analogOut[n * context->analogChannels + ch] = context->analogIn[n * context->analogChannels + ch];
+
+			// Or using the macros:
+			analogWrite(context, n, ch, analogRead(context, n, ch));
+		}
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 01-passthrough
+
+Audio and analog passthrough: input to output
+-----------------------------------------
+
+This sketch demonstrates how to read from and write to the audio and analog input and output buffers.
+
+In `render()` you'll see a nested for loop structure. You'll see this in all Bela projects. 
+The first for loop cycles through `audioFrames`, the second through 
+`audioChannels` (in this case left 0 and right 1).
+
+You can access any information about current audio and sensor settings you can do the following: 
+`context->name_of_item`. For example `context->audioChannels` returns current number of channels,
+`context->audioFrames` returns the current number of audio frames, 
+`context->audioSampleRate` returns the audio sample rate.
+
+You can look at all the information you can access in ::BeagleRTContext.
+
+Reading and writing from the audio buffers
+------------------------------------------
+
+The simplest way to read samples from the audio input buffer is with
+`audioRead()` which we pass three arguments: context, current audio 
+frame and current channel. In this example we have 
+`audioRead(context, n, ch)` where both `n` and `ch` are provided by 
+the nested for loop structure.
+
+We can write samples to the audio output buffer in a similar way using 
+`audioWrite()`. This has a fourth argument which is the value of the output.
+For example `audioWrite(context, n, ch, value_to_output)`.
+
+Reading and writing from the analog buffers
+-------------------------------------------
+
+The same is true for `analogRead()` and `analogWrite()`.
+
+Note that for the analog channels we write to and read from the buffers in a separate set 
+of nested for loops. This is because the they are sampled at half audio rate by default. 
+The first of these for loops cycles through `analogFrames`, the second through
+`analogChannels`.
+
+By setting `audioWriteFrame(context, n, ch, audioReadFrame(context, n, ch))` and
+`analogWrite(context, n, ch, analogReadFrame(context, n, ch))` we have a simple 
+passthrough of audio input to output and analog input to output.
+
+
+It is also possible to address the buffers directly, for example: 
+`context->audioOut[n * context->audioChannels + ch]`.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/01-Basics/scope/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,102 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+    Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+    Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <Bela.h>
+#include <Scope.h>
+#include <cmath>
+
+// set the frequency of the oscillators
+float gFrequency = 110.0;
+float gPhase;
+float gInverseSampleRate;
+
+// instantiate the scope
+Scope scope;
+
+bool setup(BelaContext *context, void *userData)
+{
+    // tell the scope how many channels and the sample rate
+    scope.setup(3, context->audioSampleRate);
+
+    gPhase = 0;
+    gInverseSampleRate = 1.0f/context->audioSampleRate;
+    
+	return true;
+}
+
+float lastOut = 0.0;
+float lastOut2 = 0.0;
+void render(BelaContext *context, void *userData)
+{
+    // iterate over the audio frames and create three oscillators, seperated in phase by PI/2
+    for (unsigned int n=0; n<context->audioFrames; n++){
+        float out = 0.8f * sinf(gPhase);
+        float out2 = 0.8f * sinf(gPhase - M_PI/2);
+		float out3 = 0.8f * sinf(gPhase + M_PI/2);
+		gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate;
+		if(gPhase > 2.0 * M_PI)
+			gPhase -= 2.0 * M_PI;
+			
+		// log the three oscillators to the scope
+        scope.log(out, out2, out3);
+        
+        // optional - tell the scope to trigger when oscillator 1 becomes less than oscillator 2
+        // note this has no effect unless trigger mode is set to custom in the scope UI
+        if (lastOut >= lastOut2 && out < out2){
+            scope.trigger();
+        }
+        
+        lastOut = out;
+        lastOut2 = out2;
+    }
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 01-scope
+
+Oscilloscope in-browser
+-----------------------
+
+This example demonstrates the scope feature of the IDE.
+
+The scope is instantiated at the top of the file via `Scope scope;`
+
+In `setup()` we define how many channels the scope should have and the sample 
+rate that it should run at via `scope.setup(3, context->audioSampleRate)`.
+
+In `render()` we choose what the scope log via `scope.log(out, out2, out3)`. 
+In this example the scope is logging three sine waves with different phases. To see
+the output click on the <b>Open Scope</b> button.
+
+An additional option is to set the trigger of the oscilloscope from within `render()`.
+In this example we are triggering the scope when oscillator 1 becomes less than 
+oscillator 2 via `scope.trigger()`. Note that this functionality only takes effect
+when the triggering mode is set to custom in the scope UI.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/01-Basics/sinetone/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,114 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <unistd.h>
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	float frequency = 440.0;	// Frequency of oscillator
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"frequency", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				frequency = atof(optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &frequency) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/01-Basics/sinetone/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,86 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <Bela.h>
+#include <cmath>
+
+float gFrequency = 440.0;
+float gPhase;
+float gInverseSampleRate;
+
+bool setup(BelaContext *context, void *userData)
+{
+	// Retrieve a parameter passed in from the initAudio() call
+	if(userData != 0)
+		gFrequency = *(float *)userData;
+
+	gInverseSampleRate = 1.0 / context->audioSampleRate;
+	gPhase = 0.0;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		float out = 0.8f * sinf(gPhase);
+		gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate;
+		if(gPhase > 2.0 * M_PI)
+			gPhase -= 2.0 * M_PI;
+
+		for(unsigned int channel = 0; channel < context->audioChannels; channel++) {
+			// Two equivalent ways to write this code
+
+			// The long way, using the buffers directly:
+			// context->audioOut[n * context->audioChannels + channel] = out;
+
+			// Or using the macros:
+			audioWrite(context, n, channel, out);
+		}
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 01-sinetone
+
+Producing your first bleep!
+---------------------------
+
+This sketch is the hello world of embedded interactive audio. Better known as bleep, it 
+produces a sine tone.
+
+The frequency of the sine tone is determined by a global variable, `gFrequency` 
+(line 12). The sine tone is produced by incrementing the phase of a sin function 
+on every audio frame.
+
+In render() you'll see a nested for loop structure. You'll see this in all Bela projects. 
+The first for loop cycles through 'audioFrames', the second through 'audioChannels' (in this case left 0 and right 1). 
+It is good to familiarise yourself with this structure as it's fundamental to producing sound with the system.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/02-Digital/digital-input/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,82 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <cmath>
+#include <rtdk.h>
+#include <stdlib.h>
+
+
+bool setup(BelaContext *context, void *userData)
+{
+	pinMode(context, 0, P8_08, INPUT);
+	pinMode(context, 0, P8_07, OUTPUT);
+	return true;
+}
+
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n=0; n<context->digitalFrames; n++){
+		int status=digitalRead(context, 0, P8_08); //read the value of the button
+		digitalWriteOnce(context, n, P8_07, status); //write the status to the LED
+		float out = 0.1 * status * rand() / (float)RAND_MAX * 2 - 1; //generate some noise, gated by the button
+		for(unsigned int j = 0; j < context->audioChannels; j++){
+			audioWrite(context, n, j, out); //write the audio output
+		}
+	}
+}
+
+
+void cleanup(BelaContext *context, void *userData)
+{
+	// Nothing to do here
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 02-digital-input
+
+Switching an LED on and off
+---------------------------
+
+This example brings together digital input and digital output. The program will read 
+a button and turn the LED on and off according to the state of the button.
+
+- connect an LED in series with a 470ohm resistor between P8_07 and ground.
+- connect a 1k resistor to P9_03 (+3.3V),
+- connect the other end of the resistor to both a button and P8_08
+- connect the other end of the button to ground.
+
+You will notice that the LED will normally stay on and will turn off as long as 
+the button is pressed. This is due to the fact that the LED is set to the same 
+value read at input P8_08. When the button is not pressed, P8_08 is `HIGH` and so 
+P8_07 is set to `HIGH` as well, so that the LED conducts and emits light. When 
+the button is pressed, P8_08 goes `LOW` and P8_07 is set to `LOW`, turning off the LED.
+
+As an exercise try and change the code so that the LED only turns on when 
+the button is pressed.
+*/
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/02-Digital/digital-output/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,89 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <cmath>
+#include <rtdk.h>
+
+bool setup(BelaContext *context, void *userData)
+{
+    pinMode(context, 0, P8_07, OUTPUT);
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+  static int count=0; //counts elapsed samples
+  float interval=0.5; //how often to toggle the LED (in seconds)
+  static int status=GPIO_LOW;
+	for(unsigned int n=0; n<context->digitalFrames; n++){
+    if(count==context->digitalSampleRate*interval){ //if enough samples have elapsed
+      count=0; //reset the counter
+    // status=digitalRead(context, 0, P8_07);
+      if(status==GPIO_LOW) { //toggle the status
+          digitalWrite(context, n, P8_07, status); //write the status to the LED
+          status=GPIO_HIGH;
+      }
+      else {
+          digitalWrite(context, n, P8_07, status); //write the status to the LED
+          status=GPIO_LOW;
+      }
+    }
+    count++;
+  }
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+	// Nothing to do here
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 02-digital-output
+
+Blinking an LED
+---------------
+
+This sketch shows the simplest case of digital out. 
+
+- Connect an LED in series with a 470ohm resistor between P8_07 and ground. 
+
+The led is blinked on and off by setting the digital pin `HIGH` and `LOW` every interval seconds which is set in 
+`render()`.
+
+In `setup()` the pin mode must be set to output mode via `pinMode()`. For example: 
+`pinMode(context, 0, P8_07, OUTPUT)`. 
+In `render()` the output of the digital pins is set by `digitalWrite()`. For example: 
+`digitalWrite(context, n, P8_07, status)` where `status` can be equal to 
+either `HIGH` or `LOW`. When set `HIGH` the pin will give 3.3V, when set to 
+`LOW` 0V.
+
+To keep track of elapsed time we have a sample counter count. When the count reaches 
+a certain limit it switches state to either `HIGH` or `LOW` depending on its current 
+value. In this case the limit is `context->digitalSampleRate*interval` which 
+allows us to write the desired interval in seconds, stored in `interval`.
+*/
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/02-Digital/level-meter/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,138 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <cmath>
+
+#define NUMBER_OF_SEGMENTS	10
+
+// Two levels of audio: one follows current value, the other holds
+// peaks for longer
+float gAudioLocalLevel = 0, gAudioPeakLevel = 0;
+
+// Decay rates for detecting levels
+float gLocalDecayRate = 0.99, gPeakDecayRate = 0.999;
+
+// Thresholds for LEDs: set in setup()
+float gThresholds[NUMBER_OF_SEGMENTS + 1];
+int gSamplesToLight[NUMBER_OF_SEGMENTS];
+
+// High-pass filter on the input
+float gLastX[2] = {0};
+float gLastY[2] = {0};
+
+// These coefficients make a high-pass filter at 5Hz for 44.1kHz sample rate
+double gB0 = 0.99949640;
+double gB1 = -1.99899280;
+double gB2 = gB0;
+double gA1 = -1.99899254;
+double gA2 = 0.99899305;
+
+bool setup(BelaContext *context, void *userData)
+{	
+	// This project makes the assumption that the audio and digital
+	// sample rates are the same. But check it to be sure!
+	if(context->audioFrames != context->digitalFrames) {
+		rt_printf("Error: this project needs the audio and digital sample rates to be the same.\n");
+		return false;
+	}
+	
+	// Initialise threshold levels in -3dB steps. One extra for efficiency in render()
+	// Level = 10^(dB/20)
+	for(int i = 0; i < NUMBER_OF_SEGMENTS + 1; i++) {
+		gThresholds[i] = powf(10.0f, (-1.0 * (NUMBER_OF_SEGMENTS - i)) * .05);
+	}
+	
+	for(int i = 0; i < NUMBER_OF_SEGMENTS; i++) {
+		gSamplesToLight[i] = 0;
+		pinMode(context, 0, i, OUTPUT);
+	}
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		// Get average of audio input channels
+		float sample = 0;
+		for(unsigned int ch = 0; ch < context->audioChannels; ch++) {
+			context->audioOut[n * context->audioChannels + ch] = 
+				context->audioIn[n * context->audioChannels + ch];
+			sample += context->audioIn[n * context->audioChannels + ch];
+		}
+		
+		// Do DC-blocking on the sum
+		float out = gB0 * sample + gB1 * gLastX[0] + gB2 * gLastX[1]
+						- gA1 * gLastY[0] - gA2 * gLastY[1];
+
+		gLastX[1] = gLastX[0];
+		gLastX[0] = sample;
+		gLastY[1] = gLastY[0];
+		gLastY[0] = out;
+		
+		out = fabsf(out / (float)context->audioChannels);
+		
+		// Do peak detection: fast-responding local level
+		if(out > gAudioLocalLevel)
+			gAudioLocalLevel = out;
+		else
+			gAudioLocalLevel *= gLocalDecayRate;
+		
+		// Do peak detection: slow-responding peak level
+		if(out > gAudioPeakLevel)
+			gAudioPeakLevel = out;
+		else {
+			// Make peak decay slowly by only multiplying
+			// every few samples
+			if(((context->audioFramesElapsed + n) & 31) == 0)
+				gAudioPeakLevel *= gPeakDecayRate;
+		}	
+		// LED bargraph on digital outputs 0-9
+		for(int led = 0; led < NUMBER_OF_SEGMENTS; led++) {
+			// All LEDs up to the local level light up. The LED
+			// for the peak level also remains lit.
+			int state = LOW;
+				
+			if(gAudioLocalLevel > gThresholds[led]) {
+				state = HIGH;
+				gSamplesToLight[led] = 1000;
+			}
+			/*else if(gAudioPeakLevel > gThresholds[led] && gAudioPeakLevel <= gThresholds[led + 1]) {
+				state = HIGH;
+				gSamplesToLight[led] = 1000;
+			}*/
+			else if(--gSamplesToLight[led] > 0)
+				state = HIGH;
+			
+			// Write LED
+			digitalWriteOnce(context, n, led, state);
+		}
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/03-Analog/analog-input/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,133 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+int gSensorInputFrequency = 0;
+int gSensorInputAmplitude = 1;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --frequency [-f] input:  Choose the analog input controlling frequency (0-7; default 0)\n";
+	cerr << "   --amplitude [-a] input:  Choose the analog input controlling amplitude (0-7; default 1)\n";
+	cerr << "   --help [-h]:             Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"frequency", 1, NULL, 'f'},
+		{"amplitude", 1, NULL, 'a'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:a:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				gSensorInputFrequency = atoi(optarg);
+				if(gSensorInputFrequency < 0 || gSensorInputFrequency > 7) {
+					usage(basename(argv[0]));
+					exit(0);
+				}
+				break;
+		case 'a':
+				gSensorInputAmplitude = atoi(optarg);
+				if(gSensorInputAmplitude < 0 || gSensorInputAmplitude > 7) {
+					usage(basename(argv[0]));
+					exit(0);
+				}
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, 0) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	if(settings.verbose) {
+		cout << "--> Frequency on input " << gSensorInputFrequency << endl;
+		cout << "--> Amplitude on input " << gSensorInputAmplitude << endl;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/03-Analog/analog-input/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,114 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <rtdk.h>
+#include <cmath>
+
+float gPhase;
+float gInverseSampleRate;
+int gAudioFramesPerAnalogFrame;
+
+// These settings are carried over from main.cpp
+// Setting global variables is an alternative approach
+// to passing a structure to userData in setup()
+
+extern int gSensorInputFrequency;
+extern int gSensorInputAmplitude;
+
+bool setup(BelaContext *context, void *userData)
+{
+	if(context->analogFrames == 0 || context->analogFrames > context->audioFrames) {
+		rt_printf("Error: this example needs analog enabled, with 4 or 8 channels\n");
+		return false;
+	}
+
+	gAudioFramesPerAnalogFrame = context->audioFrames / context->analogFrames;
+	gInverseSampleRate = 1.0 / context->audioSampleRate;
+	gPhase = 0.0;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	float frequency = 440.0;
+	float amplitude = 0.8;
+
+	// There are twice as many audio frames as matrix frames since audio sample rate
+	// is twice as high
+
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		if(!(n % gAudioFramesPerAnalogFrame)) {
+			// Even audio samples: update frequency and amplitude from the matrix
+			frequency = map(analogRead(context, n/gAudioFramesPerAnalogFrame, gSensorInputFrequency), 0, 1, 100, 1000);
+			amplitude = analogRead(context, n/gAudioFramesPerAnalogFrame, gSensorInputAmplitude);
+		}
+
+		float out = amplitude * sinf(gPhase);
+
+		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
+			context->audioOut[n * context->audioChannels + channel] = out;
+
+		gPhase += 2.0 * M_PI * frequency * gInverseSampleRate;
+		if(gPhase > 2.0 * M_PI)
+			gPhase -= 2.0 * M_PI;
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 03-analog-input
+
+Connecting potentiometers
+-------------------------
+
+This sketch produces a sine tone, the frequency and amplitude of which are 
+affected by data received on the analog pins. Before looping through each audio 
+frame, we declare a value for the frequency and amplitude of our sine tone 
+(line 55); we adjust these values by taking in data from analog sensors 
+(for example potentiometers) with `analogRead()`.
+
+- connect a 10K pot to 3.3V and GND on its 1st and 3rd pins.
+- connect the 2nd middle pin of the pot to analogIn 0.
+- connect another 10K pot in the same way but with the middle pin connected to analogIn 1.
+
+The important thing to notice is that audio is sampled twice as often as analog 
+data. The audio sampling rate is 44.1kHz (44100 frames per second) and the 
+analog sampling rate is 22.05kHz (22050 frames per second). On line 62 you might 
+notice that we are processing the analog data and updating frequency and 
+amplitude only on every second audio sample, since the analog sampling rate is 
+half that of the audio.
+
+Note that the pin numbers are stored in the variables `gAnalogInputFrequency` and 
+`gAnalogInputAmplitude`. These are declared in the main.cpp file; if you look in 
+that file you will see that they have the values of 0 and 1. Bear in mind that 
+these are analog input pins which is a specific header!
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/03-Analog/analog-output/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,121 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --frequency [-f] frequency: Set the frequency of the LED fade (default: 1.0)\n";
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	float frequency = 1.0;	// Frequency of LED fades
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"frequency", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// In this example, audio isn't used so might as well leave speaker muted
+	settings.beginMuted = 1;
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				frequency = atof(optarg);
+				if(frequency < 0)
+					frequency = 0;
+				if(frequency > 11025.0)
+					frequency = 11025.0;
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &frequency) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/03-Analog/analog-output/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,109 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <rtdk.h>
+#include <cmath>
+
+// Set range for analog outputs designed for driving LEDs
+const float kMinimumAmplitude = (1.5 / 5.0);
+const float kAmplitudeRange = 1.0 - kMinimumAmplitude;
+
+float gFrequency;
+float gPhase;
+float gInverseSampleRate;
+
+bool setup(BelaContext *context, void *userData)
+{
+	// Retrieve a parameter passed in from the initAudio() call
+	gFrequency = *(float *)userData;
+
+	if(context->analogFrames == 0) {
+		rt_printf("Error: this example needs the matrix enabled\n");
+		return false;
+	}
+
+	gInverseSampleRate = 1.0 / context->analogSampleRate;
+	gPhase = 0.0;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->analogFrames; n++) {
+		// Set LED to different phase for each matrix channel
+		float relativePhase = 0.0;
+		for(unsigned int channel = 0; channel < context->analogChannels; channel++) {
+			float out = kMinimumAmplitude + kAmplitudeRange * 0.5f * (1.0f + sinf(gPhase + relativePhase));
+
+			analogWrite(context, n, channel, out);
+
+			// Advance by pi/4 (1/8 of a full rotation) for each channel
+			relativePhase += M_PI * 0.25;
+		}
+
+		gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate;
+		if(gPhase > 2.0 * M_PI)
+			gPhase -= 2.0 * M_PI;
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 03-analog-output
+
+Fading LEDs
+-----------
+
+This sketch uses a sine wave to drive the brightness of a series of LEDs 
+connected to the eight analog out pins. Again you can see the nested `for` loop 
+structure but this time for the analog output channels rather than the audio.
+
+- connect an LED in series with a 470ohm resistor between each of the analogOut pins and ground.
+
+Within the first for loop in render we cycle through each frame in the analog 
+output matrix. At each frame we then cycle through the analog output channels 
+with another for loop and set the output voltage according to the phase of a 
+sine tone that acts as an LFO. The analog output pins can provide a voltage of 
+~4.092V.
+
+The output on each pin is set with `analogWrite()` within the for loop that 
+cycles through the analog output channels. This needs to be provided with 
+arguments as follows `analogWrite(context, n, channel, out)`. Channel is 
+where the you give the address of the analog output pin (in this case we cycle 
+through each pin address in the for loop), out is the variable that holds the 
+desired output (in this case set by the sine wave).
+
+Notice that the phase of the brightness cycle for each led is different. This 
+is achieved by updating a variable that stores a relative phase value. This 
+variable is advanced by pi/4 (1/8 of a full rotation) for each channel giving 
+each of the eight LEDs a different phase.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/03-Analog/scope-analog/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,102 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <cmath>
+#include <Scope.h>
+
+Scope scope;
+
+float gInverseSampleRate;
+float gPhase;
+
+bool setup(BelaContext *context, void *userData)
+{
+
+    // setup the scope with 3 channels at the audio sample rate
+	scope.setup(3, context->audioSampleRate);
+	
+	gInverseSampleRate = 1.0 / context->audioSampleRate;
+	gPhase = 0.0;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+	    
+	    // read analogIn channels 0 and 1
+	    float in1 = analogRead(context, n, 0);
+	    float in2 = analogRead(context, n, 1);
+	    
+	    // map in1 to amplitude and in2 to frequency
+	    float amplitude = in1 * 0.8f;
+	    float frequency = map(in2, 0, 1, 100, 1000);
+	    
+	    // generate a sine wave with the amplitude and frequency 
+	    float out = amplitude * sinf(gPhase);
+	    gPhase += 2.0 * M_PI * frequency * gInverseSampleRate;
+		if(gPhase > 2.0 * M_PI)
+			gPhase -= 2.0 * M_PI;
+	    
+	    // log the sine wave and sensor values on the scope
+	    scope.log(out, in1, in2);
+	    
+	    // pass the sine wave to the audio outputs
+	    for(unsigned int channel = 0; channel < context->audioChannels; channel++)
+			context->audioOut[n * context->audioChannels + channel] = out;
+
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 03-scope-analog
+
+Connecting potentiometers
+-------------------------
+
+This example reads from analogue inputs 0 and 1 via `analogReadFrame()` and 
+generates a sine wave with amplitude and frequency determined by their values. 
+It's best to connect a 10K potentiometer to each of these analog inputs. Far 
+left and far right pins of the pot go to 3.3V and GND, the middle should be 
+connected to the analog in pins.
+
+The sine wave is then plotted on the oscilloscope. Click the Open Scope button to 
+view the results. As you turn the potentiometers you will see the amplitude and 
+frequency of the sine wave change.
+
+This project also shows as example of `map()` which allows you to re-scale a number 
+from one range to another. Note that `map()` does not constrain your variable 
+within the upper and lower limits. If you want to do this use the `constrain()`
+function.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/FFT-audio-in/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,116 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --fftsize [-s] size:     Set the fSize of the FFT, in samples\n";
+	cerr << "   --help [-h]:             Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	int fftSize = 64;		    // Size of the FFT, in samples
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"fftsize", 1, NULL, 's'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+//	settings.useAnalog = 0;	// No matrix usage by default
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hs:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 's':
+				fftSize = atof(optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &fftSize) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/FFT-audio-in/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,111 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <rtdk.h>
+#include <ne10/NE10.h>					// neon library
+#include <cmath>
+
+int gFFTSize;
+
+int gReadPointer = 0;
+int gWritePointer = 0;
+
+// FFT vars
+static ne10_fft_cpx_float32_t* timeDomainIn;
+static ne10_fft_cpx_float32_t* timeDomainOut;
+static ne10_fft_cpx_float32_t* frequencyDomain;
+static ne10_fft_cfg_float32_t cfg;
+
+bool setup(BelaContext *context, void *userData)
+{
+	// Retrieve a parameter passed in from the initAudio() call
+	gFFTSize = *(int *)userData;
+
+	timeDomainIn = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+	timeDomainOut = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+	frequencyDomain = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+	cfg = ne10_fft_alloc_c2c_float32_neon (gFFTSize);
+
+	memset(timeDomainOut, 0, gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		timeDomainIn[gReadPointer].r = (ne10_float32_t) ((context->audioIn[n*context->audioChannels] +
+															context->audioIn[n*context->audioChannels+1]) * 0.5);
+		timeDomainIn[gReadPointer].i = 0;
+
+		if(++gReadPointer >= gFFTSize)
+		{
+			//FFT
+			ne10_fft_c2c_1d_float32_neon (frequencyDomain, timeDomainIn, cfg, 0);
+
+			//Do frequency domain stuff
+
+			//IFFT
+			ne10_fft_c2c_1d_float32_neon (timeDomainOut, frequencyDomain, cfg, 1);
+
+			gReadPointer = 0;
+			gWritePointer = 0;
+		}
+
+		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
+			context->audioOut[n * context->audioChannels + channel] = (float) timeDomainOut[gWritePointer].r;
+		gWritePointer++;
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+	NE10_FREE(timeDomainIn);
+	NE10_FREE(timeDomainOut);
+	NE10_FREE(frequencyDomain);
+	NE10_FREE(cfg);
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 04-FFT-audio-in
+
+Fast Fourier Transform
+----------------------
+
+This sketch performs an FFT (Fast Fourier Transform) on incoming audio. It uses 
+the NE10 library, included at the top of the file.
+
+Read the documentation on the NE10 library [here](http://projectne10.github.io/Ne10/doc/annotated.html).
+
+The variables `timeDomainIn`, `timeDomainOut` and `frequencyDomain` are 
+variables of the struct `ne10_fft_cpx_float32_t` [http://projectne10.github.io/Ne10/doc/structne10__fft__cpx__float32__t.html](http://projectne10.github.io/Ne10/doc/structne10__fft__cpx__float32__t.html). 
+These are declared at the top of the file, and memory is allocated 
+for them in `setup()`.
+
+In `render()` a `for` loop performs the FFT which is performed on each sample, 
+and the resulting output is placed on each channel.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/FFT-phase-vocoder/SampleData.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,19 @@
+/*
+ * SampleData.h
+ *
+ *  Created on: Nov 5, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef SAMPLEDATA_H_
+#define SAMPLEDATA_H_
+
+// User defined structure to pass between main and rendere complex data retrieved from file
+struct SampleData {
+	float *samples;	// Samples in file
+	int sampleLen;	// Total nume of samples
+};
+
+
+
+#endif /* SAMPLEDATA_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/FFT-phase-vocoder/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,212 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <cstdio>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <unistd.h>
+#include <sys/time.h>
+#include <sndfile.h>				// to load audio files
+#include "SampleData.h"
+#include <Bela.h>
+
+using namespace std;
+
+// Global variables used by getCurrentTime()
+unsigned long long gFirstSeconds, gFirstMicroseconds;
+
+// Load samples from file
+int initFile(string file, SampleData *smp)//float *& smp)
+{
+  	SNDFILE *sndfile ;
+	SF_INFO sfinfo ;
+
+	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
+		cout << "Couldn't open file " << file << endl;
+		return 1;
+	}
+
+	int numChan = sfinfo.channels;
+	if(numChan != 1)
+	{
+		cout << "Error: " << file << " is not a mono file" << endl;
+		return 1;
+	}
+
+	smp->sampleLen = sfinfo.frames * numChan;
+	smp->samples = new float[smp->sampleLen];
+	if(smp == NULL){
+		cout << "Could not allocate buffer" << endl;
+		return 1;
+	}
+
+	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
+	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
+
+	// Pad with zeros in case we couldn't read whole file
+	for(int k = readcount; k <smp->sampleLen; k++)
+		smp->samples[k] = 0;
+
+	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
+		double	scale ;
+		int 	m ;
+
+		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
+		if (scale < 1e-10)
+			scale = 1.0 ;
+		else
+			scale = 32700.0 / scale ;
+		cout << "File samples scale = " << scale << endl;
+
+		for (m = 0; m < smp->sampleLen; m++)
+			smp->samples[m] *= scale;
+	}
+
+	sf_close(sndfile);
+
+	return 0;
+}
+
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+/* Function which returns the time since start of the program
+ * in (fractional) seconds.
+ */
+double getCurrentTime(void) {
+	unsigned long long result;
+	struct timeval tv;
+
+	gettimeofday(&tv, NULL);
+	result = (tv.tv_sec - gFirstSeconds) * 1000000ULL + (tv.tv_usec - gFirstMicroseconds);
+	return (double)result / 1000000.0;
+}
+extern SampleData gSampleData;
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	struct timeval tv;
+	string fileName;			// Name of the sample to load
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"file", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	gSampleData.samples = 0;
+	gSampleData.sampleLen = -1;
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	settings.periodSize = 32; // Larger period size by default, for testing
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				fileName = string((char *)optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	if(fileName.empty()){
+		fileName = "sample.wav";
+	}
+
+
+	// Load file
+	if(initFile(fileName, &gSampleData) != 0)
+	{
+		cout << "Error: unable to load samples " << endl;
+		return -1;
+	}
+
+	if(settings.verbose)
+		cout << "File contains " << gSampleData.sampleLen << " samples" << endl;
+
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &gSampleData) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Initialise time
+	gettimeofday(&tv, NULL);
+	gFirstSeconds = tv.tv_sec;
+	gFirstMicroseconds = tv.tv_usec;
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/FFT-phase-vocoder/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,309 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <rtdk.h>
+#include <ne10/NE10.h>					// NEON FFT library
+#include <cmath>
+#include "SampleData.h"
+#include <Midi.h>
+
+#define BUFFER_SIZE 16384
+
+// TODO: your buffer and counter go here!
+float gInputBuffer[BUFFER_SIZE];
+int gInputBufferPointer = 0;
+float gOutputBuffer[BUFFER_SIZE];
+int gOutputBufferWritePointer = 0;
+int gOutputBufferReadPointer = 0;
+int gSampleCount = 0;
+
+float *gWindowBuffer;
+
+// -----------------------------------------------
+// These variables used internally in the example:
+int gFFTSize = 2048;
+int gHopSize = 512;
+int gPeriod = 512;
+float gFFTScaleFactor = 0;
+
+// FFT vars
+ne10_fft_cpx_float32_t* timeDomainIn;
+ne10_fft_cpx_float32_t* timeDomainOut;
+ne10_fft_cpx_float32_t* frequencyDomain;
+ne10_fft_cfg_float32_t cfg;
+
+// Sample info
+SampleData gSampleData;	// User defined structure to get complex data from main
+int gReadPtr = 0;		// Position of last read sample from file
+
+// Auxiliary task for calculating FFT
+AuxiliaryTask gFFTTask;
+int gFFTInputBufferPointer = 0;
+int gFFTOutputBufferPointer = 0;
+
+void process_fft_background();
+
+
+int gEffect = 0; // change this here or with midi CC
+enum{
+	kBypass,
+	kRobot,
+	kWhisper,
+};
+
+float gDryWet = 1; // mix between the unprocessed and processed sound
+float gPlaybackLive = 0.5f; // mix between the file playback and the live audio input
+float gGain = 1; // overall gain
+float *gInputAudio = NULL;
+Midi midi;
+
+
+void midiCallback(MidiChannelMessage message, void* arg){
+	if(message.getType() == kmmNoteOn){
+		if(message.getDataByte(1) > 0){
+			int note = message.getDataByte(0);
+			float frequency = powf(2, (note-69)/12.f)*440;
+			gPeriod = (int)(44100 / frequency + 0.5);
+			printf("\nnote: %d, frequency: %f, hop: %d\n", note, frequency, gPeriod);
+		}
+	}
+
+	bool shouldPrint = false;
+	if(message.getType() == kmmControlChange){
+		float data = message.getDataByte(1) / 127.0f;
+		switch (message.getDataByte(0)){
+		case 2 :
+			gEffect = (int)(data * 2 + 0.5); // CC2 selects an effect between 0,1,2
+			break;
+		case 3 :
+			gPlaybackLive = data;
+			break;
+		case 4 :
+			gDryWet = data;
+			break;
+		case 5:
+			gGain = data*10;
+			break;
+		default:
+			shouldPrint = true;
+		}
+	}
+	if(shouldPrint){
+		message.prettyPrint();
+	}
+}
+
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+bool setup(BelaContext* context, void* userData)
+{
+	midi.readFrom(0);
+	midi.setParserCallback(midiCallback);
+	// Retrieve a parameter passed in from the initAudio() call
+	gSampleData = *(SampleData *)userData;
+
+	gFFTScaleFactor = 1.0f / (float)gFFTSize;
+	gOutputBufferWritePointer += gHopSize;
+
+	timeDomainIn = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+	timeDomainOut = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+	frequencyDomain = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+	cfg = ne10_fft_alloc_c2c_float32_neon (gFFTSize);
+
+	memset(timeDomainOut, 0, gFFTSize * sizeof (ne10_fft_cpx_float32_t));
+	memset(gOutputBuffer, 0, BUFFER_SIZE * sizeof(float));
+
+	// Allocate buffer to mirror and modify the input
+	gInputAudio = (float *)malloc(context->audioFrames * context->audioChannels * sizeof(float));
+	if(gInputAudio == 0)
+		return false;
+
+	// Allocate the window buffer based on the FFT size
+	gWindowBuffer = (float *)malloc(gFFTSize * sizeof(float));
+	if(gWindowBuffer == 0)
+		return false;
+
+	// Calculate a Hann window
+	for(int n = 0; n < gFFTSize; n++) {
+		gWindowBuffer[n] = 0.5f * (1.0f - cosf(2.0 * M_PI * n / (float)(gFFTSize - 1)));
+	}
+
+	// Initialise auxiliary tasks
+	if((gFFTTask = Bela_createAuxiliaryTask(&process_fft_background, 90, "fft-calculation")) == 0)
+		return false;
+	rt_printf("You are listening to an FFT phase-vocoder with overlap-and-add.\n"
+	"Use Midi Control Change to control:\n"
+	"CC 2: effect type (bypass/robotization/whisperization)\n"
+	"CC 3: mix between recorded sample and live audio input\n"
+	"CC 4: mix between the unprocessed and processed sound\n"
+	"CC 5: gain\n"
+	);
+	return true;
+}
+
+// This function handles the FFT processing in this example once the buffer has
+// been assembled.
+void process_fft(float *inBuffer, int inWritePointer, float *outBuffer, int outWritePointer)
+{
+	// Copy buffer into FFT input
+	int pointer = (inWritePointer - gFFTSize + BUFFER_SIZE) % BUFFER_SIZE;
+	for(int n = 0; n < gFFTSize; n++) {
+		timeDomainIn[n].r = (ne10_float32_t) inBuffer[pointer] * gWindowBuffer[n];
+		timeDomainIn[n].i = 0;
+
+		pointer++;
+		if(pointer >= BUFFER_SIZE)
+			pointer = 0;
+	}
+
+	// Run the FFT
+	ne10_fft_c2c_1d_float32_neon (frequencyDomain, timeDomainIn, cfg, 0);
+
+	switch (gEffect){
+	  case kRobot :
+	  // Robotise the output
+	    for(int n = 0; n < gFFTSize; n++) {
+	      float amplitude = sqrtf(frequencyDomain[n].r * frequencyDomain[n].r + frequencyDomain[n].i * frequencyDomain[n].i);
+	      frequencyDomain[n].r = amplitude;
+	      frequencyDomain[n].i = 0;
+	    }
+	    break;
+	  case kWhisper :
+	    for(int n = 0; n < gFFTSize; n++) {
+	      float amplitude = sqrtf(frequencyDomain[n].r * frequencyDomain[n].r + frequencyDomain[n].i * frequencyDomain[n].i);
+	      float phase = rand()/(float)RAND_MAX * 2 * M_PI;
+	      frequencyDomain[n].r = cosf(phase) * amplitude;
+	      frequencyDomain[n].i = sinf(phase) * amplitude;
+	    }
+	    break;
+	  case kBypass:
+	    //bypass
+	    break;
+	  }
+
+	// Run the inverse FFT
+	ne10_fft_c2c_1d_float32_neon (timeDomainOut, frequencyDomain, cfg, 1);
+	// Overlap-and-add timeDomainOut into the output buffer
+	pointer = outWritePointer;
+	for(int n = 0; n < gFFTSize; n++) {
+		outBuffer[pointer] += (timeDomainOut[n].r) * gFFTScaleFactor;
+		if(isnan(outBuffer[pointer]))
+			rt_printf("outBuffer OLA\n");
+		pointer++;
+		if(pointer >= BUFFER_SIZE)
+			pointer = 0;
+	}
+}
+
+// Function to process the FFT in a thread at lower priority
+void process_fft_background() {
+	process_fft(gInputBuffer, gFFTInputBufferPointer, gOutputBuffer, gFFTOutputBufferPointer);
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+void render(BelaContext* context, void* userData)
+{
+	float* audioOut = context->audioOut;
+	int numAudioFrames = context->audioFrames;
+	int numAudioChannels = context->audioChannels;
+	// ------ this code internal to the demo; leave as is ----------------
+
+	// Prep the "input" to be the sound file played in a loop
+	for(int n = 0; n < numAudioFrames; n++) {
+		if(gReadPtr < gSampleData.sampleLen)
+			gInputAudio[2*n] = gInputAudio[2*n+1] = gSampleData.samples[gReadPtr]*(1-gPlaybackLive) +
+			gPlaybackLive*0.5f*(audioRead(context,n,0)+audioRead(context,n,1));
+		else
+			gInputAudio[2*n] = gInputAudio[2*n+1] = 0;
+		if(++gReadPtr >= gSampleData.sampleLen)
+			gReadPtr = 0;
+	}
+	// -------------------------------------------------------------------
+
+	for(int n = 0; n < numAudioFrames; n++) {
+		gInputBuffer[gInputBufferPointer] = ((gInputAudio[n*numAudioChannels] + gInputAudio[n*numAudioChannels+1]) * 0.5);
+
+		// Copy output buffer to output
+		for(int channel = 0; channel < numAudioChannels; channel++){
+			audioOut[n * numAudioChannels + channel] = gOutputBuffer[gOutputBufferReadPointer] * gGain * gDryWet + (1 - gDryWet) * gInputAudio[n * numAudioChannels + channel];
+		}
+
+		// Clear the output sample in the buffer so it is ready for the next overlap-add
+		gOutputBuffer[gOutputBufferReadPointer] = 0;
+		gOutputBufferReadPointer++;
+		if(gOutputBufferReadPointer >= BUFFER_SIZE)
+			gOutputBufferReadPointer = 0;
+		gOutputBufferWritePointer++;
+		if(gOutputBufferWritePointer >= BUFFER_SIZE)
+			gOutputBufferWritePointer = 0;
+
+		gInputBufferPointer++;
+		if(gInputBufferPointer >= BUFFER_SIZE)
+			gInputBufferPointer = 0;
+
+		gSampleCount++;
+		if(gSampleCount >= gHopSize) {
+			//process_fft(gInputBuffer, gInputBufferPointer, gOutputBuffer, gOutputBufferPointer);
+			gFFTInputBufferPointer = gInputBufferPointer;
+			gFFTOutputBufferPointer = gOutputBufferWritePointer;
+			Bela_scheduleAuxiliaryTask(gFFTTask);
+
+			gSampleCount = 0;
+		}
+	}
+	gHopSize = gPeriod;
+}
+
+// cleanup_render() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in initialise_render().
+
+void cleanup(BelaContext* context, void* userData)
+{
+	NE10_FREE(timeDomainIn);
+	NE10_FREE(timeDomainOut);
+	NE10_FREE(frequencyDomain);
+	NE10_FREE(cfg);
+	free(gInputAudio);
+	free(gWindowBuffer);
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 04_audio_FFT_phase_vocoder
+
+Phase Vocoder
+----------------------
+
+This sketch shows an implementation of a phase vocoder and builds on the previous FFT example.
+Again it uses the NE10 library, included at the top of the file.
+
+Read the documentation on the NE10 library [here](http://projectne10.github.io/Ne10/doc/annotated.html).
+*/
Binary file examples/04-Audio/FFT-phase-vocoder/sample.wav has changed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/bucket-brigade-chorus/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,144 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h> 
+#include <Scope.h>
+#include <cmath>
+
+float gPhase1, gPhase2;
+float gFrequency1, gFrequency2;
+float gInverseSampleRate;
+
+// initialise_render() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+#include <I2c_Codec.h>
+#include <PRU.h>
+extern I2c_Codec *gAudioCodec;
+extern PRU *gPRU;
+float D=5264;
+#define delayLength 256
+float delay[delayLength];
+int writePointer=0;
+int readPointer=writePointer+1;
+AuxiliaryTask updatePll;
+
+void updatePllFunction(){
+//	gPRU->setGPIOTestPin();
+	static int count = 0;
+	while(!gShouldStop){
+		gAudioCodec->setPllD(D);
+		count++;
+		if((count&4095)==0)
+			printf("sampling rate: %f\n",gAudioCodec->getAudioSamplingRate());
+		usleep(100);
+	}
+//	gPRU->clearGPIOTestPin();
+}
+
+bool setup(BelaContext *context, void *userData)
+{
+	gInverseSampleRate = 1.0/context->audioSampleRate;
+	
+	gPhase1 = 0.0;
+	gPhase2 = 0.0;
+	
+	gFrequency1 = 200.0;
+	gFrequency2 = 201.0;
+	updatePll=Bela_createAuxiliaryTask(&updatePllFunction, 91, "update PLL");
+	for(int n=0; n<delayLength; n++){
+		delay[n]=0;
+	}
+	return true; 
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+//	printf("here\n");
+	static bool init = false;
+	if(init == false){
+		Bela_scheduleAuxiliaryTask(updatePll);
+//		gAudioCodec->setPllP(2);
+//		gAudioCodec->setPllR();
+//		gAudioCodec->setAudioSamplingRate(43600);
+//		printf("samplingRate: %f, k: %f\n", gAudioCodec->getAudioSamplingRate(), gAudioCodec->getPllK());
+		init = true;
+	}
+	static int count=0;
+	static float lfoPhase=0;
+	static float feedback=0;
+	int updateRate=1;
+	if((count&(updateRate-1))==0){
+		float amplitude = 8000;
+		float rate = 2;
+		lfoPhase+=rate*2*M_PI*updateRate*context->analogFrames/context->audioSampleRate;
+		D=amplitude+amplitude*sinf(lfoPhase);
+		if((count&255)==0){
+//			rt_printf("frequency: %f\n", gAudioCodec->getAudioSamplingRate());
+//			rt_printf("D: %.0f\n", D);
+//			rt_printf("rate: %f\n", rate);
+//			rt_printf("amplitude: %.3f\n", amplitude);
+//			rt_printf("feedback: %.3f\n\n", feedback);
+		}
+	}
+	count++;
+
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		feedback = 0.4;
+		float input = audioRead(context, n, 0) + audioRead(context, n, 1);
+	    delay[writePointer++] = input + delay[readPointer]*feedback;
+	    float output = (input + 0.9*delay[readPointer++] ) * 0.5;
+		audioWrite(context, n, 0, output);
+		audioWrite(context, n, 1, output);
+		if(writePointer>=delayLength)
+			writePointer-=delayLength;
+		if(readPointer>=delayLength)
+			readPointer-=delayLength;
+
+		gPhase1 += 2.0 * M_PI * gFrequency1 * gInverseSampleRate;
+	    gPhase2 += 2.0 * M_PI * gFrequency2 * gInverseSampleRate;
+		if(gPhase1 > 2.0 * M_PI)
+			gPhase1 -= 2.0 * M_PI;
+		if(gPhase2 > 2.0 * M_PI)
+			gPhase2 -= 2.0 * M_PI;
+	}
+}
+
+// cleanup_render() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in initialise_render().
+
+void cleanup(BelaContext *context, void *userData)
+{
+    
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/filter-FIR/FIRfilter.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,51 @@
+/*
+ * FIRfilter.h
+ *
+ *  Created on: Aug 5, 2014
+ *      Author: Victor Zappi and Andrew McPherson
+ */
+
+#ifndef FIRFILTER_H_
+#define FIRFILTER_H_
+
+
+#include <ne10/NE10.h>
+
+#define FILTER_TAP_NUM 31
+
+// Coefficients for FIR High Pass Filter at 3 KHz
+ne10_float32_t filterTaps[FILTER_TAP_NUM] = {
+		-0.000055,
+		0.000318,
+		0.001401,
+		0.003333,
+		0.005827,
+		0.007995,
+		0.008335,
+		0.004991,
+		-0.003764,
+		-0.018906,
+		-0.040112,
+		-0.065486,
+		-0.091722,
+		-0.114710,
+		-0.130454,
+		0.863946,
+		-0.130454,
+		-0.114710,
+		-0.091722,
+		-0.065486,
+		-0.040112,
+		-0.018906,
+		-0.003764,
+		0.004991,
+		0.008335,
+		0.007995,
+		0.005827,
+		0.003333,
+		0.001401,
+		0.000318,
+		-0.000055
+};
+
+#endif /* FIRFILTER_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/filter-FIR/SampleData.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,19 @@
+/*
+ * SampleData.h
+ *
+ *  Created on: Nov 5, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef SAMPLEDATA_H_
+#define SAMPLEDATA_H_
+
+// User defined structure to pass between main and rendere complex data retrieved from file
+struct SampleData {
+	float *samples;	// Samples in file
+	int sampleLen;	// Total nume of samples
+};
+
+
+
+#endif /* SAMPLEDATA_H_ */
Binary file examples/04-Audio/filter-FIR/longsample.wav has changed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/filter-FIR/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,196 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <string>
+#include <getopt.h>
+#include <sndfile.h>				// to load audio files
+
+#include <Bela.h>
+#include "SampleData.h"
+
+using namespace std;
+
+// Load samples from file
+int initFile(string file, SampleData *smp)//float *& smp)
+{
+	SNDFILE *sndfile ;
+	SF_INFO sfinfo ;
+
+	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
+		cout << "Couldn't open file " << file << endl;
+		return 1;
+	}
+
+	int numChan = sfinfo.channels;
+	if(numChan != 1)
+	{
+		cout << "Error: " << file << " is not a mono file" << endl;
+		return 1;
+	}
+
+	smp->sampleLen = sfinfo.frames * numChan;
+	smp->samples = new float[smp->sampleLen];
+	if(smp == NULL){
+		cout << "Could not allocate buffer" << endl;
+		return 1;
+	}
+
+	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
+	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
+
+	// Pad with zeros in case we couldn't read whole file
+	for(int k = readcount; k <smp->sampleLen; k++)
+		smp->samples[k] = 0;
+
+	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
+		double	scale ;
+		int 	m ;
+
+		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
+		if (scale < 1e-10)
+			scale = 1.0 ;
+		else
+			scale = 32700.0 / scale ;
+		cout << "File samples scale = " << scale << endl;
+
+		for (m = 0; m < smp->sampleLen; m++)
+			smp->samples[m] *= scale;
+	}
+
+	sf_close(sndfile);
+
+	return 0;
+}
+
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	//rt_task_delete ((RT_TASK *) &gTriggerSamplesTask);
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --file [-f] filename:    Name of the file to load (default is \"longsample.wav\")\n";
+	cerr << "   --help [-h]:             Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	string fileName;			// Name of the sample to load
+
+	SampleData sampleData;		// User define structure to pass data retrieved from file to render function
+	sampleData.samples = 0;
+	sampleData.sampleLen = -1;
+
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"file", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				fileName = string((char *)optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	if(fileName.empty()){
+		fileName = "filter/longsample.wav";
+	}
+
+	if(settings.verbose) {
+		cout << "Loading file " << fileName << endl;
+	}
+
+	// Load file
+	if(initFile(fileName, &sampleData) != 0)
+	{
+		cout << "Error: unable to load samples " << endl;
+		return -1;
+	}
+
+	if(settings.verbose)
+		cout << "File contains " << sampleData.sampleLen << " samples" << endl;
+
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &sampleData) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/filter-FIR/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,172 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#define ENABLE_NE10_FIR_FLOAT_NEON	// Define needed for Ne10 library
+
+#include <Bela.h>
+#include <cmath>
+#include <ne10/NE10.h>					// neon library
+#include "SampleData.h"
+#include "FIRfilter.h"
+
+SampleData gSampleData;	// User defined structure to get complex data from main
+int gReadPtr;			// Position of last read sample from file
+
+// filter vars
+ne10_fir_instance_f32_t gFIRfilter;
+ne10_float32_t *gFIRfilterIn;
+ne10_float32_t *gFIRfilterOut;
+ne10_uint32_t blockSize;
+ne10_float32_t *gFIRfilterState;
+
+void initialise_filter(BelaContext *context);
+
+// Task for handling the update of the frequencies using the matrix
+AuxiliaryTask gTriggerSamplesTask;
+
+bool initialise_trigger();
+void trigger_samples();
+
+bool setup(BelaContext *context, void *userData)
+{
+
+	// Retrieve a parameter passed in from the initAudio() call
+	gSampleData = *(SampleData *)userData;
+
+	gReadPtr = -1;
+
+	initialise_filter(context);
+
+	// Initialise auxiliary tasks
+	if(!initialise_trigger())
+		return false;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		float in = 0;
+
+		// If triggered...
+		if(gReadPtr != -1)
+			in += gSampleData.samples[gReadPtr++];	// ...read each sample...
+
+		if(gReadPtr >= gSampleData.sampleLen)
+			gReadPtr = -1;
+
+		gFIRfilterIn[n] = in;
+	}
+
+	ne10_fir_float_neon(&gFIRfilter, gFIRfilterIn, gFIRfilterOut, blockSize);
+
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
+				context->audioOut[n * context->audioChannels + channel] = gFIRfilterOut[n];	// ...and put it in both left and right channel
+	}
+
+
+	// Request that the lower-priority task run at next opportunity
+	Bela_scheduleAuxiliaryTask(gTriggerSamplesTask);
+}
+
+// Initialise NE10 data structures to define FIR filter
+
+void initialise_filter(BelaContext *context)
+{
+	blockSize = context->audioFrames;
+	gFIRfilterState	= (ne10_float32_t *) NE10_MALLOC ((FILTER_TAP_NUM+blockSize-1) * sizeof (ne10_float32_t));
+	gFIRfilterIn = (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
+	gFIRfilterOut = (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
+	ne10_fir_init_float(&gFIRfilter, FILTER_TAP_NUM, filterTaps, gFIRfilterState, blockSize);
+}
+
+
+// Initialise the auxiliary task
+// and print info
+
+bool initialise_trigger()
+{
+	if((gTriggerSamplesTask = Bela_createAuxiliaryTask(&trigger_samples, 50, "bela-trigger-samples")) == 0)
+		return false;
+
+	rt_printf("Press 'a' to trigger sample, 's' to stop\n");
+	rt_printf("Press 'q' to quit\n");
+
+	return true;
+}
+
+// This is a lower-priority call to periodically read keyboard input
+// and trigger samples. By placing it at a lower priority,
+// it has minimal effect on the audio performance but it will take longer to
+// complete if the system is under heavy audio load.
+
+void trigger_samples()
+{
+	// This is not a real-time task!
+	// Cos getchar is a system call, not handled by Xenomai.
+	// This task will be automatically down graded.
+
+	char keyStroke = '.';
+
+	keyStroke =	getchar();
+	while(getchar()!='\n'); // to read the first stroke
+
+	switch (keyStroke)
+	{
+		case 'a':
+			gReadPtr = 0;
+			break;
+		case 's':
+			gReadPtr = -1;
+			break;
+		case 'q':
+			gShouldStop = true;
+			break;
+		default:
+			break;
+	}
+}
+
+
+void cleanup(BelaContext *context, void *userData)
+{
+	delete[] gSampleData.samples;
+
+	NE10_FREE(gFIRfilterState);
+	NE10_FREE(gFIRfilterIn);
+	NE10_FREE(gFIRfilterOut);
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 04-filter-FIR
+
+Finite Impulse Response Filter
+------------------------------
+
+This is an example of a finite impulse response filter implementation.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/filter-IIR/SampleData.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,19 @@
+/*
+ * SampleData.h
+ *
+ *  Created on: Nov 5, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef SAMPLEDATA_H_
+#define SAMPLEDATA_H_
+
+// User defined structure to pass between main and rendere complex data retrieved from file
+struct SampleData {
+	float *samples;	// Samples in file
+	int sampleLen;	// Total nume of samples
+};
+
+
+
+#endif /* SAMPLEDATA_H_ */
Binary file examples/04-Audio/filter-IIR/longsample.wav has changed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/filter-IIR/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,202 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <string>
+#include <getopt.h>
+#include <sndfile.h>				// to load audio files
+
+#include <Bela.h>
+#include "SampleData.h"
+
+using namespace std;
+
+float gCutFreq = 100;
+
+// Load samples from file
+int initFile(string file, SampleData *smp)//float *& smp)
+{
+	SNDFILE *sndfile ;
+	SF_INFO sfinfo ;
+
+	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
+		cout << "Couldn't open file " << file << endl;
+		return 1;
+	}
+
+	int numChan = sfinfo.channels;
+	if(numChan != 1)
+	{
+		cout << "Error: " << file << " is not a mono file" << endl;
+		return 1;
+	}
+
+	smp->sampleLen = sfinfo.frames * numChan;
+	smp->samples = new float[smp->sampleLen];
+	if(smp == NULL){
+		cout << "Could not allocate buffer" << endl;
+		return 1;
+	}
+
+	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
+	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
+
+	// Pad with zeros in case we couldn't read whole file
+	for(int k = readcount; k <smp->sampleLen; k++)
+		smp->samples[k] = 0;
+
+	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
+		double	scale ;
+		int 	m ;
+
+		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
+		if (scale < 1e-10)
+			scale = 1.0 ;
+		else
+			scale = 32700.0 / scale ;
+		cout << "File samples scale = " << scale << endl;
+
+		for (m = 0; m < smp->sampleLen; m++)
+			smp->samples[m] *= scale;
+	}
+
+	sf_close(sndfile);
+
+	return 0;
+}
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	//rt_task_delete ((RT_TASK *) &gTriggerSamplesTask);
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --file [-f] filename:    Name of the file to load (default is \"longsample.wav\")\n";
+	cerr << "   --cutfreq [-c] freq:     Set the cut off frequency of the filter in Hz\n";
+	cerr << "   --help [-h]:             Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	string fileName;			// Name of the sample to load
+
+	SampleData sampleData;		// User define structure to pass data retrieved from file to render function
+	sampleData.samples = 0;
+	sampleData.sampleLen = -1;
+
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"cutfreq", 1, NULL, 'c'},
+		{"file", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:c:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				fileName = string((char *)optarg);
+				break;
+		case 'c':
+				gCutFreq = atof(optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	if(fileName.empty()){
+		fileName = "filter/longsample.wav";
+	}
+
+	if(settings.verbose) {
+		cout << "Loading file " << fileName << endl;
+	}
+
+	// Load file
+	if(initFile(fileName, &sampleData) != 0)
+	{
+		cout << "Error: unable to load samples " << endl;
+		return -1;
+	}
+
+	if(settings.verbose)
+		cout << "File contains " << sampleData.sampleLen << " samples" << endl;
+
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &sampleData) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/filter-IIR/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,230 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>	// to schedule lower prio parallel process
+#include <rtdk.h>
+#include <cmath>
+#include <stdio.h>
+#include "SampleData.h"
+
+SampleData gSampleData;	// User defined structure to get complex data from main
+int gReadPtr;			// Position of last read sample from file
+
+// filter vars
+float gLastX[2];
+float gLastY[2];
+double lb0, lb1, lb2, la1, la2 = 0.0;
+
+// communication vars between the 2 auxiliary tasks
+int gChangeCoeff = 0;
+int gFreqDelta = 0;
+
+void initialise_filter(float freq);
+
+void calculate_coeff(float cutFreq);
+
+bool initialise_aux_tasks();
+
+// Task for handling the update of the frequencies using the matrix
+AuxiliaryTask gChangeCoeffTask;
+
+void check_coeff();
+
+// Task for handling the update of the frequencies using the matrix
+AuxiliaryTask gInputTask;
+
+void read_input();
+
+
+extern float gCutFreq;
+
+
+bool setup(BelaContext *context, void *userData)
+{
+
+	// Retrieve a parameter passed in from the initAudio() call
+	gSampleData = *(SampleData *)userData;
+
+	gReadPtr = -1;
+
+	initialise_filter(200);
+
+	// Initialise auxiliary tasks
+	if(!initialise_aux_tasks())
+		return false;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		float sample = 0;
+		float out = 0;
+
+		// If triggered...
+		if(gReadPtr != -1)
+			sample += gSampleData.samples[gReadPtr++];	// ...read each sample...
+
+		if(gReadPtr >= gSampleData.sampleLen)
+			gReadPtr = -1;
+
+		out = lb0*sample+lb1*gLastX[0]+lb2*gLastX[1]-la1*gLastY[0]-la2*gLastY[1];
+
+		gLastX[1] = gLastX[0];
+		gLastX[0] = out;
+		gLastY[1] = gLastY[0];
+		gLastY[0] = out;
+
+		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
+			context->audioOut[n * context->audioChannels + channel] = out;	// ...and put it in both left and right channel
+
+	}
+
+	// Request that the lower-priority tasks run at next opportunity
+	Bela_scheduleAuxiliaryTask(gChangeCoeffTask);
+	Bela_scheduleAuxiliaryTask(gInputTask);
+}
+
+// First calculation of coefficients
+
+void initialise_filter(float freq)
+{
+	calculate_coeff(freq);
+}
+
+
+// Calculate the filter coefficients
+// second order low pass butterworth filter
+
+void calculate_coeff(float cutFreq)
+{
+	// Initialise any previous state (clearing buffers etc.)
+	// to prepare for calls to render()
+	float sampleRate = 44100;
+	double f = 2*M_PI*cutFreq/sampleRate;
+	double denom = 4+2*sqrt(2)*f+f*f;
+	lb0 = f*f/denom;
+	lb1 = 2*lb0;
+	lb2 = lb0;
+	la1 = (2*f*f-8)/denom;
+	la2 = (f*f+4-2*sqrt(2)*f)/denom;
+	gLastX[0] = gLastX [1] = 0;
+	gLastY[0] = gLastY[1] = 0;
+
+}
+
+
+// Initialise the auxiliary tasks
+// and print info
+
+bool initialise_aux_tasks()
+{
+	if((gChangeCoeffTask = Bela_createAuxiliaryTask(&check_coeff, 90, "bela-check-coeff")) == 0)
+		return false;
+
+	if((gInputTask = Bela_createAuxiliaryTask(&read_input, 50, "bela-read-input")) == 0)
+		return false;
+
+	rt_printf("Press 'a' to trigger sample, 's' to stop\n");
+	rt_printf("Press 'z' to low down cut-off freq of 100 Hz, 'x' to raise it up\n");
+	rt_printf("Press 'q' to quit\n");
+
+	return true;
+}
+
+// Check if cut-off freq has been changed
+// and new coefficients are needed
+
+void check_coeff()
+{
+	if(gChangeCoeff == 1)
+	{
+		gCutFreq += gFreqDelta;
+		gCutFreq = gCutFreq < 0 ? 0 : gCutFreq;
+		gCutFreq = gCutFreq > 22050 ? 22050 : gCutFreq;
+
+		rt_printf("Cut-off frequency: %f\n", gCutFreq);
+
+		calculate_coeff(gCutFreq);
+		gChangeCoeff = 0;
+	}
+}
+
+// This is a lower-priority call to periodically read keyboard input
+// and trigger samples. By placing it at a lower priority,
+// it has minimal effect on the audio performance but it will take longer to
+// complete if the system is under heavy audio load.
+
+void read_input()
+{
+	// This is not a real-time task!
+	// Cos getchar is a system call, not handled by Xenomai.
+	// This task will be automatically down graded.
+
+	char keyStroke = '.';
+
+	keyStroke =	getchar();
+	while(getchar()!='\n'); // to read the first stroke
+
+	switch (keyStroke)
+	{
+		case 'a':
+			gReadPtr = 0;
+			break;
+		case 's':
+			gReadPtr = -1;
+			break;
+		case 'z':
+			gChangeCoeff = 1;
+			gFreqDelta = -100;
+			break;
+		case 'x':
+			gChangeCoeff = 1;
+			gFreqDelta = 100;
+			break;
+		case 'q':
+			gShouldStop = true;
+			break;
+		default:
+			break;
+	}
+}
+
+
+void cleanup(BelaContext *context, void *userData)
+{
+	delete[] gSampleData.samples;
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 04-filter-IIR
+
+Infinite Impulse Response Filter
+------------------------------
+
+This is an example of a infinite impulse response filter implementation.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/measure-noisefloor/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,120 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+extern int gBufferSize;
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --buffer-size [-b] size  Set the analysis buffer size\n";
+	cerr << "   --help [-h]:             Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"buffer-size", 1, NULL, 'b'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// By default use a longer period size because latency is not an issue
+	settings.periodSize = 32;
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hb:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'b':
+				gBufferSize = atoi(optarg);
+				break;
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+	
+	if(gBufferSize < settings.periodSize)
+		gBufferSize = settings.periodSize;
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, 0) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/measure-noisefloor/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,178 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <cmath>
+
+int gBufferSize = 8192;
+
+// Double buffers to hold samples for noise analysis
+float *gReadBuffers[10], *gWriteBuffers[10];
+float *gBuffers0[10], *gBuffers1[10];
+
+int gWriteBufferPointers[10], gReadBufferPointers[10];
+
+// Task to analyse and print results which would otherwise be too slow for render()
+AuxiliaryTask gAnalysisTask;
+
+void analyseResults();
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+bool setup(BelaContext *context, void *userData)
+{	
+	// Clear the filter data structures
+	for(int i = 0; i < 10; i++) {
+		gReadBufferPointers[i] = gWriteBufferPointers[i] = 0;
+		gBuffers0[i] = new float[gBufferSize];
+		gBuffers1[i] = new float[gBufferSize];		
+		gWriteBuffers[i] = gBuffers0[i];
+		gReadBuffers[i] = gBuffers1[i];
+		if(gBuffers0[i] == 0 || gBuffers1[i] == 0) {
+			rt_printf("Error allocating buffer %d\n", i);
+			return false;
+		}
+	}
+	
+	gAnalysisTask = Bela_createAuxiliaryTask(analyseResults, 50, "bela-analyse-results");
+
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	bool bufferIsFull = false;	// Whether at least one buffer has filled
+	
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		// Store audio inputs in buffer
+		for(unsigned int ch = 0; ch < context->audioChannels; ch++) {
+			if(gWriteBufferPointers[ch] < gBufferSize) {
+				gWriteBuffers[ch][gWriteBufferPointers[ch]] = 
+					context->audioIn[n * context->audioChannels + ch];
+				gWriteBufferPointers[ch]++;
+				if(gWriteBufferPointers[ch] >= gBufferSize)
+					bufferIsFull = true;
+			}
+		}
+	}
+	
+	if(context->analogChannels != 0) {
+		for(unsigned int n = 0; n < context->analogFrames; n++) {
+			// Store analog inputs in buffer, starting at channel 2
+			for(unsigned int ch = 0; ch < context->analogChannels; ch++) {
+				if(gWriteBufferPointers[ch + 2] < gBufferSize) {
+					gWriteBuffers[ch + 2][gWriteBufferPointers[ch + 2]] = 
+						context->analogIn[n * context->analogChannels + ch];
+					gWriteBufferPointers[ch + 2]++;
+					if(gWriteBufferPointers[ch + 2] >= gBufferSize)
+						bufferIsFull = true;
+				}
+				
+				// Set all analog outputs to halfway point so they can be more
+				// easily measured for noise
+				context->analogOut[n * context->analogChannels + ch] = 0.5;
+			}
+		}	
+	}
+	
+
+	if(bufferIsFull) {
+		// Swap buffers and reset write pointers
+		for(int ch = 0; ch < 10; ch++) {
+			gReadBufferPointers[ch] = gWriteBufferPointers[ch];
+			gWriteBufferPointers[ch] = 0;
+			
+			if(gReadBuffers[ch] == gBuffers0[ch]) {
+				gReadBuffers[ch] = gBuffers1[ch];
+				gWriteBuffers[ch] = gBuffers0[ch];
+			}
+			else {
+				gReadBuffers[ch] = gBuffers0[ch];
+				gWriteBuffers[ch] = gBuffers1[ch];				
+			}
+		}
+		
+		Bela_scheduleAuxiliaryTask(gAnalysisTask);
+	}
+}
+
+void analyseResults()
+{
+	rt_printf("\e[1;1H\e[2J");	// Command to clear the screen
+
+	// Print the analysis results. channels 0-1 are audio, channels 2-9 are analog
+	for(int ch = 0; ch < 10; ch++) {
+		// Skip unused channels
+		if(gReadBufferPointers[ch] == 0)
+			continue;
+		
+		float mean = 0;
+		for(int n = 0; n < gReadBufferPointers[ch]; n++) {
+			mean += gReadBuffers[ch][n];
+		}
+		mean /= (float)gReadBufferPointers[ch];
+		
+		float rms = 0;
+		for(int n = 0; n < gReadBufferPointers[ch]; n++) {
+			rms += (gReadBuffers[ch][n] - mean) * (gReadBuffers[ch][n] - mean);
+		}				
+		rms = sqrtf(rms / (float)gReadBufferPointers[ch]);
+		
+		if(ch == 0)
+			rt_printf("Audio In L:  ");
+		else if(ch == 1)
+			rt_printf("Audio In R:  ");
+		else
+			rt_printf("Analog In %d: ", ch - 2);
+		
+		rt_printf("Noise %6.1fdB    DC offset %6.4f (%6.1fdB)    window size: %d\n", 
+					20.0f * log10f(rms),
+					mean, 
+					20.0f * log10f(fabsf(mean)),
+					gReadBufferPointers[ch]);
+	}
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+	for(int i = 0; i < 10; i++) {
+		delete gBuffers0[i];
+		delete gBuffers1[i];
+	}
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/oscillator-bank/audio_routines.S	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,161 @@
+@
+@ audio_routines.S
+@
+@ NEON-based functions for time-critical audio processing
+@
+@ Andrew McPherson 2014
+@ Queen Mary University of London
+@
+
+	.syntax unified
+	.arch armv7-a
+	.fpu neon
+
+@ 	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
+@							  int activePartialNum, int lookupTableSize,
+@							  float *phases, float *frequencies, float *amplitudes,
+@							  float *freqDerivatives, float *ampDerivatives,
+@							  float *lookupTable);
+
+@ Registers:
+@    r0: numAudioFrames        How many frames to render
+@    r1: audioOut              Buffer for audio output samples [stereo]
+@    r2: activePartialNum      How many active partials to render
+@    r3: lookupTableSize       Size of lookup table
+@    ---- other arguments start on the stack and are moved: -----
+@    r4: phases                Phase of each oscillator (pointer)
+@    r5: frequencies           Normalised frequency of each oscillator (pointer)
+@    r6: amplitudes            Normalised amplitude of each oscillator (pointer)
+@    r7: freqDerivatives       Derivative of frequency for each oscillator (pointer)
+@    r8: ampDerivatives        Derivative of amplitude for each oscillator (pointer)
+@    r9: lookupTable           Lookup table containing one oscillation
+@
+@ Alignment requirements:
+@    audioOut: 8-byte boundary
+@    phases: 16-byte boundary
+@    frequencies: 16-byte boundary
+@    amplitudes: 16-byte boundary
+@    freqDerivatives: 16-byte bounary
+@    ampDerivatives: 16-byte boundary
+@    lookupTable: 4-byte boundary (TODO: check this)
+
+	.align	2
+	.global	oscillator_bank_neon
+	.thumb
+	.thumb_func
+	.type	oscillator_bank_neon, %function
+oscillator_bank_neon:
+
+
+dSample		.dn		D6.F32
+qPhases		.qn		Q8.F32
+dPhases_0	.dn		D16.F32
+dPhases_1	.dn		D17.F32
+qFreqs		.qn		Q9.F32
+dFreqs_0	.dn		D18.F32
+dFreqs_1	.dn		D19.F32
+qAmps		.qn		Q10.F32
+dAmps_0		.dn		D20.F32
+dAmps_1		.dn		D21.F32
+qFreqDs		.qn		Q11.F32
+dFreqDs_0	.dn		D22.F32
+dFreqDs_1	.dn		D23.F32
+qAmpDs		.qn		Q12.F32
+dAmpDs_0	.dn		D24.F32
+dAmpDs_1	.dn		D25.F32
+
+qBaseInts	.qn		Q13.U32		@ Base indexes: unsigned ints x4
+dBaseInts_0	.dn		D26.U32
+dBaseInts_1	.dn		D27.U32
+qFractions  .qn     Q14.F32		@ Fraction indexes: floats x4
+qTableBase	.qn		Q15.U32		@ Base of lookup table
+
+	cmp r0, #0					@ Check for trivial case 1: zero frames
+	it eq
+	bxeq lr						@ Return if that's the case (otherwise might have odd behaviour)
+	cmp r2, #4					@ Check for trivial case 2: zero oscillators
+	it lt
+	bxlt lr						@ Return if that's the case
+
+	push {r4-r11}				@ Now arguments start 32 bytes above SP
+    add r11, sp, #32			@ Pointer to 32 bytes into the stack
+    ldm r11, {r4-r9}			@ Load 6 arguments into registers
+
+	vdup qTableBase, r9			@ Move lookup table base index into 4 ints
+
+	@ Outer loop: iterate over the number of oscillators, choosing 4 at a
+	@ time to work with.
+oscbank_oscillator_loop:
+	vld1 {dPhases_0, dPhases_1}, [r4]		@ no increment; will store at end of sample loop
+	vld1 {dFreqs_0, dFreqs_1}, [r5]
+	vld1 {dAmps_0, dAmps_1}, [r6]
+	vld1 {dFreqDs_0, dFreqDs_1}, [r7]!		@ increment; won't update at end of sample loop
+	vld1 {dAmpDs_0, dAmpDs_1}, [r8]!
+
+	push {r0-r1,r4-r8}
+	@ --- inner loop: iterate over the number of samples ---
+oscbank_sample_loop:
+	vcvt qBaseInts, qPhases		     		@ Take floor(phases)
+	vmov q2.f32, #1.0						@ Load 1.0 into every slot of q2
+	vshl q0.U32, qBaseInts, #2				@ Shift the indexes left 2 (*4 for float addressing)
+	vcvt qFractions, qBaseInts				@ int back to float
+	vadd q0.U32, q0.U32, qTableBase			@ Find memory addresses
+
+	vmov r4, r5, d0							@ Move two indexes to ARM registers
+	vmov r6, r7, d1							@ Move two more indexes to ARM registers
+	vsub qFractions, qPhases, qFractions	@ fraction = phase - floor(phase)
+
+	vldr.64	d0, [r4]						@ Load two consecutive floats at each location
+	vldr.64 d1, [r5]						@ These hold the previous and following samples in the table
+	vldr.64	d2, [r6]						@ TODO: check whether these work at 4-byte alignment
+	vldr.64 d3, [r7]
+
+	@ Format at this point:
+	@ Osc0(before) Osc0(after) Osc1(before) Osc1(after) Osc2(before) Osc2(after) Osc3(before) Osc3(after)
+	@ We want:
+	@ Osc0(before) Osc1(before) Osc2(before) Osc3(before) Osc0(after) Osc1(after) Osc2(after) Osc3(after)
+
+	vuzp.32 q0, q1							@ Now q0 contains before, q1 contains after
+	vsub q2.f32, q2.f32, qFractions			@ q2 = 1.0 - fraction
+	vmul q1.f32, q1.f32, qFractions			@ q1 = fraction * after
+	vmul q0.f32, q0.f32, q2.f32				@ q0 = (1.0 - fraction) * before
+
+	vadd qPhases, qPhases, qFreqs			@ Update phases
+	vadd qFreqs, qFreqs, qFreqDs			@ Update frequencies
+
+	vadd q0.f32, q0.f32, q1.f32				@ Add two interpolated components to get the final sample
+	vdup q2.u32, r3							@ Put lookup table size into each element of q2
+	vcvt qBaseInts, qPhases					@ Take floor of new phases
+	vmul q0.f32, q0.f32, qAmps				@ Multiply samples by current amplitude
+
+	vld1 dSample, [r1]						@ Load the current stereo samples
+	vpadd d2.f32, d0.f32, d1.f32			@ Pairwise accumulate q0 (output sample) into d2
+
+	vand q2, q2, qBaseInts					@ Logical AND of new phase int leaves 1 bit set only if phase >= table size
+	vpadd d3.f32, d2.f32, d2.f32			@ Pairwise accumulate d2 into d0 --> d0[0] and d0[1] both hold total of 4 oscillators
+	vadd qAmps, qAmps, qAmpDs				@ Update amplitudes
+	vcvt q0.f32, q2.u32						@ Convert int back to float after AND operation
+
+	vadd  dSample, dSample, d3.f32			@ Add oscillator outputs to each channel
+
+	subs r0, r0, #1							@ numFrames--
+	vsub qPhases, qPhases, q0.f32			@ Keep phases in table range
+	vst1 dSample, [r1]!						@ Store back in buffer and increment by 8
+
+	it gt
+	bgt oscbank_sample_loop					@ Loop if numFrames > 0
+
+	@ --- end inner loop ---
+	pop {r0-r1,r4-r8}						@ Restore registers: restores audioOut and numFrames, among others
+
+	vst1 {dPhases_0, dPhases_1}, [r4]!		@ Store phases back to array
+	vst1 {dFreqs_0, dFreqs_1}, [r5]!		@ Store frequencies back to array
+	vst1 {dAmps_0, dAmps_1}, [r6]!			@ Store amplitudes back to array
+											@ No need to update r7, r8
+
+	subs r2, r2, #4							@ numPartials -= 4
+	it  gt
+	bgt oscbank_oscillator_loop	@ Loop if numPartials > 0
+
+    pop {r4-r11}
+	bx lr
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/oscillator-bank/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,132 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+int gNumOscillators = 32;
+int gWavetableLength = 1024;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --num-oscillators [-n] oscs: Set the number of oscillators to use (default: 32)\n";
+	cerr << "   --wavetable [-w] length:     Set the wavetable length in samples (default: 1024)\n";
+	cerr << "   --help [-h]:                 Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"num-oscillators", 1, NULL, 'n'},
+		{"wavetable", 1, NULL, 'w'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hn:w:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'n':
+				gNumOscillators = atoi(optarg);
+				if(gNumOscillators <= 0) {
+					usage(basename(argv[0]));
+					exit(0);
+				}
+				break;
+		case 'w':
+				gWavetableLength = atoi(optarg);
+				if(gWavetableLength < 4)
+					gWavetableLength = 4;
+				if(gWavetableLength > 16384)
+					gWavetableLength = 16384;
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, 0) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	if(settings.verbose) {
+		cout << "--> Using " << gNumOscillators << " oscillators and wavetable of " << gWavetableLength << " samples\n";
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/oscillator-bank/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,248 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <rtdk.h>
+#include <cstdlib>
+#include <cmath>
+#include <cstring>
+#include <time.h>
+
+const float kMinimumFrequency = 20.0f;
+const float kMaximumFrequency = 8000.0f;
+
+float *gWavetable;		// Buffer holding the precalculated sine lookup table
+float *gPhases;			// Buffer holding the phase of each oscillator
+float *gFrequencies;	// Buffer holding the frequencies of each oscillator
+float *gAmplitudes;		// Buffer holding the amplitudes of each oscillator
+float *gDFrequencies;	// Buffer holding the derivatives of frequency
+float *gDAmplitudes;	// Buffer holding the derivatives of amplitude
+
+float gAudioSampleRate;
+int gSampleCount;		// Sample counter for indicating when to update frequencies
+float gNewMinFrequency;
+float gNewMaxFrequency;
+
+// Task for handling the update of the frequencies using the matrix
+AuxiliaryTask gFrequencyUpdateTask;
+
+// These settings are carried over from main.cpp
+// Setting global variables is an alternative approach
+// to passing a structure to userData in setup()
+
+extern int gNumOscillators;
+extern int gWavetableLength;
+
+void recalculate_frequencies();
+
+extern "C" {
+	// Function prototype for ARM assembly implementation of oscillator bank
+	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
+							  int activePartialNum, int lookupTableSize,
+							  float *phases, float *frequencies, float *amplitudes,
+							  float *freqDerivatives, float *ampDerivatives,
+							  float *lookupTable);
+}
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+bool setup(BelaContext *context, void *userData)
+{
+	srandom(time(NULL));
+
+	if(context->audioChannels != 2) {
+		rt_printf("Error: this example needs stereo audio enabled\n");
+		return false;
+	}
+
+	// Initialise the sine wavetable
+	if(posix_memalign((void **)&gWavetable, 8, (gWavetableLength + 1) * sizeof(float))) {
+		rt_printf("Error allocating wavetable\n");
+		return false;
+	}
+	for(int n = 0; n < gWavetableLength + 1; n++)
+		gWavetable[n] = sinf(2.0 * M_PI * (float)n / (float)gWavetableLength);
+
+	// Allocate the other buffers
+	if(posix_memalign((void **)&gPhases, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating phase buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gFrequencies, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating frequency buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gAmplitudes, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating amplitude buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gDFrequencies, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating frequency derivative buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gDAmplitudes, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating amplitude derivative buffer\n");
+		return false;
+	}
+
+	// Initialise buffer contents
+
+	float freq = kMinimumFrequency;
+	float increment = (kMaximumFrequency - kMinimumFrequency) / (float)gNumOscillators;
+
+	for(int n = 0; n < gNumOscillators; n++) {
+		gPhases[n] = 0.0;
+
+		if(context->analogFrames == 0) {
+			// Random frequencies when used without matrix
+			gFrequencies[n] = kMinimumFrequency + (kMaximumFrequency - kMinimumFrequency) * ((float)random() / (float)RAND_MAX);
+		}
+		else {
+			// Constant spread of frequencies when used with matrix
+			gFrequencies[n] = freq;
+			freq += increment;
+		}
+
+		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
+		gFrequencies[n] *= (float)gWavetableLength / context->audioSampleRate;
+		gAmplitudes[n] = ((float)random() / (float)RAND_MAX) / (float)gNumOscillators;
+		gDFrequencies[n] = gDAmplitudes[n] = 0.0;
+	}
+
+	increment = 0;
+	freq = 440.0;
+
+	for(int n = 0; n < gNumOscillators; n++) {
+		// Update the frequencies to a regular spread, plus a small amount of randomness
+		// to avoid weird phase effects
+		float randScale = 0.99 + .02 * (float)random() / (float)RAND_MAX;
+		float newFreq = freq * randScale;
+
+		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
+		gFrequencies[n] = newFreq * (float)gWavetableLength / context->audioSampleRate;
+
+		freq += increment;
+	}
+
+	// Initialise auxiliary tasks
+	if((gFrequencyUpdateTask = Bela_createAuxiliaryTask(&recalculate_frequencies, 85, "bela-update-frequencies")) == 0)
+		return false;
+
+	//for(int n = 0; n < gNumOscillators; n++)
+	//	rt_printf("%f\n", gFrequencies[n]);
+
+	gAudioSampleRate = context->audioSampleRate;
+	gSampleCount = 0;
+
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	// Initialise buffer to 0
+	memset(context->audioOut, 0, 2 * context->audioFrames * sizeof(float));
+
+	// Render audio frames
+	oscillator_bank_neon(context->audioFrames, context->audioOut,
+			gNumOscillators, gWavetableLength,
+			gPhases, gFrequencies, gAmplitudes,
+			gDFrequencies, gDAmplitudes,
+			gWavetable);
+
+	if(context->analogFrames != 0 && (gSampleCount += context->audioFrames) >= 128) {
+		gSampleCount = 0;
+		gNewMinFrequency = map(context->analogIn[0], 0, 1.0, 1000.0f, 8000.0f);
+		gNewMaxFrequency = map(context->analogIn[1], 0, 1.0, 1000.0f, 8000.0f);
+
+		// Make sure max >= min
+		if(gNewMaxFrequency < gNewMinFrequency) {
+			float temp = gNewMaxFrequency;
+			gNewMaxFrequency = gNewMinFrequency;
+			gNewMinFrequency = temp;
+		}
+
+		// Request that the lower-priority task run at next opportunity
+		//Bela_scheduleAuxiliaryTask(gFrequencyUpdateTask);
+	}
+}
+
+// This is a lower-priority call to update the frequencies which will happen
+// periodically when the matrix is enabled. By placing it at a lower priority,
+// it has minimal effect on the audio performance but it will take longer to
+// complete if the system is under heavy audio load.
+
+void recalculate_frequencies()
+{
+	float freq = gNewMinFrequency;
+	float increment = (gNewMaxFrequency - gNewMinFrequency) / (float)gNumOscillators;
+
+	for(int n = 0; n < gNumOscillators; n++) {
+		// Update the frequencies to a regular spread, plus a small amount of randomness
+		// to avoid weird phase effects
+		float randScale = 0.99 + .02 * (float)random() / (float)RAND_MAX;
+		float newFreq = freq * randScale;
+
+		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
+		gFrequencies[n] = newFreq * (float)gWavetableLength / gAudioSampleRate;
+
+		freq += increment;
+	}
+}
+
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+	free(gWavetable);
+	free(gPhases);
+	free(gFrequencies);
+	free(gAmplitudes);
+	free(gDFrequencies);
+	free(gDAmplitudes);
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 04-oscillator-bank
+
+Oscillator Bank
+----------------------
+
+These files demonstrate an oscillator bank implemented in assembly code 
+that is used as part of the d-box project.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/samples/SampleData.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,19 @@
+/*
+ * SampleData.h
+ *
+ *  Created on: Nov 5, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef SAMPLEDATA_H_
+#define SAMPLEDATA_H_
+
+// User defined structure to pass between main and rendere complex data retrieved from file
+struct SampleData {
+	float *samples;	// Samples in file
+	int sampleLen;	// Total nume of samples
+};
+
+
+
+#endif /* SAMPLEDATA_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/samples/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,195 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <string>
+#include <getopt.h>
+#include <sndfile.h>				// to load audio files
+
+#include <Bela.h>
+#include "SampleData.h"
+
+using namespace std;
+
+// Load samples from file
+int initFile(string file, SampleData *smp)//float *& smp)
+{
+	SNDFILE *sndfile ;
+	SF_INFO sfinfo ;
+
+	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
+		cout << "Couldn't open file " << file << endl;
+		return 1;
+	}
+
+	int numChan = sfinfo.channels;
+	if(numChan != 1)
+	{
+		cout << "Error: " << file << " is not a mono file" << endl;
+		return 1;
+	}
+
+	smp->sampleLen = sfinfo.frames * numChan;
+	smp->samples = new float[smp->sampleLen];
+	if(smp == NULL){
+		cout << "Could not allocate buffer" << endl;
+		return 1;
+	}
+
+	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
+	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
+
+	// Pad with zeros in case we couldn't read whole file
+	for(int k = readcount; k <smp->sampleLen; k++)
+		smp->samples[k] = 0;
+
+	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
+		double	scale ;
+		int 	m ;
+
+		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
+		if (scale < 1e-10)
+			scale = 1.0 ;
+		else
+			scale = 32700.0 / scale ;
+		cout << "File samples scale = " << scale << endl;
+
+		for (m = 0; m < smp->sampleLen; m++)
+			smp->samples[m] *= scale;
+	}
+
+	sf_close(sndfile);
+
+	return 0;
+}
+
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	//rt_task_delete ((RT_TASK *) &gTriggerSamplesTask);
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --file [-f] filename:    Name of the file to load (default is \"sample.wav\")\n";
+	cerr << "   --help [-h]:             Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	string fileName;			// Name of the sample to load
+
+	SampleData sampleData;		// User define structure to pass data retrieved from file to render function
+	sampleData.samples = 0;
+	sampleData.sampleLen = -1;
+
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"file", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				fileName = string((char *)optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	if(fileName.empty()){
+		fileName = "samples/sample.wav";
+	}
+
+	if(settings.verbose) {
+		cout << "Loading file " << fileName << endl;
+	}
+
+	// Load file
+	if(initFile(fileName, &sampleData) != 0)
+	{
+		cout << "Error: unable to load samples " << endl;
+		return -1;
+	}
+
+	if(settings.verbose)
+		cout << "File contains " << sampleData.sampleLen << " samples" << endl;
+
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &sampleData) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/04-Audio/samples/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,147 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <cmath>
+#include "SampleData.h"
+
+SampleData gSampleData;	// User defined structure to get complex data from main
+int gReadPtr;			// Position of last read sample from file
+
+// Task for handling the update of the frequencies using the matrix
+AuxiliaryTask gTriggerSamplesTask;
+
+bool initialise_trigger();
+void trigger_samples();
+
+bool setup(BelaContext *context, void *userData)
+{
+
+	// Retrieve a parameter passed in from the initAudio() call
+	gSampleData = *(SampleData *)userData;
+
+	gReadPtr = -1;
+
+	// Initialise auxiliary tasks
+	if(!initialise_trigger())
+		return false;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		float out = 0;
+
+		// If triggered...
+		if(gReadPtr != -1)
+			out += gSampleData.samples[gReadPtr++];	// ...read each sample...
+
+		if(gReadPtr >= gSampleData.sampleLen)
+			gReadPtr = -1;
+
+		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
+			context->audioOut[n * context->audioChannels + channel] = out;	// ...and put it in both left and right channel
+	}
+
+	// Request that the lower-priority task run at next opportunity
+	Bela_scheduleAuxiliaryTask(gTriggerSamplesTask);
+}
+
+// Initialise the auxiliary task
+// and print info
+
+bool initialise_trigger()
+{
+	if((gTriggerSamplesTask = Bela_createAuxiliaryTask(&trigger_samples, 50, "bela-trigger-samples")) == 0)
+		return false;
+
+	rt_printf("Press 'a' to trigger sample, 's' to stop\n");
+	rt_printf("Press 'q' to quit\n");
+
+	return true;
+}
+
+// This is a lower-priority call to periodically read keyboard input
+// and trigger samples. By placing it at a lower priority,
+// it has minimal effect on the audio performance but it will take longer to
+// complete if the system is under heavy audio load.
+
+void trigger_samples()
+{
+	// This is not a real-time task!
+	// Cos getchar is a system call, not handled by Xenomai.
+	// This task will be automatically down graded.
+
+	char keyStroke = '.';
+
+	keyStroke =	getchar();
+	while(getchar()!='\n'); // to read the first stroke
+
+	switch (keyStroke)
+	{
+		case 'a':
+			gReadPtr = 0;
+			break;
+		case 's':
+			gReadPtr = -1;
+			break;
+		case 'q':
+			gShouldStop = true;
+			break;
+		default:
+			break;
+	}
+}
+
+
+void cleanup(BelaContext *context, void *userData)
+{
+	delete[] gSampleData.samples;
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 04-samples
+
+Playback WAV files
+------------------
+
+This sketch shows how to playback audio samples from a buffer.
+
+An audio file is loaded into a buffer `SampleData` as `gSampleData`. This is 
+accessed with a read pointer that is incremented at audio rate within the render 
+function: `out += gSampleData.samples[gReadPtr++]`.
+
+Note that the read pointer is stopped from incrementing past the length of the 
+`gSampleData`. This is achieved by comparing the read pointer value against the 
+sample length which we can access as follows: `gSampleData.sampleLen`.
+
+The sample is triggered by keyboard input: (a) starts sample playback, (s) 
+stops sample playback. The triggering is treated as a lower priority task than 
+the audio. You can see this at the bottom of the render function: 
+`Bela_scheduleAuxiliaryTask(gTriggerSamplesTask)`;
+*/
Binary file examples/04-Audio/samples/sample.wav has changed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/05-Communication/OSC/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,111 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <OSCServer.h>
+#include <OSCClient.h>
+
+OSCServer oscServer;
+OSCClient oscClient;
+
+// this example is designed to be run alongside resources/osc/osc.js
+
+// parse messages recieved by OSC Server
+// msg is Message class of oscpkt: http://gruntthepeon.free.fr/oscpkt/
+void parseMessage(oscpkt::Message msg){
+    
+    rt_printf("recieved message to: %s\n", msg.addressPattern().c_str());
+    
+    int intArg;
+    float floatArg;
+    if (msg.match("/osc-test").popInt32(intArg).popFloat(floatArg).isOkNoMoreArgs()){
+        rt_printf("recieved int %i and float %f\n", intArg, floatArg);
+    }
+    
+}
+
+bool setup(BelaContext *context, void *userData)
+{
+    // setup the OSC server to recieve on port 7562
+    oscServer.setup(7562);
+    // setup the OSC client to send on port 7563
+    oscClient.setup(7563);
+    
+    // the following code sends an OSC message to address /osc-setup
+    // then waits 1 second for a reply on /osc-setup-reply
+    bool handshakeRecieved = false;
+    oscClient.sendMessageNow(oscClient.newMessage.to("/osc-setup").end());
+    oscServer.recieveMessageNow(1000);
+    while (oscServer.messageWaiting()){
+        if (oscServer.popMessage().match("/osc-setup-reply")){
+            handshakeRecieved = true;
+        }
+    }
+    
+    if (handshakeRecieved){
+        rt_printf("handshake recieved!\n");
+    } else {
+        rt_printf("timeout!\n");
+    }
+    
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+    // recieve OSC messages, parse them, and send back an acknowledgment
+    while (oscServer.messageWaiting()){
+        parseMessage(oscServer.popMessage());
+        oscClient.queueMessage(oscClient.newMessage.to("/osc-acknowledge").add(5).add(4.2f).add(std::string("OSC message recieved")).end());
+    }
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 05-OSC
+
+Open Sound Control
+------------------
+
+This example shows an implementation of OSC (Open Sound Control) which was 
+developed at UC Berkeley Center for New Music and Audio Technology (CNMAT).
+
+It is designed to be run alongside resources/osc/osc.js
+
+The OSC server port on which to receive is set in `setup()` 
+via `oscServer.setup()`. Likewise the OSC client port on which to 
+send is set in `oscClient.setup()`.
+
+In `setup()` an OSC message to address `/osc-setup`, it then waits 
+1 second for a reply on `/osc-setup-reply`.
+
+in `render()` the code receives OSC messages, parses them, and sends 
+back an acknowledgment.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/05-Communication/basic-midi/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,182 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <Midi.h>
+#include <stdlib.h>
+#include <rtdk.h>
+#include <cmath>
+
+float gFreq;
+float gPhaseIncrement = 0;
+bool gIsNoteOn = 0;
+int gVelocity = 0;
+float gSamplingPeriod = 0;
+
+void midiMessageCallback(MidiChannelMessage message, void* arg){
+	if(arg != NULL){
+		rt_printf("Message from midi port %d: ", *(int*)arg);
+	}
+	message.prettyPrint();
+	if(message.getType() == kmmNoteOn){
+		gFreq = powf(2, (message.getDataByte(0)-69)/12.0f) * 440;
+		gVelocity = message.getDataByte(1);
+		gPhaseIncrement = 2 * M_PI * gFreq * gSamplingPeriod;
+		gIsNoteOn = gVelocity > 0;
+		rt_printf("v0:%f, ph: %6.5f, gVelocity: %d\n", gFreq, gPhaseIncrement, gVelocity);
+	}
+}
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+Midi midi;
+int gMidiPort0 = 0;
+bool setup(BelaContext *context, void *userData)
+{
+	midi.readFrom(gMidiPort0);
+	midi.writeTo(gMidiPort0);
+	midi.enableParser(true);
+	midi.setParserCallback(midiMessageCallback, &gMidiPort0);
+	if(context->analogFrames == 0) {
+		rt_printf("Error: this example needs the analog I/O to be enabled\n");
+		return false;
+	}
+	gSamplingPeriod = 1/context->audioSampleRate;
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+
+enum {kVelocity, kNoteOn, kNoteNumber};
+void render(BelaContext *context, void *userData)
+{
+// one way of getting the midi data is to parse them yourself
+//	(you should set midi.enableParser(false) above):
+/*
+	static midi_byte_t noteOnStatus = 0x90; //on channel 1
+	static int noteNumber = 0;
+	static int waitingFor = kNoteOn;
+	static int playingNote = -1;
+	int message;
+	while ((message = midi.getInput()) >= 0){
+		rt_printf("%d\n", message);
+		switch(waitingFor){
+		case kNoteOn:
+			if(message == noteOnStatus){
+				waitingFor = kNoteNumber;
+			}
+			break;
+		case kNoteNumber:
+			if((message & (1<<8)) == 0){
+				noteNumber = message;
+				waitingFor = kVelocity;
+			}
+			break;
+		case kVelocity:
+			if((message & (1<<8)) == 0){
+				int _velocity = message;
+				waitingFor = kNoteOn;
+				// "monophonic" behaviour, with priority to the latest note on
+				// i.e.: a note off from a previous note does not stop the current note
+				// still you might end up having a key down and no note being played if you pressed and released another
+				// key in the meantime
+				if(_velocity == 0 && noteNumber == playingNote){
+					noteOn = false;
+					playingNote = -1;
+					velocity = _velocity;
+				} else if (_velocity > 0) {
+					noteOn = true;
+					velocity = _velocity;
+					playingNote = noteNumber;
+					f0 = powf(2, (playingNote-69)/12.0f) * 440;
+					phaseIncrement = 2 * M_PI * f0 / context->audioSampleRate;
+				}
+				rt_printf("NoteOn: %d, NoteNumber: %d, velocity: %d\n", noteOn, noteNumber, velocity);
+			}
+			break;
+		}
+	}
+*/
+	/*
+	int num;
+	//alternatively, you can use the built-in parser (only processes channel messages at the moment).
+	while((num = midi.getParser()->numAvailableMessages()) > 0){
+		static MidiChannelMessage message;
+		message = midi.getParser()->getNextChannelMessage();
+		message.prettyPrint();
+		if(message.getType() == kmmNoteOn){
+			f0 = powf(2, (message.getDataByte(0)-69)/12.0f) * 440;
+			velocity = message.getDataByte(1);
+			phaseIncrement = 2 * M_PI * f0 / context->audioSampleRate;
+			noteOn = velocity > 0;
+			rt_printf("v0:%f, ph: %6.5f, velocity: %d\n", f0, phaseIncrement, gVelocity);
+		}
+	}
+	 */
+	// the following block toggles the LED on an Owl pedal
+	// and asks the pedal to return the status of the LED
+	// using MIDI control changes
+	for(unsigned int n = 0; n < context->analogFrames; n++){
+		static int count = 0;
+		static bool state = 0;
+		analogWriteOnce(context, n, 1, state);
+		if(count % 40000 == 0){
+			state = !state;
+			midi_byte_t bytes[6] = {176, 30, (char)(state*127), 176, 67, 30}; // toggle the OWL led and ask for the led status
+			midi.writeOutput(bytes, 6);
+		}
+		count++;
+	}
+	for(unsigned int n = 0; n < context->audioFrames; n++){
+		if(gIsNoteOn == 1){
+			static float phase = 0;
+			phase += gPhaseIncrement;
+			if(phase > 2 * M_PI)
+				phase -= 2 * M_PI;
+			float value = sinf(phase) * gVelocity/128.0f;
+			audioWrite(context, n, 0, value);
+			audioWrite(context, n, 1, value);
+		} else {
+			audioWrite(context, n, 0, 0);
+			audioWrite(context, n, 1, 0);
+		}
+	}
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/05-Communication/basic-network/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,114 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+	Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+	Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <unistd.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	float frequency = 440.0;		// Frequency of oscillator
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"frequency", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				frequency = atof(optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &frequency) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/05-Communication/basic-network/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,101 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+//#include <rtdk.h>
+#include <cmath>
+#include <NetworkSend.h>
+#include <ReceiveAudioThread.h>
+#include <Utilities.h>
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+NetworkSend networkSend;
+ReceiveAudioThread receive;
+float gFrequency;
+float gInverseSampleRate;
+float gPhase;
+bool setup(BelaContext *context, void *userData)
+{
+	// Retrieve a parameter passed in from the initAudio() call
+	gFrequency = *(float *)userData;
+
+	networkSend.setup(context->audioSampleRate, context->audioFrames, 0, 9999, "192.168.7.1");
+	receive.init(10000, context->audioFrames, 0);
+	receive.startThread();
+	gInverseSampleRate = 1.0 / context->audioSampleRate;
+	gPhase = 0;
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		float out = 0.7f * sinf(gPhase);
+		gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate;
+		if(gPhase > 2.0 * M_PI)
+			gPhase -= 2.0 * M_PI;
+
+		networkSend.log(out);
+		float in;
+		int ret = receive.getSamplesSrc(&in, 1, 1);
+		for(unsigned int channel = 0; channel < context->audioChannels; channel++){
+			audioWrite(context, n, channel, in);
+		}
+	}
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 05-basic-network
+
+Networking 
+----------
+
+This sketch allows you to send audio and sensor data over UDP to a 
+DAW on the host. The host needs to run Udpioplugin which you can find 
+[here](https://code.soundsoftware.ac.uk/projects/udpioplugin). 
+
+Note that this sketch and the accompanying plugin are still in testing.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/06-Sensors/basic-pulseIn/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,73 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <PulseIn.h>
+#include <stdlib.h>
+#include <rtdk.h>
+#include <cmath>
+
+PulseIn pulseIn;
+int gPulseInPin = 0;
+int gDigitalOutPin = 1;
+int gPulseLength = 1234;
+int gSamplesBetweenPulses = 10000;
+
+bool setup(BelaContext *context, void *userData)
+{
+	pinMode(context, 0, gDigitalOutPin, OUTPUT);
+	pulseIn.init(context, gPulseInPin, 1); //third parameter is direction
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+	static bool pulseOut = 0;
+	static int count = 0;
+	for(unsigned int n = 0; n < context->digitalFrames; n++){
+		// detect if a pulse just ended
+		int duration = pulseIn.hasPulsed(context, n);
+		if(duration > 0){
+			rt_printf("duration = %d\n", duration);
+		}
+
+		// generate a rectangular waveform as a test signal.
+		// Connect gDigitalOutPin to gPulseInPin
+		// to verify that the detected pulse length is gPulseLength
+		if(count == gPulseLength ){
+			pulseOut = false;
+		}
+		if(count == (gPulseLength + gSamplesBetweenPulses)){
+			pulseOut = true;
+			count = 0;
+		}
+		digitalWrite(context, n, gDigitalOutPin, pulseOut);
+		count++;
+	}
+}
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/06-Sensors/mpr121/I2C_MPR121.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,176 @@
+/*
+ * I2C_MPR121.cpp
+ *
+ *  Created on: Oct 14, 2013
+ *      Author: Victor Zappi
+ */
+
+
+#include "I2C_MPR121.h"
+
+I2C_MPR121::I2C_MPR121() {
+
+}
+
+boolean I2C_MPR121::begin(uint8_t bus, uint8_t i2caddr) {
+  _i2c_address = i2caddr;
+	
+  if(initI2C_RW(bus, i2caddr, 0) > 0)
+	  return false;
+
+  // soft reset
+  writeRegister(MPR121_SOFTRESET, 0x63);
+  usleep(1000);
+  //delay(1);
+  for (uint8_t i=0; i<0x7F; i++) {
+  //  Serial.print("$"); Serial.print(i, HEX); 
+  //  Serial.print(": 0x"); Serial.println(readRegister8(i));
+  }
+  
+
+  writeRegister(MPR121_ECR, 0x0);
+
+  uint8_t c = readRegister8(MPR121_CONFIG2);
+  
+  if (c != 0x24) {
+	  rt_printf("MPR121 read 0x%x instead of 0x24\n", c);
+	  return false;
+  }
+
+  setThresholds(12, 6);
+  writeRegister(MPR121_MHDR, 0x01);
+  writeRegister(MPR121_NHDR, 0x01);
+  writeRegister(MPR121_NCLR, 0x0E);
+  writeRegister(MPR121_FDLR, 0x00);
+
+  writeRegister(MPR121_MHDF, 0x01);
+  writeRegister(MPR121_NHDF, 0x05);
+  writeRegister(MPR121_NCLF, 0x01);
+  writeRegister(MPR121_FDLF, 0x00);
+
+  writeRegister(MPR121_NHDT, 0x00);
+  writeRegister(MPR121_NCLT, 0x00);
+  writeRegister(MPR121_FDLT, 0x00);
+
+  writeRegister(MPR121_DEBOUNCE, 0);
+  writeRegister(MPR121_CONFIG1, 0x10); // default, 16uA charge current
+  writeRegister(MPR121_CONFIG2, 0x20); // 0.5uS encoding, 1ms period
+
+//  writeRegister(MPR121_AUTOCONFIG0, 0x8F);
+
+//  writeRegister(MPR121_UPLIMIT, 150);
+//  writeRegister(MPR121_TARGETLIMIT, 100); // should be ~400 (100 shifted)
+//  writeRegister(MPR121_LOWLIMIT, 50);
+  // enable all electrodes
+  writeRegister(MPR121_ECR, 0x8F);  // start with first 5 bits of baseline tracking
+
+  return true;
+}
+
+void I2C_MPR121::setThresholds(uint8_t touch, uint8_t release) {
+  for (uint8_t i=0; i<12; i++) {
+    writeRegister(MPR121_TOUCHTH_0 + 2*i, touch);
+    writeRegister(MPR121_RELEASETH_0 + 2*i, release);
+  }
+}
+
+uint16_t  I2C_MPR121::filteredData(uint8_t t) {
+  if (t > 12) return 0;
+  return readRegister16(MPR121_FILTDATA_0L + t*2);
+}
+
+uint16_t  I2C_MPR121::baselineData(uint8_t t) {
+  if (t > 12) return 0;
+  uint16_t bl = readRegister8(MPR121_BASELINE_0 + t);
+  return (bl << 2);
+}
+
+uint16_t  I2C_MPR121::touched(void) {
+  uint16_t t = readRegister16(MPR121_TOUCHSTATUS_L);
+  return t & 0x0FFF;
+}
+
+/*********************************************************************/
+
+
+uint8_t I2C_MPR121::readRegister8(uint8_t reg) {
+    unsigned char inbuf, outbuf;
+    struct i2c_rdwr_ioctl_data packets;
+    struct i2c_msg messages[2];
+
+    /*
+     * In order to read a register, we first do a "dummy write" by writing
+     * 0 bytes to the register we want to read from.  This is similar to
+     * the packet in set_i2c_register, except it's 1 byte rather than 2.
+     */
+    outbuf = reg;
+    messages[0].addr  = 0x5A;
+    messages[0].flags = 0;
+    messages[0].len   = sizeof(outbuf);
+    messages[0].buf   = &outbuf;
+
+    /* The data will get returned in this structure */
+    messages[1].addr  = 0x5A;
+    messages[1].flags = I2C_M_RD/* | I2C_M_NOSTART*/;
+    messages[1].len   = sizeof(inbuf);
+    messages[1].buf   = &inbuf;
+
+    /* Send the request to the kernel and get the result back */
+    packets.msgs      = messages;
+    packets.nmsgs     = 2;
+    if(ioctl(i2C_file, I2C_RDWR, &packets) < 0) {
+        rt_printf("Unable to send data");
+        return 0;
+    }
+
+    return inbuf;
+}
+
+uint16_t I2C_MPR121::readRegister16(uint8_t reg) {
+    unsigned char inbuf[2], outbuf;
+    struct i2c_rdwr_ioctl_data packets;
+    struct i2c_msg messages[2];
+
+    /*
+     * In order to read a register, we first do a "dummy write" by writing
+     * 0 bytes to the register we want to read from.  This is similar to
+     * the packet in set_i2c_register, except it's 1 byte rather than 2.
+     */
+    outbuf = reg;
+    messages[0].addr  = _i2c_address;
+    messages[0].flags = 0;
+    messages[0].len   = sizeof(outbuf);
+    messages[0].buf   = &outbuf;
+
+    /* The data will get returned in this structure */
+    messages[1].addr  = _i2c_address;
+    messages[1].flags = I2C_M_RD/* | I2C_M_NOSTART*/;
+    messages[1].len   = sizeof(inbuf);
+    messages[1].buf   = inbuf;
+
+    /* Send the request to the kernel and get the result back */
+    packets.msgs      = messages;
+    packets.nmsgs     = 2;
+    if(ioctl(i2C_file, I2C_RDWR, &packets) < 0) {
+        rt_printf("Unable to send data");
+        return 0;
+    }
+
+    return (uint16_t)inbuf[0] | (((uint16_t)inbuf[1]) << 8);
+}
+
+/**************************************************************************/
+/*!
+    @brief  Writes 8-bits to the specified destination register
+*/
+/**************************************************************************/
+void I2C_MPR121::writeRegister(uint8_t reg, uint8_t value) {
+	uint8_t buf[2] = { reg, value };
+
+	if(write(i2C_file, buf, 2) != 2)
+	{
+		cout << "Failed to write register " << (int)reg << " on MPR121\n";
+		return;
+	}
+}
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/06-Sensors/mpr121/I2C_MPR121.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,82 @@
+/*
+ * MPR121 Bela demo
+ * 
+ * Andrew McPherson
+ * Based on Adafruit library by Limor Fried/Ladyada
+ */
+
+#ifndef I2CTK_H_
+#define I2CTK_H_
+
+#include <I2c.h>
+#include "Utilities.h"
+
+typedef bool boolean;
+
+#define MPR121_I2CADDR_DEFAULT 0x5A
+
+#define MPR121_TOUCHSTATUS_L 0x00
+#define MPR121_TOUCHSTATUS_H 0x01
+#define MPR121_FILTDATA_0L  0x04
+#define MPR121_FILTDATA_0H  0x05
+#define MPR121_BASELINE_0   0x1E
+#define MPR121_MHDR         0x2B
+#define MPR121_NHDR         0x2C
+#define MPR121_NCLR         0x2D
+#define MPR121_FDLR         0x2E
+#define MPR121_MHDF         0x2F
+#define MPR121_NHDF         0x30
+#define MPR121_NCLF         0x31
+#define MPR121_FDLF         0x32
+#define MPR121_NHDT         0x33
+#define MPR121_NCLT         0x34
+#define MPR121_FDLT         0x35
+
+#define MPR121_TOUCHTH_0    0x41
+#define MPR121_RELEASETH_0    0x42
+#define MPR121_DEBOUNCE 0x5B
+#define MPR121_CONFIG1 0x5C
+#define MPR121_CONFIG2 0x5D
+#define MPR121_CHARGECURR_0 0x5F
+#define MPR121_CHARGETIME_1 0x6C
+#define MPR121_ECR 0x5E
+#define MPR121_AUTOCONFIG0 0x7B
+#define MPR121_AUTOCONFIG1 0x7C
+#define MPR121_UPLIMIT   0x7D
+#define MPR121_LOWLIMIT  0x7E
+#define MPR121_TARGETLIMIT  0x7F
+
+#define MPR121_GPIODIR  0x76
+#define MPR121_GPIOEN  0x77
+#define MPR121_GPIOSET  0x78
+#define MPR121_GPIOCLR  0x79
+#define MPR121_GPIOTOGGLE  0x7A
+
+#define MPR121_SOFTRESET 0x80
+
+class I2C_MPR121 : public I2c
+{
+public:
+	// Hardware I2C
+	I2C_MPR121();
+
+	boolean begin(uint8_t bus = 1, uint8_t i2caddr = MPR121_I2CADDR_DEFAULT);
+
+	uint16_t filteredData(uint8_t t);
+	uint16_t  baselineData(uint8_t t);
+
+	uint8_t readRegister8(uint8_t reg);
+	uint16_t readRegister16(uint8_t reg);
+	void writeRegister(uint8_t reg, uint8_t value);
+	uint16_t touched(void);
+ 
+ 	void setThresholds(uint8_t touch, uint8_t release);
+	
+	int readI2C() { return 0; } // Unused
+	
+private:
+	int _i2c_address;
+};
+
+
+#endif /* I2CTK_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/06-Sensors/mpr121/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,171 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+#include <cmath>
+#include <rtdk.h>
+#include "I2C_MPR121.h"
+
+// How many pins there are
+#define NUM_TOUCH_PINS 12
+
+// Define this to print data to terminal
+#undef DEBUG_MPR121
+
+// Change this to change how often the MPR121 is read (in Hz)
+int readInterval = 50;
+
+// Change this threshold to set the minimum amount of touch
+int threshold = 40;
+
+// This array holds the continuous sensor values
+int sensorValue[NUM_TOUCH_PINS];
+
+// ---- test code stuff -- can be deleted for your example ----
+
+// 12 notes of a C major scale...
+float gFrequencies[NUM_TOUCH_PINS] = {261.63, 293.66, 329.63, 349.23, 392.00, 440.00, 493.88, 523.25, 587.33, 659.25, 698.25, 783.99};
+
+// This is internal stuff for the demo
+float gNormFrequencies[NUM_TOUCH_PINS];
+float gPhases[NUM_TOUCH_PINS] = {0};
+
+// ---- internal stuff -- do not change -----
+
+I2C_MPR121 mpr121;			// Object to handle MPR121 sensing
+AuxiliaryTask i2cTask;		// Auxiliary task to read I2C
+
+int readCount = 0;			// How long until we read again...
+int readIntervalSamples = 0; // How many samples between reads
+
+void readMPR121();
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+bool setup(BelaContext *context, void *userData)
+{
+	if(!mpr121.begin(1, 0x5A)) {
+		rt_printf("Error initialising MPR121\n");
+		return false;
+	}
+	
+	i2cTask = Bela_createAuxiliaryTask(readMPR121, 50, "bela-mpr121");
+	readIntervalSamples = context->audioSampleRate / readInterval;
+	
+	for(int i = 0; i < NUM_TOUCH_PINS; i++) {
+		gNormFrequencies[i] = 2.0 * M_PI * gFrequencies[i] / context->audioSampleRate;
+	}
+	
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numAnalogFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	for(int n = 0; n < context->audioFrames; n++) {
+		// Keep this code: it schedules the touch sensor readings
+		if(++readCount >= readIntervalSamples) {
+			readCount = 0;
+			Bela_scheduleAuxiliaryTask(i2cTask);
+		}
+		
+		float sample = 0.0;
+		
+		// This code can be replaced with your favourite audio code
+		for(int i = 0; i < NUM_TOUCH_PINS; i++) {
+			float amplitude = sensorValue[i] / 400.0;
+			
+			// Prevent clipping
+			if(amplitude > 0.5)
+				amplitude = 0.5;
+			
+			sample += amplitude * sinf(gPhases[i]);
+			gPhases[i] += gNormFrequencies[i];
+			if(gPhases[i] > 2.0 * M_PI)
+				gPhases[i] -= 2.0 * M_PI;
+		}
+		
+		for(int ch = 0; ch < context->audioChannels; ch++)
+			context->audioOut[context->audioChannels * n + ch] = sample;
+	}
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+	// Nothing to do here
+}
+
+
+// Auxiliary task to read the I2C board
+void readMPR121()
+{
+	for(int i = 0; i < NUM_TOUCH_PINS; i++) {
+		sensorValue[i] = -(mpr121.filteredData(i) - mpr121.baselineData(i));
+		sensorValue[i] -= threshold;
+		if(sensorValue[i] < 0)
+			sensorValue[i] = 0;
+#ifdef DEBUG_MPR121
+		rt_printf("%d ", sensorValue[i]);
+#endif
+	}
+#ifdef DEBUG_MPR121
+	rt_printf("\n");
+#endif
+	
+	// You can use this to read binary on/off touch state more easily
+	//rt_printf("Touched: %x\n", mpr121.touched());
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 06-capacitive-touch
+
+Capacitive touch sensing with MPR121
+---------------------------
+
+This sketch allows you to hook up an MPR121 capactive touch sensing device
+to Bela, for example the SparkFun Capacitive Touch Sensor Breakout - MPR121.
+The breakout board gives you 12 electrode connections.
+
+To get this working with Bela you need to connect the breakout board to the I2C
+terminal on the Bela board. See the Pin guide for details of which pin is which.
+
+The sensor data will then be available for you to use in the array 
+`sensorValue[NUM_TOUCH_PINS]`.
+*/
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/07-DataLogging/basic-writeFile/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,115 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <unistd.h>
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	float frequency = 440.0;	// Frequency of oscillator
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"frequency", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				frequency = atof(optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &frequency) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/07-DataLogging/basic-writeFile/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,100 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h> 
+#include <cmath>
+#include <WriteFile.h>
+
+float gPhase1, gPhase2;
+float gFrequency1, gFrequency2;
+float gInverseSampleRate;
+
+WriteFile file1;
+WriteFile file2;
+
+bool setup(BelaContext *context, void *userData)
+{
+	gInverseSampleRate = 1.0/context->audioSampleRate;
+	file1.init("out1.m"); //set the file name to write to
+	file1.setHeader("myvar=[\n"); //set a line to be printed at the beginning of the file
+	file1.setFooter("];\n"); //set a line to be printed at the end of the file
+	file1.setEcho(true); // enable echo to the console (as well as to the file)
+	file1.setFormat("%.5f %.10f %f\n"); // set the format that you want to use for your output. Please use %f only (with modifiers)
+	file2.init("out2.m");
+	file2.setHeader("input=[\n");
+	file2.setFooter("];\n");
+	file2.setEcho(false);
+	file2.setFormat("%f\n");
+	gPhase1 = 0.0;
+	gPhase2 = 0.0;
+
+	gFrequency1 = 200.0;
+	gFrequency2 = 201.0;
+	return true; 
+}
+
+void render(BelaContext *context, void *userData)
+{
+	static int count = 0;
+	if((count&16383) == 0){
+    	file2.log(context->audioIn, context->audioFrames); //write the input buffer every so often
+	}
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+	    float chn1 = sinf(gPhase1);
+	    float chn2 = sinf(gPhase2);
+	    gPhase1 += 2.0 * M_PI * gFrequency1 * gInverseSampleRate;
+	    gPhase2 += 2.0 * M_PI * gFrequency2 * gInverseSampleRate;
+		if(gPhase1 > 2.0 * M_PI)
+			gPhase1 -= 2.0 * M_PI;
+		if(gPhase2 > 2.0 * M_PI)
+			gPhase2 -= 2.0 * M_PI;
+		if( (count&511) == 0){
+			file1.log(chn1);
+			file1.log(chn2);
+			file1.log(count);
+		}
+		count++;
+	}
+}
+
+// cleanup_render() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in initialise_render().
+
+void cleanup(BelaContext *context, void *userData)
+{
+    
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 07-write-file
+
+Writing data to a file
+---------------------------
+
+This sketch demonstrates how to log values from within a project for later processing or analysis.
+
+*/
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/07-DataLogging/loggingSensors/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,115 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <unistd.h>
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	float frequency = 440.0;	// Frequency of oscillator
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"frequency", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				frequency = atof(optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &frequency) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/07-DataLogging/loggingSensors/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,73 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h> 
+#include <cmath>
+#include <WriteFile.h>
+
+WriteFile file1;
+WriteFile file2;
+
+bool setup(BelaContext *context, void *userData)
+{
+	file1.init("out.bin"); //set the file name to write to
+	file1.setEchoInterval(1000);
+	file1.setFileType(kBinary);
+	file1.setFormat("%.4f %.4f\n"); // set the format that you want to use for your output. Please use %f only (with modifiers). When in binary mode, this is used only for echoing to console
+	file2.init("out.m"); //set the file name to write to
+	file2.setHeader("myvar=[\n"); //set one or more lines to be printed at the beginning of the file
+	file2.setFooter("];\n"); //set one or more lines to be printed at the end of the file
+	file2.setFormat("%.4f\n"); // set the format that you want to use for your output. Please use %f only (with modifiers)
+	file2.setFileType(kText);
+	file2.setEchoInterval(10000); // only print to the console 1 line every other 10000
+	return true; 
+}
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->analogFrames; n++) {
+			file1.log(&(context->analogIn[n*context->analogFrames]), 2); // log an array of values
+			file2.log(context->analogIn[n*context->analogFrames]); // log a single value
+	}
+}
+
+// cleanup_render() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in initialise_render().
+
+void cleanup(BelaContext *context, void *userData)
+{
+    
+}
+
+/* ------------ Project Explantation ------------ */
+
+/**
+\example 07-logging-sensors
+
+Logging Sensor Data
+---------------------------
+
+This sketch demonstrates how to log sensor data for later processing or analysis.
+*/
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/basicAnalogIn/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,30 @@
+#N canvas 203 356 637 339 10;
+#X obj 63 170 osc~ 440;
+#X obj 63 223 dac~;
+#X obj 63 191 *~;
+#X text 440 95 ADC 3: Analog In 0;
+#X text 34 41 Basic analog in (or 'theremin');
+#X text 34 51 ===============================;
+#X text 121 105 << Use first two analog in's;
+#X text 440 49 ADC routing:;
+#X text 440 57 ------------;
+#X text 440 105 ADC 4: Analog In 1;
+#X text 440 115 ADC 5: Analog In 2;
+#X text 440 125 ADC 6: Analog In 3;
+#X text 440 135 ADC 7: Analog In 4;
+#X text 440 145 ADC 8: Analog In 5;
+#X text 440 155 ADC 9: Analog In 6;
+#X text 440 165 ADC 10: Analog In 7;
+#X text 440 75 ADC 1: Audio In L;
+#X text 440 85 ADC 2: Audio In R;
+#X obj 63 149 *~ 880;
+#X obj 63 106 adc~ 3 4;
+#X text 355 277 ---------------------------------;
+#X text 355 267 @krighxz / BELA / heavy / 12/2015;
+#X text 379 286 beaglert.cc / enzienaudio.com;
+#X connect 0 0 2 0;
+#X connect 2 0 1 0;
+#X connect 2 0 1 1;
+#X connect 18 0 0 0;
+#X connect 19 0 18 0;
+#X connect 19 1 2 1;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/basicAnalogOut/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,30 @@
+#N canvas 275 504 679 362 10;
+#X text 148 124 << Use first two analog in's;
+#X text 431 57 ------------;
+#X text 44 51 Basic analog out;
+#X text 44 61 ================;
+#X text 431 95 DAC 3: Analog Out 0;
+#X text 431 105 DAC 4: Analog Out 1;
+#X text 431 115 DAC 5: Analog Out 2;
+#X text 431 125 DAC 6: Analog Out 3;
+#X text 431 145 DAC 8: Analog Out 5;
+#X text 431 135 DAC 7: Analog Out 4;
+#X text 431 155 DAC 9: Analog Out 6;
+#X text 431 165 DAC 10: Analog Out 7;
+#X text 431 85 DAC 2: Audio Out R;
+#X text 431 75 DAC 1: Audio Out L;
+#X obj 92 201 *~;
+#X obj 92 159 *~ 10;
+#X obj 92 180 osc~ 1;
+#X obj 35 242 dac~ 1 2 3;
+#X text 143 241 << Output to first analog out;
+#X text 431 49 DAC routing:;
+#X text 432 289 ---------------------------------;
+#X text 432 279 @krighxz / BELA / heavy / 12/2015;
+#X text 456 298 beaglert.cc / enzienaudio.com;
+#X obj 92 125 adc~ 3 4;
+#X connect 14 0 17 2;
+#X connect 15 0 16 0;
+#X connect 16 0 14 0;
+#X connect 23 0 15 0;
+#X connect 23 1 14 1;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/circularBuffer/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,79 @@
+#N canvas 436 490 738 494 10;
+#X obj 135 167 loadbang;
+#X msg 135 212 1;
+#X obj 183 234 / 44.1;
+#X obj 135 253 metro;
+#X obj 135 286 f;
+#X obj 160 286 + 64;
+#X obj 160 308 % 65536, f 8;
+#X msg 160 329 start \$1;
+#X obj 160 351 tabwrite~ circbuf;
+#X obj 363 303 - 32768;
+#X obj 363 325 + 65536;
+#X obj 363 347 % 65536;
+#X obj 342 371 +~ 0;
+#X msg 381 236 0;
+#X obj 342 258 phasor~;
+#X obj 342 392 tabread4~ circbuf;
+#X obj 341 415 dac~;
+#N canvas 422 781 312 126 buf 0;
+#N canvas 0 22 450 278 (subpatch) 0;
+#X array circbuf 65536 float 2;
+#X coords 0 1 65535 -1 256 64 1 0 0;
+#X restore 23 28 graph;
+#X restore 129 441 pd buf;
+#X obj 32 101 osc~ 440;
+#X obj 342 213 samplerate~;
+#X obj 134 189 t b b b b;
+#X text 30 82 audio input;
+#X text 219 310 write pointer;
+#X text 412 349 read pointer;
+#X obj 342 282 *~ 16;
+#X obj 342 236 / 16;
+#X obj 183 214 f 16;
+#X obj 363 189 r \$0-blocksize;
+#X obj 204 186 r \$0-blocksize;
+#X obj 394 259 r \$0-blocksize;
+#X obj 390 123 s \$0-blocksize;
+#X text 34 13 VIRTUAL CIRCULAR BUFFER;
+#X text 34 23 =======================;
+#X obj 390 55 loadbang;
+#X msg 390 77 16;
+#X text 517 454 ---------------------------------;
+#X text 517 444 @krighxz / BELA / heavy / 12/2015;
+#X text 541 463 beaglert.cc / enzienaudio.com;
+#X text 426 78 << replace with target blocksize;
+#X text 446 90 BELA default: 16;
+#X text 446 102 PD default: 64;
+#X connect 0 0 20 0;
+#X connect 1 0 3 0;
+#X connect 2 0 3 1;
+#X connect 3 0 4 0;
+#X connect 4 0 5 0;
+#X connect 5 0 6 0;
+#X connect 5 0 9 0;
+#X connect 6 0 7 0;
+#X connect 6 0 4 1;
+#X connect 7 0 8 0;
+#X connect 9 0 10 0;
+#X connect 10 0 11 0;
+#X connect 11 0 12 1;
+#X connect 12 0 15 0;
+#X connect 13 0 14 1;
+#X connect 14 0 24 0;
+#X connect 15 0 16 0;
+#X connect 15 0 16 1;
+#X connect 18 0 8 0;
+#X connect 19 0 25 0;
+#X connect 20 0 1 0;
+#X connect 20 1 26 0;
+#X connect 20 2 13 0;
+#X connect 20 3 19 0;
+#X connect 24 0 12 0;
+#X connect 25 0 14 0;
+#X connect 26 0 2 0;
+#X connect 27 0 25 1;
+#X connect 28 0 26 1;
+#X connect 29 0 24 1;
+#X connect 33 0 34 0;
+#X connect 34 0 30 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/digital/bela_digital-help.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,97 @@
+#N canvas 121 46 877 628 10;
+#X obj 64 94 r bela_digitalIn11;
+#X obj 219 94 r bela_digitalIn12;
+#X obj 386 107 adc~ 13 14;
+#X text 49 64 Digital in \, message rate;
+#X text 214 69 Digital in \, message rate;
+#X text 394 85 Digital ins \, signal rate;
+#X obj 32 283 s bela_digitalOut15;
+#X text 13 309 Digital out \, message rate;
+#X obj 396 593 s bela_setDigital;
+#X obj 396 533 loadbang;
+#X msg 431 401 in 11;
+#X text 524 390 First argument is the pin direction Second argument
+is the pin number Third argument is the rate \, can be "~" or "sig"
+for signal-rate or "message" for message-rate. Defaults to message-rate.
+;
+#X msg 431 458 in 13 ~;
+#X msg 432 429 in 12 message;
+#X msg 517 506 disable 17;
+#X obj 386 139 snapshot~;
+#X obj 500 134 snapshot~;
+#X obj 529 56 loadbang;
+#X obj 384 181 print digitalIn13;
+#X obj 500 183 print digitalIn14;
+#X obj 497 105 metro 200;
+#X text 525 474 To stop using a pin (e.g.: to save CPU) \, send a disable
+message containing the pin number;
+#X obj 224 280 dac~ 16;
+#X obj 22 232 metro 500;
+#X obj 32 257 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 224 253 phasor~ 300;
+#X text 160 234 Nice \, anti-aliased square wave;
+#X obj 22 207 loadbang;
+#X text 15 181 Blink that LED!;
+#X obj 411 306 dac~ 17;
+#X obj 405 248 osc~ 300;
+#X obj 470 248 phasor~ 169;
+#X obj 457 278 -~ 0.5;
+#X text 403 210 Sum signals together for unpredictable effects;
+#X obj 612 303 dac~ 18;
+#X text 198 305 Digital out \, signal rate;
+#X text 378 325 Digital out \, signal rate;
+#X text 588 325 Digital out \, signal rate;
+#X obj 608 222 adc~;
+#X obj 612 249 *~ 0.5;
+#X obj 612 276 +~ 0.5;
+#X text 585 203 The ultimate bit-crusher;
+#X msg 426 555 in 11 \, in 12 \, in 13 ~ \, in 14 ~ \, out 15 \, out
+16 ~ \, out 17 ~ \, out 18 ~;
+#X text 23 377 Each digital channel can be configured as either an
+input or an output. Send messages to bela_setDigital to configure the
+digitalPins (needed before using them). Pins settings can be modified
+at will during execution. Message-rate input pins receive messages
+only when the input changes. Message-rate output pins will write to
+the output the most recent value you send them. Signal-rate inputs
+and outputs are handled as regular audio channels. Outputs are thresholded
+to 0.5 before being written to the actual output. Note that each pin
+that is managed adds to the CPU usage \, so only activate the pins
+you actually need.;
+#X text 34 19 Digital channels are numbered 11 to 26 \, which corresponds
+to Bela's 0 to 15 digital channels. They can be set to respond at message
+rate or at signal rate;
+#X obj 415 271 osc~ 1;
+#X obj 385 157 change;
+#X obj 500 158 change;
+#X obj 64 116 print digitalIn11;
+#X obj 219 117 print digitalIn12;
+#X connect 0 0 48 0;
+#X connect 1 0 49 0;
+#X connect 2 0 15 0;
+#X connect 2 1 16 0;
+#X connect 9 0 42 0;
+#X connect 10 0 8 0;
+#X connect 12 0 8 0;
+#X connect 13 0 8 0;
+#X connect 14 0 8 0;
+#X connect 15 0 46 0;
+#X connect 16 0 47 0;
+#X connect 17 0 20 0;
+#X connect 20 0 16 0;
+#X connect 20 0 15 0;
+#X connect 23 0 24 0;
+#X connect 24 0 6 0;
+#X connect 25 0 22 0;
+#X connect 27 0 23 0;
+#X connect 30 0 29 0;
+#X connect 31 0 32 0;
+#X connect 32 0 29 0;
+#X connect 38 0 39 0;
+#X connect 38 1 39 0;
+#X connect 39 0 40 0;
+#X connect 40 0 34 0;
+#X connect 42 0 8 0;
+#X connect 45 0 29 0;
+#X connect 46 0 18 0;
+#X connect 47 0 19 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/digital/digital_example.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,89 @@
+#N canvas 279 78 857 690 10;
+#X obj 18 477 osc~ 200;
+#X obj 57 522 *~;
+#X obj 112 405 line~ 1;
+#X msg 116 372 1 \, 0 200;
+#X obj 205 472 line~ 1;
+#X msg 209 439 1 \, 0 200;
+#X obj 210 405 select 1;
+#X obj 140 531 *~;
+#X obj 484 601 s bela_setDigital;
+#X obj 484 532 loadbang;
+#X msg 65 445 0;
+#X obj 116 332 r bela_digitalIn12;
+#X obj 422 217 r bela_digitalIn14;
+#X obj 422 243 s bela_digitalOut13;
+#X obj 285 218 r bela_digitalIn12;
+#X obj 285 243 s bela_digitalOut11;
+#X obj 374 339 timer;
+#X msg 404 304 bang;
+#X msg 368 304 bang;
+#X obj 555 184 metro 1000;
+#X obj 575 304 s bela_digitalOut15;
+#X obj 374 366 print sourceto12;
+#X obj 488 351 timer;
+#X msg 518 316 bang;
+#X msg 482 316 bang;
+#X obj 487 380 print sourceto14;
+#X obj 663 15 r bela_digitalIn16;
+#X msg 553 265 1;
+#X obj 577 214 delay 500;
+#X obj 480 275 select 1;
+#X obj 346 268 select 1;
+#X msg 584 281 0;
+#X obj 634 251 select 1 0;
+#X obj 80 622 dac~ 3 4;
+#X obj 65 566 expr~ $v1*0.5 + 0.5;
+#X obj 138 588 expr~ $v1*0.5 + 0.5;
+#X obj 65 202 dac~ 11;
+#X obj 114 83 osc~ 300;
+#X obj 176 179 dac~ 1 3;
+#X obj 34 136 adc~ 17;
+#X obj 423 504 delay 3000;
+#X msg 388 552 disable 12;
+#X msg 484 552 out 11 ~ \, out 13 \, in 14 \, out 15 \, in 16 \, in
+17 ~;
+#X connect 0 0 1 0;
+#X connect 0 0 7 0;
+#X connect 1 0 34 0;
+#X connect 2 0 1 1;
+#X connect 3 0 2 0;
+#X connect 4 0 7 1;
+#X connect 5 0 4 0;
+#X connect 6 0 5 0;
+#X connect 6 0 10 0;
+#X connect 7 0 35 0;
+#X connect 9 0 42 0;
+#X connect 9 0 40 0;
+#X connect 10 0 0 1;
+#X connect 11 0 3 0;
+#X connect 12 0 13 0;
+#X connect 12 0 29 0;
+#X connect 14 0 15 0;
+#X connect 14 0 30 0;
+#X connect 16 0 21 0;
+#X connect 17 0 16 1;
+#X connect 18 0 16 0;
+#X connect 19 0 27 0;
+#X connect 19 0 28 0;
+#X connect 22 0 25 0;
+#X connect 23 0 22 1;
+#X connect 24 0 22 0;
+#X connect 26 0 32 0;
+#X connect 27 0 20 0;
+#X connect 27 0 24 0;
+#X connect 27 0 18 0;
+#X connect 28 0 31 0;
+#X connect 29 0 23 0;
+#X connect 30 0 17 0;
+#X connect 31 0 20 0;
+#X connect 32 0 27 0;
+#X connect 32 1 31 0;
+#X connect 34 0 33 0;
+#X connect 35 0 33 1;
+#X connect 37 0 38 0;
+#X connect 37 0 38 1;
+#X connect 39 0 36 0;
+#X connect 40 0 41 0;
+#X connect 41 0 8 0;
+#X connect 42 0 8 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/envelopeTrigger/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,196 @@
+#N canvas 646 209 359 287 10;
+#X obj 28 207 dac~;
+#N canvas 324 380 556 621 env 0;
+#N canvas 886 1001 131 207 >~ 0;
+#X obj 19 -37 -~;
+#X obj 19 -58 min~;
+#X obj 19 26 *~ 1e+37;
+#X obj 19 5 +~ 1e-37;
+#X obj 19 -16 clip~ -1e-37 0;
+#X obj 19 -83 inlet~;
+#X obj 62 -83 inlet~;
+#X obj 19 48 outlet~;
+#X connect 0 0 4 0;
+#X connect 1 0 0 0;
+#X connect 2 0 7 0;
+#X connect 3 0 2 0;
+#X connect 4 0 3 0;
+#X connect 5 0 1 0;
+#X connect 6 0 0 1;
+#X connect 6 0 1 1;
+#X restore 199 106 pd >~;
+#X obj 23 173 /~ 44100;
+#X obj 68 153 samplerate~;
+#X obj 68 133 loadbang;
+#X obj 199 128 biquad~ 0 0 -1 1 0;
+#X obj 23 209 rpole~;
+#X obj 23 153 sig~ 1;
+#X obj 226 171 sig~ 0.5;
+#N canvas 0 22 110 231 <~ 0;
+#X obj 11 -41 -~;
+#X obj 11 29 +~ 1e-37;
+#X obj 11 8 clip~ -1e-37 0;
+#X obj 11 -86 inlet~;
+#X obj 55 -85 inlet~;
+#X obj 11 -62 max~;
+#X obj 11 -17 *~ -1;
+#X obj 11 81 outlet~;
+#X obj 11 50 *~ 1e+37;
+#X connect 0 0 6 0;
+#X connect 1 0 8 0;
+#X connect 2 0 1 0;
+#X connect 3 0 5 0;
+#X connect 4 0 0 1;
+#X connect 4 0 5 1;
+#X connect 5 0 0 0;
+#X connect 6 0 2 0;
+#X connect 8 0 7 0;
+#X restore 199 192 pd <~;
+#X text 237 193 (x != x);
+#X obj 23 467 *~;
+#X obj 199 150 *~ -1;
+#X obj 23 331 *~ 512;
+#X obj 23 229 clip~ 0 1;
+#X obj 23 352 tabread4~ env1;
+#X obj 23 310 *~;
+#X obj 199 64 abs~;
+#X obj 199 43 hip~ 1;
+#X obj 199 22 inlet~ excitationSignal;
+#X obj 218 269 sig~ 1;
+#X obj 218 292 /~;
+#X obj 226 86 inlet~ threshold;
+#X obj 262 229 inlet~ envDuration;
+#X obj 38 445 osc~ 220;
+#X obj 38 425 +~ 50;
+#X obj 38 385 *~;
+#X obj 38 405 *~ 1000;
+#X obj 262 269 clip~ 1e-05 100000;
+#X obj 23 487 outlet~;
+#N canvas 0 22 450 278 (subpatch) 0;
+#X array env1 512 float 3;
+#A 0 0 0.0625 0.0883883 0.108253 0.125 0.139754 0.153093 0.165359 0.176777
+0.1875 0.197642 0.207289 0.216506 0.225347 0.233854 0.242061 0.25 0.257694
+0.265165 0.272431 0.279509 0.286411 0.293151 0.299739 0.306186 0.3125
+0.318689 0.32476 0.330719 0.336573 0.342327 0.347985 0.353553 0.359035
+0.364434 0.369755 0.375 0.380173 0.385276 0.390312 0.395285 0.400195
+0.405046 0.40984 0.414578 0.419263 0.423896 0.428478 0.433013 0.4375
+0.441942 0.446339 0.450694 0.455007 0.459279 0.463512 0.467707 0.471865
+0.475986 0.480072 0.484123 0.488141 0.492125 0.496078 0.5 0.503891
+0.507752 0.511585 0.515388 0.519164 0.522913 0.526634 0.53033 0.534
+0.537645 0.541266 0.544862 0.548435 0.551985 0.555512 0.559017 0.5625
+0.565962 0.569402 0.572822 0.576222 0.579601 0.582961 0.586302 0.589624
+0.592927 0.596212 0.599479 0.602728 0.60596 0.609175 0.612372 0.615554
+0.618718 0.621867 0.625 0.628117 0.631219 0.634306 0.637377 0.640434
+0.643477 0.646505 0.649519 0.652519 0.655506 0.658478 0.661438 0.664384
+0.667317 0.670238 0.673146 0.676041 0.678924 0.681795 0.684653 0.6875
+0.690335 0.693159 0.695971 0.698771 0.701561 0.704339 0.707107 0.709864
+0.71261 0.715345 0.71807 0.720785 0.72349 0.726184 0.728869 0.731544
+0.734209 0.736864 0.73951 0.742146 0.744773 0.747391 0.75 0.7526 0.75519
+0.757772 0.760345 0.76291 0.765466 0.768013 0.770552 0.773082 0.775605
+0.778119 0.780625 0.783123 0.785613 0.788095 0.790569 0.793036 0.795495
+0.797947 0.800391 0.802827 0.805256 0.807678 0.810093 0.8125 0.8149
+0.817294 0.81968 0.822059 0.824432 0.826797 0.829156 0.831508 0.833854
+0.836193 0.838525 0.840851 0.843171 0.845484 0.847791 0.850092 0.852386
+0.854675 0.856957 0.859233 0.861503 0.863767 0.866025 0.868278 0.870524
+0.872765 0.875 0.877229 0.879453 0.881671 0.883883 0.88609 0.888292
+0.890488 0.892679 0.894864 0.897044 0.899218 0.901388 0.903552 0.905711
+0.907865 0.910014 0.912157 0.914296 0.91643 0.918559 0.920682 0.922801
+0.924916 0.927025 0.929129 0.931229 0.933324 0.935414 0.9375 0.939581
+0.941657 0.943729 0.945797 0.947859 0.949918 0.951972 0.954021 0.956066
+0.958107 0.960143 0.962175 0.964203 0.966227 0.968246 0.970261 0.972272
+0.974279 0.976281 0.97828 0.980274 0.982265 0.984251 0.986233 0.988212
+0.990186 0.992157 0.994123 0.996086 0.998045 1 0.992172 0.984375 0.976609
+0.968874 0.961169 0.953495 0.945852 0.938239 0.930657 0.923106 0.915586
+0.908097 0.900638 0.89321 0.885813 0.878447 0.871111 0.863806 0.856532
+0.849289 0.842076 0.834894 0.827743 0.820623 0.813533 0.806474 0.799446
+0.792449 0.785483 0.778547 0.771642 0.764767 0.757924 0.751111 0.744329
+0.737578 0.730857 0.724168 0.717509 0.71088 0.704283 0.697716 0.69118
+0.684675 0.678201 0.671757 0.665344 0.658962 0.652611 0.64629 0.64
+0.633741 0.627513 0.621315 0.615148 0.609012 0.602907 0.596832 0.590788
+0.584775 0.578793 0.572841 0.56692 0.56103 0.555171 0.549343 0.543545
+0.537778 0.532041 0.526336 0.520661 0.515017 0.509404 0.503822 0.49827
+0.492749 0.487259 0.481799 0.476371 0.470973 0.465605 0.460269 0.454963
+0.449689 0.444444 0.439231 0.434048 0.428897 0.423775 0.418685 0.413625
+0.408597 0.403599 0.398631 0.393695 0.388789 0.383914 0.37907 0.374256
+0.369473 0.364721 0.36 0.355309 0.35065 0.346021 0.341423 0.336855
+0.332318 0.327812 0.323337 0.318893 0.314479 0.310096 0.305744 0.301423
+0.297132 0.292872 0.288643 0.284444 0.280277 0.27614 0.272034 0.267958
+0.263914 0.2599 0.255917 0.251965 0.248043 0.244152 0.240292 0.236463
+0.232664 0.228897 0.22516 0.221453 0.217778 0.214133 0.210519 0.206936
+0.203383 0.199862 0.196371 0.19291 0.189481 0.186082 0.182714 0.179377
+0.176071 0.172795 0.16955 0.166336 0.163153 0.16 0.156878 0.153787
+0.150727 0.147697 0.144698 0.14173 0.138793 0.135886 0.13301 0.130165
+0.127351 0.124567 0.121815 0.119093 0.116401 0.113741 0.111111 0.108512
+0.105944 0.103406 0.1009 0.0984237 0.0959785 0.093564 0.0911803 0.0888274
+0.0865052 0.0842138 0.0819531 0.0797232 0.077524 0.0753556 0.073218
+0.0711111 0.069035 0.0669896 0.064975 0.0629911 0.0610381 0.0591157
+0.0572241 0.0553633 0.0535332 0.0517339 0.0499654 0.0482276 0.0465206
+0.0448443 0.0431988 0.041584 0.04 0.0384467 0.0369243 0.0354325 0.0339715
+0.0325413 0.0311419 0.0297732 0.0284352 0.027128 0.0258516 0.0246059
+0.023391 0.0222068 0.0210534 0.0199308 0.0188389 0.0177778 0.0167474
+0.0157478 0.0147789 0.0138408 0.0129335 0.0120569 0.0112111 0.010396
+0.00961169 0.00885813 0.00813533 0.00744329 0.00678201 0.00615148 0.00555171
+0.0049827 0.00444444 0.00393695 0.00346021 0.00301423 0.002599 0.00221453
+0.00186082 0.00153787 0.00124567 0.000984237 0.000753556 0.000553633
+0.000384467 0.000246059 0.000138408 6.15148e-05 1.53787e-05 0;
+#X coords 0 1 512 0 512 64 1 0 0;
+#X restore 24 536 graph;
+#X text 244 55 centre signal around zero and take abs value;
+#X text 243 107 if greater than threshold output 1;
+#X text 313 129 differentiator;
+#X text 238 150 (generates impulse on positive trigger (0->1);
+#X text 67 210 signal-rate counter;
+#X text 296 193 -> resets counter by briefly setting;
+#X text 314 204 rpole~ coeff to 0;
+#X text 96 416 kickdrum 101 :P;
+#X text 400 514 pre-generated envelope;
+#X obj 262 249 *~;
+#X text 127 352 << use envelope value for volume and frequency;
+#X connect 0 0 4 0;
+#X connect 1 0 5 0;
+#X connect 2 0 1 1;
+#X connect 3 0 2 0;
+#X connect 4 0 11 0;
+#X connect 5 0 13 0;
+#X connect 6 0 1 0;
+#X connect 7 0 8 1;
+#X connect 8 0 5 1;
+#X connect 10 0 28 0;
+#X connect 11 0 8 0;
+#X connect 12 0 14 0;
+#X connect 13 0 15 0;
+#X connect 14 0 10 0;
+#X connect 14 0 25 0;
+#X connect 14 0 25 1;
+#X connect 15 0 12 0;
+#X connect 16 0 0 0;
+#X connect 17 0 16 0;
+#X connect 18 0 17 0;
+#X connect 19 0 20 0;
+#X connect 20 0 15 1;
+#X connect 21 0 0 1;
+#X connect 22 0 39 0;
+#X connect 22 0 39 1;
+#X connect 23 0 10 1;
+#X connect 24 0 23 0;
+#X connect 25 0 26 0;
+#X connect 26 0 24 0;
+#X connect 27 0 20 1;
+#X connect 39 0 27 0;
+#X restore 28 174 pd env;
+#X obj 79 110 adc~ 9;
+#X obj 129 130 adc~ 10;
+#X text 72 90 piezo input for excitation;
+#X text 123 110 fader 1 sets threshold;
+#X text 183 130 fader 2 sets duration;
+#X text 29 17 SAMPLE-ACCURATE ENVELOPE TRIGGER;
+#X text 29 27 ================================;
+#X obj 28 90 adc~ 8;
+#X text 141 236 ---------------------------------;
+#X text 141 226 @krighxz / BELA / heavy / 12/2015;
+#X text 165 245 beaglert.cc / enzienaudio.com;
+#X connect 1 0 0 0;
+#X connect 1 0 0 1;
+#X connect 2 0 1 1;
+#X connect 3 0 1 2;
+#X connect 9 0 1 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/hello-midi/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,81 @@
+#N canvas 85 58 1280 685 10;
+#X obj 19 396 dac~ 1 2;
+#X text 39 1 Hello world!;
+#X text 37 11 ============;
+#X obj 31 353 *~ 0.1;
+#X text 202 448 ---------------------------------;
+#X text 202 438 @krighxz / BELA / heavy / 12/2015;
+#X text 226 457 beaglert.cc / enzienaudio.com;
+#X obj 39 132 mtof;
+#X floatatom 79 59 5 0 0 0 - - -, f 5;
+#X floatatom 122 59 5 0 0 0 - - -, f 5;
+#X floatatom 41 63 5 0 0 0 - - -, f 5;
+#X obj 26 170 phasor~ 440;
+#X text 100 363 (the harsh sound of success);
+#X obj 260 83 * 20;
+#X obj 261 112 + 200;
+#X obj 132 85 loadbang;
+#X msg 131 111 1000;
+#X obj 125 166 sig~;
+#X text 673 121 [pgmin];
+#X obj 218 178 * 10;
+#X obj 30 250 hip~;
+#X obj 26 275 hip~;
+#X obj 20 302 hip~;
+#X obj 19 323 hip~;
+#X obj 87 192 lop~ 4;
+#X obj 152 256 line;
+#X obj 200 200 + 50;
+#X msg 170 227 \$1 200;
+#X msg 93 114 0;
+#X msg 185 112 0;
+#X obj 29 224 vcf~ 4;
+#X obj 70 333 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 42 92 mono;
+#X obj 40 28 bela_notein 0;
+#X obj 260 60 bela_ctlin 5 0;
+#X obj 220 151 bela_ctlin 6 0;
+#X text 537 58 [ctlin];
+#X text 332 27 [notein];
+#X text 140 29 <== the bela equivalent of ==>;
+#X text 349 59 <== the bela equivalent of ==>;
+#X text 483 121 <== the bela equivalent of ==>;
+#X obj 405 175 print pgmin;
+#X obj 408 121 bela_pgmin 0;
+#X connect 3 0 0 1;
+#X connect 3 0 0 0;
+#X connect 7 0 11 0;
+#X connect 8 0 32 1;
+#X connect 10 0 32 0;
+#X connect 11 0 30 0;
+#X connect 13 0 14 0;
+#X connect 14 0 17 0;
+#X connect 15 0 16 0;
+#X connect 15 0 29 0;
+#X connect 15 0 28 0;
+#X connect 16 0 17 0;
+#X connect 17 0 24 0;
+#X connect 19 0 26 0;
+#X connect 20 0 21 0;
+#X connect 21 0 22 0;
+#X connect 22 0 23 0;
+#X connect 23 0 3 0;
+#X connect 24 0 30 1;
+#X connect 25 0 20 1;
+#X connect 25 0 21 1;
+#X connect 25 0 22 1;
+#X connect 25 0 23 1;
+#X connect 26 0 27 0;
+#X connect 27 0 25 0;
+#X connect 29 0 19 0;
+#X connect 30 0 20 0;
+#X connect 31 0 3 1;
+#X connect 32 0 7 0;
+#X connect 32 1 31 0;
+#X connect 33 0 10 0;
+#X connect 33 1 8 0;
+#X connect 33 2 9 0;
+#X connect 34 0 13 0;
+#X connect 35 0 19 0;
+#X connect 42 0 41 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/hello-midi/countTo.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,33 @@
+#N canvas 584 255 450 300 10;
+#X obj 65 196 outlet;
+#X obj 98 8 inlet;
+#X text 189 49 counts from 0 to the given number -1;
+#X text 214 95 inlet:N;
+#X text 212 112 outlet: 0 \, 1 \, 2 \, ... \, N-1;
+#X obj 143 61 f;
+#X obj 100 138 spigot;
+#X obj 129 89 <;
+#X obj 129 113 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 95 28 t b f b;
+#X obj 92 74 + 1;
+#X obj 43 73 f;
+#X obj 63 104 t b f f f;
+#X msg 92 53 0;
+#X obj 16 144 delay 0;
+#X connect 1 0 9 0;
+#X connect 5 0 7 1;
+#X connect 6 0 11 0;
+#X connect 7 0 8 0;
+#X connect 8 0 6 1;
+#X connect 9 0 11 0;
+#X connect 9 1 5 0;
+#X connect 9 1 13 0;
+#X connect 10 0 11 1;
+#X connect 11 0 12 0;
+#X connect 12 0 14 0;
+#X connect 12 1 0 0;
+#X connect 12 2 7 0;
+#X connect 12 3 10 0;
+#X connect 13 0 11 1;
+#X connect 14 0 6 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/hello-midi/maxArray.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,67 @@
+#N canvas 464 23 1002 583 10;
+#X obj 224 202 max;
+#X obj 265 200 f;
+#X obj 223 229 change;
+#X obj 223 251 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X obj 223 271 f;
+#X obj 259 323 f;
+#X obj 187 315 f;
+#X obj 268 282 t b b;
+#X obj 183 364 outlet;
+#X obj 269 379 outlet;
+#X obj 179 17 inlet;
+#X obj 225 177 tabread \$1;
+#X obj 230 88 t f f b;
+#X text 376 37 Finds a maximum value in an array;
+#X text 531 112 \$2 arrayLength;
+#X obj 292 10 inlet;
+#X text 425 146 inlets:;
+#X text 473 144 1 bang to find the maximum;
+#X text 475 157 2 arrayLength;
+#X text 407 90 creation arguments: \$1 arrayName;
+#X obj 246 38 \$2;
+#X obj 202 40 bang;
+#X obj 230 65 f;
+#X obj 224 6 loadbang;
+#X text 444 216 outlets: 1 maximum value;
+#X text 498 231 2 index of the maximum value;
+#X msg 276 141 -1e+15;
+#X obj 222 142 t f f f;
+#X floatatom 158 266 5 0 0 0 - - -, f 5;
+#X obj 332 302 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X obj 288 253 select 127;
+#X floatatom 301 324 5 0 0 0 - - -, f 5;
+#X text 362 256 heavy does not support the second inlet of select so
+I have to hard code 127 in here;
+#X obj 223 110 countTo;
+#X connect 0 0 1 0;
+#X connect 0 0 2 0;
+#X connect 0 0 6 1;
+#X connect 0 0 28 0;
+#X connect 1 0 0 1;
+#X connect 2 0 3 0;
+#X connect 3 0 4 0;
+#X connect 4 0 5 1;
+#X connect 4 0 31 0;
+#X connect 5 0 9 0;
+#X connect 6 0 8 0;
+#X connect 7 0 6 0;
+#X connect 7 1 5 0;
+#X connect 10 0 21 0;
+#X connect 11 0 0 0;
+#X connect 12 0 33 0;
+#X connect 12 2 26 0;
+#X connect 15 0 22 1;
+#X connect 20 0 22 1;
+#X connect 21 0 22 0;
+#X connect 22 0 12 0;
+#X connect 23 0 20 0;
+#X connect 26 0 0 1;
+#X connect 27 0 30 0;
+#X connect 27 1 11 0;
+#X connect 27 2 4 1;
+#X connect 30 0 7 0;
+#X connect 30 0 29 0;
+#X connect 33 0 27 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/hello-midi/mono.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,106 @@
+#N canvas 69 182 1020 452 10;
+#N canvas 0 23 450 278 (subpatch) 0;
+#X array \$0-activeNotes 127 float 2;
+#X coords 0 100 127 -1 200 140 1 0 0;
+#X restore 537 25 graph;
+#N canvas 89 91 450 300 resetArray 1;
+#X obj 121 145 tabwrite \$0-activeNotes;
+#X msg 123 115 -0.5;
+#X obj 154 90 t b f;
+#X msg 129 45 128;
+#X obj 103 14 inlet;
+#X text 212 71 countto;
+#X obj 153 69 countTo;
+#X connect 1 0 0 0;
+#X connect 2 0 1 0;
+#X connect 2 1 0 1;
+#X connect 3 0 6 0;
+#X connect 4 0 3 0;
+#X connect 6 0 2 0;
+#X restore 83 49 pd resetArray;
+#X obj 83 3 loadbang;
+#X obj 88 25 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X obj 39 188 maxArray \$0-activeNotes 128;
+#X obj 183 219 f;
+#X obj 39 237 >= 0;
+#X floatatom 148 313 5 0 0 0 - - -, f 5;
+#X obj 70 262 select 1;
+#X obj 366 87 f;
+#X obj 426 60 f;
+#X text 334 39 note;
+#X text 424 16 velocity;
+#X obj 423 87 > 0;
+#X obj 344 281 tabwrite \$0-activeNotes;
+#X obj 330 207 f;
+#X floatatom 303 178 5 0 0 0 - - -, f 5;
+#X obj 265 163 f;
+#X obj 304 150 + 1;
+#X msg 277 130 -1;
+#X msg 371 207 -0.5;
+#X obj 400 116 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 367 137 spigot;
+#X obj 447 134 spigot;
+#X obj 460 86 == 0;
+#X obj 470 109 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 434 163 t b f;
+#X obj 337 60 inlet;
+#X obj 426 32 inlet;
+#X obj 148 350 outlet;
+#X obj 275 89 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X obj 41 163 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X obj 363 172 t b f;
+#X obj 334 240 t b f;
+#X text 578 211 pass in velocity/note pairs (velocity first) and obtain
+as an output the most recent note still active;
+#X text 581 268 outlet 2 gate;
+#X obj 330 366 outlet;
+#X text 583 251 outlet 1 most recent note active (-1 otherwise);
+#X text 155 329 most recent note active;
+#X msg 338 338 0;
+#X msg 378 339 1;
+#X connect 2 0 3 0;
+#X connect 3 0 1 0;
+#X connect 3 0 19 0;
+#X connect 4 0 6 0;
+#X connect 4 1 5 1;
+#X connect 5 0 7 0;
+#X connect 5 0 40 0;
+#X connect 6 0 8 0;
+#X connect 7 0 29 0;
+#X connect 8 0 5 0;
+#X connect 8 1 39 0;
+#X connect 9 0 22 0;
+#X connect 9 0 23 0;
+#X connect 10 0 13 0;
+#X connect 10 0 24 0;
+#X connect 13 0 21 0;
+#X connect 15 0 33 0;
+#X connect 16 0 15 1;
+#X connect 17 0 18 0;
+#X connect 18 0 16 0;
+#X connect 18 0 17 1;
+#X connect 19 0 17 1;
+#X connect 20 0 33 0;
+#X connect 21 0 22 1;
+#X connect 22 0 32 0;
+#X connect 23 0 26 0;
+#X connect 24 0 25 0;
+#X connect 25 0 23 1;
+#X connect 26 0 20 0;
+#X connect 26 1 14 1;
+#X connect 27 0 9 0;
+#X connect 28 0 10 0;
+#X connect 28 0 30 0;
+#X connect 30 0 17 0;
+#X connect 31 0 4 0;
+#X connect 32 0 15 0;
+#X connect 32 1 14 1;
+#X connect 33 0 31 0;
+#X connect 33 1 14 0;
+#X connect 39 0 36 0;
+#X connect 40 0 36 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/hello-world/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,13 @@
+#N canvas 275 573 461 212 10;
+#X obj 39 143 dac~ 1 2;
+#X text 35 32 Hello world!;
+#X text 35 42 ============;
+#X text 95 79 (the sweet sound of success);
+#X obj 39 102 *~ 0.1;
+#X obj 39 81 osc~ 440;
+#X text 238 160 ---------------------------------;
+#X text 238 150 @krighxz / BELA / heavy / 12/2015;
+#X text 262 169 beaglert.cc / enzienaudio.com;
+#X connect 4 0 0 1;
+#X connect 4 0 0 0;
+#X connect 5 0 4 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/karplusStrong/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,240 @@
+#N canvas 440 516 715 490 10;
+#X obj 38 370 dac~ 1 2;
+#X text 470 442 ---------------------------------;
+#X text 470 432 @krighxz / BELA / heavy / 12/2015;
+#X text 494 451 beaglert.cc / enzienaudio.com;
+#X text 33 22 Karplus Strong;
+#X text 33 32 ==============;
+#X obj 38 323 vd~ \$0-wg1;
+#X obj 118 345 lop~ 1000;
+#X obj 118 367 *~ 0.99;
+#X obj 118 389 s~ \$0-fb1;
+#X obj 38 94 adc~ 3;
+#X obj 38 116 hip~ 100;
+#X obj 98 176 r~ \$0-fb1;
+#X obj 38 198 delwrite~ \$0-wg1 100;
+#X obj 38 268 *~;
+#X obj 38 290 *~ 10;
+#X text 99 116 <<< centre (i.e. DC block) signal with high-pass filter
+;
+#X text 100 94 <<< use accelerometer \, piezo or FSR on first analog
+in;
+#X text 105 245 <<< use potentiometer or fader on second analog in
+;
+#X text 100 75 EXCITATION INPUT:;
+#X text 100 225 DELAY PERIOD:;
+#X text 182 365 <<< and decay here [0-0.9999];
+#X text 182 345 <<< set damping here [~100-20000];
+#N canvas 335 566 461 274 gate 0;
+#N canvas 0 22 450 278 (subpatch) 0;
+#X array \$0-xfer 1024 float 3;
+#A 0 0 -0.808242 -0.804734 -0.801234 -0.797741 -0.794256 -0.790778
+-0.787309 -0.783846 -0.780392 -0.776945 -0.773506 -0.770074 -0.76665
+-0.763233 -0.759824 -0.756423 -0.75303 -0.749644 -0.746266 -0.742895
+-0.739532 -0.736176 -0.732829 -0.729488 -0.726156 -0.722831 -0.719514
+-0.716204 -0.712902 -0.709608 -0.706321 -0.703042 -0.69977 -0.696507
+-0.69325 -0.690002 -0.686761 -0.683527 -0.680302 -0.677084 -0.673873
+-0.67067 -0.667475 -0.664288 -0.661108 -0.657935 -0.654771 -0.651614
+-0.648464 -0.645323 -0.642188 -0.639062 -0.635943 -0.632832 -0.629728
+-0.626632 -0.623544 -0.620463 -0.61739 -0.614324 -0.611266 -0.608216
+-0.605174 -0.602139 -0.599111 -0.596092 -0.59308 -0.590075 -0.587078
+-0.584089 -0.581107 -0.578134 -0.575167 -0.572209 -0.569257 -0.566314
+-0.563378 -0.56045 -0.55753 -0.554617 -0.551711 -0.548814 -0.545924
+-0.543041 -0.540167 -0.537299 -0.53444 -0.531588 -0.528744 -0.525907
+-0.523078 -0.520257 -0.517443 -0.514637 -0.511839 -0.509048 -0.506265
+-0.503489 -0.500721 -0.497961 -0.495208 -0.492463 -0.489726 -0.486996
+-0.484274 -0.481559 -0.478852 -0.476153 -0.473461 -0.470777 -0.468101
+-0.465432 -0.462771 -0.460117 -0.457472 -0.454833 -0.452203 -0.44958
+-0.446964 -0.444357 -0.441757 -0.439164 -0.436579 -0.434002 -0.431432
+-0.42887 -0.426316 -0.42377 -0.42123 -0.418699 -0.416175 -0.413659
+-0.41115 -0.40865 -0.406156 -0.403671 -0.401193 -0.398722 -0.396259
+-0.393804 -0.391357 -0.388917 -0.386485 -0.38406 -0.381643 -0.379234
+-0.376832 -0.374438 -0.372051 -0.369673 -0.367301 -0.364938 -0.362582
+-0.360233 -0.357893 -0.35556 -0.353234 -0.350916 -0.348606 -0.346304
+-0.344009 -0.341721 -0.339442 -0.33717 -0.334905 -0.332649 -0.330399
+-0.328158 -0.325924 -0.323698 -0.321479 -0.319268 -0.317065 -0.314869
+-0.312681 -0.3105 -0.308328 -0.306162 -0.304005 -0.301855 -0.299713
+-0.297578 -0.295451 -0.293331 -0.291219 -0.289115 -0.287019 -0.28493
+-0.282848 -0.280775 -0.278709 -0.27665 -0.2746 -0.272556 -0.270521
+-0.268493 -0.266473 -0.26446 -0.262455 -0.260458 -0.258468 -0.256486
+-0.254511 -0.252545 -0.250585 -0.248634 -0.24669 -0.244753 -0.242825
+-0.240904 -0.23899 -0.237084 -0.235186 -0.233296 -0.231413 -0.229537
+-0.22767 -0.22581 -0.223957 -0.222112 -0.220275 -0.218446 -0.216624
+-0.21481 -0.213003 -0.211204 -0.209413 -0.207629 -0.205853 -0.204084
+-0.202323 -0.20057 -0.198824 -0.197086 -0.195356 -0.193633 -0.191918
+-0.190211 -0.188511 -0.186819 -0.185134 -0.183457 -0.181788 -0.180126
+-0.178472 -0.176826 -0.175187 -0.173556 -0.171932 -0.170316 -0.168708
+-0.167108 -0.165515 -0.163929 -0.162351 -0.160781 -0.159219 -0.157664
+-0.156117 -0.154577 -0.153045 -0.151521 -0.150004 -0.148495 -0.146993
+-0.1455 -0.144013 -0.142535 -0.141064 -0.139601 -0.138145 -0.136697
+-0.135256 -0.133824 -0.132398 -0.130981 -0.129571 -0.128169 -0.126774
+-0.125387 -0.124008 -0.122636 -0.121272 -0.119915 -0.118566 -0.117225
+-0.115891 -0.114565 -0.113247 -0.111936 -0.110633 -0.109338 -0.10805
+-0.10677 -0.105497 -0.104232 -0.102975 -0.101725 -0.100483 -0.0992487
+-0.0980219 -0.0968027 -0.0955911 -0.0943872 -0.0931909 -0.0920023 -0.0908212
+-0.0896478 -0.0884821 -0.0873239 -0.0861734 -0.0850305 -0.0838953 -0.0827676
+-0.0816476 -0.0805353 -0.0794305 -0.0783334 -0.077244 -0.0761621 -0.0750879
+-0.0740213 -0.0729623 -0.071911 -0.0708673 -0.0698312 -0.0688028 -0.067782
+-0.0667688 -0.0657632 -0.0647653 -0.063775 -0.0627924 -0.0618173 -0.0608499
+-0.0598901 -0.058938 -0.0579935 -0.0570566 -0.0561273 -0.0552057 -0.0542917
+-0.0533853 -0.0524866 -0.0515955 -0.050712 -0.0498361 -0.0489679 -0.0481073
+-0.0472543 -0.046409 -0.0455713 -0.0447412 -0.0439188 -0.0431039 -0.0422968
+-0.0414972 -0.0407053 -0.039921 -0.0391443 -0.0383752 -0.0376138 -0.03686
+-0.0361139 -0.0353754 -0.0346445 -0.0339212 -0.0332056 -0.0324976 -0.0317972
+-0.0311044 -0.0304193 -0.0297418 -0.029072 -0.0284097 -0.0277551 -0.0271082
+-0.0264688 -0.0258371 -0.025213 -0.0245966 -0.0239877 -0.0233865 -0.022793
+-0.022207 -0.0216287 -0.021058 -0.020495 -0.0199396 -0.0193918 -0.0188516
+-0.0183191 -0.0177942 -0.0172769 -0.0167673 -0.0162653 -0.0157709 -0.0152841
+-0.014805 -0.0143335 -0.0138696 -0.0134134 -0.0129648 -0.0125238 -0.0120905
+-0.0116647 -0.0112466 -0.0108362 -0.0104333 -0.0100381 -0.00965057
+-0.00927063 -0.00889832 -0.00853363 -0.00817657 -0.00782715 -0.00748535
+-0.00715118 -0.00682465 -0.00650574 -0.00619446 -0.00589081 -0.00559479
+-0.0053064 -0.00502563 -0.0047525 -0.004487 -0.00422913 -0.00397888
+-0.00373627 -0.00350128 -0.00327393 -0.0030542 -0.0028421 -0.00263763
+-0.0024408 -0.00225159 -0.00207001 -0.00189606 -0.00172974 -0.00157104
+-0.00141998 -0.00127655 -0.00114075 -0.00101257 -0.000892029 -0.000779114
+-0.000673828 -0.000576172 -0.000486145 -0.000403747 -0.000328979 -0.000261841
+-0.000202332 -0.000150452 -0.000106201 -6.95801e-05 -4.05884e-05 -1.92261e-05
+-5.49316e-06 0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0
+-0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0
+-0 -0 -0 -0 -0 -0 -0 -0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
+0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 5.49316e-06
+1.92261e-05 4.05884e-05 6.95801e-05 0.000106201 0.000150452 0.000202332
+0.000261841 0.000328979 0.000403747 0.000486145 0.000576172 0.000673828
+0.000779114 0.000892029 0.00101257 0.00114075 0.00127655 0.00141998
+0.00157104 0.00172974 0.00189606 0.00207001 0.00225159 0.0024408 0.00263763
+0.0028421 0.0030542 0.00327393 0.00350128 0.00373627 0.00397888 0.00422913
+0.004487 0.0047525 0.00502563 0.0053064 0.00559479 0.00589081 0.00619446
+0.00650574 0.00682465 0.00715118 0.00748535 0.00782715 0.00817657 0.00853363
+0.00889832 0.00927063 0.00965057 0.0100381 0.0104333 0.0108362 0.0112466
+0.0116647 0.0120905 0.0125238 0.0129648 0.0134134 0.0138696 0.0143335
+0.014805 0.0152841 0.0157709 0.0162653 0.0167673 0.0172769 0.0177942
+0.0183191 0.0188516 0.0193918 0.0199396 0.020495 0.021058 0.0216287
+0.022207 0.022793 0.0233865 0.0239877 0.0245966 0.025213 0.0258371
+0.0264688 0.0271082 0.0277551 0.0284097 0.029072 0.0297418 0.0304193
+0.0311044 0.0317972 0.0324976 0.0332056 0.0339212 0.0346445 0.0353754
+0.0361139 0.03686 0.0376138 0.0383752 0.0391443 0.039921 0.0407053
+0.0414972 0.0422968 0.0431039 0.0439188 0.0447412 0.0455713 0.046409
+0.0472543 0.0481073 0.0489679 0.0498361 0.050712 0.0515955 0.0524866
+0.0533853 0.0542917 0.0552057 0.0561273 0.0570566 0.0579935 0.058938
+0.0598901 0.0608499 0.0618173 0.0627924 0.063775 0.0647653 0.0657632
+0.0667688 0.067782 0.0688028 0.0698312 0.0708673 0.071911 0.0729623
+0.0740213 0.0750879 0.0761621 0.077244 0.0783334 0.0794305 0.0805353
+0.0816476 0.0827676 0.0838953 0.0850305 0.0861734 0.0873239 0.0884821
+0.0896478 0.0908212 0.0920023 0.0931909 0.0943872 0.0955911 0.0968027
+0.0980219 0.0992487 0.100483 0.101725 0.102975 0.104232 0.105497 0.10677
+0.10805 0.109338 0.110633 0.111936 0.113247 0.114565 0.115891 0.117225
+0.118566 0.119915 0.121272 0.122636 0.124008 0.125387 0.126774 0.128169
+0.129571 0.130981 0.132398 0.133824 0.135256 0.136697 0.138145 0.139601
+0.141064 0.142535 0.144013 0.1455 0.146993 0.148495 0.150004 0.151521
+0.153045 0.154577 0.156117 0.157664 0.159219 0.160781 0.162351 0.163929
+0.165515 0.167108 0.168708 0.170316 0.171932 0.173556 0.175187 0.176826
+0.178472 0.180126 0.181788 0.183457 0.185134 0.186819 0.188511 0.190211
+0.191918 0.193633 0.195356 0.197086 0.198824 0.20057 0.202323 0.204084
+0.205853 0.207629 0.209413 0.211204 0.213003 0.21481 0.216624 0.218446
+0.220275 0.222112 0.223957 0.22581 0.22767 0.229537 0.231413 0.233296
+0.235186 0.237084 0.23899 0.240904 0.242825 0.244753 0.24669 0.248634
+0.250585 0.252545 0.254511 0.256486 0.258468 0.260458 0.262455 0.26446
+0.266473 0.268493 0.270521 0.272556 0.2746 0.27665 0.278709 0.280775
+0.282848 0.28493 0.287019 0.289115 0.291219 0.293331 0.295451 0.297578
+0.299713 0.301855 0.304005 0.306162 0.308328 0.3105 0.312681 0.314869
+0.317065 0.319268 0.321479 0.323698 0.325924 0.328158 0.330399 0.332649
+0.334905 0.33717 0.339442 0.341721 0.344009 0.346304 0.348606 0.350916
+0.353234 0.35556 0.357893 0.360233 0.362582 0.364938 0.367301 0.369673
+0.372051 0.374438 0.376832 0.379234 0.381643 0.38406 0.386485 0.388917
+0.391357 0.393804 0.396259 0.398722 0.401193 0.403671 0.406156 0.40865
+0.41115 0.413659 0.416175 0.418699 0.42123 0.42377 0.426316 0.42887
+0.431432 0.434002 0.436579 0.439164 0.441757 0.444357 0.446964 0.44958
+0.452203 0.454833 0.457472 0.460117 0.462771 0.465432 0.468101 0.470777
+0.473461 0.476153 0.478852 0.481559 0.484274 0.486996 0.489726 0.492463
+0.495208 0.497961 0.500721 0.503489 0.506265 0.509048 0.511839 0.514637
+0.517443 0.520257 0.523078 0.525907 0.528744 0.531588 0.53444 0.537299
+0.540167 0.543041 0.545924 0.548814 0.551711 0.554617 0.55753 0.56045
+0.563378 0.566314 0.569257 0.572209 0.575167 0.578134 0.581107 0.584089
+0.587078 0.590075 0.59308 0.596092 0.599111 0.602139 0.605174 0.608216
+0.611266 0.614324 0.61739 0.620463 0.623544 0.626632 0.629728 0.632832
+0.635943 0.639062 0.642188 0.645323 0.648464 0.651614 0.654771 0.657935
+0.661108 0.664288 0.667475 0.67067 0.673873 0.677084 0.680302 0.683527
+0.686761 0.690002 0.69325 0.696507 0.69977 0.703042 0.706321 0.709608
+0.712902 0.716204 0.719514 0.722831;
+#A 1000 0.726156 0.729488 0.732829 0.736176 0.739532 0.742895 0.746266
+0.749644 0.75303 0.756423 0.759824 0.763233 0.76665 0.770074 0.773506
+0.776945 0.780392 0.783846 0.787309 0.790778 0.794256 0.797741 0.801234
+0.804734;
+#X coords 0 1 1023 -1 200 200 1 0 0;
+#X restore 230 34 graph;
+#X obj 25 27 inlet~;
+#X obj 25 49 clip~ -1 1;
+#X obj 25 71 *~ 512;
+#X obj 25 93 +~ 512;
+#X obj 25 115 tabread4~ \$0-xfer;
+#X obj 25 137 outlet~;
+#N canvas 0 22 334 332 generate-xfer 0;
+#X obj 8 290 tabwrite \$0-xfer;
+#X obj 8 109 / 1024;
+#X obj 8 129 * 2;
+#X obj 8 149 - 1;
+#X obj 8 169 moses 0;
+#X obj 8 191 + 0.1;
+#X obj 8 213 clip -1 0;
+#X obj 68 191 - 0.1;
+#X obj 68 213 clip 0 1;
+#X obj 101 16 inlet threshold;
+#X obj 101 78 count 1024;
+#X obj 128 58 s \$0-thresh;
+#X obj 95 169 r \$0-thresh;
+#X obj 101 38 t b f;
+#X obj 68 236 *;
+#X obj 8 236 *;
+#X obj 8 258 * -1;
+#X text 140 203 <<< deadzone;
+#X text 140 253 <<< smooth by taking exponent;
+#X connect 1 0 2 0;
+#X connect 2 0 3 0;
+#X connect 3 0 4 0;
+#X connect 4 0 5 0;
+#X connect 4 1 7 0;
+#X connect 5 0 6 0;
+#X connect 6 0 15 0;
+#X connect 6 0 15 1;
+#X connect 7 0 8 0;
+#X connect 8 0 14 0;
+#X connect 8 0 14 1;
+#X connect 9 0 13 0;
+#X connect 10 0 0 1;
+#X connect 10 0 1 0;
+#X connect 12 0 5 1;
+#X connect 12 0 7 1;
+#X connect 13 0 10 0;
+#X connect 13 1 11 0;
+#X connect 14 0 0 0;
+#X connect 15 0 16 0;
+#X connect 16 0 0 0;
+#X restore 26 217 pd generate-xfer;
+#X text 58 196 <<< threshold;
+#X obj 26 173 loadbang;
+#X msg 26 195 0.1;
+#X connect 1 0 2 0;
+#X connect 2 0 3 0;
+#X connect 3 0 4 0;
+#X connect 4 0 5 0;
+#X connect 5 0 6 0;
+#X connect 9 0 10 0;
+#X connect 10 0 7 0;
+#X restore 38 137 pd gate;
+#X obj 38 246 adc~ 10;
+#X text 98 137 <<< thresholding to remove any consistent noise in the
+excitation signal;
+#X connect 6 0 7 0;
+#X connect 6 0 0 0;
+#X connect 6 0 0 1;
+#X connect 7 0 8 0;
+#X connect 8 0 9 0;
+#X connect 10 0 11 0;
+#X connect 11 0 23 0;
+#X connect 12 0 13 0;
+#X connect 14 0 15 0;
+#X connect 15 0 6 0;
+#X connect 23 0 13 0;
+#X connect 24 0 14 0;
+#X connect 24 0 14 1;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/karplusStrong/count.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,30 @@
+#N canvas 592 177 306 227 10;
+#X obj 32 111 f;
+#X obj 57 111 + 1;
+#X obj 57 151 sel 0;
+#X obj 57 171 del 1e-36;
+#X obj 57 131 > 1023;
+#X msg 32 90 0;
+#X obj 19 171 - 1;
+#X obj 19 193 outlet;
+#X obj 90 87 \$1;
+#X obj 90 109 - 1;
+#X obj 90 66 loadbang;
+#X obj 32 48 inlet;
+#X obj 32 69 t b;
+#X text 33 14 temporary replacement for [until] object;
+#X text 67 195 outputs index (0->[N-1]);
+#X connect 0 0 1 0;
+#X connect 1 0 0 1;
+#X connect 1 0 4 0;
+#X connect 1 0 6 0;
+#X connect 2 0 3 0;
+#X connect 3 0 0 0;
+#X connect 4 0 2 0;
+#X connect 5 0 0 0;
+#X connect 6 0 7 0;
+#X connect 8 0 9 0;
+#X connect 9 0 4 1;
+#X connect 10 0 8 0;
+#X connect 11 0 12 0;
+#X connect 12 0 5 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,186 @@
+#N canvas 8 23 1272 752 10;
+#X text 39 1 Hello world!;
+#X text 1003 636 ---------------------------------;
+#X text 1003 626 @krighxz / BELA / heavy / 12/2015;
+#X text 1027 645 beaglert.cc / enzienaudio.com;
+#X text 306 61 [pgmin];
+#X text 416 31 [ctlin];
+#X text 825 28 [notein];
+#X text 228 32 <== the bela equivalent of ==>;
+#X text 116 61 <== the bela equivalent of ==>;
+#X obj 740 28 bela_notein 0;
+#X obj 740 57 pack 0 0;
+#X obj 740 106 pack 0 0 0;
+#X obj 596 976 dac~ 1 2;
+#X obj 646 178 s voice-note-1;
+#X obj 736 178 s voice-note-2;
+#X obj 826 178 s voice-note-3;
+#X obj 916 178 s voice-note-4;
+#X obj 594 252 catch~ output;
+#X obj 541 57 voice 1;
+#X obj 1150 21 loadbang;
+#X obj 1150 43 metro 10;
+#X obj 1150 65 s metro;
+#X obj 944 336 * 1000;
+#X obj 1036 392 s Q;
+#X obj 944 381 s attack;
+#X obj 1036 336 * 8000;
+#X obj 1036 365 + 200;
+#X obj 1116 379 s~ wobble;
+#X obj 944 358 + 5;
+#X obj 1116 336 *~ 25;
+#X obj 629 334 r Q;
+#X obj 594 309 *~ 1;
+#X obj 981 565 clip~ 0 0.999;
+#X obj 836 610 *~;
+#X obj 740 128 route 1 2 3 4, f 36;
+#X obj 836 509 +~ 100;
+#X obj 836 485 *~ 1900;
+#X obj 624 651 delwrite~ ddd 2000;
+#X obj 740 84 poly 4 1;
+#X obj 664 729 *~ 0.5;
+#X obj 598 730 *~ 0.5;
+#X obj 597 757 hip~ 5;
+#X obj 624 558 hip~ 5;
+#X obj 624 580 hv_tanh;
+#X obj 836 667 hip~ 5;
+#X obj 836 689 hv_tanh;
+#X obj 836 566 delread~ ddd 500;
+#N canvas 0 22 178 130 snapshot 0;
+#X obj 24 20 inlet~;
+#X obj 24 62 snapshot~;
+#X obj 24 84 outlet;
+#X obj 86 29 r metro;
+#X connect 0 0 1 0;
+#X connect 1 0 2 0;
+#X connect 3 0 1 0;
+#X restore 846 538 pd snapshot;
+#X obj 78 173 bela_ctlin 1 0;
+#X obj 159 203 s c1;
+#X obj 41 61 bela_pgmin 0;
+#X obj 78 233 bela_ctlin 2 0;
+#X obj 159 263 s c2;
+#X obj 78 293 bela_ctlin 3 0;
+#X obj 159 323 s c3;
+#X obj 78 353 bela_ctlin 4 0;
+#X obj 159 383 s c4;
+#X obj 498 731 r c1;
+#X obj 498 753 / 127;
+#X obj 943 311 / 127;
+#X obj 943 289 r c2;
+#X obj 1033 311 / 127;
+#X obj 1033 289 r c3;
+#X obj 1115 291 / 127;
+#X obj 1115 269 r c4;
+#X obj 1115 313 sig~;
+#X obj 78 413 bela_ctlin 5 0;
+#X obj 159 443 s c5;
+#X obj 78 473 bela_ctlin 6 0;
+#X obj 159 503 s c6;
+#X obj 828 407 / 127;
+#X obj 828 429 sig~;
+#X obj 828 385 r c5;
+#X obj 981 519 / 127;
+#X obj 981 541 sig~;
+#X obj 981 497 r c6;
+#X obj 593 389 *~ 0.5;
+#X obj 777 562 vd~ ddd;
+#X obj 597 784 hv_tanh;
+#X obj 597 818 *~;
+#X obj 593 360 lop~ 1000;
+#X obj 828 458 lop~ 10;
+#X obj 78 533 bela_ctlin 7 0;
+#X obj 159 563 s c7;
+#X obj 78 593 bela_ctlin 8 0;
+#X obj 159 623 s c8;
+#X obj 664 769 loadbang;
+#X msg 665 792 0.75;
+#X obj 1116 456 * 8000;
+#X obj 1116 485 + 200;
+#X obj 1113 431 / 127;
+#X obj 1113 409 r c7;
+#X obj 1116 512 s QP;
+#X obj 653 850 lop~ 8000;
+#X obj 1065 725 r c8;
+#X obj 689 824 r QP;
+#X obj 1065 750 / 127;
+#X obj 1065 772 s octVol;
+#X obj 541 77 voice 2;
+#X obj 541 97 voice 3;
+#X obj 541 117 voice 4;
+#X connect 9 0 10 0;
+#X connect 9 1 10 1;
+#X connect 10 0 38 0;
+#X connect 11 0 34 0;
+#X connect 17 0 31 0;
+#X connect 19 0 20 0;
+#X connect 20 0 21 0;
+#X connect 22 0 28 0;
+#X connect 25 0 26 0;
+#X connect 26 0 23 0;
+#X connect 28 0 24 0;
+#X connect 29 0 27 0;
+#X connect 30 0 80 1;
+#X connect 31 0 80 0;
+#X connect 32 0 33 1;
+#X connect 33 0 39 0;
+#X connect 33 0 44 0;
+#X connect 34 0 13 0;
+#X connect 34 1 14 0;
+#X connect 34 2 15 0;
+#X connect 34 3 16 0;
+#X connect 35 0 47 0;
+#X connect 35 0 77 0;
+#X connect 36 0 35 0;
+#X connect 38 0 11 0;
+#X connect 38 1 11 1;
+#X connect 38 2 11 2;
+#X connect 39 0 41 0;
+#X connect 40 0 41 0;
+#X connect 41 0 78 0;
+#X connect 42 0 43 0;
+#X connect 43 0 37 0;
+#X connect 44 0 45 0;
+#X connect 45 0 37 0;
+#X connect 47 0 46 0;
+#X connect 48 0 49 0;
+#X connect 51 0 52 0;
+#X connect 53 0 54 0;
+#X connect 55 0 56 0;
+#X connect 57 0 58 0;
+#X connect 58 0 79 1;
+#X connect 59 0 22 0;
+#X connect 60 0 59 0;
+#X connect 61 0 25 0;
+#X connect 62 0 61 0;
+#X connect 63 0 65 0;
+#X connect 64 0 63 0;
+#X connect 65 0 29 0;
+#X connect 66 0 67 0;
+#X connect 68 0 69 0;
+#X connect 70 0 71 0;
+#X connect 71 0 81 0;
+#X connect 72 0 70 0;
+#X connect 73 0 74 0;
+#X connect 74 0 32 0;
+#X connect 75 0 73 0;
+#X connect 76 0 40 0;
+#X connect 76 0 42 0;
+#X connect 77 0 33 0;
+#X connect 78 0 79 0;
+#X connect 79 0 93 0;
+#X connect 80 0 76 0;
+#X connect 81 0 36 0;
+#X connect 82 0 83 0;
+#X connect 84 0 85 0;
+#X connect 86 0 87 0;
+#X connect 87 0 79 1;
+#X connect 88 0 89 0;
+#X connect 89 0 92 0;
+#X connect 90 0 88 0;
+#X connect 91 0 90 0;
+#X connect 93 0 12 0;
+#X connect 93 0 12 1;
+#X connect 94 0 96 0;
+#X connect 95 0 93 1;
+#X connect 96 0 97 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/backmain.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,79 @@
+#N canvas 85 58 1280 685 10;
+#X obj 19 396 dac~ 1 2;
+#X text 39 1 Hello world!;
+#X text 37 11 ============;
+#X obj 31 353 *~ 0.1;
+#X text 202 448 ---------------------------------;
+#X text 202 438 @krighxz / BELA / heavy / 12/2015;
+#X text 226 457 beaglert.cc / enzienaudio.com;
+#X obj 39 132 mtof;
+#X floatatom 79 59 5 0 0 0 - - -, f 5;
+#X floatatom 122 59 5 0 0 0 - - -, f 5;
+#X floatatom 41 63 5 0 0 0 - - -, f 5;
+#X obj 26 170 phasor~ 440;
+#X text 100 363 (the harsh sound of success);
+#X obj 260 83 * 20;
+#X obj 261 112 + 200;
+#X obj 132 85 loadbang;
+#X msg 131 111 1000;
+#X obj 125 166 sig~;
+#X text 673 121 [pgmin];
+#X obj 218 178 * 10;
+#X obj 30 250 hip~;
+#X obj 26 275 hip~;
+#X obj 20 302 hip~;
+#X obj 19 323 hip~;
+#X obj 87 192 lop~ 4;
+#X obj 152 256 line;
+#X obj 200 200 + 50;
+#X msg 170 227 \$1 200;
+#X msg 93 114 0;
+#X msg 185 112 0;
+#X obj 29 224 vcf~ 4;
+#X obj 70 333 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 42 92 mono;
+#X obj 40 28 bela_notein 0;
+#X obj 260 60 bela_ctlin 5 0;
+#X obj 220 151 bela_ctlin 6 0;
+#X obj 408 121 bela_pgmin 3;
+#X text 537 58 [ctlin];
+#X text 332 27 [notein];
+#X text 140 29 <== the bela equivalent of ==>;
+#X text 349 59 <== the bela equivalent of ==>;
+#X text 483 121 <== the bela equivalent of ==>;
+#X connect 3 0 0 1;
+#X connect 3 0 0 0;
+#X connect 7 0 11 0;
+#X connect 8 0 32 1;
+#X connect 10 0 32 0;
+#X connect 11 0 30 0;
+#X connect 13 0 14 0;
+#X connect 14 0 17 0;
+#X connect 15 0 16 0;
+#X connect 15 0 29 0;
+#X connect 15 0 28 0;
+#X connect 16 0 17 0;
+#X connect 17 0 24 0;
+#X connect 19 0 26 0;
+#X connect 20 0 21 0;
+#X connect 21 0 22 0;
+#X connect 22 0 23 0;
+#X connect 23 0 3 0;
+#X connect 24 0 30 1;
+#X connect 25 0 20 1;
+#X connect 25 0 21 1;
+#X connect 25 0 22 1;
+#X connect 25 0 23 1;
+#X connect 26 0 27 0;
+#X connect 27 0 25 0;
+#X connect 29 0 19 0;
+#X connect 30 0 20 0;
+#X connect 31 0 3 1;
+#X connect 32 0 7 0;
+#X connect 32 1 31 0;
+#X connect 33 0 10 0;
+#X connect 33 1 8 0;
+#X connect 33 2 9 0;
+#X connect 34 0 13 0;
+#X connect 35 0 19 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/bela_ctlin.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,76 @@
+#N canvas 554 204 703 475 10;
+#X obj 53 431 outlet;
+#X obj 108 429 outlet;
+#X obj 99 127 r ctlin;
+#X obj 92 151 unpack f f f;
+#X obj 154 429 outlet;
+#X obj 86 322 spigot;
+#X text 51 450 value;
+#X text 105 450 number;
+#X obj 211 138 loadbang;
+#X obj 177 193 \$1;
+#X obj 126 214 == \$1;
+#X obj 119 299 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 84 349 unpack f f f;
+#X obj 143 247 ||;
+#X obj 177 217 == 0;
+#X text 70 11 Emulates ctlin \, except number of outlets is fixed.
+;
+#X obj 228 247 ||;
+#X obj 262 217 == 0;
+#X obj 212 214 == \$2;
+#X obj 262 193 \$2;
+#X obj 177 277 &&;
+#X obj 211 163 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X text 173 299 If (inputController==\$1 || \$1==0) && (inputChannel==\$2
+|| \$2==0);
+#X text 197 52 Known issue: trying to select inputController 0 will
+result i accepting all controllers (because a pd external cannot tell
+the difference between \$1 been explicitely set to 0 and the patch
+not having any argument);
+#X text 162 450 channel;
+#X obj 56 192 pack f f f;
+#X obj 80 408 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X floatatom 81 382 5 0 0 0 - - -, f 5;
+#X obj 121 412 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X floatatom 122 386 5 0 0 0 - - -, f 5;
+#X obj 172 406 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X floatatom 173 380 5 0 0 0 - - -, f 5;
+#X obj 122 92 s ctlin;
+#X obj 64 29 r bela_ctlin;
+#X connect 2 0 3 0;
+#X connect 3 0 25 0;
+#X connect 3 1 10 0;
+#X connect 3 1 25 1;
+#X connect 3 2 18 0;
+#X connect 3 2 25 2;
+#X connect 5 0 12 0;
+#X connect 8 0 21 0;
+#X connect 9 0 14 0;
+#X connect 10 0 13 0;
+#X connect 11 0 5 1;
+#X connect 12 0 0 0;
+#X connect 12 0 27 0;
+#X connect 12 1 1 0;
+#X connect 12 1 29 0;
+#X connect 12 2 4 0;
+#X connect 12 2 31 0;
+#X connect 13 0 20 0;
+#X connect 14 0 13 1;
+#X connect 16 0 20 1;
+#X connect 17 0 16 1;
+#X connect 18 0 16 0;
+#X connect 19 0 17 0;
+#X connect 20 0 11 0;
+#X connect 21 0 9 0;
+#X connect 21 0 19 0;
+#X connect 25 0 5 0;
+#X connect 27 0 26 0;
+#X connect 29 0 28 0;
+#X connect 31 0 30 0;
+#X connect 33 0 32 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/bela_notein.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,59 @@
+#N canvas 399 23 629 685 10;
+#X obj 94 481 outlet;
+#X obj 165 475 outlet;
+#X obj 141 184 unpack f f f;
+#X obj 221 472 outlet;
+#X obj 143 365 spigot;
+#X obj 268 181 loadbang;
+#X obj 234 236 \$1;
+#X obj 183 257 == \$1;
+#X obj 176 342 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 143 392 unpack f f f;
+#X obj 200 290 ||;
+#X obj 234 260 == 0;
+#X obj 268 206 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X text 160 65 Emulates notein \, except number of outlets is fixed.
+;
+#X text 92 499 notenumber;
+#X text 161 489 velocity;
+#X text 220 486 channel;
+#X text 229 311 If (inputChannel==\$1 || \$1==0);
+#X obj 115 232 pack f f f;
+#X obj 82 428 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X floatatom 84 410 5 0 0 0 - - -, f 5;
+#X obj 145 448 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X floatatom 147 430 5 0 0 0 - - -, f 5;
+#X obj 198 451 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X floatatom 200 433 5 0 0 0 - - -, f 5;
+#X obj 131 112 s notein;
+#X obj 140 149 r notein;
+#X obj 135 84 r bela_notein;
+#X connect 2 0 18 0;
+#X connect 2 1 18 1;
+#X connect 2 2 7 0;
+#X connect 2 2 18 2;
+#X connect 4 0 9 0;
+#X connect 5 0 12 0;
+#X connect 6 0 11 0;
+#X connect 7 0 10 0;
+#X connect 8 0 4 1;
+#X connect 9 0 0 0;
+#X connect 9 0 20 0;
+#X connect 9 1 1 0;
+#X connect 9 1 22 0;
+#X connect 9 2 3 0;
+#X connect 9 2 24 0;
+#X connect 10 0 8 0;
+#X connect 11 0 10 1;
+#X connect 12 0 6 0;
+#X connect 18 0 4 0;
+#X connect 20 0 19 0;
+#X connect 22 0 21 0;
+#X connect 24 0 23 0;
+#X connect 26 0 2 0;
+#X connect 27 0 25 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/bela_pgmin.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,53 @@
+#N canvas 623 197 587 546 10;
+#X obj 73 413 outlet;
+#X obj 174 411 outlet;
+#X obj 106 304 spigot;
+#X obj 197 126 loadbang;
+#X obj 197 175 \$1;
+#X obj 146 196 == \$1;
+#X obj 139 281 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 163 229 ||;
+#X obj 197 199 == 0;
+#X obj 197 151 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X obj 89 111 unpack f f;
+#X obj 106 332 unpack f f;
+#X text 52 436 program number;
+#X text 182 432 channel;
+#X text 193 281 If (channel==\$1 || \$1==0);
+#X text 167 45 emulates pgmin except it always has 2 outlets and requires
+the input parameters to be explicitely set. Set them to 0 to avoid
+filtering;
+#X text 262 132 It'd be nicer with [t l l] but have to compensate for
+the fact that it is not supported by heavy;
+#X obj 77 183 pack f f;
+#X floatatom 72 355 5 0 0 0 - - -, f 5;
+#X obj 63 379 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X floatatom 150 361 5 0 0 0 - - -, f 5;
+#X obj 141 385 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X obj 91 63 s pgmin;
+#X obj 93 87 r pgmin;
+#X obj 89 42 r bela_pgmin;
+#X connect 2 0 11 0;
+#X connect 3 0 9 0;
+#X connect 4 0 8 0;
+#X connect 5 0 7 0;
+#X connect 6 0 2 1;
+#X connect 7 0 6 0;
+#X connect 8 0 7 1;
+#X connect 9 0 4 0;
+#X connect 10 0 17 0;
+#X connect 10 1 5 0;
+#X connect 10 1 17 1;
+#X connect 11 0 0 0;
+#X connect 11 0 18 0;
+#X connect 11 1 1 0;
+#X connect 11 1 20 0;
+#X connect 17 0 2 0;
+#X connect 18 0 19 0;
+#X connect 20 0 21 0;
+#X connect 23 0 10 0;
+#X connect 24 0 22 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/countTo.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,33 @@
+#N canvas 584 255 450 300 10;
+#X obj 65 196 outlet;
+#X obj 98 8 inlet;
+#X text 189 49 counts from 0 to the given number -1;
+#X text 214 95 inlet:N;
+#X text 212 112 outlet: 0 \, 1 \, 2 \, ... \, N-1;
+#X obj 143 61 f;
+#X obj 100 138 spigot;
+#X obj 129 89 <;
+#X obj 129 113 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 95 28 t b f b;
+#X obj 92 74 + 1;
+#X obj 43 73 f;
+#X obj 63 104 t b f f f;
+#X msg 92 53 0;
+#X obj 16 144 delay 0;
+#X connect 1 0 9 0;
+#X connect 5 0 7 1;
+#X connect 6 0 11 0;
+#X connect 7 0 8 0;
+#X connect 8 0 6 1;
+#X connect 9 0 11 0;
+#X connect 9 1 5 0;
+#X connect 9 1 13 0;
+#X connect 10 0 11 1;
+#X connect 11 0 12 0;
+#X connect 12 0 14 0;
+#X connect 12 1 0 0;
+#X connect 12 2 7 0;
+#X connect 12 3 10 0;
+#X connect 13 0 11 1;
+#X connect 14 0 6 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/hv_tanh.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,27 @@
+#N canvas 254 82 445 331 10;
+#X obj 39 37 inlet~;
+#X obj 39 67 clip~ -3 3;
+#X obj 54 107 *~;
+#X obj 39 277 outlet~;
+#X obj 109 107 sig~ 27;
+#X obj 54 147 +~;
+#X obj 54 217 /~;
+#X obj 39 247 *~;
+#X obj 94 147 *~ 9;
+#X obj 94 187 +~;
+#X text 142 177 // http://www.musicdsp.org/showone.php?id=238;
+#X text 140 198 // y = x * (27 + x^2) / (27 + 9*x^2);
+#X text 144 152 // pade-approximation of tanh function;
+#X connect 0 0 1 0;
+#X connect 1 0 2 0;
+#X connect 1 0 2 1;
+#X connect 1 0 7 0;
+#X connect 2 0 5 0;
+#X connect 2 0 8 0;
+#X connect 4 0 5 1;
+#X connect 4 0 9 1;
+#X connect 5 0 6 0;
+#X connect 6 0 7 1;
+#X connect 7 0 3 0;
+#X connect 8 0 9 0;
+#X connect 9 0 6 1;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/maxArray.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,67 @@
+#N canvas 464 23 1002 583 10;
+#X obj 224 202 max;
+#X obj 265 200 f;
+#X obj 223 229 change;
+#X obj 223 251 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X obj 223 271 f;
+#X obj 259 323 f;
+#X obj 187 315 f;
+#X obj 268 282 t b b;
+#X obj 183 364 outlet;
+#X obj 269 379 outlet;
+#X obj 179 17 inlet;
+#X obj 225 177 tabread \$1;
+#X obj 230 88 t f f b;
+#X text 376 37 Finds a maximum value in an array;
+#X text 531 112 \$2 arrayLength;
+#X obj 292 10 inlet;
+#X text 425 146 inlets:;
+#X text 473 144 1 bang to find the maximum;
+#X text 475 157 2 arrayLength;
+#X text 407 90 creation arguments: \$1 arrayName;
+#X obj 246 38 \$2;
+#X obj 202 40 bang;
+#X obj 230 65 f;
+#X obj 224 6 loadbang;
+#X text 444 216 outlets: 1 maximum value;
+#X text 498 231 2 index of the maximum value;
+#X msg 276 141 -1e+15;
+#X obj 222 142 t f f f;
+#X floatatom 158 266 5 0 0 0 - - -, f 5;
+#X obj 332 302 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
+-1 -1;
+#X obj 288 253 select 127;
+#X floatatom 301 324 5 0 0 0 - - -, f 5;
+#X text 362 256 heavy does not support the second inlet of select so
+I have to hard code 127 in here;
+#X obj 223 110 countTo;
+#X connect 0 0 1 0;
+#X connect 0 0 2 0;
+#X connect 0 0 6 1;
+#X connect 0 0 28 0;
+#X connect 1 0 0 1;
+#X connect 2 0 3 0;
+#X connect 3 0 4 0;
+#X connect 4 0 5 1;
+#X connect 4 0 31 0;
+#X connect 5 0 9 0;
+#X connect 6 0 8 0;
+#X connect 7 0 6 0;
+#X connect 7 1 5 0;
+#X connect 10 0 21 0;
+#X connect 11 0 0 0;
+#X connect 12 0 33 0;
+#X connect 12 2 26 0;
+#X connect 15 0 22 1;
+#X connect 20 0 22 1;
+#X connect 21 0 22 0;
+#X connect 22 0 12 0;
+#X connect 23 0 20 0;
+#X connect 26 0 0 1;
+#X connect 27 0 30 0;
+#X connect 27 1 11 0;
+#X connect 27 2 4 1;
+#X connect 30 0 7 0;
+#X connect 30 0 29 0;
+#X connect 33 0 27 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/mono.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,106 @@
+#N canvas 69 182 1020 452 10;
+#N canvas 0 23 450 278 (subpatch) 0;
+#X array \$0-activeNotes 127 float 2;
+#X coords 0 100 127 -1 200 140 1 0 0;
+#X restore 537 25 graph;
+#N canvas 89 91 450 300 resetArray 1;
+#X obj 121 145 tabwrite \$0-activeNotes;
+#X msg 123 115 -0.5;
+#X obj 154 90 t b f;
+#X msg 129 45 128;
+#X obj 103 14 inlet;
+#X text 212 71 countto;
+#X obj 153 69 countTo;
+#X connect 1 0 0 0;
+#X connect 2 0 1 0;
+#X connect 2 1 0 1;
+#X connect 3 0 6 0;
+#X connect 4 0 3 0;
+#X connect 6 0 2 0;
+#X restore 83 49 pd resetArray;
+#X obj 83 3 loadbang;
+#X obj 88 25 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X obj 39 188 maxArray \$0-activeNotes 128;
+#X obj 183 219 f;
+#X obj 39 237 >= 0;
+#X floatatom 148 313 5 0 0 0 - - -, f 5;
+#X obj 70 262 select 1;
+#X obj 366 87 f;
+#X obj 426 60 f;
+#X text 334 39 note;
+#X text 424 16 velocity;
+#X obj 423 87 > 0;
+#X obj 344 281 tabwrite \$0-activeNotes;
+#X obj 330 207 f;
+#X floatatom 303 178 5 0 0 0 - - -, f 5;
+#X obj 265 163 f;
+#X obj 304 150 + 1;
+#X msg 277 130 -1;
+#X msg 371 207 -0.5;
+#X obj 400 116 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 367 137 spigot;
+#X obj 447 134 spigot;
+#X obj 460 86 == 0;
+#X obj 470 109 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
+1;
+#X obj 434 163 t b f;
+#X obj 337 60 inlet;
+#X obj 426 32 inlet;
+#X obj 148 350 outlet;
+#X obj 275 89 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X obj 41 163 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
+-1;
+#X obj 363 172 t b f;
+#X obj 334 240 t b f;
+#X text 578 211 pass in velocity/note pairs (velocity first) and obtain
+as an output the most recent note still active;
+#X text 581 268 outlet 2 gate;
+#X obj 330 366 outlet;
+#X text 583 251 outlet 1 most recent note active (-1 otherwise);
+#X text 155 329 most recent note active;
+#X msg 338 338 0;
+#X msg 378 339 1;
+#X connect 2 0 3 0;
+#X connect 3 0 1 0;
+#X connect 3 0 19 0;
+#X connect 4 0 6 0;
+#X connect 4 1 5 1;
+#X connect 5 0 7 0;
+#X connect 5 0 40 0;
+#X connect 6 0 8 0;
+#X connect 7 0 29 0;
+#X connect 8 0 5 0;
+#X connect 8 1 39 0;
+#X connect 9 0 22 0;
+#X connect 9 0 23 0;
+#X connect 10 0 13 0;
+#X connect 10 0 24 0;
+#X connect 13 0 21 0;
+#X connect 15 0 33 0;
+#X connect 16 0 15 1;
+#X connect 17 0 18 0;
+#X connect 18 0 16 0;
+#X connect 18 0 17 1;
+#X connect 19 0 17 1;
+#X connect 20 0 33 0;
+#X connect 21 0 22 1;
+#X connect 22 0 32 0;
+#X connect 23 0 26 0;
+#X connect 24 0 25 0;
+#X connect 25 0 23 1;
+#X connect 26 0 20 0;
+#X connect 26 1 14 1;
+#X connect 27 0 9 0;
+#X connect 28 0 10 0;
+#X connect 28 0 30 0;
+#X connect 30 0 17 0;
+#X connect 31 0 4 0;
+#X connect 32 0 15 0;
+#X connect 32 1 14 1;
+#X connect 33 0 31 0;
+#X connect 33 1 14 0;
+#X connect 39 0 36 0;
+#X connect 40 0 36 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/midi-poly-synth/voice.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,55 @@
+#N canvas 145 23 388 553 10;
+#X obj 45 25 r voice-note-\$1;
+#X obj 45 47 unpack 0 0;
+#X obj 45 103 mtof;
+#X obj 111 446 *~;
+#X obj 152 367 / 127;
+#X obj 111 500 throw~ output;
+#X obj 152 419 line~;
+#X obj 152 393 pack 0 0;
+#X obj 59 176 phasor~;
+#X obj 129 176 phasor~;
+#X obj 86 304 +~;
+#X obj 197 367 r attack;
+#X obj 137 115 sig~;
+#X obj 137 137 +~;
+#X obj 171 113 r~ wobble;
+#X obj 59 198 *~ 2;
+#X obj 59 220 -~ 1;
+#X obj 129 198 *~ 2;
+#X obj 129 220 -~ 1;
+#X obj 310 280 r octVol;
+#X obj 268 397 osc~;
+#X obj 268 419 *~;
+#X obj 111 371 +~;
+#X obj 123 475 *~ 0.5;
+#X obj 268 364 mtof;
+#X obj 267 323 - 12;
+#X connect 0 0 1 0;
+#X connect 1 0 2 0;
+#X connect 1 0 25 0;
+#X connect 1 1 4 0;
+#X connect 2 0 8 0;
+#X connect 2 0 12 0;
+#X connect 3 0 23 0;
+#X connect 4 0 7 0;
+#X connect 6 0 3 1;
+#X connect 7 0 6 0;
+#X connect 8 0 15 0;
+#X connect 9 0 17 0;
+#X connect 10 0 22 0;
+#X connect 11 0 7 1;
+#X connect 12 0 13 0;
+#X connect 13 0 9 0;
+#X connect 14 0 13 1;
+#X connect 15 0 16 0;
+#X connect 16 0 10 1;
+#X connect 17 0 18 0;
+#X connect 18 0 10 0;
+#X connect 19 0 21 1;
+#X connect 20 0 21 0;
+#X connect 21 0 22 1;
+#X connect 22 0 3 0;
+#X connect 23 0 5 0;
+#X connect 24 0 20 0;
+#X connect 25 0 24 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/rubberDuckie/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,247 @@
+#N canvas 275 573 442 207 10;
+#X obj 39 149 dac~ 1 2;
+#X text 218 143 ---------------------------------;
+#X text 218 133 @krighxz / BELA / heavy / 12/2015;
+#X text 242 152 beaglert.cc / enzienaudio.com;
+#X text 35 32 Rubber Duckie;
+#X text 35 42 =============;
+#N canvas 325 86 823 605 duckie 0;
+#X obj 186 510 *~;
+#X obj 306 517 *~;
+#X obj 372 468 noise~;
+#X obj 372 492 bp~ 3000 10;
+#X obj 366 162 abs~;
+#X obj 509 158 sig~ 0;
+#X obj 467 371 +~;
+#X obj 411 210 -~ 0.55;
+#X obj 411 231 /~ 0.2;
+#X obj 411 252 clip~ 0 1;
+#X obj 411 282 *~;
+#X obj 411 308 *~ 1000;
+#X obj 366 334 +~;
+#X obj 366 308 +~ 800;
+#X obj 186 536 *~ 0.2;
+#X obj 366 282 *~ 800;
+#N canvas 568 775 135 226 <~ 0;
+#X obj 20 -43 -~;
+#X obj 20 27 +~ 1e-37;
+#X obj 20 6 clip~ -1e-37 0;
+#X obj 20 -88 inlet~;
+#X obj 64 -87 inlet~;
+#X obj 20 -64 max~;
+#X obj 20 -19 *~ -1;
+#X obj 20 79 outlet~;
+#X obj 20 48 *~ 1e+37;
+#X connect 0 0 6 0;
+#X connect 1 0 8 0;
+#X connect 2 0 1 0;
+#X connect 3 0 5 0;
+#X connect 4 0 0 1;
+#X connect 4 0 5 1;
+#X connect 5 0 0 0;
+#X connect 6 0 2 0;
+#X connect 8 0 7 0;
+#X restore 186 367 pd <~ 0;
+#N canvas 568 775 135 226 <~ 0;
+#X obj 20 -43 -~;
+#X obj 20 27 +~ 1e-37;
+#X obj 20 6 clip~ -1e-37 0;
+#X obj 20 -88 inlet~;
+#X obj 64 -87 inlet~;
+#X obj 20 -64 max~;
+#X obj 20 -19 *~ -1;
+#X obj 20 79 outlet~;
+#X obj 20 48 *~ 1e+37;
+#X connect 0 0 6 0;
+#X connect 1 0 8 0;
+#X connect 2 0 1 0;
+#X connect 3 0 5 0;
+#X connect 4 0 0 1;
+#X connect 4 0 5 1;
+#X connect 5 0 0 0;
+#X connect 6 0 2 0;
+#X connect 8 0 7 0;
+#X restore 482 181 pd <~;
+#X obj 186 428 *~;
+#X obj 225 345 sig~ 1;
+#X obj 186 394 lop~ 10;
+#X text 524 393 << band-limited pulse;
+#X text 10 367 reed stops when overblown >>;
+#X obj 186 460 *~;
+#X obj 186 480 *~;
+#X text 25 470 exp amplitude response >>;
+#X text 537 238 << detune frequency on change of direction;
+#X obj 482 236 *~ -300;
+#X obj 306 485 *~;
+#X obj 306 449 *~ 0.1;
+#X text 448 493 << noise simulates non-oscillating airflow when overblown
+;
+#X obj 186 561 outlet~;
+#N canvas 58 1286 407 525 pulse 0;
+#X obj 83 340 *~;
+#X obj 83 366 *~;
+#X obj 31 366 sig~ 1;
+#X obj 52 419 /~;
+#X obj 67 393 +~;
+#X text 116 366 X^2;
+#X text 97 396 1+X^2;
+#X text 84 421 1/(1+X^2);
+#X obj 83 317 osc~;
+#X obj 167 320 clip~ 0 999999;
+#X obj 52 483 outlet~;
+#X text 271 218 << tweak all this;
+#X text 25 14 based on F04.waveshaping.pulse.pd;
+#X obj 83 84 /~ 2;
+#X obj 167 148 -~;
+#X obj 167 170 /~ 127;
+#X obj 52 452 hip~ 5;
+#X obj 167 202 *~;
+#X obj 167 232 *~;
+#X obj 167 262 *~;
+#X obj 167 294 *~ 1000;
+#X obj 118 125 sig~ 139;
+#X obj 83 58 inlet~;
+#N canvas 331 1377 215 174 _ftom 0;
+#X obj -482 -776 tabread4~ _ftom;
+#X obj -482 -818 /~ 10000;
+#X obj -482 -797 *~ 256;
+#N canvas 0 23 450 278 (subpatch) 0;
+#X array _ftom 256 float 3;
+#A 0 0 27.0762 39.0762 46.0958 51.0762 54.9394 58.0958 60.7645 63.0762
+65.1153 66.9394 68.5894 70.0958 71.4815 72.7645 73.9589 75.0762 76.1258
+77.1153 78.0514 78.9394 79.784 80.5894 81.359 82.0958 82.8025 83.4815
+84.1349 84.7645 85.372 85.9589 86.5266 87.0762 87.609 88.1258 88.6276
+89.1153 89.5897 90.0514 90.5011 90.9394 91.3669 91.784 92.1914 92.5894
+92.9785 93.359 93.7313 94.0958 94.4528 94.8025 95.1453 95.4815 95.8113
+96.1349 96.4525 96.7645 97.0709 97.372 97.6679 97.9589 98.2451 98.5266
+98.8036 99.0762 99.3446 99.609 99.8693 100.126 100.379 100.628 100.873
+101.115 101.354 101.59 101.822 102.051 102.278 102.501 102.722 102.939
+103.154 103.367 103.577 103.784 103.989 104.191 104.392 104.589 104.785
+104.978 105.17 105.359 105.546 105.731 105.914 106.096 106.275 106.453
+106.629 106.803 106.975 107.145 107.314 107.482 107.647 107.811 107.974
+108.135 108.294 108.453 108.609 108.764 108.918 109.071 109.222 109.372
+109.521 109.668 109.814 109.959 110.103 110.245 110.386 110.527 110.666
+110.804 110.94 111.076 111.211 111.345 111.477 111.609 111.74 111.869
+111.998 112.126 112.253 112.379 112.504 112.628 112.751 112.873 112.995
+113.115 113.235 113.354 113.472 113.59 113.706 113.822 113.937 114.051
+114.165 114.278 114.39 114.501 114.612 114.722 114.831 114.939 115.047
+115.154 115.261 115.367 115.472 115.577 115.681 115.784 115.887 115.989
+116.09 116.191 116.292 116.392 116.491 116.589 116.688 116.785 116.882
+116.978 117.074 117.17 117.265 117.359 117.453 117.546 117.639 117.731
+117.823 117.915 118.005 118.096 118.186 118.275 118.364 118.453 118.541
+118.629 118.716 118.803 118.889 118.975 119.06 119.145 119.23 119.314
+119.398 119.482 119.565 119.647 119.729 119.811 119.893 119.974 120.055
+120.135 120.215 120.294 120.374 120.453 120.531 120.609 120.687 120.764
+120.842 120.918 120.995 121.071 121.147 121.222 121.297 121.372 121.446
+121.521 121.594 121.668 121.741 121.814 121.887 121.959 122.031 122.103
+122.174 122.245 122.316 122.386 122.457 122.527 122.596 122.666 122.735
+122.804 122.872 122.94 123.008;
+#X coords 0 127 256 -12 50 50 1 0 0;
+#X restore -546 -793 graph;
+#X obj -482 -838 clip~ 0 10000;
+#X obj -482 -858 inlet~;
+#X obj -482 -755 outlet~;
+#X text -567 -722 shoddy temp replacement for ftom~;
+#X connect 0 0 6 0;
+#X connect 1 0 2 0;
+#X connect 2 0 0 0;
+#X connect 4 0 1 0;
+#X connect 5 0 4 0;
+#X restore 182 125 pd _ftom;
+#X connect 0 0 1 0;
+#X connect 0 0 1 1;
+#X connect 1 0 4 1;
+#X connect 2 0 3 0;
+#X connect 2 0 4 0;
+#X connect 3 0 16 0;
+#X connect 4 0 3 1;
+#X connect 8 0 0 0;
+#X connect 9 0 0 1;
+#X connect 13 0 8 0;
+#X connect 13 0 23 0;
+#X connect 14 0 15 0;
+#X connect 15 0 17 0;
+#X connect 15 0 17 1;
+#X connect 16 0 10 0;
+#X connect 17 0 18 0;
+#X connect 17 0 18 1;
+#X connect 18 0 19 0;
+#X connect 18 0 19 1;
+#X connect 19 0 20 0;
+#X connect 20 0 9 0;
+#X connect 21 0 14 0;
+#X connect 22 0 13 0;
+#X connect 23 0 14 1;
+#X restore 467 394 pd pulse;
+#X obj 366 105 inlet~;
+#X text 295 104 airflow >>;
+#X text 218 118 ('squeeze pressure');
+#X text 13 20 Sound model of a rubber duckie toy;
+#X text 14 51 Hint: use the differential of a continuous signal as
+input for a realistic response;
+#X text 13 29 ==================================;
+#X obj 366 133 lop~ 0.5;
+#X text 423 132 << airflow resistance;
+#X connect 0 0 14 0;
+#X connect 1 0 14 0;
+#X connect 2 0 3 0;
+#X connect 3 0 1 1;
+#X connect 4 0 7 0;
+#X connect 4 0 15 0;
+#X connect 4 0 29 0;
+#X connect 4 0 18 1;
+#X connect 4 0 16 0;
+#X connect 5 0 17 1;
+#X connect 6 0 32 0;
+#X connect 7 0 8 0;
+#X connect 8 0 9 0;
+#X connect 9 0 10 0;
+#X connect 9 0 10 1;
+#X connect 10 0 11 0;
+#X connect 11 0 12 1;
+#X connect 12 0 6 0;
+#X connect 13 0 12 0;
+#X connect 14 0 31 0;
+#X connect 15 0 13 0;
+#X connect 16 0 20 0;
+#X connect 17 0 27 0;
+#X connect 18 0 23 0;
+#X connect 18 0 23 1;
+#X connect 19 0 16 1;
+#X connect 20 0 18 0;
+#X connect 23 0 24 1;
+#X connect 23 0 24 0;
+#X connect 24 0 0 0;
+#X connect 27 0 6 1;
+#X connect 28 0 1 0;
+#X connect 29 0 28 1;
+#X connect 29 0 28 0;
+#X connect 32 0 0 1;
+#X connect 33 0 39 0;
+#X connect 39 0 4 0;
+#X connect 39 0 17 0;
+#X restore 39 127 pd duckie;
+#X obj 39 77 adc~ 10;
+#N canvas 0 22 504 229 differential 0;
+#X obj 27 80 rzero~ 1;
+#X obj 72 57 sig~ 1;
+#X obj 27 152 *~ 44100;
+#X obj 72 131 samplerate~;
+#X obj 72 110 loadbang;
+#X obj 27 25 inlet~;
+#X obj 27 190 outlet~;
+#X text 84 83 y[n] = x[n] - x[n-1];
+#X text 121 57 <<< heavy currently requires signal input to set coefficient
+;
+#X text 91 151 <<< multiply by samplerate to get velocity;
+#X connect 0 0 2 0;
+#X connect 1 0 0 1;
+#X connect 2 0 6 0;
+#X connect 3 0 2 1;
+#X connect 4 0 3 0;
+#X connect 5 0 0 0;
+#X restore 39 101 pd differential;
+#X connect 6 0 0 0;
+#X connect 6 0 0 1;
+#X connect 7 0 8 0;
+#X connect 8 0 6 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/samphold/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,22 @@
+#N canvas 275 504 414 247 10;
+#X obj 55 212 dac~ 1 2;
+#X obj 55 171 *~ 0.1;
+#X obj 55 150 osc~ 440;
+#X obj 55 61 noise~;
+#X obj 55 128 samphold~;
+#X text 51 21 ==================;
+#X text 51 11 Sample and hold FM;
+#X obj 55 85 lop~ 1;
+#X obj 122 107 phasor~ 880;
+#X obj 55 107 *~ 1e+06;
+#X text 193 199 ---------------------------------;
+#X text 193 189 @krighxz / BELA / heavy / 12/2015;
+#X text 217 208 beaglert.cc / enzienaudio.com;
+#X connect 1 0 0 1;
+#X connect 1 0 0 0;
+#X connect 2 0 1 0;
+#X connect 3 0 7 0;
+#X connect 4 0 2 0;
+#X connect 7 0 9 0;
+#X connect 8 0 4 1;
+#X connect 9 0 4 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/simple-fm/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,42 @@
+#N canvas 275 573 409 282 10;
+#X obj 37 230 dac~ 1 2;
+#X text 175 222 ---------------------------------;
+#X obj 37 102 *~ 1000;
+#X obj 37 199 *~ 0.1;
+#X obj 37 79 adc~ 3;
+#X obj 65 125 adc~ 4;
+#X obj 94 147 adc~ 5;
+#N canvas 81 574 322 252 FMvoice 1;
+#X obj 20 174 +~;
+#X obj 93 116 osc~;
+#X obj 93 157 *~;
+#X obj 93 76 *~;
+#X obj 158 98 *~;
+#X obj 20 225 outlet~;
+#X obj 20 200 osc~;
+#X obj 20 39 inlet~ freq;
+#X obj 108 39 inlet~ harmRatio~;
+#X obj 173 69 inlet~ modIndex~;
+#X connect 0 0 6 0;
+#X connect 1 0 2 0;
+#X connect 2 0 0 1;
+#X connect 3 0 4 0;
+#X connect 3 0 1 0;
+#X connect 4 0 2 1;
+#X connect 6 0 5 0;
+#X connect 7 0 3 0;
+#X connect 7 0 0 0;
+#X connect 8 0 3 1;
+#X connect 9 0 4 1;
+#X restore 37 169 pd FMvoice;
+#X text 33 22 Simple FM;
+#X text 33 32 =========;
+#X text 175 212 @krighxz / BELA / heavy / 12/2015;
+#X text 199 231 beaglert.cc / enzienaudio.com;
+#X connect 2 0 7 0;
+#X connect 3 0 0 1;
+#X connect 3 0 0 0;
+#X connect 4 0 2 0;
+#X connect 5 0 7 1;
+#X connect 6 0 7 2;
+#X connect 7 0 3 0;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/08-PureData/tableScrubbing/_main.pd	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,35 @@
+#N canvas 503 319 861 473 10;
+#N canvas 0 22 450 278 (subpatch) 0;
+#X array exampleTable 65238 float 2;
+#X coords 0 1 65238 -1 200 140 1 0 0;
+#X restore 21 213 graph;
+#X obj 243 329 adc~ 1 2 3;
+#X obj 300 426 dac~;
+#X text 317 264 <<< delete this object before compiling!;
+#X obj 300 392 tabread4~ exampleTable;
+#X msg 243 218 read -resize rt.wav exampleTable;
+#X obj 300 350 lop~ 1;
+#X obj 300 371 *~ 65238;
+#X obj 243 266 soundfiler;
+#X text 489 229 right-click array -> properties -> toggle "save contents"
+;
+#X text 463 216 <<< after loading a file:;
+#X text 17 23 Load sample into table and scrub using analog input;
+#X text 17 33 ===================================================;
+#X text 362 351 <<< avoid zipper noise;
+#X text 362 371 <<< length of sample;
+#X text 317 328 <<< analog input 0 (range 0-1);
+#X text 631 419 ---------------------------------;
+#X text 631 409 @krighxz / BELA / heavy / 12/2015;
+#X text 655 428 beaglert.cc / enzienaudio.com;
+#X text 30 61 N.B. the patch cannot be bigger than 512kb in size \;
+only use this for *very small soundfiles*;
+#X text 29 99 You can modify the render.cpp file to load samples into
+tables using the Heavy API: https://enzienaudio.com/docs/c.html#api-hv-table
+;
+#X connect 1 2 6 0;
+#X connect 4 0 2 0;
+#X connect 4 0 2 1;
+#X connect 5 0 8 0;
+#X connect 6 0 7 0;
+#X connect 7 0 4 0;
Binary file examples/08-PureData/tableScrubbing/rt.wav has changed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/Junction.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,95 @@
+/*
+ *
+ * Excitation Junction for two waveguides
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+#include "Junction.h"
+#include "../include/Utilities.h"
+
+Junction::Junction()	{
+
+	setFrequency(440);
+	_dt = 1.0/44100.0;
+
+	// initialize variables
+	for(int i=0;i<WG_BUFFER_SIZE;i++)	{
+		_buffer_l[i] = 0;
+		_buffer_r[i] = 0;
+	}
+	_excitation = 0;
+	_lastPlectrumDisplacement = 0;
+	_readPtr = 0;
+
+}
+
+void Junction::update(float excitation, float left, float right)	{
+
+	// 1. advance delay buffer read pointer
+
+	if(++_readPtr>=WG_BUFFER_SIZE)
+		_readPtr=0;
+
+	// 2. add excitation sample into buffer
+
+	_buffer_l[(_readPtr+_delay_l+WG_BUFFER_SIZE)%WG_BUFFER_SIZE] = excitation;
+	_buffer_r[(_readPtr+_delay_r+WG_BUFFER_SIZE)%WG_BUFFER_SIZE] = excitation;
+
+	// 3. feed right input to left output and vice versa
+
+	_buffer_l[_readPtr] += right;
+	_buffer_r[_readPtr] += left;
+
+	// 4. store excitation value for later use
+	_excitation = excitation;
+
+}
+
+float Junction::getOutput(int direction)	{
+
+	if(direction = 0)
+		return _buffer_l[_readPtr];
+	else
+		return _buffer_r[_readPtr];
+
+}
+
+float Junction::getExcitationDisplacement()	{
+
+	// string displacement and excitation force
+	// use delayed value to account for excitation position
+	float in = _buffer_l[(_readPtr+_delay_l+WG_BUFFER_SIZE)%WG_BUFFER_SIZE] + _excitation;
+
+	// integrate total force
+	float out = 0.00001 * in + 0.99999 * _lastPlectrumDisplacement;
+
+	// store variable for next iteration
+	_lastPlectrumDisplacement = out;
+
+	// multiply by delta time
+	return out * _dt;
+
+}
+
+void Junction::setPluckPosition(float pluckPos){
+
+	pluckPos = constrain(pluckPos,0,1);
+	_delay_l = pluckPos * _periodInSamples;
+	_delay_r = (1-pluckPos) * _periodInSamples;
+
+}
+
+void Junction::setPeriod(float period)	{
+
+	_periodInMilliseconds = period;
+
+}
+
+void Junction::setFrequency(float frequency)	{
+
+	_periodInMilliseconds = 1000.0/frequency;
+	_periodInSamples = (int)(_periodInMilliseconds * 44.1);
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/Junction.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,54 @@
+/*
+ *
+ * Excitation Junction for two waveguides
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+#ifndef JUNCTION_H_
+#define JUNCTION_H_
+
+#include <cmath>
+
+#ifndef WG_BUFFER_SIZE
+#define WG_BUFFER_SIZE 4096
+#endif
+
+#ifndef		M_PI
+#define		M_PI		3.14159265358979323846264338
+#endif
+
+class Junction
+{
+
+public:
+
+	Junction();
+	void setup();
+	void update(float excitation, float left, float right);
+	float getOutput(int direction);
+	float getExcitationDisplacement();
+	void setFrequency(float frequency);
+	void setPeriod(float periodInMs);
+	void setPluckPosition(float pluckPos);
+
+private:
+
+	double _dt;
+	float _periodInMilliseconds;
+	int _periodInSamples;
+
+	int _delay_l;
+	int _delay_r;
+
+	float _buffer_l[WG_BUFFER_SIZE];
+	float _buffer_r[WG_BUFFER_SIZE];
+	int _readPtr;
+
+	float _excitation;
+	float _lastPlectrumDisplacement;
+
+};
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/MassSpringDamper.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,59 @@
+/*
+ *
+ * Simple 1-Dimensional Mass Spring Damper
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+#include "MassSpringDamper.h"
+
+MassSpringDamper::MassSpringDamper(float mass, float spring, float damp)	{
+
+	_dt = 1.0/44100.0;
+	_mass = mass;
+	_spring = spring;
+	_damp = damp;
+	_position = 0;
+	_velocity = 0;
+
+}
+
+void MassSpringDamper::setup()	{
+
+}
+
+double MassSpringDamper::update(float inForce)	{
+
+	// 1. calculate spring/damper forces using current position and velocity
+
+	double out = (_position * (double)_spring * -1) + (_velocity * (double)_damp * -1);
+
+	// 2. apply external force
+
+	out += inForce;
+
+	// 3. derive acceleration (a = f/m)
+
+	out /= (double)_mass;
+
+	// 4. derive velocity (v = a*dt)
+
+	out *= _dt;
+
+	// 5. apply previous velocity
+
+	out += _velocity;
+
+	// 6. save current velocity state for next iteration
+
+	_velocity = out;
+
+	// 7. derive new position (x[n] = x[n-1] + v[n]) and save for next iteration
+
+	out += _position;
+	_position = out;
+
+	return out;
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/MassSpringDamper.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,32 @@
+/*
+ *
+ * Simple 1-Dimensional Mass Spring Damper
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+#ifndef MASSSPRINGDAMPER_H_
+#define MASSSPRINGDAMPER_H_
+
+class MassSpringDamper
+{
+
+public:
+
+	MassSpringDamper(float mass, float spring, float damp);
+	void setup();
+	double update(float inForce);
+
+private:
+
+	double _dt;
+	float _mass;
+	float _spring;
+	float _damp;
+	double _position;
+	double _velocity;
+
+};
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/Plectrum.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,61 @@
+/*
+ *
+ * Plectrum model for touching and plucking strings
+ *
+ * Christian Heinrichs 04/2015
+ *
+ * [inspired by E. Berdahl's pluck~ abstraction for the FireFader]
+ *
+ */
+
+#include "Plectrum.h"
+
+#include "../include/Utilities.h"
+#include <cmath>
+#include <stdio.h>
+#include <cstdlib>
+
+Plectrum::Plectrum()	{
+
+	_contact = 0;
+	_lastDistance = 0;
+
+}
+
+void Plectrum::setup(float spring, float damp, float hyst)	{
+
+	_spring = spring;
+	_damp = damp;
+	_hyst = hyst;
+
+}
+
+float Plectrum::update(float position, float stringPosition)	{
+
+	float distance = position - stringPosition;
+
+	// Calculate spring/damp forces based on distance to string
+
+	float springOut = distance * _spring;
+
+	float dampOut = (distance - _lastDistance) * 44100;
+
+	float out = springOut+dampOut;
+
+	// If distance crosses zero, enable contact
+
+	if((distance>0 && _lastDistance<=0)||(distance<0 && _lastDistance>=0))
+		_contact = 1;
+
+	// If distance exceeds hysteresis threshold, jump to zero (i.e. 'pluck')
+
+	if(fabs(distance)>_hyst)
+		_contact = 0;
+
+	// FIXME: contact doesn't switch back to zero if distance goes back in original direction
+
+	_lastDistance = distance;
+
+	return out * _contact;
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/Plectrum.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,35 @@
+/*
+ *
+ * Plectrum model for touching and plucking strings
+ *
+ * [inspired by E. Berdahl's pluck~ abstraction for the FireFader]
+ *
+ */
+
+#ifndef PLECTRUM_H_
+#define PLECTRUM_H_
+
+class Plectrum
+{
+
+public:
+
+	Plectrum();
+	void setup(float spring, float damp, float hyst);
+	float update(float position, float stringPosition);
+
+private:
+
+	double _dt;
+	float _spring;
+	float _damp;
+	double _position;
+	double _velocity;
+	float _hyst;
+	float _lastDistance;
+	int _contact;
+
+};
+
+#endif
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/String.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,91 @@
+/*
+ *
+ * 1-D string consisting of two waveguides and junction
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+#include "String.h"
+#include "Junction.h"
+#include "Waveguide.h"
+
+#include "../include/Utilities.h"
+#include <rtdk.h>
+#include <cmath>
+#include <stdio.h>
+#include <cstdlib>
+
+String::String(){
+
+	wg_l = Waveguide();
+	wg_r = Waveguide();
+	junction = Junction();
+
+	junction.setPluckPosition(0.5);
+
+	_previous_l = 0;
+	_previous_r = 0;
+
+}
+
+float String::update(float in)	{
+
+	// 1. send excitation signal and previous waveguide outputs into junction
+
+	junction.update(in,_previous_l,_previous_r);
+
+	// 2. save new waveguide outputs for next iteration
+
+	_previous_l = wg_l.update(junction.getOutput(0));
+	_previous_r = wg_r.update(junction.getOutput(1));
+
+	// 3. use right waveguide as output
+
+	//rt_printf("BANANA %f ",_readPtr);
+	//rt_printf("%f\n",_previous_r);
+
+	return _previous_r;
+}
+
+float String::getPlectrumDisplacement()	{
+
+	return junction.getExcitationDisplacement();
+
+}
+
+void String::setPluckPosition(float pluckPos){
+
+	junction.setPluckPosition(pluckPos);
+
+}
+
+void String::setGlobalPosition(float pos)	{
+
+	_globalPosition = pos;
+
+}
+
+float String::getGlobalPosition()	{
+
+	return _globalPosition;
+
+}
+
+void String::setMidinote(float midinote)	{
+
+	float frequency = 440.0f*(float)powf(2,(midinote-57)/12.0f);
+
+	junction.setFrequency(frequency);
+	wg_l.setFrequency(frequency);
+	wg_r.setFrequency(frequency);
+
+}
+
+void String::setFrequency(float frequency)	{
+
+	junction.setFrequency(frequency);
+	wg_l.setFrequency(frequency);
+	wg_r.setFrequency(frequency);
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/String.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,45 @@
+/*
+ *
+ * 1-D string consisting of two waveguides and junction
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+#ifndef STRING_H_
+#define STRING_H_
+
+#include <cmath>
+#include "Junction.h"
+#include "Waveguide.h"
+
+class String
+{
+
+public:
+
+	String();
+	float update(float in);
+
+	float getPlectrumDisplacement();
+	void setMidinote(float midinote);
+	void setFrequency(float frequency);
+	void setPeriod(float periodInMs);
+	void setPluckPosition(float pluckPos);
+	void setGlobalPosition(float pos);
+	float getGlobalPosition();
+
+private:
+
+	float _previous_l;
+	float _previous_r;
+
+	float _globalPosition;
+
+	Waveguide wg_l;
+	Waveguide wg_r;
+	Junction junction;
+
+};
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/Waveguide.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,128 @@
+/*
+ *
+ * Simple 1-Dimensional Waveguide
+ *
+ */
+
+#include "Waveguide.h"
+#include "../include/Utilities.h"
+#include <rtdk.h>
+#include <cmath>
+#include <stdio.h>
+#include <cstdlib>
+
+#define DECAY 0.995//0.999
+#define DAMPING 0.01//0.05
+
+// TODO: make damping and decay parametrisable
+
+Waveguide::Waveguide()	{
+
+	// initialize variables
+	a1_lp = 0;
+	a2_lp = 0;
+	b0_lp = 0;
+	b1_lp = 0;
+	b2_lp = 0;
+	_dt = 1.0/44100.0;
+	setFrequency(440);
+	updateFilterCoeffs(8000);
+	_filterReadPtr=0;
+	for(int i=0;i<FILTER_BUFFER_SIZE;i++)	{
+		_filterBuffer_x[i] = 0;
+		_filterBuffer_y[i] = 0;
+	}
+	for(int i=0;i<WG_BUFFER_SIZE;i++)	{
+		_buffer[i] = 0;
+	}
+	_lastX = 0;
+	_lastY = 0;
+	_readPtr = 0;
+
+}
+
+void Waveguide::setup()	{
+
+}
+
+float Waveguide::update(float in)	{
+
+	// 1. advance delay buffer read pointer
+
+	if(++_readPtr>=WG_BUFFER_SIZE)
+		_readPtr=0;
+
+	// 2. write input into buffer
+
+	_buffer[_readPtr] = in;
+
+	// 3. read delayed sample from buffer
+
+	float out = _buffer[(_readPtr-_periodInSamples+WG_BUFFER_SIZE)%WG_BUFFER_SIZE];
+
+	// 4. apply damping (low-pass) filter to output
+
+	if(++_filterReadPtr>=FILTER_BUFFER_SIZE)
+			_filterReadPtr=0;
+
+	out = b0_lp*out +
+					b1_lp*_filterBuffer_x[(_filterReadPtr-1+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE] +
+					b2_lp*_filterBuffer_x[(_filterReadPtr-2+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE] -
+					a1_lp*_filterBuffer_y[(_filterReadPtr-1+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE] -
+					a2_lp*_filterBuffer_y[(_filterReadPtr-2+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE];
+
+	// 5. Simple high-pass filter to block DC-offset
+	// y[n] = x[n] - x[n-1] + a * y[n-1]
+	float gain = 0.9999;
+	float temp = out;
+	out = out - _lastX + gain * _lastY;
+	_lastY = out;
+	_lastX = temp;
+
+	// 6. Apply intensity damping
+	out *= DECAY;
+
+	_filterBuffer_x[_filterReadPtr] = in;
+	_filterBuffer_y[_filterReadPtr] = out;
+
+	return out;
+
+}
+
+void Waveguide::setFrequency(float frequency)	{
+
+	// NB: currently no interpolation, so may not be ideal for dynamically changing waveguide frequency
+	_periodInMilliseconds = 1000.0/frequency;
+	_periodInSamples = (int)(_periodInMilliseconds * 44.1);
+
+}
+
+void Waveguide::updateFilterCoeffs(float frequency)	{
+
+	// FIXME: Butterworth filter doesn't work very well,
+	//		  using simple FIR in the meantime
+
+	a1_lp = 0;
+	a2_lp = 0;
+	b0_lp = 1.0 - DAMPING;
+	b1_lp = DAMPING;
+	b2_lp = 0;
+
+	/*
+	// 'w' for sake of resembling lower-case 'omega'
+	float w = 2.0 * M_PI * frequency;
+	float t = _dt;
+	// The Q for a 2nd-order Butterworth is sqrt(2)/2
+	float q = 0.707;//sqrt(2.0)/2.0;
+
+	// low-pass filter coefficients
+	float a0_lp = w*w*t*t + 2*(w/q)*t + 4.0;
+	float k = 1.0/a0_lp;
+	a1_lp = (2.0*w*w*t*t - 8.0) * k;
+	a2_lp = (4.0 - (w/q)*2.0*t + w*w*t*t) * k;
+	b0_lp = (w*w*t*t) * k;
+	b1_lp = (2.0*w*w*t*t) * k;
+	b2_lp = (w*w*t*t) * k;
+	*/
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/Waveguide.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,57 @@
+/*
+ *
+ * Simple 1-Dimensional Waveguide
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+#ifndef WAVEGUIDE_H_
+#define WAVEGUIDE_H_
+
+#include <cmath>
+
+#ifndef WG_BUFFER_SIZE
+#define WG_BUFFER_SIZE 4096
+#endif
+
+#ifndef FILTER_BUFFER_SIZE
+#define FILTER_BUFFER_SIZE 4
+#endif
+
+#ifndef		M_PI
+#define		M_PI		3.14159265358979323846264338
+#endif
+
+class Waveguide
+{
+
+public:
+
+	Waveguide();
+	void setup();
+	float update(float in);
+	void updateFilterCoeffs(float frequency);
+	void setFrequency(float frequency);
+
+private:
+
+	double _dt;
+	float _periodInMilliseconds;
+	int _periodInSamples;
+
+	float _buffer[WG_BUFFER_SIZE];
+	int _readPtr;
+
+	float _filterBuffer_x[FILTER_BUFFER_SIZE];
+	float _filterBuffer_y[FILTER_BUFFER_SIZE];
+	float _hipBuffer_x[FILTER_BUFFER_SIZE];
+	float _hipBuffer_y[FILTER_BUFFER_SIZE];
+	int _filterReadPtr;
+
+	float b0_lp,b1_lp,b2_lp,a1_lp, a2_lp;
+	float _lastY,_lastX;
+
+};
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/airharp/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,211 @@
+/*
+ * AIR-HARP
+ * Physically modelled strings using waveguide junctions and mass-spring-dampers
+ *
+ * render.cpp
+ *
+ * Christian Heinrichs 04/2015
+ *
+ */
+
+
+#include "MassSpringDamper.h"
+#include "String.h"
+#include "Plectrum.h"
+
+#include <Bela.h>
+#include <cmath>
+#include <stdio.h>
+#include <cstdlib>
+#include <rtdk.h>
+
+#define ACCEL_BUF_SIZE 8
+#define NUMBER_OF_STRINGS 9
+
+// PENTATONIC SCALE
+float gMidinotes[NUMBER_OF_STRINGS] = {40,45,50,55,57,60,62,64,67};
+
+float gInverseSampleRate;
+
+float out_gain = 5.0;
+
+int accelPin_x = 0;
+int accelPin_y = 1;
+int accelPin_z = 2;
+
+MassSpringDamper msd = MassSpringDamper(1,0.1,10);// (10,0.001,10);
+String strings[NUMBER_OF_STRINGS];
+Plectrum plectrums[NUMBER_OF_STRINGS];
+
+float gPlectrumDisplacement = 0;
+
+float gAccel_x[ACCEL_BUF_SIZE] = {0};
+int gAccelReadPtr = 0;
+
+// DC BLOCK BUTTERWORTH
+
+// Coefficients for 100hz cut-off
+float a0_l = 0.9899759179893742;
+float a1_l = -1.9799518359787485;
+float a2_l = 0.9899759179893742;
+float a3_l = -1.979851353142371;
+float a4_l = 0.9800523188151258;
+
+float a0_r = a0_l;
+float a1_r = a1_l;
+float a2_r = a2_l;
+float a3_r = a3_l;
+float a4_r = a4_l;
+
+float x1_l = 0;
+float x2_l = 0;
+float y1_l = 0;
+float y2_l = 0;
+
+float x1_r = 0;
+float x2_r = 0;
+float y1_r = 0;
+float y2_r = 0;
+
+
+bool setup(BelaContext *context, void *userData)
+{
+
+	gInverseSampleRate = 1.0 / context->audioSampleRate;
+
+	// initialise strings & plectrums
+	for(int i=0;i<NUMBER_OF_STRINGS;i++)	{
+
+		plectrums[i] = Plectrum();
+		plectrums[i].setup(250,0.25,0.05);
+
+		strings[i] = String();
+		strings[i].setMidinote(gMidinotes[i]);
+
+		float spacing = 2.0 / (NUMBER_OF_STRINGS+1);
+
+		strings[i].setGlobalPosition( -1 + spacing*(i+1) );
+
+		rt_printf("STRING %d // midinote: %f position: %f\n",i,gMidinotes[i],( -1 + spacing*(i+1) ));
+
+	}
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+
+	float lastAccel = 0;
+
+	for(int n = 0; n < context->audioFrames; n++) {
+
+		/*
+		 *
+		 * ACCELEROMETER DATA
+		 *
+		 */
+
+		// Read accelerometer data from analog input
+		float accel_x = 0;
+		if(n%2)	{
+			accel_x = (float)context->analogIn[(n/2)*8+accelPin_x] * 2 - 1;	// 15800 - 28300 - 41500
+			lastAccel = accel_x;
+		} else {
+			// grab previous value if !n%2
+			accel_x = lastAccel;
+		}
+
+		// Dead-zone avoids noise when box is lying horizontally on a surface
+
+		float accelDeadZone = 0.1;
+
+		if(accel_x <= accelDeadZone && accel_x >= -accelDeadZone)
+			accel_x = 0;
+
+		// Perform smoothing (moving average) on acceleration value
+		if(++gAccelReadPtr >= ACCEL_BUF_SIZE)
+			gAccelReadPtr = 0;
+		gAccel_x[gAccelReadPtr] = accel_x;
+		float gravity = 0;
+		for(int i=0;i<ACCEL_BUF_SIZE;i++)	{
+			gravity = gAccel_x[(gAccelReadPtr-i+ACCEL_BUF_SIZE)%ACCEL_BUF_SIZE];
+		}
+		gravity /= ACCEL_BUF_SIZE;
+
+		/*
+		 *
+		 * PHYSICS SIMULATION
+		 *
+		 */
+
+		// The horizontal force (which can be gravity if box is tipped on its side)
+		// is used as the input to a Mass-Spring-Damper model
+		// Plectrum displacement (i.e. when interacting with string) is included
+		float massPosition = (float)msd.update(gravity - gPlectrumDisplacement);
+
+		float out_l = 0;
+		float out_r = 0;
+		// Use this parameter to quickly adjust output gain
+		float gain = 0.0015;	// 0.0015 is a good value or 12 strings
+		gPlectrumDisplacement = 0;
+
+		for(int s=0;s<NUMBER_OF_STRINGS;s++)	{
+
+			float stringPosition = strings[s].getGlobalPosition();
+
+			float plectrumForce = plectrums[s].update(massPosition, stringPosition);
+			gPlectrumDisplacement += strings[s].getPlectrumDisplacement();
+
+			// calculate panning based on string position (-1->left / 1->right)
+			float panRight = map(stringPosition,1,-1,0.1,1);
+			float panLeft = map(stringPosition,-1,1,0.1,1);
+			panRight *= panRight;
+			panLeft *= panLeft;
+
+			float out = strings[s].update(plectrumForce)*gain;
+
+			out_l += out*panLeft;
+			out_r += out*panRight;
+
+		}
+
+		// APPLY DC-BLOCK FILTER TO OUTPUTS
+
+		// LEFT CHANNEL
+		float temp_in = out_l;
+		/* compute result */
+    	out_l = a0_l * out_l + a1_l * x1_l + a2_l * x2_l - a3_l * y1_l - a4_l * y2_l;
+    	/* shift x1 to x2, sample to x1 */
+    	x2_l = x1_l;
+    	x1_l = temp_in;
+    	/* shift y1 to y2, result to y1 */
+    	y2_l = y1_l;
+   	 	y1_l = out_l;
+
+   	 	// RIGHT CHANNEL
+		temp_in = out_r;
+		/* compute result */
+    	out_r = a0_r * out_r + a1_r * x1_r + a2_r * x2_r - a3_r * y1_r - a4_r * y2_r;
+    	/* shift x1 to x2, sample to x1 */
+    	x2_r = x1_r;
+    	x1_r = temp_in;
+    	/* shift y1 to y2, result to y1 */
+    	y2_r = y1_r;
+   	 	y1_r = out_r;
+
+		context->audioOut[n * context->audioChannels + 1] = out_l * out_gain;
+		context->audioOut[n * context->audioChannels + 0] = out_r * out_gain;
+
+	}
+
+}
+
+
+// cleanup_render() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in initialise_render().
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/ADSR.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,76 @@
+//
+//  ADSR.cpp
+//
+//  Created by Nigel Redmon on 12/18/12.
+//  EarLevel Engineering: earlevel.com
+//  Copyright 2012 Nigel Redmon
+//
+//  For a complete explanation of the ADSR envelope generator and code,
+//  read the series of articles by the author, starting here:
+//  http://www.earlevel.com/main/2013/06/01/envelope-generators/
+//
+//  License:
+//
+//  This source code is provided as is, without warranty.
+//  You may copy and distribute verbatim copies of this document.
+//  You may modify and use this source code to create binary code for your own purposes, free or commercial.
+//
+
+#include "ADSR.h"
+#include <math.h>
+
+
+ADSR::ADSR(void) {
+    reset();
+    setAttackRate(0);
+    setDecayRate(0);
+    setReleaseRate(0);
+    setSustainLevel(1.0);
+    setTargetRatioA(0.3);
+    setTargetRatioDR(0.0001);
+}
+
+ADSR::~ADSR(void) {
+}
+
+void ADSR::setAttackRate(float rate) {
+    attackRate = rate;
+    attackCoef = calcCoef(rate, targetRatioA);
+    attackBase = (1.0 + targetRatioA) * (1.0 - attackCoef);
+}
+
+void ADSR::setDecayRate(float rate) {
+    decayRate = rate;
+    decayCoef = calcCoef(rate, targetRatioDR);
+    decayBase = (sustainLevel - targetRatioDR) * (1.0 - decayCoef);
+}
+
+void ADSR::setReleaseRate(float rate) {
+    releaseRate = rate;
+    releaseCoef = calcCoef(rate, targetRatioDR);
+    releaseBase = -targetRatioDR * (1.0 - releaseCoef);
+}
+
+float ADSR::calcCoef(float rate, float targetRatio) {
+    return exp(-log((1.0 + targetRatio) / targetRatio) / rate);
+}
+
+void ADSR::setSustainLevel(float level) {
+    sustainLevel = level;
+    decayBase = (sustainLevel - targetRatioDR) * (1.0 - decayCoef);
+}
+
+void ADSR::setTargetRatioA(float targetRatio) {
+    if (targetRatio < 0.000000001)
+        targetRatio = 0.000000001;  // -180 dB
+    targetRatioA = targetRatio;
+    attackBase = (1.0 + targetRatioA) * (1.0 - attackCoef);
+}
+
+void ADSR::setTargetRatioDR(float targetRatio) {
+    if (targetRatio < 0.000000001)
+        targetRatio = 0.000000001;  // -180 dB
+    targetRatioDR = targetRatio;
+    decayBase = (sustainLevel - targetRatioDR) * (1.0 - decayCoef);
+    releaseBase = -targetRatioDR * (1.0 - releaseCoef);
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/ADSR.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,137 @@
+//
+//  ADRS.h
+//
+//  Created by Nigel Redmon on 12/18/12.
+//  EarLevel Engineering: earlevel.com
+//  Copyright 2012 Nigel Redmon
+//
+//  For a complete explanation of the ADSR envelope generator and code,
+//  read the series of articles by the author, starting here:
+//  http://www.earlevel.com/main/2013/06/01/envelope-generators/
+//
+//  License:
+//
+//  This source code is provided as is, without warranty.
+//  You may copy and distribute verbatim copies of this document.
+//  You may modify and use this source code to create binary code for your own purposes, free or commercial.
+//
+
+#ifndef ADRS_h
+#define ADRS_h
+
+#include <stdio.h>
+#include <string>
+
+using namespace std;
+
+enum envState {
+	env_idle = 0,
+	env_attack,
+	env_decay,
+	env_sustain,
+	env_release
+};
+
+class ADSR {
+public:
+	ADSR(void);
+	~ADSR(void);
+	float process(void);
+	float process(int sampleCount);
+    float getOutput(void);
+    int getState(void);
+	void gate(int on);
+    void setAttackRate(float rate);
+    void setDecayRate(float rate);
+    void setReleaseRate(float rate);
+	void setSustainLevel(float level);
+    void setTargetRatioA(float targetRatio);
+    void setTargetRatioDR(float targetRatio);
+    void reset(void);
+
+protected:
+	int state;
+	float output;
+	float attackRate;
+	float decayRate;
+	float releaseRate;
+	float attackCoef;
+	float decayCoef;
+	float releaseCoef;
+	float sustainLevel;
+    float targetRatioA;
+    float targetRatioDR;
+    float attackBase;
+    float decayBase;
+    float releaseBase;
+	string name;
+    float calcCoef(float rate, float targetRatio);
+};
+
+inline float ADSR::process() {
+	switch (state) {
+        case env_idle:
+            break;
+        case env_attack:
+            output = attackBase + output * attackCoef;
+            if (output >= 1.0) {
+                output = 1.0;
+                state = env_decay;
+            }
+            break;
+        case env_decay:
+            output = decayBase + output * decayCoef;
+            if (output <= sustainLevel) {
+                output = sustainLevel;
+                state = env_sustain;
+            }
+            break;
+        case env_sustain:
+            break;
+        case env_release:
+            output = releaseBase + output * releaseCoef;
+            if (output <= 0.0) {
+                output = 0.0;
+                state = env_idle;
+            }
+            break;
+	}
+	return output;
+}
+
+inline float ADSR::process(int sampleCount)
+{
+	float retVal = 0;
+
+	if(state != env_idle)
+	{
+		for(int i=0; i<sampleCount; i++)
+			retVal = process();
+	}
+
+	return retVal;
+}
+
+inline void ADSR::gate(int gate) {
+
+	if (gate)
+		state = env_attack;
+	else if (state != env_idle)
+        state = env_release;
+}
+
+inline int ADSR::getState() {
+    return state;
+}
+
+inline void ADSR::reset() {
+    state = env_idle;
+    output = 0.0;
+}
+
+inline float ADSR::getOutput() {
+	return output;
+}
+
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/AnalogInput.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,163 @@
+/*
+ * AnalogInput.cpp
+ *
+ *  Created on: Oct 17, 2013
+ *      Author: Victor Zappi
+ */
+
+
+#include "AnalogInput.h"
+
+using namespace std;
+
+
+AnalogInput::AnalogInput()
+{
+	ActivateAnalogHnd  = NULL;
+	activateAnalogPath = "";
+	analogIsSet        = false;
+
+	AnalogInHnd    = NULL;
+	analogInPath   = "";
+	helperNumFound = false;
+
+	// support var for init
+	// these are fixed for BBB
+	startPath = "/sys/devices/bone_capemgr.*/slots";
+	readPath = "";
+
+	buffer = (char*) malloc (sizeof(char)*lSize); // reading buffer
+
+	verbose = false;
+}
+
+
+AnalogInput::~AnalogInput()
+{
+	free(buffer);
+}
+
+int AnalogInput::initAnalogInputs()
+{
+	 if(analogIsSet)
+	 {
+		 if(verbose)
+			 cout << "Fine, but Analog Input already active..."<< endl;
+		 return 0;
+	 }
+
+	// title
+	 if(verbose)
+	 	cout << "Init Analog Input"<< endl;
+
+
+	  // first: activate analog pins on cape manager
+	  // cape-bone-iio > /sys/devices/bone_capemgr.*/slots
+
+	  // we have to look for the semi-random number the BBB has initialized the bone_capemgr with [value of *]
+	  // to reach /slots and set cape-bone-iio.
+	  // to do so, we use glob lib, which translates wildcards [*] into all the values found in paths
+
+
+	  glob( startPath.c_str(), 0, NULL, &globbuf);
+
+	  if(globbuf.gl_pathc >0)
+	  {
+	    if (globbuf.gl_pathc == 1 )
+	    {
+	      activateAnalogPath = globbuf.gl_pathv[0];
+
+	      // check if file is existing
+	      if((ActivateAnalogHnd = fopen(activateAnalogPath.c_str(), "r+")) != NULL)
+	      {
+	        // we found that current capemgr num
+
+			fwrite("cape-bone-iio", sizeof(char), 13, ActivateAnalogHnd); // activate pins
+
+	        analogIsSet = true;
+
+			if(verbose)
+				cout << "Analog Pins activated via cape-bone-iio at path " << activateAnalogPath << endl;
+
+			fclose(ActivateAnalogHnd); // close file
+	      }
+	    }
+	    //else
+	      //printf("toomany", );
+	  }
+
+	  globfree(&globbuf); // self freeing
+
+
+	  if(!analogIsSet)
+	  {
+	    cout << "cannot find bone_capemgr" << endl;
+	    cout << "------Init failed------" << endl;
+	    return 1;
+	  }
+
+
+	  // build read path
+	  startPath = "/sys/devices/ocp.2/helper.*";
+
+	  glob( startPath.c_str(), 0, NULL, &globbuf);
+
+	  if(globbuf.gl_pathc >0)
+	  {
+		if (globbuf.gl_pathc == 1 )
+		{
+		  analogInPath = globbuf.gl_pathv[0] + (string)"/AIN";
+		}
+	    else
+	      cout << "Too many analog inputs with this name! [I am puzzled...]" << endl;
+	  }
+	  else
+	    cout << "Cannot find analog input dir...puzzled" << endl;
+
+
+	  return 0;
+}
+
+
+int AnalogInput::read(int index)
+{
+	  // convert int index into string
+	  stringstream ss;
+	  ss << index;
+
+	  readPath = analogInPath + ss.str(); // create pin0 file path
+
+
+	  // check if file is existing
+	  if((AnalogInHnd = fopen(readPath.c_str(), "rb")) != NULL)
+	  {
+		// we found that current helper num
+
+		// prepare read buffer to reading
+		fseek (AnalogInHnd , 0 , SEEK_END);
+		lSize = ftell (AnalogInHnd);
+		rewind (AnalogInHnd);
+
+		result = fread (buffer, 1, lSize, AnalogInHnd);
+
+		fclose(AnalogInHnd); // close file
+
+		helperNumFound = true;
+
+		//cout << "Analog Pins can be read at path " << analogInPath << endl;
+		//cout << "Test reading of Pin0 gives: " << buffer << endl;
+	  }
+
+	  if(!helperNumFound)
+	  {
+	    cout << "cannot find helper" << endl;
+	    cout << "------Analog Read failed------" << endl;
+	    return -1;
+	  }
+
+	  return atoi(buffer);
+
+}
+
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/AnalogInput.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,55 @@
+/*
+ * AnalogInput.h
+ *
+ *  Created on: Oct 17, 2013
+ *      Author: Victor Zappi
+ */
+
+#ifndef ANALOGINPUT_H_
+#define ANALOGINPUT_H_
+
+#include <iostream>
+#include <sstream>
+#include <stdio.h>
+#include <stdlib.h>
+#include <glob.h>
+
+using namespace std;
+
+class AnalogInput
+{
+private:
+	FILE *ActivateAnalogHnd;
+	string activateAnalogPath;
+	bool analogIsSet;
+
+	FILE *AnalogInHnd;
+	string analogInPath;
+	bool helperNumFound;
+
+	// suport var for init
+	string startPath;
+	string readPath;
+
+	glob_t  globbuf;
+
+	// support vars for pin reading
+	long lSize;
+	char * buffer;
+	size_t result;
+
+	bool verbose;
+
+public:
+	AnalogInput();
+	~AnalogInput();
+
+	int initAnalogInputs();
+	int read(int index);
+
+};
+
+
+
+
+#endif /* ANALOGINPUT_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/Biquad.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,169 @@
+//
+//  Biquad.cpp
+//
+//  Created by Nigel Redmon on 11/24/12
+//  EarLevel Engineering: earlevel.com
+//  Copyright 2012 Nigel Redmon
+//
+//  For a complete explanation of the Biquad code:
+//  http://www.earlevel.com/main/2012/11/26/biquad-c-source-code/
+//
+//  License:
+//
+//  This source code is provided as is, without warranty.
+//  You may copy and distribute verbatim copies of this document.
+//  You may modify and use this source code to create binary code
+//  for your own purposes, free or commercial.
+//
+
+#include <math.h>
+#include "Biquad.h"
+#include <iostream>
+
+Biquad::Biquad() {
+    type = bq_type_lowpass;
+    a0 = 1.0;
+    a1 = a2 = b1 = b2 = 0.0;
+    Fc = 0.50;
+    Q = 0.707;
+    peakGain = 0.0;
+    z1 = z2 = 0.0;
+}
+
+Biquad::Biquad(int type, double Fc, double Q, double peakGainDB) {
+    setBiquad(type, Fc, Q, peakGainDB);
+    z1 = z2 = 0.0;
+}
+
+Biquad::~Biquad() {
+}
+
+void Biquad::setType(int type) {
+    this->type = type;
+    calcBiquad();
+}
+
+void Biquad::setQ(double Q) {
+    this->Q = Q;
+    calcBiquad();
+}
+
+void Biquad::setFc(double Fc) {
+    this->Fc = Fc;
+    calcBiquad();
+}
+
+void Biquad::setPeakGain(double peakGainDB) {
+    this->peakGain = peakGainDB;
+    calcBiquad();
+}
+    
+void Biquad::setBiquad(int type, double Fc, double Q, double peakGainDB) {
+    this->type = type;
+    this->Q = Q;
+    this->Fc = Fc;
+    startFc = Fc;
+    startQ = Q;
+    startPeakGain = peakGainDB;
+    setPeakGain(peakGainDB);
+}
+
+void Biquad::calcBiquad(void) {
+    double norm;
+    double V = pow(10, fabs(peakGain) / 20.0);
+    double K = tan(M_PI * Fc);
+    switch (this->type) {
+        case bq_type_lowpass:
+            norm = 1 / (1 + K / Q + K * K);
+            a0 = K * K * norm;
+            a1 = 2 * a0;
+            a2 = a0;
+            b1 = 2 * (K * K - 1) * norm;
+            b2 = (1 - K / Q + K * K) * norm;
+            break;
+            
+        case bq_type_highpass:
+            norm = 1 / (1 + K / Q + K * K);
+            a0 = 1 * norm;
+            a1 = -2 * a0;
+            a2 = a0;
+            b1 = 2 * (K * K - 1) * norm;
+            b2 = (1 - K / Q + K * K) * norm;
+            break;
+            
+        case bq_type_bandpass:
+            norm = 1 / (1 + K / Q + K * K);
+            a0 = K / Q * norm;
+            a1 = 0;
+            a2 = -a0;
+            b1 = 2 * (K * K - 1) * norm;
+            b2 = (1 - K / Q + K * K) * norm;
+            break;
+            
+        case bq_type_notch:
+            norm = 1 / (1 + K / Q + K * K);
+            a0 = (1 + K * K) * norm;
+            a1 = 2 * (K * K - 1) * norm;
+            a2 = a0;
+            b1 = a1;
+            b2 = (1 - K / Q + K * K) * norm;
+            break;
+            
+        case bq_type_peak:
+            if (peakGain >= 0) {    // boost
+                norm = 1 / (1 + 1/Q * K + K * K);
+                a0 = (1 + V/Q * K + K * K) * norm;
+                a1 = 2 * (K * K - 1) * norm;
+                a2 = (1 - V/Q * K + K * K) * norm;
+                b1 = a1;
+                b2 = (1 - 1/Q * K + K * K) * norm;
+            }
+            else {    // cut
+                norm = 1 / (1 + V/Q * K + K * K);
+                a0 = (1 + 1/Q * K + K * K) * norm;
+                a1 = 2 * (K * K - 1) * norm;
+                a2 = (1 - 1/Q * K + K * K) * norm;
+                b1 = a1;
+                b2 = (1 - V/Q * K + K * K) * norm;
+            }
+            break;
+        case bq_type_lowshelf:
+            if (peakGain >= 0) {    // boost
+                norm = 1 / (1 + sqrt(2) * K + K * K);
+                a0 = (1 + sqrt(2*V) * K + V * K * K) * norm;
+                a1 = 2 * (V * K * K - 1) * norm;
+                a2 = (1 - sqrt(2*V) * K + V * K * K) * norm;
+                b1 = 2 * (K * K - 1) * norm;
+                b2 = (1 - sqrt(2) * K + K * K) * norm;
+            }
+            else {    // cut
+                norm = 1 / (1 + sqrt(2*V) * K + V * K * K);
+                a0 = (1 + sqrt(2) * K + K * K) * norm;
+                a1 = 2 * (K * K - 1) * norm;
+                a2 = (1 - sqrt(2) * K + K * K) * norm;
+                b1 = 2 * (V * K * K - 1) * norm;
+                b2 = (1 - sqrt(2*V) * K + V * K * K) * norm;
+            }
+            break;
+        case bq_type_highshelf:
+            if (peakGain >= 0) {    // boost
+                norm = 1 / (1 + sqrt(2) * K + K * K);
+                a0 = (V + sqrt(2*V) * K + K * K) * norm;
+                a1 = 2 * (K * K - V) * norm;
+                a2 = (V - sqrt(2*V) * K + K * K) * norm;
+                b1 = 2 * (K * K - 1) * norm;
+                b2 = (1 - sqrt(2) * K + K * K) * norm;
+            }
+            else {    // cut
+                norm = 1 / (V + sqrt(2*V) * K + K * K);
+                a0 = (1 + sqrt(2) * K + K * K) * norm;
+                a1 = 2 * (K * K - 1) * norm;
+                a2 = (1 - sqrt(2) * K + K * K) * norm;
+                b1 = 2 * (K * K - V) * norm;
+                b2 = (V - sqrt(2*V) * K + K * K) * norm;
+            }
+            break;
+    }
+    
+    return;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/Biquad.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,99 @@
+//
+//  Biquad.h
+//
+//  Created by Nigel Redmon on 11/24/12
+//  EarLevel Engineering: earlevel.com
+//  Copyright 2012 Nigel Redmon
+//
+//  For a complete explanation of the Biquad code:
+//  http://www.earlevel.com/main/2012/11/25/biquad-c-source-code/
+//
+//  License:
+//
+//  This source code is provided as is, without warranty.
+//  You may copy and distribute verbatim copies of this document.
+//  You may modify and use this source code to create binary code
+//  for your own purposes, free or commercial.
+//
+
+#ifndef Biquad_h
+#define Biquad_h
+
+enum {
+    bq_type_lowpass = 0,
+    bq_type_highpass,
+    bq_type_bandpass,
+    bq_type_notch,
+    bq_type_peak,
+    bq_type_lowshelf,
+    bq_type_highshelf
+};
+
+class Biquad {
+public:
+    Biquad();
+    Biquad(int type, double Fc, double Q, double peakGainDB);
+    ~Biquad();
+    void setType(int type);
+    void setQ(double Q);
+    void setFc(double Fc);
+    void setPeakGain(double peakGainDB);
+    void setBiquad(int type, double Fc, double Q, double peakGain);
+    float process(float in);
+    
+    double getQ();
+    double getFc();
+    double getPeakGain();
+
+    double getStartingQ();
+	double getStartingFc();
+	double getStartingPeakGain();
+
+protected:
+    void calcBiquad(void);
+
+    int type;
+    double a0, a1, a2, b1, b2;
+    double Fc, Q, peakGain;
+    double startFc, startQ, startPeakGain;
+    double z1, z2;
+};
+
+inline double Biquad::getQ()
+{
+	return Q;
+}
+
+inline double Biquad::getFc()
+{
+	return Fc;
+}
+
+inline double Biquad::getPeakGain()
+{
+	return peakGain;
+}
+
+inline double Biquad::getStartingQ()
+{
+	return startQ;
+}
+
+inline double Biquad::getStartingFc()
+{
+	return startFc;
+}
+
+inline double Biquad::getStartingPeakGain()
+{
+	return startPeakGain;
+}
+
+inline float Biquad::process(float in) {
+    double out = in * a0 + z1;
+    z1 = in * a1 + z2 - b1 * out;
+    z2 = in * a2 - b2 * out;
+    return out;
+}
+
+#endif // Biquad_h
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/DBoxSynth.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,35 @@
+/*
+ * SimpleSynth.h
+ *
+ *  Created on: Oct 22, 2013
+ *      Author: Victor Zappi
+ */
+
+#ifndef DBOXSYNTH_H_
+#define DBOXSYNTH_H_
+
+#include <iostream>
+#include <string>
+#include <stdio.h>
+#include <stdlib.h>
+
+#include "Synth.h"
+
+
+class DBoxSynth : public Synth
+{
+public:
+	DBoxSynth(unsigned int rate, unsigned long buffer_size);
+	double getSample();
+	double *getBlock(int block_size);
+
+
+private:
+	Sampler *smp;
+
+};
+
+
+
+
+#endif /* DBOXSYNTH_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/DboxSensors.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,157 @@
+/*
+ * DboxSensors.cpp
+ *
+ *  Created on: May 19, 2014
+ *      Author: Victor Zappi
+ */
+
+
+#include "DboxSensors.h"
+#include "config.h"
+
+using namespace std;
+
+
+
+int DboxSensors::initSensors(int tk0_bus, int tk0_address, int tk1_bus, int tk1_address, int tk_file, int fsr_pin, int fsrmax, int sensorTypeToUse, int gpio_0, int gpio_1)
+{
+	sensorType = sensorTypeToUse;
+	// init first touch key on i2c bus
+	if(tk0_address >= 0) {
+		if(TK0.initI2C_RW(tk0_bus, tk0_address, tk_file)>0)
+			return 1;
+		if(TK0.initTouchKey(sensorType)>0)
+			return 2;
+	}
+
+	// init second touch key on i2c bus
+	if(tk1_address >= 0) {
+		if(TK1.initI2C_RW(tk1_bus, tk1_address, tk_file)>0)
+			return 1;
+		if(TK1.initTouchKey(sensorType)>0)
+			return 2;
+	}
+
+	// init fsr on analog input pin
+	fsr_pinNum	= fsr_pin;
+	fsr_max	  	= fsrmax;
+
+	if(FSR.initAnalogInputs()>0)
+		return 3;
+
+	gpio[0] = gpio_0;
+	if(gpio[0]!=-1)
+	{
+		fdDi[0] = gpio_export(gpio[0]);
+		if(fdDi[0] == -1)
+			return 4;
+	}
+	digitalIn[0] = 1;
+
+	return 0;
+}
+
+
+int DboxSensors::readSensors()
+{
+	// write data into first touch key
+	if(TK0.ready()) {
+		if(TK0.readI2C()>0)
+			return 1;
+
+		// retrieve data from first touch key
+		tk0_touchCnt = TK0.getTouchCount();
+	}
+	else
+		tk0_touchCnt = 0;
+
+	// write data into second touch key
+	if(TK1.ready()) {
+		if(TK1.readI2C()>0)
+			return 1;
+		// retrieve data from second touch key
+		tk1_touchCnt = TK1.getTouchCount();
+	}
+	else
+		tk1_touchCnt = 0;
+
+
+	int max = 3;
+	if(sensorType != kSensorTypeTouchKey)
+		max = 5;
+	// if touches detected on main touch key
+	if(tk0_touchCnt == 0 && tk1_touchCnt == 0)
+		resetSensorsData();
+	else
+	{
+		for(int i=0; i<max; i++)
+		{
+			tk0_touchPosX[i] = TK0.getSliderPosition()[i];
+			tk0_touchSize[i] = TK0.getSlidersize()[i];
+
+			tk1_touchPosX[i] = TK1.getSliderPosition()[i];
+			tk1_touchSize[i] = TK1.getSlidersize()[i];
+		}
+		tk0_touchPosY 	 = TK0.getSliderPositionH();
+		tk1_touchPosY 	 = TK1.getSliderPositionH();
+		fsr_read		 = (double)FSR.read(fsr_pinNum);
+	}
+
+	if(gpio[0]!=-1)
+	{
+		if(gpio_read(fdDi[0], &digitalIn[0])==-1)
+			return 1;
+	}
+
+	return 0;
+}
+
+
+
+DboxSensors::DboxSensors()
+{
+	resetSensorsData();
+}
+
+
+
+DboxSensors::~DboxSensors()
+{
+	if(gpio[0]!=-1)
+		gpio_dismiss(fdDi[0], gpio[0]);
+}
+
+
+
+//--------------------------------------------------------------------------------------------------------
+// private methods
+//--------------------------------------------------------------------------------------------------------
+
+// idle values
+void DboxSensors::resetSensorsData()
+{
+	int max = 3;
+	if(sensorType != kSensorTypeTouchKey)
+		max = 5;
+
+	for(int i=0; i<max; i++)
+	{
+		tk0_touchPosX[i] = -1;
+		tk0_touchPosY	 = -1;
+		tk0_touchSize[i] = 0;
+
+		tk1_touchPosX[i] = -1;
+		tk1_touchPosY	 = -1;
+		tk1_touchSize[i] = 0;
+
+		fsr_read		 = 0;
+	}
+
+	return;
+}
+
+
+
+
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/DboxSensors.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,128 @@
+/*
+ * DboxSensors.h
+ *
+ *  Created on: May 19, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef DBOXSENSORS_H_
+#define DBOXSENSORS_H_
+
+#include <stdio.h>
+#include <sys/mount.h>	// mount()
+#include <string.h> 	// strerror()
+#include <fstream> 		// fstream
+#include <iostream>
+#include <unistd.h> 	// usleep()
+#include <glob.h>		// glob()
+#include <sys/time.h>	// elapsed time
+#include <sys/stat.h>	// mkdir()
+#include <algorithm>	// reverse() [string...]
+
+#include "I2c_TouchKey.h"
+#include "AnalogInput.h"
+#include <GPIOcontrol.h>	// TODO wrap this into a class
+
+/*---------------------------------------------------------------------------------------------------------------------------------------------------
+ * This class retrieves data from all the connected sensors,
+ * logs them
+ * and exposes to the main only the values needed to synthesize sound
+ *
+ * The simple instrument has:
+ *
+ *
+ *
+ *---------------------------------------------------------------------------------------------------------------------------------------------------
+ */
+class DboxSensors
+{
+public:
+	int initSensors(int tk0_bus, int tk0_address, int tk1_bus, int tk1_address, int tk_file, int fsr_pin, int fsrmax, int sensorTypeToUse, int gpio0=-1, int gpio1=-1);
+	int readSensors();
+	int getTKTouchCount(int index);
+	float *getTKXPositions(int index);
+	float getTKYPosition(int index);
+	float *getTKTouchSize(int index);
+	double getFSRVAlue();
+	int getDigitalIn(int index);
+
+	DboxSensors();
+	~DboxSensors();
+
+private:
+	int sensorType;
+
+	I2c_TouchKey TK0;
+	int tk0_touchCnt;
+	float tk0_touchPosX[5];
+	float tk0_touchPosY;
+	float tk0_touchSize[5];
+
+	I2c_TouchKey TK1;
+	int tk1_touchCnt;
+	float tk1_touchPosX[5];
+	float tk1_touchPosY;
+	float tk1_touchSize[5];
+
+	AnalogInput FSR;
+	int fsr_pinNum;
+	double fsr_read;
+	int fsr_max;
+
+	unsigned int digitalIn[2];
+	int fdDi[2];
+	int gpio[2];
+
+	void resetSensorsData();
+
+};
+
+
+
+//--------------------------------------------------------------------------------
+// read interface
+inline int DboxSensors::getTKTouchCount(int index)
+{
+	if(index==0)
+		return tk0_touchCnt;
+	else
+		return tk1_touchCnt;
+}
+
+inline float *DboxSensors::getTKXPositions(int index)
+{
+	if(index==0)
+		return tk0_touchPosX;
+	else
+		return tk1_touchPosX;
+}
+
+inline float DboxSensors::getTKYPosition(int index)
+{
+	if(index==0)
+		return tk0_touchPosY;
+	else
+		return tk1_touchPosY;
+}
+
+inline float *DboxSensors::getTKTouchSize(int index)
+{
+	if(index==0)
+		return tk0_touchSize;
+	else
+		return tk1_touchSize;
+}
+
+inline double DboxSensors::getFSRVAlue()
+{
+	return fsr_read;
+}
+
+inline int DboxSensors::getDigitalIn(int index)
+{
+	return digitalIn[index];
+}
+//--------------------------------------------------------------------------------
+
+
+#endif /* DBOXSENSORS_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/FIRfilter.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,73 @@
+/*
+ * FIRfilter.h
+ *
+ *  Created on: Aug 5, 2014
+ *      Author: Victor Zappi and Andrew McPherson
+ */
+
+#ifndef FIRFILTER_H_
+#define FIRFILTER_H_
+
+#define ENABLE_NE10_FIR_FLOAT_NEON	// Define needed for Ne10 library
+#include <ne10/NE10.h>
+
+//#define FILTER_TAP_NUM 21
+//ne10_float32_t filterTaps[FILTER_TAP_NUM] = {
+//	0.000350,
+//	0.001133,
+//	0.002407,
+//	0.004203,
+//	0.006468,
+//	0.009057,
+//	0.011748,
+//	0.014265,
+//	0.016323,
+//	0.017671,
+//	0.018141,
+//	0.017671,
+//	0.016323,
+//	0.014265,
+//	0.011748,
+//	0.009057,
+//	0.006468,
+//	0.004203,
+//	0.002407,
+//	0.001133,
+//	0.000350
+//};
+#define FILTER_TAP_NUM 31
+ne10_float32_t filterTaps[FILTER_TAP_NUM] = {
+		0.000018,
+		0.000043,
+		0.000078,
+		0.000125,
+		0.000183,
+		0.000252,
+		0.000330,
+		0.000415,
+		0.000504,
+		0.000592,
+		0.000677,
+		0.000754,
+		0.000818,
+		0.000866,
+		0.000897,
+		0.000907,
+		0.000897,
+		0.000866,
+		0.000818,
+		0.000754,
+		0.000677,
+		0.000592,
+		0.000504,
+		0.000415,
+		0.000330,
+		0.000252,
+		0.000183,
+		0.000125,
+		0.000078,
+		0.000043,
+		0.000018
+};
+
+#endif /* FIRFILTER_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/FeedbackOscillator.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,105 @@
+/*
+ * FeedbackOscillator.cpp
+ *
+ * Recursive phase-shift oscillator implemented
+ * on the matrix
+ *
+ * Andrew McPherson 2014
+ */
+
+#include "FeedbackOscillator.h"
+#include <cstdlib>
+#include <cmath>
+
+#define COEFF_B0 	0
+#define COEFF_B1	1
+#define COEFF_A1	2
+
+FeedbackOscillator::FeedbackOscillator()
+: wavetable1(0), wavetable2(0)
+{
+
+}
+
+FeedbackOscillator::~FeedbackOscillator() {
+	if(wavetable1 != 0)
+		free(wavetable1);
+	if(wavetable2 != 0)
+		free(wavetable2);
+
+}
+
+// Initialise the settings for the feedback oscillator
+void FeedbackOscillator::initialise(int maxTableSize, float hpfCutoffFrequency, float matrixSampleRate) {
+	wavetableMaxLength = maxTableSize;
+	if(wavetable1 != 0)
+		free(wavetable1);
+	if(wavetable2 != 0)
+		free(wavetable2);
+
+	wavetable1 = (float *)malloc(maxTableSize * sizeof(float));
+	wavetable2 = (float *)malloc(maxTableSize * sizeof(float));
+
+	float omega = tan(M_PI * hpfCutoffFrequency / matrixSampleRate);
+	float n = 1.0f / (1.0f + omega);
+
+	coeffs[COEFF_A1] = (omega - 1.0f) * n;
+	coeffs[COEFF_B0] = n;
+	coeffs[COEFF_B1] = -n;
+
+	for(int n = 0; n < maxTableSize; n++)
+		wavetable1[n] = wavetable2[n] = 0;
+
+	wavetableRead = wavetable1;
+	wavetableWrite = wavetable2;
+	wavetableWritePointer = 0;
+	sampleCount = lastTriggerCount = 0;
+}
+
+// Process one sample and store the output value
+// Returns true if the wavetable needs rendering
+int FeedbackOscillator::process(float input, float *output) {
+	float outFloat = coeffs[COEFF_B0] * input + coeffs[COEFF_B1] * lastInput - coeffs[COEFF_A1] * lastOutput;
+	int requestRenderLength = 0;
+
+	if(outFloat < -0.5)
+		*output = 0;
+	else if(outFloat > 0.5)
+		*output = 1;
+	else
+		*output = outFloat + 0.5;
+
+	if(canTrigger && outFloat > 0 && lastOutput <= 0) {
+		triggered = true;
+		requestRenderLength = wavetableWritePointer;	// How many samples stored thus far?
+		if(requestRenderLength < 4)
+			requestRenderLength = 0;	// Ignore anything with fewer than 4 points
+
+		lastTriggerCount = sampleCount;
+		canTrigger = false;
+		wavetableWritePointer = 0;
+
+		// Swap buffers
+		float *temp = wavetableWrite;
+		wavetableWrite = wavetableRead;
+		wavetableRead = temp;
+	}
+
+	if(triggered) {
+		wavetableWrite[wavetableWritePointer] = outFloat;
+		if(++wavetableWritePointer >= wavetableMaxLength) {
+			triggered = false;
+			wavetableWritePointer = 0;
+		}
+	}
+
+	if(sampleCount - lastTriggerCount > 40)
+		canTrigger = true;
+
+	sampleCount++;
+
+	lastOutput = outFloat;
+	lastInput = input;
+
+	return requestRenderLength;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/FeedbackOscillator.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,41 @@
+/*
+ * FeedbackOscillator.h
+ *
+ *  Created on: June 8, 2014
+ *      Author: Andrew McPherson
+ */
+
+#ifndef FEEDBACKOSCILLATOR_H
+#define FEEDBACKOSCILLATOR_H
+
+class FeedbackOscillator
+{
+public:
+	FeedbackOscillator();
+	~FeedbackOscillator();
+
+	// Initialise the settings for the feedback oscillator
+	void initialise(int maxTableSize, float hpfCutoffFrequency, float matrixSampleRate);
+
+	// Process one sample and store the output value
+	// Returns the length of table to interpolate; or 0 if nothing to process further
+	int process(float input, float *output);
+
+	float *wavetable() { return wavetableRead; }
+
+private:
+	float coeffs[3];			// Coefficients of first-order high-pass filter
+	float lastInput;			// last input sample for HPF
+	float lastOutput;			// last output sample of HPF
+	bool triggered;				// whether we are currently saving samples
+	bool canTrigger;			// whether we are able to begin saving samples
+	int wavetableMaxLength;		// how long the stored wavetable can be
+	int sampleCount;			// how many samples have elapsed
+	int lastTriggerCount;		// sample count when we last triggered
+
+	float *wavetable1, *wavetable2;			// Two wavetables where we record samples
+	float *wavetableRead, *wavetableWrite;	// Pointers to the above wavetables
+	int wavetableWritePointer;				// Where we are currently writing
+};
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/OscillatorBank.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,1016 @@
+/*
+ * OscillatorBank.cpp
+ *
+ *  Created on: May 23, 2014
+ *      Author: Victor Zappi and Andrew McPherson
+ */
+
+
+/*
+ * There is a problem with name consistency between this class and the Parser class in spear_parser files.
+ * There, a "frame" is each of the time values where partials are sampled, while a "hop" is the actual jump between frames [calculated in samples]
+ * Here, "hop" is used with the meaning of "frame", while "frame" became the local frame of a partial
+ *
+ * example
+ *
+ * frames:	0		1		2
+ * p0				p0_0 	p0_1
+ * p1		p1_0	p1_1 	p1_2
+ * p2		p2_0	p2_1
+ *
+ * 	In this case:
+ * 		 in Parser there are 2 hops, 3 total frames and the 3 partials have respectively 2, 3 and 2 local frames
+ *
+ * 		 here there are 3 total hops [the concept of jumps is discarded, cos not in use] and the 3 partials have respectively 2, 3 and 2 frames
+ *
+ * 	This must be fixed
+*/
+
+// TODO: fix hop-frame name consistency
+
+
+#include <stdlib.h>
+
+#include "OscillatorBank.h"
+
+OscillatorBank::OscillatorBank() {
+	loaded = false;
+}
+
+OscillatorBank::OscillatorBank(string filename, int hopsize, int samplerate) {
+	loaded = false;
+	loadFile(filename.c_str(), hopsize, samplerate);
+}
+
+OscillatorBank::OscillatorBank(char *filename, int hopsize, int samplerate) {
+	loaded = false;
+	loadFile(filename, hopsize, samplerate);
+}
+
+OscillatorBank::~OscillatorBank() {
+	free(oscillatorPhases);
+	free(oscillatorNextNormFreq);
+	free(oscillatorNextAmp);
+	free(oscillatorNormFrequencies);
+	free(oscillatorAmplitudes);
+	free(oscillatorNormFreqDerivatives);
+	free(oscillatorAmplitudeDerivatives);
+	free(phaseCopies);
+	free(nextNormFreqCopies);
+	free(nextAmpCopies);
+
+	delete[] oscStatNormFrequenciesMean;
+	delete[] oscStatNumHops;
+	delete[] lookupTable;
+	delete[] indicesMapping;
+	delete[] freqFixedDeltas;
+	delete[] ampFixedDeltas;
+	delete[] nyquistCut;
+}
+
+bool OscillatorBank::initBank(int oversamp) {
+	if (!loaded)
+		return false;
+
+	//---prepare look-up table
+	lookupTableSize = 1024;
+	lookupTable		= new float[lookupTableSize + 1];
+	for (int n = 0; n < (lookupTableSize + 1); n++)
+		lookupTable[n] = sin(2.0 * M_PI * (float) n / (float) lookupTableSize);
+	frequencyScaler = (float) lookupTableSize / rate;
+	nyqNorm			= rate / 2 * frequencyScaler;
+
+	if (oversamp < 1)
+		oversamp = 1;
+
+	//---prepare oscillators
+	partials = &(parser.partials); // pointer to paser's partials
+	partialsHopSize = parser.getHopSize();
+	lastHop = partials->getHopNum(); // last bank hop is equal to last partial frame, which is equal to partial hop num
+	overSampling = oversamp;
+	hopSize = partialsHopSize / overSampling; // if oversampling, osc bank hop num > partials hop num
+	hopSizeReminder = partialsHopSize % overSampling;
+	oscBankHopSize = hopSize;
+	numOfPartials = partials->getPartialNum();
+	numOfOscillators = partials->getMaxActivePartialNum(); // get the maximum number of active partials at the same time
+
+	// set to next multiple of 4 [NEON]
+	numOfOscillators = (numOfOscillators + 3) & ~0x3; // to be sure we can add up to 3 fake oscillators
+
+	int err;
+	//---allocate buffers
+	// alligned buffers [NEON]
+	err = posix_memalign((void**) &oscillatorPhases, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &oscillatorNextNormFreq, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &oscillatorNextAmp, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &oscillatorNormFrequencies, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &oscillatorAmplitudes, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &oscillatorNormFreqDerivatives, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &oscillatorAmplitudeDerivatives, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &phaseCopies, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &nextNormFreqCopies, 16,
+			numOfOscillators * sizeof(float));
+	err += posix_memalign((void**) &nextAmpCopies, 16,
+			numOfOscillators * sizeof(float));
+
+	// regular ones
+	oscStatNormFrequenciesMean	= new float[numOfPartials];
+	oscStatNumHops				= new float[numOfPartials];
+	indicesMapping				= new int[numOfPartials];
+	freqFixedDeltas				= new float[numOfPartials];
+	ampFixedDeltas				= new float[numOfPartials];
+	nyquistCut 					= new bool[numOfPartials];
+
+	if (err > 0) {
+		dbox_printf("Failed memory allocations %@#!\n");
+		return false;
+	}
+
+	// copy stats [they do not change]
+	for (int n = 0; n < numOfPartials; n++) {
+		oscStatNormFrequenciesMean[n] = partials->partialFreqMean[n]
+				* frequencyScaler;
+		oscStatNumHops[n] = partials->partialNumFrames[n]; // in Parser and Partials "frames" are what we call here "hops" [see comment at top of file]
+	}
+
+	// deafult values
+	actPartNum		= 0;
+	loopStartHop	= 0;
+	loopEndHop		= (parser.partials.getHopNum() - 2) * overSampling;
+	ampTh			= 0.0001;
+	hopNumTh		= 0;
+	pitchMultiplier = 1;
+	freqMovement	= 1;
+	filterNum		= 0;
+	note			= false;
+	speed			= 1;
+	nextSpeed		= -1;
+	maxSpeed 		= 10;
+	minSpeed 		= 0.1;
+	jumpHop 		= -1;
+
+	// filter
+	filterMaxF		= 22000;
+	filterAmpMinF	= 10 * frequencyScaler;
+	filterAmpMaxF	= 5000 * frequencyScaler;
+	filterAmpMul	= 10.0;
+
+	// adsr
+	minAttackTime	= .0001;
+	deltaAttackTime = 2.;
+	minReleaseTime	= 1;
+	deltaReleaseTime = 2.5;
+
+	adsr.setAttackRate(minAttackTime * rate);
+	adsr.setDecayRate(.0001 * rate);
+	adsr.setSustainLevel(1);
+	adsr.setReleaseRate(minReleaseTime * rate);
+
+	state = bank_stopped;
+	return true;
+}
+
+void OscillatorBank::resetOscillators() {
+	currentHop			= -1;
+	loopDir				= 1;
+	loopDirShift 		= 0;
+	fill(nyquistCut, nyquistCut + numOfPartials, false);
+	prevAdsrVal			= 0;
+	prevAmpTh			= ampTh;
+	prevHopNumTh		= hopNumTh;
+	prevPitchMultiplier = pitchMultiplier;
+	prevFreqMovement 	= freqMovement;
+	prevFilterNum = filterNum;
+	memcpy(prevFilterFreqs, filterFreqs, filterNum * sizeof(float));
+	memcpy(prevFilterQ, filterQ, filterNum * sizeof(float));
+
+	int activePNum		= partials->activePartialNum[0];
+	unsigned int *activeP = partials->activePartials[0];
+	for (int i = 0; i < activePNum; i++) {
+		freqFixedDeltas[activeP[i]] = partials->partialFreqDelta[activeP[i]][0]
+				/ overSampling;
+		ampFixedDeltas[activeP[i]]	= partials->partialAmpDelta[activeP[i]][0]
+				/ overSampling;
+	}
+	// attack!
+	adsr.gate(1);
+	note = true;
+
+	nextHop();
+
+	state = bank_playing;
+}
+
+void OscillatorBank::nextHop() {
+	hopSize = oscBankHopSize;
+
+	// copy phases, next freqs and next amps from previous frame
+	memcpy(phaseCopies, oscillatorPhases, actPartNum * sizeof(float));
+	memcpy(nextNormFreqCopies, oscillatorNextNormFreq,
+			actPartNum * sizeof(float));
+	memcpy(nextAmpCopies, oscillatorNextAmp, actPartNum * sizeof(float));
+
+	// next frame is forward or backwards, cos we could be in the loop
+	currentHop += loopDir;
+
+	checkDirection();
+
+//	if((currentHop/overSampling)%100 == 0)
+//		dbox_printf("currentHop %d, direction: %d\n", currentHop/overSampling, loopDir);
+
+	// if needs jump, end here this method, cos jumpToHop() will do tee rest
+	if (checkJump() == 0)
+		return;
+	// otherwise, if jump is not needed or fails, continue regular stuff
+
+	if (nextEnvState() != 0)
+		return; // release has ended!
+
+	checkSpeed();
+
+	// now let's decide how to calculate next hop
+	if (!checkOversampling())
+		nextOscBankHop();
+	else
+		nextPartialHop();
+}
+
+void OscillatorBank::nextOscBankHop() {
+	int parIndex, localHop;
+	float parDamp = 1;
+	int currentPartialHop = (currentHop / overSampling) + loopDirShift;
+
+	// if going backwards in the loop, get previous frame active partials...
+	actPartNum = partials->activePartialNum[currentPartialHop - loopDirShift];
+	actPart = partials->activePartials[currentPartialHop - loopDirShift];
+	//cout << "actPartNum: " << actPartNum << endl;
+
+	envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release]
+
+	int parCnt = 0;
+	int currentHopReminder = currentHop % overSampling;
+	// steps to reach next bank hop from previous partial hop
+	int steps = currentHopReminder + 1;
+	if (loopDir < 0)
+		steps = overSampling - currentHopReminder + 1;
+
+	for (int i = 0; i < actPartNum; i++) {
+		// find partial and frame
+		parIndex = actPart[i];
+		//localHop	= partials->localPartialFrames[currentPartialHop][parIndex];
+		localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
+
+		//float delta = partials->partialFrequencies[parIndex][localHop+loopDir] - partials->partialFrequencies[parIndex][localHop];
+
+		// if this partial was over nyquist on previous hop...
+		if (nyquistCut[parIndex]) {
+			// ...restart from safe values
+			oscillatorPhases[parCnt] = 0;
+			//TODO add freqmove dependency
+			oscillatorNextNormFreq[parCnt] =
+					(partials->partialFrequencies[parIndex][localHop]
+							+ freqFixedDeltas[parIndex] * (steps - 1))
+							* frequencyScaler * prevPitchMultiplier;
+			oscillatorNextAmp[parCnt] = 0;
+		} else if (loopDir == 1) // otherwise recover phase, target freq and target amp from previous frame
+				{
+			if ((localHop != 0) || (currentHopReminder != 0)) {
+				oscillatorPhases[parCnt] =
+						phaseCopies[indicesMapping[parIndex]];
+				oscillatorNextNormFreq[parCnt] =
+						nextNormFreqCopies[indicesMapping[parIndex]];
+				oscillatorNextAmp[parCnt] =
+						nextAmpCopies[indicesMapping[parIndex]];
+			} else // first oscillator hop [both for bank and partial], so no previous data are available
+			{
+				oscillatorPhases[parCnt] = 0;
+				//TODO add freqmove dependency
+				oscillatorNextNormFreq[parCnt] =
+						partials->partialFrequencies[parIndex][localHop]
+								* frequencyScaler * prevPitchMultiplier;
+				parDamp = calculateParDamping(parIndex, prevHopNumTh,
+						prevAdsrVal, oscillatorNextNormFreq[parCnt],
+						prevFilterNum, prevFilterFreqs, prevFilterQ);
+				oscillatorNextAmp[parCnt] =
+						partials->partialAmplitudes[parIndex][localHop]
+								* parDamp;
+				if(oscillatorNextAmp[parCnt] > 1)
+						oscillatorNextAmp[parCnt] = 1;
+				freqFixedDeltas[parIndex] =
+						partials->partialFreqDelta[parIndex][localHop + loopDir]
+								* loopDir / overSampling;
+				ampFixedDeltas[parIndex] =
+						partials->partialAmpDelta[parIndex][localHop + loopDir]
+								* loopDir / overSampling;
+			}
+		} else {
+			oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]];
+			oscillatorNextNormFreq[parCnt] =
+					nextNormFreqCopies[indicesMapping[parIndex]];
+			oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]];
+		}
+
+		// remove aliasing, skipping partial over nyquist freq
+		if (oscillatorNextNormFreq[parCnt] > nyqNorm) {
+			nyquistCut[parIndex] = true;
+			continue;
+		}
+		nyquistCut[parIndex] = false;
+
+		// first set up freq, cos filter affects amplitude damping according to freq content
+		oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts
+		// save next values, current for next round
+		oscillatorNextNormFreq[parCnt] = (freqMovement
+				* (partials->partialFrequencies[parIndex][localHop]
+						+ freqFixedDeltas[parIndex] * steps) * frequencyScaler
+				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
+				* pitchMultiplier;
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
+				- oscillatorNormFrequencies[parCnt]) / hopCounter;
+		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
+		oscillatorNormFreqDerivatives[parCnt] = freqMovement
+				* oscillatorNormFreqDerivatives[parCnt]
+				+ (1 - freqMovement)
+						* ((oscStatNormFrequenciesMean[parIndex]
+								* pitchMultiplier)
+								- oscillatorNormFrequencies[parCnt])
+						/ hopCounter;
+
+		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
+				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
+
+		// now amplitudes
+		oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts
+		// save next values, current for next round
+		//delta = partials->partialAmplitudes[parIndex][localHop+loopDir] - partials->partialAmplitudes[parIndex][localHop];
+		oscillatorNextAmp[parCnt] =
+				(partials->partialAmplitudes[parIndex][localHop]
+						+ ampFixedDeltas[parIndex] * steps) * parDamp;
+		if(oscillatorNextAmp[parCnt] > 1)
+				oscillatorNextAmp[parCnt] = 1;
+		if ((loopDir == -1) && (localHop = 1) && (currentHopReminder == 1))
+			oscillatorNextAmp[parCnt] = 0;
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
+				- oscillatorAmplitudes[parCnt]) / hopCounter;
+
+		// finally update current mapping between oscillators and partials
+		indicesMapping[parIndex] = parCnt;
+		parCnt++;
+	}
+	actPartNum = parCnt;
+	// [NEON] if not multiple of 4...
+	if (actPartNum % 4 != 0)
+		addFakeOsc();
+}
+
+void OscillatorBank::nextPartialHop() {
+	unsigned int parIndex, localHop;
+	float parDamp = 1;
+	int currentPartialHop = currentHop / overSampling;
+
+	// if going backwards in the loop, get previous frame active partials...
+	actPartNum = partials->activePartialNum[currentPartialHop - loopDirShift];
+	actPart    = partials->activePartials[currentPartialHop - loopDirShift];
+
+	envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release]
+
+	int parCnt = 0;
+	int steps = overSampling - 1; // steps to reach next hop [partial or bank] from previous partial hop
+
+	for (int i = 0; i < actPartNum; i++) {
+		// find partial and frame
+		parIndex = actPart[i];
+		//localHop	= partials->localPartialFrames[currentPartialHop][parIndex];
+		localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
+
+		// if this partial was over nyquist on previous hop...
+		if (nyquistCut[parIndex]) {
+			// ...restart from safe values
+			oscillatorPhases[parCnt] = 0;
+			//TODO add freqmove dependency
+			oscillatorNextNormFreq[parCnt] =
+					(partials->partialFrequencies[parIndex][localHop]
+							+ freqFixedDeltas[parIndex] * steps
+									* (1 - loopDirShift)) * frequencyScaler
+							* prevPitchMultiplier;
+			oscillatorNextAmp[parCnt] = 0;
+		} else if (loopDir == 1) // otherwise recover phase, target freq and target amp from previous frame
+		{
+			if ((localHop != 0) || (overSampling > 1)) {
+				oscillatorPhases[parCnt] =
+						phaseCopies[indicesMapping[parIndex]];
+				oscillatorNextNormFreq[parCnt] =
+						nextNormFreqCopies[indicesMapping[parIndex]];
+				oscillatorNextAmp[parCnt] =
+						nextAmpCopies[indicesMapping[parIndex]];
+			} else // first oscillator hop [both for bank and partial], so no previous data are available
+			{
+				oscillatorPhases[parCnt] = 0;
+				//TODO add freqmove dependency
+				oscillatorNextNormFreq[parCnt] =
+						partials->partialFrequencies[parIndex][localHop]
+								* frequencyScaler * prevPitchMultiplier;
+				parDamp = calculateParDamping(parIndex, prevHopNumTh,
+						prevAdsrVal, oscillatorNextNormFreq[parCnt],
+						prevFilterNum, prevFilterFreqs, prevFilterQ);
+				oscillatorNextAmp[parCnt] =
+						partials->partialAmplitudes[parIndex][localHop]
+								* parDamp;
+				if(oscillatorNextAmp[parCnt] > 1)
+						oscillatorNextAmp[parCnt] = 1;
+				freqFixedDeltas[parIndex] =
+						partials->partialFreqDelta[parIndex][localHop + loopDir]
+								* loopDir / overSampling;
+				ampFixedDeltas[parIndex] =
+						partials->partialAmpDelta[parIndex][localHop + loopDir]
+								* loopDir / overSampling;
+			}
+		} else {
+			if (localHop != partials->partialNumFrames[parIndex] - 1) {
+				oscillatorPhases[parCnt] =
+						phaseCopies[indicesMapping[parIndex]];
+				oscillatorNextNormFreq[parCnt] =
+						nextNormFreqCopies[indicesMapping[parIndex]];
+				oscillatorNextAmp[parCnt] =
+						nextAmpCopies[indicesMapping[parIndex]];
+			} else // first oscillator hop [going backwards - both for bank and partial] , so no previous data are available
+			{
+				oscillatorPhases[parCnt] = 0;
+				//TODO add freqmove dependency
+				oscillatorNextNormFreq[parCnt] =
+						partials->partialFrequencies[parIndex][localHop]
+								* frequencyScaler * prevPitchMultiplier;
+				parDamp = calculateParDamping(parIndex, prevHopNumTh,
+						prevAdsrVal, oscillatorNextNormFreq[parCnt],
+						prevFilterNum, prevFilterFreqs, prevFilterQ);
+				oscillatorNextAmp[parCnt] =
+						partials->partialAmplitudes[parIndex][localHop]
+								* parDamp;
+				if(oscillatorNextAmp[parCnt] > 1)
+						oscillatorNextAmp[parCnt] = 1;
+				freqFixedDeltas[parIndex] =
+						partials->partialFreqDelta[parIndex][localHop + loopDir]
+								* loopDir / overSampling;
+				ampFixedDeltas[parIndex] =
+						partials->partialAmpDelta[parIndex][localHop + loopDir]
+								* loopDir / overSampling;
+			}
+		}
+		// remove aliasing, skipping partial over nyquist freq
+		if (oscillatorNextNormFreq[parCnt] > nyqNorm) {
+			//cout << nyqNorm << endl;
+			nyquistCut[parIndex] = true;
+			continue;
+		}
+		nyquistCut[parIndex] = false;
+
+		// first set up freq, cos filter affects amplitude damping according to freq content
+		oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts
+		// save next values, current for next round
+		oscillatorNextNormFreq[parCnt] = (freqMovement
+				* (partials->partialFrequencies[parIndex][localHop + loopDir]
+						- freqFixedDeltas[parIndex] * steps * loopDirShift)
+				* frequencyScaler
+				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
+				* pitchMultiplier;
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
+				- oscillatorNormFrequencies[parCnt]) / hopCounter;
+		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
+		oscillatorNormFreqDerivatives[parCnt] = freqMovement
+				* oscillatorNormFreqDerivatives[parCnt]
+				+ (1 - freqMovement)
+						* ((oscStatNormFrequenciesMean[parIndex]
+								* pitchMultiplier)
+								- oscillatorNormFrequencies[parCnt])
+						/ hopCounter;
+
+		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
+				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
+
+		// now amplitudes
+		oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts
+		// save next values, current for next round
+		//delta = partials->partialAmplitudes[parIndex][localHop+loopDir] - partials->partialAmplitudes[parIndex][localHop];
+		oscillatorNextAmp[parCnt] =
+				(partials->partialAmplitudes[parIndex][localHop + loopDir]
+						- (ampFixedDeltas[parIndex]) * steps * loopDirShift)
+						* parDamp;
+		if(oscillatorNextAmp[parCnt] > 1)
+			oscillatorNextAmp[parCnt] = 1;
+
+		// to avoid bursts when transients are played backwards
+		if ((loopDir == -1) && (localHop - 1 == 0) && (overSampling == 1)) {
+			oscillatorNextAmp[parCnt] = 0;
+		}
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
+				- oscillatorAmplitudes[parCnt]) / hopCounter;
+
+		// if next is not going to loop boundaries, get next deltas [same direction]
+		if ((((currentPartialHop + loopDir) * overSampling != loopEndHop)
+				|| (loopDir == -1))
+				&& (((currentPartialHop + loopDir) * overSampling + loopDir
+						!= loopStartHop) || (loopDir == 1))) {
+			freqFixedDeltas[parIndex] =
+					partials->partialFreqDelta[parIndex][localHop + loopDir]
+							* loopDir / overSampling;
+			ampFixedDeltas[parIndex] =
+					partials->partialAmpDelta[parIndex][localHop + loopDir]
+							* loopDir / overSampling;
+		} else // .. otherwise, keep deltas but change sign [co swe change direction]
+		{
+			freqFixedDeltas[parIndex] = -freqFixedDeltas[parIndex];
+			ampFixedDeltas[parIndex] = -ampFixedDeltas[parIndex];
+		}
+
+		// finally update current mapping between oscillators and partials
+		indicesMapping[parIndex] = parCnt;
+		parCnt++;
+	}
+	actPartNum = parCnt;
+	// [NEON] if not multiple of 4...
+	if (actPartNum % 4 != 0)
+		addFakeOsc();
+
+	updatePrevControls();
+}
+
+void OscillatorBank::addFakeOsc() {
+	// ...calculate difference
+	int newPartNum = (actPartNum + 3) & ~0x3;
+	// ...add fake oscillators until total num is multiple of 4
+	for (int i = actPartNum; i < newPartNum; i++) {
+		oscillatorAmplitudes[i] = 0;
+		oscillatorNormFrequencies[i] = 0;
+		oscillatorAmplitudeDerivatives[i] = 0;
+		oscillatorNormFreqDerivatives[i] = 0;
+		oscillatorPhases[i] = 0;
+	}
+	// ...and update num of active partials
+	actPartNum = newPartNum;
+}
+
+void OscillatorBank::play(float vel) {
+	// set attack and release params according to velocity
+	//adsr.setAttackRate((minAttackTime + ((1 - vel) * deltaAttackTime)) * rate);
+	adsr.setAttackRate(minAttackTime * rate);
+	//adsr.setReleaseRate((minReleaseTime + (1 - vel) * deltaReleaseTime) * rate);
+	adsr.setReleaseRate(minReleaseTime * rate);
+
+	// set timbre
+	hopNumTh = log((1 - vel) + 1) / log(2) * 20000;
+
+	state = bank_toreset;
+}
+
+//---------------------------------------------------------------------------------------------------------------------------
+// private methods
+//---------------------------------------------------------------------------------------------------------------------------
+
+bool OscillatorBank::loader(char *filename, int hopsize, int samplerate) {
+	rate = samplerate;
+	loaded = parser.parseFile(filename, hopsize, samplerate);
+	return loaded;
+}
+
+int OscillatorBank::jumpToHop() {
+	int jumpGap = abs(jumpHop - currentHop / overSampling); // gaps in partial reference
+
+	// can't jump to self dude
+	if (jumpGap == 0)
+		return 1;
+
+	// direction is in general maintained with jump
+	if (jumpHop == 0)
+		setDirection(1);
+	else if (jumpHop == lastHop)
+		setDirection(-1);
+
+	dbox_printf("\tJump from %d to %d\n", currentHop / overSampling, jumpHop);
+	dbox_printf("\tdirection %d\n", loopDir);
+
+	currentHop = jumpHop * overSampling;
+
+	if (nextEnvState() != 0)
+		return 0; // release has ended!
+
+	checkSpeed();
+
+	int parIndex, localHop, targetHop;
+	float parDamp = 1;
+	int currentPartialHop = currentHop / overSampling;
+	int targetPartialHop = jumpHop;
+
+	actPartNum = partials->activePartialNum[currentPartialHop];
+	actPart = partials->activePartials[currentPartialHop];
+	int targetActParNum = partials->activePartialNum[targetPartialHop];
+	unsigned int *targetActPar = partials->activePartials[targetPartialHop];
+
+	envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release]
+
+	int parCnt = 0;
+	int currentHopReminder = currentHop % overSampling;
+
+	// steps to walk where i am [bank of partial hop] from previous partial hop
+	int steps = currentHopReminder * (overSampling != 1); // no oversampling 0, oversampling and going ff currentHopReminder
+
+	for (int i = 0; i < actPartNum; i++) {
+		// find partial and frame
+		parIndex = actPart[i];
+		//localHop	= partials->localPartialFrames[currentPartialHop][parIndex];
+		localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
+
+		// if this partial was over nyquist on previous hop...
+		if (nyquistCut[parIndex]) {
+			// ...restart from safe values
+			oscillatorPhases[parCnt] = 0;
+			//TODO add freqmove dependency
+			oscillatorNextNormFreq[parCnt] =
+					(partials->partialFrequencies[parIndex][localHop]
+							+ freqFixedDeltas[parIndex] * steps * loopDir)
+							* frequencyScaler * prevPitchMultiplier;
+			oscillatorNextAmp[parCnt] = 0;
+		} else if (loopDir == 1) {// otherwise recover phase, target freq and target amp from previous frame
+			if ((localHop != 0)
+					|| ((overSampling > 1) && (currentHopReminder != 0))) {
+				oscillatorPhases[parCnt] =
+						phaseCopies[indicesMapping[parIndex]];
+				oscillatorNextNormFreq[parCnt] =
+						nextNormFreqCopies[indicesMapping[parIndex]];
+				oscillatorNextAmp[parCnt] =
+						nextAmpCopies[indicesMapping[parIndex]];
+			} else { // first oscillator hop [both for bank and partial], so no previous data are available
+				oscillatorPhases[parCnt] = 0;
+				//TODO add freqmove dependency
+				oscillatorNextNormFreq[parCnt] =
+						partials->partialFrequencies[parIndex][localHop]
+								* frequencyScaler * prevPitchMultiplier;
+				parDamp = calculateParDamping(parIndex, prevHopNumTh,
+						prevAdsrVal, oscillatorNextNormFreq[parCnt],
+						prevFilterNum, prevFilterFreqs, prevFilterQ);
+				oscillatorNextAmp[parCnt] =
+						partials->partialAmplitudes[parIndex][localHop]
+								* parDamp;
+				if(oscillatorNextAmp[parCnt] > 1)
+						oscillatorNextAmp[parCnt] = 1;
+			}
+		} else {
+			if (( (unsigned)localHop != partials->partialNumFrames[parIndex] - 1)
+					|| ((overSampling > 1) && (currentHopReminder != 0))) {
+				oscillatorPhases[parCnt] =
+						phaseCopies[indicesMapping[parIndex]];
+				oscillatorNextNormFreq[parCnt] =
+						nextNormFreqCopies[indicesMapping[parIndex]];
+				oscillatorNextAmp[parCnt] =
+						nextAmpCopies[indicesMapping[parIndex]];
+			} else // first oscillator hop [going backwards - both for bank and partial] , so no previous data are available, so retrieve where i am
+			{
+				oscillatorPhases[parCnt] = 0;
+				//TODO add freqmove dependency
+				oscillatorNextNormFreq[parCnt] =
+						partials->partialFrequencies[parIndex][localHop]
+								* frequencyScaler * prevPitchMultiplier;
+				parDamp = calculateParDamping(parIndex, prevHopNumTh,
+						prevAdsrVal, oscillatorNextNormFreq[parCnt],
+						prevFilterNum, prevFilterFreqs, prevFilterQ);
+				oscillatorNextAmp[parCnt] =
+						partials->partialAmplitudes[parIndex][localHop]
+								* parDamp;
+				if(oscillatorNextAmp[parCnt] > 1)
+						oscillatorNextAmp[parCnt] = 1;
+			}
+		}
+		// remove aliasing, skipping partial over nyquist freq
+		if (oscillatorNextNormFreq[parCnt] > nyqNorm) {
+			//cout << nyqNorm << endl;
+			nyquistCut[parIndex] = true;
+			continue;
+		}
+		nyquistCut[parIndex] = false;
+
+		// check what happens of this partial at target hop
+		float targetFreqVal, targetAmpVal;
+		//targetHop = partials->localPartialFrames[targetPartialHop][parIndex];
+		targetHop = targetPartialHop - partials->partialStartFrame[parIndex];
+
+		if (targetHop == -1)
+			targetFreqVal = targetAmpVal = 0;
+		else {
+			targetFreqVal = partials->partialFrequencies[parIndex][targetHop]
+					* frequencyScaler; // pitch shift will be multiplied later!!!
+			targetAmpVal = partials->partialFrequencies[parIndex][targetHop]; // parDamp will be multiplied later!!!
+		}
+
+		// first set up freq, cos filter affects amplitude damping according to freq content
+		oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts
+		// save next values, current for next round
+		oscillatorNextNormFreq[parCnt] = (freqMovement * targetFreqVal
+				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
+				* pitchMultiplier;
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
+				- oscillatorNormFrequencies[parCnt]) / hopCounter;
+		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
+		oscillatorNormFreqDerivatives[parCnt] = freqMovement
+				* oscillatorNormFreqDerivatives[parCnt]
+				+ (1 - freqMovement)
+						* ((oscStatNormFrequenciesMean[parIndex]
+								* pitchMultiplier)
+								- oscillatorNormFrequencies[parCnt])
+						/ hopCounter;
+
+		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
+				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
+
+		// now amplitudes
+		oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts
+		// save next values, current for next round
+		oscillatorNextAmp[parCnt] = targetAmpVal * parDamp;
+		if(oscillatorNextAmp[parCnt] > 1)
+				oscillatorNextAmp[parCnt] = 1;
+		// to avoid bursts when transients are played backwards
+		if ((loopDir == -1) && (targetHop == 0)
+				&& ((overSampling == 1) || (currentHopReminder == 0))) {
+			oscillatorNextAmp[parCnt] = 0;
+		}
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
+				- oscillatorAmplitudes[parCnt]) / hopCounter;
+
+		//if partial does not die at target, calculate deltas according to direction
+		if (targetHop != -1) {
+			freqFixedDeltas[parIndex] =
+					partials->partialFreqDelta[parIndex][targetHop] * loopDir
+							/ overSampling;
+			ampFixedDeltas[parIndex] =
+					partials->partialAmpDelta[parIndex][targetHop] * loopDir
+							/ overSampling;
+		}
+
+		// finally update current mapping between oscillators and partials
+		indicesMapping[parIndex] = parCnt;
+		parCnt++;
+	}
+	actPartNum = parCnt;
+
+	// now add the ones that start at target hop!
+	for (int i = 0; i < targetActParNum; i++) {
+		// find partial and frame
+		parIndex = targetActPar[i];
+		//targetHop	= partials->localPartialFrames[targetPartialHop][parIndex];
+		targetHop = targetPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
+
+		// check if this partials was already active before the jump
+		//localHop = partials->localPartialFrames[currentPartialHop][parIndex];
+		localHop = currentPartialHop - partials->partialStartFrame[parIndex];
+
+		// if yes, skip it
+		if (localHop != -1)
+			continue;
+
+		// otherwise add it to active bunch and calcucalte values
+
+		// first set up freq, cos filter affects amplitude damping according to freq content
+		oscillatorNormFrequencies[parCnt] = 0;
+		// save next values, current for next round
+		oscillatorNextNormFreq[parCnt] = (freqMovement
+				* partials->partialFrequencies[parIndex][targetHop]
+				* frequencyScaler
+				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
+				* pitchMultiplier;
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
+				- oscillatorNormFrequencies[parCnt]) / hopCounter;
+		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
+		oscillatorNormFreqDerivatives[parCnt] = freqMovement
+				* oscillatorNormFreqDerivatives[parCnt]
+				+ (1 - freqMovement)
+						* ((oscStatNormFrequenciesMean[parIndex]
+								* pitchMultiplier)
+								- oscillatorNormFrequencies[parCnt])
+						/ hopCounter;
+
+		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
+				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
+
+		// now amplitudes
+		oscillatorAmplitudes[parCnt] = 0;
+		// save next values, current for next round
+		oscillatorNextAmp[parCnt] =
+				partials->partialFrequencies[parIndex][targetHop] * parDamp;
+		if(oscillatorNextAmp[parCnt] > 1)
+				oscillatorNextAmp[parCnt] = 1;
+		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
+		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
+				- oscillatorAmplitudes[parCnt]) / hopCounter;
+
+		//calculate deltas according to direction
+		freqFixedDeltas[parIndex] =
+				partials->partialFreqDelta[parIndex][targetHop] * loopDir
+						/ overSampling;
+		ampFixedDeltas[parIndex] =
+				partials->partialAmpDelta[parIndex][targetHop] * loopDir
+						/ overSampling;
+
+		// finally update current mapping between oscillators and partials
+		indicesMapping[parIndex] = parCnt;
+		parCnt++;
+
+	}
+	// [NEON] if not multiple of 4...
+	if (actPartNum % 4 != 0)
+		addFakeOsc();
+
+	updatePrevControls();
+
+	jumpHop = -1;
+
+	return 0;
+}
+
+int OscillatorBank::nextEnvState() {
+	/*
+	 envState = Attack.getState();	// to determine what state we are in [attack, decay, sustain, release]
+
+	 // osc bank is playing the tail and the tail ends...
+	 if( (state == bank_playing)&&(envState == env_idle) )
+	 {
+	 state = bank_stopped;	// ...stop bank
+	 return 1;					// and return immediately
+	 }
+	 else if( (envState == env_attack) || (envState == env_decay) )
+	 {
+	 // run envelopes until next frame
+	 dampWeight	= Attack.process(hopSize);
+	 }
+	 else if(envState == env_release)
+	 {
+	 // run envelopes until next frame
+	 dampWeight 	= Attack.process(hopSize);
+	 releaseDamp = Release.process(hopSize);
+	 }*/
+
+	envState = adsr.getState();
+	// osc bank is playing the tail and the tail ends...
+	if ((state == bank_playing) && (envState == env_idle)) {
+		state = bank_stopped; // ...stop bank
+		adsrVal = 0;
+		return 1; // and return immediately
+	} else
+		adsrVal = adsr.process(hopSize);
+
+	return 0;
+}
+
+void OscillatorBank::checkDirection() {
+	// end of the loop or end of file
+	if (((currentHop >= loopEndHop) && (loopDir == 1))
+			|| ((currentHop >= lastHop) && (loopDir == 1))) {
+		// move backwards
+		setDirection(-1);
+		//dbox_printf("backward from %d\n", loopEndHop);
+	} else if (((currentHop <= loopStartHop) && (loopDir == -1))
+			|| ((currentHop <= 0) && (loopDir == -1))) // start of the loop or start of file
+			{
+		// move forward
+		setDirection(1);
+		//dbox_printf("forward from %d\n", loopStartHop);
+	}
+}
+
+void OscillatorBank::checkSpeed() {
+	// speed control [alike on highways, LOL]
+	if (nextSpeed > 0) {
+		nextSpeed = (nextSpeed < maxSpeed) ? nextSpeed : maxSpeed;
+		nextSpeed = (nextSpeed > minSpeed) ? nextSpeed : minSpeed;
+		speed = nextSpeed;
+		nextSpeed = -1;
+	}
+	hopCounter = hopSize / speed;
+}
+
+int OscillatorBank::checkJump() {
+	//check if has to jump somewhere
+	if (jumpHop > -1) {
+		// needs to jump!
+		if (jumpToHop() == 0)
+			return 0;
+	}
+	return 1; // no jump
+}
+
+bool OscillatorBank::checkOversampling() {
+	//TODO fix this, but need andrew to fix oversampling multiple of period size
+	// if partialsHopSize is not a multiple of oversampling, change hop size to periodically match next partial hop
+	if (hopSizeReminder > 0) {
+		// if next osc bank hop overtakes next partial hop...
+		if ((currentHop + loopDir) * hopSize > partialsHopSize) {
+			hopSize = hopSizeReminder; // ...shrink osc bank hop size to match partial hop
+			return true; // and set next hop as matching with next partial hop
+		}
+	} else if (((currentHop + (1 - loopDirShift)) % overSampling) == 0) // if next osc bank hop matches next partial hop
+		return true; // ...mark next hop as partial hop
+
+	return false; // ,otherwise mark next hop as osc bank hop
+}
+
+void OscillatorBank::updatePrevControls() {
+	prevAdsrVal = adsrVal;
+	prevAmpTh = ampTh;
+	prevHopNumTh = hopNumTh;
+	prevPitchMultiplier = pitchMultiplier;
+	prevFreqMovement = freqMovement;
+	prevFilterNum = filterNum;
+	memcpy(prevFilterFreqs, filterFreqs, filterNum * sizeof(float));
+	memcpy(prevFilterQ, filterQ, filterNum * sizeof(float));
+}
+
+float OscillatorBank::calculateParDamping(int parIndex, int hopNTh,
+		float adsrVl, float nextFreq, int filNum, float *filFreq, float *filQ) {
+	float parDamp = 1;
+
+	// timbre
+	parDamp = ((float) (oscStatNumHops[parIndex] + 1)) / (hopNTh + 1);
+	parDamp = (parDamp > 1) ? 1 : parDamp;
+	parDamp = adsrVl * parDamp;
+
+	//filters
+
+	float filterWeights[MAX_TOUCHES];
+	float filterDamp[MAX_TOUCHES];
+	float filDist;
+	float filterWeightsAcc;
+	float filDmp;
+	float filAmp;
+
+// 		band reject notch filter
+//		float dist, dmp;
+//		for(int k=0; k<filterNum; k++)
+//		{
+//			dist = fabs(oscillatorNextNormFreq[parCnt]-filterFreqs[k]);
+//			if(dist<=filterQ[k])
+//			{
+//				dmp 	= dist/filterQ[k];
+//				parDamp *= dmp*dmp*dmp;
+//			}
+//		}
+
+
+	// each filter is a band pass notch filter
+
+	// if at least one is active
+	if (filNum > 0) {
+		// reset values
+		filDist = 0;
+		filterWeightsAcc = 0;
+		filDmp = 0;
+		filAmp = 0;
+		// for each filter
+		for (int k = 0; k < filNum; k++) {
+			// here are a couple of kludges to boost sound output of hi freq filters
+
+			// damping effect of filter increases with distance, but decreases with filter frequency [kludge]
+			float mul = ((filterMaxF-nextFreq)/filterMaxF) * 0.9 + 0.1 ;
+			//filDist = fabs(nextFreq - filFreq[k])*( ((exp(a*4)-1)/EXP_DENOM) * 0.9 + 0.1 );
+			filDist = fabs(nextFreq - filFreq[k])*mul;
+
+			// these to merge all filters contributions according to distance
+			filterWeights[k] = filterMaxF - filDist;
+			filterWeightsAcc += filterWeights[k];
+			// freqs very close to filter center are slightly amplified
+			// the size of this amp area and the effect of amplification increase with frequency [kludge]
+			if (filDist
+					< filterAmpMinF
+							+ (filterAmpMaxF*(1-mul) - filterAmpMinF) * (1 - filQ[k]) )
+				filAmp = filQ[k] * filterAmpMul*(1-mul);
+			else
+				filAmp = 0;
+			// actual damping
+			filDmp = 1 / (filDist * filQ[k]);
+			filDmp = (filDmp > 1) ? 1 : filDmp;
+			// sum damp+amplification
+			filterDamp[k] = filDmp + filAmp;
+		}
+		// do weighted mean to merge all filters contributions
+		filDmp = 0;
+		for (int k = 0; k < filNum; k++)
+			filDmp += filterDamp[k] * filterWeights[k];
+		filDmp /= filterWeightsAcc;
+		// apply
+		parDamp *= filDmp;
+	}
+
+
+	return parDamp;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/OscillatorBank.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,240 @@
+/*
+ * OscillatorBank.h
+ *
+ *  Created on: May 23, 2014
+ *      Author: Victor Zappi and Andrew McPherson
+ */
+
+#ifndef OSCILLATORBANK_H_
+#define OSCILLATORBANK_H_
+
+
+#include <string>
+
+#include "spear_parser.h"
+#include "ADSR.h"
+#include "config.h"
+
+using namespace std;
+
+enum OscBankstates {bank_stopped, bank_playing, bank_toreset};
+
+class OscillatorBank
+{
+public:
+	OscillatorBank();
+	OscillatorBank(string filename, int hopsize=-1, int samplerate=44100);
+	OscillatorBank(char *filename, int hopsize=-1, int samplerate=44100);
+	~OscillatorBank();
+	float *oscillatorPhases;
+	float *oscillatorNormFrequencies;
+	float *oscillatorNormFreqDerivatives;
+	float *oscillatorAmplitudes;
+	float *oscillatorAmplitudeDerivatives;
+	float *oscStatNormFrequenciesMean;
+	float *oscStatNumHops;
+	OscBankstates state;
+	bool note;
+	int actPartNum;
+	unsigned int *actPart;
+	int hopCounter;
+	int lookupTableSize;
+	float *lookupTable;
+	float ampTh;
+	int hopNumTh;
+	float pitchMultiplier;
+	float freqMovement;
+	int filterNum;
+	float filterFreqs[5];
+	float filterQ[5];
+	float filterMaxF;
+	float filterAmpMinF;
+	float filterAmpMaxF;
+	float filterAmpMul;
+
+	bool loadFile(string filename, int hopsize=-1, int samplerate=44100);
+	bool loadFile(char *filename, int hopsize=-1, int samplerate=44100);
+	bool initBank(int oversamp=1);
+	void resetOscillators();
+	int getHopSize() { return hopSize; }
+	void nextHop();
+	void setLoopHops(int start, int end);
+	void play(float vel);
+	void stop();
+	void afterTouch(float vel);
+	int getEnvelopeState();
+	float getFrequencyScaler();
+	void setSpeed(float sp);
+	float getSpeed();
+	float getMaxSpeed();
+	float getMinSpeed();
+	void setJumpHop(int hop);
+	int getLastHop();
+	int getCurrentHop() { return currentHop; }
+
+private:
+
+	bool loaded;
+	int numOfPartials;
+	int numOfOscillators;
+	int partialsHopSize;
+	int overSampling;
+	int hopSize;
+	int hopSizeReminder;
+	int oscBankHopSize;
+	float frequencyScaler;
+	float nyqNorm;
+	int lastHop;
+	int currentHop;
+	int	loopDir;
+	int loopDirShift;
+	int loopStartHop;
+	int loopEndHop;
+	int *indicesMapping;
+	float *phaseCopies;
+	float *oscillatorNextNormFreq;
+	float *oscillatorNextAmp;
+	float *nextNormFreqCopies;
+	float *nextAmpCopies;
+	float *freqFixedDeltas;
+	float *ampFixedDeltas;
+	bool *nyquistCut;
+	Spear_parser parser;
+	Partials *partials;
+	ADSR adsr;
+	float minAttackTime;
+	float deltaAttackTime;
+	float minReleaseTime;
+	float deltaReleaseTime;
+	int envState;
+	int rate;
+	float speed;
+	float nextSpeed;
+	float maxSpeed;
+	float minSpeed;
+	int jumpHop;
+	float adsrVal;
+	float prevAdsrVal;
+	float prevAmpTh;
+	int prevHopNumTh;
+	float prevPitchMultiplier;
+	float prevFreqMovement;
+	int prevFilterNum;
+	float prevFilterFreqs[5];
+	float prevFilterQ[5];
+
+	bool loader(char *filename, int hopsize=-1, int samplerate=44100);
+	void addFakeOsc();
+	void nextOscBankHop();
+	void nextPartialHop();
+	int jumpToHop();
+	void setDirection(int dir);
+	int nextEnvState();
+	void checkDirection();
+	void checkSpeed();
+	int checkJump();
+	bool checkOversampling();
+	void updatePrevControls();
+	float calculateParDamping(int parIndex, int hopNTh, float adsrVl, float nextFreq,
+							  int filNum, float *filFreq, float *filQ);
+};
+
+inline bool OscillatorBank::loadFile(string filename, int hopsize, int samplerate)
+{
+	return loader((char *)filename.c_str(), hopsize, samplerate);
+}
+
+inline bool OscillatorBank::loadFile(char *filename, int hopsize, int samplerate)
+{
+	return loader(filename, hopsize, samplerate);
+}
+
+inline void OscillatorBank::setLoopHops(int start, int end)
+{
+	if(start > end)
+		end = start;
+
+	if(start<0)
+		start = 0;
+	else if(start>lastHop)
+		start = 0;
+	if(end < 1)
+		end = 1;
+	end = (end<=lastHop) ? end : lastHop;
+
+	// set it, take into consideration hop oversampling
+	loopStartHop = start*overSampling;
+	loopEndHop	 = end*overSampling;
+}
+
+inline void OscillatorBank::stop()
+{
+	note = false;
+	adsr.gate(0);
+}
+
+inline float OscillatorBank::getFrequencyScaler()
+{
+	return frequencyScaler;
+}
+
+inline void OscillatorBank::afterTouch(float vel)
+{
+	hopNumTh = log((1-vel)+1)/log(2)*20000;
+	if(adsr.getState()==env_attack)
+		adsr.setAttackRate( (minAttackTime + ( (1-vel)*deltaAttackTime )) * rate );
+	adsr.setReleaseRate( (minReleaseTime+(1-vel)*deltaReleaseTime)* rate );
+}
+
+inline int OscillatorBank::getEnvelopeState()
+{
+	return envState;
+}
+
+inline void OscillatorBank::setSpeed(float sp)
+{
+	nextSpeed = sp;
+}
+
+inline float OscillatorBank::getSpeed()
+{
+	return speed;
+}
+
+inline float OscillatorBank::getMaxSpeed()
+{
+	return maxSpeed;
+}
+
+inline float OscillatorBank::getMinSpeed()
+{
+	return minSpeed;
+}
+
+inline void OscillatorBank::setJumpHop(int hop)
+{
+	if(hop<0)
+		return;
+	hop = (hop<=lastHop) ? hop : lastHop;
+	jumpHop = hop;
+}
+
+inline void OscillatorBank::setDirection(int dir)
+{
+	if(dir>=0)
+	{
+		loopDir			= 1;
+		loopDirShift	= 0;
+	}
+	else
+	{
+		loopDir			= -1;
+		loopDirShift	= 1;
+	}
+}
+
+inline int OscillatorBank::getLastHop()
+{
+	return lastHop;
+}
+#endif /* OSCILLATORBANK_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/PinkNoise.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,16 @@
+/*
+ * PinkNoise.cpp
+ *
+ *  Created on: Oct 15, 2013
+ *      Author: Victor Zappi
+ */
+
+#include "PinkNoise.h"
+
+// miserable definition to init static const array members...otherwise gets error when PinkNoise.h is included into another header file
+const float PinkNoise::A[] = { 0.02109238, 0.07113478, 0.68873558 }; // rescaled by (1+P)/(1-P)
+const float PinkNoise::P[] = { 0.3190,  0.7756,  0.9613  };
+
+
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/PinkNoise.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,49 @@
+#ifndef _PinkNoise_H
+#define _PinkNoise_H
+
+// Technique by Larry "RidgeRat" Trammell 3/2006
+// http://home.earthlink.net/~ltrammell/tech/pinkalg.htm
+// implementation and optimization by David Lowenfels
+
+#include <cstdlib>
+#include <ctime>
+#include <stdlib.h>
+
+#define PINK_NOISE_NUM_STAGES 3
+
+class PinkNoise {
+public:
+  PinkNoise() {
+  srand ( time(NULL) ); // initialize random generator
+    clear();
+  }
+
+  void clear() {
+    for( size_t i=0; i< PINK_NOISE_NUM_STAGES; i++ )
+      state[ i ] = 0.0;
+    }
+
+  float tick() {
+    static const float RMI2 = 2.0 / float(RAND_MAX); // + 1.0; // change for range [0,1)
+    static const float offset = A[0] + A[1] + A[2];
+
+  // unrolled loop
+    float temp = float( rand() );
+    state[0] = P[0] * (state[0] - temp) + temp;
+    temp = float( rand() );
+    state[1] = P[1] * (state[1] - temp) + temp;
+    temp = float( rand() );
+    state[2] = P[2] * (state[2] - temp) + temp;
+    return ( A[0]*state[0] + A[1]*state[1] + A[2]*state[2] )*RMI2 - offset;
+  }
+
+protected:
+  float state[ PINK_NOISE_NUM_STAGES ];
+  static const float A[ PINK_NOISE_NUM_STAGES ];
+  static const float P[ PINK_NOISE_NUM_STAGES ];
+};
+
+//const float PinkNoise::A[] = { 0.02109238, 0.07113478, 0.68873558 }; // rescaled by (1+P)/(1-P)
+//const float PinkNoise::P[] = { 0.3190,  0.7756,  0.9613  };
+
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/StatusLED.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,90 @@
+/*
+ * StatusLED.cpp
+ *
+ * Routines for manipulating the status LED
+ *
+ * (c) 2014 Andrew McPherson and Victor Zappi
+ * QMUL, Centre for Digital Music
+ */
+
+#include <iostream>
+#include "StatusLED.h"
+#include <GPIOcontrol.h>
+
+extern int gShouldStop;
+extern int gVerbose;
+
+using namespace std;
+
+StatusLED::StatusLED() {
+	gpio_number = -1;
+	milliseconds_on = 0;
+	milliseconds_off = 100;
+	blink_thread = -1;
+}
+
+StatusLED::~StatusLED() {
+	if(gpio_number >= 0) {
+		this_should_stop = true;
+		pthread_join(blink_thread, NULL);
+		gpio_unexport(gpio_number);
+	}
+}
+
+bool StatusLED::init(int gpio_pin) {
+	gpio_number = gpio_pin;
+	this_should_stop = false;
+
+	if(gpio_export(gpio_number)) {
+		if(gVerbose)
+			cout << "Warning: couldn't export status LED pin\n";
+	}
+	if(gpio_set_dir(gpio_number, OUTPUT_PIN)) {
+		if(gVerbose)
+			cout << "Couldn't set direction on status LED pin\n";
+		return false;
+	}
+	if(gpio_set_value(gpio_number, LOW)) {
+		if(gVerbose)
+			cout << "Couldn't set value on status LED pin\n";
+		return false;
+	}
+
+
+	if ( pthread_create(&blink_thread, NULL, static_blink_loop, this) )
+	{
+		cout << "Error:unable to create status LED thread" << endl;
+		return false;
+	}
+
+	return true;
+}
+
+void StatusLED::on() {
+	milliseconds_on = 100;
+	milliseconds_off = 0;
+}
+
+void StatusLED::off() {
+	milliseconds_on = 0;
+	milliseconds_off = 100;
+}
+
+void StatusLED::blink(int ms_on, int ms_off) {
+	milliseconds_on = ms_on;
+	milliseconds_off = ms_off;
+}
+
+void* StatusLED::blink_loop(void *) {
+	while(!gShouldStop && !this_should_stop) {
+		if(milliseconds_on != 0)
+			gpio_set_value(gpio_number, HIGH);
+		usleep(1000 * milliseconds_on);
+		if(gShouldStop)
+			break;
+		if(milliseconds_off != 0)
+			gpio_set_value(gpio_number, LOW);
+		usleep(1000 * milliseconds_off);
+	}
+	pthread_exit(NULL);
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/StatusLED.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,39 @@
+/*
+ * StatusLED.h
+ *
+ *
+ */
+
+#ifndef STATUSLED_H_
+#define STATUSLED_H_
+
+#include <pthread.h>
+#include <unistd.h>
+
+class StatusLED
+{
+public:
+	StatusLED();
+	~StatusLED();
+
+	bool init(int gpio_pin);
+
+	void on();
+	void off();
+	void blink(int ms_on, int ms_off);
+
+	static void *static_blink_loop(void *data) {
+		((StatusLED*)data)->blink_loop(NULL);
+		return 0;
+	}
+
+	void* blink_loop(void *);
+
+private:
+	int gpio_number;
+	int milliseconds_on, milliseconds_off;
+	bool this_should_stop;
+	pthread_t blink_thread;
+};
+
+#endif // STATUSLED_H_
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/audio_routines.S	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,185 @@
+@
+@ audio_routines.S
+@
+@ NEON-based functions for time-critical audio processing
+@
+@ Andrew McPherson 2014
+@ Queen Mary University of London
+@
+
+	.syntax unified
+	.arch armv7-a
+	.fpu neon
+
+@ 	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
+@							  int activePartialNum, int lookupTableSize,
+@							  float *phases, float *frequencies, float *amplitudes,
+@							  float *freqDerivatives, float *ampDerivatives,
+@							  float *lookupTable);
+
+@ Registers:
+@    r0: numAudioFrames        How many frames to render
+@    r1: audioOut              Buffer for audio output samples [stereo]
+@    r2: activePartialNum      How many active partials to render
+@    r3: lookupTableSize       Size of lookup table
+@    ---- other arguments start on the stack and are moved: -----
+@    r4: phases                Phase of each oscillator (pointer)
+@    r5: frequencies           Normalised frequency of each oscillator (pointer)
+@    r6: amplitudes            Normalised amplitude of each oscillator (pointer)
+@    r7: freqDerivatives       Derivative of frequency for each oscillator (pointer)
+@    r8: ampDerivatives        Derivative of amplitude for each oscillator (pointer)
+@    r9: lookupTable           Lookup table containing one oscillation
+@
+@ Alignment requirements:
+@    audioOut: 8-byte boundary
+@    phases: 16-byte boundary
+@    frequencies: 16-byte boundary
+@    amplitudes: 16-byte boundary
+@    freqDerivatives: 16-byte bounary
+@    ampDerivatives: 16-byte boundary
+@    lookupTable: 4-byte boundary (TODO: check this)
+
+	.align	2
+	.global	oscillator_bank_neon
+	.thumb
+	.thumb_func
+	.type	oscillator_bank_neon, %function
+oscillator_bank_neon:
+
+
+dSample		.dn		D6.F32
+qPhases		.qn		Q8.F32
+dPhases_0	.dn		D16.F32
+dPhases_1	.dn		D17.F32
+qFreqs		.qn		Q9.F32
+dFreqs_0	.dn		D18.F32
+dFreqs_1	.dn		D19.F32
+qAmps		.qn		Q10.F32
+dAmps_0		.dn		D20.F32
+dAmps_1		.dn		D21.F32
+qFreqDs		.qn		Q11.F32
+dFreqDs_0	.dn		D22.F32
+dFreqDs_1	.dn		D23.F32
+qAmpDs		.qn		Q12.F32
+dAmpDs_0	.dn		D24.F32
+dAmpDs_1	.dn		D25.F32
+
+qBaseInts	.qn		Q13.U32		@ Base indexes: unsigned ints x4
+dBaseInts_0	.dn		D26.U32
+dBaseInts_1	.dn		D27.U32
+qFractions  .qn     Q14.F32		@ Fraction indexes: floats x4
+qTableBase	.qn		Q15.U32		@ Base of lookup table
+
+	cmp r0, #0					@ Check for trivial case 1: zero frames
+	it eq
+	bxeq lr						@ Return if that's the case (otherwise might have odd behaviour)
+	cmp r2, #4					@ Check for trivial case 2: zero oscillators
+	it lt
+	bxlt lr						@ Return if that's the case
+
+	push {r4-r11}				@ Now arguments start 32 bytes above SP
+    add r11, sp, #32			@ Pointer to 32 bytes into the stack
+    ldm r11, {r4-r9}			@ Load 6 arguments into registers
+
+	vdup qTableBase, r9			@ Move lookup table base index into 4 ints
+
+	@ Outer loop: iterate over the number of oscillators, choosing 4 at a
+	@ time to work with.
+oscbank_oscillator_loop:
+	vld1 {dPhases_0, dPhases_1}, [r4]		@ no increment; will store at end of sample loop
+	vld1 {dFreqs_0, dFreqs_1}, [r5]
+	vld1 {dAmps_0, dAmps_1}, [r6]
+	vld1 {dFreqDs_0, dFreqDs_1}, [r7]!		@ increment; won't update at end of sample loop
+	vld1 {dAmpDs_0, dAmpDs_1}, [r8]!
+
+	push {r0-r1,r4-r8}
+	@ --- inner loop: iterate over the number of samples ---
+oscbank_sample_loop:
+	vcvt qBaseInts, qPhases		     		@ Take floor(phases)
+	vmov q2.f32, #1.0						@ Load 1.0 into every slot of q2
+	vshl q0.U32, qBaseInts, #2				@ Shift the indexes left 2 (*4 for float addressing)
+	vcvt qFractions, qBaseInts				@ int back to float
+	vadd q0.U32, q0.U32, qTableBase			@ Find memory addresses
+
+	vmov r4, r5, d0							@ Move two indexes to ARM registers
+	vmov r6, r7, d1							@ Move two more indexes to ARM registers
+	vsub qFractions, qPhases, qFractions	@ fraction = phase - floor(phase)
+
+	vldr.64	d0, [r4]						@ Load two consecutive floats at each location
+	vldr.64 d1, [r5]						@ These hold the previous and following samples in the table
+	vldr.64	d2, [r6]						@ TODO: check whether these work at 4-byte alignment
+	vldr.64 d3, [r7]
+
+	@ Format at this point:
+	@ Osc0(before) Osc0(after) Osc1(before) Osc1(after) Osc2(before) Osc2(after) Osc3(before) Osc3(after)
+	@ We want:
+	@ Osc0(before) Osc1(before) Osc2(before) Osc3(before) Osc0(after) Osc1(after) Osc2(after) Osc3(after)
+
+	vuzp.32 q0, q1							@ Now q0 contains before, q1 contains after
+	vsub q2.f32, q2.f32, qFractions			@ q2 = 1.0 - fraction
+	vmul q1.f32, q1.f32, qFractions			@ q1 = fraction * after
+	vmul q0.f32, q0.f32, q2.f32				@ q0 = (1.0 - fraction) * before
+
+	vadd qPhases, qPhases, qFreqs			@ Update phases
+	vadd qFreqs, qFreqs, qFreqDs			@ Update frequencies
+
+	vadd q0.f32, q0.f32, q1.f32				@ Add two interpolated components to get the final sample
+	vdup q2.u32, r3							@ Put lookup table size into each element of q2
+	vcvt qBaseInts, qPhases					@ Take floor of new phases
+	vmul q0.f32, q0.f32, qAmps				@ Multiply samples by current amplitude
+
+	vld1 dSample, [r1]						@ Load the current stereo samples
+	vpadd d2.f32, d0.f32, d1.f32			@ Pairwise accumulate q0 (output sample) into d2
+
+	vand q2, q2, qBaseInts					@ Logical AND of new phase int leaves 1 bit set only if phase >= table size
+	vpadd d3.f32, d2.f32, d2.f32			@ Pairwise accumulate d2 into d0 --> d0[0] and d0[1] both hold total of 4 oscillators
+	vadd qAmps, qAmps, qAmpDs				@ Update amplitudes
+	vcvt q0.f32, q2.u32						@ Convert int back to float after AND operation
+
+	vadd  dSample, dSample, d3.f32			@ Add oscillator outputs to each channel
+
+	subs r0, r0, #1							@ numFrames--
+	vsub qPhases, qPhases, q0.f32			@ Keep phases in table range
+	vst1 dSample, [r1]!						@ Store back in buffer and increment by 8
+
+	it gt
+	bgt oscbank_sample_loop					@ Loop if numFrames > 0
+
+	@ --- end inner loop ---
+	pop {r0-r1,r4-r8}						@ Restore registers: restores audioOut and numFrames, among others
+
+	vst1 {dPhases_0, dPhases_1}, [r4]!		@ Store phases back to array
+	vst1 {dFreqs_0, dFreqs_1}, [r5]!		@ Store frequencies back to array
+	vst1 {dAmps_0, dAmps_1}, [r6]!			@ Store amplitudes back to array
+											@ No need to update r7, r8
+
+	subs r2, r2, #4							@ numPartials -= 4
+	it  gt
+	bgt oscbank_oscillator_loop	@ Loop if numPartials > 0
+
+    pop {r4-r11}
+	bx lr
+
+
+@   void wavetable_interpolate_neon(int numSamplesIn, int numSamplesOut,
+@                              float *tableIn, float *tableOut);
+
+@ Registers:
+@    r0: numSamplesIn          Size of the input table
+@    r1: numSamplesOut         Size of the output table
+@    r2: tableIn               Pointer to input table
+@    r3: tableOut              Pointer to output table
+
+@ Alignment requirements:
+@    tableIn: 8-byte boundary
+@    tableOut: 8-byte boundary
+
+	.align	2
+	.global	wavetable_interpolate_neon
+	.thumb
+	.thumb_func
+	.type	wavetable_interpolate_neon, %function
+wavetable_interpolate_neon:
+    @ TODO
+
+    bx lr
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/config.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,43 @@
+/*
+ * config.h
+ *
+ * Global settings for D-Box project
+ *
+ * Andrew McPherson and Victor Zappi 2014
+ */
+
+
+#ifndef DBOX_CONFIG_H_
+#define DBOX_CONFIG_H_
+
+
+/* Number of maximum touches used by the TouchKey sensors */
+#define MAX_TOUCHES 5
+
+// for sensor 1 filter
+#define EXP_DENOM 53.5981500331			// exp(4)-1
+
+/* Define this to use Xenomai real-time extensions */
+#define DBOX_USE_XENOMAI
+//#define OLD_OSCBANK
+
+#ifdef DBOX_USE_XENOMAI
+// Xenomai-specific includes
+#include <sys/mman.h>
+
+#include <native/task.h>
+#include <native/timer.h>
+#include <rtdk.h>
+#endif
+
+#ifdef DBOX_USE_XENOMAI
+
+#define dbox_printf rt_printf
+
+#else
+
+#define dbox_printf printf
+
+#endif
+
+#endif /* DBOX_CONFIG_H */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/how_to_build_dbox.txt	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,35 @@
+The D-Box code can be built on the board using the scripts included with
+Bela. To build the code, do the following from the scripts/ directory:
+
+./setup_board.sh    # Only if Bela is not already copied to the board
+./build_project.sh -n ../projects/d-box  # -n means don't run yet
+
+You will now need to copy the .dbx sound files to the box. These need to
+be in a directory: /root/d-box/sounds
+
+Log into the board via ssh to create the d-box directory, or run:
+
+ssh root@192.168.7.2 "mkdir /root/d-box"
+
+Now copy the .dbx files from a source of your choice to the board. Assuming
+they are in a directory called "sounds":
+
+scp -r sounds root@192.168.7.2:/root/d-box/
+
+Now you can run the D-Box program to make sure it plays. You will need the
+sensors and matrix connected to the box (i.e. a complete hardware D-Box).
+From the scripts/ directory:
+
+./run_project.sh -f -c "-q 24 -r 25 -t 2 -s -p 2 -l 0 -u 0 -i 1 -n f -g 3 -v"
+
+If this plays properly, the D-Box can then be set to run automatically on boot:
+
+./set_startup.sh -l -c "-q 24 -r 25 -t 2 -s -p 2 -l 0 -u 0 -i 1 -n f -g 3 -v"
+
+If at any time you want to stop the D-Box program running, you can run:
+
+./stop_running.sh
+
+To turn off further autoboot, run:
+
+./set_startup.sh -n
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/logger.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,188 @@
+/*
+ * logger.cpp
+ *
+ *  Created on: Aug 6, 2014
+ *      Author: VIctor Zappi and Andrew McPherson
+ */
+
+#include "logger.h"
+
+// main extern vars
+extern int gShouldStop;
+extern int gVerbose;
+
+// file nanme extern vars
+extern char gId;
+extern char gGroup;
+
+
+// logged extern vars
+extern int s0TouchNum;
+extern float s0Touches_[MAX_TOUCHES];
+extern float s0Size_[MAX_TOUCHES];
+extern int s0LastIndex;
+
+extern int s1TouchNum;
+extern float s1Touches_[MAX_TOUCHES];
+extern float s1Size_[MAX_TOUCHES];
+extern int s1LastIndex;
+
+extern int fsr;
+
+
+
+string logPath			= "/boot/uboot/instrumentLog";
+string logFileIncipit	= "/datalog";
+string logFileName		= "";
+ofstream logFile;
+timeval logTimeVal;
+unsigned long long logTimeOrig;
+int logCnt				= 0;	// counts how many lines so far
+int logCntSwap			= 50;	// how many log lines before closing and re-opening the file
+
+
+// create the log file, using incremental name convention
+int initLogLoop()
+{
+	if(gVerbose==1)
+		cout << "---------------->Init Log Thread" << endl;
+
+
+	// transform chars into strings via stringstream objs
+	stringstream id_ss, group_ss, freedom_ss;
+	id_ss 		<< gId;
+	group_ss 	<< gGroup;
+
+	int logNum	= -1;
+	int logMax	= -1;
+	int pathLen	= logPath.length() + logFileIncipit.length() + 4;	// + 4 is: "_", id, group, "_"
+	glob_t globbuf;
+
+	// check how many log files are already there, and choose name according to this
+	glob( (logPath + logFileIncipit + "*").c_str(), 0, NULL, &globbuf);
+
+	// cycle through all and find the highest index
+	for(unsigned int i=0; i<globbuf.gl_pathc; i++)
+	{
+		// playing with 0-9 char digits, forming a number from 0 to 9999
+		logNum  = (globbuf.gl_pathv[i][pathLen]-48)   * 1000;	// 42 to 45 are the indices of the chars forming the file index
+		logNum += (globbuf.gl_pathv[i][pathLen+1]-48) * 100;
+		logNum += (globbuf.gl_pathv[i][pathLen+2]-48) * 10;
+		logNum +=  globbuf.gl_pathv[i][pathLen+3]-48;
+		if(logNum > logMax)
+			logMax = logNum;
+	}
+	logNum = logMax + 1;	// new index
+
+	globfree(&globbuf);
+
+	ostringstream numString;
+	numString << setw (4) << setfill ('0') << logNum;	// set integer with 4 figures
+
+	// here are the new names: PATH + DIR + INCIPIT + _ + id + group + freedom + _ + NUM (4figures) + _A.txt
+	logFileName	= logPath + logFileIncipit;
+	logFileName	+= "_" + id_ss.str() + group_ss.str() + freedom_ss.str();
+	logFileName	+= "_" + numString.str();	 //static_cast<ostringstream*>( &(ostringstream() << logNum) )->str();
+	logFileName	+= ".txt";
+
+
+	// create new files
+	FILE *fp_a		= fopen(logFileName.c_str(), "wb");
+	if(!fp_a)
+	{
+		dbox_printf("Cannot create files...\n");
+		return 2;
+	}
+	fclose(fp_a);
+
+	// ready to append
+	logFile.open(logFileName.c_str(), ios::out | ios::app);
+
+	dbox_printf("Logging on file %s\n", logFileName.c_str());
+
+	return 0;
+}
+
+
+void writeData(unsigned long long time)
+{
+
+	float fsr_ = ((float)(1799-fsr)/1799.0);
+	logFile << time 				<< "\t"		// timestamp
+			<< s0TouchNum			<< "\t";	// sensor 0 touch count
+	for(int i=0; i<MAX_TOUCHES; i++)
+		logFile << s0Touches_[i] 	<< "\t";	// sensor 0 touch pos x
+	for(int i=0; i<MAX_TOUCHES; i++)
+		logFile << s0Size_[i] 		<< "\t";	// sensor 0 touch size
+	logFile << s0LastIndex 			<< "\t"		// sensor 0 last index
+			<< fsr_					<< "\t"		// sensor 0 FSR pressure
+			<< s1TouchNum			<< "\t";	// sensor 1 touch count
+	for(int i=0; i<MAX_TOUCHES; i++)
+		logFile << s1Touches_[i] 	<< "\t";	// sensor 1 touch pos x
+	for(int i=0; i<MAX_TOUCHES; i++)
+		logFile	<< s1Size_[i] 		<< "\t";	// sensor 1 touch size
+	logFile << s1LastIndex 			<< "\t"		// sensor 1 last index
+	//... AND SO ON
+			<< "\n";
+
+	//dbox_printf("%d\n", s0LastIndex);
+	//dbox_printf("s0TouchNum: %d\t s0Touches[0]: %f\t s0Size[0]: %f\t s0LastIndex: %d\n", s0TouchNum, s0Touches_[0], s0Size_[0], s0LastIndex);
+
+}
+
+void logData(unsigned long long time)
+{
+	// if it's time to change write-file
+	if(logCnt >= logCntSwap)
+	{
+		logFile.close();	// close file, dump stream
+		logCnt = 0;		// ready for another whole round
+
+		// open again, ready to append
+		logFile.open(logFileName.c_str(), ios::out | ios::app);
+	}
+
+	writeData(time);
+
+	logCnt++;
+}
+
+
+
+
+void *logLoop(void *)
+{
+	set_realtime_priority(10);
+
+	if(gVerbose==1)
+		dbox_printf("_________________Log Thread!\n");
+
+	// get time reference
+	gettimeofday(&logTimeVal, NULL);
+	logData(0);
+
+	logTimeOrig = logTimeVal.tv_usec;
+	logTimeOrig *= 0.001;					// from usec to msec
+	logTimeOrig += logTimeVal.tv_sec*1000;	// from sec to msec
+
+	usleep(5000);
+
+	while(!gShouldStop)
+	{
+		gettimeofday(&logTimeVal, NULL);
+		unsigned long long currentTime = logTimeVal.tv_usec;
+		currentTime *= 0.001;					// from usec to msec
+		currentTime += logTimeVal.tv_sec*1000;	// from sec to msec
+
+		logData(currentTime-logTimeOrig);
+
+		usleep(5000);
+	}
+
+	if(logFile!=NULL)
+		logFile.close();
+
+	dbox_printf("log thread ended\n");
+
+	return (void *)0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/logger.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,31 @@
+/*
+ * logger.h
+ *
+ *  Created on: Aug 6, 2014
+ *      Author: Victor Zappi and Andrew McPherson
+ */
+
+#ifndef LOGGER_H_
+#define LOGGER_H_
+
+#include <string.h>
+#include <pthread.h>
+#include <stdio.h>
+#include <unistd.h>
+#include <fstream>		// file handle
+#include <iostream>		// stringstream
+#include <sstream>		// stringstream
+#include <glob.h>		// alternative to dirent.h to handle files in dirs
+#include <iomanip>		// setfill
+#include <sys/time.h>	// elapsed time
+
+#include "config.h"
+#include "prio.h"
+
+using namespace std;
+
+int initLogLoop();
+void *logLoop(void *);
+
+
+#endif /* LOGGER_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,491 @@
+/*
+ *  RTAudio.cpp
+ *
+ *  Central control code for hard real-time audio on BeagleBone Black
+ *  using PRU and Xenomai Linux extensions. This code began as part
+ *  of the Hackable Instruments project (EPSRC) at Queen Mary University
+ *  of London, 2013-14.
+ *
+ *  (c) 2014 Victor Zappi and Andrew McPherson
+ *  Queen Mary University of London
+ */
+
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <strings.h>
+#include <math.h>
+#include <iostream>
+#include <signal.h>		// interrupt handler
+#include <assert.h>
+#include <vector>
+#include <dirent.h>		// to handle files in dirs
+#include <mntent.h>		// to check if device is mounted
+#include <sys/mount.h>	// mount()
+#include <sys/time.h>	// elapsed time
+#include <ne10/NE10.h>	// neon library
+
+// thread priority
+#include <pthread.h>
+#include <sched.h>
+
+// get_opt_long
+#include <getopt.h>
+
+#include <Bela.h>
+#include "config.h"
+#include "sensors.h"
+#include "OscillatorBank.h"
+#include "StatusLED.h"
+#include "logger.h"
+
+using namespace std;
+
+//----------------------------------------
+// main variables
+//----------------------------------------
+vector<OscillatorBank*> gOscBanks;
+int gCurrentOscBank = 0;
+int gNextOscBank = 0;
+int oscBnkOversampling 				 = 1;	// oscillator bank frame oversampling
+
+const int kStatusLEDPin 	 = 30;	// P9-11 controls status LED
+StatusLED gStatusLED;
+
+pthread_t keyboardThread;
+pthread_t logThread;
+
+// general settings
+int gVerbose		= 0;        			// verbose flag
+bool forceKeyboard 	= true;					// activate/deactivate keyboard control
+bool forceSensors	= false;				// activate/deactivate sensor control
+bool forceLog		= true;					// activate/deactivate log on boot partition
+bool useSD   		= true;    				// activate/deactivate file loading from SD [as opposed to emmc]
+bool useAudioTest   = false;    			// activate/deactivate sensors and test audio only
+
+// audio settings
+unsigned int gPeriodSize = 8;				// period size for audio
+char* gPartialFilename = 0;					// name of the partials file to load
+bool gAudioIn = false;						// stereo audio in status
+
+int touchSensor0Address = 0x0C;				// I2C addresses of touch sensors
+int touchSensor1Address = 0x0B;
+int sensorType = 0;
+
+char sdPath[256]			= "/dev/mmcblk0p2";			// system path of the SD, partition 2
+char mountPath[256]			= "/root/d-box/usersounds";	// mount point of SD partition 2 [where user files are]
+char gUserDirName[256] 		= "usersounds";				// Directory in which user analysis files can be found [dir of mountPath]
+char gDefaultDirName[256] 	= "/root/d-box/sounds";		// Directory in which built in analysis files can be found
+char *gDirName;
+bool gIsLoading 			= false;
+int fileCnt 				= 0;
+std::vector <std::string> files;
+
+char gId	= 'f';	// from 0 to 14, hexadecimal [0-d]! f means not set
+char gGroup	= '2';	// 0 is no info, 1 info.   2 is not set
+
+// audio in filter
+extern ne10_float32_t *filterState[2];
+extern ne10_float32_t *filterIn[2];
+extern ne10_float32_t *filterOut[2];
+
+struct arg_data
+{
+   int  argc;
+   char **argv;
+};
+
+arg_data args;
+
+
+int readFiles()
+{
+	if(useSD)
+		gDirName = gUserDirName;
+	else
+		gDirName = gDefaultDirName;
+	DIR *dir;
+	struct dirent *ent;
+
+	// From http://stackoverflow.com/questions/612097/how-can-i-get-a-list-of-files-in-a-directory-using-c-or-c
+	if ((dir = opendir (gDirName)) != NULL) {
+		/* print all the files and directories within directory */
+		while ((ent = readdir (dir)) != NULL) {
+			// Ignore dotfiles and . and .. paths
+			if(!strncmp(ent->d_name, ".", 1))
+				continue;
+
+			//dbox_printf("%s\n", ent->d_name);
+
+			// take only .dbx and .txt files
+			string name = string(ent->d_name);
+			int len		= name.length();
+
+			bool dboxFile = false;
+
+			if( (name[len-4]=='.') && (name[len-3]=='d') && (name[len-2]=='b') && (name[len-1]=='x') )
+				dboxFile = true;
+			if( (name[len-4]=='.') && (name[len-3]=='t') && (name[len-2]=='x') && (name[len-1]=='t') )
+				dboxFile = true;
+
+			if(dboxFile)
+			{
+				fileCnt++;
+				//dbox_printf("%s\n", ent->d_name);
+				files.push_back( std::string( ent->d_name ) );
+			}
+		}
+		closedir (dir);
+	} else {
+		/* could not open directory */
+		printf("Could not open directory %s\n", gDirName);
+		return 1;
+	}
+
+	// order by name
+	std::sort( files.begin(), files.end() );
+
+	if(fileCnt==0)
+	{
+		printf("No .dbx or .txt files in %s!\n", gDirName);
+		return 1;
+	}
+
+	return 0;
+}
+
+/* Load sounds from the directory */
+void loadAudioFiles(bool loadFirstFile)
+{
+	char fullFileName[256];
+
+	if(loadFirstFile) {
+		strcpy(fullFileName, gDirName);
+		strcat(fullFileName, "/");
+		strncat(fullFileName, files[0].c_str(), 255 - strlen(gDirName));
+		dbox_printf("Loading first file %s...\n", fullFileName);
+		OscillatorBank *bank = new OscillatorBank(fullFileName);
+		if(bank->initBank(oscBnkOversampling)) {
+			bank->setLoopHops(100, bank->getLastHop());
+			gOscBanks.push_back(bank);
+		}
+	}
+
+	else {
+		for(int i=1; i<fileCnt; i++){
+			strcpy(fullFileName, gDirName);
+			strcat(fullFileName, "/");
+			strncat(fullFileName, files[i].c_str(), 255 - strlen(gDirName));
+			dbox_printf("Loading file %s...\n", fullFileName);
+			OscillatorBank *bank = new OscillatorBank(fullFileName);
+			if(bank->initBank(oscBnkOversampling)) {
+				bank->setLoopHops(100, bank->getLastHop());
+				gOscBanks.push_back(bank);
+			}
+		}
+	}
+}
+
+// adapted from http://program-nix.blogspot.co.uk/2008/08/c-language-check-filesystem-is-mounted.html
+int checkIfMounted (char * dev_path)
+{
+	FILE * mtab				= NULL;
+	struct mntent * part	= NULL;
+	int is_mounted			= 0;
+
+	if ( ( mtab = setmntent ("/etc/mtab", "r") ) != NULL)
+	{
+		while ( ( part = getmntent ( mtab) ) != NULL)
+		{
+			if ( ( part->mnt_fsname != NULL ) && ( strcmp ( part->mnt_fsname, dev_path ) ) == 0 )
+			   is_mounted = 1;
+		}
+	endmntent(mtab);
+	}
+	return is_mounted;
+}
+
+int mountSDuserPartition()
+{
+	if(checkIfMounted(sdPath))
+	{
+		printf("device %s already mounted, fair enough, let's move on\n", sdPath);
+		return 0;
+	}
+	// if mount rootfs from SD [rootfs eMMC not used] or from eMMC via properly formatted SD [SD rootfs used as storage volume]
+	// we always use rootfs on SD as storage volume ----> "/dev/mmcblk0p2"
+	int ret = mount(sdPath, "/root/d-box/usersounds", "vfat",  0, NULL);
+	if (ret!=0)
+	{
+			printf("Error in mount...%s\n", strerror(ret));
+			return 1;
+	}
+	return 0;
+}
+
+int initSoundFiles()
+{
+	if(gVerbose==1)
+		cout << "---------------->Init Audio Thread" << endl;
+
+	if(useSD)
+	{
+		// mount the SD partition where user sounds are located
+		// [this is p2, p1 is already mounted and we will log data there]
+		if(mountSDuserPartition()!=0)
+			return -1;
+	}
+
+	gIsLoading = true;
+
+	// read files from SD and order them alphabetically
+	if(readFiles()!=0)
+		return 1;
+
+	// load first file into oscBank
+	loadAudioFiles(true);
+
+	return 0;
+}
+
+//---------------------------------------------------------------------------------------------------------
+
+// Handle Ctrl-C
+void interrupt_handler(int var)
+{
+	// kill keyboard thread mercilessly
+	if(forceKeyboard)
+		pthread_cancel(keyboardThread);
+
+	gShouldStop = true;
+}
+
+
+void parseArguments(arg_data args, BelaInitSettings *settings)
+{
+	// Default filename;
+	gPartialFilename = strdup("D-Box_sound_250_60_40_h88_2.txt");
+
+	const int kOptionAudioTest = 1000;
+
+	// TODO: complete this
+	struct option long_option[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"audioin", 1, NULL, 'i'},
+		{"file", 1, NULL, 'f'},
+		{"keyboard", 1, NULL, 'k'},
+		{"audio-test", 0, NULL, kOptionAudioTest},
+		{"sensor-type", 1, NULL, 't'},
+		{"sensor0", 1, NULL, 'q'},
+		{"sensor1", 1, NULL, 'r'},
+		{"log", 1, NULL, 'l'},
+		{"usesd", 1, NULL, 'u'},
+		{"oversamp", 1, NULL, 'o'},
+		{"boxnumber", 1, NULL, 'n'},
+		{"group", 1, NULL, 'g'},
+		{NULL, 0, NULL, 0},
+	};
+	int morehelp = 0;
+	int tmp = -1;
+
+	Bela_defaultSettings(settings);
+
+	while (1)
+	{
+		int c;
+		if ((c = Bela_getopt_long(args.argc, args.argv, "hf:ki:sq:r:t:l:u:o:n:g:", long_option, settings)) < 0)
+				break;
+		switch (c)
+		{
+		case 'h':
+				morehelp++;
+				break;
+		case 'f':
+				free(gPartialFilename);
+				gPartialFilename = strdup(optarg);
+				break;
+		case 'k':
+				forceKeyboard = true;
+				break;
+		case 'i':
+				gAudioIn = (atoi(optarg)==0) ? false : true;
+				break;
+		case 's':
+				forceSensors = true;
+				break;
+		case kOptionAudioTest:
+				useAudioTest = true;
+				break;
+		case 't':
+				sensorType = atoi(optarg);
+				break;
+		case 'q':
+				touchSensor0Address = atoi(optarg);
+				break;
+		case 'r':
+				touchSensor1Address = atoi(optarg);
+				break;
+		case 'l':
+				tmp = atoi(optarg);
+				if(tmp==0)
+					forceLog = false;
+				else if(tmp>0)
+					forceLog = true;
+				break;
+		case 'u':
+				tmp = atoi(optarg);
+				if(tmp==0)
+					useSD = false;
+				else if(tmp>0)
+					useSD = true;
+				break;
+		case 'o':
+				oscBnkOversampling = atoi(optarg);
+				break;
+		case 'n':
+				gId = *optarg;
+				cout << "-set box number to: " << gId << endl;
+				break;
+		case 'g':
+				gGroup = *optarg;
+				cout << "-set group to: " << gId << endl;
+				break;
+		default:
+				break;
+		}
+	}
+
+	gPeriodSize = settings->periodSize;
+	gVerbose = settings->verbose;
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	RT_TASK rtSensorThread;
+	const char rtSensorThreadName[] = "dbox-sensor";
+	int oscBankHopSize;
+
+	// Parse command-line arguments
+	args.argc = argc;
+	args.argv = argv;
+	parseArguments(args, &settings);
+
+	Bela_setVerboseLevel(gVerbose);
+	if(gVerbose == 1 && useAudioTest)
+		cout << "main() : running in audio test mode" << endl;
+
+	// Load sound files from directory
+	if(initSoundFiles() != 0)
+		return -1;
+
+	oscBankHopSize = gOscBanks[gCurrentOscBank]->getHopSize()/gOscBanks[gCurrentOscBank]->getMinSpeed();
+
+	// Initialise the audio device
+	if(Bela_initAudio(&settings, &oscBankHopSize) != 0)
+		return -1;
+
+	// Initialise the status LED
+	if(!gStatusLED.init(kStatusLEDPin)) {
+		if(gVerbose)
+			cout << "Couldn't initialise status LED pin\n";
+	}
+
+	// Free file name string which is no longer needed
+	if(gPartialFilename != 0)
+		free(gPartialFilename);
+
+	if(!useAudioTest) {
+		if(initSensorLoop(touchSensor0Address, touchSensor1Address, sensorType) != 0)
+			return -1;
+	}
+
+	if(gVerbose == 1)
+		cout << "main() : creating audio thread" << endl;
+
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// LED on...
+	gStatusLED.on();
+
+	if(forceSensors && !useAudioTest) {
+		if(gVerbose==1)
+			cout << "main() : creating control thread" << endl;
+
+		if(rt_task_create(&rtSensorThread, rtSensorThreadName, 0, BELA_AUDIO_PRIORITY - 5, T_JOINABLE | T_FPU)) {
+			  cout << "Error:unable to create Xenomai control thread" << endl;
+			  return -1;
+		}
+		if(rt_task_start(&rtSensorThread, &sensorLoop, 0)) {
+			  cout << "Error:unable to start Xenomai control thread" << endl;
+			  return -1;
+		}
+	}
+
+	if(forceKeyboard) {
+		if(gVerbose==1)
+			cout << "main() : creating keyboard thread" << endl;
+
+		if ( pthread_create(&keyboardThread, NULL, keyboardLoop, NULL) ) {
+		  cout << "Error:unable to create keyboard thread" << endl;
+		  return -1;
+		}
+	}
+
+	if(forceLog) {
+		if(gVerbose==1)
+			cout << "main() : creating log thread" << endl;
+
+		if(initLogLoop()!=0) {
+			cout << "Error:unable to create log thread" << endl;
+			return -1;
+		}
+
+		if ( pthread_create(&logThread, NULL, logLoop, NULL) ) {
+		  cout << "Error:unable to create keyboard thread" << endl;
+		  return -1;
+		}
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// load all other files into oscBanks
+	loadAudioFiles(false);
+	cout << "Finished loading analysis files\n";
+	gIsLoading = false;
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	Bela_stopAudio();
+
+	if(!useAudioTest)
+		rt_task_join(&rtSensorThread);
+
+	Bela_cleanupAudio();
+
+	pthread_join( keyboardThread, NULL);
+	pthread_join( logThread, NULL);
+
+	for(unsigned int i = 0; i < gOscBanks.size(); i++)
+		delete gOscBanks[i];
+
+	NE10_FREE(filterState[0]);
+	NE10_FREE(filterState[1]);
+	NE10_FREE(filterIn[0]);
+	NE10_FREE(filterIn[1]);
+	NE10_FREE(filterOut[0]);
+	NE10_FREE(filterOut[1]);
+
+	printf("Program ended\nBye bye\n");
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/prio.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,48 @@
+/*
+ * prio.cpp
+ *
+ *  Created on: May 14, 2014
+ *      Author: Victor Zappi
+ */
+
+#include "prio.h"
+using namespace std;
+//-----------------------------------------------------------------------------------------------------------
+// set wanted real-time priority to this thread
+//-----------------------------------------------------------------------------------------------------------
+void set_realtime_priority(int order)
+{
+    int ret;
+
+    // We'll operate on the currently running thread.
+    pthread_t this_thread = pthread_self();
+    // struct sched_param is used to store the scheduling priority
+	struct sched_param params;
+	// We'll set the priority to the maximum.
+	params.sched_priority = sched_get_priority_max(SCHED_FIFO) - order;
+
+	// Attempt to set thread real-time priority to the SCHED_FIFO policy
+	ret = pthread_setschedparam(this_thread, SCHED_FIFO, &params);
+	if (ret != 0) {
+		// Print the error
+		cout << "Unsuccessful in setting thread realtime prio" << endl;
+		return;
+	}
+
+	// Now verify the change in thread priority
+    int policy = 0;
+    ret = pthread_getschedparam(this_thread, &policy, &params);
+    if (ret != 0) {
+        cout << "Couldn't retrieve real-time scheduling parameters" << endl;
+        return;
+    }
+
+    // Check the correct policy was applied
+    if(policy != SCHED_FIFO) {
+        cout << "Scheduling is NOT SCHED_FIFO!" << endl;
+    }
+}
+
+//-----------------------------------------------------------------------------------------------------------
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/prio.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,22 @@
+/*
+ * prio.h
+ *
+ *  Created on: May 14, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef PRIO_H_
+#define PRIO_H_
+
+
+#include <pthread.h>
+#include <sched.h>
+#include <iostream>
+
+//-----------------------------------------------------------------------------------------------------------
+// set maximum real-time priority to this thread
+//-----------------------------------------------------------------------------------------------------------
+void set_realtime_priority(int order);
+//-----------------------------------------------------------------------------------------------------------
+
+#endif /* PRIO_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,870 @@
+/*
+ * render.cpp
+ *
+ *  Created on: May 28, 2014
+ *      Author: Victor Zappi
+ */
+
+#include <Bela.h>
+#include <PRU.h>
+
+#include "StatusLED.h"
+#include "config.h"
+#include "OscillatorBank.h"
+#include "FeedbackOscillator.h"
+#include "ADSR.h"
+#include "FIRfilter.h"
+#include <assert.h>
+#include <cmath>
+#include <vector>
+
+#undef DBOX_CAPE_TEST
+
+// Mappings from pin numbers on PCB to actual DAC channels
+// This gives the DAC and ADC connectors the same effective pinout
+// Update June 2016: this is no longer needed in the latest Bela
+// release, but is kept here for convenience: it used to be 
+// 6 4 2 0 1 3 5 7 for the DAC pins
+#define DAC_PIN0	0
+#define DAC_PIN1	1
+#define DAC_PIN2	2
+#define DAC_PIN3	3
+#define DAC_PIN4	4
+#define DAC_PIN5	5
+#define DAC_PIN6	6
+#define DAC_PIN7	7
+
+#define ADC_PIN0	0
+#define ADC_PIN1	1
+#define ADC_PIN2	2
+#define ADC_PIN3	3
+#define ADC_PIN4	4
+#define ADC_PIN5	5
+#define ADC_PIN6	6
+#define ADC_PIN7	7
+
+#define N_OCT		4.0	// maximum number of octaves on sensor 1
+
+extern vector<OscillatorBank*> gOscBanks;
+extern int gCurrentOscBank;
+extern int gNextOscBank;
+extern PRU *gPRU;
+extern StatusLED gStatusLED;
+extern bool gIsLoading;
+extern bool gAudioIn;
+
+float *gOscillatorBuffer1, *gOscillatorBuffer2;
+float *gOscillatorBufferRead, *gOscillatorBufferWrite;
+int gOscillatorBufferReadPointer		= 0;
+int gOscillatorBufferReadCurrentSize	= 0;
+int gOscillatorBufferWriteCurrentSize	= 0;
+bool gOscillatorNeedsRender				= false;
+
+int gMatrixSampleCount = 0;		// How many samples have elapsed on the matrix
+
+// Wavetable which changes in response to an oscillator
+float *gDynamicWavetable;
+int gDynamicWavetableLength;
+bool gDynamicWavetableNeedsRender = false;
+
+// These variables handle the hysteresis oscillator used for setting the playback speed
+bool gSpeedHysteresisOscillatorRising	= false;
+int gSpeedHysteresisLastTrigger			= 0;
+
+// These variables handle the feedback oscillator used for controlling the wavetable
+FeedbackOscillator gFeedbackOscillator;
+float *gFeedbackOscillatorTable;
+int gFeedbackOscillatorTableLength;
+
+// This comes from sensor.cpp where it records the most recent touch location on
+// sensor 0.
+extern float gSensor0LatestTouchPos;
+extern int gSensor0LatestTouchNum;
+float gPitchLatestInput = 0;
+
+extern float gSensor1LatestTouchPos[];
+//extern float gSensor1LatestTouchSizes[];
+extern int gSensor1LatestTouchCount;
+extern int gSensor1LatestTouchIndex;
+int gSensor1LastTouchIndex		= -1;
+int gSensor1InputDelayCounter	= -1;
+int gSensor1InputIndex			= 0;
+float gSensor1MatrixTouchPos[5]	= {0};
+
+// FSR value from matrix input
+extern int gLastFSRValue;
+
+// Loop points from matrix input 4
+const int gLoopPointsInputBufferSize	= 256;
+float gLoopPointsInputBuffer[gLoopPointsInputBufferSize];
+int gLoopPointsInputBufferPointer		= 0;
+float gLoopPointMin = 0, gLoopPointMax	= 0;
+
+// multiplier to activate or mute audio in
+int audioInStatus = 0;
+
+// xenomai timer
+SRTIME prevChangeNs = 0;
+
+// pitch vars
+float octaveSplitter;
+float semitones[((int)N_OCT*12)+1];
+float deltaTouch	= 0;
+float deltaWeightP	= 0.5 / 65536.0;
+float deltaWeightI	= 0.0005 / 65536.0;
+
+// filter vars
+ne10_fir_instance_f32_t filter[2];
+ne10_float32_t *filterIn[2];
+ne10_float32_t *filterOut[2];
+ne10_uint32_t blockSize;
+ne10_float32_t *filterState[2];
+ne10_float32_t prevFiltered[2];
+int filterGain = 80;
+ADSR PeakBurst[2];
+float peak[2];
+float peakThresh = 0.2;
+
+// Tasks for lower-priority calculation
+AuxiliaryTask gMediumPriorityRender, gLowPriorityRender;
+
+
+extern "C" {
+	// Function prototype for ARM assembly implementation of oscillator bank
+	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
+							  int activePartialNum, int lookupTableSize,
+							  float *phases, float *frequencies, float *amplitudes,
+							  float *freqDerivatives, float *ampDerivatives,
+							  float *lookupTable);
+
+	void wavetable_interpolate_neon(int numSamplesIn, int numSamplesOut,
+	                              float *tableIn, float *tableOut);
+}
+
+void wavetable_interpolate(int numSamplesIn, int numSamplesOut,
+                              float *tableIn, float *tableOut,
+                              float *sineTable, float sineMix);
+
+inline float hysteresis_oscillator(float input, float risingThreshold,
+									float fallingThreshold, bool *rising);
+
+void render_medium_prio();
+void render_low_prio();
+
+#ifdef DBOX_CAPE_TEST
+void render_capetest(int numMatrixFrames, int numAudioFrames, float *audioIn, float *audioOut,
+			uint16_t *matrixIn, uint16_t *matrixOut);
+#endif
+
+bool setup(BelaContext *context, void *userData) {
+	int oscBankHopSize = *(int *)userData;
+
+	if(context->analogChannels != 8) {
+		printf("Error: D-Box needs matrix enabled with 8 channels.\n");
+		return false;
+	}
+
+	// Allocate two buffers for rendering oscillator bank samples
+	// One will be used for writing in the background while the other is used for reading
+	// on the audio thread. 8-byte alignment needed for the NEON code.
+	if(posix_memalign((void **)&gOscillatorBuffer1, 8, oscBankHopSize * context->audioChannels * sizeof(float))) {
+		printf("Error allocating render buffers\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gOscillatorBuffer2, 8, oscBankHopSize * context->audioChannels * sizeof(float))) {
+		printf("Error allocating render buffers\n");
+		return false;
+	}
+	gOscillatorBufferWrite	= gOscillatorBuffer1;
+	gOscillatorBufferRead	= gOscillatorBuffer2;
+
+	memset(gOscillatorBuffer1, 0, oscBankHopSize * context->audioChannels * sizeof(float));
+	memset(gOscillatorBuffer2, 0, oscBankHopSize * context->audioChannels * sizeof(float));
+
+	// Initialise the dynamic wavetable used by the oscillator bank
+	// It should match the size of the static one already allocated in the OscillatorBank object
+	// Don't forget a guard point at the end of the table
+	gDynamicWavetableLength = gOscBanks[gCurrentOscBank]->lookupTableSize;
+	if(posix_memalign((void **)&gDynamicWavetable, 8, (gDynamicWavetableLength + 1) * sizeof(float))) {
+		printf("Error allocating wavetable\n");
+		return false;
+	}
+
+	gFeedbackOscillator.initialise(8192, 10.0, context->analogSampleRate);
+
+	for(int n = 0; n < gDynamicWavetableLength + 1; n++)
+		gDynamicWavetable[n] = 0;
+
+	// pitch
+	float midPos		= 0.5;
+	octaveSplitter		= 1.0 / N_OCT;
+	int numOfSemi		= 12*N_OCT;
+	int middleSemitone	= 12*N_OCT/2;
+	int lastSemitone	= middleSemitone+numOfSemi/2;
+	float inc 			= 1.0 / (N_OCT*12.0);
+	int i 				= -1;
+	for(int semi=middleSemitone; semi<=lastSemitone; semi++)
+		semitones[semi] = ( midPos + (++i)*inc) + 0.5;
+	i 					= 0;
+	for(int semi=middleSemitone-1; semi>=0; semi--)
+		semitones[semi] = ( midPos - (++i)*inc) + 0.5;
+
+	if(gAudioIn)
+		audioInStatus = 1;
+
+	// filter
+	blockSize		= context->audioFrames;
+	filterState[0]	= (ne10_float32_t *) NE10_MALLOC ((FILTER_TAP_NUM+blockSize-1) * sizeof (ne10_float32_t));
+	filterState[1]	= (ne10_float32_t *) NE10_MALLOC ((FILTER_TAP_NUM+blockSize-1) * sizeof (ne10_float32_t));
+	filterIn[0]		= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
+	filterIn[1]		= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
+	filterOut[0]	= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
+	filterOut[1]	= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
+	ne10_fir_init_float(&filter[0], FILTER_TAP_NUM, filterTaps, filterState[0], blockSize);
+	ne10_fir_init_float(&filter[1], FILTER_TAP_NUM, filterTaps, filterState[1], blockSize);
+
+	// peak outputs
+	PeakBurst[0].setAttackRate(.00001 * context->analogSampleRate);
+	PeakBurst[1].setAttackRate(.00001 * context->analogSampleRate);
+	PeakBurst[0].setDecayRate(.5 * context->analogSampleRate);
+	PeakBurst[1].setDecayRate(.5 * context->analogSampleRate);
+	PeakBurst[0].setSustainLevel(0.0);
+	PeakBurst[1].setSustainLevel(0.0);
+
+	// Initialise auxiliary tasks
+	if((gMediumPriorityRender = Bela_createAuxiliaryTask(&render_medium_prio, BELA_AUDIO_PRIORITY - 10, "dbox-calculation-medium")) == 0)
+		return false;
+	if((gLowPriorityRender = Bela_createAuxiliaryTask(&render_low_prio, BELA_AUDIO_PRIORITY - 15, "dbox-calculation-low")) == 0)
+		return false;
+
+	return true;
+}
+
+void render(BelaContext *context, void *userData)
+{
+#ifdef DBOX_CAPE_TEST
+	render_capetest(numMatrixFrames, numAudioFrames, audioIn, audioOut, matrixIn, matrixOut);
+#else
+	if(gOscBanks[gCurrentOscBank]->state==bank_toreset)
+		gOscBanks[gCurrentOscBank]->resetOscillators();
+
+	if(gOscBanks[gCurrentOscBank]->state==bank_playing)
+	{
+		assert(context->audioChannels == 2);
+
+#ifdef OLD_OSCBANK
+		memset(audioOut, 0, numAudioFrames *  * sizeof(float));
+
+		/* Render the oscillator bank. The oscillator bank function is written in NEON assembly
+		 * and it strips out all extra checks, so find out in advance whether we can render a whole
+		 * block or whether the frame will increment in the middle of this buffer.
+		 */
+
+		int framesRemaining = numAudioFrames;
+		float *audioOutWithOffset = audioOut;
+
+		while(framesRemaining > 0) {
+			if(gOscBanks[gCurrentOscBank]->hopCounter >= framesRemaining) {
+				/* More frames left in this hop than we need this time. Render and finish */
+				oscillator_bank_neon(framesRemaining, audioOutWithOffset,
+									 gOscBanks[gCurrentOscBank]->actPartNum, gOscBanks[gCurrentOscBank]->lookupTableSize,
+									 gOscBanks[gCurrentOscBank]->oscillatorPhases, gOscBanks[gCurrentOscBank]->oscillatorNormFrequencies,
+									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudes,
+									 gOscBanks[gCurrentOscBank]->oscillatorNormFreqDerivatives,
+									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudeDerivatives,
+									 gDynamicWavetable/*gOscBanks[gCurrentOscBank]->lookupTable*/);
+				gOscBanks[gCurrentOscBank]->hopCounter -= framesRemaining;
+				if(gOscBanks[gCurrentOscBank]->hopCounter <= 0)
+					gOscBanks[gCurrentOscBank]->nextHop();
+				framesRemaining = 0;
+			}
+			else {
+				/* More frames to render than are left in this hop. Render and decrement the
+				 * number of remaining frames; then advance to the next oscillator frame.
+				 */
+				oscillator_bank_neon(gOscBanks[gCurrentOscBank]->hopCounter, audioOutWithOffset,
+									 gOscBanks[gCurrentOscBank]->actPartNum, gOscBanks[gCurrentOscBank]->lookupTableSize,
+									 gOscBanks[gCurrentOscBank]->oscillatorPhases, gOscBanks[gCurrentOscBank]->oscillatorNormFrequencies,
+									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudes,
+									 gOscBanks[gCurrentOscBank]->oscillatorNormFreqDerivatives,
+									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudeDerivatives,
+									 gDynamicWavetable/*gOscBanks[gCurrentOscBank]->lookupTable*/);
+				framesRemaining -= gOscBanks[gCurrentOscBank]->hopCounter;
+				audioOutWithOffset +=  * gOscBanks[gCurrentOscBank]->hopCounter;
+				gOscBanks[gCurrentOscBank]->sampleCount += gOscBanks[gCurrentOscBank]->hopCounter;
+				gOscBanks[gCurrentOscBank]->nextHop();
+			}
+		}
+#else
+		for(unsigned int n = 0; n < context->audioFrames; n++) {
+			context->audioOut[2*n] 	  = gOscillatorBufferRead[gOscillatorBufferReadPointer++]+context->audioIn[2*n]*audioInStatus;
+			context->audioOut[2*n + 1] = gOscillatorBufferRead[gOscillatorBufferReadPointer++]+context->audioIn[2*n+1]*audioInStatus;
+
+			filterIn[0][n] = fabs(context->audioIn[2*n]);	// rectify for peak detection in 1
+			filterIn[1][n] = fabs(context->audioIn[2*n+1]);	// rectify for peak detection in 2
+
+			/* FIXME why doesn't this work? */
+			/*
+			if(gOscillatorBufferReadPointer == gOscillatorBufferCurrentSize/2) {
+				gOscillatorNeedsRender = true;
+				scheduleAuxiliaryTask(gLowPriorityRender);
+			} */
+
+			if(gOscillatorBufferReadPointer >= gOscillatorBufferReadCurrentSize) {
+				// Finished reading from the buffer: swap to the next buffer
+				if(gOscillatorBufferRead == gOscillatorBuffer1) {
+					gOscillatorBufferRead = gOscillatorBuffer2;
+					gOscillatorBufferWrite = gOscillatorBuffer1;
+				}
+				else {
+					gOscillatorBufferRead = gOscillatorBuffer1;
+					gOscillatorBufferWrite = gOscillatorBuffer2;
+				}
+
+				// New buffer size is whatever finished writing last hop
+				gOscillatorBufferReadCurrentSize = gOscillatorBufferWriteCurrentSize;
+				gOscillatorBufferReadPointer = 0;
+
+				gOscillatorNeedsRender = true;
+				Bela_scheduleAuxiliaryTask(gMediumPriorityRender);
+			}
+		}
+#endif
+	}
+	else
+	{
+		for(unsigned int n = 0; n < context->audioFrames; n++) {
+			context->audioOut[2*n] 	  = context->audioIn[2*n]*audioInStatus;
+			context->audioOut[2*n + 1] = context->audioIn[2*n+1]*audioInStatus;
+
+			filterIn[0][n] = fabs(context->audioIn[2*n]);	// rectify for peak detection in 1
+			filterIn[1][n] = fabs(context->audioIn[2*n+1]);	// rectify for peak detection in 2
+		}
+	}
+
+	// low pass filter audio in 1 and 2 for peak detection
+	ne10_fir_float_neon(&filter[0], filterIn[0], filterOut[0], blockSize);
+	ne10_fir_float_neon(&filter[1], filterIn[1], filterOut[1], blockSize);
+
+	for(unsigned int n = 0; n < context->analogFrames; n++) {
+
+
+		/* Matrix Out 0, In 0
+		 *
+		 * CV loop
+		 * Controls pitch of sound
+		 */
+		float touchPosInt = gSensor0LatestTouchPos;
+		if(touchPosInt < 0) touchPosInt = 0;
+		if(touchPosInt > 1.0) touchPosInt = 1.0;
+		context->analogOut[n*8 + DAC_PIN0] = touchPosInt;
+
+		gPitchLatestInput = context->analogIn[n*8 + ADC_PIN0];
+
+
+		/* Matrix Out 7
+		 *
+		 * Loop feedback with Matrix In 0
+		 * Controls discreet pitch
+		 */
+		float deltaTarget = 0;
+		int semitoneIndex = 0;
+		if(gSensor0LatestTouchNum>0)
+		{
+			// current pitch is gPitchLatestInput, already retrieved
+			semitoneIndex	= ( gPitchLatestInput * 12 * N_OCT  )+0.5;	// closest semitone
+			deltaTarget		= (semitones[semitoneIndex]-gPitchLatestInput);				// delta between pitch and target
+			deltaTouch 		+= deltaTarget*(deltaWeightI);								// update feedback [previous + current]
+		}
+		else
+			deltaTouch = 0;
+
+		float nextOut = touchPosInt  + deltaTarget*deltaWeightP + deltaTouch;			// add feedback to touch -> next out
+		if(nextOut < 0) nextOut = 0;												// clamp
+		if(nextOut > 1.0) nextOut = 1.0;										// clamp
+		context->analogOut[n*8 + DAC_PIN7] = nextOut;										// send next nextOut
+
+
+		/*
+		 * Matrix Out 1, In 1
+		 *
+		 * Hysteresis (comparator) oscillator
+		 * Controls speed of playback
+		 */
+		bool wasRising = gSpeedHysteresisOscillatorRising;
+		context->analogOut[n*8 + DAC_PIN1] = hysteresis_oscillator(context->analogIn[n*8 + ADC_PIN1], 48000.0/65536.0,
+																	16000.0/65536.0, &gSpeedHysteresisOscillatorRising);
+
+		// Find interval of zero crossing
+		if(wasRising && !gSpeedHysteresisOscillatorRising) {
+			int interval = gMatrixSampleCount - gSpeedHysteresisLastTrigger;
+
+			// Interval since last trigger will be the new hop size; calculate to set speed
+			if(interval < 1)
+				interval = 1;
+			//float speed = (float)gOscBanks[gCurrentOscBank]->getHopSize() / (float)interval;
+			float speed = 144.0 / interval;	// Normalise to a fixed expected speed
+			gOscBanks[gCurrentOscBank]->setSpeed(speed);
+
+			gSpeedHysteresisLastTrigger = gMatrixSampleCount;
+		}
+
+		/*
+		 * Matrix Out 2, In 2
+		 *
+		 * Feedback (phase shift) oscillator
+		 * Controls wavetable used for oscillator bank
+		 */
+
+		int tableLength = gFeedbackOscillator.process(context->analogIn[n*8 + ADC_PIN2], &context->analogOut[n*8 + DAC_PIN2]);
+		if(tableLength != 0) {
+			gFeedbackOscillatorTableLength = tableLength;
+			gFeedbackOscillatorTable = gFeedbackOscillator.wavetable();
+			gDynamicWavetableNeedsRender = true;
+			Bela_scheduleAuxiliaryTask(gLowPriorityRender);
+		}
+
+		/*
+		 * Matrix Out 3, In 3
+		 *
+		 * CV loop with delay for time alignment
+		 * Touch positions from sensor 1
+		 * Change every 32 samples (ca. 1.5 ms)
+		 */
+		volatile int touchCount = gSensor1LatestTouchCount;
+		if(touchCount == 0)
+			context->analogOut[n*8 + DAC_PIN3] = 0;
+		else {
+			int touchIndex = (gMatrixSampleCount >> 5) % touchCount;
+			context->analogOut[n*8 + DAC_PIN3] = gSensor1LatestTouchPos[touchIndex] * 56000.0f / 65536.0f;
+			if(touchIndex != gSensor1LastTouchIndex) {
+				// Just changed to a new touch output. Reset the counter.
+				// It will take 2*matrixFrames samples for this output to come back to the
+				// ADC input. But we also want to read near the end of the 32 sample block;
+				// let's say 24 samples into it.
+
+				// FIXME this won't work for p > 2
+				gSensor1InputDelayCounter = 24 + 2*context->analogFrames;
+				gSensor1InputIndex = touchIndex;
+			}
+			gSensor1LastTouchIndex = touchIndex;
+		}
+
+		if(gSensor1InputDelayCounter-- >= 0 && touchCount > 0) {
+			gSensor1MatrixTouchPos[gSensor1InputIndex] = context->analogIn[n*8 + ADC_PIN3];
+		}
+
+		/* Matrix Out 4
+		 *
+		 * Sensor 1 last pos
+		 */
+		touchPosInt = gSensor1LatestTouchPos[gSensor1LatestTouchIndex];
+		if(touchPosInt < 0) touchPosInt = 0;
+		if(touchPosInt > 1.0) touchPosInt = 1.0;
+		context->analogOut[n*8 + DAC_PIN4] = touchPosInt;
+
+		/* Matrix In 4
+		 *
+		 * Loop points selector
+		 */
+		gLoopPointsInputBuffer[gLoopPointsInputBufferPointer++] = context->analogIn[n*8 + ADC_PIN4];
+		if(gLoopPointsInputBufferPointer >= gLoopPointsInputBufferSize) {
+			// Find min and max values
+			float loopMax = 0, loopMin = 1.0;
+			for(int i = 0; i < gLoopPointsInputBufferSize; i++) {
+				if(gLoopPointsInputBuffer[i] < loopMin)
+					loopMin = gLoopPointsInputBuffer[i];
+				if(gLoopPointsInputBuffer[i] > loopMax/* && gLoopPointsInputBuffer[i] != 65535*/)
+					loopMax = gLoopPointsInputBuffer[i];
+			}
+
+			if(loopMin >= loopMax)
+				loopMax = loopMin;
+
+			gLoopPointMax = loopMax;
+			gLoopPointMin = loopMin;
+			gLoopPointsInputBufferPointer = 0;
+		}
+
+		/* Matrix Out 5
+		 *
+		 * Audio In 1 peak detection and peak burst output
+		 */
+
+		filterOut[0][n*2+1]	*= filterGain;
+		float burstOut		= PeakBurst[0].getOutput();
+		if( burstOut < 0.1)
+		{
+			if( (prevFiltered[0]>=peakThresh) && (prevFiltered[0]>=filterOut[0][n*2+1]) )
+			{
+				peak[0] = prevFiltered[0];
+				PeakBurst[0].gate(1);
+			}
+		}
+
+		PeakBurst[0].process(1);
+
+		float convAudio = burstOut*peak[0];
+		context->analogOut[n*8 + DAC_PIN5] = convAudio;
+		prevFiltered[0] = filterOut[0][n*2+1];
+		if(prevFiltered[0]>1)
+			prevFiltered[0] = 1;
+
+		/* Matrix In 5
+		 *
+		 * Dissonance, via changing frequency motion of partials
+		 */
+		float amount = (float)context->analogIn[n*8 + ADC_PIN5];
+		gOscBanks[gCurrentOscBank]->freqMovement = 1.0 - amount;
+
+
+
+
+		/* Matrix Out 6
+		 *
+		 * Audio In 2 peak detection and peak burst output
+		 */
+
+		filterOut[1][n*2+1]	*= filterGain;
+		burstOut			= PeakBurst[1].getOutput();
+		if( burstOut < 0.1)
+		{
+			if( (prevFiltered[1]>=peakThresh) && (prevFiltered[1]>=filterOut[1][n*2+1]) )
+			{
+				peak[1] = prevFiltered[1];
+				PeakBurst[1].gate(1);
+			}
+		}
+
+		PeakBurst[1].process(1);
+
+		convAudio = burstOut*peak[1];
+		context->analogOut[n*8 + DAC_PIN6] = convAudio;
+		prevFiltered[1] = filterOut[1][n*2+1];
+		if(prevFiltered[1]>1)
+			prevFiltered[1] = 1;
+
+		/* Matrix In 6
+		 *
+		 * Sound selector
+		 */
+		if(!gIsLoading) {
+			// Use hysteresis to avoid jumping back and forth between sounds
+			if(gOscBanks.size() > 1) {
+				float input = context->analogIn[n*8 + ADC_PIN6];
+				const float hystValue = 16000.0 / 65536.0;
+
+				float upHysteresisValue = ((gCurrentOscBank + 1) + hystValue) / gOscBanks.size();
+				float downHysteresisValue = (gCurrentOscBank - hystValue) / gOscBanks.size();
+
+				if(input > upHysteresisValue || input < downHysteresisValue) {
+					gNextOscBank = input * gOscBanks.size();
+					if(gNextOscBank < 0)
+						gNextOscBank = 0;
+					if((unsigned)gNextOscBank >= gOscBanks.size())
+						gNextOscBank = gOscBanks.size() - 1;
+				}
+			}
+		}
+
+		/*
+		 * Matrix In 7
+		 *
+		 * FSR from primary touch sensor
+		 * Value ranges from 0-1799
+		 */
+		gLastFSRValue = context->analogIn[n*8 + ADC_PIN7] * 1799.0;
+		//gLastFSRValue = 1799 - context->analogIn[n*8 + ADC_PIN7] * (1799.0 / 65535.0);
+		//dbox_printf("%i\n",gLastFSRValue);
+
+		gMatrixSampleCount++;
+	}
+
+#endif /* DBOX_CAPE_TEST */
+}
+
+// Medium-priority render function used for audio hop calculations
+void render_medium_prio()
+{
+
+	if(gOscillatorNeedsRender) {
+		gOscillatorNeedsRender = false;
+
+		/* Render one frame into the write buffer */
+		memset(gOscillatorBufferWrite, 0, gOscBanks[gCurrentOscBank]->hopCounter * 2 * sizeof(float)); /* assumes 2 audio channels */
+
+		oscillator_bank_neon(gOscBanks[gCurrentOscBank]->hopCounter, gOscillatorBufferWrite,
+							 gOscBanks[gCurrentOscBank]->actPartNum, gOscBanks[gCurrentOscBank]->lookupTableSize,
+							 gOscBanks[gCurrentOscBank]->oscillatorPhases, gOscBanks[gCurrentOscBank]->oscillatorNormFrequencies,
+							 gOscBanks[gCurrentOscBank]->oscillatorAmplitudes,
+							 gOscBanks[gCurrentOscBank]->oscillatorNormFreqDerivatives,
+							 gOscBanks[gCurrentOscBank]->oscillatorAmplitudeDerivatives,
+							 /*gOscBanks[gCurrentOscBank]->lookupTable*/gDynamicWavetable);
+
+		gOscillatorBufferWriteCurrentSize = gOscBanks[gCurrentOscBank]->hopCounter * 2;
+
+		/* Update the pitch right before the hop
+		 * Total CV range +/- N_OCT octaves
+		 */
+		float pitch = (float)gPitchLatestInput / octaveSplitter - N_OCT/2;
+		//gOscBanks[gCurrentOscBank]->pitchMultiplier = powf(2.0f, pitch);
+		gOscBanks[gCurrentOscBank]->pitchMultiplier = pow(2.0f, pitch);
+
+#ifdef FIXME_LATER // This doesn't work very well yet
+		gOscBanks[gCurrentOscBank]->filterNum = gSensor1LatestTouchCount;
+		float freqScaler = gOscBanks[gCurrentOscBank]->getFrequencyScaler();
+		for(int i=0; i < gOscBanks[gCurrentOscBank]->filterNum; i++)
+		{
+			// touch pos is linear but freqs are log
+			gOscBanks[gCurrentOscBank]->filterFreqs[i] = ((expf(gSensor1MatrixTouchPos[i]*4)-1)/(expf(4)-1))*gOscBanks[gCurrentOscBank]->filterMaxF*freqScaler;
+			gOscBanks[gCurrentOscBank]->filterQ[i] = gSensor1LatestTouchSizes[i];
+			if(gOscBanks[gCurrentOscBank]->filterFreqs[i]>500*freqScaler)
+				gOscBanks[gCurrentOscBank]->filterPadding[i] = 1+100000*( (gOscBanks[gCurrentOscBank]->filterFreqs[i]-500*freqScaler)/(gOscBanks[gCurrentOscBank]->filterMaxF-500)*freqScaler );
+			else
+				gOscBanks[gCurrentOscBank]->filterPadding[i] = 1;
+		}
+#endif
+
+		RTIME ticks		= rt_timer_read();
+		SRTIME ns		= rt_timer_tsc2ns(ticks);
+		SRTIME delta 	= ns-prevChangeNs;
+
+		// switch to next bank cannot be too frequent, to avoid seg fault! [for example sef fault happens when removing both VDD and GND from breadboard]
+		if(gNextOscBank != gCurrentOscBank && delta>100000000) {
+
+			/*printf("ticks %llu\n", (unsigned long long)ticks);
+			printf("ns %llu\n", (unsigned long long)ns);
+			printf("prevChangeNs %llu\n", (unsigned long long)prevChangeNs);
+			printf("-------------------------->%llud\n", (unsigned long long)(ns-prevChangeNs));*/
+
+			prevChangeNs = ns;
+			dbox_printf("Changing to bank %d...\n", gNextOscBank);
+			if(gOscBanks[gCurrentOscBank]->state==bank_playing){
+				gOscBanks[gCurrentOscBank]->stop();
+			}
+
+			gCurrentOscBank = gNextOscBank;
+			gOscBanks[gCurrentOscBank]->hopNumTh = 0;
+		}
+		else {
+			/* Advance to the next oscillator frame */
+			gOscBanks[gCurrentOscBank]->nextHop();
+		}
+	}
+}
+
+// Lower-priority render function which performs matrix calculations
+// State should be transferred in via global variables
+void render_low_prio()
+{
+	gPRU->setGPIOTestPin();
+	if(gDynamicWavetableNeedsRender) {
+		// Find amplitude of wavetable
+		float meanAmplitude = 0;
+		float sineMix;
+
+		for(int i = 0; i < gFeedbackOscillatorTableLength; i++) {
+			//meanAmplitude += fabsf(gFeedbackOscillatorTable[i]);
+			meanAmplitude += fabs(gFeedbackOscillatorTable[i]);
+		}
+		meanAmplitude /= (float)gFeedbackOscillatorTableLength;
+
+		if(meanAmplitude > 0.35)
+			sineMix = 0;
+		else
+			sineMix = (.35 - meanAmplitude) / .35;
+
+		//dbox_printf("amp %f mix %f\n", meanAmplitude, sineMix);
+
+		// Copy to main wavetable
+		wavetable_interpolate(gFeedbackOscillatorTableLength, gDynamicWavetableLength,
+				gFeedbackOscillatorTable, gDynamicWavetable,
+				gOscBanks[gCurrentOscBank]->lookupTable, sineMix);
+	}
+
+	if(gLoopPointMin >= 60000.0/65536.0 && gLoopPointMax >= 60000.0/65536.0) {
+		// KLUDGE!
+		if(gCurrentOscBank == 0)
+			gOscBanks[gCurrentOscBank]->setLoopHops(50, ((float)gOscBanks[gCurrentOscBank]->getLastHop() * 0.6) - 1);
+		else
+			gOscBanks[gCurrentOscBank]->setLoopHops(5, ((float)gOscBanks[gCurrentOscBank]->getLastHop() * 0.7) - 1);
+	}
+	else {
+		float normLoopPointMin = (float)gLoopPointMin * gOscBanks[gCurrentOscBank]->getLastHop();
+		float normLoopPointMax = (float)gLoopPointMax * gOscBanks[gCurrentOscBank]->getLastHop();
+
+		int intLoopPointMin = normLoopPointMin;
+		if(intLoopPointMin < 1)
+			intLoopPointMin = 1;
+		int intLoopPointMax = normLoopPointMax;
+		if(intLoopPointMax <= intLoopPointMin)
+			intLoopPointMax = intLoopPointMin + 1;
+		if(intLoopPointMax > gOscBanks[gCurrentOscBank]->getLastHop() - 1)
+			intLoopPointMax =  gOscBanks[gCurrentOscBank]->getLastHop() - 1;
+
+		//dbox_printf("Loop points %d-%d / %d-%d\n", gLoopPointMin, gLoopPointMax, intLoopPointMin, intLoopPointMax);
+
+		/* WORKS, jsut need to fix the glitch when jumps!
+		 * *int currentHop = gOscBanks[gCurrentOscBank]->getCurrentHop();
+		if(currentHop < intLoopPointMin -1 )
+			gOscBanks[gCurrentOscBank]->setJumpHop(intLoopPointMin + 1);
+		else if(currentHop > intLoopPointMax + 1)
+			gOscBanks[gCurrentOscBank]->setJumpHop(intLoopPointMax - 1);*/
+		gOscBanks[gCurrentOscBank]->setLoopHops(intLoopPointMin, intLoopPointMax);
+	}
+
+	if(gIsLoading)
+		gStatusLED.blink(25, 75);	// Blink quickly until load finished
+	else
+		gStatusLED.blink(250 / gOscBanks[gCurrentOscBank]->getSpeed(), 250 / gOscBanks[gCurrentOscBank]->getSpeed());
+	gPRU->clearGPIOTestPin();
+
+//	static int counter = 32;
+//	if(--counter == 0) {
+//		for(int i = 0; i < gLoopPointsInputBufferSize; i++) {
+//			dbox_printf("%d ", gLoopPointsInputBuffer[i]);
+//			if(i % 32 == 31)
+//				dbox_printf("\n");
+//		}
+//		dbox_printf("\n\n");
+//		counter = 32;
+//	}
+
+	//dbox_printf("min %d max %d\n", gLoopPointMin, gLoopPointMax);
+}
+
+// Clean up at the end of render
+void cleanup(BelaContext *context, void *userData)
+{
+	free(gOscillatorBuffer1);
+	free(gOscillatorBuffer2);
+	free(gDynamicWavetable);
+}
+
+// Interpolate one wavetable into another. The output size
+// does not include the guard point at the end which will be identical
+// to the first point
+void wavetable_interpolate(int numSamplesIn, int numSamplesOut,
+                           float *tableIn, float *tableOut,
+                           float *sineTable, float sineMix)
+{
+	float fractionalScaler = (float)numSamplesIn / (float)numSamplesOut;
+
+	for(int k = 0; k < numSamplesOut; k++) {
+		float fractionalIndex = (float) k * fractionalScaler;
+		//int sB = (int)floorf(fractionalIndex);
+		int sB = (int)floor(fractionalIndex);
+		int sA = sB + 1;
+		if(sA >= numSamplesIn)
+			sA = 0;
+		float fraction = fractionalIndex - sB;
+		tableOut[k] = fraction * tableIn[sA] + (1.0f - fraction) * tableIn[sB];
+		tableOut[k] = sineMix * sineTable[k] + (1.0 - sineMix) * tableOut[k];
+	}
+
+	tableOut[numSamplesOut] = tableOut[0];
+}
+
+// Create a hysteresis oscillator with a matrix input and output
+inline float hysteresis_oscillator(float input, float risingThreshold, float fallingThreshold, bool *rising)
+{
+	float value;
+
+	if(*rising) {
+		if(input > risingThreshold) {
+			*rising = false;
+			value = 0;
+		}
+		else
+			value = 1.0;
+	}
+	else {
+		if(input < fallingThreshold) {
+			*rising = true;
+			value = 1.0;
+		}
+		else
+			value = 0;
+	}
+
+	return value;
+}
+
+#ifdef DBOX_CAPE_TEST
+// Test the functionality of the D-Box cape by checking each input and output
+// Loopback cable from ADC to DAC needed
+void render_capetest(int numMatrixFrames, int numAudioFrames, float *audioIn, float *audioOut,
+			uint16_t *matrixIn, uint16_t *matrixOut)
+{
+	static float phase = 0.0;
+	static int sampleCounter = 0;
+	static int invertChannel = 0;
+
+	// Play a sine wave on the audio output
+	for(int n = 0; n < numAudioFrames; n++) {
+		audioOut[2*n] = audioOut[2*n + 1] = 0.5*sinf(phase);
+		phase += 2.0 * M_PI * 440.0 / 44100.0;
+		if(phase >= 2.0 * M_PI)
+			phase -= 2.0 * M_PI;
+	}
+
+	for(int n = 0; n < numMatrixFrames; n++) {
+		// Change outputs every 512 samples
+		if(sampleCounter < 512) {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel)
+					matrixOut[n*8 + k] = 50000;
+				else
+					matrixOut[n*8 + k] = 0;
+			}
+		}
+		else {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel)
+					matrixOut[n*8 + k] = 0;
+				else
+					matrixOut[n*8 + k] = 50000;
+			}
+		}
+
+		// Read after 256 samples: input should be low
+		if(sampleCounter == 256) {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel) {
+					if(matrixIn[n*8 + k] < 50000) {
+						dbox_printf("FAIL channel %d -- output HIGH input %d (inverted)\n", k, matrixIn[n*8 + k]);
+					}
+				}
+				else {
+					if(matrixIn[n*8 + k] > 2048) {
+						dbox_printf("FAIL channel %d -- output LOW input %d\n", k, matrixIn[n*8 + k]);
+					}
+				}
+			}
+		}
+		else if(sampleCounter == 768) {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel) {
+					if(matrixIn[n*8 + k] > 2048) {
+						dbox_printf("FAIL channel %d -- output LOW input %d (inverted)\n", k, matrixIn[n*8 + k]);
+					}
+				}
+				else {
+					if(matrixIn[n*8 + k] < 50000) {
+						dbox_printf("FAIL channel %d -- output HIGH input %d\n", k, matrixIn[n*8 + k]);
+					}
+				}
+			}
+		}
+
+		if(++sampleCounter >= 1024) {
+			sampleCounter = 0;
+			invertChannel++;
+			if(invertChannel >= 8)
+				invertChannel = 0;
+		}
+	}
+}
+#endif
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/sensors.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,642 @@
+/*
+ * sensors.cpp
+ *
+ *  Created on: May 28, 2014
+ *      Author: Victor Zappi
+ */
+
+#include <stdio.h>
+#include <pthread.h>
+#include <unistd.h>
+#include <math.h>
+#include <vector>
+#include "prio.h"
+#include "sensors.h"
+#include "OscillatorBank.h"
+#include "DboxSensors.h"
+
+
+//----------------------------------------
+// main extern variables
+//----------------------------------------
+extern vector<OscillatorBank*> gOscBanks;
+extern int gCurrentOscBank;
+extern int gNextOscBank;
+extern int gShouldStop;
+extern int gVerbose;
+
+float gSensor0LatestTouchPos = 0;	// most recent pitch touch location [0-1] on sensor 0, used by render.cpp
+int gSensor0LatestTouchNum	 = 0;	// most recent num of touches on sensor 0, used by render.cpp
+float gSensor1LatestTouchPos[5];	// most recent touche locations on sensor 1, used by render.cpp
+//float gSensor1LatestTouchSizes[5];
+int gSensor1LatestTouchCount;		// most recent number touches on sensor 1, used by render.cpp
+int gSensor1LatestTouchIndex = 0;	// index of last touch in gSensor1LatestTouchPos[5], used by render.cpp
+int gLastFSRValue = 1799;			// most recent fsr value, used by render.cpp
+
+
+DboxSensors Sensors;
+
+
+//----------------------------------------
+// var shared with logger
+//----------------------------------------
+int s0TouchNum	 	= 0;
+float s0Touches_[MAX_TOUCHES];
+float s0Size_[MAX_TOUCHES];
+int s0LastIndex;
+
+int s1TouchNum	 	= 0;
+float s1Touches_[MAX_TOUCHES];
+float s1Size_[MAX_TOUCHES];
+int s1LastIndex;
+
+int fsr				= 1799;
+
+
+
+using namespace std;
+
+int initSensorLoop(int sensorAddress0, int sensorAddress1, int sensorType)
+{
+	int tk0_bus			= 1;
+	int tk0_address		= sensorAddress0;
+	int tk1_bus			= 1;
+	int tk1_address		= sensorAddress1;
+	int tk_file			= 0;
+	int fsr_max			= 1799;
+	int fsr_pinNum		= 4;
+
+	if(gVerbose==1)
+		cout << "---------------->Init Control Thread" << endl;
+
+	if(Sensors.initSensors(tk0_bus, tk0_address, tk1_bus, tk1_address, tk_file, fsr_pinNum, fsr_max, sensorType)>0)
+	{
+		gShouldStop = 1;
+		cout << "control cannot start" << endl;
+		return -1;
+	}
+
+	for(int i=0; i<MAX_TOUCHES; i++)
+	{
+		s0Touches_[i]	= 0.0;
+		s0Size_[i]		= 0.0;
+
+		s1Touches_[i]	= 0.0;
+		s1Size_[i]	= 0.0;
+	}
+
+	return 0;
+}
+
+void sensorLoop(void *)
+{
+	timeval start, end;
+	unsigned long elapsedTime;
+	//float touchSize		= 0;	// once used for timbre
+
+
+
+	float *s0Touches;
+	float *s0Size;
+	int s0PrevTouchNum 	= 0;
+	int s0SortedTouchIndices[MAX_TOUCHES];
+	float s0SortedTouches[MAX_TOUCHES];
+	float s0PrevSortedTouches[MAX_TOUCHES];
+
+	float *s1Touches;
+	float *s1Size;
+	int s1PrevTouchNum 	= 0;
+	int s1SortedTouchIndices[MAX_TOUCHES];
+	float s1SortedTouches[MAX_TOUCHES];
+	float s1PrevSortedTouches[MAX_TOUCHES];
+
+	float freqScaler	= 0;
+	int fsrMin			= 0;//50; // was 200
+	int fsrMax			= 1799;//1300; // was 800
+	float vel			= 0;
+	float prevVel		= 0;
+	float filterMaxF	= 0;
+	if(gVerbose==1)
+		dbox_printf("__________set Control Thread priority\n");
+
+	if(gVerbose==1)
+		dbox_printf("_________________Control Thread!\n");
+
+	// get freq scaler, cos freqs must be scaled according to the wavetable used in the oscillator bank
+	freqScaler 		= gOscBanks[gCurrentOscBank]->getFrequencyScaler();
+	filterMaxF		= gOscBanks[gCurrentOscBank]->filterMaxF;
+
+	// init time vals
+	gettimeofday(&start, NULL);
+
+	// here we go, sensor loop until the end of the application
+	while(!gShouldStop)
+	{
+		gettimeofday(&end, NULL);
+		elapsedTime = ( (end.tv_sec*1000000+end.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
+		if( elapsedTime<4000 )
+			usleep(4000-elapsedTime);
+		else
+			dbox_printf("%d\n", (int)elapsedTime); // this print happens when something's gone bad...
+
+		if(Sensors.readSensors()==0)
+		{
+			s0TouchNum	= Sensors.getTKTouchCount(0);
+			s0Touches	= Sensors.getTKXPositions(0);
+			s0Size 		= Sensors.getTKTouchSize(0);
+
+			s1TouchNum	= Sensors.getTKTouchCount(1);
+			s1Touches	= Sensors.getTKXPositions(1);
+			s1Size 		= Sensors.getTKTouchSize(1);
+
+			for(int i=0; i<MAX_TOUCHES; i++)
+			{
+				s0Touches_[i]	= s0Touches[i];
+				s0Size_[i]		= s0Size[i];
+
+				s1Touches_[i]	= s1Touches[i];
+				s1Size_[i]		= s1Size[i];
+			}
+
+			gSensor0LatestTouchNum	= s0TouchNum;
+			if(s0TouchNum > 0)
+			{
+				//-----------------------------------------------------------------------------------
+				// timbre, speed and  pitch
+				//touchSize	 = 0;	\\ once used for timbre
+
+				// if we have a number of touches different from previous round, track their order of arrival [calculated using distance comparison]
+				if(s0PrevTouchNum!=s0TouchNum)
+				{
+					float distances[MAX_TOUCHES*(MAX_TOUCHES-1)]; // maximum number of current+previous touches between rounds with different num of touches
+					int ids[MAX_TOUCHES*(MAX_TOUCHES-1)];
+					// calculate all distance permutations between previous and current touches
+					for(int i=0; i<s0TouchNum; i++)
+					{
+						for(int p=0; p<s0PrevTouchNum; p++)
+						{
+							int index			= i*s0PrevTouchNum+p;	// permutation id [says between which touches we are calculating distance]
+							distances[index]	= fabs(s0Touches[i]-s0PrevSortedTouches[p]);
+							ids[index]			= index;
+							if(index>0)
+							{
+								// sort, from min to max distance
+								float tmp;
+								while(distances[index]<distances[index-1])
+								{
+									tmp				= ids[index-1];
+									ids[index-1]	= ids[index];
+									ids[index]		= tmp;
+
+									tmp				= distances[index-1];
+									distances[index-1] = distances[index];
+									distances[index] = tmp;
+
+									index--;
+
+									if(index == 0)
+										break;
+								}
+							}
+						}
+					}
+
+					int sorted = 0;
+					bool currAssigned[MAX_TOUCHES] = {false};
+					bool prevAssigned[MAX_TOUCHES] = {false};
+
+					// track touches assigning index according to shortest distance
+					for(int i=0; i<s0TouchNum*s0PrevTouchNum; i++)
+					{
+						int currentIndex	= ids[i]/s0PrevTouchNum;
+						int prevIndex		= ids[i]%s0PrevTouchNum;
+						// avoid double assignment
+						if(!currAssigned[currentIndex] && !prevAssigned[prevIndex])
+						{
+							currAssigned[currentIndex]	= true;
+							prevAssigned[prevIndex]		= true;
+							s0SortedTouchIndices[currentIndex] = prevIndex;
+							sorted++;
+						}
+					}
+					// we still have to assign a free index to new touches
+					if(s0PrevTouchNum<s0TouchNum)
+					{
+						for(int i=0; i<s0TouchNum; i++)
+						{
+							if(!currAssigned[i])
+								s0SortedTouchIndices[i] = sorted++; // assign next free index
+
+							// update tracked value
+							s0SortedTouches[s0SortedTouchIndices[i]] = s0Touches[i];
+							s0PrevSortedTouches[i]			         = s0SortedTouches[i];
+							if(s0SortedTouchIndices[i]==s0TouchNum-1)
+								s0LastIndex = i;
+
+							// accumulate sizes for timbre
+							//touchSize += s0Size[i];
+						}
+					}
+					else // some touches have disappeared...
+					{
+						// ...we have to shift all indices...
+						for(int i=s0PrevTouchNum-1; i>=0; i--)
+						{
+							if(!prevAssigned[i])
+							{
+								for(int j=0; j<s0TouchNum; j++)
+								{
+									// ...only if touches that disappeared were before the current one
+									if(s0SortedTouchIndices[j]>i)
+										s0SortedTouchIndices[j]--;
+								}
+							}
+						}
+						// done! now update
+						for(int i=0; i<s0TouchNum; i++)
+						{
+							// update tracked value
+							s0SortedTouches[s0SortedTouchIndices[i]] = s0Touches[i];
+							s0PrevSortedTouches[i]			         = s0SortedTouches[i];
+							if(s0SortedTouchIndices[i]==s0TouchNum-1)
+								s0LastIndex = i;
+
+							// accumulate sizes for timbre
+							//touchSize += s0Size[i];
+						}
+					}
+				}
+				else // nothing's changed since last round
+				{
+					for(int i=0; i<s0TouchNum; i++)
+					{
+						// update tracked value
+						s0SortedTouches[s0SortedTouchIndices[i]] = s0Touches[i];
+						s0PrevSortedTouches[i]			         = s0SortedTouches[i];
+
+						// accumulate sizes for timbre
+						//touchSize += s0Size[i];
+					}
+				}
+
+				if(s0TouchNum == 0)
+					s0LastIndex = -1;
+
+				// timbre
+				//touchSize = (touchSize > 0.7) ? 1 : touchSize/0.7;
+				//gOscBanks[gCurrentOscBank]->hopNumTh = log((1-touchSize)+1)/log(2)*20000;
+				//gOscBanks[gCurrentOscBank]->hopNumTh = 0;
+
+
+				// pitch, controlled by last touch
+				//prevTouchPos 				= touch[touchIndex];
+				//touchPos	 			 	= (s0SortedTouches[s0TouchNum-1]-0.5)/0.5;	// from [0,1] to [-1,1]
+				gSensor0LatestTouchPos      = s0SortedTouches[s0TouchNum-1];
+				//touchPos					= s0Touches[0];
+				//gOscBanks[gCurrentOscBank]->pitchMultiplier 	= pow(2, touchPos);
+				//-----------------------------------------------------------------------------------
+
+
+
+				//-----------------------------------------------------------------------------------
+				// note on
+				//if(s0PrevTouchNum == 0)
+				//	gOscBanks[gCurrentOscBank]->play();
+				// fsr = Sensors.getFSRVAlue();
+				fsr = gLastFSRValue;
+				//dbox_printf("fsr: %d\n", fsr);
+				if(!gOscBanks[gCurrentOscBank]->note)
+				{
+					vel = fsr;
+					vel /= (float)(fsrMax-fsrMin);
+
+					vel = 1-vel;
+					dbox_printf("Attack vel: %f\n", vel);
+					gOscBanks[gCurrentOscBank]->play(vel);
+					prevVel = vel;
+				}
+				else if(gOscBanks[gCurrentOscBank]->getEnvelopeState() != env_release)
+				{
+					fsr = (fsr > fsrMax) ? fsrMax : fsr;
+					vel = (fsr < fsrMin) ? fsrMin : fsr;
+					vel -= fsrMin;
+					vel /= (float)(fsrMax-fsrMin);
+					vel = 1-vel;
+					if(vel > prevVel)
+					{
+						gOscBanks[gCurrentOscBank]->afterTouch(vel);
+						prevVel = vel;
+					}
+				}
+				//-----------------------------------------------------------------------------------
+			}
+			else
+			{
+				//prevFsr = 1799;
+				//prevTouchPos = -1;
+				//-----------------------------------------------------------------------------------
+				// note off
+				if(s0PrevTouchNum > 0)
+				{
+					if(gOscBanks[gCurrentOscBank]->state==bank_playing)
+						gOscBanks[gCurrentOscBank]->stop();
+				}
+				//-----------------------------------------------------------------------------------
+			}
+
+
+
+			// sensor 2
+			//-----------------------------------------------------------------------------------
+			//filter - calculated even when no touches on first sensor, to filter also release tail
+			gOscBanks[gCurrentOscBank]->filterNum	= s1TouchNum;
+
+			gSensor1LatestTouchCount = gOscBanks[gCurrentOscBank]->filterNum;
+			for(int i = 0; i < gSensor1LatestTouchCount; i++) {
+				gSensor1LatestTouchPos[i] = s1Touches[i];
+				//gSensor1LatestTouchSizes[i] = s1Size[i];
+			}
+
+/*			for(int i=0; i<gOscBanks[gCurrentOscBank]->filterNum; i++)
+			{
+				// touch pos is linear but freqs are log
+				gOscBanks[gCurrentOscBank]->filterFreqs[i] = ((exp(s0Touches[i]*4)-1)/(exp(4)-1))*filterMaxF*freqScaler;
+				//gOscBanks[gCurrentOscBank]->filterQ[i] = size[i]*5*(1+touch[i]*1000)*freqScaler;
+				gOscBanks[gCurrentOscBank]->filterQ[i] = s0Size[i];
+				if(gOscBanks[gCurrentOscBank]->filterFreqs[i]>500*freqScaler)
+					gOscBanks[gCurrentOscBank]->filterPadding[i] = 1+100000*( (gOscBanks[gCurrentOscBank]->filterFreqs[i]-500*freqScaler)/(filterMaxF-500)*freqScaler );
+				else
+					gOscBanks[gCurrentOscBank]->filterPadding[i] = 1;
+			}*/
+
+			// each touch on sensor 2 is a notch filter, whose Q is determined by touch size
+			for(int i=0; i<gOscBanks[gCurrentOscBank]->filterNum; i++)
+			{
+				// map touch pos [which is linear] on freqs exponentially
+				float freq = ((exp(s1Touches[i]*4)-1)/EXP_DENOM)*filterMaxF;
+				gOscBanks[gCurrentOscBank]->filterFreqs[i] = freq*freqScaler;
+				// also size is mapped exponentially on Q
+				float siz = (exp(s1Size[i])-1)/1.71828;
+				gOscBanks[gCurrentOscBank]->filterQ[i] = siz*( (filterMaxF-freq)/filterMaxF * 0.9 + 0.1 );	// size weight on Q decreases with frequency
+			}
+			//-----------------------------------------------------------------------------------
+
+
+
+			//-----------------------------------------------------------------------------------
+			// sort touches on sensor 2
+			if(s1TouchNum > 0)
+			{
+				// if we have a number of touches different from previous round, track their order of arrival [calculated using distance comparison]
+				if(s1PrevTouchNum!=s1TouchNum)
+				{
+					float distances[MAX_TOUCHES*(MAX_TOUCHES-1)]; // maximum number of current+previous touches between rounds with different num of touches
+					int ids[MAX_TOUCHES*(MAX_TOUCHES-1)];
+					// calculate all distance permutations between previous and current touches
+					for(int i=0; i<s1TouchNum; i++)
+					{
+						for(int p=0; p<s1PrevTouchNum; p++)
+						{
+							int index 			= i*s1PrevTouchNum+p;	// permutation id [says between which touches we are calculating distance]
+							distances[index]	= fabs(s1Touches[i]-s1PrevSortedTouches[p]);
+							ids[index]			= index;
+							if(index>0)
+							{
+								// sort, from min to max distance
+								float tmp;
+								while(distances[index]<distances[index-1])
+								{
+									tmp 				= ids[index-1];
+									ids[index-1] 		= ids[index];
+									ids[index] 			= tmp;
+
+									tmp					= distances[index-1];
+									distances[index-1]	= distances[index];
+									distances[index] 	= tmp;
+
+									index--;
+
+									if(index == 0)
+										break;
+								}
+							}
+						}
+					}
+
+					int sorted = 0;
+					bool currAssigned[MAX_TOUCHES] = {false};
+					bool prevAssigned[MAX_TOUCHES] = {false};
+
+					// track touches assigning index according to shortest distance
+					for(int i=0; i<s1TouchNum*s1PrevTouchNum; i++)
+					{
+						int currentIndex	= ids[i]/s1PrevTouchNum;
+						int prevIndex		= ids[i]%s1PrevTouchNum;
+						// avoid double assignment
+						if(!currAssigned[currentIndex] && !prevAssigned[prevIndex])
+						{
+							currAssigned[currentIndex]			= true;
+							prevAssigned[prevIndex]				= true;
+							s1SortedTouchIndices[currentIndex] = prevIndex;
+							sorted++;
+						}
+					}
+					// we still have to assign a free index to new touches
+					if(s1PrevTouchNum<s1TouchNum)
+					{
+						for(int i=0; i<s1TouchNum; i++)
+						{
+							if(!currAssigned[i])
+								s1SortedTouchIndices[i] = sorted++; // assign next free index
+
+							// update tracked value
+							s1SortedTouches[s1SortedTouchIndices[i]] = s1Touches[i];
+							s1PrevSortedTouches[i]			       	 = s1SortedTouches[i];
+							if(s1SortedTouchIndices[i]==s1TouchNum-1)
+								s1LastIndex = i;
+						}
+					}
+					else // some touches have disappeared...
+					{
+						// ...we have to shift all indices...
+						for(int i=s1PrevTouchNum-1; i>=0; i--)
+						{
+							if(!prevAssigned[i])
+							{
+								for(int j=0; j<s1TouchNum; j++)
+								{
+									// ...only if touches that disappeared were before the current one
+									if(s1SortedTouchIndices[j]>i)
+										s1SortedTouchIndices[j]--;
+								}
+							}
+						}
+						// done! now update
+						for(int i=0; i<s1TouchNum; i++)
+						{
+							// update tracked value
+							s1SortedTouches[s1SortedTouchIndices[i]] = s1Touches[i];
+							s1PrevSortedTouches[i]			       	 = s1SortedTouches[i];
+							if(s1SortedTouchIndices[i]==s1TouchNum-1)
+								s1LastIndex = i;
+						}
+					}
+				}
+				else // nothing's changed since last round
+				{
+					for(int i=0; i<s1TouchNum; i++)
+					{
+						// update tracked value
+						s1SortedTouches[s1SortedTouchIndices[i]] = s1Touches[i];
+						s1PrevSortedTouches[i]			       	 = s1SortedTouches[i];
+					}
+				}
+			}
+
+			if(s1TouchNum > 0)
+			{
+				gSensor1LatestTouchIndex = s1LastIndex;
+			}
+			else
+				s1LastIndex = -1;
+
+/*			dbox_printf("-----------------------------\nnum: %d, latest: %d\n", s1TouchNum, gSensor1LatestTouchIndex);
+			for(int i=0; i<s1TouchNum; i++)
+				dbox_printf("\t%f\n", gSensor1LatestTouchPos[i]);
+			dbox_printf("------\n");
+			for(int i=0; i<s1TouchNum; i++)
+				dbox_printf("\t%f\n", s1SortedTouches[i]);*/
+
+
+
+			// update variables for both sensors
+			s0PrevTouchNum	= s0TouchNum;
+			s1PrevTouchNum	= s1TouchNum;
+		}
+		else
+			dbox_printf("Come on instrument!\n");	//break
+
+		gettimeofday(&start, NULL);
+	}
+
+	dbox_printf("sensor thread ended\n");
+}
+
+void *keyboardLoop(void *)
+{
+	if(gVerbose==1)
+		cout << "_________________Keyboard Control Thread!" << endl;
+
+	char keyStroke = '.';
+	cout << "Press q to quit." << endl;
+
+	float speed;
+
+	do
+	{
+		keyStroke =	getchar();
+		while(getchar()!='\n'); // to read the first stroke
+
+		switch (keyStroke)
+		{
+			//----------------------------------------------------------------------------
+			case 'a':
+				gOscBanks[gCurrentOscBank]->hopNumTh = 0;
+				gOscBanks[gCurrentOscBank]->play(1);
+				//cout << "Note on" << endl;
+				break;
+			case 's':
+				if(gOscBanks[gCurrentOscBank]->state==bank_playing)
+				{
+					gOscBanks[gCurrentOscBank]->stop();
+					//cout << "Note off" << endl;
+				}
+				break;
+			//----------------------------------------------------------------------------
+			case '[':
+				gOscBanks[gCurrentOscBank]->freqMovement-=0.05;
+				if(gOscBanks[gCurrentOscBank]->freqMovement<0)
+					gOscBanks[gCurrentOscBank]->freqMovement = 0;
+				//cout << "gOscBanks[gCurrentOscBank]->FreqMov: " << gOscBanks[gCurrentOscBank]->freqMovement << endl;
+				break;
+			case ']':
+				gOscBanks[gCurrentOscBank]->freqMovement+=0.05;
+				if(gOscBanks[gCurrentOscBank]->freqMovement>1)
+					gOscBanks[gCurrentOscBank]->freqMovement = 1;
+				//cout << "gOscBanks[gCurrentOscBank]->FreqMov: " << gOscBanks[gCurrentOscBank]->freqMovement << endl;
+				break;
+			//----------------------------------------------------------------------------
+			case '<':
+				speed = gOscBanks[gCurrentOscBank]->getSpeed() - 0.1 ;
+				gOscBanks[gCurrentOscBank]->setSpeed(speed);
+				dbox_printf("Speed: %f\n", speed);
+
+				break;
+			case '>':
+				speed = gOscBanks[gCurrentOscBank]->getSpeed() + 0.1 ;
+				gOscBanks[gCurrentOscBank]->setSpeed(speed);
+				dbox_printf("Speed: %f\n", speed);
+				break;
+			case '0':
+				speed = 0.1;
+				gOscBanks[gCurrentOscBank]->setSpeed(speed);
+				dbox_printf("Speed: %f\n", speed);
+				break;
+			case '1':
+				speed = 0.5;
+				gOscBanks[gCurrentOscBank]->setSpeed(speed);
+				dbox_printf("Speed: %f\n", speed);
+				break;
+			case '2':
+				speed = 1;
+				gOscBanks[gCurrentOscBank]->setSpeed(speed);
+				dbox_printf("Speed: %f\n", speed);
+				break;
+			case '3':
+				speed = 2;
+				gOscBanks[gCurrentOscBank]->setSpeed(speed);
+				dbox_printf("Speed: %f\n", speed);
+				break;
+			case '4':
+				speed = 3;
+				gOscBanks[gCurrentOscBank]->setSpeed(speed);
+				dbox_printf("Speed: %f\n", speed);
+				break;
+			//----------------------------------------------------------------------------
+			case 'z':
+				gOscBanks[gCurrentOscBank]->setJumpHop(0);
+				break;
+			case 'x':
+				gOscBanks[gCurrentOscBank]->setJumpHop(100);
+				break;
+			case 'c':
+				gOscBanks[gCurrentOscBank]->setJumpHop(600);
+				break;
+			case 'v':
+				gOscBanks[gCurrentOscBank]->setJumpHop(1100);
+				break;
+			case 'b':
+				gOscBanks[gCurrentOscBank]->setJumpHop(2000);
+				break;
+			case 'n':
+				gOscBanks[gCurrentOscBank]->setJumpHop(gOscBanks[gCurrentOscBank]->getLastHop());
+				break;
+			//----------------------------------------------------------------------------
+			case 'q':
+				gShouldStop = true;
+				break;
+			case 'o':
+				gNextOscBank = (gCurrentOscBank + 1) % gOscBanks.size();
+				break;
+			default:
+				break;
+			//----------------------------------------------------------------------------
+		}
+		usleep(1000); /* Wait 1ms to avoid checking too quickly */
+	}
+	while (keyStroke!='q');
+
+	cout << "keyboard thread ended" << endl;
+
+	return (void *)0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/sensors.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,19 @@
+/*
+ * sensors.h
+ *
+ *  Created on: May 28, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef SENSORS_H_
+#define SENSORS_H_
+
+#include "config.h"
+
+int initSensorLoop(int sensorAddress0, int sensorAddress1, int sensorType);
+
+void sensorLoop(void *);
+void *keyboardLoop(void *);
+
+
+#endif /* SENSORS_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/spear_parser.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,641 @@
+/*
+ * spear_parser.cpp v1.2
+ *
+ *  Created on: May 6, 2014
+ *      Author: Victor Zappi
+ */
+
+#include "spear_parser.h"
+
+using namespace std;
+
+//#define DO_CHECKS
+
+//------------------------------------------------------------------------------------------------
+// partials
+//------------------------------------------------------------------------------------------------
+
+Partials::Partials()
+{
+	partialFrequencies	= NULL;
+//	partialAmplitudes	= NULL;
+//	partialNumFrames	= NULL;
+//	partialStartSample	= NULL;
+//	partialEndSample	= NULL;
+//	partialCurrentFrame	= NULL;
+//	partialFreqDelta	= NULL;
+//	partialAmpDelta	= NULL;
+
+
+	activePartialNum	= NULL;
+//	activePartials		= NULL;
+
+	currentSample = -1;
+}
+
+Partials::~Partials()
+{
+	if(partialFrequencies != NULL)			// check on one is enough
+	{
+		if(partialFrequencies[0] != NULL)	// check on one is enough
+		{
+			for(unsigned int i=0; i<parNum; i++)
+			{
+				delete[] partialFrequencies[i];
+				delete[] partialAmplitudes[i];
+				delete[] partialFreqDelta[i];
+				delete[] partialAmpDelta[i];
+
+			}
+		}
+
+		delete[] partialFrequencies;
+		delete[] partialAmplitudes;
+		delete[] partialNumFrames;
+		delete[] partialFreqDelta;
+		delete[] partialAmpDelta;
+		delete[] partialFreqMean;
+	}
+
+	if(activePartialNum != NULL)
+	{
+		for(unsigned int i=0; i<hopNum+1; i++)
+			delete[] activePartials[i];
+
+		delete[] activePartialNum;
+		delete[] activePartials ;
+	}
+}
+
+void Partials::init(int parN, int hopS, bool isDBX)
+{
+	if(!isDBX)
+	{
+		parNum	= parN;
+		hopSize	= hopS;
+
+		partialFrequencies	= new float *[parNum];
+		partialAmplitudes	= new float *[parNum];
+		partialNumFrames	= new unsigned int[parNum];
+		partialStartFrame	= new unsigned int[parNum];
+		partialStartSample	= new unsigned int[parNum];
+		partialEndSample	= new unsigned int[parNum];
+		partialFreqDelta	= new float *[parNum];
+		partialAmpDelta		= new float *[parNum];
+		partialFreqMean		= new float[parNum];
+
+
+
+		// init in one shot
+		fill(partialFreqMean, partialFreqMean+parNum, 0);			// mean is zero
+
+		partialFrequencies[0] 	= NULL;								// for free check
+	}
+	else
+	{
+		parNum	= parN;
+		hopSize	= hopS;
+
+		partialFrequencies	= new float *[parNum];
+		partialAmplitudes	= new float *[parNum];
+		partialNumFrames	= new unsigned int[parNum];
+		partialStartFrame	= new unsigned int[parNum];
+		partialFreqDelta	= new float *[parNum];
+		partialAmpDelta		= new float *[parNum];
+		partialFreqMean		= new float[parNum];
+
+		partialFrequencies[0] 	= NULL;								// for free check
+	}
+}
+
+
+
+void Partials::update(int parIndex, int frameNum)
+{
+	partialFrequencies[parIndex] = new float[frameNum];
+	partialAmplitudes[parIndex]	 = new float[frameNum];
+	partialFreqDelta[parIndex]	 = new float[frameNum];
+	partialAmpDelta[parIndex]	 = new float[frameNum];
+
+	fill(partialFreqDelta[parIndex], partialFreqDelta[parIndex]+frameNum, 99999.0);	// in the end, only the last one will have 99999
+	fill(partialAmpDelta[parIndex], partialAmpDelta[parIndex]+frameNum, 99999.0);	// in the end, only the last one will have 99999
+}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+//------------------------------------------------------------------------------------------------
+// spear parser
+//------------------------------------------------------------------------------------------------
+Spear_parser::Spear_parser()
+{
+	// some default values
+	hopSize			= -1;
+	fileSampleRate	= -1;
+}
+
+Spear_parser::~Spear_parser()
+{
+}
+
+void Spear_parser::calculateHopSize(char *filename)
+{
+	int index 		= 0;
+	bool prevWas_ 	= false;
+	bool found_h	= false;
+	int n 			= 0;
+
+	hopSize 		= 0;
+
+	do
+	{
+		// check if '_'
+		if(filename[index] == '_')
+			prevWas_ = true;
+		else if( (filename[index] == 'h') && prevWas_) // if it is not, but it is 'h' and previous was '_', found "_h"!
+		{
+			found_h = true;
+			while(filename[index] != '\0')
+			{
+				index++;
+				if( (filename[index] == '.') || (filename[index] == '_'))
+					break;
+				else // i am not checking if char are digits...!
+				{
+					n = filename[index];
+					hopSize =  hopSize*10+(n-48);
+				}
+			}
+		}
+		else	// else, nothing
+			prevWas_ = false;
+		index++;
+	}
+	while( (filename[index] != '\0') && !found_h );
+
+	if( !found_h || (hopSize<1) )
+		hopSize = 551;	// default val
+
+}
+
+
+bool Spear_parser::parser(char *filename, int hopsize, int samplerate)
+{
+	string name = string(filename);
+	int len		= name.length();
+	// invoke correct parser according to the type of file...just checking the extension, crude but functional
+	if( (name[len-4]=='.') && (name[len-3]=='d') && (name[len-2]=='b') && (name[len-1]=='x') )
+		return DBXparser(filename, samplerate);				// .dbox
+	else
+		return TXTparser(filename, hopSize, samplerate);	// .txt, or whatever
+}
+
+
+bool Spear_parser::DBXparser(char *filename, int samplerate)
+{
+	fileSampleRate 	= samplerate;
+
+	// working vars
+	int parNum		= 0;	// total num of partials
+	int hopNum		= 0;	// total num of hops
+
+	//----------------------------------------------------------------------------------------
+	// open a file
+	ifstream fin;
+	fin.open(filename, ios::in | ios::binary);
+	if (!fin.good())
+	{
+		cout << "Parser Error: file not found" << endl;	// exit if file not found
+		return false;
+	}
+
+	gettimeofday(&start, NULL);
+	//----------------------------------------------------------------------------------------
+	// general data
+
+	// look for partial count
+	fin.read((char *) &parNum, sizeof(int));
+	partials.parNum 		= parNum;
+
+	// look for hop count
+	fin.read((char *) &hopNum, sizeof(int));
+	partials.setHopNum(hopNum);
+
+	// look for hop size
+	fin.read((char *) &hopSize, sizeof(int));
+	partials.hopSize 		= hopSize;		// it's handy for both classes to know it
+
+	// init partials data structure
+	partials.init(parNum, hopSize, true);
+
+	// look for max active par num
+	fin.read((char *) &(partials.maxActiveParNum), sizeof(int));
+
+
+
+	// partial data
+
+	// start frame of each partial
+	fin.read((char *) partials.partialStartFrame, sizeof(int)*parNum);
+
+	// num of frames of each partial
+	fin.read((char *) partials.partialNumFrames, sizeof(int)*parNum);
+
+	// frequency mean of each partial
+	fin.read((char *) partials.partialFreqMean, sizeof(int)*parNum);
+
+	for(int par=0; par<parNum; par++)
+	{
+		int frameNum = partials.partialNumFrames[par];
+		partials.update(par, frameNum);
+		fin.read((char *)partials.partialAmplitudes[par], sizeof(float)*frameNum);		// amplitude of each partial in each frame
+		fin.read((char *)partials.partialFrequencies[par], sizeof(float)*frameNum);		// frequency of each partial in each frame
+		fin.read((char *)partials.partialAmpDelta[par], sizeof(float)*frameNum);			// amplitude delta of each partial in each frame
+		fin.read((char *)partials.partialFreqDelta[par], sizeof(float)*frameNum);			// frequency delta of each partial in each frame
+	}
+
+
+
+
+	// frame data
+
+	// number of active partial per each frame
+	fin.read((char *) partials.activePartialNum, sizeof(short)*(hopNum+1));
+	// init array
+	for(int frame=0; frame<hopNum+1; frame++)
+	{
+		partials.activePartials[frame] = new unsigned int[partials.activePartialNum[frame]];
+		fin.read((char *)partials.activePartials[frame], sizeof(int)*partials.activePartialNum[frame]);			// active partials per each frame
+	}
+
+
+
+
+
+	gettimeofday(&stop, NULL);
+	parserT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
+
+
+	printf("\n-----------------------\n");
+	printf("\nFile: %s\n", filename);
+	printf("\n-----------------------\n");
+	printf("Profiler\n");
+	printf("-----------------------\n");
+	printf("File parser:\t\t\t%lu usec\n", parserT);
+	printf("\n\nTotal:\t\t%lu usec\n", parserT);
+	printf("-----------------------\n");
+
+	fin.close();
+
+	return true;
+}
+
+
+
+
+bool Spear_parser::TXTparser(char *filename, int hopsize, int samplerate)
+{
+	hopSize 		= hopsize;
+	fileSampleRate 	= samplerate;
+	if(hopsize<0)
+	{
+		gettimeofday(&start, NULL);
+		calculateHopSize(filename);
+		gettimeofday(&stop, NULL);
+		hopSizeT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
+	}
+	else
+		hopSizeT = 0;
+
+	calculateDeltaTime();
+
+	// working vars
+	char * token;			// where to save single figures from file
+	string s		= "";	// where to save lines from file
+	int parNum		= 0;	// total num of partials
+	int parIndex	= -1;	// index of current partial
+	int frameNum	= 0;	// total num of frames
+	int frameIndex	= -1;	// index of current frame
+	int startSample	= -1;	// sample value for first frame of partials
+	int endSample	= -1;	// sample value for last frame of partials
+	int maxSample	= 0;	// to calculate total number of hops in file
+	int missSampCnt = 0;	// number of mising samples
+	double freq		= 0;	// to calculate frequency delta
+	double prevFreq	= 0;	// to calculate frequency delta
+	double amp		= 0;	// to calculate amplitude delta
+	double prevAmp	= 0;	// to calculate amplitude delta
+
+
+	//----------------------------------------------------------------------------------------
+	// open a file
+	ifstream fin;
+	fin.open(filename);
+	if (!fin.good())
+	{
+		cout << "Parser Error: file not found" << endl;	// exit if file not found
+		return false;
+	}
+
+	gettimeofday(&start, NULL);
+	//----------------------------------------------------------------------------------------
+	// init partials data structure
+	getline(fin, s);
+	getline(fin, s);
+	getline(fin, s);	// third line is the first we are interested into
+
+	// look for partial count
+	token = strtok((char *)s.c_str(), " ");
+	// check if first token is there
+	if(token)
+	{
+		token = strtok(0, " ");
+		// check if second token is there
+		if(token)
+			parNum = atoi(token);
+		#ifdef DO_CHECKS
+		else
+		{
+			cout << "Parser Error: partial count not found, bad file format" << endl;	// exit if value not found
+			return false;
+		}
+		#endif
+	}
+	#ifdef DO_CHECKS
+	else
+	{
+		cout << "Parser Error: partial count not found, bad file format" << endl;		// exit if value not found
+		return false;
+	}
+	#endif
+	// from now on we take for granted that format is correct
+
+	// init partials data structure
+	partials.init(parNum, hopSize);
+
+	//----------------------------------------------------------------------------------------
+	// fill in partials data structure
+	getline(fin, s);		// get rid of intro line "partials-data"
+	getline(fin, s);		// first important line
+
+	while (!fin.eof())
+	{
+		//-------------------------------------
+		// partial specific info
+		token		= strtok((char *)s.c_str(), " ");
+		parIndex	= atoi(token);						// partial index
+
+		token		= strtok(0, " ");					// num of frames, not used, cos we will do linear interpolation for missing frames
+//		frameNum	= atoi(token);
+//		partials.partialNumFrames[parIndex]	= frameNum;
+
+		token		= strtok(0, " ");					// time of first frame, still char *
+		startSample = fromTimeToSamples(atof(token));	// convert time to samples
+		partials.partialStartSample[parIndex]	= startSample;
+
+		token		= strtok(0, " ");					// time of last frame, still char *
+		endSample	= fromTimeToSamples(atof(token)); 	// convert time to samples
+		partials.partialEndSample[parIndex]		= endSample;
+
+		frameNum	= ((endSample-startSample)/hopSize) + 1;	// num of frames, including missing consecutive ones [+1 one cos we count frames, not hops]
+		partials.partialNumFrames[parIndex]		= frameNum;
+
+
+		// check if this is the highest sample value so far
+		if(endSample > maxSample)
+			maxSample = endSample;
+
+		// update data structure
+		partials.update(parIndex, frameNum);
+
+
+		//-------------------------------------
+		// frames
+		getline(fin, s);
+		token		= strtok((char *)s.c_str(), " ");	// frame time
+		frameIndex	= -1;
+
+		// unroll first iteration, so that in the following loop we save the check on the last frame to calculate increments
+		if(token)						// all frames data are on one line, in groups of 3 entries
+		{
+			frameIndex++;
+
+			endSample	= fromTimeToSamples(atof(token));
+
+			token		= strtok(0, " ");	// frame frequency
+			prevFreq	= atof(token);
+			partials.partialFrequencies[parIndex][frameIndex]	= (float)prevFreq;
+			partials.partialFreqMean[parIndex] 					+= prevFreq;		// for frequency mean
+
+			token	 	= strtok(0, " ");	// frame amplitude
+			prevAmp  	= atof(token);
+			partials.partialAmplitudes[parIndex][frameIndex]	= (float)prevAmp;
+
+			token 		= strtok(0, " ");	// next frame frequency, to be checked
+		}
+
+		// here the loop starts
+		while(token)						// all frames data are on one line, in groups of 3 entries
+		{
+			frameIndex++;
+			missSampCnt 	= 0;
+
+			startSample		= fromTimeToSamples(atof(token));
+
+			token			= strtok(0, " ");	// frame frequency
+			freq			= atof(token);
+
+			token			= strtok(0, " ");	// frame amplitude
+			amp				= atof(token);
+			// now we know all about the current frame, but we want to know if some frames are missing between this and the last one
+
+			// while current frame sample is farther than one hopsize...
+			while(startSample > endSample+hopSize)
+			{
+				missSampCnt++;				// ...one sample is missing
+				endSample += hopSize;		// move to next hop
+			}
+
+			// if frames are missing do interpolation and update indices
+			if(missSampCnt>0)
+				startSample = interpolateSamples(parIndex, &frameIndex, missSampCnt, endSample+hopSize, freq, amp, &prevFreq, &prevAmp);
+
+			partials.partialFrequencies[parIndex][frameIndex]	= (float)freq;
+			partials.partialFreqMean[parIndex] 					+= freq;			// for frequency mean
+			partials.setFreqDelta(parIndex, frameIndex-1, (freq-prevFreq)/hopSize);	// freq delta between prev and current frame
+			prevFreq 	= freq;
+
+			partials.partialAmplitudes[parIndex][frameIndex]	= (float)amp;
+			partials.setAmpDelta(parIndex, frameIndex-1, (amp-prevAmp)/hopSize);	// amp delta between prev and current frame
+			prevAmp		= amp;
+
+			endSample	= startSample;
+			token		= strtok(0, " ");	// next frame frequency, to be checked
+		}
+		#ifdef DO_CHECKS
+		if(frameIndex != (frameNum-1))
+		{
+			cout << "Parser Error: frame count mismatch on partial " << parIndex << ", bad file format"  << endl;	// exit if mismatch
+			cout << "frameIndex: " << frameIndex << endl;
+			cout << "frameNum: " << frameNum << endl;
+			return false;
+		}
+		#endif
+
+		partials.partialFreqMean[parIndex] /= partials.partialNumFrames[parIndex];									// frequency mean
+
+		getline(fin, s);					// next partial line, to check
+	}
+	#ifdef DO_CHECKS
+	if(parIndex != (parNum-1))
+	{
+		cout << "Parser Error: partial count mismatch, bad file format"  << endl;									// exit if mismatch
+		return false;
+	}
+	#endif
+
+	partials.setHopNum(maxSample/hopSize);
+
+	gettimeofday(&stop, NULL);
+	parserT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
+
+	gettimeofday(&start, NULL);
+	staticCalculations();
+	gettimeofday(&stop, NULL);
+	staticT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
+
+	fin.close();
+
+
+	printf("\n-----------------------\n");
+	printf("\nFile: %s\n", filename);
+	printf("\n-----------------------\n");
+	printf("Profiler\n");
+	printf("-----------------------\n");
+	printf("Hop size parser:\t\t%lu usec\n", hopSizeT);
+	printf("File parser:\t\t\t%lu usec\n", parserT);
+	printf("Static calculations:\t\t%lu usec\n", staticT);
+	printf("\n\nTotal:\t\t%lu usec\n", hopSizeT+parserT+staticT);
+	printf("-----------------------\n");
+
+	return true;
+}
+
+
+int Spear_parser::interpolateSamples(int parIndex, int *frameIndex, int missCnt, int nextSample, double nextFreq, double nextAmp, double *prevFreq, double *prevAmp)
+{
+	int frame			= *frameIndex;						// current frame index
+	int sample			= nextSample - (hopSize*(missCnt)); // move from next real frame sample to first missing frame sample
+	double freq			= *prevFreq;						// freq of the prev real frame
+	double freqStep		= (nextFreq-*prevFreq)/(missCnt+1);	// fixed freq step between hops, for missing frames [linear interpolation]
+	double deltaFreq	= freqStep/hopSize;					// fixed hop freq step in samples
+	double amp			= *prevAmp;							// same for amp...
+	double ampStep		= (nextAmp-*prevAmp)/(missCnt+1);
+	double deltaAmp		= ampStep/hopSize;
+
+	// for each missing frame
+	for(int i=0; i<missCnt; i++)
+	{
+		// calculate values for current missing frame
+		freq	+= freqStep;
+		amp		+= ampStep;
+		// save values
+		partials.partialFrequencies[parIndex][frame]	= freq;
+		partials.partialAmplitudes[parIndex][frame]		= amp;
+		partials.partialFreqMean[parIndex]				+= freq;	// for frequency mean
+		// set deltas of previous frame [real or missing]
+		partials.setFreqDelta(parIndex, frame-1, deltaFreq);
+		partials.setAmpDelta(parIndex, frame-1, deltaAmp);
+		// move to next frame [missing or real]
+		sample += hopSize;
+		frame++;
+	}
+
+	// update global values
+	*frameIndex	= frame;
+	*prevFreq	= freq;
+	*prevAmp	= amp;
+
+	return sample;	// return the frame sample of the next real frame
+}
+
+
+
+// for each frame, statically calculate:
+// - which partial is active [and the total num of active partials]
+// - at which local frame each partial is
+void Spear_parser::staticCalculations()
+{
+	partials.maxActiveParNum = 0;				// init to find maximum
+
+	unsigned short *indices	= new unsigned short[partials.parNum];	// temp array to store up to the maximum num of active partial indices
+	unsigned int activeCnt	= 0;						// counts the num of active partials in each frame
+
+	unsigned int frameSample = 0;						// current frame in samples
+
+	char *partialStarted = new char [partials.parNum];	// index of the last local frame found per each partial
+	fill(partialStarted, partialStarted+partials.parNum, 0);
+
+	for(unsigned int i=0; i<partials.hopNum+1; i++)		// for each frame [not hops, this explains the +1]
+	{
+		//partials.localPartialFrames[i] = new int[partials.parNum];	// init all local frames to -1
+		//fill(partials.localPartialFrames[i], partials.localPartialFrames[i]+partials.parNum, -1);
+
+		frameSample = i*hopSize;	// current frame, expressed in samples
+		activeCnt	  = 0;			// reset a each frame
+
+		for(unsigned int j=0; j<partials.parNum; j++)	// for each partial
+		{
+			// check if inside active time region [expressed in samples]
+			if( (frameSample>=partials.partialStartSample[j]) && (frameSample<partials.partialEndSample[j]) )	// frame sample not equal to end sample, this filters out last frames and partials with one frame only
+			{
+				// activity
+				indices[activeCnt] = j;	// save active index
+				activeCnt++;			// increase counter
+
+				// partial local frames
+				if(partialStarted[j]==0)	// this partial has just started, so current local frame is first frame
+				{
+					partialStarted[j] 		 		= 1;
+					partials.partialStartFrame[j]	= i;	// here is the number of the first frame
+				}
+			}
+		}
+
+		// activity
+		partials.activePartialNum[i] = activeCnt;							// save number of active partials for this frame
+		partials.activePartials[i]	 = new unsigned int[activeCnt];					// set correct size to save all indices
+
+		// look for maximum number of active partials at the same time
+		if(activeCnt > partials.maxActiveParNum)
+			partials.maxActiveParNum = activeCnt;
+
+		// copy indices
+		for(unsigned int k=0; k<activeCnt; k++)
+			partials.activePartials[i][k] = indices[k];
+	}
+
+	delete[] indices;
+	delete[] partialStarted;
+
+	delete[] partials.partialStartSample;
+	delete[] partials.partialEndSample;
+}
+
+
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/d-box/spear_parser.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,188 @@
+/*
+ * spear_parser.h v1.2
+ *
+ *  Created on: May 6, 2014
+ *      Author: Victor Zappi
+ */
+
+#ifndef SPEAR_PARSER_H_
+#define SPEAR_PARSER_H_
+
+#include <iostream>
+#include <fstream>
+#include <cstring>
+#include <string>
+#include <stdlib.h>		// atoi, atof
+#include <math.h>
+#include <algorithm>	// std::fill
+
+#include <sys/time.h>
+
+using namespace std;
+
+
+//------------------------------------------------------------------------------------------------
+// partials
+//------------------------------------------------------------------------------------------------
+
+class Spear_parser; // for class friendship
+
+class Partials
+{
+	friend class Spear_parser;
+	friend class Dbox_parser;
+
+public:
+	int **partialSamples;				// sample at which each frame is
+	float **partialFrequencies;			// frequencies at each frame
+	float **partialAmplitudes;			// amplitudes at each frame
+	unsigned int *partialNumFrames;		// Length of each partial in frames
+	unsigned int *partialStartFrame;	// frame at which each partial begins
+	float **partialFreqDelta;			// constant frequency slope for each partial in each frame interval
+	float **partialAmpDelta;			// constant amplitude slope for each partial in each frame interval
+	float *partialFreqMean;				// frequency mean for each partial, over all its frames
+
+	unsigned short *activePartialNum;	// num of each active partial at each frame
+	unsigned int **activePartials;		// indices of all active partials at each frame
+
+
+	int getPartialNum();
+	int getHopNum();
+	int getMaxActivePartialNum();
+
+private:
+	Partials();
+	~Partials();
+
+	unsigned int *partialStartSample;	// sample at which each partial begins
+	unsigned int *partialEndSample;		// sample at which each partial ends [sample gap between 2 consecutive frames can be an integer multiple of hopSize]
+	unsigned int parNum;
+	unsigned int currentSample;
+	unsigned int hopSize;
+	unsigned int hopNum;
+	unsigned int maxActiveParNum;
+
+	void init(int parNum, int hopSize, bool isDBX=false);
+	void update(int parIndex, int frameNum);
+	void setFreqDelta(int parIndex, int frameNum, double delta);
+	void setAmpDelta(int parIndex, int frameNum, double delta);
+	void setHopNum(int hopNum);
+};
+
+inline int Partials::getPartialNum()
+{
+	return parNum;
+}
+
+inline void Partials::setHopNum(int hopN)
+{
+	hopNum = hopN;
+
+	// prepare data structures
+	activePartialNum 	= new unsigned short[hopNum+1];	// +1 cos total num of frames = num of hops+1
+	activePartials	 	= new unsigned int *[hopNum+1];
+}
+
+// useful to increase current sample using a modulo on the total number of samples [easy to be deduced from the total num or hops]
+inline int Partials::getHopNum()
+{
+	return hopNum;
+}
+
+inline void Partials::setFreqDelta(int parIndex, int frameNum, double delta)
+{
+	partialFreqDelta[parIndex][frameNum] = delta;
+}
+
+inline void Partials::setAmpDelta(int parIndex, int frameNum, double delta)
+{
+	partialAmpDelta[parIndex][frameNum] = delta;
+}
+
+inline int Partials::getMaxActivePartialNum()
+{
+	return maxActiveParNum;
+}
+
+
+
+
+
+
+
+//------------------------------------------------------------------------------------------------
+// spear parser
+//------------------------------------------------------------------------------------------------
+
+class Spear_parser
+{
+public:
+	Spear_parser();
+	~Spear_parser();
+
+	Partials partials;
+
+	bool parseFile(string filename, int hopsize=-1, int samplerate = 44100);
+	bool parseFile(char *filename, int hopsize=-1, int samplerate = 44100);
+	int getHopSize();
+	int getFileSampleRate();
+	double getDeltaTime();
+
+private:
+
+	int hopSize;
+	int fileSampleRate;
+	double deltaTime;	// min time gap between consecutive frames
+
+	timeval start, stop;
+	unsigned long hopSizeT, parserT, staticT;
+
+	void calculateDeltaTime();
+	void calculateHopSize(char *filename);
+	bool parser(char *filename, int hopsize=-1, int samplerate=44100);
+	bool DBXparser(char *filename, int samplerate=44100);
+	bool TXTparser(char *filename, int hopsize=-1, int samplerate=44100);
+	int fromTimeToSamples(float time);
+	int interpolateSamples(int parIndex, int *frameIndex, int missCnt, int nextSample,
+							double nextFreq, double nextAmp, double *prevFreq, double *prevAmp);
+	void staticCalculations();
+
+};
+
+inline bool Spear_parser::parseFile(string filename, int hopsize, int samplerate)
+{
+	return parser((char *)filename.c_str(), hopsize, samplerate);
+}
+
+inline bool Spear_parser::parseFile(char *filename, int hopsize, int samplerate)
+{
+	return parser(filename, hopsize, samplerate);
+}
+
+inline void Spear_parser::calculateDeltaTime()
+{
+	deltaTime = (double)hopSize/ (double)fileSampleRate;
+}
+
+// each time value in the file is rounded, and 2 consecutive frames can differ of a time gap = i*deltaTime, where i is a positive integer
+inline int Spear_parser::fromTimeToSamples(float time)
+{
+	return round(time/deltaTime)*hopSize;	// round is necessary since in the file log time values are rounded, so they do not apparently look like integer multiples of deltaTime
+}
+
+inline int Spear_parser::getHopSize()
+{
+	return hopSize;
+}
+
+inline int Spear_parser::getFileSampleRate()
+{
+	return fileSampleRate;
+}
+
+inline double Spear_parser::getDeltaTime()
+{
+	return deltaTime;
+}
+
+#endif /* SPEAR_PARSER_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/oscillator_bank/audio_routines.S	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,161 @@
+@
+@ audio_routines.S
+@
+@ NEON-based functions for time-critical audio processing
+@
+@ Andrew McPherson 2014
+@ Queen Mary University of London
+@
+
+	.syntax unified
+	.arch armv7-a
+	.fpu neon
+
+@ 	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
+@							  int activePartialNum, int lookupTableSize,
+@							  float *phases, float *frequencies, float *amplitudes,
+@							  float *freqDerivatives, float *ampDerivatives,
+@							  float *lookupTable);
+
+@ Registers:
+@    r0: numAudioFrames        How many frames to render
+@    r1: audioOut              Buffer for audio output samples [stereo]
+@    r2: activePartialNum      How many active partials to render
+@    r3: lookupTableSize       Size of lookup table
+@    ---- other arguments start on the stack and are moved: -----
+@    r4: phases                Phase of each oscillator (pointer)
+@    r5: frequencies           Normalised frequency of each oscillator (pointer)
+@    r6: amplitudes            Normalised amplitude of each oscillator (pointer)
+@    r7: freqDerivatives       Derivative of frequency for each oscillator (pointer)
+@    r8: ampDerivatives        Derivative of amplitude for each oscillator (pointer)
+@    r9: lookupTable           Lookup table containing one oscillation
+@
+@ Alignment requirements:
+@    audioOut: 8-byte boundary
+@    phases: 16-byte boundary
+@    frequencies: 16-byte boundary
+@    amplitudes: 16-byte boundary
+@    freqDerivatives: 16-byte bounary
+@    ampDerivatives: 16-byte boundary
+@    lookupTable: 4-byte boundary (TODO: check this)
+
+	.align	2
+	.global	oscillator_bank_neon
+	.thumb
+	.thumb_func
+	.type	oscillator_bank_neon, %function
+oscillator_bank_neon:
+
+
+dSample		.dn		D6.F32
+qPhases		.qn		Q8.F32
+dPhases_0	.dn		D16.F32
+dPhases_1	.dn		D17.F32
+qFreqs		.qn		Q9.F32
+dFreqs_0	.dn		D18.F32
+dFreqs_1	.dn		D19.F32
+qAmps		.qn		Q10.F32
+dAmps_0		.dn		D20.F32
+dAmps_1		.dn		D21.F32
+qFreqDs		.qn		Q11.F32
+dFreqDs_0	.dn		D22.F32
+dFreqDs_1	.dn		D23.F32
+qAmpDs		.qn		Q12.F32
+dAmpDs_0	.dn		D24.F32
+dAmpDs_1	.dn		D25.F32
+
+qBaseInts	.qn		Q13.U32		@ Base indexes: unsigned ints x4
+dBaseInts_0	.dn		D26.U32
+dBaseInts_1	.dn		D27.U32
+qFractions  .qn     Q14.F32		@ Fraction indexes: floats x4
+qTableBase	.qn		Q15.U32		@ Base of lookup table
+
+	cmp r0, #0					@ Check for trivial case 1: zero frames
+	it eq
+	bxeq lr						@ Return if that's the case (otherwise might have odd behaviour)
+	cmp r2, #4					@ Check for trivial case 2: zero oscillators
+	it lt
+	bxlt lr						@ Return if that's the case
+
+	push {r4-r11}				@ Now arguments start 32 bytes above SP
+    add r11, sp, #32			@ Pointer to 32 bytes into the stack
+    ldm r11, {r4-r9}			@ Load 6 arguments into registers
+
+	vdup qTableBase, r9			@ Move lookup table base index into 4 ints
+
+	@ Outer loop: iterate over the number of oscillators, choosing 4 at a
+	@ time to work with.
+oscbank_oscillator_loop:
+	vld1 {dPhases_0, dPhases_1}, [r4]		@ no increment; will store at end of sample loop
+	vld1 {dFreqs_0, dFreqs_1}, [r5]
+	vld1 {dAmps_0, dAmps_1}, [r6]
+	vld1 {dFreqDs_0, dFreqDs_1}, [r7]!		@ increment; won't update at end of sample loop
+	vld1 {dAmpDs_0, dAmpDs_1}, [r8]!
+
+	push {r0-r1,r4-r8}
+	@ --- inner loop: iterate over the number of samples ---
+oscbank_sample_loop:
+	vcvt qBaseInts, qPhases		     		@ Take floor(phases)
+	vmov q2.f32, #1.0						@ Load 1.0 into every slot of q2
+	vshl q0.U32, qBaseInts, #2				@ Shift the indexes left 2 (*4 for float addressing)
+	vcvt qFractions, qBaseInts				@ int back to float
+	vadd q0.U32, q0.U32, qTableBase			@ Find memory addresses
+
+	vmov r4, r5, d0							@ Move two indexes to ARM registers
+	vmov r6, r7, d1							@ Move two more indexes to ARM registers
+	vsub qFractions, qPhases, qFractions	@ fraction = phase - floor(phase)
+
+	vldr.64	d0, [r4]						@ Load two consecutive floats at each location
+	vldr.64 d1, [r5]						@ These hold the previous and following samples in the table
+	vldr.64	d2, [r6]						@ TODO: check whether these work at 4-byte alignment
+	vldr.64 d3, [r7]
+
+	@ Format at this point:
+	@ Osc0(before) Osc0(after) Osc1(before) Osc1(after) Osc2(before) Osc2(after) Osc3(before) Osc3(after)
+	@ We want:
+	@ Osc0(before) Osc1(before) Osc2(before) Osc3(before) Osc0(after) Osc1(after) Osc2(after) Osc3(after)
+
+	vuzp.32 q0, q1							@ Now q0 contains before, q1 contains after
+	vsub q2.f32, q2.f32, qFractions			@ q2 = 1.0 - fraction
+	vmul q1.f32, q1.f32, qFractions			@ q1 = fraction * after
+	vmul q0.f32, q0.f32, q2.f32				@ q0 = (1.0 - fraction) * before
+
+	vadd qPhases, qPhases, qFreqs			@ Update phases
+	vadd qFreqs, qFreqs, qFreqDs			@ Update frequencies
+
+	vadd q0.f32, q0.f32, q1.f32				@ Add two interpolated components to get the final sample
+	vdup q2.u32, r3							@ Put lookup table size into each element of q2
+	vcvt qBaseInts, qPhases					@ Take floor of new phases
+	vmul q0.f32, q0.f32, qAmps				@ Multiply samples by current amplitude
+
+	vld1 dSample, [r1]						@ Load the current stereo samples
+	vpadd d2.f32, d0.f32, d1.f32			@ Pairwise accumulate q0 (output sample) into d2
+
+	vand q2, q2, qBaseInts					@ Logical AND of new phase int leaves 1 bit set only if phase >= table size
+	vpadd d3.f32, d2.f32, d2.f32			@ Pairwise accumulate d2 into d0 --> d0[0] and d0[1] both hold total of 4 oscillators
+	vadd qAmps, qAmps, qAmpDs				@ Update amplitudes
+	vcvt q0.f32, q2.u32						@ Convert int back to float after AND operation
+
+	vadd  dSample, dSample, d3.f32			@ Add oscillator outputs to each channel
+
+	subs r0, r0, #1							@ numFrames--
+	vsub qPhases, qPhases, q0.f32			@ Keep phases in table range
+	vst1 dSample, [r1]!						@ Store back in buffer and increment by 8
+
+	it gt
+	bgt oscbank_sample_loop					@ Loop if numFrames > 0
+
+	@ --- end inner loop ---
+	pop {r0-r1,r4-r8}						@ Restore registers: restores audioOut and numFrames, among others
+
+	vst1 {dPhases_0, dPhases_1}, [r4]!		@ Store phases back to array
+	vst1 {dFreqs_0, dFreqs_1}, [r5]!		@ Store frequencies back to array
+	vst1 {dAmps_0, dAmps_1}, [r6]!			@ Store amplitudes back to array
+											@ No need to update r7, r8
+
+	subs r2, r2, #4							@ numPartials -= 4
+	it  gt
+	bgt oscbank_oscillator_loop	@ Loop if numPartials > 0
+
+    pop {r4-r11}
+	bx lr
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/oscillator_bank/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,116 @@
+/*
+ * main.cpp
+ *
+ *  Created on: Oct 24, 2014
+ *      Author: parallels
+ */
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+int gNumOscillators = 32;
+int gWavetableLength = 1024;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --num-oscillators [-n] oscs: Set the number of oscillators to use (default: 32)\n";
+	cerr << "   --wavetable [-w] length:     Set the wavetable length in samples (default: 1024)\n";
+	cerr << "   --help [-h]:                 Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"num-oscillators", 1, NULL, 'n'},
+		{"wavetable", 1, NULL, 'w'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hn:w:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'n':
+				gNumOscillators = atoi(optarg);
+				if(gNumOscillators <= 0) {
+					usage(basename(argv[0]));
+					exit(0);
+				}
+				break;
+		case 'w':
+				gWavetableLength = atoi(optarg);
+				if(gWavetableLength < 4)
+					gWavetableLength = 4;
+				if(gWavetableLength > 16384)
+					gWavetableLength = 16384;
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, 0) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	if(settings.verbose) {
+		cout << "--> Using " << gNumOscillators << " oscillators and wavetable of " << gWavetableLength << " samples\n";
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/oscillator_bank/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,238 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\.io
+
+ */
+
+/*
+ * render.cpp
+ *
+ *  Created on: Oct 24, 2014
+ *      Author: parallels
+ */
+
+/**
+\example 4_oscillator_bank
+
+Oscillator Bank
+----------------------
+
+These files demonstrate an oscillator bank implemented in assembly code 
+that is used as part of the d-box project.
+*/
+
+#include <Bela.h>
+#include <rtdk.h>
+#include <cstdlib>
+#include <cmath>
+#include <cstring>
+#include <time.h>
+
+const float kMinimumFrequency = 20.0f;
+const float kMaximumFrequency = 8000.0f;
+
+float *gWavetable;		// Buffer holding the precalculated sine lookup table
+float *gPhases;			// Buffer holding the phase of each oscillator
+float *gFrequencies;	// Buffer holding the frequencies of each oscillator
+float *gAmplitudes;		// Buffer holding the amplitudes of each oscillator
+float *gDFrequencies;	// Buffer holding the derivatives of frequency
+float *gDAmplitudes;	// Buffer holding the derivatives of amplitude
+
+float gAudioSampleRate;
+int gSampleCount;		// Sample counter for indicating when to update frequencies
+float gNewMinFrequency;
+float gNewMaxFrequency;
+
+// Task for handling the update of the frequencies using the matrix
+AuxiliaryTask gFrequencyUpdateTask;
+
+// These settings are carried over from main.cpp
+// Setting global variables is an alternative approach
+// to passing a structure to userData in setup()
+
+extern int gNumOscillators;
+extern int gWavetableLength;
+
+void recalculate_frequencies();
+
+extern "C" {
+	// Function prototype for ARM assembly implementation of oscillator bank
+	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
+							  int activePartialNum, int lookupTableSize,
+							  float *phases, float *frequencies, float *amplitudes,
+							  float *freqDerivatives, float *ampDerivatives,
+							  float *lookupTable);
+}
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+bool setup(BelaContext *context, void *userData)
+{
+	srandom(time(NULL));
+
+	if(context->audioChannels != 2) {
+		rt_printf("Error: this example needs stereo audio enabled\n");
+		return false;
+	}
+
+	// Initialise the sine wavetable
+	if(posix_memalign((void **)&gWavetable, 8, (gWavetableLength + 1) * sizeof(float))) {
+		rt_printf("Error allocating wavetable\n");
+		return false;
+	}
+	for(int n = 0; n < gWavetableLength + 1; n++)
+		gWavetable[n] = sinf(2.0 * M_PI * (float)n / (float)gWavetableLength);
+
+	// Allocate the other buffers
+	if(posix_memalign((void **)&gPhases, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating phase buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gFrequencies, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating frequency buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gAmplitudes, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating amplitude buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gDFrequencies, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating frequency derivative buffer\n");
+		return false;
+	}
+	if(posix_memalign((void **)&gDAmplitudes, 16, gNumOscillators * sizeof(float))) {
+		rt_printf("Error allocating amplitude derivative buffer\n");
+		return false;
+	}
+
+	// Initialise buffer contents
+
+	float freq = kMinimumFrequency;
+	float increment = (kMaximumFrequency - kMinimumFrequency) / (float)gNumOscillators;
+
+	for(int n = 0; n < gNumOscillators; n++) {
+		gPhases[n] = 0.0;
+
+		if(context->analogFrames == 0) {
+			// Random frequencies when used without matrix
+			gFrequencies[n] = kMinimumFrequency + (kMaximumFrequency - kMinimumFrequency) * ((float)random() / (float)RAND_MAX);
+		}
+		else {
+			// Constant spread of frequencies when used with matrix
+			gFrequencies[n] = freq;
+			freq += increment;
+		}
+
+		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
+		gFrequencies[n] *= (float)gWavetableLength / context->audioSampleRate;
+		gAmplitudes[n] = ((float)random() / (float)RAND_MAX) / (float)gNumOscillators;
+		gDFrequencies[n] = gDAmplitudes[n] = 0.0;
+	}
+
+	increment = 0;
+	freq = 440.0;
+
+	for(int n = 0; n < gNumOscillators; n++) {
+		// Update the frequencies to a regular spread, plus a small amount of randomness
+		// to avoid weird phase effects
+		float randScale = 0.99 + .02 * (float)random() / (float)RAND_MAX;
+		float newFreq = freq * randScale;
+
+		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
+		gFrequencies[n] = newFreq * (float)gWavetableLength / context->audioSampleRate;
+
+		freq += increment;
+	}
+
+	// Initialise auxiliary tasks
+	if((gFrequencyUpdateTask = Bela_createAuxiliaryTask(&recalculate_frequencies, 85, "bela-update-frequencies")) == 0)
+		return false;
+
+	//for(int n = 0; n < gNumOscillators; n++)
+	//	rt_printf("%f\n", gFrequencies[n]);
+
+	gAudioSampleRate = context->audioSampleRate;
+	gSampleCount = 0;
+
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	// Initialise buffer to 0
+	memset(context->audioOut, 0, 2 * context->audioFrames * sizeof(float));
+
+	// Render audio frames
+	oscillator_bank_neon(context->audioFrames, context->audioOut,
+			gNumOscillators, gWavetableLength,
+			gPhases, gFrequencies, gAmplitudes,
+			gDFrequencies, gDAmplitudes,
+			gWavetable);
+
+	if(context->analogFrames != 0 && (gSampleCount += context->audioFrames) >= 128) {
+		gSampleCount = 0;
+		gNewMinFrequency = map(context->analogIn[0], 0, 1.0, 1000.0f, 8000.0f);
+		gNewMaxFrequency = map(context->analogIn[1], 0, 1.0, 1000.0f, 8000.0f);
+
+		// Make sure max >= min
+		if(gNewMaxFrequency < gNewMinFrequency) {
+			float temp = gNewMaxFrequency;
+			gNewMaxFrequency = gNewMinFrequency;
+			gNewMinFrequency = temp;
+		}
+
+		// Request that the lower-priority task run at next opportunity
+		//Bela_scheduleAuxiliaryTask(gFrequencyUpdateTask);
+	}
+}
+
+// This is a lower-priority call to update the frequencies which will happen
+// periodically when the matrix is enabled. By placing it at a lower priority,
+// it has minimal effect on the audio performance but it will take longer to
+// complete if the system is under heavy audio load.
+
+void recalculate_frequencies()
+{
+	float freq = gNewMinFrequency;
+	float increment = (gNewMaxFrequency - gNewMinFrequency) / (float)gNumOscillators;
+
+	for(int n = 0; n < gNumOscillators; n++) {
+		// Update the frequencies to a regular spread, plus a small amount of randomness
+		// to avoid weird phase effects
+		float randScale = 0.99 + .02 * (float)random() / (float)RAND_MAX;
+		float newFreq = freq * randScale;
+
+		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
+		gFrequencies[n] = newFreq * (float)gWavetableLength / gAudioSampleRate;
+
+		freq += increment;
+	}
+}
+
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+	free(gWavetable);
+	free(gPhases);
+	free(gFrequencies);
+	free(gAmplitudes);
+	free(gDFrequencies);
+	free(gDAmplitudes);
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/tank_wars/game.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,339 @@
+/*
+ * game.cpp
+ *
+ *  Created on: Nov 10, 2014
+ *      Author: parallels
+ */
+
+#include <cmath>
+#include <cstdlib>
+#include "vector_graphics.h"
+#include <Utilities.h>
+
+// Virtual screen size
+int screenWidth, screenHeight;
+
+// Basic information on the terrain and the tanks
+float *groundLevel;  // Y coordinate of the ground for each X coordinate
+float tank1X, tank1Y, tank2X, tank2Y; // Positions of the two tanks
+float tankRadius = 20;  // Radius of the tanks
+float cannonLength = 40;   // How long the cannon on each tank extends
+float gravity = 0.05;      // Strength of gravity
+
+// Current state of the game
+int playerHasWon = 0; // 1 if player 1 wins, 2 if player 2 wins, 0 if game in progress
+bool player1Turn = true;  // true if it's player 1's turn; false otherwise
+float tank1CannonAngle = M_PI/2;
+float tank2CannonAngle = M_PI/2; // Direction the tank cannons are pointing
+float tank1CannonStrength = 3;
+float tank2CannonStrength = 3; // Strength of intended projectile launch
+
+// Location of the projectile
+bool projectileInMotion = false;
+float projectilePositionX, projectilePositionY;
+float projectileVelocityX, projectileVelocityY;
+
+// Infor needed for sound rendering
+bool collisionJustOccurred = false;
+bool tankHitJustOccurred = false;
+
+// Useful utility function for generating random floating-point values
+float randomFloat(float low, float hi)
+{
+	float r = (float)random() / (float)RAND_MAX;
+	return map(r, 0, 1, low, hi);
+}
+
+// Restart the game, without reallocating memory
+void restartGame()
+{
+	float player1Height = screenHeight * 3/4; // randomFloat(screenHeight/2, screenHeight-5);
+	float player2Height = screenHeight - 5; // randomFloat(screenHeight/2, screenHeight-5);
+	for(int i = 0; i < screenWidth * 0.2; i++) {
+		groundLevel[i] = player1Height;
+	}
+	for(int i = screenWidth * 0.2; i < screenWidth * 0.8; i++) {
+		groundLevel[i] = player1Height + (player2Height - player1Height) * (i - screenWidth*0.2)/(screenWidth*0.6);
+	}
+	for(int i = screenWidth * 0.8; i < screenWidth; i++) {
+		groundLevel[i] = player2Height;
+	}
+
+	// Set the location of the two tanks so they rest on the ground at opposite sides
+	tank1X = screenWidth * 0.1;
+	tank1Y = player1Height;
+	tank2X = screenWidth * 0.9;
+	tank2Y = player2Height;
+
+	playerHasWon = 0;
+	projectileInMotion = false;
+}
+
+// Initialise the game
+void setupGame(int width, int height)
+{
+	// Set the screen size
+	screenWidth = width;
+	screenHeight = height;
+
+	// Initialize the ground level
+	groundLevel = new float[screenWidth];
+
+	restartGame();
+}
+
+// Advance the turn to the next player
+void nextPlayersTurn() {
+	player1Turn = !player1Turn;
+}
+
+
+// Move forward one frame on the game physics
+void nextGameFrame()
+{
+	if(!projectileInMotion)
+		return;
+
+	// Update position of projectile
+	projectilePositionX += projectileVelocityX;
+	projectilePositionY += projectileVelocityY;
+	projectileVelocityY += gravity;
+
+	// Check collision with tanks first: a collision with tank 1 means player 2 wins and vice-versa
+	if((tank1X - projectilePositionX)*(tank1X - projectilePositionX) +
+		(tank1Y - projectilePositionY)*(tank1Y - projectilePositionY)
+		<= tankRadius * tankRadius)
+	{
+		projectileInMotion = false;
+		collisionJustOccurred = false;
+		tankHitJustOccurred = true;
+		playerHasWon = 2;
+	}
+	else if((tank2X - projectilePositionX)*(tank2X - projectilePositionX) +
+		(tank2Y - projectilePositionY)*(tank2Y - projectilePositionY)
+		<= tankRadius * tankRadius)
+	{
+		projectileInMotion = false;
+		collisionJustOccurred = false;
+		tankHitJustOccurred = true;
+		playerHasWon = 1;
+	}
+	else if(projectilePositionX < 0 || projectilePositionX >= screenWidth) {
+		// Check collision whether projectile has exited the screen to the left or right
+		projectileInMotion = false;
+		collisionJustOccurred = true;
+		nextPlayersTurn();
+	}
+	else if(projectilePositionY >= groundLevel[(int)floorf(projectilePositionX)]) {
+		// Check for projectile collision with ground
+		projectileInMotion = false;
+		collisionJustOccurred = true;
+		nextPlayersTurn();
+	}
+}
+
+// Updates for game state
+void setTank1CannonAngle(float angle)
+{
+	tank1CannonAngle = angle;
+}
+
+void setTank2CannonAngle(float angle)
+{
+	tank2CannonAngle = angle;
+}
+
+void setTank1CannonStrength(float strength)
+{
+	tank1CannonStrength = strength;
+}
+
+void setTank2CannonStrength(float strength)
+{
+	tank2CannonStrength = strength;
+}
+
+// FIRE!
+void fireProjectile()
+{
+	// Can't fire while projectile is already moving, or if someone has won
+	if(projectileInMotion)
+		return;
+	if(playerHasWon != 0)
+		return;
+
+    if(player1Turn) {
+		projectilePositionX = tank1X + cannonLength * cosf(tank1CannonAngle);
+		projectilePositionY = tank1Y - cannonLength * sinf(tank1CannonAngle);
+		projectileVelocityX = tank1CannonStrength * cosf(tank1CannonAngle);
+		projectileVelocityY = -tank1CannonStrength * sinf(tank1CannonAngle);
+    }
+    else {
+		projectilePositionX = tank2X + cannonLength * cosf(tank2CannonAngle);
+		projectilePositionY = tank2Y - cannonLength * sinf(tank2CannonAngle);
+		projectileVelocityX = tank2CannonStrength * cosf(tank2CannonAngle);
+		projectileVelocityY = -tank2CannonStrength * sinf(tank2CannonAngle);
+    }
+
+    // GO!
+    projectileInMotion = true;
+}
+
+// Game state queries
+bool gameStatusPlayer1Turn()
+{
+	return player1Turn;
+}
+
+bool gameStatusProjectileInMotion()
+{
+	return projectileInMotion;
+}
+
+int gameStatusWinner()
+{
+	return playerHasWon;
+}
+
+bool gameStatusCollisionOccurred()
+{
+	if(collisionJustOccurred) {
+		collisionJustOccurred = false;
+		return true;
+	}
+	return false;
+}
+
+bool gameStatusTankHitOccurred()
+{
+	if(tankHitJustOccurred) {
+		tankHitJustOccurred = false;
+		return true;
+	}
+	return false;
+}
+
+
+float gameStatusProjectileHeight()
+{
+	return projectilePositionY / (float)screenHeight;
+}
+
+// Clean up any allocated memory for the game
+void cleanupGame()
+{
+	delete groundLevel;
+}
+
+// Drawing routines. Arguments are (interleaved) buffer to render
+// into, the available size, and the target for how many samples
+// to use (actual usage might vary slightly). Regardless of
+// lengthTarget, never use more than bufferSize samples.
+
+int drawGround(float *buffer, int bufferSize, int framesTarget)
+{
+	int length;
+
+	// Calculate total length of ground line, to arrive at a speed calculation
+	float totalLineLength = 0.4f*screenWidth
+							+ sqrtf(0.36f*screenWidth*screenWidth
+									+ (tank2Y-tank1Y)*(tank2Y-tank1Y));
+
+	// Speed is calculated in pixels per frame
+	float speed = totalLineLength / (float)framesTarget;
+
+	// Draw three lines: platforms for tanks and the connecting line.
+	// Eventually, render a more complex ground from the array.
+	length = renderLine(0, tank1Y, screenWidth * 0.2, tank1Y,
+						speed, buffer, bufferSize);
+	length += renderLine(screenWidth * 0.2, tank1Y, screenWidth * 0.8, tank2Y,
+						speed, &buffer[length], bufferSize - length);
+	length += renderLine(screenWidth * 0.8, tank2Y, screenWidth, tank2Y,
+						speed, &buffer[length], bufferSize - length);
+
+	return length;
+}
+
+int drawTanks(float *buffer, int bufferSize, int framesTarget)
+{
+	int length = 0;
+
+	// Calculate total length of tank lines, to arrive at a speed calculation
+	float totalLineLength = 2.0*M_PI*tankRadius + 2.0*(cannonLength - tankRadius);
+
+	// Speed is calculated in pixels per frame
+	float speed = totalLineLength / (float)framesTarget;
+
+	if(playerHasWon != 2) {
+		// Tank 1 body = semicircle + line
+		length += renderArc(tank1X, tank1Y, tankRadius, M_PI, 2.0 * M_PI,
+							speed, buffer, bufferSize);
+		length += renderLine(tank1X + tankRadius, tank1Y,
+							 tank1X - tankRadius, tank1Y,
+							speed, &buffer[length], bufferSize - length);
+		// Tank 1 cannon (line depending on angle)
+		length += renderLine(tank1X + tankRadius * cosf(tank1CannonAngle),
+			 tank1Y - tankRadius * sinf(tank1CannonAngle),
+			 tank1X + cannonLength * cosf(tank1CannonAngle),
+			 tank1Y - cannonLength * sinf(tank1CannonAngle),
+			 speed, &buffer[length], bufferSize - length);
+	}
+
+	if(playerHasWon != 1) {
+		// Same idea for tank 2
+		length += renderArc(tank2X, tank2Y, tankRadius, M_PI, 2.0 * M_PI,
+							speed, &buffer[length], bufferSize - length);
+		length += renderLine(tank2X + tankRadius, tank2Y,
+							 tank2X - tankRadius, tank2Y,
+							 speed, &buffer[length], bufferSize - length);
+		length += renderLine(tank2X + tankRadius * cosf(tank2CannonAngle),
+			 tank2Y - tankRadius * sinf(tank2CannonAngle),
+			 tank2X + cannonLength * cosf(tank2CannonAngle),
+			 tank2Y - cannonLength * sinf(tank2CannonAngle),
+			 speed, &buffer[length], bufferSize - length);
+	}
+
+	return length;
+}
+
+int drawProjectile(float *buffer, int bufferSize, int framesTarget)
+{
+	if(!projectileInMotion)
+		return 0;
+
+	// Draw a point for a specified number of frames (each containing X and Y)
+	// Return the number of items used in the buffer, which will be twice
+	// the number of frames unless the buffer is full
+
+	if(bufferSize/2 < framesTarget) {
+		renderPoint(projectilePositionX, projectilePositionY, buffer, bufferSize/2);
+		return bufferSize;
+	}
+	else {
+		renderPoint(projectilePositionX, projectilePositionY, buffer, framesTarget);
+		return framesTarget*2;
+	}
+}
+
+// Main drawing routine entry point
+int drawGame(float *buffer, int bufferSize)
+{
+	int length;
+
+	// Based on buffer size, come up with speeds for each of the elements
+	// 50% of time to ground; 30% to the tanks and 20% to the projectile
+	// Give a margin of 25% beyond so we don't run out of buffer space
+	// if things take longer to draw than we guess they will
+	const float amountToUse = 0.375; // 0.75/2 because two samples per frame
+	const float groundFraction = 0.5 * amountToUse;
+	const float tankFraction = 0.3 * amountToUse;
+	const float projectileFraction = 0.2 * amountToUse;
+
+	length = drawGround(buffer, bufferSize, bufferSize * groundFraction);
+	length += drawTanks(&buffer[length], bufferSize - length,
+						bufferSize * tankFraction);
+	length += drawProjectile(&buffer[length], bufferSize - length,
+						bufferSize * projectileFraction);
+
+	return length;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/tank_wars/game.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,39 @@
+/*
+ * game.h
+ *
+ *  Created on: Nov 10, 2014
+ *      Author: parallels
+ */
+
+#ifndef GAME_H_
+#define GAME_H_
+
+// Initialisation
+void setupGame(int width, int height);
+void restartGame();
+
+// Update physics
+void nextGameFrame();
+
+// State updaters
+void setTank1CannonAngle(float angle);
+void setTank2CannonAngle(float angle);
+void setTank1CannonStrength(float strength);
+void setTank2CannonStrength(float strength);
+void fireProjectile();
+
+// State queries
+bool gameStatusPlayer1Turn();
+bool gameStatusProjectileInMotion();
+int gameStatusWinner();
+bool gameStatusCollisionOccurred();
+bool gameStatusTankHitOccurred();
+float gameStatusProjectileHeight();
+
+// Render screen; returns length of buffer used
+int drawGame(float *buffer, int bufferSize);
+
+// Cleanup and memory release
+void cleanupGame();
+
+#endif /* GAME_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/tank_wars/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,169 @@
+/*
+ * main.cpp
+ *
+ *  Created on: Oct 24, 2014
+ *      Author: parallels
+ */
+
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <sndfile.h>
+#include <Bela.h>
+
+extern int gScreenFramesPerSecond;
+
+float *gMusicBuffer = 0;
+int gMusicBufferLength = 0;
+float *gSoundBoomBuffer = 0;
+int gSoundBoomBufferLength = 0;
+float *gSoundHitBuffer = 0;
+int gSoundHitBufferLength = 0;
+
+
+using namespace std;
+
+// Load a sound sample from file
+int loadSoundFile(const string& path, float **buffer, int *bufferLength)
+{
+	SNDFILE *sndfile ;
+	SF_INFO sfinfo ;
+
+	if (!(sndfile = sf_open (path.c_str(), SFM_READ, &sfinfo))) {
+		cout << "Couldn't open file " << path << endl;
+		return 1;
+	}
+
+	int numChan = sfinfo.channels;
+	if(numChan != 1)
+	{
+		cout << "Error: " << path << " is not a mono file" << endl;
+		return 1;
+	}
+
+	*bufferLength = sfinfo.frames * numChan;
+	*buffer = new float[*bufferLength];
+	if(*buffer == 0){
+		cout << "Could not allocate buffer" << endl;
+		return 1;
+	}
+
+	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
+	int readcount = sf_read_float(sndfile, *buffer, *bufferLength);
+
+	// Pad with zeros in case we couldn't read whole file
+	for(int k = readcount; k < *bufferLength; k++)
+		(*buffer)[k] = 0;
+
+	sf_close(sndfile);
+	return 0;
+}
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --fps [-f] value:           Set target frames per second\n";
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	string musicFileName = "music.wav";
+	string soundBoomFileName = "boom.wav";
+	string soundHitFileName = "hit.wav";
+	
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"fps", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'f':
+				gScreenFramesPerSecond = atoi(optarg);
+				if(gScreenFramesPerSecond < 1)
+					gScreenFramesPerSecond = 1;
+				if(gScreenFramesPerSecond > 100)
+					gScreenFramesPerSecond = 100;
+				break;
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Load the sound files
+	if(loadSoundFile(musicFileName, &gMusicBuffer, &gMusicBufferLength) != 0) {
+		cout << "Warning: unable to load sound file " << musicFileName << endl;
+	}
+	if(loadSoundFile(soundBoomFileName, &gSoundBoomBuffer, &gSoundBoomBufferLength) != 0) {
+		cout << "Warning: unable to load sound file " << soundBoomFileName << endl;
+	}
+	if(loadSoundFile(soundHitFileName, &gSoundHitBuffer, &gSoundHitBufferLength) != 0) {
+		cout << "Warning: unable to load sound file " << soundHitFileName << endl;
+	}
+	
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, 0) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// Release sound files
+	if(gMusicBuffer != 0)
+		free(gMusicBuffer);
+	if(gSoundBoomBuffer != 0)
+		free(gSoundBoomBuffer);
+	if(gSoundHitBuffer != 0)
+		free(gSoundHitBuffer);
+	
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/tank_wars/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,337 @@
+/*
+ * render.cpp
+ *
+ *  Created on: Oct 24, 2014
+ *      Author: parallels
+ */
+
+
+#include <Bela.h>
+#include "game.h"
+#include <rtdk.h>
+#include <cmath>
+#include <cstdlib>
+#include <time.h>
+
+int gAudioFramesPerMatrixFrame = 2; // Ratio in audio to matrix sample rate
+
+int gInputTank1Angle = 0;		// Inputs for the cannon angles
+int gInputTank2Angle = 1;
+int gInputLauncher = 2;			// Input for launcher FSR
+
+int gOutputX = 0;				// Outputs for the scope
+int gOutputY = 1;
+int gOutputPlayer1LED = 2;
+int gOutputPlayer2LED = 3;
+
+int gGameFramesPerSecond = 60;	// How often the physics are updated
+int gGameFrameInterval;			// ...and in frames
+int gSamplesUntilNextFrame;		// Counter until next update
+int gSamplesSinceFinish = 0;	// How long since somebody won?
+bool gGameShouldRestart = false;// Whether we need to reinitiliase the game
+
+// Counter for overall number of samples that have elapsed
+unsigned int gSampleCounter = 0;
+
+// 1st-order filter and peak detector for launcher input
+float gLauncherLastSample = 0;
+float gLauncherFilterPole = 0.8;
+float gLauncherPeakValue = 0;
+float gLauncherPeakFilterPole = 0.999;
+float gLauncherNoiseThreshold = 0.01;
+float gLauncherMinimumPeak = 0.1;
+bool gLauncherTriggered = false;
+
+// Screen update rate; affects buffer size. Actual contents of buffer
+// may be smaller than this
+int gScreenWidth = 512;
+int gScreenHeight = 512;
+int gScreenFramesPerSecond = 25;
+
+// Double buffer for rendering screen. Each is an interleaved buffer
+// of XY data.
+float *gScreenBuffer1, *gScreenBuffer2;
+float *gScreenBufferWrite, *gScreenBufferRead;
+int gScreenBufferMaxLength;				// What is the total buffer allocated?
+int gScreenBufferReadLength;			// How long is the read buffer?
+int gScreenBufferWriteLength;			// How long is the write (next) buffer?
+int gScreenBufferReadPointer;			// Where are we in the read buffer now?
+int gScreenBufferNextUpdateLocation;	// When should we render the next buffer?
+bool gScreenNextBufferReady;			// Is the next buffer ready to go?
+
+// Auxiliary (low-priority) task for updating the screen
+AuxiliaryTask gScreenUpdateTask;
+
+// Buffers for music and sound effects
+extern float *gMusicBuffer;
+extern int gMusicBufferLength;
+extern float *gSoundBoomBuffer;
+extern int gSoundBoomBufferLength;
+extern float *gSoundHitBuffer;
+extern int gSoundHitBufferLength;
+
+// Current state for sound and music
+int gMusicBufferPointer = 0;	  // 0 means start of buffer...
+int gSoundBoomBufferPointer = -1; // -1 means don't play...
+int gSoundHitBufferPointer = -1;
+float gSoundProjectileOscillatorPhase = 0;
+float gSoundProjectileOscillatorGain = 0.2;
+float gOscillatorPhaseScaler = 0;
+
+void screen_update();
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+bool setup(BelaContext *context, void *userData)
+{
+	srandom(time(NULL));
+
+	// Verify we are running with matrix enabled
+	if(context->analogFrames == 0 || context->analogChannels < 4) {
+		rt_printf("Error: this example needs the matrix enabled with at least 4 channels\n");
+		return false;
+	}
+
+	// Initialise audio variables
+	gAudioFramesPerMatrixFrame = context->audioFrames / context->analogFrames;
+	gOscillatorPhaseScaler = 2.0 * M_PI / context->audioSampleRate;
+
+	// Initialise the screen buffers
+	gScreenBufferMaxLength = 2 * context->analogSampleRate / gScreenFramesPerSecond;
+	gScreenBuffer1 = new float[gScreenBufferMaxLength];
+	gScreenBuffer2 = new float[gScreenBufferMaxLength];
+	if(gScreenBuffer1 == 0 || gScreenBuffer2 == 0) {
+		rt_printf("Error initialising screen buffers\n");
+		return false;
+	}
+
+	gScreenBufferRead = gScreenBuffer1;
+	gScreenBufferWrite = gScreenBuffer2;
+	gScreenBufferReadLength = gScreenBufferWriteLength = 0;
+	gScreenBufferReadPointer = 0;
+	gScreenBufferNextUpdateLocation = 0;
+	gScreenNextBufferReady = false;
+
+	// Initialise the game
+	setupGame(gScreenWidth, gScreenHeight);
+	gGameFrameInterval = context->analogSampleRate / gGameFramesPerSecond;
+	gSamplesUntilNextFrame = gGameFrameInterval;
+
+	// Initialise auxiliary tasks
+	if((gScreenUpdateTask = Bela_createAuxiliaryTask(&screen_update, 90,
+														 "bela-screen-update")) == 0)
+		return false;
+
+	return true;
+}
+
+// Swap buffers on the screen
+void swap_buffers()
+{
+	if(gScreenBufferRead == gScreenBuffer1) {
+		gScreenBufferRead = gScreenBuffer2;
+		gScreenBufferWrite = gScreenBuffer1;
+	}
+	else {
+		gScreenBufferRead = gScreenBuffer1;
+		gScreenBufferWrite = gScreenBuffer2;
+	}
+
+	gScreenBufferReadLength = gScreenBufferWriteLength;
+	gScreenBufferReadPointer = 0;
+
+	// Schedule next update for 3/4 of the way through the buffer
+	gScreenBufferNextUpdateLocation = gScreenBufferReadLength * 0.75;
+	gScreenNextBufferReady = false;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	int audioIndex = 0;
+
+	for(unsigned int n = 0; n < context->analogFrames; n++) {
+		for(int k = 0; k < gAudioFramesPerMatrixFrame; k++) {
+			// Render music and sound
+			float audioSample = 0;
+
+			// Music plays in a loop
+			if(gMusicBuffer != 0 && gMusicBufferPointer >= 0) {
+				audioSample += gMusicBuffer[gMusicBufferPointer++];
+				if(gMusicBufferPointer >= gMusicBufferLength)
+					gMusicBufferPointer = 0;
+			}
+
+			// Sound effect plays until finished, then stops
+			if(gSoundBoomBuffer != 0 && gSoundBoomBufferPointer >= 0) {
+				audioSample += gSoundBoomBuffer[gSoundBoomBufferPointer++];
+				if(gSoundBoomBufferPointer >= gSoundBoomBufferLength)
+					gSoundBoomBufferPointer = -1;
+			}
+
+			if(gSoundHitBuffer != 0 && gSoundHitBufferPointer >= 0) {
+				audioSample += gSoundHitBuffer[gSoundHitBufferPointer++];
+				if(gSoundHitBufferPointer >= gSoundHitBufferLength)
+					gSoundHitBufferPointer = -1;
+			}
+
+			// Oscillator plays to indicate projectile height
+			if(gameStatusProjectileInMotion()) {
+				audioSample += gSoundProjectileOscillatorGain * sinf(gSoundProjectileOscillatorPhase);
+
+				gSoundProjectileOscillatorPhase += gOscillatorPhaseScaler * constrain(map(gameStatusProjectileHeight(),
+						1.0, 0, 300, 2000), 200, 6000);
+				if(gSoundProjectileOscillatorPhase > 2.0 * M_PI)
+					gSoundProjectileOscillatorPhase -= 2.0 * M_PI;
+			}
+
+			context->audioOut[2*audioIndex] = context->audioOut[2*audioIndex + 1] = audioSample;
+			audioIndex++;
+		}
+
+		// First-order lowpass filter to remove noise on launch FSR
+		float rawSample = analogRead(context, n, gInputLauncher);
+		float launchSample = gLauncherFilterPole * gLauncherLastSample +
+							(1.0f - gLauncherFilterPole) * rawSample;
+		gLauncherLastSample = launchSample;
+
+		// Peak-detect on launch signal
+		if(launchSample >= gLauncherPeakValue) {
+			gLauncherPeakValue = launchSample;
+			gLauncherTriggered = false;
+		}
+		else {
+			if(gLauncherPeakValue - launchSample > gLauncherNoiseThreshold && !gLauncherTriggered) {
+				// Detected a peak; is it big enough overall?
+				if(gLauncherPeakValue >= gLauncherMinimumPeak) {
+					gLauncherTriggered = true;
+					// Peak detected-- fire!!
+					// Set both cannon strengths but only one will
+					// fire depending on whose turn it is
+					float strength = map(gLauncherPeakValue,
+									     gLauncherMinimumPeak, 1.0,
+										 0.5f, 10.0f);
+					setTank1CannonStrength(strength);
+					setTank2CannonStrength(strength);
+					fireProjectile();
+				}
+			}
+
+			gLauncherPeakValue *= gLauncherPeakFilterPole;
+		}
+
+		if(--gSamplesUntilNextFrame <= 0) {
+			// Update game physics and cannon angles
+			gSamplesUntilNextFrame = gGameFrameInterval;
+
+			setTank1CannonAngle(map(analogRead(context, n, gInputTank1Angle),
+									0, 1.0, M_PI, 0));
+			setTank2CannonAngle(map(analogRead(context, n, gInputTank2Angle),
+									0, 1.0, M_PI, 0));
+			nextGameFrame();
+
+			// Check for collision and start sound accordingly
+			if(gameStatusCollisionOccurred()) {
+				gSoundBoomBufferPointer = 0;
+			}
+			
+			if(gameStatusTankHitOccurred()) {
+				gSoundHitBufferPointer = 0;
+			}
+		}
+
+		if(gScreenBufferReadPointer >= gScreenBufferReadLength - 1
+			&& gScreenNextBufferReady) {
+			// Got to the end; swap buffers
+			swap_buffers();
+		}
+
+		// Push current screen buffer to the matrix output
+		if(gScreenBufferReadPointer < gScreenBufferReadLength - 1) {
+			float x = gScreenBufferRead[gScreenBufferReadPointer++];
+			float y = gScreenBufferRead[gScreenBufferReadPointer++];
+
+			// Rescale screen coordinates to matrix ranges; invert the Y
+			// coordinate to go from normal screen coordinates to scope coordinates
+			analogWriteOnce(context, n, gOutputX, constrain(map(x, 0, gScreenWidth, 0, 1.0), 0, 1.0));
+			analogWriteOnce(context, n, gOutputY, constrain(map(y, 0, gScreenHeight, 1.0, 0), 0, 1.0));
+		}
+		else {
+			// Still not ready! Write 0 until something happens
+			analogWriteOnce(context, n, gOutputX, 0);
+			analogWriteOnce(context, n, gOutputY, 0);
+		}
+
+		if(gameStatusWinner() != 0) {
+			// Blink one LED to show who won
+			// Blink both LEDs when projectile is in motion
+			float val = (gSampleCounter % 4000 > 2000) ? 1.0 : 0;
+			analogWriteOnce(context, n, gOutputPlayer1LED, gameStatusWinner() == 1 ? val : 0);
+			analogWriteOnce(context, n, gOutputPlayer2LED, gameStatusWinner() == 2 ? val : 0);
+
+			// After 5 seconds, restart the game
+			gSamplesSinceFinish++;
+			if(gSamplesSinceFinish > 22050*5)
+				gGameShouldRestart = true;
+		}
+		else if(gameStatusProjectileInMotion()) {
+			// Blink both LEDs when projectile is in motion
+			float val = (gSampleCounter % 2000 > 1000) ? 1.0 : 0;
+			analogWriteOnce(context, n, gOutputPlayer1LED, val);
+			analogWriteOnce(context, n, gOutputPlayer2LED, val);
+		}
+		else if(gameStatusPlayer1Turn()) {
+			analogWriteOnce(context, n, gOutputPlayer1LED, 1.0);
+			analogWriteOnce(context, n, gOutputPlayer2LED, 0);
+		}
+		else {
+			analogWriteOnce(context, n, gOutputPlayer2LED, 1.0);
+			analogWriteOnce(context, n, gOutputPlayer1LED, 0);
+		}
+
+		// Check if we have reached the point where we should next update
+		if(gScreenBufferReadPointer >= gScreenBufferNextUpdateLocation &&
+		   !gScreenNextBufferReady) {
+			// Update the screen at lower priority than the audio thread
+			Bela_scheduleAuxiliaryTask(gScreenUpdateTask);
+		}
+
+		gSampleCounter++;
+	}
+}
+
+void screen_update()
+{
+	// If we should restart, reinitialise the game
+	if(gGameShouldRestart) {
+		restartGame();
+		gGameShouldRestart = false;
+		gSamplesSinceFinish = 0;
+	}
+
+	// Render the game based on the current state
+	gScreenBufferWriteLength = drawGame(gScreenBufferWrite, gScreenBufferMaxLength);
+
+	// Flag it as ready to go
+	gScreenNextBufferReady = true;
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+	// Clean up the game state
+	cleanupGame();
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/tank_wars/vector_graphics.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,72 @@
+/*
+ * vector_graphics.cpp
+ *
+ *  Created on: Nov 10, 2014
+ *      Author: parallels
+ */
+
+#include <cmath>
+
+// Draw a line between two points at a specified rate in
+// pixels per buffer sample. Indicate maximum available space.
+// Returns space used
+int renderLine(float x1, float y1, float x2, float y2, float speed,
+			   float *buffer, int maxLength) {
+	// Figure out length of line and therefore how many samples
+	// are needed to represent it based on the speed (rounded to nearest int)
+	float totalLineLength = sqrtf((x2 - x1)*(x2 - x1) + (y2 - y1)*(y2 - y1));
+	int samplesNeeded = floorf(totalLineLength / speed + 0.5);
+
+	// Now render into the buffer
+	int length = 0;
+	float scaleFactor = 1.0f / samplesNeeded;
+	for(int n = 0; n < samplesNeeded; n++) {
+		if(length >= maxLength - 1)
+			return length;
+		// X coordinate
+		*buffer++ = x1 + (float)n * scaleFactor * (x2 - x1);
+		// Y coordinate
+		*buffer++ = y1 + (float)n * scaleFactor * (y2 - y1);
+		length += 2;
+	}
+
+	return length;
+}
+
+// Draw an arc around a centre point at a specified rate of pixels
+// per buffer sample. Indicate maximum available space.
+// Returns space used
+int renderArc(float x, float y, float radius, float thetaMin, float thetaMax,
+			  float speed, float *buffer, int maxLength) {
+	// Figure out circumference of arc and therefore how many samples
+	// are needed to represent it based on the speed (rounded to nearest int)
+	float circumference = (thetaMax - thetaMin) * radius;
+	int samplesNeeded = floorf(circumference / speed + 0.5);
+
+	// Now render into the buffer
+	int length = 0;
+	float scaleFactor = 1.0f / samplesNeeded;
+	for(int n = 0; n < samplesNeeded; n++) {
+		if(length >= maxLength - 1)
+			return length;
+		// Get current angle
+		float theta = thetaMin + (float)n * scaleFactor * (thetaMax - thetaMin);
+
+		// Convert polar to cartesian coordinates
+		*buffer++ = x + radius * cosf(theta);
+		*buffer++ = y + radius * sinf(theta);
+
+		length += 2;
+	}
+
+	return length;
+}
+
+// Draw a single point for a specified number of frames
+void renderPoint(float x, float y, float *buffer, float length) {
+	while(length > 0) {
+		*buffer++ = x;
+		*buffer++ = y;
+		length--;
+	}
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/10-Instruments/tank_wars/vector_graphics.h	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,18 @@
+/*
+ * vector_graphics.h
+ *
+ *  Created on: Nov 10, 2014
+ *      Author: parallels
+ */
+
+#ifndef VECTOR_GRAPHICS_H_
+#define VECTOR_GRAPHICS_H_
+
+int renderLine(float x1, float y1, float x2, float y2, float speed,
+			   float *buffer, int maxLength);
+int renderArc(float x, float y, float radius, float thetaMin, float thetaMax,
+			  float speed, float *buffer, int maxLength);
+void renderPoint(float x, float y, float *buffer, float length);
+
+
+#endif /* VECTOR_GRAPHICS_H_ */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/11-Extras/7segment/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,142 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+
+#define NUM_PINS 12
+
+// Breadboard wiring layout:
+// 11 10 12 9  8  7
+// [ LED DISP ]
+// 1  2  3  6  4  5
+
+// Organised by display segments:
+// e d . X c g b X X X f a
+const int kPins[NUM_PINS] = {P8_07, P8_08, P8_09, P8_10, P8_11, P8_12,
+							 P8_15, P8_16, P8_27, P8_28, P8_29, P8_30};
+
+// Indices into the above array: pins 12, 9, 8, 6
+const int kDigits[4] = {9, 8, 7, 3};
+
+int gCurrentlyDisplayingDigit = 0;
+int gDigitDisplayTime = 0;
+const int kDigitMaxDisplayTime = 44;
+
+int gState = 0;
+int gStateCounter = 0;
+const int kMaxState = 25;
+
+// . g f e d c b a
+//const unsigned char kBELA[4] = {0x7C, 0x79, 0x38, 0x77};
+const unsigned char kBELA[4] = {0x7C, 0x7B, 0x38, 0x5F};
+const unsigned char kPerimeter[6] = {0x01, 0x02, 0x04, 0x08, 0x10, 0x20};
+
+int gCharacterToDisplay[4] = {0, 0, 0, 0};
+	
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+bool setup(BelaContext *context, void *userData)
+{	
+	// This project makes the assumption that the audio and digital
+	// sample rates are the same. But check it to be sure!
+	if(context->audioFrames != context->digitalFrames) {
+		rt_printf("Error: this project needs the audio and digital sample rates to be the same.\n");
+		return false;
+	}
+	
+	for(int i = 0; i < NUM_PINS; i++) {
+		pinMode(context, 0, kPins[i], OUTPUT);
+	}
+
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		// Check for rotation between digits
+		if(--gDigitDisplayTime <= 0) {
+			gCurrentlyDisplayingDigit = (gCurrentlyDisplayingDigit + 1) % 4;
+			gDigitDisplayTime = kDigitMaxDisplayTime;
+		}
+	
+		// Write the currently displaying digit low and the rest high
+		for(int i = 0; i < 4; i++)
+				digitalWriteOnce(context, n, kPins[kDigits[i]], HIGH);
+		digitalWriteOnce(context, n, kPins[kDigits[gCurrentlyDisplayingDigit]], LOW);
+		
+		// Write the digit to the other outputs
+		digitalWriteOnce(context, n, kPins[11],
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x01);	// a
+		digitalWriteOnce(context, n, kPins[6], 
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x02);	// b
+		digitalWriteOnce(context, n, kPins[4], 
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x04);	// c
+		digitalWriteOnce(context, n, kPins[1],
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x08);	// d
+		digitalWriteOnce(context, n, kPins[0],
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x10);	// e
+		digitalWriteOnce(context, n, kPins[10],
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x20);	// f
+		digitalWriteOnce(context, n, kPins[5],
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x40);	// g
+		digitalWriteOnce(context, n, kPins[2],
+			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x80);	// .
+			
+		// Check for changing state
+		if(--gStateCounter <= 0) {
+			gState = (gState + 1) % kMaxState;
+			if(gState != (kMaxState - 1)) {
+				for(int i = 0; i < 4; i++)
+					gCharacterToDisplay[i] = 1 << (gState % 6);
+				gStateCounter = 2000;
+			}
+			else {
+				for(int i = 0; i < 4; i++)
+					gCharacterToDisplay[i] = kBELA[i];
+				gStateCounter = 50000;
+			}
+		}
+	}
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+	
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/11-Extras/cape-test/main.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,97 @@
+/*
+ * main.cpp
+ *
+ *  Created on: Oct 24, 2014
+ *      Author: parallels
+ */
+#include <unistd.h>
+#include <iostream>
+#include <cstdlib>
+#include <libgen.h>
+#include <signal.h>
+#include <getopt.h>
+#include <Bela.h>
+
+using namespace std;
+
+// Handle Ctrl-C by requesting that the audio rendering stop
+void interrupt_handler(int var)
+{
+	gShouldStop = true;
+}
+
+// Print usage information
+void usage(const char * processName)
+{
+	cerr << "Usage: " << processName << " [options]" << endl;
+
+	Bela_usage();
+
+	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
+	cerr << "   --help [-h]:                Print this menu\n";
+}
+
+int main(int argc, char *argv[])
+{
+	BelaInitSettings settings;	// Standard audio settings
+	float frequency = 440.0;	// Frequency of oscillator
+
+	struct option customOptions[] =
+	{
+		{"help", 0, NULL, 'h'},
+		{"frequency", 1, NULL, 'f'},
+		{NULL, 0, NULL, 0}
+	};
+
+	// Set default settings
+	Bela_defaultSettings(&settings);
+
+	// Parse command-line arguments
+	while (1) {
+		int c;
+		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
+				break;
+		switch (c) {
+		case 'h':
+				usage(basename(argv[0]));
+				exit(0);
+		case 'f':
+				frequency = atof(optarg);
+				break;
+		case '?':
+		default:
+				usage(basename(argv[0]));
+				exit(1);
+		}
+	}
+
+	// Initialise the PRU audio device
+	if(Bela_initAudio(&settings, &frequency) != 0) {
+		cout << "Error: unable to initialise audio" << endl;
+		return -1;
+	}
+
+	// Start the audio device running
+	if(Bela_startAudio()) {
+		cout << "Error: unable to start real-time audio" << endl;
+		return -1;
+	}
+
+	// Set up interrupt handler to catch Control-C and SIGTERM
+	signal(SIGINT, interrupt_handler);
+	signal(SIGTERM, interrupt_handler);
+
+	// Run until told to stop
+	while(!gShouldStop) {
+		usleep(100000);
+	}
+
+	// Stop the audio device
+	Bela_stopAudio();
+
+	// Clean up any resources allocated for audio
+	Bela_cleanupAudio();
+
+	// All done!
+	return 0;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/11-Extras/cape-test/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,279 @@
+/*
+ * render.cpp
+ *
+ *  Created on: Oct 24, 2014
+ *      Author: parallels
+ */
+
+
+#include <Bela.h>
+#include <cmath>
+
+#define ANALOG_LOW	(2048.0 / 65536.0)
+#define ANALOG_HIGH (50000.0 / 65536.0)
+
+const int gDACPinOrder[] = {6, 4, 2, 0, 1, 3, 5, 7};
+
+enum {
+	kStateTestingAudioLeft = 0,
+	kStateTestingAudioRight,
+	kStateTestingAudioDone
+};
+
+uint64_t gLastErrorFrame = 0;
+uint32_t gEnvelopeSampleCount = 0;
+float gEnvelopeValueL = 0.5, gEnvelopeValueR = 0.5;
+float gEnvelopeDecayRate = 0.9995;
+int gEnvelopeLastChannel = 0;
+
+float gPositivePeakLevels[2] = {0, 0};
+float gNegativePeakLevels[2] = {0, 0};
+float gPeakLevelDecayRate = 0.999;
+const float gPeakLevelLowThreshold = 0.02;
+const float gPeakLevelHighThreshold = 0.2;
+const float gDCOffsetThreshold = 0.1;
+int gAudioTestState = kStateTestingAudioLeft;
+int gAudioTestStateSampleCount = 0;
+int gAudioTestSuccessCounter = 0;
+const int gAudioTestSuccessCounterThreshold = 64;
+const int gAudioTestStateSampleThreshold = 16384;
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+bool setup(BelaContext *context, void *userData)
+{
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	static float phase = 0.0;
+	static int sampleCounter = 0;
+	static int invertChannel = 0;
+	float frequency = 0;
+
+	// Play a sine wave on the audio output
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		
+		// Peak detection on the audio inputs, with offset to catch
+		// DC errors
+		for(int ch = 0; ch < 2; ch++) {
+			if(context->audioIn[2*n + ch] > gPositivePeakLevels[ch])
+				gPositivePeakLevels[ch] = context->audioIn[2*n + ch];
+			gPositivePeakLevels[ch] += 0.1;
+			gPositivePeakLevels[ch] *= gPeakLevelDecayRate;
+			gPositivePeakLevels[ch] -= 0.1;
+			if(context->audioIn[2*n + ch] < gNegativePeakLevels[ch])
+				gNegativePeakLevels[ch] = context->audioIn[2*n + ch];
+			gNegativePeakLevels[ch] -= 0.1;			
+			gNegativePeakLevels[ch] *= gPeakLevelDecayRate;
+			gNegativePeakLevels[ch] += 0.1;
+		}
+		
+		if(gAudioTestState == kStateTestingAudioLeft) {
+			context->audioOut[2*n] = 0.2 * sinf(phase);
+			context->audioOut[2*n + 1] = 0;		
+			
+			frequency = 3000.0;
+			phase += 2.0 * M_PI * frequency / 44100.0;
+			if(phase >= 2.0 * M_PI)
+				phase -= 2.0 * M_PI;
+			
+			gAudioTestStateSampleCount++;
+			if(gAudioTestStateSampleCount >= gAudioTestStateSampleThreshold) {
+				// Check if we have the expected input: signal on the left but not
+				// on the right. Also check that there is not too much DC offset on the
+				// inactive signal
+				if((gPositivePeakLevels[0] - gNegativePeakLevels[0]) >= gPeakLevelHighThreshold 
+					&& (gPositivePeakLevels[1] -  gNegativePeakLevels[1]) <= gPeakLevelLowThreshold &&
+					fabsf(gPositivePeakLevels[1]) < gDCOffsetThreshold &&
+					fabsf(gNegativePeakLevels[1]) < gDCOffsetThreshold) {
+					// Successful test: increment counter
+					gAudioTestSuccessCounter++;
+					if(gAudioTestSuccessCounter >= gAudioTestSuccessCounterThreshold) {
+						gAudioTestState = kStateTestingAudioRight;
+						gAudioTestStateSampleCount = 0;
+						gAudioTestSuccessCounter = 0;
+					}
+
+				}
+				else {
+					if(!((context->audioFramesElapsed + n) % 22050)) {
+						// Debugging print messages
+						if((gPositivePeakLevels[0] - gNegativePeakLevels[0]) < gPeakLevelHighThreshold)
+							rt_printf("Left Audio In FAIL: insufficient signal: %f\n", 
+										gPositivePeakLevels[0] - gNegativePeakLevels[0]);
+						else if(gPositivePeakLevels[1] -  gNegativePeakLevels[1] > gPeakLevelLowThreshold)
+							rt_printf("Right Audio In FAIL: signal present when it should not be: %f\n",
+										gPositivePeakLevels[1] -  gNegativePeakLevels[1]);
+						else if(fabsf(gPositivePeakLevels[1]) >= gDCOffsetThreshold ||
+								fabsf(gNegativePeakLevels[1]) >= gDCOffsetThreshold)
+							rt_printf("Right Audio In FAIL: DC offset: (%f, %f)\n",
+										gPositivePeakLevels[1], gNegativePeakLevels[1]);						
+					}
+					gAudioTestSuccessCounter--;
+					if(gAudioTestSuccessCounter <= 0)
+						gAudioTestSuccessCounter = 0;
+				}
+			}
+		}
+		else if(gAudioTestState == kStateTestingAudioRight) {
+			context->audioOut[2*n] = 0;
+			context->audioOut[2*n + 1] = 0.2 * sinf(phase);
+			
+			frequency = 3000.0;
+			phase += 2.0 * M_PI * frequency / 44100.0;
+			if(phase >= 2.0 * M_PI)
+				phase -= 2.0 * M_PI;
+			
+			gAudioTestStateSampleCount++;
+			if(gAudioTestStateSampleCount >= gAudioTestStateSampleThreshold) {
+				// Check if we have the expected input: signal on the left but not
+				// on the right
+				if((gPositivePeakLevels[1] - gNegativePeakLevels[1]) >= gPeakLevelHighThreshold 
+					&& (gPositivePeakLevels[0] -  gNegativePeakLevels[0]) <= gPeakLevelLowThreshold &&
+					fabsf(gPositivePeakLevels[0]) < gDCOffsetThreshold &&
+					fabsf(gNegativePeakLevels[0]) < gDCOffsetThreshold) {
+					// Successful test: increment counter
+					gAudioTestSuccessCounter++;
+					if(gAudioTestSuccessCounter >= gAudioTestSuccessCounterThreshold) {
+						gAudioTestSuccessCounter = 0;							
+						gAudioTestStateSampleCount = 0;
+						gAudioTestState = kStateTestingAudioDone;
+					}
+				}
+				else {
+					if(!((context->audioFramesElapsed + n) % 22050)) {
+						// Debugging print messages
+						if((gPositivePeakLevels[1] - gNegativePeakLevels[1]) < gPeakLevelHighThreshold)
+							rt_printf("Right Audio In FAIL: insufficient signal: %f\n", 
+										gPositivePeakLevels[1] - gNegativePeakLevels[1]);
+						else if(gPositivePeakLevels[0] -  gNegativePeakLevels[0] > gPeakLevelLowThreshold)
+							rt_printf("Left Audio In FAIL: signal present when it should not be: %f\n",
+										gPositivePeakLevels[0] -  gNegativePeakLevels[0]);
+						else if(fabsf(gPositivePeakLevels[0]) >= gDCOffsetThreshold ||
+								fabsf(gNegativePeakLevels[0]) >= gDCOffsetThreshold)
+							rt_printf("Left Audio In FAIL: DC offset: (%f, %f)\n",
+										gPositivePeakLevels[0], gNegativePeakLevels[0]);						
+					}
+					gAudioTestSuccessCounter--;
+					if(gAudioTestSuccessCounter <= 0)
+						gAudioTestSuccessCounter = 0;
+				}
+			}			
+		}
+		else {
+			// Audio input testing finished. Play tones depending on status of
+			// analog testing
+			context->audioOut[2*n] = gEnvelopeValueL * sinf(phase);
+			context->audioOut[2*n + 1] = gEnvelopeValueR * sinf(phase);
+
+			// If one second has gone by with no error, play one sound, else
+			// play another
+			if(context->audioFramesElapsed + n - gLastErrorFrame > 44100) {
+				gEnvelopeValueL *= gEnvelopeDecayRate;
+				gEnvelopeValueR *= gEnvelopeDecayRate;
+				gEnvelopeSampleCount++;
+				if(gEnvelopeSampleCount > 22050) {
+					if(gEnvelopeLastChannel == 0)
+						gEnvelopeValueR = 0.5;
+					else
+						gEnvelopeValueL = 0.5;
+					gEnvelopeLastChannel = !gEnvelopeLastChannel;
+					gEnvelopeSampleCount = 0;
+				}
+				frequency = 880.0;
+			}
+			else {
+				gEnvelopeValueL = gEnvelopeValueR = 0.5;
+				gEnvelopeLastChannel = 0;
+				frequency = 220.0;
+			}
+
+			phase += 2.0 * M_PI * frequency / 44100.0;
+			if(phase >= 2.0 * M_PI)
+				phase -= 2.0 * M_PI;
+		}
+	}
+
+	for(unsigned int n = 0; n < context->analogFrames; n++) {
+		// Change outputs every 512 samples
+		if(sampleCounter < 512) {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel)
+					context->analogOut[n*8 + gDACPinOrder[k]] = ANALOG_HIGH;
+				else
+					context->analogOut[n*8 + gDACPinOrder[k]] = 0;
+			}
+		}
+		else {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel)
+					context->analogOut[n*8 + gDACPinOrder[k]] = 0;
+				else
+					context->analogOut[n*8 + gDACPinOrder[k]] = ANALOG_HIGH;
+			}
+		}
+
+		// Read after 256 samples: input should be low
+		if(sampleCounter == 256) {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel) {
+					if(context->analogIn[n*8 + k] < ANALOG_HIGH) {
+						rt_printf("FAIL [output %d, input %d] -- output HIGH input %f (inverted)\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
+						gLastErrorFrame = context->audioFramesElapsed + n;
+					}
+				}
+				else {
+					if(context->analogIn[n*8 + k] > ANALOG_LOW) {
+						rt_printf("FAIL [output %d, input %d] -- output LOW --> input %f\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
+						gLastErrorFrame = context->audioFramesElapsed + n;
+					}
+				}
+			}
+		}
+		else if(sampleCounter == 768) {
+			for(int k = 0; k < 8; k++) {
+				if(k == invertChannel) {
+					if(context->analogIn[n*8 + k] > ANALOG_LOW) {
+						rt_printf("FAIL [output %d, input %d] -- output LOW input %f (inverted)\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
+						gLastErrorFrame = context->audioFramesElapsed + n;
+					}
+				}
+				else {
+					if(context->analogIn[n*8 + k] < ANALOG_HIGH) {
+						rt_printf("FAIL [output %d, input %d] -- output HIGH input %f\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
+						gLastErrorFrame = context->audioFramesElapsed + n;
+					}
+				}
+			}
+		}
+
+		if(++sampleCounter >= 1024) {
+			sampleCounter = 0;
+			invertChannel++;
+			if(invertChannel >= 8)
+				invertChannel = 0;
+		}
+	}
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/11-Extras/gpioAnalogLoopbackTest/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,209 @@
+#include <Bela.h>
+#include <cmath>
+#include <rtdk.h>
+#include <sys/time.h>
+#include <sys/types.h>
+#include <unistd.h>
+#include <stats.hpp>
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+
+// digital inputs can be changed at will (as they are all being processed at the same time)
+// analog channels must be as per below
+int gAnalogOutCh = 1;
+int gDigitalOutCh = 2;
+int gDigitalInACh = 0;
+int gDigitalInDCh = 3;
+int gAnalogOutLoopDelay;
+int gDigitalOutLoopDelay;
+bool setup(BelaContext *context, void *userData)
+{
+	rt_printf("For this test you need the following connections:\n"
+			"analog%d out->digital%d in, analog%d out->analog%d in, "
+			"digital%d out -> digital%d in, digital%d out-> analog%d in\n",
+			gAnalogOutCh, gDigitalInACh, gAnalogOutCh, 0, gDigitalOutCh, gDigitalInDCh, gDigitalOutCh, 0);
+	rt_printf("Running test with %d analog channels and a buffer size of %d\n",
+			context->analogChannels, context->audioFrames);
+
+	for(unsigned int n = 0; n < context->digitalFrames; n++){
+		pinMode(context, n, gDigitalInACh, INPUT);
+		pinMode(context, n, gDigitalInDCh, INPUT);
+		pinMode(context, n, gDigitalOutCh, OUTPUT);
+	}
+	switch (context->analogChannels){
+		case 2:
+			gAnalogOutLoopDelay = context->audioFrames*2 + 3;
+			gDigitalOutLoopDelay = context->audioFrames*2 + 2;
+			break;
+		case 4:
+			gAnalogOutLoopDelay = context->audioFrames*2 + 3;
+			gDigitalOutLoopDelay = context->audioFrames*2 + 2;
+			break;
+		case 8:
+			gAnalogOutLoopDelay = context->audioFrames + 3;
+			gDigitalOutLoopDelay = context->audioFrames + 1;
+			break;
+		default:
+			exit(2);
+	}
+
+    return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numAnalogFrames
+// will be 0.
+
+const int patternLength = 31;
+static int anaErrorCount = 0;
+static int digErrorCount = 0;
+void render(BelaContext *context, void *userData)
+{
+	static bool writePattern[patternLength] = {
+									0,1,0,1,0,0,1,1,
+									0,0,0,1,1,1,0,0,
+									1,1,1,1,0,0,0,0,
+									1,1,1,1,1,0,0};
+//	for(int n = 0; n < patternLength; n++){
+//		writePattern[n]=1;
+//	}
+	static int inPointer = 0;
+	static int outPointer = 0;
+	static int digitalOutPointer = 0;
+	static int digitalInPointer = 0;
+	static int analogOut = 1;
+	/** Checking offset between analog and digital
+	 * how it should be :
+	 * The PRU loop does the following (the loop runs at 88.2kHz):
+	 * - Read/write audio sample (once for the left channel, once for the right channel)
+	 * - Write DAC 0 or 0/2 or 0/2/4/6
+	 * - Read ADC 0 or 0/2 or 0/2/4/6, 2 samples (@176.4) older than NOW
+	 * - /During/ the line above, every two loops we also Read/Write GPIO,
+	 * therefore reading on ADC 0/2/4/6 a value that is being output from GPIO will lead to undefined results
+	 * - Write DAC 1 or 1/3 or 1/3/5/7
+	 * - Read ADC 1 or 1/3 or 1/3/5/7, 2 samples (@176.4) older than NOW
+	 */
+	if(1)
+	for(unsigned int n = 0; n < context->audioFrames; n++){
+		static bool analog0In = false;
+		static bool digitalAIn = false;
+		static int count = 0;
+		bool doReadWrite = context->analogChannels<=4 ? true : ((context->analogChannels == 8) && (n&1)==0);
+		if(doReadWrite){
+			digitalAIn = digitalRead(context, n, gDigitalInACh);
+			switch(context->analogChannels){
+			case 8:
+				analog0In = analogRead(context, n/2, 0) > 0.5;
+				analogWrite(context, n/2, analogOut, writePattern[outPointer]);
+				break;
+			case 4:
+				analog0In = analogRead(context, n, 0) > 0.5;
+				analogWrite(context, n, analogOut, writePattern[outPointer]);
+				break;
+			case 2:
+				analog0In = analogRead(context, n * 2 + 1, 0) > 0.5;
+				analogWrite(context, 2 * n, analogOut, writePattern[outPointer]);
+				analogWrite(context, 2 * n + 1, analogOut, writePattern[outPointer]);
+				break;
+			}
+			gAnalogOutLoopDelay--;
+			outPointer++;
+			if(gAnalogOutLoopDelay <= 0){
+				if(++inPointer == patternLength){
+					inPointer = 0;
+				}
+			}
+		}
+		bool expectedIn = writePattern[inPointer];
+		if(gAnalogOutLoopDelay <= 0 && doReadWrite == true){
+			if(analog0In != expectedIn || digitalAIn != expectedIn){
+				rt_printf("expected: %d, received: %d %d, pointer: %d, delay: %d, count: %d\n",
+					expectedIn, analog0In, digitalAIn, inPointer, gAnalogOutLoopDelay, count);
+				anaErrorCount++;
+			}
+		}
+		count++;
+		if(analog0In != digitalAIn){ // at any time the analog and digital in should be the same
+			rt_printf("ana %d_%d %d,\n", analog0In, digitalAIn, n);
+		}
+		if(outPointer == patternLength){
+			outPointer = 0;
+		}
+	}
+	if(1)
+	for(unsigned int n = 0; n < context->audioFrames; n++){
+		static int count = 0;
+		static bool analog1In = false;
+		static bool digitalDIn = false;
+/* we need to remember the pastAnalog1In because
+ *  reading GPIO takes place before writing to it, therefore
+ *  when reading a GPIOout, the GPIOin samples will always be one sample late
+ */
+		bool doReadWrite = false;
+		static bool pastAnalog1In = false;
+		digitalWriteOnce(context, n, gDigitalOutCh,  writePattern[digitalOutPointer]);
+		if(context->analogChannels == 8){
+			if((n&1) == 0){ //do it every other sample
+				pastAnalog1In = analogRead(context, n/2, 1) > 0.5;
+				digitalDIn = digitalRead(context, n, gDigitalInDCh);
+				doReadWrite = true;
+			}
+		}
+		if(context->analogChannels == 4){
+			pastAnalog1In = analogRead(context, n, 1) > 0.5;
+			digitalDIn = digitalRead(context, n, gDigitalInDCh);
+			digitalWriteOnce(context, n, gDigitalOutCh,  writePattern[digitalOutPointer]);
+			doReadWrite = true;
+		}
+		if(context->analogChannels == 2){
+			pastAnalog1In = analogRead(context, n * 2, 1) > 0.5;
+			digitalDIn = digitalRead(context, n, gDigitalInDCh);
+			digitalWriteOnce(context, n, gDigitalOutCh,  writePattern[digitalOutPointer]);
+			doReadWrite = true;
+		}
+		bool expectedDigitalIn = writePattern[digitalInPointer];
+		if(doReadWrite == true){
+			gDigitalOutLoopDelay--;
+			if(gDigitalOutLoopDelay <= 0){
+				if(expectedDigitalIn != pastAnalog1In || expectedDigitalIn != digitalDIn){
+					rt_printf("D expected: %d, received: %d %d, pointer: %d, delay: %d, count: %d\n",
+						expectedDigitalIn, pastAnalog1In, digitalDIn, inPointer, gDigitalOutLoopDelay, count);
+					digErrorCount++;
+				}
+				if(++digitalInPointer == patternLength){
+					digitalInPointer = 0;
+				}
+			}
+			pastAnalog1In = analog1In;
+			if(++digitalOutPointer == patternLength){
+				digitalOutPointer = 0;
+			}
+		}
+		count++;
+	}
+	if(context->audioFramesElapsed > 30000){
+		gShouldStop = true;
+	}
+}
+
+
+void cleanup(BelaContext *context, void *userData)
+{
+	if(anaErrorCount == 0 && digErrorCount == 0){
+		rt_printf("Test was succesful with %d analog channels and a buffer size of %d\n", context->analogChannels, context->audioFrames);
+	} else {
+		rt_printf("------------------------\n%danalog %ddigital errors over %dsamples while running test with ",
+				anaErrorCount, digErrorCount, context->audioFramesElapsed);
+		rt_printf("%d analog channels and a buffer size of %d \n\n\n",
+				context->analogChannels, context->audioFrames);
+		exit(1);
+	}
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/examples/11-Extras/stepper/render.cpp	Mon Jun 20 17:08:02 2016 +0100
@@ -0,0 +1,166 @@
+/*
+ ____  _____ _        _    
+| __ )| ____| |      / \   
+|  _ \|  _| | |     / _ \  
+| |_) | |___| |___ / ___ \ 
+|____/|_____|_____/_/   \_\
+
+The platform for ultra-low latency audio and sensor processing
+
+http://bela.io
+
+A project of the Augmented Instruments Laboratory within the
+Centre for Digital Music at Queen Mary University of London.
+http://www.eecs.qmul.ac.uk/~andrewm
+
+(c) 2016 Augmented Instruments Laboratory: Andrew McPherson,
+  Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack,
+  Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved.
+
+The Bela software is distributed under the GNU Lesser General Public License
+(LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt
+*/
+
+
+#include <Bela.h>
+
+const int kStepLengthSlow = 1000;
+const int kStepLengthFast = 500;
+
+int gStepLengthSamples = kStepLengthSlow;
+
+const int gPinA1 = P8_27;
+const int gPinA2 = P8_28;
+const int gPinB1 = P8_29;
+const int gPinB2 = P8_30;
+const int gPinServo = P9_16;
+
+int gStepCounter = 0;
+int gPhase = 0;
+
+int gServoCounter = 0;
+
+
+enum {
+	kStateMoveRight1 = 0,
+	kStateMoveLeft1,
+	kStateMoveRight2,
+	kStateMoveLeft2,
+	kStateMoveRight3,
+	kStateMoveLeft3,
+	kStateSpin,
+	kStateMax
+};
+
+int gState = 0;
+int gStateCounter = 0;
+
+// setup() is called once before the audio rendering starts.
+// Use it to perform any initialisation and allocation which is dependent
+// on the period size or sample rate.
+//
+// userData holds an opaque pointer to a data structure that was passed
+// in from the call to initAudio().
+//
+// Return true on success; returning false halts the program.
+
+bool setup(BelaContext *context, void *userData)
+{	
+	// This project makes the assumption that the audio and digital
+	// sample rates are the same. But check it to be sure!
+	if(context->audioFrames != context->digitalFrames) {
+		rt_printf("Error: this project needs the audio and digital sample rates to be the same.\n");
+		return false;
+	}
+	
+	pinMode(context, 0, gPinA1, OUTPUT);
+	pinMode(context, 0, gPinA2, OUTPUT);
+	pinMode(context, 0, gPinB1, OUTPUT);
+	pinMode(context, 0, gPinB2, OUTPUT);
+	pinMode(context, 0, gPinServo, OUTPUT);
+		
+	return true;
+}
+
+// render() is called regularly at the highest priority by the audio engine.
+// Input and output are given from the audio hardware and the other
+// ADCs and DACs (if available). If only audio is available, numMatrixFrames
+// will be 0.
+
+void render(BelaContext *context, void *userData)
+{
+	for(unsigned int n = 0; n < context->audioFrames; n++) {
+		if(gPhase == 0 || gPhase == 1) {
+			digitalWriteOnce(context, n, gPinB1, HIGH);
+			digitalWriteOnce(context, n, gPinB2, LOW);
+		}
+		else {
+			digitalWriteOnce(context, n, gPinB1, LOW);
+			digitalWriteOnce(context, n, gPinB2, HIGH);			
+		}
+		
+		if(gPhase == 1 || gPhase == 2) {
+			digitalWriteOnce(context, n, gPinA1, HIGH);
+			digitalWriteOnce(context, n, gPinA2, LOW);
+		}
+		else {
+			digitalWriteOnce(context, n, gPinA1, LOW);
+			digitalWriteOnce(context, n, gPinA2, HIGH);			
+		}
+		
+		if(--gServoCounter > 0) 
+			digitalWriteOnce(context, n, gPinServo, HIGH);
+		else
+			digitalWriteOnce(context, n, gPinServo, LOW);
+		
+		if(++gStepCounter >= gStepLengthSamples) {
+			gStateCounter++;
+			
+			switch(gState) {
+				case kStateMoveRight1:
+				case kStateMoveRight2:
+				case kStateMoveRight3:
+					gPhase = (gPhase + 1) & 3;
+					break;
+				case kStateMoveLeft1:
+				case kStateMoveLeft2:
+				case kStateMoveLeft3:				
+					gPhase = (gPhase + 3) & 3;	
+					break;
+				case kStateSpin:
+					gPhase = (gPhase + 1) & 3;
+					break;		
+			}
+			
+			if(gState == kStateSpin) {
+				if(gStateCounter >= 48) {
+					gStateCounter = 0;
+					gState = 0;
+					gStepLengthSamples = kStepLengthSlow;
+				}				
+			}
+			else {
+				if(gStateCounter >= 16) {
+					gStateCounter = 0;
+					gState++;
+					if(gState & 1)
+						gServoCounter = 120;
+					else
+						gServoCounter = 80;
+					if(gState == kStateSpin)
+						gStepLengthSamples = kStepLengthFast;
+				}
+			}
+			
+			gStepCounter = 0;
+		}
+	}
+}
+
+// cleanup() is called once at the end, after the audio has stopped.
+// Release any resources that were allocated in setup().
+
+void cleanup(BelaContext *context, void *userData)
+{
+	
+}
--- a/examples/7segment/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,126 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-
-#include <Bela.h>
-
-#define NUM_PINS 12
-
-// Breadboard wiring layout:
-// 11 10 12 9  8  7
-// [ LED DISP ]
-// 1  2  3  6  4  5
-
-// Organised by display segments:
-// e d . X c g b X X X f a
-const int kPins[NUM_PINS] = {P8_07, P8_08, P8_09, P8_10, P8_11, P8_12,
-							 P8_15, P8_16, P8_27, P8_28, P8_29, P8_30};
-
-// Indices into the above array: pins 12, 9, 8, 6
-const int kDigits[4] = {9, 8, 7, 3};
-
-int gCurrentlyDisplayingDigit = 0;
-int gDigitDisplayTime = 0;
-const int kDigitMaxDisplayTime = 44;
-
-int gState = 0;
-int gStateCounter = 0;
-const int kMaxState = 25;
-
-// . g f e d c b a
-//const unsigned char kBELA[4] = {0x7C, 0x79, 0x38, 0x77};
-const unsigned char kBELA[4] = {0x7C, 0x7B, 0x38, 0x5F};
-const unsigned char kPerimeter[6] = {0x01, 0x02, 0x04, 0x08, 0x10, 0x20};
-
-int gCharacterToDisplay[4] = {0, 0, 0, 0};
-	
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{	
-	// This project makes the assumption that the audio and digital
-	// sample rates are the same. But check it to be sure!
-	if(context->audioFrames != context->digitalFrames) {
-		rt_printf("Error: this project needs the audio and digital sample rates to be the same.\n");
-		return false;
-	}
-	
-	for(int i = 0; i < NUM_PINS; i++) {
-		pinMode(context, 0, kPins[i], OUTPUT);
-	}
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		// Check for rotation between digits
-		if(--gDigitDisplayTime <= 0) {
-			gCurrentlyDisplayingDigit = (gCurrentlyDisplayingDigit + 1) % 4;
-			gDigitDisplayTime = kDigitMaxDisplayTime;
-		}
-	
-		// Write the currently displaying digit low and the rest high
-		for(int i = 0; i < 4; i++)
-				digitalWriteOnce(context, n, kPins[kDigits[i]], HIGH);
-		digitalWriteOnce(context, n, kPins[kDigits[gCurrentlyDisplayingDigit]], LOW);
-		
-		// Write the digit to the other outputs
-		digitalWriteOnce(context, n, kPins[11],
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x01);	// a
-		digitalWriteOnce(context, n, kPins[6], 
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x02);	// b
-		digitalWriteOnce(context, n, kPins[4], 
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x04);	// c
-		digitalWriteOnce(context, n, kPins[1],
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x08);	// d
-		digitalWriteOnce(context, n, kPins[0],
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x10);	// e
-		digitalWriteOnce(context, n, kPins[10],
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x20);	// f
-		digitalWriteOnce(context, n, kPins[5],
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x40);	// g
-		digitalWriteOnce(context, n, kPins[2],
-			gCharacterToDisplay[gCurrentlyDisplayingDigit] & 0x80);	// .
-			
-		// Check for changing state
-		if(--gStateCounter <= 0) {
-			gState = (gState + 1) % kMaxState;
-			if(gState != (kMaxState - 1)) {
-				for(int i = 0; i < 4; i++)
-					gCharacterToDisplay[i] = 1 << (gState % 6);
-				gStateCounter = 2000;
-			}
-			else {
-				for(int i = 0; i < 4; i++)
-					gCharacterToDisplay[i] = kBELA[i];
-				gStateCounter = 50000;
-			}
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	
-}
--- a/examples/airharp/Junction.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,95 +0,0 @@
-/*
- *
- * Excitation Junction for two waveguides
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-#include "Junction.h"
-#include "../include/Utilities.h"
-
-Junction::Junction()	{
-
-	setFrequency(440);
-	_dt = 1.0/44100.0;
-
-	// initialize variables
-	for(int i=0;i<WG_BUFFER_SIZE;i++)	{
-		_buffer_l[i] = 0;
-		_buffer_r[i] = 0;
-	}
-	_excitation = 0;
-	_lastPlectrumDisplacement = 0;
-	_readPtr = 0;
-
-}
-
-void Junction::update(float excitation, float left, float right)	{
-
-	// 1. advance delay buffer read pointer
-
-	if(++_readPtr>=WG_BUFFER_SIZE)
-		_readPtr=0;
-
-	// 2. add excitation sample into buffer
-
-	_buffer_l[(_readPtr+_delay_l+WG_BUFFER_SIZE)%WG_BUFFER_SIZE] = excitation;
-	_buffer_r[(_readPtr+_delay_r+WG_BUFFER_SIZE)%WG_BUFFER_SIZE] = excitation;
-
-	// 3. feed right input to left output and vice versa
-
-	_buffer_l[_readPtr] += right;
-	_buffer_r[_readPtr] += left;
-
-	// 4. store excitation value for later use
-	_excitation = excitation;
-
-}
-
-float Junction::getOutput(int direction)	{
-
-	if(direction = 0)
-		return _buffer_l[_readPtr];
-	else
-		return _buffer_r[_readPtr];
-
-}
-
-float Junction::getExcitationDisplacement()	{
-
-	// string displacement and excitation force
-	// use delayed value to account for excitation position
-	float in = _buffer_l[(_readPtr+_delay_l+WG_BUFFER_SIZE)%WG_BUFFER_SIZE] + _excitation;
-
-	// integrate total force
-	float out = 0.00001 * in + 0.99999 * _lastPlectrumDisplacement;
-
-	// store variable for next iteration
-	_lastPlectrumDisplacement = out;
-
-	// multiply by delta time
-	return out * _dt;
-
-}
-
-void Junction::setPluckPosition(float pluckPos){
-
-	pluckPos = constrain(pluckPos,0,1);
-	_delay_l = pluckPos * _periodInSamples;
-	_delay_r = (1-pluckPos) * _periodInSamples;
-
-}
-
-void Junction::setPeriod(float period)	{
-
-	_periodInMilliseconds = period;
-
-}
-
-void Junction::setFrequency(float frequency)	{
-
-	_periodInMilliseconds = 1000.0/frequency;
-	_periodInSamples = (int)(_periodInMilliseconds * 44.1);
-
-}
--- a/examples/airharp/Junction.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,54 +0,0 @@
-/*
- *
- * Excitation Junction for two waveguides
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-#ifndef JUNCTION_H_
-#define JUNCTION_H_
-
-#include <cmath>
-
-#ifndef WG_BUFFER_SIZE
-#define WG_BUFFER_SIZE 4096
-#endif
-
-#ifndef		M_PI
-#define		M_PI		3.14159265358979323846264338
-#endif
-
-class Junction
-{
-
-public:
-
-	Junction();
-	void setup();
-	void update(float excitation, float left, float right);
-	float getOutput(int direction);
-	float getExcitationDisplacement();
-	void setFrequency(float frequency);
-	void setPeriod(float periodInMs);
-	void setPluckPosition(float pluckPos);
-
-private:
-
-	double _dt;
-	float _periodInMilliseconds;
-	int _periodInSamples;
-
-	int _delay_l;
-	int _delay_r;
-
-	float _buffer_l[WG_BUFFER_SIZE];
-	float _buffer_r[WG_BUFFER_SIZE];
-	int _readPtr;
-
-	float _excitation;
-	float _lastPlectrumDisplacement;
-
-};
-
-#endif
--- a/examples/airharp/MassSpringDamper.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,59 +0,0 @@
-/*
- *
- * Simple 1-Dimensional Mass Spring Damper
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-#include "MassSpringDamper.h"
-
-MassSpringDamper::MassSpringDamper(float mass, float spring, float damp)	{
-
-	_dt = 1.0/44100.0;
-	_mass = mass;
-	_spring = spring;
-	_damp = damp;
-	_position = 0;
-	_velocity = 0;
-
-}
-
-void MassSpringDamper::setup()	{
-
-}
-
-double MassSpringDamper::update(float inForce)	{
-
-	// 1. calculate spring/damper forces using current position and velocity
-
-	double out = (_position * (double)_spring * -1) + (_velocity * (double)_damp * -1);
-
-	// 2. apply external force
-
-	out += inForce;
-
-	// 3. derive acceleration (a = f/m)
-
-	out /= (double)_mass;
-
-	// 4. derive velocity (v = a*dt)
-
-	out *= _dt;
-
-	// 5. apply previous velocity
-
-	out += _velocity;
-
-	// 6. save current velocity state for next iteration
-
-	_velocity = out;
-
-	// 7. derive new position (x[n] = x[n-1] + v[n]) and save for next iteration
-
-	out += _position;
-	_position = out;
-
-	return out;
-
-}
--- a/examples/airharp/MassSpringDamper.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,32 +0,0 @@
-/*
- *
- * Simple 1-Dimensional Mass Spring Damper
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-#ifndef MASSSPRINGDAMPER_H_
-#define MASSSPRINGDAMPER_H_
-
-class MassSpringDamper
-{
-
-public:
-
-	MassSpringDamper(float mass, float spring, float damp);
-	void setup();
-	double update(float inForce);
-
-private:
-
-	double _dt;
-	float _mass;
-	float _spring;
-	float _damp;
-	double _position;
-	double _velocity;
-
-};
-
-#endif
--- a/examples/airharp/Plectrum.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,61 +0,0 @@
-/*
- *
- * Plectrum model for touching and plucking strings
- *
- * Christian Heinrichs 04/2015
- *
- * [inspired by E. Berdahl's pluck~ abstraction for the FireFader]
- *
- */
-
-#include "Plectrum.h"
-
-#include "../include/Utilities.h"
-#include <cmath>
-#include <stdio.h>
-#include <cstdlib>
-
-Plectrum::Plectrum()	{
-
-	_contact = 0;
-	_lastDistance = 0;
-
-}
-
-void Plectrum::setup(float spring, float damp, float hyst)	{
-
-	_spring = spring;
-	_damp = damp;
-	_hyst = hyst;
-
-}
-
-float Plectrum::update(float position, float stringPosition)	{
-
-	float distance = position - stringPosition;
-
-	// Calculate spring/damp forces based on distance to string
-
-	float springOut = distance * _spring;
-
-	float dampOut = (distance - _lastDistance) * 44100;
-
-	float out = springOut+dampOut;
-
-	// If distance crosses zero, enable contact
-
-	if((distance>0 && _lastDistance<=0)||(distance<0 && _lastDistance>=0))
-		_contact = 1;
-
-	// If distance exceeds hysteresis threshold, jump to zero (i.e. 'pluck')
-
-	if(fabs(distance)>_hyst)
-		_contact = 0;
-
-	// FIXME: contact doesn't switch back to zero if distance goes back in original direction
-
-	_lastDistance = distance;
-
-	return out * _contact;
-
-}
--- a/examples/airharp/Plectrum.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,35 +0,0 @@
-/*
- *
- * Plectrum model for touching and plucking strings
- *
- * [inspired by E. Berdahl's pluck~ abstraction for the FireFader]
- *
- */
-
-#ifndef PLECTRUM_H_
-#define PLECTRUM_H_
-
-class Plectrum
-{
-
-public:
-
-	Plectrum();
-	void setup(float spring, float damp, float hyst);
-	float update(float position, float stringPosition);
-
-private:
-
-	double _dt;
-	float _spring;
-	float _damp;
-	double _position;
-	double _velocity;
-	float _hyst;
-	float _lastDistance;
-	int _contact;
-
-};
-
-#endif
-
--- a/examples/airharp/String.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,91 +0,0 @@
-/*
- *
- * 1-D string consisting of two waveguides and junction
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-#include "String.h"
-#include "Junction.h"
-#include "Waveguide.h"
-
-#include "../include/Utilities.h"
-#include <rtdk.h>
-#include <cmath>
-#include <stdio.h>
-#include <cstdlib>
-
-String::String(){
-
-	wg_l = Waveguide();
-	wg_r = Waveguide();
-	junction = Junction();
-
-	junction.setPluckPosition(0.5);
-
-	_previous_l = 0;
-	_previous_r = 0;
-
-}
-
-float String::update(float in)	{
-
-	// 1. send excitation signal and previous waveguide outputs into junction
-
-	junction.update(in,_previous_l,_previous_r);
-
-	// 2. save new waveguide outputs for next iteration
-
-	_previous_l = wg_l.update(junction.getOutput(0));
-	_previous_r = wg_r.update(junction.getOutput(1));
-
-	// 3. use right waveguide as output
-
-	//rt_printf("BANANA %f ",_readPtr);
-	//rt_printf("%f\n",_previous_r);
-
-	return _previous_r;
-}
-
-float String::getPlectrumDisplacement()	{
-
-	return junction.getExcitationDisplacement();
-
-}
-
-void String::setPluckPosition(float pluckPos){
-
-	junction.setPluckPosition(pluckPos);
-
-}
-
-void String::setGlobalPosition(float pos)	{
-
-	_globalPosition = pos;
-
-}
-
-float String::getGlobalPosition()	{
-
-	return _globalPosition;
-
-}
-
-void String::setMidinote(float midinote)	{
-
-	float frequency = 440.0f*(float)powf(2,(midinote-57)/12.0f);
-
-	junction.setFrequency(frequency);
-	wg_l.setFrequency(frequency);
-	wg_r.setFrequency(frequency);
-
-}
-
-void String::setFrequency(float frequency)	{
-
-	junction.setFrequency(frequency);
-	wg_l.setFrequency(frequency);
-	wg_r.setFrequency(frequency);
-
-}
--- a/examples/airharp/String.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,45 +0,0 @@
-/*
- *
- * 1-D string consisting of two waveguides and junction
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-#ifndef STRING_H_
-#define STRING_H_
-
-#include <cmath>
-#include "Junction.h"
-#include "Waveguide.h"
-
-class String
-{
-
-public:
-
-	String();
-	float update(float in);
-
-	float getPlectrumDisplacement();
-	void setMidinote(float midinote);
-	void setFrequency(float frequency);
-	void setPeriod(float periodInMs);
-	void setPluckPosition(float pluckPos);
-	void setGlobalPosition(float pos);
-	float getGlobalPosition();
-
-private:
-
-	float _previous_l;
-	float _previous_r;
-
-	float _globalPosition;
-
-	Waveguide wg_l;
-	Waveguide wg_r;
-	Junction junction;
-
-};
-
-#endif
--- a/examples/airharp/Waveguide.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,128 +0,0 @@
-/*
- *
- * Simple 1-Dimensional Waveguide
- *
- */
-
-#include "Waveguide.h"
-#include "../include/Utilities.h"
-#include <rtdk.h>
-#include <cmath>
-#include <stdio.h>
-#include <cstdlib>
-
-#define DECAY 0.995//0.999
-#define DAMPING 0.01//0.05
-
-// TODO: make damping and decay parametrisable
-
-Waveguide::Waveguide()	{
-
-	// initialize variables
-	a1_lp = 0;
-	a2_lp = 0;
-	b0_lp = 0;
-	b1_lp = 0;
-	b2_lp = 0;
-	_dt = 1.0/44100.0;
-	setFrequency(440);
-	updateFilterCoeffs(8000);
-	_filterReadPtr=0;
-	for(int i=0;i<FILTER_BUFFER_SIZE;i++)	{
-		_filterBuffer_x[i] = 0;
-		_filterBuffer_y[i] = 0;
-	}
-	for(int i=0;i<WG_BUFFER_SIZE;i++)	{
-		_buffer[i] = 0;
-	}
-	_lastX = 0;
-	_lastY = 0;
-	_readPtr = 0;
-
-}
-
-void Waveguide::setup()	{
-
-}
-
-float Waveguide::update(float in)	{
-
-	// 1. advance delay buffer read pointer
-
-	if(++_readPtr>=WG_BUFFER_SIZE)
-		_readPtr=0;
-
-	// 2. write input into buffer
-
-	_buffer[_readPtr] = in;
-
-	// 3. read delayed sample from buffer
-
-	float out = _buffer[(_readPtr-_periodInSamples+WG_BUFFER_SIZE)%WG_BUFFER_SIZE];
-
-	// 4. apply damping (low-pass) filter to output
-
-	if(++_filterReadPtr>=FILTER_BUFFER_SIZE)
-			_filterReadPtr=0;
-
-	out = b0_lp*out +
-					b1_lp*_filterBuffer_x[(_filterReadPtr-1+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE] +
-					b2_lp*_filterBuffer_x[(_filterReadPtr-2+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE] -
-					a1_lp*_filterBuffer_y[(_filterReadPtr-1+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE] -
-					a2_lp*_filterBuffer_y[(_filterReadPtr-2+FILTER_BUFFER_SIZE)%FILTER_BUFFER_SIZE];
-
-	// 5. Simple high-pass filter to block DC-offset
-	// y[n] = x[n] - x[n-1] + a * y[n-1]
-	float gain = 0.9999;
-	float temp = out;
-	out = out - _lastX + gain * _lastY;
-	_lastY = out;
-	_lastX = temp;
-
-	// 6. Apply intensity damping
-	out *= DECAY;
-
-	_filterBuffer_x[_filterReadPtr] = in;
-	_filterBuffer_y[_filterReadPtr] = out;
-
-	return out;
-
-}
-
-void Waveguide::setFrequency(float frequency)	{
-
-	// NB: currently no interpolation, so may not be ideal for dynamically changing waveguide frequency
-	_periodInMilliseconds = 1000.0/frequency;
-	_periodInSamples = (int)(_periodInMilliseconds * 44.1);
-
-}
-
-void Waveguide::updateFilterCoeffs(float frequency)	{
-
-	// FIXME: Butterworth filter doesn't work very well,
-	//		  using simple FIR in the meantime
-
-	a1_lp = 0;
-	a2_lp = 0;
-	b0_lp = 1.0 - DAMPING;
-	b1_lp = DAMPING;
-	b2_lp = 0;
-
-	/*
-	// 'w' for sake of resembling lower-case 'omega'
-	float w = 2.0 * M_PI * frequency;
-	float t = _dt;
-	// The Q for a 2nd-order Butterworth is sqrt(2)/2
-	float q = 0.707;//sqrt(2.0)/2.0;
-
-	// low-pass filter coefficients
-	float a0_lp = w*w*t*t + 2*(w/q)*t + 4.0;
-	float k = 1.0/a0_lp;
-	a1_lp = (2.0*w*w*t*t - 8.0) * k;
-	a2_lp = (4.0 - (w/q)*2.0*t + w*w*t*t) * k;
-	b0_lp = (w*w*t*t) * k;
-	b1_lp = (2.0*w*w*t*t) * k;
-	b2_lp = (w*w*t*t) * k;
-	*/
-
-}
--- a/examples/airharp/Waveguide.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,57 +0,0 @@
-/*
- *
- * Simple 1-Dimensional Waveguide
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-#ifndef WAVEGUIDE_H_
-#define WAVEGUIDE_H_
-
-#include <cmath>
-
-#ifndef WG_BUFFER_SIZE
-#define WG_BUFFER_SIZE 4096
-#endif
-
-#ifndef FILTER_BUFFER_SIZE
-#define FILTER_BUFFER_SIZE 4
-#endif
-
-#ifndef		M_PI
-#define		M_PI		3.14159265358979323846264338
-#endif
-
-class Waveguide
-{
-
-public:
-
-	Waveguide();
-	void setup();
-	float update(float in);
-	void updateFilterCoeffs(float frequency);
-	void setFrequency(float frequency);
-
-private:
-
-	double _dt;
-	float _periodInMilliseconds;
-	int _periodInSamples;
-
-	float _buffer[WG_BUFFER_SIZE];
-	int _readPtr;
-
-	float _filterBuffer_x[FILTER_BUFFER_SIZE];
-	float _filterBuffer_y[FILTER_BUFFER_SIZE];
-	float _hipBuffer_x[FILTER_BUFFER_SIZE];
-	float _hipBuffer_y[FILTER_BUFFER_SIZE];
-	int _filterReadPtr;
-
-	float b0_lp,b1_lp,b2_lp,a1_lp, a2_lp;
-	float _lastY,_lastX;
-
-};
-
-#endif
--- a/examples/airharp/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,211 +0,0 @@
-/*
- * AIR-HARP
- * Physically modelled strings using waveguide junctions and mass-spring-dampers
- *
- * render.cpp
- *
- * Christian Heinrichs 04/2015
- *
- */
-
-
-#include "MassSpringDamper.h"
-#include "String.h"
-#include "Plectrum.h"
-
-#include <Bela.h>
-#include <cmath>
-#include <stdio.h>
-#include <cstdlib>
-#include <rtdk.h>
-
-#define ACCEL_BUF_SIZE 8
-#define NUMBER_OF_STRINGS 9
-
-// PENTATONIC SCALE
-float gMidinotes[NUMBER_OF_STRINGS] = {40,45,50,55,57,60,62,64,67};
-
-float gInverseSampleRate;
-
-float out_gain = 5.0;
-
-int accelPin_x = 0;
-int accelPin_y = 1;
-int accelPin_z = 2;
-
-MassSpringDamper msd = MassSpringDamper(1,0.1,10);// (10,0.001,10);
-String strings[NUMBER_OF_STRINGS];
-Plectrum plectrums[NUMBER_OF_STRINGS];
-
-float gPlectrumDisplacement = 0;
-
-float gAccel_x[ACCEL_BUF_SIZE] = {0};
-int gAccelReadPtr = 0;
-
-// DC BLOCK BUTTERWORTH
-
-// Coefficients for 100hz cut-off
-float a0_l = 0.9899759179893742;
-float a1_l = -1.9799518359787485;
-float a2_l = 0.9899759179893742;
-float a3_l = -1.979851353142371;
-float a4_l = 0.9800523188151258;
-
-float a0_r = a0_l;
-float a1_r = a1_l;
-float a2_r = a2_l;
-float a3_r = a3_l;
-float a4_r = a4_l;
-
-float x1_l = 0;
-float x2_l = 0;
-float y1_l = 0;
-float y2_l = 0;
-
-float x1_r = 0;
-float x2_r = 0;
-float y1_r = 0;
-float y2_r = 0;
-
-
-bool setup(BelaContext *context, void *userData)
-{
-
-	gInverseSampleRate = 1.0 / context->audioSampleRate;
-
-	// initialise strings & plectrums
-	for(int i=0;i<NUMBER_OF_STRINGS;i++)	{
-
-		plectrums[i] = Plectrum();
-		plectrums[i].setup(250,0.25,0.05);
-
-		strings[i] = String();
-		strings[i].setMidinote(gMidinotes[i]);
-
-		float spacing = 2.0 / (NUMBER_OF_STRINGS+1);
-
-		strings[i].setGlobalPosition( -1 + spacing*(i+1) );
-
-		rt_printf("STRING %d // midinote: %f position: %f\n",i,gMidinotes[i],( -1 + spacing*(i+1) ));
-
-	}
-
-	return true;
-}
-
-void render(BelaContext *context, void *userData)
-{
-
-	float lastAccel = 0;
-
-	for(int n = 0; n < context->audioFrames; n++) {
-
-		/*
-		 *
-		 * ACCELEROMETER DATA
-		 *
-		 */
-
-		// Read accelerometer data from analog input
-		float accel_x = 0;
-		if(n%2)	{
-			accel_x = (float)context->analogIn[(n/2)*8+accelPin_x] * 2 - 1;	// 15800 - 28300 - 41500
-			lastAccel = accel_x;
-		} else {
-			// grab previous value if !n%2
-			accel_x = lastAccel;
-		}
-
-		// Dead-zone avoids noise when box is lying horizontally on a surface
-
-		float accelDeadZone = 0.1;
-
-		if(accel_x <= accelDeadZone && accel_x >= -accelDeadZone)
-			accel_x = 0;
-
-		// Perform smoothing (moving average) on acceleration value
-		if(++gAccelReadPtr >= ACCEL_BUF_SIZE)
-			gAccelReadPtr = 0;
-		gAccel_x[gAccelReadPtr] = accel_x;
-		float gravity = 0;
-		for(int i=0;i<ACCEL_BUF_SIZE;i++)	{
-			gravity = gAccel_x[(gAccelReadPtr-i+ACCEL_BUF_SIZE)%ACCEL_BUF_SIZE];
-		}
-		gravity /= ACCEL_BUF_SIZE;
-
-		/*
-		 *
-		 * PHYSICS SIMULATION
-		 *
-		 */
-
-		// The horizontal force (which can be gravity if box is tipped on its side)
-		// is used as the input to a Mass-Spring-Damper model
-		// Plectrum displacement (i.e. when interacting with string) is included
-		float massPosition = (float)msd.update(gravity - gPlectrumDisplacement);
-
-		float out_l = 0;
-		float out_r = 0;
-		// Use this parameter to quickly adjust output gain
-		float gain = 0.0015;	// 0.0015 is a good value or 12 strings
-		gPlectrumDisplacement = 0;
-
-		for(int s=0;s<NUMBER_OF_STRINGS;s++)	{
-
-			float stringPosition = strings[s].getGlobalPosition();
-
-			float plectrumForce = plectrums[s].update(massPosition, stringPosition);
-			gPlectrumDisplacement += strings[s].getPlectrumDisplacement();
-
-			// calculate panning based on string position (-1->left / 1->right)
-			float panRight = map(stringPosition,1,-1,0.1,1);
-			float panLeft = map(stringPosition,-1,1,0.1,1);
-			panRight *= panRight;
-			panLeft *= panLeft;
-
-			float out = strings[s].update(plectrumForce)*gain;
-
-			out_l += out*panLeft;
-			out_r += out*panRight;
-
-		}
-
-		// APPLY DC-BLOCK FILTER TO OUTPUTS
-
-		// LEFT CHANNEL
-		float temp_in = out_l;
-		/* compute result */
-    	out_l = a0_l * out_l + a1_l * x1_l + a2_l * x2_l - a3_l * y1_l - a4_l * y2_l;
-    	/* shift x1 to x2, sample to x1 */
-    	x2_l = x1_l;
-    	x1_l = temp_in;
-    	/* shift y1 to y2, result to y1 */
-    	y2_l = y1_l;
-   	 	y1_l = out_l;
-
-   	 	// RIGHT CHANNEL
-		temp_in = out_r;
-		/* compute result */
-    	out_r = a0_r * out_r + a1_r * x1_r + a2_r * x2_r - a3_r * y1_r - a4_r * y2_r;
-    	/* shift x1 to x2, sample to x1 */
-    	x2_r = x1_r;
-    	x1_r = temp_in;
-    	/* shift y1 to y2, result to y1 */
-    	y2_r = y1_r;
-   	 	y1_r = out_r;
-
-		context->audioOut[n * context->audioChannels + 1] = out_l * out_gain;
-		context->audioOut[n * context->audioChannels + 0] = out_r * out_gain;
-
-	}
-
-}
-
-
-// cleanup_render() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in initialise_render().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/analogDigitalDemo/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,108 +0,0 @@
-/*
- * assignment1_crossover
- * RTDSP 2015
- *
- * First assignment for ECS732 RTDSP, to implement a 2-way audio crossover
- * using the BeagleBone Black.
- *
- * Andrew McPherson and Victor Zappi
- * Queen Mary, University of London
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <fcntl.h>
-
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	float frequency = 1000.0;	// Frequency of crossover
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-        case 'f':
-        		frequency = atof(optarg);
-        		if(frequency < 20.0)
-        			frequency = 20.0;
-        		if(frequency > 5000.0)
-        			frequency = 5000.0;
-        		break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &frequency) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/analogDigitalDemo/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,118 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- *
- * Andrew McPherson and Victor Zappi
- * Queen Mary, University of London
- */
-
-/**
-\example 3_analogDigitalDemo
-
-Analog digital workout
-----------------------
-
-This sketch showcases many different ways to write and read digital pins, 
-including generating clocks and creating binary counters.
-
-The code as it is will not work properly, as the direction of the pins is not 
-set. As an exercise, you will need to set the pin mode before writing or reading 
-the digital pins. 
-
-This is for advanced users only.
-*/
-
-#include <Bela.h>
-#include <cmath>
-#include <rtdk.h>
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numAnalogFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-/*
-we assume that gNumAnalogChannels=8, numAnalogFrames==8 and  numDigitalFrames==numAudioFrames
- * */
-{
-  /*
-  * TODO: as an exercise, you will need to set the pin mode before writing or reading the digital pins.
-  */
-	if((context->audioFramesElapsed&31)==0){ //every 32 frames...
-        //ANALOG channels
-		analogWrite(context, 0, 0, analogRead(context, 0,0));
-		// read the input0 at frame0  and write it to output0 frame0. Using analogWrite will fill the rest of the buffer with the same value
-                                            // The value at the last frame will persist through the successive buffers until is set again.
-                                            // This effectively is a pass-through with downsampling by 32 times
-        analogWrite(context, 0, 3, 1.0);  // write 1.0 to channel3 from frame0 to the end of the buffer
-        analogWrite(context, 4, 3, 0.1);  // write 0.1  to channel3 from frame4 to the end of the buffer
-        analogWriteOnce(context, 6, 3, 0.2); //write 0.2 to channel3 only on frame 6
-        //this buffer for channel 3 will look like this:  1 1 1 1 0.1 0.1 0.2 0.1 
-        //the next buffers for channel 3 will be filled up with 0.1 ....
-        //DIGITAL channels
-        digitalWrite(context, 0, P8_07, GPIO_HIGH); //sets all the frames  to HIGH for channel 0
-        digitalWriteOnce(context, 4, P8_07, GPIO_LOW); //only frame 4 will be LOW  for channel 0
-        // in this buffer the frames of channel 0 will look like this: 1 1 1 1 0 1 1 1 ...... 1 
-        // in the next buffer each frame of channel 0 will be initialized to 1 (the last value of this buffer)
-        digitalWrite(context, 0, P8_08, GPIO_HIGH);
-        digitalWrite(context, 2, P8_08, GPIO_LOW);
-        digitalWrite(context, 4, P8_08, GPIO_HIGH);
-        digitalWrite(context, 5, P8_08, GPIO_LOW);
-        pinMode(context, 0, P9_16, GPIO_INPUT); // set channel 10 to input
-        // in this buffer the frames of channel 1 will look like this: 1 1 0 0 1 0 0 0 .... 0
-        // in the next buffer each frame of channel 1 will be initialized to 0 (the last value of this buffer)
-	}
-	for(unsigned int n=0; n<context->audioFrames; n++){
-		for(unsigned int c=0; c<context->audioChannels; c++){
-			context->audioOut[n*context->audioChannels + c]=context->audioIn[n*context->audioChannels + c];
-		}
-        //use digital channels 2-8 to create a 7 bit binary counter
-        context->digital[n]=context->digital[n] & (~0b111111100); // set to zero (GPIO_OUTPUT) the bits in the lower word
-        context->digital[n]=context->digital[n] & ((~0b111111100<<16) | 0xffff ); //initialize to zero the bits in the higher word (output value)
-        context->digital[n]=context->digital[n] | ( ((context->audioFramesElapsed&0b1111111)<<(16+2)) ) ;  // set the bits in the higher word to the desired output value, keeping the lower word unchanged
-        digitalWrite(context, n, P8_29, digitalRead(context, n, P8_30)); // echo the input from from channel 15 to channel 14
-        digitalWrite(context, n, P8_28, digitalRead(context, n, P9_16)); // echo the input from from channel 10 to channel 13
-        pinMode(context, 0, P8_30, 0); //set channel 15 to input
-	}
-
-	for(unsigned int n=0; n<context->analogFrames; n++){
-    	analogWrite(context, n, 1, (context->audioFramesElapsed&8191)/8192.0); // writes a single frame. channel 1 is a ramp that follows gCountFrames
-    	analogWrite(context, n, 2, analogRead(context, n, 2)); // writes a single frame. channel2 is just a passthrough
-//		rt_printf("Analog out frame %d :",n);
-//		for(int c=0; c<gNumAnalogChannels; c++)
-//			rt_printf("%.1f ",analogOut[n*gNumAnalogChannels + c]);
-//		rt_printf("\n");
-	}
-	return;
-
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	// Nothing to do here
-}
--- a/examples/audio_in_FFT/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,100 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --fftsize [-s] size:     Set the fSize of the FFT, in samples\n";
-	cerr << "   --help [-h]:             Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	int fftSize = 64;		    // Size of the FFT, in samples
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"fftsize", 1, NULL, 's'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-//	settings.useAnalog = 0;	// No matrix usage by default
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hs:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 's':
-				fftSize = atof(optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &fftSize) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
-
--- a/examples/audio_in_FFT/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,118 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: Andrew McPherson, C4DM, QMUL
- */
-
-/**
-\example 4_audio_FFT
-
-Fast Fourier Transform
-----------------------
-
-This sketch performs an FFT (Fast Fourier Transform) on incoming audio. It uses 
-the NE10 library, included at the top of the file (line 11).
-
-Read the documentation on the NE10 library [here](http://projectne10.github.io/Ne10/doc/annotated.html).
-
-The variables `timeDomainIn`, `timeDomainOut` and `frequencyDomain` are 
-variables of the struct `ne10_fft_cpx_float32_t` [http://projectne10.github.io/Ne10/doc/structne10__fft__cpx__float32__t.html](http://projectne10.github.io/Ne10/doc/structne10__fft__cpx__float32__t.html). 
-These are declared at the top of the file (line 21), and memory is allocated 
-for them in `setup()` (line 41).
-
-In `render()` a `for` loop performs the FFT which is performed on each sample, 
-and the resulting output is placed on each channel.
-*/
-
-#include <Bela.h>
-#include <rtdk.h>
-#include <ne10/NE10.h>					// neon library
-#include <cmath>
-
-int gFFTSize;
-
-int gReadPointer = 0;
-int gWritePointer = 0;
-
-// FFT vars
-static ne10_fft_cpx_float32_t* timeDomainIn;
-static ne10_fft_cpx_float32_t* timeDomainOut;
-static ne10_fft_cpx_float32_t* frequencyDomain;
-static ne10_fft_cfg_float32_t cfg;
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	// Retrieve a parameter passed in from the initAudio() call
-	gFFTSize = *(int *)userData;
-
-	timeDomainIn = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-	timeDomainOut = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-	frequencyDomain = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-	cfg = ne10_fft_alloc_c2c_float32_neon (gFFTSize);
-
-	memset(timeDomainOut, 0, gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		timeDomainIn[gReadPointer].r = (ne10_float32_t) ((context->audioIn[n*context->audioChannels] +
-															context->audioIn[n*context->audioChannels+1]) * 0.5);
-		timeDomainIn[gReadPointer].i = 0;
-
-		if(++gReadPointer >= gFFTSize)
-		{
-			//FFT
-			ne10_fft_c2c_1d_float32_neon (frequencyDomain, timeDomainIn, cfg, 0);
-
-			//Do frequency domain stuff
-
-			//IFFT
-			ne10_fft_c2c_1d_float32_neon (timeDomainOut, frequencyDomain, cfg, 1);
-
-			gReadPointer = 0;
-			gWritePointer = 0;
-		}
-
-		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
-			context->audioOut[n * context->audioChannels + channel] = (float) timeDomainOut[gWritePointer].r;
-		gWritePointer++;
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	NE10_FREE(timeDomainIn);
-	NE10_FREE(timeDomainOut);
-	NE10_FREE(frequencyDomain);
-	NE10_FREE(cfg);
-}
--- a/examples/basic/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,97 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-#include <unistd.h>
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	float frequency = 440.0;	// Frequency of oscillator
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				frequency = atof(optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &frequency) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/basic/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,94 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-/**
-\example 1_basic_helloworld
-
-Producing your first bleep!
----------------------------
-
-This sketch is the hello world of embedded interactive audio. Better known as bleep, it 
-produces a sine tone.
-
-The frequency of the sine tone is determined by a global variable, `gFrequency` 
-(line 12). The sine tone is produced by incrementing the phase of a sin function 
-on every audio frame.
-
-In render() you'll see a nested for loop structure. You'll see this in all Bela projects. 
-The first for loop cycles through 'audioFrames', the second through 'audioChannels' (in this case left 0 and right 1). 
-It is good to familiarise yourself with this structure as it's fundamental to producing sound with the system.
-*/
-
-#include <Bela.h>
-#include <cmath>
-
-float gFrequency = 440.0;
-float gPhase;
-float gInverseSampleRate;
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	// Retrieve a parameter passed in from the initAudio() call
-	if(userData != 0)
-		gFrequency = *(float *)userData;
-
-	gInverseSampleRate = 1.0 / context->audioSampleRate;
-	gPhase = 0.0;
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		float out = 0.8f * sinf(gPhase);
-		gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate;
-		if(gPhase > 2.0 * M_PI)
-			gPhase -= 2.0 * M_PI;
-
-		for(unsigned int channel = 0; channel < context->audioChannels; channel++) {
-			// Two equivalent ways to write this code
-
-			// The long way, using the buffers directly:
-			// context->audioOut[n * context->audioChannels + channel] = out;
-
-			// Or using the macros:
-			audioWrite(context, n, channel, out);
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/basic_FFT_phase_vocoder/SampleData.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,19 +0,0 @@
-/*
- * SampleData.h
- *
- *  Created on: Nov 5, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef SAMPLEDATA_H_
-#define SAMPLEDATA_H_
-
-// User defined structure to pass between main and rendere complex data retrieved from file
-struct SampleData {
-	float *samples;	// Samples in file
-	int sampleLen;	// Total nume of samples
-};
-
-
-
-#endif /* SAMPLEDATA_H_ */
--- a/examples/basic_FFT_phase_vocoder/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,196 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <cstdio>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <unistd.h>
-#include <sys/time.h>
-#include <sndfile.h>				// to load audio files
-#include "SampleData.h"
-#include <Bela.h>
-
-using namespace std;
-
-// Global variables used by getCurrentTime()
-unsigned long long gFirstSeconds, gFirstMicroseconds;
-
-// Load samples from file
-int initFile(string file, SampleData *smp)//float *& smp)
-{
-  	SNDFILE *sndfile ;
-	SF_INFO sfinfo ;
-
-	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
-		cout << "Couldn't open file " << file << endl;
-		return 1;
-	}
-
-	int numChan = sfinfo.channels;
-	if(numChan != 1)
-	{
-		cout << "Error: " << file << " is not a mono file" << endl;
-		return 1;
-	}
-
-	smp->sampleLen = sfinfo.frames * numChan;
-	smp->samples = new float[smp->sampleLen];
-	if(smp == NULL){
-		cout << "Could not allocate buffer" << endl;
-		return 1;
-	}
-
-	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
-	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
-
-	// Pad with zeros in case we couldn't read whole file
-	for(int k = readcount; k <smp->sampleLen; k++)
-		smp->samples[k] = 0;
-
-	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
-		double	scale ;
-		int 	m ;
-
-		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
-		if (scale < 1e-10)
-			scale = 1.0 ;
-		else
-			scale = 32700.0 / scale ;
-		cout << "File samples scale = " << scale << endl;
-
-		for (m = 0; m < smp->sampleLen; m++)
-			smp->samples[m] *= scale;
-	}
-
-	sf_close(sndfile);
-
-	return 0;
-}
-
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-/* Function which returns the time since start of the program
- * in (fractional) seconds.
- */
-double getCurrentTime(void) {
-	unsigned long long result;
-	struct timeval tv;
-
-	gettimeofday(&tv, NULL);
-	result = (tv.tv_sec - gFirstSeconds) * 1000000ULL + (tv.tv_usec - gFirstMicroseconds);
-	return (double)result / 1000000.0;
-}
-extern SampleData gSampleData;
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	struct timeval tv;
-	string fileName;			// Name of the sample to load
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"file", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	gSampleData.samples = 0;
-	gSampleData.sampleLen = -1;
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	settings.periodSize = 32; // Larger period size by default, for testing
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				fileName = string((char *)optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	if(fileName.empty()){
-		fileName = "sample.wav";
-	}
-
-
-	// Load file
-	if(initFile(fileName, &gSampleData) != 0)
-	{
-		cout << "Error: unable to load samples " << endl;
-		return -1;
-	}
-
-	if(settings.verbose)
-		cout << "File contains " << gSampleData.sampleLen << " samples" << endl;
-
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &gSampleData) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Initialise time
-	gettimeofday(&tv, NULL);
-	gFirstSeconds = tv.tv_sec;
-	gFirstMicroseconds = tv.tv_usec;
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/basic_FFT_phase_vocoder/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,299 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-/**
-\example 4_audio_FFT_phase_vocoder
-
-Phase Vocoder
-----------------------
-
-This sketch shows an implementation of a phase vocoder and builds on the previous FFT example.
-Again it uses the NE10 library, included at the top of the file (line 11).
-
-Read the documentation on the NE10 library [here](http://projectne10.github.io/Ne10/doc/annotated.html).
-*/
-
-#include <Bela.h>
-#include <rtdk.h>
-#include <ne10/NE10.h>					// NEON FFT library
-#include <cmath>
-#include "SampleData.h"
-#include <Midi.h>
-
-#define BUFFER_SIZE 16384
-
-// TODO: your buffer and counter go here!
-float gInputBuffer[BUFFER_SIZE];
-int gInputBufferPointer = 0;
-float gOutputBuffer[BUFFER_SIZE];
-int gOutputBufferWritePointer = 0;
-int gOutputBufferReadPointer = 0;
-int gSampleCount = 0;
-
-float *gWindowBuffer;
-
-// -----------------------------------------------
-// These variables used internally in the example:
-int gFFTSize = 2048;
-int gHopSize = 512;
-int gPeriod = 512;
-float gFFTScaleFactor = 0;
-
-// FFT vars
-ne10_fft_cpx_float32_t* timeDomainIn;
-ne10_fft_cpx_float32_t* timeDomainOut;
-ne10_fft_cpx_float32_t* frequencyDomain;
-ne10_fft_cfg_float32_t cfg;
-
-// Sample info
-SampleData gSampleData;	// User defined structure to get complex data from main
-int gReadPtr = 0;		// Position of last read sample from file
-
-// Auxiliary task for calculating FFT
-AuxiliaryTask gFFTTask;
-int gFFTInputBufferPointer = 0;
-int gFFTOutputBufferPointer = 0;
-
-void process_fft_background();
-
-
-int gEffect = 0; // change this here or with midi CC
-enum{
-	kBypass,
-	kRobot,
-	kWhisper,
-};
-
-float gDryWet = 1; // mix between the unprocessed and processed sound
-float gPlaybackLive = 0.5f; // mix between the file playback and the live audio input
-float gGain = 1; // overall gain
-float *gInputAudio = NULL;
-Midi midi;
-
-
-void midiCallback(MidiChannelMessage message, void* arg){
-	if(message.getType() == kmmNoteOn){
-		if(message.getDataByte(1) > 0){
-			int note = message.getDataByte(0);
-			float frequency = powf(2, (note-69)/12.f)*440;
-			gPeriod = (int)(44100 / frequency + 0.5);
-			printf("\nnote: %d, frequency: %f, hop: %d\n", note, frequency, gPeriod);
-		}
-	}
-
-	bool shouldPrint = false;
-	if(message.getType() == kmmControlChange){
-		float data = message.getDataByte(1) / 127.0f;
-		switch (message.getDataByte(0)){
-		case 2 :
-			gEffect = (int)(data * 2 + 0.5); // CC2 selects an effect between 0,1,2
-			break;
-		case 3 :
-			gPlaybackLive = data;
-			break;
-		case 4 :
-			gDryWet = data;
-			break;
-		case 5:
-			gGain = data*10;
-			break;
-		default:
-			shouldPrint = true;
-		}
-	}
-	if(shouldPrint){
-		message.prettyPrint();
-	}
-}
-
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-bool setup(BelaContext* context, void* userData)
-{
-	midi.readFrom(0);
-	midi.setParserCallback(midiCallback);
-	// Retrieve a parameter passed in from the initAudio() call
-	gSampleData = *(SampleData *)userData;
-
-	gFFTScaleFactor = 1.0f / (float)gFFTSize;
-	gOutputBufferWritePointer += gHopSize;
-
-	timeDomainIn = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-	timeDomainOut = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-	frequencyDomain = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-	cfg = ne10_fft_alloc_c2c_float32_neon (gFFTSize);
-
-	memset(timeDomainOut, 0, gFFTSize * sizeof (ne10_fft_cpx_float32_t));
-	memset(gOutputBuffer, 0, BUFFER_SIZE * sizeof(float));
-
-	// Allocate buffer to mirror and modify the input
-	gInputAudio = (float *)malloc(context->audioFrames * context->audioChannels * sizeof(float));
-	if(gInputAudio == 0)
-		return false;
-
-	// Allocate the window buffer based on the FFT size
-	gWindowBuffer = (float *)malloc(gFFTSize * sizeof(float));
-	if(gWindowBuffer == 0)
-		return false;
-
-	// Calculate a Hann window
-	for(int n = 0; n < gFFTSize; n++) {
-		gWindowBuffer[n] = 0.5f * (1.0f - cosf(2.0 * M_PI * n / (float)(gFFTSize - 1)));
-	}
-
-	// Initialise auxiliary tasks
-	if((gFFTTask = Bela_createAuxiliaryTask(&process_fft_background, 90, "fft-calculation")) == 0)
-		return false;
-	rt_printf("You are listening to an FFT phase-vocoder with overlap-and-add.\n"
-	"Use Midi Control Change to control:\n"
-	"CC 2: effect type (bypass/robotization/whisperization)\n"
-	"CC 3: mix between recorded sample and live audio input\n"
-	"CC 4: mix between the unprocessed and processed sound\n"
-	"CC 5: gain\n"
-	);
-	return true;
-}
-
-// This function handles the FFT processing in this example once the buffer has
-// been assembled.
-void process_fft(float *inBuffer, int inWritePointer, float *outBuffer, int outWritePointer)
-{
-	// Copy buffer into FFT input
-	int pointer = (inWritePointer - gFFTSize + BUFFER_SIZE) % BUFFER_SIZE;
-	for(int n = 0; n < gFFTSize; n++) {
-		timeDomainIn[n].r = (ne10_float32_t) inBuffer[pointer] * gWindowBuffer[n];
-		timeDomainIn[n].i = 0;
-
-		pointer++;
-		if(pointer >= BUFFER_SIZE)
-			pointer = 0;
-	}
-
-	// Run the FFT
-	ne10_fft_c2c_1d_float32_neon (frequencyDomain, timeDomainIn, cfg, 0);
-
-	switch (gEffect){
-	  case kRobot :
-	  // Robotise the output
-	    for(int n = 0; n < gFFTSize; n++) {
-	      float amplitude = sqrtf(frequencyDomain[n].r * frequencyDomain[n].r + frequencyDomain[n].i * frequencyDomain[n].i);
-	      frequencyDomain[n].r = amplitude;
-	      frequencyDomain[n].i = 0;
-	    }
-	    break;
-	  case kWhisper :
-	    for(int n = 0; n < gFFTSize; n++) {
-	      float amplitude = sqrtf(frequencyDomain[n].r * frequencyDomain[n].r + frequencyDomain[n].i * frequencyDomain[n].i);
-	      float phase = rand()/(float)RAND_MAX * 2 * M_PI;
-	      frequencyDomain[n].r = cosf(phase) * amplitude;
-	      frequencyDomain[n].i = sinf(phase) * amplitude;
-	    }
-	    break;
-	  case kBypass:
-	    //bypass
-	    break;
-	  }
-
-	// Run the inverse FFT
-	ne10_fft_c2c_1d_float32_neon (timeDomainOut, frequencyDomain, cfg, 1);
-	// Overlap-and-add timeDomainOut into the output buffer
-	pointer = outWritePointer;
-	for(int n = 0; n < gFFTSize; n++) {
-		outBuffer[pointer] += (timeDomainOut[n].r) * gFFTScaleFactor;
-		if(isnan(outBuffer[pointer]))
-			rt_printf("outBuffer OLA\n");
-		pointer++;
-		if(pointer >= BUFFER_SIZE)
-			pointer = 0;
-	}
-}
-
-// Function to process the FFT in a thread at lower priority
-void process_fft_background() {
-	process_fft(gInputBuffer, gFFTInputBufferPointer, gOutputBuffer, gFFTOutputBufferPointer);
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-void render(BelaContext* context, void* userData)
-{
-	float* audioOut = context->audioOut;
-	int numAudioFrames = context->audioFrames;
-	int numAudioChannels = context->audioChannels;
-	// ------ this code internal to the demo; leave as is ----------------
-
-	// Prep the "input" to be the sound file played in a loop
-	for(int n = 0; n < numAudioFrames; n++) {
-		if(gReadPtr < gSampleData.sampleLen)
-			gInputAudio[2*n] = gInputAudio[2*n+1] = gSampleData.samples[gReadPtr]*(1-gPlaybackLive) +
-			gPlaybackLive*0.5f*(audioRead(context,n,0)+audioRead(context,n,1));
-		else
-			gInputAudio[2*n] = gInputAudio[2*n+1] = 0;
-		if(++gReadPtr >= gSampleData.sampleLen)
-			gReadPtr = 0;
-	}
-	// -------------------------------------------------------------------
-
-	for(int n = 0; n < numAudioFrames; n++) {
-		gInputBuffer[gInputBufferPointer] = ((gInputAudio[n*numAudioChannels] + gInputAudio[n*numAudioChannels+1]) * 0.5);
-
-		// Copy output buffer to output
-		for(int channel = 0; channel < numAudioChannels; channel++){
-			audioOut[n * numAudioChannels + channel] = gOutputBuffer[gOutputBufferReadPointer] * gGain * gDryWet + (1 - gDryWet) * gInputAudio[n * numAudioChannels + channel];
-		}
-
-		// Clear the output sample in the buffer so it is ready for the next overlap-add
-		gOutputBuffer[gOutputBufferReadPointer] = 0;
-		gOutputBufferReadPointer++;
-		if(gOutputBufferReadPointer >= BUFFER_SIZE)
-			gOutputBufferReadPointer = 0;
-		gOutputBufferWritePointer++;
-		if(gOutputBufferWritePointer >= BUFFER_SIZE)
-			gOutputBufferWritePointer = 0;
-
-		gInputBufferPointer++;
-		if(gInputBufferPointer >= BUFFER_SIZE)
-			gInputBufferPointer = 0;
-
-		gSampleCount++;
-		if(gSampleCount >= gHopSize) {
-			//process_fft(gInputBuffer, gInputBufferPointer, gOutputBuffer, gOutputBufferPointer);
-			gFFTInputBufferPointer = gInputBufferPointer;
-			gFFTOutputBufferPointer = gOutputBufferWritePointer;
-			Bela_scheduleAuxiliaryTask(gFFTTask);
-
-			gSampleCount = 0;
-		}
-	}
-	gHopSize = gPeriod;
-}
-
-// cleanup_render() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in initialise_render().
-
-void cleanup(BelaContext* context, void* userData)
-{
-	NE10_FREE(timeDomainIn);
-	NE10_FREE(timeDomainOut);
-	NE10_FREE(frequencyDomain);
-	NE10_FREE(cfg);
-	free(gInputAudio);
-	free(gWindowBuffer);
-}
Binary file examples/basic_FFT_phase_vocoder/sample.wav has changed
--- a/examples/basic_analog_input/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,117 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-int gSensorInputFrequency = 0;
-int gSensorInputAmplitude = 1;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --frequency [-f] input:  Choose the analog input controlling frequency (0-7; default 0)\n";
-	cerr << "   --amplitude [-a] input:  Choose the analog input controlling amplitude (0-7; default 1)\n";
-	cerr << "   --help [-h]:             Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{"amplitude", 1, NULL, 'a'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:a:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				gSensorInputFrequency = atoi(optarg);
-				if(gSensorInputFrequency < 0 || gSensorInputFrequency > 7) {
-					usage(basename(argv[0]));
-					exit(0);
-				}
-				break;
-		case 'a':
-				gSensorInputAmplitude = atoi(optarg);
-				if(gSensorInputAmplitude < 0 || gSensorInputAmplitude > 7) {
-					usage(basename(argv[0]));
-					exit(0);
-				}
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, 0) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	if(settings.verbose) {
-		cout << "--> Frequency on input " << gSensorInputFrequency << endl;
-		cout << "--> Amplitude on input " << gSensorInputAmplitude << endl;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/basic_analog_input/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,121 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-/**
-\example 3_analog_input
-
-Connecting potentiometers
--------------------------
-
-This sketch produces a sine tone, the frequency and amplitude of which are 
-affected by data received on the analog pins. Before looping through each audio 
-frame, we declare a value for the frequency and amplitude of our sine tone 
-(line 55); we adjust these values by taking in data from analog sensors 
-(for example potentiometers) with `analogRead()`.
-
-- connect a 10K pot to 3.3V and GND on its 1st and 3rd pins.
-- connect the 2nd middle pin of the pot to analogIn 0.
-- connect another 10K pot in the same way but with the middle pin connected to analogIn 1.
-
-The important thing to notice is that audio is sampled twice as often as analog 
-data. The audio sampling rate is 44.1kHz (44100 frames per second) and the 
-analog sampling rate is 22.05kHz (22050 frames per second). On line 62 you might 
-notice that we are processing the analog data and updating frequency and 
-amplitude only on every second audio sample, since the analog sampling rate is 
-half that of the audio.
-
-Note that the pin numbers are stored in the variables `gAnalogInputFrequency` and 
-`gAnalogInputAmplitude`. These are declared in the main.cpp file; if you look in 
-that file you will see that they have the values of 0 and 1. Bear in mind that 
-these are analog input pins which is a specific header!
-*/
-
-#include <Bela.h>
-#include <rtdk.h>
-#include <cmath>
-
-float gPhase;
-float gInverseSampleRate;
-int gAudioFramesPerAnalogFrame;
-
-// These settings are carried over from main.cpp
-// Setting global variables is an alternative approach
-// to passing a structure to userData in setup()
-
-extern int gSensorInputFrequency;
-extern int gSensorInputAmplitude;
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	if(context->analogFrames == 0 || context->analogFrames > context->audioFrames) {
-		rt_printf("Error: this example needs analog enabled, with 4 or 8 channels\n");
-		return false;
-	}
-
-	gAudioFramesPerAnalogFrame = context->audioFrames / context->analogFrames;
-	gInverseSampleRate = 1.0 / context->audioSampleRate;
-	gPhase = 0.0;
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	float frequency = 440.0;
-	float amplitude = 0.8;
-
-	// There are twice as many audio frames as matrix frames since audio sample rate
-	// is twice as high
-
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		if(!(n % gAudioFramesPerAnalogFrame)) {
-			// Even audio samples: update frequency and amplitude from the matrix
-			frequency = map(analogRead(context, n/gAudioFramesPerAnalogFrame, gSensorInputFrequency), 0, 1, 100, 1000);
-			amplitude = analogRead(context, n/gAudioFramesPerAnalogFrame, gSensorInputAmplitude);
-		}
-
-		float out = amplitude * sinf(gPhase);
-
-		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
-			context->audioOut[n * context->audioChannels + channel] = out;
-
-		gPhase += 2.0 * M_PI * frequency * gInverseSampleRate;
-		if(gPhase > 2.0 * M_PI)
-			gPhase -= 2.0 * M_PI;
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/basic_analog_output/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,104 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --frequency [-f] frequency: Set the frequency of the LED fade (default: 1.0)\n";
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	float frequency = 1.0;	// Frequency of LED fades
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// In this example, audio isn't used so might as well leave speaker muted
-	settings.beginMuted = 1;
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				frequency = atof(optarg);
-				if(frequency < 0)
-					frequency = 0;
-				if(frequency > 11025.0)
-					frequency = 11025.0;
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &frequency) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/basic_analog_output/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,115 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- *
- * Andrew McPherson and Victor Zappi
- * Queen Mary, University of London
- */
-
-/**
-\example 3_analog_output
-
-Fading LEDs
------------
-
-This sketch uses a sine wave to drive the brightness of a series of LEDs 
-connected to the eight analog out pins. Again you can see the nested `for` loop 
-structure but this time for the analog output channels rather than the audio.
-
-- connect an LED in series with a 470ohm resistor between each of the analogOut pins and ground.
-
-Within the first for loop in render we cycle through each frame in the analog 
-output matrix. At each frame we then cycle through the analog output channels 
-with another for loop and set the output voltage according to the phase of a 
-sine tone that acts as an LFO. The analog output pins can provide a voltage of 
-~4.092V.
-
-The output on each pin is set with `analogWrite()` within the for loop that 
-cycles through the analog output channels. This needs to be provided with 
-arguments as follows `analogWrite(context, n, channel, out)`. Channel is 
-where the you give the address of the analog output pin (in this case we cycle 
-through each pin address in the for loop), out is the variable that holds the 
-desired output (in this case set by the sine wave).
-
-Notice that the phase of the brightness cycle for each led is different. This 
-is achieved by updating a variable that stores a relative phase value. This 
-variable is advanced by pi/4 (1/8 of a full rotation) for each channel giving 
-each of the eight LEDs a different phase.
-*/
-
-#include <Bela.h>
-#include <rtdk.h>
-#include <cmath>
-
-// Set range for analog outputs designed for driving LEDs
-const float kMinimumAmplitude = (1.5 / 5.0);
-const float kAmplitudeRange = 1.0 - kMinimumAmplitude;
-
-float gFrequency;
-float gPhase;
-float gInverseSampleRate;
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	// Retrieve a parameter passed in from the initAudio() call
-	gFrequency = *(float *)userData;
-
-	if(context->analogFrames == 0) {
-		rt_printf("Error: this example needs the matrix enabled\n");
-		return false;
-	}
-
-	gInverseSampleRate = 1.0 / context->analogSampleRate;
-	gPhase = 0.0;
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->analogFrames; n++) {
-		// Set LED to different phase for each matrix channel
-		float relativePhase = 0.0;
-		for(unsigned int channel = 0; channel < context->analogChannels; channel++) {
-			float out = kMinimumAmplitude + kAmplitudeRange * 0.5f * (1.0f + sinf(gPhase + relativePhase));
-
-			analogWrite(context, n, channel, out);
-
-			// Advance by pi/4 (1/8 of a full rotation) for each channel
-			relativePhase += M_PI * 0.25;
-		}
-
-		gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate;
-		if(gPhase > 2.0 * M_PI)
-			gPhase -= 2.0 * M_PI;
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/basic_blink/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,99 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- *
- * Andrew McPherson and Victor Zappi
- * Queen Mary, University of London
- */
-
-/**
-\example 2_digital_blink
-
-Blinking an LED
----------------
-
-This sketch shows the simplest case of digital out. 
-
-- Connect an LED in series with a 470ohm resistor between P8_07 and ground. 
-
-The led is blinked on and off by setting the digital pin `HIGH` and `LOW` every interval seconds which is set in 
-`render()`.
-
-In `setup()` the pin mode must be set to output mode via `pinMode()`. For example: 
-`pinMode(context, 0, P8_07, OUTPUT)`. 
-In `render()` the output of the digital pins is set by `digitalWrite()`. For example: 
-`digitalWrite(context, n, P8_07, status)` where `status` can be equal to 
-either `HIGH` or `LOW`. When set `HIGH` the pin will give 3.3V, when set to 
-`LOW` 0V.
-
-To keep track of elapsed time we have a sample counter count. When the count reaches 
-a certain limit it switches state to either `HIGH` or `LOW` depending on its current 
-value. In this case the limit is `context->digitalSampleRate*interval` which 
-allows us to write the desired interval in seconds, stored in `interval`.
-*/
-
-#include <Bela.h>
-#include <cmath>
-#include <rtdk.h>
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-    pinMode(context, 0, P8_07, OUTPUT);
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numAnalogFrames
-// will be 0.
-
-/* basic_blink
-* Connect an LED in series with a 470ohm resistor between P8_07 and ground.
-* The LED will blink every @interval seconds.
-*/
-
-void render(BelaContext *context, void *userData)
-{
-  static int count=0; //counts elapsed samples
-  float interval=0.5; //how often to toggle the LED (in seconds)
-  static int status=GPIO_LOW;
-	for(unsigned int n=0; n<context->digitalFrames; n++){
-    if(count==context->digitalSampleRate*interval){ //if enough samples have elapsed
-      count=0; //reset the counter
-    // status=digitalRead(context, 0, P8_07);
-      if(status==GPIO_LOW) { //toggle the status
-          digitalWrite(context, n, P8_07, status); //write the status to the LED
-          status=GPIO_HIGH;
-      }
-      else {
-          digitalWrite(context, n, P8_07, status); //write the status to the LED
-          status=GPIO_LOW;
-      }
-    }
-    count++;
-  }
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	// Nothing to do here
-}
--- a/examples/basic_button/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,93 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- *
- * Andrew McPherson and Victor Zappi
- * Queen Mary, University of London
- */
-
-/**
-\example 2_digital_button
-
-Switching an LED on and off
----------------------------
-
-This example brings together digital input and digital output. The program will read 
-a button and turn the LED on and off according to the state of the button.
-
-- connect an LED in series with a 470ohm resistor between P8_07 and ground.
-- connect a 1k resistor to P9_03 (+3.3V),
-- connect the other end of the resistor to both a button and P8_08
-- connect the other end of the button to ground.
-
-You will notice that the LED will normally stay on and will turn off as long as 
-the button is pressed. This is due to the fact that the LED is set to the same 
-value read at input P8_08. When the button is not pressed, P8_08 is `HIGH` and so 
-P8_07 is set to `HIGH` as well, so that the LED conducts and emits light. When 
-the button is pressed, P8_08 goes `LOW` and P8_07 is set to `LOW`, turning off the LED.
-
-As an exercise try and change the code so that the LED only turns on when 
-the button is pressed.
-*/
-
-#include <Bela.h>
-#include <cmath>
-#include <rtdk.h>
-#include <stdlib.h>
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	pinMode(context, 0, P8_08, INPUT);
-	pinMode(context, 0, P8_07, OUTPUT);
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numAnalogFrames
-// will be 0.
-
-/* basic_button
-* - connect an LED in series with a 470ohm resistor between P8_07 and ground.
-* - connect a 1k resistor to P9_03 (+3.3V), 
-* - connect the other end of the resistor to both a button and P8_08
-* - connect the other end of the button to ground.
-* The program will read the button and make the LED blink when the button is pressed.
-*/
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n=0; n<context->digitalFrames; n++){
-		int status=digitalRead(context, 0, P8_08); //read the value of the button
-		digitalWriteOnce(context, n, P8_07, status); //write the status to the LED
-		float out = 0.1 * status * rand() / (float)RAND_MAX * 2 - 1; //generate some noise, gated by the button
-		for(unsigned int j = 0; j < context->audioChannels; j++){
-			audioWrite(context, n, j, out); //write the audio output
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	// Nothing to do here
-}
-
--- a/examples/basic_midi/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,165 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <Bela.h>
-#include <Midi.h>
-#include <stdlib.h>
-#include <rtdk.h>
-#include <cmath>
-
-float gFreq;
-float gPhaseIncrement = 0;
-bool gIsNoteOn = 0;
-int gVelocity = 0;
-float gSamplingPeriod = 0;
-
-void midiMessageCallback(MidiChannelMessage message, void* arg){
-	if(arg != NULL){
-		rt_printf("Message from midi port %d: ", *(int*)arg);
-	}
-	message.prettyPrint();
-	if(message.getType() == kmmNoteOn){
-		gFreq = powf(2, (message.getDataByte(0)-69)/12.0f) * 440;
-		gVelocity = message.getDataByte(1);
-		gPhaseIncrement = 2 * M_PI * gFreq * gSamplingPeriod;
-		gIsNoteOn = gVelocity > 0;
-		rt_printf("v0:%f, ph: %6.5f, gVelocity: %d\n", gFreq, gPhaseIncrement, gVelocity);
-	}
-}
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-Midi midi;
-int gMidiPort0 = 0;
-bool setup(BelaContext *context, void *userData)
-{
-	midi.readFrom(gMidiPort0);
-	midi.writeTo(gMidiPort0);
-	midi.enableParser(true);
-	midi.setParserCallback(midiMessageCallback, &gMidiPort0);
-	if(context->analogFrames == 0) {
-		rt_printf("Error: this example needs the analog I/O to be enabled\n");
-		return false;
-	}
-	gSamplingPeriod = 1/context->audioSampleRate;
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-
-enum {kVelocity, kNoteOn, kNoteNumber};
-void render(BelaContext *context, void *userData)
-{
-// one way of getting the midi data is to parse them yourself
-//	(you should set midi.enableParser(false) above):
-/*
-	static midi_byte_t noteOnStatus = 0x90; //on channel 1
-	static int noteNumber = 0;
-	static int waitingFor = kNoteOn;
-	static int playingNote = -1;
-	int message;
-	while ((message = midi.getInput()) >= 0){
-		rt_printf("%d\n", message);
-		switch(waitingFor){
-		case kNoteOn:
-			if(message == noteOnStatus){
-				waitingFor = kNoteNumber;
-			}
-			break;
-		case kNoteNumber:
-			if((message & (1<<8)) == 0){
-				noteNumber = message;
-				waitingFor = kVelocity;
-			}
-			break;
-		case kVelocity:
-			if((message & (1<<8)) == 0){
-				int _velocity = message;
-				waitingFor = kNoteOn;
-				// "monophonic" behaviour, with priority to the latest note on
-				// i.e.: a note off from a previous note does not stop the current note
-				// still you might end up having a key down and no note being played if you pressed and released another
-				// key in the meantime
-				if(_velocity == 0 && noteNumber == playingNote){
-					noteOn = false;
-					playingNote = -1;
-					velocity = _velocity;
-				} else if (_velocity > 0) {
-					noteOn = true;
-					velocity = _velocity;
-					playingNote = noteNumber;
-					f0 = powf(2, (playingNote-69)/12.0f) * 440;
-					phaseIncrement = 2 * M_PI * f0 / context->audioSampleRate;
-				}
-				rt_printf("NoteOn: %d, NoteNumber: %d, velocity: %d\n", noteOn, noteNumber, velocity);
-			}
-			break;
-		}
-	}
-*/
-	/*
-	int num;
-	//alternatively, you can use the built-in parser (only processes channel messages at the moment).
-	while((num = midi.getParser()->numAvailableMessages()) > 0){
-		static MidiChannelMessage message;
-		message = midi.getParser()->getNextChannelMessage();
-		message.prettyPrint();
-		if(message.getType() == kmmNoteOn){
-			f0 = powf(2, (message.getDataByte(0)-69)/12.0f) * 440;
-			velocity = message.getDataByte(1);
-			phaseIncrement = 2 * M_PI * f0 / context->audioSampleRate;
-			noteOn = velocity > 0;
-			rt_printf("v0:%f, ph: %6.5f, velocity: %d\n", f0, phaseIncrement, gVelocity);
-		}
-	}
-	 */
-	// the following block toggles the LED on an Owl pedal
-	// and asks the pedal to return the status of the LED
-	// using MIDI control changes
-	for(unsigned int n = 0; n < context->analogFrames; n++){
-		static int count = 0;
-		static bool state = 0;
-		analogWriteOnce(context, n, 1, state);
-		if(count % 40000 == 0){
-			state = !state;
-			midi_byte_t bytes[6] = {176, 30, (char)(state*127), 176, 67, 30}; // toggle the OWL led and ask for the led status
-			midi.writeOutput(bytes, 6);
-		}
-		count++;
-	}
-	for(unsigned int n = 0; n < context->audioFrames; n++){
-		if(gIsNoteOn == 1){
-			static float phase = 0;
-			phase += gPhaseIncrement;
-			if(phase > 2 * M_PI)
-				phase -= 2 * M_PI;
-			float value = sinf(phase) * gVelocity/128.0f;
-			audioWrite(context, n, 0, value);
-			audioWrite(context, n, 1, value);
-		} else {
-			audioWrite(context, n, 0, 0);
-			audioWrite(context, n, 1, 0);
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/basic_network/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,98 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <unistd.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	float frequency = 440.0;		// Frequency of oscillator
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				frequency = atof(optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &frequency) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/basic_passthru/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,91 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --help [-h]:             Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "h", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, 0) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/basic_passthru/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,127 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
- /*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-/**
-\example 1_basic_audio_analog_passthrough
-
-Audio and analog passthrough: input to output
------------------------------------------
-
-This sketch demonstrates how to read from and write to the audio and analog input and output buffers.
-
-In `render()` you'll see a nested for loop structure. You'll see this in all Bela projects. 
-The first for loop cycles through `audioFrames`, the second through 
-`audioChannels` (in this case left 0 and right 1).
-
-You can access any information about current audio and sensor settings you can do the following: 
-`context->name_of_item`. For example `context->audioChannels` returns current number of channels,
-`context->audioFrames` returns the current number of audio frames, 
-`context->audioSampleRate` returns the audio sample rate.
-
-You can look at all the information you can access in ::BeagleRTContext.
-
-Reading and writing from the audio buffers
-------------------------------------------
-
-The simplest way to read samples from the audio input buffer is with
-`audioRead()` which we pass three arguments: context, current audio 
-frame and current channel. In this example we have 
-`audioRead(context, n, ch)` where both `n` and `ch` are provided by 
-the nested for loop structure.
-
-We can write samples to the audio output buffer in a similar way using 
-`audioWrite()`. This has a fourth argument which is the value of the output.
-For example `audioWrite(context, n, ch, value_to_output)`.
-
-Reading and writing from the analog buffers
--------------------------------------------
-
-The same is true for `analogRead()` and `analogWrite()`.
-
-Note that for the analog channels we write to and read from the buffers in a separate set 
-of nested for loops. This is because the they are sampled at half audio rate by default. 
-The first of these for loops cycles through `analogFrames`, the second through
-`analogChannels`.
-
-By setting `audioWriteFrame(context, n, ch, audioReadFrame(context, n, ch))` and
-`analogWrite(context, n, ch, analogReadFrame(context, n, ch))` we have a simple 
-passthrough of audio input to output and analog input to output.
-
-
-It is also possible to address the buffers directly, for example: 
-`context->audioOut[n * context->audioChannels + ch]`.
-*/
-
-#include <Bela.h>
-#include <rtdk.h>
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	// Nothing to do here...
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	// Simplest possible case: pass inputs through to outputs
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		for(unsigned int ch = 0; ch < context->audioChannels; ch++){
-			// Two equivalent ways to write this code
-
-			// The long way, using the buffers directly:
-			// context->audioOut[n * context->audioChannels + ch] =
-			// 		context->audioIn[n * context->audioChannels + ch];
-
-			// Or using the macros:
-			audioWrite(context, n, ch, audioRead(context, n, ch));
-		}
-	}
-
-	// Same with analog channelss
-	for(unsigned int n = 0; n < context->analogFrames; n++) {
-		for(unsigned int ch = 0; ch < context->analogChannels; ch++) {
-			// Two equivalent ways to write this code
-
-			// The long way, using the buffers directly:
-			// context->analogOut[n * context->analogChannels + ch] = context->analogIn[n * context->analogChannels + ch];
-
-			// Or using the macros:
-			analogWrite(context, n, ch, analogRead(context, n, ch));
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/basic_pulseIn/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,72 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <Bela.h>
-#include <PulseIn.h>
-#include <stdlib.h>
-#include <rtdk.h>
-#include <cmath>
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-PulseIn pulseIn;
-int gPulseInPin = 0;
-int gDigitalOutPin = 1;
-int gPulseLength = 1234;
-int gSamplesBetweenPulses = 10000;
-
-bool setup(BelaContext *context, void *userData)
-{
-	pinMode(context, 0, gDigitalOutPin, OUTPUT);
-	pulseIn.init(context, gPulseInPin, 1); //third parameter is direction
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	static bool pulseOut = 0;
-	static int count = 0;
-	for(unsigned int n = 0; n < context->digitalFrames; n++){
-		// detect if a pulse just ended
-		int duration = pulseIn.hasPulsed(context, n);
-		if(duration > 0){
-			rt_printf("duration = %d\n", duration);
-		}
-
-		// generate a rectangular waveform as a test signal.
-		// Connect gDigitalOutPin to gPulseInPin
-		// to verify that the detected pulse length is gPulseLength
-		if(count == gPulseLength ){
-			pulseOut = false;
-		}
-		if(count == (gPulseLength + gSamplesBetweenPulses)){
-			pulseOut = true;
-			count = 0;
-		}
-		digitalWrite(context, n, gDigitalOutPin, pulseOut);
-		count++;
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/basic_writeFile/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,97 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-#include <unistd.h>
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	float frequency = 440.0;	// Frequency of oscillator
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				frequency = atof(optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &frequency) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/basic_writeFile/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,63 +0,0 @@
-#include <Bela.h> 
-#include <cmath>
-#include <WriteFile.h>
-
-float gPhase1, gPhase2;
-float gFrequency1, gFrequency2;
-float gInverseSampleRate;
-
-WriteFile file1;
-WriteFile file2;
-
-bool setup(BelaContext *context, void *userData)
-{
-	gInverseSampleRate = 1.0/context->audioSampleRate;
-	file1.init("out1.m"); //set the file name to write to
-	file1.setHeader("myvar=[\n"); //set a line to be printed at the beginning of the file
-	file1.setFooter("];\n"); //set a line to be printed at the end of the file
-	file1.setEcho(true); // enable echo to the console (as well as to the file)
-	file1.setFormat("%.5f %.10f %f\n"); // set the format that you want to use for your output. Please use %f only (with modifiers)
-	file2.init("out2.m");
-	file2.setHeader("input=[\n");
-	file2.setFooter("];\n");
-	file2.setEcho(false);
-	file2.setFormat("%f\n");
-	gPhase1 = 0.0;
-	gPhase2 = 0.0;
-
-	gFrequency1 = 200.0;
-	gFrequency2 = 201.0;
-	return true; 
-}
-
-void render(BelaContext *context, void *userData)
-{
-	static int count = 0;
-	if((count&16383) == 0){
-    	file2.log(context->audioIn, context->audioFrames); //write the input buffer every so often
-	}
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-	    float chn1 = sinf(gPhase1);
-	    float chn2 = sinf(gPhase2);
-	    gPhase1 += 2.0 * M_PI * gFrequency1 * gInverseSampleRate;
-	    gPhase2 += 2.0 * M_PI * gFrequency2 * gInverseSampleRate;
-		if(gPhase1 > 2.0 * M_PI)
-			gPhase1 -= 2.0 * M_PI;
-		if(gPhase2 > 2.0 * M_PI)
-			gPhase2 -= 2.0 * M_PI;
-		if( (count&511) == 0){
-			file1.log(chn1);
-			file1.log(chn2);
-			file1.log(count);
-		}
-		count++;
-	}
-}
-
-// cleanup_render() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in initialise_render().
-
-void cleanup(BelaContext *context, void *userData)
-{
-    
-}
--- a/examples/bucket_brigade_chorus/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,120 +0,0 @@
-#include <Bela.h> 
-#include <Scope.h>
-#include <cmath>
-
-float gPhase1, gPhase2;
-float gFrequency1, gFrequency2;
-float gInverseSampleRate;
-
-// initialise_render() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-#include <I2c_Codec.h>
-#include <PRU.h>
-extern I2c_Codec *gAudioCodec;
-extern PRU *gPRU;
-float D=5264;
-#define delayLength 256
-float delay[delayLength];
-int writePointer=0;
-int readPointer=writePointer+1;
-AuxiliaryTask updatePll;
-
-void updatePllFunction(){
-//	gPRU->setGPIOTestPin();
-	static int count = 0;
-	while(!gShouldStop){
-		gAudioCodec->setPllD(D);
-		count++;
-		if((count&4095)==0)
-			printf("sampling rate: %f\n",gAudioCodec->getAudioSamplingRate());
-		usleep(100);
-	}
-//	gPRU->clearGPIOTestPin();
-}
-
-bool setup(BelaContext *context, void *userData)
-{
-	gInverseSampleRate = 1.0/context->audioSampleRate;
-	
-	gPhase1 = 0.0;
-	gPhase2 = 0.0;
-	
-	gFrequency1 = 200.0;
-	gFrequency2 = 201.0;
-	updatePll=Bela_createAuxiliaryTask(&updatePllFunction, 91, "update PLL");
-	for(int n=0; n<delayLength; n++){
-		delay[n]=0;
-	}
-	return true; 
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-//	printf("here\n");
-	static bool init = false;
-	if(init == false){
-		Bela_scheduleAuxiliaryTask(updatePll);
-//		gAudioCodec->setPllP(2);
-//		gAudioCodec->setPllR();
-//		gAudioCodec->setAudioSamplingRate(43600);
-//		printf("samplingRate: %f, k: %f\n", gAudioCodec->getAudioSamplingRate(), gAudioCodec->getPllK());
-		init = true;
-	}
-	static int count=0;
-	static float lfoPhase=0;
-	static float feedback=0;
-	int updateRate=1;
-	if((count&(updateRate-1))==0){
-		float amplitude = 8000;
-		float rate = 2;
-		lfoPhase+=rate*2*M_PI*updateRate*context->analogFrames/context->audioSampleRate;
-		D=amplitude+amplitude*sinf(lfoPhase);
-		if((count&255)==0){
-//			rt_printf("frequency: %f\n", gAudioCodec->getAudioSamplingRate());
-//			rt_printf("D: %.0f\n", D);
-//			rt_printf("rate: %f\n", rate);
-//			rt_printf("amplitude: %.3f\n", amplitude);
-//			rt_printf("feedback: %.3f\n\n", feedback);
-		}
-	}
-	count++;
-
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		feedback = 0.4;
-		float input = audioRead(context, n, 0) + audioRead(context, n, 1);
-	    delay[writePointer++] = input + delay[readPointer]*feedback;
-	    float output = (input + 0.9*delay[readPointer++] ) * 0.5;
-		audioWrite(context, n, 0, output);
-		audioWrite(context, n, 1, output);
-		if(writePointer>=delayLength)
-			writePointer-=delayLength;
-		if(readPointer>=delayLength)
-			readPointer-=delayLength;
-
-		gPhase1 += 2.0 * M_PI * gFrequency1 * gInverseSampleRate;
-	    gPhase2 += 2.0 * M_PI * gFrequency2 * gInverseSampleRate;
-		if(gPhase1 > 2.0 * M_PI)
-			gPhase1 -= 2.0 * M_PI;
-		if(gPhase2 > 2.0 * M_PI)
-			gPhase2 -= 2.0 * M_PI;
-	}
-}
-
-// cleanup_render() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in initialise_render().
-
-void cleanup(BelaContext *context, void *userData)
-{
-    
-}
--- a/examples/cape_test/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,97 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-#include <unistd.h>
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	float frequency = 440.0;	// Frequency of oscillator
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				frequency = atof(optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &frequency) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/cape_test/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,279 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-
-#include <Bela.h>
-#include <cmath>
-
-#define ANALOG_LOW	(2048.0 / 65536.0)
-#define ANALOG_HIGH (50000.0 / 65536.0)
-
-const int gDACPinOrder[] = {6, 4, 2, 0, 1, 3, 5, 7};
-
-enum {
-	kStateTestingAudioLeft = 0,
-	kStateTestingAudioRight,
-	kStateTestingAudioDone
-};
-
-uint64_t gLastErrorFrame = 0;
-uint32_t gEnvelopeSampleCount = 0;
-float gEnvelopeValueL = 0.5, gEnvelopeValueR = 0.5;
-float gEnvelopeDecayRate = 0.9995;
-int gEnvelopeLastChannel = 0;
-
-float gPositivePeakLevels[2] = {0, 0};
-float gNegativePeakLevels[2] = {0, 0};
-float gPeakLevelDecayRate = 0.999;
-const float gPeakLevelLowThreshold = 0.02;
-const float gPeakLevelHighThreshold = 0.2;
-const float gDCOffsetThreshold = 0.1;
-int gAudioTestState = kStateTestingAudioLeft;
-int gAudioTestStateSampleCount = 0;
-int gAudioTestSuccessCounter = 0;
-const int gAudioTestSuccessCounterThreshold = 64;
-const int gAudioTestStateSampleThreshold = 16384;
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	static float phase = 0.0;
-	static int sampleCounter = 0;
-	static int invertChannel = 0;
-	float frequency = 0;
-
-	// Play a sine wave on the audio output
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		
-		// Peak detection on the audio inputs, with offset to catch
-		// DC errors
-		for(int ch = 0; ch < 2; ch++) {
-			if(context->audioIn[2*n + ch] > gPositivePeakLevels[ch])
-				gPositivePeakLevels[ch] = context->audioIn[2*n + ch];
-			gPositivePeakLevels[ch] += 0.1;
-			gPositivePeakLevels[ch] *= gPeakLevelDecayRate;
-			gPositivePeakLevels[ch] -= 0.1;
-			if(context->audioIn[2*n + ch] < gNegativePeakLevels[ch])
-				gNegativePeakLevels[ch] = context->audioIn[2*n + ch];
-			gNegativePeakLevels[ch] -= 0.1;			
-			gNegativePeakLevels[ch] *= gPeakLevelDecayRate;
-			gNegativePeakLevels[ch] += 0.1;
-		}
-		
-		if(gAudioTestState == kStateTestingAudioLeft) {
-			context->audioOut[2*n] = 0.2 * sinf(phase);
-			context->audioOut[2*n + 1] = 0;		
-			
-			frequency = 3000.0;
-			phase += 2.0 * M_PI * frequency / 44100.0;
-			if(phase >= 2.0 * M_PI)
-				phase -= 2.0 * M_PI;
-			
-			gAudioTestStateSampleCount++;
-			if(gAudioTestStateSampleCount >= gAudioTestStateSampleThreshold) {
-				// Check if we have the expected input: signal on the left but not
-				// on the right. Also check that there is not too much DC offset on the
-				// inactive signal
-				if((gPositivePeakLevels[0] - gNegativePeakLevels[0]) >= gPeakLevelHighThreshold 
-					&& (gPositivePeakLevels[1] -  gNegativePeakLevels[1]) <= gPeakLevelLowThreshold &&
-					fabsf(gPositivePeakLevels[1]) < gDCOffsetThreshold &&
-					fabsf(gNegativePeakLevels[1]) < gDCOffsetThreshold) {
-					// Successful test: increment counter
-					gAudioTestSuccessCounter++;
-					if(gAudioTestSuccessCounter >= gAudioTestSuccessCounterThreshold) {
-						gAudioTestState = kStateTestingAudioRight;
-						gAudioTestStateSampleCount = 0;
-						gAudioTestSuccessCounter = 0;
-					}
-
-				}
-				else {
-					if(!((context->audioFramesElapsed + n) % 22050)) {
-						// Debugging print messages
-						if((gPositivePeakLevels[0] - gNegativePeakLevels[0]) < gPeakLevelHighThreshold)
-							rt_printf("Left Audio In FAIL: insufficient signal: %f\n", 
-										gPositivePeakLevels[0] - gNegativePeakLevels[0]);
-						else if(gPositivePeakLevels[1] -  gNegativePeakLevels[1] > gPeakLevelLowThreshold)
-							rt_printf("Right Audio In FAIL: signal present when it should not be: %f\n",
-										gPositivePeakLevels[1] -  gNegativePeakLevels[1]);
-						else if(fabsf(gPositivePeakLevels[1]) >= gDCOffsetThreshold ||
-								fabsf(gNegativePeakLevels[1]) >= gDCOffsetThreshold)
-							rt_printf("Right Audio In FAIL: DC offset: (%f, %f)\n",
-										gPositivePeakLevels[1], gNegativePeakLevels[1]);						
-					}
-					gAudioTestSuccessCounter--;
-					if(gAudioTestSuccessCounter <= 0)
-						gAudioTestSuccessCounter = 0;
-				}
-			}
-		}
-		else if(gAudioTestState == kStateTestingAudioRight) {
-			context->audioOut[2*n] = 0;
-			context->audioOut[2*n + 1] = 0.2 * sinf(phase);
-			
-			frequency = 3000.0;
-			phase += 2.0 * M_PI * frequency / 44100.0;
-			if(phase >= 2.0 * M_PI)
-				phase -= 2.0 * M_PI;
-			
-			gAudioTestStateSampleCount++;
-			if(gAudioTestStateSampleCount >= gAudioTestStateSampleThreshold) {
-				// Check if we have the expected input: signal on the left but not
-				// on the right
-				if((gPositivePeakLevels[1] - gNegativePeakLevels[1]) >= gPeakLevelHighThreshold 
-					&& (gPositivePeakLevels[0] -  gNegativePeakLevels[0]) <= gPeakLevelLowThreshold &&
-					fabsf(gPositivePeakLevels[0]) < gDCOffsetThreshold &&
-					fabsf(gNegativePeakLevels[0]) < gDCOffsetThreshold) {
-					// Successful test: increment counter
-					gAudioTestSuccessCounter++;
-					if(gAudioTestSuccessCounter >= gAudioTestSuccessCounterThreshold) {
-						gAudioTestSuccessCounter = 0;							
-						gAudioTestStateSampleCount = 0;
-						gAudioTestState = kStateTestingAudioDone;
-					}
-				}
-				else {
-					if(!((context->audioFramesElapsed + n) % 22050)) {
-						// Debugging print messages
-						if((gPositivePeakLevels[1] - gNegativePeakLevels[1]) < gPeakLevelHighThreshold)
-							rt_printf("Right Audio In FAIL: insufficient signal: %f\n", 
-										gPositivePeakLevels[1] - gNegativePeakLevels[1]);
-						else if(gPositivePeakLevels[0] -  gNegativePeakLevels[0] > gPeakLevelLowThreshold)
-							rt_printf("Left Audio In FAIL: signal present when it should not be: %f\n",
-										gPositivePeakLevels[0] -  gNegativePeakLevels[0]);
-						else if(fabsf(gPositivePeakLevels[0]) >= gDCOffsetThreshold ||
-								fabsf(gNegativePeakLevels[0]) >= gDCOffsetThreshold)
-							rt_printf("Left Audio In FAIL: DC offset: (%f, %f)\n",
-										gPositivePeakLevels[0], gNegativePeakLevels[0]);						
-					}
-					gAudioTestSuccessCounter--;
-					if(gAudioTestSuccessCounter <= 0)
-						gAudioTestSuccessCounter = 0;
-				}
-			}			
-		}
-		else {
-			// Audio input testing finished. Play tones depending on status of
-			// analog testing
-			context->audioOut[2*n] = gEnvelopeValueL * sinf(phase);
-			context->audioOut[2*n + 1] = gEnvelopeValueR * sinf(phase);
-
-			// If one second has gone by with no error, play one sound, else
-			// play another
-			if(context->audioFramesElapsed + n - gLastErrorFrame > 44100) {
-				gEnvelopeValueL *= gEnvelopeDecayRate;
-				gEnvelopeValueR *= gEnvelopeDecayRate;
-				gEnvelopeSampleCount++;
-				if(gEnvelopeSampleCount > 22050) {
-					if(gEnvelopeLastChannel == 0)
-						gEnvelopeValueR = 0.5;
-					else
-						gEnvelopeValueL = 0.5;
-					gEnvelopeLastChannel = !gEnvelopeLastChannel;
-					gEnvelopeSampleCount = 0;
-				}
-				frequency = 880.0;
-			}
-			else {
-				gEnvelopeValueL = gEnvelopeValueR = 0.5;
-				gEnvelopeLastChannel = 0;
-				frequency = 220.0;
-			}
-
-			phase += 2.0 * M_PI * frequency / 44100.0;
-			if(phase >= 2.0 * M_PI)
-				phase -= 2.0 * M_PI;
-		}
-	}
-
-	for(unsigned int n = 0; n < context->analogFrames; n++) {
-		// Change outputs every 512 samples
-		if(sampleCounter < 512) {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel)
-					context->analogOut[n*8 + gDACPinOrder[k]] = ANALOG_HIGH;
-				else
-					context->analogOut[n*8 + gDACPinOrder[k]] = 0;
-			}
-		}
-		else {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel)
-					context->analogOut[n*8 + gDACPinOrder[k]] = 0;
-				else
-					context->analogOut[n*8 + gDACPinOrder[k]] = ANALOG_HIGH;
-			}
-		}
-
-		// Read after 256 samples: input should be low
-		if(sampleCounter == 256) {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel) {
-					if(context->analogIn[n*8 + k] < ANALOG_HIGH) {
-						rt_printf("FAIL [output %d, input %d] -- output HIGH input %f (inverted)\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
-						gLastErrorFrame = context->audioFramesElapsed + n;
-					}
-				}
-				else {
-					if(context->analogIn[n*8 + k] > ANALOG_LOW) {
-						rt_printf("FAIL [output %d, input %d] -- output LOW --> input %f\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
-						gLastErrorFrame = context->audioFramesElapsed + n;
-					}
-				}
-			}
-		}
-		else if(sampleCounter == 768) {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel) {
-					if(context->analogIn[n*8 + k] > ANALOG_LOW) {
-						rt_printf("FAIL [output %d, input %d] -- output LOW input %f (inverted)\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
-						gLastErrorFrame = context->audioFramesElapsed + n;
-					}
-				}
-				else {
-					if(context->analogIn[n*8 + k] < ANALOG_HIGH) {
-						rt_printf("FAIL [output %d, input %d] -- output HIGH input %f\n", gDACPinOrder[k], k, context->analogIn[n*8 + k]);
-						gLastErrorFrame = context->audioFramesElapsed + n;
-					}
-				}
-			}
-		}
-
-		if(++sampleCounter >= 1024) {
-			sampleCounter = 0;
-			invertChannel++;
-			if(invertChannel >= 8)
-				invertChannel = 0;
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/core/link_core.sh	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,3 +0,0 @@
-#!/bin/sh
-ln -f -s ../../core/ source
-ln -f -s ../../include/ include
--- a/examples/core/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,16 +0,0 @@
-#include <Bela.h>
-
-bool setup(BelaContext *context, void *userData)
-{
-	return true;
-}
-
-void render(BelaContext *context, void *userData)
-{
-
-}
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
\ No newline at end of file
--- a/examples/d-box/ADSR.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,76 +0,0 @@
-//
-//  ADSR.cpp
-//
-//  Created by Nigel Redmon on 12/18/12.
-//  EarLevel Engineering: earlevel.com
-//  Copyright 2012 Nigel Redmon
-//
-//  For a complete explanation of the ADSR envelope generator and code,
-//  read the series of articles by the author, starting here:
-//  http://www.earlevel.com/main/2013/06/01/envelope-generators/
-//
-//  License:
-//
-//  This source code is provided as is, without warranty.
-//  You may copy and distribute verbatim copies of this document.
-//  You may modify and use this source code to create binary code for your own purposes, free or commercial.
-//
-
-#include "ADSR.h"
-#include <math.h>
-
-
-ADSR::ADSR(void) {
-    reset();
-    setAttackRate(0);
-    setDecayRate(0);
-    setReleaseRate(0);
-    setSustainLevel(1.0);
-    setTargetRatioA(0.3);
-    setTargetRatioDR(0.0001);
-}
-
-ADSR::~ADSR(void) {
-}
-
-void ADSR::setAttackRate(float rate) {
-    attackRate = rate;
-    attackCoef = calcCoef(rate, targetRatioA);
-    attackBase = (1.0 + targetRatioA) * (1.0 - attackCoef);
-}
-
-void ADSR::setDecayRate(float rate) {
-    decayRate = rate;
-    decayCoef = calcCoef(rate, targetRatioDR);
-    decayBase = (sustainLevel - targetRatioDR) * (1.0 - decayCoef);
-}
-
-void ADSR::setReleaseRate(float rate) {
-    releaseRate = rate;
-    releaseCoef = calcCoef(rate, targetRatioDR);
-    releaseBase = -targetRatioDR * (1.0 - releaseCoef);
-}
-
-float ADSR::calcCoef(float rate, float targetRatio) {
-    return exp(-log((1.0 + targetRatio) / targetRatio) / rate);
-}
-
-void ADSR::setSustainLevel(float level) {
-    sustainLevel = level;
-    decayBase = (sustainLevel - targetRatioDR) * (1.0 - decayCoef);
-}
-
-void ADSR::setTargetRatioA(float targetRatio) {
-    if (targetRatio < 0.000000001)
-        targetRatio = 0.000000001;  // -180 dB
-    targetRatioA = targetRatio;
-    attackBase = (1.0 + targetRatioA) * (1.0 - attackCoef);
-}
-
-void ADSR::setTargetRatioDR(float targetRatio) {
-    if (targetRatio < 0.000000001)
-        targetRatio = 0.000000001;  // -180 dB
-    targetRatioDR = targetRatio;
-    decayBase = (sustainLevel - targetRatioDR) * (1.0 - decayCoef);
-    releaseBase = -targetRatioDR * (1.0 - releaseCoef);
-}
--- a/examples/d-box/ADSR.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,137 +0,0 @@
-//
-//  ADRS.h
-//
-//  Created by Nigel Redmon on 12/18/12.
-//  EarLevel Engineering: earlevel.com
-//  Copyright 2012 Nigel Redmon
-//
-//  For a complete explanation of the ADSR envelope generator and code,
-//  read the series of articles by the author, starting here:
-//  http://www.earlevel.com/main/2013/06/01/envelope-generators/
-//
-//  License:
-//
-//  This source code is provided as is, without warranty.
-//  You may copy and distribute verbatim copies of this document.
-//  You may modify and use this source code to create binary code for your own purposes, free or commercial.
-//
-
-#ifndef ADRS_h
-#define ADRS_h
-
-#include <stdio.h>
-#include <string>
-
-using namespace std;
-
-enum envState {
-	env_idle = 0,
-	env_attack,
-	env_decay,
-	env_sustain,
-	env_release
-};
-
-class ADSR {
-public:
-	ADSR(void);
-	~ADSR(void);
-	float process(void);
-	float process(int sampleCount);
-    float getOutput(void);
-    int getState(void);
-	void gate(int on);
-    void setAttackRate(float rate);
-    void setDecayRate(float rate);
-    void setReleaseRate(float rate);
-	void setSustainLevel(float level);
-    void setTargetRatioA(float targetRatio);
-    void setTargetRatioDR(float targetRatio);
-    void reset(void);
-
-protected:
-	int state;
-	float output;
-	float attackRate;
-	float decayRate;
-	float releaseRate;
-	float attackCoef;
-	float decayCoef;
-	float releaseCoef;
-	float sustainLevel;
-    float targetRatioA;
-    float targetRatioDR;
-    float attackBase;
-    float decayBase;
-    float releaseBase;
-	string name;
-    float calcCoef(float rate, float targetRatio);
-};
-
-inline float ADSR::process() {
-	switch (state) {
-        case env_idle:
-            break;
-        case env_attack:
-            output = attackBase + output * attackCoef;
-            if (output >= 1.0) {
-                output = 1.0;
-                state = env_decay;
-            }
-            break;
-        case env_decay:
-            output = decayBase + output * decayCoef;
-            if (output <= sustainLevel) {
-                output = sustainLevel;
-                state = env_sustain;
-            }
-            break;
-        case env_sustain:
-            break;
-        case env_release:
-            output = releaseBase + output * releaseCoef;
-            if (output <= 0.0) {
-                output = 0.0;
-                state = env_idle;
-            }
-            break;
-	}
-	return output;
-}
-
-inline float ADSR::process(int sampleCount)
-{
-	float retVal = 0;
-
-	if(state != env_idle)
-	{
-		for(int i=0; i<sampleCount; i++)
-			retVal = process();
-	}
-
-	return retVal;
-}
-
-inline void ADSR::gate(int gate) {
-
-	if (gate)
-		state = env_attack;
-	else if (state != env_idle)
-        state = env_release;
-}
-
-inline int ADSR::getState() {
-    return state;
-}
-
-inline void ADSR::reset() {
-    state = env_idle;
-    output = 0.0;
-}
-
-inline float ADSR::getOutput() {
-	return output;
-}
-
-
-#endif
--- a/examples/d-box/AnalogInput.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,163 +0,0 @@
-/*
- * AnalogInput.cpp
- *
- *  Created on: Oct 17, 2013
- *      Author: Victor Zappi
- */
-
-
-#include "AnalogInput.h"
-
-using namespace std;
-
-
-AnalogInput::AnalogInput()
-{
-	ActivateAnalogHnd  = NULL;
-	activateAnalogPath = "";
-	analogIsSet        = false;
-
-	AnalogInHnd    = NULL;
-	analogInPath   = "";
-	helperNumFound = false;
-
-	// support var for init
-	// these are fixed for BBB
-	startPath = "/sys/devices/bone_capemgr.*/slots";
-	readPath = "";
-
-	buffer = (char*) malloc (sizeof(char)*lSize); // reading buffer
-
-	verbose = false;
-}
-
-
-AnalogInput::~AnalogInput()
-{
-	free(buffer);
-}
-
-int AnalogInput::initAnalogInputs()
-{
-	 if(analogIsSet)
-	 {
-		 if(verbose)
-			 cout << "Fine, but Analog Input already active..."<< endl;
-		 return 0;
-	 }
-
-	// title
-	 if(verbose)
-	 	cout << "Init Analog Input"<< endl;
-
-
-	  // first: activate analog pins on cape manager
-	  // cape-bone-iio > /sys/devices/bone_capemgr.*/slots
-
-	  // we have to look for the semi-random number the BBB has initialized the bone_capemgr with [value of *]
-	  // to reach /slots and set cape-bone-iio.
-	  // to do so, we use glob lib, which translates wildcards [*] into all the values found in paths
-
-
-	  glob( startPath.c_str(), 0, NULL, &globbuf);
-
-	  if(globbuf.gl_pathc >0)
-	  {
-	    if (globbuf.gl_pathc == 1 )
-	    {
-	      activateAnalogPath = globbuf.gl_pathv[0];
-
-	      // check if file is existing
-	      if((ActivateAnalogHnd = fopen(activateAnalogPath.c_str(), "r+")) != NULL)
-	      {
-	        // we found that current capemgr num
-
-			fwrite("cape-bone-iio", sizeof(char), 13, ActivateAnalogHnd); // activate pins
-
-	        analogIsSet = true;
-
-			if(verbose)
-				cout << "Analog Pins activated via cape-bone-iio at path " << activateAnalogPath << endl;
-
-			fclose(ActivateAnalogHnd); // close file
-	      }
-	    }
-	    //else
-	      //printf("toomany", );
-	  }
-
-	  globfree(&globbuf); // self freeing
-
-
-	  if(!analogIsSet)
-	  {
-	    cout << "cannot find bone_capemgr" << endl;
-	    cout << "------Init failed------" << endl;
-	    return 1;
-	  }
-
-
-	  // build read path
-	  startPath = "/sys/devices/ocp.2/helper.*";
-
-	  glob( startPath.c_str(), 0, NULL, &globbuf);
-
-	  if(globbuf.gl_pathc >0)
-	  {
-		if (globbuf.gl_pathc == 1 )
-		{
-		  analogInPath = globbuf.gl_pathv[0] + (string)"/AIN";
-		}
-	    else
-	      cout << "Too many analog inputs with this name! [I am puzzled...]" << endl;
-	  }
-	  else
-	    cout << "Cannot find analog input dir...puzzled" << endl;
-
-
-	  return 0;
-}
-
-
-int AnalogInput::read(int index)
-{
-	  // convert int index into string
-	  stringstream ss;
-	  ss << index;
-
-	  readPath = analogInPath + ss.str(); // create pin0 file path
-
-
-	  // check if file is existing
-	  if((AnalogInHnd = fopen(readPath.c_str(), "rb")) != NULL)
-	  {
-		// we found that current helper num
-
-		// prepare read buffer to reading
-		fseek (AnalogInHnd , 0 , SEEK_END);
-		lSize = ftell (AnalogInHnd);
-		rewind (AnalogInHnd);
-
-		result = fread (buffer, 1, lSize, AnalogInHnd);
-
-		fclose(AnalogInHnd); // close file
-
-		helperNumFound = true;
-
-		//cout << "Analog Pins can be read at path " << analogInPath << endl;
-		//cout << "Test reading of Pin0 gives: " << buffer << endl;
-	  }
-
-	  if(!helperNumFound)
-	  {
-	    cout << "cannot find helper" << endl;
-	    cout << "------Analog Read failed------" << endl;
-	    return -1;
-	  }
-
-	  return atoi(buffer);
-
-}
-
-
-
--- a/examples/d-box/AnalogInput.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,55 +0,0 @@
-/*
- * AnalogInput.h
- *
- *  Created on: Oct 17, 2013
- *      Author: Victor Zappi
- */
-
-#ifndef ANALOGINPUT_H_
-#define ANALOGINPUT_H_
-
-#include <iostream>
-#include <sstream>
-#include <stdio.h>
-#include <stdlib.h>
-#include <glob.h>
-
-using namespace std;
-
-class AnalogInput
-{
-private:
-	FILE *ActivateAnalogHnd;
-	string activateAnalogPath;
-	bool analogIsSet;
-
-	FILE *AnalogInHnd;
-	string analogInPath;
-	bool helperNumFound;
-
-	// suport var for init
-	string startPath;
-	string readPath;
-
-	glob_t  globbuf;
-
-	// support vars for pin reading
-	long lSize;
-	char * buffer;
-	size_t result;
-
-	bool verbose;
-
-public:
-	AnalogInput();
-	~AnalogInput();
-
-	int initAnalogInputs();
-	int read(int index);
-
-};
-
-
-
-
-#endif /* ANALOGINPUT_H_ */
--- a/examples/d-box/Biquad.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,169 +0,0 @@
-//
-//  Biquad.cpp
-//
-//  Created by Nigel Redmon on 11/24/12
-//  EarLevel Engineering: earlevel.com
-//  Copyright 2012 Nigel Redmon
-//
-//  For a complete explanation of the Biquad code:
-//  http://www.earlevel.com/main/2012/11/26/biquad-c-source-code/
-//
-//  License:
-//
-//  This source code is provided as is, without warranty.
-//  You may copy and distribute verbatim copies of this document.
-//  You may modify and use this source code to create binary code
-//  for your own purposes, free or commercial.
-//
-
-#include <math.h>
-#include "Biquad.h"
-#include <iostream>
-
-Biquad::Biquad() {
-    type = bq_type_lowpass;
-    a0 = 1.0;
-    a1 = a2 = b1 = b2 = 0.0;
-    Fc = 0.50;
-    Q = 0.707;
-    peakGain = 0.0;
-    z1 = z2 = 0.0;
-}
-
-Biquad::Biquad(int type, double Fc, double Q, double peakGainDB) {
-    setBiquad(type, Fc, Q, peakGainDB);
-    z1 = z2 = 0.0;
-}
-
-Biquad::~Biquad() {
-}
-
-void Biquad::setType(int type) {
-    this->type = type;
-    calcBiquad();
-}
-
-void Biquad::setQ(double Q) {
-    this->Q = Q;
-    calcBiquad();
-}
-
-void Biquad::setFc(double Fc) {
-    this->Fc = Fc;
-    calcBiquad();
-}
-
-void Biquad::setPeakGain(double peakGainDB) {
-    this->peakGain = peakGainDB;
-    calcBiquad();
-}
-    
-void Biquad::setBiquad(int type, double Fc, double Q, double peakGainDB) {
-    this->type = type;
-    this->Q = Q;
-    this->Fc = Fc;
-    startFc = Fc;
-    startQ = Q;
-    startPeakGain = peakGainDB;
-    setPeakGain(peakGainDB);
-}
-
-void Biquad::calcBiquad(void) {
-    double norm;
-    double V = pow(10, fabs(peakGain) / 20.0);
-    double K = tan(M_PI * Fc);
-    switch (this->type) {
-        case bq_type_lowpass:
-            norm = 1 / (1 + K / Q + K * K);
-            a0 = K * K * norm;
-            a1 = 2 * a0;
-            a2 = a0;
-            b1 = 2 * (K * K - 1) * norm;
-            b2 = (1 - K / Q + K * K) * norm;
-            break;
-            
-        case bq_type_highpass:
-            norm = 1 / (1 + K / Q + K * K);
-            a0 = 1 * norm;
-            a1 = -2 * a0;
-            a2 = a0;
-            b1 = 2 * (K * K - 1) * norm;
-            b2 = (1 - K / Q + K * K) * norm;
-            break;
-            
-        case bq_type_bandpass:
-            norm = 1 / (1 + K / Q + K * K);
-            a0 = K / Q * norm;
-            a1 = 0;
-            a2 = -a0;
-            b1 = 2 * (K * K - 1) * norm;
-            b2 = (1 - K / Q + K * K) * norm;
-            break;
-            
-        case bq_type_notch:
-            norm = 1 / (1 + K / Q + K * K);
-            a0 = (1 + K * K) * norm;
-            a1 = 2 * (K * K - 1) * norm;
-            a2 = a0;
-            b1 = a1;
-            b2 = (1 - K / Q + K * K) * norm;
-            break;
-            
-        case bq_type_peak:
-            if (peakGain >= 0) {    // boost
-                norm = 1 / (1 + 1/Q * K + K * K);
-                a0 = (1 + V/Q * K + K * K) * norm;
-                a1 = 2 * (K * K - 1) * norm;
-                a2 = (1 - V/Q * K + K * K) * norm;
-                b1 = a1;
-                b2 = (1 - 1/Q * K + K * K) * norm;
-            }
-            else {    // cut
-                norm = 1 / (1 + V/Q * K + K * K);
-                a0 = (1 + 1/Q * K + K * K) * norm;
-                a1 = 2 * (K * K - 1) * norm;
-                a2 = (1 - 1/Q * K + K * K) * norm;
-                b1 = a1;
-                b2 = (1 - V/Q * K + K * K) * norm;
-            }
-            break;
-        case bq_type_lowshelf:
-            if (peakGain >= 0) {    // boost
-                norm = 1 / (1 + sqrt(2) * K + K * K);
-                a0 = (1 + sqrt(2*V) * K + V * K * K) * norm;
-                a1 = 2 * (V * K * K - 1) * norm;
-                a2 = (1 - sqrt(2*V) * K + V * K * K) * norm;
-                b1 = 2 * (K * K - 1) * norm;
-                b2 = (1 - sqrt(2) * K + K * K) * norm;
-            }
-            else {    // cut
-                norm = 1 / (1 + sqrt(2*V) * K + V * K * K);
-                a0 = (1 + sqrt(2) * K + K * K) * norm;
-                a1 = 2 * (K * K - 1) * norm;
-                a2 = (1 - sqrt(2) * K + K * K) * norm;
-                b1 = 2 * (V * K * K - 1) * norm;
-                b2 = (1 - sqrt(2*V) * K + V * K * K) * norm;
-            }
-            break;
-        case bq_type_highshelf:
-            if (peakGain >= 0) {    // boost
-                norm = 1 / (1 + sqrt(2) * K + K * K);
-                a0 = (V + sqrt(2*V) * K + K * K) * norm;
-                a1 = 2 * (K * K - V) * norm;
-                a2 = (V - sqrt(2*V) * K + K * K) * norm;
-                b1 = 2 * (K * K - 1) * norm;
-                b2 = (1 - sqrt(2) * K + K * K) * norm;
-            }
-            else {    // cut
-                norm = 1 / (V + sqrt(2*V) * K + K * K);
-                a0 = (1 + sqrt(2) * K + K * K) * norm;
-                a1 = 2 * (K * K - 1) * norm;
-                a2 = (1 - sqrt(2) * K + K * K) * norm;
-                b1 = 2 * (K * K - V) * norm;
-                b2 = (V - sqrt(2*V) * K + K * K) * norm;
-            }
-            break;
-    }
-    
-    return;
-}
--- a/examples/d-box/Biquad.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,99 +0,0 @@
-//
-//  Biquad.h
-//
-//  Created by Nigel Redmon on 11/24/12
-//  EarLevel Engineering: earlevel.com
-//  Copyright 2012 Nigel Redmon
-//
-//  For a complete explanation of the Biquad code:
-//  http://www.earlevel.com/main/2012/11/25/biquad-c-source-code/
-//
-//  License:
-//
-//  This source code is provided as is, without warranty.
-//  You may copy and distribute verbatim copies of this document.
-//  You may modify and use this source code to create binary code
-//  for your own purposes, free or commercial.
-//
-
-#ifndef Biquad_h
-#define Biquad_h
-
-enum {
-    bq_type_lowpass = 0,
-    bq_type_highpass,
-    bq_type_bandpass,
-    bq_type_notch,
-    bq_type_peak,
-    bq_type_lowshelf,
-    bq_type_highshelf
-};
-
-class Biquad {
-public:
-    Biquad();
-    Biquad(int type, double Fc, double Q, double peakGainDB);
-    ~Biquad();
-    void setType(int type);
-    void setQ(double Q);
-    void setFc(double Fc);
-    void setPeakGain(double peakGainDB);
-    void setBiquad(int type, double Fc, double Q, double peakGain);
-    float process(float in);
-    
-    double getQ();
-    double getFc();
-    double getPeakGain();
-
-    double getStartingQ();
-	double getStartingFc();
-	double getStartingPeakGain();
-
-protected:
-    void calcBiquad(void);
-
-    int type;
-    double a0, a1, a2, b1, b2;
-    double Fc, Q, peakGain;
-    double startFc, startQ, startPeakGain;
-    double z1, z2;
-};
-
-inline double Biquad::getQ()
-{
-	return Q;
-}
-
-inline double Biquad::getFc()
-{
-	return Fc;
-}
-
-inline double Biquad::getPeakGain()
-{
-	return peakGain;
-}
-
-inline double Biquad::getStartingQ()
-{
-	return startQ;
-}
-
-inline double Biquad::getStartingFc()
-{
-	return startFc;
-}
-
-inline double Biquad::getStartingPeakGain()
-{
-	return startPeakGain;
-}
-
-inline float Biquad::process(float in) {
-    double out = in * a0 + z1;
-    z1 = in * a1 + z2 - b1 * out;
-    z2 = in * a2 - b2 * out;
-    return out;
-}
-
-#endif // Biquad_h
--- a/examples/d-box/DBoxSynth.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,35 +0,0 @@
-/*
- * SimpleSynth.h
- *
- *  Created on: Oct 22, 2013
- *      Author: Victor Zappi
- */
-
-#ifndef DBOXSYNTH_H_
-#define DBOXSYNTH_H_
-
-#include <iostream>
-#include <string>
-#include <stdio.h>
-#include <stdlib.h>
-
-#include "Synth.h"
-
-
-class DBoxSynth : public Synth
-{
-public:
-	DBoxSynth(unsigned int rate, unsigned long buffer_size);
-	double getSample();
-	double *getBlock(int block_size);
-
-
-private:
-	Sampler *smp;
-
-};
-
-
-
-
-#endif /* DBOXSYNTH_H_ */
--- a/examples/d-box/DboxSensors.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,157 +0,0 @@
-/*
- * DboxSensors.cpp
- *
- *  Created on: May 19, 2014
- *      Author: Victor Zappi
- */
-
-
-#include "DboxSensors.h"
-#include "config.h"
-
-using namespace std;
-
-
-
-int DboxSensors::initSensors(int tk0_bus, int tk0_address, int tk1_bus, int tk1_address, int tk_file, int fsr_pin, int fsrmax, int sensorTypeToUse, int gpio_0, int gpio_1)
-{
-	sensorType = sensorTypeToUse;
-	// init first touch key on i2c bus
-	if(tk0_address >= 0) {
-		if(TK0.initI2C_RW(tk0_bus, tk0_address, tk_file)>0)
-			return 1;
-		if(TK0.initTouchKey(sensorType)>0)
-			return 2;
-	}
-
-	// init second touch key on i2c bus
-	if(tk1_address >= 0) {
-		if(TK1.initI2C_RW(tk1_bus, tk1_address, tk_file)>0)
-			return 1;
-		if(TK1.initTouchKey(sensorType)>0)
-			return 2;
-	}
-
-	// init fsr on analog input pin
-	fsr_pinNum	= fsr_pin;
-	fsr_max	  	= fsrmax;
-
-	if(FSR.initAnalogInputs()>0)
-		return 3;
-
-	gpio[0] = gpio_0;
-	if(gpio[0]!=-1)
-	{
-		fdDi[0] = gpio_export(gpio[0]);
-		if(fdDi[0] == -1)
-			return 4;
-	}
-	digitalIn[0] = 1;
-
-	return 0;
-}
-
-
-int DboxSensors::readSensors()
-{
-	// write data into first touch key
-	if(TK0.ready()) {
-		if(TK0.readI2C()>0)
-			return 1;
-
-		// retrieve data from first touch key
-		tk0_touchCnt = TK0.getTouchCount();
-	}
-	else
-		tk0_touchCnt = 0;
-
-	// write data into second touch key
-	if(TK1.ready()) {
-		if(TK1.readI2C()>0)
-			return 1;
-		// retrieve data from second touch key
-		tk1_touchCnt = TK1.getTouchCount();
-	}
-	else
-		tk1_touchCnt = 0;
-
-
-	int max = 3;
-	if(sensorType != kSensorTypeTouchKey)
-		max = 5;
-	// if touches detected on main touch key
-	if(tk0_touchCnt == 0 && tk1_touchCnt == 0)
-		resetSensorsData();
-	else
-	{
-		for(int i=0; i<max; i++)
-		{
-			tk0_touchPosX[i] = TK0.getSliderPosition()[i];
-			tk0_touchSize[i] = TK0.getSlidersize()[i];
-
-			tk1_touchPosX[i] = TK1.getSliderPosition()[i];
-			tk1_touchSize[i] = TK1.getSlidersize()[i];
-		}
-		tk0_touchPosY 	 = TK0.getSliderPositionH();
-		tk1_touchPosY 	 = TK1.getSliderPositionH();
-		fsr_read		 = (double)FSR.read(fsr_pinNum);
-	}
-
-	if(gpio[0]!=-1)
-	{
-		if(gpio_read(fdDi[0], &digitalIn[0])==-1)
-			return 1;
-	}
-
-	return 0;
-}
-
-
-
-DboxSensors::DboxSensors()
-{
-	resetSensorsData();
-}
-
-
-
-DboxSensors::~DboxSensors()
-{
-	if(gpio[0]!=-1)
-		gpio_dismiss(fdDi[0], gpio[0]);
-}
-
-
-
-//--------------------------------------------------------------------------------------------------------
-// private methods
-//--------------------------------------------------------------------------------------------------------
-
-// idle values
-void DboxSensors::resetSensorsData()
-{
-	int max = 3;
-	if(sensorType != kSensorTypeTouchKey)
-		max = 5;
-
-	for(int i=0; i<max; i++)
-	{
-		tk0_touchPosX[i] = -1;
-		tk0_touchPosY	 = -1;
-		tk0_touchSize[i] = 0;
-
-		tk1_touchPosX[i] = -1;
-		tk1_touchPosY	 = -1;
-		tk1_touchSize[i] = 0;
-
-		fsr_read		 = 0;
-	}
-
-	return;
-}
-
-
-
-
-
-
--- a/examples/d-box/DboxSensors.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,128 +0,0 @@
-/*
- * DboxSensors.h
- *
- *  Created on: May 19, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef DBOXSENSORS_H_
-#define DBOXSENSORS_H_
-
-#include <stdio.h>
-#include <sys/mount.h>	// mount()
-#include <string.h> 	// strerror()
-#include <fstream> 		// fstream
-#include <iostream>
-#include <unistd.h> 	// usleep()
-#include <glob.h>		// glob()
-#include <sys/time.h>	// elapsed time
-#include <sys/stat.h>	// mkdir()
-#include <algorithm>	// reverse() [string...]
-
-#include "I2c_TouchKey.h"
-#include "AnalogInput.h"
-#include <GPIOcontrol.h>	// TODO wrap this into a class
-
-/*---------------------------------------------------------------------------------------------------------------------------------------------------
- * This class retrieves data from all the connected sensors,
- * logs them
- * and exposes to the main only the values needed to synthesize sound
- *
- * The simple instrument has:
- *
- *
- *
- *---------------------------------------------------------------------------------------------------------------------------------------------------
- */
-class DboxSensors
-{
-public:
-	int initSensors(int tk0_bus, int tk0_address, int tk1_bus, int tk1_address, int tk_file, int fsr_pin, int fsrmax, int sensorTypeToUse, int gpio0=-1, int gpio1=-1);
-	int readSensors();
-	int getTKTouchCount(int index);
-	float *getTKXPositions(int index);
-	float getTKYPosition(int index);
-	float *getTKTouchSize(int index);
-	double getFSRVAlue();
-	int getDigitalIn(int index);
-
-	DboxSensors();
-	~DboxSensors();
-
-private:
-	int sensorType;
-
-	I2c_TouchKey TK0;
-	int tk0_touchCnt;
-	float tk0_touchPosX[5];
-	float tk0_touchPosY;
-	float tk0_touchSize[5];
-
-	I2c_TouchKey TK1;
-	int tk1_touchCnt;
-	float tk1_touchPosX[5];
-	float tk1_touchPosY;
-	float tk1_touchSize[5];
-
-	AnalogInput FSR;
-	int fsr_pinNum;
-	double fsr_read;
-	int fsr_max;
-
-	unsigned int digitalIn[2];
-	int fdDi[2];
-	int gpio[2];
-
-	void resetSensorsData();
-
-};
-
-
-
-//--------------------------------------------------------------------------------
-// read interface
-inline int DboxSensors::getTKTouchCount(int index)
-{
-	if(index==0)
-		return tk0_touchCnt;
-	else
-		return tk1_touchCnt;
-}
-
-inline float *DboxSensors::getTKXPositions(int index)
-{
-	if(index==0)
-		return tk0_touchPosX;
-	else
-		return tk1_touchPosX;
-}
-
-inline float DboxSensors::getTKYPosition(int index)
-{
-	if(index==0)
-		return tk0_touchPosY;
-	else
-		return tk1_touchPosY;
-}
-
-inline float *DboxSensors::getTKTouchSize(int index)
-{
-	if(index==0)
-		return tk0_touchSize;
-	else
-		return tk1_touchSize;
-}
-
-inline double DboxSensors::getFSRVAlue()
-{
-	return fsr_read;
-}
-
-inline int DboxSensors::getDigitalIn(int index)
-{
-	return digitalIn[index];
-}
-//--------------------------------------------------------------------------------
-
-
-#endif /* DBOXSENSORS_H_ */
--- a/examples/d-box/FIRfilter.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,73 +0,0 @@
-/*
- * FIRfilter.h
- *
- *  Created on: Aug 5, 2014
- *      Author: Victor Zappi and Andrew McPherson
- */
-
-#ifndef FIRFILTER_H_
-#define FIRFILTER_H_
-
-#define ENABLE_NE10_FIR_FLOAT_NEON	// Define needed for Ne10 library
-#include <ne10/NE10.h>
-
-//#define FILTER_TAP_NUM 21
-//ne10_float32_t filterTaps[FILTER_TAP_NUM] = {
-//	0.000350,
-//	0.001133,
-//	0.002407,
-//	0.004203,
-//	0.006468,
-//	0.009057,
-//	0.011748,
-//	0.014265,
-//	0.016323,
-//	0.017671,
-//	0.018141,
-//	0.017671,
-//	0.016323,
-//	0.014265,
-//	0.011748,
-//	0.009057,
-//	0.006468,
-//	0.004203,
-//	0.002407,
-//	0.001133,
-//	0.000350
-//};
-#define FILTER_TAP_NUM 31
-ne10_float32_t filterTaps[FILTER_TAP_NUM] = {
-		0.000018,
-		0.000043,
-		0.000078,
-		0.000125,
-		0.000183,
-		0.000252,
-		0.000330,
-		0.000415,
-		0.000504,
-		0.000592,
-		0.000677,
-		0.000754,
-		0.000818,
-		0.000866,
-		0.000897,
-		0.000907,
-		0.000897,
-		0.000866,
-		0.000818,
-		0.000754,
-		0.000677,
-		0.000592,
-		0.000504,
-		0.000415,
-		0.000330,
-		0.000252,
-		0.000183,
-		0.000125,
-		0.000078,
-		0.000043,
-		0.000018
-};
-
-#endif /* FIRFILTER_H_ */
--- a/examples/d-box/FeedbackOscillator.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,105 +0,0 @@
-/*
- * FeedbackOscillator.cpp
- *
- * Recursive phase-shift oscillator implemented
- * on the matrix
- *
- * Andrew McPherson 2014
- */
-
-#include "FeedbackOscillator.h"
-#include <cstdlib>
-#include <cmath>
-
-#define COEFF_B0 	0
-#define COEFF_B1	1
-#define COEFF_A1	2
-
-FeedbackOscillator::FeedbackOscillator()
-: wavetable1(0), wavetable2(0)
-{
-
-}
-
-FeedbackOscillator::~FeedbackOscillator() {
-	if(wavetable1 != 0)
-		free(wavetable1);
-	if(wavetable2 != 0)
-		free(wavetable2);
-
-}
-
-// Initialise the settings for the feedback oscillator
-void FeedbackOscillator::initialise(int maxTableSize, float hpfCutoffFrequency, float matrixSampleRate) {
-	wavetableMaxLength = maxTableSize;
-	if(wavetable1 != 0)
-		free(wavetable1);
-	if(wavetable2 != 0)
-		free(wavetable2);
-
-	wavetable1 = (float *)malloc(maxTableSize * sizeof(float));
-	wavetable2 = (float *)malloc(maxTableSize * sizeof(float));
-
-	float omega = tan(M_PI * hpfCutoffFrequency / matrixSampleRate);
-	float n = 1.0f / (1.0f + omega);
-
-	coeffs[COEFF_A1] = (omega - 1.0f) * n;
-	coeffs[COEFF_B0] = n;
-	coeffs[COEFF_B1] = -n;
-
-	for(int n = 0; n < maxTableSize; n++)
-		wavetable1[n] = wavetable2[n] = 0;
-
-	wavetableRead = wavetable1;
-	wavetableWrite = wavetable2;
-	wavetableWritePointer = 0;
-	sampleCount = lastTriggerCount = 0;
-}
-
-// Process one sample and store the output value
-// Returns true if the wavetable needs rendering
-int FeedbackOscillator::process(float input, float *output) {
-	float outFloat = coeffs[COEFF_B0] * input + coeffs[COEFF_B1] * lastInput - coeffs[COEFF_A1] * lastOutput;
-	int requestRenderLength = 0;
-
-	if(outFloat < -0.5)
-		*output = 0;
-	else if(outFloat > 0.5)
-		*output = 1;
-	else
-		*output = outFloat + 0.5;
-
-	if(canTrigger && outFloat > 0 && lastOutput <= 0) {
-		triggered = true;
-		requestRenderLength = wavetableWritePointer;	// How many samples stored thus far?
-		if(requestRenderLength < 4)
-			requestRenderLength = 0;	// Ignore anything with fewer than 4 points
-
-		lastTriggerCount = sampleCount;
-		canTrigger = false;
-		wavetableWritePointer = 0;
-
-		// Swap buffers
-		float *temp = wavetableWrite;
-		wavetableWrite = wavetableRead;
-		wavetableRead = temp;
-	}
-
-	if(triggered) {
-		wavetableWrite[wavetableWritePointer] = outFloat;
-		if(++wavetableWritePointer >= wavetableMaxLength) {
-			triggered = false;
-			wavetableWritePointer = 0;
-		}
-	}
-
-	if(sampleCount - lastTriggerCount > 40)
-		canTrigger = true;
-
-	sampleCount++;
-
-	lastOutput = outFloat;
-	lastInput = input;
-
-	return requestRenderLength;
-}
--- a/examples/d-box/FeedbackOscillator.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,41 +0,0 @@
-/*
- * FeedbackOscillator.h
- *
- *  Created on: June 8, 2014
- *      Author: Andrew McPherson
- */
-
-#ifndef FEEDBACKOSCILLATOR_H
-#define FEEDBACKOSCILLATOR_H
-
-class FeedbackOscillator
-{
-public:
-	FeedbackOscillator();
-	~FeedbackOscillator();
-
-	// Initialise the settings for the feedback oscillator
-	void initialise(int maxTableSize, float hpfCutoffFrequency, float matrixSampleRate);
-
-	// Process one sample and store the output value
-	// Returns the length of table to interpolate; or 0 if nothing to process further
-	int process(float input, float *output);
-
-	float *wavetable() { return wavetableRead; }
-
-private:
-	float coeffs[3];			// Coefficients of first-order high-pass filter
-	float lastInput;			// last input sample for HPF
-	float lastOutput;			// last output sample of HPF
-	bool triggered;				// whether we are currently saving samples
-	bool canTrigger;			// whether we are able to begin saving samples
-	int wavetableMaxLength;		// how long the stored wavetable can be
-	int sampleCount;			// how many samples have elapsed
-	int lastTriggerCount;		// sample count when we last triggered
-
-	float *wavetable1, *wavetable2;			// Two wavetables where we record samples
-	float *wavetableRead, *wavetableWrite;	// Pointers to the above wavetables
-	int wavetableWritePointer;				// Where we are currently writing
-};
-
-#endif
--- a/examples/d-box/OscillatorBank.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,1016 +0,0 @@
-/*
- * OscillatorBank.cpp
- *
- *  Created on: May 23, 2014
- *      Author: Victor Zappi and Andrew McPherson
- */
-
-
-/*
- * There is a problem with name consistency between this class and the Parser class in spear_parser files.
- * There, a "frame" is each of the time values where partials are sampled, while a "hop" is the actual jump between frames [calculated in samples]
- * Here, "hop" is used with the meaning of "frame", while "frame" became the local frame of a partial
- *
- * example
- *
- * frames:	0		1		2
- * p0				p0_0 	p0_1
- * p1		p1_0	p1_1 	p1_2
- * p2		p2_0	p2_1
- *
- * 	In this case:
- * 		 in Parser there are 2 hops, 3 total frames and the 3 partials have respectively 2, 3 and 2 local frames
- *
- * 		 here there are 3 total hops [the concept of jumps is discarded, cos not in use] and the 3 partials have respectively 2, 3 and 2 frames
- *
- * 	This must be fixed
-*/
-
-// TODO: fix hop-frame name consistency
-
-
-#include <stdlib.h>
-
-#include "OscillatorBank.h"
-
-OscillatorBank::OscillatorBank() {
-	loaded = false;
-}
-
-OscillatorBank::OscillatorBank(string filename, int hopsize, int samplerate) {
-	loaded = false;
-	loadFile(filename.c_str(), hopsize, samplerate);
-}
-
-OscillatorBank::OscillatorBank(char *filename, int hopsize, int samplerate) {
-	loaded = false;
-	loadFile(filename, hopsize, samplerate);
-}
-
-OscillatorBank::~OscillatorBank() {
-	free(oscillatorPhases);
-	free(oscillatorNextNormFreq);
-	free(oscillatorNextAmp);
-	free(oscillatorNormFrequencies);
-	free(oscillatorAmplitudes);
-	free(oscillatorNormFreqDerivatives);
-	free(oscillatorAmplitudeDerivatives);
-	free(phaseCopies);
-	free(nextNormFreqCopies);
-	free(nextAmpCopies);
-
-	delete[] oscStatNormFrequenciesMean;
-	delete[] oscStatNumHops;
-	delete[] lookupTable;
-	delete[] indicesMapping;
-	delete[] freqFixedDeltas;
-	delete[] ampFixedDeltas;
-	delete[] nyquistCut;
-}
-
-bool OscillatorBank::initBank(int oversamp) {
-	if (!loaded)
-		return false;
-
-	//---prepare look-up table
-	lookupTableSize = 1024;
-	lookupTable		= new float[lookupTableSize + 1];
-	for (int n = 0; n < (lookupTableSize + 1); n++)
-		lookupTable[n] = sin(2.0 * M_PI * (float) n / (float) lookupTableSize);
-	frequencyScaler = (float) lookupTableSize / rate;
-	nyqNorm			= rate / 2 * frequencyScaler;
-
-	if (oversamp < 1)
-		oversamp = 1;
-
-	//---prepare oscillators
-	partials = &(parser.partials); // pointer to paser's partials
-	partialsHopSize = parser.getHopSize();
-	lastHop = partials->getHopNum(); // last bank hop is equal to last partial frame, which is equal to partial hop num
-	overSampling = oversamp;
-	hopSize = partialsHopSize / overSampling; // if oversampling, osc bank hop num > partials hop num
-	hopSizeReminder = partialsHopSize % overSampling;
-	oscBankHopSize = hopSize;
-	numOfPartials = partials->getPartialNum();
-	numOfOscillators = partials->getMaxActivePartialNum(); // get the maximum number of active partials at the same time
-
-	// set to next multiple of 4 [NEON]
-	numOfOscillators = (numOfOscillators + 3) & ~0x3; // to be sure we can add up to 3 fake oscillators
-
-	int err;
-	//---allocate buffers
-	// alligned buffers [NEON]
-	err = posix_memalign((void**) &oscillatorPhases, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &oscillatorNextNormFreq, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &oscillatorNextAmp, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &oscillatorNormFrequencies, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &oscillatorAmplitudes, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &oscillatorNormFreqDerivatives, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &oscillatorAmplitudeDerivatives, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &phaseCopies, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &nextNormFreqCopies, 16,
-			numOfOscillators * sizeof(float));
-	err += posix_memalign((void**) &nextAmpCopies, 16,
-			numOfOscillators * sizeof(float));
-
-	// regular ones
-	oscStatNormFrequenciesMean	= new float[numOfPartials];
-	oscStatNumHops				= new float[numOfPartials];
-	indicesMapping				= new int[numOfPartials];
-	freqFixedDeltas				= new float[numOfPartials];
-	ampFixedDeltas				= new float[numOfPartials];
-	nyquistCut 					= new bool[numOfPartials];
-
-	if (err > 0) {
-		dbox_printf("Failed memory allocations %@#!\n");
-		return false;
-	}
-
-	// copy stats [they do not change]
-	for (int n = 0; n < numOfPartials; n++) {
-		oscStatNormFrequenciesMean[n] = partials->partialFreqMean[n]
-				* frequencyScaler;
-		oscStatNumHops[n] = partials->partialNumFrames[n]; // in Parser and Partials "frames" are what we call here "hops" [see comment at top of file]
-	}
-
-	// deafult values
-	actPartNum		= 0;
-	loopStartHop	= 0;
-	loopEndHop		= (parser.partials.getHopNum() - 2) * overSampling;
-	ampTh			= 0.0001;
-	hopNumTh		= 0;
-	pitchMultiplier = 1;
-	freqMovement	= 1;
-	filterNum		= 0;
-	note			= false;
-	speed			= 1;
-	nextSpeed		= -1;
-	maxSpeed 		= 10;
-	minSpeed 		= 0.1;
-	jumpHop 		= -1;
-
-	// filter
-	filterMaxF		= 22000;
-	filterAmpMinF	= 10 * frequencyScaler;
-	filterAmpMaxF	= 5000 * frequencyScaler;
-	filterAmpMul	= 10.0;
-
-	// adsr
-	minAttackTime	= .0001;
-	deltaAttackTime = 2.;
-	minReleaseTime	= 1;
-	deltaReleaseTime = 2.5;
-
-	adsr.setAttackRate(minAttackTime * rate);
-	adsr.setDecayRate(.0001 * rate);
-	adsr.setSustainLevel(1);
-	adsr.setReleaseRate(minReleaseTime * rate);
-
-	state = bank_stopped;
-	return true;
-}
-
-void OscillatorBank::resetOscillators() {
-	currentHop			= -1;
-	loopDir				= 1;
-	loopDirShift 		= 0;
-	fill(nyquistCut, nyquistCut + numOfPartials, false);
-	prevAdsrVal			= 0;
-	prevAmpTh			= ampTh;
-	prevHopNumTh		= hopNumTh;
-	prevPitchMultiplier = pitchMultiplier;
-	prevFreqMovement 	= freqMovement;
-	prevFilterNum = filterNum;
-	memcpy(prevFilterFreqs, filterFreqs, filterNum * sizeof(float));
-	memcpy(prevFilterQ, filterQ, filterNum * sizeof(float));
-
-	int activePNum		= partials->activePartialNum[0];
-	unsigned int *activeP = partials->activePartials[0];
-	for (int i = 0; i < activePNum; i++) {
-		freqFixedDeltas[activeP[i]] = partials->partialFreqDelta[activeP[i]][0]
-				/ overSampling;
-		ampFixedDeltas[activeP[i]]	= partials->partialAmpDelta[activeP[i]][0]
-				/ overSampling;
-	}
-	// attack!
-	adsr.gate(1);
-	note = true;
-
-	nextHop();
-
-	state = bank_playing;
-}
-
-void OscillatorBank::nextHop() {
-	hopSize = oscBankHopSize;
-
-	// copy phases, next freqs and next amps from previous frame
-	memcpy(phaseCopies, oscillatorPhases, actPartNum * sizeof(float));
-	memcpy(nextNormFreqCopies, oscillatorNextNormFreq,
-			actPartNum * sizeof(float));
-	memcpy(nextAmpCopies, oscillatorNextAmp, actPartNum * sizeof(float));
-
-	// next frame is forward or backwards, cos we could be in the loop
-	currentHop += loopDir;
-
-	checkDirection();
-
-//	if((currentHop/overSampling)%100 == 0)
-//		dbox_printf("currentHop %d, direction: %d\n", currentHop/overSampling, loopDir);
-
-	// if needs jump, end here this method, cos jumpToHop() will do tee rest
-	if (checkJump() == 0)
-		return;
-	// otherwise, if jump is not needed or fails, continue regular stuff
-
-	if (nextEnvState() != 0)
-		return; // release has ended!
-
-	checkSpeed();
-
-	// now let's decide how to calculate next hop
-	if (!checkOversampling())
-		nextOscBankHop();
-	else
-		nextPartialHop();
-}
-
-void OscillatorBank::nextOscBankHop() {
-	int parIndex, localHop;
-	float parDamp = 1;
-	int currentPartialHop = (currentHop / overSampling) + loopDirShift;
-
-	// if going backwards in the loop, get previous frame active partials...
-	actPartNum = partials->activePartialNum[currentPartialHop - loopDirShift];
-	actPart = partials->activePartials[currentPartialHop - loopDirShift];
-	//cout << "actPartNum: " << actPartNum << endl;
-
-	envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release]
-
-	int parCnt = 0;
-	int currentHopReminder = currentHop % overSampling;
-	// steps to reach next bank hop from previous partial hop
-	int steps = currentHopReminder + 1;
-	if (loopDir < 0)
-		steps = overSampling - currentHopReminder + 1;
-
-	for (int i = 0; i < actPartNum; i++) {
-		// find partial and frame
-		parIndex = actPart[i];
-		//localHop	= partials->localPartialFrames[currentPartialHop][parIndex];
-		localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
-
-		//float delta = partials->partialFrequencies[parIndex][localHop+loopDir] - partials->partialFrequencies[parIndex][localHop];
-
-		// if this partial was over nyquist on previous hop...
-		if (nyquistCut[parIndex]) {
-			// ...restart from safe values
-			oscillatorPhases[parCnt] = 0;
-			//TODO add freqmove dependency
-			oscillatorNextNormFreq[parCnt] =
-					(partials->partialFrequencies[parIndex][localHop]
-							+ freqFixedDeltas[parIndex] * (steps - 1))
-							* frequencyScaler * prevPitchMultiplier;
-			oscillatorNextAmp[parCnt] = 0;
-		} else if (loopDir == 1) // otherwise recover phase, target freq and target amp from previous frame
-				{
-			if ((localHop != 0) || (currentHopReminder != 0)) {
-				oscillatorPhases[parCnt] =
-						phaseCopies[indicesMapping[parIndex]];
-				oscillatorNextNormFreq[parCnt] =
-						nextNormFreqCopies[indicesMapping[parIndex]];
-				oscillatorNextAmp[parCnt] =
-						nextAmpCopies[indicesMapping[parIndex]];
-			} else // first oscillator hop [both for bank and partial], so no previous data are available
-			{
-				oscillatorPhases[parCnt] = 0;
-				//TODO add freqmove dependency
-				oscillatorNextNormFreq[parCnt] =
-						partials->partialFrequencies[parIndex][localHop]
-								* frequencyScaler * prevPitchMultiplier;
-				parDamp = calculateParDamping(parIndex, prevHopNumTh,
-						prevAdsrVal, oscillatorNextNormFreq[parCnt],
-						prevFilterNum, prevFilterFreqs, prevFilterQ);
-				oscillatorNextAmp[parCnt] =
-						partials->partialAmplitudes[parIndex][localHop]
-								* parDamp;
-				if(oscillatorNextAmp[parCnt] > 1)
-						oscillatorNextAmp[parCnt] = 1;
-				freqFixedDeltas[parIndex] =
-						partials->partialFreqDelta[parIndex][localHop + loopDir]
-								* loopDir / overSampling;
-				ampFixedDeltas[parIndex] =
-						partials->partialAmpDelta[parIndex][localHop + loopDir]
-								* loopDir / overSampling;
-			}
-		} else {
-			oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]];
-			oscillatorNextNormFreq[parCnt] =
-					nextNormFreqCopies[indicesMapping[parIndex]];
-			oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]];
-		}
-
-		// remove aliasing, skipping partial over nyquist freq
-		if (oscillatorNextNormFreq[parCnt] > nyqNorm) {
-			nyquistCut[parIndex] = true;
-			continue;
-		}
-		nyquistCut[parIndex] = false;
-
-		// first set up freq, cos filter affects amplitude damping according to freq content
-		oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts
-		// save next values, current for next round
-		oscillatorNextNormFreq[parCnt] = (freqMovement
-				* (partials->partialFrequencies[parIndex][localHop]
-						+ freqFixedDeltas[parIndex] * steps) * frequencyScaler
-				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
-				* pitchMultiplier;
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
-				- oscillatorNormFrequencies[parCnt]) / hopCounter;
-		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
-		oscillatorNormFreqDerivatives[parCnt] = freqMovement
-				* oscillatorNormFreqDerivatives[parCnt]
-				+ (1 - freqMovement)
-						* ((oscStatNormFrequenciesMean[parIndex]
-								* pitchMultiplier)
-								- oscillatorNormFrequencies[parCnt])
-						/ hopCounter;
-
-		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
-				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
-
-		// now amplitudes
-		oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts
-		// save next values, current for next round
-		//delta = partials->partialAmplitudes[parIndex][localHop+loopDir] - partials->partialAmplitudes[parIndex][localHop];
-		oscillatorNextAmp[parCnt] =
-				(partials->partialAmplitudes[parIndex][localHop]
-						+ ampFixedDeltas[parIndex] * steps) * parDamp;
-		if(oscillatorNextAmp[parCnt] > 1)
-				oscillatorNextAmp[parCnt] = 1;
-		if ((loopDir == -1) && (localHop = 1) && (currentHopReminder == 1))
-			oscillatorNextAmp[parCnt] = 0;
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
-				- oscillatorAmplitudes[parCnt]) / hopCounter;
-
-		// finally update current mapping between oscillators and partials
-		indicesMapping[parIndex] = parCnt;
-		parCnt++;
-	}
-	actPartNum = parCnt;
-	// [NEON] if not multiple of 4...
-	if (actPartNum % 4 != 0)
-		addFakeOsc();
-}
-
-void OscillatorBank::nextPartialHop() {
-	unsigned int parIndex, localHop;
-	float parDamp = 1;
-	int currentPartialHop = currentHop / overSampling;
-
-	// if going backwards in the loop, get previous frame active partials...
-	actPartNum = partials->activePartialNum[currentPartialHop - loopDirShift];
-	actPart    = partials->activePartials[currentPartialHop - loopDirShift];
-
-	envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release]
-
-	int parCnt = 0;
-	int steps = overSampling - 1; // steps to reach next hop [partial or bank] from previous partial hop
-
-	for (int i = 0; i < actPartNum; i++) {
-		// find partial and frame
-		parIndex = actPart[i];
-		//localHop	= partials->localPartialFrames[currentPartialHop][parIndex];
-		localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
-
-		// if this partial was over nyquist on previous hop...
-		if (nyquistCut[parIndex]) {
-			// ...restart from safe values
-			oscillatorPhases[parCnt] = 0;
-			//TODO add freqmove dependency
-			oscillatorNextNormFreq[parCnt] =
-					(partials->partialFrequencies[parIndex][localHop]
-							+ freqFixedDeltas[parIndex] * steps
-									* (1 - loopDirShift)) * frequencyScaler
-							* prevPitchMultiplier;
-			oscillatorNextAmp[parCnt] = 0;
-		} else if (loopDir == 1) // otherwise recover phase, target freq and target amp from previous frame
-		{
-			if ((localHop != 0) || (overSampling > 1)) {
-				oscillatorPhases[parCnt] =
-						phaseCopies[indicesMapping[parIndex]];
-				oscillatorNextNormFreq[parCnt] =
-						nextNormFreqCopies[indicesMapping[parIndex]];
-				oscillatorNextAmp[parCnt] =
-						nextAmpCopies[indicesMapping[parIndex]];
-			} else // first oscillator hop [both for bank and partial], so no previous data are available
-			{
-				oscillatorPhases[parCnt] = 0;
-				//TODO add freqmove dependency
-				oscillatorNextNormFreq[parCnt] =
-						partials->partialFrequencies[parIndex][localHop]
-								* frequencyScaler * prevPitchMultiplier;
-				parDamp = calculateParDamping(parIndex, prevHopNumTh,
-						prevAdsrVal, oscillatorNextNormFreq[parCnt],
-						prevFilterNum, prevFilterFreqs, prevFilterQ);
-				oscillatorNextAmp[parCnt] =
-						partials->partialAmplitudes[parIndex][localHop]
-								* parDamp;
-				if(oscillatorNextAmp[parCnt] > 1)
-						oscillatorNextAmp[parCnt] = 1;
-				freqFixedDeltas[parIndex] =
-						partials->partialFreqDelta[parIndex][localHop + loopDir]
-								* loopDir / overSampling;
-				ampFixedDeltas[parIndex] =
-						partials->partialAmpDelta[parIndex][localHop + loopDir]
-								* loopDir / overSampling;
-			}
-		} else {
-			if (localHop != partials->partialNumFrames[parIndex] - 1) {
-				oscillatorPhases[parCnt] =
-						phaseCopies[indicesMapping[parIndex]];
-				oscillatorNextNormFreq[parCnt] =
-						nextNormFreqCopies[indicesMapping[parIndex]];
-				oscillatorNextAmp[parCnt] =
-						nextAmpCopies[indicesMapping[parIndex]];
-			} else // first oscillator hop [going backwards - both for bank and partial] , so no previous data are available
-			{
-				oscillatorPhases[parCnt] = 0;
-				//TODO add freqmove dependency
-				oscillatorNextNormFreq[parCnt] =
-						partials->partialFrequencies[parIndex][localHop]
-								* frequencyScaler * prevPitchMultiplier;
-				parDamp = calculateParDamping(parIndex, prevHopNumTh,
-						prevAdsrVal, oscillatorNextNormFreq[parCnt],
-						prevFilterNum, prevFilterFreqs, prevFilterQ);
-				oscillatorNextAmp[parCnt] =
-						partials->partialAmplitudes[parIndex][localHop]
-								* parDamp;
-				if(oscillatorNextAmp[parCnt] > 1)
-						oscillatorNextAmp[parCnt] = 1;
-				freqFixedDeltas[parIndex] =
-						partials->partialFreqDelta[parIndex][localHop + loopDir]
-								* loopDir / overSampling;
-				ampFixedDeltas[parIndex] =
-						partials->partialAmpDelta[parIndex][localHop + loopDir]
-								* loopDir / overSampling;
-			}
-		}
-		// remove aliasing, skipping partial over nyquist freq
-		if (oscillatorNextNormFreq[parCnt] > nyqNorm) {
-			//cout << nyqNorm << endl;
-			nyquistCut[parIndex] = true;
-			continue;
-		}
-		nyquistCut[parIndex] = false;
-
-		// first set up freq, cos filter affects amplitude damping according to freq content
-		oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts
-		// save next values, current for next round
-		oscillatorNextNormFreq[parCnt] = (freqMovement
-				* (partials->partialFrequencies[parIndex][localHop + loopDir]
-						- freqFixedDeltas[parIndex] * steps * loopDirShift)
-				* frequencyScaler
-				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
-				* pitchMultiplier;
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
-				- oscillatorNormFrequencies[parCnt]) / hopCounter;
-		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
-		oscillatorNormFreqDerivatives[parCnt] = freqMovement
-				* oscillatorNormFreqDerivatives[parCnt]
-				+ (1 - freqMovement)
-						* ((oscStatNormFrequenciesMean[parIndex]
-								* pitchMultiplier)
-								- oscillatorNormFrequencies[parCnt])
-						/ hopCounter;
-
-		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
-				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
-
-		// now amplitudes
-		oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts
-		// save next values, current for next round
-		//delta = partials->partialAmplitudes[parIndex][localHop+loopDir] - partials->partialAmplitudes[parIndex][localHop];
-		oscillatorNextAmp[parCnt] =
-				(partials->partialAmplitudes[parIndex][localHop + loopDir]
-						- (ampFixedDeltas[parIndex]) * steps * loopDirShift)
-						* parDamp;
-		if(oscillatorNextAmp[parCnt] > 1)
-			oscillatorNextAmp[parCnt] = 1;
-
-		// to avoid bursts when transients are played backwards
-		if ((loopDir == -1) && (localHop - 1 == 0) && (overSampling == 1)) {
-			oscillatorNextAmp[parCnt] = 0;
-		}
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
-				- oscillatorAmplitudes[parCnt]) / hopCounter;
-
-		// if next is not going to loop boundaries, get next deltas [same direction]
-		if ((((currentPartialHop + loopDir) * overSampling != loopEndHop)
-				|| (loopDir == -1))
-				&& (((currentPartialHop + loopDir) * overSampling + loopDir
-						!= loopStartHop) || (loopDir == 1))) {
-			freqFixedDeltas[parIndex] =
-					partials->partialFreqDelta[parIndex][localHop + loopDir]
-							* loopDir / overSampling;
-			ampFixedDeltas[parIndex] =
-					partials->partialAmpDelta[parIndex][localHop + loopDir]
-							* loopDir / overSampling;
-		} else // .. otherwise, keep deltas but change sign [co swe change direction]
-		{
-			freqFixedDeltas[parIndex] = -freqFixedDeltas[parIndex];
-			ampFixedDeltas[parIndex] = -ampFixedDeltas[parIndex];
-		}
-
-		// finally update current mapping between oscillators and partials
-		indicesMapping[parIndex] = parCnt;
-		parCnt++;
-	}
-	actPartNum = parCnt;
-	// [NEON] if not multiple of 4...
-	if (actPartNum % 4 != 0)
-		addFakeOsc();
-
-	updatePrevControls();
-}
-
-void OscillatorBank::addFakeOsc() {
-	// ...calculate difference
-	int newPartNum = (actPartNum + 3) & ~0x3;
-	// ...add fake oscillators until total num is multiple of 4
-	for (int i = actPartNum; i < newPartNum; i++) {
-		oscillatorAmplitudes[i] = 0;
-		oscillatorNormFrequencies[i] = 0;
-		oscillatorAmplitudeDerivatives[i] = 0;
-		oscillatorNormFreqDerivatives[i] = 0;
-		oscillatorPhases[i] = 0;
-	}
-	// ...and update num of active partials
-	actPartNum = newPartNum;
-}
-
-void OscillatorBank::play(float vel) {
-	// set attack and release params according to velocity
-	//adsr.setAttackRate((minAttackTime + ((1 - vel) * deltaAttackTime)) * rate);
-	adsr.setAttackRate(minAttackTime * rate);
-	//adsr.setReleaseRate((minReleaseTime + (1 - vel) * deltaReleaseTime) * rate);
-	adsr.setReleaseRate(minReleaseTime * rate);
-
-	// set timbre
-	hopNumTh = log((1 - vel) + 1) / log(2) * 20000;
-
-	state = bank_toreset;
-}
-
-//---------------------------------------------------------------------------------------------------------------------------
-// private methods
-//---------------------------------------------------------------------------------------------------------------------------
-
-bool OscillatorBank::loader(char *filename, int hopsize, int samplerate) {
-	rate = samplerate;
-	loaded = parser.parseFile(filename, hopsize, samplerate);
-	return loaded;
-}
-
-int OscillatorBank::jumpToHop() {
-	int jumpGap = abs(jumpHop - currentHop / overSampling); // gaps in partial reference
-
-	// can't jump to self dude
-	if (jumpGap == 0)
-		return 1;
-
-	// direction is in general maintained with jump
-	if (jumpHop == 0)
-		setDirection(1);
-	else if (jumpHop == lastHop)
-		setDirection(-1);
-
-	dbox_printf("\tJump from %d to %d\n", currentHop / overSampling, jumpHop);
-	dbox_printf("\tdirection %d\n", loopDir);
-
-	currentHop = jumpHop * overSampling;
-
-	if (nextEnvState() != 0)
-		return 0; // release has ended!
-
-	checkSpeed();
-
-	int parIndex, localHop, targetHop;
-	float parDamp = 1;
-	int currentPartialHop = currentHop / overSampling;
-	int targetPartialHop = jumpHop;
-
-	actPartNum = partials->activePartialNum[currentPartialHop];
-	actPart = partials->activePartials[currentPartialHop];
-	int targetActParNum = partials->activePartialNum[targetPartialHop];
-	unsigned int *targetActPar = partials->activePartials[targetPartialHop];
-
-	envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release]
-
-	int parCnt = 0;
-	int currentHopReminder = currentHop % overSampling;
-
-	// steps to walk where i am [bank of partial hop] from previous partial hop
-	int steps = currentHopReminder * (overSampling != 1); // no oversampling 0, oversampling and going ff currentHopReminder
-
-	for (int i = 0; i < actPartNum; i++) {
-		// find partial and frame
-		parIndex = actPart[i];
-		//localHop	= partials->localPartialFrames[currentPartialHop][parIndex];
-		localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
-
-		// if this partial was over nyquist on previous hop...
-		if (nyquistCut[parIndex]) {
-			// ...restart from safe values
-			oscillatorPhases[parCnt] = 0;
-			//TODO add freqmove dependency
-			oscillatorNextNormFreq[parCnt] =
-					(partials->partialFrequencies[parIndex][localHop]
-							+ freqFixedDeltas[parIndex] * steps * loopDir)
-							* frequencyScaler * prevPitchMultiplier;
-			oscillatorNextAmp[parCnt] = 0;
-		} else if (loopDir == 1) {// otherwise recover phase, target freq and target amp from previous frame
-			if ((localHop != 0)
-					|| ((overSampling > 1) && (currentHopReminder != 0))) {
-				oscillatorPhases[parCnt] =
-						phaseCopies[indicesMapping[parIndex]];
-				oscillatorNextNormFreq[parCnt] =
-						nextNormFreqCopies[indicesMapping[parIndex]];
-				oscillatorNextAmp[parCnt] =
-						nextAmpCopies[indicesMapping[parIndex]];
-			} else { // first oscillator hop [both for bank and partial], so no previous data are available
-				oscillatorPhases[parCnt] = 0;
-				//TODO add freqmove dependency
-				oscillatorNextNormFreq[parCnt] =
-						partials->partialFrequencies[parIndex][localHop]
-								* frequencyScaler * prevPitchMultiplier;
-				parDamp = calculateParDamping(parIndex, prevHopNumTh,
-						prevAdsrVal, oscillatorNextNormFreq[parCnt],
-						prevFilterNum, prevFilterFreqs, prevFilterQ);
-				oscillatorNextAmp[parCnt] =
-						partials->partialAmplitudes[parIndex][localHop]
-								* parDamp;
-				if(oscillatorNextAmp[parCnt] > 1)
-						oscillatorNextAmp[parCnt] = 1;
-			}
-		} else {
-			if (( (unsigned)localHop != partials->partialNumFrames[parIndex] - 1)
-					|| ((overSampling > 1) && (currentHopReminder != 0))) {
-				oscillatorPhases[parCnt] =
-						phaseCopies[indicesMapping[parIndex]];
-				oscillatorNextNormFreq[parCnt] =
-						nextNormFreqCopies[indicesMapping[parIndex]];
-				oscillatorNextAmp[parCnt] =
-						nextAmpCopies[indicesMapping[parIndex]];
-			} else // first oscillator hop [going backwards - both for bank and partial] , so no previous data are available, so retrieve where i am
-			{
-				oscillatorPhases[parCnt] = 0;
-				//TODO add freqmove dependency
-				oscillatorNextNormFreq[parCnt] =
-						partials->partialFrequencies[parIndex][localHop]
-								* frequencyScaler * prevPitchMultiplier;
-				parDamp = calculateParDamping(parIndex, prevHopNumTh,
-						prevAdsrVal, oscillatorNextNormFreq[parCnt],
-						prevFilterNum, prevFilterFreqs, prevFilterQ);
-				oscillatorNextAmp[parCnt] =
-						partials->partialAmplitudes[parIndex][localHop]
-								* parDamp;
-				if(oscillatorNextAmp[parCnt] > 1)
-						oscillatorNextAmp[parCnt] = 1;
-			}
-		}
-		// remove aliasing, skipping partial over nyquist freq
-		if (oscillatorNextNormFreq[parCnt] > nyqNorm) {
-			//cout << nyqNorm << endl;
-			nyquistCut[parIndex] = true;
-			continue;
-		}
-		nyquistCut[parIndex] = false;
-
-		// check what happens of this partial at target hop
-		float targetFreqVal, targetAmpVal;
-		//targetHop = partials->localPartialFrames[targetPartialHop][parIndex];
-		targetHop = targetPartialHop - partials->partialStartFrame[parIndex];
-
-		if (targetHop == -1)
-			targetFreqVal = targetAmpVal = 0;
-		else {
-			targetFreqVal = partials->partialFrequencies[parIndex][targetHop]
-					* frequencyScaler; // pitch shift will be multiplied later!!!
-			targetAmpVal = partials->partialFrequencies[parIndex][targetHop]; // parDamp will be multiplied later!!!
-		}
-
-		// first set up freq, cos filter affects amplitude damping according to freq content
-		oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts
-		// save next values, current for next round
-		oscillatorNextNormFreq[parCnt] = (freqMovement * targetFreqVal
-				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
-				* pitchMultiplier;
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
-				- oscillatorNormFrequencies[parCnt]) / hopCounter;
-		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
-		oscillatorNormFreqDerivatives[parCnt] = freqMovement
-				* oscillatorNormFreqDerivatives[parCnt]
-				+ (1 - freqMovement)
-						* ((oscStatNormFrequenciesMean[parIndex]
-								* pitchMultiplier)
-								- oscillatorNormFrequencies[parCnt])
-						/ hopCounter;
-
-		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
-				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
-
-		// now amplitudes
-		oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts
-		// save next values, current for next round
-		oscillatorNextAmp[parCnt] = targetAmpVal * parDamp;
-		if(oscillatorNextAmp[parCnt] > 1)
-				oscillatorNextAmp[parCnt] = 1;
-		// to avoid bursts when transients are played backwards
-		if ((loopDir == -1) && (targetHop == 0)
-				&& ((overSampling == 1) || (currentHopReminder == 0))) {
-			oscillatorNextAmp[parCnt] = 0;
-		}
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
-				- oscillatorAmplitudes[parCnt]) / hopCounter;
-
-		//if partial does not die at target, calculate deltas according to direction
-		if (targetHop != -1) {
-			freqFixedDeltas[parIndex] =
-					partials->partialFreqDelta[parIndex][targetHop] * loopDir
-							/ overSampling;
-			ampFixedDeltas[parIndex] =
-					partials->partialAmpDelta[parIndex][targetHop] * loopDir
-							/ overSampling;
-		}
-
-		// finally update current mapping between oscillators and partials
-		indicesMapping[parIndex] = parCnt;
-		parCnt++;
-	}
-	actPartNum = parCnt;
-
-	// now add the ones that start at target hop!
-	for (int i = 0; i < targetActParNum; i++) {
-		// find partial and frame
-		parIndex = targetActPar[i];
-		//targetHop	= partials->localPartialFrames[targetPartialHop][parIndex];
-		targetHop = targetPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file]
-
-		// check if this partials was already active before the jump
-		//localHop = partials->localPartialFrames[currentPartialHop][parIndex];
-		localHop = currentPartialHop - partials->partialStartFrame[parIndex];
-
-		// if yes, skip it
-		if (localHop != -1)
-			continue;
-
-		// otherwise add it to active bunch and calcucalte values
-
-		// first set up freq, cos filter affects amplitude damping according to freq content
-		oscillatorNormFrequencies[parCnt] = 0;
-		// save next values, current for next round
-		oscillatorNextNormFreq[parCnt] = (freqMovement
-				* partials->partialFrequencies[parIndex][targetHop]
-				* frequencyScaler
-				+ (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex])
-				* pitchMultiplier;
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt]
-				- oscillatorNormFrequencies[parCnt]) / hopCounter;
-		// this second weird passage handles dissonance control, morphing between regular and mean frequencies
-		oscillatorNormFreqDerivatives[parCnt] = freqMovement
-				* oscillatorNormFreqDerivatives[parCnt]
-				+ (1 - freqMovement)
-						* ((oscStatNormFrequenciesMean[parIndex]
-								* pitchMultiplier)
-								- oscillatorNormFrequencies[parCnt])
-						/ hopCounter;
-
-		parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal,
-				oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ);
-
-		// now amplitudes
-		oscillatorAmplitudes[parCnt] = 0;
-		// save next values, current for next round
-		oscillatorNextAmp[parCnt] =
-				partials->partialFrequencies[parIndex][targetHop] * parDamp;
-		if(oscillatorNextAmp[parCnt] > 1)
-				oscillatorNextAmp[parCnt] = 1;
-		// derivatives are (next hop value*next damping) - (current hop value*current damping)  ---> next hop must be available, in both directions, because of control on active partials
-		oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt]
-				- oscillatorAmplitudes[parCnt]) / hopCounter;
-
-		//calculate deltas according to direction
-		freqFixedDeltas[parIndex] =
-				partials->partialFreqDelta[parIndex][targetHop] * loopDir
-						/ overSampling;
-		ampFixedDeltas[parIndex] =
-				partials->partialAmpDelta[parIndex][targetHop] * loopDir
-						/ overSampling;
-
-		// finally update current mapping between oscillators and partials
-		indicesMapping[parIndex] = parCnt;
-		parCnt++;
-
-	}
-	// [NEON] if not multiple of 4...
-	if (actPartNum % 4 != 0)
-		addFakeOsc();
-
-	updatePrevControls();
-
-	jumpHop = -1;
-
-	return 0;
-}
-
-int OscillatorBank::nextEnvState() {
-	/*
-	 envState = Attack.getState();	// to determine what state we are in [attack, decay, sustain, release]
-
-	 // osc bank is playing the tail and the tail ends...
-	 if( (state == bank_playing)&&(envState == env_idle) )
-	 {
-	 state = bank_stopped;	// ...stop bank
-	 return 1;					// and return immediately
-	 }
-	 else if( (envState == env_attack) || (envState == env_decay) )
-	 {
-	 // run envelopes until next frame
-	 dampWeight	= Attack.process(hopSize);
-	 }
-	 else if(envState == env_release)
-	 {
-	 // run envelopes until next frame
-	 dampWeight 	= Attack.process(hopSize);
-	 releaseDamp = Release.process(hopSize);
-	 }*/
-
-	envState = adsr.getState();
-	// osc bank is playing the tail and the tail ends...
-	if ((state == bank_playing) && (envState == env_idle)) {
-		state = bank_stopped; // ...stop bank
-		adsrVal = 0;
-		return 1; // and return immediately
-	} else
-		adsrVal = adsr.process(hopSize);
-
-	return 0;
-}
-
-void OscillatorBank::checkDirection() {
-	// end of the loop or end of file
-	if (((currentHop >= loopEndHop) && (loopDir == 1))
-			|| ((currentHop >= lastHop) && (loopDir == 1))) {
-		// move backwards
-		setDirection(-1);
-		//dbox_printf("backward from %d\n", loopEndHop);
-	} else if (((currentHop <= loopStartHop) && (loopDir == -1))
-			|| ((currentHop <= 0) && (loopDir == -1))) // start of the loop or start of file
-			{
-		// move forward
-		setDirection(1);
-		//dbox_printf("forward from %d\n", loopStartHop);
-	}
-}
-
-void OscillatorBank::checkSpeed() {
-	// speed control [alike on highways, LOL]
-	if (nextSpeed > 0) {
-		nextSpeed = (nextSpeed < maxSpeed) ? nextSpeed : maxSpeed;
-		nextSpeed = (nextSpeed > minSpeed) ? nextSpeed : minSpeed;
-		speed = nextSpeed;
-		nextSpeed = -1;
-	}
-	hopCounter = hopSize / speed;
-}
-
-int OscillatorBank::checkJump() {
-	//check if has to jump somewhere
-	if (jumpHop > -1) {
-		// needs to jump!
-		if (jumpToHop() == 0)
-			return 0;
-	}
-	return 1; // no jump
-}
-
-bool OscillatorBank::checkOversampling() {
-	//TODO fix this, but need andrew to fix oversampling multiple of period size
-	// if partialsHopSize is not a multiple of oversampling, change hop size to periodically match next partial hop
-	if (hopSizeReminder > 0) {
-		// if next osc bank hop overtakes next partial hop...
-		if ((currentHop + loopDir) * hopSize > partialsHopSize) {
-			hopSize = hopSizeReminder; // ...shrink osc bank hop size to match partial hop
-			return true; // and set next hop as matching with next partial hop
-		}
-	} else if (((currentHop + (1 - loopDirShift)) % overSampling) == 0) // if next osc bank hop matches next partial hop
-		return true; // ...mark next hop as partial hop
-
-	return false; // ,otherwise mark next hop as osc bank hop
-}
-
-void OscillatorBank::updatePrevControls() {
-	prevAdsrVal = adsrVal;
-	prevAmpTh = ampTh;
-	prevHopNumTh = hopNumTh;
-	prevPitchMultiplier = pitchMultiplier;
-	prevFreqMovement = freqMovement;
-	prevFilterNum = filterNum;
-	memcpy(prevFilterFreqs, filterFreqs, filterNum * sizeof(float));
-	memcpy(prevFilterQ, filterQ, filterNum * sizeof(float));
-}
-
-float OscillatorBank::calculateParDamping(int parIndex, int hopNTh,
-		float adsrVl, float nextFreq, int filNum, float *filFreq, float *filQ) {
-	float parDamp = 1;
-
-	// timbre
-	parDamp = ((float) (oscStatNumHops[parIndex] + 1)) / (hopNTh + 1);
-	parDamp = (parDamp > 1) ? 1 : parDamp;
-	parDamp = adsrVl * parDamp;
-
-	//filters
-
-	float filterWeights[MAX_TOUCHES];
-	float filterDamp[MAX_TOUCHES];
-	float filDist;
-	float filterWeightsAcc;
-	float filDmp;
-	float filAmp;
-
-// 		band reject notch filter
-//		float dist, dmp;
-//		for(int k=0; k<filterNum; k++)
-//		{
-//			dist = fabs(oscillatorNextNormFreq[parCnt]-filterFreqs[k]);
-//			if(dist<=filterQ[k])
-//			{
-//				dmp 	= dist/filterQ[k];
-//				parDamp *= dmp*dmp*dmp;
-//			}
-//		}
-
-
-	// each filter is a band pass notch filter
-
-	// if at least one is active
-	if (filNum > 0) {
-		// reset values
-		filDist = 0;
-		filterWeightsAcc = 0;
-		filDmp = 0;
-		filAmp = 0;
-		// for each filter
-		for (int k = 0; k < filNum; k++) {
-			// here are a couple of kludges to boost sound output of hi freq filters
-
-			// damping effect of filter increases with distance, but decreases with filter frequency [kludge]
-			float mul = ((filterMaxF-nextFreq)/filterMaxF) * 0.9 + 0.1 ;
-			//filDist = fabs(nextFreq - filFreq[k])*( ((exp(a*4)-1)/EXP_DENOM) * 0.9 + 0.1 );
-			filDist = fabs(nextFreq - filFreq[k])*mul;
-
-			// these to merge all filters contributions according to distance
-			filterWeights[k] = filterMaxF - filDist;
-			filterWeightsAcc += filterWeights[k];
-			// freqs very close to filter center are slightly amplified
-			// the size of this amp area and the effect of amplification increase with frequency [kludge]
-			if (filDist
-					< filterAmpMinF
-							+ (filterAmpMaxF*(1-mul) - filterAmpMinF) * (1 - filQ[k]) )
-				filAmp = filQ[k] * filterAmpMul*(1-mul);
-			else
-				filAmp = 0;
-			// actual damping
-			filDmp = 1 / (filDist * filQ[k]);
-			filDmp = (filDmp > 1) ? 1 : filDmp;
-			// sum damp+amplification
-			filterDamp[k] = filDmp + filAmp;
-		}
-		// do weighted mean to merge all filters contributions
-		filDmp = 0;
-		for (int k = 0; k < filNum; k++)
-			filDmp += filterDamp[k] * filterWeights[k];
-		filDmp /= filterWeightsAcc;
-		// apply
-		parDamp *= filDmp;
-	}
-
-
-	return parDamp;
-}
--- a/examples/d-box/OscillatorBank.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,240 +0,0 @@
-/*
- * OscillatorBank.h
- *
- *  Created on: May 23, 2014
- *      Author: Victor Zappi and Andrew McPherson
- */
-
-#ifndef OSCILLATORBANK_H_
-#define OSCILLATORBANK_H_
-
-
-#include <string>
-
-#include "spear_parser.h"
-#include "ADSR.h"
-#include "config.h"
-
-using namespace std;
-
-enum OscBankstates {bank_stopped, bank_playing, bank_toreset};
-
-class OscillatorBank
-{
-public:
-	OscillatorBank();
-	OscillatorBank(string filename, int hopsize=-1, int samplerate=44100);
-	OscillatorBank(char *filename, int hopsize=-1, int samplerate=44100);
-	~OscillatorBank();
-	float *oscillatorPhases;
-	float *oscillatorNormFrequencies;
-	float *oscillatorNormFreqDerivatives;
-	float *oscillatorAmplitudes;
-	float *oscillatorAmplitudeDerivatives;
-	float *oscStatNormFrequenciesMean;
-	float *oscStatNumHops;
-	OscBankstates state;
-	bool note;
-	int actPartNum;
-	unsigned int *actPart;
-	int hopCounter;
-	int lookupTableSize;
-	float *lookupTable;
-	float ampTh;
-	int hopNumTh;
-	float pitchMultiplier;
-	float freqMovement;
-	int filterNum;
-	float filterFreqs[5];
-	float filterQ[5];
-	float filterMaxF;
-	float filterAmpMinF;
-	float filterAmpMaxF;
-	float filterAmpMul;
-
-	bool loadFile(string filename, int hopsize=-1, int samplerate=44100);
-	bool loadFile(char *filename, int hopsize=-1, int samplerate=44100);
-	bool initBank(int oversamp=1);
-	void resetOscillators();
-	int getHopSize() { return hopSize; }
-	void nextHop();
-	void setLoopHops(int start, int end);
-	void play(float vel);
-	void stop();
-	void afterTouch(float vel);
-	int getEnvelopeState();
-	float getFrequencyScaler();
-	void setSpeed(float sp);
-	float getSpeed();
-	float getMaxSpeed();
-	float getMinSpeed();
-	void setJumpHop(int hop);
-	int getLastHop();
-	int getCurrentHop() { return currentHop; }
-
-private:
-
-	bool loaded;
-	int numOfPartials;
-	int numOfOscillators;
-	int partialsHopSize;
-	int overSampling;
-	int hopSize;
-	int hopSizeReminder;
-	int oscBankHopSize;
-	float frequencyScaler;
-	float nyqNorm;
-	int lastHop;
-	int currentHop;
-	int	loopDir;
-	int loopDirShift;
-	int loopStartHop;
-	int loopEndHop;
-	int *indicesMapping;
-	float *phaseCopies;
-	float *oscillatorNextNormFreq;
-	float *oscillatorNextAmp;
-	float *nextNormFreqCopies;
-	float *nextAmpCopies;
-	float *freqFixedDeltas;
-	float *ampFixedDeltas;
-	bool *nyquistCut;
-	Spear_parser parser;
-	Partials *partials;
-	ADSR adsr;
-	float minAttackTime;
-	float deltaAttackTime;
-	float minReleaseTime;
-	float deltaReleaseTime;
-	int envState;
-	int rate;
-	float speed;
-	float nextSpeed;
-	float maxSpeed;
-	float minSpeed;
-	int jumpHop;
-	float adsrVal;
-	float prevAdsrVal;
-	float prevAmpTh;
-	int prevHopNumTh;
-	float prevPitchMultiplier;
-	float prevFreqMovement;
-	int prevFilterNum;
-	float prevFilterFreqs[5];
-	float prevFilterQ[5];
-
-	bool loader(char *filename, int hopsize=-1, int samplerate=44100);
-	void addFakeOsc();
-	void nextOscBankHop();
-	void nextPartialHop();
-	int jumpToHop();
-	void setDirection(int dir);
-	int nextEnvState();
-	void checkDirection();
-	void checkSpeed();
-	int checkJump();
-	bool checkOversampling();
-	void updatePrevControls();
-	float calculateParDamping(int parIndex, int hopNTh, float adsrVl, float nextFreq,
-							  int filNum, float *filFreq, float *filQ);
-};
-
-inline bool OscillatorBank::loadFile(string filename, int hopsize, int samplerate)
-{
-	return loader((char *)filename.c_str(), hopsize, samplerate);
-}
-
-inline bool OscillatorBank::loadFile(char *filename, int hopsize, int samplerate)
-{
-	return loader(filename, hopsize, samplerate);
-}
-
-inline void OscillatorBank::setLoopHops(int start, int end)
-{
-	if(start > end)
-		end = start;
-
-	if(start<0)
-		start = 0;
-	else if(start>lastHop)
-		start = 0;
-	if(end < 1)
-		end = 1;
-	end = (end<=lastHop) ? end : lastHop;
-
-	// set it, take into consideration hop oversampling
-	loopStartHop = start*overSampling;
-	loopEndHop	 = end*overSampling;
-}
-
-inline void OscillatorBank::stop()
-{
-	note = false;
-	adsr.gate(0);
-}
-
-inline float OscillatorBank::getFrequencyScaler()
-{
-	return frequencyScaler;
-}
-
-inline void OscillatorBank::afterTouch(float vel)
-{
-	hopNumTh = log((1-vel)+1)/log(2)*20000;
-	if(adsr.getState()==env_attack)
-		adsr.setAttackRate( (minAttackTime + ( (1-vel)*deltaAttackTime )) * rate );
-	adsr.setReleaseRate( (minReleaseTime+(1-vel)*deltaReleaseTime)* rate );
-}
-
-inline int OscillatorBank::getEnvelopeState()
-{
-	return envState;
-}
-
-inline void OscillatorBank::setSpeed(float sp)
-{
-	nextSpeed = sp;
-}
-
-inline float OscillatorBank::getSpeed()
-{
-	return speed;
-}
-
-inline float OscillatorBank::getMaxSpeed()
-{
-	return maxSpeed;
-}
-
-inline float OscillatorBank::getMinSpeed()
-{
-	return minSpeed;
-}
-
-inline void OscillatorBank::setJumpHop(int hop)
-{
-	if(hop<0)
-		return;
-	hop = (hop<=lastHop) ? hop : lastHop;
-	jumpHop = hop;
-}
-
-inline void OscillatorBank::setDirection(int dir)
-{
-	if(dir>=0)
-	{
-		loopDir			= 1;
-		loopDirShift	= 0;
-	}
-	else
-	{
-		loopDir			= -1;
-		loopDirShift	= 1;
-	}
-}
-
-inline int OscillatorBank::getLastHop()
-{
-	return lastHop;
-}
-#endif /* OSCILLATORBANK_H_ */
--- a/examples/d-box/PinkNoise.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,16 +0,0 @@
-/*
- * PinkNoise.cpp
- *
- *  Created on: Oct 15, 2013
- *      Author: Victor Zappi
- */
-
-#include "PinkNoise.h"
-
-// miserable definition to init static const array members...otherwise gets error when PinkNoise.h is included into another header file
-const float PinkNoise::A[] = { 0.02109238, 0.07113478, 0.68873558 }; // rescaled by (1+P)/(1-P)
-const float PinkNoise::P[] = { 0.3190,  0.7756,  0.9613  };
-
-
-
-
--- a/examples/d-box/PinkNoise.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,49 +0,0 @@
-#ifndef _PinkNoise_H
-#define _PinkNoise_H
-
-// Technique by Larry "RidgeRat" Trammell 3/2006
-// http://home.earthlink.net/~ltrammell/tech/pinkalg.htm
-// implementation and optimization by David Lowenfels
-
-#include <cstdlib>
-#include <ctime>
-#include <stdlib.h>
-
-#define PINK_NOISE_NUM_STAGES 3
-
-class PinkNoise {
-public:
-  PinkNoise() {
-  srand ( time(NULL) ); // initialize random generator
-    clear();
-  }
-
-  void clear() {
-    for( size_t i=0; i< PINK_NOISE_NUM_STAGES; i++ )
-      state[ i ] = 0.0;
-    }
-
-  float tick() {
-    static const float RMI2 = 2.0 / float(RAND_MAX); // + 1.0; // change for range [0,1)
-    static const float offset = A[0] + A[1] + A[2];
-
-  // unrolled loop
-    float temp = float( rand() );
-    state[0] = P[0] * (state[0] - temp) + temp;
-    temp = float( rand() );
-    state[1] = P[1] * (state[1] - temp) + temp;
-    temp = float( rand() );
-    state[2] = P[2] * (state[2] - temp) + temp;
-    return ( A[0]*state[0] + A[1]*state[1] + A[2]*state[2] )*RMI2 - offset;
-  }
-
-protected:
-  float state[ PINK_NOISE_NUM_STAGES ];
-  static const float A[ PINK_NOISE_NUM_STAGES ];
-  static const float P[ PINK_NOISE_NUM_STAGES ];
-};
-
-//const float PinkNoise::A[] = { 0.02109238, 0.07113478, 0.68873558 }; // rescaled by (1+P)/(1-P)
-//const float PinkNoise::P[] = { 0.3190,  0.7756,  0.9613  };
-
-#endif
--- a/examples/d-box/StatusLED.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,90 +0,0 @@
-/*
- * StatusLED.cpp
- *
- * Routines for manipulating the status LED
- *
- * (c) 2014 Andrew McPherson and Victor Zappi
- * QMUL, Centre for Digital Music
- */
-
-#include <iostream>
-#include "StatusLED.h"
-#include <GPIOcontrol.h>
-
-extern int gShouldStop;
-extern int gVerbose;
-
-using namespace std;
-
-StatusLED::StatusLED() {
-	gpio_number = -1;
-	milliseconds_on = 0;
-	milliseconds_off = 100;
-	blink_thread = -1;
-}
-
-StatusLED::~StatusLED() {
-	if(gpio_number >= 0) {
-		this_should_stop = true;
-		pthread_join(blink_thread, NULL);
-		gpio_unexport(gpio_number);
-	}
-}
-
-bool StatusLED::init(int gpio_pin) {
-	gpio_number = gpio_pin;
-	this_should_stop = false;
-
-	if(gpio_export(gpio_number)) {
-		if(gVerbose)
-			cout << "Warning: couldn't export status LED pin\n";
-	}
-	if(gpio_set_dir(gpio_number, OUTPUT_PIN)) {
-		if(gVerbose)
-			cout << "Couldn't set direction on status LED pin\n";
-		return false;
-	}
-	if(gpio_set_value(gpio_number, LOW)) {
-		if(gVerbose)
-			cout << "Couldn't set value on status LED pin\n";
-		return false;
-	}
-
-
-	if ( pthread_create(&blink_thread, NULL, static_blink_loop, this) )
-	{
-		cout << "Error:unable to create status LED thread" << endl;
-		return false;
-	}
-
-	return true;
-}
-
-void StatusLED::on() {
-	milliseconds_on = 100;
-	milliseconds_off = 0;
-}
-
-void StatusLED::off() {
-	milliseconds_on = 0;
-	milliseconds_off = 100;
-}
-
-void StatusLED::blink(int ms_on, int ms_off) {
-	milliseconds_on = ms_on;
-	milliseconds_off = ms_off;
-}
-
-void* StatusLED::blink_loop(void *) {
-	while(!gShouldStop && !this_should_stop) {
-		if(milliseconds_on != 0)
-			gpio_set_value(gpio_number, HIGH);
-		usleep(1000 * milliseconds_on);
-		if(gShouldStop)
-			break;
-		if(milliseconds_off != 0)
-			gpio_set_value(gpio_number, LOW);
-		usleep(1000 * milliseconds_off);
-	}
-	pthread_exit(NULL);
-}
--- a/examples/d-box/StatusLED.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,39 +0,0 @@
-/*
- * StatusLED.h
- *
- *
- */
-
-#ifndef STATUSLED_H_
-#define STATUSLED_H_
-
-#include <pthread.h>
-#include <unistd.h>
-
-class StatusLED
-{
-public:
-	StatusLED();
-	~StatusLED();
-
-	bool init(int gpio_pin);
-
-	void on();
-	void off();
-	void blink(int ms_on, int ms_off);
-
-	static void *static_blink_loop(void *data) {
-		((StatusLED*)data)->blink_loop(NULL);
-		return 0;
-	}
-
-	void* blink_loop(void *);
-
-private:
-	int gpio_number;
-	int milliseconds_on, milliseconds_off;
-	bool this_should_stop;
-	pthread_t blink_thread;
-};
-
-#endif // STATUSLED_H_
--- a/examples/d-box/audio_routines.S	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,185 +0,0 @@
-@
-@ audio_routines.S
-@
-@ NEON-based functions for time-critical audio processing
-@
-@ Andrew McPherson 2014
-@ Queen Mary University of London
-@
-
-	.syntax unified
-	.arch armv7-a
-	.fpu neon
-
-@ 	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
-@							  int activePartialNum, int lookupTableSize,
-@							  float *phases, float *frequencies, float *amplitudes,
-@							  float *freqDerivatives, float *ampDerivatives,
-@							  float *lookupTable);
-
-@ Registers:
-@    r0: numAudioFrames        How many frames to render
-@    r1: audioOut              Buffer for audio output samples [stereo]
-@    r2: activePartialNum      How many active partials to render
-@    r3: lookupTableSize       Size of lookup table
-@    ---- other arguments start on the stack and are moved: -----
-@    r4: phases                Phase of each oscillator (pointer)
-@    r5: frequencies           Normalised frequency of each oscillator (pointer)
-@    r6: amplitudes            Normalised amplitude of each oscillator (pointer)
-@    r7: freqDerivatives       Derivative of frequency for each oscillator (pointer)
-@    r8: ampDerivatives        Derivative of amplitude for each oscillator (pointer)
-@    r9: lookupTable           Lookup table containing one oscillation
-@
-@ Alignment requirements:
-@    audioOut: 8-byte boundary
-@    phases: 16-byte boundary
-@    frequencies: 16-byte boundary
-@    amplitudes: 16-byte boundary
-@    freqDerivatives: 16-byte bounary
-@    ampDerivatives: 16-byte boundary
-@    lookupTable: 4-byte boundary (TODO: check this)
-
-	.align	2
-	.global	oscillator_bank_neon
-	.thumb
-	.thumb_func
-	.type	oscillator_bank_neon, %function
-oscillator_bank_neon:
-
-
-dSample		.dn		D6.F32
-qPhases		.qn		Q8.F32
-dPhases_0	.dn		D16.F32
-dPhases_1	.dn		D17.F32
-qFreqs		.qn		Q9.F32
-dFreqs_0	.dn		D18.F32
-dFreqs_1	.dn		D19.F32
-qAmps		.qn		Q10.F32
-dAmps_0		.dn		D20.F32
-dAmps_1		.dn		D21.F32
-qFreqDs		.qn		Q11.F32
-dFreqDs_0	.dn		D22.F32
-dFreqDs_1	.dn		D23.F32
-qAmpDs		.qn		Q12.F32
-dAmpDs_0	.dn		D24.F32
-dAmpDs_1	.dn		D25.F32
-
-qBaseInts	.qn		Q13.U32		@ Base indexes: unsigned ints x4
-dBaseInts_0	.dn		D26.U32
-dBaseInts_1	.dn		D27.U32
-qFractions  .qn     Q14.F32		@ Fraction indexes: floats x4
-qTableBase	.qn		Q15.U32		@ Base of lookup table
-
-	cmp r0, #0					@ Check for trivial case 1: zero frames
-	it eq
-	bxeq lr						@ Return if that's the case (otherwise might have odd behaviour)
-	cmp r2, #4					@ Check for trivial case 2: zero oscillators
-	it lt
-	bxlt lr						@ Return if that's the case
-
-	push {r4-r11}				@ Now arguments start 32 bytes above SP
-    add r11, sp, #32			@ Pointer to 32 bytes into the stack
-    ldm r11, {r4-r9}			@ Load 6 arguments into registers
-
-	vdup qTableBase, r9			@ Move lookup table base index into 4 ints
-
-	@ Outer loop: iterate over the number of oscillators, choosing 4 at a
-	@ time to work with.
-oscbank_oscillator_loop:
-	vld1 {dPhases_0, dPhases_1}, [r4]		@ no increment; will store at end of sample loop
-	vld1 {dFreqs_0, dFreqs_1}, [r5]
-	vld1 {dAmps_0, dAmps_1}, [r6]
-	vld1 {dFreqDs_0, dFreqDs_1}, [r7]!		@ increment; won't update at end of sample loop
-	vld1 {dAmpDs_0, dAmpDs_1}, [r8]!
-
-	push {r0-r1,r4-r8}
-	@ --- inner loop: iterate over the number of samples ---
-oscbank_sample_loop:
-	vcvt qBaseInts, qPhases		     		@ Take floor(phases)
-	vmov q2.f32, #1.0						@ Load 1.0 into every slot of q2
-	vshl q0.U32, qBaseInts, #2				@ Shift the indexes left 2 (*4 for float addressing)
-	vcvt qFractions, qBaseInts				@ int back to float
-	vadd q0.U32, q0.U32, qTableBase			@ Find memory addresses
-
-	vmov r4, r5, d0							@ Move two indexes to ARM registers
-	vmov r6, r7, d1							@ Move two more indexes to ARM registers
-	vsub qFractions, qPhases, qFractions	@ fraction = phase - floor(phase)
-
-	vldr.64	d0, [r4]						@ Load two consecutive floats at each location
-	vldr.64 d1, [r5]						@ These hold the previous and following samples in the table
-	vldr.64	d2, [r6]						@ TODO: check whether these work at 4-byte alignment
-	vldr.64 d3, [r7]
-
-	@ Format at this point:
-	@ Osc0(before) Osc0(after) Osc1(before) Osc1(after) Osc2(before) Osc2(after) Osc3(before) Osc3(after)
-	@ We want:
-	@ Osc0(before) Osc1(before) Osc2(before) Osc3(before) Osc0(after) Osc1(after) Osc2(after) Osc3(after)
-
-	vuzp.32 q0, q1							@ Now q0 contains before, q1 contains after
-	vsub q2.f32, q2.f32, qFractions			@ q2 = 1.0 - fraction
-	vmul q1.f32, q1.f32, qFractions			@ q1 = fraction * after
-	vmul q0.f32, q0.f32, q2.f32				@ q0 = (1.0 - fraction) * before
-
-	vadd qPhases, qPhases, qFreqs			@ Update phases
-	vadd qFreqs, qFreqs, qFreqDs			@ Update frequencies
-
-	vadd q0.f32, q0.f32, q1.f32				@ Add two interpolated components to get the final sample
-	vdup q2.u32, r3							@ Put lookup table size into each element of q2
-	vcvt qBaseInts, qPhases					@ Take floor of new phases
-	vmul q0.f32, q0.f32, qAmps				@ Multiply samples by current amplitude
-
-	vld1 dSample, [r1]						@ Load the current stereo samples
-	vpadd d2.f32, d0.f32, d1.f32			@ Pairwise accumulate q0 (output sample) into d2
-
-	vand q2, q2, qBaseInts					@ Logical AND of new phase int leaves 1 bit set only if phase >= table size
-	vpadd d3.f32, d2.f32, d2.f32			@ Pairwise accumulate d2 into d0 --> d0[0] and d0[1] both hold total of 4 oscillators
-	vadd qAmps, qAmps, qAmpDs				@ Update amplitudes
-	vcvt q0.f32, q2.u32						@ Convert int back to float after AND operation
-
-	vadd  dSample, dSample, d3.f32			@ Add oscillator outputs to each channel
-
-	subs r0, r0, #1							@ numFrames--
-	vsub qPhases, qPhases, q0.f32			@ Keep phases in table range
-	vst1 dSample, [r1]!						@ Store back in buffer and increment by 8
-
-	it gt
-	bgt oscbank_sample_loop					@ Loop if numFrames > 0
-
-	@ --- end inner loop ---
-	pop {r0-r1,r4-r8}						@ Restore registers: restores audioOut and numFrames, among others
-
-	vst1 {dPhases_0, dPhases_1}, [r4]!		@ Store phases back to array
-	vst1 {dFreqs_0, dFreqs_1}, [r5]!		@ Store frequencies back to array
-	vst1 {dAmps_0, dAmps_1}, [r6]!			@ Store amplitudes back to array
-											@ No need to update r7, r8
-
-	subs r2, r2, #4							@ numPartials -= 4
-	it  gt
-	bgt oscbank_oscillator_loop	@ Loop if numPartials > 0
-
-    pop {r4-r11}
-	bx lr
-
-
-@   void wavetable_interpolate_neon(int numSamplesIn, int numSamplesOut,
-@                              float *tableIn, float *tableOut);
-
-@ Registers:
-@    r0: numSamplesIn          Size of the input table
-@    r1: numSamplesOut         Size of the output table
-@    r2: tableIn               Pointer to input table
-@    r3: tableOut              Pointer to output table
-
-@ Alignment requirements:
-@    tableIn: 8-byte boundary
-@    tableOut: 8-byte boundary
-
-	.align	2
-	.global	wavetable_interpolate_neon
-	.thumb
-	.thumb_func
-	.type	wavetable_interpolate_neon, %function
-wavetable_interpolate_neon:
-    @ TODO
-
-    bx lr
--- a/examples/d-box/config.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,43 +0,0 @@
-/*
- * config.h
- *
- * Global settings for D-Box project
- *
- * Andrew McPherson and Victor Zappi 2014
- */
-
-
-#ifndef DBOX_CONFIG_H_
-#define DBOX_CONFIG_H_
-
-
-/* Number of maximum touches used by the TouchKey sensors */
-#define MAX_TOUCHES 5
-
-// for sensor 1 filter
-#define EXP_DENOM 53.5981500331			// exp(4)-1
-
-/* Define this to use Xenomai real-time extensions */
-#define DBOX_USE_XENOMAI
-//#define OLD_OSCBANK
-
-#ifdef DBOX_USE_XENOMAI
-// Xenomai-specific includes
-#include <sys/mman.h>
-
-#include <native/task.h>
-#include <native/timer.h>
-#include <rtdk.h>
-#endif
-
-#ifdef DBOX_USE_XENOMAI
-
-#define dbox_printf rt_printf
-
-#else
-
-#define dbox_printf printf
-
-#endif
-
-#endif /* DBOX_CONFIG_H */
--- a/examples/d-box/how_to_build_dbox.txt	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,35 +0,0 @@
-The D-Box code can be built on the board using the scripts included with
-Bela. To build the code, do the following from the scripts/ directory:
-
-./setup_board.sh    # Only if Bela is not already copied to the board
-./build_project.sh -n ../projects/d-box  # -n means don't run yet
-
-You will now need to copy the .dbx sound files to the box. These need to
-be in a directory: /root/d-box/sounds
-
-Log into the board via ssh to create the d-box directory, or run:
-
-ssh root@192.168.7.2 "mkdir /root/d-box"
-
-Now copy the .dbx files from a source of your choice to the board. Assuming
-they are in a directory called "sounds":
-
-scp -r sounds root@192.168.7.2:/root/d-box/
-
-Now you can run the D-Box program to make sure it plays. You will need the
-sensors and matrix connected to the box (i.e. a complete hardware D-Box).
-From the scripts/ directory:
-
-./run_project.sh -f -c "-q 24 -r 25 -t 2 -s -p 2 -l 0 -u 0 -i 1 -n f -g 3 -v"
-
-If this plays properly, the D-Box can then be set to run automatically on boot:
-
-./set_startup.sh -l -c "-q 24 -r 25 -t 2 -s -p 2 -l 0 -u 0 -i 1 -n f -g 3 -v"
-
-If at any time you want to stop the D-Box program running, you can run:
-
-./stop_running.sh
-
-To turn off further autoboot, run:
-
-./set_startup.sh -n
--- a/examples/d-box/logger.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,188 +0,0 @@
-/*
- * logger.cpp
- *
- *  Created on: Aug 6, 2014
- *      Author: VIctor Zappi and Andrew McPherson
- */
-
-#include "logger.h"
-
-// main extern vars
-extern int gShouldStop;
-extern int gVerbose;
-
-// file nanme extern vars
-extern char gId;
-extern char gGroup;
-
-
-// logged extern vars
-extern int s0TouchNum;
-extern float s0Touches_[MAX_TOUCHES];
-extern float s0Size_[MAX_TOUCHES];
-extern int s0LastIndex;
-
-extern int s1TouchNum;
-extern float s1Touches_[MAX_TOUCHES];
-extern float s1Size_[MAX_TOUCHES];
-extern int s1LastIndex;
-
-extern int fsr;
-
-
-
-string logPath			= "/boot/uboot/instrumentLog";
-string logFileIncipit	= "/datalog";
-string logFileName		= "";
-ofstream logFile;
-timeval logTimeVal;
-unsigned long long logTimeOrig;
-int logCnt				= 0;	// counts how many lines so far
-int logCntSwap			= 50;	// how many log lines before closing and re-opening the file
-
-
-// create the log file, using incremental name convention
-int initLogLoop()
-{
-	if(gVerbose==1)
-		cout << "---------------->Init Log Thread" << endl;
-
-
-	// transform chars into strings via stringstream objs
-	stringstream id_ss, group_ss, freedom_ss;
-	id_ss 		<< gId;
-	group_ss 	<< gGroup;
-
-	int logNum	= -1;
-	int logMax	= -1;
-	int pathLen	= logPath.length() + logFileIncipit.length() + 4;	// + 4 is: "_", id, group, "_"
-	glob_t globbuf;
-
-	// check how many log files are already there, and choose name according to this
-	glob( (logPath + logFileIncipit + "*").c_str(), 0, NULL, &globbuf);
-
-	// cycle through all and find the highest index
-	for(unsigned int i=0; i<globbuf.gl_pathc; i++)
-	{
-		// playing with 0-9 char digits, forming a number from 0 to 9999
-		logNum  = (globbuf.gl_pathv[i][pathLen]-48)   * 1000;	// 42 to 45 are the indices of the chars forming the file index
-		logNum += (globbuf.gl_pathv[i][pathLen+1]-48) * 100;
-		logNum += (globbuf.gl_pathv[i][pathLen+2]-48) * 10;
-		logNum +=  globbuf.gl_pathv[i][pathLen+3]-48;
-		if(logNum > logMax)
-			logMax = logNum;
-	}
-	logNum = logMax + 1;	// new index
-
-	globfree(&globbuf);
-
-	ostringstream numString;
-	numString << setw (4) << setfill ('0') << logNum;	// set integer with 4 figures
-
-	// here are the new names: PATH + DIR + INCIPIT + _ + id + group + freedom + _ + NUM (4figures) + _A.txt
-	logFileName	= logPath + logFileIncipit;
-	logFileName	+= "_" + id_ss.str() + group_ss.str() + freedom_ss.str();
-	logFileName	+= "_" + numString.str();	 //static_cast<ostringstream*>( &(ostringstream() << logNum) )->str();
-	logFileName	+= ".txt";
-
-
-	// create new files
-	FILE *fp_a		= fopen(logFileName.c_str(), "wb");
-	if(!fp_a)
-	{
-		dbox_printf("Cannot create files...\n");
-		return 2;
-	}
-	fclose(fp_a);
-
-	// ready to append
-	logFile.open(logFileName.c_str(), ios::out | ios::app);
-
-	dbox_printf("Logging on file %s\n", logFileName.c_str());
-
-	return 0;
-}
-
-
-void writeData(unsigned long long time)
-{
-
-	float fsr_ = ((float)(1799-fsr)/1799.0);
-	logFile << time 				<< "\t"		// timestamp
-			<< s0TouchNum			<< "\t";	// sensor 0 touch count
-	for(int i=0; i<MAX_TOUCHES; i++)
-		logFile << s0Touches_[i] 	<< "\t";	// sensor 0 touch pos x
-	for(int i=0; i<MAX_TOUCHES; i++)
-		logFile << s0Size_[i] 		<< "\t";	// sensor 0 touch size
-	logFile << s0LastIndex 			<< "\t"		// sensor 0 last index
-			<< fsr_					<< "\t"		// sensor 0 FSR pressure
-			<< s1TouchNum			<< "\t";	// sensor 1 touch count
-	for(int i=0; i<MAX_TOUCHES; i++)
-		logFile << s1Touches_[i] 	<< "\t";	// sensor 1 touch pos x
-	for(int i=0; i<MAX_TOUCHES; i++)
-		logFile	<< s1Size_[i] 		<< "\t";	// sensor 1 touch size
-	logFile << s1LastIndex 			<< "\t"		// sensor 1 last index
-	//... AND SO ON
-			<< "\n";
-
-	//dbox_printf("%d\n", s0LastIndex);
-	//dbox_printf("s0TouchNum: %d\t s0Touches[0]: %f\t s0Size[0]: %f\t s0LastIndex: %d\n", s0TouchNum, s0Touches_[0], s0Size_[0], s0LastIndex);
-
-}
-
-void logData(unsigned long long time)
-{
-	// if it's time to change write-file
-	if(logCnt >= logCntSwap)
-	{
-		logFile.close();	// close file, dump stream
-		logCnt = 0;		// ready for another whole round
-
-		// open again, ready to append
-		logFile.open(logFileName.c_str(), ios::out | ios::app);
-	}
-
-	writeData(time);
-
-	logCnt++;
-}
-
-
-
-
-void *logLoop(void *)
-{
-	set_realtime_priority(10);
-
-	if(gVerbose==1)
-		dbox_printf("_________________Log Thread!\n");
-
-	// get time reference
-	gettimeofday(&logTimeVal, NULL);
-	logData(0);
-
-	logTimeOrig = logTimeVal.tv_usec;
-	logTimeOrig *= 0.001;					// from usec to msec
-	logTimeOrig += logTimeVal.tv_sec*1000;	// from sec to msec
-
-	usleep(5000);
-
-	while(!gShouldStop)
-	{
-		gettimeofday(&logTimeVal, NULL);
-		unsigned long long currentTime = logTimeVal.tv_usec;
-		currentTime *= 0.001;					// from usec to msec
-		currentTime += logTimeVal.tv_sec*1000;	// from sec to msec
-
-		logData(currentTime-logTimeOrig);
-
-		usleep(5000);
-	}
-
-	if(logFile!=NULL)
-		logFile.close();
-
-	dbox_printf("log thread ended\n");
-
-	return (void *)0;
-}
--- a/examples/d-box/logger.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,31 +0,0 @@
-/*
- * logger.h
- *
- *  Created on: Aug 6, 2014
- *      Author: Victor Zappi and Andrew McPherson
- */
-
-#ifndef LOGGER_H_
-#define LOGGER_H_
-
-#include <string.h>
-#include <pthread.h>
-#include <stdio.h>
-#include <unistd.h>
-#include <fstream>		// file handle
-#include <iostream>		// stringstream
-#include <sstream>		// stringstream
-#include <glob.h>		// alternative to dirent.h to handle files in dirs
-#include <iomanip>		// setfill
-#include <sys/time.h>	// elapsed time
-
-#include "config.h"
-#include "prio.h"
-
-using namespace std;
-
-int initLogLoop();
-void *logLoop(void *);
-
-
-#endif /* LOGGER_H_ */
--- a/examples/d-box/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,491 +0,0 @@
-/*
- *  RTAudio.cpp
- *
- *  Central control code for hard real-time audio on BeagleBone Black
- *  using PRU and Xenomai Linux extensions. This code began as part
- *  of the Hackable Instruments project (EPSRC) at Queen Mary University
- *  of London, 2013-14.
- *
- *  (c) 2014 Victor Zappi and Andrew McPherson
- *  Queen Mary University of London
- */
-
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <strings.h>
-#include <math.h>
-#include <iostream>
-#include <signal.h>		// interrupt handler
-#include <assert.h>
-#include <vector>
-#include <dirent.h>		// to handle files in dirs
-#include <mntent.h>		// to check if device is mounted
-#include <sys/mount.h>	// mount()
-#include <sys/time.h>	// elapsed time
-#include <ne10/NE10.h>	// neon library
-
-// thread priority
-#include <pthread.h>
-#include <sched.h>
-
-// get_opt_long
-#include <getopt.h>
-
-#include <Bela.h>
-#include "config.h"
-#include "sensors.h"
-#include "OscillatorBank.h"
-#include "StatusLED.h"
-#include "logger.h"
-
-using namespace std;
-
-//----------------------------------------
-// main variables
-//----------------------------------------
-vector<OscillatorBank*> gOscBanks;
-int gCurrentOscBank = 0;
-int gNextOscBank = 0;
-int oscBnkOversampling 				 = 1;	// oscillator bank frame oversampling
-
-const int kStatusLEDPin 	 = 30;	// P9-11 controls status LED
-StatusLED gStatusLED;
-
-pthread_t keyboardThread;
-pthread_t logThread;
-
-// general settings
-int gVerbose		= 0;        			// verbose flag
-bool forceKeyboard 	= true;					// activate/deactivate keyboard control
-bool forceSensors	= false;				// activate/deactivate sensor control
-bool forceLog		= true;					// activate/deactivate log on boot partition
-bool useSD   		= true;    				// activate/deactivate file loading from SD [as opposed to emmc]
-bool useAudioTest   = false;    			// activate/deactivate sensors and test audio only
-
-// audio settings
-unsigned int gPeriodSize = 8;				// period size for audio
-char* gPartialFilename = 0;					// name of the partials file to load
-bool gAudioIn = false;						// stereo audio in status
-
-int touchSensor0Address = 0x0C;				// I2C addresses of touch sensors
-int touchSensor1Address = 0x0B;
-int sensorType = 0;
-
-char sdPath[256]			= "/dev/mmcblk0p2";			// system path of the SD, partition 2
-char mountPath[256]			= "/root/d-box/usersounds";	// mount point of SD partition 2 [where user files are]
-char gUserDirName[256] 		= "usersounds";				// Directory in which user analysis files can be found [dir of mountPath]
-char gDefaultDirName[256] 	= "/root/d-box/sounds";		// Directory in which built in analysis files can be found
-char *gDirName;
-bool gIsLoading 			= false;
-int fileCnt 				= 0;
-std::vector <std::string> files;
-
-char gId	= 'f';	// from 0 to 14, hexadecimal [0-d]! f means not set
-char gGroup	= '2';	// 0 is no info, 1 info.   2 is not set
-
-// audio in filter
-extern ne10_float32_t *filterState[2];
-extern ne10_float32_t *filterIn[2];
-extern ne10_float32_t *filterOut[2];
-
-struct arg_data
-{
-   int  argc;
-   char **argv;
-};
-
-arg_data args;
-
-
-int readFiles()
-{
-	if(useSD)
-		gDirName = gUserDirName;
-	else
-		gDirName = gDefaultDirName;
-	DIR *dir;
-	struct dirent *ent;
-
-	// From http://stackoverflow.com/questions/612097/how-can-i-get-a-list-of-files-in-a-directory-using-c-or-c
-	if ((dir = opendir (gDirName)) != NULL) {
-		/* print all the files and directories within directory */
-		while ((ent = readdir (dir)) != NULL) {
-			// Ignore dotfiles and . and .. paths
-			if(!strncmp(ent->d_name, ".", 1))
-				continue;
-
-			//dbox_printf("%s\n", ent->d_name);
-
-			// take only .dbx and .txt files
-			string name = string(ent->d_name);
-			int len		= name.length();
-
-			bool dboxFile = false;
-
-			if( (name[len-4]=='.') && (name[len-3]=='d') && (name[len-2]=='b') && (name[len-1]=='x') )
-				dboxFile = true;
-			if( (name[len-4]=='.') && (name[len-3]=='t') && (name[len-2]=='x') && (name[len-1]=='t') )
-				dboxFile = true;
-
-			if(dboxFile)
-			{
-				fileCnt++;
-				//dbox_printf("%s\n", ent->d_name);
-				files.push_back( std::string( ent->d_name ) );
-			}
-		}
-		closedir (dir);
-	} else {
-		/* could not open directory */
-		printf("Could not open directory %s\n", gDirName);
-		return 1;
-	}
-
-	// order by name
-	std::sort( files.begin(), files.end() );
-
-	if(fileCnt==0)
-	{
-		printf("No .dbx or .txt files in %s!\n", gDirName);
-		return 1;
-	}
-
-	return 0;
-}
-
-/* Load sounds from the directory */
-void loadAudioFiles(bool loadFirstFile)
-{
-	char fullFileName[256];
-
-	if(loadFirstFile) {
-		strcpy(fullFileName, gDirName);
-		strcat(fullFileName, "/");
-		strncat(fullFileName, files[0].c_str(), 255 - strlen(gDirName));
-		dbox_printf("Loading first file %s...\n", fullFileName);
-		OscillatorBank *bank = new OscillatorBank(fullFileName);
-		if(bank->initBank(oscBnkOversampling)) {
-			bank->setLoopHops(100, bank->getLastHop());
-			gOscBanks.push_back(bank);
-		}
-	}
-
-	else {
-		for(int i=1; i<fileCnt; i++){
-			strcpy(fullFileName, gDirName);
-			strcat(fullFileName, "/");
-			strncat(fullFileName, files[i].c_str(), 255 - strlen(gDirName));
-			dbox_printf("Loading file %s...\n", fullFileName);
-			OscillatorBank *bank = new OscillatorBank(fullFileName);
-			if(bank->initBank(oscBnkOversampling)) {
-				bank->setLoopHops(100, bank->getLastHop());
-				gOscBanks.push_back(bank);
-			}
-		}
-	}
-}
-
-// adapted from http://program-nix.blogspot.co.uk/2008/08/c-language-check-filesystem-is-mounted.html
-int checkIfMounted (char * dev_path)
-{
-	FILE * mtab				= NULL;
-	struct mntent * part	= NULL;
-	int is_mounted			= 0;
-
-	if ( ( mtab = setmntent ("/etc/mtab", "r") ) != NULL)
-	{
-		while ( ( part = getmntent ( mtab) ) != NULL)
-		{
-			if ( ( part->mnt_fsname != NULL ) && ( strcmp ( part->mnt_fsname, dev_path ) ) == 0 )
-			   is_mounted = 1;
-		}
-	endmntent(mtab);
-	}
-	return is_mounted;
-}
-
-int mountSDuserPartition()
-{
-	if(checkIfMounted(sdPath))
-	{
-		printf("device %s already mounted, fair enough, let's move on\n", sdPath);
-		return 0;
-	}
-	// if mount rootfs from SD [rootfs eMMC not used] or from eMMC via properly formatted SD [SD rootfs used as storage volume]
-	// we always use rootfs on SD as storage volume ----> "/dev/mmcblk0p2"
-	int ret = mount(sdPath, "/root/d-box/usersounds", "vfat",  0, NULL);
-	if (ret!=0)
-	{
-			printf("Error in mount...%s\n", strerror(ret));
-			return 1;
-	}
-	return 0;
-}
-
-int initSoundFiles()
-{
-	if(gVerbose==1)
-		cout << "---------------->Init Audio Thread" << endl;
-
-	if(useSD)
-	{
-		// mount the SD partition where user sounds are located
-		// [this is p2, p1 is already mounted and we will log data there]
-		if(mountSDuserPartition()!=0)
-			return -1;
-	}
-
-	gIsLoading = true;
-
-	// read files from SD and order them alphabetically
-	if(readFiles()!=0)
-		return 1;
-
-	// load first file into oscBank
-	loadAudioFiles(true);
-
-	return 0;
-}
-
-//---------------------------------------------------------------------------------------------------------
-
-// Handle Ctrl-C
-void interrupt_handler(int var)
-{
-	// kill keyboard thread mercilessly
-	if(forceKeyboard)
-		pthread_cancel(keyboardThread);
-
-	gShouldStop = true;
-}
-
-
-void parseArguments(arg_data args, BelaInitSettings *settings)
-{
-	// Default filename;
-	gPartialFilename = strdup("D-Box_sound_250_60_40_h88_2.txt");
-
-	const int kOptionAudioTest = 1000;
-
-	// TODO: complete this
-	struct option long_option[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"audioin", 1, NULL, 'i'},
-		{"file", 1, NULL, 'f'},
-		{"keyboard", 1, NULL, 'k'},
-		{"audio-test", 0, NULL, kOptionAudioTest},
-		{"sensor-type", 1, NULL, 't'},
-		{"sensor0", 1, NULL, 'q'},
-		{"sensor1", 1, NULL, 'r'},
-		{"log", 1, NULL, 'l'},
-		{"usesd", 1, NULL, 'u'},
-		{"oversamp", 1, NULL, 'o'},
-		{"boxnumber", 1, NULL, 'n'},
-		{"group", 1, NULL, 'g'},
-		{NULL, 0, NULL, 0},
-	};
-	int morehelp = 0;
-	int tmp = -1;
-
-	Bela_defaultSettings(settings);
-
-	while (1)
-	{
-		int c;
-		if ((c = Bela_getopt_long(args.argc, args.argv, "hf:ki:sq:r:t:l:u:o:n:g:", long_option, settings)) < 0)
-				break;
-		switch (c)
-		{
-		case 'h':
-				morehelp++;
-				break;
-		case 'f':
-				free(gPartialFilename);
-				gPartialFilename = strdup(optarg);
-				break;
-		case 'k':
-				forceKeyboard = true;
-				break;
-		case 'i':
-				gAudioIn = (atoi(optarg)==0) ? false : true;
-				break;
-		case 's':
-				forceSensors = true;
-				break;
-		case kOptionAudioTest:
-				useAudioTest = true;
-				break;
-		case 't':
-				sensorType = atoi(optarg);
-				break;
-		case 'q':
-				touchSensor0Address = atoi(optarg);
-				break;
-		case 'r':
-				touchSensor1Address = atoi(optarg);
-				break;
-		case 'l':
-				tmp = atoi(optarg);
-				if(tmp==0)
-					forceLog = false;
-				else if(tmp>0)
-					forceLog = true;
-				break;
-		case 'u':
-				tmp = atoi(optarg);
-				if(tmp==0)
-					useSD = false;
-				else if(tmp>0)
-					useSD = true;
-				break;
-		case 'o':
-				oscBnkOversampling = atoi(optarg);
-				break;
-		case 'n':
-				gId = *optarg;
-				cout << "-set box number to: " << gId << endl;
-				break;
-		case 'g':
-				gGroup = *optarg;
-				cout << "-set group to: " << gId << endl;
-				break;
-		default:
-				break;
-		}
-	}
-
-	gPeriodSize = settings->periodSize;
-	gVerbose = settings->verbose;
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	RT_TASK rtSensorThread;
-	const char rtSensorThreadName[] = "dbox-sensor";
-	int oscBankHopSize;
-
-	// Parse command-line arguments
-	args.argc = argc;
-	args.argv = argv;
-	parseArguments(args, &settings);
-
-	Bela_setVerboseLevel(gVerbose);
-	if(gVerbose == 1 && useAudioTest)
-		cout << "main() : running in audio test mode" << endl;
-
-	// Load sound files from directory
-	if(initSoundFiles() != 0)
-		return -1;
-
-	oscBankHopSize = gOscBanks[gCurrentOscBank]->getHopSize()/gOscBanks[gCurrentOscBank]->getMinSpeed();
-
-	// Initialise the audio device
-	if(Bela_initAudio(&settings, &oscBankHopSize) != 0)
-		return -1;
-
-	// Initialise the status LED
-	if(!gStatusLED.init(kStatusLEDPin)) {
-		if(gVerbose)
-			cout << "Couldn't initialise status LED pin\n";
-	}
-
-	// Free file name string which is no longer needed
-	if(gPartialFilename != 0)
-		free(gPartialFilename);
-
-	if(!useAudioTest) {
-		if(initSensorLoop(touchSensor0Address, touchSensor1Address, sensorType) != 0)
-			return -1;
-	}
-
-	if(gVerbose == 1)
-		cout << "main() : creating audio thread" << endl;
-
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// LED on...
-	gStatusLED.on();
-
-	if(forceSensors && !useAudioTest) {
-		if(gVerbose==1)
-			cout << "main() : creating control thread" << endl;
-
-		if(rt_task_create(&rtSensorThread, rtSensorThreadName, 0, BELA_AUDIO_PRIORITY - 5, T_JOINABLE | T_FPU)) {
-			  cout << "Error:unable to create Xenomai control thread" << endl;
-			  return -1;
-		}
-		if(rt_task_start(&rtSensorThread, &sensorLoop, 0)) {
-			  cout << "Error:unable to start Xenomai control thread" << endl;
-			  return -1;
-		}
-	}
-
-	if(forceKeyboard) {
-		if(gVerbose==1)
-			cout << "main() : creating keyboard thread" << endl;
-
-		if ( pthread_create(&keyboardThread, NULL, keyboardLoop, NULL) ) {
-		  cout << "Error:unable to create keyboard thread" << endl;
-		  return -1;
-		}
-	}
-
-	if(forceLog) {
-		if(gVerbose==1)
-			cout << "main() : creating log thread" << endl;
-
-		if(initLogLoop()!=0) {
-			cout << "Error:unable to create log thread" << endl;
-			return -1;
-		}
-
-		if ( pthread_create(&logThread, NULL, logLoop, NULL) ) {
-		  cout << "Error:unable to create keyboard thread" << endl;
-		  return -1;
-		}
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// load all other files into oscBanks
-	loadAudioFiles(false);
-	cout << "Finished loading analysis files\n";
-	gIsLoading = false;
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	Bela_stopAudio();
-
-	if(!useAudioTest)
-		rt_task_join(&rtSensorThread);
-
-	Bela_cleanupAudio();
-
-	pthread_join( keyboardThread, NULL);
-	pthread_join( logThread, NULL);
-
-	for(unsigned int i = 0; i < gOscBanks.size(); i++)
-		delete gOscBanks[i];
-
-	NE10_FREE(filterState[0]);
-	NE10_FREE(filterState[1]);
-	NE10_FREE(filterIn[0]);
-	NE10_FREE(filterIn[1]);
-	NE10_FREE(filterOut[0]);
-	NE10_FREE(filterOut[1]);
-
-	printf("Program ended\nBye bye\n");
-	return 0;
-}
--- a/examples/d-box/prio.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,48 +0,0 @@
-/*
- * prio.cpp
- *
- *  Created on: May 14, 2014
- *      Author: Victor Zappi
- */
-
-#include "prio.h"
-using namespace std;
-//-----------------------------------------------------------------------------------------------------------
-// set wanted real-time priority to this thread
-//-----------------------------------------------------------------------------------------------------------
-void set_realtime_priority(int order)
-{
-    int ret;
-
-    // We'll operate on the currently running thread.
-    pthread_t this_thread = pthread_self();
-    // struct sched_param is used to store the scheduling priority
-	struct sched_param params;
-	// We'll set the priority to the maximum.
-	params.sched_priority = sched_get_priority_max(SCHED_FIFO) - order;
-
-	// Attempt to set thread real-time priority to the SCHED_FIFO policy
-	ret = pthread_setschedparam(this_thread, SCHED_FIFO, &params);
-	if (ret != 0) {
-		// Print the error
-		cout << "Unsuccessful in setting thread realtime prio" << endl;
-		return;
-	}
-
-	// Now verify the change in thread priority
-    int policy = 0;
-    ret = pthread_getschedparam(this_thread, &policy, &params);
-    if (ret != 0) {
-        cout << "Couldn't retrieve real-time scheduling parameters" << endl;
-        return;
-    }
-
-    // Check the correct policy was applied
-    if(policy != SCHED_FIFO) {
-        cout << "Scheduling is NOT SCHED_FIFO!" << endl;
-    }
-}
-
-//-----------------------------------------------------------------------------------------------------------
-
-
--- a/examples/d-box/prio.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,22 +0,0 @@
-/*
- * prio.h
- *
- *  Created on: May 14, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef PRIO_H_
-#define PRIO_H_
-
-
-#include <pthread.h>
-#include <sched.h>
-#include <iostream>
-
-//-----------------------------------------------------------------------------------------------------------
-// set maximum real-time priority to this thread
-//-----------------------------------------------------------------------------------------------------------
-void set_realtime_priority(int order);
-//-----------------------------------------------------------------------------------------------------------
-
-#endif /* PRIO_H_ */
--- a/examples/d-box/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,870 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: May 28, 2014
- *      Author: Victor Zappi
- */
-
-#include <Bela.h>
-#include <PRU.h>
-
-#include "StatusLED.h"
-#include "config.h"
-#include "OscillatorBank.h"
-#include "FeedbackOscillator.h"
-#include "ADSR.h"
-#include "FIRfilter.h"
-#include <assert.h>
-#include <cmath>
-#include <vector>
-
-#undef DBOX_CAPE_TEST
-
-// Mappings from pin numbers on PCB to actual DAC channels
-// This gives the DAC and ADC connectors the same effective pinout
-// Update June 2016: this is no longer needed in the latest Bela
-// release, but is kept here for convenience: it used to be 
-// 6 4 2 0 1 3 5 7 for the DAC pins
-#define DAC_PIN0	0
-#define DAC_PIN1	1
-#define DAC_PIN2	2
-#define DAC_PIN3	3
-#define DAC_PIN4	4
-#define DAC_PIN5	5
-#define DAC_PIN6	6
-#define DAC_PIN7	7
-
-#define ADC_PIN0	0
-#define ADC_PIN1	1
-#define ADC_PIN2	2
-#define ADC_PIN3	3
-#define ADC_PIN4	4
-#define ADC_PIN5	5
-#define ADC_PIN6	6
-#define ADC_PIN7	7
-
-#define N_OCT		4.0	// maximum number of octaves on sensor 1
-
-extern vector<OscillatorBank*> gOscBanks;
-extern int gCurrentOscBank;
-extern int gNextOscBank;
-extern PRU *gPRU;
-extern StatusLED gStatusLED;
-extern bool gIsLoading;
-extern bool gAudioIn;
-
-float *gOscillatorBuffer1, *gOscillatorBuffer2;
-float *gOscillatorBufferRead, *gOscillatorBufferWrite;
-int gOscillatorBufferReadPointer		= 0;
-int gOscillatorBufferReadCurrentSize	= 0;
-int gOscillatorBufferWriteCurrentSize	= 0;
-bool gOscillatorNeedsRender				= false;
-
-int gMatrixSampleCount = 0;		// How many samples have elapsed on the matrix
-
-// Wavetable which changes in response to an oscillator
-float *gDynamicWavetable;
-int gDynamicWavetableLength;
-bool gDynamicWavetableNeedsRender = false;
-
-// These variables handle the hysteresis oscillator used for setting the playback speed
-bool gSpeedHysteresisOscillatorRising	= false;
-int gSpeedHysteresisLastTrigger			= 0;
-
-// These variables handle the feedback oscillator used for controlling the wavetable
-FeedbackOscillator gFeedbackOscillator;
-float *gFeedbackOscillatorTable;
-int gFeedbackOscillatorTableLength;
-
-// This comes from sensor.cpp where it records the most recent touch location on
-// sensor 0.
-extern float gSensor0LatestTouchPos;
-extern int gSensor0LatestTouchNum;
-float gPitchLatestInput = 0;
-
-extern float gSensor1LatestTouchPos[];
-//extern float gSensor1LatestTouchSizes[];
-extern int gSensor1LatestTouchCount;
-extern int gSensor1LatestTouchIndex;
-int gSensor1LastTouchIndex		= -1;
-int gSensor1InputDelayCounter	= -1;
-int gSensor1InputIndex			= 0;
-float gSensor1MatrixTouchPos[5]	= {0};
-
-// FSR value from matrix input
-extern int gLastFSRValue;
-
-// Loop points from matrix input 4
-const int gLoopPointsInputBufferSize	= 256;
-float gLoopPointsInputBuffer[gLoopPointsInputBufferSize];
-int gLoopPointsInputBufferPointer		= 0;
-float gLoopPointMin = 0, gLoopPointMax	= 0;
-
-// multiplier to activate or mute audio in
-int audioInStatus = 0;
-
-// xenomai timer
-SRTIME prevChangeNs = 0;
-
-// pitch vars
-float octaveSplitter;
-float semitones[((int)N_OCT*12)+1];
-float deltaTouch	= 0;
-float deltaWeightP	= 0.5 / 65536.0;
-float deltaWeightI	= 0.0005 / 65536.0;
-
-// filter vars
-ne10_fir_instance_f32_t filter[2];
-ne10_float32_t *filterIn[2];
-ne10_float32_t *filterOut[2];
-ne10_uint32_t blockSize;
-ne10_float32_t *filterState[2];
-ne10_float32_t prevFiltered[2];
-int filterGain = 80;
-ADSR PeakBurst[2];
-float peak[2];
-float peakThresh = 0.2;
-
-// Tasks for lower-priority calculation
-AuxiliaryTask gMediumPriorityRender, gLowPriorityRender;
-
-
-extern "C" {
-	// Function prototype for ARM assembly implementation of oscillator bank
-	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
-							  int activePartialNum, int lookupTableSize,
-							  float *phases, float *frequencies, float *amplitudes,
-							  float *freqDerivatives, float *ampDerivatives,
-							  float *lookupTable);
-
-	void wavetable_interpolate_neon(int numSamplesIn, int numSamplesOut,
-	                              float *tableIn, float *tableOut);
-}
-
-void wavetable_interpolate(int numSamplesIn, int numSamplesOut,
-                              float *tableIn, float *tableOut,
-                              float *sineTable, float sineMix);
-
-inline float hysteresis_oscillator(float input, float risingThreshold,
-									float fallingThreshold, bool *rising);
-
-void render_medium_prio();
-void render_low_prio();
-
-#ifdef DBOX_CAPE_TEST
-void render_capetest(int numMatrixFrames, int numAudioFrames, float *audioIn, float *audioOut,
-			uint16_t *matrixIn, uint16_t *matrixOut);
-#endif
-
-bool setup(BelaContext *context, void *userData) {
-	int oscBankHopSize = *(int *)userData;
-
-	if(context->analogChannels != 8) {
-		printf("Error: D-Box needs matrix enabled with 8 channels.\n");
-		return false;
-	}
-
-	// Allocate two buffers for rendering oscillator bank samples
-	// One will be used for writing in the background while the other is used for reading
-	// on the audio thread. 8-byte alignment needed for the NEON code.
-	if(posix_memalign((void **)&gOscillatorBuffer1, 8, oscBankHopSize * context->audioChannels * sizeof(float))) {
-		printf("Error allocating render buffers\n");
-		return false;
-	}
-	if(posix_memalign((void **)&gOscillatorBuffer2, 8, oscBankHopSize * context->audioChannels * sizeof(float))) {
-		printf("Error allocating render buffers\n");
-		return false;
-	}
-	gOscillatorBufferWrite	= gOscillatorBuffer1;
-	gOscillatorBufferRead	= gOscillatorBuffer2;
-
-	memset(gOscillatorBuffer1, 0, oscBankHopSize * context->audioChannels * sizeof(float));
-	memset(gOscillatorBuffer2, 0, oscBankHopSize * context->audioChannels * sizeof(float));
-
-	// Initialise the dynamic wavetable used by the oscillator bank
-	// It should match the size of the static one already allocated in the OscillatorBank object
-	// Don't forget a guard point at the end of the table
-	gDynamicWavetableLength = gOscBanks[gCurrentOscBank]->lookupTableSize;
-	if(posix_memalign((void **)&gDynamicWavetable, 8, (gDynamicWavetableLength + 1) * sizeof(float))) {
-		printf("Error allocating wavetable\n");
-		return false;
-	}
-
-	gFeedbackOscillator.initialise(8192, 10.0, context->analogSampleRate);
-
-	for(int n = 0; n < gDynamicWavetableLength + 1; n++)
-		gDynamicWavetable[n] = 0;
-
-	// pitch
-	float midPos		= 0.5;
-	octaveSplitter		= 1.0 / N_OCT;
-	int numOfSemi		= 12*N_OCT;
-	int middleSemitone	= 12*N_OCT/2;
-	int lastSemitone	= middleSemitone+numOfSemi/2;
-	float inc 			= 1.0 / (N_OCT*12.0);
-	int i 				= -1;
-	for(int semi=middleSemitone; semi<=lastSemitone; semi++)
-		semitones[semi] = ( midPos + (++i)*inc) + 0.5;
-	i 					= 0;
-	for(int semi=middleSemitone-1; semi>=0; semi--)
-		semitones[semi] = ( midPos - (++i)*inc) + 0.5;
-
-	if(gAudioIn)
-		audioInStatus = 1;
-
-	// filter
-	blockSize		= context->audioFrames;
-	filterState[0]	= (ne10_float32_t *) NE10_MALLOC ((FILTER_TAP_NUM+blockSize-1) * sizeof (ne10_float32_t));
-	filterState[1]	= (ne10_float32_t *) NE10_MALLOC ((FILTER_TAP_NUM+blockSize-1) * sizeof (ne10_float32_t));
-	filterIn[0]		= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
-	filterIn[1]		= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
-	filterOut[0]	= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
-	filterOut[1]	= (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
-	ne10_fir_init_float(&filter[0], FILTER_TAP_NUM, filterTaps, filterState[0], blockSize);
-	ne10_fir_init_float(&filter[1], FILTER_TAP_NUM, filterTaps, filterState[1], blockSize);
-
-	// peak outputs
-	PeakBurst[0].setAttackRate(.00001 * context->analogSampleRate);
-	PeakBurst[1].setAttackRate(.00001 * context->analogSampleRate);
-	PeakBurst[0].setDecayRate(.5 * context->analogSampleRate);
-	PeakBurst[1].setDecayRate(.5 * context->analogSampleRate);
-	PeakBurst[0].setSustainLevel(0.0);
-	PeakBurst[1].setSustainLevel(0.0);
-
-	// Initialise auxiliary tasks
-	if((gMediumPriorityRender = Bela_createAuxiliaryTask(&render_medium_prio, BELA_AUDIO_PRIORITY - 10, "dbox-calculation-medium")) == 0)
-		return false;
-	if((gLowPriorityRender = Bela_createAuxiliaryTask(&render_low_prio, BELA_AUDIO_PRIORITY - 15, "dbox-calculation-low")) == 0)
-		return false;
-
-	return true;
-}
-
-void render(BelaContext *context, void *userData)
-{
-#ifdef DBOX_CAPE_TEST
-	render_capetest(numMatrixFrames, numAudioFrames, audioIn, audioOut, matrixIn, matrixOut);
-#else
-	if(gOscBanks[gCurrentOscBank]->state==bank_toreset)
-		gOscBanks[gCurrentOscBank]->resetOscillators();
-
-	if(gOscBanks[gCurrentOscBank]->state==bank_playing)
-	{
-		assert(context->audioChannels == 2);
-
-#ifdef OLD_OSCBANK
-		memset(audioOut, 0, numAudioFrames *  * sizeof(float));
-
-		/* Render the oscillator bank. The oscillator bank function is written in NEON assembly
-		 * and it strips out all extra checks, so find out in advance whether we can render a whole
-		 * block or whether the frame will increment in the middle of this buffer.
-		 */
-
-		int framesRemaining = numAudioFrames;
-		float *audioOutWithOffset = audioOut;
-
-		while(framesRemaining > 0) {
-			if(gOscBanks[gCurrentOscBank]->hopCounter >= framesRemaining) {
-				/* More frames left in this hop than we need this time. Render and finish */
-				oscillator_bank_neon(framesRemaining, audioOutWithOffset,
-									 gOscBanks[gCurrentOscBank]->actPartNum, gOscBanks[gCurrentOscBank]->lookupTableSize,
-									 gOscBanks[gCurrentOscBank]->oscillatorPhases, gOscBanks[gCurrentOscBank]->oscillatorNormFrequencies,
-									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudes,
-									 gOscBanks[gCurrentOscBank]->oscillatorNormFreqDerivatives,
-									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudeDerivatives,
-									 gDynamicWavetable/*gOscBanks[gCurrentOscBank]->lookupTable*/);
-				gOscBanks[gCurrentOscBank]->hopCounter -= framesRemaining;
-				if(gOscBanks[gCurrentOscBank]->hopCounter <= 0)
-					gOscBanks[gCurrentOscBank]->nextHop();
-				framesRemaining = 0;
-			}
-			else {
-				/* More frames to render than are left in this hop. Render and decrement the
-				 * number of remaining frames; then advance to the next oscillator frame.
-				 */
-				oscillator_bank_neon(gOscBanks[gCurrentOscBank]->hopCounter, audioOutWithOffset,
-									 gOscBanks[gCurrentOscBank]->actPartNum, gOscBanks[gCurrentOscBank]->lookupTableSize,
-									 gOscBanks[gCurrentOscBank]->oscillatorPhases, gOscBanks[gCurrentOscBank]->oscillatorNormFrequencies,
-									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudes,
-									 gOscBanks[gCurrentOscBank]->oscillatorNormFreqDerivatives,
-									 gOscBanks[gCurrentOscBank]->oscillatorAmplitudeDerivatives,
-									 gDynamicWavetable/*gOscBanks[gCurrentOscBank]->lookupTable*/);
-				framesRemaining -= gOscBanks[gCurrentOscBank]->hopCounter;
-				audioOutWithOffset +=  * gOscBanks[gCurrentOscBank]->hopCounter;
-				gOscBanks[gCurrentOscBank]->sampleCount += gOscBanks[gCurrentOscBank]->hopCounter;
-				gOscBanks[gCurrentOscBank]->nextHop();
-			}
-		}
-#else
-		for(unsigned int n = 0; n < context->audioFrames; n++) {
-			context->audioOut[2*n] 	  = gOscillatorBufferRead[gOscillatorBufferReadPointer++]+context->audioIn[2*n]*audioInStatus;
-			context->audioOut[2*n + 1] = gOscillatorBufferRead[gOscillatorBufferReadPointer++]+context->audioIn[2*n+1]*audioInStatus;
-
-			filterIn[0][n] = fabs(context->audioIn[2*n]);	// rectify for peak detection in 1
-			filterIn[1][n] = fabs(context->audioIn[2*n+1]);	// rectify for peak detection in 2
-
-			/* FIXME why doesn't this work? */
-			/*
-			if(gOscillatorBufferReadPointer == gOscillatorBufferCurrentSize/2) {
-				gOscillatorNeedsRender = true;
-				scheduleAuxiliaryTask(gLowPriorityRender);
-			} */
-
-			if(gOscillatorBufferReadPointer >= gOscillatorBufferReadCurrentSize) {
-				// Finished reading from the buffer: swap to the next buffer
-				if(gOscillatorBufferRead == gOscillatorBuffer1) {
-					gOscillatorBufferRead = gOscillatorBuffer2;
-					gOscillatorBufferWrite = gOscillatorBuffer1;
-				}
-				else {
-					gOscillatorBufferRead = gOscillatorBuffer1;
-					gOscillatorBufferWrite = gOscillatorBuffer2;
-				}
-
-				// New buffer size is whatever finished writing last hop
-				gOscillatorBufferReadCurrentSize = gOscillatorBufferWriteCurrentSize;
-				gOscillatorBufferReadPointer = 0;
-
-				gOscillatorNeedsRender = true;
-				Bela_scheduleAuxiliaryTask(gMediumPriorityRender);
-			}
-		}
-#endif
-	}
-	else
-	{
-		for(unsigned int n = 0; n < context->audioFrames; n++) {
-			context->audioOut[2*n] 	  = context->audioIn[2*n]*audioInStatus;
-			context->audioOut[2*n + 1] = context->audioIn[2*n+1]*audioInStatus;
-
-			filterIn[0][n] = fabs(context->audioIn[2*n]);	// rectify for peak detection in 1
-			filterIn[1][n] = fabs(context->audioIn[2*n+1]);	// rectify for peak detection in 2
-		}
-	}
-
-	// low pass filter audio in 1 and 2 for peak detection
-	ne10_fir_float_neon(&filter[0], filterIn[0], filterOut[0], blockSize);
-	ne10_fir_float_neon(&filter[1], filterIn[1], filterOut[1], blockSize);
-
-	for(unsigned int n = 0; n < context->analogFrames; n++) {
-
-
-		/* Matrix Out 0, In 0
-		 *
-		 * CV loop
-		 * Controls pitch of sound
-		 */
-		float touchPosInt = gSensor0LatestTouchPos;
-		if(touchPosInt < 0) touchPosInt = 0;
-		if(touchPosInt > 1.0) touchPosInt = 1.0;
-		context->analogOut[n*8 + DAC_PIN0] = touchPosInt;
-
-		gPitchLatestInput = context->analogIn[n*8 + ADC_PIN0];
-
-
-		/* Matrix Out 7
-		 *
-		 * Loop feedback with Matrix In 0
-		 * Controls discreet pitch
-		 */
-		float deltaTarget = 0;
-		int semitoneIndex = 0;
-		if(gSensor0LatestTouchNum>0)
-		{
-			// current pitch is gPitchLatestInput, already retrieved
-			semitoneIndex	= ( gPitchLatestInput * 12 * N_OCT  )+0.5;	// closest semitone
-			deltaTarget		= (semitones[semitoneIndex]-gPitchLatestInput);				// delta between pitch and target
-			deltaTouch 		+= deltaTarget*(deltaWeightI);								// update feedback [previous + current]
-		}
-		else
-			deltaTouch = 0;
-
-		float nextOut = touchPosInt  + deltaTarget*deltaWeightP + deltaTouch;			// add feedback to touch -> next out
-		if(nextOut < 0) nextOut = 0;												// clamp
-		if(nextOut > 1.0) nextOut = 1.0;										// clamp
-		context->analogOut[n*8 + DAC_PIN7] = nextOut;										// send next nextOut
-
-
-		/*
-		 * Matrix Out 1, In 1
-		 *
-		 * Hysteresis (comparator) oscillator
-		 * Controls speed of playback
-		 */
-		bool wasRising = gSpeedHysteresisOscillatorRising;
-		context->analogOut[n*8 + DAC_PIN1] = hysteresis_oscillator(context->analogIn[n*8 + ADC_PIN1], 48000.0/65536.0,
-																	16000.0/65536.0, &gSpeedHysteresisOscillatorRising);
-
-		// Find interval of zero crossing
-		if(wasRising && !gSpeedHysteresisOscillatorRising) {
-			int interval = gMatrixSampleCount - gSpeedHysteresisLastTrigger;
-
-			// Interval since last trigger will be the new hop size; calculate to set speed
-			if(interval < 1)
-				interval = 1;
-			//float speed = (float)gOscBanks[gCurrentOscBank]->getHopSize() / (float)interval;
-			float speed = 144.0 / interval;	// Normalise to a fixed expected speed
-			gOscBanks[gCurrentOscBank]->setSpeed(speed);
-
-			gSpeedHysteresisLastTrigger = gMatrixSampleCount;
-		}
-
-		/*
-		 * Matrix Out 2, In 2
-		 *
-		 * Feedback (phase shift) oscillator
-		 * Controls wavetable used for oscillator bank
-		 */
-
-		int tableLength = gFeedbackOscillator.process(context->analogIn[n*8 + ADC_PIN2], &context->analogOut[n*8 + DAC_PIN2]);
-		if(tableLength != 0) {
-			gFeedbackOscillatorTableLength = tableLength;
-			gFeedbackOscillatorTable = gFeedbackOscillator.wavetable();
-			gDynamicWavetableNeedsRender = true;
-			Bela_scheduleAuxiliaryTask(gLowPriorityRender);
-		}
-
-		/*
-		 * Matrix Out 3, In 3
-		 *
-		 * CV loop with delay for time alignment
-		 * Touch positions from sensor 1
-		 * Change every 32 samples (ca. 1.5 ms)
-		 */
-		volatile int touchCount = gSensor1LatestTouchCount;
-		if(touchCount == 0)
-			context->analogOut[n*8 + DAC_PIN3] = 0;
-		else {
-			int touchIndex = (gMatrixSampleCount >> 5) % touchCount;
-			context->analogOut[n*8 + DAC_PIN3] = gSensor1LatestTouchPos[touchIndex] * 56000.0f / 65536.0f;
-			if(touchIndex != gSensor1LastTouchIndex) {
-				// Just changed to a new touch output. Reset the counter.
-				// It will take 2*matrixFrames samples for this output to come back to the
-				// ADC input. But we also want to read near the end of the 32 sample block;
-				// let's say 24 samples into it.
-
-				// FIXME this won't work for p > 2
-				gSensor1InputDelayCounter = 24 + 2*context->analogFrames;
-				gSensor1InputIndex = touchIndex;
-			}
-			gSensor1LastTouchIndex = touchIndex;
-		}
-
-		if(gSensor1InputDelayCounter-- >= 0 && touchCount > 0) {
-			gSensor1MatrixTouchPos[gSensor1InputIndex] = context->analogIn[n*8 + ADC_PIN3];
-		}
-
-		/* Matrix Out 4
-		 *
-		 * Sensor 1 last pos
-		 */
-		touchPosInt = gSensor1LatestTouchPos[gSensor1LatestTouchIndex];
-		if(touchPosInt < 0) touchPosInt = 0;
-		if(touchPosInt > 1.0) touchPosInt = 1.0;
-		context->analogOut[n*8 + DAC_PIN4] = touchPosInt;
-
-		/* Matrix In 4
-		 *
-		 * Loop points selector
-		 */
-		gLoopPointsInputBuffer[gLoopPointsInputBufferPointer++] = context->analogIn[n*8 + ADC_PIN4];
-		if(gLoopPointsInputBufferPointer >= gLoopPointsInputBufferSize) {
-			// Find min and max values
-			float loopMax = 0, loopMin = 1.0;
-			for(int i = 0; i < gLoopPointsInputBufferSize; i++) {
-				if(gLoopPointsInputBuffer[i] < loopMin)
-					loopMin = gLoopPointsInputBuffer[i];
-				if(gLoopPointsInputBuffer[i] > loopMax/* && gLoopPointsInputBuffer[i] != 65535*/)
-					loopMax = gLoopPointsInputBuffer[i];
-			}
-
-			if(loopMin >= loopMax)
-				loopMax = loopMin;
-
-			gLoopPointMax = loopMax;
-			gLoopPointMin = loopMin;
-			gLoopPointsInputBufferPointer = 0;
-		}
-
-		/* Matrix Out 5
-		 *
-		 * Audio In 1 peak detection and peak burst output
-		 */
-
-		filterOut[0][n*2+1]	*= filterGain;
-		float burstOut		= PeakBurst[0].getOutput();
-		if( burstOut < 0.1)
-		{
-			if( (prevFiltered[0]>=peakThresh) && (prevFiltered[0]>=filterOut[0][n*2+1]) )
-			{
-				peak[0] = prevFiltered[0];
-				PeakBurst[0].gate(1);
-			}
-		}
-
-		PeakBurst[0].process(1);
-
-		float convAudio = burstOut*peak[0];
-		context->analogOut[n*8 + DAC_PIN5] = convAudio;
-		prevFiltered[0] = filterOut[0][n*2+1];
-		if(prevFiltered[0]>1)
-			prevFiltered[0] = 1;
-
-		/* Matrix In 5
-		 *
-		 * Dissonance, via changing frequency motion of partials
-		 */
-		float amount = (float)context->analogIn[n*8 + ADC_PIN5];
-		gOscBanks[gCurrentOscBank]->freqMovement = 1.0 - amount;
-
-
-
-
-		/* Matrix Out 6
-		 *
-		 * Audio In 2 peak detection and peak burst output
-		 */
-
-		filterOut[1][n*2+1]	*= filterGain;
-		burstOut			= PeakBurst[1].getOutput();
-		if( burstOut < 0.1)
-		{
-			if( (prevFiltered[1]>=peakThresh) && (prevFiltered[1]>=filterOut[1][n*2+1]) )
-			{
-				peak[1] = prevFiltered[1];
-				PeakBurst[1].gate(1);
-			}
-		}
-
-		PeakBurst[1].process(1);
-
-		convAudio = burstOut*peak[1];
-		context->analogOut[n*8 + DAC_PIN6] = convAudio;
-		prevFiltered[1] = filterOut[1][n*2+1];
-		if(prevFiltered[1]>1)
-			prevFiltered[1] = 1;
-
-		/* Matrix In 6
-		 *
-		 * Sound selector
-		 */
-		if(!gIsLoading) {
-			// Use hysteresis to avoid jumping back and forth between sounds
-			if(gOscBanks.size() > 1) {
-				float input = context->analogIn[n*8 + ADC_PIN6];
-				const float hystValue = 16000.0 / 65536.0;
-
-				float upHysteresisValue = ((gCurrentOscBank + 1) + hystValue) / gOscBanks.size();
-				float downHysteresisValue = (gCurrentOscBank - hystValue) / gOscBanks.size();
-
-				if(input > upHysteresisValue || input < downHysteresisValue) {
-					gNextOscBank = input * gOscBanks.size();
-					if(gNextOscBank < 0)
-						gNextOscBank = 0;
-					if((unsigned)gNextOscBank >= gOscBanks.size())
-						gNextOscBank = gOscBanks.size() - 1;
-				}
-			}
-		}
-
-		/*
-		 * Matrix In 7
-		 *
-		 * FSR from primary touch sensor
-		 * Value ranges from 0-1799
-		 */
-		gLastFSRValue = context->analogIn[n*8 + ADC_PIN7] * 1799.0;
-		//gLastFSRValue = 1799 - context->analogIn[n*8 + ADC_PIN7] * (1799.0 / 65535.0);
-		//dbox_printf("%i\n",gLastFSRValue);
-
-		gMatrixSampleCount++;
-	}
-
-#endif /* DBOX_CAPE_TEST */
-}
-
-// Medium-priority render function used for audio hop calculations
-void render_medium_prio()
-{
-
-	if(gOscillatorNeedsRender) {
-		gOscillatorNeedsRender = false;
-
-		/* Render one frame into the write buffer */
-		memset(gOscillatorBufferWrite, 0, gOscBanks[gCurrentOscBank]->hopCounter * 2 * sizeof(float)); /* assumes 2 audio channels */
-
-		oscillator_bank_neon(gOscBanks[gCurrentOscBank]->hopCounter, gOscillatorBufferWrite,
-							 gOscBanks[gCurrentOscBank]->actPartNum, gOscBanks[gCurrentOscBank]->lookupTableSize,
-							 gOscBanks[gCurrentOscBank]->oscillatorPhases, gOscBanks[gCurrentOscBank]->oscillatorNormFrequencies,
-							 gOscBanks[gCurrentOscBank]->oscillatorAmplitudes,
-							 gOscBanks[gCurrentOscBank]->oscillatorNormFreqDerivatives,
-							 gOscBanks[gCurrentOscBank]->oscillatorAmplitudeDerivatives,
-							 /*gOscBanks[gCurrentOscBank]->lookupTable*/gDynamicWavetable);
-
-		gOscillatorBufferWriteCurrentSize = gOscBanks[gCurrentOscBank]->hopCounter * 2;
-
-		/* Update the pitch right before the hop
-		 * Total CV range +/- N_OCT octaves
-		 */
-		float pitch = (float)gPitchLatestInput / octaveSplitter - N_OCT/2;
-		//gOscBanks[gCurrentOscBank]->pitchMultiplier = powf(2.0f, pitch);
-		gOscBanks[gCurrentOscBank]->pitchMultiplier = pow(2.0f, pitch);
-
-#ifdef FIXME_LATER // This doesn't work very well yet
-		gOscBanks[gCurrentOscBank]->filterNum = gSensor1LatestTouchCount;
-		float freqScaler = gOscBanks[gCurrentOscBank]->getFrequencyScaler();
-		for(int i=0; i < gOscBanks[gCurrentOscBank]->filterNum; i++)
-		{
-			// touch pos is linear but freqs are log
-			gOscBanks[gCurrentOscBank]->filterFreqs[i] = ((expf(gSensor1MatrixTouchPos[i]*4)-1)/(expf(4)-1))*gOscBanks[gCurrentOscBank]->filterMaxF*freqScaler;
-			gOscBanks[gCurrentOscBank]->filterQ[i] = gSensor1LatestTouchSizes[i];
-			if(gOscBanks[gCurrentOscBank]->filterFreqs[i]>500*freqScaler)
-				gOscBanks[gCurrentOscBank]->filterPadding[i] = 1+100000*( (gOscBanks[gCurrentOscBank]->filterFreqs[i]-500*freqScaler)/(gOscBanks[gCurrentOscBank]->filterMaxF-500)*freqScaler );
-			else
-				gOscBanks[gCurrentOscBank]->filterPadding[i] = 1;
-		}
-#endif
-
-		RTIME ticks		= rt_timer_read();
-		SRTIME ns		= rt_timer_tsc2ns(ticks);
-		SRTIME delta 	= ns-prevChangeNs;
-
-		// switch to next bank cannot be too frequent, to avoid seg fault! [for example sef fault happens when removing both VDD and GND from breadboard]
-		if(gNextOscBank != gCurrentOscBank && delta>100000000) {
-
-			/*printf("ticks %llu\n", (unsigned long long)ticks);
-			printf("ns %llu\n", (unsigned long long)ns);
-			printf("prevChangeNs %llu\n", (unsigned long long)prevChangeNs);
-			printf("-------------------------->%llud\n", (unsigned long long)(ns-prevChangeNs));*/
-
-			prevChangeNs = ns;
-			dbox_printf("Changing to bank %d...\n", gNextOscBank);
-			if(gOscBanks[gCurrentOscBank]->state==bank_playing){
-				gOscBanks[gCurrentOscBank]->stop();
-			}
-
-			gCurrentOscBank = gNextOscBank;
-			gOscBanks[gCurrentOscBank]->hopNumTh = 0;
-		}
-		else {
-			/* Advance to the next oscillator frame */
-			gOscBanks[gCurrentOscBank]->nextHop();
-		}
-	}
-}
-
-// Lower-priority render function which performs matrix calculations
-// State should be transferred in via global variables
-void render_low_prio()
-{
-	gPRU->setGPIOTestPin();
-	if(gDynamicWavetableNeedsRender) {
-		// Find amplitude of wavetable
-		float meanAmplitude = 0;
-		float sineMix;
-
-		for(int i = 0; i < gFeedbackOscillatorTableLength; i++) {
-			//meanAmplitude += fabsf(gFeedbackOscillatorTable[i]);
-			meanAmplitude += fabs(gFeedbackOscillatorTable[i]);
-		}
-		meanAmplitude /= (float)gFeedbackOscillatorTableLength;
-
-		if(meanAmplitude > 0.35)
-			sineMix = 0;
-		else
-			sineMix = (.35 - meanAmplitude) / .35;
-
-		//dbox_printf("amp %f mix %f\n", meanAmplitude, sineMix);
-
-		// Copy to main wavetable
-		wavetable_interpolate(gFeedbackOscillatorTableLength, gDynamicWavetableLength,
-				gFeedbackOscillatorTable, gDynamicWavetable,
-				gOscBanks[gCurrentOscBank]->lookupTable, sineMix);
-	}
-
-	if(gLoopPointMin >= 60000.0/65536.0 && gLoopPointMax >= 60000.0/65536.0) {
-		// KLUDGE!
-		if(gCurrentOscBank == 0)
-			gOscBanks[gCurrentOscBank]->setLoopHops(50, ((float)gOscBanks[gCurrentOscBank]->getLastHop() * 0.6) - 1);
-		else
-			gOscBanks[gCurrentOscBank]->setLoopHops(5, ((float)gOscBanks[gCurrentOscBank]->getLastHop() * 0.7) - 1);
-	}
-	else {
-		float normLoopPointMin = (float)gLoopPointMin * gOscBanks[gCurrentOscBank]->getLastHop();
-		float normLoopPointMax = (float)gLoopPointMax * gOscBanks[gCurrentOscBank]->getLastHop();
-
-		int intLoopPointMin = normLoopPointMin;
-		if(intLoopPointMin < 1)
-			intLoopPointMin = 1;
-		int intLoopPointMax = normLoopPointMax;
-		if(intLoopPointMax <= intLoopPointMin)
-			intLoopPointMax = intLoopPointMin + 1;
-		if(intLoopPointMax > gOscBanks[gCurrentOscBank]->getLastHop() - 1)
-			intLoopPointMax =  gOscBanks[gCurrentOscBank]->getLastHop() - 1;
-
-		//dbox_printf("Loop points %d-%d / %d-%d\n", gLoopPointMin, gLoopPointMax, intLoopPointMin, intLoopPointMax);
-
-		/* WORKS, jsut need to fix the glitch when jumps!
-		 * *int currentHop = gOscBanks[gCurrentOscBank]->getCurrentHop();
-		if(currentHop < intLoopPointMin -1 )
-			gOscBanks[gCurrentOscBank]->setJumpHop(intLoopPointMin + 1);
-		else if(currentHop > intLoopPointMax + 1)
-			gOscBanks[gCurrentOscBank]->setJumpHop(intLoopPointMax - 1);*/
-		gOscBanks[gCurrentOscBank]->setLoopHops(intLoopPointMin, intLoopPointMax);
-	}
-
-	if(gIsLoading)
-		gStatusLED.blink(25, 75);	// Blink quickly until load finished
-	else
-		gStatusLED.blink(250 / gOscBanks[gCurrentOscBank]->getSpeed(), 250 / gOscBanks[gCurrentOscBank]->getSpeed());
-	gPRU->clearGPIOTestPin();
-
-//	static int counter = 32;
-//	if(--counter == 0) {
-//		for(int i = 0; i < gLoopPointsInputBufferSize; i++) {
-//			dbox_printf("%d ", gLoopPointsInputBuffer[i]);
-//			if(i % 32 == 31)
-//				dbox_printf("\n");
-//		}
-//		dbox_printf("\n\n");
-//		counter = 32;
-//	}
-
-	//dbox_printf("min %d max %d\n", gLoopPointMin, gLoopPointMax);
-}
-
-// Clean up at the end of render
-void cleanup(BelaContext *context, void *userData)
-{
-	free(gOscillatorBuffer1);
-	free(gOscillatorBuffer2);
-	free(gDynamicWavetable);
-}
-
-// Interpolate one wavetable into another. The output size
-// does not include the guard point at the end which will be identical
-// to the first point
-void wavetable_interpolate(int numSamplesIn, int numSamplesOut,
-                           float *tableIn, float *tableOut,
-                           float *sineTable, float sineMix)
-{
-	float fractionalScaler = (float)numSamplesIn / (float)numSamplesOut;
-
-	for(int k = 0; k < numSamplesOut; k++) {
-		float fractionalIndex = (float) k * fractionalScaler;
-		//int sB = (int)floorf(fractionalIndex);
-		int sB = (int)floor(fractionalIndex);
-		int sA = sB + 1;
-		if(sA >= numSamplesIn)
-			sA = 0;
-		float fraction = fractionalIndex - sB;
-		tableOut[k] = fraction * tableIn[sA] + (1.0f - fraction) * tableIn[sB];
-		tableOut[k] = sineMix * sineTable[k] + (1.0 - sineMix) * tableOut[k];
-	}
-
-	tableOut[numSamplesOut] = tableOut[0];
-}
-
-// Create a hysteresis oscillator with a matrix input and output
-inline float hysteresis_oscillator(float input, float risingThreshold, float fallingThreshold, bool *rising)
-{
-	float value;
-
-	if(*rising) {
-		if(input > risingThreshold) {
-			*rising = false;
-			value = 0;
-		}
-		else
-			value = 1.0;
-	}
-	else {
-		if(input < fallingThreshold) {
-			*rising = true;
-			value = 1.0;
-		}
-		else
-			value = 0;
-	}
-
-	return value;
-}
-
-#ifdef DBOX_CAPE_TEST
-// Test the functionality of the D-Box cape by checking each input and output
-// Loopback cable from ADC to DAC needed
-void render_capetest(int numMatrixFrames, int numAudioFrames, float *audioIn, float *audioOut,
-			uint16_t *matrixIn, uint16_t *matrixOut)
-{
-	static float phase = 0.0;
-	static int sampleCounter = 0;
-	static int invertChannel = 0;
-
-	// Play a sine wave on the audio output
-	for(int n = 0; n < numAudioFrames; n++) {
-		audioOut[2*n] = audioOut[2*n + 1] = 0.5*sinf(phase);
-		phase += 2.0 * M_PI * 440.0 / 44100.0;
-		if(phase >= 2.0 * M_PI)
-			phase -= 2.0 * M_PI;
-	}
-
-	for(int n = 0; n < numMatrixFrames; n++) {
-		// Change outputs every 512 samples
-		if(sampleCounter < 512) {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel)
-					matrixOut[n*8 + k] = 50000;
-				else
-					matrixOut[n*8 + k] = 0;
-			}
-		}
-		else {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel)
-					matrixOut[n*8 + k] = 0;
-				else
-					matrixOut[n*8 + k] = 50000;
-			}
-		}
-
-		// Read after 256 samples: input should be low
-		if(sampleCounter == 256) {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel) {
-					if(matrixIn[n*8 + k] < 50000) {
-						dbox_printf("FAIL channel %d -- output HIGH input %d (inverted)\n", k, matrixIn[n*8 + k]);
-					}
-				}
-				else {
-					if(matrixIn[n*8 + k] > 2048) {
-						dbox_printf("FAIL channel %d -- output LOW input %d\n", k, matrixIn[n*8 + k]);
-					}
-				}
-			}
-		}
-		else if(sampleCounter == 768) {
-			for(int k = 0; k < 8; k++) {
-				if(k == invertChannel) {
-					if(matrixIn[n*8 + k] > 2048) {
-						dbox_printf("FAIL channel %d -- output LOW input %d (inverted)\n", k, matrixIn[n*8 + k]);
-					}
-				}
-				else {
-					if(matrixIn[n*8 + k] < 50000) {
-						dbox_printf("FAIL channel %d -- output HIGH input %d\n", k, matrixIn[n*8 + k]);
-					}
-				}
-			}
-		}
-
-		if(++sampleCounter >= 1024) {
-			sampleCounter = 0;
-			invertChannel++;
-			if(invertChannel >= 8)
-				invertChannel = 0;
-		}
-	}
-}
-#endif
-
-
--- a/examples/d-box/sensors.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,642 +0,0 @@
-/*
- * sensors.cpp
- *
- *  Created on: May 28, 2014
- *      Author: Victor Zappi
- */
-
-#include <stdio.h>
-#include <pthread.h>
-#include <unistd.h>
-#include <math.h>
-#include <vector>
-#include "prio.h"
-#include "sensors.h"
-#include "OscillatorBank.h"
-#include "DboxSensors.h"
-
-
-//----------------------------------------
-// main extern variables
-//----------------------------------------
-extern vector<OscillatorBank*> gOscBanks;
-extern int gCurrentOscBank;
-extern int gNextOscBank;
-extern int gShouldStop;
-extern int gVerbose;
-
-float gSensor0LatestTouchPos = 0;	// most recent pitch touch location [0-1] on sensor 0, used by render.cpp
-int gSensor0LatestTouchNum	 = 0;	// most recent num of touches on sensor 0, used by render.cpp
-float gSensor1LatestTouchPos[5];	// most recent touche locations on sensor 1, used by render.cpp
-//float gSensor1LatestTouchSizes[5];
-int gSensor1LatestTouchCount;		// most recent number touches on sensor 1, used by render.cpp
-int gSensor1LatestTouchIndex = 0;	// index of last touch in gSensor1LatestTouchPos[5], used by render.cpp
-int gLastFSRValue = 1799;			// most recent fsr value, used by render.cpp
-
-
-DboxSensors Sensors;
-
-
-//----------------------------------------
-// var shared with logger
-//----------------------------------------
-int s0TouchNum	 	= 0;
-float s0Touches_[MAX_TOUCHES];
-float s0Size_[MAX_TOUCHES];
-int s0LastIndex;
-
-int s1TouchNum	 	= 0;
-float s1Touches_[MAX_TOUCHES];
-float s1Size_[MAX_TOUCHES];
-int s1LastIndex;
-
-int fsr				= 1799;
-
-
-
-using namespace std;
-
-int initSensorLoop(int sensorAddress0, int sensorAddress1, int sensorType)
-{
-	int tk0_bus			= 1;
-	int tk0_address		= sensorAddress0;
-	int tk1_bus			= 1;
-	int tk1_address		= sensorAddress1;
-	int tk_file			= 0;
-	int fsr_max			= 1799;
-	int fsr_pinNum		= 4;
-
-	if(gVerbose==1)
-		cout << "---------------->Init Control Thread" << endl;
-
-	if(Sensors.initSensors(tk0_bus, tk0_address, tk1_bus, tk1_address, tk_file, fsr_pinNum, fsr_max, sensorType)>0)
-	{
-		gShouldStop = 1;
-		cout << "control cannot start" << endl;
-		return -1;
-	}
-
-	for(int i=0; i<MAX_TOUCHES; i++)
-	{
-		s0Touches_[i]	= 0.0;
-		s0Size_[i]		= 0.0;
-
-		s1Touches_[i]	= 0.0;
-		s1Size_[i]	= 0.0;
-	}
-
-	return 0;
-}
-
-void sensorLoop(void *)
-{
-	timeval start, end;
-	unsigned long elapsedTime;
-	//float touchSize		= 0;	// once used for timbre
-
-
-
-	float *s0Touches;
-	float *s0Size;
-	int s0PrevTouchNum 	= 0;
-	int s0SortedTouchIndices[MAX_TOUCHES];
-	float s0SortedTouches[MAX_TOUCHES];
-	float s0PrevSortedTouches[MAX_TOUCHES];
-
-	float *s1Touches;
-	float *s1Size;
-	int s1PrevTouchNum 	= 0;
-	int s1SortedTouchIndices[MAX_TOUCHES];
-	float s1SortedTouches[MAX_TOUCHES];
-	float s1PrevSortedTouches[MAX_TOUCHES];
-
-	float freqScaler	= 0;
-	int fsrMin			= 0;//50; // was 200
-	int fsrMax			= 1799;//1300; // was 800
-	float vel			= 0;
-	float prevVel		= 0;
-	float filterMaxF	= 0;
-	if(gVerbose==1)
-		dbox_printf("__________set Control Thread priority\n");
-
-	if(gVerbose==1)
-		dbox_printf("_________________Control Thread!\n");
-
-	// get freq scaler, cos freqs must be scaled according to the wavetable used in the oscillator bank
-	freqScaler 		= gOscBanks[gCurrentOscBank]->getFrequencyScaler();
-	filterMaxF		= gOscBanks[gCurrentOscBank]->filterMaxF;
-
-	// init time vals
-	gettimeofday(&start, NULL);
-
-	// here we go, sensor loop until the end of the application
-	while(!gShouldStop)
-	{
-		gettimeofday(&end, NULL);
-		elapsedTime = ( (end.tv_sec*1000000+end.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
-		if( elapsedTime<4000 )
-			usleep(4000-elapsedTime);
-		else
-			dbox_printf("%d\n", (int)elapsedTime); // this print happens when something's gone bad...
-
-		if(Sensors.readSensors()==0)
-		{
-			s0TouchNum	= Sensors.getTKTouchCount(0);
-			s0Touches	= Sensors.getTKXPositions(0);
-			s0Size 		= Sensors.getTKTouchSize(0);
-
-			s1TouchNum	= Sensors.getTKTouchCount(1);
-			s1Touches	= Sensors.getTKXPositions(1);
-			s1Size 		= Sensors.getTKTouchSize(1);
-
-			for(int i=0; i<MAX_TOUCHES; i++)
-			{
-				s0Touches_[i]	= s0Touches[i];
-				s0Size_[i]		= s0Size[i];
-
-				s1Touches_[i]	= s1Touches[i];
-				s1Size_[i]		= s1Size[i];
-			}
-
-			gSensor0LatestTouchNum	= s0TouchNum;
-			if(s0TouchNum > 0)
-			{
-				//-----------------------------------------------------------------------------------
-				// timbre, speed and  pitch
-				//touchSize	 = 0;	\\ once used for timbre
-
-				// if we have a number of touches different from previous round, track their order of arrival [calculated using distance comparison]
-				if(s0PrevTouchNum!=s0TouchNum)
-				{
-					float distances[MAX_TOUCHES*(MAX_TOUCHES-1)]; // maximum number of current+previous touches between rounds with different num of touches
-					int ids[MAX_TOUCHES*(MAX_TOUCHES-1)];
-					// calculate all distance permutations between previous and current touches
-					for(int i=0; i<s0TouchNum; i++)
-					{
-						for(int p=0; p<s0PrevTouchNum; p++)
-						{
-							int index			= i*s0PrevTouchNum+p;	// permutation id [says between which touches we are calculating distance]
-							distances[index]	= fabs(s0Touches[i]-s0PrevSortedTouches[p]);
-							ids[index]			= index;
-							if(index>0)
-							{
-								// sort, from min to max distance
-								float tmp;
-								while(distances[index]<distances[index-1])
-								{
-									tmp				= ids[index-1];
-									ids[index-1]	= ids[index];
-									ids[index]		= tmp;
-
-									tmp				= distances[index-1];
-									distances[index-1] = distances[index];
-									distances[index] = tmp;
-
-									index--;
-
-									if(index == 0)
-										break;
-								}
-							}
-						}
-					}
-
-					int sorted = 0;
-					bool currAssigned[MAX_TOUCHES] = {false};
-					bool prevAssigned[MAX_TOUCHES] = {false};
-
-					// track touches assigning index according to shortest distance
-					for(int i=0; i<s0TouchNum*s0PrevTouchNum; i++)
-					{
-						int currentIndex	= ids[i]/s0PrevTouchNum;
-						int prevIndex		= ids[i]%s0PrevTouchNum;
-						// avoid double assignment
-						if(!currAssigned[currentIndex] && !prevAssigned[prevIndex])
-						{
-							currAssigned[currentIndex]	= true;
-							prevAssigned[prevIndex]		= true;
-							s0SortedTouchIndices[currentIndex] = prevIndex;
-							sorted++;
-						}
-					}
-					// we still have to assign a free index to new touches
-					if(s0PrevTouchNum<s0TouchNum)
-					{
-						for(int i=0; i<s0TouchNum; i++)
-						{
-							if(!currAssigned[i])
-								s0SortedTouchIndices[i] = sorted++; // assign next free index
-
-							// update tracked value
-							s0SortedTouches[s0SortedTouchIndices[i]] = s0Touches[i];
-							s0PrevSortedTouches[i]			         = s0SortedTouches[i];
-							if(s0SortedTouchIndices[i]==s0TouchNum-1)
-								s0LastIndex = i;
-
-							// accumulate sizes for timbre
-							//touchSize += s0Size[i];
-						}
-					}
-					else // some touches have disappeared...
-					{
-						// ...we have to shift all indices...
-						for(int i=s0PrevTouchNum-1; i>=0; i--)
-						{
-							if(!prevAssigned[i])
-							{
-								for(int j=0; j<s0TouchNum; j++)
-								{
-									// ...only if touches that disappeared were before the current one
-									if(s0SortedTouchIndices[j]>i)
-										s0SortedTouchIndices[j]--;
-								}
-							}
-						}
-						// done! now update
-						for(int i=0; i<s0TouchNum; i++)
-						{
-							// update tracked value
-							s0SortedTouches[s0SortedTouchIndices[i]] = s0Touches[i];
-							s0PrevSortedTouches[i]			         = s0SortedTouches[i];
-							if(s0SortedTouchIndices[i]==s0TouchNum-1)
-								s0LastIndex = i;
-
-							// accumulate sizes for timbre
-							//touchSize += s0Size[i];
-						}
-					}
-				}
-				else // nothing's changed since last round
-				{
-					for(int i=0; i<s0TouchNum; i++)
-					{
-						// update tracked value
-						s0SortedTouches[s0SortedTouchIndices[i]] = s0Touches[i];
-						s0PrevSortedTouches[i]			         = s0SortedTouches[i];
-
-						// accumulate sizes for timbre
-						//touchSize += s0Size[i];
-					}
-				}
-
-				if(s0TouchNum == 0)
-					s0LastIndex = -1;
-
-				// timbre
-				//touchSize = (touchSize > 0.7) ? 1 : touchSize/0.7;
-				//gOscBanks[gCurrentOscBank]->hopNumTh = log((1-touchSize)+1)/log(2)*20000;
-				//gOscBanks[gCurrentOscBank]->hopNumTh = 0;
-
-
-				// pitch, controlled by last touch
-				//prevTouchPos 				= touch[touchIndex];
-				//touchPos	 			 	= (s0SortedTouches[s0TouchNum-1]-0.5)/0.5;	// from [0,1] to [-1,1]
-				gSensor0LatestTouchPos      = s0SortedTouches[s0TouchNum-1];
-				//touchPos					= s0Touches[0];
-				//gOscBanks[gCurrentOscBank]->pitchMultiplier 	= pow(2, touchPos);
-				//-----------------------------------------------------------------------------------
-
-
-
-				//-----------------------------------------------------------------------------------
-				// note on
-				//if(s0PrevTouchNum == 0)
-				//	gOscBanks[gCurrentOscBank]->play();
-				// fsr = Sensors.getFSRVAlue();
-				fsr = gLastFSRValue;
-				//dbox_printf("fsr: %d\n", fsr);
-				if(!gOscBanks[gCurrentOscBank]->note)
-				{
-					vel = fsr;
-					vel /= (float)(fsrMax-fsrMin);
-
-					vel = 1-vel;
-					dbox_printf("Attack vel: %f\n", vel);
-					gOscBanks[gCurrentOscBank]->play(vel);
-					prevVel = vel;
-				}
-				else if(gOscBanks[gCurrentOscBank]->getEnvelopeState() != env_release)
-				{
-					fsr = (fsr > fsrMax) ? fsrMax : fsr;
-					vel = (fsr < fsrMin) ? fsrMin : fsr;
-					vel -= fsrMin;
-					vel /= (float)(fsrMax-fsrMin);
-					vel = 1-vel;
-					if(vel > prevVel)
-					{
-						gOscBanks[gCurrentOscBank]->afterTouch(vel);
-						prevVel = vel;
-					}
-				}
-				//-----------------------------------------------------------------------------------
-			}
-			else
-			{
-				//prevFsr = 1799;
-				//prevTouchPos = -1;
-				//-----------------------------------------------------------------------------------
-				// note off
-				if(s0PrevTouchNum > 0)
-				{
-					if(gOscBanks[gCurrentOscBank]->state==bank_playing)
-						gOscBanks[gCurrentOscBank]->stop();
-				}
-				//-----------------------------------------------------------------------------------
-			}
-
-
-
-			// sensor 2
-			//-----------------------------------------------------------------------------------
-			//filter - calculated even when no touches on first sensor, to filter also release tail
-			gOscBanks[gCurrentOscBank]->filterNum	= s1TouchNum;
-
-			gSensor1LatestTouchCount = gOscBanks[gCurrentOscBank]->filterNum;
-			for(int i = 0; i < gSensor1LatestTouchCount; i++) {
-				gSensor1LatestTouchPos[i] = s1Touches[i];
-				//gSensor1LatestTouchSizes[i] = s1Size[i];
-			}
-
-/*			for(int i=0; i<gOscBanks[gCurrentOscBank]->filterNum; i++)
-			{
-				// touch pos is linear but freqs are log
-				gOscBanks[gCurrentOscBank]->filterFreqs[i] = ((exp(s0Touches[i]*4)-1)/(exp(4)-1))*filterMaxF*freqScaler;
-				//gOscBanks[gCurrentOscBank]->filterQ[i] = size[i]*5*(1+touch[i]*1000)*freqScaler;
-				gOscBanks[gCurrentOscBank]->filterQ[i] = s0Size[i];
-				if(gOscBanks[gCurrentOscBank]->filterFreqs[i]>500*freqScaler)
-					gOscBanks[gCurrentOscBank]->filterPadding[i] = 1+100000*( (gOscBanks[gCurrentOscBank]->filterFreqs[i]-500*freqScaler)/(filterMaxF-500)*freqScaler );
-				else
-					gOscBanks[gCurrentOscBank]->filterPadding[i] = 1;
-			}*/
-
-			// each touch on sensor 2 is a notch filter, whose Q is determined by touch size
-			for(int i=0; i<gOscBanks[gCurrentOscBank]->filterNum; i++)
-			{
-				// map touch pos [which is linear] on freqs exponentially
-				float freq = ((exp(s1Touches[i]*4)-1)/EXP_DENOM)*filterMaxF;
-				gOscBanks[gCurrentOscBank]->filterFreqs[i] = freq*freqScaler;
-				// also size is mapped exponentially on Q
-				float siz = (exp(s1Size[i])-1)/1.71828;
-				gOscBanks[gCurrentOscBank]->filterQ[i] = siz*( (filterMaxF-freq)/filterMaxF * 0.9 + 0.1 );	// size weight on Q decreases with frequency
-			}
-			//-----------------------------------------------------------------------------------
-
-
-
-			//-----------------------------------------------------------------------------------
-			// sort touches on sensor 2
-			if(s1TouchNum > 0)
-			{
-				// if we have a number of touches different from previous round, track their order of arrival [calculated using distance comparison]
-				if(s1PrevTouchNum!=s1TouchNum)
-				{
-					float distances[MAX_TOUCHES*(MAX_TOUCHES-1)]; // maximum number of current+previous touches between rounds with different num of touches
-					int ids[MAX_TOUCHES*(MAX_TOUCHES-1)];
-					// calculate all distance permutations between previous and current touches
-					for(int i=0; i<s1TouchNum; i++)
-					{
-						for(int p=0; p<s1PrevTouchNum; p++)
-						{
-							int index 			= i*s1PrevTouchNum+p;	// permutation id [says between which touches we are calculating distance]
-							distances[index]	= fabs(s1Touches[i]-s1PrevSortedTouches[p]);
-							ids[index]			= index;
-							if(index>0)
-							{
-								// sort, from min to max distance
-								float tmp;
-								while(distances[index]<distances[index-1])
-								{
-									tmp 				= ids[index-1];
-									ids[index-1] 		= ids[index];
-									ids[index] 			= tmp;
-
-									tmp					= distances[index-1];
-									distances[index-1]	= distances[index];
-									distances[index] 	= tmp;
-
-									index--;
-
-									if(index == 0)
-										break;
-								}
-							}
-						}
-					}
-
-					int sorted = 0;
-					bool currAssigned[MAX_TOUCHES] = {false};
-					bool prevAssigned[MAX_TOUCHES] = {false};
-
-					// track touches assigning index according to shortest distance
-					for(int i=0; i<s1TouchNum*s1PrevTouchNum; i++)
-					{
-						int currentIndex	= ids[i]/s1PrevTouchNum;
-						int prevIndex		= ids[i]%s1PrevTouchNum;
-						// avoid double assignment
-						if(!currAssigned[currentIndex] && !prevAssigned[prevIndex])
-						{
-							currAssigned[currentIndex]			= true;
-							prevAssigned[prevIndex]				= true;
-							s1SortedTouchIndices[currentIndex] = prevIndex;
-							sorted++;
-						}
-					}
-					// we still have to assign a free index to new touches
-					if(s1PrevTouchNum<s1TouchNum)
-					{
-						for(int i=0; i<s1TouchNum; i++)
-						{
-							if(!currAssigned[i])
-								s1SortedTouchIndices[i] = sorted++; // assign next free index
-
-							// update tracked value
-							s1SortedTouches[s1SortedTouchIndices[i]] = s1Touches[i];
-							s1PrevSortedTouches[i]			       	 = s1SortedTouches[i];
-							if(s1SortedTouchIndices[i]==s1TouchNum-1)
-								s1LastIndex = i;
-						}
-					}
-					else // some touches have disappeared...
-					{
-						// ...we have to shift all indices...
-						for(int i=s1PrevTouchNum-1; i>=0; i--)
-						{
-							if(!prevAssigned[i])
-							{
-								for(int j=0; j<s1TouchNum; j++)
-								{
-									// ...only if touches that disappeared were before the current one
-									if(s1SortedTouchIndices[j]>i)
-										s1SortedTouchIndices[j]--;
-								}
-							}
-						}
-						// done! now update
-						for(int i=0; i<s1TouchNum; i++)
-						{
-							// update tracked value
-							s1SortedTouches[s1SortedTouchIndices[i]] = s1Touches[i];
-							s1PrevSortedTouches[i]			       	 = s1SortedTouches[i];
-							if(s1SortedTouchIndices[i]==s1TouchNum-1)
-								s1LastIndex = i;
-						}
-					}
-				}
-				else // nothing's changed since last round
-				{
-					for(int i=0; i<s1TouchNum; i++)
-					{
-						// update tracked value
-						s1SortedTouches[s1SortedTouchIndices[i]] = s1Touches[i];
-						s1PrevSortedTouches[i]			       	 = s1SortedTouches[i];
-					}
-				}
-			}
-
-			if(s1TouchNum > 0)
-			{
-				gSensor1LatestTouchIndex = s1LastIndex;
-			}
-			else
-				s1LastIndex = -1;
-
-/*			dbox_printf("-----------------------------\nnum: %d, latest: %d\n", s1TouchNum, gSensor1LatestTouchIndex);
-			for(int i=0; i<s1TouchNum; i++)
-				dbox_printf("\t%f\n", gSensor1LatestTouchPos[i]);
-			dbox_printf("------\n");
-			for(int i=0; i<s1TouchNum; i++)
-				dbox_printf("\t%f\n", s1SortedTouches[i]);*/
-
-
-
-			// update variables for both sensors
-			s0PrevTouchNum	= s0TouchNum;
-			s1PrevTouchNum	= s1TouchNum;
-		}
-		else
-			dbox_printf("Come on instrument!\n");	//break
-
-		gettimeofday(&start, NULL);
-	}
-
-	dbox_printf("sensor thread ended\n");
-}
-
-void *keyboardLoop(void *)
-{
-	if(gVerbose==1)
-		cout << "_________________Keyboard Control Thread!" << endl;
-
-	char keyStroke = '.';
-	cout << "Press q to quit." << endl;
-
-	float speed;
-
-	do
-	{
-		keyStroke =	getchar();
-		while(getchar()!='\n'); // to read the first stroke
-
-		switch (keyStroke)
-		{
-			//----------------------------------------------------------------------------
-			case 'a':
-				gOscBanks[gCurrentOscBank]->hopNumTh = 0;
-				gOscBanks[gCurrentOscBank]->play(1);
-				//cout << "Note on" << endl;
-				break;
-			case 's':
-				if(gOscBanks[gCurrentOscBank]->state==bank_playing)
-				{
-					gOscBanks[gCurrentOscBank]->stop();
-					//cout << "Note off" << endl;
-				}
-				break;
-			//----------------------------------------------------------------------------
-			case '[':
-				gOscBanks[gCurrentOscBank]->freqMovement-=0.05;
-				if(gOscBanks[gCurrentOscBank]->freqMovement<0)
-					gOscBanks[gCurrentOscBank]->freqMovement = 0;
-				//cout << "gOscBanks[gCurrentOscBank]->FreqMov: " << gOscBanks[gCurrentOscBank]->freqMovement << endl;
-				break;
-			case ']':
-				gOscBanks[gCurrentOscBank]->freqMovement+=0.05;
-				if(gOscBanks[gCurrentOscBank]->freqMovement>1)
-					gOscBanks[gCurrentOscBank]->freqMovement = 1;
-				//cout << "gOscBanks[gCurrentOscBank]->FreqMov: " << gOscBanks[gCurrentOscBank]->freqMovement << endl;
-				break;
-			//----------------------------------------------------------------------------
-			case '<':
-				speed = gOscBanks[gCurrentOscBank]->getSpeed() - 0.1 ;
-				gOscBanks[gCurrentOscBank]->setSpeed(speed);
-				dbox_printf("Speed: %f\n", speed);
-
-				break;
-			case '>':
-				speed = gOscBanks[gCurrentOscBank]->getSpeed() + 0.1 ;
-				gOscBanks[gCurrentOscBank]->setSpeed(speed);
-				dbox_printf("Speed: %f\n", speed);
-				break;
-			case '0':
-				speed = 0.1;
-				gOscBanks[gCurrentOscBank]->setSpeed(speed);
-				dbox_printf("Speed: %f\n", speed);
-				break;
-			case '1':
-				speed = 0.5;
-				gOscBanks[gCurrentOscBank]->setSpeed(speed);
-				dbox_printf("Speed: %f\n", speed);
-				break;
-			case '2':
-				speed = 1;
-				gOscBanks[gCurrentOscBank]->setSpeed(speed);
-				dbox_printf("Speed: %f\n", speed);
-				break;
-			case '3':
-				speed = 2;
-				gOscBanks[gCurrentOscBank]->setSpeed(speed);
-				dbox_printf("Speed: %f\n", speed);
-				break;
-			case '4':
-				speed = 3;
-				gOscBanks[gCurrentOscBank]->setSpeed(speed);
-				dbox_printf("Speed: %f\n", speed);
-				break;
-			//----------------------------------------------------------------------------
-			case 'z':
-				gOscBanks[gCurrentOscBank]->setJumpHop(0);
-				break;
-			case 'x':
-				gOscBanks[gCurrentOscBank]->setJumpHop(100);
-				break;
-			case 'c':
-				gOscBanks[gCurrentOscBank]->setJumpHop(600);
-				break;
-			case 'v':
-				gOscBanks[gCurrentOscBank]->setJumpHop(1100);
-				break;
-			case 'b':
-				gOscBanks[gCurrentOscBank]->setJumpHop(2000);
-				break;
-			case 'n':
-				gOscBanks[gCurrentOscBank]->setJumpHop(gOscBanks[gCurrentOscBank]->getLastHop());
-				break;
-			//----------------------------------------------------------------------------
-			case 'q':
-				gShouldStop = true;
-				break;
-			case 'o':
-				gNextOscBank = (gCurrentOscBank + 1) % gOscBanks.size();
-				break;
-			default:
-				break;
-			//----------------------------------------------------------------------------
-		}
-		usleep(1000); /* Wait 1ms to avoid checking too quickly */
-	}
-	while (keyStroke!='q');
-
-	cout << "keyboard thread ended" << endl;
-
-	return (void *)0;
-}
--- a/examples/d-box/sensors.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,19 +0,0 @@
-/*
- * sensors.h
- *
- *  Created on: May 28, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef SENSORS_H_
-#define SENSORS_H_
-
-#include "config.h"
-
-int initSensorLoop(int sensorAddress0, int sensorAddress1, int sensorType);
-
-void sensorLoop(void *);
-void *keyboardLoop(void *);
-
-
-#endif /* SENSORS_H_ */
--- a/examples/d-box/spear_parser.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,641 +0,0 @@
-/*
- * spear_parser.cpp v1.2
- *
- *  Created on: May 6, 2014
- *      Author: Victor Zappi
- */
-
-#include "spear_parser.h"
-
-using namespace std;
-
-//#define DO_CHECKS
-
-//------------------------------------------------------------------------------------------------
-// partials
-//------------------------------------------------------------------------------------------------
-
-Partials::Partials()
-{
-	partialFrequencies	= NULL;
-//	partialAmplitudes	= NULL;
-//	partialNumFrames	= NULL;
-//	partialStartSample	= NULL;
-//	partialEndSample	= NULL;
-//	partialCurrentFrame	= NULL;
-//	partialFreqDelta	= NULL;
-//	partialAmpDelta	= NULL;
-
-
-	activePartialNum	= NULL;
-//	activePartials		= NULL;
-
-	currentSample = -1;
-}
-
-Partials::~Partials()
-{
-	if(partialFrequencies != NULL)			// check on one is enough
-	{
-		if(partialFrequencies[0] != NULL)	// check on one is enough
-		{
-			for(unsigned int i=0; i<parNum; i++)
-			{
-				delete[] partialFrequencies[i];
-				delete[] partialAmplitudes[i];
-				delete[] partialFreqDelta[i];
-				delete[] partialAmpDelta[i];
-
-			}
-		}
-
-		delete[] partialFrequencies;
-		delete[] partialAmplitudes;
-		delete[] partialNumFrames;
-		delete[] partialFreqDelta;
-		delete[] partialAmpDelta;
-		delete[] partialFreqMean;
-	}
-
-	if(activePartialNum != NULL)
-	{
-		for(unsigned int i=0; i<hopNum+1; i++)
-			delete[] activePartials[i];
-
-		delete[] activePartialNum;
-		delete[] activePartials ;
-	}
-}
-
-void Partials::init(int parN, int hopS, bool isDBX)
-{
-	if(!isDBX)
-	{
-		parNum	= parN;
-		hopSize	= hopS;
-
-		partialFrequencies	= new float *[parNum];
-		partialAmplitudes	= new float *[parNum];
-		partialNumFrames	= new unsigned int[parNum];
-		partialStartFrame	= new unsigned int[parNum];
-		partialStartSample	= new unsigned int[parNum];
-		partialEndSample	= new unsigned int[parNum];
-		partialFreqDelta	= new float *[parNum];
-		partialAmpDelta		= new float *[parNum];
-		partialFreqMean		= new float[parNum];
-
-
-
-		// init in one shot
-		fill(partialFreqMean, partialFreqMean+parNum, 0);			// mean is zero
-
-		partialFrequencies[0] 	= NULL;								// for free check
-	}
-	else
-	{
-		parNum	= parN;
-		hopSize	= hopS;
-
-		partialFrequencies	= new float *[parNum];
-		partialAmplitudes	= new float *[parNum];
-		partialNumFrames	= new unsigned int[parNum];
-		partialStartFrame	= new unsigned int[parNum];
-		partialFreqDelta	= new float *[parNum];
-		partialAmpDelta		= new float *[parNum];
-		partialFreqMean		= new float[parNum];
-
-		partialFrequencies[0] 	= NULL;								// for free check
-	}
-}
-
-
-
-void Partials::update(int parIndex, int frameNum)
-{
-	partialFrequencies[parIndex] = new float[frameNum];
-	partialAmplitudes[parIndex]	 = new float[frameNum];
-	partialFreqDelta[parIndex]	 = new float[frameNum];
-	partialAmpDelta[parIndex]	 = new float[frameNum];
-
-	fill(partialFreqDelta[parIndex], partialFreqDelta[parIndex]+frameNum, 99999.0);	// in the end, only the last one will have 99999
-	fill(partialAmpDelta[parIndex], partialAmpDelta[parIndex]+frameNum, 99999.0);	// in the end, only the last one will have 99999
-}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-//------------------------------------------------------------------------------------------------
-// spear parser
-//------------------------------------------------------------------------------------------------
-Spear_parser::Spear_parser()
-{
-	// some default values
-	hopSize			= -1;
-	fileSampleRate	= -1;
-}
-
-Spear_parser::~Spear_parser()
-{
-}
-
-void Spear_parser::calculateHopSize(char *filename)
-{
-	int index 		= 0;
-	bool prevWas_ 	= false;
-	bool found_h	= false;
-	int n 			= 0;
-
-	hopSize 		= 0;
-
-	do
-	{
-		// check if '_'
-		if(filename[index] == '_')
-			prevWas_ = true;
-		else if( (filename[index] == 'h') && prevWas_) // if it is not, but it is 'h' and previous was '_', found "_h"!
-		{
-			found_h = true;
-			while(filename[index] != '\0')
-			{
-				index++;
-				if( (filename[index] == '.') || (filename[index] == '_'))
-					break;
-				else // i am not checking if char are digits...!
-				{
-					n = filename[index];
-					hopSize =  hopSize*10+(n-48);
-				}
-			}
-		}
-		else	// else, nothing
-			prevWas_ = false;
-		index++;
-	}
-	while( (filename[index] != '\0') && !found_h );
-
-	if( !found_h || (hopSize<1) )
-		hopSize = 551;	// default val
-
-}
-
-
-bool Spear_parser::parser(char *filename, int hopsize, int samplerate)
-{
-	string name = string(filename);
-	int len		= name.length();
-	// invoke correct parser according to the type of file...just checking the extension, crude but functional
-	if( (name[len-4]=='.') && (name[len-3]=='d') && (name[len-2]=='b') && (name[len-1]=='x') )
-		return DBXparser(filename, samplerate);				// .dbox
-	else
-		return TXTparser(filename, hopSize, samplerate);	// .txt, or whatever
-}
-
-
-bool Spear_parser::DBXparser(char *filename, int samplerate)
-{
-	fileSampleRate 	= samplerate;
-
-	// working vars
-	int parNum		= 0;	// total num of partials
-	int hopNum		= 0;	// total num of hops
-
-	//----------------------------------------------------------------------------------------
-	// open a file
-	ifstream fin;
-	fin.open(filename, ios::in | ios::binary);
-	if (!fin.good())
-	{
-		cout << "Parser Error: file not found" << endl;	// exit if file not found
-		return false;
-	}
-
-	gettimeofday(&start, NULL);
-	//----------------------------------------------------------------------------------------
-	// general data
-
-	// look for partial count
-	fin.read((char *) &parNum, sizeof(int));
-	partials.parNum 		= parNum;
-
-	// look for hop count
-	fin.read((char *) &hopNum, sizeof(int));
-	partials.setHopNum(hopNum);
-
-	// look for hop size
-	fin.read((char *) &hopSize, sizeof(int));
-	partials.hopSize 		= hopSize;		// it's handy for both classes to know it
-
-	// init partials data structure
-	partials.init(parNum, hopSize, true);
-
-	// look for max active par num
-	fin.read((char *) &(partials.maxActiveParNum), sizeof(int));
-
-
-
-	// partial data
-
-	// start frame of each partial
-	fin.read((char *) partials.partialStartFrame, sizeof(int)*parNum);
-
-	// num of frames of each partial
-	fin.read((char *) partials.partialNumFrames, sizeof(int)*parNum);
-
-	// frequency mean of each partial
-	fin.read((char *) partials.partialFreqMean, sizeof(int)*parNum);
-
-	for(int par=0; par<parNum; par++)
-	{
-		int frameNum = partials.partialNumFrames[par];
-		partials.update(par, frameNum);
-		fin.read((char *)partials.partialAmplitudes[par], sizeof(float)*frameNum);		// amplitude of each partial in each frame
-		fin.read((char *)partials.partialFrequencies[par], sizeof(float)*frameNum);		// frequency of each partial in each frame
-		fin.read((char *)partials.partialAmpDelta[par], sizeof(float)*frameNum);			// amplitude delta of each partial in each frame
-		fin.read((char *)partials.partialFreqDelta[par], sizeof(float)*frameNum);			// frequency delta of each partial in each frame
-	}
-
-
-
-
-	// frame data
-
-	// number of active partial per each frame
-	fin.read((char *) partials.activePartialNum, sizeof(short)*(hopNum+1));
-	// init array
-	for(int frame=0; frame<hopNum+1; frame++)
-	{
-		partials.activePartials[frame] = new unsigned int[partials.activePartialNum[frame]];
-		fin.read((char *)partials.activePartials[frame], sizeof(int)*partials.activePartialNum[frame]);			// active partials per each frame
-	}
-
-
-
-
-
-	gettimeofday(&stop, NULL);
-	parserT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
-
-
-	printf("\n-----------------------\n");
-	printf("\nFile: %s\n", filename);
-	printf("\n-----------------------\n");
-	printf("Profiler\n");
-	printf("-----------------------\n");
-	printf("File parser:\t\t\t%lu usec\n", parserT);
-	printf("\n\nTotal:\t\t%lu usec\n", parserT);
-	printf("-----------------------\n");
-
-	fin.close();
-
-	return true;
-}
-
-
-
-
-bool Spear_parser::TXTparser(char *filename, int hopsize, int samplerate)
-{
-	hopSize 		= hopsize;
-	fileSampleRate 	= samplerate;
-	if(hopsize<0)
-	{
-		gettimeofday(&start, NULL);
-		calculateHopSize(filename);
-		gettimeofday(&stop, NULL);
-		hopSizeT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
-	}
-	else
-		hopSizeT = 0;
-
-	calculateDeltaTime();
-
-	// working vars
-	char * token;			// where to save single figures from file
-	string s		= "";	// where to save lines from file
-	int parNum		= 0;	// total num of partials
-	int parIndex	= -1;	// index of current partial
-	int frameNum	= 0;	// total num of frames
-	int frameIndex	= -1;	// index of current frame
-	int startSample	= -1;	// sample value for first frame of partials
-	int endSample	= -1;	// sample value for last frame of partials
-	int maxSample	= 0;	// to calculate total number of hops in file
-	int missSampCnt = 0;	// number of mising samples
-	double freq		= 0;	// to calculate frequency delta
-	double prevFreq	= 0;	// to calculate frequency delta
-	double amp		= 0;	// to calculate amplitude delta
-	double prevAmp	= 0;	// to calculate amplitude delta
-
-
-	//----------------------------------------------------------------------------------------
-	// open a file
-	ifstream fin;
-	fin.open(filename);
-	if (!fin.good())
-	{
-		cout << "Parser Error: file not found" << endl;	// exit if file not found
-		return false;
-	}
-
-	gettimeofday(&start, NULL);
-	//----------------------------------------------------------------------------------------
-	// init partials data structure
-	getline(fin, s);
-	getline(fin, s);
-	getline(fin, s);	// third line is the first we are interested into
-
-	// look for partial count
-	token = strtok((char *)s.c_str(), " ");
-	// check if first token is there
-	if(token)
-	{
-		token = strtok(0, " ");
-		// check if second token is there
-		if(token)
-			parNum = atoi(token);
-		#ifdef DO_CHECKS
-		else
-		{
-			cout << "Parser Error: partial count not found, bad file format" << endl;	// exit if value not found
-			return false;
-		}
-		#endif
-	}
-	#ifdef DO_CHECKS
-	else
-	{
-		cout << "Parser Error: partial count not found, bad file format" << endl;		// exit if value not found
-		return false;
-	}
-	#endif
-	// from now on we take for granted that format is correct
-
-	// init partials data structure
-	partials.init(parNum, hopSize);
-
-	//----------------------------------------------------------------------------------------
-	// fill in partials data structure
-	getline(fin, s);		// get rid of intro line "partials-data"
-	getline(fin, s);		// first important line
-
-	while (!fin.eof())
-	{
-		//-------------------------------------
-		// partial specific info
-		token		= strtok((char *)s.c_str(), " ");
-		parIndex	= atoi(token);						// partial index
-
-		token		= strtok(0, " ");					// num of frames, not used, cos we will do linear interpolation for missing frames
-//		frameNum	= atoi(token);
-//		partials.partialNumFrames[parIndex]	= frameNum;
-
-		token		= strtok(0, " ");					// time of first frame, still char *
-		startSample = fromTimeToSamples(atof(token));	// convert time to samples
-		partials.partialStartSample[parIndex]	= startSample;
-
-		token		= strtok(0, " ");					// time of last frame, still char *
-		endSample	= fromTimeToSamples(atof(token)); 	// convert time to samples
-		partials.partialEndSample[parIndex]		= endSample;
-
-		frameNum	= ((endSample-startSample)/hopSize) + 1;	// num of frames, including missing consecutive ones [+1 one cos we count frames, not hops]
-		partials.partialNumFrames[parIndex]		= frameNum;
-
-
-		// check if this is the highest sample value so far
-		if(endSample > maxSample)
-			maxSample = endSample;
-
-		// update data structure
-		partials.update(parIndex, frameNum);
-
-
-		//-------------------------------------
-		// frames
-		getline(fin, s);
-		token		= strtok((char *)s.c_str(), " ");	// frame time
-		frameIndex	= -1;
-
-		// unroll first iteration, so that in the following loop we save the check on the last frame to calculate increments
-		if(token)						// all frames data are on one line, in groups of 3 entries
-		{
-			frameIndex++;
-
-			endSample	= fromTimeToSamples(atof(token));
-
-			token		= strtok(0, " ");	// frame frequency
-			prevFreq	= atof(token);
-			partials.partialFrequencies[parIndex][frameIndex]	= (float)prevFreq;
-			partials.partialFreqMean[parIndex] 					+= prevFreq;		// for frequency mean
-
-			token	 	= strtok(0, " ");	// frame amplitude
-			prevAmp  	= atof(token);
-			partials.partialAmplitudes[parIndex][frameIndex]	= (float)prevAmp;
-
-			token 		= strtok(0, " ");	// next frame frequency, to be checked
-		}
-
-		// here the loop starts
-		while(token)						// all frames data are on one line, in groups of 3 entries
-		{
-			frameIndex++;
-			missSampCnt 	= 0;
-
-			startSample		= fromTimeToSamples(atof(token));
-
-			token			= strtok(0, " ");	// frame frequency
-			freq			= atof(token);
-
-			token			= strtok(0, " ");	// frame amplitude
-			amp				= atof(token);
-			// now we know all about the current frame, but we want to know if some frames are missing between this and the last one
-
-			// while current frame sample is farther than one hopsize...
-			while(startSample > endSample+hopSize)
-			{
-				missSampCnt++;				// ...one sample is missing
-				endSample += hopSize;		// move to next hop
-			}
-
-			// if frames are missing do interpolation and update indices
-			if(missSampCnt>0)
-				startSample = interpolateSamples(parIndex, &frameIndex, missSampCnt, endSample+hopSize, freq, amp, &prevFreq, &prevAmp);
-
-			partials.partialFrequencies[parIndex][frameIndex]	= (float)freq;
-			partials.partialFreqMean[parIndex] 					+= freq;			// for frequency mean
-			partials.setFreqDelta(parIndex, frameIndex-1, (freq-prevFreq)/hopSize);	// freq delta between prev and current frame
-			prevFreq 	= freq;
-
-			partials.partialAmplitudes[parIndex][frameIndex]	= (float)amp;
-			partials.setAmpDelta(parIndex, frameIndex-1, (amp-prevAmp)/hopSize);	// amp delta between prev and current frame
-			prevAmp		= amp;
-
-			endSample	= startSample;
-			token		= strtok(0, " ");	// next frame frequency, to be checked
-		}
-		#ifdef DO_CHECKS
-		if(frameIndex != (frameNum-1))
-		{
-			cout << "Parser Error: frame count mismatch on partial " << parIndex << ", bad file format"  << endl;	// exit if mismatch
-			cout << "frameIndex: " << frameIndex << endl;
-			cout << "frameNum: " << frameNum << endl;
-			return false;
-		}
-		#endif
-
-		partials.partialFreqMean[parIndex] /= partials.partialNumFrames[parIndex];									// frequency mean
-
-		getline(fin, s);					// next partial line, to check
-	}
-	#ifdef DO_CHECKS
-	if(parIndex != (parNum-1))
-	{
-		cout << "Parser Error: partial count mismatch, bad file format"  << endl;									// exit if mismatch
-		return false;
-	}
-	#endif
-
-	partials.setHopNum(maxSample/hopSize);
-
-	gettimeofday(&stop, NULL);
-	parserT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
-
-	gettimeofday(&start, NULL);
-	staticCalculations();
-	gettimeofday(&stop, NULL);
-	staticT = ( (stop.tv_sec*1000000+stop.tv_usec) - (start.tv_sec*1000000+start.tv_usec) );
-
-	fin.close();
-
-
-	printf("\n-----------------------\n");
-	printf("\nFile: %s\n", filename);
-	printf("\n-----------------------\n");
-	printf("Profiler\n");
-	printf("-----------------------\n");
-	printf("Hop size parser:\t\t%lu usec\n", hopSizeT);
-	printf("File parser:\t\t\t%lu usec\n", parserT);
-	printf("Static calculations:\t\t%lu usec\n", staticT);
-	printf("\n\nTotal:\t\t%lu usec\n", hopSizeT+parserT+staticT);
-	printf("-----------------------\n");
-
-	return true;
-}
-
-
-int Spear_parser::interpolateSamples(int parIndex, int *frameIndex, int missCnt, int nextSample, double nextFreq, double nextAmp, double *prevFreq, double *prevAmp)
-{
-	int frame			= *frameIndex;						// current frame index
-	int sample			= nextSample - (hopSize*(missCnt)); // move from next real frame sample to first missing frame sample
-	double freq			= *prevFreq;						// freq of the prev real frame
-	double freqStep		= (nextFreq-*prevFreq)/(missCnt+1);	// fixed freq step between hops, for missing frames [linear interpolation]
-	double deltaFreq	= freqStep/hopSize;					// fixed hop freq step in samples
-	double amp			= *prevAmp;							// same for amp...
-	double ampStep		= (nextAmp-*prevAmp)/(missCnt+1);
-	double deltaAmp		= ampStep/hopSize;
-
-	// for each missing frame
-	for(int i=0; i<missCnt; i++)
-	{
-		// calculate values for current missing frame
-		freq	+= freqStep;
-		amp		+= ampStep;
-		// save values
-		partials.partialFrequencies[parIndex][frame]	= freq;
-		partials.partialAmplitudes[parIndex][frame]		= amp;
-		partials.partialFreqMean[parIndex]				+= freq;	// for frequency mean
-		// set deltas of previous frame [real or missing]
-		partials.setFreqDelta(parIndex, frame-1, deltaFreq);
-		partials.setAmpDelta(parIndex, frame-1, deltaAmp);
-		// move to next frame [missing or real]
-		sample += hopSize;
-		frame++;
-	}
-
-	// update global values
-	*frameIndex	= frame;
-	*prevFreq	= freq;
-	*prevAmp	= amp;
-
-	return sample;	// return the frame sample of the next real frame
-}
-
-
-
-// for each frame, statically calculate:
-// - which partial is active [and the total num of active partials]
-// - at which local frame each partial is
-void Spear_parser::staticCalculations()
-{
-	partials.maxActiveParNum = 0;				// init to find maximum
-
-	unsigned short *indices	= new unsigned short[partials.parNum];	// temp array to store up to the maximum num of active partial indices
-	unsigned int activeCnt	= 0;						// counts the num of active partials in each frame
-
-	unsigned int frameSample = 0;						// current frame in samples
-
-	char *partialStarted = new char [partials.parNum];	// index of the last local frame found per each partial
-	fill(partialStarted, partialStarted+partials.parNum, 0);
-
-	for(unsigned int i=0; i<partials.hopNum+1; i++)		// for each frame [not hops, this explains the +1]
-	{
-		//partials.localPartialFrames[i] = new int[partials.parNum];	// init all local frames to -1
-		//fill(partials.localPartialFrames[i], partials.localPartialFrames[i]+partials.parNum, -1);
-
-		frameSample = i*hopSize;	// current frame, expressed in samples
-		activeCnt	  = 0;			// reset a each frame
-
-		for(unsigned int j=0; j<partials.parNum; j++)	// for each partial
-		{
-			// check if inside active time region [expressed in samples]
-			if( (frameSample>=partials.partialStartSample[j]) && (frameSample<partials.partialEndSample[j]) )	// frame sample not equal to end sample, this filters out last frames and partials with one frame only
-			{
-				// activity
-				indices[activeCnt] = j;	// save active index
-				activeCnt++;			// increase counter
-
-				// partial local frames
-				if(partialStarted[j]==0)	// this partial has just started, so current local frame is first frame
-				{
-					partialStarted[j] 		 		= 1;
-					partials.partialStartFrame[j]	= i;	// here is the number of the first frame
-				}
-			}
-		}
-
-		// activity
-		partials.activePartialNum[i] = activeCnt;							// save number of active partials for this frame
-		partials.activePartials[i]	 = new unsigned int[activeCnt];					// set correct size to save all indices
-
-		// look for maximum number of active partials at the same time
-		if(activeCnt > partials.maxActiveParNum)
-			partials.maxActiveParNum = activeCnt;
-
-		// copy indices
-		for(unsigned int k=0; k<activeCnt; k++)
-			partials.activePartials[i][k] = indices[k];
-	}
-
-	delete[] indices;
-	delete[] partialStarted;
-
-	delete[] partials.partialStartSample;
-	delete[] partials.partialEndSample;
-}
-
-
-
--- a/examples/d-box/spear_parser.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,188 +0,0 @@
-/*
- * spear_parser.h v1.2
- *
- *  Created on: May 6, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef SPEAR_PARSER_H_
-#define SPEAR_PARSER_H_
-
-#include <iostream>
-#include <fstream>
-#include <cstring>
-#include <string>
-#include <stdlib.h>		// atoi, atof
-#include <math.h>
-#include <algorithm>	// std::fill
-
-#include <sys/time.h>
-
-using namespace std;
-
-
-//------------------------------------------------------------------------------------------------
-// partials
-//------------------------------------------------------------------------------------------------
-
-class Spear_parser; // for class friendship
-
-class Partials
-{
-	friend class Spear_parser;
-	friend class Dbox_parser;
-
-public:
-	int **partialSamples;				// sample at which each frame is
-	float **partialFrequencies;			// frequencies at each frame
-	float **partialAmplitudes;			// amplitudes at each frame
-	unsigned int *partialNumFrames;		// Length of each partial in frames
-	unsigned int *partialStartFrame;	// frame at which each partial begins
-	float **partialFreqDelta;			// constant frequency slope for each partial in each frame interval
-	float **partialAmpDelta;			// constant amplitude slope for each partial in each frame interval
-	float *partialFreqMean;				// frequency mean for each partial, over all its frames
-
-	unsigned short *activePartialNum;	// num of each active partial at each frame
-	unsigned int **activePartials;		// indices of all active partials at each frame
-
-
-	int getPartialNum();
-	int getHopNum();
-	int getMaxActivePartialNum();
-
-private:
-	Partials();
-	~Partials();
-
-	unsigned int *partialStartSample;	// sample at which each partial begins
-	unsigned int *partialEndSample;		// sample at which each partial ends [sample gap between 2 consecutive frames can be an integer multiple of hopSize]
-	unsigned int parNum;
-	unsigned int currentSample;
-	unsigned int hopSize;
-	unsigned int hopNum;
-	unsigned int maxActiveParNum;
-
-	void init(int parNum, int hopSize, bool isDBX=false);
-	void update(int parIndex, int frameNum);
-	void setFreqDelta(int parIndex, int frameNum, double delta);
-	void setAmpDelta(int parIndex, int frameNum, double delta);
-	void setHopNum(int hopNum);
-};
-
-inline int Partials::getPartialNum()
-{
-	return parNum;
-}
-
-inline void Partials::setHopNum(int hopN)
-{
-	hopNum = hopN;
-
-	// prepare data structures
-	activePartialNum 	= new unsigned short[hopNum+1];	// +1 cos total num of frames = num of hops+1
-	activePartials	 	= new unsigned int *[hopNum+1];
-}
-
-// useful to increase current sample using a modulo on the total number of samples [easy to be deduced from the total num or hops]
-inline int Partials::getHopNum()
-{
-	return hopNum;
-}
-
-inline void Partials::setFreqDelta(int parIndex, int frameNum, double delta)
-{
-	partialFreqDelta[parIndex][frameNum] = delta;
-}
-
-inline void Partials::setAmpDelta(int parIndex, int frameNum, double delta)
-{
-	partialAmpDelta[parIndex][frameNum] = delta;
-}
-
-inline int Partials::getMaxActivePartialNum()
-{
-	return maxActiveParNum;
-}
-
-
-
-
-
-
-
-//------------------------------------------------------------------------------------------------
-// spear parser
-//------------------------------------------------------------------------------------------------
-
-class Spear_parser
-{
-public:
-	Spear_parser();
-	~Spear_parser();
-
-	Partials partials;
-
-	bool parseFile(string filename, int hopsize=-1, int samplerate = 44100);
-	bool parseFile(char *filename, int hopsize=-1, int samplerate = 44100);
-	int getHopSize();
-	int getFileSampleRate();
-	double getDeltaTime();
-
-private:
-
-	int hopSize;
-	int fileSampleRate;
-	double deltaTime;	// min time gap between consecutive frames
-
-	timeval start, stop;
-	unsigned long hopSizeT, parserT, staticT;
-
-	void calculateDeltaTime();
-	void calculateHopSize(char *filename);
-	bool parser(char *filename, int hopsize=-1, int samplerate=44100);
-	bool DBXparser(char *filename, int samplerate=44100);
-	bool TXTparser(char *filename, int hopsize=-1, int samplerate=44100);
-	int fromTimeToSamples(float time);
-	int interpolateSamples(int parIndex, int *frameIndex, int missCnt, int nextSample,
-							double nextFreq, double nextAmp, double *prevFreq, double *prevAmp);
-	void staticCalculations();
-
-};
-
-inline bool Spear_parser::parseFile(string filename, int hopsize, int samplerate)
-{
-	return parser((char *)filename.c_str(), hopsize, samplerate);
-}
-
-inline bool Spear_parser::parseFile(char *filename, int hopsize, int samplerate)
-{
-	return parser(filename, hopsize, samplerate);
-}
-
-inline void Spear_parser::calculateDeltaTime()
-{
-	deltaTime = (double)hopSize/ (double)fileSampleRate;
-}
-
-// each time value in the file is rounded, and 2 consecutive frames can differ of a time gap = i*deltaTime, where i is a positive integer
-inline int Spear_parser::fromTimeToSamples(float time)
-{
-	return round(time/deltaTime)*hopSize;	// round is necessary since in the file log time values are rounded, so they do not apparently look like integer multiples of deltaTime
-}
-
-inline int Spear_parser::getHopSize()
-{
-	return hopSize;
-}
-
-inline int Spear_parser::getFileSampleRate()
-{
-	return fileSampleRate;
-}
-
-inline double Spear_parser::getDeltaTime()
-{
-	return deltaTime;
-}
-
-#endif /* SPEAR_PARSER_H_ */
--- a/examples/filter_FIR/FIRfilter.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,51 +0,0 @@
-/*
- * FIRfilter.h
- *
- *  Created on: Aug 5, 2014
- *      Author: Victor Zappi and Andrew McPherson
- */
-
-#ifndef FIRFILTER_H_
-#define FIRFILTER_H_
-
-
-#include <ne10/NE10.h>
-
-#define FILTER_TAP_NUM 31
-
-// Coefficients for FIR High Pass Filter at 3 KHz
-ne10_float32_t filterTaps[FILTER_TAP_NUM] = {
-		-0.000055,
-		0.000318,
-		0.001401,
-		0.003333,
-		0.005827,
-		0.007995,
-		0.008335,
-		0.004991,
-		-0.003764,
-		-0.018906,
-		-0.040112,
-		-0.065486,
-		-0.091722,
-		-0.114710,
-		-0.130454,
-		0.863946,
-		-0.130454,
-		-0.114710,
-		-0.091722,
-		-0.065486,
-		-0.040112,
-		-0.018906,
-		-0.003764,
-		0.004991,
-		0.008335,
-		0.007995,
-		0.005827,
-		0.003333,
-		0.001401,
-		0.000318,
-		-0.000055
-};
-
-#endif /* FIRFILTER_H_ */
--- a/examples/filter_FIR/SampleData.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,19 +0,0 @@
-/*
- * SampleData.h
- *
- *  Created on: Nov 5, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef SAMPLEDATA_H_
-#define SAMPLEDATA_H_
-
-// User defined structure to pass between main and rendere complex data retrieved from file
-struct SampleData {
-	float *samples;	// Samples in file
-	int sampleLen;	// Total nume of samples
-};
-
-
-
-#endif /* SAMPLEDATA_H_ */
Binary file examples/filter_FIR/longsample.wav has changed
--- a/examples/filter_FIR/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,180 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: Andrew McPherson and Victor Zappi
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <string>
-#include <getopt.h>
-#include <sndfile.h>				// to load audio files
-
-#include <Bela.h>
-#include "SampleData.h"
-
-using namespace std;
-
-// Load samples from file
-int initFile(string file, SampleData *smp)//float *& smp)
-{
-	SNDFILE *sndfile ;
-	SF_INFO sfinfo ;
-
-	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
-		cout << "Couldn't open file " << file << endl;
-		return 1;
-	}
-
-	int numChan = sfinfo.channels;
-	if(numChan != 1)
-	{
-		cout << "Error: " << file << " is not a mono file" << endl;
-		return 1;
-	}
-
-	smp->sampleLen = sfinfo.frames * numChan;
-	smp->samples = new float[smp->sampleLen];
-	if(smp == NULL){
-		cout << "Could not allocate buffer" << endl;
-		return 1;
-	}
-
-	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
-	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
-
-	// Pad with zeros in case we couldn't read whole file
-	for(int k = readcount; k <smp->sampleLen; k++)
-		smp->samples[k] = 0;
-
-	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
-		double	scale ;
-		int 	m ;
-
-		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
-		if (scale < 1e-10)
-			scale = 1.0 ;
-		else
-			scale = 32700.0 / scale ;
-		cout << "File samples scale = " << scale << endl;
-
-		for (m = 0; m < smp->sampleLen; m++)
-			smp->samples[m] *= scale;
-	}
-
-	sf_close(sndfile);
-
-	return 0;
-}
-
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	//rt_task_delete ((RT_TASK *) &gTriggerSamplesTask);
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --file [-f] filename:    Name of the file to load (default is \"longsample.wav\")\n";
-	cerr << "   --help [-h]:             Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	string fileName;			// Name of the sample to load
-
-	SampleData sampleData;		// User define structure to pass data retrieved from file to render function
-	sampleData.samples = 0;
-	sampleData.sampleLen = -1;
-
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"file", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				fileName = string((char *)optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	if(fileName.empty()){
-		fileName = "filter/longsample.wav";
-	}
-
-	if(settings.verbose) {
-		cout << "Loading file " << fileName << endl;
-	}
-
-	// Load file
-	if(initFile(fileName, &sampleData) != 0)
-	{
-		cout << "Error: unable to load samples " << endl;
-		return -1;
-	}
-
-	if(settings.verbose)
-		cout << "File contains " << sampleData.sampleLen << " samples" << endl;
-
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &sampleData) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
-
--- a/examples/filter_FIR/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,180 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
-*/
-
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: Andrew McPherson and Victor Zappi
- */
-
-/**
-\example 4_filter_FIR
-
-Finite Impulse Response Filter
-------------------------------
-
-This is an example of a finite impulse response filter implementation.
-*/
-
-#define ENABLE_NE10_FIR_FLOAT_NEON	// Define needed for Ne10 library
-
-#include <Bela.h>
-#include <cmath>
-#include <ne10/NE10.h>					// neon library
-#include "SampleData.h"
-#include "FIRfilter.h"
-
-SampleData gSampleData;	// User defined structure to get complex data from main
-int gReadPtr;			// Position of last read sample from file
-
-// filter vars
-ne10_fir_instance_f32_t gFIRfilter;
-ne10_float32_t *gFIRfilterIn;
-ne10_float32_t *gFIRfilterOut;
-ne10_uint32_t blockSize;
-ne10_float32_t *gFIRfilterState;
-
-void initialise_filter(BelaContext *context);
-
-// Task for handling the update of the frequencies using the matrix
-AuxiliaryTask gTriggerSamplesTask;
-
-bool initialise_trigger();
-void trigger_samples();
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-
-	// Retrieve a parameter passed in from the initAudio() call
-	gSampleData = *(SampleData *)userData;
-
-	gReadPtr = -1;
-
-	initialise_filter(context);
-
-	// Initialise auxiliary tasks
-	if(!initialise_trigger())
-		return false;
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		float in = 0;
-
-		// If triggered...
-		if(gReadPtr != -1)
-			in += gSampleData.samples[gReadPtr++];	// ...read each sample...
-
-		if(gReadPtr >= gSampleData.sampleLen)
-			gReadPtr = -1;
-
-		gFIRfilterIn[n] = in;
-	}
-
-	ne10_fir_float_neon(&gFIRfilter, gFIRfilterIn, gFIRfilterOut, blockSize);
-
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
-				context->audioOut[n * context->audioChannels + channel] = gFIRfilterOut[n];	// ...and put it in both left and right channel
-	}
-
-
-	// Request that the lower-priority task run at next opportunity
-	Bela_scheduleAuxiliaryTask(gTriggerSamplesTask);
-}
-
-// Initialise NE10 data structures to define FIR filter
-
-void initialise_filter(BelaContext *context)
-{
-	blockSize = context->audioFrames;
-	gFIRfilterState	= (ne10_float32_t *) NE10_MALLOC ((FILTER_TAP_NUM+blockSize-1) * sizeof (ne10_float32_t));
-	gFIRfilterIn = (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
-	gFIRfilterOut = (ne10_float32_t *) NE10_MALLOC (blockSize * sizeof (ne10_float32_t));
-	ne10_fir_init_float(&gFIRfilter, FILTER_TAP_NUM, filterTaps, gFIRfilterState, blockSize);
-}
-
-
-// Initialise the auxiliary task
-// and print info
-
-bool initialise_trigger()
-{
-	if((gTriggerSamplesTask = Bela_createAuxiliaryTask(&trigger_samples, 50, "bela-trigger-samples")) == 0)
-		return false;
-
-	rt_printf("Press 'a' to trigger sample, 's' to stop\n");
-	rt_printf("Press 'q' to quit\n");
-
-	return true;
-}
-
-// This is a lower-priority call to periodically read keyboard input
-// and trigger samples. By placing it at a lower priority,
-// it has minimal effect on the audio performance but it will take longer to
-// complete if the system is under heavy audio load.
-
-void trigger_samples()
-{
-	// This is not a real-time task!
-	// Cos getchar is a system call, not handled by Xenomai.
-	// This task will be automatically down graded.
-
-	char keyStroke = '.';
-
-	keyStroke =	getchar();
-	while(getchar()!='\n'); // to read the first stroke
-
-	switch (keyStroke)
-	{
-		case 'a':
-			gReadPtr = 0;
-			break;
-		case 's':
-			gReadPtr = -1;
-			break;
-		case 'q':
-			gShouldStop = true;
-			break;
-		default:
-			break;
-	}
-}
-
-
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	delete[] gSampleData.samples;
-
-	NE10_FREE(gFIRfilterState);
-	NE10_FREE(gFIRfilterIn);
-	NE10_FREE(gFIRfilterOut);
-}
--- a/examples/filter_IIR/SampleData.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,19 +0,0 @@
-/*
- * SampleData.h
- *
- *  Created on: Nov 5, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef SAMPLEDATA_H_
-#define SAMPLEDATA_H_
-
-// User defined structure to pass between main and rendere complex data retrieved from file
-struct SampleData {
-	float *samples;	// Samples in file
-	int sampleLen;	// Total nume of samples
-};
-
-
-
-#endif /* SAMPLEDATA_H_ */
Binary file examples/filter_IIR/longsample.wav has changed
--- a/examples/filter_IIR/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,186 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: Andrew McPherson and Victor Zappi
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <string>
-#include <getopt.h>
-#include <sndfile.h>				// to load audio files
-
-#include <Bela.h>
-#include "SampleData.h"
-
-using namespace std;
-
-float gCutFreq = 100;
-
-// Load samples from file
-int initFile(string file, SampleData *smp)//float *& smp)
-{
-	SNDFILE *sndfile ;
-	SF_INFO sfinfo ;
-
-	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
-		cout << "Couldn't open file " << file << endl;
-		return 1;
-	}
-
-	int numChan = sfinfo.channels;
-	if(numChan != 1)
-	{
-		cout << "Error: " << file << " is not a mono file" << endl;
-		return 1;
-	}
-
-	smp->sampleLen = sfinfo.frames * numChan;
-	smp->samples = new float[smp->sampleLen];
-	if(smp == NULL){
-		cout << "Could not allocate buffer" << endl;
-		return 1;
-	}
-
-	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
-	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
-
-	// Pad with zeros in case we couldn't read whole file
-	for(int k = readcount; k <smp->sampleLen; k++)
-		smp->samples[k] = 0;
-
-	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
-		double	scale ;
-		int 	m ;
-
-		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
-		if (scale < 1e-10)
-			scale = 1.0 ;
-		else
-			scale = 32700.0 / scale ;
-		cout << "File samples scale = " << scale << endl;
-
-		for (m = 0; m < smp->sampleLen; m++)
-			smp->samples[m] *= scale;
-	}
-
-	sf_close(sndfile);
-
-	return 0;
-}
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	//rt_task_delete ((RT_TASK *) &gTriggerSamplesTask);
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --file [-f] filename:    Name of the file to load (default is \"longsample.wav\")\n";
-	cerr << "   --cutfreq [-c] freq:     Set the cut off frequency of the filter in Hz\n";
-	cerr << "   --help [-h]:             Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	string fileName;			// Name of the sample to load
-
-	SampleData sampleData;		// User define structure to pass data retrieved from file to render function
-	sampleData.samples = 0;
-	sampleData.sampleLen = -1;
-
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"cutfreq", 1, NULL, 'c'},
-		{"file", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:c:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				fileName = string((char *)optarg);
-				break;
-		case 'c':
-				gCutFreq = atof(optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	if(fileName.empty()){
-		fileName = "filter/longsample.wav";
-	}
-
-	if(settings.verbose) {
-		cout << "Loading file " << fileName << endl;
-	}
-
-	// Load file
-	if(initFile(fileName, &sampleData) != 0)
-	{
-		cout << "Error: unable to load samples " << endl;
-		return -1;
-	}
-
-	if(settings.verbose)
-		cout << "File contains " << sampleData.sampleLen << " samples" << endl;
-
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &sampleData) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
-
--- a/examples/filter_IIR/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,238 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
-*/
-
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: Andrew McPherson and Victor Zappi
- */
-
-/**
-\example 4_filter_IIR
-
-Infinite Impulse Response Filter
-------------------------------
-
-This is an example of a infinite impulse response filter implementation.
-*/
-
-#include <Bela.h>	// to schedule lower prio parallel process
-#include <rtdk.h>
-#include <cmath>
-#include <stdio.h>
-#include "SampleData.h"
-
-SampleData gSampleData;	// User defined structure to get complex data from main
-int gReadPtr;			// Position of last read sample from file
-
-// filter vars
-float gLastX[2];
-float gLastY[2];
-double lb0, lb1, lb2, la1, la2 = 0.0;
-
-// communication vars between the 2 auxiliary tasks
-int gChangeCoeff = 0;
-int gFreqDelta = 0;
-
-void initialise_filter(float freq);
-
-void calculate_coeff(float cutFreq);
-
-bool initialise_aux_tasks();
-
-// Task for handling the update of the frequencies using the matrix
-AuxiliaryTask gChangeCoeffTask;
-
-void check_coeff();
-
-// Task for handling the update of the frequencies using the matrix
-AuxiliaryTask gInputTask;
-
-void read_input();
-
-
-extern float gCutFreq;
-
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-
-	// Retrieve a parameter passed in from the initAudio() call
-	gSampleData = *(SampleData *)userData;
-
-	gReadPtr = -1;
-
-	initialise_filter(200);
-
-	// Initialise auxiliary tasks
-	if(!initialise_aux_tasks())
-		return false;
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		float sample = 0;
-		float out = 0;
-
-		// If triggered...
-		if(gReadPtr != -1)
-			sample += gSampleData.samples[gReadPtr++];	// ...read each sample...
-
-		if(gReadPtr >= gSampleData.sampleLen)
-			gReadPtr = -1;
-
-		out = lb0*sample+lb1*gLastX[0]+lb2*gLastX[1]-la1*gLastY[0]-la2*gLastY[1];
-
-		gLastX[1] = gLastX[0];
-		gLastX[0] = out;
-		gLastY[1] = gLastY[0];
-		gLastY[0] = out;
-
-		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
-			context->audioOut[n * context->audioChannels + channel] = out;	// ...and put it in both left and right channel
-
-	}
-
-	// Request that the lower-priority tasks run at next opportunity
-	Bela_scheduleAuxiliaryTask(gChangeCoeffTask);
-	Bela_scheduleAuxiliaryTask(gInputTask);
-}
-
-// First calculation of coefficients
-
-void initialise_filter(float freq)
-{
-	calculate_coeff(freq);
-}
-
-
-// Calculate the filter coefficients
-// second order low pass butterworth filter
-
-void calculate_coeff(float cutFreq)
-{
-	// Initialise any previous state (clearing buffers etc.)
-	// to prepare for calls to render()
-	float sampleRate = 44100;
-	double f = 2*M_PI*cutFreq/sampleRate;
-	double denom = 4+2*sqrt(2)*f+f*f;
-	lb0 = f*f/denom;
-	lb1 = 2*lb0;
-	lb2 = lb0;
-	la1 = (2*f*f-8)/denom;
-	la2 = (f*f+4-2*sqrt(2)*f)/denom;
-	gLastX[0] = gLastX [1] = 0;
-	gLastY[0] = gLastY[1] = 0;
-
-}
-
-
-// Initialise the auxiliary tasks
-// and print info
-
-bool initialise_aux_tasks()
-{
-	if((gChangeCoeffTask = Bela_createAuxiliaryTask(&check_coeff, 90, "bela-check-coeff")) == 0)
-		return false;
-
-	if((gInputTask = Bela_createAuxiliaryTask(&read_input, 50, "bela-read-input")) == 0)
-		return false;
-
-	rt_printf("Press 'a' to trigger sample, 's' to stop\n");
-	rt_printf("Press 'z' to low down cut-off freq of 100 Hz, 'x' to raise it up\n");
-	rt_printf("Press 'q' to quit\n");
-
-	return true;
-}
-
-// Check if cut-off freq has been changed
-// and new coefficients are needed
-
-void check_coeff()
-{
-	if(gChangeCoeff == 1)
-	{
-		gCutFreq += gFreqDelta;
-		gCutFreq = gCutFreq < 0 ? 0 : gCutFreq;
-		gCutFreq = gCutFreq > 22050 ? 22050 : gCutFreq;
-
-		rt_printf("Cut-off frequency: %f\n", gCutFreq);
-
-		calculate_coeff(gCutFreq);
-		gChangeCoeff = 0;
-	}
-}
-
-// This is a lower-priority call to periodically read keyboard input
-// and trigger samples. By placing it at a lower priority,
-// it has minimal effect on the audio performance but it will take longer to
-// complete if the system is under heavy audio load.
-
-void read_input()
-{
-	// This is not a real-time task!
-	// Cos getchar is a system call, not handled by Xenomai.
-	// This task will be automatically down graded.
-
-	char keyStroke = '.';
-
-	keyStroke =	getchar();
-	while(getchar()!='\n'); // to read the first stroke
-
-	switch (keyStroke)
-	{
-		case 'a':
-			gReadPtr = 0;
-			break;
-		case 's':
-			gReadPtr = -1;
-			break;
-		case 'z':
-			gChangeCoeff = 1;
-			gFreqDelta = -100;
-			break;
-		case 'x':
-			gChangeCoeff = 1;
-			gFreqDelta = 100;
-			break;
-		case 'q':
-			gShouldStop = true;
-			break;
-		default:
-			break;
-	}
-}
-
-
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	delete[] gSampleData.samples;
-}
--- a/examples/gpioAnalogLoopbackTest/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,209 +0,0 @@
-#include <Bela.h>
-#include <cmath>
-#include <rtdk.h>
-#include <sys/time.h>
-#include <sys/types.h>
-#include <unistd.h>
-#include <stats.hpp>
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-
-// digital inputs can be changed at will (as they are all being processed at the same time)
-// analog channels must be as per below
-int gAnalogOutCh = 1;
-int gDigitalOutCh = 2;
-int gDigitalInACh = 0;
-int gDigitalInDCh = 3;
-int gAnalogOutLoopDelay;
-int gDigitalOutLoopDelay;
-bool setup(BelaContext *context, void *userData)
-{
-	rt_printf("For this test you need the following connections:\n"
-			"analog%d out->digital%d in, analog%d out->analog%d in, "
-			"digital%d out -> digital%d in, digital%d out-> analog%d in\n",
-			gAnalogOutCh, gDigitalInACh, gAnalogOutCh, 0, gDigitalOutCh, gDigitalInDCh, gDigitalOutCh, 0);
-	rt_printf("Running test with %d analog channels and a buffer size of %d\n",
-			context->analogChannels, context->audioFrames);
-
-	for(unsigned int n = 0; n < context->digitalFrames; n++){
-		pinMode(context, n, gDigitalInACh, INPUT);
-		pinMode(context, n, gDigitalInDCh, INPUT);
-		pinMode(context, n, gDigitalOutCh, OUTPUT);
-	}
-	switch (context->analogChannels){
-		case 2:
-			gAnalogOutLoopDelay = context->audioFrames*2 + 3;
-			gDigitalOutLoopDelay = context->audioFrames*2 + 2;
-			break;
-		case 4:
-			gAnalogOutLoopDelay = context->audioFrames*2 + 3;
-			gDigitalOutLoopDelay = context->audioFrames*2 + 2;
-			break;
-		case 8:
-			gAnalogOutLoopDelay = context->audioFrames + 3;
-			gDigitalOutLoopDelay = context->audioFrames + 1;
-			break;
-		default:
-			exit(2);
-	}
-
-    return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numAnalogFrames
-// will be 0.
-
-const int patternLength = 31;
-static int anaErrorCount = 0;
-static int digErrorCount = 0;
-void render(BelaContext *context, void *userData)
-{
-	static bool writePattern[patternLength] = {
-									0,1,0,1,0,0,1,1,
-									0,0,0,1,1,1,0,0,
-									1,1,1,1,0,0,0,0,
-									1,1,1,1,1,0,0};
-//	for(int n = 0; n < patternLength; n++){
-//		writePattern[n]=1;
-//	}
-	static int inPointer = 0;
-	static int outPointer = 0;
-	static int digitalOutPointer = 0;
-	static int digitalInPointer = 0;
-	static int analogOut = 1;
-	/** Checking offset between analog and digital
-	 * how it should be :
-	 * The PRU loop does the following (the loop runs at 88.2kHz):
-	 * - Read/write audio sample (once for the left channel, once for the right channel)
-	 * - Write DAC 0 or 0/2 or 0/2/4/6
-	 * - Read ADC 0 or 0/2 or 0/2/4/6, 2 samples (@176.4) older than NOW
-	 * - /During/ the line above, every two loops we also Read/Write GPIO,
-	 * therefore reading on ADC 0/2/4/6 a value that is being output from GPIO will lead to undefined results
-	 * - Write DAC 1 or 1/3 or 1/3/5/7
-	 * - Read ADC 1 or 1/3 or 1/3/5/7, 2 samples (@176.4) older than NOW
-	 */
-	if(1)
-	for(unsigned int n = 0; n < context->audioFrames; n++){
-		static bool analog0In = false;
-		static bool digitalAIn = false;
-		static int count = 0;
-		bool doReadWrite = context->analogChannels<=4 ? true : ((context->analogChannels == 8) && (n&1)==0);
-		if(doReadWrite){
-			digitalAIn = digitalRead(context, n, gDigitalInACh);
-			switch(context->analogChannels){
-			case 8:
-				analog0In = analogRead(context, n/2, 0) > 0.5;
-				analogWrite(context, n/2, analogOut, writePattern[outPointer]);
-				break;
-			case 4:
-				analog0In = analogRead(context, n, 0) > 0.5;
-				analogWrite(context, n, analogOut, writePattern[outPointer]);
-				break;
-			case 2:
-				analog0In = analogRead(context, n * 2 + 1, 0) > 0.5;
-				analogWrite(context, 2 * n, analogOut, writePattern[outPointer]);
-				analogWrite(context, 2 * n + 1, analogOut, writePattern[outPointer]);
-				break;
-			}
-			gAnalogOutLoopDelay--;
-			outPointer++;
-			if(gAnalogOutLoopDelay <= 0){
-				if(++inPointer == patternLength){
-					inPointer = 0;
-				}
-			}
-		}
-		bool expectedIn = writePattern[inPointer];
-		if(gAnalogOutLoopDelay <= 0 && doReadWrite == true){
-			if(analog0In != expectedIn || digitalAIn != expectedIn){
-				rt_printf("expected: %d, received: %d %d, pointer: %d, delay: %d, count: %d\n",
-					expectedIn, analog0In, digitalAIn, inPointer, gAnalogOutLoopDelay, count);
-				anaErrorCount++;
-			}
-		}
-		count++;
-		if(analog0In != digitalAIn){ // at any time the analog and digital in should be the same
-			rt_printf("ana %d_%d %d,\n", analog0In, digitalAIn, n);
-		}
-		if(outPointer == patternLength){
-			outPointer = 0;
-		}
-	}
-	if(1)
-	for(unsigned int n = 0; n < context->audioFrames; n++){
-		static int count = 0;
-		static bool analog1In = false;
-		static bool digitalDIn = false;
-/* we need to remember the pastAnalog1In because
- *  reading GPIO takes place before writing to it, therefore
- *  when reading a GPIOout, the GPIOin samples will always be one sample late
- */
-		bool doReadWrite = false;
-		static bool pastAnalog1In = false;
-		digitalWriteOnce(context, n, gDigitalOutCh,  writePattern[digitalOutPointer]);
-		if(context->analogChannels == 8){
-			if((n&1) == 0){ //do it every other sample
-				pastAnalog1In = analogRead(context, n/2, 1) > 0.5;
-				digitalDIn = digitalRead(context, n, gDigitalInDCh);
-				doReadWrite = true;
-			}
-		}
-		if(context->analogChannels == 4){
-			pastAnalog1In = analogRead(context, n, 1) > 0.5;
-			digitalDIn = digitalRead(context, n, gDigitalInDCh);
-			digitalWriteOnce(context, n, gDigitalOutCh,  writePattern[digitalOutPointer]);
-			doReadWrite = true;
-		}
-		if(context->analogChannels == 2){
-			pastAnalog1In = analogRead(context, n * 2, 1) > 0.5;
-			digitalDIn = digitalRead(context, n, gDigitalInDCh);
-			digitalWriteOnce(context, n, gDigitalOutCh,  writePattern[digitalOutPointer]);
-			doReadWrite = true;
-		}
-		bool expectedDigitalIn = writePattern[digitalInPointer];
-		if(doReadWrite == true){
-			gDigitalOutLoopDelay--;
-			if(gDigitalOutLoopDelay <= 0){
-				if(expectedDigitalIn != pastAnalog1In || expectedDigitalIn != digitalDIn){
-					rt_printf("D expected: %d, received: %d %d, pointer: %d, delay: %d, count: %d\n",
-						expectedDigitalIn, pastAnalog1In, digitalDIn, inPointer, gDigitalOutLoopDelay, count);
-					digErrorCount++;
-				}
-				if(++digitalInPointer == patternLength){
-					digitalInPointer = 0;
-				}
-			}
-			pastAnalog1In = analog1In;
-			if(++digitalOutPointer == patternLength){
-				digitalOutPointer = 0;
-			}
-		}
-		count++;
-	}
-	if(context->audioFramesElapsed > 30000){
-		gShouldStop = true;
-	}
-}
-
-
-void cleanup(BelaContext *context, void *userData)
-{
-	if(anaErrorCount == 0 && digErrorCount == 0){
-		rt_printf("Test was succesful with %d analog channels and a buffer size of %d\n", context->analogChannels, context->audioFrames);
-	} else {
-		rt_printf("------------------------\n%danalog %ddigital errors over %dsamples while running test with ",
-				anaErrorCount, digErrorCount, context->audioFramesElapsed);
-		rt_printf("%d analog channels and a buffer size of %d \n\n\n",
-				context->analogChannels, context->audioFrames);
-		exit(1);
-	}
-}
--- a/examples/level_meter/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,139 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-
-#include <Bela.h>
-#include <cmath>
-
-#define NUMBER_OF_SEGMENTS	10
-
-// Two levels of audio: one follows current value, the other holds
-// peaks for longer
-float gAudioLocalLevel = 0, gAudioPeakLevel = 0;
-
-// Decay rates for detecting levels
-float gLocalDecayRate = 0.99, gPeakDecayRate = 0.999;
-
-// Thresholds for LEDs: set in setup()
-float gThresholds[NUMBER_OF_SEGMENTS + 1];
-int gSamplesToLight[NUMBER_OF_SEGMENTS];
-
-// High-pass filter on the input
-float gLastX[2] = {0};
-float gLastY[2] = {0};
-
-// These coefficients make a high-pass filter at 5Hz for 44.1kHz sample rate
-double gB0 = 0.99949640;
-double gB1 = -1.99899280;
-double gB2 = gB0;
-double gA1 = -1.99899254;
-double gA2 = 0.99899305;
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{	
-	// This project makes the assumption that the audio and digital
-	// sample rates are the same. But check it to be sure!
-	if(context->audioFrames != context->digitalFrames) {
-		rt_printf("Error: this project needs the audio and digital sample rates to be the same.\n");
-		return false;
-	}
-	
-	// Initialise threshold levels in -3dB steps. One extra for efficiency in render()
-	// Level = 10^(dB/20)
-	for(int i = 0; i < NUMBER_OF_SEGMENTS + 1; i++) {
-		gThresholds[i] = powf(10.0f, (-1.0 * (NUMBER_OF_SEGMENTS - i)) * .05);
-	}
-	
-	for(int i = 0; i < NUMBER_OF_SEGMENTS; i++) {
-		gSamplesToLight[i] = 0;
-		pinMode(context, 0, i, OUTPUT);
-	}
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		// Get average of audio input channels
-		float sample = 0;
-		for(unsigned int ch = 0; ch < context->audioChannels; ch++) {
-			context->audioOut[n * context->audioChannels + ch] = 
-				context->audioIn[n * context->audioChannels + ch];
-			sample += context->audioIn[n * context->audioChannels + ch];
-		}
-		
-		// Do DC-blocking on the sum
-		float out = gB0 * sample + gB1 * gLastX[0] + gB2 * gLastX[1]
-						- gA1 * gLastY[0] - gA2 * gLastY[1];
-
-		gLastX[1] = gLastX[0];
-		gLastX[0] = sample;
-		gLastY[1] = gLastY[0];
-		gLastY[0] = out;
-		
-		out = fabsf(out / (float)context->audioChannels);
-		
-		// Do peak detection: fast-responding local level
-		if(out > gAudioLocalLevel)
-			gAudioLocalLevel = out;
-		else
-			gAudioLocalLevel *= gLocalDecayRate;
-		
-		// Do peak detection: slow-responding peak level
-		if(out > gAudioPeakLevel)
-			gAudioPeakLevel = out;
-		else {
-			// Make peak decay slowly by only multiplying
-			// every few samples
-			if(((context->audioFramesElapsed + n) & 31) == 0)
-				gAudioPeakLevel *= gPeakDecayRate;
-		}	
-		// LED bargraph on digital outputs 0-9
-		for(int led = 0; led < NUMBER_OF_SEGMENTS; led++) {
-			// All LEDs up to the local level light up. The LED
-			// for the peak level also remains lit.
-			int state = LOW;
-				
-			if(gAudioLocalLevel > gThresholds[led]) {
-				state = HIGH;
-				gSamplesToLight[led] = 1000;
-			}
-			/*else if(gAudioPeakLevel > gThresholds[led] && gAudioPeakLevel <= gThresholds[led + 1]) {
-				state = HIGH;
-				gSamplesToLight[led] = 1000;
-			}*/
-			else if(--gSamplesToLight[led] > 0)
-				state = HIGH;
-			
-			// Write LED
-			digitalWriteOnce(context, n, led, state);
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/loggingSensors/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,97 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-#include <unistd.h>
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --frequency [-f] frequency: Set the frequency of the oscillator\n";
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	float frequency = 440.0;	// Frequency of oscillator
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"frequency", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				frequency = atof(optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &frequency) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/loggingSensors/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,37 +0,0 @@
-#include <Bela.h> 
-#include <cmath>
-#include <WriteFile.h>
-
-WriteFile file1;
-WriteFile file2;
-
-bool setup(BelaContext *context, void *userData)
-{
-	file1.init("out.bin"); //set the file name to write to
-	file1.setEchoInterval(1000);
-	file1.setFileType(kBinary);
-	file1.setFormat("%.4f %.4f\n"); // set the format that you want to use for your output. Please use %f only (with modifiers). When in binary mode, this is used only for echoing to console
-	file2.init("out.m"); //set the file name to write to
-	file2.setHeader("myvar=[\n"); //set one or more lines to be printed at the beginning of the file
-	file2.setFooter("];\n"); //set one or more lines to be printed at the end of the file
-	file2.setFormat("%.4f\n"); // set the format that you want to use for your output. Please use %f only (with modifiers)
-	file2.setFileType(kText);
-	file2.setEchoInterval(10000); // only print to the console 1 line every other 10000
-	return true; 
-}
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->analogFrames; n++) {
-			file1.log(&(context->analogIn[n*context->analogFrames]), 2); // log an array of values
-			file2.log(context->analogIn[n*context->analogFrames]); // log a single value
-	}
-}
-
-// cleanup_render() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in initialise_render().
-
-void cleanup(BelaContext *context, void *userData)
-{
-    
-}
--- a/examples/measure_noisefloor/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,104 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-extern int gBufferSize;
-
-using namespace std;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --buffer-size [-b] size  Set the analysis buffer size\n";
-	cerr << "   --help [-h]:             Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"buffer-size", 1, NULL, 'b'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// By default use a longer period size because latency is not an issue
-	settings.periodSize = 32;
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hb:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'b':
-				gBufferSize = atoi(optarg);
-				break;
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-	
-	if(gBufferSize < settings.periodSize)
-		gBufferSize = settings.periodSize;
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, 0) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/measure_noisefloor/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,162 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-
-#include <Bela.h>
-#include <cmath>
-
-int gBufferSize = 8192;
-
-// Double buffers to hold samples for noise analysis
-float *gReadBuffers[10], *gWriteBuffers[10];
-float *gBuffers0[10], *gBuffers1[10];
-
-int gWriteBufferPointers[10], gReadBufferPointers[10];
-
-// Task to analyse and print results which would otherwise be too slow for render()
-AuxiliaryTask gAnalysisTask;
-
-void analyseResults();
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{	
-	// Clear the filter data structures
-	for(int i = 0; i < 10; i++) {
-		gReadBufferPointers[i] = gWriteBufferPointers[i] = 0;
-		gBuffers0[i] = new float[gBufferSize];
-		gBuffers1[i] = new float[gBufferSize];		
-		gWriteBuffers[i] = gBuffers0[i];
-		gReadBuffers[i] = gBuffers1[i];
-		if(gBuffers0[i] == 0 || gBuffers1[i] == 0) {
-			rt_printf("Error allocating buffer %d\n", i);
-			return false;
-		}
-	}
-	
-	gAnalysisTask = Bela_createAuxiliaryTask(analyseResults, 50, "bela-analyse-results");
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	bool bufferIsFull = false;	// Whether at least one buffer has filled
-	
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		// Store audio inputs in buffer
-		for(unsigned int ch = 0; ch < context->audioChannels; ch++) {
-			if(gWriteBufferPointers[ch] < gBufferSize) {
-				gWriteBuffers[ch][gWriteBufferPointers[ch]] = 
-					context->audioIn[n * context->audioChannels + ch];
-				gWriteBufferPointers[ch]++;
-				if(gWriteBufferPointers[ch] >= gBufferSize)
-					bufferIsFull = true;
-			}
-		}
-	}
-	
-	if(context->analogChannels != 0) {
-		for(unsigned int n = 0; n < context->analogFrames; n++) {
-			// Store analog inputs in buffer, starting at channel 2
-			for(unsigned int ch = 0; ch < context->analogChannels; ch++) {
-				if(gWriteBufferPointers[ch + 2] < gBufferSize) {
-					gWriteBuffers[ch + 2][gWriteBufferPointers[ch + 2]] = 
-						context->analogIn[n * context->analogChannels + ch];
-					gWriteBufferPointers[ch + 2]++;
-					if(gWriteBufferPointers[ch + 2] >= gBufferSize)
-						bufferIsFull = true;
-				}
-				
-				// Set all analog outputs to halfway point so they can be more
-				// easily measured for noise
-				context->analogOut[n * context->analogChannels + ch] = 0.5;
-			}
-		}	
-	}
-	
-
-	if(bufferIsFull) {
-		// Swap buffers and reset write pointers
-		for(int ch = 0; ch < 10; ch++) {
-			gReadBufferPointers[ch] = gWriteBufferPointers[ch];
-			gWriteBufferPointers[ch] = 0;
-			
-			if(gReadBuffers[ch] == gBuffers0[ch]) {
-				gReadBuffers[ch] = gBuffers1[ch];
-				gWriteBuffers[ch] = gBuffers0[ch];
-			}
-			else {
-				gReadBuffers[ch] = gBuffers0[ch];
-				gWriteBuffers[ch] = gBuffers1[ch];				
-			}
-		}
-		
-		Bela_scheduleAuxiliaryTask(gAnalysisTask);
-	}
-}
-
-void analyseResults()
-{
-	rt_printf("\e[1;1H\e[2J");	// Command to clear the screen
-
-	// Print the analysis results. channels 0-1 are audio, channels 2-9 are analog
-	for(int ch = 0; ch < 10; ch++) {
-		// Skip unused channels
-		if(gReadBufferPointers[ch] == 0)
-			continue;
-		
-		float mean = 0;
-		for(int n = 0; n < gReadBufferPointers[ch]; n++) {
-			mean += gReadBuffers[ch][n];
-		}
-		mean /= (float)gReadBufferPointers[ch];
-		
-		float rms = 0;
-		for(int n = 0; n < gReadBufferPointers[ch]; n++) {
-			rms += (gReadBuffers[ch][n] - mean) * (gReadBuffers[ch][n] - mean);
-		}				
-		rms = sqrtf(rms / (float)gReadBufferPointers[ch]);
-		
-		if(ch == 0)
-			rt_printf("Audio In L:  ");
-		else if(ch == 1)
-			rt_printf("Audio In R:  ");
-		else
-			rt_printf("Analog In %d: ", ch - 2);
-		
-		rt_printf("Noise %6.1fdB    DC offset %6.4f (%6.1fdB)    window size: %d\n", 
-					20.0f * log10f(rms),
-					mean, 
-					20.0f * log10f(fabsf(mean)),
-					gReadBufferPointers[ch]);
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	for(int i = 0; i < 10; i++) {
-		delete gBuffers0[i];
-		delete gBuffers1[i];
-	}
-}
--- a/examples/minimal/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,16 +0,0 @@
-#include <Bela.h>
-
-bool setup(BelaContext *context, void *userData)
-{
-	return true;
-}
-
-void render(BelaContext *context, void *userData)
-{
-
-}
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
\ No newline at end of file
--- a/examples/mpr121/I2C_MPR121.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,176 +0,0 @@
-/*
- * I2C_MPR121.cpp
- *
- *  Created on: Oct 14, 2013
- *      Author: Victor Zappi
- */
-
-
-#include "I2C_MPR121.h"
-
-I2C_MPR121::I2C_MPR121() {
-
-}
-
-boolean I2C_MPR121::begin(uint8_t bus, uint8_t i2caddr) {
-  _i2c_address = i2caddr;
-	
-  if(initI2C_RW(bus, i2caddr, 0) > 0)
-	  return false;
-
-  // soft reset
-  writeRegister(MPR121_SOFTRESET, 0x63);
-  usleep(1000);
-  //delay(1);
-  for (uint8_t i=0; i<0x7F; i++) {
-  //  Serial.print("$"); Serial.print(i, HEX); 
-  //  Serial.print(": 0x"); Serial.println(readRegister8(i));
-  }
-  
-
-  writeRegister(MPR121_ECR, 0x0);
-
-  uint8_t c = readRegister8(MPR121_CONFIG2);
-  
-  if (c != 0x24) {
-	  rt_printf("MPR121 read 0x%x instead of 0x24\n", c);
-	  return false;
-  }
-
-  setThresholds(12, 6);
-  writeRegister(MPR121_MHDR, 0x01);
-  writeRegister(MPR121_NHDR, 0x01);
-  writeRegister(MPR121_NCLR, 0x0E);
-  writeRegister(MPR121_FDLR, 0x00);
-
-  writeRegister(MPR121_MHDF, 0x01);
-  writeRegister(MPR121_NHDF, 0x05);
-  writeRegister(MPR121_NCLF, 0x01);
-  writeRegister(MPR121_FDLF, 0x00);
-
-  writeRegister(MPR121_NHDT, 0x00);
-  writeRegister(MPR121_NCLT, 0x00);
-  writeRegister(MPR121_FDLT, 0x00);
-
-  writeRegister(MPR121_DEBOUNCE, 0);
-  writeRegister(MPR121_CONFIG1, 0x10); // default, 16uA charge current
-  writeRegister(MPR121_CONFIG2, 0x20); // 0.5uS encoding, 1ms period
-
-//  writeRegister(MPR121_AUTOCONFIG0, 0x8F);
-
-//  writeRegister(MPR121_UPLIMIT, 150);
-//  writeRegister(MPR121_TARGETLIMIT, 100); // should be ~400 (100 shifted)
-//  writeRegister(MPR121_LOWLIMIT, 50);
-  // enable all electrodes
-  writeRegister(MPR121_ECR, 0x8F);  // start with first 5 bits of baseline tracking
-
-  return true;
-}
-
-void I2C_MPR121::setThresholds(uint8_t touch, uint8_t release) {
-  for (uint8_t i=0; i<12; i++) {
-    writeRegister(MPR121_TOUCHTH_0 + 2*i, touch);
-    writeRegister(MPR121_RELEASETH_0 + 2*i, release);
-  }
-}
-
-uint16_t  I2C_MPR121::filteredData(uint8_t t) {
-  if (t > 12) return 0;
-  return readRegister16(MPR121_FILTDATA_0L + t*2);
-}
-
-uint16_t  I2C_MPR121::baselineData(uint8_t t) {
-  if (t > 12) return 0;
-  uint16_t bl = readRegister8(MPR121_BASELINE_0 + t);
-  return (bl << 2);
-}
-
-uint16_t  I2C_MPR121::touched(void) {
-  uint16_t t = readRegister16(MPR121_TOUCHSTATUS_L);
-  return t & 0x0FFF;
-}
-
-/*********************************************************************/
-
-
-uint8_t I2C_MPR121::readRegister8(uint8_t reg) {
-    unsigned char inbuf, outbuf;
-    struct i2c_rdwr_ioctl_data packets;
-    struct i2c_msg messages[2];
-
-    /*
-     * In order to read a register, we first do a "dummy write" by writing
-     * 0 bytes to the register we want to read from.  This is similar to
-     * the packet in set_i2c_register, except it's 1 byte rather than 2.
-     */
-    outbuf = reg;
-    messages[0].addr  = 0x5A;
-    messages[0].flags = 0;
-    messages[0].len   = sizeof(outbuf);
-    messages[0].buf   = &outbuf;
-
-    /* The data will get returned in this structure */
-    messages[1].addr  = 0x5A;
-    messages[1].flags = I2C_M_RD/* | I2C_M_NOSTART*/;
-    messages[1].len   = sizeof(inbuf);
-    messages[1].buf   = &inbuf;
-
-    /* Send the request to the kernel and get the result back */
-    packets.msgs      = messages;
-    packets.nmsgs     = 2;
-    if(ioctl(i2C_file, I2C_RDWR, &packets) < 0) {
-        rt_printf("Unable to send data");
-        return 0;
-    }
-
-    return inbuf;
-}
-
-uint16_t I2C_MPR121::readRegister16(uint8_t reg) {
-    unsigned char inbuf[2], outbuf;
-    struct i2c_rdwr_ioctl_data packets;
-    struct i2c_msg messages[2];
-
-    /*
-     * In order to read a register, we first do a "dummy write" by writing
-     * 0 bytes to the register we want to read from.  This is similar to
-     * the packet in set_i2c_register, except it's 1 byte rather than 2.
-     */
-    outbuf = reg;
-    messages[0].addr  = _i2c_address;
-    messages[0].flags = 0;
-    messages[0].len   = sizeof(outbuf);
-    messages[0].buf   = &outbuf;
-
-    /* The data will get returned in this structure */
-    messages[1].addr  = _i2c_address;
-    messages[1].flags = I2C_M_RD/* | I2C_M_NOSTART*/;
-    messages[1].len   = sizeof(inbuf);
-    messages[1].buf   = inbuf;
-
-    /* Send the request to the kernel and get the result back */
-    packets.msgs      = messages;
-    packets.nmsgs     = 2;
-    if(ioctl(i2C_file, I2C_RDWR, &packets) < 0) {
-        rt_printf("Unable to send data");
-        return 0;
-    }
-
-    return (uint16_t)inbuf[0] | (((uint16_t)inbuf[1]) << 8);
-}
-
-/**************************************************************************/
-/*!
-    @brief  Writes 8-bits to the specified destination register
-*/
-/**************************************************************************/
-void I2C_MPR121::writeRegister(uint8_t reg, uint8_t value) {
-	uint8_t buf[2] = { reg, value };
-
-	if(write(i2C_file, buf, 2) != 2)
-	{
-		cout << "Failed to write register " << (int)reg << " on MPR121\n";
-		return;
-	}
-}
-
--- a/examples/mpr121/I2C_MPR121.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,82 +0,0 @@
-/*
- * MPR121 Bela demo
- * 
- * Andrew McPherson
- * Based on Adafruit library by Limor Fried/Ladyada
- */
-
-#ifndef I2CTK_H_
-#define I2CTK_H_
-
-#include <I2c.h>
-#include "Utilities.h"
-
-typedef bool boolean;
-
-#define MPR121_I2CADDR_DEFAULT 0x5A
-
-#define MPR121_TOUCHSTATUS_L 0x00
-#define MPR121_TOUCHSTATUS_H 0x01
-#define MPR121_FILTDATA_0L  0x04
-#define MPR121_FILTDATA_0H  0x05
-#define MPR121_BASELINE_0   0x1E
-#define MPR121_MHDR         0x2B
-#define MPR121_NHDR         0x2C
-#define MPR121_NCLR         0x2D
-#define MPR121_FDLR         0x2E
-#define MPR121_MHDF         0x2F
-#define MPR121_NHDF         0x30
-#define MPR121_NCLF         0x31
-#define MPR121_FDLF         0x32
-#define MPR121_NHDT         0x33
-#define MPR121_NCLT         0x34
-#define MPR121_FDLT         0x35
-
-#define MPR121_TOUCHTH_0    0x41
-#define MPR121_RELEASETH_0    0x42
-#define MPR121_DEBOUNCE 0x5B
-#define MPR121_CONFIG1 0x5C
-#define MPR121_CONFIG2 0x5D
-#define MPR121_CHARGECURR_0 0x5F
-#define MPR121_CHARGETIME_1 0x6C
-#define MPR121_ECR 0x5E
-#define MPR121_AUTOCONFIG0 0x7B
-#define MPR121_AUTOCONFIG1 0x7C
-#define MPR121_UPLIMIT   0x7D
-#define MPR121_LOWLIMIT  0x7E
-#define MPR121_TARGETLIMIT  0x7F
-
-#define MPR121_GPIODIR  0x76
-#define MPR121_GPIOEN  0x77
-#define MPR121_GPIOSET  0x78
-#define MPR121_GPIOCLR  0x79
-#define MPR121_GPIOTOGGLE  0x7A
-
-#define MPR121_SOFTRESET 0x80
-
-class I2C_MPR121 : public I2c
-{
-public:
-	// Hardware I2C
-	I2C_MPR121();
-
-	boolean begin(uint8_t bus = 1, uint8_t i2caddr = MPR121_I2CADDR_DEFAULT);
-
-	uint16_t filteredData(uint8_t t);
-	uint16_t  baselineData(uint8_t t);
-
-	uint8_t readRegister8(uint8_t reg);
-	uint16_t readRegister16(uint8_t reg);
-	void writeRegister(uint8_t reg, uint8_t value);
-	uint16_t touched(void);
- 
- 	void setThresholds(uint8_t touch, uint8_t release);
-	
-	int readI2C() { return 0; } // Unused
-	
-private:
-	int _i2c_address;
-};
-
-
-#endif /* I2CTK_H_ */
--- a/examples/mpr121/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,128 +0,0 @@
-#include <Bela.h>
-#include <cmath>
-#include <rtdk.h>
-#include "I2C_MPR121.h"
-
-// How many pins there are
-#define NUM_TOUCH_PINS 12
-
-// Define this to print data to terminal
-#undef DEBUG_MPR121
-
-// Change this to change how often the MPR121 is read (in Hz)
-int readInterval = 50;
-
-// Change this threshold to set the minimum amount of touch
-int threshold = 40;
-
-// This array holds the continuous sensor values
-int sensorValue[NUM_TOUCH_PINS];
-
-// ---- test code stuff -- can be deleted for your example ----
-
-// 12 notes of a C major scale...
-float gFrequencies[NUM_TOUCH_PINS] = {261.63, 293.66, 329.63, 349.23, 392.00, 440.00, 493.88, 523.25, 587.33, 659.25, 698.25, 783.99};
-
-// This is internal stuff for the demo
-float gNormFrequencies[NUM_TOUCH_PINS];
-float gPhases[NUM_TOUCH_PINS] = {0};
-
-// ---- internal stuff -- do not change -----
-
-I2C_MPR121 mpr121;			// Object to handle MPR121 sensing
-AuxiliaryTask i2cTask;		// Auxiliary task to read I2C
-
-int readCount = 0;			// How long until we read again...
-int readIntervalSamples = 0; // How many samples between reads
-
-void readMPR121();
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	if(!mpr121.begin(1, 0x5A)) {
-		rt_printf("Error initialising MPR121\n");
-		return false;
-	}
-	
-	i2cTask = Bela_createAuxiliaryTask(readMPR121, 50, "bela-mpr121");
-	readIntervalSamples = context->audioSampleRate / readInterval;
-	
-	for(int i = 0; i < NUM_TOUCH_PINS; i++) {
-		gNormFrequencies[i] = 2.0 * M_PI * gFrequencies[i] / context->audioSampleRate;
-	}
-	
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numAnalogFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(int n = 0; n < context->audioFrames; n++) {
-		// Keep this code: it schedules the touch sensor readings
-		if(++readCount >= readIntervalSamples) {
-			readCount = 0;
-			Bela_scheduleAuxiliaryTask(i2cTask);
-		}
-		
-		float sample = 0.0;
-		
-		// This code can be replaced with your favourite audio code
-		for(int i = 0; i < NUM_TOUCH_PINS; i++) {
-			float amplitude = sensorValue[i] / 400.0;
-			
-			// Prevent clipping
-			if(amplitude > 0.5)
-				amplitude = 0.5;
-			
-			sample += amplitude * sinf(gPhases[i]);
-			gPhases[i] += gNormFrequencies[i];
-			if(gPhases[i] > 2.0 * M_PI)
-				gPhases[i] -= 2.0 * M_PI;
-		}
-		
-		for(int ch = 0; ch < context->audioChannels; ch++)
-			context->audioOut[context->audioChannels * n + ch] = sample;
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	// Nothing to do here
-}
-
-
-// Auxiliary task to read the I2C board
-void readMPR121()
-{
-	for(int i = 0; i < NUM_TOUCH_PINS; i++) {
-		sensorValue[i] = -(mpr121.filteredData(i) - mpr121.baselineData(i));
-		sensorValue[i] -= threshold;
-		if(sensorValue[i] < 0)
-			sensorValue[i] = 0;
-#ifdef DEBUG_MPR121
-		rt_printf("%d ", sensorValue[i]);
-#endif
-	}
-#ifdef DEBUG_MPR121
-	rt_printf("\n");
-#endif
-	
-	// You can use this to read binary on/off touch state more easily
-	//rt_printf("Touched: %x\n", mpr121.touched());
-}
--- a/examples/osc/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,94 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/**
-\example 5_osc
-
-Open Sound Control
-------------------
-
-This example shows an implementation of OSC (Open Sound Control) which was 
-developed at UC Berkeley Center for New Music and Audio Technology (CNMAT).
-
-It is designed to be run alongside resources/osc/osc.js
-
-The OSC server port on which to receive is set in `setup()` 
-via `oscServer.setup()`. Likewise the OSC client port on which to 
-send is set in `oscClient.setup()`.
-
-In `setup()` an OSC message to address `/osc-setup`, it then waits 
-1 second for a reply on `/osc-setup-reply`.
-
-in `render()` the code receives OSC messages, parses them, and sends 
-back an acknowledgment.
-*/
-
-#include <Bela.h>
-#include <OSCServer.h>
-#include <OSCClient.h>
-
-OSCServer oscServer;
-OSCClient oscClient;
-
-// this example is designed to be run alongside resources/osc/osc.js
-
-// parse messages recieved by OSC Server
-// msg is Message class of oscpkt: http://gruntthepeon.free.fr/oscpkt/
-void parseMessage(oscpkt::Message msg){
-    
-    rt_printf("recieved message to: %s\n", msg.addressPattern().c_str());
-    
-    int intArg;
-    float floatArg;
-    if (msg.match("/osc-test").popInt32(intArg).popFloat(floatArg).isOkNoMoreArgs()){
-        rt_printf("recieved int %i and float %f\n", intArg, floatArg);
-    }
-    
-}
-
-bool setup(BelaContext *context, void *userData)
-{
-    // setup the OSC server to recieve on port 7562
-    oscServer.setup(7562);
-    // setup the OSC client to send on port 7563
-    oscClient.setup(7563);
-    
-    // the following code sends an OSC message to address /osc-setup
-    // then waits 1 second for a reply on /osc-setup-reply
-    bool handshakeRecieved = false;
-    oscClient.sendMessageNow(oscClient.newMessage.to("/osc-setup").end());
-    oscServer.recieveMessageNow(1000);
-    while (oscServer.messageWaiting()){
-        if (oscServer.popMessage().match("/osc-setup-reply")){
-            handshakeRecieved = true;
-        }
-    }
-    
-    if (handshakeRecieved){
-        rt_printf("handshake recieved!\n");
-    } else {
-        rt_printf("timeout!\n");
-    }
-    
-	return true;
-}
-
-void render(BelaContext *context, void *userData)
-{
-    // recieve OSC messages, parse them, and send back an acknowledgment
-    while (oscServer.messageWaiting()){
-        parseMessage(oscServer.popMessage());
-        oscClient.queueMessage(oscClient.newMessage.to("/osc-acknowledge").add(5).add(4.2f).add(std::string("OSC message recieved")).end());
-    }
-}
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/oscillator_bank/audio_routines.S	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,161 +0,0 @@
-@
-@ audio_routines.S
-@
-@ NEON-based functions for time-critical audio processing
-@
-@ Andrew McPherson 2014
-@ Queen Mary University of London
-@
-
-	.syntax unified
-	.arch armv7-a
-	.fpu neon
-
-@ 	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
-@							  int activePartialNum, int lookupTableSize,
-@							  float *phases, float *frequencies, float *amplitudes,
-@							  float *freqDerivatives, float *ampDerivatives,
-@							  float *lookupTable);
-
-@ Registers:
-@    r0: numAudioFrames        How many frames to render
-@    r1: audioOut              Buffer for audio output samples [stereo]
-@    r2: activePartialNum      How many active partials to render
-@    r3: lookupTableSize       Size of lookup table
-@    ---- other arguments start on the stack and are moved: -----
-@    r4: phases                Phase of each oscillator (pointer)
-@    r5: frequencies           Normalised frequency of each oscillator (pointer)
-@    r6: amplitudes            Normalised amplitude of each oscillator (pointer)
-@    r7: freqDerivatives       Derivative of frequency for each oscillator (pointer)
-@    r8: ampDerivatives        Derivative of amplitude for each oscillator (pointer)
-@    r9: lookupTable           Lookup table containing one oscillation
-@
-@ Alignment requirements:
-@    audioOut: 8-byte boundary
-@    phases: 16-byte boundary
-@    frequencies: 16-byte boundary
-@    amplitudes: 16-byte boundary
-@    freqDerivatives: 16-byte bounary
-@    ampDerivatives: 16-byte boundary
-@    lookupTable: 4-byte boundary (TODO: check this)
-
-	.align	2
-	.global	oscillator_bank_neon
-	.thumb
-	.thumb_func
-	.type	oscillator_bank_neon, %function
-oscillator_bank_neon:
-
-
-dSample		.dn		D6.F32
-qPhases		.qn		Q8.F32
-dPhases_0	.dn		D16.F32
-dPhases_1	.dn		D17.F32
-qFreqs		.qn		Q9.F32
-dFreqs_0	.dn		D18.F32
-dFreqs_1	.dn		D19.F32
-qAmps		.qn		Q10.F32
-dAmps_0		.dn		D20.F32
-dAmps_1		.dn		D21.F32
-qFreqDs		.qn		Q11.F32
-dFreqDs_0	.dn		D22.F32
-dFreqDs_1	.dn		D23.F32
-qAmpDs		.qn		Q12.F32
-dAmpDs_0	.dn		D24.F32
-dAmpDs_1	.dn		D25.F32
-
-qBaseInts	.qn		Q13.U32		@ Base indexes: unsigned ints x4
-dBaseInts_0	.dn		D26.U32
-dBaseInts_1	.dn		D27.U32
-qFractions  .qn     Q14.F32		@ Fraction indexes: floats x4
-qTableBase	.qn		Q15.U32		@ Base of lookup table
-
-	cmp r0, #0					@ Check for trivial case 1: zero frames
-	it eq
-	bxeq lr						@ Return if that's the case (otherwise might have odd behaviour)
-	cmp r2, #4					@ Check for trivial case 2: zero oscillators
-	it lt
-	bxlt lr						@ Return if that's the case
-
-	push {r4-r11}				@ Now arguments start 32 bytes above SP
-    add r11, sp, #32			@ Pointer to 32 bytes into the stack
-    ldm r11, {r4-r9}			@ Load 6 arguments into registers
-
-	vdup qTableBase, r9			@ Move lookup table base index into 4 ints
-
-	@ Outer loop: iterate over the number of oscillators, choosing 4 at a
-	@ time to work with.
-oscbank_oscillator_loop:
-	vld1 {dPhases_0, dPhases_1}, [r4]		@ no increment; will store at end of sample loop
-	vld1 {dFreqs_0, dFreqs_1}, [r5]
-	vld1 {dAmps_0, dAmps_1}, [r6]
-	vld1 {dFreqDs_0, dFreqDs_1}, [r7]!		@ increment; won't update at end of sample loop
-	vld1 {dAmpDs_0, dAmpDs_1}, [r8]!
-
-	push {r0-r1,r4-r8}
-	@ --- inner loop: iterate over the number of samples ---
-oscbank_sample_loop:
-	vcvt qBaseInts, qPhases		     		@ Take floor(phases)
-	vmov q2.f32, #1.0						@ Load 1.0 into every slot of q2
-	vshl q0.U32, qBaseInts, #2				@ Shift the indexes left 2 (*4 for float addressing)
-	vcvt qFractions, qBaseInts				@ int back to float
-	vadd q0.U32, q0.U32, qTableBase			@ Find memory addresses
-
-	vmov r4, r5, d0							@ Move two indexes to ARM registers
-	vmov r6, r7, d1							@ Move two more indexes to ARM registers
-	vsub qFractions, qPhases, qFractions	@ fraction = phase - floor(phase)
-
-	vldr.64	d0, [r4]						@ Load two consecutive floats at each location
-	vldr.64 d1, [r5]						@ These hold the previous and following samples in the table
-	vldr.64	d2, [r6]						@ TODO: check whether these work at 4-byte alignment
-	vldr.64 d3, [r7]
-
-	@ Format at this point:
-	@ Osc0(before) Osc0(after) Osc1(before) Osc1(after) Osc2(before) Osc2(after) Osc3(before) Osc3(after)
-	@ We want:
-	@ Osc0(before) Osc1(before) Osc2(before) Osc3(before) Osc0(after) Osc1(after) Osc2(after) Osc3(after)
-
-	vuzp.32 q0, q1							@ Now q0 contains before, q1 contains after
-	vsub q2.f32, q2.f32, qFractions			@ q2 = 1.0 - fraction
-	vmul q1.f32, q1.f32, qFractions			@ q1 = fraction * after
-	vmul q0.f32, q0.f32, q2.f32				@ q0 = (1.0 - fraction) * before
-
-	vadd qPhases, qPhases, qFreqs			@ Update phases
-	vadd qFreqs, qFreqs, qFreqDs			@ Update frequencies
-
-	vadd q0.f32, q0.f32, q1.f32				@ Add two interpolated components to get the final sample
-	vdup q2.u32, r3							@ Put lookup table size into each element of q2
-	vcvt qBaseInts, qPhases					@ Take floor of new phases
-	vmul q0.f32, q0.f32, qAmps				@ Multiply samples by current amplitude
-
-	vld1 dSample, [r1]						@ Load the current stereo samples
-	vpadd d2.f32, d0.f32, d1.f32			@ Pairwise accumulate q0 (output sample) into d2
-
-	vand q2, q2, qBaseInts					@ Logical AND of new phase int leaves 1 bit set only if phase >= table size
-	vpadd d3.f32, d2.f32, d2.f32			@ Pairwise accumulate d2 into d0 --> d0[0] and d0[1] both hold total of 4 oscillators
-	vadd qAmps, qAmps, qAmpDs				@ Update amplitudes
-	vcvt q0.f32, q2.u32						@ Convert int back to float after AND operation
-
-	vadd  dSample, dSample, d3.f32			@ Add oscillator outputs to each channel
-
-	subs r0, r0, #1							@ numFrames--
-	vsub qPhases, qPhases, q0.f32			@ Keep phases in table range
-	vst1 dSample, [r1]!						@ Store back in buffer and increment by 8
-
-	it gt
-	bgt oscbank_sample_loop					@ Loop if numFrames > 0
-
-	@ --- end inner loop ---
-	pop {r0-r1,r4-r8}						@ Restore registers: restores audioOut and numFrames, among others
-
-	vst1 {dPhases_0, dPhases_1}, [r4]!		@ Store phases back to array
-	vst1 {dFreqs_0, dFreqs_1}, [r5]!		@ Store frequencies back to array
-	vst1 {dAmps_0, dAmps_1}, [r6]!			@ Store amplitudes back to array
-											@ No need to update r7, r8
-
-	subs r2, r2, #4							@ numPartials -= 4
-	it  gt
-	bgt oscbank_oscillator_loop	@ Loop if numPartials > 0
-
-    pop {r4-r11}
-	bx lr
--- a/examples/oscillator_bank/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,116 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <Bela.h>
-
-using namespace std;
-
-int gNumOscillators = 32;
-int gWavetableLength = 1024;
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --num-oscillators [-n] oscs: Set the number of oscillators to use (default: 32)\n";
-	cerr << "   --wavetable [-w] length:     Set the wavetable length in samples (default: 1024)\n";
-	cerr << "   --help [-h]:                 Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"num-oscillators", 1, NULL, 'n'},
-		{"wavetable", 1, NULL, 'w'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hn:w:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'n':
-				gNumOscillators = atoi(optarg);
-				if(gNumOscillators <= 0) {
-					usage(basename(argv[0]));
-					exit(0);
-				}
-				break;
-		case 'w':
-				gWavetableLength = atoi(optarg);
-				if(gWavetableLength < 4)
-					gWavetableLength = 4;
-				if(gWavetableLength > 16384)
-					gWavetableLength = 16384;
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, 0) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	if(settings.verbose) {
-		cout << "--> Using " << gNumOscillators << " oscillators and wavetable of " << gWavetableLength << " samples\n";
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/oscillator_bank/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,238 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-/**
-\example 4_oscillator_bank
-
-Oscillator Bank
-----------------------
-
-These files demonstrate an oscillator bank implemented in assembly code 
-that is used as part of the d-box project.
-*/
-
-#include <Bela.h>
-#include <rtdk.h>
-#include <cstdlib>
-#include <cmath>
-#include <cstring>
-#include <time.h>
-
-const float kMinimumFrequency = 20.0f;
-const float kMaximumFrequency = 8000.0f;
-
-float *gWavetable;		// Buffer holding the precalculated sine lookup table
-float *gPhases;			// Buffer holding the phase of each oscillator
-float *gFrequencies;	// Buffer holding the frequencies of each oscillator
-float *gAmplitudes;		// Buffer holding the amplitudes of each oscillator
-float *gDFrequencies;	// Buffer holding the derivatives of frequency
-float *gDAmplitudes;	// Buffer holding the derivatives of amplitude
-
-float gAudioSampleRate;
-int gSampleCount;		// Sample counter for indicating when to update frequencies
-float gNewMinFrequency;
-float gNewMaxFrequency;
-
-// Task for handling the update of the frequencies using the matrix
-AuxiliaryTask gFrequencyUpdateTask;
-
-// These settings are carried over from main.cpp
-// Setting global variables is an alternative approach
-// to passing a structure to userData in setup()
-
-extern int gNumOscillators;
-extern int gWavetableLength;
-
-void recalculate_frequencies();
-
-extern "C" {
-	// Function prototype for ARM assembly implementation of oscillator bank
-	void oscillator_bank_neon(int numAudioFrames, float *audioOut,
-							  int activePartialNum, int lookupTableSize,
-							  float *phases, float *frequencies, float *amplitudes,
-							  float *freqDerivatives, float *ampDerivatives,
-							  float *lookupTable);
-}
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-bool setup(BelaContext *context, void *userData)
-{
-	srandom(time(NULL));
-
-	if(context->audioChannels != 2) {
-		rt_printf("Error: this example needs stereo audio enabled\n");
-		return false;
-	}
-
-	// Initialise the sine wavetable
-	if(posix_memalign((void **)&gWavetable, 8, (gWavetableLength + 1) * sizeof(float))) {
-		rt_printf("Error allocating wavetable\n");
-		return false;
-	}
-	for(int n = 0; n < gWavetableLength + 1; n++)
-		gWavetable[n] = sinf(2.0 * M_PI * (float)n / (float)gWavetableLength);
-
-	// Allocate the other buffers
-	if(posix_memalign((void **)&gPhases, 16, gNumOscillators * sizeof(float))) {
-		rt_printf("Error allocating phase buffer\n");
-		return false;
-	}
-	if(posix_memalign((void **)&gFrequencies, 16, gNumOscillators * sizeof(float))) {
-		rt_printf("Error allocating frequency buffer\n");
-		return false;
-	}
-	if(posix_memalign((void **)&gAmplitudes, 16, gNumOscillators * sizeof(float))) {
-		rt_printf("Error allocating amplitude buffer\n");
-		return false;
-	}
-	if(posix_memalign((void **)&gDFrequencies, 16, gNumOscillators * sizeof(float))) {
-		rt_printf("Error allocating frequency derivative buffer\n");
-		return false;
-	}
-	if(posix_memalign((void **)&gDAmplitudes, 16, gNumOscillators * sizeof(float))) {
-		rt_printf("Error allocating amplitude derivative buffer\n");
-		return false;
-	}
-
-	// Initialise buffer contents
-
-	float freq = kMinimumFrequency;
-	float increment = (kMaximumFrequency - kMinimumFrequency) / (float)gNumOscillators;
-
-	for(int n = 0; n < gNumOscillators; n++) {
-		gPhases[n] = 0.0;
-
-		if(context->analogFrames == 0) {
-			// Random frequencies when used without matrix
-			gFrequencies[n] = kMinimumFrequency + (kMaximumFrequency - kMinimumFrequency) * ((float)random() / (float)RAND_MAX);
-		}
-		else {
-			// Constant spread of frequencies when used with matrix
-			gFrequencies[n] = freq;
-			freq += increment;
-		}
-
-		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
-		gFrequencies[n] *= (float)gWavetableLength / context->audioSampleRate;
-		gAmplitudes[n] = ((float)random() / (float)RAND_MAX) / (float)gNumOscillators;
-		gDFrequencies[n] = gDAmplitudes[n] = 0.0;
-	}
-
-	increment = 0;
-	freq = 440.0;
-
-	for(int n = 0; n < gNumOscillators; n++) {
-		// Update the frequencies to a regular spread, plus a small amount of randomness
-		// to avoid weird phase effects
-		float randScale = 0.99 + .02 * (float)random() / (float)RAND_MAX;
-		float newFreq = freq * randScale;
-
-		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
-		gFrequencies[n] = newFreq * (float)gWavetableLength / context->audioSampleRate;
-
-		freq += increment;
-	}
-
-	// Initialise auxiliary tasks
-	if((gFrequencyUpdateTask = Bela_createAuxiliaryTask(&recalculate_frequencies, 85, "bela-update-frequencies")) == 0)
-		return false;
-
-	//for(int n = 0; n < gNumOscillators; n++)
-	//	rt_printf("%f\n", gFrequencies[n]);
-
-	gAudioSampleRate = context->audioSampleRate;
-	gSampleCount = 0;
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	// Initialise buffer to 0
-	memset(context->audioOut, 0, 2 * context->audioFrames * sizeof(float));
-
-	// Render audio frames
-	oscillator_bank_neon(context->audioFrames, context->audioOut,
-			gNumOscillators, gWavetableLength,
-			gPhases, gFrequencies, gAmplitudes,
-			gDFrequencies, gDAmplitudes,
-			gWavetable);
-
-	if(context->analogFrames != 0 && (gSampleCount += context->audioFrames) >= 128) {
-		gSampleCount = 0;
-		gNewMinFrequency = map(context->analogIn[0], 0, 1.0, 1000.0f, 8000.0f);
-		gNewMaxFrequency = map(context->analogIn[1], 0, 1.0, 1000.0f, 8000.0f);
-
-		// Make sure max >= min
-		if(gNewMaxFrequency < gNewMinFrequency) {
-			float temp = gNewMaxFrequency;
-			gNewMaxFrequency = gNewMinFrequency;
-			gNewMinFrequency = temp;
-		}
-
-		// Request that the lower-priority task run at next opportunity
-		//Bela_scheduleAuxiliaryTask(gFrequencyUpdateTask);
-	}
-}
-
-// This is a lower-priority call to update the frequencies which will happen
-// periodically when the matrix is enabled. By placing it at a lower priority,
-// it has minimal effect on the audio performance but it will take longer to
-// complete if the system is under heavy audio load.
-
-void recalculate_frequencies()
-{
-	float freq = gNewMinFrequency;
-	float increment = (gNewMaxFrequency - gNewMinFrequency) / (float)gNumOscillators;
-
-	for(int n = 0; n < gNumOscillators; n++) {
-		// Update the frequencies to a regular spread, plus a small amount of randomness
-		// to avoid weird phase effects
-		float randScale = 0.99 + .02 * (float)random() / (float)RAND_MAX;
-		float newFreq = freq * randScale;
-
-		// For efficiency, frequency is expressed in change in wavetable position per sample, not Hz or radians
-		gFrequencies[n] = newFreq * (float)gWavetableLength / gAudioSampleRate;
-
-		freq += increment;
-	}
-}
-
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	free(gWavetable);
-	free(gPhases);
-	free(gFrequencies);
-	free(gAmplitudes);
-	free(gDFrequencies);
-	free(gDAmplitudes);
-}
--- a/examples/pd/basicAnalogIn/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,30 +0,0 @@
-#N canvas 203 356 637 339 10;
-#X obj 63 170 osc~ 440;
-#X obj 63 223 dac~;
-#X obj 63 191 *~;
-#X text 440 95 ADC 3: Analog In 0;
-#X text 34 41 Basic analog in (or 'theremin');
-#X text 34 51 ===============================;
-#X text 121 105 << Use first two analog in's;
-#X text 440 49 ADC routing:;
-#X text 440 57 ------------;
-#X text 440 105 ADC 4: Analog In 1;
-#X text 440 115 ADC 5: Analog In 2;
-#X text 440 125 ADC 6: Analog In 3;
-#X text 440 135 ADC 7: Analog In 4;
-#X text 440 145 ADC 8: Analog In 5;
-#X text 440 155 ADC 9: Analog In 6;
-#X text 440 165 ADC 10: Analog In 7;
-#X text 440 75 ADC 1: Audio In L;
-#X text 440 85 ADC 2: Audio In R;
-#X obj 63 149 *~ 880;
-#X obj 63 106 adc~ 3 4;
-#X text 355 277 ---------------------------------;
-#X text 355 267 @krighxz / BELA / heavy / 12/2015;
-#X text 379 286 beaglert.cc / enzienaudio.com;
-#X connect 0 0 2 0;
-#X connect 2 0 1 0;
-#X connect 2 0 1 1;
-#X connect 18 0 0 0;
-#X connect 19 0 18 0;
-#X connect 19 1 2 1;
--- a/examples/pd/basicAnalogOut/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,30 +0,0 @@
-#N canvas 275 504 679 362 10;
-#X text 148 124 << Use first two analog in's;
-#X text 431 57 ------------;
-#X text 44 51 Basic analog out;
-#X text 44 61 ================;
-#X text 431 95 DAC 3: Analog Out 0;
-#X text 431 105 DAC 4: Analog Out 1;
-#X text 431 115 DAC 5: Analog Out 2;
-#X text 431 125 DAC 6: Analog Out 3;
-#X text 431 145 DAC 8: Analog Out 5;
-#X text 431 135 DAC 7: Analog Out 4;
-#X text 431 155 DAC 9: Analog Out 6;
-#X text 431 165 DAC 10: Analog Out 7;
-#X text 431 85 DAC 2: Audio Out R;
-#X text 431 75 DAC 1: Audio Out L;
-#X obj 92 201 *~;
-#X obj 92 159 *~ 10;
-#X obj 92 180 osc~ 1;
-#X obj 35 242 dac~ 1 2 3;
-#X text 143 241 << Output to first analog out;
-#X text 431 49 DAC routing:;
-#X text 432 289 ---------------------------------;
-#X text 432 279 @krighxz / BELA / heavy / 12/2015;
-#X text 456 298 beaglert.cc / enzienaudio.com;
-#X obj 92 125 adc~ 3 4;
-#X connect 14 0 17 2;
-#X connect 15 0 16 0;
-#X connect 16 0 14 0;
-#X connect 23 0 15 0;
-#X connect 23 1 14 1;
--- a/examples/pd/circularBuffer/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,79 +0,0 @@
-#N canvas 436 490 738 494 10;
-#X obj 135 167 loadbang;
-#X msg 135 212 1;
-#X obj 183 234 / 44.1;
-#X obj 135 253 metro;
-#X obj 135 286 f;
-#X obj 160 286 + 64;
-#X obj 160 308 % 65536, f 8;
-#X msg 160 329 start \$1;
-#X obj 160 351 tabwrite~ circbuf;
-#X obj 363 303 - 32768;
-#X obj 363 325 + 65536;
-#X obj 363 347 % 65536;
-#X obj 342 371 +~ 0;
-#X msg 381 236 0;
-#X obj 342 258 phasor~;
-#X obj 342 392 tabread4~ circbuf;
-#X obj 341 415 dac~;
-#N canvas 422 781 312 126 buf 0;
-#N canvas 0 22 450 278 (subpatch) 0;
-#X array circbuf 65536 float 2;
-#X coords 0 1 65535 -1 256 64 1 0 0;
-#X restore 23 28 graph;
-#X restore 129 441 pd buf;
-#X obj 32 101 osc~ 440;
-#X obj 342 213 samplerate~;
-#X obj 134 189 t b b b b;
-#X text 30 82 audio input;
-#X text 219 310 write pointer;
-#X text 412 349 read pointer;
-#X obj 342 282 *~ 16;
-#X obj 342 236 / 16;
-#X obj 183 214 f 16;
-#X obj 363 189 r \$0-blocksize;
-#X obj 204 186 r \$0-blocksize;
-#X obj 394 259 r \$0-blocksize;
-#X obj 390 123 s \$0-blocksize;
-#X text 34 13 VIRTUAL CIRCULAR BUFFER;
-#X text 34 23 =======================;
-#X obj 390 55 loadbang;
-#X msg 390 77 16;
-#X text 517 454 ---------------------------------;
-#X text 517 444 @krighxz / BELA / heavy / 12/2015;
-#X text 541 463 beaglert.cc / enzienaudio.com;
-#X text 426 78 << replace with target blocksize;
-#X text 446 90 BELA default: 16;
-#X text 446 102 PD default: 64;
-#X connect 0 0 20 0;
-#X connect 1 0 3 0;
-#X connect 2 0 3 1;
-#X connect 3 0 4 0;
-#X connect 4 0 5 0;
-#X connect 5 0 6 0;
-#X connect 5 0 9 0;
-#X connect 6 0 7 0;
-#X connect 6 0 4 1;
-#X connect 7 0 8 0;
-#X connect 9 0 10 0;
-#X connect 10 0 11 0;
-#X connect 11 0 12 1;
-#X connect 12 0 15 0;
-#X connect 13 0 14 1;
-#X connect 14 0 24 0;
-#X connect 15 0 16 0;
-#X connect 15 0 16 1;
-#X connect 18 0 8 0;
-#X connect 19 0 25 0;
-#X connect 20 0 1 0;
-#X connect 20 1 26 0;
-#X connect 20 2 13 0;
-#X connect 20 3 19 0;
-#X connect 24 0 12 0;
-#X connect 25 0 14 0;
-#X connect 26 0 2 0;
-#X connect 27 0 25 1;
-#X connect 28 0 26 1;
-#X connect 29 0 24 1;
-#X connect 33 0 34 0;
-#X connect 34 0 30 0;
--- a/examples/pd/digital/bela_digital-help.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,97 +0,0 @@
-#N canvas 121 46 877 628 10;
-#X obj 64 94 r bela_digitalIn11;
-#X obj 219 94 r bela_digitalIn12;
-#X obj 386 107 adc~ 13 14;
-#X text 49 64 Digital in \, message rate;
-#X text 214 69 Digital in \, message rate;
-#X text 394 85 Digital ins \, signal rate;
-#X obj 32 283 s bela_digitalOut15;
-#X text 13 309 Digital out \, message rate;
-#X obj 396 593 s bela_setDigital;
-#X obj 396 533 loadbang;
-#X msg 431 401 in 11;
-#X text 524 390 First argument is the pin direction Second argument
-is the pin number Third argument is the rate \, can be "~" or "sig"
-for signal-rate or "message" for message-rate. Defaults to message-rate.
-;
-#X msg 431 458 in 13 ~;
-#X msg 432 429 in 12 message;
-#X msg 517 506 disable 17;
-#X obj 386 139 snapshot~;
-#X obj 500 134 snapshot~;
-#X obj 529 56 loadbang;
-#X obj 384 181 print digitalIn13;
-#X obj 500 183 print digitalIn14;
-#X obj 497 105 metro 200;
-#X text 525 474 To stop using a pin (e.g.: to save CPU) \, send a disable
-message containing the pin number;
-#X obj 224 280 dac~ 16;
-#X obj 22 232 metro 500;
-#X obj 32 257 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
-1;
-#X obj 224 253 phasor~ 300;
-#X text 160 234 Nice \, anti-aliased square wave;
-#X obj 22 207 loadbang;
-#X text 15 181 Blink that LED!;
-#X obj 411 306 dac~ 17;
-#X obj 405 248 osc~ 300;
-#X obj 470 248 phasor~ 169;
-#X obj 457 278 -~ 0.5;
-#X text 403 210 Sum signals together for unpredictable effects;
-#X obj 612 303 dac~ 18;
-#X text 198 305 Digital out \, signal rate;
-#X text 378 325 Digital out \, signal rate;
-#X text 588 325 Digital out \, signal rate;
-#X obj 608 222 adc~;
-#X obj 612 249 *~ 0.5;
-#X obj 612 276 +~ 0.5;
-#X text 585 203 The ultimate bit-crusher;
-#X msg 426 555 in 11 \, in 12 \, in 13 ~ \, in 14 ~ \, out 15 \, out
-16 ~ \, out 17 ~ \, out 18 ~;
-#X text 23 377 Each digital channel can be configured as either an
-input or an output. Send messages to bela_setDigital to configure the
-digitalPins (needed before using them). Pins settings can be modified
-at will during execution. Message-rate input pins receive messages
-only when the input changes. Message-rate output pins will write to
-the output the most recent value you send them. Signal-rate inputs
-and outputs are handled as regular audio channels. Outputs are thresholded
-to 0.5 before being written to the actual output. Note that each pin
-that is managed adds to the CPU usage \, so only activate the pins
-you actually need.;
-#X text 34 19 Digital channels are numbered 11 to 26 \, which corresponds
-to Bela's 0 to 15 digital channels. They can be set to respond at message
-rate or at signal rate;
-#X obj 415 271 osc~ 1;
-#X obj 385 157 change;
-#X obj 500 158 change;
-#X obj 64 116 print digitalIn11;
-#X obj 219 117 print digitalIn12;
-#X connect 0 0 48 0;
-#X connect 1 0 49 0;
-#X connect 2 0 15 0;
-#X connect 2 1 16 0;
-#X connect 9 0 42 0;
-#X connect 10 0 8 0;
-#X connect 12 0 8 0;
-#X connect 13 0 8 0;
-#X connect 14 0 8 0;
-#X connect 15 0 46 0;
-#X connect 16 0 47 0;
-#X connect 17 0 20 0;
-#X connect 20 0 16 0;
-#X connect 20 0 15 0;
-#X connect 23 0 24 0;
-#X connect 24 0 6 0;
-#X connect 25 0 22 0;
-#X connect 27 0 23 0;
-#X connect 30 0 29 0;
-#X connect 31 0 32 0;
-#X connect 32 0 29 0;
-#X connect 38 0 39 0;
-#X connect 38 1 39 0;
-#X connect 39 0 40 0;
-#X connect 40 0 34 0;
-#X connect 42 0 8 0;
-#X connect 45 0 29 0;
-#X connect 46 0 18 0;
-#X connect 47 0 19 0;
--- a/examples/pd/digital/digital_example.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,89 +0,0 @@
-#N canvas 279 78 857 690 10;
-#X obj 18 477 osc~ 200;
-#X obj 57 522 *~;
-#X obj 112 405 line~ 1;
-#X msg 116 372 1 \, 0 200;
-#X obj 205 472 line~ 1;
-#X msg 209 439 1 \, 0 200;
-#X obj 210 405 select 1;
-#X obj 140 531 *~;
-#X obj 484 601 s bela_setDigital;
-#X obj 484 532 loadbang;
-#X msg 65 445 0;
-#X obj 116 332 r bela_digitalIn12;
-#X obj 422 217 r bela_digitalIn14;
-#X obj 422 243 s bela_digitalOut13;
-#X obj 285 218 r bela_digitalIn12;
-#X obj 285 243 s bela_digitalOut11;
-#X obj 374 339 timer;
-#X msg 404 304 bang;
-#X msg 368 304 bang;
-#X obj 555 184 metro 1000;
-#X obj 575 304 s bela_digitalOut15;
-#X obj 374 366 print sourceto12;
-#X obj 488 351 timer;
-#X msg 518 316 bang;
-#X msg 482 316 bang;
-#X obj 487 380 print sourceto14;
-#X obj 663 15 r bela_digitalIn16;
-#X msg 553 265 1;
-#X obj 577 214 delay 500;
-#X obj 480 275 select 1;
-#X obj 346 268 select 1;
-#X msg 584 281 0;
-#X obj 634 251 select 1 0;
-#X obj 80 622 dac~ 3 4;
-#X obj 65 566 expr~ $v1*0.5 + 0.5;
-#X obj 138 588 expr~ $v1*0.5 + 0.5;
-#X obj 65 202 dac~ 11;
-#X obj 114 83 osc~ 300;
-#X obj 176 179 dac~ 1 3;
-#X obj 34 136 adc~ 17;
-#X obj 423 504 delay 3000;
-#X msg 388 552 disable 12;
-#X msg 484 552 out 11 ~ \, out 13 \, in 14 \, out 15 \, in 16 \, in
-17 ~;
-#X connect 0 0 1 0;
-#X connect 0 0 7 0;
-#X connect 1 0 34 0;
-#X connect 2 0 1 1;
-#X connect 3 0 2 0;
-#X connect 4 0 7 1;
-#X connect 5 0 4 0;
-#X connect 6 0 5 0;
-#X connect 6 0 10 0;
-#X connect 7 0 35 0;
-#X connect 9 0 42 0;
-#X connect 9 0 40 0;
-#X connect 10 0 0 1;
-#X connect 11 0 3 0;
-#X connect 12 0 13 0;
-#X connect 12 0 29 0;
-#X connect 14 0 15 0;
-#X connect 14 0 30 0;
-#X connect 16 0 21 0;
-#X connect 17 0 16 1;
-#X connect 18 0 16 0;
-#X connect 19 0 27 0;
-#X connect 19 0 28 0;
-#X connect 22 0 25 0;
-#X connect 23 0 22 1;
-#X connect 24 0 22 0;
-#X connect 26 0 32 0;
-#X connect 27 0 20 0;
-#X connect 27 0 24 0;
-#X connect 27 0 18 0;
-#X connect 28 0 31 0;
-#X connect 29 0 23 0;
-#X connect 30 0 17 0;
-#X connect 31 0 20 0;
-#X connect 32 0 27 0;
-#X connect 32 1 31 0;
-#X connect 34 0 33 0;
-#X connect 35 0 33 1;
-#X connect 37 0 38 0;
-#X connect 37 0 38 1;
-#X connect 39 0 36 0;
-#X connect 40 0 41 0;
-#X connect 41 0 8 0;
-#X connect 42 0 8 0;
--- a/examples/pd/envelopeTrigger/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,196 +0,0 @@
-#N canvas 646 209 359 287 10;
-#X obj 28 207 dac~;
-#N canvas 324 380 556 621 env 0;
-#N canvas 886 1001 131 207 >~ 0;
-#X obj 19 -37 -~;
-#X obj 19 -58 min~;
-#X obj 19 26 *~ 1e+37;
-#X obj 19 5 +~ 1e-37;
-#X obj 19 -16 clip~ -1e-37 0;
-#X obj 19 -83 inlet~;
-#X obj 62 -83 inlet~;
-#X obj 19 48 outlet~;
-#X connect 0 0 4 0;
-#X connect 1 0 0 0;
-#X connect 2 0 7 0;
-#X connect 3 0 2 0;
-#X connect 4 0 3 0;
-#X connect 5 0 1 0;
-#X connect 6 0 0 1;
-#X connect 6 0 1 1;
-#X restore 199 106 pd >~;
-#X obj 23 173 /~ 44100;
-#X obj 68 153 samplerate~;
-#X obj 68 133 loadbang;
-#X obj 199 128 biquad~ 0 0 -1 1 0;
-#X obj 23 209 rpole~;
-#X obj 23 153 sig~ 1;
-#X obj 226 171 sig~ 0.5;
-#N canvas 0 22 110 231 <~ 0;
-#X obj 11 -41 -~;
-#X obj 11 29 +~ 1e-37;
-#X obj 11 8 clip~ -1e-37 0;
-#X obj 11 -86 inlet~;
-#X obj 55 -85 inlet~;
-#X obj 11 -62 max~;
-#X obj 11 -17 *~ -1;
-#X obj 11 81 outlet~;
-#X obj 11 50 *~ 1e+37;
-#X connect 0 0 6 0;
-#X connect 1 0 8 0;
-#X connect 2 0 1 0;
-#X connect 3 0 5 0;
-#X connect 4 0 0 1;
-#X connect 4 0 5 1;
-#X connect 5 0 0 0;
-#X connect 6 0 2 0;
-#X connect 8 0 7 0;
-#X restore 199 192 pd <~;
-#X text 237 193 (x != x);
-#X obj 23 467 *~;
-#X obj 199 150 *~ -1;
-#X obj 23 331 *~ 512;
-#X obj 23 229 clip~ 0 1;
-#X obj 23 352 tabread4~ env1;
-#X obj 23 310 *~;
-#X obj 199 64 abs~;
-#X obj 199 43 hip~ 1;
-#X obj 199 22 inlet~ excitationSignal;
-#X obj 218 269 sig~ 1;
-#X obj 218 292 /~;
-#X obj 226 86 inlet~ threshold;
-#X obj 262 229 inlet~ envDuration;
-#X obj 38 445 osc~ 220;
-#X obj 38 425 +~ 50;
-#X obj 38 385 *~;
-#X obj 38 405 *~ 1000;
-#X obj 262 269 clip~ 1e-05 100000;
-#X obj 23 487 outlet~;
-#N canvas 0 22 450 278 (subpatch) 0;
-#X array env1 512 float 3;
-#A 0 0 0.0625 0.0883883 0.108253 0.125 0.139754 0.153093 0.165359 0.176777
-0.1875 0.197642 0.207289 0.216506 0.225347 0.233854 0.242061 0.25 0.257694
-0.265165 0.272431 0.279509 0.286411 0.293151 0.299739 0.306186 0.3125
-0.318689 0.32476 0.330719 0.336573 0.342327 0.347985 0.353553 0.359035
-0.364434 0.369755 0.375 0.380173 0.385276 0.390312 0.395285 0.400195
-0.405046 0.40984 0.414578 0.419263 0.423896 0.428478 0.433013 0.4375
-0.441942 0.446339 0.450694 0.455007 0.459279 0.463512 0.467707 0.471865
-0.475986 0.480072 0.484123 0.488141 0.492125 0.496078 0.5 0.503891
-0.507752 0.511585 0.515388 0.519164 0.522913 0.526634 0.53033 0.534
-0.537645 0.541266 0.544862 0.548435 0.551985 0.555512 0.559017 0.5625
-0.565962 0.569402 0.572822 0.576222 0.579601 0.582961 0.586302 0.589624
-0.592927 0.596212 0.599479 0.602728 0.60596 0.609175 0.612372 0.615554
-0.618718 0.621867 0.625 0.628117 0.631219 0.634306 0.637377 0.640434
-0.643477 0.646505 0.649519 0.652519 0.655506 0.658478 0.661438 0.664384
-0.667317 0.670238 0.673146 0.676041 0.678924 0.681795 0.684653 0.6875
-0.690335 0.693159 0.695971 0.698771 0.701561 0.704339 0.707107 0.709864
-0.71261 0.715345 0.71807 0.720785 0.72349 0.726184 0.728869 0.731544
-0.734209 0.736864 0.73951 0.742146 0.744773 0.747391 0.75 0.7526 0.75519
-0.757772 0.760345 0.76291 0.765466 0.768013 0.770552 0.773082 0.775605
-0.778119 0.780625 0.783123 0.785613 0.788095 0.790569 0.793036 0.795495
-0.797947 0.800391 0.802827 0.805256 0.807678 0.810093 0.8125 0.8149
-0.817294 0.81968 0.822059 0.824432 0.826797 0.829156 0.831508 0.833854
-0.836193 0.838525 0.840851 0.843171 0.845484 0.847791 0.850092 0.852386
-0.854675 0.856957 0.859233 0.861503 0.863767 0.866025 0.868278 0.870524
-0.872765 0.875 0.877229 0.879453 0.881671 0.883883 0.88609 0.888292
-0.890488 0.892679 0.894864 0.897044 0.899218 0.901388 0.903552 0.905711
-0.907865 0.910014 0.912157 0.914296 0.91643 0.918559 0.920682 0.922801
-0.924916 0.927025 0.929129 0.931229 0.933324 0.935414 0.9375 0.939581
-0.941657 0.943729 0.945797 0.947859 0.949918 0.951972 0.954021 0.956066
-0.958107 0.960143 0.962175 0.964203 0.966227 0.968246 0.970261 0.972272
-0.974279 0.976281 0.97828 0.980274 0.982265 0.984251 0.986233 0.988212
-0.990186 0.992157 0.994123 0.996086 0.998045 1 0.992172 0.984375 0.976609
-0.968874 0.961169 0.953495 0.945852 0.938239 0.930657 0.923106 0.915586
-0.908097 0.900638 0.89321 0.885813 0.878447 0.871111 0.863806 0.856532
-0.849289 0.842076 0.834894 0.827743 0.820623 0.813533 0.806474 0.799446
-0.792449 0.785483 0.778547 0.771642 0.764767 0.757924 0.751111 0.744329
-0.737578 0.730857 0.724168 0.717509 0.71088 0.704283 0.697716 0.69118
-0.684675 0.678201 0.671757 0.665344 0.658962 0.652611 0.64629 0.64
-0.633741 0.627513 0.621315 0.615148 0.609012 0.602907 0.596832 0.590788
-0.584775 0.578793 0.572841 0.56692 0.56103 0.555171 0.549343 0.543545
-0.537778 0.532041 0.526336 0.520661 0.515017 0.509404 0.503822 0.49827
-0.492749 0.487259 0.481799 0.476371 0.470973 0.465605 0.460269 0.454963
-0.449689 0.444444 0.439231 0.434048 0.428897 0.423775 0.418685 0.413625
-0.408597 0.403599 0.398631 0.393695 0.388789 0.383914 0.37907 0.374256
-0.369473 0.364721 0.36 0.355309 0.35065 0.346021 0.341423 0.336855
-0.332318 0.327812 0.323337 0.318893 0.314479 0.310096 0.305744 0.301423
-0.297132 0.292872 0.288643 0.284444 0.280277 0.27614 0.272034 0.267958
-0.263914 0.2599 0.255917 0.251965 0.248043 0.244152 0.240292 0.236463
-0.232664 0.228897 0.22516 0.221453 0.217778 0.214133 0.210519 0.206936
-0.203383 0.199862 0.196371 0.19291 0.189481 0.186082 0.182714 0.179377
-0.176071 0.172795 0.16955 0.166336 0.163153 0.16 0.156878 0.153787
-0.150727 0.147697 0.144698 0.14173 0.138793 0.135886 0.13301 0.130165
-0.127351 0.124567 0.121815 0.119093 0.116401 0.113741 0.111111 0.108512
-0.105944 0.103406 0.1009 0.0984237 0.0959785 0.093564 0.0911803 0.0888274
-0.0865052 0.0842138 0.0819531 0.0797232 0.077524 0.0753556 0.073218
-0.0711111 0.069035 0.0669896 0.064975 0.0629911 0.0610381 0.0591157
-0.0572241 0.0553633 0.0535332 0.0517339 0.0499654 0.0482276 0.0465206
-0.0448443 0.0431988 0.041584 0.04 0.0384467 0.0369243 0.0354325 0.0339715
-0.0325413 0.0311419 0.0297732 0.0284352 0.027128 0.0258516 0.0246059
-0.023391 0.0222068 0.0210534 0.0199308 0.0188389 0.0177778 0.0167474
-0.0157478 0.0147789 0.0138408 0.0129335 0.0120569 0.0112111 0.010396
-0.00961169 0.00885813 0.00813533 0.00744329 0.00678201 0.00615148 0.00555171
-0.0049827 0.00444444 0.00393695 0.00346021 0.00301423 0.002599 0.00221453
-0.00186082 0.00153787 0.00124567 0.000984237 0.000753556 0.000553633
-0.000384467 0.000246059 0.000138408 6.15148e-05 1.53787e-05 0;
-#X coords 0 1 512 0 512 64 1 0 0;
-#X restore 24 536 graph;
-#X text 244 55 centre signal around zero and take abs value;
-#X text 243 107 if greater than threshold output 1;
-#X text 313 129 differentiator;
-#X text 238 150 (generates impulse on positive trigger (0->1);
-#X text 67 210 signal-rate counter;
-#X text 296 193 -> resets counter by briefly setting;
-#X text 314 204 rpole~ coeff to 0;
-#X text 96 416 kickdrum 101 :P;
-#X text 400 514 pre-generated envelope;
-#X obj 262 249 *~;
-#X text 127 352 << use envelope value for volume and frequency;
-#X connect 0 0 4 0;
-#X connect 1 0 5 0;
-#X connect 2 0 1 1;
-#X connect 3 0 2 0;
-#X connect 4 0 11 0;
-#X connect 5 0 13 0;
-#X connect 6 0 1 0;
-#X connect 7 0 8 1;
-#X connect 8 0 5 1;
-#X connect 10 0 28 0;
-#X connect 11 0 8 0;
-#X connect 12 0 14 0;
-#X connect 13 0 15 0;
-#X connect 14 0 10 0;
-#X connect 14 0 25 0;
-#X connect 14 0 25 1;
-#X connect 15 0 12 0;
-#X connect 16 0 0 0;
-#X connect 17 0 16 0;
-#X connect 18 0 17 0;
-#X connect 19 0 20 0;
-#X connect 20 0 15 1;
-#X connect 21 0 0 1;
-#X connect 22 0 39 0;
-#X connect 22 0 39 1;
-#X connect 23 0 10 1;
-#X connect 24 0 23 0;
-#X connect 25 0 26 0;
-#X connect 26 0 24 0;
-#X connect 27 0 20 1;
-#X connect 39 0 27 0;
-#X restore 28 174 pd env;
-#X obj 79 110 adc~ 9;
-#X obj 129 130 adc~ 10;
-#X text 72 90 piezo input for excitation;
-#X text 123 110 fader 1 sets threshold;
-#X text 183 130 fader 2 sets duration;
-#X text 29 17 SAMPLE-ACCURATE ENVELOPE TRIGGER;
-#X text 29 27 ================================;
-#X obj 28 90 adc~ 8;
-#X text 141 236 ---------------------------------;
-#X text 141 226 @krighxz / BELA / heavy / 12/2015;
-#X text 165 245 beaglert.cc / enzienaudio.com;
-#X connect 1 0 0 0;
-#X connect 1 0 0 1;
-#X connect 2 0 1 1;
-#X connect 3 0 1 2;
-#X connect 9 0 1 0;
--- a/examples/pd/hello-midi/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,81 +0,0 @@
-#N canvas 85 58 1280 685 10;
-#X obj 19 396 dac~ 1 2;
-#X text 39 1 Hello world!;
-#X text 37 11 ============;
-#X obj 31 353 *~ 0.1;
-#X text 202 448 ---------------------------------;
-#X text 202 438 @krighxz / BELA / heavy / 12/2015;
-#X text 226 457 beaglert.cc / enzienaudio.com;
-#X obj 39 132 mtof;
-#X floatatom 79 59 5 0 0 0 - - -, f 5;
-#X floatatom 122 59 5 0 0 0 - - -, f 5;
-#X floatatom 41 63 5 0 0 0 - - -, f 5;
-#X obj 26 170 phasor~ 440;
-#X text 100 363 (the harsh sound of success);
-#X obj 260 83 * 20;
-#X obj 261 112 + 200;
-#X obj 132 85 loadbang;
-#X msg 131 111 1000;
-#X obj 125 166 sig~;
-#X text 673 121 [pgmin];
-#X obj 218 178 * 10;
-#X obj 30 250 hip~;
-#X obj 26 275 hip~;
-#X obj 20 302 hip~;
-#X obj 19 323 hip~;
-#X obj 87 192 lop~ 4;
-#X obj 152 256 line;
-#X obj 200 200 + 50;
-#X msg 170 227 \$1 200;
-#X msg 93 114 0;
-#X msg 185 112 0;
-#X obj 29 224 vcf~ 4;
-#X obj 70 333 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
-1;
-#X obj 42 92 mono;
-#X obj 40 28 bela_notein 0;
-#X obj 260 60 bela_ctlin 5 0;
-#X obj 220 151 bela_ctlin 6 0;
-#X text 537 58 [ctlin];
-#X text 332 27 [notein];
-#X text 140 29 <== the bela equivalent of ==>;
-#X text 349 59 <== the bela equivalent of ==>;
-#X text 483 121 <== the bela equivalent of ==>;
-#X obj 405 175 print pgmin;
-#X obj 408 121 bela_pgmin 0;
-#X connect 3 0 0 1;
-#X connect 3 0 0 0;
-#X connect 7 0 11 0;
-#X connect 8 0 32 1;
-#X connect 10 0 32 0;
-#X connect 11 0 30 0;
-#X connect 13 0 14 0;
-#X connect 14 0 17 0;
-#X connect 15 0 16 0;
-#X connect 15 0 29 0;
-#X connect 15 0 28 0;
-#X connect 16 0 17 0;
-#X connect 17 0 24 0;
-#X connect 19 0 26 0;
-#X connect 20 0 21 0;
-#X connect 21 0 22 0;
-#X connect 22 0 23 0;
-#X connect 23 0 3 0;
-#X connect 24 0 30 1;
-#X connect 25 0 20 1;
-#X connect 25 0 21 1;
-#X connect 25 0 22 1;
-#X connect 25 0 23 1;
-#X connect 26 0 27 0;
-#X connect 27 0 25 0;
-#X connect 29 0 19 0;
-#X connect 30 0 20 0;
-#X connect 31 0 3 1;
-#X connect 32 0 7 0;
-#X connect 32 1 31 0;
-#X connect 33 0 10 0;
-#X connect 33 1 8 0;
-#X connect 33 2 9 0;
-#X connect 34 0 13 0;
-#X connect 35 0 19 0;
-#X connect 42 0 41 0;
--- a/examples/pd/hello-midi/countTo.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,33 +0,0 @@
-#N canvas 584 255 450 300 10;
-#X obj 65 196 outlet;
-#X obj 98 8 inlet;
-#X text 189 49 counts from 0 to the given number -1;
-#X text 214 95 inlet:N;
-#X text 212 112 outlet: 0 \, 1 \, 2 \, ... \, N-1;
-#X obj 143 61 f;
-#X obj 100 138 spigot;
-#X obj 129 89 <;
-#X obj 129 113 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
-1;
-#X obj 95 28 t b f b;
-#X obj 92 74 + 1;
-#X obj 43 73 f;
-#X obj 63 104 t b f f f;
-#X msg 92 53 0;
-#X obj 16 144 delay 0;
-#X connect 1 0 9 0;
-#X connect 5 0 7 1;
-#X connect 6 0 11 0;
-#X connect 7 0 8 0;
-#X connect 8 0 6 1;
-#X connect 9 0 11 0;
-#X connect 9 1 5 0;
-#X connect 9 1 13 0;
-#X connect 10 0 11 1;
-#X connect 11 0 12 0;
-#X connect 12 0 14 0;
-#X connect 12 1 0 0;
-#X connect 12 2 7 0;
-#X connect 12 3 10 0;
-#X connect 13 0 11 1;
-#X connect 14 0 6 0;
--- a/examples/pd/hello-midi/maxArray.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,67 +0,0 @@
-#N canvas 464 23 1002 583 10;
-#X obj 224 202 max;
-#X obj 265 200 f;
-#X obj 223 229 change;
-#X obj 223 251 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
--1 -1;
-#X obj 223 271 f;
-#X obj 259 323 f;
-#X obj 187 315 f;
-#X obj 268 282 t b b;
-#X obj 183 364 outlet;
-#X obj 269 379 outlet;
-#X obj 179 17 inlet;
-#X obj 225 177 tabread \$1;
-#X obj 230 88 t f f b;
-#X text 376 37 Finds a maximum value in an array;
-#X text 531 112 \$2 arrayLength;
-#X obj 292 10 inlet;
-#X text 425 146 inlets:;
-#X text 473 144 1 bang to find the maximum;
-#X text 475 157 2 arrayLength;
-#X text 407 90 creation arguments: \$1 arrayName;
-#X obj 246 38 \$2;
-#X obj 202 40 bang;
-#X obj 230 65 f;
-#X obj 224 6 loadbang;
-#X text 444 216 outlets: 1 maximum value;
-#X text 498 231 2 index of the maximum value;
-#X msg 276 141 -1e+15;
-#X obj 222 142 t f f f;
-#X floatatom 158 266 5 0 0 0 - - -, f 5;
-#X obj 332 302 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144
--1 -1;
-#X obj 288 253 select 127;
-#X floatatom 301 324 5 0 0 0 - - -, f 5;
-#X text 362 256 heavy does not support the second inlet of select so
-I have to hard code 127 in here;
-#X obj 223 110 countTo;
-#X connect 0 0 1 0;
-#X connect 0 0 2 0;
-#X connect 0 0 6 1;
-#X connect 0 0 28 0;
-#X connect 1 0 0 1;
-#X connect 2 0 3 0;
-#X connect 3 0 4 0;
-#X connect 4 0 5 1;
-#X connect 4 0 31 0;
-#X connect 5 0 9 0;
-#X connect 6 0 8 0;
-#X connect 7 0 6 0;
-#X connect 7 1 5 0;
-#X connect 10 0 21 0;
-#X connect 11 0 0 0;
-#X connect 12 0 33 0;
-#X connect 12 2 26 0;
-#X connect 15 0 22 1;
-#X connect 20 0 22 1;
-#X connect 21 0 22 0;
-#X connect 22 0 12 0;
-#X connect 23 0 20 0;
-#X connect 26 0 0 1;
-#X connect 27 0 30 0;
-#X connect 27 1 11 0;
-#X connect 27 2 4 1;
-#X connect 30 0 7 0;
-#X connect 30 0 29 0;
-#X connect 33 0 27 0;
--- a/examples/pd/hello-midi/mono.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,106 +0,0 @@
-#N canvas 69 182 1020 452 10;
-#N canvas 0 23 450 278 (subpatch) 0;
-#X array \$0-activeNotes 127 float 2;
-#X coords 0 100 127 -1 200 140 1 0 0;
-#X restore 537 25 graph;
-#N canvas 89 91 450 300 resetArray 1;
-#X obj 121 145 tabwrite \$0-activeNotes;
-#X msg 123 115 -0.5;
-#X obj 154 90 t b f;
-#X msg 129 45 128;
-#X obj 103 14 inlet;
-#X text 212 71 countto;
-#X obj 153 69 countTo;
-#X connect 1 0 0 0;
-#X connect 2 0 1 0;
-#X connect 2 1 0 1;
-#X connect 3 0 6 0;
-#X connect 4 0 3 0;
-#X connect 6 0 2 0;
-#X restore 83 49 pd resetArray;
-#X obj 83 3 loadbang;
-#X obj 88 25 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
--1;
-#X obj 39 188 maxArray \$0-activeNotes 128;
-#X obj 183 219 f;
-#X obj 39 237 >= 0;
-#X floatatom 148 313 5 0 0 0 - - -, f 5;
-#X obj 70 262 select 1;
-#X obj 366 87 f;
-#X obj 426 60 f;
-#X text 334 39 note;
-#X text 424 16 velocity;
-#X obj 423 87 > 0;
-#X obj 344 281 tabwrite \$0-activeNotes;
-#X obj 330 207 f;
-#X floatatom 303 178 5 0 0 0 - - -, f 5;
-#X obj 265 163 f;
-#X obj 304 150 + 1;
-#X msg 277 130 -1;
-#X msg 371 207 -0.5;
-#X obj 400 116 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
-1;
-#X obj 367 137 spigot;
-#X obj 447 134 spigot;
-#X obj 460 86 == 0;
-#X obj 470 109 tgl 15 0 empty empty empty 17 7 0 10 -262144 -1 -1 0
-1;
-#X obj 434 163 t b f;
-#X obj 337 60 inlet;
-#X obj 426 32 inlet;
-#X obj 148 350 outlet;
-#X obj 275 89 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
--1;
-#X obj 41 163 bng 15 250 50 0 empty empty empty 17 7 0 10 -262144 -1
--1;
-#X obj 363 172 t b f;
-#X obj 334 240 t b f;
-#X text 578 211 pass in velocity/note pairs (velocity first) and obtain
-as an output the most recent note still active;
-#X text 581 268 outlet 2 gate;
-#X obj 330 366 outlet;
-#X text 583 251 outlet 1 most recent note active (-1 otherwise);
-#X text 155 329 most recent note active;
-#X msg 338 338 0;
-#X msg 378 339 1;
-#X connect 2 0 3 0;
-#X connect 3 0 1 0;
-#X connect 3 0 19 0;
-#X connect 4 0 6 0;
-#X connect 4 1 5 1;
-#X connect 5 0 7 0;
-#X connect 5 0 40 0;
-#X connect 6 0 8 0;
-#X connect 7 0 29 0;
-#X connect 8 0 5 0;
-#X connect 8 1 39 0;
-#X connect 9 0 22 0;
-#X connect 9 0 23 0;
-#X connect 10 0 13 0;
-#X connect 10 0 24 0;
-#X connect 13 0 21 0;
-#X connect 15 0 33 0;
-#X connect 16 0 15 1;
-#X connect 17 0 18 0;
-#X connect 18 0 16 0;
-#X connect 18 0 17 1;
-#X connect 19 0 17 1;
-#X connect 20 0 33 0;
-#X connect 21 0 22 1;
-#X connect 22 0 32 0;
-#X connect 23 0 26 0;
-#X connect 24 0 25 0;
-#X connect 25 0 23 1;
-#X connect 26 0 20 0;
-#X connect 26 1 14 1;
-#X connect 27 0 9 0;
-#X connect 28 0 10 0;
-#X connect 28 0 30 0;
-#X connect 30 0 17 0;
-#X connect 31 0 4 0;
-#X connect 32 0 15 0;
-#X connect 32 1 14 1;
-#X connect 33 0 31 0;
-#X connect 33 1 14 0;
-#X connect 39 0 36 0;
-#X connect 40 0 36 0;
--- a/examples/pd/hello-world/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,13 +0,0 @@
-#N canvas 275 573 461 212 10;
-#X obj 39 143 dac~ 1 2;
-#X text 35 32 Hello world!;
-#X text 35 42 ============;
-#X text 95 79 (the sweet sound of success);
-#X obj 39 102 *~ 0.1;
-#X obj 39 81 osc~ 440;
-#X text 238 160 ---------------------------------;
-#X text 238 150 @krighxz / BELA / heavy / 12/2015;
-#X text 262 169 beaglert.cc / enzienaudio.com;
-#X connect 4 0 0 1;
-#X connect 4 0 0 0;
-#X connect 5 0 4 0;
--- a/examples/pd/karplusStrong/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,240 +0,0 @@
-#N canvas 440 516 715 490 10;
-#X obj 38 370 dac~ 1 2;
-#X text 470 442 ---------------------------------;
-#X text 470 432 @krighxz / BELA / heavy / 12/2015;
-#X text 494 451 beaglert.cc / enzienaudio.com;
-#X text 33 22 Karplus Strong;
-#X text 33 32 ==============;
-#X obj 38 323 vd~ \$0-wg1;
-#X obj 118 345 lop~ 1000;
-#X obj 118 367 *~ 0.99;
-#X obj 118 389 s~ \$0-fb1;
-#X obj 38 94 adc~ 3;
-#X obj 38 116 hip~ 100;
-#X obj 98 176 r~ \$0-fb1;
-#X obj 38 198 delwrite~ \$0-wg1 100;
-#X obj 38 268 *~;
-#X obj 38 290 *~ 10;
-#X text 99 116 <<< centre (i.e. DC block) signal with high-pass filter
-;
-#X text 100 94 <<< use accelerometer \, piezo or FSR on first analog
-in;
-#X text 105 245 <<< use potentiometer or fader on second analog in
-;
-#X text 100 75 EXCITATION INPUT:;
-#X text 100 225 DELAY PERIOD:;
-#X text 182 365 <<< and decay here [0-0.9999];
-#X text 182 345 <<< set damping here [~100-20000];
-#N canvas 335 566 461 274 gate 0;
-#N canvas 0 22 450 278 (subpatch) 0;
-#X array \$0-xfer 1024 float 3;
-#A 0 0 -0.808242 -0.804734 -0.801234 -0.797741 -0.794256 -0.790778
--0.787309 -0.783846 -0.780392 -0.776945 -0.773506 -0.770074 -0.76665
--0.763233 -0.759824 -0.756423 -0.75303 -0.749644 -0.746266 -0.742895
--0.739532 -0.736176 -0.732829 -0.729488 -0.726156 -0.722831 -0.719514
--0.716204 -0.712902 -0.709608 -0.706321 -0.703042 -0.69977 -0.696507
--0.69325 -0.690002 -0.686761 -0.683527 -0.680302 -0.677084 -0.673873
--0.67067 -0.667475 -0.664288 -0.661108 -0.657935 -0.654771 -0.651614
--0.648464 -0.645323 -0.642188 -0.639062 -0.635943 -0.632832 -0.629728
--0.626632 -0.623544 -0.620463 -0.61739 -0.614324 -0.611266 -0.608216
--0.605174 -0.602139 -0.599111 -0.596092 -0.59308 -0.590075 -0.587078
--0.584089 -0.581107 -0.578134 -0.575167 -0.572209 -0.569257 -0.566314
--0.563378 -0.56045 -0.55753 -0.554617 -0.551711 -0.548814 -0.545924
--0.543041 -0.540167 -0.537299 -0.53444 -0.531588 -0.528744 -0.525907
--0.523078 -0.520257 -0.517443 -0.514637 -0.511839 -0.509048 -0.506265
--0.503489 -0.500721 -0.497961 -0.495208 -0.492463 -0.489726 -0.486996
--0.484274 -0.481559 -0.478852 -0.476153 -0.473461 -0.470777 -0.468101
--0.465432 -0.462771 -0.460117 -0.457472 -0.454833 -0.452203 -0.44958
--0.446964 -0.444357 -0.441757 -0.439164 -0.436579 -0.434002 -0.431432
--0.42887 -0.426316 -0.42377 -0.42123 -0.418699 -0.416175 -0.413659
--0.41115 -0.40865 -0.406156 -0.403671 -0.401193 -0.398722 -0.396259
--0.393804 -0.391357 -0.388917 -0.386485 -0.38406 -0.381643 -0.379234
--0.376832 -0.374438 -0.372051 -0.369673 -0.367301 -0.364938 -0.362582
--0.360233 -0.357893 -0.35556 -0.353234 -0.350916 -0.348606 -0.346304
--0.344009 -0.341721 -0.339442 -0.33717 -0.334905 -0.332649 -0.330399
--0.328158 -0.325924 -0.323698 -0.321479 -0.319268 -0.317065 -0.314869
--0.312681 -0.3105 -0.308328 -0.306162 -0.304005 -0.301855 -0.299713
--0.297578 -0.295451 -0.293331 -0.291219 -0.289115 -0.287019 -0.28493
--0.282848 -0.280775 -0.278709 -0.27665 -0.2746 -0.272556 -0.270521
--0.268493 -0.266473 -0.26446 -0.262455 -0.260458 -0.258468 -0.256486
--0.254511 -0.252545 -0.250585 -0.248634 -0.24669 -0.244753 -0.242825
--0.240904 -0.23899 -0.237084 -0.235186 -0.233296 -0.231413 -0.229537
--0.22767 -0.22581 -0.223957 -0.222112 -0.220275 -0.218446 -0.216624
--0.21481 -0.213003 -0.211204 -0.209413 -0.207629 -0.205853 -0.204084
--0.202323 -0.20057 -0.198824 -0.197086 -0.195356 -0.193633 -0.191918
--0.190211 -0.188511 -0.186819 -0.185134 -0.183457 -0.181788 -0.180126
--0.178472 -0.176826 -0.175187 -0.173556 -0.171932 -0.170316 -0.168708
--0.167108 -0.165515 -0.163929 -0.162351 -0.160781 -0.159219 -0.157664
--0.156117 -0.154577 -0.153045 -0.151521 -0.150004 -0.148495 -0.146993
--0.1455 -0.144013 -0.142535 -0.141064 -0.139601 -0.138145 -0.136697
--0.135256 -0.133824 -0.132398 -0.130981 -0.129571 -0.128169 -0.126774
--0.125387 -0.124008 -0.122636 -0.121272 -0.119915 -0.118566 -0.117225
--0.115891 -0.114565 -0.113247 -0.111936 -0.110633 -0.109338 -0.10805
--0.10677 -0.105497 -0.104232 -0.102975 -0.101725 -0.100483 -0.0992487
--0.0980219 -0.0968027 -0.0955911 -0.0943872 -0.0931909 -0.0920023 -0.0908212
--0.0896478 -0.0884821 -0.0873239 -0.0861734 -0.0850305 -0.0838953 -0.0827676
--0.0816476 -0.0805353 -0.0794305 -0.0783334 -0.077244 -0.0761621 -0.0750879
--0.0740213 -0.0729623 -0.071911 -0.0708673 -0.0698312 -0.0688028 -0.067782
--0.0667688 -0.0657632 -0.0647653 -0.063775 -0.0627924 -0.0618173 -0.0608499
--0.0598901 -0.058938 -0.0579935 -0.0570566 -0.0561273 -0.0552057 -0.0542917
--0.0533853 -0.0524866 -0.0515955 -0.050712 -0.0498361 -0.0489679 -0.0481073
--0.0472543 -0.046409 -0.0455713 -0.0447412 -0.0439188 -0.0431039 -0.0422968
--0.0414972 -0.0407053 -0.039921 -0.0391443 -0.0383752 -0.0376138 -0.03686
--0.0361139 -0.0353754 -0.0346445 -0.0339212 -0.0332056 -0.0324976 -0.0317972
--0.0311044 -0.0304193 -0.0297418 -0.029072 -0.0284097 -0.0277551 -0.0271082
--0.0264688 -0.0258371 -0.025213 -0.0245966 -0.0239877 -0.0233865 -0.022793
--0.022207 -0.0216287 -0.021058 -0.020495 -0.0199396 -0.0193918 -0.0188516
--0.0183191 -0.0177942 -0.0172769 -0.0167673 -0.0162653 -0.0157709 -0.0152841
--0.014805 -0.0143335 -0.0138696 -0.0134134 -0.0129648 -0.0125238 -0.0120905
--0.0116647 -0.0112466 -0.0108362 -0.0104333 -0.0100381 -0.00965057
--0.00927063 -0.00889832 -0.00853363 -0.00817657 -0.00782715 -0.00748535
--0.00715118 -0.00682465 -0.00650574 -0.00619446 -0.00589081 -0.00559479
--0.0053064 -0.00502563 -0.0047525 -0.004487 -0.00422913 -0.00397888
--0.00373627 -0.00350128 -0.00327393 -0.0030542 -0.0028421 -0.00263763
--0.0024408 -0.00225159 -0.00207001 -0.00189606 -0.00172974 -0.00157104
--0.00141998 -0.00127655 -0.00114075 -0.00101257 -0.000892029 -0.000779114
--0.000673828 -0.000576172 -0.000486145 -0.000403747 -0.000328979 -0.000261841
--0.000202332 -0.000150452 -0.000106201 -6.95801e-05 -4.05884e-05 -1.92261e-05
--5.49316e-06 0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0
--0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0 -0
--0 -0 -0 -0 -0 -0 -0 -0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
-0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 5.49316e-06
-1.92261e-05 4.05884e-05 6.95801e-05 0.000106201 0.000150452 0.000202332
-0.000261841 0.000328979 0.000403747 0.000486145 0.000576172 0.000673828
-0.000779114 0.000892029 0.00101257 0.00114075 0.00127655 0.00141998
-0.00157104 0.00172974 0.00189606 0.00207001 0.00225159 0.0024408 0.00263763
-0.0028421 0.0030542 0.00327393 0.00350128 0.00373627 0.00397888 0.00422913
-0.004487 0.0047525 0.00502563 0.0053064 0.00559479 0.00589081 0.00619446
-0.00650574 0.00682465 0.00715118 0.00748535 0.00782715 0.00817657 0.00853363
-0.00889832 0.00927063 0.00965057 0.0100381 0.0104333 0.0108362 0.0112466
-0.0116647 0.0120905 0.0125238 0.0129648 0.0134134 0.0138696 0.0143335
-0.014805 0.0152841 0.0157709 0.0162653 0.0167673 0.0172769 0.0177942
-0.0183191 0.0188516 0.0193918 0.0199396 0.020495 0.021058 0.0216287
-0.022207 0.022793 0.0233865 0.0239877 0.0245966 0.025213 0.0258371
-0.0264688 0.0271082 0.0277551 0.0284097 0.029072 0.0297418 0.0304193
-0.0311044 0.0317972 0.0324976 0.0332056 0.0339212 0.0346445 0.0353754
-0.0361139 0.03686 0.0376138 0.0383752 0.0391443 0.039921 0.0407053
-0.0414972 0.0422968 0.0431039 0.0439188 0.0447412 0.0455713 0.046409
-0.0472543 0.0481073 0.0489679 0.0498361 0.050712 0.0515955 0.0524866
-0.0533853 0.0542917 0.0552057 0.0561273 0.0570566 0.0579935 0.058938
-0.0598901 0.0608499 0.0618173 0.0627924 0.063775 0.0647653 0.0657632
-0.0667688 0.067782 0.0688028 0.0698312 0.0708673 0.071911 0.0729623
-0.0740213 0.0750879 0.0761621 0.077244 0.0783334 0.0794305 0.0805353
-0.0816476 0.0827676 0.0838953 0.0850305 0.0861734 0.0873239 0.0884821
-0.0896478 0.0908212 0.0920023 0.0931909 0.0943872 0.0955911 0.0968027
-0.0980219 0.0992487 0.100483 0.101725 0.102975 0.104232 0.105497 0.10677
-0.10805 0.109338 0.110633 0.111936 0.113247 0.114565 0.115891 0.117225
-0.118566 0.119915 0.121272 0.122636 0.124008 0.125387 0.126774 0.128169
-0.129571 0.130981 0.132398 0.133824 0.135256 0.136697 0.138145 0.139601
-0.141064 0.142535 0.144013 0.1455 0.146993 0.148495 0.150004 0.151521
-0.153045 0.154577 0.156117 0.157664 0.159219 0.160781 0.162351 0.163929
-0.165515 0.167108 0.168708 0.170316 0.171932 0.173556 0.175187 0.176826
-0.178472 0.180126 0.181788 0.183457 0.185134 0.186819 0.188511 0.190211
-0.191918 0.193633 0.195356 0.197086 0.198824 0.20057 0.202323 0.204084
-0.205853 0.207629 0.209413 0.211204 0.213003 0.21481 0.216624 0.218446
-0.220275 0.222112 0.223957 0.22581 0.22767 0.229537 0.231413 0.233296
-0.235186 0.237084 0.23899 0.240904 0.242825 0.244753 0.24669 0.248634
-0.250585 0.252545 0.254511 0.256486 0.258468 0.260458 0.262455 0.26446
-0.266473 0.268493 0.270521 0.272556 0.2746 0.27665 0.278709 0.280775
-0.282848 0.28493 0.287019 0.289115 0.291219 0.293331 0.295451 0.297578
-0.299713 0.301855 0.304005 0.306162 0.308328 0.3105 0.312681 0.314869
-0.317065 0.319268 0.321479 0.323698 0.325924 0.328158 0.330399 0.332649
-0.334905 0.33717 0.339442 0.341721 0.344009 0.346304 0.348606 0.350916
-0.353234 0.35556 0.357893 0.360233 0.362582 0.364938 0.367301 0.369673
-0.372051 0.374438 0.376832 0.379234 0.381643 0.38406 0.386485 0.388917
-0.391357 0.393804 0.396259 0.398722 0.401193 0.403671 0.406156 0.40865
-0.41115 0.413659 0.416175 0.418699 0.42123 0.42377 0.426316 0.42887
-0.431432 0.434002 0.436579 0.439164 0.441757 0.444357 0.446964 0.44958
-0.452203 0.454833 0.457472 0.460117 0.462771 0.465432 0.468101 0.470777
-0.473461 0.476153 0.478852 0.481559 0.484274 0.486996 0.489726 0.492463
-0.495208 0.497961 0.500721 0.503489 0.506265 0.509048 0.511839 0.514637
-0.517443 0.520257 0.523078 0.525907 0.528744 0.531588 0.53444 0.537299
-0.540167 0.543041 0.545924 0.548814 0.551711 0.554617 0.55753 0.56045
-0.563378 0.566314 0.569257 0.572209 0.575167 0.578134 0.581107 0.584089
-0.587078 0.590075 0.59308 0.596092 0.599111 0.602139 0.605174 0.608216
-0.611266 0.614324 0.61739 0.620463 0.623544 0.626632 0.629728 0.632832
-0.635943 0.639062 0.642188 0.645323 0.648464 0.651614 0.654771 0.657935
-0.661108 0.664288 0.667475 0.67067 0.673873 0.677084 0.680302 0.683527
-0.686761 0.690002 0.69325 0.696507 0.69977 0.703042 0.706321 0.709608
-0.712902 0.716204 0.719514 0.722831;
-#A 1000 0.726156 0.729488 0.732829 0.736176 0.739532 0.742895 0.746266
-0.749644 0.75303 0.756423 0.759824 0.763233 0.76665 0.770074 0.773506
-0.776945 0.780392 0.783846 0.787309 0.790778 0.794256 0.797741 0.801234
-0.804734;
-#X coords 0 1 1023 -1 200 200 1 0 0;
-#X restore 230 34 graph;
-#X obj 25 27 inlet~;
-#X obj 25 49 clip~ -1 1;
-#X obj 25 71 *~ 512;
-#X obj 25 93 +~ 512;
-#X obj 25 115 tabread4~ \$0-xfer;
-#X obj 25 137 outlet~;
-#N canvas 0 22 334 332 generate-xfer 0;
-#X obj 8 290 tabwrite \$0-xfer;
-#X obj 8 109 / 1024;
-#X obj 8 129 * 2;
-#X obj 8 149 - 1;
-#X obj 8 169 moses 0;
-#X obj 8 191 + 0.1;
-#X obj 8 213 clip -1 0;
-#X obj 68 191 - 0.1;
-#X obj 68 213 clip 0 1;
-#X obj 101 16 inlet threshold;
-#X obj 101 78 count 1024;
-#X obj 128 58 s \$0-thresh;
-#X obj 95 169 r \$0-thresh;
-#X obj 101 38 t b f;
-#X obj 68 236 *;
-#X obj 8 236 *;
-#X obj 8 258 * -1;
-#X text 140 203 <<< deadzone;
-#X text 140 253 <<< smooth by taking exponent;
-#X connect 1 0 2 0;
-#X connect 2 0 3 0;
-#X connect 3 0 4 0;
-#X connect 4 0 5 0;
-#X connect 4 1 7 0;
-#X connect 5 0 6 0;
-#X connect 6 0 15 0;
-#X connect 6 0 15 1;
-#X connect 7 0 8 0;
-#X connect 8 0 14 0;
-#X connect 8 0 14 1;
-#X connect 9 0 13 0;
-#X connect 10 0 0 1;
-#X connect 10 0 1 0;
-#X connect 12 0 5 1;
-#X connect 12 0 7 1;
-#X connect 13 0 10 0;
-#X connect 13 1 11 0;
-#X connect 14 0 0 0;
-#X connect 15 0 16 0;
-#X connect 16 0 0 0;
-#X restore 26 217 pd generate-xfer;
-#X text 58 196 <<< threshold;
-#X obj 26 173 loadbang;
-#X msg 26 195 0.1;
-#X connect 1 0 2 0;
-#X connect 2 0 3 0;
-#X connect 3 0 4 0;
-#X connect 4 0 5 0;
-#X connect 5 0 6 0;
-#X connect 9 0 10 0;
-#X connect 10 0 7 0;
-#X restore 38 137 pd gate;
-#X obj 38 246 adc~ 10;
-#X text 98 137 <<< thresholding to remove any consistent noise in the
-excitation signal;
-#X connect 6 0 7 0;
-#X connect 6 0 0 0;
-#X connect 6 0 0 1;
-#X connect 7 0 8 0;
-#X connect 8 0 9 0;
-#X connect 10 0 11 0;
-#X connect 11 0 23 0;
-#X connect 12 0 13 0;
-#X connect 14 0 15 0;
-#X connect 15 0 6 0;
-#X connect 23 0 13 0;
-#X connect 24 0 14 0;
-#X connect 24 0 14 1;
--- a/examples/pd/karplusStrong/count.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,30 +0,0 @@
-#N canvas 592 177 306 227 10;
-#X obj 32 111 f;
-#X obj 57 111 + 1;
-#X obj 57 151 sel 0;
-#X obj 57 171 del 1e-36;
-#X obj 57 131 > 1023;
-#X msg 32 90 0;
-#X obj 19 171 - 1;
-#X obj 19 193 outlet;
-#X obj 90 87 \$1;
-#X obj 90 109 - 1;
-#X obj 90 66 loadbang;
-#X obj 32 48 inlet;
-#X obj 32 69 t b;
-#X text 33 14 temporary replacement for [until] object;
-#X text 67 195 outputs index (0->[N-1]);
-#X connect 0 0 1 0;
-#X connect 1 0 0 1;
-#X connect 1 0 4 0;
-#X connect 1 0 6 0;
-#X connect 2 0 3 0;
-#X connect 3 0 0 0;
-#X connect 4 0 2 0;
-#X connect 5 0 0 0;
-#X connect 6 0 7 0;
-#X connect 8 0 9 0;
-#X connect 9 0 4 1;
-#X connect 10 0 8 0;
-#X connect 11 0 12 0;
-#X connect 12 0 5 0;
--- a/examples/pd/rubberDuckie/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,247 +0,0 @@
-#N canvas 275 573 442 207 10;
-#X obj 39 149 dac~ 1 2;
-#X text 218 143 ---------------------------------;
-#X text 218 133 @krighxz / BELA / heavy / 12/2015;
-#X text 242 152 beaglert.cc / enzienaudio.com;
-#X text 35 32 Rubber Duckie;
-#X text 35 42 =============;
-#N canvas 325 86 823 605 duckie 0;
-#X obj 186 510 *~;
-#X obj 306 517 *~;
-#X obj 372 468 noise~;
-#X obj 372 492 bp~ 3000 10;
-#X obj 366 162 abs~;
-#X obj 509 158 sig~ 0;
-#X obj 467 371 +~;
-#X obj 411 210 -~ 0.55;
-#X obj 411 231 /~ 0.2;
-#X obj 411 252 clip~ 0 1;
-#X obj 411 282 *~;
-#X obj 411 308 *~ 1000;
-#X obj 366 334 +~;
-#X obj 366 308 +~ 800;
-#X obj 186 536 *~ 0.2;
-#X obj 366 282 *~ 800;
-#N canvas 568 775 135 226 <~ 0;
-#X obj 20 -43 -~;
-#X obj 20 27 +~ 1e-37;
-#X obj 20 6 clip~ -1e-37 0;
-#X obj 20 -88 inlet~;
-#X obj 64 -87 inlet~;
-#X obj 20 -64 max~;
-#X obj 20 -19 *~ -1;
-#X obj 20 79 outlet~;
-#X obj 20 48 *~ 1e+37;
-#X connect 0 0 6 0;
-#X connect 1 0 8 0;
-#X connect 2 0 1 0;
-#X connect 3 0 5 0;
-#X connect 4 0 0 1;
-#X connect 4 0 5 1;
-#X connect 5 0 0 0;
-#X connect 6 0 2 0;
-#X connect 8 0 7 0;
-#X restore 186 367 pd <~ 0;
-#N canvas 568 775 135 226 <~ 0;
-#X obj 20 -43 -~;
-#X obj 20 27 +~ 1e-37;
-#X obj 20 6 clip~ -1e-37 0;
-#X obj 20 -88 inlet~;
-#X obj 64 -87 inlet~;
-#X obj 20 -64 max~;
-#X obj 20 -19 *~ -1;
-#X obj 20 79 outlet~;
-#X obj 20 48 *~ 1e+37;
-#X connect 0 0 6 0;
-#X connect 1 0 8 0;
-#X connect 2 0 1 0;
-#X connect 3 0 5 0;
-#X connect 4 0 0 1;
-#X connect 4 0 5 1;
-#X connect 5 0 0 0;
-#X connect 6 0 2 0;
-#X connect 8 0 7 0;
-#X restore 482 181 pd <~;
-#X obj 186 428 *~;
-#X obj 225 345 sig~ 1;
-#X obj 186 394 lop~ 10;
-#X text 524 393 << band-limited pulse;
-#X text 10 367 reed stops when overblown >>;
-#X obj 186 460 *~;
-#X obj 186 480 *~;
-#X text 25 470 exp amplitude response >>;
-#X text 537 238 << detune frequency on change of direction;
-#X obj 482 236 *~ -300;
-#X obj 306 485 *~;
-#X obj 306 449 *~ 0.1;
-#X text 448 493 << noise simulates non-oscillating airflow when overblown
-;
-#X obj 186 561 outlet~;
-#N canvas 58 1286 407 525 pulse 0;
-#X obj 83 340 *~;
-#X obj 83 366 *~;
-#X obj 31 366 sig~ 1;
-#X obj 52 419 /~;
-#X obj 67 393 +~;
-#X text 116 366 X^2;
-#X text 97 396 1+X^2;
-#X text 84 421 1/(1+X^2);
-#X obj 83 317 osc~;
-#X obj 167 320 clip~ 0 999999;
-#X obj 52 483 outlet~;
-#X text 271 218 << tweak all this;
-#X text 25 14 based on F04.waveshaping.pulse.pd;
-#X obj 83 84 /~ 2;
-#X obj 167 148 -~;
-#X obj 167 170 /~ 127;
-#X obj 52 452 hip~ 5;
-#X obj 167 202 *~;
-#X obj 167 232 *~;
-#X obj 167 262 *~;
-#X obj 167 294 *~ 1000;
-#X obj 118 125 sig~ 139;
-#X obj 83 58 inlet~;
-#N canvas 331 1377 215 174 _ftom 0;
-#X obj -482 -776 tabread4~ _ftom;
-#X obj -482 -818 /~ 10000;
-#X obj -482 -797 *~ 256;
-#N canvas 0 23 450 278 (subpatch) 0;
-#X array _ftom 256 float 3;
-#A 0 0 27.0762 39.0762 46.0958 51.0762 54.9394 58.0958 60.7645 63.0762
-65.1153 66.9394 68.5894 70.0958 71.4815 72.7645 73.9589 75.0762 76.1258
-77.1153 78.0514 78.9394 79.784 80.5894 81.359 82.0958 82.8025 83.4815
-84.1349 84.7645 85.372 85.9589 86.5266 87.0762 87.609 88.1258 88.6276
-89.1153 89.5897 90.0514 90.5011 90.9394 91.3669 91.784 92.1914 92.5894
-92.9785 93.359 93.7313 94.0958 94.4528 94.8025 95.1453 95.4815 95.8113
-96.1349 96.4525 96.7645 97.0709 97.372 97.6679 97.9589 98.2451 98.5266
-98.8036 99.0762 99.3446 99.609 99.8693 100.126 100.379 100.628 100.873
-101.115 101.354 101.59 101.822 102.051 102.278 102.501 102.722 102.939
-103.154 103.367 103.577 103.784 103.989 104.191 104.392 104.589 104.785
-104.978 105.17 105.359 105.546 105.731 105.914 106.096 106.275 106.453
-106.629 106.803 106.975 107.145 107.314 107.482 107.647 107.811 107.974
-108.135 108.294 108.453 108.609 108.764 108.918 109.071 109.222 109.372
-109.521 109.668 109.814 109.959 110.103 110.245 110.386 110.527 110.666
-110.804 110.94 111.076 111.211 111.345 111.477 111.609 111.74 111.869
-111.998 112.126 112.253 112.379 112.504 112.628 112.751 112.873 112.995
-113.115 113.235 113.354 113.472 113.59 113.706 113.822 113.937 114.051
-114.165 114.278 114.39 114.501 114.612 114.722 114.831 114.939 115.047
-115.154 115.261 115.367 115.472 115.577 115.681 115.784 115.887 115.989
-116.09 116.191 116.292 116.392 116.491 116.589 116.688 116.785 116.882
-116.978 117.074 117.17 117.265 117.359 117.453 117.546 117.639 117.731
-117.823 117.915 118.005 118.096 118.186 118.275 118.364 118.453 118.541
-118.629 118.716 118.803 118.889 118.975 119.06 119.145 119.23 119.314
-119.398 119.482 119.565 119.647 119.729 119.811 119.893 119.974 120.055
-120.135 120.215 120.294 120.374 120.453 120.531 120.609 120.687 120.764
-120.842 120.918 120.995 121.071 121.147 121.222 121.297 121.372 121.446
-121.521 121.594 121.668 121.741 121.814 121.887 121.959 122.031 122.103
-122.174 122.245 122.316 122.386 122.457 122.527 122.596 122.666 122.735
-122.804 122.872 122.94 123.008;
-#X coords 0 127 256 -12 50 50 1 0 0;
-#X restore -546 -793 graph;
-#X obj -482 -838 clip~ 0 10000;
-#X obj -482 -858 inlet~;
-#X obj -482 -755 outlet~;
-#X text -567 -722 shoddy temp replacement for ftom~;
-#X connect 0 0 6 0;
-#X connect 1 0 2 0;
-#X connect 2 0 0 0;
-#X connect 4 0 1 0;
-#X connect 5 0 4 0;
-#X restore 182 125 pd _ftom;
-#X connect 0 0 1 0;
-#X connect 0 0 1 1;
-#X connect 1 0 4 1;
-#X connect 2 0 3 0;
-#X connect 2 0 4 0;
-#X connect 3 0 16 0;
-#X connect 4 0 3 1;
-#X connect 8 0 0 0;
-#X connect 9 0 0 1;
-#X connect 13 0 8 0;
-#X connect 13 0 23 0;
-#X connect 14 0 15 0;
-#X connect 15 0 17 0;
-#X connect 15 0 17 1;
-#X connect 16 0 10 0;
-#X connect 17 0 18 0;
-#X connect 17 0 18 1;
-#X connect 18 0 19 0;
-#X connect 18 0 19 1;
-#X connect 19 0 20 0;
-#X connect 20 0 9 0;
-#X connect 21 0 14 0;
-#X connect 22 0 13 0;
-#X connect 23 0 14 1;
-#X restore 467 394 pd pulse;
-#X obj 366 105 inlet~;
-#X text 295 104 airflow >>;
-#X text 218 118 ('squeeze pressure');
-#X text 13 20 Sound model of a rubber duckie toy;
-#X text 14 51 Hint: use the differential of a continuous signal as
-input for a realistic response;
-#X text 13 29 ==================================;
-#X obj 366 133 lop~ 0.5;
-#X text 423 132 << airflow resistance;
-#X connect 0 0 14 0;
-#X connect 1 0 14 0;
-#X connect 2 0 3 0;
-#X connect 3 0 1 1;
-#X connect 4 0 7 0;
-#X connect 4 0 15 0;
-#X connect 4 0 29 0;
-#X connect 4 0 18 1;
-#X connect 4 0 16 0;
-#X connect 5 0 17 1;
-#X connect 6 0 32 0;
-#X connect 7 0 8 0;
-#X connect 8 0 9 0;
-#X connect 9 0 10 0;
-#X connect 9 0 10 1;
-#X connect 10 0 11 0;
-#X connect 11 0 12 1;
-#X connect 12 0 6 0;
-#X connect 13 0 12 0;
-#X connect 14 0 31 0;
-#X connect 15 0 13 0;
-#X connect 16 0 20 0;
-#X connect 17 0 27 0;
-#X connect 18 0 23 0;
-#X connect 18 0 23 1;
-#X connect 19 0 16 1;
-#X connect 20 0 18 0;
-#X connect 23 0 24 1;
-#X connect 23 0 24 0;
-#X connect 24 0 0 0;
-#X connect 27 0 6 1;
-#X connect 28 0 1 0;
-#X connect 29 0 28 1;
-#X connect 29 0 28 0;
-#X connect 32 0 0 1;
-#X connect 33 0 39 0;
-#X connect 39 0 4 0;
-#X connect 39 0 17 0;
-#X restore 39 127 pd duckie;
-#X obj 39 77 adc~ 10;
-#N canvas 0 22 504 229 differential 0;
-#X obj 27 80 rzero~ 1;
-#X obj 72 57 sig~ 1;
-#X obj 27 152 *~ 44100;
-#X obj 72 131 samplerate~;
-#X obj 72 110 loadbang;
-#X obj 27 25 inlet~;
-#X obj 27 190 outlet~;
-#X text 84 83 y[n] = x[n] - x[n-1];
-#X text 121 57 <<< heavy currently requires signal input to set coefficient
-;
-#X text 91 151 <<< multiply by samplerate to get velocity;
-#X connect 0 0 2 0;
-#X connect 1 0 0 1;
-#X connect 2 0 6 0;
-#X connect 3 0 2 1;
-#X connect 4 0 3 0;
-#X connect 5 0 0 0;
-#X restore 39 101 pd differential;
-#X connect 6 0 0 0;
-#X connect 6 0 0 1;
-#X connect 7 0 8 0;
-#X connect 8 0 6 0;
--- a/examples/pd/samphold/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,22 +0,0 @@
-#N canvas 275 504 414 247 10;
-#X obj 55 212 dac~ 1 2;
-#X obj 55 171 *~ 0.1;
-#X obj 55 150 osc~ 440;
-#X obj 55 61 noise~;
-#X obj 55 128 samphold~;
-#X text 51 21 ==================;
-#X text 51 11 Sample and hold FM;
-#X obj 55 85 lop~ 1;
-#X obj 122 107 phasor~ 880;
-#X obj 55 107 *~ 1e+06;
-#X text 193 199 ---------------------------------;
-#X text 193 189 @krighxz / BELA / heavy / 12/2015;
-#X text 217 208 beaglert.cc / enzienaudio.com;
-#X connect 1 0 0 1;
-#X connect 1 0 0 0;
-#X connect 2 0 1 0;
-#X connect 3 0 7 0;
-#X connect 4 0 2 0;
-#X connect 7 0 9 0;
-#X connect 8 0 4 1;
-#X connect 9 0 4 0;
--- a/examples/pd/simple-fm/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,42 +0,0 @@
-#N canvas 275 573 409 282 10;
-#X obj 37 230 dac~ 1 2;
-#X text 175 222 ---------------------------------;
-#X obj 37 102 *~ 1000;
-#X obj 37 199 *~ 0.1;
-#X obj 37 79 adc~ 3;
-#X obj 65 125 adc~ 4;
-#X obj 94 147 adc~ 5;
-#N canvas 81 574 322 252 FMvoice 1;
-#X obj 20 174 +~;
-#X obj 93 116 osc~;
-#X obj 93 157 *~;
-#X obj 93 76 *~;
-#X obj 158 98 *~;
-#X obj 20 225 outlet~;
-#X obj 20 200 osc~;
-#X obj 20 39 inlet~ freq;
-#X obj 108 39 inlet~ harmRatio~;
-#X obj 173 69 inlet~ modIndex~;
-#X connect 0 0 6 0;
-#X connect 1 0 2 0;
-#X connect 2 0 0 1;
-#X connect 3 0 4 0;
-#X connect 3 0 1 0;
-#X connect 4 0 2 1;
-#X connect 6 0 5 0;
-#X connect 7 0 3 0;
-#X connect 7 0 0 0;
-#X connect 8 0 3 1;
-#X connect 9 0 4 1;
-#X restore 37 169 pd FMvoice;
-#X text 33 22 Simple FM;
-#X text 33 32 =========;
-#X text 175 212 @krighxz / BELA / heavy / 12/2015;
-#X text 199 231 beaglert.cc / enzienaudio.com;
-#X connect 2 0 7 0;
-#X connect 3 0 0 1;
-#X connect 3 0 0 0;
-#X connect 4 0 2 0;
-#X connect 5 0 7 1;
-#X connect 6 0 7 2;
-#X connect 7 0 3 0;
--- a/examples/pd/tableScrubbing/_main.pd	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,35 +0,0 @@
-#N canvas 503 319 861 473 10;
-#N canvas 0 22 450 278 (subpatch) 0;
-#X array exampleTable 65238 float 2;
-#X coords 0 1 65238 -1 200 140 1 0 0;
-#X restore 21 213 graph;
-#X obj 243 329 adc~ 1 2 3;
-#X obj 300 426 dac~;
-#X text 317 264 <<< delete this object before compiling!;
-#X obj 300 392 tabread4~ exampleTable;
-#X msg 243 218 read -resize rt.wav exampleTable;
-#X obj 300 350 lop~ 1;
-#X obj 300 371 *~ 65238;
-#X obj 243 266 soundfiler;
-#X text 489 229 right-click array -> properties -> toggle "save contents"
-;
-#X text 463 216 <<< after loading a file:;
-#X text 17 23 Load sample into table and scrub using analog input;
-#X text 17 33 ===================================================;
-#X text 362 351 <<< avoid zipper noise;
-#X text 362 371 <<< length of sample;
-#X text 317 328 <<< analog input 0 (range 0-1);
-#X text 631 419 ---------------------------------;
-#X text 631 409 @krighxz / BELA / heavy / 12/2015;
-#X text 655 428 beaglert.cc / enzienaudio.com;
-#X text 30 61 N.B. the patch cannot be bigger than 512kb in size \;
-only use this for *very small soundfiles*;
-#X text 29 99 You can modify the render.cpp file to load samples into
-tables using the Heavy API: https://enzienaudio.com/docs/c.html#api-hv-table
-;
-#X connect 1 2 6 0;
-#X connect 4 0 2 0;
-#X connect 4 0 2 1;
-#X connect 5 0 8 0;
-#X connect 6 0 7 0;
-#X connect 7 0 4 0;
Binary file examples/pd/tableScrubbing/rt.wav has changed
--- a/examples/samples/SampleData.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,19 +0,0 @@
-/*
- * SampleData.h
- *
- *  Created on: Nov 5, 2014
- *      Author: Victor Zappi
- */
-
-#ifndef SAMPLEDATA_H_
-#define SAMPLEDATA_H_
-
-// User defined structure to pass between main and rendere complex data retrieved from file
-struct SampleData {
-	float *samples;	// Samples in file
-	int sampleLen;	// Total nume of samples
-};
-
-
-
-#endif /* SAMPLEDATA_H_ */
--- a/examples/samples/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,179 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: Andrew McPherson and Victor Zappi
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <string>
-#include <getopt.h>
-#include <sndfile.h>				// to load audio files
-
-#include <Bela.h>
-#include "SampleData.h"
-
-using namespace std;
-
-// Load samples from file
-int initFile(string file, SampleData *smp)//float *& smp)
-{
-	SNDFILE *sndfile ;
-	SF_INFO sfinfo ;
-
-	if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
-		cout << "Couldn't open file " << file << endl;
-		return 1;
-	}
-
-	int numChan = sfinfo.channels;
-	if(numChan != 1)
-	{
-		cout << "Error: " << file << " is not a mono file" << endl;
-		return 1;
-	}
-
-	smp->sampleLen = sfinfo.frames * numChan;
-	smp->samples = new float[smp->sampleLen];
-	if(smp == NULL){
-		cout << "Could not allocate buffer" << endl;
-		return 1;
-	}
-
-	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
-	int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
-
-	// Pad with zeros in case we couldn't read whole file
-	for(int k = readcount; k <smp->sampleLen; k++)
-		smp->samples[k] = 0;
-
-	if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
-		double	scale ;
-		int 	m ;
-
-		sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
-		if (scale < 1e-10)
-			scale = 1.0 ;
-		else
-			scale = 32700.0 / scale ;
-		cout << "File samples scale = " << scale << endl;
-
-		for (m = 0; m < smp->sampleLen; m++)
-			smp->samples[m] *= scale;
-	}
-
-	sf_close(sndfile);
-
-	return 0;
-}
-
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	//rt_task_delete ((RT_TASK *) &gTriggerSamplesTask);
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --file [-f] filename:    Name of the file to load (default is \"sample.wav\")\n";
-	cerr << "   --help [-h]:             Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	string fileName;			// Name of the sample to load
-
-	SampleData sampleData;		// User define structure to pass data retrieved from file to render function
-	sampleData.samples = 0;
-	sampleData.sampleLen = -1;
-
-
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"file", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case 'f':
-				fileName = string((char *)optarg);
-				break;
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	if(fileName.empty()){
-		fileName = "samples/sample.wav";
-	}
-
-	if(settings.verbose) {
-		cout << "Loading file " << fileName << endl;
-	}
-
-	// Load file
-	if(initFile(fileName, &sampleData) != 0)
-	{
-		cout << "Error: unable to load samples " << endl;
-		return -1;
-	}
-
-	if(settings.verbose)
-		cout << "File contains " << sampleData.sampleLen << " samples" << endl;
-
-
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, &sampleData) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// All done!
-	return 0;
-}
--- a/examples/samples/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,154 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/*
- *
- * Andrew McPherson and Victor Zappi
- * Queen Mary, University of London
- */
-
-/**
-\example 4_audio_samples
-
-Playback WAV files
-------------------
-
-This sketch shows how to playback audio samples from a buffer.
-
-An audio file is loaded into a buffer `SampleData` as `gSampleData`. This is 
-accessed with a read pointer that is incremented at audio rate within the render 
-function: `out += gSampleData.samples[gReadPtr++]`.
-
-Note that the read pointer is stopped from incrementing past the length of the 
-`gSampleData`. This is achieved by comparing the read pointer value against the 
-sample length which we can access as follows: `gSampleData.sampleLen`.
-
-The sample is triggered by keyboard input: (a) starts sample playback, (s) 
-stops sample playback. The triggering is treated as a lower priority task than 
-the audio. You can see this at the bottom of the render function: 
-`Bela_scheduleAuxiliaryTask(gTriggerSamplesTask)`;
-*/
-
-#include <Bela.h>
-#include <cmath>
-#include "SampleData.h"
-
-SampleData gSampleData;	// User defined structure to get complex data from main
-int gReadPtr;			// Position of last read sample from file
-
-// Task for handling the update of the frequencies using the matrix
-AuxiliaryTask gTriggerSamplesTask;
-
-bool initialise_trigger();
-void trigger_samples();
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-
-	// Retrieve a parameter passed in from the initAudio() call
-	gSampleData = *(SampleData *)userData;
-
-	gReadPtr = -1;
-
-	// Initialise auxiliary tasks
-	if(!initialise_trigger())
-		return false;
-
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		float out = 0;
-
-		// If triggered...
-		if(gReadPtr != -1)
-			out += gSampleData.samples[gReadPtr++];	// ...read each sample...
-
-		if(gReadPtr >= gSampleData.sampleLen)
-			gReadPtr = -1;
-
-		for(unsigned int channel = 0; channel < context->audioChannels; channel++)
-			context->audioOut[n * context->audioChannels + channel] = out;	// ...and put it in both left and right channel
-	}
-
-	// Request that the lower-priority task run at next opportunity
-	Bela_scheduleAuxiliaryTask(gTriggerSamplesTask);
-}
-
-// Initialise the auxiliary task
-// and print info
-
-bool initialise_trigger()
-{
-	if((gTriggerSamplesTask = Bela_createAuxiliaryTask(&trigger_samples, 50, "bela-trigger-samples")) == 0)
-		return false;
-
-	rt_printf("Press 'a' to trigger sample, 's' to stop\n");
-	rt_printf("Press 'q' to quit\n");
-
-	return true;
-}
-
-// This is a lower-priority call to periodically read keyboard input
-// and trigger samples. By placing it at a lower priority,
-// it has minimal effect on the audio performance but it will take longer to
-// complete if the system is under heavy audio load.
-
-void trigger_samples()
-{
-	// This is not a real-time task!
-	// Cos getchar is a system call, not handled by Xenomai.
-	// This task will be automatically down graded.
-
-	char keyStroke = '.';
-
-	keyStroke =	getchar();
-	while(getchar()!='\n'); // to read the first stroke
-
-	switch (keyStroke)
-	{
-		case 'a':
-			gReadPtr = 0;
-			break;
-		case 's':
-			gReadPtr = -1;
-			break;
-		case 'q':
-			gShouldStop = true;
-			break;
-		default:
-			break;
-	}
-}
-
-
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	delete[] gSampleData.samples;
-}
Binary file examples/samples/sample.wav has changed
--- a/examples/scope_analogue/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,85 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/**
-\example 3_scope_analog
-
-Connecting potentiometers
--------------------------
-
-This example reads from analogue inputs 0 and 1 via `analogReadFrame()` and 
-generates a sine wave with amplitude and frequency determined by their values. 
-It's best to connect a 10K potentiometer to each of these analog inputs. Far 
-left and far right pins of the pot go to 3.3V and GND, the middle should be 
-connected to the analog in pins.
-
-The sine wave is then plotted on the oscilloscope. Click the Open Scope button to 
-view the results. As you turn the potentiometers you will see the amplitude and 
-frequency of the sine wave change.
-
-This project also shows as example of `map()` which allows you to re-scale a number 
-from one range to another. Note that `map()` does not constrain your variable 
-within the upper and lower limits. If you want to do this use the `constrain()`
-function.
-*/
-
-#include <Bela.h>
-#include <cmath>
-#include <Scope.h>
-
-Scope scope;
-
-float gInverseSampleRate;
-float gPhase;
-
-bool setup(BelaContext *context, void *userData)
-{
-
-    // setup the scope with 3 channels at the audio sample rate
-	scope.setup(3, context->audioSampleRate);
-	
-	gInverseSampleRate = 1.0 / context->audioSampleRate;
-	gPhase = 0.0;
-
-	return true;
-}
-
-void render(BelaContext *context, void *userData)
-{
-
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-	    
-	    // read analogIn channels 0 and 1
-	    float in1 = analogRead(context, n, 0);
-	    float in2 = analogRead(context, n, 1);
-	    
-	    // map in1 to amplitude and in2 to frequency
-	    float amplitude = in1 * 0.8f;
-	    float frequency = map(in2, 0, 1, 100, 1000);
-	    
-	    // generate a sine wave with the amplitude and frequency 
-	    float out = amplitude * sinf(gPhase);
-	    gPhase += 2.0 * M_PI * frequency * gInverseSampleRate;
-		if(gPhase > 2.0 * M_PI)
-			gPhase -= 2.0 * M_PI;
-	    
-	    // log the sine wave and sensor values on the scope
-	    scope.log(out, in1, in2);
-	    
-	    // pass the sine wave to the audio outputs
-	    for(unsigned int channel = 0; channel < context->audioChannels; channel++)
-			context->audioOut[n * context->audioChannels + channel] = out;
-
-	}
-}
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/scope_basic/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,86 +0,0 @@
-/*
- ____  _____ _        _    
-| __ )| ____| |      / \   
-|  _ \|  _| | |     / _ \  
-| |_) | |___| |___ / ___ \ 
-|____/|_____|_____/_/   \_\.io
-
- */
-
-/**
-\example 1_scope_basic
-
-Oscilloscope in-browser
------------------------
-
-This example demonstrates the scope feature of the IDE.
-
-The scope is instantiated at the top of the file via `Scope scope;`
-
-In `setup()` we define how many channels the scope should have and the sample 
-rate that it should run at via `scope.setup(3, context->audioSampleRate)`.
-
-In `render()` we choose what the scope log via `scope.log(out, out2, out3)`. 
-In this example the scope is logging three sine waves with different phases. To see
-the output click on the <b>Open Scope</b> button.
-
-An additional option is to set the trigger of the oscilloscope from within `render()`.
-In this example we are triggering the scope when oscillator 1 becomes less than 
-oscillator 2 via `scope.trigger()`. Note that this functionality only takes effect
-when the triggering mode is set to custom in the scope UI.
-*/
-
-#include <Bela.h>
-#include <Scope.h>
-#include <cmath>
-
-// set the frequency of the oscillators
-float gFrequency = 110.0;
-float gPhase;
-float gInverseSampleRate;
-
-// instantiate the scope
-Scope scope;
-
-bool setup(BelaContext *context, void *userData)
-{
-    // tell the scope how many channels and the sample rate
-    scope.setup(3, context->audioSampleRate);
-
-    gPhase = 0;
-    gInverseSampleRate = 1.0f/context->audioSampleRate;
-    
-	return true;
-}
-
-float lastOut = 0.0;
-float lastOut2 = 0.0;
-void render(BelaContext *context, void *userData)
-{
-    // iterate over the audio frames and create three oscillators, seperated in phase by PI/2
-    for (unsigned int n=0; n<context->audioFrames; n++){
-        float out = 0.8f * sinf(gPhase);
-        float out2 = 0.8f * sinf(gPhase - M_PI/2);
-		float out3 = 0.8f * sinf(gPhase + M_PI/2);
-		gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate;
-		if(gPhase > 2.0 * M_PI)
-			gPhase -= 2.0 * M_PI;
-			
-		// log the three oscillators to the scope
-        scope.log(out, out2, out3);
-        
-        // optional - tell the scope to trigger when oscillator 1 becomes less than oscillator 2
-        // note this has no effect unless trigger mode is set to custom in the scope UI
-        if (lastOut >= lastOut2 && out < out2){
-            scope.trigger();
-        }
-        
-        lastOut = out;
-        lastOut2 = out2;
-    }
-}
-
-void cleanup(BelaContext *context, void *userData)
-{
-
-}
--- a/examples/stepper/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,150 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-
-#include <Bela.h>
-
-const int kStepLengthSlow = 1000;
-const int kStepLengthFast = 500;
-
-int gStepLengthSamples = kStepLengthSlow;
-
-const int gPinA1 = P8_27;
-const int gPinA2 = P8_28;
-const int gPinB1 = P8_29;
-const int gPinB2 = P8_30;
-const int gPinServo = P9_16;
-
-int gStepCounter = 0;
-int gPhase = 0;
-
-int gServoCounter = 0;
-
-
-enum {
-	kStateMoveRight1 = 0,
-	kStateMoveLeft1,
-	kStateMoveRight2,
-	kStateMoveLeft2,
-	kStateMoveRight3,
-	kStateMoveLeft3,
-	kStateSpin,
-	kStateMax
-};
-
-int gState = 0;
-int gStateCounter = 0;
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{	
-	// This project makes the assumption that the audio and digital
-	// sample rates are the same. But check it to be sure!
-	if(context->audioFrames != context->digitalFrames) {
-		rt_printf("Error: this project needs the audio and digital sample rates to be the same.\n");
-		return false;
-	}
-	
-	pinMode(context, 0, gPinA1, OUTPUT);
-	pinMode(context, 0, gPinA2, OUTPUT);
-	pinMode(context, 0, gPinB1, OUTPUT);
-	pinMode(context, 0, gPinB2, OUTPUT);
-	pinMode(context, 0, gPinServo, OUTPUT);
-		
-	return true;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	for(unsigned int n = 0; n < context->audioFrames; n++) {
-		if(gPhase == 0 || gPhase == 1) {
-			digitalWriteOnce(context, n, gPinB1, HIGH);
-			digitalWriteOnce(context, n, gPinB2, LOW);
-		}
-		else {
-			digitalWriteOnce(context, n, gPinB1, LOW);
-			digitalWriteOnce(context, n, gPinB2, HIGH);			
-		}
-		
-		if(gPhase == 1 || gPhase == 2) {
-			digitalWriteOnce(context, n, gPinA1, HIGH);
-			digitalWriteOnce(context, n, gPinA2, LOW);
-		}
-		else {
-			digitalWriteOnce(context, n, gPinA1, LOW);
-			digitalWriteOnce(context, n, gPinA2, HIGH);			
-		}
-		
-		if(--gServoCounter > 0) 
-			digitalWriteOnce(context, n, gPinServo, HIGH);
-		else
-			digitalWriteOnce(context, n, gPinServo, LOW);
-		
-		if(++gStepCounter >= gStepLengthSamples) {
-			gStateCounter++;
-			
-			switch(gState) {
-				case kStateMoveRight1:
-				case kStateMoveRight2:
-				case kStateMoveRight3:
-					gPhase = (gPhase + 1) & 3;
-					break;
-				case kStateMoveLeft1:
-				case kStateMoveLeft2:
-				case kStateMoveLeft3:				
-					gPhase = (gPhase + 3) & 3;	
-					break;
-				case kStateSpin:
-					gPhase = (gPhase + 1) & 3;
-					break;		
-			}
-			
-			if(gState == kStateSpin) {
-				if(gStateCounter >= 48) {
-					gStateCounter = 0;
-					gState = 0;
-					gStepLengthSamples = kStepLengthSlow;
-				}				
-			}
-			else {
-				if(gStateCounter >= 16) {
-					gStateCounter = 0;
-					gState++;
-					if(gState & 1)
-						gServoCounter = 120;
-					else
-						gServoCounter = 80;
-					if(gState == kStateSpin)
-						gStepLengthSamples = kStepLengthFast;
-				}
-			}
-			
-			gStepCounter = 0;
-		}
-	}
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	
-}
--- a/examples/tank_wars/game.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,339 +0,0 @@
-/*
- * game.cpp
- *
- *  Created on: Nov 10, 2014
- *      Author: parallels
- */
-
-#include <cmath>
-#include <cstdlib>
-#include "vector_graphics.h"
-#include <Utilities.h>
-
-// Virtual screen size
-int screenWidth, screenHeight;
-
-// Basic information on the terrain and the tanks
-float *groundLevel;  // Y coordinate of the ground for each X coordinate
-float tank1X, tank1Y, tank2X, tank2Y; // Positions of the two tanks
-float tankRadius = 20;  // Radius of the tanks
-float cannonLength = 40;   // How long the cannon on each tank extends
-float gravity = 0.05;      // Strength of gravity
-
-// Current state of the game
-int playerHasWon = 0; // 1 if player 1 wins, 2 if player 2 wins, 0 if game in progress
-bool player1Turn = true;  // true if it's player 1's turn; false otherwise
-float tank1CannonAngle = M_PI/2;
-float tank2CannonAngle = M_PI/2; // Direction the tank cannons are pointing
-float tank1CannonStrength = 3;
-float tank2CannonStrength = 3; // Strength of intended projectile launch
-
-// Location of the projectile
-bool projectileInMotion = false;
-float projectilePositionX, projectilePositionY;
-float projectileVelocityX, projectileVelocityY;
-
-// Infor needed for sound rendering
-bool collisionJustOccurred = false;
-bool tankHitJustOccurred = false;
-
-// Useful utility function for generating random floating-point values
-float randomFloat(float low, float hi)
-{
-	float r = (float)random() / (float)RAND_MAX;
-	return map(r, 0, 1, low, hi);
-}
-
-// Restart the game, without reallocating memory
-void restartGame()
-{
-	float player1Height = screenHeight * 3/4; // randomFloat(screenHeight/2, screenHeight-5);
-	float player2Height = screenHeight - 5; // randomFloat(screenHeight/2, screenHeight-5);
-	for(int i = 0; i < screenWidth * 0.2; i++) {
-		groundLevel[i] = player1Height;
-	}
-	for(int i = screenWidth * 0.2; i < screenWidth * 0.8; i++) {
-		groundLevel[i] = player1Height + (player2Height - player1Height) * (i - screenWidth*0.2)/(screenWidth*0.6);
-	}
-	for(int i = screenWidth * 0.8; i < screenWidth; i++) {
-		groundLevel[i] = player2Height;
-	}
-
-	// Set the location of the two tanks so they rest on the ground at opposite sides
-	tank1X = screenWidth * 0.1;
-	tank1Y = player1Height;
-	tank2X = screenWidth * 0.9;
-	tank2Y = player2Height;
-
-	playerHasWon = 0;
-	projectileInMotion = false;
-}
-
-// Initialise the game
-void setupGame(int width, int height)
-{
-	// Set the screen size
-	screenWidth = width;
-	screenHeight = height;
-
-	// Initialize the ground level
-	groundLevel = new float[screenWidth];
-
-	restartGame();
-}
-
-// Advance the turn to the next player
-void nextPlayersTurn() {
-	player1Turn = !player1Turn;
-}
-
-
-// Move forward one frame on the game physics
-void nextGameFrame()
-{
-	if(!projectileInMotion)
-		return;
-
-	// Update position of projectile
-	projectilePositionX += projectileVelocityX;
-	projectilePositionY += projectileVelocityY;
-	projectileVelocityY += gravity;
-
-	// Check collision with tanks first: a collision with tank 1 means player 2 wins and vice-versa
-	if((tank1X - projectilePositionX)*(tank1X - projectilePositionX) +
-		(tank1Y - projectilePositionY)*(tank1Y - projectilePositionY)
-		<= tankRadius * tankRadius)
-	{
-		projectileInMotion = false;
-		collisionJustOccurred = false;
-		tankHitJustOccurred = true;
-		playerHasWon = 2;
-	}
-	else if((tank2X - projectilePositionX)*(tank2X - projectilePositionX) +
-		(tank2Y - projectilePositionY)*(tank2Y - projectilePositionY)
-		<= tankRadius * tankRadius)
-	{
-		projectileInMotion = false;
-		collisionJustOccurred = false;
-		tankHitJustOccurred = true;
-		playerHasWon = 1;
-	}
-	else if(projectilePositionX < 0 || projectilePositionX >= screenWidth) {
-		// Check collision whether projectile has exited the screen to the left or right
-		projectileInMotion = false;
-		collisionJustOccurred = true;
-		nextPlayersTurn();
-	}
-	else if(projectilePositionY >= groundLevel[(int)floorf(projectilePositionX)]) {
-		// Check for projectile collision with ground
-		projectileInMotion = false;
-		collisionJustOccurred = true;
-		nextPlayersTurn();
-	}
-}
-
-// Updates for game state
-void setTank1CannonAngle(float angle)
-{
-	tank1CannonAngle = angle;
-}
-
-void setTank2CannonAngle(float angle)
-{
-	tank2CannonAngle = angle;
-}
-
-void setTank1CannonStrength(float strength)
-{
-	tank1CannonStrength = strength;
-}
-
-void setTank2CannonStrength(float strength)
-{
-	tank2CannonStrength = strength;
-}
-
-// FIRE!
-void fireProjectile()
-{
-	// Can't fire while projectile is already moving, or if someone has won
-	if(projectileInMotion)
-		return;
-	if(playerHasWon != 0)
-		return;
-
-    if(player1Turn) {
-		projectilePositionX = tank1X + cannonLength * cosf(tank1CannonAngle);
-		projectilePositionY = tank1Y - cannonLength * sinf(tank1CannonAngle);
-		projectileVelocityX = tank1CannonStrength * cosf(tank1CannonAngle);
-		projectileVelocityY = -tank1CannonStrength * sinf(tank1CannonAngle);
-    }
-    else {
-		projectilePositionX = tank2X + cannonLength * cosf(tank2CannonAngle);
-		projectilePositionY = tank2Y - cannonLength * sinf(tank2CannonAngle);
-		projectileVelocityX = tank2CannonStrength * cosf(tank2CannonAngle);
-		projectileVelocityY = -tank2CannonStrength * sinf(tank2CannonAngle);
-    }
-
-    // GO!
-    projectileInMotion = true;
-}
-
-// Game state queries
-bool gameStatusPlayer1Turn()
-{
-	return player1Turn;
-}
-
-bool gameStatusProjectileInMotion()
-{
-	return projectileInMotion;
-}
-
-int gameStatusWinner()
-{
-	return playerHasWon;
-}
-
-bool gameStatusCollisionOccurred()
-{
-	if(collisionJustOccurred) {
-		collisionJustOccurred = false;
-		return true;
-	}
-	return false;
-}
-
-bool gameStatusTankHitOccurred()
-{
-	if(tankHitJustOccurred) {
-		tankHitJustOccurred = false;
-		return true;
-	}
-	return false;
-}
-
-
-float gameStatusProjectileHeight()
-{
-	return projectilePositionY / (float)screenHeight;
-}
-
-// Clean up any allocated memory for the game
-void cleanupGame()
-{
-	delete groundLevel;
-}
-
-// Drawing routines. Arguments are (interleaved) buffer to render
-// into, the available size, and the target for how many samples
-// to use (actual usage might vary slightly). Regardless of
-// lengthTarget, never use more than bufferSize samples.
-
-int drawGround(float *buffer, int bufferSize, int framesTarget)
-{
-	int length;
-
-	// Calculate total length of ground line, to arrive at a speed calculation
-	float totalLineLength = 0.4f*screenWidth
-							+ sqrtf(0.36f*screenWidth*screenWidth
-									+ (tank2Y-tank1Y)*(tank2Y-tank1Y));
-
-	// Speed is calculated in pixels per frame
-	float speed = totalLineLength / (float)framesTarget;
-
-	// Draw three lines: platforms for tanks and the connecting line.
-	// Eventually, render a more complex ground from the array.
-	length = renderLine(0, tank1Y, screenWidth * 0.2, tank1Y,
-						speed, buffer, bufferSize);
-	length += renderLine(screenWidth * 0.2, tank1Y, screenWidth * 0.8, tank2Y,
-						speed, &buffer[length], bufferSize - length);
-	length += renderLine(screenWidth * 0.8, tank2Y, screenWidth, tank2Y,
-						speed, &buffer[length], bufferSize - length);
-
-	return length;
-}
-
-int drawTanks(float *buffer, int bufferSize, int framesTarget)
-{
-	int length = 0;
-
-	// Calculate total length of tank lines, to arrive at a speed calculation
-	float totalLineLength = 2.0*M_PI*tankRadius + 2.0*(cannonLength - tankRadius);
-
-	// Speed is calculated in pixels per frame
-	float speed = totalLineLength / (float)framesTarget;
-
-	if(playerHasWon != 2) {
-		// Tank 1 body = semicircle + line
-		length += renderArc(tank1X, tank1Y, tankRadius, M_PI, 2.0 * M_PI,
-							speed, buffer, bufferSize);
-		length += renderLine(tank1X + tankRadius, tank1Y,
-							 tank1X - tankRadius, tank1Y,
-							speed, &buffer[length], bufferSize - length);
-		// Tank 1 cannon (line depending on angle)
-		length += renderLine(tank1X + tankRadius * cosf(tank1CannonAngle),
-			 tank1Y - tankRadius * sinf(tank1CannonAngle),
-			 tank1X + cannonLength * cosf(tank1CannonAngle),
-			 tank1Y - cannonLength * sinf(tank1CannonAngle),
-			 speed, &buffer[length], bufferSize - length);
-	}
-
-	if(playerHasWon != 1) {
-		// Same idea for tank 2
-		length += renderArc(tank2X, tank2Y, tankRadius, M_PI, 2.0 * M_PI,
-							speed, &buffer[length], bufferSize - length);
-		length += renderLine(tank2X + tankRadius, tank2Y,
-							 tank2X - tankRadius, tank2Y,
-							 speed, &buffer[length], bufferSize - length);
-		length += renderLine(tank2X + tankRadius * cosf(tank2CannonAngle),
-			 tank2Y - tankRadius * sinf(tank2CannonAngle),
-			 tank2X + cannonLength * cosf(tank2CannonAngle),
-			 tank2Y - cannonLength * sinf(tank2CannonAngle),
-			 speed, &buffer[length], bufferSize - length);
-	}
-
-	return length;
-}
-
-int drawProjectile(float *buffer, int bufferSize, int framesTarget)
-{
-	if(!projectileInMotion)
-		return 0;
-
-	// Draw a point for a specified number of frames (each containing X and Y)
-	// Return the number of items used in the buffer, which will be twice
-	// the number of frames unless the buffer is full
-
-	if(bufferSize/2 < framesTarget) {
-		renderPoint(projectilePositionX, projectilePositionY, buffer, bufferSize/2);
-		return bufferSize;
-	}
-	else {
-		renderPoint(projectilePositionX, projectilePositionY, buffer, framesTarget);
-		return framesTarget*2;
-	}
-}
-
-// Main drawing routine entry point
-int drawGame(float *buffer, int bufferSize)
-{
-	int length;
-
-	// Based on buffer size, come up with speeds for each of the elements
-	// 50% of time to ground; 30% to the tanks and 20% to the projectile
-	// Give a margin of 25% beyond so we don't run out of buffer space
-	// if things take longer to draw than we guess they will
-	const float amountToUse = 0.375; // 0.75/2 because two samples per frame
-	const float groundFraction = 0.5 * amountToUse;
-	const float tankFraction = 0.3 * amountToUse;
-	const float projectileFraction = 0.2 * amountToUse;
-
-	length = drawGround(buffer, bufferSize, bufferSize * groundFraction);
-	length += drawTanks(&buffer[length], bufferSize - length,
-						bufferSize * tankFraction);
-	length += drawProjectile(&buffer[length], bufferSize - length,
-						bufferSize * projectileFraction);
-
-	return length;
-}
--- a/examples/tank_wars/game.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,39 +0,0 @@
-/*
- * game.h
- *
- *  Created on: Nov 10, 2014
- *      Author: parallels
- */
-
-#ifndef GAME_H_
-#define GAME_H_
-
-// Initialisation
-void setupGame(int width, int height);
-void restartGame();
-
-// Update physics
-void nextGameFrame();
-
-// State updaters
-void setTank1CannonAngle(float angle);
-void setTank2CannonAngle(float angle);
-void setTank1CannonStrength(float strength);
-void setTank2CannonStrength(float strength);
-void fireProjectile();
-
-// State queries
-bool gameStatusPlayer1Turn();
-bool gameStatusProjectileInMotion();
-int gameStatusWinner();
-bool gameStatusCollisionOccurred();
-bool gameStatusTankHitOccurred();
-float gameStatusProjectileHeight();
-
-// Render screen; returns length of buffer used
-int drawGame(float *buffer, int bufferSize);
-
-// Cleanup and memory release
-void cleanupGame();
-
-#endif /* GAME_H_ */
--- a/examples/tank_wars/main.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,169 +0,0 @@
-/*
- * main.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-#include <iostream>
-#include <cstdlib>
-#include <libgen.h>
-#include <signal.h>
-#include <getopt.h>
-#include <sndfile.h>
-#include <Bela.h>
-
-extern int gScreenFramesPerSecond;
-
-float *gMusicBuffer = 0;
-int gMusicBufferLength = 0;
-float *gSoundBoomBuffer = 0;
-int gSoundBoomBufferLength = 0;
-float *gSoundHitBuffer = 0;
-int gSoundHitBufferLength = 0;
-
-
-using namespace std;
-
-// Load a sound sample from file
-int loadSoundFile(const string& path, float **buffer, int *bufferLength)
-{
-	SNDFILE *sndfile ;
-	SF_INFO sfinfo ;
-
-	if (!(sndfile = sf_open (path.c_str(), SFM_READ, &sfinfo))) {
-		cout << "Couldn't open file " << path << endl;
-		return 1;
-	}
-
-	int numChan = sfinfo.channels;
-	if(numChan != 1)
-	{
-		cout << "Error: " << path << " is not a mono file" << endl;
-		return 1;
-	}
-
-	*bufferLength = sfinfo.frames * numChan;
-	*buffer = new float[*bufferLength];
-	if(*buffer == 0){
-		cout << "Could not allocate buffer" << endl;
-		return 1;
-	}
-
-	int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
-	int readcount = sf_read_float(sndfile, *buffer, *bufferLength);
-
-	// Pad with zeros in case we couldn't read whole file
-	for(int k = readcount; k < *bufferLength; k++)
-		(*buffer)[k] = 0;
-
-	sf_close(sndfile);
-	return 0;
-}
-
-// Handle Ctrl-C by requesting that the audio rendering stop
-void interrupt_handler(int var)
-{
-	gShouldStop = true;
-}
-
-// Print usage information
-void usage(const char * processName)
-{
-	cerr << "Usage: " << processName << " [options]" << endl;
-
-	Bela_usage();
-
-	cerr << "   --fps [-f] value:           Set target frames per second\n";
-	cerr << "   --help [-h]:                Print this menu\n";
-}
-
-int main(int argc, char *argv[])
-{
-	BelaInitSettings settings;	// Standard audio settings
-	string musicFileName = "music.wav";
-	string soundBoomFileName = "boom.wav";
-	string soundHitFileName = "hit.wav";
-	
-	struct option customOptions[] =
-	{
-		{"help", 0, NULL, 'h'},
-		{"fps", 1, NULL, 'f'},
-		{NULL, 0, NULL, 0}
-	};
-
-	// Set default settings
-	Bela_defaultSettings(&settings);
-
-	// Parse command-line arguments
-	while (1) {
-		int c;
-		if ((c = Bela_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0)
-				break;
-		switch (c) {
-		case 'f':
-				gScreenFramesPerSecond = atoi(optarg);
-				if(gScreenFramesPerSecond < 1)
-					gScreenFramesPerSecond = 1;
-				if(gScreenFramesPerSecond > 100)
-					gScreenFramesPerSecond = 100;
-				break;
-		case 'h':
-				usage(basename(argv[0]));
-				exit(0);
-		case '?':
-		default:
-				usage(basename(argv[0]));
-				exit(1);
-		}
-	}
-
-	// Load the sound files
-	if(loadSoundFile(musicFileName, &gMusicBuffer, &gMusicBufferLength) != 0) {
-		cout << "Warning: unable to load sound file " << musicFileName << endl;
-	}
-	if(loadSoundFile(soundBoomFileName, &gSoundBoomBuffer, &gSoundBoomBufferLength) != 0) {
-		cout << "Warning: unable to load sound file " << soundBoomFileName << endl;
-	}
-	if(loadSoundFile(soundHitFileName, &gSoundHitBuffer, &gSoundHitBufferLength) != 0) {
-		cout << "Warning: unable to load sound file " << soundHitFileName << endl;
-	}
-	
-	// Initialise the PRU audio device
-	if(Bela_initAudio(&settings, 0) != 0) {
-		cout << "Error: unable to initialise audio" << endl;
-		return -1;
-	}
-
-	// Start the audio device running
-	if(Bela_startAudio()) {
-		cout << "Error: unable to start real-time audio" << endl;
-		return -1;
-	}
-
-	// Set up interrupt handler to catch Control-C and SIGTERM
-	signal(SIGINT, interrupt_handler);
-	signal(SIGTERM, interrupt_handler);
-
-	// Run until told to stop
-	while(!gShouldStop) {
-		usleep(100000);
-	}
-
-	// Stop the audio device
-	Bela_stopAudio();
-
-	// Clean up any resources allocated for audio
-	Bela_cleanupAudio();
-
-	// Release sound files
-	if(gMusicBuffer != 0)
-		free(gMusicBuffer);
-	if(gSoundBoomBuffer != 0)
-		free(gSoundBoomBuffer);
-	if(gSoundHitBuffer != 0)
-		free(gSoundHitBuffer);
-	
-	// All done!
-	return 0;
-}
--- a/examples/tank_wars/render.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,337 +0,0 @@
-/*
- * render.cpp
- *
- *  Created on: Oct 24, 2014
- *      Author: parallels
- */
-
-
-#include <Bela.h>
-#include "game.h"
-#include <rtdk.h>
-#include <cmath>
-#include <cstdlib>
-#include <time.h>
-
-int gAudioFramesPerMatrixFrame = 2; // Ratio in audio to matrix sample rate
-
-int gInputTank1Angle = 0;		// Inputs for the cannon angles
-int gInputTank2Angle = 1;
-int gInputLauncher = 2;			// Input for launcher FSR
-
-int gOutputX = 0;				// Outputs for the scope
-int gOutputY = 1;
-int gOutputPlayer1LED = 2;
-int gOutputPlayer2LED = 3;
-
-int gGameFramesPerSecond = 60;	// How often the physics are updated
-int gGameFrameInterval;			// ...and in frames
-int gSamplesUntilNextFrame;		// Counter until next update
-int gSamplesSinceFinish = 0;	// How long since somebody won?
-bool gGameShouldRestart = false;// Whether we need to reinitiliase the game
-
-// Counter for overall number of samples that have elapsed
-unsigned int gSampleCounter = 0;
-
-// 1st-order filter and peak detector for launcher input
-float gLauncherLastSample = 0;
-float gLauncherFilterPole = 0.8;
-float gLauncherPeakValue = 0;
-float gLauncherPeakFilterPole = 0.999;
-float gLauncherNoiseThreshold = 0.01;
-float gLauncherMinimumPeak = 0.1;
-bool gLauncherTriggered = false;
-
-// Screen update rate; affects buffer size. Actual contents of buffer
-// may be smaller than this
-int gScreenWidth = 512;
-int gScreenHeight = 512;
-int gScreenFramesPerSecond = 25;
-
-// Double buffer for rendering screen. Each is an interleaved buffer
-// of XY data.
-float *gScreenBuffer1, *gScreenBuffer2;
-float *gScreenBufferWrite, *gScreenBufferRead;
-int gScreenBufferMaxLength;				// What is the total buffer allocated?
-int gScreenBufferReadLength;			// How long is the read buffer?
-int gScreenBufferWriteLength;			// How long is the write (next) buffer?
-int gScreenBufferReadPointer;			// Where are we in the read buffer now?
-int gScreenBufferNextUpdateLocation;	// When should we render the next buffer?
-bool gScreenNextBufferReady;			// Is the next buffer ready to go?
-
-// Auxiliary (low-priority) task for updating the screen
-AuxiliaryTask gScreenUpdateTask;
-
-// Buffers for music and sound effects
-extern float *gMusicBuffer;
-extern int gMusicBufferLength;
-extern float *gSoundBoomBuffer;
-extern int gSoundBoomBufferLength;
-extern float *gSoundHitBuffer;
-extern int gSoundHitBufferLength;
-
-// Current state for sound and music
-int gMusicBufferPointer = 0;	  // 0 means start of buffer...
-int gSoundBoomBufferPointer = -1; // -1 means don't play...
-int gSoundHitBufferPointer = -1;
-float gSoundProjectileOscillatorPhase = 0;
-float gSoundProjectileOscillatorGain = 0.2;
-float gOscillatorPhaseScaler = 0;
-
-void screen_update();
-
-// setup() is called once before the audio rendering starts.
-// Use it to perform any initialisation and allocation which is dependent
-// on the period size or sample rate.
-//
-// userData holds an opaque pointer to a data structure that was passed
-// in from the call to initAudio().
-//
-// Return true on success; returning false halts the program.
-
-bool setup(BelaContext *context, void *userData)
-{
-	srandom(time(NULL));
-
-	// Verify we are running with matrix enabled
-	if(context->analogFrames == 0 || context->analogChannels < 4) {
-		rt_printf("Error: this example needs the matrix enabled with at least 4 channels\n");
-		return false;
-	}
-
-	// Initialise audio variables
-	gAudioFramesPerMatrixFrame = context->audioFrames / context->analogFrames;
-	gOscillatorPhaseScaler = 2.0 * M_PI / context->audioSampleRate;
-
-	// Initialise the screen buffers
-	gScreenBufferMaxLength = 2 * context->analogSampleRate / gScreenFramesPerSecond;
-	gScreenBuffer1 = new float[gScreenBufferMaxLength];
-	gScreenBuffer2 = new float[gScreenBufferMaxLength];
-	if(gScreenBuffer1 == 0 || gScreenBuffer2 == 0) {
-		rt_printf("Error initialising screen buffers\n");
-		return false;
-	}
-
-	gScreenBufferRead = gScreenBuffer1;
-	gScreenBufferWrite = gScreenBuffer2;
-	gScreenBufferReadLength = gScreenBufferWriteLength = 0;
-	gScreenBufferReadPointer = 0;
-	gScreenBufferNextUpdateLocation = 0;
-	gScreenNextBufferReady = false;
-
-	// Initialise the game
-	setupGame(gScreenWidth, gScreenHeight);
-	gGameFrameInterval = context->analogSampleRate / gGameFramesPerSecond;
-	gSamplesUntilNextFrame = gGameFrameInterval;
-
-	// Initialise auxiliary tasks
-	if((gScreenUpdateTask = Bela_createAuxiliaryTask(&screen_update, 90,
-														 "bela-screen-update")) == 0)
-		return false;
-
-	return true;
-}
-
-// Swap buffers on the screen
-void swap_buffers()
-{
-	if(gScreenBufferRead == gScreenBuffer1) {
-		gScreenBufferRead = gScreenBuffer2;
-		gScreenBufferWrite = gScreenBuffer1;
-	}
-	else {
-		gScreenBufferRead = gScreenBuffer1;
-		gScreenBufferWrite = gScreenBuffer2;
-	}
-
-	gScreenBufferReadLength = gScreenBufferWriteLength;
-	gScreenBufferReadPointer = 0;
-
-	// Schedule next update for 3/4 of the way through the buffer
-	gScreenBufferNextUpdateLocation = gScreenBufferReadLength * 0.75;
-	gScreenNextBufferReady = false;
-}
-
-// render() is called regularly at the highest priority by the audio engine.
-// Input and output are given from the audio hardware and the other
-// ADCs and DACs (if available). If only audio is available, numMatrixFrames
-// will be 0.
-
-void render(BelaContext *context, void *userData)
-{
-	int audioIndex = 0;
-
-	for(unsigned int n = 0; n < context->analogFrames; n++) {
-		for(int k = 0; k < gAudioFramesPerMatrixFrame; k++) {
-			// Render music and sound
-			float audioSample = 0;
-
-			// Music plays in a loop
-			if(gMusicBuffer != 0 && gMusicBufferPointer >= 0) {
-				audioSample += gMusicBuffer[gMusicBufferPointer++];
-				if(gMusicBufferPointer >= gMusicBufferLength)
-					gMusicBufferPointer = 0;
-			}
-
-			// Sound effect plays until finished, then stops
-			if(gSoundBoomBuffer != 0 && gSoundBoomBufferPointer >= 0) {
-				audioSample += gSoundBoomBuffer[gSoundBoomBufferPointer++];
-				if(gSoundBoomBufferPointer >= gSoundBoomBufferLength)
-					gSoundBoomBufferPointer = -1;
-			}
-
-			if(gSoundHitBuffer != 0 && gSoundHitBufferPointer >= 0) {
-				audioSample += gSoundHitBuffer[gSoundHitBufferPointer++];
-				if(gSoundHitBufferPointer >= gSoundHitBufferLength)
-					gSoundHitBufferPointer = -1;
-			}
-
-			// Oscillator plays to indicate projectile height
-			if(gameStatusProjectileInMotion()) {
-				audioSample += gSoundProjectileOscillatorGain * sinf(gSoundProjectileOscillatorPhase);
-
-				gSoundProjectileOscillatorPhase += gOscillatorPhaseScaler * constrain(map(gameStatusProjectileHeight(),
-						1.0, 0, 300, 2000), 200, 6000);
-				if(gSoundProjectileOscillatorPhase > 2.0 * M_PI)
-					gSoundProjectileOscillatorPhase -= 2.0 * M_PI;
-			}
-
-			context->audioOut[2*audioIndex] = context->audioOut[2*audioIndex + 1] = audioSample;
-			audioIndex++;
-		}
-
-		// First-order lowpass filter to remove noise on launch FSR
-		float rawSample = analogRead(context, n, gInputLauncher);
-		float launchSample = gLauncherFilterPole * gLauncherLastSample +
-							(1.0f - gLauncherFilterPole) * rawSample;
-		gLauncherLastSample = launchSample;
-
-		// Peak-detect on launch signal
-		if(launchSample >= gLauncherPeakValue) {
-			gLauncherPeakValue = launchSample;
-			gLauncherTriggered = false;
-		}
-		else {
-			if(gLauncherPeakValue - launchSample > gLauncherNoiseThreshold && !gLauncherTriggered) {
-				// Detected a peak; is it big enough overall?
-				if(gLauncherPeakValue >= gLauncherMinimumPeak) {
-					gLauncherTriggered = true;
-					// Peak detected-- fire!!
-					// Set both cannon strengths but only one will
-					// fire depending on whose turn it is
-					float strength = map(gLauncherPeakValue,
-									     gLauncherMinimumPeak, 1.0,
-										 0.5f, 10.0f);
-					setTank1CannonStrength(strength);
-					setTank2CannonStrength(strength);
-					fireProjectile();
-				}
-			}
-
-			gLauncherPeakValue *= gLauncherPeakFilterPole;
-		}
-
-		if(--gSamplesUntilNextFrame <= 0) {
-			// Update game physics and cannon angles
-			gSamplesUntilNextFrame = gGameFrameInterval;
-
-			setTank1CannonAngle(map(analogRead(context, n, gInputTank1Angle),
-									0, 1.0, M_PI, 0));
-			setTank2CannonAngle(map(analogRead(context, n, gInputTank2Angle),
-									0, 1.0, M_PI, 0));
-			nextGameFrame();
-
-			// Check for collision and start sound accordingly
-			if(gameStatusCollisionOccurred()) {
-				gSoundBoomBufferPointer = 0;
-			}
-			
-			if(gameStatusTankHitOccurred()) {
-				gSoundHitBufferPointer = 0;
-			}
-		}
-
-		if(gScreenBufferReadPointer >= gScreenBufferReadLength - 1
-			&& gScreenNextBufferReady) {
-			// Got to the end; swap buffers
-			swap_buffers();
-		}
-
-		// Push current screen buffer to the matrix output
-		if(gScreenBufferReadPointer < gScreenBufferReadLength - 1) {
-			float x = gScreenBufferRead[gScreenBufferReadPointer++];
-			float y = gScreenBufferRead[gScreenBufferReadPointer++];
-
-			// Rescale screen coordinates to matrix ranges; invert the Y
-			// coordinate to go from normal screen coordinates to scope coordinates
-			analogWriteOnce(context, n, gOutputX, constrain(map(x, 0, gScreenWidth, 0, 1.0), 0, 1.0));
-			analogWriteOnce(context, n, gOutputY, constrain(map(y, 0, gScreenHeight, 1.0, 0), 0, 1.0));
-		}
-		else {
-			// Still not ready! Write 0 until something happens
-			analogWriteOnce(context, n, gOutputX, 0);
-			analogWriteOnce(context, n, gOutputY, 0);
-		}
-
-		if(gameStatusWinner() != 0) {
-			// Blink one LED to show who won
-			// Blink both LEDs when projectile is in motion
-			float val = (gSampleCounter % 4000 > 2000) ? 1.0 : 0;
-			analogWriteOnce(context, n, gOutputPlayer1LED, gameStatusWinner() == 1 ? val : 0);
-			analogWriteOnce(context, n, gOutputPlayer2LED, gameStatusWinner() == 2 ? val : 0);
-
-			// After 5 seconds, restart the game
-			gSamplesSinceFinish++;
-			if(gSamplesSinceFinish > 22050*5)
-				gGameShouldRestart = true;
-		}
-		else if(gameStatusProjectileInMotion()) {
-			// Blink both LEDs when projectile is in motion
-			float val = (gSampleCounter % 2000 > 1000) ? 1.0 : 0;
-			analogWriteOnce(context, n, gOutputPlayer1LED, val);
-			analogWriteOnce(context, n, gOutputPlayer2LED, val);
-		}
-		else if(gameStatusPlayer1Turn()) {
-			analogWriteOnce(context, n, gOutputPlayer1LED, 1.0);
-			analogWriteOnce(context, n, gOutputPlayer2LED, 0);
-		}
-		else {
-			analogWriteOnce(context, n, gOutputPlayer2LED, 1.0);
-			analogWriteOnce(context, n, gOutputPlayer1LED, 0);
-		}
-
-		// Check if we have reached the point where we should next update
-		if(gScreenBufferReadPointer >= gScreenBufferNextUpdateLocation &&
-		   !gScreenNextBufferReady) {
-			// Update the screen at lower priority than the audio thread
-			Bela_scheduleAuxiliaryTask(gScreenUpdateTask);
-		}
-
-		gSampleCounter++;
-	}
-}
-
-void screen_update()
-{
-	// If we should restart, reinitialise the game
-	if(gGameShouldRestart) {
-		restartGame();
-		gGameShouldRestart = false;
-		gSamplesSinceFinish = 0;
-	}
-
-	// Render the game based on the current state
-	gScreenBufferWriteLength = drawGame(gScreenBufferWrite, gScreenBufferMaxLength);
-
-	// Flag it as ready to go
-	gScreenNextBufferReady = true;
-}
-
-// cleanup() is called once at the end, after the audio has stopped.
-// Release any resources that were allocated in setup().
-
-void cleanup(BelaContext *context, void *userData)
-{
-	// Clean up the game state
-	cleanupGame();
-}
--- a/examples/tank_wars/vector_graphics.cpp	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,72 +0,0 @@
-/*
- * vector_graphics.cpp
- *
- *  Created on: Nov 10, 2014
- *      Author: parallels
- */
-
-#include <cmath>
-
-// Draw a line between two points at a specified rate in
-// pixels per buffer sample. Indicate maximum available space.
-// Returns space used
-int renderLine(float x1, float y1, float x2, float y2, float speed,
-			   float *buffer, int maxLength) {
-	// Figure out length of line and therefore how many samples
-	// are needed to represent it based on the speed (rounded to nearest int)
-	float totalLineLength = sqrtf((x2 - x1)*(x2 - x1) + (y2 - y1)*(y2 - y1));
-	int samplesNeeded = floorf(totalLineLength / speed + 0.5);
-
-	// Now render into the buffer
-	int length = 0;
-	float scaleFactor = 1.0f / samplesNeeded;
-	for(int n = 0; n < samplesNeeded; n++) {
-		if(length >= maxLength - 1)
-			return length;
-		// X coordinate
-		*buffer++ = x1 + (float)n * scaleFactor * (x2 - x1);
-		// Y coordinate
-		*buffer++ = y1 + (float)n * scaleFactor * (y2 - y1);
-		length += 2;
-	}
-
-	return length;
-}
-
-// Draw an arc around a centre point at a specified rate of pixels
-// per buffer sample. Indicate maximum available space.
-// Returns space used
-int renderArc(float x, float y, float radius, float thetaMin, float thetaMax,
-			  float speed, float *buffer, int maxLength) {
-	// Figure out circumference of arc and therefore how many samples
-	// are needed to represent it based on the speed (rounded to nearest int)
-	float circumference = (thetaMax - thetaMin) * radius;
-	int samplesNeeded = floorf(circumference / speed + 0.5);
-
-	// Now render into the buffer
-	int length = 0;
-	float scaleFactor = 1.0f / samplesNeeded;
-	for(int n = 0; n < samplesNeeded; n++) {
-		if(length >= maxLength - 1)
-			return length;
-		// Get current angle
-		float theta = thetaMin + (float)n * scaleFactor * (thetaMax - thetaMin);
-
-		// Convert polar to cartesian coordinates
-		*buffer++ = x + radius * cosf(theta);
-		*buffer++ = y + radius * sinf(theta);
-
-		length += 2;
-	}
-
-	return length;
-}
-
-// Draw a single point for a specified number of frames
-void renderPoint(float x, float y, float *buffer, float length) {
-	while(length > 0) {
-		*buffer++ = x;
-		*buffer++ = y;
-		length--;
-	}
-}
--- a/examples/tank_wars/vector_graphics.h	Mon Jun 20 16:57:35 2016 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,18 +0,0 @@
-/*
- * vector_graphics.h
- *
- *  Created on: Nov 10, 2014
- *      Author: parallels
- */
-
-#ifndef VECTOR_GRAPHICS_H_
-#define VECTOR_GRAPHICS_H_
-
-int renderLine(float x1, float y1, float x2, float y2, float speed,
-			   float *buffer, int maxLength);
-int renderArc(float x, float y, float radius, float thetaMin, float thetaMax,
-			  float speed, float *buffer, int maxLength);
-void renderPoint(float x, float y, float *buffer, float length);
-
-
-#endif /* VECTOR_GRAPHICS_H_ */