This adds both a midi generator and a midi type 1 simulator. While I would have preferred to use an existing tool for this, I found that timidity++ does not emit pitch wheel events correctly, and I don't know of another widely-distributed tool that does midi-to-midi format conversion. The c++ and python versions were co-developed. I wrote one to test the other. There is more cleanup to do, but `roundtrip.cpp` produces a valid type 0 midi file given a type 1 or type 0 midi file as input.
148 lines
3.4 KiB
C++
148 lines
3.4 KiB
C++
#include <iostream>
|
|
#include <fstream>
|
|
#include <tuple>
|
|
#include <cassert>
|
|
#include <tuple>
|
|
|
|
#include "parse.hpp"
|
|
#include "strings.hpp"
|
|
|
|
static uint16_t _cent_to_fns[] = {
|
|
0x0,
|
|
0x3d,
|
|
0x7d,
|
|
0xc2,
|
|
0x10a,
|
|
0x157,
|
|
0x1a8,
|
|
0x1fe,
|
|
0x25a,
|
|
0x2ba,
|
|
0x321,
|
|
0x38d,
|
|
};
|
|
|
|
constexpr std::tuple<uint16_t, uint16_t>
|
|
midi_note_to_oct_fns(const int32_t midi_note)
|
|
{
|
|
int32_t a440_note = midi_note - 69;
|
|
uint16_t oct = (a440_note / 12) & 0xf;
|
|
uint32_t cent = static_cast<uint32_t>(a440_note) % 12;
|
|
uint16_t fns = _cent_to_fns[cent];
|
|
return {oct, fns};
|
|
}
|
|
|
|
void
|
|
dump_midi(midi::midi_event_t& midi_event)
|
|
{
|
|
switch (midi_event.type) {
|
|
case midi::midi_event_t::type_t::note_on:
|
|
{
|
|
std::cout << " note_on " << (int)midi_event.data.note_on.note << '\n';
|
|
auto&& [oct, fns] = midi_note_to_oct_fns(midi_event.data.note_on.note);
|
|
std::cout << " oct " << oct << " fns " << fns << '\n';
|
|
}
|
|
break;
|
|
case midi::midi_event_t::type_t::note_off:
|
|
{
|
|
std::cout << " note_off " << (int)midi_event.data.note_on.note << '\n';
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
int parse(uint8_t const * start)
|
|
{
|
|
uint8_t const * buf = &start[0];
|
|
auto header_o = midi::parse::header(buf);
|
|
if (!header_o) {
|
|
std::cerr << "invalid header\n";
|
|
return -1;
|
|
}
|
|
midi::header_t header;
|
|
std::tie(buf, header) = *header_o;
|
|
|
|
std::cout << "header.format: " << midi::strings::header_format(header.format) << '\n';
|
|
std::cout << "header.ntrks: " << header.ntrks << '\n';
|
|
|
|
for (int32_t i = 0; i < header.ntrks; i++) {
|
|
std::cout << "track[" << i << "]:\n";
|
|
|
|
auto track_o = midi::parse::track(buf);
|
|
if (!track_o) {
|
|
std::cerr << "invalid track\n";
|
|
return -1;
|
|
}
|
|
|
|
uint32_t track_length;
|
|
std::tie(buf, track_length) = *track_o;
|
|
|
|
std::cout << " track_length: " << track_length << '\n';
|
|
|
|
uint8_t const * track_start = buf;
|
|
while (buf - track_start < track_length) {
|
|
std::cout << " event:\n";
|
|
auto mtrk_event_o = midi::parse::mtrk_event(buf);
|
|
if (!mtrk_event_o) {
|
|
std::cout << " invalid mtrk_event\n";
|
|
std::cout << std::hex << buf[0] << ' ' << buf[1] << ' ' << buf[2] << ' ' << buf[3];
|
|
return -1;
|
|
}
|
|
|
|
midi::mtrk_event_t mtrk_event;
|
|
std::tie(buf, mtrk_event) = *mtrk_event_o;
|
|
std::cout << " delta_time: " << mtrk_event.delta_time << '\n';
|
|
switch (mtrk_event.event.type) {
|
|
case midi::event_t::type_t::midi:
|
|
std::cout << " midi: " << '\n';
|
|
dump_midi(mtrk_event.event.event.midi);
|
|
break;
|
|
case midi::event_t::type_t::sysex:
|
|
std::cout << " sysex: " << '\n';
|
|
break;
|
|
case midi::event_t::type_t::meta:
|
|
std::cout << " meta: " << '\n';
|
|
break;
|
|
default:
|
|
assert(false);
|
|
}
|
|
}
|
|
|
|
assert(buf - track_start == track_length);
|
|
std::cout << "trailing/unparsed data: " << track_length - (buf - track_start) << '\n';
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
if (argc < 2) {
|
|
std::cerr << "argc < 2\n";
|
|
return -1;
|
|
}
|
|
|
|
std::cerr << argv[1] << '\n';
|
|
|
|
std::ifstream ifs;
|
|
ifs.open(argv[1], std::ios::in | std::ios::binary | std::ios::trunc);
|
|
if (!ifs.is_open()) {
|
|
std::cerr << "ifstream\n";
|
|
return -1;
|
|
}
|
|
|
|
auto size = static_cast<int32_t>(ifs.tellg());
|
|
uint8_t start[size];
|
|
ifs.seekg(0);
|
|
if (!ifs.read(reinterpret_cast<char*>(&start[0]), size)) {
|
|
std::cerr << "read\n";
|
|
return -1;
|
|
}
|
|
ifs.close();
|
|
|
|
parse(start);
|
|
|
|
return 0;
|
|
}
|