mirror of
https://github.com/xemu-project/xemu.git
synced 2024-12-18 01:08:51 +00:00
Block layer patches
- coroutine: Fix crashes due to too large pool batch size - fdc: Prevent end-of-track overrun - nbd: MULTI_CONN for shared writable exports - iotests test runner improvements -----BEGIN PGP SIGNATURE----- iQJFBAABCAAvFiEE3D3rFZqa+V09dFb+fwmycsiPL9YFAmJ9KCkRHGt3b2xmQHJl ZGhhdC5jb20ACgkQfwmycsiPL9ZtSRAAmYDFBPqxfutpFXM7kIKwL6COXJC12MOx Tmu8cDiGB/jNChdi3kl6I5h5njzo3U0ZlL/Ign6EzHoeoXLAPSeUWmuRsARwsZ+A rL61gf6yrMjAo45FZuIS0GlMDk8BauRwPl9qPWeqQcrtOMYpxwZfyFGmcMpQgAOI MSC1I8p3FA7oJhGpKIHDPOjaZA97Lm2rLnDIwZ4f0YgssbybFBcFCXOQbhpsVhLy Tjp/L+qRUtna9xBsPHQvHZW0kITQbCQPdX+oVqqUmwzSvuHqfXKe1YppyPjBt/S0 H7nxtx4HOgP0lP5Kea+wbIRAk9Da5uaOW8hlMWRLShEKv1iTUenQSKteBB6CD03t GD9ze1kGoR9b6szw795BXxZxcWii0cn359lIVHeKR/U8zDuz5w3zhyl0klK8xeJy nj+JErLwQ7BD8kNR+7WAfXTF3tk2dQao1AvsBjn087KjMiJ/Mg8HY4K2zrjBUrHL DLTyAIjzct3BWJDZ02fb5jb8pHmIP3JO6m9Zvjm7ibP65BqJOwIXUTFpbgnrOg45 oFLDV4JgC4Hh4GEtdm+UhQE51A0VVW5pDaqWTdWkCcuk3QgxUdM3Wm3SW6pw1Gvb T0X0j5RgF/k3YrW576R/VIy6z4YPbzAtiG4O/zSlsujHoDcVNWnxApgSB/unaDh8 LNkFPGEMeSs= =JmTm -----END PGP SIGNATURE----- Merge tag 'for-upstream' of git://repo.or.cz/qemu/kevin into staging Block layer patches - coroutine: Fix crashes due to too large pool batch size - fdc: Prevent end-of-track overrun - nbd: MULTI_CONN for shared writable exports - iotests test runner improvements # -----BEGIN PGP SIGNATURE----- # # iQJFBAABCAAvFiEE3D3rFZqa+V09dFb+fwmycsiPL9YFAmJ9KCkRHGt3b2xmQHJl # ZGhhdC5jb20ACgkQfwmycsiPL9ZtSRAAmYDFBPqxfutpFXM7kIKwL6COXJC12MOx # Tmu8cDiGB/jNChdi3kl6I5h5njzo3U0ZlL/Ign6EzHoeoXLAPSeUWmuRsARwsZ+A # rL61gf6yrMjAo45FZuIS0GlMDk8BauRwPl9qPWeqQcrtOMYpxwZfyFGmcMpQgAOI # MSC1I8p3FA7oJhGpKIHDPOjaZA97Lm2rLnDIwZ4f0YgssbybFBcFCXOQbhpsVhLy # Tjp/L+qRUtna9xBsPHQvHZW0kITQbCQPdX+oVqqUmwzSvuHqfXKe1YppyPjBt/S0 # H7nxtx4HOgP0lP5Kea+wbIRAk9Da5uaOW8hlMWRLShEKv1iTUenQSKteBB6CD03t # GD9ze1kGoR9b6szw795BXxZxcWii0cn359lIVHeKR/U8zDuz5w3zhyl0klK8xeJy # nj+JErLwQ7BD8kNR+7WAfXTF3tk2dQao1AvsBjn087KjMiJ/Mg8HY4K2zrjBUrHL # DLTyAIjzct3BWJDZ02fb5jb8pHmIP3JO6m9Zvjm7ibP65BqJOwIXUTFpbgnrOg45 # oFLDV4JgC4Hh4GEtdm+UhQE51A0VVW5pDaqWTdWkCcuk3QgxUdM3Wm3SW6pw1Gvb # T0X0j5RgF/k3YrW576R/VIy6z4YPbzAtiG4O/zSlsujHoDcVNWnxApgSB/unaDh8 # LNkFPGEMeSs= # =JmTm # -----END PGP SIGNATURE----- # gpg: Signature made Thu 12 May 2022 08:30:49 AM PDT # gpg: using RSA key DC3DEB159A9AF95D3D7456FE7F09B272C88F2FD6 # gpg: issuer "kwolf@redhat.com" # gpg: Good signature from "Kevin Wolf <kwolf@redhat.com>" [full] * tag 'for-upstream' of git://repo.or.cz/qemu/kevin: qemu-iotests: inline common.config into common.rc nbd/server: Allow MULTI_CONN for shared writable exports qemu-nbd: Pass max connections to blockdev layer tests/qtest/fdc-test: Add a regression test for CVE-2021-3507 hw/block/fdc: Prevent end-of-track overrun (CVE-2021-3507) .gitlab-ci.d: export meson testlog.txt as an artifact tests/qemu-iotests: print intent to run a test in TAP mode iotests/testrunner: Flush after run_test() coroutine: Revert to constant batch size coroutine: Rename qemu_coroutine_inc/dec_pool_size() Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
commit
b32b3897f8
@ -26,7 +26,7 @@
|
||||
make -j"$JOBS" $MAKE_CHECK_ARGS ;
|
||||
fi
|
||||
|
||||
.native_test_job_template:
|
||||
.common_test_job_template:
|
||||
stage: test
|
||||
image: $CI_REGISTRY_IMAGE/qemu/$IMAGE:latest
|
||||
script:
|
||||
@ -37,8 +37,16 @@
|
||||
# Avoid recompiling by hiding ninja with NINJA=":"
|
||||
- make NINJA=":" $MAKE_CHECK_ARGS
|
||||
|
||||
.native_test_job_template:
|
||||
extends: .common_test_job_template
|
||||
artifacts:
|
||||
name: "$CI_JOB_NAME-$CI_COMMIT_REF_SLUG"
|
||||
expire_in: 7 days
|
||||
paths:
|
||||
- build/meson-logs/testlog.txt
|
||||
|
||||
.avocado_test_job_template:
|
||||
extends: .native_test_job_template
|
||||
extends: .common_test_job_template
|
||||
cache:
|
||||
key: "${CI_JOB_NAME}-cache"
|
||||
paths:
|
||||
|
@ -3367,6 +3367,7 @@ F: qemu-nbd.*
|
||||
F: blockdev-nbd.c
|
||||
F: docs/interop/nbd.txt
|
||||
F: docs/tools/qemu-nbd.rst
|
||||
F: tests/qemu-iotests/tests/*nbd*
|
||||
T: git https://repo.or.cz/qemu/ericb.git nbd
|
||||
T: git https://src.openvz.org/scm/~vsementsov/qemu.git nbd
|
||||
|
||||
|
@ -30,18 +30,23 @@ typedef struct NBDServerData {
|
||||
} NBDServerData;
|
||||
|
||||
static NBDServerData *nbd_server;
|
||||
static bool is_qemu_nbd;
|
||||
static int qemu_nbd_connections = -1; /* Non-negative if this is qemu-nbd */
|
||||
|
||||
static void nbd_update_server_watch(NBDServerData *s);
|
||||
|
||||
void nbd_server_is_qemu_nbd(bool value)
|
||||
void nbd_server_is_qemu_nbd(int max_connections)
|
||||
{
|
||||
is_qemu_nbd = value;
|
||||
qemu_nbd_connections = max_connections;
|
||||
}
|
||||
|
||||
bool nbd_server_is_running(void)
|
||||
{
|
||||
return nbd_server || is_qemu_nbd;
|
||||
return nbd_server || qemu_nbd_connections >= 0;
|
||||
}
|
||||
|
||||
int nbd_server_max_connections(void)
|
||||
{
|
||||
return nbd_server ? nbd_server->max_connections : qemu_nbd_connections;
|
||||
}
|
||||
|
||||
static void nbd_blockdev_client_closed(NBDClient *client, bool ignored)
|
||||
|
@ -68,3 +68,4 @@ NBD_CMD_BLOCK_STATUS for "qemu:dirty-bitmap:", NBD_CMD_CACHE
|
||||
* 4.2: NBD_FLAG_CAN_MULTI_CONN for shareable read-only exports,
|
||||
NBD_CMD_FLAG_FAST_ZERO
|
||||
* 5.2: NBD_CMD_BLOCK_STATUS for "qemu:allocation-depth"
|
||||
* 7.1: NBD_FLAG_CAN_MULTI_CONN for shareable writable exports
|
||||
|
@ -139,8 +139,7 @@ driver options if :option:`--image-opts` is specified.
|
||||
.. option:: -e, --shared=NUM
|
||||
|
||||
Allow up to *NUM* clients to share the device (default
|
||||
``1``), 0 for unlimited. Safe for readers, but for now,
|
||||
consistency is not guaranteed between multiple writers.
|
||||
``1``), 0 for unlimited.
|
||||
|
||||
.. option:: -t, --persistent
|
||||
|
||||
|
@ -1530,6 +1530,14 @@ static void fdctrl_start_transfer(FDCtrl *fdctrl, int direction)
|
||||
int tmp;
|
||||
fdctrl->data_len = 128 << (fdctrl->fifo[5] > 7 ? 7 : fdctrl->fifo[5]);
|
||||
tmp = (fdctrl->fifo[6] - ks + 1);
|
||||
if (tmp < 0) {
|
||||
FLOPPY_DPRINTF("invalid EOT: %d\n", tmp);
|
||||
fdctrl_stop_transfer(fdctrl, FD_SR0_ABNTERM, FD_SR1_MA, 0x00);
|
||||
fdctrl->fifo[3] = kt;
|
||||
fdctrl->fifo[4] = kh;
|
||||
fdctrl->fifo[5] = ks;
|
||||
return;
|
||||
}
|
||||
if (fdctrl->fifo[0] & 0x80)
|
||||
tmp += fdctrl->fifo[6];
|
||||
fdctrl->data_len *= tmp;
|
||||
|
@ -1215,8 +1215,7 @@ static void virtio_blk_device_realize(DeviceState *dev, Error **errp)
|
||||
for (i = 0; i < conf->num_queues; i++) {
|
||||
virtio_add_queue(vdev, conf->queue_size, virtio_blk_handle_output);
|
||||
}
|
||||
qemu_coroutine_increase_pool_batch_size(conf->num_queues * conf->queue_size
|
||||
/ 2);
|
||||
qemu_coroutine_inc_pool_size(conf->num_queues * conf->queue_size / 2);
|
||||
virtio_blk_data_plane_create(vdev, conf, &s->dataplane, &err);
|
||||
if (err != NULL) {
|
||||
error_propagate(errp, err);
|
||||
@ -1253,8 +1252,7 @@ static void virtio_blk_device_unrealize(DeviceState *dev)
|
||||
for (i = 0; i < conf->num_queues; i++) {
|
||||
virtio_del_queue(vdev, i);
|
||||
}
|
||||
qemu_coroutine_decrease_pool_batch_size(conf->num_queues * conf->queue_size
|
||||
/ 2);
|
||||
qemu_coroutine_dec_pool_size(conf->num_queues * conf->queue_size / 2);
|
||||
qemu_del_vm_change_state_handler(s->change);
|
||||
blockdev_mark_auto_del(s->blk);
|
||||
virtio_cleanup(vdev);
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (C) 2016-2020 Red Hat, Inc.
|
||||
* Copyright (C) 2016-2022 Red Hat, Inc.
|
||||
* Copyright (C) 2005 Anthony Liguori <anthony@codemonkey.ws>
|
||||
*
|
||||
* Network Block Device
|
||||
@ -344,8 +344,9 @@ void nbd_client_new(QIOChannelSocket *sioc,
|
||||
void nbd_client_get(NBDClient *client);
|
||||
void nbd_client_put(NBDClient *client);
|
||||
|
||||
void nbd_server_is_qemu_nbd(bool value);
|
||||
void nbd_server_is_qemu_nbd(int max_connections);
|
||||
bool nbd_server_is_running(void);
|
||||
int nbd_server_max_connections(void);
|
||||
void nbd_server_start(SocketAddress *addr, const char *tls_creds,
|
||||
const char *tls_authz, uint32_t max_connections,
|
||||
Error **errp);
|
||||
|
@ -334,12 +334,12 @@ void coroutine_fn yield_until_fd_readable(int fd);
|
||||
/**
|
||||
* Increase coroutine pool size
|
||||
*/
|
||||
void qemu_coroutine_increase_pool_batch_size(unsigned int additional_pool_size);
|
||||
void qemu_coroutine_inc_pool_size(unsigned int additional_pool_size);
|
||||
|
||||
/**
|
||||
* Devcrease coroutine pool size
|
||||
* Decrease coroutine pool size
|
||||
*/
|
||||
void qemu_coroutine_decrease_pool_batch_size(unsigned int additional_pool_size);
|
||||
void qemu_coroutine_dec_pool_size(unsigned int additional_pool_size);
|
||||
|
||||
#include "qemu/lockable.h"
|
||||
|
||||
|
10
nbd/server.c
10
nbd/server.c
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (C) 2016-2021 Red Hat, Inc.
|
||||
* Copyright (C) 2016-2022 Red Hat, Inc.
|
||||
* Copyright (C) 2005 Anthony Liguori <anthony@codemonkey.ws>
|
||||
*
|
||||
* Network Block Device Server Side
|
||||
@ -1642,7 +1642,6 @@ static int nbd_export_create(BlockExport *blk_exp, BlockExportOptions *exp_args,
|
||||
int64_t size;
|
||||
uint64_t perm, shared_perm;
|
||||
bool readonly = !exp_args->writable;
|
||||
bool shared = !exp_args->writable;
|
||||
BlockDirtyBitmapOrStrList *bitmaps;
|
||||
size_t i;
|
||||
int ret;
|
||||
@ -1693,11 +1692,12 @@ static int nbd_export_create(BlockExport *blk_exp, BlockExportOptions *exp_args,
|
||||
exp->description = g_strdup(arg->description);
|
||||
exp->nbdflags = (NBD_FLAG_HAS_FLAGS | NBD_FLAG_SEND_FLUSH |
|
||||
NBD_FLAG_SEND_FUA | NBD_FLAG_SEND_CACHE);
|
||||
|
||||
if (nbd_server_max_connections() != 1) {
|
||||
exp->nbdflags |= NBD_FLAG_CAN_MULTI_CONN;
|
||||
}
|
||||
if (readonly) {
|
||||
exp->nbdflags |= NBD_FLAG_READ_ONLY;
|
||||
if (shared) {
|
||||
exp->nbdflags |= NBD_FLAG_CAN_MULTI_CONN;
|
||||
}
|
||||
} else {
|
||||
exp->nbdflags |= (NBD_FLAG_SEND_TRIM | NBD_FLAG_SEND_WRITE_ZEROES |
|
||||
NBD_FLAG_SEND_FAST_ZERO);
|
||||
|
@ -22,7 +22,9 @@
|
||||
# recreated on the fly while the NBD server is active.
|
||||
# If missing, it will default to denying access (since 4.0).
|
||||
# @max-connections: The maximum number of connections to allow at the same
|
||||
# time, 0 for unlimited. (since 5.2; default: 0)
|
||||
# time, 0 for unlimited. Setting this to 1 also stops
|
||||
# the server from advertising multiple client support
|
||||
# (since 5.2; default: 0)
|
||||
#
|
||||
# Since: 4.2
|
||||
##
|
||||
@ -51,7 +53,9 @@
|
||||
# recreated on the fly while the NBD server is active.
|
||||
# If missing, it will default to denying access (since 4.0).
|
||||
# @max-connections: The maximum number of connections to allow at the same
|
||||
# time, 0 for unlimited. (since 5.2; default: 0)
|
||||
# time, 0 for unlimited. Setting this to 1 also stops
|
||||
# the server from advertising multiple client support
|
||||
# (since 5.2; default: 0).
|
||||
#
|
||||
# Returns: error if the server is already running.
|
||||
#
|
||||
|
@ -1095,7 +1095,7 @@ int main(int argc, char **argv)
|
||||
|
||||
bs->detect_zeroes = detect_zeroes;
|
||||
|
||||
nbd_server_is_qemu_nbd(true);
|
||||
nbd_server_is_qemu_nbd(shared);
|
||||
|
||||
export_opts = g_new(BlockExportOptions, 1);
|
||||
*export_opts = (BlockExportOptions) {
|
||||
|
@ -1,41 +0,0 @@
|
||||
#!/usr/bin/env bash
|
||||
#
|
||||
# Copyright (C) 2009 Red Hat, Inc.
|
||||
# Copyright (c) 2000-2003,2006 Silicon Graphics, Inc. All Rights Reserved.
|
||||
#
|
||||
# This program is free software; you can redistribute it and/or
|
||||
# modify it under the terms of the GNU General Public License as
|
||||
# published by the Free Software Foundation.
|
||||
#
|
||||
# This program is distributed in the hope that it would be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
#
|
||||
# all tests should use a common language setting to prevent golden
|
||||
# output mismatches.
|
||||
export LANG=C
|
||||
|
||||
PATH=".:$PATH"
|
||||
|
||||
HOSTOS=$(uname -s)
|
||||
arch=$(uname -m)
|
||||
[[ "$arch" =~ "ppc64" ]] && qemu_arch=ppc64 || qemu_arch="$arch"
|
||||
|
||||
# make sure we have a standard umask
|
||||
umask 022
|
||||
|
||||
_optstr_add()
|
||||
{
|
||||
if [ -n "$1" ]; then
|
||||
echo "$1,$2"
|
||||
else
|
||||
echo "$2"
|
||||
fi
|
||||
}
|
||||
|
||||
# make sure this script returns success
|
||||
true
|
@ -17,6 +17,17 @@
|
||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
#
|
||||
|
||||
export LANG=C
|
||||
|
||||
PATH=".:$PATH"
|
||||
|
||||
HOSTOS=$(uname -s)
|
||||
arch=$(uname -m)
|
||||
[[ "$arch" =~ "ppc64" ]] && qemu_arch=ppc64 || qemu_arch="$arch"
|
||||
|
||||
# make sure we have a standard umask
|
||||
umask 022
|
||||
|
||||
# bail out, setting up .notrun file
|
||||
_notrun()
|
||||
{
|
||||
@ -120,18 +131,14 @@ peek_file_raw()
|
||||
dd if="$1" bs=1 skip="$2" count="$3" status=none
|
||||
}
|
||||
|
||||
config=common.config
|
||||
test -f $config || config=../common.config
|
||||
if ! test -f $config
|
||||
then
|
||||
echo "$0: failed to find common.config"
|
||||
exit 1
|
||||
fi
|
||||
if ! . $config
|
||||
then
|
||||
echo "$0: failed to source common.config"
|
||||
exit 1
|
||||
fi
|
||||
_optstr_add()
|
||||
{
|
||||
if [ -n "$1" ]; then
|
||||
echo "$1,$2"
|
||||
else
|
||||
echo "$2"
|
||||
fi
|
||||
}
|
||||
|
||||
# Set the variables to the empty string to turn Valgrind off
|
||||
# for specific processes, e.g.
|
||||
|
@ -361,6 +361,9 @@ class TestRunner(ContextManager['TestRunner']):
|
||||
starttime=start,
|
||||
lasttime=last_el,
|
||||
end = '\n' if mp else '\r')
|
||||
else:
|
||||
testname = os.path.basename(test)
|
||||
print(f'# running {self.env.imgfmt} {testname}')
|
||||
|
||||
res = self.do_run_test(test, mp)
|
||||
|
||||
@ -378,6 +381,7 @@ class TestRunner(ContextManager['TestRunner']):
|
||||
else:
|
||||
print(res.casenotrun)
|
||||
|
||||
sys.stdout.flush()
|
||||
return res
|
||||
|
||||
def run_tests(self, tests: List[str], jobs: int = 1) -> bool:
|
||||
|
145
tests/qemu-iotests/tests/nbd-multiconn
Executable file
145
tests/qemu-iotests/tests/nbd-multiconn
Executable file
@ -0,0 +1,145 @@
|
||||
#!/usr/bin/env python3
|
||||
# group: rw auto quick
|
||||
#
|
||||
# Test cases for NBD multi-conn advertisement
|
||||
#
|
||||
# Copyright (C) 2022 Red Hat, Inc.
|
||||
#
|
||||
# This program is free software; you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation; either version 2 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
import os
|
||||
from contextlib import contextmanager
|
||||
import iotests
|
||||
from iotests import qemu_img_create, qemu_io
|
||||
|
||||
|
||||
disk = os.path.join(iotests.test_dir, 'disk')
|
||||
size = '4M'
|
||||
nbd_sock = os.path.join(iotests.sock_dir, 'nbd_sock')
|
||||
nbd_uri = 'nbd+unix:///{}?socket=' + nbd_sock
|
||||
|
||||
|
||||
@contextmanager
|
||||
def open_nbd(export_name):
|
||||
h = nbd.NBD()
|
||||
try:
|
||||
h.connect_uri(nbd_uri.format(export_name))
|
||||
yield h
|
||||
finally:
|
||||
h.shutdown()
|
||||
|
||||
class TestNbdMulticonn(iotests.QMPTestCase):
|
||||
def setUp(self):
|
||||
qemu_img_create('-f', iotests.imgfmt, disk, size)
|
||||
qemu_io('-c', 'w -P 1 0 2M', '-c', 'w -P 2 2M 2M', disk)
|
||||
|
||||
self.vm = iotests.VM()
|
||||
self.vm.launch()
|
||||
result = self.vm.qmp('blockdev-add', {
|
||||
'driver': 'qcow2',
|
||||
'node-name': 'n',
|
||||
'file': {'driver': 'file', 'filename': disk}
|
||||
})
|
||||
self.assert_qmp(result, 'return', {})
|
||||
|
||||
def tearDown(self):
|
||||
self.vm.shutdown()
|
||||
os.remove(disk)
|
||||
try:
|
||||
os.remove(nbd_sock)
|
||||
except OSError:
|
||||
pass
|
||||
|
||||
@contextmanager
|
||||
def run_server(self, max_connections=None):
|
||||
args = {
|
||||
'addr': {
|
||||
'type': 'unix',
|
||||
'data': {'path': nbd_sock}
|
||||
}
|
||||
}
|
||||
if max_connections is not None:
|
||||
args['max-connections'] = max_connections
|
||||
|
||||
result = self.vm.qmp('nbd-server-start', args)
|
||||
self.assert_qmp(result, 'return', {})
|
||||
yield
|
||||
|
||||
result = self.vm.qmp('nbd-server-stop')
|
||||
self.assert_qmp(result, 'return', {})
|
||||
|
||||
def add_export(self, name, writable=None):
|
||||
args = {
|
||||
'type': 'nbd',
|
||||
'id': name,
|
||||
'node-name': 'n',
|
||||
'name': name,
|
||||
}
|
||||
if writable is not None:
|
||||
args['writable'] = writable
|
||||
|
||||
result = self.vm.qmp('block-export-add', args)
|
||||
self.assert_qmp(result, 'return', {})
|
||||
|
||||
def test_default_settings(self):
|
||||
with self.run_server():
|
||||
self.add_export('r')
|
||||
self.add_export('w', writable=True)
|
||||
with open_nbd('r') as h:
|
||||
self.assertTrue(h.can_multi_conn())
|
||||
with open_nbd('w') as h:
|
||||
self.assertTrue(h.can_multi_conn())
|
||||
|
||||
def test_limited_connections(self):
|
||||
with self.run_server(max_connections=1):
|
||||
self.add_export('r')
|
||||
self.add_export('w', writable=True)
|
||||
with open_nbd('r') as h:
|
||||
self.assertFalse(h.can_multi_conn())
|
||||
with open_nbd('w') as h:
|
||||
self.assertFalse(h.can_multi_conn())
|
||||
|
||||
def test_parallel_writes(self):
|
||||
with self.run_server():
|
||||
self.add_export('w', writable=True)
|
||||
|
||||
clients = [nbd.NBD() for _ in range(3)]
|
||||
for c in clients:
|
||||
c.connect_uri(nbd_uri.format('w'))
|
||||
self.assertTrue(c.can_multi_conn())
|
||||
|
||||
initial_data = clients[0].pread(1024 * 1024, 0)
|
||||
self.assertEqual(initial_data, b'\x01' * 1024 * 1024)
|
||||
|
||||
updated_data = b'\x03' * 1024 * 1024
|
||||
clients[1].pwrite(updated_data, 0)
|
||||
clients[2].flush()
|
||||
current_data = clients[0].pread(1024 * 1024, 0)
|
||||
|
||||
self.assertEqual(updated_data, current_data)
|
||||
|
||||
for i in range(3):
|
||||
clients[i].shutdown()
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
try:
|
||||
# Easier to use libnbd than to try and set up parallel
|
||||
# 'qemu-nbd --list' or 'qemu-io' processes, but not all systems
|
||||
# have libnbd installed.
|
||||
import nbd # type: ignore
|
||||
|
||||
iotests.main(supported_fmts=['qcow2'])
|
||||
except ImportError:
|
||||
iotests.notrun('libnbd not installed')
|
5
tests/qemu-iotests/tests/nbd-multiconn.out
Normal file
5
tests/qemu-iotests/tests/nbd-multiconn.out
Normal file
@ -0,0 +1,5 @@
|
||||
...
|
||||
----------------------------------------------------------------------
|
||||
Ran 3 tests
|
||||
|
||||
OK
|
@ -17,7 +17,7 @@ wrote 2097152/2097152 bytes at offset 1048576
|
||||
exports available: 1
|
||||
export: ''
|
||||
size: 4194304
|
||||
flags: 0x58f ( readonly flush fua df multi cache )
|
||||
flags: 0x48f ( readonly flush fua df cache )
|
||||
min block: 1
|
||||
opt block: 4096
|
||||
max block: 33554432
|
||||
|
@ -582,6 +582,26 @@ static void test_cve_2021_20196(void)
|
||||
qtest_quit(s);
|
||||
}
|
||||
|
||||
static void test_cve_2021_3507(void)
|
||||
{
|
||||
QTestState *s;
|
||||
|
||||
s = qtest_initf("-nographic -m 32M -nodefaults "
|
||||
"-drive file=%s,format=raw,if=floppy,snapshot=on",
|
||||
test_image);
|
||||
qtest_outl(s, 0x9, 0x0a0206);
|
||||
qtest_outw(s, 0x3f4, 0x1600);
|
||||
qtest_outw(s, 0x3f4, 0x0000);
|
||||
qtest_outw(s, 0x3f4, 0x0000);
|
||||
qtest_outw(s, 0x3f4, 0x0000);
|
||||
qtest_outw(s, 0x3f4, 0x0200);
|
||||
qtest_outw(s, 0x3f4, 0x0200);
|
||||
qtest_outw(s, 0x3f4, 0x0000);
|
||||
qtest_outw(s, 0x3f4, 0x0000);
|
||||
qtest_outw(s, 0x3f4, 0x0000);
|
||||
qtest_quit(s);
|
||||
}
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
int fd;
|
||||
@ -613,6 +633,7 @@ int main(int argc, char **argv)
|
||||
qtest_add_func("/fdc/read_no_dma_19", test_read_no_dma_19);
|
||||
qtest_add_func("/fdc/fuzz-registers", fuzz_registers);
|
||||
qtest_add_func("/fdc/fuzz/cve_2021_20196", test_cve_2021_20196);
|
||||
qtest_add_func("/fdc/fuzz/cve_2021_3507", test_cve_2021_3507);
|
||||
|
||||
ret = g_test_run();
|
||||
|
||||
|
@ -21,14 +21,20 @@
|
||||
#include "qemu/coroutine-tls.h"
|
||||
#include "block/aio.h"
|
||||
|
||||
/** Initial batch size is 64, and is increased on demand */
|
||||
/**
|
||||
* The minimal batch size is always 64, coroutines from the release_pool are
|
||||
* reused as soon as there are 64 coroutines in it. The maximum pool size starts
|
||||
* with 64 and is increased on demand so that coroutines are not deleted even if
|
||||
* they are not immediately reused.
|
||||
*/
|
||||
enum {
|
||||
POOL_INITIAL_BATCH_SIZE = 64,
|
||||
POOL_MIN_BATCH_SIZE = 64,
|
||||
POOL_INITIAL_MAX_SIZE = 64,
|
||||
};
|
||||
|
||||
/** Free list to speed up creation */
|
||||
static QSLIST_HEAD(, Coroutine) release_pool = QSLIST_HEAD_INITIALIZER(pool);
|
||||
static unsigned int pool_batch_size = POOL_INITIAL_BATCH_SIZE;
|
||||
static unsigned int pool_max_size = POOL_INITIAL_MAX_SIZE;
|
||||
static unsigned int release_pool_size;
|
||||
|
||||
typedef QSLIST_HEAD(, Coroutine) CoroutineQSList;
|
||||
@ -57,7 +63,7 @@ Coroutine *qemu_coroutine_create(CoroutineEntry *entry, void *opaque)
|
||||
|
||||
co = QSLIST_FIRST(alloc_pool);
|
||||
if (!co) {
|
||||
if (release_pool_size > qatomic_read(&pool_batch_size)) {
|
||||
if (release_pool_size > POOL_MIN_BATCH_SIZE) {
|
||||
/* Slow path; a good place to register the destructor, too. */
|
||||
Notifier *notifier = get_ptr_coroutine_pool_cleanup_notifier();
|
||||
if (!notifier->notify) {
|
||||
@ -95,12 +101,12 @@ static void coroutine_delete(Coroutine *co)
|
||||
co->caller = NULL;
|
||||
|
||||
if (CONFIG_COROUTINE_POOL) {
|
||||
if (release_pool_size < qatomic_read(&pool_batch_size) * 2) {
|
||||
if (release_pool_size < qatomic_read(&pool_max_size) * 2) {
|
||||
QSLIST_INSERT_HEAD_ATOMIC(&release_pool, co, pool_next);
|
||||
qatomic_inc(&release_pool_size);
|
||||
return;
|
||||
}
|
||||
if (get_alloc_pool_size() < qatomic_read(&pool_batch_size)) {
|
||||
if (get_alloc_pool_size() < qatomic_read(&pool_max_size)) {
|
||||
QSLIST_INSERT_HEAD(get_ptr_alloc_pool(), co, pool_next);
|
||||
set_alloc_pool_size(get_alloc_pool_size() + 1);
|
||||
return;
|
||||
@ -212,12 +218,12 @@ AioContext *coroutine_fn qemu_coroutine_get_aio_context(Coroutine *co)
|
||||
return co->ctx;
|
||||
}
|
||||
|
||||
void qemu_coroutine_increase_pool_batch_size(unsigned int additional_pool_size)
|
||||
void qemu_coroutine_inc_pool_size(unsigned int additional_pool_size)
|
||||
{
|
||||
qatomic_add(&pool_batch_size, additional_pool_size);
|
||||
qatomic_add(&pool_max_size, additional_pool_size);
|
||||
}
|
||||
|
||||
void qemu_coroutine_decrease_pool_batch_size(unsigned int removing_pool_size)
|
||||
void qemu_coroutine_dec_pool_size(unsigned int removing_pool_size)
|
||||
{
|
||||
qatomic_sub(&pool_batch_size, removing_pool_size);
|
||||
qatomic_sub(&pool_max_size, removing_pool_size);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user