On Thu, Jun 25, 2020 at 10:59:31AM +0200, Borislav Petkov wrote: > On Thu, Jun 18, 2020 at 01:08:25AM +0300, Jarkko Sakkinen wrote: > > diff --git a/arch/x86/mm/fault.c b/arch/x86/mm/fault.c > > index 66be9bd60307..25d48aae36c1 100644 > > --- a/arch/x86/mm/fault.c > > +++ b/arch/x86/mm/fault.c > > @@ -1055,6 +1055,19 @@ access_error(unsigned long error_code, struct vm_area_struct *vma) > > if (error_code & X86_PF_PK) > > return 1; > > > > + /* > > + * Access is blocked by the Enclave Page Cache Map (EPCM), i.e. the > > + * access is allowed by the PTE but not the EPCM. This usually happens > > + * when the EPCM is yanked out from under us, e.g. by hardware after a > > + * suspend/resume cycle. In any case, software, i.e. the kernel, can't > > + * fix the source of the fault as the EPCM can't be directly modified by > > + * software. Handle the fault as an access error in order to signal > > + * userspace so that userspace can rebuild their enclave(s), even though > > + * userspace may not have actually violated access permissions. > > + */ > > Lemme check whether I understand this correctly: userspace must check > whether the SIGSEGV is generated on an access to an enclave page? Sort of. Technically it's that's an accurate statement, but practically speaking userspace can only access enclave pages when it is executing in the enclave, and exceptions in enclaves have unique behavior. Exceptions in enclaves essentially bounce through a userspace-software-defined location prior to being delivered to the kernel. The trampoline is done by the CPU so that the CPU can scrub the GPRs, XSAVE state, etc... and hide the true RIP of the exception. The pre-exception enclave state is saved into protected memory and restored when userspace resumes the enclave. Enterring or resuming an enclave can only be done through dedicted ENCLU instructions, so really it ends up being that the SIGSEGV handler needs to check the IP that "caused" the fault, which is actually the IP of the trampoline. But, that's only the first half of the story... > Also, do I see it correctly that when this happens, dmesg will have > > printk("%s%s[%d]: segfault at %lx ip %px sp %px error %lx", > > due to: > > if (likely(show_unhandled_signals)) > show_signal_msg(regs, error_code, address, tsk); > > which does: > > if (!unhandled_signal(tsk, SIGSEGV)) > return; > > or is the task expected to register a SIGSEGV handler so that the > segfault doesn't land in dmesg? Yes, without extra help, any task running an enclave is expected to register a SIGSEGV handler so that the task can restart the enclave if the EPC is "lost". However, building and running enclaves is complex, and the vast majority of SGX enabled applications are expected to leverage a library of one kind or another to hand the bulk of the gory details. But, signal handling in libraries is a mess, e.g. requires filtering/forwarding, resignaling, etc... To that end, in v14 of this patch[1], Andy Lutomirski came up with the idea of adding a vDSO function to provide the low level enclave EENTER/ERESUME and trampoline, and then teaching the kernel to do exception fixup on the relevant instructions in the vDSO. The vDSO's exception fixup then returns to normal userspace, with a (technically optional) struct holding the details of the exception. That allows for synchronous delivery of exceptions in enclaves, obviates the need for userspace to regsiter a SIGSEGV handler, and also means the SIGSEGV will never show up in dmesg so long as userspace is using the vDSO. The kernel still supports direct EENTER/ERESUME, but AFAIK everyone is moving (or has moved) to the vDSO interface. The vDSO stuff is in patches 15-18 of this series. There's a gigantic thread on all the alternatives that were considered[2]. [1] https://lkml.kernel.org/r/CALCETrXByb2UVuZ6AXUeOd8y90NAikbZuvdN3wf_TjHZ+CxNhA@xxxxxxxxxxxxxx [2] https://lkml.kernel.org/r/CALCETrWdpoDkbZjkucKL91GWpDPG9p=VqYrULade2pFDR7S=GQ@xxxxxxxxxxxxxx > > If so, are we documenting this? > > If not, then we should not issue any "segfault" messages to dmesg > because that would be wrong. > > Or maybe I'm not seeing it right but I don't have the hardware to test > this out... > > Thx. > > -- > Regards/Gruss, > Boris. > > https://people.kernel.org/tglx/notes-about-netiquette