C++ 中从内存中解码 libvorbis 音频

libvorbis audio decode from memory in C++

本文关键字:libvorbis 音频 解码 内存 C++      更新时间:2023-10-16

给定一个C++编码的缓冲区,使用oggvorbis结构解码已经在内存中的数据的步骤是什么?

OggVorbis_File无法使用,因为资产位于压缩存档中。

我正在尝试研究必要的结构和方法,但我对音频编码和解码相当陌生。

任何可以帮助我进一步阅读的资源也值得赞赏!

应该澄清一下,我打算使用解码的数据流式传输到 OpenAL。

谢谢。

回答我自己的问题。

这可以通过向 vorbis 提供自定义回调来完成。

struct ogg_file
{
    char* curPtr;
    char* filePtr;
    size_t fileSize;
};
size_t AR_readOgg(void* dst, size_t size1, size_t size2, void* fh)
{
    ogg_file* of = reinterpret_cast<ogg_file*>(fh);
    size_t len = size1 * size2;
    if ( of->curPtr + len > of->filePtr + of->fileSize )
    {
        len = of->filePtr + of->fileSize - of->curPtr;
    }
    memcpy( dst, of->curPtr, len );
    of->curPtr += len;
    return len;
}
int AR_seekOgg( void *fh, ogg_int64_t to, int type ) {
    ogg_file* of = reinterpret_cast<ogg_file*>(fh);
    switch( type ) {
        case SEEK_CUR:
            of->curPtr += to;
            break;
        case SEEK_END:
            of->curPtr = of->filePtr + of->fileSize - to;
            break;
        case SEEK_SET:
            of->curPtr = of->filePtr + to;
            break;
        default:
            return -1;
    }
    if ( of->curPtr < of->filePtr ) {
        of->curPtr = of->filePtr;
        return -1;
    }
    if ( of->curPtr > of->filePtr + of->fileSize ) {
        of->curPtr = of->filePtr + of->fileSize;
        return -1;
    }
    return 0;
}
int AR_closeOgg(void* fh)
{
    return 0;
}
long AR_tellOgg( void *fh )
{
    ogg_file* of = reinterpret_cast<ogg_file*>(fh);
    return (of->curPtr - of->filePtr);
}

用法

ov_callbacks callbacks;
ogg_file t;
t.curPtr = t.filePtr = compressedData;
t.fileSize = compressedDataSize;
OggVorbis_File* ov = new OggVorbis_File;
mOggFile = ov;
memset( ov, 0, sizeof( OggVorbis_File ) );
callbacks.read_func = AR_readOgg;
callbacks.seek_func = AR_seekOgg;
callbacks.close_func = AR_closeOgg;
callbacks.tell_func = AR_tellOgg;
int ret = ov_open_callbacks((void *)&t, ov, NULL, -1, callbacks);
vorbis_info* vi = ov_info(ov, -1);
assert(vi);
/* compressed data is available to use, to uncompress look into ov_read */

特别感谢Doom3 GPL源代码对此的大部分帮助,它可以查看地点 : 这里

你也可以不要

重新发明轮子并使用这样的fmemopen

FILE* memfile = fmemopen(data, len, "r");

其中data是指向内存开始的指针,len是数据的长度。然后像常规对象一样将memfile传递给ov_open FILE对象。

但是,

也有缺点:此功能似乎是特定于Linux的(但是可以在arduino中找到,所以我对它的状态有点困惑),因此您在其他系统上没有它。但是有一些实现(检查窗口或苹果操作系统的libconfuse)。