From 7e3a993406a08f2c260d4dfa90d228894edad6d4 Mon Sep 17 00:00:00 2001 From: Giulio Moro Date: Sat, 8 Sep 2018 19:26:27 +0100 Subject: [PATCH] All the bela changes, in one single commit, rebased on 3.11.x --- CMakeLists.txt | 2 +- HelpSource/Classes/AnalogIn.schelp | 50 + HelpSource/Classes/AnalogOut.schelp | 55 + HelpSource/Classes/DigitalIO.schelp | 72 + HelpSource/Classes/DigitalIn.schelp | 55 + HelpSource/Classes/DigitalOut.schelp | 71 + HelpSource/Classes/MultiplexAnalogIn.schelp | 58 + README_BELA.md | 359 +++++ SCClassLibrary/Common/Audio/bela/BELAUGens.sc | 94 ++ SCClassLibrary/Common/Control/Server.sc | 58 + bela_analog_audio_io.png | Bin 0 -> 22655 bytes cmake_modules/FindBela.cmake | 79 + cmake_modules/FindXenomai.cmake | 156 ++ examples/bela/bela_example_analogin.scd | 25 + examples/bela/bela_example_analogin_2.scd | 20 + examples/bela/bela_example_analogout.scd | 16 + examples/bela/bela_example_digital.scd | 18 + examples/bela/bela_example_digitalio.scd | 16 + examples/bela/bela_example_digitalout.scd | 16 + examples/bela/bela_start_scsynth.scd | 21 + examples/bela/bela_start_scsynth_2.scd | 22 + examples/bela/bela_test_cases.scd | 351 +++++ include/plugin_interface/SC_World.h | 20 + include/server/SC_WorldOptions.h | 15 + lang/CMakeLists.txt | 2 +- server/plugins/BELAUGens.cpp | 1386 +++++++++++++++++ server/plugins/CMakeLists.txt | 47 + server/scsynth/CMakeLists.txt | 31 +- server/scsynth/SC_Bela.cpp | 502 ++++++ server/scsynth/SC_CoreAudio.cpp | 19 +- server/scsynth/SC_CoreAudio.h | 1 + server/scsynth/SC_World.cpp | 34 + server/scsynth/scsynth_main.cpp | 76 +- 33 files changed, 3735 insertions(+), 12 deletions(-) create mode 100644 HelpSource/Classes/AnalogIn.schelp create mode 100644 HelpSource/Classes/AnalogOut.schelp create mode 100644 HelpSource/Classes/DigitalIO.schelp create mode 100644 HelpSource/Classes/DigitalIn.schelp create mode 100644 HelpSource/Classes/DigitalOut.schelp create mode 100644 HelpSource/Classes/MultiplexAnalogIn.schelp create mode 100644 README_BELA.md create mode 100644 SCClassLibrary/Common/Audio/bela/BELAUGens.sc create mode 100644 bela_analog_audio_io.png create mode 100644 cmake_modules/FindBela.cmake create mode 100644 cmake_modules/FindXenomai.cmake create mode 100644 examples/bela/bela_example_analogin.scd create mode 100644 examples/bela/bela_example_analogin_2.scd create mode 100644 examples/bela/bela_example_analogout.scd create mode 100644 examples/bela/bela_example_digital.scd create mode 100644 examples/bela/bela_example_digitalio.scd create mode 100644 examples/bela/bela_example_digitalout.scd create mode 100644 examples/bela/bela_start_scsynth.scd create mode 100644 examples/bela/bela_start_scsynth_2.scd create mode 100644 examples/bela/bela_test_cases.scd create mode 100644 server/plugins/BELAUGens.cpp create mode 100644 server/scsynth/SC_Bela.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 432de06c668..b0c202f1105 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -164,7 +164,7 @@ else() # ARM platforms do not have SSE set(SSE2 OFF) endif() -set(AUDIOAPI "default" CACHE STRING "Audio API to use (one of {default,coreaudio,jack,portaudio})") +set(AUDIOAPI "default" CACHE STRING "Audio API to use (one of {default,coreaudio,jack,portaudio,bela})") if (AUDIOAPI STREQUAL jack) # here we check for JACK metadata API diff --git a/HelpSource/Classes/AnalogIn.schelp b/HelpSource/Classes/AnalogIn.schelp new file mode 100644 index 00000000000..a788318a4ec --- /dev/null +++ b/HelpSource/Classes/AnalogIn.schelp @@ -0,0 +1,50 @@ +class:: AnalogIn +summary:: Read data from an analog input of the BELA board +related:: Classes/AnalogOut, Classes/DigitalIn, Classes/DigitalOut, Classes/DigitalIO +categories:: UGens>BELA + + +Description:: + +Reads analog data from an analog input of the BELA board. + +note:: +This UGen only works on BeLa +:: + +classmethods:: + +method::ar + +argument::analogPin + +Analog pin number to read. Pin numbers begin at 0. This value can be modulated at audiorate. + +argument::mul + +argument::add + +method::kr + +argument::analogPin + +Analog pin number to read. Pin numbers begin at 0. + +argument::mul + +argument::add + +Examples:: + +code:: +// modulate frequency of a sine oscillator + +( +SynthDef("help-AnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( AnalogIn.ar( 0 ).exprange( 200, 5000 ), 0, 0.1 ) + ) +}).play; +) +:: + diff --git a/HelpSource/Classes/AnalogOut.schelp b/HelpSource/Classes/AnalogOut.schelp new file mode 100644 index 00000000000..37b6e4a4c09 --- /dev/null +++ b/HelpSource/Classes/AnalogOut.schelp @@ -0,0 +1,55 @@ +class:: AnalogOut +summary:: Write data to an analog output of the BELA board +related:: Classes/AnalogIn, Classes/DigitalIn, Classes/DigitalOut, Classes/DigitalIO +categories:: UGens>BELA + + +Description:: + +Writes analog data to an analog output of the BELA board. + +note:: +This UGen only works on BeLa +:: + +classmethods:: + +method::ar + +argument::analogPin + +Analog pin number to read. Pin numbers begin at 0. This value can be modulated at audiorate. + +argument::output + +Value to write out to the pin. + +argument::mul + +argument::add + +method::kr + +argument::analogPin + +Analog pin number to read. Pin numbers begin at 0. + +argument::output + +Value to write out to the pin. + +argument::mul + +argument::add + +Examples:: + +code:: +// write a sine oscillator's output to a pin + +( +SynthDef("help-AnalogOut",{ arg out=0; + AnalogOut.ar( 0 , SinOsc.ar( 10 ) ); +}).play; +) +:: diff --git a/HelpSource/Classes/DigitalIO.schelp b/HelpSource/Classes/DigitalIO.schelp new file mode 100644 index 00000000000..7cfaccdd2f5 --- /dev/null +++ b/HelpSource/Classes/DigitalIO.schelp @@ -0,0 +1,72 @@ +class:: DigitalIO +summary:: Read or write data to a digital pin of the BELA board +related:: Classes/AnalogIn, Classes/AnalogOut, Classes/DigitalIn, Classes/DigitalOut +categories:: UGens>BELA + + +Description:: + +Read or write digital data from or to a digital pin. The pin number of this UGen can be modulated, as well as its I/O mode. + +note:: +This UGen only works on BeLa. +:: + +note:: +If you do not need to change the pin mode or the pin, you should use the UGen link::Classes/DigitalIn:: or link::Classes/DigitalOut:: +:: + +classmethods:: + +method::ar + +The output of this UGen is always the last value read when the digital pin was an input. + +argument::digitalPin + +Digital pin number to write to. Pin numbers begin at 0. This value can be modulated at audiorate. + +argument::output + +Value to write out to the pin - the value will be 1 when the argument is larger than 0, otherwise 0. This value can be modulated at audio rate. + +argument::pinMode + +Value to write out to the pin - the pin will be an input when the argument is smaller than 0.5, otherwise an output. This value can be modulated at audiorate. + +argument::mul + +argument::add + +method::kr + +The output of this UGen is always the last value read when the digital pin was an input. + +argument::digitalPin + +Digital pin number to write to. Pin numbers begin at 0. + +argument::output + +Value to write out to the pin - the value will be 1 when the argument is larger than 0, otherwise 0. + +argument::pinMode + +Value to write out to the pin - the pin will be an input when the argument is smaller than 0.5, otherwise an output. + +argument::mul + +argument::add + +Examples:: + +code:: +// write a sine oscillator's output to a pin, and read the pin value at other times + +( +SynthDef("help-DigitalIO",{ arg out=0; + DigitalIO.ar( 0, SinOsc.ar( 10 ),LFPulse.kr( 0.1 ) ).poll; +}).play; +) + +:: diff --git a/HelpSource/Classes/DigitalIn.schelp b/HelpSource/Classes/DigitalIn.schelp new file mode 100644 index 00000000000..1adc44f9459 --- /dev/null +++ b/HelpSource/Classes/DigitalIn.schelp @@ -0,0 +1,55 @@ +class:: DigitalIn +summary:: Read data from a digital input of the BELA board +related:: Classes/AnalogIn, Classes/AnalogOut, Classes/DigitalOut, Classes/DigitalIO +categories:: UGens>BELA + + +Description:: + +Reads digital data from an digital input of the BELA board. + +note:: +This UGen only works on BeLa. +:: + +note:: +If you want to modulate the pin number, you should use the UGen link::Classes/DigitalIO:: +:: + +classmethods:: + +method::ar + +argument::digitalPin + +Digital pin number to read. Pin numbers begin at 0. This value cannot be modulated. + +argument::mul + +argument::add + +method::kr + +argument::digitalPin + +Digital pin number to read. Pin numbers begin at 0. This value cannot be modulated. + +argument::mul + +argument::add + + +Examples:: + +code:: +// turn on and off a sine oscillator + +( +SynthDef("help-DigitalIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( 500, 0, 0.1 * DigitalIn.ar( 0 ) ) + ) +}).play; +) +:: + diff --git a/HelpSource/Classes/DigitalOut.schelp b/HelpSource/Classes/DigitalOut.schelp new file mode 100644 index 00000000000..d5d59131cc4 --- /dev/null +++ b/HelpSource/Classes/DigitalOut.schelp @@ -0,0 +1,71 @@ +class:: DigitalOut +summary:: Write data to a digital input of the BELA board +related:: Classes/AnalogIn, Classes/AnalogOut, Classes/DigitalIn, Classes/DigitalIO +categories:: UGens>BELA + + +Description:: + +Write digital data to a digital output of the BELA board. + +note:: +This UGen only works on BeLa. +:: + +note:: +If you want to modulate the pin number, you should use the UGen link::Classes/DigitalIO:: +:: + +classmethods:: + +method::ar + +argument::digitalPin + +Digital pin number to write to. Pin numbers begin at 0. This value cannot be modulated. + +argument::output + +Value to write out to the pin - the value will be 1 when the argument is larger than 0, otherwise 0. + +argument::writeMode + +Mode of writing to the output, this can be 0 (using Bela's DigitalWrite and only when value changes) or 1 (using Bela's DigitalWriteOnce). This value cannot be modulated. + +argument::mul + +argument::add + + +method::kr + +argument::digitalPin + +Digital pin number to write to. Pin numbers begin at 0. This value cannot be modulated. + +argument::output + +Value to write out to the pin - the value will be 1 when the argument is larger than 0, otherwise 0. + +argument::writeMode + +Mode of writing to the output, this can be 0 (using Bela's DigitalWrite and only when value changes) or 1 (using Bela's DigitalWriteOnce). This value cannot be modulated. + +argument::mul + +argument::add + + +Examples:: + +code:: +// write a sine oscillator's output to a pin + +( +SynthDef("help-DigitalOut",{ arg out=0; + DigitalOut.ar( 0, SinOsc.ar( 10 ) ); +}).play; +) + +:: + diff --git a/HelpSource/Classes/MultiplexAnalogIn.schelp b/HelpSource/Classes/MultiplexAnalogIn.schelp new file mode 100644 index 00000000000..0ed49fc4095 --- /dev/null +++ b/HelpSource/Classes/MultiplexAnalogIn.schelp @@ -0,0 +1,58 @@ +class:: MultiplexAnalogIn +summary:: Read data from an analog input of the BELA board +related:: Classes/AnalogIn, Classes/AnalogOut, Classes/DigitalIn, Classes/DigitalOut, Classes/DigitalIO +categories:: UGens>BELA + + +Description:: + +Reads analog data from a multiplexed analog input of the BELA board, with the additional Multiplexer board. + +note:: +This UGen only works on BeLa +:: + +classmethods:: + +method::ar + +argument::analogPin + +Analog pin number to read. Pin numbers begin at 0. This value can be modulated at audiorate. + +argument::muxChannel + +Multiplex channel to read. Pin numbers begin at 0. This value can be modulated at audiorate. + +argument::mul + +argument::add + +method::kr + +argument::analogPin + +Analog pin number to read. Pin numbers begin at 0. + +argument::muxChannel + +Multiplex channel to read. Pin numbers begin at 0. + +argument::mul + +argument::add + +Examples:: + +code:: +// modulate frequency of a sine oscillator + +( +SynthDef("help-MultiplexAnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( MultiplexAnalogIn.ar( 0, 1 ).exprange( 200, 5000 ), 0, 0.1 ) + ) +}).play; +) +:: + diff --git a/README_BELA.md b/README_BELA.md new file mode 100644 index 00000000000..17cedb40eca --- /dev/null +++ b/README_BELA.md @@ -0,0 +1,359 @@ +This code will build on Bela image 3.0 and above (requires libbela and xenomai 3). +====================================== + +See [SuperCollider-on-Bela](https://github.com/BelaPlatform/Bela/wiki/SuperCollider-on-Bela) for general information on SuperCollider on Bela. + +Compiling SuperCollider scsynth on Bela +======================================= + +See [README.md](README.md) for the main SuperCollider readme. + +This file is Dan's, Marije's and Giulio's notes about compiling SC on [Bela](http://bela.io) platform. + +This branch contains that plus other modifications to get the SC source code master branch building. +The main addition in this branch is a **Xenomai/Bela audio driver for scsynth**, to use Bela's ultra-low-latency audio thread *instead* of jack/portaudio, and **plugins to access the analog and digital channels of the Bela-cape** + +> *NOTE:* This guide assumes you have the [Bela image v0.3.0](https://github.com/BelaPlatform/bela-image-builder/releases/tag/v0.3.0) or later. + +> *NOTE:* You need to get the latest version of the Bela code in order for Supercollider to compile. + +All of the commands here are to be executed *on the Bela device itself*. Normally you would SSH to it from a computer connected by USB, in order to do the following stuff. + +Preparation +=========== + +Make sure the system time on the board is up to date. This is ensured by simply opening the Bela IDE in a web browser and loading the page, or from the console, if the board is connected to the internet: + + dpkg-reconfigure tzdata + ntpdate pool.ntp.org + date # make sure this gives the right result + +or from the console, if the board is NOT connected to the internet, running from the host something like + + ssh -tt -o StrictHostKeyChecking=no -o ConnectTimeout=5 root@192.168.7.2 "sudo date -s \"`date '+%Y%m%d %T %z'`\"" + +Get the source code +=================== + +My modified source code is in this git branch here. If your Bela is connected to the network you can grab it directly: + + cd ~/ + git clone --recursive -b bela https://github.com/BelaPlatform/supercollider.git + cd supercollider + +Otherwise, `git clone` it on your computer and `scp` it over. + +### On Bela image v0.2.x(Debian Wheezy): + +This version of Supercollider will not compile on this image. Please update your image: https://github.com/BelaPlatform/bela-image-builder/releases + +### On Bela image v0.3.x(Debian Stretch): + +Before we compile, here are two optional steps to make your workflow faster + +1. installing `ccache` makes repeated builds faster, if you have spare disk space for it. It's especially helpful if you're going to be changing the cmake build scripts. + +``` +apt-get install ccache # requires internet access +mkdir /root/.ccache +echo "cache_dir = '/extrabela/ccache'" >> ~/.ccache/ccache.conf +``` + +2. alternatively, use `distcc` to make all your builds faster by off-loading the actual compilation to your host computer. You need to: +* install a cross-compiler for gcc-6.3 or clang 3.9 on your host (e.g.: [this](http://files.bela.io/gcc/arm-bela-linux-gnueabihf.zip) or download clang 3.9 from the LLVM website for Mac or a `g++-6.3-arm-linux-gnueabihf` package for your Linux distro). +* then follow instructions here to setup a working distcc environment https://forum.bela.io/d/724-distcc-distributed-compilation-with-bela +* on the host, launch `distccd` with something like `distccd --verbose --no-detach --daemon --allow 192.168.7.2 --log-level error --log-file ~/distccd.log` (and then `tail ~/distccd.log` for errors) +* if you get an error during compilation where it cannot find some stdlib includes (e.g.: `#include `), it may well be that `cmake` is trying to force `clang` to use `libc++` (with `-stdlib=libc++`). You can override this by editing your distcc-... executable and adding `-stdlib=libstdc++` at the end of the `$@` line. +* then on the board run the following before the `cmake` commands below: + +``` +export DISTCC_HOSTS="192.168.7.1" +export CC="distcc-clang" # or other as appropriate, see forum post above +export CXX="distcc-clang++" # or other as appropriate, see forum post above +``` + +NOTE: make sure you don't pass `-march=native` to the compiler when using `distcc`, or it will compile natively. Therefore, make sure you do NOT pass `-DNATIVE=ON` to `cmake`, as per below + +Then here's how to build: + + mkdir ~/supercollider/build + cd ~/supercollider/build + +Several options here: + + # here's the command WITHOUT ccache + cmake .. -DCMAKE_C_COMPILER=clang -DCMAKE_CXX_COMPILER=clang++ -DNOVA_SIMD=ON -DSSE=OFF -DSSE2=OFF -DINSTALL_HELP=OFF -DNO_X11=ON -DSC_QT=OFF -DSC_IDE=OFF -DSC_EL=OFF -DSC_ED=OFF -DSC_VIM=OFF -DSC_HIDAPI=OFF -DSUPERNOVA=OFF -DNO_AVAHI=ON -DNATIVE=ON -DENABLE_TESTSUITE=OFF -DAUDIOAPI=bela + + # or here's the command WITH ccache + cmake .. -DCMAKE_C_COMPILER=/usr/lib/ccache/clang -DCMAKE_CXX_COMPILER=/usr/lib/ccache/clang-3.9 -DNOVA_SIMD=ON -DSSE=OFF -DSSE2=OFF -DNO_X11=ON -DINSTALL_HELP=OFF -DSC_QT=OFF -DSC_IDE=OFF -DSC_EL=OFF -DSC_ED=OFF -DSC_VIM=OFF -DSC_HIDAPI=OFF -DSUPERNOVA=OFF -DNO_AVAHI=ON -DNATIVE=ON -DENABLE_TESTSUITE=OFF -DAUDIOAPI=bela + + # or here's the command WITH distcc (it will infer the compilers from the `export CC CXX` above + cmake .. -DNOVA_SIMD=ON -DSSE=OFF -DSSE2=OFF -DNO_X11=ON -DINSTALL_HELP=OFF -DSC_QT=OFF -DSC_IDE=OFF -DSC_EL=OFF -DSC_ED=OFF -DSC_VIM=OFF -DSC_HIDAPI=OFF -DSUPERNOVA=OFF -DNO_AVAHI=ON -DENABLE_TESTSUITE=OFF -DAUDIOAPI=bela + make + +The `make` step will take a little while, about 30 minutes when using plain `gcc` or `ccache` (you can try `make -j2` or `make -j3` with `distcc`), more like 10 minutes when using `distcc`. It seems it is stuck for a long time at compiling the `BinaryOpUGens.cpp`, but it will get past that. + +Next we install: + + make install + +### Cross-compiling + +On a Linux machine: + +- you need a built bela-image-builder, or just copy the whole filesystem from the baord you want to build for +- download a cross compiler, e.g., for Bela Images v0.3.x : http://releases.linaro.org/components/toolchain/binaries/6.3-2017.05/arm-linux-gnueabihf/gcc-linaro-6.3.1-2017.05-x86_64_arm-linux-gnueabihf.tar.xz +- create a file with the cross-toolchain details, e.g.: `~/Toolchain-arm-linux-gnueabihf.cmake` which contains something like this (edit the `SYSROOT` and `GCC_BASE` variables) + +```` +SET(CMAKE_SYSTEM_NAME Linux) +SET(CMAKE_SYSTEM_VERSION 1) + +set(CMAKE_SYSTEM_PROCESSOR arm ) +SET(CMAKE_LIBRARY_ARCHITECTURE arm-linux-gnueabihf) + +# where is the target environment +SET(SYSROOT "$ENV{HOME}/bela-image-builder-stretch/rootfs/") + +# RPATH - a list of directories which is linked into the executable, +# supported on most UNIX systems. It is ignored if RUNPATH is present. +set(FLAGS "${FLAGS} -Wl,-rpath-link,${SYSROOT}/lib/arm-linux-gnueabihf") +set(FLAGS "${FLAGS} -Wl,-rpath-link,${SYSROOT}/usr/lib/arm-linux-gnueabihf") +set(FLAGS "${FLAGS} -Wl,-rpath-link,${SYSROOT}/usr/local/lib") +set(RPATH_FLAGS ${FLAGS}) + +# specify the cross compiler +set(GCC_BASE $ENV{HOME}/gcc-linaro-6.3.1-2017.05-x86_64_arm-linux-gnueabihf/bin/arm-linux-gnueabihf-) +SET(CMAKE_C_COMPILER ${GCC_BASE}gcc) +SET(CMAKE_CXX_COMPILER ${GCC_BASE}g++) +SET(CMAKE_ASM_COMPILER ${GCC_BASE}as) +SET(CMAKE_RANLIB ${CROSS_COMPILER}ranlib) +UNSET(CMAKE_C_FLAGS CACHE) +UNSET(CMAKE_CXX_FLAGS CACHE) + +#link_libraries("-no-pie") +set(COMMON_FLAGS "-no-pie -fno-pie") +set(CMAKE_EXE_LINKER_FLAGS "-no-pie") +set(CMAKE_SHARED_LINKER_FLAGS "-no-pie") +set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${COMMON_FLAGS} ${RPATH_FLAGS}" CACHE STRING "c++ flags" FORCE) +set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${COMMON_FLAGS} ${RPATH_FLAGS}" CACHE STRING "c flags" FORCE) + +set(CMAKE_C_LINK_FLAGS "${CMAKE_C_LINK_FLAGS}" CACHE INTERNAL "c link flags" FORCE) +set(CMAKE_CXX_LINK_FLAGS "${CMAKE_CXX_LINK_FLAGS}" CACHE INTERNAL "c++ link flags" FORCE) + +SET(CMAKE_SYSROOT ${SYSROOT}) +SET(CMAKE_FIND_ROOT_PATH ${SYSROOT}) + +# search for programs in the build host directories +SET(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER) # for xeno-config an bela-config we have to manually override this there +# for libraries and headers in the target directories +SET(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY) +SET(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY) +``` +- create a couple of symlinks to make things easier: +``` +ln -s ~/bela-image-builder-stretch/rootfs/usr/xenomai /usr/xenomai +ln -s ~/bela-image-builder-stretch/rootfs/lib/arm-linux-gnueabihf/ /lib/arm-linux-gnueabihf +``` +- configure: +``` +cd supercollider +mkdir build && cd build +cmake .. -DCMAKE_TOOLCHAIN_FILE=~/Toolchain-arm-linux-gnueabihf.cmake -DNOVA_SIMD=ON -DSSE=OFF -DSSE2=OFF -DINSTALL_HELP=OFF -DNO_X11=ON -DSC_QT=OFF -DSC_IDE=OFF -DSC_EL=OFF -DSC_ED=OFF -DSC_VIM=OFF -DSC_HIDAPI=OFF -DSUPERNOVA=OFF -DNO_AVAHI=ON -DENABLE_TESTSUITE=OFF -DAUDIOAPI=bela -DSC_ABLETON_LINK=ON -DCMAKE_INSTALL_PREFIX=/usr/ +``` +- build: +``` +make -j20 +``` +- create a debian package to be installed on the board: +``` +cpack -G DEB -D CPACK_PACKAGE_CONTACT="Your Name " -D CPACK_DEBIAN_PACKAGE_ARCHITECTURE="armhf" -D CPACK_CMAKE_GENERATOR=Ninja +``` +(the `CPACK_CMAKE_GENERATOR=Ninja` is a trick to prevent the `preinstall` target from rebuilding the whole thing slower (see [here](https://stackoverflow.com/a/57530945/2958741)). + +Running it +========== + +Just run the executable like this: + + scsynth -u 57110 -z 16 + +The `-u` flag tells it which UDP port to listen on, and with the `-z` flag we choose scsynth's internal blocksize. We need to do this because scsynth's default internal buffer size (64) is bigger than the hardware buffer size (16), so dividing hardware by internal returned 0 buffers per callback. To make it run, you need to add the command-line argument "-z 16" (or presumably make the hardware buffer size bigger). + +So now you should have scsynth running on the device. You should be able to send OSC commands to it from SuperCollider running on your main computer: + + // These commands are to be run in SUPERCOLLIDER running on your MAIN computer. (I guess you could run them on the device too if you wanted.) + Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); + s.initTree; + s.startAliveThread; + SynthDef("funsound", { Out.ar(0, 0.5 * Pan2.ar(SinOsc.ar(LFNoise1.kr(2).exprange(100, 1000)), LFNoise1.kr(2))) }).add; + x = Synth("funsound"); + SynthDef("bish", { Out.ar(0, PinkNoise.ar * EnvGen.ar(Env.perc, Impulse.kr(2))) }).add; + y = Synth("bish"); + + // then when you want to stop the sounds: + x.free; + y.free; + + // You could use this to test mic input - be careful of feedback! + SynthDef("mic", { Out.ar(0, SoundIn.ar([0,1])) }).add; + z = Synth("mic"); + z.free; + +BELA I/O's +========== + +I/O support for the Bela is implemented. + +The startup flag ```-J``` defines how many analog input channels will be enabled, the startup flag ```-K``` how many analog output channels will be enabled, the startup flag ```-G``` how many digital channels will be enabled; by default all are set to 0. + +So for all analog and digital channels to be enabled run scsynth like this: + + scsynth -u 57110 -z 16 -J 8 -K 8 -G 16 + +To use the analog channels all as audio I/O + + scsynth -u 57110 -z 16 -J 8 -K 8 -G 16 -i 10 -o 10 + +This will start scsynth with 10 inputs and outputs, inputs/outputs 2 - 9 are the analog pins + +To use the analog channels all via the UGens only: + + scsynth -u 57110 -z 16 -J 8 -K 8 -G 16 -i 2 -o 2 + +This will start scsynth with 2 audio inputs and outputs, the analog I/O will only be accessible through UGens, but are all enabled. + +If you want higher sample rates of the analog I/O, you can set the number of channels to 4; the number of available channels is then 4. + + scsynth -u 57110 -z 16 -J 4 -K 4 -G 16 -i 2 -o 2 + +The amount of analog inputs and outputs actually used will be rounded to a multiple of 4, so the actual options are 0, 4 or 8 analog channels. This is because in SuperCollider we cannot sample the analog channels faster than audio rate (right now). + +![Channel explanation](bela_analog_audio_io.png) + +The ```ServerOptions``` class has appropriate variables to set the command line arguments, so you can set them with (but also see the comment below): + + s.options.numAnalogInChannels = 8; + s.options.numAnalogOutChannels = 8; + s.options.numDigitalChannels = 16; + + +The UGens ```AnalogIn```, ```AnalogOut```, ```DigitalIn```, ```DigitalOut```, ```DigitalIO``` give access to the pins; they all have helpfiles with examples of usage. + + +Examples +====================================================== + +Example files are available in the folder ```examples/bela```, and will be installed to ```/usr/local/share/SuperCollider/examples/bela```. + + +Running scsynth *and* sclang +====================================================== + +You can start the server as normal from the language. To set the settings for the analog I/O you should set them to some reasonable values. The defaults are to not pass the flags to scsynth. + + s = Server.default; + + s.options.numAnalogInChannels = 8; + s.options.numAnalogOutChannels = 8; + s.options.numDigitalChannels = 16; + + s.options.blockSize = 16; + s.options.numInputBusChannels = 2; + s.options.numOutputBusChannels = 2; + + s.waitForBoot({ + "THE SERVER IS BOOTED! Start of my actually interesting code".postln; + }); + +Alternatively, you can start scsynth manually, and then connect to it from a separate instance of sclang. + +So make one connection and start scsynth: + + scsynth -u 57110 -z 16 -J 8 -K 8 -G 16 -i 2 -o 2 + +And another to start sclang: + + sclang examples/bela/bela_example_analogin_2.scd + +Options Overview +---------------------------- + +Here is a breakdown of the options for running *scsynth* and how to set them up with either *scsynth* or *sclang* + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
paramscsynthsclang
audio computation block size-z # s.options.blockSize = 16;
number analog input channels enabled [0, 4, 8]-J #s.options.numAnalogInChannels = 0;
number analog output channels enabled [0, 4, 8]-K #s.options.numAnalogOutChannels = 0;
number digital channels enabled-G #s.options.numDigitalChannels = 16;
number of input buffer channels-i #s.options.numInputBusChannels = 2;
number of output buffer channels-o #s.options.numOutputBusChannels = 2;
+ + +Monitoring its performance +====================================================== +Here's a tip on how to check CPU load and "mode switches" for the running program, to make sure it's running properly in realtime etc. (A "mode switch" is something to be avoided: it means the code is dropping out of the Xenomai real-time execution and into normal Linux mode, which can be caused by certain operations in the realtime thread.) + + watch -n 0.5 cat /proc/xenomai/sched/stat + +which produces output like: + +
+Every 0.5s: cat /proc/xenomai/sched/stat                                                                                                               bela: Fri Jan 11 00:39:12 2019
+
+CPU  PID    MSW        CSW        XSC        PF    STAT       %CPU  NAME
+  0  0      0          835043     0          0     00018000   79.0  [ROOT]
+  0  14390  5          5          14         1     000480c0    0.0  scsynth
+  0  14404  34655      69562      69517      0     00048042    4.4  mAudioSyncSignalTask
+  0  14405  1          69782      104649     0     00048046   14.3  bela-audio
+  0  0      0          744555     0          0     00000000    0.8  [IRQ16: [timer]]
+  0  0      0          34899      0          0     00000000    1.0  [IRQ181: rtdm_pruss_irq_irq]
+
+ +the "MSW" column indicates mode switches; this number should NEVER increase in the bela-audio thread. It is fine if it increases on a task that runs occasionally, but keep in mind that each mode switch carries an additional overhead. + +Optional: Bonus level: Even more plugins (sc3-plugins) +====================================================== + +SuperCollider comes with a built-in set of UGen plugins but there's an extra set in the community **sc3-plugins** project. So if you want, you can also install those: + + cd /extrabela + git clone --recursive https://github.com/supercollider/sc3-plugins.git + cd sc3-plugins + mkdir build + cd build + cmake -DSC_PATH=~/supercollider -DCMAKE_C_COMPILER=/usr/lib/ccache/gcc-4.8 -DCMAKE_CXX_COMPILER=/usr/lib/ccache/g++-4.8 -DCMAKE_C_FLAGS="-march=armv7-a -mtune=cortex-a8 -mfloat-abi=hard -mfpu=neon -O2" -DCMAKE_CPP_FLAGS="-march=armv7-a -mtune=cortex-a8 -mfloat-abi=hard -mfpu=neon -O2" .. + make + make install + +These are basically just the instructions from the README of [the sc3-plugins project](https://github.com/supercollider/sc3-plugins/). diff --git a/SCClassLibrary/Common/Audio/bela/BELAUGens.sc b/SCClassLibrary/Common/Audio/bela/BELAUGens.sc new file mode 100644 index 00000000000..79bd9fa7629 --- /dev/null +++ b/SCClassLibrary/Common/Audio/bela/BELAUGens.sc @@ -0,0 +1,94 @@ +/* + * BELAUGens to access the analog and digital I/O + * created by nescivi, (c) 2016 + * https://www.nescivi.eu + */ + +/* input: id of analog pin to read; can be modulated at audiorate + * output: value of analog analogPin + */ +MultiplexAnalogIn : UGen { + signalRange { ^\unipolar } + + *ar { arg analogPin = 0, muxChannel=0, mul=1.0, add=0.0; + ^this.multiNew('audio', analogPin, muxChannel ).madd(mul,add) + } + *kr { arg analogPin = 0, muxChannel=0, mul=1.0, add=0.0; + ^this.multiNew('control', analogPin, muxChannel ).madd(mul,add) + } +} + + +/* input: id of analog pin to read; can be modulated at audiorate + * output: value of analog analogPin + */ +AnalogIn : UGen { + signalRange { ^\unipolar } + + *ar { arg analogPin = 0, mul=1.0, add=0.0; + ^this.multiNew('audio', analogPin ).madd(mul,add) + } + *kr { arg analogPin = 0, mul=1.0, add=0.0; + ^this.multiNew('control', analogPin ).madd(mul,add) + } +} + +/* input 1: id of analog pin to read; can be modulated at audiorate + * input 2: value to write out + * output: none + */ +AnalogOut : UGen { + *ar { arg analogPin = 0, output=0, mul=1.0, add=0.0; + ^this.multiNew('audio', analogPin, output ).madd(mul,add) + } + *kr { arg analogPin = 0, output=0, mul=1.0, add=0.0; + ^this.multiNew('control', analogPin, output ).madd(mul,add) + } + numOutputs { ^0 } + writeOutputSpecs {} +} + +/* input: id of digital pin to read; cannot be modulated + * output: value of digital pin + */ +DigitalIn : UGen { + signalRange { ^\unipolar } + + *ar { arg digitalPin = 0, mul=1.0, add=0.0; + ^this.multiNew('audio', digitalPin ).madd(mul,add) + } + *kr { arg digitalPin = 0, mul=1.0, add=0.0; + ^this.multiNew('control', digitalPin ).madd(mul,add) + } +} + +/* input 1: id of digital pin to read; cannot be modulated + * input 2: value to write out + * output: none + */ +DigitalOut : UGen { + *ar { arg digitalPin = 0, output=0, writeMode=0, mul=1.0, add=0.0; + ^this.multiNew('audio', digitalPin, output, writeMode ).madd(mul,add) + } + *kr { arg digitalPin = 0, output=0, writeMode=0, mul=1.0, add=0.0; + ^this.multiNew('control', digitalPin, output, writeMode ).madd(mul,add) + } + numOutputs { ^0 } + writeOutputSpecs {} +} + +/* input 1: id of digital pin to read; cannot be modulated + * input 2: value to write out + * input 3: pin mode ( < 0.5 = input, otherwise output) + * output: value of digital pin (last read value) + */ +DigitalIO : UGen { + signalRange { ^\unipolar } + + *ar { arg digitalPin = 0, output=0, pinMode=0, mul=1.0, add=0.0; + ^this.multiNew('audio', digitalPin, output, pinMode ).madd(mul,add) + } + *kr { arg digitalPin = 0, output=0, pinMode=0, mul=1.0, add=0.0; + ^this.multiNew('control', digitalPin, output, pinMode ).madd(mul,add) + } +} diff --git a/SCClassLibrary/Common/Control/Server.sc b/SCClassLibrary/Common/Control/Server.sc index ed3ec6b7dd1..3cf4ca868ba 100644 --- a/SCClassLibrary/Common/Control/Server.sc +++ b/SCClassLibrary/Common/Control/Server.sc @@ -58,6 +58,19 @@ ServerOptions { var <>safetyClipThreshold; + // extension for BELA + var <>numAnalogInChannels; + var <>numAnalogOutChannels; + var <>numDigitalChannels; + var <>headphoneLevel; + var <>pgaGainLeft; + var <>pgaGainRight; + var <>speakerMuted; + var <>dacLevel; + var <>adcLevel; + var <>numMultiplexChannels; + var <>belaPRU; + *initClass { defaultValues = IdentityDictionary.newFrom( ( @@ -101,6 +114,17 @@ ServerOptions { recBufSize: nil, bindAddress: "127.0.0.1", safetyClipThreshold: 1.26 // ca. 2 dB + numAnalogInChannels: 2, + numAnalogOutChannels: 2, + numDigitalChannels: 16, + headphoneLevel: -6, + pgaGainLeft: 10, + pgaGainRight: 10, + speakerMuted: 0, + dacLevel: 0, + adcLevel: 0, + numMultiplexChannels: 0, + belaPRU: 1, ) ) } @@ -229,6 +253,40 @@ ServerOptions { }); if (thisProcess.platform.name === \osx && Server.program.asString.endsWith("supernova").not && safetyClipThreshold.notNil, { o = o ++ " -s " ++ safetyClipThreshold; + + // additions for BELA + if (numAnalogInChannels.notNil, { + o = o ++ " -J " ++ numAnalogInChannels; + }); + if (numAnalogOutChannels.notNil, { + o = o ++ " -K " ++ numAnalogOutChannels; + }); + if (numDigitalChannels.notNil, { + o = o ++ " -G " ++ numDigitalChannels; + }); + if (headphoneLevel.notNil, { + o = o ++ " -Q " ++ headphoneLevel; + }); + if (pgaGainLeft.notNil, { + o = o ++ " -X " ++ pgaGainLeft; + }); + if (pgaGainRight.notNil, { + o = o ++ " -Y " ++ pgaGainRight; + }); + if (speakerMuted.notNil, { + o = o ++ " -s " ++ speakerMuted; + }); + if (dacLevel.notNil, { + o = o ++ " -x " ++ dacLevel; + }); + if (adcLevel.notNil, { + o = o ++ " -y " ++ adcLevel; + }); + if (numMultiplexChannels.notNil, { + o = o ++ " -g " ++ numMultiplexChannels; + }); + if (belaPRU.notNil, { + o = o ++ " -T " ++ belaPRU; }); ^o } diff --git a/bela_analog_audio_io.png b/bela_analog_audio_io.png new file mode 100644 index 0000000000000000000000000000000000000000..88befd3e40f9f6f656d710bb8c1a3d67440341cb GIT binary patch literal 22655 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%0AL}}YQ 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 literal 0 HcmV?d00001 diff --git a/cmake_modules/FindBela.cmake b/cmake_modules/FindBela.cmake new file mode 100644 index 00000000000..2581e815937 --- /dev/null +++ b/cmake_modules/FindBela.cmake @@ -0,0 +1,79 @@ +# - Try to find Bela (BeagleRT) +# Once done this will define +# +# BELA_FOUND - system has bela +# BELA_INCLUDE_DIRS - the bela include directory +# BELA_SOURCES - the bela source files to compile in +# BELA_LIBRARIES - Link these too please +# BELA_DEFINITIONS - Compiler switches required for using bela +# +# Copyright (c) 2008 Andreas Schneider +# Modified for other libraries by Lasse Kärkkäinen +# +# Redistribution and use is allowed according to the terms of the New +# BSD license. +# For details see the accompanying COPYING-CMAKE-SCRIPTS file. +# + +if (BELA_CFLAGS AND BELA_CXXFLAGS AND BELA_LDFLAGS) + # in cache already + set(BELA_FOUND TRUE) +else (BELA_CFLAGS AND BELA_CXXFLAGS AND BELA_LDFLAGS) + # Bela comes with its own ...-config program to get configuration flags + if (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) + # if cross compiling, we want to find this program only from the sysroot + set(CACHED ${CMAKE_FIND_ROOT_PATH_MODE_PROGRAM}) + set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM ONLY) + endif (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) + find_program(BELA_CONFIG + NAMES + bela-config + PATHS + /root/Bela/resources/bin + $ENV{BELA_ROOT}/resources/bin + /usr/local/bin + /usr/bin + ) + if (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) + # restore the previous value + set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM ${CACHED}) + endif (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) +message("Searching for BELA man: config ${BELA_CONFIG}") + + if (BELA_CONFIG) + execute_process(COMMAND ${BELA_CONFIG} --defines OUTPUT_VARIABLE BELA_DEFINITIONS) + string(STRIP "${BELA_DEFINITIONS}" BELA_DEFINITIONS) + execute_process(COMMAND ${BELA_CONFIG} --includes OUTPUT_VARIABLE BELA_INCLUDE_DIRS) + string(STRIP "${BELA_INCLUDE_DIRS}" BELA_INCLUDE_DIRS) + execute_process(COMMAND ${BELA_CONFIG} --libraries OUTPUT_VARIABLE BELA_LIBRARIES) + string(STRIP "${BELA_LIBRARIES}" BELA_LIBRARIES) + execute_process(COMMAND ${BELA_CONFIG} --cflags OUTPUT_VARIABLE BELA_C_FLAGS) + string(STRIP "${BELA_C_FLAGS}" BELA_C_FLAGS) + execute_process(COMMAND ${BELA_CONFIG} --cxxflags OUTPUT_VARIABLE BELA_CXX_FLAGS) + string(STRIP "${BELA_CXX_FLAGS}" BELA_CXX_FLAGS) + SET(BELA_CXX_FLAGS "${BELA_CXX_FLAGS} -DBELA_DONT_INCLUDE_UTILITIES") + endif (BELA_CONFIG) + + if (BELA_CONFIG) + set(BELA_FOUND TRUE) + endif (BELA_CONFIG) + + if (BELA_FOUND) + if (NOT BELA_FIND_QUIETLY) + execute_process(COMMAND ${BELA_CONFIG} --prefix OUTPUT_VARIABLE BELA_PREFIX) + message(STATUS "Found Bela: ${BELA_PREFIX}") + message(STATUS "BELA_DEFINITIONS: ${BELA_DEFINITIONS}") + message(STATUS "BELA_INCLUDE_DIRS: ${BELA_INCLUDE_DIRS}") + message(STATUS "BELA_LIBRARIES: ${BELA_LIBRARIES}") + message(STATUS "BELA_C_FLAGS: ${BELA_C_FLAGS}") + message(STATUS "BELA_CXX_FLAGS: ${BELA_CXX_FLAGS}") + endif (NOT BELA_FIND_QUIETLY) + else (BELA_FOUND) + if (BELA_FIND_REQUIRED) + message(FATAL_ERROR "Could not find BELA") + endif (BELA_FIND_REQUIRED) + endif (BELA_FOUND) +# show the BELA_ variables only in the advanced view +mark_as_advanced(BELA_CFLAGS BELA_CXXFLAGS BELA_LDFLAGS) + +endif (BELA_CFLAGS AND BELA_CXXFLAGS AND BELA_LDFLAGS) diff --git a/cmake_modules/FindXenomai.cmake b/cmake_modules/FindXenomai.cmake new file mode 100644 index 00000000000..89c03f9dab4 --- /dev/null +++ b/cmake_modules/FindXenomai.cmake @@ -0,0 +1,156 @@ +# - Try to find xenomai +# Once done this will define +# +# XENOMAI_FOUND - system has xenomai +# XENOMAI_INCLUDE_DIRS - the xenomai include directory +# XENOMAI_LIBRARIES - Link these to use xenomai +# XENOMAI_DEFINITIONS - Compiler switches required for using xenomai +# +# Copyright (c) 2008 Andreas Schneider +# Modified for other libraries by Lasse Kärkkäinen +# +# Redistribution and use is allowed according to the terms of the New +# BSD license. +# For details see the accompanying COPYING-CMAKE-SCRIPTS file. +# + +# https://cmake.org/Wiki/CMakeMacroParseArguments + +MACRO(PARSE_ARGUMENTS prefix arg_names option_names) + SET(DEFAULT_ARGS) + FOREACH(arg_name ${arg_names}) + SET(${prefix}_${arg_name}) + ENDFOREACH(arg_name) + FOREACH(option ${option_names}) + SET(${prefix}_${option} FALSE) + ENDFOREACH(option) + + SET(current_arg_name DEFAULT_ARGS) + SET(current_arg_list) + FOREACH(arg ${ARGN}) + SET(larg_names ${arg_names}) + LIST(FIND larg_names "${arg}" is_arg_name) + IF (is_arg_name GREATER -1) + SET(${prefix}_${current_arg_name} ${current_arg_list}) + SET(current_arg_name ${arg}) + SET(current_arg_list) + ELSE (is_arg_name GREATER -1) + SET(loption_names ${option_names}) + LIST(FIND loption_names "${arg}" is_option) + IF (is_option GREATER -1) + SET(${prefix}_${arg} TRUE) + ELSE (is_option GREATER -1) + SET(current_arg_list ${current_arg_list} ${arg}) + ENDIF (is_option GREATER -1) + ENDIF (is_arg_name GREATER -1) + ENDFOREACH(arg) + SET(${prefix}_${current_arg_name} ${current_arg_list}) +ENDMACRO(PARSE_ARGUMENTS) + +# LIST_FILTER( [ ...] +# [OUTPUT_VARIABLE ]) +# Removes items from which do not match any of the specified +# regular expressions. An optional argument OUTPUT_VARIABLE +# specifies a variable in which to store the matched items instead of +# updating +# As regular expressions can not be given to macros (see bug #5389), we pass +# variable names whose content is the regular expressions. +# Note that this macro requires PARSE_ARGUMENTS macro, available here: +# http://www.cmake.org/Wiki/CMakeMacroParseArguments +MACRO(LIST_FILTER) + PARSE_ARGUMENTS(LIST_FILTER "OUTPUT_VARIABLE" "" ${ARGV}) + # Check arguments. + LIST(LENGTH LIST_FILTER_DEFAULT_ARGS LIST_FILTER_default_length) + IF(${LIST_FILTER_default_length} EQUAL 0) + MESSAGE(FATAL_ERROR "LIST_FILTER: missing list variable.") + ENDIF(${LIST_FILTER_default_length} EQUAL 0) + IF(${LIST_FILTER_default_length} EQUAL 1) + MESSAGE(FATAL_ERROR "LIST_FILTER: missing regular expression variable.") + ENDIF(${LIST_FILTER_default_length} EQUAL 1) + # Reset output variable + IF(NOT LIST_FILTER_OUTPUT_VARIABLE) + SET(LIST_FILTER_OUTPUT_VARIABLE "LIST_FILTER_internal_output") + ENDIF(NOT LIST_FILTER_OUTPUT_VARIABLE) + SET(${LIST_FILTER_OUTPUT_VARIABLE}) + # Extract input list from arguments + LIST(GET LIST_FILTER_DEFAULT_ARGS 0 LIST_FILTER_input_list) + LIST(REMOVE_AT LIST_FILTER_DEFAULT_ARGS 0) + FOREACH(LIST_FILTER_item ${${LIST_FILTER_input_list}}) + FOREACH(LIST_FILTER_regexp_var ${LIST_FILTER_DEFAULT_ARGS}) + FOREACH(LIST_FILTER_regexp ${${LIST_FILTER_regexp_var}}) + IF(${LIST_FILTER_item} MATCHES ${LIST_FILTER_regexp}) + LIST(APPEND ${LIST_FILTER_OUTPUT_VARIABLE} ${LIST_FILTER_item}) + ENDIF(${LIST_FILTER_item} MATCHES ${LIST_FILTER_regexp}) + ENDFOREACH(LIST_FILTER_regexp ${${LIST_FILTER_regexp_var}}) + ENDFOREACH(LIST_FILTER_regexp_var) + ENDFOREACH(LIST_FILTER_item) + # If OUTPUT_VARIABLE is not specified, overwrite the input list. + IF(${LIST_FILTER_OUTPUT_VARIABLE} STREQUAL "LIST_FILTER_internal_output") + SET(${LIST_FILTER_input_list} ${${LIST_FILTER_OUTPUT_VARIABLE}}) + ENDIF(${LIST_FILTER_OUTPUT_VARIABLE} STREQUAL "LIST_FILTER_internal_output") +ENDMACRO(LIST_FILTER) + + +if (XENOMAI_LIRARIES AND XENOMAI_INCLUDE_DIRS AND XENOMAI_DEFINITIONS) + # in cache already + set(XENOMAI_FOUND TRUE) +else (XENOMAI_LIRARIES AND XENOMAI_INCLUDE_DIRS AND XENOMAI_DEFINITIONS) + # Xenomai comes with its own ...-config program to get cflags and ldflags + if (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) + # if cross compiling, we want to find this program only from the sysroot + set(CACHED ${CMAKE_FIND_ROOT_PATH_MODE_PROGRAM}) + set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM ONLY) + endif (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) + find_program(XENOMAI_XENO_CONFIG + NAMES + xeno-config + PATHS + ${_XENOMAI_INCLUDEDIR} + /usr/xenomai/bin + /usr/local/bin + /usr/bin + ) + if (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) + # restore the previous value + set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM ${CACHED}) + endif (CMAKE_FIND_ROOT_PATH_MODE_PROGRAM) + + if (XENOMAI_XENO_CONFIG) + set(XENOMAI_FOUND TRUE) + execute_process(COMMAND ${XENOMAI_XENO_CONFIG} --skin=posix --cflags OUTPUT_VARIABLE XENOMAI_CFLAGS) + string(STRIP "${XENOMAI_CFLAGS}" XENOMAI_CFLAGS) + # use grep to separate out defines and includes + execute_process( + COMMAND bash -c "A= ; for a in ${XENOMAI_CFLAGS}; do echo $a | grep -q \"\\-D.*\" && A=\"$A $a\"; done; echo $A" + OUTPUT_VARIABLE XENOMAI_DEFINITIONS) + string(STRIP "${XENOMAI_DEFINITIONS}" XENOMAI_DEFINITIONS) + + execute_process( + COMMAND bash -c "A= ; for a in ${XENOMAI_CFLAGS}; do echo $a | grep -q \"\\-I.*\" && A=\"$A `echo $a|sed s/-I//`;\"; done; echo $A" + OUTPUT_VARIABLE XENOMAI_INCLUDE_DIRS) + string(STRIP "${XENOMAI_INCLUDE_DIRS}" XENOMAI_INCLUDE_DIRS) + + execute_process( + COMMAND ${XENOMAI_XENO_CONFIG} --skin=posix --ldflags --no-auto-init + COMMAND sed "s/-Wl,@.*wrappers//g" + OUTPUT_VARIABLE XENOMAI_LIBRARIES) + string(STRIP "${XENOMAI_LIBRARIES}" XENOMAI_LIBRARIES) + endif (XENOMAI_XENO_CONFIG) + + if (XENOMAI_FOUND) + if (NOT XENOMAI_FIND_QUIETLY) + execute_process(COMMAND ${XENOMAI_XENO_CONFIG} --prefix OUTPUT_VARIABLE XENOMAI_PREFIX) + message(STATUS "Found xenomai: ${XENOMAI_PREFIX}") + message(STATUS "XENOMAI_LIBRARIES: ${XENOMAI_LIBRARIES}") + message(STATUS "XENOMAI_INCLUDE_DIRS: ${XENOMAI_INCLUDE_DIRS}") + message(STATUS "XENOMAI_DEFINITIONS: ${XENOMAI_DEFINITIONS}") + endif (NOT XENOMAI_FIND_QUIETLY) + else (XENOMAI_FOUND) + if (XENOMAI_FIND_REQUIRED) + message(FATAL_ERROR "Could not find XENOMAI") + endif (XENOMAI_FIND_REQUIRED) + endif (XENOMAI_FOUND) +# show the XENOMAI_ variables only in the advanced view +mark_as_advanced(XENOMAI_LIRARIES XENOMAI_INCLUDE_DIRS XENOMAI_DEFINITIONS) + +endif (XENOMAI_LIRARIES AND XENOMAI_INCLUDE_DIRS AND XENOMAI_DEFINITIONS) diff --git a/examples/bela/bela_example_analogin.scd b/examples/bela/bela_example_analogin.scd new file mode 100644 index 00000000000..88960c18663 --- /dev/null +++ b/examples/bela/bela_example_analogin.scd @@ -0,0 +1,25 @@ +s = Server.default; + +s.options.numAnalogInChannels = 8; +s.options.numAnalogOutChannels = 8; +s.options.numDigitalChannels = 16; + +s.options.blockSize = 16; +s.options.numInputBusChannels = 2; +s.options.numOutputBusChannels = 2; + +s.options.postln; + +s.waitForBoot({ + // modulate frequency of a sine oscillator + ( + SynthDef("help-AnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( AnalogIn.ar( DC.ar( 0 ) ).exprange( 200, 5000 ), 0, 0.1 ) + ) + }).send(s); + ); + + s.sync; + Synth.new("help-AnalogIn", target: s); +}); diff --git a/examples/bela/bela_example_analogin_2.scd b/examples/bela/bela_example_analogin_2.scd new file mode 100644 index 00000000000..fb509ffb307 --- /dev/null +++ b/examples/bela/bela_example_analogin_2.scd @@ -0,0 +1,20 @@ +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + +fork{ + s.sync; + + ( + SynthDef("help-AnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( AnalogIn.ar( DC.ar( 0 ) ).exprange( 200, 5000 ), 0, 0.1 ) + ) + }).send(s); + ); + + s.sync; + Synth.new("help-AnalogIn", target: s).postln; +}; + +s.freeAll; \ No newline at end of file diff --git a/examples/bela/bela_example_analogout.scd b/examples/bela/bela_example_analogout.scd new file mode 100644 index 00000000000..afd727908ec --- /dev/null +++ b/examples/bela/bela_example_analogout.scd @@ -0,0 +1,16 @@ +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + +fork{ + s.sync; + + ( + SynthDef("help-AnalogOut",{ arg out=0; + AnalogOut.ar( DC.ar( 0 ), SinOsc.ar( 10, 0, 0.5, 0.5 ) ); + }).send(s); + ); + + s.sync; + Synth.new("help-AnalogOut", target: s).postln; +}; diff --git a/examples/bela/bela_example_digital.scd b/examples/bela/bela_example_digital.scd new file mode 100644 index 00000000000..72a0765f3fc --- /dev/null +++ b/examples/bela/bela_example_digital.scd @@ -0,0 +1,18 @@ +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + +fork{ + s.sync; + + ( + SynthDef("help-DigitalIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( 500, 0, 0.1 * DigitalIn.ar( 0 ) ) + ) + }).send(s); + ); + + s.sync; + Synth.new("help-DigitalIn", target: s).postln; +}; diff --git a/examples/bela/bela_example_digitalio.scd b/examples/bela/bela_example_digitalio.scd new file mode 100644 index 00000000000..c0231d5d85d --- /dev/null +++ b/examples/bela/bela_example_digitalio.scd @@ -0,0 +1,16 @@ +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + +fork{ + s.sync; + + ( + SynthDef("help-DigitalIO",{ arg out=0; + DigitalIO.ar( DC.ar( 0 ), SinOsc.ar( 10 ), K2A.ar( LFPulse.kr( 0.1 ) ) ).poll; + }).send(s); + ); + + s.sync; + Synth.new("help-DigitalIO", target: s).postln; +}; diff --git a/examples/bela/bela_example_digitalout.scd b/examples/bela/bela_example_digitalout.scd new file mode 100644 index 00000000000..522306fa389 --- /dev/null +++ b/examples/bela/bela_example_digitalout.scd @@ -0,0 +1,16 @@ +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + +fork{ + s.sync; + + ( + SynthDef("help-DigitalOut",{ arg out=0; + DigitalOut.ar( 0, SinOsc.ar( 10 ) ); + }).send(s); + ); + + s.sync; + Synth.new("help-DigitalOut", target: s).postln; +}; diff --git a/examples/bela/bela_start_scsynth.scd b/examples/bela/bela_start_scsynth.scd new file mode 100644 index 00000000000..16860ecdd3d --- /dev/null +++ b/examples/bela/bela_start_scsynth.scd @@ -0,0 +1,21 @@ +s = Server.default; + +s.options.numAnalogInChannels = 8; +s.options.numAnalogOutChannels = 8; +s.options.numDigitalChannels = 16; + +s.options.pgaGainLeft = 4; +s.options.pgaGainRight = 5; +s.options.headphoneLevel = -8; +s.options.speakerMuted = 0; +s.options.dacLevel = -5; +s.options.adcLevel = -3; +s.options.numMultiplexChannels = 0; + +s.options.blockSize = 16; +s.options.numInputBusChannels = 2; +s.options.numOutputBusChannels = 2; + +s.options.postln; + +s.boot; diff --git a/examples/bela/bela_start_scsynth_2.scd b/examples/bela/bela_start_scsynth_2.scd new file mode 100644 index 00000000000..aeb26d865de --- /dev/null +++ b/examples/bela/bela_start_scsynth_2.scd @@ -0,0 +1,22 @@ +s = Server.default; + +s.options.numAnalogInChannels = 8; +s.options.numAnalogOutChannels = 8; +s.options.numDigitalChannels = 16; + +s.options.pgaGainLeft = 4; +s.options.pgaGainRight = 5; +s.options.headphoneLevel = -8; +s.options.speakerMuted = 1; +s.options.dacLevel = -5; +s.options.adcLevel = -3; +s.options.numMultiplexChannels = 6; +s.options.belaPRU = 1; + +s.options.blockSize = 16; +s.options.numInputBusChannels = 2; +s.options.numOutputBusChannels = 2; + +s.options.postln; + +s.boot; diff --git a/examples/bela/bela_test_cases.scd b/examples/bela/bela_test_cases.scd new file mode 100644 index 00000000000..d1209ad7e8e --- /dev/null +++ b/examples/bela/bela_test_cases.scd @@ -0,0 +1,351 @@ +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + + +/// AnalogIn + +// ak +( +SynthDef("AnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( AnalogIn.ar( 0 ).poll.exprange( 200, 5000 ), 0, 0.1 ) + ) +}).send(s); +); + +a = Synth.new("AnalogIn", target: s).postln; +a.free; + +// ak +( +SynthDef("AnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( AnalogIn.ar( + Stepper.kr(Impulse.kr(1), 0, 0, 7, 1 ).poll + ).poll.exprange( 200, 5000 ), 0, 0.1 ) + ) +}).send(s); +); + +a = Synth.new("AnalogIn", target: s).postln; +a.free; + +// kk +( +SynthDef("AnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( AnalogIn.kr( + 0 + ).poll.exprange( 200, 5000 ), 0, 0.1 ) + ) +}).send(s); +); + +a = Synth.new("AnalogIn", target: s).postln; +a.free; + +// aa +( +SynthDef("AnalogIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( AnalogIn.ar( Stepper.ar(Impulse.ar(1), 0, 0, 7, 1 ).poll ).poll.exprange( 200, 5000 ), 0, 0.1 ) + ) +}).send(s); +); + +a = Synth.new("AnalogIn", target: s).postln; +a.free; + + +/// AnalogOut + +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + + +// aaa +( +SynthDef("AnalogOut",{ arg out=0; + AnalogOut.ar( DC.ar( 0 ), SinOsc.ar( 10, 0, 0.5, 0.5 ) ); +}).send(s); +); + +a = Synth.new("AnalogOut", target: s).postln; +a.free + +// aka +( +SynthDef("AnalogOut",{ arg out=0; + AnalogOut.ar( 0, SinOsc.ar( 10, 0, 0.5, 0.5 ) ); +}).send(s); +); + +a = Synth.new("AnalogOut", target: s).postln; +a.free; + +// aak +( +SynthDef("AnalogOut",{ arg out=0; + AnalogOut.ar( DC.ar( 0 ), SinOsc.kr( 10, 0, 0.5, 0.5 ) ); +}).send(s); +); + +a = Synth.new("AnalogOut", target: s).postln; +a.free + +// kk +( // diverted to kk - AnalogOut becomes .kr effectively +SynthDef("AnalogOut",{ arg out=0; + AnalogOut.ar( 0, SinOsc.kr( 10, 0, 0.5, 0.5 ) ); +}).send(s); +); + +a = Synth.new("AnalogOut", target: s).postln; +a.free; + + +// kk +( +SynthDef("AnalogOut",{ arg out=0; + AnalogOut.kr( 0, SinOsc.kr( 10, 0, 0.5, 0.5 ) ); +}).send(s); +); + +a = Synth.new("AnalogOut", target: s).postln; +a.free; + +// kk + warning +( +SynthDef("AnalogOut",{ arg out=0; + AnalogOut.kr( 0, SinOsc.ar( 10, 0, 0.5, 0.5 ) ); +}).send(s); +); + +a = Synth.new("AnalogOut", target: s).postln; +a.free; + + + +// DigitalIn + +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + + +// a +( +SynthDef("DigitalIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( 500, 0, 0.1 * DigitalIn.ar( 0 ) ) + ) +}).send(s); +); + +a = Synth.new("DigitalIn", target: s).postln; +a.free; + +// k +( +SynthDef("DigitalIn",{ arg out=0; + Out.ar(out, + SinOsc.ar( 500, 0, 0.1 * DigitalIn.kr( 0 ) ) + ) +}).send(s); +); + +a = Synth.new("DigitalIn", target: s).postln; +a.free; + + + +// DigitalOut + +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + +// a +( +SynthDef("DigitalOut",{ arg out=0; + DigitalOut.ar( 0, SinOsc.ar( 10 ) ); +}).send(s); +); + +a = Synth.new("DigitalOut", target: s).postln; +a.free; + +// k + warning +( +SynthDef("DigitalOut",{ arg out=0; + DigitalOut.kr( 0, SinOsc.ar( 10 ) ); +}).send(s); +); + +a = Synth.new("DigitalOut", target: s).postln; +a.free; + + +// k +( +SynthDef("DigitalOut",{ arg out=0; + DigitalOut.kr( 0, SinOsc.kr( 10 ) ); +}).send(s); +); + +a = Synth.new("DigitalOut", target: s).postln; +a.free; + +// k + warning +( +SynthDef("DigitalOut",{ arg out=0; + DigitalOut.ar( 0, SinOsc.kr( 10 ) ); +}).send(s); +); + +a = Synth.new("DigitalOut", target: s).postln; +a.free; + + +// a-once +( +SynthDef("DigitalOut",{ arg out=0; + DigitalOut.ar( 0, SinOsc.ar( 10 ), 1 ); +}).send(s); +); + +a = Synth.new("DigitalOut", target: s).postln; +a.free; + + + + +// DigitalIO + + +Server.default = s = Server("belaServer", NetAddr("192.168.7.2", 57110)); +s.initTree; +s.startAliveThread; + + +( // aaaa once - everything audio rate +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( DC.ar( 0 ), SinOsc.ar( 10 ), K2A.ar( LFPulse.kr( 0.1 ) ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + +( // aaak once - pinmode control rate +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( DC.ar( 0 ), SinOsc.ar( 10 ), LFPulse.kr( 0.1 ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + +( // aaka once - output value control rate +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( DC.ar( 0 ), SinOsc.kr( 10 ), K2A.ar( LFPulse.kr( 0.1 ) ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + +( // aakk once - output value control rate, pinmode control rate +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( DC.ar( 0 ), SinOsc.kr( 10 ), LFPulse.kr( 0.1 ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + +( // akaa once - pin changed at control rate, rest audio +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( 0, SinOsc.ar( 10 ), K2A.ar( LFPulse.kr( 0.1 ) ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + +( // akak once - pin changed at control rate, output value at audio rate, mode at control rate +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( 0, SinOsc.ar( 10 ), LFPulse.kr( 0.1 ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + +( // ak once - pin changed at control rate, output control rate, pinmode control +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( 0, SinOsc.kr( 10 ), LFPulse.kr( 0.1 ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + +( // akka once - pin changed at control rate, output control rate, pinmode control +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.ar( 0, SinOsc.kr( 10 ), K2A.ar( LFPulse.kr( 0.1 ) ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + + +( // kk - warning +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.kr( 0, SinOsc.kr( 10 ), K2A.ar( LFPulse.kr( 0.1 ) ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + +( // kk - warning +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.kr( 0, SinOsc.ar( 10 ), LFPulse.kr( 0.1 ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + +( // kk - warning +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.kr( Stepper.ar( Impulse.ar(1), 0, 7, 1 ), SinOsc.kr( 10 ), LFPulse.kr( 0.1 ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + + +( // kk +SynthDef("DigitalIO",{ arg out=0; + DigitalIO.kr( 0, SinOsc.kr( 10 ), LFPulse.kr( 0.1 ) ).poll; +}).send(s); +); + +a = Synth.new("DigitalIO", target: s).postln; +a.free; + +// +s.freeAll; \ No newline at end of file diff --git a/include/plugin_interface/SC_World.h b/include/plugin_interface/SC_World.h index c768385d78d..9d68c3d0096 100644 --- a/include/plugin_interface/SC_World.h +++ b/include/plugin_interface/SC_World.h @@ -21,6 +21,10 @@ #pragma once +#ifdef BELA +# include "Bela.h" +#endif + #include "SC_Types.h" #include "SC_Rate.h" #include "SC_SndBuf.h" @@ -103,6 +107,22 @@ struct World { nova::padded_rw_spinlock* mAudioBusLocks; nova::spin_lock* mControlBusLock; #endif + +#ifdef BELA + BelaContext* mBelaContext; + // uint32 mBelaAnalogChannels; + uint32 mBelaAnalogInputChannels; + uint32 mBelaAnalogOutputChannels; + uint32 mBelaDigitalChannels; + float mBelaHeadphoneLevel; + float mBelaPGAGainLeft; + float mBelaPGAGainRight; + bool mBelaSpeakerMuted; + float mBelaDACLevel; + float mBelaADCLevel; + uint32 mBelaNumMuxChannels; + uint32 mBelaPRU; +#endif }; inline SndBuf* World_GetBuf(struct World* inWorld, uint32 index) { diff --git a/include/server/SC_WorldOptions.h b/include/server/SC_WorldOptions.h index 024bab3941f..af1b1cbc9ae 100644 --- a/include/server/SC_WorldOptions.h +++ b/include/server/SC_WorldOptions.h @@ -84,6 +84,21 @@ struct WorldOptions { const char* mRestrictedPath = nullptr; int mSharedMemoryID = 0; + +#ifdef BELA + // uint32 mBelaAnalogChannels; + uint32 mBelaAnalogInputChannels; + uint32 mBelaAnalogOutputChannels; + uint32 mBelaDigitalChannels; + float mBelaHeadphoneLevel; + float mBelaPGAGainLeft; + float mBelaPGAGainRight; + bool mBelaSpeakerMuted; + float mBelaDACLevel; + float mBelaADCLevel; + uint32 mBelaNumMuxChannels; + uint32 mBelaPRU; +#endif }; struct SndBuf; diff --git a/lang/CMakeLists.txt b/lang/CMakeLists.txt index 56142d61704..6141e8b8e73 100644 --- a/lang/CMakeLists.txt +++ b/lang/CMakeLists.txt @@ -116,7 +116,7 @@ if(AUDIOAPI STREQUAL "default") endif(APPLE) endif() -if(NOT AUDIOAPI MATCHES "^(jack|coreaudio|portaudio)$") +if(NOT AUDIOAPI MATCHES "^(jack|coreaudio|portaudio|bela)$") message(FATAL_ERROR "Unrecognised audio API: ${AUDIOAPI}") endif() diff --git a/server/plugins/BELAUGens.cpp b/server/plugins/BELAUGens.cpp new file mode 100644 index 00000000000..235797d576b --- /dev/null +++ b/server/plugins/BELAUGens.cpp @@ -0,0 +1,1386 @@ +/* + SuperCollider real time audio synthesis system + Copyright (c) 2002 James McCartney. All rights reserved. + http://www.audiosynth.com + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA +*/ + +/* + * BELA I/O UGens created by nescivi, (c) 2016 + * https://www.nescivi.eu + */ + +// #include + +#include + +#include "Bela.h" +// These functions are provided by xenomai +int rt_printf(const char* format, ...); +int rt_fprintf(FILE* stream, const char* format, ...); + +#include "SC_PlugIn.h" + +static InterfaceTable* ft; + +struct MultiplexAnalogIn : public Unit { + // TODO: can we remove this ? +}; + + +struct AnalogIn : public Unit { + // TODO: can we remove this ? +}; + +struct AnalogOut : public Unit { + // TODO: can we remove this ? +}; + +// static digital pin, static function (in) +struct DigitalIn : public Unit { + int mDigitalPin; +}; + +// static digital pin, static function (out) - uses DigitalWrite and a check whether value changed +struct DigitalOut : public Unit { + int mDigitalPin; + int mLastOut; +}; + +// static digital pin, static function (out) - uses DigitalWriteOnce +struct DigitalOutA : public Unit { + int mDigitalPin; + int mLastOut; +}; + +// flexible digital pin, flexible function (in or out) +struct DigitalIO : public Unit { + int mLastDigitalIn; + int mLastDigitalOut; +}; + +/* +struct BelaScope : public Unit +{ +}; + +struct BelaScopeChannel : public Unit +{ + int mScopeChannel; +}; +*/ + +////////////////////////////////////////////////////////////////////////////////////////////////// + +void MultiplexAnalogIn_next_aaa(MultiplexAnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* fin = IN(0); // analog in pin, can be modulated + float* fmux = IN(1); // mux channel, can be modulated + float* out = ZOUT(0); + int analogPin = 0; + int muxChannel = 0; + float analogValue = 0; + + // context->audioFrames should be equal to inNumSamples + // for(unsigned int n = 0; n < context->audioFrames; n++) { + for (unsigned int n = 0; n < inNumSamples; n++) { + analogPin = (int)fin[n]; + muxChannel = (int)fmux[n]; + if ((analogPin < 0) || (analogPin >= context->analogInChannels) || (muxChannel < 0) + || (muxChannel > context->multiplexerChannels)) { + rt_printf("MultiplexAnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogInChannels, analogPin); + rt_printf("MultiplexAnalogIn warning: muxChannel must be between %i and %i, it is %i \n", 0, + context->multiplexerChannels, muxChannel); + } else { + analogValue = multiplexerAnalogRead( + context, analogPin, muxChannel); // is there something like NI? analogReadNI(context, 0, analogPin); + // if(analogPin == 0) + // { + // static int count = 0; + // count++; + // if(count % 20000 == 0) + // rt_printf("MultiPlexed AnalogValue = %.3f\n", analogValue); + // } + } + *++out = analogValue; + } +} + +void MultiplexAnalogIn_next_aak(MultiplexAnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* fin = IN(0); // analog in pin, can be modulated + int muxChannel = (float)IN0(1); + float* out = ZOUT(0); + int analogPin = 0; + float analogValue = 0; + + // context->audioFrames should be equal to inNumSamples + // for(unsigned int n = 0; n < context->audioFrames; n++) { + for (unsigned int n = 0; n < inNumSamples; n++) { + analogPin = (int)fin[n]; + if ((analogPin < 0) || (analogPin >= context->analogInChannels) || (muxChannel < 0) + || (muxChannel > context->multiplexerChannels)) { + rt_printf("MultiplexAnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogInChannels, analogPin); + rt_printf("MultiplexAnalogIn warning: muxChannel must be between %i and %i, it is %i \n", 0, + context->multiplexerChannels, muxChannel); + } else { + analogValue = multiplexerAnalogRead( + context, analogPin, muxChannel); // is there something like NI? analogReadNI(context, 0, analogPin); + // if(analogPin == 0) + // { + // static int count = 0; + // count++; + // if(count % 20000 == 0) + // rt_printf("MultiPlexed AnalogValue = %.3f\n", analogValue); + // } + } + *++out = analogValue; + } +} + +void MultiplexAnalogIn_next_aka(MultiplexAnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int analogPin = (float)IN0(0); + float* fmux = IN(1); // mux channel, can be modulated + float* out = ZOUT(0); + int muxChannel = 0; + float analogValue = 0; + + // context->audioFrames should be equal to inNumSamples + // for(unsigned int n = 0; n < context->audioFrames; n++) { + for (unsigned int n = 0; n < inNumSamples; n++) { + muxChannel = (int)fmux[n]; + if ((analogPin < 0) || (analogPin >= context->analogInChannels) || (muxChannel < 0) + || (muxChannel > context->multiplexerChannels)) { + rt_printf("MultiplexAnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogInChannels, analogPin); + rt_printf("MultiplexAnalogIn warning: muxChannel must be between %i and %i, it is %i \n", 0, + context->multiplexerChannels, muxChannel); + } else { + analogValue = multiplexerAnalogRead( + context, analogPin, muxChannel); // is there something like NI? analogReadNI(context, 0, analogPin); + // if(analogPin == 0) + // { + // static int count = 0; + // count++; + // if(count % 20000 == 0) + // rt_printf("MultiPlexed AnalogValue = %.3f\n", analogValue); + // } + } + *++out = analogValue; + } +} + +void MultiplexAnalogIn_next_akk(MultiplexAnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int analogPin = (float)IN0(0); + int muxChannel = (float)IN0(1); + float* out = ZOUT(0); + float analogValue = 0; + + if ((analogPin < 0) || (analogPin >= context->analogInChannels) || (muxChannel < 0) + || (muxChannel > context->multiplexerChannels)) { + rt_printf("MultiplexAnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogInChannels, analogPin); + rt_printf("MultiplexAnalogIn warning: muxChannel must be between %i and %i, it is %i \n", 0, + context->multiplexerChannels, muxChannel); + for (unsigned int n = 0; n < inNumSamples; n++) { + *++out = 0; + } + } else { + for (unsigned int n = 0; n < inNumSamples; n++) { + analogValue = multiplexerAnalogRead( + context, analogPin, muxChannel); // is there something like NI? analogReadNI(context, 0, analogPin); + // if(analogPin == 0) + // { + // static int count = 0; + // count++; + // if(count % 20000 == 0) + // rt_printf("MultiPlexed AnalogValue = %.3f\n", analogValue); + // } + *++out = analogValue; + } + } +} + +void MultiplexAnalogIn_next_kkk(MultiplexAnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int analogPin = (float)IN0(0); + int muxChannel = (float)IN0(1); + + if ((analogPin < 0) || (analogPin >= context->analogInChannels)) { + rt_printf("MultiplexAnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogInChannels, analogPin); + ZOUT0(0) = 0.0; + } else if ((muxChannel < 0) || (muxChannel > context->multiplexerChannels)) { + rt_printf("MultiplexAnalogIn warning: muxChannel must be between %i and %i, it is %i \n", 0, + context->multiplexerChannels, muxChannel); + ZOUT0(0) = 0.0; + } else { + ZOUT0(0) = multiplexerAnalogRead( + context, analogPin, muxChannel); // is there something like NI? analogReadNI(context, 0, analogPin); + } +} + +void MultiplexAnalogIn_Ctor(MultiplexAnalogIn* unit) { + BelaContext* context = unit->mWorld->mBelaContext; + + if (context->analogFrames == 0 || context->analogFrames > context->audioFrames) { + rt_printf("MultiplexAnalogIn Error: the UGen needs BELA analog enabled, with 4 or 8 channels\n"); + return; + } + if (context->multiplexerChannels == 0) { + rt_printf("MultiplexAnalogIn Error: the UGen needs BELA Multiplexer Capelet enabled\n"); + return; + } + + // initiate first sample + MultiplexAnalogIn_next_kkk(unit, 1); + // set calculation method + if (unit->mCalcRate == calc_FullRate) { + if (INRATE(0) == calc_FullRate) { + if (INRATE(1) == calc_FullRate) { + SETCALC(MultiplexAnalogIn_next_aaa); + } else { + // rt_printf("AnalogIn: aa\n"); + SETCALC(MultiplexAnalogIn_next_aak); + } + } else { + if (INRATE(1) == calc_FullRate) { + SETCALC(MultiplexAnalogIn_next_aka); + } else { + // rt_printf("AnalogIn: ak\n"); + SETCALC(MultiplexAnalogIn_next_akk); + } + } + } else { + if ((INRATE(0) == calc_FullRate) || (INRATE(1) == calc_FullRate)) { + rt_printf("MultiplexAnalogIn warning: output rate is control rate, so cannot change analog pin or " + "multiplex channel at audio rate\n"); + } + // rt_printf("AnalogIn: kk\n"); + SETCALC(MultiplexAnalogIn_next_kkk); + } +} + +////////////////////////////////////////////////////////////////////////////////////////////////// + +void AnalogIn_next_aa(AnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* fin = IN(0); // analog in pin, can be modulated + float* out = ZOUT(0); + int analogPin = 0; + float analogValue = 0; + + // context->audioFrames should be equal to inNumSamples + // for(unsigned int n = 0; n < context->audioFrames; n++) { + for (unsigned int n = 0; n < inNumSamples; n++) { + analogPin = (int)fin[n]; + // analogPin = sc_clip( analogPin, 0.0, context->analogInChannels ); + if ((analogPin < 0) || (analogPin >= context->analogInChannels)) { + rt_printf("AnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogInChannels, analogPin); + } else { + analogValue = analogReadNI(context, n, analogPin); + // if(analogPin == 0) + // { + // static int count = 0; + // count++; + // if(count % 20000 == 0) + // rt_printf("AnalogValue = %.3f\n", analogValue); + // } + } + *++out = analogValue; + } +} + +void AnalogIn_next_ak(AnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int analogPin = (float)IN0(0); + float* out = ZOUT(0); + float analogValue = 0; + + if ((analogPin < 0) || (analogPin >= context->analogInChannels)) { + rt_printf("AnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, context->analogInChannels, + analogPin); + for (unsigned int n = 0; n < inNumSamples; n++) { + *++out = 0; + } + } else { + for (unsigned int n = 0; n < inNumSamples; n++) { + analogValue = analogReadNI(context, n, analogPin); + // if(analogPin == 0) + // { + // static int count = 0; + // count++; + // if(count % 20000 == 0) + // rt_printf("AnalogValue = %.3f\n", analogValue); + // } + *++out = analogValue; + } + } +} + + +void AnalogIn_next_kk(AnalogIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int analogPin = (float)IN0(0); + + if ((analogPin < 0) || (analogPin >= context->analogInChannels)) { + rt_printf("AnalogIn warning: analog pin must be between %i and %i, it is %i \n", 0, context->analogInChannels, + analogPin); + ZOUT0(0) = 0.0; + } else { + ZOUT0(0) = analogReadNI(context, 0, analogPin); + } +} + +void AnalogIn_Ctor(AnalogIn* unit) { + BelaContext* context = unit->mWorld->mBelaContext; + + if (context->analogFrames == 0 || context->analogFrames > context->audioFrames) { + rt_printf("AnalogIn Error: the UGen needs BELA analog enabled, with 4 or 8 channels\n"); + return; + } + + // initiate first sample + AnalogIn_next_kk(unit, 1); + // set calculation method + if (unit->mCalcRate == calc_FullRate) { + if (INRATE(0) == calc_FullRate) { + // rt_printf("AnalogIn: aa\n"); + SETCALC(AnalogIn_next_aa); + } else { + // rt_printf("AnalogIn: ak\n"); + SETCALC(AnalogIn_next_ak); + } + } else { + if (INRATE(0) == calc_FullRate) { + rt_printf("AnalogIn warning: output rate is control rate, so cannot change analog pin at audio rate\n"); + } + // rt_printf("AnalogIn: kk\n"); + SETCALC(AnalogIn_next_kk); + } +} + +////////////////////////////////////////////////////////////////////////////////////////////////// + +void AnalogOut_next_aaa(AnalogOut* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* fin = IN(0); // analog in pin, can be modulated + float* in = IN(1); + + int analogPin = 0; + float newinput = 0; + for (unsigned int n = 0; n < inNumSamples; n++) { + // read input + analogPin = (int)fin[n]; + if ((analogPin < 0) || (analogPin >= context->analogOutChannels)) { + rt_printf("AnalogOut warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogOutChannels, analogPin); + } else { + newinput = in[n]; // read next input sample + analogWriteOnceNI(context, n, analogPin, newinput); + } + } +} + +void AnalogOut_next_aka(AnalogOut* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int analogPin = (int)IN0(0); // analog in pin, can be modulated + float* in = IN(1); + + float newinput = 0; + if ((analogPin < 0) || (analogPin >= context->analogOutChannels)) { + rt_printf("AnalogOut warning: analog pin must be between %i and %i, it is %i \n", 0, context->analogOutChannels, + analogPin); + } else { + for (unsigned int n = 0; n < inNumSamples; n++) { + newinput = in[n]; // read next input sample + analogWriteOnceNI(context, n, analogPin, newinput); + } + } +} + +void AnalogOut_next_aak(AnalogOut* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* fin = IN(0); // analog in pin, can be modulated + float in = IN0(1); + + int analogPin = 0; + for (unsigned int n = 0; n < inNumSamples; n++) { + // read input + analogPin = (int)fin[n]; + if ((analogPin < 0) || (analogPin >= context->analogOutChannels)) { + rt_printf("AnalogOut warning: analog pin must be between %i and %i, it is %i \n", 0, + context->analogOutChannels, analogPin); + } else { + analogWriteOnceNI(context, n, analogPin, in); + } + } +} + + +void AnalogOut_next_kk(AnalogOut* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int analogPin = (int)IN0(0); // analog in pin, can be modulated + float in = IN0(1); + + if ((analogPin < 0) || (analogPin >= context->analogOutChannels)) { + rt_printf("AnalogOut warning: analog pin must be between %i and %i, it is %i \n", 0, context->analogOutChannels, + analogPin); + } else { + analogWriteNI(context, 0, analogPin, in); + } +} + +void AnalogOut_Ctor(AnalogOut* unit) { + BelaContext* context = unit->mWorld->mBelaContext; + + if (context->analogFrames == 0) { + rt_printf("AnalogOut Error: the UGen needs BELA analog enabled\n"); + return; + } + + // initiate first sample + AnalogOut_next_kk(unit, 1); + + if (unit->mCalcRate == calc_FullRate) { // ugen running at audio rate; + if (INRATE(0) == calc_FullRate) { // pin changed at audio rate + if (INRATE(1) == calc_FullRate) { // output changed at audio rate + SETCALC(AnalogOut_next_aaa); + // rt_printf("AnalogOut: aaa\n"); + } else { + SETCALC(AnalogOut_next_aak); + // rt_printf("AnalogOut: aak\n"); + } + } else { // pin changed at control rate + if (INRATE(1) == calc_FullRate) { // output changed at audio rate + SETCALC(AnalogOut_next_aka); + // rt_printf("AnalogOut: aka\n"); + } else { // analog output only changes at control rate anyways + rt_printf("AnalogOut warning: inputs are control rate, so AnalogOut is also running at control rate\n"); + // rt_printf("AnalogOut: kk\n"); + SETCALC(AnalogOut_next_kk); + } + } + } else { // ugen at control rate + if ((INRATE(0) == calc_FullRate) || (INRATE(1) == calc_FullRate)) { + rt_printf("AnalogOut warning: output rate is control rate, so cannot change inputs at audio rate\n"); + } + // rt_printf("AnalogOut: kk\n"); + SETCALC(AnalogOut_next_kk); + } +} + +////////////////////////////////////////////////////////////////////////////////////////////////// + +void DigitalIn_next_a(DigitalIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int pinid = unit->mDigitalPin; + int digitalValue; + float* out = ZOUT(0); + + // context->audioFrames should be equal to inNumSamples + // for(unsigned int n = 0; n < context->audioFrames; n++) { + for (unsigned int n = 0; n < inNumSamples; n++) { + digitalValue = digitalRead(context, n, pinid); // read the value of the button + *++out = (float)digitalValue; + } +} + +void DigitalIn_next_k(DigitalIn* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int pinid = unit->mDigitalPin; + int digitalValue = digitalRead(context, 0, pinid); // read the value of the button + ZOUT0(0) = (float)digitalValue; +} + +void DigitalIn_next_dummy_a(DigitalIn* unit, int inNumSamples) { + float* out = ZOUT(0); + + for (unsigned int n = 0; n < inNumSamples; n++) { + *++out = 0.0; + } +} + +void DigitalIn_next_dummy_k(DigitalIn* unit, int inNumSamples) { ZOUT0(0) = 0.0; } + +void DigitalIn_Ctor(DigitalIn* unit) { + BelaContext* context = unit->mWorld->mBelaContext; + + float fDigitalIn = ZIN0(0); // digital in pin -- cannot change after construction + unit->mDigitalPin = (int)fDigitalIn; + // unit->mDigitalPin = (int) sc_clip( fDigitalIn, 0., 15.0 ); + if ((unit->mDigitalPin < 0) || (unit->mDigitalPin >= context->digitalChannels)) { + rt_printf("DigitalIn warning: digital pin must be between %i and %i, it is %i \n", 0, context->digitalChannels, + unit->mDigitalPin); + // initiate first sample + if (unit->mCalcRate == calc_FullRate) { // ugen running at audio rate; + DigitalIn_next_dummy_a(unit, 1); + } else { + DigitalIn_next_dummy_k(unit, 1); + } + } else { + pinMode(context, 0, unit->mDigitalPin, INPUT); + // initiate first sample + DigitalIn_next_k(unit, 1); + // set calculation method + if (unit->mCalcRate == calc_FullRate) { // ugen running at audio rate; + SETCALC(DigitalIn_next_a); + // rt_printf("DigitalIn: a\n"); + } else { + SETCALC(DigitalIn_next_k); + // rt_printf("DigitalIn: k\n"); + } + } +} + +////////////////////////////////////////////////////////////////////////////////////////////////// + +void DigitalOut_next_a_once(DigitalOut* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int pinid = unit->mDigitalPin; + float* in = IN(1); + + float newinput = 0; + // int lastOut = unit->mLastOut; + + for (unsigned int n = 0; n < inNumSamples; n++) { + // read input + newinput = in[n]; + if (newinput > 0.5) { + digitalWriteOnce(context, n, pinid, 1); + } else { + digitalWriteOnce(context, n, pinid, 0); + } + // else if ( lastOut == 1 ) { + // digitalWrite(context, n, pinid, 0 ); + // } + } + // unit->mLastOut = lastOut; +} + +void DigitalOut_next_a(DigitalOut* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int pinid = unit->mDigitalPin; + float* in = IN(1); + + float newinput = 0; + int lastOut = unit->mLastOut; + + for (unsigned int n = 0; n < inNumSamples; n++) { + // read input + newinput = in[n]; + if (newinput > 0.5) { + if (lastOut == 0) { + lastOut = 1; + digitalWrite(context, n, pinid, 1); + } + } else if (lastOut == 1) { + lastOut = 0; + digitalWrite(context, n, pinid, 0); + } + } + unit->mLastOut = lastOut; +} + +void DigitalOut_next_k(DigitalOut* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int pinid = unit->mDigitalPin; + float in = IN0(1); + + int lastOut = unit->mLastOut; + if (in > 0.5) { + if (lastOut == 0) { + lastOut = 1; + digitalWrite(context, 0, pinid, 1); + } + } else if (lastOut == 1) { + lastOut = 0; + digitalWrite(context, 0, pinid, 0); + } + unit->mLastOut = lastOut; +} + +void DigitalOut_next_dummy(DigitalOut* unit, int inNumSamples) {} + +void DigitalOut_Ctor(DigitalOut* unit) { + BelaContext* context = unit->mWorld->mBelaContext; + + float fDigital = ZIN0(0); // digital in pin -- cannot change after construction + int writeMode = + (int)ZIN0(2); // method of writing; 1 = writeOnce; 0 = write on change -- cannot change after construction + unit->mDigitalPin = (int)fDigital; + unit->mLastOut = 0; + + if ((unit->mDigitalPin < 0) || (unit->mDigitalPin >= context->digitalChannels)) { + rt_printf("DigitalOut warning: digital pin must be between %i and %i, it is %i \n", 0, context->digitalChannels, + unit->mDigitalPin); + // initiate first sample + DigitalOut_next_dummy(unit, 1); + // set calculation method + SETCALC(DigitalOut_next_dummy); + } else { + pinMode(context, 0, unit->mDigitalPin, OUTPUT); + // initiate first sample + DigitalOut_next_k(unit, 1); + + if (unit->mCalcRate == calc_FullRate) { // ugen running at audio rate; + if (INRATE(1) == calc_FullRate) { // output changed at audio rate + if (writeMode) { + // rt_printf("DigitalOut: a once\n"); + SETCALC(DigitalOut_next_a_once); + } else { + // rt_printf("DigitalOut: a\n"); + SETCALC(DigitalOut_next_a); + } + } else { // not much reason to actually do audiorate output + rt_printf("DigitalOut warning: inputs are control rate, so DigitalOut will run at control rate\n"); + // rt_printf("DigitalOut: k\n"); + SETCALC(DigitalOut_next_k); + } + } else { // ugen at control rate + if (INRATE(1) == calc_FullRate) { + rt_printf("DigitalOut warning: UGen rate is control rate, so cannot change inputs at audio rate\n"); + } + // rt_printf("DigitalOut: k\n"); + SETCALC(DigitalOut_next_k); + } + } +} + +////////////////////////////////////////////////////////////////////////////////////////////////// + +void DigitalIO_next_aaaa_once(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* pinid = IN(0); + float* in = IN(1); // input value + float* iomode = IN(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newpin; + float newmode = 0; // input + + int newDigInInt = unit->mLastDigitalIn; + float newDigIn = (float)newDigInInt; + int newDigOut = unit->mLastDigitalOut; + + for (unsigned int n = 0; n < inNumSamples; n++) { + // read input + newpin = (int)pinid[n]; + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, + context->digitalChannels, newpin); + } else { + newDigOut = (int)in[n]; + newmode = iomode[n]; + if (newmode < 0.5) { + pinModeOnce(context, n, newpin, INPUT); + newDigInInt = digitalRead(context, n, newpin); + } else { + pinModeOnce(context, n, newpin, OUTPUT); + digitalWriteOnce(context, n, newpin, newDigOut); + } + } + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + + +void DigitalIO_next_aaak_once(DigitalIO* unit, int inNumSamples) { + // pinMode at control rate + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* pinid = IN(0); + float* in = IN(1); // input value + float iomode = IN0(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newDigInInt = unit->mLastDigitalIn; + float newDigIn = (float)newDigInInt; + + int newDigOut = unit->mLastDigitalOut; + // float newinput; + + int newpin; + if (iomode < 0.5) { + for (unsigned int n = 0; n < inNumSamples; n++) { + newpin = (int)pinid[n]; + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, + context->digitalChannels, newpin); + } else { + pinModeOnce(context, n, newpin, INPUT); + newDigInInt = digitalRead(context, n, newpin); + } + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + } else { + for (unsigned int n = 0; n < inNumSamples; n++) { + newpin = (int)pinid[n]; + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, + context->digitalChannels, newpin); + } else { + pinModeOnce(context, n, newpin, OUTPUT); + newDigOut = (int)in[n]; + digitalWriteOnce(context, n, newpin, newDigOut); + } + *++out = (float)newDigInInt; + } + } + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + +// output changing at control rate, rest audio +void DigitalIO_next_aaka_once(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* pinid = IN(0); + float in = IN0(1); // input value + float* iomode = IN(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newpin; + float newmode = 0; // input + + int newDigInInt = unit->mLastDigitalIn; + float newDigIn = (float)newDigInInt; + // int newDigOut = unit->mLastDigitalOut; + int newDigOut = (int)in; + + for (unsigned int n = 0; n < inNumSamples; n++) { + // read input + newpin = (int)pinid[n]; + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, + context->digitalChannels, newpin); + } else { + newmode = iomode[n]; + if (newmode < 0.5) { + pinModeOnce(context, n, newpin, INPUT); + newDigInInt = digitalRead(context, n, newpin); + } else { + pinModeOnce(context, n, newpin, OUTPUT); + digitalWriteOnce(context, n, newpin, newDigOut); + } + } + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + + +// output changing at control rate, and pin mode at control rate +void DigitalIO_next_aakk_once(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float* pinid = IN(0); + float in = IN0(1); // input value + float iomode = IN0(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newpin; + float newmode = 0; // input + + int newDigInInt = unit->mLastDigitalIn; + float newDigIn = (float)newDigInInt; + // int newDigOut = unit->mLastDigitalOut; + int newDigOut = (int)in; + + if (iomode < 0.5) { + for (unsigned int n = 0; n < inNumSamples; n++) { + newpin = (int)pinid[n]; + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, + context->digitalChannels, newpin); + } else { + pinModeOnce(context, n, newpin, INPUT); + newDigInInt = digitalRead(context, n, newpin); + } + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + } else { + for (unsigned int n = 0; n < inNumSamples; n++) { + newpin = (int)pinid[n]; + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, + context->digitalChannels, newpin); + } else { + pinModeOnce(context, n, newpin, OUTPUT); + digitalWriteOnce(context, n, newpin, newDigOut); + } + *++out = (float)newDigInInt; + } + } + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + + +// pin changing at control rate, output control rate, rest audio rate +void DigitalIO_next_akaa_once(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float pinid = IN0(0); + float in = IN0(1); // input value + float* iomode = IN(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newpin = (int)pinid; + float newmode = 0; // input + // float newinput; + + int newDigInInt = unit->mLastDigitalIn; + float newDigIn = (float)newDigInInt; + + int newDigOut = (int)in; + + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, context->digitalChannels, + newpin); + } else { + for (unsigned int n = 0; n < inNumSamples; n++) { + // newinput = in[n]; + newmode = iomode[n]; + if (newmode < 0.5) { + pinModeOnce(context, n, newpin, INPUT); + newDigInInt = digitalRead(context, n, newpin); + } else { + pinModeOnce(context, n, newpin, OUTPUT); + digitalWriteOnce(context, n, newpin, newDigOut); + } + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + } + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + +// result audio rate, pin changing at control rate, output value audio rate, pin mode change control rate +void DigitalIO_next_akak_once(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float pinid = IN0(0); + float* in = IN(1); // input value + float iomode = IN0(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newpin = (int)pinid; + float newmode = 0; // input + // float newinput; + + int newDigInInt = unit->mLastDigitalIn; + float newDigIn = (float)newDigInInt; + + int newDigOut = (int)in; + + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, context->digitalChannels, + newpin); + } else { + if (iomode < 0.5) { + pinMode(context, 0, newpin, INPUT); + for (unsigned int n = 0; n < inNumSamples; n++) { + newDigInInt = digitalRead(context, n, newpin); + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + } else { + pinMode(context, 0, newpin, OUTPUT); + for (unsigned int n = 0; n < inNumSamples; n++) { + newDigOut = (int)in[n]; + digitalWriteOnce(context, n, newpin, newDigOut); + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + } + } + + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + +// audio rate ugen output, pin changing at control rate, output at control rate, mode at audio rate +void DigitalIO_next_akka_once(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float pinid = IN0(0); + float in = IN0(1); // input value + float* iomode = IN(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newpin = (int)pinid; + float newinput = in; + + float newmode = 0; // input + + int newDigInInt = unit->mLastDigitalIn; + int newDigOut = unit->mLastDigitalOut; + + if ((newpin < 0) || (newpin >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, context->digitalChannels, + newpin); + } + + for (unsigned int n = 0; n < inNumSamples; n++) { + newmode = iomode[n]; + if (newmode < 0.5) { // digital read + pinModeOnce(context, n, newpin, INPUT); + newDigInInt = digitalRead(context, n, newpin); + } else { // digital write + pinModeOnce(context, n, newpin, OUTPUT); + if (newinput > 0.5) { + newDigOut = 1; + } else { + newDigOut = 0; + } + digitalWriteOnce(context, n, newpin, newDigOut); + } + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + + +// all inputs at control rate, output at audio rate +void DigitalIO_next_ak(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + float pinid = IN0(0); + float in = IN0(1); // input value + float iomode = IN0(2); // IO mode : < 0.5 = input, else output + float* out = ZOUT(0); // output value = last output value + + int newpin = (int)pinid; + + int newDigInInt = unit->mLastDigitalIn; + float newDigIn = (float)newDigInInt; + int newDigOut = (int)in; + + if ((pinid < 0) || (pinid >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, context->digitalChannels, + newpin); + } else { + if (iomode < 0.5) { + pinMode(context, 0, newpin, INPUT); + for (unsigned int n = 0; n < inNumSamples; n++) { + // read input + newDigInInt = digitalRead(context, n, newpin); + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + } else { + pinMode(context, 0, newpin, OUTPUT); + for (unsigned int n = 0; n < inNumSamples; n++) { + if (in > 0.5) { + newDigOut = 1; + } else { + newDigOut = 0; + } + digitalWriteOnce(context, n, newpin, newDigOut); + // always write to the output of the UGen + *++out = (float)newDigInInt; + } + } + } + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + +// all at control rate, output at control rate +void DigitalIO_next_kk(DigitalIO* unit, int inNumSamples) { + World* world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext* context = world->mBelaContext; + + int pinid = (int)IN0(0); + float in = IN0(1); // input value + float iomode = IN0(2); // IO mode : < 0.5 = input, else output + // float *out = ZOUT(0); // output value = last output value + + int newDigInInt = unit->mLastDigitalIn; + int newDigOut = unit->mLastDigitalOut; + + if ((pinid < 0) || (pinid >= context->digitalChannels)) { + rt_printf("DigitalIO warning: digital pin must be between %i and %i, it is %i \n", 0, context->digitalChannels, + pinid); + } else { + if (iomode < 0.5) { + pinMode(context, 0, pinid, INPUT); + newDigInInt = digitalRead(context, 0, pinid); + } else { + pinMode(context, 0, pinid, OUTPUT); + if (in > 0.5) { + newDigOut = 1; + } else { + newDigOut = 0; + } + digitalWrite(context, 0, pinid, newDigOut); + } + } + ZOUT0(0) = (float)newDigInInt; + + unit->mLastDigitalIn = newDigInInt; + unit->mLastDigitalOut = newDigOut; +} + +/* +void DigitalIO_next(DigitalIO *unit, int inNumSamples) +{ + World *world = unit->mWorld; + int bufLength = world->mBufLength; + BelaContext *context = world->mBelaContext; + + float *pinid = IN(0); + float *in = IN(1); // input value + float *iomode = IN(2); // IO mode : < 0.5 = input, else output + float *out = ZOUT(0); // output value = last output value + + int newpin; + float newmode = 0; // input + float newinput = 0; + int newinputInt = 0; + int newoutput = unit->mLastIn; + + // context->audioFrames should be equal to inNumSamples +// for(unsigned int n = 0; n < context->digitalFrames; n++) { + for(unsigned int n = 0; n < inNumSamples; n++) { + // read input + newpin = (int) pinid[n]; + if ( (newpin < 0) || (newpin >= context->digitalChannels) ){ + rt_printf( "digital pin must be between %i and %i, it is %i", 0, context->digitalChannels, newpin ); + } else { + newinput = in[n]; + newmode = iomode[n]; + if ( newmode < 0.5 ){ + // pinModeOnce( context, n, newpin, INPUT ); + pinMode( context, n, newpin, INPUT ); + newoutput = digitalRead(context, n, newpin); + } else { + // pinModeOnce( context, n, newpin, OUTPUT ); + pinMode( context, n, newpin, OUTPUT ); + if ( newinput > 0.5 ){ + newinputInt = 1; + } else { + newinputInt = 0; + } + // digitalWriteOnce(context, n, newpin, newinputInt); + digitalWrite(context, n, newpin, newinputInt); + } + } + // always write to the output of the UGen + *++out = (float) newoutput; + } + unit->mLastDigitalIn = newoutput; + unit->mLastDigitalOut = newinput; +} +*/ + +void DigitalIO_Ctor(DigitalIO* unit) { + BelaContext* context = unit->mWorld->mBelaContext; + + unit->mLastDigitalIn = 0; + unit->mLastDigitalOut = 0; + + // int writeMode = (int) ZIN0(3); // method of writing; 1 = writeOnce; 0 = write on change + + // initiate first sample + DigitalIO_next_kk(unit, 1); + // set calculation method + // SETCALC(DigitalIO_next); + if (unit->mCalcRate == calc_FullRate) { // ugen running at audio rate; + if (INRATE(0) == calc_FullRate) { // pin changed at audio rate + if (INRATE(1) == calc_FullRate) { // output changed at audio rate + if (INRATE(2) == calc_FullRate) { // pinmode changed at audio rate + // if ( writeMode ){ + // rt_printf("DigitalIO: aaaa once\n"); + SETCALC(DigitalIO_next_aaaa_once); + // } else { + // SETCALC(DigitalIO_next_aaaa); + // } + } else { + // if ( writeMode ){ + // rt_printf("DigitalIO: aaak once\n"); + SETCALC(DigitalIO_next_aaak_once); + // } else { + // SETCALC(DigitalIO_next_aaak); + // } + } + } else { // output changed at control rate + if (INRATE(2) == calc_FullRate) { // pinmode changed at audio rate + // if ( writeMode ){ + // rt_printf("DigitalIO: aaka once\n"); + SETCALC(DigitalIO_next_aaka_once); + // } else { + // SETCALC(DigitalIO_next_aaka); + // } + } else { + // if ( writeMode ){ + // rt_printf("DigitalIO: aakk once\n"); + SETCALC(DigitalIO_next_aakk_once); + // } else { + // SETCALC(DigitalIO_next_aakk); + // } + } + } + } else { // pin changed at control rate + if (INRATE(1) == calc_FullRate) { // output changed at audio rate + if (INRATE(2) == calc_FullRate) { // pinmode changed at audio rate + // if ( writeMode ){ + // rt_printf("DigitalIO: akaa once\n"); + SETCALC(DigitalIO_next_akaa_once); + // } else { + // SETCALC(DigitalIO_next_akaa); + // } + } else { + // if ( writeMode ){ + // rt_printf("DigitalIO: akak once\n"); + SETCALC(DigitalIO_next_akak_once); + // } else { + // SETCALC(DigitalIO_next_akak); + // } + } + } else { // output changed at control rate + if (INRATE(2) == calc_FullRate) { // pinmode changed at audio rate + // if ( writeMode ){ + // rt_printf("DigitalIO: akka once\n"); + SETCALC(DigitalIO_next_akka_once); + // } else { + // SETCALC(DigitalIO_next_akka); + // } + } else { // pinmode at control rate + // rt_printf("DigitalIO: ak once\n"); + SETCALC(DigitalIO_next_ak); + } + } + } + } else { // ugen at control rate + if ((INRATE(0) == calc_FullRate) || (INRATE(1) == calc_FullRate) || (INRATE(2) == calc_FullRate)) { + rt_printf("DigitalIO warning: UGen rate is control rate, so cannot change inputs at audio rate\n"); + } + // rt_printf("DigitalIO: kk\n"); + SETCALC(DigitalIO_next_kk); + } +} + +////////////////////////////////////////////////////////////////////////////////////////////////// + +/* +int noScopeChannels = 0; +Scope * belaScope; + + +void BelaScopeChannel_next( BelaScope *unit ) +{ + int scopeChannel = unit->mScopeChannel; + float *in = IN(1); + + for(unsigned int n = 0; n < inNumSamples; n++) { + belaScope->logChannel( scopeChannel, in[n] ); + } +} + + +void BelaScopeChannel_Ctor(BelaScope *unit) +{ + BelaContext *context = unit->mWorld->mBelaContext; + +// belaScope = Scope(); + // which channel is an input variable +// belaScope->setup(3, context->audioSampleRate); + float fChan = ZIN0(0); // number of channels + mScopeChannel = (int ) fChan; + // check whether channel is within number of channels of scope + if ( mScopeChannel > noScopeChannels ){ + // error + } + // initiate first sample + + BelaScopeChannel_next( unit, 1); + // set calculation method + SETCALC(BelaScopeChannel_next); +} + + + +void BelaScope_next(BelaScope *unit) +{ +} + +void BelaScope_Ctor(BelaScope *unit) +{ + BelaContext *context = unit->mWorld->mBelaContext; + + float fChan = ZIN0(0); // number of channels + noScopeChannels = (int) fChan; + + belaScope = Scope(); + // number of channels is a variable + belaScope->setup(noScopeChannels, context->audioSampleRate); + + // initiate first sample + BelaScope_next( unit, 1); + // set calculation method + SETCALC(BelaScope_next); +} + +void BelaScope_Dtor(BelaScope *unit) +{ + belaScope->stop(); + delete belaScope; + noScopeChannels = 0; +} +*/ + +////////////////////////////////////////////////////////////////////////////////////////////////// + +// extern "C" +// { +// +// +// } + +// // the functions below are needed?? +// +// void render(BelaContext *belaContext, void *userData) +// { +// // SC_BelaDriver *driver = (SC_BelaDriver*)userData; +// // driver->BelaAudioCallback(belaContext); +// } +// // setup() is called once before the audio rendering starts. +// // Use it to perform any initialisation and allocation which is dependent +// // on the period size or sample rate. +// // +// // userData holds an opaque pointer to a data structure that was passed +// // in from the call to initAudio(). +// // +// // Return true on success; returning false halts the program. +// bool setup(BelaContext* belaContext, void* userData) +// { +// if(userData == 0){ +// printf("BelaPLUGINS: error, setup() got no user data\n"); +// return false; +// } +// +// return true; +// } +// +// // cleanup() is called once at the end, after the audio has stopped. +// // Release any resources that were allocated in setup(). +// void cleanup(BelaContext *belaContext, void *userData) +// { +// } + + +PluginLoad(BELA) { + ft = inTable; + + DefineSimpleUnit(MultiplexAnalogIn); + DefineSimpleUnit(AnalogIn); + DefineSimpleUnit(AnalogOut); + DefineSimpleUnit(DigitalIn); + DefineSimpleUnit(DigitalOut); + DefineSimpleUnit(DigitalIO); +} + + +// C_LINKAGE SC_API_EXPORT void unload(InterfaceTable *inTable) +// { +// +// } diff --git a/server/plugins/CMakeLists.txt b/server/plugins/CMakeLists.txt index b1838ceb084..98647935e38 100644 --- a/server/plugins/CMakeLists.txt +++ b/server/plugins/CMakeLists.txt @@ -69,6 +69,47 @@ foreach(plugin ${plugin_sources}) list(APPEND plugins ${plugin_name}) endforeach(plugin) +if(AUDIOAPI STREQUAL bela) + # The Bela lib will have its own set of libraries and includes + # However, the Bela audio backend for SC has some direct calls to Xenomai + # Therefore we need to get flags for those as well + find_package(Xenomai) + if(NOT XENOMAI_FOUND) + message(FATAL_ERROR "Bela selected as audio API, but Xenomai development files not found") + endif() + find_package(Bela) + if(NOT BELA_FOUND) + message(FATAL_ERROR "Bela selected as audio API, but Bela development files not found") + endif() + message(STATUS "bela libs: ${BELA_LIBRARIES}") + if (CMAKE_COMPILER_IS_GNUCXX OR CMAKE_COMPILER_IS_CLANG) + # recommended compile flags for beaglebone etc; set here because bela api flag directly implies the architecture + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${BELA_C_FLAGS}") + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${BELA_CXX_FLAGS}") + endif() +endif() + +if (BELA_FOUND) + add_library(BELAUGens MODULE + BELAUGens.cpp +# ${BELA_SOURCES} +# ${scplugin_common_sources} + ) + set(belaugens BELAUGens) + add_definitions("-DBELA" ${XENOMAI_DEFINITIONS} ${BELA_DEFINITIONS}) + include_directories(${XENOMAI_INCLUDE_DIRS}) + include_directories(${BELA_INCLUDE_DIRS}) + +# set(CMAKE_EXECUTABLE_RUNTIME_C_FLAG "-Wl,-wrap,clock_gettime,-rpath,") +# set(CMAKE_EXECUTABLE_RUNTIME_CXX_FLAG "-Wl,-wrap,clock_gettime,-rpath,") +# set(CMAKE_SHARED_LIBRARY_RUNTIME_C_FLAG "-Wl,-wrap,clock_gettime,-rpath,") +# set(CMAKE_SHARED_LIBRARY_RUNTIME_CXX_FLAG "-Wl,-wrap,clock_gettime,-rpath,") +# target_link_libraries(BELAUGens ${XENOMAI_LIBRARIES} ${BELA_LIBRARIES} libscsynth.a) + target_link_libraries(BELAUGens libscsynth) + + list(APPEND plugins BELAUGens) +endif() + if(NOT NO_X11) if (APPLE) add_library(UIUGens MODULE UIUGens.mm) @@ -215,6 +256,12 @@ if(NOT NO_X11) endforeach() endif() +if( BELA_FOUND ) + foreach(ugen ${belaugens}) + target_link_libraries(${ugen} ${XENOMAI_LIBRARIES} ${BELA_LIBRARIES}) + endforeach() +endif() + foreach(plugin ${plugins}) if(WIN32) target_link_libraries(${plugin} wsock32 ws2_32) diff --git a/server/scsynth/CMakeLists.txt b/server/scsynth/CMakeLists.txt index 4bf5602d539..c69e742fec3 100644 --- a/server/scsynth/CMakeLists.txt +++ b/server/scsynth/CMakeLists.txt @@ -33,7 +33,7 @@ if(AUDIOAPI STREQUAL "default") endif(APPLE) endif() -if(NOT AUDIOAPI MATCHES "^(jack|coreaudio|portaudio)$") +if(NOT AUDIOAPI MATCHES "^(jack|coreaudio|portaudio|bela)$") message(FATAL_ERROR "Unrecognised audio API: ${AUDIOAPI}") endif() @@ -47,6 +47,24 @@ elseif(AUDIOAPI STREQUAL portaudio AND SYSTEM_PORTAUDIO) if(NOT PORTAUDIO_FOUND) message(FATAL_ERROR "Portaudio selected as audio API, but development files not found") endif() +elseif(AUDIOAPI STREQUAL bela) + # The Bela lib will have its own set of libraries and includes + # However, the Bela audio backend for SC has some direct calls to Xenomai + # Therefore we need to get flags for those as well + find_package(Xenomai) + if(NOT XENOMAI_FOUND) + message(FATAL_ERROR "Bela selected as audio API, but Xenomai development files not found") + endif() + find_package(Bela) + if(NOT BELA_FOUND) + message(FATAL_ERROR "Bela selected as audio API, but Bela development files not found") + endif() + message(STATUS "bela libs: ${BELA_LIBRARIES}") + if (CMAKE_COMPILER_IS_GNUCXX OR CMAKE_COMPILER_IS_CLANG) + # recommended compile flags for beaglebone etc; set here because bela api flag directly implies the architecture + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${BELA_C_FLAGS}") + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${BELA_CXX_FLAGS}") + endif() endif() message(STATUS "Audio API (scsynth): ${AUDIOAPI}") @@ -124,6 +142,11 @@ elseif (AUDIOAPI STREQUAL portaudio) list(APPEND scsynth_sources SC_PortAudio.cpp ${CMAKE_SOURCE_DIR}/common/SC_PaUtils.cpp) add_definitions("-DSC_AUDIO_API=SC_AUDIO_API_PORTAUDIO" ${PORTAUDIO_DEFINITIONS}) include_directories(${PORTAUDIO_INCLUDE_DIRS}) +elseif (AUDIOAPI STREQUAL bela) + list(APPEND scsynth_sources SC_Bela.cpp) + add_definitions("-DSC_AUDIO_API=SC_AUDIO_API_BELA" "-DBELA" ${XENOMAI_DEFINITIONS} ${BELA_DEFINITIONS}) + include_directories(${XENOMAI_INCLUDE_DIRS}) + include_directories(${BELA_INCLUDE_DIRS}) endif() set (FINAL_BUILD 0) # disable final build for scsynth @@ -191,6 +214,8 @@ if (AUDIOAPI STREQUAL jack) target_link_libraries(libscsynth ${JACK_LIBRARIES}) elseif(AUDIOAPI STREQUAL portaudio) target_link_libraries(libscsynth ${PORTAUDIO_LIBRARIES}) +elseif(AUDIOAPI STREQUAL bela) + target_link_libraries(libscsynth ${XENOMAI_LIBRARIES} ${BELA_LIBRARIES}) elseif(AUDIOAPI STREQUAL coreaudio) target_link_libraries(libscsynth "-framework CoreAudio") endif() @@ -235,6 +260,10 @@ add_executable(scsynth ) target_link_libraries(scsynth libscsynth) +if(AUDIOAPI STREQUAL bela) + target_link_libraries(scsynth ${XENOMAI_LIBRARIES} ${BELA_LIBRARIES}) +endif() + if (PTHREADS_FOUND) target_link_libraries(scsynth ${PTHREADS_LIBRARIES}) endif() diff --git a/server/scsynth/SC_Bela.cpp b/server/scsynth/SC_Bela.cpp new file mode 100644 index 00000000000..691eb47c4c5 --- /dev/null +++ b/server/scsynth/SC_Bela.cpp @@ -0,0 +1,502 @@ +/* + Bela audio driver for SuperCollider. + Copyright (c) 2015 Dan Stowell. All rights reserved. + Copyright (c) 2016 Marije Baalman. All rights reserved. + Copyright (c) 2016 Giulio Moro. All rights reserved. + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + + This file contains elements from SC_PortAudio.cpp and SC_Jack.cpp, + copyright their authors, and published under the same licence. +*/ +#include "SC_CoreAudio.h" +#include +#include "SC_Prototypes.h" +#include "SC_HiddenWorld.h" +#include "SC_WorldOptions.h" +#include "SC_Time.hpp" +#include +#include + +#ifdef XENOMAI_SKIN_native +# include // needed for CLOCK_HOST_REALTIME +#endif +#ifdef XENOMAI_SKIN_posix +# include // needed for CLOCK_HOST_REALTIME +#endif + +extern "C" { +// This will be wrapped by Xenomai without requiring linker flags +int __wrap_clock_gettime(clockid_t clock_id, struct timespec* tp); +// this is provided by Xenomai +int rt_vprintf(const char* format, va_list ap); +} + +#include "Bela.h" +// Xenomai-specific includes +#include + +using namespace std; + +int32 server_timeseed() { return timeSeed(); } + +int64 gOSCoffset = 0; + +static inline int64 GetCurrentOSCTime() { return OSCTime(getTime()); } + +int64 oscTimeNow() { return GetCurrentOSCTime(); } + +void initializeScheduler() { gOSCoffset = GetCurrentOSCTime(); } + + +class SC_BelaDriver : public SC_AudioDriver { + int mInputChannelCount, mOutputChannelCount; + +protected: + // Driver interface methods + virtual bool DriverSetup(int* outNumSamplesPerCallback, double* outSampleRate); + virtual bool DriverStart(); + virtual bool DriverStop(); + +public: + SC_BelaDriver(struct World* inWorld); + virtual ~SC_BelaDriver(); + + void BelaAudioCallback(BelaContext* belaContext); + void SignalReceived(int); + static void staticMAudioSyncSignal(void*); + static AuxiliaryTask mAudioSyncSignalTask; + static int countInstances; + static SC_SyncCondition* staticMAudioSync; + +private: + uint32 mSCBufLength; +}; + +AuxiliaryTask SC_BelaDriver::mAudioSyncSignalTask; +int SC_BelaDriver::countInstances; +SC_SyncCondition* SC_BelaDriver::staticMAudioSync; +SC_BelaDriver* mBelaDriverInstance = 0; + +SC_AudioDriver* SC_NewAudioDriver(struct World* inWorld) { + if (mBelaDriverInstance != 0) { + scprintf("Warning: SC_NewAudioDriver called with an existing SC_BelaDriver instance.\n"); + } + + mBelaDriverInstance = new SC_BelaDriver(inWorld); + return mBelaDriverInstance; +} + +SC_BelaDriver::SC_BelaDriver(struct World* inWorld): SC_AudioDriver(inWorld) { + mStartHostSecs = 0; + mSCBufLength = inWorld->mBufLength; + + staticMAudioSync = &mAudioSync; + ++countInstances; + if (countInstances != 1) { + fprintf(stderr, "Error: there are %d instances of SC_BelaDriver running at the same time. Exiting\n", + countInstances); + exit(1); + } +} + +SC_BelaDriver::~SC_BelaDriver() { + // Clean up any resources allocated for audio + Bela_cleanupAudio(); + scprintf("SC_BelaDriver: >>Bela_cleanupAudio\n"); + --countInstances; + mBelaDriverInstance = 0; +} + +static float gBelaSampleRate; +// Return true on success; returning false halts the program. +bool sc_belaSetup(BelaContext* belaContext, void* userData) { + // cast void pointer + // SC_BelaDriver *belaDriver = (SC_BelaDriver*) userData; + gBelaSampleRate = belaContext->audioSampleRate; + return true; +} + +void sc_belaRender(BelaContext* belaContext, void* userData) { + SC_BelaDriver* driver = (SC_BelaDriver*)userData; + + driver->BelaAudioCallback(belaContext); +} + +void sc_belaAudioThreadDone(BelaContext*, void* userData) { + SC_BelaDriver* driver = (SC_BelaDriver*)userData; + if (driver) + driver->SignalReceived(0); +} + +void sc_belaSignal(int arg) { + if (mBelaDriverInstance != 0) + mBelaDriverInstance->SignalReceived(arg); +} + +void sc_SetDenormalFlags(); + +void SC_BelaDriver::BelaAudioCallback(BelaContext* belaContext) { + struct timespec tspec; + + sc_SetDenormalFlags(); + World* world = mWorld; + // add a pointer to belaWorld + // mWorld->mBelaContext = belaContext; + world->mBelaContext = belaContext; + + // NOTE: code here is adapted from the SC_Jack.cpp, the version not using the DLL + + // Use Xenomai-friendly clock_gettime() -- note that this requires a -wrap argument to build + __wrap_clock_gettime(CLOCK_HOST_REALTIME, &tspec); + + double hostSecs = (double)tspec.tv_sec + (double)tspec.tv_nsec * 1.0e-9; + double sampleTime = static_cast(belaContext->audioFramesElapsed); + + if (mStartHostSecs == 0) { + mStartHostSecs = hostSecs; + mStartSampleTime = sampleTime; + } else { + double instSampleRate = (sampleTime - mPrevSampleTime) / (hostSecs - mPrevHostSecs); + double smoothSampleRate = mSmoothSampleRate; + smoothSampleRate = smoothSampleRate + 0.002 * (instSampleRate - smoothSampleRate); + if (fabs(smoothSampleRate - mSampleRate) > 10.) { + smoothSampleRate = mSampleRate; + } + mOSCincrement = (int64)(mOSCincrementNumerator / smoothSampleRate); + mSmoothSampleRate = smoothSampleRate; + } + + mPrevHostSecs = hostSecs; + mPrevSampleTime = sampleTime; + + try { + mFromEngine.Free(); + mToEngine.Perform(); + mOscPacketsToEngine.Perform(); + + const uint32_t numInputs = belaContext->audioInChannels; + const uint32_t numOutputs = belaContext->audioOutChannels; + + int numSamples = NumSamplesPerCallback(); + int bufFrames = mWorld->mBufLength; + int numBufs = numSamples / bufFrames; + + float* inBuses = mWorld->mAudioBus + mWorld->mNumOutputs * bufFrames; + float* outBuses = mWorld->mAudioBus; + int32* inTouched = mWorld->mAudioBusTouched + mWorld->mNumOutputs; + int32* outTouched = mWorld->mAudioBusTouched; + + int minInputs = sc_min(numInputs, mWorld->mNumInputs); + int minOutputs = sc_min(numOutputs, mWorld->mNumOutputs); + + int anaInputs = 0; + if (numInputs < (int)mWorld->mNumInputs) { + anaInputs = sc_min(belaContext->analogInChannels, (int)(mWorld->mNumInputs - numInputs)); + } + int anaOutputs = 0; + if (numOutputs < (int)mWorld->mNumOutputs) { + anaOutputs = sc_min(belaContext->analogOutChannels, (int)(mWorld->mNumOutputs - numOutputs)); + } + + int bufFramePos = 0; + + // THIS IS TO DO LATER -- LOOK AT CACHEING AND CONSTING TO IMPROVE EFFICIENCY + // cache I/O buffers + // for (int i = 0; i < minInputs; ++i) { + // inBuffers[i] = (sc_jack_sample_t*)jack_port_get_buffer(inPorts[i], numSamples); + //} + // + // for (int i = 0; i < minOutputs; ++i) { + // outBuffers[i] = (sc_jack_sample_t*)jack_port_get_buffer(outPorts[i], numSamples); + //} + + // main loop + int64 oscTime = mOSCbuftime = + ((int64)(tspec.tv_sec + kSECONDS_FROM_1900_to_1970) << 32) + (int64)(tspec.tv_nsec * kNanosToOSCunits); + + int64 oscInc = mOSCincrement; + double oscToSamples = mOSCtoSamples; + + // clear out anything left over in audioOut buffer + for (int i = 0; i < belaContext->audioFrames * belaContext->audioOutChannels; i++) { + belaContext->audioOut[i] = 0; + } + + for (int i = 0; i < numBufs; ++i, mWorld->mBufCounter++, bufFramePos += bufFrames) { + int32 bufCounter = mWorld->mBufCounter; + int32* tch; + + // copy+touch inputs + tch = inTouched; + memcpy(inBuses, belaContext->audioIn, sizeof(belaContext->audioIn[0]) * bufFrames * minInputs); + for (int k = 0; k < minInputs; ++k) { + *tch++ = bufCounter; + } + + memcpy(inBuses + minInputs * bufFrames, belaContext->analogIn, + sizeof(belaContext->analogIn[0]) * bufFrames * anaInputs); + for (int k = minInputs; k < (minInputs + anaInputs); ++k) { + *tch++ = bufCounter; + } + + // run engine + int64 schedTime; + int64 nextTime = oscTime + oscInc; + + while ((schedTime = mScheduler.NextTime()) <= nextTime) { + float diffTime = (float)(schedTime - oscTime) * oscToSamples + 0.5; + float diffTimeFloor = floor(diffTime); + world->mSampleOffset = (int)diffTimeFloor; + world->mSubsampleOffset = diffTime - diffTimeFloor; + + if (world->mSampleOffset < 0) + world->mSampleOffset = 0; + else if (world->mSampleOffset >= world->mBufLength) + world->mSampleOffset = world->mBufLength - 1; + + SC_ScheduledEvent event = mScheduler.Remove(); + event.Perform(); + } + + world->mSampleOffset = 0; + world->mSubsampleOffset = 0.f; + World_Run(world); + + // copy touched outputs + tch = outTouched; + + for (int k = 0; k < minOutputs; ++k) { + if (*tch++ == bufCounter) { + memcpy(belaContext->audioOut + k * bufFrames, outBuses + k * bufFrames, + sizeof(belaContext->audioOut[0]) * bufFrames); + } + } + + for (int k = minOutputs; k < (minOutputs + anaOutputs); ++k) { + if (*tch++ == bufCounter) { + unsigned int analogChannel = k - minOutputs; // starting at 0 + memcpy(belaContext->analogOut + analogChannel * bufFrames, outBuses + k * bufFrames, + sizeof(belaContext->analogOut[0]) * bufFrames); + } + } + + // advance OSC time + mOSCbuftime = oscTime = nextTime; + } + } catch (std::exception& exc) { + scprintf("SC_BelaDriver: exception in real time: %s\n", exc.what()); + } catch (...) { + scprintf("SC_BelaDriver: unknown exception in real time\n"); + } + + // this avoids Xenomai mode switches in the audio thread ... + Bela_scheduleAuxiliaryTask(mAudioSyncSignalTask); +} + +void SC_BelaDriver::staticMAudioSyncSignal(void*) { + // ... but mode switches are still happening here, in a lower priority thread. + // FIXME: this triggers a mode switch in Xenomai. + staticMAudioSync->Signal(); +} +// ==================================================================== + +bool SC_BelaDriver::DriverSetup(int* outNumSamples, double* outSampleRate) { + BelaInitSettings* settings = Bela_InitSettings_alloc(); + Bela_defaultSettings(settings); + settings->setup = sc_belaSetup; + settings->render = sc_belaRender; +#if (BELA_MAJOR_VERSION == 1 && BELA_MINOR_VERSION >= 8) || (BELA_MAJOR_VERSION > 1) + // if the feature is supported on Bela, add a callback to be called when + // the audio thread stops. This is useful e.g.: to gracefully exit from + // scsynth when pressing the Bela button + settings->audioThreadDone = sc_belaAudioThreadDone; +#endif // BELA >= 1.8 + settings->interleave = 0; + settings->uniformSampleRate = 1; + settings->analogOutputsPersist = 0; + + if (mPreferredHardwareBufferFrameSize) { + settings->periodSize = mPreferredHardwareBufferFrameSize; + } + if (settings->periodSize != mSCBufLength) { + scprintf("Warning in SC_BelaDriver::DriverSetup(): hardware buffer size (%i) different from SC audio buffer " + "size (%i). Changed the hardware buffer size to be equal to the SC audio buffer size .\n", + settings->periodSize, mSCBufLength); + settings->periodSize = mSCBufLength; + } + // note that Bela doesn't give us an option to choose samplerate, since it's baked-in. + // This can be retrieved in sc_belaSetup() + + // configure the number of analog channels - this will determine their internal samplerate + settings->useAnalog = 0; + + // explicitly requested number of analog channels + int numAnalogIn = mWorld->mBelaAnalogInputChannels; + int numAnalogOut = mWorld->mBelaAnalogOutputChannels; + + int extraAudioIn = mWorld->mNumInputs - settings->numAudioInChannels; + int extraAudioOut = mWorld->mNumOutputs - settings->numAudioOutChannels; + // if we need more audio channels than there actually are audio + // channels, make sure we have some extra analogs + if (extraAudioIn > 0) { + numAnalogIn = sc_max(numAnalogIn, extraAudioIn); + } + if (extraAudioOut > 0) { + numAnalogOut = sc_max(numAnalogOut, extraAudioOut); + } + + // snap the number of requested analog channels to the 0, 4, 8. + // 4 will give same actual sample rate as audio, 8 will give half of it. + if (numAnalogIn > 0) { + if (numAnalogIn < 5) { + numAnalogIn = 4; + } else { + numAnalogIn = 8; + } + } + + if (numAnalogOut > 0) { + if (numAnalogOut < 5) { + numAnalogOut = 4; + } else { + numAnalogOut = 8; + } + } + + // final check: right now the number of analog output channels on bela needs to be the same as analog input + // channels. this is likely to change in the future, that is why we factored it out + if (numAnalogOut != numAnalogIn) { + // Chosing the maximum of the two + numAnalogOut = sc_max(numAnalogOut, numAnalogIn); + numAnalogIn = numAnalogOut; + printf("Number of analog input channels must match number of analog outputs. Using %u for both\n", numAnalogIn); + } + settings->numAnalogInChannels = numAnalogOut; + settings->numAnalogOutChannels = numAnalogIn; + + if (settings->numAnalogInChannels > 0 || settings->numAnalogOutChannels > 0) { + settings->useAnalog = 1; + } + + // enable the audio expander capelet for the first few "analog as audio" channels + // inputs and ... + for (int n = 0; n < extraAudioIn; ++n) { + printf("Using analog in %d as audio in %d\n", n, n + settings->numAudioInChannels); + settings->audioExpanderInputs |= (1 << n); + } + + // ... outputs + for (int n = 0; n < extraAudioOut; ++n) { + printf("Using analog out %d as audio out %d\n", n, n + settings->numAudioOutChannels); + settings->audioExpanderOutputs |= (1 << n); + } + + // configure the number of digital channels + settings->useDigital = 0; + + if (mWorld->mBelaDigitalChannels > 0) { + settings->numDigitalChannels = mWorld->mBelaDigitalChannels; + settings->useDigital = 1; + } + if ((mWorld->mBelaHeadphoneLevel >= -63.5) + && (mWorld->mBelaHeadphoneLevel <= 0.)) { // headphone output level (0dB max; -63.5dB min) + settings->headphoneLevel = mWorld->mBelaHeadphoneLevel; + } + if ((mWorld->mBelaPGAGainLeft >= 0) && (mWorld->mBelaPGAGainLeft <= 59.5)) { // (0db min; 59.5db max) + settings->pgaGain[0] = mWorld->mBelaPGAGainLeft; + } + if ((mWorld->mBelaPGAGainRight >= 0) && (mWorld->mBelaPGAGainRight <= 59.5)) { // (0db min; 59.5db max) + settings->pgaGain[1] = mWorld->mBelaPGAGainRight; + } + + if (mWorld->mBelaSpeakerMuted) { + settings->beginMuted = 1; + } else { + settings->beginMuted = 0; + } + if ((mWorld->mBelaDACLevel >= -63.5) && (mWorld->mBelaDACLevel <= 0.)) { // (0dB max; -63.5dB min) + settings->dacLevel = mWorld->mBelaDACLevel; + } + if ((mWorld->mBelaADCLevel >= -12) && (mWorld->mBelaADCLevel <= 0.)) { // (0dB max; -12dB min) + settings->adcLevel = mWorld->mBelaADCLevel; + } + + settings->numMuxChannels = mWorld->mBelaNumMuxChannels; + + if ((mWorld->mBelaPRU == 0) || (mWorld->mBelaPRU == 1)) { + settings->pruNumber = mWorld->mBelaPRU; + } + + scprintf("SC_BelaDriver: >>DriverSetup - Running on PRU (%i)\nConfigured with \n (%i) analog input and (%i) analog " + "output channels, (%i) digital channels, and (%i) multiplexer channels.\n HeadphoneLevel (%f dB), " + "pga_gain_left (%f dB) and pga_gain_right (%f dB)\n DAC Level (%f dB), ADC Level (%f dB)\n", + settings->pruNumber, settings->numAnalogInChannels, settings->numAnalogOutChannels, + settings->numDigitalChannels, settings->numMuxChannels, settings->headphoneLevel, settings->pgaGain[0], + settings->pgaGain[1], settings->dacLevel, settings->adcLevel); + if (settings->beginMuted == 1) { + scprintf("Speakers are muted.\n"); + } else { + scprintf("Speakers are not muted.\n"); + } + + settings->verbose = mWorld->mVerbosity; + // Initialise the PRU audio device. This function prepares audio rendering in Bela. It should be called from main() + // sometime after command line option parsing has finished. It will initialise the rendering system, which in the + // process will result in a call to the user-defined setup() function. + if (Bela_initAudio(settings, this) != 0) { + scprintf("Error in SC_BelaDriver::DriverSetup(): unable to initialise audio\n"); + return false; + } + mAudioSyncSignalTask = Bela_createAuxiliaryTask( + staticMAudioSyncSignal, 90, "mAudioSyncSignalTask"); // needs to be created after the call to Bela_initAudio() + if (!mAudioSyncSignalTask) { + fprintf(stderr, "Error: unable to create Bela auxiliary task\n"); + exit(1); + } + + *outNumSamples = settings->periodSize; + *outSampleRate = gBelaSampleRate; + Bela_InitSettings_free(settings); + + // Set up interrupt handler to catch Control-C and SIGTERM + signal(SIGINT, sc_belaSignal); + signal(SIGTERM, sc_belaSignal); + + return true; +} + +bool SC_BelaDriver::DriverStart() { + SetPrintFunc((PrintFunc)rt_vprintf); // Use Xenomai's realtime-friendly printing function +#ifdef XENOMAI_SKIN_native + rt_print_auto_init(1); // Make sure the buffers for rt_vprintf are actually initialized. +#endif + if (Bela_startAudio()) { + scprintf("Error in SC_BelaDriver::DriverStart(): unable to start real-time audio\n"); + return false; + } + return true; +} + +bool SC_BelaDriver::DriverStop() { + Bela_stopAudio(); + return true; +} + +void SC_BelaDriver::SignalReceived(int signal) { + scprintf("SC_BelaDriver: signal received: %d; terminating\n", signal); + mWorld->hw->mTerminating = true; + mWorld->hw->mQuitProgram->post(); +} diff --git a/server/scsynth/SC_CoreAudio.cpp b/server/scsynth/SC_CoreAudio.cpp index 81371514648..45abd7f50a3 100644 --- a/server/scsynth/SC_CoreAudio.cpp +++ b/server/scsynth/SC_CoreAudio.cpp @@ -359,6 +359,9 @@ void SC_AudioDriver::RunThread() { while (mRunThreadFlag) { // wait for sync mAudioSync.WaitNext(); +#ifdef BELA + rt_print_flush_buffers(); +#endif /* BELA */ reinterpret_cast(mWorld->mNRTLock)->lock(); @@ -661,7 +664,7 @@ bool SC_CoreAudioDriver::DriverSetup(int* outNumSamplesPerCallback, double* outS count = sizeof(mOutputDevice); // get the output device: // err = AudioHardwareGetProperty(kAudioHardwarePropertyDefaultOutputDevice, &count, (void *) & - // mOutputDevice); + //mOutputDevice); propertyAddress.mSelector = kAudioHardwarePropertyDefaultOutputDevice; @@ -678,7 +681,7 @@ bool SC_CoreAudioDriver::DriverSetup(int* outNumSamplesPerCallback, double* outS if (mInputDevice == kAudioDeviceUnknown) { count = sizeof(mInputDevice); // err = AudioHardwareGetProperty(kAudioHardwarePropertyDefaultInputDevice, &count, (void *) & - // mInputDevice); + //mInputDevice); propertyAddress.mSelector = kAudioHardwarePropertyDefaultInputDevice; @@ -1624,16 +1627,16 @@ bool SC_CoreAudioDriver::DriverStart() { try { if (UseSeparateIO()) { - // err = AudioDeviceAddIOProc(mOutputDevice, appIOProc, (void *) this); // setup Out device with an - // IO proc + // err = AudioDeviceAddIOProc(mOutputDevice, appIOProc, (void *) this); // setup Out device with an IO + //proc err = AudioDeviceCreateIOProcID(mOutputDevice, appIOProcFunc, (void*)this, &mOutputID); if (err != kAudioHardwareNoError) { scprintf("AudioDeviceAddIOProc failed %s %d\n", &err, (int)err); return false; } - // err = AudioDeviceAddIOProc(mInputDevice, appIOProcSeparateIn, (void *) this); // setup In - // device with an IO proc + // err = AudioDeviceAddIOProc(mInputDevice, appIOProcSeparateIn, (void *) this); // setup In device + //with an IO proc err = AudioDeviceCreateIOProcID(mInputDevice, appIOProcSeparateIn, (void*)this, &mInputID); if (err != kAudioHardwareNoError) { @@ -1772,8 +1775,8 @@ bool SC_CoreAudioDriver::DriverStart() { AudioObjectSetPropertyData(kAudioObjectSystemObject, &theAddress, 0, NULL, sizeof(CFRunLoopRef), &theRunLoop); // for now no spotting of hardware changes, assumption is that ServerOptions inviolate. However, if a device was - // unplugged, could react to loss of that device by switching to system default? note that the number of listeners - // is stripped down to only one for now, to react to headphone swaps in the case of Built-in Output + // unplugged, could react to loss of that device by switching to system default? note that the number of listeners is + // stripped down to only one for now, to react to headphone swaps in the case of Built-in Output AddDeviceListeners(mOutputDevice, this); return true; diff --git a/server/scsynth/SC_CoreAudio.h b/server/scsynth/SC_CoreAudio.h index bd012073158..a95e9150e62 100644 --- a/server/scsynth/SC_CoreAudio.h +++ b/server/scsynth/SC_CoreAudio.h @@ -34,6 +34,7 @@ #define SC_AUDIO_API_PORTAUDIO 3 #define SC_AUDIO_API_AUDIOUNITS 4 #define SC_AUDIO_API_COREAUDIOIPHONE 5 +#define SC_AUDIO_API_BELA 6 #ifdef SC_IPHONE # define SC_AUDIO_API SC_AUDIO_API_COREAUDIOIPHONE diff --git a/server/scsynth/SC_World.cpp b/server/scsynth/SC_World.cpp index 1f9689cbe98..4d5915459ae 100644 --- a/server/scsynth/SC_World.cpp +++ b/server/scsynth/SC_World.cpp @@ -159,6 +159,25 @@ void sc_SetDenormalFlags() { #if BOOST_HW_SIMD_X86 >= BOOST_HW_SIMD_X86_SSE_VERSION _MM_SET_FLUSH_ZERO_MODE(_MM_FLUSH_ZERO_ON); _mm_setcsr(_mm_getcsr() | 0x40); // DAZ +#elif defined(__VFP_FP__) + // the Cortex A8, along with the SIMD Neon unit. + // This function turns on "fast mode" by enabling + // flushing denormals to zero on the VFP. + // The NEON already flushe to zero, so it requires + // no specific settings. + /* This code is from math-neon/math_runfast.c + Copyright (c) 2015 Lachlan Tychsen-Smith (lachlan.ts@gmail.com) + MIT License + */ + static const unsigned int x = 0x04086060; + static const unsigned int y = 0x03000000; + int r; + asm volatile("fmrx %0, fpscr \n\t" // r0 = FPSCR + "and %0, %0, %1 \n\t" // r0 = r0 & 0x04086060 + "orr %0, %0, %2 \n\t" // r0 = r0 | 0x03000000 + "fmxr fpscr, %0 \n\t" // FPSCR = r0 + : "=r"(r) + : "r"(x), "r"(y)); #endif } @@ -390,6 +409,21 @@ World* World_New(WorldOptions* inOptions) { } else { world->hw->mPassword[0] = 0; } +#ifdef BELA + // world->mBelaAnalogChannels = inOptions->mBelaAnalogChannels; + // scprintf("INFO: WORLD: number of analog channels %i.\n", world->mBelaAnalogChannels ); + world->mBelaAnalogInputChannels = inOptions->mBelaAnalogInputChannels; + world->mBelaAnalogOutputChannels = inOptions->mBelaAnalogOutputChannels; + world->mBelaDigitalChannels = inOptions->mBelaDigitalChannels; + world->mBelaHeadphoneLevel = inOptions->mBelaHeadphoneLevel; + world->mBelaPGAGainLeft = inOptions->mBelaPGAGainLeft; + world->mBelaPGAGainRight = inOptions->mBelaPGAGainRight; + world->mBelaSpeakerMuted = inOptions->mBelaSpeakerMuted; + world->mBelaDACLevel = inOptions->mBelaDACLevel; + world->mBelaADCLevel = inOptions->mBelaADCLevel; + world->mBelaNumMuxChannels = inOptions->mBelaNumMuxChannels; + world->mBelaPRU = inOptions->mBelaPRU; +#endif #ifdef __APPLE__ world->hw->mInputStreamsEnabled = inOptions->mInputStreamsEnabled; diff --git a/server/scsynth/scsynth_main.cpp b/server/scsynth/scsynth_main.cpp index f9489d188cf..6111f3cedc4 100644 --- a/server/scsynth/scsynth_main.cpp +++ b/server/scsynth/scsynth_main.cpp @@ -87,6 +87,20 @@ void Usage() { " -I \n" " -O \n" #endif +#ifdef BELA + // " -J \n" + " -J \n" + " -K \n" + " -G \n" + " -Q (0dB max, -63.5dB min)\n" + " -X \n" + " -Y \n" + " -s \n" + " -x \n" + " -y \n" + " -g \n" + " -T \n" +#endif #if (_POSIX_MEMLOCK - 0) >= 200112L " -L enable memory locking\n" #endif @@ -147,9 +161,27 @@ int scsynth_main(int argc, char** argv) { WorldOptions options; +#ifdef BELA + // defaults + options.mBelaAnalogInputChannels = 0; + options.mBelaAnalogOutputChannels = 0; + options.mBelaDigitalChannels = 0; + options.mBelaHeadphoneLevel = -6.; + options.mBelaPGAGainLeft = 20; + options.mBelaPGAGainRight = 20; + options.mBelaSpeakerMuted = 0; + options.mBelaADCLevel = 0; + options.mBelaDACLevel = 0; + options.mBelaNumMuxChannels = 0; + options.mBelaPRU = 1; +#endif + for (int i = 1; i < argc;) { +#ifdef BELA +#define EXTRA_OPTIONS "JKGQXYxygTO" +#endif // BELA if (argv[i][0] != '-' || argv[i][1] == 0 - || strchr("utBaioczblndpmwZrCNSDIOsMHvVRUhPL", argv[i][1]) == nullptr) { + || strchr("utBaioczblndpmwZrCNSDIOsMHvVRUhPL" EXTRA_OPTIONS, argv[i][1]) == nullptr) { scprintf("ERROR: Invalid option %s\n", argv[i]); Usage(); } @@ -280,6 +312,48 @@ int scsynth_main(int argc, char** argv) { options.mMemoryLocking = false; #endif break; +#ifdef BELA + case 'J': + checkNumArgs(2); + options.mBelaAnalogInputChannels = atoi(argv[j + 1]); + break; + case 'K': + checkNumArgs(2); + options.mBelaAnalogOutputChannels = atoi(argv[j + 1]); + break; + case 'G': + checkNumArgs(2); + options.mBelaDigitalChannels = atoi(argv[j + 1]); + break; + case 'Q': + checkNumArgs(2); + options.mBelaHeadphoneLevel = atof(argv[j + 1]); + break; + case 'X': + checkNumArgs(2); + options.mBelaPGAGainLeft = atof(argv[j + 1]); + break; + case 'Y': + checkNumArgs(2); + options.mBelaPGAGainRight = atof(argv[j + 1]); + break; + case 'x': + checkNumArgs(2); + options.mBelaDACLevel = atof(argv[j + 1]); + break; + case 'y': + checkNumArgs(2); + options.mBelaADCLevel = atof(argv[j + 1]); + break; + case 'g': + checkNumArgs(2); + options.mBelaNumMuxChannels = atoi(argv[j + 1]); + break; + case 'T': + checkNumArgs(2); + options.mBelaPRU = atoi(argv[j + 1]); + break; +#endif case 'V': checkNumArgs(2); options.mVerbosity = atoi(argv[j + 1]);