e4978addd1
The code is based on parts of the SF patch "bxmount".
1073 lines
33 KiB
C++
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;
|
|
}
|