| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | /*-
 | 
					
						
							|  |  |  |  *   BSD LICENSE | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Copyright (c) Intel Corporation. | 
					
						
							|  |  |  |  *   All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  *   modification, are permitted provided that the following conditions | 
					
						
							|  |  |  |  *   are met: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     * Redistributions of source code must retain the above copyright | 
					
						
							|  |  |  |  *       notice, this list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  *     * Redistributions in binary form must reproduce the above copyright | 
					
						
							|  |  |  |  *       notice, this list of conditions and the following disclaimer in | 
					
						
							|  |  |  |  *       the documentation and/or other materials provided with the | 
					
						
							|  |  |  |  *       distribution. | 
					
						
							|  |  |  |  *     * Neither the name of Intel Corporation nor the names of its | 
					
						
							|  |  |  |  *       contributors may be used to endorse or promote products derived | 
					
						
							|  |  |  |  *       from this software without specific prior written permission. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
					
						
							|  |  |  |  *   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
					
						
							|  |  |  |  *   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | 
					
						
							|  |  |  |  *   A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | 
					
						
							|  |  |  |  *   OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | 
					
						
							|  |  |  |  *   SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | 
					
						
							|  |  |  |  *   LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
					
						
							|  |  |  |  *   DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
					
						
							|  |  |  |  *   THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
					
						
							|  |  |  |  *   (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  *   OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "gpt.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 21:44:43 +00:00
										 |  |  | #include "spdk/crc32.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | #include "spdk/endian.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-21 21:44:43 +00:00
										 |  |  | #include "spdk/event.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-21 19:49:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-06 16:16:26 +00:00
										 |  |  | #include "spdk/log.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define GPT_PRIMARY_PARTITION_TABLE_LBA 0x1
 | 
					
						
							|  |  |  | #define PRIMARY_PARTITION_NUMBER 4
 | 
					
						
							|  |  |  | #define GPT_PROTECTIVE_MBR 1
 | 
					
						
							|  |  |  | #define SPDK_MAX_NUM_PARTITION_ENTRIES 128
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | static uint64_t | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_get_expected_head_lba(struct spdk_gpt *gpt) | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	switch (gpt->parse_phase) { | 
					
						
							|  |  |  | 	case SPDK_GPT_PARSE_PHASE_PRIMARY: | 
					
						
							|  |  |  | 		return GPT_PRIMARY_PARTITION_TABLE_LBA; | 
					
						
							|  |  |  | 	case SPDK_GPT_PARSE_PHASE_SECONDARY: | 
					
						
							|  |  |  | 		return gpt->lba_end; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		assert(false); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct spdk_gpt_header * | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_get_header_buf(struct spdk_gpt *gpt) | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	switch (gpt->parse_phase) { | 
					
						
							|  |  |  | 	case SPDK_GPT_PARSE_PHASE_PRIMARY: | 
					
						
							|  |  |  | 		return (struct spdk_gpt_header *) | 
					
						
							|  |  |  | 		       (gpt->buf + GPT_PRIMARY_PARTITION_TABLE_LBA * gpt->sector_size); | 
					
						
							|  |  |  | 	case SPDK_GPT_PARSE_PHASE_SECONDARY: | 
					
						
							|  |  |  | 		return (struct spdk_gpt_header *) | 
					
						
							|  |  |  | 		       (gpt->buf + (gpt->buf_size - gpt->sector_size)); | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		assert(false); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct spdk_gpt_partition_entry * | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_get_partitions_buf(struct spdk_gpt *gpt, uint64_t total_partition_size, | 
					
						
							|  |  |  | 		       uint64_t partition_start_lba) | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	uint64_t secondary_total_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (gpt->parse_phase) { | 
					
						
							|  |  |  | 	case SPDK_GPT_PARSE_PHASE_PRIMARY: | 
					
						
							|  |  |  | 		if ((total_partition_size + partition_start_lba * gpt->sector_size) > | 
					
						
							|  |  |  | 		    gpt->buf_size) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("Buffer size is not enough\n"); | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return (struct spdk_gpt_partition_entry *) | 
					
						
							|  |  |  | 		       (gpt->buf + partition_start_lba * gpt->sector_size); | 
					
						
							|  |  |  | 	case SPDK_GPT_PARSE_PHASE_SECONDARY: | 
					
						
							|  |  |  | 		secondary_total_size = (gpt->lba_end - partition_start_lba + 1) * gpt->sector_size; | 
					
						
							|  |  |  | 		if (secondary_total_size > gpt->buf_size) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("Buffer size is not enough\n"); | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return (struct spdk_gpt_partition_entry *) | 
					
						
							|  |  |  | 		       (gpt->buf + (gpt->buf_size - secondary_total_size)); | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		assert(false); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_read_partitions(struct spdk_gpt *gpt) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	uint32_t total_partition_size, num_partition_entries, partition_entry_size; | 
					
						
							|  |  |  | 	uint64_t partition_start_lba; | 
					
						
							|  |  |  | 	struct spdk_gpt_header *head = gpt->header; | 
					
						
							|  |  |  | 	uint32_t crc32; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	num_partition_entries = from_le32(&head->num_partition_entries); | 
					
						
							|  |  |  | 	if (num_partition_entries > SPDK_MAX_NUM_PARTITION_ENTRIES) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Num_partition_entries=%u which exceeds max=%u\n", | 
					
						
							|  |  |  | 			    num_partition_entries, SPDK_MAX_NUM_PARTITION_ENTRIES); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	partition_entry_size = from_le32(&head->size_of_partition_entry); | 
					
						
							|  |  |  | 	if (partition_entry_size != sizeof(struct spdk_gpt_partition_entry)) { | 
					
						
							| 
									
										
										
										
											2020-11-17 16:29:21 +00:00
										 |  |  | 		SPDK_ERRLOG("Partition_entry_size(%x) != expected(%zx)\n", | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 			    partition_entry_size, sizeof(struct spdk_gpt_partition_entry)); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	total_partition_size = num_partition_entries * partition_entry_size; | 
					
						
							|  |  |  | 	partition_start_lba = from_le64(&head->partition_entry_lba); | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | 	gpt->partitions = gpt_get_partitions_buf(gpt, total_partition_size, | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 			  partition_start_lba); | 
					
						
							|  |  |  | 	if (!gpt->partitions) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Failed to get gpt partitions buf\n"); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 21:44:43 +00:00
										 |  |  | 	crc32 = spdk_crc32_ieee_update(gpt->partitions, total_partition_size, ~0); | 
					
						
							|  |  |  | 	crc32 ^= ~0; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (crc32 != from_le32(&head->partition_entry_array_crc32)) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("GPT partition entry array crc32 did not match\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_lba_range_check(struct spdk_gpt_header *head, uint64_t lba_end) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	uint64_t usable_lba_start, usable_lba_end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	usable_lba_start = from_le64(&head->first_usable_lba); | 
					
						
							|  |  |  | 	usable_lba_end = from_le64(&head->last_usable_lba); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (usable_lba_end < usable_lba_start) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Head's usable_lba_end(%" PRIu64 ") < usable_lba_start(%" PRIu64 ")\n", | 
					
						
							|  |  |  | 			    usable_lba_end, usable_lba_start); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (usable_lba_end > lba_end) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Head's usable_lba_end(%" PRIu64 ") > lba_end(%" PRIu64 ")\n", | 
					
						
							|  |  |  | 			    usable_lba_end, lba_end); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((usable_lba_start < GPT_PRIMARY_PARTITION_TABLE_LBA) && | 
					
						
							|  |  |  | 	    (GPT_PRIMARY_PARTITION_TABLE_LBA < usable_lba_end)) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Head lba is not in the usable range\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_read_header(struct spdk_gpt *gpt) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	uint32_t head_size; | 
					
						
							|  |  |  | 	uint32_t new_crc, original_crc; | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 	uint64_t my_lba, head_lba; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	struct spdk_gpt_header *head; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | 	head = gpt_get_header_buf(gpt); | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 	if (!head) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Failed to get gpt header buf\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	head_size = from_le32(&head->header_size); | 
					
						
							|  |  |  | 	if (head_size < sizeof(*head) || head_size > gpt->sector_size) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("head_size=%u\n", head_size); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	original_crc = from_le32(&head->header_crc32); | 
					
						
							|  |  |  | 	head->header_crc32 = 0; | 
					
						
							| 
									
										
										
										
											2017-07-21 21:44:43 +00:00
										 |  |  | 	new_crc = spdk_crc32_ieee_update(head, from_le32(&head->header_size), ~0); | 
					
						
							|  |  |  | 	new_crc ^= ~0; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	/* restore header crc32 */ | 
					
						
							|  |  |  | 	to_le32(&head->header_crc32, original_crc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (new_crc != original_crc) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("head crc32 does not match, provided=%u, caculated=%u\n", | 
					
						
							|  |  |  | 			    original_crc, new_crc); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (memcmp(SPDK_GPT_SIGNATURE, head->gpt_signature, | 
					
						
							|  |  |  | 		   sizeof(head->gpt_signature))) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("signature did not match\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | 	head_lba = gpt_get_expected_head_lba(gpt); | 
					
						
							| 
									
										
										
										
											2019-01-09 15:32:35 +00:00
										 |  |  | 	my_lba = from_le64(&head->my_lba); | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 	if (my_lba != head_lba) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("head my_lba(%" PRIu64 ") != expected(%" PRIu64 ")\n", | 
					
						
							|  |  |  | 			    my_lba, head_lba); | 
					
						
							| 
									
										
										
										
											2019-01-09 15:32:35 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | 	if (gpt_lba_range_check(head, gpt->lba_end)) { | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		SPDK_ERRLOG("lba range check error\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gpt->header = head; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_check_mbr(struct spdk_gpt *gpt) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i, primary_partition = 0; | 
					
						
							|  |  |  | 	uint32_t total_lba_size = 0, ret = 0, expected_start_lba; | 
					
						
							|  |  |  | 	struct spdk_mbr *mbr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mbr = (struct spdk_mbr *)gpt->buf; | 
					
						
							|  |  |  | 	if (from_le16(&mbr->mbr_signature) != SPDK_MBR_SIGNATURE) { | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 		SPDK_DEBUGLOG(gpt_parse, "Signature mismatch, provided=%x," | 
					
						
							| 
									
										
										
										
											2017-07-07 01:38:22 +00:00
										 |  |  | 			      "expected=%x\n", from_le16(&mbr->disk_signature), | 
					
						
							|  |  |  | 			      SPDK_MBR_SIGNATURE); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < PRIMARY_PARTITION_NUMBER; i++) { | 
					
						
							|  |  |  | 		if (mbr->partitions[i].os_type == SPDK_MBR_OS_TYPE_GPT_PROTECTIVE) { | 
					
						
							|  |  |  | 			primary_partition = i; | 
					
						
							|  |  |  | 			ret = GPT_PROTECTIVE_MBR; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ret == GPT_PROTECTIVE_MBR) { | 
					
						
							| 
									
										
										
										
											2017-07-21 20:02:57 +00:00
										 |  |  | 		expected_start_lba = GPT_PRIMARY_PARTITION_TABLE_LBA; | 
					
						
							|  |  |  | 		if (from_le32(&mbr->partitions[primary_partition].start_lba) != expected_start_lba) { | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(gpt_parse, "start lba mismatch, provided=%u, expected=%u\n", | 
					
						
							| 
									
										
										
										
											2017-07-21 20:02:57 +00:00
										 |  |  | 				      from_le32(&mbr->partitions[primary_partition].start_lba), | 
					
						
							|  |  |  | 				      expected_start_lba); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		total_lba_size = from_le32(&mbr->partitions[primary_partition].size_lba); | 
					
						
							|  |  |  | 		if ((total_lba_size != ((uint32_t) gpt->total_sectors - 1)) && | 
					
						
							|  |  |  | 		    (total_lba_size != 0xFFFFFFFF)) { | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 			SPDK_DEBUGLOG(gpt_parse, | 
					
						
							| 
									
										
										
										
											2018-11-05 08:12:35 +00:00
										 |  |  | 				      "GPT Primary MBR size does not equal: (record_size %u != actual_size %u)!\n", | 
					
						
							|  |  |  | 				      total_lba_size, (uint32_t) gpt->total_sectors - 1); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 		SPDK_DEBUGLOG(gpt_parse, "Currently only support GPT Protective MBR format\n"); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_parse_mbr(struct spdk_gpt *gpt) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!gpt || !gpt->buf) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Gpt and the related buffer should not be NULL\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | 	rc = gpt_check_mbr(gpt); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | 		SPDK_DEBUGLOG(gpt_parse, "Failed to detect gpt in MBR\n"); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | gpt_parse_partition_table(struct spdk_gpt *gpt) | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | 	rc = gpt_read_header(gpt); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Failed to read gpt header\n"); | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-10 07:22:54 +00:00
										 |  |  | 	rc = gpt_read_partitions(gpt); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Failed to read gpt partitions\n"); | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-07-07 01:38:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 11:27:29 +00:00
										 |  |  | SPDK_LOG_REGISTER_COMPONENT(gpt_parse) |