| /* | |
| * 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 */ | |