| 
									
										
										
										
											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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * This driver reads a GPT partition table from a bdev and exposes a virtual block device for | 
					
						
							|  |  |  |  * each partition. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 19:49:09 +00:00
										 |  |  | #include "gpt.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "spdk/conf.h"
 | 
					
						
							|  |  |  | #include "spdk/endian.h"
 | 
					
						
							|  |  |  | #include "spdk/env.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-11 20:32:15 +00:00
										 |  |  | #include "spdk/thread.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-21 19:49:09 +00:00
										 |  |  | #include "spdk/rpc.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | #include "spdk/string.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-07 22:47:02 +00:00
										 |  |  | #include "spdk/util.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-23 21:01:03 +00:00
										 |  |  | #include "spdk/bdev_module.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | #include "spdk_internal/log.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-06 18:52:46 +00:00
										 |  |  | static int vbdev_gpt_init(void); | 
					
						
							|  |  |  | static void vbdev_gpt_examine(struct spdk_bdev *bdev); | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | static int vbdev_gpt_get_ctx_size(void); | 
					
						
							| 
									
										
										
										
											2018-03-06 18:52:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-09 22:20:21 +00:00
										 |  |  | static struct spdk_bdev_module gpt_if = { | 
					
						
							| 
									
										
										
										
											2018-03-06 18:52:46 +00:00
										 |  |  | 	.name = "gpt", | 
					
						
							|  |  |  | 	.module_init = vbdev_gpt_init, | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | 	.get_ctx_size = vbdev_gpt_get_ctx_size, | 
					
						
							| 
									
										
										
										
											2018-06-26 12:07:11 +00:00
										 |  |  | 	.examine_disk = vbdev_gpt_examine, | 
					
						
							| 
									
										
										
										
											2018-03-06 18:52:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-02-05 10:46:48 +00:00
										 |  |  | SPDK_BDEV_MODULE_REGISTER(gpt, &gpt_if) | 
					
						
							| 
									
										
										
										
											2017-07-07 00:36:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | /* Base block device gpt context */ | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | struct gpt_base { | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 	struct spdk_gpt			gpt; | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	struct spdk_bdev_part_base	*part_base; | 
					
						
							| 
									
										
										
										
											2018-11-26 13:13:21 +00:00
										 |  |  | 	SPDK_BDEV_PART_TAILQ		parts; | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* This channel is only used for reading the partition table. */ | 
					
						
							|  |  |  | 	struct spdk_io_channel		*ch; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Context for each gpt virtual bdev */ | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | struct gpt_disk { | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 	struct spdk_bdev_part	part; | 
					
						
							| 
									
										
										
										
											2017-07-07 02:56:26 +00:00
										 |  |  | 	uint32_t		partition_index; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | struct gpt_channel { | 
					
						
							|  |  |  | 	struct spdk_bdev_part_channel	part_ch; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | struct gpt_io { | 
					
						
							|  |  |  | 	struct spdk_io_channel *ch; | 
					
						
							|  |  |  | 	struct spdk_bdev_io *bdev_io; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* for bdev_io_wait */ | 
					
						
							|  |  |  | 	struct spdk_bdev_io_wait_entry bdev_io_wait; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | static bool g_gpt_disabled; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | spdk_gpt_base_free(void *ctx) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	struct gpt_base *gpt_base = ctx; | 
					
						
							| 
									
										
										
										
											2017-09-14 18:34:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 08:46:35 +00:00
										 |  |  | 	spdk_free(gpt_base->gpt.buf); | 
					
						
							| 
									
										
										
										
											2017-09-14 18:34:36 +00:00
										 |  |  | 	free(gpt_base); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-17 19:35:48 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2018-11-26 12:45:53 +00:00
										 |  |  | spdk_gpt_base_bdev_hotremove_cb(void *_part_base) | 
					
						
							| 
									
										
										
										
											2017-07-17 19:35:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-26 12:45:53 +00:00
										 |  |  | 	struct spdk_bdev_part_base *part_base = _part_base; | 
					
						
							| 
									
										
										
										
											2018-11-26 13:13:21 +00:00
										 |  |  | 	struct gpt_base *gpt_base = spdk_bdev_part_base_get_ctx(part_base); | 
					
						
							| 
									
										
										
										
											2018-11-26 12:45:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-26 13:13:21 +00:00
										 |  |  | 	spdk_bdev_part_base_hotremove(part_base, &gpt_base->parts); | 
					
						
							| 
									
										
										
										
											2017-07-17 19:35:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | static int vbdev_gpt_destruct(void *ctx); | 
					
						
							|  |  |  | static void vbdev_gpt_submit_request(struct spdk_io_channel *_ch, struct spdk_bdev_io *bdev_io); | 
					
						
							| 
									
										
										
										
											2018-02-22 12:48:13 +00:00
										 |  |  | static int vbdev_gpt_dump_info_json(void *ctx, struct spdk_json_write_ctx *w); | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct spdk_bdev_fn_table vbdev_gpt_fn_table = { | 
					
						
							|  |  |  | 	.destruct		= vbdev_gpt_destruct, | 
					
						
							|  |  |  | 	.submit_request		= vbdev_gpt_submit_request, | 
					
						
							| 
									
										
										
										
											2018-02-22 12:48:13 +00:00
										 |  |  | 	.dump_info_json		= vbdev_gpt_dump_info_json, | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | static struct gpt_base * | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | spdk_gpt_base_bdev_init(struct spdk_bdev *bdev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	struct gpt_base *gpt_base; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	struct spdk_gpt *gpt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	gpt_base = calloc(1, sizeof(*gpt_base)); | 
					
						
							|  |  |  | 	if (!gpt_base) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Cannot alloc memory for gpt_base pointer\n"); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-26 13:13:21 +00:00
										 |  |  | 	TAILQ_INIT(&gpt_base->parts); | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	gpt_base->part_base = spdk_bdev_part_base_construct(bdev, | 
					
						
							|  |  |  | 			      spdk_gpt_base_bdev_hotremove_cb, | 
					
						
							|  |  |  | 			      &gpt_if, &vbdev_gpt_fn_table, | 
					
						
							| 
									
										
										
										
											2018-11-26 13:13:21 +00:00
										 |  |  | 			      &gpt_base->parts, spdk_gpt_base_free, gpt_base, | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 			      sizeof(struct gpt_channel), NULL, NULL); | 
					
						
							|  |  |  | 	if (!gpt_base->part_base) { | 
					
						
							|  |  |  | 		free(gpt_base); | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 		SPDK_ERRLOG("cannot construct gpt_base"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	gpt = &gpt_base->gpt; | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 	gpt->parse_phase = SPDK_GPT_PARSE_PHASE_PRIMARY; | 
					
						
							| 
									
										
										
										
											2017-10-23 18:59:27 +00:00
										 |  |  | 	gpt->buf_size = spdk_max(SPDK_GPT_BUFFER_SIZE, bdev->blocklen); | 
					
						
							| 
									
										
										
										
											2019-04-05 08:46:35 +00:00
										 |  |  | 	gpt->buf = spdk_zmalloc(gpt->buf_size, spdk_bdev_get_buf_align(bdev), NULL, | 
					
						
							|  |  |  | 				SPDK_ENV_LCORE_ID_ANY, SPDK_MALLOC_DMA); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	if (!gpt->buf) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Cannot alloc buf\n"); | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 		spdk_bdev_part_base_free(gpt_base->part_base); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gpt->sector_size = bdev->blocklen; | 
					
						
							|  |  |  | 	gpt->total_sectors = bdev->blockcnt; | 
					
						
							|  |  |  | 	gpt->lba_start = 0; | 
					
						
							|  |  |  | 	gpt->lba_end = gpt->total_sectors - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	return gpt_base; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | vbdev_gpt_destruct(void *ctx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	struct gpt_disk *gpt_disk = ctx; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-16 21:45:53 +00:00
										 |  |  | 	return spdk_bdev_part_free(&gpt_disk->part); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-11 06:03:41 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | _vbdev_gpt_submit_request(struct spdk_io_channel *_ch, struct spdk_bdev_io *bdev_io); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | vbdev_gpt_resubmit_request(void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct gpt_io *io = (struct gpt_io *)arg; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-11 06:03:41 +00:00
										 |  |  | 	_vbdev_gpt_submit_request(io->ch, io->bdev_io); | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | vbdev_gpt_queue_io(struct gpt_io *io) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	io->bdev_io_wait.bdev = io->bdev_io->bdev; | 
					
						
							|  |  |  | 	io->bdev_io_wait.cb_fn = vbdev_gpt_resubmit_request; | 
					
						
							|  |  |  | 	io->bdev_io_wait.cb_arg = io; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = spdk_bdev_queue_io_wait(io->bdev_io->bdev, | 
					
						
							|  |  |  | 				     io->ch, &io->bdev_io_wait); | 
					
						
							|  |  |  | 	if (rc != 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Queue io failed in vbdev_gpt_queue_io, rc=%d.\n", rc); | 
					
						
							|  |  |  | 		spdk_bdev_io_complete(io->bdev_io, SPDK_BDEV_IO_STATUS_FAILED); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-07-11 06:03:41 +00:00
										 |  |  | vbdev_gpt_get_buf_cb(struct spdk_io_channel *ch, struct spdk_bdev_io *bdev_io, bool success) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!success) { | 
					
						
							|  |  |  | 		spdk_bdev_io_complete(bdev_io, SPDK_BDEV_IO_STATUS_FAILED); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_vbdev_gpt_submit_request(ch, bdev_io); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | _vbdev_gpt_submit_request(struct spdk_io_channel *_ch, struct spdk_bdev_io *bdev_io) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 	struct gpt_channel *ch = spdk_io_channel_get_ctx(_ch); | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | 	struct gpt_io *io = (struct gpt_io *)bdev_io->driver_ctx; | 
					
						
							| 
									
										
										
										
											2018-09-25 03:26:06 +00:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-25 03:26:06 +00:00
										 |  |  | 	rc = spdk_bdev_part_submit_request(&ch->part_ch, bdev_io); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | 		if (rc == -ENOMEM) { | 
					
						
							|  |  |  | 			SPDK_DEBUGLOG(SPDK_LOG_VBDEV_GPT, "gpt: no memory, queue io\n"); | 
					
						
							|  |  |  | 			io->ch = _ch; | 
					
						
							|  |  |  | 			io->bdev_io = bdev_io; | 
					
						
							|  |  |  | 			vbdev_gpt_queue_io(io); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("gpt: error on bdev_io submission, rc=%d.\n", rc); | 
					
						
							|  |  |  | 			spdk_bdev_io_complete(bdev_io, SPDK_BDEV_IO_STATUS_FAILED); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-25 03:26:06 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-11 06:03:41 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | vbdev_gpt_submit_request(struct spdk_io_channel *_ch, struct spdk_bdev_io *bdev_io) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (bdev_io->type) { | 
					
						
							|  |  |  | 	case SPDK_BDEV_IO_TYPE_READ: | 
					
						
							|  |  |  | 		spdk_bdev_io_get_buf(bdev_io, vbdev_gpt_get_buf_cb, | 
					
						
							|  |  |  | 				     bdev_io->u.bdev.num_blocks * bdev_io->bdev->blocklen); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		_vbdev_gpt_submit_request(_ch, bdev_io); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 18:19:58 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | write_guid(struct spdk_json_write_ctx *w, const struct spdk_gpt_guid *guid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	spdk_json_write_string_fmt(w, "%08x-%04x-%04x-%04x-%04x%08x", | 
					
						
							|  |  |  | 				   from_le32(&guid->raw[0]), | 
					
						
							|  |  |  | 				   from_le16(&guid->raw[4]), | 
					
						
							|  |  |  | 				   from_le16(&guid->raw[6]), | 
					
						
							|  |  |  | 				   from_be16(&guid->raw[8]), | 
					
						
							|  |  |  | 				   from_be16(&guid->raw[10]), | 
					
						
							|  |  |  | 				   from_be32(&guid->raw[12])); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 22:47:02 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | write_string_utf16le(struct spdk_json_write_ctx *w, const uint16_t *str, size_t max_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	size_t len; | 
					
						
							|  |  |  | 	const uint16_t *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (len = 0, p = str; len < max_len && *p; p++) { | 
					
						
							|  |  |  | 		len++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_json_write_string_utf16le_raw(w, str, len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2018-02-22 12:48:13 +00:00
										 |  |  | vbdev_gpt_dump_info_json(void *ctx, struct spdk_json_write_ctx *w) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	struct gpt_disk *gpt_disk = SPDK_CONTAINEROF(ctx, struct gpt_disk, part); | 
					
						
							| 
									
										
										
										
											2018-06-28 17:37:00 +00:00
										 |  |  | 	struct spdk_bdev_part_base *base_bdev = spdk_bdev_part_get_base(&gpt_disk->part); | 
					
						
							|  |  |  | 	struct gpt_base *gpt_base = spdk_bdev_part_base_get_ctx(base_bdev); | 
					
						
							|  |  |  | 	struct spdk_bdev *part_base_bdev = spdk_bdev_part_base_get_bdev(base_bdev); | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 	struct spdk_gpt *gpt = &gpt_base->gpt; | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	struct spdk_gpt_partition_entry *gpt_entry = &gpt->partitions[gpt_disk->partition_index]; | 
					
						
							| 
									
										
										
										
											2018-06-28 17:37:00 +00:00
										 |  |  | 	uint64_t offset_blocks = spdk_bdev_part_get_offset_blocks(&gpt_disk->part); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:34:45 +00:00
										 |  |  | 	spdk_json_write_named_object_begin(w, "gpt"); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:34:45 +00:00
										 |  |  | 	spdk_json_write_named_string(w, "base_bdev", spdk_bdev_get_name(part_base_bdev)); | 
					
						
							| 
									
										
										
										
											2017-07-07 18:19:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-01 05:34:45 +00:00
										 |  |  | 	spdk_json_write_named_uint64(w, "offset_blocks", offset_blocks); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 18:19:58 +00:00
										 |  |  | 	spdk_json_write_name(w, "partition_type_guid"); | 
					
						
							|  |  |  | 	write_guid(w, &gpt_entry->part_type_guid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_json_write_name(w, "unique_partition_guid"); | 
					
						
							|  |  |  | 	write_guid(w, &gpt_entry->unique_partition_guid); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 22:47:02 +00:00
										 |  |  | 	spdk_json_write_name(w, "partition_name"); | 
					
						
							|  |  |  | 	write_string_utf16le(w, gpt_entry->partition_name, SPDK_COUNTOF(gpt_entry->partition_name)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	spdk_json_write_object_end(w); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | vbdev_gpt_create_bdevs(struct gpt_base *gpt_base) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	uint32_t num_partition_entries; | 
					
						
							|  |  |  | 	uint64_t i, head_lba_start, head_lba_end; | 
					
						
							| 
									
										
										
										
											2018-07-03 16:57:23 +00:00
										 |  |  | 	uint32_t num_partitions; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	struct spdk_gpt_partition_entry *p; | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	struct gpt_disk *d; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	struct spdk_gpt *gpt; | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 	char *name; | 
					
						
							|  |  |  | 	struct spdk_bdev *base_bdev; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	gpt = &gpt_base->gpt; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	num_partition_entries = from_le32(&gpt->header->num_partition_entries); | 
					
						
							|  |  |  | 	head_lba_start = from_le64(&gpt->header->first_usable_lba); | 
					
						
							|  |  |  | 	head_lba_end = from_le64(&gpt->header->last_usable_lba); | 
					
						
							| 
									
										
										
										
											2018-07-03 16:57:23 +00:00
										 |  |  | 	num_partitions = 0; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < num_partition_entries; i++) { | 
					
						
							|  |  |  | 		p = &gpt->partitions[i]; | 
					
						
							|  |  |  | 		uint64_t lba_start = from_le64(&p->starting_lba); | 
					
						
							|  |  |  | 		uint64_t lba_end = from_le64(&p->ending_lba); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 17:51:47 +00:00
										 |  |  | 		if (!SPDK_GPT_GUID_EQUAL(&gpt->partitions[i].part_type_guid, | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 					 &SPDK_GPT_PART_TYPE_GUID) || | 
					
						
							|  |  |  | 		    lba_start == 0) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (lba_start < head_lba_start || lba_end > head_lba_end) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		d = calloc(1, sizeof(*d)); | 
					
						
							|  |  |  | 		if (!d) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("Memory allocation failure\n"); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* index start at 1 instead of 0 to match the existing style */ | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 		base_bdev = spdk_bdev_part_base_get_bdev(gpt_base->part_base); | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 		name = spdk_sprintf_alloc("%sp%" PRIu64, spdk_bdev_get_name(base_bdev), i + 1); | 
					
						
							|  |  |  | 		if (!name) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("name allocation failure\n"); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 			free(d); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 		rc = spdk_bdev_part_construct(&d->part, gpt_base->part_base, name, | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 					      lba_start, lba_end - lba_start, "GPT Disk"); | 
					
						
							| 
									
										
										
										
											2018-08-30 14:59:10 +00:00
										 |  |  | 		free(name); | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 		if (rc) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("could not construct bdev part\n"); | 
					
						
							|  |  |  | 			/* spdk_bdev_part_construct will free name on failure */ | 
					
						
							|  |  |  | 			free(d); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-07-03 16:57:23 +00:00
										 |  |  | 		num_partitions++; | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 		d->partition_index = i; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 16:57:23 +00:00
										 |  |  | 	return num_partitions; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2019-01-19 07:05:54 +00:00
										 |  |  | spdk_gpt_read_secondary_table_complete(struct spdk_bdev_io *bdev_io, bool status, void *arg) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	struct gpt_base *gpt_base = (struct gpt_base *)arg; | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	struct spdk_bdev *bdev = spdk_bdev_part_base_get_bdev(gpt_base->part_base); | 
					
						
							| 
									
										
										
										
											2018-07-03 16:57:23 +00:00
										 |  |  | 	int rc, num_partitions = 0; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 	spdk_bdev_free_io(bdev_io); | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	spdk_put_io_channel(gpt_base->ch); | 
					
						
							|  |  |  | 	gpt_base->ch = NULL; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (status != SPDK_BDEV_IO_STATUS_SUCCESS) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Gpt: bdev=%s io error status=%d\n", | 
					
						
							| 
									
										
										
										
											2017-07-07 23:04:52 +00:00
										 |  |  | 			    spdk_bdev_get_name(bdev), status); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		goto end; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-19 07:05:54 +00:00
										 |  |  | 	rc = spdk_gpt_parse_partition_table(&gpt_base->gpt); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		SPDK_DEBUGLOG(SPDK_LOG_VBDEV_GPT, "Failed to parse secondary partition table\n"); | 
					
						
							|  |  |  | 		goto end; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	SPDK_WARNLOG("Gpt: bdev=%s primary partition table broken, use the secondary\n", | 
					
						
							|  |  |  | 		     spdk_bdev_get_name(bdev)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	num_partitions = vbdev_gpt_create_bdevs(gpt_base); | 
					
						
							|  |  |  | 	if (num_partitions < 0) { | 
					
						
							|  |  |  | 		SPDK_DEBUGLOG(SPDK_LOG_VBDEV_GPT, "Failed to split dev=%s by gpt table\n", | 
					
						
							|  |  |  | 			      spdk_bdev_get_name(bdev)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | end: | 
					
						
							|  |  |  | 	spdk_bdev_module_examine_done(&gpt_if); | 
					
						
							|  |  |  | 	if (num_partitions <= 0) { | 
					
						
							|  |  |  | 		/* If no gpt_disk instances were created, free the base context */ | 
					
						
							|  |  |  | 		spdk_bdev_part_base_free(gpt_base->part_base); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | vbdev_gpt_read_secondary_table(struct gpt_base *gpt_base) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spdk_gpt *gpt; | 
					
						
							|  |  |  | 	struct spdk_bdev_desc *part_base_desc; | 
					
						
							|  |  |  | 	uint64_t secondary_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gpt = &gpt_base->gpt; | 
					
						
							|  |  |  | 	gpt->parse_phase = SPDK_GPT_PARSE_PHASE_SECONDARY; | 
					
						
							|  |  |  | 	gpt->header = NULL; | 
					
						
							|  |  |  | 	gpt->partitions = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	part_base_desc = spdk_bdev_part_base_get_desc(gpt_base->part_base); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	secondary_offset = gpt->total_sectors * gpt->sector_size - gpt->buf_size; | 
					
						
							|  |  |  | 	return spdk_bdev_read(part_base_desc, gpt_base->ch, gpt_base->gpt.buf, secondary_offset, | 
					
						
							|  |  |  | 			      gpt_base->gpt.buf_size, spdk_gpt_read_secondary_table_complete, | 
					
						
							|  |  |  | 			      gpt_base); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | spdk_gpt_bdev_complete(struct spdk_bdev_io *bdev_io, bool status, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct gpt_base *gpt_base = (struct gpt_base *)arg; | 
					
						
							|  |  |  | 	struct spdk_bdev *bdev = spdk_bdev_part_base_get_bdev(gpt_base->part_base); | 
					
						
							|  |  |  | 	int rc, num_partitions = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spdk_bdev_free_io(bdev_io); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (status != SPDK_BDEV_IO_STATUS_SUCCESS) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Gpt: bdev=%s io error status=%d\n", | 
					
						
							|  |  |  | 			    spdk_bdev_get_name(bdev), status); | 
					
						
							|  |  |  | 		goto end; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 	rc = spdk_gpt_parse_mbr(&gpt_base->gpt); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2019-01-19 06:56:14 +00:00
										 |  |  | 		SPDK_DEBUGLOG(SPDK_LOG_VBDEV_GPT, "Failed to parse mbr\n"); | 
					
						
							|  |  |  | 		goto end; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = spdk_gpt_parse_partition_table(&gpt_base->gpt); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		SPDK_DEBUGLOG(SPDK_LOG_VBDEV_GPT, "Failed to parse primary partition table\n"); | 
					
						
							| 
									
										
										
										
											2019-01-19 07:05:54 +00:00
										 |  |  | 		rc = vbdev_gpt_read_secondary_table(gpt_base); | 
					
						
							|  |  |  | 		if (rc) { | 
					
						
							|  |  |  | 			SPDK_ERRLOG("Failed to read secondary table\n"); | 
					
						
							|  |  |  | 			goto end; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 16:57:23 +00:00
										 |  |  | 	num_partitions = vbdev_gpt_create_bdevs(gpt_base); | 
					
						
							|  |  |  | 	if (num_partitions < 0) { | 
					
						
							| 
									
										
										
										
											2017-08-30 18:06:33 +00:00
										 |  |  | 		SPDK_DEBUGLOG(SPDK_LOG_VBDEV_GPT, "Failed to split dev=%s by gpt table\n", | 
					
						
							| 
									
										
										
										
											2017-07-07 23:04:52 +00:00
										 |  |  | 			      spdk_bdev_get_name(bdev)); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | end: | 
					
						
							| 
									
										
										
										
											2019-01-19 07:05:54 +00:00
										 |  |  | 	spdk_put_io_channel(gpt_base->ch); | 
					
						
							|  |  |  | 	gpt_base->ch = NULL; | 
					
						
							| 
									
										
										
										
											2017-07-07 16:14:33 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Notify the generic bdev layer that the actions related to the original examine | 
					
						
							|  |  |  | 	 *  callback are now completed. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2018-03-06 18:52:46 +00:00
										 |  |  | 	spdk_bdev_module_examine_done(&gpt_if); | 
					
						
							| 
									
										
										
										
											2017-07-07 16:14:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 16:57:23 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * vbdev_gpt_create_bdevs returns the number of bdevs created upon success. | 
					
						
							|  |  |  | 	 * We can branch on this value. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (num_partitions <= 0) { | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 		/* If no gpt_disk instances were created, free the base context */ | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 		spdk_bdev_part_base_free(gpt_base->part_base); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | vbdev_gpt_read_gpt(struct spdk_bdev *bdev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	struct gpt_base *gpt_base; | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	struct spdk_bdev_desc *part_base_desc; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 18:45:02 +00:00
										 |  |  | 	gpt_base = spdk_gpt_base_bdev_init(bdev); | 
					
						
							|  |  |  | 	if (!gpt_base) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Cannot allocated gpt_base\n"); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	part_base_desc = spdk_bdev_part_base_get_desc(gpt_base->part_base); | 
					
						
							|  |  |  | 	gpt_base->ch = spdk_bdev_get_io_channel(part_base_desc); | 
					
						
							| 
									
										
										
										
											2017-10-18 08:59:43 +00:00
										 |  |  | 	if (gpt_base->ch == NULL) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("Failed to get an io_channel.\n"); | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 		spdk_bdev_part_base_free(gpt_base->part_base); | 
					
						
							| 
									
										
										
										
											2017-10-18 08:59:43 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-08-30 19:25:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 	rc = spdk_bdev_read(part_base_desc, gpt_base->ch, gpt_base->gpt.buf, 0, | 
					
						
							| 
									
										
										
										
											2017-10-23 18:59:27 +00:00
										 |  |  | 			    gpt_base->gpt.buf_size, spdk_gpt_bdev_complete, gpt_base); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	if (rc < 0) { | 
					
						
							| 
									
										
										
										
											2017-09-11 15:04:55 +00:00
										 |  |  | 		spdk_put_io_channel(gpt_base->ch); | 
					
						
							| 
									
										
										
										
											2018-06-21 22:23:42 +00:00
										 |  |  | 		spdk_bdev_part_base_free(gpt_base->part_base); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		SPDK_ERRLOG("Failed to send bdev_io command\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-13 17:36:19 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | vbdev_gpt_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spdk_conf_section *sp = spdk_conf_find_section(NULL, "Gpt"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (sp && spdk_conf_section_get_boolval(sp, "Disable", false)) { | 
					
						
							|  |  |  | 		/* Disable Gpt probe */ | 
					
						
							|  |  |  | 		g_gpt_disabled = true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-13 17:36:19 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-08 09:16:08 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | vbdev_gpt_get_ctx_size(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return sizeof(struct gpt_io); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2017-07-11 07:04:40 +00:00
										 |  |  | vbdev_gpt_examine(struct spdk_bdev *bdev) | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-07-07 16:14:33 +00:00
										 |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-05 02:06:21 +00:00
										 |  |  | 	/* A bdev with fewer than 2 blocks cannot have a GPT. Block 0 has
 | 
					
						
							|  |  |  | 	 * the MBR and block 1 has the GPT header. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (g_gpt_disabled || spdk_bdev_get_num_blocks(bdev) < 2) { | 
					
						
							| 
									
										
										
										
											2018-03-06 18:52:46 +00:00
										 |  |  | 		spdk_bdev_module_examine_done(&gpt_if); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 15:43:32 +00:00
										 |  |  | 	if (spdk_bdev_get_block_size(bdev) % 512 != 0) { | 
					
						
							|  |  |  | 		SPDK_ERRLOG("GPT module does not support block size %" PRIu32 " for bdev %s\n", | 
					
						
							|  |  |  | 			    spdk_bdev_get_block_size(bdev), spdk_bdev_get_name(bdev)); | 
					
						
							|  |  |  | 		spdk_bdev_module_examine_done(&gpt_if); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 16:14:33 +00:00
										 |  |  | 	rc = vbdev_gpt_read_gpt(bdev); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2018-03-06 18:52:46 +00:00
										 |  |  | 		spdk_bdev_module_examine_done(&gpt_if); | 
					
						
							| 
									
										
										
										
											2017-07-07 16:14:33 +00:00
										 |  |  | 		SPDK_ERRLOG("Failed to read info from bdev %s\n", spdk_bdev_get_name(bdev)); | 
					
						
							| 
									
										
										
										
											2017-06-01 05:02:54 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-30 18:06:33 +00:00
										 |  |  | SPDK_LOG_REGISTER_COMPONENT("vbdev_gpt", SPDK_LOG_VBDEV_GPT) |