zynaddsubfx

ZynAddSubFX open source synthesizer
Log | Files | Refs | Submodules | LICENSE

commit 0a044a611d2fa2ca221e401f9f4b4a4fc6583285
parent e3dc53122bc55c61fd0b8778061fb2783abc3c50
Author: fundamental <[email protected]>
Date:   Wed,  8 Apr 2015 16:45:05 -0400

Merge branch 'master' of ssh://git.code.sf.net/p/zynaddsubfx/code

Diffstat:
A.travis.yml | 17+++++++++++++++++
Adoc/architecture.txt | 155+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Msrc/Misc/Part.cpp | 98++++++++++++++++++++++++++++++++++++++++----------------------------------------
Msrc/Misc/Part.h | 12++++++------
Msrc/Misc/Util.cpp | 43+++++++++++++++++++++++++++++++++++++++++--
Msrc/Misc/Util.h | 3+++
Msrc/Nio/JackEngine.cpp | 6++++++
Msrc/Nio/JackMultiEngine.cpp | 3+++
Msrc/Nio/Nio.cpp | 7++++---
Msrc/Nio/Nio.h | 1+
Msrc/Synth/ADnote.cpp | 20++++++++++----------
Msrc/Synth/ADnote.h | 2+-
Msrc/Synth/Envelope.cpp | 14+++++++-------
Msrc/Synth/Envelope.h | 4++--
Msrc/Synth/PADnote.cpp | 8++++----
Msrc/Synth/PADnote.h | 2+-
Msrc/Synth/SUBnote.cpp | 12++++++------
Msrc/Synth/SUBnote.h | 2+-
Msrc/Synth/SynthNote.h | 2+-
Msrc/Tests/AdNoteTest.h | 2+-
Msrc/Tests/PadNoteTest.h | 2+-
Msrc/Tests/SubNoteTest.h | 2+-
Msrc/UI/ADnoteUI.fl | 16++++++++--------
Msrc/UI/Connection.cpp | 2++
Msrc/UI/EnvelopeUI.fl | 6+++---
Msrc/UI/MasterUI.fl | 10+++++-----
Msrc/UI/PartUI.fl | 2+-
Msrc/UI/VirKeyboard.fl | 32++++++++++++++++----------------
Msrc/globals.h | 4++--
Msrc/main.cpp | 14++++++++++++--
30 files changed, 370 insertions(+), 133 deletions(-)

diff --git a/.travis.yml b/.travis.yml @@ -0,0 +1,17 @@ +language: cpp + +compiler: + - gcc + +before_install: + - sudo add-apt-repository -y ppa:ubuntu-toolchain-r/test + - sudo apt-get -qq update + - sudo apt-get -qq install g++-4.8 + - sudo update-alternatives --install /usr/bin/g++ g++ /usr/bin/g++-4.8 90 + - sudo apt-get install zlib1g-dev libmxml-dev libfftw3-dev liblo-dev + +script: + - mkdir build && cd build + - cmake .. + - make + - ctest --output-on-failure diff --git a/doc/architecture.txt b/doc/architecture.txt @@ -0,0 +1,155 @@ +ZynAddSubFX Architecture +======================== +:author: Mark McCurry + +In order to understand how to effectively navigate the source code and to +better understand the relationships between the internal components of +ZynAddSubFX. +To start off, the coarsest division of the codebase can be in breaking the +source into three areas: + +Backend:: + Realtime Data and Audio/Midi Handling +Middware:: + Non-Realtime algorithms used to initialize realtime data and communication + core +UI:: + Any User Interface (graphical or otherwise) that binds to a middleware + instance in order to permit modification of any parameters used in + synthesis. + +These three components communicate to each other _almost_ exclusively through +the use of OSC messages using librtosc or liblo. +In this document, I hope to define all of the edge cases where communication +(by design) does not use message passing as interactions are somewhat +complicated lock free operations. + +Before getting into each layer's details, the following threads may exist: + +Main Thread:: + The original program thread, responsible for repeatedly polling events + from NSM, LASH, general in-process UI, and middleware +Middleware Helper Thread:: + Responsible for handling any procedures which may block normal event + processing such as XML de-serialization +Audio Output Thread:: + Thread responsible for performing synthesis and passing it to driver level + API for the sound to be output +MIDI Input Thread:: + Thread responsible for handling midi events. This thread is only active if + the audio output and the midi input drivers are not the same type. + +Now for the meat of things: + +The Backend +----------- + +Historically the realtime side of things has revolved around a single instance +of the aptly named class 'Master', which is the host to numerous +implementation pointers and instances of all Parts which in turn contain more +parameters and notes. +This instance generally assumes that it is in full control of all of its data +and it gets regularly called from the IO subsystem to produce some output +audio in increments of some set block size. +All classes that operate under a given instance of 'Master' assume that they +have a fixed: + +Buffer Size:: + Unit of time to calculate at once and interval to perform interpolations + over +Oscillator Size:: + Size of the Additive Synthesis Oscillator Interpolation Buffer +Sample Rate:: + Number of samples per second +Allocator:: + Source for memory allocations from a resizable memory pool + +Changing any of these essentially requires rebuilding all child data +structures at the moment. + +Most of the children objects can be placed into the categories: + +Parameter Objects:: + Objects which contain serialize able parameters for synthesis and little to + no complex math +Synthesis Objects:: + Objects which initialize with parameter objects and generate output audio or + control values for other synthesis objects +Container Objects:: + Objects which are responsible for organizing a variety of synthesis and + parameter objects and for routing the outputs of each child object to the + right destination (e.g. 'Part' and 'Master') +Hybrid Objects:: + Objects which have _Odd_ divisions between what is a parameter, and what + is destined for synthesis (e.g. 'PADnoteParameters' and 'OscilGen') + +The normal behavior of these objects can be seen by observing a call of +_OutMgr::tick_ which first flushes the midi queue possibly constructing a few +new notes via _Part::NoteOn_, then _Master::AudioOut_ is called. +This is the root of the synthesis calls, but before anything is synthesized, +OSC messages are dispatched which typically update system parameter and +coefficients which cannot be calculated in realtime such as padnote based +wavetables. +Most data is allocated on the initialization of the add/sub/pad synthesis +engine, however anything which cannot be bounded easily then is allocated via +the tlsf based allocator. + + +The MiddleWare +-------------- + +Now in the previous section, details on how exactly messages were delivered +was only vaguely mentioned. +Anything unclear should hopefully be clarified here. +The primary message handling is taken care of by two ring buffers 'uToB' and +'bToU' which are, respectively, the user interface to backend and the backend +to user interface ringbuffers. +Additionally, Middleware handles non-realtime processing, such as 'Oscilgen' +and 'PADnoteParameters'. + +To handle these cases, any message from a user interface is intercepted. +Non-realtime requests are handled in middleware itself and other messages are +forwarded along. +This permits some internal messages to be sent that the UI has never directly +requested. +A similar process occurs for messages originating from the backend. + +A large portion of the middleware code is designed to manage up-to-date +pointers to the internal datastructures, in order to avoid directly accessing +anything via the pointer to the 'Master' datastructure. + +Loading +~~~~~~~ + +In order to avoid access to the backend datastructures typically a replacement +object is sent to the backend to be copied from or from which a pointer swap +can occur. + +Saving +~~~~~~ + +This is where the nice pristine hands off approach sadly comes to an end. +There simply isn't an effective means of capturing all parameters without +taking a large amount of time. +In order to permit the serialization of parameter objects, the backend is +partially 'frozen'. +This essentially prevents the backend from processing most messages from the +user interface and when this occurs the parameters which are too be serialized +can be guaranteed to be constant and thus safe to access across threads. +This class of read-only-operation can be seen as used in parameter copy/paste +operations and in saving full instances as well as instruments. + + +The User Interface +------------------ + +From an architectural standpoint the important thing to note about the user +interface is that virtual every widget has a location which is composed of a +base path and a path extension. +Most messages going to a widget are solely to this widget's one location +(occasionally they're to a few associated paths). + +This structure makes it possible for a set of widgets to get relocated +(rebase/reext) to another path. +This occurs quite frequently (e.g. "/part0/PVolume" -> "/part1/PVolume") and +it may be the occasional source of bugs. diff --git a/src/Misc/Part.cpp b/src/Misc/Part.cpp @@ -200,7 +200,7 @@ Part::Part(Allocator &alloc, Microtonal *microtonal_, FFTwrapper *fft_) oldfreq = -1.0f; - for(int i = 0; i < POLIPHONY; ++i) { + for(int i = 0; i < POLYPHONY; ++i) { partnote[i].status = KEY_OFF; partnote[i].note = -1; partnote[i].itemsplaying = 0; @@ -307,7 +307,7 @@ void Part::defaultsinstrument() */ void Part::cleanup(bool final_) { - for(int k = 0; k < POLIPHONY; ++k) + for(int k = 0; k < POLYPHONY; ++k) KillNotePos(k); for(int i = 0; i < synth->buffersize; ++i) { partoutl[i] = final_ ? 0.0f : denormalkillbuf[i]; @@ -352,7 +352,7 @@ void Part::NoteOn(unsigned char note, int masterkeyshift) { // Legato and MonoMem used vars: - int posb = POLIPHONY - 1; // Just a dummy initial value. + int posb = POLYPHONY - 1; // Just a dummy initial value. bool legatomodevalid = false; //true when legato mode is determined applicable. bool doinglegato = false; // true when we determined we do a legato note. bool ismonofirstnote = false; /*(In Mono/Legato) true when we determined @@ -368,7 +368,7 @@ void Part::NoteOn(unsigned char note, monomem[note].velocity = velocity; // Store this note's velocity. monomem[note].mkeyshift = masterkeyshift; /* Store masterkeyshift too*/ if((partnote[lastpos].status != KEY_PLAYING) - && (partnote[lastpos].status != KEY_RELASED_AND_SUSTAINED)) + && (partnote[lastpos].status != KEY_RELEASED_AND_SUSTAINED)) ismonofirstnote = true; // No other keys are held or sustained. } else if(!monomemEmpty()) monomemClear(); @@ -376,7 +376,7 @@ void Part::NoteOn(unsigned char note, lastnote = note; int pos = -1; - for(int i = 0; i < POLIPHONY; ++i) + for(int i = 0; i < POLYPHONY; ++i) if(partnote[i].status == KEY_OFF) { pos = i; break; @@ -401,14 +401,14 @@ void Part::NoteOn(unsigned char note, } else { // Legato mode is valid, but this is only a first note. - for(int i = 0; i < POLIPHONY; ++i) + for(int i = 0; i < POLYPHONY; ++i) if((partnote[i].status == KEY_PLAYING) - || (partnote[i].status == KEY_RELASED_AND_SUSTAINED)) - RelaseNotePos(i); + || (partnote[i].status == KEY_RELEASED_AND_SUSTAINED)) + ReleaseNotePos(i); // Set posb - posb = (pos + 1) % POLIPHONY; //We really want it (if the following fails) - for(int i = 0; i < POLIPHONY; ++i) + posb = (pos + 1) % POLYPHONY; //We really want it (if the following fails) + for(int i = 0; i < POLYPHONY; ++i) if((partnote[i].status == KEY_OFF) && (pos != i)) { posb = i; break; @@ -419,17 +419,17 @@ void Part::NoteOn(unsigned char note, } else // Legato mode is either off or non-applicable. if(!Ppolymode) { //if the mode is 'mono' turn off all other notes - for(int i = 0; i < POLIPHONY; ++i) + for(int i = 0; i < POLYPHONY; ++i) if(partnote[i].status == KEY_PLAYING) - RelaseNotePos(i); - RelaseSustainedKeys(); + ReleaseNotePos(i); + ReleaseSustainedKeys(); } lastlegatomodevalid = legatomodevalid; if(pos == -1) fprintf(stderr, "%s", - "NOTES TOO MANY (> POLIPHONY) - (Part.cpp::NoteOn(..))\n"); + "NOTES TOO MANY (> POLYPHONY) - (Part.cpp::NoteOn(..))\n"); else { //start the note partnote[pos].status = KEY_PLAYING; @@ -639,29 +639,29 @@ void Part::NoteOn(unsigned char note, } } - //this only relase the keys if there is maximum number of keys allowed + //this only release the keys if there is maximum number of keys allowed setkeylimit(Pkeylimit); } /* * Note Off Messages */ -void Part::NoteOff(unsigned char note) //relase the key +void Part::NoteOff(unsigned char note) //release the key { // This note is released, so we remove it from the list. if(!monomemEmpty()) monomemPop(note); - for(int i = POLIPHONY - 1; i >= 0; i--) //first note in, is first out if there are same note multiple times + for(int i = POLYPHONY - 1; i >= 0; i--) //first note in, is first out if there are same note multiple times if((partnote[i].status == KEY_PLAYING) && (partnote[i].note == note)) { if(!ctl.sustain.sustain) { //the sustain pedal is not pushed if(!Ppolymode && !monomemEmpty()) MonoMemRenote();//Play most recent still active note else - RelaseNotePos(i); + ReleaseNotePos(i); } else //the sustain pedal is pushed - partnote[i].status = KEY_RELASED_AND_SUSTAINED; + partnote[i].status = KEY_RELEASED_AND_SUSTAINED; } } @@ -679,7 +679,7 @@ void Part::PolyphonicAftertouch(unsigned char note, monomem[note].velocity = velocity; // Store this note's velocity. - for(int i = 0; i < POLIPHONY; ++i) + for(int i = 0; i < POLYPHONY; ++i) if((partnote[i].note == note) && (partnote[i].status == KEY_PLAYING)) { /* update velocity */ // compute the velocity offset @@ -755,14 +755,14 @@ void Part::SetController(unsigned int type, int par) case C_sustain: ctl.setsustain(par); if(ctl.sustain.sustain == 0) - RelaseSustainedKeys(); + ReleaseSustainedKeys(); break; case C_allsoundsoff: AllNotesOff(); //Panic break; case C_resetallcontrollers: ctl.resetall(); - RelaseSustainedKeys(); + ReleaseSustainedKeys(); if(ctl.volume.receive != 0) volume = ctl.volume.volume; else @@ -782,7 +782,7 @@ void Part::SetController(unsigned int type, int par) //more update to add here if I add controllers break; case C_allnotesoff: - RelaseAllKeys(); + ReleaseAllKeys(); break; case C_resonance_center: ctl.setresonancecenter(par); @@ -802,31 +802,31 @@ void Part::SetController(unsigned int type, int par) } } /* - * Relase the sustained keys + * Release the sustained keys */ -void Part::RelaseSustainedKeys() +void Part::ReleaseSustainedKeys() { // Let's call MonoMemRenote() on some conditions: if(Ppolymode == 0 && !monomemEmpty()) if(monomemBack() != lastnote) // Sustain controller manipulation would cause repeated same note respawn without this check. MonoMemRenote(); // To play most recent still held note. - for(int i = 0; i < POLIPHONY; ++i) - if(partnote[i].status == KEY_RELASED_AND_SUSTAINED) - RelaseNotePos(i); + for(int i = 0; i < POLYPHONY; ++i) + if(partnote[i].status == KEY_RELEASED_AND_SUSTAINED) + ReleaseNotePos(i); } /* - * Relase all keys + * Release all keys */ -void Part::RelaseAllKeys() +void Part::ReleaseAllKeys() { - for(int i = 0; i < POLIPHONY; ++i) - if((partnote[i].status != KEY_RELASED) + for(int i = 0; i < POLYPHONY; ++i) + if((partnote[i].status != KEY_RELEASED) && (partnote[i].status != KEY_OFF)) //thanks to Frank Neumann - RelaseNotePos(i); + ReleaseNotePos(i); } // Call NoteOn(...) with the most recent still held key as new note @@ -836,7 +836,7 @@ void Part::MonoMemRenote() unsigned char mmrtempnote = monomemBack(); // Last list element. monomemPop(mmrtempnote); // We remove it, will be added again in NoteOn(...). if(Pnoteon == 0) - RelaseNotePos(lastpos); + ReleaseNotePos(lastpos); else NoteOn(mmrtempnote, monomem[mmrtempnote].velocity, monomem[mmrtempnote].mkeyshift); @@ -845,19 +845,19 @@ void Part::MonoMemRenote() /* * Release note at position */ -void Part::RelaseNotePos(int pos) +void Part::ReleaseNotePos(int pos) { for(int j = 0; j < NUM_KIT_ITEMS; ++j) { if(partnote[pos].kititem[j].adnote) - partnote[pos].kititem[j].adnote->relasekey(); + partnote[pos].kititem[j].adnote->releasekey(); if(partnote[pos].kititem[j].subnote) - partnote[pos].kititem[j].subnote->relasekey(); + partnote[pos].kititem[j].subnote->releasekey(); if(partnote[pos].kititem[j].padnote) - partnote[pos].kititem[j].padnote->relasekey(); + partnote[pos].kititem[j].padnote->releasekey(); } - partnote[pos].status = KEY_RELASED; + partnote[pos].status = KEY_RELEASED; } @@ -891,26 +891,26 @@ void Part::setkeylimit(unsigned char Pkeylimit) this->Pkeylimit = Pkeylimit; int keylimit = Pkeylimit; if(keylimit == 0) - keylimit = POLIPHONY - 5; + keylimit = POLYPHONY - 5; //release old keys if the number of notes>keylimit if(Ppolymode != 0) { int notecount = 0; - for(int i = 0; i < POLIPHONY; ++i) - if((partnote[i].status == KEY_PLAYING) || (partnote[i].status == KEY_RELASED_AND_SUSTAINED)) + for(int i = 0; i < POLYPHONY; ++i) + if((partnote[i].status == KEY_PLAYING) || (partnote[i].status == KEY_RELEASED_AND_SUSTAINED)) notecount++; int oldestnotepos = -1; if(notecount > keylimit) //find out the oldest note - for(int i = 0; i < POLIPHONY; ++i) { + for(int i = 0; i < POLYPHONY; ++i) { int maxtime = 0; - if(((partnote[i].status == KEY_PLAYING) || (partnote[i].status == KEY_RELASED_AND_SUSTAINED)) && (partnote[i].time > maxtime)) { + if(((partnote[i].status == KEY_PLAYING) || (partnote[i].status == KEY_RELEASED_AND_SUSTAINED)) && (partnote[i].time > maxtime)) { maxtime = partnote[i].time; oldestnotepos = i; } } if(oldestnotepos != -1) - RelaseNotePos(oldestnotepos); + ReleaseNotePos(oldestnotepos); } } @@ -973,7 +973,7 @@ void Part::ComputePartSmps() partfxinputr[nefx][i] = 0.0f; } - for(unsigned k = 0; k < POLIPHONY; ++k) { + for(unsigned k = 0; k < POLYPHONY; ++k) { if(partnote[k].status == KEY_OFF) continue; partnote[k].time++; @@ -1009,7 +1009,7 @@ void Part::ComputePartSmps() partoutl[i] *= tmp; partoutr[i] *= tmp; } - for(int k = 0; k < POLIPHONY; ++k) + for(int k = 0; k < POLYPHONY; ++k) KillNotePos(k); killallnotes = false; for(int nefx = 0; nefx < NUM_PART_EFX; ++nefx) @@ -1057,7 +1057,7 @@ void Part::setkititemstatus(unsigned kititem, bool Penabled_) kkit.Pname[0] = '\0'; //Reset notes s.t. stale buffers will not get read - for(int k = 0; k < POLIPHONY; ++k) + for(int k = 0; k < POLYPHONY; ++k) KillNotePos(k); } else { @@ -1216,7 +1216,7 @@ void Part::kill_rt(void) { for(int i=0; i<NUM_PART_EFX; ++i) partefx[i]->kill(); - for(int k = 0; k < POLIPHONY; ++k) + for(int k = 0; k < POLYPHONY; ++k) KillNotePos(k); } diff --git a/src/Misc/Part.h b/src/Misc/Part.h @@ -54,8 +54,8 @@ class Part int masterkeyshift) REALTIME; void AllNotesOff() REALTIME; //panic void SetController(unsigned int type, int par) REALTIME; - void RelaseSustainedKeys() REALTIME; //this is called when the sustain pedal is relased - void RelaseAllKeys() REALTIME; //this is called on AllNotesOff controller + void ReleaseSustainedKeys() REALTIME; //this is called when the sustain pedal is released + void ReleaseAllKeys() REALTIME; //this is called on AllNotesOff controller /* The synthesizer part output */ void ComputePartSmps() REALTIME; //Part output @@ -119,7 +119,7 @@ class Part bool Ppolymode; //Part mode - 0=monophonic , 1=polyphonic bool Plegatomode; // 0=normal, 1=legato - unsigned char Pkeylimit; //how many keys are alowed to be played same time (0=off), the older will be relased + unsigned char Pkeylimit; //how many keys are alowed to be played same time (0=off), the older will be released char *Pname; //name of the instrument struct { //instrument additional information @@ -136,7 +136,7 @@ class Part *partfxinputr[NUM_PART_EFX + 1]; //partfxinput l/r [NUM_PART_EFX] is for "no effect" buffer enum NoteStatus { - KEY_OFF, KEY_PLAYING, KEY_RELASED_AND_SUSTAINED, KEY_RELASED + KEY_OFF, KEY_PLAYING, KEY_RELEASED_AND_SUSTAINED, KEY_RELEASED }; float volume, oldvolumel, oldvolumer; //this is applied by Master @@ -155,7 +155,7 @@ class Part private: void RunNote(unsigned k); void KillNotePos(int pos); - void RelaseNotePos(int pos); + void ReleaseNotePos(int pos); void MonoMemRenote(); // MonoMem stuff. int killallnotes; //is set to 1 if I want to kill all notes @@ -191,7 +191,7 @@ class Part store the velocity and masterkeyshift values of a given note (the list only store note values). For example 'monomem[note].velocity' would be the velocity value of the note 'note'.*/ - PartNotes partnote[POLIPHONY]; + PartNotes partnote[POLYPHONY]; float oldfreq; //this is used for portamento Microtonal *microtonal; diff --git a/src/Misc/Util.cpp b/src/Misc/Util.cpp @@ -23,8 +23,9 @@ #include "Util.h" #include <vector> #include <cassert> -#include <math.h> -#include <stdio.h> +#include <cmath> +#include <cstdio> +#include <fstream> #include <err.h> #include <sys/types.h> @@ -136,6 +137,43 @@ void os_sleep(long length) usleep(length); } +//!< maximum lenght a pid has on any POSIX system +//!< this is an estimation, but more than 12 looks insane +constexpr std::size_t max_pid_len = 12; + +//!< safe pid lenght guess, posix conform +std::size_t os_guess_pid_length() +{ + const char* pid_max_file = "/proc/sys/kernel/pid_max"; + if(-1 == access(pid_max_file, R_OK)) { + return max_pid_len; + } + else { + std::ifstream is(pid_max_file); + if(!is.good()) + return max_pid_len; + else { + std::string s; + is >> s; + for(const auto& c : s) + if(c < '0' || c > '9') + return max_pid_len; + return std::min(s.length(), max_pid_len); + } + } +} + +//!< returns pid padded, posix conform +std::string os_pid_as_padded_string() +{ + char result_str[max_pid_len << 1]; + std::fill_n(result_str, max_pid_len, '0'); + std::size_t written = snprintf(result_str + max_pid_len, max_pid_len, + "%d", (int)getpid()); + // the below pointer should never cause segfaults: + return result_str + max_pid_len + written - os_guess_pid_length(); +} + std::string legalizeFilename(std::string filename) { for(int i = 0; i < (int) filename.size(); ++i) { @@ -179,3 +217,4 @@ const char *message_snip(const char *m) while(*m && *m!='/')++m; return *m?m+1:m; } + diff --git a/src/Misc/Util.h b/src/Misc/Util.h @@ -52,6 +52,9 @@ void set_realtime(); /**Os independent sleep in microsecond*/ void os_sleep(long length); +//! returns pid padded to maximum pid lenght, posix conform +std::string os_pid_as_padded_string(); + std::string legalizeFilename(std::string filename); extern float *denormalkillbuf; /**<the buffer to add noise in order to avoid denormalisation*/ diff --git a/src/Nio/JackEngine.cpp b/src/Nio/JackEngine.cpp @@ -30,10 +30,13 @@ #include <sys/stat.h> #include <cassert> #include <cstring> +#include <unistd.h> // access() +#include <fstream> // std::istream #include "Nio.h" #include "OutMgr.h" #include "InMgr.h" +#include "Misc/Util.h" #include "JackEngine.h" @@ -65,6 +68,9 @@ bool JackEngine::connectServer(string server) string postfix = Nio::getPostfix(); if(!postfix.empty()) clientname += "_" + postfix; + if(Nio::pidInClientName) + clientname += "_" + os_pid_as_padded_string(); + jack_status_t jackstatus; bool use_server_name = server.size() && server.compare("default") != 0; jack_options_t jopts = (jack_options_t) diff --git a/src/Nio/JackMultiEngine.cpp b/src/Nio/JackMultiEngine.cpp @@ -28,6 +28,7 @@ #include <cassert> #include "Nio.h" +#include "Misc/Util.h" #include "../Misc/Master.h" #include "../Misc/Part.h" #include "../Misc/MiddleWare.h" @@ -82,6 +83,8 @@ bool JackMultiEngine::Start(void) string postfix = Nio::getPostfix(); if(!postfix.empty()) clientname += "_" + postfix; + if(Nio::pidInClientName) + clientname += "_" + os_pid_as_padded_string(); jack_status_t jackstatus; impl->client = jack_client_open(clientname.c_str(), JackNullOption, &jackstatus); diff --git a/src/Nio/Nio.cpp b/src/Nio/Nio.cpp @@ -24,9 +24,10 @@ OutMgr *out = NULL; EngineMgr *eng = NULL; string postfix; -bool Nio::autoConnect = false; -string Nio::defaultSource = IN_DEFAULT; -string Nio::defaultSink = OUT_DEFAULT; +bool Nio::autoConnect = false; +bool Nio::pidInClientName = false; +string Nio::defaultSource = IN_DEFAULT; +string Nio::defaultSink = OUT_DEFAULT; void Nio::init(class Master *master) { diff --git a/src/Nio/Nio.h b/src/Nio/Nio.h @@ -43,6 +43,7 @@ namespace Nio void waveEnd(void); extern bool autoConnect; + extern bool pidInClientName; extern std::string defaultSource; extern std::string defaultSink; }; diff --git a/src/Synth/ADnote.cpp b/src/Synth/ADnote.cpp @@ -1694,15 +1694,15 @@ int ADnote::noteout(float *outl, float *outr) /* - * Relase the key (NoteOff) + * Release the key (NoteOff) */ -void ADnote::relasekey() +void ADnote::releasekey() { for(int nvoice = 0; nvoice < NUM_VOICES; ++nvoice) NoteVoicePar[nvoice].releasekey(); - NoteGlobalPar.FreqEnvelope->relasekey(); - NoteGlobalPar.FilterEnvelope->relasekey(); - NoteGlobalPar.AmpEnvelope->relasekey(); + NoteGlobalPar.FreqEnvelope->releasekey(); + NoteGlobalPar.FilterEnvelope->releasekey(); + NoteGlobalPar.AmpEnvelope->releasekey(); } /* @@ -1721,15 +1721,15 @@ void ADnote::Voice::releasekey() if(!Enabled) return; if(AmpEnvelope) - AmpEnvelope->relasekey(); + AmpEnvelope->releasekey(); if(FreqEnvelope) - FreqEnvelope->relasekey(); + FreqEnvelope->releasekey(); if(FilterEnvelope) - FilterEnvelope->relasekey(); + FilterEnvelope->releasekey(); if(FMFreqEnvelope) - FMFreqEnvelope->relasekey(); + FMFreqEnvelope->releasekey(); if(FMAmpEnvelope) - FMAmpEnvelope->relasekey(); + FMAmpEnvelope->releasekey(); } void ADnote::Voice::kill(Allocator &memory) diff --git a/src/Synth/ADnote.h b/src/Synth/ADnote.h @@ -51,7 +51,7 @@ class ADnote:public SynthNote void legatonote(LegatoParams pars); int noteout(float *outl, float *outr); - void relasekey(); + void releasekey(); int finished() const; private: diff --git a/src/Synth/Envelope.cpp b/src/Synth/Envelope.cpp @@ -30,7 +30,7 @@ Envelope::Envelope(EnvelopeParams &pars, float basefreq) if(envpoints > MAX_ENVELOPE_POINTS) envpoints = MAX_ENVELOPE_POINTS; envsustain = (pars.Penvsustain == 0) ? -1 : pars.Penvsustain; - forcedrelase = pars.Pforcedrelease; + forcedrelease = pars.Pforcedrelease; envstretch = powf(440.0f / basefreq, pars.Penvstretch / 64.0f); linearenvelope = pars.Plinearenvelope; @@ -92,14 +92,14 @@ Envelope::~Envelope() /* - * Relase the key (note envelope) + * Release the key (note envelope) */ -void Envelope::relasekey() +void Envelope::releasekey() { if(keyreleased) return; keyreleased = true; - if(forcedrelase != 0) + if(forcedrelease != 0) t = 0.0f; } @@ -119,7 +119,7 @@ float Envelope::envout() return envoutval; } - if(keyreleased && (forcedrelase != 0)) { //do the forced release + if(keyreleased && (forcedrelease != 0)) { //do the forced release int tmp = (envsustain < 0) ? (envpoints - 1) : (envsustain + 1); //if there is no sustain point, use the last point for release if(envdt[tmp] < 0.00000001f) @@ -130,7 +130,7 @@ float Envelope::envout() if(t >= 1.0f) { currentpoint = envsustain + 2; - forcedrelase = 0; + forcedrelease = 0; t = 0.0f; inct = envdt[currentpoint]; if((currentpoint >= envpoints) || (envsustain < 0)) @@ -175,7 +175,7 @@ float Envelope::envout_dB() if(linearenvelope != 0) return envout(); - if((currentpoint == 1) && (!keyreleased || (forcedrelase == 0))) { //first point is always lineary interpolated + if((currentpoint == 1) && (!keyreleased || (forcedrelease == 0))) { //first point is always lineary interpolated float v1 = env_dB2rap(envval[0]); float v2 = env_dB2rap(envval[1]); out = v1 + (v2 - v1) * t; diff --git a/src/Synth/Envelope.h b/src/Synth/Envelope.h @@ -34,7 +34,7 @@ class Envelope Envelope(class EnvelopeParams &pars, float basefreq); /**Destructor*/ ~Envelope(); - void relasekey(); + void releasekey(); float envout(); float envout_dB(); /**Determines the status of the Envelope @@ -51,7 +51,7 @@ class Envelope int linearenvelope; int currentpoint; //current envelope point (starts from 1) - int forcedrelase; + int forcedrelease; bool keyreleased; //if the key was released bool envfinish; float t; // the time from the last point diff --git a/src/Synth/PADnote.cpp b/src/Synth/PADnote.cpp @@ -413,9 +413,9 @@ int PADnote::finished() const return finished_; } -void PADnote::relasekey() +void PADnote::releasekey() { - NoteGlobalPar.FreqEnvelope->relasekey(); - NoteGlobalPar.FilterEnvelope->relasekey(); - NoteGlobalPar.AmpEnvelope->relasekey(); + NoteGlobalPar.FreqEnvelope->releasekey(); + NoteGlobalPar.FilterEnvelope->releasekey(); + NoteGlobalPar.AmpEnvelope->releasekey(); } diff --git a/src/Synth/PADnote.h b/src/Synth/PADnote.h @@ -37,7 +37,7 @@ class PADnote:public SynthNote int noteout(float *outl, float *outr); int finished() const; - void relasekey(); + void releasekey(); private: void setup(float freq, float velocity, int portamento_, int midinote, bool legato = false); diff --git a/src/Synth/SUBnote.cpp b/src/Synth/SUBnote.cpp @@ -578,17 +578,17 @@ int SUBnote::noteout(float *outl, float *outr) } /* - * Relase Key (Note Off) + * Release Key (Note Off) */ -void SUBnote::relasekey() +void SUBnote::releasekey() { - AmpEnvelope->relasekey(); + AmpEnvelope->releasekey(); if(FreqEnvelope) - FreqEnvelope->relasekey(); + FreqEnvelope->releasekey(); if(BandWidthEnvelope) - BandWidthEnvelope->relasekey(); + BandWidthEnvelope->releasekey(); if(GlobalFilterEnvelope) - GlobalFilterEnvelope->relasekey(); + GlobalFilterEnvelope->releasekey(); } /* diff --git a/src/Synth/SUBnote.h b/src/Synth/SUBnote.h @@ -36,7 +36,7 @@ class SUBnote:public SynthNote void legatonote(LegatoParams pars); int noteout(float *outl, float *outr); //note output,return 0 if the note is finished - void relasekey(); + void releasekey(); int finished() const; private: diff --git a/src/Synth/SynthNote.h b/src/Synth/SynthNote.h @@ -57,7 +57,7 @@ class SynthNote //TODO fix this spelling error [noisey commit] /**Release the key for the note and start release portion of envelopes.*/ - virtual void relasekey() = 0; + virtual void releasekey() = 0; /**Return if note is finished. * @return finished=1 unfinished=0*/ diff --git a/src/Tests/AdNoteTest.h b/src/Tests/AdNoteTest.h @@ -146,7 +146,7 @@ class AdNoteTest:public CxxTest::TestSuite TS_ASSERT_DELTA(outL[255], 0.254609f, 0.0001f); - note->relasekey(); + note->releasekey(); note->noteout(outL, outR); diff --git a/src/Tests/PadNoteTest.h b/src/Tests/PadNoteTest.h @@ -161,7 +161,7 @@ class PadNoteTest:public CxxTest::TestSuite TS_ASSERT_DELTA(outL[255], 0.0660f, 0.0005f); - note->relasekey(); + note->releasekey(); note->noteout(outL, outR); diff --git a/src/Tests/SubNoteTest.h b/src/Tests/SubNoteTest.h @@ -128,7 +128,7 @@ class SubNoteTest:public CxxTest::TestSuite TS_ASSERT_DELTA(outL[255], 0.0000f, 0.0001f); - note->relasekey(); + note->releasekey(); note->noteout(outL, outR); diff --git a/src/UI/ADnoteUI.fl b/src/UI/ADnoteUI.fl @@ -235,7 +235,7 @@ class ADvoiceUI {open : {public Fl_Group} callback {if (o->value()==0) voiceFMfreqenvgroup->deactivate(); else voiceFMfreqenvgroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {545 305 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {545 305 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PFMFreqEnvelopeEnabled");} class Fl_Osc_Check } @@ -301,7 +301,7 @@ o->redraw();} callback {if (o->value()==0) voiceFMampenvgroup->deactivate(); else voiceFMampenvgroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {545 150 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {545 150 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PFMAmpEnvelopeEnabled");} class Fl_Osc_Check } @@ -433,7 +433,7 @@ o->redraw();} callback {if (o->value()==0) voicefreqenvgroup->deactivate(); else voicefreqenvgroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {15 310 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {15 310 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PFreqEnvelopeEnabled");} class Fl_Osc_Check } @@ -448,7 +448,7 @@ o->redraw();} callback {if (o->value()==0) voicefreqlfogroup->deactivate(); else voicefreqlfogroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {225 311 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {225 311 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PFreqLfoEnabled");} class Fl_Osc_Check } @@ -644,7 +644,7 @@ voiceonbutton->redraw();} open callback {if (o->value()==0) voiceampenvgroup->deactivate(); else voiceampenvgroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {15 110 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {15 110 50 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PAmpEnvelopeEnabled");} class Fl_Osc_Check } @@ -659,7 +659,7 @@ o->redraw();} callback {if (o->value()==0) voiceamplfogroup->deactivate(); else voiceamplfogroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {15 185 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {15 185 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PAmpLfoEnabled");} class Fl_Osc_Check } @@ -692,7 +692,7 @@ o->redraw();} callback {if (o->value()==0) voicefilterenvgroup->deactivate(); else voicefilterenvgroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {255 119 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {255 119 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PFilterEnvelopeEnabled");} class Fl_Osc_Check } @@ -707,7 +707,7 @@ o->redraw();} callback {if (o->value()==0) voicefilterlfogroup->deactivate(); else voicefilterlfogroup->activate(); o->redraw();} - tooltip {Forced Relase} xywh {255 196 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 + tooltip {Forced Release} xywh {255 196 55 10} down_box DOWN_BOX labelfont 1 labelsize 11 code0 {o->init("PFilterLfoEnabled");} class Fl_Osc_Check } diff --git a/src/UI/Connection.cpp b/src/UI/Connection.cpp @@ -176,6 +176,8 @@ rtosc::Ports uiPorts::ports = { void GUI::raiseUi(ui_handle_t gui, const char *message) { + if(!gui) + return; MasterUI *mui = (MasterUI*)gui; mui->osc->tryLink(message); //printf("got message for UI '%s'\n", message); diff --git a/src/UI/EnvelopeUI.fl b/src/UI/EnvelopeUI.fl @@ -134,7 +134,7 @@ freeedit->redraw();} } Fl_Check_Button forcedreleasecheck { label frcR - tooltip {Forced Relase} xywh {410 165 40 15} down_box DOWN_BOX labelsize 10 + tooltip {Forced Release} xywh {410 165 40 15} down_box DOWN_BOX labelsize 10 code0 {o->init("Pforcedrelease");} code1 {//TODO if (Pfreemode==0) o->hide();} class Fl_Osc_Check @@ -218,7 +218,7 @@ envfree->redraw();} } Fl_Check_Button e1forcedrelease { label frcR - tooltip {Forced Relase} xywh {180 35 20 15} down_box DOWN_BOX labelsize 10 align 6 + tooltip {Forced Release} xywh {180 35 20 15} down_box DOWN_BOX labelsize 10 align 6 code0 {o->init("Pforcedrelease");} class Fl_Osc_Check } @@ -380,7 +380,7 @@ envfree->redraw();} } Fl_Check_Button e3forcedrelease { label frcR - tooltip {Forced Relase} xywh {250 30 15 20} down_box DOWN_BOX labelsize 10 align 6 + tooltip {Forced Release} xywh {250 30 15 20} down_box DOWN_BOX labelsize 10 align 6 code0 {o->init("Pforcedrelease");} class Fl_Osc_Check } diff --git a/src/UI/MasterUI.fl b/src/UI/MasterUI.fl @@ -462,7 +462,7 @@ if (fl_choice("The file *might* exist. \\nOverwrite it?","No","Yes",NULL)) { } Fl_Button {} { label {Panic!} - callback {virkeyboard->relaseallkeys(); + callback {virkeyboard->releaseallkeys(); o->oscWrite("Panic");} xywh {285 34 105 53} color 90 labelfont 1 class Fl_Osc_Button @@ -1034,7 +1034,7 @@ bankui->show();} } Fl_Choice partrcv { label {Midi Channel Receive} - callback {virkeys->relaseallkeys(0); + callback {virkeys->releaseallkeys(0); virkeys->midich=(int) o->value();} open tooltip {receive from Midi channel} xywh {140 157 65 18} down_box BORDER_BOX labelsize 10 align 130 textfont 1 code0 {char nrstr[10]; for(int i=0;i<NUM_MIDI_CHANNELS;i++){sprintf(nrstr,"Ch%d",i+1);if (i!=9) o->add(nrstr); else o->add("Dr10");};} @@ -1375,7 +1375,7 @@ pthread_mutex_unlock(&master->mutex);*/} } Fl_Button {} { label {Stop ALL sounds!} - callback {virkeyboard->relaseallkeys(); + callback {virkeyboard->releaseallkeys(); o->oscWrite("Panic");} xywh {5 149 115 31} color 90 labelfont 1 labelsize 10 class Fl_Osc_Button @@ -1394,7 +1394,7 @@ simpleglobalfinedetuneslider->do_callback();} } Fl_Counter simplenpartcounter { label Part - callback {virkeys->relaseallkeys(0); + callback {virkeys->releaseallkeys(0); npartcounter->value(o->value()); npart=(int) o->value()-1; @@ -1406,7 +1406,7 @@ simplerefresh();} } Fl_Counter {} { label {Keyb.Oct.} - callback {virkeys->relaseallkeys(0); + callback {virkeys->releaseallkeys(0); virkeys->midioct=(int) o->value(); virkeys->take_focus();} tooltip {Midi Octave} xywh {5 190 55 20} type Simple labelsize 11 align 8 when 6 minimum 0 maximum 5 step 1 textfont 1 textsize 11 diff --git a/src/UI/PartUI.fl b/src/UI/PartUI.fl @@ -476,7 +476,7 @@ osc->write("/setkeylimit", "i", val);} open label Sustain callback {//TODO /*if (ctl.sustain.receive==0) { - RelaseSustainedKeys(); + ReleaseSustainedKeys(); ctl.setsustain(0); };*/} tooltip {Sustain pedal enable} xywh {205 80 60 20} box THIN_UP_BOX down_box DOWN_BOX labelsize 10 diff --git a/src/UI/VirKeyboard.fl b/src/UI/VirKeyboard.fl @@ -221,10 +221,10 @@ if ((event==FL_PUSH)||(event==FL_DRAG)||(event==FL_RELEASE)){ if ((event==FL_PUSH)&&(Fl::event_shift()!=0)) { if (pressed[kpos]==0) presskey(kpos,0,1); - else relasekey(kpos,1); + else releasekey(kpos,1); }; if ((event==FL_RELEASE)&&(Fl::event_shift()==0)) - relaseallkeys(1); + releaseallkeys(1); take_focus(); }; @@ -255,7 +255,7 @@ if ((event==FL_KEYDOWN)||(event==FL_KEYUP)){ if (kpos==-1) return(0); if ((event==FL_KEYUP) && (Fl::event_key(key)==0) && (Fl::get_key(key)!=0)) return(0); if (event==FL_KEYDOWN) presskey(kpos,0,2); - else relasekey(kpos,2); + else releasekey(kpos,2); }; return(1);} {} @@ -265,13 +265,13 @@ return(1);} {} //when the user uses the shift key if (nk>=N_OCT*12) return; if ((nk<0)&&(exclusive==0)) { - relaseallkeys(type); + releaseallkeys(type); return; }; if (nk<0) return; if (pressed[nk]!=0) return;//the key is already pressed -if (exclusive!=0) relaseallkeys(type); +if (exclusive!=0) releaseallkeys(type); pressed[nk]=type; damage(1); @@ -282,7 +282,7 @@ if (rndvelocity!=0){ osc->write(loc+"noteOn", "iii", midich,nk+midioct*12,(int)vel);} {} } - Function {relasekey(int nk,int type)} {} { + Function {releasekey(int nk,int type)} {} { code {if ((nk<0)||(nk>=N_OCT*12)) return; if (pressed[nk]==0) return;//the key is not pressed if ((type!=0)&&(pressed[nk]!=type)) return; @@ -294,8 +294,8 @@ damage(1); osc->write(loc+"noteOff", "ii", midich,nk+12*midioct);} {} } - Function {relaseallkeys(int type)} {} { - code {for (int i=0;i<N_OCT*12;i++) relasekey(i,type);} {} + Function {releaseallkeys(int type)} {} { + code {for (int i=0;i<N_OCT*12;i++) releasekey(i,type);} {} } decl {int pressed[N_OCT*12+1];} {private local } @@ -319,7 +319,7 @@ class VirKeyboard {open } { Fl_Window virkeyboardwindow { label {Virtual Keyboard - ZynAddSubFX} - callback {relaseallkeys(); + callback {releaseallkeys(); virkeyboardwindow->hide();} open xywh {103 620 650 130} type Double visible } { @@ -331,7 +331,7 @@ virkeyboardwindow->hide();} open } Fl_Counter {} { label {"qwer.." Oct} - callback {relaseallkeys(); + callback {releaseallkeys(); virkeys->keyoct1=(int) o->value(); virkeys->take_focus();} tooltip {keys "q2w3er5t6y..." octave} xywh {380 95 45 15} type Simple labelsize 10 align 4 when 6 minimum 0 maximum 5 step 1 textfont 1 textsize 10 @@ -339,7 +339,7 @@ virkeys->take_focus();} } Fl_Counter {} { label {"zxcv.." Oct} - callback {relaseallkeys(); + callback {releaseallkeys(); virkeys->keyoct2=(int) o->value(); virkeys->take_focus();} tooltip {keys "zsxdcvgbh..." octave} xywh {380 110 45 15} type Simple labelsize 10 align 4 when 6 minimum 0 maximum 5 step 1 textfont 1 textsize 10 @@ -354,7 +354,7 @@ virkeys->take_focus();} } Fl_Counter {} { label {Oct.} - callback {relaseallkeys(); + callback {releaseallkeys(); virkeys->midioct=(int) o->value(); virkeys->take_focus();} tooltip {Midi Octave} xywh {255 100 55 20} type Simple labelsize 11 align 4 when 6 minimum 0 maximum 5 step 1 textfont 1 textsize 11 @@ -362,7 +362,7 @@ virkeys->take_focus();} } Fl_Button {} { label Close - callback {relaseallkeys(); + callback {releaseallkeys(); virkeyboardwindow->hide();} xywh {545 105 55 20} box THIN_UP_BOX } @@ -469,7 +469,7 @@ pitchwheelroller->do_callback();} } Fl_Choice partrcv { label {MIDI Ch.} - callback {relaseallkeys(); + callback {releaseallkeys(); virkeys->midich=(int) o->value(); virkeys->take_focus();} open tooltip {Send to Midi Channel} xywh {20 105 65 20} down_box BORDER_BOX labelsize 10 align 5 textfont 1 textsize 10 @@ -493,9 +493,9 @@ make_window();} {} } { code {virkeyboardwindow->show();} {} } - Function {relaseallkeys()} {open + Function {releaseallkeys()} {open } { - code {virkeys->relaseallkeys(0);} {} + code {virkeys->releaseallkeys(0);} {} } decl {int midictl;} {private local } diff --git a/src/globals.h b/src/globals.h @@ -110,9 +110,9 @@ typedef std::complex<fftw_real> fft_t; #define NUM_VOICES 8 /* - * The poliphony (notes) + * The polyphony (notes) */ -#define POLIPHONY 60 +#define POLYPHONY 60 /* * Number of system effects diff --git a/src/main.cpp b/src/main.cpp @@ -202,6 +202,9 @@ int main(int argc, char *argv[]) "auto-connect", 0, NULL, 'a' }, { + "pid-in-client-name", 0, NULL, 'p' + }, + { "output", 1, NULL, 'O' }, { @@ -228,7 +231,7 @@ int main(int argc, char *argv[]) /**\todo check this process for a small memory leak*/ opt = getopt_long(argc, argv, - "l:L:r:b:o:I:O:N:e:hvaSDUY", + "l:L:r:b:o:I:O:N:e:hvapSDUY", opts, &option_index); char *optarguments = optarg; @@ -315,6 +318,9 @@ int main(int argc, char *argv[]) case 'a': Nio::autoConnect = true; break; + case 'p': + Nio::pidInClientName = true; + break; case 'e': GETOP(execAfterInit); break; @@ -361,6 +367,8 @@ int main(int argc, char *argv[]) " -U , --no-gui\t\t\t\t Run ZynAddSubFX without user interface\n" << " -N , --named\t\t\t\t Postfix IO Name when possible\n" << " -a , --auto-connect\t\t\t AutoConnect when using JACK\n" + << " -p , --pid-in-client-name\t\t Append PID to (JACK) " + "client name\n" << " -O , --output\t\t\t\t Set Output Engine\n" << " -I , --input\t\t\t\t Set Input Engine\n" << " -e , --exec-after-init\t\t Run post-initialization script\n" @@ -424,7 +432,9 @@ int main(int argc, char *argv[]) } - gui = GUI::createUi(middleware->spawnUiApi(), &Pexitprogram); + gui = NULL; + if(!noui) + gui = GUI::createUi(middleware->spawnUiApi(), &Pexitprogram); middleware->setUiCallback(GUI::raiseUi, gui); middleware->setIdleCallback([](){GUI::tickUi(gui);}); middlewarepointer = middleware;