Декодировать поле переменной длины MIDI
Как декодировать поля переменной длины в MIDI? Описание, приведенное в [1], приводит только некоторые примеры без упоминания того, что происходит, если установлен старший значащий бит. Это мой текущий код:
uint32_t MuStudio::MIDI::FileReader::varfieldGet()
{
// Note: always swap bytes on return since the loop reads in LE fashion since it
// it cannot be known a priori how many bytes to read
uint32_t ret=0;
uint8_t byte_in;
uint8_t k=0;
while(!m_source.eof())
{
byte_in=m_source.byteGet();
if(byte_in&0x80)
{
if(k==sizeof(ret))
{break;}
// How to continue here?
}
else
{return __builtin_bswap32(ret|(byte_in<<k));}
}
while(!m_source.eof() && byte_in&0x80)
{byte_in=m_source.byteGet();}
return __builtin_bswap32(ret);
}
[1] http://www.music.mcgill.ca/~ich/classes/mumt306/midiformat.pdf
1 ответ
Решение
Значения переменной длины не являются прямым порядком байтов, но имеют большой порядок байтов, но это не имеет значения, потому что вы читаете их по одному байту за раз и создаете результат в своем родном порядке байтов.
Когда вы объединяете байт с битами предыдущего чтения, вы должны сместить предыдущие биты, а не текущий байт.
Когда установлен самый значимый бит, вам не нужно делать ничего особенного; когда бит ясен, вы должны остановиться:
uint32_t varfieldGet()
{
uint32_t ret = 0;
uint8_t byte_in;
for (;;)
{
if (m_source.eof())
// return error
byte_in = m_source.byteGet();
ret = (ret << 7) | (byte_in & 0x7f);
if (!(byte_in & 0x80))
return ret;
}
}