EMILE/libstream/stream_open.c

266 lines
5.2 KiB
C
Raw Normal View History

2005-11-21 23:55:56 +00:00
/*
*
* (c) 2005 Laurent Vivier <Laurent@lvivier.info>
2005-11-21 23:55:56 +00:00
*
*/
2005-11-22 23:11:42 +00:00
#include <stdio.h>
2005-11-21 23:55:56 +00:00
#include <string.h>
#include <stdlib.h>
#include "libstream.h"
2005-11-26 08:53:02 +00:00
#ifdef FLOPPY_SUPPORT
2005-11-21 23:55:56 +00:00
#include <libfloppy.h>
2005-11-26 08:53:02 +00:00
#endif
#ifdef SCSI_SUPPORT
2005-11-21 23:55:56 +00:00
#include <libscsi.h>
2005-11-26 08:53:02 +00:00
#endif
#ifdef BLOCK_SUPPORT
2005-11-21 23:55:56 +00:00
#include <libblock.h>
2005-11-26 08:53:02 +00:00
#endif
#ifdef CONTAINER_SUPPORT
2005-11-21 23:55:56 +00:00
#include <libcontainer.h>
2005-11-26 08:53:02 +00:00
#endif
#ifdef ISO9660_SUPPORT
2005-11-21 23:55:56 +00:00
#include <libiso9660.h>
2005-11-26 08:53:02 +00:00
#endif
2007-10-10 21:13:35 +00:00
#ifdef MAP_SUPPORT
#include <libmap.h>
#endif
2005-11-21 23:55:56 +00:00
typedef enum {
device_FLOPPY,
device_SCSI,
} device_t;
typedef enum {
fs_BLOCK,
fs_CONTAINER,
fs_ISO9660,
} fs_t;
static char* get_fs(char *path, fs_t *fs)
{
if (strncmp("block:", path, 6) == 0)
{
*fs = fs_BLOCK;
return path + 6;
2005-11-26 08:53:02 +00:00
}
if (strncmp("container:", path, 10) == 0)
2005-11-21 23:55:56 +00:00
{
*fs = fs_CONTAINER;
return path + 10;
2005-11-26 08:53:02 +00:00
}
if (strncmp("iso9660:", path, 8) == 0)
2005-11-21 23:55:56 +00:00
{
*fs = fs_ISO9660;
return path + 8;
}
return NULL;
}
static char *get_device(char* path,
device_t *device, int *unit, int* partition)
{
int nb;
if (*path != '(')
return NULL;
path++;
if (strncmp("fd", path, 2) == 0) {
*device = device_FLOPPY;
path += 2;
} else if (strncmp("sd", path, 2) == 0) {
*device = device_SCSI;
path += 2;
} else
return NULL;
nb = 0;
while ( (*path >= '0') && (*path <= '9') ) {
nb = (nb * 10) + (*path - '0');
path++;
}
*unit = nb;
*partition = -1;
if ( (*path == 0) || (*path == ')') )
{
path++;
return path;
}
if (*path != ',')
return NULL;
path++;
nb = 0;
while ( (*path >= '0') && (*path <= '9') ) {
nb = (nb * 10) + (*path - '0');
path++;
}
*partition = nb;
if ( (*path == 0) || (*path == ')') )
{
path++;
return path;
}
return NULL;
}
stream_t *stream_open(char *dev)
{
stream_t *stream;
fs_t fs;
device_t device;
int unit, partition;
char *current;
current = get_fs(dev, &fs);
if (current == NULL)
2005-11-22 23:11:42 +00:00
{
printf("Cannot identify given filesystem\n");
2005-11-21 23:55:56 +00:00
return NULL;
2005-11-22 23:11:42 +00:00
}
2005-11-21 23:55:56 +00:00
current = get_device(current, &device, &unit, &partition);
if (current == NULL)
2005-11-22 23:11:42 +00:00
{
printf("Cannot identify given device\n");
2005-11-21 23:55:56 +00:00
return NULL;
2005-11-22 23:11:42 +00:00
}
2005-11-21 23:55:56 +00:00
stream = (stream_t*)malloc(sizeof(stream_t));
switch(device)
{
2005-11-26 08:53:02 +00:00
#ifdef FLOPPY_SUPPORT
2005-11-21 23:55:56 +00:00
case device_FLOPPY:
if (partition != -1)
{
free(stream);
return NULL;
}
stream->device.data = floppy_open(unit);
if (stream->device.data == NULL)
{
free(stream);
return NULL;
}
2005-11-26 08:53:02 +00:00
stream->device.read_sector =
(stream_read_sector_t)floppy_read_sector;
2005-11-21 23:55:56 +00:00
stream->device.close = (stream_close_t)floppy_close;
2005-11-26 08:53:02 +00:00
stream->device.get_blocksize =
(stream_get_blocksize_t)floppy_get_blocksize;
2005-11-21 23:55:56 +00:00
break;
2005-11-26 08:53:02 +00:00
#endif /* FLOPPY_SUPPORT */
2005-11-21 23:55:56 +00:00
2005-11-26 08:53:02 +00:00
#ifdef SCSI_SUPPORT
2005-11-21 23:55:56 +00:00
case device_SCSI:
stream->device.data = scsi_open(unit);
if (stream->device.data == NULL)
{
free(stream);
return NULL;
}
2005-11-26 08:53:02 +00:00
stream->device.read_sector =
(stream_read_sector_t)scsi_read_sector;
2005-11-21 23:55:56 +00:00
stream->device.close = (stream_close_t)scsi_close;
2005-11-26 08:53:02 +00:00
stream->device.get_blocksize =
(stream_get_blocksize_t)scsi_get_blocksize;
2005-11-21 23:55:56 +00:00
break;
2005-11-26 08:53:02 +00:00
#endif /* SCSI_SUPPORT */
2005-11-21 23:55:56 +00:00
default:
free(stream);
2005-11-26 08:53:02 +00:00
return NULL;
2005-11-21 23:55:56 +00:00
break;
}
2007-10-10 21:13:35 +00:00
if (partition != -1)
{
#ifdef MAP_SUPPORT
int ret;
map_t *map;
map = map_open(&stream->device);
if (map == NULL)
goto map_error;
stream->device.data = map;
ret = map_read(map, partition);
if (ret == -1)
goto map_read_error;
stream->device.read_sector = (stream_read_sector_t)map_read_sector;
stream->device.close = (stream_close_t)map_close;
map_read_error:
map_close(map);
map_error:
#endif /* MAP_SUPPORT */
stream->device.close(&stream->device);
free(stream);
return NULL;
}
2005-11-21 23:55:56 +00:00
switch(fs)
{
2005-11-26 08:53:02 +00:00
#ifdef BLOCK_SUPPORT
2005-11-21 23:55:56 +00:00
case fs_BLOCK:
2005-11-23 22:37:27 +00:00
stream->fs.volume = NULL;
stream->fs.file = block_open(&stream->device, current);
if (stream->fs.file == NULL)
2005-11-21 23:55:56 +00:00
goto outfs;
2005-11-23 00:09:00 +00:00
stream->fs.read = (stream_read_t)block_read;
2005-11-23 22:37:27 +00:00
stream->fs.lseek = (stream_lseek_t)block_lseek;
2005-11-23 00:09:00 +00:00
stream->fs.close = (stream_close_t)block_close;
2005-11-23 22:37:27 +00:00
stream->fs.umount = NULL;
2005-11-23 00:09:00 +00:00
stream->fs.fstat = (stream_fstat_t)block_fstat;
2005-11-21 23:55:56 +00:00
break;
2005-11-26 08:53:02 +00:00
#endif /* BLOCK_SUPPORT */
#ifdef CONTAINER_SUPPORT
2005-11-21 23:55:56 +00:00
case fs_CONTAINER:
2005-11-23 22:37:27 +00:00
stream->fs.volume = NULL;
stream->fs.file = container_open(&stream->device, current);
2005-12-01 22:17:09 +00:00
if (stream->fs.file == NULL)
2005-11-21 23:55:56 +00:00
goto outfs;
2005-11-23 00:09:00 +00:00
stream->fs.read = (stream_read_t)container_read;
2005-12-01 22:17:09 +00:00
stream->fs.lseek = (stream_lseek_t)container_lseek;
2005-11-23 00:09:00 +00:00
stream->fs.close = (stream_close_t)container_close;
2005-12-01 22:17:09 +00:00
stream->fs.umount = NULL;
2005-11-23 00:09:00 +00:00
stream->fs.fstat = (stream_fstat_t)container_fstat;
2005-11-21 23:55:56 +00:00
break;
2005-11-26 08:53:02 +00:00
#endif /* CONTAINER_SUPPORT */
#ifdef ISO9660_SUPPORT
2005-11-21 23:55:56 +00:00
case fs_ISO9660:
stream->fs.volume = iso9660_mount(&stream->device);
if (stream->fs.volume == NULL)
2005-11-22 23:11:42 +00:00
{
printf("Cannot mount volume ISO9660\n");
2005-11-21 23:55:56 +00:00
goto outfs;
2005-11-22 23:11:42 +00:00
}
2005-11-21 23:55:56 +00:00
stream->fs.file = iso9660_open(stream->fs.volume, current);
if (stream->fs.file == NULL)
{
iso9660_umount(stream->fs.volume);
goto outfs;
}
stream->fs.read = (stream_read_t)iso9660_read;
stream->fs.lseek = (stream_lseek_t)iso9660_lseek;
stream->fs.close = (stream_close_t)iso9660_close;
stream->fs.umount = (stream_umount_t)iso9660_umount;
2005-11-23 00:09:00 +00:00
stream->fs.fstat = (stream_fstat_t)iso9660_fstat;
2005-11-21 23:55:56 +00:00
break;
2005-11-26 08:53:02 +00:00
#endif /* ISO9660_SUPPORT */
2005-11-21 23:55:56 +00:00
default:
outfs:
stream->device.close(stream->device.data);
free(stream);
return NULL;
}
return stream;
}