| [258] | 1 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 2 | // Date     : 01/06/2015 | 
|---|
|  | 3 | // Authors  : Alain Greiner | 
|---|
| [258] | 4 | // Copyright (c) UPMC-LIP6 | 
|---|
|  | 5 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [569] | 6 | // The fat32.h and fat32.c files define a library of access functions | 
|---|
| [587] | 7 | // to a FAT32 disk on a block device. It is intended to be used by both | 
|---|
|  | 8 | // the boot code and the kernel code. | 
|---|
| [258] | 9 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 10 | // Implementation notes: | 
|---|
|  | 11 | // 1. the "lba" (Logical Block Address) is the physical sector index on | 
|---|
|  | 12 | //    the block device. The physical sector size is supposed to be 512 bytes. | 
|---|
|  | 13 | // 2. the "cluster" variable is actually a cluster index. A cluster contains | 
|---|
| [587] | 14 | //    8 sectors (4K bytes) and the cluster index is a 32 bits word. | 
|---|
|  | 15 | // 3. Each file or directory referenced by the software is represented | 
|---|
|  | 16 | //    by an "inode". The set of "inodes" is organised as a tree, that is | 
|---|
|  | 17 | //    a sub-tree of the complete file system existing on the block device. | 
|---|
|  | 18 | // 4. A given file can be referenced by several software tasks, and each task | 
|---|
|  | 19 | //    will use a private handler, called a "file descriptor", allocated by the OS | 
|---|
|  | 20 | //    when the task open the file, that is organised as an indexed array. | 
|---|
|  | 21 | // 5. This FAT32 library implements (N+1) caches : one private "File_ Cache" | 
|---|
|  | 22 | //    for each referenced file or directory, and a specific "Fat_Cache" for | 
|---|
|  | 23 | //    the FAT itself. Each cache contain a variable number of clusters that are | 
|---|
|  | 24 | //    dynamically allocated when they are accessed, and organised as a 64-Tree. | 
|---|
| [258] | 25 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 26 | // General Debug Policy: | 
|---|
|  | 27 | // The global variable GIET_DEBUG_FAT is defined in the giet_config.h file. | 
|---|
|  | 28 | // The debug is activated if (proctime > GIET_DEBUG_FAT) && (GIET_DEBUG_FAT != 0) | 
|---|
|  | 29 | // The GIET_DEBUG_FAT bit 0 defines the level of debug: | 
|---|
|  | 30 | //    if   (GIET_DEBUG_FAT & 0x1)    => detailed debug | 
|---|
|  | 31 | //    else                           => external functions only | 
|---|
|  | 32 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [258] | 33 |  | 
|---|
|  | 34 | #include <giet_config.h> | 
|---|
| [530] | 35 | #include <hard_config.h> | 
|---|
| [258] | 36 | #include <fat32.h> | 
|---|
| [530] | 37 | #include <utils.h> | 
|---|
|  | 38 | #include <vmem.h> | 
|---|
| [587] | 39 | #include <kernel_malloc.h> | 
|---|
| [530] | 40 | #include <bdv_driver.h> | 
|---|
|  | 41 | #include <hba_driver.h> | 
|---|
|  | 42 | #include <sdc_driver.h> | 
|---|
|  | 43 | #include <rdk_driver.h> | 
|---|
|  | 44 | #include <mmc_driver.h> | 
|---|
| [458] | 45 | #include <tty0.h> | 
|---|
| [258] | 46 |  | 
|---|
|  | 47 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 48 | //               Global variables | 
|---|
| [258] | 49 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 50 |  | 
|---|
| [587] | 51 | // Fat-Descriptor | 
|---|
|  | 52 | __attribute__((section(".kdata"))) | 
|---|
|  | 53 | fat_desc_t _fat __attribute__((aligned(64))); | 
|---|
| [258] | 54 |  | 
|---|
| [587] | 55 | // buffer used by boot code as a simple cache when scanning FAT | 
|---|
|  | 56 | __attribute__((section(".kdata"))) | 
|---|
|  | 57 | unsigned char  _fat_buffer_fat[4096] __attribute__((aligned(64))); | 
|---|
| [530] | 58 |  | 
|---|
| [587] | 59 | // buffer used by boot code as a simple cache when scanning a directory in DATA region | 
|---|
|  | 60 | __attribute__((section(".kdata"))) | 
|---|
|  | 61 | unsigned char  _fat_buffer_data[4096] __attribute__((aligned(64))); | 
|---|
| [530] | 62 |  | 
|---|
| [587] | 63 | // lba of cluster in fat_buffer_fat | 
|---|
|  | 64 | __attribute__((section(".kdata"))) | 
|---|
|  | 65 | unsigned int   _fat_buffer_fat_lba; | 
|---|
|  | 66 |  | 
|---|
|  | 67 | // lba of cluster in fat_buffer_data | 
|---|
|  | 68 | __attribute__((section(".kdata"))) | 
|---|
|  | 69 | unsigned int   _fat_buffer_data_lba; | 
|---|
|  | 70 |  | 
|---|
|  | 71 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 72 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 73 | //                  Static functions declaration | 
|---|
|  | 74 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 75 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 76 |  | 
|---|
|  | 77 |  | 
|---|
|  | 78 | /////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 79 | // This debug function displays the content of a 512 bytes buffer "buf", | 
|---|
|  | 80 | // with an identifier defined by the "string" and "block_id" arguments. | 
|---|
|  | 81 | /////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 82 |  | 
|---|
|  | 83 | #if GIET_DEBUG_FAT | 
|---|
|  | 84 | static void _display_one_block( unsigned char* buf, | 
|---|
|  | 85 | char*          string, | 
|---|
|  | 86 | unsigned int   block_id ); | 
|---|
| [530] | 87 | #endif | 
|---|
|  | 88 |  | 
|---|
| [587] | 89 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 90 | // This debug function displays the FAT descriptor. | 
|---|
|  | 91 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [530] | 92 |  | 
|---|
| [587] | 93 | #if GIET_DEBUG_FAT | 
|---|
|  | 94 | static void _display_fat_descriptor(); | 
|---|
| [551] | 95 | #endif | 
|---|
| [530] | 96 |  | 
|---|
| [587] | 97 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 98 | // This debug function displays the sequence of clusters allocated to a | 
|---|
|  | 99 | // file (or directory) identified by the "inode" argument. | 
|---|
|  | 100 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [530] | 101 |  | 
|---|
| [587] | 102 | #if GIET_DEBUG_FAT | 
|---|
|  | 103 | static void _display_clusters_list( fat_inode_t* inode ); | 
|---|
| [530] | 104 | #endif | 
|---|
|  | 105 |  | 
|---|
| [587] | 106 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 107 | // The following function transfers one or several blocks between the device | 
|---|
|  | 108 | // and a memory buffer identified by a virtual address. | 
|---|
|  | 109 | // It computes the memory buffer physical address, and calls the proper | 
|---|
|  | 110 | // IOC driver depending on the subtype (BDV / HBA / SDC / SPI / RDK). | 
|---|
|  | 111 | // The use_irq argument allows to activate the descheduling mode, | 
|---|
|  | 112 | // if it supported by the IOC driver subtype | 
|---|
| [674] | 113 | // It returns 0 on success. | 
|---|
|  | 114 | // It returns -1 on error. | 
|---|
| [587] | 115 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [530] | 116 |  | 
|---|
| [587] | 117 | static int _fat_ioc_access( unsigned int use_irq, | 
|---|
|  | 118 | unsigned int to_mem, | 
|---|
|  | 119 | unsigned int lba, | 
|---|
|  | 120 | unsigned int buf_vaddr, | 
|---|
|  | 121 | unsigned int count ); | 
|---|
| [530] | 122 |  | 
|---|
| [258] | 123 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 124 | // The following function returns in the "desc" argument a pointer on a buffer | 
|---|
|  | 125 | // descriptor contained in a File_Cache, or in the Fat_Cache. | 
|---|
|  | 126 | // The searched buffer is idenfified by the "inode" and "cluster_id" arguments. | 
|---|
|  | 127 | // If the "inode" pointer is not NULL, the searched cache is a File-Cache. | 
|---|
|  | 128 | // If the "inode" pointer is NULL, the searched cache is the Fat-Cache, | 
|---|
|  | 129 | // The "cluster_id" argument is the buffer index in the file (or in the FAT). | 
|---|
|  | 130 | // In case of miss, it allocate a 4 Kbytes buffer and a cluster descriptor | 
|---|
|  | 131 | // from the local kernel heap, and calls the _fat_ioc_access() function to load | 
|---|
|  | 132 | // the missing cluster from the block device. | 
|---|
| [674] | 133 | // It returns 0 on success. | 
|---|
|  | 134 | // It returns 1 on error. | 
|---|
| [258] | 135 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 136 |  | 
|---|
|  | 137 | static unsigned int _get_buffer_from_cache( fat_inode_t*        inode, | 
|---|
|  | 138 | unsigned int        cluster_id, | 
|---|
|  | 139 | fat_cache_desc_t**  desc ); | 
|---|
|  | 140 |  | 
|---|
|  | 141 | //////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 142 | // This function extract a (partial) name from a LFN directory entry. | 
|---|
|  | 143 | //////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 144 |  | 
|---|
|  | 145 | static void _get_name_from_long( unsigned char* buffer, | 
|---|
|  | 146 | char*          name ); | 
|---|
|  | 147 |  | 
|---|
|  | 148 | //////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 149 | // The following function extract a name from a NORMAL directory entry. | 
|---|
|  | 150 | //////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 151 |  | 
|---|
|  | 152 | static void _get_name_from_short( unsigned char* buffer, | 
|---|
|  | 153 | char*          name ); | 
|---|
|  | 154 |  | 
|---|
|  | 155 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 156 | // This function returns the number of levels of a File-Cache (or Fat-Cache) | 
|---|
|  | 157 | // from the size of the file (or FAT). | 
|---|
|  | 158 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 159 |  | 
|---|
|  | 160 | static inline unsigned int _get_levels_from_size( unsigned int size ); | 
|---|
|  | 161 |  | 
|---|
|  | 162 | /////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 163 | // The following function analyses the "pathname" argument, from the character | 
|---|
|  | 164 | // defined by the "nb_read" argument. | 
|---|
|  | 165 | // It copies the found name in the "name" buffer (without '/'), | 
|---|
|  | 166 | // and updates the "nb_read" argument. | 
|---|
| [674] | 167 | // It returns 0 on success. | 
|---|
| [587] | 168 | // It returns 1 if one name length > NAME_MAX_SIZE characters | 
|---|
|  | 169 | /////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 170 |  | 
|---|
|  | 171 | static unsigned int _get_name_from_path( char*          pathname, | 
|---|
|  | 172 | char*          name, | 
|---|
|  | 173 | unsigned int*  nb_read ); | 
|---|
|  | 174 |  | 
|---|
|  | 175 | //////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 176 | // The following function scan the "pathname" argument, and copies in the | 
|---|
|  | 177 | // "name" buffer the last name in path (leaf name). | 
|---|
| [674] | 178 | // It returns 0 on success. | 
|---|
| [587] | 179 | // It returns 1 if one name length > NAME_MAX_SIZE characters | 
|---|
|  | 180 | //////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 181 | static unsigned int _get_last_name( char*   pathname, | 
|---|
|  | 182 | char*   name ); | 
|---|
|  | 183 |  | 
|---|
|  | 184 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [674] | 185 | // The following function accesses the Fat-Cache and returns in the "value" | 
|---|
| [587] | 186 | // argument the content of the FAT slot identified by the "cluster" argument. | 
|---|
|  | 187 | // It loads the missing cluster from block device into cache in case of miss. | 
|---|
| [674] | 188 | // It returns 0 on success. | 
|---|
|  | 189 | // It returns 1 on error. | 
|---|
| [587] | 190 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 191 |  | 
|---|
|  | 192 | static unsigned int _get_fat_entry( unsigned int  cluster, | 
|---|
|  | 193 | unsigned int* value ); | 
|---|
|  | 194 |  | 
|---|
|  | 195 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 196 | // The following function writes a new "value" in the Fat-Cache, in the slot | 
|---|
|  | 197 | // identified by the "cluster" argument. | 
|---|
|  | 198 | // It loads the missing cluster from block device into cache in case of miss. | 
|---|
| [674] | 199 | // It returns 0 on success, | 
|---|
|  | 200 | // It returns 1 on error. | 
|---|
| [587] | 201 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 202 |  | 
|---|
|  | 203 | static unsigned int _set_fat_entry( unsigned int  cluster, | 
|---|
|  | 204 | unsigned int  value ); | 
|---|
|  | 205 |  | 
|---|
|  | 206 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 207 | // The following function introduces the inode identified by the "child" argument, | 
|---|
|  | 208 | // as a new child of the "parent" inode in the Inode-Tree. | 
|---|
|  | 209 | // All checking are supposed to be done by the caller. | 
|---|
|  | 210 | // Nor the File-Cache, neither the block device are modified. | 
|---|
|  | 211 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 212 |  | 
|---|
|  | 213 | static void _add_inode_in_tree( fat_inode_t*  child, | 
|---|
|  | 214 | fat_inode_t*  parent ); | 
|---|
|  | 215 |  | 
|---|
|  | 216 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 217 | // The following function removes one inode identified by the "inode" argument | 
|---|
|  | 218 | // from the Inode-Tree. All checking are supposed to be done by the caller. | 
|---|
|  | 219 | // Nor the File-Cache, neither the block device are modified. | 
|---|
|  | 220 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 221 |  | 
|---|
|  | 222 | static void _remove_inode_from_tree( fat_inode_t* inode ); | 
|---|
|  | 223 |  | 
|---|
|  | 224 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 225 | // This recursive function scan one File-Cache (or Fat-Cache) from root to leaves, | 
|---|
|  | 226 | // to writes all dirty clusters to block device, and reset the dirty bits. | 
|---|
|  | 227 | // The cache is identified by the "root" an "levels" arguments. | 
|---|
|  | 228 | // The "string" argument is only used for debug : inode name or Fat. | 
|---|
| [674] | 229 | // It returns 0 on success. | 
|---|
|  | 230 | // It returns 1 on error. | 
|---|
| [587] | 231 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 232 |  | 
|---|
|  | 233 | static unsigned int _update_device_from_cache( unsigned int      levels, | 
|---|
|  | 234 | fat_cache_node_t* root, | 
|---|
|  | 235 | char*             string ); | 
|---|
|  | 236 |  | 
|---|
|  | 237 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [674] | 238 | // The following function accesses directly the FS_INFO block on the block device, | 
|---|
| [587] | 239 | // to update the "first_free_cluster" and "free_clusters_number" values, | 
|---|
|  | 240 | // using only the Fat-Descriptor single block buffer. | 
|---|
| [674] | 241 | // It return 0 on success. | 
|---|
|  | 242 | // It return 1 on error. | 
|---|
| [587] | 243 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 244 |  | 
|---|
|  | 245 | static unsigned int _update_fs_info(); | 
|---|
|  | 246 |  | 
|---|
|  | 247 | ////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 248 | // The following function read a data field (from one to four bytes) | 
|---|
|  | 249 | // from an unsigned char[] buffer, taking endianness into account. | 
|---|
|  | 250 | // The analysed field is defined by the "offset" and "size" arguments. | 
|---|
|  | 251 | ////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 252 |  | 
|---|
|  | 253 | static unsigned int _read_entry( unsigned int    offset, | 
|---|
|  | 254 | unsigned int    size, | 
|---|
|  | 255 | unsigned char*  buffer, | 
|---|
|  | 256 | unsigned int    little_indian ); | 
|---|
|  | 257 |  | 
|---|
|  | 258 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 259 | // The following function returns the lba of first sector in DATA region | 
|---|
|  | 260 | // from the cluster index. The cluster index must be larger than 2. | 
|---|
|  | 261 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 262 |  | 
|---|
|  | 263 | static unsigned int _cluster_to_lba( unsigned int cluster ); | 
|---|
|  | 264 |  | 
|---|
|  | 265 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 266 | // The following function returns in the "nb_entries" argument the number of files | 
|---|
| [620] | 267 | // (or directories) contained in a directory identified by the "inode " pointer. | 
|---|
| [674] | 268 | // It returns 0 on success. | 
|---|
|  | 269 | // It returns 1 on error. | 
|---|
| [587] | 270 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 271 |  | 
|---|
|  | 272 | static unsigned int _get_nb_entries( fat_inode_t*   inode, | 
|---|
|  | 273 | unsigned int*  nb_entries ); | 
|---|
|  | 274 |  | 
|---|
|  | 275 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 276 | // The following function search in the directory identified by the "parent" | 
|---|
|  | 277 | // inode pointer a child (file or directory) identified by its "name". | 
|---|
|  | 278 | // It returns in the "inode" argument the searched child inode pointer. | 
|---|
| [674] | 279 | // If the searched name is not found in the Inode-Tree, the function accesses | 
|---|
| [587] | 280 | // the "file_cache" associated to the parent directory. | 
|---|
| [674] | 281 | // If the child exists on block device, the Inode-Tree is updated, and | 
|---|
| [587] | 282 | // a success code is returned. | 
|---|
|  | 283 | // If the file/dir does not exist on block device, a error code is returned. | 
|---|
|  | 284 | // It returns 0 if inode found. | 
|---|
|  | 285 | // It returns 1 if inode not found. | 
|---|
| [674] | 286 | // It returns 2 on error in cache access. | 
|---|
| [587] | 287 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 288 |  | 
|---|
|  | 289 | static unsigned int _get_child_from_parent( fat_inode_t*   parent, | 
|---|
|  | 290 | char*          name, | 
|---|
|  | 291 | fat_inode_t**  inode ); | 
|---|
|  | 292 |  | 
|---|
|  | 293 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 294 | // For a file (or a directory) identified by the "pathname" argument, the | 
|---|
|  | 295 | // following function returns in the "inode" argument the inode pointer | 
|---|
|  | 296 | // associated to the searched file (or directory), with code (0). | 
|---|
|  | 297 | // If the searched file (or directory) is not found, but the parent directory | 
|---|
|  | 298 | // is found, it returns in the "inode" argument the pointer on the parent inode, | 
|---|
|  | 299 | // with code (1).  Finally, code (2) and code (3) are error codes. | 
|---|
|  | 300 | // Both the Inode-Tree and the involved Cache-Files are updated from the block | 
|---|
|  | 301 | // device in case of miss on one inode during the search in path. | 
|---|
|  | 302 | // Neither the Fat-Cache, nor the block device are updated. | 
|---|
|  | 303 | // It returns 0 if searched inode found | 
|---|
|  | 304 | // It returns 1 if searched inode not found but parent directory found | 
|---|
|  | 305 | // It returns 2 if searched inode not found and parent directory not found | 
|---|
|  | 306 | // It returns 3 if one name too long | 
|---|
|  | 307 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 308 |  | 
|---|
|  | 309 | static unsigned int _get_inode_from_path( char*          pathname, | 
|---|
|  | 310 | fat_inode_t**  inode ); | 
|---|
|  | 311 |  | 
|---|
|  | 312 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [638] | 313 | // The following function checks if node "a" is an ancestor of inode "b". | 
|---|
|  | 314 | // It returns 0 on failure. | 
|---|
|  | 315 | // It returns 1 otherwise. | 
|---|
|  | 316 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 317 |  | 
|---|
|  | 318 | static unsigned int _is_ancestor( fat_inode_t* a, | 
|---|
|  | 319 | fat_inode_t* b); | 
|---|
|  | 320 |  | 
|---|
|  | 321 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 322 | // This function computes the length and the number of LFN entries required | 
|---|
|  | 323 | // to store a node name in the "length" and "nb_lfn" arguments. | 
|---|
|  | 324 | // Short name (less than 13 characters) require 1 LFN entry. | 
|---|
|  | 325 | // Medium names (from 14 to 26 characters require 2 LFN entries. | 
|---|
|  | 326 | // Large names (up to 31 characters) require 3 LFN entries. | 
|---|
| [674] | 327 | // It returns 0 on success. | 
|---|
| [587] | 328 | // It returns 1 if length larger than 31 characters. | 
|---|
|  | 329 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 330 |  | 
|---|
|  | 331 | static unsigned int _check_name_length( char* name, | 
|---|
|  | 332 | unsigned int* length, | 
|---|
|  | 333 | unsigned int* nb_lfn ); | 
|---|
|  | 334 |  | 
|---|
|  | 335 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 336 | // For a node identified by the "inode" argument, this function updates the | 
|---|
|  | 337 | // "size" and "cluster" values in the entry of the parent directory File-Cache. | 
|---|
|  | 338 | // It set the dirty bit in the modified buffer of the parent directory File-Cache. | 
|---|
|  | 339 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 340 |  | 
|---|
|  | 341 | static unsigned int _update_dir_entry( fat_inode_t*  inode ); | 
|---|
|  | 342 |  | 
|---|
|  | 343 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 344 | // The following function add new "child" in Cache-File of "parent" directory. | 
|---|
| [674] | 345 | // It accesses the File_Cache associated to the parent directory, and scan the | 
|---|
| [587] | 346 | // clusters allocated to this directory to find the NO_MORE entry. | 
|---|
|  | 347 | // This entry will be the first modified entry in the directory. | 
|---|
|  | 348 | // Regarding the name storage, it uses LFN entries for all names. | 
|---|
|  | 349 | // Therefore, it writes 1, 2, or 3 LFN entries (depending on the child name | 
|---|
|  | 350 | // actual length, it writes one NORMAL entry, and writes the new NOMORE entry. | 
|---|
|  | 351 | // It updates the dentry field in the child inode. | 
|---|
|  | 352 | // It set the dirty bit for all modified File-Cache buffers. | 
|---|
|  | 353 | // The block device is not modified by this function. | 
|---|
|  | 354 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 355 |  | 
|---|
|  | 356 | static unsigned int _add_dir_entry( fat_inode_t* child, | 
|---|
|  | 357 | fat_inode_t* parent ); | 
|---|
|  | 358 |  | 
|---|
|  | 359 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 360 | // The following function invalidates all dir_entries associated to the "inode" | 
|---|
|  | 361 | // argument from its parent directory. | 
|---|
|  | 362 | // It set the dirty bit for all modified buffers in parent directory Cache-File. | 
|---|
|  | 363 | // The inode itself is not modified by this function. | 
|---|
|  | 364 | // The block device is not modified by this function. | 
|---|
|  | 365 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 366 |  | 
|---|
|  | 367 | static unsigned int _remove_dir_entry( fat_inode_t*  inode ); | 
|---|
|  | 368 |  | 
|---|
|  | 369 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 370 | // The following function add the special entries "." and ".." in the File-Cache | 
|---|
|  | 371 | // of the directory identified by the "child" argument. | 
|---|
|  | 372 | // The parent directory is defined by the "parent" argument. | 
|---|
|  | 373 | // The child directory File-Cache is supposed to be empty. | 
|---|
|  | 374 | // We use two NORMAL entries for these "." and ".." entries. | 
|---|
|  | 375 | // The block device is not modified by this function. | 
|---|
|  | 376 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 377 |  | 
|---|
|  | 378 | static void _add_special_directories( fat_inode_t* child, | 
|---|
|  | 379 | fat_inode_t* parent ); | 
|---|
|  | 380 |  | 
|---|
|  | 381 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 382 | // The following function releases all clusters allocated to a file or directory, | 
|---|
|  | 383 | // from the cluster index defined by the "cluster" argument, until the end | 
|---|
|  | 384 | // of the FAT linked list. | 
|---|
|  | 385 | // It calls _get_fat_entry() and _set_fat_entry() functions to scan the FAT, | 
|---|
|  | 386 | // and to update the clusters chaining. | 
|---|
|  | 387 | // The FAT region on block device is updated. | 
|---|
| [674] | 388 | // It returns 0 on success. | 
|---|
|  | 389 | // It returns 1 on error. | 
|---|
| [587] | 390 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 391 |  | 
|---|
|  | 392 | static unsigned int _clusters_release( unsigned int cluster ); | 
|---|
|  | 393 |  | 
|---|
|  | 394 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 395 | // This function allocate "nb_clusters_more" new clusters to a file (or directory) | 
|---|
|  | 396 | // identified by the "inode" pointer. It allocates also the associated buffers | 
|---|
|  | 397 | // and buffer descriptors in the Cache-File. | 
|---|
|  | 398 | // It calls _get_fat_entry() and _set_fat_entry() functions to update the | 
|---|
|  | 399 | // clusters chaining in the Cache-Fat. The FAT region on block device is updated. | 
|---|
| [674] | 400 | // It returns 0 on success. | 
|---|
|  | 401 | // It returns 1 on error. | 
|---|
| [587] | 402 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 403 |  | 
|---|
|  | 404 | static unsigned int _clusters_allocate( fat_inode_t* inode, | 
|---|
|  | 405 | unsigned int nb_clusters_current, | 
|---|
|  | 406 | unsigned int nb_clusters_more ); | 
|---|
|  | 407 |  | 
|---|
|  | 408 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [651] | 409 | // This recursive function scans one File-Cache (or Fat-Cache) from root to | 
|---|
|  | 410 | // leaves. All memory allocated for 4KB buffers, and buffer descriptors (in | 
|---|
|  | 411 | // leaves) is released, along with the 64-Tree structure (root node is kept). | 
|---|
|  | 412 | // The cache is identified by the "root" and "levels" arguments. | 
|---|
| [587] | 413 | // It should not contain any dirty clusters. | 
|---|
|  | 414 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 415 |  | 
|---|
| [651] | 416 | static void _release_cache_memory( fat_cache_node_t*  root, | 
|---|
|  | 417 | unsigned int       levels ); | 
|---|
| [587] | 418 |  | 
|---|
|  | 419 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [639] | 420 | // The following function allocates and initializes a new Fat-Cache node. | 
|---|
|  | 421 | // Its first child can be specified (used when adding a cache level). | 
|---|
|  | 422 | // The Fat-Cache is initialized as empty: all children set to NULL. | 
|---|
|  | 423 | // It returns a pointer to a new Fat-Cache node. | 
|---|
|  | 424 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 425 |  | 
|---|
|  | 426 | static fat_cache_node_t* _allocate_one_cache_node( fat_cache_node_t* first_child ); | 
|---|
|  | 427 |  | 
|---|
|  | 428 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 429 | // The following function allocates and initializes a new inode, | 
|---|
|  | 430 | // using the values defined by the arguments. | 
|---|
|  | 431 | // If the "cache_allocate" argument is true ans empty cache is allocated. | 
|---|
|  | 432 | // It returns a pointer on the new inode. | 
|---|
|  | 433 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 434 |  | 
|---|
|  | 435 | static fat_inode_t* _allocate_one_inode( char*        name, | 
|---|
|  | 436 | unsigned int is_dir, | 
|---|
|  | 437 | unsigned int cluster, | 
|---|
|  | 438 | unsigned int size, | 
|---|
|  | 439 | unsigned int count, | 
|---|
|  | 440 | unsigned int dentry, | 
|---|
|  | 441 | unsigned int cache_allocate ); | 
|---|
|  | 442 |  | 
|---|
|  | 443 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 444 | // The following function allocates one 4 Kbytes buffer and associated cluster | 
|---|
|  | 445 | // descriptor for the file (or directory) identified by the "inode" argument, | 
|---|
|  | 446 | // and updates the Cache_File slot identified by the "cluster_id" argument. | 
|---|
|  | 447 | // The File-Cache slot must be empty. | 
|---|
|  | 448 | // It updates the cluster descriptor, using the "cluster" argument, that is | 
|---|
|  | 449 | // the cluster index in FAT.  The cluster descriptor dirty field is set. | 
|---|
|  | 450 | // It traverse the 64-tree Cache-file from top to bottom to find the last level. | 
|---|
|  | 451 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 452 |  | 
|---|
|  | 453 | static void _allocate_one_buffer( fat_inode_t*    inode, | 
|---|
|  | 454 | unsigned int    cluster_id, | 
|---|
|  | 455 | unsigned int    cluster ); | 
|---|
|  | 456 |  | 
|---|
|  | 457 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 458 | // The following function allocates one free cluster from the FAT "heap" of free | 
|---|
|  | 459 | // clusters, and returns the cluster index in the "cluster" argument. | 
|---|
|  | 460 | // It updates the FAT slot, and the two FAT global variables: first_free_cluster, | 
|---|
|  | 461 | // and free_clusters_number. | 
|---|
| [674] | 462 | // It returns 0 on success. | 
|---|
|  | 463 | // It returns 1 on error. | 
|---|
| [587] | 464 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 465 |  | 
|---|
|  | 466 | static unsigned int _allocate_one_cluster( unsigned int*  cluster ); | 
|---|
|  | 467 |  | 
|---|
|  | 468 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 469 | // This function remove from the file system a file or a directory | 
|---|
|  | 470 | // identified by the "inode" argument. | 
|---|
|  | 471 | // The remove condition must be checked by the caller. | 
|---|
|  | 472 | // The relevant lock(s) must have been taken by te caller. | 
|---|
| [674] | 473 | // It returns 0 on success. | 
|---|
|  | 474 | // It returns 1 on error. | 
|---|
| [587] | 475 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 476 |  | 
|---|
|  | 477 | static unsigned int _remove_node_from_fs( fat_inode_t* inode ); | 
|---|
|  | 478 |  | 
|---|
|  | 479 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 480 | // This function return the cluster index and the size for a file | 
|---|
|  | 481 | // identified by the "pathname" argument, scanning directly the block | 
|---|
|  | 482 | // device DATA region. | 
|---|
|  | 483 | // It is intended to be called only by the _fat_load_no_cache() function, | 
|---|
|  | 484 | // it does not use the dynamically allocated File Caches, but uses only | 
|---|
|  | 485 | // the 4 Kbytes _fat_buffer_data. | 
|---|
| [674] | 486 | // It returns 0 on success. | 
|---|
|  | 487 | // It returns 1 on error. | 
|---|
| [587] | 488 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 489 |  | 
|---|
|  | 490 | static unsigned int _file_info_no_cache( char*          pathname, | 
|---|
|  | 491 | unsigned int*  file_cluster, | 
|---|
|  | 492 | unsigned int*  file_size ); | 
|---|
|  | 493 |  | 
|---|
|  | 494 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 495 | // This function scan directly the FAT region on the block device, | 
|---|
|  | 496 | // and returns in the "next" argument the value stored in the fat slot | 
|---|
|  | 497 | // identified by the "cluster" argument. | 
|---|
|  | 498 | // It is intended to be called only by the _fat_load_no_cache() function, | 
|---|
|  | 499 | // as it does not use the dynamically allocated Fat-Cache, but uses only | 
|---|
|  | 500 | // the 4 Kbytes _fat_buffer_fat. | 
|---|
| [674] | 501 | // It returns 0 on success. | 
|---|
|  | 502 | // It returns 1 on error. | 
|---|
| [587] | 503 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 504 |  | 
|---|
|  | 505 | static unsigned int _next_cluster_no_cache( unsigned int   cluster, | 
|---|
|  | 506 | unsigned int*  next ); | 
|---|
|  | 507 |  | 
|---|
|  | 508 |  | 
|---|
|  | 509 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 510 | // The following functions return the length or the size of a FAT field, | 
|---|
|  | 511 | // identified by an (offset,length) mnemonic defined in the fat32.h file. | 
|---|
|  | 512 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 513 |  | 
|---|
|  | 514 | static inline int get_length( int offset , int length ) { return length; } | 
|---|
|  | 515 |  | 
|---|
|  | 516 | static inline int get_offset( int offset , int length ) { return offset; } | 
|---|
|  | 517 |  | 
|---|
|  | 518 |  | 
|---|
|  | 519 |  | 
|---|
|  | 520 |  | 
|---|
|  | 521 |  | 
|---|
|  | 522 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 523 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 524 | //                  Static functions definition | 
|---|
|  | 525 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 526 | ////////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 527 |  | 
|---|
|  | 528 | #if GIET_DEBUG_FAT | 
|---|
|  | 529 | /////////////////////////////////////////////////// | 
|---|
|  | 530 | static void _display_one_block( unsigned char* buf, | 
|---|
|  | 531 | char*          string, | 
|---|
|  | 532 | unsigned int   block_id ) | 
|---|
| [258] | 533 | { | 
|---|
|  | 534 | unsigned int line; | 
|---|
|  | 535 | unsigned int word; | 
|---|
|  | 536 |  | 
|---|
| [587] | 537 | _printf("\n***  <%s>  block %x  ***********************************\n", | 
|---|
|  | 538 | string , block_id ); | 
|---|
| [258] | 539 |  | 
|---|
| [417] | 540 | for ( line = 0 ; line < 16 ; line++ ) | 
|---|
| [258] | 541 | { | 
|---|
| [417] | 542 | // display line index | 
|---|
| [587] | 543 | _printf("%x : ", line ); | 
|---|
| [258] | 544 |  | 
|---|
| [417] | 545 | // display 8*4 bytes hexa | 
|---|
| [258] | 546 | for ( word=0 ; word<8 ; word++ ) | 
|---|
|  | 547 | { | 
|---|
|  | 548 | unsigned int byte  = (line<<5) + (word<<2); | 
|---|
| [587] | 549 | unsigned int hexa  = (buf[byte  ]<<24) | | 
|---|
|  | 550 | (buf[byte+1]<<16) | | 
|---|
|  | 551 | (buf[byte+2]<< 8) | | 
|---|
|  | 552 | (buf[byte+3]      ); | 
|---|
|  | 553 | _printf(" %X |", hexa ); | 
|---|
| [258] | 554 | } | 
|---|
| [587] | 555 | _printf("\n"); | 
|---|
| [258] | 556 | } | 
|---|
| [587] | 557 | _printf("*******************************************************************\n"); | 
|---|
|  | 558 | } // end _display_one_block() | 
|---|
| [530] | 559 | #endif | 
|---|
| [291] | 560 |  | 
|---|
| [587] | 561 |  | 
|---|
|  | 562 |  | 
|---|
| [530] | 563 | #if GIET_DEBUG_FAT | 
|---|
| [587] | 564 | ///////////////////////////////////// | 
|---|
|  | 565 | static void _display_fat_descriptor() | 
|---|
| [530] | 566 | { | 
|---|
| [587] | 567 | _printf("\n###############  FAT DESCRIPTOR  ################################" | 
|---|
| [674] | 568 | "\nFAT initialized                  %x" | 
|---|
| [587] | 569 | "\nBlock Size  (bytes)              %x" | 
|---|
|  | 570 | "\nCluster Size  (bytes)            %x" | 
|---|
| [530] | 571 | "\nFAT region first lba             %x" | 
|---|
| [587] | 572 | "\nFAT region size (blocks)         %x" | 
|---|
|  | 573 | "\nDATA region first lba            %x" | 
|---|
|  | 574 | "\nDATA region size (blocks)        %x" | 
|---|
| [530] | 575 | "\nNumber of free clusters          %x" | 
|---|
| [587] | 576 | "\nFirst free cluster index         %x" | 
|---|
|  | 577 | "\nFat_cache_levels                 %d" | 
|---|
| [530] | 578 | "\n#################################################################\n", | 
|---|
| [674] | 579 | _fat.initialized, | 
|---|
| [530] | 580 | _fat.sector_size, | 
|---|
| [587] | 581 | _fat.cluster_size, | 
|---|
| [530] | 582 | _fat.fat_lba, | 
|---|
| [587] | 583 | _fat.fat_sectors, | 
|---|
| [530] | 584 | _fat.data_lba, | 
|---|
| [587] | 585 | _fat.data_sectors, | 
|---|
|  | 586 | _fat.free_clusters_number, | 
|---|
|  | 587 | _fat.first_free_cluster, | 
|---|
|  | 588 | _fat.fat_cache_levels ); | 
|---|
|  | 589 |  | 
|---|
|  | 590 | } // end _display_fat_descriptor() | 
|---|
| [530] | 591 | #endif | 
|---|
|  | 592 |  | 
|---|
| [587] | 593 |  | 
|---|
|  | 594 |  | 
|---|
|  | 595 | #if GIET_DEBUG_FAT | 
|---|
|  | 596 | //////////////////////////////////////////////////////// | 
|---|
|  | 597 | static void _display_clusters_list( fat_inode_t* inode ) | 
|---|
| [258] | 598 | { | 
|---|
| [587] | 599 | _printf("\n**************** clusters for <%s> ***********************\n", inode->name ); | 
|---|
|  | 600 | unsigned int next; | 
|---|
|  | 601 | unsigned int n       = 0; | 
|---|
|  | 602 | unsigned int current = inode->cluster; | 
|---|
|  | 603 | while( (current < END_OF_CHAIN_CLUSTER_MIN) && (n < 1024) ) | 
|---|
|  | 604 | { | 
|---|
|  | 605 | _get_fat_entry( current , &next ); | 
|---|
|  | 606 | _printf(" > %X", current ); | 
|---|
|  | 607 | n++; | 
|---|
|  | 608 | if ( (n & 0x7) == 0 ) _printf("\n"); | 
|---|
|  | 609 | current = next; | 
|---|
|  | 610 | } | 
|---|
|  | 611 | _printf("\n"); | 
|---|
|  | 612 | }  // end _display_clusters_list() | 
|---|
|  | 613 | #endif | 
|---|
| [258] | 614 |  | 
|---|
| [587] | 615 |  | 
|---|
|  | 616 |  | 
|---|
|  | 617 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 618 | static int _fat_ioc_access( unsigned int use_irq,       // descheduling if non zero | 
|---|
|  | 619 | unsigned int to_mem,        // read / write | 
|---|
|  | 620 | unsigned int lba,           // first sector on device | 
|---|
|  | 621 | unsigned int buf_vaddr,     // memory buffer vaddr | 
|---|
|  | 622 | unsigned int count )        // number of sectors | 
|---|
| [291] | 623 | { | 
|---|
| [587] | 624 | // compute memory buffer physical address | 
|---|
|  | 625 | unsigned int       flags;         // for _v2p_translate | 
|---|
|  | 626 | unsigned long long buf_paddr;     // buffer physical address | 
|---|
| [291] | 627 |  | 
|---|
| [587] | 628 | if ( ((_get_mmu_mode() & 0x4) == 0 ) || USE_IOC_RDK )  // identity | 
|---|
|  | 629 | { | 
|---|
|  | 630 | buf_paddr = (unsigned long long)buf_vaddr; | 
|---|
|  | 631 | } | 
|---|
|  | 632 | else                                // V2P translation required | 
|---|
| [291] | 633 | { | 
|---|
| [587] | 634 | buf_paddr = _v2p_translate( buf_vaddr , &flags ); | 
|---|
| [291] | 635 | } | 
|---|
| [587] | 636 |  | 
|---|
|  | 637 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 638 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 639 | _printf("\n[DEBUG FAT] _fat_ioc_access(): enters at cycle %d\n" | 
|---|
| [587] | 640 | "  to_mem = %d / vaddr = %x / paddr = %l / sectors = %d / lba = %x\n", | 
|---|
|  | 641 | _get_proctime(), to_mem, buf_vaddr, buf_paddr, count, lba ); | 
|---|
|  | 642 | #endif | 
|---|
|  | 643 |  | 
|---|
|  | 644 |  | 
|---|
|  | 645 | #if GIET_NO_HARD_CC     // L1 cache inval (virtual addresses) | 
|---|
|  | 646 | if ( to_mem ) _dcache_buf_invalidate( buf_vaddr, count<<9 ); | 
|---|
|  | 647 | #endif | 
|---|
|  | 648 |  | 
|---|
|  | 649 |  | 
|---|
|  | 650 | #if   ( USE_IOC_BDV )   // call the proper driver | 
|---|
|  | 651 | return( _bdv_access( use_irq , to_mem , lba , buf_paddr , count ) ); | 
|---|
|  | 652 | #elif ( USE_IOC_HBA ) | 
|---|
|  | 653 | return( _hba_access( use_irq , to_mem , lba , buf_paddr , count ) ); | 
|---|
|  | 654 | #elif ( USE_IOC_SDC ) | 
|---|
|  | 655 | return( _sdc_access( use_irq , to_mem , lba , buf_paddr , count ) ); | 
|---|
|  | 656 | #elif ( USE_IOC_SPI ) | 
|---|
|  | 657 | return( _spi_access( use_irq , to_mem , lba , buf_paddr , count ) ); | 
|---|
|  | 658 | #elif ( USE_IOC_RDK ) | 
|---|
|  | 659 | return( _rdk_access( use_irq , to_mem , lba , buf_paddr , count ) ); | 
|---|
|  | 660 | #else | 
|---|
| [664] | 661 | _printf("\n[FAT ERROR] _fat_ioc_access(): no IOC driver\n"); | 
|---|
| [587] | 662 | _exit(); | 
|---|
|  | 663 | #endif | 
|---|
|  | 664 |  | 
|---|
|  | 665 | }  // end _fat_ioc_access() | 
|---|
|  | 666 |  | 
|---|
|  | 667 |  | 
|---|
|  | 668 |  | 
|---|
|  | 669 |  | 
|---|
|  | 670 | ///////////////////////////////////////////////////////////////////// | 
|---|
|  | 671 | static inline unsigned int _get_levels_from_size( unsigned int size ) | 
|---|
|  | 672 | { | 
|---|
|  | 673 | if      ( size <= (1<<18) ) return 1;     // 64 clusters == 256 Kbytes | 
|---|
|  | 674 | else if ( size <= (1<<24) ) return 2;     // 64 * 64 clusters => 16 Mbytes | 
|---|
|  | 675 | else if ( size <= (1<<30) ) return 3;     // 64 * 64 * 64 cluster => 1 Gbytes | 
|---|
|  | 676 | else                        return 4;     // 64 * 64 * 64 * 64 clusters | 
|---|
| [291] | 677 | } | 
|---|
|  | 678 |  | 
|---|
| [587] | 679 |  | 
|---|
|  | 680 |  | 
|---|
|  | 681 | //////////////////////////////////////////////////////// | 
|---|
|  | 682 | static unsigned int _read_entry( unsigned int    offset, | 
|---|
|  | 683 | unsigned int    size, | 
|---|
|  | 684 | unsigned char*  buffer, | 
|---|
|  | 685 | unsigned int    little_endian ) | 
|---|
| [258] | 686 | { | 
|---|
| [587] | 687 | unsigned int n; | 
|---|
| [258] | 688 | unsigned int res  = 0; | 
|---|
|  | 689 |  | 
|---|
| [587] | 690 | if ( little_endian) | 
|---|
| [258] | 691 | { | 
|---|
| [587] | 692 | for( n = size ; n > 0 ; n-- ) res = (res<<8) | buffer[offset+n-1]; | 
|---|
| [258] | 693 | } | 
|---|
|  | 694 | else | 
|---|
|  | 695 | { | 
|---|
| [587] | 696 | for( n = 0 ; n < size ; n++ ) res = (res<<8) | buffer[offset+n]; | 
|---|
| [258] | 697 | } | 
|---|
|  | 698 | return res; | 
|---|
|  | 699 |  | 
|---|
| [587] | 700 | }  // end _read_entry | 
|---|
| [258] | 701 |  | 
|---|
|  | 702 |  | 
|---|
|  | 703 |  | 
|---|
| [587] | 704 | ////////////////////////////////////////////////////////////////// | 
|---|
|  | 705 | static inline unsigned int _cluster_to_lba( unsigned int cluster ) | 
|---|
| [258] | 706 | { | 
|---|
| [587] | 707 | if ( cluster < 2 ) | 
|---|
|  | 708 | { | 
|---|
| [664] | 709 | _printf("\n[FAT ERROR] _cluster_to_lba(): cluster smaller than 2\n"); | 
|---|
| [587] | 710 | _exit(); | 
|---|
| [530] | 711 | } | 
|---|
| [258] | 712 |  | 
|---|
| [587] | 713 | return  ((cluster - 2) << 3) + _fat.data_lba; | 
|---|
| [258] | 714 | } | 
|---|
|  | 715 |  | 
|---|
| [587] | 716 |  | 
|---|
| [258] | 717 | ////////////////////////////////////////////////////// | 
|---|
| [596] | 718 | static inline unsigned char _to_lower(unsigned char c) | 
|---|
|  | 719 | { | 
|---|
|  | 720 | if (c >= 'A' && c <= 'Z') return (c | 0x20); | 
|---|
|  | 721 | else                      return c; | 
|---|
|  | 722 | } | 
|---|
|  | 723 |  | 
|---|
|  | 724 |  | 
|---|
|  | 725 | ////////////////////////////////////////////////////// | 
|---|
| [587] | 726 | static inline unsigned char _to_upper(unsigned char c) | 
|---|
| [258] | 727 | { | 
|---|
|  | 728 | if (c >= 'a' && c <= 'z') return (c & ~(0x20)); | 
|---|
|  | 729 | else                      return c; | 
|---|
|  | 730 | } | 
|---|
|  | 731 |  | 
|---|
|  | 732 |  | 
|---|
|  | 733 |  | 
|---|
| [587] | 734 | /////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 735 | static unsigned int _get_name_from_path( char*          pathname,  // input | 
|---|
|  | 736 | char*          name,      // output | 
|---|
|  | 737 | unsigned int*  nb_read )  // input & output | 
|---|
| [291] | 738 | { | 
|---|
| [587] | 739 | // skip leading "/" character | 
|---|
|  | 740 | if ( pathname[*nb_read] == '/' ) *nb_read = *nb_read + 1; | 
|---|
| [291] | 741 |  | 
|---|
| [587] | 742 | // initialises current indexes | 
|---|
|  | 743 | unsigned int i = *nb_read; | 
|---|
|  | 744 | unsigned int j = 0; | 
|---|
| [291] | 745 |  | 
|---|
| [587] | 746 | while ( (pathname[i] != '/') && (pathname[i] != 0) ) | 
|---|
| [291] | 747 | { | 
|---|
| [587] | 748 | name[j++] = pathname[i++]; | 
|---|
|  | 749 | if ( j > NAME_MAX_SIZE ) return 1; | 
|---|
| [291] | 750 | } | 
|---|
| [587] | 751 |  | 
|---|
|  | 752 | // set end of string | 
|---|
| [291] | 753 | name[j] = 0; | 
|---|
|  | 754 |  | 
|---|
| [587] | 755 | // skip trailing "/" character | 
|---|
| [291] | 756 | if ( pathname[i] == '/' ) *nb_read += j+1; | 
|---|
|  | 757 | else                      *nb_read += j; | 
|---|
|  | 758 |  | 
|---|
| [587] | 759 | return 0; | 
|---|
| [291] | 760 | } | 
|---|
|  | 761 |  | 
|---|
| [587] | 762 |  | 
|---|
|  | 763 |  | 
|---|
|  | 764 | //////////////////////////////////////////////////////////////////// | 
|---|
|  | 765 | static unsigned int _get_last_name( char*   pathname,       // input | 
|---|
|  | 766 | char*   name )          // output | 
|---|
|  | 767 | { | 
|---|
|  | 768 | unsigned int nb_read = 0; | 
|---|
|  | 769 | while ( pathname[nb_read] != 0 ) | 
|---|
|  | 770 | { | 
|---|
|  | 771 | if ( _get_name_from_path( pathname, name, &nb_read ) ) return 1; | 
|---|
|  | 772 | } | 
|---|
|  | 773 |  | 
|---|
|  | 774 | return 0; | 
|---|
|  | 775 | }   // end _get_last_name() | 
|---|
|  | 776 |  | 
|---|
|  | 777 |  | 
|---|
|  | 778 |  | 
|---|
| [258] | 779 | //////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 780 | static void _get_name_from_short( unsigned char* buffer,  // input:  SFN dir_entry | 
|---|
|  | 781 | char*          name )   // output: name | 
|---|
| [258] | 782 | { | 
|---|
| [596] | 783 | unsigned int i; | 
|---|
|  | 784 | unsigned int j = 0; | 
|---|
| [258] | 785 |  | 
|---|
| [596] | 786 | // get name | 
|---|
|  | 787 | for ( i = 0; i < 8 && buffer[i] != ' '; i++ ) | 
|---|
| [258] | 788 | { | 
|---|
| [596] | 789 | name[j] = _to_lower( buffer[i] ); | 
|---|
|  | 790 | j++; | 
|---|
| [258] | 791 | } | 
|---|
| [596] | 792 |  | 
|---|
|  | 793 | // get extension | 
|---|
|  | 794 | for ( i = 8; i < 8 + 3 && buffer[i] != ' '; i++ ) | 
|---|
|  | 795 | { | 
|---|
|  | 796 | // we entered the loop so there is an extension. add the dot | 
|---|
|  | 797 | if ( i == 8 ) | 
|---|
|  | 798 | { | 
|---|
|  | 799 | name[j] = '.'; | 
|---|
|  | 800 | j++; | 
|---|
|  | 801 | } | 
|---|
|  | 802 |  | 
|---|
|  | 803 | name[j] = _to_lower( buffer[i] ); | 
|---|
|  | 804 | j++; | 
|---|
|  | 805 | } | 
|---|
|  | 806 |  | 
|---|
|  | 807 | name[j] = '\0'; | 
|---|
| [258] | 808 | } | 
|---|
| [291] | 809 |  | 
|---|
| [258] | 810 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 811 | static void _get_name_from_long( unsigned char*  buffer, // input : LFN dir_entry | 
|---|
|  | 812 | char*           name )  // output : name | 
|---|
| [258] | 813 | { | 
|---|
| [587] | 814 | unsigned int   name_offset         = 0; | 
|---|
|  | 815 | unsigned int   buffer_offset       = get_length(LDIR_ORD); | 
|---|
| [258] | 816 | unsigned int   l_name_1            = get_length(LDIR_NAME_1); | 
|---|
|  | 817 | unsigned int   l_name_2            = get_length(LDIR_NAME_2); | 
|---|
|  | 818 | unsigned int   l_name_3            = get_length(LDIR_NAME_3); | 
|---|
|  | 819 | unsigned int   l_attr              = get_length(LDIR_ATTR); | 
|---|
|  | 820 | unsigned int   l_type              = get_length(LDIR_TYPE); | 
|---|
|  | 821 | unsigned int   l_chksum            = get_length(LDIR_CHKSUM); | 
|---|
|  | 822 | unsigned int   l_rsvd              = get_length(LDIR_RSVD); | 
|---|
|  | 823 |  | 
|---|
|  | 824 | unsigned int j            = 0; | 
|---|
|  | 825 | unsigned int eof          = 0; | 
|---|
|  | 826 |  | 
|---|
| [587] | 827 | while ( (buffer_offset != DIR_ENTRY_SIZE)  && (!eof) ) | 
|---|
| [258] | 828 | { | 
|---|
|  | 829 | while (j != l_name_1 && !eof ) | 
|---|
|  | 830 | { | 
|---|
| [587] | 831 | if ( (buffer[buffer_offset] == 0x00) || | 
|---|
|  | 832 | (buffer[buffer_offset] == 0xFF) ) | 
|---|
| [258] | 833 | { | 
|---|
|  | 834 | eof = 1; | 
|---|
|  | 835 | continue; | 
|---|
|  | 836 | } | 
|---|
| [587] | 837 | name[name_offset] = buffer[buffer_offset]; | 
|---|
|  | 838 | buffer_offset += 2; | 
|---|
| [258] | 839 | j += 2; | 
|---|
| [587] | 840 | name_offset++; | 
|---|
| [258] | 841 | } | 
|---|
|  | 842 |  | 
|---|
| [587] | 843 | buffer_offset += (l_attr + l_type + l_chksum); | 
|---|
| [258] | 844 | j = 0; | 
|---|
|  | 845 |  | 
|---|
|  | 846 | while (j != l_name_2 && !eof ) | 
|---|
|  | 847 | { | 
|---|
| [587] | 848 | if ( (buffer[buffer_offset] == 0x00) || | 
|---|
|  | 849 | (buffer[buffer_offset] == 0xFF) ) | 
|---|
| [258] | 850 | { | 
|---|
|  | 851 | eof = 1; | 
|---|
|  | 852 | continue; | 
|---|
|  | 853 | } | 
|---|
| [587] | 854 | name[name_offset] = buffer[buffer_offset]; | 
|---|
|  | 855 | buffer_offset += 2; | 
|---|
| [258] | 856 | j += 2; | 
|---|
| [587] | 857 | name_offset++; | 
|---|
| [258] | 858 | } | 
|---|
|  | 859 |  | 
|---|
| [587] | 860 | buffer_offset += l_rsvd; | 
|---|
| [258] | 861 | j = 0; | 
|---|
|  | 862 |  | 
|---|
|  | 863 | while (j != l_name_3 && !eof ) | 
|---|
|  | 864 | { | 
|---|
| [587] | 865 | if ( (buffer[buffer_offset] == 0x00) || | 
|---|
|  | 866 | (buffer[buffer_offset] == 0xFF) ) | 
|---|
| [258] | 867 | { | 
|---|
|  | 868 | eof = 1; | 
|---|
|  | 869 | continue; | 
|---|
|  | 870 | } | 
|---|
| [587] | 871 | name[name_offset] = buffer[buffer_offset]; | 
|---|
|  | 872 | buffer_offset += 2; | 
|---|
| [258] | 873 | j += 2; | 
|---|
| [587] | 874 | name_offset++; | 
|---|
| [258] | 875 | } | 
|---|
|  | 876 | } | 
|---|
| [587] | 877 | name[name_offset] = 0; | 
|---|
| [258] | 878 | } // end get_name_from_long() | 
|---|
|  | 879 |  | 
|---|
| [587] | 880 |  | 
|---|
|  | 881 |  | 
|---|
| [639] | 882 | //////////////////////////////////////////////////////////// | 
|---|
|  | 883 | static fat_cache_node_t* _allocate_one_cache_node( fat_cache_node_t* first_child ) | 
|---|
|  | 884 | { | 
|---|
|  | 885 | fat_cache_node_t* cnode; | 
|---|
|  | 886 | unsigned int i; | 
|---|
| [587] | 887 |  | 
|---|
| [639] | 888 | cnode = _malloc( sizeof(fat_cache_node_t) ); | 
|---|
|  | 889 |  | 
|---|
|  | 890 | cnode->children[0] = first_child; | 
|---|
|  | 891 | for ( i = 1 ; i < 64 ; i++ ) | 
|---|
|  | 892 | cnode->children[i] = NULL; | 
|---|
|  | 893 |  | 
|---|
|  | 894 | return cnode; | 
|---|
|  | 895 | }   // end _allocate_one_cache_node() | 
|---|
|  | 896 |  | 
|---|
|  | 897 |  | 
|---|
|  | 898 |  | 
|---|
| [587] | 899 | //////////////////////////////////////////////////////////// | 
|---|
|  | 900 | static fat_inode_t* _allocate_one_inode( char*        name, | 
|---|
|  | 901 | unsigned int is_dir, | 
|---|
|  | 902 | unsigned int cluster, | 
|---|
|  | 903 | unsigned int size, | 
|---|
|  | 904 | unsigned int count, | 
|---|
|  | 905 | unsigned int dentry, | 
|---|
|  | 906 | unsigned int cache_allocate ) | 
|---|
| [291] | 907 | { | 
|---|
| [587] | 908 | fat_inode_t* new_inode  = _malloc( sizeof(fat_inode_t) ); | 
|---|
| [291] | 909 |  | 
|---|
| [587] | 910 | new_inode->parent   = NULL;                 // set by _add_inode_in_tree() | 
|---|
|  | 911 | new_inode->next     = NULL;                 // set by _add_inode_in_tree() | 
|---|
|  | 912 | new_inode->child    = NULL;                 // set by _add_inode_in_tree() | 
|---|
|  | 913 | new_inode->cluster  = cluster; | 
|---|
|  | 914 | new_inode->size     = size; | 
|---|
|  | 915 | new_inode->cache    = NULL; | 
|---|
|  | 916 | new_inode->levels   = 0; | 
|---|
|  | 917 | new_inode->count    = count; | 
|---|
|  | 918 | new_inode->is_dir   = (is_dir != 0); | 
|---|
|  | 919 | new_inode->dentry   = dentry; | 
|---|
| [291] | 920 |  | 
|---|
| [587] | 921 | _strcpy( new_inode->name , name ); | 
|---|
|  | 922 |  | 
|---|
|  | 923 | if ( cache_allocate ) | 
|---|
| [291] | 924 | { | 
|---|
| [639] | 925 | new_inode->cache    = _allocate_one_cache_node( NULL ); | 
|---|
| [587] | 926 | new_inode->levels   = _get_levels_from_size( size ); | 
|---|
| [291] | 927 | } | 
|---|
|  | 928 |  | 
|---|
| [587] | 929 | return new_inode; | 
|---|
|  | 930 | }   // end _allocate_one_inode() | 
|---|
| [291] | 931 |  | 
|---|
| [587] | 932 |  | 
|---|
|  | 933 |  | 
|---|
|  | 934 |  | 
|---|
|  | 935 | //////////////////////////////////////////////////// | 
|---|
|  | 936 | static void _add_inode_in_tree( fat_inode_t*  child, | 
|---|
|  | 937 | fat_inode_t*  parent ) | 
|---|
|  | 938 | { | 
|---|
|  | 939 | child->parent = parent; | 
|---|
|  | 940 | child->next   = parent->child; | 
|---|
|  | 941 | parent->child = child; | 
|---|
|  | 942 | }   // end _add_inode-in_tree() | 
|---|
|  | 943 |  | 
|---|
|  | 944 |  | 
|---|
|  | 945 |  | 
|---|
|  | 946 |  | 
|---|
|  | 947 | ////////////////////////////////////////////////////////// | 
|---|
|  | 948 | static void _remove_inode_from_tree( fat_inode_t*  inode ) | 
|---|
|  | 949 | { | 
|---|
|  | 950 | fat_inode_t*  current; | 
|---|
|  | 951 | fat_inode_t*  prev = inode->parent->child; | 
|---|
|  | 952 |  | 
|---|
|  | 953 | if ( inode == prev )  // removed inode is first in its linked list | 
|---|
| [291] | 954 | { | 
|---|
| [587] | 955 | inode->parent->child = inode->next; | 
|---|
| [291] | 956 | } | 
|---|
| [587] | 957 | else                  // removed inode is not the first | 
|---|
| [291] | 958 | { | 
|---|
| [587] | 959 | for( current = prev->next ; current ; current = current->next ) | 
|---|
| [291] | 960 | { | 
|---|
| [587] | 961 | if ( current == inode ) | 
|---|
| [291] | 962 | { | 
|---|
| [587] | 963 | prev->next = current->next; | 
|---|
| [291] | 964 | } | 
|---|
| [587] | 965 | prev = current; | 
|---|
|  | 966 | } | 
|---|
|  | 967 | } | 
|---|
|  | 968 | }  // end _delete_one_inode() | 
|---|
| [291] | 969 |  | 
|---|
|  | 970 |  | 
|---|
| [587] | 971 |  | 
|---|
|  | 972 |  | 
|---|
|  | 973 | ////////////////////////////////////////////////////////////////////// | 
|---|
|  | 974 | static unsigned int _get_buffer_from_cache( fat_inode_t*        inode, | 
|---|
|  | 975 | unsigned int        cluster_id, | 
|---|
|  | 976 | fat_cache_desc_t**  desc ) | 
|---|
|  | 977 | { | 
|---|
|  | 978 | // get cache pointer and levels | 
|---|
|  | 979 | fat_cache_node_t*   node;         // pointer on a 64-tree node | 
|---|
|  | 980 | unsigned int        level;        // cache level | 
|---|
|  | 981 |  | 
|---|
|  | 982 | if ( inode == NULL )   // searched cache is the Fat-Cache | 
|---|
|  | 983 | { | 
|---|
|  | 984 | node   = _fat.fat_cache_root; | 
|---|
|  | 985 | level  = _fat.fat_cache_levels; | 
|---|
|  | 986 |  | 
|---|
|  | 987 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 988 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 989 | _printf("\n[DEBUG FAT] _get_buffer_from_cache(): enters in FAT-Cache" | 
|---|
| [587] | 990 | " for cluster_id = %d\n", cluster_id ); | 
|---|
|  | 991 | #endif | 
|---|
|  | 992 | } | 
|---|
|  | 993 | else                   // searched cache is a File-Cache | 
|---|
|  | 994 | { | 
|---|
| [665] | 995 | // add cache levels if needed | 
|---|
|  | 996 | while ( _get_levels_from_size( (cluster_id + 1) * 4096 ) > inode->levels ) | 
|---|
|  | 997 | { | 
|---|
|  | 998 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 999 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
|  | 1000 | _printf("\n[DEBUG FAT] _get_buffer_from_cache(): adding a File-Cache level\n" ); | 
|---|
|  | 1001 | #endif | 
|---|
|  | 1002 |  | 
|---|
|  | 1003 | inode->cache = _allocate_one_cache_node( inode->cache ); | 
|---|
|  | 1004 | inode->levels++; | 
|---|
|  | 1005 | } | 
|---|
|  | 1006 |  | 
|---|
| [587] | 1007 | node   = inode->cache; | 
|---|
|  | 1008 | level  = inode->levels; | 
|---|
|  | 1009 |  | 
|---|
|  | 1010 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1011 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1012 | _printf("\n[DEBUG FAT] _get_buffer_from_cache(): enters in File-Cache <%s>" | 
|---|
| [587] | 1013 | " for cluster_id = %d\n", inode->name , cluster_id ); | 
|---|
|  | 1014 | #endif | 
|---|
|  | 1015 | } | 
|---|
|  | 1016 |  | 
|---|
|  | 1017 | // search the 64-tree cache from top to bottom | 
|---|
|  | 1018 | while ( level ) | 
|---|
|  | 1019 | { | 
|---|
|  | 1020 | // compute child index at each level | 
|---|
|  | 1021 | unsigned int index = (cluster_id >> (6*(level-1))) & 0x3F; | 
|---|
|  | 1022 |  | 
|---|
|  | 1023 | if ( level == 1 )        // last level => children are cluster descriptors | 
|---|
| [291] | 1024 | { | 
|---|
| [587] | 1025 | fat_cache_desc_t* pdesc = (fat_cache_desc_t*)node->children[index]; | 
|---|
| [291] | 1026 |  | 
|---|
| [587] | 1027 | if ( pdesc == NULL )      // miss | 
|---|
| [291] | 1028 | { | 
|---|
| [587] | 1029 | // get missing cluster index lba | 
|---|
|  | 1030 | unsigned int lba; | 
|---|
|  | 1031 | unsigned int next; | 
|---|
|  | 1032 | unsigned int current = inode->cluster; | 
|---|
|  | 1033 | unsigned int count   = cluster_id; | 
|---|
|  | 1034 |  | 
|---|
|  | 1035 | if ( inode == NULL )      // searched cache is the Fat-Cache | 
|---|
| [291] | 1036 | { | 
|---|
| [587] | 1037 |  | 
|---|
|  | 1038 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1039 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1040 | _printf("\n[DEBUG FAT] _get_buffer_from_cache(): miss in FAT-Cache for cluster_id %d\n", | 
|---|
| [587] | 1041 | cluster_id ); | 
|---|
|  | 1042 | #endif | 
|---|
|  | 1043 | lba = _fat.fat_lba + (cluster_id << 3); | 
|---|
| [291] | 1044 | } | 
|---|
| [587] | 1045 | else                      // searched cache is a File-Cache | 
|---|
|  | 1046 | { | 
|---|
|  | 1047 |  | 
|---|
|  | 1048 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1049 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1050 | _printf("\n[DEBUG FAT] _get_buffer_from_cache(): miss in File-Cache <%s> " | 
|---|
| [587] | 1051 | "for cluster_id %d\n", inode->name, cluster_id ); | 
|---|
|  | 1052 | #endif | 
|---|
|  | 1053 | while ( count ) | 
|---|
|  | 1054 | { | 
|---|
|  | 1055 | if ( _get_fat_entry( current , &next ) ) return 1; | 
|---|
|  | 1056 | current = next; | 
|---|
|  | 1057 | count--; | 
|---|
|  | 1058 | } | 
|---|
|  | 1059 | lba = _cluster_to_lba( current ); | 
|---|
|  | 1060 | } | 
|---|
|  | 1061 |  | 
|---|
| [653] | 1062 | // allocate 4K buffer | 
|---|
|  | 1063 | void* buf = _malloc( 4096 ); | 
|---|
|  | 1064 |  | 
|---|
| [587] | 1065 | // load one cluster (8 blocks) from block device | 
|---|
|  | 1066 | if ( _fat_ioc_access( 1,         // descheduling | 
|---|
|  | 1067 | 1,         // to memory | 
|---|
|  | 1068 | lba, | 
|---|
|  | 1069 | (unsigned int)buf, | 
|---|
|  | 1070 | 8 ) ) | 
|---|
|  | 1071 | { | 
|---|
| [653] | 1072 | _free( buf ); | 
|---|
| [664] | 1073 | _printf("\n[FAT ERROR] _get_buffer_from_cache()" | 
|---|
|  | 1074 | ": cannot access block device for lba = %x\n", lba ); | 
|---|
| [587] | 1075 | return 1; | 
|---|
|  | 1076 | } | 
|---|
|  | 1077 |  | 
|---|
| [653] | 1078 | // allocate buffer descriptor | 
|---|
|  | 1079 | pdesc          = _malloc( sizeof(fat_cache_desc_t) ); | 
|---|
| [587] | 1080 | pdesc->lba     = lba; | 
|---|
|  | 1081 | pdesc->buffer  = buf; | 
|---|
|  | 1082 | pdesc->dirty   = 0; | 
|---|
| [653] | 1083 | node->children[index] = pdesc; | 
|---|
| [587] | 1084 |  | 
|---|
|  | 1085 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1086 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1087 | _printf("\n[DEBUG FAT] _get_buffer_from_cache(): buffer loaded from device" | 
|---|
| [587] | 1088 | " at vaddr = %x\n", (unsigned int)buf ); | 
|---|
|  | 1089 | #endif | 
|---|
| [291] | 1090 | } | 
|---|
| [587] | 1091 |  | 
|---|
|  | 1092 | // return pdesc pointer | 
|---|
|  | 1093 | *desc = pdesc; | 
|---|
|  | 1094 |  | 
|---|
|  | 1095 | // prepare next iteration | 
|---|
|  | 1096 | level--; | 
|---|
|  | 1097 | } | 
|---|
|  | 1098 | else                      // not last level => children are 64-tree nodes | 
|---|
|  | 1099 | { | 
|---|
|  | 1100 | fat_cache_node_t* child = (fat_cache_node_t*)node->children[index]; | 
|---|
|  | 1101 | if ( child == NULL )  // miss | 
|---|
| [291] | 1102 | { | 
|---|
| [587] | 1103 | // allocate a cache node if miss | 
|---|
| [650] | 1104 | child = _allocate_one_cache_node( NULL ); | 
|---|
| [587] | 1105 | node->children[index] = child; | 
|---|
| [291] | 1106 | } | 
|---|
| [587] | 1107 |  | 
|---|
|  | 1108 | // prepare next iteration | 
|---|
|  | 1109 | node = child; | 
|---|
|  | 1110 | level--; | 
|---|
| [291] | 1111 | } | 
|---|
| [587] | 1112 | } // end while | 
|---|
| [291] | 1113 |  | 
|---|
| [530] | 1114 | return 0; | 
|---|
| [587] | 1115 | }  // end _get_buffer_from_cache() | 
|---|
| [530] | 1116 |  | 
|---|
|  | 1117 |  | 
|---|
| [291] | 1118 |  | 
|---|
|  | 1119 |  | 
|---|
| [587] | 1120 | ///////////////////////////////////// | 
|---|
|  | 1121 | static unsigned int _update_fs_info() | 
|---|
|  | 1122 | { | 
|---|
|  | 1123 | // update buffer if miss | 
|---|
|  | 1124 | if ( _fat.fs_info_lba != _fat.block_buffer_lba ) | 
|---|
| [291] | 1125 | { | 
|---|
| [587] | 1126 | if ( _fat_ioc_access( 1,                 // descheduling | 
|---|
| [530] | 1127 | 1,                 // read | 
|---|
| [587] | 1128 | _fat.fs_info_lba, | 
|---|
|  | 1129 | (unsigned int)_fat.block_buffer, | 
|---|
|  | 1130 | 1 ) )              // one block | 
|---|
| [291] | 1131 | { | 
|---|
| [664] | 1132 | _printf("\n[FAT_ERROR] _update_fs_info(): cannot read block\n"); | 
|---|
| [291] | 1133 | return 1; | 
|---|
|  | 1134 | } | 
|---|
| [587] | 1135 | _fat.block_buffer_lba = _fat.fs_info_lba; | 
|---|
| [291] | 1136 | } | 
|---|
|  | 1137 |  | 
|---|
| [587] | 1138 | // update FAT buffer | 
|---|
|  | 1139 | unsigned int* ptr; | 
|---|
|  | 1140 | ptr  = (unsigned int*)(_fat.block_buffer + get_offset(FS_FREE_CLUSTERS) ); | 
|---|
|  | 1141 | *ptr = _fat.free_clusters_number; | 
|---|
|  | 1142 |  | 
|---|
|  | 1143 | ptr  = (unsigned int*)(_fat.block_buffer + get_offset(FS_FREE_CLUSTER_HINT) ); | 
|---|
|  | 1144 | *ptr = _fat.first_free_cluster; | 
|---|
| [530] | 1145 |  | 
|---|
|  | 1146 | // write bloc to FAT | 
|---|
| [587] | 1147 | if ( _fat_ioc_access( 1,                // descheduling | 
|---|
| [530] | 1148 | 0,                // write | 
|---|
| [587] | 1149 | _fat.fs_info_lba, | 
|---|
|  | 1150 | (unsigned int)_fat.block_buffer, | 
|---|
|  | 1151 | 1 ) )             // one block | 
|---|
| [530] | 1152 | { | 
|---|
| [664] | 1153 | _printf("\n[FAT_ERROR] _update_fs_info(): cannot write block\n"); | 
|---|
| [530] | 1154 | return 1; | 
|---|
|  | 1155 | } | 
|---|
|  | 1156 |  | 
|---|
| [587] | 1157 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1158 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1159 | _printf("\n[DEBUG FAT] _update_fs_info(): nb_free = %x / first_free = %x\n", | 
|---|
| [587] | 1160 | _fat.free_clusters_number , _fat.first_free_cluster ); | 
|---|
|  | 1161 | #endif | 
|---|
|  | 1162 |  | 
|---|
| [530] | 1163 | return 0; | 
|---|
|  | 1164 | }  // end _update_fs_info() | 
|---|
|  | 1165 |  | 
|---|
| [587] | 1166 |  | 
|---|
|  | 1167 |  | 
|---|
|  | 1168 | ///////////////////////////////////////////////////////////////// | 
|---|
|  | 1169 | static inline unsigned int _get_fat_entry( unsigned int  cluster, | 
|---|
|  | 1170 | unsigned int* value ) | 
|---|
| [291] | 1171 | { | 
|---|
| [587] | 1172 | // compute cluster_id & entry_id in FAT | 
|---|
|  | 1173 | // a FAT cluster is an array of 1024 unsigned int entries | 
|---|
|  | 1174 | unsigned int       cluster_id = cluster >> 10; | 
|---|
|  | 1175 | unsigned int       entry_id   = cluster & 0x3FF; | 
|---|
| [291] | 1176 |  | 
|---|
| [587] | 1177 | // get pointer on the relevant cluster descriptor in FAT cache | 
|---|
|  | 1178 | fat_cache_desc_t*  pdesc; | 
|---|
|  | 1179 | unsigned int*      buffer; | 
|---|
|  | 1180 | if ( _get_buffer_from_cache( NULL,               // Fat-Cache | 
|---|
|  | 1181 | cluster_id, | 
|---|
|  | 1182 | &pdesc ) ) return 1; | 
|---|
|  | 1183 |  | 
|---|
|  | 1184 | // get value from FAT slot | 
|---|
|  | 1185 | buffer = (unsigned int*)pdesc->buffer; | 
|---|
|  | 1186 | *value = buffer[entry_id]; | 
|---|
|  | 1187 |  | 
|---|
|  | 1188 | return 0; | 
|---|
|  | 1189 | }  // end _get_fat_entry() | 
|---|
|  | 1190 |  | 
|---|
|  | 1191 |  | 
|---|
|  | 1192 |  | 
|---|
|  | 1193 | //////////////////////////////////////////////////////////////// | 
|---|
|  | 1194 | static inline unsigned int _set_fat_entry( unsigned int cluster, | 
|---|
|  | 1195 | unsigned int value  ) | 
|---|
|  | 1196 | { | 
|---|
|  | 1197 | // compute cluster_id & entry_id in FAT | 
|---|
|  | 1198 | // a FAT cluster is an array of 1024 unsigned int entries | 
|---|
|  | 1199 | unsigned int cluster_id = cluster >> 10; | 
|---|
|  | 1200 | unsigned int entry_id   = cluster & 0x3FF; | 
|---|
|  | 1201 |  | 
|---|
|  | 1202 | // get pointer on the relevant cluster descriptor in FAT cache | 
|---|
|  | 1203 | fat_cache_desc_t*  pdesc; | 
|---|
|  | 1204 | unsigned int*      buffer; | 
|---|
|  | 1205 | if ( _get_buffer_from_cache( NULL,               // Fat-Cache | 
|---|
|  | 1206 | cluster_id, | 
|---|
|  | 1207 | &pdesc ) )  return 1; | 
|---|
|  | 1208 |  | 
|---|
|  | 1209 | // set value into FAT slot | 
|---|
|  | 1210 | buffer           = (unsigned int*)pdesc->buffer; | 
|---|
|  | 1211 | buffer[entry_id] = value; | 
|---|
|  | 1212 | pdesc->dirty     = 1; | 
|---|
|  | 1213 |  | 
|---|
|  | 1214 | return 0; | 
|---|
|  | 1215 | } // end _set_fat_entry() | 
|---|
|  | 1216 |  | 
|---|
|  | 1217 |  | 
|---|
|  | 1218 |  | 
|---|
|  | 1219 | ////////////////////////////////////////////////////// | 
|---|
|  | 1220 | static void _allocate_one_buffer( fat_inode_t*  inode, | 
|---|
|  | 1221 | unsigned int  cluster_id, | 
|---|
|  | 1222 | unsigned int  cluster ) | 
|---|
|  | 1223 | { | 
|---|
| [639] | 1224 | // add cache levels if needed | 
|---|
|  | 1225 | while ( _get_levels_from_size( (cluster_id + 1) * 4096 ) > inode->levels ) | 
|---|
|  | 1226 | { | 
|---|
|  | 1227 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1228 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1229 | _printf("\n[DEBUG FAT] _allocate_one_buffer(): adding a cache level\n" ); | 
|---|
| [639] | 1230 | #endif | 
|---|
|  | 1231 |  | 
|---|
|  | 1232 | inode->cache = _allocate_one_cache_node( inode->cache ); | 
|---|
|  | 1233 | inode->levels++; | 
|---|
|  | 1234 | } | 
|---|
|  | 1235 |  | 
|---|
| [587] | 1236 | // search the 64-tree cache from top to bottom | 
|---|
|  | 1237 | fat_cache_node_t*  node   = inode->cache; | 
|---|
| [653] | 1238 | unsigned int       level; | 
|---|
| [587] | 1239 |  | 
|---|
| [653] | 1240 | for ( level = inode->levels; level != 0; level-- ) | 
|---|
| [587] | 1241 | { | 
|---|
|  | 1242 | // compute child index | 
|---|
|  | 1243 | unsigned int index = (cluster_id >> (6*(level-1))) & 0x3F; | 
|---|
|  | 1244 |  | 
|---|
|  | 1245 | if ( level == 1 )        // last level => children are cluster descriptors | 
|---|
|  | 1246 | { | 
|---|
|  | 1247 | fat_cache_desc_t* pdesc = (fat_cache_desc_t*)node->children[index]; | 
|---|
|  | 1248 | if ( pdesc != NULL )      // slot not empty!!! | 
|---|
|  | 1249 | { | 
|---|
| [664] | 1250 | _printf("\n[FAT ERROR] allocate_one buffer(): slot not empty " | 
|---|
| [587] | 1251 | "in File-Cache <%s> / cluster_id = %d\n", inode->name , cluster_id ); | 
|---|
|  | 1252 | _exit(); | 
|---|
|  | 1253 | } | 
|---|
|  | 1254 |  | 
|---|
|  | 1255 | #if (GIET_DEBUG_FAT & 1) | 
|---|
| [569] | 1256 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1257 | _printf("\n[DEBUG FAT] _allocate_one_buffer(): buffer allocated to <%s> for cluster_id %d\n", | 
|---|
| [587] | 1258 | inode->name, cluster_id ); | 
|---|
| [291] | 1259 | #endif | 
|---|
|  | 1260 |  | 
|---|
| [653] | 1261 | // allocate buffer descriptor | 
|---|
|  | 1262 | pdesc = _malloc( sizeof(fat_cache_desc_t) ); | 
|---|
| [587] | 1263 | pdesc->lba     = _cluster_to_lba( cluster ); | 
|---|
| [653] | 1264 | pdesc->buffer  = _malloc( 4096 ); | 
|---|
| [587] | 1265 | pdesc->dirty   = 1; | 
|---|
| [653] | 1266 | node->children[index] = pdesc; | 
|---|
| [587] | 1267 | } | 
|---|
|  | 1268 | else                      // not last level => children are 64-tree nodes | 
|---|
| [291] | 1269 | { | 
|---|
| [587] | 1270 | fat_cache_node_t* child = (fat_cache_node_t*)node->children[index]; | 
|---|
|  | 1271 | if ( child == NULL )  // miss | 
|---|
|  | 1272 | { | 
|---|
|  | 1273 | // allocate a cache node if miss | 
|---|
| [650] | 1274 | child = _allocate_one_cache_node( NULL ); | 
|---|
| [587] | 1275 | node->children[index] = child; | 
|---|
|  | 1276 | } | 
|---|
|  | 1277 |  | 
|---|
|  | 1278 | // prepare next iteration | 
|---|
|  | 1279 | node  = child; | 
|---|
| [291] | 1280 | } | 
|---|
| [653] | 1281 | } // end for | 
|---|
| [587] | 1282 | } // end _allocate_one_buffer | 
|---|
| [530] | 1283 |  | 
|---|
|  | 1284 |  | 
|---|
| [587] | 1285 |  | 
|---|
|  | 1286 |  | 
|---|
|  | 1287 | /////////////////////////////////////////////////////////////////// | 
|---|
|  | 1288 | static unsigned int _allocate_one_cluster( unsigned int*  cluster ) | 
|---|
|  | 1289 | { | 
|---|
|  | 1290 | unsigned int nb_free = _fat.free_clusters_number; | 
|---|
|  | 1291 | unsigned int free    = _fat.first_free_cluster; | 
|---|
|  | 1292 |  | 
|---|
|  | 1293 | // scan FAT to get next free cluster index | 
|---|
|  | 1294 | unsigned int current = free; | 
|---|
|  | 1295 | unsigned int found   = 0; | 
|---|
|  | 1296 | unsigned int max     = (_fat.data_sectors >> 3); | 
|---|
|  | 1297 | unsigned int value; | 
|---|
|  | 1298 | do | 
|---|
| [530] | 1299 | { | 
|---|
| [587] | 1300 | // increment current | 
|---|
|  | 1301 | current++; | 
|---|
|  | 1302 |  | 
|---|
|  | 1303 | // get FAT entry indexed by current | 
|---|
|  | 1304 | if ( _get_fat_entry( current , &value ) ) return 1; | 
|---|
|  | 1305 | // test if free | 
|---|
|  | 1306 | if ( value == FREE_CLUSTER ) found = 1; | 
|---|
|  | 1307 | } | 
|---|
|  | 1308 | while ( (current < max) && (found == 0) ); | 
|---|
|  | 1309 |  | 
|---|
|  | 1310 | // check found | 
|---|
|  | 1311 | if ( found == 0 ) | 
|---|
|  | 1312 | { | 
|---|
| [664] | 1313 | _printf("\n[FAT_ERROR] _allocate_one_cluster(): unconsistent FAT state"); | 
|---|
| [530] | 1314 | return 1; | 
|---|
|  | 1315 | } | 
|---|
|  | 1316 |  | 
|---|
| [587] | 1317 | // update allocated FAT slot | 
|---|
|  | 1318 | if ( _set_fat_entry( free , END_OF_CHAIN_CLUSTER_MAX ) ) return 1; | 
|---|
|  | 1319 |  | 
|---|
|  | 1320 | // update FAT descriptor global variables | 
|---|
|  | 1321 | _fat.free_clusters_number = nb_free - 1; | 
|---|
|  | 1322 | _fat.first_free_cluster   = current; | 
|---|
|  | 1323 |  | 
|---|
|  | 1324 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1325 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1326 | _printf("\n[DEBUG FAT] _allocate_one_cluster(): cluster = %x / first_free = %x\n", | 
|---|
| [587] | 1327 | free , current ); | 
|---|
|  | 1328 | #endif | 
|---|
|  | 1329 |  | 
|---|
|  | 1330 | // returns free cluster index | 
|---|
|  | 1331 | *cluster = free; | 
|---|
| [530] | 1332 | return 0; | 
|---|
| [291] | 1333 |  | 
|---|
| [587] | 1334 | }  // end _allocate_one_cluster() | 
|---|
|  | 1335 |  | 
|---|
|  | 1336 |  | 
|---|
|  | 1337 |  | 
|---|
|  | 1338 |  | 
|---|
|  | 1339 | ////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 1340 | static unsigned int _update_device_from_cache( unsigned int        levels, | 
|---|
|  | 1341 | fat_cache_node_t*   root, | 
|---|
|  | 1342 | char*               string ) | 
|---|
| [291] | 1343 | { | 
|---|
| [587] | 1344 | unsigned int index; | 
|---|
|  | 1345 | unsigned int ret = 0; | 
|---|
| [291] | 1346 |  | 
|---|
| [587] | 1347 | if ( levels == 1 )  // last level => children are cluster descriptors | 
|---|
| [291] | 1348 | { | 
|---|
| [587] | 1349 | for( index = 0 ; index < 64 ; index++ ) | 
|---|
|  | 1350 | { | 
|---|
|  | 1351 | fat_cache_desc_t* pdesc = root->children[index]; | 
|---|
|  | 1352 | if ( pdesc != NULL ) | 
|---|
|  | 1353 | { | 
|---|
|  | 1354 | // update cluster on device if dirty | 
|---|
|  | 1355 | if ( pdesc->dirty ) | 
|---|
|  | 1356 | { | 
|---|
|  | 1357 | if ( _fat_ioc_access( 1,           // descheduling | 
|---|
|  | 1358 | 0,           // to block device | 
|---|
|  | 1359 | pdesc->lba, | 
|---|
|  | 1360 | (unsigned int)pdesc->buffer, | 
|---|
|  | 1361 | 8 ) ) | 
|---|
|  | 1362 | { | 
|---|
| [664] | 1363 | _printf("\n[FAT_ERROR] _update_device from_cache(): " | 
|---|
| [587] | 1364 | " cannot access lba = %x\n", pdesc->lba ); | 
|---|
|  | 1365 | ret = 1; | 
|---|
|  | 1366 | } | 
|---|
|  | 1367 | else | 
|---|
|  | 1368 | { | 
|---|
|  | 1369 | pdesc->dirty = 0; | 
|---|
|  | 1370 |  | 
|---|
|  | 1371 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1372 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1373 | _printf("\n[DEBUG FAT] _update_device_from_cache(): cluster_id = %d for <%s>\n", | 
|---|
| [587] | 1374 | index , string ); | 
|---|
|  | 1375 | #endif | 
|---|
|  | 1376 |  | 
|---|
|  | 1377 | } | 
|---|
|  | 1378 | } | 
|---|
|  | 1379 | } | 
|---|
|  | 1380 | } | 
|---|
| [291] | 1381 | } | 
|---|
| [587] | 1382 | else               // not the last level = recursive call on each children | 
|---|
| [291] | 1383 | { | 
|---|
| [587] | 1384 | for( index = 0 ; index < 64 ; index++ ) | 
|---|
|  | 1385 | { | 
|---|
|  | 1386 | fat_cache_node_t* pnode = root->children[index]; | 
|---|
|  | 1387 | if ( pnode != NULL ) | 
|---|
|  | 1388 | { | 
|---|
|  | 1389 | if ( _update_device_from_cache( levels - 1, | 
|---|
|  | 1390 | root->children[index], | 
|---|
|  | 1391 | string ) ) ret = 1; | 
|---|
|  | 1392 | } | 
|---|
|  | 1393 | } | 
|---|
| [291] | 1394 | } | 
|---|
| [587] | 1395 | return ret; | 
|---|
|  | 1396 | }  // end _update_device_from_cache() | 
|---|
| [291] | 1397 |  | 
|---|
|  | 1398 |  | 
|---|
| [587] | 1399 |  | 
|---|
|  | 1400 | /////////////////////////////////////////////////////////////////// | 
|---|
| [651] | 1401 | static void _release_cache_memory( fat_cache_node_t*  root, | 
|---|
|  | 1402 | unsigned int       levels ) | 
|---|
| [587] | 1403 | { | 
|---|
| [651] | 1404 | unsigned int i; | 
|---|
| [587] | 1405 |  | 
|---|
|  | 1406 | if ( levels == 1 )  // last level => children are cluster descriptors | 
|---|
| [503] | 1407 | { | 
|---|
| [651] | 1408 | for( i = 0 ; i < 64 ; i++ ) | 
|---|
| [587] | 1409 | { | 
|---|
| [651] | 1410 | fat_cache_desc_t* pdesc = root->children[i]; | 
|---|
| [587] | 1411 |  | 
|---|
|  | 1412 | if ( pdesc != NULL ) | 
|---|
|  | 1413 | { | 
|---|
|  | 1414 | // check dirty | 
|---|
|  | 1415 | if ( pdesc->dirty ) | 
|---|
| [664] | 1416 | _printf("\n[FAT ERROR] _release_cache_memory(): dirty cluster\n"); | 
|---|
| [651] | 1417 |  | 
|---|
|  | 1418 | _free( pdesc->buffer ); | 
|---|
|  | 1419 | _free( pdesc ); | 
|---|
|  | 1420 | root->children[i] = NULL; | 
|---|
| [587] | 1421 | } | 
|---|
|  | 1422 | } | 
|---|
| [503] | 1423 | } | 
|---|
| [587] | 1424 | else               // not the last level = recursive call on each children | 
|---|
|  | 1425 | { | 
|---|
| [651] | 1426 | for( i = 0 ; i < 64 ; i++ ) | 
|---|
| [587] | 1427 | { | 
|---|
| [651] | 1428 | fat_cache_node_t* cnode = root->children[i]; | 
|---|
| [291] | 1429 |  | 
|---|
| [651] | 1430 | if ( cnode != NULL ) | 
|---|
|  | 1431 | { | 
|---|
|  | 1432 | _release_cache_memory( cnode, levels - 1 ); | 
|---|
|  | 1433 | _free( cnode ); | 
|---|
|  | 1434 | root->children[i] = NULL; | 
|---|
|  | 1435 | } | 
|---|
| [587] | 1436 | } | 
|---|
|  | 1437 | } | 
|---|
|  | 1438 | }  // end _release_cache_memory() | 
|---|
|  | 1439 |  | 
|---|
|  | 1440 |  | 
|---|
|  | 1441 |  | 
|---|
|  | 1442 |  | 
|---|
|  | 1443 |  | 
|---|
|  | 1444 | ///////////////////////////////////////////////////////////// | 
|---|
|  | 1445 | static unsigned int _clusters_allocate( fat_inode_t* inode, | 
|---|
|  | 1446 | unsigned int nb_current_clusters, | 
|---|
|  | 1447 | unsigned int nb_required_clusters ) | 
|---|
|  | 1448 | { | 
|---|
|  | 1449 | // Check if FAT contains enough free clusters | 
|---|
|  | 1450 | if ( nb_required_clusters > _fat.free_clusters_number ) | 
|---|
| [291] | 1451 | { | 
|---|
| [664] | 1452 | _printf("\n[FAT ERROR] _clusters_allocate(): required_clusters = %d" | 
|---|
| [587] | 1453 | " / free_clusters = %d\n", nb_required_clusters , _fat.free_clusters_number ); | 
|---|
|  | 1454 | return 1; | 
|---|
|  | 1455 | } | 
|---|
| [291] | 1456 |  | 
|---|
| [587] | 1457 | #if (GIET_DEBUG_FAT & 1) | 
|---|
| [569] | 1458 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1459 | _printf("\n[DEBUG FAT] _clusters_allocate(): enters for <%s> / nb_current_clusters = %d " | 
|---|
| [587] | 1460 | "/ nb_required_clusters = %d\n", | 
|---|
|  | 1461 | inode->name , nb_current_clusters , nb_required_clusters ); | 
|---|
| [291] | 1462 | #endif | 
|---|
| [587] | 1463 |  | 
|---|
|  | 1464 | // compute last allocated cluster index when (nb_current_clusters > 0) | 
|---|
|  | 1465 | unsigned int current = inode->cluster; | 
|---|
|  | 1466 | unsigned int next; | 
|---|
|  | 1467 | unsigned int last; | 
|---|
|  | 1468 | if ( nb_current_clusters )   // clusters allocated => search last | 
|---|
|  | 1469 | { | 
|---|
|  | 1470 | while ( current < END_OF_CHAIN_CLUSTER_MIN ) | 
|---|
| [291] | 1471 | { | 
|---|
| [587] | 1472 | // get next cluster | 
|---|
|  | 1473 | if ( _get_fat_entry( current , &next ) )  return 1; | 
|---|
|  | 1474 | last    = current; | 
|---|
|  | 1475 | current = next; | 
|---|
| [291] | 1476 | } | 
|---|
| [587] | 1477 | } | 
|---|
| [291] | 1478 |  | 
|---|
| [587] | 1479 | // Loop on the new clusters to be allocated | 
|---|
|  | 1480 | // if (nb_current_clusters == 0) the first new cluster index must | 
|---|
|  | 1481 | //                               be written in inode->cluster field | 
|---|
|  | 1482 | // if (nb_current_clusters >  0) the first new cluster index must | 
|---|
|  | 1483 | //                               be written in FAT[last] | 
|---|
|  | 1484 | unsigned int      cluster_id; | 
|---|
|  | 1485 | unsigned int      new; | 
|---|
|  | 1486 | for ( cluster_id = nb_current_clusters ; | 
|---|
|  | 1487 | cluster_id < (nb_current_clusters + nb_required_clusters) ; | 
|---|
|  | 1488 | cluster_id ++ ) | 
|---|
|  | 1489 | { | 
|---|
|  | 1490 | // allocate one cluster on block device | 
|---|
|  | 1491 | if ( _allocate_one_cluster( &new ) ) return 1; | 
|---|
| [291] | 1492 |  | 
|---|
| [587] | 1493 | // allocate one 4K buffer to File-Cache | 
|---|
|  | 1494 | _allocate_one_buffer( inode, | 
|---|
|  | 1495 | cluster_id, | 
|---|
|  | 1496 | new ); | 
|---|
|  | 1497 |  | 
|---|
|  | 1498 | if ( cluster_id == 0 )  // update inode | 
|---|
| [291] | 1499 | { | 
|---|
| [587] | 1500 | inode->cluster = new; | 
|---|
| [291] | 1501 | } | 
|---|
| [587] | 1502 | else                    // update FAT | 
|---|
| [291] | 1503 | { | 
|---|
| [587] | 1504 | if ( _set_fat_entry( last , new ) ) return 1; | 
|---|
| [291] | 1505 | } | 
|---|
|  | 1506 |  | 
|---|
| [587] | 1507 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1508 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1509 | _printf("\n[DEBUG FAT] _clusters_allocate(): done for cluster_id = %d / cluster = %x\n", | 
|---|
| [587] | 1510 | cluster_id , new ); | 
|---|
|  | 1511 | #endif | 
|---|
| [291] | 1512 |  | 
|---|
| [587] | 1513 | // update loop variables | 
|---|
|  | 1514 | last = new; | 
|---|
|  | 1515 |  | 
|---|
|  | 1516 | } // end for loop | 
|---|
|  | 1517 |  | 
|---|
|  | 1518 | // update FAT : last slot should contain END_OF_CHAIN_CLUSTER_MAX | 
|---|
|  | 1519 | if ( _set_fat_entry( last , END_OF_CHAIN_CLUSTER_MAX ) )  return 1; | 
|---|
|  | 1520 |  | 
|---|
|  | 1521 | // update the FAT on block device | 
|---|
|  | 1522 | if ( _update_device_from_cache( _fat.fat_cache_levels, | 
|---|
|  | 1523 | _fat.fat_cache_root, | 
|---|
|  | 1524 | "FAT" ) )              return 1; | 
|---|
|  | 1525 | return 0; | 
|---|
|  | 1526 | }  // end _clusters_allocate() | 
|---|
|  | 1527 |  | 
|---|
|  | 1528 |  | 
|---|
|  | 1529 |  | 
|---|
|  | 1530 | ////////////////////////////////////////////////////////////// | 
|---|
|  | 1531 | static unsigned int _clusters_release( unsigned int cluster ) | 
|---|
|  | 1532 | { | 
|---|
|  | 1533 | // scan the FAT | 
|---|
|  | 1534 | unsigned int current = cluster; | 
|---|
|  | 1535 | unsigned int next; | 
|---|
|  | 1536 | do | 
|---|
|  | 1537 | { | 
|---|
|  | 1538 | // get next_cluster index | 
|---|
|  | 1539 | if ( _get_fat_entry( current , &next ) )  return 1; | 
|---|
|  | 1540 |  | 
|---|
|  | 1541 | // release current_cluster | 
|---|
|  | 1542 | if ( _set_fat_entry( current , FREE_CLUSTER ) )   return 1; | 
|---|
|  | 1543 |  | 
|---|
|  | 1544 | // update first_free_cluster and free_clusters_number in FAT descriptor | 
|---|
|  | 1545 | _fat.free_clusters_number = _fat.free_clusters_number + 1; | 
|---|
|  | 1546 | if ( _fat.first_free_cluster > current ) _fat.first_free_cluster = current; | 
|---|
|  | 1547 |  | 
|---|
|  | 1548 | // update loop variable | 
|---|
|  | 1549 | current = next; | 
|---|
| [291] | 1550 | } | 
|---|
| [587] | 1551 | while ( next < END_OF_CHAIN_CLUSTER_MIN ); | 
|---|
| [291] | 1552 |  | 
|---|
| [587] | 1553 | // update the FAT on block device | 
|---|
|  | 1554 | if ( _update_device_from_cache( _fat.fat_cache_levels, | 
|---|
|  | 1555 | _fat.fat_cache_root, | 
|---|
|  | 1556 | "FAT" ) )                return 1; | 
|---|
| [291] | 1557 | return 0; | 
|---|
| [587] | 1558 | }  // end _clusters_release() | 
|---|
| [291] | 1559 |  | 
|---|
| [587] | 1560 |  | 
|---|
|  | 1561 |  | 
|---|
|  | 1562 | /////////////////////////////////////////////////////////// | 
|---|
|  | 1563 | static void _add_special_directories( fat_inode_t*  child, | 
|---|
|  | 1564 | fat_inode_t*  parent ) | 
|---|
| [258] | 1565 | { | 
|---|
| [587] | 1566 | // get first File-Cache buffer for child | 
|---|
|  | 1567 | fat_cache_desc_t*   pdesc  = (fat_cache_desc_t*)child->cache->children[0]; | 
|---|
|  | 1568 | unsigned char*      entry; | 
|---|
| [258] | 1569 |  | 
|---|
| [587] | 1570 | unsigned int i; | 
|---|
|  | 1571 | unsigned int cluster; | 
|---|
|  | 1572 | unsigned int size; | 
|---|
| [417] | 1573 |  | 
|---|
| [587] | 1574 | // set "." entry (32 bytes) | 
|---|
|  | 1575 | cluster = child->cluster; | 
|---|
|  | 1576 | size    = child->size; | 
|---|
|  | 1577 | entry   = pdesc->buffer; | 
|---|
|  | 1578 |  | 
|---|
|  | 1579 | for ( i = 0 ; i < 32 ; i++ ) | 
|---|
|  | 1580 | { | 
|---|
|  | 1581 | if      (i == 0 )     entry[i] = 0x2E;          // SFN | 
|---|
|  | 1582 | else if (i <  11)     entry[i] = 0x20;          // SFN | 
|---|
|  | 1583 | else if (i == 11)     entry[i] = 0x10;          // ATTR == dir | 
|---|
|  | 1584 | else if (i == 20)     entry[i] = cluster>>16;   // cluster.B2 | 
|---|
|  | 1585 | else if (i == 21)     entry[i] = cluster>>24;   // cluster.B3 | 
|---|
|  | 1586 | else if (i == 26)     entry[i] = cluster>>0;    // cluster.B0 | 
|---|
|  | 1587 | else if (i == 27)     entry[i] = cluster>>8;    // cluster.B1 | 
|---|
|  | 1588 | else if (i == 28)     entry[i] = size>>0;       // size.B0 | 
|---|
|  | 1589 | else if (i == 29)     entry[i] = size>>8;       // size.B1 | 
|---|
|  | 1590 | else if (i == 30)     entry[i] = size>>16;      // size.B2 | 
|---|
|  | 1591 | else if (i == 31)     entry[i] = size>>24;      // size.B3 | 
|---|
|  | 1592 | else                  entry[i] = 0x00; | 
|---|
|  | 1593 | } | 
|---|
| [258] | 1594 |  | 
|---|
| [587] | 1595 | // set ".." entry (32 bytes) | 
|---|
|  | 1596 | cluster = parent->cluster; | 
|---|
|  | 1597 | size    = parent->size; | 
|---|
|  | 1598 | entry   = pdesc->buffer + 32; | 
|---|
| [258] | 1599 |  | 
|---|
| [587] | 1600 | for ( i = 0 ; i < 32 ; i++ ) | 
|---|
| [258] | 1601 | { | 
|---|
| [587] | 1602 | if      (i <  2 )     entry[i] = 0x2E;          // SFN | 
|---|
|  | 1603 | else if (i <  11)     entry[i] = 0x20;          // SFN | 
|---|
|  | 1604 | else if (i == 11)     entry[i] = 0x10;          // ATTR == dir | 
|---|
|  | 1605 | else if (i == 20)     entry[i] = cluster>>16;   // cluster.B2 | 
|---|
|  | 1606 | else if (i == 21)     entry[i] = cluster>>24;   // cluster.B3 | 
|---|
|  | 1607 | else if (i == 26)     entry[i] = cluster>>0;    // cluster.B0 | 
|---|
|  | 1608 | else if (i == 27)     entry[i] = cluster>>8;    // cluster.B1 | 
|---|
|  | 1609 | else if (i == 28)     entry[i] = size>>0;       // size.B0 | 
|---|
|  | 1610 | else if (i == 29)     entry[i] = size>>8;       // size.B1 | 
|---|
|  | 1611 | else if (i == 30)     entry[i] = size>>16;      // size.B2 | 
|---|
|  | 1612 | else if (i == 31)     entry[i] = size>>24;      // size.B3 | 
|---|
|  | 1613 | else                  entry[i] = 0x00; | 
|---|
| [258] | 1614 | } | 
|---|
| [587] | 1615 | }  // end _add_special_directories | 
|---|
| [258] | 1616 |  | 
|---|
| [358] | 1617 |  | 
|---|
| [587] | 1618 |  | 
|---|
|  | 1619 | //////////////////////////////////////////////////////////// | 
|---|
| [638] | 1620 | static unsigned int _is_ancestor( fat_inode_t* a, | 
|---|
|  | 1621 | fat_inode_t* b ) | 
|---|
|  | 1622 | { | 
|---|
|  | 1623 | while ( b ) | 
|---|
|  | 1624 | { | 
|---|
|  | 1625 | if ( a == b ) | 
|---|
|  | 1626 | return 1; | 
|---|
|  | 1627 |  | 
|---|
|  | 1628 | b = b->parent; | 
|---|
|  | 1629 | } | 
|---|
|  | 1630 |  | 
|---|
|  | 1631 | return 0; | 
|---|
|  | 1632 | } // _is_ancestor() | 
|---|
|  | 1633 |  | 
|---|
|  | 1634 |  | 
|---|
|  | 1635 |  | 
|---|
|  | 1636 | //////////////////////////////////////////////////////////// | 
|---|
| [587] | 1637 | static unsigned int _check_name_length( char* name, | 
|---|
|  | 1638 | unsigned int* length, | 
|---|
|  | 1639 | unsigned int* nb_lfn ) | 
|---|
|  | 1640 | { | 
|---|
|  | 1641 | unsigned int len = _strlen( name ); | 
|---|
|  | 1642 | if      ( len <= 13 ) | 
|---|
| [258] | 1643 | { | 
|---|
| [587] | 1644 | *length  = len; | 
|---|
|  | 1645 | *nb_lfn  = 1; | 
|---|
|  | 1646 | return 0; | 
|---|
|  | 1647 | } | 
|---|
|  | 1648 | else if ( len <= 26 ) | 
|---|
|  | 1649 | { | 
|---|
|  | 1650 | *length  = len; | 
|---|
|  | 1651 | *nb_lfn  = 2; | 
|---|
|  | 1652 | return 0; | 
|---|
|  | 1653 | } | 
|---|
|  | 1654 | else if ( len <= 31 ) | 
|---|
|  | 1655 | { | 
|---|
|  | 1656 | *length  = len; | 
|---|
|  | 1657 | *nb_lfn  = 3; | 
|---|
|  | 1658 | return 0; | 
|---|
|  | 1659 | } | 
|---|
|  | 1660 | else | 
|---|
|  | 1661 | { | 
|---|
|  | 1662 | return 1; | 
|---|
|  | 1663 | } | 
|---|
|  | 1664 | }  // _check_name_length() | 
|---|
|  | 1665 |  | 
|---|
|  | 1666 |  | 
|---|
|  | 1667 |  | 
|---|
|  | 1668 |  | 
|---|
|  | 1669 | /////////////////////////////////////////////////////////// | 
|---|
|  | 1670 | static unsigned int _get_nb_entries( fat_inode_t*   inode, | 
|---|
|  | 1671 | unsigned int*  nb_entries ) | 
|---|
|  | 1672 | { | 
|---|
|  | 1673 | // scan directory until "end of directory" with two embedded loops: | 
|---|
|  | 1674 | // - scan the clusters allocated to this directory | 
|---|
|  | 1675 | // - scan the entries to find NO_MORE_ENTRY | 
|---|
|  | 1676 | fat_cache_desc_t*  pdesc;                      // pointer on buffer descriptor | 
|---|
|  | 1677 | unsigned char*     buffer;                     // 4 Kbytes buffer (one cluster) | 
|---|
|  | 1678 | unsigned int       ord;                        // ORD field in directory entry | 
|---|
|  | 1679 | unsigned int       attr;                       // ATTR field in directory entry | 
|---|
|  | 1680 | unsigned int       cluster_id = 0;             // cluster index in directory | 
|---|
|  | 1681 | unsigned int       offset     = 0;             // position in scanned buffer | 
|---|
|  | 1682 | unsigned int       found      = 0;             // NO_MORE_ENTRY found | 
|---|
|  | 1683 | unsigned int       count      = 0;             // number of valid NORMAL entries | 
|---|
|  | 1684 |  | 
|---|
|  | 1685 | // loop on clusters allocated to directory | 
|---|
|  | 1686 | while ( found == 0 ) | 
|---|
|  | 1687 | { | 
|---|
|  | 1688 | // get one 4 Kytes buffer from File_Cache | 
|---|
|  | 1689 | if ( _get_buffer_from_cache( inode, | 
|---|
|  | 1690 | cluster_id, | 
|---|
|  | 1691 | &pdesc ) )   return 1; | 
|---|
|  | 1692 | buffer = pdesc->buffer; | 
|---|
|  | 1693 |  | 
|---|
|  | 1694 | // loop on directory entries in buffer | 
|---|
|  | 1695 | while ( (offset < 4096) && (found == 0) ) | 
|---|
| [258] | 1696 | { | 
|---|
| [587] | 1697 | attr = _read_entry( DIR_ATTR , buffer + offset , 0 ); | 
|---|
|  | 1698 | ord  = _read_entry( LDIR_ORD , buffer + offset , 0 ); | 
|---|
|  | 1699 |  | 
|---|
|  | 1700 | if ( ord == NO_MORE_ENTRY ) | 
|---|
| [258] | 1701 | { | 
|---|
| [587] | 1702 | found = 1; | 
|---|
|  | 1703 | } | 
|---|
|  | 1704 | else if ( ord == FREE_ENTRY )             // free entry => skip | 
|---|
|  | 1705 | { | 
|---|
|  | 1706 | offset = offset + 32; | 
|---|
| [258] | 1707 | } | 
|---|
| [587] | 1708 | else if ( attr == ATTR_LONG_NAME_MASK )   // LFN entry => skip | 
|---|
| [258] | 1709 | { | 
|---|
| [587] | 1710 | offset = offset + 32; | 
|---|
|  | 1711 | } | 
|---|
|  | 1712 | else                                      // NORMAL entry | 
|---|
|  | 1713 | { | 
|---|
| [620] | 1714 | offset = offset + 32; | 
|---|
| [587] | 1715 | count++; | 
|---|
|  | 1716 | } | 
|---|
|  | 1717 | }  // end loop on directory entries | 
|---|
|  | 1718 |  | 
|---|
|  | 1719 | cluster_id++; | 
|---|
|  | 1720 | offset = 0; | 
|---|
|  | 1721 |  | 
|---|
|  | 1722 | }  // end loop on clusters | 
|---|
|  | 1723 |  | 
|---|
|  | 1724 | // return nb_entries | 
|---|
|  | 1725 | *nb_entries = count; | 
|---|
|  | 1726 |  | 
|---|
|  | 1727 | return 0; | 
|---|
|  | 1728 | }  // end dir_not_empty() | 
|---|
|  | 1729 |  | 
|---|
|  | 1730 |  | 
|---|
|  | 1731 |  | 
|---|
|  | 1732 | /////////////////////////////////////////////////////////// | 
|---|
|  | 1733 | static unsigned int _add_dir_entry( fat_inode_t*   child, | 
|---|
|  | 1734 | fat_inode_t*   parent ) | 
|---|
|  | 1735 | { | 
|---|
|  | 1736 | // get child attributes | 
|---|
|  | 1737 | unsigned int      is_dir  = child->is_dir; | 
|---|
|  | 1738 | unsigned int      size    = child->size; | 
|---|
|  | 1739 | unsigned int      cluster = child->cluster; | 
|---|
|  | 1740 |  | 
|---|
|  | 1741 | // compute number of required entries to store child->name | 
|---|
|  | 1742 | // - Short name (less than 13 characters) require 3 entries: | 
|---|
|  | 1743 | //   one LFN entry / one NORMAL entry / one NO_MORE_ENTRY entry. | 
|---|
|  | 1744 | // - Longer names (up to 31 characters) can require 4 or 5 entries: | 
|---|
|  | 1745 | //   2 or 3 LFN entries / one NORMAL entry / one NO_MORE entry. | 
|---|
|  | 1746 | unsigned int length; | 
|---|
|  | 1747 | unsigned int nb_lfn; | 
|---|
|  | 1748 | if ( _check_name_length( child->name, | 
|---|
|  | 1749 | &length, | 
|---|
|  | 1750 | &nb_lfn ) )  return 1; | 
|---|
|  | 1751 |  | 
|---|
|  | 1752 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1753 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1754 | _printf("\n[DEBUG FAT] _add_dir_entry(): try to add <%s> in <%s> / nb_lfn = %d\n", | 
|---|
| [587] | 1755 | child->name , parent->name, nb_lfn ); | 
|---|
|  | 1756 | #endif | 
|---|
|  | 1757 |  | 
|---|
|  | 1758 | // Find end of directory : two embedded loops: | 
|---|
|  | 1759 | // - scan the clusters allocated to this directory | 
|---|
|  | 1760 | // - scan the entries to find NO_MORE_ENTRY | 
|---|
|  | 1761 | fat_cache_desc_t*  pdesc;                      // pointer on buffer descriptor | 
|---|
|  | 1762 | unsigned char*     buffer;                     // 4 Kbytes buffer (one cluster) | 
|---|
|  | 1763 | unsigned int       cluster_id = 0;             // cluster index in directory | 
|---|
|  | 1764 | unsigned int       offset     = 0;             // position in scanned buffer | 
|---|
|  | 1765 | unsigned int       found      = 0;             // NO_MORE_ENTRY found | 
|---|
|  | 1766 |  | 
|---|
|  | 1767 | // loop on clusters allocated to directory | 
|---|
|  | 1768 | while ( found == 0 ) | 
|---|
|  | 1769 | { | 
|---|
|  | 1770 | // get one 4 Kytes buffer from File_Cache | 
|---|
|  | 1771 | if ( _get_buffer_from_cache( parent, | 
|---|
|  | 1772 | cluster_id, | 
|---|
|  | 1773 | &pdesc ) )   return 1; | 
|---|
|  | 1774 |  | 
|---|
|  | 1775 | buffer = pdesc->buffer; | 
|---|
|  | 1776 |  | 
|---|
|  | 1777 | // loop on directory entries in buffer | 
|---|
|  | 1778 | while ( (offset < 4096) && (found == 0) ) | 
|---|
|  | 1779 | { | 
|---|
|  | 1780 | if ( _read_entry( LDIR_ORD , buffer + offset , 0 ) == NO_MORE_ENTRY ) | 
|---|
| [258] | 1781 | { | 
|---|
| [587] | 1782 | found        = 1; | 
|---|
|  | 1783 | pdesc->dirty = 1; | 
|---|
|  | 1784 | } | 
|---|
|  | 1785 | else | 
|---|
|  | 1786 | { | 
|---|
|  | 1787 | offset = offset + 32; | 
|---|
| [258] | 1788 | } | 
|---|
| [587] | 1789 | }  // end loop on entries | 
|---|
|  | 1790 | if ( found == 0 ) | 
|---|
|  | 1791 | { | 
|---|
|  | 1792 | cluster_id++; | 
|---|
|  | 1793 | offset = 0; | 
|---|
| [258] | 1794 | } | 
|---|
| [587] | 1795 | }  // end loop on clusters | 
|---|
| [258] | 1796 |  | 
|---|
| [587] | 1797 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1798 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1799 | _printf("\n[DEBUG FAT] _add_dir_entry(): get NO_MORE directory entry : " | 
|---|
| [587] | 1800 | " buffer = %x / offset = %x / cluster_id = %d\n", | 
|---|
|  | 1801 | (unsigned int)buffer , offset , cluster_id ); | 
|---|
|  | 1802 | #endif | 
|---|
| [258] | 1803 |  | 
|---|
| [587] | 1804 | // enter FSM : | 
|---|
|  | 1805 | // The new child requires to write 3, 4, or 5 directory entries. | 
|---|
|  | 1806 | // To actually register the new child, we use a 5 steps FSM | 
|---|
|  | 1807 | // (one state per entry to be written), that is traversed as: | 
|---|
|  | 1808 | //    LFN3 -> LFN2 -> LFN1 -> NORMAL -> NOMORE | 
|---|
|  | 1809 | // The buffer and first directory entry to be  written are identified | 
|---|
|  | 1810 | // by the variables : buffer / cluster_id / offset | 
|---|
| [417] | 1811 |  | 
|---|
| [587] | 1812 | unsigned char* name  = (unsigned char*)child->name; | 
|---|
|  | 1813 |  | 
|---|
|  | 1814 | unsigned int step;          // FSM state | 
|---|
|  | 1815 |  | 
|---|
|  | 1816 | if      ( nb_lfn == 1 ) step = 3; | 
|---|
|  | 1817 | else if ( nb_lfn == 2 ) step = 4; | 
|---|
|  | 1818 | else if ( nb_lfn == 3 ) step = 5; | 
|---|
|  | 1819 |  | 
|---|
|  | 1820 | unsigned int   i;           // byte index in 32 bytes directory | 
|---|
|  | 1821 | unsigned int   c;           // character index in name | 
|---|
|  | 1822 | unsigned char* entry;       // buffer + offset; | 
|---|
|  | 1823 |  | 
|---|
|  | 1824 | while ( step ) | 
|---|
|  | 1825 | { | 
|---|
|  | 1826 | // get another buffer if required | 
|---|
|  | 1827 | if ( offset >= 4096 )  // new buffer required | 
|---|
| [417] | 1828 | { | 
|---|
| [665] | 1829 | if ( _get_buffer_from_cache( parent, | 
|---|
|  | 1830 | cluster_id + 1, | 
|---|
|  | 1831 | &pdesc ) )      return 1; | 
|---|
|  | 1832 | buffer       = pdesc->buffer; | 
|---|
|  | 1833 | pdesc->dirty = 1; | 
|---|
|  | 1834 | offset       = 0; | 
|---|
| [587] | 1835 | } | 
|---|
| [417] | 1836 |  | 
|---|
| [587] | 1837 | // compute directory entry address | 
|---|
|  | 1838 | entry = buffer + offset; | 
|---|
| [417] | 1839 |  | 
|---|
| [587] | 1840 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1841 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 1842 | _printf("\n[DEBUG FAT] _add_dir_entry(): FSM step = %d /" | 
|---|
| [587] | 1843 | " offset = %x / nb_lfn = %d\n", step, offset, nb_lfn ); | 
|---|
|  | 1844 | #endif | 
|---|
|  | 1845 |  | 
|---|
|  | 1846 | // write one 32 bytes directory entry per iteration | 
|---|
|  | 1847 | switch ( step ) | 
|---|
|  | 1848 | { | 
|---|
|  | 1849 | case 5:   // write LFN3 entry | 
|---|
|  | 1850 | { | 
|---|
|  | 1851 | c = 26; | 
|---|
|  | 1852 | // scan the 32 bytes in dir_entry | 
|---|
|  | 1853 | for ( i = 0 ; i < 32 ; i++ ) | 
|---|
| [258] | 1854 | { | 
|---|
| [587] | 1855 | if (i == 0) | 
|---|
|  | 1856 | { | 
|---|
|  | 1857 | if ( nb_lfn == 3) entry[i] = 0x43; | 
|---|
|  | 1858 | else              entry[i] = 0x03; | 
|---|
|  | 1859 | } | 
|---|
|  | 1860 | else if ( ( ((i >= 1 ) && (i<=10) && ((i&1)==1))   || | 
|---|
|  | 1861 | ((i >= 14) && (i<=25) && ((i&1)==0))   || | 
|---|
|  | 1862 | ((i >= 28) && (i<=31) && ((i&1)==0)) ) && | 
|---|
|  | 1863 | ( c < length ) ) | 
|---|
|  | 1864 | { | 
|---|
|  | 1865 | entry[i] = name[c]; | 
|---|
|  | 1866 | c++; | 
|---|
|  | 1867 | } | 
|---|
|  | 1868 | else if (i == 11)     entry[i] = 0x0F; | 
|---|
|  | 1869 | else if (i == 12)     entry[i] = 0xCA; | 
|---|
|  | 1870 | else                  entry[i] = 0x00; | 
|---|
| [258] | 1871 | } | 
|---|
| [587] | 1872 | step--; | 
|---|
|  | 1873 | break; | 
|---|
|  | 1874 | } | 
|---|
|  | 1875 | case 4:   // write LFN2 entry | 
|---|
|  | 1876 | { | 
|---|
|  | 1877 | c = 13; | 
|---|
|  | 1878 | // scan the 32 bytes in dir_entry | 
|---|
|  | 1879 | for ( i = 0 ; i < 32 ; i++ ) | 
|---|
| [258] | 1880 | { | 
|---|
| [587] | 1881 | if (i == 0) | 
|---|
|  | 1882 | { | 
|---|
|  | 1883 | if ( nb_lfn == 2) entry[i] = 0x42; | 
|---|
|  | 1884 | else              entry[i] = 0x02; | 
|---|
|  | 1885 | } | 
|---|
|  | 1886 | else if ( ( ((i >= 1 ) && (i<=10) && ((i&1)==1))   || | 
|---|
|  | 1887 | ((i >= 14) && (i<=25) && ((i&1)==0))   || | 
|---|
|  | 1888 | ((i >= 28) && (i<=31) && ((i&1)==0)) ) && | 
|---|
|  | 1889 | ( c < length ) ) | 
|---|
|  | 1890 | { | 
|---|
|  | 1891 | entry[i] = name[c]; | 
|---|
|  | 1892 | c++; | 
|---|
|  | 1893 | } | 
|---|
|  | 1894 | else if (i == 11)     entry[i] = 0x0F; | 
|---|
|  | 1895 | else if (i == 12)     entry[i] = 0xCA; | 
|---|
|  | 1896 | else                  entry[i] = 0x00; | 
|---|
| [258] | 1897 | } | 
|---|
| [587] | 1898 | step--; | 
|---|
|  | 1899 | break; | 
|---|
| [258] | 1900 | } | 
|---|
| [587] | 1901 | case 3:   // Write LFN1 entry | 
|---|
| [258] | 1902 | { | 
|---|
| [587] | 1903 | c = 0; | 
|---|
|  | 1904 | // scan the 32 bytes in dir_entry | 
|---|
|  | 1905 | for ( i = 0 ; i < 32 ; i++ ) | 
|---|
|  | 1906 | { | 
|---|
|  | 1907 | if (i == 0) | 
|---|
|  | 1908 | { | 
|---|
|  | 1909 | if ( nb_lfn == 1) entry[i] = 0x41; | 
|---|
|  | 1910 | else              entry[i] = 0x01; | 
|---|
|  | 1911 | } | 
|---|
|  | 1912 | else if ( ( ((i >= 1 ) && (i<=10) && ((i&1)==1))   || | 
|---|
|  | 1913 | ((i >= 14) && (i<=25) && ((i&1)==0))   || | 
|---|
|  | 1914 | ((i >= 28) && (i<=31) && ((i&1)==0)) ) && | 
|---|
|  | 1915 | ( c < length ) ) | 
|---|
|  | 1916 | { | 
|---|
|  | 1917 | entry[i] = name[c]; | 
|---|
|  | 1918 | c++; | 
|---|
|  | 1919 | } | 
|---|
|  | 1920 | else if (i == 11)     entry[i] = 0x0F; | 
|---|
|  | 1921 | else if (i == 12)     entry[i] = 0xCA; | 
|---|
|  | 1922 | else                  entry[i] = 0x00; | 
|---|
|  | 1923 | } | 
|---|
|  | 1924 | step--; | 
|---|
|  | 1925 | break; | 
|---|
| [258] | 1926 | } | 
|---|
| [587] | 1927 | case 2:   // write NORMAL entry | 
|---|
| [258] | 1928 | { | 
|---|
| [587] | 1929 | c = 0; | 
|---|
|  | 1930 | // scan the 32 bytes in dir_entry | 
|---|
|  | 1931 | for ( i = 0 ; i < 32 ; i++ ) | 
|---|
| [258] | 1932 | { | 
|---|
| [587] | 1933 | if      ( (i < 8) && (c < length) )             // SFN | 
|---|
| [417] | 1934 | { | 
|---|
| [587] | 1935 | entry[i] = _to_upper( name[c] ); | 
|---|
|  | 1936 | c++; | 
|---|
| [417] | 1937 | } | 
|---|
| [587] | 1938 | else if (i <  11)     entry[i] = 0x20;          // EXT | 
|---|
|  | 1939 | else if (i == 11)                               // ATTR | 
|---|
|  | 1940 | { | 
|---|
|  | 1941 | if (is_dir)       entry[i] = 0x10; | 
|---|
|  | 1942 | else              entry[i] = 0x20; | 
|---|
|  | 1943 | } | 
|---|
|  | 1944 | else if (i == 20)     entry[i] = cluster>>16;   // cluster.B2 | 
|---|
|  | 1945 | else if (i == 21)     entry[i] = cluster>>24;   // cluster.B3 | 
|---|
|  | 1946 | else if (i == 26)     entry[i] = cluster>>0;    // cluster.B0 | 
|---|
|  | 1947 | else if (i == 27)     entry[i] = cluster>>8;    // cluster.B1 | 
|---|
|  | 1948 | else if (i == 28)     entry[i] = size>>0;       // size.B0 | 
|---|
|  | 1949 | else if (i == 29)     entry[i] = size>>8;       // size.B1 | 
|---|
|  | 1950 | else if (i == 30)     entry[i] = size>>16;      // size.B2 | 
|---|
|  | 1951 | else if (i == 31)     entry[i] = size>>24;      // size.B3 | 
|---|
|  | 1952 | else                  entry[i] = 0x00; | 
|---|
| [258] | 1953 | } | 
|---|
| [587] | 1954 |  | 
|---|
|  | 1955 | // update the dentry field in child inode | 
|---|
|  | 1956 | child->dentry = ((cluster_id<<12) + offset)>>5; | 
|---|
|  | 1957 |  | 
|---|
|  | 1958 | step--; | 
|---|
|  | 1959 | break; | 
|---|
| [417] | 1960 | } | 
|---|
| [587] | 1961 | case 1:   // write NOMORE entry | 
|---|
| [417] | 1962 | { | 
|---|
| [587] | 1963 | step--; | 
|---|
|  | 1964 | entry [0] = 0x00; | 
|---|
|  | 1965 | break; | 
|---|
|  | 1966 | } | 
|---|
|  | 1967 | } // end switch step | 
|---|
|  | 1968 | offset += 32; | 
|---|
|  | 1969 | } // exit while => exit FSM | 
|---|
|  | 1970 |  | 
|---|
|  | 1971 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 1972 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
|  | 1973 | { | 
|---|
| [664] | 1974 | _printf("\n[DEBUG FAT] _add_dir_entry(): <%s> successfully added in <%s>\n", | 
|---|
| [587] | 1975 | child->name , parent->name ); | 
|---|
|  | 1976 | } | 
|---|
|  | 1977 | #endif | 
|---|
|  | 1978 |  | 
|---|
|  | 1979 | return 0; | 
|---|
|  | 1980 | } // end _add_dir_entry | 
|---|
|  | 1981 |  | 
|---|
|  | 1982 |  | 
|---|
|  | 1983 |  | 
|---|
|  | 1984 | //////////////////////////////////////////////////////////// | 
|---|
|  | 1985 | static unsigned int _remove_dir_entry( fat_inode_t*  inode ) | 
|---|
|  | 1986 | { | 
|---|
|  | 1987 | // compute number of LFN entries | 
|---|
|  | 1988 | unsigned int length; | 
|---|
|  | 1989 | unsigned int nb_lfn; | 
|---|
|  | 1990 | if ( _check_name_length( inode->name, | 
|---|
|  | 1991 | &length, | 
|---|
|  | 1992 | &nb_lfn ) )  return 1; | 
|---|
|  | 1993 |  | 
|---|
|  | 1994 | // get cluster_id and offset in parent directory cache | 
|---|
|  | 1995 | unsigned int  dentry     = inode->dentry; | 
|---|
|  | 1996 | unsigned int  cluster_id = dentry >> 7; | 
|---|
|  | 1997 | unsigned int  offset     = (dentry & 0x7F)<<5; | 
|---|
|  | 1998 |  | 
|---|
|  | 1999 | // get buffer from parent directory cache | 
|---|
|  | 2000 | unsigned char*     buffer; | 
|---|
|  | 2001 | fat_cache_desc_t*  pdesc; | 
|---|
|  | 2002 |  | 
|---|
|  | 2003 | if ( _get_buffer_from_cache( inode->parent, | 
|---|
|  | 2004 | cluster_id, | 
|---|
|  | 2005 | &pdesc ) ) return 1; | 
|---|
|  | 2006 | buffer       = pdesc->buffer; | 
|---|
|  | 2007 | pdesc->dirty = 1; | 
|---|
|  | 2008 |  | 
|---|
|  | 2009 | // invalidate NORMAL entry in directory cache | 
|---|
|  | 2010 | buffer[offset] = 0xE5; | 
|---|
|  | 2011 |  | 
|---|
|  | 2012 | // invalidate LFN entries | 
|---|
|  | 2013 | while ( nb_lfn ) | 
|---|
|  | 2014 | { | 
|---|
| [616] | 2015 | if (offset == 0)  // we must load buffer for (cluster_id - 1) | 
|---|
| [587] | 2016 | { | 
|---|
| [636] | 2017 | if ( cluster_id == 0 ) | 
|---|
|  | 2018 | break; | 
|---|
|  | 2019 |  | 
|---|
| [587] | 2020 | if ( _get_buffer_from_cache( inode->parent, | 
|---|
|  | 2021 | cluster_id - 1, | 
|---|
|  | 2022 | &pdesc ) )   return 1; | 
|---|
|  | 2023 | buffer       = pdesc->buffer; | 
|---|
|  | 2024 | pdesc->dirty = 1; | 
|---|
| [616] | 2025 | offset       = 4096; | 
|---|
| [587] | 2026 | } | 
|---|
|  | 2027 |  | 
|---|
| [616] | 2028 | offset = offset - 32; | 
|---|
|  | 2029 |  | 
|---|
|  | 2030 | // check for LFN entry | 
|---|
|  | 2031 | if ( _read_entry( DIR_ATTR , buffer + offset , 0 ) != ATTR_LONG_NAME_MASK ) | 
|---|
|  | 2032 | break; | 
|---|
|  | 2033 |  | 
|---|
| [587] | 2034 | // invalidate LFN entry | 
|---|
|  | 2035 | buffer[offset] = 0xE5; | 
|---|
|  | 2036 |  | 
|---|
|  | 2037 | nb_lfn--; | 
|---|
|  | 2038 | } | 
|---|
|  | 2039 |  | 
|---|
|  | 2040 | return 0; | 
|---|
|  | 2041 | }  // end _remove_dir_entry | 
|---|
|  | 2042 |  | 
|---|
|  | 2043 |  | 
|---|
|  | 2044 |  | 
|---|
|  | 2045 |  | 
|---|
|  | 2046 | //////////////////////////////////////////////////////////// | 
|---|
|  | 2047 | static unsigned int _update_dir_entry( fat_inode_t*  inode ) | 
|---|
|  | 2048 | { | 
|---|
|  | 2049 | // get Cache-File buffer containing the parent directory entry | 
|---|
|  | 2050 | // 128 directories entries in one 4 Kbytes buffer | 
|---|
|  | 2051 | fat_cache_desc_t*  pdesc; | 
|---|
|  | 2052 | unsigned char*     buffer; | 
|---|
|  | 2053 | unsigned int       cluster_id = inode->dentry>>7; | 
|---|
| [626] | 2054 | unsigned int       offset     = (inode->dentry & 0x7F)<<5; | 
|---|
| [587] | 2055 |  | 
|---|
|  | 2056 | if ( _get_buffer_from_cache( inode->parent, | 
|---|
|  | 2057 | cluster_id, | 
|---|
|  | 2058 | &pdesc ) )    return 1; | 
|---|
|  | 2059 | buffer       = pdesc->buffer; | 
|---|
|  | 2060 | pdesc->dirty = 1; | 
|---|
|  | 2061 |  | 
|---|
|  | 2062 | // update size field | 
|---|
|  | 2063 | buffer[offset + 28] = inode->size>>0;       // size.B0 | 
|---|
|  | 2064 | buffer[offset + 29] = inode->size>>8;       // size.B1 | 
|---|
|  | 2065 | buffer[offset + 30] = inode->size>>16;      // size.B2 | 
|---|
|  | 2066 | buffer[offset + 31] = inode->size>>24;      // size.B3 | 
|---|
|  | 2067 |  | 
|---|
|  | 2068 | // update cluster field | 
|---|
|  | 2069 | buffer[offset + 26] = inode->cluster>>0;    // cluster.B0 | 
|---|
|  | 2070 | buffer[offset + 27] = inode->cluster>>8;    // cluster.B1 | 
|---|
|  | 2071 | buffer[offset + 20] = inode->cluster>>16;   // cluster.B2 | 
|---|
|  | 2072 | buffer[offset + 21] = inode->cluster>>24;   // cluster.B3 | 
|---|
|  | 2073 |  | 
|---|
|  | 2074 | return 0; | 
|---|
|  | 2075 | } // end _update_dir_entry() | 
|---|
|  | 2076 |  | 
|---|
|  | 2077 |  | 
|---|
|  | 2078 |  | 
|---|
|  | 2079 |  | 
|---|
|  | 2080 | ////////////////////////////////////////////////////////////////// | 
|---|
|  | 2081 | static unsigned int _get_child_from_parent( fat_inode_t*   parent, | 
|---|
|  | 2082 | char*          name, | 
|---|
|  | 2083 | fat_inode_t**  inode ) | 
|---|
|  | 2084 | { | 
|---|
|  | 2085 | fat_inode_t*   current; | 
|---|
|  | 2086 |  | 
|---|
|  | 2087 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2088 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2089 | _printf("\n[DEBUG FAT] _get_child_from_parent(): search <%s> in directory <%s>\n", | 
|---|
| [587] | 2090 | name , parent->name ); | 
|---|
|  | 2091 | #endif | 
|---|
|  | 2092 |  | 
|---|
|  | 2093 | // scan inodes in the parent directory | 
|---|
|  | 2094 | for ( current = parent->child ; current ; current = current->next ) | 
|---|
|  | 2095 | { | 
|---|
| [619] | 2096 | if ( _strcmp( name , current->name ) == 0 ) | 
|---|
| [587] | 2097 | { | 
|---|
|  | 2098 |  | 
|---|
|  | 2099 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2100 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2101 | _printf("\n[DEBUG FAT] _get_child_from_parent(): found inode <%s> in directory <%s>\n", | 
|---|
| [587] | 2102 | name , parent->name ); | 
|---|
|  | 2103 | #endif | 
|---|
|  | 2104 | *inode = current; | 
|---|
|  | 2105 | return 0;           // name found | 
|---|
|  | 2106 | } | 
|---|
|  | 2107 | } | 
|---|
|  | 2108 |  | 
|---|
|  | 2109 | // not found in Inode-Tree => access the parent directory | 
|---|
|  | 2110 | // file_cache.  Two embedded loops: | 
|---|
|  | 2111 | // - scan the clusters allocated to this directory | 
|---|
|  | 2112 | // - scan the directory entries in each 4 Kbytes buffer | 
|---|
|  | 2113 |  | 
|---|
|  | 2114 | unsigned char*    buffer;           // pointer on one cache buffer | 
|---|
|  | 2115 | char              cname[32];        // buffer for one full entry name | 
|---|
|  | 2116 | char              lfn1[16];         // buffer for one partial name | 
|---|
|  | 2117 | char              lfn2[16];         // buffer for one partial name | 
|---|
|  | 2118 | char              lfn3[16];         // buffer for one partial name | 
|---|
|  | 2119 | unsigned int      size;             // searched file/dir size (bytes) | 
|---|
|  | 2120 | unsigned int      cluster;          // searched file/dir cluster index | 
|---|
|  | 2121 | unsigned int      is_dir;           // searched file/dir type | 
|---|
|  | 2122 | unsigned int      attr;             // directory entry ATTR field | 
|---|
|  | 2123 | unsigned int      ord;              // directory entry ORD field | 
|---|
| [597] | 2124 | unsigned int      lfn = 0;          // LFN entries number | 
|---|
| [587] | 2125 | unsigned int      dentry;           // directory entry index | 
|---|
|  | 2126 | unsigned int      offset     = 0;   // byte offset in buffer | 
|---|
|  | 2127 | unsigned int      cluster_id = 0;   // cluster index in directory | 
|---|
|  | 2128 | int               found      = 0;   // not found (0) / name found (1) / end of dir (-1) | 
|---|
|  | 2129 |  | 
|---|
|  | 2130 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2131 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2132 | _printf("\n[DEBUG FAT] _get_child_from_parent(): does not found inode <%s>" | 
|---|
| [587] | 2133 | " in directory <%s> => search in cache\n", name , parent->name ); | 
|---|
|  | 2134 | #endif | 
|---|
|  | 2135 |  | 
|---|
|  | 2136 | // scan the clusters allocated to parent directory | 
|---|
|  | 2137 | while ( found == 0 ) | 
|---|
|  | 2138 | { | 
|---|
|  | 2139 | // get one 4 Kytes buffer from parent File_Cache | 
|---|
|  | 2140 | fat_cache_desc_t*  pdesc; | 
|---|
|  | 2141 | if ( _get_buffer_from_cache( parent, | 
|---|
|  | 2142 | cluster_id, | 
|---|
|  | 2143 | &pdesc ) )    return 2; | 
|---|
|  | 2144 | buffer = pdesc->buffer; | 
|---|
|  | 2145 |  | 
|---|
|  | 2146 | // scan this buffer until end of directory, end of buffer, or name found | 
|---|
|  | 2147 | while( (offset < 4096) && (found == 0) ) | 
|---|
|  | 2148 | { | 
|---|
|  | 2149 |  | 
|---|
|  | 2150 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2151 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2152 | _printf("\n[DEBUG FAT] _get_child_from_parent(): scan buffer %d for <%s>\n", | 
|---|
| [587] | 2153 | cluster_id , name ); | 
|---|
|  | 2154 | #endif | 
|---|
|  | 2155 | attr = _read_entry( DIR_ATTR , buffer + offset , 0 ); | 
|---|
|  | 2156 | ord  = _read_entry( LDIR_ORD , buffer + offset , 0 ); | 
|---|
|  | 2157 |  | 
|---|
|  | 2158 | if (ord == NO_MORE_ENTRY)                 // no more entry in directory => break | 
|---|
|  | 2159 | { | 
|---|
|  | 2160 | found = -1; | 
|---|
|  | 2161 | } | 
|---|
|  | 2162 | else if ( ord == FREE_ENTRY )             // free entry => skip | 
|---|
|  | 2163 | { | 
|---|
|  | 2164 | offset = offset + 32; | 
|---|
|  | 2165 | } | 
|---|
|  | 2166 | else if ( attr == ATTR_LONG_NAME_MASK )   // LFN entry => get partial name | 
|---|
|  | 2167 | { | 
|---|
|  | 2168 | unsigned int seq = ord & 0x3; | 
|---|
|  | 2169 | lfn = (seq > lfn) ? seq : lfn; | 
|---|
|  | 2170 | if      ( seq == 1 ) _get_name_from_long( buffer + offset, lfn1 ); | 
|---|
|  | 2171 | else if ( seq == 2 ) _get_name_from_long( buffer + offset, lfn2 ); | 
|---|
|  | 2172 | else if ( seq == 3 ) _get_name_from_long( buffer + offset, lfn3 ); | 
|---|
|  | 2173 | offset = offset + 32; | 
|---|
|  | 2174 | } | 
|---|
|  | 2175 | else                                 // NORMAL entry | 
|---|
|  | 2176 | { | 
|---|
|  | 2177 | // build the extracted name | 
|---|
|  | 2178 | if      ( lfn == 0 ) | 
|---|
| [258] | 2179 | { | 
|---|
| [587] | 2180 | _get_name_from_short( buffer + offset , cname ); | 
|---|
| [258] | 2181 | } | 
|---|
| [587] | 2182 | else if ( lfn == 1 ) | 
|---|
| [417] | 2183 | { | 
|---|
| [587] | 2184 | _strcpy( cname      , lfn1 ); | 
|---|
|  | 2185 | } | 
|---|
|  | 2186 | else if ( lfn == 2 ) | 
|---|
|  | 2187 | { | 
|---|
|  | 2188 | _strcpy( cname      , lfn1 ); | 
|---|
|  | 2189 | _strcpy( cname + 13 , lfn2 ); | 
|---|
| [417] | 2190 | } | 
|---|
| [587] | 2191 | else if ( lfn == 3 ) | 
|---|
|  | 2192 | { | 
|---|
|  | 2193 | _strcpy( cname      , lfn1 ); | 
|---|
|  | 2194 | _strcpy( cname + 13 , lfn2 ); | 
|---|
|  | 2195 | _strcpy( cname + 26 , lfn3 ); | 
|---|
|  | 2196 | } | 
|---|
|  | 2197 |  | 
|---|
|  | 2198 | // test if extracted name == searched name | 
|---|
| [619] | 2199 | if ( _strcmp( name , cname ) == 0 ) | 
|---|
| [587] | 2200 | { | 
|---|
|  | 2201 | cluster = (_read_entry( DIR_FST_CLUS_HI , buffer + offset , 1 ) << 16) | | 
|---|
|  | 2202 | (_read_entry( DIR_FST_CLUS_LO , buffer + offset , 1 )      ) ; | 
|---|
|  | 2203 | dentry  = ((cluster_id<<12) + offset)>>5; | 
|---|
|  | 2204 | is_dir  = ((attr & ATTR_DIRECTORY) == ATTR_DIRECTORY); | 
|---|
|  | 2205 | size    = _read_entry( DIR_FILE_SIZE , buffer + offset , 1 ); | 
|---|
|  | 2206 | found   = 1; | 
|---|
|  | 2207 | } | 
|---|
|  | 2208 | offset = offset + 32; | 
|---|
|  | 2209 | lfn    = 0; | 
|---|
| [258] | 2210 | } | 
|---|
| [587] | 2211 | }  // end loop on directory entries | 
|---|
|  | 2212 | cluster_id++; | 
|---|
|  | 2213 | offset = 0; | 
|---|
|  | 2214 | }  // end loop on buffers | 
|---|
|  | 2215 |  | 
|---|
|  | 2216 | if ( found == -1 )  // found end of directory in parent directory | 
|---|
|  | 2217 | { | 
|---|
|  | 2218 |  | 
|---|
|  | 2219 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2220 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2221 | _printf("\n[DEBUG FAT] _get_child_from_parent(): found end of directory in <%s>\n", | 
|---|
| [587] | 2222 | parent->name ); | 
|---|
|  | 2223 | #endif | 
|---|
|  | 2224 | *inode = NULL; | 
|---|
|  | 2225 | return 1; | 
|---|
|  | 2226 | } | 
|---|
|  | 2227 | else               // found searched name in parent directory | 
|---|
|  | 2228 | { | 
|---|
|  | 2229 | // allocate a new inode and an empty Cache-File | 
|---|
|  | 2230 | *inode = _allocate_one_inode( name, | 
|---|
|  | 2231 | is_dir, | 
|---|
|  | 2232 | cluster, | 
|---|
|  | 2233 | size, | 
|---|
|  | 2234 | 0,             // count | 
|---|
|  | 2235 | dentry, | 
|---|
|  | 2236 | 1 );           // cache_allocate | 
|---|
|  | 2237 |  | 
|---|
|  | 2238 | // introduce it in Inode-Tree | 
|---|
|  | 2239 | _add_inode_in_tree( *inode , parent ); | 
|---|
|  | 2240 |  | 
|---|
|  | 2241 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2242 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2243 | _printf("\n[DEBUG FAT] _get_child_from_parent(): found <%s> on device\n", name ); | 
|---|
| [587] | 2244 | #endif | 
|---|
|  | 2245 | return 0; | 
|---|
|  | 2246 | } | 
|---|
|  | 2247 | }  // end _get_child_from_parent() | 
|---|
|  | 2248 |  | 
|---|
|  | 2249 |  | 
|---|
|  | 2250 |  | 
|---|
|  | 2251 |  | 
|---|
|  | 2252 | ////////////////////////////////////////////////////////////////// | 
|---|
|  | 2253 | static unsigned int _get_inode_from_path( char*          pathname, | 
|---|
|  | 2254 | fat_inode_t**  inode ) | 
|---|
|  | 2255 | { | 
|---|
|  | 2256 | char                 name[32];         // buffer for one name in pathname | 
|---|
|  | 2257 | unsigned int         nb_read;              // number of characters written in name[] | 
|---|
|  | 2258 | fat_inode_t*         parent;           // parent inode | 
|---|
|  | 2259 | fat_inode_t*         child;            // child inode | 
|---|
|  | 2260 | unsigned int         last;             // while exit condition | 
|---|
|  | 2261 | unsigned int         code;             // return value | 
|---|
|  | 2262 |  | 
|---|
|  | 2263 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2264 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2265 | _printf("\n[DEBUG FAT] _get_inode_from_path(): enters for path <%s>\n", pathname ); | 
|---|
| [587] | 2266 | #endif | 
|---|
|  | 2267 |  | 
|---|
|  | 2268 | // handle root directory case | 
|---|
| [619] | 2269 | if ( _strcmp( pathname , "/" ) == 0 ) | 
|---|
| [587] | 2270 | { | 
|---|
|  | 2271 |  | 
|---|
|  | 2272 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2273 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2274 | _printf("\n[DEBUG FAT] _get_inode_from_path(): found root inode for <%s>\n", | 
|---|
| [587] | 2275 | pathname ); | 
|---|
|  | 2276 | #endif | 
|---|
|  | 2277 | *inode  = _fat.inode_tree_root; | 
|---|
|  | 2278 | return 0; | 
|---|
|  | 2279 | } | 
|---|
|  | 2280 |  | 
|---|
|  | 2281 | // If the pathname is not "/", we traverse the inode tree from the root. | 
|---|
|  | 2282 | // We use _get_name_from_path() to scan pathname and extract inode names. | 
|---|
|  | 2283 | // We use _get_child_from_parent() to scan each directory in the path. | 
|---|
|  | 2284 |  | 
|---|
|  | 2285 | last       = 0; | 
|---|
|  | 2286 | nb_read    = 0;                      // number of characters analysed in path | 
|---|
|  | 2287 | parent     = _fat.inode_tree_root;   // Inode-Tree root | 
|---|
|  | 2288 |  | 
|---|
| [617] | 2289 | while ( !last ) | 
|---|
| [587] | 2290 | { | 
|---|
|  | 2291 | // get searched file/dir name | 
|---|
|  | 2292 | if ( _get_name_from_path( pathname, name, &nb_read ) ) | 
|---|
|  | 2293 | { | 
|---|
|  | 2294 | return 3;   // error : name too long | 
|---|
| [258] | 2295 | } | 
|---|
|  | 2296 |  | 
|---|
| [587] | 2297 | // compute last iteration condition | 
|---|
|  | 2298 | last = (pathname[nb_read] == 0); | 
|---|
| [258] | 2299 |  | 
|---|
| [587] | 2300 | #if (GIET_DEBUG_FAT & 1) | 
|---|
| [569] | 2301 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2302 | _printf("\n[DEBUG FAT] _get_inode_from_path(): got name <%s>\n", name ); | 
|---|
| [417] | 2303 | #endif | 
|---|
|  | 2304 |  | 
|---|
| [619] | 2305 | if ( _strcmp( name, ".." ) == 0) | 
|---|
| [587] | 2306 | { | 
|---|
| [617] | 2307 | // found special name "..", try to go up | 
|---|
|  | 2308 | code = 0; | 
|---|
|  | 2309 | if ( parent->parent ) | 
|---|
|  | 2310 | child = parent->parent; | 
|---|
|  | 2311 | else | 
|---|
|  | 2312 | child = parent; | 
|---|
|  | 2313 | } | 
|---|
| [619] | 2314 | else if ( _strcmp( name, "." ) == 0 ) | 
|---|
| [617] | 2315 | { | 
|---|
|  | 2316 | // found special name ".", stay on the same level | 
|---|
|  | 2317 | code = 0; | 
|---|
|  | 2318 | child = parent; | 
|---|
|  | 2319 | } | 
|---|
|  | 2320 | else | 
|---|
|  | 2321 | { | 
|---|
|  | 2322 | // get child inode from parent directory | 
|---|
|  | 2323 | code = _get_child_from_parent( parent, | 
|---|
|  | 2324 | name, | 
|---|
|  | 2325 | &child ); | 
|---|
| [417] | 2326 |  | 
|---|
| [617] | 2327 | // we need to find the child inode for all non terminal names | 
|---|
|  | 2328 | if ( (code == 2) || ((code == 1 ) && !last) ) | 
|---|
|  | 2329 | { | 
|---|
|  | 2330 |  | 
|---|
|  | 2331 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2332 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2333 | _printf("\n[DEBUG FAT] _get_inode_from_path(): neither parent, nor child found for <%s>\n", | 
|---|
| [617] | 2334 | pathname ); | 
|---|
|  | 2335 | #endif | 
|---|
|  | 2336 | return 2;  // error : parent inode not found | 
|---|
|  | 2337 | } | 
|---|
| [587] | 2338 | } | 
|---|
| [617] | 2339 |  | 
|---|
| [587] | 2340 | // update parent if not the last iteration | 
|---|
| [617] | 2341 | if ( !last ) | 
|---|
|  | 2342 | parent = child; | 
|---|
| [587] | 2343 | } // end while | 
|---|
|  | 2344 |  | 
|---|
|  | 2345 | // returns inode pointer | 
|---|
|  | 2346 | if (code == 0 ) | 
|---|
|  | 2347 | { | 
|---|
|  | 2348 |  | 
|---|
|  | 2349 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2350 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2351 | _printf("\n[DEBUG FAT] _get_inode_from_path(): found inode for <%s>\n", | 
|---|
| [587] | 2352 | pathname ); | 
|---|
|  | 2353 | #endif | 
|---|
|  | 2354 | *inode  = child; | 
|---|
|  | 2355 | } | 
|---|
|  | 2356 | else | 
|---|
|  | 2357 | { | 
|---|
|  | 2358 |  | 
|---|
|  | 2359 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2360 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2361 | _printf("\n[DEBUG FAT] _get_inode_from_path(): found only parent inode for <%s>\n", | 
|---|
| [587] | 2362 | pathname ); | 
|---|
|  | 2363 | #endif | 
|---|
|  | 2364 | *inode  = parent; | 
|---|
|  | 2365 | } | 
|---|
|  | 2366 |  | 
|---|
|  | 2367 | return code;                 // can be 0 (found) or 1 (not found) | 
|---|
|  | 2368 |  | 
|---|
|  | 2369 | }  // end _get_inode_from_path() | 
|---|
|  | 2370 |  | 
|---|
|  | 2371 |  | 
|---|
|  | 2372 |  | 
|---|
|  | 2373 |  | 
|---|
|  | 2374 | ////////////////////////////////////////////////////////////// | 
|---|
|  | 2375 | static unsigned int _remove_node_from_fs( fat_inode_t* inode ) | 
|---|
| [258] | 2376 | { | 
|---|
| [637] | 2377 | // check for root node | 
|---|
|  | 2378 | if ( !inode->parent ) return 1; | 
|---|
|  | 2379 |  | 
|---|
| [587] | 2380 | // remove entry in parent directory | 
|---|
|  | 2381 | if ( _remove_dir_entry( inode ) ) return 1; | 
|---|
|  | 2382 |  | 
|---|
|  | 2383 | // update parent directory on device | 
|---|
|  | 2384 | if ( _update_device_from_cache( inode->parent->levels, | 
|---|
|  | 2385 | inode->parent->cache, | 
|---|
|  | 2386 | inode->parent->name ) ) return 1; | 
|---|
|  | 2387 |  | 
|---|
|  | 2388 | // release clusters allocated to file/dir in DATA region | 
|---|
|  | 2389 | if ( _clusters_release( inode->cluster ) ) return 1; | 
|---|
|  | 2390 |  | 
|---|
|  | 2391 | // release File-Cache | 
|---|
| [651] | 2392 | _release_cache_memory( inode->cache, inode->levels ); | 
|---|
|  | 2393 | _free ( inode->cache ); | 
|---|
| [587] | 2394 |  | 
|---|
|  | 2395 | // remove inode from Inode-Tree | 
|---|
|  | 2396 | _remove_inode_from_tree( inode ); | 
|---|
|  | 2397 |  | 
|---|
|  | 2398 | // release inode | 
|---|
|  | 2399 | _free ( inode ); | 
|---|
|  | 2400 |  | 
|---|
| [258] | 2401 | return 0; | 
|---|
| [587] | 2402 | }  // end _remove_node_from_fs() | 
|---|
| [258] | 2403 |  | 
|---|
|  | 2404 |  | 
|---|
| [587] | 2405 | ////////////////////////////////////////////////////////////////// | 
|---|
|  | 2406 | static unsigned int _next_cluster_no_cache( unsigned int   cluster, | 
|---|
|  | 2407 | unsigned int*  next ) | 
|---|
|  | 2408 | { | 
|---|
|  | 2409 | // compute cluster_id and slot_id | 
|---|
|  | 2410 | // each cluster contains 1024 slots (4 bytes per slot) | 
|---|
|  | 2411 | unsigned int cluster_id  = cluster >> 10; | 
|---|
|  | 2412 | unsigned int slot_id     = cluster & 0x3FF; | 
|---|
| [258] | 2413 |  | 
|---|
| [587] | 2414 | // compute lba of cluster identified by cluster_id | 
|---|
|  | 2415 | unsigned int lba = _fat.fat_lba + (cluster_id << 3); | 
|---|
| [258] | 2416 |  | 
|---|
| [587] | 2417 | // get cluster containing the adressed FAT slot in FAT buffer | 
|---|
|  | 2418 | if ( _fat_buffer_fat_lba != lba ) | 
|---|
|  | 2419 | { | 
|---|
|  | 2420 | if ( _fat_ioc_access( 0,         // no descheduling | 
|---|
|  | 2421 | 1,         // read | 
|---|
|  | 2422 | lba, | 
|---|
|  | 2423 | (unsigned int)_fat_buffer_fat, | 
|---|
|  | 2424 | 8 ) ) | 
|---|
|  | 2425 | { | 
|---|
| [664] | 2426 | _printf("\n[FAT ERROR] _next_cluster_no_cache(): " | 
|---|
| [587] | 2427 | "cannot load lba = %x into fat_buffer\n", lba ); | 
|---|
|  | 2428 | return 1; | 
|---|
|  | 2429 | } | 
|---|
|  | 2430 |  | 
|---|
|  | 2431 | _fat_buffer_fat_lba = lba; | 
|---|
|  | 2432 | } | 
|---|
|  | 2433 |  | 
|---|
|  | 2434 | // return next cluster index | 
|---|
|  | 2435 | unsigned int* buf = (unsigned int*)_fat_buffer_fat; | 
|---|
|  | 2436 | *next = buf[slot_id]; | 
|---|
|  | 2437 | return 0; | 
|---|
|  | 2438 |  | 
|---|
|  | 2439 | }  // end _next_cluster_no_cache() | 
|---|
|  | 2440 |  | 
|---|
|  | 2441 |  | 
|---|
|  | 2442 |  | 
|---|
|  | 2443 |  | 
|---|
|  | 2444 | ///////////////////////////////////////////////////////////////// | 
|---|
|  | 2445 | static unsigned int _file_info_no_cache( char*          pathname, | 
|---|
|  | 2446 | unsigned int*  file_cluster, | 
|---|
|  | 2447 | unsigned int*  file_size ) | 
|---|
| [258] | 2448 | { | 
|---|
| [587] | 2449 |  | 
|---|
|  | 2450 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2451 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2452 | _printf("\n[DEBUG FAT] _file_info_no_cache(): enters for path <%s>\n", pathname ); | 
|---|
| [587] | 2453 | #endif | 
|---|
| [258] | 2454 |  | 
|---|
| [587] | 2455 | char            name[32];             // buffer for one name in the analysed pathname | 
|---|
|  | 2456 | char            lfn1[16];             // buffer for a partial name in LFN entry | 
|---|
|  | 2457 | char            lfn2[16];             // buffer for a partial name in LFN entry | 
|---|
|  | 2458 | char            lfn3[16];             // buffer for a partial name in LFN entry | 
|---|
|  | 2459 | char            cname[32];            // buffer for a full name in a directory entry | 
|---|
|  | 2460 | unsigned int    nb_read;              // number of characters analysed in path | 
|---|
|  | 2461 | unsigned int    parent_cluster;       // cluster index for the parent directory | 
|---|
| [597] | 2462 | unsigned int    child_cluster = 0;    // cluster index for the searched file/dir | 
|---|
|  | 2463 | unsigned int    child_size = 0;       // size of the searched file/dir | 
|---|
| [587] | 2464 | unsigned int    child_is_dir;         // type of the searched file/dir | 
|---|
|  | 2465 | unsigned int    offset;               // offset in a 4 Kbytes buffer | 
|---|
|  | 2466 | unsigned int    ord;                  // ORD field in a directory entry | 
|---|
|  | 2467 | unsigned int    attr;                 // ATTR field in a directory entry | 
|---|
| [597] | 2468 | unsigned int    lfn = 0;              // number of lfn entries | 
|---|
| [587] | 2469 | unsigned char*  buf;                  // pointer on a 4 Kbytes buffer | 
|---|
|  | 2470 | unsigned int    found;                // name found in current directory entry | 
|---|
|  | 2471 |  | 
|---|
|  | 2472 | // Three embedded loops: | 
|---|
|  | 2473 | // - scan pathname to extract file/dir names, | 
|---|
|  | 2474 | // - for each name, scan the clusters of the parent directory | 
|---|
|  | 2475 | // - for each cluster, scan the 4 Kbytes buffer to find the file/dir name | 
|---|
|  | 2476 | // The starting point is the root directory (cluster 2) | 
|---|
|  | 2477 |  | 
|---|
|  | 2478 | nb_read        = 0; | 
|---|
|  | 2479 | parent_cluster = 2; | 
|---|
|  | 2480 |  | 
|---|
|  | 2481 | // scan pathname | 
|---|
|  | 2482 | while ( pathname[nb_read] != 0 ) | 
|---|
|  | 2483 | { | 
|---|
|  | 2484 | // get searched file/dir name | 
|---|
|  | 2485 | if ( _get_name_from_path( pathname, name, &nb_read ) ) return 1; | 
|---|
|  | 2486 |  | 
|---|
|  | 2487 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2488 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2489 | _printf("\n[DEBUG FAT] _file_info_no_cache(): search name <%s>" | 
|---|
| [587] | 2490 | " in cluster %x\n", name , parent_cluster ); | 
|---|
|  | 2491 | #endif | 
|---|
|  | 2492 | found  = 0; | 
|---|
|  | 2493 |  | 
|---|
|  | 2494 | // scan clusters containing the parent directory | 
|---|
|  | 2495 | while ( found == 0 ) | 
|---|
|  | 2496 | { | 
|---|
|  | 2497 | // compute lba | 
|---|
|  | 2498 | unsigned int lba = _cluster_to_lba( parent_cluster ); | 
|---|
|  | 2499 |  | 
|---|
|  | 2500 | // load one cluster of the parent directory into data_buffer | 
|---|
|  | 2501 | if ( _fat_buffer_data_lba != lba ) | 
|---|
|  | 2502 | { | 
|---|
|  | 2503 | if ( _fat_ioc_access( 0,         // no descheduling | 
|---|
|  | 2504 | 1,         // read | 
|---|
|  | 2505 | lba, | 
|---|
|  | 2506 | (unsigned int)_fat_buffer_data, | 
|---|
|  | 2507 | 8 ) ) | 
|---|
|  | 2508 | { | 
|---|
| [664] | 2509 | _printf("\n[FAT ERROR] _file_info_no_cache(): " | 
|---|
| [587] | 2510 | "cannot load lba = %x into data_buffer\n", lba ); | 
|---|
|  | 2511 | return 1; | 
|---|
|  | 2512 | } | 
|---|
|  | 2513 |  | 
|---|
|  | 2514 | _fat_buffer_data_lba = lba; | 
|---|
|  | 2515 | } | 
|---|
|  | 2516 |  | 
|---|
|  | 2517 | offset = 0; | 
|---|
|  | 2518 |  | 
|---|
|  | 2519 | // scan this 4 Kbytes buffer | 
|---|
|  | 2520 | while ( (offset < 4096) && (found == 0) ) | 
|---|
|  | 2521 | { | 
|---|
|  | 2522 | buf  = _fat_buffer_data + offset; | 
|---|
|  | 2523 | attr = _read_entry( DIR_ATTR , buf , 0 ); | 
|---|
|  | 2524 | ord  = _read_entry( LDIR_ORD , buf , 0 ); | 
|---|
|  | 2525 |  | 
|---|
|  | 2526 | if (ord == NO_MORE_ENTRY)               // no more entry => break | 
|---|
|  | 2527 | { | 
|---|
|  | 2528 | found = 2; | 
|---|
|  | 2529 | } | 
|---|
|  | 2530 | else if ( ord == FREE_ENTRY )           // free entry => skip | 
|---|
|  | 2531 | { | 
|---|
|  | 2532 | offset = offset + 32; | 
|---|
|  | 2533 | } | 
|---|
|  | 2534 | else if ( attr == ATTR_LONG_NAME_MASK ) // LFN entry => get partial name | 
|---|
|  | 2535 | { | 
|---|
|  | 2536 | unsigned int seq = ord & 0x3; | 
|---|
|  | 2537 | lfn = (seq > lfn) ? seq : lfn; | 
|---|
|  | 2538 | if      ( seq == 1 ) _get_name_from_long( buf, lfn1 ); | 
|---|
|  | 2539 | else if ( seq == 2 ) _get_name_from_long( buf, lfn2 ); | 
|---|
|  | 2540 | else if ( seq == 3 ) _get_name_from_long( buf, lfn3 ); | 
|---|
|  | 2541 | offset = offset + 32; | 
|---|
|  | 2542 | } | 
|---|
|  | 2543 | else                                    // NORMAL entry | 
|---|
|  | 2544 | { | 
|---|
|  | 2545 | // build the full mame for current directory entry | 
|---|
|  | 2546 | if      ( lfn == 0 ) | 
|---|
|  | 2547 | { | 
|---|
|  | 2548 | _get_name_from_short( buf , cname ); | 
|---|
|  | 2549 | } | 
|---|
|  | 2550 | else if ( lfn == 1 ) | 
|---|
|  | 2551 | { | 
|---|
|  | 2552 | _strcpy( cname      , lfn1 ); | 
|---|
|  | 2553 | } | 
|---|
|  | 2554 | else if ( lfn == 2 ) | 
|---|
|  | 2555 | { | 
|---|
|  | 2556 | _strcpy( cname      , lfn1 ); | 
|---|
|  | 2557 | _strcpy( cname + 13 , lfn2 ); | 
|---|
|  | 2558 | } | 
|---|
|  | 2559 | else if ( lfn == 3 ) | 
|---|
|  | 2560 | { | 
|---|
|  | 2561 | _strcpy( cname      , lfn1 ); | 
|---|
|  | 2562 | _strcpy( cname + 13 , lfn2 ); | 
|---|
|  | 2563 | _strcpy( cname + 26 , lfn3 ); | 
|---|
|  | 2564 | } | 
|---|
|  | 2565 |  | 
|---|
|  | 2566 | // test if extracted name == searched name | 
|---|
| [619] | 2567 | if ( _strcmp( name , cname ) == 0 ) | 
|---|
| [587] | 2568 | { | 
|---|
|  | 2569 | child_cluster = (_read_entry( DIR_FST_CLUS_HI , buf , 1 ) << 16) | | 
|---|
|  | 2570 | (_read_entry( DIR_FST_CLUS_LO , buf , 1 )      ) ; | 
|---|
|  | 2571 | child_is_dir  = ((attr & ATTR_DIRECTORY) == ATTR_DIRECTORY); | 
|---|
|  | 2572 | child_size    = _read_entry( DIR_FILE_SIZE , buf , 1 ); | 
|---|
|  | 2573 | found         = 1; | 
|---|
|  | 2574 | } | 
|---|
|  | 2575 | offset = offset + 32; | 
|---|
|  | 2576 | lfn = 0; | 
|---|
|  | 2577 | } | 
|---|
|  | 2578 | }  // en loop on directory entries | 
|---|
|  | 2579 |  | 
|---|
|  | 2580 | // compute next cluster index | 
|---|
|  | 2581 | unsigned int next; | 
|---|
|  | 2582 | if ( _next_cluster_no_cache ( parent_cluster , &next ) ) return 1; | 
|---|
|  | 2583 | parent_cluster = next; | 
|---|
|  | 2584 | } // end loop on clusters | 
|---|
|  | 2585 |  | 
|---|
|  | 2586 | if ( found == 2 )  // found end of directory => error | 
|---|
|  | 2587 | { | 
|---|
| [664] | 2588 | _printf("\n[FAT ERROR] _file_info_no_cache(): <%s> not found\n", | 
|---|
| [587] | 2589 | name ); | 
|---|
|  | 2590 | return 1; | 
|---|
|  | 2591 | } | 
|---|
|  | 2592 |  | 
|---|
|  | 2593 | // check type | 
|---|
|  | 2594 | if ( ((pathname[nb_read] == 0) && (child_is_dir != 0)) || | 
|---|
|  | 2595 | ((pathname[nb_read] != 0) && (child_is_dir == 0)) ) | 
|---|
|  | 2596 | { | 
|---|
| [664] | 2597 | _printf("\n[FAT ERROR] _file_info_no_cache(): illegal type for <%s>\n", name ); | 
|---|
| [587] | 2598 | return 1; | 
|---|
|  | 2599 | } | 
|---|
|  | 2600 |  | 
|---|
|  | 2601 | // update parent_cluster for next name | 
|---|
|  | 2602 | parent_cluster = child_cluster; | 
|---|
|  | 2603 |  | 
|---|
|  | 2604 | }  // end loop on names | 
|---|
|  | 2605 |  | 
|---|
|  | 2606 | #if (GIET_DEBUG_FAT & 1) | 
|---|
|  | 2607 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2608 | _printf("\n[DEBUG FAT] _file_info_no_cache(): success for <%s> / " | 
|---|
| [587] | 2609 | "file_size = %x / file_cluster = %x\n", pathname, child_size, child_cluster ); | 
|---|
|  | 2610 | #endif | 
|---|
|  | 2611 |  | 
|---|
|  | 2612 | // return file cluster and size | 
|---|
|  | 2613 | *file_size    = child_size; | 
|---|
|  | 2614 | *file_cluster = child_cluster; | 
|---|
|  | 2615 | return 0; | 
|---|
|  | 2616 |  | 
|---|
|  | 2617 | }  // end _file_info_no_cache() | 
|---|
|  | 2618 |  | 
|---|
|  | 2619 |  | 
|---|
|  | 2620 |  | 
|---|
|  | 2621 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 2622 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 2623 | //             Extern functions | 
|---|
|  | 2624 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 2625 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 2626 |  | 
|---|
|  | 2627 |  | 
|---|
|  | 2628 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 2629 | // This function initializes the FAT structures. | 
|---|
| [674] | 2630 | // - The Fat-Descriptor is always initialized. | 
|---|
| [587] | 2631 | // - The dynamically allocated structures (the Inode-Tre, the Fat_Cache, | 
|---|
|  | 2632 | //   and the File-Cache for the root directory) are only allocated | 
|---|
| [674] | 2633 | //   and initialized if the "kernel_mode" argument is set. | 
|---|
| [587] | 2634 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 2635 | // Implementation note: | 
|---|
|  | 2636 | // This function is called twice, by the boot-loader, and by the kernel_init. | 
|---|
|  | 2637 | // It does not use dynamic memory allocation from the distributed heap. | 
|---|
|  | 2638 | // It use informations found in the boot sector and FS-INFO sector. | 
|---|
|  | 2639 | // that are loaded in the Fat-Descriptor temporary block_buffer. | 
|---|
|  | 2640 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 2641 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 2642 | // Returns a negative value on error: | 
|---|
|  | 2643 | //   GIET_FAT32_IO_ERROR, | 
|---|
| [674] | 2644 | //   GIET_FAT32_INVALID_BOOT_SECTOR | 
|---|
| [587] | 2645 | ///////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 2646 | int _fat_init( unsigned int kernel_mode ) | 
|---|
|  | 2647 | { | 
|---|
|  | 2648 |  | 
|---|
| [258] | 2649 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 2650 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2651 | _printf("\n[DEBUG FAT] _fat_init(): enters at cycle %d\n", _get_proctime() ); | 
|---|
| [258] | 2652 | #endif | 
|---|
| [458] | 2653 |  | 
|---|
|  | 2654 | // FAT initialisation should be done only once | 
|---|
| [674] | 2655 | if ( _fat.initialized == FAT_INITIALIZED ) | 
|---|
| [458] | 2656 | { | 
|---|
| [664] | 2657 | _printf("\n[FAT WARNING] _fat_init(): FAT already initialized\n"); | 
|---|
|  | 2658 | return GIET_FAT32_OK; | 
|---|
| [458] | 2659 | } | 
|---|
|  | 2660 |  | 
|---|
| [587] | 2661 | // load Boot sector (VBR) into FAT buffer | 
|---|
|  | 2662 | if ( _fat_ioc_access( 0,                                  // no descheduling | 
|---|
|  | 2663 | 1,                                  // read | 
|---|
|  | 2664 | 0,                                  // block index | 
|---|
|  | 2665 | (unsigned int)_fat.block_buffer, | 
|---|
|  | 2666 | 1 ) )                               // one block | 
|---|
| [258] | 2667 | { | 
|---|
| [664] | 2668 | _printf("\n[FAT ERROR] _fat_init(): cannot load VBR\n"); | 
|---|
|  | 2669 | return GIET_FAT32_IO_ERROR; | 
|---|
| [258] | 2670 | } | 
|---|
|  | 2671 |  | 
|---|
| [587] | 2672 | _fat.block_buffer_lba = 0; | 
|---|
|  | 2673 |  | 
|---|
|  | 2674 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 2675 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [587] | 2676 | { | 
|---|
| [664] | 2677 | _printf("\n[DEBUG FAT] _fat_init(): Boot sector loaded\n"); | 
|---|
| [587] | 2678 | _display_one_block( _fat.block_buffer, "block device", _fat.block_buffer_lba ); | 
|---|
|  | 2679 | } | 
|---|
| [258] | 2680 | #endif | 
|---|
|  | 2681 |  | 
|---|
|  | 2682 | // checking various FAT32 assuptions from boot sector | 
|---|
| [587] | 2683 | if( _read_entry( BPB_BYTSPERSEC, _fat.block_buffer, 1 ) != 512 ) | 
|---|
| [258] | 2684 | { | 
|---|
| [664] | 2685 | _printf("\n[FAT ERROR] _fat_init(): The sector size must be 512 bytes\n"); | 
|---|
|  | 2686 | return GIET_FAT32_INVALID_BOOT_SECTOR; | 
|---|
| [258] | 2687 | } | 
|---|
| [587] | 2688 | if( _read_entry( BPB_SECPERCLUS, _fat.block_buffer, 1 ) != 8 ) | 
|---|
| [258] | 2689 | { | 
|---|
| [664] | 2690 | _printf("\n[FAT ERROR] _fat_init(): The cluster size must be 8 blocks\n"); | 
|---|
|  | 2691 | return GIET_FAT32_INVALID_BOOT_SECTOR; | 
|---|
| [587] | 2692 | } | 
|---|
|  | 2693 | if( _read_entry( BPB_NUMFATS, _fat.block_buffer, 1 ) != 1 ) | 
|---|
|  | 2694 | { | 
|---|
| [664] | 2695 | _printf("\n[FAT ERROR] _fat_init(): The number of FAT copies in FAT region must be 1\n"); | 
|---|
|  | 2696 | return GIET_FAT32_INVALID_BOOT_SECTOR; | 
|---|
| [258] | 2697 | } | 
|---|
| [587] | 2698 | if( (_read_entry( BPB_FAT32_FATSZ32, _fat.block_buffer, 1 ) & 0xF) != 0 ) | 
|---|
| [258] | 2699 | { | 
|---|
| [664] | 2700 | _printf("\n[FAT ERROR] _fat_init(): The FAT region must be multiple of 16 sectors\n"); | 
|---|
|  | 2701 | return GIET_FAT32_INVALID_BOOT_SECTOR; | 
|---|
| [258] | 2702 | } | 
|---|
| [587] | 2703 | if( _read_entry( BPB_FAT32_ROOTCLUS, _fat.block_buffer, 1 ) != 2 ) | 
|---|
| [258] | 2704 | { | 
|---|
| [664] | 2705 | _printf("\n[FAT ERROR] _fat_init(): The root directory must be at cluster 2\n"); | 
|---|
|  | 2706 | return GIET_FAT32_INVALID_BOOT_SECTOR; | 
|---|
| [258] | 2707 | } | 
|---|
|  | 2708 |  | 
|---|
| [587] | 2709 | // initialise Fat-Descriptor from VBR | 
|---|
|  | 2710 | _fat.sector_size         = 512; | 
|---|
|  | 2711 | _fat.cluster_size        = 4096; | 
|---|
|  | 2712 | _fat.fat_sectors         = _read_entry( BPB_FAT32_FATSZ32 , _fat.block_buffer , 1 ); | 
|---|
|  | 2713 | _fat.fat_lba             = _read_entry( BPB_RSVDSECCNT , _fat.block_buffer , 1 ); | 
|---|
|  | 2714 | _fat.data_sectors        = _fat.fat_sectors << 10; | 
|---|
| [530] | 2715 | _fat.data_lba            = _fat.fat_lba + _fat.fat_sectors; | 
|---|
| [587] | 2716 | _fat.fs_info_lba         = _read_entry( BPB_FAT32_FSINFO , _fat.block_buffer , 1 ); | 
|---|
|  | 2717 | _fat_buffer_fat_lba      = 0xFFFFFFFF; | 
|---|
|  | 2718 | _fat_buffer_data_lba     = 0xFFFFFFFF; | 
|---|
| [674] | 2719 | _fat.initialized         = FAT_INITIALIZED; | 
|---|
| [258] | 2720 |  | 
|---|
| [587] | 2721 | // load FS_INFO sector into FAT buffer | 
|---|
|  | 2722 | if ( _fat_ioc_access( 0,                                // no descheduling | 
|---|
|  | 2723 | 1,                                // read | 
|---|
|  | 2724 | _fat.fs_info_lba,                 // lba | 
|---|
|  | 2725 | (unsigned int)_fat.block_buffer, | 
|---|
|  | 2726 | 1 ) )                             // one block | 
|---|
|  | 2727 | { | 
|---|
| [664] | 2728 | _printf("\n[FAT ERROR] _fat_init(): cannot load FS_INFO Sector\n"); | 
|---|
|  | 2729 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 2730 | } | 
|---|
| [458] | 2731 |  | 
|---|
| [587] | 2732 | _fat.block_buffer_lba = _fat.fs_info_lba; | 
|---|
| [258] | 2733 |  | 
|---|
| [291] | 2734 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 2735 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [587] | 2736 | { | 
|---|
| [664] | 2737 | _printf("\n[DEBUG FAT] _fat_init(): FS-INFO sector loaded\n"); | 
|---|
| [587] | 2738 | _display_one_block( _fat.block_buffer, "block device", _fat.block_buffer_lba ); | 
|---|
|  | 2739 | } | 
|---|
| [291] | 2740 | #endif | 
|---|
|  | 2741 |  | 
|---|
| [587] | 2742 | // initialise Fat-Descriptor from FS_INFO | 
|---|
|  | 2743 | _fat.free_clusters_number   = _read_entry( FS_FREE_CLUSTERS    , _fat.block_buffer, 1); | 
|---|
|  | 2744 | _fat.first_free_cluster     = _read_entry( FS_FREE_CLUSTER_HINT, _fat.block_buffer, 1); | 
|---|
| [291] | 2745 |  | 
|---|
| [587] | 2746 | // This is done only when the _fat_init() is called in kernel mode | 
|---|
| [291] | 2747 |  | 
|---|
| [587] | 2748 | if ( kernel_mode ) | 
|---|
|  | 2749 | { | 
|---|
| [652] | 2750 | unsigned int i; | 
|---|
| [587] | 2751 |  | 
|---|
| [652] | 2752 | // create Inode-Tree root | 
|---|
|  | 2753 | _fat.inode_tree_root = _allocate_one_inode("/", // dir name | 
|---|
|  | 2754 | 1,   // directory | 
|---|
|  | 2755 | 2,   // cluster id | 
|---|
|  | 2756 | 0,   // no size | 
|---|
|  | 2757 | 0,   // no children | 
|---|
|  | 2758 | 0,   // no dentry | 
|---|
|  | 2759 | 1);  // allocate cache | 
|---|
| [587] | 2760 |  | 
|---|
| [652] | 2761 | // initialize lock | 
|---|
| [587] | 2762 | _spin_lock_init( &_fat.fat_lock ); | 
|---|
|  | 2763 |  | 
|---|
| [652] | 2764 | // initialize File Descriptor Array | 
|---|
|  | 2765 | for( i = 0 ; i < GIET_OPEN_FILES_MAX ; i++ ) _fat.fd[i].allocated = 0; | 
|---|
| [587] | 2766 |  | 
|---|
| [652] | 2767 | // initialize fat_cache root | 
|---|
| [650] | 2768 | _fat.fat_cache_root   = _allocate_one_cache_node( NULL ); | 
|---|
| [587] | 2769 | _fat.fat_cache_levels = _get_levels_from_size( _fat.fat_sectors << 9 ); | 
|---|
|  | 2770 | }  // end if kernel_mode | 
|---|
|  | 2771 |  | 
|---|
| [291] | 2772 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 2773 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [587] | 2774 | _display_fat_descriptor(); | 
|---|
| [291] | 2775 | #endif | 
|---|
|  | 2776 |  | 
|---|
| [664] | 2777 | return GIET_FAT32_OK; | 
|---|
| [258] | 2778 | }  // end _fat_init() | 
|---|
|  | 2779 |  | 
|---|
| [587] | 2780 |  | 
|---|
|  | 2781 |  | 
|---|
|  | 2782 |  | 
|---|
| [258] | 2783 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 2784 | // This function implements the giet_fat_open() system call. | 
|---|
|  | 2785 | // The semantic is similar to the UNIX open() function, but only the O_CREATE | 
|---|
|  | 2786 | // and O_RDONLY flags are supported. The UNIX access rights are not supported. | 
|---|
|  | 2787 | // If the file does not exist in the specified directory, it is created. | 
|---|
|  | 2788 | // If the specified directory does not exist, an error is returned. | 
|---|
|  | 2789 | // It allocates a file descriptor to the calling task, for the file identified | 
|---|
|  | 2790 | // by "pathname". If several tasks try to open the same file, each task | 
|---|
|  | 2791 | // obtains a private file descriptor. | 
|---|
|  | 2792 | // A node name (file or directory) cannot be larger than 31 characters. | 
|---|
| [258] | 2793 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 2794 | // Returns a file descriptor index on success. | 
|---|
|  | 2795 | // Returns a negative value on error: | 
|---|
|  | 2796 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 2797 | //   GIET_FAT32_FILE_NOT_FOUND, | 
|---|
|  | 2798 | //   GIET_FAT32_NAME_TOO_LONG, | 
|---|
|  | 2799 | //   GIET_FAT32_IO_ERROR, | 
|---|
|  | 2800 | //   GIET_FAT32_TOO_MANY_OPEN_FILES | 
|---|
| [258] | 2801 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 2802 | int _fat_open( char*        pathname,     // absolute path from root | 
|---|
|  | 2803 | unsigned int flags )       // O_CREATE and O_RDONLY | 
|---|
| [258] | 2804 | { | 
|---|
| [587] | 2805 | unsigned int         fd_id;            // index in File-Descriptor-Array | 
|---|
|  | 2806 | unsigned int         code;             // error code | 
|---|
|  | 2807 | fat_inode_t*         inode;            // anonymous inode pointer | 
|---|
|  | 2808 | fat_inode_t*         child;            // pointer on searched file inode | 
|---|
|  | 2809 | fat_inode_t*         parent;           // pointer on parent directory inode | 
|---|
| [258] | 2810 |  | 
|---|
| [587] | 2811 | // get flags | 
|---|
|  | 2812 | unsigned int create    = ((flags & O_CREATE) != 0); | 
|---|
| [654] | 2813 | unsigned int read_only = ((flags & O_RDONLY) != 0); | 
|---|
|  | 2814 | unsigned int truncate  = ((flags & O_TRUNC)  != 0); | 
|---|
| [587] | 2815 |  | 
|---|
| [258] | 2816 | #if GIET_DEBUG_FAT | 
|---|
| [295] | 2817 | unsigned int procid  = _get_procid(); | 
|---|
| [429] | 2818 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 2819 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 2820 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
| [569] | 2821 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2822 | _printf("\n[DEBUG FAT] _fat_open(): P[%d,%d,%d] enters for path <%s> / " | 
|---|
| [587] | 2823 | " create = %d / read_only = %d\n", | 
|---|
|  | 2824 | x, y, p, pathname , create , read_only ); | 
|---|
| [258] | 2825 | #endif | 
|---|
|  | 2826 |  | 
|---|
| [674] | 2827 | // checking FAT initialized | 
|---|
|  | 2828 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [458] | 2829 | { | 
|---|
| [664] | 2830 | _printf("\n[FAT ERROR] _fat_open(): FAT not initialized\n"); | 
|---|
|  | 2831 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [458] | 2832 | } | 
|---|
| [587] | 2833 |  | 
|---|
|  | 2834 | // takes the lock | 
|---|
| [530] | 2835 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
| [295] | 2836 |  | 
|---|
| [587] | 2837 | // get inode pointer | 
|---|
|  | 2838 | code = _get_inode_from_path( pathname , &inode ); | 
|---|
|  | 2839 |  | 
|---|
|  | 2840 | if ( code == 2 ) | 
|---|
| [258] | 2841 | { | 
|---|
| [587] | 2842 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2843 | _printf("\n[FAT ERROR] _fat_open(): path to parent not found" | 
|---|
| [587] | 2844 | " for file <%s>\n", pathname ); | 
|---|
| [664] | 2845 | return GIET_FAT32_FILE_NOT_FOUND; | 
|---|
| [587] | 2846 | } | 
|---|
|  | 2847 | else if ( code == 3 ) | 
|---|
|  | 2848 | { | 
|---|
|  | 2849 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2850 | _printf("\n[FAT ERROR] _fat_open(): one name in path too long" | 
|---|
| [587] | 2851 | " for file <%s>\n", pathname ); | 
|---|
| [664] | 2852 | return GIET_FAT32_NAME_TOO_LONG; | 
|---|
| [587] | 2853 | } | 
|---|
|  | 2854 | else if ( (code == 1) && (create == 0) ) | 
|---|
|  | 2855 | { | 
|---|
|  | 2856 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2857 | _printf("\n[FAT ERROR] _fat_open(): file not found" | 
|---|
| [587] | 2858 | " for file <%s>\n", pathname ); | 
|---|
| [664] | 2859 | return GIET_FAT32_FILE_NOT_FOUND; | 
|---|
| [587] | 2860 | } | 
|---|
|  | 2861 | else if ( (code == 1) && (create != 0) )   // file name not found => create | 
|---|
|  | 2862 | { | 
|---|
|  | 2863 | // set parent inode pointer | 
|---|
|  | 2864 | parent = inode; | 
|---|
| [258] | 2865 |  | 
|---|
|  | 2866 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 2867 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2868 | _printf("\n[DEBUG FAT] _fat_open(): P[%d,%d,%d] create a new file <%s>\n", | 
|---|
| [587] | 2869 | x , y , p , pathname ); | 
|---|
| [258] | 2870 | #endif | 
|---|
| [291] | 2871 |  | 
|---|
| [587] | 2872 | // get new file name / error check already done by _get_inode_from_path() | 
|---|
|  | 2873 | char name[32]; | 
|---|
|  | 2874 | _get_last_name( pathname , name ); | 
|---|
|  | 2875 |  | 
|---|
|  | 2876 | // allocate a new inode and an empty Cache-File | 
|---|
|  | 2877 | child = _allocate_one_inode( name, | 
|---|
|  | 2878 | 0,                         // not a directory | 
|---|
|  | 2879 | END_OF_CHAIN_CLUSTER_MAX,  // no cluster allocated | 
|---|
|  | 2880 | 0,                         // size : new file is empty | 
|---|
|  | 2881 | 0,                         // count incremented later | 
|---|
|  | 2882 | 0,                         // dentry set by add_dir_entry | 
|---|
|  | 2883 | 1 );                       // cache_allocate | 
|---|
|  | 2884 |  | 
|---|
|  | 2885 | // introduce inode into Inode-Tree | 
|---|
|  | 2886 | _add_inode_in_tree( child , parent ); | 
|---|
|  | 2887 |  | 
|---|
|  | 2888 | // add an entry in the parent directory Cache_file | 
|---|
|  | 2889 | if ( _add_dir_entry( child , parent ) ) | 
|---|
| [258] | 2890 | { | 
|---|
| [587] | 2891 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2892 | _printf("\n[FAT ERROR] _fat_open(): cannot update parent directory" | 
|---|
| [587] | 2893 | " for file <%s>\n" , pathname ); | 
|---|
| [664] | 2894 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 2895 | } | 
|---|
|  | 2896 |  | 
|---|
|  | 2897 | // update DATA region on block device for parent directory | 
|---|
|  | 2898 | if ( _update_device_from_cache( parent->levels, | 
|---|
|  | 2899 | parent->cache, | 
|---|
|  | 2900 | parent->name ) ) | 
|---|
|  | 2901 | { | 
|---|
|  | 2902 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2903 | _printf("\n[FAT ERROR] _fat_open(): cannot update DATA region " | 
|---|
| [587] | 2904 | " for parent of file <%s>\n", pathname ); | 
|---|
| [664] | 2905 | return GIET_FAT32_IO_ERROR; | 
|---|
| [258] | 2906 | } | 
|---|
|  | 2907 |  | 
|---|
| [587] | 2908 | // update FAT region on block device | 
|---|
|  | 2909 | if ( _update_device_from_cache( _fat.fat_cache_levels, | 
|---|
|  | 2910 | _fat.fat_cache_root, | 
|---|
|  | 2911 | "FAT" ) ) | 
|---|
| [258] | 2912 | { | 
|---|
| [587] | 2913 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2914 | _printf("\n[FAT ERROR] _fat_open(): cannot update FAT region" | 
|---|
| [587] | 2915 | " for file <%s>\n", pathname ); | 
|---|
| [664] | 2916 | return GIET_FAT32_IO_ERROR; | 
|---|
| [258] | 2917 | } | 
|---|
| [587] | 2918 |  | 
|---|
|  | 2919 | // update FS_INFO sector | 
|---|
|  | 2920 | if ( _update_fs_info() ) | 
|---|
| [258] | 2921 | { | 
|---|
| [530] | 2922 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2923 | _printf("\n[FAT ERROR] _fat_open(): cannot update FS-INFO" | 
|---|
| [587] | 2924 | " for file <%s>\n", pathname ); | 
|---|
| [664] | 2925 | return GIET_FAT32_IO_ERROR; | 
|---|
| [258] | 2926 | } | 
|---|
| [654] | 2927 |  | 
|---|
|  | 2928 | // no need to truncate a new file | 
|---|
|  | 2929 | truncate = 0; | 
|---|
| [258] | 2930 | } | 
|---|
| [587] | 2931 | else // code == 0 | 
|---|
|  | 2932 | { | 
|---|
|  | 2933 | // set searched file inode pointer | 
|---|
|  | 2934 | child = inode; | 
|---|
| [258] | 2935 |  | 
|---|
| [417] | 2936 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 2937 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2938 | _printf("\n[DEBUG FAT] _fat_open(): P[%d,%d,%d] found file <%s> on device : inode = %x\n", | 
|---|
| [587] | 2939 | x , y , p , pathname , child ); | 
|---|
| [417] | 2940 | #endif | 
|---|
| [587] | 2941 | } | 
|---|
| [258] | 2942 |  | 
|---|
| [587] | 2943 | // Search an empty slot in file descriptors array | 
|---|
|  | 2944 | fd_id = 0; | 
|---|
|  | 2945 | while ( (_fat.fd[fd_id].allocated) != 0 && (fd_id < GIET_OPEN_FILES_MAX) ) | 
|---|
| [258] | 2946 | { | 
|---|
| [587] | 2947 | fd_id++; | 
|---|
|  | 2948 | } | 
|---|
|  | 2949 |  | 
|---|
|  | 2950 | // set file descriptor if an empty slot has been found | 
|---|
| [654] | 2951 | if ( fd_id >= GIET_OPEN_FILES_MAX ) | 
|---|
| [587] | 2952 | { | 
|---|
| [654] | 2953 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2954 | _printf("\n[FAT ERROR] _fat_open(): File-Descriptors-Array full\n"); | 
|---|
|  | 2955 | return GIET_FAT32_TOO_MANY_OPEN_FILES; | 
|---|
| [654] | 2956 | } | 
|---|
| [587] | 2957 |  | 
|---|
| [654] | 2958 | // update file descriptor | 
|---|
|  | 2959 | _fat.fd[fd_id].allocated  = 1; | 
|---|
|  | 2960 | _fat.fd[fd_id].seek       = 0; | 
|---|
|  | 2961 | _fat.fd[fd_id].read_only  = read_only; | 
|---|
|  | 2962 | _fat.fd[fd_id].inode      = child; | 
|---|
| [587] | 2963 |  | 
|---|
| [654] | 2964 | // increment the refcount | 
|---|
|  | 2965 | child->count = child->count + 1; | 
|---|
| [587] | 2966 |  | 
|---|
| [654] | 2967 | // truncate the file if requested | 
|---|
|  | 2968 | if ( truncate && !read_only && !child->is_dir ) | 
|---|
|  | 2969 | { | 
|---|
|  | 2970 | // empty file | 
|---|
|  | 2971 | child->size = 0; | 
|---|
|  | 2972 | child->levels = _get_levels_from_size( child->size ); | 
|---|
|  | 2973 |  | 
|---|
|  | 2974 | // release File-Cache (keep root node) | 
|---|
|  | 2975 | _release_cache_memory( child->cache, child->levels ); | 
|---|
|  | 2976 |  | 
|---|
|  | 2977 | // release clusters allocated to file/dir in DATA region | 
|---|
|  | 2978 | if ( _clusters_release( child->cluster ) ) | 
|---|
|  | 2979 | { | 
|---|
|  | 2980 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2981 | _printf("\n[FAT ERROR] _fat_open(): can't truncate file\n"); | 
|---|
|  | 2982 | return GIET_FAT32_IO_ERROR; | 
|---|
| [654] | 2983 | } | 
|---|
|  | 2984 |  | 
|---|
|  | 2985 | // update parent directory entry (size and cluster index) | 
|---|
|  | 2986 | if ( _update_dir_entry( child ) ) | 
|---|
|  | 2987 | { | 
|---|
|  | 2988 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 2989 | _printf("\n[FAT ERROR] _fat_open(): can't truncate file\n"); | 
|---|
|  | 2990 | return GIET_FAT32_IO_ERROR; | 
|---|
| [654] | 2991 | } | 
|---|
|  | 2992 | } | 
|---|
|  | 2993 |  | 
|---|
|  | 2994 | // releases the lock | 
|---|
|  | 2995 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 2996 |  | 
|---|
| [587] | 2997 | #if GIET_DEBUG_FAT | 
|---|
|  | 2998 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 2999 | _printf("\n[DEBUG FAT] _fat_open(): P[%d,%d,%d] got fd = %d for <%s> / " | 
|---|
| [587] | 3000 | "read_only = %d\n", | 
|---|
|  | 3001 | x , y , p , fd_id , pathname , read_only ); | 
|---|
|  | 3002 | #endif | 
|---|
| [654] | 3003 | return fd_id; | 
|---|
| [587] | 3004 | } // end _fat_open() | 
|---|
|  | 3005 |  | 
|---|
|  | 3006 |  | 
|---|
|  | 3007 |  | 
|---|
|  | 3008 |  | 
|---|
|  | 3009 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3010 | // This function implements the "giet_fat_close()" system call. | 
|---|
|  | 3011 | // It decrements the inode reference count, and release the fd_id entry | 
|---|
|  | 3012 | // in the file descriptors array. | 
|---|
|  | 3013 | // If the reference count is zero, it writes all dirty clusters on block device, | 
|---|
| [651] | 3014 | // and releases the memory allocated to the File_Cache. | 
|---|
| [587] | 3015 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 3016 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 3017 | // Returns a negative value on error: | 
|---|
|  | 3018 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 3019 | //   GIET_FAT32_INVALID_FD, | 
|---|
|  | 3020 | //   GIET_FAT32_NOT_OPEN, | 
|---|
|  | 3021 | //   GIET_FAT32_IO_ERROR | 
|---|
| [587] | 3022 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3023 | int _fat_close( unsigned int fd_id ) | 
|---|
|  | 3024 | { | 
|---|
| [674] | 3025 | // checking FAT initialized | 
|---|
|  | 3026 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [587] | 3027 | { | 
|---|
| [664] | 3028 | _printf("\n[FAT ERROR] _fat_close(): FAT not initialized\n"); | 
|---|
|  | 3029 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [587] | 3030 | } | 
|---|
|  | 3031 |  | 
|---|
|  | 3032 | if( (fd_id >= GIET_OPEN_FILES_MAX) ) | 
|---|
|  | 3033 | { | 
|---|
| [664] | 3034 | _printf("\n[FAT ERROR] _fat_close(): illegal file descriptor index\n"); | 
|---|
|  | 3035 | return GIET_FAT32_INVALID_FD; | 
|---|
| [587] | 3036 | } | 
|---|
|  | 3037 |  | 
|---|
|  | 3038 | // takes lock | 
|---|
|  | 3039 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
|  | 3040 |  | 
|---|
| [625] | 3041 | if( _fat.fd[fd_id].allocated == 0 ) | 
|---|
|  | 3042 | { | 
|---|
|  | 3043 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3044 | _printf("\n[FAT ERROR] _fat_close(): file not open\n"); | 
|---|
|  | 3045 | return GIET_FAT32_NOT_OPEN; | 
|---|
| [625] | 3046 | } | 
|---|
|  | 3047 |  | 
|---|
| [587] | 3048 | // get the inode pointer | 
|---|
|  | 3049 | fat_inode_t*  inode = _fat.fd[fd_id].inode; | 
|---|
|  | 3050 |  | 
|---|
|  | 3051 | // decrement reference count | 
|---|
|  | 3052 | inode->count = inode->count - 1; | 
|---|
|  | 3053 |  | 
|---|
|  | 3054 | #if GIET_DEBUG_FAT | 
|---|
|  | 3055 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
|  | 3056 | _printf("\n[FAT DEBUG] _fat_close() for file <%s> : refcount = %d\n", | 
|---|
|  | 3057 | inode->name , inode->count ); | 
|---|
|  | 3058 | #endif | 
|---|
|  | 3059 |  | 
|---|
|  | 3060 | // update block device and release File-Cache if no more references | 
|---|
|  | 3061 | if ( inode->count == 0 ) | 
|---|
|  | 3062 | { | 
|---|
|  | 3063 | // update all dirty clusters for closed file | 
|---|
|  | 3064 | if ( _update_device_from_cache( inode->levels, | 
|---|
|  | 3065 | inode->cache, | 
|---|
|  | 3066 | inode->name ) ) | 
|---|
| [258] | 3067 | { | 
|---|
| [587] | 3068 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3069 | _printf("\n[FAT ERROR] _fat_close(): cannot write dirty clusters " | 
|---|
| [587] | 3070 | "for file <%s>\n", inode->name ); | 
|---|
| [664] | 3071 | return GIET_FAT32_IO_ERROR; | 
|---|
| [258] | 3072 | } | 
|---|
|  | 3073 |  | 
|---|
| [587] | 3074 | #if GIET_DEBUG_FAT | 
|---|
|  | 3075 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
|  | 3076 | _printf("\n[FAT DEBUG] _fat_close() update device for file <%s>\n", inode->name ); | 
|---|
|  | 3077 | #endif | 
|---|
|  | 3078 |  | 
|---|
|  | 3079 | // update directory dirty clusters for parent directory | 
|---|
| [624] | 3080 | if ( inode->parent && | 
|---|
|  | 3081 | _update_device_from_cache( inode->parent->levels, | 
|---|
| [587] | 3082 | inode->parent->cache, | 
|---|
| [624] | 3083 | inode->parent->name ) ) | 
|---|
| [258] | 3084 | { | 
|---|
| [587] | 3085 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3086 | _printf("\n[FAT ERROR] _fat_close(): cannot write dirty clusters " | 
|---|
| [587] | 3087 | "for directory <%s>\n", inode->parent->name ); | 
|---|
| [664] | 3088 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3089 | } | 
|---|
| [258] | 3090 |  | 
|---|
|  | 3091 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 3092 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [587] | 3093 | _printf("\n[FAT DEBUG] _fat_close() update device for parent directory <%s>\n", | 
|---|
|  | 3094 | inode->parent->name ); | 
|---|
| [258] | 3095 | #endif | 
|---|
|  | 3096 |  | 
|---|
| [651] | 3097 | // release memory allocated to File-Cache (keep cache root node) | 
|---|
|  | 3098 | _release_cache_memory( inode->cache, inode->levels ); | 
|---|
| [587] | 3099 |  | 
|---|
|  | 3100 | #if GIET_DEBUG_FAT | 
|---|
|  | 3101 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
|  | 3102 | _printf("\n[FAT DEBUG] _fat_close() release memory for File-Cache <%s>\n", | 
|---|
|  | 3103 | inode->name ); | 
|---|
|  | 3104 | #endif | 
|---|
|  | 3105 |  | 
|---|
| [258] | 3106 | } | 
|---|
| [587] | 3107 |  | 
|---|
|  | 3108 | // release fd_id entry in file descriptor array | 
|---|
|  | 3109 | _fat.fd[fd_id].allocated = 0; | 
|---|
|  | 3110 |  | 
|---|
|  | 3111 | // release lock | 
|---|
|  | 3112 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 3113 |  | 
|---|
| [664] | 3114 | return GIET_FAT32_OK; | 
|---|
| [587] | 3115 | } // end fat_close() | 
|---|
|  | 3116 |  | 
|---|
|  | 3117 |  | 
|---|
|  | 3118 |  | 
|---|
|  | 3119 |  | 
|---|
|  | 3120 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3121 | // This function implements the giet_fat_file_info() system call. | 
|---|
| [623] | 3122 | // It returns the size, the current offset and the directory info for a file | 
|---|
|  | 3123 | // identified by the "fd_id" argument. | 
|---|
| [587] | 3124 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 3125 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 3126 | // Returns a negative value on error: | 
|---|
|  | 3127 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 3128 | //   GIET_FAT32_INVALID_FD, | 
|---|
|  | 3129 | //   GIET_FAT32_NOT_OPEN | 
|---|
| [587] | 3130 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [658] | 3131 | int _fat_file_info( unsigned int     fd_id, | 
|---|
|  | 3132 | fat_file_info_t* info ) | 
|---|
| [587] | 3133 | { | 
|---|
| [674] | 3134 | if ( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [258] | 3135 | { | 
|---|
| [664] | 3136 | _printf("\n[FAT ERROR] _fat_file_info(): FAT not initialized\n"); | 
|---|
|  | 3137 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [258] | 3138 | } | 
|---|
|  | 3139 |  | 
|---|
| [623] | 3140 | if ( fd_id >= GIET_OPEN_FILES_MAX ) | 
|---|
| [587] | 3141 | { | 
|---|
| [664] | 3142 | _printf("\n[FAT ERROR] _fat_file_info(): illegal file descriptor index\n"); | 
|---|
|  | 3143 | return GIET_FAT32_INVALID_FD; | 
|---|
| [587] | 3144 | } | 
|---|
| [258] | 3145 |  | 
|---|
| [587] | 3146 | if ( _fat.fd[fd_id].allocated == 0 ) | 
|---|
|  | 3147 | { | 
|---|
| [664] | 3148 | _printf("\n[FAT ERROR] _fat_file_info(): file not open\n"); | 
|---|
|  | 3149 | return GIET_FAT32_NOT_OPEN; | 
|---|
| [258] | 3150 | } | 
|---|
| [587] | 3151 |  | 
|---|
| [623] | 3152 | info->size   = _fat.fd[fd_id].inode->size; | 
|---|
|  | 3153 | info->offset = _fat.fd[fd_id].seek; | 
|---|
|  | 3154 | info->is_dir = _fat.fd[fd_id].inode->is_dir; | 
|---|
|  | 3155 |  | 
|---|
| [664] | 3156 | return GIET_FAT32_OK; | 
|---|
| [587] | 3157 | } // end _fat_file_info() | 
|---|
|  | 3158 |  | 
|---|
|  | 3159 |  | 
|---|
|  | 3160 |  | 
|---|
|  | 3161 |  | 
|---|
|  | 3162 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3163 | // The following function implements the "giet_fat_read()" system call. | 
|---|
|  | 3164 | // It transfers "count" bytes from the File_Cache associated to the file | 
|---|
|  | 3165 | // identified by "fd_id", to the user "buffer", from the current file offset. | 
|---|
|  | 3166 | // In case of miss in the File_Cache, it loads all involved clusters into cache. | 
|---|
|  | 3167 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [674] | 3168 | // Returns the number of bytes actually transfered on success. | 
|---|
| [664] | 3169 | // Returns 0 if EOF is encountered (offset + count > file_size). | 
|---|
|  | 3170 | // Returns a negative value on error: | 
|---|
|  | 3171 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 3172 | //   GIET_FAT32_INVALID_FD, | 
|---|
|  | 3173 | //   GIET_FAT32_NOT_OPEN, | 
|---|
|  | 3174 | //   GIET_FAT32_IO_ERROR | 
|---|
| [587] | 3175 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3176 | int _fat_read( unsigned int fd_id,     // file descriptor index | 
|---|
|  | 3177 | void*        buffer,    // destination buffer | 
|---|
|  | 3178 | unsigned int count )    // number of bytes to read | 
|---|
|  | 3179 | { | 
|---|
| [674] | 3180 | // checking FAT initialized | 
|---|
|  | 3181 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [258] | 3182 | { | 
|---|
| [664] | 3183 | _printf("\n[FAT ERROR] _fat_write(): FAT not initialized\n"); | 
|---|
|  | 3184 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [258] | 3185 | } | 
|---|
| [587] | 3186 |  | 
|---|
|  | 3187 | // check fd_id overflow | 
|---|
|  | 3188 | if ( fd_id >= GIET_OPEN_FILES_MAX ) | 
|---|
| [258] | 3189 | { | 
|---|
| [664] | 3190 | _printf("\n[FAT ERROR] _fat_read(): illegal file descriptor\n"); | 
|---|
|  | 3191 | return GIET_FAT32_INVALID_FD; | 
|---|
| [258] | 3192 | } | 
|---|
| [358] | 3193 |  | 
|---|
| [587] | 3194 | // check file is open | 
|---|
|  | 3195 | if ( _fat.fd[fd_id].allocated == 0 ) | 
|---|
|  | 3196 | { | 
|---|
| [664] | 3197 | _printf("\n[FAT ERROR] _fat_read(): file not open\n"); | 
|---|
|  | 3198 | return GIET_FAT32_NOT_OPEN; | 
|---|
| [587] | 3199 | } | 
|---|
| [358] | 3200 |  | 
|---|
| [587] | 3201 | // takes lock | 
|---|
|  | 3202 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
|  | 3203 |  | 
|---|
|  | 3204 | // get file inode pointer and offset | 
|---|
|  | 3205 | fat_inode_t* inode  = _fat.fd[fd_id].inode; | 
|---|
|  | 3206 | unsigned int seek   = _fat.fd[fd_id].seek; | 
|---|
|  | 3207 |  | 
|---|
|  | 3208 | // check count & seek versus file size | 
|---|
| [658] | 3209 | if ( count + seek > inode->size && !inode->is_dir ) | 
|---|
| [258] | 3210 | { | 
|---|
| [587] | 3211 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3212 | _printf("\n[FAT ERROR] _fat_read(): file too small" | 
|---|
| [587] | 3213 | " / seek = %x / count = %x / file_size = %x\n", | 
|---|
|  | 3214 | seek , count , inode->size ); | 
|---|
|  | 3215 | return 0; | 
|---|
| [258] | 3216 | } | 
|---|
|  | 3217 |  | 
|---|
| [587] | 3218 | // compute first_cluster_id and first_byte_to_move | 
|---|
|  | 3219 | unsigned int first_cluster_id   = seek >> 12; | 
|---|
|  | 3220 | unsigned int first_byte_to_move = seek & 0xFFF; | 
|---|
| [258] | 3221 |  | 
|---|
| [587] | 3222 | // compute last_cluster and last_byte_to_move | 
|---|
|  | 3223 | unsigned int last_cluster_id   = (seek + count - 1) >> 12; | 
|---|
|  | 3224 | unsigned int last_byte_to_move = (seek + count - 1) & 0xFFF; | 
|---|
| [258] | 3225 |  | 
|---|
|  | 3226 | #if GIET_DEBUG_FAT | 
|---|
| [354] | 3227 | unsigned int procid  = _get_procid(); | 
|---|
| [429] | 3228 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 3229 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 3230 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
| [569] | 3231 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3232 | _printf("\n[DEBUG FAT] _fat_read(): P[%d,%d,%d] enters for file <%s> " | 
|---|
| [587] | 3233 | " / bytes = %x / offset = %x\n" | 
|---|
|  | 3234 | "first_cluster_id = %x / first_byte_to_move = %x" | 
|---|
|  | 3235 | " / last_cluster_id = %x / last_byte_to_move = %x\n", | 
|---|
|  | 3236 | x , y , p , inode->name , count , seek , | 
|---|
|  | 3237 | first_cluster_id , first_byte_to_move , last_cluster_id , last_byte_to_move ); | 
|---|
| [258] | 3238 | #endif | 
|---|
|  | 3239 |  | 
|---|
| [587] | 3240 | // loop on all cluster covering the requested transfer | 
|---|
|  | 3241 | unsigned int cluster_id; | 
|---|
|  | 3242 | unsigned int done = 0; | 
|---|
|  | 3243 | for ( cluster_id = first_cluster_id ; cluster_id <= last_cluster_id ; cluster_id++ ) | 
|---|
| [258] | 3244 | { | 
|---|
| [587] | 3245 | // get pointer on the cluster_id buffer in cache | 
|---|
|  | 3246 | unsigned char*     cbuf; | 
|---|
|  | 3247 | fat_cache_desc_t*  pdesc; | 
|---|
|  | 3248 | if ( _get_buffer_from_cache( inode, | 
|---|
|  | 3249 | cluster_id, | 
|---|
|  | 3250 | &pdesc ) ) | 
|---|
|  | 3251 | { | 
|---|
|  | 3252 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3253 | _printf("\n[FAT ERROR] _fat_read(): cannot load file <%s>\n", | 
|---|
| [587] | 3254 | inode->name ); | 
|---|
| [664] | 3255 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3256 | } | 
|---|
|  | 3257 | cbuf = pdesc->buffer; | 
|---|
|  | 3258 |  | 
|---|
|  | 3259 | #if GIET_DEBUG_FAT | 
|---|
|  | 3260 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3261 | _printf("\n[DEBUG FAT] _fat_read(): P[%d,%d,%d] moves cluster_id %d from Cache-File <%s>\n", | 
|---|
| [587] | 3262 | x , y , p , cluster_id, inode->name ); | 
|---|
|  | 3263 | #endif | 
|---|
|  | 3264 |  | 
|---|
|  | 3265 | // compute memcpy arguments | 
|---|
|  | 3266 | unsigned char*  source; | 
|---|
|  | 3267 | unsigned int    nbytes; | 
|---|
|  | 3268 | unsigned char*  dest = (unsigned char*)buffer + done; | 
|---|
|  | 3269 | if ( (cluster_id == first_cluster_id) && (cluster_id == last_cluster_id) ) | 
|---|
|  | 3270 | { | 
|---|
|  | 3271 | source = cbuf + first_byte_to_move; | 
|---|
|  | 3272 | nbytes = last_byte_to_move - first_byte_to_move + 1; | 
|---|
|  | 3273 | } | 
|---|
|  | 3274 | else if ( cluster_id == first_cluster_id ) | 
|---|
|  | 3275 | { | 
|---|
|  | 3276 | source = cbuf + first_byte_to_move; | 
|---|
|  | 3277 | nbytes = 4096 - first_byte_to_move; | 
|---|
|  | 3278 | } | 
|---|
|  | 3279 | else if ( cluster_id == last_cluster_id ) | 
|---|
|  | 3280 | { | 
|---|
|  | 3281 | source = cbuf; | 
|---|
|  | 3282 | nbytes = last_byte_to_move + 1; | 
|---|
|  | 3283 | } | 
|---|
|  | 3284 | else  // not first / not last | 
|---|
|  | 3285 | { | 
|---|
|  | 3286 | source = cbuf; | 
|---|
|  | 3287 | nbytes = 4096; | 
|---|
|  | 3288 | } | 
|---|
|  | 3289 |  | 
|---|
|  | 3290 | // move data | 
|---|
|  | 3291 | memcpy( dest , source , nbytes ); | 
|---|
|  | 3292 | done = done + nbytes; | 
|---|
| [258] | 3293 | } | 
|---|
|  | 3294 |  | 
|---|
| [587] | 3295 | #if GIET_DEBUG_FAT | 
|---|
|  | 3296 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3297 | _printf("\n[DEBUG FAT] _fat_read(): P[%d,%d,%d] loaded file <%s> from Cache-File\n", | 
|---|
| [587] | 3298 | x , y , p , inode->name ); | 
|---|
|  | 3299 | #endif | 
|---|
| [258] | 3300 |  | 
|---|
| [587] | 3301 | // update seek | 
|---|
|  | 3302 | _fat.fd[fd_id].seek += done; | 
|---|
| [258] | 3303 |  | 
|---|
| [587] | 3304 | // release lock | 
|---|
|  | 3305 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 3306 |  | 
|---|
|  | 3307 | return done; | 
|---|
|  | 3308 | } // end _fat_read() | 
|---|
|  | 3309 |  | 
|---|
|  | 3310 |  | 
|---|
|  | 3311 |  | 
|---|
|  | 3312 |  | 
|---|
|  | 3313 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3314 | // The following function implements the "giet_fat_write()" system call. | 
|---|
|  | 3315 | // It transfers "count" bytes to the fat_cache associated to the file | 
|---|
|  | 3316 | // identified by "fd_id", from the user "buffer", using the current file offset. | 
|---|
|  | 3317 | // It increases the file size and allocate new clusters if (count + offset) | 
|---|
|  | 3318 | // is larger than the current file size. Then it loads and updates all | 
|---|
|  | 3319 | // involved clusters in the cache. | 
|---|
|  | 3320 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 3321 | // Returns number of bytes actually written on success. | 
|---|
|  | 3322 | // Returns a negative value on error: | 
|---|
|  | 3323 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 3324 | //   GIET_FAT32_INVALID_FD, | 
|---|
|  | 3325 | //   GIET_FAT32_NOT_OPEN, | 
|---|
|  | 3326 | //   GIET_FAT32_READ_ONLY, | 
|---|
|  | 3327 | //   GIET_FAT32_NO_FREE_SPACE, | 
|---|
|  | 3328 | //   GIET_FAT32_IO_ERROR | 
|---|
| [587] | 3329 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3330 | int _fat_write( unsigned int fd_id,    // file descriptor index | 
|---|
|  | 3331 | void*        buffer,   // source buffer | 
|---|
|  | 3332 | unsigned int count )   // number of bytes to write | 
|---|
|  | 3333 | { | 
|---|
| [674] | 3334 | // checking FAT initialized | 
|---|
|  | 3335 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [258] | 3336 | { | 
|---|
| [674] | 3337 | _printf("\n[FAT ERROR] _fat_write(): FAT not initialized\n"); | 
|---|
| [664] | 3338 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [587] | 3339 | } | 
|---|
| [258] | 3340 |  | 
|---|
| [587] | 3341 | // takes lock | 
|---|
|  | 3342 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
|  | 3343 |  | 
|---|
|  | 3344 | // check fd_id overflow | 
|---|
|  | 3345 | if ( fd_id >= GIET_OPEN_FILES_MAX ) | 
|---|
|  | 3346 | { | 
|---|
|  | 3347 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3348 | _printf("\n[FAT ERROR] _fat_write(): illegal file descriptor\n"); | 
|---|
|  | 3349 | return GIET_FAT32_INVALID_FD; | 
|---|
| [587] | 3350 | } | 
|---|
|  | 3351 |  | 
|---|
|  | 3352 | // check file open | 
|---|
|  | 3353 | if ( _fat.fd[fd_id].allocated == 0 ) | 
|---|
|  | 3354 | { | 
|---|
|  | 3355 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3356 | _printf("\n[FAT ERROR] _fat_write(): file not open\n" ); | 
|---|
|  | 3357 | return GIET_FAT32_NOT_OPEN; | 
|---|
| [587] | 3358 | } | 
|---|
|  | 3359 |  | 
|---|
|  | 3360 | // check file writable | 
|---|
|  | 3361 | if ( _fat.fd[fd_id].read_only ) | 
|---|
|  | 3362 | { | 
|---|
|  | 3363 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3364 | _printf("\n[FAT ERROR] _fat_write(): file <%s> is read-only\n", | 
|---|
| [587] | 3365 | _fat.fd[fd_id].inode->name ); | 
|---|
| [664] | 3366 | return GIET_FAT32_READ_ONLY; | 
|---|
| [587] | 3367 | } | 
|---|
|  | 3368 |  | 
|---|
|  | 3369 | // get file inode pointer and seek | 
|---|
|  | 3370 | fat_inode_t* inode  = _fat.fd[fd_id].inode; | 
|---|
|  | 3371 | unsigned int seek   = _fat.fd[fd_id].seek; | 
|---|
|  | 3372 |  | 
|---|
| [258] | 3373 | #if GIET_DEBUG_FAT | 
|---|
| [587] | 3374 | unsigned int procid  = _get_procid(); | 
|---|
|  | 3375 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 3376 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 3377 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
| [569] | 3378 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3379 | _printf("\n[DEBUG FAT] _fat_write(): P[%d,%d,%d] enters for file <%s> " | 
|---|
| [587] | 3380 | " / bytes = %x / seek = %x\n", | 
|---|
|  | 3381 | x , y , p , inode->name , count , seek ); | 
|---|
| [258] | 3382 | #endif | 
|---|
|  | 3383 |  | 
|---|
| [587] | 3384 | // chek if file size must be incremented | 
|---|
|  | 3385 | // and allocate new clusters from FAT if required | 
|---|
|  | 3386 | unsigned int old_size = inode->size; | 
|---|
|  | 3387 | unsigned int new_size = seek + count; | 
|---|
|  | 3388 | if ( new_size > old_size ) | 
|---|
|  | 3389 | { | 
|---|
|  | 3390 | // update size in inode | 
|---|
|  | 3391 | inode->size = new_size; | 
|---|
|  | 3392 |  | 
|---|
|  | 3393 | // compute current and required numbers of clusters | 
|---|
|  | 3394 | unsigned old_clusters = old_size >> 12; | 
|---|
|  | 3395 | if ( old_size & 0xFFF ) old_clusters++; | 
|---|
|  | 3396 |  | 
|---|
|  | 3397 | unsigned new_clusters = new_size >> 12; | 
|---|
|  | 3398 | if ( new_size & 0xFFF ) new_clusters++; | 
|---|
|  | 3399 |  | 
|---|
|  | 3400 | // allocate new clusters from FAT if required | 
|---|
|  | 3401 | if ( new_clusters > old_clusters ) | 
|---|
| [258] | 3402 | { | 
|---|
| [587] | 3403 |  | 
|---|
|  | 3404 | #if GIET_DEBUG_FAT | 
|---|
|  | 3405 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3406 | _printf("\n[DEBUG FAT] _fat_write(): P[%d,%d,%d] allocates new clusters for file <%s>" | 
|---|
| [587] | 3407 | " / current = %d / required = %d\n", | 
|---|
|  | 3408 | x , y , p , inode->name , old_clusters , new_clusters ); | 
|---|
|  | 3409 | #endif | 
|---|
|  | 3410 | // allocate missing clusters | 
|---|
|  | 3411 | if ( _clusters_allocate( inode, | 
|---|
|  | 3412 | old_clusters, | 
|---|
|  | 3413 | new_clusters - old_clusters ) ) | 
|---|
|  | 3414 | { | 
|---|
|  | 3415 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3416 | _printf("\n[FAT ERROR] _fat_write(): no free clusters" | 
|---|
| [587] | 3417 | " for file <%s>\n", _fat.fd[fd_id].inode->name ); | 
|---|
| [664] | 3418 | return GIET_FAT32_NO_FREE_SPACE; | 
|---|
| [587] | 3419 | } | 
|---|
| [258] | 3420 | } | 
|---|
|  | 3421 |  | 
|---|
| [654] | 3422 | // update parent directory entry (size and cluster index) | 
|---|
| [587] | 3423 | if ( _update_dir_entry( inode ) ) | 
|---|
|  | 3424 | { | 
|---|
|  | 3425 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3426 | _printf("\n[FAT ERROR] _fat_write(): cannot update parent directory entry" | 
|---|
| [587] | 3427 | " for file <%s>\n", _fat.fd[fd_id].inode->name ); | 
|---|
| [664] | 3428 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3429 | } | 
|---|
|  | 3430 |  | 
|---|
|  | 3431 |  | 
|---|
|  | 3432 | #if GIET_DEBUG_FAT | 
|---|
|  | 3433 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3434 | _printf("\n[DEBUG FAT] _fat_write(): P[%d,%d,%d] updates size for file <%s> / size = %x\n", | 
|---|
| [587] | 3435 | x , y , p , inode->name , (new_size - old_size) ); | 
|---|
|  | 3436 | #endif | 
|---|
|  | 3437 |  | 
|---|
| [258] | 3438 | } | 
|---|
|  | 3439 |  | 
|---|
| [587] | 3440 | // compute first_cluster_id and first_byte_to_move | 
|---|
|  | 3441 | unsigned int first_cluster_id   = seek >> 12; | 
|---|
|  | 3442 | unsigned int first_byte_to_move = seek & 0xFFF; | 
|---|
| [258] | 3443 |  | 
|---|
| [587] | 3444 | // compute last_cluster and last_byte_to_move | 
|---|
|  | 3445 | unsigned int last_cluster_id   = (seek + count - 1) >> 12; | 
|---|
|  | 3446 | unsigned int last_byte_to_move = (seek + count - 1) & 0xFFF; | 
|---|
|  | 3447 |  | 
|---|
|  | 3448 | #if GIET_DEBUG_FAT | 
|---|
|  | 3449 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3450 | _printf("\n[DEBUG FAT] _fat_write(): P[%d,%d,%d] starts loop on clusters for file <%s>\n" | 
|---|
| [587] | 3451 | "  first_cluster_id = %d / first_byte_to_move = %x" | 
|---|
|  | 3452 | " / last_cluster_id = %d / last_byte_to_move = %x\n", | 
|---|
|  | 3453 | x , y , p , inode->name , | 
|---|
|  | 3454 | first_cluster_id , first_byte_to_move , last_cluster_id , last_byte_to_move ); | 
|---|
|  | 3455 | #endif | 
|---|
|  | 3456 |  | 
|---|
|  | 3457 | // loop on all clusters covering the requested transfer | 
|---|
|  | 3458 | unsigned int cluster_id; | 
|---|
|  | 3459 | unsigned int done = 0; | 
|---|
|  | 3460 | for ( cluster_id = first_cluster_id ; cluster_id <= last_cluster_id ; cluster_id++ ) | 
|---|
|  | 3461 | { | 
|---|
|  | 3462 | // get pointer on one 4K buffer in File-Cache | 
|---|
|  | 3463 | unsigned char*     cbuf; | 
|---|
|  | 3464 | fat_cache_desc_t*  pdesc; | 
|---|
|  | 3465 | if ( _get_buffer_from_cache( inode, | 
|---|
|  | 3466 | cluster_id, | 
|---|
|  | 3467 | &pdesc ) ) | 
|---|
|  | 3468 | { | 
|---|
|  | 3469 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3470 | _printf("\n[FAT ERROR] _fat_write(): cannot load file <%s>\n", | 
|---|
| [587] | 3471 | inode->name ); | 
|---|
| [664] | 3472 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3473 | } | 
|---|
|  | 3474 |  | 
|---|
|  | 3475 | cbuf         = pdesc->buffer; | 
|---|
|  | 3476 | pdesc->dirty = 1; | 
|---|
|  | 3477 |  | 
|---|
|  | 3478 | #if GIET_DEBUG_FAT | 
|---|
|  | 3479 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3480 | _printf("\n[DEBUG FAT] _fat_write(): P[%d,%d,%d] move cluster_id %d to Cache-file <%s>\n", | 
|---|
| [587] | 3481 | x , y , p , cluster_id, inode->name ); | 
|---|
|  | 3482 | #endif | 
|---|
|  | 3483 |  | 
|---|
|  | 3484 | // compute memcpy arguments | 
|---|
|  | 3485 | unsigned char* source = (unsigned char*)buffer + done; | 
|---|
|  | 3486 | unsigned char* dest; | 
|---|
|  | 3487 | unsigned int   nbytes; | 
|---|
|  | 3488 | if ( (cluster_id == first_cluster_id) && (cluster_id == last_cluster_id) ) | 
|---|
|  | 3489 | { | 
|---|
|  | 3490 | dest   = cbuf + first_byte_to_move; | 
|---|
|  | 3491 | nbytes = last_byte_to_move - first_byte_to_move + 1; | 
|---|
|  | 3492 | } | 
|---|
|  | 3493 | else if ( cluster_id == first_cluster_id ) | 
|---|
|  | 3494 | { | 
|---|
|  | 3495 | dest   = cbuf + first_byte_to_move; | 
|---|
|  | 3496 | nbytes = 4096 - first_byte_to_move; | 
|---|
|  | 3497 | } | 
|---|
|  | 3498 | else if ( cluster_id == last_cluster_id ) | 
|---|
|  | 3499 | { | 
|---|
|  | 3500 | dest   = cbuf; | 
|---|
|  | 3501 | nbytes = last_byte_to_move + 1; | 
|---|
|  | 3502 | } | 
|---|
|  | 3503 | else | 
|---|
|  | 3504 | { | 
|---|
|  | 3505 | dest   = cbuf; | 
|---|
|  | 3506 | nbytes = 4096; | 
|---|
|  | 3507 | } | 
|---|
|  | 3508 |  | 
|---|
|  | 3509 | //move date | 
|---|
|  | 3510 | memcpy( dest , source , nbytes ); | 
|---|
|  | 3511 | done = done + nbytes; | 
|---|
|  | 3512 |  | 
|---|
|  | 3513 | } // end for clusters | 
|---|
|  | 3514 |  | 
|---|
|  | 3515 | // update seek | 
|---|
|  | 3516 | _fat.fd[fd_id].seek += done; | 
|---|
|  | 3517 |  | 
|---|
|  | 3518 | #if GIET_DEBUG_FAT | 
|---|
|  | 3519 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3520 | _printf("\n[DEBUG FAT] _fat_write(): P[%d,%d,%d] store file <%s> into Cache-File\n", | 
|---|
| [587] | 3521 | x , y , p , inode->name ); | 
|---|
|  | 3522 | #endif | 
|---|
|  | 3523 |  | 
|---|
|  | 3524 | // release lock | 
|---|
|  | 3525 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 3526 |  | 
|---|
|  | 3527 | return done; | 
|---|
|  | 3528 | } // end _fat_write() | 
|---|
|  | 3529 |  | 
|---|
|  | 3530 |  | 
|---|
|  | 3531 |  | 
|---|
|  | 3532 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3533 | // The following function implements the "giet_fat_lseek()" system call. | 
|---|
|  | 3534 | // It repositions the seek in the file descriptor "fd_id", according to | 
|---|
|  | 3535 | // the "seek" and "whence" arguments. | 
|---|
|  | 3536 | // It has the same semantic as the UNIX lseek() function. | 
|---|
|  | 3537 | // Accepted values for whence are SEEK_SET and SEEK_CUR. | 
|---|
|  | 3538 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 3539 | // Returns new seek value (in bytes) on success. | 
|---|
|  | 3540 | // Returns a negative value on error: | 
|---|
|  | 3541 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 3542 | //   GIET_FAT32_INVALID_FD, | 
|---|
|  | 3543 | //   GIET_FAT32_NOT_OPEN, | 
|---|
|  | 3544 | //   GIET_FAT32_INVALID_ARG | 
|---|
| [587] | 3545 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3546 | int _fat_lseek( unsigned int fd_id, | 
|---|
|  | 3547 | unsigned int seek, | 
|---|
|  | 3548 | unsigned int whence ) | 
|---|
| [258] | 3549 | { | 
|---|
| [674] | 3550 | // checking FAT initialized | 
|---|
|  | 3551 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [587] | 3552 | { | 
|---|
| [664] | 3553 | _printf("\n[FAT ERROR] _fat_lseek(): FAT not initialized\n"); | 
|---|
|  | 3554 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [587] | 3555 | } | 
|---|
| [259] | 3556 |  | 
|---|
| [587] | 3557 | // check fd_id overflow | 
|---|
|  | 3558 | if ( fd_id >= GIET_OPEN_FILES_MAX ) | 
|---|
|  | 3559 | { | 
|---|
| [664] | 3560 | _printf("\n[FAT ERROR] _fat_lseek(): illegal file descriptor\n"); | 
|---|
|  | 3561 | return GIET_FAT32_INVALID_FD; | 
|---|
| [587] | 3562 | } | 
|---|
| [259] | 3563 |  | 
|---|
| [587] | 3564 | // takes lock | 
|---|
|  | 3565 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
| [259] | 3566 |  | 
|---|
| [587] | 3567 | // check file open | 
|---|
|  | 3568 | if ( _fat.fd[fd_id].allocated == 0 ) | 
|---|
|  | 3569 | { | 
|---|
|  | 3570 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3571 | _printf("\n[FAT ERROR] _fat_lseek(): file not open\n"); | 
|---|
|  | 3572 | return GIET_FAT32_NOT_OPEN; | 
|---|
| [587] | 3573 | } | 
|---|
| [259] | 3574 |  | 
|---|
| [587] | 3575 | unsigned int  new_seek; | 
|---|
| [291] | 3576 |  | 
|---|
| [587] | 3577 | // compute new seek | 
|---|
|  | 3578 | if      ( whence == SEEK_CUR ) new_seek = _fat.fd[fd_id].seek + seek; | 
|---|
|  | 3579 | else if ( whence == SEEK_SET ) new_seek = seek; | 
|---|
|  | 3580 | else | 
|---|
|  | 3581 | { | 
|---|
|  | 3582 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3583 | _printf("\n[FAT ERROR] _fat_lseek(): illegal whence value\n"); | 
|---|
|  | 3584 | return GIET_FAT32_INVALID_ARG; | 
|---|
| [587] | 3585 | } | 
|---|
| [291] | 3586 |  | 
|---|
| [587] | 3587 | // update file descriptor offset | 
|---|
|  | 3588 | _fat.fd[fd_id].seek = new_seek; | 
|---|
| [291] | 3589 |  | 
|---|
| [259] | 3590 | #if GIET_DEBUG_FAT | 
|---|
| [295] | 3591 | unsigned int procid  = _get_procid(); | 
|---|
| [429] | 3592 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 3593 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 3594 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
| [569] | 3595 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3596 | _printf("\n[DEBUG FAT] _fat_lseek(): P[%d,%d,%d] set seek = %x for file <%s>\n", | 
|---|
| [587] | 3597 | x , y , p , new_seek , _fat.fd[fd_id].inode->name ); | 
|---|
| [259] | 3598 | #endif | 
|---|
|  | 3599 |  | 
|---|
| [587] | 3600 | // release lock | 
|---|
|  | 3601 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 3602 |  | 
|---|
|  | 3603 | return new_seek; | 
|---|
|  | 3604 | }  // end _fat_lseek() | 
|---|
|  | 3605 |  | 
|---|
|  | 3606 |  | 
|---|
|  | 3607 |  | 
|---|
|  | 3608 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3609 | // The following function implements the giet_fat_remove() system call. | 
|---|
|  | 3610 | // It deletes the file/directory identified by the "pathname" argument from | 
|---|
|  | 3611 | // the file system, if the remove condition is fulfilled (directory empty, | 
|---|
|  | 3612 | // or file not referenced). | 
|---|
|  | 3613 | // All clusters allocated in the block device DATA region are released. | 
|---|
|  | 3614 | // The FAT region is updated on the block device. | 
|---|
|  | 3615 | // The Inode-Tree is updated. | 
|---|
|  | 3616 | // The associated File_Cache is released. | 
|---|
|  | 3617 | // The Fat_Cache is updated. | 
|---|
|  | 3618 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 3619 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 3620 | // Returns a negative value on error: | 
|---|
|  | 3621 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 3622 | //   GIET_FAT32_FILE_NOT_FOUND, | 
|---|
|  | 3623 | //   GIET_FAT32_NAME_TOO_LONG, | 
|---|
|  | 3624 | //   GIET_FAT32_IS_DIRECTORY, | 
|---|
|  | 3625 | //   GIET_FAT32_NOT_A_DIRECTORY, | 
|---|
|  | 3626 | //   GIET_FAT32_IS_OPEN, | 
|---|
|  | 3627 | //   GIET_FAT32_IO_ERROR, | 
|---|
|  | 3628 | //   GIET_FAT32_DIRECTORY_NOT_EMPTY | 
|---|
| [587] | 3629 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3630 | int _fat_remove( char*        pathname, | 
|---|
|  | 3631 | unsigned int should_be_dir ) | 
|---|
|  | 3632 | { | 
|---|
|  | 3633 | fat_inode_t*  inode;            // searched file inode pointer | 
|---|
|  | 3634 |  | 
|---|
|  | 3635 | #if GIET_DEBUG_FAT | 
|---|
|  | 3636 | unsigned int procid  = _get_procid(); | 
|---|
|  | 3637 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 3638 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 3639 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
|  | 3640 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3641 | _printf("\n[DEBUG FAT] _fat_remove(): P[%d,%d,%d] enters for path <%s>\n", | 
|---|
| [587] | 3642 | x, y, p, pathname ); | 
|---|
|  | 3643 | #endif | 
|---|
|  | 3644 |  | 
|---|
| [674] | 3645 | // checking FAT initialized | 
|---|
|  | 3646 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [587] | 3647 | { | 
|---|
| [664] | 3648 | _printf("\n[FAT ERROR] _fat_remove(): FAT not initialized\n"); | 
|---|
|  | 3649 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [259] | 3650 | } | 
|---|
| [587] | 3651 |  | 
|---|
|  | 3652 | // take the lock | 
|---|
|  | 3653 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
|  | 3654 |  | 
|---|
|  | 3655 | // get searched file inode | 
|---|
|  | 3656 | unsigned int code = _get_inode_from_path( pathname , &inode ); | 
|---|
|  | 3657 |  | 
|---|
|  | 3658 | #if GIET_DEBUG_FAT | 
|---|
|  | 3659 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3660 | _printf("\n[DEBUG FAT] _fat_remove(): P[%d,%d,%d] found inode %x for <%s> / code = %d\n", | 
|---|
| [587] | 3661 | x , y , p , (unsigned int)inode , pathname , code ); | 
|---|
|  | 3662 | #endif | 
|---|
|  | 3663 |  | 
|---|
|  | 3664 | if ( (code == 1) || (code == 2) ) | 
|---|
| [259] | 3665 | { | 
|---|
| [587] | 3666 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3667 | _printf("\n[FAT ERROR] _fat_remove(): file <%s> not found\n", | 
|---|
| [587] | 3668 | pathname ); | 
|---|
| [664] | 3669 | return GIET_FAT32_FILE_NOT_FOUND; | 
|---|
| [259] | 3670 | } | 
|---|
| [587] | 3671 | else if ( code == 3 ) | 
|---|
| [259] | 3672 | { | 
|---|
| [587] | 3673 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3674 | _printf("\n[FAT ERROR] _fat_remove(): name too long in <%s>\n", | 
|---|
| [587] | 3675 | pathname ); | 
|---|
| [664] | 3676 | return GIET_FAT32_NAME_TOO_LONG; | 
|---|
| [259] | 3677 | } | 
|---|
|  | 3678 |  | 
|---|
| [587] | 3679 | // check inode type | 
|---|
|  | 3680 | if ( (inode->is_dir != 0) && (should_be_dir == 0) ) | 
|---|
| [291] | 3681 | { | 
|---|
| [587] | 3682 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3683 | _printf("\n[FAT ERROR] _fat_remove(): <%s> is a directory\n", | 
|---|
| [587] | 3684 | pathname ); | 
|---|
| [664] | 3685 | return GIET_FAT32_IS_DIRECTORY; | 
|---|
| [587] | 3686 | } | 
|---|
|  | 3687 | if ( (inode->is_dir == 0) && (should_be_dir != 0) ) | 
|---|
|  | 3688 | { | 
|---|
|  | 3689 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3690 | _printf("\n[FAT ERROR] _fat_remove(): <%s> is not a directory\n", | 
|---|
| [587] | 3691 | pathname ); | 
|---|
| [664] | 3692 | return GIET_FAT32_NOT_A_DIRECTORY; | 
|---|
| [587] | 3693 | } | 
|---|
|  | 3694 |  | 
|---|
|  | 3695 | #if GIET_DEBUG_FAT | 
|---|
|  | 3696 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3697 | _printf("\n[DEBUG FAT] _fat_remove(): P[%d,%d,%d] checked inode type for <%s>\n", | 
|---|
| [587] | 3698 | x , y , p , pathname ); | 
|---|
|  | 3699 | #endif | 
|---|
|  | 3700 |  | 
|---|
|  | 3701 | // check references count for a file | 
|---|
|  | 3702 | if ( (inode->is_dir == 0) && (inode->count != 0) ) | 
|---|
|  | 3703 | { | 
|---|
|  | 3704 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3705 | _printf("\n[FAT ERROR] _fat_remove(): file <%s> still referenced\n", | 
|---|
| [587] | 3706 | pathname ); | 
|---|
| [664] | 3707 | return GIET_FAT32_IS_OPEN; | 
|---|
| [587] | 3708 | } | 
|---|
|  | 3709 |  | 
|---|
|  | 3710 | //  check empty for a directory | 
|---|
|  | 3711 | if ( inode->is_dir ) | 
|---|
|  | 3712 | { | 
|---|
|  | 3713 | unsigned int entries; | 
|---|
|  | 3714 | if ( _get_nb_entries( inode , &entries ) ) | 
|---|
| [291] | 3715 | { | 
|---|
| [587] | 3716 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3717 | _printf("\n[FAT ERROR] _fat_remove(): cannot scan directory <%s>\n", | 
|---|
| [587] | 3718 | pathname ); | 
|---|
| [664] | 3719 | return GIET_FAT32_IO_ERROR; | 
|---|
| [291] | 3720 | } | 
|---|
| [587] | 3721 | else if ( entries > 2 ) | 
|---|
|  | 3722 | { | 
|---|
|  | 3723 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3724 | _printf("\n[FAT ERROR] _fat_remove(): directory <%s> not empty\n", | 
|---|
| [587] | 3725 | pathname ); | 
|---|
| [664] | 3726 | return GIET_FAT32_DIRECTORY_NOT_EMPTY; | 
|---|
| [587] | 3727 | } | 
|---|
| [291] | 3728 | } | 
|---|
|  | 3729 |  | 
|---|
| [587] | 3730 | #if GIET_DEBUG_FAT | 
|---|
|  | 3731 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3732 | _printf("\n[DEBUG FAT] _fat_remove(): P[%d,%d,%d] checked remove condition OK for <%s>\n", | 
|---|
| [587] | 3733 | x , y , p , pathname ); | 
|---|
|  | 3734 | #endif | 
|---|
| [259] | 3735 |  | 
|---|
| [587] | 3736 | // remove the file or directory from the file system | 
|---|
|  | 3737 | if ( _remove_node_from_fs( inode ) ) | 
|---|
|  | 3738 | { | 
|---|
|  | 3739 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3740 | _printf("\n[FAT ERROR] _fat_remove(): cannot remove <%s> from FS\n", | 
|---|
| [587] | 3741 | pathname ); | 
|---|
| [664] | 3742 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3743 | } | 
|---|
| [259] | 3744 |  | 
|---|
| [587] | 3745 | // release lock and return success | 
|---|
|  | 3746 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 3747 |  | 
|---|
| [259] | 3748 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 3749 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3750 | _printf("\n[DEBUG FAT] _fat_remove(): P[%d,%d,%d] removed  <%s> from FS\n", | 
|---|
| [587] | 3751 | x, y, p, pathname ); | 
|---|
| [259] | 3752 | #endif | 
|---|
| [587] | 3753 |  | 
|---|
| [664] | 3754 | return GIET_FAT32_OK; | 
|---|
| [587] | 3755 |  | 
|---|
|  | 3756 | }  // end _fat_remove() | 
|---|
| [259] | 3757 |  | 
|---|
| [587] | 3758 |  | 
|---|
|  | 3759 |  | 
|---|
|  | 3760 |  | 
|---|
|  | 3761 |  | 
|---|
|  | 3762 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3763 | // This function implements the giet_fat_rename() system call. | 
|---|
|  | 3764 | // It moves an existing file or directory from one node (defined by "old_path" | 
|---|
|  | 3765 | // argument) to another node (defined by "new_path" argument) in the FS tree. | 
|---|
|  | 3766 | // The type (file/directory) and content are not modified. | 
|---|
|  | 3767 | // If the new_path file/dir exist, it is removed from the file system, but only | 
|---|
|  | 3768 | // if the remove condition is respected (directory empty / file not referenced). | 
|---|
|  | 3769 | // The removed entry is only removed after the new entry is actually created. | 
|---|
|  | 3770 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 3771 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 3772 | // Returns a negative value on error: | 
|---|
|  | 3773 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 3774 | //   GIET_FAT32_FILE_NOT_FOUND, | 
|---|
|  | 3775 | //   GIET_FAT32_MOVE_INTO_SUBDIR, | 
|---|
|  | 3776 | //   GIET_FAT32_IO_ERROR, | 
|---|
|  | 3777 | //   GIET_FAT32_DIRECTORY_NOT_EMPTY, | 
|---|
|  | 3778 | //   GIET_FAT32_IS_OPEN | 
|---|
| [587] | 3779 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 3780 | int _fat_rename( char*  old_path, | 
|---|
|  | 3781 | char*  new_path ) | 
|---|
|  | 3782 | { | 
|---|
|  | 3783 | fat_inode_t*  inode;        // anonymous inode pointer | 
|---|
|  | 3784 | fat_inode_t*  old;          // inode identified by old_path      => to be deleted | 
|---|
|  | 3785 | fat_inode_t*  new;          // inode identified by new_path      => to be created | 
|---|
|  | 3786 | fat_inode_t*  old_parent;   // parent inode  in old_path         => to be modified | 
|---|
|  | 3787 | fat_inode_t*  new_parent;   // parent inode  in new_path         => to be modified | 
|---|
|  | 3788 | fat_inode_t*  to_remove;    // previouly identified by new_path  => to be removed | 
|---|
|  | 3789 | unsigned int  code; | 
|---|
|  | 3790 |  | 
|---|
|  | 3791 | #if GIET_DEBUG_FAT | 
|---|
|  | 3792 | unsigned int procid  = _get_procid(); | 
|---|
|  | 3793 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 3794 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 3795 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
|  | 3796 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3797 | _printf("\n[DEBUG FAT] _fat_rename(): P[%d,%d,%d] enters to move <%s> to <%s>\n", | 
|---|
| [587] | 3798 | x , y , p , old_path , new_path ); | 
|---|
|  | 3799 | #endif | 
|---|
|  | 3800 |  | 
|---|
| [674] | 3801 | // checking FAT initialized | 
|---|
|  | 3802 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [259] | 3803 | { | 
|---|
| [664] | 3804 | _printf("\n[FAT ERROR] _fat_rename(): FAT not initialized\n"); | 
|---|
|  | 3805 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [259] | 3806 | } | 
|---|
|  | 3807 |  | 
|---|
| [587] | 3808 | // take the lock | 
|---|
|  | 3809 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
| [259] | 3810 |  | 
|---|
| [587] | 3811 | // get "old" and "old_parent" inode pointers | 
|---|
|  | 3812 | if ( _get_inode_from_path( old_path , &inode ) ) | 
|---|
|  | 3813 | { | 
|---|
|  | 3814 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3815 | _printf("\n[FAT ERROR] _fat_rename(): <%s> not found\n", old_path ); | 
|---|
|  | 3816 | return GIET_FAT32_FILE_NOT_FOUND; | 
|---|
| [587] | 3817 | } | 
|---|
|  | 3818 | else | 
|---|
|  | 3819 | { | 
|---|
|  | 3820 | old        = inode; | 
|---|
|  | 3821 | old_parent = inode->parent; | 
|---|
|  | 3822 | } | 
|---|
| [259] | 3823 |  | 
|---|
| [587] | 3824 | // get "to_removed" and "new_parent" inode pointers | 
|---|
|  | 3825 | code = _get_inode_from_path( new_path , &inode ); | 
|---|
|  | 3826 |  | 
|---|
|  | 3827 | if ( code == 0 )       // new_path inode already exist | 
|---|
| [259] | 3828 | { | 
|---|
| [621] | 3829 | if ( inode == old )  // the file will replace itself, do nothing | 
|---|
|  | 3830 | { | 
|---|
|  | 3831 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3832 | return GIET_FAT32_OK; | 
|---|
| [621] | 3833 | } | 
|---|
|  | 3834 |  | 
|---|
| [587] | 3835 | to_remove        = inode; | 
|---|
| [621] | 3836 | new_parent       = inode->parent; | 
|---|
| [587] | 3837 | } | 
|---|
|  | 3838 | else if ( code == 1 )  // to_remove does not exist but parent exist | 
|---|
|  | 3839 | { | 
|---|
|  | 3840 | to_remove        = NULL; | 
|---|
|  | 3841 | new_parent       = inode; | 
|---|
|  | 3842 | } | 
|---|
|  | 3843 | else                   // parent directory in new_path not found | 
|---|
|  | 3844 | { | 
|---|
|  | 3845 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3846 | _printf("\n[FAT ERROR] _fat_rename(): <%s> not found\n", new_path ); | 
|---|
|  | 3847 | return GIET_FAT32_FILE_NOT_FOUND; | 
|---|
| [587] | 3848 | } | 
|---|
| [259] | 3849 |  | 
|---|
| [638] | 3850 | // check for move into own subdirectory | 
|---|
|  | 3851 | if ( _is_ancestor( old, new_parent ) ) | 
|---|
|  | 3852 | { | 
|---|
|  | 3853 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3854 | _printf("\n[FAT ERROR] _fat_rename(): can't move %s into its own subdirectory\n", old_path ); | 
|---|
|  | 3855 | return GIET_FAT32_MOVE_INTO_SUBDIR; | 
|---|
| [638] | 3856 | } | 
|---|
|  | 3857 |  | 
|---|
| [259] | 3858 | #if GIET_DEBUG_FAT | 
|---|
| [569] | 3859 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [587] | 3860 | { | 
|---|
|  | 3861 | if ( to_remove ) | 
|---|
| [664] | 3862 | _printf("\n[DEBUG FAT] _fat_rename(): old_parent = %s / old = %s / new_parent = %s " | 
|---|
| [587] | 3863 | "/ to_remove = %s\n", | 
|---|
|  | 3864 | old_parent->name , old->name , new_parent->name , to_remove->name ); | 
|---|
|  | 3865 | else | 
|---|
| [664] | 3866 | _printf("\n[DEBUG FAT] _fat_rename(): old_parent = %s / old = %s / new_parent = %s " | 
|---|
| [587] | 3867 | "/ no remove\n", | 
|---|
|  | 3868 | old_parent->name , old->name , new_parent->name ); | 
|---|
|  | 3869 | } | 
|---|
| [259] | 3870 | #endif | 
|---|
|  | 3871 |  | 
|---|
| [587] | 3872 | // check remove condition for "to_remove" inode | 
|---|
|  | 3873 | if ( to_remove ) | 
|---|
|  | 3874 | { | 
|---|
|  | 3875 | if ( to_remove->is_dir )   // it's a directory | 
|---|
| [259] | 3876 | { | 
|---|
| [587] | 3877 | unsigned int entries; | 
|---|
|  | 3878 | if ( _get_nb_entries( to_remove , &entries ) ) | 
|---|
|  | 3879 | { | 
|---|
|  | 3880 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3881 | _printf("\n[FAT ERROR] _fat_rename(): cannot scan directory <%s>\n", | 
|---|
| [587] | 3882 | to_remove->name ); | 
|---|
| [664] | 3883 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3884 | } | 
|---|
|  | 3885 | else if ( entries > 2 ) | 
|---|
|  | 3886 | { | 
|---|
|  | 3887 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3888 | _printf("\n[FAT ERROR] _fat_rename(): directory <%s> not empty\n", | 
|---|
| [587] | 3889 | to_remove->name ); | 
|---|
| [664] | 3890 | return GIET_FAT32_DIRECTORY_NOT_EMPTY; | 
|---|
| [587] | 3891 | } | 
|---|
| [259] | 3892 | } | 
|---|
| [587] | 3893 | else                       // it's a file | 
|---|
|  | 3894 | { | 
|---|
|  | 3895 | if ( to_remove->count ) | 
|---|
|  | 3896 | { | 
|---|
|  | 3897 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3898 | _printf("\n[FAT ERROR] _fat_rename(): file <%s> still referenced\n", | 
|---|
| [587] | 3899 | to_remove->name ); | 
|---|
| [664] | 3900 | return GIET_FAT32_IS_OPEN; | 
|---|
| [587] | 3901 | } | 
|---|
|  | 3902 | } | 
|---|
| [259] | 3903 | } | 
|---|
| [291] | 3904 |  | 
|---|
| [587] | 3905 | #if GIET_DEBUG_FAT | 
|---|
|  | 3906 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 3907 | _printf("\n[FAT DEBUG] _fat_rename(): P[%d,%d,%d] checked remove condition OK\n", | 
|---|
| [587] | 3908 | x , y , p ); | 
|---|
|  | 3909 | #endif | 
|---|
|  | 3910 |  | 
|---|
|  | 3911 | // get new last name / error checking already done by _get_inode_from_path() | 
|---|
|  | 3912 | char  new_name[32]; | 
|---|
|  | 3913 | _get_last_name( new_path , new_name ); | 
|---|
|  | 3914 |  | 
|---|
|  | 3915 | // allocate "new" inode | 
|---|
|  | 3916 | new = _allocate_one_inode( new_name, | 
|---|
|  | 3917 | old->is_dir, | 
|---|
|  | 3918 | old->cluster, | 
|---|
|  | 3919 | old->size, | 
|---|
|  | 3920 | 0,              // count | 
|---|
|  | 3921 | 0,              // dentry | 
|---|
|  | 3922 | 0 );            // no cache_allocate | 
|---|
|  | 3923 |  | 
|---|
|  | 3924 | // give the "old" File-Cache to the "new inode | 
|---|
|  | 3925 | new->levels = old->levels; | 
|---|
|  | 3926 | new->cache  = old->cache; | 
|---|
|  | 3927 |  | 
|---|
|  | 3928 | // add "new" to "new_parent" directory File-Cache | 
|---|
|  | 3929 | if ( _add_dir_entry( new , new_parent ) ) | 
|---|
| [291] | 3930 | { | 
|---|
| [587] | 3931 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3932 | _printf("\n[FAT ERROR] _fat_rename(): cannot add <%s> into <%s>\n", | 
|---|
| [587] | 3933 | new->name , new_parent->name ); | 
|---|
| [664] | 3934 | return GIET_FAT32_IO_ERROR; | 
|---|
| [291] | 3935 | } | 
|---|
|  | 3936 |  | 
|---|
| [587] | 3937 | // add "new" to "new_parent" directory in Inode-Tree | 
|---|
|  | 3938 | _add_inode_in_tree( new , new_parent ); | 
|---|
|  | 3939 |  | 
|---|
|  | 3940 | // updates "new_parent" directory on device | 
|---|
|  | 3941 | if ( _update_device_from_cache( new_parent->levels, | 
|---|
|  | 3942 | new_parent->cache, | 
|---|
|  | 3943 | new_parent->name ) ) | 
|---|
| [291] | 3944 | { | 
|---|
| [587] | 3945 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3946 | _printf("\n[FAT ERROR] _fat_rename(): cannot update <%s> on device\n", | 
|---|
| [587] | 3947 | new_parent->name ); | 
|---|
| [664] | 3948 | return GIET_FAT32_IO_ERROR; | 
|---|
| [291] | 3949 | } | 
|---|
| [258] | 3950 |  | 
|---|
| [587] | 3951 | // remove "old" from "old_parent" File-Cache | 
|---|
|  | 3952 | if ( _remove_dir_entry( old ) ) | 
|---|
|  | 3953 | { | 
|---|
|  | 3954 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3955 | _printf("\n[FAT ERROR] _fat_rename(): cannot remove <%s> from <%s>\n", | 
|---|
| [587] | 3956 | old->name , old_parent->name ); | 
|---|
| [664] | 3957 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3958 | } | 
|---|
|  | 3959 |  | 
|---|
|  | 3960 | // remove "old" inode from Inode-Tree | 
|---|
|  | 3961 | _remove_inode_from_tree( old ); | 
|---|
|  | 3962 |  | 
|---|
| [622] | 3963 | // release "old" inode | 
|---|
|  | 3964 | _free( old ); | 
|---|
|  | 3965 |  | 
|---|
| [587] | 3966 | // updates "old_parent" directory on device | 
|---|
|  | 3967 | if ( _update_device_from_cache( old_parent->levels, | 
|---|
|  | 3968 | old_parent->cache, | 
|---|
|  | 3969 | old_parent->name ) ) | 
|---|
|  | 3970 | { | 
|---|
|  | 3971 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3972 | _printf("\n[FAT ERROR] _fat_rename(): cannot update <%s> on device\n", | 
|---|
| [587] | 3973 | old_parent->name ); | 
|---|
| [664] | 3974 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3975 | } | 
|---|
|  | 3976 |  | 
|---|
|  | 3977 | // remove "to_remove" from File System (if required) | 
|---|
|  | 3978 | if ( to_remove ) | 
|---|
|  | 3979 | { | 
|---|
|  | 3980 | if ( _remove_node_from_fs( to_remove ) ) | 
|---|
|  | 3981 | { | 
|---|
|  | 3982 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 3983 | _printf("\n[FAT ERROR] _fat_rename(): cannot remove <%s> from FS\n", | 
|---|
| [587] | 3984 | to_remove->name ); | 
|---|
| [664] | 3985 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 3986 | } | 
|---|
|  | 3987 | } | 
|---|
|  | 3988 |  | 
|---|
|  | 3989 | // release lock | 
|---|
|  | 3990 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 3991 |  | 
|---|
| [664] | 3992 | return GIET_FAT32_OK; | 
|---|
| [587] | 3993 | }  // end _fat_rename() | 
|---|
|  | 3994 |  | 
|---|
|  | 3995 |  | 
|---|
|  | 3996 |  | 
|---|
|  | 3997 |  | 
|---|
| [258] | 3998 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 3999 | // The following function implements the giet_fat_mkdir() system call. | 
|---|
|  | 4000 | // It creates in file system the directory specified by the "pathname" argument. | 
|---|
|  | 4001 | // The Inode-Tree is updated. | 
|---|
|  | 4002 | // One cluster is allocated to the new directory. | 
|---|
|  | 4003 | // The associated File-Cache is created. | 
|---|
|  | 4004 | // The FAT region on block device is updated. | 
|---|
|  | 4005 | // The DATA region on block device is updated. | 
|---|
| [260] | 4006 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 4007 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 4008 | // Returns a negative value on error: | 
|---|
|  | 4009 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 4010 | //   GIET_FAT32_FILE_NOT_FOUND, | 
|---|
|  | 4011 | //   GIET_FAT32_NAME_TOO_LONG, | 
|---|
|  | 4012 | //   GIET_FAT32_FILE_EXISTS, | 
|---|
|  | 4013 | //   GIET_FAT32_NO_FREE_SPACE, | 
|---|
|  | 4014 | //   GIET_FAT32_IO_ERROR | 
|---|
| [260] | 4015 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [587] | 4016 | int _fat_mkdir( char* pathname ) | 
|---|
| [260] | 4017 | { | 
|---|
| [587] | 4018 | fat_inode_t*         inode;            // anonymous inode pointer | 
|---|
|  | 4019 | fat_inode_t*         child;            // searched directory inode pointer | 
|---|
|  | 4020 | fat_inode_t*         parent;           // parent directory inode pointer | 
|---|
| [260] | 4021 |  | 
|---|
| [587] | 4022 | #if GIET_DEBUG_FAT | 
|---|
|  | 4023 | unsigned int procid  = _get_procid(); | 
|---|
|  | 4024 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 4025 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 4026 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
|  | 4027 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 4028 | _printf("\n[DEBUG FAT] _fat_mkdir(): P[%d,%d,%d] enters for path <%s>\n", | 
|---|
| [587] | 4029 | x, y, p, pathname ); | 
|---|
|  | 4030 | #endif | 
|---|
|  | 4031 |  | 
|---|
| [674] | 4032 | // checking FAT initialized | 
|---|
|  | 4033 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [260] | 4034 | { | 
|---|
| [664] | 4035 | _printf("\n[FAT ERROR] _fat_mkdir(): FAT not initialized\n"); | 
|---|
|  | 4036 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [587] | 4037 | } | 
|---|
| [260] | 4038 |  | 
|---|
| [587] | 4039 | // takes the lock | 
|---|
|  | 4040 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
|  | 4041 |  | 
|---|
|  | 4042 | // get inode | 
|---|
|  | 4043 | unsigned int code = _get_inode_from_path( pathname , &inode ); | 
|---|
| [260] | 4044 |  | 
|---|
| [587] | 4045 | if ( code == 2 ) | 
|---|
|  | 4046 | { | 
|---|
|  | 4047 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4048 | _printf("\n[FAT ERROR] _fat_mkdir(): path to parent not found" | 
|---|
| [587] | 4049 | " for directory <%s>\n", pathname ); | 
|---|
| [664] | 4050 | return GIET_FAT32_FILE_NOT_FOUND; | 
|---|
| [260] | 4051 | } | 
|---|
| [587] | 4052 | else if ( code == 3 ) | 
|---|
| [260] | 4053 | { | 
|---|
| [587] | 4054 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4055 | _printf("\n[FAT ERROR] _fat_mkdir(): one name in path too long" | 
|---|
| [587] | 4056 | " for directory  <%s>\n", pathname ); | 
|---|
| [664] | 4057 | return GIET_FAT32_NAME_TOO_LONG; | 
|---|
| [587] | 4058 | } | 
|---|
|  | 4059 | else if ( code == 0 ) | 
|---|
|  | 4060 | { | 
|---|
|  | 4061 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4062 | _printf("\n[FAT ERROR] _fat_mkdir(): directory <%s> already exist\n", | 
|---|
| [587] | 4063 | pathname ); | 
|---|
| [664] | 4064 | return GIET_FAT32_FILE_EXISTS; | 
|---|
| [587] | 4065 | } | 
|---|
|  | 4066 | else if ( code == 1 )   // directory not found => create | 
|---|
|  | 4067 | { | 
|---|
|  | 4068 | parent = inode; | 
|---|
| [260] | 4069 |  | 
|---|
| [587] | 4070 | #if GIET_DEBUG_FAT | 
|---|
|  | 4071 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 4072 | _printf("\n[DEBUG FAT] _fat_mkdir(): P[%d,%d,%d] create new directory <%s>\n", | 
|---|
| [587] | 4073 | x , y , p , pathname ); | 
|---|
|  | 4074 | #endif | 
|---|
|  | 4075 |  | 
|---|
|  | 4076 | // get directory name / error check already done by _get_inode_from_path() | 
|---|
|  | 4077 | char name[32]; | 
|---|
|  | 4078 | _get_last_name( pathname , name ); | 
|---|
|  | 4079 |  | 
|---|
|  | 4080 | // allocate one cluster from FAT for the new directory | 
|---|
|  | 4081 | unsigned int cluster; | 
|---|
|  | 4082 | if ( _allocate_one_cluster( &cluster ) ) | 
|---|
|  | 4083 | { | 
|---|
|  | 4084 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4085 | _printf("\n[FAT ERROR] _fat_mkdir(): no free cluster" | 
|---|
| [587] | 4086 | " for directory <%s>\n" , pathname ); | 
|---|
| [664] | 4087 | return GIET_FAT32_NO_FREE_SPACE; | 
|---|
| [587] | 4088 | } | 
|---|
|  | 4089 |  | 
|---|
|  | 4090 | // allocate a new inode and an empty Cache-File | 
|---|
|  | 4091 | child = _allocate_one_inode( name, | 
|---|
|  | 4092 | 1,           // it's a directory | 
|---|
|  | 4093 | cluster, | 
|---|
|  | 4094 | 0,           // size not defined | 
|---|
|  | 4095 | 0,           // count | 
|---|
|  | 4096 | 0,           // dentry set by _add_dir_entry() | 
|---|
|  | 4097 | 1 );         // cache_allocate | 
|---|
|  | 4098 |  | 
|---|
|  | 4099 | // introduce inode in Inode-Tree | 
|---|
|  | 4100 | _add_inode_in_tree( child , parent ); | 
|---|
|  | 4101 |  | 
|---|
|  | 4102 | // allocate and initialise one 4 Kbytes buffer and associated descriptor | 
|---|
|  | 4103 | _allocate_one_buffer( child, | 
|---|
|  | 4104 | 0,            // cluster_id, | 
|---|
|  | 4105 | cluster ); | 
|---|
|  | 4106 |  | 
|---|
|  | 4107 | _add_special_directories( child, | 
|---|
|  | 4108 | parent ); | 
|---|
|  | 4109 |  | 
|---|
|  | 4110 | // add an entry in the parent directory Cache_file | 
|---|
|  | 4111 | if ( _add_dir_entry( child , parent ) ) | 
|---|
|  | 4112 | { | 
|---|
|  | 4113 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4114 | _printf("\n[FAT ERROR] _fat_mkdir(): cannot update parent directory" | 
|---|
| [587] | 4115 | " for directory <%s>\n" , pathname ); | 
|---|
| [664] | 4116 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 4117 | } | 
|---|
|  | 4118 |  | 
|---|
|  | 4119 | // update DATA region on block device for parent directory | 
|---|
|  | 4120 | if ( _update_device_from_cache( parent->levels, | 
|---|
|  | 4121 | parent->cache, | 
|---|
|  | 4122 | parent->name ) ) | 
|---|
|  | 4123 | { | 
|---|
|  | 4124 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4125 | _printf("\n[FAT ERROR] _fat_mkdir(): cannot update DATA region " | 
|---|
| [587] | 4126 | " for parent of directory <%s>\n", pathname ); | 
|---|
| [664] | 4127 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 4128 | } | 
|---|
|  | 4129 |  | 
|---|
|  | 4130 | // update FAT region on block device | 
|---|
|  | 4131 | if ( _update_device_from_cache( _fat.fat_cache_levels, | 
|---|
|  | 4132 | _fat.fat_cache_root, | 
|---|
|  | 4133 | "FAT" ) ) | 
|---|
|  | 4134 | { | 
|---|
|  | 4135 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4136 | _printf("\n[FAT ERROR] _fat_mkdir(): cannot update FAT region" | 
|---|
| [587] | 4137 | " for directory <%s>\n", pathname ); | 
|---|
| [664] | 4138 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 4139 | } | 
|---|
|  | 4140 |  | 
|---|
|  | 4141 | // update FS_INFO sector | 
|---|
|  | 4142 | if ( _update_fs_info() ) | 
|---|
|  | 4143 | { | 
|---|
|  | 4144 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4145 | _printf("\n[FAT ERROR] _fat_mkdir(): cannot update FS-INFO" | 
|---|
| [587] | 4146 | " for directory <%s>\n", pathname ); | 
|---|
| [664] | 4147 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 4148 | } | 
|---|
|  | 4149 |  | 
|---|
|  | 4150 | // update DATA region on block device for the new directory | 
|---|
|  | 4151 | if ( _update_device_from_cache( child->levels, | 
|---|
|  | 4152 | child->cache, | 
|---|
|  | 4153 | child->name ) ) | 
|---|
|  | 4154 | { | 
|---|
|  | 4155 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
| [664] | 4156 | _printf("\n[FAT ERROR] _fat_mkdir(): cannot update DATA region" | 
|---|
| [587] | 4157 | " for directory <%s>\n", pathname ); | 
|---|
| [664] | 4158 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 4159 | } | 
|---|
|  | 4160 | }  // end create directory | 
|---|
|  | 4161 |  | 
|---|
| [606] | 4162 | // release lock | 
|---|
|  | 4163 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 4164 |  | 
|---|
| [664] | 4165 | return GIET_FAT32_OK; | 
|---|
| [587] | 4166 | }  // end _fat_mkdir() | 
|---|
|  | 4167 |  | 
|---|
|  | 4168 |  | 
|---|
|  | 4169 |  | 
|---|
|  | 4170 |  | 
|---|
| [658] | 4171 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 4172 | // This function implements the giet_fat_opendir() system call. | 
|---|
|  | 4173 | // The semantic is similar to the UNIX opendir() function. | 
|---|
|  | 4174 | // If the specified directory does not exist, an error is returned. | 
|---|
|  | 4175 | // It allocates a file descriptor to the calling task, for the directory | 
|---|
|  | 4176 | // identified by "pathname". If several tasks try to open the same directory, | 
|---|
|  | 4177 | // each task obtains a private file descriptor. | 
|---|
|  | 4178 | // A node name cannot be larger than 31 characters. | 
|---|
|  | 4179 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 4180 | // Returns a file descriptor for the directory index on success | 
|---|
| [658] | 4181 | // Returns a negative value on error: | 
|---|
| [664] | 4182 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 4183 | //   GIET_FAT32_NAME_TOO_LONG, | 
|---|
|  | 4184 | //   GIET_FAT32_FILE_NOT_FOUND, | 
|---|
|  | 4185 | //   GIET_FAT32_TOO_MANY_OPEN_FILES, | 
|---|
|  | 4186 | //   GIET_FAT32_NOT_A_DIRECTORY | 
|---|
| [658] | 4187 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 4188 | extern int _fat_opendir( char* pathname ) | 
|---|
|  | 4189 | { | 
|---|
|  | 4190 | int fd_id = _fat_open( pathname, O_RDONLY ); | 
|---|
| [587] | 4191 |  | 
|---|
| [658] | 4192 | if ( fd_id < 0 ) | 
|---|
|  | 4193 | return fd_id; | 
|---|
|  | 4194 |  | 
|---|
|  | 4195 | if ( !_fat.fd[fd_id].inode->is_dir ) | 
|---|
|  | 4196 | { | 
|---|
| [664] | 4197 | _printf("\n[FAT ERROR] _fat_opendir(): <%s> is not a directory\n", | 
|---|
| [658] | 4198 | pathname ); | 
|---|
| [664] | 4199 | return GIET_FAT32_NOT_A_DIRECTORY; | 
|---|
| [658] | 4200 | } | 
|---|
|  | 4201 |  | 
|---|
|  | 4202 | return fd_id; | 
|---|
|  | 4203 | } | 
|---|
|  | 4204 |  | 
|---|
|  | 4205 |  | 
|---|
|  | 4206 |  | 
|---|
|  | 4207 |  | 
|---|
| [260] | 4208 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [658] | 4209 | // This function implements the "giet_fat_closedir()" system call. | 
|---|
|  | 4210 | // Same behavior as _fat_close(), no check for directory. | 
|---|
|  | 4211 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 4212 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 4213 | // Returns a negative value on error: | 
|---|
|  | 4214 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 4215 | //   GIET_FAT32_INVALID_FD, | 
|---|
|  | 4216 | //   GIET_FAT32_NOT_OPEN, | 
|---|
|  | 4217 | //   GIET_FAT32_IO_ERROR | 
|---|
| [658] | 4218 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 4219 | extern int _fat_closedir( unsigned int fd_id ) | 
|---|
|  | 4220 | { | 
|---|
|  | 4221 | return _fat_close( fd_id ); | 
|---|
|  | 4222 | } | 
|---|
|  | 4223 |  | 
|---|
|  | 4224 |  | 
|---|
|  | 4225 |  | 
|---|
|  | 4226 |  | 
|---|
|  | 4227 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 4228 | // This function implements the "giet_fat_readdir()" system call. | 
|---|
|  | 4229 | // It reads one directory entry from the file descriptor opened by | 
|---|
|  | 4230 | // "giet_fat_opendir()" and writes its info to the "entry" argument. | 
|---|
|  | 4231 | // This includes the cluster, size, is_dir and name info for each entry. | 
|---|
|  | 4232 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 4233 | // Returns GIET_FAT32_OK on success. | 
|---|
| [658] | 4234 | // Returns a negative value on error: | 
|---|
| [664] | 4235 | //   GIET_FAT32_NOT_INITIALIZED, | 
|---|
|  | 4236 | //   GIET_FAT32_INVALID_FD, | 
|---|
|  | 4237 | //   GIET_FAT32_NOT_OPEN, | 
|---|
|  | 4238 | //   GIET_FAT32_NOT_A_DIRECTORY, | 
|---|
|  | 4239 | //   GIET_FAT32_IO_ERROR, | 
|---|
|  | 4240 | //   GIET_FAT32_NO_MORE_ENTRIES | 
|---|
| [658] | 4241 | ///////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 4242 | extern int _fat_readdir( unsigned int  fd_id, | 
|---|
|  | 4243 | fat_dirent_t* entry ) | 
|---|
|  | 4244 | { | 
|---|
|  | 4245 | unsigned int  lfn   = 0;            // lfn entries count | 
|---|
|  | 4246 | unsigned int  attr;                 // ATTR field value | 
|---|
|  | 4247 | unsigned int  ord;                  // ORD field value | 
|---|
|  | 4248 | char          lfn1[16];             // temporary buffer for string in LFN1 | 
|---|
|  | 4249 | char          lfn2[16];             // temporary buffer for string in LFN2 | 
|---|
|  | 4250 | char          lfn3[16];             // temporary buffer for string in LFN3 | 
|---|
|  | 4251 | unsigned char buf[DIR_ENTRY_SIZE];  // raw entry buffer | 
|---|
|  | 4252 | fat_file_info_t info; | 
|---|
|  | 4253 |  | 
|---|
|  | 4254 | // check for directory | 
|---|
|  | 4255 | int ret = _fat_file_info( fd_id, &info ); | 
|---|
|  | 4256 | if (ret < 0) | 
|---|
|  | 4257 | { | 
|---|
|  | 4258 | return ret; | 
|---|
|  | 4259 | } | 
|---|
|  | 4260 | else if ( !info.is_dir ) | 
|---|
|  | 4261 | { | 
|---|
| [664] | 4262 | _printf("\n[FAT ERROR] _fat_readdir(): not a directory\n" ); | 
|---|
|  | 4263 | return GIET_FAT32_NOT_A_DIRECTORY; | 
|---|
| [658] | 4264 | } | 
|---|
|  | 4265 |  | 
|---|
|  | 4266 | while ( 1 ) | 
|---|
|  | 4267 | { | 
|---|
|  | 4268 | if ( _fat_read( fd_id, &buf, sizeof(buf) ) != sizeof(buf) ) | 
|---|
|  | 4269 | { | 
|---|
| [664] | 4270 | _printf("\n[FAT ERROR] _fat_readdir(): can't read entry\n" ); | 
|---|
|  | 4271 | return GIET_FAT32_IO_ERROR; | 
|---|
| [658] | 4272 | } | 
|---|
|  | 4273 |  | 
|---|
|  | 4274 | attr = _read_entry( DIR_ATTR, buf, 0 ); | 
|---|
|  | 4275 | ord  = _read_entry( LDIR_ORD, buf, 0 ); | 
|---|
|  | 4276 |  | 
|---|
|  | 4277 | if (ord == NO_MORE_ENTRY)               // no more entry in directory => stop | 
|---|
|  | 4278 | { | 
|---|
|  | 4279 | // seek back to this entry | 
|---|
|  | 4280 | _spin_lock_acquire( &_fat.fat_lock ); | 
|---|
|  | 4281 | _fat.fd[fd_id].seek -= DIR_ENTRY_SIZE; | 
|---|
|  | 4282 | _spin_lock_release( &_fat.fat_lock ); | 
|---|
|  | 4283 |  | 
|---|
| [664] | 4284 | return GIET_FAT32_NO_MORE_ENTRIES; | 
|---|
| [658] | 4285 | } | 
|---|
|  | 4286 | else if ( ord == FREE_ENTRY )           // free entry => skip | 
|---|
|  | 4287 | { | 
|---|
|  | 4288 | continue; | 
|---|
|  | 4289 | } | 
|---|
|  | 4290 | else if ( attr == ATTR_LONG_NAME_MASK ) // LFN entry => get partial names | 
|---|
|  | 4291 | { | 
|---|
|  | 4292 | unsigned int seq = ord & 0x3; | 
|---|
|  | 4293 | lfn = (seq > lfn) ? seq : lfn; | 
|---|
|  | 4294 | if      ( seq == 1 ) _get_name_from_long( buf, lfn1 ); | 
|---|
|  | 4295 | else if ( seq == 2 ) _get_name_from_long( buf, lfn2 ); | 
|---|
|  | 4296 | else if ( seq == 3 ) _get_name_from_long( buf, lfn3 ); | 
|---|
|  | 4297 | continue; | 
|---|
|  | 4298 | } | 
|---|
|  | 4299 | else                                    // NORMAL entry => stop | 
|---|
|  | 4300 | { | 
|---|
|  | 4301 | break; | 
|---|
|  | 4302 | } | 
|---|
|  | 4303 | } | 
|---|
|  | 4304 |  | 
|---|
|  | 4305 | // TODO handle is_vid | 
|---|
|  | 4306 | entry->cluster = (_read_entry( DIR_FST_CLUS_HI, buf, 1 ) << 16) | | 
|---|
|  | 4307 | (_read_entry( DIR_FST_CLUS_LO, buf, 1 )      ) ; | 
|---|
|  | 4308 | entry->size    = (_read_entry( DIR_FILE_SIZE  , buf, 1 )      ) ; | 
|---|
|  | 4309 | entry->is_dir  = ((attr & ATTR_DIRECTORY) == ATTR_DIRECTORY); | 
|---|
|  | 4310 |  | 
|---|
|  | 4311 | if      ( lfn == 0 ) | 
|---|
|  | 4312 | { | 
|---|
|  | 4313 | _get_name_from_short( buf, entry->name ); | 
|---|
|  | 4314 | } | 
|---|
|  | 4315 | else if ( lfn == 1 ) | 
|---|
|  | 4316 | { | 
|---|
|  | 4317 | _strcpy( entry->name     , lfn1 ); | 
|---|
|  | 4318 | } | 
|---|
|  | 4319 | else if ( lfn == 2 ) | 
|---|
|  | 4320 | { | 
|---|
|  | 4321 | _strcpy( entry->name     , lfn1 ); | 
|---|
|  | 4322 | _strcpy( entry->name + 13, lfn2 ); | 
|---|
|  | 4323 | } | 
|---|
|  | 4324 | else if ( lfn == 3 ) | 
|---|
|  | 4325 | { | 
|---|
|  | 4326 | _strcpy( entry->name     , lfn1 ); | 
|---|
|  | 4327 | _strcpy( entry->name + 13, lfn2 ); | 
|---|
|  | 4328 | _strcpy( entry->name + 26, lfn3 ); | 
|---|
|  | 4329 | } | 
|---|
|  | 4330 |  | 
|---|
| [664] | 4331 | return GIET_FAT32_OK; | 
|---|
| [658] | 4332 | } | 
|---|
|  | 4333 |  | 
|---|
|  | 4334 |  | 
|---|
|  | 4335 |  | 
|---|
|  | 4336 |  | 
|---|
| [587] | 4337 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| [674] | 4338 | // This function loads a file identified by the "pathname" argument into the | 
|---|
| [587] | 4339 | // memory buffer defined by the "buffer_vbase" and "buffer_size" arguments. | 
|---|
|  | 4340 | // It is intended to be called by the boot-loader, as it does not use the | 
|---|
|  | 4341 | // dynamically allocated FAT structures (Inode-Tree, Fat_Cache or File-Cache, | 
|---|
|  | 4342 | // File-Descriptor-Array). | 
|---|
|  | 4343 | // It uses only the 512 bytes buffer defined in the FAT descriptor. | 
|---|
|  | 4344 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| [664] | 4345 | // Returns GIET_FAT32_OK on success. | 
|---|
|  | 4346 | // Returns negative value on error: | 
|---|
|  | 4347 | //   GIET_FAT32_NOT_INITIALIZED | 
|---|
|  | 4348 | //   GIET_FAT32_FILE_NOT_FOUND | 
|---|
|  | 4349 | //   GIET_FAT32_BUFFER_TOO_SMALL | 
|---|
|  | 4350 | //   GIET_FAT32_IO_ERROR | 
|---|
| [587] | 4351 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
|  | 4352 | int _fat_load_no_cache( char*        pathname, | 
|---|
|  | 4353 | unsigned int buffer_vbase, | 
|---|
|  | 4354 | unsigned int buffer_size ) | 
|---|
| [258] | 4355 | { | 
|---|
| [674] | 4356 | // checking FAT initialized | 
|---|
|  | 4357 | if( _fat.initialized != FAT_INITIALIZED ) | 
|---|
| [587] | 4358 | { | 
|---|
| [664] | 4359 | _printf("\n[FAT ERROR] _fat_load_no_cache(): FAT not initialized\n"); | 
|---|
|  | 4360 | return GIET_FAT32_NOT_INITIALIZED; | 
|---|
| [587] | 4361 | } | 
|---|
| [258] | 4362 |  | 
|---|
| [587] | 4363 | unsigned int  file_size; | 
|---|
|  | 4364 | unsigned int  cluster; | 
|---|
|  | 4365 |  | 
|---|
|  | 4366 | #if GIET_DEBUG_FAT | 
|---|
|  | 4367 | unsigned int procid  = _get_procid(); | 
|---|
|  | 4368 | unsigned int x       = procid >> (Y_WIDTH + P_WIDTH); | 
|---|
|  | 4369 | unsigned int y       = (procid >> P_WIDTH) & ((1<<Y_WIDTH)-1); | 
|---|
|  | 4370 | unsigned int p       = procid & ((1<<P_WIDTH)-1); | 
|---|
|  | 4371 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 4372 | _printf("\n[DEBUG FAT] _fat_load_no_cache(): P[%d,%d,%d] enters for file <%s>\n", | 
|---|
| [587] | 4373 | x , y , p , pathname ); | 
|---|
|  | 4374 | #endif | 
|---|
|  | 4375 |  | 
|---|
|  | 4376 | // get file size, and cluster index in FAT | 
|---|
|  | 4377 | if ( _file_info_no_cache( pathname, | 
|---|
|  | 4378 | &cluster, | 
|---|
|  | 4379 | &file_size ) ) | 
|---|
|  | 4380 | { | 
|---|
| [664] | 4381 | _printf("\n[FAT ERROR] _fat_load_no_cache(): file <%s> not found\n", | 
|---|
| [587] | 4382 | pathname ); | 
|---|
| [664] | 4383 | return GIET_FAT32_FILE_NOT_FOUND; | 
|---|
| [587] | 4384 | } | 
|---|
|  | 4385 |  | 
|---|
|  | 4386 | // check buffer size | 
|---|
|  | 4387 | if ( file_size > buffer_size ) | 
|---|
|  | 4388 | { | 
|---|
| [664] | 4389 | _printf("\n[FAT ERROR] _fat_load_no_cache(): buffer too small : " | 
|---|
| [587] | 4390 | "file_size = %x / buffer_size = %x", file_size , buffer_size ); | 
|---|
| [664] | 4391 | return GIET_FAT32_BUFFER_TOO_SMALL; | 
|---|
| [587] | 4392 | } | 
|---|
|  | 4393 |  | 
|---|
|  | 4394 | // compute total number of clusters to read | 
|---|
|  | 4395 | unsigned int nb_clusters = file_size >> 12; | 
|---|
|  | 4396 | if ( file_size & 0xFFF ) nb_clusters++; | 
|---|
|  | 4397 |  | 
|---|
|  | 4398 | // initialise buffer address | 
|---|
|  | 4399 | unsigned int dst = buffer_vbase; | 
|---|
|  | 4400 |  | 
|---|
|  | 4401 | // loop on the clusters containing the file | 
|---|
|  | 4402 | while ( nb_clusters > 0 ) | 
|---|
|  | 4403 | { | 
|---|
|  | 4404 | unsigned int lba = _cluster_to_lba( cluster ); | 
|---|
|  | 4405 |  | 
|---|
|  | 4406 | if( _fat_ioc_access( 0,         // no descheduling | 
|---|
|  | 4407 | 1,         // read | 
|---|
|  | 4408 | lba, | 
|---|
|  | 4409 | dst, | 
|---|
|  | 4410 | 8 ) )      // 8 blocks | 
|---|
|  | 4411 | { | 
|---|
| [664] | 4412 | _printf("\n[FAT ERROR] _fat_load_no_cache(): cannot load lba %x", lba ); | 
|---|
|  | 4413 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 4414 | } | 
|---|
|  | 4415 |  | 
|---|
|  | 4416 |  | 
|---|
|  | 4417 | // compute next cluster index | 
|---|
|  | 4418 | unsigned int next; | 
|---|
|  | 4419 | if ( _next_cluster_no_cache( cluster , &next ) ) | 
|---|
|  | 4420 | { | 
|---|
| [664] | 4421 | _printf("\n[FAT ERROR] _fat_load_no_cache(): cannot get next cluster " | 
|---|
| [587] | 4422 | " for cluster = %x\n", cluster ); | 
|---|
| [664] | 4423 | return GIET_FAT32_IO_ERROR; | 
|---|
| [587] | 4424 | } | 
|---|
|  | 4425 |  | 
|---|
|  | 4426 | // update variables for next iteration | 
|---|
|  | 4427 | nb_clusters = nb_clusters - 1; | 
|---|
|  | 4428 | dst         = dst + 4096; | 
|---|
|  | 4429 | cluster     = next; | 
|---|
|  | 4430 | } | 
|---|
|  | 4431 |  | 
|---|
|  | 4432 | #if GIET_DEBUG_FAT | 
|---|
|  | 4433 | if ( _get_proctime() > GIET_DEBUG_FAT ) | 
|---|
| [664] | 4434 | _printf("\n[DEBUG FAT] _fat_load_no_cache(): P[%d,%d,%d] loaded <%s> at vaddr = %x" | 
|---|
| [587] | 4435 | " / size = %x\n", x , y , p , pathname , buffer_vbase , file_size ); | 
|---|
|  | 4436 | #endif | 
|---|
|  | 4437 |  | 
|---|
| [664] | 4438 | return GIET_FAT32_OK; | 
|---|
| [587] | 4439 | }  // end _fat_load_no_cache() | 
|---|
| [258] | 4440 |  | 
|---|
|  | 4441 |  | 
|---|
| [587] | 4442 |  | 
|---|
| [258] | 4443 | // Local Variables: | 
|---|
|  | 4444 | // tab-width: 4 | 
|---|
|  | 4445 | // c-basic-offset: 4 | 
|---|
|  | 4446 | // c-file-offsets:((innamespace . 0)(inline-open . 0)) | 
|---|
|  | 4447 | // indent-tabs-mode: nil | 
|---|
|  | 4448 | // End: | 
|---|
|  | 4449 | // vim: filetype=c:expandtab:shiftwidth=4:tabstop=4:softtabstop=4 | 
|---|
|  | 4450 |  | 
|---|