summaryrefslogtreecommitdiffstats
path: root/src/spdk/dpdk/examples/vhost_blk/blk.c
blob: f8c8549b3acc7baf53af57422dbfea60ee44311e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/* SPDX-License-Identifier: BSD-3-Clause
 * Copyright(c) 2010-2019 Intel Corporation
 */

/**
 * This work is largely based on the "vhost-user-blk" implementation by
 * SPDK(https://github.com/spdk/spdk).
 */

#include <stdio.h>
#include <stdint.h>
#include <unistd.h>
#include <assert.h>
#include <ctype.h>
#include <string.h>
#include <stddef.h>

#include <rte_atomic.h>
#include <rte_cycles.h>
#include <rte_log.h>
#include <rte_malloc.h>
#include <rte_byteorder.h>
#include <rte_string_fns.h>

#include "vhost_blk.h"
#include "blk_spec.h"

static void
vhost_strcpy_pad(void *dst, const char *src, size_t size, int pad)
{
	size_t len;

	len = strlen(src);
	if (len < size) {
		memcpy(dst, src, len);
		memset((char *)dst + len, pad, size - len);
	} else {
		memcpy(dst, src, size);
	}
}

static int
vhost_bdev_blk_readwrite(struct vhost_block_dev *bdev,
			  struct vhost_blk_task *task,
			  uint64_t lba_512, __rte_unused uint32_t xfer_len)
{
	uint32_t i;
	uint64_t offset;
	uint32_t nbytes = 0;

	offset = lba_512 * 512;

	/* iovs[0] is the head and iovs[iovs_cnt - 1] is the tail
	 * Middle is the data range
	 */
	for (i = 1; i < task->iovs_cnt - 1; i++) {
		if (task->dxfer_dir == BLK_DIR_TO_DEV)
			memcpy(bdev->data + offset, task->iovs[i].iov_base,
			       task->iovs[i].iov_len);
		else
			memcpy(task->iovs[i].iov_base, bdev->data + offset,
			       task->iovs[i].iov_len);
		offset += task->iovs[i].iov_len;
		nbytes += task->iovs[i].iov_len;
	}

	return nbytes;
}

int
vhost_bdev_process_blk_commands(struct vhost_block_dev *bdev,
				 struct vhost_blk_task *task)
{
	size_t used_len;

	if (unlikely(task->data_len > (bdev->blockcnt * bdev->blocklen))) {
		fprintf(stderr, "read or write beyond capacity\n");
		return VIRTIO_BLK_S_UNSUPP;
	}

	switch (task->req->type) {
	case VIRTIO_BLK_T_IN:
		if (unlikely(task->data_len == 0 ||
			(task->data_len & (512 - 1)) != 0)) {
			fprintf(stderr,
				"%s - passed IO buffer is not multiple of 512b"
				"(req_idx = %"PRIu16").\n",
				task->req->type ? "WRITE" : "READ",
				task->req_idx);
			return VIRTIO_BLK_S_UNSUPP;
		}

		task->dxfer_dir = BLK_DIR_FROM_DEV;
		vhost_bdev_blk_readwrite(bdev, task,
					 task->req->sector, task->data_len);
		break;
	case VIRTIO_BLK_T_OUT:
		if (unlikely(task->data_len == 0 ||
			(task->data_len & (512 - 1)) != 0)) {
			fprintf(stderr,
				"%s - passed IO buffer is not multiple of 512b"
				"(req_idx = %"PRIu16").\n",
				task->req->type ? "WRITE" : "READ",
				task->req_idx);
			return VIRTIO_BLK_S_UNSUPP;
		}

		task->dxfer_dir = BLK_DIR_TO_DEV;
		vhost_bdev_blk_readwrite(bdev, task,
					 task->req->sector, task->data_len);
		break;
	case VIRTIO_BLK_T_GET_ID:
		if (!task->iovs_cnt || task->data_len)
			return VIRTIO_BLK_S_UNSUPP;
		used_len = RTE_MIN((size_t)VIRTIO_BLK_ID_BYTES, task->data_len);
		vhost_strcpy_pad(task->iovs[0].iov_base,
				 bdev->product_name, used_len, ' ');
		break;
	default:
		fprintf(stderr, "unsupported cmd\n");
		return VIRTIO_BLK_S_UNSUPP;
	}

	return VIRTIO_BLK_S_OK;
}