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-2014 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 25 // Constructor. Its main task is to initialize the data list, but only 26 // if this is the first instance, since it's a static linked list. 27 // Partition type codes are MBR type codes multiplied by 0x0100, with 28 // additional related codes taking on following numbers. For instance, 29 // the FreeBSD disklabel code in MBR is 0xa5; here, it's 0xa500, with 30 // additional FreeBSD codes being 0xa501, 0xa502, and so on. This gives 31 // related codes similar numbers and (given appropriate entry positions 32 // in the linked list) keeps them together in the listings generated 33 // by typing "L" at the main gdisk menu. 34 PartType::PartType(void) : GUIDData() { 35 numInstances++; 36 if (numInstances == 1) { 37 AddAllTypes(); 38 } // if 39 } // default constructor 40 41 PartType::PartType(const PartType & orig) : GUIDData(orig) { 42 numInstances++; 43 if (numInstances == 1) { // should never happen; just being paranoid 44 AddAllTypes(); 45 } // if 46 } // PartType copy constructor 47 48 PartType::PartType(const GUIDData & orig) : GUIDData(orig) { 49 numInstances++; 50 if (numInstances == 1) { 51 AddAllTypes(); 52 } // if 53 } // PartType copy constructor 54 55 PartType::~PartType(void) { 56 AType* tempType; 57 58 numInstances--; 59 if (numInstances == 0) { 60 while (allTypes != NULL) { 61 tempType = allTypes; 62 allTypes = allTypes->next; 63 delete tempType; 64 } // while 65 } // if 66 } // destructor 67 68 // Add all partition type codes to the internal linked-list structure. 69 // Used by constructors. 70 // See http://www.win.tue.nl/~aeb/partitions/partition_types-1.html 71 // for a list of MBR partition type codes. 72 void PartType::AddAllTypes(void) { 73 // Start with the "unused entry," which should normally appear only 74 // on empty partition table entries.... 75 AddType(0x0000, "00000000-0000-0000-0000-000000000000", "Unused entry", 0); 76 77 // DOS/Windows partition types, most of which are hidden from the "L" listing 78 // (they're available mainly for MBR-to-GPT conversions). 79 AddType(0x0100, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-12 80 AddType(0x0400, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-16 < 32M 81 AddType(0x0600, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-16 82 AddType(0x0700, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 1); // NTFS (or HPFS) 83 AddType(0x0b00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-32 84 AddType(0x0c00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-32 LBA 85 AddType(0x0c01, "E3C9E316-0B5C-4DB8-817D-F92DF00215AE", "Microsoft reserved"); 86 AddType(0x0e00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // FAT-16 LBA 87 AddType(0x1100, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-12 88 AddType(0x1400, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-16 < 32M 89 AddType(0x1600, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-16 90 AddType(0x1700, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden NTFS (or HPFS) 91 AddType(0x1b00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-32 92 AddType(0x1c00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-32 LBA 93 AddType(0x1e00, "EBD0A0A2-B9E5-4433-87C0-68B6B72699C7", "Microsoft basic data", 0); // Hidden FAT-16 LBA 94 AddType(0x2700, "DE94BBA4-06D1-4D40-A16A-BFD50179D6AC", "Windows RE"); 95 96 // Open Network Install Environment (ONIE) specific types. 97 // See http://www.onie.org/ and 98 // https://github.com/onie/onie/blob/master/rootconf/x86_64/sysroot-lib-onie/onie-blkdev-common 99 AddType(0x3000, "7412F7D5-A156-4B13-81DC-867174929325", "ONIE boot"); 100 AddType(0x3001, "D4E6E2CD-4469-46F3-B5CB-1BFF57AFC149", "ONIE config"); 101 102 // PowerPC reference platform boot partition 103 AddType(0x4100, "9E1A2D38-C612-4316-AA26-8B49521E5A8B", "PowerPC PReP boot"); 104 105 // Windows LDM ("dynamic disk") types 106 AddType(0x4200, "AF9B60A0-1431-4F62-BC68-3311714A69AD", "Windows LDM data"); // Logical disk manager 107 AddType(0x4201, "5808C8AA-7E8F-42E0-85D2-E1E90434CFB3", "Windows LDM metadata"); // Logical disk manager 108 109 // An oddball IBM filesystem.... 110 AddType(0x7501, "37AFFC90-EF7D-4E96-91C3-2D7AE055B174", "IBM GPFS"); // General Parallel File System (GPFS) 111 112 // ChromeOS-specific partition types... 113 // Values taken from vboot_reference/firmware/lib/cgptlib/include/gpt.h in 114 // ChromeOS source code, retrieved 12/23/2010. They're also at 115 // http://www.chromium.org/chromium-os/chromiumos-design-docs/disk-format. 116 // These have no MBR equivalents, AFAIK, so I'm using 0x7Fxx values, since they're close 117 // to the Linux values. 118 AddType(0x7f00, "FE3A2A5D-4F32-41A7-B725-ACCC3285A309", "ChromeOS kernel"); 119 AddType(0x7f01, "3CB8E202-3B7E-47DD-8A3C-7FF2A13CFCEC", "ChromeOS root"); 120 AddType(0x7f02, "2E0A753D-9E48-43B0-8337-B15192CB1B5E", "ChromeOS reserved"); 121 122 // Linux-specific partition types.... 123 AddType(0x8200, "0657FD6D-A4AB-43C4-84E5-0933C84B4F4F", "Linux swap"); // Linux swap (or Solaris on MBR) 124 AddType(0x8300, "0FC63DAF-8483-4772-8E79-3D69D8477DE4", "Linux filesystem"); // Linux native 125 AddType(0x8301, "8DA63339-0007-60C0-C436-083AC8230908", "Linux reserved"); 126 // See http://www.freedesktop.org/software/systemd/man/systemd-gpt-auto-generator.html 127 // and http://www.freedesktop.org/wiki/Specifications/DiscoverablePartitionsSpec/ 128 AddType(0x8302, "933AC7E1-2EB4-4F13-B844-0E14E2AEF915", "Linux /home"); // Linux /home (auto-mounted by systemd) 129 AddType(0x8303, "44479540-F297-41B2-9AF7-D131D5F0458A", "Linux x86 root (/)"); // Linux / on x86 (auto-mounted by systemd) 130 AddType(0x8304, "4F68BCE3-E8CD-4DB1-96E7-FBCAF984B709", "Linux x86-64 root (/)"); // Linux / on x86-64 (auto-mounted by systemd) 131 AddType(0x8305, "B921B045-1DF0-41C3-AF44-4C6F280D3FAE", "Linux ARM64 root (/)"); // Linux / on 64-bit ARM (auto-mounted by systemd) 132 AddType(0x8306, "3B8F8425-20E0-4F3B-907F-1A25A76F98E8", "Linux /srv"); // Linux /srv (auto-mounted by systemd) 133 134 // Used by Intel Rapid Start technology 135 AddType(0x8400, "D3BFE2DE-3DAF-11DF-BA40-E3A556D89593", "Intel Rapid Start"); 136 137 // Another Linux type code.... 138 AddType(0x8e00, "E6D6D379-F507-44C2-A23C-238F2A3DF928", "Linux LVM"); 139 140 // FreeBSD partition types.... 141 // Note: Rather than extract FreeBSD disklabel data, convert FreeBSD 142 // partitions in-place, and let FreeBSD sort out the details.... 143 AddType(0xa500, "516E7CB4-6ECF-11D6-8FF8-00022D09712B", "FreeBSD disklabel"); 144 AddType(0xa501, "83BD6B9D-7F41-11DC-BE0B-001560B84F0F", "FreeBSD boot"); 145 AddType(0xa502, "516E7CB5-6ECF-11D6-8FF8-00022D09712B", "FreeBSD swap"); 146 AddType(0xa503, "516E7CB6-6ECF-11D6-8FF8-00022D09712B", "FreeBSD UFS"); 147 AddType(0xa504, "516E7CBA-6ECF-11D6-8FF8-00022D09712B", "FreeBSD ZFS"); 148 AddType(0xa505, "516E7CB8-6ECF-11D6-8FF8-00022D09712B", "FreeBSD Vinum/RAID"); 149 150 // Midnight BSD partition types.... 151 AddType(0xa580, "85D5E45A-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD data"); 152 AddType(0xa581, "85D5E45E-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD boot"); 153 AddType(0xa582, "85D5E45B-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD swap"); 154 AddType(0xa583, "0394Ef8B-237E-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD UFS"); 155 AddType(0xa584, "85D5E45D-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD ZFS"); 156 AddType(0xa585, "85D5E45C-237C-11E1-B4B3-E89A8F7FC3A7", "Midnight BSD Vinum"); 157 158 // A MacOS partition type, separated from others by NetBSD partition types... 159 AddType(0xa800, "55465300-0000-11AA-AA11-00306543ECAC", "Apple UFS"); // Mac OS X 160 161 // NetBSD partition types. Note that the main entry sets it up as a 162 // FreeBSD disklabel. I'm not 100% certain this is the correct behavior. 163 AddType(0xa900, "516E7CB4-6ECF-11D6-8FF8-00022D09712B", "FreeBSD disklabel", 0); // NetBSD disklabel 164 AddType(0xa901, "49F48D32-B10E-11DC-B99B-0019D1879648", "NetBSD swap"); 165 AddType(0xa902, "49F48D5A-B10E-11DC-B99B-0019D1879648", "NetBSD FFS"); 166 AddType(0xa903, "49F48D82-B10E-11DC-B99B-0019D1879648", "NetBSD LFS"); 167 AddType(0xa904, "2DB519C4-B10F-11DC-B99B-0019D1879648", "NetBSD concatenated"); 168 AddType(0xa905, "2DB519EC-B10F-11DC-B99B-0019D1879648", "NetBSD encrypted"); 169 AddType(0xa906, "49F48DAA-B10E-11DC-B99B-0019D1879648", "NetBSD RAID"); 170 171 // Mac OS partition types (See also 0xa800, above).... 172 AddType(0xab00, "426F6F74-0000-11AA-AA11-00306543ECAC", "Apple boot"); 173 AddType(0xaf00, "48465300-0000-11AA-AA11-00306543ECAC", "Apple HFS/HFS+"); 174 AddType(0xaf01, "52414944-0000-11AA-AA11-00306543ECAC", "Apple RAID"); 175 AddType(0xaf02, "52414944-5F4F-11AA-AA11-00306543ECAC", "Apple RAID offline"); 176 AddType(0xaf03, "4C616265-6C00-11AA-AA11-00306543ECAC", "Apple label"); 177 AddType(0xaf04, "5265636F-7665-11AA-AA11-00306543ECAC", "AppleTV recovery"); 178 AddType(0xaf05, "53746F72-6167-11AA-AA11-00306543ECAC", "Apple Core Storage"); 179 180 // Solaris partition types (one of which is shared with MacOS) 181 AddType(0xbe00, "6A82CB45-1DD2-11B2-99A6-080020736631", "Solaris boot"); 182 AddType(0xbf00, "6A85CF4D-1DD2-11B2-99A6-080020736631", "Solaris root"); 183 AddType(0xbf01, "6A898CC3-1DD2-11B2-99A6-080020736631", "Solaris /usr & Mac ZFS"); // Solaris/MacOS 184 AddType(0xbf02, "6A87C46F-1DD2-11B2-99A6-080020736631", "Solaris swap"); 185 AddType(0xbf03, "6A8B642B-1DD2-11B2-99A6-080020736631", "Solaris backup"); 186 AddType(0xbf04, "6A8EF2E9-1DD2-11B2-99A6-080020736631", "Solaris /var"); 187 AddType(0xbf05, "6A90BA39-1DD2-11B2-99A6-080020736631", "Solaris /home"); 188 AddType(0xbf06, "6A9283A5-1DD2-11B2-99A6-080020736631", "Solaris alternate sector"); 189 AddType(0xbf07, "6A945A3B-1DD2-11B2-99A6-080020736631", "Solaris Reserved 1"); 190 AddType(0xbf08, "6A9630D1-1DD2-11B2-99A6-080020736631", "Solaris Reserved 2"); 191 AddType(0xbf09, "6A980767-1DD2-11B2-99A6-080020736631", "Solaris Reserved 3"); 192 AddType(0xbf0a, "6A96237F-1DD2-11B2-99A6-080020736631", "Solaris Reserved 4"); 193 AddType(0xbf0b, "6A8D2AC7-1DD2-11B2-99A6-080020736631", "Solaris Reserved 5"); 194 195 // I can find no MBR equivalents for these, but they're on the 196 // Wikipedia page for GPT, so here we go.... 197 AddType(0xc001, "75894C1E-3AEB-11D3-B7C1-7B03A0000000", "HP-UX data"); 198 AddType(0xc002, "E2A1E728-32E3-11D6-A682-7B03A0000000", "HP-UX service"); 199 200 // See http://www.freedesktop.org/wiki/Specifications/BootLoaderSpec 201 AddType(0xea00, "BC13C2FF-59E6-4262-A352-B275FD6F7172", "Freedesktop $BOOT"); 202 203 // Type code for Haiku; uses BeOS MBR code as hex code base 204 AddType(0xeb00, "42465331-3BA3-10F1-802A-4861696B7521", "Haiku BFS"); 205 206 // Manufacturer-specific ESP-like partitions (in order in which they were added) 207 AddType(0xed00, "F4019732-066E-4E12-8273-346C5641494F", "Sony system partition"); 208 AddType(0xed01, "BFBFAFE7-A34F-448A-9A5B-6213EB736C22", "Lenovo system partition"); 209 210 // EFI system and related partitions 211 AddType(0xef00, "C12A7328-F81F-11D2-BA4B-00A0C93EC93B", "EFI System"); // Parted identifies these as having the "boot flag" set 212 AddType(0xef01, "024DEE41-33E7-11D3-9D69-0008C781F39F", "MBR partition scheme"); // Used to nest MBR in GPT 213 AddType(0xef02, "21686148-6449-6E6F-744E-656564454649", "BIOS boot partition"); // Used by GRUB 214 215 // Ceph type codes; see https://github.com/ceph/ceph/blob/9bcc42a3e6b08521694b5c0228b2c6ed7b3d312e/src/ceph-disk#L76-L81 216 AddType(0xf800, "4FBD7E29-9D25-41B8-AFD0-062C0CEFF05D", "Ceph OSD"); // Ceph Object Storage Daemon 217 AddType(0xf801, "4FBD7E29-9D25-41B8-AFD0-5EC00CEFF05D", "Ceph dm-crypt OSD"); // Ceph Object Storage Daemon (encrypted) 218 AddType(0xf802, "BFBFAFE7-A34F-448A-9A5B-6213EB736C22", "Ceph journal"); 219 AddType(0xf803, "45B0969E-9B03-4F30-B4C6-5EC00CEFF106", "Ceph dm-crypt journal"); 220 AddType(0xf804, "89C57F98-2FE5-4DC0-89C1-F3AD0CEFF2BE", "Ceph disk in creation"); 221 AddType(0xf805, "89C57F98-2FE5-4DC0-89C1-5EC00CEFF2BE", "Ceph dm-crypt disk in creation"); 222 223 // VMWare ESX partition types codes 224 AddType(0xfb00, "AA31E02A-400F-11DB-9590-000C2911D1B8", "VMWare VMFS"); 225 AddType(0xfb01, "9198EFFC-31C0-11DB-8F78-000C2911D1B8", "VMWare reserved"); 226 AddType(0xfc00, "9D275380-40AD-11DB-BF97-000C2911D1B8", "VMWare kcore crash protection"); 227 228 // A straggler Linux partition type.... 229 AddType(0xfd00, "A19D880F-05FC-4D3B-A006-743F0F84911E", "Linux RAID"); 230 231 // Note: DO NOT use the 0xffff code; that's reserved to indicate an 232 // unknown GUID type code. 233 } // PartType::AddAllTypes() 234 235 // Add a single type to the linked list of types. Returns 1 if operation 236 // succeeds, 0 otherwise. 237 int PartType::AddType(uint16_t mbrType, const char * guidData, const char * name, 238 int toDisplay) { 239 AType* tempType; 240 int allOK = 1; 241 242 tempType = new AType; 243 if (tempType != NULL) { 244 tempType->MBRType = mbrType; 245 tempType->GUIDType = guidData; 246 tempType->name = name; 247 tempType->display = toDisplay; 248 tempType->next = NULL; 249 if (allTypes == NULL) { // first entry 250 allTypes = tempType; 251 } else { 252 lastType->next = tempType; 253 } // if/else 254 lastType = tempType; 255 } else { 256 cerr << "Unable to allocate memory in PartType::AddType()! Partition type list will\n"; 257 cerr << "be incomplete!\n"; 258 allOK = 0; 259 } // if/else 260 return allOK; 261 } // GUID::AddType(const char* variant) 262 263 // Assignment operator by string. If the original string is short, 264 // interpret it as a gdisk hex code; if it's longer, interpret it as 265 // a direct entry of a GUID value. If a short string isn't a hex 266 // number, do nothing. 267 PartType & PartType::operator=(const string & orig) { 268 uint32_t hexCode; 269 270 if (orig.length() < 32) { 271 if (IsHex(orig)) { 272 sscanf(orig.c_str(), "%x", &hexCode); 273 *this = hexCode; 274 } 275 } else { 276 GUIDData::operator=(orig); 277 } // if/else hexCode or GUID 278 return *this; 279 } // PartType::operator=(const char * orig) 280 281 // Assignment from C-style string; rely on C++ casting.... 282 PartType & PartType::operator=(const char * orig) { 283 return operator=((string) orig); 284 } // PartType::operator=(const char * orig) 285 286 // Assign a GUID based on my custom 2-byte (16-bit) MBR hex ID variant 287 PartType & PartType::operator=(uint16_t ID) { 288 AType* theItem = allTypes; 289 int found = 0; 290 291 // Now search the type list for a match to the ID.... 292 while ((theItem != NULL) && (!found)) { 293 if (theItem->MBRType == ID) { 294 GUIDData::operator=(theItem->GUIDType); 295 found = 1; 296 } else { 297 theItem = theItem->next; 298 } // if/else 299 } // while 300 if (!found) { 301 // Assign a default value.... 302 operator=(DEFAULT_GPT_TYPE); 303 cout.setf(ios::uppercase); 304 cout.fill('0'); 305 cout << "Exact type match not found for type code "; 306 cout.width(4); 307 cout << hex << ID << "; assigning type code for\n'" << TypeName() << "'\n" << dec; 308 cout.fill(' '); 309 } // if (!found) 310 return *this; 311 } // PartType::operator=(uint16_t ID) 312 313 // Return the English description of the partition type (e.g., "Linux filesystem") 314 string PartType::TypeName(void) const { 315 AType* theItem = allTypes; 316 int found = 0; 317 string typeName; 318 319 while ((theItem != NULL) && (!found)) { 320 if (theItem->GUIDType == *this) { // found it! 321 typeName = theItem->name; 322 found = 1; 323 } else { 324 theItem = theItem->next; 325 } // if/else 326 } // while 327 if (!found) { 328 typeName = "Unknown"; 329 } // if (!found) 330 return typeName; 331 } // PartType::TypeName() 332 333 #ifdef USE_UTF16 334 // Return the Unicode description of the partition type (e.g., "Linux filesystem") 335 UnicodeString PartType::UTypeName(void) const { 336 AType* theItem = allTypes; 337 int found = 0; 338 UnicodeString typeName; 339 340 while ((theItem != NULL) && (!found)) { 341 if (theItem->GUIDType == *this) { // found it! 342 typeName = theItem->name.c_str(); 343 found = 1; 344 } else { 345 theItem = theItem->next; 346 } // if/else 347 } // while 348 if (!found) { 349 typeName = "Unknown"; 350 } // if (!found) 351 return typeName; 352 } // PartType::TypeName() 353 #endif 354 355 // Return the custom GPT fdisk 2-byte (16-bit) hex code for this GUID partition type 356 // Note that this function ignores entries for which the display variable 357 // is set to 0. This enables control of which values get returned when 358 // there are multiple possibilities, but opens the algorithm up to the 359 // potential for problems should the data in the list be bad. 360 uint16_t PartType::GetHexType() const { 361 AType* theItem = allTypes; 362 int found = 0; 363 uint16_t theID = 0xFFFF; 364 365 while ((theItem != NULL) && (!found)) { 366 if ((theItem->GUIDType == *this) && (theItem->display == 1)) { // found it! 367 theID = theItem->MBRType; 368 found = 1; 369 } else { 370 theItem = theItem->next; 371 } // if/else 372 } // while 373 if (!found) { 374 theID = 0xFFFF; 375 } // if (!found) 376 return theID; 377 } // PartType::GetHex() 378 379 // Displays the available types and my extended MBR codes for same.... 380 // Note: This function assumes an 80-column display. On wider displays, 381 // it stops at under 80 columns; on narrower displays, lines will wrap 382 // in an ugly way. The maxLines value is the maximum number of lines 383 // to display before prompting to continue, or 0 (or a negative value) 384 // for no limit. 385 void PartType::ShowAllTypes(int maxLines) const { 386 int colCount = 1, lineCount = 1; 387 size_t i; 388 AType* thisType = allTypes; 389 string line; 390 391 cout.unsetf(ios::uppercase); 392 while (thisType != NULL) { 393 if (thisType->display == 1) { // show it 394 cout.fill('0'); 395 cout.width(4); 396 cout << hex << thisType->MBRType << " "; 397 cout << thisType->name.substr(0, 20); 398 for (i = 0; i < (20 - (thisType->name.substr(0, 20).length())); i++) 399 cout << " "; 400 if ((colCount % 3) == 0) { 401 if (thisType->next) { 402 cout << "\n"; 403 if ((maxLines > 0) && (lineCount++ % maxLines) == 0) { 404 cout << "Press the <Enter> key to see more codes: "; 405 getline(cin, line); 406 } // if reached screen line limit 407 } // if there's another entry following this one 408 } else { 409 cout << " "; 410 } 411 colCount++; 412 } // if 413 thisType = thisType->next; 414 } // while 415 cout.fill(' '); 416 cout << "\n" << dec; 417 } // PartType::ShowAllTypes(int maxLines) 418 419 // Returns 1 if code is a valid extended MBR code, 0 if it's not 420 int PartType::Valid(uint16_t code) const { 421 AType* thisType = allTypes; 422 int found = 0; 423 424 while ((thisType != NULL) && (!found)) { 425 if (thisType->MBRType == code) { 426 found = 1; 427 } // if 428 thisType = thisType->next; 429 } // while 430 return found; 431 } // PartType::Valid() 432