This is Petit FAT File System Module for mbed NXP LPC1768. The ported library from http://elm-chan.org/fsw/ff/00index_p.html made by Mr. ChaN. The connection is same as SDCard library here http://mbed.org/projects/cookbook/wiki/SDCard . If you need change a pin, You can find the definition at libpff/connect.h :)

Dependencies:   mbed

Committer:
shintamainjp
Date:
Fri Jun 04 23:47:39 2010 +0000
Revision:
1:a24950493dec
Parent:
0:b5453be5ad07

        

Who changed what in which revision?

UserRevisionLine numberNew contents of line
shintamainjp 0:b5453be5ad07 1 /*----------------------------------------------------------------------------/
shintamainjp 0:b5453be5ad07 2 / Petit FatFs - FAT file system module R0.02 (C)ChaN, 2009
shintamainjp 0:b5453be5ad07 3 /-----------------------------------------------------------------------------/
shintamainjp 0:b5453be5ad07 4 / Petit FatFs module is an open source software to implement FAT file system to
shintamainjp 0:b5453be5ad07 5 / small embedded systems. This is a free software and is opened for education,
shintamainjp 0:b5453be5ad07 6 / research and commercial developments under license policy of following trems.
shintamainjp 0:b5453be5ad07 7 /
shintamainjp 0:b5453be5ad07 8 / Copyright (C) 2009, ChaN, all right reserved.
shintamainjp 0:b5453be5ad07 9 /
shintamainjp 0:b5453be5ad07 10 / * The Petit FatFs module is a free software and there is NO WARRANTY.
shintamainjp 0:b5453be5ad07 11 / * No restriction on use. You can use, modify and redistribute it for
shintamainjp 0:b5453be5ad07 12 / personal, non-profit or commercial use UNDER YOUR RESPONSIBILITY.
shintamainjp 0:b5453be5ad07 13 / * Redistributions of source code must retain the above copyright notice.
shintamainjp 0:b5453be5ad07 14 /
shintamainjp 0:b5453be5ad07 15 /-----------------------------------------------------------------------------/
shintamainjp 0:b5453be5ad07 16 / Jun 15,'09 R0.01a First release. (Branched from FatFs R0.07b.)
shintamainjp 0:b5453be5ad07 17 /
shintamainjp 0:b5453be5ad07 18 / Dec 14,'09 R0.02 Added multiple code page support.
shintamainjp 0:b5453be5ad07 19 / Added write funciton.
shintamainjp 0:b5453be5ad07 20 / Changed stream read mode interface.
shintamainjp 0:b5453be5ad07 21 /----------------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 22
shintamainjp 0:b5453be5ad07 23 #include "pff.h" /* Petit FatFs configurations and declarations */
shintamainjp 0:b5453be5ad07 24 #include "diskio.h" /* Declarations of low level disk I/O functions */
shintamainjp 0:b5453be5ad07 25
shintamainjp 0:b5453be5ad07 26
shintamainjp 0:b5453be5ad07 27 /*--------------------------------------------------------------------------
shintamainjp 0:b5453be5ad07 28
shintamainjp 0:b5453be5ad07 29 Private Functions
shintamainjp 0:b5453be5ad07 30
shintamainjp 0:b5453be5ad07 31 ---------------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 32
shintamainjp 0:b5453be5ad07 33 static
shintamainjp 0:b5453be5ad07 34 FATFS *FatFs; /* Pointer to the file system object (logical drive) */
shintamainjp 0:b5453be5ad07 35
shintamainjp 0:b5453be5ad07 36
shintamainjp 0:b5453be5ad07 37
shintamainjp 0:b5453be5ad07 38 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 39 /* String functions */
shintamainjp 0:b5453be5ad07 40 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 41
shintamainjp 0:b5453be5ad07 42 /* Fill memory */
shintamainjp 0:b5453be5ad07 43 static
shintamainjp 0:b5453be5ad07 44 void mem_set (void* dst, int val, int cnt) {
shintamainjp 0:b5453be5ad07 45 char *d = (char*)dst;
shintamainjp 0:b5453be5ad07 46 while (cnt--) *d++ = (char)val;
shintamainjp 0:b5453be5ad07 47 }
shintamainjp 0:b5453be5ad07 48
shintamainjp 0:b5453be5ad07 49 /* Compare memory to memory */
shintamainjp 0:b5453be5ad07 50 static
shintamainjp 0:b5453be5ad07 51 int mem_cmp (const void* dst, const void* src, int cnt) {
shintamainjp 0:b5453be5ad07 52 const char *d = (const char *)dst, *s = (const char *)src;
shintamainjp 0:b5453be5ad07 53 int r = 0;
shintamainjp 0:b5453be5ad07 54 while (cnt-- && (r = *d++ - *s++) == 0) ;
shintamainjp 0:b5453be5ad07 55 return r;
shintamainjp 0:b5453be5ad07 56 }
shintamainjp 0:b5453be5ad07 57
shintamainjp 0:b5453be5ad07 58
shintamainjp 0:b5453be5ad07 59
shintamainjp 0:b5453be5ad07 60 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 61 /* FAT access - Read value of a FAT entry */
shintamainjp 0:b5453be5ad07 62 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 63
shintamainjp 0:b5453be5ad07 64 static
shintamainjp 0:b5453be5ad07 65 CLUST get_fat ( /* 1:IO error, Else:Cluster status */
shintamainjp 0:b5453be5ad07 66 CLUST clst /* Cluster# to get the link information */
shintamainjp 0:b5453be5ad07 67 ) {
shintamainjp 0:b5453be5ad07 68 WORD wc, bc, ofs;
shintamainjp 0:b5453be5ad07 69 BYTE buf[4];
shintamainjp 0:b5453be5ad07 70 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 71
shintamainjp 0:b5453be5ad07 72
shintamainjp 0:b5453be5ad07 73 if (clst < 2 || clst >= fs->max_clust) /* Range check */
shintamainjp 0:b5453be5ad07 74 return 1;
shintamainjp 0:b5453be5ad07 75
shintamainjp 0:b5453be5ad07 76 switch (fs->fs_type) {
shintamainjp 0:b5453be5ad07 77 case FS_FAT12 :
shintamainjp 0:b5453be5ad07 78 bc = (WORD)clst;
shintamainjp 0:b5453be5ad07 79 bc += bc / 2;
shintamainjp 0:b5453be5ad07 80 ofs = bc % 512;
shintamainjp 0:b5453be5ad07 81 bc /= 512;
shintamainjp 0:b5453be5ad07 82 if (ofs != 511) {
shintamainjp 0:b5453be5ad07 83 if (disk_readp(buf, fs->fatbase + bc, ofs, 2)) break;
shintamainjp 0:b5453be5ad07 84 } else {
shintamainjp 0:b5453be5ad07 85 if (disk_readp(buf, fs->fatbase + bc, 511, 1)) break;
shintamainjp 0:b5453be5ad07 86 if (disk_readp(buf+1, fs->fatbase + bc + 1, 0, 1)) break;
shintamainjp 0:b5453be5ad07 87 }
shintamainjp 0:b5453be5ad07 88 wc = LD_WORD(buf);
shintamainjp 0:b5453be5ad07 89 return (clst & 1) ? (wc >> 4) : (wc & 0xFFF);
shintamainjp 0:b5453be5ad07 90
shintamainjp 0:b5453be5ad07 91 case FS_FAT16 :
shintamainjp 0:b5453be5ad07 92 if (disk_readp(buf, fs->fatbase + clst / 256, (WORD)(((WORD)clst % 256) * 2), 2)) break;
shintamainjp 0:b5453be5ad07 93 return LD_WORD(buf);
shintamainjp 0:b5453be5ad07 94 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 95 case FS_FAT32 :
shintamainjp 0:b5453be5ad07 96 if (disk_readp(buf, fs->fatbase + clst / 128, (WORD)(((WORD)clst % 128) * 4), 4)) break;
shintamainjp 0:b5453be5ad07 97 return LD_DWORD(buf) & 0x0FFFFFFF;
shintamainjp 0:b5453be5ad07 98 #endif
shintamainjp 0:b5453be5ad07 99 }
shintamainjp 0:b5453be5ad07 100
shintamainjp 0:b5453be5ad07 101 return 1; /* An error occured at the disk I/O layer */
shintamainjp 0:b5453be5ad07 102 }
shintamainjp 0:b5453be5ad07 103
shintamainjp 0:b5453be5ad07 104
shintamainjp 0:b5453be5ad07 105
shintamainjp 0:b5453be5ad07 106
shintamainjp 0:b5453be5ad07 107 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 108 /* Get sector# from cluster# */
shintamainjp 0:b5453be5ad07 109 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 110
shintamainjp 0:b5453be5ad07 111 static
shintamainjp 0:b5453be5ad07 112 DWORD clust2sect ( /* !=0: Sector number, 0: Failed - invalid cluster# */
shintamainjp 0:b5453be5ad07 113 CLUST clst /* Cluster# to be converted */
shintamainjp 0:b5453be5ad07 114 ) {
shintamainjp 0:b5453be5ad07 115 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 116
shintamainjp 0:b5453be5ad07 117
shintamainjp 0:b5453be5ad07 118 clst -= 2;
shintamainjp 0:b5453be5ad07 119 if (clst >= (fs->max_clust - 2)) return 0; /* Invalid cluster# */
shintamainjp 0:b5453be5ad07 120 return (DWORD)clst * fs->csize + fs->database;
shintamainjp 0:b5453be5ad07 121 }
shintamainjp 0:b5453be5ad07 122
shintamainjp 0:b5453be5ad07 123
shintamainjp 0:b5453be5ad07 124
shintamainjp 0:b5453be5ad07 125
shintamainjp 0:b5453be5ad07 126 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 127 /* Directory handling - Rewind directory index */
shintamainjp 0:b5453be5ad07 128 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 129
shintamainjp 0:b5453be5ad07 130 static
shintamainjp 0:b5453be5ad07 131 FRESULT dir_rewind (
shintamainjp 0:b5453be5ad07 132 FATDIR *dj /* Pointer to directory object */
shintamainjp 0:b5453be5ad07 133 ) {
shintamainjp 0:b5453be5ad07 134 CLUST clst;
shintamainjp 0:b5453be5ad07 135 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 136
shintamainjp 0:b5453be5ad07 137
shintamainjp 0:b5453be5ad07 138 dj->index = 0;
shintamainjp 0:b5453be5ad07 139 clst = dj->sclust;
shintamainjp 0:b5453be5ad07 140 if (clst == 1 || clst >= fs->max_clust) /* Check start cluster range */
shintamainjp 0:b5453be5ad07 141 return FR_DISK_ERR;
shintamainjp 0:b5453be5ad07 142 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 143 if (!clst && fs->fs_type == FS_FAT32) /* Replace cluster# 0 with root cluster# if in FAT32 */
shintamainjp 0:b5453be5ad07 144 clst = fs->dirbase;
shintamainjp 0:b5453be5ad07 145 #endif
shintamainjp 0:b5453be5ad07 146 dj->clust = clst; /* Current cluster */
shintamainjp 0:b5453be5ad07 147 dj->sect = clst ? clust2sect(clst) : fs->dirbase; /* Current sector */
shintamainjp 0:b5453be5ad07 148
shintamainjp 0:b5453be5ad07 149 return FR_OK; /* Seek succeeded */
shintamainjp 0:b5453be5ad07 150 }
shintamainjp 0:b5453be5ad07 151
shintamainjp 0:b5453be5ad07 152
shintamainjp 0:b5453be5ad07 153
shintamainjp 0:b5453be5ad07 154
shintamainjp 0:b5453be5ad07 155 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 156 /* Directory handling - Move directory index next */
shintamainjp 0:b5453be5ad07 157 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 158
shintamainjp 0:b5453be5ad07 159 static
shintamainjp 0:b5453be5ad07 160 FRESULT dir_next ( /* FR_OK:Succeeded, FR_NO_FILE:End of table */
shintamainjp 0:b5453be5ad07 161 FATDIR *dj /* Pointer to directory object */
shintamainjp 0:b5453be5ad07 162 ) {
shintamainjp 0:b5453be5ad07 163 CLUST clst;
shintamainjp 0:b5453be5ad07 164 WORD i;
shintamainjp 0:b5453be5ad07 165 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 166
shintamainjp 0:b5453be5ad07 167
shintamainjp 0:b5453be5ad07 168 i = dj->index + 1;
shintamainjp 0:b5453be5ad07 169 if (!i || !dj->sect) /* Report EOT when index has reached 65535 */
shintamainjp 0:b5453be5ad07 170 return FR_NO_FILE;
shintamainjp 0:b5453be5ad07 171
shintamainjp 0:b5453be5ad07 172 if (!(i & (16-1))) { /* Sector changed? */
shintamainjp 0:b5453be5ad07 173 dj->sect++; /* Next sector */
shintamainjp 0:b5453be5ad07 174
shintamainjp 0:b5453be5ad07 175 if (dj->clust == 0) { /* Static table */
shintamainjp 0:b5453be5ad07 176 if (i >= fs->n_rootdir) /* Report EOT when end of table */
shintamainjp 0:b5453be5ad07 177 return FR_NO_FILE;
shintamainjp 0:b5453be5ad07 178 } else { /* Dynamic table */
shintamainjp 0:b5453be5ad07 179 if (((i / 16) & (fs->csize-1)) == 0) { /* Cluster changed? */
shintamainjp 0:b5453be5ad07 180 clst = get_fat(dj->clust); /* Get next cluster */
shintamainjp 0:b5453be5ad07 181 if (clst <= 1) return FR_DISK_ERR;
shintamainjp 0:b5453be5ad07 182 if (clst >= fs->max_clust) /* When it reached end of dynamic table */
shintamainjp 0:b5453be5ad07 183 return FR_NO_FILE; /* Report EOT */
shintamainjp 0:b5453be5ad07 184 dj->clust = clst; /* Initialize data for new cluster */
shintamainjp 0:b5453be5ad07 185 dj->sect = clust2sect(clst);
shintamainjp 0:b5453be5ad07 186 }
shintamainjp 0:b5453be5ad07 187 }
shintamainjp 0:b5453be5ad07 188 }
shintamainjp 0:b5453be5ad07 189
shintamainjp 0:b5453be5ad07 190 dj->index = i;
shintamainjp 0:b5453be5ad07 191
shintamainjp 0:b5453be5ad07 192 return FR_OK;
shintamainjp 0:b5453be5ad07 193 }
shintamainjp 0:b5453be5ad07 194
shintamainjp 0:b5453be5ad07 195
shintamainjp 0:b5453be5ad07 196
shintamainjp 0:b5453be5ad07 197
shintamainjp 0:b5453be5ad07 198 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 199 /* Directory handling - Find an object in the directory */
shintamainjp 0:b5453be5ad07 200 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 201
shintamainjp 0:b5453be5ad07 202 static
shintamainjp 0:b5453be5ad07 203 FRESULT dir_find (
shintamainjp 0:b5453be5ad07 204 FATDIR *dj /* Pointer to the directory object linked to the file name */
shintamainjp 0:b5453be5ad07 205 ) {
shintamainjp 0:b5453be5ad07 206 FRESULT res;
shintamainjp 0:b5453be5ad07 207 BYTE c, *dir;
shintamainjp 0:b5453be5ad07 208
shintamainjp 0:b5453be5ad07 209
shintamainjp 0:b5453be5ad07 210 res = dir_rewind(dj); /* Rewind directory object */
shintamainjp 0:b5453be5ad07 211 if (res != FR_OK) return res;
shintamainjp 0:b5453be5ad07 212
shintamainjp 0:b5453be5ad07 213 dir = FatFs->buf;
shintamainjp 0:b5453be5ad07 214 do {
shintamainjp 0:b5453be5ad07 215 res = disk_readp(dir, dj->sect, (WORD)((dj->index % 16) * 32), 32) /* Read an entry */
shintamainjp 0:b5453be5ad07 216 ? FR_DISK_ERR : FR_OK;
shintamainjp 0:b5453be5ad07 217 if (res != FR_OK) break;
shintamainjp 0:b5453be5ad07 218 c = dir[DIR_Name]; /* First character */
shintamainjp 0:b5453be5ad07 219 if (c == 0) {
shintamainjp 0:b5453be5ad07 220 res = FR_NO_FILE; /* Reached to end of table */
shintamainjp 0:b5453be5ad07 221 break;
shintamainjp 0:b5453be5ad07 222 }
shintamainjp 0:b5453be5ad07 223 if (!(dir[DIR_Attr] & AM_VOL) && !mem_cmp(dir, dj->fn, 11)) /* Is it a valid entry? */
shintamainjp 0:b5453be5ad07 224 break;
shintamainjp 0:b5453be5ad07 225 res = dir_next(dj); /* Next entry */
shintamainjp 0:b5453be5ad07 226 } while (res == FR_OK);
shintamainjp 0:b5453be5ad07 227
shintamainjp 0:b5453be5ad07 228 return res;
shintamainjp 0:b5453be5ad07 229 }
shintamainjp 0:b5453be5ad07 230
shintamainjp 0:b5453be5ad07 231
shintamainjp 0:b5453be5ad07 232
shintamainjp 0:b5453be5ad07 233
shintamainjp 0:b5453be5ad07 234 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 235 /* Read an object from the directory */
shintamainjp 0:b5453be5ad07 236 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 237 #if _USE_DIR
shintamainjp 0:b5453be5ad07 238 static
shintamainjp 0:b5453be5ad07 239 FRESULT dir_read (
shintamainjp 0:b5453be5ad07 240 FATDIR *dj /* Pointer to the directory object to store read object name */
shintamainjp 0:b5453be5ad07 241 ) {
shintamainjp 0:b5453be5ad07 242 FRESULT res;
shintamainjp 0:b5453be5ad07 243 BYTE a, c, *dir;
shintamainjp 0:b5453be5ad07 244
shintamainjp 0:b5453be5ad07 245
shintamainjp 0:b5453be5ad07 246 res = FR_NO_FILE;
shintamainjp 0:b5453be5ad07 247 dir = FatFs->buf;
shintamainjp 0:b5453be5ad07 248 while (dj->sect) {
shintamainjp 0:b5453be5ad07 249 res = disk_readp(dir, dj->sect, (WORD)((dj->index % 16) * 32), 32) /* Read an entry */
shintamainjp 0:b5453be5ad07 250 ? FR_DISK_ERR : FR_OK;
shintamainjp 0:b5453be5ad07 251 if (res != FR_OK) break;
shintamainjp 0:b5453be5ad07 252 c = dir[DIR_Name];
shintamainjp 0:b5453be5ad07 253 if (c == 0) {
shintamainjp 0:b5453be5ad07 254 res = FR_NO_FILE; /* Reached to end of table */
shintamainjp 0:b5453be5ad07 255 break;
shintamainjp 0:b5453be5ad07 256 }
shintamainjp 0:b5453be5ad07 257 a = dir[DIR_Attr] & AM_MASK;
shintamainjp 0:b5453be5ad07 258 if (c != 0xE5 && c != '.' && !(a & AM_VOL)) /* Is it a valid entry? */
shintamainjp 0:b5453be5ad07 259 break;
shintamainjp 0:b5453be5ad07 260 res = dir_next(dj); /* Next entry */
shintamainjp 0:b5453be5ad07 261 if (res != FR_OK) break;
shintamainjp 0:b5453be5ad07 262 }
shintamainjp 0:b5453be5ad07 263
shintamainjp 0:b5453be5ad07 264 if (res != FR_OK) dj->sect = 0;
shintamainjp 0:b5453be5ad07 265
shintamainjp 0:b5453be5ad07 266 return res;
shintamainjp 0:b5453be5ad07 267 }
shintamainjp 0:b5453be5ad07 268 #endif
shintamainjp 0:b5453be5ad07 269
shintamainjp 0:b5453be5ad07 270
shintamainjp 0:b5453be5ad07 271
shintamainjp 0:b5453be5ad07 272 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 273 /* Pick a segment and create the object name in directory form */
shintamainjp 0:b5453be5ad07 274 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 275
shintamainjp 0:b5453be5ad07 276 #ifdef _EXCVT
shintamainjp 0:b5453be5ad07 277 static const BYTE cvt[] = _EXCVT;
shintamainjp 0:b5453be5ad07 278 #endif
shintamainjp 0:b5453be5ad07 279
shintamainjp 0:b5453be5ad07 280 static
shintamainjp 0:b5453be5ad07 281 FRESULT create_name (
shintamainjp 0:b5453be5ad07 282 FATDIR *dj, /* Pointer to the directory object */
shintamainjp 0:b5453be5ad07 283 const char **path /* Pointer to pointer to the segment in the path string */
shintamainjp 0:b5453be5ad07 284 ) {
shintamainjp 0:b5453be5ad07 285 BYTE c, d, ni, si, i, *sfn;
shintamainjp 0:b5453be5ad07 286 const char *p;
shintamainjp 0:b5453be5ad07 287
shintamainjp 0:b5453be5ad07 288 /* Create file name in directory form */
shintamainjp 0:b5453be5ad07 289 sfn = dj->fn;
shintamainjp 0:b5453be5ad07 290 mem_set(sfn, ' ', 11);
shintamainjp 0:b5453be5ad07 291 si = i = 0;
shintamainjp 0:b5453be5ad07 292 ni = 8;
shintamainjp 0:b5453be5ad07 293 p = *path;
shintamainjp 0:b5453be5ad07 294 for (;;) {
shintamainjp 0:b5453be5ad07 295 c = p[si++];
shintamainjp 0:b5453be5ad07 296 if (c <= ' ' || c == '/') break; /* Break on end of segment */
shintamainjp 0:b5453be5ad07 297 if (c == '.' || i >= ni) {
shintamainjp 0:b5453be5ad07 298 if (ni != 8 || c != '.') break;
shintamainjp 0:b5453be5ad07 299 i = 8;
shintamainjp 0:b5453be5ad07 300 ni = 11;
shintamainjp 0:b5453be5ad07 301 continue;
shintamainjp 0:b5453be5ad07 302 }
shintamainjp 0:b5453be5ad07 303 #ifdef _EXCVT
shintamainjp 0:b5453be5ad07 304 if (c >= 0x80) /* To upper extended char (SBCS) */
shintamainjp 0:b5453be5ad07 305 c = cvt[c - 0x80];
shintamainjp 0:b5453be5ad07 306 #endif
shintamainjp 0:b5453be5ad07 307 if (IsDBCS1(c) && i >= ni - 1) { /* DBC 1st byte? */
shintamainjp 0:b5453be5ad07 308 d = p[si++]; /* Get 2nd byte */
shintamainjp 0:b5453be5ad07 309 sfn[i++] = c;
shintamainjp 0:b5453be5ad07 310 sfn[i++] = d;
shintamainjp 0:b5453be5ad07 311 } else { /* Single byte code */
shintamainjp 0:b5453be5ad07 312 if (IsLower(c)) c -= 0x20; /* toupper */
shintamainjp 0:b5453be5ad07 313 sfn[i++] = c;
shintamainjp 0:b5453be5ad07 314 }
shintamainjp 0:b5453be5ad07 315 }
shintamainjp 0:b5453be5ad07 316 *path = &p[si]; /* Rerurn pointer to the next segment */
shintamainjp 0:b5453be5ad07 317
shintamainjp 0:b5453be5ad07 318 sfn[11] = (c <= ' ') ? 1 : 0; /* Set last segment flag if end of path */
shintamainjp 0:b5453be5ad07 319
shintamainjp 0:b5453be5ad07 320 return FR_OK;
shintamainjp 0:b5453be5ad07 321 }
shintamainjp 0:b5453be5ad07 322
shintamainjp 0:b5453be5ad07 323
shintamainjp 0:b5453be5ad07 324
shintamainjp 0:b5453be5ad07 325
shintamainjp 0:b5453be5ad07 326 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 327 /* Get file information from directory entry */
shintamainjp 0:b5453be5ad07 328 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 329 #if _USE_DIR
shintamainjp 0:b5453be5ad07 330 static
shintamainjp 0:b5453be5ad07 331 void get_fileinfo ( /* No return code */
shintamainjp 0:b5453be5ad07 332 FATDIR *dj, /* Pointer to the directory object */
shintamainjp 0:b5453be5ad07 333 FILINFO *fno /* Pointer to store the file information */
shintamainjp 0:b5453be5ad07 334 ) {
shintamainjp 0:b5453be5ad07 335 BYTE i, c, *dir;
shintamainjp 0:b5453be5ad07 336 char *p;
shintamainjp 0:b5453be5ad07 337
shintamainjp 0:b5453be5ad07 338
shintamainjp 0:b5453be5ad07 339 p = fno->fname;
shintamainjp 0:b5453be5ad07 340 if (dj->sect) {
shintamainjp 0:b5453be5ad07 341 dir = FatFs->buf;
shintamainjp 0:b5453be5ad07 342 for (i = 0; i < 8; i++) { /* Copy file name body */
shintamainjp 0:b5453be5ad07 343 c = dir[i];
shintamainjp 0:b5453be5ad07 344 if (c == ' ') break;
shintamainjp 0:b5453be5ad07 345 if (c == 0x05) c = 0xE5;
shintamainjp 0:b5453be5ad07 346 *p++ = c;
shintamainjp 0:b5453be5ad07 347 }
shintamainjp 0:b5453be5ad07 348 if (dir[8] != ' ') { /* Copy file name extension */
shintamainjp 0:b5453be5ad07 349 *p++ = '.';
shintamainjp 0:b5453be5ad07 350 for (i = 8; i < 11; i++) {
shintamainjp 0:b5453be5ad07 351 c = dir[i];
shintamainjp 0:b5453be5ad07 352 if (c == ' ') break;
shintamainjp 0:b5453be5ad07 353 *p++ = c;
shintamainjp 0:b5453be5ad07 354 }
shintamainjp 0:b5453be5ad07 355 }
shintamainjp 0:b5453be5ad07 356 fno->fattrib = dir[DIR_Attr]; /* Attribute */
shintamainjp 0:b5453be5ad07 357 fno->fsize = LD_DWORD(dir+DIR_FileSize); /* Size */
shintamainjp 0:b5453be5ad07 358 fno->fdate = LD_WORD(dir+DIR_WrtDate); /* Date */
shintamainjp 0:b5453be5ad07 359 fno->ftime = LD_WORD(dir+DIR_WrtTime); /* Time */
shintamainjp 0:b5453be5ad07 360 }
shintamainjp 0:b5453be5ad07 361 *p = 0;
shintamainjp 0:b5453be5ad07 362 }
shintamainjp 0:b5453be5ad07 363 #endif /* _USE_DIR */
shintamainjp 0:b5453be5ad07 364
shintamainjp 0:b5453be5ad07 365
shintamainjp 0:b5453be5ad07 366
shintamainjp 0:b5453be5ad07 367 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 368 /* Follow a file path */
shintamainjp 0:b5453be5ad07 369 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 370
shintamainjp 0:b5453be5ad07 371 static
shintamainjp 0:b5453be5ad07 372 FRESULT follow_path ( /* FR_OK(0): successful, !=0: error code */
shintamainjp 0:b5453be5ad07 373 FATDIR *dj, /* Directory object to return last directory and found object */
shintamainjp 0:b5453be5ad07 374 const char *path /* Full-path string to find a file or directory */
shintamainjp 0:b5453be5ad07 375 ) {
shintamainjp 0:b5453be5ad07 376 FRESULT res;
shintamainjp 0:b5453be5ad07 377 BYTE *dir;
shintamainjp 0:b5453be5ad07 378
shintamainjp 0:b5453be5ad07 379
shintamainjp 0:b5453be5ad07 380 while (*path == ' ') path++; /* Skip leading spaces */
shintamainjp 0:b5453be5ad07 381 if (*path == '/') path++; /* Strip heading separator */
shintamainjp 0:b5453be5ad07 382 dj->sclust = 0; /* Set start directory (always root dir) */
shintamainjp 0:b5453be5ad07 383
shintamainjp 0:b5453be5ad07 384 if ((BYTE)*path <= ' ') { /* Null path means the root directory */
shintamainjp 0:b5453be5ad07 385 res = dir_rewind(dj);
shintamainjp 0:b5453be5ad07 386 FatFs->buf[0] = 0;
shintamainjp 0:b5453be5ad07 387
shintamainjp 0:b5453be5ad07 388 } else { /* Follow path */
shintamainjp 0:b5453be5ad07 389 for (;;) {
shintamainjp 0:b5453be5ad07 390 res = create_name(dj, &path); /* Get a segment */
shintamainjp 0:b5453be5ad07 391 if (res != FR_OK) break;
shintamainjp 0:b5453be5ad07 392 res = dir_find(dj); /* Find it */
shintamainjp 0:b5453be5ad07 393 if (res != FR_OK) { /* Could not find the object */
shintamainjp 0:b5453be5ad07 394 if (res == FR_NO_FILE && !*(dj->fn+11))
shintamainjp 0:b5453be5ad07 395 res = FR_NO_PATH;
shintamainjp 0:b5453be5ad07 396 break;
shintamainjp 0:b5453be5ad07 397 }
shintamainjp 0:b5453be5ad07 398 if (*(dj->fn+11)) break; /* Last segment match. Function completed. */
shintamainjp 0:b5453be5ad07 399 dir = FatFs->buf; /* There is next segment. Follow the sub directory */
shintamainjp 0:b5453be5ad07 400 if (!(dir[DIR_Attr] & AM_DIR)) { /* Cannot follow because it is a file */
shintamainjp 0:b5453be5ad07 401 res = FR_NO_PATH;
shintamainjp 0:b5453be5ad07 402 break;
shintamainjp 0:b5453be5ad07 403 }
shintamainjp 0:b5453be5ad07 404 dj->sclust =
shintamainjp 0:b5453be5ad07 405 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 406 ((DWORD)LD_WORD(dir+DIR_FstClusHI) << 16) |
shintamainjp 0:b5453be5ad07 407 #endif
shintamainjp 0:b5453be5ad07 408 LD_WORD(dir+DIR_FstClusLO);
shintamainjp 0:b5453be5ad07 409 }
shintamainjp 0:b5453be5ad07 410 }
shintamainjp 0:b5453be5ad07 411
shintamainjp 0:b5453be5ad07 412 return res;
shintamainjp 0:b5453be5ad07 413 }
shintamainjp 0:b5453be5ad07 414
shintamainjp 0:b5453be5ad07 415
shintamainjp 0:b5453be5ad07 416
shintamainjp 0:b5453be5ad07 417
shintamainjp 0:b5453be5ad07 418 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 419 /* Check a sector if it is an FAT boot record */
shintamainjp 0:b5453be5ad07 420 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 421
shintamainjp 0:b5453be5ad07 422 static
shintamainjp 0:b5453be5ad07 423 BYTE check_fs ( /* 0:The FAT boot record, 1:Valid boot record but not an FAT, 2:Not a boot record, 3:Error */
shintamainjp 0:b5453be5ad07 424 BYTE *buf, /* Working buffer */
shintamainjp 0:b5453be5ad07 425 DWORD sect /* Sector# (lba) to check if it is an FAT boot record or not */
shintamainjp 0:b5453be5ad07 426 ) {
shintamainjp 0:b5453be5ad07 427 if (disk_readp(buf, sect, 510, 2)) /* Read the boot sector */
shintamainjp 0:b5453be5ad07 428 return 3;
shintamainjp 0:b5453be5ad07 429 if (LD_WORD(buf) != 0xAA55) /* Check record signature */
shintamainjp 0:b5453be5ad07 430 return 2;
shintamainjp 0:b5453be5ad07 431
shintamainjp 0:b5453be5ad07 432 if (!disk_readp(buf, sect, BS_FilSysType, 2) && LD_WORD(buf) == 0x4146) /* Check FAT12/16 */
shintamainjp 0:b5453be5ad07 433 return 0;
shintamainjp 0:b5453be5ad07 434 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 435 if (!disk_readp(buf, sect, BS_FilSysType32, 2) && LD_WORD(buf) == 0x4146) /* Check FAT32 */
shintamainjp 0:b5453be5ad07 436 return 0;
shintamainjp 0:b5453be5ad07 437 #endif
shintamainjp 0:b5453be5ad07 438 return 1;
shintamainjp 0:b5453be5ad07 439 }
shintamainjp 0:b5453be5ad07 440
shintamainjp 0:b5453be5ad07 441
shintamainjp 0:b5453be5ad07 442
shintamainjp 0:b5453be5ad07 443
shintamainjp 0:b5453be5ad07 444 /*--------------------------------------------------------------------------
shintamainjp 0:b5453be5ad07 445
shintamainjp 0:b5453be5ad07 446 Public Functions
shintamainjp 0:b5453be5ad07 447
shintamainjp 0:b5453be5ad07 448 --------------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 449
shintamainjp 0:b5453be5ad07 450
shintamainjp 0:b5453be5ad07 451
shintamainjp 0:b5453be5ad07 452 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 453 /* Mount/Unmount a Locical Drive */
shintamainjp 0:b5453be5ad07 454 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 455
shintamainjp 0:b5453be5ad07 456 FRESULT pf_mount (
shintamainjp 0:b5453be5ad07 457 FATFS *fs /* Pointer to new file system object (NULL: Unmount) */
shintamainjp 0:b5453be5ad07 458 ) {
shintamainjp 0:b5453be5ad07 459 BYTE fmt, buf[36];
shintamainjp 0:b5453be5ad07 460 DWORD bsect, fsize, tsect, mclst;
shintamainjp 0:b5453be5ad07 461
shintamainjp 0:b5453be5ad07 462
shintamainjp 0:b5453be5ad07 463 FatFs = 0;
shintamainjp 0:b5453be5ad07 464 if (!fs) return FR_OK; /* Unregister fs object */
shintamainjp 0:b5453be5ad07 465
shintamainjp 0:b5453be5ad07 466 if (disk_initialize() & STA_NOINIT) /* Check if the drive is ready or not */
shintamainjp 0:b5453be5ad07 467 return FR_NOT_READY;
shintamainjp 0:b5453be5ad07 468
shintamainjp 0:b5453be5ad07 469 /* Search FAT partition on the drive */
shintamainjp 0:b5453be5ad07 470 bsect = 0;
shintamainjp 0:b5453be5ad07 471 fmt = check_fs(buf, bsect); /* Check sector 0 as an SFD format */
shintamainjp 0:b5453be5ad07 472 if (fmt == 1) { /* Not an FAT boot record, it may be FDISK format */
shintamainjp 0:b5453be5ad07 473 /* Check a partition listed in top of the partition table */
shintamainjp 0:b5453be5ad07 474 if (disk_readp(buf, bsect, MBR_Table, 16)) { /* 1st partition entry */
shintamainjp 0:b5453be5ad07 475 fmt = 3;
shintamainjp 0:b5453be5ad07 476 } else {
shintamainjp 0:b5453be5ad07 477 if (buf[4]) { /* Is the partition existing? */
shintamainjp 0:b5453be5ad07 478 bsect = LD_DWORD(&buf[8]); /* Partition offset in LBA */
shintamainjp 0:b5453be5ad07 479 fmt = check_fs(buf, bsect); /* Check the partition */
shintamainjp 0:b5453be5ad07 480 }
shintamainjp 0:b5453be5ad07 481 }
shintamainjp 0:b5453be5ad07 482 }
shintamainjp 0:b5453be5ad07 483 if (fmt == 3) return FR_DISK_ERR;
shintamainjp 0:b5453be5ad07 484 if (fmt) return FR_NO_FILESYSTEM; /* No valid FAT patition is found */
shintamainjp 0:b5453be5ad07 485
shintamainjp 0:b5453be5ad07 486 /* Initialize the file system object */
shintamainjp 0:b5453be5ad07 487 if (disk_readp(buf, bsect, 13, sizeof(buf))) return FR_DISK_ERR;
shintamainjp 0:b5453be5ad07 488
shintamainjp 0:b5453be5ad07 489 fsize = LD_WORD(buf+BPB_FATSz16-13); /* Number of sectors per FAT */
shintamainjp 0:b5453be5ad07 490 if (!fsize) fsize = LD_DWORD(buf+BPB_FATSz32-13);
shintamainjp 0:b5453be5ad07 491
shintamainjp 0:b5453be5ad07 492 fsize *= buf[BPB_NumFATs-13]; /* Number of sectors in FAT area */
shintamainjp 0:b5453be5ad07 493 fs->fatbase = bsect + LD_WORD(buf+BPB_RsvdSecCnt-13); /* FAT start sector (lba) */
shintamainjp 0:b5453be5ad07 494 fs->csize = buf[BPB_SecPerClus-13]; /* Number of sectors per cluster */
shintamainjp 0:b5453be5ad07 495 fs->n_rootdir = LD_WORD(buf+BPB_RootEntCnt-13); /* Nmuber of root directory entries */
shintamainjp 0:b5453be5ad07 496 tsect = LD_WORD(buf+BPB_TotSec16-13); /* Number of sectors on the file system */
shintamainjp 0:b5453be5ad07 497 if (!tsect) tsect = LD_DWORD(buf+BPB_TotSec32-13);
shintamainjp 0:b5453be5ad07 498 mclst = (tsect /* Last cluster# + 1 */
shintamainjp 0:b5453be5ad07 499 - LD_WORD(buf+BPB_RsvdSecCnt-13) - fsize - fs->n_rootdir / 16
shintamainjp 0:b5453be5ad07 500 ) / fs->csize + 2;
shintamainjp 0:b5453be5ad07 501 fs->max_clust = (CLUST)mclst;
shintamainjp 0:b5453be5ad07 502
shintamainjp 0:b5453be5ad07 503 fmt = FS_FAT12; /* Determine the FAT sub type */
shintamainjp 0:b5453be5ad07 504 if (mclst >= 0xFF7) fmt = FS_FAT16; /* Number of clusters >= 0xFF5 */
shintamainjp 0:b5453be5ad07 505 if (mclst >= 0xFFF7) /* Number of clusters >= 0xFFF5 */
shintamainjp 0:b5453be5ad07 506 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 507 fmt = FS_FAT32;
shintamainjp 0:b5453be5ad07 508 #else
shintamainjp 0:b5453be5ad07 509 return FR_NO_FILESYSTEM;
shintamainjp 0:b5453be5ad07 510 #endif
shintamainjp 0:b5453be5ad07 511
shintamainjp 0:b5453be5ad07 512 fs->fs_type = fmt; /* FAT sub-type */
shintamainjp 0:b5453be5ad07 513 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 514 if (fmt == FS_FAT32)
shintamainjp 0:b5453be5ad07 515 fs->dirbase = LD_DWORD(buf+(BPB_RootClus-13)); /* Root directory start cluster */
shintamainjp 0:b5453be5ad07 516 else
shintamainjp 0:b5453be5ad07 517 #endif
shintamainjp 0:b5453be5ad07 518 fs->dirbase = fs->fatbase + fsize; /* Root directory start sector (lba) */
shintamainjp 0:b5453be5ad07 519 fs->database = fs->fatbase + fsize + fs->n_rootdir / 16; /* Data start sector (lba) */
shintamainjp 0:b5453be5ad07 520
shintamainjp 0:b5453be5ad07 521 fs->flag = 0;
shintamainjp 0:b5453be5ad07 522 FatFs = fs;
shintamainjp 0:b5453be5ad07 523
shintamainjp 0:b5453be5ad07 524 return FR_OK;
shintamainjp 0:b5453be5ad07 525 }
shintamainjp 0:b5453be5ad07 526
shintamainjp 0:b5453be5ad07 527
shintamainjp 0:b5453be5ad07 528
shintamainjp 0:b5453be5ad07 529
shintamainjp 0:b5453be5ad07 530 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 531 /* Open or Create a File */
shintamainjp 0:b5453be5ad07 532 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 533
shintamainjp 0:b5453be5ad07 534 FRESULT pf_open (
shintamainjp 0:b5453be5ad07 535 const char *path /* Pointer to the file name */
shintamainjp 0:b5453be5ad07 536 ) {
shintamainjp 0:b5453be5ad07 537 FRESULT res;
shintamainjp 0:b5453be5ad07 538 FATDIR dj;
shintamainjp 0:b5453be5ad07 539 BYTE sp[12], dir[32];
shintamainjp 0:b5453be5ad07 540 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 541
shintamainjp 0:b5453be5ad07 542
shintamainjp 0:b5453be5ad07 543 if (!fs) /* Check file system */
shintamainjp 0:b5453be5ad07 544 return FR_NOT_ENABLED;
shintamainjp 0:b5453be5ad07 545
shintamainjp 0:b5453be5ad07 546 fs->flag = 0;
shintamainjp 0:b5453be5ad07 547 fs->buf = dir;
shintamainjp 0:b5453be5ad07 548 dj.fn = sp;
shintamainjp 0:b5453be5ad07 549 res = follow_path(&dj, path); /* Follow the file path */
shintamainjp 0:b5453be5ad07 550 if (res != FR_OK) return res; /* Follow failed */
shintamainjp 0:b5453be5ad07 551 if (!dir[0] || (dir[DIR_Attr] & AM_DIR)) /* It is a directory */
shintamainjp 0:b5453be5ad07 552 return FR_NO_FILE;
shintamainjp 0:b5453be5ad07 553
shintamainjp 0:b5453be5ad07 554 fs->org_clust = /* File start cluster */
shintamainjp 0:b5453be5ad07 555 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 556 ((DWORD)LD_WORD(dir+DIR_FstClusHI) << 16) |
shintamainjp 0:b5453be5ad07 557 #endif
shintamainjp 0:b5453be5ad07 558 LD_WORD(dir+DIR_FstClusLO);
shintamainjp 0:b5453be5ad07 559 fs->fsize = LD_DWORD(dir+DIR_FileSize); /* File size */
shintamainjp 0:b5453be5ad07 560 fs->fptr = 0; /* File pointer */
shintamainjp 0:b5453be5ad07 561 fs->flag = FA_OPENED;
shintamainjp 0:b5453be5ad07 562
shintamainjp 0:b5453be5ad07 563 return FR_OK;
shintamainjp 0:b5453be5ad07 564 }
shintamainjp 0:b5453be5ad07 565
shintamainjp 0:b5453be5ad07 566
shintamainjp 0:b5453be5ad07 567
shintamainjp 0:b5453be5ad07 568
shintamainjp 0:b5453be5ad07 569 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 570 /* Read File */
shintamainjp 0:b5453be5ad07 571 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 572 #if _USE_READ
shintamainjp 0:b5453be5ad07 573
shintamainjp 0:b5453be5ad07 574 FRESULT pf_read (
shintamainjp 0:b5453be5ad07 575 void* buff, /* Pointer to the read buffer (NULL:Forward data to the stream)*/
shintamainjp 0:b5453be5ad07 576 WORD btr, /* Number of bytes to read */
shintamainjp 0:b5453be5ad07 577 WORD* br /* Pointer to number of bytes read */
shintamainjp 0:b5453be5ad07 578 ) {
shintamainjp 0:b5453be5ad07 579 DRESULT dr;
shintamainjp 0:b5453be5ad07 580 CLUST clst;
shintamainjp 0:b5453be5ad07 581 DWORD sect, remain;
shintamainjp 0:b5453be5ad07 582 BYTE *rbuff = (BYTE *)buff;
shintamainjp 0:b5453be5ad07 583 WORD rcnt;
shintamainjp 0:b5453be5ad07 584 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 585
shintamainjp 0:b5453be5ad07 586
shintamainjp 0:b5453be5ad07 587 *br = 0;
shintamainjp 0:b5453be5ad07 588 if (!fs) return FR_NOT_ENABLED; /* Check file system */
shintamainjp 0:b5453be5ad07 589 if (!(fs->flag & FA_OPENED)) /* Check if opened */
shintamainjp 0:b5453be5ad07 590 return FR_NOT_OPENED;
shintamainjp 0:b5453be5ad07 591
shintamainjp 0:b5453be5ad07 592 remain = fs->fsize - fs->fptr;
shintamainjp 0:b5453be5ad07 593 if (btr > remain) btr = (WORD)remain; /* Truncate btr by remaining bytes */
shintamainjp 0:b5453be5ad07 594
shintamainjp 0:b5453be5ad07 595 while (btr) { /* Repeat until all data transferred */
shintamainjp 0:b5453be5ad07 596 if ((fs->fptr % 512) == 0) { /* On the sector boundary? */
shintamainjp 0:b5453be5ad07 597 if ((fs->fptr / 512 % fs->csize) == 0) { /* On the cluster boundary? */
shintamainjp 0:b5453be5ad07 598 clst = (fs->fptr == 0) ? /* On the top of the file? */
shintamainjp 0:b5453be5ad07 599 fs->org_clust : get_fat(fs->curr_clust);
shintamainjp 0:b5453be5ad07 600 if (clst <= 1) goto fr_abort;
shintamainjp 0:b5453be5ad07 601 fs->curr_clust = clst; /* Update current cluster */
shintamainjp 0:b5453be5ad07 602 fs->csect = 0; /* Reset sector offset in the cluster */
shintamainjp 0:b5453be5ad07 603 }
shintamainjp 0:b5453be5ad07 604 sect = clust2sect(fs->curr_clust); /* Get current sector */
shintamainjp 0:b5453be5ad07 605 if (!sect) goto fr_abort;
shintamainjp 0:b5453be5ad07 606 fs->dsect = sect + fs->csect++;
shintamainjp 0:b5453be5ad07 607 }
shintamainjp 0:b5453be5ad07 608 rcnt = 512 - ((WORD)fs->fptr % 512); /* Get partial sector data from sector buffer */
shintamainjp 0:b5453be5ad07 609 if (rcnt > btr) rcnt = btr;
shintamainjp 0:b5453be5ad07 610 dr = disk_readp(!buff ? 0 : rbuff, fs->dsect, (WORD)(fs->fptr % 512), rcnt);
shintamainjp 0:b5453be5ad07 611 if (dr) goto fr_abort;
shintamainjp 0:b5453be5ad07 612 fs->fptr += rcnt;
shintamainjp 0:b5453be5ad07 613 rbuff += rcnt; /* Update pointers and counters */
shintamainjp 0:b5453be5ad07 614 btr -= rcnt;
shintamainjp 0:b5453be5ad07 615 *br += rcnt;
shintamainjp 0:b5453be5ad07 616 }
shintamainjp 0:b5453be5ad07 617
shintamainjp 0:b5453be5ad07 618 return FR_OK;
shintamainjp 0:b5453be5ad07 619
shintamainjp 0:b5453be5ad07 620 fr_abort:
shintamainjp 0:b5453be5ad07 621 fs->flag = 0;
shintamainjp 0:b5453be5ad07 622 return FR_DISK_ERR;
shintamainjp 0:b5453be5ad07 623 }
shintamainjp 0:b5453be5ad07 624 #endif
shintamainjp 0:b5453be5ad07 625
shintamainjp 0:b5453be5ad07 626
shintamainjp 0:b5453be5ad07 627
shintamainjp 0:b5453be5ad07 628 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 629 /* Write File */
shintamainjp 0:b5453be5ad07 630 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 631 #if _USE_WRITE
shintamainjp 0:b5453be5ad07 632
shintamainjp 0:b5453be5ad07 633 FRESULT pf_write (
shintamainjp 0:b5453be5ad07 634 const void* buff, /* Pointer to the data to be written */
shintamainjp 0:b5453be5ad07 635 WORD btw, /* Number of bytes to write (0:Finalize the current write operation) */
shintamainjp 0:b5453be5ad07 636 WORD* bw /* Pointer to number of bytes written */
shintamainjp 0:b5453be5ad07 637 ) {
shintamainjp 0:b5453be5ad07 638 CLUST clst;
shintamainjp 0:b5453be5ad07 639 DWORD sect, remain;
shintamainjp 0:b5453be5ad07 640 const BYTE *p = (const BYTE *)buff;
shintamainjp 0:b5453be5ad07 641 WORD wcnt;
shintamainjp 0:b5453be5ad07 642 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 643
shintamainjp 0:b5453be5ad07 644
shintamainjp 0:b5453be5ad07 645 *bw = 0;
shintamainjp 0:b5453be5ad07 646 if (!fs) return FR_NOT_ENABLED; /* Check file system */
shintamainjp 0:b5453be5ad07 647 if (!(fs->flag & FA_OPENED)) /* Check if opened */
shintamainjp 0:b5453be5ad07 648 return FR_NOT_OPENED;
shintamainjp 0:b5453be5ad07 649
shintamainjp 0:b5453be5ad07 650 if (!btw) { /* Finalize request */
shintamainjp 0:b5453be5ad07 651 if ((fs->flag & FA__WIP) && disk_writep(0, 0)) goto fw_abort;
shintamainjp 0:b5453be5ad07 652 fs->flag &= ~FA__WIP;
shintamainjp 0:b5453be5ad07 653 return FR_OK;
shintamainjp 0:b5453be5ad07 654 } else { /* Write data request */
shintamainjp 0:b5453be5ad07 655 if (!(fs->flag & FA__WIP)) /* Round down fptr to the sector boundary */
shintamainjp 0:b5453be5ad07 656 fs->fptr &= 0xFFFFFE00;
shintamainjp 0:b5453be5ad07 657 }
shintamainjp 0:b5453be5ad07 658 remain = fs->fsize - fs->fptr;
shintamainjp 0:b5453be5ad07 659 if (btw > remain) btw = (WORD)remain; /* Truncate btw by remaining bytes */
shintamainjp 0:b5453be5ad07 660
shintamainjp 0:b5453be5ad07 661 while (btw) { /* Repeat until all data transferred */
shintamainjp 0:b5453be5ad07 662 if (((WORD)fs->fptr % 512) == 0) { /* On the sector boundary? */
shintamainjp 0:b5453be5ad07 663 if ((fs->fptr / 512 % fs->csize) == 0) { /* On the cluster boundary? */
shintamainjp 0:b5453be5ad07 664 clst = (fs->fptr == 0) ? /* On the top of the file? */
shintamainjp 0:b5453be5ad07 665 fs->org_clust : get_fat(fs->curr_clust);
shintamainjp 0:b5453be5ad07 666 if (clst <= 1) goto fw_abort;
shintamainjp 0:b5453be5ad07 667 fs->curr_clust = clst; /* Update current cluster */
shintamainjp 0:b5453be5ad07 668 fs->csect = 0; /* Reset sector offset in the cluster */
shintamainjp 0:b5453be5ad07 669 }
shintamainjp 0:b5453be5ad07 670 sect = clust2sect(fs->curr_clust); /* Get current sector */
shintamainjp 0:b5453be5ad07 671 if (!sect) goto fw_abort;
shintamainjp 0:b5453be5ad07 672 fs->dsect = sect + fs->csect++;
shintamainjp 0:b5453be5ad07 673 if (disk_writep(0, fs->dsect)) goto fw_abort; /* Initiate a sector write operation */
shintamainjp 0:b5453be5ad07 674 fs->flag |= FA__WIP;
shintamainjp 0:b5453be5ad07 675 }
shintamainjp 0:b5453be5ad07 676 wcnt = 512 - ((WORD)fs->fptr % 512); /* Number of bytes to write to the sector */
shintamainjp 0:b5453be5ad07 677 if (wcnt > btw) wcnt = btw;
shintamainjp 0:b5453be5ad07 678 if (disk_writep(p, wcnt)) goto fw_abort; /* Send data to the sector */
shintamainjp 0:b5453be5ad07 679 fs->fptr += wcnt;
shintamainjp 0:b5453be5ad07 680 p += wcnt; /* Update pointers and counters */
shintamainjp 0:b5453be5ad07 681 btw -= wcnt;
shintamainjp 0:b5453be5ad07 682 *bw += wcnt;
shintamainjp 0:b5453be5ad07 683 if (((WORD)fs->fptr % 512) == 0) {
shintamainjp 0:b5453be5ad07 684 if (disk_writep(0, 0)) goto fw_abort; /* Finalize the currtent secter write operation */
shintamainjp 0:b5453be5ad07 685 fs->flag &= ~FA__WIP;
shintamainjp 0:b5453be5ad07 686 }
shintamainjp 0:b5453be5ad07 687 }
shintamainjp 0:b5453be5ad07 688
shintamainjp 0:b5453be5ad07 689 return FR_OK;
shintamainjp 0:b5453be5ad07 690
shintamainjp 0:b5453be5ad07 691 fw_abort:
shintamainjp 0:b5453be5ad07 692 fs->flag = 0;
shintamainjp 0:b5453be5ad07 693 return FR_DISK_ERR;
shintamainjp 0:b5453be5ad07 694 }
shintamainjp 0:b5453be5ad07 695 #endif
shintamainjp 0:b5453be5ad07 696
shintamainjp 0:b5453be5ad07 697
shintamainjp 0:b5453be5ad07 698
shintamainjp 0:b5453be5ad07 699 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 700 /* Seek File R/W Pointer */
shintamainjp 0:b5453be5ad07 701 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 702 #if _USE_LSEEK
shintamainjp 0:b5453be5ad07 703
shintamainjp 0:b5453be5ad07 704 FRESULT pf_lseek (
shintamainjp 0:b5453be5ad07 705 DWORD ofs /* File pointer from top of file */
shintamainjp 0:b5453be5ad07 706 ) {
shintamainjp 0:b5453be5ad07 707 CLUST clst;
shintamainjp 0:b5453be5ad07 708 DWORD bcs, sect, ifptr;
shintamainjp 0:b5453be5ad07 709 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 710
shintamainjp 0:b5453be5ad07 711
shintamainjp 0:b5453be5ad07 712 if (!fs) return FR_NOT_ENABLED; /* Check file system */
shintamainjp 0:b5453be5ad07 713 if (!(fs->flag & FA_OPENED)) /* Check if opened */
shintamainjp 0:b5453be5ad07 714 return FR_NOT_OPENED;
shintamainjp 0:b5453be5ad07 715
shintamainjp 0:b5453be5ad07 716 if (ofs > fs->fsize) ofs = fs->fsize; /* Clip offset with the file size */
shintamainjp 0:b5453be5ad07 717 ifptr = fs->fptr;
shintamainjp 0:b5453be5ad07 718 fs->fptr = 0;
shintamainjp 0:b5453be5ad07 719 if (ofs > 0) {
shintamainjp 0:b5453be5ad07 720 bcs = (DWORD)fs->csize * 512; /* Cluster size (byte) */
shintamainjp 0:b5453be5ad07 721 if (ifptr > 0 &&
shintamainjp 0:b5453be5ad07 722 (ofs - 1) / bcs >= (ifptr - 1) / bcs) { /* When seek to same or following cluster, */
shintamainjp 0:b5453be5ad07 723 fs->fptr = (ifptr - 1) & ~(bcs - 1); /* start from the current cluster */
shintamainjp 0:b5453be5ad07 724 ofs -= fs->fptr;
shintamainjp 0:b5453be5ad07 725 clst = fs->curr_clust;
shintamainjp 0:b5453be5ad07 726 } else { /* When seek to back cluster, */
shintamainjp 0:b5453be5ad07 727 clst = fs->org_clust; /* start from the first cluster */
shintamainjp 0:b5453be5ad07 728 fs->curr_clust = clst;
shintamainjp 0:b5453be5ad07 729 }
shintamainjp 0:b5453be5ad07 730 while (ofs > bcs) { /* Cluster following loop */
shintamainjp 0:b5453be5ad07 731 clst = get_fat(clst); /* Follow cluster chain */
shintamainjp 0:b5453be5ad07 732 if (clst <= 1 || clst >= fs->max_clust) goto fe_abort;
shintamainjp 0:b5453be5ad07 733 fs->curr_clust = clst;
shintamainjp 0:b5453be5ad07 734 fs->fptr += bcs;
shintamainjp 0:b5453be5ad07 735 ofs -= bcs;
shintamainjp 0:b5453be5ad07 736 }
shintamainjp 0:b5453be5ad07 737 fs->fptr += ofs;
shintamainjp 0:b5453be5ad07 738 sect = clust2sect(clst); /* Current sector */
shintamainjp 0:b5453be5ad07 739 if (!sect) goto fe_abort;
shintamainjp 0:b5453be5ad07 740 fs->csect = (BYTE)(ofs / 512); /* Sector offset in the cluster */
shintamainjp 0:b5453be5ad07 741 if (ofs % 512)
shintamainjp 0:b5453be5ad07 742 fs->dsect = sect + fs->csect++;
shintamainjp 0:b5453be5ad07 743 }
shintamainjp 0:b5453be5ad07 744
shintamainjp 0:b5453be5ad07 745 return FR_OK;
shintamainjp 0:b5453be5ad07 746
shintamainjp 0:b5453be5ad07 747 fe_abort:
shintamainjp 0:b5453be5ad07 748 fs->flag = 0;
shintamainjp 0:b5453be5ad07 749 return FR_DISK_ERR;
shintamainjp 0:b5453be5ad07 750 }
shintamainjp 0:b5453be5ad07 751 #endif
shintamainjp 0:b5453be5ad07 752
shintamainjp 0:b5453be5ad07 753
shintamainjp 0:b5453be5ad07 754
shintamainjp 0:b5453be5ad07 755 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 756 /* Create a Directroy Object */
shintamainjp 0:b5453be5ad07 757 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 758 #if _USE_DIR
shintamainjp 0:b5453be5ad07 759
shintamainjp 0:b5453be5ad07 760 FRESULT pf_opendir (
shintamainjp 0:b5453be5ad07 761 FATDIR *dj, /* Pointer to directory object to create */
shintamainjp 0:b5453be5ad07 762 const char *path /* Pointer to the directory path */
shintamainjp 0:b5453be5ad07 763 ) {
shintamainjp 0:b5453be5ad07 764 FRESULT res;
shintamainjp 0:b5453be5ad07 765 BYTE sp[12], dir[32];
shintamainjp 0:b5453be5ad07 766 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 767
shintamainjp 0:b5453be5ad07 768
shintamainjp 0:b5453be5ad07 769 if (!fs) { /* Check file system */
shintamainjp 0:b5453be5ad07 770 res = FR_NOT_ENABLED;
shintamainjp 0:b5453be5ad07 771 } else {
shintamainjp 0:b5453be5ad07 772 fs->buf = dir;
shintamainjp 0:b5453be5ad07 773 dj->fn = sp;
shintamainjp 0:b5453be5ad07 774 res = follow_path(dj, path); /* Follow the path to the directory */
shintamainjp 0:b5453be5ad07 775 if (res == FR_OK) { /* Follow completed */
shintamainjp 0:b5453be5ad07 776 if (dir[0]) { /* It is not the root dir */
shintamainjp 0:b5453be5ad07 777 if (dir[DIR_Attr] & AM_DIR) { /* The object is a directory */
shintamainjp 0:b5453be5ad07 778 dj->sclust =
shintamainjp 0:b5453be5ad07 779 #if _FS_FAT32
shintamainjp 0:b5453be5ad07 780 ((DWORD)LD_WORD(dir+DIR_FstClusHI) << 16) |
shintamainjp 0:b5453be5ad07 781 #endif
shintamainjp 0:b5453be5ad07 782 LD_WORD(dir+DIR_FstClusLO);
shintamainjp 0:b5453be5ad07 783 } else { /* The object is not a directory */
shintamainjp 0:b5453be5ad07 784 res = FR_NO_PATH;
shintamainjp 0:b5453be5ad07 785 }
shintamainjp 0:b5453be5ad07 786 }
shintamainjp 0:b5453be5ad07 787 if (res == FR_OK)
shintamainjp 0:b5453be5ad07 788 res = dir_rewind(dj); /* Rewind dir */
shintamainjp 0:b5453be5ad07 789 }
shintamainjp 0:b5453be5ad07 790 if (res == FR_NO_FILE) res = FR_NO_PATH;
shintamainjp 0:b5453be5ad07 791 }
shintamainjp 0:b5453be5ad07 792
shintamainjp 0:b5453be5ad07 793 return res;
shintamainjp 0:b5453be5ad07 794 }
shintamainjp 0:b5453be5ad07 795
shintamainjp 0:b5453be5ad07 796
shintamainjp 0:b5453be5ad07 797
shintamainjp 0:b5453be5ad07 798
shintamainjp 0:b5453be5ad07 799 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 800 /* Read Directory Entry in Sequense */
shintamainjp 0:b5453be5ad07 801 /*-----------------------------------------------------------------------*/
shintamainjp 0:b5453be5ad07 802
shintamainjp 0:b5453be5ad07 803 FRESULT pf_readdir (
shintamainjp 0:b5453be5ad07 804 FATDIR *dj, /* Pointer to the open directory object */
shintamainjp 0:b5453be5ad07 805 FILINFO *fno /* Pointer to file information to return */
shintamainjp 0:b5453be5ad07 806 ) {
shintamainjp 0:b5453be5ad07 807 FRESULT res;
shintamainjp 0:b5453be5ad07 808 BYTE sp[12], dir[32];
shintamainjp 0:b5453be5ad07 809 FATFS *fs = FatFs;
shintamainjp 0:b5453be5ad07 810
shintamainjp 0:b5453be5ad07 811
shintamainjp 0:b5453be5ad07 812 if (!fs) { /* Check file system */
shintamainjp 0:b5453be5ad07 813 res = FR_NOT_ENABLED;
shintamainjp 0:b5453be5ad07 814 } else {
shintamainjp 0:b5453be5ad07 815 fs->buf = dir;
shintamainjp 0:b5453be5ad07 816 dj->fn = sp;
shintamainjp 0:b5453be5ad07 817 if (!fno) {
shintamainjp 0:b5453be5ad07 818 res = dir_rewind(dj);
shintamainjp 0:b5453be5ad07 819 } else {
shintamainjp 0:b5453be5ad07 820 res = dir_read(dj);
shintamainjp 0:b5453be5ad07 821 if (res == FR_NO_FILE) {
shintamainjp 0:b5453be5ad07 822 dj->sect = 0;
shintamainjp 0:b5453be5ad07 823 res = FR_OK;
shintamainjp 0:b5453be5ad07 824 }
shintamainjp 0:b5453be5ad07 825 if (res == FR_OK) { /* A valid entry is found */
shintamainjp 0:b5453be5ad07 826 get_fileinfo(dj, fno); /* Get the object information */
shintamainjp 0:b5453be5ad07 827 res = dir_next(dj); /* Increment index for next */
shintamainjp 0:b5453be5ad07 828 if (res == FR_NO_FILE) {
shintamainjp 0:b5453be5ad07 829 dj->sect = 0;
shintamainjp 0:b5453be5ad07 830 res = FR_OK;
shintamainjp 0:b5453be5ad07 831 }
shintamainjp 0:b5453be5ad07 832 }
shintamainjp 0:b5453be5ad07 833 }
shintamainjp 0:b5453be5ad07 834 }
shintamainjp 0:b5453be5ad07 835
shintamainjp 0:b5453be5ad07 836 return res;
shintamainjp 0:b5453be5ad07 837 }
shintamainjp 0:b5453be5ad07 838
shintamainjp 0:b5453be5ad07 839 #endif /* _USE_DIR */
shintamainjp 0:b5453be5ad07 840