mirror of https://github.com/ventoy/Ventoy.git
479 lines
16 KiB
C
479 lines
16 KiB
C
//-----------------------------------------------------------------------------
|
|
//-----------------------------------------------------------------------------
|
|
// FAT16/32 File IO Library
|
|
// V2.6
|
|
// Ultra-Embedded.com
|
|
// Copyright 2003 - 2012
|
|
//
|
|
// Email: admin@ultra-embedded.com
|
|
//
|
|
// License: GPL
|
|
// If you would like a version with a more permissive license for use in
|
|
// closed source commercial applications please contact me for details.
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
// This file is part of FAT File IO Library.
|
|
//
|
|
// FAT File IO Library 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 2 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// FAT File IO Library 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 FAT File IO Library; if not, write to the Free Software
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
//-----------------------------------------------------------------------------
|
|
//-----------------------------------------------------------------------------
|
|
#include <string.h>
|
|
#include "fat_defs.h"
|
|
#include "fat_access.h"
|
|
#include "fat_table.h"
|
|
|
|
#ifndef FAT_BUFFERS
|
|
#define FAT_BUFFERS 1
|
|
#endif
|
|
|
|
#ifndef FAT_BUFFER_SECTORS
|
|
#define FAT_BUFFER_SECTORS 1
|
|
#endif
|
|
|
|
#if FAT_BUFFERS < 1 || FAT_BUFFER_SECTORS < 1
|
|
#error "FAT_BUFFERS & FAT_BUFFER_SECTORS must be at least 1"
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// FAT Sector Buffer
|
|
//-----------------------------------------------------------------------------
|
|
#define FAT32_GET_32BIT_WORD(pbuf, location) ( GET_32BIT_WORD(pbuf->ptr, location) )
|
|
#define FAT32_SET_32BIT_WORD(pbuf, location, value) { SET_32BIT_WORD(pbuf->ptr, location, value); pbuf->dirty = 1; }
|
|
#define FAT16_GET_16BIT_WORD(pbuf, location) ( GET_16BIT_WORD(pbuf->ptr, location) )
|
|
#define FAT16_SET_16BIT_WORD(pbuf, location, value) { SET_16BIT_WORD(pbuf->ptr, location, value); pbuf->dirty = 1; }
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_fat_init:
|
|
//-----------------------------------------------------------------------------
|
|
void fatfs_fat_init(struct fatfs *fs)
|
|
{
|
|
int i;
|
|
|
|
// FAT buffer chain head
|
|
fs->fat_buffer_head = NULL;
|
|
|
|
for (i=0;i<FAT_BUFFERS;i++)
|
|
{
|
|
// Initialise buffers to invalid
|
|
fs->fat_buffers[i].address = FAT32_INVALID_CLUSTER;
|
|
fs->fat_buffers[i].dirty = 0;
|
|
memset(fs->fat_buffers[i].sector, 0x00, sizeof(fs->fat_buffers[i].sector));
|
|
fs->fat_buffers[i].ptr = NULL;
|
|
|
|
// Add to head of queue
|
|
fs->fat_buffers[i].next = fs->fat_buffer_head;
|
|
fs->fat_buffer_head = &fs->fat_buffers[i];
|
|
}
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_fat_writeback: Writeback 'dirty' FAT sectors to disk
|
|
//-----------------------------------------------------------------------------
|
|
static int fatfs_fat_writeback(struct fatfs *fs, struct fat_buffer *pcur)
|
|
{
|
|
if (pcur)
|
|
{
|
|
// Writeback sector if changed
|
|
if (pcur->dirty)
|
|
{
|
|
if (fs->disk_io.write_media)
|
|
{
|
|
uint32 sectors = FAT_BUFFER_SECTORS;
|
|
uint32 offset = pcur->address - fs->fat_begin_lba;
|
|
|
|
// Limit to sectors used for the FAT
|
|
if ((offset + FAT_BUFFER_SECTORS) <= fs->fat_sectors)
|
|
sectors = FAT_BUFFER_SECTORS;
|
|
else
|
|
sectors = fs->fat_sectors - offset;
|
|
|
|
if (!fs->disk_io.write_media(pcur->address, pcur->sector, sectors))
|
|
return 0;
|
|
}
|
|
|
|
pcur->dirty = 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_fat_read_sector: Read a FAT sector
|
|
//-----------------------------------------------------------------------------
|
|
static struct fat_buffer *fatfs_fat_read_sector(struct fatfs *fs, uint32 sector)
|
|
{
|
|
struct fat_buffer *last = NULL;
|
|
struct fat_buffer *pcur = fs->fat_buffer_head;
|
|
|
|
// Itterate through sector buffer list
|
|
while (pcur)
|
|
{
|
|
// Sector within this buffer?
|
|
if ((sector >= pcur->address) && (sector < (pcur->address + FAT_BUFFER_SECTORS)))
|
|
break;
|
|
|
|
// End of list?
|
|
if (pcur->next == NULL)
|
|
{
|
|
// Remove buffer from list
|
|
if (last)
|
|
last->next = NULL;
|
|
// We the first and last buffer in the chain?
|
|
else
|
|
fs->fat_buffer_head = NULL;
|
|
}
|
|
|
|
last = pcur;
|
|
pcur = pcur->next;
|
|
}
|
|
|
|
// We found the sector already in FAT buffer chain
|
|
if (pcur)
|
|
{
|
|
pcur->ptr = (uint8 *)(pcur->sector + ((sector - pcur->address) * FAT_SECTOR_SIZE));
|
|
return pcur;
|
|
}
|
|
|
|
// Else, we removed the last item from the list
|
|
pcur = last;
|
|
|
|
// Add to start of sector buffer list (now newest sector)
|
|
pcur->next = fs->fat_buffer_head;
|
|
fs->fat_buffer_head = pcur;
|
|
|
|
// Writeback sector if changed
|
|
if (pcur->dirty)
|
|
if (!fatfs_fat_writeback(fs, pcur))
|
|
return 0;
|
|
|
|
// Address is now new sector
|
|
pcur->address = sector;
|
|
|
|
// Read next sector
|
|
if (!fs->disk_io.read_media(pcur->address, pcur->sector, FAT_BUFFER_SECTORS))
|
|
{
|
|
// Read failed, invalidate buffer address
|
|
pcur->address = FAT32_INVALID_CLUSTER;
|
|
return NULL;
|
|
}
|
|
|
|
pcur->ptr = pcur->sector;
|
|
return pcur;
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_fat_purge: Purge 'dirty' FAT sectors to disk
|
|
//-----------------------------------------------------------------------------
|
|
int fatfs_fat_purge(struct fatfs *fs)
|
|
{
|
|
struct fat_buffer *pcur = fs->fat_buffer_head;
|
|
|
|
// Itterate through sector buffer list
|
|
while (pcur)
|
|
{
|
|
// Writeback sector if changed
|
|
if (pcur->dirty)
|
|
if (!fatfs_fat_writeback(fs, pcur))
|
|
return 0;
|
|
|
|
pcur = pcur->next;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// General FAT Table Operations
|
|
//-----------------------------------------------------------------------------
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_find_next_cluster: Return cluster number of next cluster in chain by
|
|
// reading FAT table and traversing it. Return 0xffffffff for end of chain.
|
|
//-----------------------------------------------------------------------------
|
|
uint32 fatfs_find_next_cluster(struct fatfs *fs, uint32 current_cluster)
|
|
{
|
|
uint32 fat_sector_offset, position;
|
|
uint32 nextcluster;
|
|
struct fat_buffer *pbuf;
|
|
|
|
// Why is '..' labelled with cluster 0 when it should be 2 ??
|
|
if (current_cluster == 0)
|
|
current_cluster = 2;
|
|
|
|
// Find which sector of FAT table to read
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
fat_sector_offset = current_cluster / 256;
|
|
else
|
|
fat_sector_offset = current_cluster / 128;
|
|
|
|
// Read FAT sector into buffer
|
|
pbuf = fatfs_fat_read_sector(fs, fs->fat_begin_lba+fat_sector_offset);
|
|
if (!pbuf)
|
|
return (FAT32_LAST_CLUSTER);
|
|
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
{
|
|
// Find 32 bit entry of current sector relating to cluster number
|
|
position = (current_cluster - (fat_sector_offset * 256)) * 2;
|
|
|
|
// Read Next Clusters value from Sector Buffer
|
|
nextcluster = FAT16_GET_16BIT_WORD(pbuf, (uint16)position);
|
|
|
|
// If end of chain found
|
|
if (nextcluster >= 0xFFF8 && nextcluster <= 0xFFFF)
|
|
return (FAT32_LAST_CLUSTER);
|
|
}
|
|
else
|
|
{
|
|
// Find 32 bit entry of current sector relating to cluster number
|
|
position = (current_cluster - (fat_sector_offset * 128)) * 4;
|
|
|
|
// Read Next Clusters value from Sector Buffer
|
|
nextcluster = FAT32_GET_32BIT_WORD(pbuf, (uint16)position);
|
|
|
|
// Mask out MS 4 bits (its 28bit addressing)
|
|
nextcluster = nextcluster & 0x0FFFFFFF;
|
|
|
|
// If end of chain found
|
|
if (nextcluster >= 0x0FFFFFF8 && nextcluster <= 0x0FFFFFFF)
|
|
return (FAT32_LAST_CLUSTER);
|
|
}
|
|
|
|
// Else return next cluster
|
|
return (nextcluster);
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_set_fs_info_next_free_cluster: Write the next free cluster to the FSINFO table
|
|
//-----------------------------------------------------------------------------
|
|
void fatfs_set_fs_info_next_free_cluster(struct fatfs *fs, uint32 newValue)
|
|
{
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
;
|
|
else
|
|
{
|
|
// Load sector to change it
|
|
struct fat_buffer *pbuf = fatfs_fat_read_sector(fs, fs->lba_begin+fs->fs_info_sector);
|
|
if (!pbuf)
|
|
return ;
|
|
|
|
// Change
|
|
FAT32_SET_32BIT_WORD(pbuf, 492, newValue);
|
|
fs->next_free_cluster = newValue;
|
|
|
|
// Write back FSINFO sector to disk
|
|
if (fs->disk_io.write_media)
|
|
fs->disk_io.write_media(pbuf->address, pbuf->sector, 1);
|
|
|
|
// Invalidate cache entry
|
|
pbuf->address = FAT32_INVALID_CLUSTER;
|
|
pbuf->dirty = 0;
|
|
}
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_find_blank_cluster: Find a free cluster entry by reading the FAT
|
|
//-----------------------------------------------------------------------------
|
|
#if FATFS_INC_WRITE_SUPPORT
|
|
int fatfs_find_blank_cluster(struct fatfs *fs, uint32 start_cluster, uint32 *free_cluster)
|
|
{
|
|
uint32 fat_sector_offset, position;
|
|
uint32 nextcluster;
|
|
uint32 current_cluster = start_cluster;
|
|
struct fat_buffer *pbuf;
|
|
|
|
do
|
|
{
|
|
// Find which sector of FAT table to read
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
fat_sector_offset = current_cluster / 256;
|
|
else
|
|
fat_sector_offset = current_cluster / 128;
|
|
|
|
if ( fat_sector_offset < fs->fat_sectors)
|
|
{
|
|
// Read FAT sector into buffer
|
|
pbuf = fatfs_fat_read_sector(fs, fs->fat_begin_lba+fat_sector_offset);
|
|
if (!pbuf)
|
|
return 0;
|
|
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
{
|
|
// Find 32 bit entry of current sector relating to cluster number
|
|
position = (current_cluster - (fat_sector_offset * 256)) * 2;
|
|
|
|
// Read Next Clusters value from Sector Buffer
|
|
nextcluster = FAT16_GET_16BIT_WORD(pbuf, (uint16)position);
|
|
}
|
|
else
|
|
{
|
|
// Find 32 bit entry of current sector relating to cluster number
|
|
position = (current_cluster - (fat_sector_offset * 128)) * 4;
|
|
|
|
// Read Next Clusters value from Sector Buffer
|
|
nextcluster = FAT32_GET_32BIT_WORD(pbuf, (uint16)position);
|
|
|
|
// Mask out MS 4 bits (its 28bit addressing)
|
|
nextcluster = nextcluster & 0x0FFFFFFF;
|
|
}
|
|
|
|
if (nextcluster !=0 )
|
|
current_cluster++;
|
|
}
|
|
else
|
|
// Otherwise, run out of FAT sectors to check...
|
|
return 0;
|
|
}
|
|
while (nextcluster != 0x0);
|
|
|
|
// Found blank entry
|
|
*free_cluster = current_cluster;
|
|
return 1;
|
|
}
|
|
#endif
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_fat_set_cluster: Set a cluster link in the chain. NOTE: Immediate
|
|
// write (slow).
|
|
//-----------------------------------------------------------------------------
|
|
#if FATFS_INC_WRITE_SUPPORT
|
|
int fatfs_fat_set_cluster(struct fatfs *fs, uint32 cluster, uint32 next_cluster)
|
|
{
|
|
struct fat_buffer *pbuf;
|
|
uint32 fat_sector_offset, position;
|
|
|
|
// Find which sector of FAT table to read
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
fat_sector_offset = cluster / 256;
|
|
else
|
|
fat_sector_offset = cluster / 128;
|
|
|
|
// Read FAT sector into buffer
|
|
pbuf = fatfs_fat_read_sector(fs, fs->fat_begin_lba+fat_sector_offset);
|
|
if (!pbuf)
|
|
return 0;
|
|
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
{
|
|
// Find 16 bit entry of current sector relating to cluster number
|
|
position = (cluster - (fat_sector_offset * 256)) * 2;
|
|
|
|
// Write Next Clusters value to Sector Buffer
|
|
FAT16_SET_16BIT_WORD(pbuf, (uint16)position, ((uint16)next_cluster));
|
|
}
|
|
else
|
|
{
|
|
// Find 32 bit entry of current sector relating to cluster number
|
|
position = (cluster - (fat_sector_offset * 128)) * 4;
|
|
|
|
// Write Next Clusters value to Sector Buffer
|
|
FAT32_SET_32BIT_WORD(pbuf, (uint16)position, next_cluster);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
#endif
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_free_cluster_chain: Follow a chain marking each element as free
|
|
//-----------------------------------------------------------------------------
|
|
#if FATFS_INC_WRITE_SUPPORT
|
|
int fatfs_free_cluster_chain(struct fatfs *fs, uint32 start_cluster)
|
|
{
|
|
uint32 last_cluster;
|
|
uint32 next_cluster = start_cluster;
|
|
|
|
// Loop until end of chain
|
|
while ( (next_cluster != FAT32_LAST_CLUSTER) && (next_cluster != 0x00000000) )
|
|
{
|
|
last_cluster = next_cluster;
|
|
|
|
// Find next link
|
|
next_cluster = fatfs_find_next_cluster(fs, next_cluster);
|
|
|
|
// Clear last link
|
|
fatfs_fat_set_cluster(fs, last_cluster, 0x00000000);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
#endif
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_fat_add_cluster_to_chain: Follow a chain marking and then add a new entry
|
|
// to the current tail.
|
|
//-----------------------------------------------------------------------------
|
|
#if FATFS_INC_WRITE_SUPPORT
|
|
int fatfs_fat_add_cluster_to_chain(struct fatfs *fs, uint32 start_cluster, uint32 newEntry)
|
|
{
|
|
uint32 last_cluster = FAT32_LAST_CLUSTER;
|
|
uint32 next_cluster = start_cluster;
|
|
|
|
if (start_cluster == FAT32_LAST_CLUSTER)
|
|
return 0;
|
|
|
|
// Loop until end of chain
|
|
while ( next_cluster != FAT32_LAST_CLUSTER )
|
|
{
|
|
last_cluster = next_cluster;
|
|
|
|
// Find next link
|
|
next_cluster = fatfs_find_next_cluster(fs, next_cluster);
|
|
if (!next_cluster)
|
|
return 0;
|
|
}
|
|
|
|
// Add link in for new cluster
|
|
fatfs_fat_set_cluster(fs, last_cluster, newEntry);
|
|
|
|
// Mark new cluster as end of chain
|
|
fatfs_fat_set_cluster(fs, newEntry, FAT32_LAST_CLUSTER);
|
|
|
|
return 1;
|
|
}
|
|
#endif
|
|
//-----------------------------------------------------------------------------
|
|
// fatfs_count_free_clusters:
|
|
//-----------------------------------------------------------------------------
|
|
uint32 fatfs_count_free_clusters(struct fatfs *fs)
|
|
{
|
|
uint32 i,j;
|
|
uint32 count = 0;
|
|
struct fat_buffer *pbuf;
|
|
|
|
for (i = 0; i < fs->fat_sectors; i++)
|
|
{
|
|
// Read FAT sector into buffer
|
|
pbuf = fatfs_fat_read_sector(fs, fs->fat_begin_lba + i);
|
|
if (!pbuf)
|
|
break;
|
|
|
|
for (j = 0; j < FAT_SECTOR_SIZE; )
|
|
{
|
|
if (fs->fat_type == FAT_TYPE_16)
|
|
{
|
|
if (FAT16_GET_16BIT_WORD(pbuf, (uint16)j) == 0)
|
|
count++;
|
|
|
|
j += 2;
|
|
}
|
|
else
|
|
{
|
|
if (FAT32_GET_32BIT_WORD(pbuf, (uint16)j) == 0)
|
|
count++;
|
|
|
|
j += 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
return count;
|
|
}
|