2005-11-06 14:07:01 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2011-02-25 01:05:47 +03:00
|
|
|
// $Id$
|
2005-11-06 14:07:01 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2021-01-06 00:57:13 +03:00
|
|
|
// Copyright (C) 2005-2021 The Bochs Project
|
2005-11-06 14:07:01 +03:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library 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
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
2009-02-08 12:05:52 +03:00
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2006-03-25 21:04:15 +03:00
|
|
|
#ifndef BX_HDIMAGE_H
|
|
|
|
#define BX_HDIMAGE_H
|
|
|
|
|
2014-03-23 13:19:44 +04:00
|
|
|
// required for access() checks
|
|
|
|
#ifndef F_OK
|
|
|
|
#define F_OK 0
|
|
|
|
#endif
|
|
|
|
|
2010-12-22 00:47:41 +03:00
|
|
|
// hdimage capabilities
|
2011-01-21 19:00:38 +03:00
|
|
|
#define HDIMAGE_READONLY 1
|
|
|
|
#define HDIMAGE_HAS_GEOMETRY 2
|
|
|
|
#define HDIMAGE_AUTO_GEOMETRY 4
|
2010-12-22 00:47:41 +03:00
|
|
|
|
2012-10-03 20:17:37 +04:00
|
|
|
// hdimage format check return values
|
|
|
|
#define HDIMAGE_FORMAT_OK 0
|
|
|
|
#define HDIMAGE_SIZE_ERROR -1
|
|
|
|
#define HDIMAGE_READ_ERROR -2
|
|
|
|
#define HDIMAGE_NO_SIGNATURE -3
|
|
|
|
#define HDIMAGE_TYPE_ERROR -4
|
|
|
|
#define HDIMAGE_VERSION_ERROR -5
|
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
// SPARSE IMAGES HEADER
|
|
|
|
#define SPARSE_HEADER_MAGIC (0x02468ace)
|
2006-06-16 11:29:33 +04:00
|
|
|
#define SPARSE_HEADER_VERSION 2
|
|
|
|
#define SPARSE_HEADER_V1 1
|
2005-11-06 14:07:01 +03:00
|
|
|
#define SPARSE_HEADER_SIZE (256) // Plenty of room for later
|
|
|
|
#define SPARSE_PAGE_NOT_ALLOCATED (0xffffffff)
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
Bit32u magic;
|
|
|
|
Bit32u version;
|
|
|
|
Bit32u pagesize;
|
|
|
|
Bit32u numpages;
|
2006-06-16 11:29:33 +04:00
|
|
|
Bit64u disk;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2006-06-16 11:29:33 +04:00
|
|
|
Bit32u padding[58];
|
2005-11-06 14:07:01 +03:00
|
|
|
} sparse_header_t;
|
|
|
|
|
|
|
|
#define STANDARD_HEADER_MAGIC "Bochs Virtual HD Image"
|
2006-05-15 01:15:33 +04:00
|
|
|
#define STANDARD_HEADER_V1 (0x00010000)
|
|
|
|
#define STANDARD_HEADER_VERSION (0x00020000)
|
2005-11-06 14:07:01 +03:00
|
|
|
#define STANDARD_HEADER_SIZE (512)
|
|
|
|
|
|
|
|
|
|
|
|
// WARNING : headers are kept in x86 (little) endianness
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
Bit8u magic[32];
|
|
|
|
Bit8u type[16];
|
|
|
|
Bit8u subtype[16];
|
|
|
|
Bit32u version;
|
|
|
|
Bit32u header;
|
|
|
|
} standard_header_t;
|
|
|
|
|
|
|
|
#define REDOLOG_TYPE "Redolog"
|
|
|
|
#define REDOLOG_SUBTYPE_UNDOABLE "Undoable"
|
|
|
|
#define REDOLOG_SUBTYPE_VOLATILE "Volatile"
|
|
|
|
#define REDOLOG_SUBTYPE_GROWING "Growing"
|
|
|
|
|
|
|
|
#define REDOLOG_PAGE_NOT_ALLOCATED (0xffffffff)
|
|
|
|
|
|
|
|
#define UNDOABLE_REDOLOG_EXTENSION ".redolog"
|
|
|
|
#define UNDOABLE_REDOLOG_EXTENSION_LENGTH (strlen(UNDOABLE_REDOLOG_EXTENSION))
|
|
|
|
#define VOLATILE_REDOLOG_EXTENSION ".XXXXXX"
|
|
|
|
#define VOLATILE_REDOLOG_EXTENSION_LENGTH (strlen(VOLATILE_REDOLOG_EXTENSION))
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
// the fields in the header are kept in little endian
|
|
|
|
Bit32u catalog; // #entries in the catalog
|
|
|
|
Bit32u bitmap; // bitmap size in bytes
|
|
|
|
Bit32u extent; // extent size in bytes
|
2012-07-21 16:00:46 +04:00
|
|
|
Bit32u timestamp; // modification time in FAT format (subtype 'undoable' only)
|
2005-11-06 14:07:01 +03:00
|
|
|
Bit64u disk; // disk size in bytes
|
|
|
|
} redolog_specific_header_t;
|
|
|
|
|
2006-05-15 01:15:33 +04:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
// the fields in the header are kept in little endian
|
|
|
|
Bit32u catalog; // #entries in the catalog
|
|
|
|
Bit32u bitmap; // bitmap size in bytes
|
|
|
|
Bit32u extent; // extent size in bytes
|
|
|
|
Bit64u disk; // disk size in bytes
|
|
|
|
} redolog_specific_header_v1_t;
|
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
standard_header_t standard;
|
|
|
|
redolog_specific_header_t specific;
|
|
|
|
|
|
|
|
Bit8u padding[STANDARD_HEADER_SIZE - (sizeof (standard_header_t) + sizeof (redolog_specific_header_t))];
|
|
|
|
} redolog_header_t;
|
|
|
|
|
2006-05-15 01:15:33 +04:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
standard_header_t standard;
|
|
|
|
redolog_specific_header_v1_t specific;
|
|
|
|
|
|
|
|
Bit8u padding[STANDARD_HEADER_SIZE - (sizeof (standard_header_t) + sizeof (redolog_specific_header_v1_t))];
|
|
|
|
} redolog_header_v1_t;
|
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
// htod : convert host to disk (little) endianness
|
|
|
|
// dtoh : convert disk (little) to host endianness
|
|
|
|
#if defined (BX_LITTLE_ENDIAN)
|
2013-10-13 18:33:55 +04:00
|
|
|
#define htod16(val) (val)
|
|
|
|
#define dtoh16(val) (val)
|
2005-11-06 14:07:01 +03:00
|
|
|
#define htod32(val) (val)
|
|
|
|
#define dtoh32(val) (val)
|
|
|
|
#define htod64(val) (val)
|
|
|
|
#define dtoh64(val) (val)
|
|
|
|
#else
|
2013-10-13 18:33:55 +04:00
|
|
|
#define htod16(val) ((((val)&0xff00)>>8) | (((val)&0xff)<<8))
|
|
|
|
#define dtoh16(val) htod16(val)
|
2011-09-08 21:53:28 +04:00
|
|
|
#define htod32(val) bx_bswap32(val)
|
2005-11-06 14:07:01 +03:00
|
|
|
#define dtoh32(val) htod32(val)
|
2011-09-08 21:53:28 +04:00
|
|
|
#define htod64(val) bx_bswap64(val)
|
2005-11-06 14:07:01 +03:00
|
|
|
#define dtoh64(val) htod64(val)
|
|
|
|
#endif
|
|
|
|
|
2013-10-13 18:33:55 +04:00
|
|
|
class device_image_t;
|
|
|
|
class redolog_t;
|
|
|
|
|
2021-01-16 15:47:02 +03:00
|
|
|
#ifdef BXIMAGE
|
|
|
|
int bx_create_image_file(const char *filename);
|
|
|
|
#endif
|
2020-12-27 20:26:33 +03:00
|
|
|
BOCHSAPI_MSVCONLY int bx_read_image(int fd, Bit64s offset, void *buf, int count);
|
|
|
|
BOCHSAPI_MSVCONLY int bx_write_image(int fd, Bit64s offset, void *buf, int count);
|
|
|
|
BOCHSAPI_MSVCONLY int bx_close_image(int fd, const char *pathname);
|
2012-09-27 22:38:30 +04:00
|
|
|
#ifndef WIN32
|
|
|
|
int hdimage_open_file(const char *pathname, int flags, Bit64u *fsize, time_t *mtime);
|
|
|
|
#else
|
2020-12-27 20:26:33 +03:00
|
|
|
BOCHSAPI_MSVCONLY int hdimage_open_file(const char *pathname, int flags, Bit64u *fsize, FILETIME *mtime);
|
2012-09-27 22:38:30 +04:00
|
|
|
#endif
|
2021-01-06 00:57:13 +03:00
|
|
|
bool hdimage_detect_image_mode(const char *pathname, const char **image_mode);
|
2021-01-31 18:44:39 +03:00
|
|
|
BOCHSAPI_MSVCONLY bool hdimage_backup_file(int fd, const char *backup_fname);
|
|
|
|
BOCHSAPI_MSVCONLY bool hdimage_copy_file(const char *src, const char *dst);
|
|
|
|
bool coherency_check(device_image_t *ro_disk, redolog_t *redolog);
|
2013-10-13 18:33:55 +04:00
|
|
|
#ifndef WIN32
|
|
|
|
Bit16u fat_datetime(time_t time, int return_time);
|
|
|
|
#else
|
2020-12-27 20:26:33 +03:00
|
|
|
Bit16u BOCHSAPI_MSVCONLY fat_datetime(FILETIME time, int return_time);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-08-05 13:33:33 +04:00
|
|
|
|
2012-09-27 22:38:30 +04:00
|
|
|
// base class
|
2020-12-27 20:26:33 +03:00
|
|
|
class BOCHSAPI_MSVCONLY device_image_t
|
2005-11-06 14:07:01 +03:00
|
|
|
{
|
|
|
|
public:
|
2006-06-16 11:29:33 +04:00
|
|
|
// Default constructor
|
|
|
|
device_image_t();
|
2006-12-27 18:21:03 +03:00
|
|
|
virtual ~device_image_t() {}
|
2006-06-16 11:29:33 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
// Open a image. Returns non-negative if successful.
|
2012-10-07 22:36:22 +04:00
|
|
|
virtual int open(const char* pathname);
|
|
|
|
|
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
|
|
|
virtual int open(const char* pathname, int flags) = 0;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Close the image.
|
2006-11-18 14:51:07 +03:00
|
|
|
virtual void close() = 0;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
2006-11-18 14:51:07 +03:00
|
|
|
virtual Bit64s lseek(Bit64s offset, int whence) = 0;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
2006-11-18 14:51:07 +03:00
|
|
|
virtual ssize_t read(void* buf, size_t count) = 0;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
2006-11-18 14:51:07 +03:00
|
|
|
virtual ssize_t write(const void* buf, size_t count) = 0;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2010-12-22 00:47:41 +03:00
|
|
|
// Get image capabilities
|
|
|
|
virtual Bit32u get_capabilities();
|
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
// Get modification time in FAT format
|
2014-04-04 23:14:32 +04:00
|
|
|
virtual Bit32u get_timestamp();
|
2012-10-07 22:36:22 +04:00
|
|
|
|
2012-10-03 20:17:37 +04:00
|
|
|
// Check image format
|
|
|
|
static int check_format(int fd, Bit64u imgsize) {return HDIMAGE_NO_SIGNATURE;}
|
|
|
|
|
2021-01-16 15:47:02 +03:00
|
|
|
#ifdef BXIMAGE
|
|
|
|
// Create new image file
|
|
|
|
virtual int create_image(const char *pathname, Bit64u size) {return 0;}
|
|
|
|
#else
|
2012-09-16 23:51:30 +04:00
|
|
|
// Save/restore support
|
|
|
|
virtual void register_state(bx_list_c *parent);
|
2021-01-31 18:44:39 +03:00
|
|
|
virtual bool save_state(const char *backup_fname) {return 0;}
|
2012-09-20 01:05:18 +04:00
|
|
|
virtual void restore_state(const char *backup_fname) {}
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-16 23:51:30 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
unsigned cylinders;
|
|
|
|
unsigned heads;
|
2011-11-26 19:09:00 +04:00
|
|
|
unsigned spt;
|
2018-03-18 12:07:31 +03:00
|
|
|
unsigned sect_size;
|
2006-06-09 00:32:00 +04:00
|
|
|
Bit64u hd_size;
|
2012-10-07 22:36:22 +04:00
|
|
|
protected:
|
|
|
|
#ifndef WIN32
|
|
|
|
time_t mtime;
|
|
|
|
#else
|
|
|
|
FILETIME mtime;
|
|
|
|
#endif
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
// FLAT MODE
|
2014-01-23 22:35:18 +04:00
|
|
|
class flat_image_t : public device_image_t
|
2005-11-06 14:07:01 +03:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
2006-11-18 14:51:07 +03:00
|
|
|
int open(const char* pathname, int flags);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Close the image.
|
2006-11-18 14:51:07 +03:00
|
|
|
void close();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
2006-11-18 14:51:07 +03:00
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
2006-11-18 14:51:07 +03:00
|
|
|
ssize_t read(void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
2006-11-18 14:51:07 +03:00
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-10-03 20:17:37 +04:00
|
|
|
// Check image format
|
|
|
|
static int check_format(int fd, Bit64u imgsize);
|
|
|
|
|
2013-10-13 18:33:55 +04:00
|
|
|
#ifndef BXIMAGE
|
2012-09-16 23:51:30 +04:00
|
|
|
// Save/restore support
|
2021-01-31 18:44:39 +03:00
|
|
|
bool save_state(const char *backup_fname);
|
2012-09-22 14:04:28 +04:00
|
|
|
void restore_state(const char *backup_fname);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-16 23:51:30 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
private:
|
|
|
|
int fd;
|
2012-09-25 09:12:46 +04:00
|
|
|
const char *pathname;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
// CONCAT MODE
|
|
|
|
class concat_image_t : public device_image_t
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Default constructor
|
|
|
|
concat_image_t();
|
2008-01-27 01:24:03 +03:00
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
|
|
|
int open(const char* pathname, int flags);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Close the image.
|
2006-06-09 00:32:00 +04:00
|
|
|
void close();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
2006-06-09 00:32:00 +04:00
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
2006-06-09 00:32:00 +04:00
|
|
|
ssize_t read(void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
2006-06-09 00:32:00 +04:00
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2013-10-13 18:33:55 +04:00
|
|
|
#ifndef BXIMAGE
|
2012-09-18 23:00:25 +04:00
|
|
|
// Save/restore support
|
2021-01-31 18:44:39 +03:00
|
|
|
bool save_state(const char *backup_fname);
|
2012-09-25 09:12:46 +04:00
|
|
|
void restore_state(const char *backup_fname);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-18 23:00:25 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
private:
|
|
|
|
#define BX_CONCAT_MAX_IMAGES 8
|
|
|
|
int fd_table[BX_CONCAT_MAX_IMAGES];
|
2015-10-17 13:06:36 +03:00
|
|
|
Bit64u start_offset_table[BX_CONCAT_MAX_IMAGES];
|
|
|
|
Bit64u length_table[BX_CONCAT_MAX_IMAGES];
|
2006-06-09 00:32:00 +04:00
|
|
|
void increment_string(char *str);
|
2005-11-06 14:07:01 +03:00
|
|
|
int maxfd; // number of entries in tables that are valid
|
|
|
|
|
|
|
|
// notice if anyone does sequential read or write without seek in between.
|
|
|
|
// This can be supported pretty easily, but needs additional checks.
|
|
|
|
// 0=something other than seek was last operation
|
|
|
|
// 1=seek was last operation
|
2020-09-01 23:13:56 +03:00
|
|
|
//int seek_was_last_op;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// the following variables tell which partial image file to use for
|
|
|
|
// the next read and write.
|
|
|
|
int index; // index into table
|
2015-10-17 13:06:36 +03:00
|
|
|
int curr_fd; // fd to use for reads and writes
|
|
|
|
Bit64u curr_min, curr_max; // byte offset boundary of this image
|
|
|
|
Bit64u total_offset; // current byte offset
|
2012-09-25 09:12:46 +04:00
|
|
|
const char *pathname0;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
// SPARSE MODE
|
|
|
|
class sparse_image_t : public device_image_t
|
|
|
|
{
|
|
|
|
|
|
|
|
// Format of a sparse file:
|
|
|
|
// 256 byte header, containing details such as page size and number of pages
|
|
|
|
// Page indirection table, mapping virtual pages to physical pages within file
|
|
|
|
// Physical pages till end of file
|
|
|
|
|
|
|
|
public:
|
2012-07-24 23:27:22 +04:00
|
|
|
// Default constructor
|
|
|
|
sparse_image_t();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
|
|
|
int open(const char* pathname, int flags);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
// Close the image.
|
|
|
|
void close();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
|
|
|
ssize_t read(void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-10-03 20:17:37 +04:00
|
|
|
// Check image format
|
|
|
|
static int check_format(int fd, Bit64u imgsize);
|
|
|
|
|
2021-01-16 15:47:02 +03:00
|
|
|
#ifdef BXIMAGE
|
|
|
|
// Create new image file
|
|
|
|
int create_image(const char *pathname, Bit64u size);
|
|
|
|
#else
|
2012-09-18 23:00:25 +04:00
|
|
|
// Save/restore support
|
2021-01-31 18:44:39 +03:00
|
|
|
bool save_state(const char *backup_fname);
|
2012-09-23 15:03:52 +04:00
|
|
|
void restore_state(const char *backup_fname);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-18 23:00:25 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
private:
|
2012-07-24 23:27:22 +04:00
|
|
|
int fd;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
#ifdef _POSIX_MAPPED_FILES
|
2012-07-24 23:27:22 +04:00
|
|
|
void * mmap_header;
|
|
|
|
size_t mmap_length;
|
|
|
|
size_t system_pagesize_mask;
|
2005-11-06 14:07:01 +03:00
|
|
|
#endif
|
2012-07-24 23:27:22 +04:00
|
|
|
Bit32u *pagetable;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
// Header is written to disk in little-endian (x86) format
|
|
|
|
// Thus needs to be converted on big-endian systems before read
|
|
|
|
// The pagetable is also kept little endian
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
sparse_header_t header;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
Bit32u pagesize;
|
|
|
|
int pagesize_shift;
|
|
|
|
Bit32u pagesize_mask;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
Bit64s data_start;
|
2012-10-03 20:17:37 +04:00
|
|
|
Bit64u underlying_filesize;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
char *pathname;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2020-09-01 23:13:56 +03:00
|
|
|
//Bit64s position;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
Bit32u position_virtual_page;
|
|
|
|
Bit32u position_physical_page;
|
|
|
|
Bit32u position_page_offset;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
Bit64s underlying_current_filepos;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
Bit64s total_size;
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
void panic(const char * message);
|
|
|
|
Bit64s get_physical_offset();
|
|
|
|
void set_virtual_page(Bit32u new_virtual_page);
|
2012-10-03 20:17:37 +04:00
|
|
|
int read_header();
|
2012-07-24 23:27:22 +04:00
|
|
|
ssize_t read_page_fragment(Bit32u read_virtual_page, Bit32u read_page_offset, size_t read_size, void * buf);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-07-24 23:27:22 +04:00
|
|
|
sparse_image_t *parent_image;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
2013-03-08 22:25:32 +04:00
|
|
|
#ifdef WIN32
|
2005-11-06 14:07:01 +03:00
|
|
|
class dll_image_t : public device_image_t
|
|
|
|
{
|
|
|
|
public:
|
2013-03-08 22:25:32 +04:00
|
|
|
dll_image_t();
|
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
|
|
|
int open(const char* pathname, int flags);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Close the image.
|
2006-11-18 14:51:07 +03:00
|
|
|
void close();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
2006-11-18 14:51:07 +03:00
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
2006-11-18 14:51:07 +03:00
|
|
|
ssize_t read(void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
2006-11-18 14:51:07 +03:00
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
private:
|
2013-03-08 22:25:32 +04:00
|
|
|
int vunit;
|
|
|
|
Bit64s vblk;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// REDOLOG class
|
2020-12-27 20:26:33 +03:00
|
|
|
class BOCHSAPI_MSVCONLY redolog_t
|
2005-11-06 14:07:01 +03:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
redolog_t();
|
2006-06-05 12:00:21 +04:00
|
|
|
int make_header(const char* type, Bit64u size);
|
|
|
|
int create(const char* filename, const char* type, Bit64u size);
|
|
|
|
int create(int filedes, const char* type, Bit64u size);
|
2006-06-09 00:32:00 +04:00
|
|
|
int open(const char* filename, const char* type);
|
2012-10-04 21:01:17 +04:00
|
|
|
int open(const char* filename, const char* type, int flags);
|
2006-06-05 12:00:21 +04:00
|
|
|
void close();
|
|
|
|
Bit64u get_size();
|
2012-07-21 16:00:46 +04:00
|
|
|
Bit32u get_timestamp();
|
2021-01-31 18:44:39 +03:00
|
|
|
bool set_timestamp(Bit32u timestamp);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2006-06-09 00:32:00 +04:00
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2006-06-05 12:00:21 +04:00
|
|
|
ssize_t read(void* buf, size_t count);
|
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-10-03 20:17:37 +04:00
|
|
|
static int check_format(int fd, const char *subtype);
|
|
|
|
|
2013-10-23 12:35:21 +04:00
|
|
|
#ifdef BXIMAGE
|
|
|
|
int commit(device_image_t *base_image);
|
|
|
|
#else
|
2021-01-31 18:44:39 +03:00
|
|
|
bool save_state(const char *backup_fname);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-17 23:30:40 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
private:
|
|
|
|
void print_header();
|
2017-01-30 22:08:37 +03:00
|
|
|
char *pathname;
|
2005-11-06 14:07:01 +03:00
|
|
|
int fd;
|
|
|
|
redolog_header_t header; // Header is kept in x86 (little) endianness
|
|
|
|
Bit32u *catalog;
|
|
|
|
Bit8u *bitmap;
|
2021-01-31 18:44:39 +03:00
|
|
|
bool bitmap_update;
|
2005-11-06 14:07:01 +03:00
|
|
|
Bit32u extent_index;
|
|
|
|
Bit32u extent_offset;
|
|
|
|
Bit32u extent_next;
|
|
|
|
|
2011-01-07 21:35:34 +03:00
|
|
|
Bit32u bitmap_blocks;
|
|
|
|
Bit32u extent_blocks;
|
|
|
|
|
|
|
|
Bit64s imagepos;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
// GROWING MODE
|
|
|
|
class growing_image_t : public device_image_t
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Contructor
|
2006-06-09 00:32:00 +04:00
|
|
|
growing_image_t();
|
2007-03-10 15:53:54 +03:00
|
|
|
virtual ~growing_image_t();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
|
|
|
int open(const char* pathname, int flags);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Close the image.
|
2006-06-05 12:00:21 +04:00
|
|
|
void close();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
2006-06-09 00:32:00 +04:00
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
2006-06-05 12:00:21 +04:00
|
|
|
ssize_t read(void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
2006-06-05 12:00:21 +04:00
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2014-04-04 23:14:32 +04:00
|
|
|
// Get modification time in FAT format
|
|
|
|
virtual Bit32u get_timestamp();
|
|
|
|
|
2012-10-03 20:17:37 +04:00
|
|
|
// Check image format
|
|
|
|
static int check_format(int fd, Bit64u imgsize);
|
|
|
|
|
2021-01-16 15:47:02 +03:00
|
|
|
#ifdef BXIMAGE
|
|
|
|
// Create new image file
|
|
|
|
int create_image(const char *pathname, Bit64u size);
|
|
|
|
#else
|
2012-09-17 23:30:40 +04:00
|
|
|
// Save/restore support
|
2021-01-31 18:44:39 +03:00
|
|
|
bool save_state(const char *backup_fname);
|
2012-09-23 23:28:28 +04:00
|
|
|
void restore_state(const char *backup_fname);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-17 23:30:40 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
private:
|
|
|
|
redolog_t *redolog;
|
2012-09-25 09:12:46 +04:00
|
|
|
const char *pathname;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
// UNDOABLE MODE
|
|
|
|
class undoable_image_t : public device_image_t
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Contructor
|
2006-06-09 00:32:00 +04:00
|
|
|
undoable_image_t(const char* redolog_name);
|
2007-03-10 15:53:54 +03:00
|
|
|
virtual ~undoable_image_t();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
|
|
|
int open(const char* pathname, int flags);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Close the image.
|
2006-06-09 00:32:00 +04:00
|
|
|
void close();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
2006-06-09 00:32:00 +04:00
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
2006-06-09 00:32:00 +04:00
|
|
|
ssize_t read(void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
2006-06-09 00:32:00 +04:00
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2018-03-31 20:07:44 +03:00
|
|
|
// Get image capabilities
|
|
|
|
virtual Bit32u get_capabilities() {return caps;}
|
|
|
|
|
2013-10-13 18:33:55 +04:00
|
|
|
#ifndef BXIMAGE
|
2012-09-17 23:30:40 +04:00
|
|
|
// Save/restore support
|
2021-01-31 18:44:39 +03:00
|
|
|
bool save_state(const char *backup_fname);
|
2012-09-22 00:25:23 +04:00
|
|
|
void restore_state(const char *backup_fname);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-17 23:30:40 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
private:
|
|
|
|
redolog_t *redolog; // Redolog instance
|
2012-10-07 22:36:22 +04:00
|
|
|
device_image_t *ro_disk; // Read-only base disk instance
|
2005-11-06 14:07:01 +03:00
|
|
|
char *redolog_name; // Redolog name
|
2018-03-31 20:07:44 +03:00
|
|
|
Bit32u caps;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// VOLATILE MODE
|
|
|
|
class volatile_image_t : public device_image_t
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Contructor
|
2006-06-09 00:32:00 +04:00
|
|
|
volatile_image_t(const char* redolog_name);
|
2007-03-10 15:53:54 +03:00
|
|
|
virtual ~volatile_image_t();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2012-10-07 22:36:22 +04:00
|
|
|
// Open an image with specific flags. Returns non-negative if successful.
|
|
|
|
int open(const char* pathname, int flags);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Close the image.
|
2006-06-09 00:32:00 +04:00
|
|
|
void close();
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Position ourselves. Return the resulting offset from the
|
|
|
|
// beginning of the file.
|
2006-06-09 00:32:00 +04:00
|
|
|
Bit64s lseek(Bit64s offset, int whence);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Read count bytes to the buffer buf. Return the number of
|
|
|
|
// bytes read (count).
|
2006-06-09 00:32:00 +04:00
|
|
|
ssize_t read(void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
|
|
|
// Write count bytes from buf. Return the number of bytes
|
|
|
|
// written (count).
|
2006-06-09 00:32:00 +04:00
|
|
|
ssize_t write(const void* buf, size_t count);
|
2005-11-06 14:07:01 +03:00
|
|
|
|
2018-03-31 20:07:44 +03:00
|
|
|
// Get image capabilities
|
|
|
|
virtual Bit32u get_capabilities() {return caps;}
|
|
|
|
|
2013-10-13 18:33:55 +04:00
|
|
|
#ifndef BXIMAGE
|
2012-09-17 23:30:40 +04:00
|
|
|
// Save/restore support
|
2021-01-31 18:44:39 +03:00
|
|
|
bool save_state(const char *backup_fname);
|
2012-09-22 00:25:23 +04:00
|
|
|
void restore_state(const char *backup_fname);
|
2013-10-13 18:33:55 +04:00
|
|
|
#endif
|
2012-09-17 23:30:40 +04:00
|
|
|
|
2005-11-06 14:07:01 +03:00
|
|
|
private:
|
|
|
|
redolog_t *redolog; // Redolog instance
|
2012-10-07 22:36:22 +04:00
|
|
|
device_image_t *ro_disk; // Read-only base disk instance
|
2005-11-06 14:07:01 +03:00
|
|
|
char *redolog_name; // Redolog name
|
|
|
|
char *redolog_temp; // Redolog temporary file name
|
2018-03-31 20:07:44 +03:00
|
|
|
Bit32u caps;
|
2005-11-06 14:07:01 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-10-13 18:33:55 +04:00
|
|
|
#ifndef BXIMAGE
|
2020-12-27 20:26:33 +03:00
|
|
|
|
|
|
|
#define DEV_hdimage_init_image(a,b,c) bx_hdimage_ctl.init_image(a,b,c)
|
|
|
|
#define DEV_hdimage_init_cdrom(a) bx_hdimage_ctl.init_cdrom(a)
|
|
|
|
|
|
|
|
class BOCHSAPI bx_hdimage_ctl_c : public logfunctions {
|
2011-01-13 01:34:42 +03:00
|
|
|
public:
|
2011-01-14 19:43:55 +03:00
|
|
|
bx_hdimage_ctl_c();
|
2011-01-13 01:34:42 +03:00
|
|
|
virtual ~bx_hdimage_ctl_c() {}
|
2020-12-27 20:26:33 +03:00
|
|
|
void init(void);
|
2021-01-06 00:57:13 +03:00
|
|
|
const char **get_mode_names();
|
|
|
|
int get_mode_id(const char *mode);
|
|
|
|
void list_modules(void);
|
2020-12-27 20:26:33 +03:00
|
|
|
void exit(void);
|
2021-01-06 00:57:13 +03:00
|
|
|
device_image_t *init_image(const char *image_mode, Bit64u disk_size, const char *journal);
|
2020-12-27 20:26:33 +03:00
|
|
|
cdrom_base_c *init_cdrom(const char *dev);
|
2011-01-13 01:34:42 +03:00
|
|
|
};
|
2020-12-27 20:26:33 +03:00
|
|
|
|
|
|
|
BOCHSAPI extern bx_hdimage_ctl_c bx_hdimage_ctl;
|
|
|
|
|
2021-01-10 13:18:23 +03:00
|
|
|
#endif // ifndef BXIMAGE
|
|
|
|
|
2020-12-27 20:26:33 +03:00
|
|
|
//
|
|
|
|
// The hdimage_locator class is used by device_image_t classes to register
|
|
|
|
// their name. The common hdimage code uses the static 'create' method
|
|
|
|
// to locate and instantiate a device_image_t class.
|
|
|
|
//
|
|
|
|
class BOCHSAPI_MSVCONLY hdimage_locator_c {
|
|
|
|
public:
|
2021-01-06 00:57:13 +03:00
|
|
|
static bool module_present(const char *mode);
|
|
|
|
static Bit8u get_modules_count(void);
|
|
|
|
static const char* get_module_name(Bit8u index);
|
|
|
|
static void cleanup(void);
|
2020-12-27 20:26:33 +03:00
|
|
|
static device_image_t *create(const char *mode, Bit64u disk_size, const char *journal);
|
2021-01-09 00:17:47 +03:00
|
|
|
static bool detect_image_mode(int fd, Bit64u disk_size, const char **image_mode);
|
2020-12-27 20:26:33 +03:00
|
|
|
protected:
|
|
|
|
hdimage_locator_c(const char *mode);
|
|
|
|
virtual ~hdimage_locator_c();
|
|
|
|
virtual device_image_t *allocate(Bit64u disk_size, const char *journal) = 0;
|
2021-01-09 00:17:47 +03:00
|
|
|
virtual int check_format(int fd, Bit64u disk_size) {return -1;}
|
2020-12-27 20:26:33 +03:00
|
|
|
private:
|
2021-01-06 00:57:13 +03:00
|
|
|
static Bit8u count;
|
2020-12-27 20:26:33 +03:00
|
|
|
static hdimage_locator_c *all;
|
|
|
|
hdimage_locator_c *next;
|
|
|
|
const char *mode;
|
|
|
|
};
|
|
|
|
|
2006-03-25 21:04:15 +03:00
|
|
|
#endif
|