Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

440 lines
9.8 KiB

  1. /*++
  2. Copyright (c) 1997 Microsoft Corporation
  3. Module Name:
  4. bootstat.c
  5. Abstract:
  6. Manipulates the boot status data file.
  7. Author:
  8. Peter Wieland (peterwie) 01-18-01
  9. Revision History:
  10. --*/
  11. #include "bldr.h"
  12. #include "bootstatus.h"
  13. #include <stdio.h>
  14. #define FIELD_SIZE(type, field) sizeof(((type *)0)->field)
  15. #define FIELD_OFFSET_AND_SIZE(n) {FIELD_OFFSET(BSD_BOOT_STATUS_DATA, n), FIELD_SIZE(BSD_BOOT_STATUS_DATA, n)}
  16. VOID
  17. BlAutoAdvancedBoot(
  18. IN OUT PCHAR *LoadOptions,
  19. IN BSD_LAST_BOOT_STATUS LastBootStatus,
  20. IN ULONG AdvancedBootMode
  21. )
  22. {
  23. UCHAR bootStatusString[32];
  24. PUCHAR advancedBootString = NULL;
  25. ULONG newLoadOptionsLength;
  26. PUCHAR newLoadOptions;
  27. //
  28. // Write the last boot status into a string.
  29. //
  30. sprintf(bootStatusString, "LastBootStatus=%d", LastBootStatus);
  31. //
  32. // Based on the advanced boot mode indicated by the caller, adjust the
  33. // boot options.
  34. //
  35. if (AdvancedBootMode != -1) {
  36. advancedBootString = BlGetAdvancedBootLoadOptions(AdvancedBootMode);
  37. }
  38. //
  39. // Determine the length of the new load options string.
  40. //
  41. newLoadOptionsLength = strlen(bootStatusString) + 1;
  42. if(*LoadOptions != NULL) {
  43. newLoadOptionsLength += strlen(*LoadOptions) + 1;
  44. }
  45. if(advancedBootString) {
  46. newLoadOptionsLength += strlen(advancedBootString) + 1;
  47. }
  48. newLoadOptions = BlAllocateHeap(newLoadOptionsLength * sizeof(UCHAR));
  49. if(newLoadOptions == NULL) {
  50. return;
  51. }
  52. //
  53. // Concatenate all the strings together.
  54. //
  55. sprintf(newLoadOptions, "%s %s %s",
  56. ((*LoadOptions != NULL) ? *LoadOptions : ""),
  57. ((advancedBootString != NULL) ? advancedBootString : ""),
  58. bootStatusString);
  59. if(AdvancedBootMode != -1) {
  60. BlDoAdvancedBootLoadProcessing(AdvancedBootMode);
  61. }
  62. *LoadOptions = newLoadOptions;
  63. return;
  64. }
  65. ARC_STATUS
  66. BlGetSetBootStatusData(
  67. IN PVOID DataHandle,
  68. IN BOOLEAN Get,
  69. IN RTL_BSD_ITEM_TYPE DataItem,
  70. IN PVOID DataBuffer,
  71. IN ULONG DataBufferLength,
  72. OUT PULONG BytesReturned OPTIONAL
  73. )
  74. {
  75. ULONG fileId = (ULONG) ((ULONG_PTR) DataHandle);
  76. struct {
  77. ULONG FieldOffset;
  78. ULONG FieldLength;
  79. } bootStatusFields[] = {
  80. FIELD_OFFSET_AND_SIZE(Version),
  81. FIELD_OFFSET_AND_SIZE(ProductType),
  82. FIELD_OFFSET_AND_SIZE(AutoAdvancedBoot),
  83. FIELD_OFFSET_AND_SIZE(AdvancedBootMenuTimeout),
  84. FIELD_OFFSET_AND_SIZE(LastBootSucceeded),
  85. FIELD_OFFSET_AND_SIZE(LastBootShutdown)
  86. };
  87. ULONG dataFileVersion;
  88. LARGE_INTEGER fileOffset;
  89. ULONG itemLength;
  90. ULONG bytesRead;
  91. ARC_STATUS status;
  92. ASSERT(RtlBsdItemMax == (sizeof(bootStatusFields) / sizeof(bootStatusFields[0])));
  93. //
  94. // Read the version number out of the data file.
  95. //
  96. fileOffset.QuadPart = 0;
  97. status = BlSeek(fileId, &fileOffset, SeekAbsolute);
  98. if(status != ESUCCESS) {
  99. return status;
  100. }
  101. status = BlRead(fileId,
  102. &dataFileVersion,
  103. sizeof(ULONG),
  104. &bytesRead);
  105. if(status != ESUCCESS) {
  106. return status;
  107. }
  108. //
  109. // If the data item requsted isn't one we have code to handle then
  110. // return invalid parameter.
  111. //
  112. if(DataItem >= (sizeof(bootStatusFields) / sizeof(bootStatusFields[0]))) {
  113. return EINVAL;
  114. }
  115. fileOffset.QuadPart = bootStatusFields[DataItem].FieldOffset;
  116. itemLength = bootStatusFields[DataItem].FieldLength;
  117. //
  118. // If the data item offset is beyond the end of the file then return a
  119. // versioning error.
  120. //
  121. if((fileOffset.QuadPart + itemLength) > dataFileVersion) {
  122. return STATUS_REVISION_MISMATCH;
  123. }
  124. if(DataBufferLength < itemLength) {
  125. DataBufferLength = itemLength;
  126. return EINVAL;
  127. }
  128. status = BlSeek(fileId, &fileOffset, SeekAbsolute);
  129. if(status != ESUCCESS) {
  130. return status;
  131. }
  132. if(Get) {
  133. status = BlRead(fileId,
  134. DataBuffer,
  135. itemLength,
  136. &bytesRead);
  137. } else {
  138. status = BlWrite(fileId,
  139. DataBuffer,
  140. itemLength,
  141. &bytesRead);
  142. }
  143. if((status == ESUCCESS) && ARGUMENT_PRESENT(BytesReturned)) {
  144. *BytesReturned = bytesRead;
  145. }
  146. return status;
  147. }
  148. ARC_STATUS
  149. BlLockBootStatusData(
  150. IN ULONG SystemPartitionId,
  151. IN PCHAR SystemPartition,
  152. IN PCHAR SystemDirectory,
  153. OUT PVOID *DataHandle
  154. )
  155. /*++
  156. Routine Description:
  157. This routine opens the boot status data file.
  158. Arguments:
  159. SystemPartitionId - if non-zero this is the arc file id of the system
  160. partition. This will be used to locate the system
  161. directory instead of the system partition name (below).
  162. SystemPartition - the arc name of the system partition. Ignored if
  163. SystemPartitionId is non-zero.
  164. SystemDirectory - the name of the system directory.
  165. DataHandle - returns a handle to the boot status data.
  166. Return Value:
  167. ESUCCESS if the status data could be locked, or error indicating why not.
  168. --*/
  169. {
  170. ULONG driveId;
  171. UCHAR filePath[100];
  172. ULONG fileId;
  173. ARC_STATUS status;
  174. if(SystemPartitionId == 0) {
  175. //
  176. // Attempt to open the system partition
  177. //
  178. status = ArcOpen(SystemPartition, ArcOpenReadWrite, &driveId);
  179. if(status != ESUCCESS) {
  180. return status;
  181. }
  182. } else {
  183. driveId = SystemPartitionId;
  184. }
  185. //
  186. // Now attempt to open the file <SystemDirectory>\bootstat.dat
  187. //
  188. strcpy(filePath, SystemDirectory);
  189. strcat(filePath, BSD_FILE_NAME);
  190. status = BlOpen(driveId, filePath, ArcOpenReadWrite, &fileId);
  191. if(SystemPartitionId == 0) {
  192. //
  193. // Close the drive.
  194. //
  195. ArcClose(driveId);
  196. }
  197. //
  198. // The file doesn't exist so we don't know the state of the last boot.
  199. //
  200. if(status != ESUCCESS) {
  201. return status;
  202. }
  203. *DataHandle = (PVOID) ((ULONG_PTR) fileId);
  204. return ESUCCESS;
  205. }
  206. VOID
  207. BlUnlockBootStatusData(
  208. IN PVOID DataHandle
  209. )
  210. {
  211. ULONG fileId = (ULONG) ((ULONG_PTR) DataHandle);
  212. BlClose(fileId);
  213. return;
  214. }
  215. ULONG
  216. BlGetLastBootStatus(
  217. IN PVOID DataHandle,
  218. OUT BSD_LAST_BOOT_STATUS *LastBootStatus
  219. )
  220. {
  221. UCHAR lastBootGood;
  222. UCHAR lastShutdownGood;
  223. UCHAR aabEnabled;
  224. ULONG advancedBootMode = -1;
  225. ARC_STATUS status;
  226. *LastBootStatus = BsdLastBootGood;
  227. //
  228. // The file contains a simple data structure so i can avoid parsing an
  229. // INI file. If this proves to be insufficient for policy management then
  230. // we'll change it into an ini file.
  231. //
  232. //
  233. // Read the last boot status.
  234. //
  235. status = BlGetSetBootStatusData(DataHandle,
  236. TRUE,
  237. RtlBsdItemBootGood,
  238. &lastBootGood,
  239. sizeof(UCHAR),
  240. NULL);
  241. if(status != ESUCCESS) {
  242. *LastBootStatus = BsdLastBootUnknown;
  243. return advancedBootMode;
  244. }
  245. status = BlGetSetBootStatusData(DataHandle,
  246. TRUE,
  247. RtlBsdItemBootShutdown,
  248. &lastShutdownGood,
  249. sizeof(UCHAR),
  250. NULL);
  251. if(status != ESUCCESS) {
  252. *LastBootStatus = BsdLastBootUnknown;
  253. return advancedBootMode;
  254. }
  255. status = BlGetSetBootStatusData(DataHandle,
  256. TRUE,
  257. RtlBsdItemAabEnabled,
  258. &aabEnabled,
  259. sizeof(UCHAR),
  260. NULL);
  261. if(status != ESUCCESS) {
  262. *LastBootStatus = BsdLastBootUnknown;
  263. return advancedBootMode;
  264. }
  265. //
  266. // If the system was shutdown cleanly then don't bother to check if the
  267. // boot was good.
  268. //
  269. if(lastShutdownGood) {
  270. return advancedBootMode;
  271. }
  272. //
  273. // Determine the last boot status & what action to take.
  274. //
  275. if(lastBootGood == FALSE) {
  276. //
  277. // Enable last known good.
  278. //
  279. advancedBootMode = 6;
  280. *LastBootStatus = BsdLastBootFailed;
  281. } else if(lastShutdownGood == FALSE) {
  282. //
  283. // Enable safe mode without networking.
  284. //
  285. advancedBootMode = 0;
  286. *LastBootStatus = BsdLastBootNotShutdown;
  287. }
  288. //
  289. // Now disable auto safemode actions if requested.
  290. //
  291. if(aabEnabled == FALSE) {
  292. advancedBootMode = -1;
  293. }
  294. return advancedBootMode;
  295. }
  296. VOID
  297. BlWriteBootStatusFlags(
  298. IN ULONG SystemPartitionId,
  299. IN PUCHAR SystemDirectory,
  300. IN BOOLEAN LastBootSucceeded,
  301. IN BOOLEAN LastBootShutdown
  302. )
  303. {
  304. PVOID dataHandle;
  305. ARC_STATUS status;
  306. status = BlLockBootStatusData(SystemPartitionId,
  307. NULL,
  308. SystemDirectory,
  309. &dataHandle);
  310. if(status == ESUCCESS) {
  311. BlGetSetBootStatusData(dataHandle,
  312. FALSE,
  313. RtlBsdItemBootGood,
  314. &LastBootSucceeded,
  315. sizeof(UCHAR),
  316. NULL);
  317. BlGetSetBootStatusData(dataHandle,
  318. FALSE,
  319. RtlBsdItemBootShutdown,
  320. &LastBootShutdown,
  321. sizeof(UCHAR),
  322. NULL);
  323. BlUnlockBootStatusData(dataHandle);
  324. }
  325. return;
  326. }