Bochs/bochs/misc/bximage.cc
Volker Ruppert e4978addd1 bximage: added simple partition table viewer to the info function.
The code is based on parts of the SF patch "bxmount".
2024-04-14 18:15:52 +02:00

1073 lines
33 KiB
C++

/////////////////////////////////////////////////////////////////////////
// $Id$
/////////////////////////////////////////////////////////////////////////
//
// Copyright (C) 2001-2024 The Bochs Project
//
// 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
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
/////////////////////////////////////////////////////////////////////////
// Create empty hard disk or floppy disk images for bochs.
// Convert a hard disk image from one format (mode) to another.
// Resize a hard disk image.
// Commit a redolog file to a supported base image.
#ifndef BXIMAGE
#define BXIMAGE
#endif
#include "config.h"
#include "bxcompat.h"
#if defined(WIN32) && !defined(__CYGWIN__)
# include <conio.h>
# include <winioctl.h>
#endif
#include <ctype.h>
#include "osdep.h"
#include "bswap.h"
#include "iodev/hdimage/hdimage.h"
#include "iodev/hdimage/vmware3.h"
#include "iodev/hdimage/vmware4.h"
#include "iodev/hdimage/vpc.h"
#include "iodev/hdimage/vbox.h"
#define BXIMAGE_FUNC_NULL 0
#define BXIMAGE_FUNC_CREATE_IMAGE 1
#define BXIMAGE_FUNC_CONVERT_IMAGE 2
#define BXIMAGE_FUNC_RESIZE_IMAGE 3
#define BXIMAGE_FUNC_COMMIT_UNDOABLE 4
#define BXIMAGE_FUNC_IMAGE_INFO 5
#define BX_MAX_CYL_BITS 24 // 8 TB
#define SECTOR_SIZE 512
const int bx_max_hd_megs = (int)(((1 << BX_MAX_CYL_BITS) - 1) * 16.0 * 63.0 / 2048.0);
int bximage_func;
int bx_hdimage;
int bx_fdsize_idx;
int bx_min_hd_megs = 10;
int bx_hdsize;
int bx_imagemode;
int bx_backup;
int bx_interactive;
int bx_sectsize_idx;
Bit16u bx_sectsize_val;
char bx_filename_1[512];
char bx_filename_2[522];
const char *EOF_ERR = "ERROR: End of input";
const char *svnid = "$Id$";
const char *divider = "========================================================================";
const char *main_menu_prompt =
"\n"
"1. Create new floppy or hard disk image\n"
"2. Convert hard disk image to other format (mode)\n"
"3. Resize hard disk image\n"
"4. Commit 'undoable' redolog to base image\n"
"5. Disk image info\n"
"\n"
"0. Quit\n"
"\n"
"Please choose one ";
// menu data for choosing floppy/hard disk
const char *fdhd_menu = "\nDo you want to create a floppy disk image or a hard disk image?\nPlease type hd or fd. ";
const char *fdhd_choices[] = { "fd", "hd" };
int fdhd_n_choices = 2;
// menu data for choosing floppy size
const char *fdsize_menu = "\nChoose the size of floppy disk image to create.\nPlease type 160k, 180k, 320k, 360k, 720k, 1.2M, 1.44M, 1.68M, 1.72M, or 2.88M.\n ";
const char *fdsize_choices[] = { "160k","180k","320k","360k","720k","1.2M","1.44M","1.68M","1.72M","2.88M" };
const unsigned fdsize_sectors[] = { 320, 360, 640, 720, 1440, 2400, 2880, 3360, 3444, 5760 };
int fdsize_n_choices = 10;
// menu data for choosing disk mode
const char *hdmode_menu = "\nWhat kind of image should I create?\nPlease type flat, sparse, growing, vpc or vmware4. ";
const char *hdmode_choices[] = {"flat", "sparse", "growing", "vpc", "vmware4" };
int hdmode_n_choices = 5;
// menu data for choosing hard disk sector size
const char *sectsize_menu = "\nChoose the size of hard disk sectors.\nPlease type 512, 1024 or 4096. ";
const char *sectsize_choices[] = { "512","1024","4096" };
int sectsize_n_choices = 3;
#if !BX_HAVE_SNPRINTF
#include <stdarg.h>
/* XXX use real snprintf */
/* if they don't have snprintf, just use sprintf */
int snprintf(char *s, size_t maxlen, const char *format, ...)
{
va_list arg;
int done;
va_start(arg, format);
done = vsprintf(s, format, arg);
va_end(arg);
return done;
}
#endif /* !BX_HAVE_SNPRINTF */
#if !BX_HAVE_MKSTEMP
int bx_mkstemp(char *tpl)
{
mktemp(tpl);
return ::open(tpl, O_RDWR | O_CREAT | O_TRUNC
#ifdef O_BINARY
| O_BINARY
#endif
, S_IWUSR | S_IRUSR | S_IRGRP | S_IWGRP);
}
#endif // !BX_HAVE_MKSTEMP
void myexit(int code)
{
#if defined(WIN32) && !defined(__CYGWIN__)
printf("\nPress any key to continue\n");
getch();
#endif
exit(code);
}
// stolen from main.cc
void bx_center_print(FILE *file, const char *line, int maxwidth)
{
int imax;
int i;
imax = (maxwidth - strlen(line)) >> 1;
for (i=0; i<imax; i++) fputc(' ', file);
fputs(line, file);
}
void print_banner()
{
printf("%s\n", divider);
bx_center_print(stdout, "bximage\n", 72);
bx_center_print(stdout, "Disk Image Creation / Conversion / Resize and Commit Tool for Bochs\n", 72);
bx_center_print(stdout, svnid, 72);
printf("\n%s\n", divider);
}
// this is how we crash
void fatal(const char *c)
{
printf("%s\n", c);
myexit(1);
}
/* check if the argument string is present in the list -
returns index on success, -1 on failure. */
int get_menu_index(char *arg, int n_choices, const char *choice[])
{
int i;
for (i=0; i<n_choices; i++) {
if (!strcmp(choice[i], arg)) {
// matched, return the choice number
return i;
}
}
return -1;
}
// remove leading spaces, newline junk at end. returns pointer to
// cleaned string, which is between s0 and the null
char *clean_string(char *s0)
{
char *s = s0;
char *ptr;
// find first nonblank
while (isspace(*s))
s++;
// truncate string at first non-alphanumeric
ptr = s;
while (isprint(*ptr))
ptr++;
*ptr = 0;
return s;
}
// returns 0 on success, -1 on failure. The value goes into out.
int ask_int(const char *prompt, int min, int max, int the_default, int *out)
{
int n = max + 1;
char buffer[1024];
char *clean;
int illegal;
while (1) {
printf("%s", prompt);
printf("[%d] ", the_default);
fflush(stdout);
if (!fgets(buffer, sizeof(buffer), stdin))
return -1;
clean = clean_string(buffer);
if (strlen(clean) < 1) {
// empty line, use the default
*out = the_default;
return 0;
}
illegal = (1 != sscanf(buffer, "%d", &n));
if (illegal || n<min || n>max) {
fprintf(stderr, "Your choice (%s) was not an integer between %d and %d.\n\n",
clean, min, max);
} else {
// choice is okay
*out = n;
return 0;
}
}
}
int ask_menu(const char *prompt, int n_choices, const char *choice[], int the_default, int *out)
{
char buffer[1024];
char *clean;
int i;
*out = -1;
while(1) {
printf("%s", prompt);
printf("[%s] ", choice[the_default]);
fflush(stdout);
if (!fgets(buffer, sizeof(buffer), stdin))
return -1;
clean = clean_string(buffer);
if (strlen(clean) < 1) {
// empty line, use the default
*out = the_default;
return 0;
}
for (i=0; i<n_choices; i++) {
if (!strcmp(choice[i], clean)) {
// matched, return the choice number
*out = i;
return 0;
}
}
printf("Your choice (%s) did not match any of the choices:\n", clean);
for (i=0; i<n_choices; i++) {
if (i>0) printf(", ");
printf("%s", choice[i]);
}
printf("\n");
fflush(stdout);
}
}
int ask_yn(const char *prompt, int the_default, int *out)
{
char buffer[16];
char *clean;
*out = -1;
while (1) {
printf("%s", prompt);
printf("[%s] ", the_default?"yes":"no");
fflush(stdout);
if (!fgets(buffer, sizeof(buffer), stdin))
return -1;
clean = clean_string(buffer);
if (strlen(clean) < 1) {
// empty line, use the default
*out = the_default;
return 0;
}
switch (tolower(clean[0])) {
case 'y': *out=1; return 0;
case 'n': *out=0; return 0;
}
fprintf(stderr, "Please type either yes or no.\n");
}
}
int ask_string(const char *prompt, char *the_default, char *out)
{
char buffer[1024];
char *clean;
printf("%s", prompt);
printf("[%s] ", the_default);
fflush(stdout);
if (!fgets(buffer, sizeof(buffer), stdin))
return -1;
clean = clean_string(buffer);
if (strlen(clean) < 1) {
// empty line, use the default
strcpy(out, the_default);
return 0;
}
strcpy(out, clean);
return 0;
}
device_image_t* init_image(const char *imgmode)
{
device_image_t *hdimage = NULL;
// instantiate the right class
if (!strcmp(imgmode, "flat")) {
hdimage = new flat_image_t();
} else if (!strcmp(imgmode, "concat")) {
hdimage = new concat_image_t();
#ifdef WIN32
} else if (!strcmp(imgmode, "dll")) {
hdimage = new dll_image_t();
#endif
} else if (!strcmp(imgmode, "sparse")) {
hdimage = new sparse_image_t();
} else if (!strcmp(imgmode, "vmware3")) {
hdimage = new vmware3_image_t();
} else if (!strcmp(imgmode, "vmware4")) {
hdimage = new vmware4_image_t();
} else if (!strcmp(imgmode, "growing")) {
hdimage = new growing_image_t();
} else if (!strcmp(imgmode, "vpc")) {
hdimage = new vpc_image_t();
} else if (!strcmp(imgmode, "vbox")) {
hdimage = new vbox_image_t();
} else {
fatal("unsupported disk image mode");
}
return hdimage;
}
void create_flat_image(const char *filename, Bit64u size)
{
char buffer[512];
int fd = bx_create_image_file(filename);
if (fd < 0)
fatal("ERROR: failed to create flat image file");
memset(buffer, 0, 512);
if (bx_write_image(fd, size - 512, buffer, 512) != 512)
fatal("ERROR: while writing block in flat file !");
close(fd);
}
#ifdef WIN32
void create_flat_image_win32(const char *filename, Bit64u size)
{
HANDLE hFile;
LARGE_INTEGER pos;
DWORD dwCount, errCode;
USHORT mode;
char buffer[1024];
hFile = CreateFile(filename, GENERIC_WRITE|GENERIC_READ, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
if (hFile == INVALID_HANDLE_VALUE) {
// attempt to print an error
#ifdef HAVE_PERROR
sprintf(buffer, "while opening '%s' for writing", filename);
perror(buffer);
#endif
fatal("ERROR: Could not write disk image");
}
SetLastError(NO_ERROR);
mode = COMPRESSION_FORMAT_DEFAULT;
dwCount = 0;
memset(buffer, 0, 512);
WriteFile(hFile, buffer, 512, &dwCount, NULL); // set the first sector to 0, Win98 doesn't zero out the file
// if there is a write at/over the end
DeviceIoControl(hFile, FSCTL_SET_COMPRESSION, &mode, sizeof(mode), NULL, 0, &dwCount, NULL);
pos.u.LowPart = (unsigned long)((size - 512));
pos.u.HighPart = (unsigned long)((size - 512) >> 32);
pos.u.LowPart = SetFilePointer(hFile, pos.u.LowPart, &pos.u.HighPart, FILE_BEGIN);
memset(buffer, 0, 512);
if ((pos.u.LowPart == 0xffffffff && GetLastError() != NO_ERROR) ||
!WriteFile(hFile, buffer, 512, &dwCount, NULL) || (dwCount != 512)) {
errCode = GetLastError();
CloseHandle(hFile);
if (errCode == ERROR_DISK_FULL) {
fatal("\nERROR: Not enough space on disk for image!");
} else {
sprintf(buffer, "\nERROR: Disk image creation failed with error code %i!", errCode);
fatal(buffer);
}
}
CloseHandle(hFile);
}
#endif
device_image_t* create_hard_disk_image(const char *filename, const char *imgmode, Bit64u size)
{
device_image_t *hdimage = init_image(imgmode);
if(!strcmp(imgmode, "flat")) {
#ifndef WIN32
create_flat_image(filename, size);
#else
create_flat_image_win32(filename, size);
#endif
} else if(!strcmp(imgmode, "growing")) {
hdimage->create_image(filename, size);
} else if(!strcmp(imgmode, "sparse")) {
hdimage->create_image(filename, size);
} else if(!strcmp(imgmode, "vpc")) {
hdimage->create_image(filename, size);
} else if(!strcmp(imgmode, "vmware4")) {
hdimage->create_image(filename, size);
} else {
fatal("image mode not implemented yet");
}
return hdimage;
}
void convert_image(const char *newimgmode, Bit64u newsize)
{
device_image_t *source_image, *dest_image;
Bit64u i, sc, s;
char buffer[512], null_sector[512];
const char *imgmode = NULL;
bool error = false;
printf("\n");
memset(null_sector, 0, 512);
if (newsize == 0) {
if (!strncmp(bx_filename_1, "concat:", 7)) {
imgmode = "concat";
strcpy(bx_filename_1, &bx_filename_1[7]);
#ifdef WIN32
} else if (!strncmp(bx_filename_1, "dll:", 4)) {
imgmode = "dll";
strcpy(bx_filename_1, &bx_filename_1[4]);
#endif
}
}
if (access(bx_filename_1, F_OK) < 0) {
fatal("source disk image doesn't exist");
}
if (imgmode == NULL) {
hdimage_detect_image_mode(bx_filename_1, &imgmode);
}
if (imgmode == NULL) {
fatal("source disk image mode not detected");
} else {
printf("source image mode = '%s'\n", imgmode);
}
source_image = init_image(imgmode);
if (source_image->open(bx_filename_1, O_RDONLY) < 0)
fatal("cannot open source disk image");
if (newsize > 0) {
dest_image = create_hard_disk_image(bx_filename_2, newimgmode, newsize);
} else {
dest_image = create_hard_disk_image(bx_filename_2, newimgmode, source_image->hd_size);
}
if (dest_image->open(bx_filename_2) < 0)
fatal("cannot open destination disk image");
printf("\nConverting image file: [ 0%%]");
sc = source_image->hd_size / 512;
s = 0;
for (i = 0; i < source_image->hd_size; i+=512) {
printf("\x8\x8\x8\x8\x8%3d%%]", (int)((s+1)*100/sc));
fflush(stdout);
if (source_image->lseek(i, SEEK_SET) < 0) {
error = true;
break;
}
if ((source_image->read(buffer, 512) == 512) &&
(memcmp(buffer, null_sector, 512) != 0)) {
if (dest_image->lseek(i, SEEK_SET) < 0) {
error = true;
break;
}
if (dest_image->write(buffer, 512) < 0) {
error = true;
break;
}
}
s++;
}
source_image->close();
dest_image->close();
delete dest_image;
delete source_image;
if (error) {
fatal("image conversion failed");
} else {
printf(" Done.\n");
}
}
void commit_redolog()
{
device_image_t *base_image;
redolog_t *redolog;
int ret;
const char *imgmode = NULL;
printf("\n");
if (access(bx_filename_1, F_OK) < 0) {
fatal("base disk image doesn't exist");
}
if (!hdimage_detect_image_mode(bx_filename_1, &imgmode))
fatal("base disk image mode not detected");
base_image = init_image(imgmode);
if (base_image->open(bx_filename_1) < 0)
fatal("cannot open base disk image");
redolog = new redolog_t;
if (redolog->open(bx_filename_2, REDOLOG_SUBTYPE_UNDOABLE) < 0)
fatal("Can't open redolog");
if (!coherency_check(base_image, redolog))
fatal("coherency check failed");
ret = redolog->commit(base_image);
base_image->close();
redolog->close();
delete base_image;
delete redolog;
if (ret < 0) {
fatal("redolog commit failed");
} else {
printf(" Done.\n\n");
}
}
void print_usage()
{
fprintf(stderr,
"Usage: bximage [options] [filename1] [filename2]\n\n"
"Supported options:\n"
" -func=... operation to perform (create, convert, resize, commit, info)\n"
" -fd=... create: floppy image with size code\n"
" -hd=... create/resize: hard disk image with size in megabytes (M)\n"
" or gigabytes (G)\n"
" -imgmode=... create/convert: hard disk image mode (default = flat)\n"
" -sectsize=... create: hard disk sector size (default = 512)\n"
" -b convert/resize: create a backup of the source image\n"
" commit: create backups of the base image and redolog file\n"
" -q quiet mode (don't prompt for user input)\n"
" --help display this help and exit\n\n"
"Other arguments:\n"
" filename1 create: new image file\n"
" convert/resize: source image file\n"
" commit: base image file\n"
" filename2 convert/resize: destination image file\n"
" commit: redolog (journal) file\n\n");
}
void set_default_values()
{
if (bximage_func == BXIMAGE_FUNC_CREATE_IMAGE) {
if (bx_hdimage == -1) {
bx_hdimage = 1;
bx_fdsize_idx = 6;
bx_interactive = 1;
}
if (bx_hdimage == 1) {
if (bx_imagemode == -1) {
bx_imagemode = 0;
bx_interactive = 1;
}
if (bx_hdsize == 0) {
bx_hdsize = 10;
bx_interactive = 1;
}
} else {
if (bx_fdsize_idx == -1) {
bx_fdsize_idx = 6;
bx_interactive = 1;
}
}
} else if (bximage_func == BXIMAGE_FUNC_CONVERT_IMAGE) {
if (bx_imagemode == -1) {
bx_imagemode = 0;
bx_interactive = 1;
}
} else if (bximage_func == BXIMAGE_FUNC_RESIZE_IMAGE) {
if (bx_hdsize == 0) {
bx_interactive = 1;
}
}
}
int parse_cmdline(int argc, char *argv[])
{
int arg = 1;
int ret = 1;
int fnargs = 0;
char *suffix;
bximage_func = BXIMAGE_FUNC_NULL;
bx_hdimage = -1;
bx_fdsize_idx = -1;
bx_hdsize = 0;
bx_imagemode = -1;
bx_backup = 0;
bx_interactive = 1;
bx_sectsize_idx = 0;
bx_sectsize_val = 512;
bx_filename_1[0] = 0;
bx_filename_2[0] = 0;
while ((arg < argc) && (ret == 1)) {
// parse next arg
if (!strcmp("--help", argv[arg]) || !strncmp("/?", argv[arg], 2)) {
print_usage();
ret = 0;
}
else if (!strncmp("-func=", argv[arg], 6)) {
if (bximage_func != BXIMAGE_FUNC_NULL) {
printf("bximage mode already defined\n\n");
ret = 0;
} else if (!strcmp(&argv[arg][6], "create")) {
bximage_func = BXIMAGE_FUNC_CREATE_IMAGE;
} else if (!strcmp(&argv[arg][6], "convert")) {
bximage_func = BXIMAGE_FUNC_CONVERT_IMAGE;
} else if (!strcmp(&argv[arg][6], "resize")) {
bximage_func = BXIMAGE_FUNC_RESIZE_IMAGE;
} else if (!strcmp(&argv[arg][6], "commit")) {
bximage_func = BXIMAGE_FUNC_COMMIT_UNDOABLE;
} else if (!strcmp(&argv[arg][6], "info")) {
bximage_func = BXIMAGE_FUNC_IMAGE_INFO;
} else {
printf("Unknown bximage mode '%s'\n\n", &argv[arg][6]);
ret = 0;
}
}
else if (!strncmp("-fd=", argv[arg], 4)) {
bx_hdimage = 0;
bx_fdsize_idx = get_menu_index(&argv[arg][4], fdsize_n_choices, fdsize_choices);
if (bx_fdsize_idx < 0) {
printf("Unknown floppy image size: %s\n\n", &argv[arg][6]);
ret = 0;
}
}
else if (!strncmp("-hd=", argv[arg], 4)) {
bx_hdimage = 1;
bx_imagemode = 0;
bx_hdsize = (int)strtol(&argv[arg][4], &suffix, 10);
if (bx_hdsize <= 0) {
printf("Error in hard disk image size argument: %s\n\n", &argv[arg][4]);
ret = 0;
} else if (!strcmp(suffix, "G")) {
bx_hdsize <<= 10;
} else if ((strlen(suffix) > 0) && strcmp(suffix, "M")) {
printf("Error in hard disk image size suffix: %s\n\n", &argv[arg][4]);
ret = 0;
}
if ((bx_hdsize < bx_min_hd_megs) || (bx_hdsize > bx_max_hd_megs)) {
printf("Hard disk image size out of range\n\n");
ret = 0;
}
}
else if (!strncmp("-imgmode=", argv[arg], 9)) {
bx_imagemode = get_menu_index(&argv[arg][9], hdmode_n_choices, hdmode_choices);
if (bx_imagemode < 0) {
printf("Unknown image mode: %s\n\n", &argv[arg][9]);
ret = 0;
}
}
else if (!strncmp("-sectsize=", argv[arg], 10)) {
bx_sectsize_idx = get_menu_index(&argv[arg][10], sectsize_n_choices, sectsize_choices);
if (bx_sectsize_idx < 0) {
printf("Unsupported hard disk sector size: %s\n\n", &argv[arg][10]);
ret = 0;
} else {
bx_sectsize_val = atoi(sectsize_choices[bx_sectsize_idx]);
}
}
else if (!strcmp("-b", argv[arg])) {
bx_backup = 1;
}
else if (!strcmp("-q", argv[arg])) {
bx_interactive = 0;
}
else if (argv[arg][0] == '-') {
printf("Unknown option: %s\n\n", argv[arg]);
ret = 0;
} else {
if (fnargs == 0) {
strcpy(bx_filename_1, argv[arg]);
} else if (fnargs == 1) {
strcpy(bx_filename_2, argv[arg]);
} else {
printf("Ignoring extra parameter: %s\n\n", argv[arg]);
}
fnargs++;
}
arg++;
}
if (bximage_func == BXIMAGE_FUNC_NULL) {
printf("\nERROR: Parameter -func missing - switching to interactive mode.\n\n");
bx_interactive = 1;
} else {
set_default_values();
if (fnargs < 1) {
bx_interactive = 1;
printf("\nERROR; Filename missing - switching to interactive mode.\n\n");
}
if ((bximage_func == BXIMAGE_FUNC_COMMIT_UNDOABLE) && (fnargs == 1)) {
snprintf(bx_filename_2, 520, "%s%s", bx_filename_1, UNDOABLE_REDOLOG_EXTENSION);
}
}
return ret;
}
void image_overwrite_check(const char *filename)
{
char buffer[512];
int fd = hdimage_open_file(filename, O_RDONLY, NULL, NULL);
if (fd >= 0) {
close(fd);
int confirm;
sprintf(buffer, "\nThe disk image '%s' already exists. Are you sure you want to replace it?\nPlease type yes or no. ", filename);
if (ask_yn(buffer, 0, &confirm) < 0)
fatal(EOF_ERR);
if (!confirm)
fatal("ERROR: Aborted");
}
}
void check_image_names()
{
char backup_fname[520];
if (!strlen(bx_filename_2)) {
strcpy(bx_filename_2, bx_filename_1);
}
if ((!strcmp(bx_filename_1, bx_filename_2)) && bx_backup) {
snprintf(backup_fname, 517, "%s%s", bx_filename_1, ".orig");
if (hdimage_copy_file(bx_filename_1, backup_fname) != 1)
fatal("backup of source image file failed");
} else if (strcmp(bx_filename_1, bx_filename_2)) {
image_overwrite_check(bx_filename_2);
}
}
int get_image_mode_and_hdsize(const char *filename, const char **image_mode)
{
device_image_t *source_image;
int hdsize_megs = 0;
if (!hdimage_detect_image_mode(bx_filename_1, image_mode))
fatal("source disk image mode not detected");
source_image = init_image(*image_mode);
if (source_image->open(bx_filename_1, O_RDONLY) < 0) {
fatal("cannot open source disk image");
} else {
hdsize_megs = (int)(source_image->hd_size >> 20);
source_image->close();
}
return hdsize_megs;
}
void print_partitiontable(device_image_t *hdimage)
{
size_t ret = 0;
unsigned int n = 0;
Bit8u data[SECTOR_SIZE];
ret = hdimage->lseek(0, SEEK_SET);
if (ret < 0) {
fatal("lseek() to MBR failed");
}
ret = hdimage->read(data, SECTOR_SIZE);
if (ret != SECTOR_SIZE) {
fatal("failed to read MBR");
}
// check bootsector ID, not present = no partition table present
if ((data[0x1FE] != 0x55) && (data[0x1FE] != 0xAA)) {
printf("No partition-table detected.\n\n");
return;
}
printf("Partition table\n");
printf(" boot\t sector start\tsize\ttype\n");
for (n = 0; n < 4; n++) {
// calculate position of partition record (n) in table
Bit8u *p = &data[0x1BE + (16 * n)];
Bit8u bootable = p[0x00]; // 0x80 = bootable, 0x00 = not bootable
Bit8u type = p[0x04];
Bit32u start = *((Bit32u *)&p[0x08]);
Bit32u size = *((Bit32u *)&p[0x0C]);
if (type > 0) {
// print partition number
printf("%d. ", (n + 1));
// print '*' if partition is bootable
if (bootable) printf("*\t "); else printf("\t ");
// print start of partition in sectors
printf("%12u", start);
// print size of partition in sectors
size = (size >> 1) / 1024;
if (size > (4 * 1024))
printf("%6uG\t", (size / 1024));
else
printf("%6uM\t", size);
printf("%02X\n", type);
}
}
printf("\n");
}
int CDECL main(int argc, char *argv[])
{
char bochsrc_line[1024], prompt[80], tmpfname[528];
Bit64u hdsize = 0;
device_image_t *hdimage;
const char *image_mode = NULL;
if (!parse_cmdline(argc, argv))
myexit(1);
print_banner();
if (bx_interactive) {
if (ask_int(main_menu_prompt, 0, 5, bximage_func, &bximage_func) < 0)
fatal(EOF_ERR);
set_default_values();
switch (bximage_func) {
case BXIMAGE_FUNC_NULL:
myexit(0);
break;
case BXIMAGE_FUNC_CREATE_IMAGE:
printf("\nCreate image\n");
if (ask_menu(fdhd_menu, fdhd_n_choices, fdhd_choices, bx_hdimage, &bx_hdimage) < 0)
fatal(EOF_ERR);
if (bx_hdimage == 0) { // floppy
if (ask_menu(fdsize_menu, fdsize_n_choices, fdsize_choices, bx_fdsize_idx, &bx_fdsize_idx) < 0)
fatal(EOF_ERR);
if (!strlen(bx_filename_1)) {
strcpy(bx_filename_1, "a.img");
}
if (ask_string("\nWhat should be the name of the image?\n", bx_filename_1, bx_filename_1) < 0)
fatal(EOF_ERR);
} else { // hard disk
if (ask_menu(hdmode_menu, hdmode_n_choices, hdmode_choices, bx_imagemode, &bx_imagemode) < 0)
fatal(EOF_ERR);
image_mode = hdmode_choices[bx_imagemode];
if (!strcmp(image_mode, "flat") ||
!strcmp(image_mode, "sparse") ||
!strcmp(image_mode, "growing")) {
if (ask_menu(sectsize_menu, sectsize_n_choices, sectsize_choices, bx_sectsize_idx, &bx_sectsize_idx) < 0)
fatal(EOF_ERR);
bx_sectsize_val = atoi(sectsize_choices[bx_sectsize_idx]);
} else if (bx_sectsize_val != 512) {
fatal(EOF_ERR);
}
sprintf(prompt, "\nEnter the hard disk size in megabytes, between %d and %d\n",
bx_min_hd_megs, bx_max_hd_megs);
if (ask_int(prompt, bx_min_hd_megs, bx_max_hd_megs, bx_hdsize, &bx_hdsize) < 0)
fatal(EOF_ERR);
if (!strlen(bx_filename_1)) {
strcpy(bx_filename_1, "c.img");
}
if (ask_string("\nWhat should be the name of the image?\n", bx_filename_1, bx_filename_1) < 0)
fatal(EOF_ERR);
}
break;
case BXIMAGE_FUNC_CONVERT_IMAGE:
printf("\nConvert image\n");
if (!strlen(bx_filename_1)) {
strcpy(bx_filename_1, "c.img");
}
if (ask_string("\nWhat is the name of the source image?\n", bx_filename_1, bx_filename_1) < 0)
fatal(EOF_ERR);
if (!strlen(bx_filename_2)) {
strcpy(tmpfname, bx_filename_1);
} else {
strcpy(tmpfname, bx_filename_2);
}
if (ask_string("\nWhat should be the name of the new image?\n", tmpfname, bx_filename_2) < 0)
fatal(EOF_ERR);
if (ask_menu(hdmode_menu, hdmode_n_choices, hdmode_choices, bx_imagemode, &bx_imagemode) < 0)
fatal(EOF_ERR);
if (!strcmp(bx_filename_1, bx_filename_2)) {
if (ask_yn("\nShould I create a backup of the source image\n", 0, &bx_backup) < 0)
fatal(EOF_ERR);
}
break;
case BXIMAGE_FUNC_RESIZE_IMAGE:
printf("\nResize image\n");
if (!strlen(bx_filename_1)) {
strcpy(bx_filename_1, "c.img");
}
if (ask_string("\nWhat is the name of the source image?\n", bx_filename_1, bx_filename_1) < 0)
fatal(EOF_ERR);
if (!strlen(bx_filename_2)) {
strcpy(tmpfname, bx_filename_1);
} else {
strcpy(tmpfname, bx_filename_2);
}
if (ask_string("\nWhat should be the name of the new image?\n", tmpfname, bx_filename_2) < 0)
fatal(EOF_ERR);
bx_min_hd_megs = get_image_mode_and_hdsize(bx_filename_1, &image_mode);
sprintf(prompt, "\nEnter the new hard disk size in megabytes, between %d and %d\n",
bx_min_hd_megs, bx_max_hd_megs);
if (bx_hdsize < bx_min_hd_megs) bx_hdsize = bx_min_hd_megs;
if (ask_int(prompt, bx_min_hd_megs, bx_max_hd_megs, bx_hdsize, &bx_hdsize) < 0)
fatal(EOF_ERR);
if (!strcmp(bx_filename_1, bx_filename_2)) {
if (ask_yn("\nShould I create a backup of the source image\n", 0, &bx_backup) < 0)
fatal(EOF_ERR);
}
break;
case BXIMAGE_FUNC_COMMIT_UNDOABLE:
printf("\nCommit redolog\n");
if (!strlen(bx_filename_1)) {
strcpy(bx_filename_1, "c.img");
}
if (ask_string("\nWhat is the name of the base image?\n", bx_filename_1, bx_filename_1) < 0)
fatal(EOF_ERR);
if (!strlen(bx_filename_2)) {
snprintf(tmpfname, 520, "%s%s", bx_filename_1, UNDOABLE_REDOLOG_EXTENSION);
} else {
strcpy(tmpfname, bx_filename_2);
}
if (ask_string("\nWhat is the redolog name?\n", tmpfname, bx_filename_2) < 0)
fatal(EOF_ERR);
if (ask_yn("\nShould I create a backup of base image and redolog\n", 0, &bx_backup) < 0)
fatal(EOF_ERR);
break;
case BXIMAGE_FUNC_IMAGE_INFO:
printf("\nDisk image info\n");
if (!strlen(bx_filename_1)) {
strcpy(bx_filename_1, "c.img");
}
if (ask_string("\nWhat is the name of the image?\n", bx_filename_1, bx_filename_1) < 0)
fatal(EOF_ERR);
break;
default:
fatal("\nbximage: unknown mode");
}
}
switch (bximage_func) {
case BXIMAGE_FUNC_CREATE_IMAGE:
image_overwrite_check(bx_filename_1);
if (bx_hdimage == 0) {
sprintf(bochsrc_line, "floppya: image=\"%s\", status=inserted", bx_filename_1);
printf("\nCreating floppy image '%s' with %u sectors\n", bx_filename_1, fdsize_sectors[bx_fdsize_idx]);
create_flat_image(bx_filename_1, fdsize_sectors[bx_fdsize_idx] * 512);
} else {
int heads = 16, spt = 63;
if (bx_sectsize_val == 512) {
sprintf(bochsrc_line, "ata0-master: type=disk, path=\"%s\", mode=%s",
bx_filename_1, hdmode_choices[bx_imagemode]);
} else {
sprintf(bochsrc_line, "ata0-master: type=disk, path=\"%s\", mode=%s, sect_size=%d",
bx_filename_1, hdmode_choices[bx_imagemode], bx_sectsize_val);
}
image_mode = hdmode_choices[bx_imagemode];
hdsize = ((Bit64u)bx_hdsize) << 20;
Bit64u cyl = (Bit64u)(hdsize/16.0/63.0/bx_sectsize_val);
if (cyl >= (1 << BX_MAX_CYL_BITS))
fatal("ERROR: number of cylinders out of range !\n");
printf("\nCreating hard disk image '%s' with CHS=" FMT_LL "d/%d/%d (sector size = %d)\n",
bx_filename_1, cyl, heads, spt, bx_sectsize_val);
hdsize = cyl * heads * spt * bx_sectsize_val;
hdimage = create_hard_disk_image(bx_filename_1, image_mode, hdsize);
delete hdimage;
}
printf("\nThe following line should appear in your bochsrc:\n");
printf(" %s\n", bochsrc_line);
#ifdef WIN32
if (OpenClipboard(NULL)) {
HGLOBAL hgClip;
EmptyClipboard();
hgClip = GlobalAlloc(GMEM_DDESHARE, (strlen(bochsrc_line) + 1));
strcpy((char *)GlobalLock(hgClip), bochsrc_line);
GlobalUnlock(hgClip);
SetClipboardData(CF_TEXT, hgClip);
CloseClipboard();
printf("(The line is stored in your windows clipboard, use CTRL-V to paste)\n");
}
#endif
break;
case BXIMAGE_FUNC_CONVERT_IMAGE:
check_image_names();
image_mode = hdmode_choices[bx_imagemode];
convert_image(image_mode, 0);
break;
case BXIMAGE_FUNC_RESIZE_IMAGE:
bx_min_hd_megs = get_image_mode_and_hdsize(bx_filename_1, &image_mode);
if (bx_hdsize < bx_min_hd_megs)
fatal("invalid image size");
hdsize = ((Bit64u)bx_hdsize) << 20;
check_image_names();
convert_image(image_mode, hdsize);
break;
case BXIMAGE_FUNC_COMMIT_UNDOABLE:
if (bx_backup) {
snprintf(tmpfname, 517, "%s%s", bx_filename_1, ".orig");
if (hdimage_copy_file(bx_filename_1, tmpfname) != 1)
fatal("backup of base image file failed");
}
commit_redolog();
if (bx_backup) {
snprintf(tmpfname, 527, "%s%s", bx_filename_2, ".orig");
if (rename(bx_filename_2, tmpfname) != 0)
fatal("rename of redolog file failed");
} else {
if (unlink(bx_filename_2) != 0)
fatal("ERROR: while removing the redolog !\n");
}
break;
case BXIMAGE_FUNC_IMAGE_INFO:
if (!hdimage_detect_image_mode(bx_filename_1, &image_mode)) {
fatal("disk image mode not detected");
} else {
printf("\ndisk image mode = '%s'\n", image_mode);
}
hdimage = init_image(image_mode);
if (hdimage->open(bx_filename_1, O_RDONLY) < 0) {
fatal("cannot open source disk image");
} else {
if (hdimage->get_capabilities() & HDIMAGE_AUTO_GEOMETRY) {
Bit64u cyl = (Bit64u)(hdimage->hd_size/16.0/63.0/512.0);
printf("geometry = " FMT_LL "d/16/63 (" FMT_LL "d MB)\n\n", cyl,
hdimage->hd_size >> 20);
} else {
printf("geometry = %d/%d/%d (" FMT_LL "d MB)\n\n", hdimage->cylinders,
hdimage->heads, hdimage->spt, hdimage->hd_size >> 20);
}
print_partitiontable(hdimage);
hdimage->close();
}
break;
}
myexit(0);
return 0;
}