Difference between revisions of "Unit FATFS"
| (14 intermediate revisions by the same user not shown) | |||
| Line 5: | Line 5: | ||
| ---- | ---- | ||
| − | '' | + | '''Ultibo FAT12/16/32/exFAT Interface unit''' | 
| + | |||
| + | ''Notes:'' | ||
| + | |||
| + | Based on information from numerous sources, primarily the document fatgen103.pdf from Microsoft. FAT Disk Blocks are based on reading a group of sectors into a block at once. The BlockNo of each block is therefore the FAT entry no of the first FAT in the block. | ||
| + | |||
| + | The use of IsEightDotThree and GenerateShortName(Ex) from the UltiboUtils unit should be replaced by internal routines optimised for maximum performance. All Block and Entry Offsets have been expanded to 32bit to avoid 16bit overflow on 64K clusters. | ||
| === Constants === | === Constants === | ||
| ---- | ---- | ||
| − | '' | + | |
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial; padding-top: 20px; padding-bottom: 15px;"> | ||
| + | <div style="font-size: 14px; padding-left: 12px;">'''FATFS specific constants''' <code> fat* </code></div> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | | <code>fatFAT12 = 0;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatFAT16 = 1;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatFAT32 = 2;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatDotName = '.';</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatDotDotName = '..';</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatNameCount = 8;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatExtCount = 3;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatLabelMask = '*';</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatLabelCount = 11;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatOemName = 'MSWIN4.1';</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatDefaultName = 'NO NAME';</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatFlagNone = $00;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatFlagName = $08;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatFlagExt = $10;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatFlagBoth = $18;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>lfnMaxName = 2048;</code> | ||
| + | | Buffer size for lfn handling | ||
| + | |- | ||
| + | | <code>lfnMaxAlias = 999999;</code> | ||
| + | | Max number of lfn aliases (ie ~999999) | ||
| + | |- | ||
| + | | <code>lfnMaxNameAlias = 4;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>lfnMaxHashAlias = 999;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>lfnMaxHashShift = 16;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatMaxFileSize = 4294967295;</code> | ||
| + | | Max size in bytes of a File (4GB) | ||
| + | |- | ||
| + | | <code>fatMaxDirectorySize = 2097152;</code> | ||
| + | | Max size in bytes of a Directory (2MB = 65536 * 32) | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatMinClusterSize = 512;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatMaxClusterSize = 65536;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat12MinClusters = 3;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat16MinClusters = 4085;</code> | ||
| + | | Smallest Number of Clusters on a FAT16 volume, is FAT12 if less | ||
| + | |- | ||
| + | | <code>fat32MinClusters = 65525;</code> | ||
| + | | Smallest Number of Clusters on a FAT32 volume, is FAT16 if less | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat12MinSectors = 6;</code> | ||
| + | | Number of Sectors at which we cannot create a FAT12 (Not possible due to 512 Root entries) | ||
| + | |- | ||
| + | | <code>fat12MinSectors = 798;</code> | ||
| + | | Number of Sectors at which we cannot create a FAT12 | ||
| + | |- | ||
| + | | <code>fat12MaxSectors = 32623;</code> | ||
| + | | Number of Sectors at which we insist on creating FAT16 | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat16MinSectors = 8400;</code> | ||
| + | | Number of Sectors at which we cannot create a FAT16 | ||
| + | |- | ||
| + | | <code>fat16MaxSectors = 1048576;</code> | ||
| + | | Number of Sectors at which we insist on creating FAT32 | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat32MinSectors = 66600;</code> | ||
| + | | Number of Sectors at which we cannot create a FAT32 | ||
| + | |- | ||
| + | | <code>fat32MaxSectors = $FFFFFFFF;</code> | ||
| + | | No limit on maximum size of a FAT32 | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | |colspan="2"|Note: In FAT12 EOC will be >= $0FF0 due to Media ID on Floppy | ||
| + | |- | ||
| + | | <code>fat12EndOfFile = $0FF0;</code> | ||
| + | | Some Code sets values from $0FF0 - $0FFF as EOF | ||
| + | |- | ||
| + | | <code>fat12EndOfFile = $0FF8;</code> | ||
| + | | Some Code sets values from $0FF8 - $0FFF as EOF | ||
| + | |- | ||
| + | | <code>fat12EndOfCluster = $0FFF;</code> | ||
| + | | EOF Mark, Check for >= $0FF0 - See Above | ||
| + | |- | ||
| + | | <code>fat12FreeCluster = $0000;</code> | ||
| + | | Available for Use | ||
| + | |- | ||
| + | | <code>fat12BadCluster = $0FF7;</code> | ||
| + | | Bad Cluster Mark - Do Not Use | ||
| + | |- | ||
| + | | <code>fat12MediaCluster = $0F00;</code> | ||
| + | | First Cluster - $0F00 + MediaID from BPB Usually $0FF8 | ||
| + | |- | ||
| + | | <code>fat12Reserved0 = $0FF0;</code> | ||
| + | | Reserved Cluster Nos - Do Not Use (Due to Media ID on Floppy) | ||
| + | |- | ||
| + | | <code>fat12Reserved1 = $0FF1;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat12Reserved2 = $0FF2;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat12Reserved3 = $0FF3;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat12Reserved4 = $0FF4;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat12Reserved5 = $0FF5;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat12Reserved6 = $0FF6;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat16EndOfFile = $FFF8;</code> | ||
| + | | Some Code sets values from $FFF8 - $FFFF as EOF | ||
| + | |- | ||
| + | | <code>fat16EndOfCluster = $FFFF;</code> | ||
| + | | EOF Mark, Check for >= $FFF8 - See Above | ||
| + | |- | ||
| + | | <code>fat16FreeCluster = $0000;</code> | ||
| + | | Available for Use | ||
| + | |- | ||
| + | | <code>fat16BadCluster = $FFF7;</code> | ||
| + | | Bad Cluster Mark - Do Not Use | ||
| + | |- | ||
| + | | <code>fat16MediaCluster = $FF00;</code> | ||
| + | | First Cluster - $FF00 + MediaID from BPB Usually $FFF8 | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat32EndOfFile = $0FFFFFF8;</code> | ||
| + | | Some Code sets values from $0FFFFFF8 - $0FFFFFFF as EOF | ||
| + | |- | ||
| + | | <code>fat32EndOfCluster = $0FFFFFFF;</code> | ||
| + | | EOF Mark, Check for >= $0FFFFFF8 - See Above | ||
| + | |- | ||
| + | | <code>fat32FreeCluster = $00000000;</code> | ||
| + | | Available for Use | ||
| + | |- | ||
| + | | <code>fat32BadCluster = $0FFFFFF7;</code> | ||
| + | | Bad Cluster Mark - Do Not Use | ||
| + | |- | ||
| + | | <code>fat32MediaCluster = $0FFFFF00;</code> | ||
| + | | First Cluster - $0FFFFF00 + MediaID from BPB Usually $0FFFFFF8 | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat16CleanShutdown = $8000;</code> | ||
| + | | Second Cluster - EndOfCluster + Bits | ||
| + | |- | ||
| + | | <code>fat16HardError = $4000;</code> | ||
| + | | Second Cluster - EndOfCluster + Bits | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat32CleanShutdown = $08000000;</code> | ||
| + | | Second Cluster - EndOfCluster + Bits | ||
| + | |- | ||
| + | | <code>fat32HardError = $04000000;</code> | ||
| + | | Second Cluster - EndOfCluster + Bits | ||
| + | |- | ||
| + | | <code>fat32ReservedBits = $F0000000;</code> | ||
| + | | Bits 31-29 must be preserved on FAT32 | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatBootSector = 0;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatFreeCluster = 0;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatStartCluster = 2;</code> | ||
| + | | Lowest allocatable cluster number | ||
| + | |- | ||
| + | | <code>fatUnevenCluster = $00000001;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatUnknownCluster = $FFFFFFFF;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatEntryFree = $E5;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatEntryFreeAll = $00;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatEntrySpecial = $05;</code> | ||
| + | | If real first char is $E5 | ||
| + | |- | ||
| + | | <code>fatEntryPadding = $20;</code> | ||
| + | | Padded with spaces (32) | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatEntrySize = 32;</code> | ||
| + | | Size of a FAT Directory Entry (or LFNDirectoryEntry) | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>lfnEntryMax= $3F;</code> | ||
| + | | Maximum count of an LFN entry | ||
| + | |- | ||
| + | | <code>lfnEntryLast = $40;</code> | ||
| + | | Marker for LFN last entry | ||
| + | |- | ||
| + | | <code>lfnEntryMask = $BF;</code> | ||
| + | | Mask for LFN entry count | ||
| + | |- | ||
| + | | <code>lfnEntryChars = 13;</code> | ||
| + | | Number of characters per LFN entry | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>lfnEntryNull = $0000;</code> | ||
| + | | Null terminator on LFN when applicable | ||
| + | |- | ||
| + | | <code>lfnEntryPadding = $FFFF;</code> | ||
| + | | Empty LFN chars padded with (65535) | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>lfnAttributeMask = (faReadOnly or faHidden or faSysFile or faVolumeID);</code> | ||
| + | | Not Used - See faLongName | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatBootSignature = $29;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat32LeadSignature = $41615252;</code> | ||
| + | | RRaA | ||
| + | |- | ||
| + | | <code>fat32StructSignature = $61417272;</code> | ||
| + | | rrAa | ||
| + | |- | ||
| + | | <code>fat32TrailSignature = $AA550000;</code> | ||
| + | | Standard Boot Sector signature | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatDotChar:Byte = $2E;</code> | ||
| + | | Checked separately from Invalid chars due to . and .. entries | ||
| + | |- | ||
| + | | <code>fatSpaceChar:Byte = $20;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fatBaseChars:set of Byte = [$00..$19];</code> | ||
| + | | 0..31 (Dont include $20 as that is checked separately) | ||
| + | |- | ||
| + | | <code>fatInvalidChars:set of Byte = [$22,$2A,$2B,$2C,$2F,$3A,$3B,$3C,$3D,$3E,$3F,$5B,$5C,$5D,$7C];</code> | ||
| + | | 34,42,43,44,47,58,59,60,61,62,63,91,92,93,124 (Dont include $2E as that is checked separately) | ||
| + | |- | ||
| + | | <code>fatLowercaseChars:set of Byte = [$61..$7A];</code> | ||
| + | | 97..122 (These dont account for the code page) | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | <br /> | ||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial; padding-top: 20px; padding-bottom: 15px;"> | ||
| + | <div style="font-size: 14px; padding-left: 12px;">'''FATFS params table''' <code> fatParams* </code></div> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Used for FAT entry block loading | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fatMaxParams = 11;</code> | ||
| + | | style="width: 50%;"|  | ||
| + | |- | ||
| + | | <code>fatParams:array[0..fatMaxParams] of TFATParams = (</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''SectorSize,FATType,BlockShiftCount,EntriesPerBlock,SectorsPerBlock'' | ||
| + | |- | ||
| + | |colspan="2"|''FAT12'' | ||
| + | |- | ||
| + | | <code>(SectorSize:512; FATType:ftFAT12; BlockShiftCount:10; EntriesPerBlock:1024; SectorsPerBlock:3),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:1024; FATType:ftFAT12; BlockShiftCount:11 ;EntriesPerBlock:2048; SectorsPerBlock:3),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:2048; FATType:ftFAT12; BlockShiftCount:12; EntriesPerBlock:4096; SectorsPerBlock:3),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:4096; FATType:ftFAT12; BlockShiftCount:13; EntriesPerBlock:8192; SectorsPerBlock:3),</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''FAT16'' | ||
| + | |- | ||
| + | | <code>(SectorSize:512; FATType:ftFAT16; BlockShiftCount:10; EntriesPerBlock:1024; SectorsPerBlock:4),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:1024; FATType:ftFAT16; BlockShiftCount:11; EntriesPerBlock:2048; SectorsPerBlock:4),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:2048; FATType:ftFAT16; BlockShiftCount:12; EntriesPerBlock:4096; SectorsPerBlock:4),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:4096 ;FATType:ftFAT16 ;BlockShiftCount:13; EntriesPerBlock:8192; SectorsPerBlock:4),</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''FAT32'' | ||
| + | |- | ||
| + | | <code>(SectorSize:512; FATType:ftFAT32; BlockShiftCount:10; EntriesPerBlock:1024; SectorsPerBlock:8),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:1024; FATType:ftFAT32; BlockShiftCount:11; EntriesPerBlock:2048; SectorsPerBlock:8),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:2048; FATType:ftFAT32; BlockShiftCount:12; EntriesPerBlock:4096; SectorsPerBlock:8),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(SectorSize:4096; FATType:ftFAT32; BlockShiftCount:13; EntriesPerBlock:8192; SectorsPerBlock:8));</code> | ||
| + | |   | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | <br /> | ||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial; padding-top: 20px; padding-bottom: 15px;"> | ||
| + | <div style="font-size: 14px; padding-left: 12px;">'''FATFS geometry tables''' <code> fat12Geometry* </code></div> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Used for FAT formatting (Floppy Disk only) | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat12MaxGeometry = 4;</code> | ||
| + | | style="width: 50%;"|  | ||
| + | |- | ||
| + | | <code>fat12Geometry:array[0..fat12MaxGeometry] of TFATGeometry = (</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''FloppyType,SectorsPerFat,SectorsPerCluster,RootEntryCount,MediaId'' | ||
| + | |- | ||
| + | | <code>(FloppyType:ft360K; SectorsPerFat:2; SectorsPerCluster:2; RootEntryCount:112; MediaId:$FD),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(FloppyType:ft12M;SectorsPerFat:7;SectorsPerCluster:1;RootEntryCount:224;MediaId:$F9),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(FloppyType:ft720K; SectorsPerFat:3; SectorsPerCluster:2; RootEntryCount:112; MediaId:$F9),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(FloppyType:ft144M; SectorsPerFat:9; SectorsPerCluster:1; RootEntryCount:224; MediaId:$F0),</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>(FloppyType:ft288M; SectorsPerFat:9; SectorsPerCluster:2; RootEntryCount:240; MediaId:$F0));</code> | ||
| + | |   | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | <br /> | ||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial; padding-top: 20px; padding-bottom: 15px;"> | ||
| + | <div style="font-size: 14px; padding-left: 12px;">'''FATFS cluster size tables''' <code> fat_ClusterSize* </code></div> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Used for FAT formatting (Fixed Disk only) | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat12MaxClusterSize = 2;</code> | ||
| + | | style="width: 50%;"|  | ||
| + | |- | ||
| + | | <code>fat12ClusterSize:array[0..fat12MaxClusterSize] of TFATClusterSize = (</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''SectorCount,SectorsPerCluster'' | ||
| + | |- | ||
| + | | <code>(SectorCount:6; SectorsPerCluster:0),</code> | ||
| + | | up to 3 KB, the 0 value for SectorsPerCluster trips an error (Not possible due to 512 Root entries) | ||
| + | |- | ||
| + | | <code>(SectorCount:798; SectorsPerCluster:0),</code> | ||
| + | | up to 399 KB, the 0 value for SectorsPerCluster trips an error | ||
| + | |- | ||
| + | | <code>(SectorCount:32623; SectorsPerCluster:8),</code> | ||
| + | | up to 15.9 MB, 4k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:$FFFFFFFF; SectorsPerCluster:0));</code> | ||
| + | | greater than 15.9 MB, 0 value for SectorsPerCluster trips an error | ||
| + | |- | ||
| + | |colspan="2"|See FAT12 Notes for Min/Max and Values | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat16MaxClusterSize = 8;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat16ClusterSize:array[0..fat16MaxClusterSize] of TFATClusterSize = (</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''SectorCount,SectorsPerCluster'' | ||
| + | |- | ||
| + | | <code>(SectorCount:8400; SectorsPerCluster:0),</code> | ||
| + | | up to 4.1 MB, the 0 value for SectorsPerCluster trips an error | ||
| + | |- | ||
| + | | <code>(SectorCount:32680; SectorsPerCluster:2),</code> | ||
| + | | up to 16 MB, 1k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:262144; SectorsPerCluster:4),</code> | ||
| + | | up to 128 MB, 2k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:524288; SectorsPerCluster:8),</code> | ||
| + | | up to 256 MB, 4k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:1048576; SectorsPerCluster:16),</code> | ||
| + | | up to 512 MB, 8k cluster | ||
| + | |- | ||
| + | |colspan="2"|The entries after this point are not used unless FAT16 is forced | ||
| + | |- | ||
| + | | <code>(SectorCount:2097152; SectorsPerCluster:32),</code> | ||
| + | | up to 1 GB, 16k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:4194304; SectorsPerCluster:64),</code> | ||
| + | | up to 2 GB, 32k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:8388608; SectorsPerCluster:128),</code> | ||
| + | | up to 4 GB, 64k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:$FFFFFFFF;SectorsPerCluster:0));</code> | ||
| + | | greater than 4 GB, 0 value for SectorsPerCluster trips an error | ||
| + | |- | ||
| + | |colspan="2"|  | ||
| + | |- | ||
| + | | <code>fat32MaxClusterSize = 5;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>fat32ClusterSize:array[0..fat32MaxClusterSize] of TFATClusterSize = (</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''SectorCount,SectorsPerCluster'' | ||
| + | |- | ||
| + | | <code>(SectorCount:66600; SectorsPerCluster:0),</code> | ||
| + | | up to 32.5 MB, the 0 value for SectorsPerCluster trips an error | ||
| + | |- | ||
| + | | <code>(SectorCount:532480 ;SectorsPerCluster:1),</code> | ||
| + | | up to 260 MB, .5k cluster | ||
| + | |- | ||
| + | |colspan="2"|The entries before this point are not used unless FAT32 is forced | ||
| + | |- | ||
| + | | <code>(SectorCount:16777216; SectorsPerCluster:8),</code> | ||
| + | | up to 8 GB, 4k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:33554432; SectorsPerCluster:16),</code> | ||
| + | | up to 16 GB, 8k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:67108864; SectorsPerCluster:32),</code> | ||
| + | | up to 32 GB, 16k cluster | ||
| + | |- | ||
| + | | <code>(SectorCount:$FFFFFFFF; SectorsPerCluster:64));</code> | ||
| + | | greater than 32 GB, 32k cluster | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | <br /> | ||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial; padding-top: 20px; padding-bottom: 15px;"> | ||
| + | <div style="font-size: 14px; padding-left: 12px;">'''FATFS partition types''' <code> fatPartitionType* </code></div> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | | <code>fatMaxPartitionType = 5;</code> | ||
| + | | style="width: 50%;"|  | ||
| + | |- | ||
| + | | <code>fatPartitionType:array[0..fatMaxPartitionType] of TFATPartitionType = (</code> | ||
| + | |   | ||
| + | |- | ||
| + | |colspan="2"|''Excluded,Extended,SectorCount,PartitionId,LBAType'' | ||
| + | |- | ||
| + | |colspan="2"|''Extended Partitions'' | ||
| + | |- | ||
| + | | <code>(Excluded:False; Extended:True; SectorCount:$FFFFFFFF ;PartitionId:pidExtended; LBAType:pidExtLBA),</code> | ||
| + | | any size, Extended | ||
| + | |- | ||
| + | |colspan="2"|Primary and Logical Partitions | ||
| + | |- | ||
| + | |colspan="2"|''Unused'' | ||
| + | |- | ||
| + | | <code>(Excluded:False; Extended:False; SectorCount:798; PartitionId:pidUnused; LBAType:pidUnused),</code> | ||
| + | | up to 399 KB, Unused (Unused type trips an error) | ||
| + | |- | ||
| + | |colspan="2"|''FAT12'' | ||
| + | |- | ||
| + | | <code>(Excluded:False; Extended:False; SectorCount:32680; PartitionId:pidFAT12; LBAType:pidFAT12),</code> | ||
| + | | up to 16 MB, FAT12 | ||
| + | |- | ||
| + | |colspan="2"|''FAT16'' | ||
| + | |- | ||
| + | | <code>(Excluded:False; Extended:False; SectorCount:65360; PartitionId:pidFAT16; LBAType:pidFAT16LBA),</code> | ||
| + | | up to 32 MB, FAT16 | ||
| + | |- | ||
| + | | <code>(Excluded:False; Extended:False; SectorCount:1048576; PartitionId:pidFAT16HUGE; LBAType:pidFAT16LBA),</code> | ||
| + | | up to 512 MB, FAT16 HUGE | ||
| + | |- | ||
| + | |colspan="2"|''FAT32'' | ||
| + | |- | ||
| + | | <code>(Excluded:False; Extended:False; SectorCount:$FFFFFFFF; PartitionId:pidFAT32; LBAType:pidFAT32LBA));</code> | ||
| + | | above 512 MB, FAT32 | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | <br /> | ||
| === Type definitions === | === Type definitions === | ||
| ---- | ---- | ||
| − | '' | + | |
| + | '''FATFS specific types''' | ||
| + | |||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | <code>TFATType = (ftNONE,ftFAT12,ftFAT16,ftFAT32);</code> | ||
| + | | style="width: 40%;"| | ||
| + | |- | ||
| + | |}  | ||
| + | |||
| + | '''FATFS params'''  | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | |||
| + | <code>TFATParams = record </code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Note: (Not Packed) Block values for various Sector sizes. | ||
| + | |- | ||
| + | | <code>SectorSize:Word;</code> | ||
| + | | style="width: 50%;"|  | ||
| + | |- | ||
| + | | <code>FATType:TFATType;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>BlockShiftCount:Word;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>EntriesPerBlock:LongWord;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>SectorsPerBlock:LongWord;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |} | ||
| + | </div></div>   | ||
| + | |||
| + | '''FATFS geometry'''   | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | |||
| + | <code>TFATGeometry = record</code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Note: (Not Packed) FAT12 values for various Floppy Types. | ||
| + | |- | ||
| + | | <code>FloppyType:TFloppyType;</code> | ||
| + | | style="width: 50%;"|  | ||
| + | |- | ||
| + | | <code>SectorsPerFat:LongWord;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>SectorsPerCluster:LongWord;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>RootEntryCount:LongWord;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>MediaId:Byte;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | |||
| + | '''FATFS cluster size'''  | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | |||
| + | <code>TFATClusterSize = record</code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Note: (Not Packed) Sectors per Cluster for various disk sizes (Fixed Disk only). | ||
| + | |- | ||
| + | | <code>SectorCount:LongWord;</code> | ||
| + | | Based on 512 bytes per sector only | ||
| + | |- | ||
| + | | <code>SectorsPerCluster:LongWord;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | |||
| + | '''FATFS partition type'''  | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | |||
| + | <code>TFATPartitionType = record</code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Note: (Not Packed) Partition Type for various disk sizes (Fixed Disk only). | ||
| + | |- | ||
| + | | <code>Excluded:Boolean;</code> | ||
| + | | Entry is excluded from FAT Partition Type check | ||
| + | |- | ||
| + | | <code>Extended:Boolean;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>SectorCount:LongWord;</code> | ||
| + | | Based on 512 bytes per sector only | ||
| + | |- | ||
| + | | <code>PartitionId:Byte;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>LBAType:Byte;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | |||
| + | '''FATFS name'''  | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | <code>PFATName = ^TFATName;</code> | ||
| + | |||
| + | <code>TFATName = packed record</code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Note: 11 Bytes | ||
| + | |- | ||
| + | | <code>Name:array[0..7] of Char;</code> | ||
| + | | File Name | ||
| + | |- | ||
| + | | <code>Ext:array[0..2] of Char;</code> | ||
| + | | File Extension | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | |||
| + | '''FATSFS directory'''  | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | <code>PFATDirectory = ^TFATDirectory;</code> | ||
| + | |||
| + | <code>TFATDirectory = packed record</code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Note: 32 Bytes | ||
| + | |- | ||
| + | | <code>Name:array[0..7] of Char;</code> | ||
| + | | File Name | ||
| + | |- | ||
| + | | <code>Ext:array[0..2] of Char;</code> | ||
| + | | File Extension | ||
| + | |- | ||
| + | | <code>Attribute:Byte;</code> | ||
| + | | File Attributes - See Standard Values | ||
| + | |- | ||
| + | | <code>Reserved:Byte;</code> | ||
| + | | Always 0 (See below) | ||
| + | |- | ||
| + | | <code>CaseFlags:Byte;</code> | ||
| + | | Set to $08 if Name, $10 if Extension or $18 if both | ||
| + | |- | ||
| + | | <code>CreateTimeMsecs:Byte;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>CreateTime:Word;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>CreateDate:Word;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>LastAccessDate:Word;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>FirstClusterHigh:Word;</code> | ||
| + | | High word of the start cluster (FAT32 Only) | ||
| + | |- | ||
| + | | <code>WriteTime:Word;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>WriteDate:Word;</code> | ||
| + | |   | ||
| + | |- | ||
| + | | <code>FirstClusterLow:Word;</code> | ||
| + | | Low Word of Start Cluster all versions | ||
| + | |- | ||
| + | | <code>Length:LongWord;</code> | ||
| + | |   | ||
| + | |- | ||
| + | |} | ||
| + | </div></div>   | ||
| + | |||
| + | '''LFN directory'''   | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | <code>PLFNDirectory = ^TLFNDirectory;</code> | ||
| + | |||
| + | <code>TLFNDirectory = packed record</code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | |colspan="2"|Note: 32 Bytes | ||
| + | |- | ||
| + | | <code>Order:Byte;</code> | ||
| + | | Bit 6 = 1 if last LFN in chain, Bits 5-0 = Order of LFN chunks for this File | ||
| + | |- | ||
| + | | <code>Name1:array[0..4] of WideChar;</code> | ||
| + | | First 5 Characters in this chunk | ||
| + | |- | ||
| + | | <code>Attribute:Byte;</code> | ||
| + | | Always 0Fh - Volume, System, Hidden, ReadOnly | ||
| + | |- | ||
| + | | <code>FileType:Byte;</code> | ||
| + | | Usually 00h | ||
| + | |- | ||
| + | | <code>Checksum:Byte;</code> | ||
| + | | Checksum calculated on 8.3 name | ||
| + | |- | ||
| + | | <code>Name2:array[0..5] of WideChar;</code> | ||
| + | | Next 6 Characters in this chunk | ||
| + | |- | ||
| + | | <code>Reserved:Word;</code> | ||
| + | | Always 0000h - FirstCluster in Standard Entry | ||
| + | |- | ||
| + | | <code>Name3:array[0..1] of WideChar;</code> | ||
| + | | Next 2 Characters in this chunk | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | |||
| + | '''FATFS info sector''' | ||
| + | |||
| + | <div class="toccolours mw-collapsible mw-collapsed" style="border: 1; font-family: arial;"> | ||
| + | <code>PFATInfoSector = ^TFATInfoSector;</code> | ||
| + | |||
| + | <code>TFATInfoSector = packed record</code> | ||
| + | <div class="mw-collapsible-content" style="text-align: left; padding-left: 5px;"> | ||
| + | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| + | |- | ||
| + | | <code>LeadSignature:LongWord;</code> | ||
| + | | Signature "RRaA" or 41615252h | ||
| + | |- | ||
| + | | <code>Reserved1:array[0..479] of Byte;</code> | ||
| + | | Always 0 | ||
| + | |- | ||
| + | | <code>StructureSignature:LongWord;</code> | ||
| + | | Signature "rrAa" or 61417272h | ||
| + | |- | ||
| + | | <code>FreeClusterCount:LongWord;</code> | ||
| + | | Free Clusters $FFFFFFFF if unknown | ||
| + | |- | ||
| + | | <code>LastFreeCluster:LongWord;</code> | ||
| + | | Most recently allocated cluster $FFFFFFFF if unknown | ||
| + | |- | ||
| + | | <code>Reserved2:array[0..11] of Byte;</code> | ||
| + | | Always 0 | ||
| + | |- | ||
| + | | <code>TrailSignature:LongWord;</code> | ||
| + | | Signature AA550000h | ||
| + | |- | ||
| + | |} | ||
| + | </div></div> | ||
| + | <br /> | ||
| + | |||
| + | === Class definitions === | ||
| + | ---- | ||
| + | |||
| + | |||
| + | '''FATFS specific classes''' | ||
| + | |||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATRecognizer|<code>TFATRecognizer = class(TRecognizer)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATPartitioner|<code>TFATPartitioner = class(TDiskPartitioner)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATFormatter|<code>TFATFormatter = class(TDiskFormatter)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATDefragger|<code>TFATDefragger = class(TDiskDefragger)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATRepairer|<code>TFATRepairer = class(TDiskRepairer)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATFileSystem|<code>TFATFileSystem = class(TFileSystem)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATDiskTable|<code>TFATDiskTable = class(TDiskTable)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATDiskBlock|<code>TFATDiskBlock = class(TDiskBlock)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | {| class="wikitable" style="font-size: 14px; text-align: left; width: 100%; height: 50px;" | ||
| + | |- | ||
| + | | [[TFATDiskEntry|<code>TFATDiskEntry = class(TDiskEntry)</code>]] | ||
| + | |- | ||
| + | |} | ||
| + | <br /> | ||
| === Public variables === | === Public variables === | ||
| ---- | ---- | ||
| − | '' | + | ''None defined'' | 
| === Function declarations === | === Function declarations === | ||
| Line 34: | Line 922: | ||
| {| class="wikitable" style="font-size: 14px; background: white;" | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| |- | |- | ||
| − | !  | + | ! Note | 
| | None documented | | None documented | ||
| |- | |- | ||
| Line 46: | Line 934: | ||
| {| class="wikitable" style="font-size: 14px; background: white;" | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| |- | |- | ||
| − | !  | + | ! Note | 
| | None documented | | None documented | ||
| |- | |- | ||
| Line 61: | Line 949: | ||
| {| class="wikitable" style="font-size: 14px; background: white;" | {| class="wikitable" style="font-size: 14px; background: white;" | ||
| |- | |- | ||
| − | !  | + | ! Note | 
| | None documented | | None documented | ||
| |- | |- | ||
Latest revision as of 05:36, 31 August 2021
Return to Unit Reference
Contents
Description
Ultibo FAT12/16/32/exFAT Interface unit
Notes:
Based on information from numerous sources, primarily the document fatgen103.pdf from Microsoft. FAT Disk Blocks are based on reading a group of sectors into a block at once. The BlockNo of each block is therefore the FAT entry no of the first FAT in the block.
The use of IsEightDotThree and GenerateShortName(Ex) from the UltiboUtils unit should be replaced by internal routines optimised for maximum performance. All Block and Entry Offsets have been expanded to 32bit to avoid 16bit overflow on 64K clusters.
Constants
 fat* | fatFAT12 = 0; | |
| fatFAT16 = 1; | |
| fatFAT32 = 2; | |
| fatDotName = '.'; | |
| fatDotDotName = '..'; | |
| fatNameCount = 8; | |
| fatExtCount = 3; | |
| fatLabelMask = '*'; | |
| fatLabelCount = 11; | |
| fatOemName = 'MSWIN4.1'; | |
| fatDefaultName = 'NO NAME'; | |
| fatFlagNone = $00; | |
| fatFlagName = $08; | |
| fatFlagExt = $10; | |
| fatFlagBoth = $18; | |
| lfnMaxName = 2048; | Buffer size for lfn handling | 
| lfnMaxAlias = 999999; | Max number of lfn aliases (ie ~999999) | 
| lfnMaxNameAlias = 4; | |
| lfnMaxHashAlias = 999; | |
| lfnMaxHashShift = 16; | |
| fatMaxFileSize = 4294967295; | Max size in bytes of a File (4GB) | 
| fatMaxDirectorySize = 2097152; | Max size in bytes of a Directory (2MB = 65536 * 32) | 
| fatMinClusterSize = 512; | |
| fatMaxClusterSize = 65536; | |
| fat12MinClusters = 3; | |
| fat16MinClusters = 4085; | Smallest Number of Clusters on a FAT16 volume, is FAT12 if less | 
| fat32MinClusters = 65525; | Smallest Number of Clusters on a FAT32 volume, is FAT16 if less | 
| fat12MinSectors = 6; | Number of Sectors at which we cannot create a FAT12 (Not possible due to 512 Root entries) | 
| fat12MinSectors = 798; | Number of Sectors at which we cannot create a FAT12 | 
| fat12MaxSectors = 32623; | Number of Sectors at which we insist on creating FAT16 | 
| fat16MinSectors = 8400; | Number of Sectors at which we cannot create a FAT16 | 
| fat16MaxSectors = 1048576; | Number of Sectors at which we insist on creating FAT32 | 
| fat32MinSectors = 66600; | Number of Sectors at which we cannot create a FAT32 | 
| fat32MaxSectors = $FFFFFFFF; | No limit on maximum size of a FAT32 | 
| Note: In FAT12 EOC will be >= $0FF0 due to Media ID on Floppy | |
| fat12EndOfFile = $0FF0; | Some Code sets values from $0FF0 - $0FFF as EOF | 
| fat12EndOfFile = $0FF8; | Some Code sets values from $0FF8 - $0FFF as EOF | 
| fat12EndOfCluster = $0FFF; | EOF Mark, Check for >= $0FF0 - See Above | 
| fat12FreeCluster = $0000; | Available for Use | 
| fat12BadCluster = $0FF7; | Bad Cluster Mark - Do Not Use | 
| fat12MediaCluster = $0F00; | First Cluster - $0F00 + MediaID from BPB Usually $0FF8 | 
| fat12Reserved0 = $0FF0; | Reserved Cluster Nos - Do Not Use (Due to Media ID on Floppy) | 
| fat12Reserved1 = $0FF1; | |
| fat12Reserved2 = $0FF2; | |
| fat12Reserved3 = $0FF3; | |
| fat12Reserved4 = $0FF4; | |
| fat12Reserved5 = $0FF5; | |
| fat12Reserved6 = $0FF6; | |
| fat16EndOfFile = $FFF8; | Some Code sets values from $FFF8 - $FFFF as EOF | 
| fat16EndOfCluster = $FFFF; | EOF Mark, Check for >= $FFF8 - See Above | 
| fat16FreeCluster = $0000; | Available for Use | 
| fat16BadCluster = $FFF7; | Bad Cluster Mark - Do Not Use | 
| fat16MediaCluster = $FF00; | First Cluster - $FF00 + MediaID from BPB Usually $FFF8 | 
| fat32EndOfFile = $0FFFFFF8; | Some Code sets values from $0FFFFFF8 - $0FFFFFFF as EOF | 
| fat32EndOfCluster = $0FFFFFFF; | EOF Mark, Check for >= $0FFFFFF8 - See Above | 
| fat32FreeCluster = $00000000; | Available for Use | 
| fat32BadCluster = $0FFFFFF7; | Bad Cluster Mark - Do Not Use | 
| fat32MediaCluster = $0FFFFF00; | First Cluster - $0FFFFF00 + MediaID from BPB Usually $0FFFFFF8 | 
| fat16CleanShutdown = $8000; | Second Cluster - EndOfCluster + Bits | 
| fat16HardError = $4000; | Second Cluster - EndOfCluster + Bits | 
| fat32CleanShutdown = $08000000; | Second Cluster - EndOfCluster + Bits | 
| fat32HardError = $04000000; | Second Cluster - EndOfCluster + Bits | 
| fat32ReservedBits = $F0000000; | Bits 31-29 must be preserved on FAT32 | 
| fatBootSector = 0; | |
| fatFreeCluster = 0; | |
| fatStartCluster = 2; | Lowest allocatable cluster number | 
| fatUnevenCluster = $00000001; | |
| fatUnknownCluster = $FFFFFFFF; | |
| fatEntryFree = $E5; | |
| fatEntryFreeAll = $00; | |
| fatEntrySpecial = $05; | If real first char is $E5 | 
| fatEntryPadding = $20; | Padded with spaces (32) | 
| fatEntrySize = 32; | Size of a FAT Directory Entry (or LFNDirectoryEntry) | 
| lfnEntryMax= $3F; | Maximum count of an LFN entry | 
| lfnEntryLast = $40; | Marker for LFN last entry | 
| lfnEntryMask = $BF; | Mask for LFN entry count | 
| lfnEntryChars = 13; | Number of characters per LFN entry | 
| lfnEntryNull = $0000; | Null terminator on LFN when applicable | 
| lfnEntryPadding = $FFFF; | Empty LFN chars padded with (65535) | 
| lfnAttributeMask = (faReadOnly or faHidden or faSysFile or faVolumeID); | Not Used - See faLongName | 
| fatBootSignature = $29; | |
| fat32LeadSignature = $41615252; | RRaA | 
| fat32StructSignature = $61417272; | rrAa | 
| fat32TrailSignature = $AA550000; | Standard Boot Sector signature | 
| fatDotChar:Byte = $2E; | Checked separately from Invalid chars due to . and .. entries | 
| fatSpaceChar:Byte = $20; | |
| fatBaseChars:set of Byte = [$00..$19]; | 0..31 (Dont include $20 as that is checked separately) | 
| fatInvalidChars:set of Byte = [$22,$2A,$2B,$2C,$2F,$3A,$3B,$3C,$3D,$3E,$3F,$5B,$5C,$5D,$7C]; | 34,42,43,44,47,58,59,60,61,62,63,91,92,93,124 (Dont include $2E as that is checked separately) | 
| fatLowercaseChars:set of Byte = [$61..$7A]; | 97..122 (These dont account for the code page) | 
 fatParams* | Used for FAT entry block loading | |
| fatMaxParams = 11; | |
| fatParams:array[0..fatMaxParams] of TFATParams = ( | |
| SectorSize,FATType,BlockShiftCount,EntriesPerBlock,SectorsPerBlock | |
| FAT12 | |
| (SectorSize:512; FATType:ftFAT12; BlockShiftCount:10; EntriesPerBlock:1024; SectorsPerBlock:3), | |
| (SectorSize:1024; FATType:ftFAT12; BlockShiftCount:11 ;EntriesPerBlock:2048; SectorsPerBlock:3), | |
| (SectorSize:2048; FATType:ftFAT12; BlockShiftCount:12; EntriesPerBlock:4096; SectorsPerBlock:3), | |
| (SectorSize:4096; FATType:ftFAT12; BlockShiftCount:13; EntriesPerBlock:8192; SectorsPerBlock:3), | |
| FAT16 | |
| (SectorSize:512; FATType:ftFAT16; BlockShiftCount:10; EntriesPerBlock:1024; SectorsPerBlock:4), | |
| (SectorSize:1024; FATType:ftFAT16; BlockShiftCount:11; EntriesPerBlock:2048; SectorsPerBlock:4), | |
| (SectorSize:2048; FATType:ftFAT16; BlockShiftCount:12; EntriesPerBlock:4096; SectorsPerBlock:4), | |
| (SectorSize:4096 ;FATType:ftFAT16 ;BlockShiftCount:13; EntriesPerBlock:8192; SectorsPerBlock:4), | |
| FAT32 | |
| (SectorSize:512; FATType:ftFAT32; BlockShiftCount:10; EntriesPerBlock:1024; SectorsPerBlock:8), | |
| (SectorSize:1024; FATType:ftFAT32; BlockShiftCount:11; EntriesPerBlock:2048; SectorsPerBlock:8), | |
| (SectorSize:2048; FATType:ftFAT32; BlockShiftCount:12; EntriesPerBlock:4096; SectorsPerBlock:8), | |
| (SectorSize:4096; FATType:ftFAT32; BlockShiftCount:13; EntriesPerBlock:8192; SectorsPerBlock:8)); | |
 fat12Geometry* | Used for FAT formatting (Floppy Disk only) | |
| fat12MaxGeometry = 4; | |
| fat12Geometry:array[0..fat12MaxGeometry] of TFATGeometry = ( | |
| FloppyType,SectorsPerFat,SectorsPerCluster,RootEntryCount,MediaId | |
| (FloppyType:ft360K; SectorsPerFat:2; SectorsPerCluster:2; RootEntryCount:112; MediaId:$FD), | |
| (FloppyType:ft12M;SectorsPerFat:7;SectorsPerCluster:1;RootEntryCount:224;MediaId:$F9), | |
| (FloppyType:ft720K; SectorsPerFat:3; SectorsPerCluster:2; RootEntryCount:112; MediaId:$F9), | |
| (FloppyType:ft144M; SectorsPerFat:9; SectorsPerCluster:1; RootEntryCount:224; MediaId:$F0), | |
| (FloppyType:ft288M; SectorsPerFat:9; SectorsPerCluster:2; RootEntryCount:240; MediaId:$F0)); | |
 fat_ClusterSize* | Used for FAT formatting (Fixed Disk only) | |
| fat12MaxClusterSize = 2; | |
| fat12ClusterSize:array[0..fat12MaxClusterSize] of TFATClusterSize = ( | |
| SectorCount,SectorsPerCluster | |
| (SectorCount:6; SectorsPerCluster:0), | up to 3 KB, the 0 value for SectorsPerCluster trips an error (Not possible due to 512 Root entries) | 
| (SectorCount:798; SectorsPerCluster:0), | up to 399 KB, the 0 value for SectorsPerCluster trips an error | 
| (SectorCount:32623; SectorsPerCluster:8), | up to 15.9 MB, 4k cluster | 
| (SectorCount:$FFFFFFFF; SectorsPerCluster:0)); | greater than 15.9 MB, 0 value for SectorsPerCluster trips an error | 
| See FAT12 Notes for Min/Max and Values | |
| fat16MaxClusterSize = 8; | |
| fat16ClusterSize:array[0..fat16MaxClusterSize] of TFATClusterSize = ( | |
| SectorCount,SectorsPerCluster | |
| (SectorCount:8400; SectorsPerCluster:0), | up to 4.1 MB, the 0 value for SectorsPerCluster trips an error | 
| (SectorCount:32680; SectorsPerCluster:2), | up to 16 MB, 1k cluster | 
| (SectorCount:262144; SectorsPerCluster:4), | up to 128 MB, 2k cluster | 
| (SectorCount:524288; SectorsPerCluster:8), | up to 256 MB, 4k cluster | 
| (SectorCount:1048576; SectorsPerCluster:16), | up to 512 MB, 8k cluster | 
| The entries after this point are not used unless FAT16 is forced | |
| (SectorCount:2097152; SectorsPerCluster:32), | up to 1 GB, 16k cluster | 
| (SectorCount:4194304; SectorsPerCluster:64), | up to 2 GB, 32k cluster | 
| (SectorCount:8388608; SectorsPerCluster:128), | up to 4 GB, 64k cluster | 
| (SectorCount:$FFFFFFFF;SectorsPerCluster:0)); | greater than 4 GB, 0 value for SectorsPerCluster trips an error | 
| fat32MaxClusterSize = 5; | |
| fat32ClusterSize:array[0..fat32MaxClusterSize] of TFATClusterSize = ( | |
| SectorCount,SectorsPerCluster | |
| (SectorCount:66600; SectorsPerCluster:0), | up to 32.5 MB, the 0 value for SectorsPerCluster trips an error | 
| (SectorCount:532480 ;SectorsPerCluster:1), | up to 260 MB, .5k cluster | 
| The entries before this point are not used unless FAT32 is forced | |
| (SectorCount:16777216; SectorsPerCluster:8), | up to 8 GB, 4k cluster | 
| (SectorCount:33554432; SectorsPerCluster:16), | up to 16 GB, 8k cluster | 
| (SectorCount:67108864; SectorsPerCluster:32), | up to 32 GB, 16k cluster | 
| (SectorCount:$FFFFFFFF; SectorsPerCluster:64)); | greater than 32 GB, 32k cluster | 
 fatPartitionType* | fatMaxPartitionType = 5; | |
| fatPartitionType:array[0..fatMaxPartitionType] of TFATPartitionType = ( | |
| Excluded,Extended,SectorCount,PartitionId,LBAType | |
| Extended Partitions | |
| (Excluded:False; Extended:True; SectorCount:$FFFFFFFF ;PartitionId:pidExtended; LBAType:pidExtLBA), | any size, Extended | 
| Primary and Logical Partitions | |
| Unused | |
| (Excluded:False; Extended:False; SectorCount:798; PartitionId:pidUnused; LBAType:pidUnused), | up to 399 KB, Unused (Unused type trips an error) | 
| FAT12 | |
| (Excluded:False; Extended:False; SectorCount:32680; PartitionId:pidFAT12; LBAType:pidFAT12), | up to 16 MB, FAT12 | 
| FAT16 | |
| (Excluded:False; Extended:False; SectorCount:65360; PartitionId:pidFAT16; LBAType:pidFAT16LBA), | up to 32 MB, FAT16 | 
| (Excluded:False; Extended:False; SectorCount:1048576; PartitionId:pidFAT16HUGE; LBAType:pidFAT16LBA), | up to 512 MB, FAT16 HUGE | 
| FAT32 | |
| (Excluded:False; Extended:False; SectorCount:$FFFFFFFF; PartitionId:pidFAT32; LBAType:pidFAT32LBA)); | above 512 MB, FAT32 | 
Type definitions
FATFS specific types
| TFATType = (ftNONE,ftFAT12,ftFAT16,ftFAT32); | 
FATFS params
TFATParams = record 
| Note: (Not Packed) Block values for various Sector sizes. | |
| SectorSize:Word; | |
| FATType:TFATType; | |
| BlockShiftCount:Word; | |
| EntriesPerBlock:LongWord; | |
| SectorsPerBlock:LongWord; | |
FATFS geometry
TFATGeometry = record
| Note: (Not Packed) FAT12 values for various Floppy Types. | |
| FloppyType:TFloppyType; | |
| SectorsPerFat:LongWord; | |
| SectorsPerCluster:LongWord; | |
| RootEntryCount:LongWord; | |
| MediaId:Byte; | |
FATFS cluster size
TFATClusterSize = record
| Note: (Not Packed) Sectors per Cluster for various disk sizes (Fixed Disk only). | |
| SectorCount:LongWord; | Based on 512 bytes per sector only | 
| SectorsPerCluster:LongWord; | |
FATFS partition type
TFATPartitionType = record
| Note: (Not Packed) Partition Type for various disk sizes (Fixed Disk only). | |
| Excluded:Boolean; | Entry is excluded from FAT Partition Type check | 
| Extended:Boolean; | |
| SectorCount:LongWord; | Based on 512 bytes per sector only | 
| PartitionId:Byte; | |
| LBAType:Byte; | |
FATFS name
PFATName = ^TFATName;
TFATName = packed record
| Note: 11 Bytes | |
| Name:array[0..7] of Char; | File Name | 
| Ext:array[0..2] of Char; | File Extension | 
FATSFS directory
PFATDirectory = ^TFATDirectory;
TFATDirectory = packed record
| Note: 32 Bytes | |
| Name:array[0..7] of Char; | File Name | 
| Ext:array[0..2] of Char; | File Extension | 
| Attribute:Byte; | File Attributes - See Standard Values | 
| Reserved:Byte; | Always 0 (See below) | 
| CaseFlags:Byte; | Set to $08 if Name, $10 if Extension or $18 if both | 
| CreateTimeMsecs:Byte; | |
| CreateTime:Word; | |
| CreateDate:Word; | |
| LastAccessDate:Word; | |
| FirstClusterHigh:Word; | High word of the start cluster (FAT32 Only) | 
| WriteTime:Word; | |
| WriteDate:Word; | |
| FirstClusterLow:Word; | Low Word of Start Cluster all versions | 
| Length:LongWord; | |
LFN directory
PLFNDirectory = ^TLFNDirectory;
TLFNDirectory = packed record
| Note: 32 Bytes | |
| Order:Byte; | Bit 6 = 1 if last LFN in chain, Bits 5-0 = Order of LFN chunks for this File | 
| Name1:array[0..4] of WideChar; | First 5 Characters in this chunk | 
| Attribute:Byte; | Always 0Fh - Volume, System, Hidden, ReadOnly | 
| FileType:Byte; | Usually 00h | 
| Checksum:Byte; | Checksum calculated on 8.3 name | 
| Name2:array[0..5] of WideChar; | Next 6 Characters in this chunk | 
| Reserved:Word; | Always 0000h - FirstCluster in Standard Entry | 
| Name3:array[0..1] of WideChar; | Next 2 Characters in this chunk | 
FATFS info sector
PFATInfoSector = ^TFATInfoSector;
TFATInfoSector = packed record
| LeadSignature:LongWord; | Signature "RRaA" or 41615252h | 
| Reserved1:array[0..479] of Byte; | Always 0 | 
| StructureSignature:LongWord; | Signature "rrAa" or 61417272h | 
| FreeClusterCount:LongWord; | Free Clusters $FFFFFFFF if unknown | 
| LastFreeCluster:LongWord; | Most recently allocated cluster $FFFFFFFF if unknown | 
| Reserved2:array[0..11] of Byte; | Always 0 | 
| TrailSignature:LongWord; | Signature AA550000h | 
Class definitions
FATFS specific classes
| TFATRecognizer = class(TRecognizer) | 
| TFATPartitioner = class(TDiskPartitioner) | 
| TFATFormatter = class(TDiskFormatter) | 
| TFATDefragger = class(TDiskDefragger) | 
| TFATRepairer = class(TDiskRepairer) | 
| TFATFileSystem = class(TFileSystem) | 
| TFATDiskTable = class(TDiskTable) | 
| TFATDiskBlock = class(TDiskBlock) | 
| TFATDiskEntry = class(TDiskEntry) | 
Public variables
None defined
Function declarations
Initialization functions
procedure FATFSInit;
| Note | None documented | 
|---|
procedure FATFSQuit;
| Note | None documented | 
|---|
FATFS helper functions
function FATTypeToString(AType:TFATType):String;
| Note | None documented | 
|---|
Return to Unit Reference

