asm-generic: implement virt_xxx memory barriers
Guests running within virtual machines might be affected by SMP effects even if the guest itself is compiled without SMP support. This is an artifact of interfacing with an SMP host while running an UP kernel. Using mandatory barriers for this use-case would be possible but is often suboptimal. In particular, virtio uses a bunch of confusing ifdefs to work around this, while xen just uses the mandatory barriers. To better handle this case, low-level virt_mb() etc macros are made available. These are implemented trivially using the low-level __smp_xxx macros, the purpose of these wrappers is to annotate those specific cases. These have the same effect as smp_mb() etc when SMP is enabled, but generate identical code for SMP and non-SMP systems. For example, virtual machine guests should use virt_mb() rather than smp_mb() when synchronizing against a (possibly SMP) host. Suggested-by: David Miller <davem@davemloft.net> Signed-off-by: Michael S. Tsirkin <mst@redhat.com> Acked-by: Peter Zijlstra (Intel) <peterz@infradead.org>
This commit is contained in:
Родитель
1638fb7207
Коммит
6a65d26385
|
@ -1655,17 +1655,18 @@ macro is a good place to start looking.
|
||||||
SMP memory barriers are reduced to compiler barriers on uniprocessor compiled
|
SMP memory barriers are reduced to compiler barriers on uniprocessor compiled
|
||||||
systems because it is assumed that a CPU will appear to be self-consistent,
|
systems because it is assumed that a CPU will appear to be self-consistent,
|
||||||
and will order overlapping accesses correctly with respect to itself.
|
and will order overlapping accesses correctly with respect to itself.
|
||||||
|
However, see the subsection on "Virtual Machine Guests" below.
|
||||||
|
|
||||||
[!] Note that SMP memory barriers _must_ be used to control the ordering of
|
[!] Note that SMP memory barriers _must_ be used to control the ordering of
|
||||||
references to shared memory on SMP systems, though the use of locking instead
|
references to shared memory on SMP systems, though the use of locking instead
|
||||||
is sufficient.
|
is sufficient.
|
||||||
|
|
||||||
Mandatory barriers should not be used to control SMP effects, since mandatory
|
Mandatory barriers should not be used to control SMP effects, since mandatory
|
||||||
barriers unnecessarily impose overhead on UP systems. They may, however, be
|
barriers impose unnecessary overhead on both SMP and UP systems. They may,
|
||||||
used to control MMIO effects on accesses through relaxed memory I/O windows.
|
however, be used to control MMIO effects on accesses through relaxed memory I/O
|
||||||
These are required even on non-SMP systems as they affect the order in which
|
windows. These barriers are required even on non-SMP systems as they affect
|
||||||
memory operations appear to a device by prohibiting both the compiler and the
|
the order in which memory operations appear to a device by prohibiting both the
|
||||||
CPU from reordering them.
|
compiler and the CPU from reordering them.
|
||||||
|
|
||||||
|
|
||||||
There are some more advanced barrier functions:
|
There are some more advanced barrier functions:
|
||||||
|
@ -2948,6 +2949,23 @@ The Alpha defines the Linux kernel's memory barrier model.
|
||||||
|
|
||||||
See the subsection on "Cache Coherency" above.
|
See the subsection on "Cache Coherency" above.
|
||||||
|
|
||||||
|
VIRTUAL MACHINE GUESTS
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
Guests running within virtual machines might be affected by SMP effects even if
|
||||||
|
the guest itself is compiled without SMP support. This is an artifact of
|
||||||
|
interfacing with an SMP host while running an UP kernel. Using mandatory
|
||||||
|
barriers for this use-case would be possible but is often suboptimal.
|
||||||
|
|
||||||
|
To handle this case optimally, low-level virt_mb() etc macros are available.
|
||||||
|
These have the same effect as smp_mb() etc when SMP is enabled, but generate
|
||||||
|
identical code for SMP and non-SMP systems. For example, virtual machine guests
|
||||||
|
should use virt_mb() rather than smp_mb() when synchronizing against a
|
||||||
|
(possibly SMP) host.
|
||||||
|
|
||||||
|
These are equivalent to smp_mb() etc counterparts in all other respects,
|
||||||
|
in particular, they do not control MMIO effects: to control
|
||||||
|
MMIO effects, use mandatory barriers.
|
||||||
|
|
||||||
============
|
============
|
||||||
EXAMPLE USES
|
EXAMPLE USES
|
||||||
|
|
|
@ -196,5 +196,16 @@ do { \
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Barriers for virtual machine guests when talking to an SMP host */
|
||||||
|
#define virt_mb() __smp_mb()
|
||||||
|
#define virt_rmb() __smp_rmb()
|
||||||
|
#define virt_wmb() __smp_wmb()
|
||||||
|
#define virt_read_barrier_depends() __smp_read_barrier_depends()
|
||||||
|
#define virt_store_mb(var, value) __smp_store_mb(var, value)
|
||||||
|
#define virt_mb__before_atomic() __smp_mb__before_atomic()
|
||||||
|
#define virt_mb__after_atomic() __smp_mb__after_atomic()
|
||||||
|
#define virt_store_release(p, v) __smp_store_release(p, v)
|
||||||
|
#define virt_load_acquire(p) __smp_load_acquire(p)
|
||||||
|
|
||||||
#endif /* !__ASSEMBLY__ */
|
#endif /* !__ASSEMBLY__ */
|
||||||
#endif /* __ASM_GENERIC_BARRIER_H */
|
#endif /* __ASM_GENERIC_BARRIER_H */
|
||||||
|
|
Загрузка…
Ссылка в новой задаче