I64 ClusNumNext(CDrive *drive, I64 c, I64 count=1) {//Return next count'th cluster in chain. Bool unlock; DriveCheck(drive); if (count <= 0) return c; try { unlock = DriveLock(drive); switch (drive->fs_type) { case FSt_REDSEA: c += count; break; case FSt_FAT32: while (count-- > 0 && 0 < c < 0x0FFFFFF8) { DriveFATBlkSet(drive, c); c = drive->cur_fat_blk[c & (BLK_SIZE / 4 - 1)]; } if (!(0 < c < 0x0FFFFFF8)) c = INVALID_CLUS; break; default: throw('Drive'); } if (unlock) DriveUnlock(drive); } catch if (unlock) DriveUnlock(drive); return c; } I64 Clus2Blk(CDrive *drive, I64 c) {//Drive cluster num to block num. DriveCheck(drive); switch (drive->fs_type) { case FSt_REDSEA: return c; case FSt_FAT32: return drive->data_area + c * drive->spc; default: throw('Drive'); } } I64 ClusBlkRead(CDrive *drive, U8 *buf, I64 c, I64 blks) {//Accepts block count, so padding on last cluster is not read. I64 i; Bool unlock; DriveCheck(drive); if (blks <= 0) return c; try { unlock = DriveLock(drive); switch (drive->fs_type) { case FSt_REDSEA: BlkRead(drive, buf, c, blks); c += blks; break; case FSt_FAT32: while (blks && 0 < c < 0x0FFFFFF8) { i = blks; if (i > drive->spc) i = drive->spc; BlkRead(drive, buf, drive->data_area + c * drive->spc, i); buf+=i << BLK_SIZE_BITS; c = ClusNumNext(drive, c, 1); blks -= i; } if (blks) throw('Drive'); break; default: throw('Drive'); } if (unlock) DriveUnlock(drive); } catch if (unlock) DriveUnlock(drive); return c; } I64 ClusRead(CDrive *drive, U8 *buf, I64 c, I64 count) {//Read cluster count from drive to buf. return ClusBlkRead(drive, buf, c, count*drive->spc); } I64 ClusBlkWrite(CDrive *drive, U8 *buf, I64 c, I64 blks) {//Accepts block count, so padding on last cluster is not written. I64 i; Bool unlock; DriveCheck(drive); if (blks <= 0) return c; try { unlock = DriveLock(drive); switch (drive->fs_type) { case FSt_REDSEA: BlkWrite(drive, buf, c, blks); c = 0; break; case FSt_FAT32: while (blks) { if (!(0 < c < 0x0FFFFFF8)) throw('Drive'); i = blks; if (i > drive->spc) i = drive->spc; BlkWrite(drive, buf, drive->data_area + c * drive->spc, i); buf += i << BLK_SIZE_BITS; c = ClusNumNext(drive, c); blks -= i; } break; default: throw('Drive'); } if (unlock) DriveUnlock(drive); } catch if (unlock) DriveUnlock(drive); return c; } I64 ClusWrite(CDrive *drive, U8 *buf, I64 c, I64 count) {//Write cluster count from buf to drive. return ClusBlkWrite(drive, buf, c, count * drive->spc); } I64 ClusAlloc(CDrive *drive, I64 c=0, I64 count=1, Bool contiguous=FALSE) {//Alloc cluster count into chain. //c=0 means first cluster in chain DriveCheck(drive); if (count <= 0) return c; switch (drive->fs_type) { case FSt_REDSEA: return RedSeaAllocClus(drive, count); case FSt_FAT32: if (contiguous) { if (c) throw('File'); return FAT32AllocContiguousClus(drive, count); } else return FAT32AllocClus(drive, c, count); default: throw('Drive'); } }