在安卓系统中使用OpenSL从内存中播放PCM WAVE音频

3
我正在尝试设置OpenSL AudioPlayer,以使用我分配的内存来播放wav文件。我这样做是为了可以拥有多个AudioPlayers共享相同的数据并节省内存。
我尝试将整个文件提供给openSL,并告诉它这是一个WAVE格式的文件,使用format_mime。
 SLDataLocator_Address loc_fd = {SL_DATALOCATOR_ADDRESS, data, size};
SLDataFormat_MIME format_mime = { SL_DATAFORMAT_MIME, (SLchar*)"audio/x-wav",SL_CONTAINERTYPE_WAV};
SLDataSource audioSrc = { &loc_fd, &format_mime };
// configure audio sink
SLDataLocator_OutputMix loc_outmix = { SL_DATALOCATOR_OUTPUTMIX,outputMixObject };
SLDataSink audioSnk = { &loc_outmix, 0 };
// create audio player
const SLInterfaceID ids[2] = { SL_IID_SEEK, SL_IID_PLAYBACKRATE };
const SLboolean req[2] = { SL_BOOLEAN_FALSE, SL_BOOLEAN_FALSE };
result = (*engineEngine)->CreateAudioPlayer(engineEngine,&uriPlayerObject[cntSOUND],&audioSrc, &audioSnk, 0, ids, req);

我已经自己解析了WAVE数据,并加载了format_pcm格式。

SLDataFormat_PCM format_pcm;
format_pcm.formatType = SL_DATAFORMAT_PCM;
char* wavParser = isWAVE(data);
if(wavParser == NULL)
{
    Log("NOT A WAVE!");
    return -1;
}
char* fmtChunk = getChunk("fmt ", data, size);
parsefmtChunk(fmtChunk, &format_pcm);
char* dataChunk = getChunk("data",data, size);
dataChunk += 4;
unsigned int dataSize = *((unsigned int*)dataChunk);
dataChunk += 4;
format_pcm.channelMask = 0;
format_pcm.containerSize = 16;
format_pcm.endianness = SL_BYTEORDER_LITTLEENDIAN;
loc_fd.pAddress = dataChunk;
loc_fd.length = dataSize;

parsefmtChunk函数是

void parsefmtChunk(char* fmtchunk, SLDataFormat_PCM* pcm)
{
char* data = fmtchunk + 8;
unsigned short audioFormat = *((unsigned short*)data);
if(audioFormat != 1)
{
    Log("Not PCM!");
    Log("Reached Line:%d in File %s", __LINE__, __FILE__);
    return;
}
data += 2;
pcm->numChannels = *((unsigned short*)data);
data += 2;
pcm->samplesPerSec = *((unsigned int*)data);
data += 4;
//Byte Rate
data += 4;
//Block Align
data += 2;
//BitsPerSample
pcm->bitsPerSample = *((unsigned short*)data);

(Byte Rate和Block Align是否应该用于填充pcm结构?)
但是,每当我创建audioplayer时,我都会收到SL_RESULT_CONTENT_UNSUPPORTED错误。
这是我从parsefmt函数记录的内容 通道数:2 每秒采样率:44100 每个样本的位数:16
2个回答

3

从android-ndk-r8b/docs/opensles/index.html

PCM数据格式

PCM数据格式只能与缓冲队列一起使用。

所以,像我之前想的那样使用SLDataLocator_Address就无法使用SLDataFormat_PCM。

相反,我可以使用一个大的缓冲队列来代替,如下所示:

bufferqueueitf->Enqueue(bufferqueueitf,dataChunk,dataSize);

1
你试过这个吗?
SLDataFormat_MIME format_mime = {SL_DATAFORMAT_MIME, NULL, SL_CONTAINERTYPE_UNSPECIFIED};

Android实现的OpenSL ES并不完全兼容,http://mobilepearls.com/labs/native-android-api/ndk/docs/opensles/ 建议如下:

Android实现的OpenSL ES要求mimeType必须初始化为NULL或有效的UTF-8字符串,并且containerType必须初始化为有效值。在没有其他考虑因素,如可移植性到其他实现,或无法通过头标识的内容格式等情况下,我们建议将mimeType设置为NULL, containerType设置为SL_CONTAINERTYPE_UNSPECIFIED。

此外,请确保提供有效的URI。

不使用URI(尽管规范没有提到,但MIME也适用于Android文件描述符)。 - agoaj

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接