424 lines
13 KiB
C
424 lines
13 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
/**
|
|
* Copyright(c) 2016-20 Intel Corporation.
|
|
*
|
|
* Intel Software Guard Extensions (SGX) support.
|
|
*/
|
|
#ifndef _ASM_X86_SGX_H
|
|
#define _ASM_X86_SGX_H
|
|
|
|
#include <linux/bits.h>
|
|
#include <linux/types.h>
|
|
|
|
/*
|
|
* This file contains both data structures defined by SGX architecture and Linux
|
|
* defined software data structures and functions. The two should not be mixed
|
|
* together for better readability. The architectural definitions come first.
|
|
*/
|
|
|
|
/* The SGX specific CPUID function. */
|
|
#define SGX_CPUID 0x12
|
|
/* EPC enumeration. */
|
|
#define SGX_CPUID_EPC 2
|
|
/* An invalid EPC section, i.e. the end marker. */
|
|
#define SGX_CPUID_EPC_INVALID 0x0
|
|
/* A valid EPC section. */
|
|
#define SGX_CPUID_EPC_SECTION 0x1
|
|
/* The bitmask for the EPC section type. */
|
|
#define SGX_CPUID_EPC_MASK GENMASK(3, 0)
|
|
|
|
enum sgx_encls_function {
|
|
ECREATE = 0x00,
|
|
EADD = 0x01,
|
|
EINIT = 0x02,
|
|
EREMOVE = 0x03,
|
|
EDGBRD = 0x04,
|
|
EDGBWR = 0x05,
|
|
EEXTEND = 0x06,
|
|
ELDU = 0x08,
|
|
EBLOCK = 0x09,
|
|
EPA = 0x0A,
|
|
EWB = 0x0B,
|
|
ETRACK = 0x0C,
|
|
EAUG = 0x0D,
|
|
EMODPR = 0x0E,
|
|
EMODT = 0x0F,
|
|
};
|
|
|
|
/**
|
|
* SGX_ENCLS_FAULT_FLAG - flag signifying an ENCLS return code is a trapnr
|
|
*
|
|
* ENCLS has its own (positive value) error codes and also generates
|
|
* ENCLS specific #GP and #PF faults. And the ENCLS values get munged
|
|
* with system error codes as everything percolates back up the stack.
|
|
* Unfortunately (for us), we need to precisely identify each unique
|
|
* error code, e.g. the action taken if EWB fails varies based on the
|
|
* type of fault and on the exact SGX error code, i.e. we can't simply
|
|
* convert all faults to -EFAULT.
|
|
*
|
|
* To make all three error types coexist, we set bit 30 to identify an
|
|
* ENCLS fault. Bit 31 (technically bits N:31) is used to differentiate
|
|
* between positive (faults and SGX error codes) and negative (system
|
|
* error codes) values.
|
|
*/
|
|
#define SGX_ENCLS_FAULT_FLAG 0x40000000
|
|
|
|
/**
|
|
* enum sgx_return_code - The return code type for ENCLS, ENCLU and ENCLV
|
|
* %SGX_EPC_PAGE_CONFLICT: Page is being written by other ENCLS function.
|
|
* %SGX_NOT_TRACKED: Previous ETRACK's shootdown sequence has not
|
|
* been completed yet.
|
|
* %SGX_CHILD_PRESENT SECS has child pages present in the EPC.
|
|
* %SGX_INVALID_EINITTOKEN: EINITTOKEN is invalid and enclave signer's
|
|
* public key does not match IA32_SGXLEPUBKEYHASH.
|
|
* %SGX_PAGE_NOT_MODIFIABLE: The EPC page cannot be modified because it
|
|
* is in the PENDING or MODIFIED state.
|
|
* %SGX_UNMASKED_EVENT: An unmasked event, e.g. INTR, was received
|
|
*/
|
|
enum sgx_return_code {
|
|
SGX_EPC_PAGE_CONFLICT = 7,
|
|
SGX_NOT_TRACKED = 11,
|
|
SGX_CHILD_PRESENT = 13,
|
|
SGX_INVALID_EINITTOKEN = 16,
|
|
SGX_PAGE_NOT_MODIFIABLE = 20,
|
|
SGX_UNMASKED_EVENT = 128,
|
|
};
|
|
|
|
/* The modulus size for 3072-bit RSA keys. */
|
|
#define SGX_MODULUS_SIZE 384
|
|
|
|
/**
|
|
* enum sgx_miscselect - additional information to an SSA frame
|
|
* %SGX_MISC_EXINFO: Report #PF or #GP to the SSA frame.
|
|
*
|
|
* Save State Area (SSA) is a stack inside the enclave used to store processor
|
|
* state when an exception or interrupt occurs. This enum defines additional
|
|
* information stored to an SSA frame.
|
|
*/
|
|
enum sgx_miscselect {
|
|
SGX_MISC_EXINFO = BIT(0),
|
|
};
|
|
|
|
#define SGX_MISC_RESERVED_MASK GENMASK_ULL(63, 1)
|
|
|
|
#define SGX_SSA_GPRS_SIZE 184
|
|
#define SGX_SSA_MISC_EXINFO_SIZE 16
|
|
|
|
/**
|
|
* enum sgx_attributes - the attributes field in &struct sgx_secs
|
|
* %SGX_ATTR_INIT: Enclave can be entered (is initialized).
|
|
* %SGX_ATTR_DEBUG: Allow ENCLS(EDBGRD) and ENCLS(EDBGWR).
|
|
* %SGX_ATTR_MODE64BIT: Tell that this a 64-bit enclave.
|
|
* %SGX_ATTR_PROVISIONKEY: Allow to use provisioning keys for remote
|
|
* attestation.
|
|
* %SGX_ATTR_KSS: Allow to use key separation and sharing (KSS).
|
|
* %SGX_ATTR_EINITTOKENKEY: Allow to use token signing key that is used to
|
|
* sign cryptographic tokens that can be passed to
|
|
* EINIT as an authorization to run an enclave.
|
|
* %SGX_ATTR_ASYNC_EXIT_NOTIFY: Allow enclaves to be notified after an
|
|
* asynchronous exit has occurred.
|
|
*/
|
|
enum sgx_attribute {
|
|
SGX_ATTR_INIT = BIT(0),
|
|
SGX_ATTR_DEBUG = BIT(1),
|
|
SGX_ATTR_MODE64BIT = BIT(2),
|
|
/* BIT(3) is reserved */
|
|
SGX_ATTR_PROVISIONKEY = BIT(4),
|
|
SGX_ATTR_EINITTOKENKEY = BIT(5),
|
|
/* BIT(6) is for CET */
|
|
SGX_ATTR_KSS = BIT(7),
|
|
/* BIT(8) is reserved */
|
|
/* BIT(9) is reserved */
|
|
SGX_ATTR_ASYNC_EXIT_NOTIFY = BIT(10),
|
|
};
|
|
|
|
#define SGX_ATTR_RESERVED_MASK (BIT_ULL(3) | \
|
|
BIT_ULL(6) | \
|
|
BIT_ULL(8) | \
|
|
BIT_ULL(9) | \
|
|
GENMASK_ULL(63, 11))
|
|
|
|
#define SGX_ATTR_UNPRIV_MASK (SGX_ATTR_DEBUG | \
|
|
SGX_ATTR_MODE64BIT | \
|
|
SGX_ATTR_KSS | \
|
|
SGX_ATTR_ASYNC_EXIT_NOTIFY)
|
|
|
|
#define SGX_ATTR_PRIV_MASK (SGX_ATTR_PROVISIONKEY | \
|
|
SGX_ATTR_EINITTOKENKEY)
|
|
|
|
/**
|
|
* struct sgx_secs - SGX Enclave Control Structure (SECS)
|
|
* @size: size of the address space
|
|
* @base: base address of the address space
|
|
* @ssa_frame_size: size of an SSA frame
|
|
* @miscselect: additional information stored to an SSA frame
|
|
* @attributes: attributes for enclave
|
|
* @xfrm: XSave-Feature Request Mask (subset of XCR0)
|
|
* @mrenclave: SHA256-hash of the enclave contents
|
|
* @mrsigner: SHA256-hash of the public key used to sign the SIGSTRUCT
|
|
* @config_id: a user-defined value that is used in key derivation
|
|
* @isv_prod_id: a user-defined value that is used in key derivation
|
|
* @isv_svn: a user-defined value that is used in key derivation
|
|
* @config_svn: a user-defined value that is used in key derivation
|
|
*
|
|
* SGX Enclave Control Structure (SECS) is a special enclave page that is not
|
|
* visible in the address space. In fact, this structure defines the address
|
|
* range and other global attributes for the enclave and it is the first EPC
|
|
* page created for any enclave. It is moved from a temporary buffer to an EPC
|
|
* by the means of ENCLS[ECREATE] function.
|
|
*/
|
|
struct sgx_secs {
|
|
u64 size;
|
|
u64 base;
|
|
u32 ssa_frame_size;
|
|
u32 miscselect;
|
|
u8 reserved1[24];
|
|
u64 attributes;
|
|
u64 xfrm;
|
|
u32 mrenclave[8];
|
|
u8 reserved2[32];
|
|
u32 mrsigner[8];
|
|
u8 reserved3[32];
|
|
u32 config_id[16];
|
|
u16 isv_prod_id;
|
|
u16 isv_svn;
|
|
u16 config_svn;
|
|
u8 reserved4[3834];
|
|
} __packed;
|
|
|
|
/**
|
|
* enum sgx_tcs_flags - execution flags for TCS
|
|
* %SGX_TCS_DBGOPTIN: If enabled allows single-stepping and breakpoints
|
|
* inside an enclave. It is cleared by EADD but can
|
|
* be set later with EDBGWR.
|
|
*/
|
|
enum sgx_tcs_flags {
|
|
SGX_TCS_DBGOPTIN = 0x01,
|
|
};
|
|
|
|
#define SGX_TCS_RESERVED_MASK GENMASK_ULL(63, 1)
|
|
#define SGX_TCS_RESERVED_SIZE 4024
|
|
|
|
/**
|
|
* struct sgx_tcs - Thread Control Structure (TCS)
|
|
* @state: used to mark an entered TCS
|
|
* @flags: execution flags (cleared by EADD)
|
|
* @ssa_offset: SSA stack offset relative to the enclave base
|
|
* @ssa_index: the current SSA frame index (cleard by EADD)
|
|
* @nr_ssa_frames: the number of frame in the SSA stack
|
|
* @entry_offset: entry point offset relative to the enclave base
|
|
* @exit_addr: address outside the enclave to exit on an exception or
|
|
* interrupt
|
|
* @fs_offset: offset relative to the enclave base to become FS
|
|
* segment inside the enclave
|
|
* @gs_offset: offset relative to the enclave base to become GS
|
|
* segment inside the enclave
|
|
* @fs_limit: size to become a new FS-limit (only 32-bit enclaves)
|
|
* @gs_limit: size to become a new GS-limit (only 32-bit enclaves)
|
|
*
|
|
* Thread Control Structure (TCS) is an enclave page visible in its address
|
|
* space that defines an entry point inside the enclave. A thread enters inside
|
|
* an enclave by supplying address of TCS to ENCLU(EENTER). A TCS can be entered
|
|
* by only one thread at a time.
|
|
*/
|
|
struct sgx_tcs {
|
|
u64 state;
|
|
u64 flags;
|
|
u64 ssa_offset;
|
|
u32 ssa_index;
|
|
u32 nr_ssa_frames;
|
|
u64 entry_offset;
|
|
u64 exit_addr;
|
|
u64 fs_offset;
|
|
u64 gs_offset;
|
|
u32 fs_limit;
|
|
u32 gs_limit;
|
|
u8 reserved[SGX_TCS_RESERVED_SIZE];
|
|
} __packed;
|
|
|
|
/**
|
|
* struct sgx_pageinfo - an enclave page descriptor
|
|
* @addr: address of the enclave page
|
|
* @contents: pointer to the page contents
|
|
* @metadata: pointer either to a SECINFO or PCMD instance
|
|
* @secs: address of the SECS page
|
|
*/
|
|
struct sgx_pageinfo {
|
|
u64 addr;
|
|
u64 contents;
|
|
u64 metadata;
|
|
u64 secs;
|
|
} __packed __aligned(32);
|
|
|
|
|
|
/**
|
|
* enum sgx_page_type - bits in the SECINFO flags defining the page type
|
|
* %SGX_PAGE_TYPE_SECS: a SECS page
|
|
* %SGX_PAGE_TYPE_TCS: a TCS page
|
|
* %SGX_PAGE_TYPE_REG: a regular page
|
|
* %SGX_PAGE_TYPE_VA: a VA page
|
|
* %SGX_PAGE_TYPE_TRIM: a page in trimmed state
|
|
*
|
|
* Make sure when making changes to this enum that its values can still fit
|
|
* in the bitfield within &struct sgx_encl_page
|
|
*/
|
|
enum sgx_page_type {
|
|
SGX_PAGE_TYPE_SECS,
|
|
SGX_PAGE_TYPE_TCS,
|
|
SGX_PAGE_TYPE_REG,
|
|
SGX_PAGE_TYPE_VA,
|
|
SGX_PAGE_TYPE_TRIM,
|
|
};
|
|
|
|
#define SGX_NR_PAGE_TYPES 5
|
|
#define SGX_PAGE_TYPE_MASK GENMASK(7, 0)
|
|
|
|
/**
|
|
* enum sgx_secinfo_flags - the flags field in &struct sgx_secinfo
|
|
* %SGX_SECINFO_R: allow read
|
|
* %SGX_SECINFO_W: allow write
|
|
* %SGX_SECINFO_X: allow execution
|
|
* %SGX_SECINFO_SECS: a SECS page
|
|
* %SGX_SECINFO_TCS: a TCS page
|
|
* %SGX_SECINFO_REG: a regular page
|
|
* %SGX_SECINFO_VA: a VA page
|
|
* %SGX_SECINFO_TRIM: a page in trimmed state
|
|
*/
|
|
enum sgx_secinfo_flags {
|
|
SGX_SECINFO_R = BIT(0),
|
|
SGX_SECINFO_W = BIT(1),
|
|
SGX_SECINFO_X = BIT(2),
|
|
SGX_SECINFO_SECS = (SGX_PAGE_TYPE_SECS << 8),
|
|
SGX_SECINFO_TCS = (SGX_PAGE_TYPE_TCS << 8),
|
|
SGX_SECINFO_REG = (SGX_PAGE_TYPE_REG << 8),
|
|
SGX_SECINFO_VA = (SGX_PAGE_TYPE_VA << 8),
|
|
SGX_SECINFO_TRIM = (SGX_PAGE_TYPE_TRIM << 8),
|
|
};
|
|
|
|
#define SGX_SECINFO_PERMISSION_MASK GENMASK_ULL(2, 0)
|
|
#define SGX_SECINFO_PAGE_TYPE_MASK (SGX_PAGE_TYPE_MASK << 8)
|
|
#define SGX_SECINFO_RESERVED_MASK ~(SGX_SECINFO_PERMISSION_MASK | \
|
|
SGX_SECINFO_PAGE_TYPE_MASK)
|
|
|
|
/**
|
|
* struct sgx_secinfo - describes attributes of an EPC page
|
|
* @flags: permissions and type
|
|
*
|
|
* Used together with ENCLS leaves that add or modify an EPC page to an
|
|
* enclave to define page permissions and type.
|
|
*/
|
|
struct sgx_secinfo {
|
|
u64 flags;
|
|
u8 reserved[56];
|
|
} __packed __aligned(64);
|
|
|
|
#define SGX_PCMD_RESERVED_SIZE 40
|
|
|
|
/**
|
|
* struct sgx_pcmd - Paging Crypto Metadata (PCMD)
|
|
* @enclave_id: enclave identifier
|
|
* @mac: MAC over PCMD, page contents and isvsvn
|
|
*
|
|
* PCMD is stored for every swapped page to the regular memory. When ELDU loads
|
|
* the page back it recalculates the MAC by using a isvsvn number stored in a
|
|
* VA page. Together these two structures bring integrity and rollback
|
|
* protection.
|
|
*/
|
|
struct sgx_pcmd {
|
|
struct sgx_secinfo secinfo;
|
|
u64 enclave_id;
|
|
u8 reserved[SGX_PCMD_RESERVED_SIZE];
|
|
u8 mac[16];
|
|
} __packed __aligned(128);
|
|
|
|
#define SGX_SIGSTRUCT_RESERVED1_SIZE 84
|
|
#define SGX_SIGSTRUCT_RESERVED2_SIZE 20
|
|
#define SGX_SIGSTRUCT_RESERVED3_SIZE 32
|
|
#define SGX_SIGSTRUCT_RESERVED4_SIZE 12
|
|
|
|
/**
|
|
* struct sgx_sigstruct_header - defines author of the enclave
|
|
* @header1: constant byte string
|
|
* @vendor: must be either 0x0000 or 0x8086
|
|
* @date: YYYYMMDD in BCD
|
|
* @header2: constant byte string
|
|
* @swdefined: software defined value
|
|
*/
|
|
struct sgx_sigstruct_header {
|
|
u64 header1[2];
|
|
u32 vendor;
|
|
u32 date;
|
|
u64 header2[2];
|
|
u32 swdefined;
|
|
u8 reserved1[84];
|
|
} __packed;
|
|
|
|
/**
|
|
* struct sgx_sigstruct_body - defines contents of the enclave
|
|
* @miscselect: additional information stored to an SSA frame
|
|
* @misc_mask: required miscselect in SECS
|
|
* @attributes: attributes for enclave
|
|
* @xfrm: XSave-Feature Request Mask (subset of XCR0)
|
|
* @attributes_mask: required attributes in SECS
|
|
* @xfrm_mask: required XFRM in SECS
|
|
* @mrenclave: SHA256-hash of the enclave contents
|
|
* @isvprodid: a user-defined value that is used in key derivation
|
|
* @isvsvn: a user-defined value that is used in key derivation
|
|
*/
|
|
struct sgx_sigstruct_body {
|
|
u32 miscselect;
|
|
u32 misc_mask;
|
|
u8 reserved2[20];
|
|
u64 attributes;
|
|
u64 xfrm;
|
|
u64 attributes_mask;
|
|
u64 xfrm_mask;
|
|
u8 mrenclave[32];
|
|
u8 reserved3[32];
|
|
u16 isvprodid;
|
|
u16 isvsvn;
|
|
} __packed;
|
|
|
|
/**
|
|
* struct sgx_sigstruct - an enclave signature
|
|
* @header: defines author of the enclave
|
|
* @modulus: the modulus of the public key
|
|
* @exponent: the exponent of the public key
|
|
* @signature: the signature calculated over the fields except modulus,
|
|
* @body: defines contents of the enclave
|
|
* @q1: a value used in RSA signature verification
|
|
* @q2: a value used in RSA signature verification
|
|
*
|
|
* Header and body are the parts that are actual signed. The remaining fields
|
|
* define the signature of the enclave.
|
|
*/
|
|
struct sgx_sigstruct {
|
|
struct sgx_sigstruct_header header;
|
|
u8 modulus[SGX_MODULUS_SIZE];
|
|
u32 exponent;
|
|
u8 signature[SGX_MODULUS_SIZE];
|
|
struct sgx_sigstruct_body body;
|
|
u8 reserved4[12];
|
|
u8 q1[SGX_MODULUS_SIZE];
|
|
u8 q2[SGX_MODULUS_SIZE];
|
|
} __packed;
|
|
|
|
#define SGX_LAUNCH_TOKEN_SIZE 304
|
|
|
|
/*
|
|
* Do not put any hardware-defined SGX structure representations below this
|
|
* comment!
|
|
*/
|
|
|
|
#ifdef CONFIG_X86_SGX_KVM
|
|
int sgx_virt_ecreate(struct sgx_pageinfo *pageinfo, void __user *secs,
|
|
int *trapnr);
|
|
int sgx_virt_einit(void __user *sigstruct, void __user *token,
|
|
void __user *secs, u64 *lepubkeyhash, int *trapnr);
|
|
#endif
|
|
|
|
int sgx_set_attribute(unsigned long *allowed_attributes,
|
|
unsigned int attribute_fd);
|
|
|
|
#endif /* _ASM_X86_SGX_H */
|