1cf200d32Sopenharmony_ci/* 2cf200d32Sopenharmony_ci Implementation of GPTData class derivative with popt-based command 3cf200d32Sopenharmony_ci line processing 4cf200d32Sopenharmony_ci Copyright (C) 2010-2024 Roderick W. Smith 5cf200d32Sopenharmony_ci 6cf200d32Sopenharmony_ci This program is free software; you can redistribute it and/or modify 7cf200d32Sopenharmony_ci it under the terms of the GNU General Public License as published by 8cf200d32Sopenharmony_ci the Free Software Foundation; either version 2 of the License, or 9cf200d32Sopenharmony_ci (at your option) any later version. 10cf200d32Sopenharmony_ci 11cf200d32Sopenharmony_ci This program is distributed in the hope that it will be useful, 12cf200d32Sopenharmony_ci but WITHOUT ANY WARRANTY; without even the implied warranty of 13cf200d32Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 14cf200d32Sopenharmony_ci GNU General Public License for more details. 15cf200d32Sopenharmony_ci 16cf200d32Sopenharmony_ci You should have received a copy of the GNU General Public License along 17cf200d32Sopenharmony_ci with this program; if not, write to the Free Software Foundation, Inc., 18cf200d32Sopenharmony_ci 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. 19cf200d32Sopenharmony_ci*/ 20cf200d32Sopenharmony_ci 21cf200d32Sopenharmony_ci#include <string.h> 22cf200d32Sopenharmony_ci#include <string> 23cf200d32Sopenharmony_ci#include <iostream> 24cf200d32Sopenharmony_ci#include <sstream> 25cf200d32Sopenharmony_ci#include <errno.h> 26cf200d32Sopenharmony_ci#include <popt.h> 27cf200d32Sopenharmony_ci#include "gptcl.h" 28cf200d32Sopenharmony_ci 29cf200d32Sopenharmony_ciusing namespace std; 30cf200d32Sopenharmony_ci 31cf200d32Sopenharmony_ciGPTDataCL::GPTDataCL(void) { 32cf200d32Sopenharmony_ci attributeOperation = backupFile = partName = hybrids = newPartInfo = NULL; 33cf200d32Sopenharmony_ci mbrParts = twoParts = outDevice = typeCode = partGUID = diskGUID = NULL; 34cf200d32Sopenharmony_ci alignment = DEFAULT_ALIGNMENT; 35cf200d32Sopenharmony_ci alignEnd = false; 36cf200d32Sopenharmony_ci deletePartNum = infoPartNum = largestPartNum = bsdPartNum = 0; 37cf200d32Sopenharmony_ci tableSize = GPT_SIZE; 38cf200d32Sopenharmony_ci} // GPTDataCL constructor 39cf200d32Sopenharmony_ci 40cf200d32Sopenharmony_ciGPTDataCL::GPTDataCL(string filename) { 41cf200d32Sopenharmony_ci} // GPTDataCL constructor with filename 42cf200d32Sopenharmony_ci 43cf200d32Sopenharmony_ciGPTDataCL::~GPTDataCL(void) { 44cf200d32Sopenharmony_ci} // GPTDataCL destructor 45cf200d32Sopenharmony_ci 46cf200d32Sopenharmony_civoid GPTDataCL::LoadBackupFile(string backupFile, int &saveData, int &neverSaveData) { 47cf200d32Sopenharmony_ci if (LoadGPTBackup(backupFile) == 1) { 48cf200d32Sopenharmony_ci JustLooking(0); 49cf200d32Sopenharmony_ci saveData = 1; 50cf200d32Sopenharmony_ci } else { 51cf200d32Sopenharmony_ci saveData = 0; 52cf200d32Sopenharmony_ci neverSaveData = 1; 53cf200d32Sopenharmony_ci cerr << "Error loading backup file!\n"; 54cf200d32Sopenharmony_ci } // else 55cf200d32Sopenharmony_ci} // GPTDataCL::LoadBackupFile() 56cf200d32Sopenharmony_ci 57cf200d32Sopenharmony_ci// Perform the actions specified on the command line. This is necessarily one 58cf200d32Sopenharmony_ci// monster of a function! 59cf200d32Sopenharmony_ci// Returns values: 60cf200d32Sopenharmony_ci// 0 = success 61cf200d32Sopenharmony_ci// 1 = too few arguments 62cf200d32Sopenharmony_ci// 2 = error when reading partition table 63cf200d32Sopenharmony_ci// 3 = non-GPT disk and no -g option 64cf200d32Sopenharmony_ci// 4 = unable to save changes 65cf200d32Sopenharmony_ci// 8 = disk replication operation (-R) failed 66cf200d32Sopenharmony_ciint GPTDataCL::DoOptions(int argc, char* argv[]) { 67cf200d32Sopenharmony_ci GPTData secondDevice; 68cf200d32Sopenharmony_ci int opt, numOptions = 0, saveData = 0, neverSaveData = 0; 69cf200d32Sopenharmony_ci int partNum = 0, newPartNum = -1, saveNonGPT = 1, retval = 0, pretend = 0; 70cf200d32Sopenharmony_ci int byteSwapPartNum = 0; 71cf200d32Sopenharmony_ci uint64_t low, high, startSector, endSector, sSize, mainTableLBA, secondTableLBA; 72cf200d32Sopenharmony_ci uint64_t temp; // temporary variable; free to use in any case 73cf200d32Sopenharmony_ci char *device; 74cf200d32Sopenharmony_ci string cmd, typeGUID, name; 75cf200d32Sopenharmony_ci PartType typeHelper; 76cf200d32Sopenharmony_ci 77cf200d32Sopenharmony_ci struct poptOption theOptions[] = 78cf200d32Sopenharmony_ci { 79cf200d32Sopenharmony_ci {"attributes", 'A', POPT_ARG_STRING, &attributeOperation, 'A', "operate on partition attributes", 80cf200d32Sopenharmony_ci "list|[partnum:show|or|nand|xor|=|set|clear|toggle|get[:bitnum|hexbitmask]]"}, 81cf200d32Sopenharmony_ci {"set-alignment", 'a', POPT_ARG_INT, &alignment, 'a', "set sector alignment", "value"}, 82cf200d32Sopenharmony_ci {"backup", 'b', POPT_ARG_STRING, &backupFile, 'b', "backup GPT to file", "file"}, 83cf200d32Sopenharmony_ci {"byte-swap-name", 'B', POPT_ARG_INT, &byteSwapPartNum, 'B', "byte-swap partition's name", "partnum"}, 84cf200d32Sopenharmony_ci {"change-name", 'c', POPT_ARG_STRING, &partName, 'c', "change partition's name", "partnum:name"}, 85cf200d32Sopenharmony_ci {"recompute-chs", 'C', POPT_ARG_NONE, NULL, 'C', "recompute CHS values in protective/hybrid MBR", ""}, 86cf200d32Sopenharmony_ci {"delete", 'd', POPT_ARG_INT, &deletePartNum, 'd', "delete a partition", "partnum"}, 87cf200d32Sopenharmony_ci {"display-alignment", 'D', POPT_ARG_NONE, NULL, 'D', "show number of sectors per allocation block", ""}, 88cf200d32Sopenharmony_ci {"move-second-header", 'e', POPT_ARG_NONE, NULL, 'e', "move second/backup header to end of disk", ""}, 89cf200d32Sopenharmony_ci {"end-of-largest", 'E', POPT_ARG_NONE, NULL, 'E', "show end of largest free block", ""}, 90cf200d32Sopenharmony_ci {"first-in-largest", 'f', POPT_ARG_NONE, NULL, 'f', "show start of the largest free block", ""}, 91cf200d32Sopenharmony_ci {"first-aligned-in-largest", 'F', POPT_ARG_NONE, NULL, 'F', "show start of the largest free block, aligned", ""}, 92cf200d32Sopenharmony_ci {"mbrtogpt", 'g', POPT_ARG_NONE, NULL, 'g', "convert MBR to GPT", ""}, 93cf200d32Sopenharmony_ci {"randomize-guids", 'G', POPT_ARG_NONE, NULL, 'G', "randomize disk and partition GUIDs", ""}, 94cf200d32Sopenharmony_ci {"hybrid", 'h', POPT_ARG_STRING, &hybrids, 'h', "create hybrid MBR", "partnum[:partnum...][:EE]"}, 95cf200d32Sopenharmony_ci {"info", 'i', POPT_ARG_INT, &infoPartNum, 'i', "show detailed information on partition", "partnum"}, 96cf200d32Sopenharmony_ci {"align-end", 'I', POPT_ARG_NONE, NULL, 'I', "align partition end points", ""}, 97cf200d32Sopenharmony_ci {"move-main-table", 'j', POPT_ARG_INT, &mainTableLBA, 'j', "change the start sector of the main partition table", "sector"}, 98cf200d32Sopenharmony_ci {"move-backup-table", 'k', POPT_ARG_INT, &secondTableLBA, 'k', "change the start sector of the second/backup partition table", "sector"}, 99cf200d32Sopenharmony_ci {"load-backup", 'l', POPT_ARG_STRING, &backupFile, 'l', "load GPT backup from file", "file"}, 100cf200d32Sopenharmony_ci {"list-types", 'L', POPT_ARG_NONE, NULL, 'L', "list known partition types", ""}, 101cf200d32Sopenharmony_ci {"gpttombr", 'm', POPT_ARG_STRING, &mbrParts, 'm', "convert GPT to MBR", "partnum[:partnum...]"}, 102cf200d32Sopenharmony_ci {"new", 'n', POPT_ARG_STRING, &newPartInfo, 'n', "create new partition", "partnum:start:end"}, 103cf200d32Sopenharmony_ci {"largest-new", 'N', POPT_ARG_INT, &largestPartNum, 'N', "create largest possible new partition", "partnum"}, 104cf200d32Sopenharmony_ci {"clear", 'o', POPT_ARG_NONE, NULL, 'o', "clear partition table", ""}, 105cf200d32Sopenharmony_ci {"print-mbr", 'O', POPT_ARG_NONE, NULL, 'O', "print MBR partition table", ""}, 106cf200d32Sopenharmony_ci {"print", 'p', POPT_ARG_NONE, NULL, 'p', "print partition table", ""}, 107cf200d32Sopenharmony_ci {"pretend", 'P', POPT_ARG_NONE, NULL, 'P', "make changes in memory, but don't write them", ""}, 108cf200d32Sopenharmony_ci {"transpose", 'r', POPT_ARG_STRING, &twoParts, 'r', "transpose two partitions", "partnum:partnum"}, 109cf200d32Sopenharmony_ci {"replicate", 'R', POPT_ARG_STRING, &outDevice, 'R', "replicate partition table", "device_filename"}, 110cf200d32Sopenharmony_ci {"sort", 's', POPT_ARG_NONE, NULL, 's', "sort partition table entries", ""}, 111cf200d32Sopenharmony_ci {"resize-table", 'S', POPT_ARG_INT, &tableSize, 'S', "resize partition table", "numparts"}, 112cf200d32Sopenharmony_ci {"typecode", 't', POPT_ARG_STRING, &typeCode, 't', "change partition type code", "partnum:{hexcode|GUID}"}, 113cf200d32Sopenharmony_ci {"transform-bsd", 'T', POPT_ARG_INT, &bsdPartNum, 'T', "transform BSD disklabel partition to GPT", "partnum"}, 114cf200d32Sopenharmony_ci {"partition-guid", 'u', POPT_ARG_STRING, &partGUID, 'u', "set partition GUID", "partnum:guid"}, 115cf200d32Sopenharmony_ci {"disk-guid", 'U', POPT_ARG_STRING, &diskGUID, 'U', "set disk GUID", "guid"}, 116cf200d32Sopenharmony_ci {"verify", 'v', POPT_ARG_NONE, NULL, 'v', "check partition table integrity", ""}, 117cf200d32Sopenharmony_ci {"version", 'V', POPT_ARG_NONE, NULL, 'V', "display version information", ""}, 118cf200d32Sopenharmony_ci {"zap", 'z', POPT_ARG_NONE, NULL, 'z', "zap (destroy) GPT (but not MBR) data structures", ""}, 119cf200d32Sopenharmony_ci {"zap-all", 'Z', POPT_ARG_NONE, NULL, 'Z', "zap (destroy) GPT and MBR data structures", ""}, 120cf200d32Sopenharmony_ci POPT_AUTOHELP { NULL, 0, 0, NULL, 0 } 121cf200d32Sopenharmony_ci }; 122cf200d32Sopenharmony_ci 123cf200d32Sopenharmony_ci // Create popt context... 124cf200d32Sopenharmony_ci poptCon = poptGetContext(NULL, argc, (const char**) argv, theOptions, 0); 125cf200d32Sopenharmony_ci 126cf200d32Sopenharmony_ci poptSetOtherOptionHelp(poptCon, " [OPTION...] <device>"); 127cf200d32Sopenharmony_ci 128cf200d32Sopenharmony_ci if (argc < 2) { 129cf200d32Sopenharmony_ci poptPrintUsage(poptCon, stderr, 0); 130cf200d32Sopenharmony_ci return 1; 131cf200d32Sopenharmony_ci } 132cf200d32Sopenharmony_ci 133cf200d32Sopenharmony_ci // Do one loop through the options to find the device filename and deal 134cf200d32Sopenharmony_ci // with options that don't require a device filename, to flag destructive 135cf200d32Sopenharmony_ci // (o, z, or Z) options, and to flag presence of a --pretend/-P option 136cf200d32Sopenharmony_ci while ((opt = poptGetNextOpt(poptCon)) > 0) { 137cf200d32Sopenharmony_ci switch (opt) { 138cf200d32Sopenharmony_ci case 'A': 139cf200d32Sopenharmony_ci cmd = GetString(attributeOperation, 1); 140cf200d32Sopenharmony_ci if (cmd == "list") 141cf200d32Sopenharmony_ci Attributes::ListAttributes(); 142cf200d32Sopenharmony_ci break; 143cf200d32Sopenharmony_ci case 'L': 144cf200d32Sopenharmony_ci typeHelper.ShowAllTypes(0); 145cf200d32Sopenharmony_ci break; 146cf200d32Sopenharmony_ci case 'P': 147cf200d32Sopenharmony_ci pretend = 1; 148cf200d32Sopenharmony_ci break; 149cf200d32Sopenharmony_ci case 'V': 150cf200d32Sopenharmony_ci cout << "GPT fdisk (sgdisk) version " << GPTFDISK_VERSION << "\n\n"; 151cf200d32Sopenharmony_ci break; 152cf200d32Sopenharmony_ci default: 153cf200d32Sopenharmony_ci break; 154cf200d32Sopenharmony_ci } // switch 155cf200d32Sopenharmony_ci numOptions++; 156cf200d32Sopenharmony_ci } // while 157cf200d32Sopenharmony_ci 158cf200d32Sopenharmony_ci // Assume first non-option argument is the device filename.... 159cf200d32Sopenharmony_ci device = (char*) poptGetArg(poptCon); 160cf200d32Sopenharmony_ci 161cf200d32Sopenharmony_ci if (device != NULL) { 162cf200d32Sopenharmony_ci device = strdup(device); 163cf200d32Sopenharmony_ci poptResetContext(poptCon); 164cf200d32Sopenharmony_ci JustLooking(); // reset as necessary 165cf200d32Sopenharmony_ci BeQuiet(); // Tell called functions to be less verbose & interactive 166cf200d32Sopenharmony_ci if (LoadPartitions((string) device)) { 167cf200d32Sopenharmony_ci if ((WhichWasUsed() == use_mbr) || (WhichWasUsed() == use_bsd)) 168cf200d32Sopenharmony_ci saveNonGPT = 0; // flag so we don't overwrite unless directed to do so 169cf200d32Sopenharmony_ci sSize = GetBlockSize(); 170cf200d32Sopenharmony_ci while ((opt = poptGetNextOpt(poptCon)) > 0) { 171cf200d32Sopenharmony_ci switch (opt) { 172cf200d32Sopenharmony_ci case 'A': { 173cf200d32Sopenharmony_ci if (cmd != "list") { 174cf200d32Sopenharmony_ci partNum = (int) GetInt(attributeOperation, 1) - 1; 175cf200d32Sopenharmony_ci if (partNum < 0) 176cf200d32Sopenharmony_ci partNum = newPartNum; 177cf200d32Sopenharmony_ci if ((partNum >= 0) && (partNum < (int) GetNumParts())) { 178cf200d32Sopenharmony_ci switch (ManageAttributes(partNum, GetString(attributeOperation, 2), 179cf200d32Sopenharmony_ci GetString(attributeOperation, 3))) { 180cf200d32Sopenharmony_ci case -1: 181cf200d32Sopenharmony_ci saveData = 0; 182cf200d32Sopenharmony_ci neverSaveData = 1; 183cf200d32Sopenharmony_ci break; 184cf200d32Sopenharmony_ci case 1: 185cf200d32Sopenharmony_ci JustLooking(0); 186cf200d32Sopenharmony_ci saveData = 1; 187cf200d32Sopenharmony_ci break; 188cf200d32Sopenharmony_ci default: 189cf200d32Sopenharmony_ci break; 190cf200d32Sopenharmony_ci } // switch 191cf200d32Sopenharmony_ci } else { 192cf200d32Sopenharmony_ci cerr << "Error: Invalid partition number " << partNum + 1 << "\n"; 193cf200d32Sopenharmony_ci saveData = 0; 194cf200d32Sopenharmony_ci neverSaveData = 1; 195cf200d32Sopenharmony_ci } // if/else reasonable partition # 196cf200d32Sopenharmony_ci } // if (cmd != "list") 197cf200d32Sopenharmony_ci break; 198cf200d32Sopenharmony_ci } // case 'A': 199cf200d32Sopenharmony_ci case 'a': 200cf200d32Sopenharmony_ci SetAlignment(alignment); 201cf200d32Sopenharmony_ci break; 202cf200d32Sopenharmony_ci case 'B': 203cf200d32Sopenharmony_ci if (IsUsedPartNum(byteSwapPartNum - 1)) { 204cf200d32Sopenharmony_ci partitions[byteSwapPartNum - 1].ReverseNameBytes(); 205cf200d32Sopenharmony_ci cout << "Changed partition " << byteSwapPartNum << "'s name to " 206cf200d32Sopenharmony_ci << partitions[byteSwapPartNum - 1].GetDescription() << "\n"; 207cf200d32Sopenharmony_ci JustLooking(0); 208cf200d32Sopenharmony_ci saveData = 1; 209cf200d32Sopenharmony_ci } 210cf200d32Sopenharmony_ci break; 211cf200d32Sopenharmony_ci case 'b': 212cf200d32Sopenharmony_ci SaveGPTBackup(backupFile); 213cf200d32Sopenharmony_ci free(backupFile); 214cf200d32Sopenharmony_ci break; 215cf200d32Sopenharmony_ci case 'c': 216cf200d32Sopenharmony_ci JustLooking(0); 217cf200d32Sopenharmony_ci partNum = (int) GetInt(partName, 1) - 1; 218cf200d32Sopenharmony_ci if (partNum < 0) 219cf200d32Sopenharmony_ci partNum = newPartNum; 220cf200d32Sopenharmony_ci if ((partNum >= 0) && (partNum < (int) GetNumParts())) { 221cf200d32Sopenharmony_ci name = GetString(partName, 2); 222cf200d32Sopenharmony_ci if (SetName(partNum, (UnicodeString) name.c_str())) { 223cf200d32Sopenharmony_ci saveData = 1; 224cf200d32Sopenharmony_ci } else { 225cf200d32Sopenharmony_ci cerr << "Unable to set partition " << partNum + 1 226cf200d32Sopenharmony_ci << "'s name to '" << GetString(partName, 2) << "'!\n"; 227cf200d32Sopenharmony_ci neverSaveData = 1; 228cf200d32Sopenharmony_ci } // if/else 229cf200d32Sopenharmony_ci free(partName); 230cf200d32Sopenharmony_ci } 231cf200d32Sopenharmony_ci break; 232cf200d32Sopenharmony_ci case 'C': 233cf200d32Sopenharmony_ci JustLooking(0); 234cf200d32Sopenharmony_ci RecomputeCHS(); 235cf200d32Sopenharmony_ci saveData = 1; 236cf200d32Sopenharmony_ci break; 237cf200d32Sopenharmony_ci case 'd': 238cf200d32Sopenharmony_ci JustLooking(0); 239cf200d32Sopenharmony_ci if (DeletePartition(deletePartNum - 1) == 0) { 240cf200d32Sopenharmony_ci cerr << "Error " << errno << " deleting partition!\n"; 241cf200d32Sopenharmony_ci neverSaveData = 1; 242cf200d32Sopenharmony_ci } else saveData = 1; 243cf200d32Sopenharmony_ci break; 244cf200d32Sopenharmony_ci case 'D': 245cf200d32Sopenharmony_ci cout << GetAlignment() << "\n"; 246cf200d32Sopenharmony_ci break; 247cf200d32Sopenharmony_ci case 'e': 248cf200d32Sopenharmony_ci JustLooking(0); 249cf200d32Sopenharmony_ci MoveSecondHeaderToEnd(); 250cf200d32Sopenharmony_ci saveData = 1; 251cf200d32Sopenharmony_ci break; 252cf200d32Sopenharmony_ci case 'E': 253cf200d32Sopenharmony_ci cout << FindLastInFree(FindFirstInLargest()) << "\n"; 254cf200d32Sopenharmony_ci break; 255cf200d32Sopenharmony_ci case 'f': 256cf200d32Sopenharmony_ci cout << FindFirstInLargest() << "\n"; 257cf200d32Sopenharmony_ci break; 258cf200d32Sopenharmony_ci case 'F': 259cf200d32Sopenharmony_ci temp = FindFirstInLargest(); 260cf200d32Sopenharmony_ci Align(&temp); 261cf200d32Sopenharmony_ci cout << temp << "\n"; 262cf200d32Sopenharmony_ci break; 263cf200d32Sopenharmony_ci case 'g': 264cf200d32Sopenharmony_ci JustLooking(0); 265cf200d32Sopenharmony_ci saveData = 1; 266cf200d32Sopenharmony_ci saveNonGPT = 1; 267cf200d32Sopenharmony_ci break; 268cf200d32Sopenharmony_ci case 'G': 269cf200d32Sopenharmony_ci JustLooking(0); 270cf200d32Sopenharmony_ci saveData = 1; 271cf200d32Sopenharmony_ci RandomizeGUIDs(); 272cf200d32Sopenharmony_ci break; 273cf200d32Sopenharmony_ci case 'h': 274cf200d32Sopenharmony_ci JustLooking(0); 275cf200d32Sopenharmony_ci if (BuildMBR(hybrids, 1) == 1) 276cf200d32Sopenharmony_ci saveData = 1; 277cf200d32Sopenharmony_ci break; 278cf200d32Sopenharmony_ci case 'i': 279cf200d32Sopenharmony_ci ShowPartDetails(infoPartNum - 1); 280cf200d32Sopenharmony_ci break; 281cf200d32Sopenharmony_ci case 'I': 282cf200d32Sopenharmony_ci alignEnd = true; 283cf200d32Sopenharmony_ci break; 284cf200d32Sopenharmony_ci case 'j': 285cf200d32Sopenharmony_ci if (MoveMainTable(mainTableLBA)) { 286cf200d32Sopenharmony_ci JustLooking(0); 287cf200d32Sopenharmony_ci saveData = 1; 288cf200d32Sopenharmony_ci } else { 289cf200d32Sopenharmony_ci neverSaveData = 1; 290cf200d32Sopenharmony_ci } // if/else 291cf200d32Sopenharmony_ci break; 292cf200d32Sopenharmony_ci case 'k': 293cf200d32Sopenharmony_ci if (MoveSecondTable(secondTableLBA)) { 294cf200d32Sopenharmony_ci JustLooking(0); 295cf200d32Sopenharmony_ci saveData = 1; 296cf200d32Sopenharmony_ci } else { 297cf200d32Sopenharmony_ci neverSaveData = 1; 298cf200d32Sopenharmony_ci } // if/else 299cf200d32Sopenharmony_ci break; 300cf200d32Sopenharmony_ci case 'l': 301cf200d32Sopenharmony_ci LoadBackupFile(backupFile, saveData, neverSaveData); 302cf200d32Sopenharmony_ci free(backupFile); 303cf200d32Sopenharmony_ci break; 304cf200d32Sopenharmony_ci case 'L': 305cf200d32Sopenharmony_ci break; 306cf200d32Sopenharmony_ci case 'm': 307cf200d32Sopenharmony_ci JustLooking(0); 308cf200d32Sopenharmony_ci if (BuildMBR(mbrParts, 0) == 1) { 309cf200d32Sopenharmony_ci if (!pretend) { 310cf200d32Sopenharmony_ci if (SaveMBR()) { 311cf200d32Sopenharmony_ci DestroyGPT(); 312cf200d32Sopenharmony_ci } else 313cf200d32Sopenharmony_ci cerr << "Problem saving MBR!\n"; 314cf200d32Sopenharmony_ci } // if 315cf200d32Sopenharmony_ci saveNonGPT = 0; 316cf200d32Sopenharmony_ci pretend = 1; // Not really, but works around problem if -g is used with this... 317cf200d32Sopenharmony_ci saveData = 0; 318cf200d32Sopenharmony_ci } // if 319cf200d32Sopenharmony_ci break; 320cf200d32Sopenharmony_ci case 'n': 321cf200d32Sopenharmony_ci JustLooking(0); 322cf200d32Sopenharmony_ci newPartNum = (int) GetInt(newPartInfo, 1) - 1; 323cf200d32Sopenharmony_ci if (newPartNum < 0) 324cf200d32Sopenharmony_ci newPartNum = FindFirstFreePart(); 325cf200d32Sopenharmony_ci low = FindFirstInLargest(); 326cf200d32Sopenharmony_ci Align(&low); 327cf200d32Sopenharmony_ci high = FindLastInFree(low, alignEnd); 328cf200d32Sopenharmony_ci startSector = IeeeToInt(GetString(newPartInfo, 2), sSize, low, high, sectorAlignment, low); 329cf200d32Sopenharmony_ci endSector = IeeeToInt(GetString(newPartInfo, 3), sSize, startSector, high, sectorAlignment, high); 330cf200d32Sopenharmony_ci if (CreatePartition(newPartNum, startSector, endSector)) { 331cf200d32Sopenharmony_ci saveData = 1; 332cf200d32Sopenharmony_ci } else { 333cf200d32Sopenharmony_ci cerr << "Could not create partition " << newPartNum + 1 << " from " 334cf200d32Sopenharmony_ci << startSector << " to " << endSector << "\n"; 335cf200d32Sopenharmony_ci neverSaveData = 1; 336cf200d32Sopenharmony_ci } // if/else 337cf200d32Sopenharmony_ci free(newPartInfo); 338cf200d32Sopenharmony_ci break; 339cf200d32Sopenharmony_ci case 'N': 340cf200d32Sopenharmony_ci JustLooking(0); 341cf200d32Sopenharmony_ci startSector = FindFirstInLargest(); 342cf200d32Sopenharmony_ci Align(&startSector); 343cf200d32Sopenharmony_ci endSector = FindLastInFree(startSector, alignEnd); 344cf200d32Sopenharmony_ci if (largestPartNum <= 0) { 345cf200d32Sopenharmony_ci largestPartNum = FindFirstFreePart() + 1; 346cf200d32Sopenharmony_ci newPartNum = largestPartNum - 1; 347cf200d32Sopenharmony_ci } 348cf200d32Sopenharmony_ci if (CreatePartition(largestPartNum - 1, startSector, endSector)) { 349cf200d32Sopenharmony_ci saveData = 1; 350cf200d32Sopenharmony_ci } else { 351cf200d32Sopenharmony_ci cerr << "Could not create partition " << largestPartNum << " from " 352cf200d32Sopenharmony_ci << startSector << " to " << endSector << "\n"; 353cf200d32Sopenharmony_ci neverSaveData = 1; 354cf200d32Sopenharmony_ci } // if/else 355cf200d32Sopenharmony_ci break; 356cf200d32Sopenharmony_ci case 'o': 357cf200d32Sopenharmony_ci JustLooking(0); 358cf200d32Sopenharmony_ci ClearGPTData(); 359cf200d32Sopenharmony_ci saveData = 1; 360cf200d32Sopenharmony_ci break; 361cf200d32Sopenharmony_ci case 'O': 362cf200d32Sopenharmony_ci DisplayMBRData(); 363cf200d32Sopenharmony_ci break; 364cf200d32Sopenharmony_ci case 'p': 365cf200d32Sopenharmony_ci DisplayGPTData(); 366cf200d32Sopenharmony_ci break; 367cf200d32Sopenharmony_ci case 'P': 368cf200d32Sopenharmony_ci pretend = 1; 369cf200d32Sopenharmony_ci break; 370cf200d32Sopenharmony_ci case 'r': 371cf200d32Sopenharmony_ci JustLooking(0); 372cf200d32Sopenharmony_ci uint64_t p1, p2; 373cf200d32Sopenharmony_ci p1 = GetInt(twoParts, 1) - 1; 374cf200d32Sopenharmony_ci p2 = GetInt(twoParts, 2) - 1; 375cf200d32Sopenharmony_ci if (SwapPartitions((uint32_t) p1, (uint32_t) p2) == 0) { 376cf200d32Sopenharmony_ci neverSaveData = 1; 377cf200d32Sopenharmony_ci cerr << "Cannot swap partitions " << p1 + 1 << " and " << p2 + 1 << "\n"; 378cf200d32Sopenharmony_ci } else saveData = 1; 379cf200d32Sopenharmony_ci break; 380cf200d32Sopenharmony_ci case 'R': 381cf200d32Sopenharmony_ci secondDevice = *this; 382cf200d32Sopenharmony_ci secondDevice.SetDisk(outDevice); 383cf200d32Sopenharmony_ci secondDevice.JustLooking(0); 384cf200d32Sopenharmony_ci if (!secondDevice.SaveGPTData(1)) 385cf200d32Sopenharmony_ci retval = 8; 386cf200d32Sopenharmony_ci break; 387cf200d32Sopenharmony_ci case 's': 388cf200d32Sopenharmony_ci JustLooking(0); 389cf200d32Sopenharmony_ci SortGPT(); 390cf200d32Sopenharmony_ci saveData = 1; 391cf200d32Sopenharmony_ci break; 392cf200d32Sopenharmony_ci case 'S': 393cf200d32Sopenharmony_ci JustLooking(0); 394cf200d32Sopenharmony_ci if (SetGPTSize(tableSize) == 0) 395cf200d32Sopenharmony_ci neverSaveData = 1; 396cf200d32Sopenharmony_ci else 397cf200d32Sopenharmony_ci saveData = 1; 398cf200d32Sopenharmony_ci break; 399cf200d32Sopenharmony_ci case 't': 400cf200d32Sopenharmony_ci JustLooking(0); 401cf200d32Sopenharmony_ci partNum = (int) GetInt(typeCode, 1) - 1; 402cf200d32Sopenharmony_ci if (partNum < 0) 403cf200d32Sopenharmony_ci partNum = newPartNum; 404cf200d32Sopenharmony_ci if ((partNum >= 0) && (partNum < (int) GetNumParts())) { 405cf200d32Sopenharmony_ci typeHelper = GetString(typeCode, 2); 406cf200d32Sopenharmony_ci if ((typeHelper != PartType::unusedPartType) && 407cf200d32Sopenharmony_ci (ChangePartType(partNum, typeHelper))) { 408cf200d32Sopenharmony_ci saveData = 1; 409cf200d32Sopenharmony_ci } else { 410cf200d32Sopenharmony_ci cerr << "Could not change partition " << partNum + 1 411cf200d32Sopenharmony_ci << "'s type code to " << GetString(typeCode, 2) << "!\n"; 412cf200d32Sopenharmony_ci neverSaveData = 1; 413cf200d32Sopenharmony_ci } // if/else 414cf200d32Sopenharmony_ci free(typeCode); 415cf200d32Sopenharmony_ci } 416cf200d32Sopenharmony_ci break; 417cf200d32Sopenharmony_ci case 'T': 418cf200d32Sopenharmony_ci JustLooking(0); 419cf200d32Sopenharmony_ci XFormDisklabel(bsdPartNum - 1); 420cf200d32Sopenharmony_ci saveData = 1; 421cf200d32Sopenharmony_ci break; 422cf200d32Sopenharmony_ci case 'u': 423cf200d32Sopenharmony_ci JustLooking(0); 424cf200d32Sopenharmony_ci saveData = 1; 425cf200d32Sopenharmony_ci partNum = (int) GetInt(partGUID, 1) - 1; 426cf200d32Sopenharmony_ci if (partNum < 0) 427cf200d32Sopenharmony_ci partNum = newPartNum; 428cf200d32Sopenharmony_ci if ((partNum >= 0) && (partNum < (int) GetNumParts())) { 429cf200d32Sopenharmony_ci SetPartitionGUID(partNum, GetString(partGUID, 2).c_str()); 430cf200d32Sopenharmony_ci } 431cf200d32Sopenharmony_ci break; 432cf200d32Sopenharmony_ci case 'U': 433cf200d32Sopenharmony_ci JustLooking(0); 434cf200d32Sopenharmony_ci saveData = 1; 435cf200d32Sopenharmony_ci SetDiskGUID(diskGUID); 436cf200d32Sopenharmony_ci break; 437cf200d32Sopenharmony_ci case 'v': 438cf200d32Sopenharmony_ci Verify(); 439cf200d32Sopenharmony_ci break; 440cf200d32Sopenharmony_ci case 'z': 441cf200d32Sopenharmony_ci if (!pretend) { 442cf200d32Sopenharmony_ci DestroyGPT(); 443cf200d32Sopenharmony_ci } // if 444cf200d32Sopenharmony_ci saveNonGPT = 1; 445cf200d32Sopenharmony_ci saveData = 0; 446cf200d32Sopenharmony_ci break; 447cf200d32Sopenharmony_ci case 'Z': 448cf200d32Sopenharmony_ci if (!pretend) { 449cf200d32Sopenharmony_ci DestroyGPT(); 450cf200d32Sopenharmony_ci DestroyMBR(); 451cf200d32Sopenharmony_ci } // if 452cf200d32Sopenharmony_ci saveNonGPT = 1; 453cf200d32Sopenharmony_ci saveData = 0; 454cf200d32Sopenharmony_ci break; 455cf200d32Sopenharmony_ci default: 456cf200d32Sopenharmony_ci cerr << "Unknown option (-" << opt << ")!\n"; 457cf200d32Sopenharmony_ci break; 458cf200d32Sopenharmony_ci } // switch 459cf200d32Sopenharmony_ci } // while 460cf200d32Sopenharmony_ci } else { // if loaded OK 461cf200d32Sopenharmony_ci poptResetContext(poptCon); 462cf200d32Sopenharmony_ci // Do a few types of operations even if there are problems.... 463cf200d32Sopenharmony_ci while ((opt = poptGetNextOpt(poptCon)) > 0) { 464cf200d32Sopenharmony_ci switch (opt) { 465cf200d32Sopenharmony_ci case 'l': 466cf200d32Sopenharmony_ci LoadBackupFile(backupFile, saveData, neverSaveData); 467cf200d32Sopenharmony_ci cout << "Information: Loading backup partition table; will override earlier problems!\n"; 468cf200d32Sopenharmony_ci free(backupFile); 469cf200d32Sopenharmony_ci retval = 0; 470cf200d32Sopenharmony_ci break; 471cf200d32Sopenharmony_ci case 'o': 472cf200d32Sopenharmony_ci JustLooking(0); 473cf200d32Sopenharmony_ci ClearGPTData(); 474cf200d32Sopenharmony_ci saveData = 1; 475cf200d32Sopenharmony_ci cout << "Information: Creating fresh partition table; will override earlier problems!\n"; 476cf200d32Sopenharmony_ci retval = 0; 477cf200d32Sopenharmony_ci break; 478cf200d32Sopenharmony_ci case 'v': 479cf200d32Sopenharmony_ci cout << "Verification may miss some problems or report too many!\n"; 480cf200d32Sopenharmony_ci Verify(); 481cf200d32Sopenharmony_ci break; 482cf200d32Sopenharmony_ci case 'z': 483cf200d32Sopenharmony_ci if (!pretend) { 484cf200d32Sopenharmony_ci DestroyGPT(); 485cf200d32Sopenharmony_ci } // if 486cf200d32Sopenharmony_ci saveNonGPT = 1; 487cf200d32Sopenharmony_ci saveData = 0; 488cf200d32Sopenharmony_ci break; 489cf200d32Sopenharmony_ci case 'Z': 490cf200d32Sopenharmony_ci if (!pretend) { 491cf200d32Sopenharmony_ci DestroyGPT(); 492cf200d32Sopenharmony_ci DestroyMBR(); 493cf200d32Sopenharmony_ci } // if 494cf200d32Sopenharmony_ci saveNonGPT = 1; 495cf200d32Sopenharmony_ci saveData = 0; 496cf200d32Sopenharmony_ci break; 497cf200d32Sopenharmony_ci } // switch 498cf200d32Sopenharmony_ci } // while 499cf200d32Sopenharmony_ci retval = 2; 500cf200d32Sopenharmony_ci } // if/else loaded OK 501cf200d32Sopenharmony_ci if ((saveData) && (!neverSaveData) && (saveNonGPT) && (!pretend)) { 502cf200d32Sopenharmony_ci if (!SaveGPTData(1)) 503cf200d32Sopenharmony_ci retval = 4; 504cf200d32Sopenharmony_ci } 505cf200d32Sopenharmony_ci if (saveData && (!saveNonGPT)) { 506cf200d32Sopenharmony_ci cout << "Non-GPT disk; not saving changes. Use -g to override.\n"; 507cf200d32Sopenharmony_ci retval = 3; 508cf200d32Sopenharmony_ci } // if 509cf200d32Sopenharmony_ci if (neverSaveData) { 510cf200d32Sopenharmony_ci cerr << "Error encountered; not saving changes.\n"; 511cf200d32Sopenharmony_ci retval = 4; 512cf200d32Sopenharmony_ci } // if 513cf200d32Sopenharmony_ci free(device); 514cf200d32Sopenharmony_ci } // if (device != NULL) 515cf200d32Sopenharmony_ci poptFreeContext(poptCon); 516cf200d32Sopenharmony_ci return retval; 517cf200d32Sopenharmony_ci} // GPTDataCL::DoOptions() 518cf200d32Sopenharmony_ci 519cf200d32Sopenharmony_ci// Create a hybrid or regular MBR from GPT data structures 520cf200d32Sopenharmony_ciint GPTDataCL::BuildMBR(char* argument, int isHybrid) { 521cf200d32Sopenharmony_ci int numParts, allOK = 1, i, origPartNum; 522cf200d32Sopenharmony_ci int eeLast = 0, mbrNum = 0; 523cf200d32Sopenharmony_ci MBRPart newPart; 524cf200d32Sopenharmony_ci BasicMBRData newMBR; 525cf200d32Sopenharmony_ci 526cf200d32Sopenharmony_ci if (argument != NULL) { 527cf200d32Sopenharmony_ci numParts = CountColons(argument) + 1; 528cf200d32Sopenharmony_ci if (isHybrid) { 529cf200d32Sopenharmony_ci eeLast = GetString(argument, numParts) == "EE"; 530cf200d32Sopenharmony_ci if (eeLast) { 531cf200d32Sopenharmony_ci numParts--; 532cf200d32Sopenharmony_ci } 533cf200d32Sopenharmony_ci } 534cf200d32Sopenharmony_ci 535cf200d32Sopenharmony_ci if (numParts <= (4 - isHybrid)) { 536cf200d32Sopenharmony_ci newMBR.SetDisk(GetDisk()); 537cf200d32Sopenharmony_ci for (i = 0; i < numParts; i++) { 538cf200d32Sopenharmony_ci origPartNum = GetInt(argument, i + 1) - 1; 539cf200d32Sopenharmony_ci if (IsUsedPartNum(origPartNum) && (partitions[origPartNum].IsSizedForMBR() == MBR_SIZED_GOOD)) { 540cf200d32Sopenharmony_ci mbrNum = i + (isHybrid && ! eeLast); 541cf200d32Sopenharmony_ci newPart.SetInclusion(PRIMARY); 542cf200d32Sopenharmony_ci newPart.SetLocation(operator[](origPartNum).GetFirstLBA(), 543cf200d32Sopenharmony_ci operator[](origPartNum).GetLengthLBA()); 544cf200d32Sopenharmony_ci newPart.SetStatus(0); 545cf200d32Sopenharmony_ci newPart.SetType((uint8_t)(operator[](origPartNum).GetHexType() / 0x0100)); 546cf200d32Sopenharmony_ci newMBR.AddPart(mbrNum, newPart); 547cf200d32Sopenharmony_ci } else { 548cf200d32Sopenharmony_ci cerr << "Original partition " << origPartNum + 1 << " does not exist or is too big! Aborting operation!\n"; 549cf200d32Sopenharmony_ci allOK = 0; 550cf200d32Sopenharmony_ci } // if/else 551cf200d32Sopenharmony_ci } // for 552cf200d32Sopenharmony_ci if (isHybrid) { 553cf200d32Sopenharmony_ci if (eeLast) { 554cf200d32Sopenharmony_ci mbrNum = i; 555cf200d32Sopenharmony_ci } else { 556cf200d32Sopenharmony_ci mbrNum = 0; 557cf200d32Sopenharmony_ci } 558cf200d32Sopenharmony_ci newPart.SetInclusion(PRIMARY); 559cf200d32Sopenharmony_ci newPart.SetLocation(1, newMBR.FindLastInFree(1)); 560cf200d32Sopenharmony_ci newPart.SetStatus(0); 561cf200d32Sopenharmony_ci newPart.SetType(0xEE); 562cf200d32Sopenharmony_ci newMBR.AddPart(mbrNum, newPart); 563cf200d32Sopenharmony_ci } // if 564cf200d32Sopenharmony_ci if (allOK) 565cf200d32Sopenharmony_ci SetProtectiveMBR(newMBR); 566cf200d32Sopenharmony_ci } else allOK = 0; 567cf200d32Sopenharmony_ci } else allOK = 0; 568cf200d32Sopenharmony_ci if (!allOK) 569cf200d32Sopenharmony_ci cerr << "Problem creating MBR!\n"; 570cf200d32Sopenharmony_ci return allOK; 571cf200d32Sopenharmony_ci} // GPTDataCL::BuildMBR() 572cf200d32Sopenharmony_ci 573cf200d32Sopenharmony_ci// Returns the number of colons in argument string, ignoring the 574cf200d32Sopenharmony_ci// first character (thus, a leading colon is ignored, as GetString() 575cf200d32Sopenharmony_ci// does). 576cf200d32Sopenharmony_ciint CountColons(char* argument) { 577cf200d32Sopenharmony_ci int num = 0; 578cf200d32Sopenharmony_ci 579cf200d32Sopenharmony_ci while ((argument[0] != '\0') && (argument = strchr(&argument[1], ':'))) 580cf200d32Sopenharmony_ci num++; 581cf200d32Sopenharmony_ci 582cf200d32Sopenharmony_ci return num; 583cf200d32Sopenharmony_ci} // GPTDataCL::CountColons() 584cf200d32Sopenharmony_ci 585cf200d32Sopenharmony_ci// Extract integer data from argument string, which should be colon-delimited 586cf200d32Sopenharmony_ciuint64_t GetInt(const string & argument, int itemNum) { 587cf200d32Sopenharmony_ci uint64_t retval; 588cf200d32Sopenharmony_ci 589cf200d32Sopenharmony_ci istringstream inString(GetString(argument, itemNum)); 590cf200d32Sopenharmony_ci inString >> retval; 591cf200d32Sopenharmony_ci return retval; 592cf200d32Sopenharmony_ci} // GPTDataCL::GetInt() 593cf200d32Sopenharmony_ci 594cf200d32Sopenharmony_ci// Extract string data from argument string, which should be colon-delimited 595cf200d32Sopenharmony_ci// If string begins with a colon, that colon is skipped in the counting. If an 596cf200d32Sopenharmony_ci// invalid itemNum is specified, returns an empty string. 597cf200d32Sopenharmony_cistring GetString(string argument, int itemNum) { 598cf200d32Sopenharmony_ci size_t startPos = 0, endPos = 0; 599cf200d32Sopenharmony_ci string retVal = ""; 600cf200d32Sopenharmony_ci int foundLast = 0; 601cf200d32Sopenharmony_ci int numFound = 0; 602cf200d32Sopenharmony_ci 603cf200d32Sopenharmony_ci if (argument[0] == ':') 604cf200d32Sopenharmony_ci argument.erase(0, 1); 605cf200d32Sopenharmony_ci while ((numFound < itemNum) && (!foundLast)) { 606cf200d32Sopenharmony_ci endPos = argument.find(':', startPos); 607cf200d32Sopenharmony_ci numFound++; 608cf200d32Sopenharmony_ci if (endPos == string::npos) { 609cf200d32Sopenharmony_ci foundLast = 1; 610cf200d32Sopenharmony_ci endPos = argument.length(); 611cf200d32Sopenharmony_ci } else if (numFound < itemNum) { 612cf200d32Sopenharmony_ci startPos = endPos + 1; 613cf200d32Sopenharmony_ci } // if/elseif 614cf200d32Sopenharmony_ci } // while 615cf200d32Sopenharmony_ci if ((numFound == itemNum) && (numFound > 0)) 616cf200d32Sopenharmony_ci retVal = argument.substr(startPos, endPos - startPos); 617cf200d32Sopenharmony_ci 618cf200d32Sopenharmony_ci return retVal; 619cf200d32Sopenharmony_ci} // GetString() 620