1acc4229cSAndra Paraschiv // SPDX-License-Identifier: GPL-2.0
2acc4229cSAndra Paraschiv /*
3059ebe4fSAndra Paraschiv  * Copyright 2020-2021 Amazon.com, Inc. or its affiliates. All Rights Reserved.
4acc4229cSAndra Paraschiv  */
5acc4229cSAndra Paraschiv 
6acc4229cSAndra Paraschiv /**
7acc4229cSAndra Paraschiv  * DOC: Sample flow of using the ioctl interface provided by the Nitro Enclaves (NE)
8acc4229cSAndra Paraschiv  * kernel driver.
9acc4229cSAndra Paraschiv  *
10acc4229cSAndra Paraschiv  * Usage
11acc4229cSAndra Paraschiv  * -----
12acc4229cSAndra Paraschiv  *
13acc4229cSAndra Paraschiv  * Load the nitro_enclaves module, setting also the enclave CPU pool. The
14acc4229cSAndra Paraschiv  * enclave CPUs need to be full cores from the same NUMA node. CPU 0 and its
15acc4229cSAndra Paraschiv  * siblings have to remain available for the primary / parent VM, so they
16acc4229cSAndra Paraschiv  * cannot be included in the enclave CPU pool.
17acc4229cSAndra Paraschiv  *
18acc4229cSAndra Paraschiv  * See the cpu list section from the kernel documentation.
19acc4229cSAndra Paraschiv  * https://www.kernel.org/doc/html/latest/admin-guide/kernel-parameters.html#cpu-lists
20acc4229cSAndra Paraschiv  *
21acc4229cSAndra Paraschiv  *	insmod drivers/virt/nitro_enclaves/nitro_enclaves.ko
22acc4229cSAndra Paraschiv  *	lsmod
23acc4229cSAndra Paraschiv  *
24acc4229cSAndra Paraschiv  *	The CPU pool can be set at runtime, after the kernel module is loaded.
25acc4229cSAndra Paraschiv  *
26acc4229cSAndra Paraschiv  *	echo <cpu-list> > /sys/module/nitro_enclaves/parameters/ne_cpus
27acc4229cSAndra Paraschiv  *
28acc4229cSAndra Paraschiv  *	NUMA and CPU siblings information can be found using:
29acc4229cSAndra Paraschiv  *
30acc4229cSAndra Paraschiv  *	lscpu
31acc4229cSAndra Paraschiv  *	/proc/cpuinfo
32acc4229cSAndra Paraschiv  *
33acc4229cSAndra Paraschiv  * Check the online / offline CPU list. The CPUs from the pool should be
34acc4229cSAndra Paraschiv  * offlined.
35acc4229cSAndra Paraschiv  *
36acc4229cSAndra Paraschiv  *	lscpu
37acc4229cSAndra Paraschiv  *
38acc4229cSAndra Paraschiv  * Check dmesg for any warnings / errors through the NE driver lifetime / usage.
39acc4229cSAndra Paraschiv  * The NE logs contain the "nitro_enclaves" or "pci 0000:00:02.0" pattern.
40acc4229cSAndra Paraschiv  *
41acc4229cSAndra Paraschiv  *	dmesg
42acc4229cSAndra Paraschiv  *
43acc4229cSAndra Paraschiv  * Setup hugetlbfs huge pages. The memory needs to be from the same NUMA node as
44acc4229cSAndra Paraschiv  * the enclave CPUs.
45acc4229cSAndra Paraschiv  *
46acc4229cSAndra Paraschiv  * https://www.kernel.org/doc/html/latest/admin-guide/mm/hugetlbpage.html
47acc4229cSAndra Paraschiv  *
48acc4229cSAndra Paraschiv  * By default, the allocation of hugetlb pages are distributed on all possible
49acc4229cSAndra Paraschiv  * NUMA nodes. Use the following configuration files to set the number of huge
50acc4229cSAndra Paraschiv  * pages from a NUMA node:
51acc4229cSAndra Paraschiv  *
52acc4229cSAndra Paraschiv  *	/sys/devices/system/node/node<X>/hugepages/hugepages-2048kB/nr_hugepages
53acc4229cSAndra Paraschiv  *	/sys/devices/system/node/node<X>/hugepages/hugepages-1048576kB/nr_hugepages
54acc4229cSAndra Paraschiv  *
55acc4229cSAndra Paraschiv  *	or, if not on a system with multiple NUMA nodes, can also set the number
56acc4229cSAndra Paraschiv  *	of 2 MiB / 1 GiB huge pages using
57acc4229cSAndra Paraschiv  *
58acc4229cSAndra Paraschiv  *	/sys/kernel/mm/hugepages/hugepages-2048kB/nr_hugepages
59acc4229cSAndra Paraschiv  *	/sys/kernel/mm/hugepages/hugepages-1048576kB/nr_hugepages
60acc4229cSAndra Paraschiv  *
61acc4229cSAndra Paraschiv  *	In this example 256 hugepages of 2 MiB are used.
62acc4229cSAndra Paraschiv  *
63acc4229cSAndra Paraschiv  * Build and run the NE sample.
64acc4229cSAndra Paraschiv  *
65acc4229cSAndra Paraschiv  *	make -C samples/nitro_enclaves clean
66acc4229cSAndra Paraschiv  *	make -C samples/nitro_enclaves
67acc4229cSAndra Paraschiv  *	./samples/nitro_enclaves/ne_ioctl_sample <path_to_enclave_image>
68acc4229cSAndra Paraschiv  *
69acc4229cSAndra Paraschiv  * Unload the nitro_enclaves module.
70acc4229cSAndra Paraschiv  *
71acc4229cSAndra Paraschiv  *	rmmod nitro_enclaves
72acc4229cSAndra Paraschiv  *	lsmod
73acc4229cSAndra Paraschiv  */
74acc4229cSAndra Paraschiv 
75acc4229cSAndra Paraschiv #include <stdio.h>
76acc4229cSAndra Paraschiv #include <stdlib.h>
77acc4229cSAndra Paraschiv #include <errno.h>
78acc4229cSAndra Paraschiv #include <fcntl.h>
79acc4229cSAndra Paraschiv #include <limits.h>
80acc4229cSAndra Paraschiv #include <poll.h>
81acc4229cSAndra Paraschiv #include <pthread.h>
82acc4229cSAndra Paraschiv #include <string.h>
83acc4229cSAndra Paraschiv #include <sys/eventfd.h>
84acc4229cSAndra Paraschiv #include <sys/ioctl.h>
85acc4229cSAndra Paraschiv #include <sys/mman.h>
86acc4229cSAndra Paraschiv #include <sys/socket.h>
87acc4229cSAndra Paraschiv #include <sys/stat.h>
88acc4229cSAndra Paraschiv #include <sys/types.h>
89acc4229cSAndra Paraschiv #include <unistd.h>
90acc4229cSAndra Paraschiv 
91acc4229cSAndra Paraschiv #include <linux/mman.h>
92acc4229cSAndra Paraschiv #include <linux/nitro_enclaves.h>
93acc4229cSAndra Paraschiv #include <linux/vm_sockets.h>
94acc4229cSAndra Paraschiv 
95acc4229cSAndra Paraschiv /**
96acc4229cSAndra Paraschiv  * NE_DEV_NAME - Nitro Enclaves (NE) misc device that provides the ioctl interface.
97acc4229cSAndra Paraschiv  */
98acc4229cSAndra Paraschiv #define NE_DEV_NAME			"/dev/nitro_enclaves"
99acc4229cSAndra Paraschiv 
100acc4229cSAndra Paraschiv /**
101acc4229cSAndra Paraschiv  * NE_POLL_WAIT_TIME - Timeout in seconds for each poll event.
102acc4229cSAndra Paraschiv  */
103acc4229cSAndra Paraschiv #define NE_POLL_WAIT_TIME		(60)
104acc4229cSAndra Paraschiv /**
105acc4229cSAndra Paraschiv  * NE_POLL_WAIT_TIME_MS - Timeout in milliseconds for each poll event.
106acc4229cSAndra Paraschiv  */
107acc4229cSAndra Paraschiv #define NE_POLL_WAIT_TIME_MS		(NE_POLL_WAIT_TIME * 1000)
108acc4229cSAndra Paraschiv 
109acc4229cSAndra Paraschiv /**
110acc4229cSAndra Paraschiv  * NE_SLEEP_TIME - Amount of time in seconds for the process to keep the enclave alive.
111acc4229cSAndra Paraschiv  */
112acc4229cSAndra Paraschiv #define NE_SLEEP_TIME			(300)
113acc4229cSAndra Paraschiv 
114acc4229cSAndra Paraschiv /**
115acc4229cSAndra Paraschiv  * NE_DEFAULT_NR_VCPUS - Default number of vCPUs set for an enclave.
116acc4229cSAndra Paraschiv  */
117acc4229cSAndra Paraschiv #define NE_DEFAULT_NR_VCPUS		(2)
118acc4229cSAndra Paraschiv 
119acc4229cSAndra Paraschiv /**
120acc4229cSAndra Paraschiv  * NE_MIN_MEM_REGION_SIZE - Minimum size of a memory region - 2 MiB.
121acc4229cSAndra Paraschiv  */
122acc4229cSAndra Paraschiv #define NE_MIN_MEM_REGION_SIZE		(2 * 1024 * 1024)
123acc4229cSAndra Paraschiv 
124acc4229cSAndra Paraschiv /**
125acc4229cSAndra Paraschiv  * NE_DEFAULT_NR_MEM_REGIONS - Default number of memory regions of 2 MiB set for
126acc4229cSAndra Paraschiv  *			       an enclave.
127acc4229cSAndra Paraschiv  */
128acc4229cSAndra Paraschiv #define NE_DEFAULT_NR_MEM_REGIONS	(256)
129acc4229cSAndra Paraschiv 
130acc4229cSAndra Paraschiv /**
131acc4229cSAndra Paraschiv  * NE_IMAGE_LOAD_HEARTBEAT_CID - Vsock CID for enclave image loading heartbeat logic.
132acc4229cSAndra Paraschiv  */
133acc4229cSAndra Paraschiv #define NE_IMAGE_LOAD_HEARTBEAT_CID	(3)
134acc4229cSAndra Paraschiv /**
135acc4229cSAndra Paraschiv  * NE_IMAGE_LOAD_HEARTBEAT_PORT - Vsock port for enclave image loading heartbeat logic.
136acc4229cSAndra Paraschiv  */
137acc4229cSAndra Paraschiv #define NE_IMAGE_LOAD_HEARTBEAT_PORT	(9000)
138acc4229cSAndra Paraschiv /**
139acc4229cSAndra Paraschiv  * NE_IMAGE_LOAD_HEARTBEAT_VALUE - Heartbeat value for enclave image loading.
140acc4229cSAndra Paraschiv  */
141acc4229cSAndra Paraschiv #define NE_IMAGE_LOAD_HEARTBEAT_VALUE	(0xb7)
142acc4229cSAndra Paraschiv 
143acc4229cSAndra Paraschiv /**
144acc4229cSAndra Paraschiv  * struct ne_user_mem_region - User space memory region set for an enclave.
145acc4229cSAndra Paraschiv  * @userspace_addr:	Address of the user space memory region.
146acc4229cSAndra Paraschiv  * @memory_size:	Size of the user space memory region.
147acc4229cSAndra Paraschiv  */
148acc4229cSAndra Paraschiv struct ne_user_mem_region {
149acc4229cSAndra Paraschiv 	void	*userspace_addr;
150acc4229cSAndra Paraschiv 	size_t	memory_size;
151acc4229cSAndra Paraschiv };
152acc4229cSAndra Paraschiv 
153acc4229cSAndra Paraschiv /**
154acc4229cSAndra Paraschiv  * ne_create_vm() - Create a slot for the enclave VM.
155acc4229cSAndra Paraschiv  * @ne_dev_fd:		The file descriptor of the NE misc device.
156acc4229cSAndra Paraschiv  * @slot_uid:		The generated slot uid for the enclave.
157acc4229cSAndra Paraschiv  * @enclave_fd :	The generated file descriptor for the enclave.
158acc4229cSAndra Paraschiv  *
159acc4229cSAndra Paraschiv  * Context: Process context.
160acc4229cSAndra Paraschiv  * Return:
161acc4229cSAndra Paraschiv  * * 0 on success.
162acc4229cSAndra Paraschiv  * * Negative return value on failure.
163acc4229cSAndra Paraschiv  */
ne_create_vm(int ne_dev_fd,unsigned long * slot_uid,int * enclave_fd)164acc4229cSAndra Paraschiv static int ne_create_vm(int ne_dev_fd, unsigned long *slot_uid, int *enclave_fd)
165acc4229cSAndra Paraschiv {
166acc4229cSAndra Paraschiv 	int rc = -EINVAL;
167acc4229cSAndra Paraschiv 	*enclave_fd = ioctl(ne_dev_fd, NE_CREATE_VM, slot_uid);
168acc4229cSAndra Paraschiv 
169acc4229cSAndra Paraschiv 	if (*enclave_fd < 0) {
170acc4229cSAndra Paraschiv 		rc = *enclave_fd;
171acc4229cSAndra Paraschiv 		switch (errno) {
172acc4229cSAndra Paraschiv 		case NE_ERR_NO_CPUS_AVAIL_IN_POOL: {
173acc4229cSAndra Paraschiv 			printf("Error in create VM, no CPUs available in the NE CPU pool\n");
174acc4229cSAndra Paraschiv 
175acc4229cSAndra Paraschiv 			break;
176acc4229cSAndra Paraschiv 		}
177acc4229cSAndra Paraschiv 
178acc4229cSAndra Paraschiv 		default:
179acc4229cSAndra Paraschiv 			printf("Error in create VM [%m]\n");
180acc4229cSAndra Paraschiv 		}
181acc4229cSAndra Paraschiv 
182acc4229cSAndra Paraschiv 		return rc;
183acc4229cSAndra Paraschiv 	}
184acc4229cSAndra Paraschiv 
185acc4229cSAndra Paraschiv 	return 0;
186acc4229cSAndra Paraschiv }
187acc4229cSAndra Paraschiv 
188acc4229cSAndra Paraschiv /**
189acc4229cSAndra Paraschiv  * ne_poll_enclave_fd() - Thread function for polling the enclave fd.
190acc4229cSAndra Paraschiv  * @data:	Argument provided for the polling function.
191acc4229cSAndra Paraschiv  *
192acc4229cSAndra Paraschiv  * Context: Process context.
193acc4229cSAndra Paraschiv  * Return:
194acc4229cSAndra Paraschiv  * * NULL on success / failure.
195acc4229cSAndra Paraschiv  */
ne_poll_enclave_fd(void * data)196acc4229cSAndra Paraschiv void *ne_poll_enclave_fd(void *data)
197acc4229cSAndra Paraschiv {
198acc4229cSAndra Paraschiv 	int enclave_fd = *(int *)data;
199acc4229cSAndra Paraschiv 	struct pollfd fds[1] = {};
200acc4229cSAndra Paraschiv 	int i = 0;
201acc4229cSAndra Paraschiv 	int rc = -EINVAL;
202acc4229cSAndra Paraschiv 
203acc4229cSAndra Paraschiv 	printf("Running from poll thread, enclave fd %d\n", enclave_fd);
204acc4229cSAndra Paraschiv 
205acc4229cSAndra Paraschiv 	fds[0].fd = enclave_fd;
206acc4229cSAndra Paraschiv 	fds[0].events = POLLIN | POLLERR | POLLHUP;
207acc4229cSAndra Paraschiv 
208acc4229cSAndra Paraschiv 	/* Keep on polling until the current process is terminated. */
209acc4229cSAndra Paraschiv 	while (1) {
210acc4229cSAndra Paraschiv 		printf("[iter %d] Polling ...\n", i);
211acc4229cSAndra Paraschiv 
212acc4229cSAndra Paraschiv 		rc = poll(fds, 1, NE_POLL_WAIT_TIME_MS);
213acc4229cSAndra Paraschiv 		if (rc < 0) {
214acc4229cSAndra Paraschiv 			printf("Error in poll [%m]\n");
215acc4229cSAndra Paraschiv 
216acc4229cSAndra Paraschiv 			return NULL;
217acc4229cSAndra Paraschiv 		}
218acc4229cSAndra Paraschiv 
219acc4229cSAndra Paraschiv 		i++;
220acc4229cSAndra Paraschiv 
221acc4229cSAndra Paraschiv 		if (!rc) {
222acc4229cSAndra Paraschiv 			printf("Poll: %d seconds elapsed\n",
223acc4229cSAndra Paraschiv 			       i * NE_POLL_WAIT_TIME);
224acc4229cSAndra Paraschiv 
225acc4229cSAndra Paraschiv 			continue;
226acc4229cSAndra Paraschiv 		}
227acc4229cSAndra Paraschiv 
228acc4229cSAndra Paraschiv 		printf("Poll received value 0x%x\n", fds[0].revents);
229acc4229cSAndra Paraschiv 
230acc4229cSAndra Paraschiv 		if (fds[0].revents & POLLHUP) {
231acc4229cSAndra Paraschiv 			printf("Received POLLHUP\n");
232acc4229cSAndra Paraschiv 
233acc4229cSAndra Paraschiv 			return NULL;
234acc4229cSAndra Paraschiv 		}
235acc4229cSAndra Paraschiv 
236acc4229cSAndra Paraschiv 		if (fds[0].revents & POLLNVAL) {
237acc4229cSAndra Paraschiv 			printf("Received POLLNVAL\n");
238acc4229cSAndra Paraschiv 
239acc4229cSAndra Paraschiv 			return NULL;
240acc4229cSAndra Paraschiv 		}
241acc4229cSAndra Paraschiv 	}
242acc4229cSAndra Paraschiv 
243acc4229cSAndra Paraschiv 	return NULL;
244acc4229cSAndra Paraschiv }
245acc4229cSAndra Paraschiv 
246acc4229cSAndra Paraschiv /**
247acc4229cSAndra Paraschiv  * ne_alloc_user_mem_region() - Allocate a user space memory region for an enclave.
248acc4229cSAndra Paraschiv  * @ne_user_mem_region:	User space memory region allocated using hugetlbfs.
249acc4229cSAndra Paraschiv  *
250acc4229cSAndra Paraschiv  * Context: Process context.
251acc4229cSAndra Paraschiv  * Return:
252acc4229cSAndra Paraschiv  * * 0 on success.
253acc4229cSAndra Paraschiv  * * Negative return value on failure.
254acc4229cSAndra Paraschiv  */
ne_alloc_user_mem_region(struct ne_user_mem_region * ne_user_mem_region)255acc4229cSAndra Paraschiv static int ne_alloc_user_mem_region(struct ne_user_mem_region *ne_user_mem_region)
256acc4229cSAndra Paraschiv {
257acc4229cSAndra Paraschiv 	/**
258acc4229cSAndra Paraschiv 	 * Check available hugetlb encodings for different huge page sizes in
259acc4229cSAndra Paraschiv 	 * include/uapi/linux/mman.h.
260acc4229cSAndra Paraschiv 	 */
261acc4229cSAndra Paraschiv 	ne_user_mem_region->userspace_addr = mmap(NULL, ne_user_mem_region->memory_size,
262acc4229cSAndra Paraschiv 						  PROT_READ | PROT_WRITE,
263acc4229cSAndra Paraschiv 						  MAP_PRIVATE | MAP_ANONYMOUS |
264acc4229cSAndra Paraschiv 						  MAP_HUGETLB | MAP_HUGE_2MB, -1, 0);
265acc4229cSAndra Paraschiv 	if (ne_user_mem_region->userspace_addr == MAP_FAILED) {
266acc4229cSAndra Paraschiv 		printf("Error in mmap memory [%m]\n");
267acc4229cSAndra Paraschiv 
268acc4229cSAndra Paraschiv 		return -1;
269acc4229cSAndra Paraschiv 	}
270acc4229cSAndra Paraschiv 
271acc4229cSAndra Paraschiv 	return 0;
272acc4229cSAndra Paraschiv }
273acc4229cSAndra Paraschiv 
274acc4229cSAndra Paraschiv /**
275acc4229cSAndra Paraschiv  * ne_load_enclave_image() - Place the enclave image in the enclave memory.
276acc4229cSAndra Paraschiv  * @enclave_fd :		The file descriptor associated with the enclave.
277acc4229cSAndra Paraschiv  * @ne_user_mem_regions:	User space memory regions allocated for the enclave.
278acc4229cSAndra Paraschiv  * @enclave_image_path :	The file path of the enclave image.
279acc4229cSAndra Paraschiv  *
280acc4229cSAndra Paraschiv  * Context: Process context.
281acc4229cSAndra Paraschiv  * Return:
282acc4229cSAndra Paraschiv  * * 0 on success.
283acc4229cSAndra Paraschiv  * * Negative return value on failure.
284acc4229cSAndra Paraschiv  */
ne_load_enclave_image(int enclave_fd,struct ne_user_mem_region ne_user_mem_regions[],char * enclave_image_path)285acc4229cSAndra Paraschiv static int ne_load_enclave_image(int enclave_fd, struct ne_user_mem_region ne_user_mem_regions[],
286acc4229cSAndra Paraschiv 				 char *enclave_image_path)
287acc4229cSAndra Paraschiv {
288acc4229cSAndra Paraschiv 	unsigned char *enclave_image = NULL;
289acc4229cSAndra Paraschiv 	int enclave_image_fd = -1;
290acc4229cSAndra Paraschiv 	size_t enclave_image_size = 0;
291acc4229cSAndra Paraschiv 	size_t enclave_memory_size = 0;
292acc4229cSAndra Paraschiv 	unsigned long i = 0;
293acc4229cSAndra Paraschiv 	size_t image_written_bytes = 0;
294acc4229cSAndra Paraschiv 	struct ne_image_load_info image_load_info = {
295acc4229cSAndra Paraschiv 		.flags = NE_EIF_IMAGE,
296acc4229cSAndra Paraschiv 	};
297acc4229cSAndra Paraschiv 	struct stat image_stat_buf = {};
298acc4229cSAndra Paraschiv 	int rc = -EINVAL;
299acc4229cSAndra Paraschiv 	size_t temp_image_offset = 0;
300acc4229cSAndra Paraschiv 
301acc4229cSAndra Paraschiv 	for (i = 0; i < NE_DEFAULT_NR_MEM_REGIONS; i++)
302acc4229cSAndra Paraschiv 		enclave_memory_size += ne_user_mem_regions[i].memory_size;
303acc4229cSAndra Paraschiv 
304acc4229cSAndra Paraschiv 	rc = stat(enclave_image_path, &image_stat_buf);
305acc4229cSAndra Paraschiv 	if (rc < 0) {
306acc4229cSAndra Paraschiv 		printf("Error in get image stat info [%m]\n");
307acc4229cSAndra Paraschiv 
308acc4229cSAndra Paraschiv 		return rc;
309acc4229cSAndra Paraschiv 	}
310acc4229cSAndra Paraschiv 
311acc4229cSAndra Paraschiv 	enclave_image_size = image_stat_buf.st_size;
312acc4229cSAndra Paraschiv 
313acc4229cSAndra Paraschiv 	if (enclave_memory_size < enclave_image_size) {
314acc4229cSAndra Paraschiv 		printf("The enclave memory is smaller than the enclave image size\n");
315acc4229cSAndra Paraschiv 
316acc4229cSAndra Paraschiv 		return -ENOMEM;
317acc4229cSAndra Paraschiv 	}
318acc4229cSAndra Paraschiv 
319acc4229cSAndra Paraschiv 	rc = ioctl(enclave_fd, NE_GET_IMAGE_LOAD_INFO, &image_load_info);
320acc4229cSAndra Paraschiv 	if (rc < 0) {
321acc4229cSAndra Paraschiv 		switch (errno) {
322acc4229cSAndra Paraschiv 		case NE_ERR_NOT_IN_INIT_STATE: {
323acc4229cSAndra Paraschiv 			printf("Error in get image load info, enclave not in init state\n");
324acc4229cSAndra Paraschiv 
325acc4229cSAndra Paraschiv 			break;
326acc4229cSAndra Paraschiv 		}
327acc4229cSAndra Paraschiv 
328acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_FLAG_VALUE: {
329acc4229cSAndra Paraschiv 			printf("Error in get image load info, provided invalid flag\n");
330acc4229cSAndra Paraschiv 
331acc4229cSAndra Paraschiv 			break;
332acc4229cSAndra Paraschiv 		}
333acc4229cSAndra Paraschiv 
334acc4229cSAndra Paraschiv 		default:
335acc4229cSAndra Paraschiv 			printf("Error in get image load info [%m]\n");
336acc4229cSAndra Paraschiv 		}
337acc4229cSAndra Paraschiv 
338acc4229cSAndra Paraschiv 		return rc;
339acc4229cSAndra Paraschiv 	}
340acc4229cSAndra Paraschiv 
341acc4229cSAndra Paraschiv 	printf("Enclave image offset in enclave memory is %lld\n",
342acc4229cSAndra Paraschiv 	       image_load_info.memory_offset);
343acc4229cSAndra Paraschiv 
344acc4229cSAndra Paraschiv 	enclave_image_fd = open(enclave_image_path, O_RDONLY);
345acc4229cSAndra Paraschiv 	if (enclave_image_fd < 0) {
346acc4229cSAndra Paraschiv 		printf("Error in open enclave image file [%m]\n");
347acc4229cSAndra Paraschiv 
348acc4229cSAndra Paraschiv 		return enclave_image_fd;
349acc4229cSAndra Paraschiv 	}
350acc4229cSAndra Paraschiv 
351acc4229cSAndra Paraschiv 	enclave_image = mmap(NULL, enclave_image_size, PROT_READ,
352acc4229cSAndra Paraschiv 			     MAP_PRIVATE, enclave_image_fd, 0);
353acc4229cSAndra Paraschiv 	if (enclave_image == MAP_FAILED) {
354acc4229cSAndra Paraschiv 		printf("Error in mmap enclave image [%m]\n");
355acc4229cSAndra Paraschiv 
356acc4229cSAndra Paraschiv 		return -1;
357acc4229cSAndra Paraschiv 	}
358acc4229cSAndra Paraschiv 
359acc4229cSAndra Paraschiv 	temp_image_offset = image_load_info.memory_offset;
360acc4229cSAndra Paraschiv 
361acc4229cSAndra Paraschiv 	for (i = 0; i < NE_DEFAULT_NR_MEM_REGIONS; i++) {
362acc4229cSAndra Paraschiv 		size_t bytes_to_write = 0;
363acc4229cSAndra Paraschiv 		size_t memory_offset = 0;
364acc4229cSAndra Paraschiv 		size_t memory_size = ne_user_mem_regions[i].memory_size;
365acc4229cSAndra Paraschiv 		size_t remaining_bytes = 0;
366acc4229cSAndra Paraschiv 		void *userspace_addr = ne_user_mem_regions[i].userspace_addr;
367acc4229cSAndra Paraschiv 
368acc4229cSAndra Paraschiv 		if (temp_image_offset >= memory_size) {
369acc4229cSAndra Paraschiv 			temp_image_offset -= memory_size;
370acc4229cSAndra Paraschiv 
371acc4229cSAndra Paraschiv 			continue;
372acc4229cSAndra Paraschiv 		} else if (temp_image_offset != 0) {
373acc4229cSAndra Paraschiv 			memory_offset = temp_image_offset;
374acc4229cSAndra Paraschiv 			memory_size -= temp_image_offset;
375acc4229cSAndra Paraschiv 			temp_image_offset = 0;
376acc4229cSAndra Paraschiv 		}
377acc4229cSAndra Paraschiv 
378acc4229cSAndra Paraschiv 		remaining_bytes = enclave_image_size - image_written_bytes;
379acc4229cSAndra Paraschiv 		bytes_to_write = memory_size < remaining_bytes ?
380acc4229cSAndra Paraschiv 				 memory_size : remaining_bytes;
381acc4229cSAndra Paraschiv 
382acc4229cSAndra Paraschiv 		memcpy(userspace_addr + memory_offset,
383acc4229cSAndra Paraschiv 		       enclave_image + image_written_bytes, bytes_to_write);
384acc4229cSAndra Paraschiv 
385acc4229cSAndra Paraschiv 		image_written_bytes += bytes_to_write;
386acc4229cSAndra Paraschiv 
387acc4229cSAndra Paraschiv 		if (image_written_bytes == enclave_image_size)
388acc4229cSAndra Paraschiv 			break;
389acc4229cSAndra Paraschiv 	}
390acc4229cSAndra Paraschiv 
391acc4229cSAndra Paraschiv 	munmap(enclave_image, enclave_image_size);
392acc4229cSAndra Paraschiv 
393acc4229cSAndra Paraschiv 	close(enclave_image_fd);
394acc4229cSAndra Paraschiv 
395acc4229cSAndra Paraschiv 	return 0;
396acc4229cSAndra Paraschiv }
397acc4229cSAndra Paraschiv 
398acc4229cSAndra Paraschiv /**
399acc4229cSAndra Paraschiv  * ne_set_user_mem_region() - Set a user space memory region for the given enclave.
400acc4229cSAndra Paraschiv  * @enclave_fd :		The file descriptor associated with the enclave.
401acc4229cSAndra Paraschiv  * @ne_user_mem_region :	User space memory region to be set for the enclave.
402acc4229cSAndra Paraschiv  *
403acc4229cSAndra Paraschiv  * Context: Process context.
404acc4229cSAndra Paraschiv  * Return:
405acc4229cSAndra Paraschiv  * * 0 on success.
406acc4229cSAndra Paraschiv  * * Negative return value on failure.
407acc4229cSAndra Paraschiv  */
ne_set_user_mem_region(int enclave_fd,struct ne_user_mem_region ne_user_mem_region)408acc4229cSAndra Paraschiv static int ne_set_user_mem_region(int enclave_fd, struct ne_user_mem_region ne_user_mem_region)
409acc4229cSAndra Paraschiv {
410acc4229cSAndra Paraschiv 	struct ne_user_memory_region mem_region = {
411acc4229cSAndra Paraschiv 		.flags = NE_DEFAULT_MEMORY_REGION,
412acc4229cSAndra Paraschiv 		.memory_size = ne_user_mem_region.memory_size,
413acc4229cSAndra Paraschiv 		.userspace_addr = (__u64)ne_user_mem_region.userspace_addr,
414acc4229cSAndra Paraschiv 	};
415acc4229cSAndra Paraschiv 	int rc = -EINVAL;
416acc4229cSAndra Paraschiv 
417acc4229cSAndra Paraschiv 	rc = ioctl(enclave_fd, NE_SET_USER_MEMORY_REGION, &mem_region);
418acc4229cSAndra Paraschiv 	if (rc < 0) {
419acc4229cSAndra Paraschiv 		switch (errno) {
420acc4229cSAndra Paraschiv 		case NE_ERR_NOT_IN_INIT_STATE: {
421acc4229cSAndra Paraschiv 			printf("Error in set user memory region, enclave not in init state\n");
422acc4229cSAndra Paraschiv 
423acc4229cSAndra Paraschiv 			break;
424acc4229cSAndra Paraschiv 		}
425acc4229cSAndra Paraschiv 
426acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_MEM_REGION_SIZE: {
427acc4229cSAndra Paraschiv 			printf("Error in set user memory region, mem size not multiple of 2 MiB\n");
428acc4229cSAndra Paraschiv 
429acc4229cSAndra Paraschiv 			break;
430acc4229cSAndra Paraschiv 		}
431acc4229cSAndra Paraschiv 
432acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_MEM_REGION_ADDR: {
433acc4229cSAndra Paraschiv 			printf("Error in set user memory region, invalid user space address\n");
434acc4229cSAndra Paraschiv 
435acc4229cSAndra Paraschiv 			break;
436acc4229cSAndra Paraschiv 		}
437acc4229cSAndra Paraschiv 
438acc4229cSAndra Paraschiv 		case NE_ERR_UNALIGNED_MEM_REGION_ADDR: {
439acc4229cSAndra Paraschiv 			printf("Error in set user memory region, unaligned user space address\n");
440acc4229cSAndra Paraschiv 
441acc4229cSAndra Paraschiv 			break;
442acc4229cSAndra Paraschiv 		}
443acc4229cSAndra Paraschiv 
444acc4229cSAndra Paraschiv 		case NE_ERR_MEM_REGION_ALREADY_USED: {
445acc4229cSAndra Paraschiv 			printf("Error in set user memory region, memory region already used\n");
446acc4229cSAndra Paraschiv 
447acc4229cSAndra Paraschiv 			break;
448acc4229cSAndra Paraschiv 		}
449acc4229cSAndra Paraschiv 
450acc4229cSAndra Paraschiv 		case NE_ERR_MEM_NOT_HUGE_PAGE: {
451acc4229cSAndra Paraschiv 			printf("Error in set user memory region, not backed by huge pages\n");
452acc4229cSAndra Paraschiv 
453acc4229cSAndra Paraschiv 			break;
454acc4229cSAndra Paraschiv 		}
455acc4229cSAndra Paraschiv 
456acc4229cSAndra Paraschiv 		case NE_ERR_MEM_DIFFERENT_NUMA_NODE: {
457acc4229cSAndra Paraschiv 			printf("Error in set user memory region, different NUMA node than CPUs\n");
458acc4229cSAndra Paraschiv 
459acc4229cSAndra Paraschiv 			break;
460acc4229cSAndra Paraschiv 		}
461acc4229cSAndra Paraschiv 
462acc4229cSAndra Paraschiv 		case NE_ERR_MEM_MAX_REGIONS: {
463acc4229cSAndra Paraschiv 			printf("Error in set user memory region, max memory regions reached\n");
464acc4229cSAndra Paraschiv 
465acc4229cSAndra Paraschiv 			break;
466acc4229cSAndra Paraschiv 		}
467acc4229cSAndra Paraschiv 
468acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_PAGE_SIZE: {
469acc4229cSAndra Paraschiv 			printf("Error in set user memory region, has page not multiple of 2 MiB\n");
470acc4229cSAndra Paraschiv 
471acc4229cSAndra Paraschiv 			break;
472acc4229cSAndra Paraschiv 		}
473acc4229cSAndra Paraschiv 
474acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_FLAG_VALUE: {
475acc4229cSAndra Paraschiv 			printf("Error in set user memory region, provided invalid flag\n");
476acc4229cSAndra Paraschiv 
477acc4229cSAndra Paraschiv 			break;
478acc4229cSAndra Paraschiv 		}
479acc4229cSAndra Paraschiv 
480acc4229cSAndra Paraschiv 		default:
481acc4229cSAndra Paraschiv 			printf("Error in set user memory region [%m]\n");
482acc4229cSAndra Paraschiv 		}
483acc4229cSAndra Paraschiv 
484acc4229cSAndra Paraschiv 		return rc;
485acc4229cSAndra Paraschiv 	}
486acc4229cSAndra Paraschiv 
487acc4229cSAndra Paraschiv 	return 0;
488acc4229cSAndra Paraschiv }
489acc4229cSAndra Paraschiv 
490acc4229cSAndra Paraschiv /**
491acc4229cSAndra Paraschiv  * ne_free_mem_regions() - Unmap all the user space memory regions that were set
492acc4229cSAndra Paraschiv  *			   aside for the enclave.
493acc4229cSAndra Paraschiv  * @ne_user_mem_regions:	The user space memory regions associated with an enclave.
494acc4229cSAndra Paraschiv  *
495acc4229cSAndra Paraschiv  * Context: Process context.
496acc4229cSAndra Paraschiv  */
ne_free_mem_regions(struct ne_user_mem_region ne_user_mem_regions[])497acc4229cSAndra Paraschiv static void ne_free_mem_regions(struct ne_user_mem_region ne_user_mem_regions[])
498acc4229cSAndra Paraschiv {
499acc4229cSAndra Paraschiv 	unsigned int i = 0;
500acc4229cSAndra Paraschiv 
501acc4229cSAndra Paraschiv 	for (i = 0; i < NE_DEFAULT_NR_MEM_REGIONS; i++)
502acc4229cSAndra Paraschiv 		munmap(ne_user_mem_regions[i].userspace_addr,
503acc4229cSAndra Paraschiv 		       ne_user_mem_regions[i].memory_size);
504acc4229cSAndra Paraschiv }
505acc4229cSAndra Paraschiv 
506acc4229cSAndra Paraschiv /**
507acc4229cSAndra Paraschiv  * ne_add_vcpu() - Add a vCPU to the given enclave.
508acc4229cSAndra Paraschiv  * @enclave_fd :	The file descriptor associated with the enclave.
509acc4229cSAndra Paraschiv  * @vcpu_id:		vCPU id to be set for the enclave, either provided or
510acc4229cSAndra Paraschiv  *			auto-generated (if provided vCPU id is 0).
511acc4229cSAndra Paraschiv  *
512acc4229cSAndra Paraschiv  * Context: Process context.
513acc4229cSAndra Paraschiv  * Return:
514acc4229cSAndra Paraschiv  * * 0 on success.
515acc4229cSAndra Paraschiv  * * Negative return value on failure.
516acc4229cSAndra Paraschiv  */
ne_add_vcpu(int enclave_fd,unsigned int * vcpu_id)517acc4229cSAndra Paraschiv static int ne_add_vcpu(int enclave_fd, unsigned int *vcpu_id)
518acc4229cSAndra Paraschiv {
519acc4229cSAndra Paraschiv 	int rc = -EINVAL;
520acc4229cSAndra Paraschiv 
521acc4229cSAndra Paraschiv 	rc = ioctl(enclave_fd, NE_ADD_VCPU, vcpu_id);
522acc4229cSAndra Paraschiv 	if (rc < 0) {
523acc4229cSAndra Paraschiv 		switch (errno) {
524acc4229cSAndra Paraschiv 		case NE_ERR_NO_CPUS_AVAIL_IN_POOL: {
525acc4229cSAndra Paraschiv 			printf("Error in add vcpu, no CPUs available in the NE CPU pool\n");
526acc4229cSAndra Paraschiv 
527acc4229cSAndra Paraschiv 			break;
528acc4229cSAndra Paraschiv 		}
529acc4229cSAndra Paraschiv 
530acc4229cSAndra Paraschiv 		case NE_ERR_VCPU_ALREADY_USED: {
531acc4229cSAndra Paraschiv 			printf("Error in add vcpu, the provided vCPU is already used\n");
532acc4229cSAndra Paraschiv 
533acc4229cSAndra Paraschiv 			break;
534acc4229cSAndra Paraschiv 		}
535acc4229cSAndra Paraschiv 
536acc4229cSAndra Paraschiv 		case NE_ERR_VCPU_NOT_IN_CPU_POOL: {
537acc4229cSAndra Paraschiv 			printf("Error in add vcpu, the provided vCPU is not in the NE CPU pool\n");
538acc4229cSAndra Paraschiv 
539acc4229cSAndra Paraschiv 			break;
540acc4229cSAndra Paraschiv 		}
541acc4229cSAndra Paraschiv 
542acc4229cSAndra Paraschiv 		case NE_ERR_VCPU_INVALID_CPU_CORE: {
543acc4229cSAndra Paraschiv 			printf("Error in add vcpu, the core id of the provided vCPU is invalid\n");
544acc4229cSAndra Paraschiv 
545acc4229cSAndra Paraschiv 			break;
546acc4229cSAndra Paraschiv 		}
547acc4229cSAndra Paraschiv 
548acc4229cSAndra Paraschiv 		case NE_ERR_NOT_IN_INIT_STATE: {
549acc4229cSAndra Paraschiv 			printf("Error in add vcpu, enclave not in init state\n");
550acc4229cSAndra Paraschiv 
551acc4229cSAndra Paraschiv 			break;
552acc4229cSAndra Paraschiv 		}
553acc4229cSAndra Paraschiv 
554acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_VCPU: {
555acc4229cSAndra Paraschiv 			printf("Error in add vcpu, the provided vCPU is out of avail CPUs range\n");
556acc4229cSAndra Paraschiv 
557acc4229cSAndra Paraschiv 			break;
558acc4229cSAndra Paraschiv 		}
559acc4229cSAndra Paraschiv 
560acc4229cSAndra Paraschiv 		default:
561acc4229cSAndra Paraschiv 			printf("Error in add vcpu [%m]\n");
562acc4229cSAndra Paraschiv 		}
563*da1c396aSAndra Paraschiv 
564acc4229cSAndra Paraschiv 		return rc;
565acc4229cSAndra Paraschiv 	}
566acc4229cSAndra Paraschiv 
567acc4229cSAndra Paraschiv 	return 0;
568acc4229cSAndra Paraschiv }
569acc4229cSAndra Paraschiv 
570acc4229cSAndra Paraschiv /**
571acc4229cSAndra Paraschiv  * ne_start_enclave() - Start the given enclave.
572acc4229cSAndra Paraschiv  * @enclave_fd :		The file descriptor associated with the enclave.
573acc4229cSAndra Paraschiv  * @enclave_start_info :	Enclave metadata used for starting e.g. vsock CID.
574acc4229cSAndra Paraschiv  *
575acc4229cSAndra Paraschiv  * Context: Process context.
576acc4229cSAndra Paraschiv  * Return:
577acc4229cSAndra Paraschiv  * * 0 on success.
578acc4229cSAndra Paraschiv  * * Negative return value on failure.
579acc4229cSAndra Paraschiv  */
ne_start_enclave(int enclave_fd,struct ne_enclave_start_info * enclave_start_info)580acc4229cSAndra Paraschiv static int ne_start_enclave(int enclave_fd,  struct ne_enclave_start_info *enclave_start_info)
581acc4229cSAndra Paraschiv {
582acc4229cSAndra Paraschiv 	int rc = -EINVAL;
583acc4229cSAndra Paraschiv 
584acc4229cSAndra Paraschiv 	rc = ioctl(enclave_fd, NE_START_ENCLAVE, enclave_start_info);
585acc4229cSAndra Paraschiv 	if (rc < 0) {
586acc4229cSAndra Paraschiv 		switch (errno) {
587acc4229cSAndra Paraschiv 		case NE_ERR_NOT_IN_INIT_STATE: {
588acc4229cSAndra Paraschiv 			printf("Error in start enclave, enclave not in init state\n");
589acc4229cSAndra Paraschiv 
590acc4229cSAndra Paraschiv 			break;
591acc4229cSAndra Paraschiv 		}
592acc4229cSAndra Paraschiv 
593acc4229cSAndra Paraschiv 		case NE_ERR_NO_MEM_REGIONS_ADDED: {
594acc4229cSAndra Paraschiv 			printf("Error in start enclave, no memory regions have been added\n");
595acc4229cSAndra Paraschiv 
596acc4229cSAndra Paraschiv 			break;
597acc4229cSAndra Paraschiv 		}
598acc4229cSAndra Paraschiv 
599acc4229cSAndra Paraschiv 		case NE_ERR_NO_VCPUS_ADDED: {
600acc4229cSAndra Paraschiv 			printf("Error in start enclave, no vCPUs have been added\n");
601acc4229cSAndra Paraschiv 
602acc4229cSAndra Paraschiv 			break;
603acc4229cSAndra Paraschiv 		}
604acc4229cSAndra Paraschiv 
605acc4229cSAndra Paraschiv 		case NE_ERR_FULL_CORES_NOT_USED: {
606acc4229cSAndra Paraschiv 			printf("Error in start enclave, enclave has no full cores set\n");
607acc4229cSAndra Paraschiv 
608acc4229cSAndra Paraschiv 			break;
609acc4229cSAndra Paraschiv 		}
610acc4229cSAndra Paraschiv 
611acc4229cSAndra Paraschiv 		case NE_ERR_ENCLAVE_MEM_MIN_SIZE: {
612acc4229cSAndra Paraschiv 			printf("Error in start enclave, enclave memory is less than min size\n");
613acc4229cSAndra Paraschiv 
614acc4229cSAndra Paraschiv 			break;
615acc4229cSAndra Paraschiv 		}
616acc4229cSAndra Paraschiv 
617acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_FLAG_VALUE: {
618acc4229cSAndra Paraschiv 			printf("Error in start enclave, provided invalid flag\n");
619acc4229cSAndra Paraschiv 
620acc4229cSAndra Paraschiv 			break;
621acc4229cSAndra Paraschiv 		}
622acc4229cSAndra Paraschiv 
623acc4229cSAndra Paraschiv 		case NE_ERR_INVALID_ENCLAVE_CID: {
624acc4229cSAndra Paraschiv 			printf("Error in start enclave, provided invalid enclave CID\n");
625acc4229cSAndra Paraschiv 
626acc4229cSAndra Paraschiv 			break;
627acc4229cSAndra Paraschiv 		}
628acc4229cSAndra Paraschiv 
629acc4229cSAndra Paraschiv 		default:
630acc4229cSAndra Paraschiv 			printf("Error in start enclave [%m]\n");
631acc4229cSAndra Paraschiv 		}
632acc4229cSAndra Paraschiv 
633acc4229cSAndra Paraschiv 		return rc;
634acc4229cSAndra Paraschiv 	}
635acc4229cSAndra Paraschiv 
636acc4229cSAndra Paraschiv 	return 0;
637acc4229cSAndra Paraschiv }
638acc4229cSAndra Paraschiv 
639acc4229cSAndra Paraschiv /**
640059ebe4fSAndra Paraschiv  * ne_start_enclave_check_booted() - Start the enclave and wait for a heartbeat
641acc4229cSAndra Paraschiv  *				     from it, on a newly created vsock channel,
642acc4229cSAndra Paraschiv  *				     to check it has booted.
643acc4229cSAndra Paraschiv  * @enclave_fd :	The file descriptor associated with the enclave.
644acc4229cSAndra Paraschiv  *
645acc4229cSAndra Paraschiv  * Context: Process context.
646acc4229cSAndra Paraschiv  * Return:
647acc4229cSAndra Paraschiv  * * 0 on success.
648acc4229cSAndra Paraschiv  * * Negative return value on failure.
649acc4229cSAndra Paraschiv  */
ne_start_enclave_check_booted(int enclave_fd)650acc4229cSAndra Paraschiv static int ne_start_enclave_check_booted(int enclave_fd)
651acc4229cSAndra Paraschiv {
652acc4229cSAndra Paraschiv 	struct sockaddr_vm client_vsock_addr = {};
653acc4229cSAndra Paraschiv 	int client_vsock_fd = -1;
654acc4229cSAndra Paraschiv 	socklen_t client_vsock_len = sizeof(client_vsock_addr);
655acc4229cSAndra Paraschiv 	struct ne_enclave_start_info enclave_start_info = {};
656acc4229cSAndra Paraschiv 	struct pollfd fds[1] = {};
657acc4229cSAndra Paraschiv 	int rc = -EINVAL;
658acc4229cSAndra Paraschiv 	unsigned char recv_buf = 0;
659acc4229cSAndra Paraschiv 	struct sockaddr_vm server_vsock_addr = {
660acc4229cSAndra Paraschiv 		.svm_family = AF_VSOCK,
661acc4229cSAndra Paraschiv 		.svm_cid = NE_IMAGE_LOAD_HEARTBEAT_CID,
662acc4229cSAndra Paraschiv 		.svm_port = NE_IMAGE_LOAD_HEARTBEAT_PORT,
663acc4229cSAndra Paraschiv 	};
664acc4229cSAndra Paraschiv 	int server_vsock_fd = -1;
665acc4229cSAndra Paraschiv 
666acc4229cSAndra Paraschiv 	server_vsock_fd = socket(AF_VSOCK, SOCK_STREAM, 0);
667acc4229cSAndra Paraschiv 	if (server_vsock_fd < 0) {
668acc4229cSAndra Paraschiv 		rc = server_vsock_fd;
669acc4229cSAndra Paraschiv 
670acc4229cSAndra Paraschiv 		printf("Error in socket [%m]\n");
671acc4229cSAndra Paraschiv 
672acc4229cSAndra Paraschiv 		return rc;
673acc4229cSAndra Paraschiv 	}
674acc4229cSAndra Paraschiv 
675acc4229cSAndra Paraschiv 	rc = bind(server_vsock_fd, (struct sockaddr *)&server_vsock_addr,
676acc4229cSAndra Paraschiv 		  sizeof(server_vsock_addr));
677acc4229cSAndra Paraschiv 	if (rc < 0) {
678acc4229cSAndra Paraschiv 		printf("Error in bind [%m]\n");
679acc4229cSAndra Paraschiv 
680acc4229cSAndra Paraschiv 		goto out;
681acc4229cSAndra Paraschiv 	}
682acc4229cSAndra Paraschiv 
683acc4229cSAndra Paraschiv 	rc = listen(server_vsock_fd, 1);
684acc4229cSAndra Paraschiv 	if (rc < 0) {
685acc4229cSAndra Paraschiv 		printf("Error in listen [%m]\n");
686acc4229cSAndra Paraschiv 
687acc4229cSAndra Paraschiv 		goto out;
688acc4229cSAndra Paraschiv 	}
689acc4229cSAndra Paraschiv 
690acc4229cSAndra Paraschiv 	rc = ne_start_enclave(enclave_fd, &enclave_start_info);
691acc4229cSAndra Paraschiv 	if (rc < 0)
692acc4229cSAndra Paraschiv 		goto out;
693acc4229cSAndra Paraschiv 
694acc4229cSAndra Paraschiv 	printf("Enclave started, CID %llu\n", enclave_start_info.enclave_cid);
695acc4229cSAndra Paraschiv 
696acc4229cSAndra Paraschiv 	fds[0].fd = server_vsock_fd;
697acc4229cSAndra Paraschiv 	fds[0].events = POLLIN;
698acc4229cSAndra Paraschiv 
699acc4229cSAndra Paraschiv 	rc = poll(fds, 1, NE_POLL_WAIT_TIME_MS);
700acc4229cSAndra Paraschiv 	if (rc < 0) {
701acc4229cSAndra Paraschiv 		printf("Error in poll [%m]\n");
702acc4229cSAndra Paraschiv 
703acc4229cSAndra Paraschiv 		goto out;
704acc4229cSAndra Paraschiv 	}
705acc4229cSAndra Paraschiv 
706acc4229cSAndra Paraschiv 	if (!rc) {
707acc4229cSAndra Paraschiv 		printf("Poll timeout, %d seconds elapsed\n", NE_POLL_WAIT_TIME);
708acc4229cSAndra Paraschiv 
709acc4229cSAndra Paraschiv 		rc = -ETIMEDOUT;
710acc4229cSAndra Paraschiv 
711acc4229cSAndra Paraschiv 		goto out;
712acc4229cSAndra Paraschiv 	}
713acc4229cSAndra Paraschiv 
714acc4229cSAndra Paraschiv 	if ((fds[0].revents & POLLIN) == 0) {
715acc4229cSAndra Paraschiv 		printf("Poll received value %d\n", fds[0].revents);
716acc4229cSAndra Paraschiv 
717acc4229cSAndra Paraschiv 		rc = -EINVAL;
718acc4229cSAndra Paraschiv 
719acc4229cSAndra Paraschiv 		goto out;
720acc4229cSAndra Paraschiv 	}
721acc4229cSAndra Paraschiv 
722acc4229cSAndra Paraschiv 	rc = accept(server_vsock_fd, (struct sockaddr *)&client_vsock_addr,
723acc4229cSAndra Paraschiv 		    &client_vsock_len);
724acc4229cSAndra Paraschiv 	if (rc < 0) {
725acc4229cSAndra Paraschiv 		printf("Error in accept [%m]\n");
726acc4229cSAndra Paraschiv 
727acc4229cSAndra Paraschiv 		goto out;
728acc4229cSAndra Paraschiv 	}
729acc4229cSAndra Paraschiv 
730acc4229cSAndra Paraschiv 	client_vsock_fd = rc;
731acc4229cSAndra Paraschiv 
732acc4229cSAndra Paraschiv 	/*
733acc4229cSAndra Paraschiv 	 * Read the heartbeat value that the init process in the enclave sends
734acc4229cSAndra Paraschiv 	 * after vsock connect.
735acc4229cSAndra Paraschiv 	 */
736acc4229cSAndra Paraschiv 	rc = read(client_vsock_fd, &recv_buf, sizeof(recv_buf));
737acc4229cSAndra Paraschiv 	if (rc < 0) {
738acc4229cSAndra Paraschiv 		printf("Error in read [%m]\n");
739acc4229cSAndra Paraschiv 
740acc4229cSAndra Paraschiv 		goto out;
741acc4229cSAndra Paraschiv 	}
742acc4229cSAndra Paraschiv 
743acc4229cSAndra Paraschiv 	if (rc != sizeof(recv_buf) || recv_buf != NE_IMAGE_LOAD_HEARTBEAT_VALUE) {
744acc4229cSAndra Paraschiv 		printf("Read %d instead of %d\n", recv_buf,
745acc4229cSAndra Paraschiv 		       NE_IMAGE_LOAD_HEARTBEAT_VALUE);
746acc4229cSAndra Paraschiv 
747acc4229cSAndra Paraschiv 		goto out;
748acc4229cSAndra Paraschiv 	}
749acc4229cSAndra Paraschiv 
750acc4229cSAndra Paraschiv 	/* Write the heartbeat value back. */
751acc4229cSAndra Paraschiv 	rc = write(client_vsock_fd, &recv_buf, sizeof(recv_buf));
752acc4229cSAndra Paraschiv 	if (rc < 0) {
753acc4229cSAndra Paraschiv 		printf("Error in write [%m]\n");
754acc4229cSAndra Paraschiv 
755acc4229cSAndra Paraschiv 		goto out;
756acc4229cSAndra Paraschiv 	}
757acc4229cSAndra Paraschiv 
758acc4229cSAndra Paraschiv 	rc = 0;
759acc4229cSAndra Paraschiv 
760acc4229cSAndra Paraschiv out:
761acc4229cSAndra Paraschiv 	close(server_vsock_fd);
762acc4229cSAndra Paraschiv 
763acc4229cSAndra Paraschiv 	return rc;
764acc4229cSAndra Paraschiv }
765acc4229cSAndra Paraschiv 
main(int argc,char * argv[])766acc4229cSAndra Paraschiv int main(int argc, char *argv[])
767acc4229cSAndra Paraschiv {
768acc4229cSAndra Paraschiv 	int enclave_fd = -1;
769acc4229cSAndra Paraschiv 	unsigned int i = 0;
770acc4229cSAndra Paraschiv 	int ne_dev_fd = -1;
771acc4229cSAndra Paraschiv 	struct ne_user_mem_region ne_user_mem_regions[NE_DEFAULT_NR_MEM_REGIONS] = {};
772acc4229cSAndra Paraschiv 	unsigned int ne_vcpus[NE_DEFAULT_NR_VCPUS] = {};
773acc4229cSAndra Paraschiv 	int rc = -EINVAL;
774acc4229cSAndra Paraschiv 	pthread_t thread_id = 0;
775acc4229cSAndra Paraschiv 	unsigned long slot_uid = 0;
776acc4229cSAndra Paraschiv 
777acc4229cSAndra Paraschiv 	if (argc != 2) {
778acc4229cSAndra Paraschiv 		printf("Usage: %s <path_to_enclave_image>\n", argv[0]);
779acc4229cSAndra Paraschiv 
780acc4229cSAndra Paraschiv 		exit(EXIT_FAILURE);
781acc4229cSAndra Paraschiv 	}
782acc4229cSAndra Paraschiv 
783acc4229cSAndra Paraschiv 	if (strlen(argv[1]) >= PATH_MAX) {
784acc4229cSAndra Paraschiv 		printf("The size of the path to enclave image is higher than max path\n");
785acc4229cSAndra Paraschiv 
786acc4229cSAndra Paraschiv 		exit(EXIT_FAILURE);
787acc4229cSAndra Paraschiv 	}
788acc4229cSAndra Paraschiv 
789acc4229cSAndra Paraschiv 	ne_dev_fd = open(NE_DEV_NAME, O_RDWR | O_CLOEXEC);
790acc4229cSAndra Paraschiv 	if (ne_dev_fd < 0) {
791acc4229cSAndra Paraschiv 		printf("Error in open NE device [%m]\n");
792acc4229cSAndra Paraschiv 
793acc4229cSAndra Paraschiv 		exit(EXIT_FAILURE);
794acc4229cSAndra Paraschiv 	}
795acc4229cSAndra Paraschiv 
796acc4229cSAndra Paraschiv 	printf("Creating enclave slot ...\n");
797acc4229cSAndra Paraschiv 
798acc4229cSAndra Paraschiv 	rc = ne_create_vm(ne_dev_fd, &slot_uid, &enclave_fd);
799acc4229cSAndra Paraschiv 
800acc4229cSAndra Paraschiv 	close(ne_dev_fd);
801acc4229cSAndra Paraschiv 
802acc4229cSAndra Paraschiv 	if (rc < 0)
803acc4229cSAndra Paraschiv 		exit(EXIT_FAILURE);
804acc4229cSAndra Paraschiv 
805acc4229cSAndra Paraschiv 	printf("Enclave fd %d\n", enclave_fd);
806acc4229cSAndra Paraschiv 
807acc4229cSAndra Paraschiv 	rc = pthread_create(&thread_id, NULL, ne_poll_enclave_fd, (void *)&enclave_fd);
808acc4229cSAndra Paraschiv 	if (rc < 0) {
809acc4229cSAndra Paraschiv 		printf("Error in thread create [%m]\n");
810acc4229cSAndra Paraschiv 
811acc4229cSAndra Paraschiv 		close(enclave_fd);
812acc4229cSAndra Paraschiv 
813acc4229cSAndra Paraschiv 		exit(EXIT_FAILURE);
814acc4229cSAndra Paraschiv 	}
815acc4229cSAndra Paraschiv 
816acc4229cSAndra Paraschiv 	for (i = 0; i < NE_DEFAULT_NR_MEM_REGIONS; i++) {
817acc4229cSAndra Paraschiv 		ne_user_mem_regions[i].memory_size = NE_MIN_MEM_REGION_SIZE;
818acc4229cSAndra Paraschiv 
819acc4229cSAndra Paraschiv 		rc = ne_alloc_user_mem_region(&ne_user_mem_regions[i]);
820acc4229cSAndra Paraschiv 		if (rc < 0) {
821acc4229cSAndra Paraschiv 			printf("Error in alloc userspace memory region, iter %d\n", i);
822acc4229cSAndra Paraschiv 
823acc4229cSAndra Paraschiv 			goto release_enclave_fd;
824acc4229cSAndra Paraschiv 		}
825acc4229cSAndra Paraschiv 	}
826acc4229cSAndra Paraschiv 
827acc4229cSAndra Paraschiv 	rc = ne_load_enclave_image(enclave_fd, ne_user_mem_regions, argv[1]);
828acc4229cSAndra Paraschiv 	if (rc < 0)
829acc4229cSAndra Paraschiv 		goto release_enclave_fd;
830acc4229cSAndra Paraschiv 
831acc4229cSAndra Paraschiv 	for (i = 0; i < NE_DEFAULT_NR_MEM_REGIONS; i++) {
832acc4229cSAndra Paraschiv 		rc = ne_set_user_mem_region(enclave_fd, ne_user_mem_regions[i]);
833acc4229cSAndra Paraschiv 		if (rc < 0) {
834acc4229cSAndra Paraschiv 			printf("Error in set memory region, iter %d\n", i);
835acc4229cSAndra Paraschiv 
836acc4229cSAndra Paraschiv 			goto release_enclave_fd;
837acc4229cSAndra Paraschiv 		}
838acc4229cSAndra Paraschiv 	}
839acc4229cSAndra Paraschiv 
840acc4229cSAndra Paraschiv 	printf("Enclave memory regions were added\n");
841acc4229cSAndra Paraschiv 
842acc4229cSAndra Paraschiv 	for (i = 0; i < NE_DEFAULT_NR_VCPUS; i++) {
843acc4229cSAndra Paraschiv 		/*
844acc4229cSAndra Paraschiv 		 * The vCPU is chosen from the enclave vCPU pool, if the value
845acc4229cSAndra Paraschiv 		 * of the vcpu_id is 0.
846acc4229cSAndra Paraschiv 		 */
847acc4229cSAndra Paraschiv 		ne_vcpus[i] = 0;
848acc4229cSAndra Paraschiv 		rc = ne_add_vcpu(enclave_fd, &ne_vcpus[i]);
849acc4229cSAndra Paraschiv 		if (rc < 0) {
850acc4229cSAndra Paraschiv 			printf("Error in add vcpu, iter %d\n", i);
851acc4229cSAndra Paraschiv 
852acc4229cSAndra Paraschiv 			goto release_enclave_fd;
853acc4229cSAndra Paraschiv 		}
854acc4229cSAndra Paraschiv 
855acc4229cSAndra Paraschiv 		printf("Added vCPU %d to the enclave\n", ne_vcpus[i]);
856acc4229cSAndra Paraschiv 	}
857acc4229cSAndra Paraschiv 
858acc4229cSAndra Paraschiv 	printf("Enclave vCPUs were added\n");
859acc4229cSAndra Paraschiv 
860acc4229cSAndra Paraschiv 	rc = ne_start_enclave_check_booted(enclave_fd);
861acc4229cSAndra Paraschiv 	if (rc < 0) {
862acc4229cSAndra Paraschiv 		printf("Error in the enclave start / image loading heartbeat logic [rc=%d]\n", rc);
863acc4229cSAndra Paraschiv 
864acc4229cSAndra Paraschiv 		goto release_enclave_fd;
865acc4229cSAndra Paraschiv 	}
866acc4229cSAndra Paraschiv 
867acc4229cSAndra Paraschiv 	printf("Entering sleep for %d seconds ...\n", NE_SLEEP_TIME);
868acc4229cSAndra Paraschiv 
869acc4229cSAndra Paraschiv 	sleep(NE_SLEEP_TIME);
870acc4229cSAndra Paraschiv 
871acc4229cSAndra Paraschiv 	close(enclave_fd);
872acc4229cSAndra Paraschiv 
873acc4229cSAndra Paraschiv 	ne_free_mem_regions(ne_user_mem_regions);
874acc4229cSAndra Paraschiv 
875acc4229cSAndra Paraschiv 	exit(EXIT_SUCCESS);
876acc4229cSAndra Paraschiv 
877acc4229cSAndra Paraschiv release_enclave_fd:
878acc4229cSAndra Paraschiv 	close(enclave_fd);
879acc4229cSAndra Paraschiv 	ne_free_mem_regions(ne_user_mem_regions);
880acc4229cSAndra Paraschiv 
881acc4229cSAndra Paraschiv 	exit(EXIT_FAILURE);
882acc4229cSAndra Paraschiv }
883