Update contrib.
1 // Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // mmfswaudioinput.cpp
17 #include "mmfswaudioinput.h"
18 #include "mmfswaudioinputpriv.h"
19 #include <d32soundsc.h>
21 #include "mmf/utils/rateconvert.h" // if we need to resample
22 _LIT(KPddFileName,"SOUNDSC.PDD");
23 _LIT(KLddFileName,"ESOUNDSC.LDD");
25 #ifdef SYMBIAN_SWCODEC_LOGGING
27 const TText* const KStateNames[] = // must agree with TState
30 _S("EStateInitialized2"),
31 _S("EStateRecordWait2"),
32 _S("EStateRecordWaitAck2"),
35 static const TText* StateName(TInt aState)
37 return KStateNames[aState];
40 const TText* const KRStateNames[] = // must agree with TRunningState
44 _S("ERStateFinishing"),
45 _S("ERStateFinished"),
49 static const TText* RStateName(TInt aState)
51 return KRStateNames[aState];
54 #endif // SYMBIAN_SWCODEC_LOGGING
58 static void Panic(TInt aPanic)
60 _LIT(KPanicString, "SwAudioInput");
61 User::Panic(KPanicString, aPanic);
64 void CAudioInput::CheckFullInvariant()
67 CheckActiveRecorders();
70 void CAudioInput::CheckInvariant(TBool aKnownConstructed)
72 // full check would be that each recorder is in one, and only one, queue.
73 // However, since the queues share the same infrastructure, checking the overall length of queues
74 // is correct should suffice. During construction or deletion this may obviously vary
75 TInt totalLength = QLength(iIdleQueue) + QLength(iRecordingQueue) +
76 QLength(iPendingQueue) + QLength(iBusyQueue);
77 if (aKnownConstructed)
79 __ASSERT_DEBUG(totalLength==KNumRecorders, Panic(KPanicBadTotalQueueLength));
83 __ASSERT_DEBUG(totalLength<=KNumRecorders, Panic(KPanicBadTotalQueueLength2));
85 __ASSERT_DEBUG(QLength(iBusyQueue)<=1, Panic(KPanicBadTotalQueueLength));
90 // inline versions that do nothing...
92 inline void CAudioInput::CheckFullInvariant()
96 inline void CAudioInput::CheckInvariant(TBool /*aKnownConstructed*/)
102 const TInt KMinBufferSize = 4; // assume a good default?
103 //Shared chunk driver does not support max. buffer size. 16K is given in order to simulate the old driver behavior.
104 const TInt KMaxBufferSize = 0x4000;
106 //Table that maps given linear value of volume to the corresponding decibel value.
107 const TUint8 KLinearToDbConstantLookup[] =
369 const TInt KNumSampleRates = 9;
371 struct TSampleRateEnumTable
374 TSoundRate iRateEnum;
377 //Table that maps given samples per second to the corresponding enums in RSoundSc
378 const TSampleRateEnumTable KRateEnumLookup[] =
380 {48000,ESoundRate48000Hz,KSoundRate48000Hz},
381 {44100,ESoundRate44100Hz,KSoundRate44100Hz},
382 {32000,ESoundRate32000Hz,KSoundRate32000Hz},
383 {24000,ESoundRate24000Hz,KSoundRate24000Hz},
384 {22050,ESoundRate22050Hz,KSoundRate22050Hz},
385 {16000,ESoundRate16000Hz,KSoundRate16000Hz},
386 {12000,ESoundRate12000Hz,KSoundRate12000Hz},
387 {11025,ESoundRate11025Hz,KSoundRate11025Hz},
388 {8000, ESoundRate8000Hz, KSoundRate8000Hz}
393 EXPORT_C TAudioInputParams::TAudioInputParams() :
394 iSampleRate(0), iNominalBufferSize(0)
401 EXPORT_C MAudioInput* MAudioInput::CreateL(MAudioInputObserver& aObserver)
403 MAudioInput* result = CAudioInput::NewL(aObserver);
407 CAudioInput* CAudioInput::NewL(MAudioInputObserver& aObserver)
409 CAudioInput* result = new CAudioInput(aObserver);
410 CleanupStack::PushL(result);
411 result->ConstructL();
412 CleanupStack::Pop(result);
416 CAudioInput::CAudioInput(MAudioInputObserver& aObserver) :
417 iObserver(aObserver),
418 iIdleQueue(_FOFF(CRecorder,iLink)),
419 iRecordingQueue(_FOFF(CRecorder,iLink)),
420 iPendingQueue(_FOFF(CRecorder,iLink)),
421 iBusyQueue(_FOFF(CRecorder,iLink))
423 ASSERT(iState == EStateCreated2); // assume zero'ing initialises correctly
426 void CAudioInput::Release()
427 // effective destructor call
432 TAny* CAudioInput::Interface(TUid aInterfaceUid)
434 if (aInterfaceUid == KUidAIParamInterface)
436 MAIParamInterface* self = this;
442 RSoundSc& CAudioInput::RecordSoundDevice()
444 ASSERT(iRecordSoundDevice.Handle()!=0); // should be open
445 return iRecordSoundDevice;
448 CAudioInput::~CAudioInput()
450 CheckInvariant(EFalse); // may not be constructed
452 for (TInt i = 0; i < KNumRecorders; i++)
454 // just in case, call cancel directly from this point too
455 // Cancel depends on the active queue, and might not be quite the same.
456 CRecorder* recorder = iRecorders[i];
463 delete iAsyncCallBack;
465 iRecordSoundDevice.Close();
469 void CAudioInput::Cancel()
471 #ifdef SYMBIAN_SWCODEC_LOGGING
472 RDebug::Print(_L("--->CAudioInput::Cancel()"));
477 iAsyncCallBack->Cancel();
479 #ifdef SYMBIAN_SWCODEC_LOGGING
480 RDebug::Print(_L("<---CAudioInput::Cancel()"));
484 void CAudioInput::CancelRecorders()
485 // if a recorder is active, then cancel it. Also move the list if required.
487 CheckInvariant(); // semi-invariant check - this is called from destructor
490 while (QPop(recorder, iRecordingQueue))
493 iIdleQueue.AddLast(*recorder);
495 CheckFullInvariant();
498 void CAudioInput::CancelPendingRecorders()
499 // take any recorder in the pending queue. ack the buffer and send to idle
501 CheckFullInvariant();
504 while (QPop(recorder, iPendingQueue))
506 recorder->ReleaseBuffer();
507 iIdleQueue.AddLast(*recorder);
509 CheckFullInvariant();
512 void CAudioInput::CancelBusyRecorder()
513 // take busy recorder. ack the buffer and send to idle
515 CheckFullInvariant();
518 if (QPop(recorder, iBusyQueue))
520 recorder->ReleaseBuffer();
521 iIdleQueue.AddLast(*recorder);
523 CheckFullInvariant();
526 void CAudioInput::RecordAllIdle()
527 // take any recorder in idle queue and set recording
529 CheckFullInvariant();
532 while (QPop(recorder, iIdleQueue))
534 recorder->RecordData();
535 iRecordingQueue.AddLast(*recorder);
537 CheckFullInvariant();
540 void CAudioInput::ConstructL()
542 for (TInt i = 0; i < KNumRecorders; i++)
544 iRecorders[i] = new (ELeave) CRecorder(*this, i);
545 iIdleQueue.AddLast(*(iRecorders[i]));
547 iAsyncCallBack = new (ELeave) CAsyncCallBack(CActive::EPriorityStandard);
548 TCallBack callback(Callback, this);
549 iAsyncCallBack->Set(callback);
550 User::LoadPhysicalDevice(KPddFileName);
551 User::LoadLogicalDevice(KLddFileName);
552 CheckFullInvariant();
555 TInt CAudioInput::Initialize(const TAudioInputParams& aParams)
557 #ifdef SYMBIAN_SWCODEC_LOGGING
558 RDebug::Print(_L("--->CAudioInput::Initialize() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
560 CheckFullInvariant();
561 TInt error = KErrNone;
562 if (iState == EStateCreated2)
564 if (!iRecordSoundDevice.Handle())
566 error = iRecordSoundDevice.Open(KSoundScRxUnit0);
569 Close(); // TODO Close() required?
574 iBufferLength = aParams.iNominalBufferSize; // will be updated by SetFormat() if required
575 error = SetFormat(aParams);
578 iRecordBufferConfig.iNumBuffers = KNumRecorders*2; // for each AO we create two buffers
579 iRecordBufferConfig.iFlags = 0;
580 iRecordBufferConfig.iBufferSizeInBytes = iBufferLength;
581 ASSERT(iChunk.Handle()==0); // should not be already open
582 TPckg<TRecordSharedChunkBufConfig> bufferConfigBuf(
583 iRecordBufferConfig);
584 error = iRecordSoundDevice.SetBufferChunkCreate(
585 bufferConfigBuf, iChunk);
586 #ifdef SYMBIAN_SWCODEC_LOGGING
588 _L("iRecordBufferConfig.iNumBuffers = [%d]"),iRecordBufferConfig.iNumBuffers);
590 _L("iRecordBufferConfig.iFlags = [%d]"),iRecordBufferConfig.iFlags);
592 _L("iRecordBufferConfig.iBufferSizeInBytes = [%d]"),iRecordBufferConfig.iBufferSizeInBytes);
594 if (error == KErrNone)
596 ASSERT(iChunk.Handle()); // should now be open
597 iRecordSoundDevice.GetBufferConfig(bufferConfigBuf); // overwrite iRecordBufferConfig
598 SetGain(aParams.iInitialGain);
599 iState = EStateInitialized2;
606 error = KErrNotReady;
608 #ifdef SYMBIAN_SWCODEC_LOGGING
609 RDebug::Print(_L("<---CAudioInput::Initialize(%d) state=%s rstate=%s"), error, StateName(
610 iState), RStateName(iRState));
612 CheckFullInvariant();
616 void CAudioInput::Close()
618 #ifdef SYMBIAN_SWCODEC_LOGGING
619 RDebug::Print(_L("--->CAudioInput::Close() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
621 CheckFullInvariant();
622 InternalStop(); // Technically this should not be required, as client should Stop() first, but just in case
623 if (iState == EStateInitialized2)
625 iRecordSoundDevice.Close();
628 iState = EStateCreated2;
630 ASSERT(iState==EStateCreated2);
631 ASSERT(QLength(iIdleQueue)==KNumRecorders);
632 #ifdef SYMBIAN_SWCODEC_LOGGING
633 RDebug::Print(_L("<---CAudioInput::Close() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
635 CheckFullInvariant();
638 TInt CAudioInput::Start()
640 #ifdef SYMBIAN_SWCODEC_LOGGING
641 RDebug::Print(_L("--->CAudioInput::Start() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
643 CheckFullInvariant();
644 TInt error = KErrNone;
645 if (iState == EStateInitialized2)
648 iState = EStateRecordWait2;
649 iRState = ERStateRunning;
653 error = KErrNotReady;
655 #ifdef SYMBIAN_SWCODEC_LOGGING
656 RDebug::Print(_L("<---CAudioInput::Start(%d) state=%s rstate=%s"),
657 error, StateName(iState), RStateName(iRState));
659 CheckFullInvariant();
663 void CAudioInput::BufferAck()
665 #ifdef SYMBIAN_SWCODEC_LOGGING
666 RDebug::Print(_L("--->CAudioInput::BufferAck() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
668 CheckFullInvariant();
669 ASSERT(iState==EStateRecordWaitAck2);
671 iState = EStateRecordWait2;
673 #ifdef SYMBIAN_SWCODEC_LOGGING
674 RDebug::Print(_L("<---CAudioInput::BufferAck() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
676 CheckFullInvariant();
679 void CAudioInput::HandleBufferAck()
681 CRecorder* recorder = QPop(iBusyQueue);
682 recorder->ReleaseBuffer();
683 if (iRState == ERStateRunning)
685 recorder->RecordData();
686 iRecordingQueue.AddLast(*recorder);
690 iIdleQueue.AddLast(*recorder);
691 if (iRState == ERStatePaused && (QLength(iRecordingQueue)+QLength(iPendingQueue) == 0))
693 iRState = ERStateFinishing;
698 TInt CAudioInput::Pause()
700 #ifdef SYMBIAN_SWCODEC_LOGGING
701 RDebug::Print(_L("--->CAudioInput::Pause() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
703 CheckFullInvariant();
704 TInt err = KErrNone; // note we are silent if called in wrong state
706 #ifdef SYMBIAN_SWCODEC_LOGGING
708 _L("***!pause irecordingquelength %d pendingquelength %d ibusyquelength=%d"), QLength(iRecordingQueue),
709 QLength(iPendingQueue), QLength(iBusyQueue));
711 if ((iState == EStateRecordWait2 || iState == EStateRecordWaitAck2) && iRState==ERStateRunning)
713 iRecordSoundDevice.Pause();
715 iRState = ERStatePaused;
717 #ifdef SYMBIAN_SWCODEC_LOGGING
718 RDebug::Print(_L("<---CAudioInput::Pause(%d) state=%s err=%d"), err, StateName(iState), RStateName(iRState));
720 CheckFullInvariant();
724 TInt CAudioInput::Resume()
726 #ifdef SYMBIAN_SWCODEC_LOGGING
727 RDebug::Print(_L("--->CAudioInput::Resume() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
729 CheckFullInvariant();
730 TInt err = KErrNone; // note we are silent if called in the wrong state
731 if ((iState == EStateRecordWait2 || iState == EStateRecordWaitAck2) &&
732 ((iRState==ERStatePaused || iRState==ERStateFinishing || iRState==ERStateFinished)))
734 err = RecordSoundDevice().Resume();
738 iRState = ERStateRunning;
740 #ifdef SYMBIAN_SWCODEC_LOGGING
741 RDebug::Print(_L("<---CAudioInput::Resume(%d) state=%s rstate=%s"), err, StateName(iState), RStateName(iRState));
743 CheckFullInvariant();
747 TInt CAudioInput::Flush()
749 #ifdef SYMBIAN_SWCODEC_LOGGING
750 RDebug::Print(_L("--->CAudioInput::Flush() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
752 CheckFullInvariant();
753 TInt error = KErrNotReady;
754 if (iRState==ERStatePaused)
756 if (iState == EStateRecordWait2)
759 ASSERT(iState == EStateRecordWait2); // stay put
762 else if (iState == EStateRecordWaitAck2)
765 iState = EStateRecordWait2;
769 #ifdef SYMBIAN_SWCODEC_LOGGING
770 RDebug::Print(_L("--->CAudioInput::Flush(%d) state=%s rstate=%s"), error, StateName(iState), RStateName(iRState));
772 CheckFullInvariant();
776 void CAudioInput::Stop()
778 #ifdef SYMBIAN_SWCODEC_LOGGING
779 RDebug::Print(_L("--->CAudioInput::Stop() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
781 CheckFullInvariant();
783 #ifdef SYMBIAN_SWCODEC_LOGGING
784 RDebug::Print(_L("<---CAudioInput::Stop() state=%s rstate=%s"), StateName(iState), RStateName(iRState));
788 void CAudioInput::InternalStop()
789 // This stops all recording and returns pending and busy buffers to idle. Must be called when
790 // client knows the buffer has been grabbed (e.g. _not_ before error callback)
792 CheckInvariant(); // Can be called from buffer error, so can't check full invariant.
793 if (iState != EStateInitialized2 && iState != EStateCreated2)
796 iState = EStateInitialized2;
798 CheckFullInvariant();
799 ASSERT((QLength(iRecordingQueue) + QLength(iPendingQueue) +
800 QLength(iBusyQueue))==0); // everything is stopped
803 void CAudioInput::InternalFlush()
806 CancelPendingRecorders();
807 CancelBusyRecorder();
810 void CAudioInput::BufferArrives(CRecorder* aRecorder)
812 #ifdef SYMBIAN_SWCODEC_LOGGING
814 _L("--->CAudioInput::BufferArrives(%d,%d) state=%s rstate=%s"), aRecorder->Index(),
815 aRecorder->StatusOrOffset(), StateName(iState), RStateName(iRState));
817 CheckInvariant(); // Can't use CheckFullInvariant() from RunL
818 ASSERT(iState==EStateRecordWait2 || iState==EStateRecordWaitAck2);
819 ASSERT(aRecorder->Offset()>=0); // assert we're not here due to an error
820 iRecordingQueue.Remove(*aRecorder);
821 iPendingQueue.AddLast(*aRecorder);
822 if (iState==EStateRecordWait2)
826 #ifdef SYMBIAN_SWCODEC_LOGGING
827 RDebug::Print(_L("<---CAudioInput::BufferArrives() state=%s rstate=%s"),
828 StateName(iState), RStateName(iRState));
830 CheckFullInvariant();
833 void CAudioInput::UseBuffer(CRecorder* aRecorder)
834 // incomming buffer is pointed to by iBufPtr. Either directly or via convert, use for callback
836 iBufPtr.Set(iChunk.Base() + aRecorder->Offset(), aRecorder->Length());
839 #ifdef SYMBIAN_SWCODEC_LOGGING
840 RDebug::Print(_L("iBufPtr length [%d] iconvbuff length [%d,%d]"),
841 iBufPtr.Length(), iConvBuff.Length(), iConvBuff.MaxLength());
843 __DEBUG_ONLY(TInt converted =) iConverter->Convert(iBufPtr, iConvBuff);
844 // the following assert should check we convert the log.
845 // Actually we sometimes fail at the end of the operation with what is effectively
846 // the last buffer. Arguably a driver fault, but there we are
847 // ASSERT(converted==iBufPtr.Length());
848 #ifdef SYMBIAN_SWCODEC_LOGGING
849 RDebug::Print(_L("iBufPtr length [%d] iconvbuff length after [%d,%d]"),
850 iBufPtr.Length(), iConvBuff.Length(), iConvBuff.MaxLength());
852 iObserver.InputBufferAvailable(iConvBuff);
856 iObserver.InputBufferAvailable(iBufPtr);
858 #ifdef SYMBIAN_SWCODEC_LOGGING
859 RDebug::Print(_L("12345 ibufptr = [0x%x]"),iBufPtr.Ptr());
863 void CAudioInput::BufferError(CRecorder* aRecorder, TInt aError)
865 #ifdef SYMBIAN_SWCODEC_LOGGING
867 _L("--->CAudioInput::BufferError(%d,%d) state=%s rstate=%s"), aRecorder->Index(),
868 aError, StateName(iState), RStateName(iRState));
870 CheckInvariant(); // Can't use CheckFullInvariant() from RunL
871 if (aError==KErrCancel || aError==KErrOverflow)
873 // Cancel: sign of a Pause operation. If paused etc, then merely add to idle list. potentially generate finished signal
874 // if not paused, then not clear but just in case request record again
875 // Overflow: basically try again, but if paused merely add to idle. Check for last buffer just in case
876 if (iRState!=ERStateRunning)
878 iRecordingQueue.Remove(*aRecorder);
879 iIdleQueue.AddLast(*aRecorder);
880 #ifdef SYMBIAN_SWCODEC_LOGGING
882 _L("***! irecordingquelength %d pendingquelength %d ibusyquelength=%d"), QLength(iRecordingQueue),
883 QLength(iPendingQueue), QLength(iBusyQueue));
885 if (iRState == ERStatePaused && (QLength(iRecordingQueue)+QLength(iPendingQueue)+QLength(iBusyQueue) == 0))
887 iRState = ERStateFinishing;
893 aRecorder->RecordData();
898 iRecordingQueue.Remove(*aRecorder);
899 iIdleQueue.AddLast(*aRecorder);
900 iRState = ERStateFailed;
901 iObserver.InputError(aError);
903 #ifdef SYMBIAN_SWCODEC_LOGGING
904 RDebug::Print(_L("<---CAudioInput::BufferError() state=%s rstate=%s"),
905 StateName(iState), RStateName(iRState));
907 CheckFullInvariant();
910 TInt CAudioInput::Callback(TAny* aPtr)
912 CAudioInput* self = static_cast<CAudioInput*> (aPtr);
913 TRAPD(error,self->AsyncCallbackL());
914 return error; // TODO really have to handle error
917 void CAudioInput::RequestCallback()
919 // ensure iAsyncCallBack is active
920 if (!iAsyncCallBack->IsActive())
922 iAsyncCallBack->Call();
926 void CAudioInput::AsyncCallbackL()
928 #ifdef SYMBIAN_SWCODEC_LOGGING
929 RDebug::Print(_L("--->CAudioInput::AsyncCallbackL() state=%s rstate=%s"),
930 StateName(iState), RStateName(iRState));
932 CheckFullInvariant();
933 ASSERT(iState==EStateRecordWait2 || iState==EStateRecordWaitAck2); // should not occur in other states. Actually ignore in 2nd
934 if (iState==EStateRecordWait2)
936 if (QLength(iPendingQueue)>0)
938 ASSERT(QLength(iBusyQueue)==0);
939 iState = EStateRecordWaitAck2; // change state prior to callback, in case sync call from callback
940 CRecorder* recorder = QPop(iPendingQueue);
941 iBusyQueue.AddLast(*recorder);
946 if (iRState == ERStateFinishing)
948 ASSERT(QLength(iRecordingQueue)+QLength(iPendingQueue)+QLength(iBusyQueue) == 0); // should be true
949 iRState = ERStateFinished;
950 iObserver.InputFinished();
954 #ifdef SYMBIAN_SWCODEC_LOGGING
955 RDebug::Print(_L("<---CAudioInput::AsyncCallbackL() state=%s rstate=%s"),
956 StateName(iState), RStateName(iRState));
958 CheckFullInvariant();
961 TInt CAudioInput::GetBufferSizes(TInt& aMinSize, TInt& aMaxSize)
963 aMinSize = KMinBufferSize;
964 aMaxSize = KMaxBufferSize;
968 TInt CAudioInput::SetGain(TInt aGain)
970 TInt error = KErrNone; // note: silent if in wrong state
971 if (iRecordSoundDevice.Handle())
973 // we have to switch from level to dB value
974 if(aGain >=0 && aGain<=KSoundMaxVolume)
976 error = iRecordSoundDevice.SetVolume(KLinearToDbConstantLookup[aGain]);
980 error = KErrArgument;
986 TInt CAudioInput::SetFormat(const TAudioInputParams& aFormat)
988 TInt err = KErrNotFound;
989 TCurrentSoundFormatV02Buf formatBuf;
990 TFormatData formatData;
993 iConverter = NULL; // setting this to NULL indicates we are not using converter. No other flag
995 TInt wantedRate = aFormat.iSampleRate;
996 for (TInt index = 0; index < KNumSampleRates; index++)
998 if (wantedRate == KRateEnumLookup[index].iRate)
1000 formatBuf().iRate = KRateEnumLookup[index].iRateEnum;
1001 formatData.iSampleRate = wantedRate;
1007 if (err == KErrNone)
1009 formatBuf().iChannels = aFormat.iNumChannels;
1010 formatBuf().iEncoding = ESoundEncoding16BitPCM;
1011 formatBuf().iDataFormat = ESoundDataFormatInterleaved;
1012 err = iRecordSoundDevice.SetAudioFormat(formatBuf);
1013 #if defined(SYMBIAN_SOUNDADAPTER_FORCECDRATES) || defined (SYMBIAN_SOUNDADAPTER_FORCESTEREO)
1014 err = KErrNotSupported; // force Negotiate - for debugging
1016 if (err == KErrNotSupported)
1018 // don't support directly. Perhaps can rate convert?
1019 err = NegotiateFormat(aFormat, formatData);
1025 TInt CAudioInput::NegotiateFormat(const TAudioInputParams& aFormat, TFormatData &aFormatData)
1027 TInt err = KErrNotFound;
1028 TCurrentSoundFormatV02Buf formatBuf;
1030 TInt origBufferLength = iBufferLength; // cache in case we change
1032 // find out first what the driver supports
1033 TSoundFormatsSupportedV02Buf supportedFormat;
1034 iRecordSoundDevice.Caps(supportedFormat);
1035 TUint32 supportedRates = supportedFormat().iRates;
1036 #ifdef SYMBIAN_SOUNDADAPTER_FORCECDRATES
1037 supportedRates &= KSoundRate11025Hz | KSoundRate22050Hz
1038 | KSoundRate44100Hz; // only use CD rates - for debugging
1042 // We want the next rate above consistently - we go down from this to the requested rate.
1043 // If there is one, we don't support - we _never_ upsample.
1044 // note that the table is given in descending order, so we start with the highest
1045 TInt wantedRate = aFormat.iSampleRate;
1046 TInt takeTheFirst = EFalse;
1047 TInt nextUpValidIndex = -1;
1048 for (TInt index = 0; index < KNumSampleRates; index++)
1050 TBool lookingAtRequestedRate = wantedRate
1051 == KRateEnumLookup[index].iRate;
1052 TSoundRate wantedEnum = KRateEnumLookup[index].iRateEnum;
1053 TUint32 equivBitmap = KRateEnumLookup[index].iRateConstant;
1054 TBool isSupported = (equivBitmap & supportedRates) != EFalse;
1055 if (lookingAtRequestedRate || takeTheFirst)
1059 // this rate is supported
1060 formatBuf().iRate = wantedEnum;
1061 aFormatData.iActualRate = KRateEnumLookup[index].iRate;
1066 else if (!takeTheFirst)
1068 // while we are still looking for the rate, want to cache any supported index
1069 // at end of loop, this will be the first rate above ours that is supported
1070 // use for fallback if required
1073 nextUpValidIndex = index;
1076 if (lookingAtRequestedRate)
1078 // For record we just abort.
1085 // if there is one above the requested rate, use that
1086 if (nextUpValidIndex >= 0)
1088 TSoundRate wantedEnum =
1089 KRateEnumLookup[nextUpValidIndex].iRateEnum;
1090 formatBuf().iRate = wantedEnum;
1091 aFormatData.iActualRate = KRateEnumLookup[nextUpValidIndex].iRate;
1098 // should have something!
1102 aFormatData.iSampleRate = wantedRate; // iSampleRate is our requested/apparent rate, not the device rate.
1104 TUint32 channelsSupported = supportedFormat().iChannels;
1105 #ifdef SYMBIAN_SOUNDADAPTER_FORCESTEREO
1106 channelsSupported &= KSoundStereoChannel; // don't use mono - for debugging
1109 if (aFormat.iNumChannels == 1)
1111 aFormatData.iRequestedChannels = 1;
1113 if (channelsSupported & KSoundMonoChannel)
1115 // mono is supported, as usual
1116 aFormatData.iActualChannels = 1;
1118 else if (channelsSupported & KSoundStereoChannel)
1120 aFormatData.iActualChannels = 2;
1121 iBufferLength *= 2; // double size, will do stereo->mono
1125 return KErrNotSupported; // should not get this far for real
1128 else if (aFormat.iNumChannels == 2)
1130 aFormatData.iRequestedChannels = 2;
1132 if (channelsSupported & KSoundStereoChannel)
1134 // stereo is supported, as usual
1135 aFormatData.iActualChannels = 2;
1137 else if (channelsSupported & KSoundMonoChannel)
1139 aFormatData.iActualChannels = 1;
1140 iBufferLength /= 2; // halve size, will do mono->stereo
1144 return KErrNotSupported; // should not get this far for real
1149 return KErrNotSupported; // unknown number of channels requested!
1152 formatBuf().iChannels = aFormatData.iActualChannels;
1154 formatBuf().iEncoding = ESoundEncoding16BitPCM;
1155 formatBuf().iDataFormat = ESoundDataFormatInterleaved;
1156 err = iRecordSoundDevice.SetAudioFormat(formatBuf);
1160 ASSERT(!iConverter); // pre-condition at top of function anyway
1161 // when recording we convert from actual to requested
1162 TInt outputRateToUse = aFormatData.iSampleRate;
1163 #ifdef SYMBIAN_SKIP_RESAMPLE_ON_RECORD
1164 // with this macro just channel convert at most
1165 outputRateToUse = aFormatData.iActualRate;
1167 #ifdef SYMBIAN_SOUNDADAPTER_DEBUG
1168 RDebug::Print(_L("RMdaDevSound::CBody::NegotiateFormat: Convert:CreateL from %d/%d to %d/%d"),
1169 aFormatData.iActualRate, aFormatData.iActualChannels,
1170 aFormatData.iSampleRate, aFormatData.iRequestedChannels);
1172 TRAP(err, iConverter = CChannelAndSampleRateConverter::CreateL(aFormatData.iActualRate,
1173 aFormatData.iActualChannels,
1175 aFormatData.iRequestedChannels));
1177 if (!err && iConverter)
1179 err = iConvBuff.Create(origBufferLength);
1180 #ifdef SYMBIAN_SWCODEC_LOGGING
1181 RDebug::Print(_L("iBufferLength length [%d] iconvbuff length [%d,%d]"),
1182 iBufferLength, iConvBuff.Length(), iConvBuff.MaxLength());
1189 TInt CAudioInput::GetSupportedSampleRates(RArray<TInt>& aSupportedSampleRates)
1191 TInt err = KErrNone;
1193 if (iRecordSoundDevice.Handle())
1195 GetSupportedSampleRates(aSupportedSampleRates, iRecordSoundDevice);
1198 {//temporarily open the device if we can
1200 err = tempsound.Open(KSoundScRxUnit0);
1203 err = GetSupportedSampleRates(aSupportedSampleRates, tempsound);
1210 TInt CAudioInput::GetSupportedSampleRates(
1211 RArray<TInt>& aSupportedSampleRates, RSoundSc& aSoundDevice)
1213 ASSERT(aSoundDevice.Handle()); // parent to ensure this is open
1215 TInt err = KErrNone;
1217 TSoundFormatsSupportedV02Buf supportedFormat;
1218 aSoundDevice.Caps(supportedFormat);
1219 TUint32 rates = supportedFormat().iRates;
1221 for (TInt i = KNumSampleRates - 1; i > 0; i--)//min to max
1223 if (rates & KRateEnumLookup[i].iRateConstant)
1225 err = aSupportedSampleRates.Append(KRateEnumLookup[i].iRate);
1235 TInt CAudioInput::QLength(TSglQue<CRecorder>& aQueue)
1236 // count elements in List/Q. Have to use iterator to do this - it seems.
1238 TSglQueIter<CRecorder> iter(aQueue);
1242 // like old-fashioned C string manipulations. iterate through all members
1248 CAudioInput::CRecorder* CAudioInput::QPop(TSglQue<CRecorder>& aQueue)
1250 CRecorder* recorder = NULL;
1251 if (! aQueue.IsEmpty())
1253 recorder = aQueue.First();
1254 aQueue.Remove(*recorder);
1261 // these functions are used in invariant checking only
1263 void CAudioInput::CheckActiveRecorders(TSglQue<CRecorder>& aQueue, TBool aExpected, TInt aPanicCode)
1264 // check that all the elements in the given Q are IsActive() or vice-versa
1266 TSglQueIter<CRecorder> iter(aQueue);
1268 CRecorder* recorder;
1269 while ((recorder=iter++)!=NULL)
1271 TBool expected = aExpected != EFalse; // ensure these are either true or false
1272 TBool active = recorder->IsActive() != EFalse;
1273 __ASSERT_DEBUG(expected == active, Panic(aPanicCode));
1277 void CAudioInput::CheckActiveRecorders()
1278 // check that all the elements in the recordingQueue are IsActive() etc
1279 // can't be used as CRecorder::RunL() pre-condition
1281 CheckActiveRecorders(iRecordingQueue, ETrue, EPanicBusyRecorderNotActive);
1282 CheckActiveRecorders(iIdleQueue, EFalse, EPanicNonBusyRecorderActive);
1283 CheckActiveRecorders(iPendingQueue, EFalse, EPanicNonBusyRecorderActive);
1284 CheckActiveRecorders(iBusyQueue, EFalse, EPanicNonBusyRecorderActive);
1294 CAudioInput::CRecorder::CRecorder(CAudioInput& aParent, TInt aIndex) :
1295 CActive(EPriorityStandard), iParent(aParent), iIndex(aIndex)
1297 CActiveScheduler::Add(this);
1300 CAudioInput::CRecorder::~CRecorder()
1305 void CAudioInput::CRecorder::Cancel()
1307 // this override takes into account that ReleaseBuffer must be called - this is not the
1308 // normal pattern where following Cancel() we're not concerned with the results
1311 ASSERT(!BufferHeld()); // if active then buffer held should be clear. don't reset then
1313 ReleaseBuffer(ETrue); // release - might have been a successful run!
1317 ReleaseBuffer(); // this will release buffer if still outstanding
1321 void CAudioInput::CRecorder::RunL()
1323 #ifdef SYMBIAN_SWCODEC_LOGGING
1324 RDebug::Print(_L("--->CAudioInput::CRecorder::RunL(%d, %d)"), Index(),
1327 TInt errorOrOffset = iStatus.Int(); // negative -> error. non-negative is offset in chunk
1329 if (errorOrOffset < 0)
1331 #ifdef SYMBIAN_SWCODEC_LOGGING
1332 RDebug::Print(_L("errorOrOffset = [%d]"),errorOrOffset);
1334 // ReleaseBuffer(ETrue); // calls ReleaseBuffer() on error code. Driver requires this, even though seems wrong
1335 iParent.BufferError(this, errorOrOffset);
1339 ASSERT(!iBufferHeld);
1340 iBufferHeld = ETrue;
1342 #ifdef SYMBIAN_SWCODEC_LOGGING
1343 RDebug::Print(_L("errorOrOffset = [%d]"),errorOrOffset);
1345 // If a buffer larger than expected arrives truncate it.
1346 iLength = Min(iLength,iParent.iBufferLength);
1347 iParent.BufferArrives(this);
1349 #ifdef SYMBIAN_SWCODEC_LOGGING
1350 RDebug::Print(_L("<---CAudioInput::CRecorder::RunL(%d)"), Index());
1354 void CAudioInput::CRecorder::RecordData()
1356 #ifdef SYMBIAN_SWCODEC_LOGGING
1357 RDebug::Print(_L("--->CAudioInput::CRecorder::RecordData(%d)"), Index());
1359 ASSERT(!iBufferHeld);
1360 Deque(); // ensure we append to the AO queue, so if it comes to it we process oldest request first
1361 CActiveScheduler::Add(this);
1362 iLength = iParent.BufferLength(); // TODO do we have to set this first or is it an OUT param purely
1363 iParent.RecordSoundDevice().RecordData(iStatus, iLength);
1366 #ifdef SYMBIAN_SWCODEC_LOGGING
1367 RDebug::Print(_L("###****#####!!!! Buffer length [%d], status [%d] "), iLength,
1370 #ifdef SYMBIAN_SWCODEC_LOGGING
1371 RDebug::Print(_L("<---CAudioInput::CRecorder::RecordData(%d)"), Index());
1375 void CAudioInput::CRecorder::DoCancel()
1377 #ifdef SYMBIAN_SWCODEC_LOGGING
1378 RDebug::Print(_L("--->CAudioInput::CRecorder::DoCancel(%d)"), Index());
1380 iParent.RecordSoundDevice().Cancel(iStatus);
1381 #ifdef SYMBIAN_SWCODEC_LOGGING
1382 RDebug::Print(_L("<---CAudioInput::CRecorder::DoCancel(%d)"), Index());
1386 void CAudioInput::CRecorder::ReleaseBuffer(TBool aDoAnyway)
1388 if (iBufferHeld || aDoAnyway)
1390 iParent.RecordSoundDevice().ReleaseBuffer(iStatus.Int());
1391 iBufferHeld = EFalse;
1395 TInt CAudioInput::CRecorder::Index() const
1400 TInt CAudioInput::CRecorder::Length() const
1405 TBool CAudioInput::CRecorder::IsBusy() const
1407 return IsActive() || BufferHeld();
1410 TBool CAudioInput::CRecorder::BufferHeld() const
1411 // BufferHeld() means we're in control of a passed buffer
1416 TInt CAudioInput::CRecorder::Offset() const
1417 // If we call this, we've discounted errors so can assert non-negative
1419 TInt result = StatusOrOffset();
1424 TInt CAudioInput::CRecorder::StatusOrOffset() const
1425 // The iStatus assuming is valid
1427 ASSERT(!IsActive()); // or would not be valid
1428 TInt result = iStatus.Int();