xref: /openbmc/linux/virt/kvm/binary_stats.c (revision f7eeb00845934851b580b188f079545ab176fa5c)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * KVM binary statistics interface implementation
4  *
5  * Copyright 2021 Google LLC
6  */
7 
8 #include <linux/kvm_host.h>
9 #include <linux/kvm.h>
10 #include <linux/errno.h>
11 #include <linux/uaccess.h>
12 
13 /**
14  * kvm_stats_read() - Common function to read from the binary statistics
15  * file descriptor.
16  *
17  * @id: identification string of the stats
18  * @header: stats header for a vm or a vcpu
19  * @desc: start address of an array of stats descriptors for a vm or a vcpu
20  * @stats: start address of stats data block for a vm or a vcpu
21  * @size_stats: the size of stats data block pointed by @stats
22  * @user_buffer: start address of userspace buffer
23  * @size: requested read size from userspace
24  * @offset: the start position from which the content will be read for the
25  *          corresponding vm or vcp file descriptor
26  *
27  * The file content of a vm/vcpu file descriptor is now defined as below:
28  * +-------------+
29  * |   Header    |
30  * +-------------+
31  * |  id string  |
32  * +-------------+
33  * | Descriptors |
34  * +-------------+
35  * | Stats Data  |
36  * +-------------+
37  * Although this function allows userspace to read any amount of data (as long
38  * as in the limit) from any position, the typical usage would follow below
39  * steps:
40  * 1. Read header from offset 0. Get the offset of descriptors and stats data
41  *    and some other necessary information. This is a one-time work for the
42  *    lifecycle of the corresponding vm/vcpu stats fd.
43  * 2. Read id string from its offset. This is a one-time work for the lifecycle
44  *    of the corresponding vm/vcpu stats fd.
45  * 3. Read descriptors from its offset and discover all the stats by parsing
46  *    descriptors. This is a one-time work for the lifecycle of the
47  *    corresponding vm/vcpu stats fd.
48  * 4. Periodically read stats data from its offset using pread.
49  *
50  * Return: the number of bytes that has been successfully read
51  */
52 ssize_t kvm_stats_read(char *id, const struct kvm_stats_header *header,
53 		       const struct _kvm_stats_desc *desc,
54 		       void *stats, size_t size_stats,
55 		       char __user *user_buffer, size_t size, loff_t *offset)
56 {
57 	ssize_t len;
58 	ssize_t copylen;
59 	ssize_t remain = size;
60 	size_t size_desc;
61 	size_t size_header;
62 	void *src;
63 	loff_t pos = *offset;
64 	char __user *dest = user_buffer;
65 
66 	size_header = sizeof(*header);
67 	size_desc = header->num_desc * sizeof(*desc);
68 
69 	len = KVM_STATS_NAME_SIZE + size_header + size_desc + size_stats - pos;
70 	len = min(len, remain);
71 	if (len <= 0)
72 		return 0;
73 	remain = len;
74 
75 	/*
76 	 * Copy kvm stats header.
77 	 * The header is the first block of content userspace usually read out.
78 	 * The pos is 0 and the copylen and remain would be the size of header.
79 	 * The copy of the header would be skipped if offset is larger than the
80 	 * size of header. That usually happens when userspace reads stats
81 	 * descriptors and stats data.
82 	 */
83 	copylen = size_header - pos;
84 	copylen = min(copylen, remain);
85 	if (copylen > 0) {
86 		src = (void *)header + pos;
87 		if (copy_to_user(dest, src, copylen))
88 			return -EFAULT;
89 		remain -= copylen;
90 		pos += copylen;
91 		dest += copylen;
92 	}
93 
94 	/*
95 	 * Copy kvm stats header id string.
96 	 * The id string is unique for every vm/vcpu, which is stored in kvm
97 	 * and kvm_vcpu structure.
98 	 * The id string is part of the stat header from the perspective of
99 	 * userspace, it is usually read out together with previous constant
100 	 * header part and could be skipped for later descriptors and stats
101 	 * data readings.
102 	 */
103 	copylen = header->id_offset + KVM_STATS_NAME_SIZE - pos;
104 	copylen = min(copylen, remain);
105 	if (copylen > 0) {
106 		src = id + pos - header->id_offset;
107 		if (copy_to_user(dest, src, copylen))
108 			return -EFAULT;
109 		remain -= copylen;
110 		pos += copylen;
111 		dest += copylen;
112 	}
113 
114 	/*
115 	 * Copy kvm stats descriptors.
116 	 * The descriptors copy would be skipped in the typical case that
117 	 * userspace periodically read stats data, since the pos would be
118 	 * greater than the end address of descriptors
119 	 * (header->header.desc_offset + size_desc) causing copylen <= 0.
120 	 */
121 	copylen = header->desc_offset + size_desc - pos;
122 	copylen = min(copylen, remain);
123 	if (copylen > 0) {
124 		src = (void *)desc + pos - header->desc_offset;
125 		if (copy_to_user(dest, src, copylen))
126 			return -EFAULT;
127 		remain -= copylen;
128 		pos += copylen;
129 		dest += copylen;
130 	}
131 
132 	/* Copy kvm stats values */
133 	copylen = header->data_offset + size_stats - pos;
134 	copylen = min(copylen, remain);
135 	if (copylen > 0) {
136 		src = stats + pos - header->data_offset;
137 		if (copy_to_user(dest, src, copylen))
138 			return -EFAULT;
139 		pos += copylen;
140 	}
141 
142 	*offset = pos;
143 	return len;
144 }
145