blob: e11e008192ba45625426a8c44f899f9c94a38a89 [file] [log] [blame]
/*
* Copyright (C) 2011 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.
*/
/**
******************************************************************************
* @file M4OSA_FileReader_optim.c
* @brief
* @note This file implements functions to manipulate filesystem access
******************************************************************************
*/
/** Addition of Trace ID **/
#include "M4OSA_CoreID.h"
#include "M4OSA_Error.h"
#ifdef M4TRACE_ID
#undef M4TRACE_ID
#endif
#define M4TRACE_ID M4OSA_FILE_READER
#include "M4OSA_FileCommon.h"
#include "M4OSA_FileReader.h"
#include "M4OSA_FileWriter.h"
#include "M4OSA_Memory.h"
#include "M4OSA_Debug.h"
#include "LVOSA_FileReader_optim.h"
#define M4OSA_READER_OPTIM_USE_OSAL_IF
#ifndef M4OSA_READER_OPTIM_USE_OSAL_IF
#include "M4OSA_FileAccess.h"
#endif
#define M4ERR_CHECK_NULL_RETURN_VALUE(retval, pointer) if ((pointer) == M4OSA_NULL) return (retval);
/**
******************************************************************************
* File reader cache buffers parameters (size, number of buffers, etc)
******************************************************************************
*/
#define M4OSA_READBUFFER_SIZE 1024*16
#define M4OSA_READBUFFER_NB 2
#define M4OSA_READBUFFER_NONE -1
#define M4OSA_EOF -1
#define MAX_FILLS_SINCE_LAST_ACCESS M4OSA_READBUFFER_NB*2
/**
******************************************************************************
* structure M4OSA_FileReader_Buffer
* @brief This structure defines the File reader Buffers context (private)
******************************************************************************
*/
typedef struct
{
M4OSA_MemAddr8 data; /**< buffer data */
M4OSA_FilePosition size; /**< size of the buffer */
M4OSA_FilePosition filepos; /**< position in the file where the buffer starts */
M4OSA_FilePosition remain; /**< data amount not already copied from buffer */
M4OSA_UInt32 nbFillSinceLastAcess; /**< To know since how many time we didn't use this buffer */
} M4OSA_FileReader_Buffer_optim;
/**
******************************************************************************
* structure M4OSA_FileReader_Context
* @brief This structure defines the File reader context (private)
* @note This structure is used for all File Reader calls to store the context
******************************************************************************
*/
typedef struct
{
M4OSA_Bool IsOpened; /**< Micro state machine */
M4OSA_FileAttribute FileAttribute; /**< Opening mode */
M4OSA_FilePosition readFilePos; /**< Effective position of the GFL read pointer */
M4OSA_FilePosition absolutePos; /**< Virtual position for next reading */
M4OSA_FilePosition fileSize; /**< Size of the file */
M4OSA_FileReader_Buffer_optim buffer[M4OSA_READBUFFER_NB]; /**< Read buffers */
M4OSA_Void* aFileDesc; /**< File descriptor */
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
M4OSA_FileReadPointer* FS; /**< Filesystem interface */
#else
M4OSA_FileSystem_FctPtr *FS; /**< Filesystem interface */
#endif
} M4OSA_FileReader_Context_optim;
/* __________________________________________________________ */
/*| |*/
/*| Global function for handling low level read access |*/
/*|__________________________________________________________|*/
static M4OSA_FileReadPointer* gv_NXPSW_READOPT_lowLevelFunctions;
M4OSA_ERR NXPSW_FileReaderOptim_init(M4OSA_Void *lowLevel_functionPointers, M4OSA_Void *optimized_functionPointers)
{
M4OSA_FileReadPointer* lowLevel_fp = (M4OSA_FileReadPointer*) lowLevel_functionPointers;
M4OSA_FileReadPointer* optimized_fp = (M4OSA_FileReadPointer*) optimized_functionPointers;
//Set the optimized functions, to be called by the user
optimized_fp->openRead = M4OSA_fileReadOpen_optim;
optimized_fp->readData = M4OSA_fileReadData_optim;
optimized_fp->seek = M4OSA_fileReadSeek_optim;
optimized_fp->closeRead = M4OSA_fileReadClose_optim;
optimized_fp->setOption = M4OSA_fileReadSetOption_optim;
optimized_fp->getOption = M4OSA_fileReadGetOption_optim;
return M4NO_ERROR;
}
M4OSA_ERR NXPSW_FileReaderOptim_cleanUp()
{
gv_NXPSW_READOPT_lowLevelFunctions = M4OSA_NULL;
return M4NO_ERROR;
}
M4OSA_ERR NXPSW_FileReaderOptim_getLowLevelFunctions(M4OSA_Void **FS)
{
M4OSA_FileReadPointer** pFunctionsPointer = (M4OSA_FileReadPointer**) FS;
*pFunctionsPointer = gv_NXPSW_READOPT_lowLevelFunctions;
return M4NO_ERROR;
}
/* __________________________________________________________ */
/*| |*/
/*| Buffer handling functions for Read access |*/
/*|__________________________________________________________|*/
/**************************************************************/
M4OSA_ERR M4OSA_FileReader_BufferInit(M4OSA_FileReader_Context_optim* apContext)
/**************************************************************/
{
M4OSA_UInt8 i;
for(i=0; i<M4OSA_READBUFFER_NB; i++)
{
apContext->buffer[i].data = M4OSA_NULL;
apContext->buffer[i].size = 0;
apContext->buffer[i].filepos = 0;
apContext->buffer[i].remain = 0;
}
for(i=0; i<M4OSA_READBUFFER_NB; i++)
{
apContext->buffer[i].data = (M4OSA_MemAddr8) M4OSA_32bitAlignedMalloc(M4OSA_READBUFFER_SIZE,
M4OSA_FILE_READER, (M4OSA_Char *)"M4OSA_FileReader_BufferInit");
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_ALLOC, apContext->buffer[i].data);
}
return M4NO_ERROR;
}
/**************************************************************/
M4OSA_Void M4OSA_FileReader_BufferFree(M4OSA_FileReader_Context_optim* apContext)
/**************************************************************/
{
M4OSA_Int8 i;
for(i=0; i<M4OSA_READBUFFER_NB; i++)
if(apContext->buffer[i].data != M4OSA_NULL)
free(apContext->buffer[i].data);
}
/**************************************************************/
M4OSA_FilePosition M4OSA_FileReader_BufferCopy(M4OSA_FileReader_Context_optim* apContext,
M4OSA_Int8 i, M4OSA_FilePosition pos,
M4OSA_FilePosition size, M4OSA_MemAddr8 pData)
/**************************************************************/
{
M4OSA_FilePosition copysize;
M4OSA_FilePosition offset;
if(apContext->buffer[i].size == M4OSA_EOF) return M4OSA_EOF;
if( (pos < apContext->buffer[i].filepos)
|| (pos > (apContext->buffer[i].filepos + apContext->buffer[i].size - 1)) )
{
return 0; /* nothing copied */
}
offset = pos - apContext->buffer[i].filepos;
copysize = apContext->buffer[i].size - offset;
copysize = (size < copysize) ? size : copysize;
memcpy((void *)pData, (void *)(apContext->buffer[i].data + offset), copysize);
apContext->buffer[i].remain -= copysize;
apContext->buffer[i].nbFillSinceLastAcess = 0;
return copysize;
}
/**************************************************************/
M4OSA_ERR M4OSA_FileReader_BufferFill(M4OSA_FileReader_Context_optim* apContext,
M4OSA_Int8 i, M4OSA_FilePosition pos)
/**************************************************************/
{
M4OSA_FilePosition gridPos;
M4OSA_FilePosition tempPos;
M4OSA_UInt32 bufferSize;
M4OSA_FilePosition diff;
M4OSA_FilePosition size;
M4OSA_ERR err = M4NO_ERROR;
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
M4OSA_ERR errno = M4NO_ERROR;
M4OSA_UInt32 fileReadSize = 0;
M4OSA_FilePosition fileSeekPosition = 0;
#else
M4OSA_Int32 ret_val;
M4OSA_UInt16 errno;
#endif
M4OSA_TRACE3_4("BufferFill i = %d pos = %ld read = %ld old = %ld", i, pos,
apContext->readFilePos, apContext->buffer[i].filepos);
/* Avoid cycling statement because of EOF */
if(pos >= apContext->fileSize)
return M4WAR_NO_MORE_AU;
/* Relocate to absolute postion if necessary */
bufferSize = M4OSA_READBUFFER_SIZE;
tempPos = (M4OSA_FilePosition) (pos / bufferSize);
gridPos = tempPos * M4OSA_READBUFFER_SIZE;
diff = gridPos - apContext->readFilePos;
if(diff != 0)
{
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
fileSeekPosition = diff;
errno = apContext->FS->seek(apContext->aFileDesc, M4OSA_kFileSeekCurrent,
&fileSeekPosition);
apContext->readFilePos = gridPos;
if(M4NO_ERROR != errno)
{
err = errno;
M4OSA_TRACE1_1("M4OSA_FileReader_BufferFill ERR1 = 0x%x", err);
return err;
}
#else
ret_val = apContext->FS->pFctPtr_Seek(apContext->aFileDesc, diff,
M4OSA_kFileSeekCurrent, &errno);
apContext->readFilePos = gridPos;
if(ret_val != 0)
{
err = M4OSA_ERR_CREATE(M4_ERR, M4OSA_FILE_READER, errno);
M4OSA_TRACE1_1("M4OSA_FileReader_BufferFill ERR1 = 0x%x", err);
return err;
}
#endif /*M4OSA_READER_OPTIM_USE_OSAL_IF*/
}
apContext->buffer[i].filepos = apContext->readFilePos;
/* Read Data */
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
fileReadSize = M4OSA_READBUFFER_SIZE;
errno = apContext->FS->readData(apContext->aFileDesc,
(M4OSA_MemAddr8)apContext->buffer[i].data, &fileReadSize);
size = (M4OSA_FilePosition)fileReadSize;
if ((M4NO_ERROR != errno)&&(M4WAR_NO_DATA_YET != errno))
{
apContext->buffer[i].size = M4OSA_EOF;
apContext->buffer[i].remain = 0;
err = errno;
M4OSA_TRACE1_1("M4OSA_FileReader_BufferFill ERR2 = 0x%x", err);
return err;
}
#else
size = apContext->FS->pFctPtr_Read(apContext->aFileDesc,
(M4OSA_UInt8 *)apContext->buffer[i].data, M4OSA_READBUFFER_SIZE, &errno);
if(size == -1)
{
apContext->buffer[i].size = M4OSA_EOF;
apContext->buffer[i].remain = 0;
err = M4OSA_ERR_CREATE(M4_ERR, M4OSA_FILE_READER, errno);
M4OSA_TRACE1_1("M4OSA_FileReader_BufferFill ERR2 = 0x%x", err);
return err;
}
#endif
apContext->buffer[i].size = size;
apContext->buffer[i].remain = size;
apContext->buffer[i].nbFillSinceLastAcess = 0;
/* Retrieve current position */
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
errno = apContext->FS->getOption(apContext->aFileDesc,
M4OSA_kFileReadGetFilePosition,
(M4OSA_DataOption*) &apContext->readFilePos);
if (M4NO_ERROR != errno)
{
err = errno;
M4OSA_TRACE1_1("M4OSA_FileReader_BufferFill ERR3 = 0x%x", err);
}
else if( (apContext->buffer[i].size >= 0)
&& (apContext->buffer[i].size < M4OSA_READBUFFER_SIZE) )
{
err = M4WAR_NO_DATA_YET;
M4OSA_TRACE2_0("M4OSA_FileReader_BufferFill returns NO DATA YET");
return err;
}
#else
apContext->readFilePos = apContext->FS->pFctPtr_Tell(apContext->aFileDesc, &errno);
if( (apContext->buffer[i].size >= 0)
&& (apContext->buffer[i].size < M4OSA_READBUFFER_SIZE) )
{
err = M4WAR_NO_DATA_YET;
M4OSA_TRACE1_1("M4OSA_FileReader_BufferFill ERR3 = 0x%x", err);
return err;
}
#endif /*M4OSA_READER_OPTIM_USE_OSAL_IF*/
/* Return without error */
return M4NO_ERROR;
}
/**************************************************************/
M4OSA_Int8 M4OSA_FileReader_BufferMatch(M4OSA_FileReader_Context_optim* apContext,
M4OSA_FilePosition pos)
/**************************************************************/
{
M4OSA_Int8 i;
/* Select the buffer which matches with given pos */
for(i=0; i<M4OSA_READBUFFER_NB; i++)
{
if( (pos >= apContext->buffer[i].filepos)
&& (pos < (apContext->buffer[i].filepos + apContext->buffer[i].size)) )
{
return i;
}
}
return M4OSA_READBUFFER_NONE;
}
/**************************************************************/
M4OSA_Int8 M4OSA_FileReader_BufferSelect(M4OSA_FileReader_Context_optim* apContext,
M4OSA_Int8 current_i)
/**************************************************************/
{
M4OSA_Int8 i,j;
M4OSA_FilePosition min_amount,max_amount;
M4OSA_Int8 min_i,max_count;
/* update nbFillSinceLastAcess field */
for(i=0; i<M4OSA_READBUFFER_NB; i++)
{
apContext->buffer[i].nbFillSinceLastAcess ++;
}
/* Plan A : Scan for empty buffer */
for(i=0; i<M4OSA_READBUFFER_NB; i++)
{
if(apContext->buffer[i].remain == 0)
{
return i;
}
}
max_count = M4OSA_READBUFFER_NB;
max_amount = MAX_FILLS_SINCE_LAST_ACCESS;
/* Plan B : Scan for dead buffer */
for(i=0; i<M4OSA_READBUFFER_NB; i++)
{
if(apContext->buffer[i].nbFillSinceLastAcess >= (M4OSA_UInt32) max_amount)
{
max_amount = apContext->buffer[i].nbFillSinceLastAcess;
max_count = i;
}
}
if(max_count<M4OSA_READBUFFER_NB)
{
M4OSA_TRACE2_2("DEAD BUFFER: %d, %d",max_count,apContext->buffer[max_count].nbFillSinceLastAcess);
return max_count;
}
min_i = current_i;
min_amount = M4OSA_READBUFFER_SIZE;
/* Select the buffer which is the most "empty" */
for(i=0; i<M4OSA_READBUFFER_NB; i++)
{
j = (i+current_i)%M4OSA_READBUFFER_NB;
if(apContext->buffer[j].remain < min_amount)
{
min_amount = apContext->buffer[j].remain;
min_i = j;
}
}
return min_i;
}
/**************************************************************/
M4OSA_ERR M4OSA_FileReader_CalculateSize(M4OSA_FileReader_Context_optim* apContext)
/**************************************************************/
{
M4OSA_ERR err = M4NO_ERROR;
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
M4OSA_ERR errno = M4NO_ERROR;
#else
M4OSA_Int32 ret_val;
M4OSA_UInt16 errno;
#endif
/* go to the end of file*/
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
errno = apContext->FS->getOption(apContext->aFileDesc, M4OSA_kFileReadGetFileSize,
(M4OSA_DataOption*) &apContext->fileSize);
if (M4NO_ERROR != errno)
{
err = errno;
M4OSA_TRACE1_1("M4OSA_FileReader_CalculateSize ERR = 0x%x", err);
}
#else
ret_val = apContext->FS->pFctPtr_Seek(apContext->aFileDesc, 0, M4OSA_kFileSeekEnd, &errno);
if (ret_val != 0)
{
apContext->readFilePos = M4OSA_EOF;
err = M4OSA_ERR_CREATE(M4_ERR, M4OSA_FILE_READER, errno);
M4OSA_TRACE1_1("M4OSA_FileReader_CalculateSize ERR = 0x%x", err);
}
else
{
/* Retrieve size of the file */
apContext->fileSize = apContext->FS->pFctPtr_Tell(apContext->aFileDesc, &errno);
apContext->readFilePos = apContext->fileSize;
}
#endif /*M4OSA_READER_OPTIM_USE_OSAL_IF*/
return err;
}
/* __________________________________________________________ */
/*| |*/
/*| OSAL filesystem API |*/
/*|__________________________________________________________|*/
/**
******************************************************************************
* @brief This method opens the provided fileDescriptor and returns its context.
* @param pContext: (OUT) File reader context.
* @param pFileDescriptor : (IN) File Descriptor of the input file.
* @param FileModeAccess : (IN) File mode access.
* @return M4NO_ERROR: there is no error
* @return M4ERR_PARAMETER pContext or fileDescriptor is NULL
* @return M4ERR_ALLOC there is no more memory available
* @return M4ERR_FILE_BAD_MODE_ACCESS the file mode access is not correct (it must be either isTextMode or read)
* @return M4ERR_FILE_NOT_FOUND The file can not be opened.
******************************************************************************
*/
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
M4OSA_ERR M4OSA_fileReadOpen_optim(M4OSA_Context* pContext,
M4OSA_Void* pFileDescriptor,
M4OSA_UInt32 FileModeAccess)
#else
M4OSA_ERR M4OSA_fileReadOpen_optim(M4OSA_Context* pContext,
M4OSA_Void* pFileDescriptor,
M4OSA_UInt32 FileModeAccess,
M4OSA_FileSystem_FctPtr *FS)
#endif
{
M4OSA_FileReader_Context_optim* apContext = M4OSA_NULL;
M4OSA_ERR err = M4NO_ERROR;
M4OSA_Void* aFileDesc = M4OSA_NULL;
M4OSA_Bool buffers_allocated = M4OSA_FALSE;
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
M4OSA_ERR errno = M4NO_ERROR;
#else
M4OSA_UInt16 errno;
#endif /*M4OSA_READER_OPTIM_USE_OSAL_IF*/
M4OSA_TRACE2_3("M4OSA_fileReadOpen_optim p = 0x%p fd = %s mode = %lu", pContext,
pFileDescriptor, FileModeAccess);
/* Check input parameters */
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_PARAMETER, pContext);
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_PARAMETER, pFileDescriptor);
*pContext = M4OSA_NULL;
/* Allocate memory for the File reader context. */
apContext = (M4OSA_FileReader_Context_optim *)M4OSA_32bitAlignedMalloc(sizeof(M4OSA_FileReader_Context_optim),
M4OSA_FILE_READER, (M4OSA_Char *)"M4OSA_FileReader_Context_optim");
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_ALLOC, apContext);
/* Set filesystem interface */
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
/*Set the optimized functions, to be called by the user*/
apContext->FS = (M4OSA_FileReadPointer*) M4OSA_32bitAlignedMalloc(sizeof(M4OSA_FileReadPointer),
M4OSA_FILE_READER, (M4OSA_Char *)"M4OSA_FileReaderOptim_init");
if (M4OSA_NULL==apContext->FS)
{
M4OSA_TRACE1_0("M4OSA_FileReaderOptim_init - ERROR : allocation failed");
return M4ERR_ALLOC;
}
apContext->FS->openRead = M4OSA_fileReadOpen;
apContext->FS->readData = M4OSA_fileReadData;
apContext->FS->seek = M4OSA_fileReadSeek;
apContext->FS->closeRead = M4OSA_fileReadClose;
apContext->FS->setOption = M4OSA_fileReadSetOption;
apContext->FS->getOption = M4OSA_fileReadGetOption;
#else
apContext->FS = FS;
#endif
/* Verify access mode */
if ( ((FileModeAccess & M4OSA_kFileAppend) != 0)
|| ((FileModeAccess & M4OSA_kFileRead) == 0))
{
err = M4ERR_FILE_BAD_MODE_ACCESS;
goto cleanup;
}
/* Open file in read mode */
if((FileModeAccess & M4OSA_kFileCreate) != 0)
{
err = M4ERR_FILE_BAD_MODE_ACCESS;
}
else
{
if ((FileModeAccess & M4OSA_kFileRead))
{
/* File is opened in read only*/
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
errno = apContext->FS->openRead(&aFileDesc, pFileDescriptor, FileModeAccess);
if ((aFileDesc == M4OSA_NULL)||(M4NO_ERROR != errno))
{
/* converts the error to PSW format*/
err = errno;
M4OSA_TRACE2_1("M4OSA_fileReadOpen_optim ERR1 = 0x%x", err);
apContext->IsOpened = M4OSA_FALSE;
}
#else
aFileDesc = apContext->FS->pFctPtr_Open(pFileDescriptor, FileModeAccess, &errno);
if (aFileDesc == M4OSA_NULL)
{
/* converts the error to PSW format*/
err = M4OSA_ERR_CREATE(M4_ERR, M4OSA_FILE_READER, errno);
M4OSA_TRACE2_1("M4OSA_fileReadOpen_optim ERR1 = 0x%x", err);
apContext->IsOpened = M4OSA_FALSE;
}
#endif
else
{
apContext->IsOpened = M4OSA_TRUE;
}
}
else
{
err = M4ERR_FILE_BAD_MODE_ACCESS;
}
}
if (M4NO_ERROR != err) goto cleanup;
/* Allocate buffers */
err = M4OSA_FileReader_BufferInit(apContext);
buffers_allocated = M4OSA_TRUE;
if (M4NO_ERROR != err) goto cleanup;
/* Initialize parameters */
apContext->fileSize = 0;
apContext->absolutePos = 0;
apContext->readFilePos = 0;
/* Retrieve the File Descriptor*/
apContext->aFileDesc = aFileDesc;
/* Retrieve the File mode Access */
apContext->FileAttribute.modeAccess = (M4OSA_FileModeAccess) FileModeAccess;
/*Retrieve the File reader context */
*pContext= (M4OSA_Context)apContext;
/* Compute file size */
err = M4OSA_FileReader_CalculateSize(apContext);
if (M4NO_ERROR != err) goto cleanup;
return M4NO_ERROR;
cleanup:
/* free context */
if (M4OSA_NULL != apContext)
{
if(buffers_allocated == M4OSA_TRUE)
{
M4OSA_FileReader_BufferFree(apContext);
}
free( apContext);
*pContext = M4OSA_NULL;
}
M4OSA_TRACE2_1 ("M4OSA_fileReadOpen_optim: returns error 0x%0x", err)
return err;
}
/**
******************************************************************************
* @brief This method reads the 'size' bytes in the core file reader (selected by its 'context')
* and writes the data to the 'data' pointer. If 'size' byte can not be read in the core file reader,
* 'size' parameter is updated to match the correct number of read bytes.
* @param pContext: (IN) File reader context.
* @param pData : (OUT) Data pointer of the read data.
* @param pSize : (INOUT) Size of the data to read (in byte).
* @return M4NO_ERROR: there is no error
* @return M4ERR_PARAMETER pSize, fileDescriptor or pData is NULL
* @return M4ERR_ALLOC there is no more memory available
* @return M4ERR_BAD_CONTEXT provided context is not a valid one.
******************************************************************************
*/
M4OSA_ERR M4OSA_fileReadData_optim(M4OSA_Context pContext,M4OSA_MemAddr8 pData,
M4OSA_UInt32* pSize)
{
M4OSA_FileReader_Context_optim* apContext =
(M4OSA_FileReader_Context_optim*) pContext;
M4OSA_ERR err;
M4OSA_FilePosition aSize;
M4OSA_FilePosition copiedSize;
M4OSA_Int8 selected_buffer, current_buffer;
M4OSA_TRACE3_3("M4OSA_fileReadData_optim p = 0x%p d = 0x%p s = %lu",
pContext, pData, *pSize);
/* Check input parameters */
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_BAD_CONTEXT, apContext);
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_PARAMETER, pData);
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_PARAMETER, pSize);
if (apContext->IsOpened != M4OSA_TRUE)
{
return M4ERR_BAD_CONTEXT;
}
/* Prevent reading beyond EOF */
if((*pSize > 0) && (apContext->absolutePos >= apContext->fileSize))
{
copiedSize = 0;
err = M4WAR_NO_MORE_AU;
goto cleanup;
}
/* Check if data can be read from a buffer */
/* If not, fill one according to quantized positions */
copiedSize = 0;
err = M4NO_ERROR;
selected_buffer = M4OSA_FileReader_BufferMatch(apContext, apContext->absolutePos);
if(selected_buffer == M4OSA_READBUFFER_NONE)
{
selected_buffer = M4OSA_FileReader_BufferSelect(apContext, 0);
err = M4OSA_FileReader_BufferFill(apContext, selected_buffer,
apContext->absolutePos);
}
if(err != M4NO_ERROR)
{
if(err == M4WAR_NO_DATA_YET)
{
if (*pSize <= (M4OSA_UInt32)apContext->buffer[selected_buffer].size)
{
err = M4NO_ERROR;
}
else
{
copiedSize = (M4OSA_UInt32)apContext->buffer[selected_buffer].size;
/*copy the content into pData*/
M4OSA_FileReader_BufferCopy(apContext, selected_buffer,
apContext->absolutePos, copiedSize, pData);
goto cleanup;
}
}
else
{
goto cleanup;
}
}
M4OSA_TRACE3_3("read size = %lu buffer = %d pos = %ld", *pSize,
selected_buffer, apContext->absolutePos);
/* Copy buffer into pData */
while(((M4OSA_UInt32)copiedSize < *pSize) && (err == M4NO_ERROR))
{
aSize = M4OSA_FileReader_BufferCopy(apContext, selected_buffer,
apContext->absolutePos+copiedSize,
*pSize-copiedSize, pData+copiedSize);
copiedSize += aSize;
if(aSize == 0)
{
err = M4WAR_NO_DATA_YET;
}
else
{
if((M4OSA_UInt32)copiedSize < *pSize)
{
current_buffer = selected_buffer;
selected_buffer = M4OSA_FileReader_BufferMatch(apContext,
apContext->absolutePos+copiedSize);
if(selected_buffer == M4OSA_READBUFFER_NONE)
{
selected_buffer = M4OSA_FileReader_BufferSelect(apContext,
current_buffer);
err = M4OSA_FileReader_BufferFill(apContext, selected_buffer,
apContext->absolutePos+copiedSize);
if(err != M4NO_ERROR)
{
if(err == M4WAR_NO_DATA_YET)
{
/*If we got all the data that we wanted, we should return no error*/
if ((*pSize-copiedSize) <= (M4OSA_UInt32)apContext->buffer[selected_buffer].size)
{
err = M4NO_ERROR;
}
/*If we did not get enough data, we will return NO_DATA_YET*/
/*copy the data read*/
aSize = M4OSA_FileReader_BufferCopy(apContext, selected_buffer,
apContext->absolutePos+copiedSize,
*pSize-copiedSize, pData+copiedSize);
copiedSize += aSize;
/*we reached end of file, so stop trying to read*/
goto cleanup;
}
if (err == M4WAR_NO_MORE_AU)
{
err = M4WAR_NO_DATA_YET;
/*copy the data read*/
aSize = M4OSA_FileReader_BufferCopy(apContext, selected_buffer,
apContext->absolutePos+copiedSize,
*pSize-copiedSize, pData+copiedSize);
copiedSize += aSize;
/*we reached end of file, so stop trying to read*/
goto cleanup;
}
else
{
goto cleanup;
}
}
}
}
}
}
cleanup :
/* Update the new position of the pointer */
apContext->absolutePos = apContext->absolutePos + copiedSize;
if((err != M4NO_ERROR)&&(err!=M4WAR_NO_DATA_YET))
{
M4OSA_TRACE2_3("M4OSA_fileReadData_optim size = %ld copied = %ld err = 0x%x",
*pSize, copiedSize, err);
}
/* Effective copied size must be returned */
*pSize = copiedSize;
/* Read is done */
return err;
}
/**
******************************************************************************
* @brief This method seeks at the provided position in the core file reader (selected by its 'context').
* The position is related to the seekMode parameter it can be either :
* From the beginning (position MUST be positive) : end position = position
* From the end (position MUST be negative) : end position = file size + position
* From the current position (signed offset) : end position = current position + position.
* @param pContext: (IN) File reader context.
* @param SeekMode : (IN) Seek access mode.
* @param pPosition : (IN) Position in the file.
* @return M4NO_ERROR: there is no error
* @return M4ERR_PARAMETER Seekmode or fileDescriptor is NULL
* @return M4ERR_ALLOC there is no more memory available
* @return M4ERR_BAD_CONTEXT provided context is not a valid one.
* @return M4ERR_FILE_INVALID_POSITION the position cannot be reached.
******************************************************************************
*/
M4OSA_ERR M4OSA_fileReadSeek_optim( M4OSA_Context pContext, M4OSA_FileSeekAccessMode SeekMode,
M4OSA_FilePosition* pPosition)
{
M4OSA_FileReader_Context_optim* apContext = (M4OSA_FileReader_Context_optim*) pContext;
M4OSA_ERR err = M4NO_ERROR;
M4OSA_TRACE3_3("M4OSA_fileReadSeek_optim p = 0x%p mode = %d pos = %d", pContext,
SeekMode, *pPosition);
/* Check input parameters */
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_BAD_CONTEXT, apContext);
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_PARAMETER, pPosition);
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_PARAMETER, SeekMode);
if (apContext->IsOpened != M4OSA_TRUE)
{
return M4ERR_BAD_CONTEXT; /*< The context can not be correct */
}
/* Go to the desired position */
switch(SeekMode)
{
case M4OSA_kFileSeekBeginning :
if(*pPosition < 0) {
return M4ERR_PARAMETER; /**< Bad SeekAcess mode */
}
apContext->absolutePos = *pPosition;
*pPosition = apContext->absolutePos;
break;
case M4OSA_kFileSeekEnd :
if(*pPosition > 0) {
return M4ERR_PARAMETER; /**< Bad SeekAcess mode */
}
apContext->absolutePos = apContext->fileSize + *pPosition;
*pPosition = apContext->absolutePos;
break;
case M4OSA_kFileSeekCurrent :
if(((apContext->absolutePos + *pPosition) > apContext->fileSize) ||
((apContext->absolutePos + *pPosition) < 0)){
return M4ERR_PARAMETER; /**< Bad SeekAcess mode */
}
apContext->absolutePos = apContext->absolutePos + *pPosition;
*pPosition = apContext->absolutePos;
break;
default :
err = M4ERR_PARAMETER; /**< Bad SeekAcess mode */
break;
}
/* Return without error */
return err;
}
/**
******************************************************************************
* @brief This method asks the core file reader to close the file
* (associated to the context) and also frees the context.
* @param pContext: (IN) File reader context.
* @return M4NO_ERROR: there is no error
* @return M4ERR_BAD_CONTEXT provided context is not a valid one.
******************************************************************************
*/
M4OSA_ERR M4OSA_fileReadClose_optim(M4OSA_Context pContext)
{
M4OSA_FileReader_Context_optim* apContext = (M4OSA_FileReader_Context_optim*) pContext;
M4OSA_ERR err = M4NO_ERROR;
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
M4OSA_ERR errno = M4NO_ERROR;
#else
M4OSA_UInt16 errno;
#endif
M4OSA_TRACE2_1("M4OSA_fileReadClose_optim p = 0x%p", pContext );
/* Check input parameters */
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_BAD_CONTEXT, apContext);
if (apContext->IsOpened != M4OSA_TRUE)
{
return M4ERR_BAD_CONTEXT; /**< The context can not be correct */
}
/* buffer */
M4OSA_FileReader_BufferFree(apContext);
/* Close the file */
#ifdef M4OSA_READER_OPTIM_USE_OSAL_IF
errno = apContext->FS->closeRead(apContext->aFileDesc);
if (M4NO_ERROR != errno)
{
/* converts the error to PSW format*/
err = errno;
M4OSA_TRACE2_1("M4OSA_fileReadClose_optim ERR1 = 0x%x", err);
}
#else
aRet_Val = apContext->FS->pFctPtr_Close(apContext->aFileDesc, &errno);
if (aRet_Val != 0)
{
/* converts the error to PSW format*/
err = M4OSA_ERR_CREATE(M4_ERR, M4OSA_FILE_READER, errno);
M4OSA_TRACE2_1("M4OSA_fileReadClose_optim ERR1 = 0x%x", err);
}
#endif /*M4OSA_READER_OPTIM_USE_OSAL_IF*/
apContext->IsOpened = M4OSA_FALSE;
//>>>> GLM20090212 : set the low level function statically
if (apContext->FS != M4OSA_NULL)
{
free( apContext->FS);
}
//<<<< GLM20090212 : set the low level function statically
/* Free the context */
free(apContext);
/* Return without error */
return err;
}
/**
******************************************************************************
* @brief This is a dummy function required to maintain function pointer
* structure.
* @note This is a dummy function required to maintain function pointer
* structure.
* @param pContext: (IN) Execution context.
* @param OptionId : (IN) Id of the option to set.
* @param OptionValue : (IN) Value of the option.
* @return M4NO_ERROR: there is no error
******************************************************************************
*/
M4OSA_ERR M4OSA_fileReadSetOption_optim(M4OSA_Context pContext,
M4OSA_FileReadOptionID OptionID,
M4OSA_DataOption OptionValue)
{
M4OSA_ERR err = M4NO_ERROR;
return err;
}
/**
******************************************************************************
* @brief This method asks the core file reader to return the value associated
* with the optionID.The caller is responsible for allocating/de-allocating
* the memory of the value field.
* @note The options handled by the component depend on the implementation
* of the component.
* @param pContext: (IN) Execution context.
* @param OptionId : (IN) Id of the option to set.
* @param pOptionValue : (OUT) Value of the option.
* @return M4NO_ERROR: there is no error
* @return M4ERR_BAD_CONTEXT pContext is NULL
* @return M4ERR_BAD_OPTION_ID the option id is not valid.
* @return M4ERR_NOT_IMPLEMENTED The option is not implemented yet.
******************************************************************************
*/
M4OSA_ERR M4OSA_fileReadGetOption_optim(M4OSA_Context pContext,
M4OSA_FileReadOptionID OptionID,
M4OSA_DataOption* pOptionValue)
{
M4OSA_FileReader_Context_optim* apContext = (M4OSA_FileReader_Context_optim*) pContext;
M4OSA_ERR err = M4NO_ERROR;
/* Check input parameters */
M4ERR_CHECK_NULL_RETURN_VALUE(M4ERR_BAD_CONTEXT, apContext);
if (apContext->IsOpened != M4OSA_TRUE)
{
return M4ERR_BAD_CONTEXT; /**< The context can not be correct */
}
/* Get the desired option if it is avalaible */
switch(OptionID)
{
/* Get File Size */
case M4OSA_kFileReadGetFileSize:/**< Get size of the file, limited to 32 bit size */
(*(M4OSA_UInt32 *)pOptionValue) = apContext->fileSize;
break;
/* Check End of file Occurs */
case M4OSA_kFileReadIsEOF : /**< See if we are at the end of the file */
(*(M4OSA_Bool *)pOptionValue) = (apContext->absolutePos >= apContext->fileSize) ? M4OSA_TRUE : M4OSA_FALSE;
break;
/* Get File Position */
case M4OSA_kFileReadGetFilePosition : /**< Get file position */
*(M4OSA_FilePosition *)pOptionValue = apContext->absolutePos;
break;
/* Get Attribute */
case M4OSA_kFileReadGetFileAttribute : /**< Get the file attribute = access mode */
(*(M4OSA_FileAttribute *)pOptionValue).modeAccess = apContext->FileAttribute.modeAccess;
break;
default:
/**< Bad option ID */
err = M4ERR_BAD_OPTION_ID;
break;
}
/*Return without error */
return err;
}