Декодировать поле переменной длины 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;
    }
}
Другие вопросы по тегам