----- On Nov 14, 2017, at 9:53 AM, Avi Kivity avi@xxxxxxxxxxxx wrote:
On 11/13/2017 06:56 PM, Mathieu Desnoyers wrote:Very good points! Perhaps those guarantees should be documented somewhere ?
----- On Nov 10, 2017, at 4:57 PM, Mathieu DesnoyersI think that's safe. The kernel has to execute a MOV CR3 instruction
mathieu.desnoyers@xxxxxxxxxxxx wrote:
----- On Nov 10, 2017, at 4:36 PM, Linus Torvalds torvalds@xxxxxxxxxxxxxxxxxxxxI figured out what you're pointing to: if exec() is executed by a previously
wrote:
On Fri, Nov 10, 2017 at 1:12 PM, Mathieu DesnoyersThat core serializing instruction is not that much about I$ vs D$
<mathieu.desnoyers@xxxxxxxxxxxx> wrote:
x86 can return to user-space through sysexit and sysretq, which are notWe should check with Intel. I would actually be surprised if the I$
core serializing. This breaks expectations from user-space about
sequential consistency from a single-threaded self-modifying program
point of view in specific migration patterns.
Feedback is welcome,
can be out of sync with the D$ after a sysretq. It would actually
break things like "read code from disk" too in theory.
consistency, but rather about the processor speculatively executing code
ahead of its retirement point. Ref. Intel Architecture Software Developer's
Manual, Volume 3: System Programming.
7.1.3. "Handling Self- and Cross-Modifying Code":
"The act of a processor writing data into a currently executing code segment
with the intent of
executing that data as code is called self-modifying code. Intel Architecture
processors exhibit
model-specific behavior when executing self-modified code, depending upon how
far ahead of
the current execution pointer the code has been modified. As processor
architectures become
more complex and start to speculatively execute code ahead of the retirement
point (as in the P6
family processors), the rules regarding which code should execute, pre- or
post-modification,
become blurred. [...]"
AFAIU, this core serializing instruction seems to be needed for use-cases of
self-modifying code, but not for the initial load of a program from disk,
as the processor has no way to have speculatively executed any of its
instructions.
running thread, and there is no core serializing instruction between program
load and return to user-space, the kernel ends up acting like a JIT, indeed.
before it can execute code loaded by exec, and that is a serializing
instruction. Loading and unloading shared libraries is made safe by the
IRET executed by page faults (loading) and TLB shootdown IPIs (unloading).
Directly modifying code in userspace is unsafe if there is someAndroid guys at Google have reproducers of context synchronization issues
non-coherent instruction cache. Instruction fetch and speculative
execution are non-coherent, but they're probably too short (in current
processors) to matter. Trace caches are probably large enough, but I
don't know whether they are coherent or not.
on arm 64 in JIT scenarios. Based on the information I got, flushing the
instruction caches is not enough: they also need to issue a context
synchronizing instruction.
Perhaps the current Intel processors may have short enough speculative
execution and small enough trace caches, but relying on this without
a clear statement from Intel seems fragile.
I've tried to create a small single-threaded self-modifying loop in
user-space to trigger a trace cache or speculative execution quirk,
but I have not succeeded yet. I suspect that I would need to know
more about the internals of the processor architecture to create the
right stalls that would allow speculative execution to move further
ahead, and trigger an incoherent execution flow. Ideas on how to
trigger this would be welcome.