2010-12-23 19:17:12 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2011-02-25 01:05:47 +03:00
|
|
|
// $Id$
|
2010-12-23 19:17:12 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2012-08-31 16:08:19 +04:00
|
|
|
// Virtual VFAT image support (shadows a local directory)
|
|
|
|
// ported from QEMU block driver with some additions (see vvfat.cc)
|
2010-12-23 19:17:12 +03:00
|
|
|
//
|
2012-08-31 16:08:19 +04:00
|
|
|
// Copyright (c) 2004,2005 Johannes E. Schindelin
|
|
|
|
// Copyright (C) 2010-2012 The Bochs Project
|
2010-12-23 19:17:12 +03:00
|
|
|
//
|
2012-08-31 16:08:19 +04:00
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
2010-12-23 19:17:12 +03:00
|
|
|
//
|
2012-08-31 16:08:19 +04:00
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE.
|
2010-12-23 19:17:12 +03:00
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#ifndef BX_VVFAT_H
|
|
|
|
#define BX_VVFAT_H
|
|
|
|
|
|
|
|
typedef struct array_t {
|
|
|
|
char *pointer;
|
|
|
|
unsigned int size, next, item_size;
|
|
|
|
} array_t;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
Bit8u head;
|
|
|
|
Bit8u sector;
|
|
|
|
Bit8u cylinder;
|
|
|
|
} mbr_chs_t;
|
|
|
|
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER<1300)
|
|
|
|
#pragma pack(push, 1)
|
|
|
|
#elif defined(__MWERKS__) && defined(macintosh)
|
|
|
|
#pragma options align=packed
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER>=1300)
|
|
|
|
__declspec(align(1))
|
|
|
|
#endif
|
|
|
|
struct direntry_t {
|
|
|
|
Bit8u name[8];
|
|
|
|
Bit8u extension[3];
|
|
|
|
Bit8u attributes;
|
|
|
|
Bit8u reserved[2];
|
|
|
|
Bit16u ctime;
|
|
|
|
Bit16u cdate;
|
|
|
|
Bit16u adate;
|
|
|
|
Bit16u begin_hi;
|
|
|
|
Bit16u mtime;
|
|
|
|
Bit16u mdate;
|
|
|
|
Bit16u begin;
|
|
|
|
Bit32u size;
|
|
|
|
}
|
|
|
|
#if !defined(_MSC_VER)
|
|
|
|
GCC_ATTRIBUTE((packed))
|
|
|
|
#endif
|
|
|
|
direntry_t;
|
|
|
|
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER<1300)
|
|
|
|
#pragma pack(pop)
|
|
|
|
#elif defined(__MWERKS__) && defined(macintosh)
|
|
|
|
#pragma options align=reset
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// this structure are used to transparently access the files
|
|
|
|
|
|
|
|
enum {
|
|
|
|
MODE_UNDEFINED = 0, MODE_NORMAL = 1, MODE_MODIFIED = 2,
|
|
|
|
MODE_DIRECTORY = 4, MODE_FAKED = 8,
|
|
|
|
MODE_DELETED = 16, MODE_RENAMED = 32
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct mapping_t {
|
|
|
|
// begin is the first cluster, end is the last+1
|
|
|
|
Bit32u begin, end;
|
2011-01-09 03:36:42 +03:00
|
|
|
// as this->directory is growable, no pointer may be used here
|
2010-12-23 19:17:12 +03:00
|
|
|
unsigned int dir_index;
|
|
|
|
// the clusters of a file may be in any order; this points to the first
|
|
|
|
int first_mapping_index;
|
|
|
|
union {
|
|
|
|
/* offset is
|
|
|
|
* - the offset in the file (in clusters) for a file, or
|
|
|
|
* - the next cluster of the directory for a directory, and
|
|
|
|
* - the address of the buffer for a faked entry
|
|
|
|
*/
|
|
|
|
struct {
|
|
|
|
Bit32u offset;
|
|
|
|
} file;
|
|
|
|
struct {
|
|
|
|
int parent_mapping_index;
|
|
|
|
int first_dir_index;
|
|
|
|
} dir;
|
|
|
|
} info;
|
2011-01-09 03:36:42 +03:00
|
|
|
// path contains the full path, i.e. it always starts with vvfat_path
|
2010-12-23 19:17:12 +03:00
|
|
|
char *path;
|
|
|
|
|
|
|
|
Bit8u mode;
|
|
|
|
|
|
|
|
int read_only;
|
|
|
|
} mapping_t;
|
|
|
|
|
|
|
|
class vvfat_image_t : public device_image_t
|
|
|
|
{
|
|
|
|
public:
|
2011-01-11 23:14:21 +03:00
|
|
|
vvfat_image_t(Bit64u size, const char* redolog_name);
|
2010-12-23 19:17:12 +03:00
|
|
|
virtual ~vvfat_image_t();
|
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
int open(const char* pathname, int flags);
|
2010-12-23 19:17:12 +03:00
|
|
|
void close();
|
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
|
|
|
ssize_t read(void* buf, size_t count);
|
|
|
|
ssize_t write(const void* buf, size_t count);
|
|
|
|
Bit32u get_capabilities();
|
|
|
|
|
|
|
|
private:
|
|
|
|
bx_bool sector2CHS(Bit32u spos, mbr_chs_t *chs);
|
|
|
|
void init_mbr();
|
|
|
|
direntry_t* create_long_filename(const char* filename);
|
|
|
|
void fat_set(unsigned int cluster, Bit32u value);
|
|
|
|
void init_fat();
|
|
|
|
direntry_t* create_short_and_long_name(unsigned int directory_start,
|
|
|
|
const char* filename, int is_dot);
|
|
|
|
int read_directory(int mapping_index);
|
|
|
|
Bit32u sector2cluster(off_t sector_num);
|
2011-01-09 03:36:42 +03:00
|
|
|
off_t cluster2sector(Bit32u cluster_num);
|
2010-12-23 19:17:12 +03:00
|
|
|
int init_directories(const char* dirname);
|
2011-01-09 03:36:42 +03:00
|
|
|
bx_bool read_sector_from_file(const char *path, Bit8u *buffer, Bit32u sector);
|
2011-01-11 00:15:05 +03:00
|
|
|
void set_file_attributes(void);
|
2011-01-09 03:36:42 +03:00
|
|
|
Bit32u fat_get_next(Bit32u current);
|
|
|
|
bx_bool write_file(const char *path, direntry_t *entry, bx_bool create);
|
|
|
|
direntry_t* read_direntry(Bit8u *buffer, char *filename);
|
|
|
|
void parse_directory(const char *path, Bit32u start_cluster);
|
2011-01-11 00:15:05 +03:00
|
|
|
void commit_changes(void);
|
2010-12-23 19:17:12 +03:00
|
|
|
void close_current_file(void);
|
|
|
|
int open_file(mapping_t* mapping);
|
|
|
|
int find_mapping_for_cluster_aux(int cluster_num, int index1, int index2);
|
|
|
|
mapping_t* find_mapping_for_cluster(int cluster_num);
|
2011-01-09 22:20:11 +03:00
|
|
|
mapping_t* find_mapping_for_path(const char* path);
|
2010-12-23 19:17:12 +03:00
|
|
|
int read_cluster(int cluster_num);
|
|
|
|
|
|
|
|
Bit8u *first_sectors;
|
2010-12-27 02:13:29 +03:00
|
|
|
Bit32u offset_to_bootsector;
|
2010-12-30 15:30:58 +03:00
|
|
|
Bit32u offset_to_fat;
|
2010-12-27 02:13:29 +03:00
|
|
|
Bit32u offset_to_root_dir;
|
|
|
|
Bit32u offset_to_data;
|
2010-12-23 19:17:12 +03:00
|
|
|
|
|
|
|
Bit16u cluster_size;
|
|
|
|
Bit8u sectors_per_cluster;
|
2010-12-31 18:39:27 +03:00
|
|
|
Bit32u sectors_per_fat;
|
2010-12-23 19:17:12 +03:00
|
|
|
Bit32u sector_count;
|
|
|
|
Bit32u cluster_count; // total number of clusters of this partition
|
|
|
|
Bit32u max_fat_value;
|
2010-12-27 02:13:29 +03:00
|
|
|
Bit32u first_cluster_of_root_dir;
|
|
|
|
Bit16u root_entries;
|
2010-12-30 15:30:58 +03:00
|
|
|
Bit16u reserved_sectors;
|
2010-12-27 02:13:29 +03:00
|
|
|
|
|
|
|
Bit8u fat_type;
|
|
|
|
array_t fat, directory, mapping;
|
2010-12-23 19:17:12 +03:00
|
|
|
|
|
|
|
int current_fd;
|
|
|
|
mapping_t* current_mapping;
|
|
|
|
Bit8u *cluster; // points to current cluster
|
|
|
|
Bit8u *cluster_buffer; // points to a buffer to hold temp data
|
|
|
|
Bit16u current_cluster;
|
|
|
|
|
2011-01-09 03:36:42 +03:00
|
|
|
const char *vvfat_path;
|
2010-12-23 19:17:12 +03:00
|
|
|
Bit32u sector_num;
|
2010-12-31 18:39:27 +03:00
|
|
|
|
|
|
|
bx_bool use_mbr_file;
|
|
|
|
bx_bool use_boot_file;
|
2011-01-11 00:15:05 +03:00
|
|
|
FILE *vvfat_attr_fd;
|
2011-01-01 22:14:25 +03:00
|
|
|
|
2011-01-09 03:36:42 +03:00
|
|
|
bx_bool vvfat_modified;
|
|
|
|
void *fat2;
|
2011-01-01 22:14:25 +03:00
|
|
|
redolog_t *redolog; // Redolog instance
|
|
|
|
char *redolog_name; // Redolog name
|
|
|
|
char *redolog_temp; // Redolog temporary file name
|
2010-12-23 19:17:12 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|