blob: 3a6ca633a5e4fe0053cdf57af55cf2385e7d3c7a [file] [log] [blame]
/*
* FreeRTOS+FAT SL V1.0.1 (C) 2014 HCC Embedded
*
* The FreeRTOS+FAT SL license terms are different to the FreeRTOS license
* terms.
*
* FreeRTOS+FAT SL uses a dual license model that allows the software to be used
* under a standard GPL open source license, or a commercial license. The
* standard GPL license (unlike the modified GPL license under which FreeRTOS
* itself is distributed) requires that all software statically linked with
* FreeRTOS+FAT SL is also distributed under the same GPL V2 license terms.
* Details of both license options follow:
*
* - Open source licensing -
* FreeRTOS+FAT SL is a free download and may be used, modified, evaluated and
* distributed without charge provided the user adheres to version two of the
* GNU General Public License (GPL) and does not remove the copyright notice or
* this text. The GPL V2 text is available on the gnu.org web site, and on the
* following URL: http://www.FreeRTOS.org/gpl-2.0.txt.
*
* - Commercial licensing -
* Businesses and individuals who for commercial or other reasons cannot comply
* with the terms of the GPL V2 license must obtain a commercial license before
* incorporating FreeRTOS+FAT SL into proprietary software for distribution in
* any form. Commercial licenses can be purchased from
* http://shop.freertos.org/fat_sl and do not require any source files to be
* changed.
*
* FreeRTOS+FAT SL is distributed in the hope that it will be useful. You
* cannot use FreeRTOS+FAT SL unless you agree that you use the software 'as
* is'. FreeRTOS+FAT SL is provided WITHOUT ANY WARRANTY; without even the
* implied warranties of NON-INFRINGEMENT, MERCHANTABILITY or FITNESS FOR A
* PARTICULAR PURPOSE. Real Time Engineers Ltd. and HCC Embedded disclaims all
* conditions and terms, be they implied, expressed, or statutory.
*
* http://www.FreeRTOS.org
* http://www.FreeRTOS.org/FreeRTOS-Plus
*
*/
#define FS_MUTEX_DEFINED
#include "../../api/fat_sl.h"
#include "../../version/ver_fat_sl.h"
#if VER_FAT_SL_MAJOR != 5 || VER_FAT_SL_MINOR != 2
#error Incompatible FAT_SL version number!
#endif
#if F_FS_THREAD_AWARE == 1
xSemaphoreHandle fs_lock_semaphore;
/*
** fr_findfirst
**
** find first time a file using wildcards
**
** INPUT : filename - name of the file
** *find - pointer to a pre-define F_FIND structure
** RETURN: F_NOERR - on success
** F_ERR_NOTFOUND - if not found
*/
unsigned char fr_findfirst ( const char * filename, F_FIND * find )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_findfirst( filename, find );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_findnext
**
** find next time a file using wildcards
**
** INPUT : *find - pointer to a pre-define F_FIND structure
** RETURN: F_NOERR - on success
** F_ERR_NOTFOUND - if not found
*/
unsigned char fr_findnext ( F_FIND * find )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_findnext( find );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_filelength
**
** Get the length of a file
**
** INPUT : filename - name of the file
** RETURN: size of the file or F_ERR_INVALID if not exists or volume not working
*/
long fr_filelength ( const char * filename )
{
unsigned long rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_filelength( filename );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = 0;
}
return rc;
}
/*
** fr_open
**
** open a file
**
** INPUT : filename - file to be opened
** mode - open method (r,w,a,r+,w+,a+)
** RETURN: pointer to a file descriptor or 0 on error
*/
F_FILE * fr_open ( const char * filename, const char * mode )
{
F_FILE * rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_open( filename, mode );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = NULL;
}
return rc;
}
/*
** fr_close
**
** Close a previously opened file.
**
** INPUT : *filehandle - pointer to the file descriptor
** RETURN: F_NOERR on success, other if error
*/
unsigned char fr_close ( F_FILE * filehandle )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_close( filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_read
**
** Read from a file.
**
** INPUT : buf - buffer to read data
** size - number of unique
** size_st - size of unique
** *filehandle - pointer to file descriptor
** OUTPUT: number of read bytes
*/
long fr_read ( void * bbuf, long size, long size_st, F_FILE * filehandle )
{
long rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_read( bbuf, size, size_st, filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = 0;
}
return rc;
}
/*
** fr_write
**
** INPUT : bbuf - buffer to write from
** size - number of unique
** size_st - size of unique
** *filehandle - pointer to the file descriptor
** RETURN: number of written bytes
*/
long fr_write ( const void * bbuf, long size, long size_st, F_FILE * filehandle )
{
long rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_write( bbuf, size, size_st, filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = 0;
}
return rc;
}
/*
** fr_seek
**
** Seek in a file.
**
** INPUT : *filehandle - pointer to a file descriptor
** offset - offset
** whence - F_SEEK_SET: position = offset
** F_SEEK_CUR: position = position + offset
** F_SEEK_END: position = end of file (offset=0)
** RETURN: F_NOERR on succes, other if error.
*/
unsigned char fr_seek ( F_FILE * filehandle, long offset, unsigned char whence )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_seek( filehandle, offset, whence );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_tell
**
** get current position in the file
**
** INPUT : *filehandle - pointer to a file descriptor
** RETURN: -1 on error or current position.
*/
long fr_tell ( F_FILE * filehandle )
{
long rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_tell( filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = -1;
}
return rc;
}
/*
** fr_getc
**
** read one byte from a file
**
** INPUT : *filehandle - pointer to a file descriptor
** RETURN: -1 if error, otherwise the read character.
*/
int fr_getc ( F_FILE * filehandle )
{
int rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_getc( filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = -1;
}
return rc;
}
/*
** fr_putc
**
** write one byte to a file
**
** INPUT : ch - character to write
** *filehandle - pointer to a file handler
** RETURN: ch on success, -1 on error
*/
int fr_putc ( int ch, F_FILE * filehandle )
{
int rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_putc( ch, filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = -1;
}
return rc;
}
/*
** fr_rewind
**
** set current position in the file to the beginning
**
** INPUT : *filehandle - pointer to a file descriptor
** RETURN: F_NOERR on succes, other if error.
*/
unsigned char fr_rewind ( F_FILE * filehandle )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_rewind( filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_eof
**
** check if current position is at the end of the file.
**
** INPUT : *filehandle - pointer to a file descriptor
** RETURN: F_ERR_EOF - at the end of the file
** F_NOERR - no error, end of the file not reached
** other - on error
*/
unsigned char fr_eof ( F_FILE * filehandle )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_eof( filehandle );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** Format the device
*/
unsigned char fr_hardformat ( unsigned char fattype )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_hardformat( fattype );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_get_serial
**
** Get serial number
**
** OUTPUT: serial - where to write the serial number
** RETURN: error code
*/
unsigned char fr_getserial ( unsigned long * serial )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_getserial( serial );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_delete
**
** Delete a file. Removes the chain that belongs to the file and inserts a new descriptor
** to the directory with first_cluster set to 0.
**
** INPUT : filename - name of the file to delete
** RETURN: F_NOERR on success, other if error.
*/
unsigned char fr_delete ( const char * filename )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_delete( filename );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_truncate
**
** Open a file and set end of file
**
** INPUT: filename - name of the file
** filesize - required new size
** RETURN: NULL on error, otherwise file pointer
*/
F_FILE * fr_truncate ( const char * filename, long filesize )
{
F_FILE * f;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
f = fn_truncate( filename, filesize );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
f = NULL;
}
return f;
}
/*
** fr_getfreespace
**
** Get free space on the volume
**
** OUTPUT: *sp - pre-defined F_SPACE structure, where information will be stored
** RETURN: F_NOERR - on success
** F_ERR_NOTFORMATTED - if volume is not formatted
*/
unsigned char fr_getfreespace ( F_SPACE * sp )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_getfreespace( sp );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_chdir
**
** Change to a directory
**
** INPUT: path - path to the dircetory
** RETURN: 0 - on success, other if error
*/
unsigned char fr_chdir ( const char * path )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_chdir( path );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_mkdir
**
** Create a directory
**
** INPUT: path - new directory path
** RETURN: 0 - on success, other if error
*/
unsigned char fr_mkdir ( const char * path )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_mkdir( path );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_rmdir
**
** Removes a directory
**
** INPUT: path - path to remove
** RETURN: 0 - on success, other if error
*/
unsigned char fr_rmdir ( const char * path )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_rmdir( path );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_getcwd
**
** Get current working directory
**
** INPUT: maxlen - maximum length allowed
** OUTPUT: path - current working directory
** RETURN: 0 - on success, other if error
*/
unsigned char fr_getcwd ( char * path, unsigned char maxlen, char root )
{
unsigned char rc;
if( xSemaphoreTake( fs_lock_semaphore, F_MAX_LOCK_WAIT_TICKS ) == pdPASS )
{
rc = fn_getcwd( path, maxlen, root );
xSemaphoreGive( fs_lock_semaphore );
}
else
{
rc = F_ERR_OS;
}
return rc;
}
/*
** fr_init
**
** Initialize FAT_SL OS module
**
** RETURN: F_NO_ERROR or F_ERR_OS
*/
unsigned char fr_init ( void )
{
return F_NO_ERROR;
}
#endif /* F_FS_THREAD_AWARE */