/* * 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: # slesTest_recBuffQueue /sdcard/myrec.raw 4 These use adb on host to retrive the file: % adb pull /sdcard/myrec.raw myrec.raw How to examine the output with Audacity: Project / Import raw data Select myrec.raw file, then click Open button Choose these options: Signed 16-bit PCM Little-endian 1 Channel (Mono) Sample rate 22050 Hz Click Import button How to convert with sox: sox -r 22050 -s -2 myrec.raw myrec.wav */ #include <stdlib.h> #include <stdio.h> #include <string.h> #include <unistd.h> #include <sys/time.h> #include <fcntl.h> #include <SLES/OpenSLES.h> #include <SLES/OpenSLES_Android.h> /* 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_SAMPLES 1024 #define BUFFER_SIZE_IN_BYTES (2*BUFFER_SIZE_IN_SAMPLES) /* Local storage for Audio data */ int8_t pcmData[NB_BUFFERS_IN_QUEUE * BUFFER_SIZE_IN_BYTES]; /* destination for recorded data */ static FILE* gFp; //----------------------------------------------------------------- /* 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(stdout, "%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, SLuint32 event) { if (SL_RECORDEVENT_HEADATNEWPOS & event) { SLmillisecond pMsec = 0; (*caller)->GetPosition(caller, &pMsec); fprintf(stdout, "SL_RECORDEVENT_HEADATNEWPOS current position=%ums\n", pMsec); } if (SL_RECORDEVENT_HEADATMARKER & event) { SLmillisecond pMsec = 0; (*caller)->GetPosition(caller, &pMsec); fprintf(stdout, "SL_RECORDEVENT_HEADATMARKER current position=%ums\n", pMsec); } } //----------------------------------------------------------------- /* Callback for recording buffer queue events */ void RecBufferQueueCallback( SLAndroidSimpleBufferQueueItf queueItf, void *pContext) { //fprintf(stdout, "RecBufferQueueCallback called\n"); CallbackCntxt *pCntxt = (CallbackCntxt*)pContext; /* Save the recorded data */ fwrite(pCntxt->pDataBase, BUFFER_SIZE_IN_BYTES, 1, gFp); /* 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);*/ } //----------------------------------------------------------------- /* Record to an audio path by opening a file descriptor on that path */ void TestRecToBuffQueue( SLObjectItf sl, const char* path, SLAint64 durationInSeconds) { gFp = fopen(path, "w"); if (NULL == gFp) { 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; SLDataFormat_PCM pcm; SLboolean required[NUM_EXPLICIT_INTERFACES_FOR_RECORDER]; SLInterfaceID iidArray[NUM_EXPLICIT_INTERFACES_FOR_RECORDER]; /* 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 < NUM_EXPLICIT_INTERFACES_FOR_RECORDER ; i++) { required[i] = SL_BOOLEAN_FALSE; iidArray[i] = SL_IID_NULL; } /* ------------------------------------------------------ */ /* Configuration of the recorder */ /* Request the AndroidSimpleBufferQueue and AndroidConfiguration interfaces */ required[0] = SL_BOOLEAN_TRUE; iidArray[0] = SL_IID_ANDROIDSIMPLEBUFFERQUEUE; required[1] = SL_BOOLEAN_TRUE; iidArray[1] = SL_IID_ANDROIDCONFIGURATION; /* 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 = SL_DATAFORMAT_PCM; pcm.numChannels = 1; pcm.samplesPerSec = SL_SAMPLINGRATE_22_05; pcm.bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_16; pcm.containerSize = 16; pcm.channelMask = SL_SPEAKER_FRONT_LEFT; pcm.endianness = SL_BYTEORDER_LITTLEENDIAN; recDest.pLocator = (void *) &recBuffQueue; recDest.pFormat = (void * ) &pcm; /* Create the audio recorder */ result = (*EngineItf)->CreateAudioRecorder(EngineItf, &recorder, &recSource, &recDest, NUM_EXPLICIT_INTERFACES_FOR_RECORDER, iidArray, required); ExitOnError(result); fprintf(stdout, "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); fprintf(stdout, "Recorder parameterized with preset %u\n", presetValue); } else { printf("Using default record preset\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); fprintf(stdout, "Recorder realized\n"); /* 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); fprintf(stdout, "Recorder callback registered\n"); /* 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 */ fprintf(stdout,"Enqueueing buffer "); for(int i = 0 ; i < NB_BUFFERS_IN_QUEUE ; i++) { fprintf(stdout,"%d ", i); result = (*recBuffQueueItf)->Enqueue(recBuffQueueItf, cntxt.pData, BUFFER_SIZE_IN_BYTES); ExitOnError(result); cntxt.pData += BUFFER_SIZE_IN_BYTES; } fprintf(stdout,"\n"); cntxt.pData = cntxt.pDataBase; /* ------------------------------------------------------ */ /* Start recording */ result = (*recordItf)->SetRecordState(recordItf, SL_RECORDSTATE_RECORDING); ExitOnError(result); fprintf(stdout, "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); fprintf(stdout, "Stopped recording\n"); /* Destroy the AudioRecorder object */ (*recorder)->Destroy(recorder); fclose(gFp); } //----------------------------------------------------------------- int main(int argc, char* const argv[]) { SLresult result; SLObjectItf sl; const char *prog = argv[0]; fprintf(stdout, "OpenSL ES test %s: exercises SLRecordItf and SLAndroidSimpleBufferQueueItf ", prog); fprintf(stdout, "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 'p': // preset number presetValue = atoi(&arg[2]); break; default: fprintf(stderr, "%s: unknown option %s\n", prog, arg); break; } } if (argc-i < 2) { printf("Usage: \t%s [-p#] destination_file duration_in_seconds\n", prog); 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("Example: \"%s /sdcard/myrec.raw 4\" \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], (SLAint64)atoi(argv[i+1])); /* Shutdown OpenSL ES */ (*sl)->Destroy(sl); return EXIT_SUCCESS; }