Воспроизведение звука PCM WAVE из памяти с помощью OpenSL на Android

Я пытаюсь настроить OpenSL AudioPlayer для использования памяти, выделенной для воспроизведения файла WAV. Я хочу сделать это, чтобы у меня было несколько AudioPlayer, которые совместно используют одни и те же данные и сохраняют память.

Я попытался дать 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?)

но всякий раз, когда я создаю аудиоплеер, я получаю SL_RESULT_CONTENT_UNSUPPORTED

Это то, что я регистрирую из моей функции parsefmt
Каналы: 2
samplesPerSec: 44100
BitsPerSample: 16

3

Решение

из android-ndk-r8b / docs / opensles / index.html

Формат данных PCM

Формат данных PCM может использоваться только с буферными очередями.

Итак, SLDataFormat_PCM НЕ МОГУ использоваться с SLDataLocator_Address, как я и предполагал.

Вместо этого я могу делать то, что хочу, с помощью очереди буферов, используя только одну большую очередь, как

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

Другие решения

Вы пробовали это?

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.

1

По вопросам рекламы [email protected]