rufus/src/syslinux/libfat/cache.c

76 lines
2.1 KiB
C

/* ----------------------------------------------------------------------- *
*
* 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 <stdlib.h>
#include <malloc.h>
#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);
}
}