linux/virt/kvm/binary_stats.c
<<
>>
Prefs
   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 */
  52ssize_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