sl@0
|
1 |
// Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
//
|
sl@0
|
15 |
|
sl@0
|
16 |
#include "MMFAudioCodec.h"
|
sl@0
|
17 |
#include <mmf/common/mmfpaniccodes.h>
|
sl@0
|
18 |
#include "MmfImaAdpcmtopcm16hwdevice.h"
|
sl@0
|
19 |
#include <mmf/server/mmfswcodecwrappercustominterfacesuids.hrh>
|
sl@0
|
20 |
|
sl@0
|
21 |
/**
|
sl@0
|
22 |
*
|
sl@0
|
23 |
* NewL
|
sl@0
|
24 |
*
|
sl@0
|
25 |
*/
|
sl@0
|
26 |
CMMFImaAdpcmToPcm16CodecHwDevice* CMMFImaAdpcmToPcm16CodecHwDevice::NewL()
|
sl@0
|
27 |
{
|
sl@0
|
28 |
CMMFImaAdpcmToPcm16CodecHwDevice* self=new(ELeave)CMMFImaAdpcmToPcm16CodecHwDevice();
|
sl@0
|
29 |
CleanupStack::PushL(self);
|
sl@0
|
30 |
self->ConstructL();
|
sl@0
|
31 |
CleanupStack::Pop(self);
|
sl@0
|
32 |
return self;
|
sl@0
|
33 |
}
|
sl@0
|
34 |
|
sl@0
|
35 |
/**
|
sl@0
|
36 |
*
|
sl@0
|
37 |
* ConstructL
|
sl@0
|
38 |
*
|
sl@0
|
39 |
*/
|
sl@0
|
40 |
void CMMFImaAdpcmToPcm16CodecHwDevice::ConstructL()
|
sl@0
|
41 |
{
|
sl@0
|
42 |
iCodec = new (ELeave) CMMFImaAdpcmToPcm16Codec();
|
sl@0
|
43 |
}
|
sl@0
|
44 |
|
sl@0
|
45 |
/**
|
sl@0
|
46 |
*
|
sl@0
|
47 |
* ~CMMFMulawPcm16HwDevice
|
sl@0
|
48 |
*
|
sl@0
|
49 |
*/
|
sl@0
|
50 |
CMMFImaAdpcmToPcm16CodecHwDevice::~CMMFImaAdpcmToPcm16CodecHwDevice()
|
sl@0
|
51 |
{
|
sl@0
|
52 |
}
|
sl@0
|
53 |
|
sl@0
|
54 |
/**
|
sl@0
|
55 |
*
|
sl@0
|
56 |
* Codec
|
sl@0
|
57 |
*
|
sl@0
|
58 |
*/
|
sl@0
|
59 |
CMMFSwCodec &CMMFImaAdpcmToPcm16CodecHwDevice::Codec()
|
sl@0
|
60 |
{
|
sl@0
|
61 |
return *iCodec;
|
sl@0
|
62 |
}
|
sl@0
|
63 |
|
sl@0
|
64 |
/**
|
sl@0
|
65 |
Retrieves a custom interface to the device.
|
sl@0
|
66 |
The reference CMMFImaAdpcmToPcm16CodecHwDevice supports one FileBlockLength custom interface,
|
sl@0
|
67 |
or else calls two standard (CMMFSwCodecWrapper) custom interfaces,TPlayCustomInterface and TRecordCustomInterface.
|
sl@0
|
68 |
|
sl@0
|
69 |
@param aInterface
|
sl@0
|
70 |
Interface UID, defined with the custom interface.
|
sl@0
|
71 |
aInterface = KUidCustomInterfaceDevSoundFileBlockLength,
|
sl@0
|
72 |
aInterface = KMmfPlayCustomInterface for TPlayCustomInterface,
|
sl@0
|
73 |
aInterface = KMmfRecordCustomInterface for TRecordCustomInterface.
|
sl@0
|
74 |
@return A pointer to the interface implementation, The return value must be cast to the
|
sl@0
|
75 |
correct type by the user.
|
sl@0
|
76 |
*/
|
sl@0
|
77 |
TAny* CMMFImaAdpcmToPcm16CodecHwDevice::CustomInterface(TUid aInterfaceId)
|
sl@0
|
78 |
{
|
sl@0
|
79 |
if(aInterfaceId == KUidCustomInterfaceDevSoundFileBlockLength)
|
sl@0
|
80 |
{
|
sl@0
|
81 |
return static_cast<MMMFDevSoundCustomInterfaceFileBlockLength*>(this);
|
sl@0
|
82 |
}
|
sl@0
|
83 |
else
|
sl@0
|
84 |
{
|
sl@0
|
85 |
return CMMFSwCodecWrapper::CustomInterface(aInterfaceId);
|
sl@0
|
86 |
}
|
sl@0
|
87 |
}
|
sl@0
|
88 |
|
sl@0
|
89 |
/**
|
sl@0
|
90 |
@see CMMFSwCodecWrapper::Start
|
sl@0
|
91 |
|
sl@0
|
92 |
this function sets SampleRate and Channels for CMMFImaAdpcmToPcm16Codec
|
sl@0
|
93 |
*/
|
sl@0
|
94 |
TInt CMMFImaAdpcmToPcm16CodecHwDevice::Start(TDeviceFunc aFuncCmd, TDeviceFlow aFlowCmd)
|
sl@0
|
95 |
{
|
sl@0
|
96 |
TInt err = static_cast<CMMFImaAdpcmToPcm16Codec*>(iCodec)->Configure(iChannels, iSampleRate, iBlockAlign);
|
sl@0
|
97 |
if (err == KErrNone)
|
sl@0
|
98 |
{
|
sl@0
|
99 |
err = CMMFSwCodecWrapper::Start(aFuncCmd, aFlowCmd);
|
sl@0
|
100 |
}
|
sl@0
|
101 |
|
sl@0
|
102 |
return err;
|
sl@0
|
103 |
}
|
sl@0
|
104 |
|
sl@0
|
105 |
/**
|
sl@0
|
106 |
SetFileBlockLength
|
sl@0
|
107 |
|
sl@0
|
108 |
This function sets file's block length for CMMFImaAdpcmToPcm16CodecHwDevice
|
sl@0
|
109 |
|
sl@0
|
110 |
@param aBlockAlign
|
sl@0
|
111 |
The file's block length
|
sl@0
|
112 |
*/
|
sl@0
|
113 |
void CMMFImaAdpcmToPcm16CodecHwDevice::SetFileBlockLength(TUint aBlockAlign)
|
sl@0
|
114 |
{
|
sl@0
|
115 |
iBlockAlign = aBlockAlign;
|
sl@0
|
116 |
}
|
sl@0
|
117 |
|
sl@0
|
118 |
CMMFImaAdpcmToPcm16Codec::CMMFImaAdpcmToPcm16Codec()
|
sl@0
|
119 |
{
|
sl@0
|
120 |
iChannels = 1;
|
sl@0
|
121 |
iSampleRate = 0;
|
sl@0
|
122 |
iBlockAlign = KImaAdpcmBlockAlign;
|
sl@0
|
123 |
iSamplesPerBlock = KImaAdpcmSamplesPerBlock;
|
sl@0
|
124 |
}
|
sl@0
|
125 |
/**
|
sl@0
|
126 |
*
|
sl@0
|
127 |
* ProcessL
|
sl@0
|
128 |
* @param aSrc
|
sl@0
|
129 |
* @param aDst
|
sl@0
|
130 |
* @pre position of buffer aSrc is 0
|
sl@0
|
131 |
* @pre position of buffer aDst is 0
|
sl@0
|
132 |
* @pre sufficient bytes in output to consume input
|
sl@0
|
133 |
* @return TCodecProcessResult
|
sl@0
|
134 |
* This function converts IMA ADPCM samples to PCM samples.
|
sl@0
|
135 |
*
|
sl@0
|
136 |
*/
|
sl@0
|
137 |
CMMFSwCodec::TCodecProcessResult CMMFImaAdpcmToPcm16Codec::ProcessL(const CMMFBuffer& aSrc, CMMFBuffer& aDst)
|
sl@0
|
138 |
{
|
sl@0
|
139 |
CMMFSwCodec::TCodecProcessResult result;
|
sl@0
|
140 |
result.iCodecProcessStatus = TCodecProcessResult::EProcessComplete;
|
sl@0
|
141 |
|
sl@0
|
142 |
//convert from generic CMMFBuffer to CMMFDataBuffer
|
sl@0
|
143 |
const CMMFDataBuffer* src = STATIC_CAST(const CMMFDataBuffer*, &aSrc);
|
sl@0
|
144 |
CMMFDataBuffer* dst = STATIC_CAST(CMMFDataBuffer*, &aDst);
|
sl@0
|
145 |
|
sl@0
|
146 |
if( !CheckPreconditions( src, dst ) )
|
sl@0
|
147 |
{
|
sl@0
|
148 |
//[ precondition(s) violation ]
|
sl@0
|
149 |
User::Leave(KErrArgument);
|
sl@0
|
150 |
}
|
sl@0
|
151 |
|
sl@0
|
152 |
//calculate how much source is required to fill the destination buffer
|
sl@0
|
153 |
TUint blocksRemaining = src->Data().Length() / iBlockAlign;
|
sl@0
|
154 |
|
sl@0
|
155 |
//we need to cast away CONST even on the source, as the TClass needs a TUint8*
|
sl@0
|
156 |
TUint8* pSrc = CONST_CAST(TUint8*,src->Data().Ptr());
|
sl@0
|
157 |
TUint8* pDst = CONST_CAST(TUint8*,dst->Data().Ptr());
|
sl@0
|
158 |
|
sl@0
|
159 |
//[ [process full blocks ]
|
sl@0
|
160 |
TUint dstBytesAdded = 0;
|
sl@0
|
161 |
for( TUint count = 0; count < blocksRemaining; count++ )
|
sl@0
|
162 |
{
|
sl@0
|
163 |
iImaAdpcmTo16Pcm.Convert(pSrc, pDst, iSamplesPerBlock);
|
sl@0
|
164 |
pSrc += iBlockAlign;
|
sl@0
|
165 |
pDst += (iSamplesPerBlock * sizeof(TInt16));
|
sl@0
|
166 |
dstBytesAdded += (iSamplesPerBlock * sizeof(TInt16));
|
sl@0
|
167 |
}
|
sl@0
|
168 |
|
sl@0
|
169 |
result.iCodecProcessStatus = TCodecProcessResult::EProcessComplete;
|
sl@0
|
170 |
result.iSrcBytesProcessed = blocksRemaining * iBlockAlign;
|
sl@0
|
171 |
result.iDstBytesAdded = dstBytesAdded;
|
sl@0
|
172 |
dst->Data().SetLength(result.iDstBytesAdded);
|
sl@0
|
173 |
|
sl@0
|
174 |
//[ check post conditions
|
sl@0
|
175 |
__ASSERT_DEBUG( (src->Position() == 0), TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
|
sl@0
|
176 |
__ASSERT_DEBUG( (dst->Position() == 0), TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
|
sl@0
|
177 |
TInt r1 = src->Data().Length();
|
sl@0
|
178 |
r1 /= iBlockAlign;
|
sl@0
|
179 |
TInt r2 = dst->Data().Length();
|
sl@0
|
180 |
r2 /=(iSamplesPerBlock * sizeof(TInt16));
|
sl@0
|
181 |
__ASSERT_DEBUG( r1== r2, TMmfAudioCodecPanicsNameSpace::Panic(TMmfAudioCodecPanicsNameSpace::EPostConditionViolation ));
|
sl@0
|
182 |
__ASSERT_DEBUG( dst->Data().Length() % 2 == 0, TMmfAudioCodecPanicsNameSpace::Panic( TMmfAudioCodecPanicsNameSpace::EPostConditionViolation )); // pcm output
|
sl@0
|
183 |
|
sl@0
|
184 |
return result;
|
sl@0
|
185 |
}
|
sl@0
|
186 |
|
sl@0
|
187 |
/**
|
sl@0
|
188 |
*
|
sl@0
|
189 |
* Preconditions
|
sl@0
|
190 |
* This methos tests the preconditions of the ProcessL method
|
sl@0
|
191 |
* @return TBool ETrue for sucess and EFalse for failure of the preconditions
|
sl@0
|
192 |
*
|
sl@0
|
193 |
**/
|
sl@0
|
194 |
TBool CMMFImaAdpcmToPcm16Codec::CheckPreconditions( const CMMFDataBuffer* aSrcBuffer, CMMFDataBuffer* aDestBuffer )
|
sl@0
|
195 |
{
|
sl@0
|
196 |
TBool result = EFalse;
|
sl@0
|
197 |
|
sl@0
|
198 |
if(! aSrcBuffer )
|
sl@0
|
199 |
{
|
sl@0
|
200 |
return result;
|
sl@0
|
201 |
}
|
sl@0
|
202 |
|
sl@0
|
203 |
if( ! aDestBuffer )
|
sl@0
|
204 |
{
|
sl@0
|
205 |
return result;
|
sl@0
|
206 |
}
|
sl@0
|
207 |
|
sl@0
|
208 |
// Check position of src and dest are 0
|
sl@0
|
209 |
if( aSrcBuffer->Position() )
|
sl@0
|
210 |
{
|
sl@0
|
211 |
return result;
|
sl@0
|
212 |
}
|
sl@0
|
213 |
|
sl@0
|
214 |
// Check position of src and dest are 0
|
sl@0
|
215 |
if( aDestBuffer->Position() )
|
sl@0
|
216 |
{
|
sl@0
|
217 |
return result;
|
sl@0
|
218 |
}
|
sl@0
|
219 |
|
sl@0
|
220 |
// check there are sufficient bytes in the output to consume the input
|
sl@0
|
221 |
const TUint KTempBufferSize = iSamplesPerBlock * 2;
|
sl@0
|
222 |
TInt numInputSubFrames = aSrcBuffer->Data().Length() / iBlockAlign;
|
sl@0
|
223 |
TInt numOutputSubFrames = aDestBuffer->Data().MaxLength() / KTempBufferSize;
|
sl@0
|
224 |
|
sl@0
|
225 |
//[ we need modulo 1010 bytes on all src frames that are not the last
|
sl@0
|
226 |
// frame
|
sl@0
|
227 |
// For the last frame we will code only whole frames and effectively
|
sl@0
|
228 |
// drop any remaining samples]
|
sl@0
|
229 |
TBool validInputDataLength = (aSrcBuffer->Data().Length() % iBlockAlign == 0) ;
|
sl@0
|
230 |
|
sl@0
|
231 |
if( (numInputSubFrames > numOutputSubFrames) || // sufficient space in the output for the input
|
sl@0
|
232 |
(aSrcBuffer->Position() > 0 ) || // position must be zero since we can eat all the data
|
sl@0
|
233 |
(aDestBuffer->Position() > 0 ) ||
|
sl@0
|
234 |
(!validInputDataLength)) //position must be zero
|
sl@0
|
235 |
{
|
sl@0
|
236 |
return result;
|
sl@0
|
237 |
}
|
sl@0
|
238 |
|
sl@0
|
239 |
result = ETrue; // preconditions have been satisfied
|
sl@0
|
240 |
|
sl@0
|
241 |
return result;
|
sl@0
|
242 |
}
|
sl@0
|
243 |
|
sl@0
|
244 |
/**
|
sl@0
|
245 |
Configure
|
sl@0
|
246 |
|
sl@0
|
247 |
This function sets file's block length, channels & sample rate.
|
sl@0
|
248 |
|
sl@0
|
249 |
@param aChannels
|
sl@0
|
250 |
The file's number of channels
|
sl@0
|
251 |
@param aSampleRate
|
sl@0
|
252 |
The file's sample rate
|
sl@0
|
253 |
@param aBlockAlign
|
sl@0
|
254 |
The file's block length
|
sl@0
|
255 |
*/
|
sl@0
|
256 |
TInt CMMFImaAdpcmToPcm16Codec::Configure(TUint aChannels, TUint aSampleRate, TUint aBlockAlign)
|
sl@0
|
257 |
{
|
sl@0
|
258 |
iChannels = aChannels;
|
sl@0
|
259 |
iSampleRate = aSampleRate;
|
sl@0
|
260 |
|
sl@0
|
261 |
if (aBlockAlign < 256)
|
sl@0
|
262 |
{
|
sl@0
|
263 |
switch (iSampleRate * iChannels)
|
sl@0
|
264 |
{
|
sl@0
|
265 |
case 8000: // fall through, same as 11025
|
sl@0
|
266 |
case 11025:
|
sl@0
|
267 |
case 16000:
|
sl@0
|
268 |
iBlockAlign = 256;
|
sl@0
|
269 |
break;
|
sl@0
|
270 |
case 22050:
|
sl@0
|
271 |
iBlockAlign = 512;
|
sl@0
|
272 |
break;
|
sl@0
|
273 |
|
sl@0
|
274 |
case 44100:
|
sl@0
|
275 |
iBlockAlign = 1024;
|
sl@0
|
276 |
break;
|
sl@0
|
277 |
|
sl@0
|
278 |
case 88200:
|
sl@0
|
279 |
iBlockAlign = 2048;
|
sl@0
|
280 |
break;
|
sl@0
|
281 |
|
sl@0
|
282 |
default:
|
sl@0
|
283 |
return KErrArgument;
|
sl@0
|
284 |
}
|
sl@0
|
285 |
}
|
sl@0
|
286 |
else
|
sl@0
|
287 |
{
|
sl@0
|
288 |
iBlockAlign = aBlockAlign;
|
sl@0
|
289 |
}
|
sl@0
|
290 |
|
sl@0
|
291 |
const TUint KImaAdpcmBitsPerSample = 4;
|
sl@0
|
292 |
iSamplesPerBlock = (iBlockAlign - 4 * iChannels) * 8 / (KImaAdpcmBitsPerSample * iChannels) + 1;
|
sl@0
|
293 |
|
sl@0
|
294 |
return KErrNone;
|
sl@0
|
295 |
}
|
sl@0
|
296 |
|
sl@0
|
297 |
|
sl@0
|
298 |
|
sl@0
|
299 |
|
sl@0
|
300 |
|
sl@0
|
301 |
|