MiniDexed
MiniDexed copied to clipboard
Allow sending larger SysEx messages
Cartridge dumps are larger than we can send out over MIDI. Currently,
- With 512 bytes of data, it WORKS both over USB MIDI and rtpMIDI
- With 514 bytes, it WORKS over rtpMIDI but NOT over USB MIDI
- With 1024 bytes, it WORKS over rtpMIDI but NOT over USB MIDI
It seems like we need to break down ("chunk") larger SysEx messages into parts.
However, it appears that Circle refuses to send "incomplete" SysEx messages.
Reference:
- https://github.com/rsta2/circle/issues/563
//
// midichunker.h
//
// MiniDexed - Dexed FM synthesizer for bare metal Raspberry Pi
// Copyright (C) 2022-25 The MiniDexed Team
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
#pragma once
#include <cstddef>
#include <vector>
#include <cstdint>
class MIDISysExChunker {
public:
MIDISysExChunker(const uint8_t* data, size_t length, size_t chunkSize = 256);
bool hasNext() const;
std::vector<uint8_t> next();
void reset();
private:
const uint8_t* m_data;
size_t m_length;
size_t m_chunkSize;
size_t m_offset;
};
//
// midichunker.cpp
//
// MiniDexed - Dexed FM synthesizer for bare metal Raspberry Pi
// Copyright (C) 2022-25 The MiniDexed Team
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
#include "midichunker.h"
#include <algorithm>
#include <circle/logger.h>
static const char* From = "midichunker";
MIDISysExChunker::MIDISysExChunker(const uint8_t* data, size_t length, size_t chunkSize)
: m_data(data), m_length(length), m_chunkSize(chunkSize), m_offset(0) {}
bool MIDISysExChunker::hasNext() const {
return m_offset < m_length;
}
std::vector<uint8_t> MIDISysExChunker::next() {
if (!hasNext()) return {};
size_t remaining = m_length - m_offset;
size_t chunkLen = std::min(m_chunkSize, remaining);
// Only the last chunk should contain the final 0xF7
if (m_offset + chunkLen >= m_length && m_data[m_length-1] == 0xF7) {
chunkLen = m_length - m_offset;
} else if (m_offset + chunkLen > 0 && m_data[m_offset + chunkLen - 1] == 0xF7) {
chunkLen--;
}
LOGNOTE("Chunker: m_offset=%d, chunkLen=%d, remaining=%d, m_length=%d, m_chunkSize=%d", (int)m_offset, (int)chunkLen, (int)remaining, (int)m_length, (int)m_chunkSize);
std::vector<uint8_t> chunk(m_data + m_offset, m_data + m_offset + chunkLen);
m_offset += chunkLen;
return chunk;
}
void MIDISysExChunker::reset() {
m_offset = 0;
}