Difference between revisions of "Contributing to Zynthian Development backup"
| m (Wyleu moved page Contributing to Zynthian Development to Contributing to Zynthian Development backup) | |||
| (139 intermediate revisions by 6 users not shown) | |||
| Line 1: | Line 1: | ||
| − | |||
| − | ===  | + | === So what is an IDE ?=== | 
| + | We could operate all of this using the command line and git commands, but unless you really want to, don't. Use an IDE. | ||
| − | + | An IDE (Integrated Development Environment) is a collection of programs that work together to ease the difficulties of acquiring, writing, modifying, testing and debugging code that you write along with the possibility of many, many other functions you probably never considered. At the moment there is one that seems to be really sweeping the board with this (yes, I know, emacs users, but I worked on pycharm and intellij and I know which I prefer out of the three). | |
| − | + | It's Microsoft's Visual Studio code,  https://code.visualstudio.com/ know to friends and enemies alike as VSC (or VSCode). | |
| − | + | [[File:Zynthian-visual-studio-code.png|600px|center]] | |
| − | + | ==== So what is a host machine? ==== | |
| − | |||
| − | |||
| − | |||
| − | + | Visual Studio Code is a considerable piece of software, Microsoft know an awful lot about writing code and they have placed a suberb, well maintained IDE into the hands of absolutely anybody with a desire to write code. (If you date back far enough to remember "a computer on every desk running Microsoft software" you might generate a rye smile at this point, but the software world is a very different place now, and Microsoft are perfectly willing to produce their IDE for every conceivable platform and operating system they come across and make it work. Anyone who plays around with Arduinos and such will find VSC's platformIO plugin is a considerable improvement on the Arduino's own IDE, and plugins like this are the secret of VSC.) The Python environment is a plugin, tools to view git repositories are a plugin, remote access components are a plugin. I haven't looked but I don't doubt there is some MIDI related plugin out there...  | |
| + | So we have a considerable chunk of code to run, which we are going to use to write a considerable chunk of code on a Raspberry Pi... Can you run VSC on a Raspberry Pi? Well yes you can, indeed I have run my VSC environment on a Pi5. But I wouldn't run zynthian and the VSC core UI code on a zynthian. There is a cleverer way. You use the remote connection plugin, to access a separate zynthian doing musical stuff. The VSC remote component installs itself onto the Pi, once you present it with the correct password (Did I mention about getting your passwords sorted out...?), and it handles many, many areas of this process that you can lose yourself into setting up yourself. I have written this process up for other environments and it was..... Involved. OK VSC advert over.   | ||
| − | + | So our development environment consists of two machines, one is the zynthian itself, and the other is another computer running the VSC GI code.   | |
| − | + | This other machine is the host computer and it can be any number of bizarre, powerful, stylish, home built, clapped out, PC's and goodness knows what else, but as long as it runs VSC and has a network connection you are good to go. As I say my host machine is now a Pi5 running VSC very effectively, but your's could be MAC or a PC or goodness knows what else. This is why using a common IDE provides dividends because we don't need to explain all the different access mechanisms for all these machines. Look up how you install VSC on your host machine and do that. It's pretty easy and if anyone wants to write up specific versions then great! Just say so on the forum. | |
| − | + | ==== How do I install VSC on my host machine? ==== | |
| − | |||
| − | + | Go to [https://code.visualstudio.com/Download VSC Download page] and pick the appropriate version. For your particular Pi OS. If you are running 64 bit software pick .deb arm64, if you are on a 32 bit Pi OS then pick .deb arm32.If you are on MAC or PC then download accordingly. | |
| − | + | [[File:Microsft-vsc-download page.png|600px|thumb|center|VSC Download Page]] | |
| − | This  | + | This will download the selected version and you install as appropriate to your operating system. On my Pi5 I open the download folder  | 
| − | = | + | <br clear=all> | 
| − | + | [[File:File-download.png|600px|thumb|center|Pi Desktop Download Folder]] | |
| − | + | and right click on the downloaded file and select Package Install.   | |
| − | = | + | <br clear=all> | 
| − | |||
| − | |||
| − | |||
| − | + | It will and you will have a new icon appear in your desktop menu in the Programming section.   | |
| − | + | [[File:Visual-studio-code-menu-item.png|600px|thumb|center|Pi Desktop VSC menu item]] | |
| − | + | Simply pressing on the icon will run Visual Studio Code, but for the moment there is a bit more to consider... | |
| − | |||
| − | + | <br clear=all> | |
| − | |||
| − | + | ==== Where are the zynthian source files located on a zynthian? ==== | |
| − | + | The zynthian software exists as a set of directories (or folders if you prefer - but get used to us using the more accurate "directories") within a directory on the zynthian called... /zynthian.   | |
| − | + | [[File:Zynthian-home-directory.png|600px|thumb|center|Zynthian home directory]] | |
| − | + | <br clear=all> | |
| + | The GUI software running on the zynth that operates the touchscreen are located within the zynthian-ui directory. and there is a file called zynthian.sh. This is a shell script which isn't written in python but in a ''bash''. It is a collection of commands one runs on the terminal we mentioned earlier an allows various things to be configured, before the actual python script that does the work, which is called zynthian_main.py, is run.  | ||
| − | + | [[File:Zynthian-ui-folder.png|600px|thumb|center|Zynthian-ui directory]] | |
| − | |||
| − | + | One thing the shell script sets up is the debugging level which decides what level of message the zynthian generates when it encounters a logging command in the python script... | |
| − | |||
| − | + | <br clear=all> | |
| − | + | ==== What happens if zynthian crashes and how do I stop it restarting on errors ? ==== | |
| + | When a zynthian starts up, most of the early functions and what gets run gets handled by a Linux tool called systemd. This organises what order things happen in and nursemaids programs, setting up procedures for what to do when programs crash or misbehave. For instance there is no point in starting up the network devices if there is no network, and this is the sort of thing systemd does on your behalf. In the zynthian case it starts up the zynthian program using the scripts we discussed in /zynthian/zynthian-ui and handles a zynthian by restarting the UI if the software crashes for some reason. This is what you want if you are performing but is decidedly irritating if you are trying to write and test code.  | ||
| + | There are magic incantations to control zynthian via systemd  | ||
| − | + |  systemctl stop zynthian | |
| − | + |  systemctl start zynthian  | |
| + |   systemtl status zynthian | ||
| − | + | So to stop a running zynthian you issue the first of these and this will tell systemd to ignore the restart so you can take over the running of it. | |
| − | + | ==== How do I start a X11 windows server so I can run the code in a debugger? ==== | |
| − | |||
| − | + | Once you have stopped the zynthian process, the GUI layer (X11) required to run zynthian is also stopped so you can start zynthian within the VSC debugger. To allow this to happen you have to start the X11 system to allow you to have something to connect to. | |
| − | + | To do this in the window you typed "systemctl stop zynthian", type | |
| − | |||
| − | + |  X -r -s 0 | |
| − | + | You can then start the debugger in VSC. | |
| − | + | === How do I Debug a Zynthian? === | |
| − | + | A careful balancing act is required here because there are a lot of entities attempting to work harmoniously here and a Pi can be a small world. | |
| − | + | * The zynthian running on Pi, preferably as meaty as one can get. A Pi3 for instance will run out of resources with more recent engines. | |
| + | * A Separate Computer Host from where the operation is all controlled. | ||
| + | * Some extra code loaded on the remote machine that communicates with the controlling programme on the host machine. | ||
| + | * A VNC server (A copy of the GUI image that runs in a browser) running on the zynthian to allow the screen to be examined during debugging.  | ||
| + | * Some alterations to the launch.json file in .vscode to set things up nicely. | ||
| − | + | Visual Studio Code is an excellent environment for this and it fairly effortlessly sets up most of the environment for you. | |
| − | + | It will need to be running on your host machine, be it Windows, Mac or Linux which could be running on a Pi. VSC runs on them all. Indeed this is being typed on a Raspberry Pi desktop machine which has VSC open and already connected to Pi4 (zynthian-rack6.local in this exercise). Running in the Linux subsystem on Chromebook also works (which is how riban had developed much of the recent zynthian code changes)! | |
| − | + | [[File:VSC connected to zynthian.png|1200px|thumb|center]] | |
| − | + | There are several things to notice in this screen because it is very carefully designed to be informative and make best use of screen space. | |
| − | [ | + | The icons on the left are worth considering. | 
| + | Here's the [https://code.visualstudio.com/docs/getstarted/userinterface official docs...] | ||
| + | =====Explorer===== | ||
| + | Microsoft attempting to purloin a name. It's files and Directories and when configured as we intend to do here it's the files on the remote Pi4. What you edit here will change on the Pi we are examining.  | ||
| + | =====Search===== | ||
| + | It's very useful to be able to perform a word or phrase search right across the entire chunk of code you are working on. This where you do that. | ||
| + | =====Source Control===== | ||
| + | It would be temping to keep all your code efforts safely backed up somewhere and if it could give you a history of what you have done in the past that would be really useful. Welcome to ''git''. It does that for you. You store the code on github.com and VSC handles the transfer of code to and from this repository for you. It also allows you to upload your code for a Pull Request (the ultimate accolade!) consideration for entering in to the zynthian code base, because you are working on a your personal fork on github of the zynthian code, aren't you?  | ||
| + | =====Run & Debug===== | ||
| + | This is where we start the code examination and debugging sessions.  | ||
| + | We don't start the zynthian up with the debugger running as you might suspect, it's a little bit more contrived than that. But we will explain the current (2024-09-04) dance below... | ||
| − | + | The actual controls to run the debugger are in the little window floating up at the top right of the screen. The VSC documentation [https://code.visualstudio.com/Docs/editor/debugging is here...] | |
| − | + | =====Extensions===== | |
| − | |||
| − | + | Chunks of code written by many contributors including Microsoft themselves that allow extra functionality to be loaded into VSC. The debugger itself is one such component, as is the Remote Access mechanism and an extension to work intelligently with Python code. | |
| + | =====Remote Explorer===== | ||
| + | Access to the Remote Machine. This an extension that needs to be installed to allow remote debugging. | ||
| − | + | =====Testing===== | |
| + | Testing Module. I've not had a lot of success with this. | ||
| − | ====  | + | =====Accounts===== | 
| + | Microsofts domain control | ||
| + | =====Manage===== | ||
| + | Admin Tasks | ||
| − | + | ====What should I do before starting VSC for debugging?==== | |
| + | Check the VNC connection from your host machine's browser to the zynthian, by selecting VNC. | ||
| − | + | ====How do I set up VSC to use the debugger?==== | |
| + | The first requirement is the Remote Explorer mentioned above, use the extensions panel to search and install. Beyond this, the zynthian has specific setup elements that allow the system to run properly. These are outlined in a file called launch.json which VSC can generate if requested but can be manually created in a directory /zynthian/zynthian-ui/.vscode | ||
| − | + | Here is an example launc.json for debugging zynthian: | |
| − | + | { | |
| + |     // Use IntelliSense to learn about possible attributes. | ||
| + |     // Hover to view descriptions of existing attributes. | ||
| + |     // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 | ||
| + |     "version": "0.2.0", | ||
| + |     "configurations": [ | ||
| + |         { | ||
| + |             "name": "Zynthian", | ||
| + |             "type": "debugpy", | ||
| + |             "request": "launch", | ||
| + |             "program": "zynthian_main.py", | ||
| + |             "python": "/zynthian/venv/bin/python3", | ||
| + |             "console": "integratedTerminal", | ||
| + |             "justMyCode": true, | ||
| + |             "env": { | ||
| + |                 "DISPLAY": ":0" | ||
| + |             } | ||
| + |         }, | ||
| + |         { | ||
| + |             "name": "Zynthian Debug", | ||
| + |             "type": "debugpy", | ||
| + |             "request": "launch", | ||
| + |             "program": "zynthian_main.py", | ||
| + |             "python": "/zynthian/venv/bin/python3", | ||
| + |             "console": "integratedTerminal", | ||
| + |             "justMyCode": true, | ||
| + |             "subProcess": true, | ||
| + |             "env": { | ||
| + |                 "DISPLAY": ":0", | ||
| + |                 "ZYNTHIAN_LOG_LEVEL": "10" | ||
| + |             } | ||
| + |         } | ||
| + |     ] | ||
| + | } | ||
| − | |||
| − | + | This allows two different configurations to be used for debugging with different levels of log reporting. | |
| − | + | ====How do I stop the zynthian from within VSC?==== | |
| + | Once you have a remote terminal within VSC running you type | ||
| + | systemctl stop zynthian.   | ||
| − | + | This will stop the zynthian process, and your GUI Screen will disappear and the zynth wont make any sound. ( well it shouldn't). other processes like webconf will continue to work so we only need to restart the zynthian from within VSC to allow us access to VSC debugging facilities. | |
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | + | ====How do I re-allocate the screen?==== | |
| + | After typing systemctl stop zynthian  | ||
| − | + | type: | |
| + | X -r -s 0 | ||
| − | + | This sorts out the graphics subsystem to work in the debugging situation.   | |
| − | + | ====How do I actually start the VSC Debugger?==== | |
| + | Simply press the Debug Green Arrow  button | ||
| <br clear=all> | <br clear=all> | ||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| ==== How do I tell the world about my contribution...? ==== | ==== How do I tell the world about my contribution...? ==== | ||
| − | |||
| You put it on Github because that is where the zynthian software itself resides ... | You put it on Github because that is where the zynthian software itself resides ... | ||
| Line 174: | Line 198: | ||
| You need to submit a pull request. | You need to submit a pull request. | ||
| + | ==== How do I submit a pull request...? ==== | ||
| This is a concept within the github environment, and to do that you need to have an account at github yourself. So you need to dance your way throu their act of anointing ([http://www.github.com www.github.com]). | This is a concept within the github environment, and to do that you need to have an account at github yourself. So you need to dance your way throu their act of anointing ([http://www.github.com www.github.com]). | ||
| Line 190: | Line 215: | ||
| <br clear=all> | <br clear=all> | ||
| + | |||
| === It is 'suggested' one creates a task/issue? how do I do that ? === | === It is 'suggested' one creates a task/issue? how do I do that ? === | ||
| − |   " | + |   "It's a good habit to create a task/issue associated with more relevant changes" | 
| + | |||
| + | You will need a GitHub account to report issues. | ||
| + | |||
| + | |||
| + | So you will have to got to https://github.com/     and open an account there. | ||
| + | It's a Microsoft owned site so vsc helps to view the code, if you want to see the zynthian components at some stage. | ||
| + | |||
| + | You are redirected by the Report issue button to  | ||
| + | |||
| + | https://github.com/zynthian/zynthian-issue-tracking   | ||
| − | + | Which is the location of zynthian issue reporting. Please do it here, not on the individual code branches !! | |
| + | |||
| + | |||
| + | This is so that we know who reported it and can engage with them in its progress. | ||
| + | |||
| + | So to repeat, | ||
| + | |||
| + |  The best way to report a bug is to use the "Report Issue" button in the webconf. This creates a new issue in GitHub issue tracker, from a template with some data populated that is useful to the developers. | ||
| [[File:Zynthian-issue-tracker-page.png|600px|thumb|center]] | [[File:Zynthian-issue-tracker-page.png|600px|thumb|center]] | ||
| Line 207: | Line 250: | ||
| ==== So what have we gained here ? ==== | ==== So what have we gained here ? ==== | ||
| − | Well, the communal  | + | Well, the communal zynthian entity gains an area of interest. Even if the task, dies on the vine, and nothing more gets done, there is a possible development signposted. You also get a unique identity number that goes into the zynthian event queue. #941 in this case. This is an identifier you can add into a comment with Pull requests so the call for work and the doing of work can be related, tracked and pull requested, accepted tested and completed. | 
| − | + | This sort of structure within a project like zynthian is a very good thing. And if precisely scratching your name into a spaceship excites you, then take this as an early buzz. | |
| + | [[File:Zynthian Issues page.png|600px|center]] | ||
| <br clear=all> | <br clear=all> | ||
| − | + | An issue could be allocated to somebody else at some later time which means the code magically appears from someone you probably would never have met. Of course if you go ahead yourself and write it then so much the better. You move that one step closer to the "Had An Open Source PR accepted" | |
| + | |||
| + | The GitHub environment allows these relationships to be managed and the more structured this stuff is the less intimidating it becomes... | ||
| + | |||
| + | <br clear=all> | ||
| − | [https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/working-with-forks/fork-a-repo The  | + | ==== How Do I fork the zynthian GitHub repository? ==== | 
| − | [[File:Zynthian-ui fork page.png|600px|thumb|center|Zynthian page at  | + | [https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/working-with-forks/fork-a-repo The GitHub write up is here...] But in summary it's done from the appropriate zynthian page end of the process. | 
| + | [[File:Zynthian-ui fork page.png|600px|thumb|center|Zynthian page at GitHub]] | ||
| <br clear=all>   | <br clear=all>   | ||
| Line 226: | Line 275: | ||
| <br clear=all> | <br clear=all> | ||
| − | Here is the confirmation in my personal  | + | Here is the confirmation in my personal GitHub space. | 
| [[File:Zynthian-ui-page-forked to wyleu.png|600px|thumb|center]] | [[File:Zynthian-ui-page-forked to wyleu.png|600px|thumb|center]] | ||
| Line 232: | Line 281: | ||
| <br clear=all> | <br clear=all> | ||
| − | ==== Can I view files on my  | + | ==== Can I view files on my zynthian-ui GitHub repository? ==== | 
| [[File:Zynthian-ui-page-forked to wyleu.png|600px|thumb|center]] | [[File:Zynthian-ui-page-forked to wyleu.png|600px|thumb|center]] | ||
| − | In fact there is a very useful  | + | In fact there is a very useful GitHub element that allows you to view code on gihub within a browser based tool that is nothing more or less than VSC. | 
| − | It's no coincidence Microsoft bought  | + | It's no coincidence Microsoft bought GitHub... | 
| To operate this feature simple select the appropriate repository home page... https://github.com/wyleu/zynthian-ui/ | To operate this feature simple select the appropriate repository home page... https://github.com/wyleu/zynthian-ui/ | ||
| Line 249: | Line 298: | ||
| <br clear=all> | <br clear=all> | ||
| − | + | and here in VSC in GitHub showing a ctldev file ready to be examined.   | |
| − | and here in VSC in  | ||
| [[File:Github-vsc-editing-zynthian-ui-testing-ctrldev file.png|600px|thumb|center]] | [[File:Github-vsc-editing-zynthian-ui-testing-ctrldev file.png|600px|thumb|center]] | ||
| Line 279: | Line 327: | ||
| ==== How Do I access code on my zynthian from my host machine? ==== | ==== How Do I access code on my zynthian from my host machine? ==== | ||
| − | We need to use  Visual Studio Code(VSC) Remote Explorer to allow us to add on to the zynthian and install enough of an environment to allow VSC to seamlessly pass code between the environment and the desktop machine.Frankly sometimes it feels like magic, and for anyone who has read my previous attempts to do this sort of thing you spent a lot of time making sure the shaky old connections required behaved as machines are restarted and stopped and broken and goodness knows what  | + | We need to use  Visual Studio Code(VSC) Remote Explorer (remember to install the extension!) to allow us to add on to the zynthian and install enough of an environment to allow VSC to seamlessly pass code between the environment and the desktop machine. Frankly sometimes it feels like magic, and for anyone who has read my previous attempts to do this sort of thing you spent a lot of time making sure the shaky old connections required behaved as machines are restarted and stopped and broken and goodness knows what else. VSC really sorts this out nicely, and it's a blessing. Indeed it is so tenacious in how it manages connections that the following image is actual a bit of a manufactured effort, but worry no it does it all very well.   | 
| Line 436: | Line 484: | ||
| <br clear=all> | <br clear=all> | ||
| − | ==== Can I see a diagram of the various branches? ==== | + | ==== Can I see a diagram of the my various zynthian various branches? ==== | 
| Here is the git graph vsc plugin showing tramlines for this repository. You will need to install this plugin to get this.   | Here is the git graph vsc plugin showing tramlines for this repository. You will need to install this plugin to get this.   | ||
| Line 448: | Line 496: | ||
| From the github.com/zynthian/zynthian-ui/   you can select insights in the top menu and then Network in the left hand vertical menu. Github will chug away for a little while having a think then show you this, which now features the duo-piano-device branch waiting for me to submit the completion of the code with some more commits and then the pull request which, if successful will result in the branch being merged into testing, and at some later date in stable .  . .   | From the github.com/zynthian/zynthian-ui/   you can select insights in the top menu and then Network in the left hand vertical menu. Github will chug away for a little while having a think then show you this, which now features the duo-piano-device branch waiting for me to submit the completion of the code with some more commits and then the pull request which, if successful will result in the branch being merged into testing, and at some later date in stable .  . .   | ||
| [[File:Zynthian-ui-networks.png|600px|thumb|center]] | [[File:Zynthian-ui-networks.png|600px|thumb|center]] | ||
| + | <br clear=all> | ||
| + | =So what is LV2?= | ||
| + | ==[https://en.wikipedia.org/wiki/LV2 From wikipedia...]== | ||
| + | LV2 (LADSPA Version 2) is a set of royalty-free open standards[2] for music production plug-ins and matching host applications. It includes support for the synthesis and processing of digital audio and CV,[3] events such as MIDI and OSC, and provides a free alternative to audio plug-in standards such as Virtual Studio Technology (VST) and Audio Units (AU). | ||
| + | |||
| + | ==[https://lv2plug.in/ns/lv2core# And from the actual LV2 site...]== | ||
| + | |||
| + | LV2 is an interface for writing audio plugins in C or compatible languages, which can be dynamically loaded into many host applications. This core specification is simple and minimal, but is designed so that extensions can be defined to add more advanced features, making it possible to implement nearly any feature. | ||
| + | |||
| + | LV2 maintains a strong distinction between code and data. Plugin code is in a shared library, while data is in a companion data file written in Turtle. Code, data, and any other resources (such as waveforms) are shipped together in a bundle directory. The code contains only the executable portions of the plugin. All other data is provided in the data file(s). This makes plugin data flexible and extensible, and allows the host to do everything but run the plugin without loading or executing any code. Among other advantages, this makes hosts more robust (broken plugins can't crash a host during discovery) and allows generic tools written in any language to work with LV2 data. The LV2 specification itself is distributed in a similar way. | ||
| + | |||
| + | LV2 is an extensible framework, allowing a program to load a plugin to do some processing. Note that the terms used here are generic on purpose because LV2 allows any type of data to be exchanged between the host and the plugin. | ||
| + | |||
| + | == In Summary== | ||
| + | |||
| + | So there is a clear distinction between the definitions of the facilities which are held in carefully defined definition files and the actual software that accesses the facilities by reading the definition files. This way there is a controlled interaction betwenn the host and the plugin and each can be protected from the vagueries of the other. | ||
| + | |||
| + | It is a list of definitions that refer back to core definitions and these relationships are maintained in a format called RDF . . . | ||
| + | |||
| + | ==So what is RDF ?== | ||
| + | [https://en.wikipedia.org/wiki/Resource_Description_Framework Once more from wikipedia...] | ||
| + | |||
| + | The Resource Description Framework (RDF) is a method to describe and exchange graph data. It was originally designed as a data model for metadata by the World Wide Web Consortium (W3C). It provides a variety of syntax notations and formats, of which the most widely used is Turtle (Terse RDF Triple Language). | ||
| + | |||
| + | It describes objects and the relationships between them . . .  | ||
| + | |||
| + | RDF represents information using semantic triples, which comprise a subject, predicate, and object. Each item in the triple is expressed as a Web URI. Turtle provides a way to group three URIs to make a triple, and provides ways to abbreviate such information, for example by factoring out common portions of URIs. For example, information about Huckleberry Finn could be expressed as: | ||
| + | |||
| + |  <http://example.org/books/Huckleberry_Finn> | ||
| + |    <http://example.org/relation/author> | ||
| + |    <http://example.org/person/Mark_Twain> . | ||
| + | |||
| + | In the zynthian plug in world it's primarily about the definition, allocation and mapping of audio and midi ports and parameters in the zynthian world to parameters provided by the LV2 plugin we are exchanging data with. | ||
| + | |||
| + | But first you establish some prefixes so you aren't typing out URL all the time... | ||
| + | Here the classes that are defined by lv2 are specified ( follow the link)  | ||
| + | and similarly DOAP defines structure for a software project, and SPDX is the software package data exchange. Thus a set of definitions are described that can be understood by humans ( to a certain extent) and machines (VERY specifically)  | ||
| + | |||
| + | |||
| + |  @prefix lv2:  <http://lv2plug.in/ns/lv2core#>. | ||
| + |  @prefix doap: <http://usefulinc.com/ns/doap#>. | ||
| + |  @prefix spdx: <http://spdx.org/rdf/terms#>. | ||
| + | |||
| + | Then you use these prefixes to define the relationships we are concerned with... | ||
| + | |||
| + | The 'a' is an atom. More news as I get it. . .  | ||
| + | |||
| + |  <http://example.org/lv2/wikipediaexample/silence> | ||
| + |   a lv2:Plugin; | ||
| + |   lv2:binary <silence.so>; | ||
| + |   doap:name "Silence"; | ||
| + |   doap:license spdx:GPL-3.0-or-later; | ||
| + |   rdfs:comment "This is an example plugin that includes an example plugin description." | ||
| + | |||
| + |   lv2:port [ | ||
| + |     a lv2:AudioPort, lv2:OutputPort; | ||
| + |     lv2:index 0; | ||
| + |     lv2:symbol "output"; | ||
| + |     lv2:name "Output"; | ||
| + |   ]. | ||
| + | |||
| + | |||
| + | === How Do I start writing a lv2 component compatible with Zynthian? === | ||
| + | |||
| + | ====Which zynthian components do I need to clone in Github to fully develop a plug in?==== | ||
| + | |||
| + | Not sure. | ||
| + | |||
| + | ====What do I need in code terms in addition to the normal zynthian files?==== | ||
| + | |||
| + | Somewhere to develop and test the code as we have described above. | ||
| + | Now at this point we have to dip below the passive waters of Python where everything is slow, leisurely and well behaved. And embrace the proscribed world of C & C++ where code runs fast and life is cheap. | ||
| + | |||
| + | You will also need components that actually understand the concepts lv2 embodies. ( hopefully someone ( you know who you are)) will jump in and give us the history of this particular chunk of the mostly virtual universe we inhabit.  | ||
| + | |||
| + | I'm working off a Grok summary on building a 50Hz & 60Hz notch filter and will attempt to ruthlessly hone this to something vaguely useful. Alternatively, I might go bell ringing. | ||
| + | |||
| + | You will need some lv3 libraries... | ||
| + | |||
| + |  '''sudo apt install libasound2-dev lv2-dev''' | ||
| + | |||
| + | =====libasound2-dev===== | ||
| + | |||
| + | From the README.md file. | ||
| + | ''The alsa-lib is a library to interface with ALSA in the Linux kernel and virtual devices using a plugin system.'' | ||
| + | |||
| + | [https://github.com/pop-os/libasound2 Alsa Lib asound2 . . . ] | ||
| + | |||
| + | These are the alsa libraries, the essential glue to plug into the operating systems understanding of sound & MIDI on a Pi. Obviously this (at the moment) will only run on a Pi, so code elements at this level are aimed at that particular machine. That doesn't actually mean you can't write it on some other operating system, just be aware when you might be trying to put pears in containers designed for apples. . . | ||
| + | |||
| + | These libraries are the components that allow you to run the functionality described. But only that. If you wish to connect to them from your own programme then we need to install a little bit more, like descriptions of the hooks that those code pieces require and this is what the -dev on the end of the library name describes. Easy mistake to make it the early days, not installing the dev editions. | ||
| + | |||
| + | Why the slightly different names asound2 and alsa?  | ||
| + | Alsa has always been a bit impenetrable . | ||
| + | |||
| + | =====lv2-dev===== | ||
| + | |||
| + | ''LV2 is a plugin standard for audio systems. It defines an extensible C API for plugins, and a format for self-contained "bundle" directories that contain plugins, metadata, and other resources'' | ||
| + | |||
| + | LV2 components and the appropriate development files to muck around with lv2 files. | ||
| + | |||
| + | ====Anything else?==== | ||
| + | |||
| + | You are best to use a LV2 compatibleframework to construct the components to allow integration with the Linux system | ||
| + | |||
| + | There are: | ||
| + | # Juce | ||
| + | # DPF | ||
| + | |||
| + | ====Juce==== | ||
| + | |||
| + | https://github.com/rec/echomesh/blob/master/documentation/Building%20Juce%20applications%20on%20the%20Raspberry%20Pi.md  | ||
| + | |||
| + |  Juce is a popular and stable cross-platform open-source C++ development system that lets you write an application as a single codebase that works on Windows, OS/X and Linux. | ||
| + | |||
| + |  Juce has a huge number of components relating to pretty well anything you need - GUI, audio processing, networking, etc. - and is particularly popular amongst audio software developers. You can find a lot more     information and some snappy demos on the Juce site. | ||
| + | |||
| + |  It turns out that Juce supports the Raspberry Pi "out of the box" as long as you install a few libraries before you start. I created a sample application on the Mac, copied it to a Raspberry Pi and compiled it,  and it worked right the first time! | ||
| + | |||
| + | ====DPF==== | ||
| + |  '''git clone --recursive https://github.com/DISTRHO/DPF.git''' | ||
| + | |||
| + | This is a tool to assist plug in writing.... | ||
| + | |||
| + |  ''DPF is designed to make development of new plugins an easy and enjoyable task.'' | ||
| + |  ''It allows developers to create plugins with custom UIs using a simple C++ API.'' | ||
| + |  ''The framework facilitates exporting various different plugin formats from the same code-base.'' | ||
| + | |||
| + |   ''DPF can build for LADSPA, DSSI, LV2, VST2, VST3 and CLAP formats.'' | ||
| + |   ''A JACK/Standalone mode is also available, allowing you to quickly test plugins.'' | ||
| + | |||
| + | Then make a  directory. | ||
| + |  '''mkdir SimpleGainPlugin''' | ||
| + |  '''cd SimpleGainPlugin''' | ||
| + | |||
| + | Save the three files below (DistrhoPluginMain.cpp, DistrhoPluginInfo.h, Makefile) in this directory. | ||
| + | |||
| + | [[File:Lv2simple.png|800px|thumb|center]] | ||
| + | |||
| + | ====So which is better?==== | ||
| + | Grok provided a comparison. | ||
| + | ''For developing LV2 plugins on Zynthian, DPF is the better choice due to its native LV2 support, lightweight design, and alignment with Zynthian’s open-source, Linux-based ecosystem. It integrates seamlessly with Zynthian’s plugin system and performs well on its hardware. Use JUCE only if you need cross-platform compatibility (e.g., developing for VST3/AU alongside LV2) or prefer its polished development tools, but be prepared to optimize heavily for Zynthian’s constraints.'' | ||
| + | |||
| + | =LV2 Components= | ||
| + | |||
| + | ==A Notch Filter== | ||
| + | === The Definition Files=== | ||
| + | === The Code components === | ||
| − | <br clear=all> | + | =====DisthroPluginInfo.h===== | 
| + | |||
| + |  #ifndef DISTRHO_PLUGIN_INFO_H_INCLUDED | ||
| + |  #define DISTRHO_PLUGIN_INFO_H_INCLUDED | ||
| + | |||
| + |  #define DISTRHO_PLUGIN_BRAND "Wyleu" | ||
| + |  #define DISTRHO_PLUGIN_NAME "SimpleGain" | ||
| + |  #define DISTRHO_PLUGIN_URI "http://zynthian.org/plugins/lv2/simplegain" | ||
| + |  #define DISTRHO_PLUGIN_HAS_UI 0 | ||
| + |  #define DISTRHO_PLUGIN_IS_RT_SAFE 1 | ||
| + |  #define DISTRHO_PLUGIN_NUM_INPUTS 2 | ||
| + |  #define DISTRHO_PLUGIN_NUM_OUTPUTS 2 | ||
| + |  #define DISTRHO_PLUGIN_WANT_LATENCY 0 | ||
| + |  #define DISTRHO_PLUGIN_WANT_PROGRAMS 0 | ||
| + |  #define DISTRHO_PLUGIN_WANT_STATE 0 | ||
| + |  #define DISTRHO_PLUGIN_WANT_TIMEPOS 0 | ||
| + |  #define DISTRHO_PLUGIN_IS_SYNTH 0 | ||
| + | |||
| + |  #endif | ||
| + | |||
| + | =====Partial Explanation of DistrhoPluginInfo.h===== | ||
| + | |||
| + | This defines the plugin’s metadata. | ||
| + | |||
| + | You get to define some information about you and what you are trying to do. | ||
| + | You make decisions like am I going to use a GUI? | ||
| + | How many inputs and outputs it will have | ||
| + | and the expectations as how it will interact with the environment. | ||
| + | |||
| + | All wrapped in a test which is then settrue as part of programme flow. . . Very neat. | ||
| + | |||
| + |  #ifndef DISTRHO_PLUGIN_INFO_H_INCLUDED | ||
| + |  #define DISTRHO_PLUGIN_INFO_H_INCLUDED | ||
| + |  ... | ||
| + |  #endif | ||
| + | |||
| + | The Python programmer in me would like a little indentation but such is life... | ||
| + | |||
| + | =====DisthroPluginMain.cpp===== | ||
| + | |||
| + |  #include "DistrhoPlugin.hpp" | ||
| + | |||
| + |  START_NAMESPACE_DPF | ||
| + | |||
| + |  class SimpleGainPlugin : public Plugin { | ||
| + |  public: | ||
| + |     SimpleGainPlugin() : Plugin(1, 0, 0) { // 1 parameter, 0 programs, 0 states | ||
| + |         gain = 1.0f; // Default gain (no change) | ||
| + |     } | ||
| + | |||
| + |  protected: | ||
| + |     const char* getLabel() const override { return "SimpleGain"; } | ||
| + |     const char* getDescription() const override { return "A simple gain plugin for Zynthian"; } | ||
| + |     const char* getMaker() const override { return "YourName"; } | ||
| + |     const char* getLicense() const override { return "MIT"; } | ||
| + |     uint32_t getVersion() const override { return d_version(1, 0, 0); } | ||
| + |     int64_t getUniqueId() const override { return d_cconst('S', 'G', 'a', 'n'); } | ||
| + | |||
| + |     void initParameter(uint32_t index, Parameter& parameter) override { | ||
| + |         if (index != 0) return; | ||
| + |         parameter.hints = kParameterIsAutomable; | ||
| + |         parameter.name = "Gain"; | ||
| + |         parameter.symbol = "gain"; | ||
| + |         parameter.unit = "x"; | ||
| + |         parameter.ranges.def = 1.0f; | ||
| + |         parameter.ranges.min = 0.0f; | ||
| + |         parameter.ranges.max = 2.0f; | ||
| + |     } | ||
| + | |||
| + |     float getParameterValue(uint32_t index) const override { | ||
| + |         if (index != 0) return 0.0f; | ||
| + |         return gain; | ||
| + |     } | ||
| + | |||
| + |     void setParameterValue(uint32_t index, float value) override { | ||
| + |         if (index != 0) return; | ||
| + |         gain = value; | ||
| + |     } | ||
| + | |||
| + |     void run(const float** inputs, float** outputs, uint32_t frames) override { | ||
| + |         const float* inL = inputs[0]; | ||
| + |         const float* inR = inputs[1]; | ||
| + |         float* outL = outputs[0]; | ||
| + |         float* outR = outputs[1]; | ||
| + | |||
| + |         for (uint32_t i = 0; i < frames; ++i) { | ||
| + |             outL[i] = inL[i] * gain; | ||
| + |             outR[i] = inR[i] * gain; | ||
| + |         } | ||
| + |     } | ||
| + | |||
| + |  private: | ||
| + |     float gain; | ||
| + |     DISTRHO_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(SimpleGainPlugin) | ||
| + |  }; | ||
| + | |||
| + |  Plugin* createPlugin() { | ||
| + |     return new SimpleGainPlugin(); | ||
| + |  } | ||
| + | |||
| + |  END_NAMESPACE_DPF | ||
| + | |||
| + | =====Explanation of DistrhoPluginMain.cpp===== | ||
| + | |||
| + | This is the main plugin implementation. | ||
| + | |||
| + | It includes the header files  | ||
| + |  #include "DistrhoPlugin.hpp" | ||
| + | |||
| + | we define a class: | ||
| + |  class SimpleGainPlugin : public Plugin { | ||
| + |  public: | ||
| + |     SimpleGainPlugin() : Plugin(1, 0, 0) { // 1 parameter, 0 programs, 0 states | ||
| + |         gain = 1.0f; // Default gain (no change) | ||
| + |     } | ||
| + | |||
| + | And within that class we set the gain to 1.0f      | ||
| + | |||
| + | This means a float (A floating point number 1.0345 sorts of things not 1,2,3,4,5). Basically what it gets in it puts straight out in this case and if we substituted 1.0345f for the 1.0f the signal would get a little bit louder. | ||
| + | A digital Patch lead! | ||
| + | |||
| + | This class will be returned as an object of type Plugin. But quite how and where Plugin is defined remains to be found.... | ||
| + | |||
| + | |||
| + | At the bottom of the code we instantiate (generate, create, make alive...) the object and return it to the calling function.  | ||
| + | |||
| + |  Plugin* createPlugin() { | ||
| + |     return new SimpleGainPlugin(); | ||
| + |  } | ||
| + | |||
| + | So this is a little factory that makes a Plugin object that can interact with the rest of the alsa audio framework. | ||
| + | |||
| + | So what's the rest of it... | ||
| + | |||
| + | Two sections  | ||
| + | # '''protected:''' | ||
| + | # '''private:''' | ||
| + | |||
| + | |||
| + | ======Protected:====== | ||
| + | Members declared as protected are accessible within the same class and in derived classes  | ||
| + | Some functions are declared  | ||
| + | |||
| + | Firstly those that return character strings of text (char*)  | ||
| + |  getLabel() const overide { return "SimpleGain"; } | ||
| + |  getDescription() | ||
| + |  getMaker() | ||
| + |  getLicense() | ||
| + | |||
| + | then a function that returns an unsigned 32 bit integer 0 -> 4 Gig | ||
| + | |||
| + | uint32_t getVersion() const override { return d_version(1, 0, 0); } | ||
| + | |||
| + | and an even more massive number 64 bit integer ( don't ask)  | ||
| + | |||
| + |     int64_t getUniqueId() const override { return d_cconst('S', 'G', 'a', 'n'); } | ||
| + | |||
| + | Next we have a function that initializes some Parameters and define a range of values for that handy gain number we'd like to alter.  | ||
| + | |||
| + |     void initParameter(uint32_t index, Parameter& parameter) override { | ||
| + |         if (index != 0) return; | ||
| + |         parameter.hints = kParameterIsAutomable; | ||
| + |         parameter.name = "Gain"; | ||
| + |         parameter.symbol = "gain"; | ||
| + |         parameter.unit = "x"; | ||
| + |         parameter.ranges.def = 1.0f; | ||
| + |         parameter.ranges.min = 0.0f; | ||
| + |         parameter.ranges.max = 2.0f; | ||
| + | |||
| + | This applies extra values to the external implementation of this parameter so the two environments can pass data effectively.  | ||
| + | |||
| + | index is required to have a value 0 to set these values to what are presumably defaults.  | ||
| + | |||
| + | Next we have a Getter & a Setter  | ||
| + | |||
| + |     float getParameterValue(uint32_t index) const override { | ||
| + |         if (index != 0) return 0.0f; | ||
| + |         return gain; | ||
| + |     } | ||
| + | |||
| + |     void setParameterValue(uint32_t index, float value) override { | ||
| + |         if (index != 0) return; | ||
| + |         gain = value; | ||
| + |     } | ||
| + | |||
| + | Again requiring a 0 index to operate. | ||
| + | |||
| + | Lastly in the protected section | ||
| + | |||
| + |     void run(const float** inputs, float** outputs, uint32_t frames) override { | ||
| + |         const float* inL = inputs[0]; | ||
| + |         const float* inR = inputs[1]; | ||
| + |         float* outL = outputs[0]; | ||
| + |         float* outR = outputs[1]; | ||
| + | |||
| + |         for (uint32_t i = 0; i < frames; ++i) { | ||
| + |             outL[i] = inL[i] * gain; | ||
| + |             outR[i] = inR[i] * gain; | ||
| + |         } | ||
| + |     } | ||
| + | |||
| + | The run function of this class, which will be called by the underlying system to process audio samples appearing at the input and generating appropriate output at the correct time. | ||
| + | |||
| + | i.e. doing the clever stuff between these two... | ||
| + | |||
| + |  void run(const float** inputs, float** outputs, uint32_t frames) override | ||
| + | It's defined as a void so it does not return a value, interacting via the arrays passed to it.  | ||
| + | |||
| + | The inputs to the tun function are a  | ||
| + |  an array of inputs  | ||
| + |  an array of outputs  | ||
| + |  A frame count. The number of frames in the array.  | ||
| + | |||
| + | Don't know what override does. | ||
| + | |||
| + | The arrays are allocated to variables  ( notice the assumption here that we only handle the first two arrays handed to us. How it deals with other arrays is a point of discussion.) | ||
| + | |||
| + |         const float* inL = inputs[0]; | ||
| + |         const float* inR = inputs[1]; | ||
| + |         float* outL = outputs[0]; | ||
| + |         float* outR = outputs[1]; | ||
| + | |||
| + | So for each frame we loop a counter i with that numbered frame of audio  | ||
| + | |||
| + |         for (uint32_t i = 0; i < frames; ++i) { | ||
| + | |||
| + | and then do stuff  | ||
| + |             outL[i] = inL[i] * gain; | ||
| + |             outR[i] = inR[i] * gain; | ||
| + |         } | ||
| + | |||
| + | At the end of the process the out arrays have been filled with the adjusted values from the input arrays. . . . | ||
| + | |||
| + | And these can be handled by the framework infrastructure to throw it at the next device down the chain. | ||
| + | |||
| + | <br clear all> | ||
| + | |||
| + | ======Private:====== | ||
| + | Members declared as private are accessible only within the same class. | ||
| + | They cannot be accessed by derived classes, objects of the class, or external code. | ||
| + | Used to encapsulate data and hide implementation details. | ||
| + | |||
| + | ======So where do these undefined chunks of code come from?====== | ||
| + | |||
| + | We haven't had any nice import functions to do this for us so how can these references to external functions be addressed? | ||
| + | |||
| + | We need to 'link' the code we have written with the development hooks of the libraries we imported earlier.  | ||
| + | |||
| + | We need something to 'make' our programme. Enter the makefile . . .  | ||
| + | |||
| + | =====makefile===== | ||
| + | |||
| + |  #!/usr/bin/make -f | ||
| + | |||
| + |  NAME=SimpleGain | ||
| + |  DPF_PATH=../../DPF | ||
| + | |||
| + |  include $(DPF_PATH)/Makefile.plugins.mk | ||
| + | |||
| + |  TARGETS=lv2 | ||
| + | |||
| + |  all: $(TARGETS) | ||
| + | |||
| + |  include $(DPF_PATH)/Makefile.base.mk | ||
| + | |||
| + | =====makefile what does it do ?===== | ||
| + | |||
| + | |||
| + | This attempts to builds the plugin as an LV2 bundle. | ||
| + | |||
| + | The two files mentioned here are frankly enormous. I assume these are built by code not hand but have absolutely no desire to venture into them with any exploratory desire.  | ||
| + | |||
| + | I wonder if 'make' has any way of reporting the world it believes it's constructing ahead of time? | ||
| + | |||
| + |  wyleu@raspberrypi:~/Code/lv2/DPF/SimpleGainPlugin $ make -n | ||
| + |  mkdir -p /bin/SimpleGain.lv2 | ||
| + |  echo "Creating LV2 plugin for SimpleGain" | ||
| + |  g++ /build/SimpleGain/DistrhoPluginMain_LV2.cpp.o -Wall -Wextra -pipe -MD -MP -fno-gnu- unique -fPIC -DPIC -DNDEBUG -O3 -ffast-math -fdata-sections -ffunction-sections - fvisibility=hidden -DHAVE_ALSA -DHAVE_JACK -DHAVE_RTAUDIO  -DHAVE_X11  -DHAVE_XEXT - DHAVE_XSYNC -DHAVE_OPENGL -std=gnu++11 -fvisibility-inlines-hidden -I. -I../../DPF/distrho - I../../DPF/dgl -I../../DPF/dgl/src/pugl-upstream/include -fdata-sections -ffunction-sections  -Wl,-O1,--as-needed,--gc-sections -Wl,--strip-all  -Wl,--no-undefined -ldl     -shared -Wl,-- version-script=../../DPF/utils/symbols/lv2.version -o /bin/SimpleGain.lv2/SimpleGain.so | ||
| + | |||
| + | Not sure where the line breaks and if there are any in that mouthful. | ||
| + | |||
| + | make -p generates enormous amounts of information.. | ||
| + | |||
| + | make -d provides what we already know.... | ||
| + | |||
| + |  Live child 0x555658e367a0 (/bin/SimpleGain.lv2/SimpleGain.so) PID 51392  | ||
| + |  Reaping winning child 0x555658e367a0 PID 51392  | ||
| + |  Live child 0x555658e367a0 (/bin/SimpleGain.lv2/SimpleGain.so) PID 51393  | ||
| + |  Creating LV2 plugin for SimpleGain | ||
| + |  Reaping winning child 0x555658e367a0 PID 51393  | ||
| + |  Live child 0x555658e367a0 (/bin/SimpleGain.lv2/SimpleGain.so) PID 51394  | ||
| + |  /usr/bin/ld: cannot open output file /bin/SimpleGain.lv2/SimpleGain.so: Permission denied | ||
| + | |||
| + | ==Auto Leveller== | ||
| + | |||
| + | A Place holder for information on the construction of this component and a casual implementation of some kind of document structure. | ||
| + | |||
| + | [[File:Leveller.png|thumb|Auto Leveller]] | ||
| + | |||
| + | [[File:AutoLeveller GUI.png|thumb|AutoLeveller GUI]] | ||
| + | |||
| + | ===What is it?=== | ||
| + | You can think of it as your angry sound engineer in a live situation, both during soundcheck and during the gig. Initially it will turn your output volume down to a reasonable level. It will also do that if anything unforseen happens in the chain (you add another plugin which features an integer gain multiplier and is set to gain=20). When set the right values (try the preset “Tuned”) it should level any source to a loudness level somewhere into yellow meter territory. | ||
| + | |||
| + | It also features two programs: | ||
| + | |||
| + | Init: All set to zero. It will not perform anything unless your level is above 0 dBfs, which is always bad. | ||
| + | Tuned: Set the parameters to reasonable values: Pre Gain will boost the signal by +6 dB to compensate quiet engines like soundfonts, peak threshold is set to -4 dBfs and RMS to -18 dBFS to keep more or less transient heavy signals somewhere in yellow area of your mixer. | ||
| + | |||
| + | ===Parameters=== | ||
| + | ====Pre Gain (dBfs)==== | ||
| + | Adjust the pre amplification in case you deal with quieter sources (like soundfonts) you want to boost before getting into the automatic levelling. | ||
| + | |||
| + | ====Theshold Peak (dBfs)==== | ||
| + | Set your maximum peak levels you want to have. Setting the threshold will reset your former gain reduction so it can listen from the start again. When “Listen” is on, it will only turn the volume downwards. Recommended values are between 0 and -4 dBfs. | ||
| + | |||
| + | ====Theshold RMS (dBfs)==== | ||
| + | Set your maximum RMS levels you want to have. Setting the threshold will reset your former gain reduction so it can listen from the start again. When “Listen” is on, it will only turn the volume downwards. The detector is estimating RMS values of a 300ms window by a using a lowpass filter. Recommended values are between -14 and -18 dBfs. | ||
| + | |||
| + | ====Listen (bool)==== | ||
| + | When engaged (by default) it will perform detection and level matching. When disengaged it will keep the current gain, but will not further listen. If engaged again, the gain reduction will be reset. | ||
| + | |||
| + | =LV2 custom control groups (ttl)= | ||
| + | If not edited further, LV2 parameters will be presented in a manner being sorted by their parameter ID and grouped in groups of four without helpful group names. We can edit a specific file to get some more helpful named and ordered parameter groups. It is highly recommended to study other custom control ttl files before edit your own one. | ||
| + | |||
| + | ==Location== | ||
| + | We start by searching for the original ttl file in the zynthian filesystem. Typically we'll find them in /usr/lib/lv2/[PLUGIN].lv2 or whereever this plugin lives. We copy the whole [PLUGIN].lv2 directory over to /zynthian/zynthian-data/lv2-custom (check the [https://github.com/zynthian/zynthian-data/tree/oram/lv2-custom other custom ttls] for reference). When we look inside, we'll find several *.so and *.ttl files. The file we are searching for is not the manifest.ttl but the other one. We can delete any other file from that directory. | ||
| + | |||
| + | On any zynthian update database regeneration zynthian will replace the original ttl file with the ones in /zynthian/zynthian-data/lv2-custom. | ||
| + | |||
| + | ==Editing the ttl file== | ||
| + | ===Namespace prefixes=== | ||
| + | |||
| + | Most of the time we have to add some lines to the header, if not already present: | ||
| + | |||
| + |  @prefix rdf:  <http://www.w3.org/1999/02/22-rdf-syntax-ns#> . | ||
| + |  @prefix pg:   <http://lv2plug.in/ns/ext/port-groups#> . | ||
| + |  @prefix plug: <COPY URN FROM PLUGIN BLOCK> . | ||
| + | |||
| + | These [https://lv2plug.in/book/ namespace prefixes] act like unique identifiers for specific tasks. The URN to copy we'll find on top of the plugin block, which is the section that holds all the control ports. Just look into another custom ttl and compare the top of a plugin block with the URN set under the prefix "plug:". | ||
| + | |||
| + | ===Control groups=== | ||
| + | |||
| + | Add control groups like that outside the plugin block, for example right at the end of the file. Groups with higher displayPriority values will be displayed on top of your parameter group view. GROUP1 and the symbol "group1" can be named yourself (only letters, numbers or _", name "Group1 Display Name" you can also edited and will show on your UI. | ||
| + | |||
| + |  plug:GROUP1 | ||
| + |     a pg:ControlGroup ; | ||
| + |     lv2:name "Group1 Display Name" ; | ||
| + |     lv2:symbol "group1" ; | ||
| + |     lv2:displayPriority 3 . | ||
| + | |||
| + |  plug:GROUP2 | ||
| + |     a pg:ControlGroup ; | ||
| + |     lv2:name "Group2 Display Name" ; | ||
| + |     lv2:symbol "group2" ; | ||
| + |     lv2:displayPriority 2 . | ||
| + | |||
| + |  ... | ||
| + | |||
| + | ===Parameter assignment=== | ||
| + | |||
| + | Add following lines to every control port representing a parameter (contains "a lv2:ControlPort", so not the audio input and output ports), while display priority goes from higher to lower numbers as well: | ||
| + | |||
| + |  pg:group plug:GROUP1 ; | ||
| + |  lv2:displayPriority 1 ; | ||
| + | |||
| + | ===Scale Points=== | ||
| + | |||
| + | This is especially helpful for integer or boolean like enumeration parameters and other parameters which should have stepped controls. Here we have an example for a boolean parameter, which should only be in state "On" or "Off": | ||
| + | |||
| + |  lv2:scalePoint [ | ||
| + |     rdf:value 0.0 ; | ||
| + |     rdfs:label "OFF" ; | ||
| + |     rdfs:comment "OFF" ; | ||
| + |  ], [ | ||
| + |     rdf:value 1.0 ; | ||
| + |     rdfs:label "ON" ; | ||
| + |     rdfs:comment "ON" ; | ||
| + |  ]; | ||
| + | |||
| + | It has to be like that because typically (oftentimes, but not necessarily) parameters are represented as floating point values (0.000f to 1.000f), even if they're supposed to be boolean or integer, even if LV2 would support these kind of data. Other parameters could be of "integer logic". Let's assume the parameter "Waveform" would have the states "Saw, Sine, Square, Triange", we may assume that every floating point value from 0.0f to <0.25f represents "Saw" and so on. Then we might try this (while value and label are mandatory, comment is not): | ||
| + | |||
| + |  lv2:scalePoint [ | ||
| + |     rdf:value 0.0 ; | ||
| + |     rdfs:label "Saw" ; | ||
| + |     rdfs:comment "Saw" ; | ||
| + |  ], [ | ||
| + |     rdf:value 0.25 ; | ||
| + |     rdfs:label "Sine" ; | ||
| + |     rdfs:comment "Sine" ; | ||
| + |  ], [ | ||
| + |     rdf:value 0.5 ; | ||
| + |     rdfs:label "Square" ; | ||
| + |     rdfs:comment "Square" ; | ||
| + |  ], [ | ||
| + |     rdf:value 0.75 ; | ||
| + |     rdfs:label "Triangle" ; | ||
| + |     rdfs:comment "Triangle" ; | ||
| + |  ]; | ||
| + | |||
| + | ===Disable parameter from being shown on UI=== | ||
| + | |||
| + | Sometimes we have parameters which have no function and must not be displayed. You add the following line to the parameter definition: | ||
| + | |||
| + |  lv2:portProperty pp:notOnGUI ; | ||
| + | |||
| + | ===Summary parameter definition=== | ||
| + | So a single parameter should for example look like this: | ||
| + | |||
| + |     [ | ||
| + |         a lv2:InputPort, lv2:ControlPort ; | ||
| + |         lv2:index 8 ; | ||
| + |         lv2:symbol "delaytimesync" ; | ||
| + |         lv2:name "Host Sync" ; | ||
| + |         lv2:default 0.473684 ; | ||
| + |         lv2:minimum 0.0 ; | ||
| + |         lv2:maximum 1.0 ; | ||
| + |         '''pg:group plug:DELAY ;''' | ||
| + |         lv2:displayPriority 2 ; | ||
| + |         lv2:scalePoint [ | ||
| + |             rdf:value 0.0 ; | ||
| + |             rdfs:label "Free" ; | ||
| + |             rdfs:comment "Free" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.052631 ; | ||
| + |             rdfs:label "1/16" ; | ||
| + |             rdfs:comment "1/16" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.105263 ; | ||
| + |             rdfs:label "1/8" ; | ||
| + |             rdfs:comment "1/8" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.157895 ; | ||
| + |             rdfs:label "1/4" ; | ||
| + |             rdfs:comment "1/4" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.211053 ; | ||
| + |             rdfs:label "1/2" ; | ||
| + |             rdfs:comment "1/2" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.263158 ; | ||
| + |             rdfs:label "1/1" ; | ||
| + |             rdfs:comment "1/1" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.315789 ; | ||
| + |             rdfs:label "2/1" ; | ||
| + |             rdfs:comment "2/1" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.368421 ; | ||
| + |             rdfs:label "1/16." ; | ||
| + |             rdfs:comment "1/16." ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.421052 ; | ||
| + |             rdfs:label "1/8." ; | ||
| + |             rdfs:comment "1/8." ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.473684 ; | ||
| + |             rdfs:label "1/4." ; | ||
| + |             rdfs:comment "1/4." ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.526316 ; | ||
| + |             rdfs:label "1/2." ; | ||
| + |             rdfs:comment "1/2." ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.578947 ; | ||
| + |             rdfs:label "1/1." ; | ||
| + |             rdfs:comment "1/1." ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.631579 ; | ||
| + |             rdfs:label "2/1." ; | ||
| + |             rdfs:comment "2/1." ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.684211 ; | ||
| + |             rdfs:label "1/16T" ; | ||
| + |             rdfs:comment "1/16T" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.736842 ; | ||
| + |             rdfs:label "1/8T" ; | ||
| + |             rdfs:comment "1/8T" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.789474 ; | ||
| + |             rdfs:label "1/4T" ; | ||
| + |             rdfs:comment "1/4T" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.842105 ; | ||
| + |             rdfs:label "1/2T" ; | ||
| + |             rdfs:comment "1/2T" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.894736 ; | ||
| + |             rdfs:label "1/1T" ; | ||
| + |             rdfs:comment "1/1T" ; | ||
| + |         ], [ | ||
| + |             rdf:value 0.947368 ; | ||
| + |             rdfs:label "2/1T" ; | ||
| + |             rdfs:comment "2/1T" ; | ||
| + |         ], [ | ||
| + |             rdf:value 1.0 ; | ||
| + |             rdfs:label "-" ; | ||
| + |             rdfs:comment "-" ; | ||
| + |         ]; | ||
| + |     ] , | ||
| + | |||
| + | The bold line is the one we started to insert things. You typically should only edit the "name" value, which is the display label and the newly added values. In this case we added the control to the previously added group named "DELAY", added the displayPriority, so that it will be displayed within the group above all parameters with value <2 and below paramters with value >2. Then we added stepped controls for delay times defined by subdivisions of the host tempo, where the values represent 1/19, 2/19, 3/19... Continuous parameters typically do not need scale points at all, rather they would become stepped parameters as well if we'd add them. | ||
| + | |||
| + | ==Recommended workflow== | ||
| + | Instead of just copying the ttl over to the respective directory and edit it for your own use you can make things easier for the devs to implement your custom feature and share it with everybody this way: | ||
| + | |||
| + | * Fork the zynthian/zynthian-data repositories vangelis branch | ||
| + | * Checkout to that branch in your zynthian | ||
| + | * Develop using your fork | ||
| + | * Make a pull request | ||
Latest revision as of 16:21, 5 September 2025
1 So what is an IDE ?
We could operate all of this using the command line and git commands, but unless you really want to, don't. Use an IDE.
An IDE (Integrated Development Environment) is a collection of programs that work together to ease the difficulties of acquiring, writing, modifying, testing and debugging code that you write along with the possibility of many, many other functions you probably never considered. At the moment there is one that seems to be really sweeping the board with this (yes, I know, emacs users, but I worked on pycharm and intellij and I know which I prefer out of the three).
It's Microsoft's Visual Studio code, https://code.visualstudio.com/ know to friends and enemies alike as VSC (or VSCode).
1.1 So what is a host machine?
Visual Studio Code is a considerable piece of software, Microsoft know an awful lot about writing code and they have placed a suberb, well maintained IDE into the hands of absolutely anybody with a desire to write code. (If you date back far enough to remember "a computer on every desk running Microsoft software" you might generate a rye smile at this point, but the software world is a very different place now, and Microsoft are perfectly willing to produce their IDE for every conceivable platform and operating system they come across and make it work. Anyone who plays around with Arduinos and such will find VSC's platformIO plugin is a considerable improvement on the Arduino's own IDE, and plugins like this are the secret of VSC.) The Python environment is a plugin, tools to view git repositories are a plugin, remote access components are a plugin. I haven't looked but I don't doubt there is some MIDI related plugin out there... So we have a considerable chunk of code to run, which we are going to use to write a considerable chunk of code on a Raspberry Pi... Can you run VSC on a Raspberry Pi? Well yes you can, indeed I have run my VSC environment on a Pi5. But I wouldn't run zynthian and the VSC core UI code on a zynthian. There is a cleverer way. You use the remote connection plugin, to access a separate zynthian doing musical stuff. The VSC remote component installs itself onto the Pi, once you present it with the correct password (Did I mention about getting your passwords sorted out...?), and it handles many, many areas of this process that you can lose yourself into setting up yourself. I have written this process up for other environments and it was..... Involved. OK VSC advert over.
So our development environment consists of two machines, one is the zynthian itself, and the other is another computer running the VSC GI code.
This other machine is the host computer and it can be any number of bizarre, powerful, stylish, home built, clapped out, PC's and goodness knows what else, but as long as it runs VSC and has a network connection you are good to go. As I say my host machine is now a Pi5 running VSC very effectively, but your's could be MAC or a PC or goodness knows what else. This is why using a common IDE provides dividends because we don't need to explain all the different access mechanisms for all these machines. Look up how you install VSC on your host machine and do that. It's pretty easy and if anyone wants to write up specific versions then great! Just say so on the forum.
1.2 How do I install VSC on my host machine?
Go to VSC Download page and pick the appropriate version. For your particular Pi OS. If you are running 64 bit software pick .deb arm64, if you are on a 32 bit Pi OS then pick .deb arm32.If you are on MAC or PC then download accordingly.
This will download the selected version and you install as appropriate to your operating system. On my Pi5 I open the download folder
and right click on the downloaded file and select Package Install.
It will and you will have a new icon appear in your desktop menu in the Programming section.
Simply pressing on the icon will run Visual Studio Code, but for the moment there is a bit more to consider...
1.3 Where are the zynthian source files located on a zynthian?
The zynthian software exists as a set of directories (or folders if you prefer - but get used to us using the more accurate "directories") within a directory on the zynthian called... /zynthian.
The GUI software running on the zynth that operates the touchscreen are located within the zynthian-ui directory. and there is a file called zynthian.sh. This is a shell script which isn't written in python but in a bash. It is a collection of commands one runs on the terminal we mentioned earlier an allows various things to be configured, before the actual python script that does the work, which is called zynthian_main.py, is run.
One thing the shell script sets up is the debugging level which decides what level of message the zynthian generates when it encounters a logging command in the python script...
1.4 What happens if zynthian crashes and how do I stop it restarting on errors ?
When a zynthian starts up, most of the early functions and what gets run gets handled by a Linux tool called systemd. This organises what order things happen in and nursemaids programs, setting up procedures for what to do when programs crash or misbehave. For instance there is no point in starting up the network devices if there is no network, and this is the sort of thing systemd does on your behalf. In the zynthian case it starts up the zynthian program using the scripts we discussed in /zynthian/zynthian-ui and handles a zynthian by restarting the UI if the software crashes for some reason. This is what you want if you are performing but is decidedly irritating if you are trying to write and test code. There are magic incantations to control zynthian via systemd
systemctl stop zynthian systemctl start zynthian systemtl status zynthian
So to stop a running zynthian you issue the first of these and this will tell systemd to ignore the restart so you can take over the running of it.
1.5 How do I start a X11 windows server so I can run the code in a debugger?
Once you have stopped the zynthian process, the GUI layer (X11) required to run zynthian is also stopped so you can start zynthian within the VSC debugger. To allow this to happen you have to start the X11 system to allow you to have something to connect to.
To do this in the window you typed "systemctl stop zynthian", type
X -r -s 0
You can then start the debugger in VSC.
2 How do I Debug a Zynthian?
A careful balancing act is required here because there are a lot of entities attempting to work harmoniously here and a Pi can be a small world.
- The zynthian running on Pi, preferably as meaty as one can get. A Pi3 for instance will run out of resources with more recent engines.
- A Separate Computer Host from where the operation is all controlled.
- Some extra code loaded on the remote machine that communicates with the controlling programme on the host machine.
- A VNC server (A copy of the GUI image that runs in a browser) running on the zynthian to allow the screen to be examined during debugging.
- Some alterations to the launch.json file in .vscode to set things up nicely.
Visual Studio Code is an excellent environment for this and it fairly effortlessly sets up most of the environment for you. It will need to be running on your host machine, be it Windows, Mac or Linux which could be running on a Pi. VSC runs on them all. Indeed this is being typed on a Raspberry Pi desktop machine which has VSC open and already connected to Pi4 (zynthian-rack6.local in this exercise). Running in the Linux subsystem on Chromebook also works (which is how riban had developed much of the recent zynthian code changes)!
There are several things to notice in this screen because it is very carefully designed to be informative and make best use of screen space.
The icons on the left are worth considering. Here's the official docs...
2.1 Explorer
Microsoft attempting to purloin a name. It's files and Directories and when configured as we intend to do here it's the files on the remote Pi4. What you edit here will change on the Pi we are examining.
2.2 Search
It's very useful to be able to perform a word or phrase search right across the entire chunk of code you are working on. This where you do that.
2.3 Source Control
It would be temping to keep all your code efforts safely backed up somewhere and if it could give you a history of what you have done in the past that would be really useful. Welcome to git. It does that for you. You store the code on github.com and VSC handles the transfer of code to and from this repository for you. It also allows you to upload your code for a Pull Request (the ultimate accolade!) consideration for entering in to the zynthian code base, because you are working on a your personal fork on github of the zynthian code, aren't you?
2.4 Run & Debug
This is where we start the code examination and debugging sessions. We don't start the zynthian up with the debugger running as you might suspect, it's a little bit more contrived than that. But we will explain the current (2024-09-04) dance below...
The actual controls to run the debugger are in the little window floating up at the top right of the screen. The VSC documentation is here...
2.5 Extensions
Chunks of code written by many contributors including Microsoft themselves that allow extra functionality to be loaded into VSC. The debugger itself is one such component, as is the Remote Access mechanism and an extension to work intelligently with Python code.
2.6 Remote Explorer
Access to the Remote Machine. This an extension that needs to be installed to allow remote debugging.
2.7 Testing
Testing Module. I've not had a lot of success with this.
2.8 Accounts
Microsofts domain control
2.9 Manage
Admin Tasks
2.10 What should I do before starting VSC for debugging?
Check the VNC connection from your host machine's browser to the zynthian, by selecting VNC.
2.11 How do I set up VSC to use the debugger?
The first requirement is the Remote Explorer mentioned above, use the extensions panel to search and install. Beyond this, the zynthian has specific setup elements that allow the system to run properly. These are outlined in a file called launch.json which VSC can generate if requested but can be manually created in a directory /zynthian/zynthian-ui/.vscode
Here is an example launc.json for debugging zynthian:
{
// Use IntelliSense to learn about possible attributes. // Hover to view descriptions of existing attributes. // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 "version": "0.2.0", "configurations": [ { "name": "Zynthian", "type": "debugpy", "request": "launch", "program": "zynthian_main.py", "python": "/zynthian/venv/bin/python3", "console": "integratedTerminal", "justMyCode": true, "env": { "DISPLAY": ":0" } }, { "name": "Zynthian Debug", "type": "debugpy", "request": "launch", "program": "zynthian_main.py", "python": "/zynthian/venv/bin/python3", "console": "integratedTerminal", "justMyCode": true, "subProcess": true, "env": { "DISPLAY": ":0", "ZYNTHIAN_LOG_LEVEL": "10" } } ]
}
This allows two different configurations to be used for debugging with different levels of log reporting.
2.12 How do I stop the zynthian from within VSC?
Once you have a remote terminal within VSC running you type systemctl stop zynthian.
This will stop the zynthian process, and your GUI Screen will disappear and the zynth wont make any sound. ( well it shouldn't). other processes like webconf will continue to work so we only need to restart the zynthian from within VSC to allow us access to VSC debugging facilities.
2.13 How do I re-allocate the screen?
After typing systemctl stop zynthian
type: X -r -s 0
This sorts out the graphics subsystem to work in the debugging situation.
2.14 How do I actually start the VSC Debugger?
Simply press the Debug Green Arrow button
2.15 How do I tell the world about my contribution...?
You put it on Github because that is where the zynthian software itself resides ...
But that is where Zynthian tells the world about its contribution and you need the good people at zynthian to approve your contribution, and they wont do that till you ask in the right way. . .
You need to submit a pull request.
2.16 How do I submit a pull request...?
This is a concept within the github environment, and to do that you need to have an account at github yourself. So you need to dance your way throu their act of anointing (www.github.com).
Now a Pull request is not a specific git command, it's an agreement amongst mortals that the submitted code is suitable to be included in what zynthian publish to the world. One of these mortals then performs a git merge to take your offering, It's called a branch, it has a name, and it contains the code you have written, and combines it into the 'testing' branch so it can be tested.
But, you say, How do I get my code up to github in the first place, and that is, unfortunately, the wrong way to think of it.
You don't put the code there, you make your own copy of the whole zynthian code base in your own github account and download it from there to your machine.
This is called a fork and it means that your submission is derived from the original git hub zynthian site and can, assuming there aren't disagreements of changes two people make (a conflict), then be merged into the code base, using the git merge command and your code is now in testing ...
Congratulations if you get this far. You have just won an open source gold star by having a PR accepted by an open source project.!
So we better write some code.
3 It is 'suggested' one creates a task/issue? how do I do that ?
"It's a good habit to create a task/issue associated with more relevant changes"
You will need a GitHub account to report issues.
So you will have to got to https://github.com/     and open an account there.
It's a Microsoft owned site so vsc helps to view the code, if you want to see the zynthian components at some stage.
You are redirected by the Report issue button to
https://github.com/zynthian/zynthian-issue-tracking
Which is the location of zynthian issue reporting. Please do it here, not on the individual code branches !!
This is so that we know who reported it and can engage with them in its progress.
So to repeat,
The best way to report a bug is to use the "Report Issue" button in the webconf. This creates a new issue in GitHub issue tracker, from a template with some data populated that is useful to the developers.
Then fill out something relevant
3.1 So what have we gained here ?
Well, the communal zynthian entity gains an area of interest. Even if the task, dies on the vine, and nothing more gets done, there is a possible development signposted. You also get a unique identity number that goes into the zynthian event queue. #941 in this case. This is an identifier you can add into a comment with Pull requests so the call for work and the doing of work can be related, tracked and pull requested, accepted tested and completed.
This sort of structure within a project like zynthian is a very good thing. And if precisely scratching your name into a spaceship excites you, then take this as an early buzz.
An issue could be allocated to somebody else at some later time which means the code magically appears from someone you probably would never have met. Of course if you go ahead yourself and write it then so much the better. You move that one step closer to the "Had An Open Source PR accepted"
The GitHub environment allows these relationships to be managed and the more structured this stuff is the less intimidating it becomes...
3.2 How Do I fork the zynthian GitHub repository?
The GitHub write up is here... But in summary it's done from the appropriate zynthian page end of the process.
 
And here are the default options which will suit nearly all circumstances. You should only strike from the testing branch.
Here is the confirmation in my personal GitHub space.
3.3 Can I view files on my zynthian-ui GitHub repository?
In fact there is a very useful GitHub element that allows you to view code on gihub within a browser based tool that is nothing more or less than VSC.
It's no coincidence Microsoft bought GitHub...
To operate this feature simple select the appropriate repository home page... https://github.com/wyleu/zynthian-ui/
and press the . key...
and here in VSC in GitHub showing a ctldev file ready to be examined.
4 How Do I pass code from my Zynthian-ui github repository to my Zynthian on my desk?
This is where we return to applications running on our host computer as opposed to browser based github tools . . and use Visual Studio Code(VSC) actually as an application running on our host machine...
So we start it up, select Clone git repository and it's clever enough to populate the drop down if you ask it to select from GitHub.
Give it a local directory. I store my stuff under Code but it really is up to you. Try to be structural.
And there we have a git repository located on our host machine, which is useful upto a point but isn't actually located on the zynthian . . .
4.1 How Do I access code on my zynthian from my host machine?
We need to use Visual Studio Code(VSC) Remote Explorer (remember to install the extension!) to allow us to add on to the zynthian and install enough of an environment to allow VSC to seamlessly pass code between the environment and the desktop machine. Frankly sometimes it feels like magic, and for anyone who has read my previous attempts to do this sort of thing you spent a lot of time making sure the shaky old connections required behaved as machines are restarted and stopped and broken and goodness knows what else. VSC really sorts this out nicely, and it's a blessing. Indeed it is so tenacious in how it manages connections that the following image is actual a bit of a manufactured effort, but worry no it does it all very well.
 
select Connect to and enter root@zynthian.local or if you have renamed your zynthian as I tend to do to something like root@zynthian-rack2.local you put that in and enter the password for the machine. You will find yourself doing this password entry a fair bit. There are ways to make this once only per machine turning on but we will skip over that for now.
And we now get a new item in the Remote Explorer desktop and are returned to the VSC setup page to describe where we want to locate this connection to the remote machine and it is important that you put
/home/pi/zynthian-ui in here. If you have /zynthian/zynthian-ui in her which might seem like it should work it wont for reasons to do with the long term migration to a more structured system. Don't worry it's in hand.
So we now can examine and edit the code on the zynthian as thou' it is on our desktop... Which is kind of cool.
4.2 How Do I stop being asked for my password?
on your host machine type:
ssh-copy-id root@zynthian-your-machine-name.local
This will copy the key from your host computer to your remote host, where it will allow you to log on without providing the machine password each time, just a first confirmation by asking for your passphrase NOT your password. and when you need to enter the password the first time you will be asked for you pass phrase, if you set one, and since you have probably set this up for github you will have done, This should reduce the chatter a fair bit, but it may still occasionally ask just out of a certain degree of paranoia.. .
4.3 How Do I redirect the zynthian code in zynthian-ui to my personnal repository rather than the zynthian master I forked from ?
We need to edit a specific file on the zynthian within the zynthian-ui folder. It's in a hidden directory .git within the /home/pi/zynthian-ui folder so it won't show up with the ls command, use the ls -las form instead to see it.. Now all this can be done from within VSC using a terminal os you have a nicely set up ssh connection to the machine all maintained by VSC. This terminal is in the lower right hand of the window and it saves you the trouble of having to set up separate connections to do basic housekeeping. Because this file is actually specifically kept out of the repository it doesn't appear in the VSC code listing so we have to go and abuse it by hand, which is why we are using the terminal . . . The terminal opens up in the zynthian-ui folder because that's how we configured it so all we have to do is type
cd .git
if you now type
ls -las
you should see this sort of thing ...
root@zynthian-rack2:/home/pi/zynthian-ui/.git# ls -las total 92
4 drwxr-xr-x 8 root root 4096 Jan 20 21:46 . 4 drwxr-xr-x 14 root root 4096 Jan 15 11:21 .. 4 -rw-r--r-- 1 root root 16 Jan 19 15:47 COMMIT_EDITMSG 4 -rw-r--r-- 1 root root 220 Jan 20 21:02 FETCH_HEAD 4 -rw-r--r-- 1 root root 24 Jan 20 21:44 HEAD 4 -rw-r--r-- 1 root root 41 Jan 20 21:02 ORIG_HEAD 4 drwxr-xr-x 2 root root 4096 Sep 6 2020 branches 4 -rw-r--r-- 1 root root 726 Jan 20 20:55 config 4 -rw-r--r-- 1 root root 73 Sep 5 2020 description 4 drwxr-xr-x 2 root root 4096 Sep 6 2020 hooks 32 -rw-r--r-- 1 root root 32619 Jan 20 21:45 index 4 drwxr-xr-x 2 root root 4096 Sep 6 2020 info 4 drwxr-xr-x 3 root root 4096 Sep 6 2020 logs 4 drwxr-xr-x 133 root root 4096 Jan 19 15:47 objects 4 -rw-r--r-- 1 root root 334 Aug 25 00:11 packed-refs 4 drwxr-xr-x 5 root root 4096 Sep 6 2020 refs
we want to modify the config file in this folder... So lets ook at this file first to see whats in it . . .
cat config
root@zynthian-rack2:/home/pi/zynthian-ui/.git# cat config
[core]
       repositoryformatversion = 0
       filemode = true
       bare = false
       logallrefupdates = true
[remote "origin"]   
       url = https://github.com/zynthian/zynthian-ui.git
       fetch = +refs/heads/*:refs/remotes/origin/*
[branch "master"]
       remote = origin
       merge = refs/heads/master
[branch "testing"]
       remote = origin
       merge = refs/heads/testing
[branch "staging-2209"]
       remote = origin
       merge = refs/heads/staging-2209
[branch "stable"]
       remote = origin
       merge = refs/heads/stable
[branch "staging-2307"]
       remote = origin
       merge = refs/heads/staging-2307
[branch "chain_manager"]
       remote = origin
       merge = refs/heads/chain_manager
We want to change the line 
url = https://github.com/zynthian/zynthian-ui.git
to point to our own repository, which in my case is
url = https://github.com/wyleu/zynthian-ui.git
to do this we use a little command line text editor called nano... so we type
nano config
Control-X to leave Control-O to write out the file and arrow keys move you around in the file
And you have now redirected your zynthian to look at your repository on github as it's ultimate origin file . . .
git pull will now cement this relationship.
4.4 How Do I Start a new branch and what should I call it ?
The VSC status bar at the very bottom of the screen contains a lit of useful information. It shows the remote machine you are logged onto and it shows the current branch you are located on.
Indeed there are useful plugins that can be added that will show git tram lines and such which are very useful . . . but if you press on the testing button in the status bar you will be presented with a dialog for adding a new branch .
Quite what to call it is upto you but remember it will live long in the github zynthian home, so choose something relevant that you can remember what it's about at some point in the future. If it's too scurrilous your Pull Request will be rejected and you will be asked to change it to something more aligned with house style ... Benevolent dictators are perfectly entitled to behave like that .
In summary. git doesn't forget unless you beat it very hard with a stick cos remembering what happened is it's very purpose in life.
4.5 How do I switch to different branches when I'm developing?
Simply use the branch name switch down on the status line at the bottom left of the vsc window. Here is is set to the branch name I created Duo-piano-device
You can also see that I have added three device files into the ctrldev directory which is the code I working on...
4.6 Can I see a diagram of the my various zynthian various branches?
Here is the git graph vsc plugin showing tramlines for this repository. You will need to install this plugin to get this.
4.7 Can I see a diagram of the various branches in the zynthian github page ?
From the github.com/zynthian/zynthian-ui/ you can select insights in the top menu and then Network in the left hand vertical menu. Github will chug away for a little while having a think then show you this, which now features the duo-piano-device branch waiting for me to submit the completion of the code with some more commits and then the pull request which, if successful will result in the branch being merged into testing, and at some later date in stable . . .
5 So what is LV2?
5.1 From wikipedia...
LV2 (LADSPA Version 2) is a set of royalty-free open standards[2] for music production plug-ins and matching host applications. It includes support for the synthesis and processing of digital audio and CV,[3] events such as MIDI and OSC, and provides a free alternative to audio plug-in standards such as Virtual Studio Technology (VST) and Audio Units (AU).
5.2 And from the actual LV2 site...
LV2 is an interface for writing audio plugins in C or compatible languages, which can be dynamically loaded into many host applications. This core specification is simple and minimal, but is designed so that extensions can be defined to add more advanced features, making it possible to implement nearly any feature.
LV2 maintains a strong distinction between code and data. Plugin code is in a shared library, while data is in a companion data file written in Turtle. Code, data, and any other resources (such as waveforms) are shipped together in a bundle directory. The code contains only the executable portions of the plugin. All other data is provided in the data file(s). This makes plugin data flexible and extensible, and allows the host to do everything but run the plugin without loading or executing any code. Among other advantages, this makes hosts more robust (broken plugins can't crash a host during discovery) and allows generic tools written in any language to work with LV2 data. The LV2 specification itself is distributed in a similar way.
LV2 is an extensible framework, allowing a program to load a plugin to do some processing. Note that the terms used here are generic on purpose because LV2 allows any type of data to be exchanged between the host and the plugin.
5.3 In Summary
So there is a clear distinction between the definitions of the facilities which are held in carefully defined definition files and the actual software that accesses the facilities by reading the definition files. This way there is a controlled interaction betwenn the host and the plugin and each can be protected from the vagueries of the other.
It is a list of definitions that refer back to core definitions and these relationships are maintained in a format called RDF . . .
5.4 So what is RDF ?
The Resource Description Framework (RDF) is a method to describe and exchange graph data. It was originally designed as a data model for metadata by the World Wide Web Consortium (W3C). It provides a variety of syntax notations and formats, of which the most widely used is Turtle (Terse RDF Triple Language).
It describes objects and the relationships between them . . .
RDF represents information using semantic triples, which comprise a subject, predicate, and object. Each item in the triple is expressed as a Web URI. Turtle provides a way to group three URIs to make a triple, and provides ways to abbreviate such information, for example by factoring out common portions of URIs. For example, information about Huckleberry Finn could be expressed as:
<http://example.org/books/Huckleberry_Finn> <http://example.org/relation/author> <http://example.org/person/Mark_Twain> .
In the zynthian plug in world it's primarily about the definition, allocation and mapping of audio and midi ports and parameters in the zynthian world to parameters provided by the LV2 plugin we are exchanging data with.
But first you establish some prefixes so you aren't typing out URL all the time... Here the classes that are defined by lv2 are specified ( follow the link) and similarly DOAP defines structure for a software project, and SPDX is the software package data exchange. Thus a set of definitions are described that can be understood by humans ( to a certain extent) and machines (VERY specifically)
@prefix lv2: <http://lv2plug.in/ns/lv2core#>. @prefix doap: <http://usefulinc.com/ns/doap#>. @prefix spdx: <http://spdx.org/rdf/terms#>.
Then you use these prefixes to define the relationships we are concerned with...
The 'a' is an atom. More news as I get it. . .
<http://example.org/lv2/wikipediaexample/silence> a lv2:Plugin; lv2:binary <silence.so>; doap:name "Silence"; doap:license spdx:GPL-3.0-or-later; rdfs:comment "This is an example plugin that includes an example plugin description."
lv2:port [ a lv2:AudioPort, lv2:OutputPort; lv2:index 0; lv2:symbol "output"; lv2:name "Output"; ].
5.4.1 How Do I start writing a lv2 component compatible with Zynthian?
5.4.1.1 Which zynthian components do I need to clone in Github to fully develop a plug in?
Not sure.
5.4.1.2 What do I need in code terms in addition to the normal zynthian files?
Somewhere to develop and test the code as we have described above. Now at this point we have to dip below the passive waters of Python where everything is slow, leisurely and well behaved. And embrace the proscribed world of C & C++ where code runs fast and life is cheap.
You will also need components that actually understand the concepts lv2 embodies. ( hopefully someone ( you know who you are)) will jump in and give us the history of this particular chunk of the mostly virtual universe we inhabit.
I'm working off a Grok summary on building a 50Hz & 60Hz notch filter and will attempt to ruthlessly hone this to something vaguely useful. Alternatively, I might go bell ringing.
You will need some lv3 libraries...
sudo apt install libasound2-dev lv2-dev
5.4.1.2.1 libasound2-dev
From the README.md file. The alsa-lib is a library to interface with ALSA in the Linux kernel and virtual devices using a plugin system.
These are the alsa libraries, the essential glue to plug into the operating systems understanding of sound & MIDI on a Pi. Obviously this (at the moment) will only run on a Pi, so code elements at this level are aimed at that particular machine. That doesn't actually mean you can't write it on some other operating system, just be aware when you might be trying to put pears in containers designed for apples. . .
These libraries are the components that allow you to run the functionality described. But only that. If you wish to connect to them from your own programme then we need to install a little bit more, like descriptions of the hooks that those code pieces require and this is what the -dev on the end of the library name describes. Easy mistake to make it the early days, not installing the dev editions.
Why the slightly different names asound2 and alsa? Alsa has always been a bit impenetrable .
5.4.1.2.2 lv2-dev
LV2 is a plugin standard for audio systems. It defines an extensible C API for plugins, and a format for self-contained "bundle" directories that contain plugins, metadata, and other resources
LV2 components and the appropriate development files to muck around with lv2 files.
5.4.1.3 Anything else?
You are best to use a LV2 compatibleframework to construct the components to allow integration with the Linux system
There are:
- Juce
- DPF
5.4.1.4 Juce
Juce is a popular and stable cross-platform open-source C++ development system that lets you write an application as a single codebase that works on Windows, OS/X and Linux. Juce has a huge number of components relating to pretty well anything you need - GUI, audio processing, networking, etc. - and is particularly popular amongst audio software developers. You can find a lot more information and some snappy demos on the Juce site. It turns out that Juce supports the Raspberry Pi "out of the box" as long as you install a few libraries before you start. I created a sample application on the Mac, copied it to a Raspberry Pi and compiled it, and it worked right the first time!
5.4.1.5 DPF
git clone --recursive https://github.com/DISTRHO/DPF.git
This is a tool to assist plug in writing....
DPF is designed to make development of new plugins an easy and enjoyable task. It allows developers to create plugins with custom UIs using a simple C++ API. The framework facilitates exporting various different plugin formats from the same code-base.
DPF can build for LADSPA, DSSI, LV2, VST2, VST3 and CLAP formats. A JACK/Standalone mode is also available, allowing you to quickly test plugins.
Then make a directory.
mkdir SimpleGainPlugin cd SimpleGainPlugin
Save the three files below (DistrhoPluginMain.cpp, DistrhoPluginInfo.h, Makefile) in this directory.
5.4.1.6 So which is better?
Grok provided a comparison. For developing LV2 plugins on Zynthian, DPF is the better choice due to its native LV2 support, lightweight design, and alignment with Zynthian’s open-source, Linux-based ecosystem. It integrates seamlessly with Zynthian’s plugin system and performs well on its hardware. Use JUCE only if you need cross-platform compatibility (e.g., developing for VST3/AU alongside LV2) or prefer its polished development tools, but be prepared to optimize heavily for Zynthian’s constraints.
6 LV2 Components
6.1 A Notch Filter
6.1.1 The Definition Files
6.1.2 The Code components
6.1.2.1 DisthroPluginInfo.h
#ifndef DISTRHO_PLUGIN_INFO_H_INCLUDED #define DISTRHO_PLUGIN_INFO_H_INCLUDED #define DISTRHO_PLUGIN_BRAND "Wyleu" #define DISTRHO_PLUGIN_NAME "SimpleGain" #define DISTRHO_PLUGIN_URI "http://zynthian.org/plugins/lv2/simplegain" #define DISTRHO_PLUGIN_HAS_UI 0 #define DISTRHO_PLUGIN_IS_RT_SAFE 1 #define DISTRHO_PLUGIN_NUM_INPUTS 2 #define DISTRHO_PLUGIN_NUM_OUTPUTS 2 #define DISTRHO_PLUGIN_WANT_LATENCY 0 #define DISTRHO_PLUGIN_WANT_PROGRAMS 0 #define DISTRHO_PLUGIN_WANT_STATE 0 #define DISTRHO_PLUGIN_WANT_TIMEPOS 0 #define DISTRHO_PLUGIN_IS_SYNTH 0 #endif
6.1.2.2 Partial Explanation of DistrhoPluginInfo.h
This defines the plugin’s metadata.
You get to define some information about you and what you are trying to do. You make decisions like am I going to use a GUI? How many inputs and outputs it will have and the expectations as how it will interact with the environment.
All wrapped in a test which is then settrue as part of programme flow. . . Very neat.
#ifndef DISTRHO_PLUGIN_INFO_H_INCLUDED #define DISTRHO_PLUGIN_INFO_H_INCLUDED ... #endif
The Python programmer in me would like a little indentation but such is life...
6.1.2.3 DisthroPluginMain.cpp
#include "DistrhoPlugin.hpp"
START_NAMESPACE_DPF
class SimpleGainPlugin : public Plugin {
public:
   SimpleGainPlugin() : Plugin(1, 0, 0) { // 1 parameter, 0 programs, 0 states
       gain = 1.0f; // Default gain (no change)
   }
protected:
   const char* getLabel() const override { return "SimpleGain"; }
   const char* getDescription() const override { return "A simple gain plugin for Zynthian"; }
   const char* getMaker() const override { return "YourName"; }
   const char* getLicense() const override { return "MIT"; }
   uint32_t getVersion() const override { return d_version(1, 0, 0); }
   int64_t getUniqueId() const override { return d_cconst('S', 'G', 'a', 'n'); }
   void initParameter(uint32_t index, Parameter& parameter) override {
       if (index != 0) return;
       parameter.hints = kParameterIsAutomable;
       parameter.name = "Gain";
       parameter.symbol = "gain";
       parameter.unit = "x";
       parameter.ranges.def = 1.0f;
       parameter.ranges.min = 0.0f;
       parameter.ranges.max = 2.0f;
   }
   float getParameterValue(uint32_t index) const override {
       if (index != 0) return 0.0f;
       return gain;
   }
   void setParameterValue(uint32_t index, float value) override {
       if (index != 0) return;
       gain = value;
   }
   void run(const float** inputs, float** outputs, uint32_t frames) override {
       const float* inL = inputs[0];
       const float* inR = inputs[1];
       float* outL = outputs[0];
       float* outR = outputs[1];
       for (uint32_t i = 0; i < frames; ++i) {
           outL[i] = inL[i] * gain;
           outR[i] = inR[i] * gain;
       }
   }
private:
   float gain;
   DISTRHO_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(SimpleGainPlugin)
};
Plugin* createPlugin() {
   return new SimpleGainPlugin();
}
END_NAMESPACE_DPF
6.1.2.4 Explanation of DistrhoPluginMain.cpp
This is the main plugin implementation.
It includes the header files
#include "DistrhoPlugin.hpp"
we define a class:
class SimpleGainPlugin : public Plugin {
public:
   SimpleGainPlugin() : Plugin(1, 0, 0) { // 1 parameter, 0 programs, 0 states
       gain = 1.0f; // Default gain (no change)
   }
And within that class we set the gain to 1.0f
This means a float (A floating point number 1.0345 sorts of things not 1,2,3,4,5). Basically what it gets in it puts straight out in this case and if we substituted 1.0345f for the 1.0f the signal would get a little bit louder. A digital Patch lead!
This class will be returned as an object of type Plugin. But quite how and where Plugin is defined remains to be found....
At the bottom of the code we instantiate (generate, create, make alive...) the object and return it to the calling function. 
Plugin* createPlugin() {
   return new SimpleGainPlugin();
}
So this is a little factory that makes a Plugin object that can interact with the rest of the alsa audio framework.
So what's the rest of it...
Two sections
- protected:
- private:
6.1.2.4.1 Protected:
Members declared as protected are accessible within the same class and in derived classes Some functions are declared
Firstly those that return character strings of text (char*)
getLabel() const overide { return "SimpleGain"; }
getDescription()
getMaker()
getLicense()
then a function that returns an unsigned 32 bit integer 0 -> 4 Gig
uint32_t getVersion() const override { return d_version(1, 0, 0); }
and an even more massive number 64 bit integer ( don't ask)
   int64_t getUniqueId() const override { return d_cconst('S', 'G', 'a', 'n'); }
Next we have a function that initializes some Parameters and define a range of values for that handy gain number we'd like to alter.
   void initParameter(uint32_t index, Parameter& parameter) override {
       if (index != 0) return;
       parameter.hints = kParameterIsAutomable;
       parameter.name = "Gain";
       parameter.symbol = "gain";
       parameter.unit = "x";
       parameter.ranges.def = 1.0f;
       parameter.ranges.min = 0.0f;
       parameter.ranges.max = 2.0f;
This applies extra values to the external implementation of this parameter so the two environments can pass data effectively.
index is required to have a value 0 to set these values to what are presumably defaults.
Next we have a Getter & a Setter
   float getParameterValue(uint32_t index) const override {
       if (index != 0) return 0.0f;
       return gain;
   }
   void setParameterValue(uint32_t index, float value) override {
       if (index != 0) return;
       gain = value;
   }
Again requiring a 0 index to operate.
Lastly in the protected section
   void run(const float** inputs, float** outputs, uint32_t frames) override {
       const float* inL = inputs[0];
       const float* inR = inputs[1];
       float* outL = outputs[0];
       float* outR = outputs[1];
       for (uint32_t i = 0; i < frames; ++i) {
           outL[i] = inL[i] * gain;
           outR[i] = inR[i] * gain;
       }
   }
The run function of this class, which will be called by the underlying system to process audio samples appearing at the input and generating appropriate output at the correct time.
i.e. doing the clever stuff between these two...
void run(const float** inputs, float** outputs, uint32_t frames) override
It's defined as a void so it does not return a value, interacting via the arrays passed to it.
The inputs to the tun function are a
an array of inputs an array of outputs A frame count. The number of frames in the array.
Don't know what override does.
The arrays are allocated to variables ( notice the assumption here that we only handle the first two arrays handed to us. How it deals with other arrays is a point of discussion.)
       const float* inL = inputs[0];
       const float* inR = inputs[1];
       float* outL = outputs[0];
       float* outR = outputs[1];
So for each frame we loop a counter i with that numbered frame of audio
       for (uint32_t i = 0; i < frames; ++i) {
and then do stuff
           outL[i] = inL[i] * gain;
           outR[i] = inR[i] * gain;
       }
At the end of the process the out arrays have been filled with the adjusted values from the input arrays. . . .
And these can be handled by the framework infrastructure to throw it at the next device down the chain.
6.1.2.4.2 Private:
Members declared as private are accessible only within the same class. They cannot be accessed by derived classes, objects of the class, or external code. Used to encapsulate data and hide implementation details.
6.1.2.4.3 So where do these undefined chunks of code come from?
We haven't had any nice import functions to do this for us so how can these references to external functions be addressed?
We need to 'link' the code we have written with the development hooks of the libraries we imported earlier.
We need something to 'make' our programme. Enter the makefile . . .
6.1.2.5 makefile
#!/usr/bin/make -f NAME=SimpleGain DPF_PATH=../../DPF include $(DPF_PATH)/Makefile.plugins.mk TARGETS=lv2 all: $(TARGETS) include $(DPF_PATH)/Makefile.base.mk
6.1.2.6 makefile what does it do ?
This attempts to builds the plugin as an LV2 bundle.
The two files mentioned here are frankly enormous. I assume these are built by code not hand but have absolutely no desire to venture into them with any exploratory desire.
I wonder if 'make' has any way of reporting the world it believes it's constructing ahead of time?
wyleu@raspberrypi:~/Code/lv2/DPF/SimpleGainPlugin $ make -n mkdir -p /bin/SimpleGain.lv2 echo "Creating LV2 plugin for SimpleGain" g++ /build/SimpleGain/DistrhoPluginMain_LV2.cpp.o -Wall -Wextra -pipe -MD -MP -fno-gnu- unique -fPIC -DPIC -DNDEBUG -O3 -ffast-math -fdata-sections -ffunction-sections - fvisibility=hidden -DHAVE_ALSA -DHAVE_JACK -DHAVE_RTAUDIO -DHAVE_X11 -DHAVE_XEXT - DHAVE_XSYNC -DHAVE_OPENGL -std=gnu++11 -fvisibility-inlines-hidden -I. -I../../DPF/distrho - I../../DPF/dgl -I../../DPF/dgl/src/pugl-upstream/include -fdata-sections -ffunction-sections -Wl,-O1,--as-needed,--gc-sections -Wl,--strip-all -Wl,--no-undefined -ldl -shared -Wl,-- version-script=../../DPF/utils/symbols/lv2.version -o /bin/SimpleGain.lv2/SimpleGain.so
Not sure where the line breaks and if there are any in that mouthful.
make -p generates enormous amounts of information..
make -d provides what we already know....
Live child 0x555658e367a0 (/bin/SimpleGain.lv2/SimpleGain.so) PID 51392 Reaping winning child 0x555658e367a0 PID 51392 Live child 0x555658e367a0 (/bin/SimpleGain.lv2/SimpleGain.so) PID 51393 Creating LV2 plugin for SimpleGain Reaping winning child 0x555658e367a0 PID 51393 Live child 0x555658e367a0 (/bin/SimpleGain.lv2/SimpleGain.so) PID 51394 /usr/bin/ld: cannot open output file /bin/SimpleGain.lv2/SimpleGain.so: Permission denied
6.2 Auto Leveller
A Place holder for information on the construction of this component and a casual implementation of some kind of document structure.
6.2.1 What is it?
You can think of it as your angry sound engineer in a live situation, both during soundcheck and during the gig. Initially it will turn your output volume down to a reasonable level. It will also do that if anything unforseen happens in the chain (you add another plugin which features an integer gain multiplier and is set to gain=20). When set the right values (try the preset “Tuned”) it should level any source to a loudness level somewhere into yellow meter territory.
It also features two programs:
Init: All set to zero. It will not perform anything unless your level is above 0 dBfs, which is always bad. Tuned: Set the parameters to reasonable values: Pre Gain will boost the signal by +6 dB to compensate quiet engines like soundfonts, peak threshold is set to -4 dBfs and RMS to -18 dBFS to keep more or less transient heavy signals somewhere in yellow area of your mixer.
6.2.2 Parameters
6.2.2.1 Pre Gain (dBfs)
Adjust the pre amplification in case you deal with quieter sources (like soundfonts) you want to boost before getting into the automatic levelling.
6.2.2.2 Theshold Peak (dBfs)
Set your maximum peak levels you want to have. Setting the threshold will reset your former gain reduction so it can listen from the start again. When “Listen” is on, it will only turn the volume downwards. Recommended values are between 0 and -4 dBfs.
6.2.2.3 Theshold RMS (dBfs)
Set your maximum RMS levels you want to have. Setting the threshold will reset your former gain reduction so it can listen from the start again. When “Listen” is on, it will only turn the volume downwards. The detector is estimating RMS values of a 300ms window by a using a lowpass filter. Recommended values are between -14 and -18 dBfs.
6.2.2.4 Listen (bool)
When engaged (by default) it will perform detection and level matching. When disengaged it will keep the current gain, but will not further listen. If engaged again, the gain reduction will be reset.
7 LV2 custom control groups (ttl)
If not edited further, LV2 parameters will be presented in a manner being sorted by their parameter ID and grouped in groups of four without helpful group names. We can edit a specific file to get some more helpful named and ordered parameter groups. It is highly recommended to study other custom control ttl files before edit your own one.
7.1 Location
We start by searching for the original ttl file in the zynthian filesystem. Typically we'll find them in /usr/lib/lv2/[PLUGIN].lv2 or whereever this plugin lives. We copy the whole [PLUGIN].lv2 directory over to /zynthian/zynthian-data/lv2-custom (check the other custom ttls for reference). When we look inside, we'll find several *.so and *.ttl files. The file we are searching for is not the manifest.ttl but the other one. We can delete any other file from that directory.
On any zynthian update database regeneration zynthian will replace the original ttl file with the ones in /zynthian/zynthian-data/lv2-custom.
7.2 Editing the ttl file
7.2.1 Namespace prefixes
Most of the time we have to add some lines to the header, if not already present:
@prefix rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> . @prefix pg: <http://lv2plug.in/ns/ext/port-groups#> . @prefix plug: <COPY URN FROM PLUGIN BLOCK> .
These namespace prefixes act like unique identifiers for specific tasks. The URN to copy we'll find on top of the plugin block, which is the section that holds all the control ports. Just look into another custom ttl and compare the top of a plugin block with the URN set under the prefix "plug:".
7.2.2 Control groups
Add control groups like that outside the plugin block, for example right at the end of the file. Groups with higher displayPriority values will be displayed on top of your parameter group view. GROUP1 and the symbol "group1" can be named yourself (only letters, numbers or _", name "Group1 Display Name" you can also edited and will show on your UI.
plug:GROUP1 a pg:ControlGroup ; lv2:name "Group1 Display Name" ; lv2:symbol "group1" ; lv2:displayPriority 3 . plug:GROUP2 a pg:ControlGroup ; lv2:name "Group2 Display Name" ; lv2:symbol "group2" ; lv2:displayPriority 2 . ...
7.2.3 Parameter assignment
Add following lines to every control port representing a parameter (contains "a lv2:ControlPort", so not the audio input and output ports), while display priority goes from higher to lower numbers as well:
pg:group plug:GROUP1 ; lv2:displayPriority 1 ;
7.2.4 Scale Points
This is especially helpful for integer or boolean like enumeration parameters and other parameters which should have stepped controls. Here we have an example for a boolean parameter, which should only be in state "On" or "Off":
lv2:scalePoint [ rdf:value 0.0 ; rdfs:label "OFF" ; rdfs:comment "OFF" ; ], [ rdf:value 1.0 ; rdfs:label "ON" ; rdfs:comment "ON" ; ];
It has to be like that because typically (oftentimes, but not necessarily) parameters are represented as floating point values (0.000f to 1.000f), even if they're supposed to be boolean or integer, even if LV2 would support these kind of data. Other parameters could be of "integer logic". Let's assume the parameter "Waveform" would have the states "Saw, Sine, Square, Triange", we may assume that every floating point value from 0.0f to <0.25f represents "Saw" and so on. Then we might try this (while value and label are mandatory, comment is not):
lv2:scalePoint [ rdf:value 0.0 ; rdfs:label "Saw" ; rdfs:comment "Saw" ; ], [ rdf:value 0.25 ; rdfs:label "Sine" ; rdfs:comment "Sine" ; ], [ rdf:value 0.5 ; rdfs:label "Square" ; rdfs:comment "Square" ; ], [ rdf:value 0.75 ; rdfs:label "Triangle" ; rdfs:comment "Triangle" ; ];
7.2.5 Disable parameter from being shown on UI
Sometimes we have parameters which have no function and must not be displayed. You add the following line to the parameter definition:
lv2:portProperty pp:notOnGUI ;
7.2.6 Summary parameter definition
So a single parameter should for example look like this:
   [
       a lv2:InputPort, lv2:ControlPort ;
       lv2:index 8 ;
       lv2:symbol "delaytimesync" ;
       lv2:name "Host Sync" ;
       lv2:default 0.473684 ;
       lv2:minimum 0.0 ;
       lv2:maximum 1.0 ;
       pg:group plug:DELAY ;
       lv2:displayPriority 2 ;
       lv2:scalePoint [
           rdf:value 0.0 ;
           rdfs:label "Free" ;
           rdfs:comment "Free" ;
       ], [
           rdf:value 0.052631 ;
           rdfs:label "1/16" ;
           rdfs:comment "1/16" ;
       ], [
           rdf:value 0.105263 ;
           rdfs:label "1/8" ;
           rdfs:comment "1/8" ;
       ], [
           rdf:value 0.157895 ;
           rdfs:label "1/4" ;
           rdfs:comment "1/4" ;
       ], [
           rdf:value 0.211053 ;
           rdfs:label "1/2" ;
           rdfs:comment "1/2" ;
       ], [
           rdf:value 0.263158 ;
           rdfs:label "1/1" ;
           rdfs:comment "1/1" ;
       ], [
           rdf:value 0.315789 ;
           rdfs:label "2/1" ;
           rdfs:comment "2/1" ;
       ], [
           rdf:value 0.368421 ;
           rdfs:label "1/16." ;
           rdfs:comment "1/16." ;
       ], [
           rdf:value 0.421052 ;
           rdfs:label "1/8." ;
           rdfs:comment "1/8." ;
       ], [
           rdf:value 0.473684 ;
           rdfs:label "1/4." ;
           rdfs:comment "1/4." ;
       ], [
           rdf:value 0.526316 ;
           rdfs:label "1/2." ;
           rdfs:comment "1/2." ;
       ], [
           rdf:value 0.578947 ;
           rdfs:label "1/1." ;
           rdfs:comment "1/1." ;
       ], [
           rdf:value 0.631579 ;
           rdfs:label "2/1." ;
           rdfs:comment "2/1." ;
       ], [
           rdf:value 0.684211 ;
           rdfs:label "1/16T" ;
           rdfs:comment "1/16T" ;
       ], [
           rdf:value 0.736842 ;
           rdfs:label "1/8T" ;
           rdfs:comment "1/8T" ;
       ], [
           rdf:value 0.789474 ;
           rdfs:label "1/4T" ;
           rdfs:comment "1/4T" ;
       ], [
           rdf:value 0.842105 ;
           rdfs:label "1/2T" ;
           rdfs:comment "1/2T" ;
       ], [
           rdf:value 0.894736 ;
           rdfs:label "1/1T" ;
           rdfs:comment "1/1T" ;
       ], [
           rdf:value 0.947368 ;
           rdfs:label "2/1T" ;
           rdfs:comment "2/1T" ;
       ], [
           rdf:value 1.0 ;
           rdfs:label "-" ;
           rdfs:comment "-" ;
       ];
   ] ,
The bold line is the one we started to insert things. You typically should only edit the "name" value, which is the display label and the newly added values. In this case we added the control to the previously added group named "DELAY", added the displayPriority, so that it will be displayed within the group above all parameters with value <2 and below paramters with value >2. Then we added stepped controls for delay times defined by subdivisions of the host tempo, where the values represent 1/19, 2/19, 3/19... Continuous parameters typically do not need scale points at all, rather they would become stepped parameters as well if we'd add them.
7.3 Recommended workflow
Instead of just copying the ttl over to the respective directory and edit it for your own use you can make things easier for the devs to implement your custom feature and share it with everybody this way:
- Fork the zynthian/zynthian-data repositories vangelis branch
- Checkout to that branch in your zynthian
- Develop using your fork
- Make a pull request





























