testdisk/src/hdcache.c
2021-11-20 11:26:50 +01:00

341 lines
11 KiB
C

/*
File: hdcache.c
Copyright (C) 2005-2008 Christophe GRENIER <grenier@cgsecurity.org>
This software 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; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License along
with this program; if not, write the Free Software Foundation, Inc., 51
Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#ifdef HAVE_STDLIB_H
#include <stdlib.h>
#endif
#ifdef HAVE_STRING_H
#include <string.h>
#endif
#include "types.h"
#include "common.h"
#include "hdcache.h"
#include "log.h"
#define CACHE_BUFFER_NBR 16
#define CACHE_DEFAULT_SIZE 64*512
//#define DEBUG_CACHE 1
struct cache_buffer_struct
{
unsigned char *buffer;
unsigned int buffer_size;
unsigned int cache_size;
uint64_t cache_offset;
int cache_status;
};
struct cache_struct
{
disk_t *disk_car;
struct cache_buffer_struct cache[CACHE_BUFFER_NBR];
#ifdef DEBUG_CACHE
uint64_t nbr_fnct_sect;
uint64_t nbr_pread_sect;
unsigned int nbr_fnct_call;
unsigned int nbr_pread_call;
#endif
unsigned int cache_buffer_nbr;
unsigned int cache_size_min;
unsigned int last_io_error_nbr;
};
/*@
@ requires \valid(disk_car);
@ requires valid_disk(disk_car);
@ requires \valid((char *)buffer + (0 .. count-1));
@ requires separation: \separated(disk_car, (char *)buffer + (0 .. count-1));
@*/
static int cache_pread_aux(disk_t *disk_car, void *buffer, const unsigned int count, const uint64_t offset, const unsigned int read_ahead)
{
struct cache_struct *data=(struct cache_struct *)disk_car->data;
#ifdef DEBUG_CACHE
log_info("cache_pread(buffer, count=%u, offset=%llu, read_ahead=%u)\n", count,(long long unsigned)offset, read_ahead);
data->nbr_fnct_call++;
#endif
{
unsigned int i;
unsigned int cache_buffer_nbr;
/* Data is probably in the last buffers */
for(i=0, cache_buffer_nbr=data->cache_buffer_nbr;
i<CACHE_BUFFER_NBR;
i++, cache_buffer_nbr=(cache_buffer_nbr+CACHE_BUFFER_NBR-1)%CACHE_BUFFER_NBR)
{
const struct cache_buffer_struct *cache=&data->cache[cache_buffer_nbr];
if(cache->cache_offset <= offset &&
offset < cache->cache_offset +cache->cache_size &&
cache->buffer!=NULL && cache->cache_size>0)
{
const unsigned int data_available= cache->cache_size + cache->cache_offset - offset;
const int res=cache->cache_status + cache->cache_offset - offset;
if(count<=data_available)
{
#ifdef DEBUG_CACHE
log_info("cache use %5u count=%u, coffset=%llu, cstatus=%d\n",
cache_buffer_nbr, cache->cache_size, (long long unsigned)cache->cache_offset,
cache->cache_status);
data->nbr_fnct_sect+=count;
#endif
memcpy(buffer, cache->buffer + offset - cache->cache_offset, count);
return (res < (signed)count ? res : (signed)count );
}
else
{
#ifdef DEBUG_CACHE
log_info("cache USE %5u count=%u, coffset=%llu, ctstatus=%d, call again cache_pread_aux\n",
cache_buffer_nbr, cache->cache_size, (long long unsigned)cache->cache_offset,
cache->cache_status);
data->nbr_fnct_sect+=data_available;
#endif
memcpy(buffer, cache->buffer + offset - cache->cache_offset, data_available);
return res + cache_pread_aux(disk_car, (unsigned char*)buffer+data_available,
count-data_available, offset+data_available, read_ahead);
}
}
}
}
{
struct cache_buffer_struct *cache;
const unsigned int count_new=(read_ahead!=0 && count<data->cache_size_min && (offset+data->cache_size_min<data->disk_car->disk_real_size)?data->cache_size_min:count);
data->cache_buffer_nbr=(data->cache_buffer_nbr+1)%CACHE_BUFFER_NBR;
cache=&data->cache[data->cache_buffer_nbr];
if(cache->buffer_size < count_new)
{ /* Buffer is too small, drop it */
free(cache->buffer);
cache->buffer=NULL;
}
if(cache->buffer==NULL)
{ /* Allocate buffer */
cache->buffer_size=(count_new<CACHE_DEFAULT_SIZE?CACHE_DEFAULT_SIZE:count_new);
cache->buffer=(unsigned char *)MALLOC(cache->buffer_size);
}
cache->cache_size=count_new;
cache->cache_offset=offset;
cache->cache_status=data->disk_car->pread(data->disk_car, cache->buffer, count_new, offset);
#ifdef DEBUG_CACHE
data->nbr_fnct_sect+=count;
data->nbr_pread_call++;
data->nbr_pread_sect+=count_new;
log_info("cache PREAD(buffer[%u], count=%u, count_new=%u, offset=%llu, cstatus=%d)\n",
data->cache_buffer_nbr, count, count_new, (long long unsigned)offset,
cache->cache_status);
#endif
if(cache->cache_status >= (signed)count)
{
data->last_io_error_nbr=0;
memcpy(buffer, cache->buffer, count);
return count;
}
/* Read failure */
data->last_io_error_nbr++;
if(count_new<=disk_car->sector_size || disk_car->sector_size<=0 || data->last_io_error_nbr>1)
{
memcpy(buffer, cache->buffer, count);
return cache->cache_status;
}
/* Free the existing cache */
cache->cache_size=0;
/* split the read sector by sector */
{
unsigned int off;
memset(buffer, 0, count);
for(off=0; off<count; off+=disk_car->sector_size)
{
if(cache_pread_aux(disk_car,
(unsigned char*)buffer+off,
(disk_car->sector_size < count - off ? disk_car->sector_size : count - off),
offset+off, 0) <= 0)
{
return off;
}
}
return count;
}
}
}
/*@
@ requires \valid(disk_car);
@ requires valid_disk(disk_car);
@ requires \valid((char *)buffer + (0 .. count-1));
@ requires separation: \separated(disk_car, (char *)buffer + (0 .. count-1));
@*/
static int cache_pread(disk_t *disk_car, void *buffer, const unsigned int count, const uint64_t offset)
{
const struct cache_struct *data=(const struct cache_struct *)disk_car->data;
return cache_pread_aux(disk_car, buffer, count, offset, (data->last_io_error_nbr==0));
}
/*@
@ requires \valid(disk_car);
@ requires valid_disk(disk_car);
@ requires \valid_read((char *)buffer + (0 .. count-1));
@ requires separation: \separated(disk_car, (const char *)buffer + (0 .. count-1));
@*/
static int cache_pwrite(disk_t *disk_car, const void *buffer, const unsigned int count, const uint64_t offset)
{
struct cache_struct *data=(struct cache_struct *)disk_car->data;
unsigned int i;
for(i=0;i<CACHE_BUFFER_NBR;i++)
{
struct cache_buffer_struct *cache=&data->cache[i];
if(!(cache->cache_offset+cache->cache_size-1 < offset || offset+count-1 < cache->cache_offset))
{
/* Discard the cache */
cache->cache_size=0;
}
}
disk_car->write_used=1;
return data->disk_car->pwrite(data->disk_car, buffer, count, offset);
}
/*@
@ requires \valid(disk_car);
@ requires valid_disk(disk_car);
@*/
static void cache_clean(disk_t *disk_car)
{
if(disk_car->data)
{
struct cache_struct *data=(struct cache_struct *)disk_car->data;
unsigned int i;
#ifdef DEBUG_CACHE
log_info("%s\ncache_pread total_call=%u, total_count=%llu\n read total_call=%u, total_count=%llu\n",
data->disk_car->description(data->disk_car),
data->nbr_fnct_call, (long long unsigned)data->nbr_fnct_sect,
data->nbr_pread_call, (long long unsigned)data->nbr_pread_sect);
#endif
data->disk_car->clean(data->disk_car);
for(i=0;i<CACHE_BUFFER_NBR;i++)
{
struct cache_buffer_struct *cache=&data->cache[i];
free(cache->buffer);
}
free(disk_car->data);
disk_car->data=NULL;
}
free(disk_car);
}
/*@
@ requires \valid(disk_car);
@*/
static int cache_sync(disk_t *disk_car)
{
struct cache_struct *data=(struct cache_struct *)disk_car->data;
return data->disk_car->sync(data->disk_car);
}
/*@
@ requires \valid_read(CHS_source);
@ requires \valid(CHS_dst);
@ requires separation: \separated(CHS_dst, CHS_source);
@ assigns CHS_dst->cylinders, CHS_dst->heads_per_cylinder, CHS_dst->sectors_per_head;
@ ensures CHS_dst->cylinders==CHS_source->cylinders;
@ ensures CHS_dst->heads_per_cylinder==CHS_source->heads_per_cylinder;
@ ensures CHS_dst->sectors_per_head==CHS_source->sectors_per_head;
@*/
static void dup_geometry(CHSgeometry_t * CHS_dst, const CHSgeometry_t * CHS_source)
{
CHS_dst->cylinders=CHS_source->cylinders;
CHS_dst->heads_per_cylinder=CHS_source->heads_per_cylinder;
CHS_dst->sectors_per_head=CHS_source->sectors_per_head;
}
/*@
@ requires \valid(disk_car);
@ requires valid_disk(disk_car);
@ ensures valid_read_string(\result);
@*/
static const char *cache_description(disk_t *disk_car)
{
const char *tmp;
struct cache_struct *data=(struct cache_struct *)disk_car->data;
dup_geometry(&data->disk_car->geom,&disk_car->geom);
data->disk_car->disk_size=disk_car->disk_size;
tmp=data->disk_car->description(data->disk_car);
/*@ assert valid_read_string(tmp); */
return tmp;
}
/*@
@ requires \valid(disk_car);
@ requires valid_disk(disk_car);
@ ensures valid_read_string(\result);
@*/
static const char *cache_description_short(disk_t *disk_car)
{
const char *tmp;
struct cache_struct *data=(struct cache_struct *)disk_car->data;
dup_geometry(&data->disk_car->geom,&disk_car->geom);
data->disk_car->disk_size=disk_car->disk_size;
tmp=data->disk_car->description_short(data->disk_car);
/*@ assert valid_read_string(tmp); */
return tmp;
}
disk_t *new_diskcache(disk_t *disk_car, const unsigned int testdisk_mode)
{
unsigned int i;
struct cache_struct*data=(struct cache_struct*)MALLOC(sizeof(*data));
disk_t *new_disk_car=(disk_t *)MALLOC(sizeof(*new_disk_car));
memcpy(new_disk_car,disk_car,sizeof(*new_disk_car));
data->disk_car=disk_car;
#ifdef DEBUG_CACHE
data->nbr_fnct_sect=0;
data->nbr_pread_sect=0;
data->nbr_fnct_call=0;
data->nbr_pread_call=0;
#endif
data->cache_buffer_nbr=0;
data->last_io_error_nbr=0;
if(testdisk_mode&TESTDISK_O_READAHEAD_8K)
data->cache_size_min=16*512;
else if(testdisk_mode&TESTDISK_O_READAHEAD_32K)
data->cache_size_min=64*512;
else
data->cache_size_min=0;
dup_geometry(&new_disk_car->geom,&disk_car->geom);
new_disk_car->disk_size=disk_car->disk_size;
new_disk_car->disk_real_size=disk_car->disk_real_size;
new_disk_car->write_used=0;
new_disk_car->data=data;
new_disk_car->pread=&cache_pread;
new_disk_car->pwrite=&cache_pwrite;
new_disk_car->sync=&cache_sync;
new_disk_car->clean=&cache_clean;
new_disk_car->description=&cache_description;
new_disk_car->description_short=&cache_description_short;
new_disk_car->rbuffer=NULL;
new_disk_car->wbuffer=NULL;
new_disk_car->rbuffer_size=0;
new_disk_car->wbuffer_size=0;
for(i=0;i<CACHE_BUFFER_NBR;i++)
{
data->cache[i].buffer=NULL;
data->cache[i].buffer_size=0;
}
return new_disk_car;
}