| 1 | /* | 
|---|
| 2 | * Copyright (c) 2008 Jakub Jermar | 
|---|
| 3 | * Copyright (c) 2011 Oleg Romanenko | 
|---|
| 4 | * All rights reserved. | 
|---|
| 5 | * | 
|---|
| 6 | * Redistribution and use in source and binary forms, with or without | 
|---|
| 7 | * modification, are permitted provided that the following conditions | 
|---|
| 8 | * are met: | 
|---|
| 9 | * | 
|---|
| 10 | * - Redistributions of source code must retain the above copyright | 
|---|
| 11 | *   notice, this list of conditions and the following disclaimer. | 
|---|
| 12 | * - Redistributions in binary form must reproduce the above copyright | 
|---|
| 13 | *   notice, this list of conditions and the following disclaimer in the | 
|---|
| 14 | *   documentation and/or other materials provided with the distribution. | 
|---|
| 15 | * - The name of the author may not be used to endorse or promote products | 
|---|
| 16 | *   derived from this software without specific prior written permission. | 
|---|
| 17 | * | 
|---|
| 18 | * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR | 
|---|
| 19 | * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | 
|---|
| 20 | * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. | 
|---|
| 21 | * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, | 
|---|
| 22 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | 
|---|
| 23 | * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
|---|
| 24 | * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
|---|
| 25 | * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|---|
| 26 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 
|---|
| 27 | * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|---|
| 28 | */ | 
|---|
| 29 |  | 
|---|
| 30 | /** @addtogroup fs | 
|---|
| 31 | * @{ | 
|---|
| 32 | */ | 
|---|
| 33 |  | 
|---|
| 34 | #ifndef FAT_FAT_H_ | 
|---|
| 35 | #define FAT_FAT_H_ | 
|---|
| 36 |  | 
|---|
| 37 | #include "fat_fat.h" | 
|---|
| 38 | #include <fibril_synch.h> | 
|---|
| 39 | #include <libfs.h> | 
|---|
| 40 | #include <atomic.h> | 
|---|
| 41 | #include <sys/types.h> | 
|---|
| 42 | #include <bool.h> | 
|---|
| 43 | #include "../../vfs/vfs.h" | 
|---|
| 44 |  | 
|---|
| 45 | #ifndef dprintf | 
|---|
| 46 | #define dprintf(...)    printf(__VA_ARGS__) | 
|---|
| 47 | #endif | 
|---|
| 48 |  | 
|---|
| 49 | #define min(a, b)       ((a) < (b) ? (a) : (b)) | 
|---|
| 50 |  | 
|---|
| 51 | /* | 
|---|
| 52 | * Convenience macros for accessing some frequently used boot sector members. | 
|---|
| 53 | */ | 
|---|
| 54 | #define BPS(bs)         uint16_t_le2host((bs)->bps) | 
|---|
| 55 | #define SPC(bs)         (bs)->spc | 
|---|
| 56 | #define RSCNT(bs)       uint16_t_le2host((bs)->rscnt) | 
|---|
| 57 | #define FATCNT(bs)      (bs)->fatcnt | 
|---|
| 58 |  | 
|---|
| 59 | #define SF(bs)          (uint16_t_le2host((bs)->sec_per_fat) ? \ | 
|---|
| 60 | uint16_t_le2host((bs)->sec_per_fat) : \ | 
|---|
| 61 | uint32_t_le2host(bs->fat32.sectors_per_fat)) | 
|---|
| 62 |  | 
|---|
| 63 | #define RDE(bs)         uint16_t_le2host((bs)->root_ent_max) | 
|---|
| 64 |  | 
|---|
| 65 | #define TS(bs)          (uint16_t_le2host((bs)->totsec16) ? \ | 
|---|
| 66 | uint16_t_le2host((bs)->totsec16) : \ | 
|---|
| 67 | uint32_t_le2host(bs->totsec32)) | 
|---|
| 68 |  | 
|---|
| 69 | #define BS_BLOCK        0 | 
|---|
| 70 | #define BS_SIZE         512 | 
|---|
| 71 |  | 
|---|
| 72 | typedef struct fat_bs { | 
|---|
| 73 | uint8_t         ji[3];          /**< Jump instruction. */ | 
|---|
| 74 | uint8_t         oem_name[8]; | 
|---|
| 75 | /* BIOS Parameter Block */ | 
|---|
| 76 | uint16_t        bps;            /**< Bytes per sector. */ | 
|---|
| 77 | uint8_t         spc;            /**< Sectors per cluster. */ | 
|---|
| 78 | uint16_t        rscnt;          /**< Reserved sector count. */ | 
|---|
| 79 | uint8_t         fatcnt;         /**< Number of FATs. */ | 
|---|
| 80 | uint16_t        root_ent_max;   /**< Maximum number of root directory | 
|---|
| 81 | entries. */ | 
|---|
| 82 | uint16_t        totsec16;       /**< Total sectors. 16-bit version. */ | 
|---|
| 83 | uint8_t         mdesc;          /**< Media descriptor. */ | 
|---|
| 84 | uint16_t        sec_per_fat;    /**< Sectors per FAT12/FAT16. */ | 
|---|
| 85 | uint16_t        sec_per_track;  /**< Sectors per track. */ | 
|---|
| 86 | uint16_t        headcnt;        /**< Number of heads. */ | 
|---|
| 87 | uint32_t        hidden_sec;     /**< Hidden sectors. */ | 
|---|
| 88 | uint32_t        totsec32;       /**< Total sectors. 32-bit version. */ | 
|---|
| 89 |  | 
|---|
| 90 | union { | 
|---|
| 91 | struct { | 
|---|
| 92 | /* FAT12/FAT16 only: Extended BIOS Parameter Block */ | 
|---|
| 93 | /** Physical drive number. */ | 
|---|
| 94 | uint8_t         pdn; | 
|---|
| 95 | uint8_t         reserved; | 
|---|
| 96 | /** Extended boot signature. */ | 
|---|
| 97 | uint8_t         ebs; | 
|---|
| 98 | /** Serial number. */ | 
|---|
| 99 | uint32_t        id; | 
|---|
| 100 | /** Volume label. */ | 
|---|
| 101 | uint8_t         label[11]; | 
|---|
| 102 | /** FAT type. */ | 
|---|
| 103 | uint8_t         type[8]; | 
|---|
| 104 | /** Boot code. */ | 
|---|
| 105 | uint8_t         boot_code[448]; | 
|---|
| 106 | /** Boot sector signature. */ | 
|---|
| 107 | uint16_t        signature; | 
|---|
| 108 | } __attribute__ ((packed)); | 
|---|
| 109 | struct { | 
|---|
| 110 | /* FAT32 only */ | 
|---|
| 111 | /** Sectors per FAT. */ | 
|---|
| 112 | uint32_t        sectors_per_fat; | 
|---|
| 113 | /** FAT flags. */ | 
|---|
| 114 | uint16_t        flags; | 
|---|
| 115 | /** Version. */ | 
|---|
| 116 | uint16_t        version; | 
|---|
| 117 | /** Cluster number of root directory. */ | 
|---|
| 118 | uint32_t        root_cluster; | 
|---|
| 119 | /** Sector number of file system information sector. */ | 
|---|
| 120 | uint16_t        fsinfo_sec; | 
|---|
| 121 | /** Sector number of boot sector copy. */ | 
|---|
| 122 | uint16_t        bscopy_sec; | 
|---|
| 123 | uint8_t         reserved1[12]; | 
|---|
| 124 | /** Physical drive number. */ | 
|---|
| 125 | uint8_t         pdn; | 
|---|
| 126 | uint8_t         reserved2; | 
|---|
| 127 | /** Extended boot signature. */ | 
|---|
| 128 | uint8_t         ebs; | 
|---|
| 129 | /** Serial number. */ | 
|---|
| 130 | uint32_t        id; | 
|---|
| 131 | /** Volume label. */ | 
|---|
| 132 | uint8_t         label[11]; | 
|---|
| 133 | /** FAT type. */ | 
|---|
| 134 | uint8_t         type[8]; | 
|---|
| 135 | /** Boot code. */ | 
|---|
| 136 | uint8_t         boot_code[420]; | 
|---|
| 137 | /** Signature. */ | 
|---|
| 138 | uint16_t        signature; | 
|---|
| 139 | } fat32 __attribute__ ((packed)); | 
|---|
| 140 | }; | 
|---|
| 141 | } __attribute__ ((packed)) fat_bs_t; | 
|---|
| 142 |  | 
|---|
| 143 | typedef enum { | 
|---|
| 144 | FAT_INVALID, | 
|---|
| 145 | FAT_DIRECTORY, | 
|---|
| 146 | FAT_FILE | 
|---|
| 147 | } fat_node_type_t; | 
|---|
| 148 |  | 
|---|
| 149 | struct fat_node; | 
|---|
| 150 |  | 
|---|
| 151 | /** FAT index structure. | 
|---|
| 152 | * | 
|---|
| 153 | * This structure exists to help us to overcome certain limitations of the FAT | 
|---|
| 154 | * file system design.  The problem with FAT is that it is hard to find | 
|---|
| 155 | * an entity which could represent a VFS index.  There are two candidates: | 
|---|
| 156 | * | 
|---|
| 157 | * a) number of the node's first cluster | 
|---|
| 158 | * b) the pair of the parent directory's first cluster and the dentry index | 
|---|
| 159 | *    within the parent directory | 
|---|
| 160 | * | 
|---|
| 161 | * We need VFS indices to be: | 
|---|
| 162 | * A) unique | 
|---|
| 163 | * B) stable in time, at least until the next mount | 
|---|
| 164 | * | 
|---|
| 165 | * Unfortunately a) does not meet the A) criterion because zero-length files | 
|---|
| 166 | * will have the first cluster field cleared.  And b) does not meet the B) | 
|---|
| 167 | * criterion because unlink() and rename() will both free up the original | 
|---|
| 168 | * dentry, which contains all the essential info about the file. | 
|---|
| 169 | * | 
|---|
| 170 | * Therefore, a completely opaque indices are used and the FAT server maintains | 
|---|
| 171 | * a mapping between them and otherwise nice b) variant.  On rename(), the VFS | 
|---|
| 172 | * index stays unaltered, while the internal FAT "physical tree address" | 
|---|
| 173 | * changes.  The unlink case is also handled this way thanks to an in-core node | 
|---|
| 174 | * pointer embedded in the index structure. | 
|---|
| 175 | */ | 
|---|
| 176 | typedef struct { | 
|---|
| 177 | /** Used indices (position) hash table link. */ | 
|---|
| 178 | link_t          uph_link; | 
|---|
| 179 | /** Used indices (index) hash table link. */ | 
|---|
| 180 | link_t          uih_link; | 
|---|
| 181 |  | 
|---|
| 182 | fibril_mutex_t  lock; | 
|---|
| 183 | service_id_t    service_id; | 
|---|
| 184 | fs_index_t      index; | 
|---|
| 185 | /** | 
|---|
| 186 | * Parent node's first cluster. | 
|---|
| 187 | * Zero is used if this node is not linked, in which case nodep must | 
|---|
| 188 | * contain a pointer to the in-core node structure. | 
|---|
| 189 | * One is used when the parent is the root directory. | 
|---|
| 190 | */ | 
|---|
| 191 | fat_cluster_t   pfc; | 
|---|
| 192 | /** Directory entry index within the parent node. */ | 
|---|
| 193 | unsigned        pdi; | 
|---|
| 194 | /** Pointer to in-core node instance. */ | 
|---|
| 195 | struct fat_node *nodep; | 
|---|
| 196 | } fat_idx_t; | 
|---|
| 197 |  | 
|---|
| 198 | /** FAT in-core node. */ | 
|---|
| 199 | typedef struct fat_node { | 
|---|
| 200 | /** Back pointer to the FS node. */ | 
|---|
| 201 | fs_node_t               *bp; | 
|---|
| 202 |  | 
|---|
| 203 | fibril_mutex_t          lock; | 
|---|
| 204 | fat_node_type_t         type; | 
|---|
| 205 | fat_idx_t               *idx; | 
|---|
| 206 | /** | 
|---|
| 207 | *  Node's first cluster. | 
|---|
| 208 | *  Zero is used for zero-length nodes. | 
|---|
| 209 | *  One is used to mark root directory. | 
|---|
| 210 | */ | 
|---|
| 211 | fat_cluster_t           firstc; | 
|---|
| 212 | /** FAT in-core node free list link. */ | 
|---|
| 213 | link_t                  ffn_link; | 
|---|
| 214 | aoff64_t                size; | 
|---|
| 215 | unsigned                lnkcnt; | 
|---|
| 216 | unsigned                refcnt; | 
|---|
| 217 | bool                    dirty; | 
|---|
| 218 |  | 
|---|
| 219 | /* | 
|---|
| 220 | * Cache of the node's last and "current" cluster to avoid some | 
|---|
| 221 | * unnecessary FAT walks. | 
|---|
| 222 | */ | 
|---|
| 223 | /* Node's last cluster in FAT. */ | 
|---|
| 224 | bool            lastc_cached_valid; | 
|---|
| 225 | fat_cluster_t   lastc_cached_value; | 
|---|
| 226 | /* Node's "current" cluster, i.e. where the last I/O took place. */ | 
|---|
| 227 | bool            currc_cached_valid; | 
|---|
| 228 | aoff64_t        currc_cached_bn; | 
|---|
| 229 | fat_cluster_t   currc_cached_value; | 
|---|
| 230 | } fat_node_t; | 
|---|
| 231 |  | 
|---|
| 232 | extern vfs_out_ops_t fat_ops; | 
|---|
| 233 | extern libfs_ops_t fat_libfs_ops; | 
|---|
| 234 |  | 
|---|
| 235 | extern int fat_idx_get_new(fat_idx_t **, service_id_t); | 
|---|
| 236 | extern fat_idx_t *fat_idx_get_by_pos(service_id_t, fat_cluster_t, unsigned); | 
|---|
| 237 | extern fat_idx_t *fat_idx_get_by_index(service_id_t, fs_index_t); | 
|---|
| 238 | extern void fat_idx_destroy(fat_idx_t *); | 
|---|
| 239 | extern void fat_idx_hashin(fat_idx_t *); | 
|---|
| 240 | extern void fat_idx_hashout(fat_idx_t *); | 
|---|
| 241 |  | 
|---|
| 242 | extern int fat_idx_init(void); | 
|---|
| 243 | extern void fat_idx_fini(void); | 
|---|
| 244 | extern int fat_idx_init_by_service_id(service_id_t); | 
|---|
| 245 | extern void fat_idx_fini_by_service_id(service_id_t); | 
|---|
| 246 |  | 
|---|
| 247 | #endif | 
|---|
| 248 |  | 
|---|
| 249 | /** | 
|---|
| 250 | * @} | 
|---|
| 251 | */ | 
|---|