2 * fat-handler - FAT12/16/32 filesystem handler
4 * Copyright © 2006 Marek Szyprowski
5 * Copyright © 2007-2015 The AROS Development Team
7 * This program is free software; you can redistribute it and/or modify it
8 * under the same terms as AROS itself.
16 #define DEBUG_DIRENTRY 0
21 #define DEBUG_NOTIFY 0
23 #define DEBUG_PACKETS 0
24 #define DEBUG_CACHESTATS 0
28 #include <exec/interrupts.h>
30 #include "fat_struct.h"
34 /* Filesystem structures */
36 #define ID_FAT_DISK 0x46415400UL
38 #define ID_FAT12_DISK 0x46415400UL
39 #define ID_FAT16_DISK 0x46415401UL
40 #define ID_FAT32_DISK 0x46415402UL
42 #define ACTION_VOLUME_ADD 16000
43 #define ACTION_VOLUME_REMOVE 16001
45 #define DEF_POOL_SIZE 65536
46 #define DEF_POOL_THRESHOLD DEF_POOL_SIZE
49 /* A handle on something, file or directory */
52 struct FSSuper
*sb
; /* filesystem data */
54 ULONG first_cluster
; /* first cluster of this file */
55 ULONG cur_cluster
; /* cluster that the current sector is within */
57 ULONG cluster_offset
; /* cluster number of this cluster within the current file */
59 ULONG first_sector
; /* first sector in the first cluster, for fat12/16 root dir */
60 ULONG cur_sector
; /* sector number our block is currently in */
62 ULONG sector_offset
; /* current sector as an offset in the current cluster
63 i.e. cur = sector(cur_cluster) + offset */
65 APTR block
; /* current block from the cache */
66 UBYTE
*data
; /* current data buffer (from cache) */
69 /* A handle on a directory */
74 ULONG cur_index
; /* last entry returned, for GetNextDirEntry */
77 /* Single directory entry */
80 struct FSSuper
*sb
; /* filesystem data */
82 ULONG cluster
; /* cluster the containing directory starts at */
83 ULONG index
; /* index of this entry */
85 ULONG pos
; /* byte offset within directory that the entry came from */
88 struct FATDirEntry entry
;
89 struct FATLongDirEntry long_entry
;
93 #define FAT_ROOTDIR_MARK 0xFFFFFFFFlu
103 struct MsgPort
* fl_Task
;
106 ULONG magic
; /* we set this to ID_FAT_DISK so we can tell
107 our locks from those of other filesystems */
109 struct MinNode node
; /* node in the list of locks attached to the global lock */
110 struct GlobalLock
*gl
; /* pointer to the global lock for this file */
112 struct IOHandle ioh
; /* handle for reads and writes */
113 ULONG pos
; /* current seek position within the file */
115 BOOL do_notify
; /* if set, send notification on file close (ACTION_END) */
116 struct FSSuper
*sb
; /* pointer to sb, for unlocking when volume is removed */
123 ULONG dir_cluster
; /* first cluster of the directory we're in */
124 ULONG dir_entry
; /* this is our dir entry within dir_cluster */
126 LONG access
; /* access mode, shared or exclusive */
128 ULONG first_cluster
; /* first data cluster */
130 ULONG attr
; /* file attributes, from the dir entry */
131 ULONG size
; /* file size, from the dir entry */
133 UBYTE name
[FAT_MAX_LONG_FILENAME
]; /* copy of the name (bstr) */
135 UBYTE shortname
[FAT_MAX_SHORT_NAME
+ 2]; /* copy of the short name (bstr) */
138 struct MinList locks
; /* list of ExtFileLocks opened on this file */
141 /* A node in the list of notification requests */
146 struct GlobalLock
*gl
; /* pointer to global lock if this file is
147 locked. if it's not, this is NULL */
149 struct NotifyRequest
*nr
; /* the request that DOS passed us */
156 struct MinList locks
; /* global locks */
157 struct GlobalLock root_lock
;
158 struct MinList notifies
;
161 struct VolumeIdentity
163 UBYTE name
[FAT_MAX_SHORT_NAME
+ 2]; /* BCPL string */
164 struct DateStamp create_time
;
171 struct Globals
*glob
;
172 struct DosList
*doslist
;
174 struct VolumeInfo
*info
;
177 ULONG first_device_sector
;
180 ULONG sectorsize_bits
;
182 ULONG cluster_sectors
;
184 ULONG clustersize_bits
;
185 ULONG cluster_sectors_bits
;
187 ULONG first_fat_sector
;
188 ULONG first_data_sector
;
189 ULONG first_rootdir_sector
;
191 ULONG rootdir_sectors
;
194 ULONG clusters_count
;
207 ULONG fat_blocks_count
;
210 ULONG fat_cachesize_bits
;
211 ULONG fat_cache_block
;
212 UWORD fat_cache_no
; /* FAT number that cached FAT blocks belong to */
215 struct FATFSInfo
*fsinfo_buffer
;
217 ULONG rootdir_cluster
;
218 ULONG rootdir_sector
;
220 struct VolumeIdentity volume
;
223 ULONG (*func_get_fat_entry
)(struct FSSuper
*sb
, ULONG n
);
224 BOOL (*func_set_fat_entry
)(struct FSSuper
*sb
, ULONG n
, ULONG val
);
231 struct ExecBase
*gl_SysBase
;
232 struct DosLibrary
*gl_DOSBase
;
233 struct Library
*gl_UtilityBase
;
234 struct Device
*gl_TimerBase
;
237 struct Task
*ourtask
;
238 struct MsgPort
*ourport
;
241 struct MsgPort
*notifyport
;
244 struct DosList
*devnode
;
245 struct FileSysStartupMsg
*fssm
;
247 struct DosPacket
*death_packet
;
251 struct IOExtTD
*diskioreq
;
252 struct IOExtTD
*diskchgreq
;
253 struct MsgPort
*diskport
;
254 ULONG diskchgsig_bit
;
255 struct timerequest
*timereq
;
256 struct MsgPort
*timerport
;
257 ULONG last_num
; /* last block number that was outside boundaries */
264 struct FSSuper
*sb
; /* current sb */
265 struct MinList sblist
; /* sbs with outstanding locks or notifies */
272 /* Character sets translation */
273 UBYTE from_unicode
[65536];
274 UWORD to_unicode
[256];
276 /* Disk change interrupt */
278 struct Interrupt Interrupt
;
279 struct ExecBase
*SysBase
;
285 #define DOSBase (glob->gl_DOSBase)
286 #define SysBase (glob->gl_SysBase)
287 #define UtilityBase (glob->gl_UtilityBase)
288 #define TimerBase (glob->gl_TimerBase)
292 /* Get the first sector of a cluster */
293 #define SECTOR_FROM_CLUSTER(sb,cl) \
294 ((ULONG) (((cl-2) << sb->cluster_sectors_bits) + sb->first_data_sector))
296 #define FIRST_FILE_CLUSTER(de) \
297 (AROS_LE2WORD((de)->e.entry.first_cluster_lo) | \
298 (((ULONG) AROS_LE2WORD((de)->e.entry.first_cluster_hi)) << 16))
300 #define RESET_HANDLE(ioh) \
303 (ioh)->cluster_offset = (ioh)->sector_offset = 0xffffffff; \
304 if ((ioh)->block != NULL) \
306 Cache_FreeBlock((ioh)->sb->cache, (ioh)->block); \
307 (ioh)->block = NULL; \
312 #define RESET_DIRHANDLE(dh) \
315 RESET_HANDLE(&((dh)->ioh)); \
316 (dh)->cur_index = 0xffffffff; \
320 #define GET_NEXT_CLUSTER(sb,cl) (sb->func_get_fat_entry(sb,cl))
321 #define SET_NEXT_CLUSTER(sb,cl,val) (sb->func_set_fat_entry(sb,cl,val))
323 #define CALC_SHORT_NAME_CHECKSUM(name,checksum) \
328 for (i = 0; i < FAT_MAX_SHORT_NAME; i++) \
330 ((checksum & 1) ? 0x80 : 0) + (checksum >> 1) + name[i]; \
334 #define LOCKFROMNODE(A) \
335 ((struct ExtFileLock *) \
336 (((BYTE *)(A)) - (IPTR)&((struct ExtFileLock *)NULL)->node))