|  | /* | 
|  | * Copyright 2014, NICTA | 
|  | * | 
|  | * This software may be distributed and modified according to the terms of | 
|  | * the BSD 2-Clause license. Note that NO WARRANTY is provided. | 
|  | * See "LICENSE_BSD2.txt" for details. | 
|  | * | 
|  | * @TAG(NICTA_BSD) | 
|  | */ | 
|  | #ifndef SEL4UTILS_PROCESS_H | 
|  | #define SEL4UTILS_PROCESS_H | 
|  |  | 
|  | #include <autoconf.h> | 
|  |  | 
|  | #if (defined CONFIG_LIB_SEL4_VSPACE && defined CONFIG_LIB_SEL4_VKA) | 
|  |  | 
|  | #include <vka/vka.h> | 
|  | #include <vspace/vspace.h> | 
|  |  | 
|  | #include <sel4utils/thread.h> | 
|  | #include <sel4utils/vspace.h> | 
|  | #include <sel4utils/elf.h> | 
|  |  | 
|  | #define WORD_STRING_SIZE ((CONFIG_WORD_SIZE / 3) + 1) | 
|  |  | 
|  | typedef struct object_node object_node_t; | 
|  |  | 
|  | struct object_node { | 
|  | vka_object_t object; | 
|  | object_node_t *next; | 
|  | }; | 
|  |  | 
|  | typedef struct { | 
|  | vka_object_t pd; | 
|  | vspace_t vspace; | 
|  | sel4utils_alloc_data_t data; | 
|  | vka_object_t cspace; | 
|  | uint32_t cspace_size; | 
|  | uint32_t cspace_next_free; | 
|  | sel4utils_thread_t thread; | 
|  | vka_object_t fault_endpoint; | 
|  | void *entry_point; | 
|  | uintptr_t sysinfo; | 
|  | object_node_t *allocated_object_list_head; | 
|  | /* if the elf wasn't loaded into the address space, this describes the regions. | 
|  | * this permits lazy loading / copy on write / page sharing / whatever crazy thing | 
|  | * you want to implement */ | 
|  | int num_elf_regions; | 
|  | sel4utils_elf_region_t *elf_regions; | 
|  | bool own_vspace; | 
|  | bool own_cspace; | 
|  | } sel4utils_process_t; | 
|  |  | 
|  | /* sel4utils processes start with some caps in their cspace. | 
|  | * These are the caps | 
|  | */ | 
|  | enum sel4utils_cspace_layout { | 
|  | /* no cap in NULL */ | 
|  | SEL4UTILS_NULL_SLOT = 0, | 
|  | /* | 
|  | * The root cnode (with appropriate guard) | 
|  | */ | 
|  | SEL4UTILS_CNODE_SLOT = 1, | 
|  | /* The slot on the cspace that fault_endpoint is put if | 
|  | * sel4utils_configure_process is used. | 
|  | */ | 
|  | SEL4UTILS_ENDPOINT_SLOT = 2, | 
|  |  | 
|  | /* The page directory slot */ | 
|  | SEL4UTILS_PD_SLOT = 3, | 
|  |  | 
|  | /* the slot for the asid pool that this thread is in and can create threads | 
|  | * in. 0 if this kernel does not support asid pools */ | 
|  | SEL4UTILS_ASID_POOL_SLOT = 4, | 
|  |  | 
|  | /* the slot for this processes tcb */ | 
|  | SEL4UTILS_TCB_SLOT = 5, | 
|  |  | 
|  | /* First free slot in the cspace configured by sel4utils */ | 
|  | SEL4UTILS_FIRST_FREE = 6 | 
|  | }; | 
|  |  | 
|  | typedef struct { | 
|  | /* should we handle elf logic at all? */ | 
|  | bool is_elf; | 
|  | /* if so what is the image name? */ | 
|  | char *image_name; | 
|  | /* Do you want the elf image preloaded? */ | 
|  | bool do_elf_load; | 
|  |  | 
|  | /* otherwise what is the entry point and sysinfo? */ | 
|  | void *entry_point; | 
|  | uintptr_t sysinfo; | 
|  |  | 
|  | /* should we create a default single level cspace? */ | 
|  | bool create_cspace; | 
|  | /* if so how big ? */ | 
|  | int one_level_cspace_size_bits; | 
|  |  | 
|  | /* otherwise what is the root cnode ?*/ | 
|  | /* Note if you use a custom cspace then | 
|  | * sel4utils_copy_cap_to_process etc will not work */ | 
|  | vka_object_t cnode; | 
|  |  | 
|  | /* do you want us to create a vspace for you? */ | 
|  | bool create_vspace; | 
|  | /* if not what is the page dir, and what is the vspace */ | 
|  | vspace_t *vspace; | 
|  | vka_object_t page_dir; | 
|  |  | 
|  | /* if so, is there a regions you want left clear?*/ | 
|  | sel4utils_elf_region_t *reservations; | 
|  | int num_reservations; | 
|  |  | 
|  | /* do you want a fault endpoint created? */ | 
|  | bool create_fault_endpoint; | 
|  | /* otherwise what is it */ | 
|  | vka_object_t fault_endpoint; | 
|  |  | 
|  | int priority; | 
|  | #ifndef CONFIG_KERNEL_STABLE | 
|  | seL4_CPtr asid_pool; | 
|  | #endif | 
|  | } sel4utils_process_config_t; | 
|  |  | 
|  | /** | 
|  | * Start a process, and copy arguments into the processes address space. | 
|  | * | 
|  | * This is intended to use when loading applications of the format: | 
|  | * | 
|  | * int main(int argc, char **argv) { }; | 
|  | * | 
|  | * The third argument (in r2 for arm, on stack for ia32) will be the address of the ipc buffer. | 
|  | * This is intended to be loaded by the crt into the data word of the ipc buffer, | 
|  | * or you can just add it to your function prototype. | 
|  | * | 
|  | * Add the following to your crt to use it this way: | 
|  | * | 
|  | * arm: str r2, [r2, #484] | 
|  | * ia32: popl %ebp | 
|  | *       movl %ebp, 484(%ebp) | 
|  | * | 
|  | * @param process initialised sel4utils process struct. | 
|  | * @param vka     vka interface to use for allocation of frames. | 
|  | * @param vspace  the current vspace. | 
|  | * @param argc    the number of arguments. | 
|  | * @param argv    a pointer to an array of strings in the current vspace. | 
|  | * @param resume  1 to start the process, 0 to leave suspended. | 
|  | * | 
|  | * @return -1 on error, 0 on success. | 
|  | * | 
|  | */ | 
|  | int sel4utils_spawn_process(sel4utils_process_t *process, vka_t *vka, vspace_t *vspace, | 
|  | int argc, char *argv[], int resume); | 
|  |  | 
|  | /** | 
|  | * Start a process, and copy arguments into the processes address space. | 
|  | * | 
|  | * This is intended for use when loading applications that have a System V ABI compliant | 
|  | * entry point. This means that the entry point should *not* be a 'main' that is expecting | 
|  | * argc and argv in the form as passed here, but should be an _start routine that will | 
|  | * take a stack frame with the arguments on it and construct an appropriate invocation | 
|  | * to a main function. The stack frame passed to the entry point will have an auxv, envp | 
|  | * and argv | 
|  | * | 
|  | * @param process initialised sel4utils process struct. | 
|  | * @param vka     vka interface to use for allocation of frames. | 
|  | * @param vspace  the current vspace. | 
|  | * @param argc    the number of arguments. | 
|  | * @param argv    a pointer to an array of strings in the current vspace. | 
|  | * @param resume  1 to start the process, 0 to leave suspended. | 
|  | * | 
|  | * @return -1 on error, 0 on success. | 
|  | * | 
|  | */ | 
|  | int sel4utils_spawn_process_v(sel4utils_process_t *process, vka_t *vka, vspace_t *vspace, | 
|  | int argc, char *argv[], int resume); | 
|  |  | 
|  | /** | 
|  | * This is the function to use if you just want to set up a process as fast as possible. | 
|  | * It creates a simple cspace and vspace for you, allocates a fault endpoint and puts | 
|  | * it into the new cspace. | 
|  | * | 
|  | * It loads the elf file into the new vspace, parses the elf file and stores the entry point | 
|  | * into process->entry_point. | 
|  | * | 
|  | * It uses the same vka for both allocations - the new process will not have an allocator. | 
|  | * | 
|  | * The process will have just one thread. | 
|  | * | 
|  | * Otherwise, use it as a model for loading processes and use the elf_load. | 
|  | * | 
|  | * WARNING: when launching processes on ia32 please ensure your calling convention | 
|  | * matches that documented in sel4utils_start_thread. Otherwise your arguments | 
|  | * won't come through correctly. | 
|  | * | 
|  | * @param process      uninitialised process struct. | 
|  | * @param vka          allocator to use to allocate objects. | 
|  | * @param vspace vspace allocator for the current vspace. | 
|  | * @param priority     priority to configure the process to run as. | 
|  | * @param image_name   name of the elf image to load from the cpio archive. | 
|  | * | 
|  | * @return 0 on success, -1 on error. | 
|  | */ | 
|  | int sel4utils_configure_process(sel4utils_process_t *process, vka_t *vka, vspace_t *vspace, | 
|  | uint8_t priority, char *image_name); | 
|  |  | 
|  | /** | 
|  | * Configure a process with more customisations (Create your own vspace, customise cspace size). | 
|  | * | 
|  | * @param process               uninitialised process struct | 
|  | * @param vka                   allocator to use to allocate objects. | 
|  | * @param spawner_vspace        vspace to use to allocate virtual memory in the current address space. | 
|  | * @param config process config. | 
|  | * | 
|  | * @return 0 on success, -1 on error. | 
|  | */ | 
|  | int sel4utils_configure_process_custom(sel4utils_process_t *process, vka_t *target_vka, | 
|  | vspace_t *spawner_vspace, sel4utils_process_config_t config); | 
|  |  | 
|  | /** | 
|  | * Copy a cap into a process' cspace. | 
|  | * | 
|  | * This will only work if you configured the process using one of the above functions, or | 
|  | * have mimicked their functionality. | 
|  | * | 
|  | * @param process process to copy the cap to | 
|  | * @param src     path in the current cspace to copy the cap from | 
|  | * | 
|  | * @return 0 on failure, otherwise the slot in the processes cspace. | 
|  | */ | 
|  | seL4_CPtr sel4utils_copy_cap_to_process(sel4utils_process_t *process, cspacepath_t src); | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Move a cap into a process' cspace. | 
|  | * | 
|  | * This will only work if you configured the process using one of the above functions, or | 
|  | * have mimicked their functionality. | 
|  | * | 
|  | * @param process process to move the cap to | 
|  | * @param src     path in the current cspace to move the cap from | 
|  | * @param vka     the allocator that owns the cslot the cap is currently in, so it may be freed after the move. | 
|  | * | 
|  | * @return 0 on failure, otherwise the slot in the processes cspace. | 
|  | */ | 
|  | seL4_CPtr sel4utils_move_cap_to_process(sel4utils_process_t *process, cspacepath_t src, vka_t *from_vka); | 
|  |  | 
|  | /** | 
|  | * | 
|  | * Mint a cap into a process' cspace. | 
|  | * | 
|  | * As above, except mint the cap with rights and data. | 
|  | * | 
|  | * @return 0 on failure, otherwise the slot in the cspace where the new cap is. | 
|  | */ | 
|  | seL4_CPtr sel4utils_mint_cap_to_process(sel4utils_process_t *process, cspacepath_t src, seL4_CapRights rights, seL4_CapData_t data); | 
|  |  | 
|  | /** | 
|  | * Destroy a process. | 
|  | * | 
|  | * This will free everything possible associated with a process and teardown the vspace. | 
|  | * | 
|  | * @param process process to destroy | 
|  | * @param vka allocator used to allocate objects for this process | 
|  | */ | 
|  | void sel4utils_destroy_process(sel4utils_process_t *process, vka_t *vka); | 
|  |  | 
|  |  | 
|  | /* | 
|  | * sel4utils default allocated object function for vspaces. | 
|  | * | 
|  | * Stores a list of allocated objects in the process struct and frees them | 
|  | * when sel4utils_destroy_process is called. | 
|  | */ | 
|  | void sel4utils_allocated_object(void *cookie, vka_object_t object); | 
|  |  | 
|  | /* | 
|  | * Create c-formatted argument list to pass to a process from arbitrarily long amount of words. | 
|  | * | 
|  | * @param strings empty 2d array of chars to populate with word strings. | 
|  | * @param argv empty 1d array of char pointers which will be set up with pointers to | 
|  | *             strings in strings. | 
|  | * @param argc number of words | 
|  | * @param ... list of words to create arguments from. | 
|  | * | 
|  | */ | 
|  | void sel4utils_create_word_args(char strings[][WORD_STRING_SIZE], char *argv[], int argc, ...); | 
|  |  | 
|  | #endif /* (defined CONFIG_LIB_SEL4_VSPACE && defined CONFIG_LIB_SEL4_VKA) */ | 
|  | #endif /* SEL4UTILS_PROCESS_H */ |