mirror of
https://github.com/reactos/reactos.git
synced 2025-01-12 01:00:06 +00:00
c424146e2c
svn path=/branches/cmake-bringup/; revision=48236
909 lines
27 KiB
C
909 lines
27 KiB
C
/*****************************************************************************
|
|
* FullFAT - High Performance, Thread-Safe Embedded FAT File-System *
|
|
* Copyright (C) 2009 James Walmsley (james@worm.me.uk) *
|
|
* *
|
|
* This program is free software: you can redistribute it and/or modify *
|
|
* it under the terms of the GNU General Public License as published by *
|
|
* the Free Software Foundation, either version 3 of the License, or *
|
|
* (at your option) any later version. *
|
|
* *
|
|
* This program is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU General Public License *
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>. *
|
|
* *
|
|
* IMPORTANT NOTICE: *
|
|
* ================= *
|
|
* Alternative Licensing is available directly from the Copyright holder, *
|
|
* (James Walmsley). For more information consult LICENSING.TXT to obtain *
|
|
* a Commercial license. *
|
|
* *
|
|
* See RESTRICTIONS.TXT for extra restrictions on the use of FullFAT. *
|
|
* *
|
|
* Removing the above notice is illegal and will invalidate this license. *
|
|
*****************************************************************************
|
|
* See http://worm.me.uk/fullfat for more information. *
|
|
* Or http://fullfat.googlecode.com/ for latest releases and the wiki. *
|
|
*****************************************************************************/
|
|
|
|
/**
|
|
* @file ff_ioman.c
|
|
* @author James Walmsley
|
|
* @ingroup IOMAN
|
|
*
|
|
* @defgroup IOMAN I/O Manager
|
|
* @brief Handles IO buffers for FullFAT safely.
|
|
*
|
|
* Provides a simple static interface to the rest of FullFAT to manage
|
|
* buffers. It also defines the public interfaces for Creating and
|
|
* Destroying a FullFAT IO object.
|
|
**/
|
|
|
|
#include "ff_ioman.h" // Includes ff_types.h, ff_safety.h, <stdio.h>
|
|
#include "fat.h"
|
|
|
|
extern FF_T_UINT32 FF_FindFreeCluster (FF_IOMAN *pIoman);
|
|
extern FF_T_UINT32 FF_CountFreeClusters (FF_IOMAN *pIoman);
|
|
|
|
static void FF_IOMAN_InitBufferDescriptors(FF_IOMAN *pIoman);
|
|
|
|
/**
|
|
* @public
|
|
* @brief Creates an FF_IOMAN object, to initialise FullFAT
|
|
*
|
|
* @param pCacheMem Pointer to a buffer for the cache. (NULL if ok to Malloc).
|
|
* @param Size The size of the provided buffer, or size of the cache to be created.
|
|
* @param BlkSize The block size of devices to be attached. If in doubt use 512.
|
|
* @param pError Pointer to a signed byte for error checking. Can be NULL if not required.
|
|
* @param pError To be checked when a NULL pointer is returned.
|
|
*
|
|
* @return Returns a pointer to an FF_IOMAN type object. NULL on Error, check the contents of
|
|
* @return pError
|
|
**/
|
|
FF_IOMAN *FF_CreateIOMAN(FF_T_UINT8 *pCacheMem, FF_T_UINT32 Size, FF_T_UINT16 BlkSize, FF_ERROR *pError) {
|
|
|
|
FF_IOMAN *pIoman = NULL;
|
|
FF_T_UINT32 *pLong = NULL; // Force malloc to malloc memory on a 32-bit boundary.
|
|
#ifdef FF_PATH_CACHE
|
|
FF_T_UINT32 i;
|
|
#endif
|
|
|
|
if(pError) {
|
|
*pError = FF_ERR_NONE;
|
|
}
|
|
|
|
if((BlkSize % 512) != 0 || Size == 0) {
|
|
if(pError) {
|
|
*pError = FF_ERR_IOMAN_BAD_BLKSIZE;
|
|
}
|
|
return NULL; // BlkSize Size not a multiple of 512 > 0
|
|
}
|
|
|
|
if((Size % BlkSize) != 0 || Size == 0) {
|
|
if(pError) {
|
|
*pError = FF_ERR_IOMAN_BAD_MEMSIZE;
|
|
}
|
|
return NULL; // Memory Size not a multiple of BlkSize > 0
|
|
}
|
|
|
|
pIoman = (FF_IOMAN *) FF_Malloc(sizeof(FF_IOMAN));
|
|
|
|
if(!pIoman) { // Ensure malloc() succeeded.
|
|
if(pError) {
|
|
*pError = FF_ERR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// This is just a bit-mask, to use a byte to keep track of memory.
|
|
// pIoman->MemAllocation = 0x00; // Unset all allocation identifiers.
|
|
pIoman->pBlkDevice = NULL;
|
|
pIoman->pBuffers = NULL;
|
|
pIoman->pCacheMem = NULL;
|
|
pIoman->pPartition = NULL;
|
|
pIoman->pSemaphore = NULL;
|
|
|
|
pIoman->pPartition = (FF_PARTITION *) FF_Malloc(sizeof(FF_PARTITION));
|
|
if(pIoman->pPartition) { // If succeeded, flag that allocation.
|
|
pIoman->MemAllocation |= FF_IOMAN_ALLOC_PART;
|
|
pIoman->pPartition->LastFreeCluster = 0;
|
|
pIoman->pPartition->PartitionMounted = FF_FALSE; // This should be checked by FF_Open();
|
|
#ifdef FF_PATH_CACHE
|
|
pIoman->pPartition->PCIndex = 0;
|
|
for(i = 0; i < FF_PATH_CACHE_DEPTH; i++) {
|
|
pIoman->pPartition->PathCache[i].DirCluster = 0;
|
|
pIoman->pPartition->PathCache[i].Path[0] = '\0';
|
|
#ifdef FF_HASH_TABLE_SUPPORT
|
|
pIoman->pPartition->PathCache[i].pHashTable = FF_CreateHashTable();
|
|
pIoman->pPartition->PathCache[i].bHashed = FF_FALSE;
|
|
#endif
|
|
}
|
|
#endif
|
|
} else {
|
|
FF_DestroyIOMAN(pIoman);
|
|
return NULL;
|
|
}
|
|
|
|
pIoman->pBlkDevice = (FF_BLK_DEVICE *) FF_Malloc(sizeof(FF_BLK_DEVICE));
|
|
if(pIoman->pBlkDevice) { // If succeeded, flag that allocation.
|
|
pIoman->MemAllocation |= FF_IOMAN_ALLOC_BLKDEV;
|
|
|
|
// Make sure all pointers are NULL
|
|
pIoman->pBlkDevice->fnReadBlocks = NULL;
|
|
pIoman->pBlkDevice->fnWriteBlocks = NULL;
|
|
pIoman->pBlkDevice->pParam = NULL;
|
|
|
|
} else {
|
|
FF_DestroyIOMAN(pIoman);
|
|
return NULL;
|
|
}
|
|
|
|
// Organise the memory provided, or create our own!
|
|
if(pCacheMem) {
|
|
pIoman->pCacheMem = pCacheMem;
|
|
}else { // No-Cache buffer provided (malloc)
|
|
pLong = (FF_T_UINT32 *) FF_Malloc(Size);
|
|
pIoman->pCacheMem = (FF_T_UINT8 *) pLong;
|
|
if(!pIoman->pCacheMem) {
|
|
pIoman->MemAllocation |= FF_IOMAN_ALLOC_BUFFERS;
|
|
FF_DestroyIOMAN(pIoman);
|
|
return NULL;
|
|
}
|
|
|
|
}
|
|
|
|
pIoman->BlkSize = BlkSize;
|
|
pIoman->CacheSize = (FF_T_UINT16) (Size / BlkSize);
|
|
pIoman->FirstFile = NULL;
|
|
pIoman->Locks = 0;
|
|
|
|
/* Malloc() memory for buffer objects. (FullFAT never refers to a buffer directly
|
|
but uses buffer objects instead. Allows us to provide thread safety.
|
|
*/
|
|
pIoman->pBuffers = (FF_BUFFER *) FF_Malloc(sizeof(FF_BUFFER) * pIoman->CacheSize);
|
|
|
|
if(pIoman->pBuffers) {
|
|
pIoman->MemAllocation |= FF_IOMAN_ALLOC_BUFDESCR;
|
|
FF_IOMAN_InitBufferDescriptors(pIoman);
|
|
} else {
|
|
FF_DestroyIOMAN(pIoman);
|
|
}
|
|
|
|
// Finally create a Semaphore for Buffer Description modifications.
|
|
pIoman->pSemaphore = FF_CreateSemaphore();
|
|
|
|
return pIoman; // Sucess, return the created object.
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
* @brief Destroys an FF_IOMAN object, and frees all assigned memory.
|
|
*
|
|
* @param pIoman Pointer to an FF_IOMAN object, as returned from FF_CreateIOMAN.
|
|
*
|
|
* @return FF_ERR_NONE on sucess, or a documented error code on failure. (FF_ERR_NULL_POINTER)
|
|
*
|
|
**/
|
|
FF_ERROR FF_DestroyIOMAN(FF_IOMAN *pIoman) {
|
|
|
|
// Ensure no NULL pointer was provided.
|
|
if(!pIoman) {
|
|
return FF_ERR_NULL_POINTER;
|
|
}
|
|
|
|
// Ensure pPartition pointer was allocated.
|
|
if((pIoman->MemAllocation & FF_IOMAN_ALLOC_PART)) {
|
|
FF_Free(pIoman->pPartition);
|
|
}
|
|
|
|
// Ensure pBlkDevice pointer was allocated.
|
|
if((pIoman->MemAllocation & FF_IOMAN_ALLOC_BLKDEV)) {
|
|
FF_Free(pIoman->pBlkDevice);
|
|
}
|
|
|
|
// Ensure pBuffers pointer was allocated.
|
|
if((pIoman->MemAllocation & FF_IOMAN_ALLOC_BUFDESCR)) {
|
|
FF_Free(pIoman->pBuffers);
|
|
}
|
|
|
|
// Ensure pCacheMem pointer was allocated.
|
|
if((pIoman->MemAllocation & FF_IOMAN_ALLOC_BUFFERS)) {
|
|
FF_Free(pIoman->pCacheMem);
|
|
}
|
|
|
|
// Destroy any Semaphore that was created.
|
|
if(pIoman->pSemaphore) {
|
|
FF_DestroySemaphore(pIoman->pSemaphore);
|
|
}
|
|
|
|
// Finally free the FF_IOMAN object.
|
|
FF_Free(pIoman);
|
|
|
|
return FF_ERR_NONE;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
* @brief Initialises Buffer Descriptions as part of the FF_IOMAN object initialisation.
|
|
*
|
|
* @param pIoman IOMAN Object.
|
|
*
|
|
**/
|
|
static void FF_IOMAN_InitBufferDescriptors(FF_IOMAN *pIoman) {
|
|
FF_T_UINT16 i;
|
|
FF_BUFFER *pBuffer = pIoman->pBuffers;
|
|
pIoman->LastReplaced = 0;
|
|
for(i = 0; i < pIoman->CacheSize; i++) {
|
|
pBuffer->Mode = 0;
|
|
pBuffer->NumHandles = 0;
|
|
pBuffer->Persistance = 0;
|
|
pBuffer->LRU = 0;
|
|
pBuffer->Sector = 0;
|
|
pBuffer->pBuffer = (FF_T_UINT8 *)((pIoman->pCacheMem) + (pIoman->BlkSize * i));
|
|
pBuffer->Modified = FF_FALSE;
|
|
pBuffer->Valid = FF_FALSE;
|
|
pBuffer++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
* @brief Tests the Mode for validity.
|
|
*
|
|
* @param Mode Mode of buffer to check.
|
|
*
|
|
* @return FF_TRUE when valid, else FF_FALSE.
|
|
**/
|
|
/*static FF_T_BOOL FF_IOMAN_ModeValid(FF_T_UINT8 Mode) {
|
|
if(Mode == FF_MODE_READ || Mode == FF_MODE_WRITE) {
|
|
return FF_TRUE;
|
|
}
|
|
return FF_FALSE;
|
|
}*/
|
|
|
|
|
|
/**
|
|
* @private
|
|
* @brief Fills a buffer with the appropriate sector via the device driver.
|
|
*
|
|
* @param pIoman FF_IOMAN object.
|
|
* @param Sector LBA address of the sector to fetch.
|
|
* @param pBuffer Pointer to a byte-wise buffer to store the fetched data.
|
|
*
|
|
* @return FF_TRUE when valid, else FF_FALSE.
|
|
**/
|
|
static FF_ERROR FF_IOMAN_FillBuffer(FF_IOMAN *pIoman, FF_T_UINT32 Sector, FF_T_UINT8 *pBuffer) {
|
|
FF_T_SINT32 retVal = 0;
|
|
if(pIoman->pBlkDevice->fnReadBlocks) { // Make sure we don't execute a NULL.
|
|
do{
|
|
retVal = pIoman->pBlkDevice->fnReadBlocks(pBuffer, Sector, 1, pIoman->pBlkDevice->pParam);
|
|
if(retVal == FF_ERR_DRIVER_BUSY) {
|
|
FF_Sleep(FF_DRIVER_BUSY_SLEEP);
|
|
}
|
|
} while(retVal == FF_ERR_DRIVER_BUSY);
|
|
if(retVal < 0) {
|
|
return -1; // FF_ERR_DRIVER_FATAL_ERROR was returned Fail!
|
|
} else {
|
|
if(retVal == 1) {
|
|
return 0; // 1 Block was sucessfully read.
|
|
} else {
|
|
return -1; // 0 Blocks we're read, Error!
|
|
}
|
|
}
|
|
}
|
|
return -1; // error no device diver registered.
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
* @brief Flushes a buffer to the device driver.
|
|
*
|
|
* @param pIoman FF_IOMAN object.
|
|
* @param Sector LBA address of the sector to fetch.
|
|
* @param pBuffer Pointer to a byte-wise buffer to store the fetched data.
|
|
*
|
|
* @return FF_TRUE when valid, else FF_FALSE.
|
|
**/
|
|
static FF_ERROR FF_IOMAN_FlushBuffer(FF_IOMAN *pIoman, FF_T_UINT32 Sector, FF_T_UINT8 *pBuffer) {
|
|
FF_T_SINT32 retVal = 0;
|
|
if(pIoman->pBlkDevice->fnWriteBlocks) { // Make sure we don't execute a NULL.
|
|
do{
|
|
retVal = pIoman->pBlkDevice->fnWriteBlocks(pBuffer, Sector, 1, pIoman->pBlkDevice->pParam);
|
|
if(retVal == FF_ERR_DRIVER_BUSY) {
|
|
FF_Sleep(FF_DRIVER_BUSY_SLEEP);
|
|
}
|
|
} while(retVal == FF_ERR_DRIVER_BUSY);
|
|
if(retVal < 0) {
|
|
return -1; // FF_ERR_DRIVER_FATAL_ERROR was returned Fail!
|
|
} else {
|
|
if(retVal == 1) {
|
|
return FF_ERR_NONE; // 1 Block was sucessfully written.
|
|
} else {
|
|
return -1; // 0 Blocks we're written, Error!
|
|
}
|
|
}
|
|
}
|
|
return -1; // error no device diver registered.
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
* @brief Flushes all Write cache buffers with no active Handles.
|
|
*
|
|
* @param pIoman IOMAN Object.
|
|
*
|
|
* @return FF_ERR_NONE on Success.
|
|
**/
|
|
FF_ERROR FF_FlushCache(FF_IOMAN *pIoman) {
|
|
|
|
FF_T_UINT16 i,x;
|
|
|
|
if(!pIoman) {
|
|
return FF_ERR_NULL_POINTER;
|
|
}
|
|
|
|
FF_PendSemaphore(pIoman->pSemaphore);
|
|
{
|
|
for(i = 0; i < pIoman->CacheSize; i++) {
|
|
if((pIoman->pBuffers + i)->NumHandles == 0 && (pIoman->pBuffers + i)->Modified == FF_TRUE) {
|
|
|
|
FF_IOMAN_FlushBuffer(pIoman, (pIoman->pBuffers + i)->Sector, (pIoman->pBuffers + i)->pBuffer);
|
|
|
|
// Buffer has now been flushed, mark it as a read buffer and unmodified.
|
|
(pIoman->pBuffers + i)->Mode = FF_MODE_READ;
|
|
(pIoman->pBuffers + i)->Modified = FF_FALSE;
|
|
|
|
// Search for other buffers that used this sector, and mark them as modified
|
|
// So that further requests will result in the new sector being fetched.
|
|
for(x = 0; x < pIoman->CacheSize; x++) {
|
|
if(x != i) {
|
|
if((pIoman->pBuffers + x)->Sector == (pIoman->pBuffers + i)->Sector && (pIoman->pBuffers + x)->Mode == FF_MODE_READ) {
|
|
(pIoman->pBuffers + x)->Modified = FF_TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
|
|
return FF_ERR_NONE;
|
|
}
|
|
|
|
/*static FF_T_BOOL FF_isFATSector(FF_IOMAN *pIoman, FF_T_UINT32 Sector) {
|
|
if(Sector >= pIoman->pPartition->FatBeginLBA && Sector < (pIoman->pPartition->FatBeginLBA + pIoman->pPartition->ReservedSectors)) {
|
|
return FF_TRUE;
|
|
}
|
|
return FF_FALSE;
|
|
}*/
|
|
|
|
FF_BUFFER *FF_GetBuffer(FF_IOMAN *pIoman, FF_T_UINT32 Sector, FF_T_UINT8 Mode) {
|
|
FF_BUFFER *pBuffer;
|
|
FF_BUFFER *pBufLRU = NULL;
|
|
FF_BUFFER *pBufLHITS = NULL;
|
|
FF_BUFFER *pBufMatch = NULL;
|
|
|
|
FF_T_UINT32 i;
|
|
|
|
while(!pBufMatch) {
|
|
FF_PendSemaphore(pIoman->pSemaphore);
|
|
{
|
|
for(i = 0; i < pIoman->CacheSize; i++) {
|
|
pBuffer = (pIoman->pBuffers + i);
|
|
if(pBuffer->Sector == Sector && pBuffer->Valid == FF_TRUE) {
|
|
pBufMatch = pBuffer;
|
|
} else {
|
|
if(pBuffer->NumHandles == 0) {
|
|
pBuffer->LRU += 1;
|
|
|
|
if(!pBufLRU) {
|
|
pBufLRU = pBuffer;
|
|
}
|
|
if(!pBufLHITS) {
|
|
pBufLHITS = pBuffer;
|
|
}
|
|
|
|
if(pBuffer->LRU >= pBufLRU->LRU) {
|
|
if(pBuffer->LRU == pBufLRU->LRU) {
|
|
if(pBuffer->Persistance > pBufLRU->Persistance) {
|
|
pBufLRU = pBuffer;
|
|
}
|
|
} else {
|
|
pBufLRU = pBuffer;
|
|
}
|
|
}
|
|
|
|
if(pBuffer->Persistance < pBufLHITS->Persistance) {
|
|
pBufLHITS = pBuffer;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(pBufMatch) {
|
|
// A Match was found process!
|
|
if(Mode == FF_MODE_READ && pBufMatch->Mode == FF_MODE_READ) {
|
|
pBufMatch->NumHandles += 1;
|
|
pBufMatch->Persistance += 1;
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
return pBufMatch;
|
|
}
|
|
|
|
if(pBufMatch->Mode == FF_MODE_WRITE && pBufMatch->NumHandles == 0) { // This buffer has no attached handles.
|
|
pBufMatch->Mode = Mode;
|
|
pBufMatch->NumHandles = 1;
|
|
pBufMatch->Persistance += 1;
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
return pBufMatch;
|
|
}
|
|
|
|
if(pBufMatch->Mode == FF_MODE_READ && Mode == FF_MODE_WRITE && pBufMatch->NumHandles == 0) {
|
|
pBufMatch->Mode = Mode;
|
|
pBufMatch->Modified = FF_TRUE;
|
|
pBufMatch->NumHandles = 1;
|
|
pBufMatch->Persistance += 1;
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
return pBufMatch;
|
|
}
|
|
|
|
pBufMatch = NULL; // Sector is already in use, keep yielding until its available!
|
|
|
|
} else {
|
|
// Choose a suitable buffer!
|
|
if(pBufLRU) {
|
|
// Process the suitable candidate.
|
|
if(pBufLRU->Modified == FF_TRUE) {
|
|
FF_IOMAN_FlushBuffer(pIoman, pBufLRU->Sector, pBufLRU->pBuffer);
|
|
}
|
|
pBufLRU->Mode = Mode;
|
|
pBufLRU->Persistance = 1;
|
|
pBufLRU->LRU = 0;
|
|
pBufLRU->NumHandles = 1;
|
|
pBufLRU->Sector = Sector;
|
|
|
|
if(Mode == FF_MODE_WRITE) {
|
|
pBufLRU->Modified = FF_TRUE;
|
|
} else {
|
|
pBufLRU->Modified = FF_FALSE;
|
|
}
|
|
|
|
FF_IOMAN_FillBuffer(pIoman, Sector, pBufLRU->pBuffer);
|
|
pBufLRU->Valid = FF_TRUE;
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
return pBufLRU;
|
|
}
|
|
|
|
}
|
|
}
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
FF_Yield();
|
|
}
|
|
|
|
return pBufMatch; // Return the Matched Buffer!
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
* @brief Releases a buffer resource.
|
|
*
|
|
* @param pIoman Pointer to an FF_IOMAN object.
|
|
* @param pBuffer Pointer to an FF_BUFFER object.
|
|
*
|
|
**/
|
|
void FF_ReleaseBuffer(FF_IOMAN *pIoman, FF_BUFFER *pBuffer) {
|
|
// Protect description changes with a semaphore.
|
|
FF_PendSemaphore(pIoman->pSemaphore);
|
|
{
|
|
pBuffer->NumHandles--;
|
|
}
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
* @brief Registers a device driver with FullFAT
|
|
*
|
|
* The device drivers must adhere to the specification provided by
|
|
* FF_WRITE_BLOCKS and FF_READ_BLOCKS.
|
|
*
|
|
* @param pIoman FF_IOMAN object.
|
|
* @param BlkSize Block Size that the driver deals in. (Minimum 512, larger values must be a multiple of 512).
|
|
* @param fnWriteBlocks Pointer to the Write Blocks to device function, as described by FF_WRITE_BLOCKS.
|
|
* @param fnReadBlocks Pointer to the Read Blocks from device function, as described by FF_READ_BLOCKS.
|
|
* @param pParam Pointer to a parameter for use in the functions.
|
|
*
|
|
* @return 0 on success, FF_ERR_IOMAN_DEV_ALREADY_REGD if a device was already hooked, FF_ERR_IOMAN_NULL_POINTER if a pIoman object wasn't provided.
|
|
**/
|
|
FF_ERROR FF_RegisterBlkDevice(FF_IOMAN *pIoman, FF_T_UINT16 BlkSize, FF_WRITE_BLOCKS fnWriteBlocks, FF_READ_BLOCKS fnReadBlocks, void *pParam) {
|
|
if(!pIoman) { // We can't do anything without an IOMAN object.
|
|
return FF_ERR_NULL_POINTER;
|
|
}
|
|
|
|
if((BlkSize % 512) != 0 || BlkSize == 0) {
|
|
return FF_ERR_IOMAN_DEV_INVALID_BLKSIZE; // BlkSize Size not a multiple of IOMAN's Expected BlockSize > 0
|
|
}
|
|
|
|
if((BlkSize % pIoman->BlkSize) != 0 || BlkSize == 0) {
|
|
return FF_ERR_IOMAN_DEV_INVALID_BLKSIZE; // BlkSize Size not a multiple of IOMAN's Expected BlockSize > 0
|
|
}
|
|
|
|
// Ensure that a device cannot be re-registered "mid-flight"
|
|
// Doing so would corrupt the context of FullFAT
|
|
if(pIoman->pBlkDevice->fnReadBlocks) {
|
|
return FF_ERR_IOMAN_DEV_ALREADY_REGD;
|
|
}
|
|
if(pIoman->pBlkDevice->fnWriteBlocks) {
|
|
return FF_ERR_IOMAN_DEV_ALREADY_REGD;
|
|
}
|
|
if(pIoman->pBlkDevice->pParam) {
|
|
return FF_ERR_IOMAN_DEV_ALREADY_REGD;
|
|
}
|
|
|
|
// Here we shall just set the values.
|
|
// FullFAT checks before using any of these values.
|
|
pIoman->pBlkDevice->devBlkSize = BlkSize;
|
|
pIoman->pBlkDevice->fnReadBlocks = fnReadBlocks;
|
|
pIoman->pBlkDevice->fnWriteBlocks = fnWriteBlocks;
|
|
pIoman->pBlkDevice->pParam = pParam;
|
|
|
|
return FF_ERR_NONE; // Success
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
**/
|
|
static FF_ERROR FF_DetermineFatType(FF_IOMAN *pIoman) {
|
|
|
|
FF_PARTITION *pPart;
|
|
FF_BUFFER *pBuffer;
|
|
FF_T_UINT32 testLong;
|
|
if(pIoman) {
|
|
pPart = pIoman->pPartition;
|
|
|
|
if(pPart->NumClusters < 4085) {
|
|
// FAT12
|
|
pPart->Type = FF_T_FAT12;
|
|
#ifdef FF_FAT_CHECK
|
|
#ifdef FF_FAT12_SUPPORT
|
|
pBuffer = FF_GetBuffer(pIoman, pIoman->pPartition->FatBeginLBA, FF_MODE_READ);
|
|
{
|
|
if(!pBuffer) {
|
|
return FF_ERR_DEVICE_DRIVER_FAILED;
|
|
}
|
|
testLong = (FF_T_UINT32) FF_getShort(pBuffer->pBuffer, 0x0000);
|
|
}
|
|
FF_ReleaseBuffer(pIoman, pBuffer);
|
|
if((testLong & 0x3FF) != 0x3F8) {
|
|
return FF_ERR_IOMAN_NOT_FAT_FORMATTED;
|
|
}
|
|
#else
|
|
return FF_ERR_IOMAN_NOT_FAT_FORMATTED;
|
|
#endif
|
|
#endif
|
|
#ifdef FF_FAT12_SUPPORT
|
|
return FF_ERR_NONE;
|
|
#endif
|
|
|
|
} else if(pPart->NumClusters < 65525) {
|
|
// FAT 16
|
|
pPart->Type = FF_T_FAT16;
|
|
#ifdef FF_FAT_CHECK
|
|
pBuffer = FF_GetBuffer(pIoman, pIoman->pPartition->FatBeginLBA, FF_MODE_READ);
|
|
{
|
|
if(!pBuffer) {
|
|
return FF_ERR_DEVICE_DRIVER_FAILED;
|
|
}
|
|
testLong = (FF_T_UINT32) FF_getShort(pBuffer->pBuffer, 0x0000);
|
|
}
|
|
FF_ReleaseBuffer(pIoman, pBuffer);
|
|
if(testLong != 0xFFF8) {
|
|
return FF_ERR_IOMAN_NOT_FAT_FORMATTED;
|
|
}
|
|
#endif
|
|
return FF_ERR_NONE;
|
|
}
|
|
else {
|
|
// FAT 32!
|
|
pPart->Type = FF_T_FAT32;
|
|
#ifdef FF_FAT_CHECK
|
|
pBuffer = FF_GetBuffer(pIoman, pIoman->pPartition->FatBeginLBA, FF_MODE_READ);
|
|
{
|
|
if(!pBuffer) {
|
|
return FF_ERR_DEVICE_DRIVER_FAILED;
|
|
}
|
|
testLong = FF_getLong(pBuffer->pBuffer, 0x0000);
|
|
}
|
|
FF_ReleaseBuffer(pIoman, pBuffer);
|
|
if((testLong & 0x0FFFFFF8) != 0x0FFFFFF8) {
|
|
return FF_ERR_IOMAN_NOT_FAT_FORMATTED;
|
|
}
|
|
#endif
|
|
return FF_ERR_NONE;
|
|
}
|
|
}
|
|
|
|
return FF_ERR_IOMAN_NOT_FAT_FORMATTED;
|
|
}
|
|
/**
|
|
* @public
|
|
* @brief Mounts the Specified partition, the volume specified by the FF_IOMAN object provided.
|
|
*
|
|
* The device drivers must adhere to the specification provided by
|
|
* FF_WRITE_BLOCKS and FF_READ_BLOCKS.
|
|
*
|
|
* @param pIoman FF_IOMAN object.
|
|
* @param PartitionNumber The primary partition number to be mounted. (0 - 3).
|
|
*
|
|
* @return 0 on success.
|
|
* @return FF_ERR_NULL_POINTER if a pIoman object wasn't provided.
|
|
* @return FF_ERR_IOMAN_INVALID_PARTITION_NUM if the partition number is out of range.
|
|
* @return FF_ERR_IOMAN_NO_MOUNTABLE_PARTITION if no partition was found.
|
|
* @return FF_ERR_IOMAN_INVALID_FORMAT if the master boot record or partition boot block didn't provide sensible data.
|
|
* @return FF_ERR_IOMAN_NOT_FAT_FORMATTED if the volume or partition couldn't be determined to be FAT. (@see ff_config.h)
|
|
*
|
|
**/
|
|
FF_ERROR FF_MountPartition(FF_IOMAN *pIoman, FF_T_UINT8 PartitionNumber) {
|
|
FF_PARTITION *pPart;
|
|
FF_BUFFER *pBuffer = 0;
|
|
|
|
if(!pIoman) {
|
|
return FF_ERR_NULL_POINTER;
|
|
}
|
|
|
|
if(PartitionNumber > 3) {
|
|
return FF_ERR_IOMAN_INVALID_PARTITION_NUM;
|
|
}
|
|
|
|
pPart = pIoman->pPartition;
|
|
|
|
pBuffer = FF_GetBuffer(pIoman, 0, FF_MODE_READ);
|
|
if(!pBuffer) {
|
|
return FF_ERR_DEVICE_DRIVER_FAILED;
|
|
}
|
|
pPart->BlkSize = FF_getShort(pBuffer->pBuffer, FF_FAT_BYTES_PER_SECTOR);
|
|
|
|
if((pPart->BlkSize % 512) == 0 && pPart->BlkSize > 0) {
|
|
// Volume is not partitioned (MBR Found)
|
|
pPart->BeginLBA = 0;
|
|
} else {
|
|
// Primary Partitions to deal with!
|
|
pPart->BeginLBA = FF_getLong(pBuffer->pBuffer, (FF_T_UINT16)(FF_FAT_PTBL + FF_FAT_PTBL_LBA + (16 * PartitionNumber)));
|
|
FF_ReleaseBuffer(pIoman, pBuffer);
|
|
|
|
if(!pPart->BeginLBA) {
|
|
return FF_ERR_IOMAN_NO_MOUNTABLE_PARTITION;
|
|
}
|
|
// Now we get the Partition sector.
|
|
pBuffer = FF_GetBuffer(pIoman, pPart->BeginLBA, FF_MODE_READ);
|
|
if(!pBuffer) {
|
|
return FF_ERR_DEVICE_DRIVER_FAILED;
|
|
}
|
|
pPart->BlkSize = FF_getShort(pBuffer->pBuffer, FF_FAT_BYTES_PER_SECTOR);
|
|
if((pPart->BlkSize % 512) != 0 || pPart->BlkSize == 0) {
|
|
FF_ReleaseBuffer(pIoman, pBuffer);
|
|
return FF_ERR_IOMAN_INVALID_FORMAT;
|
|
}
|
|
}
|
|
// Assume FAT16, then we'll adjust if its FAT32
|
|
pPart->ReservedSectors = FF_getShort(pBuffer->pBuffer, FF_FAT_RESERVED_SECTORS);
|
|
pPart->FatBeginLBA = pPart->BeginLBA + pPart->ReservedSectors;
|
|
|
|
pPart->NumFATS = (FF_T_UINT8) FF_getShort(pBuffer->pBuffer, FF_FAT_NUMBER_OF_FATS);
|
|
pPart->SectorsPerFAT = (FF_T_UINT32) FF_getShort(pBuffer->pBuffer, FF_FAT_16_SECTORS_PER_FAT);
|
|
|
|
pPart->SectorsPerCluster = FF_getChar(pBuffer->pBuffer, FF_FAT_SECTORS_PER_CLUS);
|
|
|
|
pPart->BlkFactor = (FF_T_UINT8) (pPart->BlkSize / pIoman->BlkSize); // Set the BlockFactor (How many real-blocks in a fake block!).
|
|
|
|
if(pPart->SectorsPerFAT == 0) { // FAT32
|
|
pPart->SectorsPerFAT = FF_getLong(pBuffer->pBuffer, FF_FAT_32_SECTORS_PER_FAT);
|
|
pPart->RootDirCluster = FF_getLong(pBuffer->pBuffer, FF_FAT_ROOT_DIR_CLUSTER);
|
|
pPart->ClusterBeginLBA = pPart->BeginLBA + pPart->ReservedSectors + (pPart->NumFATS * pPart->SectorsPerFAT);
|
|
pPart->TotalSectors = (FF_T_UINT32) FF_getShort(pBuffer->pBuffer, FF_FAT_16_TOTAL_SECTORS);
|
|
if(pPart->TotalSectors == 0) {
|
|
pPart->TotalSectors = FF_getLong(pBuffer->pBuffer, FF_FAT_32_TOTAL_SECTORS);
|
|
}
|
|
} else { // FAT16
|
|
pPart->ClusterBeginLBA = pPart->BeginLBA + pPart->ReservedSectors + (pPart->NumFATS * pPart->SectorsPerFAT);
|
|
pPart->TotalSectors = (FF_T_UINT32) FF_getShort(pBuffer->pBuffer, FF_FAT_16_TOTAL_SECTORS);
|
|
pPart->RootDirCluster = 1; // 1st Cluster is RootDir!
|
|
if(pPart->TotalSectors == 0) {
|
|
pPart->TotalSectors = FF_getLong(pBuffer->pBuffer, FF_FAT_32_TOTAL_SECTORS);
|
|
}
|
|
}
|
|
|
|
FF_ReleaseBuffer(pIoman, pBuffer); // Release the buffer finally!
|
|
pPart->RootDirSectors = ((FF_getShort(pBuffer->pBuffer, FF_FAT_ROOT_ENTRY_COUNT) * 32) + pPart->BlkSize - 1) / pPart->BlkSize;
|
|
pPart->FirstDataSector = pPart->ClusterBeginLBA + pPart->RootDirSectors;
|
|
pPart->DataSectors = pPart->TotalSectors - (pPart->ReservedSectors + (pPart->NumFATS * pPart->SectorsPerFAT) + pPart->RootDirSectors);
|
|
pPart->NumClusters = pPart->DataSectors / pPart->SectorsPerCluster;
|
|
|
|
if(FF_DetermineFatType(pIoman)) {
|
|
return FF_ERR_IOMAN_NOT_FAT_FORMATTED;
|
|
}
|
|
|
|
#ifdef FF_MOUNT_FIND_FREE
|
|
pPart->LastFreeCluster = FF_FindFreeCluster(pIoman);
|
|
pPart->FreeClusterCount = FF_CountFreeClusters(pIoman);
|
|
#else
|
|
pPart->LastFreeCluster = 0;
|
|
pPart->FreeClusterCount = 0;
|
|
#endif
|
|
|
|
return FF_ERR_NONE;
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
* @brief Unregister a Blockdevice, so that the IOMAN can be re-used for another device.
|
|
*
|
|
* Any active partitions must be Unmounted first.
|
|
*
|
|
* @param pIoman FF_IOMAN object.
|
|
*
|
|
* @return FF_ERR_NONE on success.
|
|
**/
|
|
FF_ERROR FF_UnregisterBlkDevice(FF_IOMAN *pIoman) {
|
|
|
|
FF_T_SINT8 RetVal = FF_ERR_NONE;
|
|
|
|
if(!pIoman) {
|
|
return FF_ERR_NULL_POINTER;
|
|
}
|
|
|
|
FF_PendSemaphore(pIoman->pSemaphore);
|
|
{
|
|
if(pIoman->pPartition->PartitionMounted == FF_FALSE) {
|
|
pIoman->pBlkDevice->devBlkSize = 0;
|
|
pIoman->pBlkDevice->fnReadBlocks = NULL;
|
|
pIoman->pBlkDevice->fnWriteBlocks = NULL;
|
|
pIoman->pBlkDevice->pParam = NULL;
|
|
} else {
|
|
RetVal = FF_ERR_IOMAN_PARTITION_MOUNTED;
|
|
}
|
|
}
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
|
|
return RetVal;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
* @brief Checks the cache for Active Handles
|
|
*
|
|
* @param pIoman FF_IOMAN Object.
|
|
*
|
|
* @return FF_TRUE if an active handle is found, else FF_FALSE.
|
|
*
|
|
* @pre This function must be wrapped with the cache handling semaphore.
|
|
**/
|
|
static FF_T_BOOL FF_ActiveHandles(FF_IOMAN *pIoman) {
|
|
FF_T_UINT32 i;
|
|
FF_BUFFER *pBuffer;
|
|
|
|
for(i = 0; i < pIoman->CacheSize; i++) {
|
|
pBuffer = (pIoman->pBuffers + i);
|
|
if(pBuffer->NumHandles) {
|
|
return FF_TRUE;
|
|
}
|
|
}
|
|
|
|
return FF_FALSE;
|
|
}
|
|
|
|
|
|
/**
|
|
* @public
|
|
* @brief Unmounts the active partition.
|
|
*
|
|
* @param pIoman FF_IOMAN Object.
|
|
*
|
|
* @return FF_ERR_NONE on success.
|
|
**/
|
|
FF_ERROR FF_UnmountPartition(FF_IOMAN *pIoman) {
|
|
FF_T_SINT8 RetVal = FF_ERR_NONE;
|
|
|
|
if(!pIoman) {
|
|
return FF_ERR_NULL_POINTER;
|
|
}
|
|
|
|
FF_PendSemaphore(pIoman->pSemaphore); // Ensure that there are no File Handles
|
|
{
|
|
if(!FF_ActiveHandles(pIoman)) {
|
|
if(pIoman->FirstFile == NULL) {
|
|
FF_FlushCache(pIoman); // Flush any unwritten sectors to disk.
|
|
pIoman->pPartition->PartitionMounted = FF_FALSE;
|
|
} else {
|
|
RetVal = FF_ERR_IOMAN_ACTIVE_HANDLES;
|
|
}
|
|
} else {
|
|
RetVal = FF_ERR_IOMAN_ACTIVE_HANDLES; // Active handles found on the cache.
|
|
}
|
|
}
|
|
FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
|
|
return RetVal;
|
|
}
|
|
|
|
|
|
FF_ERROR FF_IncreaseFreeClusters(FF_IOMAN *pIoman, FF_T_UINT32 Count) {
|
|
|
|
//FF_PendSemaphore(pIoman->pSemaphore);
|
|
//{
|
|
if(!pIoman->pPartition->FreeClusterCount) {
|
|
pIoman->pPartition->FreeClusterCount = FF_CountFreeClusters(pIoman);
|
|
}
|
|
pIoman->pPartition->FreeClusterCount += Count;
|
|
//}
|
|
//FF_ReleaseSemaphore(pIoman->pSemaphore);
|
|
|
|
return FF_ERR_NONE;
|
|
}
|
|
|
|
FF_ERROR FF_DecreaseFreeClusters(FF_IOMAN *pIoman, FF_T_UINT32 Count) {
|
|
|
|
//FF_lockFAT(pIoman);
|
|
//{
|
|
if(!pIoman->pPartition->FreeClusterCount) {
|
|
pIoman->pPartition->FreeClusterCount = FF_CountFreeClusters(pIoman);
|
|
}
|
|
pIoman->pPartition->FreeClusterCount -= Count;
|
|
//}
|
|
//FF_unlockFAT(pIoman);
|
|
|
|
return FF_ERR_NONE;
|
|
}
|
|
|
|
|
|
/**
|
|
* @brief Returns the Block-size of a mounted Partition
|
|
*
|
|
* The purpose of this function is to provide API access to information
|
|
* that might be useful in special cases. Like USB sticks that require a sector
|
|
* knocking sequence for security. After the sector knock, some secure USB
|
|
* sticks then present a different BlockSize.
|
|
*
|
|
* @param pIoman FF_IOMAN Object returned from FF_CreateIOMAN()
|
|
*
|
|
* @return The blocksize of the partition. A value less than 0 when an error occurs.
|
|
* @return Any negative value can be cast to the FF_ERROR type.
|
|
**/
|
|
FF_T_SINT32 FF_GetPartitionBlockSize(FF_IOMAN *pIoman) {
|
|
|
|
if(pIoman) {
|
|
return (FF_T_SINT32) pIoman->pPartition->BlkSize;
|
|
}
|
|
|
|
return FF_ERR_NULL_POINTER;
|
|
}
|
|
|
|
#ifdef FF_64_NUM_SUPPORT
|
|
/**
|
|
* @brief Returns the number of bytes contained within the mounted partition or volume.
|
|
*
|
|
* @param pIoman FF_IOMAN Object returned from FF_CreateIOMAN()
|
|
*
|
|
* @return The total number of bytes that the mounted partition or volume contains.
|
|
*
|
|
**/
|
|
FF_T_UINT64 FF_GetVolumeSize(FF_IOMAN *pIoman) {
|
|
if(pIoman) {
|
|
FF_T_UINT32 TotalClusters = pIoman->pPartition->DataSectors / pIoman->pPartition->SectorsPerCluster;
|
|
return (FF_T_UINT64) ((FF_T_UINT64)TotalClusters * (FF_T_UINT64)((FF_T_UINT64)pIoman->pPartition->SectorsPerCluster * (FF_T_UINT64)pIoman->pPartition->BlkSize));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#else
|
|
FF_T_UINT32 FF_GetVolumeSize(FF_IOMAN *pIoman) {
|
|
FF_T_UINT32 TotalClusters = pIoman->pPartition->DataSectors / pIoman->pPartition->SectorsPerCluster;
|
|
return (FF_T_UINT32) (TotalClusters * (pIoman->pPartition->SectorsPerCluster * pIoman->pPartition->BlkSize));
|
|
}
|
|
#endif
|
|
|