job: Add lifecycle QMP commands
This adds QMP commands that control the transition between states of the job lifecycle. Signed-off-by: Kevin Wolf <kwolf@redhat.com>
This commit is contained in:
parent
1dac83f1a1
commit
1a90bc8128
@ -1370,6 +1370,7 @@ S: Supported
|
||||
F: blockjob.c
|
||||
F: include/block/blockjob.h
|
||||
F: job.c
|
||||
F: job-qmp.c
|
||||
F: include/block/job.h
|
||||
F: block/backup.c
|
||||
F: block/commit.c
|
||||
|
@ -97,6 +97,7 @@ io-obj-y = io/
|
||||
ifeq ($(CONFIG_SOFTMMU),y)
|
||||
common-obj-y = blockdev.o blockdev-nbd.o block/
|
||||
common-obj-y += bootdevice.o iothread.o
|
||||
common-obj-y += job-qmp.o
|
||||
common-obj-y += net/
|
||||
common-obj-y += qdev-monitor.o device-hotplug.o
|
||||
common-obj-$(CONFIG_WIN32) += os-win32.o
|
||||
|
134
job-qmp.c
Normal file
134
job-qmp.c
Normal file
@ -0,0 +1,134 @@
|
||||
/*
|
||||
* QMP interface for background jobs
|
||||
*
|
||||
* Copyright (c) 2011 IBM Corp.
|
||||
* Copyright (c) 2012, 2018 Red Hat, Inc.
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
#include "qemu/osdep.h"
|
||||
#include "qemu-common.h"
|
||||
#include "qemu/job.h"
|
||||
#include "qapi/qapi-commands-job.h"
|
||||
#include "qapi/error.h"
|
||||
#include "trace-root.h"
|
||||
|
||||
/* Get a job using its ID and acquire its AioContext */
|
||||
static Job *find_job(const char *id, AioContext **aio_context, Error **errp)
|
||||
{
|
||||
Job *job;
|
||||
|
||||
*aio_context = NULL;
|
||||
|
||||
job = job_get(id);
|
||||
if (!job) {
|
||||
error_setg(errp, "Job not found");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
*aio_context = job->aio_context;
|
||||
aio_context_acquire(*aio_context);
|
||||
|
||||
return job;
|
||||
}
|
||||
|
||||
void qmp_job_cancel(const char *id, Error **errp)
|
||||
{
|
||||
AioContext *aio_context;
|
||||
Job *job = find_job(id, &aio_context, errp);
|
||||
|
||||
if (!job) {
|
||||
return;
|
||||
}
|
||||
|
||||
trace_qmp_job_cancel(job);
|
||||
job_user_cancel(job, true, errp);
|
||||
aio_context_release(aio_context);
|
||||
}
|
||||
|
||||
void qmp_job_pause(const char *id, Error **errp)
|
||||
{
|
||||
AioContext *aio_context;
|
||||
Job *job = find_job(id, &aio_context, errp);
|
||||
|
||||
if (!job) {
|
||||
return;
|
||||
}
|
||||
|
||||
trace_qmp_job_pause(job);
|
||||
job_user_pause(job, errp);
|
||||
aio_context_release(aio_context);
|
||||
}
|
||||
|
||||
void qmp_job_resume(const char *id, Error **errp)
|
||||
{
|
||||
AioContext *aio_context;
|
||||
Job *job = find_job(id, &aio_context, errp);
|
||||
|
||||
if (!job) {
|
||||
return;
|
||||
}
|
||||
|
||||
trace_qmp_job_resume(job);
|
||||
job_user_resume(job, errp);
|
||||
aio_context_release(aio_context);
|
||||
}
|
||||
|
||||
void qmp_job_complete(const char *id, Error **errp)
|
||||
{
|
||||
AioContext *aio_context;
|
||||
Job *job = find_job(id, &aio_context, errp);
|
||||
|
||||
if (!job) {
|
||||
return;
|
||||
}
|
||||
|
||||
trace_qmp_job_complete(job);
|
||||
job_complete(job, errp);
|
||||
aio_context_release(aio_context);
|
||||
}
|
||||
|
||||
void qmp_job_finalize(const char *id, Error **errp)
|
||||
{
|
||||
AioContext *aio_context;
|
||||
Job *job = find_job(id, &aio_context, errp);
|
||||
|
||||
if (!job) {
|
||||
return;
|
||||
}
|
||||
|
||||
trace_qmp_job_finalize(job);
|
||||
job_finalize(job, errp);
|
||||
aio_context_release(aio_context);
|
||||
}
|
||||
|
||||
void qmp_job_dismiss(const char *id, Error **errp)
|
||||
{
|
||||
AioContext *aio_context;
|
||||
Job *job = find_job(id, &aio_context, errp);
|
||||
|
||||
if (!job) {
|
||||
return;
|
||||
}
|
||||
|
||||
trace_qmp_job_dismiss(job);
|
||||
job_dismiss(&job, errp);
|
||||
aio_context_release(aio_context);
|
||||
}
|
@ -106,3 +106,102 @@
|
||||
{ 'event': 'JOB_STATUS_CHANGE',
|
||||
'data': { 'id': 'str',
|
||||
'status': 'JobStatus' } }
|
||||
|
||||
##
|
||||
# @job-pause:
|
||||
#
|
||||
# Pause an active job.
|
||||
#
|
||||
# This command returns immediately after marking the active job for pausing.
|
||||
# Pausing an already paused job is an error.
|
||||
#
|
||||
# The job will pause as soon as possible, which means transitioning into the
|
||||
# PAUSED state if it was RUNNING, or into STANDBY if it was READY. The
|
||||
# corresponding JOB_STATUS_CHANGE event will be emitted.
|
||||
#
|
||||
# Cancelling a paused job automatically resumes it.
|
||||
#
|
||||
# @id: The job identifier.
|
||||
#
|
||||
# Since: 2.13
|
||||
##
|
||||
{ 'command': 'job-pause', 'data': { 'id': 'str' } }
|
||||
|
||||
##
|
||||
# @job-resume:
|
||||
#
|
||||
# Resume a paused job.
|
||||
#
|
||||
# This command returns immediately after resuming a paused job. Resuming an
|
||||
# already running job is an error.
|
||||
#
|
||||
# @id : The job identifier.
|
||||
#
|
||||
# Since: 2.13
|
||||
##
|
||||
{ 'command': 'job-resume', 'data': { 'id': 'str' } }
|
||||
|
||||
##
|
||||
# @job-cancel:
|
||||
#
|
||||
# Instruct an active background job to cancel at the next opportunity.
|
||||
# This command returns immediately after marking the active job for
|
||||
# cancellation.
|
||||
#
|
||||
# The job will cancel as soon as possible and then emit a JOB_STATUS_CHANGE
|
||||
# event. Usually, the status will change to ABORTING, but it is possible that
|
||||
# a job successfully completes (e.g. because it was almost done and there was
|
||||
# no opportunity to cancel earlier than completing the job) and transitions to
|
||||
# PENDING instead.
|
||||
#
|
||||
# @id: The job identifier.
|
||||
#
|
||||
# Since: 2.13
|
||||
##
|
||||
{ 'command': 'job-cancel', 'data': { 'id': 'str' } }
|
||||
|
||||
|
||||
##
|
||||
# @job-complete:
|
||||
#
|
||||
# Manually trigger completion of an active job in the READY state.
|
||||
#
|
||||
# @id: The job identifier.
|
||||
#
|
||||
# Since: 2.13
|
||||
##
|
||||
{ 'command': 'job-complete', 'data': { 'id': 'str' } }
|
||||
|
||||
##
|
||||
# @job-dismiss:
|
||||
#
|
||||
# Deletes a job that is in the CONCLUDED state. This command only needs to be
|
||||
# run explicitly for jobs that don't have automatic dismiss enabled.
|
||||
#
|
||||
# This command will refuse to operate on any job that has not yet reached its
|
||||
# terminal state, JOB_STATUS_CONCLUDED. For jobs that make use of JOB_READY
|
||||
# event, job-cancel or job-complete will still need to be used as appropriate.
|
||||
#
|
||||
# @id: The job identifier.
|
||||
#
|
||||
# Since: 2.13
|
||||
##
|
||||
{ 'command': 'job-dismiss', 'data': { 'id': 'str' } }
|
||||
|
||||
##
|
||||
# @job-finalize:
|
||||
#
|
||||
# Instructs all jobs in a transaction (or a single job if it is not part of any
|
||||
# transaction) to finalize any graph changes and do any necessary cleanup. This
|
||||
# command requires that all involved jobs are in the PENDING state.
|
||||
#
|
||||
# For jobs in a transaction, instructing one job to finalize will force
|
||||
# ALL jobs in the transaction to finalize, so it is only necessary to instruct
|
||||
# a single member job to finalize.
|
||||
#
|
||||
# @id: The identifier of any job in the transaction, or of a job that is not
|
||||
# part of any transaction.
|
||||
#
|
||||
# Since: 2.13
|
||||
##
|
||||
{ 'command': 'job-finalize', 'data': { 'id': 'str' } }
|
||||
|
@ -109,6 +109,15 @@ job_state_transition(void *job, int ret, const char *legal, const char *s0, con
|
||||
job_apply_verb(void *job, const char *state, const char *verb, const char *legal) "job %p in state %s; applying verb %s (%s)"
|
||||
job_completed(void *job, int ret, int jret) "job %p ret %d corrected ret %d"
|
||||
|
||||
# job-qmp.c
|
||||
qmp_job_cancel(void *job) "job %p"
|
||||
qmp_job_pause(void *job) "job %p"
|
||||
qmp_job_resume(void *job) "job %p"
|
||||
qmp_job_complete(void *job) "job %p"
|
||||
qmp_job_finalize(void *job) "job %p"
|
||||
qmp_job_dismiss(void *job) "job %p"
|
||||
|
||||
|
||||
### Guest events, keep at bottom
|
||||
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user