Services that create SGX enclaves depend on the Intel SGX SDK. This must be installed in the build environment, as well as the runtime environment.
An easy way to get this environment is to build in the docker image that we use for CI.
The dockerfile for this image lives in docker/Dockerfile
.
You can use ./mob prompt
to pull this image, (or to build it locally), and get a prompt
in this environment.
# From the root of the repo
./mob prompt
# At the container prompt
cargo build
If you have SGX-enabled hardware (activated in BIOS, and with SGX kernel module installed),
you can use ./mob prompt --hw
to get SGX in the container. Then you can both build and
run the tests in SGX_MODE=HW
. (See below for an explanation.)
A docker-less build also works fine for development:
- Set up your environment like the Dockerfile
There are two project-wide SGX-related configuration variables SGX_MODE
and IAS_MODE
.
These are set by environment variables, and they must be the same for all artifacts,
even those that don't depend directly on SGX. E.g. mobilecoind
must have the same configuration
as view_service
for Intel Remote Attestation to work, otherwise an error will occur at runtime.
For testing, you should usually use SGX_MODE=SW
and IAS_MODE=DEV
.
SGX_MODE=SW
means that the enclaves won't be "real" enclaves -- the service will link
to Intel-provided "_sim" versions of the Intel SGX SDK, and the enclave will be loaded approximately
like a shared library being dlopen
'ed. This means that you will be able to use gdb
and get
backtraces normally through the enclave code. In this mode, the CPU does not securely compute
measurements of the enclave, and attestation doesn't prove the integrity of the enclave.
SGX_MODE=HW
means that the real Intel libraries are used, and the enclave is loaded securely.
This mode is required for Intel Remote Attestation to work and provide security.
The clients and servers must all agree about this setting, or attestation will fail.
IAS_MODE=DEV
means that we will hit the Intel-provided "dev endpoints" during remote attestation.
These won't require the real production signing key in connection to the MRENCLAVE measurements.
IAS_MODE=PROD
means that we will hit the real Intel-provided production endpoints for remote attestation.
In code, this discrepancy is largely handled by the attest-net
crate.
The clients and servers must all agree about this setting, or attestation will fail.
cargo
supports crate-level features, and feature unification across the build plan.
cargo
does not support any notion of "global project-wide configuration".
In practice, it's too hard to invoke cargo to get all the features enabled exactly correctly on
all the right crates, if every crate has an sgx_mode
and ias_mode
feature.
Even if cargo had workspace-level features, which it doesn't, that wouldn't be good enough for us
because our build requires using multiple workspaces. We must keep the cargo features on some
targets separated and not unified.
Unifying cargo features across enclave targets and server targets will break the enclave builds.
This is because the enclave builds in a special no_std
environment.
Making SGX_MODE
and IAS_MODE
environment variables, and making build.rs
scripts that read
them and set features on these crates as needed, is the simplest way to make sure that there is
one source of truth for these values for all of the artifacts in the whole build.
The SGX_MODE
environment variable configuration is also used throughout Intel SGX SDK examples.
In order to run SGX securely with SGX_MODE=HW
you may need to change certain settings in your BIOS. These settings reduce the risk of certain side-channel attacks and are required
by the MobileCoin Fog network and clients to accept an enclave quote.
These may look different depending on your BIOS type.
Software Guard Extensions (SGX)
must be enabled. This is usually found under CPU Configuration
Hyperthreading
must be disabled. This is also usually found under CPU Configuration
Integrated Graphics
must be disabled. This is usually found under Display
For technical reasons, the ingest_enclave
, view_enclave
, and ledger_enclave
must be in a separate workspace.
They are also built using cargo build
.
The enclave build is invoked automatically if needed from the *_service
build.rs
.
To reproducibly build the enclave, (get exactly the right MRENCLAVE value), you must build in the container.
For local testing, you don't need to get exactly the right MRENCLAVE value. You can set up test networks with whatever MRENCLAVE your build produces, and clients that check this value using the Remote Attestation process.
If you want to download a prebuilt enclave, signed using the production signing key, in order to use IAS_MODE=PROD
and participate in a production-environment network, you will need to follow the instructions in Enclave Signing Material.
The enclave needs to be signed in order to run in production. The MobileCoin Foundation manages the key that signs the enclave which is used in the production MobileCoin Fog services. You can pull down the publicly available signature material in order to run the enclave that will attest with MobileCoin clients.
Building locally does not require providing a private key, as a random key will be generated during build.
There are two ways to use materials from a previously signed enclave to build your enclave locally.
The TestNet signature artifacts are available via
curl -O https://enclave-distribution.test.mobilecoin.com/production.json
This retrieves a json record of:
{
"consensus": {
"enclave": "pool/<git revision>/<signing hash>/<filename>",
"sigstruct": "pool/<git revision>/<signing hash>/<filename>"
},
"ingest": {
"enclave": "pool/<git revision>/<signing hash>/<filename>",
"sigstruct": "pool/<git revision>/<signing hash>/<filename>"
},
"ledger": {
"enclave": "pool/<git revision>/<signing hash>/<filename>",
"sigstruct": "pool/<git revision>/<signing hash>/<filename>"
},
"view": {
"enclave": "pool/<git revision>/<signing hash>/<filename>",
"sigstruct": "pool/<git revision>/<signing hash>/<filename>"
}
}
The git revision refers to the TestNet release version.
Once you have the desired artifact, you will need to extract either the signed enclave or the sigstruct file to build:
MobileCoin's TestNet Consensus Signed Enclave materials are available at, for example:
curl -O https://enclave-distribution.test.mobilecoin.com/pool/bceca6256b2ad9a6ccc1b88c109687365677f0c9/bf7fa957a6a94acb588851bc8767eca5776c79f4fc2aa6bcb99312c3c386c/libconsensus-enclave.signed.so
curl -O https://enclave-distribution.test.mobilecoin.com/pool/bceca6256b2ad9a6ccc1b88c109687365677f0c9/bf7fa957a6a94acb588851bc8767eca5776c79f4fc2aa6bcb99312c3c386c/consensus-enclave.css
Then, when you build, you will provide either CONSENSUS_ENCLAVE_SIGNED=$(pwd)/libconsensus-enclave.signed.so
or CONSENSUS_ENCLAVE_CSS=$(pwd)/consensus-enclave.css
.