diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 56ef1fd..c819939 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -19,6 +19,12 @@ jobs: - name: Get specific commits of git submodules run: | sh -ex ./submod.sh + - name: Apply patches + run: | + # Put git hash in startup message + sed -i "s/Loading.../$(date +%Y%m%d)-$(git rev-parse --short HEAD)/g" src/userinterface.cpp + # https://github.com/rsta2/circle/discussions/427#discussioncomment-11198505 + sed -i -e 's|TTLShort = 120|TTLShort = 15|g' circle-stdlib/libs/circle/include/circle/net/mdnspublisher.h - name: Install toolchains run: | set -ex @@ -27,6 +33,12 @@ jobs: wget -q https://developer.arm.com/-/media/Files/downloads/gnu-a/10.3-2021.07/binrel/gcc-arm-10.3-2021.07-x86_64-arm-none-eabi.tar.xz tar xf *-arm-none-eabi.tar.xz mkdir -p kernels + - name: Build for Raspberry Pi 5 + run: | + set -ex + export PATH=$(readlink -f ./gcc-*aarch64-none*/bin/):$PATH + RPI=5 bash -ex build.sh + cp ./src/kernel*.img ./kernels/ - name: Build for Raspberry Pi 4 run: | set -ex @@ -51,7 +63,7 @@ jobs: export PATH=$(readlink -f ./gcc-*arm-none*/bin/):$PATH RPI=1 bash -ex build.sh cp ./src/kernel*.img ./kernels/ - - name: Get Raspberry Pi boot files + - name: Get Raspberry Pi boot files and WLAN firmware run: | set -ex export PATH=$(readlink -f ./gcc-*aarch64-none*/bin/):$PATH @@ -68,12 +80,18 @@ jobs: cd sdcard cp ../kernels/* . || true cd - + # WLAN firmware + cp circle-stdlib/libs/circle/addon/wlan/sample/hello_wlan/wpa_supplicant.conf sdcard/ + mkdir -p sdcard/firmware + cd sdcard/firmware + make -f ../../circle-stdlib/libs/circle/addon/wlan/firmware/Makefile + cd - - name: Get performance files run: | git clone https://github.com/Banana71/Soundplantage --depth 1 # depth 1 means only the latest commit cp -r ./Soundplantage/performance ./Soundplantage/*.pdf ./sdcard/ cd sdcard - zip -r ../MiniDexed_$GITHUB_RUN_NUMBER_$(date +%Y-%m-%d)-$(git rev-parse --short HEAD).zip * + zip -r ../MiniDexed_$GITHUB_RUN_NUMBER_$(date +%Y%m%d)-$(git rev-parse --short HEAD).zip * echo "artifactName=MiniDexed_$GITHUB_RUN_NUMBER_$(date +%Y-%m-%d)-$(git rev-parse --short HEAD)" >> $GITHUB_ENV cd - - uses: actions/upload-artifact@v3 diff --git a/.github/workflows/pr-comment.yml b/.github/workflows/pr-comment.yml index c5c8ebb..863a6d8 100644 --- a/.github/workflows/pr-comment.yml +++ b/.github/workflows/pr-comment.yml @@ -1,64 +1,32 @@ -# https://nightly.link/ -# https://github.com/oprypin/nightly.link/blob/master/.github/workflows/pr-comment.yml +# https://github.com/subsurface/subsurface/blob/master/.github/workflows/artifact-links.yml + +name: Add artifact links to pull request -name: Comment on pull request on: workflow_run: - workflows: ['Build'] + workflows: ["Build"] types: [completed] + jobs: - pr_comment: - if: github.event.workflow_run.event == 'pull_request' && github.event.workflow_run.conclusion == 'success' - runs-on: ubuntu-latest + artifacts-url-comments: + name: Add artifact links to PR and issues + runs-on: ubuntu-22.04 + + # Restrict permissions for the GITHUB_TOKEN, https://docs.github.com/en/actions/using-jobs/assigning-permissions-to-jobs + permissions: + issues: write + pull-requests: write + actions: read + steps: - - uses: actions/github-script@v6 - with: - # This snippet is public-domain, taken from - # https://github.com/oprypin/nightly.link/blob/master/.github/workflows/pr-comment.yml - script: | - async function upsertComment(owner, repo, issue_number, purpose, body) { - const {data: comments} = await github.rest.issues.listComments( - {owner, repo, issue_number}); - - const marker = ``; - body = marker + "\n" + body; - - const existing = comments.filter((c) => c.body.includes(marker)); - if (existing.length > 0) { - const last = existing[existing.length - 1]; - core.info(`Updating comment ${last.id}`); - await github.rest.issues.updateComment({ - owner, repo, - body, - comment_id: last.id, - }); - } else { - core.info(`Creating a comment in issue / PR #${issue_number}`); - await github.rest.issues.createComment({issue_number, body, owner, repo}); - } - } - - const {owner, repo} = context.repo; - const run_id = ${{github.event.workflow_run.id}}; - - const pull_requests = ${{ toJSON(github.event.workflow_run.pull_requests) }}; - if (!pull_requests.length) { - return core.error("This workflow doesn't match any pull requests!"); - } - - const artifacts = await github.paginate( - github.rest.actions.listWorkflowRunArtifacts, {owner, repo, run_id}); - if (!artifacts.length) { - return core.error(`No artifacts found`); - } - let body = `Download the artifacts for this pull request:\n`; - for (const art of artifacts) { - body += `\n* [${art.name}.zip](https://nightly.link/${owner}/${repo}/actions/artifacts/${art.id}.zip)`; - } - - core.info("Review thread message body:", body); - - for (const pr of pull_requests) { - await upsertComment(owner, repo, pr.number, - "nightly-link", body); - } + - name: Add artifact links to PR and issues + if: github.event.workflow_run.event == 'pull_request' + uses: tonyhallett/artifacts-url-comments@0965ff1a7ae03c5c1644d3c30f956effea4e05ef # v1.1.0 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + with: + prefix: "Build for testing:" + suffix: "Use at your own risk." + format: name + addTo: pull + errorNoArtifacts: false diff --git a/README.md b/README.md index 8623d9b..68cdb02 100644 --- a/README.md +++ b/README.md @@ -1,15 +1,19 @@ -# MiniDexed ![](https://github.com/probonopd/MiniDexed/actions/workflows/build.yml/badge.svg) +# MiniDexed ![Github Build Status](https://github.com/probonopd/MiniDexed/actions/workflows/build.yml/badge.svg) ![minidexed](https://user-images.githubusercontent.com/2480569/161813414-bb156a1c-efec-44c0-802a-8926412a08e0.jpg) -MiniDexed is a FM synthesizer closely modeled on the famous DX7 by a well-known Japanese manufacturer running on a bare metal Raspberry Pi (without a Linux kernel or operating system). On Raspberry Pi 2 and larger, it can run 8 tone generators, not unlike the TX816/TX802 (8 DX7 instances without the keyboard in one box). [Featured by HACKADAY](https://hackaday.com/2022/04/19/bare-metal-gives-this-pi-some-classic-synths/), [adafruit](https://blog.adafruit.com/2022/04/25/free-yamaha-dx7-synth-emulator-on-a-raspberry-pi/), and [Synth Geekery](https://www.youtube.com/watch?v=TDSy5nnm0jA). +MiniDexed is a FM synthesizer closely modeled on the famous DX7 by a well-known Japanese manufacturer running on a bare metal Raspberry Pi (without a Linux kernel or operating system). On Raspberry Pi 2 and larger, it can run 8 tone generators, not unlike the TX816/TX802 (8 DX7 instances without the keyboard in one box). [Featured by HACKADAY](https://hackaday.com/2022/04/19/bare-metal-gives-this-pi-some-classic-synths/), [Adafruit](https://blog.adafruit.com/2022/04/25/free-yamaha-dx7-synth-emulator-on-a-raspberry-pi/), and [Synth Geekery](https://www.youtube.com/watch?v=TDSy5nnm0jA). + +## Demo songs + +Listen to some examples made with MiniDexed by Banana71 [here](https://soundcloud.com/soundplantage/sets/minidexed2). ## Features - [x] Uses [Synth_Dexed](https://codeberg.org/dcoredump/Synth_Dexed) with [circle-stdlib](https://github.com/smuehlst/circle-stdlib) - [x] SD card contents can be downloaded from [GitHub Releases](../../releases) - [x] Runs on all Raspberry Pi models (except Pico); see below for details -- [x] Produces sound on the headphone jack, HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) (better), or a [dedicated DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2c-dac) (best) +- [x] Produces sound on the headphone jack, HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) (better), or a [dedicated DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2s-dac) (best) - [x] Supports multiple voices through Program Change and Bank Change LSB/MSB MIDI messages - [x] Loads voices from `.syx` files from SD card (e.g., using `getsysex.sh` or from [Dexed_cart_1.0.zip](http://hsjp.eu/downloads/Dexed/Dexed_cart_1.0.zip)) - [x] Menu structure on optional [HD44780 display](https://www.berrybase.de/sensoren-module/displays/alphanumerische-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) and rotary encoder @@ -18,48 +22,49 @@ MiniDexed is a FM synthesizer closely modeled on the famous DX7 by a well-known - [x] Allows to configure multiple Dexed instances through `performance.ini` files (e.g., [converted](https://github.com/BobanSpasic/MDX_Vault) from DX1, DX5, TX816, DX7II, TX802) - [x] Compressor effect - [x] Reverb effect -- [x] Voices can be edited over MIDI, e.g., using the [synthmata](https://synthmata.github.io/volca-fm/) online editor (requires [additional hardware](https://github.com/probonopd/MiniDexed/wiki/Hardware#usb-midi-device)) +- [x] Voices can be edited over MIDI, e.g., using the [synthmata](https://synthmata.github.io/volca-fm/) online editor (requires [additional hardware](https://github.com/probonopd/MiniDexed/wiki/Hardware#usb-midi-devices)) ## Introduction Video about this project by [Floyd Steinberg](https://www.youtube.com/watch?v=Z3t94ceMHJo): -[![](https://i.ytimg.com/vi/Z3t94ceMHJo/sddefault.jpg)](https://www.youtube.com/watch?v=Z3t94ceMHJo) +[![YouTube Video about MiniDexed (Floyd Steinberg)](https://i.ytimg.com/vi/Z3t94ceMHJo/sddefault.jpg)](https://www.youtube.com/watch?v=Z3t94ceMHJo) ## System Requirements -* Raspberry Pi 1, 2, 3, 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2s DAC for audio out). On Raspberry Pi 1 and on Raspberry Pi Zero there will be severely limited functionality (only one tone generator instead of 8) -* A [PCM5102A or PCM5122 based DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2c-dac), HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) for good sound quality. If you don't have this, you can use the headphone jack on the Raspberry Pi but on anything but the Raspberry 4 the sound quality will be seriously limited -* Optionally (but highly recommended), an [LCDC1602 Display](https://www.berrybase.de/en/sensors-modules/displays/alphanumeric-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) (with or without i2c "backpack" board) and a [KY-040 rotary encoder](https://www.berrybase.de/en/components/passive-components/potentiometer/rotary-encoder/drehregler/rotary-encoder-mit-breakoutboard-ohne-gewinde-und-mutter) +- Raspberry Pi 1, 2, 3, 4, or 400. Raspberry Pi Zero and Zero 2 can be used but need HDMI or a supported i2s DAC for audio out. On Raspberry Pi 1 and on Raspberry Pi Zero there will be severely limited functionality (only one tone generator instead of 8) +- Raspberry Pi 5 can be used but currently support is experimental: HDMI sound and USB Gadget mode are not available yet, and it is not clear if there are implications for cooling from running MiniDexed. Also, MiniDexed is currently not taking advantage of the higher processing power of the Raspberry Pi 5 yet. *Hence, you may consider using one of the less expensive, older Raspberry Pi boards for your first build.* +- A [PCM5102A or PCM5122 based DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2s-dac), HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) for good sound quality. If you don't have this, you can use the headphone jack on the Raspberry Pi but on anything but the Raspberry 4 the sound quality will be seriously limited +- Optionally (but highly recommended), an [LCDC1602 Display](https://www.berrybase.de/en/sensors-modules/displays/alphanumeric-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) (with or without i2c "backpack" board) and a [KY-040 rotary encoder](https://www.berrybase.de/en/components/passive-components/potentiometer/rotary-encoder/drehregler/rotary-encoder-mit-breakoutboard-ohne-gewinde-und-mutter) ## Usage -* In the case of Raspberry Pi 4, Update the firmware and bootloader to the latest version (not doing this may cause USB reliability issues) -* Download from [GitHub Releases](../../releases) -* Unzip -* Put the files into the root directory of a FAT32 formatted partition on SD/microSD card (Note for small SD cards which are no longer sold: If less than 65525 clusters, you may need to format as FAT16.) -* Put SD/microSD card into Raspberry Pi 1, 2, 3 or 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2c DAC for audio out) -* Attach headphones to the headphone jack using `SoundDevice=pwm` in `minidexed.ini` (default) (poor audio quality) -* Alternatively, attach a PCM5102A or PCM5122 based DAC and select i2c sound output using `SoundDevice=i2s` in `minidexed.ini` (best audio quality) -* Alternatively, attach a HDMI display with sound and select HDMI sound output using `SoundDevice=hdmi` in `minidexed.ini` (this may introduce slight latency) -* Attach a MIDI keyboard via USB (alternatively you can build a circuit that allows you to attach a "traditional" MIDI keyboard using a DIN connector, or use a DIN-MIDI-to-USB adapter) -* If you are using a LCDC1602 with an i2c "backpack" board, then you need to set `LCDI2CAddress=0x27` (or another address your i2c "backpack" board is set to) in `minidexed.ini` -* Boot -* Start playing -* If the system seems to become unresponsive after a few seconds, remove `usbspeed=full` from `cmdline.txt` and repeat ([details](https://github.com/probonopd/MiniDexed/issues/39)) -* Optionally, put voices in `.syx` files onto the SD card (e.g., using `getsysex.sh`) -* See the Wiki for [Menu](https://github.com/probonopd/MiniDexed/wiki/Menu) operation -* For voice programming, use any DX series editor (using MIDI sysex), including Dexed -* For library management, use the dedicated [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software -* If something is unclear or does not work, don't hesitate to [ask](https://github.com/probonopd/MiniDexed/discussions/)! +- In the case of Raspberry Pi 4, Update the firmware and bootloader to the latest version (not doing this may cause USB reliability issues) +- Download from [GitHub Releases](../../releases) +- Unzip +- Put the files into the root directory of a FAT32 formatted partition on SD/microSD card (Note for small SD cards which are no longer sold: If less than 65525 clusters, you may need to format as FAT16.) +- Put SD/microSD card into Raspberry Pi 1, 2, 3 or 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2c DAC for audio out) +- Attach headphones to the headphone jack using `SoundDevice=pwm` in `minidexed.ini` (default) (poor audio quality) +- Alternatively, attach a PCM5102A or PCM5122 based DAC and select i2c sound output using `SoundDevice=i2s` in `minidexed.ini` (best audio quality) +- Alternatively, attach a HDMI display with sound and select HDMI sound output using `SoundDevice=hdmi` in `minidexed.ini` (this may introduce slight latency) +- Attach a MIDI keyboard via USB (alternatively you can build a circuit that allows you to attach a "traditional" MIDI keyboard using a DIN connector, or use a DIN-MIDI-to-USB adapter) +- If you are using a LCDC1602 with an i2c "backpack" board, then you need to set `LCDI2CAddress=0x27` (or another address your i2c "backpack" board is set to) in `minidexed.ini` +- Boot +- Start playing +- If the system seems to become unresponsive after a few seconds, remove `usbspeed=full` from `cmdline.txt` and repeat ([details](https://github.com/probonopd/MiniDexed/issues/39)) +- Optionally, put voices in `.syx` files onto the SD card (e.g., using `getsysex.sh`) +- See the Wiki for [Menu](https://github.com/probonopd/MiniDexed/wiki/Menu) operation +- For voice programming, use any DX series editor (using MIDI sysex), including Dexed +- For library management, use the dedicated [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software +- If something is unclear or does not work, don't hesitate to [ask](https://github.com/probonopd/MiniDexed/discussions/)! ## Pinout All devices on Raspberry Pi GPIOs are **optional**. -![](https://user-images.githubusercontent.com/2480569/166105580-da11481c-8fc7-4375-8ab1-3031ab5c6ad0.png) +![Raspberry Pi Pinout/GPIO Diagram](https://user-images.githubusercontent.com/2480569/166105580-da11481c-8fc7-4375-8ab1-3031ab5c6ad0.png) -Please the the [wiki](https://github.com/probonopd/MiniDexed/wiki) for more information. +Please see the [wiki](https://github.com/probonopd/MiniDexed/wiki) for more information. ## Downloading @@ -71,24 +76,28 @@ Please see the [wiki](https://github.com/probonopd/MiniDexed/wiki/Development#bu ## Contributing -This project lives from the contributions of skilled C++ developers, testers, writers, etc. Please see https://github.com/probonopd/MiniDexed/issues. +This project lives from the contributions of skilled C++ developers, testers, writers, etc. Please see . ## Discussions -We are happy to hear from you. Please join the discussions on https://github.com/probonopd/MiniDexed/discussions. +We are happy to hear from you. Please join the discussions on . ## Documentation -Project documentation is at https://github.com/probonopd/MiniDexed/wiki. +Project documentation is at . ## Acknowledgements This project stands on the shoulders of giants. Special thanks to: -* [raphlinus](https://github.com/raphlinus) for the [MSFA](https://github.com/google/music-synthesizer-for-android) sound engine -* [asb2m10](https://github.com/asb2m10/dexed) for the [Dexed](https://github.com/asb2m10/dexed) software -* [dcoredump](https://github.com/dcoredump) for https://codeberg.org/dcoredump/Synth_Dexed, a port of Dexed for embedded systems -* [rsta2](https://github.com/rsta2) for https://github.com/rsta2/circle, the library to run code on bare metal Raspberry Pi (without a Linux kernel or operating system) and for the bulk of the MiniDexed code -* [smuehlst](https://github.com/smuehlst) for https://github.com/smuehlst/circle-stdlib, a version with Standard C and C++ library support -* [Banana71](https://github.com/Banana71) for the sound design of the [Soundplantage](https://github.com/Banana71/Soundplantage) performances shipped with MiniDexed -* [BobanSpasic](https://github.com/BobanSpasic) for the [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software, [MiniDexed performance converter](https://github.com/BobanSpasic/MDX_PerfConv) and [collection of performances for MiniDexed](https://github.com/BobanSpasic/MDX_Vault) +- [raphlinus](https://github.com/raphlinus) for the [MSFA](https://github.com/google/music-synthesizer-for-android) sound engine +- [asb2m10](https://github.com/asb2m10/dexed) for the [Dexed](https://github.com/asb2m10/dexed) software +- [dcoredump](https://github.com/dcoredump) for [Synth Dexed](https://codeberg.org/dcoredump/Synth_Dexed), a port of Dexed for embedded systems +- [rsta2](https://github.com/rsta2) for [Circle](https://github.com/rsta2/circle), the library to run code on bare metal Raspberry Pi (without a Linux kernel or operating system) and for the bulk of the MiniDexed code +- [smuehlst](https://github.com/smuehlst) for [circle-stdlib](https://github.com/smuehlst/circle-stdlib), a version with Standard C and C++ library support +- [Banana71](https://github.com/Banana71) for the sound design of the [Soundplantage](https://github.com/Banana71/Soundplantage) performances shipped with MiniDexed +- [BobanSpasic](https://github.com/BobanSpasic) for the [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software, [MiniDexed performance converter](https://github.com/BobanSpasic/MDX_PerfConv) and [collection of performances for MiniDexed](https://github.com/BobanSpasic/MDX_Vault) +- [diyelectromusic](https://github.com/diyelectromusic/) for many [contributions](https://github.com/probonopd/MiniDexed/commits?author=diyelectromusic) + +## Stargazers over time +[![Stargazers over time](https://starchart.cc/probonopd/MiniDexed.svg?variant=adaptive)](https://starchart.cc/probonopd/MiniDexed) diff --git a/Synth_Dexed b/Synth_Dexed index 8c677ce..c9f5274 160000 --- a/Synth_Dexed +++ b/Synth_Dexed @@ -1 +1 @@ -Subproject commit 8c677ceb4b3fb73f8643e30ff6cf4158dc8b9e53 +Subproject commit c9f52741a802ad9bb01263823650f7cc3b0b5108 diff --git a/circle-stdlib b/circle-stdlib index 61cf3a4..3bd135d 160000 --- a/circle-stdlib +++ b/circle-stdlib @@ -1 +1 @@ -Subproject commit 61cf3a47bf93628039078b7c840e44432e52343e +Subproject commit 3bd135d35ac23c7c726924cb85b5890cbf80bbfe diff --git a/src/config.cpp b/src/config.cpp index ca958ef..70c76f3 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -36,20 +36,43 @@ void CConfig::Load (void) { m_Properties.Load (); - m_bUSBGadgetMode = m_Properties.GetNumber ("USBGadget", 0) != 0; + // Number of Tone Generators and Polyphony + m_nToneGenerators = m_Properties.GetNumber ("ToneGenerators", DefToneGenerators); + m_nPolyphony = m_Properties.GetNumber ("Polyphony", DefaultNotes); + // At present there are only two options for tone generators: min or max + // and for the Pi 1,2,3 these are the same anyway. + if ((m_nToneGenerators != MinToneGenerators) && (m_nToneGenerators != AllToneGenerators)) + { + m_nToneGenerators = DefToneGenerators; + } + if (m_nPolyphony > MaxNotes) + { + m_nPolyphony = DefaultNotes; + } + + m_bUSBGadget = m_Properties.GetNumber ("USBGadget", 0) != 0; + m_nUSBGadgetPin = m_Properties.GetNumber ("USBGadgetPin", 0); // Default OFF + SetUSBGadgetMode(m_bUSBGadget); // Might get overriden later by USBGadgetPin state m_SoundDevice = m_Properties.GetString ("SoundDevice", "pwm"); m_nSampleRate = m_Properties.GetNumber ("SampleRate", 48000); + m_bQuadDAC8Chan = m_Properties.GetNumber ("QuadDAC8Chan", 0) != 0; + if (m_SoundDevice == "hdmi") { + m_nChunkSize = m_Properties.GetNumber ("ChunkSize", 384*6); + } + else + { #ifdef ARM_ALLOW_MULTI_CORE - m_nChunkSize = m_Properties.GetNumber ("ChunkSize", m_SoundDevice == "hdmi" ? 384*6 : 256); + m_nChunkSize = m_Properties.GetNumber ("ChunkSize", m_bQuadDAC8Chan ? 1024 : 256); // 128 per channel #else - m_nChunkSize = m_Properties.GetNumber ("ChunkSize", m_SoundDevice == "hdmi" ? 384*6 : 1024); + m_nChunkSize = m_Properties.GetNumber ("ChunkSize", 1024); #endif + } m_nDACI2CAddress = m_Properties.GetNumber ("DACI2CAddress", 0); m_bChannelsSwapped = m_Properties.GetNumber ("ChannelsSwapped", 0) != 0; - unsigned newEngineType = m_Properties.GetNumber ("EngineType", 1); + unsigned newEngineType = m_Properties.GetNumber ("EngineType", 1); if (newEngineType == 2) { m_EngineType = MKI; } else if (newEngineType == 3) { @@ -82,9 +105,13 @@ void CConfig::Load (void) m_bMIDIRXProgramChange = m_Properties.GetNumber ("MIDIRXProgramChange", 1) != 0; m_bIgnoreAllNotesOff = m_Properties.GetNumber ("IgnoreAllNotesOff", 0) != 0; - m_bMIDIAutoVoiceDumpOnPC = m_Properties.GetNumber ("MIDIAutoVoiceDumpOnPC", 1) != 0; + m_bMIDIAutoVoiceDumpOnPC = m_Properties.GetNumber ("MIDIAutoVoiceDumpOnPC", 0) != 0; m_bHeaderlessSysExVoices = m_Properties.GetNumber ("HeaderlessSysExVoices", 0) != 0; m_bExpandPCAcrossBanks = m_Properties.GetNumber ("ExpandPCAcrossBanks", 1) != 0; + + m_nMIDISystemCCVol = m_Properties.GetNumber ("MIDISystemCCVol", 0); + m_nMIDISystemCCPan = m_Properties.GetNumber ("MIDISystemCCPan", 0); + m_nMIDISystemCCDetune = m_Properties.GetNumber ("MIDISystemCCDetune", 0); m_bLCDEnabled = m_Properties.GetNumber ("LCDEnabled", 0) != 0; m_nLCDPinEnable = m_Properties.GetNumber ("LCDPinEnable", 4); @@ -102,6 +129,20 @@ void CConfig::Load (void) m_bSSD1306LCDRotate = m_Properties.GetNumber ("SSD1306LCDRotate", 0) != 0; m_bSSD1306LCDMirror = m_Properties.GetNumber ("SSD1306LCDMirror", 0) != 0; + m_nSPIBus = m_Properties.GetNumber ("SPIBus", SPI_INACTIVE); // Disabled by default + m_nSPIMode = m_Properties.GetNumber ("SPIMode", SPI_DEF_MODE); + m_nSPIClockKHz = m_Properties.GetNumber ("SPIClockKHz", SPI_DEF_CLOCK); + + m_bST7789Enabled = m_Properties.GetNumber ("ST7789Enabled", 0) != 0; + m_nST7789Data = m_Properties.GetNumber ("ST7789Data", 0); + m_nST7789Select = m_Properties.GetNumber ("ST7789Select", 0); + m_nST7789Reset = m_Properties.GetNumber ("ST7789Reset", 0); // optional + m_nST7789Backlight = m_Properties.GetNumber ("ST7789Backlight", 0); // optional + m_nST7789Width = m_Properties.GetNumber ("ST7789Width", 240); + m_nST7789Height = m_Properties.GetNumber ("ST7789Height", 240); + m_nST7789Rotation = m_Properties.GetNumber ("ST7789Rotation", 0); + m_bST7789SmallFont = m_Properties.GetNumber ("ST7789SmallFont", 0) != 0; + m_nLCDColumns = m_Properties.GetNumber ("LCDColumns", 16); m_nLCDRows = m_Properties.GetNumber ("LCDRows", 2); @@ -164,11 +205,68 @@ void CConfig::Load (void) m_INetworkDNSServer = m_Properties.GetIPAddress("NetworkDNSServer") != 0; } +unsigned CConfig::GetToneGenerators (void) const +{ + return m_nToneGenerators; +} + +unsigned CConfig::GetPolyphony (void) const +{ + return m_nPolyphony; +} + +unsigned CConfig::GetTGsCore1 (void) const +{ +#ifndef ARM_ALLOW_MULTI_CORE + return 0; +#else + if (m_nToneGenerators > MinToneGenerators) + { + return TGsCore1 + TGsCore1Opt; + } + else + { + return TGsCore1; + } +#endif +} + +unsigned CConfig::GetTGsCore23 (void) const +{ +#ifndef ARM_ALLOW_MULTI_CORE + return 0; +#else + if (m_nToneGenerators > MinToneGenerators) + { + return TGsCore23 + TGsCore23Opt; + } + else + { + return TGsCore23; + } +#endif +} + +bool CConfig::GetUSBGadget (void) const +{ + return m_bUSBGadget; +} + +unsigned CConfig::GetUSBGadgetPin (void) const +{ + return m_nUSBGadgetPin; +} + bool CConfig::GetUSBGadgetMode (void) const { return m_bUSBGadgetMode; } +void CConfig::SetUSBGadgetMode (bool USBGadgetMode) +{ + m_bUSBGadgetMode = USBGadgetMode; +} + const char *CConfig::GetSoundDevice (void) const { return m_SoundDevice.c_str (); @@ -199,6 +297,11 @@ unsigned CConfig::GetEngineType (void) const return m_EngineType; } +bool CConfig::GetQuadDAC8Chan (void) const +{ + return m_bQuadDAC8Chan; +} + unsigned CConfig::GetMIDIBaudRate (void) const { return m_nMIDIBaudRate; @@ -239,6 +342,21 @@ bool CConfig::GetExpandPCAcrossBanks (void) const return m_bExpandPCAcrossBanks; } +unsigned CConfig::GetMIDISystemCCVol (void) const +{ + return m_nMIDISystemCCVol; +} + +unsigned CConfig::GetMIDISystemCCPan (void) const +{ + return m_nMIDISystemCCPan; +} + +unsigned CConfig::GetMIDISystemCCDetune (void) const +{ + return m_nMIDISystemCCDetune; +} + bool CConfig::GetLCDEnabled (void) const { return m_bLCDEnabled; @@ -309,6 +427,65 @@ bool CConfig::GetSSD1306LCDMirror (void) const return m_bSSD1306LCDMirror; } +unsigned CConfig::GetSPIBus (void) const +{ + return m_nSPIBus; +} + +unsigned CConfig::GetSPIMode (void) const +{ + return m_nSPIMode; +} + +unsigned CConfig::GetSPIClockKHz (void) const +{ + return m_nSPIClockKHz; +} + +bool CConfig::GetST7789Enabled (void) const +{ + return m_bST7789Enabled; +} + +unsigned CConfig::GetST7789Data (void) const +{ + return m_nST7789Data; +} + +unsigned CConfig::GetST7789Select (void) const +{ + return m_nST7789Select; +} + +unsigned CConfig::GetST7789Reset (void) const +{ + return m_nST7789Reset; +} + +unsigned CConfig::GetST7789Backlight (void) const +{ + return m_nST7789Backlight; +} + +unsigned CConfig::GetST7789Width (void) const +{ + return m_nST7789Width; +} + +unsigned CConfig::GetST7789Height (void) const +{ + return m_nST7789Height; +} + +unsigned CConfig::GetST7789Rotation (void) const +{ + return m_nST7789Rotation; +} + +bool CConfig::GetST7789SmallFont (void) const +{ + return m_bST7789SmallFont; +} unsigned CConfig::GetLCDColumns (void) const { return m_nLCDColumns; diff --git a/src/config.h b/src/config.h index 794b9ce..3103fca 100644 --- a/src/config.h +++ b/src/config.h @@ -29,21 +29,56 @@ #include #include +#define SPI_INACTIVE 255 +#define SPI_DEF_CLOCK 15000 // kHz +#define SPI_DEF_MODE 0 // Default mode (0,1,2,3) + class CConfig // Configuration for MiniDexed { public: +// Set maximum, minimum and default numbers of tone generators, depending on Pi version. +// Actual number in can be changed via config settings for some Pis. #ifndef ARM_ALLOW_MULTI_CORE - static const unsigned ToneGenerators = 1; + // Pi V1 or Zero (single core) + static const unsigned MinToneGenerators = 1; + static const unsigned AllToneGenerators = 1; + static const unsigned DefToneGenerators = AllToneGenerators; +#else +#if (RASPPI==4 || RASPPI==5) + // Pi 4 and 5 quad core + // These are max values, default is to support 8 in total with optional 16 TGs + static const unsigned TGsCore1 = 2; // process 2 TGs on core 1 + static const unsigned TGsCore23 = 3; // process 3 TGs on core 2 and 3 each + static const unsigned TGsCore1Opt = 2; // process optional additional 2 TGs on core 1 + static const unsigned TGsCore23Opt = 3; // process optional additional 3 TGs on core 2 and 3 each + static const unsigned MinToneGenerators = TGsCore1 + 2*TGsCore23; + static const unsigned AllToneGenerators = TGsCore1 + TGsCore1Opt + 2*TGsCore23 + 2*TGsCore23Opt; + static const unsigned DefToneGenerators = MinToneGenerators; #else + // Pi 2 or 3 quad core static const unsigned TGsCore1 = 2; // process 2 TGs on core 1 static const unsigned TGsCore23 = 3; // process 3 TGs on core 2 and 3 each - static const unsigned ToneGenerators = TGsCore1 + 2*TGsCore23; + static const unsigned TGsCore1Opt = 0; + static const unsigned TGsCore23Opt = 0; + static const unsigned MinToneGenerators = TGsCore1 + 2*TGsCore23; + static const unsigned AllToneGenerators = MinToneGenerators; + static const unsigned DefToneGenerators = AllToneGenerators; #endif - +#endif + +// Set maximum polyphony, depending on PI version. This can be changed via config settings #if RASPPI == 1 - static const unsigned MaxNotes = 8; // polyphony + static const unsigned MaxNotes = 8; + static const unsigned DefaultNotes = 8; +#elif RASPPI == 4 + static const unsigned MaxNotes = 32; + static const unsigned DefaultNotes = 24; +#elif RASPPI == 5 + static const unsigned MaxNotes = 32; + static const unsigned DefaultNotes = 32; #else static const unsigned MaxNotes = 16; + static const unsigned DefaultNotes = 16; #endif static const unsigned MaxChunkSize = 4096; @@ -64,8 +99,17 @@ public: void Load (void); + // TGs and Polyphony + unsigned GetToneGenerators (void) const; + unsigned GetPolyphony (void) const; + unsigned GetTGsCore1 (void) const; + unsigned GetTGsCore23 (void) const; + // USB Mode - bool GetUSBGadgetMode (void) const; // true if in USB gadget mode + bool GetUSBGadget (void) const; + unsigned GetUSBGadgetPin (void) const; + bool GetUSBGadgetMode (void) const; // true if in USB gadget mode depending on USBGadget and USBGadgetPin + void SetUSBGadgetMode (bool USBGadgetMode); // Sound device const char *GetSoundDevice (void) const; @@ -74,6 +118,7 @@ public: unsigned GetDACI2CAddress (void) const; // 0 for auto probing bool GetChannelsSwapped (void) const; unsigned GetEngineType (void) const; + bool GetQuadDAC8Chan (void) const; // false if not specified // MIDI unsigned GetMIDIBaudRate (void) const; @@ -81,9 +126,12 @@ public: const char *GetMIDIThruOut (void) const; // "" if not specified bool GetMIDIRXProgramChange (void) const; // true if not specified bool GetIgnoreAllNotesOff (void) const; - bool GetMIDIAutoVoiceDumpOnPC (void) const; // true if not specified + bool GetMIDIAutoVoiceDumpOnPC (void) const; // false if not specified bool GetHeaderlessSysExVoices (void) const; // false if not specified bool GetExpandPCAcrossBanks (void) const; // true if not specified + unsigned GetMIDISystemCCVol (void) const; + unsigned GetMIDISystemCCPan (void) const; + unsigned GetMIDISystemCCDetune (void) const; // HD44780 LCD // GPIO pin numbers are chip numbers, not header positions @@ -104,6 +152,22 @@ public: bool GetSSD1306LCDRotate (void) const; bool GetSSD1306LCDMirror (void) const; + // SPI support + unsigned GetSPIBus (void) const; + unsigned GetSPIMode (void) const; + unsigned GetSPIClockKHz (void) const; + + // ST7789 LCD + bool GetST7789Enabled (void) const; + unsigned GetST7789Data (void) const; + unsigned GetST7789Select (void) const; + unsigned GetST7789Reset (void) const; + unsigned GetST7789Backlight (void) const; + unsigned GetST7789Width (void) const; + unsigned GetST7789Height (void) const; + unsigned GetST7789Rotation (void) const; + bool GetST7789SmallFont (void) const; + unsigned GetLCDColumns (void) const; unsigned GetLCDRows (void) const; @@ -182,6 +246,11 @@ public: private: CPropertiesFatFsFile m_Properties; + unsigned m_nToneGenerators; + unsigned m_nPolyphony; + + bool m_bUSBGadget; + unsigned m_nUSBGadgetPin; bool m_bUSBGadgetMode; std::string m_SoundDevice; @@ -190,6 +259,7 @@ private: unsigned m_nDACI2CAddress; bool m_bChannelsSwapped; unsigned m_EngineType; + bool m_bQuadDAC8Chan; unsigned m_nMIDIBaudRate; std::string m_MIDIThruIn; @@ -199,6 +269,9 @@ private: bool m_bMIDIAutoVoiceDumpOnPC; bool m_bHeaderlessSysExVoices; bool m_bExpandPCAcrossBanks; + unsigned m_nMIDISystemCCVol; + unsigned m_nMIDISystemCCPan; + unsigned m_nMIDISystemCCDetune; bool m_bLCDEnabled; unsigned m_nLCDPinEnable; @@ -215,7 +288,21 @@ private: unsigned m_nSSD1306LCDHeight; bool m_bSSD1306LCDRotate; bool m_bSSD1306LCDMirror; - + + unsigned m_nSPIBus; + unsigned m_nSPIMode; + unsigned m_nSPIClockKHz; + + bool m_bST7789Enabled; + unsigned m_nST7789Data; + unsigned m_nST7789Select; + unsigned m_nST7789Reset; + unsigned m_nST7789Backlight; + unsigned m_nST7789Width; + unsigned m_nST7789Height; + unsigned m_nST7789Rotation; + unsigned m_bST7789SmallFont; + unsigned m_nLCDColumns; unsigned m_nLCDRows; diff --git a/src/config.txt b/src/config.txt index 7957b5c..c10dfa1 100644 --- a/src/config.txt +++ b/src/config.txt @@ -10,7 +10,7 @@ gpu_mem=16 disable_overscan=0 # -# Use 64-bit for RPi 3, 4, 400 and Zero 2, and 32-bit for all other models +# Use 64-bit for RPi 3, 4, 400, 5 and Zero 2, and 32-bit for all other models # [pi3] @@ -24,3 +24,7 @@ kernel=kernel8-rpi4.img # Zero 2 W [pi02] arm_64bit=1 + +[pi5] +arm_64bit=1 +kernel=kernel_2712.img diff --git a/src/kernel.cpp b/src/kernel.cpp index 3d80d71..c6c86d9 100644 --- a/src/kernel.cpp +++ b/src/kernel.cpp @@ -20,6 +20,7 @@ #include "kernel.h" #include #include +#include #include #include #include "usbminidexedmidigadget.h" @@ -36,6 +37,7 @@ CKernel::CKernel (void) m_Config (&mFileSystem), m_GPIOManager (&mInterrupt), m_I2CMaster (CMachineInfo::Get ()->GetDevice (DeviceI2CMaster), TRUE), + m_pSPIMaster (nullptr), m_pDexed (0) { s_pThis = this; @@ -69,23 +71,76 @@ bool CKernel::Initialize (void) m_Config.Load (); - if (m_Config.GetUSBGadgetMode()) + unsigned nSPIMaster = m_Config.GetSPIBus(); + unsigned nSPIMode = m_Config.GetSPIMode(); + unsigned long nSPIClock = 1000 * m_Config.GetSPIClockKHz(); +#if RASPPI<4 + // By default older RPI versions use SPI 0. + // It is possible to build circle to support SPI 1 for + // devices that use the 40-pin header, but that isn't + // enabled at present... + if (nSPIMaster == 0) +#else + // RPI 4+ has several possible SPI Bus Configurations. + // As mentioned above, SPI 1 is not built by default. + // See circle/include/circle/spimaster.h + if (nSPIMaster == 0 || nSPIMaster == 3 || nSPIMaster == 4 || nSPIMaster == 5 || nSPIMaster == 6) +#endif { + unsigned nCPHA = (nSPIMode & 1) ? 1 : 0; + unsigned nCPOL = (nSPIMode & 2) ? 1 : 0; + m_pSPIMaster = new CSPIMaster (nSPIClock, nCPOL, nCPHA, nSPIMaster); + if (!m_pSPIMaster->Initialize()) + { + delete (m_pSPIMaster); + m_pSPIMaster = nullptr; + } + } + + bool bUSBGadgetMode = false; + if (m_Config.GetUSBGadget()) + { + unsigned nUSBGadgetPin = m_Config.GetUSBGadgetPin(); + if (nUSBGadgetPin == 0) + { + // No hardware config option + bUSBGadgetMode = true; + } + else + { + // State of USB Gadget Mode determined by state of the pin. + // Pulled down = enable USB Gadget mode + CGPIOPin usbGadgetPin(nUSBGadgetPin, GPIOModeInputPullUp); + + if (usbGadgetPin.Read() == 0) + { + bUSBGadgetMode = true; + } + } + } + + if (bUSBGadgetMode) + { +#if RASPPI==5 +#warning No support for USB Gadget Mode on RPI 5 yet +#else // Run the USB stack in USB Gadget (device) mode m_pUSB = new CUSBMiniDexedMIDIGadget (&mInterrupt); +#endif } else { // Run the USB stack in USB Host (default) mode m_pUSB = new CUSBHCIDevice (&mInterrupt, &mTimer, TRUE); } + m_Config.SetUSBGadgetMode(bUSBGadgetMode); if (!m_pUSB->Initialize ()) { return FALSE; } - m_pDexed = new CMiniDexed (&m_Config, &mInterrupt, &m_GPIOManager, &m_I2CMaster, + m_pDexed = new CMiniDexed (&m_Config, &mInterrupt, &m_GPIOManager, &m_I2CMaster, m_pSPIMaster, &mFileSystem); assert (m_pDexed); diff --git a/src/kernel.h b/src/kernel.h index 8167c61..25ab3ed 100644 --- a/src/kernel.h +++ b/src/kernel.h @@ -24,6 +24,7 @@ #include #include #include +#include #include #include #include "config.h" @@ -55,6 +56,7 @@ private: CCPUThrottle m_CPUThrottle; CGPIOManager m_GPIOManager; CI2CMaster m_I2CMaster; + CSPIMaster *m_pSPIMaster; CMiniDexed *m_pDexed; CUSBController *m_pUSB; diff --git a/src/mididevice.cpp b/src/mididevice.cpp index cc5a79c..8ea5600 100644 --- a/src/mididevice.cpp +++ b/src/mididevice.cpp @@ -53,6 +53,21 @@ LOGMODULE ("mididevice"); #define MIDI_PROGRAM_CHANGE 0b1100 #define MIDI_PITCH_BEND 0b1110 +// MIDI "System" level (i.e. all TG) custom CC maps +// Note: Even if number of TGs is not 8, there are only 8 +// available to be used in the mappings here. +#define NUM_MIDI_CC_MAPS 8 +const unsigned MIDISystemCCMap[NUM_MIDI_CC_MAPS][8] = { + {0,0,0,0,0,0,0,0}, // 0 = disabled + {16,17,18,19,80,81,82,83}, // 1 = General Purpose Controllers 1-8 + {20,21,22,23,24,25,26,27}, + {52,53,54,55,56,57,58,59}, + {102,103,104,105,106,107,108,109}, + {110,111,112,113,114,115,116,117}, + {3,9,14,15,28,29,30,31}, + {35,41,46,47,60,61,62,63} +}; + #define MIDI_SYSTEM_EXCLUSIVE_BEGIN 0xF0 #define MIDI_SYSTEM_EXCLUSIVE_END 0xF7 #define MIDI_TIMING_CLOCK 0xF8 @@ -65,10 +80,38 @@ CMIDIDevice::CMIDIDevice (CMiniDexed *pSynthesizer, CConfig *pConfig, CUserInter m_pConfig (pConfig), m_pUI (pUI) { - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < CConfig::AllToneGenerators; nTG++) { m_ChannelMap[nTG] = Disabled; } + + m_nMIDISystemCCVol = m_pConfig->GetMIDISystemCCVol(); + m_nMIDISystemCCPan = m_pConfig->GetMIDISystemCCPan(); + m_nMIDISystemCCDetune = m_pConfig->GetMIDISystemCCDetune(); + + m_MIDISystemCCBitmap[0] = 0; + m_MIDISystemCCBitmap[1] = 0; + m_MIDISystemCCBitmap[2] = 0; + m_MIDISystemCCBitmap[3] = 0; + + for (int tg=0; tg<8; tg++) + { + if (m_nMIDISystemCCVol != 0) { + u8 cc = MIDISystemCCMap[m_nMIDISystemCCVol][tg]; + m_MIDISystemCCBitmap[cc>>5] |= (1<<(cc%32)); + } + if (m_nMIDISystemCCPan != 0) { + u8 cc = MIDISystemCCMap[m_nMIDISystemCCPan][tg]; + m_MIDISystemCCBitmap[cc>>5] |= (1<<(cc%32)); + } + if (m_nMIDISystemCCDetune != 0) { + u8 cc = MIDISystemCCMap[m_nMIDISystemCCDetune][tg]; + m_MIDISystemCCBitmap[cc>>5] |= (1<<(cc%32)); + } + } + if (m_pConfig->GetMIDIDumpEnabled ()) { + LOGNOTE("MIDI System CC Map: %08X %08X %08X %08X", m_MIDISystemCCBitmap[3],m_MIDISystemCCBitmap[2],m_MIDISystemCCBitmap[1],m_MIDISystemCCBitmap[0]); + } } CMIDIDevice::~CMIDIDevice (void) @@ -78,13 +121,13 @@ CMIDIDevice::~CMIDIDevice (void) void CMIDIDevice::SetChannel (u8 ucChannel, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_ChannelMap[nTG] = ucChannel; } u8 CMIDIDevice::GetChannel (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_ChannelMap[nTG]; } @@ -184,9 +227,35 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign else { // Perform any MiniDexed level MIDI handling before specific Tone Generators + unsigned nPerfCh = m_pSynthesizer->GetPerformanceSelectChannel(); switch (ucType) { case MIDI_CONTROL_CHANGE: + // Check for performance PC messages + if (nPerfCh != Disabled) + { + if ((ucChannel == nPerfCh) || (nPerfCh == OmniMode)) + { + if (pMessage[1] == MIDI_CC_BANK_SELECT_MSB) + { + m_pSynthesizer->BankSelectMSBPerformance (pMessage[2]); + } + else if (pMessage[1] == MIDI_CC_BANK_SELECT_LSB) + { + m_pSynthesizer->BankSelectLSBPerformance (pMessage[2]); + } + else + { + // Ignore any other CC messages at this time + } + } + } + if (nLength == 3) + { + m_pUI->UIMIDICmdHandler (ucChannel, ucStatus & 0xF0, pMessage[1], pMessage[2]); + } + break; + case MIDI_NOTE_OFF: case MIDI_NOTE_ON: if (nLength < 3) @@ -195,11 +264,11 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign } m_pUI->UIMIDICmdHandler (ucChannel, ucStatus & 0xF0, pMessage[1], pMessage[2]); break; + case MIDI_PROGRAM_CHANGE: // Check for performance PC messages if( m_pConfig->GetMIDIRXProgramChange() ) { - unsigned nPerfCh = m_pSynthesizer->GetPerformanceSelectChannel(); if( nPerfCh != Disabled) { if ((ucChannel == nPerfCh) || (nPerfCh == OmniMode)) @@ -211,8 +280,10 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign break; } - // Process MIDI for each Tone Generator - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + // Process MIDI for each active Tone Generator + bool bSystemCCHandled = false; + bool bSystemCCChecked = false; + for (unsigned nTG = 0; nTG < m_pConfig->GetToneGenerators() && !bSystemCCHandled; nTG++) { if (ucStatus == MIDI_SYSTEM_EXCLUSIVE_BEGIN) { @@ -345,6 +416,17 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign m_pSynthesizer->notesOff (pMessage[2], nTG); } break; + + default: + // Check for system-level, cross-TG MIDI Controls, but only do it once. + // Also, if successfully handled, then no need to process other TGs, + // so it is possible to break out of the main TG loop too. + // Note: We handle this here so we get the TG MIDI channel checking. + if (!bSystemCCChecked) { + bSystemCCHandled = HandleMIDISystemCC(pMessage[1], pMessage[2]); + bSystemCCChecked = true; + } + break; } break; @@ -390,6 +472,52 @@ void CMIDIDevice::AddDevice (const char *pDeviceName) s_DeviceMap.insert (std::pair (pDeviceName, this)); } +bool CMIDIDevice::HandleMIDISystemCC(const u8 ucCC, const u8 ucCCval) +{ + // This only makes sense when there are at least 8 TGs. + // Note: If more than 8 TGs then only 8 TGs are controllable this way. + if (m_pConfig->GetToneGenerators() < 8) { + return false; + } + + // Quickly reject any CCs not in the configured maps + if ((m_MIDISystemCCBitmap[ucCC>>5] & (1<<(ucCC%32))) == 0) { + // Not in the map + return false; + } + + // Not looking for duplicate CCs so return once handled + for (unsigned tg=0; tg<8; tg++) { + if (m_nMIDISystemCCVol != 0) { + if (ucCC == MIDISystemCCMap[m_nMIDISystemCCVol][tg]) { + m_pSynthesizer->SetVolume (ucCCval, tg); + return true; + } + } + if (m_nMIDISystemCCPan != 0) { + if (ucCC == MIDISystemCCMap[m_nMIDISystemCCPan][tg]) { + m_pSynthesizer->SetPan (ucCCval, tg); + return true; + } + } + if (m_nMIDISystemCCDetune != 0) { + if (ucCC == MIDISystemCCMap[m_nMIDISystemCCDetune][tg]) { + if (ucCCval == 0) + { + m_pSynthesizer->SetMasterTune (0, tg); + } + else + { + m_pSynthesizer->SetMasterTune (maplong (ucCCval, 1, 127, -99, 99), tg); + } + return true; + } + } + } + + return false; +} + void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nLength, const unsigned nCable, const uint8_t nTG) { int16_t sysex_return; diff --git a/src/mididevice.h b/src/mididevice.h index 44be25a..44f1691 100644 --- a/src/mididevice.h +++ b/src/mididevice.h @@ -30,6 +30,9 @@ #include #include "userinterface.h" +#define MAX_DX7_SYSEX_LENGTH 4104 +#define MAX_MIDI_MESSAGE MAX_DX7_SYSEX_LENGTH + class CMiniDexed; class CMIDIDevice @@ -57,12 +60,21 @@ protected: void MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsigned nCable = 0); void AddDevice (const char *pDeviceName); void HandleSystemExclusive(const uint8_t* pMessage, const size_t nLength, const unsigned nCable, const uint8_t nTG); + +private: + bool HandleMIDISystemCC(const u8 ucCC, const u8 ucCCval); + private: CMiniDexed *m_pSynthesizer; CConfig *m_pConfig; CUserInterface *m_pUI; - u8 m_ChannelMap[CConfig::ToneGenerators]; + u8 m_ChannelMap[CConfig::AllToneGenerators]; + + unsigned m_nMIDISystemCCVol; + unsigned m_nMIDISystemCCPan; + unsigned m_nMIDISystemCCDetune; + u32 m_MIDISystemCCBitmap[4]; // to allow for 128 bit entries std::string m_DeviceName; diff --git a/src/midikeyboard.cpp b/src/midikeyboard.cpp index 169f165..2eae9d8 100644 --- a/src/midikeyboard.cpp +++ b/src/midikeyboard.cpp @@ -37,6 +37,7 @@ TMIDIPacketHandler * const CMIDIKeyboard::s_pMIDIPacketHandler[MaxInstances] = CMIDIKeyboard::CMIDIKeyboard (CMiniDexed *pSynthesizer, CConfig *pConfig, CUserInterface *pUI, unsigned nInstance) : CMIDIDevice (pSynthesizer, pConfig, pUI), + m_nSysExIdx (0), m_nInstance (nInstance), m_pMIDIDevice (0) { @@ -100,28 +101,86 @@ void CMIDIKeyboard::Send (const u8 *pMessage, size_t nLength, unsigned nCable) m_SendQueue.push (Entry); } +// Most packets will be passed straight onto the main MIDI message handler +// but SysEx messages are multiple USB packets and so will need building up +// before parsing. +void CMIDIKeyboard::USBMIDIMessageHandler (u8 *pPacket, unsigned nLength, unsigned nCable) +{ + if ((pPacket[0] == 0xF0) && (m_nSysExIdx == 0)) + { + // Start of SysEx message + //printf("SysEx Start Idx=%d, (%d)\n", m_nSysExIdx, nLength); + for (unsigned i=0; i= USB_SYSEX_BUFFER_SIZE) { + // Run out of space, so reset and ignore rest of the message + m_nSysExIdx = 0; + break; + } + else if (pPacket[i] == 0xF7) { + // End of SysEx message + m_SysEx[m_nSysExIdx++] = pPacket[i]; + //printf ("SysEx End Idx=%d\n", m_nSysExIdx); + MIDIMessageHandler (m_SysEx, m_nSysExIdx, nCable); + // Reset ready for next time + m_nSysExIdx = 0; + } + else if ((pPacket[i] & 0x80) != 0) { + // Received another command, so reset processing as something has gone wrong + //printf ("SysEx Reset\n"); + m_nSysExIdx = 0; + break; + } + else + { + // Store the byte + m_SysEx[m_nSysExIdx++] = pPacket[i]; + } + } + } + else + { + // Assume it is a standard message + MIDIMessageHandler (pPacket, nLength, nCable); + } +} + void CMIDIKeyboard::MIDIPacketHandler0 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[0] != 0); - s_pThis[0]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[0]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::MIDIPacketHandler1 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[1] != 0); - s_pThis[1]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[1]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::MIDIPacketHandler2 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[2] != 0); - s_pThis[2]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[2]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::MIDIPacketHandler3 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[3] != 0); - s_pThis[3]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[3]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::DeviceRemovedHandler (CDevice *pDevice, void *pContext) diff --git a/src/midikeyboard.h b/src/midikeyboard.h index 0868f9c..047fa52 100644 --- a/src/midikeyboard.h +++ b/src/midikeyboard.h @@ -31,6 +31,8 @@ #include #include +#define USB_SYSEX_BUFFER_SIZE (MAX_DX7_SYSEX_LENGTH+128) // Allow a bit spare to handle unexpected SysEx messages + class CMiniDexed; class CMIDIKeyboard : public CMIDIDevice @@ -53,6 +55,8 @@ private: static void MIDIPacketHandler3 (unsigned nCable, u8 *pPacket, unsigned nLength); static void DeviceRemovedHandler (CDevice *pDevice, void *pContext); + + void USBMIDIMessageHandler (u8 *pPacket, unsigned nLength, unsigned nCable); private: struct TSendQueueEntry @@ -61,6 +65,8 @@ private: size_t nLength; unsigned nCable; }; + uint8_t m_SysEx[USB_SYSEX_BUFFER_SIZE]; + unsigned m_nSysExIdx; private: unsigned m_nInstance; @@ -73,6 +79,7 @@ private: static CMIDIKeyboard *s_pThis[MaxInstances]; static TMIDIPacketHandler * const s_pMIDIPacketHandler[MaxInstances]; + }; #endif diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 78f8e69..3c2a183 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -37,21 +37,22 @@ const char WLANConfigFile[] = "SD:wpa_supplicant.conf"; LOGMODULE ("minidexed"); CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, - CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, FATFS *pFileSystem) + CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, FATFS *pFileSystem) : #ifdef ARM_ALLOW_MULTI_CORE CMultiCoreSupport (CMemorySystem::Get ()), #endif m_pConfig (pConfig), - m_UI (this, pGPIOManager, pI2CMaster, pConfig), + m_UI (this, pGPIOManager, pI2CMaster, pSPIMaster, pConfig), m_PerformanceConfig (pFileSystem), m_PCKeyboard (this, pConfig, &m_UI), m_SerialMIDI (this, pInterrupt, pConfig, &m_UI), m_bUseSerial (false), + m_bQuadDAC8Chan (false), m_pSoundDevice (0), m_bChannelsSwapped (pConfig->GetChannelsSwapped ()), #ifdef ARM_ALLOW_MULTI_CORE - m_nActiveTGsLog2 (0), +// m_nActiveTGsLog2 (0), #endif m_GetChunkTimer ("GetChunk", 1000000U * pConfig->GetChunkSize ()/2 / pConfig->GetSampleRate ()), @@ -66,12 +67,19 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_bSavePerformance (false), m_bSavePerformanceNewFile (false), m_bSetNewPerformance (false), + m_bSetNewPerformanceBank (false), + m_bSetFirstPerformance (false), m_bDeletePerformance (false), - m_bLoadPerformanceBusy(false) + m_bLoadPerformanceBusy(false), + m_bLoadPerformanceBankBusy(false) { assert (m_pConfig); + + m_nToneGenerators = m_pConfig->GetToneGenerators(); + m_nPolyphony = m_pConfig->GetPolyphony(); + LOGNOTE("Tone Generators=%d, Polyphony=%d", m_nToneGenerators, m_nPolyphony); - for (unsigned i = 0; i < CConfig::ToneGenerators; i++) + for (unsigned i = 0; i < CConfig::AllToneGenerators; i++) { m_nVoiceBankID[i] = 0; m_nVoiceBankIDMSB[i] = 0; @@ -102,22 +110,57 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_nAftertouchTarget[i]=0; m_nReverbSend[i] = 0; - m_uchOPMask[i] = 0b111111; // All operators on - m_pTG[i] = new CDexedAdapter (CConfig::MaxNotes, pConfig->GetSampleRate ()); - assert (m_pTG[i]); - - m_pTG[i]->setEngineType(pConfig->GetEngineType ()); - m_pTG[i]->activate (); + // Active the required number of active TGs + if (iGetSampleRate ()); + assert (m_pTG[i]); + + m_pTG[i]->setEngineType(pConfig->GetEngineType ()); + m_pTG[i]->activate (); + } } + + unsigned nUSBGadgetPin = pConfig->GetUSBGadgetPin(); + bool bUSBGadget = pConfig->GetUSBGadget(); + bool bUSBGadgetMode = pConfig->GetUSBGadgetMode(); - if (pConfig->GetUSBGadgetMode()) + if (bUSBGadgetMode) { - LOGNOTE ("USB In Gadget (Device) Mode"); +#if RASPPI==5 + LOGNOTE ("USB Gadget (Device) Mode NOT supported on RPI 5"); +#else + if (nUSBGadgetPin == 0) + { + LOGNOTE ("USB In Gadget (Device) Mode"); + } + else + { + LOGNOTE ("USB In Gadget (Device) Mode [USBGadgetPin %d = LOW]", nUSBGadgetPin); + } +#endif } else { - LOGNOTE ("USB In Host Mode"); + if (bUSBGadget) + { + if (nUSBGadgetPin == 0) + { + // This shouldn't be possible... + LOGNOTE ("USB State Unknown"); + } + else + { + LOGNOTE ("USB In Host Mode [USBGadgetPin %d = HIGH]", nUSBGadgetPin); + } + } + else + { + LOGNOTE ("USB In Host Mode"); + } } for (unsigned i = 0; i < CConfig::MaxUSBMIDIDevices; i++) @@ -131,13 +174,37 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, if (strcmp (pDeviceName, "i2s") == 0) { LOGNOTE ("I2S mode"); - - m_pSoundDevice = new CI2SSoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), - pConfig->GetChunkSize (), false, - pI2CMaster, pConfig->GetDACI2CAddress ()); +#if RASPPI==5 + // Quad DAC 8-channel mono only an option for RPI 5 + m_bQuadDAC8Chan = pConfig->GetQuadDAC8Chan (); +#endif + if (m_bQuadDAC8Chan && (m_nToneGenerators != 8)) + { + LOGNOTE("ERROR: Quad DAC Mode is only valid when number of TGs = 8. Defaulting to non-Quad DAC mode,"); + m_bQuadDAC8Chan = false; + } + if (m_bQuadDAC8Chan) { + LOGNOTE ("Configured for Quad DAC 8-channel Mono audio"); + m_pSoundDevice = new CI2SSoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), + pConfig->GetChunkSize (), false, + pI2CMaster, pConfig->GetDACI2CAddress (), + CI2SSoundBaseDevice::DeviceModeTXOnly, + 8); // 8 channels - L+R x4 across 4 I2S lanes + } + else + { + m_pSoundDevice = new CI2SSoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), + pConfig->GetChunkSize (), false, + pI2CMaster, pConfig->GetDACI2CAddress (), + CI2SSoundBaseDevice::DeviceModeTXOnly, + 2); // 2 channels - L+R + } } else if (strcmp (pDeviceName, "hdmi") == 0) { +#if RASPPI==5 + LOGNOTE ("HDMI mode NOT supported on RPI 5."); +#else LOGNOTE ("HDMI mode"); m_pSoundDevice = new CHDMISoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), @@ -146,6 +213,7 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, // The channels are swapped by default in the HDMI sound driver. // TODO: Remove this line, when this has been fixed in the driver. m_bChannelsSwapped = !m_bChannelsSwapped; +#endif } else { @@ -165,11 +233,11 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, setMasterVolume(1.0); // BEGIN setup tg_mixer - tg_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); + tg_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); // END setup tgmixer // BEGIN setup reverb - reverb_send_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); + reverb_send_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); reverb = new AudioEffectPlateReverb(pConfig->GetSampleRate()); SetParameter (ParameterReverbEnable, 1); SetParameter (ParameterReverbSize, 70); @@ -183,6 +251,8 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, SetParameter (ParameterCompressorEnable, 1); SetPerformanceSelectChannel(m_pConfig->GetPerformanceSelectChannel()); + + SetParameter (ParameterPerformanceBank, 0); }; bool CMiniDexed::Initialize (void) @@ -218,7 +288,7 @@ bool CMiniDexed::Initialize (void) LOGNOTE("Program Change: Disabled"); } - for (unsigned i = 0; i < CConfig::ToneGenerators; i++) + for (unsigned i = 0; i < m_nToneGenerators; i++) { assert (m_pTG[i]); @@ -240,6 +310,7 @@ bool CMiniDexed::Initialize (void) reverb_send_mixer->gain(i,mapfloat(m_nReverbSend[i],0,99,0.0f,1.0f)); } + m_PerformanceConfig.Init(m_nToneGenerators); if (m_PerformanceConfig.Load ()) { LoadPerformanceParameters(); @@ -249,25 +320,31 @@ bool CMiniDexed::Initialize (void) SetMIDIChannel (CMIDIDevice::OmniMode, 0); } - // load performances file list, and attempt to create the performance folder - if (!m_PerformanceConfig.ListPerformances()) + // setup and start the sound device + int Channels = 1; // 16-bit Mono +#ifdef ARM_ALLOW_MULTI_CORE + if (m_bQuadDAC8Chan) { - LOGERR ("Cannot create internal Performance folder, new performances can't be created"); + Channels = 8; // 16-bit 8-channel mono } - - // setup and start the sound device - if (!m_pSoundDevice->AllocateQueueFrames (m_pConfig->GetChunkSize ())) + else + { + Channels = 2; // 16-bit Stereo + } +#endif + // Need 2 x ChunkSize / Channel queue frames as the audio driver uses + // two DMA channels each of ChunkSize and one single single frame + // contains a sample for each of all the channels. + // + // See discussion here: https://github.com/rsta2/circle/discussions/453 + if (!m_pSoundDevice->AllocateQueueFrames (2 * m_pConfig->GetChunkSize () / Channels)) { LOGERR ("Cannot allocate sound queue"); return false; } -#ifndef ARM_ALLOW_MULTI_CORE - m_pSoundDevice->SetWriteFormat (SoundFormatSigned16, 1); // 16-bit Mono -#else - m_pSoundDevice->SetWriteFormat (SoundFormatSigned16, 2); // 16-bit Stereo -#endif + m_pSoundDevice->SetWriteFormat (SoundFormatSigned16, Channels); m_nQueueSizeFrames = m_pSoundDevice->GetQueueSizeFrames (); @@ -320,14 +397,30 @@ void CMiniDexed::Process (bool bPlugAndPlayUpdated) m_bSavePerformanceNewFile = false; } - if (m_bSetNewPerformance && !m_bLoadPerformanceBusy) + if (m_bSetNewPerformanceBank && !m_bLoadPerformanceBusy && !m_bLoadPerformanceBankBusy) + { + DoSetNewPerformanceBank (); + if (m_nSetNewPerformanceBankID == GetActualPerformanceBankID()) + { + m_bSetNewPerformanceBank = false; + } + + // If there is no pending SetNewPerformance already, then see if we need to find the first performance to load + // NB: If called from the UI, then there will not be a SetNewPerformance, so load the first existing one. + // If called from MIDI, there will probably be a SetNewPerformance alongside the Bank select. + if (!m_bSetNewPerformance && m_bSetFirstPerformance) + { + DoSetFirstPerformance(); + } + } + + if (m_bSetNewPerformance && !m_bSetNewPerformanceBank && !m_bLoadPerformanceBusy && !m_bLoadPerformanceBankBusy) { DoSetNewPerformance (); if (m_nSetNewPerformanceID == GetActualPerformanceID()) { m_bSetNewPerformance = false; } - } if(m_bDeletePerformance) @@ -392,12 +485,16 @@ void CMiniDexed::Run (unsigned nCore) // process the TGs, assigned to this core (2 or 3) - assert (m_nFramesToProcess <= CConfig::MaxChunkSize); - unsigned nTG = CConfig::TGsCore1 + (nCore-2)*CConfig::TGsCore23; - for (unsigned i = 0; i < CConfig::TGsCore23; i++, nTG++) + assert (m_nFramesToProcess <= m_pConfig->MaxChunkSize); + unsigned nTG = m_pConfig->GetTGsCore1() + (nCore-2)*m_pConfig->GetTGsCore23(); + for (unsigned i = 0; i < m_pConfig->GetTGsCore23(); i++, nTG++) { - assert (m_pTG[nTG]); - m_pTG[nTG]->getSamples (m_OutputLevel[nTG],m_nFramesToProcess); + assert (nTG < CConfig::AllToneGenerators); + if (nTG < m_pConfig->GetToneGenerators()) + { + assert (m_pTG[nTG]); + m_pTG[nTG]->getSamples (m_OutputLevel[nTG],m_nFramesToProcess); + } } } } @@ -410,11 +507,17 @@ CSysExFileLoader *CMiniDexed::GetSysExFileLoader (void) return &m_SysExFileLoader; } +CPerformanceConfig *CMiniDexed::GetPerformanceConfig (void) +{ + return &m_PerformanceConfig; +} + void CMiniDexed::BankSelect (unsigned nBank, unsigned nTG) { nBank=constrain((int)nBank,0,16383); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG if (GetSysExFileLoader ()->IsValidBank(nBank)) { @@ -425,11 +528,27 @@ void CMiniDexed::BankSelect (unsigned nBank, unsigned nTG) } } +void CMiniDexed::BankSelectPerformance (unsigned nBank) +{ + nBank=constrain((int)nBank,0,16383); + + if (GetPerformanceConfig ()->IsValidPerformanceBank(nBank)) + { + // Only change if we have the bank loaded + m_nVoiceBankIDPerformance = nBank; + SetNewPerformanceBank (nBank); + + m_UI.ParameterChanged (); + } +} + void CMiniDexed::BankSelectMSB (unsigned nBankMSB, unsigned nTG) { nBankMSB=constrain((int)nBankMSB,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + // MIDI Spec 1.0 "BANK SELECT" states: // "The transmitter must transmit the MSB and LSB as a pair, // and the Program Change must be sent immediately after @@ -440,11 +559,19 @@ void CMiniDexed::BankSelectMSB (unsigned nBankMSB, unsigned nTG) m_nVoiceBankIDMSB[nTG] = nBankMSB; } +void CMiniDexed::BankSelectMSBPerformance (unsigned nBankMSB) +{ + nBankMSB=constrain((int)nBankMSB,0,127); + m_nVoiceBankIDMSBPerformance = nBankMSB; +} + void CMiniDexed::BankSelectLSB (unsigned nBankLSB, unsigned nTG) { nBankLSB=constrain((int)nBankLSB,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + unsigned nBank = m_nVoiceBankID[nTG]; unsigned nBankMSB = m_nVoiceBankIDMSB[nTG]; nBank = (nBankMSB << 7) + nBankLSB; @@ -453,6 +580,18 @@ void CMiniDexed::BankSelectLSB (unsigned nBankLSB, unsigned nTG) BankSelect(nBank, nTG); } +void CMiniDexed::BankSelectLSBPerformance (unsigned nBankLSB) +{ + nBankLSB=constrain((int)nBankLSB,0,127); + + unsigned nBank = m_nVoiceBankIDPerformance; + unsigned nBankMSB = m_nVoiceBankIDMSBPerformance; + nBank = (nBankMSB << 7) + nBankLSB; + + // Now should have both MSB and LSB so enable the BankSelect + BankSelectPerformance(nBank); +} + void CMiniDexed::ProgramChange (unsigned nProgram, unsigned nTG) { assert (m_pConfig); @@ -480,7 +619,9 @@ void CMiniDexed::ProgramChange (unsigned nProgram, unsigned nTG) nProgram=constrain((int)nProgram,0,31); } - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nProgram[nTG] = nProgram; uint8_t Buffer[156]; @@ -507,10 +648,7 @@ void CMiniDexed::ProgramChangePerformance (unsigned nProgram) if (m_nParameter[ParameterPerformanceSelectChannel] != CMIDIDevice::Disabled) { // Program Change messages change Performances. - unsigned nLastPerformance = m_PerformanceConfig.GetLastPerformance(); - - // GetLastPerformance actually returns 1-indexed, number of performances - if (nProgram < nLastPerformance - 1) + if (m_PerformanceConfig.IsValidPerformance(nProgram)) { SetNewPerformance(nProgram); } @@ -522,7 +660,9 @@ void CMiniDexed::SetVolume (unsigned nVolume, unsigned nTG) { nVolume=constrain((int)nVolume,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nVolume[nTG] = nVolume; assert (m_pTG[nTG]); @@ -535,7 +675,9 @@ void CMiniDexed::SetPan (unsigned nPan, unsigned nTG) { nPan=constrain((int)nPan,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nPan[nTG] = nPan; tg_mixer->pan(nTG,mapfloat(nPan,0,127,0.0f,1.0f)); @@ -548,7 +690,9 @@ void CMiniDexed::SetReverbSend (unsigned nReverbSend, unsigned nTG) { nReverbSend=constrain((int)nReverbSend,0,99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nReverbSend[nTG] = nReverbSend; reverb_send_mixer->gain(nTG,mapfloat(nReverbSend,0,99,0.0f,1.0f)); @@ -560,7 +704,9 @@ void CMiniDexed::SetMasterTune (int nMasterTune, unsigned nTG) { nMasterTune=constrain((int)nMasterTune,-99,99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nMasterTune[nTG] = nMasterTune; assert (m_pTG[nTG]); @@ -573,7 +719,9 @@ void CMiniDexed::SetCutoff (int nCutoff, unsigned nTG) { nCutoff = constrain (nCutoff, 0, 99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nCutoff[nTG] = nCutoff; assert (m_pTG[nTG]); @@ -586,7 +734,9 @@ void CMiniDexed::SetResonance (int nResonance, unsigned nTG) { nResonance = constrain (nResonance, 0, 99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nResonance[nTG] = nResonance; assert (m_pTG[nTG]); @@ -599,7 +749,9 @@ void CMiniDexed::SetResonance (int nResonance, unsigned nTG) void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (uchChannel < CMIDIDevice::ChannelUnknown); m_nMIDIChannel[nTG] = uchChannel; @@ -619,6 +771,7 @@ void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) m_UDPMIDI.SetChannel (uchChannel, nTG); #ifdef ARM_ALLOW_MULTI_CORE +/* This doesn't appear to be used anywhere... unsigned nActiveTGs = 0; for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) { @@ -631,6 +784,7 @@ void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) assert (nActiveTGs <= 8); static const unsigned Log2[] = {0, 0, 1, 2, 2, 3, 3, 3, 3}; m_nActiveTGsLog2 = Log2[nActiveTGs]; +*/ #endif m_UI.ParameterChanged (); @@ -638,7 +792,9 @@ void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) void CMiniDexed::keyup (int16_t pitch, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); pitch = ApplyNoteLimits (pitch, nTG); @@ -650,7 +806,9 @@ void CMiniDexed::keyup (int16_t pitch, unsigned nTG) void CMiniDexed::keydown (int16_t pitch, uint8_t velocity, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); pitch = ApplyNoteLimits (pitch, nTG); @@ -662,7 +820,8 @@ void CMiniDexed::keydown (int16_t pitch, uint8_t velocity, unsigned nTG) int16_t CMiniDexed::ApplyNoteLimits (int16_t pitch, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return -1; // Not an active TG if ( pitch < (int16_t) m_nNoteLimitLow[nTG] || pitch > (int16_t) m_nNoteLimitHigh[nTG]) @@ -683,14 +842,18 @@ int16_t CMiniDexed::ApplyNoteLimits (int16_t pitch, unsigned nTG) void CMiniDexed::setSustain(bool sustain, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setSustain (sustain); } void CMiniDexed::panic(uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); if (value == 0) { m_pTG[nTG]->panic (); @@ -699,7 +862,9 @@ void CMiniDexed::panic(uint8_t value, unsigned nTG) void CMiniDexed::notesOff(uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); if (value == 0) { m_pTG[nTG]->notesOff (); @@ -708,7 +873,9 @@ void CMiniDexed::notesOff(uint8_t value, unsigned nTG) void CMiniDexed::setModWheel (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setModWheel (value); } @@ -716,35 +883,45 @@ void CMiniDexed::setModWheel (uint8_t value, unsigned nTG) void CMiniDexed::setFootController (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setFootController (value); } void CMiniDexed::setBreathController (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setBreathController (value); } void CMiniDexed::setAftertouch (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setAftertouch (value); } void CMiniDexed::setPitchbend (int16_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setPitchbend (value); } void CMiniDexed::ControllersRefresh (unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->ControllersRefresh (); } @@ -759,7 +936,7 @@ void CMiniDexed::SetParameter (TParameter Parameter, int nValue) switch (Parameter) { case ParameterCompressorEnable: - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < m_nToneGenerators; nTG++) { assert (m_pTG[nTG]); m_pTG[nTG]->setCompressor (!!nValue); @@ -819,6 +996,10 @@ void CMiniDexed::SetParameter (TParameter Parameter, int nValue) // Nothing more to do break; + case ParameterPerformanceBank: + BankSelectPerformance(nValue); + break; + default: assert (0); break; @@ -833,7 +1014,8 @@ int CMiniDexed::GetParameter (TParameter Parameter) void CMiniDexed::SetTGParameter (TTGParameter Parameter, int nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG switch (Parameter) { @@ -888,7 +1070,7 @@ void CMiniDexed::SetTGParameter (TTGParameter Parameter, int nValue, unsigned nT int CMiniDexed::GetTGParameter (TTGParameter Parameter, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); switch (Parameter) { @@ -939,7 +1121,9 @@ int CMiniDexed::GetTGParameter (TTGParameter Parameter, unsigned nTG) void CMiniDexed::SetVoiceParameter (uint8_t uchOffset, uint8_t uchValue, unsigned nOP, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); assert (nOP <= 6); @@ -972,7 +1156,9 @@ void CMiniDexed::SetVoiceParameter (uint8_t uchOffset, uint8_t uchValue, unsigne uint8_t CMiniDexed::GetVoiceParameter (uint8_t uchOffset, unsigned nOP, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return 0; // Not an active TG + assert (m_pTG[nTG]); assert (nOP <= 6); @@ -996,13 +1182,15 @@ std::string CMiniDexed::GetVoiceName (unsigned nTG) { char VoiceName[11]; memset (VoiceName, 0, sizeof VoiceName); + VoiceName[0] = 32; // space + assert (nTG < CConfig::AllToneGenerators); - assert (nTG < CConfig::ToneGenerators); - assert (m_pTG[nTG]); - m_pTG[nTG]->setName (VoiceName); - + if (nTG < m_nToneGenerators) + { + assert (m_pTG[nTG]); + m_pTG[nTG]->setName (VoiceName); + } std::string Result (VoiceName); - return Result; } @@ -1044,6 +1232,7 @@ void CMiniDexed::ProcessSound (void) void CMiniDexed::ProcessSound (void) { assert (m_pSoundDevice); + assert (m_pConfig); unsigned nFrames = m_nQueueSizeFrames - m_pSoundDevice->GetQueueFramesAvail (); if (nFrames >= m_nQueueSizeFrames/2) @@ -1064,7 +1253,7 @@ void CMiniDexed::ProcessSound (void) // process the TGs assigned to core 1 assert (nFrames <= CConfig::MaxChunkSize); - for (unsigned i = 0; i < CConfig::TGsCore1; i++) + for (unsigned i = 0; i < m_pConfig->GetTGsCore1(); i++) { assert (m_pTG[i]); m_pTG[i]->getSamples (m_OutputLevel[i], nFrames); @@ -1083,87 +1272,133 @@ void CMiniDexed::ProcessSound (void) // Audio signal path after tone generators starts here // - assert (CConfig::ToneGenerators == 8); + if (m_bQuadDAC8Chan) { + // This is only supported when there are 8 TGs + assert (m_nToneGenerators == 8); - uint8_t indexL=0, indexR=1; - - // BEGIN TG mixing - float32_t tmp_float[nFrames*2]; - int16_t tmp_int[nFrames*2]; + // No mixing is performed by MiniDexed, sound is output in 8 channels. + // Note: one TG per audio channel; output=mono; no processing. + const int Channels = 8; // One TG per channel + float32_t tmp_float[nFrames*Channels]; + int16_t tmp_int[nFrames*Channels]; - if(nMasterVolume > 0.0) - { - for (uint8_t i = 0; i < CConfig::ToneGenerators; i++) + if(nMasterVolume > 0.0) { - tg_mixer->doAddMix(i,m_OutputLevel[i]); - reverb_send_mixer->doAddMix(i,m_OutputLevel[i]); + // Convert dual float array (8 chan) to single int16 array (8 chan) + for(uint16_t i=0; i0.0 && nMasterVolume <1.0) + { + tmp_float[(i*Channels)+tg]=m_OutputLevel[tg][i] * nMasterVolume; + } + else if(nMasterVolume == 1.0) + { + tmp_float[(i*Channels)+tg]=m_OutputLevel[tg][i]; + } + } + } + arm_float_to_q15(tmp_float,tmp_int,nFrames*Channels); } - // END TG mixing - - // BEGIN create SampleBuffer for holding audio data - float32_t SampleBuffer[2][nFrames]; - // END create SampleBuffer for holding audio data - - // get the mix of all TGs - tg_mixer->getMix(SampleBuffer[indexL], SampleBuffer[indexR]); - - // BEGIN adding reverb - if (m_nParameter[ParameterReverbEnable]) + else { - float32_t ReverbBuffer[2][nFrames]; - float32_t ReverbSendBuffer[2][nFrames]; - - arm_fill_f32(0.0f, ReverbBuffer[indexL], nFrames); - arm_fill_f32(0.0f, ReverbBuffer[indexR], nFrames); - arm_fill_f32(0.0f, ReverbSendBuffer[indexR], nFrames); - arm_fill_f32(0.0f, ReverbSendBuffer[indexL], nFrames); - - m_ReverbSpinLock.Acquire (); - - reverb_send_mixer->getMix(ReverbSendBuffer[indexL], ReverbSendBuffer[indexR]); - reverb->doReverb(ReverbSendBuffer[indexL],ReverbSendBuffer[indexR],ReverbBuffer[indexL], ReverbBuffer[indexR],nFrames); - - // scale down and add left reverb buffer by reverb level - arm_scale_f32(ReverbBuffer[indexL], reverb->get_level(), ReverbBuffer[indexL], nFrames); - arm_add_f32(SampleBuffer[indexL], ReverbBuffer[indexL], SampleBuffer[indexL], nFrames); - // scale down and add right reverb buffer by reverb level - arm_scale_f32(ReverbBuffer[indexR], reverb->get_level(), ReverbBuffer[indexR], nFrames); - arm_add_f32(SampleBuffer[indexR], ReverbBuffer[indexR], SampleBuffer[indexR], nFrames); - - m_ReverbSpinLock.Release (); + arm_fill_q15(0, tmp_int, nFrames*Channels); } - // END adding reverb - - // swap stereo channels if needed prior to writing back out - if (m_bChannelsSwapped) + + if (m_pSoundDevice->Write (tmp_int, sizeof(tmp_int)) != (int) sizeof(tmp_int)) { - indexL=1; - indexR=0; + LOGERR ("Sound data dropped"); } + } + else + { + // Mix everything down to stereo + uint8_t indexL=0, indexR=1; - // Convert dual float array (left, right) to single int16 array (left/right) - for(uint16_t i=0; i 0.0) { - if(nMasterVolume >0.0 && nMasterVolume <1.0) + for (uint8_t i = 0; i < m_nToneGenerators; i++) { - tmp_float[i*2]=SampleBuffer[indexL][i] * nMasterVolume; - tmp_float[(i*2)+1]=SampleBuffer[indexR][i] * nMasterVolume; + tg_mixer->doAddMix(i,m_OutputLevel[i]); + reverb_send_mixer->doAddMix(i,m_OutputLevel[i]); } - else if(nMasterVolume == 1.0) + // END TG mixing + + // BEGIN create SampleBuffer for holding audio data + float32_t SampleBuffer[2][nFrames]; + // END create SampleBuffer for holding audio data + + // get the mix of all TGs + tg_mixer->getMix(SampleBuffer[indexL], SampleBuffer[indexR]); + + // BEGIN adding reverb + if (m_nParameter[ParameterReverbEnable]) { - tmp_float[i*2]=SampleBuffer[indexL][i]; - tmp_float[(i*2)+1]=SampleBuffer[indexR][i]; + float32_t ReverbBuffer[2][nFrames]; + float32_t ReverbSendBuffer[2][nFrames]; + + arm_fill_f32(0.0f, ReverbBuffer[indexL], nFrames); + arm_fill_f32(0.0f, ReverbBuffer[indexR], nFrames); + arm_fill_f32(0.0f, ReverbSendBuffer[indexR], nFrames); + arm_fill_f32(0.0f, ReverbSendBuffer[indexL], nFrames); + + m_ReverbSpinLock.Acquire (); + + reverb_send_mixer->getMix(ReverbSendBuffer[indexL], ReverbSendBuffer[indexR]); + reverb->doReverb(ReverbSendBuffer[indexL],ReverbSendBuffer[indexR],ReverbBuffer[indexL], ReverbBuffer[indexR],nFrames); + + // scale down and add left reverb buffer by reverb level + arm_scale_f32(ReverbBuffer[indexL], reverb->get_level(), ReverbBuffer[indexL], nFrames); + arm_add_f32(SampleBuffer[indexL], ReverbBuffer[indexL], SampleBuffer[indexL], nFrames); + // scale down and add right reverb buffer by reverb level + arm_scale_f32(ReverbBuffer[indexR], reverb->get_level(), ReverbBuffer[indexR], nFrames); + arm_add_f32(SampleBuffer[indexR], ReverbBuffer[indexR], SampleBuffer[indexR], nFrames); + + m_ReverbSpinLock.Release (); + } + // END adding reverb + + // swap stereo channels if needed prior to writing back out + if (m_bChannelsSwapped) + { + indexL=1; + indexR=0; } + + // Convert dual float array (left, right) to single int16 array (left/right) + for(uint16_t i=0; i0.0 && nMasterVolume <1.0) + { + tmp_float[i*2]=SampleBuffer[indexL][i] * nMasterVolume; + tmp_float[(i*2)+1]=SampleBuffer[indexR][i] * nMasterVolume; + } + else if(nMasterVolume == 1.0) + { + tmp_float[i*2]=SampleBuffer[indexL][i]; + tmp_float[(i*2)+1]=SampleBuffer[indexR][i]; + } + } + arm_float_to_q15(tmp_float,tmp_int,nFrames*2); + } + else + { + arm_fill_q15(0, tmp_int, nFrames * 2); } - arm_float_to_q15(tmp_float,tmp_int,nFrames*2); - } - else - arm_fill_q15(0, tmp_int, nFrames * 2); - if (m_pSoundDevice->Write (tmp_int, sizeof(tmp_int)) != (int) sizeof(tmp_int)) - { - LOGERR ("Sound data dropped"); - } + if (m_pSoundDevice->Write (tmp_int, sizeof(tmp_int)) != (int) sizeof(tmp_int)) + { + LOGERR ("Sound data dropped"); + } + } // End of Stereo mixing if (m_bProfileEnabled) { @@ -1200,15 +1435,22 @@ void CMiniDexed::SetPerformanceSelectChannel (unsigned uCh) bool CMiniDexed::SavePerformance (bool bSaveAsDeault) { - m_bSavePerformance = true; - m_bSaveAsDeault=bSaveAsDeault; + if (m_PerformanceConfig.GetInternalFolderOk()) + { + m_bSavePerformance = true; + m_bSaveAsDeault=bSaveAsDeault; - return true; + return true; + } + else + { + return false; + } } bool CMiniDexed::DoSavePerformance (void) { - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < CConfig::AllToneGenerators; nTG++) { m_PerformanceConfig.SetBankNumber (m_nVoiceBankID[nTG], nTG); m_PerformanceConfig.SetVoiceNumber (m_nProgram[nTG], nTG); @@ -1227,8 +1469,14 @@ bool CMiniDexed::DoSavePerformance (void) m_PerformanceConfig.SetNoteLimitLow (m_nNoteLimitLow[nTG], nTG); m_PerformanceConfig.SetNoteLimitHigh (m_nNoteLimitHigh[nTG], nTG); m_PerformanceConfig.SetNoteShift (m_nNoteShift[nTG], nTG); - m_pTG[nTG]->getVoiceData(m_nRawVoiceData); - m_PerformanceConfig.SetVoiceDataToTxt (m_nRawVoiceData, nTG); + if (nTG < m_pConfig->GetToneGenerators()) + { + m_pTG[nTG]->getVoiceData(m_nRawVoiceData); + } else { + // Not an active TG so provide default voice by asking for an invalid voice ID. + m_SysExFileLoader.GetVoice(CSysExFileLoader::MaxVoiceBankID, CSysExFileLoader::VoicesPerBank+1, m_nRawVoiceData); + } + m_PerformanceConfig.SetVoiceDataToTxt (m_nRawVoiceData, nTG); m_PerformanceConfig.SetMonoMode (m_bMonoMode[nTG], nTG); m_PerformanceConfig.SetModulationWheelRange (m_nModulationWheelRange[nTG], nTG); @@ -1262,7 +1510,9 @@ bool CMiniDexed::DoSavePerformance (void) void CMiniDexed::setMonoMode(uint8_t mono, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_bMonoMode[nTG]= mono != 0; m_pTG[nTG]->setMonoMode(constrain(mono, 0, 1)); @@ -1273,7 +1523,9 @@ void CMiniDexed::setMonoMode(uint8_t mono, uint8_t nTG) void CMiniDexed::setPitchbendRange(uint8_t range, uint8_t nTG) { range = constrain (range, 0, 12); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPitchBendRange[nTG] = range; @@ -1285,7 +1537,9 @@ void CMiniDexed::setPitchbendRange(uint8_t range, uint8_t nTG) void CMiniDexed::setPitchbendStep(uint8_t step, uint8_t nTG) { step= constrain (step, 0, 12); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPitchBendStep[nTG] = step; @@ -1298,7 +1552,9 @@ void CMiniDexed::setPortamentoMode(uint8_t mode, uint8_t nTG) { mode= constrain (mode, 0, 1); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPortamentoMode[nTG] = mode; @@ -1310,7 +1566,9 @@ void CMiniDexed::setPortamentoMode(uint8_t mode, uint8_t nTG) void CMiniDexed::setPortamentoGlissando(uint8_t glissando, uint8_t nTG) { glissando = constrain (glissando, 0, 1); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPortamentoGlissando[nTG] = glissando; @@ -1322,7 +1580,9 @@ void CMiniDexed::setPortamentoGlissando(uint8_t glissando, uint8_t nTG) void CMiniDexed::setPortamentoTime(uint8_t time, uint8_t nTG) { time = constrain (time, 0, 99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPortamentoTime[nTG] = time; @@ -1333,7 +1593,9 @@ void CMiniDexed::setPortamentoTime(uint8_t time, uint8_t nTG) void CMiniDexed::setModWheelRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nModulationWheelRange[nTG] = range; @@ -1346,7 +1608,9 @@ void CMiniDexed::setModWheelRange(uint8_t range, uint8_t nTG) void CMiniDexed::setModWheelTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nModulationWheelTarget[nTG] = target; @@ -1358,7 +1622,9 @@ void CMiniDexed::setModWheelTarget(uint8_t target, uint8_t nTG) void CMiniDexed::setFootControllerRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nFootControlRange[nTG]=range; @@ -1371,7 +1637,9 @@ void CMiniDexed::setFootControllerRange(uint8_t range, uint8_t nTG) void CMiniDexed::setFootControllerTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nFootControlTarget[nTG] = target; @@ -1383,7 +1651,9 @@ void CMiniDexed::setFootControllerTarget(uint8_t target, uint8_t nTG) void CMiniDexed::setBreathControllerRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nBreathControlRange[nTG]=range; @@ -1396,7 +1666,9 @@ void CMiniDexed::setBreathControllerRange(uint8_t range, uint8_t nTG) void CMiniDexed::setBreathControllerTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nBreathControlTarget[nTG]=target; @@ -1408,7 +1680,9 @@ void CMiniDexed::setBreathControllerTarget(uint8_t target, uint8_t nTG) void CMiniDexed::setAftertouchRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nAftertouchRange[nTG]=range; @@ -1421,7 +1695,9 @@ void CMiniDexed::setAftertouchRange(uint8_t range, uint8_t nTG) void CMiniDexed::setAftertouchTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nAftertouchTarget[nTG]=target; @@ -1433,7 +1709,9 @@ void CMiniDexed::setAftertouchTarget(uint8_t target, uint8_t nTG) void CMiniDexed::loadVoiceParameters(const uint8_t* data, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); uint8_t voice[161]; @@ -1454,7 +1732,9 @@ void CMiniDexed::loadVoiceParameters(const uint8_t* data, uint8_t nTG) void CMiniDexed::setVoiceDataElement(uint8_t data, uint8_t number, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setVoiceDataElement(constrain(data, 0, 155),constrain(number, 0, 99)); @@ -1464,7 +1744,9 @@ void CMiniDexed::setVoiceDataElement(uint8_t data, uint8_t number, uint8_t nTG) int16_t CMiniDexed::checkSystemExclusive(const uint8_t* pMessage,const uint16_t nLength, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return 0; // Not an active TG + assert (m_pTG[nTG]); return(m_pTG[nTG]->checkSystemExclusive(pMessage, nLength)); @@ -1475,10 +1757,17 @@ void CMiniDexed::getSysExVoiceDump(uint8_t* dest, uint8_t nTG) uint8_t checksum = 0; uint8_t data[155]; - assert (nTG < CConfig::ToneGenerators); - assert (m_pTG[nTG]); - - m_pTG[nTG]->getVoiceData(data); + assert (nTG < CConfig::AllToneGenerators); + if (nTG < m_nToneGenerators) + { + assert (m_pTG[nTG]); + m_pTG[nTG]->getVoiceData(data); + } + else + { + // Not an active TG so grab a default voice + m_SysExFileLoader.GetVoice(CSysExFileLoader::MaxVoiceBankID, CSysExFileLoader::VoicesPerBank+1, data); + } dest[0] = 0xF0; // SysEx start dest[1] = 0x43; // ID=Yamaha @@ -1520,6 +1809,16 @@ unsigned CMiniDexed::GetLastPerformance() return m_PerformanceConfig.GetLastPerformance(); } +unsigned CMiniDexed::GetPerformanceBank() +{ + return m_PerformanceConfig.GetPerformanceBank(); +} + +unsigned CMiniDexed::GetLastPerformanceBank() +{ + return m_PerformanceConfig.GetLastPerformanceBank(); +} + unsigned CMiniDexed::GetActualPerformanceID() { return m_PerformanceConfig.GetActualPerformanceID(); @@ -1530,6 +1829,16 @@ void CMiniDexed::SetActualPerformanceID(unsigned nID) m_PerformanceConfig.SetActualPerformanceID(nID); } +unsigned CMiniDexed::GetActualPerformanceBankID() +{ + return m_PerformanceConfig.GetActualPerformanceBankID(); +} + +void CMiniDexed::SetActualPerformanceBankID(unsigned nBankID) +{ + m_PerformanceConfig.SetActualPerformanceBankID(nBankID); +} + bool CMiniDexed::SetNewPerformance(unsigned nID) { m_bSetNewPerformance = true; @@ -1538,6 +1847,20 @@ bool CMiniDexed::SetNewPerformance(unsigned nID) return true; } +bool CMiniDexed::SetNewPerformanceBank(unsigned nBankID) +{ + m_bSetNewPerformanceBank = true; + m_nSetNewPerformanceBankID = nBankID; + + return true; +} + +void CMiniDexed::SetFirstPerformance(void) +{ + m_bSetFirstPerformance = true; + return; +} + bool CMiniDexed::DoSetNewPerformance (void) { m_bLoadPerformanceBusy = true; @@ -1559,6 +1882,25 @@ bool CMiniDexed::DoSetNewPerformance (void) } } +bool CMiniDexed::DoSetNewPerformanceBank (void) +{ + m_bLoadPerformanceBankBusy = true; + + unsigned nBankID = m_nSetNewPerformanceBankID; + m_PerformanceConfig.SetNewPerformanceBank(nBankID); + + m_bLoadPerformanceBankBusy = false; + return true; +} + +void CMiniDexed::DoSetFirstPerformance(void) +{ + unsigned nID = m_PerformanceConfig.FindFirstPerformance(); + SetNewPerformance(nID); + m_bSetFirstPerformance = false; + return; +} + bool CMiniDexed::SavePerformanceNewFile () { m_bSavePerformanceNewFile = m_PerformanceConfig.GetInternalFolderOk() && m_PerformanceConfig.CheckFreePerformanceSlot(); @@ -1588,7 +1930,7 @@ bool CMiniDexed::DoSavePerformanceNewFile (void) void CMiniDexed::LoadPerformanceParameters(void) { - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < CConfig::AllToneGenerators; nTG++) { BankSelect (m_PerformanceConfig.GetBankNumber (nTG), nTG); @@ -1650,21 +1992,41 @@ void CMiniDexed::SetNewPerformanceName(std::string nName) m_PerformanceConfig.SetNewPerformanceName(nName); } +bool CMiniDexed::IsValidPerformance(unsigned nID) +{ + return m_PerformanceConfig.IsValidPerformance(nID); +} + +bool CMiniDexed::IsValidPerformanceBank(unsigned nBankID) +{ + return m_PerformanceConfig.IsValidPerformanceBank(nBankID); +} + void CMiniDexed::SetVoiceName (std::string VoiceName, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); - char Name[10]; + char Name[11]; strncpy(Name, VoiceName.c_str(),10); + Name[10] = '\0'; m_pTG[nTG]->getName (Name); } bool CMiniDexed::DeletePerformance(unsigned nID) { - m_bDeletePerformance = true; - m_nDeletePerformanceID = nID; + if (m_PerformanceConfig.IsValidPerformance(nID) && m_PerformanceConfig.GetInternalFolderOk()) + { + m_bDeletePerformance = true; + m_nDeletePerformanceID = nID; - return true; + return true; + } + else + { + return false; + } } bool CMiniDexed::DoDeletePerformance(void) @@ -1860,12 +2222,19 @@ void CMiniDexed::UpdateNetwork() m_UI.DisplayWrite ("IP", "Network", IPString, 0, 1); - m_pmDNSPublisher = new CmDNSPublisher (m_pNet); - assert (m_pmDNSPublisher); - - //static const char *ppText[] = {"RTP-MIDI Receiver", nullptr}; // dont bother adding additional data - if (!m_pmDNSPublisher->PublishService (MDNSSERVICENAME, CmDNSPublisher::ServiceTypeAppleMIDI, - 5004)) + CmDNSPublisher *pmDNSPublisher = new CmDNSPublisher (m_pNet); + assert (pmDNSPublisher); + static const char ServiceName[] = "MiniDexed"; + static const char *ppText[] = {"RTP-MIDI Receiver", nullptr}; // TXT record strings + if (!pmDNSPublisher->PublishService (ServiceName, CmDNSPublisher::ServiceTypeAppleMIDI, + 5004, ppText)) + { + LOGPANIC ("Cannot publish mdns service"); + } + static const char *ppText2[] = {"FTP Server", nullptr}; // TXT record strings + static constexpr const char *ServiceTypeFTP = "_ftp._tcp"; + if (!pmDNSPublisher->PublishService (ServiceName, ServiceTypeFTP, + 21, ppText2)) { LOGPANIC ("Cannot publish mdns service"); } @@ -1896,7 +2265,7 @@ bool CMiniDexed::InitNetwork() if (m_pConfig->GetNetworkEnabled () && (strcmp(m_pConfig->GetNetworkType(), "wifi") == 0)) { - LOGNOTE("Initializing Wi-Fi"); + LOGNOTE("Initializing WLAN"); if (m_WLAN.Initialize() && m_WPASupplicant.Initialize()) { @@ -1905,7 +2274,7 @@ bool CMiniDexed::InitNetwork() } else - LOGERR("Failed to initialize Wi-Fi"); + LOGERR("Failed to initialize WLAN"); } else if (m_pConfig->GetNetworkEnabled () && (strcmp(m_pConfig->GetNetworkType(), "ethernet") == 0)) { @@ -1937,4 +2306,4 @@ bool CMiniDexed::InitNetwork() m_pNetDevice = CNetDevice::GetNetDevice(NetDeviceType); } return m_pNet != nullptr; -} \ No newline at end of file +} diff --git a/src/minidexed.h b/src/minidexed.h index 189e45e..7a691c4 100644 --- a/src/minidexed.h +++ b/src/minidexed.h @@ -36,6 +36,7 @@ #include #include #include +#include #include #include #include @@ -58,7 +59,7 @@ class CMiniDexed { public: CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, - CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, FATFS *pFileSystem); + CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, FATFS *pFileSystem); bool Initialize (void); @@ -68,10 +69,14 @@ public: void Run (unsigned nCore); #endif CSysExFileLoader *GetSysExFileLoader (void); + CPerformanceConfig *GetPerformanceConfig (void); void BankSelect (unsigned nBank, unsigned nTG); + void BankSelectPerformance (unsigned nBank); void BankSelectMSB (unsigned nBankMSB, unsigned nTG); + void BankSelectMSBPerformance (unsigned nBankMSB); void BankSelectLSB (unsigned nBankLSB, unsigned nTG); + void BankSelectLSBPerformance (unsigned nBankLSB); void ProgramChange (unsigned nProgram, unsigned nTG); void ProgramChangePerformance (unsigned nProgram); void SetVolume (unsigned nVolume, unsigned nTG); @@ -123,17 +128,27 @@ public: std::string GetPerformanceFileName(unsigned nID); std::string GetPerformanceName(unsigned nID); unsigned GetLastPerformance(); + unsigned GetPerformanceBank(); + unsigned GetLastPerformanceBank(); unsigned GetActualPerformanceID(); void SetActualPerformanceID(unsigned nID); + unsigned GetActualPerformanceBankID(); + void SetActualPerformanceBankID(unsigned nBankID); bool SetNewPerformance(unsigned nID); + bool SetNewPerformanceBank(unsigned nBankID); + void SetFirstPerformance(void); + void DoSetFirstPerformance(void); bool SavePerformanceNewFile (); bool DoSavePerformanceNewFile (void); bool DoSetNewPerformance (void); + bool DoSetNewPerformanceBank (void); bool GetPerformanceSelectToLoad(void); bool SavePerformance (bool bSaveAsDeault); unsigned GetPerformanceSelectChannel (void); void SetPerformanceSelectChannel (unsigned uCh); + bool IsValidPerformance(unsigned nID); + bool IsValidPerformanceBank(unsigned nBankID); // Must match the order in CUIMenu::TParameter enum TParameter @@ -147,6 +162,7 @@ public: ParameterReverbDiffusion, ParameterReverbLevel, ParameterPerformanceSelectChannel, + ParameterPerformanceBank, ParameterUnknown }; @@ -222,7 +238,7 @@ public: private: int16_t ApplyNoteLimits (int16_t pitch, unsigned nTG); // returns < 0 to ignore note - uint8_t m_uchOPMask[CConfig::ToneGenerators]; + uint8_t m_uchOPMask[CConfig::AllToneGenerators]; void LoadPerformanceParameters(void); void ProcessSound (void); const char* GetNetworkDeviceShortName() const; @@ -242,39 +258,44 @@ private: CConfig *m_pConfig; int m_nParameter[ParameterUnknown]; // global (non-TG) parameters - - CDexedAdapter *m_pTG[CConfig::ToneGenerators]; - - unsigned m_nVoiceBankID[CConfig::ToneGenerators]; - unsigned m_nVoiceBankIDMSB[CConfig::ToneGenerators]; - unsigned m_nProgram[CConfig::ToneGenerators]; - unsigned m_nVolume[CConfig::ToneGenerators]; - unsigned m_nPan[CConfig::ToneGenerators]; - int m_nMasterTune[CConfig::ToneGenerators]; - int m_nCutoff[CConfig::ToneGenerators]; - int m_nResonance[CConfig::ToneGenerators]; - unsigned m_nMIDIChannel[CConfig::ToneGenerators]; - unsigned m_nPitchBendRange[CConfig::ToneGenerators]; - unsigned m_nPitchBendStep[CConfig::ToneGenerators]; - unsigned m_nPortamentoMode[CConfig::ToneGenerators]; - unsigned m_nPortamentoGlissando[CConfig::ToneGenerators]; - unsigned m_nPortamentoTime[CConfig::ToneGenerators]; - bool m_bMonoMode[CConfig::ToneGenerators]; + + unsigned m_nToneGenerators; + unsigned m_nPolyphony; + + CDexedAdapter *m_pTG[CConfig::AllToneGenerators]; + + unsigned m_nVoiceBankID[CConfig::AllToneGenerators]; + unsigned m_nVoiceBankIDMSB[CConfig::AllToneGenerators]; + unsigned m_nVoiceBankIDPerformance; + unsigned m_nVoiceBankIDMSBPerformance; + unsigned m_nProgram[CConfig::AllToneGenerators]; + unsigned m_nVolume[CConfig::AllToneGenerators]; + unsigned m_nPan[CConfig::AllToneGenerators]; + int m_nMasterTune[CConfig::AllToneGenerators]; + int m_nCutoff[CConfig::AllToneGenerators]; + int m_nResonance[CConfig::AllToneGenerators]; + unsigned m_nMIDIChannel[CConfig::AllToneGenerators]; + unsigned m_nPitchBendRange[CConfig::AllToneGenerators]; + unsigned m_nPitchBendStep[CConfig::AllToneGenerators]; + unsigned m_nPortamentoMode[CConfig::AllToneGenerators]; + unsigned m_nPortamentoGlissando[CConfig::AllToneGenerators]; + unsigned m_nPortamentoTime[CConfig::AllToneGenerators]; + bool m_bMonoMode[CConfig::AllToneGenerators]; - unsigned m_nModulationWheelRange[CConfig::ToneGenerators]; - unsigned m_nModulationWheelTarget[CConfig::ToneGenerators]; - unsigned m_nFootControlRange[CConfig::ToneGenerators]; - unsigned m_nFootControlTarget[CConfig::ToneGenerators]; - unsigned m_nBreathControlRange[CConfig::ToneGenerators]; - unsigned m_nBreathControlTarget[CConfig::ToneGenerators]; - unsigned m_nAftertouchRange[CConfig::ToneGenerators]; - unsigned m_nAftertouchTarget[CConfig::ToneGenerators]; + unsigned m_nModulationWheelRange[CConfig::AllToneGenerators]; + unsigned m_nModulationWheelTarget[CConfig::AllToneGenerators]; + unsigned m_nFootControlRange[CConfig::AllToneGenerators]; + unsigned m_nFootControlTarget[CConfig::AllToneGenerators]; + unsigned m_nBreathControlRange[CConfig::AllToneGenerators]; + unsigned m_nBreathControlTarget[CConfig::AllToneGenerators]; + unsigned m_nAftertouchRange[CConfig::AllToneGenerators]; + unsigned m_nAftertouchTarget[CConfig::AllToneGenerators]; - unsigned m_nNoteLimitLow[CConfig::ToneGenerators]; - unsigned m_nNoteLimitHigh[CConfig::ToneGenerators]; - int m_nNoteShift[CConfig::ToneGenerators]; + unsigned m_nNoteLimitLow[CConfig::AllToneGenerators]; + unsigned m_nNoteLimitHigh[CConfig::AllToneGenerators]; + int m_nNoteShift[CConfig::AllToneGenerators]; - unsigned m_nReverbSend[CConfig::ToneGenerators]; + unsigned m_nReverbSend[CConfig::AllToneGenerators]; uint8_t m_nRawVoiceData[156]; @@ -289,24 +310,25 @@ private: CPCKeyboard m_PCKeyboard; CSerialMIDIDevice m_SerialMIDI; bool m_bUseSerial; + bool m_bQuadDAC8Chan; CSoundBaseDevice *m_pSoundDevice; bool m_bChannelsSwapped; unsigned m_nQueueSizeFrames; #ifdef ARM_ALLOW_MULTI_CORE - unsigned m_nActiveTGsLog2; +// unsigned m_nActiveTGsLog2; volatile TCoreStatus m_CoreStatus[CORES]; volatile unsigned m_nFramesToProcess; - float32_t m_OutputLevel[CConfig::ToneGenerators][CConfig::MaxChunkSize]; + float32_t m_OutputLevel[CConfig::AllToneGenerators][CConfig::MaxChunkSize]; #endif CPerformanceTimer m_GetChunkTimer; bool m_bProfileEnabled; AudioEffectPlateReverb* reverb; - AudioStereoMixer* tg_mixer; - AudioStereoMixer* reverb_send_mixer; + AudioStereoMixer* tg_mixer; + AudioStereoMixer* reverb_send_mixer; CSpinLock m_ReverbSpinLock; @@ -324,9 +346,13 @@ private: bool m_bSavePerformanceNewFile; bool m_bSetNewPerformance; unsigned m_nSetNewPerformanceID; + bool m_bSetNewPerformanceBank; + unsigned m_nSetNewPerformanceBankID; + bool m_bSetFirstPerformance; bool m_bDeletePerformance; unsigned m_nDeletePerformanceID; bool m_bLoadPerformanceBusy; + bool m_bLoadPerformanceBankBusy; bool m_bSaveAsDeault; diff --git a/src/minidexed.ini b/src/minidexed.ini index cd5ea1a..41c5a12 100644 --- a/src/minidexed.ini +++ b/src/minidexed.ini @@ -17,7 +17,7 @@ EngineType=1 MIDIBaudRate=31250 #MIDIThru=umidi1,ttyS1 IgnoreAllNotesOff=0 -MIDIAutoVoiceDumpOnPC=1 +MIDIAutoVoiceDumpOnPC=0 HeaderlessSysExVoices=0 # Program Change enable # 0 = Ignore all Program Change messages. @@ -55,6 +55,27 @@ SSD1306LCDHeight=32 SSD1306LCDRotate=0 SSD1306LCDMirror=0 +# ST7789 LCD +# SPIBus=0 for any RPi (GPIO 10,11,8,7). +# Note: Leave blank (default) if no SPI device required. +# Select=0|1 for CE0 or CE1 +# Data = GPIO pin number +# Optional: Reset, Backlight = GPIO pin numbers +# Rotation=0,90,180,270 +# SmallFont=0 (default), 1 +# +# For a 240 wide display set LCDColumns=15 with LCDRows=2 +SPIBus= +ST7789Enabled=0 +ST7789Data= +ST7789Select= +ST7789Reset= +ST7789Backlight= +ST7789Width=240 +ST7789Height=240 +ST7789Rotation=0 +ST7789SmallFont=0 + # Default is 16x2 display (e.g. HD44780) LCDColumns=16 LCDRows=2 @@ -116,5 +137,10 @@ EncoderPinData=9 MIDIDumpEnabled=0 ProfileEnabled=0 +# Network +NetworkEnabled=0 +NetworkDHCP=1 +NetworkType=wifi + # Performance PerformanceSelectToLoad=1 diff --git a/src/performanceconfig.cpp b/src/performanceconfig.cpp index 853ccf5..8a182e7 100644 --- a/src/performanceconfig.cpp +++ b/src/performanceconfig.cpp @@ -28,8 +28,15 @@ LOGMODULE ("Performance"); +//#define VERBOSE_DEBUG + +#define PERFORMANCE_DIR "performance" +#define DEFAULT_PERFORMANCE_FILENAME "performance.ini" +#define DEFAULT_PERFORMANCE_NAME "Default" +#define DEFAULT_PERFORMANCE_BANK_NAME "Default" + CPerformanceConfig::CPerformanceConfig (FATFS *pFileSystem) -: m_Properties ("performance.ini", pFileSystem) +: m_Properties (DEFAULT_PERFORMANCE_FILENAME, pFileSystem) { m_pFileSystem = pFileSystem; } @@ -38,6 +45,59 @@ CPerformanceConfig::~CPerformanceConfig (void) { } +bool CPerformanceConfig::Init (unsigned nToneGenerators) +{ + // Different versions of Pi allow different TG configurations. + // On loading, performances will load up to the number of + // supported/active TGs. + // + // On saving, the active/supported number of TGs is used. + // + // This means that if an 8TG performance is loaded into + // a 16 TG system and then saved, the saved performance + // will include all 16 TG configurations. + // + m_nToneGenerators = nToneGenerators; + + // Check intermal performance directory exists + DIR Directory; + FRESULT Result; + //Check if internal "performance" directory exists + Result = f_opendir (&Directory, "SD:/" PERFORMANCE_DIR); + if (Result == FR_OK) + { + m_bPerformanceDirectoryExists=true; + Result = f_closedir (&Directory); + } + else + { + m_bPerformanceDirectoryExists = false; + } + + // List banks if present + ListPerformanceBanks(); + +#ifdef VERBOSE_DEBUG +#warning "PerformanceConfig in verbose debug printing mode" + LOGNOTE("Testing loading of banks"); + for (unsigned i=0; i= NUM_PERFORMANCES) { + if (!m_bPerformanceDirectoryExists) + { + // Nothing can be done if there is no performance directory + LOGNOTE("Performance directory does not exist"); + return false; + } + if (m_nLastPerformance >= NUM_PERFORMANCES) { // No space left for new performances LOGWARN ("No space left for new performance"); return false; } + + // Note: New performances are created at the end of the currently selected bank. + // Sould we default to a new bank just for user-performances? + // + // There is the possibility of MIDI changing the Bank Number and the user + // not spotting the bank has changed... + // + // Another option would be to find empty slots in the current bank + // rather than always add at the end. + // + // Sorting this out is left for a future update. std::string sPerformanceName = NewPerformanceName; NewPerformanceName=""; - nActualPerformance=nLastPerformance; + unsigned nNewPerformance = m_nLastPerformance + 1; std::string nFileName; std::string nPath; std::string nIndex = "000000"; - nIndex += std::to_string(++nLastFileIndex); + nIndex += std::to_string(nNewPerformance+1); // Index on disk = index in memory+1 nIndex = nIndex.substr(nIndex.length()-6,6); @@ -793,10 +942,11 @@ bool CPerformanceConfig::CreateNewPerformanceFile(void) nFileName +=sPerformanceName.substr(0,14); } nFileName += ".ini"; - m_nPerformanceFileName[nLastPerformance]= nFileName; + m_PerformanceFileName[nNewPerformance]= sPerformanceName; nPath = "SD:/" ; nPath += PERFORMANCE_DIR; + nPath += AddPerformanceBankDirName(m_nPerformanceBank); nPath += "/"; nFileName = nPath + nFileName; @@ -804,17 +954,18 @@ bool CPerformanceConfig::CreateNewPerformanceFile(void) FRESULT Result = f_open (&File, nFileName.c_str(), FA_WRITE | FA_CREATE_ALWAYS); if (Result != FR_OK) { - m_nPerformanceFileName[nLastPerformance]=nullptr; + m_PerformanceFileName[nNewPerformance]=nullptr; return false; } if (f_close (&File) != FR_OK) { - m_nPerformanceFileName[nLastPerformance]=nullptr; + m_PerformanceFileName[nNewPerformance]=nullptr; return false; } - nLastPerformance++; + m_nLastPerformance = nNewPerformance; + m_nActualPerformance = nNewPerformance; new (&m_Properties) CPropertiesFatFsFile(nFileName.c_str(), m_pFileSystem); return true; @@ -822,88 +973,124 @@ bool CPerformanceConfig::CreateNewPerformanceFile(void) bool CPerformanceConfig::ListPerformances() { - nInternalFolderOk=false; - nExternalFolderOk=false; // for future USB implementation - nLastPerformance=0; - nLastFileIndex=0; - m_nPerformanceFileName[nLastPerformance++]="performance.ini"; // in order to assure retrocompatibility - - unsigned nPIndex; - DIR Directory; - FILINFO FileInfo; - FRESULT Result; - //Check if internal "performance" directory exists - Result = f_opendir (&Directory, "SD:/" PERFORMANCE_DIR); - if (Result == FR_OK) + // Clear any existing lists of performances + for (unsigned i=0; i= NUM_PERFORMANCES) { - LOGNOTE ("Skipping performance %s", FileInfo.fname); - } else { - if (!(FileInfo.fattrib & (AM_HID | AM_SYS))) + if (!(FileInfo.fattrib & (AM_HID | AM_SYS))) + { + std::string OriFileName = FileInfo.fname; + size_t nLen = OriFileName.length(); + if ( nLen > 8 && nLen <26 && strcmp(OriFileName.substr(6,1).c_str(), "_")==0) { - std::string FileName = FileInfo.fname; - size_t nLen = FileName.length(); - if ( nLen > 8 && nLen <26 && strcmp(FileName.substr(6,1).c_str(), "_")==0) - { - nPIndex=stoi(FileName.substr(0,6)); - if(nPIndex > nLastFileIndex) + // Note: m_nLastPerformance - refers to the number (index) of the last performance in memory, + // which includes a default performance. + // + // Filenames on the disk start from 1 to match what the user might see in MIDI. + // So this means that actually file 000001_ will correspond to index position [0]. + // For the default bank though, ID 1 is the default performance, so will already exist. + // m_PerformanceFileName[0] = default performance (file 000001) + // m_PerformanceFileName[1] = first available on-disk performance (file 000002) + // + // Note2: filenames assume 6 digits, underscore, name, finally ".ini" + // i.e. 123456_Performance Name.ini + // + nPIndex=stoi(OriFileName.substr(0,6)); + if ((nPIndex < 1) || (nPIndex >= (NUM_PERFORMANCES+1))) + { + // Index is out of range - skip to next file + LOGNOTE ("Performance number out of range: %s (%d to %d)", FileInfo.fname, 1, NUM_PERFORMANCES); + } + else + { + // Convert from "user facing" 1..indexed number to internal 0..indexed + nPIndex = nPIndex-1; + if (m_PerformanceFileName[nPIndex].empty()) { - nLastFileIndex=nPIndex; - } + if(nPIndex > m_nLastPerformance) + { + m_nLastPerformance=nPIndex; + } + + std::string FileName = OriFileName.substr(0,OriFileName.length()-4).substr(7,14); - m_nPerformanceFileName[nLastPerformance++]= FileName; - } + m_PerformanceFileName[nPIndex] = FileName; +#ifdef VERBOSE_DEBUG + LOGNOTE ("Loading performance %s (%d, %s)", OriFileName.c_str(), nPIndex, FileName.c_str()); +#endif + } + else + { + LOGNOTE ("Duplicate performance %s", OriFileName.c_str()); + } + } } } Result = f_findnext (&Directory, &FileInfo); } - // sort by performance number-name - if (nLastPerformance > 2) - { - sort (m_nPerformanceFileName+1, m_nPerformanceFileName + nLastPerformance); // default is always on first place. %%%%%%%%%%%%%%%% - } + f_closedir (&Directory); } - LOGNOTE ("Number of Performances: %d", nLastPerformance); - - return nInternalFolderOk; -} - + return true; +} void CPerformanceConfig::SetNewPerformance (unsigned nID) { - nActualPerformance=nID; - std::string FileN = ""; - if (nID != 0) // in order to assure retrocompatibility + assert (nID < NUM_PERFORMANCES); + m_nActualPerformance=nID; + std::string FileN = GetPerformanceFullFilePath(nID); + + new (&m_Properties) CPropertiesFatFsFile(FileN.c_str(), m_pFileSystem); +#ifdef VERBOSE_DEBUG + LOGNOTE("Selecting Performance: %d (%s)", nID+1, FileN.c_str()); +#endif +} + +unsigned CPerformanceConfig::FindFirstPerformance (void) +{ + for (int nID=0; nID < NUM_PERFORMANCES; nID++) + { + if (IsValidPerformance(nID)) { - FileN += PERFORMANCE_DIR; - FileN += "/"; + return nID; } - FileN += m_nPerformanceFileName[nID]; - new (&m_Properties) CPropertiesFatFsFile(FileN.c_str(), m_pFileSystem); - + } + + return 0; // Even though 0 is a valid performance, not much else to do } std::string CPerformanceConfig::GetNewPerformanceDefaultName(void) { std::string nIndex = "000000"; - nIndex += std::to_string(nLastFileIndex+1); + nIndex += std::to_string(m_nLastPerformance+1+1); // Convert from internal 0.. index to a file-based 1.. index to show the user nIndex = nIndex.substr(nIndex.length()-6,6); return "Perf" + nIndex; } @@ -923,31 +1110,229 @@ void CPerformanceConfig::SetNewPerformanceName(std::string nName) bool CPerformanceConfig::DeletePerformance(unsigned nID) { + if (!m_bPerformanceDirectoryExists) + { + // Nothing can be done if there is no performance directory + LOGNOTE("Performance directory does not exist"); + return false; + } bool bOK = false; - if(nID == 0){return bOK;} // default (performance.ini at root directory) can't be deleted + if((m_nPerformanceBank == 0) && (nID == 0)){return bOK;} // default (performance.ini at root directory) can't be deleted DIR Directory; FILINFO FileInfo; std::string FileN = "SD:/"; FileN += PERFORMANCE_DIR; - + FileN += AddPerformanceBankDirName(m_nPerformanceBank); - FRESULT Result = f_findfirst (&Directory, &FileInfo, FileN.c_str(), m_nPerformanceFileName[nID].c_str()); + FRESULT Result = f_findfirst (&Directory, &FileInfo, FileN.c_str(), GetPerformanceFileName(nID).c_str()); if (Result == FR_OK && FileInfo.fname[0]) { FileN += "/"; - FileN += m_nPerformanceFileName[nID]; + FileN += GetPerformanceFileName(nID); Result=f_unlink (FileN.c_str()); if (Result == FR_OK) { SetNewPerformance(0); - nActualPerformance =0; + m_nActualPerformance =0; //nMenuSelectedPerformance=0; - m_nPerformanceFileName[nID]="ZZZZZZ"; - sort (m_nPerformanceFileName+1, m_nPerformanceFileName + nLastPerformance); // test si va con -1 o no - --nLastPerformance; - m_nPerformanceFileName[nLastPerformance]=nullptr; + m_PerformanceFileName[nID].clear(); + // If this was the last performance in the bank... + if (nID == m_nLastPerformance) + { + do + { + // Find the new last performance + m_nLastPerformance--; + } while (!IsValidPerformance(m_nLastPerformance) && (m_nLastPerformance > 0)); + } bOK=true; } + else + { + LOGNOTE ("Failed to delete %s", FileN.c_str()); + } } return bOK; } + +bool CPerformanceConfig::ListPerformanceBanks() +{ + m_nPerformanceBank = 0; + m_nLastPerformance = 0; + m_nLastPerformanceBank = 0; + + // Open performance directory + DIR Directory; + FILINFO FileInfo; + FRESULT Result; + Result = f_opendir (&Directory, "SD:/" PERFORMANCE_DIR); + if (Result != FR_OK) + { + // No performance directory, so no performance banks. + // So nothing else to do here + LOGNOTE ("No performance banks detected"); + m_bPerformanceDirectoryExists = false; + return false; + } + + unsigned nNumBanks = 0; + + // Add in the default performance directory as the first bank + m_PerformanceBankName[0] = DEFAULT_PERFORMANCE_BANK_NAME; + nNumBanks = 1; + m_nLastPerformanceBank = 0; + + // List directories with names in format 01_Perf Bank Name + Result = f_findfirst (&Directory, &FileInfo, "SD:/" PERFORMANCE_DIR, "*"); + for (unsigned i = 0; Result == FR_OK && FileInfo.fname[0]; i++) + { + // Check to see if it is a directory + if ((FileInfo.fattrib & AM_DIR) != 0) + { + // Looking for Performance banks of the form: 01_Perf Bank Name + // So positions 0,1,2 = decimal digit + // position 3 = "_" + // positions 4.. = actual name + // + std::string OriFileName = FileInfo.fname; + size_t nLen = OriFileName.length(); + if ( nLen > 4 && nLen <26 && strcmp(OriFileName.substr(3,1).c_str(), "_")==0) + { + unsigned nBankIndex = stoi(OriFileName.substr(0,3)); + // Recall user index numbered 002..NUM_PERFORMANCE_BANKS + // NB: Bank 001 is reserved for the default performance directory + if ((nBankIndex > 0) && (nBankIndex <= NUM_PERFORMANCE_BANKS)) + { + // Convert from "user facing" 1..indexed number to internal 0..indexed + nBankIndex = nBankIndex-1; + if (m_PerformanceBankName[nBankIndex].empty()) + { + std::string BankName = OriFileName.substr(4,nLen); + + m_PerformanceBankName[nBankIndex] = BankName; +#ifdef VERBOSE_DEBUG + LOGNOTE ("Found performance bank %s (%d, %s)", OriFileName.c_str(), nBankIndex, BankName.c_str()); +#endif + nNumBanks++; + if (nBankIndex > m_nLastPerformanceBank) + { + m_nLastPerformanceBank = nBankIndex; + } + } + else + { + LOGNOTE ("Duplicate Performance Bank: %s", FileInfo.fname); + if (nBankIndex==0) + { + LOGNOTE ("(Bank 001 is the default performance directory)"); + } + } + } + else + { + LOGNOTE ("Performance Bank number out of range: %s (%d to %d)", FileInfo.fname, 1, NUM_PERFORMANCE_BANKS); + } + } + else + { +#ifdef VERBOSE_DEBUG + LOGNOTE ("Skipping: %s", FileInfo.fname); +#endif + } + } + + Result = f_findnext (&Directory, &FileInfo); + } + + if (nNumBanks > 0) + { + LOGNOTE ("Number of Performance Banks: %d (last = %d)", nNumBanks, m_nLastPerformanceBank+1); + } + + f_closedir (&Directory); + return true; +} + +void CPerformanceConfig::SetNewPerformanceBank(unsigned nBankID) +{ + assert (nBankID < NUM_PERFORMANCE_BANKS); + if (IsValidPerformanceBank(nBankID)) + { +#ifdef VERBOSE_DEBUG + LOGNOTE("Selecting Performance Bank: %d", nBankID+1); +#endif + m_nPerformanceBank = nBankID; + m_nActualPerformanceBank = nBankID; + ListPerformances(); + } + else + { +#ifdef VERBOSE_DEBUG + LOGNOTE("Not selecting invalid Performance Bank: %d", nBankID+1); +#endif + } +} + +unsigned CPerformanceConfig::GetPerformanceBank(void) +{ + return m_nPerformanceBank; +} + +std::string CPerformanceConfig::GetPerformanceBankName(unsigned nBankID) +{ + assert (nBankID < NUM_PERFORMANCE_BANKS); + if (IsValidPerformanceBank(nBankID)) + { + return m_PerformanceBankName[nBankID]; + } + else + { + return DEFAULT_PERFORMANCE_BANK_NAME; + } +} + +std::string CPerformanceConfig::AddPerformanceBankDirName(unsigned nBankID) +{ + assert (nBankID < NUM_PERFORMANCE_BANKS); + if (IsValidPerformanceBank(nBankID)) + { + // Performance Banks directories in format "001_Bank Name" + std::string Index; + if (nBankID == 0) + { + // Legacy: Bank 1 is the default performance directory + return ""; + } + + if (nBankID < 9) + { + Index = "00" + std::to_string(nBankID+1); + } + else if (nBankID < 99) + { + Index = "0" + std::to_string(nBankID+1); + } + else + { + Index = std::to_string(nBankID+1); + } + + return "/" + Index + "_" + m_PerformanceBankName[nBankID]; + } + else + { + return ""; + } +} + +bool CPerformanceConfig::IsValidPerformanceBank(unsigned nBankID) +{ + if (nBankID >= NUM_PERFORMANCE_BANKS) { + return false; + } + if (m_PerformanceBankName[nBankID].empty()) + { + return false; + } + return true; +} diff --git a/src/performanceconfig.h b/src/performanceconfig.h index c151c7a..c9c4b27 100644 --- a/src/performanceconfig.h +++ b/src/performanceconfig.h @@ -27,14 +27,16 @@ #include #include #define NUM_VOICE_PARAM 156 -#define PERFORMANCE_DIR "performance" -#define NUM_PERFORMANCES 256 +#define NUM_PERFORMANCES 128 +#define NUM_PERFORMANCE_BANKS 128 class CPerformanceConfig // Performance configuration { public: CPerformanceConfig (FATFS *pFileSystem); ~CPerformanceConfig (void); + + bool Init (unsigned nToneGenerators); bool Load (void); @@ -122,59 +124,76 @@ public: bool ListPerformances(); //std::string m_DirName; void SetNewPerformance (unsigned nID); + unsigned FindFirstPerformance (void); std::string GetPerformanceFileName(unsigned nID); + std::string GetPerformanceFullFilePath(unsigned nID); std::string GetPerformanceName(unsigned nID); unsigned GetLastPerformance(); + unsigned GetLastPerformanceBank(); void SetActualPerformanceID(unsigned nID); unsigned GetActualPerformanceID(); + void SetActualPerformanceBankID(unsigned nBankID); + unsigned GetActualPerformanceBankID(); bool CreateNewPerformanceFile(void); bool GetInternalFolderOk(); std::string GetNewPerformanceDefaultName(void); void SetNewPerformanceName(std::string nName); bool DeletePerformance(unsigned nID); bool CheckFreePerformanceSlot(void); + std::string AddPerformanceBankDirName(unsigned nBankID); + bool IsValidPerformance(unsigned nID); + + bool ListPerformanceBanks(void); + void SetNewPerformanceBank(unsigned nBankID); + unsigned GetPerformanceBank(void); + std::string GetPerformanceBankName(unsigned nBankID); + bool IsValidPerformanceBank(unsigned nBankID); private: CPropertiesFatFsFile m_Properties; - - unsigned m_nBankNumber[CConfig::ToneGenerators]; - unsigned m_nVoiceNumber[CConfig::ToneGenerators]; - unsigned m_nMIDIChannel[CConfig::ToneGenerators]; - unsigned m_nVolume[CConfig::ToneGenerators]; - unsigned m_nPan[CConfig::ToneGenerators]; - int m_nDetune[CConfig::ToneGenerators]; - unsigned m_nCutoff[CConfig::ToneGenerators]; - unsigned m_nResonance[CConfig::ToneGenerators]; - unsigned m_nNoteLimitLow[CConfig::ToneGenerators]; - unsigned m_nNoteLimitHigh[CConfig::ToneGenerators]; - int m_nNoteShift[CConfig::ToneGenerators]; - int m_nReverbSend[CConfig::ToneGenerators]; - unsigned m_nPitchBendRange[CConfig::ToneGenerators]; - unsigned m_nPitchBendStep[CConfig::ToneGenerators]; - unsigned m_nPortamentoMode[CConfig::ToneGenerators]; - unsigned m_nPortamentoGlissando[CConfig::ToneGenerators]; - unsigned m_nPortamentoTime[CConfig::ToneGenerators]; - std::string m_nVoiceDataTxt[CConfig::ToneGenerators]; - bool m_bMonoMode[CConfig::ToneGenerators]; - - unsigned m_nModulationWheelRange[CConfig::ToneGenerators]; - unsigned m_nModulationWheelTarget[CConfig::ToneGenerators]; - unsigned m_nFootControlRange[CConfig::ToneGenerators]; - unsigned m_nFootControlTarget[CConfig::ToneGenerators]; - unsigned m_nBreathControlRange[CConfig::ToneGenerators]; - unsigned m_nBreathControlTarget[CConfig::ToneGenerators]; - unsigned m_nAftertouchRange[CConfig::ToneGenerators]; - unsigned m_nAftertouchTarget[CConfig::ToneGenerators]; - - unsigned nLastPerformance; - unsigned nLastFileIndex; - unsigned nActualPerformance = 0; + + unsigned m_nToneGenerators; + + unsigned m_nBankNumber[CConfig::AllToneGenerators]; + unsigned m_nVoiceNumber[CConfig::AllToneGenerators]; + unsigned m_nMIDIChannel[CConfig::AllToneGenerators]; + unsigned m_nVolume[CConfig::AllToneGenerators]; + unsigned m_nPan[CConfig::AllToneGenerators]; + int m_nDetune[CConfig::AllToneGenerators]; + unsigned m_nCutoff[CConfig::AllToneGenerators]; + unsigned m_nResonance[CConfig::AllToneGenerators]; + unsigned m_nNoteLimitLow[CConfig::AllToneGenerators]; + unsigned m_nNoteLimitHigh[CConfig::AllToneGenerators]; + int m_nNoteShift[CConfig::AllToneGenerators]; + int m_nReverbSend[CConfig::AllToneGenerators]; + unsigned m_nPitchBendRange[CConfig::AllToneGenerators]; + unsigned m_nPitchBendStep[CConfig::AllToneGenerators]; + unsigned m_nPortamentoMode[CConfig::AllToneGenerators]; + unsigned m_nPortamentoGlissando[CConfig::AllToneGenerators]; + unsigned m_nPortamentoTime[CConfig::AllToneGenerators]; + std::string m_nVoiceDataTxt[CConfig::AllToneGenerators]; + bool m_bMonoMode[CConfig::AllToneGenerators]; + + unsigned m_nModulationWheelRange[CConfig::AllToneGenerators]; + unsigned m_nModulationWheelTarget[CConfig::AllToneGenerators]; + unsigned m_nFootControlRange[CConfig::AllToneGenerators]; + unsigned m_nFootControlTarget[CConfig::AllToneGenerators]; + unsigned m_nBreathControlRange[CConfig::AllToneGenerators]; + unsigned m_nBreathControlTarget[CConfig::AllToneGenerators]; + unsigned m_nAftertouchRange[CConfig::AllToneGenerators]; + unsigned m_nAftertouchTarget[CConfig::AllToneGenerators]; + + unsigned m_nLastPerformance; + unsigned m_nActualPerformance = 0; + unsigned m_nActualPerformanceBank = 0; + unsigned m_nPerformanceBank; + unsigned m_nLastPerformanceBank; + bool m_bPerformanceDirectoryExists; //unsigned nMenuSelectedPerformance = 0; - std::string m_nPerformanceFileName[NUM_PERFORMANCES]; + std::string m_PerformanceFileName[NUM_PERFORMANCES]; + std::string m_PerformanceBankName[NUM_PERFORMANCE_BANKS]; FATFS *m_pFileSystem; - bool nInternalFolderOk=false; - bool nExternalFolderOk=false; // for future USB implementation std::string NewPerformanceName=""; bool m_bCompressorEnable; diff --git a/src/serialmididevice.cpp b/src/serialmididevice.cpp index 883fd4d..3fe01e8 100644 --- a/src/serialmididevice.cpp +++ b/src/serialmididevice.cpp @@ -28,11 +28,15 @@ LOGMODULE("serialmididevice"); +// There are several UART options - see circle/include/serial.h +// 0 corresponds to GP14/GP15 on all RPi versions. +#define SERIAL_MIDI_DEVICE 0 + CSerialMIDIDevice::CSerialMIDIDevice (CMiniDexed *pSynthesizer, CInterruptSystem *pInterrupt, CConfig *pConfig, CUserInterface *pUI) : CMIDIDevice (pSynthesizer, pConfig, pUI), m_pConfig (pConfig), - m_Serial (pInterrupt, TRUE), + m_Serial (pInterrupt, TRUE, SERIAL_MIDI_DEVICE), m_nSerialState (0), m_nSysEx (0), m_SendBuffer (&m_Serial) @@ -71,7 +75,7 @@ void CSerialMIDIDevice::Process (void) return; } - if (m_pConfig->GetMIDIDumpEnabled ()) +/* if (m_pConfig->GetMIDIDumpEnabled ()) { printf("Incoming MIDI data:"); for (uint16_t i = 0; i < nResult; i++) @@ -81,7 +85,7 @@ void CSerialMIDIDevice::Process (void) printf(" 0x%02x",Buffer[i]); } printf("\n"); - } + }*/ // Process MIDI messages // See: https://www.midi.org/specifications/item/table-1-summary-of-midi-message diff --git a/src/serialmididevice.h b/src/serialmididevice.h index 1a5b465..fdf61b2 100644 --- a/src/serialmididevice.h +++ b/src/serialmididevice.h @@ -30,9 +30,6 @@ #include #include -#define MAX_DX7_SYSEX_LENGTH 4104 -#define MAX_MIDI_MESSAGE MAX_DX7_SYSEX_LENGTH - class CMiniDexed; class CSerialMIDIDevice : public CMIDIDevice diff --git a/src/sysexfileloader.cpp b/src/sysexfileloader.cpp index ea5ba5c..6b83f5b 100644 --- a/src/sysexfileloader.cpp +++ b/src/sysexfileloader.cpp @@ -273,6 +273,23 @@ std::string CSysExFileLoader::GetBankName (unsigned nBankID) return "NO NAME"; } +std::string CSysExFileLoader::GetVoiceName (unsigned nBankID, unsigned nVoiceID) +{ + if ((nBankID <= MaxVoiceBankID) && (nVoiceID < VoicesPerBank)) + { + if (IsValidBank(nBankID)) + { + // The name is the last 10 characters of the voice data + char sVoiceName[11]; + strncpy (sVoiceName, (char *)((char *)&(m_pVoiceBank[nBankID]->Voice[nVoiceID]) + SizePackedVoice - 10), 10); + sVoiceName[10] = 0; + std::string result(sVoiceName); + return result; + } + } + return "INIT VOICE"; +} + unsigned CSysExFileLoader::GetNextBankUp (unsigned nBankID) { // Find the next loaded bank "up" from the provided bank ID diff --git a/src/sysexfileloader.h b/src/sysexfileloader.h index 4918db6..272d775 100644 --- a/src/sysexfileloader.h +++ b/src/sysexfileloader.h @@ -60,6 +60,7 @@ public: void Load (bool bHeaderlessSysExVoices = false); std::string GetBankName (unsigned nBankID); // 0 .. MaxVoiceBankID + std::string GetVoiceName (unsigned nBankID, unsigned nVoice); // 0 .. MaxVoiceBankID, 0 .. VoicesPerBank-1 unsigned GetNumHighestBank (); // 0 .. MaxVoiceBankID bool IsValidBank (unsigned nBankID); unsigned GetNextBankUp (unsigned nBankID); diff --git a/src/uibuttons.h b/src/uibuttons.h index 47d128d..44dcca0 100644 --- a/src/uibuttons.h +++ b/src/uibuttons.h @@ -26,7 +26,7 @@ #include "config.h" #define BUTTONS_UPDATE_NUM_TICKS 100 -#define DEBOUNCE_TIME 100 +#define DEBOUNCE_TIME 20 #define MAX_GPIO_BUTTONS 9 // 5 UI buttons, 4 Program/TG Select buttons #define MAX_MIDI_BUTTONS 9 #define MAX_BUTTONS (MAX_GPIO_BUTTONS+MAX_MIDI_BUTTONS) diff --git a/src/uimenu.cpp b/src/uimenu.cpp index 93e6b0e..8b72425 100644 --- a/src/uimenu.cpp +++ b/src/uimenu.cpp @@ -51,6 +51,16 @@ const CUIMenu::TMenuItem CUIMenu::s_MainMenu[] = {"TG6", MenuHandler, s_TGMenu, 5}, {"TG7", MenuHandler, s_TGMenu, 6}, {"TG8", MenuHandler, s_TGMenu, 7}, +#if (RASPPI==4 || RASPPI==5) + {"TG9", MenuHandler, s_TGMenu, 8}, + {"TG10", MenuHandler, s_TGMenu, 9}, + {"TG11", MenuHandler, s_TGMenu, 10}, + {"TG12", MenuHandler, s_TGMenu, 11}, + {"TG13", MenuHandler, s_TGMenu, 12}, + {"TG14", MenuHandler, s_TGMenu, 13}, + {"TG15", MenuHandler, s_TGMenu, 14}, + {"TG16", MenuHandler, s_TGMenu, 15}, +#endif #endif {"Effects", MenuHandler, s_EffectsMenu}, {"Performance", MenuHandler, s_PerformanceMenu}, @@ -214,7 +224,8 @@ const CUIMenu::TParameter CUIMenu::s_GlobalParameter[CMiniDexed::ParameterUnknow {0, 99, 1}, // ParameterReverbLowPass {0, 99, 1}, // ParameterReverbDiffusion {0, 99, 1}, // ParameterReverbLevel - {0, CMIDIDevice::ChannelUnknown-1, 1, ToMIDIChannel} // ParameterPerformanceSelectChannel + {0, CMIDIDevice::ChannelUnknown-1, 1, ToMIDIChannel}, // ParameterPerformanceSelectChannel + {0, NUM_PERFORMANCE_BANKS, 1} // ParameterPerformanceBank }; // must match CMiniDexed::TTGParameter @@ -327,14 +338,16 @@ const CUIMenu::TMenuItem CUIMenu::s_PerformanceMenu[] = {"Load", PerformanceMenu, 0, 0}, {"Save", MenuHandler, s_SaveMenu}, {"Delete", PerformanceMenu, 0, 1}, + {"Bank", EditPerformanceBankNumber, 0, 0}, {"PCCH", EditGlobalParameter, 0, CMiniDexed::ParameterPerformanceSelectChannel}, {0} }; -CUIMenu::CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed) +CUIMenu::CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed, CConfig *pConfig) : m_pUI (pUI), m_pMiniDexed (pMiniDexed), + m_pConfig (pConfig), m_pParentMenu (s_MenuRoot), m_pCurrentMenu (s_MainMenu), m_nCurrentMenuItem (0), @@ -342,23 +355,27 @@ CUIMenu::CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed) m_nCurrentParameter (0), m_nCurrentMenuDepth (0) { -#ifndef ARM_ALLOW_MULTI_CORE - // If there is just one core, then there is only a single - // tone generator so start on the TG1 menu... - m_pParentMenu = s_MainMenu; - m_pCurrentMenu = s_TGMenu; - m_nCurrentMenuItem = 0; - m_nCurrentSelection = 0; - m_nCurrentParameter = 0; - m_nCurrentMenuDepth = 1; + assert (m_pConfig); + m_nToneGenerators = m_pConfig->GetToneGenerators(); - // Place the "root" menu at the top of the stack - m_MenuStackParent[0] = s_MenuRoot; - m_MenuStackMenu[0] = s_MainMenu; - m_nMenuStackItem[0] = 0; - m_nMenuStackSelection[0] = 0; - m_nMenuStackParameter[0] = 0; -#endif + if (m_nToneGenerators == 1) + { + // If there is just one core, then there is only a single + // tone generator so start on the TG1 menu... + m_pParentMenu = s_MainMenu; + m_pCurrentMenu = s_TGMenu; + m_nCurrentMenuItem = 0; + m_nCurrentSelection = 0; + m_nCurrentParameter = 0; + m_nCurrentMenuDepth = 1; + + // Place the "root" menu at the top of the stack + m_MenuStackParent[0] = s_MenuRoot; + m_MenuStackMenu[0] = s_MainMenu; + m_nMenuStackItem[0] = 0; + m_nMenuStackSelection[0] = 0; + m_nMenuStackParameter[0] = 0; + } } void CUIMenu::EventHandler (TMenuEvent Event) @@ -381,28 +398,31 @@ void CUIMenu::EventHandler (TMenuEvent Event) break; case MenuEventHome: -#ifdef ARM_ALLOW_MULTI_CORE - m_pParentMenu = s_MenuRoot; - m_pCurrentMenu = s_MainMenu; - m_nCurrentMenuItem = 0; - m_nCurrentSelection = 0; - m_nCurrentParameter = 0; - m_nCurrentMenuDepth = 0; -#else - // "Home" is the TG0 menu if only one TG active - m_pParentMenu = s_MainMenu; - m_pCurrentMenu = s_TGMenu; - m_nCurrentMenuItem = 0; - m_nCurrentSelection = 0; - m_nCurrentParameter = 0; - m_nCurrentMenuDepth = 1; - // Place the "root" menu at the top of the stack - m_MenuStackParent[0] = s_MenuRoot; - m_MenuStackMenu[0] = s_MainMenu; - m_nMenuStackItem[0] = 0; - m_nMenuStackSelection[0] = 0; - m_nMenuStackParameter[0] = 0; -#endif + if (m_nToneGenerators == 1) + { + // "Home" is the TG0 menu if only one TG active + m_pParentMenu = s_MainMenu; + m_pCurrentMenu = s_TGMenu; + m_nCurrentMenuItem = 0; + m_nCurrentSelection = 0; + m_nCurrentParameter = 0; + m_nCurrentMenuDepth = 1; + // Place the "root" menu at the top of the stack + m_MenuStackParent[0] = s_MenuRoot; + m_MenuStackMenu[0] = s_MainMenu; + m_nMenuStackItem[0] = 0; + m_nMenuStackSelection[0] = 0; + m_nMenuStackParameter[0] = 0; + } + else + { + m_pParentMenu = s_MenuRoot; + m_pCurrentMenu = s_MainMenu; + m_nCurrentMenuItem = 0; + m_nCurrentSelection = 0; + m_nCurrentParameter = 0; + m_nCurrentMenuDepth = 0; + } EventHandler (MenuEventUpdate); break; @@ -451,7 +471,30 @@ void CUIMenu::MenuHandler (CUIMenu *pUIMenu, TMenuEvent Event) break; case MenuEventStepDown: - if (pUIMenu->m_nCurrentSelection > 0) + if (pUIMenu->m_nCurrentSelection == 0) + { + // If in main mennu, wrap around + if (pUIMenu->m_pCurrentMenu == s_MainMenu) + { + // Find last entry with a name + while (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection+1].Name) + { + pUIMenu->m_nCurrentSelection++; + } + } + } + else if (pUIMenu->m_nCurrentSelection > 0) + { + pUIMenu->m_nCurrentSelection--; + } + // Might need to trim menu if number of TGs is configured to be less than the maximum supported + while ((pUIMenu->m_pCurrentMenu == s_MainMenu) && (pUIMenu->m_nCurrentSelection > 0) && + ( // Skip any unused menus + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].MenuItem == s_TGMenu) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter >= pUIMenu->m_nToneGenerators) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter < CConfig::AllToneGenerators) + ) + ) { pUIMenu->m_nCurrentSelection--; } @@ -461,7 +504,27 @@ void CUIMenu::MenuHandler (CUIMenu *pUIMenu, TMenuEvent Event) ++pUIMenu->m_nCurrentSelection; if (!pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Name) // more entries? { - pUIMenu->m_nCurrentSelection--; + if (pUIMenu->m_pCurrentMenu == s_MainMenu) + { + // If in main mennu, wrap around + pUIMenu->m_nCurrentSelection = 0; + } + else + { + // Return to last known good item + pUIMenu->m_nCurrentSelection--; + } + } + // Might need to trim menu if number of TGs is configured to be less than the maximum supported + while ((pUIMenu->m_pCurrentMenu == s_MainMenu) && (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection+1].Name) && + ( // Skip any unused TG menus + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].MenuItem == s_TGMenu) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter >= pUIMenu->m_nToneGenerators) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter < CConfig::AllToneGenerators) + ) + ) + { + pUIMenu->m_nCurrentSelection++; } break; @@ -1149,7 +1212,7 @@ void CUIMenu::TGShortcutHandler (TMenuEvent Event) assert (m_nCurrentMenuDepth >= 2); assert (m_MenuStackMenu[0] = s_MainMenu); unsigned nTG = m_nMenuStackSelection[0]; - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); assert (m_nMenuStackItem[1] == nTG); assert (m_nMenuStackParameter[1] == nTG); @@ -1164,7 +1227,7 @@ void CUIMenu::TGShortcutHandler (TMenuEvent Event) nTG++; } - if (nTG < CConfig::ToneGenerators) + if (nTG < m_nToneGenerators) { m_nMenuStackSelection[0] = nTG; m_nMenuStackItem[1] = nTG; @@ -1211,24 +1274,43 @@ void CUIMenu::PgmUpDownHandler (TMenuEvent Event) // Program Up/Down acts on performances unsigned nLastPerformance = m_pMiniDexed->GetLastPerformance(); unsigned nPerformance = m_pMiniDexed->GetActualPerformanceID(); + unsigned nStart = nPerformance; //LOGNOTE("Performance actual=%d, last=%d", nPerformance, nLastPerformance); if (Event == MenuEventPgmDown) { - if (nPerformance > 0) + do { - m_nSelectedPerformanceID = nPerformance-1; - m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); - //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); - } + if (nPerformance == 0) + { + // Wrap around + nPerformance = nLastPerformance; + } + else if (nPerformance > 0) + { + --nPerformance; + } + } while ((m_pMiniDexed->IsValidPerformance(nPerformance) != true) && (nPerformance != nStart)); + m_nSelectedPerformanceID = nPerformance; + m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); + //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); } - else + else // MenuEventPgmUp { - if (nPerformance < nLastPerformance-1) + do { - m_nSelectedPerformanceID = nPerformance+1; - m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); - //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); - } + if (nPerformance == nLastPerformance) + { + // Wrap around + nPerformance = 0; + } + else if (nPerformance < nLastPerformance) + { + ++nPerformance; + } + } while ((m_pMiniDexed->IsValidPerformance(nPerformance) != true) && (nPerformance != nStart)); + m_nSelectedPerformanceID = nPerformance; + m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); + //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); } } else @@ -1241,51 +1323,53 @@ void CUIMenu::PgmUpDownHandler (TMenuEvent Event) if (m_MenuStackMenu[0] == s_MainMenu && (m_pCurrentMenu == s_TGMenu) || (m_MenuStackMenu[1] == s_TGMenu)) { nTG = m_nMenuStackSelection[0]; } - assert (nTG < CConfig::ToneGenerators); - - int nPgm = m_pMiniDexed->GetTGParameter (CMiniDexed::TGParameterProgram, nTG); - - assert (Event == MenuEventPgmDown || Event == MenuEventPgmUp); - if (Event == MenuEventPgmDown) + assert (nTG < CConfig::AllToneGenerators); + if (nTG < m_nToneGenerators) { - //LOGNOTE("PgmDown"); - if (--nPgm < 0) + int nPgm = m_pMiniDexed->GetTGParameter (CMiniDexed::TGParameterProgram, nTG); + + assert (Event == MenuEventPgmDown || Event == MenuEventPgmUp); + if (Event == MenuEventPgmDown) { - // Switch down a voice bank and set to the last voice - nPgm = CSysExFileLoader::VoicesPerBank-1; - int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); - nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankDown(nVB); - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + //LOGNOTE("PgmDown"); + if (--nPgm < 0) + { + // Switch down a voice bank and set to the last voice + nPgm = CSysExFileLoader::VoicesPerBank-1; + int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); + nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankDown(nVB); + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + } + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); } - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); - } - else - { - //LOGNOTE("PgmUp"); - if (++nPgm > (int) CSysExFileLoader::VoicesPerBank-1) + else { - // Switch up a voice bank and reset to voice 0 - nPgm = 0; - int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); - nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankUp(nVB); - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + //LOGNOTE("PgmUp"); + if (++nPgm > (int) CSysExFileLoader::VoicesPerBank-1) + { + // Switch up a voice bank and reset to voice 0 + nPgm = 0; + int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); + nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankUp(nVB); + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + } + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); } - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); - } - // Skip empty voices. - // Use same criteria in EditProgramNumber () too. - string voiceName = m_pMiniDexed->GetVoiceName (nTG); - if (voiceName == "EMPTY " - || voiceName == " " - || voiceName == "----------" - || voiceName == "~~~~~~~~~~" ) - { - if (Event == MenuEventPgmUp) { - PgmUpDownHandler (MenuEventPgmUp); - } - if (Event == MenuEventPgmDown) { - PgmUpDownHandler (MenuEventPgmDown); + // Skip empty voices. + // Use same criteria in EditProgramNumber () too. + string voiceName = m_pMiniDexed->GetVoiceName (nTG); + if (voiceName == "EMPTY " + || voiceName == " " + || voiceName == "----------" + || voiceName == "~~~~~~~~~~" ) + { + if (Event == MenuEventPgmUp) { + PgmUpDownHandler (MenuEventPgmUp); + } + if (Event == MenuEventPgmDown) { + PgmUpDownHandler (MenuEventPgmDown); + } } } } @@ -1296,7 +1380,7 @@ void CUIMenu::TGUpDownHandler (TMenuEvent Event) // This will update the menus to position it for the next TG up or down unsigned nTG = 0; - if (CConfig::ToneGenerators <= 1) { + if (m_nToneGenerators <= 1) { // Nothing to do if only a single TG return; } @@ -1307,7 +1391,7 @@ void CUIMenu::TGUpDownHandler (TMenuEvent Event) nTG = m_nMenuStackSelection[0]; } - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); assert (Event == MenuEventTGDown || Event == MenuEventTGUp); if (Event == MenuEventTGDown) { @@ -1319,7 +1403,7 @@ void CUIMenu::TGUpDownHandler (TMenuEvent Event) else { //LOGNOTE("TGUp"); - if (nTG < CConfig::ToneGenerators - 1) { + if (nTG < m_nToneGenerators - 1) { nTG++; } } @@ -1366,7 +1450,15 @@ void CUIMenu::TimerHandlerNoBack (TKernelTimerHandle hTimer, void *pParam, void void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) { bool bPerformanceSelectToLoad = pUIMenu->m_pMiniDexed->GetPerformanceSelectToLoad(); + unsigned nLastPerformance = pUIMenu->m_pMiniDexed->GetLastPerformance(); unsigned nValue = pUIMenu->m_nSelectedPerformanceID; + unsigned nStart = nValue; + if (pUIMenu->m_pMiniDexed->IsValidPerformance(nValue) != true) + { + // A bank change has left the selected performance out of sync + nValue = pUIMenu->m_pMiniDexed->GetActualPerformanceID(); + pUIMenu->m_nSelectedPerformanceID = nValue; + } std::string Value; if (Event == MenuEventUpdate) @@ -1380,17 +1472,25 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) } if(!pUIMenu->m_bPerformanceDeleteMode) - { + { switch (Event) { case MenuEventUpdate: break; case MenuEventStepDown: - if (nValue > 0) + do { - --nValue; - } + if (nValue == 0) + { + // Wrap around + nValue = nLastPerformance; + } + else if (nValue > 0) + { + --nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformance(nValue) != true) && (nValue != nStart)); pUIMenu->m_nSelectedPerformanceID = nValue; if (!bPerformanceSelectToLoad && pUIMenu->m_nCurrentParameter==0) { @@ -1399,10 +1499,18 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) break; case MenuEventStepUp: - if (++nValue > (unsigned) pUIMenu->m_pMiniDexed->GetLastPerformance()-1) + do { - nValue = pUIMenu->m_pMiniDexed->GetLastPerformance()-1; - } + if (nValue == nLastPerformance) + { + // Wrap around + nValue = 0; + } + else if (nValue < nLastPerformance) + { + ++nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformance(nValue) != true) && (nValue != nStart)); pUIMenu->m_nSelectedPerformanceID = nValue; if (!bPerformanceSelectToLoad && pUIMenu->m_nCurrentParameter==0) { @@ -1421,7 +1529,7 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) break; case 1: - if (pUIMenu->m_nSelectedPerformanceID != 0) + if (pUIMenu->m_pMiniDexed->IsValidPerformance(pUIMenu->m_nSelectedPerformanceID)) { pUIMenu->m_bPerformanceDeleteMode=true; pUIMenu->m_bConfirmDeletePerformance=false; @@ -1474,17 +1582,24 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) if(!pUIMenu->m_bPerformanceDeleteMode) { Value = pUIMenu->m_pMiniDexed->GetPerformanceName(nValue); + unsigned nBankNum = pUIMenu->m_pMiniDexed->GetPerformanceBank(); - - std::string nPSelected = ""; + std::string nPSelected = "000"; + nPSelected += std::to_string(nBankNum+1); // Convert to user-facing bank number rather than index + nPSelected = nPSelected.substr(nPSelected.length()-3,3); + std::string nPPerf = "000"; + nPPerf += std::to_string(nValue+1); // Convert to user-facing performance number rather than index + nPPerf = nPPerf.substr(nPPerf.length()-3,3); + + nPSelected += ":"+nPPerf; if(nValue == pUIMenu->m_pMiniDexed->GetActualPerformanceID()) { - nPSelected= "[L]"; + nPSelected += " [L]"; } pUIMenu->m_pUI->DisplayWrite (pUIMenu->m_pParentMenu[pUIMenu->m_nCurrentMenuItem].Name, nPSelected.c_str(), - Value.c_str (), - (int) nValue > 0, (int) nValue < (int) pUIMenu->m_pMiniDexed->GetLastPerformance()-1); + Value.c_str (), true, true); +// (int) nValue > 0, (int) nValue < (int) pUIMenu->m_pMiniDexed->GetLastPerformance()); } else { @@ -1492,6 +1607,90 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) } } +void CUIMenu::EditPerformanceBankNumber (CUIMenu *pUIMenu, TMenuEvent Event) +{ + bool bPerformanceSelectToLoad = pUIMenu->m_pMiniDexed->GetPerformanceSelectToLoad(); + unsigned nLastPerformanceBank = pUIMenu->m_pMiniDexed->GetLastPerformanceBank(); + unsigned nValue = pUIMenu->m_nSelectedPerformanceBankID; + unsigned nStart = nValue; + std::string Value; + + switch (Event) + { + case MenuEventUpdate: + break; + + case MenuEventStepDown: + do + { + if (nValue == 0) + { + // Wrap around + nValue = nLastPerformanceBank; + } + else if (nValue > 0) + { + --nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformanceBank(nValue) != true) && (nValue != nStart)); + pUIMenu->m_nSelectedPerformanceBankID = nValue; + if (!bPerformanceSelectToLoad) + { + // Switch to the new bank and select the first performance voice + pUIMenu->m_pMiniDexed->SetParameter (CMiniDexed::ParameterPerformanceBank, nValue); + pUIMenu->m_pMiniDexed->SetFirstPerformance(); + } + break; + + case MenuEventStepUp: + do + { + if (nValue == nLastPerformanceBank) + { + // Wrap around + nValue = 0; + } + else if (nValue < nLastPerformanceBank) + { + ++nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformanceBank(nValue) != true) && (nValue != nStart)); + pUIMenu->m_nSelectedPerformanceBankID = nValue; + if (!bPerformanceSelectToLoad) + { + pUIMenu->m_pMiniDexed->SetParameter (CMiniDexed::ParameterPerformanceBank, nValue); + pUIMenu->m_pMiniDexed->SetFirstPerformance(); + } + break; + + case MenuEventSelect: + if (bPerformanceSelectToLoad) + { + pUIMenu->m_pMiniDexed->SetParameter (CMiniDexed::ParameterPerformanceBank, nValue); + pUIMenu->m_pMiniDexed->SetFirstPerformance(); + } + break; + + default: + return; + } + + Value = pUIMenu->m_pMiniDexed->GetPerformanceConfig ()->GetPerformanceBankName(nValue); + std::string nPSelected = "000"; + nPSelected += std::to_string(nValue+1); // Convert to user-facing number rather than index + nPSelected = nPSelected.substr(nPSelected.length()-3,3); + + if(nValue == (unsigned)pUIMenu->m_pMiniDexed->GetParameter (CMiniDexed::ParameterPerformanceBank)) + { + nPSelected += " [L]"; + } + + pUIMenu->m_pUI->DisplayWrite (pUIMenu->m_pParentMenu[pUIMenu->m_nCurrentMenuItem].Name, nPSelected.c_str(), + Value.c_str (), + nValue > 0, + nValue < pUIMenu->m_pMiniDexed->GetLastPerformanceBank()-1); +} + void CUIMenu::InputTxt (CUIMenu *pUIMenu, TMenuEvent Event) { unsigned nTG=0; diff --git a/src/uimenu.h b/src/uimenu.h index b66d65c..0034278 100644 --- a/src/uimenu.h +++ b/src/uimenu.h @@ -25,6 +25,7 @@ #include #include +#include "config.h" class CMiniDexed; class CUserInterface; @@ -53,7 +54,7 @@ public: }; public: - CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed); + CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed, CConfig *pConfig); void EventHandler (TMenuEvent Event); @@ -91,6 +92,7 @@ private: static void EditTGParameterModulation (CUIMenu *pUIMenu, TMenuEvent Event); static void PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event); static void SavePerformanceNewFile (CUIMenu *pUIMenu, TMenuEvent Event); + static void EditPerformanceBankNumber (CUIMenu *pUIMenu, TMenuEvent Event); static std::string GetGlobalValueString (unsigned nParameter, int nValue); static std::string GetTGValueString (unsigned nTGParameter, int nValue); @@ -127,6 +129,9 @@ private: private: CUserInterface *m_pUI; CMiniDexed *m_pMiniDexed; + CConfig *m_pConfig; + + unsigned m_nToneGenerators; const TMenuItem *m_pParentMenu; const TMenuItem *m_pCurrentMenu; @@ -169,6 +174,7 @@ private: bool m_bPerformanceDeleteMode=false; bool m_bConfirmDeletePerformance=false; unsigned m_nSelectedPerformanceID =0; + unsigned m_nSelectedPerformanceBankID =0; bool m_bSplashShow=false; }; diff --git a/src/usbminidexedmidigadget.h b/src/usbminidexedmidigadget.h index bbb5f44..8513549 100644 --- a/src/usbminidexedmidigadget.h +++ b/src/usbminidexedmidigadget.h @@ -22,6 +22,23 @@ #ifndef _usbminidexedmidigadget_h #define _usbminidexedmidigadget_h +#if RASPPI==5 +#include +#include + +#warning No support for USB Gadget Mode on RPI 5 yet +class CUSBMiniDexedMIDIGadget +{ +public: + CUSBMiniDexedMIDIGadget (CInterruptSystem *pInterruptSystem) + { + } + + ~CUSBMiniDexedMIDIGadget (void) + { + } +}; +#else #include #include #include @@ -82,5 +99,6 @@ protected: return CUSBMIDIGadget::GetDescriptor(wValue, wIndex, pLength); } }; +#endif #endif diff --git a/src/userinterface.cpp b/src/userinterface.cpp index 1c0716e..aa46f9e 100644 --- a/src/userinterface.cpp +++ b/src/userinterface.cpp @@ -27,17 +27,18 @@ LOGMODULE ("ui"); -CUserInterface::CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CConfig *pConfig) +CUserInterface::CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, CConfig *pConfig) : m_pMiniDexed (pMiniDexed), m_pGPIOManager (pGPIOManager), m_pI2CMaster (pI2CMaster), + m_pSPIMaster (pSPIMaster), m_pConfig (pConfig), m_pLCD (0), m_pLCDBuffered (0), m_pUIButtons (0), m_pRotaryEncoder (0), m_bSwitchPressed (false), - m_Menu (this, pMiniDexed) + m_Menu (this, pMiniDexed, pConfig) { } @@ -57,17 +58,69 @@ bool CUserInterface::Initialize (void) { unsigned i2caddr = m_pConfig->GetLCDI2CAddress (); unsigned ssd1306addr = m_pConfig->GetSSD1306LCDI2CAddress (); + bool st7789 = m_pConfig->GetST7789Enabled (); if (ssd1306addr != 0) { m_pSSD1306 = new CSSD1306Device (m_pConfig->GetSSD1306LCDWidth (), m_pConfig->GetSSD1306LCDHeight (), m_pI2CMaster, ssd1306addr, m_pConfig->GetSSD1306LCDRotate (), m_pConfig->GetSSD1306LCDMirror ()); - LOGDBG ("LCD: SSD1306"); if (!m_pSSD1306->Initialize ()) { + LOGDBG("LCD: SSD1306 initialization failed"); return false; } + LOGDBG ("LCD: SSD1306"); m_pLCD = m_pSSD1306; - } else if (i2caddr == 0) + } + else if (st7789) + { + if (m_pSPIMaster == nullptr) + { + LOGDBG("LCD: ST7789 Enabled but SPI Initialisation Failed"); + return false; + } + + unsigned long nSPIClock = 1000 * m_pConfig->GetSPIClockKHz(); + unsigned nSPIMode = m_pConfig->GetSPIMode(); + unsigned nCPHA = (nSPIMode & 1) ? 1 : 0; + unsigned nCPOL = (nSPIMode & 2) ? 1 : 0; + LOGDBG("SPI: CPOL=%u; CPHA=%u; CLK=%u",nCPOL,nCPHA,nSPIClock); + m_pST7789Display = new CST7789Display (m_pSPIMaster, + m_pConfig->GetST7789Data(), + m_pConfig->GetST7789Reset(), + m_pConfig->GetST7789Backlight(), + m_pConfig->GetST7789Width(), + m_pConfig->GetST7789Height(), + nCPOL, nCPHA, nSPIClock, + m_pConfig->GetST7789Select()); + if (m_pST7789Display->Initialize()) + { + m_pST7789Display->SetRotation (m_pConfig->GetST7789Rotation()); + bool bLargeFont = !(m_pConfig->GetST7789SmallFont()); + m_pST7789 = new CST7789Device (m_pSPIMaster, m_pST7789Display, m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows (), bLargeFont, bLargeFont); + if (m_pST7789->Initialize()) + { + LOGDBG ("LCD: ST7789"); + m_pLCD = m_pST7789; + } + else + { + LOGDBG ("LCD: Failed to initalize ST7789 character device"); + delete (m_pST7789); + delete (m_pST7789Display); + m_pST7789 = nullptr; + m_pST7789Display = nullptr; + return false; + } + } + else + { + LOGDBG ("LCD: Failed to initialize ST7789 display"); + delete (m_pST7789Display); + m_pST7789Display = nullptr; + return false; + } + } + else if (i2caddr == 0) { m_pHD44780 = new CHD44780Device (m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows (), m_pConfig->GetLCDPinData4 (), @@ -77,22 +130,24 @@ bool CUserInterface::Initialize (void) m_pConfig->GetLCDPinEnable (), m_pConfig->GetLCDPinRegisterSelect (), m_pConfig->GetLCDPinReadWrite ()); - LOGDBG ("LCD: HD44780"); if (!m_pHD44780->Initialize ()) { + LOGDBG("LCD: HD44780 initialization failed"); return false; } + LOGDBG ("LCD: HD44780"); m_pLCD = m_pHD44780; } else { m_pHD44780 = new CHD44780Device (m_pI2CMaster, i2caddr, m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows ()); - LOGDBG ("LCD: HD44780 I2C"); if (!m_pHD44780->Initialize ()) { + LOGDBG("LCD: HD44780 (I2C) initialization failed"); return false; } + LOGDBG ("LCD: HD44780 I2C"); m_pLCD = m_pHD44780; } assert (m_pLCD); @@ -217,7 +272,7 @@ void CUserInterface::DisplayWrite (const char *pMenu, const char *pParam, const CString Value (" "); if (bArrowDown) { - Value = "\x7F"; // arrow left character + Value = "<"; // arrow left character } Value.Append (pValue); @@ -232,7 +287,7 @@ void CUserInterface::DisplayWrite (const char *pMenu, const char *pParam, const } } - Value.Append ("\x7E"); // arrow right character + Value.Append (">"); // arrow right character } Msg.Append (Value); @@ -388,13 +443,16 @@ void CUserInterface::UISetMIDIButtonChannel (unsigned uCh) if (uCh == 0) { m_nMIDIButtonCh = CMIDIDevice::Disabled; + LOGNOTE("MIDI Button channel not set"); } - else if (uCh < CMIDIDevice::Channels) + else if (uCh <= CMIDIDevice::Channels) { m_nMIDIButtonCh = uCh - 1; + LOGNOTE("MIDI Button channel set to: %d", m_nMIDIButtonCh+1); } else { m_nMIDIButtonCh = CMIDIDevice::OmniMode; + LOGNOTE("MIDI Button channel set to: OMNI"); } } \ No newline at end of file diff --git a/src/userinterface.h b/src/userinterface.h index 5de2846..a8026db 100644 --- a/src/userinterface.h +++ b/src/userinterface.h @@ -26,16 +26,18 @@ #include #include #include +#include #include #include #include +#include class CMiniDexed; class CUserInterface { public: - CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CConfig *pConfig); + CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, CConfig *pConfig); ~CUserInterface (void); bool Initialize (void); @@ -68,11 +70,14 @@ private: CMiniDexed *m_pMiniDexed; CGPIOManager *m_pGPIOManager; CI2CMaster *m_pI2CMaster; + CSPIMaster *m_pSPIMaster; CConfig *m_pConfig; CCharDevice *m_pLCD; CHD44780Device *m_pHD44780; CSSD1306Device *m_pSSD1306; + CST7789Display *m_pST7789Display; + CST7789Device *m_pST7789; CWriteBufferDevice *m_pLCDBuffered; CUIButtons *m_pUIButtons; diff --git a/submod.sh b/submod.sh index 559aeb0..78531dc 100755 --- a/submod.sh +++ b/submod.sh @@ -6,13 +6,13 @@ git submodule update --init --recursive # # Use fixed master branch of circle-stdlib then re-update cd circle-stdlib/ -git checkout 695ab4a +git checkout 3bd135d git submodule update --init --recursive cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout tags/Step48 +git checkout c243194 cd - cd circle-stdlib/libs/circle-newlib #git checkout develop