2011-11-27 02:09:45 +00:00
|
|
|
/*
|
|
|
|
* Rufus: The Resourceful USB Formatting Utility
|
|
|
|
* MS-DOS boot file extraction, from the FAT12 floppy image in diskcopy.dll
|
|
|
|
* Copyright (c) 2011 Pete Batard <pete@akeo.ie>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Memory leaks detection - define _CRTDBG_MAP_ALLOC as preprocessor macro */
|
|
|
|
#ifdef _CRTDBG_MAP_ALLOC
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <crtdbg.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "rufus.h"
|
|
|
|
#include "msdos.h"
|
|
|
|
|
|
|
|
static BYTE* DiskImage;
|
|
|
|
static size_t DiskImageSize;
|
|
|
|
|
2011-11-30 17:42:25 +00:00
|
|
|
/*
|
|
|
|
* FAT time conversion, from ReactOS' time.c
|
|
|
|
*/
|
|
|
|
#define TICKSPERMIN 600000000
|
|
|
|
#define TICKSPERSEC 10000000
|
|
|
|
#define TICKSPERMSEC 10000
|
|
|
|
#define SECSPERDAY 86400
|
|
|
|
#define SECSPERHOUR 3600
|
|
|
|
#define SECSPERMIN 60
|
|
|
|
#define MINSPERHOUR 60
|
|
|
|
#define HOURSPERDAY 24
|
|
|
|
#define EPOCHWEEKDAY 1
|
|
|
|
#define DAYSPERWEEK 7
|
|
|
|
#define EPOCHYEAR 1601
|
|
|
|
#define DAYSPERNORMALYEAR 365
|
|
|
|
#define DAYSPERLEAPYEAR 366
|
|
|
|
#define MONSPERYEAR 12
|
|
|
|
|
|
|
|
typedef struct _TIME_FIELDS {
|
|
|
|
short Year;
|
|
|
|
short Month;
|
|
|
|
short Day;
|
|
|
|
short Hour;
|
|
|
|
short Minute;
|
|
|
|
short Second;
|
|
|
|
short Milliseconds;
|
|
|
|
short Weekday;
|
|
|
|
} TIME_FIELDS, *PTIME_FIELDS;
|
|
|
|
|
|
|
|
#define ARGUMENT_PRESENT(ArgumentPointer) \
|
|
|
|
((CHAR*)((ULONG_PTR)(ArgumentPointer)) != (CHAR*)NULL)
|
|
|
|
|
|
|
|
static const int YearLengths[2] =
|
|
|
|
{
|
|
|
|
DAYSPERNORMALYEAR, DAYSPERLEAPYEAR
|
|
|
|
};
|
|
|
|
static const UCHAR MonthLengths[2][MONSPERYEAR] =
|
|
|
|
{
|
|
|
|
{ 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },
|
|
|
|
{ 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }
|
|
|
|
};
|
|
|
|
|
|
|
|
static __inline int IsLeapYear(int Year)
|
|
|
|
{
|
|
|
|
return Year % 4 == 0 && (Year % 100 != 0 || Year % 400 == 0) ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int DaysSinceEpoch(int Year)
|
|
|
|
{
|
|
|
|
int Days;
|
|
|
|
Year--; /* Don't include a leap day from the current year */
|
|
|
|
Days = Year * DAYSPERNORMALYEAR + Year / 4 - Year / 100 + Year / 400;
|
|
|
|
Days -= (EPOCHYEAR - 1) * DAYSPERNORMALYEAR + (EPOCHYEAR - 1) / 4 - (EPOCHYEAR - 1) / 100 + (EPOCHYEAR - 1) / 400;
|
|
|
|
return Days;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOLEAN RtlTimeFieldsToTime(PTIME_FIELDS TimeFields, PLARGE_INTEGER Time)
|
|
|
|
{
|
|
|
|
int CurMonth;
|
|
|
|
TIME_FIELDS IntTimeFields;
|
|
|
|
|
|
|
|
memcpy(&IntTimeFields,
|
|
|
|
TimeFields,
|
|
|
|
sizeof(TIME_FIELDS));
|
|
|
|
|
|
|
|
if (TimeFields->Milliseconds < 0 || TimeFields->Milliseconds > 999 ||
|
|
|
|
TimeFields->Second < 0 || TimeFields->Second > 59 ||
|
|
|
|
TimeFields->Minute < 0 || TimeFields->Minute > 59 ||
|
|
|
|
TimeFields->Hour < 0 || TimeFields->Hour > 23 ||
|
|
|
|
TimeFields->Month < 1 || TimeFields->Month > 12 ||
|
|
|
|
TimeFields->Day < 1 ||
|
|
|
|
TimeFields->Day >
|
|
|
|
MonthLengths[IsLeapYear(TimeFields->Year)][TimeFields->Month - 1] ||
|
|
|
|
TimeFields->Year < 1601) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compute the time */
|
|
|
|
Time->QuadPart = DaysSinceEpoch(IntTimeFields.Year);
|
|
|
|
for (CurMonth = 1; CurMonth < IntTimeFields.Month; CurMonth++) {
|
|
|
|
Time->QuadPart += MonthLengths[IsLeapYear(IntTimeFields.Year)][CurMonth - 1];
|
|
|
|
}
|
|
|
|
Time->QuadPart += IntTimeFields.Day - 1;
|
|
|
|
Time->QuadPart *= SECSPERDAY;
|
|
|
|
Time->QuadPart += IntTimeFields.Hour * SECSPERHOUR + IntTimeFields.Minute * SECSPERMIN +
|
|
|
|
IntTimeFields.Second;
|
|
|
|
Time->QuadPart *= TICKSPERSEC;
|
|
|
|
Time->QuadPart += IntTimeFields.Milliseconds * TICKSPERMSEC;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void FatDateTimeToSystemTime(PLARGE_INTEGER SystemTime, PFAT_DATETIME FatDateTime, UCHAR TenMs OPTIONAL)
|
|
|
|
{
|
|
|
|
TIME_FIELDS TimeFields;
|
|
|
|
|
|
|
|
/* Setup time fields */
|
|
|
|
TimeFields.Year = FatDateTime->Date.Year + 1980;
|
|
|
|
TimeFields.Month = FatDateTime->Date.Month;
|
|
|
|
TimeFields.Day = FatDateTime->Date.Day;
|
|
|
|
TimeFields.Hour = FatDateTime->Time.Hour;
|
|
|
|
TimeFields.Minute = FatDateTime->Time.Minute;
|
|
|
|
TimeFields.Second = (FatDateTime->Time.DoubleSeconds << 1);
|
|
|
|
|
|
|
|
/* Adjust up to 10 milliseconds
|
|
|
|
* if the parameter was supplied
|
|
|
|
*/
|
|
|
|
if (ARGUMENT_PRESENT(TenMs)) {
|
|
|
|
TimeFields.Second += TenMs / 100;
|
|
|
|
TimeFields.Milliseconds = (TenMs % 100) * 10;
|
|
|
|
} else {
|
|
|
|
TimeFields.Milliseconds = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fix seconds value that might get beyond the bound */
|
|
|
|
if (TimeFields.Second > 59) TimeFields.Second = 0;
|
|
|
|
|
|
|
|
/* Perform ceonversion to system time if possible */
|
|
|
|
if (!RtlTimeFieldsToTime(&TimeFields, SystemTime)) {
|
|
|
|
/* Set to default time if conversion failed */
|
|
|
|
SystemTime->QuadPart = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* http://www.multiboot.ru/msdos8.htm & http://en.wikipedia.org/wiki/Windows_Me#Real_mode_DOS
|
|
|
|
* COMMAND.COM and IO.SYS from diskcopy.dll are from the WinME crippled version
|
|
|
|
* that removed real mode DOS => they must be patched:
|
|
|
|
* IO.SYS 000003AA 75 -> EB
|
|
|
|
* COMMAND.COM 00006510 75 -> EB
|
|
|
|
*/
|
|
|
|
static BOOL Patch_COMMAND_COM(size_t filestart, size_t filesize)
|
|
|
|
{
|
|
|
|
const BYTE expected[8] = { 0x15, 0x80, 0xFA, 0x03, 0x75, 0x10, 0xB8, 0x0E };
|
|
|
|
|
|
|
|
uprintf("Patching COMMAND.COM...\n");
|
|
|
|
if (filesize != 93040) {
|
|
|
|
uprintf(" unexpected file size\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (memcmp(&DiskImage[filestart+0x650c], expected, sizeof(expected)) != 0) {
|
|
|
|
uprintf(" unexpected binary data\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
DiskImage[filestart+0x6510] = 0xeb;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL Patch_IO_SYS(size_t filestart, size_t filesize)
|
|
|
|
{
|
|
|
|
const BYTE expected[8] = { 0xFA, 0x80, 0x75, 0x09, 0x8D, 0xB6, 0x99, 0x00 };
|
|
|
|
|
|
|
|
uprintf("Patching IO.SYS...\n");
|
|
|
|
if (filesize != 116736) {
|
|
|
|
uprintf(" unexpected file size\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (memcmp(&DiskImage[filestart+0x3a8], expected, sizeof(expected)) != 0) {
|
|
|
|
uprintf(" unexpected binary data\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
DiskImage[filestart+0x3aa] = 0xeb;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2011-11-27 02:09:45 +00:00
|
|
|
/* Extract the file identified by FAT RootDir index 'entry' to 'path' */
|
|
|
|
static BOOL ExtractFAT(int entry, const char* path)
|
|
|
|
{
|
2011-11-30 17:42:25 +00:00
|
|
|
HANDLE hFile;
|
|
|
|
DWORD Size;
|
2011-11-27 02:09:45 +00:00
|
|
|
char filename[MAX_PATH];
|
2011-11-30 17:42:25 +00:00
|
|
|
size_t i, pos, fnamepos;
|
|
|
|
size_t filestart, filesize;
|
|
|
|
FAT_DATETIME LastAccessTime;
|
|
|
|
LARGE_INTEGER liCreationTime, liLastAccessTime, liLastWriteTime;
|
|
|
|
FILETIME ftCreationTime, ftLastAccessTime, ftLastWriteTime;
|
|
|
|
PDIR_ENTRY dir_entry = (PDIR_ENTRY)&DiskImage[FAT12_ROOTDIR_OFFSET + entry*FAT_BYTES_PER_DIRENT];
|
2011-11-27 02:09:45 +00:00
|
|
|
|
|
|
|
if ((path == NULL) || ((safe_strlen(path) + 14) > sizeof(filename))) {
|
|
|
|
uprintf("invalid path supplied for MS-DOS FAT extraction\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
strcpy(filename, path);
|
|
|
|
pos = strlen(path);
|
|
|
|
filename[pos++] = '\\';
|
2011-11-30 17:42:25 +00:00
|
|
|
fnamepos = pos;
|
2011-11-27 02:09:45 +00:00
|
|
|
|
|
|
|
for(i=0; i<8; i++) {
|
2011-11-30 17:42:25 +00:00
|
|
|
if (dir_entry->FileName[i] == ' ')
|
2011-11-27 02:09:45 +00:00
|
|
|
break;
|
2011-11-30 17:42:25 +00:00
|
|
|
filename[pos++] = dir_entry->FileName[i];
|
2011-11-27 02:09:45 +00:00
|
|
|
}
|
|
|
|
filename[pos++] = '.';
|
|
|
|
for (i=8; i<11; i++) {
|
2011-11-30 17:42:25 +00:00
|
|
|
if (dir_entry->FileName[i] == ' ')
|
2011-11-27 02:09:45 +00:00
|
|
|
break;
|
2011-11-30 17:42:25 +00:00
|
|
|
filename[pos++] = dir_entry->FileName[i];
|
2011-11-27 02:09:45 +00:00
|
|
|
}
|
|
|
|
filename[pos] = 0;
|
2011-11-30 17:42:25 +00:00
|
|
|
filestart = (dir_entry->FirstCluster + FAT12_CLUSTER_OFFSET)*FAT12_CLUSTER_SIZE;
|
|
|
|
filesize = dir_entry->FileSize;
|
2011-11-27 02:09:45 +00:00
|
|
|
if ((filestart + filesize) > DiskImageSize) {
|
2011-11-30 17:42:25 +00:00
|
|
|
uprintf("FAT File %s would be out of bounds: %X, %X\n", filename, filestart, filesize);
|
|
|
|
uprintf("%X, %X\n", dir_entry->FirstCluster, dir_entry->FileSize);
|
2011-11-27 02:09:45 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2011-11-30 17:42:25 +00:00
|
|
|
/* WinME DOS files need to be patched */
|
|
|
|
if (strcmp(&filename[fnamepos], "COMMAND.COM") == 0) {
|
|
|
|
Patch_COMMAND_COM(filestart, filesize);
|
|
|
|
} else if (strcmp(&filename[fnamepos], "IO.SYS") == 0) {
|
|
|
|
Patch_IO_SYS(filestart, filesize);
|
2011-11-27 02:09:45 +00:00
|
|
|
}
|
|
|
|
|
2011-11-30 17:42:25 +00:00
|
|
|
/* Create a file, using the same attributes as found in the FAT */
|
|
|
|
hFile = CreateFileA(filename, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE,
|
|
|
|
NULL, CREATE_ALWAYS, dir_entry->Attributes, 0);
|
|
|
|
if (hFile == INVALID_HANDLE_VALUE) {
|
|
|
|
uprintf("Unable to create file '%s': %s.\n", filename, WindowsErrorString());
|
2011-11-27 02:09:45 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2011-11-30 17:42:25 +00:00
|
|
|
if ((!WriteFile(hFile, &DiskImage[filestart], (DWORD)filesize, &Size, 0)) || (filesize != Size)) {
|
|
|
|
uprintf("Couldn't write file '%s': %s.\n", filename, WindowsErrorString());
|
|
|
|
safe_closehandle(hFile);
|
|
|
|
return FALSE; safe_closehandle(hFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Restore timestamps from FAT */
|
|
|
|
FatDateTimeToSystemTime(&liCreationTime, &dir_entry->CreationDateTime, dir_entry->CreationTimeTenMs);
|
|
|
|
ftCreationTime.dwHighDateTime = liCreationTime.HighPart;
|
|
|
|
ftCreationTime.dwLowDateTime = liCreationTime.LowPart;
|
|
|
|
LastAccessTime.Value = 0;
|
|
|
|
LastAccessTime.Date = dir_entry->LastAccessDate;
|
|
|
|
FatDateTimeToSystemTime(&liLastAccessTime, &LastAccessTime, 0);
|
|
|
|
ftLastAccessTime.dwHighDateTime = liLastAccessTime.HighPart;
|
|
|
|
ftLastAccessTime.dwLowDateTime = liLastAccessTime.LowPart;
|
|
|
|
FatDateTimeToSystemTime(&liLastWriteTime, &dir_entry->LastWriteDateTime, 0);
|
|
|
|
ftLastWriteTime.dwHighDateTime = liLastWriteTime.HighPart;
|
|
|
|
ftLastWriteTime.dwLowDateTime = liLastWriteTime.LowPart;
|
|
|
|
if (!SetFileTime(hFile, &ftCreationTime, &ftLastAccessTime, &ftLastWriteTime)) {
|
|
|
|
uprintf("Could not set timestamps: %s\n", WindowsErrorString());
|
|
|
|
}
|
|
|
|
|
|
|
|
safe_closehandle(hFile);
|
2011-11-27 02:09:45 +00:00
|
|
|
uprintf("Succesfully wrote '%s' (%d bytes)\n", filename, filesize);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extract the MS-DOS files contained in the FAT12 1.4MB floppy
|
|
|
|
image included as resource "BINFILE" in diskcopy.dll */
|
|
|
|
BOOL ExtractMSDOS(const char* path)
|
|
|
|
{
|
|
|
|
char dllname[MAX_PATH] = "C:\\Windows\\System32";
|
|
|
|
int i, j;
|
|
|
|
HMODULE hDLL;
|
|
|
|
HRSRC hDiskImage;
|
|
|
|
|
|
|
|
// TODO: optionally extract some more, including "deleted" entries
|
|
|
|
char* extractlist[] = {"MSDOS SYS", "COMMAND COM", "IO SYS"};
|
|
|
|
|
|
|
|
GetSystemDirectoryA(dllname, sizeof(dllname));
|
|
|
|
safe_strcat(dllname, sizeof(dllname), "\\diskcopy.dll");
|
|
|
|
hDLL = LoadLibraryA(dllname);
|
|
|
|
if (hDLL == NULL) {
|
|
|
|
uprintf("Unable to open %s: %s\n", dllname, WindowsErrorString());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
hDiskImage = FindResourceA(hDLL, MAKEINTRESOURCEA(1), "BINFILE");
|
|
|
|
if (hDiskImage == NULL) {
|
|
|
|
uprintf("Unable to locate disk image in %s: %s\n", dllname, WindowsErrorString());
|
|
|
|
FreeLibrary(hDLL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
DiskImage = (BYTE*)LockResource(LoadResource(hDLL, hDiskImage));
|
|
|
|
if (DiskImage == NULL) {
|
|
|
|
uprintf("Unable to access disk image in %s: %s\n", dllname, WindowsErrorString());
|
|
|
|
FreeLibrary(hDLL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
DiskImageSize = (size_t)SizeofResource(hDLL, hDiskImage);
|
|
|
|
// Sanity check
|
|
|
|
if (DiskImageSize < 700*1024) {
|
|
|
|
uprintf("MS-DOS disk image is too small (%d bytes)\n", dllname, DiskImageSize);
|
|
|
|
FreeLibrary(hDLL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2011-11-30 17:42:25 +00:00
|
|
|
for (i=0; i<FAT_FN_DIR_ENTRY_LAST; i++) {
|
|
|
|
if (DiskImage[FAT12_ROOTDIR_OFFSET + i*FAT_BYTES_PER_DIRENT] == FAT_DIRENT_DELETED)
|
2011-11-27 02:09:45 +00:00
|
|
|
continue;
|
|
|
|
for (j=0; j<ARRAYSIZE(extractlist); j++) {
|
2011-11-30 17:42:25 +00:00
|
|
|
if (memcmp(extractlist[j], &DiskImage[FAT12_ROOTDIR_OFFSET + i*FAT_BYTES_PER_DIRENT], 8+3) == 0) {
|
2011-11-27 02:09:45 +00:00
|
|
|
ExtractFAT(i, path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FreeLibrary(hDLL);
|
2011-11-30 17:42:25 +00:00
|
|
|
|
2011-11-27 02:09:45 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|