/* * 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. */ /* Audio Record Test First run the program from shell: % adb shell slesTest_recBuffQueue /sdcard/myrec.wav These use adb on host to retrive the file: % adb pull /sdcard/myrec.wav */ #include #include #include #include #include #include #include #include #include #include #include audio_format_t transferFormat = AUDIO_FORMAT_DEFAULT; uint32_t sampleRate = 48000; int channelCount = 1; bool useIndexChannelMask = false; size_t frameSize; uint32_t performanceMode = SL_ANDROID_PERFORMANCE_LATENCY; bool aec = false; bool agc = false; bool ns = false; /* Preset number to use for recording */ SLuint32 presetValue = SL_ANDROID_RECORDING_PRESET_NONE; /* Explicitly requesting SL_IID_ANDROIDSIMPLEBUFFERQUEUE and SL_IID_ANDROIDCONFIGURATION * on the AudioRecorder object */ #define NUM_EXPLICIT_INTERFACES_FOR_RECORDER 2 /* Size of the recording buffer queue */ #define NB_BUFFERS_IN_QUEUE 1 /* Size of each buffer in the queue */ #define BUFFER_SIZE_IN_BYTES 2048 /* Local storage for Audio data */ int8_t pcmData[NB_BUFFERS_IN_QUEUE * BUFFER_SIZE_IN_BYTES]; /* destination for recorded data */ SNDFILE *sndfile; //----------------------------------------------------------------- /* Exits the application if an error is encountered */ #define ExitOnError(x) ExitOnErrorFunc(x,__LINE__) void ExitOnErrorFunc( SLresult result , int line) { if (SL_RESULT_SUCCESS != result) { fprintf(stderr, "%u error code encountered at line %d, exiting\n", result, line); exit(EXIT_FAILURE); } } //----------------------------------------------------------------- /* Structure for passing information to callback function */ typedef struct CallbackCntxt_ { SLPlayItf playItf; SLuint32 size; SLint8* pDataBase; // Base address of local audio data storage SLint8* pData; // Current address of local audio data storage } CallbackCntxt; //----------------------------------------------------------------- /* Callback for recording buffer queue events */ void RecCallback( SLRecordItf caller, void *pContext __unused, SLuint32 event) { if (SL_RECORDEVENT_HEADATNEWPOS & event) { SLmillisecond pMsec = 0; (*caller)->GetPosition(caller, &pMsec); printf("SL_RECORDEVENT_HEADATNEWPOS current position=%ums\n", pMsec); } if (SL_RECORDEVENT_HEADATMARKER & event) { SLmillisecond pMsec = 0; (*caller)->GetPosition(caller, &pMsec); printf("SL_RECORDEVENT_HEADATMARKER current position=%ums\n", pMsec); } } //----------------------------------------------------------------- /* Callback for recording buffer queue events */ void RecBufferQueueCallback( SLAndroidSimpleBufferQueueItf queueItf, void *pContext) { //printf("RecBufferQueueCallback called\n"); CallbackCntxt *pCntxt = (CallbackCntxt*)pContext; /* Save the recorded data */ sf_count_t frameCount = BUFFER_SIZE_IN_BYTES / frameSize; switch (transferFormat) { case AUDIO_FORMAT_PCM_8_BIT: { short temp[BUFFER_SIZE_IN_BYTES]; memcpy_to_i16_from_u8(temp, (const unsigned char *) pCntxt->pDataBase, BUFFER_SIZE_IN_BYTES); (void) sf_writef_short(sndfile, (const short *) temp, frameCount); } break; case AUDIO_FORMAT_PCM_16_BIT: (void) sf_writef_short(sndfile, (const short *) pCntxt->pDataBase, frameCount); break; case AUDIO_FORMAT_PCM_32_BIT: (void) sf_writef_int(sndfile, (const int *) pCntxt->pDataBase, frameCount); break; case AUDIO_FORMAT_PCM_FLOAT: (void) sf_writef_float(sndfile, (const float *) pCntxt->pDataBase, frameCount); break; default: fprintf(stderr, "Unsupported transfer format %d\n", transferFormat); exit(EXIT_FAILURE); } /* Increase data pointer by buffer size */ pCntxt->pData += BUFFER_SIZE_IN_BYTES; if (pCntxt->pData >= pCntxt->pDataBase + (NB_BUFFERS_IN_QUEUE * BUFFER_SIZE_IN_BYTES)) { pCntxt->pData = pCntxt->pDataBase; } ExitOnError( (*queueItf)->Enqueue(queueItf, pCntxt->pDataBase, BUFFER_SIZE_IN_BYTES) ); SLAndroidSimpleBufferQueueState recQueueState; ExitOnError( (*queueItf)->GetState(queueItf, &recQueueState) ); /*fprintf(stderr, "\tRecBufferQueueCallback now has pCntxt->pData=%p queue: " "count=%u playIndex=%u\n", pCntxt->pData, recQueueState.count, recQueueState.index);*/ //printf("RecBufferQueueCallback returned\n"); } //----------------------------------------------------------------- /* Record to an audio path by opening a file descriptor on that path */ void TestRecToBuffQueue( SLObjectItf sl, const char* path, SLAint64 durationInSeconds) { SF_INFO info; info.frames = 0; info.samplerate = sampleRate; info.channels = channelCount; switch (transferFormat) { case AUDIO_FORMAT_PCM_8_BIT: info.format = SF_FORMAT_WAV | SF_FORMAT_PCM_U8; break; case AUDIO_FORMAT_PCM_16_BIT: info.format = SF_FORMAT_WAV | SF_FORMAT_PCM_16; break; case AUDIO_FORMAT_PCM_32_BIT: info.format = SF_FORMAT_WAV | SF_FORMAT_PCM_32; break; case AUDIO_FORMAT_PCM_FLOAT: info.format = SF_FORMAT_WAV | SF_FORMAT_FLOAT; break; default: fprintf(stderr, "Unsupported transfer format %d\n", transferFormat); exit(EXIT_FAILURE); } sndfile = sf_open(path, SFM_WRITE, &info); if (sndfile == NULL) { ExitOnError(SL_RESULT_RESOURCE_ERROR); } SLresult result; SLEngineItf EngineItf; /* Objects this application uses: one audio recorder */ SLObjectItf recorder; /* Interfaces for the audio recorder */ SLAndroidSimpleBufferQueueItf recBuffQueueItf; SLRecordItf recordItf; SLAndroidConfigurationItf configItf; /* Source of audio data for the recording */ SLDataSource recSource; SLDataLocator_IODevice ioDevice; /* Data sink for recorded audio */ SLDataSink recDest; SLDataLocator_AndroidSimpleBufferQueue recBuffQueue; /* As mentioned in the Android extension API documentation this is identical to * OpenSL ES 1.1 SLDataFormat_PCM_EX, and can be used for an instance of SLDataFormat_PCM. */ SLAndroidDataFormat_PCM_EX pcm; int numInterfaces = NUM_EXPLICIT_INTERFACES_FOR_RECORDER; if (aec) numInterfaces++; if (agc) numInterfaces++; if (ns) numInterfaces++; SLboolean required[numInterfaces]; SLInterfaceID iidArray[numInterfaces]; /* Get the SL Engine Interface which is implicit */ result = (*sl)->GetInterface(sl, SL_IID_ENGINE, (void*)&EngineItf); ExitOnError(result); /* Initialize arrays required[] and iidArray[] */ for (int i=0 ; i < numInterfaces ; i++) { required[i] = SL_BOOLEAN_FALSE; iidArray[i] = SL_IID_NULL; } /* ------------------------------------------------------ */ /* Configuration of the recorder */ /* Request the AndroidSimpleBufferQueue and AndroidConfiguration interfaces */ int index = 0; required[index] = SL_BOOLEAN_TRUE; iidArray[index++] = SL_IID_ANDROIDSIMPLEBUFFERQUEUE; required[index] = SL_BOOLEAN_TRUE; iidArray[index++] = SL_IID_ANDROIDCONFIGURATION; if (aec) { iidArray[index++] = SL_IID_ANDROIDACOUSTICECHOCANCELLATION; } if (agc) { iidArray[index++] = SL_IID_ANDROIDAUTOMATICGAINCONTROL; } if (ns) { iidArray[index++] = SL_IID_ANDROIDNOISESUPPRESSION; } /* Setup the data source */ ioDevice.locatorType = SL_DATALOCATOR_IODEVICE; ioDevice.deviceType = SL_IODEVICE_AUDIOINPUT; ioDevice.deviceID = SL_DEFAULTDEVICEID_AUDIOINPUT; ioDevice.device = NULL; recSource.pLocator = (void *) &ioDevice; recSource.pFormat = NULL; /* Setup the data sink */ recBuffQueue.locatorType = SL_DATALOCATOR_ANDROIDSIMPLEBUFFERQUEUE; recBuffQueue.numBuffers = NB_BUFFERS_IN_QUEUE; /* set up the format of the data in the buffer queue */ pcm.formatType = transferFormat == AUDIO_FORMAT_PCM_FLOAT || transferFormat == AUDIO_FORMAT_PCM_8_BIT ? SL_ANDROID_DATAFORMAT_PCM_EX : SL_DATAFORMAT_PCM; pcm.numChannels = channelCount; pcm.sampleRate = sampleRate * 1000; // milliHz pcm.representation = SL_ANDROID_PCM_REPRESENTATION_SIGNED_INT; switch (transferFormat) { case AUDIO_FORMAT_PCM_16_BIT: pcm.bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_16; pcm.containerSize = 16; break; case AUDIO_FORMAT_PCM_32_BIT: pcm.bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_32; pcm.containerSize = 32; break; case AUDIO_FORMAT_PCM_8_BIT: pcm.bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_8; pcm.containerSize = 8; pcm.representation = SL_ANDROID_PCM_REPRESENTATION_UNSIGNED_INT; break; case AUDIO_FORMAT_PCM_FLOAT: pcm.bitsPerSample = 32; pcm.containerSize = 32; pcm.representation = SL_ANDROID_PCM_REPRESENTATION_FLOAT; break; default: fprintf(stderr, "Unsupported transfer format %d\n", transferFormat); exit(EXIT_FAILURE); } if (useIndexChannelMask) { pcm.channelMask = (1 << channelCount) - 1; } else { switch (channelCount) { case 1: pcm.channelMask = SL_SPEAKER_FRONT_LEFT; break; case 2: pcm.channelMask = SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT; break; default: fprintf(stderr, "Unsupported channel count %d\n", channelCount); exit(EXIT_FAILURE); } } pcm.endianness = SL_BYTEORDER_LITTLEENDIAN; recDest.pLocator = (void *) &recBuffQueue; recDest.pFormat = (void * ) &pcm; /* Create the audio recorder */ result = (*EngineItf)->CreateAudioRecorder(EngineItf, &recorder, &recSource, &recDest, numInterfaces, iidArray, required); ExitOnError(result); printf("Recorder created\n"); /* Get the Android configuration interface which is explicit */ result = (*recorder)->GetInterface(recorder, SL_IID_ANDROIDCONFIGURATION, (void*)&configItf); ExitOnError(result); /* Use the configuration interface to configure the recorder before it's realized */ if (presetValue != SL_ANDROID_RECORDING_PRESET_NONE) { result = (*configItf)->SetConfiguration(configItf, SL_ANDROID_KEY_RECORDING_PRESET, &presetValue, sizeof(SLuint32)); ExitOnError(result); printf("Recorder configured with preset %u\n", presetValue); } else { printf("Using default record preset\n"); } if (performanceMode != SL_ANDROID_PERFORMANCE_LATENCY) { result = (*configItf)->SetConfiguration(configItf, SL_ANDROID_KEY_PERFORMANCE_MODE, &performanceMode, sizeof(SLuint32)); ExitOnError(result); printf("Recorder configured with performance mode %u\n", performanceMode); } else { printf("Using default performance mode\n"); } SLuint32 presetRetrieved = SL_ANDROID_RECORDING_PRESET_NONE; SLuint32 presetSize = 2*sizeof(SLuint32); // intentionally too big result = (*configItf)->GetConfiguration(configItf, SL_ANDROID_KEY_RECORDING_PRESET, &presetSize, (void*)&presetRetrieved); ExitOnError(result); if (presetValue == SL_ANDROID_RECORDING_PRESET_NONE) { printf("The default record preset appears to be %u\n", presetRetrieved); } else if (presetValue != presetRetrieved) { fprintf(stderr, "Error retrieving recording preset as %u instead of %u\n", presetRetrieved, presetValue); ExitOnError(SL_RESULT_INTERNAL_ERROR); } /* Realize the recorder in synchronous mode. */ result = (*recorder)->Realize(recorder, SL_BOOLEAN_FALSE); ExitOnError(result); printf("Recorder realized\n"); /* Check actual performance mode granted*/ SLuint32 modeRetrieved = SL_ANDROID_PERFORMANCE_NONE; SLuint32 modeSize = sizeof(SLuint32); result = (*configItf)->GetConfiguration(configItf, SL_ANDROID_KEY_PERFORMANCE_MODE, &modeSize, (void*)&modeRetrieved); ExitOnError(result); printf("Actual performance mode is %u\n", modeRetrieved); /* Get the record interface which is implicit */ result = (*recorder)->GetInterface(recorder, SL_IID_RECORD, (void*)&recordItf); ExitOnError(result); /* Set up the recorder callback to get events during the recording */ result = (*recordItf)->SetMarkerPosition(recordItf, 2000); ExitOnError(result); result = (*recordItf)->SetPositionUpdatePeriod(recordItf, 500); ExitOnError(result); result = (*recordItf)->SetCallbackEventsMask(recordItf, SL_RECORDEVENT_HEADATMARKER | SL_RECORDEVENT_HEADATNEWPOS); ExitOnError(result); result = (*recordItf)->RegisterCallback(recordItf, RecCallback, NULL); ExitOnError(result); printf("Recorder callback registered\n"); /* Enable AEC if requested */ if (aec) { SLAndroidAcousticEchoCancellationItf aecItf; result = (*recorder)->GetInterface( recorder, SL_IID_ANDROIDACOUSTICECHOCANCELLATION, (void*)&aecItf); printf("AEC is %savailable\n", SL_RESULT_SUCCESS == result ? "" : "not "); if (SL_RESULT_SUCCESS == result) { result = (*aecItf)->SetEnabled(aecItf, true); ExitOnError(result); SLboolean enabled; result = (*aecItf)->IsEnabled(aecItf, &enabled); ExitOnError(result); printf("AEC is %s\n", enabled ? "enabled" : "not enabled"); } } /* Enable AGC if requested */ if (agc) { SLAndroidAutomaticGainControlItf agcItf; result = (*recorder)->GetInterface( recorder, SL_IID_ANDROIDAUTOMATICGAINCONTROL, (void*)&agcItf); printf("AGC is %savailable\n", SL_RESULT_SUCCESS == result ? "" : "not "); if (SL_RESULT_SUCCESS == result) { result = (*agcItf)->SetEnabled(agcItf, true); ExitOnError(result); SLboolean enabled; result = (*agcItf)->IsEnabled(agcItf, &enabled); ExitOnError(result); printf("AGC is %s\n", enabled ? "enabled" : "not enabled"); } } /* Enable NS if requested */ if (ns) { SLAndroidNoiseSuppressionItf nsItf; result = (*recorder)->GetInterface( recorder, SL_IID_ANDROIDNOISESUPPRESSION, (void*)&nsItf); printf("NS is %savailable\n", SL_RESULT_SUCCESS == result ? "" : "not "); if (SL_RESULT_SUCCESS == result) { result = (*nsItf)->SetEnabled(nsItf, true); ExitOnError(result); SLboolean enabled; result = (*nsItf)->IsEnabled(nsItf, &enabled); ExitOnError(result); printf("NS is %s\n", enabled ? "enabled" : "not enabled"); } } /* Get the buffer queue interface which was explicitly requested */ result = (*recorder)->GetInterface(recorder, SL_IID_ANDROIDSIMPLEBUFFERQUEUE, (void*)&recBuffQueueItf); ExitOnError(result); /* ------------------------------------------------------ */ /* Initialize the callback and its context for the recording buffer queue */ CallbackCntxt cntxt; cntxt.pDataBase = (int8_t*)&pcmData; cntxt.pData = cntxt.pDataBase; cntxt.size = sizeof(pcmData); result = (*recBuffQueueItf)->RegisterCallback(recBuffQueueItf, RecBufferQueueCallback, &cntxt); ExitOnError(result); /* Enqueue buffers to map the region of memory allocated to store the recorded data */ printf("Enqueueing buffer "); for(int i = 0 ; i < NB_BUFFERS_IN_QUEUE ; i++) { printf("%d ", i); result = (*recBuffQueueItf)->Enqueue(recBuffQueueItf, cntxt.pData, BUFFER_SIZE_IN_BYTES); ExitOnError(result); cntxt.pData += BUFFER_SIZE_IN_BYTES; } printf("\n"); cntxt.pData = cntxt.pDataBase; /* ------------------------------------------------------ */ /* Start recording */ result = (*recordItf)->SetRecordState(recordItf, SL_RECORDSTATE_RECORDING); ExitOnError(result); printf("Starting to record\n"); /* Record for at least a second */ if (durationInSeconds < 1) { durationInSeconds = 1; } usleep(durationInSeconds * 1000 * 1000); /* ------------------------------------------------------ */ /* End of recording */ /* Stop recording */ result = (*recordItf)->SetRecordState(recordItf, SL_RECORDSTATE_STOPPED); ExitOnError(result); printf("Stopped recording\n"); /* Destroy the AudioRecorder object */ (*recorder)->Destroy(recorder); sf_close(sndfile); } //----------------------------------------------------------------- int main(int argc, char* const argv[]) { int durationInSeconds = 10; SLresult result; SLObjectItf sl; const char *prog = argv[0]; printf("OpenSL ES test %s: exercises SLRecordItf and SLAndroidSimpleBufferQueueItf ", prog); printf("on an AudioRecorder object\n"); int i; for (i = 1; i < argc; ++i) { const char *arg = argv[i]; if (arg[0] != '-') { break; } switch (arg[1]) { case 'c': // channel count channelCount = atoi(&arg[2]); break; case 'd': // duration in seconds durationInSeconds = atoi(&arg[2]); break; case 'f': transferFormat = AUDIO_FORMAT_PCM_FLOAT; break; case 'i': useIndexChannelMask = true; break; case 'p': // preset number presetValue = atoi(&arg[2]); break; case 'r': sampleRate = atoi(&arg[2]); break; case '1': transferFormat = AUDIO_FORMAT_PCM_8_BIT; break; case '2': transferFormat = AUDIO_FORMAT_PCM_16_BIT; break; case '4': transferFormat = AUDIO_FORMAT_PCM_32_BIT; break; case 'm': performanceMode = atoi(&arg[2]); break; case 'x': if (strstr(arg, "e") != NULL) { aec = true; } if (strstr(arg, "a") != NULL) { agc = true; } if (strstr(arg, "n") != NULL) { ns = true; } break; default: fprintf(stderr, "%s: unknown option %s\n", prog, arg); break; } } if (transferFormat == AUDIO_FORMAT_DEFAULT) { transferFormat = AUDIO_FORMAT_PCM_16_BIT; } frameSize = audio_bytes_per_sample(transferFormat) * channelCount; if (argc-i != 1) { printf("Usage: \t%s [-c#] [-d#] [-i] [-p#] [-r#] [-1/2/4/f] destination_file\n", prog); printf(" -c# channel count, defaults to 1\n"); printf(" -d# duration in seconds, default to 10\n"); printf(" -i index channel mask, not yet implemented\n"); printf(" -p# is the preset value which defaults to SL_ANDROID_RECORDING_PRESET_NONE\n"); printf(" possible values are:\n"); printf(" -p%d SL_ANDROID_RECORDING_PRESET_NONE\n", SL_ANDROID_RECORDING_PRESET_NONE); printf(" -p%d SL_ANDROID_RECORDING_PRESET_GENERIC\n", SL_ANDROID_RECORDING_PRESET_GENERIC); printf(" -p%d SL_ANDROID_RECORDING_PRESET_CAMCORDER\n", SL_ANDROID_RECORDING_PRESET_CAMCORDER); printf(" -p%d SL_ANDROID_RECORDING_PRESET_VOICE_RECOGNITION\n", SL_ANDROID_RECORDING_PRESET_VOICE_RECOGNITION); printf(" -p%d SL_ANDROID_RECORDING_PRESET_VOICE_COMMUNICATION\n", SL_ANDROID_RECORDING_PRESET_VOICE_COMMUNICATION); printf(" -r# sample rate in Hz, defaults to 48000\n"); printf(" -[1/2/4/f] sample format: 8-bit unsigned, 16-bit signed, 32-bit signed, float, " "defaults to 16-bit signed\n"); printf(" -m# is the performance mode value which defaults to" " SL_ANDROID_PERFORMANCE_LATENCY\n"); printf(" possible values are:\n"); printf(" -m%d SL_ANDROID_PERFORMANCE_NONE\n", SL_ANDROID_PERFORMANCE_NONE); printf(" -m%d SL_ANDROID_PERFORMANCE_LATENCY\n", SL_ANDROID_PERFORMANCE_LATENCY); printf(" -m%d SL_ANDROID_PERFORMANCE_LATENCY_EFFECTS\n", SL_ANDROID_PERFORMANCE_LATENCY_EFFECTS); printf(" -m%d SL_ANDROID_PERFORMANCE_POWER_SAVING\n", SL_ANDROID_PERFORMANCE_POWER_SAVING); printf(" -x[e][a][n] for pre processing:\n" " - e: Echo canceler\n" " - a: Automatic Gain Control\n" " - n: Noise Suppression\n"); printf("Example: \"%s /sdcard/myrec.wav\" \n", prog); exit(EXIT_FAILURE); } SLEngineOption EngineOption[] = { {(SLuint32) SL_ENGINEOPTION_THREADSAFE, (SLuint32) SL_BOOLEAN_TRUE} }; result = slCreateEngine( &sl, 1, EngineOption, 0, NULL, NULL); ExitOnError(result); /* Realizing the SL Engine in synchronous mode. */ result = (*sl)->Realize(sl, SL_BOOLEAN_FALSE); ExitOnError(result); TestRecToBuffQueue(sl, argv[i], durationInSeconds); /* Shutdown OpenSL ES */ (*sl)->Destroy(sl); return EXIT_SUCCESS; }