/* * ReactOS kernel * Copyright (C) 2002, 2003, 2004 ReactOS Team * * 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 2 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, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ /* * COPYRIGHT: See COPYING in the top level directory * PROJECT: ReactOS text-mode setup * FILE: subsys/system/usetup/usetup.c * PURPOSE: Text-mode setup * PROGRAMMER: Eric Kohl * Casper S. Hornstrup (chorns@users.sourceforge.net) * Hervé Poussineau (hpoussin@reactos.org) */ #include "usetup.h" #define NDEBUG #include /* GLOBALS ******************************************************************/ HANDLE ProcessHeap; UNICODE_STRING SourceRootPath; UNICODE_STRING SourceRootDir; UNICODE_STRING SourcePath; BOOLEAN IsUnattendedSetup = FALSE; LONG UnattendDestinationDiskNumber; LONG UnattendDestinationPartitionNumber; LONG UnattendMBRInstallType = -1; LONG UnattendFormatPartition = 0; LONG AutoPartition = 0; WCHAR UnattendInstallationDirectory[MAX_PATH]; PWCHAR SelectedLanguageId; WCHAR LocaleID[9]; WCHAR DefaultLanguage[20]; WCHAR DefaultKBLayout[20]; BOOLEAN RepairUpdateFlag = FALSE; HANDLE hPnpThread = INVALID_HANDLE_VALUE; PPARTLIST PartitionList = NULL; /* LOCALS *******************************************************************/ static PFILE_SYSTEM_LIST FileSystemList = NULL; static UNICODE_STRING InstallPath; /* Path to the install directory */ static UNICODE_STRING DestinationPath; static UNICODE_STRING DestinationArcPath; static UNICODE_STRING DestinationRootPath; /* Path to the active partition (boot manager) */ static UNICODE_STRING SystemRootPath; static HINF SetupInf; static HSPFILEQ SetupFileQueue = NULL; static BOOLEAN WarnLinuxPartitions = TRUE; static PGENERIC_LIST ComputerList = NULL; static PGENERIC_LIST DisplayList = NULL; static PGENERIC_LIST KeyboardList = NULL; static PGENERIC_LIST LayoutList = NULL; static PGENERIC_LIST LanguageList = NULL; static LANGID LanguageId = 0; static ULONG RequiredPartitionDiskSpace = ~0; /* FUNCTIONS ****************************************************************/ static VOID PrintString(char* fmt,...) { char buffer[512]; va_list ap; UNICODE_STRING UnicodeString; ANSI_STRING AnsiString; va_start(ap, fmt); vsprintf(buffer, fmt, ap); va_end(ap); RtlInitAnsiString(&AnsiString, buffer); RtlAnsiStringToUnicodeString(&UnicodeString, &AnsiString, TRUE); NtDisplayString(&UnicodeString); RtlFreeUnicodeString(&UnicodeString); } static VOID DrawBox(IN SHORT xLeft, IN SHORT yTop, IN SHORT Width, IN SHORT Height) { COORD coPos; DWORD Written; /* draw upper left corner */ coPos.X = xLeft; coPos.Y = yTop; FillConsoleOutputCharacterA(StdOutput, 0xDA, // '+', 1, coPos, &Written); /* draw upper edge */ coPos.X = xLeft + 1; coPos.Y = yTop; FillConsoleOutputCharacterA(StdOutput, 0xC4, // '-', Width - 2, coPos, &Written); /* draw upper right corner */ coPos.X = xLeft + Width - 1; coPos.Y = yTop; FillConsoleOutputCharacterA(StdOutput, 0xBF, // '+', 1, coPos, &Written); /* Draw right edge, inner space and left edge */ for (coPos.Y = yTop + 1; coPos.Y < yTop + Height - 1; coPos.Y++) { coPos.X = xLeft; FillConsoleOutputCharacterA(StdOutput, 0xB3, // '|', 1, coPos, &Written); coPos.X = xLeft + 1; FillConsoleOutputCharacterA(StdOutput, ' ', Width - 2, coPos, &Written); coPos.X = xLeft + Width - 1; FillConsoleOutputCharacterA(StdOutput, 0xB3, // '|', 1, coPos, &Written); } /* draw lower left corner */ coPos.X = xLeft; coPos.Y = yTop + Height - 1; FillConsoleOutputCharacterA(StdOutput, 0xC0, // '+', 1, coPos, &Written); /* draw lower edge */ coPos.X = xLeft + 1; coPos.Y = yTop + Height - 1; FillConsoleOutputCharacterA(StdOutput, 0xC4, // '-', Width - 2, coPos, &Written); /* draw lower right corner */ coPos.X = xLeft + Width - 1; coPos.Y = yTop + Height - 1; FillConsoleOutputCharacterA(StdOutput, 0xD9, // '+', 1, coPos, &Written); } VOID PopupError(PCCH Text, PCCH Status, PINPUT_RECORD Ir, ULONG WaitEvent) { SHORT yTop; SHORT xLeft; COORD coPos; DWORD Written; ULONG Length; ULONG MaxLength; ULONG Lines; PCHAR p; PCCH pnext; BOOLEAN LastLine; SHORT Width; SHORT Height; /* Count text lines and longest line */ MaxLength = 0; Lines = 0; pnext = Text; while (TRUE) { p = strchr(pnext, '\n'); if (p == NULL) { Length = strlen(pnext); LastLine = TRUE; } else { Length = (ULONG)(p - pnext); LastLine = FALSE; } Lines++; if (Length > MaxLength) MaxLength = Length; if (LastLine == TRUE) break; pnext = p + 1; } /* Check length of status line */ if (Status != NULL) { Length = strlen(Status); if (Length > MaxLength) MaxLength = Length; } Width = MaxLength + 4; Height = Lines + 2; if (Status != NULL) Height += 2; yTop = (yScreen - Height) / 2; xLeft = (xScreen - Width) / 2; /* Set screen attributes */ coPos.X = xLeft; for (coPos.Y = yTop; coPos.Y < yTop + Height; coPos.Y++) { FillConsoleOutputAttribute(StdOutput, FOREGROUND_RED | BACKGROUND_WHITE, Width, coPos, &Written); } DrawBox(xLeft, yTop, Width, Height); /* Print message text */ coPos.Y = yTop + 1; pnext = Text; while (TRUE) { p = strchr(pnext, '\n'); if (p == NULL) { Length = strlen(pnext); LastLine = TRUE; } else { Length = (ULONG)(p - pnext); LastLine = FALSE; } if (Length != 0) { coPos.X = xLeft + 2; WriteConsoleOutputCharacterA(StdOutput, pnext, Length, coPos, &Written); } if (LastLine == TRUE) break; coPos.Y++; pnext = p + 1; } /* Print separator line and status text */ if (Status != NULL) { coPos.Y = yTop + Height - 3; coPos.X = xLeft; FillConsoleOutputCharacterA(StdOutput, 0xC3, // '+', 1, coPos, &Written); coPos.X = xLeft + 1; FillConsoleOutputCharacterA(StdOutput, 0xC4, // '-', Width - 2, coPos, &Written); coPos.X = xLeft + Width - 1; FillConsoleOutputCharacterA(StdOutput, 0xB4, // '+', 1, coPos, &Written); coPos.Y++; coPos.X = xLeft + 2; WriteConsoleOutputCharacterA(StdOutput, Status, min(strlen(Status), (SIZE_T)Width - 4), coPos, &Written); } if (WaitEvent == POPUP_WAIT_NONE) return; while (TRUE) { CONSOLE_ConInKey(Ir); if (WaitEvent == POPUP_WAIT_ANY_KEY || Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) { return; } } } /* * Confirm quit setup * RETURNS * TRUE: Quit setup. * FALSE: Don't quit setup. */ static BOOL ConfirmQuit(PINPUT_RECORD Ir) { BOOL Result = FALSE; MUIDisplayError(ERROR_NOT_INSTALLED, NULL, POPUP_WAIT_NONE); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { Result = TRUE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { Result = FALSE; break; } } return Result; } VOID CheckUnattendedSetup(VOID) { WCHAR UnattendInfPath[MAX_PATH]; INFCONTEXT Context; HINF UnattendInf; UINT ErrorLine; INT IntValue; PWCHAR Value; if (DoesFileExist(SourcePath.Buffer, L"unattend.inf") == FALSE) { DPRINT("Does not exist: %S\\%S\n", SourcePath.Buffer, L"unattend.inf"); return; } wcscpy(UnattendInfPath, SourcePath.Buffer); wcscat(UnattendInfPath, L"\\unattend.inf"); /* Load 'unattend.inf' from install media. */ UnattendInf = SetupOpenInfFileW(UnattendInfPath, NULL, INF_STYLE_WIN4, LanguageId, &ErrorLine); if (UnattendInf == INVALID_HANDLE_VALUE) { DPRINT("SetupOpenInfFileW() failed\n"); return; } /* Open 'Unattend' section */ if (!SetupFindFirstLineW(UnattendInf, L"Unattend", L"Signature", &Context)) { DPRINT("SetupFindFirstLineW() failed for section 'Unattend'\n"); SetupCloseInfFile(UnattendInf); return; } /* Get pointer 'Signature' key */ if (!INF_GetData(&Context, NULL, &Value)) { DPRINT("INF_GetData() failed for key 'Signature'\n"); SetupCloseInfFile(UnattendInf); return; } /* Check 'Signature' string */ if (_wcsicmp(Value, L"$ReactOS$") != 0) { DPRINT("Signature not $ReactOS$\n"); SetupCloseInfFile(UnattendInf); return; } /* Check if Unattend setup is enabled */ if (!SetupFindFirstLineW(UnattendInf, L"Unattend", L"UnattendSetupEnabled", &Context)) { DPRINT("Can't find key 'UnattendSetupEnabled'\n"); SetupCloseInfFile(UnattendInf); return; } if (!INF_GetData(&Context, NULL, &Value)) { DPRINT("Can't read key 'UnattendSetupEnabled'\n"); SetupCloseInfFile(UnattendInf); return; } if (_wcsicmp(Value, L"yes") != 0) { DPRINT("Unattend setup is disabled by 'UnattendSetupEnabled' key!\n"); SetupCloseInfFile(UnattendInf); return; } /* Search for 'DestinationDiskNumber' in the 'Unattend' section */ if (!SetupFindFirstLineW(UnattendInf, L"Unattend", L"DestinationDiskNumber", &Context)) { DPRINT("SetupFindFirstLine() failed for key 'DestinationDiskNumber'\n"); SetupCloseInfFile(UnattendInf); return; } if (!SetupGetIntField(&Context, 1, &IntValue)) { DPRINT("SetupGetIntField() failed for key 'DestinationDiskNumber'\n"); SetupCloseInfFile(UnattendInf); return; } UnattendDestinationDiskNumber = (LONG)IntValue; /* Search for 'DestinationPartitionNumber' in the 'Unattend' section */ if (!SetupFindFirstLineW(UnattendInf, L"Unattend", L"DestinationPartitionNumber", &Context)) { DPRINT("SetupFindFirstLine() failed for key 'DestinationPartitionNumber'\n"); SetupCloseInfFile(UnattendInf); return; } if (!SetupGetIntField(&Context, 1, &IntValue)) { DPRINT("SetupGetIntField() failed for key 'DestinationPartitionNumber'\n"); SetupCloseInfFile(UnattendInf); return; } UnattendDestinationPartitionNumber = IntValue; /* Search for 'DestinationPartitionNumber' in the 'Unattend' section */ if (!SetupFindFirstLineW(UnattendInf, L"Unattend", L"DestinationPartitionNumber", &Context)) { DPRINT("SetupFindFirstLine() failed for key 'DestinationPartitionNumber'\n"); SetupCloseInfFile(UnattendInf); return; } /* Get pointer 'InstallationDirectory' key */ if (!INF_GetData(&Context, NULL, &Value)) { DPRINT("INF_GetData() failed for key 'InstallationDirectory'\n"); SetupCloseInfFile(UnattendInf); return; } wcscpy(UnattendInstallationDirectory, Value); IsUnattendedSetup = TRUE; /* Search for 'MBRInstallType' in the 'Unattend' section */ if (SetupFindFirstLineW(UnattendInf, L"Unattend", L"MBRInstallType", &Context)) { if (SetupGetIntField(&Context, 1, &IntValue)) { UnattendMBRInstallType = IntValue; } } /* Search for 'FormatPartition' in the 'Unattend' section */ if (SetupFindFirstLineW(UnattendInf, L"Unattend", L"FormatPartition", &Context)) { if (SetupGetIntField(&Context, 1, &IntValue)) { UnattendFormatPartition = IntValue; } } if (SetupFindFirstLineW(UnattendInf, L"Unattend", L"AutoPartition", &Context)) { if (SetupGetIntField(&Context, 1, &IntValue)) { AutoPartition = IntValue; } } /* search for LocaleID in the 'Unattend' section*/ if (SetupFindFirstLineW (UnattendInf, L"Unattend", L"LocaleID", &Context)) { if (INF_GetData (&Context, NULL, &Value)) { LONG Id = wcstol(Value, NULL, 16); swprintf(LocaleID,L"%08lx", Id); } } SetupCloseInfFile(UnattendInf); DPRINT("Running unattended setup\n"); } VOID UpdateKBLayout(VOID) { PGENERIC_LIST_ENTRY ListEntry; LPCWSTR pszNewLayout; pszNewLayout = MUIDefaultKeyboardLayout(); if (LayoutList == NULL) { LayoutList = CreateKeyboardLayoutList(SetupInf, DefaultKBLayout); if (LayoutList == NULL) { /* FIXME: Handle error! */ return; } } ListEntry = GetFirstListEntry(LayoutList); /* Search for default layout (if provided) */ if (pszNewLayout != NULL) { while (ListEntry != NULL) { if (!wcscmp(pszNewLayout, GetListEntryUserData(ListEntry))) { SetCurrentListEntry(LayoutList, ListEntry); break; } ListEntry = GetNextListEntry(ListEntry); } } } static PAGE_NUMBER LanguagePage(PINPUT_RECORD Ir) { PWCHAR NewLanguageId; BOOL RefreshPage = FALSE; /* Initialize the computer settings list */ if (LanguageList == NULL) { LanguageList = CreateLanguageList(SetupInf, DefaultLanguage); if (LanguageList == NULL) { PopupError("Setup failed to initialize available translations", NULL, NULL, POPUP_WAIT_NONE); return INTRO_PAGE; } } /* Load the font */ SelectedLanguageId = DefaultLanguage; SetConsoleCodePage(); DrawGenericList(LanguageList, 2, 18, xScreen - 3, yScreen - 3); ScrollToPositionGenericList (LanguageList, GetDefaultLanguageIndex()); MUIDisplayPage(LANGUAGE_PAGE); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { ScrollDownGenericList (LanguageList); RefreshPage = TRUE; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { ScrollUpGenericList(LanguageList); RefreshPage = TRUE; } if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_NEXT)) /* PAGE DOWN */ { ScrollPageDownGenericList(LanguageList); RefreshPage = TRUE; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_PRIOR)) /* PAGE UP */ { ScrollPageUpGenericList(LanguageList); RefreshPage = TRUE; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { SelectedLanguageId = (PWCHAR)GetListEntryUserData(GetCurrentListEntry(LanguageList)); LanguageId = (LANGID)(wcstol(SelectedLanguageId, NULL, 16) & 0xFFFF); if (wcscmp(SelectedLanguageId, DefaultLanguage)) { UpdateKBLayout(); } /* Load the font */ SetConsoleCodePage(); return INTRO_PAGE; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar > 0x60) && (Ir->Event.KeyEvent.uChar.AsciiChar < 0x7b)) { /* a-z */ GenericListKeyPress (LanguageList, Ir->Event.KeyEvent.uChar.AsciiChar); RefreshPage = TRUE; } if (RefreshPage) { NewLanguageId = (PWCHAR)GetListEntryUserData(GetCurrentListEntry(LanguageList)); if (SelectedLanguageId != NewLanguageId) { /* Clear the language page */ MUIClearPage(LANGUAGE_PAGE); SelectedLanguageId = NewLanguageId; /* Load the font */ SetConsoleCodePage(); /* Redraw language selection page in native language */ MUIDisplayPage(LANGUAGE_PAGE); } RefreshPage = FALSE; } } return INTRO_PAGE; } /* * Start page * RETURNS * Number of the next page. */ static PAGE_NUMBER SetupStartPage(PINPUT_RECORD Ir) { SYSTEM_DEVICE_INFORMATION Sdi; NTSTATUS Status; WCHAR FileNameBuffer[MAX_PATH]; INFCONTEXT Context; PWCHAR Value; UINT ErrorLine; ULONG ReturnSize; PGENERIC_LIST_ENTRY ListEntry; INT IntValue; CONSOLE_SetStatusText(MUIGetString(STRING_PLEASEWAIT)); /* Check whether a harddisk is available */ Status = NtQuerySystemInformation(SystemDeviceInformation, &Sdi, sizeof(SYSTEM_DEVICE_INFORMATION), &ReturnSize); if (!NT_SUCCESS(Status)) { CONSOLE_PrintTextXY(6, 15, "NtQuerySystemInformation() failed (Status 0x%08lx)", Status); MUIDisplayError(ERROR_DRIVE_INFORMATION, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } if (Sdi.NumberOfDisks == 0) { MUIDisplayError(ERROR_NO_HDD, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Get the source path and source root path */ Status = GetSourcePaths(&SourcePath, &SourceRootPath, &SourceRootDir); if (!NT_SUCCESS(Status)) { CONSOLE_PrintTextXY(6, 15, "GetSourcePaths() failed (Status 0x%08lx)", Status); MUIDisplayError(ERROR_NO_SOURCE_DRIVE, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } #if 0 else { CONSOLE_PrintTextXY(6, 15, "SourcePath: '%wZ'", &SourcePath); CONSOLE_PrintTextXY(6, 16, "SourceRootPath: '%wZ'", &SourceRootPath); CONSOLE_PrintTextXY(6, 17, "SourceRootDir: '%wZ'", &SourceRootDir); } #endif /* Load txtsetup.sif from install media. */ wcscpy(FileNameBuffer, SourcePath.Buffer); wcscat(FileNameBuffer, L"\\txtsetup.sif"); SetupInf = SetupOpenInfFileW(FileNameBuffer, NULL, INF_STYLE_WIN4, LanguageId, &ErrorLine); if (SetupInf == INVALID_HANDLE_VALUE) { MUIDisplayError(ERROR_LOAD_TXTSETUPSIF, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Open 'Version' section */ if (!SetupFindFirstLineW(SetupInf, L"Version", L"Signature", &Context)) { MUIDisplayError(ERROR_CORRUPT_TXTSETUPSIF, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Get pointer 'Signature' key */ if (!INF_GetData(&Context, NULL, &Value)) { MUIDisplayError(ERROR_CORRUPT_TXTSETUPSIF, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Check 'Signature' string */ if (_wcsicmp(Value, L"$ReactOS$") != 0) { MUIDisplayError(ERROR_SIGNATURE_TXTSETUPSIF, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Open 'DiskSpaceRequirements' section */ if (!SetupFindFirstLineW(SetupInf, L"DiskSpaceRequirements", L"FreeSysPartDiskSpace", &Context)) { MUIDisplayError(ERROR_CORRUPT_TXTSETUPSIF, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Get the 'FreeSysPartDiskSpace' value */ if (!SetupGetIntField(&Context, 1, &IntValue)) { MUIDisplayError(ERROR_CORRUPT_TXTSETUPSIF, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } RequiredPartitionDiskSpace = (ULONG)IntValue; /* Start PnP thread */ if (hPnpThread != INVALID_HANDLE_VALUE) { NtResumeThread(hPnpThread, NULL); hPnpThread = INVALID_HANDLE_VALUE; } CheckUnattendedSetup(); if (IsUnattendedSetup) { //TODO //read options from inf ComputerList = CreateComputerTypeList(SetupInf); DisplayList = CreateDisplayDriverList(SetupInf); KeyboardList = CreateKeyboardDriverList(SetupInf); LayoutList = CreateKeyboardLayoutList(SetupInf, DefaultKBLayout); LanguageList = CreateLanguageList(SetupInf, DefaultLanguage); /* new part */ wcscpy(SelectedLanguageId,LocaleID); /* first we hack LanguageList */ ListEntry = GetFirstListEntry(LanguageList); while (ListEntry != NULL) { if (!wcsicmp(LocaleID, GetListEntryUserData(ListEntry))) { DPRINT("found %S in LanguageList\n",GetListEntryUserData(ListEntry)); SetCurrentListEntry(LanguageList, ListEntry); break; } ListEntry = GetNextListEntry(ListEntry); } /* now LayoutList */ ListEntry = GetFirstListEntry(LayoutList); while (ListEntry != NULL) { if (!wcsicmp(LocaleID, GetListEntryUserData(ListEntry))) { DPRINT("found %S in LayoutList\n",GetListEntryUserData(ListEntry)); SetCurrentListEntry(LayoutList, ListEntry); break; } ListEntry = GetNextListEntry(ListEntry); } SetConsoleCodePage(); return INSTALL_INTRO_PAGE; } return LANGUAGE_PAGE; } /* * First setup page * RETURNS * Next page number. */ static PAGE_NUMBER IntroPage(PINPUT_RECORD Ir) { MUIDisplayPage(START_PAGE); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return INSTALL_INTRO_PAGE; break; } else if (toupper(Ir->Event.KeyEvent.uChar.AsciiChar) == 'R') /* R */ { return REPAIR_INTRO_PAGE; break; } else if (toupper(Ir->Event.KeyEvent.uChar.AsciiChar) == 'L') /* R */ { return LICENSE_PAGE; break; } } return INTRO_PAGE; } /* * License Page * RETURNS * Back to main setup page. */ static PAGE_NUMBER LicensePage(PINPUT_RECORD Ir) { MUIDisplayPage(LICENSE_PAGE); while (TRUE) { CONSOLE_ConInKey(Ir); if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return INTRO_PAGE; break; } } return LICENSE_PAGE; } static PAGE_NUMBER RepairIntroPage(PINPUT_RECORD Ir) { MUIDisplayPage(REPAIR_INTRO_PAGE); while(TRUE) { CONSOLE_ConInKey(Ir); if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return REBOOT_PAGE; } else if (toupper(Ir->Event.KeyEvent.uChar.AsciiChar) == 'U') /* U */ { RepairUpdateFlag = TRUE; return INSTALL_INTRO_PAGE; } else if (toupper(Ir->Event.KeyEvent.uChar.AsciiChar) == 'R') /* R */ { return INTRO_PAGE; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE)) /* ESC */ { return INTRO_PAGE; } } return REPAIR_INTRO_PAGE; } static PAGE_NUMBER InstallIntroPage(PINPUT_RECORD Ir) { MUIDisplayPage(INSTALL_INTRO_PAGE); if (RepairUpdateFlag) { //return SELECT_PARTITION_PAGE; return DEVICE_SETTINGS_PAGE; } if (IsUnattendedSetup) { return SELECT_PARTITION_PAGE; } while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return DEVICE_SETTINGS_PAGE; // return SCSI_CONTROLLER_PAGE; } } return INSTALL_INTRO_PAGE; } #if 0 static PAGE_NUMBER ScsiControllerPage(PINPUT_RECORD Ir) { SetTextXY(6, 8, "Setup detected the following mass storage devices:"); /* FIXME: print loaded mass storage driver descriptions */ #if 0 SetTextXY(8, 10, "TEST device"); #endif SetStatusText(" ENTER = Continue F3 = Quit"); while (TRUE) { ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return DEVICE_SETTINGS_PAGE; } } return SCSI_CONTROLLER_PAGE; } #endif static PAGE_NUMBER DeviceSettingsPage(PINPUT_RECORD Ir) { static ULONG Line = 16; MUIDisplayPage(DEVICE_SETTINGS_PAGE); /* Initialize the computer settings list */ if (ComputerList == NULL) { ComputerList = CreateComputerTypeList(SetupInf); if (ComputerList == NULL) { MUIDisplayError(ERROR_LOAD_COMPUTER, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } } /* Initialize the display settings list */ if (DisplayList == NULL) { DisplayList = CreateDisplayDriverList(SetupInf); if (DisplayList == NULL) { MUIDisplayError(ERROR_LOAD_DISPLAY, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } } /* Initialize the keyboard settings list */ if (KeyboardList == NULL) { KeyboardList = CreateKeyboardDriverList(SetupInf); if (KeyboardList == NULL) { MUIDisplayError(ERROR_LOAD_KEYBOARD, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } } /* Initialize the keyboard layout list */ if (LayoutList == NULL) { LayoutList = CreateKeyboardLayoutList(SetupInf, DefaultKBLayout); if (LayoutList == NULL) { /* FIXME: report error */ MUIDisplayError(ERROR_LOAD_KBLAYOUT, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } } MUIDisplayPage(DEVICE_SETTINGS_PAGE); CONSOLE_SetTextXY(25, 11, GetListEntryText(GetCurrentListEntry((ComputerList)))); CONSOLE_SetTextXY(25, 12, GetListEntryText(GetCurrentListEntry((DisplayList)))); CONSOLE_SetTextXY(25, 13, GetListEntryText(GetCurrentListEntry((KeyboardList)))); CONSOLE_SetTextXY(25, 14, GetListEntryText(GetCurrentListEntry((LayoutList)))); CONSOLE_InvertTextXY(24, Line, 48, 1); if (RepairUpdateFlag) { return SELECT_PARTITION_PAGE; } while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { CONSOLE_NormalTextXY(24, Line, 48, 1); if (Line == 14) Line = 16; else if (Line == 16) Line = 11; else Line++; CONSOLE_InvertTextXY(24, Line, 48, 1); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { CONSOLE_NormalTextXY(24, Line, 48, 1); if (Line == 11) Line = 16; else if (Line == 16) Line = 14; else Line--; CONSOLE_InvertTextXY(24, Line, 48, 1); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { if (Line == 11) return COMPUTER_SETTINGS_PAGE; else if (Line == 12) return DISPLAY_SETTINGS_PAGE; else if (Line == 13) return KEYBOARD_SETTINGS_PAGE; else if (Line == 14) return LAYOUT_SETTINGS_PAGE; else if (Line == 16) return SELECT_PARTITION_PAGE; } } return DEVICE_SETTINGS_PAGE; } static PAGE_NUMBER ComputerSettingsPage(PINPUT_RECORD Ir) { MUIDisplayPage(COMPUTER_SETTINGS_PAGE); DrawGenericList(ComputerList, 2, 18, xScreen - 3, yScreen - 3); SaveGenericListState(ComputerList); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { ScrollDownGenericList(ComputerList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { ScrollUpGenericList(ComputerList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE)) /* ESC */ { RestoreGenericListState(ComputerList); return DEVICE_SETTINGS_PAGE; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return DEVICE_SETTINGS_PAGE; } } return COMPUTER_SETTINGS_PAGE; } static PAGE_NUMBER DisplaySettingsPage(PINPUT_RECORD Ir) { MUIDisplayPage(DISPLAY_SETTINGS_PAGE); DrawGenericList(DisplayList, 2, 18, xScreen - 3, yScreen - 3); SaveGenericListState(DisplayList); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { ScrollDownGenericList(DisplayList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { ScrollUpGenericList(DisplayList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) { return QUIT_PAGE; } break; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE)) /* ESC */ { RestoreGenericListState(DisplayList); return DEVICE_SETTINGS_PAGE; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return DEVICE_SETTINGS_PAGE; } } return DISPLAY_SETTINGS_PAGE; } static PAGE_NUMBER KeyboardSettingsPage(PINPUT_RECORD Ir) { MUIDisplayPage(KEYBOARD_SETTINGS_PAGE); DrawGenericList(KeyboardList, 2, 18, xScreen - 3, yScreen - 3); SaveGenericListState(KeyboardList); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { ScrollDownGenericList(KeyboardList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { ScrollUpGenericList(KeyboardList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE)) /* ESC */ { RestoreGenericListState(KeyboardList); return DEVICE_SETTINGS_PAGE; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return DEVICE_SETTINGS_PAGE; } } return DISPLAY_SETTINGS_PAGE; } static PAGE_NUMBER LayoutSettingsPage(PINPUT_RECORD Ir) { MUIDisplayPage(LAYOUT_SETTINGS_PAGE); DrawGenericList(LayoutList, 2, 18, xScreen - 3, yScreen - 3); SaveGenericListState(LayoutList); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { ScrollDownGenericList(LayoutList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { ScrollUpGenericList(LayoutList); } if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_NEXT)) /* PAGE DOWN */ { ScrollPageDownGenericList(LayoutList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_PRIOR)) /* PAGE UP */ { ScrollPageUpGenericList(LayoutList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE)) /* ESC */ { RestoreGenericListState(LayoutList); return DEVICE_SETTINGS_PAGE; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return DEVICE_SETTINGS_PAGE; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar > 0x60) && (Ir->Event.KeyEvent.uChar.AsciiChar < 0x7b)) { /* a-z */ GenericListKeyPress(LayoutList , Ir->Event.KeyEvent.uChar.AsciiChar); } } return DISPLAY_SETTINGS_PAGE; } static BOOL IsDiskSizeValid(PPARTENTRY PartEntry) { ULONGLONG m1, m2; /* check for unpartitioned space */ m1 = PartEntry->UnpartitionedLength; m1 = (m1 + (1 << 19)) >> 20; /* in MBytes (rounded) */ if( m1 > RequiredPartitionDiskSpace) { return TRUE; } /* check for partitioned space */ m2 = PartEntry->PartInfo[0].PartitionLength.QuadPart; m2 = (m2 + (1 << 19)) >> 20; /* in MBytes (rounded) */ if (m2 < RequiredPartitionDiskSpace) { /* partition is too small so ask for another partion */ DPRINT1("Partition is too small(unpartitioned: %I64u MB, partitioned: %I64u MB), required disk space is %lu MB\n", m1, m2, RequiredPartitionDiskSpace); return FALSE; } else { return TRUE; } } static PAGE_NUMBER SelectPartitionPage(PINPUT_RECORD Ir) { MUIDisplayPage(SELECT_PARTITION_PAGE); if (PartitionList == NULL) { PartitionList = CreatePartitionList(2, 19, xScreen - 3, yScreen - 3); if (PartitionList == NULL) { /* FIXME: show an error dialog */ return QUIT_PAGE; } } DrawPartitionList(PartitionList); /* Warn about partitions created by Linux Fdisk */ if (WarnLinuxPartitions == TRUE && CheckForLinuxFdiskPartitions(PartitionList) == TRUE) { MUIDisplayError(ERROR_WARN_PARTITION, NULL, POPUP_WAIT_NONE); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { return QUIT_PAGE; } else if (Ir->Event.KeyEvent.wVirtualKeyCode == VK_RETURN) /* ENTER */ { WarnLinuxPartitions = FALSE; return SELECT_PARTITION_PAGE; } } } if (IsUnattendedSetup) { if (!SelectPartition(PartitionList, UnattendDestinationDiskNumber, UnattendDestinationPartitionNumber)) { if (AutoPartition) { PPARTENTRY PartEntry = PartitionList->CurrentPartition; ULONG MaxSize = (PartEntry->UnpartitionedLength + (1 << 19)) >> 20; /* in MBytes (rounded) */ if(!IsDiskSizeValid(PartitionList->CurrentPartition)) { MUIDisplayError(ERROR_INSUFFICIENT_DISKSPACE, Ir, POPUP_WAIT_ANY_KEY); return SELECT_PARTITION_PAGE; /* let the user select another partition */ } CreateNewPartition(PartitionList, MaxSize, TRUE); return SELECT_FILE_SYSTEM_PAGE; } } else { if(!IsDiskSizeValid(PartitionList->CurrentPartition)) { MUIDisplayError(ERROR_INSUFFICIENT_DISKSPACE, Ir, POPUP_WAIT_ANY_KEY); return SELECT_PARTITION_PAGE; /* let the user select another partition */ } return SELECT_FILE_SYSTEM_PAGE; } } while (TRUE) { /* Update status text */ if (PartitionList->CurrentPartition == NULL || PartitionList->CurrentPartition->Unpartitioned == TRUE) { CONSOLE_SetStatusText(MUIGetString(STRING_INSTALLCREATEPARTITION)); } else { CONSOLE_SetStatusText(MUIGetString(STRING_INSTALLDELETEPARTITION)); } CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) { DestroyPartitionList(PartitionList); PartitionList = NULL; return QUIT_PAGE; } break; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { ScrollDownPartitionList(PartitionList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { ScrollUpPartitionList(PartitionList); } else if (Ir->Event.KeyEvent.wVirtualKeyCode == VK_RETURN) /* ENTER */ { if(!IsDiskSizeValid(PartitionList->CurrentPartition)) { MUIDisplayError(ERROR_INSUFFICIENT_DISKSPACE, Ir, POPUP_WAIT_ANY_KEY); return SELECT_PARTITION_PAGE; /* let the user select another partition */ } if (PartitionList->CurrentPartition == NULL || PartitionList->CurrentPartition->Unpartitioned == TRUE) { CreateNewPartition(PartitionList, 0ULL, TRUE); } return SELECT_FILE_SYSTEM_PAGE; } else if (Ir->Event.KeyEvent.wVirtualKeyCode == 'C') /* C */ { if (PartitionList->CurrentPartition->Unpartitioned == FALSE) { MUIDisplayError(ERROR_NEW_PARTITION, Ir, POPUP_WAIT_ANY_KEY); return SELECT_PARTITION_PAGE; } return CREATE_PARTITION_PAGE; } else if (Ir->Event.KeyEvent.wVirtualKeyCode == 'D') /* D */ { if (PartitionList->CurrentPartition->Unpartitioned == TRUE) { MUIDisplayError(ERROR_DELETE_SPACE, Ir, POPUP_WAIT_ANY_KEY); return SELECT_PARTITION_PAGE; } return DELETE_PARTITION_PAGE; } } return SELECT_PARTITION_PAGE; } static VOID DrawInputField(ULONG FieldLength, SHORT Left, SHORT Top, PCHAR FieldContent) { CHAR buf[100]; COORD coPos; DWORD Written; coPos.X = Left; coPos.Y = Top; memset(buf, '_', sizeof(buf)); buf[FieldLength - strlen(FieldContent)] = 0; strcat(buf, FieldContent); WriteConsoleOutputCharacterA(StdOutput, buf, strlen(buf), coPos, &Written); } #define PARTITION_SIZE_INPUT_FIELD_LENGTH 6 /* Restriction for MaxSize: pow(10, PARTITION_SIZE_INPUT_FIELD_LENGTH)-1 */ #define PARTITION_MAXSIZE 999999 static VOID ShowPartitionSizeInputBox(SHORT Left, SHORT Top, SHORT Right, SHORT Bottom, ULONG MaxSize, PCHAR InputBuffer, PBOOLEAN Quit, PBOOLEAN Cancel) { INPUT_RECORD Ir; COORD coPos; DWORD Written; CHAR Buffer[100]; ULONG Index; CHAR ch; SHORT iLeft; SHORT iTop; if (Quit != NULL) *Quit = FALSE; if (Cancel != NULL) *Cancel = FALSE; DrawBox(Left, Top, Right - Left + 1, Bottom - Top + 1); /* Print message */ coPos.X = Left + 2; coPos.Y = Top + 2; strcpy(Buffer, MUIGetString(STRING_PARTITIONSIZE)); iLeft = coPos.X + strlen(Buffer) + 1; iTop = coPos.Y; WriteConsoleOutputCharacterA(StdOutput, Buffer, strlen (Buffer), coPos, &Written); sprintf(Buffer, MUIGetString(STRING_MAXSIZE), MaxSize); coPos.X = iLeft + PARTITION_SIZE_INPUT_FIELD_LENGTH + 1; coPos.Y = iTop; WriteConsoleOutputCharacterA(StdOutput, Buffer, strlen (Buffer), coPos, &Written); sprintf(Buffer, "%lu", MaxSize); Index = strlen(Buffer); DrawInputField(PARTITION_SIZE_INPUT_FIELD_LENGTH, iLeft, iTop, Buffer); while (TRUE) { CONSOLE_ConInKey(&Ir); if ((Ir.Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir.Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (Quit != NULL) *Quit = TRUE; Buffer[0] = 0; break; } else if (Ir.Event.KeyEvent.wVirtualKeyCode == VK_RETURN) /* ENTER */ { break; } else if (Ir.Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE) /* ESCAPE */ { if (Cancel != NULL) *Cancel = TRUE; Buffer[0] = 0; break; } else if ((Ir.Event.KeyEvent.wVirtualKeyCode == VK_BACK) && /* BACKSPACE */ (Index > 0)) { Index--; Buffer[Index] = 0; DrawInputField(PARTITION_SIZE_INPUT_FIELD_LENGTH, iLeft, iTop, Buffer); } else if ((Ir.Event.KeyEvent.uChar.AsciiChar != 0x00) && (Index < PARTITION_SIZE_INPUT_FIELD_LENGTH)) { ch = Ir.Event.KeyEvent.uChar.AsciiChar; if ((ch >= '0') && (ch <= '9')) { Buffer[Index] = ch; Index++; Buffer[Index] = 0; DrawInputField(PARTITION_SIZE_INPUT_FIELD_LENGTH, iLeft, iTop, Buffer); } } } strcpy(InputBuffer, Buffer); } static PAGE_NUMBER CreatePartitionPage(PINPUT_RECORD Ir) { PDISKENTRY DiskEntry; PPARTENTRY PartEntry; BOOLEAN Quit; BOOLEAN Cancel; CHAR InputBuffer[50]; ULONG MaxSize; ULONGLONG PartSize; ULONGLONG DiskSize; PCHAR Unit; if (PartitionList == NULL || PartitionList->CurrentDisk == NULL || PartitionList->CurrentPartition == NULL) { /* FIXME: show an error dialog */ return QUIT_PAGE; } DiskEntry = PartitionList->CurrentDisk; PartEntry = PartitionList->CurrentPartition; CONSOLE_SetStatusText(MUIGetString(STRING_PLEASEWAIT)); CONSOLE_SetTextXY(6, 8, MUIGetString(STRING_CHOOSENEWPARTITION)); #if 0 if (DiskEntry->DiskSize >= 0x280000000ULL) /* 10 GB */ { DiskSize = (DiskEntry->DiskSize + (1 << 29)) >> 30; Unit = MUIGetString(STRING_GB); } else #endif { DiskSize = (DiskEntry->DiskSize + (1 << 19)) >> 20; if (DiskSize == 0) DiskSize = 1; Unit = MUIGetString(STRING_MB); } if (DiskEntry->DriverName.Length > 0) { CONSOLE_PrintTextXY(6, 10, MUIGetString(STRING_HDINFOPARTCREATE), DiskSize, Unit, DiskEntry->DiskNumber, DiskEntry->Port, DiskEntry->Bus, DiskEntry->Id, &DiskEntry->DriverName); } else { CONSOLE_PrintTextXY(6, 10, MUIGetString(STRING_HDDINFOUNK1), DiskSize, Unit, DiskEntry->DiskNumber, DiskEntry->Port, DiskEntry->Bus, DiskEntry->Id); } CONSOLE_SetTextXY(6, 12, MUIGetString(STRING_HDDSIZE)); #if 0 CONSOLE_PrintTextXY(8, 10, "Maximum size of the new partition is %I64u MB", PartitionList->CurrentPartition->UnpartitionedLength / (1024*1024)); #endif CONSOLE_SetStatusText(MUIGetString(STRING_CREATEPARTITION)); PartEntry = PartitionList->CurrentPartition; while (TRUE) { MaxSize = (PartEntry->UnpartitionedLength + (1 << 19)) >> 20; /* in MBytes (rounded) */ if (MaxSize > PARTITION_MAXSIZE) MaxSize = PARTITION_MAXSIZE; ShowPartitionSizeInputBox(12, 14, xScreen - 12, 17, /* left, top, right, bottom */ MaxSize, InputBuffer, &Quit, &Cancel); if (Quit == TRUE) { if (ConfirmQuit (Ir) == TRUE) { return QUIT_PAGE; } } else if (Cancel == TRUE) { return SELECT_PARTITION_PAGE; } else { PartSize = atoi(InputBuffer); if (PartSize < 1) { /* Too small */ continue; } if (PartSize > MaxSize) { /* Too large */ continue; } /* Convert to bytes */ if (PartSize == MaxSize) { /* Use all of the unpartitioned disk space */ PartSize = PartEntry->UnpartitionedLength; } else { /* Round-up by cylinder size */ PartSize = (PartSize * 1024 * 1024 + DiskEntry->CylinderSize - 1) / DiskEntry->CylinderSize * DiskEntry->CylinderSize; /* But never get larger than the unpartitioned disk space */ if (PartSize > PartEntry->UnpartitionedLength) PartSize = PartEntry->UnpartitionedLength; } DPRINT ("Partition size: %I64u bytes\n", PartSize); CreateNewPartition(PartitionList, PartSize, FALSE); return SELECT_PARTITION_PAGE; } } return CREATE_PARTITION_PAGE; } static PAGE_NUMBER DeletePartitionPage(PINPUT_RECORD Ir) { PDISKENTRY DiskEntry; PPARTENTRY PartEntry; ULONGLONG DiskSize; ULONGLONG PartSize; PCHAR Unit; PCHAR PartType; UCHAR PartNumber; if (PartitionList == NULL || PartitionList->CurrentDisk == NULL || PartitionList->CurrentPartition == NULL) { /* FIXME: show an error dialog */ return QUIT_PAGE; } DiskEntry = PartitionList->CurrentDisk; PartEntry = PartitionList->CurrentPartition; PartNumber = PartitionList->CurrentPartitionNumber; MUIDisplayPage(DELETE_PARTITION_PAGE); /* Determine partition type */ PartType = NULL; if (PartEntry->New == TRUE) { PartType = MUIGetString(STRING_UNFORMATTED); } else if (PartEntry->Unpartitioned == FALSE) { if ((PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT_12) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT_16) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_HUGE) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_XINT13)) { PartType = "FAT"; } else if ((PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT32) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT32_XINT13)) { PartType = "FAT32"; } else if (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_EXT2) { PartType = "EXT2"; } else if (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_IFS) { PartType = "NTFS"; /* FIXME: Not quite correct! */ } } #if 0 if (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart >= 0x280000000LL) /* 10 GB */ { PartSize = (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart + (1 << 29)) >> 30; Unit = MUIGetString(STRING_GB); } else #endif if (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart >= 0xA00000LL) /* 10 MB */ { PartSize = (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart + (1 << 19)) >> 20; Unit = MUIGetString(STRING_MB); } else { PartSize = (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart + (1 << 9)) >> 10; Unit = MUIGetString(STRING_KB); } if (PartType == NULL) { CONSOLE_PrintTextXY(6, 10, MUIGetString(STRING_HDDINFOUNK2), (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : PartEntry->DriveLetter[PartNumber], (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : ':', PartEntry->PartInfo[PartNumber].PartitionType, PartSize, Unit); } else { CONSOLE_PrintTextXY(6, 10, " %c%c %s %I64u %s", (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : PartEntry->DriveLetter[PartNumber], (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : ':', PartType, PartSize, Unit); } #if 0 if (DiskEntry->DiskSize >= 0x280000000ULL) /* 10 GB */ { DiskSize = (DiskEntry->DiskSize + (1 << 29)) >> 30; Unit = MUIGetString(STRING_GB); } else #endif { DiskSize = (DiskEntry->DiskSize + (1 << 19)) >> 20; if (DiskSize == 0) DiskSize = 1; Unit = MUIGetString(STRING_MB); } if (DiskEntry->DriverName.Length > 0) { CONSOLE_PrintTextXY(6, 12, MUIGetString(STRING_HDINFOPARTDELETE), DiskSize, Unit, DiskEntry->DiskNumber, DiskEntry->Port, DiskEntry->Bus, DiskEntry->Id, &DiskEntry->DriverName); } else { CONSOLE_PrintTextXY(6, 12, MUIGetString(STRING_HDDINFOUNK3), DiskSize, Unit, DiskEntry->DiskNumber, DiskEntry->Port, DiskEntry->Bus, DiskEntry->Id); } while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) { return QUIT_PAGE; } break; } else if (Ir->Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE) /* ESC */ { return SELECT_PARTITION_PAGE; } else if (Ir->Event.KeyEvent.wVirtualKeyCode == 'D') /* D */ { DeleteCurrentPartition(PartitionList); return SELECT_PARTITION_PAGE; } } return DELETE_PARTITION_PAGE; } static PAGE_NUMBER SelectFileSystemPage(PINPUT_RECORD Ir) { PDISKENTRY DiskEntry; PPARTENTRY PartEntry; UCHAR PartNumber; ULONGLONG DiskSize; ULONGLONG PartSize; PCHAR DiskUnit; PCHAR PartUnit; PCHAR PartType; if (PartitionList == NULL || PartitionList->CurrentDisk == NULL || PartitionList->CurrentPartition == NULL) { /* FIXME: show an error dialog */ return QUIT_PAGE; } DiskEntry = PartitionList->CurrentDisk; PartEntry = PartitionList->CurrentPartition; PartNumber = PartitionList->CurrentPartitionNumber; /* adjust disk size */ if (DiskEntry->DiskSize >= 0x280000000ULL) /* 10 GB */ { DiskSize = (DiskEntry->DiskSize + (1 << 29)) >> 30; DiskUnit = MUIGetString(STRING_GB); } else { DiskSize = (DiskEntry->DiskSize + (1 << 19)) >> 20; DiskUnit = MUIGetString(STRING_MB); } /* adjust partition size */ if (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart >= 0x280000000LL) /* 10 GB */ { PartSize = (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart + (1 << 29)) >> 30; PartUnit = MUIGetString(STRING_GB); } else { PartSize = (PartEntry->PartInfo[PartNumber].PartitionLength.QuadPart + (1 << 19)) >> 20; PartUnit = MUIGetString(STRING_MB); } /* adjust partition type */ if ((PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT_12) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT_16) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_HUGE) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_XINT13)) { PartType = "FAT"; } else if ((PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT32) || (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_FAT32_XINT13)) { PartType = "FAT32"; } else if (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_EXT2) { PartType = "EXT2"; } else if (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_IFS) { PartType = "NTFS"; /* FIXME: Not quite correct! */ } else if (PartEntry->PartInfo[PartNumber].PartitionType == PARTITION_ENTRY_UNUSED) { PartType = MUIGetString(STRING_FORMATUNUSED); } else { PartType = MUIGetString(STRING_FORMATUNKNOWN); } if (PartEntry->AutoCreate == TRUE) { CONSOLE_SetTextXY(6, 8, MUIGetString(STRING_NEWPARTITION)); #if 0 CONSOLE_PrintTextXY(8, 10, "Partition %lu (%I64u %s) %s of", PartEntry->PartInfo[PartNumber].PartitionNumber, PartSize, PartUnit, PartType); #endif CONSOLE_PrintTextXY(8, 10, MUIGetString(STRING_HDINFOPARTZEROED), DiskEntry->DiskNumber, DiskSize, DiskUnit, DiskEntry->Port, DiskEntry->Bus, DiskEntry->Id, &DiskEntry->DriverName); CONSOLE_SetTextXY(6, 12, MUIGetString(STRING_PARTFORMAT)); PartEntry->AutoCreate = FALSE; } else if (PartEntry->New == TRUE) { CONSOLE_SetTextXY(6, 8, MUIGetString(STRING_NONFORMATTEDPART)); CONSOLE_SetTextXY(6, 10, MUIGetString(STRING_PARTFORMAT)); } else { CONSOLE_SetTextXY(6, 8, MUIGetString(STRING_INSTALLONPART)); if (PartType == NULL) { CONSOLE_PrintTextXY(8, 10, MUIGetString(STRING_HDDINFOUNK4), (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : PartEntry->DriveLetter[PartNumber], (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : ':', PartEntry->PartInfo[PartNumber].PartitionType, PartSize, PartUnit); } else { CONSOLE_PrintTextXY(8, 10, "%c%c %s %I64u %s", (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : PartEntry->DriveLetter[PartNumber], (PartEntry->DriveLetter[PartNumber] == 0) ? '-' : ':', PartType, PartSize, PartUnit); } CONSOLE_PrintTextXY(6, 12, MUIGetString(STRING_HDINFOPARTEXISTS), DiskEntry->DiskNumber, DiskSize, DiskUnit, DiskEntry->Port, DiskEntry->Bus, DiskEntry->Id, &DiskEntry->DriverName); } MUIDisplayPage(SELECT_FILE_SYSTEM_PAGE); if (FileSystemList == NULL) { FileSystemList = CreateFileSystemList(6, 26, PartEntry->New, L"FAT"); if (FileSystemList == NULL) { /* FIXME: show an error dialog */ return QUIT_PAGE; } /* FIXME: Add file systems to list */ } DrawFileSystemList(FileSystemList); if (RepairUpdateFlag) { return CHECK_FILE_SYSTEM_PAGE; //return SELECT_PARTITION_PAGE; } if (IsUnattendedSetup) { if (UnattendFormatPartition) { return FORMAT_PARTITION_PAGE; } return CHECK_FILE_SYSTEM_PAGE; } while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) { return QUIT_PAGE; } break; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_ESCAPE)) /* ESC */ { return SELECT_PARTITION_PAGE; } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { ScrollDownFileSystemList(FileSystemList); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { ScrollUpFileSystemList(FileSystemList); } else if (Ir->Event.KeyEvent.wVirtualKeyCode == VK_RETURN) /* ENTER */ { if (!FileSystemList->Selected->FormatFunc) { return CHECK_FILE_SYSTEM_PAGE; } else { return FORMAT_PARTITION_PAGE; } } } return SELECT_FILE_SYSTEM_PAGE; } static ULONG FormatPartitionPage(PINPUT_RECORD Ir) { WCHAR PathBuffer[MAX_PATH]; PPARTENTRY PartEntry; UCHAR PartNum; NTSTATUS Status; #ifndef NDEBUG PDISKENTRY DiskEntry; ULONG Line; ULONG i; PLIST_ENTRY Entry; #endif MUIDisplayPage(FORMAT_PARTITION_PAGE); if (PartitionList == NULL || PartitionList->CurrentDisk == NULL || PartitionList->CurrentPartition == NULL) { /* FIXME: show an error dialog */ return QUIT_PAGE; } #ifndef NDEBUG DiskEntry = PartitionList->CurrentDisk; #endif PartEntry = PartitionList->CurrentPartition; PartNum = PartitionList->CurrentPartitionNumber; while (TRUE) { if (!IsUnattendedSetup) { CONSOLE_ConInKey(Ir); } if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) { return QUIT_PAGE; } break; } else if (Ir->Event.KeyEvent.wVirtualKeyCode == VK_RETURN || IsUnattendedSetup) /* ENTER */ { CONSOLE_SetStatusText(MUIGetString(STRING_PLEASEWAIT)); if (wcscmp(FileSystemList->Selected->FileSystem, L"FAT") == 0) { if (PartEntry->PartInfo[PartNum].PartitionLength.QuadPart < (4200LL * 1024LL)) { /* FAT12 CHS partition (disk is smaller than 4.1MB) */ PartEntry->PartInfo[PartNum].PartitionType = PARTITION_FAT_12; } else if (PartEntry->PartInfo[PartNum].StartingOffset.QuadPart < (1024LL * 255LL * 63LL * 512LL)) { /* Partition starts below the 8.4GB boundary ==> CHS partition */ if (PartEntry->PartInfo[PartNum].PartitionLength.QuadPart < (32LL * 1024LL * 1024LL)) { /* FAT16 CHS partition (partiton size < 32MB) */ PartEntry->PartInfo[PartNum].PartitionType = PARTITION_FAT_16; } else if (PartEntry->PartInfo[PartNum].PartitionLength.QuadPart < (512LL * 1024LL * 1024LL)) { /* FAT16 CHS partition (partition size < 512MB) */ PartEntry->PartInfo[PartNum].PartitionType = PARTITION_HUGE; } else { /* FAT32 CHS partition (partition size >= 512MB) */ PartEntry->PartInfo[PartNum].PartitionType = PARTITION_FAT32; } } else { /* Partition starts above the 8.4GB boundary ==> LBA partition */ if (PartEntry->PartInfo[PartNum].PartitionLength.QuadPart < (512LL * 1024LL * 1024LL)) { /* FAT16 LBA partition (partition size < 512MB) */ PartEntry->PartInfo[PartNum].PartitionType = PARTITION_XINT13; } else { /* FAT32 LBA partition (partition size >= 512MB) */ PartEntry->PartInfo[PartNum].PartitionType = PARTITION_FAT32_XINT13; } } } else if (wcscmp(FileSystemList->Selected->FileSystem, L"EXT2") == 0) PartEntry->PartInfo[PartNum].PartitionType = PARTITION_EXT2; else if (!FileSystemList->Selected->FormatFunc) return QUIT_PAGE; #ifndef NDEBUG CONSOLE_PrintTextXY(6, 12, "Disk: %I64u Cylinder: %I64u Track: %I64u", DiskEntry->DiskSize, DiskEntry->CylinderSize, DiskEntry->TrackSize); Line = 13; DiskEntry = PartitionList->CurrentDisk; Entry = DiskEntry->PartListHead.Flink; while (Entry != &DiskEntry->PartListHead) { PartEntry = CONTAINING_RECORD(Entry, PARTENTRY, ListEntry); if (PartEntry->Unpartitioned == FALSE) { for (i = 0; i < 4; i++) { CONSOLE_PrintTextXY(6, Line, "%2u: %2u %c %12I64u %12I64u %2u %c", i, PartEntry->PartInfo[i].PartitionNumber, PartEntry->PartInfo[i].BootIndicator ? 'A' : '-', PartEntry->PartInfo[i].StartingOffset.QuadPart, PartEntry->PartInfo[i].PartitionLength.QuadPart, PartEntry->PartInfo[i].PartitionType, PartEntry->PartInfo[i].RewritePartition ? '*' : ' '); Line++; } Line++; } Entry = Entry->Flink; } /* Restore the old entry */ PartEntry = PartitionList->CurrentPartition; #endif if (WritePartitionsToDisk(PartitionList) == FALSE) { DPRINT("WritePartitionsToDisk() failed\n"); MUIDisplayError(ERROR_WRITE_PTABLE, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Set DestinationRootPath */ RtlFreeUnicodeString(&DestinationRootPath); swprintf(PathBuffer, L"\\Device\\Harddisk%lu\\Partition%lu", PartitionList->CurrentDisk->DiskNumber, PartitionList->CurrentPartition->PartInfo[PartNum].PartitionNumber); RtlCreateUnicodeString(&DestinationRootPath, PathBuffer); DPRINT("DestinationRootPath: %wZ\n", &DestinationRootPath); if (FileSystemList->Selected->FormatFunc) { Status = FormatPartition(&DestinationRootPath, FileSystemList->Selected); if (!NT_SUCCESS(Status)) { DPRINT1("FormatPartition() failed with status 0x%08lx\n", Status); /* FIXME: show an error dialog */ return QUIT_PAGE; } PartEntry->New = FALSE; CheckActiveBootPartition(PartitionList); } #ifndef NDEBUG CONSOLE_SetStatusText(" Done. Press any key ..."); CONSOLE_ConInKey(Ir); #endif DestroyFileSystemList(FileSystemList); FileSystemList = NULL; return INSTALL_DIRECTORY_PAGE; } } return FORMAT_PARTITION_PAGE; } static ULONG CheckFileSystemPage(PINPUT_RECORD Ir) { PFILE_SYSTEM_ITEM CurrentFileSystem; WCHAR PathBuffer[MAX_PATH]; CHAR Buffer[MAX_PATH]; NTSTATUS Status; UCHAR PartNum = PartitionList->CurrentPartitionNumber; /* FIXME: code duplicated in FormatPartitionPage */ /* Set DestinationRootPath */ RtlFreeUnicodeString(&DestinationRootPath); swprintf(PathBuffer, L"\\Device\\Harddisk%lu\\Partition%lu", PartitionList->CurrentDisk->DiskNumber, PartitionList->CurrentPartition->PartInfo[PartNum].PartitionNumber); RtlCreateUnicodeString(&DestinationRootPath, PathBuffer); DPRINT("DestinationRootPath: %wZ\n", &DestinationRootPath); CONSOLE_SetTextXY(6, 8, MUIGetString(STRING_CHECKINGPART)); CONSOLE_SetStatusText(MUIGetString(STRING_PLEASEWAIT)); /* WRONG: first filesystem is not necesseraly the one of the current partition! */ CurrentFileSystem = CONTAINING_RECORD(FileSystemList->ListHead.Flink, FILE_SYSTEM_ITEM, ListEntry); if (!CurrentFileSystem->ChkdskFunc) { sprintf(Buffer, "Setup is currently unable to check a partition formatted in %S.\n" "\n" " \x07 Press ENTER to continue Setup.\n" " \x07 Press F3 to quit Setup.", CurrentFileSystem->FileSystem); PopupError(Buffer, MUIGetString(STRING_QUITCONTINUE), NULL, POPUP_WAIT_NONE); while (TRUE) { CONSOLE_ConInKey(Ir); if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x00 && Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3) /* F3 */ { if (ConfirmQuit(Ir)) return QUIT_PAGE; else return CHECK_FILE_SYSTEM_PAGE; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == VK_RETURN) /* ENTER */ { return INSTALL_DIRECTORY_PAGE; } } } else { Status = ChkdskPartition(&DestinationRootPath, CurrentFileSystem); if (!NT_SUCCESS(Status)) { DPRINT("ChkdskPartition() failed with status 0x%08lx\n", Status); sprintf(Buffer, "Setup failed to verify the selected partition.\n" "(Status 0x%08lx).\n", Status); PopupError(Buffer, MUIGetString(STRING_REBOOTCOMPUTER), Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } return INSTALL_DIRECTORY_PAGE; } } static PAGE_NUMBER InstallDirectoryPage1(PWCHAR InstallDir, PDISKENTRY DiskEntry, PPARTENTRY PartEntry, UCHAR PartNum) { WCHAR PathBuffer[MAX_PATH]; /* Create 'InstallPath' string */ RtlFreeUnicodeString(&InstallPath); RtlCreateUnicodeString(&InstallPath, InstallDir); /* Create 'DestinationPath' string */ RtlFreeUnicodeString(&DestinationPath); wcscpy(PathBuffer, DestinationRootPath.Buffer); if (InstallDir[0] != L'\\') wcscat(PathBuffer, L"\\"); wcscat(PathBuffer, InstallDir); RtlCreateUnicodeString(&DestinationPath, PathBuffer); /* Create 'DestinationArcPath' */ RtlFreeUnicodeString(&DestinationArcPath); swprintf(PathBuffer, L"multi(0)disk(0)rdisk(%lu)partition(%lu)", DiskEntry->BiosDiskNumber, PartEntry->PartInfo[PartNum].PartitionNumber); if (InstallDir[0] != L'\\') wcscat(PathBuffer, L"\\"); wcscat(PathBuffer, InstallDir); RtlCreateUnicodeString(&DestinationArcPath, PathBuffer); return PREPARE_COPY_PAGE; } static PAGE_NUMBER InstallDirectoryPage(PINPUT_RECORD Ir) { PDISKENTRY DiskEntry; PPARTENTRY PartEntry; WCHAR InstallDir[51]; PWCHAR DefaultPath; INFCONTEXT Context; ULONG Length; if (PartitionList == NULL || PartitionList->CurrentDisk == NULL || PartitionList->CurrentPartition == NULL) { /* FIXME: show an error dialog */ return QUIT_PAGE; } DiskEntry = PartitionList->CurrentDisk; PartEntry = PartitionList->CurrentPartition; /* Search for 'DefaultPath' in the 'SetupData' section */ if (!SetupFindFirstLineW(SetupInf, L"SetupData", L"DefaultPath", &Context)) { MUIDisplayError(ERROR_FIND_SETUPDATA, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Read the 'DefaultPath' data */ if (INF_GetData(&Context, NULL, &DefaultPath)) { wcscpy(InstallDir, DefaultPath); } else { wcscpy(InstallDir, L"\\ReactOS"); } Length = wcslen(InstallDir); CONSOLE_SetInputTextXY(8, 11, 51, InstallDir); MUIDisplayPage(INSTALL_DIRECTORY_PAGE); if (IsUnattendedSetup) { return InstallDirectoryPage1(InstallDir, DiskEntry, PartEntry, PartitionList->CurrentPartitionNumber); } while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return InstallDirectoryPage1(InstallDir, DiskEntry, PartEntry, PartitionList->CurrentPartitionNumber); } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x08) /* BACKSPACE */ { if (Length > 0) { Length--; InstallDir[Length] = 0; CONSOLE_SetInputTextXY(8, 11, 51, InstallDir); } } else if (isprint(Ir->Event.KeyEvent.uChar.AsciiChar)) { if (Length < 50) { InstallDir[Length] = (WCHAR)Ir->Event.KeyEvent.uChar.AsciiChar; Length++; InstallDir[Length] = 0; CONSOLE_SetInputTextXY(8, 11, 51, InstallDir); } } } return INSTALL_DIRECTORY_PAGE; } static BOOLEAN AddSectionToCopyQueueCab(HINF InfFile, PWCHAR SectionName, PWCHAR SourceCabinet, PCUNICODE_STRING DestinationPath, PINPUT_RECORD Ir) { INFCONTEXT FilesContext; INFCONTEXT DirContext; PWCHAR FileKeyName; PWCHAR FileKeyValue; PWCHAR DirKeyValue; PWCHAR TargetFileName; /* Search for the SectionName section */ if (!SetupFindFirstLineW(InfFile, SectionName, NULL, &FilesContext)) { char Buffer[128]; sprintf(Buffer, MUIGetString(STRING_TXTSETUPFAILED), SectionName); PopupError(Buffer, MUIGetString(STRING_REBOOTCOMPUTER), Ir, POPUP_WAIT_ENTER); return FALSE; } /* * Enumerate the files in the section * and add them to the file queue. */ do { /* Get source file name and target directory id */ if (!INF_GetData(&FilesContext, &FileKeyName, &FileKeyValue)) { /* FIXME: Handle error! */ DPRINT1("INF_GetData() failed\n"); break; } /* Get optional target file name */ if (!INF_GetDataField(&FilesContext, 2, &TargetFileName)) TargetFileName = NULL; DPRINT("FileKeyName: '%S' FileKeyValue: '%S'\n", FileKeyName, FileKeyValue); /* Lookup target directory */ if (!SetupFindFirstLineW(InfFile, L"Directories", FileKeyValue, &DirContext)) { /* FIXME: Handle error! */ DPRINT1("SetupFindFirstLine() failed\n"); break; } if (!INF_GetData(&DirContext, NULL, &DirKeyValue)) { /* FIXME: Handle error! */ DPRINT1("INF_GetData() failed\n"); break; } if (!SetupQueueCopy(SetupFileQueue, SourceCabinet, SourceRootPath.Buffer, SourceRootDir.Buffer, FileKeyName, DirKeyValue, TargetFileName)) { /* FIXME: Handle error! */ DPRINT1("SetupQueueCopy() failed\n"); } } while (SetupFindNextLine(&FilesContext, &FilesContext)); return TRUE; } static BOOLEAN AddSectionToCopyQueue(HINF InfFile, PWCHAR SectionName, PWCHAR SourceCabinet, PCUNICODE_STRING DestinationPath, PINPUT_RECORD Ir) { INFCONTEXT FilesContext; INFCONTEXT DirContext; PWCHAR FileKeyName; PWCHAR FileKeyValue; PWCHAR DirKeyValue; PWCHAR TargetFileName; WCHAR CompleteOrigFileName[512]; if (SourceCabinet) return AddSectionToCopyQueueCab(InfFile, L"SourceFiles", SourceCabinet, DestinationPath, Ir); /* Search for the SectionName section */ if (!SetupFindFirstLineW(InfFile, SectionName, NULL, &FilesContext)) { char Buffer[128]; sprintf(Buffer, MUIGetString(STRING_TXTSETUPFAILED), SectionName); PopupError(Buffer, MUIGetString(STRING_REBOOTCOMPUTER), Ir, POPUP_WAIT_ENTER); return FALSE; } /* * Enumerate the files in the section * and add them to the file queue. */ do { /* Get source file name and target directory id */ if (!INF_GetData(&FilesContext, &FileKeyName, &FileKeyValue)) { /* FIXME: Handle error! */ DPRINT1("INF_GetData() failed\n"); break; } /* Get target directory id */ if (!INF_GetDataField(&FilesContext, 13, &FileKeyValue)) { /* FIXME: Handle error! */ DPRINT1("INF_GetData() failed\n"); break; } /* Get optional target file name */ if (!INF_GetDataField(&FilesContext, 11, &TargetFileName)) TargetFileName = NULL; else if (!*TargetFileName) TargetFileName = NULL; DPRINT("FileKeyName: '%S' FileKeyValue: '%S'\n", FileKeyName, FileKeyValue); /* Lookup target directory */ if (!SetupFindFirstLineW(InfFile, L"Directories", FileKeyValue, &DirContext)) { /* FIXME: Handle error! */ DPRINT1("SetupFindFirstLine() failed\n"); break; } if (!INF_GetData(&DirContext, NULL, &DirKeyValue)) { /* FIXME: Handle error! */ DPRINT1("INF_GetData() failed\n"); break; } wcscpy(CompleteOrigFileName, SourceRootDir.Buffer); wcscat(CompleteOrigFileName, L"\\"); wcscat(CompleteOrigFileName, DirKeyValue); if (!SetupQueueCopy(SetupFileQueue, SourceCabinet, SourceRootPath.Buffer, CompleteOrigFileName, FileKeyName, DirKeyValue, TargetFileName)) { /* FIXME: Handle error! */ DPRINT1("SetupQueueCopy() failed\n"); } } while (SetupFindNextLine(&FilesContext, &FilesContext)); return TRUE; } static BOOLEAN PrepareCopyPageInfFile(HINF InfFile, PWCHAR SourceCabinet, PINPUT_RECORD Ir) { WCHAR PathBuffer[MAX_PATH]; INFCONTEXT DirContext; PWCHAR AdditionalSectionName = NULL; PWCHAR KeyValue; ULONG Length; NTSTATUS Status; /* Add common files */ if (!AddSectionToCopyQueue(InfFile, L"SourceDisksFiles", SourceCabinet, &DestinationPath, Ir)) return FALSE; /* Add specific files depending of computer type */ if (SourceCabinet == NULL) { if (!ProcessComputerFiles(InfFile, ComputerList, &AdditionalSectionName)) return FALSE; if (AdditionalSectionName) { if (!AddSectionToCopyQueue(InfFile, AdditionalSectionName, SourceCabinet, &DestinationPath, Ir)) return FALSE; } } /* Create directories */ /* * FIXME: * Install directories like '\reactos\test' are not handled yet. */ /* Get destination path */ wcscpy(PathBuffer, DestinationPath.Buffer); /* Remove trailing backslash */ Length = wcslen(PathBuffer); if ((Length > 0) && (PathBuffer[Length - 1] == '\\')) { PathBuffer[Length - 1] = 0; } /* Create the install directory */ Status = SetupCreateDirectory(PathBuffer); if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_COLLISION) { DPRINT("Creating directory '%S' failed: Status = 0x%08lx", PathBuffer, Status); MUIDisplayError(ERROR_CREATE_INSTALL_DIR, Ir, POPUP_WAIT_ENTER); return FALSE; } /* Search for the 'Directories' section */ if (!SetupFindFirstLineW(InfFile, L"Directories", NULL, &DirContext)) { if (SourceCabinet) { MUIDisplayError(ERROR_CABINET_SECTION, Ir, POPUP_WAIT_ENTER); } else { MUIDisplayError(ERROR_TXTSETUP_SECTION, Ir, POPUP_WAIT_ENTER); } return FALSE; } /* Enumerate the directory values and create the subdirectories */ do { if (!INF_GetData(&DirContext, NULL, &KeyValue)) { DPRINT1("break\n"); break; } if (KeyValue[0] == L'\\' && KeyValue[1] != 0) { DPRINT("Absolute Path: '%S'\n", KeyValue); wcscpy(PathBuffer, DestinationRootPath.Buffer); wcscat(PathBuffer, KeyValue); DPRINT("FullPath: '%S'\n", PathBuffer); } else if (KeyValue[0] != L'\\') { DPRINT("RelativePath: '%S'\n", KeyValue); wcscpy(PathBuffer, DestinationPath.Buffer); wcscat(PathBuffer, L"\\"); wcscat(PathBuffer, KeyValue); DPRINT("FullPath: '%S'\n", PathBuffer); Status = SetupCreateDirectory(PathBuffer); if (!NT_SUCCESS(Status) && Status != STATUS_OBJECT_NAME_COLLISION) { DPRINT("Creating directory '%S' failed: Status = 0x%08lx", PathBuffer, Status); MUIDisplayError(ERROR_CREATE_DIR, Ir, POPUP_WAIT_ENTER); return FALSE; } } } while (SetupFindNextLine (&DirContext, &DirContext)); return TRUE; } static PAGE_NUMBER PrepareCopyPage(PINPUT_RECORD Ir) { HINF InfHandle; WCHAR PathBuffer[MAX_PATH]; INFCONTEXT CabinetsContext; ULONG InfFileSize; PWCHAR KeyValue; UINT ErrorLine; PVOID InfFileData; MUIDisplayPage(PREPARE_COPY_PAGE); /* Create the file queue */ SetupFileQueue = SetupOpenFileQueue(); if (SetupFileQueue == NULL) { MUIDisplayError(ERROR_COPY_QUEUE, Ir, POPUP_WAIT_ENTER); return(QUIT_PAGE); } if (!PrepareCopyPageInfFile(SetupInf, NULL, Ir)) { return QUIT_PAGE; } /* Search for the 'Cabinets' section */ if (!SetupFindFirstLineW(SetupInf, L"Cabinets", NULL, &CabinetsContext)) { return FILE_COPY_PAGE; } /* * Enumerate the directory values in the 'Cabinets' * section and parse their inf files. */ do { if (!INF_GetData(&CabinetsContext, NULL, &KeyValue)) break; wcscpy(PathBuffer, SourcePath.Buffer); wcscat(PathBuffer, L"\\"); wcscat(PathBuffer, KeyValue); #ifdef __REACTOS__ CabinetInitialize(); CabinetSetEventHandlers(NULL, NULL, NULL); CabinetSetCabinetName(PathBuffer); if (CabinetOpen() == CAB_STATUS_SUCCESS) { DPRINT("Cabinet %S\n", CabinetGetCabinetName()); InfFileData = CabinetGetCabinetReservedArea(&InfFileSize); if (InfFileData == NULL) { MUIDisplayError(ERROR_CABINET_SCRIPT, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } } else { DPRINT("Cannot open cabinet: %S.\n", CabinetGetCabinetName()); MUIDisplayError(ERROR_CABINET_MISSING, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } InfHandle = INF_OpenBufferedFileA((CHAR*) InfFileData, InfFileSize, (const CHAR*) NULL, INF_STYLE_WIN4, LanguageId, &ErrorLine); if (InfHandle == INVALID_HANDLE_VALUE) { MUIDisplayError(ERROR_INVALID_CABINET_INF, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } CabinetCleanup(); if (!PrepareCopyPageInfFile(InfHandle, KeyValue, Ir)) { return QUIT_PAGE; } #endif } while (SetupFindNextLine(&CabinetsContext, &CabinetsContext)); return FILE_COPY_PAGE; } VOID NTAPI SetupUpdateMemoryInfo(IN PCOPYCONTEXT CopyContext, IN BOOLEAN First) { SYSTEM_PERFORMANCE_INFORMATION PerfInfo; /* Get the memory information from the system */ NtQuerySystemInformation(SystemPerformanceInformation, &PerfInfo, sizeof(PerfInfo), NULL); /* Check if this is initial setup */ if (First) { /* Set maximum limits to be total RAM pages */ ProgressSetStepCount(CopyContext->MemoryBars[0], PerfInfo.CommitLimit); ProgressSetStepCount(CopyContext->MemoryBars[1], PerfInfo.CommitLimit); ProgressSetStepCount(CopyContext->MemoryBars[2], PerfInfo.CommitLimit); } /* Set current values */ ProgressSetStep(CopyContext->MemoryBars[0], PerfInfo.PagedPoolPages + PerfInfo.NonPagedPoolPages); ProgressSetStep(CopyContext->MemoryBars[1], PerfInfo.ResidentSystemCachePage); ProgressSetStep(CopyContext->MemoryBars[2], PerfInfo.AvailablePages); } static UINT CALLBACK FileCopyCallback(PVOID Context, UINT Notification, UINT_PTR Param1, UINT_PTR Param2) { PCOPYCONTEXT CopyContext; CopyContext = (PCOPYCONTEXT)Context; switch (Notification) { case SPFILENOTIFY_STARTSUBQUEUE: CopyContext->TotalOperations = (ULONG)Param2; ProgressSetStepCount(CopyContext->ProgressBar, CopyContext->TotalOperations); SetupUpdateMemoryInfo(CopyContext, TRUE); break; case SPFILENOTIFY_STARTCOPY: /* Display copy message */ CONSOLE_SetStatusText(MUIGetString(STRING_COPYING), (PWSTR)Param1); SetupUpdateMemoryInfo(CopyContext, FALSE); break; case SPFILENOTIFY_ENDCOPY: CopyContext->CompletedOperations++; ProgressNextStep(CopyContext->ProgressBar); SetupUpdateMemoryInfo(CopyContext, FALSE); break; } return 0; } static PAGE_NUMBER FileCopyPage(PINPUT_RECORD Ir) { COPYCONTEXT CopyContext; unsigned int mem_bar_width; MUIDisplayPage(FILE_COPY_PAGE); /* Create context for the copy process */ CopyContext.DestinationRootPath = DestinationRootPath.Buffer; CopyContext.InstallPath = InstallPath.Buffer; CopyContext.TotalOperations = 0; CopyContext.CompletedOperations = 0; /* Create the progress bar as well */ CopyContext.ProgressBar = CreateProgressBar(13, 26, xScreen - 13, yScreen - 20, 10, 24, TRUE, MUIGetString(STRING_SETUPCOPYINGFILES)); // fit memory bars to screen width, distribute them uniform mem_bar_width = (xScreen - 26) / 5; mem_bar_width -= mem_bar_width % 2; // make even /* ATTENTION: The following progress bars are debug stuff, which should not be translated!! */ /* Create the paged pool progress bar */ CopyContext.MemoryBars[0] = CreateProgressBar(13, 40, 13 + mem_bar_width, 43, 13, 44, FALSE, "Kernel Pool"); /* Create the non paged pool progress bar */ CopyContext.MemoryBars[1] = CreateProgressBar((xScreen / 2)- (mem_bar_width / 2), 40, (xScreen / 2) + (mem_bar_width / 2), 43, (xScreen / 2)- (mem_bar_width / 2), 44, FALSE, "Kernel Cache"); /* Create the global memory progress bar */ CopyContext.MemoryBars[2] = CreateProgressBar(xScreen - 13 - mem_bar_width, 40, xScreen - 13, 43, xScreen - 13 - mem_bar_width, 44, FALSE, "Free Memory"); /* Do the file copying */ SetupCommitFileQueueW(NULL, SetupFileQueue, FileCopyCallback, &CopyContext); /* If we get here, we're done, so cleanup the queue and progress bar */ SetupCloseFileQueue(SetupFileQueue); DestroyProgressBar(CopyContext.ProgressBar); DestroyProgressBar(CopyContext.MemoryBars[0]); DestroyProgressBar(CopyContext.MemoryBars[1]); DestroyProgressBar(CopyContext.MemoryBars[2]); /* Go display the next page */ return REGISTRY_PAGE; } static PAGE_NUMBER RegistryPage(PINPUT_RECORD Ir) { INFCONTEXT InfContext; PWSTR Action; PWSTR File; PWSTR Section; BOOLEAN Delete; NTSTATUS Status; MUIDisplayPage(REGISTRY_PAGE); if (RepairUpdateFlag) { return SUCCESS_PAGE; } if (!SetInstallPathValue(&DestinationPath)) { DPRINT("SetInstallPathValue() failed\n"); MUIDisplayError(ERROR_INITIALIZE_REGISTRY, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Create the default hives */ #ifdef __REACTOS__ Status = NtInitializeRegistry(CM_BOOT_FLAG_SETUP); if (!NT_SUCCESS(Status)) { DPRINT("NtInitializeRegistry() failed (Status %lx)\n", Status); MUIDisplayError(ERROR_CREATE_HIVE, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } #else RegInitializeRegistry(); #endif /* Update registry */ CONSOLE_SetStatusText(MUIGetString(STRING_REGHIVEUPDATE)); if (!SetupFindFirstLineW(SetupInf, L"HiveInfs.Install", NULL, &InfContext)) { DPRINT1("SetupFindFirstLine() failed\n"); MUIDisplayError(ERROR_FIND_REGISTRY, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } do { INF_GetDataField (&InfContext, 0, &Action); INF_GetDataField (&InfContext, 1, &File); INF_GetDataField (&InfContext, 2, &Section); DPRINT("Action: %S File: %S Section %S\n", Action, File, Section); if (Action == NULL) break; // Hackfix if (!_wcsicmp (Action, L"AddReg")) { Delete = FALSE; } else if (!_wcsicmp (Action, L"DelReg")) { Delete = TRUE; } else { continue; } CONSOLE_SetStatusText(MUIGetString(STRING_IMPORTFILE), File); if (!ImportRegistryFile(File, Section, LanguageId, Delete)) { DPRINT("Importing %S failed\n", File); MUIDisplayError(ERROR_IMPORT_HIVE, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } } while (SetupFindNextLine(&InfContext, &InfContext)); /* Update display registry settings */ CONSOLE_SetStatusText(MUIGetString(STRING_DISPLAYETTINGSUPDATE)); if (!ProcessDisplayRegistry(SetupInf, DisplayList)) { MUIDisplayError(ERROR_UPDATE_DISPLAY_SETTINGS, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Set the locale */ CONSOLE_SetStatusText(MUIGetString(STRING_LOCALESETTINGSUPDATE)); if (!ProcessLocaleRegistry(LanguageList)) { MUIDisplayError(ERROR_UPDATE_LOCALESETTINGS, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Add keyboard layouts */ CONSOLE_SetStatusText(MUIGetString(STRING_ADDKBLAYOUTS)); if (!AddKeyboardLayouts()) { MUIDisplayError(ERROR_ADDING_KBLAYOUTS, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Set GeoID */ if (!SetGeoID(MUIGetGeoID())) { MUIDisplayError(ERROR_UPDATE_GEOID, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } if (!IsUnattendedSetup) { /* Update keyboard layout settings */ CONSOLE_SetStatusText(MUIGetString(STRING_KEYBOARDSETTINGSUPDATE)); if (!ProcessKeyboardLayoutRegistry(LayoutList)) { MUIDisplayError(ERROR_UPDATE_KBSETTINGS, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } } /* Add codepage information to registry */ CONSOLE_SetStatusText(MUIGetString(STRING_CODEPAGEINFOUPDATE)); if (!AddCodePage()) { MUIDisplayError(ERROR_ADDING_CODEPAGE, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Update the mounted devices list */ SetMountedDeviceValues(PartitionList); CONSOLE_SetStatusText(MUIGetString(STRING_DONE)); return BOOT_LOADER_PAGE; } static PAGE_NUMBER BootLoaderPage(PINPUT_RECORD Ir) { UCHAR PartitionType; BOOLEAN InstallOnFloppy; USHORT Line = 12; WCHAR PathBuffer[MAX_PATH]; CONSOLE_SetStatusText(MUIGetString(STRING_PLEASEWAIT)); CheckActiveBootPartition(PartitionList); RtlFreeUnicodeString(&SystemRootPath); swprintf(PathBuffer, L"\\Device\\Harddisk%lu\\Partition%lu", PartitionList->ActiveBootDisk->DiskNumber, PartitionList->ActiveBootPartition-> PartInfo[PartitionList->ActiveBootPartitionNumber].PartitionNumber); RtlCreateUnicodeString(&SystemRootPath, PathBuffer); DPRINT("SystemRootPath: %wZ\n", &SystemRootPath); PartitionType = PartitionList->ActiveBootPartition-> PartInfo[PartitionList->ActiveBootPartitionNumber].PartitionType; if (IsUnattendedSetup) { if (UnattendMBRInstallType == 0) /* skip MBR installation */ { return SUCCESS_PAGE; } else if (UnattendMBRInstallType == 1) /* install on floppy */ { return BOOT_LOADER_FLOPPY_PAGE; } } if (PartitionType == PARTITION_ENTRY_UNUSED) { DPRINT("Error: active partition invalid (unused)\n"); InstallOnFloppy = TRUE; } else if (PartitionType == 0x0A) { /* OS/2 boot manager partition */ DPRINT("Found OS/2 boot manager partition\n"); InstallOnFloppy = TRUE; } else if (PartitionType == 0x83) { /* Linux ext2 partition */ DPRINT("Found Linux ext2 partition\n"); InstallOnFloppy = TRUE; } else if (PartitionType == PARTITION_IFS) { /* NTFS partition */ DPRINT("Found NTFS partition\n"); InstallOnFloppy = TRUE; } else if ((PartitionType == PARTITION_FAT_12) || (PartitionType == PARTITION_FAT_16) || (PartitionType == PARTITION_HUGE) || (PartitionType == PARTITION_XINT13) || (PartitionType == PARTITION_FAT32) || (PartitionType == PARTITION_FAT32_XINT13)) { DPRINT("Found FAT partition\n"); InstallOnFloppy = FALSE; } else { /* Unknown partition */ DPRINT("Unknown partition found\n"); InstallOnFloppy = TRUE; } if (InstallOnFloppy == TRUE) { return BOOT_LOADER_FLOPPY_PAGE; } /* Unattended install on hdd? */ if (IsUnattendedSetup && UnattendMBRInstallType == 2) { return BOOT_LOADER_HARDDISK_MBR_PAGE; } MUIDisplayPage(BOOT_LOADER_PAGE); CONSOLE_InvertTextXY(8, Line, 60, 1); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_DOWN)) /* DOWN */ { CONSOLE_NormalTextXY(8, Line, 60, 1); Line++; if (Line<12) Line=15; if (Line>15) Line=12; CONSOLE_InvertTextXY(8, Line, 60, 1); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_UP)) /* UP */ { CONSOLE_NormalTextXY(8, Line, 60, 1); Line--; if (Line<12) Line=15; if (Line>15) Line=12; CONSOLE_InvertTextXY(8, Line, 60, 1); } else if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { if (Line == 12) { return BOOT_LOADER_HARDDISK_MBR_PAGE; } else if (Line == 13) { return BOOT_LOADER_HARDDISK_VBR_PAGE; } else if (Line == 14) { return BOOT_LOADER_FLOPPY_PAGE; } else if (Line == 15) { return SUCCESS_PAGE; } return BOOT_LOADER_PAGE; } } return BOOT_LOADER_PAGE; } static PAGE_NUMBER BootLoaderFloppyPage(PINPUT_RECORD Ir) { NTSTATUS Status; MUIDisplayPage(BOOT_LOADER_FLOPPY_PAGE); // SetStatusText(" Please wait..."); while (TRUE) { CONSOLE_ConInKey(Ir); if ((Ir->Event.KeyEvent.uChar.AsciiChar == 0x00) && (Ir->Event.KeyEvent.wVirtualKeyCode == VK_F3)) /* F3 */ { if (ConfirmQuit(Ir) == TRUE) return QUIT_PAGE; break; } else if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { if (DoesFileExist(L"\\Device\\Floppy0", L"\\") == FALSE) { MUIDisplayError(ERROR_NO_FLOPPY, Ir, POPUP_WAIT_ENTER); return BOOT_LOADER_FLOPPY_PAGE; } Status = InstallFatBootcodeToFloppy(&SourceRootPath, &DestinationArcPath); if (!NT_SUCCESS(Status)) { /* Print error message */ return BOOT_LOADER_FLOPPY_PAGE; } return SUCCESS_PAGE; } } return BOOT_LOADER_FLOPPY_PAGE; } static PAGE_NUMBER BootLoaderHarddiskVbrPage(PINPUT_RECORD Ir) { UCHAR PartitionType; NTSTATUS Status; PartitionType = PartitionList->ActiveBootPartition-> PartInfo[PartitionList->ActiveBootPartitionNumber].PartitionType; Status = InstallVBRToPartition(&SystemRootPath, &SourceRootPath, &DestinationArcPath, PartitionType); if (!NT_SUCCESS(Status)) { MUIDisplayError(ERROR_WRITE_BOOT, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } return SUCCESS_PAGE; } static PAGE_NUMBER BootLoaderHarddiskMbrPage(PINPUT_RECORD Ir) { UCHAR PartitionType; NTSTATUS Status; WCHAR DestinationDevicePathBuffer[MAX_PATH]; WCHAR SourceMbrPathBuffer[MAX_PATH]; /* Step 1: Write the VBR */ PartitionType = PartitionList->ActiveBootPartition-> PartInfo[PartitionList->ActiveBootPartitionNumber].PartitionType; Status = InstallVBRToPartition(&SystemRootPath, &SourceRootPath, &DestinationArcPath, PartitionType); if (!NT_SUCCESS(Status)) { MUIDisplayError(ERROR_WRITE_BOOT, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } /* Step 2: Write the MBR */ swprintf(DestinationDevicePathBuffer, L"\\Device\\Harddisk%d\\Partition0", PartitionList->ActiveBootDisk->DiskNumber); wcscpy(SourceMbrPathBuffer, SourceRootPath.Buffer); wcscat(SourceMbrPathBuffer, L"\\loader\\dosmbr.bin"); DPRINT("Install MBR bootcode: %S ==> %S\n", SourceMbrPathBuffer, DestinationDevicePathBuffer); Status = InstallMbrBootCodeToDisk(SourceMbrPathBuffer, DestinationDevicePathBuffer); if (!NT_SUCCESS (Status)) { DPRINT1("InstallMbrBootCodeToDisk() failed (Status %lx)\n", Status); MUIDisplayError(ERROR_INSTALL_BOOTCODE, Ir, POPUP_WAIT_ENTER); return QUIT_PAGE; } return SUCCESS_PAGE; } static PAGE_NUMBER QuitPage(PINPUT_RECORD Ir) { MUIDisplayPage(QUIT_PAGE); /* Destroy partition list */ if (PartitionList != NULL) { DestroyPartitionList (PartitionList); PartitionList = NULL; } /* Destroy filesystem list */ if (FileSystemList != NULL) { DestroyFileSystemList (FileSystemList); FileSystemList = NULL; } /* Destroy computer settings list */ if (ComputerList != NULL) { DestroyGenericList(ComputerList, TRUE); ComputerList = NULL; } /* Destroy display settings list */ if (DisplayList != NULL) { DestroyGenericList(DisplayList, TRUE); DisplayList = NULL; } /* Destroy keyboard settings list */ if (KeyboardList != NULL) { DestroyGenericList(KeyboardList, TRUE); KeyboardList = NULL; } /* Destroy keyboard layout list */ if (LayoutList != NULL) { DestroyGenericList(LayoutList, TRUE); LayoutList = NULL; } if (LanguageList != NULL) { DestroyGenericList(LanguageList, FALSE); LanguageList = NULL; } CONSOLE_SetStatusText(MUIGetString(STRING_REBOOTCOMPUTER2)); while (TRUE) { CONSOLE_ConInKey(Ir); if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return FLUSH_PAGE; } } } static PAGE_NUMBER SuccessPage(PINPUT_RECORD Ir) { MUIDisplayPage(SUCCESS_PAGE); if (IsUnattendedSetup) { return FLUSH_PAGE; } while (TRUE) { CONSOLE_ConInKey(Ir); if (Ir->Event.KeyEvent.uChar.AsciiChar == 0x0D) /* ENTER */ { return FLUSH_PAGE; } } } static PAGE_NUMBER FlushPage(PINPUT_RECORD Ir) { MUIDisplayPage(FLUSH_PAGE); return REBOOT_PAGE; } DWORD WINAPI PnpEventThread(IN LPVOID lpParameter); VOID RunUSetup(VOID) { INPUT_RECORD Ir; PAGE_NUMBER Page; LARGE_INTEGER Time; NTSTATUS Status; NtQuerySystemTime(&Time); Status = RtlCreateUserThread(NtCurrentProcess(), NULL, TRUE, 0, 0, 0, PnpEventThread, &SetupInf, &hPnpThread, NULL); if (!NT_SUCCESS(Status)) hPnpThread = INVALID_HANDLE_VALUE; if (!CONSOLE_Init()) { PrintString(MUIGetString(STRING_CONSOLEFAIL1)); PrintString(MUIGetString(STRING_CONSOLEFAIL2)); PrintString(MUIGetString(STRING_CONSOLEFAIL3)); /* Raise a hard error (crash the system/BSOD) */ NtRaiseHardError(STATUS_SYSTEM_PROCESS_TERMINATED, 0,0,0,0,0); } /* Initialize global unicode strings */ RtlInitUnicodeString(&SourcePath, NULL); RtlInitUnicodeString(&SourceRootPath, NULL); RtlInitUnicodeString(&SourceRootDir, NULL); RtlInitUnicodeString(&InstallPath, NULL); RtlInitUnicodeString(&DestinationPath, NULL); RtlInitUnicodeString(&DestinationArcPath, NULL); RtlInitUnicodeString(&DestinationRootPath, NULL); RtlInitUnicodeString(&SystemRootPath, NULL); /* Hide the cursor */ CONSOLE_SetCursorType(TRUE, FALSE); Page = START_PAGE; while (Page != REBOOT_PAGE) { CONSOLE_ClearScreen(); CONSOLE_Flush(); //CONSOLE_SetUnderlinedTextXY(4, 3, " ReactOS " KERNEL_VERSION_STR " Setup "); //CONSOLE_Flush(); switch (Page) { /* Start page */ case START_PAGE: Page = SetupStartPage(&Ir); break; /* Language page */ case LANGUAGE_PAGE: Page = LanguagePage(&Ir); break; /* License page */ case LICENSE_PAGE: Page = LicensePage(&Ir); break; /* Intro page */ case INTRO_PAGE: Page = IntroPage(&Ir); break; /* Install pages */ case INSTALL_INTRO_PAGE: Page = InstallIntroPage(&Ir); break; #if 0 case SCSI_CONTROLLER_PAGE: Page = ScsiControllerPage(&Ir); break; #endif #if 0 case OEM_DRIVER_PAGE: Page = OemDriverPage(&Ir); break; #endif case DEVICE_SETTINGS_PAGE: Page = DeviceSettingsPage(&Ir); break; case COMPUTER_SETTINGS_PAGE: Page = ComputerSettingsPage(&Ir); break; case DISPLAY_SETTINGS_PAGE: Page = DisplaySettingsPage(&Ir); break; case KEYBOARD_SETTINGS_PAGE: Page = KeyboardSettingsPage(&Ir); break; case LAYOUT_SETTINGS_PAGE: Page = LayoutSettingsPage(&Ir); break; case SELECT_PARTITION_PAGE: Page = SelectPartitionPage(&Ir); break; case CREATE_PARTITION_PAGE: Page = CreatePartitionPage(&Ir); break; case DELETE_PARTITION_PAGE: Page = DeletePartitionPage(&Ir); break; case SELECT_FILE_SYSTEM_PAGE: Page = SelectFileSystemPage(&Ir); break; case FORMAT_PARTITION_PAGE: Page = (PAGE_NUMBER) FormatPartitionPage(&Ir); break; case CHECK_FILE_SYSTEM_PAGE: Page = (PAGE_NUMBER) CheckFileSystemPage(&Ir); break; case INSTALL_DIRECTORY_PAGE: Page = InstallDirectoryPage(&Ir); break; case PREPARE_COPY_PAGE: Page = PrepareCopyPage(&Ir); break; case FILE_COPY_PAGE: Page = FileCopyPage(&Ir); break; case REGISTRY_PAGE: Page = RegistryPage(&Ir); break; case BOOT_LOADER_PAGE: Page = BootLoaderPage(&Ir); break; case BOOT_LOADER_FLOPPY_PAGE: Page = BootLoaderFloppyPage(&Ir); break; case BOOT_LOADER_HARDDISK_MBR_PAGE: Page = BootLoaderHarddiskMbrPage(&Ir); break; case BOOT_LOADER_HARDDISK_VBR_PAGE: Page = BootLoaderHarddiskVbrPage(&Ir); break; /* Repair pages */ case REPAIR_INTRO_PAGE: Page = RepairIntroPage(&Ir); break; case SUCCESS_PAGE: Page = SuccessPage(&Ir); break; case FLUSH_PAGE: Page = FlushPage(&Ir); break; case QUIT_PAGE: Page = QuitPage(&Ir); break; case REBOOT_PAGE: break; } } FreeConsole(); /* Avoid bugcheck */ Time.QuadPart += 50000000; NtDelayExecution(FALSE, &Time); /* Reboot */ NtShutdownSystem(ShutdownReboot); NtTerminateProcess(NtCurrentProcess(), 0); } #ifdef __REACTOS__ VOID NTAPI NtProcessStartup(PPEB Peb) { RtlNormalizeProcessParams(Peb->ProcessParameters); ProcessHeap = Peb->ProcessHeap; InfSetHeap(ProcessHeap); RunUSetup(); } #endif /* __REACTOS__ */ /* EOF */