| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * KVM binary statistics interface implementation |
| * |
| * Copyright 2021 Google LLC |
| */ |
| |
| #include <linux/kvm_host.h> |
| #include <linux/kvm.h> |
| #include <linux/errno.h> |
| #include <linux/uaccess.h> |
| |
| /** |
| * kvm_stats_read() - Common function to read from the binary statistics |
| * file descriptor. |
| * |
| * @id: identification string of the stats |
| * @header: stats header for a vm or a vcpu |
| * @desc: start address of an array of stats descriptors for a vm or a vcpu |
| * @stats: start address of stats data block for a vm or a vcpu |
| * @size_stats: the size of stats data block pointed by @stats |
| * @user_buffer: start address of userspace buffer |
| * @size: requested read size from userspace |
| * @offset: the start position from which the content will be read for the |
| * corresponding vm or vcp file descriptor |
| * |
| * The file content of a vm/vcpu file descriptor is now defined as below: |
| * +-------------+ |
| * | Header | |
| * +-------------+ |
| * | id string | |
| * +-------------+ |
| * | Descriptors | |
| * +-------------+ |
| * | Stats Data | |
| * +-------------+ |
| * Although this function allows userspace to read any amount of data (as long |
| * as in the limit) from any position, the typical usage would follow below |
| * steps: |
| * 1. Read header from offset 0. Get the offset of descriptors and stats data |
| * and some other necessary information. This is a one-time work for the |
| * lifecycle of the corresponding vm/vcpu stats fd. |
| * 2. Read id string from its offset. This is a one-time work for the lifecycle |
| * of the corresponding vm/vcpu stats fd. |
| * 3. Read descriptors from its offset and discover all the stats by parsing |
| * descriptors. This is a one-time work for the lifecycle of the |
| * corresponding vm/vcpu stats fd. |
| * 4. Periodically read stats data from its offset using pread. |
| * |
| * Return: the number of bytes that has been successfully read |
| */ |
| ssize_t kvm_stats_read(char *id, const struct kvm_stats_header *header, |
| const struct _kvm_stats_desc *desc, |
| void *stats, size_t size_stats, |
| char __user *user_buffer, size_t size, loff_t *offset) |
| { |
| ssize_t len; |
| ssize_t copylen; |
| ssize_t remain = size; |
| size_t size_desc; |
| size_t size_header; |
| void *src; |
| loff_t pos = *offset; |
| char __user *dest = user_buffer; |
| |
| size_header = sizeof(*header); |
| size_desc = header->num_desc * sizeof(*desc); |
| |
| len = KVM_STATS_NAME_SIZE + size_header + size_desc + size_stats - pos; |
| len = min(len, remain); |
| if (len <= 0) |
| return 0; |
| remain = len; |
| |
| /* |
| * Copy kvm stats header. |
| * The header is the first block of content userspace usually read out. |
| * The pos is 0 and the copylen and remain would be the size of header. |
| * The copy of the header would be skipped if offset is larger than the |
| * size of header. That usually happens when userspace reads stats |
| * descriptors and stats data. |
| */ |
| copylen = size_header - pos; |
| copylen = min(copylen, remain); |
| if (copylen > 0) { |
| src = (void *)header + pos; |
| if (copy_to_user(dest, src, copylen)) |
| return -EFAULT; |
| remain -= copylen; |
| pos += copylen; |
| dest += copylen; |
| } |
| |
| /* |
| * Copy kvm stats header id string. |
| * The id string is unique for every vm/vcpu, which is stored in kvm |
| * and kvm_vcpu structure. |
| * The id string is part of the stat header from the perspective of |
| * userspace, it is usually read out together with previous constant |
| * header part and could be skipped for later descriptors and stats |
| * data readings. |
| */ |
| copylen = header->id_offset + KVM_STATS_NAME_SIZE - pos; |
| copylen = min(copylen, remain); |
| if (copylen > 0) { |
| src = id + pos - header->id_offset; |
| if (copy_to_user(dest, src, copylen)) |
| return -EFAULT; |
| remain -= copylen; |
| pos += copylen; |
| dest += copylen; |
| } |
| |
| /* |
| * Copy kvm stats descriptors. |
| * The descriptors copy would be skipped in the typical case that |
| * userspace periodically read stats data, since the pos would be |
| * greater than the end address of descriptors |
| * (header->header.desc_offset + size_desc) causing copylen <= 0. |
| */ |
| copylen = header->desc_offset + size_desc - pos; |
| copylen = min(copylen, remain); |
| if (copylen > 0) { |
| src = (void *)desc + pos - header->desc_offset; |
| if (copy_to_user(dest, src, copylen)) |
| return -EFAULT; |
| remain -= copylen; |
| pos += copylen; |
| dest += copylen; |
| } |
| |
| /* Copy kvm stats values */ |
| copylen = header->data_offset + size_stats - pos; |
| copylen = min(copylen, remain); |
| if (copylen > 0) { |
| src = stats + pos - header->data_offset; |
| if (copy_to_user(dest, src, copylen)) |
| return -EFAULT; |
| pos += copylen; |
| } |
| |
| *offset = pos; |
| return len; |
| } |