blob: 165d683d52d8c39e2a3982d3a10de12be30d2ac0 [file] [log] [blame] [edit]
/*
**
** Copyright 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 "AudioMIO"
#include <utils/Log.h>
#include "android_audio_mio.h"
#include "pvlogger.h"
#include "pv_mime_string_utils.h"
#include "oscl_snprintf.h"
#include "oscl_mem.h"
#include "oscl_dll.h"
#include "oscl_mem.h"
#include <sys/prctl.h>
#include <utils/Timers.h>
#include <sys/resource.h>
#include <limits.h>
#include <utils/threads.h>
#include <media/AudioTrack.h>
using namespace android;
// depth of buffer/command queues in MIO
static const int kCommandQueueDepth = 10;
// Define entry point for this DLL
OSCL_DLL_ENTRY_POINT_DEFAULT()
// Audio Media IO component base class implementation
OSCL_EXPORT_REF AndroidAudioMIO::AndroidAudioMIO(const char* name)
: OsclTimerObject(OsclActiveObject::EPriorityNominal, name),
iWriteCompleteAO(NULL)
{
initData();
}
OSCL_EXPORT_REF AndroidAudioMIO::~AndroidAudioMIO()
{
LOGV("destructor");
Cleanup();
}
void AndroidAudioMIO::initData()
{
LOGV("initData in");
ResetData();
iCommandCounter = 0;
iLogger = NULL;
iCommandResponseQueue.reserve(kCommandQueueDepth);
iWriteResponseQueueLock.Create();
iWriteResponseQueue.reserve(kCommandQueueDepth);
iObserver = NULL;
iLogger = NULL;
iPeer = NULL;
iState = STATE_MIO_IDLE;
iWriteBusy = false;
iFlushPending = false;
iDataQueued = 0;
LOGV("initData out");
}
void AndroidAudioMIO::ResetData()
//reset all data from this session.
{
LOGV("ResetData in");
// reset all the received media parameters.
iAudioFormat = PVMF_MIME_FORMAT_UNKNOWN;
iAudioNumChannelsValid = false;
iAudioSamplingRateValid = false;
iAudioThreadCreatedAndMIOConfigured = false;
LOGV("ResetData out");
}
//cleanup all allocated memory and release resources.
void AndroidAudioMIO::Cleanup()
{
LOGV("Cleanup in");
while (!iCommandResponseQueue.empty()) {
if (iObserver) {
iObserver->RequestCompleted(PVMFCmdResp(iCommandResponseQueue[0].iCmdId,
iCommandResponseQueue[0].iContext, iCommandResponseQueue[0].iStatus));
}
iCommandResponseQueue.erase(&iCommandResponseQueue[0]);
}
// return empty buffers
returnAllBuffers();
// delete the request active object
if (iWriteCompleteAO) {
OSCL_DELETE(iWriteCompleteAO);
iWriteCompleteAO = NULL;
}
iWriteResponseQueueLock.Close();
LOGV("Cleanup out");
}
PVMFStatus AndroidAudioMIO::connect(PvmiMIOSession& aSession, PvmiMIOObserver* aObserver)
{
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::connect() called"));
// currently supports only one session
if (iObserver) return PVMFFailure;
iObserver = aObserver;
return PVMFSuccess;
}
PVMFStatus AndroidAudioMIO::disconnect(PvmiMIOSession aSession)
{
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::disconnect() called"));
iObserver = NULL;
return PVMFSuccess;
}
PvmiMediaTransfer* AndroidAudioMIO::createMediaTransfer(PvmiMIOSession& aSession,
PvmiKvp* read_formats, int32 read_flags,
PvmiKvp* write_formats, int32 write_flags)
{
// create the request active object
// such when audio output thread is done with a buffer
// it can put the buffer on the write response queue
// and schedule this MIO to run, to return the buffer
// to the engine
iWriteCompleteAO = OSCL_NEW(AndroidAudioOutputThreadSafeCallbackAO,(this, 5));
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::createMediaTransfer() called"));
return (PvmiMediaTransfer*)this;
}
PVMFCommandId AndroidAudioMIO::QueueCmdResponse(PVMFStatus status, const OsclAny* aContext)
{
PVMFCommandId cmdId = iCommandCounter++;
CommandResponse resp(status, cmdId, aContext);
iCommandResponseQueue.push_back(resp);
// cancel any timer delay so the command response will happen ASAP.
if (IsBusy()) Cancel();
RunIfNotReady();
return cmdId;
}
// return any held buffers to the engine
void AndroidAudioMIO::ProcessWriteResponseQueue()
{
//LOGV("ProcessWriteResponseQueue in [Response Q size %d]",iWriteResponseQueue.size());
PVMFStatus status = 0;
PVMFCommandId cmdId = 0;
const OsclAny* context = 0;
iWriteResponseQueueLock.Lock();
while (!iWriteResponseQueue.empty()) {
status = iWriteResponseQueue[0].iStatus;
cmdId = iWriteResponseQueue[0].iCmdId;
context = (OsclAny*)iWriteResponseQueue[0].iContext;
iWriteResponseQueue.erase(&iWriteResponseQueue[0]);
iWriteResponseQueueLock.Unlock();
if (iPeer) {
LOGV("Return buffer(%d) status(%d) context(%p)", cmdId,status,context);
iPeer->writeComplete(status, cmdId, (OsclAny*)context);
}
iWriteResponseQueueLock.Lock();
}
iWriteResponseQueueLock.Unlock();
//LOGV("ProcessWriteResponseQueue out");
}
PVMFCommandId AndroidAudioMIO::QueryUUID(const PvmfMimeString& aMimeType,
Oscl_Vector<PVUuid, OsclMemAllocator>& aUuids,
bool aExactUuidsOnly, const OsclAny* aContext)
{
LOGV("QueryUUID");
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::QueryUUID() called"));
return QueueCmdResponse(PVMFFailure, aContext);
}
PVMFCommandId AndroidAudioMIO::QueryInterface(const PVUuid& aUuid, PVInterface*& aInterfacePtr, const OsclAny* aContext)
{
LOGV("QueryInterface");
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::QueryInterface() called"));
PVMFStatus status=PVMFFailure;
if (aUuid == PVMI_CAPABILITY_AND_CONFIG_PVUUID) {
PvmiCapabilityAndConfig* myInterface = OSCL_STATIC_CAST(PvmiCapabilityAndConfig*, this);
aInterfacePtr = OSCL_STATIC_CAST(PVInterface*, myInterface);
status = PVMFSuccess;
}
return QueueCmdResponse(status, aContext);
}
PVMFCommandId AndroidAudioMIO::Init(const OsclAny* aContext)
{
LOGV("Init (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::Init() called"));
iState=STATE_MIO_INITIALIZED;
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::Reset(const OsclAny* aContext)
{
LOGV("Reset (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::Reset() called"));
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::Start(const OsclAny* aContext)
{
// Start is NO-OP
LOGV("Start (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::Start() called"));
iState = STATE_MIO_STARTED;
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::Pause(const OsclAny* aContext)
{
LOGV("Pause (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::Pause() called"));
iState = STATE_MIO_PAUSED;
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::Flush(const OsclAny* aContext)
{
LOGV("Flush (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::Flush() called"));
iState = STATE_MIO_INITIALIZED;
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::DiscardData(const OsclAny* aContext)
{
LOGV("DiscardData (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::DiscardData() called"));
return DiscardData(UINT_MAX, aContext);
}
PVMFCommandId AndroidAudioMIO::DiscardData(PVMFTimestamp aTimestamp, const OsclAny* aContext)
{
LOGV("DiscardData (%u, %p)", aTimestamp, aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::DiscardData() called"));
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::Stop(const OsclAny* aContext)
{
LOGV("Stop (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::Stop() called"));
iState = STATE_MIO_INITIALIZED;
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::CancelAllCommands(const OsclAny* aContext)
{
LOGV("CancelAllCommands (%p)", aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::CancelAllCommands() called"));
//commands are executed immediately upon being received, so
//it isn't really possible to cancel them.
return QueueCmdResponse(PVMFSuccess, aContext);
}
PVMFCommandId AndroidAudioMIO::CancelCommand(PVMFCommandId aCmdId, const OsclAny* aContext)
{
LOGV("CancelCommand (%u, %p)", aCmdId, aContext);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::CancelCommand() called"));
//see if the response is still queued.
PVMFStatus status = PVMFFailure;
for (uint32 i = 0; i < iCommandResponseQueue.size(); i++) {
if (iCommandResponseQueue[i].iCmdId == aCmdId) {
status = PVMFSuccess;
break;
}
}
return QueueCmdResponse(PVMFSuccess, aContext);
}
void AndroidAudioMIO::ThreadLogon()
{
LOGV("ThreadLogon() called ");
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::ThreadLogon() called"));
if (iState == STATE_MIO_IDLE) {
iLogger = PVLogger::GetLoggerObject("AndroidAudioMIO\n");
AddToScheduler();
iState=STATE_MIO_LOGGED_ON;
}
}
void AndroidAudioMIO::ThreadLogoff()
{
LOGV("ThreadLogoff() called");
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::ThreadLogoff() called"));
if (iState!=STATE_MIO_IDLE) {
RemoveFromScheduler();
iLogger = NULL;
iState = STATE_MIO_IDLE;
ResetData();
}
}
void AndroidAudioMIO::setPeer(PvmiMediaTransfer* aPeer)
{
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE, (0,"AndroidAudioMIO::setPeer() called"));
// Set the observer
iPeer = aPeer;
}
//This routine will determine whether data can be accepted in a writeAsync
//call and if not, will return true;
bool AndroidAudioMIO::CheckWriteBusy(uint32 aSeqNum)
{
// FIXME: this line screws up video output - why?
// return (iOSSRequestQueue.size() >= 5);
return false;
}
PVMFCommandId AndroidAudioMIO::writeAsync(uint8 aFormatType, int32 aFormatIndex, uint8* aData, uint32 aDataLen,
const PvmiMediaXferHeader& data_header_info, OsclAny* aContext)
{
// Do a leave if MIO is not configured except when it is an EOS
if (!iAudioThreadCreatedAndMIOConfigured
&&
!((PVMI_MEDIAXFER_FMT_TYPE_NOTIFICATION == aFormatType)
&& (PVMI_MEDIAXFER_FMT_INDEX_END_OF_STREAM == aFormatIndex)))
{
LOGE("ERROR :: data is pumped in before mio configured");
OSCL_LEAVE(OsclErrInvalidState);
return -1;
}
uint32 aSeqNum = data_header_info.seq_num;
PVMFTimestamp aTimestamp = data_header_info.timestamp;
uint32 flags = data_header_info.flags;
bool bWriteComplete = true;
//LOGV("writeAsync() called seqnum %d ts %d flags %d context %d formattype %d formatindex %d",aSeqNum, aTimestamp, flags,aContext,aFormatType,aFormatIndex);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE,
(0,"AndroidAudioMIO::writeAsync() seqnum %d ts %d flags %d context %d",
aSeqNum, aTimestamp, flags,aContext));
PVMFStatus status=PVMFFailure;
switch(aFormatType) {
case PVMI_MEDIAXFER_FMT_TYPE_COMMAND :
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE,
(0,"AndroidAudioMIO::writeAsync() called with Command info."));
//ignore
LOGV("PVMI_MEDIAXFER_FMT_TYPE_COMMAND");
status = PVMFSuccess;
break;
case PVMI_MEDIAXFER_FMT_TYPE_NOTIFICATION :
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE,
(0,"AndroidAudioMIO::writeAsync() called with Notification info."));
switch(aFormatIndex) {
case PVMI_MEDIAXFER_FMT_INDEX_END_OF_STREAM:
LOGV("PVMI_MEDIAXFER_FMT_INDEX_END_OF_STREAM");
bWriteComplete = false; //force an empty buffer through the audio thread
break;
default:
break;
}
//ignore
status = PVMFSuccess;
break;
case PVMI_MEDIAXFER_FMT_TYPE_DATA :
switch(aFormatIndex) {
case PVMI_MEDIAXFER_FMT_INDEX_FMT_SPECIFIC_INFO:
status = PVMFSuccess;
LOGV("PVMI_MEDIAXFER_FMT_INDEX_FMT_SPECIFIC_INFO");
break;
case PVMI_MEDIAXFER_FMT_INDEX_DATA:
LOGV("PVMI_MEDIAXFER_FMT_INDEX_DATA");
//data contains the media bitstream.
//Check whether we can accept data now and leave if we can't.
if (CheckWriteBusy(aSeqNum)) {
PVLOGGER_LOGMSG(PVLOGMSG_INST_REL, iLogger, PVLOGMSG_ERR,
(0,"AndroidAudioMIO::writeAsync: Entering busy state"));
//schedule an event to re-start the data flow.
iWriteBusy = true;
bWriteComplete = false;
// Rich:: commenting in this line.
// Need some timeout here just in case.
// I have no evidence of any problems.
RunIfNotReady(10 * 1000);
OSCL_LEAVE(OsclErrBusy);
} else {
if (aDataLen > 0) {
// this buffer will be queued by the audio output thread to process
// this buffer cannot be write completed until
// it has been processed by the audio output thread
bWriteComplete = false;
} else {
LOGE("writeAsync() called with aDataLen==0");
PVLOGGER_LOGMSG(PVLOGMSG_INST_HLDBG, iLogger, PVLOGMSG_INFO,
(0,"AndroidAudioMIO::writeAsync() called aDataLen==0."));
}
status = PVMFSuccess;
}
break;
default:
LOGE("Error unrecognized format index =%u", aFormatIndex);
PVLOGGER_LOGMSG(PVLOGMSG_INST_REL, iLogger, PVLOGMSG_ERR,
(0,"AndroidAudioMIO::writeAsync: Error - unrecognized format index"));
status = PVMFFailure;
break;
}
break;
default:
LOGE("Error unrecognized format type =%u", aFormatType);
PVLOGGER_LOGMSG(PVLOGMSG_INST_REL, iLogger, PVLOGMSG_ERR,
(0,"AndroidAudioMIO::writeAsync: Error - unrecognized format type"));
status = PVMFFailure;
break;
}
//Schedule asynchronous response
PVMFCommandId cmdid=iCommandCounter++;
if (bWriteComplete) {
LOGV("write complete (%d)", cmdid);
WriteResponse resp(status, cmdid, aContext, aTimestamp);
iWriteResponseQueueLock.Lock();
iWriteResponseQueue.push_back(resp);
iWriteResponseQueueLock.Unlock();
RunIfNotReady();
} else if (!iWriteBusy) {
writeAudioBuffer(aData, aDataLen, cmdid, aContext, aTimestamp);
}
LOGV("data queued = %u", iDataQueued);
return cmdid;
}
PVMFStatus AndroidAudioMIO::getParametersSync(PvmiMIOSession aSession, PvmiKeyType aIdentifier,
PvmiKvp*& aParameters, int& num_parameter_elements,
PvmiCapabilityContext aContext)
{
LOGV("getParametersSync in %s",aIdentifier);
OSCL_UNUSED_ARG(aSession);
OSCL_UNUSED_ARG(aContext);
aParameters=NULL;
num_parameter_elements=0;
PVMFStatus status = PVMFFailure;
if(pv_mime_strcmp(aIdentifier, INPUT_FORMATS_CAP_QUERY) == 0) {
//This is a query for the list of supported formats.
aParameters=(PvmiKvp*)oscl_malloc(2 * sizeof(PvmiKvp));
if (aParameters) {
aParameters[num_parameter_elements++].value.pChar_value = (char*)PVMF_MIME_PCM16;
aParameters[num_parameter_elements++].value.pChar_value = (char*)PVMF_MIME_PCM8;
status = PVMFSuccess;
}
else{
status = PVMFErrNoMemory;
}
}
//other queries are not currently supported.
//unrecognized key.
LOGV("getParametersSync out status %d",status);
return status;
}
PVMFStatus AndroidAudioMIO::releaseParameters(PvmiMIOSession aSession, PvmiKvp* aParameters, int num_elements)
{
LOGV("releaseParameters in");
OSCL_UNUSED_ARG(aSession);
OSCL_UNUSED_ARG(num_elements);
PVMFStatus status = PVMFFailure;
//release parameters that were allocated by this component.
if (aParameters) {
oscl_free(aParameters);
status = PVMFSuccess;
}
LOGV("releaseParameters out status %d",status);
return status;
}
PVMFStatus AndroidAudioMIO::verifyParametersSync(PvmiMIOSession aSession, PvmiKvp* aParameters, int num_elements)
{
OSCL_UNUSED_ARG(aSession);
//LOGV("verifyParametersSync in");
// Go through each parameter
for (int32 i=0; i<num_elements; i++) {
char* compstr = NULL;
pv_mime_string_extract_type(0, aParameters[i].key, compstr);
if (pv_mime_strcmp(compstr, _STRLIT_CHAR("x-pvmf/media/format-type")) == 0) {
if ((pv_mime_strcmp(aParameters[i].value.pChar_value, PVMF_MIME_PCM8) == 0) ||
(pv_mime_strcmp(aParameters[i].value.pChar_value, PVMF_MIME_PCM16) == 0)
) {
return PVMFSuccess;
}
else {
return PVMFErrNotSupported;
}
}
}
return PVMFSuccess;
//LOGV("verifyParametersSync out");
}
void AndroidAudioMIO::setParametersSync(PvmiMIOSession aSession, PvmiKvp* aParameters,
int num_elements, PvmiKvp * & aRet_kvp)
{
OSCL_UNUSED_ARG(aSession);
//LOGV("setParametersSync in");
aRet_kvp = NULL;
for (int32 i=0;i<num_elements;i++) {
//Check against known audio parameter keys...
if (pv_mime_strcmp(aParameters[i].key, MOUT_AUDIO_FORMAT_KEY) == 0) {
LOGV("Audio format: %s", aParameters[i].value.pChar_value);
iAudioFormat=aParameters[i].value.pChar_value;
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE,
(0,"AndroidAudioOutput::setParametersSync() Audio Format Key, Value %s",aParameters[i].value.pChar_value));
} else if (pv_mime_strcmp(aParameters[i].key, MOUT_AUDIO_SAMPLING_RATE_KEY) == 0) {
iAudioSamplingRate=(int32)aParameters[i].value.uint32_value;
iAudioSamplingRateValid=true;
// LOGD("iAudioSamplingRate=%d", iAudioSamplingRate);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE,
(0,"AndroidAudioMIO::setParametersSync() Audio Sampling Rate Key, Value %d",iAudioSamplingRate));
} else if (pv_mime_strcmp(aParameters[i].key, MOUT_AUDIO_NUM_CHANNELS_KEY) == 0) {
iAudioNumChannels=(int32)aParameters[i].value.uint32_value;
iAudioNumChannelsValid=true;
// LOGD("iAudioNumChannels=%d", iAudioNumChannels);
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE,
(0,"AndroidAudioMIO::setParametersSync() Audio Num Channels Key, Value %d",iAudioNumChannels));
} else {
//if we get here the key is unrecognized.
PVLOGGER_LOGMSG(PVLOGMSG_INST_LLDBG, iLogger, PVLOGMSG_STACK_TRACE,
(0,"AndroidAudioMIO::setParametersSync() Error, unrecognized key %s",aParameters[i].key));
//set the return value to indicate the unrecognized key
//and return.
aRet_kvp = &aParameters[i];
return;
}
}
//LOGV("setParametersSync out");
}
//
// Private section
//
void AndroidAudioMIO::Run()
{
while (!iCommandResponseQueue.empty()) {
if (iObserver) {
iObserver->RequestCompleted(PVMFCmdResp(iCommandResponseQueue[0].iCmdId,
iCommandResponseQueue[0].iContext,
iCommandResponseQueue[0].iStatus));
}
iCommandResponseQueue.erase(&iCommandResponseQueue[0]);
}
//send async write completion
ProcessWriteResponseQueue();
//Re-start the data transfer if needed.
if (iWriteBusy) {
iWriteBusy = false;
iPeer->statusUpdate(PVMI_MEDIAXFER_STATUS_WRITE);
}
}
// send response to MIO
void AndroidAudioMIO::sendResponse(PVMFCommandId cmdid, const OsclAny* context, PVMFTimestamp timestamp)
{
LOGV("sendResponse :: return buffer (%d) timestamp(%d) context(%p)", cmdid, timestamp, context);
WriteResponse resp(PVMFSuccess, cmdid, context, timestamp);
iWriteResponseQueueLock.Lock();
if (iWriteResponseQueue.size() < iWriteResponseQueue.capacity()) {
iWriteResponseQueue.push_back(resp);
} else {
LOGE("Exceeded response queue capacity");
}
iWriteResponseQueueLock.Unlock();
// Create an event for the threadsafe callback AO
OsclAny* P = NULL;
iWriteCompleteAO->ReceiveEvent(P);
}
void AndroidAudioMIO::setAudioSink(const sp<MediaPlayerInterface::AudioSink>& audioSink)
{
LOGV("setAudioSink in");
mAudioSink = audioSink;
}
//------------------------------------------------------------------------
// Active timing support
//
OSCL_EXPORT_REF PVMFStatus AndroidAudioMIOActiveTimingSupport::SetClock(PVMFMediaClock *clockVal)
{
LOGV("ATS :: SetClock in");
iClock=clockVal;
return PVMFSuccess;
}
void AndroidAudioMIOActiveTimingSupport::NotificationsInterfaceDestroyed()
{
LOGV("ATS :: NotificationsInterfaceDestroyed in");
iClockNotificationsInf=NULL;
}
OSCL_EXPORT_REF bool AndroidAudioMIOActiveTimingSupport::queryInterface(const PVUuid& aUuid, PVInterface*& aInterface)
{
aInterface = NULL;
PVUuid uuid;
queryUuid(uuid);
bool status = false;
if (uuid == aUuid) {
PvmiClockExtensionInterface* myInterface = OSCL_STATIC_CAST(PvmiClockExtensionInterface*, this);
aInterface = OSCL_STATIC_CAST(PVInterface*, myInterface);
status = true;
}
LOGV("ATS :: queryInterface out status %d",status);
return status;
}
void AndroidAudioMIOActiveTimingSupport::queryUuid(PVUuid& uuid)
{
LOGV("ATS :: queryUuid in");
uuid = PvmiClockExtensionInterfaceUuid;
}
void AndroidAudioMIOActiveTimingSupport::ClockStateUpdated()
{
LOGV("ATS :: ClockStateUpdated in");
if (iClock) {
PVMFMediaClock::PVMFMediaClockState newClockState = iClock->GetState();
if (newClockState != iClockState) {
iClockState = newClockState;
switch (iClockState) {
case PVMFMediaClock::STOPPED:
LOGV("A/V clock stopped");
break;
case PVMFMediaClock::RUNNING:
LOGV("A/V clock running");
// must be seeking, get new clock offset for A/V sync
if (iUpdateClock) {
bool overflowFlag = false;
uint32 currentTimeBase32 = 0;
iClock->GetCurrentTime32(iStartTime, overflowFlag, PVMF_MEDIA_CLOCK_MSEC, currentTimeBase32);
iFrameCount = 0;
iUpdateClock = false;
LOGV("update iStartTime: %d", iStartTime);
}
LOGV("signal thread to start");
if (iAudioThreadSem) iAudioThreadSem->Signal();
break;
case PVMFMediaClock::PAUSED:
LOGV("A/V clock paused");
break;
default:
LOGE("Wrong clock state!");
break;
}
}
}
LOGV("ATS :: ClockStateUpdated out");
}
void AndroidAudioMIOActiveTimingSupport::UpdateClock()
{
LOGV("ATS :: UpdateClock in");
if (iClock && (iClockState == PVMFMediaClock::RUNNING)) {
uint32 clockTime32, timeBaseTime32, updateClock32;
int32 correction = 0;
bool overflowFlag = false;
// get current time
iClock->GetCurrentTime32(clockTime32, overflowFlag, PVMF_MEDIA_CLOCK_MSEC, timeBaseTime32);
LOGV("PV clock current = %u", clockTime32);
// calculate sample clock
updateClock32 = iFrameCount * iMsecsPerFrame;
LOGV("sample clock = %u frameCount(%u) msecsPerFrame(%f)", updateClock32,iFrameCount,iMsecsPerFrame);
// startup - force clock backwards to compensate for latency
if (updateClock32 < iDriverLatency) {
LOGV("iStartTime = %u , iDriverLatency = %u",iStartTime, iDriverLatency);
correction = iStartTime - clockTime32;
LOGV("latency stall - forcing clock (correction = %d)",correction);
}
// normal play mode - check delta between PV engine clock and sample clock
else {
correction = (updateClock32 - iDriverLatency) - (clockTime32 - iStartTime);
LOGV("clock drift (correction = (updateClock32(%d)-iDriverLatency(%d))-(clockTime32(%d)-iStartTime(%d))= %d)",updateClock32,iDriverLatency,clockTime32,iStartTime,correction);
}
// do clock correction if drift exceeds threshold
if (OSCL_ABS(correction) > iMinCorrection) {
if (correction > iMaxCorrection) {
correction = iMaxCorrection;
} else if (correction < -iMaxCorrection) {
correction = -iMaxCorrection;
}
updateClock32 = clockTime32 + correction;
LOGV("drift correction = %d, new clock = %u", correction, updateClock32);
PVMFMediaClockAdjustTimeStatus adjustmentstatus =
iClock->AdjustClockTime32(clockTime32, timeBaseTime32, updateClock32,PVMF_MEDIA_CLOCK_MSEC,overflowFlag);
if ((PVMF_MEDIA_CLOCK_ADJUST_SUCCESS != adjustmentstatus))
{
LOGE("Error adjusting clock status = %d",adjustmentstatus);
}
if(overflowFlag) {
LOGE("Adjusting clock caused overflow");
}
}
else{
LOGV("ATS :: sample clock and pv clock are close enough no need to update");
}
}
LOGV("ATS :: UpdateClock out");
}
void AndroidAudioMIOActiveTimingSupport::setDriverLatency(uint32 latency)
{
LOGV("ATS :: setDriverLatency in");
iDriverLatency = latency;
if (iClock){
LOGV("register latency to media clock and set clockobserver");
PVMFStatus ret = iClock->ConstructMediaClockNotificationsInterface(iClockNotificationsInf,*this,latency);
if(iClockNotificationsInf && (PVMFSuccess == ret))
{
iClockNotificationsInf->SetClockStateObserver(*this);
}
else
{
LOGE("latency could NOT be set !! set it later ");
}
}
LOGV("ATS :: setDriverLatency out");
}