os/mm/devsound/devsoundrefplugin/src/plugin/audio/mmfpcm16ToImaAdpcm.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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 "mmfpcm16ToImaAdpcm.h"
sl@0
    17
sl@0
    18
/**
sl@0
    19
*
sl@0
    20
* NewL
sl@0
    21
*
sl@0
    22
*/
sl@0
    23
CMMFPcm16ToImaAdpcmHwDevice* CMMFPcm16ToImaAdpcmHwDevice::NewL()
sl@0
    24
	{
sl@0
    25
	CMMFPcm16ToImaAdpcmHwDevice* self=new(ELeave) CMMFPcm16ToImaAdpcmHwDevice();
sl@0
    26
	CleanupStack::PushL(self);
sl@0
    27
	self->ConstructL();
sl@0
    28
	CleanupStack::Pop(self);
sl@0
    29
	return self;
sl@0
    30
	}
sl@0
    31
sl@0
    32
/**
sl@0
    33
*
sl@0
    34
* ~CMMFPcm16ToAlawHwDevice
sl@0
    35
*
sl@0
    36
*/
sl@0
    37
CMMFPcm16ToImaAdpcmHwDevice::~CMMFPcm16ToImaAdpcmHwDevice()
sl@0
    38
	{
sl@0
    39
	}
sl@0
    40
sl@0
    41
/**
sl@0
    42
*
sl@0
    43
* ConstructL
sl@0
    44
*
sl@0
    45
*/
sl@0
    46
void CMMFPcm16ToImaAdpcmHwDevice::ConstructL()
sl@0
    47
	{
sl@0
    48
	iCodec = new (ELeave) CMMFPcm16ToImaAdpcmCodec();
sl@0
    49
	}
sl@0
    50
sl@0
    51
/**
sl@0
    52
*
sl@0
    53
* Codec
sl@0
    54
*
sl@0
    55
*/
sl@0
    56
CMMFSwCodec &CMMFPcm16ToImaAdpcmHwDevice::Codec()
sl@0
    57
	{
sl@0
    58
	return *iCodec;
sl@0
    59
	}
sl@0
    60
sl@0
    61
/**
sl@0
    62
*
sl@0
    63
* ResetL
sl@0
    64
*
sl@0
    65
*/
sl@0
    66
void CMMFPcm16ToImaAdpcmCodec::ResetL()
sl@0
    67
	{
sl@0
    68
	//Reset the actual codec
sl@0
    69
	TMMFImaAdpcmCodecState state;
sl@0
    70
	state.iIndex = 0;
sl@0
    71
	state.iPredicted = 0;
sl@0
    72
	iPcm16ToImaAdpcm.SetState(state);
sl@0
    73
	}
sl@0
    74
	
sl@0
    75
/**
sl@0
    76
*
sl@0
    77
* ProcessL
sl@0
    78
* @param aSrc src buffer
sl@0
    79
* @param aDst destination buffer
sl@0
    80
* @return CMMFSwCodec::TCodecProcessResult
sl@0
    81
*  This function converts PCM samples to IMA ADPCM samples in
sl@0
    82
*  blocks of KImaAdpcmBlockAlign (256) bytes. 1010 source 
sl@0
    83
*  bytes are required to fill a 256 byte block.
sl@0
    84
* @pre if last buffer and src contains < 1010 bytes discard input
sl@0
    85
* This function throws away the last buffer if it contains < 1010 bytes
sl@0
    86
* (ie we must have sufficient data to process an entire frame )
sl@0
    87
* All other src buffers must contain
sl@0
    88
* All destination buffer must contain
sl@0
    89
*
sl@0
    90
**/
sl@0
    91
CMMFSwCodec::TCodecProcessResult CMMFPcm16ToImaAdpcmCodec::ProcessL(const CMMFBuffer& aSrc, CMMFBuffer& aDst)
sl@0
    92
	{
sl@0
    93
	CMMFSwCodec::TCodecProcessResult result;
sl@0
    94
	result.iCodecProcessStatus = TCodecProcessResult::EProcessComplete;
sl@0
    95
	
sl@0
    96
	//convert from generic CMMFBuffer to CMMFDataBuffer
sl@0
    97
	const CMMFDataBuffer* source = STATIC_CAST(const CMMFDataBuffer*, &aSrc);
sl@0
    98
	CMMFDataBuffer* destination  = STATIC_CAST(CMMFDataBuffer*, &aDst);
sl@0
    99
	
sl@0
   100
	//[ source and destination must not be null ]
sl@0
   101
	if( !source || !destination )
sl@0
   102
		User::Leave( KErrArgument );
sl@0
   103
	
sl@0
   104
	//[ check preconditions ]
sl@0
   105
	if( !BuffersStatus( source, destination ))
sl@0
   106
		{
sl@0
   107
		User::Leave( KErrArgument );
sl@0
   108
		}
sl@0
   109
	
sl@0
   110
	//[ code the buffers ]
sl@0
   111
	ProcessBuffers( *source, *destination, result );
sl@0
   112
	
sl@0
   113
	return result;	
sl@0
   114
	}
sl@0
   115
sl@0
   116
/**
sl@0
   117
*
sl@0
   118
* ProcessBuffers
sl@0
   119
* @param aSource
sl@0
   120
* @param aDestination
sl@0
   121
* @param aResult 
sl@0
   122
* all we have to do is find out how many source frames there 
sl@0
   123
* are to process and process them
sl@0
   124
* finally returning process complete and fillin the status of the result
sl@0
   125
*
sl@0
   126
**/
sl@0
   127
void CMMFPcm16ToImaAdpcmCodec::ProcessBuffers(const CMMFDataBuffer& aSource, CMMFDataBuffer& aDestination, CMMFSwCodec::TCodecProcessResult& aResult )
sl@0
   128
	{
sl@0
   129
	//[ calculate how many full buffers are to be processed ]
sl@0
   130
    const TUint srcLen    = aSource.Data().Length();
sl@0
   131
    TInt numFullSrcFrames = srcLen/KSourceFrameSize;
sl@0
   132
    
sl@0
   133
	TUint8* pSrc = const_cast<TUint8*>(aSource.Data().Ptr());
sl@0
   134
	TUint8* pDst = const_cast<TUint8*>(aDestination.Data().Ptr());
sl@0
   135
    TInt dstBytesAdded = 0;
sl@0
   136
	// calculate number of pcm samples per source frame
sl@0
   137
	const TInt KSamplesPerFrame = KSourceFrameSize/(sizeof(TInt16)); 
sl@0
   138
	
sl@0
   139
	//[ convert all the buffers ]
sl@0
   140
	for( TInt count = 0; count < numFullSrcFrames; count++ )
sl@0
   141
		{
sl@0
   142
		i16PcmToImaAdpcm.Convert(pSrc, pDst, KSamplesPerFrame );				
sl@0
   143
		pSrc          += KSourceFrameSize;
sl@0
   144
		pDst          += KCodedFrameSize;
sl@0
   145
		dstBytesAdded += KCodedFrameSize;
sl@0
   146
		}
sl@0
   147
	aResult.iSrcBytesProcessed = numFullSrcFrames*KSourceFrameSize;
sl@0
   148
	aResult.iDstBytesAdded = dstBytesAdded;		
sl@0
   149
	aDestination.Data().SetLength( aResult.iDstBytesAdded);
sl@0
   150
	}
sl@0
   151
sl@0
   152
/**
sl@0
   153
*
sl@0
   154
* BuffersStatus
sl@0
   155
* @param source buffer containing the data to be coded
sl@0
   156
* @param destination buffer containing the coded data
sl@0
   157
* @return TBool EFalse indicates bad buffers
sl@0
   158
*
sl@0
   159
**/
sl@0
   160
TBool CMMFPcm16ToImaAdpcmCodec::BuffersStatus( const CMMFDataBuffer* source, const CMMFDataBuffer* destination )
sl@0
   161
	{
sl@0
   162
	TBool status = EFalse;
sl@0
   163
	
sl@0
   164
	//[ demand source and destination positions are zero ]
sl@0
   165
    CMMFDataBuffer* pDst = const_cast<CMMFDataBuffer*>( destination );
sl@0
   166
	if( source->Position() || destination->Position() )
sl@0
   167
		{
sl@0
   168
		return status;
sl@0
   169
		}
sl@0
   170
	
sl@0
   171
	//[ Have we got full buffers ]
sl@0
   172
	TInt sourceBuffers = source->Data().Length()/KSourceFrameSize;
sl@0
   173
	TInt destBuffers   = (pDst->Data().MaxLength())/KCodedFrameSize;
sl@0
   174
	
sl@0
   175
	if( sourceBuffers <= destBuffers )  // the sink can process the source
sl@0
   176
		{
sl@0
   177
		return ETrue;                       // note this precondition has been weakened in line with other codecs
sl@0
   178
		}                                   // such that it can process partially full buffers
sl@0
   179
	                                        // ie you can if you wish use larger buffers than needed and only partially
sl@0
   180
	                                        // fill them. We do however expect all the input to be processed.
sl@0
   181
	return status;
sl@0
   182
	}