|
|
1bdc94 |
From f92e2e46e8d14701afb20a12a45423cd06d7f5c6 Mon Sep 17 00:00:00 2001
|
|
|
1bdc94 |
From: Kevin Wolf <kwolf@redhat.com>
|
|
|
1bdc94 |
Date: Tue, 26 Jun 2018 09:47:54 +0200
|
|
|
1bdc94 |
Subject: [PATCH 25/89] blockjob: Implement block_job_set_speed() centrally
|
|
|
1bdc94 |
|
|
|
1bdc94 |
RH-Author: Kevin Wolf <kwolf@redhat.com>
|
|
|
1bdc94 |
Message-id: <20180626094856.6924-12-kwolf@redhat.com>
|
|
|
1bdc94 |
Patchwork-id: 81102
|
|
|
1bdc94 |
O-Subject: [RHV-7.6 qemu-kvm-rhev PATCH v2 11/73] blockjob: Implement block_job_set_speed() centrally
|
|
|
1bdc94 |
Bugzilla: 1513543
|
|
|
1bdc94 |
RH-Acked-by: Jeffrey Cody <jcody@redhat.com>
|
|
|
1bdc94 |
RH-Acked-by: Max Reitz <mreitz@redhat.com>
|
|
|
1bdc94 |
RH-Acked-by: Fam Zheng <famz@redhat.com>
|
|
|
1bdc94 |
|
|
|
1bdc94 |
All block job drivers support .set_speed and all of them duplicate the
|
|
|
1bdc94 |
same code to implement it. Move that code to blockjob.c and remove the
|
|
|
1bdc94 |
now useless callback.
|
|
|
1bdc94 |
|
|
|
1bdc94 |
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
|
|
|
1bdc94 |
Reviewed-by: Eric Blake <eblake@redhat.com>
|
|
|
1bdc94 |
Reviewed-by: Max Reitz <mreitz@redhat.com>
|
|
|
1bdc94 |
Reviewed-by: John Snow <jsnow@redhat.com>
|
|
|
1bdc94 |
(cherry picked from commit 18bb69287ea522ab696e1bea818b93e5eaa85745)
|
|
|
1bdc94 |
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
|
|
|
1bdc94 |
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
|
|
|
1bdc94 |
---
|
|
|
1bdc94 |
block/backup.c | 13 -------------
|
|
|
1bdc94 |
block/commit.c | 14 --------------
|
|
|
1bdc94 |
block/mirror.c | 26 ++++++--------------------
|
|
|
1bdc94 |
block/stream.c | 14 --------------
|
|
|
1bdc94 |
blockjob.c | 12 ++++--------
|
|
|
1bdc94 |
include/block/blockjob.h | 2 ++
|
|
|
1bdc94 |
include/block/blockjob_int.h | 3 ---
|
|
|
1bdc94 |
7 files changed, 12 insertions(+), 72 deletions(-)
|
|
|
1bdc94 |
|
|
|
1bdc94 |
diff --git a/block/backup.c b/block/backup.c
|
|
|
1bdc94 |
index 7585c43..8468fd9 100644
|
|
|
1bdc94 |
--- a/block/backup.c
|
|
|
1bdc94 |
+++ b/block/backup.c
|
|
|
1bdc94 |
@@ -27,7 +27,6 @@
|
|
|
1bdc94 |
#include "qemu/error-report.h"
|
|
|
1bdc94 |
|
|
|
1bdc94 |
#define BACKUP_CLUSTER_SIZE_DEFAULT (1 << 16)
|
|
|
1bdc94 |
-#define SLICE_TIME 100000000ULL /* ns */
|
|
|
1bdc94 |
|
|
|
1bdc94 |
typedef struct BackupBlockJob {
|
|
|
1bdc94 |
BlockJob common;
|
|
|
1bdc94 |
@@ -190,17 +189,6 @@ static int coroutine_fn backup_before_write_notify(
|
|
|
1bdc94 |
return backup_do_cow(job, req->offset, req->bytes, NULL, true);
|
|
|
1bdc94 |
}
|
|
|
1bdc94 |
|
|
|
1bdc94 |
-static void backup_set_speed(BlockJob *job, int64_t speed, Error **errp)
|
|
|
1bdc94 |
-{
|
|
|
1bdc94 |
- BackupBlockJob *s = container_of(job, BackupBlockJob, common);
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
- if (speed < 0) {
|
|
|
1bdc94 |
- error_setg(errp, QERR_INVALID_PARAMETER, "speed");
|
|
|
1bdc94 |
- return;
|
|
|
1bdc94 |
- }
|
|
|
1bdc94 |
- ratelimit_set_speed(&s->common.limit, speed, SLICE_TIME);
|
|
|
1bdc94 |
-}
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
static void backup_cleanup_sync_bitmap(BackupBlockJob *job, int ret)
|
|
|
1bdc94 |
{
|
|
|
1bdc94 |
BdrvDirtyBitmap *bm;
|
|
|
1bdc94 |
@@ -540,7 +528,6 @@ static const BlockJobDriver backup_job_driver = {
|
|
|
1bdc94 |
.instance_size = sizeof(BackupBlockJob),
|
|
|
1bdc94 |
.job_type = BLOCK_JOB_TYPE_BACKUP,
|
|
|
1bdc94 |
.start = backup_run,
|
|
|
1bdc94 |
- .set_speed = backup_set_speed,
|
|
|
1bdc94 |
.commit = backup_commit,
|
|
|
1bdc94 |
.abort = backup_abort,
|
|
|
1bdc94 |
.clean = backup_clean,
|
|
|
1bdc94 |
diff --git a/block/commit.c b/block/commit.c
|
|
|
1bdc94 |
index beec5d0..46cbeae 100644
|
|
|
1bdc94 |
--- a/block/commit.c
|
|
|
1bdc94 |
+++ b/block/commit.c
|
|
|
1bdc94 |
@@ -31,8 +31,6 @@ enum {
|
|
|
1bdc94 |
COMMIT_BUFFER_SIZE = 512 * 1024, /* in bytes */
|
|
|
1bdc94 |
};
|
|
|
1bdc94 |
|
|
|
1bdc94 |
-#define SLICE_TIME 100000000ULL /* ns */
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
typedef struct CommitBlockJob {
|
|
|
1bdc94 |
BlockJob common;
|
|
|
1bdc94 |
BlockDriverState *commit_top_bs;
|
|
|
1bdc94 |
@@ -216,21 +214,9 @@ out:
|
|
|
1bdc94 |
block_job_defer_to_main_loop(&s->common, commit_complete, data);
|
|
|
1bdc94 |
}
|
|
|
1bdc94 |
|
|
|
1bdc94 |
-static void commit_set_speed(BlockJob *job, int64_t speed, Error **errp)
|
|
|
1bdc94 |
-{
|
|
|
1bdc94 |
- CommitBlockJob *s = container_of(job, CommitBlockJob, common);
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
- if (speed < 0) {
|
|
|
1bdc94 |
- error_setg(errp, QERR_INVALID_PARAMETER, "speed");
|
|
|
1bdc94 |
- return;
|
|
|
1bdc94 |
- }
|
|
|
1bdc94 |
- ratelimit_set_speed(&s->common.limit, speed, SLICE_TIME);
|
|
|
1bdc94 |
-}
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
static const BlockJobDriver commit_job_driver = {
|
|
|
1bdc94 |
.instance_size = sizeof(CommitBlockJob),
|
|
|
1bdc94 |
.job_type = BLOCK_JOB_TYPE_COMMIT,
|
|
|
1bdc94 |
- .set_speed = commit_set_speed,
|
|
|
1bdc94 |
.start = commit_run,
|
|
|
1bdc94 |
};
|
|
|
1bdc94 |
|
|
|
1bdc94 |
diff --git a/block/mirror.c b/block/mirror.c
|
|
|
1bdc94 |
index a515ec1..d5e0ff2 100644
|
|
|
1bdc94 |
--- a/block/mirror.c
|
|
|
1bdc94 |
+++ b/block/mirror.c
|
|
|
1bdc94 |
@@ -22,7 +22,6 @@
|
|
|
1bdc94 |
#include "qemu/ratelimit.h"
|
|
|
1bdc94 |
#include "qemu/bitmap.h"
|
|
|
1bdc94 |
|
|
|
1bdc94 |
-#define SLICE_TIME 100000000ULL /* ns */
|
|
|
1bdc94 |
#define MAX_IN_FLIGHT 16
|
|
|
1bdc94 |
#define MAX_IO_BYTES (1 << 20) /* 1 Mb */
|
|
|
1bdc94 |
#define DEFAULT_MIRROR_BUF_SIZE (MAX_IN_FLIGHT * MAX_IO_BYTES)
|
|
|
1bdc94 |
@@ -596,7 +595,7 @@ static void mirror_throttle(MirrorBlockJob *s)
|
|
|
1bdc94 |
{
|
|
|
1bdc94 |
int64_t now = qemu_clock_get_ns(QEMU_CLOCK_REALTIME);
|
|
|
1bdc94 |
|
|
|
1bdc94 |
- if (now - s->last_pause_ns > SLICE_TIME) {
|
|
|
1bdc94 |
+ if (now - s->last_pause_ns > BLOCK_JOB_SLICE_TIME) {
|
|
|
1bdc94 |
s->last_pause_ns = now;
|
|
|
1bdc94 |
block_job_sleep_ns(&s->common, 0);
|
|
|
1bdc94 |
} else {
|
|
|
1bdc94 |
@@ -799,11 +798,10 @@ static void coroutine_fn mirror_run(void *opaque)
|
|
|
1bdc94 |
|
|
|
1bdc94 |
/* Note that even when no rate limit is applied we need to yield
|
|
|
1bdc94 |
* periodically with no pending I/O so that bdrv_drain_all() returns.
|
|
|
1bdc94 |
- * We do so every SLICE_TIME nanoseconds, or when there is an error,
|
|
|
1bdc94 |
- * or when the source is clean, whichever comes first.
|
|
|
1bdc94 |
- */
|
|
|
1bdc94 |
+ * We do so every BLKOCK_JOB_SLICE_TIME nanoseconds, or when there is
|
|
|
1bdc94 |
+ * an error, or when the source is clean, whichever comes first. */
|
|
|
1bdc94 |
delta = qemu_clock_get_ns(QEMU_CLOCK_REALTIME) - s->last_pause_ns;
|
|
|
1bdc94 |
- if (delta < SLICE_TIME &&
|
|
|
1bdc94 |
+ if (delta < BLOCK_JOB_SLICE_TIME &&
|
|
|
1bdc94 |
s->common.iostatus == BLOCK_DEVICE_IO_STATUS_OK) {
|
|
|
1bdc94 |
if (s->in_flight >= MAX_IN_FLIGHT || s->buf_free_count == 0 ||
|
|
|
1bdc94 |
(cnt == 0 && s->in_flight > 0)) {
|
|
|
1bdc94 |
@@ -869,7 +867,8 @@ static void coroutine_fn mirror_run(void *opaque)
|
|
|
1bdc94 |
ret = 0;
|
|
|
1bdc94 |
|
|
|
1bdc94 |
if (s->synced && !should_complete) {
|
|
|
1bdc94 |
- delay_ns = (s->in_flight == 0 && cnt == 0 ? SLICE_TIME : 0);
|
|
|
1bdc94 |
+ delay_ns = (s->in_flight == 0 &&
|
|
|
1bdc94 |
+ cnt == 0 ? BLOCK_JOB_SLICE_TIME : 0);
|
|
|
1bdc94 |
}
|
|
|
1bdc94 |
trace_mirror_before_sleep(s, cnt, s->synced, delay_ns);
|
|
|
1bdc94 |
block_job_sleep_ns(&s->common, delay_ns);
|
|
|
1bdc94 |
@@ -908,17 +907,6 @@ immediate_exit:
|
|
|
1bdc94 |
block_job_defer_to_main_loop(&s->common, mirror_exit, data);
|
|
|
1bdc94 |
}
|
|
|
1bdc94 |
|
|
|
1bdc94 |
-static void mirror_set_speed(BlockJob *job, int64_t speed, Error **errp)
|
|
|
1bdc94 |
-{
|
|
|
1bdc94 |
- MirrorBlockJob *s = container_of(job, MirrorBlockJob, common);
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
- if (speed < 0) {
|
|
|
1bdc94 |
- error_setg(errp, QERR_INVALID_PARAMETER, "speed");
|
|
|
1bdc94 |
- return;
|
|
|
1bdc94 |
- }
|
|
|
1bdc94 |
- ratelimit_set_speed(&s->common.limit, speed, SLICE_TIME);
|
|
|
1bdc94 |
-}
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
static void mirror_complete(BlockJob *job, Error **errp)
|
|
|
1bdc94 |
{
|
|
|
1bdc94 |
MirrorBlockJob *s = container_of(job, MirrorBlockJob, common);
|
|
|
1bdc94 |
@@ -1003,7 +991,6 @@ static void mirror_drain(BlockJob *job)
|
|
|
1bdc94 |
static const BlockJobDriver mirror_job_driver = {
|
|
|
1bdc94 |
.instance_size = sizeof(MirrorBlockJob),
|
|
|
1bdc94 |
.job_type = BLOCK_JOB_TYPE_MIRROR,
|
|
|
1bdc94 |
- .set_speed = mirror_set_speed,
|
|
|
1bdc94 |
.start = mirror_run,
|
|
|
1bdc94 |
.complete = mirror_complete,
|
|
|
1bdc94 |
.pause = mirror_pause,
|
|
|
1bdc94 |
@@ -1014,7 +1001,6 @@ static const BlockJobDriver mirror_job_driver = {
|
|
|
1bdc94 |
static const BlockJobDriver commit_active_job_driver = {
|
|
|
1bdc94 |
.instance_size = sizeof(MirrorBlockJob),
|
|
|
1bdc94 |
.job_type = BLOCK_JOB_TYPE_COMMIT,
|
|
|
1bdc94 |
- .set_speed = mirror_set_speed,
|
|
|
1bdc94 |
.start = mirror_run,
|
|
|
1bdc94 |
.complete = mirror_complete,
|
|
|
1bdc94 |
.pause = mirror_pause,
|
|
|
1bdc94 |
diff --git a/block/stream.c b/block/stream.c
|
|
|
1bdc94 |
index a1d4768..797d7c4 100644
|
|
|
1bdc94 |
--- a/block/stream.c
|
|
|
1bdc94 |
+++ b/block/stream.c
|
|
|
1bdc94 |
@@ -29,8 +29,6 @@ enum {
|
|
|
1bdc94 |
STREAM_BUFFER_SIZE = 512 * 1024, /* in bytes */
|
|
|
1bdc94 |
};
|
|
|
1bdc94 |
|
|
|
1bdc94 |
-#define SLICE_TIME 100000000ULL /* ns */
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
typedef struct StreamBlockJob {
|
|
|
1bdc94 |
BlockJob common;
|
|
|
1bdc94 |
BlockDriverState *base;
|
|
|
1bdc94 |
@@ -210,21 +208,9 @@ out:
|
|
|
1bdc94 |
block_job_defer_to_main_loop(&s->common, stream_complete, data);
|
|
|
1bdc94 |
}
|
|
|
1bdc94 |
|
|
|
1bdc94 |
-static void stream_set_speed(BlockJob *job, int64_t speed, Error **errp)
|
|
|
1bdc94 |
-{
|
|
|
1bdc94 |
- StreamBlockJob *s = container_of(job, StreamBlockJob, common);
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
- if (speed < 0) {
|
|
|
1bdc94 |
- error_setg(errp, QERR_INVALID_PARAMETER, "speed");
|
|
|
1bdc94 |
- return;
|
|
|
1bdc94 |
- }
|
|
|
1bdc94 |
- ratelimit_set_speed(&s->common.limit, speed, SLICE_TIME);
|
|
|
1bdc94 |
-}
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
static const BlockJobDriver stream_job_driver = {
|
|
|
1bdc94 |
.instance_size = sizeof(StreamBlockJob),
|
|
|
1bdc94 |
.job_type = BLOCK_JOB_TYPE_STREAM,
|
|
|
1bdc94 |
- .set_speed = stream_set_speed,
|
|
|
1bdc94 |
.start = stream_run,
|
|
|
1bdc94 |
};
|
|
|
1bdc94 |
|
|
|
1bdc94 |
diff --git a/blockjob.c b/blockjob.c
|
|
|
1bdc94 |
index d0a2ac5..0f7214c 100644
|
|
|
1bdc94 |
--- a/blockjob.c
|
|
|
1bdc94 |
+++ b/blockjob.c
|
|
|
1bdc94 |
@@ -667,22 +667,18 @@ static void block_job_completed_txn_success(BlockJob *job)
|
|
|
1bdc94 |
|
|
|
1bdc94 |
void block_job_set_speed(BlockJob *job, int64_t speed, Error **errp)
|
|
|
1bdc94 |
{
|
|
|
1bdc94 |
- Error *local_err = NULL;
|
|
|
1bdc94 |
int64_t old_speed = job->speed;
|
|
|
1bdc94 |
|
|
|
1bdc94 |
- if (!job->driver->set_speed) {
|
|
|
1bdc94 |
- error_setg(errp, QERR_UNSUPPORTED);
|
|
|
1bdc94 |
- return;
|
|
|
1bdc94 |
- }
|
|
|
1bdc94 |
if (block_job_apply_verb(job, BLOCK_JOB_VERB_SET_SPEED, errp)) {
|
|
|
1bdc94 |
return;
|
|
|
1bdc94 |
}
|
|
|
1bdc94 |
- job->driver->set_speed(job, speed, &local_err);
|
|
|
1bdc94 |
- if (local_err) {
|
|
|
1bdc94 |
- error_propagate(errp, local_err);
|
|
|
1bdc94 |
+ if (speed < 0) {
|
|
|
1bdc94 |
+ error_setg(errp, QERR_INVALID_PARAMETER, "speed");
|
|
|
1bdc94 |
return;
|
|
|
1bdc94 |
}
|
|
|
1bdc94 |
|
|
|
1bdc94 |
+ ratelimit_set_speed(&job->limit, speed, BLOCK_JOB_SLICE_TIME);
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
job->speed = speed;
|
|
|
1bdc94 |
if (speed && speed <= old_speed) {
|
|
|
1bdc94 |
return;
|
|
|
1bdc94 |
diff --git a/include/block/blockjob.h b/include/block/blockjob.h
|
|
|
1bdc94 |
index 22bf418..82f52f4 100644
|
|
|
1bdc94 |
--- a/include/block/blockjob.h
|
|
|
1bdc94 |
+++ b/include/block/blockjob.h
|
|
|
1bdc94 |
@@ -29,6 +29,8 @@
|
|
|
1bdc94 |
#include "block/block.h"
|
|
|
1bdc94 |
#include "qemu/ratelimit.h"
|
|
|
1bdc94 |
|
|
|
1bdc94 |
+#define BLOCK_JOB_SLICE_TIME 100000000ULL /* ns */
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
typedef struct BlockJobDriver BlockJobDriver;
|
|
|
1bdc94 |
typedef struct BlockJobTxn BlockJobTxn;
|
|
|
1bdc94 |
|
|
|
1bdc94 |
diff --git a/include/block/blockjob_int.h b/include/block/blockjob_int.h
|
|
|
1bdc94 |
index d5a515d..ad510d5 100644
|
|
|
1bdc94 |
--- a/include/block/blockjob_int.h
|
|
|
1bdc94 |
+++ b/include/block/blockjob_int.h
|
|
|
1bdc94 |
@@ -41,9 +41,6 @@ struct BlockJobDriver {
|
|
|
1bdc94 |
/** String describing the operation, part of query-block-jobs QMP API */
|
|
|
1bdc94 |
BlockJobType job_type;
|
|
|
1bdc94 |
|
|
|
1bdc94 |
- /** Optional callback for job types that support setting a speed limit */
|
|
|
1bdc94 |
- void (*set_speed)(BlockJob *job, int64_t speed, Error **errp);
|
|
|
1bdc94 |
-
|
|
|
1bdc94 |
/** Mandatory: Entrypoint for the Coroutine. */
|
|
|
1bdc94 |
CoroutineEntry *start;
|
|
|
1bdc94 |
|
|
|
1bdc94 |
--
|
|
|
1bdc94 |
1.8.3.1
|
|
|
1bdc94 |
|