2003-05-03 20:37:18 +04:00
/*
2001-06-01 08:20:26 +04:00
* misc / bximage . c
2011-02-25 01:05:47 +03:00
* $ Id $
2001-06-01 07:30:40 +04:00
*
* Create empty hard disk or floppy disk images for bochs .
*
*/
2004-04-30 21:26:38 +04:00
# ifdef WIN32
# include <conio.h>
2004-08-19 20:03:03 +04:00
# include <windows.h>
# include <winioctl.h>
2004-08-19 23:42:22 +04:00
# ifdef _MSC_VER
# include <io.h>
# endif
2004-04-30 21:26:38 +04:00
# endif
2001-06-01 07:30:40 +04:00
# include <stdio.h>
# include <stdlib.h>
2001-06-01 09:11:45 +04:00
# include <string.h>
# include <ctype.h>
2001-06-01 07:30:40 +04:00
# include <assert.h>
# include "config.h"
2003-05-03 20:37:18 +04:00
# include <string.h>
2004-08-19 23:42:22 +04:00
# include "../osdep.h"
2012-01-04 23:20:52 +04:00
# include "bswap.h"
2004-08-19 23:42:22 +04:00
2005-11-06 14:07:01 +03:00
# define HDIMAGE_HEADERS_ONLY 1
# include "../iodev/hdimage.h"
2003-05-03 20:37:18 +04:00
2012-07-26 20:15:48 +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 ) ;
2012-07-24 23:15:00 +04:00
2004-08-29 23:31:09 +04:00
int bx_hdimage ;
int bx_fdsize_idx ;
int bx_hdsize ;
int bx_hdimagemode ;
int bx_interactive ;
char bx_filename [ 256 ] ;
2004-08-19 20:03:03 +04:00
typedef int ( * WRITE_IMAGE ) ( FILE * , Bit64u ) ;
# ifdef WIN32
typedef int ( * WRITE_IMAGE_WIN32 ) ( HANDLE , Bit64u ) ;
# endif
2001-06-01 08:20:26 +04:00
char * EOF_ERR = " ERROR: End of input " ;
2011-02-25 01:05:47 +03:00
char * rcsid = " $Id$ " ;
2001-06-01 08:20:26 +04:00
char * divider = " ======================================================================== " ;
2001-06-01 08:07:21 +04:00
2001-06-01 08:20:26 +04:00
/* menu data for choosing floppy/hard disk */
2003-05-03 20:37:18 +04:00
char * fdhd_menu = " \n Do you want to create a floppy disk image or a hard disk image? \n Please type hd or fd. " ;
2001-06-01 08:20:26 +04:00
char * fdhd_choices [ ] = { " fd " , " hd " } ;
int fdhd_n_choices = 2 ;
/* menu data for choosing floppy size */
2005-11-20 23:26:35 +03:00
char * fdsize_menu = " \n Choose the size of floppy disk image to create, in megabytes. \n Please type 0.16, 0.18, 0.32, 0.36, 0.72, 1.2, 1.44, 1.68, 1.72, or 2.88. \n " ;
char * fdsize_choices [ ] = { " 0.16 " , " 0.18 " , " 0.32 " , " 0.36 " , " 0.72 " , " 1.2 " , " 1.44 " , " 1.68 " , " 1.72 " , " 2.88 " } ;
int fdsize_n_choices = 10 ;
2001-06-01 08:20:26 +04:00
2003-05-03 20:37:18 +04:00
/* menu data for choosing disk mode */
2003-08-01 05:20:00 +04:00
char * hdmode_menu = " \n What kind of image should I create? \n Please type flat, sparse or growing. " ;
char * hdmode_choices [ ] = { " flat " , " sparse " , " growing " } ;
2003-05-03 20:37:18 +04:00
int hdmode_n_choices = 3 ;
2009-04-14 13:45:22 +04:00
void myexit ( int code )
2002-11-21 09:36:22 +03:00
{
# ifdef WIN32
2009-04-14 13:45:22 +04:00
printf ( " \n Press any key to continue \n " ) ;
2002-11-21 09:36:22 +03:00
getch ( ) ;
# endif
exit ( code ) ;
}
2001-06-01 08:20:26 +04:00
/* stolen from main.cc */
2009-04-14 13:45:22 +04:00
void bx_center_print ( FILE * file , char * line , int maxwidth )
2001-06-01 08:20:26 +04:00
{
int imax ;
int i ;
imax = ( maxwidth - strlen ( line ) ) > > 1 ;
2009-04-14 13:45:22 +04:00
for ( i = 0 ; i < imax ; i + + ) fputc ( ' ' , file ) ;
fputs ( line , file ) ;
2001-06-01 08:20:26 +04:00
}
2009-04-14 13:45:22 +04:00
void print_banner ( )
2001-06-01 08:20:26 +04:00
{
2009-04-14 13:45:22 +04:00
printf ( " %s \n " , divider ) ;
bx_center_print ( stdout , " bximage \n " , 72 ) ;
bx_center_print ( stdout , " Disk Image Creation Tool for Bochs \n " , 72 ) ;
bx_center_print ( stdout , rcsid , 72 ) ;
printf ( " \n %s \n " , divider ) ;
2001-06-01 08:20:26 +04:00
}
/* this is how we crash */
2009-04-14 13:45:22 +04:00
void fatal ( const char * c )
2001-06-01 08:20:26 +04:00
{
2009-04-14 13:45:22 +04:00
printf ( " %s \n " , c ) ;
myexit ( 1 ) ;
2001-06-01 08:20:26 +04:00
}
2004-08-29 23:31:09 +04:00
/* 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 , char * choice [ ] )
{
int i ;
for ( i = 0 ; i < n_choices ; i + + ) {
2009-04-14 13:45:22 +04:00
if ( ! strcmp ( choice [ i ] , arg ) ) {
2004-08-29 23:31:09 +04:00
// matched, return the choice number
return i ;
}
}
return - 1 ;
}
2008-02-06 01:57:43 +03:00
/* remove leading spaces, newline junk at end. returns pointer to
2001-06-01 08:20:26 +04:00
cleaned string , which is between s0 and the null */
2009-04-14 13:45:22 +04:00
char * clean_string ( char * s0 )
2001-06-01 07:30:40 +04:00
{
char * s = s0 ;
char * ptr ;
/* find first nonblank */
2009-04-14 13:45:22 +04:00
while ( isspace ( * s ) )
2001-06-01 07:30:40 +04:00
s + + ;
/* truncate string at first non-alphanumeric */
ptr = s ;
2009-04-14 13:45:22 +04:00
while ( isprint ( * ptr ) )
2001-06-01 07:30:40 +04:00
ptr + + ;
* ptr = 0 ;
return s ;
}
2001-06-01 08:20:26 +04:00
/* returns 0 on success, -1 on failure. The value goes into out. */
2009-04-14 13:45:22 +04:00
int ask_int ( const char * prompt , int min , int max , int the_default , int * out )
2001-06-01 07:30:40 +04:00
{
int n = max + 1 ;
char buffer [ 1024 ] ;
char * clean ;
int illegal ;
2009-04-14 13:45:22 +04:00
while ( 1 ) {
printf ( " %s " , prompt ) ;
printf ( " [%d] " , the_default ) ;
if ( ! fgets ( buffer , sizeof ( buffer ) , stdin ) )
2001-06-01 07:30:40 +04:00
return - 1 ;
2009-04-14 13:45:22 +04:00
clean = clean_string ( buffer ) ;
2001-06-01 07:30:40 +04:00
if ( strlen ( clean ) < 1 ) {
// empty line, use the default
* out = the_default ;
return 0 ;
}
2009-04-14 13:45:22 +04:00
illegal = ( 1 ! = sscanf ( buffer , " %d " , & n ) ) ;
2001-06-01 07:30:40 +04:00
if ( illegal | | n < min | | n > max ) {
2009-04-14 13:45:22 +04:00
printf ( " Your choice (%s) was not an integer between %d and %d. \n \n " ,
2001-06-01 07:30:40 +04:00
clean , min , max ) ;
} else {
// choice is okay
* out = n ;
return 0 ;
}
}
}
2009-04-14 13:45:22 +04:00
int ask_menu ( const char * prompt , int n_choices , char * choice [ ] , int the_default , int * out )
2001-06-01 07:30:40 +04:00
{
char buffer [ 1024 ] ;
char * clean ;
int i ;
* out = - 1 ;
2009-04-14 13:45:22 +04:00
while ( 1 ) {
printf ( " %s " , prompt ) ;
printf ( " [%s] " , choice [ the_default ] ) ;
if ( ! fgets ( buffer , sizeof ( buffer ) , stdin ) )
2001-06-01 07:30:40 +04:00
return - 1 ;
2009-04-14 13:45:22 +04:00
clean = clean_string ( buffer ) ;
2001-06-01 07:30:40 +04:00
if ( strlen ( clean ) < 1 ) {
// empty line, use the default
* out = the_default ;
return 0 ;
}
for ( i = 0 ; i < n_choices ; i + + ) {
2009-04-14 13:45:22 +04:00
if ( ! strcmp ( choice [ i ] , clean ) ) {
2001-06-01 07:30:40 +04:00
// matched, return the choice number
* out = i ;
return 0 ;
}
}
2009-04-14 13:45:22 +04:00
printf ( " Your choice (%s) did not match any of the choices: \n " , clean ) ;
2001-06-01 07:30:40 +04:00
for ( i = 0 ; i < n_choices ; i + + ) {
2009-04-14 13:45:22 +04:00
if ( i > 0 ) printf ( " , " ) ;
printf ( " %s " , choice [ i ] ) ;
2001-06-01 07:30:40 +04:00
}
2009-04-14 13:45:22 +04:00
printf ( " \n " ) ;
2001-06-01 07:30:40 +04:00
}
}
2009-04-14 13:45:22 +04:00
int ask_yn ( const char * prompt , int the_default , int * out )
2001-06-01 08:07:21 +04:00
{
char buffer [ 16 ] ;
char * clean ;
* out = - 1 ;
while ( 1 ) {
2009-04-14 13:45:22 +04:00
printf ( " %s " , prompt ) ;
printf ( " [%s] " , the_default ? " yes " : " no " ) ;
if ( ! fgets ( buffer , sizeof ( buffer ) , stdin ) )
2001-06-01 08:07:21 +04:00
return - 1 ;
2009-04-14 13:45:22 +04:00
clean = clean_string ( buffer ) ;
2001-06-01 08:07:21 +04:00
if ( strlen ( clean ) < 1 ) {
// empty line, use the default
* out = the_default ;
return 0 ;
}
2009-04-14 13:45:22 +04:00
switch ( tolower ( clean [ 0 ] ) ) {
2001-06-01 08:07:21 +04:00
case ' y ' : * out = 1 ; return 0 ;
case ' n ' : * out = 0 ; return 0 ;
}
2009-04-14 13:45:22 +04:00
printf ( " Please type either yes or no. \n " ) ;
2001-06-01 08:07:21 +04:00
}
}
2009-04-14 13:45:22 +04:00
int ask_string ( const char * prompt , char * the_default , char * out )
2001-06-01 07:30:40 +04:00
{
char buffer [ 1024 ] ;
char * clean ;
out [ 0 ] = 0 ;
2009-04-14 13:45:22 +04:00
printf ( " %s " , prompt ) ;
printf ( " [%s] " , the_default ) ;
if ( ! fgets ( buffer , sizeof ( buffer ) , stdin ) )
2001-06-01 07:30:40 +04:00
return - 1 ;
2009-04-14 13:45:22 +04:00
clean = clean_string ( buffer ) ;
2001-06-01 07:30:40 +04:00
if ( strlen ( clean ) < 1 ) {
// empty line, use the default
2009-04-14 13:45:22 +04:00
strcpy ( out , the_default ) ;
2001-06-01 07:30:40 +04:00
return 0 ;
}
2009-04-14 13:45:22 +04:00
strcpy ( out , clean ) ;
2001-06-01 07:30:40 +04:00
return 0 ;
}
2003-05-03 20:37:18 +04:00
// fileset is like memset but for a file handle
void fileset ( FILE * fp , int c , size_t n )
{
# define BLOCK_SIZE (1024)
int block [ BLOCK_SIZE ] ;
size_t left_to_write = n ;
memset ( block , c , sizeof ( block ) ) ;
2009-04-14 13:45:22 +04:00
while ( left_to_write > 0 )
2003-05-03 20:37:18 +04:00
{
size_t write = sizeof ( block ) ;
if ( write > left_to_write ) write = left_to_write ;
if ( 1 ! = fwrite ( block , write , 1 , fp ) )
{
2009-04-14 13:45:22 +04:00
fclose ( fp ) ;
fatal ( " ERROR: The disk image is not complete - could not write data block! " ) ;
2003-05-03 20:37:18 +04:00
}
left_to_write - = write ;
}
}
/* Create a suited redolog header */
void make_redolog_header ( redolog_header_t * header , const char * type , Bit64u size )
{
2009-04-14 13:45:22 +04:00
Bit32u entries , extent_size , bitmap_size ;
Bit64u maxsize ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
// Set standard header values
strcpy ( ( char * ) header - > standard . magic , STANDARD_HEADER_MAGIC ) ;
strcpy ( ( char * ) header - > standard . type , REDOLOG_TYPE ) ;
strcpy ( ( char * ) header - > standard . subtype , type ) ;
header - > standard . version = htod32 ( STANDARD_HEADER_VERSION ) ;
header - > standard . header = htod32 ( STANDARD_HEADER_SIZE ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
entries = 512 ;
bitmap_size = 1 ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
// Compute #entries and extent size values
do {
static Bit32u flip = 0 ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
extent_size = 8 * bitmap_size * 512 ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
header - > specific . catalog = htod32 ( entries ) ;
header - > specific . bitmap = htod32 ( bitmap_size ) ;
header - > specific . extent = htod32 ( extent_size ) ;
2008-02-06 01:57:43 +03:00
2009-04-14 13:45:22 +04:00
maxsize = ( Bit64u ) entries * ( Bit64u ) extent_size ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
flip + + ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
if ( flip & 0x01 ) bitmap_size * = 2 ;
else entries * = 2 ;
} while ( maxsize < size ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
header - > specific . disk = htod64 ( size ) ;
2003-05-03 20:37:18 +04:00
}
/* produce a flat image file */
2008-02-06 01:57:43 +03:00
# ifdef WIN32
2004-08-19 20:03:03 +04:00
int make_flat_image_win32 ( HANDLE hFile , Bit64u sec )
{
LARGE_INTEGER pos ;
2005-11-06 14:07:01 +03:00
DWORD dwCount , errCode ;
2004-08-19 20:03:03 +04:00
USHORT mode ;
char buffer [ 1024 ] ;
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 ) ;
2005-06-04 21:44:59 +04:00
pos . u . LowPart = ( unsigned long ) ( ( sec - 1 ) < < 9 ) ;
pos . u . HighPart = ( unsigned long ) ( ( sec - 1 ) > > 23 ) ;
2004-08-19 20:03:03 +04:00
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 )
{
2005-11-06 14:07:01 +03:00
errCode = GetLastError ( ) ;
2004-08-19 20:03:03 +04:00
CloseHandle ( hFile ) ;
2005-11-06 14:07:01 +03:00
if ( errCode = = ERROR_DISK_FULL ) {
2009-04-14 13:45:22 +04:00
fatal ( " \n ERROR: Not enough space on disk for image! " ) ;
2005-11-06 14:07:01 +03:00
} else {
sprintf ( buffer , " \n ERROR: Disk image creation failed with error code %i! " , errCode ) ;
2009-04-14 13:45:22 +04:00
fatal ( buffer ) ;
2005-11-06 14:07:01 +03:00
}
2004-08-19 20:03:03 +04:00
}
return 0 ;
}
# endif
2003-05-03 20:37:18 +04:00
int make_flat_image ( FILE * fp , Bit64u sec )
{
2009-04-14 13:45:22 +04:00
/*
* seek to sec * 512 - 1 and write a single character .
* can ' t just do : fseek ( fp , 512 * sec - 1 , SEEK_SET )
* because 512 * sec may be too large for signed int .
*/
while ( sec > 0 )
{
/* temp <-- min(sec, 4194303)
2003-05-03 20:37:18 +04:00
* 4194303 is ( int ) ( 0x7FFFFFFF / 512 )
*/
2009-04-14 13:45:22 +04:00
long temp = ( long ) ( ( sec < 4194303 ) ? sec : 4194303 ) ;
fseek ( fp , 512 * temp , SEEK_CUR ) ;
sec - = temp ;
}
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
fseek ( fp , - 1 , SEEK_CUR ) ;
if ( fputc ( ' \0 ' , fp ) = = EOF )
{
fclose ( fp ) ;
fatal ( " \n ERROR: The disk image is not complete! (image larger then free space?) " ) ;
}
return 0 ;
2003-05-03 20:37:18 +04:00
}
/* produce a sparse image file */
int make_sparse_image ( FILE * fp , Bit64u sec )
{
2009-04-14 13:45:22 +04:00
Bit64u numpages ;
sparse_header_t header ;
size_t sizesofar ;
size_t padtopagesize ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
memset ( & header , 0 , sizeof ( header ) ) ;
header . magic = htod32 ( SPARSE_HEADER_MAGIC ) ;
header . version = htod32 ( SPARSE_HEADER_VERSION ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
header . pagesize = htod32 ( ( 1 < < 10 ) * 32 ) ; // Use 32 KB Pages - could be configurable
numpages = ( sec / ( dtoh32 ( header . pagesize ) / 512 ) ) + 1 ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
header . numpages = htod32 ( ( Bit32u ) numpages ) ;
header . disk = htod64 ( sec * 512 ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
if ( numpages ! = dtoh32 ( header . numpages ) )
{
fclose ( fp ) ;
fatal ( " ERROR: The disk image is too large for a sparse image! " ) ;
// Could increase page size here.
// But note this only happens at 128 Terabytes!
}
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
if ( fwrite ( & header , sizeof ( header ) , 1 , fp ) ! = 1 )
{
fclose ( fp ) ;
fatal ( " ERROR: The disk image is not complete - could not write header! " ) ;
}
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
fileset ( fp , 0xff , 4 * dtoh32 ( header . numpages ) ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
sizesofar = SPARSE_HEADER_SIZE + ( 4 * dtoh32 ( header . numpages ) ) ;
padtopagesize = dtoh32 ( header . pagesize ) - ( sizesofar & ( dtoh32 ( header . pagesize ) - 1 ) ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
fileset ( fp , 0 , padtopagesize ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
return 0 ;
2003-05-03 20:37:18 +04:00
}
2003-08-01 05:20:00 +04:00
/* produce a growing image file */
int make_growing_image ( FILE * fp , Bit64u sec )
2003-05-03 20:37:18 +04:00
{
2009-04-14 13:45:22 +04:00
redolog_header_t header ;
Bit32u i , not_allocated = htod32 ( REDOLOG_PAGE_NOT_ALLOCATED ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
memset ( & header , 0 , sizeof ( header ) ) ;
make_redolog_header ( & header , REDOLOG_SUBTYPE_GROWING , sec * 512 ) ;
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
if ( fwrite ( & header , sizeof ( header ) , 1 , fp ) ! = 1 ) {
fclose ( fp ) ;
fatal ( " ERROR: The disk image is not complete - could not write header! " ) ;
}
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
for ( i = 0 ; i < dtoh32 ( header . specific . catalog ) ; i + + ) {
if ( fwrite ( & not_allocated , sizeof ( Bit32u ) , 1 , fp ) ! = 1 ) {
fclose ( fp ) ;
fatal ( " ERROR: The disk image is not complete - could not write catalog! " ) ;
}
}
2003-05-03 20:37:18 +04:00
2009-04-14 13:45:22 +04:00
return 0 ;
2003-05-03 20:37:18 +04:00
}
2001-06-01 08:20:26 +04:00
/* produce the image file */
2004-08-19 20:03:03 +04:00
# ifdef WIN32
int make_image_win32 ( Bit64u sec , char * filename , WRITE_IMAGE_WIN32 write_image )
2001-06-01 07:30:40 +04:00
{
2004-08-19 20:03:03 +04:00
HANDLE hFile ;
char buffer [ 1024 ] ;
// check if it exists before trashing someone's disk image
hFile = CreateFile ( filename , GENERIC_READ , 0 , NULL , OPEN_EXISTING , FILE_ATTRIBUTE_NORMAL , NULL ) ;
if ( hFile ! = INVALID_HANDLE_VALUE ) {
int confirm ;
2009-04-14 13:45:22 +04:00
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 ) ;
2004-08-19 20:03:03 +04:00
if ( ! confirm )
2009-04-14 13:45:22 +04:00
fatal ( " ERROR: Aborted " ) ;
2004-08-19 20:03:03 +04:00
CloseHandle ( hFile ) ;
}
// okay, now open it for writing
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
2009-04-14 13:45:22 +04:00
sprintf ( buffer , " while opening '%s' for writing " , filename ) ;
perror ( buffer ) ;
2004-08-19 20:03:03 +04:00
# endif
2009-04-14 13:45:22 +04:00
fatal ( " ERROR: Could not write disk image " ) ;
2004-08-19 20:03:03 +04:00
}
2009-04-14 13:45:22 +04:00
printf ( " \n Writing: [ " ) ;
2004-08-19 20:03:03 +04:00
2009-04-14 13:45:22 +04:00
if ( ( * write_image ) ( hFile , sec ) ! = 0 )
fatal ( " ERROR: while writing disk image! " ) ;
2004-08-19 20:03:03 +04:00
2009-04-14 13:45:22 +04:00
printf ( " ] Done. \n " ) ;
2004-08-19 20:03:03 +04:00
CloseHandle ( hFile ) ;
return 0 ;
}
# endif
2009-04-14 13:45:22 +04:00
int make_image ( Bit64u sec , char * filename , WRITE_IMAGE write_image )
2004-08-19 20:03:03 +04:00
{
FILE * fp ;
2001-06-01 07:30:40 +04:00
char buffer [ 1024 ] ;
2008-02-06 01:57:43 +03:00
2001-06-01 08:07:21 +04:00
// check if it exists before trashing someone's disk image
2009-04-14 13:45:22 +04:00
fp = fopen ( filename , " r " ) ;
2001-06-01 08:07:21 +04:00
if ( fp ) {
int confirm ;
2009-04-14 13:45:22 +04:00
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 ) ;
2003-05-03 20:37:18 +04:00
if ( ! confirm )
2009-04-14 13:45:22 +04:00
fatal ( " ERROR: Aborted " ) ;
fclose ( fp ) ;
2001-06-01 08:07:21 +04:00
}
// okay, now open it for writing
2009-04-14 13:45:22 +04:00
fp = fopen ( filename , " w " ) ;
2001-06-01 07:30:40 +04:00
if ( fp = = NULL ) {
// attempt to print an error
# ifdef HAVE_PERROR
2009-04-14 13:45:22 +04:00
sprintf ( buffer , " while opening '%s' for writing " , filename ) ;
perror ( buffer ) ;
2001-06-01 07:30:40 +04:00
# endif
2009-04-14 13:45:22 +04:00
fatal ( " ERROR: Could not write disk image " ) ;
2001-06-01 07:30:40 +04:00
}
2002-05-21 11:23:09 +04:00
2009-04-14 13:45:22 +04:00
printf ( " \n Writing: [ " ) ;
2002-05-21 11:23:09 +04:00
2009-04-14 13:45:22 +04:00
if ( ( * write_image ) ( fp , sec ) ! = 0 )
fatal ( " ERROR: while writing disk image! " ) ;
2002-05-21 11:23:09 +04:00
2009-04-14 13:45:22 +04:00
printf ( " ] Done. \n " ) ;
fclose ( fp ) ;
2001-06-01 07:30:40 +04:00
return 0 ;
}
2009-04-14 13:45:22 +04:00
void print_usage ( )
2004-08-29 23:31:09 +04:00
{
2008-02-06 01:57:43 +03:00
fprintf ( stderr ,
2004-08-29 23:31:09 +04:00
" Usage: bximage [options] [filename] \n \n "
" Supported options: \n "
" -fd create floppy image \n "
" -hd create hard disk image \n "
" -mode=... image mode (hard disks only) \n "
" -size=... image size in megabytes \n "
" -q quiet mode (don't prompt for user input) \n "
" --help display this help and exit \n \n " ) ;
}
2009-04-14 13:45:22 +04:00
int parse_cmdline ( int argc , char * argv [ ] )
2004-08-29 23:31:09 +04:00
{
int arg = 1 ;
int ret = 1 ;
bx_hdimage = - 1 ;
bx_fdsize_idx = - 1 ;
bx_hdsize = - 1 ;
bx_hdimagemode = - 1 ;
bx_interactive = 1 ;
bx_filename [ 0 ] = 0 ;
while ( ( arg < argc ) & & ( ret = = 1 ) ) {
// parse next arg
2009-04-14 13:45:22 +04:00
if ( ! strcmp ( " --help " , argv [ arg ] ) | | ! strncmp ( " /? " , argv [ arg ] , 2 ) ) {
2004-08-29 23:31:09 +04:00
print_usage ( ) ;
ret = 0 ;
}
2009-04-14 13:45:22 +04:00
else if ( ! strcmp ( " -fd " , argv [ arg ] ) ) {
2004-08-29 23:31:09 +04:00
bx_hdimage = 0 ;
bx_hdimagemode = 0 ;
}
2009-04-14 13:45:22 +04:00
else if ( ! strcmp ( " -hd " , argv [ arg ] ) ) {
2004-08-29 23:31:09 +04:00
bx_hdimage = 1 ;
bx_fdsize_idx = 0 ;
}
2009-04-14 13:45:22 +04:00
else if ( ! strncmp ( " -mode= " , argv [ arg ] , 6 ) ) {
2004-08-29 23:31:09 +04:00
if ( bx_hdimage = = 1 ) {
bx_hdimagemode = get_menu_index ( & argv [ arg ] [ 6 ] , hdmode_n_choices , hdmode_choices ) ;
if ( bx_hdimagemode < 0 ) {
2009-04-14 13:45:22 +04:00
printf ( " Unknown image mode: %s \n \n " , & argv [ arg ] [ 6 ] ) ;
2004-08-29 23:31:09 +04:00
ret = 0 ;
}
} else {
2009-04-14 13:45:22 +04:00
printf ( " Image mode option only supported for hard disks \n \n " ) ;
2004-08-29 23:31:09 +04:00
ret = 0 ;
}
}
2009-04-14 13:45:22 +04:00
else if ( ! strncmp ( " -size= " , argv [ arg ] , 6 ) ) {
2004-08-29 23:31:09 +04:00
if ( bx_hdimage = = 0 ) {
bx_fdsize_idx = get_menu_index ( & argv [ arg ] [ 6 ] , fdsize_n_choices , fdsize_choices ) ;
if ( bx_fdsize_idx < 0 ) {
2009-04-14 13:45:22 +04:00
printf ( " Unknown floppy image size: %s \n \n " , & argv [ arg ] [ 6 ] ) ;
2004-08-29 23:31:09 +04:00
ret = 0 ;
}
} else if ( bx_hdimage = = 1 ) {
2009-04-14 13:45:22 +04:00
if ( sscanf ( & argv [ arg ] [ 6 ] , " %d " , & bx_hdsize ) ! = 1 ) {
printf ( " Error in hard disk image size argument: %s \n \n " , & argv [ arg ] [ 6 ] ) ;
2004-08-29 23:31:09 +04:00
ret = 0 ;
2012-07-26 20:15:48 +04:00
} else if ( ( bx_hdsize < 1 ) | | ( bx_hdsize > bx_max_hd_megs ) ) {
2009-04-14 13:45:22 +04:00
printf ( " Hard disk image size out of range \n \n " ) ;
2004-08-29 23:31:09 +04:00
ret = 0 ;
2012-07-23 00:40:31 +04:00
}
2004-08-29 23:31:09 +04:00
} else {
2009-04-14 13:45:22 +04:00
printf ( " Image type (fd/hd) not specified \n \n " ) ;
2004-08-29 23:31:09 +04:00
}
}
2009-04-14 13:45:22 +04:00
else if ( ! strcmp ( " -q " , argv [ arg ] ) ) {
2004-08-29 23:31:09 +04:00
bx_interactive = 0 ;
}
else if ( argv [ arg ] [ 0 ] = = ' - ' ) {
2009-04-14 13:45:22 +04:00
printf ( " Unknown option: %s \n \n " , argv [ arg ] ) ;
2004-08-29 23:31:09 +04:00
ret = 0 ;
} else {
strcpy ( bx_filename , argv [ arg ] ) ;
}
arg + + ;
}
if ( bx_hdimage = = - 1 ) {
bx_hdimage = 1 ;
2005-11-20 23:26:35 +03:00
bx_fdsize_idx = 6 ;
2004-08-29 23:31:09 +04:00
bx_interactive = 1 ;
}
if ( bx_hdimage = = 1 ) {
if ( bx_hdimagemode = = - 1 ) {
bx_hdimagemode = 0 ;
bx_interactive = 1 ;
}
if ( bx_hdsize = = - 1 ) {
bx_hdsize = 10 ;
bx_interactive = 1 ;
}
} else {
if ( bx_fdsize_idx = = - 1 ) {
2005-11-20 23:26:35 +03:00
bx_fdsize_idx = 6 ;
2004-08-29 23:31:09 +04:00
bx_interactive = 1 ;
}
}
if ( ! strlen ( bx_filename ) ) {
bx_interactive = 1 ;
}
return ret ;
}
2009-04-14 13:45:22 +04:00
int CDECL main ( int argc , char * argv [ ] )
2001-06-01 07:30:40 +04:00
{
2002-11-15 21:14:08 +03:00
Bit64s sectors = 0 ;
2001-06-01 07:30:40 +04:00
char filename [ 256 ] ;
char bochsrc_line [ 256 ] ;
2012-07-24 23:15:00 +04:00
char prompt [ 80 ] ;
2003-05-03 20:37:18 +04:00
2004-08-19 20:03:03 +04:00
WRITE_IMAGE write_function = NULL ;
# ifdef WIN32
WRITE_IMAGE_WIN32 writefn_win32 = NULL ;
# endif
2008-02-06 01:57:43 +03:00
2009-04-14 13:45:22 +04:00
if ( ! parse_cmdline ( argc , argv ) )
2004-08-29 23:31:09 +04:00
myexit ( 1 ) ;
2009-04-14 13:45:22 +04:00
print_banner ( ) ;
2004-08-29 23:31:09 +04:00
if ( bx_interactive ) {
2009-04-14 13:45:22 +04:00
if ( ask_menu ( fdhd_menu , fdhd_n_choices , fdhd_choices , bx_hdimage , & bx_hdimage ) < 0 )
fatal ( EOF_ERR ) ;
2004-08-29 23:31:09 +04:00
}
if ( bx_hdimage ) {
2012-07-24 23:15:00 +04:00
Bit64u cyl ;
2005-11-06 19:50:30 +03:00
int hdsize , heads = 16 , spt = 63 ;
2003-05-03 20:37:18 +04:00
int mode ;
2004-08-29 23:31:09 +04:00
if ( bx_interactive ) {
2009-04-14 13:45:22 +04:00
if ( ask_menu ( hdmode_menu , hdmode_n_choices , hdmode_choices , bx_hdimagemode , & mode ) < 0 )
2004-08-29 23:31:09 +04:00
fatal ( EOF_ERR ) ;
2012-07-26 20:15:48 +04:00
sprintf ( prompt , " \n Enter the hard disk size in megabytes, between 1 and %d \n " , bx_max_hd_megs ) ;
if ( ask_int ( prompt , 1 , bx_max_hd_megs , bx_hdsize , & hdsize ) < 0 )
2009-04-14 13:45:22 +04:00
fatal ( EOF_ERR ) ;
2004-08-29 23:31:09 +04:00
} else {
mode = bx_hdimagemode ;
hdsize = bx_hdsize ;
}
2012-07-24 23:15:00 +04:00
cyl = ( Bit64u ) ( hdsize * 1024.0 * 1024.0 / 16.0 / 63.0 / 512.0 ) ;
assert ( cyl < ( 1 < < BX_MAX_CYL_BITS ) ) ;
2001-06-01 07:30:40 +04:00
sectors = cyl * heads * spt ;
2009-04-14 13:45:22 +04:00
printf ( " \n I will create a '%s' hard disk image with \n " , hdmode_choices [ mode ] ) ;
2012-07-24 23:15:00 +04:00
printf ( " cyl= " FMT_LL " d \n " , cyl ) ;
2009-04-14 13:45:22 +04:00
printf ( " heads=%d \n " , heads ) ;
printf ( " sectors per track=%d \n " , spt ) ;
printf ( " total sectors= " FMT_LL " d \n " , sectors ) ;
printf ( " total size=%.2f megabytes \n " , ( float ) ( Bit64s ) ( sectors / 2 ) / 1024.0 ) ;
2004-08-29 23:31:09 +04:00
if ( bx_interactive ) {
if ( ! strlen ( bx_filename ) ) strcpy ( bx_filename , " c.img " ) ;
2009-04-14 13:45:22 +04:00
if ( ask_string ( " \n What should I name the image? \n " , bx_filename , filename ) < 0 )
fatal ( EOF_ERR ) ;
2004-08-29 23:31:09 +04:00
} else {
strcpy ( filename , bx_filename ) ;
}
2003-05-03 20:37:18 +04:00
2012-07-24 23:15:00 +04:00
sprintf ( bochsrc_line , " ata0-master: type=disk, path= \" %s \" , mode=%s, cylinders= " FMT_LL " d, heads=%d, spt=%d " , filename , hdmode_choices [ mode ] , cyl , heads , spt ) ;
2003-05-03 20:37:18 +04:00
switch ( mode ) {
case 1 :
write_function = make_sparse_image ;
break ;
case 2 :
2003-08-01 05:20:00 +04:00
write_function = make_growing_image ;
2003-05-03 20:37:18 +04:00
break ;
default :
2004-08-19 20:03:03 +04:00
# ifdef WIN32
writefn_win32 = make_flat_image_win32 ;
# else
2003-05-03 20:37:18 +04:00
write_function = make_flat_image ;
2004-08-19 20:03:03 +04:00
# endif
2003-05-03 20:37:18 +04:00
}
2001-06-01 07:30:40 +04:00
} else {
2001-06-01 09:11:45 +04:00
int fdsize , cyl = 0 , heads = 0 , spt = 0 ;
2004-08-29 23:31:09 +04:00
if ( bx_interactive ) {
2009-04-14 13:45:22 +04:00
if ( ask_menu ( fdsize_menu , fdsize_n_choices , fdsize_choices , bx_fdsize_idx , & fdsize ) < 0 )
fatal ( EOF_ERR ) ;
2004-08-29 23:31:09 +04:00
} else {
fdsize = bx_fdsize_idx ;
}
2001-06-01 07:30:40 +04:00
switch ( fdsize ) {
2005-11-20 23:26:35 +03:00
case 0 : cyl = 40 ; heads = 1 ; spt = 8 ; break ; /* 0.16 meg */
case 1 : cyl = 40 ; heads = 1 ; spt = 9 ; break ; /* 0.18 meg */
case 2 : cyl = 40 ; heads = 2 ; spt = 8 ; break ; /* 0.32 meg */
case 3 : cyl = 40 ; heads = 2 ; spt = 9 ; break ; /* 0.36 meg */
case 4 : cyl = 80 ; heads = 2 ; spt = 9 ; break ; /* 0.72 meg */
case 5 : cyl = 80 ; heads = 2 ; spt = 15 ; break ; /* 1.2 meg */
case 6 : cyl = 80 ; heads = 2 ; spt = 18 ; break ; /* 1.44 meg */
case 7 : cyl = 80 ; heads = 2 ; spt = 21 ; break ; /* 1.68 meg */
case 8 : cyl = 82 ; heads = 2 ; spt = 21 ; break ; /* 1.72 meg */
case 9 : cyl = 80 ; heads = 2 ; spt = 36 ; break ; /* 2.88 meg */
default :
2009-04-14 13:45:22 +04:00
fatal ( " ERROR: fdsize out of range " ) ;
2001-06-01 07:30:40 +04:00
}
sectors = cyl * heads * spt ;
2009-04-14 13:45:22 +04:00
printf ( " I will create a floppy image with \n " ) ;
printf ( " cyl=%d \n " , cyl ) ;
printf ( " heads=%d \n " , heads ) ;
printf ( " sectors per track=%d \n " , spt ) ;
printf ( " total sectors= " FMT_LL " d \n " , sectors ) ;
printf ( " total bytes= " FMT_LL " d \n " , sectors * 512 ) ;
2004-08-29 23:31:09 +04:00
if ( bx_interactive ) {
if ( ! strlen ( bx_filename ) ) strcpy ( bx_filename , " a.img " ) ;
2009-04-14 13:45:22 +04:00
if ( ask_string ( " \n What should I name the image? \n " , bx_filename , filename ) < 0 )
fatal ( EOF_ERR ) ;
2004-08-29 23:31:09 +04:00
} else {
strcpy ( filename , bx_filename ) ;
}
2009-04-14 13:45:22 +04:00
sprintf ( bochsrc_line , " floppya: image= \" %s \" , status=inserted " , filename ) ;
2003-05-03 20:37:18 +04:00
write_function = make_flat_image ;
2001-06-01 07:30:40 +04:00
}
if ( sectors < 1 )
2009-04-14 13:45:22 +04:00
fatal ( " ERROR: Illegal disk size! " ) ;
2001-06-01 07:30:40 +04:00
if ( strlen ( filename ) < 1 )
2009-04-14 13:45:22 +04:00
fatal ( " ERROR: Illegal filename " ) ;
2004-08-19 20:03:03 +04:00
# ifdef WIN32
if ( writefn_win32 ! = NULL ) {
2009-04-14 13:45:22 +04:00
make_image_win32 ( sectors , filename , writefn_win32 ) ;
2004-08-19 20:03:03 +04:00
}
else
2004-08-24 15:42:18 +04:00
# endif
2004-08-19 20:03:03 +04:00
{
2009-04-14 13:45:22 +04:00
make_image ( sectors , filename , write_function ) ;
2004-08-19 20:03:03 +04:00
}
2009-04-14 13:45:22 +04:00
printf ( " \n I wrote " FMT_LL " u bytes to " , sectors * 512 ) ;
printf ( " %s. \n " , filename ) ;
printf ( " \n The following line should appear in your bochsrc: \n " ) ;
printf ( " %s \n " , bochsrc_line ) ;
2004-04-30 21:26:38 +04:00
# 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 ( ) ;
2005-01-28 13:25:06 +03:00
printf ( " (The line is stored in your windows clipboard, use CTRL-V to paste) \n " ) ;
2004-04-30 21:26:38 +04:00
}
# endif
2002-11-21 09:36:22 +03:00
myexit ( 0 ) ;
2002-11-26 14:21:31 +03:00
// make picky compilers (c++, gcc) happy,
2003-05-03 20:37:18 +04:00
// even though we leave via 'myexit' just above
2002-11-26 14:21:31 +03:00
return 0 ;
2001-06-01 07:30:40 +04:00
}