/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // Test various combinations of data sources and sinks #include <assert.h> #include <stdio.h> #include <stdlib.h> #include <SLES/OpenSLES.h> int main(int argc __unused, char **argv __unused) { SLresult result; SLObjectItf engineObject; // create engine result = slCreateEngine(&engineObject, 0, NULL, 0, NULL, NULL); assert(SL_RESULT_SUCCESS == result); SLEngineItf engineEngine; result = (*engineObject)->Realize(engineObject, SL_BOOLEAN_FALSE); assert(SL_RESULT_SUCCESS == result); result = (*engineObject)->GetInterface(engineObject, SL_IID_ENGINE, &engineEngine); assert(SL_RESULT_SUCCESS == result); // configure a typical audio source of 44.1 kHz stereo 16-bit little endian SLDataLocator_BufferQueue loc_bufq; loc_bufq.locatorType = SL_DATALOCATOR_BUFFERQUEUE; loc_bufq.numBuffers = 1; SLDataFormat_PCM format_pcm; format_pcm.formatType = SL_DATAFORMAT_PCM; format_pcm.numChannels = 2; format_pcm.samplesPerSec = SL_SAMPLINGRATE_44_1; format_pcm.bitsPerSample = 16; format_pcm.containerSize = 16; format_pcm.channelMask = SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT; format_pcm.endianness = SL_BYTEORDER_LITTLEENDIAN; SLDataSource audioSrc; audioSrc.pLocator = &loc_bufq; audioSrc.pFormat = &format_pcm; // configure audio sink SLDataLocator_OutputMix loc_outmix; loc_outmix.locatorType = SL_DATALOCATOR_OUTPUTMIX; loc_outmix.outputMix = NULL; SLDataSink audioSnk; audioSnk.pLocator = &loc_outmix; audioSnk.pFormat = NULL; // create audio player using a NULL output mix SLInterfaceID ids[1] = {SL_IID_BUFFERQUEUE}; SLboolean req[1] = {SL_BOOLEAN_TRUE}; SLObjectItf playerObject; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); // create audio player using an engine as the output mix loc_outmix.outputMix = engineObject; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); // create output mix but don't realize it yet SLObjectItf outputMixObject; result = (*engineEngine)->CreateOutputMix(engineEngine, &outputMixObject, 0, NULL, NULL); assert(SL_RESULT_SUCCESS == result); // create audio player using the unrealized output mix loc_outmix.outputMix = outputMixObject; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PRECONDITIONS_VIOLATED == result); assert(NULL == playerObject); // now realize the output mix result = (*outputMixObject)->Realize(outputMixObject, SL_BOOLEAN_FALSE); assert(SL_RESULT_SUCCESS == result); // create audio player using the realized output mix // and a bogus data format for the sink (ignored per spec) audioSnk.pFormat = (void *) 0xDEADBEEF; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_SUCCESS == result); assert(NULL != playerObject); audioSnk.pFormat = NULL; // destroy player (*playerObject)->Destroy(playerObject); // now try to create audio player using various unusual parameters // number of channels format_pcm.numChannels = 0; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.numChannels = 3; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.numChannels = 2; // sample rate format_pcm.samplesPerSec = 0; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.samplesPerSec = 1000; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_CONTENT_UNSUPPORTED == result); assert(NULL == playerObject); format_pcm.samplesPerSec = SL_SAMPLINGRATE_44_1; // bits per sample format_pcm.bitsPerSample = 17; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.bitsPerSample = 24; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.bitsPerSample = 16; // container size format_pcm.containerSize = 8; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.containerSize = 32; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_CONTENT_UNSUPPORTED == result); assert(NULL == playerObject); format_pcm.containerSize = 16; // channel mask format_pcm.channelMask = 0; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_SUCCESS == result); assert(NULL != playerObject); (*playerObject)->Destroy(playerObject); format_pcm.channelMask = SL_SPEAKER_FRONT_CENTER; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.channelMask = SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT | SL_SPEAKER_FRONT_CENTER; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.numChannels = 1; format_pcm.channelMask = 0; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_SUCCESS == result); assert(NULL != playerObject); (*playerObject)->Destroy(playerObject); format_pcm.channelMask = SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.numChannels = 2; // endianness format_pcm.endianness = SL_BYTEORDER_BIGENDIAN; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); #ifdef ANDROID // known bug on SDL assert(SL_RESULT_CONTENT_UNSUPPORTED == result); assert(NULL == playerObject); #else if (SL_RESULT_CONTENT_UNSUPPORTED != result) { printf("ERROR: expected SL_RESULT_CONTENT_UNSUPPORTED\n"); if (NULL != playerObject) (*playerObject)->Destroy(playerObject); } #endif format_pcm.endianness = 0; result = (*engineEngine)->CreateAudioPlayer(engineEngine, &playerObject, &audioSrc, &audioSnk, 1, ids, req); assert(SL_RESULT_PARAMETER_INVALID == result); assert(NULL == playerObject); format_pcm.endianness = SL_BYTEORDER_LITTLEENDIAN; // destroy output mix (*outputMixObject)->Destroy(outputMixObject); // destroy engine (*engineObject)->Destroy(engineObject); return EXIT_SUCCESS; }