• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // parttypes.cc
2 // Class to manage partition type codes -- a slight variant on MBR type
3 // codes, GUID type codes, and associated names.
4 
5 /* This program is copyright (c) 2009-2018 by Roderick W. Smith. It is distributed
6   under the terms of the GNU GPL version 2, as detailed in the COPYING file. */
7 
8 #define __STDC_LIMIT_MACROS
9 #ifndef __STDC_CONSTANT_MACROS
10 #define __STDC_CONSTANT_MACROS
11 #endif
12 
13 #include <string.h>
14 #include <stdint.h>
15 #include <stdio.h>
16 #include <iostream>
17 #include "parttypes.h"
18 
19 using namespace std;
20 
21 int PartType::numInstances = 0;
22 AType* PartType::allTypes = NULL;
23 AType* PartType::lastType = NULL;
24 const PartType PartType::unusedPartType = (GUIDData) "00000000-0000-0000-0000-000000000000";
25 
26 #define SCREEN_WIDTH 80
27 #define NUM_COLUMNS 2
28 #define DESC_LENGTH (SCREEN_WIDTH - (6 * NUM_COLUMNS)) / NUM_COLUMNS
29 
30 // Constructor. Its main task is to initialize the data list, but only
31 // if this is the first instance, since it's a static linked list.
32 // Partition type codes are MBR type codes multiplied by 0x0100, with
33 // additional related codes taking on following numbers. For instance,
34 // the FreeBSD disklabel code in MBR is 0xa5; here, it's 0xa500, with
35 // additional FreeBSD codes being 0xa501, 0xa502, and so on. This gives
36 // related codes similar numbers and (given appropriate entry positions
37 // in the linked list) keeps them together in the listings generated
38 // by typing "L" at the main gdisk menu.
PartType(void)39 PartType::PartType(void) : GUIDData() {
40    numInstances++;
41    if (numInstances == 1) {
42       AddAllTypes();
43    } // if
44 } // default constructor
45 
PartType(const PartType & orig)46 PartType::PartType(const PartType & orig) : GUIDData(orig) {
47    numInstances++;
48    if (numInstances == 1) { // should never happen; just being paranoid
49       AddAllTypes();
50    } // if
51 } // PartType copy constructor
52 
PartType(const GUIDData & orig)53 PartType::PartType(const GUIDData & orig) : GUIDData(orig) {
54    numInstances++;
55    if (numInstances == 1) {
56       AddAllTypes();
57    } // if
58 } // PartType copy constructor
59 
~PartType(void)60 PartType::~PartType(void) {
61    AType* tempType;
62 
63    numInstances--;
64    if (numInstances == 0) {
65       while (allTypes != NULL) {
66          tempType = allTypes;
67          allTypes = allTypes->next;
68          delete tempType;
69       } // while
70    } // if
71 } // destructor
72 
73 // Add all partition type codes to the internal linked-list structure.
74 // Used by constructors.
75 // See http://www.win.tue.nl/~aeb/partitions/partition_types-1.html
76 // for a list of MBR partition type codes.
AddAllTypes(void)77 void PartType::AddAllTypes(void) {
78    // Start with the "unused entry," which should normally appear only
79    // on empty partition table entries....
80    AddType(0x0000, "00000000-0000-0000-0000-000000000000", "Unused entry", 0);
81 
82    // DOS/Windows partition types, most of which are hidden from the "L" listing
83    // (they're available mainly for MBR-to-GPT conversions).
84    AddType(0x0100, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-12
85    AddType(0x0400, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-16 < 32M
86    AddType(0x0600, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-16
87    AddType(0x0700, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 1); // NTFS (or HPFS)
88    AddType(0x0b00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-32
89    AddType(0x0c00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-32 LBA
90    AddType(0x0c01, "E3C9E316-0B5C-4DB8-817D-F92DF00215AE", "Microsoft reserved");
91    AddType(0x0e00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-16 LBA
92    AddType(0x1100, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-12
93    AddType(0x1400, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-16 < 32M
94    AddType(0x1600, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-16
95    AddType(0x1700, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden NTFS (or HPFS)
96    AddType(0x1b00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-32
97    AddType(0x1c00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-32 LBA
98    AddType(0x1e00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-16 LBA
99    AddType(0x2700, "DE94BBA4-06D1-4D40-A16A-BFD50179D6AC", "Windows RE");
100 
101    // Open Network Install Environment (ONIE) specific types.
102    // See http://www.onie.org/ and
103    // https://github.com/opencomputeproject/onie/blob/master/patches/gptfdisk/add-onie-partition-types.patch
104    AddType(0x3000, "7412F7D5-A156-4B13-81DC-867174929325", "ONIE boot");
105    AddType(0x3001, "D4E6E2CD-4469-46F3-B5CB-1BFF57AFC149", "ONIE config");
106 
107    // Plan 9; see http://man.cat-v.org/9front/8/prep
108    AddType(0x3900, "C91818F9-8025-47AF-89D2-F030D7000C2C", "Plan 9");
109 
110    // PowerPC reference platform boot partition
111    AddType(0x4100, "9E1A2D38-C612-4316-AA26-8B49521E5A8B", "PowerPC PReP boot");
112 
113    // Windows LDM ("dynamic disk") types
114    AddType(0x4200, "AF9B60A0-1431-4F62-BC68-3311714A69AD", "Windows LDM data"); // Logical disk manager
115    AddType(0x4201, "5808C8AA-7E8F-42E0-85D2-E1E90434CFB3", "Windows LDM metadata"); // Logical disk manager
116    AddType(0x4202, "E75CAF8F-F680-4CEE-AFA3-B001E56EFC2D", "Windows Storage Spaces"); // A newer LDM-type setup
117 
118    // An oddball IBM filesystem....
119    AddType(0x7501, "37AFFC90-EF7D-4E96-91C3-2D7AE055B174", "IBM GPFS"); // General Parallel File System (GPFS)
120 
121    // ChromeOS-specific partition types...
122    // Values taken from vboot_reference/firmware/lib/cgptlib/include/gpt.h in
123    // ChromeOS source code, retrieved 12/23/2010. They're also at
124    // http://www.chromium.org/chromium-os/chromiumos-design-docs/disk-format.
125    // These have no MBR equivalents, AFAIK, so I'm using 0x7Fxx values, since they're close
126    // to the Linux values.
127    AddType(0x7f00, "FE3A2A5D-4F32-41A7-B725-ACCC3285A309", "ChromeOS kernel");
128    AddType(0x7f01, "3CB8E202-3B7E-47DD-8A3C-7FF2A13CFCEC", "ChromeOS root");
129    AddType(0x7f02, "2E0A753D-9E48-43B0-8337-B15192CB1B5E", "ChromeOS reserved");
130 
131    // Linux-specific partition types....
132    AddType(0x8200, "0657FD6D-A4AB-43C4-84E5-0933C84B4F4F", "Linux swap"); // Linux swap (or Solaris on MBR)
133    AddType(0x8300, "0FC63DAF-8483-4772-8E79-3D69D8477DE4", "Linux filesystem"); // Linux native
134    AddType(0x8301, "8DA63339-0007-60C0-C436-083AC8230908", "Linux reserved");
135    // See https://www.freedesktop.org/software/systemd/man/systemd-gpt-auto-generator.html
136    // and https://systemd.io/DISCOVERABLE_PARTITIONS
137    AddType(0x8302, "933AC7E1-2EB4-4F13-B844-0E14E2AEF915", "Linux /home"); // Linux /home (auto-mounted by systemd)
138    AddType(0x8303, "44479540-F297-41B2-9AF7-D131D5F0458A", "Linux x86 root (/)"); // Linux / on x86 (auto-mounted by systemd)
139    AddType(0x8304, "4F68BCE3-E8CD-4DB1-96E7-FBCAF984B709", "Linux x86-64 root (/)"); // Linux / on x86-64 (auto-mounted by systemd)
140    AddType(0x8305, "B921B045-1DF0-41C3-AF44-4C6F280D3FAE", "Linux ARM64 root (/)"); // Linux / on 64-bit ARM (auto-mounted by systemd)
141    AddType(0x8306, "3B8F8425-20E0-4F3B-907F-1A25A76F98E8", "Linux /srv"); // Linux /srv (auto-mounted by systemd)
142    AddType(0x8307, "69DAD710-2CE4-4E3C-B16C-21A1D49ABED3", "Linux ARM32 root (/)"); // Linux / on 32-bit ARM (auto-mounted by systemd)
143    AddType(0x8308, "7FFEC5C9-2D00-49B7-8941-3EA10A5586B7", "Linux dm-crypt");
144    AddType(0x8309, "CA7D7CCB-63ED-4C53-861C-1742536059CC", "Linux LUKS");
145    AddType(0x830A, "993D8D3D-F80E-4225-855A-9DAF8ED7EA97", "Linux IA-64 root (/)"); // Linux / on Itanium (auto-mounted by systemd)
146    AddType(0x830B, "D13C5D3B-B5D1-422A-B29F-9454FDC89D76", "Linux x86 root verity");
147    AddType(0x830C, "2C7357ED-EBD2-46D9-AEC1-23D437EC2BF5", "Linux x86-64 root verity");
148    AddType(0x830D, "7386CDF2-203C-47A9-A498-F2ECCE45A2D6", "Linux ARM32 root verity");
149    AddType(0x830E, "DF3300CE-D69F-4C92-978C-9BFB0F38D820", "Linux ARM64 root verity");
150    AddType(0x830F, "86ED10D5-B607-45BB-8957-D350F23D0571", "Linux IA-64 root verity");
151    AddType(0x8310, "4D21B016-B534-45C2-A9FB-5C16E091FD2D", "Linux /var"); // Linux /var (auto-mounted by systemd)
152    AddType(0x8311, "7EC6F557-3BC5-4ACA-B293-16EF5DF639D1", "Linux /var/tmp"); // Linux /var/tmp (auto-mounted by systemd)
153 
154    // Used by Intel Rapid Start technology
155    AddType(0x8400, "D3BFE2DE-3DAF-11DF-BA40-E3A556D89593", "Intel Rapid Start");
156 
157    // Type codes for Container Linux (formerly CoreOS; https://coreos.com)
158    AddType(0x8500, "5DFBF5F4-2848-4BAC-AA5E-0D9A20B745A6", "Container Linux /usr");
159    AddType(0x8501, "3884DD41-8582-4404-B9A8-E9B84F2DF50E", "Container Linux resizable rootfs");
160    AddType(0x8502, "C95DC21A-DF0E-4340-8D7B-26CBFA9A03E0", "Container Linux /OEM customizations");
161    AddType(0x8503, "BE9067B9-EA49-4F15-B4F6-F36F8C9E1818", "Container Linux root on RAID");
162 
163    // Another Linux type code....
164    AddType(0x8e00, "E6D6D379-F507-44C2-A23C-238F2A3DF928", "Linux LVM");
165 
166    // Android type codes....
167    // from Wikipedia, https://gist.github.com/culots/704afd126dec2f45c22d0c9d42cb7fab,
168    // and my own Android devices' partition tables
169    AddType(0xa000, "2568845D-2332-4675-BC39-8FA5A4748D15", "Android bootloader");
170    AddType(0xa001, "114EAFFE-1552-4022-B26E-9B053604CF84", "Android bootloader 2");
171    AddType(0xa002, "49A4D17F-93A3-45C1-A0DE-F50B2EBE2599", "Android boot 1");
172    AddType(0xa003, "4177C722-9E92-4AAB-8644-43502BFD5506", "Android recovery 1");
173    AddType(0xa004, "EF32A33B-A409-486C-9141-9FFB711F6266", "Android misc");
174    AddType(0xa005, "20AC26BE-20B7-11E3-84C5-6CFDB94711E9", "Android metadata");
175    AddType(0xa006, "38F428E6-D326-425D-9140-6E0EA133647C", "Android system 1");
176    AddType(0xa007, "A893EF21-E428-470A-9E55-0668FD91A2D9", "Android cache");
177    AddType(0xa008, "DC76DDA9-5AC1-491C-AF42-A82591580C0D", "Android data");
178    AddType(0xa009, "EBC597D0-2053-4B15-8B64-E0AAC75F4DB1", "Android persistent");
179    AddType(0xa00a, "8F68CC74-C5E5-48DA-BE91-A0C8C15E9C80", "Android factory");
180    AddType(0xa00b, "767941D0-2085-11E3-AD3B-6CFDB94711E9", "Android fastboot/tertiary");
181    AddType(0xa00c, "AC6D7924-EB71-4DF8-B48D-E267B27148FF", "Android OEM");
182    AddType(0xa00d, "C5A0AEEC-13EA-11E5-A1B1-001E67CA0C3C", "Android vendor");
183    AddType(0xa00e, "BD59408B-4514-490D-BF12-9878D963F378", "Android config");
184    AddType(0xa00f, "9FDAA6EF-4B3F-40D2-BA8D-BFF16BFB887B", "Android factory (alt)");
185    AddType(0xa010, "19A710A2-B3CA-11E4-B026-10604B889DCF", "Android meta");
186    AddType(0xa011, "193D1EA4-B3CA-11E4-B075-10604B889DCF", "Android EXT");
187    AddType(0xa012, "DEA0BA2C-CBDD-4805-B4F9-F428251C3E98", "Android SBL1");
188    AddType(0xa013, "8C6B52AD-8A9E-4398-AD09-AE916E53AE2D", "Android SBL2");
189    AddType(0xa014, "05E044DF-92F1-4325-B69E-374A82E97D6E", "Android SBL3");
190    AddType(0xa015, "400FFDCD-22E0-47E7-9A23-F16ED9382388", "Android APPSBL");
191    AddType(0xa016, "A053AA7F-40B8-4B1C-BA08-2F68AC71A4F4", "Android QSEE/tz");
192    AddType(0xa017, "E1A6A689-0C8D-4CC6-B4E8-55A4320FBD8A", "Android QHEE/hyp");
193    AddType(0xa018, "098DF793-D712-413D-9D4E-89D711772228", "Android RPM");
194    AddType(0xa019, "D4E0D938-B7FA-48C1-9D21-BC5ED5C4B203", "Android WDOG debug/sdi");
195    AddType(0xa01a, "20A0C19C-286A-42FA-9CE7-F64C3226A794", "Android DDR");
196    AddType(0xa01b, "A19F205F-CCD8-4B6D-8F1E-2D9BC24CFFB1", "Android CDT");
197    AddType(0xa01c, "66C9B323-F7FC-48B6-BF96-6F32E335A428", "Android RAM dump");
198    AddType(0xa01d, "303E6AC3-AF15-4C54-9E9B-D9A8FBECF401", "Android SEC");
199    AddType(0xa01e, "C00EEF24-7709-43D6-9799-DD2B411E7A3C", "Android PMIC");
200    AddType(0xa01f, "82ACC91F-357C-4A68-9C8F-689E1B1A23A1", "Android misc 1");
201    AddType(0xa020, "E2802D54-0545-E8A1-A1E8-C7A3E245ACD4", "Android misc 2");
202    AddType(0xa021, "65ADDCF4-0C5C-4D9A-AC2D-D90B5CBFCD03", "Android device info");
203    AddType(0xa022, "E6E98DA2-E22A-4D12-AB33-169E7DEAA507", "Android APDP");
204    AddType(0xa023, "ED9E8101-05FA-46B7-82AA-8D58770D200B", "Android MSADP");
205    AddType(0xa024, "11406F35-1173-4869-807B-27DF71802812", "Android DPO");
206    AddType(0xa025, "9D72D4E4-9958-42DA-AC26-BEA7A90B0434", "Android recovery 2");
207    AddType(0xa026, "6C95E238-E343-4BA8-B489-8681ED22AD0B", "Android persist");
208    AddType(0xa027, "EBBEADAF-22C9-E33B-8F5D-0E81686A68CB", "Android modem ST1");
209    AddType(0xa028, "0A288B1F-22C9-E33B-8F5D-0E81686A68CB", "Android modem ST2");
210    AddType(0xa029, "57B90A16-22C9-E33B-8F5D-0E81686A68CB", "Android FSC");
211    AddType(0xa02a, "638FF8E2-22C9-E33B-8F5D-0E81686A68CB", "Android FSG 1");
212    AddType(0xa02b, "2013373E-1AC4-4131-BFD8-B6A7AC638772", "Android FSG 2");
213    AddType(0xa02c, "2C86E742-745E-4FDD-BFD8-B6A7AC638772", "Android SSD");
214    AddType(0xa02d, "DE7D4029-0F5B-41C8-AE7E-F6C023A02B33", "Android keystore");
215    AddType(0xa02e, "323EF595-AF7A-4AFA-8060-97BE72841BB9", "Android encrypt");
216    AddType(0xa02f, "45864011-CF89-46E6-A445-85262E065604", "Android EKSST");
217    AddType(0xa030, "8ED8AE95-597F-4C8A-A5BD-A7FF8E4DFAA9", "Android RCT");
218    AddType(0xa031, "DF24E5ED-8C96-4B86-B00B-79667DC6DE11", "Android spare1");
219    AddType(0xa032, "7C29D3AD-78B9-452E-9DEB-D098D542F092", "Android spare2");
220    AddType(0xa033, "379D107E-229E-499D-AD4F-61F5BCF87BD4", "Android spare3");
221    AddType(0xa034, "0DEA65E5-A676-4CDF-823C-77568B577ED5", "Android spare4");
222    AddType(0xa035, "4627AE27-CFEF-48A1-88FE-99C3509ADE26", "Android raw resources");
223    AddType(0xa036, "20117F86-E985-4357-B9EE-374BC1D8487D", "Android boot 2");
224    AddType(0xa037, "86A7CB80-84E1-408C-99AB-694F1A410FC7", "Android FOTA");
225    AddType(0xa038, "97D7B011-54DA-4835-B3C4-917AD6E73D74", "Android system 2");
226    AddType(0xa039, "5594C694-C871-4B5F-90B1-690A6F68E0F7", "Android cache");
227    AddType(0xa03a, "1B81E7E6-F50D-419B-A739-2AEEF8DA3335", "Android user data");
228    AddType(0xa03b, "98523EC6-90FE-4C67-B50A-0FC59ED6F56D", "LG (Android) advanced flasher");
229    AddType(0xa03c, "2644BCC0-F36A-4792-9533-1738BED53EE3", "Android PG1FS");
230    AddType(0xa03d, "DD7C91E9-38C9-45C5-8A12-4A80F7E14057", "Android PG2FS");
231    AddType(0xa03e, "7696D5B6-43FD-4664-A228-C563C4A1E8CC", "Android board info");
232    AddType(0xa03f, "0D802D54-058D-4A20-AD2D-C7A362CEACD4", "Android MFG");
233    AddType(0xa040, "10A0C19C-516A-5444-5CE3-664C3226A794", "Android limits");
234 
235    // Atari TOS partition type
236    AddType(0xa200, "734E5AFE-F61A-11E6-BC64-92361F002671", "Atari TOS basic data");
237 
238    // FreeBSD partition types....
239    // Note: Rather than extract FreeBSD disklabel data, convert FreeBSD
240    // partitions in-place, and let FreeBSD sort out the details....
241    AddType(0xa500, "516E7CB4-6ECF-11D6-8FF8-00022D09712B", "FreeBSD disklabel");
242    AddType(0xa501, "83BD6B9D-7F41-11DC-BE0B-001560B84F0F", "FreeBSD boot");
243    AddType(0xa502, "516E7CB5-6ECF-11D6-8FF8-00022D09712B", "FreeBSD swap");
244    AddType(0xa503, "516E7CB6-6ECF-11D6-8FF8-00022D09712B", "FreeBSD UFS");
245    AddType(0xa504, "516E7CBA-6ECF-11D6-8FF8-00022D09712B", "FreeBSD ZFS");
246    AddType(0xa505, "516E7CB8-6ECF-11D6-8FF8-00022D09712B", "FreeBSD Vinum/RAID");
247 
248    // Midnight BSD partition types....
249    AddType(0xa580, "85D5E45A-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD data");
250    AddType(0xa581, "85D5E45E-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD boot");
251    AddType(0xa582, "85D5E45B-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD swap");
252    AddType(0xa583, "0394Ef8B-237E-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD UFS");
253    AddType(0xa584, "85D5E45D-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD ZFS");
254    AddType(0xa585, "85D5E45C-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD Vinum");
255 
256    // OpenBSD partition type....
257    AddType(0xa600, "824CC7A0-36A8-11E3-890A-952519AD3F61", "OpenBSD disklabel");
258 
259    // A MacOS partition type, separated from others by NetBSD partition types...
260    AddType(0xa800, "55465300-0000-11AA-AA11-00306543ECAC", "Apple UFS"); // Mac OS X
261 
262    // NetBSD partition types. Note that the main entry sets it up as a
263    // FreeBSD disklabel. I'm not 100% certain this is the correct behavior.
264    AddType(0xa900, "516E7CB4-6ECF-11D6-8FF8-00022D09712B", "FreeBSD disklabel", 0); // NetBSD disklabel
265    AddType(0xa901, "49F48D32-B10E-11DC-B99B-0019D1879648", "NetBSD swap");
266    AddType(0xa902, "49F48D5A-B10E-11DC-B99B-0019D1879648", "NetBSD FFS");
267    AddType(0xa903, "49F48D82-B10E-11DC-B99B-0019D1879648", "NetBSD LFS");
268    AddType(0xa904, "2DB519C4-B10F-11DC-B99B-0019D1879648", "NetBSD concatenated");
269    AddType(0xa905, "2DB519EC-B10F-11DC-B99B-0019D1879648", "NetBSD encrypted");
270    AddType(0xa906, "49F48DAA-B10E-11DC-B99B-0019D1879648", "NetBSD RAID");
271 
272    // Mac OS partition types (See also 0xa800, above)....
273    AddType(0xab00, "426F6F74-0000-11AA-AA11-00306543ECAC", "Recovery HD");
274    AddType(0xaf00, "48465300-0000-11AA-AA11-00306543ECAC", "Apple HFS/HFS+");
275    AddType(0xaf01, "52414944-0000-11AA-AA11-00306543ECAC", "Apple RAID");
276    AddType(0xaf02, "52414944-5F4F-11AA-AA11-00306543ECAC", "Apple RAID offline");
277    AddType(0xaf03, "4C616265-6C00-11AA-AA11-00306543ECAC", "Apple label");
278    AddType(0xaf04, "5265636F-7665-11AA-AA11-00306543ECAC", "AppleTV recovery");
279    AddType(0xaf05, "53746F72-6167-11AA-AA11-00306543ECAC", "Apple Core Storage");
280    AddType(0xaf06, "B6FA30DA-92D2-4A9A-96F1-871EC6486200", "Apple SoftRAID Status");
281    AddType(0xaf07, "2E313465-19B9-463F-8126-8A7993773801", "Apple SoftRAID Scratch");
282    AddType(0xaf08, "FA709C7E-65B1-4593-BFD5-E71D61DE9B02", "Apple SoftRAID Volume");
283    AddType(0xaf09, "BBBA6DF5-F46F-4A89-8F59-8765B2727503", "Apple SoftRAID Cache");
284    AddType(0xaf0a, "7C3457EF-0000-11AA-AA11-00306543ECAC", "Apple APFS");
285 
286    // QNX Power-Safe (QNX6)
287    AddType(0xb300, "CEF5A9AD-73BC-4601-89F3-CDEEEEE321A1", "QNX6 Power-Safe");
288 
289    // Acronis Secure Zone
290    AddType(0xbc00, "0311FC50-01CA-4725-AD77-9ADBB20ACE98", "Acronis Secure Zone");
291 
292    // Solaris partition types (one of which is shared with MacOS)
293    AddType(0xbe00, "6A82CB45-1DD2-11B2-99A6-080020736631", "Solaris boot");
294    AddType(0xbf00, "6A85CF4D-1DD2-11B2-99A6-080020736631", "Solaris root");
295    AddType(0xbf01, "6A898CC3-1DD2-11B2-99A6-080020736631", "Solaris /usr & Mac ZFS"); // Solaris/MacOS
296    AddType(0xbf02, "6A87C46F-1DD2-11B2-99A6-080020736631", "Solaris swap");
297    AddType(0xbf03, "6A8B642B-1DD2-11B2-99A6-080020736631", "Solaris backup");
298    AddType(0xbf04, "6A8EF2E9-1DD2-11B2-99A6-080020736631", "Solaris /var");
299    AddType(0xbf05, "6A90BA39-1DD2-11B2-99A6-080020736631", "Solaris /home");
300    AddType(0xbf06, "6A9283A5-1DD2-11B2-99A6-080020736631", "Solaris alternate sector");
301    AddType(0xbf07, "6A945A3B-1DD2-11B2-99A6-080020736631", "Solaris Reserved 1");
302    AddType(0xbf08, "6A9630D1-1DD2-11B2-99A6-080020736631", "Solaris Reserved 2");
303    AddType(0xbf09, "6A980767-1DD2-11B2-99A6-080020736631", "Solaris Reserved 3");
304    AddType(0xbf0a, "6A96237F-1DD2-11B2-99A6-080020736631", "Solaris Reserved 4");
305    AddType(0xbf0b, "6A8D2AC7-1DD2-11B2-99A6-080020736631", "Solaris Reserved 5");
306 
307    // I can find no MBR equivalents for these, but they're on the
308    // Wikipedia page for GPT, so here we go....
309    AddType(0xc001, "75894C1E-3AEB-11D3-B7C1-7B03A0000000", "HP-UX data");
310    AddType(0xc002, "E2A1E728-32E3-11D6-A682-7B03A0000000", "HP-UX service");
311 
312    // Open Network Install Environment (ONIE) partitions....
313    AddType(0xe100, "7412F7D5-A156-4B13-81DC-867174929325", "ONIE boot");
314    AddType(0xe101, "D4E6E2CD-4469-46F3-B5CB-1BFF57AFC149", "ONIE config");
315 
316    // Veracrypt (https://www.veracrypt.fr/en/Home.html) encrypted partition
317    AddType(0xe900, "8C8F8EFF-AC95-4770-814A-21994F2DBC8F", "Veracrypt data");
318 
319    // See http://www.freedesktop.org/wiki/Specifications/BootLoaderSpec
320    AddType(0xea00, "BC13C2FF-59E6-4262-A352-B275FD6F7172", "Freedesktop $BOOT");
321 
322    // Type code for Haiku; uses BeOS MBR code as hex code base
323    AddType(0xeb00, "42465331-3BA3-10F1-802A-4861696B7521", "Haiku BFS");
324 
325    // Manufacturer-specific ESP-like partitions (in order in which they were added)
326    AddType(0xed00, "F4019732-066E-4E12-8273-346C5641494F", "Sony system partition");
327    AddType(0xed01, "BFBFAFE7-A34F-448A-9A5B-6213EB736C22", "Lenovo system partition");
328 
329    // EFI system and related partitions
330    AddType(0xef00, "C12A7328-F81F-11D2-BA4B-00A0C93EC93B", "EFI system partition"); // Parted identifies these as having the "boot flag" set
331    AddType(0xef01, "024DEE41-33E7-11D3-9D69-0008C781F39F", "MBR partition scheme"); // Used to nest MBR in GPT
332    AddType(0xef02, "21686148-6449-6E6F-744E-656564454649", "BIOS boot partition"); // Used by GRUB
333 
334    // Ceph type codes; see https://github.com/ceph/ceph/blob/9bcc42a3e6b08521694b5c0228b2c6ed7b3d312e/src/ceph-disk#L76-L81
335    // and Wikipedia
336    AddType(0xf800, "4FBD7E29-9D25-41B8-AFD0-062C0CEFF05D", "Ceph OSD"); // Ceph Object Storage Daemon
337    AddType(0xf801, "4FBD7E29-9D25-41B8-AFD0-5EC00CEFF05D", "Ceph dm-crypt OSD"); // Ceph Object Storage Daemon (encrypted)
338    AddType(0xf802, "45B0969E-9B03-4F30-B4C6-B4B80CEFF106", "Ceph journal");
339    AddType(0xf803, "45B0969E-9B03-4F30-B4C6-5EC00CEFF106", "Ceph dm-crypt journal");
340    AddType(0xf804, "89C57F98-2FE5-4DC0-89C1-F3AD0CEFF2BE", "Ceph disk in creation");
341    AddType(0xf805, "89C57F98-2FE5-4DC0-89C1-5EC00CEFF2BE", "Ceph dm-crypt disk in creation");
342    AddType(0xf806, "CAFECAFE-9B03-4F30-B4C6-B4B80CEFF106", "Ceph block");
343    AddType(0xf807, "30CD0809-C2B2-499C-8879-2D6B78529876", "Ceph block DB");
344    AddType(0xf808, "5CE17FCE-4087-4169-B7FF-056CC58473F9", "Ceph block write-ahead log");
345    AddType(0xf809, "FB3AABF9-D25F-47CC-BF5E-721D1816496B", "Ceph lockbox for dm-crypt keys");
346    AddType(0xf80a, "4FBD7E29-8AE0-4982-BF9D-5A8D867AF560", "Ceph multipath OSD");
347    AddType(0xf80b, "45B0969E-8AE0-4982-BF9D-5A8D867AF560", "Ceph multipath journal");
348    AddType(0xf80c, "CAFECAFE-8AE0-4982-BF9D-5A8D867AF560", "Ceph multipath block 1");
349    AddType(0xf80d, "7F4A666A-16F3-47A2-8445-152EF4D03F6C", "Ceph multipath block 2");
350    AddType(0xf80e, "EC6D6385-E346-45DC-BE91-DA2A7C8B3261", "Ceph multipath block DB");
351    AddType(0xf80f, "01B41E1B-002A-453C-9F17-88793989FF8F", "Ceph multipath block write-ahead log");
352    AddType(0xf810, "CAFECAFE-9B03-4F30-B4C6-5EC00CEFF106", "Ceph dm-crypt block");
353    AddType(0xf811, "93B0052D-02D9-4D8A-A43B-33A3EE4DFBC3", "Ceph dm-crypt block DB");
354    AddType(0xf812, "306E8683-4FE2-4330-B7C0-00A917C16966", "Ceph dm-crypt block write-ahead log");
355    AddType(0xf813, "45B0969E-9B03-4F30-B4C6-35865CEFF106", "Ceph dm-crypt LUKS journal");
356    AddType(0xf814, "CAFECAFE-9B03-4F30-B4C6-35865CEFF106", "Ceph dm-crypt LUKS block");
357    AddType(0xf815, "166418DA-C469-4022-ADF4-B30AFD37F176", "Ceph dm-crypt LUKS block DB");
358    AddType(0xf816, "86A32090-3647-40B9-BBBD-38D8C573AA86", "Ceph dm-crypt LUKS block write-ahead log");
359    AddType(0xf817, "4FBD7E29-9D25-41B8-AFD0-35865CEFF05D", "Ceph dm-crypt LUKS OSD");
360 
361    // VMWare ESX partition types codes
362    AddType(0xfb00, "AA31E02A-400F-11DB-9590-000C2911D1B8", "VMWare VMFS");
363    AddType(0xfb01, "9198EFFC-31C0-11DB-8F78-000C2911D1B8", "VMWare reserved");
364    AddType(0xfc00, "9D275380-40AD-11DB-BF97-000C2911D1B8", "VMWare kcore crash protection");
365 
366    // A straggler Linux partition type....
367    AddType(0xfd00, "A19D880F-05FC-4D3B-A006-743F0F84911E", "Linux RAID");
368 
369    // Note: DO NOT use the 0xffff code; that's reserved to indicate an
370    // unknown GUID type code.
371 } // PartType::AddAllTypes()
372 
373 // Add a single type to the linked list of types. Returns 1 if operation
374 // succeeds, 0 otherwise.
AddType(uint16_t mbrType,const char * guidData,const char * name,int toDisplay)375 int PartType::AddType(uint16_t mbrType, const char * guidData, const char * name,
376                       int toDisplay) {
377    AType* tempType;
378    int allOK = 1;
379 
380    tempType = new AType;
381    if (tempType != NULL) {
382       tempType->MBRType = mbrType;
383       tempType->GUIDType = guidData;
384       tempType->name = name;
385       tempType->display = toDisplay;
386       tempType->next = NULL;
387       if (allTypes == NULL) { // first entry
388          allTypes = tempType;
389       } else {
390          lastType->next = tempType;
391       } // if/else
392       lastType = tempType;
393    } else {
394       cerr << "Unable to allocate memory in PartType::AddType()! Partition type list will\n";
395       cerr << "be incomplete!\n";
396       allOK = 0;
397    } // if/else
398    return allOK;
399 } // GUID::AddType(const char* variant)
400 
401 // Assignment operator by string. If the original string is short,
402 // interpret it as a gdisk hex code; if it's longer, interpret it as
403 // a direct entry of a GUID value. If a short string isn't a hex
404 // number, do nothing.
operator =(const string & orig)405 PartType & PartType::operator=(const string & orig) {
406    uint32_t hexCode;
407 
408    if (orig.length() < 32) {
409       if (IsHex(orig)) {
410          sscanf(orig.c_str(), "%x", &hexCode);
411          *this = hexCode;
412       }
413    } else {
414       GUIDData::operator=(orig);
415    } // if/else hexCode or GUID
416    return *this;
417 } // PartType::operator=(const char * orig)
418 
419 // Assignment from C-style string; rely on C++ casting....
operator =(const char * orig)420 PartType & PartType::operator=(const char * orig) {
421    return operator=((string) orig);
422 } // PartType::operator=(const char * orig)
423 
424 // Assign a GUID based on my custom 2-byte (16-bit) MBR hex ID variant
operator =(uint16_t ID)425 PartType & PartType::operator=(uint16_t ID) {
426    AType* theItem = allTypes;
427    int found = 0;
428 
429    // Now search the type list for a match to the ID....
430    while ((theItem != NULL) && (!found)) {
431       if (theItem->MBRType == ID)  {
432          GUIDData::operator=(theItem->GUIDType);
433          found = 1;
434       } else {
435          theItem = theItem->next;
436       } // if/else
437    } // while
438    if (!found) {
439       // Assign a default value....
440       operator=(DEFAULT_GPT_TYPE);
441       cout.setf(ios::uppercase);
442       cout.fill('0');
443       cout << "Exact type match not found for type code ";
444       cout.width(4);
445       cout << hex << ID << "; assigning type code for\n'" << TypeName() << "'\n" << dec;
446       cout.fill(' ');
447    } // if (!found)
448    return *this;
449 } // PartType::operator=(uint16_t ID)
450 
451 // Return the English description of the partition type (e.g., "Linux filesystem")
TypeName(void) const452 string PartType::TypeName(void) const {
453    AType* theItem = allTypes;
454    int found = 0;
455    string typeName;
456 
457    while ((theItem != NULL) && (!found)) {
458       if (theItem->GUIDType == *this) { // found it!
459          typeName = theItem->name;
460          found = 1;
461       } else {
462          theItem = theItem->next;
463       } // if/else
464    } // while
465    if (!found) {
466       typeName = "Unknown";
467    } // if (!found)
468    return typeName;
469 } // PartType::TypeName()
470 
471 #ifdef USE_UTF16
472 // Return the Unicode description of the partition type (e.g., "Linux filesystem")
UTypeName(void) const473 UnicodeString PartType::UTypeName(void) const {
474    AType* theItem = allTypes;
475    int found = 0;
476    UnicodeString typeName;
477 
478    while ((theItem != NULL) && (!found)) {
479       if (theItem->GUIDType == *this) { // found it!
480          typeName = theItem->name.c_str();
481          found = 1;
482       } else {
483          theItem = theItem->next;
484       } // if/else
485    } // while
486    if (!found) {
487       typeName = "Unknown";
488    } // if (!found)
489    return typeName;
490 } // PartType::TypeName()
491 #endif
492 
493 // Return the custom GPT fdisk 2-byte (16-bit) hex code for this GUID partition type
494 // Note that this function ignores entries for which the display variable
495 // is set to 0. This enables control of which values get returned when
496 // there are multiple possibilities, but opens the algorithm up to the
497 // potential for problems should the data in the list be bad.
GetHexType() const498 uint16_t PartType::GetHexType() const {
499    AType* theItem = allTypes;
500    int found = 0;
501    uint16_t theID = 0xFFFF;
502 
503    while ((theItem != NULL) && (!found)) {
504       if ((theItem->GUIDType == *this) && (theItem->display == 1)) { // found it!
505          theID = theItem->MBRType;
506          found = 1;
507       } else {
508          theItem = theItem->next;
509       } // if/else
510    } // while
511    if (!found) {
512       theID = 0xFFFF;
513    } // if (!found)
514    return theID;
515 } // PartType::GetHex()
516 
517 // Displays the available types and my extended MBR codes for same....
518 // Note: This function assumes an 80-column display. On wider displays,
519 // it stops at under 80 columns; on narrower displays, lines will wrap
520 // in an ugly way. The maxLines value is the maximum number of lines
521 // to display before prompting to continue, or 0 (or a negative value)
522 // for no limit. If (maxLines > 0), this function will prompt for a
523 // substring to search for in the partition type description, so it's
524 // imperative that maxLines be set to 0 in non-interactive contexts
525 // (namely, sgdisk).
ShowAllTypes(int maxLines) const526 void PartType::ShowAllTypes(int maxLines) const {
527    int colCount = 1, lineCount = 1;
528    size_t i;
529    AType* thisType = allTypes;
530    string line, matchString = "";
531    size_t found;
532 
533    cout.unsetf(ios::uppercase);
534    if (maxLines > 0) {
535       cout << "Type search string, or <Enter> to show all codes: ";
536       matchString = ToLower(ReadString());
537    } // if
538    while (thisType != NULL) {
539       found = ToLower(thisType->name).find(matchString);
540       if ((thisType->display == 1) && (found != string::npos)) { // show it
541          cout.fill('0');
542          cout.width(4);
543          cout << hex << thisType->MBRType << " ";
544          cout << thisType->name.substr(0, DESC_LENGTH);
545          for (i = 0; i < (DESC_LENGTH - (thisType->name.substr(0, DESC_LENGTH).length())); i++)
546             cout << " ";
547          if ((colCount % NUM_COLUMNS) == 0) {
548             if (thisType->next) {
549                cout << "\n";
550                if ((maxLines > 0) && (lineCount++ % maxLines) == 0) {
551                   cout << "Press the <Enter> key to see more codes, q to quit: ";
552                   getline(cin, line);
553                   if ((line[0] =='q') || (line[0] =='Q'))
554                      break;
555                } // if reached screen line limit
556             } // if there's another entry following this one
557          } else {
558             cout << "  ";
559          }
560          colCount++;
561       } // if
562       thisType = thisType->next;
563    } // while
564    cout.fill(' ');
565    cout << "\n" << dec;
566 } // PartType::ShowAllTypes(int maxLines)
567 
568 // Returns 1 if code is a valid extended MBR code, 0 if it's not
Valid(uint16_t code) const569 int PartType::Valid(uint16_t code) const {
570    AType* thisType = allTypes;
571    int found = 0;
572 
573    while ((thisType != NULL) && (!found)) {
574       if (thisType->MBRType == code) {
575          found = 1;
576       } // if
577       thisType = thisType->next;
578    } // while
579    return found;
580 } // PartType::Valid()
581