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