* Fuzzing improvements (Qiuhao, Alexander)

* i386: Fix BMI decoding for instructions with the 0x66 prefix (David)
 * initial attempt at fixing event_notifier emulation (Maxim)
 * i386: PKS emulation, fix for "qemu-system-i386 -cpu host" (myself)
 * meson: RBD test fixes (myself)
 * meson: TCI warnings (Philippe)
 * Leaner build for --disable-guest-agent, --disable-system and
   --disable-tools (Philippe, Stefan)
 * --enable-tcg-interpreter fix (Richard)
 * i386: SVM feature bits (Wei)
 * KVM bugfix (Thomas H.)
 * Add missing MemoryRegionOps callbacks (PJP)
 -----BEGIN PGP SIGNATURE-----
 
 iQFIBAABCAAyFiEE8TM4V0tmI4mGbHaCv/vSX3jHroMFAmAhR4cUHHBib256aW5p
 QHJlZGhhdC5jb20ACgkQv/vSX3jHroOviAf/Ymk/KwHZKySVqHOjDZnvP5PXMAru
 p1zlMRLAorXK+CTbshkIliaQyD8ggzT4HCinJ2NisdfTWMmlWbgr8gahNqKyZ5UG
 HlL28va3dvGhelswh/CNso1ZhVb2Q+aAYn/c6LXQva2r0xi26ohJTkIkSCPP/bnI
 +73dGzwAilBOsBVbn4cCm/70XtwDpPkw41IZIDoy/4lhL8ZdpHMz8oOjNIlOdlcU
 aEDfM8vYE4C70OtUlRZ1OwVxzcjS1Bf6dQYcpg5gAKy/jAAqR+v2PStxXiUuj5D3
 cAzd03Goh78Wcre+CbWxDKGcGtiooUT+J09wmvDPYVUHcpQMbumf4MufrQ==
 =INB5
 -----END PGP SIGNATURE-----

Merge remote-tracking branch 'remotes/bonzini-gitlab/tags/for-upstream' into staging

* Fuzzing improvements (Qiuhao, Alexander)
* i386: Fix BMI decoding for instructions with the 0x66 prefix (David)
* initial attempt at fixing event_notifier emulation (Maxim)
* i386: PKS emulation, fix for "qemu-system-i386 -cpu host" (myself)
* meson: RBD test fixes (myself)
* meson: TCI warnings (Philippe)
* Leaner build for --disable-guest-agent, --disable-system and
  --disable-tools (Philippe, Stefan)
* --enable-tcg-interpreter fix (Richard)
* i386: SVM feature bits (Wei)
* KVM bugfix (Thomas H.)
* Add missing MemoryRegionOps callbacks (PJP)

# gpg: Signature made Mon 08 Feb 2021 14:15:35 GMT
# gpg:                using RSA key F13338574B662389866C7682BFFBD25F78C7AE83
# gpg:                issuer "pbonzini@redhat.com"
# gpg: Good signature from "Paolo Bonzini <bonzini@gnu.org>" [full]
# gpg:                 aka "Paolo Bonzini <pbonzini@redhat.com>" [full]
# Primary key fingerprint: 46F5 9FBD 57D6 12E7 BFD4  E2F7 7E15 100C CD36 69B1
#      Subkey fingerprint: F133 3857 4B66 2389 866C  7682 BFFB D25F 78C7 AE83

* remotes/bonzini-gitlab/tags/for-upstream: (46 commits)
  target/i386: Expose VMX entry/exit load pkrs control bits
  target/i386: Add support for save/load IA32_PKRS MSR
  imx7-ccm: add digprog mmio write method
  tz-ppc: add dummy read/write methods
  spapr_pci: add spapr msi read method
  nvram: add nrf51_soc flash read method
  prep: add ppc-parity write method
  vfio: add quirk device write method
  pci-host: designware: add pcie-msi read method
  hw/pci-host: add pci-intack write method
  cpu-throttle: Remove timer_mod() from cpu_throttle_set()
  replay: rng-builtin support
  pc-bios/descriptors: fix paths in json files
  replay: fix replay of the interrupts
  accel/kvm/kvm-all: Fix wrong return code handling in dirty log code
  qapi/meson: Restrict UI module to system emulation and tools
  qapi/meson: Restrict system-mode specific modules
  qapi/meson: Remove QMP from user-mode emulation
  qapi/meson: Restrict qdev code to system-mode emulation
  meson: Restrict emulation code
  ...

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
Peter Maydell 2021-02-09 10:04:51 +00:00
commit 41d306ec7d
47 changed files with 573 additions and 191 deletions

View File

@ -2541,6 +2541,7 @@ F: qapi/qom.json
F: qapi/qdev.json F: qapi/qdev.json
F: scripts/coccinelle/qom-parent-type.cocci F: scripts/coccinelle/qom-parent-type.cocci
F: softmmu/qdev-monitor.c F: softmmu/qdev-monitor.c
F: stubs/qdev.c
F: qom/ F: qom/
F: tests/check-qom-interface.c F: tests/check-qom-interface.c
F: tests/check-qom-proplist.c F: tests/check-qom-proplist.c

View File

@ -644,15 +644,18 @@ static int kvm_physical_sync_dirty_bitmap(KVMMemoryListener *kml,
d.dirty_bitmap = mem->dirty_bmap; d.dirty_bitmap = mem->dirty_bmap;
d.slot = mem->slot | (kml->as_id << 16); d.slot = mem->slot | (kml->as_id << 16);
if (kvm_vm_ioctl(s, KVM_GET_DIRTY_LOG, &d) == -1) { ret = kvm_vm_ioctl(s, KVM_GET_DIRTY_LOG, &d);
DPRINTF("ioctl failed %d\n", errno); if (ret == -ENOENT) {
ret = -1; /* kernel does not have dirty bitmap in this slot */
ret = 0;
} else if (ret < 0) {
error_report("ioctl KVM_GET_DIRTY_LOG failed: %d", errno);
goto out; goto out;
} } else {
subsection.offset_within_region += slot_offset; subsection.offset_within_region += slot_offset;
subsection.size = int128_make64(slot_size); subsection.size = int128_make64(slot_size);
kvm_get_dirty_pages_log_range(&subsection, d.dirty_bitmap); kvm_get_dirty_pages_log_range(&subsection, d.dirty_bitmap);
}
slot_offset += slot_size; slot_offset += slot_size;
start_addr += slot_size; start_addr += slot_size;
@ -750,8 +753,8 @@ static int kvm_log_clear_one_slot(KVMSlot *mem, int as_id, uint64_t start,
d.num_pages = bmap_npages; d.num_pages = bmap_npages;
d.slot = mem->slot | (as_id << 16); d.slot = mem->slot | (as_id << 16);
if (kvm_vm_ioctl(s, KVM_CLEAR_DIRTY_LOG, &d) == -1) { ret = kvm_vm_ioctl(s, KVM_CLEAR_DIRTY_LOG, &d);
ret = -errno; if (ret < 0 && ret != -ENOENT) {
error_report("%s: KVM_CLEAR_DIRTY_LOG failed, slot=%d, " error_report("%s: KVM_CLEAR_DIRTY_LOG failed, slot=%d, "
"start=0x%"PRIx64", size=0x%"PRIx32", errno=%d", "start=0x%"PRIx64", size=0x%"PRIx32", errno=%d",
__func__, d.slot, (uint64_t)d.first_page, __func__, d.slot, (uint64_t)d.first_page,

View File

@ -81,7 +81,13 @@ void icount_handle_deadline(void)
int64_t deadline = qemu_clock_deadline_ns_all(QEMU_CLOCK_VIRTUAL, int64_t deadline = qemu_clock_deadline_ns_all(QEMU_CLOCK_VIRTUAL,
QEMU_TIMER_ATTR_ALL); QEMU_TIMER_ATTR_ALL);
if (deadline == 0) { /*
* Instructions, interrupts, and exceptions are processed in cpu-exec.
* Don't interrupt cpu thread, when these events are waiting
* (i.e., there is no checkpoint)
*/
if (deadline == 0
&& (replay_mode != REPLAY_MODE_PLAY || replay_has_checkpoint())) {
icount_notify_aio_contexts(); icount_notify_aio_contexts();
} }
} }

View File

@ -494,6 +494,16 @@ host_memory_backend_class_init(ObjectClass *oc, void *data)
host_memory_backend_get_share, host_memory_backend_set_share); host_memory_backend_get_share, host_memory_backend_set_share);
object_class_property_set_description(oc, "share", object_class_property_set_description(oc, "share",
"Mark the memory as private to QEMU or shared"); "Mark the memory as private to QEMU or shared");
/*
* Do not delete/rename option. This option must be considered stable
* (as if it didn't have the 'x-' prefix including deprecation period) as
* long as 4.0 and older machine types exists.
* Option will be used by upper layers to override (disable) canonical path
* for ramblock-id set by compat properties on old machine types ( <= 4.0),
* to keep migration working when backend is used for main RAM with
* -machine memory-backend= option (main RAM historically used prefix-less
* ramblock-id).
*/
object_class_property_add_bool(oc, "x-use-canonical-path-for-ramblock-id", object_class_property_add_bool(oc, "x-use-canonical-path-for-ramblock-id",
host_memory_backend_get_use_canonical_path, host_memory_backend_get_use_canonical_path,
host_memory_backend_set_use_canonical_path); host_memory_backend_set_use_canonical_path);

View File

@ -10,6 +10,7 @@
#include "qemu/main-loop.h" #include "qemu/main-loop.h"
#include "qemu/guest-random.h" #include "qemu/guest-random.h"
#include "qom/object.h" #include "qom/object.h"
#include "sysemu/replay.h"
OBJECT_DECLARE_SIMPLE_TYPE(RngBuiltin, RNG_BUILTIN) OBJECT_DECLARE_SIMPLE_TYPE(RngBuiltin, RNG_BUILTIN)
@ -37,7 +38,7 @@ static void rng_builtin_request_entropy(RngBackend *b, RngRequest *req)
{ {
RngBuiltin *s = RNG_BUILTIN(b); RngBuiltin *s = RNG_BUILTIN(b);
qemu_bh_schedule(s->bh); replay_bh_schedule_event(s->bh);
} }
static void rng_builtin_init(Object *obj) static void rng_builtin_init(Object *obj)

2
configure vendored
View File

@ -1764,7 +1764,7 @@ Advanced options (experts only):
--with-trace-file=NAME Full PATH,NAME of file to store traces --with-trace-file=NAME Full PATH,NAME of file to store traces
Default:trace-<pid> Default:trace-<pid>
--disable-slirp disable SLIRP userspace network connectivity --disable-slirp disable SLIRP userspace network connectivity
--enable-tcg-interpreter enable TCG with bytecode interpreter (TCI) --enable-tcg-interpreter enable TCI (TCG with bytecode interpreter, experimental and slow)
--enable-malloc-trim enable libc malloc_trim() for memory optimization --enable-malloc-trim enable libc malloc_trim() for memory optimization
--oss-lib path to OSS library --oss-lib path to OSS library
--cpu=CPU Build for host CPU [$cpu] --cpu=CPU Build for host CPU [$cpu]

View File

@ -100,7 +100,7 @@ In meson.build::
# Detect dependency # Detect dependency
sdl_image = dependency('SDL2_image', required: get_option('sdl_image'), sdl_image = dependency('SDL2_image', required: get_option('sdl_image'),
method: 'pkg-config', method: 'pkg-config',
static: enable_static) kwargs: static_kwargs)
# Create config-host.h (if applicable) # Create config-host.h (if applicable)
config_host_data.set('CONFIG_SDL_IMAGE', sdl_image.found()) config_host_data.set('CONFIG_SDL_IMAGE', sdl_image.found())

View File

@ -119,7 +119,7 @@ Adding a new fuzzer
Coverage over virtual devices can be improved by adding additional fuzzers. Coverage over virtual devices can be improved by adding additional fuzzers.
Fuzzers are kept in ``tests/qtest/fuzz/`` and should be added to Fuzzers are kept in ``tests/qtest/fuzz/`` and should be added to
``tests/qtest/fuzz/Makefile.include`` ``tests/qtest/fuzz/meson.build``
Fuzzers can rely on both qtest and libqos to communicate with virtual devices. Fuzzers can rely on both qtest and libqos to communicate with virtual devices.
@ -128,8 +128,7 @@ Fuzzers can rely on both qtest and libqos to communicate with virtual devices.
2. Write the fuzzing code using the libqtest/libqos API. See existing fuzzers 2. Write the fuzzing code using the libqtest/libqos API. See existing fuzzers
for reference. for reference.
3. Register the fuzzer in ``tests/fuzz/Makefile.include`` by appending the 3. Add the fuzzer to ``tests/qtest/fuzz/meson.build``.
corresponding object to fuzz-obj-y
Fuzzers can be more-or-less thought of as special qtest programs which can Fuzzers can be more-or-less thought of as special qtest programs which can
modify the qtest commands and/or qtest command arguments based on inputs modify the qtest commands and/or qtest command arguments based on inputs
@ -181,6 +180,36 @@ To ensure that these env variables have been configured correctly, we can use::
The output should contain a complete list of matched MemoryRegions. The output should contain a complete list of matched MemoryRegions.
OSS-Fuzz
--------
QEMU is continuously fuzzed on `OSS-Fuzz` __(https://github.com/google/oss-fuzz).
By default, the OSS-Fuzz build will try to fuzz every fuzz-target. Since the
generic-fuzz target requires additional information provided in environment
variables, we pre-define some generic-fuzz configs in
``tests/qtest/fuzz/generic_fuzz_configs.h``. Each config must specify:
- ``.name``: To identify the fuzzer config
- ``.args`` OR ``.argfunc``: A string or pointer to a function returning a
string. These strings are used to specify the ``QEMU_FUZZ_ARGS``
environment variable. ``argfunc`` is useful when the config relies on e.g.
a dynamically created temp directory, or a free tcp/udp port.
- ``.objects``: A string that specifies the ``QEMU_FUZZ_OBJECTS`` environment
variable.
To fuzz additional devices/device configuration on OSS-Fuzz, send patches for
either a new device-specific fuzzer or a new generic-fuzz config.
Build details:
- The Dockerfile that sets up the environment for building QEMU's
fuzzers on OSS-Fuzz can be fund in the OSS-Fuzz repository
__(https://github.com/google/oss-fuzz/blob/master/projects/qemu/Dockerfile)
- The script responsible for building the fuzzers can be found in the
QEMU source tree at ``scripts/oss-fuzz/build.sh``
Implementation Details / Fuzzer Lifecycle Implementation Details / Fuzzer Lifecycle
----------------------------------------- -----------------------------------------

View File

@ -46,9 +46,11 @@ if build_docs
meson.source_root() / 'docs/sphinx/qmp_lexer.py', meson.source_root() / 'docs/sphinx/qmp_lexer.py',
qapi_gen_depends ] qapi_gen_depends ]
have_ga = have_tools and config_host.has_key('CONFIG_GUEST_AGENT')
man_pages = { man_pages = {
'qemu-ga.8': (have_tools ? 'man8' : ''), 'qemu-ga.8': (have_ga ? 'man8' : ''),
'qemu-ga-ref.7': 'man7', 'qemu-ga-ref.7': (have_ga ? 'man7' : ''),
'qemu-qmp-ref.7': 'man7', 'qemu-qmp-ref.7': 'man7',
'qemu-storage-daemon-qmp-ref.7': (have_tools ? 'man7' : ''), 'qemu-storage-daemon-qmp-ref.7': (have_tools ? 'man7' : ''),
'qemu-img.1': (have_tools ? 'man1' : ''), 'qemu-img.1': (have_tools ? 'man1' : ''),

View File

@ -131,8 +131,16 @@ static const struct MemoryRegionOps imx7_set_clr_tog_ops = {
}, },
}; };
static void imx7_digprog_write(void *opaque, hwaddr addr,
uint64_t data, unsigned size)
{
qemu_log_mask(LOG_GUEST_ERROR,
"Guest write to read-only ANALOG_DIGPROG register\n");
}
static const struct MemoryRegionOps imx7_digprog_ops = { static const struct MemoryRegionOps imx7_digprog_ops = {
.read = imx7_set_clr_tog_read, .read = imx7_set_clr_tog_read,
.write = imx7_digprog_write,
.endianness = DEVICE_NATIVE_ENDIAN, .endianness = DEVICE_NATIVE_ENDIAN,
.impl = { .impl = {
.min_access_size = 4, .min_access_size = 4,

View File

@ -196,7 +196,21 @@ static bool tz_ppc_dummy_accepts(void *opaque, hwaddr addr,
g_assert_not_reached(); g_assert_not_reached();
} }
static uint64_t tz_ppc_dummy_read(void *opaque, hwaddr addr, unsigned size)
{
g_assert_not_reached();
}
static void tz_ppc_dummy_write(void *opaque, hwaddr addr,
uint64_t data, unsigned size)
{
g_assert_not_reached();
}
static const MemoryRegionOps tz_ppc_dummy_ops = { static const MemoryRegionOps tz_ppc_dummy_ops = {
/* define r/w methods to avoid assert failure in memory_region_init_io */
.read = tz_ppc_dummy_read,
.write = tz_ppc_dummy_write,
.valid.accepts = tz_ppc_dummy_accepts, .valid.accepts = tz_ppc_dummy_accepts,
}; };

View File

@ -273,6 +273,15 @@ static const MemoryRegionOps io_ops = {
.endianness = DEVICE_LITTLE_ENDIAN, .endianness = DEVICE_LITTLE_ENDIAN,
}; };
static uint64_t flash_read(void *opaque, hwaddr offset, unsigned size)
{
/*
* This is a rom_device MemoryRegion which is always in
* romd_mode (we never put it in MMIO mode), so reads always
* go directly to RAM and never come here.
*/
g_assert_not_reached();
}
static void flash_write(void *opaque, hwaddr offset, uint64_t value, static void flash_write(void *opaque, hwaddr offset, uint64_t value,
unsigned int size) unsigned int size)
@ -300,6 +309,7 @@ static void flash_write(void *opaque, hwaddr offset, uint64_t value,
static const MemoryRegionOps flash_ops = { static const MemoryRegionOps flash_ops = {
.read = flash_read,
.write = flash_write, .write = flash_write,
.valid.min_access_size = 4, .valid.min_access_size = 4,
.valid.max_access_size = 4, .valid.max_access_size = 4,

View File

@ -21,6 +21,7 @@
#include "qemu/osdep.h" #include "qemu/osdep.h"
#include "qapi/error.h" #include "qapi/error.h"
#include "qemu/module.h" #include "qemu/module.h"
#include "qemu/log.h"
#include "hw/pci/msi.h" #include "hw/pci/msi.h"
#include "hw/pci/pci_bridge.h" #include "hw/pci/pci_bridge.h"
#include "hw/pci/pci_host.h" #include "hw/pci/pci_host.h"
@ -63,6 +64,23 @@ designware_pcie_root_to_host(DesignwarePCIERoot *root)
return DESIGNWARE_PCIE_HOST(bus->parent); return DESIGNWARE_PCIE_HOST(bus->parent);
} }
static uint64_t designware_pcie_root_msi_read(void *opaque, hwaddr addr,
unsigned size)
{
/*
* Attempts to read from the MSI address are undefined in
* the PCI specifications. For this hardware, the datasheet
* specifies that a read from the magic address is simply not
* intercepted by the MSI controller, and will go out to the
* AHB/AXI bus like any other PCI-device-initiated DMA read.
* This is not trivial to implement in QEMU, so since
* well-behaved guests won't ever ask a PCI device to DMA from
* this address we just log the missing functionality.
*/
qemu_log_mask(LOG_UNIMP, "%s not implemented\n", __func__);
return 0;
}
static void designware_pcie_root_msi_write(void *opaque, hwaddr addr, static void designware_pcie_root_msi_write(void *opaque, hwaddr addr,
uint64_t val, unsigned len) uint64_t val, unsigned len)
{ {
@ -77,6 +95,7 @@ static void designware_pcie_root_msi_write(void *opaque, hwaddr addr,
} }
static const MemoryRegionOps designware_pci_host_msi_ops = { static const MemoryRegionOps designware_pci_host_msi_ops = {
.read = designware_pcie_root_msi_read,
.write = designware_pcie_root_msi_write, .write = designware_pcie_root_msi_write,
.endianness = DEVICE_LITTLE_ENDIAN, .endianness = DEVICE_LITTLE_ENDIAN,
.valid = { .valid = {

View File

@ -27,6 +27,7 @@
#include "qemu-common.h" #include "qemu-common.h"
#include "qemu/datadir.h" #include "qemu/datadir.h"
#include "qemu/units.h" #include "qemu/units.h"
#include "qemu/log.h"
#include "qapi/error.h" #include "qapi/error.h"
#include "hw/pci/pci.h" #include "hw/pci/pci.h"
#include "hw/pci/pci_bus.h" #include "hw/pci/pci_bus.h"
@ -121,8 +122,15 @@ static uint64_t raven_intack_read(void *opaque, hwaddr addr,
return pic_read_irq(isa_pic); return pic_read_irq(isa_pic);
} }
static void raven_intack_write(void *opaque, hwaddr addr,
uint64_t data, unsigned size)
{
qemu_log_mask(LOG_UNIMP, "%s not implemented\n", __func__);
}
static const MemoryRegionOps raven_intack_ops = { static const MemoryRegionOps raven_intack_ops = {
.read = raven_intack_read, .read = raven_intack_read,
.write = raven_intack_write,
.valid = { .valid = {
.max_access_size = 1, .max_access_size = 1,
}, },

View File

@ -23,6 +23,7 @@
*/ */
#include "qemu/osdep.h" #include "qemu/osdep.h"
#include "qemu/log.h"
#include "hw/irq.h" #include "hw/irq.h"
#include "hw/isa/isa.h" #include "hw/isa/isa.h"
#include "hw/qdev-properties.h" #include "hw/qdev-properties.h"
@ -235,8 +236,15 @@ static uint64_t ppc_parity_error_readl(void *opaque, hwaddr addr,
return val; return val;
} }
static void ppc_parity_error_writel(void *opaque, hwaddr addr,
uint64_t data, unsigned size)
{
qemu_log_mask(LOG_GUEST_ERROR, "%s: invalid access\n", __func__);
}
static const MemoryRegionOps ppc_parity_error_ops = { static const MemoryRegionOps ppc_parity_error_ops = {
.read = ppc_parity_error_readl, .read = ppc_parity_error_readl,
.write = ppc_parity_error_writel,
.valid = { .valid = {
.min_access_size = 4, .min_access_size = 4,
.max_access_size = 4, .max_access_size = 4,

View File

@ -53,6 +53,7 @@
#include "sysemu/hostmem.h" #include "sysemu/hostmem.h"
#include "sysemu/numa.h" #include "sysemu/numa.h"
#include "hw/ppc/spapr_numa.h" #include "hw/ppc/spapr_numa.h"
#include "qemu/log.h"
/* Copied from the kernel arch/powerpc/platforms/pseries/msi.c */ /* Copied from the kernel arch/powerpc/platforms/pseries/msi.c */
#define RTAS_QUERY_FN 0 #define RTAS_QUERY_FN 0
@ -739,6 +740,12 @@ static PCIINTxRoute spapr_route_intx_pin_to_irq(void *opaque, int pin)
return route; return route;
} }
static uint64_t spapr_msi_read(void *opaque, hwaddr addr, unsigned size)
{
qemu_log_mask(LOG_GUEST_ERROR, "%s: invalid access\n", __func__);
return 0;
}
/* /*
* MSI/MSIX memory region implementation. * MSI/MSIX memory region implementation.
* The handler handles both MSI and MSIX. * The handler handles both MSI and MSIX.
@ -756,8 +763,11 @@ static void spapr_msi_write(void *opaque, hwaddr addr,
} }
static const MemoryRegionOps spapr_msi_ops = { static const MemoryRegionOps spapr_msi_ops = {
/* There is no .read as the read result is undefined by PCI spec */ /*
.read = NULL, * .read result is undefined by PCI spec.
* define .read method to avoid assert failure in memory_region_init_io
*/
.read = spapr_msi_read,
.write = spapr_msi_write, .write = spapr_msi_write,
.endianness = DEVICE_LITTLE_ENDIAN .endianness = DEVICE_LITTLE_ENDIAN
}; };

View File

@ -126,6 +126,7 @@ int virtio_scsi_dataplane_start(VirtIODevice *vdev)
{ {
int i; int i;
int rc; int rc;
int vq_init_count = 0;
BusState *qbus = qdev_get_parent_bus(DEVICE(vdev)); BusState *qbus = qdev_get_parent_bus(DEVICE(vdev));
VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus); VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus);
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(vdev); VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(vdev);
@ -153,17 +154,22 @@ int virtio_scsi_dataplane_start(VirtIODevice *vdev)
if (rc) { if (rc) {
goto fail_vrings; goto fail_vrings;
} }
vq_init_count++;
rc = virtio_scsi_vring_init(s, vs->event_vq, 1, rc = virtio_scsi_vring_init(s, vs->event_vq, 1,
virtio_scsi_data_plane_handle_event); virtio_scsi_data_plane_handle_event);
if (rc) { if (rc) {
goto fail_vrings; goto fail_vrings;
} }
vq_init_count++;
for (i = 0; i < vs->conf.num_queues; i++) { for (i = 0; i < vs->conf.num_queues; i++) {
rc = virtio_scsi_vring_init(s, vs->cmd_vqs[i], i + 2, rc = virtio_scsi_vring_init(s, vs->cmd_vqs[i], i + 2,
virtio_scsi_data_plane_handle_cmd); virtio_scsi_data_plane_handle_cmd);
if (rc) { if (rc) {
goto fail_vrings; goto fail_vrings;
} }
vq_init_count++;
} }
s->dataplane_starting = false; s->dataplane_starting = false;
@ -174,7 +180,7 @@ int virtio_scsi_dataplane_start(VirtIODevice *vdev)
fail_vrings: fail_vrings:
aio_wait_bh_oneshot(s->ctx, virtio_scsi_dataplane_stop_bh, s); aio_wait_bh_oneshot(s->ctx, virtio_scsi_dataplane_stop_bh, s);
aio_context_release(s->ctx); aio_context_release(s->ctx);
for (i = 0; i < vs->conf.num_queues + 2; i++) { for (i = 0; i < vq_init_count; i++) {
virtio_bus_set_host_notifier(VIRTIO_BUS(qbus), i, false); virtio_bus_set_host_notifier(VIRTIO_BUS(qbus), i, false);
virtio_bus_cleanup_host_notifier(VIRTIO_BUS(qbus), i); virtio_bus_cleanup_host_notifier(VIRTIO_BUS(qbus), i);
} }

View File

@ -14,6 +14,7 @@
#include CONFIG_DEVICES #include CONFIG_DEVICES
#include "exec/memop.h" #include "exec/memop.h"
#include "qemu/units.h" #include "qemu/units.h"
#include "qemu/log.h"
#include "qemu/error-report.h" #include "qemu/error-report.h"
#include "qemu/main-loop.h" #include "qemu/main-loop.h"
#include "qemu/module.h" #include "qemu/module.h"
@ -264,8 +265,15 @@ static uint64_t vfio_ati_3c3_quirk_read(void *opaque,
return data; return data;
} }
static void vfio_ati_3c3_quirk_write(void *opaque, hwaddr addr,
uint64_t data, unsigned size)
{
qemu_log_mask(LOG_GUEST_ERROR, "%s: invalid access\n", __func__);
}
static const MemoryRegionOps vfio_ati_3c3_quirk = { static const MemoryRegionOps vfio_ati_3c3_quirk = {
.read = vfio_ati_3c3_quirk_read, .read = vfio_ati_3c3_quirk_read,
.write = vfio_ati_3c3_quirk_write,
.endianness = DEVICE_LITTLE_ENDIAN, .endianness = DEVICE_LITTLE_ENDIAN,
}; };

View File

@ -45,13 +45,11 @@ DECLARE_OBJ_CHECKERS(IOMMUMemoryRegion, IOMMUMemoryRegionClass,
#ifdef CONFIG_FUZZ #ifdef CONFIG_FUZZ
void fuzz_dma_read_cb(size_t addr, void fuzz_dma_read_cb(size_t addr,
size_t len, size_t len,
MemoryRegion *mr, MemoryRegion *mr);
bool is_write);
#else #else
static inline void fuzz_dma_read_cb(size_t addr, static inline void fuzz_dma_read_cb(size_t addr,
size_t len, size_t len,
MemoryRegion *mr, MemoryRegion *mr)
bool is_write)
{ {
/* Do Nothing */ /* Do Nothing */
} }
@ -2514,7 +2512,7 @@ address_space_read_cached(MemoryRegionCache *cache, hwaddr addr,
void *buf, hwaddr len) void *buf, hwaddr len)
{ {
assert(addr < cache->len && len <= cache->len - addr); assert(addr < cache->len && len <= cache->len - addr);
fuzz_dma_read_cb(cache->xlat + addr, len, cache->mrs.mr, false); fuzz_dma_read_cb(cache->xlat + addr, len, cache->mrs.mr);
if (likely(cache->ptr)) { if (likely(cache->ptr)) {
memcpy(buf, cache->ptr + addr, len); memcpy(buf, cache->ptr + addr, len);
return MEMTX_OK; return MEMTX_OK;

View File

@ -28,7 +28,7 @@ static inline uint32_t ADDRESS_SPACE_LD_CACHED(l)(MemoryRegionCache *cache,
hwaddr addr, MemTxAttrs attrs, MemTxResult *result) hwaddr addr, MemTxAttrs attrs, MemTxResult *result)
{ {
assert(addr < cache->len && 4 <= cache->len - addr); assert(addr < cache->len && 4 <= cache->len - addr);
fuzz_dma_read_cb(cache->xlat + addr, 4, cache->mrs.mr, false); fuzz_dma_read_cb(cache->xlat + addr, 4, cache->mrs.mr);
if (likely(cache->ptr)) { if (likely(cache->ptr)) {
return LD_P(l)(cache->ptr + addr); return LD_P(l)(cache->ptr + addr);
} else { } else {
@ -40,7 +40,7 @@ static inline uint64_t ADDRESS_SPACE_LD_CACHED(q)(MemoryRegionCache *cache,
hwaddr addr, MemTxAttrs attrs, MemTxResult *result) hwaddr addr, MemTxAttrs attrs, MemTxResult *result)
{ {
assert(addr < cache->len && 8 <= cache->len - addr); assert(addr < cache->len && 8 <= cache->len - addr);
fuzz_dma_read_cb(cache->xlat + addr, 8, cache->mrs.mr, false); fuzz_dma_read_cb(cache->xlat + addr, 8, cache->mrs.mr);
if (likely(cache->ptr)) { if (likely(cache->ptr)) {
return LD_P(q)(cache->ptr + addr); return LD_P(q)(cache->ptr + addr);
} else { } else {
@ -52,7 +52,7 @@ static inline uint32_t ADDRESS_SPACE_LD_CACHED(uw)(MemoryRegionCache *cache,
hwaddr addr, MemTxAttrs attrs, MemTxResult *result) hwaddr addr, MemTxAttrs attrs, MemTxResult *result)
{ {
assert(addr < cache->len && 2 <= cache->len - addr); assert(addr < cache->len && 2 <= cache->len - addr);
fuzz_dma_read_cb(cache->xlat + addr, 2, cache->mrs.mr, false); fuzz_dma_read_cb(cache->xlat + addr, 2, cache->mrs.mr);
if (likely(cache->ptr)) { if (likely(cache->ptr)) {
return LD_P(uw)(cache->ptr + addr); return LD_P(uw)(cache->ptr + addr);
} else { } else {

View File

@ -24,6 +24,7 @@ struct EventNotifier {
#else #else
int rfd; int rfd;
int wfd; int wfd;
bool initialized;
#endif #endif
}; };

View File

@ -42,7 +42,7 @@ static inline uint32_t glue(address_space_ldl_internal, SUFFIX)(ARG1_DECL,
MO_32 | devend_memop(endian), attrs); MO_32 | devend_memop(endian), attrs);
} else { } else {
/* RAM case */ /* RAM case */
fuzz_dma_read_cb(addr, 4, mr, false); fuzz_dma_read_cb(addr, 4, mr);
ptr = qemu_map_ram_ptr(mr->ram_block, addr1); ptr = qemu_map_ram_ptr(mr->ram_block, addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:
@ -111,7 +111,7 @@ static inline uint64_t glue(address_space_ldq_internal, SUFFIX)(ARG1_DECL,
MO_64 | devend_memop(endian), attrs); MO_64 | devend_memop(endian), attrs);
} else { } else {
/* RAM case */ /* RAM case */
fuzz_dma_read_cb(addr, 8, mr, false); fuzz_dma_read_cb(addr, 8, mr);
ptr = qemu_map_ram_ptr(mr->ram_block, addr1); ptr = qemu_map_ram_ptr(mr->ram_block, addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:
@ -177,7 +177,7 @@ uint32_t glue(address_space_ldub, SUFFIX)(ARG1_DECL,
r = memory_region_dispatch_read(mr, addr1, &val, MO_8, attrs); r = memory_region_dispatch_read(mr, addr1, &val, MO_8, attrs);
} else { } else {
/* RAM case */ /* RAM case */
fuzz_dma_read_cb(addr, 1, mr, false); fuzz_dma_read_cb(addr, 1, mr);
ptr = qemu_map_ram_ptr(mr->ram_block, addr1); ptr = qemu_map_ram_ptr(mr->ram_block, addr1);
val = ldub_p(ptr); val = ldub_p(ptr);
r = MEMTX_OK; r = MEMTX_OK;
@ -215,7 +215,7 @@ static inline uint32_t glue(address_space_lduw_internal, SUFFIX)(ARG1_DECL,
MO_16 | devend_memop(endian), attrs); MO_16 | devend_memop(endian), attrs);
} else { } else {
/* RAM case */ /* RAM case */
fuzz_dma_read_cb(addr, 2, mr, false); fuzz_dma_read_cb(addr, 2, mr);
ptr = qemu_map_ram_ptr(mr->ram_block, addr1); ptr = qemu_map_ram_ptr(mr->ram_block, addr1);
switch (endian) { switch (endian) {
case DEVICE_LITTLE_ENDIAN: case DEVICE_LITTLE_ENDIAN:

View File

@ -18,6 +18,9 @@ config_host = keyval.load(meson.current_build_dir() / 'config-host.mak')
enable_modules = 'CONFIG_MODULES' in config_host enable_modules = 'CONFIG_MODULES' in config_host
enable_static = 'CONFIG_STATIC' in config_host enable_static = 'CONFIG_STATIC' in config_host
# Allow both shared and static libraries unless --enable-static
static_kwargs = enable_static ? {'static': true} : {}
# Temporary directory used for files created while # Temporary directory used for files created while
# configure runs. Since it is in the build directory # configure runs. Since it is in the build directory
# we can safely blow away any previous version of it # we can safely blow away any previous version of it
@ -224,10 +227,17 @@ tcg_arch = config_host['ARCH']
if not get_option('tcg').disabled() if not get_option('tcg').disabled()
if cpu not in supported_cpus if cpu not in supported_cpus
if get_option('tcg_interpreter') if get_option('tcg_interpreter')
warning('Unsupported CPU @0@, will use TCG with TCI (experimental)'.format(cpu)) warning('Unsupported CPU @0@, will use TCG with TCI (experimental and slow)'.format(cpu))
else else
error('Unsupported CPU @0@, try --enable-tcg-interpreter'.format(cpu)) error('Unsupported CPU @0@, try --enable-tcg-interpreter'.format(cpu))
endif endif
elif get_option('tcg_interpreter')
warning('Use of the TCG interpretor is not recommended on this host')
warning('architecture. There is a native TCG execution backend available')
warning('which provides substantially better performance and reliability.')
warning('It is strongly recommended to remove the --enable-tcg-interpreter')
warning('configuration option on this architecture to use the native')
warning('backend.')
endif endif
if get_option('tcg_interpreter') if get_option('tcg_interpreter')
tcg_arch = 'tci' tcg_arch = 'tci'
@ -311,14 +321,14 @@ endif
pixman = not_found pixman = not_found
if have_system or have_tools if have_system or have_tools
pixman = dependency('pixman-1', required: have_system, version:'>=0.21.8', pixman = dependency('pixman-1', required: have_system, version:'>=0.21.8',
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
endif endif
pam = not_found pam = not_found
if 'CONFIG_AUTH_PAM' in config_host if 'CONFIG_AUTH_PAM' in config_host
pam = cc.find_library('pam') pam = cc.find_library('pam')
endif endif
libaio = cc.find_library('aio', required: false) libaio = cc.find_library('aio', required: false)
zlib = dependency('zlib', required: true, static: enable_static) zlib = dependency('zlib', required: true, kwargs: static_kwargs)
linux_io_uring = not_found linux_io_uring = not_found
if 'CONFIG_LINUX_IO_URING' in config_host if 'CONFIG_LINUX_IO_URING' in config_host
linux_io_uring = declare_dependency(compile_args: config_host['LINUX_IO_URING_CFLAGS'].split(), linux_io_uring = declare_dependency(compile_args: config_host['LINUX_IO_URING_CFLAGS'].split(),
@ -333,7 +343,7 @@ libnfs = not_found
if not get_option('libnfs').auto() or have_block if not get_option('libnfs').auto() or have_block
libnfs = dependency('libnfs', version: '>=1.9.3', libnfs = dependency('libnfs', version: '>=1.9.3',
required: get_option('libnfs'), required: get_option('libnfs'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
endif endif
libattr_test = ''' libattr_test = '''
@ -354,7 +364,7 @@ if not get_option('attr').disabled()
else else
libattr = cc.find_library('attr', has_headers: ['attr/xattr.h'], libattr = cc.find_library('attr', has_headers: ['attr/xattr.h'],
required: get_option('attr'), required: get_option('attr'),
static: enable_static) kwargs: static_kwargs)
if libattr.found() and not \ if libattr.found() and not \
cc.links(libattr_test, dependencies: libattr, args: '-DCONFIG_LIBATTR') cc.links(libattr_test, dependencies: libattr, args: '-DCONFIG_LIBATTR')
libattr = not_found libattr = not_found
@ -381,14 +391,14 @@ seccomp = not_found
if not get_option('seccomp').auto() or have_system or have_tools if not get_option('seccomp').auto() or have_system or have_tools
seccomp = dependency('libseccomp', version: '>=2.3.0', seccomp = dependency('libseccomp', version: '>=2.3.0',
required: get_option('seccomp'), required: get_option('seccomp'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
endif endif
libcap_ng = not_found libcap_ng = not_found
if not get_option('cap_ng').auto() or have_system or have_tools if not get_option('cap_ng').auto() or have_system or have_tools
libcap_ng = cc.find_library('cap-ng', has_headers: ['cap-ng.h'], libcap_ng = cc.find_library('cap-ng', has_headers: ['cap-ng.h'],
required: get_option('cap_ng'), required: get_option('cap_ng'),
static: enable_static) kwargs: static_kwargs)
endif endif
if libcap_ng.found() and not cc.links(''' if libcap_ng.found() and not cc.links('''
#include <cap-ng.h> #include <cap-ng.h>
@ -409,7 +419,7 @@ if get_option('xkbcommon').auto() and not have_system and not have_tools
xkbcommon = not_found xkbcommon = not_found
else else
xkbcommon = dependency('xkbcommon', required: get_option('xkbcommon'), xkbcommon = dependency('xkbcommon', required: get_option('xkbcommon'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
endif endif
vde = not_found vde = not_found
if config_host.has_key('CONFIG_VDE') if config_host.has_key('CONFIG_VDE')
@ -445,13 +455,13 @@ libiscsi = not_found
if not get_option('libiscsi').auto() or have_block if not get_option('libiscsi').auto() or have_block
libiscsi = dependency('libiscsi', version: '>=1.9.0', libiscsi = dependency('libiscsi', version: '>=1.9.0',
required: get_option('libiscsi'), required: get_option('libiscsi'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
endif endif
zstd = not_found zstd = not_found
if not get_option('zstd').auto() or have_block if not get_option('zstd').auto() or have_block
zstd = dependency('libzstd', version: '>=1.4.0', zstd = dependency('libzstd', version: '>=1.4.0',
required: get_option('zstd'), required: get_option('zstd'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
endif endif
gbm = not_found gbm = not_found
if 'CONFIG_GBM' in config_host if 'CONFIG_GBM' in config_host
@ -468,14 +478,14 @@ if not get_option('curl').auto() or have_block
curl = dependency('libcurl', version: '>=7.29.0', curl = dependency('libcurl', version: '>=7.29.0',
method: 'pkg-config', method: 'pkg-config',
required: get_option('curl'), required: get_option('curl'),
static: enable_static) kwargs: static_kwargs)
endif endif
libudev = not_found libudev = not_found
if targetos == 'linux' and (have_system or have_tools) if targetos == 'linux' and (have_system or have_tools)
libudev = dependency('libudev', libudev = dependency('libudev',
method: 'pkg-config', method: 'pkg-config',
required: get_option('libudev'), required: get_option('libudev'),
static: enable_static) kwargs: static_kwargs)
endif endif
mpathlibs = [libudev] mpathlibs = [libudev]
@ -511,17 +521,17 @@ if targetos == 'linux' and have_tools and not get_option('mpath').disabled()
}''' }'''
libmpathpersist = cc.find_library('mpathpersist', libmpathpersist = cc.find_library('mpathpersist',
required: get_option('mpath'), required: get_option('mpath'),
static: enable_static) kwargs: static_kwargs)
if libmpathpersist.found() if libmpathpersist.found()
mpathlibs += libmpathpersist mpathlibs += libmpathpersist
if enable_static if enable_static
mpathlibs += cc.find_library('devmapper', mpathlibs += cc.find_library('devmapper',
required: get_option('mpath'), required: get_option('mpath'),
static: enable_static) kwargs: static_kwargs)
endif endif
mpathlibs += cc.find_library('multipath', mpathlibs += cc.find_library('multipath',
required: get_option('mpath'), required: get_option('mpath'),
static: enable_static) kwargs: static_kwargs)
foreach lib: mpathlibs foreach lib: mpathlibs
if not lib.found() if not lib.found()
mpathlibs = [] mpathlibs = []
@ -571,7 +581,7 @@ if have_system and not get_option('curses').disabled()
curses = dependency(curses_dep, curses = dependency(curses_dep,
required: false, required: false,
method: 'pkg-config', method: 'pkg-config',
static: enable_static) kwargs: static_kwargs)
endif endif
endforeach endforeach
msg = get_option('curses').enabled() ? 'curses library not found' : '' msg = get_option('curses').enabled() ? 'curses library not found' : ''
@ -596,7 +606,7 @@ if have_system and not get_option('curses').disabled()
foreach curses_libname : curses_libname_list foreach curses_libname : curses_libname_list
libcurses = cc.find_library(curses_libname, libcurses = cc.find_library(curses_libname,
required: false, required: false,
static: enable_static) kwargs: static_kwargs)
if libcurses.found() if libcurses.found()
if cc.links(curses_test, args: curses_compile_args, dependencies: libcurses) if cc.links(curses_test, args: curses_compile_args, dependencies: libcurses)
curses = declare_dependency(compile_args: curses_compile_args, curses = declare_dependency(compile_args: curses_compile_args,
@ -647,7 +657,7 @@ brlapi = not_found
if not get_option('brlapi').auto() or have_system if not get_option('brlapi').auto() or have_system
brlapi = cc.find_library('brlapi', has_headers: ['brlapi.h'], brlapi = cc.find_library('brlapi', has_headers: ['brlapi.h'],
required: get_option('brlapi'), required: get_option('brlapi'),
static: enable_static) kwargs: static_kwargs)
if brlapi.found() and not cc.links(''' if brlapi.found() and not cc.links('''
#include <brlapi.h> #include <brlapi.h>
#include <stddef.h> #include <stddef.h>
@ -663,7 +673,7 @@ endif
sdl = not_found sdl = not_found
if not get_option('sdl').auto() or (have_system and not cocoa.found()) if not get_option('sdl').auto() or (have_system and not cocoa.found())
sdl = dependency('sdl2', required: get_option('sdl'), static: enable_static) sdl = dependency('sdl2', required: get_option('sdl'), kwargs: static_kwargs)
sdl_image = not_found sdl_image = not_found
endif endif
if sdl.found() if sdl.found()
@ -671,7 +681,7 @@ if sdl.found()
sdl = declare_dependency(compile_args: '-Wno-undef', sdl = declare_dependency(compile_args: '-Wno-undef',
dependencies: sdl) dependencies: sdl)
sdl_image = dependency('SDL2_image', required: get_option('sdl_image'), sdl_image = dependency('SDL2_image', required: get_option('sdl_image'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
else else
if get_option('sdl_image').enabled() if get_option('sdl_image').enabled()
error('sdl-image required, but SDL was @0@'.format( error('sdl-image required, but SDL was @0@'.format(
@ -683,11 +693,12 @@ endif
rbd = not_found rbd = not_found
if not get_option('rbd').auto() or have_block if not get_option('rbd').auto() or have_block
librados = cc.find_library('rados', required: get_option('rbd'), librados = cc.find_library('rados', required: get_option('rbd'),
static: enable_static) kwargs: static_kwargs)
librbd = cc.find_library('rbd', has_headers: ['rbd/librbd.h'], librbd = cc.find_library('rbd', has_headers: ['rbd/librbd.h'],
required: get_option('rbd'), required: get_option('rbd'),
static: enable_static) kwargs: static_kwargs)
if librados.found() and librbd.found() and cc.links(''' if librados.found() and librbd.found()
if cc.links('''
#include <stdio.h> #include <stdio.h>
#include <rbd/librbd.h> #include <rbd/librbd.h>
int main(void) { int main(void) {
@ -696,6 +707,11 @@ if not get_option('rbd').auto() or have_block
return 0; return 0;
}''', dependencies: [librbd, librados]) }''', dependencies: [librbd, librados])
rbd = declare_dependency(dependencies: [librbd, librados]) rbd = declare_dependency(dependencies: [librbd, librados])
elif get_option('rbd').enabled()
error('could not link librados')
else
warning('could not link librados, disabling')
endif
endif endif
endif endif
@ -705,7 +721,7 @@ glusterfs_iocb_has_stat = false
if not get_option('glusterfs').auto() or have_block if not get_option('glusterfs').auto() or have_block
glusterfs = dependency('glusterfs-api', version: '>=3', glusterfs = dependency('glusterfs-api', version: '>=3',
required: get_option('glusterfs'), required: get_option('glusterfs'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
if glusterfs.found() if glusterfs.found()
glusterfs_ftruncate_has_stat = cc.links(''' glusterfs_ftruncate_has_stat = cc.links('''
#include <glusterfs/api/glfs.h> #include <glusterfs/api/glfs.h>
@ -744,7 +760,7 @@ libbzip2 = not_found
if not get_option('bzip2').auto() or have_block if not get_option('bzip2').auto() or have_block
libbzip2 = cc.find_library('bz2', has_headers: ['bzlib.h'], libbzip2 = cc.find_library('bz2', has_headers: ['bzlib.h'],
required: get_option('bzip2'), required: get_option('bzip2'),
static: enable_static) kwargs: static_kwargs)
if libbzip2.found() and not cc.links(''' if libbzip2.found() and not cc.links('''
#include <bzlib.h> #include <bzlib.h>
int main(void) { BZ2_bzlibVersion(); return 0; }''', dependencies: libbzip2) int main(void) { BZ2_bzlibVersion(); return 0; }''', dependencies: libbzip2)
@ -761,7 +777,7 @@ liblzfse = not_found
if not get_option('lzfse').auto() or have_block if not get_option('lzfse').auto() or have_block
liblzfse = cc.find_library('lzfse', has_headers: ['lzfse.h'], liblzfse = cc.find_library('lzfse', has_headers: ['lzfse.h'],
required: get_option('lzfse'), required: get_option('lzfse'),
static: enable_static) kwargs: static_kwargs)
endif endif
if liblzfse.found() and not cc.links(''' if liblzfse.found() and not cc.links('''
#include <lzfse.h> #include <lzfse.h>
@ -798,12 +814,12 @@ if not get_option('gtk').auto() or (have_system and not cocoa.found())
gtk = dependency('gtk+-3.0', version: '>=3.22.0', gtk = dependency('gtk+-3.0', version: '>=3.22.0',
method: 'pkg-config', method: 'pkg-config',
required: get_option('gtk'), required: get_option('gtk'),
static: enable_static) kwargs: static_kwargs)
if gtk.found() if gtk.found()
gtkx11 = dependency('gtk+-x11-3.0', version: '>=3.22.0', gtkx11 = dependency('gtk+-x11-3.0', version: '>=3.22.0',
method: 'pkg-config', method: 'pkg-config',
required: false, required: false,
static: enable_static) kwargs: static_kwargs)
gtk = declare_dependency(dependencies: [gtk, gtkx11]) gtk = declare_dependency(dependencies: [gtk, gtkx11])
endif endif
endif endif
@ -816,7 +832,7 @@ endif
x11 = not_found x11 = not_found
if gtkx11.found() or 'lm32-softmmu' in target_dirs if gtkx11.found() or 'lm32-softmmu' in target_dirs
x11 = dependency('x11', method: 'pkg-config', required: gtkx11.found(), x11 = dependency('x11', method: 'pkg-config', required: gtkx11.found(),
static: enable_static) kwargs: static_kwargs)
endif endif
vnc = not_found vnc = not_found
png = not_found png = not_found
@ -825,12 +841,12 @@ sasl = not_found
if get_option('vnc').enabled() if get_option('vnc').enabled()
vnc = declare_dependency() # dummy dependency vnc = declare_dependency() # dummy dependency
png = dependency('libpng', required: get_option('vnc_png'), png = dependency('libpng', required: get_option('vnc_png'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
jpeg = dependency('libjpeg', required: get_option('vnc_jpeg'), jpeg = dependency('libjpeg', required: get_option('vnc_jpeg'),
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
sasl = cc.find_library('sasl2', has_headers: ['sasl/sasl.h'], sasl = cc.find_library('sasl2', has_headers: ['sasl/sasl.h'],
required: get_option('vnc_sasl'), required: get_option('vnc_sasl'),
static: enable_static) kwargs: static_kwargs)
if sasl.found() if sasl.found()
sasl = declare_dependency(dependencies: sasl, sasl = declare_dependency(dependencies: sasl,
compile_args: '-DSTRUCT_IOVEC_DEFINED') compile_args: '-DSTRUCT_IOVEC_DEFINED')
@ -841,7 +857,7 @@ snappy = not_found
if not get_option('snappy').auto() or have_system if not get_option('snappy').auto() or have_system
snappy = cc.find_library('snappy', has_headers: ['snappy-c.h'], snappy = cc.find_library('snappy', has_headers: ['snappy-c.h'],
required: get_option('snappy'), required: get_option('snappy'),
static: enable_static) kwargs: static_kwargs)
endif endif
if snappy.found() and not cc.links(''' if snappy.found() and not cc.links('''
#include <snappy-c.h> #include <snappy-c.h>
@ -858,7 +874,7 @@ lzo = not_found
if not get_option('lzo').auto() or have_system if not get_option('lzo').auto() or have_system
lzo = cc.find_library('lzo2', has_headers: ['lzo/lzo1x.h'], lzo = cc.find_library('lzo2', has_headers: ['lzo/lzo1x.h'],
required: get_option('lzo'), required: get_option('lzo'),
static: enable_static) kwargs: static_kwargs)
endif endif
if lzo.found() and not cc.links(''' if lzo.found() and not cc.links('''
#include <lzo/lzo1x.h> #include <lzo/lzo1x.h>
@ -893,7 +909,7 @@ u2f = not_found
if have_system if have_system
u2f = dependency('u2f-emu', required: get_option('u2f'), u2f = dependency('u2f-emu', required: get_option('u2f'),
method: 'pkg-config', method: 'pkg-config',
static: enable_static) kwargs: static_kwargs)
endif endif
usbredir = not_found usbredir = not_found
if 'CONFIG_USB_REDIR' in config_host if 'CONFIG_USB_REDIR' in config_host
@ -920,7 +936,7 @@ if 'CONFIG_TASN1' in config_host
link_args: config_host['TASN1_LIBS'].split()) link_args: config_host['TASN1_LIBS'].split())
endif endif
keyutils = dependency('libkeyutils', required: false, keyutils = dependency('libkeyutils', required: false,
method: 'pkg-config', static: enable_static) method: 'pkg-config', kwargs: static_kwargs)
has_gettid = cc.has_function('gettid') has_gettid = cc.has_function('gettid')
@ -979,7 +995,7 @@ endif
fuse = dependency('fuse3', required: get_option('fuse'), fuse = dependency('fuse3', required: get_option('fuse'),
version: '>=3.1', method: 'pkg-config', version: '>=3.1', method: 'pkg-config',
static: enable_static) kwargs: static_kwargs)
fuse_lseek = not_found fuse_lseek = not_found
if not get_option('fuse_lseek').disabled() if not get_option('fuse_lseek').disabled()
@ -1367,7 +1383,7 @@ capstone_opt = get_option('capstone')
if capstone_opt in ['enabled', 'auto', 'system'] if capstone_opt in ['enabled', 'auto', 'system']
have_internal = fs.exists(meson.current_source_dir() / 'capstone/Makefile') have_internal = fs.exists(meson.current_source_dir() / 'capstone/Makefile')
capstone = dependency('capstone', version: '>=4.0', capstone = dependency('capstone', version: '>=4.0',
static: enable_static, method: 'pkg-config', kwargs: static_kwargs, method: 'pkg-config',
required: capstone_opt == 'system' or required: capstone_opt == 'system' or
capstone_opt == 'enabled' and not have_internal) capstone_opt == 'enabled' and not have_internal)
if capstone.found() if capstone.found()
@ -1477,7 +1493,7 @@ if have_system
slirp_opt = get_option('slirp') slirp_opt = get_option('slirp')
if slirp_opt in ['enabled', 'auto', 'system'] if slirp_opt in ['enabled', 'auto', 'system']
have_internal = fs.exists(meson.current_source_dir() / 'slirp/meson.build') have_internal = fs.exists(meson.current_source_dir() / 'slirp/meson.build')
slirp = dependency('slirp', static: enable_static, slirp = dependency('slirp', kwargs: static_kwargs,
method: 'pkg-config', method: 'pkg-config',
required: slirp_opt == 'system' or required: slirp_opt == 'system' or
slirp_opt == 'enabled' and not have_internal) slirp_opt == 'enabled' and not have_internal)
@ -1556,7 +1572,7 @@ fdt_opt = get_option('fdt')
if have_system if have_system
if fdt_opt in ['enabled', 'auto', 'system'] if fdt_opt in ['enabled', 'auto', 'system']
have_internal = fs.exists(meson.current_source_dir() / 'dtc/libfdt/Makefile.libfdt') have_internal = fs.exists(meson.current_source_dir() / 'dtc/libfdt/Makefile.libfdt')
fdt = cc.find_library('fdt', static: enable_static, fdt = cc.find_library('fdt', kwargs: static_kwargs,
required: fdt_opt == 'system' or required: fdt_opt == 'system' or
fdt_opt == 'enabled' and not have_internal) fdt_opt == 'enabled' and not have_internal)
if fdt.found() and cc.links(''' if fdt.found() and cc.links('''
@ -1725,10 +1741,11 @@ target_softmmu_arch = {}
# TODO: add each directory to the subdirs from its own meson.build, once # TODO: add each directory to the subdirs from its own meson.build, once
# we have those # we have those
trace_events_subdirs = [ trace_events_subdirs = [
'accel/kvm',
'accel/tcg',
'crypto', 'crypto',
'qapi',
'qom',
'monitor', 'monitor',
'util',
] ]
if have_user if have_user
trace_events_subdirs += [ 'linux-user' ] trace_events_subdirs += [ 'linux-user' ]
@ -1744,6 +1761,7 @@ if have_block
endif endif
if have_system if have_system
trace_events_subdirs += [ trace_events_subdirs += [
'accel/kvm',
'audio', 'audio',
'backends', 'backends',
'backends/tpm', 'backends/tpm',
@ -1801,10 +1819,10 @@ if have_system
'ui', 'ui',
] ]
endif endif
trace_events_subdirs += [ if have_system or have_user
trace_events_subdirs += [
'accel/tcg',
'hw/core', 'hw/core',
'qapi',
'qom',
'target/arm', 'target/arm',
'target/hppa', 'target/hppa',
'target/i386', 'target/i386',
@ -1814,8 +1832,8 @@ trace_events_subdirs += [
'target/riscv', 'target/riscv',
'target/s390x', 'target/s390x',
'target/sparc', 'target/sparc',
'util', ]
] endif
vhost_user = not_found vhost_user = not_found
if 'CONFIG_VHOST_USER' in config_host if 'CONFIG_VHOST_USER' in config_host
@ -1849,41 +1867,45 @@ libqemuutil = static_library('qemuutil',
qemuutil = declare_dependency(link_with: libqemuutil, qemuutil = declare_dependency(link_with: libqemuutil,
sources: genh + version_res) sources: genh + version_res)
decodetree = generator(find_program('scripts/decodetree.py'), if have_system or have_user
decodetree = generator(find_program('scripts/decodetree.py'),
output: 'decode-@BASENAME@.c.inc', output: 'decode-@BASENAME@.c.inc',
arguments: ['@INPUT@', '@EXTRA_ARGS@', '-o', '@OUTPUT@']) arguments: ['@INPUT@', '@EXTRA_ARGS@', '-o', '@OUTPUT@'])
subdir('libdecnumber')
subdir('target')
endif
subdir('audio') subdir('audio')
subdir('io') subdir('io')
subdir('chardev') subdir('chardev')
subdir('fsdev') subdir('fsdev')
subdir('libdecnumber')
subdir('target')
subdir('dump') subdir('dump')
block_ss.add(files( if have_block
block_ss.add(files(
'block.c', 'block.c',
'blockjob.c', 'blockjob.c',
'job.c', 'job.c',
'qemu-io-cmds.c', 'qemu-io-cmds.c',
)) ))
block_ss.add(when: 'CONFIG_REPLICATION', if_true: files('replication.c')) block_ss.add(when: 'CONFIG_REPLICATION', if_true: files('replication.c'))
subdir('nbd') subdir('nbd')
subdir('scsi') subdir('scsi')
subdir('block') subdir('block')
blockdev_ss.add(files( blockdev_ss.add(files(
'blockdev.c', 'blockdev.c',
'blockdev-nbd.c', 'blockdev-nbd.c',
'iothread.c', 'iothread.c',
'job-qmp.c', 'job-qmp.c',
), gnutls) ), gnutls)
# os-posix.c contains POSIX-specific functions used by qemu-storage-daemon, # os-posix.c contains POSIX-specific functions used by qemu-storage-daemon,
# os-win32.c does not # os-win32.c does not
blockdev_ss.add(when: 'CONFIG_POSIX', if_true: files('os-posix.c')) blockdev_ss.add(when: 'CONFIG_POSIX', if_true: files('os-posix.c'))
softmmu_ss.add(when: 'CONFIG_WIN32', if_true: [files('os-win32.c')]) softmmu_ss.add(when: 'CONFIG_WIN32', if_true: [files('os-win32.c')])
endif
common_ss.add(files('cpus-common.c')) common_ss.add(files('cpus-common.c'))
@ -2500,8 +2522,12 @@ if have_system
endif endif
summary_info += {'TCG support': config_all.has_key('CONFIG_TCG')} summary_info += {'TCG support': config_all.has_key('CONFIG_TCG')}
if config_all.has_key('CONFIG_TCG') if config_all.has_key('CONFIG_TCG')
if get_option('tcg_interpreter')
summary_info += {'TCG backend': 'TCI (TCG with bytecode interpreter, experimental and slow)'}
else
summary_info += {'TCG backend': 'native (@0@)'.format(cpu)}
endif
summary_info += {'TCG debug enabled': config_host.has_key('CONFIG_DEBUG_TCG')} summary_info += {'TCG debug enabled': config_host.has_key('CONFIG_DEBUG_TCG')}
summary_info += {'TCG interpreter': tcg_arch == 'tci'}
endif endif
summary_info += {'target list': ' '.join(target_dirs)} summary_info += {'target list': ' '.join(target_dirs)}
if have_system if have_system

View File

@ -40,7 +40,7 @@ option('xen_pci_passthrough', type: 'feature', value: 'auto',
option('tcg', type: 'feature', value: 'auto', option('tcg', type: 'feature', value: 'auto',
description: 'TCG support') description: 'TCG support')
option('tcg_interpreter', type: 'boolean', value: false, option('tcg_interpreter', type: 'boolean', value: false,
description: 'TCG bytecode interpreter (TCI)') description: 'TCG with bytecode interpreter (experimental and slow)')
option('cfi', type: 'boolean', value: 'false', option('cfi', type: 'boolean', value: 'false',
description: 'Control-Flow Integrity (CFI)') description: 'Control-Flow Integrity (CFI)')
option('cfi_debug', type: 'boolean', value: 'false', option('cfi_debug', type: 'boolean', value: 'false',

View File

@ -9,7 +9,7 @@ if install_edk2_blobs
] ]
configure_file(input: files(f), configure_file(input: files(f),
output: f, output: f,
configuration: {'DATADIR': qemu_datadir}, configuration: {'DATADIR': get_option('prefix') / qemu_datadir},
install: get_option('install_blobs'), install: get_option('install_blobs'),
install_dir: qemu_datadir / 'firmware') install_dir: qemu_datadir / 'firmware')
endforeach endforeach

View File

@ -12,6 +12,7 @@ if install_edk2_blobs
foreach f : fds foreach f : fds
custom_target(f, custom_target(f,
build_by_default: have_system,
output: f, output: f,
input: '@0@.bz2'.format(f), input: '@0@.bz2'.format(f),
capture: true, capture: true,

View File

@ -4,18 +4,20 @@ util_ss.add(files(
'qapi-dealloc-visitor.c', 'qapi-dealloc-visitor.c',
'qapi-util.c', 'qapi-util.c',
'qapi-visit-core.c', 'qapi-visit-core.c',
'qmp-dispatch.c',
'qmp-event.c',
'qmp-registry.c',
'qobject-input-visitor.c', 'qobject-input-visitor.c',
'qobject-output-visitor.c', 'qobject-output-visitor.c',
'string-input-visitor.c', 'string-input-visitor.c',
'string-output-visitor.c', 'string-output-visitor.c',
)) ))
if have_system or have_tools
util_ss.add(files(
'qmp-dispatch.c',
'qmp-event.c',
'qmp-registry.c',
))
endif
qapi_all_modules = [ qapi_all_modules = [
'acpi',
'audio',
'authz', 'authz',
'block', 'block',
'block-core', 'block-core',
@ -35,20 +37,30 @@ qapi_all_modules = [
'misc-target', 'misc-target',
'net', 'net',
'pragma', 'pragma',
'qdev',
'pci',
'qom', 'qom',
'rdma',
'replay', 'replay',
'rocker',
'run-state', 'run-state',
'sockets', 'sockets',
'tpm',
'trace', 'trace',
'transaction', 'transaction',
'ui',
'yank', 'yank',
] ]
if have_system
qapi_all_modules += [
'acpi',
'audio',
'qdev',
'pci',
'rdma',
'rocker',
'tpm',
]
endif
if have_system or have_tools
qapi_all_modules += [
'ui',
]
endif
qapi_storage_daemon_modules = [ qapi_storage_daemon_modules = [
'block-core', 'block-core',

View File

@ -35,7 +35,8 @@ DEF("machine", HAS_ARG, QEMU_OPTION_machine, \
" suppress-vmdesc=on|off disables self-describing migration (default=off)\n" " suppress-vmdesc=on|off disables self-describing migration (default=off)\n"
" nvdimm=on|off controls NVDIMM support (default=off)\n" " nvdimm=on|off controls NVDIMM support (default=off)\n"
" memory-encryption=@var{} memory encryption object to use (default=none)\n" " memory-encryption=@var{} memory encryption object to use (default=none)\n"
" hmat=on|off controls ACPI HMAT support (default=off)\n", " hmat=on|off controls ACPI HMAT support (default=off)\n"
" memory-backend='backend-id' specifies explicitly provided backend for main RAM (default=none)\n",
QEMU_ARCH_ALL) QEMU_ARCH_ALL)
SRST SRST
``-machine [type=]name[,prop=value[,...]]`` ``-machine [type=]name[,prop=value[,...]]``
@ -96,6 +97,29 @@ SRST
``hmat=on|off`` ``hmat=on|off``
Enables or disables ACPI Heterogeneous Memory Attribute Table Enables or disables ACPI Heterogeneous Memory Attribute Table
(HMAT) support. The default is off. (HMAT) support. The default is off.
``memory-backend='id'``
An alternative to legacy ``-mem-path`` and ``mem-prealloc`` options.
Allows to use a memory backend as main RAM.
For example:
::
-object memory-backend-file,id=pc.ram,size=512M,mem-path=/hugetlbfs,prealloc=on,share=on
-machine memory-backend=pc.ram
-m 512M
Migration compatibility note:
a) as backend id one shall use value of 'default-ram-id', advertised by
machine type (available via ``query-machines`` QMP command), if migration
to/from old QEMU (<5.0) is expected.
b) for machine types 4.0 and older, user shall
use ``x-use-canonical-path-for-ramblock-id=off`` backend option
if migration to/from old QEMU (<5.0) is expected.
For example:
::
-object memory-backend-ram,id=pc.ram,size=512M,x-use-canonical-path-for-ramblock-id=off
-machine memory-backend=pc.ram
-m 512M
ERST ERST
HXCOMM Deprecated by -machine HXCOMM Deprecated by -machine

View File

@ -261,7 +261,7 @@ def clear_bits(newtrace, outpath):
data_try = hex(int("".join(data_bin_list), 2)) data_try = hex(int("".join(data_bin_list), 2))
# It seems qtest only accepts padded hex-values. # It seems qtest only accepts padded hex-values.
if len(data_try) % 2 == 1: if len(data_try) % 2 == 1:
data_try = data_try[:2] + "0" + data_try[2:-1] data_try = data_try[:2] + "0" + data_try[2:]
newtrace[i] = "{prefix} {data_try}\n".format( newtrace[i] = "{prefix} {data_try}\n".format(
prefix=prefix, prefix=prefix,

View File

@ -90,14 +90,21 @@ static void cpu_throttle_timer_tick(void *opaque)
void cpu_throttle_set(int new_throttle_pct) void cpu_throttle_set(int new_throttle_pct)
{ {
/*
* boolean to store whether throttle is already active or not,
* before modifying throttle_percentage
*/
bool throttle_active = cpu_throttle_active();
/* Ensure throttle percentage is within valid range */ /* Ensure throttle percentage is within valid range */
new_throttle_pct = MIN(new_throttle_pct, CPU_THROTTLE_PCT_MAX); new_throttle_pct = MIN(new_throttle_pct, CPU_THROTTLE_PCT_MAX);
new_throttle_pct = MAX(new_throttle_pct, CPU_THROTTLE_PCT_MIN); new_throttle_pct = MAX(new_throttle_pct, CPU_THROTTLE_PCT_MIN);
qatomic_set(&throttle_percentage, new_throttle_pct); qatomic_set(&throttle_percentage, new_throttle_pct);
timer_mod(throttle_timer, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL_RT) + if (!throttle_active) {
CPU_THROTTLE_TIMESLICE_NS); cpu_throttle_timer_tick(NULL);
}
} }
void cpu_throttle_stop(void) void cpu_throttle_stop(void)

View File

@ -1440,7 +1440,7 @@ MemTxResult memory_region_dispatch_read(MemoryRegion *mr,
unsigned size = memop_size(op); unsigned size = memop_size(op);
MemTxResult r; MemTxResult r;
fuzz_dma_read_cb(addr, size, mr, false); fuzz_dma_read_cb(addr, size, mr);
if (!memory_region_access_valid(mr, addr, size, false, attrs)) { if (!memory_region_access_valid(mr, addr, size, false, attrs)) {
*pval = unassigned_mem_read(mr, addr, size); *pval = unassigned_mem_read(mr, addr, size);
return MEMTX_DECODE_ERROR; return MEMTX_DECODE_ERROR;
@ -3285,8 +3285,7 @@ void memory_region_init_rom_device(MemoryRegion *mr,
#ifdef CONFIG_FUZZ #ifdef CONFIG_FUZZ
void __attribute__((weak)) fuzz_dma_read_cb(size_t addr, void __attribute__((weak)) fuzz_dma_read_cb(size_t addr,
size_t len, size_t len,
MemoryRegion *mr, MemoryRegion *mr)
bool is_write)
{ {
} }
#endif #endif

View File

@ -2839,7 +2839,7 @@ MemTxResult flatview_read_continue(FlatView *fv, hwaddr addr,
stn_he_p(buf, l, val); stn_he_p(buf, l, val);
} else { } else {
/* RAM case */ /* RAM case */
fuzz_dma_read_cb(addr, len, mr, false); fuzz_dma_read_cb(addr, len, mr);
ram_ptr = qemu_ram_ptr_length(mr->ram_block, addr1, &l, false); ram_ptr = qemu_ram_ptr_length(mr->ram_block, addr1, &l, false);
memcpy(buf, ram_ptr, l); memcpy(buf, ram_ptr, l);
} }
@ -3200,7 +3200,7 @@ void *address_space_map(AddressSpace *as,
memory_region_ref(mr); memory_region_ref(mr);
*plen = flatview_extend_translation(fv, addr, len, mr, xlat, *plen = flatview_extend_translation(fv, addr, len, mr, xlat,
l, is_write, attrs); l, is_write, attrs);
fuzz_dma_read_cb(addr, *plen, mr, is_write); fuzz_dma_read_cb(addr, *plen, mr);
ptr = qemu_ram_ptr_length(mr->ram_block, xlat, plen, true); ptr = qemu_ram_ptr_length(mr->ram_block, xlat, plen, true);
return ptr; return ptr;

View File

@ -53,4 +53,6 @@ endif
if have_system if have_system
stub_ss.add(files('semihost.c')) stub_ss.add(files('semihost.c'))
stub_ss.add(files('xen-hw-stub.c')) stub_ss.add(files('xen-hw-stub.c'))
else
stub_ss.add(files('qdev.c'))
endif endif

23
stubs/qdev.c Normal file
View File

@ -0,0 +1,23 @@
/*
* QOM stubs
*
* Copyright (c) 2021 Red Hat, Inc.
*
* Author:
* Philippe Mathieu-Daudé <philmd@redhat.com>
*
* SPDX-License-Identifier: GPL-2.0-or-later
*
* This work is licensed under the terms of the GNU GPL, version 2 or later.
* See the COPYING file in the top-level directory.
*/
#include "qemu/osdep.h"
#include "qapi/qapi-events-qdev.h"
void qapi_event_send_device_deleted(bool has_device,
const char *device,
const char *path)
{
/* Nothing to do. */
}

View File

@ -667,7 +667,7 @@ static void x86_cpu_vendor_words2str(char *dst, uint32_t vendor1,
CPUID_7_0_EBX_RDSEED */ CPUID_7_0_EBX_RDSEED */
#define TCG_7_0_ECX_FEATURES (CPUID_7_0_ECX_PKU | \ #define TCG_7_0_ECX_FEATURES (CPUID_7_0_ECX_PKU | \
/* CPUID_7_0_ECX_OSPKE is dynamic */ \ /* CPUID_7_0_ECX_OSPKE is dynamic */ \
CPUID_7_0_ECX_LA57) CPUID_7_0_ECX_LA57 | CPUID_7_0_ECX_PKS)
#define TCG_7_0_EDX_FEATURES 0 #define TCG_7_0_EDX_FEATURES 0
#define TCG_7_1_EAX_FEATURES 0 #define TCG_7_1_EAX_FEATURES 0
#define TCG_APM_FEATURES 0 #define TCG_APM_FEATURES 0
@ -926,11 +926,11 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
"npt", "lbrv", "svm-lock", "nrip-save", "npt", "lbrv", "svm-lock", "nrip-save",
"tsc-scale", "vmcb-clean", "flushbyasid", "decodeassists", "tsc-scale", "vmcb-clean", "flushbyasid", "decodeassists",
NULL, NULL, "pause-filter", NULL, NULL, NULL, "pause-filter", NULL,
"pfthreshold", NULL, NULL, NULL, "pfthreshold", "avic", NULL, "v-vmsave-vmload",
NULL, NULL, NULL, NULL, "vgif", NULL, NULL, NULL,
NULL, NULL, NULL, NULL,
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
"svme-addr-chk", NULL, NULL, NULL,
}, },
.cpuid = { .eax = 0x8000000A, .reg = R_EDX, }, .cpuid = { .eax = 0x8000000A, .reg = R_EDX, },
.tcg_features = TCG_SVM_FEATURES, .tcg_features = TCG_SVM_FEATURES,
@ -964,7 +964,7 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
"la57", NULL, NULL, NULL, "la57", NULL, NULL, NULL,
NULL, NULL, "rdpid", NULL, NULL, NULL, "rdpid", NULL,
NULL, "cldemote", NULL, "movdiri", NULL, "cldemote", NULL, "movdiri",
"movdir64b", NULL, NULL, NULL, "movdir64b", NULL, NULL, "pks",
}, },
.cpuid = { .cpuid = {
.eax = 7, .eax = 7,
@ -1215,7 +1215,7 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
"vmx-exit-save-efer", "vmx-exit-load-efer", "vmx-exit-save-efer", "vmx-exit-load-efer",
"vmx-exit-save-preemption-timer", "vmx-exit-clear-bndcfgs", "vmx-exit-save-preemption-timer", "vmx-exit-clear-bndcfgs",
NULL, "vmx-exit-clear-rtit-ctl", NULL, NULL, NULL, "vmx-exit-clear-rtit-ctl", NULL, NULL,
NULL, NULL, NULL, NULL, NULL, "vmx-exit-load-pkrs", NULL, NULL,
}, },
.msr = { .msr = {
.index = MSR_IA32_VMX_TRUE_EXIT_CTLS, .index = MSR_IA32_VMX_TRUE_EXIT_CTLS,
@ -1230,7 +1230,7 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
NULL, "vmx-entry-ia32e-mode", NULL, NULL, NULL, "vmx-entry-ia32e-mode", NULL, NULL,
NULL, "vmx-entry-load-perf-global-ctrl", "vmx-entry-load-pat", "vmx-entry-load-efer", NULL, "vmx-entry-load-perf-global-ctrl", "vmx-entry-load-pat", "vmx-entry-load-efer",
"vmx-entry-load-bndcfgs", NULL, "vmx-entry-load-rtit-ctl", NULL, "vmx-entry-load-bndcfgs", NULL, "vmx-entry-load-rtit-ctl", NULL,
NULL, NULL, NULL, NULL, NULL, NULL, "vmx-entry-load-pkrs", NULL,
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
}, },
@ -5073,6 +5073,11 @@ static uint64_t x86_cpu_get_supported_feature_word(FeatureWord w,
} else { } else {
return ~0; return ~0;
} }
#ifndef TARGET_X86_64
if (w == FEAT_8000_0001_EDX) {
r &= ~CPUID_EXT2_LM;
}
#endif
if (migratable_only) { if (migratable_only) {
r &= x86_cpu_get_migratable_flags(w); r &= x86_cpu_get_migratable_flags(w);
} }

View File

@ -247,6 +247,7 @@ typedef enum X86Seg {
#define CR4_SMEP_MASK (1U << 20) #define CR4_SMEP_MASK (1U << 20)
#define CR4_SMAP_MASK (1U << 21) #define CR4_SMAP_MASK (1U << 21)
#define CR4_PKE_MASK (1U << 22) #define CR4_PKE_MASK (1U << 22)
#define CR4_PKS_MASK (1U << 24)
#define DR6_BD (1 << 13) #define DR6_BD (1 << 13)
#define DR6_BS (1 << 14) #define DR6_BS (1 << 14)
@ -357,6 +358,7 @@ typedef enum X86Seg {
#define MSR_IA32_TSX_CTRL 0x122 #define MSR_IA32_TSX_CTRL 0x122
#define MSR_IA32_TSCDEADLINE 0x6e0 #define MSR_IA32_TSCDEADLINE 0x6e0
#define MSR_IA32_PKRS 0x6e1
#define FEATURE_CONTROL_LOCKED (1<<0) #define FEATURE_CONTROL_LOCKED (1<<0)
#define FEATURE_CONTROL_VMXON_ENABLED_OUTSIDE_SMX (1<<2) #define FEATURE_CONTROL_VMXON_ENABLED_OUTSIDE_SMX (1<<2)
@ -680,6 +682,10 @@ typedef uint64_t FeatureWordArray[FEATURE_WORDS];
#define CPUID_SVM_DECODEASSIST (1U << 7) #define CPUID_SVM_DECODEASSIST (1U << 7)
#define CPUID_SVM_PAUSEFILTER (1U << 10) #define CPUID_SVM_PAUSEFILTER (1U << 10)
#define CPUID_SVM_PFTHRESHOLD (1U << 12) #define CPUID_SVM_PFTHRESHOLD (1U << 12)
#define CPUID_SVM_AVIC (1U << 13)
#define CPUID_SVM_V_VMSAVE_VMLOAD (1U << 15)
#define CPUID_SVM_VGIF (1U << 16)
#define CPUID_SVM_SVME_ADDR_CHK (1U << 28)
/* Support RDFSBASE/RDGSBASE/WRFSBASE/WRGSBASE */ /* Support RDFSBASE/RDGSBASE/WRFSBASE/WRGSBASE */
#define CPUID_7_0_EBX_FSGSBASE (1U << 0) #define CPUID_7_0_EBX_FSGSBASE (1U << 0)
@ -768,6 +774,8 @@ typedef uint64_t FeatureWordArray[FEATURE_WORDS];
#define CPUID_7_0_ECX_MOVDIRI (1U << 27) #define CPUID_7_0_ECX_MOVDIRI (1U << 27)
/* Move 64 Bytes as Direct Store Instruction */ /* Move 64 Bytes as Direct Store Instruction */
#define CPUID_7_0_ECX_MOVDIR64B (1U << 28) #define CPUID_7_0_ECX_MOVDIR64B (1U << 28)
/* Protection Keys for Supervisor-mode Pages */
#define CPUID_7_0_ECX_PKS (1U << 31)
/* AVX512 Neural Network Instructions */ /* AVX512 Neural Network Instructions */
#define CPUID_7_0_EDX_AVX512_4VNNIW (1U << 2) #define CPUID_7_0_EDX_AVX512_4VNNIW (1U << 2)
@ -965,6 +973,7 @@ typedef uint64_t FeatureWordArray[FEATURE_WORDS];
#define VMX_VM_EXIT_CLEAR_BNDCFGS 0x00800000 #define VMX_VM_EXIT_CLEAR_BNDCFGS 0x00800000
#define VMX_VM_EXIT_PT_CONCEAL_PIP 0x01000000 #define VMX_VM_EXIT_PT_CONCEAL_PIP 0x01000000
#define VMX_VM_EXIT_CLEAR_IA32_RTIT_CTL 0x02000000 #define VMX_VM_EXIT_CLEAR_IA32_RTIT_CTL 0x02000000
#define VMX_VM_EXIT_LOAD_IA32_PKRS 0x20000000
#define VMX_VM_ENTRY_LOAD_DEBUG_CONTROLS 0x00000004 #define VMX_VM_ENTRY_LOAD_DEBUG_CONTROLS 0x00000004
#define VMX_VM_ENTRY_IA32E_MODE 0x00000200 #define VMX_VM_ENTRY_IA32E_MODE 0x00000200
@ -976,6 +985,7 @@ typedef uint64_t FeatureWordArray[FEATURE_WORDS];
#define VMX_VM_ENTRY_LOAD_BNDCFGS 0x00010000 #define VMX_VM_ENTRY_LOAD_BNDCFGS 0x00010000
#define VMX_VM_ENTRY_PT_CONCEAL_PIP 0x00020000 #define VMX_VM_ENTRY_PT_CONCEAL_PIP 0x00020000
#define VMX_VM_ENTRY_LOAD_IA32_RTIT_CTL 0x00040000 #define VMX_VM_ENTRY_LOAD_IA32_RTIT_CTL 0x00040000
#define VMX_VM_ENTRY_LOAD_IA32_PKRS 0x00400000
/* Supported Hyper-V Enlightenments */ /* Supported Hyper-V Enlightenments */
#define HYPERV_FEAT_RELAXED 0 #define HYPERV_FEAT_RELAXED 0
@ -1483,6 +1493,7 @@ typedef struct CPUX86State {
uint64_t msr_smi_count; uint64_t msr_smi_count;
uint32_t pkru; uint32_t pkru;
uint32_t pkrs;
uint32_t tsx_ctrl; uint32_t tsx_ctrl;
uint64_t spec_ctrl; uint64_t spec_ctrl;

View File

@ -194,6 +194,9 @@ void cpu_x86_update_cr4(CPUX86State *env, uint32_t new_cr4)
if (!(env->features[FEAT_7_0_ECX] & CPUID_7_0_ECX_PKU)) { if (!(env->features[FEAT_7_0_ECX] & CPUID_7_0_ECX_PKU)) {
new_cr4 &= ~CR4_PKE_MASK; new_cr4 &= ~CR4_PKE_MASK;
} }
if (!(env->features[FEAT_7_0_ECX] & CPUID_7_0_ECX_PKS)) {
new_cr4 &= ~CR4_PKS_MASK;
}
env->cr[4] = new_cr4; env->cr[4] = new_cr4;
env->hflags = hflags; env->hflags = hflags;

View File

@ -113,6 +113,7 @@ static bool has_msr_vmx_vmfunc;
static bool has_msr_ucode_rev; static bool has_msr_ucode_rev;
static bool has_msr_vmx_procbased_ctls2; static bool has_msr_vmx_procbased_ctls2;
static bool has_msr_perf_capabs; static bool has_msr_perf_capabs;
static bool has_msr_pkrs;
static uint32_t has_architectural_pmu_version; static uint32_t has_architectural_pmu_version;
static uint32_t num_architectural_pmu_gp_counters; static uint32_t num_architectural_pmu_gp_counters;
@ -2087,6 +2088,9 @@ static int kvm_get_supported_msrs(KVMState *s)
case MSR_IA32_VMX_PROCBASED_CTLS2: case MSR_IA32_VMX_PROCBASED_CTLS2:
has_msr_vmx_procbased_ctls2 = true; has_msr_vmx_procbased_ctls2 = true;
break; break;
case MSR_IA32_PKRS:
has_msr_pkrs = true;
break;
} }
} }
} }
@ -2814,6 +2818,9 @@ static int kvm_put_msrs(X86CPU *cpu, int level)
if (has_msr_smi_count) { if (has_msr_smi_count) {
kvm_msr_entry_add(cpu, MSR_SMI_COUNT, env->msr_smi_count); kvm_msr_entry_add(cpu, MSR_SMI_COUNT, env->msr_smi_count);
} }
if (has_msr_pkrs) {
kvm_msr_entry_add(cpu, MSR_IA32_PKRS, env->pkrs);
}
if (has_msr_bndcfgs) { if (has_msr_bndcfgs) {
kvm_msr_entry_add(cpu, MSR_IA32_BNDCFGS, env->msr_bndcfgs); kvm_msr_entry_add(cpu, MSR_IA32_BNDCFGS, env->msr_bndcfgs);
} }
@ -3205,6 +3212,9 @@ static int kvm_get_msrs(X86CPU *cpu)
if (has_msr_feature_control) { if (has_msr_feature_control) {
kvm_msr_entry_add(cpu, MSR_IA32_FEATURE_CONTROL, 0); kvm_msr_entry_add(cpu, MSR_IA32_FEATURE_CONTROL, 0);
} }
if (has_msr_pkrs) {
kvm_msr_entry_add(cpu, MSR_IA32_PKRS, 0);
}
if (has_msr_bndcfgs) { if (has_msr_bndcfgs) {
kvm_msr_entry_add(cpu, MSR_IA32_BNDCFGS, 0); kvm_msr_entry_add(cpu, MSR_IA32_BNDCFGS, 0);
} }
@ -3475,6 +3485,9 @@ static int kvm_get_msrs(X86CPU *cpu)
case MSR_IA32_UMWAIT_CONTROL: case MSR_IA32_UMWAIT_CONTROL:
env->umwait = msrs[i].data; env->umwait = msrs[i].data;
break; break;
case MSR_IA32_PKRS:
env->pkrs = msrs[i].data;
break;
default: default:
if (msrs[i].index >= MSR_MC0_CTL && if (msrs[i].index >= MSR_MC0_CTL &&
msrs[i].index < MSR_MC0_CTL + (env->mcg_cap & 0xff) * 4) { msrs[i].index < MSR_MC0_CTL + (env->mcg_cap & 0xff) * 4) {

View File

@ -980,7 +980,6 @@ static const VMStateDescription vmstate_umwait = {
} }
}; };
#ifdef TARGET_X86_64
static bool pkru_needed(void *opaque) static bool pkru_needed(void *opaque)
{ {
X86CPU *cpu = opaque; X86CPU *cpu = opaque;
@ -999,7 +998,25 @@ static const VMStateDescription vmstate_pkru = {
VMSTATE_END_OF_LIST() VMSTATE_END_OF_LIST()
} }
}; };
#endif
static bool pkrs_needed(void *opaque)
{
X86CPU *cpu = opaque;
CPUX86State *env = &cpu->env;
return env->pkrs != 0;
}
static const VMStateDescription vmstate_pkrs = {
.name = "cpu/pkrs",
.version_id = 1,
.minimum_version_id = 1,
.needed = pkrs_needed,
.fields = (VMStateField[]){
VMSTATE_UINT32(env.pkrs, X86CPU),
VMSTATE_END_OF_LIST()
}
};
static bool tsc_khz_needed(void *opaque) static bool tsc_khz_needed(void *opaque)
{ {
@ -1480,9 +1497,8 @@ VMStateDescription vmstate_x86_cpu = {
&vmstate_umwait, &vmstate_umwait,
&vmstate_tsc_khz, &vmstate_tsc_khz,
&vmstate_msr_smi_count, &vmstate_msr_smi_count,
#ifdef TARGET_X86_64
&vmstate_pkru, &vmstate_pkru,
#endif &vmstate_pkrs,
&vmstate_spec_ctrl, &vmstate_spec_ctrl,
&vmstate_mcg_ext_ctl, &vmstate_mcg_ext_ctl,
&vmstate_msr_intel_pt, &vmstate_msr_intel_pt,

View File

@ -361,6 +361,7 @@ static int handle_mmu_fault(CPUState *cs, vaddr addr, int size,
uint64_t rsvd_mask = PG_HI_RSVD_MASK; uint64_t rsvd_mask = PG_HI_RSVD_MASK;
uint32_t page_offset; uint32_t page_offset;
target_ulong vaddr; target_ulong vaddr;
uint32_t pkr;
is_user = mmu_idx == MMU_USER_IDX; is_user = mmu_idx == MMU_USER_IDX;
#if defined(DEBUG_MMU) #if defined(DEBUG_MMU)
@ -588,21 +589,28 @@ do_check_protect_pse36:
!((env->cr[4] & CR4_SMEP_MASK) && (ptep & PG_USER_MASK)))) { !((env->cr[4] & CR4_SMEP_MASK) && (ptep & PG_USER_MASK)))) {
prot |= PAGE_EXEC; prot |= PAGE_EXEC;
} }
if ((env->cr[4] & CR4_PKE_MASK) && (env->hflags & HF_LMA_MASK) &&
(ptep & PG_USER_MASK) && env->pkru) {
uint32_t pk = (pte & PG_PKRU_MASK) >> PG_PKRU_BIT;
uint32_t pkru_ad = (env->pkru >> pk * 2) & 1;
uint32_t pkru_wd = (env->pkru >> pk * 2) & 2;
uint32_t pkru_prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC;
if (pkru_ad) { if (!(env->hflags & HF_LMA_MASK)) {
pkru_prot &= ~(PAGE_READ | PAGE_WRITE); pkr = 0;
} else if (pkru_wd && (is_user || env->cr[0] & CR0_WP_MASK)) { } else if (ptep & PG_USER_MASK) {
pkru_prot &= ~PAGE_WRITE; pkr = env->cr[4] & CR4_PKE_MASK ? env->pkru : 0;
} else {
pkr = env->cr[4] & CR4_PKS_MASK ? env->pkrs : 0;
}
if (pkr) {
uint32_t pk = (pte & PG_PKRU_MASK) >> PG_PKRU_BIT;
uint32_t pkr_ad = (pkr >> pk * 2) & 1;
uint32_t pkr_wd = (pkr >> pk * 2) & 2;
uint32_t pkr_prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC;
if (pkr_ad) {
pkr_prot &= ~(PAGE_READ | PAGE_WRITE);
} else if (pkr_wd && (is_user || env->cr[0] & CR0_WP_MASK)) {
pkr_prot &= ~PAGE_WRITE;
} }
prot &= pkru_prot; prot &= pkr_prot;
if ((pkru_prot & (1 << is_write1)) == 0) { if ((pkr_prot & (1 << is_write1)) == 0) {
assert(is_write1 != 2); assert(is_write1 != 2);
error_code |= PG_ERROR_PK_MASK; error_code |= PG_ERROR_PK_MASK;
goto do_fault_protect; goto do_fault_protect;

View File

@ -244,6 +244,7 @@ void helper_rdmsr(CPUX86State *env)
void helper_wrmsr(CPUX86State *env) void helper_wrmsr(CPUX86State *env)
{ {
uint64_t val; uint64_t val;
CPUState *cs = env_cpu(env);
cpu_svm_check_intercept_param(env, SVM_EXIT_MSR, 1, GETPC()); cpu_svm_check_intercept_param(env, SVM_EXIT_MSR, 1, GETPC());
@ -296,6 +297,13 @@ void helper_wrmsr(CPUX86State *env)
case MSR_PAT: case MSR_PAT:
env->pat = val; env->pat = val;
break; break;
case MSR_IA32_PKRS:
if (val & 0xFFFFFFFF00000000ull) {
goto error;
}
env->pkrs = val;
tlb_flush(cs);
break;
case MSR_VM_HSAVE_PA: case MSR_VM_HSAVE_PA:
env->vm_hsave = val; env->vm_hsave = val;
break; break;
@ -399,6 +407,9 @@ void helper_wrmsr(CPUX86State *env)
/* XXX: exception? */ /* XXX: exception? */
break; break;
} }
return;
error:
raise_exception_err_ra(env, EXCP0D_GPF, 0, GETPC());
} }
void helper_rdmsr(CPUX86State *env) void helper_rdmsr(CPUX86State *env)
@ -430,6 +441,9 @@ void helper_rdmsr(CPUX86State *env)
case MSR_PAT: case MSR_PAT:
val = env->pat; val = env->pat;
break; break;
case MSR_IA32_PKRS:
val = env->pkrs;
break;
case MSR_VM_HSAVE_PA: case MSR_VM_HSAVE_PA:
val = env->vm_hsave; val = env->vm_hsave;
break; break;

View File

@ -3075,7 +3075,7 @@ static void gen_sse(CPUX86State *env, DisasContext *s, int b,
} }
if (is_xmm if (is_xmm
&& !(s->flags & HF_OSFXSR_MASK) && !(s->flags & HF_OSFXSR_MASK)
&& ((b != 0x38 && b != 0x3a) || (s->prefix & PREFIX_DATA))) { && (b != 0x38 && b != 0x3a)) {
goto unknown_op; goto unknown_op;
} }
if (b == 0x0e) { if (b == 0x0e) {

View File

@ -86,7 +86,6 @@ tests = {
'test-qobject-input-visitor': [testqapi], 'test-qobject-input-visitor': [testqapi],
'test-string-input-visitor': [testqapi], 'test-string-input-visitor': [testqapi],
'test-string-output-visitor': [testqapi], 'test-string-output-visitor': [testqapi],
'test-qmp-event': [testqapi],
'test-opts-visitor': [testqapi], 'test-opts-visitor': [testqapi],
'test-visitor-serialization': [testqapi], 'test-visitor-serialization': [testqapi],
'test-bitmap': [], 'test-bitmap': [],
@ -117,6 +116,12 @@ tests = {
'test-qapi-util': [], 'test-qapi-util': [],
} }
if have_system or have_tools
tests += {
'test-qmp-event': [testqapi],
}
endif
test_deps = { test_deps = {
'test-qht-par': qht_bench, 'test-qht-par': qht_bench,
} }
@ -276,7 +281,9 @@ test('decodetree', sh,
workdir: meson.current_source_dir() / 'decode', workdir: meson.current_source_dir() / 'decode',
suite: 'decodetree') suite: 'decodetree')
subdir('fp') if 'CONFIG_TCG' in config_all
subdir('fp')
endif
if not get_option('tcg').disabled() if not get_option('tcg').disabled()
if 'CONFIG_PLUGIN' in config_host if 'CONFIG_PLUGIN' in config_host

View File

@ -159,6 +159,8 @@ int LLVMFuzzerInitialize(int *argc, char ***argv, char ***envp)
char *target_name; char *target_name;
const char *bindir; const char *bindir;
char *datadir; char *datadir;
GString *cmd_line;
gchar *pretty_cmd_line;
bool serialize = false; bool serialize = false;
/* Initialize qgraph and modules */ /* Initialize qgraph and modules */
@ -217,7 +219,7 @@ int LLVMFuzzerInitialize(int *argc, char ***argv, char ***envp)
} }
/* Run QEMU's softmmu main with the fuzz-target dependent arguments */ /* Run QEMU's softmmu main with the fuzz-target dependent arguments */
GString *cmd_line = fuzz_target->get_init_cmdline(fuzz_target); cmd_line = fuzz_target->get_init_cmdline(fuzz_target);
g_string_append_printf(cmd_line, " %s -qtest /dev/null ", g_string_append_printf(cmd_line, " %s -qtest /dev/null ",
getenv("QTEST_LOG") ? "" : "-qtest-log none"); getenv("QTEST_LOG") ? "" : "-qtest-log none");
@ -226,6 +228,13 @@ int LLVMFuzzerInitialize(int *argc, char ***argv, char ***envp)
wordexp(cmd_line->str, &result, 0); wordexp(cmd_line->str, &result, 0);
g_string_free(cmd_line, true); g_string_free(cmd_line, true);
if (getenv("QTEST_LOG")) {
pretty_cmd_line = g_strjoinv(" ", result.we_wordv + 1);
printf("Starting %s with Arguments: %s\n",
result.we_wordv[0], pretty_cmd_line);
g_free(pretty_cmd_line);
}
qemu_init(result.we_wordc, result.we_wordv, NULL); qemu_init(result.we_wordc, result.we_wordv, NULL);
/* re-enable the rcu atfork, which was previously disabled in qemu_init */ /* re-enable the rcu atfork, which was previously disabled in qemu_init */

View File

@ -175,7 +175,7 @@ static int memory_access_size(MemoryRegion *mr, unsigned l, hwaddr addr)
* generic_fuzz(), avoiding potential race-conditions, which we don't have * generic_fuzz(), avoiding potential race-conditions, which we don't have
* a good way for reproducing right now. * a good way for reproducing right now.
*/ */
void fuzz_dma_read_cb(size_t addr, size_t len, MemoryRegion *mr, bool is_write) void fuzz_dma_read_cb(size_t addr, size_t len, MemoryRegion *mr)
{ {
/* Are we in the generic-fuzzer or are we using another fuzz-target? */ /* Are we in the generic-fuzzer or are we using another fuzz-target? */
if (!qts_global) { if (!qts_global) {
@ -187,14 +187,11 @@ void fuzz_dma_read_cb(size_t addr, size_t len, MemoryRegion *mr, bool is_write)
* - We have no DMA patterns defined * - We have no DMA patterns defined
* - The length of the DMA read request is zero * - The length of the DMA read request is zero
* - The DMA read is hitting an MR other than the machine's main RAM * - The DMA read is hitting an MR other than the machine's main RAM
* - The DMA request is not a read (what happens for a address_space_map
* with is_write=True? Can the device use the same pointer to do reads?)
* - The DMA request hits past the bounds of our RAM * - The DMA request hits past the bounds of our RAM
*/ */
if (dma_patterns->len == 0 if (dma_patterns->len == 0
|| len == 0 || len == 0
|| mr != current_machine->ram || mr != current_machine->ram
|| is_write
|| addr > current_machine->ram_size) { || addr > current_machine->ram_size) {
return; return;
} }
@ -213,12 +210,12 @@ void fuzz_dma_read_cb(size_t addr, size_t len, MemoryRegion *mr, bool is_write)
double_fetch = true; double_fetch = true;
if (addr < region.addr if (addr < region.addr
&& avoid_double_fetches) { && avoid_double_fetches) {
fuzz_dma_read_cb(addr, region.addr - addr, mr, is_write); fuzz_dma_read_cb(addr, region.addr - addr, mr);
} }
if (addr + len > region.addr + region.size if (addr + len > region.addr + region.size
&& avoid_double_fetches) { && avoid_double_fetches) {
fuzz_dma_read_cb(region.addr + region.size, fuzz_dma_read_cb(region.addr + region.size,
addr + len - (region.addr + region.size), mr, is_write); addr + len - (region.addr + region.size), mr);
} }
return; return;
} }
@ -936,12 +933,20 @@ static GString *generic_fuzz_cmdline(FuzzTarget *t)
static GString *generic_fuzz_predefined_config_cmdline(FuzzTarget *t) static GString *generic_fuzz_predefined_config_cmdline(FuzzTarget *t)
{ {
gchar *args;
const generic_fuzz_config *config; const generic_fuzz_config *config;
g_assert(t->opaque); g_assert(t->opaque);
config = t->opaque; config = t->opaque;
setenv("QEMU_AVOID_DOUBLE_FETCH", "1", 1); setenv("QEMU_AVOID_DOUBLE_FETCH", "1", 1);
if (config->argfunc) {
args = config->argfunc();
setenv("QEMU_FUZZ_ARGS", args, 1);
g_free(args);
} else {
g_assert_nonnull(config->args);
setenv("QEMU_FUZZ_ARGS", config->args, 1); setenv("QEMU_FUZZ_ARGS", config->args, 1);
}
setenv("QEMU_FUZZ_OBJECTS", config->objects, 1); setenv("QEMU_FUZZ_OBJECTS", config->objects, 1);
return generic_fuzz_cmdline(t); return generic_fuzz_cmdline(t);
} }

View File

@ -16,8 +16,19 @@
typedef struct generic_fuzz_config { typedef struct generic_fuzz_config {
const char *name, *args, *objects; const char *name, *args, *objects;
gchar* (*argfunc)(void); /* Result must be freeable by g_free() */
} generic_fuzz_config; } generic_fuzz_config;
static inline gchar *generic_fuzzer_virtio_9p_args(void){
char tmpdir[] = "/tmp/qemu-fuzz.XXXXXX";
g_assert_nonnull(mkdtemp(tmpdir));
return g_strdup_printf("-machine q35 -nodefaults "
"-device virtio-9p,fsdev=hshare,mount_tag=hshare "
"-fsdev local,id=hshare,path=%s,security_model=mapped-xattr,"
"writeout=immediate,fmode=0600,dmode=0700", tmpdir);
}
const generic_fuzz_config predefined_configs[] = { const generic_fuzz_config predefined_configs[] = {
{ {
.name = "virtio-net-pci-slirp", .name = "virtio-net-pci-slirp",
@ -59,6 +70,16 @@ const generic_fuzz_config predefined_configs[] = {
.name = "virtio-mouse", .name = "virtio-mouse",
.args = "-machine q35 -nodefaults -device virtio-mouse", .args = "-machine q35 -nodefaults -device virtio-mouse",
.objects = "virtio*", .objects = "virtio*",
},{
.name = "virtio-9p",
.argfunc = generic_fuzzer_virtio_9p_args,
.objects = "virtio*",
},{
.name = "virtio-9p-synth",
.args = "-machine q35 -nodefaults "
"-device virtio-9p,fsdev=hshare,mount_tag=hshare "
"-fsdev synth,id=hshare",
.objects = "virtio*",
},{ },{
.name = "e1000", .name = "e1000",
.args = "-M q35 -nodefaults " .args = "-M q35 -nodefaults "
@ -85,10 +106,28 @@ const generic_fuzz_config predefined_configs[] = {
.objects = "intel-hda", .objects = "intel-hda",
},{ },{
.name = "ide-hd", .name = "ide-hd",
.args = "-machine pc -nodefaults "
"-drive file=null-co://,if=none,format=raw,id=disk0 "
"-device ide-hd,drive=disk0",
.objects = "*ide*",
},{
.name = "ide-atapi",
.args = "-machine pc -nodefaults "
"-drive file=null-co://,if=none,format=raw,id=disk0 "
"-device ide-cd,drive=disk0",
.objects = "*ide*",
},{
.name = "ahci-hd",
.args = "-machine q35 -nodefaults " .args = "-machine q35 -nodefaults "
"-drive file=null-co://,if=none,format=raw,id=disk0 " "-drive file=null-co://,if=none,format=raw,id=disk0 "
"-device ide-hd,drive=disk0", "-device ide-hd,drive=disk0",
.objects = "ahci*", .objects = "*ahci*",
},{
.name = "ahci-atapi",
.args = "-machine q35 -nodefaults "
"-drive file=null-co://,if=none,format=raw,id=disk0 "
"-device ide-cd,drive=disk0",
.objects = "*ahci*",
},{ },{
.name = "floppy", .name = "floppy",
.args = "-machine pc -nodefaults -device floppy,id=floppy0 " .args = "-machine pc -nodefaults -device floppy,id=floppy0 "

View File

@ -29,6 +29,7 @@ void event_notifier_init_fd(EventNotifier *e, int fd)
{ {
e->rfd = fd; e->rfd = fd;
e->wfd = fd; e->wfd = fd;
e->initialized = true;
} }
#endif #endif
@ -68,6 +69,7 @@ int event_notifier_init(EventNotifier *e, int active)
if (active) { if (active) {
event_notifier_set(e); event_notifier_set(e);
} }
e->initialized = true;
return 0; return 0;
fail: fail:
@ -78,12 +80,18 @@ fail:
void event_notifier_cleanup(EventNotifier *e) void event_notifier_cleanup(EventNotifier *e)
{ {
if (!e->initialized) {
return;
}
if (e->rfd != e->wfd) { if (e->rfd != e->wfd) {
close(e->rfd); close(e->rfd);
} }
e->rfd = -1; e->rfd = -1;
close(e->wfd); close(e->wfd);
e->wfd = -1; e->wfd = -1;
e->initialized = false;
} }
int event_notifier_get_fd(const EventNotifier *e) int event_notifier_get_fd(const EventNotifier *e)
@ -96,6 +104,10 @@ int event_notifier_set(EventNotifier *e)
static const uint64_t value = 1; static const uint64_t value = 1;
ssize_t ret; ssize_t ret;
if (!e->initialized) {
return -1;
}
do { do {
ret = write(e->wfd, &value, sizeof(value)); ret = write(e->wfd, &value, sizeof(value));
} while (ret < 0 && errno == EINTR); } while (ret < 0 && errno == EINTR);
@ -113,6 +125,10 @@ int event_notifier_test_and_clear(EventNotifier *e)
ssize_t len; ssize_t len;
char buffer[512]; char buffer[512];
if (!e->initialized) {
return 0;
}
/* Drain the notify pipe. For eventfd, only 8 bytes will be read. */ /* Drain the notify pipe. For eventfd, only 8 bytes will be read. */
value = 0; value = 0;
do { do {