/* ----------------------------------------------------------------------- * * * Copyright 2004-2008 H. Peter Anvin - All Rights Reserved * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, Inc., 53 Temple Place Ste 330, * Boston MA 02111-1307, USA; either version 2 of the License, or * (at your option) any later version; incorporated herein by reference. * * ----------------------------------------------------------------------- */ /* * open.c * * Open a FAT filesystem and compute some initial values; return NULL * on failure. */ #include #include "libfatint.h" #include "ulint.h" struct libfat_filesystem * libfat_open(int (*readfunc) (intptr_t, void *, size_t, libfat_sector_t), intptr_t readptr) { struct libfat_filesystem *fs = NULL; struct fat_bootsect *bs; int i; uint32_t sectors, fatsize, minfatsize, rootdirsize; uint32_t nclusters; fs = malloc(sizeof(struct libfat_filesystem)); if (!fs) goto barf; fs->sectors = NULL; fs->read = readfunc; fs->readptr = readptr; bs = libfat_get_sector(fs, 0); if (!bs) goto barf; if (read16(&bs->bsBytesPerSec) != LIBFAT_SECTOR_SIZE) goto barf; for (i = 0; i <= 8; i++) { if ((uint8_t) (1 << i) == read8(&bs->bsSecPerClust)) break; } if (i > 8) goto barf; fs->clustsize = 1 << i; /* Treat 0 as 2^8 = 64K */ fs->clustshift = i; sectors = read16(&bs->bsSectors); if (!sectors) sectors = read32(&bs->bsHugeSectors); fs->end = sectors; fs->fat = read16(&bs->bsResSectors); fatsize = read16(&bs->bsFATsecs); if (!fatsize) fatsize = read32(&bs->u.fat32.bpb_fatsz32); fs->rootdir = fs->fat + fatsize * read8(&bs->bsFATs); rootdirsize = ((read16(&bs->bsRootDirEnts) << 5) + LIBFAT_SECTOR_MASK) >> LIBFAT_SECTOR_SHIFT; fs->data = fs->rootdir + rootdirsize; /* Sanity checking */ if (fs->data >= fs->end) goto barf; /* Figure out how many clusters */ nclusters = (fs->end - fs->data) >> fs->clustshift; fs->endcluster = nclusters + 2; if (nclusters <= 0xff4) { fs->fat_type = FAT12; minfatsize = fs->endcluster + (fs->endcluster >> 1); } else if (nclusters <= 0xfff4) { fs->fat_type = FAT16; minfatsize = fs->endcluster << 1; } else if (nclusters <= 0xffffff4) { fs->fat_type = FAT28; minfatsize = fs->endcluster << 2; } else goto barf; /* Impossibly many clusters */ minfatsize = (minfatsize + LIBFAT_SECTOR_SIZE - 1) >> LIBFAT_SECTOR_SHIFT; if (minfatsize > fatsize) goto barf; /* The FATs don't fit */ if (fs->fat_type == FAT28) fs->rootcluster = read32(&bs->u.fat32.bpb_rootclus); else fs->rootcluster = 0; return fs; /* All good */ barf: if (fs) free(fs); return NULL; } void libfat_close(struct libfat_filesystem *fs) { libfat_flush(fs); free(fs); }