21 #include "user_config.h"
37 #if FF_DEFINED != 86606
38 #error Wrong include file (ff.h).
43 #define MAX_DIR 0x200000
44 #define MAX_DIR_EX 0x10000000
45 #define MAX_FAT12 0xFF5
46 #define MAX_FAT16 0xFFF5
47 #define MAX_FAT32 0x0FFFFFF5
48 #define MAX_EXFAT 0x7FFFFFFD
52 #define IsUpper(c) ((c) >= 'A' && (c) <= 'Z')
53 #define IsLower(c) ((c) >= 'a' && (c) <= 'z')
54 #define IsDigit(c) ((c) >= '0' && (c) <= '9')
55 #define IsSurrogate(c) ((c) >= 0xD800 && (c) <= 0xDFFF)
56 #define IsSurrogateH(c) ((c) >= 0xD800 && (c) <= 0xDBFF)
57 #define IsSurrogateL(c) ((c) >= 0xDC00 && (c) <= 0xDFFF)
61 #define FA_SEEKEND 0x20
62 #define FA_MODIFIED 0x40
81 #define NS_NONAME 0x80
85 #define ET_BITMAP 0x81
86 #define ET_UPCASE 0x82
87 #define ET_VLABEL 0x83
88 #define ET_FILEDIR 0x85
89 #define ET_STREAM 0xC0
90 #define ET_FILENAME 0xC1
98 #define BPB_BytsPerSec 11
99 #define BPB_SecPerClus 13
100 #define BPB_RsvdSecCnt 14
101 #define BPB_NumFATs 16
102 #define BPB_RootEntCnt 17
103 #define BPB_TotSec16 19
105 #define BPB_FATSz16 22
106 #define BPB_SecPerTrk 24
107 #define BPB_NumHeads 26
108 #define BPB_HiddSec 28
109 #define BPB_TotSec32 32
112 #define BS_BootSig 38
115 #define BS_FilSysType 54
116 #define BS_BootCode 62
119 #define BPB_FATSz32 36
120 #define BPB_ExtFlags32 40
121 #define BPB_FSVer32 42
122 #define BPB_RootClus32 44
123 #define BPB_FSInfo32 48
124 #define BPB_BkBootSec32 50
125 #define BS_DrvNum32 64
126 #define BS_NTres32 65
127 #define BS_BootSig32 66
128 #define BS_VolID32 67
129 #define BS_VolLab32 71
130 #define BS_FilSysType32 82
131 #define BS_BootCode32 90
133 #define BPB_ZeroedEx 11
134 #define BPB_VolOfsEx 64
135 #define BPB_TotSecEx 72
136 #define BPB_FatOfsEx 80
137 #define BPB_FatSzEx 84
138 #define BPB_DataOfsEx 88
139 #define BPB_NumClusEx 92
140 #define BPB_RootClusEx 96
141 #define BPB_VolIDEx 100
142 #define BPB_FSVerEx 104
143 #define BPB_VolFlagEx 106
144 #define BPB_BytsPerSecEx 108
145 #define BPB_SecPerClusEx 109
146 #define BPB_NumFATsEx 110
147 #define BPB_DrvNumEx 111
148 #define BPB_PercInUseEx 112
149 #define BPB_RsvdEx 113
150 #define BS_BootCodeEx 120
155 #define DIR_CrtTime10 13
156 #define DIR_CrtTime 14
157 #define DIR_LstAccDate 18
158 #define DIR_FstClusHI 20
159 #define DIR_ModTime 22
160 #define DIR_FstClusLO 26
161 #define DIR_FileSize 28
165 #define LDIR_Chksum 13
166 #define LDIR_FstClusLO 26
168 #define XDIR_NumLabel 1
170 #define XDIR_CaseSum 4
171 #define XDIR_NumSec 1
172 #define XDIR_SetSum 2
174 #define XDIR_CrtTime 8
175 #define XDIR_ModTime 12
176 #define XDIR_AccTime 16
177 #define XDIR_CrtTime10 20
178 #define XDIR_ModTime10 21
179 #define XDIR_CrtTZ 22
180 #define XDIR_ModTZ 23
181 #define XDIR_AccTZ 24
182 #define XDIR_GenFlags 33
183 #define XDIR_NumName 35
184 #define XDIR_NameHash 36
185 #define XDIR_ValidFileSize 40
186 #define XDIR_FstClus 52
187 #define XDIR_FileSize 56
194 #define FSI_LeadSig 0
195 #define FSI_StrucSig 484
196 #define FSI_Free_Count 488
197 #define FSI_Nxt_Free 492
199 #define MBR_Table 446
210 #define PTE_SizLba 12
216 #define GPTH_CurLba 24
217 #define GPTH_BakLba 32
218 #define GPTH_FstLba 40
219 #define GPTH_LstLba 48
220 #define GPTH_DskGuid 56
221 #define GPTH_PtOfs 72
222 #define GPTH_PtNum 80
223 #define GPTH_PteSize 84
224 #define GPTH_PtBcc 88
226 #define GPTE_PtGuid 0
227 #define GPTE_UpGuid 16
228 #define GPTE_FstLba 32
229 #define GPTE_LstLba 40
230 #define GPTE_Flags 48
235 #define ABORT(fs, res) { fp->err = (BYTE)(res); LEAVE_FF(fs, res); }
241 #error Static LFN work area cannot be used at thread-safe configuration
243 #define LEAVE_FF(fs, res) { unlock_fs(fs, res); return res; }
245 #define LEAVE_FF(fs, res) return res
250 #if FF_MULTI_PARTITION
251 #define LD2PD(vol) VolToPart[vol].pd
252 #define LD2PT(vol) VolToPart[vol].pt
254 #define LD2PD(vol) (BYTE)(vol)
260 #if (FF_MAX_SS < FF_MIN_SS) || (FF_MAX_SS != 512 && FF_MAX_SS != 1024 && FF_MAX_SS != 2048 && FF_MAX_SS != 4096) || (FF_MIN_SS != 512 && FF_MIN_SS != 1024 && FF_MIN_SS != 2048 && FF_MIN_SS != 4096)
261 #error Wrong sector size configuration
263 #if FF_MAX_SS == FF_MIN_SS
264 #define SS(fs) ((UINT)FF_MAX_SS)
266 #define SS(fs) ((fs)->ssize)
272 #if FF_NORTC_YEAR < 1980 || FF_NORTC_YEAR > 2107 || FF_NORTC_MON < 1 || FF_NORTC_MON > 12 || FF_NORTC_MDAY < 1 || FF_NORTC_MDAY > 31
273 #error Invalid FF_FS_NORTC settings
275 #define GET_FATTIME() ((DWORD)(FF_NORTC_YEAR - 1980) << 25 | (DWORD)FF_NORTC_MON << 21 | (DWORD)FF_NORTC_MDAY << 16)
277 #define GET_FATTIME() get_fattime()
284 #error FF_FS_LOCK must be 0 at read-only configuration
296 #define TBL_CT437 {0x80,0x9A,0x45,0x41,0x8E,0x41,0x8F,0x80,0x45,0x45,0x45,0x49,0x49,0x49,0x8E,0x8F, \
297 0x90,0x92,0x92,0x4F,0x99,0x4F,0x55,0x55,0x59,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
298 0x41,0x49,0x4F,0x55,0xA5,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
299 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
300 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
301 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
302 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
303 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
304 #define TBL_CT720 {0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
305 0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
306 0xA0,0xA1,0xA2,0xA3,0xA4,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
307 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
308 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
309 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
310 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
311 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
312 #define TBL_CT737 {0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
313 0x90,0x92,0x92,0x93,0x94,0x95,0x96,0x97,0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87, \
314 0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F,0x90,0x91,0xAA,0x92,0x93,0x94,0x95,0x96, \
315 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
316 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
317 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
318 0x97,0xEA,0xEB,0xEC,0xE4,0xED,0xEE,0xEF,0xF5,0xF0,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
319 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
320 #define TBL_CT771 {0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
321 0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
322 0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
323 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
324 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
325 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDC,0xDE,0xDE, \
326 0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
327 0xF0,0xF0,0xF2,0xF2,0xF4,0xF4,0xF6,0xF6,0xF8,0xF8,0xFA,0xFA,0xFC,0xFC,0xFE,0xFF}
328 #define TBL_CT775 {0x80,0x9A,0x91,0xA0,0x8E,0x95,0x8F,0x80,0xAD,0xED,0x8A,0x8A,0xA1,0x8D,0x8E,0x8F, \
329 0x90,0x92,0x92,0xE2,0x99,0x95,0x96,0x97,0x97,0x99,0x9A,0x9D,0x9C,0x9D,0x9E,0x9F, \
330 0xA0,0xA1,0xE0,0xA3,0xA3,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
331 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
332 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
333 0xB5,0xB6,0xB7,0xB8,0xBD,0xBE,0xC6,0xC7,0xA5,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
334 0xE0,0xE1,0xE2,0xE3,0xE5,0xE5,0xE6,0xE3,0xE8,0xE8,0xEA,0xEA,0xEE,0xED,0xEE,0xEF, \
335 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
336 #define TBL_CT850 {0x43,0x55,0x45,0x41,0x41,0x41,0x41,0x43,0x45,0x45,0x45,0x49,0x49,0x49,0x41,0x41, \
337 0x45,0x92,0x92,0x4F,0x4F,0x4F,0x55,0x55,0x59,0x4F,0x55,0x4F,0x9C,0x4F,0x9E,0x9F, \
338 0x41,0x49,0x4F,0x55,0xA5,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
339 0xB0,0xB1,0xB2,0xB3,0xB4,0x41,0x41,0x41,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
340 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0x41,0x41,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
341 0xD1,0xD1,0x45,0x45,0x45,0x49,0x49,0x49,0x49,0xD9,0xDA,0xDB,0xDC,0xDD,0x49,0xDF, \
342 0x4F,0xE1,0x4F,0x4F,0x4F,0x4F,0xE6,0xE8,0xE8,0x55,0x55,0x55,0x59,0x59,0xEE,0xEF, \
343 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
344 #define TBL_CT852 {0x80,0x9A,0x90,0xB6,0x8E,0xDE,0x8F,0x80,0x9D,0xD3,0x8A,0x8A,0xD7,0x8D,0x8E,0x8F, \
345 0x90,0x91,0x91,0xE2,0x99,0x95,0x95,0x97,0x97,0x99,0x9A,0x9B,0x9B,0x9D,0x9E,0xAC, \
346 0xB5,0xD6,0xE0,0xE9,0xA4,0xA4,0xA6,0xA6,0xA8,0xA8,0xAA,0x8D,0xAC,0xB8,0xAE,0xAF, \
347 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBD,0xBF, \
348 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC6,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
349 0xD1,0xD1,0xD2,0xD3,0xD2,0xD5,0xD6,0xD7,0xB7,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
350 0xE0,0xE1,0xE2,0xE3,0xE3,0xD5,0xE6,0xE6,0xE8,0xE9,0xE8,0xEB,0xED,0xED,0xDD,0xEF, \
351 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xEB,0xFC,0xFC,0xFE,0xFF}
352 #define TBL_CT855 {0x81,0x81,0x83,0x83,0x85,0x85,0x87,0x87,0x89,0x89,0x8B,0x8B,0x8D,0x8D,0x8F,0x8F, \
353 0x91,0x91,0x93,0x93,0x95,0x95,0x97,0x97,0x99,0x99,0x9B,0x9B,0x9D,0x9D,0x9F,0x9F, \
354 0xA1,0xA1,0xA3,0xA3,0xA5,0xA5,0xA7,0xA7,0xA9,0xA9,0xAB,0xAB,0xAD,0xAD,0xAE,0xAF, \
355 0xB0,0xB1,0xB2,0xB3,0xB4,0xB6,0xB6,0xB8,0xB8,0xB9,0xBA,0xBB,0xBC,0xBE,0xBE,0xBF, \
356 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC7,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
357 0xD1,0xD1,0xD3,0xD3,0xD5,0xD5,0xD7,0xD7,0xDD,0xD9,0xDA,0xDB,0xDC,0xDD,0xE0,0xDF, \
358 0xE0,0xE2,0xE2,0xE4,0xE4,0xE6,0xE6,0xE8,0xE8,0xEA,0xEA,0xEC,0xEC,0xEE,0xEE,0xEF, \
359 0xF0,0xF2,0xF2,0xF4,0xF4,0xF6,0xF6,0xF8,0xF8,0xFA,0xFA,0xFC,0xFC,0xFD,0xFE,0xFF}
360 #define TBL_CT857 {0x80,0x9A,0x90,0xB6,0x8E,0xB7,0x8F,0x80,0xD2,0xD3,0xD4,0xD8,0xD7,0x49,0x8E,0x8F, \
361 0x90,0x92,0x92,0xE2,0x99,0xE3,0xEA,0xEB,0x98,0x99,0x9A,0x9D,0x9C,0x9D,0x9E,0x9E, \
362 0xB5,0xD6,0xE0,0xE9,0xA5,0xA5,0xA6,0xA6,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
363 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
364 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC7,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
365 0xD0,0xD1,0xD2,0xD3,0xD4,0x49,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
366 0xE0,0xE1,0xE2,0xE3,0xE5,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xDE,0xED,0xEE,0xEF, \
367 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
368 #define TBL_CT860 {0x80,0x9A,0x90,0x8F,0x8E,0x91,0x86,0x80,0x89,0x89,0x92,0x8B,0x8C,0x98,0x8E,0x8F, \
369 0x90,0x91,0x92,0x8C,0x99,0xA9,0x96,0x9D,0x98,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
370 0x86,0x8B,0x9F,0x96,0xA5,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
371 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
372 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
373 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
374 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
375 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
376 #define TBL_CT861 {0x80,0x9A,0x90,0x41,0x8E,0x41,0x8F,0x80,0x45,0x45,0x45,0x8B,0x8B,0x8D,0x8E,0x8F, \
377 0x90,0x92,0x92,0x4F,0x99,0x8D,0x55,0x97,0x97,0x99,0x9A,0x9D,0x9C,0x9D,0x9E,0x9F, \
378 0xA4,0xA5,0xA6,0xA7,0xA4,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
379 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
380 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
381 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
382 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
383 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
384 #define TBL_CT862 {0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
385 0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
386 0x41,0x49,0x4F,0x55,0xA5,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
387 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
388 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
389 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
390 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
391 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
392 #define TBL_CT863 {0x43,0x55,0x45,0x41,0x41,0x41,0x86,0x43,0x45,0x45,0x45,0x49,0x49,0x8D,0x41,0x8F, \
393 0x45,0x45,0x45,0x4F,0x45,0x49,0x55,0x55,0x98,0x4F,0x55,0x9B,0x9C,0x55,0x55,0x9F, \
394 0xA0,0xA1,0x4F,0x55,0xA4,0xA5,0xA6,0xA7,0x49,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
395 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
396 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
397 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
398 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
399 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
400 #define TBL_CT864 {0x80,0x9A,0x45,0x41,0x8E,0x41,0x8F,0x80,0x45,0x45,0x45,0x49,0x49,0x49,0x8E,0x8F, \
401 0x90,0x92,0x92,0x4F,0x99,0x4F,0x55,0x55,0x59,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
402 0x41,0x49,0x4F,0x55,0xA5,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
403 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
404 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
405 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
406 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
407 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
408 #define TBL_CT865 {0x80,0x9A,0x90,0x41,0x8E,0x41,0x8F,0x80,0x45,0x45,0x45,0x49,0x49,0x49,0x8E,0x8F, \
409 0x90,0x92,0x92,0x4F,0x99,0x4F,0x55,0x55,0x59,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
410 0x41,0x49,0x4F,0x55,0xA5,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
411 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
412 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
413 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
414 0xE0,0xE1,0xE2,0xE3,0xE4,0xE5,0xE6,0xE7,0xE8,0xE9,0xEA,0xEB,0xEC,0xED,0xEE,0xEF, \
415 0xF0,0xF1,0xF2,0xF3,0xF4,0xF5,0xF6,0xF7,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
416 #define TBL_CT866 {0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
417 0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
418 0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
419 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
420 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
421 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xD6,0xD7,0xD8,0xD9,0xDA,0xDB,0xDC,0xDD,0xDE,0xDF, \
422 0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9A,0x9B,0x9C,0x9D,0x9E,0x9F, \
423 0xF0,0xF0,0xF2,0xF2,0xF4,0xF4,0xF6,0xF6,0xF8,0xF9,0xFA,0xFB,0xFC,0xFD,0xFE,0xFF}
424 #define TBL_CT869 {0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8A,0x8B,0x8C,0x8D,0x8E,0x8F, \
425 0x90,0x91,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9A,0x86,0x9C,0x8D,0x8F,0x90, \
426 0x91,0x90,0x92,0x95,0xA4,0xA5,0xA6,0xA7,0xA8,0xA9,0xAA,0xAB,0xAC,0xAD,0xAE,0xAF, \
427 0xB0,0xB1,0xB2,0xB3,0xB4,0xB5,0xB6,0xB7,0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0xBE,0xBF, \
428 0xC0,0xC1,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7,0xC8,0xC9,0xCA,0xCB,0xCC,0xCD,0xCE,0xCF, \
429 0xD0,0xD1,0xD2,0xD3,0xD4,0xD5,0xA4,0xA5,0xA6,0xD9,0xDA,0xDB,0xDC,0xA7,0xA8,0xDF, \
430 0xA9,0xAA,0xAC,0xAD,0xB5,0xB6,0xB7,0xB8,0xBD,0xBE,0xC6,0xC7,0xCF,0xCF,0xD0,0xEF, \
431 0xF0,0xF1,0xD1,0xD2,0xD3,0xF5,0xD4,0xF7,0xF8,0xF9,0xD5,0x96,0x95,0x98,0xFE,0xFF}
436 #define TBL_DC932 {0x81, 0x9F, 0xE0, 0xFC, 0x40, 0x7E, 0x80, 0xFC, 0x00, 0x00}
437 #define TBL_DC936 {0x81, 0xFE, 0x00, 0x00, 0x40, 0x7E, 0x80, 0xFE, 0x00, 0x00}
438 #define TBL_DC949 {0x81, 0xFE, 0x00, 0x00, 0x41, 0x5A, 0x61, 0x7A, 0x81, 0xFE}
439 #define TBL_DC950 {0x81, 0xFE, 0x00, 0x00, 0x40, 0x7E, 0xA1, 0xFE, 0x00, 0x00}
443 #define MERGE_2STR(a, b) a ## b
444 #define MKCVTBL(hd, cp) MERGE_2STR(hd, cp)
462 #if FF_VOLUMES < 1 || FF_VOLUMES > 10
463 #error Wrong FF_VOLUMES setting
477 #ifdef FF_VOLUME_STRS
483 #if FF_MIN_GPT > 0x100000000
484 #error Wrong FF_MIN_GPT setting
486 static const BYTE GUID_MS_Basic[16] = {0xA2,0xA0,0xD0,0xEB,0xE5,0xB9,0x33,0x44,0x87,0xC0,0x68,0xB6,0xB7,0x26,0x99,0xC7};
497 #error LFN must be enabled when enable exFAT
500 #define INIT_NAMBUF(fs)
501 #define FREE_NAMBUF()
502 #define LEAVE_MKFS(res) return res
505 #if FF_MAX_LFN < 12 || FF_MAX_LFN > 255
506 #error Wrong setting of FF_MAX_LFN
508 #if FF_LFN_BUF < FF_SFN_BUF || FF_SFN_BUF < 12
509 #error Wrong setting of FF_LFN_BUF or FF_SFN_BUF
511 #if FF_LFN_UNICODE < 0 || FF_LFN_UNICODE > 3
512 #error Wrong setting of FF_LFN_UNICODE
514 static const BYTE LfnOfs[] = {1,3,5,7,9,14,16,18,20,22,24,28,30};
515 #define MAXDIRB(nc) ((nc + 44U) / 15 * SZDIRE)
523 #define INIT_NAMBUF(fs)
524 #define FREE_NAMBUF()
525 #define LEAVE_MKFS(res) return res
527 #elif FF_USE_LFN == 2
529 #define DEF_NAMBUF WCHAR lbuf[FF_MAX_LFN+1]; BYTE dbuf[MAXDIRB(FF_MAX_LFN)];
530 #define INIT_NAMBUF(fs) { (fs)->lfnbuf = lbuf; (fs)->dirbuf = dbuf; }
531 #define FREE_NAMBUF()
533 #define DEF_NAMBUF WCHAR lbuf[FF_MAX_LFN+1];
534 #define INIT_NAMBUF(fs) { (fs)->lfnbuf = lbuf; }
535 #define FREE_NAMBUF()
537 #define LEAVE_MKFS(res) return res
539 #elif FF_USE_LFN == 3
541 #define DEF_NAMBUF WCHAR *lfn;
542 #define INIT_NAMBUF(fs) { lfn = ff_memalloc((FF_MAX_LFN+1)*2 + MAXDIRB(FF_MAX_LFN)); if (!lfn) LEAVE_FF(fs, FR_NOT_ENOUGH_CORE); (fs)->lfnbuf = lfn; (fs)->dirbuf = (BYTE*)(lfn+FF_MAX_LFN+1); }
543 #define FREE_NAMBUF() ff_memfree(lfn)
545 #define DEF_NAMBUF WCHAR *lfn;
546 #define INIT_NAMBUF(fs) { lfn = ff_memalloc((FF_MAX_LFN+1)*2); if (!lfn) LEAVE_FF(fs, FR_NOT_ENOUGH_CORE); (fs)->lfnbuf = lfn; }
547 #define FREE_NAMBUF() ff_memfree(lfn)
549 #define LEAVE_MKFS(res) { if (!work) ff_memfree(buf); return res; }
551 #define MAX_MALLOC (FF_MAX_SS*4)
554 #error Wrong setting of FF_USE_LFN
565 #if FF_CODE_PAGE == 0
566 #define CODEPAGE CodePage
567 static WORD CodePage;
592 #elif FF_CODE_PAGE < 900
593 #define CODEPAGE FF_CODE_PAGE
597 #define CODEPAGE FF_CODE_PAGE
621 rv = rv << 8 | ptr[0];
630 rv = rv << 8 | ptr[2];
631 rv = rv << 8 | ptr[1];
632 rv = rv << 8 | ptr[0];
637 static QWORD ld_qword (
const BYTE* ptr)
642 rv = rv << 8 | ptr[6];
643 rv = rv << 8 | ptr[5];
644 rv = rv << 8 | ptr[4];
645 rv = rv << 8 | ptr[3];
646 rv = rv << 8 | ptr[2];
647 rv = rv << 8 | ptr[1];
648 rv = rv << 8 | ptr[0];
656 *ptr++ = (
BYTE)val; val >>= 8;
662 *ptr++ = (
BYTE)val; val >>= 8;
663 *ptr++ = (
BYTE)val; val >>= 8;
664 *ptr++ = (
BYTE)val; val >>= 8;
669 static void st_qword (
BYTE* ptr, QWORD val)
671 *ptr++ = (
BYTE)val; val >>= 8;
672 *ptr++ = (
BYTE)val; val >>= 8;
673 *ptr++ = (
BYTE)val; val >>= 8;
674 *ptr++ = (
BYTE)val; val >>= 8;
675 *ptr++ = (
BYTE)val; val >>= 8;
676 *ptr++ = (
BYTE)val; val >>= 8;
677 *ptr++ = (
BYTE)val; val >>= 8;
722 }
while (--cnt && r == 0);
731 while (*str && *str != chr) str++;
739 #if FF_CODE_PAGE == 0
740 if (DbcTbl && c >= DbcTbl[0]) {
741 if (c <= DbcTbl[1])
return 1;
742 if (c >= DbcTbl[2] && c <= DbcTbl[3])
return 1;
744 #elif FF_CODE_PAGE >= 900
745 if (c >= DbcTbl[0]) {
746 if (c <= DbcTbl[1])
return 1;
747 if (c >= DbcTbl[2] && c <= DbcTbl[3])
return 1;
750 if (c != 0)
return 0;
759 #if FF_CODE_PAGE == 0
760 if (DbcTbl && c >= DbcTbl[4]) {
761 if (c <= DbcTbl[5])
return 1;
762 if (c >= DbcTbl[6] && c <= DbcTbl[7])
return 1;
763 if (c >= DbcTbl[8] && c <= DbcTbl[9])
return 1;
765 #elif FF_CODE_PAGE >= 900
766 if (c >= DbcTbl[4]) {
767 if (c <= DbcTbl[5])
return 1;
768 if (c >= DbcTbl[6] && c <= DbcTbl[7])
return 1;
769 if (c >= DbcTbl[8] && c <= DbcTbl[9])
return 1;
772 if (c != 0)
return 0;
786 const TCHAR *p = *str;
788 #if FF_LFN_UNICODE == 1
798 #elif FF_LFN_UNICODE == 2
804 if ((uc & 0xE0) == 0xC0) {
807 if ((uc & 0xF0) == 0xE0) {
810 if ((uc & 0xF8) == 0xF0) {
819 if ((b & 0xC0) != 0x80)
return 0xFFFFFFFF;
820 uc = uc << 6 | (b & 0x3F);
822 if (uc < 0x80 ||
IsSurrogate(uc) || uc >= 0x110000)
return 0xFFFFFFFF;
823 if (uc >= 0x010000) uc = 0xD800DC00 | ((uc - 0x10000) << 6 & 0x3FF0000) | (uc & 0x3FF);
826 #elif FF_LFN_UNICODE == 3
828 if (uc >= 0x110000 ||
IsSurrogate(uc))
return 0xFFFFFFFF;
829 if (uc >= 0x010000) uc = 0xD800DC00 | ((uc - 0x10000) << 6 & 0x3FF0000) | (uc & 0x3FF);
838 if (!
dbc_2nd(b))
return 0xFFFFFFFF;
843 if (wc == 0)
return 0xFFFFFFFF;
860 #if FF_LFN_UNICODE == 1
863 hs = (
WCHAR)(chr >> 16);
875 #elif FF_LFN_UNICODE == 2
879 if (szb < 1)
return 0;
884 if (szb < 2)
return 0;
885 *buf++ = (
TCHAR)(0xC0 | (chr >> 6 & 0x1F));
886 *buf++ = (
TCHAR)(0x80 | (chr >> 0 & 0x3F));
891 *buf++ = (
TCHAR)(0xE0 | (chr >> 12 & 0x0F));
892 *buf++ = (
TCHAR)(0x80 | (chr >> 6 & 0x3F));
893 *buf++ = (
TCHAR)(0x80 | (chr >> 0 & 0x3F));
897 if (szb < 4)
return 0;
898 hc = ((chr & 0xFFFF0000) - 0xD8000000) >> 6;
899 chr = (chr & 0xFFFF) - 0xDC00;
900 if (hc >= 0x100000 || chr >= 0x400)
return 0;
901 chr = (hc | chr) + 0x10000;
902 *buf++ = (
TCHAR)(0xF0 | (chr >> 18 & 0x07));
903 *buf++ = (
TCHAR)(0x80 | (chr >> 12 & 0x3F));
904 *buf++ = (
TCHAR)(0x80 | (chr >> 6 & 0x3F));
905 *buf++ = (
TCHAR)(0x80 | (chr >> 0 & 0x3F));
908 #elif FF_LFN_UNICODE == 3
911 if (szb < 1)
return 0;
912 if (chr >= 0x10000) {
913 hc = ((chr & 0xFFFF0000) - 0xD8000000) >> 6;
914 chr = (chr & 0xFFFF) - 0xDC00;
915 if (hc >= 0x100000 || chr >= 0x400)
return 0;
916 chr = (hc | chr) + 0x10000;
926 if (szb < 2)
return 0;
927 *buf++ = (char)(wc >> 8);
931 if (wc == 0 || szb < 1)
return 0;
947 return ff_req_grant(fs->sobj);
951 static void unlock_fs (
957 ff_rel_grant(fs->sobj);
981 if (Files[i].fs == dp->
obj.
fs &&
983 Files[i].ofs == dp->
dptr)
break;
997 static int enq_lock (
void)
1001 for (i = 0; i <
FF_FS_LOCK && Files[i].fs; i++) ;
1006 static UINT inc_lock (
1015 if (Files[i].fs == dp->
obj.
fs
1017 && Files[i].ofs == dp->
dptr)
break;
1021 for (i = 0; i <
FF_FS_LOCK && Files[i].fs; i++) ;
1023 Files[i].fs = dp->
obj.
fs;
1025 Files[i].ofs = dp->
dptr;
1029 if (acc >= 1 && Files[i].ctr)
return 0;
1031 Files[i].ctr = acc ? 0x100 : Files[i].ctr + 1;
1047 if (n == 0x100) n = 0;
1050 if (n == 0) Files[i].fs = 0;
1059 static void clear_lock (
1066 if (Files[i].fs == fs) Files[i].fs = 0;
1114 sect = (
LBA_t)0 - 1;
1174 if (clst >= fs->
n_fatent - 2)
return 0;
1195 if (clst < 2 || clst >= fs->
n_fatent) {
1203 bc = (
UINT)clst; bc += bc / 2;
1205 wc = fs->
win[bc++ %
SS(fs)];
1207 wc |= fs->
win[bc %
SS(fs)] << 8;
1208 val = (clst & 1) ? (wc >> 4) : (wc & 0xFFF);
1226 if (obj->
stat == 2 && cofs <= clen) {
1227 val = (cofs == clen) ? 0x7FFFFFFF : clst + 1;
1230 if (obj->
stat == 3 && cofs < obj->n_cont) {
1234 if (obj->
stat != 2) {
1235 if (obj->n_frag != 0) {
1273 if (clst >= 2 && clst < fs->n_fatent) {
1276 bc = (
UINT)clst; bc += bc / 2;
1278 if (res !=
FR_OK)
break;
1279 p = fs->
win + bc++ %
SS(fs);
1280 *p = (clst & 1) ? ((*p & 0x0F) | ((
BYTE)val << 4)) : (
BYTE)val;
1283 if (res !=
FR_OK)
break;
1284 p = fs->
win + bc %
SS(fs);
1285 *p = (clst & 1) ? (
BYTE)(val >> 4) : ((*p & 0xF0) | ((
BYTE)(val >> 8) & 0x0F));
1291 if (res !=
FR_OK)
break;
1301 if (res !=
FR_OK)
break;
1303 val = (val & 0x0FFFFFFF) | (
ld_dword(fs->
win + clst * 4 %
SS(fs)) & 0xF0000000);
1318 #if FF_FS_EXFAT && !FF_FS_READONLY
1327 static DWORD find_bitmap (
1335 DWORD val, scl, ctr;
1339 if (clst >= fs->
n_fatent - 2) clst = 0;
1340 scl = val = clst; ctr = 0;
1343 i = val / 8 %
SS(fs); bm = 1 << (val % 8);
1346 bv = fs->
win[i] & bm; bm <<= 1;
1348 val = 0; bm = 0; i =
SS(fs);
1351 if (++ctr == ncl)
return scl + 2;
1355 if (val == clst)
return 0;
1358 }
while (++i <
SS(fs));
1367 static FRESULT change_bitmap (
1380 sect = fs->bitbase + clst / 8 /
SS(fs);
1381 i = clst / 8 %
SS(fs);
1382 bm = 1 << (clst % 8);
1390 if (--ncl == 0)
return FR_OK;
1393 }
while (++i <
SS(fs));
1403 static FRESULT fill_first_frag (
1411 if (obj->
stat == 3) {
1412 for (cl = obj->
sclust, n = obj->n_cont; n; cl++, n--) {
1414 if (res !=
FR_OK)
return res;
1426 static FRESULT fill_last_frag (
1435 while (obj->n_frag > 0) {
1436 res =
put_fat(obj->
fs, lcl - obj->n_frag + 1, (obj->n_frag > 1) ? lcl - obj->n_frag + 2 : term);
1437 if (res !=
FR_OK)
return res;
1461 #if FF_FS_EXFAT || FF_USE_TRIM
1462 DWORD scl = clst, ecl = clst;
1472 res =
put_fat(fs, pclst, 0xFFFFFFFF);
1473 if (res !=
FR_OK)
return res;
1479 if (nxt == 0)
break;
1484 if (res !=
FR_OK)
return res;
1490 #if FF_FS_EXFAT || FF_USE_TRIM
1491 if (ecl + 1 == nxt) {
1496 res = change_bitmap(fs, scl, ecl - scl + 1, 0);
1497 if (res !=
FR_OK)
return res;
1509 }
while (clst < fs->n_fatent);
1517 if (obj->
stat == 0) {
1519 while (clst != pclst) {
1523 if (nxt != clst + 1)
break;
1526 if (clst == pclst) {
1530 if (obj->
stat == 3 && pclst >= obj->
sclust && pclst <= obj->sclust + obj->n_cont) {
1559 if (scl == 0 || scl >= fs->
n_fatent) scl = 1;
1563 if (cs < 2)
return 1;
1564 if (cs == 0xFFFFFFFF)
return cs;
1565 if (cs < fs->n_fatent)
return cs;
1572 ncl = find_bitmap(fs, scl, 1);
1573 if (ncl == 0 || ncl == 0xFFFFFFFF)
return ncl;
1574 res = change_bitmap(fs, ncl, 1, 1);
1580 if (obj->
stat == 2 && ncl != scl + 1) {
1581 obj->n_cont = scl - obj->
sclust;
1585 if (obj->
stat != 2) {
1586 if (ncl == clst + 1) {
1587 obj->n_frag = obj->n_frag ? obj->n_frag + 1 : 2;
1589 if (obj->n_frag == 0) obj->n_frag = 1;
1590 res = fill_last_frag(obj, clst, ncl);
1591 if (res ==
FR_OK) obj->n_frag = 1;
1602 if (cs == 1 || cs == 0xFFFFFFFF)
return cs;
1605 if (cs >= 2 && cs < fs->n_fatent) scl = cs;
1615 if (ncl > scl)
return 0;
1619 if (cs == 1 || cs == 0xFFFFFFFF)
return cs;
1620 if (ncl == scl)
return 0;
1623 res =
put_fat(fs, ncl, 0xFFFFFFFF);
1624 if (res ==
FR_OK && clst != 0) {
1655 DWORD cl, ncl, *tbl;
1659 tbl = fp->
cltbl + 1;
1663 if (ncl == 0)
return 0;
1664 if (cl < ncl)
break;
1705 ibuf = fs->
win; szb = 1;
1744 while (ofs >= csz) {
1754 dp->
sect += ofs /
SS(fs);
1755 dp->
dir = fs->
win + (ofs %
SS(fs));
1780 if (ofs %
SS(fs) == 0) {
1783 if (dp->
clust == 0) {
1789 if ((ofs /
SS(fs) & (fs->
csize - 1)) == 0) {
1805 if (!stretch) dp->
sect = 0;
1843 if (res !=
FR_OK)
break;
1849 if (++n == nent)
break;
1854 }
while (res ==
FR_OK);
1908 const WCHAR* lfnbuf,
1918 i = ((dir[
LDIR_Ord] & 0x3F) - 1) * 13;
1920 for (wc = 1, s = 0; s < 13; s++) {
1928 if (uc != 0xFFFF)
return 0;
1932 if ((dir[
LDIR_Ord] &
LLEF) && wc && lfnbuf[i])
return 0;
1938 #if FF_FS_MINIMIZE <= 1 || FF_FS_RPATH >= 2 || FF_USE_LABEL || FF_FS_EXFAT
1956 for (wc = 1, s = 0; s < 13; s++) {
1960 lfnbuf[i++] = wc = uc;
1962 if (uc != 0xFFFF)
return 0;
2000 if (wc != 0xFFFF) wc = lfn[i++];
2002 if (wc == 0) wc = 0xFFFF;
2004 if (wc == 0xFFFF || !lfn[i]) ord |=
LLEF;
2013 #if FF_USE_LFN && !FF_FS_READONLY
2037 for (i = 0; i < 16; i++) {
2038 sreg = (sreg << 1) + (wc & 1);
2040 if (sreg & 0x10000) sreg ^= 0x11021;
2049 c = (
BYTE)((seq % 16) +
'0');
2050 if (c >
'9') c += 7;
2057 for (j = 0; j < i &&
dst[j] !=
' '; j++) {
2059 if (j == i - 1)
break;
2064 dst[j++] = (i < 8) ? ns[i++] :
' ';
2098 static WORD xdir_sum (
2107 for (i =
sum = 0; i < szblk; i++) {
2111 sum = ((
sum & 1) ? 0x8000 : 0) + (
sum >> 1) + dir[i];
2119 static WORD xname_sum (
2127 while ((chr = *name++) != 0) {
2129 sum = ((
sum & 1) ? 0x8000 : 0) + (
sum >> 1) + (chr & 0xFF);
2130 sum = ((
sum & 1) ? 0x8000 : 0) + (
sum >> 1) + (chr >> 8);
2136 #if !FF_FS_READONLY && FF_USE_MKFS
2137 static DWORD xsum32 (
2142 sum = ((
sum & 1) ? 0x80000000 : 0) + (
sum >> 1) + dat;
2148 #if FF_FS_MINIMIZE <= 1 || FF_FS_RPATH >= 2
2153 static void get_xfileinfo (
2163 nc = 0; hs = 0; di = 0;
2166 if ((si %
SZDIRE) == 0) si += 2;
2167 wc =
ld_word(dirb + si); si += 2; nc++;
2172 if (wc == 0) { di = 0;
break; }
2176 if (hs != 0) di = 0;
2177 if (di == 0) fno->
fname[di++] =
'?';
2205 if (res !=
FR_OK)
return res;
2214 if (res !=
FR_OK)
return res;
2216 if (res !=
FR_OK)
return res;
2226 if (res !=
FR_OK)
return res;
2228 if (res !=
FR_OK)
return res;
2231 }
while ((i +=
SZDIRE) < sz_ent);
2245 static void init_alloc_info (
2258 #if !FF_FS_READONLY || FF_FS_RPATH != 0
2263 static FRESULT load_obj_xdir (
2280 res = load_xdir(dp);
2306 while (res ==
FR_OK) {
2308 if (res !=
FR_OK)
break;
2311 if (--nent == 0)
break;
2324 static void create_xdir (
2341 nlen = nc1 = 0; wc = 1;
2345 if (wc != 0 && (wc = lfn[nlen]) != 0) nlen++;
2348 }
while (i %
SZDIRE != 0);
2350 }
while (lfn[nlen]);
2362 #if FF_FS_MINIMIZE <= 1 || FF_FS_RPATH >= 2 || FF_USE_LABEL || FF_FS_EXFAT
2367 #define DIR_READ_FILE(dp) dir_read(dp, 0)
2368 #define DIR_READ_LABEL(dp) dir_read(dp, 1)
2384 if (res !=
FR_OK)
break;
2396 res = load_xdir(dp);
2433 if (res !=
FR_OK)
break;
2460 if (res !=
FR_OK)
return res;
2468 #if FF_MAX_LFN < 255
2472 for (nc = fs->dirbuf[
XDIR_NumName], di =
SZDIRE * 2, ni = 0; nc; nc--, di += 2, ni++) {
2473 if ((di %
SZDIRE) == 0) di += 2;
2476 if (nc == 0 && !fs->
lfnbuf[ni])
break;
2487 if (res !=
FR_OK)
break;
2493 ord = 0xFF; dp->
blk_ofs = 0xFFFFFFFF;
2508 ord = 0xFF; dp->
blk_ofs = 0xFFFFFFFF;
2516 }
while (res ==
FR_OK);
2541 for (nlen = 0; fs->
lfnbuf[nlen]; nlen++) ;
2545 nent = (nlen + 14) / 15 + 2;
2547 if (res !=
FR_OK)
return res;
2552 res = fill_first_frag(&dp->
obj);
2553 if (res !=
FR_OK)
return res;
2554 res = fill_last_frag(&dp->
obj, dp->
clust, 0xFFFFFFFF);
2555 if (res !=
FR_OK)
return res;
2559 res = load_obj_xdir(&dj, &dp->
obj);
2560 if (res !=
FR_OK)
return res;
2565 res = store_xdir(&dj);
2566 if (res !=
FR_OK)
return res;
2570 create_xdir(fs->dirbuf, fs->
lfnbuf);
2578 for (n = 1; n < 100; n++) {
2581 if (res !=
FR_OK)
break;
2589 nent = (sn[
NSFLAG] &
NS_LFN) ? (nlen + 12) / 13 + 1 : 1;
2591 if (res ==
FR_OK && --nent) {
2597 if (res !=
FR_OK)
break;
2601 }
while (res ==
FR_OK && --nent);
2630 #if !FF_FS_READONLY && FF_FS_MINIMIZE == 0
2648 if (res !=
FR_OK)
break;
2655 if (dp->
dptr >= last)
break;
2657 }
while (res ==
FR_OK);
2676 #if FF_FS_MINIMIZE <= 1 || FF_FS_RPATH >= 2
2697 if (dp->
sect == 0)
return;
2702 get_xfileinfo(fs->dirbuf, fno);
2707 if (dp->
blk_ofs != 0xFFFFFFFF) {
2709 while (fs->
lfnbuf[si] != 0) {
2715 if (wc == 0) { di = 0;
break; }
2719 if (hs != 0) di = 0;
2727 if (wc ==
' ')
continue;
2730 #if FF_LFN_UNICODE >= 1
2732 wc = wc << 8 | dp->
dir[si++];
2735 if (wc == 0) { di = 0;
break; }
2737 if (wc == 0) { di = 0;
break; }
2745 if (fno->
fname[0] == 0) {
2747 fno->
fname[di++] =
'?';
2751 if (wc ==
'.') lcf =
NS_EXT;
2764 if (c ==
' ')
continue;
2766 if (si == 9) fno->
fname[di++] =
'.';
2767 fno->
fname[di++] = c;
2782 #if FF_USE_FIND && FF_FS_MINIMIZE <= 1
2794 #if FF_USE_LFN && FF_LFN_UNICODE >= 1
2796 if (chr == 0xFFFFFFFF) chr = 0;
2800 chr = (
BYTE)*(*ptr)++;
2801 if (
IsLower(chr)) chr -= 0x20;
2802 #if FF_CODE_PAGE == 0
2803 if (
ExCvt && chr >= 0x80) chr =
ExCvt[chr - 0x80];
2804 #elif FF_CODE_PAGE < 900
2805 if (chr >= 0x80) chr =
ExCvt[chr - 0x80];
2807 #if FF_CODE_PAGE == 0 || FF_CODE_PAGE >= 900
2825 const TCHAR *pp, *np;
2833 if (*pat == 0 && inf)
return 1;
2838 if (*pp ==
'?' || *pp ==
'*') {
2841 if (*pp++ ==
'?') nm++;
else nx = 1;
2842 }
while (*pp ==
'?' || *pp ==
'*');
2848 if (pc != nc)
break;
2849 if (pc == 0)
return 1;
2852 }
while (inf && nc);
2883 if (uc >= 0x10000) lfn[di++] = (
WCHAR)(uc >> 16);
2885 if (wc <
' ' || wc ==
'/' || wc ==
'\\')
break;
2894 while (*p ==
'/' || *p ==
'\\') p++;
2898 #if FF_FS_RPATH != 0
2899 if ((di == 1 && lfn[di - 1] ==
'.') ||
2900 (di == 2 && lfn[di - 1] ==
'.' && lfn[di - 2] ==
'.')) {
2902 for (i = 0; i < 11; i++) {
2903 dp->
fn[i] = (i < di) ?
'.' :
' ';
2911 if (wc !=
' ' && wc !=
'.')
break;
2918 for (si = 0; lfn[si] ==
' '; si++) ;
2920 while (di > 0 && lfn[di - 1] !=
'.') di--;
2927 if (wc ==
' ' || (wc ==
'.' && si != di)) {
2932 if (i >= ni || si == di) {
2939 si = di; i = 8; ni = 11; b <<= 2;
2945 #if FF_CODE_PAGE == 0
2948 if (wc & 0x80) wc =
ExCvt[wc & 0x7F];
2952 #elif FF_CODE_PAGE < 900
2954 if (wc & 0x80) wc =
ExCvt[wc & 0x7F];
2965 dp->
fn[i++] = (
BYTE)(wc >> 8);
2967 if (wc == 0 ||
chk_chr(
"+,;=[]", wc)) {
2983 if (ni == 8) b <<= 2;
2984 if ((b & 0x0C) == 0x0C || (b & 0x03) == 0x03) cf |=
NS_LFN;
2986 if (b & 0x01) cf |=
NS_EXT;
3001 p = *path; sfn = dp->
fn;
3004 #if FF_FS_RPATH != 0
3008 if (c !=
'.' || si >= 3)
break;
3019 if (c <=
' ')
break;
3020 if (c ==
'/' || c ==
'\\') {
3021 while (p[si] ==
'/' || p[si] ==
'\\') si++;
3024 if (c ==
'.' || i >= ni) {
3029 #if FF_CODE_PAGE == 0
3030 if (
ExCvt && c >= 0x80) {
3031 c =
ExCvt[c & 0x7F];
3033 #elif FF_CODE_PAGE < 900
3035 c =
ExCvt[c & 0x7F];
3076 #if FF_FS_RPATH != 0
3077 if (*path !=
'/' && *path !=
'\\') {
3082 while (*path ==
'/' || *path ==
'\\') path++;
3087 #if FF_FS_RPATH != 0
3091 dp->
obj.c_scl = fs->cdc_scl;
3092 dp->
obj.c_size = fs->cdc_size;
3093 dp->
obj.c_ofs = fs->cdc_ofs;
3094 res = load_obj_xdir(&dj, &dp->
obj);
3095 if (res !=
FR_OK)
return res;
3102 if ((
UINT)*path <
' ') {
3109 if (res !=
FR_OK)
break;
3115 if (!(ns &
NS_LAST))
continue;
3134 init_alloc_info(fs, &dp->
obj);
3157 const TCHAR *tp, *tt;
3160 #if FF_STR_VOLUME_ID
3166 if (!tp)
return vol;
3167 do tc = *tt++;
while ((
UINT)tc >= (
FF_USE_LFN ?
' ' :
'!') && tc !=
':');
3171 if (
IsDigit(*tp) && tp + 2 == tt) {
3174 #if FF_STR_VOLUME_ID == 1
3178 sp = VolumeStr[i]; tp = *path;
3180 c = *sp++; tc = *tp++;
3183 }
while (c && (
TCHAR)c == tc);
3184 }
while ((c || tp != tt) && ++i <
FF_VOLUMES);
3193 #if FF_STR_VOLUME_ID == 2
3197 sp = VolumeStr[i]; tp = *path;
3199 c = *sp++; tc = *(++tp);
3202 }
while (c && (
TCHAR)c == tc);
3212 #if FF_FS_RPATH != 0
3231 static DWORD crc32 (
3239 for (b = 1; b; b <<= 1) {
3240 crc ^= (d & b) ? 1 : 0;
3241 crc = (crc & 1) ? crc >> 1 ^ 0xEDB88320 : crc >> 1;
3249 static int test_gpt_header (
3257 if (
mem_cmp(gpth +
GPTH_Sign,
"EFI PART" "\0\0\1\0" "\x5C\0\0", 16))
return 0;
3258 for (i = 0, bcc = 0xFFFFFFFF; i < 92; i++) {
3259 bcc = crc32(bcc, i -
GPTH_Bcc < 4 ? 0 : gpth[i]);
3268 #if !FF_FS_READONLY && FF_USE_MKFS
3271 static DWORD make_rand (
3280 if (seed == 0) seed = 1;
3282 for (r = 0; r < 8; r++) seed = seed & 1 ? seed >> 1 ^ 0xA3000000 : seed >> 1;
3283 *buff++ = (
BYTE)seed;
3332 if (fmt != 2 && (fmt >= 3 || part == 0))
return fmt;
3338 DWORD n_ent, v_ent, ofs;
3342 if (!test_gpt_header(fs->
win))
return 3;
3345 for (v_ent = i = 0; i < n_ent; i++) {
3351 if (part == 0 && fmt <= 1)
return fmt;
3352 if (part != 0 && v_ent == part)
return fmt;
3359 for (i = 0; i < 4; i++) {
3362 i = part ? part - 1 : 0;
3364 fmt = mbr_pt[i] ?
check_fs(fs, mbr_pt[i]) : 3;
3365 }
while (part == 0 && fmt >= 2 && ++i < 4);
3385 DWORD tsect, sysect, fasize, nclst, szbfat;
3427 #if FF_MAX_SS != FF_MIN_SS
3443 DWORD so, cv, bcl, i;
3493 if (cv == 0xFFFFFFFF)
break;
3529 nclst = (tsect - sysect) / fs->
csize;
3559 #if (FF_FS_NOFSINFO & 3) != 3
3569 #if (FF_FS_NOFSINFO & 1) == 0
3572 #if (FF_FS_NOFSINFO & 2) == 0
3586 fs->dirbuf = DirBuf;
3589 #if FF_FS_RPATH != 0
3615 if (lock_fs(obj->
fs)) {
3630 *rfs = (res ==
FR_OK) ? obj->
fs : 0;
3658 const TCHAR *rp = path;
3671 if (!ff_del_syncobj(cfs->sobj))
return FR_INT_ERR;
3684 if (opt == 0)
return FR_OK;
3707 DWORD cl, bcs, clst;
3730 res = chk_lock(&dj, (mode & ~
FA_READ) ? 1 : 0);
3758 init_alloc_info(fs, &fp->
obj);
3760 mem_set(fs->dirbuf + 2, 0, 30);
3761 mem_set(fs->dirbuf + 38, 0, 26);
3765 res = store_xdir(&dj);
3807 fp->
obj.lockid = inc_lock(&dj, (mode & ~
FA_READ) ? 1 : 0);
3829 init_alloc_info(fs, &fp->
obj);
3859 if (res ==
FR_OK && ofs %
SS(fs)) {
3901 UINT rcnt, cc, csect;
3910 if (btr > remain) btr = (
UINT)remain;
3913 btr -= rcnt, *br += rcnt, rbuff += rcnt, fp->
fptr += rcnt) {
3914 if (fp->
fptr %
SS(fs) == 0) {
3917 if (fp->
fptr == 0) {
3938 if (csect + cc > fs->
csize) {
3939 cc = fs->
csize - csect;
3942 #if !FF_FS_READONLY && FF_FS_MINIMIZE <= 2
3957 if (fp->
sect != sect) {
3970 if (rcnt > btr) rcnt = btr;
4001 UINT wcnt, cc, csect;
4002 const BYTE *wbuff = (
const BYTE*)buff;
4017 if (fp->
fptr %
SS(fs) == 0) {
4020 if (fp->
fptr == 0) {
4035 if (clst == 0)
break;
4054 if (csect + cc > fs->
csize) {
4055 cc = fs->
csize - csect;
4058 #if FF_FS_MINIMIZE <= 2
4060 if (fs->
winsect - sect < cc) {
4065 if (fp->
sect - sect < cc) {
4080 if (fp->
sect != sect &&
4089 if (wcnt > btw) wcnt = btw;
4135 res = fill_first_frag(&fp->
obj);
4137 res = fill_last_frag(&fp->
obj, fp->
clust, 0xFFFFFFFF);
4144 res = load_obj_xdir(&dj, &fp->
obj);
4154 res = store_xdir(&dj);
4208 res = dec_lock(fp->
obj.lockid);
4214 unlock_fs(fs,
FR_OK);
4224 #if FF_FS_RPATH >= 1
4250 #if FF_STR_VOLUME_ID == 2
4270 fs->cdc_scl = dj.
obj.c_scl;
4271 fs->cdc_size = dj.
obj.c_size;
4272 fs->cdc_ofs = dj.
obj.c_ofs;
4295 #if FF_STR_VOLUME_ID == 2
4307 #if FF_FS_RPATH >= 2
4321 #if FF_STR_VOLUME_ID
4342 if (res !=
FR_OK)
break;
4344 if (res !=
FR_OK)
break;
4347 if (res !=
FR_OK)
break;
4350 if (res !=
FR_OK)
break;
4353 }
while (res ==
FR_OK);
4355 if (res !=
FR_OK)
break;
4357 for (n = 0; fno.
fname[n]; n++) ;
4361 while (n) buff[--i] = fno.
fname[--n];
4366 if (i == len) buff[--i] =
'/';
4369 #if FF_STR_VOLUME_ID >= 1
4370 for (n = 0, vp = (
const char*)VolumeStr[
CurrVol]; vp[n]; n++) ;
4373 for (vl = 0; vl < n; *tp++ = (
TCHAR)vp[vl], vl++) ;
4388 do *tp++ = buff[i++];
while (i < len);
4403 #if FF_FS_MINIMIZE <= 2
4419 DWORD cl, pcl, ncl, tcl, tlen, ulen, *tbl;
4427 res = fill_last_frag(&fp->
obj, fp->
clust, 0xFFFFFFFF);
4436 tlen = *tbl++; ulen = 2;
4441 tcl = cl; ncl = 0; ulen += 2;
4447 }
while (cl == pcl + 1);
4449 *tbl++ = ncl; *tbl++ = tcl;
4451 }
while (cl < fs->n_fatent);
4493 fp->
fptr = nsect = 0;
4497 (ofs - 1) / bcs >= (ifptr - 1) / bcs) {
4515 ofs -= bcs; fp->
fptr += bcs;
4539 nsect += (
DWORD)(ofs /
SS(fs));
4547 if (fp->
fptr %
SS(fs) && nsect != fp->
sect) {
4566 #if FF_FS_MINIMIZE <= 1
4597 init_alloc_info(fs, &dp->
obj);
4613 dp->
obj.lockid = inc_lock(dp, 0);
4648 if (dp->
obj.lockid) res = dec_lock(dp->
obj.lockid);
4654 unlock_fs(fs,
FR_OK);
4713 if (res !=
FR_OK || !fno || !fno->
fname[0])
break;
4715 #if FF_USE_LFN && FF_USE_FIND == 2
4732 const TCHAR* pattern
4750 #if FF_FS_MINIMIZE == 0
4815 clst = 2; obj.
fs = fs;
4820 if (
stat == 0) nfree++;
4821 }
while (++clst < fs->n_fatent);
4834 if (res !=
FR_OK)
break;
4836 for (b = 8, bm = fs->
win[i]; b && clst; b--, clst--) {
4837 if (!(bm & 1)) nfree++;
4840 i = (i + 1) %
SS(fs);
4851 if (res !=
FR_OK)
break;
4857 if ((
ld_dword(fs->
win + i) & 0x0FFFFFFF) == 0) nfree++;
4894 if (fp->
fptr == 0) {
4902 if (res ==
FR_OK && ncl < fs->n_fatent) {
4954 if (res ==
FR_OK) res = chk_lock(&dj, 2);
4968 init_alloc_info(fs, &obj);
4976 #if FF_FS_RPATH != 0
4977 if (dclst == fs->
cdir) {
5001 if (res ==
FR_OK && dclst != 0) {
5079 res = store_xdir(&dj);
5109 const TCHAR* path_old,
5110 const TCHAR* path_new
5130 res = chk_lock(&djo, 2);
5140 mem_cpy(&djn, &djo,
sizeof djo);
5155 res = store_xdir(&djn);
5183 if (res ==
FR_OK && dir[1] ==
'.') {
5213 #if FF_USE_CHMOD && !FF_FS_READONLY
5241 res = store_xdir(&dj);
5286 res = store_xdir(&dj);
5328 if (res ==
FR_OK && label) {
5344 if (wc == 0) { di = 0;
break; }
5348 if (hs != 0) di = 0;
5356 #if FF_USE_LFN && FF_LFN_UNICODE >= 1
5359 if (wc != 0) wc =
put_utf(wc, &label[di], 4);
5360 if (wc == 0) { di = 0;
break; }
5363 label[di++] = (
TCHAR)wc;
5369 }
while (label[--di] ==
' ');
5380 if (res ==
FR_OK && vsn) {
5417 static const char badchr[] =
"+.,;=[]/\\\"*:<>\?|\x7F";
5430 while ((
UINT)*label >=
' ') {
5432 if (dc >= 0x10000) {
5433 if (dc == 0xFFFFFFFF || di >= 10) {
5439 if (dc == 0 ||
chk_chr(badchr + 7, (
int)dc) || di >= 11) {
5449 while ((
UINT)*label >=
' ') {
5454 wc = (
BYTE)*label++;
5457 #if FF_CODE_PAGE == 0
5458 if (
ExCvt && wc >= 0x80) wc =
ExCvt[wc - 0x80];
5459 #elif FF_CODE_PAGE < 900
5460 if (wc >= 0x80) wc =
ExCvt[wc - 0x80];
5463 if (wc == 0 ||
chk_chr(badchr + 0, (
int)wc) || di >= (
UINT)((wc >= 0x100) ? 10 : 11)) {
5466 if (wc >= 0x100) dirvn[di++] = (
BYTE)(wc >> 8);
5467 dirvn[di++] = (
BYTE)wc;
5470 while (di && dirvn[di - 1] ==
' ') di--;
5522 #if FF_USE_EXPAND && !FF_FS_READONLY
5535 DWORD n, clst, stcl, scl, ncl, tcl, lclst;
5545 tcl = (
DWORD)(fsz / n) + ((fsz & (n - 1)) ? 1 : 0);
5547 if (stcl < 2 || stcl >= fs->
n_fatent) stcl = 2;
5551 scl = find_bitmap(fs, stcl, tcl);
5556 res = change_bitmap(fs, scl, tcl, 1);
5557 lclst = scl + tcl - 1;
5565 scl = clst = stcl; ncl = 0;
5568 if (++clst >= fs->
n_fatent) clst = 2;
5570 if (n == 0xFFFFFFFF) { res =
FR_DISK_ERR;
break; }
5572 if (++ncl == tcl)
break;
5574 scl = clst; ncl = 0;
5576 if (clst == stcl) { res =
FR_DENIED;
break; }
5580 for (clst = scl, n = tcl; n; clst++, n--) {
5581 res =
put_fat(fs, clst, (n == 1) ? 0xFFFFFFFF : clst + 1);
5582 if (res !=
FR_OK)
break;
5639 if (btf > remain) btf = (
UINT)remain;
5641 for ( ; btf && (*func)(0, 0);
5642 fp->
fptr += rcnt, *bf += rcnt, btf -= rcnt) {
5644 if (fp->
fptr %
SS(fs) == 0) {
5646 clst = (fp->
fptr == 0) ?
5660 if (fp->
sect != sect) {
5673 if (rcnt > btf) rcnt = btf;
5674 rcnt = (*func)(dbuf + ((
UINT)fp->
fptr %
SS(fs)), rcnt);
5684 #if !FF_FS_READONLY && FF_USE_MKFS
5689 #define N_SEC_TRACK 63
5690 #define GPT_ALIGN 0x100000
5691 #define GPT_ITEMS 128
5705 DWORD sz_drv32, s_lba32, n_lba32;
5706 BYTE *pte, hd, n_hd, sc, n_sc;
5714 UINT sz_pt, pi, si, ofs;
5715 DWORD bcc, rnd, align;
5716 QWORD s_lba64, n_lba64, sz_pool, s_bpt;
5717 static const BYTE gpt_mbr[16] = {0x00, 0x00, 0x02, 0x00, 0xEE, 0xFE, 0xFF, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF};
5719 #if FF_MAX_SS != FF_MIN_SS
5728 s_bpt = sz_drv - sz_pt - 1;
5729 s_lba64 = 2 + sz_pt;
5730 sz_pool = s_bpt - s_lba64;
5731 bcc = 0xFFFFFFFF; n_lba64 = 1;
5736 s_lba64 = (s_lba64 + align - 1) & ((QWORD)0 - align);
5737 n_lba64 = plst[si++];
5738 if (n_lba64 <= 100) {
5739 n_lba64 = sz_pool * n_lba64 / 100;
5740 n_lba64 = (n_lba64 + align - 1) & ((QWORD)0 - align);
5742 if (s_lba64 + n_lba64 > s_bpt) {
5743 n_lba64 = (s_lba64 < s_bpt) ? s_bpt - s_lba64 : 0;
5749 rnd = make_rand(rnd, buf + ofs +
GPTE_UpGuid, 16);
5751 st_qword(buf + ofs +
GPTE_LstLba, s_lba64 + n_lba64 - 1);
5754 if ((pi + 1) *
SZ_GPTE % ss == 0) {
5755 for (i = 0; i < ss; bcc = crc32(bcc, buf[i++])) ;
5773 for (i = 0, bcc= 0xFFFFFFFF; i < 92; bcc = crc32(bcc, buf[i++])) ;
5782 for (i = 0, bcc= 0xFFFFFFFF; i < 92; bcc = crc32(bcc, buf[i++])) ;
5795 sz_drv32 = (
DWORD)sz_drv;
5797 for (n_hd = 8; n_hd != 0 && sz_drv32 / n_hd / n_sc > 1024; n_hd *= 2) ;
5798 if (n_hd == 0) n_hd = 255;
5802 for (i = 0, s_lba32 = n_sc; i < 4 && s_lba32 != 0 && s_lba32 < sz_drv32; i++, s_lba32 += n_lba32) {
5803 n_lba32 = (
DWORD)plst[i];
5804 if (n_lba32 <= 100) n_lba32 = (n_lba32 == 100) ? sz_drv32 : sz_drv32 / 100 * n_lba32;
5805 if (s_lba32 + n_lba32 > sz_drv32 || s_lba32 + n_lba32 < s_lba32) n_lba32 = sz_drv32 - s_lba32;
5806 if (n_lba32 == 0)
break;
5812 cy = (
UINT)(s_lba32 / n_sc / n_hd);
5813 hd = (
BYTE)(s_lba32 / n_sc % n_hd);
5814 sc = (
BYTE)(s_lba32 % n_sc + 1);
5819 cy = (
UINT)((s_lba32 + n_lba32 - 1) / n_sc / n_hd);
5820 hd = (
BYTE)((s_lba32 + n_lba32 - 1) / n_sc % n_hd);
5821 sc = (
BYTE)((s_lba32 + n_lba32 - 1) % n_sc + 1);
5845 static const WORD cst[] = {1, 4, 16, 64, 256, 512, 0};
5846 static const WORD cst32[] = {1, 2, 4, 8, 16, 32, 0};
5848 BYTE fsopt, fsty, sys, *buf, *pte, pdrv, ipart;
5850 DWORD sz_buf, sz_blk, n_clst, pau, nsect, n;
5851 LBA_t sz_vol, b_vol, b_fat, b_data;
5853 DWORD sz_rsv, sz_fat, sz_dir, sz_au;
5854 UINT n_fat, n_root, i;
5866 if (!opt) opt = &defopt;
5872 sz_blk = opt->
align;
5874 if (sz_blk == 0 || sz_blk > 0x8000 || (sz_blk & (sz_blk - 1))) sz_blk = 1;
5875 #if FF_MAX_SS != FF_MIN_SS
5918 sz_vol = ld_qword(buf + ofs +
GPTE_LstLba) - b_vol + 1;
5921 n_ent--; ofs = (ofs +
SZ_GPTE) % ss;
5956 if ((fsopt &
FM_ANY) ==
FM_EXFAT || sz_vol >= 0x4000000 || sz_au > 128) {
5963 if (sz_au > 128) sz_au = 128;
5975 DWORD szb_bit, szb_case,
sum, nb, cl, tbl[3];
5982 lba[0] = b_vol; lba[1] = b_vol + sz_vol - 1;
5988 if (sz_vol >= 0x80000) sz_au = 64;
5989 if (sz_vol >= 0x4000000) sz_au = 256;
5992 sz_fat = (
DWORD)((sz_vol / sz_au + 2) * 4 + ss - 1) / ss;
5993 b_data = (b_fat + sz_fat + sz_blk - 1) & ~((
LBA_t)sz_blk - 1);
5995 n_clst = (
DWORD)(sz_vol - (b_data - b_vol)) / sz_au;
5999 szb_bit = (n_clst + 7) / 8;
6000 tbl[0] = (szb_bit + sz_au * ss - 1) / (sz_au * ss);
6003 sect = b_data + sz_au * tbl[0];
6005 st = 0; si = 0; i = 0; j = 0; szb_case = 0;
6015 ch = 0xFFFF; st = 2;
break;
6021 if (--j == 0) st = 0;
6029 sum = xsum32(buf[i + 1] = (
BYTE)(ch >> 8),
sum);
6030 i += 2; szb_case += 2;
6031 if (si == 0 || i == sz_buf * ss) {
6032 n = (i + ss - 1) / ss;
6037 tbl[1] = (szb_case + sz_au * ss - 1) / (sz_au * ss);
6041 sect = b_data; nsect = (szb_bit + ss - 1) / ss;
6042 nb = tbl[0] + tbl[1] + tbl[2];
6045 for (i = 0; nb >= 8 && i < sz_buf * ss; buf[i++] = 0xFF, nb -= 8) ;
6046 for (b = 1; nb != 0 && i < sz_buf * ss; buf[i] |= b, b <<= 1, nb--) ;
6047 n = (nsect > sz_buf) ? sz_buf : nsect;
6049 sect += n; nsect -= n;
6053 sect = b_fat; nsect = sz_fat;
6056 mem_set(buf, 0, sz_buf * ss); i = 0;
6058 st_dword(buf + i, 0xFFFFFFF8); i += 4; cl++;
6059 st_dword(buf + i, 0xFFFFFFFF); i += 4; cl++;
6062 while (nb != 0 && i < sz_buf * ss) {
6063 st_dword(buf + i, (nb > 1) ? cl + 1 : 0xFFFFFFFF);
6066 if (nb == 0 && j < 3) nb = tbl[j++];
6067 }
while (nb != 0 && i < sz_buf * ss);
6068 n = (nsect > sz_buf) ? sz_buf : nsect;
6070 sect += n; nsect -= n;
6083 sect = b_data + sz_au * (tbl[0] + tbl[1]); nsect = sz_au;
6085 n = (nsect > sz_buf) ? sz_buf : nsect;
6088 sect += n; nsect -= n;
6093 for (n = 0; n < 2; n++) {
6112 for (i =
sum = 0; i < ss; i++) {
6118 st_word(buf + ss - 2, 0xAA55);
6119 for (j = 1; j < 9; j++) {
6120 for (i = 0; i < ss;
sum = xsum32(buf[i++],
sum)) ;
6125 for ( ; j < 11; j++) {
6126 for (i = 0; i < ss;
sum = xsum32(buf[i++],
sum)) ;
6130 for (i = 0; i < ss; i += 4)
st_dword(buf + i,
sum);
6142 n = (
DWORD)sz_vol / 0x20000;
6143 for (i = 0, pau = 1; cst32[i] && cst32[i] <= n; i++, pau <<= 1) ;
6145 n_clst = (
DWORD)sz_vol / pau;
6146 sz_fat = (n_clst * 4 + 8 + ss - 1) / ss;
6152 n = (
DWORD)sz_vol / 0x1000;
6153 for (i = 0, pau = 1; cst[i] && cst[i] <= n; i++, pau <<= 1) ;
6155 n_clst = (
DWORD)sz_vol / pau;
6160 n = (n_clst * 3 + 1) / 2 + 3;
6162 sz_fat = (n + ss - 1) / ss;
6166 b_fat = b_vol + sz_rsv;
6167 b_data = b_fat + sz_fat * n_fat + sz_dir;
6170 n = (
DWORD)(((b_data + sz_blk - 1) & ~(sz_blk - 1)) - b_data);
6172 sz_rsv += n; b_fat += n;
6175 n--; sz_rsv++; b_fat++;
6177 sz_fat += n / n_fat;
6182 n_clst = ((
DWORD)sz_vol - sz_rsv - sz_fat * n_fat - sz_dir) / pau;
6185 if (sz_au == 0 && (sz_au = pau / 2) != 0)
continue;
6191 if (sz_au == 0 && (pau * 2) <= 64) {
6192 sz_au = pau * 2;
continue;
6197 if (sz_au == 0 && (sz_au = pau * 2) <= 128)
continue;
6201 if (sz_au == 0 && (sz_au = pau * 2) <= 128)
continue;
6212 lba[0] = b_vol; lba[1] = b_vol + sz_vol - 1;
6223 if (sz_vol < 0x10000) {
6267 for (i = 0; i < n_fat; i++) {
6277 n = (nsect > sz_buf) ? sz_buf : nsect;
6280 sect += n; nsect -= n;
6285 nsect = (fsty ==
FS_FAT32) ? pau : sz_dir;
6287 n = (nsect > sz_buf) ? sz_buf : nsect;
6289 sect += n; nsect -= n;
6302 if (sz_vol >= 0x10000) {
6305 sys = (fsty ==
FS_FAT16) ? 0x04 : 0x01;
6312 if (!
FF_LBA64 || !(fsopt & 0x80)) {
6320 lba[0] = sz_vol, lba[1] = 0;
6334 #if FF_MULTI_PARTITION
6367 #if FF_USE_LFN && FF_LFN_UNICODE && (FF_STRF_ENCODE < 0 || FF_STRF_ENCODE > 3)
6368 #error Wrong FF_STRF_ENCODE setting
6385 #if FF_USE_LFN && FF_LFN_UNICODE && FF_STRF_ENCODE <= 2
6388 #if FF_USE_LFN && FF_LFN_UNICODE && FF_STRF_ENCODE == 3
6392 #if FF_USE_LFN && FF_LFN_UNICODE
6398 #if FF_STRF_ENCODE == 0
6404 if (rc != 1 || !
dbc_2nd(s[0]))
continue;
6405 wc = wc << 8 | s[0];
6408 if (dc == 0)
continue;
6409 #elif FF_STRF_ENCODE == 1 || FF_STRF_ENCODE == 2
6419 dc = ((dc & 0x3FF) + 0x40) << 10 | (wc & 0x3FF);
6427 if ((dc & 0xE0) == 0xC0) { dc &= 0x1F; ct = 1; }
6428 if ((dc & 0xF0) == 0xE0) { dc &= 0x0F; ct = 2; }
6429 if ((dc & 0xF8) == 0xF0) { dc &= 0x07; ct = 3; }
6430 if (ct == 0)
continue;
6432 if (rc != ct)
break;
6435 if ((s[rc] & 0xC0) != 0x80)
break;
6436 dc = dc << 6 | (s[rc] & 0x3F);
6437 }
while (++rc < ct);
6438 if (rc != ct || dc < 0x80 ||
IsSurrogate(dc) || dc >= 0x110000)
continue;
6444 #if FF_LFN_UNICODE == 1 || FF_LFN_UNICODE == 3
6446 *p++ = (
TCHAR)(0xD800 | ((dc >> 10) - 0x40)); nc++;
6447 dc = 0xDC00 | (dc & 0x3FF);
6449 *p++ = (
TCHAR)dc; nc++;
6450 if (dc ==
'\n')
break;
6451 #elif FF_LFN_UNICODE == 2
6455 if (dc ==
'\n')
break;
6458 *p++ = (
TCHAR)(0xC0 | (dc >> 6 & 0x1F));
6459 *p++ = (
TCHAR)(0x80 | (dc >> 0 & 0x3F));
6463 *p++ = (
TCHAR)(0xE0 | (dc >> 12 & 0x0F));
6464 *p++ = (
TCHAR)(0x80 | (dc >> 6 & 0x3F));
6465 *p++ = (
TCHAR)(0x80 | (dc >> 0 & 0x3F));
6468 *p++ = (
TCHAR)(0xF0 | (dc >> 18 & 0x07));
6469 *p++ = (
TCHAR)(0x80 | (dc >> 12 & 0x3F));
6470 *p++ = (
TCHAR)(0x80 | (dc >> 6 & 0x3F));
6471 *p++ = (
TCHAR)(0x80 | (dc >> 0 & 0x3F));
6486 *p++ = (
TCHAR)dc; nc++;
6487 if (dc ==
'\n')
break;
6492 return nc ? buff : 0;
6509 #if FF_USE_LFN && FF_LFN_UNICODE == 1
6511 #elif FF_USE_LFN && FF_LFN_UNICODE == 2
6521 static void putc_bfd (putbuff* pb,
TCHAR c)
6525 #if FF_USE_LFN && FF_LFN_UNICODE
6527 #if FF_LFN_UNICODE == 2
6541 #if FF_USE_LFN && FF_LFN_UNICODE
6542 #if FF_LFN_UNICODE == 1
6546 hs = pb->hs; pb->hs = 0;
6553 #elif FF_LFN_UNICODE == 2
6556 pb->bs[pb->wi = 0] = (
BYTE)c;
6557 if ((
BYTE)c < 0x80)
break;
6558 if (((
BYTE)c & 0xE0) == 0xC0) pb->ct = 1;
6559 if (((
BYTE)c & 0xF0) == 0xE0) pb->ct = 2;
6560 if (((
BYTE)c & 0xF1) == 0xF0) pb->ct = 3;
6563 if (((
BYTE)c & 0xC0) != 0x80) {
6564 pb->ct = 0;
continue;
6566 pb->bs[++pb->wi] = (
BYTE)c;
6567 if (--pb->ct == 0)
break;
6571 tp = (
TCHAR*)pb->bs;
6573 if (dc == 0xFFFFFFFF)
return;
6575 hs = (
WCHAR)(dc >> 16);
6576 #elif FF_LFN_UNICODE == 3
6579 hs = (
WCHAR)(0xD800 | ((c >> 10) - 0x40));
6580 wc = 0xDC00 | (c & 0x3FF);
6588 #if FF_STRF_ENCODE == 1
6596 #elif FF_STRF_ENCODE == 2
6598 pb->buf[i++] = (
BYTE)(hs >> 8);
6599 pb->buf[i++] = (
BYTE)hs;
6602 pb->buf[i++] = (
BYTE)(wc >> 8);
6603 pb->buf[i++] = (
BYTE)wc;
6604 #elif FF_STRF_ENCODE == 3
6607 hs = (hs & 0x3FF) + 0x40;
6608 pb->buf[i++] = (
BYTE)(0xF0 | hs >> 8);
6609 pb->buf[i++] = (
BYTE)(0x80 | (hs >> 2 & 0x3F));
6610 pb->buf[i++] = (
BYTE)(0x80 | (hs & 3) << 4 | (wc >> 6 & 0x0F));
6611 pb->buf[i++] = (
BYTE)(0x80 | (wc & 0x3F));
6614 pb->buf[i++] = (
BYTE)wc;
6618 pb->buf[i++] = (
BYTE)(0xC0 | wc >> 6);
6621 pb->buf[i++] = (
BYTE)(0xE0 | wc >> 12);
6622 pb->buf[i++] = (
BYTE)(0x80 | (wc >> 6 & 0x3F));
6624 pb->buf[i++] = (
BYTE)(0x80 | (wc & 0x3F));
6628 if (hs != 0)
return;
6630 if (wc == 0)
return;
6632 pb->buf[i++] = (
BYTE)(wc >> 8); nc++;
6634 pb->buf[i++] = (
BYTE)wc;
6638 pb->buf[i++] = (
BYTE)c;
6641 if (i >= (
int)(
sizeof pb->buf) - 4) {
6643 i = (n == (
UINT)i) ? 0 : -1;
6652 static int putc_flush (putbuff* pb)
6658 && (
UINT)pb->idx == nw)
return pb->nchr;
6665 static void putc_init (putbuff* pb,
FIL* fp)
6667 mem_set(pb, 0,
sizeof (putbuff));
6683 return putc_flush(&pb);
6702 while (*str) putc_bfd(&pb, *str++);
6703 return putc_flush(&pb);
6724 TCHAR c, d, str[32], *p;
6748 w = va_arg(arp,
int);
6752 w = w * 10 + c -
'0';
6756 if (c ==
'l' || c ==
'L') {
6764 p = va_arg(arp,
TCHAR*);
6765 for (j = 0; p[j]; j++) ;
6767 while (j++ < w) putc_bfd(&pb,
' ') ;
6769 while (*p) putc_bfd(&pb, *p++) ;
6770 while (j++ < w) putc_bfd(&pb,
' ') ;
6774 putc_bfd(&pb, (
TCHAR)va_arg(arp,
int));
continue;
6790 putc_bfd(&pb, c);
continue;
6794 v = (f & 4) ? (
DWORD)va_arg(arp,
long) : ((d ==
'D') ? (
DWORD)(long)va_arg(arp, int) : (
DWORD)va_arg(arp, unsigned int));
6795 if (d ==
'D' && (v & 0x80000000)) {
6801 d = (
TCHAR)(v % r); v /= r;
6802 if (d > 9) d += (c ==
'x') ? 0x27 : 0x07;
6804 }
while (v && i <
sizeof str /
sizeof *str);
6805 if (f & 8) str[i++] =
'-';
6806 j = i; d = (f & 1) ?
'0' :
' ';
6808 while (j++ < w) putc_bfd(&pb, d);
6811 putc_bfd(&pb, str[--i]);
6813 while (j++ < w) putc_bfd(&pb, d);
6818 return putc_flush(&pb);
6826 #if FF_CODE_PAGE == 0
6835 static const WORD validcp[] = { 437, 720, 737, 771, 775, 850, 852, 857, 860, 861, 862, 863, 864, 865, 866, 869, 932, 936, 949, 950, 0};
6836 static const BYTE*
const tables[] = {Ct437, Ct720, Ct737, Ct771, Ct775, Ct850, Ct852, Ct857, Ct860, Ct861, Ct862, Ct863, Ct864, Ct865, Ct866, Ct869, Dc932, Dc936, Dc949, Dc950, 0};
6840 for (i = 0; validcp[i] != 0 && validcp[i] != cp; i++) ;