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