blob: 027e37f11d996304e054e00a4641996081473564 [file] [log] [blame]
// sgdisk.cc
// Command-line-based version of gdisk. This program is named after sfdisk,
// and it can serve a similar role (easily scripted, etc.), but it's used
// strictly via command-line arguments, and it doesn't bear much resemblance
// to sfdisk in actual use.
//
// by Rod Smith, project began February 2009; sgdisk begun January 2010.
/* This program is copyright (c) 2009, 2010 by Roderick W. Smith. It is distributed
under the terms of the GNU GPL version 2, as detailed in the COPYING file. */
#include <stdio.h>
#include <string>
#include <popt.h>
#include <errno.h>
#include "mbr.h"
#include "gpt.h"
#include "support.h"
#include "parttypes.h"
using namespace std;
#define MAX_OPTIONS 50
uint64_t GetInt(char* Info, int itemNum);
string GetString(char* Info, int itemNum);
int main(int argc, char *argv[]) {
GPTData theGPT;
int opt, i, numOptions = 0, saveData = 0, neverSaveData = 0;
int partNum = 0, deletePartNum = 0, infoPartNum = 0, bsdPartNum = 0, saveNonGPT = 1;
int alignment = 8, retval = 0, pretend = 0;
uint16_t hexCode;
uint32_t tableSize = 128;
uint64_t startSector, endSector;
char* device = NULL;
char *argument = NULL, *newPartInfo = NULL, *typeCode = NULL, *partName;
char *backupFile = NULL;
PartTypes typeHelper;
poptContext poptCon;
struct poptOption theOptions[] =
{
{"set-alignment", 'a', POPT_ARG_INT, &alignment, 'a', "set sector alignment", "value"},
{"backup", 'b', POPT_ARG_STRING, &backupFile, 'b', "backup GPT to file", "file"},
{"change-name", 'c', POPT_ARG_STRING, &partName, 'c', "change partition's name", "partnum:name"},
{"delete", 'd', POPT_ARG_INT, &deletePartNum, 'd', "delete a partition", "partnum"},
{"move-second-header", 'e', POPT_ARG_NONE, NULL, 'e', "move second header to end of disk", ""},
{"end-of-largest", 'E', POPT_ARG_NONE, NULL, 'E', "show end of largest free block", ""},
{"first-in-largest", 'f', POPT_ARG_NONE, NULL, 'f', "show start of the largest free block", ""},
{"mbrtogpt", 'g', POPT_ARG_NONE, NULL, 'g', "convert MBR to GPT", ""},
{"info", 'i', POPT_ARG_INT, &infoPartNum, 'i', "show detailed information on partition", "partnum"},
{"load-backup", 'l', POPT_ARG_STRING, &backupFile, 'l', "load GPT backup from file", "file"},
{"list-types", 'L', POPT_ARG_NONE, NULL, 'L', "list known partition types", ""},
{"new", 'n', POPT_ARG_STRING, &newPartInfo, 'n', "create new partition", "partnum:start:end"},
{"clear", 'o', POPT_ARG_NONE, NULL, 'o', "clear partition table", ""},
{"print", 'p', POPT_ARG_NONE, NULL, 'p', "print partition table", ""},
{"pretend", 'P', POPT_ARG_NONE, NULL, 'P', "make changes in memory, but don't write them", ""},
{"sort", 's', POPT_ARG_NONE, NULL, 's', "sort partition table entries", ""},
{"resize-table", 'S', POPT_ARG_INT, &tableSize, 'S', "resize partition table", "numparts"},
{"typecode", 't', POPT_ARG_STRING, &typeCode, 't', "change partition type code", "partnum:hexcode"},
{"transform-bsd", 'T', POPT_ARG_INT, &bsdPartNum, 'T', "transform BSD disklabel partition to GPT", "partnum"},
{"verify", 'v', POPT_ARG_NONE, NULL, 'v', "check partition table integrity", ""},
{"version", 'V', POPT_ARG_NONE, NULL, 'V', "display version information", ""},
{"zap", 'z', POPT_ARG_NONE, NULL, 'z', "zap (destroy) GPT data structures", ""},
POPT_AUTOHELP { NULL, 0, 0, NULL, 0 }
};
// Create popt context...
poptCon = poptGetContext(NULL, argc, (const char**) argv, theOptions, 0);
poptSetOtherOptionHelp(poptCon, " [OPTION...] <device>");
if (argc < 2) {
poptPrintUsage(poptCon, stderr, 0);
exit(1);
}
// Do one loop through the options to find the device filename and deal
// with options that don't require a device filename....
while ((opt = poptGetNextOpt(poptCon)) > 0) {
switch (opt) {
case 'L':
typeHelper.ShowTypes();
break;
case 'P':
pretend = 1;
break;
case 'V':
printf("GPT fdisk (sgdisk) version 0.5.4-pre1\n\n");
break;
default:
break;
} // switch
numOptions++;
} // while
// Assume first non-option argument is the device filename....
device = (char*) poptGetArg(poptCon);
poptResetContext(poptCon);
if (device != NULL) {
theGPT.JustLooking(); // reset as necessary
theGPT.BeQuiet(); // Tell called functions to be less verbose & interactive
if (theGPT.LoadPartitions(device)) {
if ((theGPT.WhichWasUsed() == use_mbr) || (theGPT.WhichWasUsed() == use_bsd))
saveNonGPT = 0; // flag so we don't overwrite unless directed to do so
while ((opt = poptGetNextOpt(poptCon)) > 0) {
switch (opt) {
case 'a':
theGPT.SetAlignment(alignment);
break;
case 'b':
theGPT.SaveGPTBackup(backupFile);
free(backupFile);
break;
case 'c':
theGPT.JustLooking(0);
partNum = (int) GetInt(partName, 1) - 1;
if (theGPT.SetName(partNum, (char*) GetString(partName, 2).c_str())) {
saveData = 1;
} else {
fprintf(stderr, "Unable set set partition %d's name to '%s'!\n",
partNum + 1, GetString(partName, 2).c_str());
neverSaveData = 1;
} // if/else
free(partName);
break;
case 'd':
theGPT.JustLooking(0);
if (theGPT.DeletePartition(deletePartNum - 1) == 0) {
fprintf(stderr, "Error %d deleting partition!\n", errno);
neverSaveData = 1;
} else saveData = 1;
break;
case 'e':
theGPT.JustLooking(0);
theGPT.MoveSecondHeaderToEnd();
saveData = 1;
break;
case 'E':
printf("%llu\n", (unsigned long long) theGPT.FindLastAvailable(theGPT.FindFirstInLargest()));
break;
case 'f':
printf("%llu\n", (unsigned long long) theGPT.FindFirstInLargest());
break;
case 'g':
theGPT.JustLooking(0);
saveData = 1;
saveNonGPT = 1;
break;
case 'i':
theGPT.ShowPartDetails(infoPartNum - 1);
break;
case 'l':
if (theGPT.LoadGPTBackup(backupFile) == 1)
saveData = 1;
else {
saveData = 0;
neverSaveData = 1;
fprintf(stderr, "Error loading backup file!\n");
} // else
free(backupFile);
break;
case 'L':
break;
case 'n':
theGPT.JustLooking(0);
partNum = (int) GetInt(newPartInfo, 1) - 1;
startSector = GetInt(newPartInfo, 2);
endSector = GetInt(newPartInfo, 3);
if (theGPT.CreatePartition(partNum, startSector, endSector)) {
saveData = 1;
} else {
fprintf(stderr, "Could not create partition %d from %llu to %llu!\n",
partNum, startSector, endSector);
neverSaveData = 1;
} // if/else
free(newPartInfo);
break;
case 'o':
theGPT.JustLooking(0);
theGPT.ClearGPTData();
saveData = 1;
break;
case 'p':
theGPT.DisplayGPTData();
break;
case 'P':
pretend = 1;
break;
case 's':
theGPT.JustLooking(0);
theGPT.SortGPT();
saveData = 1;
break;
case 'S':
theGPT.JustLooking(0);
if (theGPT.SetGPTSize(tableSize) == 0)
neverSaveData = 1;
else
saveData = 1;
break;
case 't':
theGPT.JustLooking(0);
partNum = (int) GetInt(typeCode, 1) - 1;
sscanf(GetString(typeCode, 2).c_str(), "%x", &hexCode);
if (theGPT.ChangePartType(partNum, hexCode)) {
saveData = 1;
} else {
fprintf(stderr, "Could not change partition %d's type code to %x!\n",
partNum + 1, hexCode);
neverSaveData = 1;
} // if/else
free(typeCode);
break;
case 'T':
theGPT.JustLooking(0);
theGPT.XFormDisklabel(bsdPartNum);
saveData = 1;
break;
case 'v':
theGPT.Verify();
break;
case 'z':
if (!pretend)
theGPT.DestroyGPT(-1);
saveNonGPT = 0;
break;
default:
printf("Unknown option (-%c)!\n", opt);
break;
} // switch
} // while
if ((saveData) && (!neverSaveData) && (saveNonGPT) && (!pretend))
theGPT.SaveGPTData(1);
if (saveData && (!saveNonGPT)) {
printf("Non-GPT disk; not saving changes. Use -g to override.\n");
retval = 3;
} // if
if (neverSaveData) {
printf("Error encountered; not saving changes.\n");
retval = 4;
} // if
} else { // if loaded OK
retval = 2;
} // if/else loaded OK
} // if (device != NULL)
poptFreeContext(poptCon);
return retval;
} // main
// Extract integer data from argument string, which should be colon-delimited
uint64_t GetInt(char* argument, int itemNum) {
int startPos = -1, endPos = -1;
uint64_t retval = 0;
string Info;
Info = argument;
while (itemNum-- > 0) {
startPos = endPos + 1;
endPos = Info.find(':', startPos);
}
if (endPos == string::npos)
endPos = Info.length();
endPos--;
sscanf(Info.substr(startPos, endPos - startPos + 1).c_str(), "%llu", &retval);
/* printf("In GetInt(), startPos = %d, endPos = %d, retval = %llu\n", startPos,
endPos, (unsigned long long) retval); */
return retval;
} // GetInt()
// Extract string data from argument string, which should be colon-delimited
string GetString(char* argument, int itemNum) {
int startPos = -1, endPos = -1;
string Info;
Info = argument;
while (itemNum-- > 0) {
startPos = endPos + 1;
endPos = Info.find(':', startPos);
}
if (endPos == string::npos)
endPos = Info.length();
endPos--;
return Info.substr(startPos, endPos - startPos + 1);
} // GetString()