Blame SOURCES/kvm-job-Move-state-transitions-to-Job.patch

1bdc94
From bb250647bab72a20db342030756d1837b91bc91a Mon Sep 17 00:00:00 2001
1bdc94
From: Kevin Wolf <kwolf@redhat.com>
1bdc94
Date: Tue, 26 Jun 2018 09:48:04 +0200
1bdc94
Subject: [PATCH 35/89] job: Move state transitions to Job
1bdc94
1bdc94
RH-Author: Kevin Wolf <kwolf@redhat.com>
1bdc94
Message-id: <20180626094856.6924-22-kwolf@redhat.com>
1bdc94
Patchwork-id: 81093
1bdc94
O-Subject: [RHV-7.6 qemu-kvm-rhev PATCH v2 21/73] job: Move state transitions to Job
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
This moves BlockJob.status and the closely related functions
1bdc94
(block_)job_state_transition() and (block_)job_apply_verb to Job. The
1bdc94
two QAPI enums are renamed to JobStatus and JobVerb.
1bdc94
1bdc94
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
1bdc94
Reviewed-by: Max Reitz <mreitz@redhat.com>
1bdc94
Reviewed-by: John Snow <jsnow@redhat.com>
1bdc94
Reviewed-by: Eric Blake <eblake@redhat.com>
1bdc94
(cherry picked from commit a50c2ab858fe613fb805e53b4f6b970ab936706d)
1bdc94
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
1bdc94
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
1bdc94
---
1bdc94
 block/trace-events       |   2 -
1bdc94
 blockjob.c               | 102 +++++++++++------------------------------------
1bdc94
 include/block/blockjob.h |   3 --
1bdc94
 include/qemu/job.h       |  13 ++++++
1bdc94
 job.c                    |  56 ++++++++++++++++++++++++++
1bdc94
 qapi/block-core.json     |  16 ++++----
1bdc94
 tests/test-blockjob.c    |  39 +++++++++---------
1bdc94
 trace-events             |   4 ++
1bdc94
 8 files changed, 123 insertions(+), 112 deletions(-)
1bdc94
1bdc94
diff --git a/block/trace-events b/block/trace-events
1bdc94
index f8c50b4..93b9279 100644
1bdc94
--- a/block/trace-events
1bdc94
+++ b/block/trace-events
1bdc94
@@ -6,8 +6,6 @@ bdrv_lock_medium(void *bs, bool locked) "bs %p locked %d"
1bdc94
 
1bdc94
 # blockjob.c
1bdc94
 block_job_completed(void *job, int ret, int jret) "job %p ret %d corrected ret %d"
1bdc94
-block_job_state_transition(void *job,  int ret, const char *legal, const char *s0, const char *s1) "job %p (ret: %d) attempting %s transition (%s-->%s)"
1bdc94
-block_job_apply_verb(void *job, const char *state, const char *verb, const char *legal) "job %p in state %s; applying verb %s (%s)"
1bdc94
 
1bdc94
 # block/block-backend.c
1bdc94
 blk_co_preadv(void *blk, void *bs, int64_t offset, unsigned int bytes, int flags) "blk %p bs %p offset %"PRId64" bytes %u flags 0x%x"
1bdc94
diff --git a/blockjob.c b/blockjob.c
1bdc94
index c69b2e7..0fba01e 100644
1bdc94
--- a/blockjob.c
1bdc94
+++ b/blockjob.c
1bdc94
@@ -41,61 +41,6 @@
1bdc94
  * block_job_enter. */
1bdc94
 static QemuMutex block_job_mutex;
1bdc94
 
1bdc94
-/* BlockJob State Transition Table */
1bdc94
-bool BlockJobSTT[BLOCK_JOB_STATUS__MAX][BLOCK_JOB_STATUS__MAX] = {
1bdc94
-                                          /* U, C, R, P, Y, S, W, D, X, E, N */
1bdc94
-    /* U: */ [BLOCK_JOB_STATUS_UNDEFINED] = {0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0},
1bdc94
-    /* C: */ [BLOCK_JOB_STATUS_CREATED]   = {0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 1},
1bdc94
-    /* R: */ [BLOCK_JOB_STATUS_RUNNING]   = {0, 0, 0, 1, 1, 0, 1, 0, 1, 0, 0},
1bdc94
-    /* P: */ [BLOCK_JOB_STATUS_PAUSED]    = {0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0},
1bdc94
-    /* Y: */ [BLOCK_JOB_STATUS_READY]     = {0, 0, 0, 0, 0, 1, 1, 0, 1, 0, 0},
1bdc94
-    /* S: */ [BLOCK_JOB_STATUS_STANDBY]   = {0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0},
1bdc94
-    /* W: */ [BLOCK_JOB_STATUS_WAITING]   = {0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0},
1bdc94
-    /* D: */ [BLOCK_JOB_STATUS_PENDING]   = {0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0},
1bdc94
-    /* X: */ [BLOCK_JOB_STATUS_ABORTING]  = {0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0},
1bdc94
-    /* E: */ [BLOCK_JOB_STATUS_CONCLUDED] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
1bdc94
-    /* N: */ [BLOCK_JOB_STATUS_NULL]      = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
1bdc94
-};
1bdc94
-
1bdc94
-bool BlockJobVerbTable[BLOCK_JOB_VERB__MAX][BLOCK_JOB_STATUS__MAX] = {
1bdc94
-                                          /* U, C, R, P, Y, S, W, D, X, E, N */
1bdc94
-    [BLOCK_JOB_VERB_CANCEL]               = {0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0},
1bdc94
-    [BLOCK_JOB_VERB_PAUSE]                = {0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0},
1bdc94
-    [BLOCK_JOB_VERB_RESUME]               = {0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0},
1bdc94
-    [BLOCK_JOB_VERB_SET_SPEED]            = {0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0},
1bdc94
-    [BLOCK_JOB_VERB_COMPLETE]             = {0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0},
1bdc94
-    [BLOCK_JOB_VERB_FINALIZE]             = {0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0},
1bdc94
-    [BLOCK_JOB_VERB_DISMISS]              = {0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0},
1bdc94
-};
1bdc94
-
1bdc94
-static void block_job_state_transition(BlockJob *job, BlockJobStatus s1)
1bdc94
-{
1bdc94
-    BlockJobStatus s0 = job->status;
1bdc94
-    assert(s1 >= 0 && s1 <= BLOCK_JOB_STATUS__MAX);
1bdc94
-    trace_block_job_state_transition(job, job->ret, BlockJobSTT[s0][s1] ?
1bdc94
-                                     "allowed" : "disallowed",
1bdc94
-                                     BlockJobStatus_str(s0),
1bdc94
-                                     BlockJobStatus_str(s1));
1bdc94
-    assert(BlockJobSTT[s0][s1]);
1bdc94
-    job->status = s1;
1bdc94
-}
1bdc94
-
1bdc94
-static int block_job_apply_verb(BlockJob *job, BlockJobVerb bv, Error **errp)
1bdc94
-{
1bdc94
-    assert(bv >= 0 && bv <= BLOCK_JOB_VERB__MAX);
1bdc94
-    trace_block_job_apply_verb(job, BlockJobStatus_str(job->status),
1bdc94
-                               BlockJobVerb_str(bv),
1bdc94
-                               BlockJobVerbTable[bv][job->status] ?
1bdc94
-                               "allowed" : "prohibited");
1bdc94
-    if (BlockJobVerbTable[bv][job->status]) {
1bdc94
-        return 0;
1bdc94
-    }
1bdc94
-    error_setg(errp, "Job '%s' in state '%s' cannot accept command verb '%s'",
1bdc94
-               job->job.id, BlockJobStatus_str(job->status),
1bdc94
-               BlockJobVerb_str(bv));
1bdc94
-    return -EPERM;
1bdc94
-}
1bdc94
-
1bdc94
 static void block_job_lock(void)
1bdc94
 {
1bdc94
     qemu_mutex_lock(&block_job_mutex);
1bdc94
@@ -257,7 +202,7 @@ static void block_job_detach_aio_context(void *opaque);
1bdc94
 void block_job_unref(BlockJob *job)
1bdc94
 {
1bdc94
     if (--job->refcnt == 0) {
1bdc94
-        assert(job->status == BLOCK_JOB_STATUS_NULL);
1bdc94
+        assert(job->job.status == JOB_STATUS_NULL);
1bdc94
         assert(!job->txn);
1bdc94
         BlockDriverState *bs = blk_bs(job->blk);
1bdc94
         bs->job = NULL;
1bdc94
@@ -409,7 +354,7 @@ void block_job_start(BlockJob *job)
1bdc94
     job->pause_count--;
1bdc94
     job->busy = true;
1bdc94
     job->paused = false;
1bdc94
-    block_job_state_transition(job, BLOCK_JOB_STATUS_RUNNING);
1bdc94
+    job_state_transition(&job->job, JOB_STATUS_RUNNING);
1bdc94
     bdrv_coroutine_enter(blk_bs(job->blk), job->co);
1bdc94
 }
1bdc94
 
1bdc94
@@ -421,7 +366,7 @@ static void block_job_decommission(BlockJob *job)
1bdc94
     job->paused = false;
1bdc94
     job->deferred_to_main_loop = true;
1bdc94
     block_job_txn_del_job(job);
1bdc94
-    block_job_state_transition(job, BLOCK_JOB_STATUS_NULL);
1bdc94
+    job_state_transition(&job->job, JOB_STATUS_NULL);
1bdc94
     block_job_unref(job);
1bdc94
 }
1bdc94
 
1bdc94
@@ -432,7 +377,7 @@ static void block_job_do_dismiss(BlockJob *job)
1bdc94
 
1bdc94
 static void block_job_conclude(BlockJob *job)
1bdc94
 {
1bdc94
-    block_job_state_transition(job, BLOCK_JOB_STATUS_CONCLUDED);
1bdc94
+    job_state_transition(&job->job, JOB_STATUS_CONCLUDED);
1bdc94
     if (job->auto_dismiss || !block_job_started(job)) {
1bdc94
         block_job_do_dismiss(job);
1bdc94
     }
1bdc94
@@ -444,7 +389,7 @@ static void block_job_update_rc(BlockJob *job)
1bdc94
         job->ret = -ECANCELED;
1bdc94
     }
1bdc94
     if (job->ret) {
1bdc94
-        block_job_state_transition(job, BLOCK_JOB_STATUS_ABORTING);
1bdc94
+        job_state_transition(&job->job, JOB_STATUS_ABORTING);
1bdc94
     }
1bdc94
 }
1bdc94
 
1bdc94
@@ -652,7 +597,7 @@ static void block_job_completed_txn_success(BlockJob *job)
1bdc94
     BlockJobTxn *txn = job->txn;
1bdc94
     BlockJob *other_job;
1bdc94
 
1bdc94
-    block_job_state_transition(job, BLOCK_JOB_STATUS_WAITING);
1bdc94
+    job_state_transition(&job->job, JOB_STATUS_WAITING);
1bdc94
 
1bdc94
     /*
1bdc94
      * Successful completion, see if there are other running jobs in this
1bdc94
@@ -677,7 +622,7 @@ void block_job_set_speed(BlockJob *job, int64_t speed, Error **errp)
1bdc94
 {
1bdc94
     int64_t old_speed = job->speed;
1bdc94
 
1bdc94
-    if (block_job_apply_verb(job, BLOCK_JOB_VERB_SET_SPEED, errp)) {
1bdc94
+    if (job_apply_verb(&job->job, JOB_VERB_SET_SPEED, errp)) {
1bdc94
         return;
1bdc94
     }
1bdc94
     if (speed < 0) {
1bdc94
@@ -709,7 +654,7 @@ void block_job_complete(BlockJob *job, Error **errp)
1bdc94
 {
1bdc94
     /* Should not be reachable via external interface for internal jobs */
1bdc94
     assert(job->job.id);
1bdc94
-    if (block_job_apply_verb(job, BLOCK_JOB_VERB_COMPLETE, errp)) {
1bdc94
+    if (job_apply_verb(&job->job, JOB_VERB_COMPLETE, errp)) {
1bdc94
         return;
1bdc94
     }
1bdc94
     if (job->pause_count || job->cancelled || !job->driver->complete) {
1bdc94
@@ -724,7 +669,7 @@ void block_job_complete(BlockJob *job, Error **errp)
1bdc94
 void block_job_finalize(BlockJob *job, Error **errp)
1bdc94
 {
1bdc94
     assert(job && job->job.id);
1bdc94
-    if (block_job_apply_verb(job, BLOCK_JOB_VERB_FINALIZE, errp)) {
1bdc94
+    if (job_apply_verb(&job->job, JOB_VERB_FINALIZE, errp)) {
1bdc94
         return;
1bdc94
     }
1bdc94
     block_job_do_finalize(job);
1bdc94
@@ -735,7 +680,7 @@ void block_job_dismiss(BlockJob **jobptr, Error **errp)
1bdc94
     BlockJob *job = *jobptr;
1bdc94
     /* similarly to _complete, this is QMP-interface only. */
1bdc94
     assert(job->job.id);
1bdc94
-    if (block_job_apply_verb(job, BLOCK_JOB_VERB_DISMISS, errp)) {
1bdc94
+    if (job_apply_verb(&job->job, JOB_VERB_DISMISS, errp)) {
1bdc94
         return;
1bdc94
     }
1bdc94
 
1bdc94
@@ -745,7 +690,7 @@ void block_job_dismiss(BlockJob **jobptr, Error **errp)
1bdc94
 
1bdc94
 void block_job_user_pause(BlockJob *job, Error **errp)
1bdc94
 {
1bdc94
-    if (block_job_apply_verb(job, BLOCK_JOB_VERB_PAUSE, errp)) {
1bdc94
+    if (job_apply_verb(&job->job, JOB_VERB_PAUSE, errp)) {
1bdc94
         return;
1bdc94
     }
1bdc94
     if (job->user_paused) {
1bdc94
@@ -768,7 +713,7 @@ void block_job_user_resume(BlockJob *job, Error **errp)
1bdc94
         error_setg(errp, "Can't resume a job that was not paused");
1bdc94
         return;
1bdc94
     }
1bdc94
-    if (block_job_apply_verb(job, BLOCK_JOB_VERB_RESUME, errp)) {
1bdc94
+    if (job_apply_verb(&job->job, JOB_VERB_RESUME, errp)) {
1bdc94
         return;
1bdc94
     }
1bdc94
     block_job_iostatus_reset(job);
1bdc94
@@ -778,7 +723,7 @@ void block_job_user_resume(BlockJob *job, Error **errp)
1bdc94
 
1bdc94
 void block_job_cancel(BlockJob *job, bool force)
1bdc94
 {
1bdc94
-    if (job->status == BLOCK_JOB_STATUS_CONCLUDED) {
1bdc94
+    if (job->job.status == JOB_STATUS_CONCLUDED) {
1bdc94
         block_job_do_dismiss(job);
1bdc94
         return;
1bdc94
     }
1bdc94
@@ -794,7 +739,7 @@ void block_job_cancel(BlockJob *job, bool force)
1bdc94
 
1bdc94
 void block_job_user_cancel(BlockJob *job, bool force, Error **errp)
1bdc94
 {
1bdc94
-    if (block_job_apply_verb(job, BLOCK_JOB_VERB_CANCEL, errp)) {
1bdc94
+    if (job_apply_verb(&job->job, JOB_VERB_CANCEL, errp)) {
1bdc94
         return;
1bdc94
     }
1bdc94
     block_job_cancel(job, force);
1bdc94
@@ -859,7 +804,7 @@ BlockJobInfo *block_job_query(BlockJob *job, Error **errp)
1bdc94
     info->speed     = job->speed;
1bdc94
     info->io_status = job->iostatus;
1bdc94
     info->ready     = job->ready;
1bdc94
-    info->status    = job->status;
1bdc94
+    info->status    = job->job.status;
1bdc94
     info->auto_finalize = job->auto_finalize;
1bdc94
     info->auto_dismiss  = job->auto_dismiss;
1bdc94
     info->has_error = job->ret != 0;
1bdc94
@@ -907,7 +852,7 @@ static void block_job_event_completed(BlockJob *job, const char *msg)
1bdc94
 
1bdc94
 static int block_job_event_pending(BlockJob *job)
1bdc94
 {
1bdc94
-    block_job_state_transition(job, BLOCK_JOB_STATUS_PENDING);
1bdc94
+    job_state_transition(&job->job, JOB_STATUS_PENDING);
1bdc94
     if (!job->auto_finalize && !block_job_is_internal(job)) {
1bdc94
         qapi_event_send_block_job_pending(job_type(&job->job),
1bdc94
                                           job->job.id,
1bdc94
@@ -975,7 +920,6 @@ void *block_job_create(const char *job_id, const BlockJobDriver *driver,
1bdc94
     job->refcnt        = 1;
1bdc94
     job->auto_finalize = !(flags & BLOCK_JOB_MANUAL_FINALIZE);
1bdc94
     job->auto_dismiss  = !(flags & BLOCK_JOB_MANUAL_DISMISS);
1bdc94
-    block_job_state_transition(job, BLOCK_JOB_STATUS_CREATED);
1bdc94
     aio_timer_init(qemu_get_aio_context(), &job->sleep_timer,
1bdc94
                    QEMU_CLOCK_REALTIME, SCALE_NS,
1bdc94
                    block_job_sleep_timer_cb, job);
1bdc94
@@ -1017,7 +961,7 @@ void *block_job_create(const char *job_id, const BlockJobDriver *driver,
1bdc94
 
1bdc94
 void block_job_early_fail(BlockJob *job)
1bdc94
 {
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_CREATED);
1bdc94
+    assert(job->job.status == JOB_STATUS_CREATED);
1bdc94
     block_job_decommission(job);
1bdc94
 }
1bdc94
 
1bdc94
@@ -1077,14 +1021,14 @@ void coroutine_fn block_job_pause_point(BlockJob *job)
1bdc94
     }
1bdc94
 
1bdc94
     if (block_job_should_pause(job) && !block_job_is_cancelled(job)) {
1bdc94
-        BlockJobStatus status = job->status;
1bdc94
-        block_job_state_transition(job, status == BLOCK_JOB_STATUS_READY ? \
1bdc94
-                                   BLOCK_JOB_STATUS_STANDBY :           \
1bdc94
-                                   BLOCK_JOB_STATUS_PAUSED);
1bdc94
+        JobStatus status = job->job.status;
1bdc94
+        job_state_transition(&job->job, status == JOB_STATUS_READY
1bdc94
+                                        ? JOB_STATUS_STANDBY
1bdc94
+                                        : JOB_STATUS_PAUSED);
1bdc94
         job->paused = true;
1bdc94
         block_job_do_yield(job, -1);
1bdc94
         job->paused = false;
1bdc94
-        block_job_state_transition(job, status);
1bdc94
+        job_state_transition(&job->job, status);
1bdc94
     }
1bdc94
 
1bdc94
     if (job->driver->resume) {
1bdc94
@@ -1176,7 +1120,7 @@ void block_job_iostatus_reset(BlockJob *job)
1bdc94
 
1bdc94
 void block_job_event_ready(BlockJob *job)
1bdc94
 {
1bdc94
-    block_job_state_transition(job, BLOCK_JOB_STATUS_READY);
1bdc94
+    job_state_transition(&job->job, JOB_STATUS_READY);
1bdc94
     job->ready = true;
1bdc94
 
1bdc94
     if (block_job_is_internal(job)) {
1bdc94
diff --git a/include/block/blockjob.h b/include/block/blockjob.h
1bdc94
index 10bd9f7..01cdee6 100644
1bdc94
--- a/include/block/blockjob.h
1bdc94
+++ b/include/block/blockjob.h
1bdc94
@@ -147,9 +147,6 @@ typedef struct BlockJob {
1bdc94
      */
1bdc94
     QEMUTimer sleep_timer;
1bdc94
 
1bdc94
-    /** Current state; See @BlockJobStatus for details. */
1bdc94
-    BlockJobStatus status;
1bdc94
-
1bdc94
     /** True if this job should automatically finalize itself */
1bdc94
     bool auto_finalize;
1bdc94
 
1bdc94
diff --git a/include/qemu/job.h b/include/qemu/job.h
1bdc94
index bae2b09..0b78778 100644
1bdc94
--- a/include/qemu/job.h
1bdc94
+++ b/include/qemu/job.h
1bdc94
@@ -41,6 +41,9 @@ typedef struct Job {
1bdc94
     /** The type of this job. */
1bdc94
     const JobDriver *driver;
1bdc94
 
1bdc94
+    /** Current state; See @JobStatus for details. */
1bdc94
+    JobStatus status;
1bdc94
+
1bdc94
     /** Element of the list of jobs */
1bdc94
     QLIST_ENTRY(Job) job_list;
1bdc94
 } Job;
1bdc94
@@ -90,4 +93,14 @@ Job *job_next(Job *job);
1bdc94
  */
1bdc94
 Job *job_get(const char *id);
1bdc94
 
1bdc94
+/**
1bdc94
+ * Check whether the verb @verb can be applied to @job in its current state.
1bdc94
+ * Returns 0 if the verb can be applied; otherwise errp is set and -EPERM
1bdc94
+ * returned.
1bdc94
+ */
1bdc94
+int job_apply_verb(Job *job, JobVerb verb, Error **errp);
1bdc94
+
1bdc94
+/* TODO To be removed from the public interface */
1bdc94
+void job_state_transition(Job *job, JobStatus s1);
1bdc94
+
1bdc94
 #endif
1bdc94
diff --git a/job.c b/job.c
1bdc94
index e57303c..b049a32 100644
1bdc94
--- a/job.c
1bdc94
+++ b/job.c
1bdc94
@@ -28,9 +28,63 @@
1bdc94
 #include "qapi/error.h"
1bdc94
 #include "qemu/job.h"
1bdc94
 #include "qemu/id.h"
1bdc94
+#include "trace-root.h"
1bdc94
 
1bdc94
 static QLIST_HEAD(, Job) jobs = QLIST_HEAD_INITIALIZER(jobs);
1bdc94
 
1bdc94
+/* Job State Transition Table */
1bdc94
+bool JobSTT[JOB_STATUS__MAX][JOB_STATUS__MAX] = {
1bdc94
+                                    /* U, C, R, P, Y, S, W, D, X, E, N */
1bdc94
+    /* U: */ [JOB_STATUS_UNDEFINED] = {0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0},
1bdc94
+    /* C: */ [JOB_STATUS_CREATED]   = {0, 0, 1, 0, 0, 0, 0, 0, 1, 0, 1},
1bdc94
+    /* R: */ [JOB_STATUS_RUNNING]   = {0, 0, 0, 1, 1, 0, 1, 0, 1, 0, 0},
1bdc94
+    /* P: */ [JOB_STATUS_PAUSED]    = {0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0},
1bdc94
+    /* Y: */ [JOB_STATUS_READY]     = {0, 0, 0, 0, 0, 1, 1, 0, 1, 0, 0},
1bdc94
+    /* S: */ [JOB_STATUS_STANDBY]   = {0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0},
1bdc94
+    /* W: */ [JOB_STATUS_WAITING]   = {0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0},
1bdc94
+    /* D: */ [JOB_STATUS_PENDING]   = {0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0},
1bdc94
+    /* X: */ [JOB_STATUS_ABORTING]  = {0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0},
1bdc94
+    /* E: */ [JOB_STATUS_CONCLUDED] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
1bdc94
+    /* N: */ [JOB_STATUS_NULL]      = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
1bdc94
+};
1bdc94
+
1bdc94
+bool JobVerbTable[JOB_VERB__MAX][JOB_STATUS__MAX] = {
1bdc94
+                                    /* U, C, R, P, Y, S, W, D, X, E, N */
1bdc94
+    [JOB_VERB_CANCEL]               = {0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0},
1bdc94
+    [JOB_VERB_PAUSE]                = {0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0},
1bdc94
+    [JOB_VERB_RESUME]               = {0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0},
1bdc94
+    [JOB_VERB_SET_SPEED]            = {0, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0},
1bdc94
+    [JOB_VERB_COMPLETE]             = {0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0},
1bdc94
+    [JOB_VERB_FINALIZE]             = {0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0},
1bdc94
+    [JOB_VERB_DISMISS]              = {0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0},
1bdc94
+};
1bdc94
+
1bdc94
+/* TODO Make static once the whole state machine is in job.c */
1bdc94
+void job_state_transition(Job *job, JobStatus s1)
1bdc94
+{
1bdc94
+    JobStatus s0 = job->status;
1bdc94
+    assert(s1 >= 0 && s1 <= JOB_STATUS__MAX);
1bdc94
+    trace_job_state_transition(job, /* TODO re-enable: job->ret */ 0,
1bdc94
+                               JobSTT[s0][s1] ? "allowed" : "disallowed",
1bdc94
+                               JobStatus_str(s0), JobStatus_str(s1));
1bdc94
+    assert(JobSTT[s0][s1]);
1bdc94
+    job->status = s1;
1bdc94
+}
1bdc94
+
1bdc94
+int job_apply_verb(Job *job, JobVerb verb, Error **errp)
1bdc94
+{
1bdc94
+    JobStatus s0 = job->status;
1bdc94
+    assert(verb >= 0 && verb <= JOB_VERB__MAX);
1bdc94
+    trace_job_apply_verb(job, JobStatus_str(s0), JobVerb_str(verb),
1bdc94
+                         JobVerbTable[verb][s0] ? "allowed" : "prohibited");
1bdc94
+    if (JobVerbTable[verb][s0]) {
1bdc94
+        return 0;
1bdc94
+    }
1bdc94
+    error_setg(errp, "Job '%s' in state '%s' cannot accept command verb '%s'",
1bdc94
+               job->id, JobStatus_str(s0), JobVerb_str(verb));
1bdc94
+    return -EPERM;
1bdc94
+}
1bdc94
+
1bdc94
 JobType job_type(const Job *job)
1bdc94
 {
1bdc94
     return job->driver->job_type;
1bdc94
@@ -81,6 +135,8 @@ void *job_create(const char *job_id, const JobDriver *driver, Error **errp)
1bdc94
     job->driver        = driver;
1bdc94
     job->id            = g_strdup(job_id);
1bdc94
 
1bdc94
+    job_state_transition(job, JOB_STATUS_CREATED);
1bdc94
+
1bdc94
     QLIST_INSERT_HEAD(&jobs, job, job_list);
1bdc94
 
1bdc94
     return job;
1bdc94
diff --git a/qapi/block-core.json b/qapi/block-core.json
1bdc94
index 607e5c5..85bf353 100644
1bdc94
--- a/qapi/block-core.json
1bdc94
+++ b/qapi/block-core.json
1bdc94
@@ -1068,9 +1068,9 @@
1bdc94
   'data': ['commit', 'stream', 'mirror', 'backup'] }
1bdc94
 
1bdc94
 ##
1bdc94
-# @BlockJobVerb:
1bdc94
+# @JobVerb:
1bdc94
 #
1bdc94
-# Represents command verbs that can be applied to a blockjob.
1bdc94
+# Represents command verbs that can be applied to a job.
1bdc94
 #
1bdc94
 # @cancel: see @block-job-cancel
1bdc94
 #
1bdc94
@@ -1088,14 +1088,14 @@
1bdc94
 #
1bdc94
 # Since: 2.12
1bdc94
 ##
1bdc94
-{ 'enum': 'BlockJobVerb',
1bdc94
+{ 'enum': 'JobVerb',
1bdc94
   'data': ['cancel', 'pause', 'resume', 'set-speed', 'complete', 'dismiss',
1bdc94
            'finalize' ] }
1bdc94
 
1bdc94
 ##
1bdc94
-# @BlockJobStatus:
1bdc94
+# @JobStatus:
1bdc94
 #
1bdc94
-# Indicates the present state of a given blockjob in its lifetime.
1bdc94
+# Indicates the present state of a given job in its lifetime.
1bdc94
 #
1bdc94
 # @undefined: Erroneous, default state. Should not ever be visible.
1bdc94
 #
1bdc94
@@ -1134,7 +1134,7 @@
1bdc94
 #
1bdc94
 # Since: 2.12
1bdc94
 ##
1bdc94
-{ 'enum': 'BlockJobStatus',
1bdc94
+{ 'enum': 'JobStatus',
1bdc94
   'data': ['undefined', 'created', 'running', 'paused', 'ready', 'standby',
1bdc94
            'waiting', 'pending', 'aborting', 'concluded', 'null' ] }
1bdc94
 
1bdc94
@@ -1184,7 +1184,7 @@
1bdc94
   'data': {'type': 'str', 'device': 'str', 'len': 'int',
1bdc94
            'offset': 'int', 'busy': 'bool', 'paused': 'bool', 'speed': 'int',
1bdc94
            'io-status': 'BlockDeviceIoStatus', 'ready': 'bool',
1bdc94
-           'status': 'BlockJobStatus',
1bdc94
+           'status': 'JobStatus',
1bdc94
            'auto-finalize': 'bool', 'auto-dismiss': 'bool',
1bdc94
            '*error': 'str' } }
1bdc94
 
1bdc94
@@ -2416,7 +2416,7 @@
1bdc94
 # QEMU 2.12+ job lifetime management semantics.
1bdc94
 #
1bdc94
 # This command will refuse to operate on any job that has not yet reached
1bdc94
-# its terminal state, BLOCK_JOB_STATUS_CONCLUDED. For jobs that make use of
1bdc94
+# its terminal state, JOB_STATUS_CONCLUDED. For jobs that make use of the
1bdc94
 # BLOCK_JOB_READY event, block-job-cancel or block-job-complete will still need
1bdc94
 # to be used as appropriate.
1bdc94
 #
1bdc94
diff --git a/tests/test-blockjob.c b/tests/test-blockjob.c
1bdc94
index b820261..6ccd585 100644
1bdc94
--- a/tests/test-blockjob.c
1bdc94
+++ b/tests/test-blockjob.c
1bdc94
@@ -211,7 +211,7 @@ static CancelJob *create_common(BlockJob **pjob)
1bdc94
     job = mk_job(blk, "Steve", &test_cancel_driver, true,
1bdc94
                  BLOCK_JOB_MANUAL_FINALIZE | BLOCK_JOB_MANUAL_DISMISS);
1bdc94
     block_job_ref(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_CREATED);
1bdc94
+    assert(job->job.status == JOB_STATUS_CREATED);
1bdc94
     s = container_of(job, CancelJob, common);
1bdc94
     s->blk = blk;
1bdc94
 
1bdc94
@@ -223,15 +223,14 @@ static void cancel_common(CancelJob *s)
1bdc94
 {
1bdc94
     BlockJob *job = &s->common;
1bdc94
     BlockBackend *blk = s->blk;
1bdc94
-    BlockJobStatus sts = job->status;
1bdc94
+    JobStatus sts = job->job.status;
1bdc94
 
1bdc94
     block_job_cancel_sync(job);
1bdc94
-    if ((sts != BLOCK_JOB_STATUS_CREATED) &&
1bdc94
-        (sts != BLOCK_JOB_STATUS_CONCLUDED)) {
1bdc94
+    if (sts != JOB_STATUS_CREATED && sts != JOB_STATUS_CONCLUDED) {
1bdc94
         BlockJob *dummy = job;
1bdc94
         block_job_dismiss(&dummy, &error_abort);
1bdc94
     }
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_NULL);
1bdc94
+    assert(job->job.status == JOB_STATUS_NULL);
1bdc94
     block_job_unref(job);
1bdc94
     destroy_blk(blk);
1bdc94
 }
1bdc94
@@ -253,7 +252,7 @@ static void test_cancel_running(void)
1bdc94
     s = create_common(&job;;
1bdc94
 
1bdc94
     block_job_start(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_RUNNING);
1bdc94
+    assert(job->job.status == JOB_STATUS_RUNNING);
1bdc94
 
1bdc94
     cancel_common(s);
1bdc94
 }
1bdc94
@@ -266,11 +265,11 @@ static void test_cancel_paused(void)
1bdc94
     s = create_common(&job;;
1bdc94
 
1bdc94
     block_job_start(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_RUNNING);
1bdc94
+    assert(job->job.status == JOB_STATUS_RUNNING);
1bdc94
 
1bdc94
     block_job_user_pause(job, &error_abort);
1bdc94
     block_job_enter(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_PAUSED);
1bdc94
+    assert(job->job.status == JOB_STATUS_PAUSED);
1bdc94
 
1bdc94
     cancel_common(s);
1bdc94
 }
1bdc94
@@ -283,11 +282,11 @@ static void test_cancel_ready(void)
1bdc94
     s = create_common(&job;;
1bdc94
 
1bdc94
     block_job_start(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_RUNNING);
1bdc94
+    assert(job->job.status == JOB_STATUS_RUNNING);
1bdc94
 
1bdc94
     s->should_converge = true;
1bdc94
     block_job_enter(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_READY);
1bdc94
+    assert(job->job.status == JOB_STATUS_READY);
1bdc94
 
1bdc94
     cancel_common(s);
1bdc94
 }
1bdc94
@@ -300,15 +299,15 @@ static void test_cancel_standby(void)
1bdc94
     s = create_common(&job;;
1bdc94
 
1bdc94
     block_job_start(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_RUNNING);
1bdc94
+    assert(job->job.status == JOB_STATUS_RUNNING);
1bdc94
 
1bdc94
     s->should_converge = true;
1bdc94
     block_job_enter(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_READY);
1bdc94
+    assert(job->job.status == JOB_STATUS_READY);
1bdc94
 
1bdc94
     block_job_user_pause(job, &error_abort);
1bdc94
     block_job_enter(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_STANDBY);
1bdc94
+    assert(job->job.status == JOB_STATUS_STANDBY);
1bdc94
 
1bdc94
     cancel_common(s);
1bdc94
 }
1bdc94
@@ -321,18 +320,18 @@ static void test_cancel_pending(void)
1bdc94
     s = create_common(&job;;
1bdc94
 
1bdc94
     block_job_start(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_RUNNING);
1bdc94
+    assert(job->job.status == JOB_STATUS_RUNNING);
1bdc94
 
1bdc94
     s->should_converge = true;
1bdc94
     block_job_enter(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_READY);
1bdc94
+    assert(job->job.status == JOB_STATUS_READY);
1bdc94
 
1bdc94
     block_job_complete(job, &error_abort);
1bdc94
     block_job_enter(job);
1bdc94
     while (!s->completed) {
1bdc94
         aio_poll(qemu_get_aio_context(), true);
1bdc94
     }
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_PENDING);
1bdc94
+    assert(job->job.status == JOB_STATUS_PENDING);
1bdc94
 
1bdc94
     cancel_common(s);
1bdc94
 }
1bdc94
@@ -345,21 +344,21 @@ static void test_cancel_concluded(void)
1bdc94
     s = create_common(&job;;
1bdc94
 
1bdc94
     block_job_start(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_RUNNING);
1bdc94
+    assert(job->job.status == JOB_STATUS_RUNNING);
1bdc94
 
1bdc94
     s->should_converge = true;
1bdc94
     block_job_enter(job);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_READY);
1bdc94
+    assert(job->job.status == JOB_STATUS_READY);
1bdc94
 
1bdc94
     block_job_complete(job, &error_abort);
1bdc94
     block_job_enter(job);
1bdc94
     while (!s->completed) {
1bdc94
         aio_poll(qemu_get_aio_context(), true);
1bdc94
     }
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_PENDING);
1bdc94
+    assert(job->job.status == JOB_STATUS_PENDING);
1bdc94
 
1bdc94
     block_job_finalize(job, &error_abort);
1bdc94
-    assert(job->status == BLOCK_JOB_STATUS_CONCLUDED);
1bdc94
+    assert(job->job.status == JOB_STATUS_CONCLUDED);
1bdc94
 
1bdc94
     cancel_common(s);
1bdc94
 }
1bdc94
diff --git a/trace-events b/trace-events
1bdc94
index ed71f44..2507e13 100644
1bdc94
--- a/trace-events
1bdc94
+++ b/trace-events
1bdc94
@@ -104,6 +104,10 @@ gdbstub_err_invalid_rle(void) "got invalid RLE sequence"
1bdc94
 gdbstub_err_checksum_invalid(uint8_t ch) "got invalid command checksum digit: 0x%02x"
1bdc94
 gdbstub_err_checksum_incorrect(uint8_t expected, uint8_t got) "got command packet with incorrect checksum, expected=0x%02x, received=0x%02x"
1bdc94
 
1bdc94
+# job.c
1bdc94
+job_state_transition(void *job,  int ret, const char *legal, const char *s0, const char *s1) "job %p (ret: %d) attempting %s transition (%s-->%s)"
1bdc94
+job_apply_verb(void *job, const char *state, const char *verb, const char *legal) "job %p in state %s; applying verb %s (%s)"
1bdc94
+
1bdc94
 ### Guest events, keep at bottom
1bdc94
 
1bdc94
 
1bdc94
-- 
1bdc94
1.8.3.1
1bdc94