blob: 1186dbf240ef1c5ab7a3d928cab482c2238b239a [file] [log] [blame]
/*
* 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 */