FIPSSPEC.CPP
上传用户:datang2001
上传日期:2007-02-01
资源大小:53269k
文件大小:7k
源码类别:

操作系统开发

开发平台:

C/C++

  1. /*
  2. FIPS - the First nondestructive Interactive Partition Splitting program
  3. Module fipsspec.cpp
  4. RCS - Header:
  5. $Header: c:/daten/fips/source/main/RCS/fipsspec.cpp 1.4 1995/01/19 00:00:53 schaefer Exp schaefer $
  6. Copyright (C) 1993 Arno Schaefer
  7. This program is free software; you can redistribute it and/or modify
  8. it under the terms of the GNU General Public License as published by
  9. the Free Software Foundation; either version 2 of the License, or
  10. (at your option) any later version.
  11. This program is distributed in the hope that it will be useful,
  12. but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  14. GNU General Public License for more details.
  15. You should have received a copy of the GNU General Public License
  16. along with this program; if not, write to the Free Software
  17. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  18. Report problems and direct all questions to:
  19. schaefer@rbg.informatik.th-darmstadt.de
  20. */
  21. #include <dos.h>
  22. #include "fipsspec.h"
  23. #include "global.h"
  24. #include "input.h"
  25. #define FIRST_CHECK false
  26. #define FINAL_CHECK true
  27. #define DISK_INT 0x13
  28. #define RESET_DISK 0
  29. #define GET_DRIVE_PARAMS 8
  30. void fips_bpb::print (void)
  31. {
  32. printx ("Bytes per sector: %un",bytes_per_sector);
  33. printx ("Sectors per cluster: %un",sectors_per_cluster);
  34. printx ("Reserved sectors: %un",reserved_sectors);
  35. printx ("Number of FATs: %un",no_of_fats);
  36. printx ("Number of rootdirectory entries: %un",no_of_rootdir_entries);
  37. printx ("Number of sectors (short): %un",no_of_sectors);
  38. printx ("Media descriptor byte: %02Xhn",media_descriptor);
  39. printx ("Sectors per FAT: %un",sectors_per_fat);
  40. printx ("Sectors per track: %un",sectors_per_track);
  41. printx ("Drive heads: %un",drive_heads);
  42. printx ("Hidden sectors: %lun",hidden_sectors);
  43. printx ("Number of sectors (long): %lun",no_of_sectors_long);
  44. printx ("Physical drive number: %02Xhn",phys_drive_no);
  45. printx ("Signature: %02Xhnn",signature);
  46. }
  47. void fips_partition_table::print (void)
  48. {
  49. printx ("     |        |     Start      |      |      End       | Start  |Number of|n");
  50. printx ("Part.|bootable|Head Cyl. Sector|System|Head Cyl. Sector| Sector |Sectors  |  MBn");
  51. printx ("-----+--------+----------------+------+----------------+--------+---------+----n");
  52. for (int i=0;i<4;i++)
  53. {
  54. printx ("%u    |    %s |%4u %4u   %4u|   %02Xh|%4u %4u   %4u|%8lu| %8lu|%4lun",i+1,
  55. partition_info[i].bootable ? "yes" : " no",
  56. partition_info[i].start_head,partition_info[i].start_cylinder,partition_info[i].start_sector,
  57. partition_info[i].system,partition_info[i].end_head,partition_info[i].end_cylinder,partition_info[i].end_sector,
  58. partition_info[i].start_sector_abs,partition_info[i].no_of_sectors_abs,partition_info[i].no_of_sectors_abs / 2048);
  59. }
  60. }
  61. void fips_harddrive::get_geometry (void)
  62. {
  63. union REGS regs;
  64. regs.h.ah = GET_DRIVE_PARAMS;
  65. regs.h.dl = number;
  66. int86 (DISK_INT,&regs,&regs);
  67. if (global.debug_mode)
  68. {
  69. fprintf (global.debugfile,"nRegisters after call to int 13h 08h (drive %02Xh):nn",number);
  70. fprintf (global.debugfile,"   00       sc/cl    hdn");
  71. fprintf (global.debugfile,"al ah bl bh cl ch dl dh   si    di    cflgs flagsn");
  72. hexwrite ((byte *) &regs,16,global.debugfile);
  73. }
  74. if ((errorcode = regs.h.ah) != 0) return;
  75. geometry.heads = (dword) regs.h.dh + 1;
  76. geometry.sectors = (dword) regs.h.cl & 0x3f;
  77. geometry.cylinders = ((dword) regs.h.ch | (((dword) regs.h.cl << 2) & 0x300)) + 1;
  78. if (global.debug_mode)
  79. {
  80. fprintf (global.debugfile, "nGeometry reported by BIOS:n");
  81. fprintf
  82. (
  83. global.debugfile,
  84. "%ld cylinders, %ld heads, %ld sectorsn",
  85. geometry.cylinders,
  86. geometry.heads,
  87. geometry.sectors
  88. );
  89. }
  90. }
  91. void fips_harddrive::reset (void)
  92. {
  93. union REGS regs;
  94. regs.h.ah = RESET_DISK;
  95. regs.h.dl = number;
  96. int86 (DISK_INT,&regs,&regs);
  97. if (global.debug_mode)
  98. {
  99. fprintf (global.debugfile,"nRegisters after call to int 13h 00h (drive %02Xh):nn",number);
  100. fprintf (global.debugfile,"al ah bl bh cl ch dl dh   si    di    cflgs flagsn");
  101. hexwrite ((byte *) &regs,16,global.debugfile);
  102. }
  103. errorcode = regs.h.ah;
  104. }
  105. void fips_logdrive_info::put_debug_info (void)
  106. {
  107. fprintf (global.debugfile,"Calculated Partition Characteristica:nn");
  108. fprintf (global.debugfile,"Start of FAT 1: %lun",start_fat1);
  109. fprintf (global.debugfile,"Start of FAT 2: %lun",start_fat2);
  110. fprintf (global.debugfile,"Start of Rootdirectory: %lun",start_rootdir);
  111. fprintf (global.debugfile,"Start of Data: %lun",start_data);
  112. fprintf (global.debugfile,"Number of Clusters: %lun",no_of_clusters);
  113. }
  114. dword fips_partition::min_cylinder (fat16 fat, drive_geometry geometry)
  115. {
  116. dword new_part_min_sector =
  117. info().start_data
  118. + (dword) 4085
  119. * bpb().sectors_per_cluster;
  120. dword new_part_min_cylinder =
  121. (
  122. new_part_min_sector
  123. + partition_info->start_sector_abs
  124. - 1
  125. )
  126. / (geometry.heads * geometry.sectors)
  127. + 1;
  128. if (new_part_min_cylinder > partition_info->end_cylinder)
  129. error ("Partition too small - can't split");
  130. dword min_free_cluster = fat.min_free_cluster ();
  131. dword min_free_sector =
  132. info().start_data
  133. + (min_free_cluster - 2)
  134. * (dword) bpb().sectors_per_cluster;
  135. dword min_free_cylinder =
  136. (
  137. min_free_sector
  138. + partition_info->start_sector_abs
  139. - 1
  140. )
  141. / (geometry.heads * geometry.sectors)
  142. + 1;
  143. if (min_free_cylinder > partition_info->end_cylinder)
  144. error ("Last cylinder is not free");
  145. if (new_part_min_cylinder < min_free_cylinder)
  146. new_part_min_cylinder = min_free_cylinder;
  147. return (new_part_min_cylinder);
  148. }
  149. boolean fips_partition::split (fips_harddrive hd)
  150. {
  151. if (read_boot_sector ())
  152. error ("Error reading boot sector");
  153. if (global.debug_mode)
  154. {
  155. fprintf
  156. (
  157. global.debugfile,
  158. "nBoot sector drive %02Xh, partition %u:nn",
  159. hd.number,
  160. number + 1
  161. );
  162. hexwrite
  163. (
  164. boot_sector->data,
  165. 512,
  166. global.debugfile
  167. );
  168. }
  169. get_bpb ();
  170. printx ("nBoot sector:nn");
  171. print_bpb ();
  172. get_info ();
  173. if (global.debug_mode)
  174. write_info_debugfile ();
  175. check ();
  176. fat16 fat1 (this,1);
  177. fat16 fat2 (this,2);
  178. fat1.check_against (&fat2);
  179. dword new_part_min_cylinder =
  180. min_cylinder (fat2, hd.geometry);
  181. if (ask_if_save()) save_root_and_boot(&hd,this);
  182. dword new_start_cylinder =
  183. ask_for_new_start_cylinder
  184. (
  185. partition_info->start_cylinder,
  186. new_part_min_cylinder,
  187. partition_info->end_cylinder,
  188. hd.geometry.heads * hd.geometry.sectors
  189. );
  190. fat2.check_empty
  191. (
  192. new_start_cylinder
  193. * hd.geometry.heads
  194. * hd.geometry.sectors
  195. - partition_info->start_sector_abs
  196. );
  197. hd.calculate_new_root (new_start_cylinder, this);
  198. hd.put_partition_table();
  199. hd.get_partition_table();
  200. printx ("nNew partition table:nn");
  201. hd.print_partition_table ();
  202. hd.check (FINAL_CHECK);
  203. if (ask_if_continue () == false)
  204. {
  205. return (false);
  206. }
  207. calculate_new_boot ();
  208. put_bpb ();
  209. get_bpb ();
  210. printx ("nNew boot sector:nn");
  211. print_bpb ();
  212. get_info ();
  213. if (global.debug_mode)
  214. write_info_debugfile ();
  215. check();
  216. if (!global.test_mode)
  217. {
  218. ask_for_write_permission ();
  219. if (hd.write_root_sector ())
  220. error ("Error writing root sector");
  221. if (write_boot_sector ())
  222. error ("Error writing boot sector");
  223. printx ("Repartitioning completen");
  224. }
  225. return (true);
  226. }