2017-02-28 12:31:46 +03:00
|
|
|
/*
|
|
|
|
* Fsdev Throttle
|
|
|
|
*
|
|
|
|
* Copyright (C) 2016 Huawei Technologies Duesseldorf GmbH
|
|
|
|
*
|
|
|
|
* Author: Pradeep Jagadeesh <pradeep.jagadeesh@huawei.com>
|
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2 or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* See the COPYING file in the top-level directory for details.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "qemu/osdep.h"
|
|
|
|
#include "qemu/error-report.h"
|
|
|
|
#include "qemu-fsdev-throttle.h"
|
|
|
|
#include "qemu/iov.h"
|
Include qemu/main-loop.h less
In my "build everything" tree, changing qemu/main-loop.h triggers a
recompile of some 5600 out of 6600 objects (not counting tests and
objects that don't depend on qemu/osdep.h). It includes block/aio.h,
which in turn includes qemu/event_notifier.h, qemu/notify.h,
qemu/processor.h, qemu/qsp.h, qemu/queue.h, qemu/thread-posix.h,
qemu/thread.h, qemu/timer.h, and a few more.
Include qemu/main-loop.h only where it's needed. Touching it now
recompiles only some 1700 objects. For block/aio.h and
qemu/event_notifier.h, these numbers drop from 5600 to 2800. For the
others, they shrink only slightly.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-Id: <20190812052359.30071-21-armbru@redhat.com>
Reviewed-by: Alex Bennée <alex.bennee@linaro.org>
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Tested-by: Philippe Mathieu-Daudé <philmd@redhat.com>
2019-08-12 08:23:50 +03:00
|
|
|
#include "qemu/main-loop.h"
|
2018-02-01 14:18:46 +03:00
|
|
|
#include "qemu/option.h"
|
2017-02-28 12:31:46 +03:00
|
|
|
|
|
|
|
static void fsdev_throttle_read_timer_cb(void *opaque)
|
|
|
|
{
|
|
|
|
FsThrottle *fst = opaque;
|
2018-02-03 18:39:34 +03:00
|
|
|
qemu_co_enter_next(&fst->throttled_reqs[false], NULL);
|
2017-02-28 12:31:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void fsdev_throttle_write_timer_cb(void *opaque)
|
|
|
|
{
|
|
|
|
FsThrottle *fst = opaque;
|
2018-02-03 18:39:34 +03:00
|
|
|
qemu_co_enter_next(&fst->throttled_reqs[true], NULL);
|
2017-02-28 12:31:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void fsdev_throttle_parse_opts(QemuOpts *opts, FsThrottle *fst, Error **errp)
|
|
|
|
{
|
|
|
|
throttle_config_init(&fst->cfg);
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_TOTAL].avg =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-total", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_READ].avg =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-read", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_WRITE].avg =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-write", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_TOTAL].avg =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-total", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_READ].avg =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-read", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_WRITE].avg =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-write", 0);
|
|
|
|
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_TOTAL].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-total-max", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_READ].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-read-max", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_WRITE].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-write-max", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_TOTAL].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-total-max", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_READ].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-read-max", 0);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_WRITE].max =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-write-max", 0);
|
|
|
|
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_TOTAL].burst_length =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-total-max-length", 1);
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_READ].burst_length =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-read-max-length", 1);
|
|
|
|
fst->cfg.buckets[THROTTLE_BPS_WRITE].burst_length =
|
|
|
|
qemu_opt_get_number(opts, "throttling.bps-write-max-length", 1);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_TOTAL].burst_length =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-total-max-length", 1);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_READ].burst_length =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-read-max-length", 1);
|
|
|
|
fst->cfg.buckets[THROTTLE_OPS_WRITE].burst_length =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-write-max-length", 1);
|
|
|
|
fst->cfg.op_size =
|
|
|
|
qemu_opt_get_number(opts, "throttling.iops-size", 0);
|
|
|
|
|
|
|
|
throttle_is_valid(&fst->cfg, errp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void fsdev_throttle_init(FsThrottle *fst)
|
|
|
|
{
|
|
|
|
if (throttle_enabled(&fst->cfg)) {
|
|
|
|
throttle_init(&fst->ts);
|
|
|
|
throttle_timers_init(&fst->tt,
|
|
|
|
qemu_get_aio_context(),
|
|
|
|
QEMU_CLOCK_REALTIME,
|
|
|
|
fsdev_throttle_read_timer_cb,
|
|
|
|
fsdev_throttle_write_timer_cb,
|
|
|
|
fst);
|
2017-07-02 13:06:46 +03:00
|
|
|
throttle_config(&fst->ts, QEMU_CLOCK_REALTIME, &fst->cfg);
|
2017-02-28 12:31:46 +03:00
|
|
|
qemu_co_queue_init(&fst->throttled_reqs[0]);
|
|
|
|
qemu_co_queue_init(&fst->throttled_reqs[1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void coroutine_fn fsdev_co_throttle_request(FsThrottle *fst, bool is_write,
|
|
|
|
struct iovec *iov, int iovcnt)
|
|
|
|
{
|
|
|
|
if (throttle_enabled(&fst->cfg)) {
|
|
|
|
if (throttle_schedule_timer(&fst->ts, &fst->tt, is_write) ||
|
|
|
|
!qemu_co_queue_empty(&fst->throttled_reqs[is_write])) {
|
|
|
|
qemu_co_queue_wait(&fst->throttled_reqs[is_write], NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
throttle_account(&fst->ts, is_write, iov_size(iov, iovcnt));
|
|
|
|
|
|
|
|
if (!qemu_co_queue_empty(&fst->throttled_reqs[is_write]) &&
|
|
|
|
!throttle_schedule_timer(&fst->ts, &fst->tt, is_write)) {
|
|
|
|
qemu_co_queue_next(&fst->throttled_reqs[is_write]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void fsdev_throttle_cleanup(FsThrottle *fst)
|
|
|
|
{
|
|
|
|
if (throttle_enabled(&fst->cfg)) {
|
|
|
|
throttle_timers_destroy(&fst->tt);
|
|
|
|
}
|
|
|
|
}
|