EMILE/libcontainer/container_read.c
2005-12-01 22:14:27 +00:00

99 lines
2.0 KiB
C

/*
*
* (c) 2005 Laurent Vivier <LaurentVivier@wanadoo.fr>
*
*/
#include <sys/types.h>
#include <string.h>
#include "libcontainer.h"
static unsigned long seek_block(container_FILE *file)
{
struct emile_container *container = file->container;
ssize_t current;
int i;
unsigned long offset = file->offset;
int block_size = file->device.get_blocksize(file->device.data);
/* search forward */
for (i = file->last_index,
current = file->last_current;
i < container->blocks[i].count; i++)
{
int extent_size = block_size *
container->blocks[i].count;
if ( (current <= offset) && (offset < current + extent_size) )
{
file->last_current = current;
file->last_index = i;
return container->blocks[i].offset +
(offset - current) / block_size;
}
current += extent_size;
}
/* search backward */
for (i = file->last_index,
current = file->last_current;
i > 0; i--)
{
int extent_size = block_size *
container->blocks[i - 1].count;
current -= extent_size;
if ( (current <= offset) && (offset < current + extent_size) )
{
file->last_current = current;
file->last_index = i - 1;
return container->blocks[i].offset +
(offset - current) / block_size;
}
}
return 0;
}
ssize_t container_read(container_FILE *file, void *ptr, size_t size)
{
int err;
ssize_t read = 0;
int part;
int block_size = file->device.get_blocksize(file->device.data);
while (size != 0)
{
unsigned long block_nb = seek_block(file);
int block_offset = file->offset % block_size;
if (block_nb != file->current_block)
{
err = file->device.read_sector(
file->device.data,
block_nb,
file->buffer,
block_size);
if (err == -1)
return -1;
file->current_block = block_nb;
}
part = block_size - block_offset;
if (part > size)
part = size;
memcpy(ptr, file->buffer + block_offset, part);
size -= part;
ptr = (char*)ptr + part;
file->offset += part;
read += part;
}
return read;
}