Occlum now can use PKU, a hardware feature, to enforce isolation between LibOS and (LibOS's)userspace applications. Here is its manual.
Note: Userspace applications in this manual always refers to the LibOS's userpsace applications.
PKU (Protection Keys for Userspace) (aka. MPK: Memory Protection Keys) has been introduced into Linux since 2015 (details can be found in the lwn page). It is a lightweight intra-process isolation mechanism for userspace (Ring 3) software. Since its memory access policy is restricted by MMU (Memory Management Unit), it incurs almost non-zero overhead at runtime compared to Software Fault Isolation (SFI), and the memory access permission switch overhead is low. More details can be found in the manual page.
Currently, Occlum lacks the ability to isolate LibOS from userspace applications. Though userspace applications are considered benign in Occlum, but it is bug-prone inevitably. Potential illegal memory accesses may affect the correctness of computation, even lead to the crash of the whole enclave. Intra-enclave isolation is helpful for developers to uncover bugs beforehand.
It necessary to enforce the isolation in Occlum, and leveraging PKU is a good choice.
Security Concerns: PKU is only an option for users to enhance Occlum's security. Userspace applications are still in TCB (Trusted Computing Base). It is should be emphasized that OS has the full control of enclave's page table and control registers (e.g. CR4), and OS has mainly two ways to enforce data access policy related to PKU:
- OS is able to set CR4.PKE to 0, rendering intra-enclave isolation useless;
- OS is able to misconfigure pkeys in PTE.
So OS is able to perform attacks easily by misconfiguring PTEs or CR4. However, such configurations can only lead to DoS (Denial-of-Service) attcks, and DoS is not considered in SGX's threat model. If users worry that PKU feature in Occlum opens a new attack vector, they can turn off PKU feature in production environment. Users can still turn on PKU feature in test environment before releasing their APPs.
-
The CPU supports PKU feature, which can checked by:
cat /proc/cpuinfo | grep pku
-
OS has set CR4.PKE to 1, which can be checked by:
cat /proc/cpuinfo | grep ospke
-
Occlum relies on syscalls to configure PKU:
pkey_alloc()
pkey_mprotect()
pkey_free()
Occlum always runs in docker environment. Docker uses secure computing mode (
seccomp
) provided by Linux to specify the available within the container (details). Currently, the defaultseccomp
profile has not added PKU related syscall into its white list (we have reported such issue). To work around it, developers needs to add--privileged
flag indocker run
command, or use our customized profile by adding--security-opt
flag.To check whether PKU related syscalls are allowed in container environment, users can run sample code in Occlum's container environment:
#define _GNU_SOURCE #include <stdlib.h> #include <stdio.h> #include <sys/mman.h> #define BUF_SIZE (0x1000) int main() { int ret = -1, pkey = -1; ret = pkey_alloc(0, 0); if (ret < 0) { perror("Cannot invoke pkey_alloc() successfully"); exit(-1); } else { pkey = ret; char *buf = (char *)mmap(NULL, BUF_SIZE, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANON, -1, 0); if (buf == MAP_FAILED) { perror("Cannot invoke mmap() successfully"); exit(-1); } ret = pkey_mprotect(buf, BUF_SIZE, PROT_READ | PROT_WRITE, pkey); if (ret < 0) { perror("Cannot invoke pkey_mprotect() successfully"); exit(-1); } ret = munmap(buf, BUF_SIZE); if (ret < 0) { perror("Cannot invoke munmap() successfully"); exit(-1); } ret = pkey_free(pkey); if (ret < 0) { perror("Cannot invoke pkey_free() successfully"); exit(-1); } } printf("PKU related syscalls are allowed in container environment!\n"); return 0; }
The following output:
PKU related syscalls are allowed in container environment!
indicates PKU related syscalls are allowed in container environment.
PKU feature can only be enabled in HW
mode. Whether to turn on PKU feature is controlled by Occlum.json
.
{
// ...
"metadata": {
// "pkru" = 0: PKU feature must be disabled
// "pkru" = 1: PKU feature must be enabled
// "pkru" = 2: PKU feature is enabled if the platform supports it
"pkru": 0
}
// ...
}
Users have three options for PKU feature:
-
If
metadata.pkru
== 0: The PKU feature must disabled in Occlum. It is the default value. Since docker has not supported PKU related syscalls in container environment by default, Occlum turns off PKU feature in default configuration. -
If
metadata.pkru
== 1: The PKU feature must enabled in Occlum. If CPU on the platform does not support PKU, then enclave cannot be successfully initialized. -
If
metadata.pkru
== 2: If CPU supports PKU, also OS enables PKU feature, PKU feature is turned on in Occlum.