#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Kernel { class Process { BAN_NON_COPYABLE(Process); BAN_NON_MOVABLE(Process); public: using entry_t = Thread::entry_t; public: static Process* create_kernel(entry_t, void*); static BAN::ErrorOr create_userspace(BAN::StringView); ~Process(); [[noreturn]] void exit(); void add_thread(Thread*); void on_thread_exit(Thread&); BAN::ErrorOr init_stdio(); BAN::ErrorOr set_termios(const termios&); pid_t pid() const { return m_pid; } BAN::ErrorOr fork(uintptr_t rsp, uintptr_t rip); BAN::ErrorOr open(BAN::StringView, int); BAN::ErrorOr close(int fd); BAN::ErrorOr read(int fd, void* buffer, size_t count); BAN::ErrorOr write(int fd, const void* buffer, size_t count); BAN::ErrorOr creat(BAN::StringView name, mode_t); BAN::ErrorOr seek(int fd, off_t offset, int whence); BAN::ErrorOr tell(int fd); BAN::ErrorOr fstat(int fd, struct stat*); BAN::ErrorOr stat(BAN::StringView path, struct stat*); BAN::ErrorOr mount(BAN::StringView source, BAN::StringView target); BAN::ErrorOr> read_directory_entries(int); BAN::ErrorOr working_directory() const; BAN::ErrorOr set_working_directory(BAN::StringView); TTY& tty() { ASSERT(m_tty); return *m_tty; } BAN::ErrorOr allocate(size_t); void free(void*); void termid(char*) const; static Process& current() { return Thread::current().process(); } MMU& mmu() { return m_mmu ? *m_mmu : MMU::kernel(); } private: Process(pid_t); static Process* create_process(); static void register_process(Process*); BAN::ErrorOr absolute_path_of(BAN::StringView) const; private: struct OpenFileDescription { BAN::RefPtr inode; BAN::String path; off_t offset { 0 }; uint8_t flags { 0 }; }; BAN::ErrorOr validate_fd(int); OpenFileDescription& open_file_description(int); BAN::ErrorOr get_free_fd(); BAN::Vector m_open_files; BAN::Vector m_mapped_ranges; mutable RecursiveSpinLock m_lock; const pid_t m_pid = 0; BAN::String m_working_directory; BAN::Vector m_threads; BAN::Vector m_fixed_width_allocators; GeneralAllocator* m_general_allocator; MMU* m_mmu { nullptr }; TTY* m_tty { nullptr }; }; }