- kvm-blkdebug-Add-delay-ns-option.patch [RHEL-121686] - kvm-block-Add-blk_co_start-end_request-and-BDRV_REQ_NO_Q.patch [RHEL-121686] - kvm-block-Add-flags-parameter-to-blk_-_pdiscard.patch [RHEL-121686] - kvm-ide-Minimal-fix-for-deadlock-between-TRIM-and-drain.patch [RHEL-121686] - kvm-ide-Clean-up-ide_trim_co_entry-to-be-idiomatic-corou.patch [RHEL-121686] - kvm-ide-test-Factor-out-wait_dma_completion.patch [RHEL-121686] - kvm-ide-test-Test-reset-during-TRIM.patch [RHEL-121686] - kvm-block-graph-lock-fix-missed-wakeup-in-bdrv_graph_co_.patch [RHEL-186384] - kvm-block-curl-fix-curl-internal-handles-handling.patch [RHEL-186384] - kvm-block-curl.c-Use-explicit-long-constants-in-curl_eas.patch [RHEL-186384] - kvm-block-curl.c-Fix-CURLOPT_VERBOSE-parameter-type.patch [RHEL-186384] - kvm-block-curl-fix-concurrent-completion-handling.patch [RHEL-186384] - kvm-block-curl-free-s-password-in-cleanup-paths.patch [RHEL-186384] - kvm-nvme-Kick-and-check-completions-in-BDS-context.patch [RHEL-186384] - kvm-nvme-Note-in-which-AioContext-some-functions-run.patch [RHEL-186384] - kvm-block-remove-detached-header-option-from-opts-after-.patch [RHEL-186384] - kvm-block-fix-luks-amend-when-run-in-coroutine.patch [RHEL-186384] - kvm-qed-Don-t-try-to-flush-during-incoming-migration.patch [RHEL-186384] - kvm-block-vmdk-fix-OOB-read-in-vmdk_read_extent.patch [RHEL-186384] - kvm-block-throttle-groups-fix-deadlock-with-iolimits-and.patch [RHEL-186384] - kvm-throttle-group-Fix-race-condition-in-throttle_group_.patch [RHEL-186384] - kvm-qemu-img-Fix-amend-option-parse-error-handling.patch [RHEL-186384] - kvm-qemu-img-rebase-don-t-exceed-IO_BUF_SIZE-in-one-oper.patch [RHEL-186384] - kvm-python-backport-drop-Python3.6-workarounds.patch [RHEL-186384] - kvm-python-backport-Remove-deprecated-get_event_loop-cal.patch [RHEL-186384] - kvm-python-backport-avoid-creating-additional-event-loop.patch [RHEL-186384] - kvm-iotests-147-ensure-temporary-sockets-are-closed-befo.patch [RHEL-186384] - kvm-iotests-151-ensure-subprocesses-are-cleaned-up.patch [RHEL-186384] - kvm-tests-qemu-iotest-fix-iotest-024-with-qed-images.patch [RHEL-186384] - kvm-tests-qemu-iotests-Fix-check-for-existing-file-in-_r.patch [RHEL-186384] - kvm-async-access-bottom-half-flags-with-qatomic_read.patch [RHEL-186384] - kvm-block-linux-aio-bound-ioq_submit-recursion-depth.patch [RHEL-186384] - kvm-block-io-fallback-to-bounce-buffer-if-BLKZEROOUT-is-.patch [RHEL-186384] - kvm-file-posix-populate-pwrite_zeroes_alignment.patch [RHEL-186384] - kvm-block-use-pwrite_zeroes_alignment-when-writing-first.patch [RHEL-186384] - kvm-iotests-add-Linux-loop-device-image-creation-test.patch [RHEL-186384] - kvm-virtio-Fix-crash-when-sriov-pf-is-set-for-non-PCI-Ex.patch [RHEL-186384] - kvm-virtio-scsi-pass-the-same-cdb_size-to-virtio_scsi_po.patch [RHEL-186384] - kvm-hw-scsi-avoid-deadlock-upon-TMF-request-cancelling-w.patch [RHEL-186384] - kvm-virtio-blk-fix-zone-report-buffer-out-of-memory-CVE-.patch [RHEL-186384] - kvm-ide-Fix-potential-assertion-failure-on-VM-stop-for-P.patch [RHEL-186384] - kvm-block-Create-DEFAULT_BLOCK_CONF-macro.patch [RHEL-186384] - kvm-block-Add-more-defaults-to-DEFAULT_BLOCK_CONF.patch [RHEL-186384] - kvm-block-mirror-check-range-when-setting-zero-bitmap-fo.patch [RHEL-186384] - kvm-iotests-test-active-mirror-with-unaligned-small-writ.patch [RHEL-186384] - kvm-block-mirror-fix-assertion-failure-upon-duplicate-co.patch [RHEL-186384] - kvm-commit-Drain-nodes-across-all-of-bdrv_commit.patch [RHEL-186384] - kvm-qemu-io-Add-aio_discard-command.patch [RHEL-186384] - kvm-qcow2-Fix-corruption-on-discard-during-write-with-CO.patch [RHEL-186384] - kvm-iotests-046-Test-that-discard-write_zeroes-wait-for-.patch [RHEL-186384] - kvm-qcow2-Fix-data-loss-on-zero-write-with-detect-zeroes.patch [RHEL-186384] - kvm-block-Fix-crash-after-setting-latency-historygram-wi.patch [RHEL-186384] - Resolves: RHEL-121686 (qemu-kvm hung during drain after double pause) - Resolves: RHEL-186384 (virt-storage: Backport stable branch fixes)
211 lines
8.4 KiB
Diff
211 lines
8.4 KiB
Diff
From 3739feb5280370c3439030e1f1fe0ec0d0eb2ccb Mon Sep 17 00:00:00 2001
|
|
From: John Snow <jsnow@redhat.com>
|
|
Date: Wed, 3 Sep 2025 01:06:30 -0400
|
|
Subject: [PATCH 26/52] python: backport 'avoid creating additional event loops
|
|
per thread'
|
|
MIME-Version: 1.0
|
|
Content-Type: text/plain; charset=UTF-8
|
|
Content-Transfer-Encoding: 8bit
|
|
|
|
RH-Author: Kevin Wolf <kwolf@redhat.com>
|
|
RH-MergeRequest: 504: virt-storage: Backport stable branch fixes
|
|
RH-Jira: RHEL-186384
|
|
RH-Acked-by: Stefan Hajnoczi <stefanha@redhat.com>
|
|
RH-Acked-by: Miroslav Rezanina <mrezanin@redhat.com>
|
|
RH-Commit: [19/45] 1f870d1d45d5158c428bac46820d5aa7621f810f (kmwolf/centos-qemu-kvm)
|
|
|
|
This commit is two backports squashed into one to avoid regressions.
|
|
|
|
python: *really* remove get_event_loop
|
|
|
|
A prior commit, aa1ff990, switched away from using get_event_loop *by
|
|
default*, but this is not good enough to avoid deprecation warnings as
|
|
`asyncio.get_event_loop_policy().get_event_loop()` is *also*
|
|
deprecated. Replace this mechanism with explicit calls to
|
|
asyncio.get_new_loop() and revise the cleanup mechanisms in __del__ to
|
|
match.
|
|
|
|
python: avoid creating additional event loops per thread
|
|
|
|
"Too hasty by far!", commit 21ce2ee4 attempted to avoid deprecated
|
|
behavior altogether by calling new_event_loop() directly if there was no
|
|
loop currently running, but this has the unfortunate side effect of
|
|
potentially creating multiple event loops per thread if tests
|
|
instantiate multiple QMP connections in a single thread. This behavior
|
|
is apparently not well-defined and causes problems in some, but not all,
|
|
combinations of Python interpreter version and platform environment.
|
|
|
|
Partially revert to Daniel Berrange's original patch, which calls
|
|
get_event_loop and simply suppresses the deprecation warning in
|
|
Python<=3.13. This time, however, additionally register new loops
|
|
created with new_event_loop() so that future calls to get_event_loop()
|
|
will return the loop already created.
|
|
|
|
Reported-by: Richard W.M. Jones <rjones@redhat.com>
|
|
Reported-by: Daniel P. Berrangé <berrange@redhat.com>
|
|
Signed-off-by: John Snow <jsnow@redhat.com>
|
|
cherry picked from commit python-qemu-qmp@21ce2ee4f2df87efe84a27b9c5112487f4670622
|
|
cherry picked from commit python-qemu-qmp@c08fb82b38212956ccffc03fc6d015c3979f42fe
|
|
Signed-off-by: John Snow <jsnow@redhat.com>
|
|
Reviewed-by: Daniel P. Berrangé <berrange@redhat.com>
|
|
(cherry picked from commit 85f223e5b031eb8ab63fbca314a4fb296a3a2632)
|
|
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
|
|
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
|
|
---
|
|
python/qemu/qmp/legacy.py | 46 +++++++++++++++++++++++---------------
|
|
python/qemu/qmp/qmp_tui.py | 10 ++-------
|
|
python/qemu/qmp/util.py | 27 ++++++++++++++++++++++
|
|
3 files changed, 57 insertions(+), 26 deletions(-)
|
|
|
|
diff --git a/python/qemu/qmp/legacy.py b/python/qemu/qmp/legacy.py
|
|
index ea9b8032c3..c732212c04 100644
|
|
--- a/python/qemu/qmp/legacy.py
|
|
+++ b/python/qemu/qmp/legacy.py
|
|
@@ -38,6 +38,7 @@
|
|
from .error import QMPError
|
|
from .protocol import Runstate, SocketAddrT
|
|
from .qmp_client import QMPClient
|
|
+from .util import get_or_create_event_loop
|
|
|
|
|
|
#: QMPMessage is an entire QMP message of any kind.
|
|
@@ -86,17 +87,13 @@ def __init__(self,
|
|
"server argument should be False when passing a socket")
|
|
|
|
self._qmp = QMPClient(nickname)
|
|
-
|
|
- try:
|
|
- self._aloop = asyncio.get_running_loop()
|
|
- except RuntimeError:
|
|
- # No running loop; since this is a sync shim likely to be
|
|
- # used in fully sync programs, create one if neccessary.
|
|
- self._aloop = asyncio.get_event_loop_policy().get_event_loop()
|
|
-
|
|
self._address = address
|
|
self._timeout: Optional[float] = None
|
|
|
|
+ # This is a sync shim intended for use in fully synchronous
|
|
+ # programs. Create and set an event loop if necessary.
|
|
+ self._aloop = get_or_create_event_loop()
|
|
+
|
|
if server:
|
|
assert not isinstance(self._address, socket.socket)
|
|
self._sync(self._qmp.start_server(self._address))
|
|
@@ -310,17 +307,30 @@ def send_fd_scm(self, fd: int) -> None:
|
|
self._qmp.send_fd_scm(fd)
|
|
|
|
def __del__(self) -> None:
|
|
- if self._qmp.runstate == Runstate.IDLE:
|
|
- return
|
|
+ if self._qmp.runstate != Runstate.IDLE:
|
|
+ self._qmp.logger.warning(
|
|
+ "QEMUMonitorProtocol object garbage collected without a prior "
|
|
+ "call to close()"
|
|
+ )
|
|
|
|
if not self._aloop.is_running():
|
|
- self.close()
|
|
- else:
|
|
- # Garbage collection ran while the event loop was running.
|
|
- # Nothing we can do about it now, but if we don't raise our
|
|
- # own error, the user will be treated to a lot of traceback
|
|
- # they might not understand.
|
|
+ if self._qmp.runstate != Runstate.IDLE:
|
|
+ # If the user neglected to close the QMP session and we
|
|
+ # are not currently running in an asyncio context, we
|
|
+ # have the opportunity to close the QMP session. If we
|
|
+ # do not do this, the error messages presented over
|
|
+ # dangling async resources may not make any sense to the
|
|
+ # user.
|
|
+ self.close()
|
|
+
|
|
+ if self._qmp.runstate != Runstate.IDLE:
|
|
+ # If QMP is still not quiesced, it means that the garbage
|
|
+ # collector ran from a context within the event loop and we
|
|
+ # are simply too late to take any corrective action. Raise
|
|
+ # our own error to give meaningful feedback to the user in
|
|
+ # order to prevent pages of asyncio stacktrace jargon.
|
|
raise QMPError(
|
|
- "QEMUMonitorProtocol.close()"
|
|
- " was not called before object was garbage collected"
|
|
+ "QEMUMonitorProtocol.close() was not called before object was "
|
|
+ "garbage collected, and could not be closed due to GC running "
|
|
+ "in the event loop"
|
|
)
|
|
diff --git a/python/qemu/qmp/qmp_tui.py b/python/qemu/qmp/qmp_tui.py
|
|
index 651f611316..89b0f5e081 100644
|
|
--- a/python/qemu/qmp/qmp_tui.py
|
|
+++ b/python/qemu/qmp/qmp_tui.py
|
|
@@ -40,7 +40,7 @@
|
|
from .message import DeserializationError, Message, UnexpectedTypeError
|
|
from .protocol import ConnectError, Runstate
|
|
from .qmp_client import ExecInterruptedError, QMPClient
|
|
-from .util import pretty_traceback
|
|
+from .util import get_or_create_event_loop, pretty_traceback
|
|
|
|
|
|
# The name of the signal that is used to update the history list
|
|
@@ -376,13 +376,7 @@ def run(self, debug: bool = False) -> None:
|
|
"""
|
|
screen = urwid.raw_display.Screen()
|
|
screen.set_terminal_properties(256)
|
|
-
|
|
- try:
|
|
- self.aloop = asyncio.get_running_loop()
|
|
- except RuntimeError:
|
|
- # No running asyncio event loop. Create one if necessary.
|
|
- self.aloop = asyncio.get_event_loop_policy().get_event_loop()
|
|
-
|
|
+ self.aloop = get_or_create_event_loop()
|
|
self.aloop.set_debug(debug)
|
|
|
|
# Gracefully handle SIGTERM and SIGINT signals
|
|
diff --git a/python/qemu/qmp/util.py b/python/qemu/qmp/util.py
|
|
index 0b3e781373..47ec39a8b5 100644
|
|
--- a/python/qemu/qmp/util.py
|
|
+++ b/python/qemu/qmp/util.py
|
|
@@ -10,6 +10,7 @@
|
|
import sys
|
|
import traceback
|
|
from typing import TypeVar, cast
|
|
+import warnings
|
|
|
|
|
|
T = TypeVar('T')
|
|
@@ -20,6 +21,32 @@
|
|
# --------------------------
|
|
|
|
|
|
+def get_or_create_event_loop() -> asyncio.AbstractEventLoop:
|
|
+ """
|
|
+ Return this thread's current event loop, or create a new one.
|
|
+
|
|
+ This function behaves similarly to asyncio.get_event_loop() in
|
|
+ Python<=3.13, where if there is no event loop currently associated
|
|
+ with the current context, it will create and register one. It should
|
|
+ generally not be used in any asyncio-native applications.
|
|
+ """
|
|
+ try:
|
|
+ with warnings.catch_warnings():
|
|
+ # Python <= 3.13 will trigger deprecation warnings if no
|
|
+ # event loop is set, but will create and set a new loop.
|
|
+ warnings.simplefilter("ignore")
|
|
+ loop = asyncio.get_event_loop()
|
|
+ except RuntimeError:
|
|
+ # Python 3.14+: No event loop set for this thread,
|
|
+ # create and set one.
|
|
+ loop = asyncio.new_event_loop()
|
|
+ # Set this loop as the current thread's loop, to be returned
|
|
+ # by calls to get_event_loop() in the future.
|
|
+ asyncio.set_event_loop(loop)
|
|
+
|
|
+ return loop
|
|
+
|
|
+
|
|
async def flush(writer: asyncio.StreamWriter) -> None:
|
|
"""
|
|
Utility function to ensure a StreamWriter is *fully* drained.
|
|
--
|
|
2.52.0
|
|
|