#pragma once #include #include #include struct vfs_mount; #define HANDLE_MAX 16 enum process_state { PS_RUNNING, PS_DEAD, // return message not collected PS_WAITS4CHILDDEATH, PS_WAITS4FS, PS_WAITS4REQUEST, PS_WAITS4PIPE, PS_WAITS4TIMER, PS_LAST, }; struct process { struct pagedir *pages; /* if NULL, refcount == 1. kmalloc'd */ uint64_t *pages_refcount; struct registers regs; struct process *sibling, *child, *parent; enum process_state state; union { /* saved value, meaning depends on .state */ int death_msg; // PS_DEAD struct { char __user *buf; size_t max_len; struct fs_wait_response __user *res; } awaited_req; // PS_WAITS4REQUEST struct { struct handle *pipe; char __user *buf; size_t len; struct process *next; } waits4pipe; struct { /* managed by timer_schedule */ uint64_t goal; struct process *next; } waits4timer; }; struct vfs_mount *mount; struct handle *handles[HANDLE_MAX]; uint32_t id; /* only for debugging, don't expose to userland */ bool noreap; /* allocated once, the requests from WAITS4FS get stored here */ struct vfs_request *reqslot; /* vfs_backend controlled (not exclusively) by this process */ struct vfs_backend *controlled; struct vfs_request *handled_req; struct { void *buf; size_t len; size_t pos; } execbuf; }; extern struct process *process_current; /** Creates the root process. */ struct process *process_seed(void *data, size_t datalen); struct process *process_fork(struct process *parent, int flags); void process_kill(struct process *proc, int ret); /** Tries to free a process / collect its return value. */ void process_try2collect(struct process *dead); /** Switches execution to any running process. */ _Noreturn void process_switch_any(void); /** Used for iterating over all processes */ struct process *process_next(struct process *); handle_t process_find_free_handle(struct process *proc, handle_t start_at); struct handle *process_handle_get(struct process *, handle_t); void process_transition(struct process *, enum process_state);