/* ----------------------------------------------------------------------- * * * 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. * * ----------------------------------------------------------------------- */ /* * cache.c * * Simple sector cache */ #include #include #include "libfatint.h" /* * NB: We need to align our sector buffers to at least the 8-byte mark, as some Windows * disk devices, notably O2Micro PCI-E SD card readers, return ERROR_INVALID_PARAMETER * when attempting to use ReadFile() against a non 8-byte aligned buffer. * For good measure, we'll go further and align our buffers on a 16-byte boundary. * Also, since struct libfat_sector's data[0] is our buffer, this means we must BOTH * align that member in the struct declaration, and use aligned malloc/free. */ extern void _uprintf(const char *format, ...); void *libfat_get_sector(struct libfat_filesystem *fs, libfat_sector_t n) { struct libfat_sector *ls; for (ls = fs->sectors; ls; ls = ls->next) { if (ls->n == n) return ls->data; /* Found in cache */ } /* Not found in cache */ ls = _mm_malloc(sizeof(struct libfat_sector) + LIBFAT_SECTOR_SIZE, 16); if (!ls) { libfat_flush(fs); ls = _mm_malloc(sizeof(struct libfat_sector) + LIBFAT_SECTOR_SIZE, 16); if (!ls) return NULL; /* Can't allocate memory */ } if (fs->read(fs->readptr, ls->data, LIBFAT_SECTOR_SIZE, n) != LIBFAT_SECTOR_SIZE) { _mm_free(ls); return NULL; /* I/O error */ } ls->n = n; ls->next = fs->sectors; fs->sectors = ls; return ls->data; } void libfat_flush(struct libfat_filesystem *fs) { struct libfat_sector *ls, *lsnext; lsnext = fs->sectors; fs->sectors = NULL; for (ls = lsnext; ls; ls = lsnext) { lsnext = ls->next; _mm_free(ls); } }