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

时间:2012-11-18 04:13:52

标签: c++ oggvorbis vorbis

考虑到C ++中的编码缓冲区,使用oggvorbis结构解码已经存储在内的数据的步骤是什么?

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

我试图研究必要的结构和方法,但我对音频编码和解码相当新。

任何可以帮助我进一步阅读的资源也会受到赞赏!

我应该澄清一下,我打算使用解码后的数据流入OpenAL。

感谢。

2 个答案:

答案 0 :(得分:9)

回答我自己的问题。

这可以通过向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源代码的大部分帮助,它可以 请查看:here

答案 1 :(得分:1)

您也不能重新发明轮子,而是像这样使用fmemopen

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

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

但是,它有一个缺点:该函数似乎是特定于linux的(但是可以在arduino中找到,所以我对其状态有些困惑),所以在其他系统上则没有它。但是它们有一些实现方式(检查libconfuse的窗口或apple OSes)。