2013-10-14 00:30:51 +04:00
/////////////////////////////////////////////////////////////////////////
// $Id$
/////////////////////////////////////////////////////////////////////////
//
2023-12-10 13:17:21 +03:00
// Copyright (C) 2001-2023 The Bochs Project
2013-10-14 00:30:51 +04: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
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
/////////////////////////////////////////////////////////////////////////
2013-10-18 23:36:50 +04:00
// Create empty hard disk or floppy disk images for bochs.
2013-10-14 00:30:51 +04:00
// Convert a hard disk image from one format (mode) to another.
2013-10-18 23:36:50 +04:00
// Resize a hard disk image.
2013-10-14 00:30:51 +04:00
// Commit a redolog file to a supported base image.
2013-10-29 00:31:34 +04:00
# ifndef BXIMAGE
# define BXIMAGE
# endif
2013-10-17 20:00:26 +04:00
# include "config.h"
# include "bxcompat.h"
2013-11-01 22:19:52 +04:00
# if defined(WIN32) && !defined(__CYGWIN__)
2013-10-14 00:30:51 +04:00
# include <conio.h>
2013-10-16 23:33:30 +04:00
# include <winioctl.h>
2013-10-14 00:30:51 +04:00
# endif
# include <ctype.h>
2013-10-22 23:26:42 +04:00
# include "osdep.h"
# include "bswap.h"
2013-10-14 00:30:51 +04:00
# include "iodev/hdimage/hdimage.h"
# include "iodev/hdimage/vmware3.h"
# include "iodev/hdimage/vmware4.h"
2020-12-30 20:56:27 +03:00
# include "iodev/hdimage/vpc.h"
2021-01-08 16:19:08 +03:00
# include "iodev/hdimage/vbox.h"
2013-10-14 00:30:51 +04:00
2021-01-09 19:51:52 +03:00
# 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
2013-10-14 00:30:51 +04:00
2013-10-15 00:40:41 +04:00
# define BX_MAX_CYL_BITS 24 // 8 TB
const int bx_max_hd_megs = ( int ) ( ( ( 1 < < BX_MAX_CYL_BITS ) - 1 ) * 16.0 * 63.0 / 2048.0 ) ;
2021-01-09 19:51:52 +03:00
int bximage_func ;
2013-10-15 00:40:41 +04:00
int bx_hdimage ;
int bx_fdsize_idx ;
2013-10-18 23:36:50 +04:00
int bx_min_hd_megs = 10 ;
2013-10-15 00:40:41 +04:00
int bx_hdsize ;
int bx_imagemode ;
2013-11-02 19:58:59 +04:00
int bx_backup ;
2013-10-14 00:30:51 +04:00
int bx_interactive ;
2018-03-31 00:04:04 +03:00
int bx_sectsize_idx ;
2020-09-01 23:13:56 +03:00
Bit16u bx_sectsize_val ;
2013-10-14 00:30:51 +04:00
char bx_filename_1 [ 512 ] ;
2020-09-01 23:13:56 +03:00
char bx_filename_2 [ 522 ] ;
2013-10-14 00:30:51 +04:00
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 "
2013-10-18 23:36:50 +04:00
" 3. Resize hard disk image \n "
" 4. Commit 'undoable' redolog to base image \n "
2013-11-03 20:13:35 +04:00
" 5. Disk image info \n "
2013-10-14 00:30:51 +04:00
" \n "
" 0. Quit \n "
" \n "
" Please choose one " ;
2013-10-15 00:40:41 +04:00
// menu data for choosing floppy/hard disk
const char * fdhd_menu = " \n Do you want to create a floppy disk image or a hard disk image? \n Please type hd or fd. " ;
const char * fdhd_choices [ ] = { " fd " , " hd " } ;
int fdhd_n_choices = 2 ;
// menu data for choosing floppy size
2017-02-12 19:51:52 +03:00
const char * fdsize_menu = " \n Choose the size of floppy disk image to create. \n Please type 160k, 180k, 320k, 360k, 720k, 1.2M, 1.44M, 1.68M, 1.72M, or 2.88M. \n " ;
2013-10-15 00:40:41 +04:00
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 ;
2013-10-14 00:30:51 +04:00
// menu data for choosing disk mode
2013-11-03 11:41:29 +04:00
const char * hdmode_menu = " \n What kind of image should I create? \n Please type flat, sparse, growing, vpc or vmware4. " ;
const char * hdmode_choices [ ] = { " flat " , " sparse " , " growing " , " vpc " , " vmware4 " } ;
int hdmode_n_choices = 5 ;
2013-10-14 00:30:51 +04:00
2018-03-31 00:04:04 +03:00
// menu data for choosing hard disk sector size
const char * sectsize_menu = " \n Choose the size of hard disk sectors. \n Please type 512, 1024 or 4096. " ;
const char * sectsize_choices [ ] = { " 512 " , " 1024 " , " 4096 " } ;
int sectsize_n_choices = 3 ;
2013-10-14 00:30:51 +04:00
# 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 )
{
2013-11-01 22:19:52 +04:00
# if defined(WIN32) && !defined(__CYGWIN__)
2013-10-14 00:30:51 +04:00
printf ( " \n Press 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 ) ;
2013-10-21 22:26:00 +04:00
bx_center_print ( stdout , " bximage \n " , 72 ) ;
bx_center_print ( stdout , " Disk Image Creation / Conversion / Resize and Commit Tool for Bochs \n " , 72 ) ;
2013-10-14 00:30:51 +04:00
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 ) ;
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 ] ) ;
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 " ) ;
}
}
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 " ) ;
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 ) ;
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 ;
}
2021-01-09 19:51:52 +03:00
device_image_t * init_image ( const char * imgmode )
2013-10-14 00:30:51 +04:00
{
device_image_t * hdimage = NULL ;
// instantiate the right class
2021-01-09 19:51:52 +03:00
if ( ! strcmp ( imgmode , " flat " ) ) {
hdimage = new flat_image_t ( ) ;
} else if ( ! strcmp ( imgmode , " concat " ) ) {
hdimage = new concat_image_t ( ) ;
2013-10-25 23:10:09 +04:00
# ifdef WIN32
2021-01-09 19:51:52 +03:00
} else if ( ! strcmp ( imgmode , " dll " ) ) {
hdimage = new dll_image_t ( ) ;
2013-10-25 23:10:09 +04:00
# endif
2021-01-09 19:51:52 +03:00
} 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 " ) ;
2013-10-14 00:30:51 +04:00
}
return hdimage ;
}
2013-10-15 22:00:39 +04:00
void create_flat_image ( const char * filename , Bit64u size )
2013-10-14 00:30:51 +04:00
{
char buffer [ 512 ] ;
2021-01-16 15:47:02 +03:00
int fd = bx_create_image_file ( filename ) ;
if ( fd < 0 )
fatal ( " ERROR: failed to create flat image file " ) ;
2013-10-14 00:30:51 +04:00
memset ( buffer , 0 , 512 ) ;
2013-10-15 00:40:41 +04:00
if ( bx_write_image ( fd , size - 512 , buffer , 512 ) ! = 512 )
2013-10-14 00:30:51 +04:00
fatal ( " ERROR: while writing block in flat file ! " ) ;
close ( fd ) ;
2013-10-15 22:00:39 +04:00
}
2013-10-14 00:30:51 +04:00
2013-10-15 22:00:39 +04:00
# ifdef WIN32
void create_flat_image_win32 ( const char * filename , Bit64u size )
{
2013-10-16 22:55:43 +04:00
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 ( " \n ERROR: Not enough space on disk for image! " ) ;
} else {
sprintf ( buffer , " \n ERROR: Disk image creation failed with error code %i! " , errCode ) ;
fatal ( buffer ) ;
}
}
CloseHandle ( hFile ) ;
2013-10-14 00:30:51 +04:00
}
2013-10-15 22:00:39 +04:00
# endif
2013-10-14 00:30:51 +04:00
2021-01-16 15:47:02 +03:00
device_image_t * create_hard_disk_image ( const char * filename , const char * imgmode , Bit64u size )
2013-10-15 22:00:39 +04:00
{
2021-01-16 15:47:02 +03:00
device_image_t * hdimage = init_image ( imgmode ) ;
2021-01-09 19:51:52 +03:00
if ( ! strcmp ( imgmode , " flat " ) ) {
2013-10-16 22:55:43 +04:00
# ifndef WIN32
2021-01-09 19:51:52 +03:00
create_flat_image ( filename , size ) ;
2013-10-16 22:55:43 +04:00
# else
2021-01-09 19:51:52 +03:00
create_flat_image_win32 ( filename , size ) ;
2013-10-16 22:55:43 +04:00
# endif
2021-01-09 19:51:52 +03:00
} else if ( ! strcmp ( imgmode , " growing " ) ) {
2021-01-16 15:47:02 +03:00
hdimage - > create_image ( filename , size ) ;
2021-01-09 19:51:52 +03:00
} else if ( ! strcmp ( imgmode , " sparse " ) ) {
2021-01-16 15:47:02 +03:00
hdimage - > create_image ( filename , size ) ;
2021-01-09 19:51:52 +03:00
} else if ( ! strcmp ( imgmode , " vpc " ) ) {
2021-01-16 15:47:02 +03:00
hdimage - > create_image ( filename , size ) ;
2021-01-09 19:51:52 +03:00
} else if ( ! strcmp ( imgmode , " vmware4 " ) ) {
2021-01-16 15:47:02 +03:00
hdimage - > create_image ( filename , size ) ;
2021-01-09 19:51:52 +03:00
} else {
fatal ( " image mode not implemented yet " ) ;
2013-10-15 22:00:39 +04:00
}
2021-01-16 15:47:02 +03:00
return hdimage ;
2013-10-14 00:30:51 +04:00
}
2021-01-09 19:51:52 +03:00
void convert_image ( const char * newimgmode , Bit64u newsize )
2013-10-14 00:30:51 +04:00
{
device_image_t * source_image , * dest_image ;
Bit64u i , sc , s ;
char buffer [ 512 ] , null_sector [ 512 ] ;
2021-01-09 19:51:52 +03:00
const char * imgmode = NULL ;
2021-01-30 20:37:42 +03:00
bool error = false ;
2013-10-14 00:30:51 +04:00
printf ( " \n " ) ;
memset ( null_sector , 0 , 512 ) ;
2013-10-25 23:10:09 +04:00
if ( newsize = = 0 ) {
if ( ! strncmp ( bx_filename_1 , " concat: " , 7 ) ) {
2021-01-09 19:51:52 +03:00
imgmode = " concat " ;
2013-10-25 23:10:09 +04:00
strcpy ( bx_filename_1 , & bx_filename_1 [ 7 ] ) ;
# ifdef WIN32
} else if ( ! strncmp ( bx_filename_1 , " dll: " , 4 ) ) {
2021-01-09 19:51:52 +03:00
imgmode = " dll " ;
2013-10-25 23:10:09 +04:00
strcpy ( bx_filename_1 , & bx_filename_1 [ 4 ] ) ;
# endif
}
}
2014-03-23 13:19:44 +04:00
if ( access ( bx_filename_1 , F_OK ) < 0 ) {
fatal ( " source disk image doesn't exist " ) ;
}
2021-01-09 19:51:52 +03:00
if ( imgmode = = NULL ) {
hdimage_detect_image_mode ( bx_filename_1 , & imgmode ) ;
2013-10-25 23:10:09 +04:00
}
2021-01-09 19:51:52 +03:00
if ( imgmode = = NULL ) {
2013-10-18 23:36:50 +04:00
fatal ( " source disk image mode not detected " ) ;
2013-10-27 23:23:59 +04:00
} else {
2021-01-09 19:51:52 +03:00
printf ( " source image mode = '%s' \n " , imgmode ) ;
2013-10-27 23:23:59 +04:00
}
2013-10-14 00:30:51 +04:00
2021-01-09 19:51:52 +03:00
source_image = init_image ( imgmode ) ;
2013-10-14 00:30:51 +04:00
if ( source_image - > open ( bx_filename_1 , O_RDONLY ) < 0 )
fatal ( " cannot open source disk image " ) ;
2013-10-18 23:36:50 +04:00
if ( newsize > 0 ) {
2021-01-16 15:47:02 +03:00
dest_image = create_hard_disk_image ( bx_filename_2 , newimgmode , newsize ) ;
2013-10-18 23:36:50 +04:00
} else {
2021-01-16 15:47:02 +03:00
dest_image = create_hard_disk_image ( bx_filename_2 , newimgmode , source_image - > hd_size ) ;
2013-10-18 23:36:50 +04:00
}
2013-10-14 00:30:51 +04:00
if ( dest_image - > open ( bx_filename_2 ) < 0 )
fatal ( " cannot open destination disk image " ) ;
printf ( " \n Converting 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 ) {
2021-01-30 20:37:42 +03:00
error = true ;
2013-10-14 00:30:51 +04:00
break ;
}
if ( ( source_image - > read ( buffer , 512 ) = = 512 ) & &
( memcmp ( buffer , null_sector , 512 ) ! = 0 ) ) {
if ( dest_image - > lseek ( i , SEEK_SET ) < 0 ) {
2021-01-30 20:37:42 +03:00
error = true ;
2013-10-14 00:30:51 +04:00
break ;
}
if ( dest_image - > write ( buffer , 512 ) < 0 ) {
2021-01-30 20:37:42 +03:00
error = true ;
2013-10-14 00:30:51 +04:00
break ;
}
}
s + + ;
}
source_image - > close ( ) ;
dest_image - > close ( ) ;
delete dest_image ;
delete source_image ;
if ( error ) {
fatal ( " image conversion failed " ) ;
} else {
printf ( " Done. \n " ) ;
}
}
2013-10-15 22:00:39 +04:00
void commit_redolog ( )
2013-10-14 00:30:51 +04:00
{
device_image_t * base_image ;
redolog_t * redolog ;
2021-01-09 19:51:52 +03:00
int ret ;
const char * imgmode = NULL ;
2013-10-14 00:30:51 +04:00
printf ( " \n " ) ;
2014-03-23 13:19:44 +04:00
if ( access ( bx_filename_1 , F_OK ) < 0 ) {
fatal ( " base disk image doesn't exist " ) ;
}
2021-01-09 19:51:52 +03:00
if ( ! hdimage_detect_image_mode ( bx_filename_1 , & imgmode ) )
2013-10-14 00:30:51 +04:00
fatal ( " base disk image mode not detected " ) ;
2021-01-09 19:51:52 +03:00
base_image = init_image ( imgmode ) ;
2013-10-14 00:30:51 +04:00
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 " ) ;
2013-10-23 12:35:21 +04:00
ret = redolog - > commit ( base_image ) ;
2013-10-14 00:30:51 +04:00
base_image - > close ( ) ;
redolog - > close ( ) ;
delete base_image ;
delete redolog ;
2013-10-23 12:35:21 +04:00
if ( ret < 0 ) {
2013-10-14 00:30:51 +04:00
fatal ( " redolog commit failed " ) ;
} else {
2013-10-23 12:35:21 +04:00
printf ( " Done. \n \n " ) ;
2013-10-14 00:30:51 +04:00
}
}
void print_usage ( )
{
fprintf ( stderr ,
2013-10-21 22:26:00 +04:00
" Usage: bximage [options] [filename1] [filename2] \n \n "
2013-10-14 00:30:51 +04:00
" Supported options: \n "
2021-01-09 19:51:52 +03:00
" -func=... operation to perform (create, convert, resize, commit, info) \n "
2013-10-15 00:40:41 +04:00
" -fd=... create: floppy image with size code \n "
2013-10-18 23:36:50 +04:00
" -hd=... create/resize: hard disk image with size in megabytes (M) \n "
" or gigabytes (G) \n "
2023-12-10 13:17:21 +03:00
" -imgmode=... create/convert: hard disk image mode (default = flat) \n "
" -sectsize=... create: hard disk sector size (default = 512) \n "
2013-11-02 19:58:59 +04:00
" -b convert/resize: create a backup of the source image \n "
" commit: create backups of the base image and redolog file \n "
2013-10-14 00:30:51 +04:00
" -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 "
2013-10-18 23:36:50 +04:00
" convert/resize: source image file \n "
2013-10-14 00:30:51 +04:00
" commit: base image file \n "
2013-10-18 23:36:50 +04:00
" filename2 convert/resize: destination image file \n "
2013-10-14 00:30:51 +04:00
" commit: redolog (journal) file \n \n " ) ;
}
2013-10-15 00:40:41 +04:00
void set_default_values ( )
{
2021-01-09 19:51:52 +03:00
if ( bximage_func = = BXIMAGE_FUNC_CREATE_IMAGE ) {
2013-10-15 00:40:41 +04:00
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 ;
}
}
2021-01-09 19:51:52 +03:00
} else if ( bximage_func = = BXIMAGE_FUNC_CONVERT_IMAGE ) {
2013-10-15 00:40:41 +04:00
if ( bx_imagemode = = - 1 ) {
bx_imagemode = 0 ;
bx_interactive = 1 ;
}
2021-01-09 19:51:52 +03:00
} else if ( bximage_func = = BXIMAGE_FUNC_RESIZE_IMAGE ) {
2013-10-22 23:26:42 +04:00
if ( bx_hdsize = = 0 ) {
bx_interactive = 1 ;
}
2013-10-15 00:40:41 +04:00
}
}
2013-10-14 00:30:51 +04:00
int parse_cmdline ( int argc , char * argv [ ] )
{
int arg = 1 ;
int ret = 1 ;
int fnargs = 0 ;
2013-10-17 20:00:26 +04:00
char * suffix ;
2013-10-14 00:30:51 +04:00
2021-01-09 19:51:52 +03:00
bximage_func = BXIMAGE_FUNC_NULL ;
2013-10-15 00:40:41 +04:00
bx_hdimage = - 1 ;
bx_fdsize_idx = - 1 ;
bx_hdsize = 0 ;
bx_imagemode = - 1 ;
2013-11-02 19:58:59 +04:00
bx_backup = 0 ;
2013-10-14 00:30:51 +04:00
bx_interactive = 1 ;
2018-03-31 00:04:04 +03:00
bx_sectsize_idx = 0 ;
bx_sectsize_val = 512 ;
2013-10-14 00:30:51 +04:00
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 ;
}
2021-01-09 19:51:52 +03:00
else if ( ! strncmp ( " -func= " , argv [ arg ] , 6 ) ) {
if ( bximage_func ! = BXIMAGE_FUNC_NULL ) {
2013-10-19 14:20:37 +04:00
printf ( " bximage mode already defined \n \n " ) ;
ret = 0 ;
} else if ( ! strcmp ( & argv [ arg ] [ 6 ] , " create " ) ) {
2021-01-09 19:51:52 +03:00
bximage_func = BXIMAGE_FUNC_CREATE_IMAGE ;
2013-10-14 00:30:51 +04:00
} else if ( ! strcmp ( & argv [ arg ] [ 6 ] , " convert " ) ) {
2021-01-09 19:51:52 +03:00
bximage_func = BXIMAGE_FUNC_CONVERT_IMAGE ;
2013-10-18 23:36:50 +04:00
} else if ( ! strcmp ( & argv [ arg ] [ 6 ] , " resize " ) ) {
2021-01-09 19:51:52 +03:00
bximage_func = BXIMAGE_FUNC_RESIZE_IMAGE ;
2013-10-14 00:30:51 +04:00
} else if ( ! strcmp ( & argv [ arg ] [ 6 ] , " commit " ) ) {
2021-01-09 19:51:52 +03:00
bximage_func = BXIMAGE_FUNC_COMMIT_UNDOABLE ;
2013-11-03 20:13:35 +04:00
} else if ( ! strcmp ( & argv [ arg ] [ 6 ] , " info " ) ) {
2021-01-09 19:51:52 +03:00
bximage_func = BXIMAGE_FUNC_IMAGE_INFO ;
2013-10-14 00:30:51 +04:00
} else {
printf ( " Unknown bximage mode '%s' \n \n " , & argv [ arg ] [ 6 ] ) ;
2013-10-19 14:20:37 +04:00
ret = 0 ;
2013-10-14 00:30:51 +04:00
}
}
2013-10-15 00:40:41 +04:00
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 ;
2013-10-17 20:00:26 +04:00
bx_hdsize = ( int ) strtol ( & argv [ arg ] [ 4 ] , & suffix , 10 ) ;
if ( bx_hdsize < = 0 ) {
2013-10-15 00:40:41 +04:00
printf ( " Error in hard disk image size argument: %s \n \n " , & argv [ arg ] [ 4 ] ) ;
ret = 0 ;
2013-10-17 20:00:26 +04:00
} else if ( ! strcmp ( suffix , " G " ) ) {
bx_hdsize < < = 10 ;
2013-10-22 23:26:42 +04:00
} else if ( ( strlen ( suffix ) > 0 ) & & strcmp ( suffix , " M " ) ) {
2013-10-17 20:00:26 +04:00
printf ( " Error in hard disk image size suffix: %s \n \n " , & argv [ arg ] [ 4 ] ) ;
ret = 0 ;
}
2013-10-18 23:36:50 +04:00
if ( ( bx_hdsize < bx_min_hd_megs ) | | ( bx_hdsize > bx_max_hd_megs ) ) {
2013-10-15 00:40:41 +04:00
printf ( " Hard disk image size out of range \n \n " ) ;
ret = 0 ;
}
}
2013-10-14 00:30:51 +04:00
else if ( ! strncmp ( " -imgmode= " , argv [ arg ] , 9 ) ) {
2013-10-15 00:40:41 +04:00
bx_imagemode = get_menu_index ( & argv [ arg ] [ 9 ] , hdmode_n_choices , hdmode_choices ) ;
if ( bx_imagemode < 0 ) {
2013-10-14 00:30:51 +04:00
printf ( " Unknown image mode: %s \n \n " , & argv [ arg ] [ 9 ] ) ;
ret = 0 ;
}
}
2018-03-31 00:04:04 +03:00
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 ] ) ;
}
}
2013-11-02 19:58:59 +04:00
else if ( ! strcmp ( " -b " , argv [ arg ] ) ) {
bx_backup = 1 ;
2013-10-14 00:30:51 +04:00
}
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 + + ;
}
2021-01-09 19:51:52 +03:00
if ( bximage_func = = BXIMAGE_FUNC_NULL ) {
2023-12-10 13:17:21 +03:00
printf ( " \n ERROR: Parameter -func missing - switching to interactive mode. \n \n " ) ;
2013-10-14 00:30:51 +04:00
bx_interactive = 1 ;
2013-10-15 00:40:41 +04:00
} else {
set_default_values ( ) ;
2013-10-20 11:30:36 +04:00
if ( fnargs < 1 ) {
2013-10-15 00:40:41 +04:00
bx_interactive = 1 ;
2023-12-10 13:17:21 +03:00
printf ( " \n ERROR; Filename missing - switching to interactive mode. \n \n " ) ;
2013-10-15 00:40:41 +04:00
}
2021-01-09 19:51:52 +03:00
if ( ( bximage_func = = BXIMAGE_FUNC_COMMIT_UNDOABLE ) & & ( fnargs = = 1 ) ) {
2020-09-01 23:13:56 +03:00
snprintf ( bx_filename_2 , 520 , " %s%s " , bx_filename_1 , UNDOABLE_REDOLOG_EXTENSION ) ;
2013-10-20 11:30:36 +04:00
}
2013-10-14 00:30:51 +04:00
}
return ret ;
}
2013-10-15 22:00:39 +04:00
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 , " \n The disk image '%s' already exists. Are you sure you want to replace it? \n Please type yes or no. " , filename ) ;
if ( ask_yn ( buffer , 0 , & confirm ) < 0 )
fatal ( EOF_ERR ) ;
if ( ! confirm )
fatal ( " ERROR: Aborted " ) ;
}
}
2013-10-18 23:36:50 +04:00
void check_image_names ( )
{
2020-09-01 23:13:56 +03:00
char backup_fname [ 520 ] ;
2013-11-02 19:58:59 +04:00
2013-10-18 23:36:50 +04:00
if ( ! strlen ( bx_filename_2 ) ) {
strcpy ( bx_filename_2 , bx_filename_1 ) ;
}
2013-11-02 19:58:59 +04:00
if ( ( ! strcmp ( bx_filename_1 , bx_filename_2 ) ) & & bx_backup ) {
2020-09-01 23:13:56 +03:00
snprintf ( backup_fname , 517 , " %s%s " , bx_filename_1 , " .orig " ) ;
2013-11-02 19:58:59 +04:00
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 ) ;
2013-10-18 23:36:50 +04:00
}
}
2021-01-09 19:51:52 +03:00
int get_image_mode_and_hdsize ( const char * filename , const char * * image_mode )
2013-10-19 14:20:37 +04:00
{
device_image_t * source_image ;
2021-01-09 19:51:52 +03:00
int hdsize_megs = 0 ;
2013-10-19 14:20:37 +04:00
2021-01-09 19:51:52 +03:00
if ( ! hdimage_detect_image_mode ( bx_filename_1 , image_mode ) )
2013-10-19 14:20:37 +04:00
fatal ( " source disk image mode not detected " ) ;
2021-01-09 19:51:52 +03:00
source_image = init_image ( * image_mode ) ;
2013-10-19 14:20:37 +04:00
if ( source_image - > open ( bx_filename_1 , O_RDONLY ) < 0 ) {
fatal ( " cannot open source disk image " ) ;
} else {
2021-01-09 19:51:52 +03:00
hdsize_megs = ( int ) ( source_image - > hd_size > > 20 ) ;
2013-10-19 14:20:37 +04:00
source_image - > close ( ) ;
}
2021-01-09 19:51:52 +03:00
return hdsize_megs ;
2013-10-19 14:20:37 +04:00
}
2014-06-07 11:32:06 +04:00
int CDECL main ( int argc , char * argv [ ] )
2013-10-14 00:30:51 +04:00
{
2020-09-01 23:13:56 +03:00
char bochsrc_line [ 1024 ] , prompt [ 80 ] , tmpfname [ 528 ] ;
2013-10-15 22:00:39 +04:00
Bit64u hdsize = 0 ;
2013-11-03 20:13:35 +04:00
device_image_t * hdimage ;
2021-01-06 00:57:13 +03:00
const char * image_mode = NULL ;
2013-10-14 00:30:51 +04:00
if ( ! parse_cmdline ( argc , argv ) )
myexit ( 1 ) ;
print_banner ( ) ;
if ( bx_interactive ) {
2021-01-09 19:51:52 +03:00
if ( ask_int ( main_menu_prompt , 0 , 5 , bximage_func , & bximage_func ) < 0 )
2013-10-14 00:30:51 +04:00
fatal ( EOF_ERR ) ;
2013-10-15 00:40:41 +04:00
set_default_values ( ) ;
2021-01-09 19:51:52 +03:00
switch ( bximage_func ) {
case BXIMAGE_FUNC_NULL :
2013-10-14 00:30:51 +04:00
myexit ( 0 ) ;
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_CREATE_IMAGE :
2013-10-19 14:20:37 +04:00
printf ( " \n Create image \n " ) ;
2013-10-15 00:40:41 +04:00
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 ( " \n What 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 ) ;
2021-01-09 19:51:52 +03:00
image_mode = hdmode_choices [ bx_imagemode ] ;
if ( ! strcmp ( image_mode , " flat " ) | |
! strcmp ( image_mode , " sparse " ) | |
! strcmp ( image_mode , " growing " ) ) {
2018-03-31 00:04:04 +03:00
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 ) ;
}
2013-10-18 23:36:50 +04:00
sprintf ( prompt , " \n Enter 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 )
2013-10-15 00:40:41 +04:00
fatal ( EOF_ERR ) ;
if ( ! strlen ( bx_filename_1 ) ) {
strcpy ( bx_filename_1 , " c.img " ) ;
}
if ( ask_string ( " \n What should be the name of the image? \n " , bx_filename_1 , bx_filename_1 ) < 0 )
fatal ( EOF_ERR ) ;
}
2013-10-14 19:57:10 +04:00
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_CONVERT_IMAGE :
2013-10-19 14:20:37 +04:00
printf ( " \n Convert image \n " ) ;
2013-10-14 00:30:51 +04:00
if ( ! strlen ( bx_filename_1 ) ) {
strcpy ( bx_filename_1 , " c.img " ) ;
}
if ( ask_string ( " \n What is the name of the source image? \n " , bx_filename_1 , bx_filename_1 ) < 0 )
fatal ( EOF_ERR ) ;
if ( ! strlen ( bx_filename_2 ) ) {
2013-10-25 23:10:09 +04:00
strcpy ( tmpfname , bx_filename_1 ) ;
2013-10-14 00:30:51 +04:00
} else {
2013-10-18 23:36:50 +04:00
strcpy ( tmpfname , bx_filename_2 ) ;
2013-10-14 00:30:51 +04:00
}
2013-10-18 23:36:50 +04:00
if ( ask_string ( " \n What should be the name of the new image? \n " , tmpfname , bx_filename_2 ) < 0 )
2013-10-14 00:30:51 +04:00
fatal ( EOF_ERR ) ;
2013-10-15 00:40:41 +04:00
if ( ask_menu ( hdmode_menu , hdmode_n_choices , hdmode_choices , bx_imagemode , & bx_imagemode ) < 0 )
2013-10-14 00:30:51 +04:00
fatal ( EOF_ERR ) ;
2013-11-02 19:58:59 +04:00
if ( ! strcmp ( bx_filename_1 , bx_filename_2 ) ) {
if ( ask_yn ( " \n Should I create a backup of the source image \n " , 0 , & bx_backup ) < 0 )
fatal ( EOF_ERR ) ;
}
2013-10-14 00:30:51 +04:00
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_RESIZE_IMAGE :
2013-10-19 14:20:37 +04:00
printf ( " \n Resize image \n " ) ;
2013-10-18 23:36:50 +04:00
if ( ! strlen ( bx_filename_1 ) ) {
strcpy ( bx_filename_1 , " c.img " ) ;
}
if ( ask_string ( " \n What is the name of the source image? \n " , bx_filename_1 , bx_filename_1 ) < 0 )
fatal ( EOF_ERR ) ;
if ( ! strlen ( bx_filename_2 ) ) {
2013-10-25 23:10:09 +04:00
strcpy ( tmpfname , bx_filename_1 ) ;
2013-10-18 23:36:50 +04:00
} else {
strcpy ( tmpfname , bx_filename_2 ) ;
}
if ( ask_string ( " \n What should be the name of the new image? \n " , tmpfname , bx_filename_2 ) < 0 )
fatal ( EOF_ERR ) ;
2021-01-09 19:51:52 +03:00
bx_min_hd_megs = get_image_mode_and_hdsize ( bx_filename_1 , & image_mode ) ;
2013-10-18 23:36:50 +04:00
sprintf ( prompt , " \n Enter 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 ) ;
2013-11-02 19:58:59 +04:00
if ( ! strcmp ( bx_filename_1 , bx_filename_2 ) ) {
if ( ask_yn ( " \n Should I create a backup of the source image \n " , 0 , & bx_backup ) < 0 )
fatal ( EOF_ERR ) ;
}
2013-10-18 23:36:50 +04:00
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_COMMIT_UNDOABLE :
2013-10-19 14:20:37 +04:00
printf ( " \n Commit redolog \n " ) ;
2013-10-14 00:30:51 +04:00
if ( ! strlen ( bx_filename_1 ) ) {
strcpy ( bx_filename_1 , " c.img " ) ;
}
if ( ask_string ( " \n What is the name of the base image? \n " , bx_filename_1 , bx_filename_1 ) < 0 )
fatal ( EOF_ERR ) ;
if ( ! strlen ( bx_filename_2 ) ) {
2020-09-01 23:13:56 +03:00
snprintf ( tmpfname , 520 , " %s%s " , bx_filename_1 , UNDOABLE_REDOLOG_EXTENSION ) ;
2013-10-14 00:30:51 +04:00
} else {
2013-10-18 23:36:50 +04:00
strcpy ( tmpfname , bx_filename_2 ) ;
2013-10-14 00:30:51 +04:00
}
2013-10-18 23:36:50 +04:00
if ( ask_string ( " \n What is the redolog name? \n " , tmpfname , bx_filename_2 ) < 0 )
2013-10-14 00:30:51 +04:00
fatal ( EOF_ERR ) ;
2013-11-02 19:58:59 +04:00
if ( ask_yn ( " \n Should I create a backup of base image and redolog \n " , 0 , & bx_backup ) < 0 )
2013-10-14 00:30:51 +04:00
fatal ( EOF_ERR ) ;
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_IMAGE_INFO :
2013-11-03 20:13:35 +04:00
printf ( " \n Disk image info \n " ) ;
if ( ! strlen ( bx_filename_1 ) ) {
strcpy ( bx_filename_1 , " c.img " ) ;
}
if ( ask_string ( " \n What is the name of the image? \n " , bx_filename_1 , bx_filename_1 ) < 0 )
fatal ( EOF_ERR ) ;
break ;
2013-10-14 00:30:51 +04:00
default :
2013-10-21 22:26:00 +04:00
fatal ( " \n bximage: unknown mode " ) ;
2013-10-14 00:30:51 +04:00
}
}
2021-01-09 19:51:52 +03:00
switch ( bximage_func ) {
case BXIMAGE_FUNC_CREATE_IMAGE :
2013-10-19 14:20:37 +04:00
image_overwrite_check ( bx_filename_1 ) ;
if ( bx_hdimage = = 0 ) {
sprintf ( bochsrc_line , " floppya: image= \" %s \" , status=inserted " , bx_filename_1 ) ;
2015-01-26 00:24:13 +03:00
printf ( " \n Creating floppy image '%s' with %u sectors \n " , bx_filename_1 , fdsize_sectors [ bx_fdsize_idx ] ) ;
2013-10-19 14:20:37 +04:00
create_flat_image ( bx_filename_1 , fdsize_sectors [ bx_fdsize_idx ] * 512 ) ;
} else {
2013-10-25 23:10:09 +04:00
int heads = 16 , spt = 63 ;
2018-03-31 00:04:04 +03:00
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 ) ;
}
2021-01-09 19:51:52 +03:00
image_mode = hdmode_choices [ bx_imagemode ] ;
2013-10-19 14:20:37 +04:00
hdsize = ( ( Bit64u ) bx_hdsize ) < < 20 ;
2018-03-31 00:04:04 +03:00
Bit64u cyl = ( Bit64u ) ( hdsize / 16.0 / 63.0 / bx_sectsize_val ) ;
2013-10-19 14:20:37 +04:00
if ( cyl > = ( 1 < < BX_MAX_CYL_BITS ) )
fatal ( " ERROR: number of cylinders out of range ! \n " ) ;
2018-03-31 00:04:04 +03:00
printf ( " \n Creating 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 ;
2021-01-16 15:47:02 +03:00
hdimage = create_hard_disk_image ( bx_filename_1 , image_mode , hdsize ) ;
delete hdimage ;
2013-10-19 14:20:37 +04:00
}
printf ( " \n The following line should appear in your bochsrc: \n " ) ;
printf ( " %s \n " , bochsrc_line ) ;
2013-10-15 22:00:39 +04:00
# ifdef WIN32
2013-10-19 14:20:37 +04:00
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 " ) ;
}
2013-10-15 22:00:39 +04:00
# endif
2013-10-19 14:20:37 +04:00
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_CONVERT_IMAGE :
2013-10-19 14:20:37 +04:00
check_image_names ( ) ;
2021-01-09 19:51:52 +03:00
image_mode = hdmode_choices [ bx_imagemode ] ;
convert_image ( image_mode , 0 ) ;
2013-10-19 14:20:37 +04:00
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_RESIZE_IMAGE :
bx_min_hd_megs = get_image_mode_and_hdsize ( bx_filename_1 , & image_mode ) ;
2013-10-19 14:20:37 +04:00
if ( bx_hdsize < bx_min_hd_megs )
fatal ( " invalid image size " ) ;
hdsize = ( ( Bit64u ) bx_hdsize ) < < 20 ;
check_image_names ( ) ;
2021-01-09 19:51:52 +03:00
convert_image ( image_mode , hdsize ) ;
2013-10-19 14:20:37 +04:00
break ;
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_COMMIT_UNDOABLE :
2013-11-02 19:58:59 +04:00
if ( bx_backup ) {
2020-09-01 23:13:56 +03:00
snprintf ( tmpfname , 517 , " %s%s " , bx_filename_1 , " .orig " ) ;
2013-11-02 19:58:59 +04:00
if ( hdimage_copy_file ( bx_filename_1 , tmpfname ) ! = 1 )
fatal ( " backup of base image file failed " ) ;
}
2013-10-19 14:20:37 +04:00
commit_redolog ( ) ;
2013-11-02 19:58:59 +04:00
if ( bx_backup ) {
2020-09-01 23:13:56 +03:00
snprintf ( tmpfname , 527 , " %s%s " , bx_filename_2 , " .orig " ) ;
2013-11-02 19:58:59 +04:00
if ( rename ( bx_filename_2 , tmpfname ) ! = 0 )
fatal ( " rename of redolog file failed " ) ;
} else {
2013-10-19 14:20:37 +04:00
if ( unlink ( bx_filename_2 ) ! = 0 )
fatal ( " ERROR: while removing the redolog ! \n " ) ;
}
break ;
2013-11-03 20:13:35 +04:00
2021-01-09 19:51:52 +03:00
case BXIMAGE_FUNC_IMAGE_INFO :
if ( ! hdimage_detect_image_mode ( bx_filename_1 , & image_mode ) ) {
2013-11-03 20:13:35 +04:00
fatal ( " disk image mode not detected " ) ;
} else {
2021-01-09 19:51:52 +03:00
printf ( " \n disk image mode = '%s' \n " , image_mode ) ;
2013-11-03 20:13:35 +04:00
}
2021-01-09 19:51:52 +03:00
hdimage = init_image ( image_mode ) ;
2013-11-03 20:13:35 +04:00
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 ) ;
2014-12-01 20:06:00 +03:00
printf ( " geometry = " FMT_LL " d/16/63 ( " FMT_LL " d MB) \n \n " , cyl ,
2013-11-03 20:13:35 +04:00
hdimage - > hd_size > > 20 ) ;
2014-04-20 16:50:05 +04:00
} else {
2014-12-01 20:06:00 +03:00
printf ( " geometry = %d/%d/%d ( " FMT_LL " d MB) \n \n " , hdimage - > cylinders ,
2014-04-20 16:50:05 +04:00
hdimage - > heads , hdimage - > spt , hdimage - > hd_size > > 20 ) ;
2013-11-03 20:13:35 +04:00
}
hdimage - > close ( ) ;
}
break ;
2013-10-14 00:30:51 +04:00
}
2013-10-15 22:00:39 +04:00
myexit ( 0 ) ;
2013-10-14 00:30:51 +04:00
return 0 ;
}