#include #include #include #include #include "test.h" /* * Checks whether the regular expression matches the entire MIDI data, printed * as hex. */ static int midi_matches_regex(unsigned char *midi, int count, const char *regex) { char *text; regex_t re; regmatch_t match; int i; text = malloc(2 * count + 1); if (!text) return 0; for (i = 0; i < count; ++i) sprintf(text + 2 * i, "%02x", midi[i]); if (regcomp(&re, regex, REG_EXTENDED) != 0) { free(text); return 0; } i = regexec(&re, text, 1, &match, 0); i = i == 0 && match.rm_so == 0 && match.rm_eo == strlen(text); regfree(&re); free(text); return i; } static void test_decode(void) { snd_midi_event_t *midi_event; snd_seq_event_t ev; unsigned char buf[50]; int count; if (ALSA_CHECK(snd_midi_event_new(256 /* ? */, &midi_event)) < 0) return; #define DECODE() snd_midi_event_decode(midi_event, buf, sizeof(buf), &ev) #define BUF_MATCHES(str) midi_matches_regex(buf, count, str) #define DECODES_TO(str) ((count = DECODE()), BUF_MATCHES(str)) snd_seq_ev_clear(&ev); snd_seq_ev_set_fixed(&ev); ev.type = SND_SEQ_EVENT_NONE; TEST_CHECK(DECODE() == -ENOENT); snd_seq_ev_set_noteoff(&ev, 1, 2, 3); TEST_CHECK(DECODES_TO("810203")); snd_seq_ev_set_noteon(&ev, 4, 5, 6); TEST_CHECK(DECODES_TO("940506")); snd_seq_ev_set_keypress(&ev, 7, 8, 9); TEST_CHECK(DECODES_TO("a70809")); snd_seq_ev_set_controller(&ev, 10, 11, 12); TEST_CHECK(DECODES_TO("ba0b0c")); snd_seq_ev_set_pgmchange(&ev, 13, 14); TEST_CHECK(DECODES_TO("cd0e")); snd_seq_ev_set_chanpress(&ev, 15, 16); TEST_CHECK(DECODES_TO("df10")); snd_seq_ev_set_pitchbend(&ev, 1, 0x222); TEST_CHECK(DECODES_TO("e12244")); snd_seq_ev_set_sysex(&ev, 6, "\xf0\x7e\x7f\x06\x01\xf7"); TEST_CHECK(DECODES_TO("f07e7f0601f7")); snd_seq_ev_set_fixed(&ev); ev.type = SND_SEQ_EVENT_QFRAME; ev.data.control.value = 3; TEST_CHECK(DECODES_TO("f103")); ev.type = SND_SEQ_EVENT_SONGPOS; ev.data.control.value = 0x444; TEST_CHECK(DECODES_TO("f24408")); ev.type = SND_SEQ_EVENT_SONGSEL; ev.data.control.value = 5; TEST_CHECK(DECODES_TO("f305")); ev.type = SND_SEQ_EVENT_TUNE_REQUEST; TEST_CHECK(DECODES_TO("f6")); ev.type = SND_SEQ_EVENT_CLOCK; TEST_CHECK(DECODES_TO("f8")); ev.type = SND_SEQ_EVENT_START; TEST_CHECK(DECODES_TO("fa")); ev.type = SND_SEQ_EVENT_CONTINUE; TEST_CHECK(DECODES_TO("fb")); ev.type = SND_SEQ_EVENT_STOP; TEST_CHECK(DECODES_TO("fc")); ev.type = SND_SEQ_EVENT_SENSING; TEST_CHECK(DECODES_TO("fe")); ev.type = SND_SEQ_EVENT_RESET; TEST_CHECK(DECODES_TO("ff")); ev.type = SND_SEQ_EVENT_CONTROL14; ev.data.control.channel = 6; ev.data.control.param = 7; ev.data.control.value = 0x888; /* * This regular expression catches all allowed combinations of LSB/MSB * order and running status. */ TEST_CHECK(DECODES_TO("b6(0711(b6)?2708|2708(b6)?0711)")); ev.type = SND_SEQ_EVENT_NONREGPARAM; ev.data.control.channel = 9; ev.data.control.param = 0xaaa; ev.data.control.value = 0xbbb; TEST_CHECK(DECODES_TO("b9(622a(b9)?6315|6315(b9)?622a)(b9)?(0617(b9)?263b|263b(b9)?0617)")); ev.type = SND_SEQ_EVENT_REGPARAM; ev.data.control.channel = 12; ev.data.control.param = 0xddd; ev.data.control.value = 0xeee; TEST_CHECK(DECODES_TO("bc(645d(bc)?651b|651b(bc)?645d)(bc)?(061d(bc)?266e|266e(bc)?061d)")); /* no running status after SysEx */ snd_seq_ev_set_pgmchange(&ev, 0, 0x11); TEST_CHECK(DECODES_TO("c011")); snd_seq_ev_set_sysex(&ev, 6, "\xf0\x7e\x7f\x09\x02\xf7"); TEST_CHECK(DECODES_TO("f07e7f0902f7")); snd_seq_ev_set_pgmchange(&ev, 0, 0x11); TEST_CHECK(DECODES_TO("c011")); /* no running status for non-realtime common messages */ ev.type = SND_SEQ_EVENT_QFRAME; ev.data.control.value = 0x11; TEST_CHECK(DECODES_TO("f111")); TEST_CHECK(DECODES_TO("f111")); /* buffer overflow */ TEST_CHECK(snd_midi_event_decode(midi_event, buf, 1, &ev) == -ENOMEM); snd_midi_event_free(midi_event); } static void test_reset_decode(void) { snd_midi_event_t *midi_event; snd_seq_event_t ev; unsigned char buf[50]; int count; if (ALSA_CHECK(snd_midi_event_new(256 /* ? */, &midi_event)) < 0) return; snd_seq_ev_clear(&ev); snd_seq_ev_set_noteon(&ev, 1, 2, 3); TEST_CHECK(DECODES_TO("910203")); snd_midi_event_reset_decode(midi_event); TEST_CHECK(DECODES_TO("910203")); snd_midi_event_free(midi_event); } static void test_encode(void) { snd_midi_event_t *midi_event; snd_seq_event_t ev; if (ALSA_CHECK(snd_midi_event_new(256, &midi_event)) < 0) return; #define ENCODE(str) snd_midi_event_encode(midi_event, \ (const unsigned char *)str, \ sizeof(str) - 1, &ev) TEST_CHECK(ENCODE("\x81\x02\x03") == 3); TEST_CHECK(ev.type == SND_SEQ_EVENT_NOTEOFF); TEST_CHECK((ev.flags & SND_SEQ_EVENT_LENGTH_MASK) == SND_SEQ_EVENT_LENGTH_FIXED); TEST_CHECK(ev.data.note.channel == 1); TEST_CHECK(ev.data.note.note == 2); TEST_CHECK(ev.data.note.velocity == 3); TEST_CHECK(ENCODE("\x94\x05\x06") == 3); TEST_CHECK(ev.type == SND_SEQ_EVENT_NOTEON); TEST_CHECK(ev.data.note.channel == 4); TEST_CHECK(ev.data.note.note == 5); TEST_CHECK(ev.data.note.velocity == 6); TEST_CHECK(ENCODE("\xa7\x08\x09") == 3); TEST_CHECK(ev.type == SND_SEQ_EVENT_KEYPRESS); TEST_CHECK(ev.data.note.channel == 7); TEST_CHECK(ev.data.note.note == 8); TEST_CHECK(ev.data.note.velocity == 9); TEST_CHECK(ENCODE("\xba\x0b\x0c") == 3); TEST_CHECK(ev.type == SND_SEQ_EVENT_CONTROLLER); TEST_CHECK(ev.data.control.channel == 10); TEST_CHECK(ev.data.control.param == 11); TEST_CHECK(ev.data.control.value == 12); TEST_CHECK(ENCODE("\xcd\x0e") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_PGMCHANGE); TEST_CHECK(ev.data.control.channel == 13); TEST_CHECK(ev.data.control.value == 14); TEST_CHECK(ENCODE("\xdf\x10") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_CHANPRESS); TEST_CHECK(ev.data.control.channel == 15); TEST_CHECK(ev.data.control.value == 16); TEST_CHECK(ENCODE("\xe1\x22\x33") == 3); TEST_CHECK(ev.type == SND_SEQ_EVENT_PITCHBEND); TEST_CHECK(ev.data.control.channel == 1); TEST_CHECK(ev.data.control.value == -1630); TEST_CHECK(ENCODE("\xf0\x7f\x7f\x04\x01\x7f\x7f\xf7") == 8); TEST_CHECK(ev.type == SND_SEQ_EVENT_SYSEX); TEST_CHECK((ev.flags & SND_SEQ_EVENT_LENGTH_MASK) == SND_SEQ_EVENT_LENGTH_VARIABLE); TEST_CHECK(ev.data.ext.len == 8); TEST_CHECK(!memcmp(ev.data.ext.ptr, "\xf0\x7f\x7f\x04\x01\x7f\x7f\xf7", 8)); TEST_CHECK(ENCODE("\xf1\x04") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_QFRAME); TEST_CHECK(ev.data.control.value == 4); TEST_CHECK(ENCODE("\xf2\x55\x66") == 3); TEST_CHECK(ev.type == SND_SEQ_EVENT_SONGPOS); TEST_CHECK(ev.data.control.value == 13141); TEST_CHECK(ENCODE("\xf3\x07") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_SONGSEL); TEST_CHECK(ev.data.control.value == 7); TEST_CHECK(ENCODE("\xf6") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_TUNE_REQUEST); TEST_CHECK(ENCODE("\xf8") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_CLOCK); TEST_CHECK(ENCODE("\xfa") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_START); TEST_CHECK(ENCODE("\xfb") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_CONTINUE); TEST_CHECK(ENCODE("\xfc") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_STOP); TEST_CHECK(ENCODE("\xfe") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_SENSING); TEST_CHECK(ENCODE("\xff") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_RESET); TEST_CHECK(ENCODE("\xc1\xf8") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_CLOCK); TEST_CHECK(ENCODE("\x22") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_PGMCHANGE); TEST_CHECK(ev.data.control.channel == 1); TEST_CHECK(ev.data.control.value == 0x22); TEST_CHECK(ENCODE("\xf8") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_CLOCK); TEST_CHECK(ENCODE("\x33") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_PGMCHANGE); TEST_CHECK(ev.data.control.channel == 1); TEST_CHECK(ev.data.control.value == 0x33); TEST_CHECK(ENCODE("\xc1\xf6") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_TUNE_REQUEST); TEST_CHECK(ENCODE("\x44\x44") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_NONE); snd_midi_event_free(midi_event); } static void test_reset_encode(void) { snd_midi_event_t *midi_event; snd_seq_event_t ev; if (ALSA_CHECK(snd_midi_event_new(256, &midi_event)) < 0) return; TEST_CHECK(ENCODE("\x91\x02") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_NONE); snd_midi_event_reset_encode(midi_event); TEST_CHECK(ENCODE("\x03") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_NONE); snd_midi_event_free(midi_event); } static void test_init(void) { snd_midi_event_t *midi_event; snd_seq_event_t ev; unsigned char buf[50]; int count; if (ALSA_CHECK(snd_midi_event_new(256, &midi_event)) < 0) return; snd_seq_ev_set_noteon(&ev, 1, 2, 3); TEST_CHECK(DECODES_TO("910203")); TEST_CHECK(ENCODE("\x94\x05") == 2); TEST_CHECK(ev.type == SND_SEQ_EVENT_NONE); snd_midi_event_init(midi_event); snd_seq_ev_set_noteon(&ev, 1, 2, 3); TEST_CHECK(DECODES_TO("910203")); TEST_CHECK(ENCODE("\x06") == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_NONE); snd_midi_event_free(midi_event); } static void test_encode_byte(void) { snd_midi_event_t *midi_event; snd_seq_event_t ev; if (ALSA_CHECK(snd_midi_event_new(256, &midi_event)) < 0) return; #define ENCODE_BYTE(c) snd_midi_event_encode_byte(midi_event, c, &ev) TEST_CHECK(ENCODE_BYTE(0x81) == 0); TEST_CHECK(ENCODE_BYTE(0x02) == 0); TEST_CHECK(ENCODE_BYTE(0x03) == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_NOTEOFF); TEST_CHECK((ev.flags & SND_SEQ_EVENT_LENGTH_MASK) == SND_SEQ_EVENT_LENGTH_FIXED); TEST_CHECK(ev.data.note.channel == 1); TEST_CHECK(ev.data.note.note == 2); TEST_CHECK(ev.data.note.velocity == 3); TEST_CHECK(ENCODE_BYTE(0x04) == 0); TEST_CHECK(ENCODE_BYTE(0xf8) == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_CLOCK); TEST_CHECK(ENCODE_BYTE(0x05) == 1); TEST_CHECK(ev.type == SND_SEQ_EVENT_NOTEOFF); TEST_CHECK(ev.data.note.channel == 1); TEST_CHECK(ev.data.note.note == 4); TEST_CHECK(ev.data.note.velocity == 5); snd_midi_event_free(midi_event); } int main(void) { test_decode(); test_reset_decode(); test_encode(); test_reset_encode(); test_encode_byte(); test_init(); return TEST_EXIT_CODE(); }