diff options
author | Paolo Bonzini <pbonzini@redhat.com> | 2018-02-16 10:05:23 +0100 |
---|---|---|
committer | Paolo Bonzini <pbonzini@redhat.com> | 2018-03-12 16:12:47 +0100 |
commit | a40161cbe9ccbcbab798c3e4d257c4bba99d153a (patch) | |
tree | fe6cc9bb0a5a3253a59d0187a5266573a2b81e16 /util/sys_membarrier.c | |
parent | c8d3877e48c4f57381d72eaf8d016bff12ce2d7c (diff) | |
download | qemu-a40161cbe9ccbcbab798c3e4d257c4bba99d153a.zip qemu-a40161cbe9ccbcbab798c3e4d257c4bba99d153a.tar.gz qemu-a40161cbe9ccbcbab798c3e4d257c4bba99d153a.tar.bz2 |
membarrier: add --enable-membarrier
Actually enable the global memory barriers if supported by the OS.
Because only recent versions of Linux include the support, they
are disabled by default. Note that it also has to be disabled
for QEMU to run under Wine.
Before this patch, rcutorture reports 85 ns/read for my machine,
after the patch it reports 12.5 ns/read. On the other hand updates
go from 50 *micro*seconds to 20 *milli*seconds.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Diffstat (limited to 'util/sys_membarrier.c')
-rw-r--r-- | util/sys_membarrier.c | 50 |
1 files changed, 50 insertions, 0 deletions
diff --git a/util/sys_membarrier.c b/util/sys_membarrier.c new file mode 100644 index 0000000..8dcb53e --- /dev/null +++ b/util/sys_membarrier.c @@ -0,0 +1,50 @@ +/* + * Process-global memory barriers + * + * Copyright (c) 2018 Red Hat, Inc. + * + * Author: Paolo Bonzini <pbonzini@redhat.com> + */ + +#include <qemu/osdep.h> +#include <qemu/sys_membarrier.h> +#include <qemu/error-report.h> + +#ifdef CONFIG_LINUX +#include <linux/membarrier.h> +#include <sys/syscall.h> + +static int +membarrier(int cmd, int flags) +{ + return syscall(__NR_membarrier, cmd, flags); +} +#endif + +void smp_mb_global(void) +{ +#if defined CONFIG_WIN32 + FlushProcessWriteBuffers(); +#elif defined CONFIG_LINUX + membarrier(MEMBARRIER_CMD_SHARED, 0); +#else +#error --enable-membarrier is not supported on this operating system. +#endif +} + +void smp_mb_global_init(void) +{ +#ifdef CONFIG_LINUX + int ret = membarrier(MEMBARRIER_CMD_QUERY, 0); + if (ret < 0) { + error_report("This QEMU binary requires the membarrier system call."); + error_report("Please upgrade your system to a newer version of Linux"); + exit(1); + } + if (!(ret & MEMBARRIER_CMD_SHARED)) { + error_report("This QEMU binary requires MEMBARRIER_CMD_SHARED support."); + error_report("Please upgrade your system to a newer version of Linux"); + exit(1); + } +#endif +} |