blob: c8a841202d703d2f29c8651be4b2c5933ad91d9c [file] [log] [blame]
/*
* Copyright (c) 2008, 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.
*/
//#define LOG_NDEBUG 0
#define LOG_TAG "audio_input"
#include <utils/Log.h>
#include <utils/threads.h>
#include "oscl_base.h"
#include "android_audio_input.h"
#include "pv_mime_string_utils.h"
#include "oscl_dll.h"
#include "pvmf_duration_infomessage.h"
#include <media/AudioRecord.h>
#include <sys/prctl.h>
using namespace android;
// TODO: get buffer size from AudioFlinger
static const int kBufferSize = 2048;
// Define entry point for this DLL
OSCL_DLL_ENTRY_POINT_DEFAULT()
// At the start of a recording we are going to mute the first 600ms in
// order to eliminate recording of the videorecorder signaltone.
static const int32 AUTO_RAMP_START_MS = 600;
// After the initial mute we're going to linearly ramp up the volume
// over the next 300ms.
static const int32 AUTO_RAMP_DURATION_MS = 300;
////////////////////////////////////////////////////////////////////////////
AndroidAudioInput::AndroidAudioInput(uint32 audioSource)
: OsclTimerObject(OsclActiveObject::EPriorityNominal, "AndroidAudioInput"),
iCmdIdCounter(0),
iPeer(NULL),
iThreadLoggedOn(false),
iAudioNumChannels(DEFAULT_AUDIO_NUMBER_OF_CHANNELS),
iAudioSamplingRate(DEFAULT_AUDIO_SAMPLING_RATE),
iAudioSource(audioSource),
iDataEventCounter(0),
iWriteCompleteAO(NULL),
iTimeStamp(0),
iMediaBufferMemPool(NULL),
iState(STATE_IDLE),
iMaxAmplitude(0),
iTrackMaxAmplitude(false),
iAudioThreadStarted(false),
iAuthorClock(NULL),
iClockNotificationsInf(NULL),
iFirstFrameReceived(false),
iFirstFrameTs(0)
{
LOGV("AndroidAudioInput constructor %p", this);
// semaphore used to communicate between this mio and the audio output thread
iAudioThreadSem = new OsclSemaphore();
iAudioThreadSem->Create(0);
iAudioThreadTermSem = new OsclSemaphore();
iAudioThreadTermSem->Create(0);
iAudioThreadStartLock = new Mutex();
iAudioThreadStartCV = new Condition();
//locks to access the queues by this mio and by the audio output thread
iWriteResponseQueueLock.Create();
iOSSRequestQueueLock.Create();
{
iAudioFormat=PVMF_MIME_FORMAT_UNKNOWN;
iExitAudioThread=false;
iCommandCounter=0;
iCommandResponseQueue.reserve(5);
iWriteResponseQueue.reserve(5);
iOSSRequestQueue.reserve(5);
iObserver=NULL;
iWriteBusy=false;
iWriteBusySeqNum=0;
}
}
////////////////////////////////////////////////////////////////////////////
AndroidAudioInput::~AndroidAudioInput()
{
LOGV("AndroidAudioInput destructor %p", this);
if(iWriteCompleteAO)
{
OSCL_DELETE(iWriteCompleteAO);
iWriteCompleteAO = NULL;
}
if(iMediaBufferMemPool)
{
OSCL_DELETE(iMediaBufferMemPool);
iMediaBufferMemPool = NULL;
}
iWriteResponseQueueLock.Close();
iOSSRequestQueueLock.Close();
iAudioThreadSem->Close();
delete iAudioThreadSem;
iAudioThreadTermSem->Close();
delete iAudioThreadTermSem;
delete iAudioThreadStartLock;
delete iAudioThreadStartCV;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::connect(PvmiMIOSession& aSession, PvmiMIOObserver* aObserver)
{
LOGV("connect");
if(!aObserver)
{
LOGE("connect: aObserver is NULL");
return PVMFFailure;
}
int32 err = 0;
OSCL_TRY(err, iObservers.push_back(aObserver));
OSCL_FIRST_CATCH_ANY(err, return PVMFErrNoMemory);
aSession = (PvmiMIOSession)(iObservers.size() - 1); // Session ID is the index of observer in the vector
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::disconnect(PvmiMIOSession aSession)
{
LOGV("disconnect");
uint32 index = (uint32)aSession;
if(index >= iObservers.size())
{
LOGE("disconnect: Invalid session ID: %d", index);
return PVMFFailure;
}
iObservers.erase(iObservers.begin()+index);
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
PvmiMediaTransfer* AndroidAudioInput::createMediaTransfer(PvmiMIOSession& aSession,
PvmiKvp* read_formats,
int32 read_flags,
PvmiKvp* write_formats,
int32 write_flags)
{
LOGV("createMediaTransfer %p", this);
OSCL_UNUSED_ARG(read_formats);
OSCL_UNUSED_ARG(read_flags);
OSCL_UNUSED_ARG(write_formats);
OSCL_UNUSED_ARG(write_flags);
uint32 index = (uint32)aSession;
if(index >= iObservers.size())
{
LOGE("Invalid sessions ID: index %d, size %d", index, iObservers.size());
// Invalid session ID
OSCL_LEAVE(OsclErrArgument);
return NULL;
}
iWriteCompleteAO = OSCL_NEW(AndroidAudioInputThreadSafeCallbackAO,(this,5));
return (PvmiMediaTransfer*)this;
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::deleteMediaTransfer(PvmiMIOSession& aSession,
PvmiMediaTransfer* media_transfer)
{
LOGV("deleteMediaTransfer %p", this);
uint32 index = (uint32)aSession;
if(!media_transfer || index >= iObservers.size())
{
LOGE("Invalid sessions ID: index %d, size %d", index, iObservers.size());
// Invalid session ID
OSCL_LEAVE(OsclErrArgument);
}
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::QueryUUID(const PvmfMimeString& aMimeType,
Oscl_Vector<PVUuid, OsclMemAllocator>& aUuids,
bool aExactUuidsOnly,
const OsclAny* aContext)
{
OSCL_UNUSED_ARG(aMimeType);
OSCL_UNUSED_ARG(aExactUuidsOnly);
int32 err = 0;
OSCL_TRY(err, aUuids.push_back(PVMI_CAPABILITY_AND_CONFIG_PVUUID););
OSCL_FIRST_CATCH_ANY(err, OSCL_LEAVE(OsclErrNoMemory););
return AddCmdToQueue(AI_CMD_QUERY_UUID, aContext);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::QueryInterface(const PVUuid& aUuid,
PVInterface*& aInterfacePtr,
const OsclAny* aContext)
{
if(aUuid == PVMI_CAPABILITY_AND_CONFIG_PVUUID)
{
PvmiCapabilityAndConfig* myInterface = OSCL_STATIC_CAST(PvmiCapabilityAndConfig*,this);
aInterfacePtr = OSCL_STATIC_CAST(PVInterface*, myInterface);
}
else
{
aInterfacePtr = NULL;
}
return AddCmdToQueue(AI_CMD_QUERY_INTERFACE, aContext, (OsclAny*)&aInterfacePtr);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::Init(const OsclAny* aContext)
{
LOGV("Init");
if(iState != STATE_IDLE)
{
LOGE("Init: Invalid state (%d)", iState);
OSCL_LEAVE(OsclErrInvalidState);
return -1;
}
return AddCmdToQueue(AI_CMD_INIT, aContext);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::Start(const OsclAny* aContext)
{
LOGV("Start");
if(iState != STATE_INITIALIZED && iState != STATE_PAUSED)
{
LOGE("Start: Invalid state (%d)", iState);
OSCL_LEAVE(OsclErrInvalidState);
return -1;
}
return AddCmdToQueue(AI_CMD_START, aContext);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::Pause(const OsclAny* aContext)
{
LOGV("Pause");
if(iState != STATE_STARTED)
{
LOGE("Pause: Invalid state (%d)", iState);
OSCL_LEAVE(OsclErrInvalidState);
return -1;
}
return AddCmdToQueue(AI_CMD_PAUSE, aContext);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::Flush(const OsclAny* aContext)
{
LOGV("Flush");
if(iState != STATE_STARTED || iState != STATE_PAUSED)
{
LOGE("Flush: Invalid state (%d)", iState);
OSCL_LEAVE(OsclErrInvalidState);
return -1;
}
return AddCmdToQueue(AI_CMD_FLUSH, aContext);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::Reset(const OsclAny* aContext)
{
LOGV("Reset");
return AddCmdToQueue(AI_CMD_RESET, aContext);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::DiscardData(PVMFTimestamp aTimestamp, const OsclAny* aContext)
{
OSCL_UNUSED_ARG(aContext);
OSCL_UNUSED_ARG(aTimestamp);
OSCL_LEAVE(OsclErrNotSupported);
return -1;
}
PVMFCommandId AndroidAudioInput::DiscardData(const OsclAny* aContext)
{
OSCL_UNUSED_ARG(aContext);
OSCL_LEAVE(OsclErrNotSupported);
return -1;
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::Stop(const OsclAny* aContext)
{
LOGV("Stop %p", this);
if(iState != STATE_STARTED && iState != STATE_PAUSED)
{
LOGE("Stop: Invalid state (%d)", iState);
OSCL_LEAVE(OsclErrInvalidState);
return -1;
}
return AddCmdToQueue(AI_CMD_STOP, aContext);
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::ThreadLogon()
{
LOGV("ThreadLogon %p", this);
if(!iThreadLoggedOn)
{
AddToScheduler();
iThreadLoggedOn = true;
}
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::ThreadLogoff()
{
LOGV("ThreadLogoff");
if(iThreadLoggedOn)
{
RemoveFromScheduler();
iThreadLoggedOn = false;
}
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::CancelAllCommands( const OsclAny* aContext)
{
OSCL_UNUSED_ARG(aContext);
OSCL_LEAVE(OsclErrNotSupported);
return -1;
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::CancelCommand( PVMFCommandId aCmdId, const OsclAny* aContext)
{
OSCL_UNUSED_ARG(aCmdId);
OSCL_UNUSED_ARG(aContext);
OSCL_LEAVE(OsclErrNotSupported);
return -1;
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::setPeer(PvmiMediaTransfer* aPeer)
{
LOGV("setPeer");
if(iPeer && aPeer)
{
LOGE("setPeer failed: iPeer(%p) and aPeer(%p)", iPeer, aPeer);
OSCL_LEAVE(OsclErrGeneral);
return;
}
iPeer = aPeer;
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::useMemoryAllocators(OsclMemAllocator* write_alloc)
{
OSCL_UNUSED_ARG(write_alloc);
OSCL_LEAVE(OsclErrNotSupported);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::writeAsync(uint8 aFormatType, int32 aFormatIndex,
uint8* aData, uint32 aDataLen,
const PvmiMediaXferHeader& data_header_info,
OsclAny* aContext)
{
OSCL_UNUSED_ARG(aFormatType);
OSCL_UNUSED_ARG(aFormatIndex);
OSCL_UNUSED_ARG(aData);
OSCL_UNUSED_ARG(aDataLen);
OSCL_UNUSED_ARG(data_header_info);
OSCL_UNUSED_ARG(aContext);
// This is an active data source. writeAsync is not supported.
OSCL_LEAVE(OsclErrNotSupported);
return -1;
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::writeComplete(PVMFStatus aStatus, PVMFCommandId write_cmd_id,
OsclAny* aContext)
{
LOGV("writeComplete(%d, %p)", write_cmd_id, aContext);
OSCL_UNUSED_ARG(aContext);
if(aStatus != PVMFSuccess)
{
return;
}
if(iSentMediaData.empty())
{
// Error: Nothing to complete
return;
}
Oscl_Vector<AndroidAudioInputMediaData, OsclMemAllocator>::iterator it;
for(it = iSentMediaData.begin(); it != iSentMediaData.end(); it++)
{
if( it->iId == write_cmd_id )
{
iMediaBufferMemPool->deallocate(it->iData);
iSentMediaData.erase(it);
//check mem callback
AddDataEventToQueue(0);
//LOGE("writeComplete SentMediaQ out\n");
return;
}
}
// Error: unmatching ID. They should be in sequence
//LOGE("writeComplete ERROR SentMediaQ[0].iId %d write_cmd_id %d\n", iSentMediaData[0].iId, write_cmd_id);
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::readAsync(uint8* data, uint32 max_data_len,
OsclAny* aContext, int32* formats, uint16 num_formats)
{
OSCL_UNUSED_ARG(data);
OSCL_UNUSED_ARG(max_data_len);
OSCL_UNUSED_ARG(aContext);
OSCL_UNUSED_ARG(formats);
OSCL_UNUSED_ARG(num_formats);
// This is an active data source. readAsync is not supported.
OSCL_LEAVE(OsclErrNotSupported);
return -1;
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::readComplete(PVMFStatus aStatus, PVMFCommandId read_cmd_id,
int32 format_index, const PvmiMediaXferHeader& data_header_info,
OsclAny* aContext)
{
OSCL_UNUSED_ARG(aStatus);
OSCL_UNUSED_ARG(read_cmd_id);
OSCL_UNUSED_ARG(format_index);
OSCL_UNUSED_ARG(data_header_info);
OSCL_UNUSED_ARG(aContext);
// This is an active data source. readComplete is not supported.
OSCL_LEAVE(OsclErrNotSupported);
return;
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::statusUpdate(uint32 status_flags)
{
LOGV("statusUpdate");
if (status_flags != PVMI_MEDIAXFER_STATUS_WRITE)
{
LOGE("stateUpdate: unsupported status flags(%d)", status_flags);
OSCL_LEAVE(OsclErrNotSupported);
}
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::cancelCommand(PVMFCommandId aCmdId)
{
OSCL_UNUSED_ARG(aCmdId);
// This cancel command ( with a small "c" in cancel ) is for the media transfer interface.
// implementation is similar to the cancel command of the media I/O interface.
OSCL_LEAVE(OsclErrNotSupported);
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::cancelAllCommands()
{
OSCL_LEAVE(OsclErrNotSupported);
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::setObserver(PvmiConfigAndCapabilityCmdObserver* aObserver)
{
OSCL_UNUSED_ARG(aObserver);
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::getParametersSync(PvmiMIOSession session,
PvmiKeyType identifier,
PvmiKvp*& parameters,
int& num_parameter_elements,
PvmiCapabilityContext context)
{
LOGV("getParametersSync(%s)", identifier);
OSCL_UNUSED_ARG(session);
OSCL_UNUSED_ARG(context);
parameters = NULL;
num_parameter_elements = 0;
PVMFStatus status = PVMFFailure;
if( pv_mime_strcmp(identifier, OUTPUT_FORMATS_CAP_QUERY) == 0 ||
pv_mime_strcmp(identifier, OUTPUT_FORMATS_CUR_QUERY) == 0)
{
num_parameter_elements = 1;
status = AllocateKvp(parameters, (PvmiKeyType)OUTPUT_FORMATS_VALTYPE, num_parameter_elements);
if(status != PVMFSuccess)
{
LOGE("AndroidAudioInput::getParametersSync() OUTPUT_FORMATS_VALTYPE AllocateKvp failed");
return status;
}
parameters[0].value.pChar_value = (char*)PVMF_MIME_PCM16;
}
else if(pv_mime_strcmp(identifier, OUTPUT_TIMESCALE_CUR_QUERY) == 0)
{
num_parameter_elements = 1;
status = AllocateKvp(parameters, (PvmiKeyType)OUTPUT_TIMESCALE_CUR_VALUE, num_parameter_elements);
if(status != PVMFSuccess)
{
LOGE("AndroidAudioInput::getParametersSync() OUTPUT_TIMESCALE_CUR_VALUE AllocateKvp failed");
return status;
}
// XXX is it okay to hardcode this as the timescale?
parameters[0].value.uint32_value = 1000;
}
else if (pv_mime_strcmp(identifier, AUDIO_OUTPUT_SAMPLING_RATE_CUR_QUERY) == 0)
{
num_parameter_elements = 1;
status = AllocateKvp(parameters, (PvmiKeyType)AUDIO_OUTPUT_SAMPLING_RATE_CUR_QUERY, num_parameter_elements);
if (status != PVMFSuccess)
{
LOGE("AndroidAudioInput::getParametersSync() AUDIO_OUTPUT_SAMPLING_RATE_CUR_QUERY AllocateKvp failed");
return status;
}
parameters[0].value.uint32_value = iAudioSamplingRate;
}
else if (pv_mime_strcmp(identifier, AUDIO_OUTPUT_NUM_CHANNELS_CUR_QUERY) == 0)
{
num_parameter_elements = 1;
status = AllocateKvp(parameters, (PvmiKeyType)AUDIO_OUTPUT_NUM_CHANNELS_CUR_QUERY, num_parameter_elements);
if (status != PVMFSuccess)
{
LOGE("AndroidAudioInput::getParametersSync() AUDIO_OUTPUT_NUM_CHANNELS_CUR_QUERY AllocateKvp failed");
return status;
}
parameters[0].value.uint32_value = iAudioNumChannels;
}
return status;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::releaseParameters(PvmiMIOSession session,
PvmiKvp* parameters,
int num_elements)
{
OSCL_UNUSED_ARG(session);
OSCL_UNUSED_ARG(num_elements);
if(parameters)
{
iAlloc.deallocate((OsclAny*)parameters);
return PVMFSuccess;
}
else
{
LOGE("Attempt to release NULL parameters");
return PVMFFailure;
}
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::createContext(PvmiMIOSession session, PvmiCapabilityContext& context)
{
OSCL_UNUSED_ARG(session);
OSCL_UNUSED_ARG(context);
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::setContextParameters(PvmiMIOSession session,
PvmiCapabilityContext& context,
PvmiKvp* parameters, int num_parameter_elements)
{
OSCL_UNUSED_ARG(session);
OSCL_UNUSED_ARG(context);
OSCL_UNUSED_ARG(parameters);
OSCL_UNUSED_ARG(num_parameter_elements);
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::DeleteContext(PvmiMIOSession session, PvmiCapabilityContext& context)
{
OSCL_UNUSED_ARG(session);
OSCL_UNUSED_ARG(context);
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::setParametersSync(PvmiMIOSession session, PvmiKvp* parameters,
int num_elements, PvmiKvp*& ret_kvp)
{
LOGV("setParametersSync");
OSCL_UNUSED_ARG(session);
PVMFStatus status = PVMFSuccess;
ret_kvp = NULL;
for(int32 i = 0; i < num_elements; i++)
{
status = VerifyAndSetParameter(&(parameters[i]), true);
if(status != PVMFSuccess)
{
LOGE("VerifyAndSetParameter failed");
ret_kvp = &(parameters[i]);
OSCL_LEAVE(OsclErrArgument);
}
}
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::setParametersAsync(PvmiMIOSession session,
PvmiKvp* parameters,
int num_elements,
PvmiKvp*& ret_kvp,
OsclAny* context)
{
OSCL_UNUSED_ARG(session);
OSCL_UNUSED_ARG(parameters);
OSCL_UNUSED_ARG(num_elements);
OSCL_UNUSED_ARG(ret_kvp);
OSCL_UNUSED_ARG(context);
OSCL_LEAVE(OsclErrNotSupported);
return -1;
}
////////////////////////////////////////////////////////////////////////////
uint32 AndroidAudioInput::getCapabilityMetric (PvmiMIOSession session)
{
OSCL_UNUSED_ARG(session);
return 0;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::verifyParametersSync(PvmiMIOSession session,
PvmiKvp* parameters, int num_elements)
{
OSCL_UNUSED_ARG(session);
OSCL_UNUSED_ARG(parameters);
OSCL_UNUSED_ARG(num_elements);
return PVMFErrNotSupported;
}
////////////////////////////////////////////////////////////////////////////
bool AndroidAudioInput::setAudioSamplingRate(int32 iSamplingRate)
{
LOGV("AndroidAudioInput::setAudioSamplingRate( %d )", iSamplingRate);
if (iSamplingRate == 0)
{
// Setting sampling rate to zero will cause a crash
LOGE("AndroidAudioInput::setAudioSamplingRate() invalid sampling rate. Return false.");
return false;
}
iAudioSamplingRate = iSamplingRate;
LOGV("AndroidAudioInput::setAudioSamplingRate() iAudioSamplingRate %d set", iAudioSamplingRate);
return true;
}
////////////////////////////////////////////////////////////////////////////
bool AndroidAudioInput::setAudioNumChannels(int32 iNumChannels)
{
LOGV("AndroidAudioInput::setAudioNumChannels( %d )", iNumChannels);
iAudioNumChannels = iNumChannels;
LOGV("AndroidAudioInput::setAudioNumChannels() iAudioNumChannels %d set", iAudioNumChannels);
return true;
}
////////////////////////////////////////////////////////////////////////////
bool AndroidAudioInput::setAudioSource(uint32 iSource)
{
iAudioSource = iSource;
LOGV("AndroidAudioInput::setAudioSource() iAudioSource %d set", iAudioSource);
return true;
}
////////////////////////////////////////////////////////////////////////////
// Private methods
////////////////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::Run()
{
if(!iCmdQueue.empty())
{
AndroidAudioInputCmd cmd = iCmdQueue[0];
iCmdQueue.erase(iCmdQueue.begin());
switch(cmd.iType)
{
case AI_CMD_INIT:
DoRequestCompleted(cmd, DoInit());
break;
case AI_CMD_START:
DoRequestCompleted(cmd, DoStart());
break;
case AI_CMD_PAUSE:
DoRequestCompleted(cmd, DoPause());
break;
case AI_CMD_FLUSH:
DoRequestCompleted(cmd, DoFlush());
break;
case AI_CMD_RESET:
DoRequestCompleted(cmd, DoReset());
break;
case AI_CMD_STOP:
DoRequestCompleted(cmd, DoStop());
break;
case AI_DATA_WRITE_EVENT:
DoRead();
break;
case AI_CMD_QUERY_UUID:
case AI_CMD_QUERY_INTERFACE:
DoRequestCompleted(cmd, PVMFSuccess);
break;
case AI_CMD_CANCEL_ALL_COMMANDS:
case AI_CMD_CANCEL_COMMAND:
DoRequestCompleted(cmd, PVMFFailure);
break;
default:
break;
}
}
if ((iState == STATE_STARTED) && (iStartCmd.iType == AI_CMD_START)) {
// This means Audio MIO is waiting for
// first audio frame to be received
if (iFirstFrameReceived) {
// Set the clock with iFirstFrameTs
if (iAuthorClock) {
bool tmpbool = false;
iAuthorClock->SetStartTime32((uint32&)iFirstFrameTs, PVMF_MEDIA_CLOCK_MSEC, tmpbool);
}
LOGV("First frame received, Complete the start");
// First frame is received now complete Start
DoRequestCompleted(iStartCmd, PVMFSuccess);
// Set the iStartCmd type to be invalid now
iStartCmd.iType = AI_INVALID_CMD;
}
}
if(!iCmdQueue.empty())
{
// Run again if there are more things to process
RunIfNotReady();
}
if((iState == STATE_STARTED) && (iFirstFrameReceived))
{
SendMicData();
}
}
////////////////////////////////////////////////////////////////////////////
PVMFCommandId AndroidAudioInput::AddCmdToQueue(AndroidAudioInputCmdType aType,
const OsclAny* aContext, OsclAny* aData1)
{
if(aType == AI_DATA_WRITE_EVENT)
OSCL_LEAVE(OsclErrArgument);
AndroidAudioInputCmd cmd;
cmd.iType = aType;
cmd.iContext = OSCL_STATIC_CAST(OsclAny*, aContext);
cmd.iData1 = aData1;
cmd.iId = iCmdIdCounter;
++iCmdIdCounter;
iCmdQueue.push_back(cmd);
RunIfNotReady();
return cmd.iId;
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::AddDataEventToQueue(uint32 aMicroSecondsToEvent)
{
// If not added to the scheduler just return
if (!IsAdded())
return;
AndroidAudioInputCmd cmd;
cmd.iType = AI_DATA_WRITE_EVENT;
iCmdQueue.push_back(cmd);
RunIfNotReady(aMicroSecondsToEvent);
}
////////////////////////////////////////////////////////////////////////////
void AndroidAudioInput::DoRequestCompleted(const AndroidAudioInputCmd& aCmd, PVMFStatus aStatus, OsclAny* aEventData)
{
LOGV("DoRequestCompleted(%d, %d) this %p", aCmd.iId, aStatus, this);
if ((aStatus == PVMFPending) && (aCmd.iType == AI_CMD_START)) {
LOGV("Start is pending, Return here, wait for success or failure");
// Copy the command
iStartCmd = aCmd;
return;
}
PVMFCmdResp response(aCmd.iId, aCmd.iContext, aStatus, aEventData);
for(uint32 i = 0; i < iObservers.size(); i++)
iObservers[i]->RequestCompleted(response);
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::DoInit()
{
LOGV("DoInit");
//calculate time for a buffer to fill
iMicroSecondsPerDataEvent = (int32)((1000000/iAudioSamplingRate) / iAudioNumChannels);
LOGV("AndroidAudioInput::DoInit() iMicroSecondsPerDataEvent %d", iMicroSecondsPerDataEvent);
iDataEventCounter = 0;
// Create memory pool for the media data, using the maximum frame size found earlier
int32 err = 0;
OSCL_TRY(err,
if(iMediaBufferMemPool)
{
OSCL_DELETE(iMediaBufferMemPool);
iMediaBufferMemPool = NULL;
}
iMediaBufferMemPool = OSCL_NEW(OsclMemPoolFixedChunkAllocator, (4));
if(!iMediaBufferMemPool) {
LOGE("AndroidAudioInput::DoInit() unable to create memory pool. Return PVMFErrNoMemory.");
OSCL_LEAVE(OsclErrNoMemory);
}
);
OSCL_FIRST_CATCH_ANY(err, return PVMFErrNoMemory);
iState = STATE_INITIALIZED;
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::DoStart()
{
LOGV("DoStart");
// Set the clock state observer
if (iAuthorClock) {
iAuthorClock->ConstructMediaClockNotificationsInterface(iClockNotificationsInf, *this);
if (iClockNotificationsInf == NULL) {
return PVMFErrNoMemory;
}
iClockNotificationsInf->SetClockStateObserver(*this);
}
iAudioThreadStartLock->lock();
iAudioThreadStarted = false;
OsclThread AudioInput_Thread;
OsclProcStatus::eOsclProcError ret = AudioInput_Thread.Create(
(TOsclThreadFuncPtr)start_audin_thread_func, 0,
(TOsclThreadFuncArg)this, Start_on_creation);
if ( OsclProcStatus::SUCCESS_ERROR != ret)
{ // thread creation failed
LOGE("Failed to create thread (%d)", ret);
iAudioThreadStartLock->unlock();
return PVMFFailure;
}
// wait for thread to set up AudioRecord
while (!iAudioThreadStarted)
iAudioThreadStartCV->wait(*iAudioThreadStartLock);
status_t startResult = iAudioThreadStartResult;
iAudioThreadStartLock->unlock();
if (startResult != NO_ERROR)
return PVMFFailure; // thread failed to set up AudioRecord
iState = STATE_STARTED;
AddDataEventToQueue(0);
// Hold back onto Start until the first audio frame is received by
// the audio thread.
// We want to hold back the start until first audio frame since
// recording time origin will start with the first audio sample.
// First audio sample will serve as the reference for audio and video
// frames timestamps in msecs.
if (!iFirstFrameReceived) {
LOGV("First Frame not received, hold back the start");
return PVMFPending;
}
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
int AndroidAudioInput::start_audin_thread_func(TOsclThreadFuncArg arg)
{
prctl(PR_SET_NAME, (unsigned long) "audio in", 0, 0, 0);
sp<AndroidAudioInput> obj = (AndroidAudioInput *)arg;
return obj->audin_thread_func();
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::DoPause()
{
LOGV("DoPause");
iExitAudioThread = true;
iState = STATE_PAUSED;
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::DoReset()
{
LOGD("DoReset: E");
// Remove and destroy the clock state observer
RemoveDestroyClockStateObs();
iExitAudioThread = true;
iDataEventCounter = 0;
iFirstFrameReceived = false;
iFirstFrameTs = 0;
if(iAudioThreadStarted ){
iAudioThreadSem->Signal();
iAudioThreadTermSem->Wait();
iAudioThreadStarted = false;
}
while(!iCmdQueue.empty())
{
AndroidAudioInputCmd cmd = iCmdQueue[0];
iCmdQueue.erase(iCmdQueue.begin());
}
Cancel();
while(!iOSSRequestQueue.empty())
{
uint8* data = iOSSRequestQueue[0];
iMediaBufferMemPool->deallocate(data);
iOSSRequestQueue.erase(&iOSSRequestQueue[0]);
}
while(!iWriteResponseQueue.empty())
{
MicData micdata = iWriteResponseQueue[0];
iMediaBufferMemPool->deallocate(micdata.iData);
iWriteResponseQueue.erase(&iWriteResponseQueue[0]);
}
iState = STATE_IDLE;
LOGD("DoReset: X");
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::DoFlush()
{
// This method should stop capturing media data but continue to send captured
// media data that is already in buffer and then go to stopped state.
// However, in this case of file input we do not have such a buffer for
// captured data, so this behaves the same way as stop.
LOGV("DoFlush");
return DoStop();
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::DoStop()
{
LOGD("DoStop: E");
// Remove and destroy the clock state observer
RemoveDestroyClockStateObs();
iExitAudioThread = true;
iDataEventCounter = 0;
iFirstFrameReceived = false;
iFirstFrameTs = 0;
iState = STATE_STOPPED;
if(iAudioThreadStarted ){
iAudioThreadSem->Signal();
iAudioThreadTermSem->Wait();
iAudioThreadStarted = false;
}
LOGD("DoStop: X");
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::DoRead()
{
LOGV("DoRead");
// Just copy from PVMFFileInputNode::HandleEventPortActivity. The only difference
// is that data buffer is allocated by calling iMediaBufferMemPool->allocate(bytesToRead)
// and there's no need to wrap it in a PVMFSharedMediaDataPtr. Also, you'll need to
// keep track of the data pointer and the write command id received from peer->writeAsync
// and put it in the iSentMediaData queue
if(iState != STATE_STARTED)
{
LOGV("not started");
return PVMFSuccess;
}
if (NULL == iPeer)
{
LOGV("iPeer Null");
return PVMFSuccess;
}
uint32 timeStamp = 0;
uint32 writeAsyncID = 0;
++iDataEventCounter;
iOSSRequestQueueLock.Lock();
//allocate mem for the audio capture thread
// Create new media data buffer
int32 error = 0;
uint8* data = NULL;
OSCL_TRY(error,
data = (uint8*)iMediaBufferMemPool->allocate(kBufferSize);
);
if (error != OsclErrNone || !data )
{
LOGV("no data available");
iOSSRequestQueueLock.Unlock();
//TBD wait for mem callback
return PVMFSuccess;
}
else
{
iOSSRequestQueue.push_back(data);
iOSSRequestQueueLock.Unlock();
iAudioThreadSem->Signal();
}
// Queue the next data event
AddDataEventToQueue(iMicroSecondsPerDataEvent);
return PVMFSuccess;
}
void AndroidAudioInput::RampVolume(
int32 timeInFrames,
int32 kAutoRampDurationFrames,
void *_data,
size_t numBytes) const {
int16 *data = (int16 *)_data;
// Apply the ramp to the entire buffer or to the end of the ramp duration
// whichever comes first.
int32 kStopFrame = timeInFrames + numBytes / sizeof(int16);
if (kStopFrame > kAutoRampDurationFrames) {
kStopFrame = kAutoRampDurationFrames;
}
const int32 kShift = 14;
int32 fixedMultiplier =
(timeInFrames << kShift) / kAutoRampDurationFrames;
if (iAudioNumChannels == 1) {
while (timeInFrames < kStopFrame) {
data[0] = (data[0] * fixedMultiplier) >> kShift;
++data;
++timeInFrames;
if ((timeInFrames & 3) == 0) {
// Update the multiplier every 4 frames.
fixedMultiplier =
(timeInFrames << kShift) / kAutoRampDurationFrames;
}
}
} else {
LOG_ALWAYS_FATAL_IF(
iAudioNumChannels != 2,
"We only support mono or stereo audio data here.");
// Stereo
while (timeInFrames < kStopFrame) {
data[0] = (data[0] * fixedMultiplier) >> kShift;
data[1] = (data[1] * fixedMultiplier) >> kShift;
data += 2;
++timeInFrames;
if ((timeInFrames & 3) == 0) {
// Update the multiplier every 4 frames.
fixedMultiplier =
(timeInFrames << kShift) / kAutoRampDurationFrames;
}
}
}
}
////////////////////////////////////////////////////////////////////////////
int AndroidAudioInput::audin_thread_func() {
// setup audio record session
iAudioThreadStartLock->lock();
// set microphone input flags to turn on AGC and noise suppression
uint32_t flags = AudioRecord::RECORD_AGC_ENABLE |
AudioRecord::RECORD_NS_ENABLE |
AudioRecord::RECORD_IIR_ENABLE;
LOGV("create AudioRecord %p", this);
AudioRecord
* record = new AudioRecord(
iAudioSource, iAudioSamplingRate,
android::AudioSystem::PCM_16_BIT,
(iAudioNumChannels > 1) ? AudioSystem::CHANNEL_IN_STEREO : AudioSystem::CHANNEL_IN_MONO,
4*kBufferSize/iAudioNumChannels/sizeof(int16), flags);
LOGV("AudioRecord created %p, this %p", record, this);
status_t res = record->initCheck();
if (res == NO_ERROR)
res = record->start();
iAudioThreadStartResult = res;
iAudioThreadStarted = true;
iAudioThreadStartCV->signal();
iAudioThreadStartLock->unlock();
if (res == NO_ERROR) {
// We are going to ramp up the volume from 0 to full at the
// start of recording.
int64_t numFramesRecorded = 0;
const int32 kAutoRampStartFrames =
AUTO_RAMP_START_MS * iAudioSamplingRate / 1000;
const int32 kAutoRampDurationFrames =
AUTO_RAMP_DURATION_MS * iAudioSamplingRate / 1000;
while (!iExitAudioThread) {
iOSSRequestQueueLock.Lock();
if (iOSSRequestQueue.empty()) {
iOSSRequestQueueLock.Unlock();
iAudioThreadSem->Wait();
continue;
}
uint8* data = iOSSRequestQueue[0];
iOSSRequestQueue.erase(&iOSSRequestQueue[0]);
iOSSRequestQueueLock.Unlock();
uint32 inputFramesLost = record->getInputFramesLost();
if (inputFramesLost > 0) {
uint32 dataDurationLost = inputFramesLost * 1000 / iAudioSamplingRate;
LOGW("Notifying %d observers of %u ms of lost audio", iObservers.size(), dataDurationLost);
for(uint32 i = 0; i < iObservers.size(); i++) {
int32 leavecode = 0;
PVMFDurationInfoMessage* eventMsg = NULL;
OSCL_TRY(leavecode, eventMsg = OSCL_NEW(PVMFDurationInfoMessage, (dataDurationLost)));
iObservers[i]->ReportInfoEvent(PVMFInfoOverflow, OSCL_STATIC_CAST(PVInterface*, eventMsg));
}
}
int numOfBytes = record->read(data, kBufferSize);
if (numOfBytes <= 0) {
// FIXME:
// When numOfBytes is not greater than 0, instead of terminating the audio
// recording thread immediately, wait for next incoming audio frame or stop/reset
// command to terminate the thread. Lets log the case to see whether the deadlock
// root cause is here. To resolve the problem, change the break to continue.
LOGW("numOfBytes (%d) <= 0.", numOfBytes);
break;
}
if (iFirstFrameReceived == false) {
iFirstFrameReceived = true;
// Get the AudioRecord latency and
// get the system clock at this point
// The difference in 2 will give the actual time
// of first audio capture.
uint32 systime = (uint32) (systemTime() / 1000000L);
uint32 recordLatency = 0;
// Get the latency now
size_t kernelBufferSize = 0;
status_t ret = AudioSystem::getInputBufferSize(iAudioSamplingRate,
AudioSystem::PCM_16_BIT,
iAudioNumChannels, &kernelBufferSize);
if (ret == NO_ERROR) {
uint32 readBufferFrames = kBufferSize/2/iAudioNumChannels;
uint32 kernelFrames = kernelBufferSize/2/iAudioNumChannels;
recordLatency = ((readBufferFrames - 1)/kernelFrames + 1) * (kernelFrames*1000)/iAudioSamplingRate;
} else {
LOGE("AudioSystem::getInputBufferSize returned error");
}
iFirstFrameTs = systime - recordLatency;
LOGV("First Audio Frame received systime %d, recordLatency %d, iFirstFrameTs %d", systime, recordLatency, iFirstFrameTs);
}
if (numFramesRecorded < kAutoRampStartFrames) {
// Start with silence...
memset(data, 0, numOfBytes);
} else {
// Then ramp up the volume...
int64_t delta = numFramesRecorded - kAutoRampStartFrames;
if (delta < kAutoRampDurationFrames) {
RampVolume(
delta, kAutoRampDurationFrames, data, numOfBytes);
}
}
if (iTrackMaxAmplitude) {
int16 *p = (int16*) data;
for (int i = numOfBytes >> 1; i > 0; --i) {
int16 v = *p++;
if (v < 0) {
v = -v;
}
if (v > iMaxAmplitude) {
iMaxAmplitude = v;
}
}
}
int32 dataFrames = numOfBytes / sizeof(int16) / iAudioNumChannels;
numFramesRecorded += dataFrames;
int dataDuration = dataFrames * 1000 / iAudioSamplingRate; //ms
MicData micdata(data, numOfBytes, iTimeStamp,
dataDuration);
iWriteResponseQueueLock.Lock();
iWriteResponseQueue.push_back(micdata);
iWriteResponseQueueLock.Unlock();
iTimeStamp += dataDuration;
// Queue the next data event
OsclAny* P = NULL;
iWriteCompleteAO->ReceiveEvent(P);
}
LOGV("record->stop %p, this %p", record, this);
record->stop();
}
LOGV("delete record %p, this %p", record, this);
delete record;
iAudioThreadTermSem->Signal();
return 0;
}
void AndroidAudioInput::SendMicData(void)
{
//LOGE("SendMicData in\n");
//ASSUMPTION: the output queue is always available. no wait
if(iState != STATE_STARTED)
{
LOGV("not started");
return;
}
if (NULL == iPeer)
{
LOGV("iPeer Null");
return;
}
iWriteResponseQueueLock.Lock();
//LOGE("SendMicData QSize %d\n", iWriteResponseQueue.size());
if(iWriteResponseQueue.empty())
{
iWriteResponseQueueLock.Unlock();
return;
}
MicData &data = iWriteResponseQueue[0];
// send data to Peer & store the id
PvmiMediaXferHeader data_hdr;
data_hdr.seq_num=iDataEventCounter-1;
data_hdr.timestamp = data.iTimestamp;
data_hdr.flags=0;
data_hdr.duration = data.iDuration;
data_hdr.stream_id=0;
int32 err = 0;
PVMFCommandId writeAsyncID = 0;
OSCL_TRY(err,
writeAsyncID = iPeer->writeAsync(PVMI_MEDIAXFER_FMT_TYPE_DATA, 0, data.iData, data.iDataLen, data_hdr);
);
OSCL_FIRST_CATCH_ANY(err,
// send data failed, data sent out in wrong state.
LOGE("send data failed");
iWriteResponseQueueLock.Unlock();
return;
);
// Save the id and data pointer on iSentMediaData queue for writeComplete call
AndroidAudioInputMediaData sentData;
sentData.iId = writeAsyncID;
sentData.iData = data.iData;
iWriteResponseQueue.erase(&iWriteResponseQueue[0]);
iWriteResponseQueueLock.Unlock();
iSentMediaData.push_back(sentData);
//LOGE("SendMicData out SentQ size %d\n", iSentMediaData.size());
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::AllocateKvp(PvmiKvp*& aKvp, PvmiKeyType aKey, int32 aNumParams)
{
uint8* buf = NULL;
uint32 keyLen = oscl_strlen(aKey) + 1;
int32 err = 0;
OSCL_TRY(err,
buf = (uint8*)iAlloc.allocate(aNumParams * (sizeof(PvmiKvp) + keyLen));
if(!buf) {
LOGE("Failed to allocate memory for Kvp");
OSCL_LEAVE(OsclErrNoMemory);
}
);
OSCL_FIRST_CATCH_ANY(err, LOGE("allocation error"); return PVMFErrNoMemory;);
int32 i = 0;
PvmiKvp* curKvp = aKvp = new (buf) PvmiKvp;
buf += sizeof(PvmiKvp);
for(i = 1; i < aNumParams; i++)
{
curKvp += i;
curKvp = new (buf) PvmiKvp;
buf += sizeof(PvmiKvp);
}
for(i = 0; i < aNumParams; i++)
{
aKvp[i].key = (char*)buf;
oscl_strncpy(aKvp[i].key, aKey, keyLen);
buf += keyLen;
}
return PVMFSuccess;
}
////////////////////////////////////////////////////////////////////////////
PVMFStatus AndroidAudioInput::VerifyAndSetParameter(PvmiKvp* aKvp, bool aSetParam)
{
if(!aKvp)
{
LOGE("aKvp is a NULL pointer");
return PVMFFailure;
}
if(pv_mime_strcmp(aKvp->key, OUTPUT_FORMATS_VALTYPE) == 0)
{
if(pv_mime_strcmp(aKvp->value.pChar_value, PVMF_MIME_PCM16) == 0)
{
return PVMFSuccess;
}
else
{
LOGE("unsupported format (%s) for key %s", aKvp->value.pChar_value, aKvp->key);
return PVMFFailure;
}
}
else if (pv_mime_strcmp(aKvp->key, PVMF_AUTHORING_CLOCK_KEY) == 0)
{
LOGV("AndroidAudioInput::VerifyAndSetParameter() PVMF_AUTHORING_CLOCK_KEY value %p", aKvp->value.key_specific_value);
if( (NULL == aKvp->value.key_specific_value) && ( iAuthorClock ) )
{
RemoveDestroyClockStateObs();
}
iAuthorClock = (PVMFMediaClock*) aKvp->value.key_specific_value;
return PVMFSuccess;
}
LOGE("unsupported parameter: %s", aKvp->key);
return PVMFFailure;
}
int AndroidAudioInput::maxAmplitude()
{
// first call to this function activates tracking
if (!iTrackMaxAmplitude) {
iTrackMaxAmplitude = true;
}
int result = iMaxAmplitude;
iMaxAmplitude = 0;
return result;
}
void AndroidAudioInput::NotificationsInterfaceDestroyed()
{
iClockNotificationsInf = NULL;
}
void AndroidAudioInput::ClockStateUpdated()
{
PVMFMediaClock::PVMFMediaClockState iClockState = iAuthorClock->GetState();
uint32 currentTime = 0;
bool tmpbool = false;
iAuthorClock->GetCurrentTime32(currentTime, tmpbool, PVMF_MEDIA_CLOCK_MSEC);
LOGV("ClockStateUpdated State %d Current clock value %d", iClockState, currentTime);
}
void AndroidAudioInput::RemoveDestroyClockStateObs()
{
if (iAuthorClock != NULL) {
if (iClockNotificationsInf != NULL) {
iClockNotificationsInf->RemoveClockStateObserver(*this);
iAuthorClock->DestroyMediaClockNotificationsInterface(iClockNotificationsInf);
iClockNotificationsInf = NULL;
}
}
}