Skip to content

Allow sending larger SysEx messages #904

@probonopd

Description

@probonopd

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:

//
// 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;
}

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions