1*57696d54SAkhilesh Sanikop /* gpt.h -- GPT and data structure definitions, types, and 2*57696d54SAkhilesh Sanikop functions */ 3*57696d54SAkhilesh Sanikop 4*57696d54SAkhilesh Sanikop /* This program is copyright (c) 2009-2022 by Roderick W. Smith. It is distributed 5*57696d54SAkhilesh Sanikop under the terms of the GNU GPL version 2, as detailed in the COPYING file. */ 6*57696d54SAkhilesh Sanikop 7*57696d54SAkhilesh Sanikop #ifndef __GPTSTRUCTS 8*57696d54SAkhilesh Sanikop #define __GPTSTRUCTS 9*57696d54SAkhilesh Sanikop 10*57696d54SAkhilesh Sanikop #include <stdint.h> 11*57696d54SAkhilesh Sanikop #include <sys/types.h> 12*57696d54SAkhilesh Sanikop #include "gptpart.h" 13*57696d54SAkhilesh Sanikop #include "support.h" 14*57696d54SAkhilesh Sanikop #include "mbr.h" 15*57696d54SAkhilesh Sanikop #include "bsd.h" 16*57696d54SAkhilesh Sanikop #include "gptpart.h" 17*57696d54SAkhilesh Sanikop 18*57696d54SAkhilesh Sanikop // Default values for sector alignment 19*57696d54SAkhilesh Sanikop #define DEFAULT_ALIGNMENT 2048 20*57696d54SAkhilesh Sanikop #define MAX_ALIGNMENT 65536 21*57696d54SAkhilesh Sanikop #define MIN_AF_ALIGNMENT 8 22*57696d54SAkhilesh Sanikop 23*57696d54SAkhilesh Sanikop // Below constant corresponds to a ~279GiB (300GB) disk, since the 24*57696d54SAkhilesh Sanikop // smallest Advanced Format drive I know of is 320GB in size 25*57696d54SAkhilesh Sanikop #define SMALLEST_ADVANCED_FORMAT UINT64_C(585937500) 26*57696d54SAkhilesh Sanikop 27*57696d54SAkhilesh Sanikop /**************************************** 28*57696d54SAkhilesh Sanikop * * 29*57696d54SAkhilesh Sanikop * GPTData class and related structures * 30*57696d54SAkhilesh Sanikop * * 31*57696d54SAkhilesh Sanikop ****************************************/ 32*57696d54SAkhilesh Sanikop 33*57696d54SAkhilesh Sanikop // Validity state of GPT data 34*57696d54SAkhilesh Sanikop enum GPTValidity {gpt_valid, gpt_corrupt, gpt_invalid}; 35*57696d54SAkhilesh Sanikop 36*57696d54SAkhilesh Sanikop // Which set of partition data to use 37*57696d54SAkhilesh Sanikop enum WhichToUse {use_gpt, use_mbr, use_bsd, use_new, use_abort}; 38*57696d54SAkhilesh Sanikop 39*57696d54SAkhilesh Sanikop // Header (first 512 bytes) of GPT table 40*57696d54SAkhilesh Sanikop #pragma pack(1) 41*57696d54SAkhilesh Sanikop struct GPTHeader { 42*57696d54SAkhilesh Sanikop uint64_t signature; 43*57696d54SAkhilesh Sanikop uint32_t revision; 44*57696d54SAkhilesh Sanikop uint32_t headerSize; 45*57696d54SAkhilesh Sanikop uint32_t headerCRC; 46*57696d54SAkhilesh Sanikop uint32_t reserved; 47*57696d54SAkhilesh Sanikop uint64_t currentLBA; 48*57696d54SAkhilesh Sanikop uint64_t backupLBA; 49*57696d54SAkhilesh Sanikop uint64_t firstUsableLBA; 50*57696d54SAkhilesh Sanikop uint64_t lastUsableLBA; 51*57696d54SAkhilesh Sanikop GUIDData diskGUID; 52*57696d54SAkhilesh Sanikop uint64_t partitionEntriesLBA; 53*57696d54SAkhilesh Sanikop uint32_t numParts; 54*57696d54SAkhilesh Sanikop uint32_t sizeOfPartitionEntries; 55*57696d54SAkhilesh Sanikop uint32_t partitionEntriesCRC; 56*57696d54SAkhilesh Sanikop unsigned char reserved2[GPT_RESERVED]; 57*57696d54SAkhilesh Sanikop }; // struct GPTHeader 58*57696d54SAkhilesh Sanikop #pragma pack () 59*57696d54SAkhilesh Sanikop 60*57696d54SAkhilesh Sanikop // Data in GPT format 61*57696d54SAkhilesh Sanikop class GPTData { 62*57696d54SAkhilesh Sanikop protected: 63*57696d54SAkhilesh Sanikop struct GPTHeader mainHeader; 64*57696d54SAkhilesh Sanikop GPTPart *partitions; 65*57696d54SAkhilesh Sanikop uint32_t numParts; // # of partitions the table can hold 66*57696d54SAkhilesh Sanikop struct GPTHeader secondHeader; 67*57696d54SAkhilesh Sanikop MBRData protectiveMBR; 68*57696d54SAkhilesh Sanikop std::string device; // device filename 69*57696d54SAkhilesh Sanikop DiskIO myDisk; 70*57696d54SAkhilesh Sanikop uint32_t blockSize; // device logical block size 71*57696d54SAkhilesh Sanikop uint32_t physBlockSize; // device physical block size (or 0 if it can't be determined) 72*57696d54SAkhilesh Sanikop uint64_t diskSize; // size of device, in logical blocks 73*57696d54SAkhilesh Sanikop GPTValidity state; // is GPT valid? 74*57696d54SAkhilesh Sanikop int justLooking; // Set to 1 if program launched with "-l" or if read-only 75*57696d54SAkhilesh Sanikop int mainCrcOk; 76*57696d54SAkhilesh Sanikop int secondCrcOk; 77*57696d54SAkhilesh Sanikop int mainPartsCrcOk; 78*57696d54SAkhilesh Sanikop int secondPartsCrcOk; 79*57696d54SAkhilesh Sanikop int apmFound; // set to 1 if APM detected 80*57696d54SAkhilesh Sanikop int bsdFound; // set to 1 if BSD disklabel detected in MBR 81*57696d54SAkhilesh Sanikop uint32_t sectorAlignment; // Start partitions at multiples of sectorAlignment 82*57696d54SAkhilesh Sanikop int beQuiet; 83*57696d54SAkhilesh Sanikop WhichToUse whichWasUsed; 84*57696d54SAkhilesh Sanikop 85*57696d54SAkhilesh Sanikop int LoadHeader(struct GPTHeader *header, DiskIO & disk, uint64_t sector, int *crcOk); 86*57696d54SAkhilesh Sanikop int LoadPartitionTable(const struct GPTHeader & header, DiskIO & disk, uint64_t sector = 0); 87*57696d54SAkhilesh Sanikop int CheckTable(struct GPTHeader *header); 88*57696d54SAkhilesh Sanikop int SaveHeader(struct GPTHeader *header, DiskIO & disk, uint64_t sector); 89*57696d54SAkhilesh Sanikop int SavePartitionTable(DiskIO & disk, uint64_t sector); 90*57696d54SAkhilesh Sanikop public: 91*57696d54SAkhilesh Sanikop // Basic necessary functions.... 92*57696d54SAkhilesh Sanikop GPTData(void); 93*57696d54SAkhilesh Sanikop GPTData(const GPTData &); 94*57696d54SAkhilesh Sanikop GPTData(std::string deviceFilename); 95*57696d54SAkhilesh Sanikop virtual ~GPTData(void); 96*57696d54SAkhilesh Sanikop GPTData & operator=(const GPTData & orig); 97*57696d54SAkhilesh Sanikop 98*57696d54SAkhilesh Sanikop // Verify (or update) data integrity 99*57696d54SAkhilesh Sanikop int Verify(void); 100*57696d54SAkhilesh Sanikop int CheckGPTSize(void); 101*57696d54SAkhilesh Sanikop int CheckHeaderValidity(void); 102*57696d54SAkhilesh Sanikop int CheckHeaderCRC(struct GPTHeader* header, int warn = 0); 103*57696d54SAkhilesh Sanikop void RecomputeCRCs(void); 104*57696d54SAkhilesh Sanikop void RebuildMainHeader(void); 105*57696d54SAkhilesh Sanikop void RebuildSecondHeader(void); VerifyMBR(void)106*57696d54SAkhilesh Sanikop int VerifyMBR(void) {return protectiveMBR.FindOverlaps();} 107*57696d54SAkhilesh Sanikop int FindHybridMismatches(void); 108*57696d54SAkhilesh Sanikop int FindOverlaps(void); 109*57696d54SAkhilesh Sanikop int FindInsanePartitions(void); 110*57696d54SAkhilesh Sanikop 111*57696d54SAkhilesh Sanikop // Load or save data from/to disk 112*57696d54SAkhilesh Sanikop int SetDisk(const std::string & deviceFilename); 113*57696d54SAkhilesh Sanikop int SetDisk(const DiskIO & disk); GetDisk(void)114*57696d54SAkhilesh Sanikop DiskIO* GetDisk(void) {return &myDisk;} LoadMBR(const std::string & f)115*57696d54SAkhilesh Sanikop int LoadMBR(const std::string & f) {return protectiveMBR.ReadMBRData(f);} WriteProtectiveMBR(void)116*57696d54SAkhilesh Sanikop int WriteProtectiveMBR(void) {return protectiveMBR.WriteMBRData(&myDisk);} 117*57696d54SAkhilesh Sanikop void PartitionScan(void); 118*57696d54SAkhilesh Sanikop int LoadPartitions(const std::string & deviceFilename); 119*57696d54SAkhilesh Sanikop int ForceLoadGPTData(void); 120*57696d54SAkhilesh Sanikop int LoadMainTable(void); 121*57696d54SAkhilesh Sanikop int LoadSecondTableAsMain(void); 122*57696d54SAkhilesh Sanikop int SaveGPTData(int quiet = 0); 123*57696d54SAkhilesh Sanikop int SaveGPTBackup(const std::string & filename); 124*57696d54SAkhilesh Sanikop int LoadGPTBackup(const std::string & filename); 125*57696d54SAkhilesh Sanikop int SaveMBR(void); 126*57696d54SAkhilesh Sanikop int DestroyGPT(void); 127*57696d54SAkhilesh Sanikop int DestroyMBR(void); 128*57696d54SAkhilesh Sanikop 129*57696d54SAkhilesh Sanikop // Display data.... 130*57696d54SAkhilesh Sanikop void ShowAPMState(void); 131*57696d54SAkhilesh Sanikop void ShowGPTState(void); 132*57696d54SAkhilesh Sanikop void DisplayGPTData(void); DisplayMBRData(void)133*57696d54SAkhilesh Sanikop void DisplayMBRData(void) {protectiveMBR.DisplayMBRData();} 134*57696d54SAkhilesh Sanikop void ShowPartDetails(uint32_t partNum); 135*57696d54SAkhilesh Sanikop 136*57696d54SAkhilesh Sanikop // Convert between GPT and other formats 137*57696d54SAkhilesh Sanikop virtual WhichToUse UseWhichPartitions(void); 138*57696d54SAkhilesh Sanikop void XFormPartitions(void); 139*57696d54SAkhilesh Sanikop int XFormDisklabel(uint32_t partNum); 140*57696d54SAkhilesh Sanikop int XFormDisklabel(BSDData* disklabel); 141*57696d54SAkhilesh Sanikop int OnePartToMBR(uint32_t gptPart, int mbrPart); // add one partition to MBR. Returns 1 if successful 142*57696d54SAkhilesh Sanikop 143*57696d54SAkhilesh Sanikop // Adjust GPT structures WITHOUT user interaction... 144*57696d54SAkhilesh Sanikop int SetGPTSize(uint32_t numEntries, int fillGPTSectors = 1); 145*57696d54SAkhilesh Sanikop int MoveMainTable(uint64_t pteSector); 146*57696d54SAkhilesh Sanikop void BlankPartitions(void); 147*57696d54SAkhilesh Sanikop int DeletePartition(uint32_t partNum); 148*57696d54SAkhilesh Sanikop uint32_t CreatePartition(uint32_t partNum, uint64_t startSector, uint64_t endSector); 149*57696d54SAkhilesh Sanikop void SortGPT(void); 150*57696d54SAkhilesh Sanikop int SwapPartitions(uint32_t partNum1, uint32_t partNum2); 151*57696d54SAkhilesh Sanikop int ClearGPTData(void); 152*57696d54SAkhilesh Sanikop void MoveSecondHeaderToEnd(); 153*57696d54SAkhilesh Sanikop int SetName(uint32_t partNum, const UnicodeString & theName); 154*57696d54SAkhilesh Sanikop void SetDiskGUID(GUIDData newGUID); 155*57696d54SAkhilesh Sanikop int SetPartitionGUID(uint32_t pn, GUIDData theGUID); 156*57696d54SAkhilesh Sanikop void RandomizeGUIDs(void); 157*57696d54SAkhilesh Sanikop int ChangePartType(uint32_t pn, PartType theGUID); MakeProtectiveMBR(void)158*57696d54SAkhilesh Sanikop void MakeProtectiveMBR(void) {protectiveMBR.MakeProtectiveMBR();} 159*57696d54SAkhilesh Sanikop void RecomputeCHS(void); 160*57696d54SAkhilesh Sanikop int Align(uint64_t* sector); SetProtectiveMBR(BasicMBRData & newMBR)161*57696d54SAkhilesh Sanikop void SetProtectiveMBR(BasicMBRData & newMBR) {protectiveMBR = newMBR;} 162*57696d54SAkhilesh Sanikop 163*57696d54SAkhilesh Sanikop // Return data about the GPT structures.... GetState(void)164*57696d54SAkhilesh Sanikop WhichToUse GetState(void) {return whichWasUsed;} 165*57696d54SAkhilesh Sanikop int GetPartRange(uint32_t* low, uint32_t* high); 166*57696d54SAkhilesh Sanikop int FindFirstFreePart(void); GetNumParts(void)167*57696d54SAkhilesh Sanikop uint32_t GetNumParts(void) {return mainHeader.numParts;} GetTableSizeInSectors(void)168*57696d54SAkhilesh Sanikop uint64_t GetTableSizeInSectors(void) {return (((numParts * GPT_SIZE) / blockSize) + 169*57696d54SAkhilesh Sanikop (((numParts * GPT_SIZE) % blockSize) != 0)); } GetMainHeaderLBA(void)170*57696d54SAkhilesh Sanikop uint64_t GetMainHeaderLBA(void) {return mainHeader.currentLBA;} GetSecondHeaderLBA(void)171*57696d54SAkhilesh Sanikop uint64_t GetSecondHeaderLBA(void) {return secondHeader.currentLBA;} GetMainPartsLBA(void)172*57696d54SAkhilesh Sanikop uint64_t GetMainPartsLBA(void) {return mainHeader.partitionEntriesLBA;} GetSecondPartsLBA(void)173*57696d54SAkhilesh Sanikop uint64_t GetSecondPartsLBA(void) {return secondHeader.partitionEntriesLBA;} GetFirstUsableLBA(void)174*57696d54SAkhilesh Sanikop uint64_t GetFirstUsableLBA(void) {return mainHeader.firstUsableLBA;} GetLastUsableLBA(void)175*57696d54SAkhilesh Sanikop uint64_t GetLastUsableLBA(void) {return mainHeader.lastUsableLBA;} 176*57696d54SAkhilesh Sanikop uint32_t CountParts(void); 177*57696d54SAkhilesh Sanikop bool ValidPartNum (const uint32_t partNum); 178*57696d54SAkhilesh Sanikop const GPTPart & operator[](uint32_t partNum) const; 179*57696d54SAkhilesh Sanikop const GUIDData & GetDiskGUID(void) const; GetBlockSize(void)180*57696d54SAkhilesh Sanikop uint32_t GetBlockSize(void) {return blockSize;} 181*57696d54SAkhilesh Sanikop 182*57696d54SAkhilesh Sanikop // Find information about free space 183*57696d54SAkhilesh Sanikop uint64_t FindFirstAvailable(uint64_t start = 0); 184*57696d54SAkhilesh Sanikop uint64_t FindFirstUsedLBA(void); 185*57696d54SAkhilesh Sanikop uint64_t FindFirstInLargest(void); 186*57696d54SAkhilesh Sanikop uint64_t FindLastAvailable(); 187*57696d54SAkhilesh Sanikop uint64_t FindLastInFree(uint64_t start, bool align = false); 188*57696d54SAkhilesh Sanikop uint64_t FindFreeBlocks(uint32_t *numSegments, uint64_t *largestSegment); 189*57696d54SAkhilesh Sanikop int IsFree(uint64_t sector, uint32_t *partNum = NULL); 190*57696d54SAkhilesh Sanikop int IsFreePartNum(uint32_t partNum); 191*57696d54SAkhilesh Sanikop int IsUsedPartNum(uint32_t partNum); 192*57696d54SAkhilesh Sanikop 193*57696d54SAkhilesh Sanikop // Change how functions work, or return information on same 194*57696d54SAkhilesh Sanikop void SetAlignment(uint32_t n); 195*57696d54SAkhilesh Sanikop uint32_t ComputeAlignment(void); // Set alignment based on current partitions GetAlignment(void)196*57696d54SAkhilesh Sanikop uint32_t GetAlignment(void) {return sectorAlignment;} 197*57696d54SAkhilesh Sanikop void JustLooking(int i = 1) {justLooking = i;} 198*57696d54SAkhilesh Sanikop void BeQuiet(int i = 1) {beQuiet = i;} WhichWasUsed(void)199*57696d54SAkhilesh Sanikop WhichToUse WhichWasUsed(void) {return whichWasUsed;} 200*57696d54SAkhilesh Sanikop 201*57696d54SAkhilesh Sanikop // Endianness functions 202*57696d54SAkhilesh Sanikop void ReverseHeaderBytes(struct GPTHeader* header); 203*57696d54SAkhilesh Sanikop void ReversePartitionBytes(); // for endianness 204*57696d54SAkhilesh Sanikop 205*57696d54SAkhilesh Sanikop // Attributes functions 206*57696d54SAkhilesh Sanikop int ManageAttributes(int partNum, const std::string & command, const std::string & bits); 207*57696d54SAkhilesh Sanikop void ShowAttributes(const uint32_t partNum); 208*57696d54SAkhilesh Sanikop void GetAttribute(const uint32_t partNum, const std::string& attributeBits); 209*57696d54SAkhilesh Sanikop 210*57696d54SAkhilesh Sanikop }; // class GPTData 211*57696d54SAkhilesh Sanikop 212*57696d54SAkhilesh Sanikop // Function prototypes.... 213*57696d54SAkhilesh Sanikop int SizesOK(void); 214*57696d54SAkhilesh Sanikop 215*57696d54SAkhilesh Sanikop #endif 216